libbe_build: Synchronize Node.cpp with the main one.

Mostly formatting changes with a few minor bugfixes it appears.
This commit is contained in:
Augustin Cavalier 2021-11-18 11:18:49 -05:00
parent a7639d2389
commit 28c8a16a5a
2 changed files with 279 additions and 429 deletions

View File

@ -28,11 +28,7 @@ struct node_ref {
bool operator!=(const node_ref &ref) const; bool operator!=(const node_ref &ref) const;
node_ref& operator=(const node_ref &ref); node_ref& operator=(const node_ref &ref);
bool operator<(const node_ref &ref) const bool operator<(const node_ref &ref) const;
{
return device < ref.device
|| (device == ref.device && node < ref.node);
}
dev_t device; dev_t device;
ino_t node; ino_t node;

View File

@ -1,22 +1,26 @@
//---------------------------------------------------------------------- /*
// This software is part of the Haiku distribution and is covered * Copyright 2002-2011 Haiku, Inc. All rights reserved.
// by the MIT license. * Distributed under the terms of the MIT License.
//---------------------------------------------------------------------- *
/*! * Authors:
\file Node.cpp * Tyler Dauwalder
BNode implementation. * Ingo Weinhold, bonefish@users.sf.net
*/ */
#include <Node.h>
#include <errno.h> #include <errno.h>
#include <fcntl.h> #include <fcntl.h>
#include <fs_attr.h> // for struct attr_info
#include <new> #include <new>
#include <string.h> #include <string.h>
#include <unistd.h> #include <unistd.h>
#include <compat/sys/stat.h>
#include <Directory.h> #include <Directory.h>
#include <Entry.h> #include <Entry.h>
#include <Node.h> #include <fs_attr.h>
#include <String.h> #include <String.h>
#include <TypeConstants.h> #include <TypeConstants.h>
@ -24,243 +28,178 @@
#include "storage_support.h" #include "storage_support.h"
//----------------------------------------------------------------------
// node_ref
//----------------------------------------------------------------------
// constructor // #pragma mark - node_ref
/*! \brief Creates an uninitialized node_ref object.
*/
node_ref::node_ref() node_ref::node_ref()
: device((dev_t)-1), :
device((dev_t)-1),
node((ino_t)-1) node((ino_t)-1)
{ {
} }
// copy constructor
/*! \brief Creates a copy of the given node_ref object. node_ref::node_ref(const node_ref& other)
\param ref the node_ref to be copied :
*/ device((dev_t)-1),
node_ref::node_ref(const node_ref &ref)
: device((dev_t)-1),
node((ino_t)-1) node((ino_t)-1)
{ {
*this = ref; *this = other;
} }
// ==
/*! \brief Tests whether this node_ref and the supplied one are equal.
\param ref the node_ref to be compared with
\return \c true, if the objects are equal, \c false otherwise
*/
bool bool
node_ref::operator==(const node_ref &ref) const node_ref::operator==(const node_ref& other) const
{ {
return (device == ref.device && node == ref.node); return (device == other.device && node == other.node);
} }
// !=
/*! \brief Tests whether this node_ref and the supplied one are not equal.
\param ref the node_ref to be compared with
\return \c false, if the objects are equal, \c true otherwise
*/
bool bool
node_ref::operator!=(const node_ref &ref) const node_ref::operator!=(const node_ref& other) const
{ {
return !(*this == ref); return !(*this == other);
} }
// =
/*! \brief Makes this node ref a copy of the supplied one. bool
\param ref the node_ref to be copied node_ref::operator<(const node_ref& other) const
\return a reference to this object {
*/ if (this->device != other.device)
return this->device < other.device;
return this->node < other.node;
}
node_ref& node_ref&
node_ref::operator=(const node_ref &ref) node_ref::operator=(const node_ref& other)
{ {
device = ref.device; device = other.device;
node = ref.node; node = other.node;
return *this; return *this;
} }
//----------------------------------------------------------------------
// BNode
//----------------------------------------------------------------------
/*! \brief Creates an uninitialized BNode object // #pragma mark - BNode
*/
BNode::BNode() BNode::BNode()
: fFd(-1), :
fFd(-1),
fAttrFd(-1), fAttrFd(-1),
fCStatus(B_NO_INIT) fCStatus(B_NO_INIT)
{ {
} }
/*! \brief Creates a BNode object and initializes it to the specified
entry_ref. BNode::BNode(const entry_ref* ref)
\param ref the entry_ref referring to the entry :
*/ fFd(-1),
BNode::BNode(const entry_ref *ref)
: fFd(-1),
fAttrFd(-1), fAttrFd(-1),
fCStatus(B_NO_INIT) fCStatus(B_NO_INIT)
{ {
SetTo(ref); // fCStatus is set by SetTo(), ignore return value
(void)SetTo(ref);
} }
/*! \brief Creates a BNode object and initializes it to the specified
filesystem entry. BNode::BNode(const BEntry* entry)
\param entry the BEntry representing the entry :
*/ fFd(-1),
BNode::BNode(const BEntry *entry)
: fFd(-1),
fAttrFd(-1), fAttrFd(-1),
fCStatus(B_NO_INIT) fCStatus(B_NO_INIT)
{ {
SetTo(entry); // fCStatus is set by SetTo(), ignore return value
(void)SetTo(entry);
} }
/*! \brief Creates a BNode object and initializes it to the entry referred
to by the specified path. BNode::BNode(const char* path)
\param path the path referring to the entry :
*/ fFd(-1),
BNode::BNode(const char *path)
: fFd(-1),
fAttrFd(-1), fAttrFd(-1),
fCStatus(B_NO_INIT) fCStatus(B_NO_INIT)
{ {
SetTo(path); // fCStatus is set by SetTo(), ignore return value
(void)SetTo(path);
} }
/*! \brief Creates a BNode object and initializes it to the entry referred
to by the specified path rooted in the specified directory. BNode::BNode(const BDirectory* dir, const char* path)
\param dir the BDirectory, relative to which the entry's path name is :
given fFd(-1),
\param path the entry's path name relative to \a dir
*/
BNode::BNode(const BDirectory *dir, const char *path)
: fFd(-1),
fAttrFd(-1), fAttrFd(-1),
fCStatus(B_NO_INIT) fCStatus(B_NO_INIT)
{ {
SetTo(dir, path); // fCStatus is set by SetTo(), ignore return value
(void)SetTo(dir, path);
} }
/*! \brief Creates a copy of the given BNode.
\param node the BNode to be copied BNode::BNode(const BNode& node)
*/ :
BNode::BNode(const BNode &node) fFd(-1),
: fFd(-1),
fAttrFd(-1), fAttrFd(-1),
fCStatus(B_NO_INIT) fCStatus(B_NO_INIT)
{ {
*this = node; *this = node;
} }
/*! \brief Frees all resources associated with this BNode.
*/
BNode::~BNode() BNode::~BNode()
{ {
Unset(); Unset();
} }
/*! \brief Checks whether the object has been properly initialized or not.
\return
- \c B_OK, if the object has been properly initialized,
- an error code, otherwise.
*/
status_t status_t
BNode::InitCheck() const BNode::InitCheck() const
{ {
return fCStatus; return fCStatus;
} }
/*! \brief Fills in the given stat structure with \code stat() \endcode
information for this object.
\param st a pointer to a stat structure to be filled in
\return
- \c B_OK: Everything went fine.
- \c B_BAD_VALUE: \c NULL \a st.
- another error code, e.g., if the object wasn't properly initialized
*/
status_t
BNode::GetStat(struct stat *st) const
{
return (fCStatus != B_OK)
? fCStatus
: _kern_read_stat(fFd, NULL, false, st, sizeof(struct stat));
}
/*! \brief Reinitializes the object to the specified entry_ref.
\param ref the entry_ref referring to the entry
\return
- \c B_OK: Everything went fine.
- \c B_BAD_VALUE: \c NULL \a ref.
- \c B_ENTRY_NOT_FOUND: The entry could not be found.
- \c B_BUSY: The entry is locked.
*/
status_t status_t
BNode::SetTo(const entry_ref *ref) BNode::SetTo(const entry_ref* ref)
{ {
return _SetTo(ref, false); return _SetTo(ref, false);
} }
/*! \brief Reinitializes the object to the specified filesystem entry.
\param entry the BEntry representing the entry
\return
- \c B_OK: Everything went fine.
- \c B_BAD_VALUE: \c NULL \a entry.
- \c B_ENTRY_NOT_FOUND: The entry could not be found.
- \c B_BUSY: The entry is locked.
*/
status_t status_t
BNode::SetTo(const BEntry *entry) BNode::SetTo(const BEntry* entry)
{ {
if (!entry) { if (entry == NULL) {
Unset(); Unset();
return (fCStatus = B_BAD_VALUE); return (fCStatus = B_BAD_VALUE);
} }
return _SetTo(entry->fDirFd, entry->fName, false); return _SetTo(entry->fDirFd, entry->fName, false);
} }
/*! \brief Reinitializes the object to the entry referred to by the specified
path.
\param path the path referring to the entry
\return
- \c B_OK: Everything went fine.
- \c B_BAD_VALUE: \c NULL \a path.
- \c B_ENTRY_NOT_FOUND: The entry could not be found.
- \c B_BUSY: The entry is locked.
*/
status_t status_t
BNode::SetTo(const char *path) BNode::SetTo(const char* path)
{ {
return _SetTo(-1, path, false); return _SetTo(-1, path, false);
} }
/*! \brief Reinitializes the object to the entry referred to by the specified
path rooted in the specified directory.
\param dir the BDirectory, relative to which the entry's path name is
given
\param path the entry's path name relative to \a dir
\return
- \c B_OK: Everything went fine.
- \c B_BAD_VALUE: \c NULL \a dir or \a path.
- \c B_ENTRY_NOT_FOUND: The entry could not be found.
- \c B_BUSY: The entry is locked.
*/
status_t status_t
BNode::SetTo(const BDirectory *dir, const char *path) BNode::SetTo(const BDirectory* dir, const char* path)
{ {
if (!dir || !path || BPrivate::Storage::is_absolute_path(path)) { if (dir == NULL || path == NULL
|| BPrivate::Storage::is_absolute_path(path)) {
Unset(); Unset();
return (fCStatus = B_BAD_VALUE); return (fCStatus = B_BAD_VALUE);
} }
return _SetTo(dir->fDirFd, path, false); return _SetTo(dir->fDirFd, path, false);
} }
/*! \brief Returns the object to an uninitialized state.
*/
void void
BNode::Unset() BNode::Unset()
{ {
@ -268,186 +207,107 @@ BNode::Unset()
fCStatus = B_NO_INIT; fCStatus = B_NO_INIT;
} }
/*! \brief Attains an exclusive lock on the data referred to by this node, so
that it may not be modified by any other objects or methods.
\return
- \c B_OK: Everything went fine.
- \c B_FILE_ERROR: The object is not initialized.
- \c B_BUSY: The node is already locked.
*/
status_t status_t
BNode::Lock() BNode::Lock()
{ {
if (fCStatus != B_OK) if (fCStatus != B_OK)
return fCStatus; return fCStatus;
return _kern_lock_node(fFd); return _kern_lock_node(fFd);
} }
/*! \brief Unlocks the node.
\return
- \c B_OK: Everything went fine.
- \c B_FILE_ERROR: The object is not initialized.
- \c B_BAD_VALUE: The node is not locked.
*/
status_t status_t
BNode::Unlock() BNode::Unlock()
{ {
if (fCStatus != B_OK) if (fCStatus != B_OK)
return fCStatus; return fCStatus;
return _kern_unlock_node(fFd); return _kern_unlock_node(fFd);
} }
/*! \brief Immediately performs any pending disk actions on the node.
\return
- \c B_OK: Everything went fine.
- an error code, if something went wrong.
*/
status_t status_t
BNode::Sync() BNode::Sync()
{ {
return (fCStatus != B_OK) ? B_FILE_ERROR : _kern_fsync(fFd); return (fCStatus != B_OK) ? B_FILE_ERROR : _kern_fsync(fFd);
} }
/*! \brief Writes data from a buffer to an attribute.
Write the \a len bytes of data from \a buffer to
the attribute specified by \a name after erasing any data
that existed previously. The type specified by \a type \em is
remembered, and may be queried with GetAttrInfo(). The value of
\a offset is currently ignored.
\param attr the name of the attribute
\param type the type of the attribute
\param offset the index at which to write the data (currently ignored)
\param buffer the buffer containing the data to be written
\param len the number of bytes to be written
\return
- the number of bytes actually written
- \c B_BAD_VALUE: \c NULL \a attr or \a buffer
- \c B_FILE_ERROR: The object is not initialized or the node it refers to
is read only.
- \c B_NOT_ALLOWED: The node resides on a read only volume.
- \c B_DEVICE_FULL: Insufficient disk space.
- \c B_NO_MEMORY: Insufficient memory to complete the operation.
*/
ssize_t ssize_t
BNode::WriteAttr(const char *attr, type_code type, off_t offset, BNode::WriteAttr(const char* attr, type_code type, off_t offset,
const void *buffer, size_t len) const void* buffer, size_t length)
{ {
if (fCStatus != B_OK) if (fCStatus != B_OK)
return B_FILE_ERROR; return B_FILE_ERROR;
if (!attr || !buffer)
if (attr == NULL || buffer == NULL)
return B_BAD_VALUE; return B_BAD_VALUE;
ssize_t result = fs_write_attr (fFd, attr, type, offset, buffer, len);
return (result < 0 ? errno : result); ssize_t result = fs_write_attr(fFd, attr, type, offset, buffer, length);
return result < 0 ? errno : result;
} }
/*! \brief Reads data from an attribute into a buffer.
Reads the data of the attribute given by \a name into
the buffer specified by \a buffer with length specified
by \a len. \a type and \a offset are currently ignored.
\param attr the name of the attribute
\param type the type of the attribute (currently ignored)
\param offset the index from which to read the data (currently ignored)
\param buffer the buffer for the data to be read
\param len the number of bytes to be read
\return
- the number of bytes actually read
- \c B_BAD_VALUE: \c NULL \a attr or \a buffer
- \c B_FILE_ERROR: The object is not initialized.
- \c B_ENTRY_NOT_FOUND: The node has no attribute \a attr.
*/
ssize_t ssize_t
BNode::ReadAttr(const char *attr, type_code type, off_t offset, BNode::ReadAttr(const char* attr, type_code type, off_t offset,
void *buffer, size_t len) const void* buffer, size_t length) const
{ {
if (fCStatus != B_OK) if (fCStatus != B_OK)
return B_FILE_ERROR; return B_FILE_ERROR;
if (!attr || !buffer)
if (attr == NULL || buffer == NULL)
return B_BAD_VALUE; return B_BAD_VALUE;
ssize_t result = fs_read_attr(fFd, attr, type, offset, buffer, len );
return (result == -1 ? errno : result); ssize_t result = fs_read_attr(fFd, attr, type, offset, buffer, length);
return result == -1 ? errno : result;
} }
/*! \brief Deletes the attribute given by \a name.
\param name the name of the attribute
- \c B_OK: Everything went fine.
- \c B_BAD_VALUE: \c NULL \a name
- \c B_FILE_ERROR: The object is not initialized or the node it refers to
is read only.
- \c B_ENTRY_NOT_FOUND: The node has no attribute \a name.
- \c B_NOT_ALLOWED: The node resides on a read only volume.
*/
status_t status_t
BNode::RemoveAttr(const char *name) BNode::RemoveAttr(const char* name)
{ {
return (fCStatus != B_OK) ? B_FILE_ERROR : _kern_remove_attr(fFd, name); return fCStatus != B_OK ? B_FILE_ERROR : _kern_remove_attr(fFd, name);
} }
/*! \brief Moves the attribute given by \a oldname to \a newname.
If \a newname already exists, the current data is clobbered.
\param oldname the name of the attribute to be renamed
\param newname the new name for the attribute
\return
- \c B_OK: Everything went fine.
- \c B_BAD_VALUE: \c NULL \a oldname or \a newname
- \c B_FILE_ERROR: The object is not initialized or the node it refers to
is read only.
- \c B_ENTRY_NOT_FOUND: The node has no attribute \a oldname.
- \c B_NOT_ALLOWED: The node resides on a read only volume.
*/
status_t status_t
BNode::RenameAttr(const char *oldname, const char *newname) BNode::RenameAttr(const char* oldName, const char* newName)
{ {
if (fCStatus != B_OK) if (fCStatus != B_OK)
return B_FILE_ERROR; return B_FILE_ERROR;
return _kern_rename_attr(fFd, oldname, fFd, newname);
return _kern_rename_attr(fFd, oldName, fFd, newName);
} }
/*! \brief Fills in the pre-allocated attr_info struct pointed to by \a info
with useful information about the attribute specified by \a name.
\param name the name of the attribute
\param info the attr_info structure to be filled in
\return
- \c B_OK: Everything went fine.
- \c B_BAD_VALUE: \c NULL \a name
- \c B_FILE_ERROR: The object is not initialized.
- \c B_ENTRY_NOT_FOUND: The node has no attribute \a name.
*/
status_t status_t
BNode::GetAttrInfo(const char *name, struct attr_info *info) const BNode::GetAttrInfo(const char* name, struct attr_info* info) const
{ {
if (fCStatus != B_OK) if (fCStatus != B_OK)
return B_FILE_ERROR; return B_FILE_ERROR;
if (!name || !info)
if (name == NULL || info == NULL)
return B_BAD_VALUE; return B_BAD_VALUE;
return (fs_stat_attr(fFd, name, info) < 0) ? errno : B_OK ;
return fs_stat_attr(fFd, name, info) < 0 ? errno : B_OK ;
} }
/*! \brief Returns the next attribute in the node's list of attributes.
Every BNode maintains a pointer to its list of attributes.
GetNextAttrName() retrieves the name of the attribute that the pointer is
currently pointing to, and then bumps the pointer to the next attribute.
The name is copied into the buffer, which should be at least
B_ATTR_NAME_LENGTH characters long. The copied name is NULL-terminated.
When you've asked for every name in the list, GetNextAttrName()
returns \c B_ENTRY_NOT_FOUND.
\param buffer the buffer the name of the next attribute shall be stored in
(must be at least \c B_ATTR_NAME_LENGTH bytes long)
\return
- \c B_OK: Everything went fine.
- \c B_BAD_VALUE: \c NULL \a buffer.
- \c B_FILE_ERROR: The object is not initialized.
- \c B_ENTRY_NOT_FOUND: There are no more attributes, the last attribute
name has already been returned.
*/
status_t status_t
BNode::GetNextAttrName(char *buffer) BNode::GetNextAttrName(char* buffer)
{ {
// We're allowed to assume buffer is at least // We're allowed to assume buffer is at least
// B_ATTR_NAME_LENGTH chars long, but NULLs // B_ATTR_NAME_LENGTH chars long, but NULLs
// are not acceptable. // are not acceptable.
// BeOS R5 crashed when passed NULL
if (buffer == NULL) if (buffer == NULL)
return B_BAD_VALUE; // /new R5 crashed when passed NULL return B_BAD_VALUE;
if (InitAttrDir() != B_OK) if (InitAttrDir() != B_OK)
return B_FILE_ERROR; return B_FILE_ERROR;
@ -455,66 +315,46 @@ BNode::GetNextAttrName(char *buffer)
ssize_t result = _kern_read_dir(fAttrFd, &entry, sizeof(entry), 1); ssize_t result = _kern_read_dir(fAttrFd, &entry, sizeof(entry), 1);
if (result < 0) if (result < 0)
return result; return result;
if (result == 0) if (result == 0)
return B_ENTRY_NOT_FOUND; return B_ENTRY_NOT_FOUND;
strlcpy(buffer, entry.d_name, B_ATTR_NAME_LENGTH); strlcpy(buffer, entry.d_name, B_ATTR_NAME_LENGTH);
return B_OK; return B_OK;
} }
/*! \brief Resets the object's attribute pointer to the first attribute in the
list.
\return
- \c B_OK: Everything went fine.
- \c B_FILE_ERROR: Some error occured.
*/
status_t status_t
BNode::RewindAttrs() BNode::RewindAttrs()
{ {
if (InitAttrDir() != B_OK) if (InitAttrDir() != B_OK)
return B_FILE_ERROR; return B_FILE_ERROR;
return _kern_rewind_dir(fAttrFd); return _kern_rewind_dir(fAttrFd);
} }
/*! Writes the specified string to the specified attribute, clobbering any
previous data.
\param name the name of the attribute
\param data the BString to be written to the attribute
- \c B_OK: Everything went fine.
- \c B_BAD_VALUE: \c NULL \a name or \a data
- \c B_FILE_ERROR: The object is not initialized or the node it refers to
is read only.
- \c B_NOT_ALLOWED: The node resides on a read only volume.
- \c B_DEVICE_FULL: Insufficient disk space.
- \c B_NO_MEMORY: Insufficient memory to complete the operation.
*/
status_t status_t
BNode::WriteAttrString(const char *name, const BString *data) BNode::WriteAttrString(const char* name, const BString* data)
{ {
status_t error = (!name || !data) ? B_BAD_VALUE : B_OK; status_t error = (!name || !data) ? B_BAD_VALUE : B_OK;
if (error == B_OK) { if (error == B_OK) {
int32 len = data->Length() + 1; int32 length = data->Length() + 1;
ssize_t sizeWritten = WriteAttr(name, B_STRING_TYPE, 0, data->String(), ssize_t sizeWritten = WriteAttr(name, B_STRING_TYPE, 0, data->String(),
len); length);
if (sizeWritten != len) if (sizeWritten != length)
error = sizeWritten; error = sizeWritten;
} }
return error; return error;
} }
/*! \brief Reads the data of the specified attribute into the pre-allocated
\a result.
\param name the name of the attribute
\param result the BString to be set to the value of the attribute
\return
- \c B_OK: Everything went fine.
- \c B_BAD_VALUE: \c NULL \a name or \a result
- \c B_FILE_ERROR: The object is not initialized.
- \c B_ENTRY_NOT_FOUND: The node has no attribute \a attr.
*/
status_t status_t
BNode::ReadAttrString(const char *name, BString *result) const BNode::ReadAttrString(const char* name, BString* result) const
{ {
if (!name || !result) if (name == NULL || result == NULL)
return B_BAD_VALUE; return B_BAD_VALUE;
attr_info info; attr_info info;
@ -523,89 +363,85 @@ BNode::ReadAttrString(const char *name, BString *result) const
error = GetAttrInfo(name, &info); error = GetAttrInfo(name, &info);
if (error != B_OK) if (error != B_OK)
return error; return error;
// Lock the string's buffer so we can meddle with it // Lock the string's buffer so we can meddle with it
char *data = result->LockBuffer(info.size+1); char* data = result->LockBuffer(info.size + 1);
if (!data) if (data == NULL)
return B_NO_MEMORY; return B_NO_MEMORY;
// Read the attribute // Read the attribute
ssize_t bytes = ReadAttr(name, B_STRING_TYPE, 0, data, info.size); ssize_t bytes = ReadAttr(name, B_STRING_TYPE, 0, data, info.size);
// Check for failure // Check for failure
if (bytes < 0) { if (bytes < 0) {
error = bytes; error = bytes;
bytes = 0; // In this instance, we simply clear the string bytes = 0;
// In this instance, we simply clear the string
} else } else
error = B_OK; error = B_OK;
// Null terminate the new string just to be sure (since it *is* // Null terminate the new string just to be sure (since it *is*
// possible to read and write non-NULL-terminated strings) // possible to read and write non-NULL-terminated strings)
data[bytes] = 0; data[bytes] = 0;
result->UnlockBuffer(); result->UnlockBuffer();
return error; return error;
} }
/*! \brief Reinitializes the object as a copy of the \a node.
\param node the BNode to be copied
\return a reference to this BNode object.
*/
BNode& BNode&
BNode::operator=(const BNode &node) BNode::operator=(const BNode& node)
{ {
// No need to do any assignment if already equal // No need to do any assignment if already equal
if (*this == node) if (*this == node)
return *this; return *this;
// Close down out current state // Close down out current state
Unset(); Unset();
// We have to manually dup the node, because R5::BNode::Dup() // We have to manually dup the node, because R5::BNode::Dup()
// is not declared to be const (which IMO is retarded). // is not declared to be const (which IMO is retarded).
fFd = _kern_dup(node.fFd); fFd = _kern_dup(node.fFd);
fCStatus = (fFd < 0) ? B_NO_INIT : B_OK ; fCStatus = (fFd < 0) ? B_NO_INIT : B_OK ;
return *this; return *this;
} }
/*! Tests whether this and the supplied BNode object are equal.
Two BNode objects are said to be equal if they're set to the same node,
or if they're both \c B_NO_INIT.
\param node the BNode to be compared with
\return \c true, if the BNode objects are equal, \c false otherwise
*/
bool bool
BNode::operator==(const BNode &node) const BNode::operator==(const BNode& node) const
{ {
if (fCStatus == B_NO_INIT && node.InitCheck() == B_NO_INIT) if (fCStatus == B_NO_INIT && node.InitCheck() == B_NO_INIT)
return true; return true;
if (fCStatus == B_OK && node.InitCheck() == B_OK) { if (fCStatus == B_OK && node.InitCheck() == B_OK) {
// compare the node_refs // compare the node_refs
node_ref ref1, ref2; node_ref ref1, ref2;
if (GetNodeRef(&ref1) != B_OK) if (GetNodeRef(&ref1) != B_OK)
return false; return false;
if (node.GetNodeRef(&ref2) != B_OK) if (node.GetNodeRef(&ref2) != B_OK)
return false; return false;
return (ref1 == ref2); return (ref1 == ref2);
} }
return false; return false;
} }
/*! Tests whether this and the supplied BNode object are not equal.
Two BNode objects are said to be equal if they're set to the same node,
or if they're both \c B_NO_INIT.
\param node the BNode to be compared with
\return \c false, if the BNode objects are equal, \c true otherwise
*/
bool bool
BNode::operator!=(const BNode &node) const BNode::operator!=(const BNode& node) const
{ {
return !(*this == node); return !(*this == node);
} }
/*! \brief Returns a POSIX file descriptor to the node this object refers to.
Remember to call close() on the file descriptor when you're through with
it.
\return a valid file descriptor, or -1, if something went wrong.
*/
int int
BNode::Dup() BNode::Dup()
{ {
int fd = _kern_dup(fFd); int fd = _kern_dup(fFd);
return (fd >= 0 ? fd : -1); // comply with R5 return value
return (fd >= 0 ? fd : -1);
// comply with R5 return value
} }
@ -617,27 +453,37 @@ void BNode::_RudeNode4() { }
void BNode::_RudeNode5() { } void BNode::_RudeNode5() { }
void BNode::_RudeNode6() { } void BNode::_RudeNode6() { }
/*! \brief Sets the node's file descriptor.
/*! Sets the node's file descriptor.
Used by each implementation (i.e. BNode, BFile, BDirectory, etc.) to set Used by each implementation (i.e. BNode, BFile, BDirectory, etc.) to set
the node's file descriptor. This allows each subclass to use the various the node's file descriptor. This allows each subclass to use the various
file-type specific system calls for opening file descriptors. file-type specific system calls for opening file descriptors.
\param fd the file descriptor this BNode should be set to (may be -1)
\return \c B_OK, if everything went fine, an error code otherwise.
\note This method calls close_fd() to close previously opened FDs. Thus \note This method calls close_fd() to close previously opened FDs. Thus
derived classes should take care to first call set_fd() and set derived classes should take care to first call set_fd() and set
class specific resources freed in their close_fd() version class specific resources freed in their close_fd() version
thereafter. thereafter.
\param fd the file descriptor this BNode should be set to (may be -1).
\returns \c B_OK if everything went fine, or an error code if something
went wrong.
*/ */
status_t status_t
BNode::set_fd(int fd) BNode::set_fd(int fd)
{ {
if (fFd != -1) if (fFd != -1)
close_fd(); close_fd();
fFd = fd; fFd = fd;
return B_OK; return B_OK;
} }
/*! \brief Closes the node's file descriptor(s).
/*! Closes the node's file descriptor(s).
To be implemented by subclasses to close the file descriptor using the To be implemented by subclasses to close the file descriptor using the
proper system call for the given file-type. This implementation calls proper system call for the given file-type. This implementation calls
_kern_close(fFd) and also _kern_close(fAttrDir) if necessary. _kern_close(fFd) and also _kern_close(fAttrDir) if necessary.
@ -645,8 +491,7 @@ BNode::set_fd(int fd)
void void
BNode::close_fd() BNode::close_fd()
{ {
if (fAttrFd >= 0) if (fAttrFd >= 0) {
{
_kern_close(fAttrFd); _kern_close(fAttrFd);
fAttrFd = -1; fAttrFd = -1;
} }
@ -656,10 +501,12 @@ BNode::close_fd()
} }
} }
// set_status
/*! \brief Sets the BNode's status. /*! Sets the BNode's status.
To be used by derived classes instead of accessing the BNode's private To be used by derived classes instead of accessing the BNode's private
\c fCStatus member directly. \c fCStatus member directly.
\param newStatus the new value for the status variable. \param newStatus the new value for the status variable.
*/ */
void void
@ -668,8 +515,8 @@ BNode::set_status(status_t newStatus)
fCStatus = newStatus; fCStatus = newStatus;
} }
// _SetTo
/*! \brief Initializes the BNode's file descriptor to the node referred to /*! Initializes the BNode's file descriptor to the node referred to
by the given FD and path combo. by the given FD and path combo.
\a path must either be \c NULL, an absolute or a relative path. \a path must either be \c NULL, an absolute or a relative path.
@ -691,30 +538,31 @@ BNode::set_status(status_t newStatus)
current working directory. current working directory.
\param traverse If the node identified by \a fd and \a path is a symlink \param traverse If the node identified by \a fd and \a path is a symlink
and \a traverse is \c true, the symlink will be resolved recursively. and \a traverse is \c true, the symlink will be resolved recursively.
\return \c B_OK, if everything went fine, another error code otherwise.
\returns \c B_OK if everything went fine, or an error code otherwise.
*/ */
status_t status_t
BNode::_SetTo(int fd, const char *path, bool traverse) BNode::_SetTo(int fd, const char* path, bool traverse)
{ {
Unset(); Unset();
status_t error = (fd >= 0 || path ? B_OK : B_BAD_VALUE); status_t error = (fd >= 0 || path ? B_OK : B_BAD_VALUE);
if (error == B_OK) { if (error == B_OK) {
int traverseFlag = (traverse ? 0 : O_NOTRAVERSE); int traverseFlag = (traverse ? 0 : O_NOTRAVERSE);
fFd = _kern_open(fd, path, O_RDWR | traverseFlag, 0); fFd = _kern_open(fd, path, O_RDWR | O_CLOEXEC | traverseFlag, 0);
if (fFd < B_OK && fFd != B_ENTRY_NOT_FOUND) { if (fFd < B_OK && fFd != B_ENTRY_NOT_FOUND) {
// opening read-write failed, re-try read-only // opening read-write failed, re-try read-only
fFd = _kern_open(fd, path, O_RDONLY | traverseFlag, 0); fFd = _kern_open(fd, path, O_RDONLY | O_CLOEXEC | traverseFlag, 0);
} }
if (fFd < 0) if (fFd < 0)
error = fFd; error = fFd;
else
fcntl(fFd, F_SETFD, FD_CLOEXEC);
} }
return fCStatus = error; return fCStatus = error;
} }
// _SetTo
/*! \brief Initializes the BNode's file descriptor to the node referred to /*! Initializes the BNode's file descriptor to the node referred to
by the given entry_ref. by the given entry_ref.
The method will first try to open the node with read and write permission. The method will first try to open the node with read and write permission.
@ -724,52 +572,61 @@ BNode::_SetTo(int fd, const char *path, bool traverse)
The \a fCStatus member will be set to the return value of this method. The \a fCStatus member will be set to the return value of this method.
\param ref An entry_ref identifying the node to be opened. \param ref An entry_ref identifying the node to be opened.
\param traverse If the node identified by \a ref is a symlink \param traverse If the node identified by \a ref is a symlink and
and \a traverse is \c true, the symlink will be resolved recursively. \a traverse is \c true, the symlink will be resolved recursively.
\return \c B_OK, if everything went fine, another error code otherwise.
\returns \c B_OK if everything went fine, or an error code otherwise.
*/ */
status_t status_t
BNode::_SetTo(const entry_ref *ref, bool traverse) BNode::_SetTo(const entry_ref* ref, bool traverse)
{ {
Unset(); Unset();
status_t error = (ref ? B_OK : B_BAD_VALUE);
if (error == B_OK) { status_t result = (ref ? B_OK : B_BAD_VALUE);
if (result == B_OK) {
int traverseFlag = (traverse ? 0 : O_NOTRAVERSE); int traverseFlag = (traverse ? 0 : O_NOTRAVERSE);
fFd = _kern_open_entry_ref(ref->device, ref->directory, ref->name, fFd = _kern_open_entry_ref(ref->device, ref->directory, ref->name,
O_RDWR | traverseFlag, 0); O_RDWR | O_CLOEXEC | traverseFlag, 0);
if (fFd < B_OK && fFd != B_ENTRY_NOT_FOUND) { if (fFd < B_OK && fFd != B_ENTRY_NOT_FOUND) {
// opening read-write failed, re-try read-only // opening read-write failed, re-try read-only
fFd = _kern_open_entry_ref(ref->device, ref->directory, ref->name, fFd = _kern_open_entry_ref(ref->device, ref->directory, ref->name,
O_RDONLY | traverseFlag, 0); O_RDONLY | O_CLOEXEC | traverseFlag, 0);
} }
if (fFd < 0) if (fFd < 0)
error = fFd; result = fFd;
else
fcntl(fFd, F_SETFD, FD_CLOEXEC);
} }
return fCStatus = error;
return fCStatus = result;
} }
/*! \brief Modifies a certain setting for this node based on \a what and the
/*! Modifies a certain setting for this node based on \a what and the
corresponding value in \a st. corresponding value in \a st.
Inherited from and called by BStatable. Inherited from and called by BStatable.
\param st a stat structure containing the value to be set
\param what specifies what setting to be modified \param st a stat structure containing the value to be set.
\return \c B_OK if everything went fine, an error code otherwise. \param what specifies what setting to be modified.
\returns \c B_OK if everything went fine, or an error code otherwise.
*/ */
status_t status_t
BNode::set_stat(struct stat &st, uint32 what) BNode::set_stat(struct stat& stat, uint32 what)
{ {
if (fCStatus != B_OK) if (fCStatus != B_OK)
return B_FILE_ERROR; return B_FILE_ERROR;
return _kern_write_stat(fFd, NULL, false, &st, sizeof(struct stat),
return _kern_write_stat(fFd, NULL, false, &stat, sizeof(struct stat),
what); what);
} }
/*! \brief Verifies that the BNode has been properly initialized, and then
/*! Verifies that the BNode has been properly initialized, and then
(if necessary) opens the attribute directory on the node's file (if necessary) opens the attribute directory on the node's file
descriptor, storing it in fAttrDir. descriptor, storing it in fAttrDir.
\return \c B_OK if everything went fine, an error code otherwise.
\returns \c B_OK if everything went fine, or an error code otherwise.
*/ */
status_t status_t
BNode::InitAttrDir() BNode::InitAttrDir()
@ -782,18 +639,15 @@ BNode::InitAttrDir()
// set close on exec flag // set close on exec flag
fcntl(fAttrFd, F_SETFD, FD_CLOEXEC); fcntl(fAttrFd, F_SETFD, FD_CLOEXEC);
} }
return fCStatus; return fCStatus;
} }
/*! \var BNode::fFd
File descriptor for the given node.
*/
/*! \var BNode::fAttrFd
File descriptor for the attribute directory of the node. Initialized lazily.
*/
/*! \var BNode::fCStatus
The object's initialization status.
*/
status_t
BNode::GetStat(struct stat* stat) const
{
return fCStatus != B_OK
? fCStatus
: _kern_read_stat(fFd, NULL, false, stat, sizeof(struct stat));
}