mirror of
https://review.haiku-os.org/haiku
synced 2025-02-02 19:57:42 +01:00
98833b15bf
git-svn-id: file:///srv/svn/repos/haiku/trunk/current@3475 a95241bf-73f2-0310-859d-f6bbb57e9c96
722 lines
18 KiB
C++
722 lines
18 KiB
C++
//----------------------------------------------------------------------
|
|
// This software is part of the OpenBeOS distribution and is covered
|
|
// by the OpenBeOS license.
|
|
//---------------------------------------------------------------------
|
|
/*!
|
|
\file Path.cpp
|
|
BPath implementation.
|
|
*/
|
|
|
|
#include <new>
|
|
|
|
#include <Path.h>
|
|
#include <Directory.h>
|
|
#include <Entry.h>
|
|
#include <StorageDefs.h>
|
|
#include <String.h>
|
|
#include "kernel_interface.h"
|
|
#include "storage_support.h"
|
|
|
|
using namespace std;
|
|
|
|
#ifdef USE_OPENBEOS_NAMESPACE
|
|
using namespace OpenBeOS;
|
|
#endif
|
|
|
|
//! Creates an uninitialized BPath object.
|
|
BPath::BPath()
|
|
: fName(NULL),
|
|
fCStatus(B_NO_INIT)
|
|
{
|
|
}
|
|
|
|
//! Creates a copy of the given BPath object.
|
|
/*! \param path the object to be copied
|
|
*/
|
|
BPath::BPath(const BPath &path)
|
|
: fName(NULL),
|
|
fCStatus(B_NO_INIT)
|
|
{
|
|
*this = path;
|
|
}
|
|
|
|
/*! \brief Creates a BPath object and initializes it to the filesystem entry
|
|
specified by the given entry_ref struct.
|
|
\param ref the entry_ref
|
|
*/
|
|
BPath::BPath(const entry_ref *ref)
|
|
: fName(NULL),
|
|
fCStatus(B_NO_INIT)
|
|
{
|
|
SetTo(ref);
|
|
}
|
|
|
|
/*! Creates a BPath object and initializes it to the filesystem entry
|
|
specified by the given BEntry object.
|
|
\param entry the BEntry object
|
|
*/
|
|
BPath::BPath(const BEntry *entry)
|
|
: fName(NULL),
|
|
fCStatus(B_NO_INIT)
|
|
{
|
|
SetTo(entry);
|
|
}
|
|
|
|
/*! \brief Creates a BPath object and initializes it to the specified path or
|
|
path and filename combination.
|
|
\param dir The base component of the pathname. May be absolute or relative.
|
|
If relative, it is reckoned off the current working directory.
|
|
\param leaf The (optional) leaf component of the pathname. Must be
|
|
relative. The value of leaf is concatenated to the end of \a dir
|
|
(a "/" will be added as a separator, if necessary).
|
|
\param normalize boolean flag used to force normalization; normalization
|
|
may occur even if false (see \ref MustNormalize).
|
|
*/
|
|
BPath::BPath(const char *dir, const char *leaf, bool normalize)
|
|
: fName(NULL),
|
|
fCStatus(B_NO_INIT)
|
|
{
|
|
SetTo(dir, leaf, normalize);
|
|
}
|
|
|
|
/*! \brief Creates a BPath object and initializes it to the specified directory
|
|
and filename combination.
|
|
\param dir Refers to the directory that provides the base component of the
|
|
pathname.
|
|
\param leaf The (optional) leaf component of the pathname. Must be
|
|
relative. The value of leaf is concatenated to the end of \a dir
|
|
(a "/" will be added as a separator, if necessary).
|
|
\param normalize boolean flag used to force normalization; normalization
|
|
may occur even if false (see \ref MustNormalize).
|
|
*/
|
|
BPath::BPath(const BDirectory *dir, const char *leaf, bool normalize)
|
|
: fName(NULL),
|
|
fCStatus(B_NO_INIT)
|
|
{
|
|
SetTo(dir, leaf, normalize);
|
|
}
|
|
|
|
//! Destroys the BPath object and frees any of its associated resources.
|
|
BPath::~BPath()
|
|
{
|
|
Unset();
|
|
}
|
|
|
|
//! Returns the status of the most recent construction or SetTo() call.
|
|
/*! \return \c B_OK, if the BPath object is properly initialized, an error
|
|
code otherwise.
|
|
*/
|
|
status_t
|
|
BPath::InitCheck() const
|
|
{
|
|
return fCStatus;
|
|
}
|
|
|
|
/*! \brief Reinitializes the object to the filesystem entry specified by the
|
|
given entry_ref struct.
|
|
\param ref the entry_ref
|
|
\return
|
|
- \c B_OK: The initialization was successful.
|
|
- \c B_BAD_VALUE: \c NULL \a ref.
|
|
- \c B_NAME_TOO_LONG: The pathname is longer than \c B_PATH_NAME_LENGTH.
|
|
- other error codes.
|
|
*/
|
|
status_t
|
|
BPath::SetTo(const entry_ref *ref)
|
|
{
|
|
Unset();
|
|
status_t error = (ref ? B_OK : B_BAD_VALUE);
|
|
if (error == B_OK) {
|
|
char path[B_PATH_NAME_LENGTH];
|
|
error = BPrivate::Storage::entry_ref_to_path(ref, path, sizeof(path));
|
|
if (error == B_OK)
|
|
error = set_path(path); // the path is already normalized
|
|
}
|
|
fCStatus = error;
|
|
return error;
|
|
}
|
|
|
|
/*! \brief Reinitializes the object to the specified filesystem entry.
|
|
\param entry the BEntry
|
|
\return
|
|
- \c B_OK: The initialization was successful.
|
|
- \c B_BAD_VALUE: \c NULL \a entry.
|
|
- \c B_NAME_TOO_LONG: The pathname is longer than \c B_PATH_NAME_LENGTH.
|
|
- other error codes.
|
|
*/
|
|
status_t
|
|
BPath::SetTo(const BEntry *entry)
|
|
{
|
|
Unset();
|
|
status_t error = (entry ? B_OK : B_BAD_VALUE);
|
|
entry_ref ref;
|
|
if (error == B_OK)
|
|
error = entry->GetRef(&ref);
|
|
if (error == B_OK)
|
|
error = SetTo(&ref);
|
|
fCStatus = error;
|
|
return error;
|
|
}
|
|
|
|
/*! \brief Reinitializes the object to the specified path or path and file
|
|
name combination.
|
|
\param path the path name
|
|
\param leaf the leaf name (may be \c NULL)
|
|
\param normalize boolean flag used to force normalization; normalization
|
|
may occur even if false (see \ref MustNormalize).
|
|
\return
|
|
- \c B_OK: The initialization was successful.
|
|
- \c B_BAD_VALUE: \c NULL \a path or absolute \a leaf.
|
|
- \c B_NAME_TOO_LONG: The pathname is longer than \c B_PATH_NAME_LENGTH.
|
|
- other error codes.
|
|
\note \code path.SetTo(path.Path(), "new leaf") \endcode is safe.
|
|
*/
|
|
status_t
|
|
BPath::SetTo(const char *path, const char *leaf, bool normalize)
|
|
{
|
|
status_t error = (path ? B_OK : B_BAD_VALUE);
|
|
if (error == B_OK && leaf && BPrivate::Storage::is_absolute_path(leaf))
|
|
error = B_BAD_VALUE;
|
|
char newPath[B_PATH_NAME_LENGTH];
|
|
if (error == B_OK) {
|
|
// we always normalize relative paths
|
|
normalize |= !BPrivate::Storage::is_absolute_path(path);
|
|
// build a new path from path and leaf
|
|
// copy path first
|
|
uint32 pathLen = strlen(path);
|
|
if (pathLen >= sizeof(newPath))
|
|
error = B_NAME_TOO_LONG;
|
|
if (error == B_OK)
|
|
strcpy(newPath, path);
|
|
// append leaf, if supplied
|
|
if (error == B_OK && leaf) {
|
|
bool needsSeparator = (pathLen > 0 && path[pathLen - 1] != '/');
|
|
uint32 wholeLen = pathLen + (needsSeparator ? 1 : 0)
|
|
+ strlen(leaf);
|
|
if (wholeLen >= sizeof(newPath))
|
|
error = B_NAME_TOO_LONG;
|
|
if (error == B_OK) {
|
|
if (needsSeparator) {
|
|
newPath[pathLen] = '/';
|
|
pathLen++;
|
|
}
|
|
strcpy(newPath + pathLen, leaf);
|
|
}
|
|
}
|
|
// check, if necessary to normalize
|
|
if (error == B_OK && !normalize) {
|
|
try {
|
|
normalize = normalize || MustNormalize(newPath);
|
|
} catch (BPath::EBadInput) {
|
|
error = B_BAD_VALUE;
|
|
}
|
|
}
|
|
// normalize the path, if necessary, otherwise just set it
|
|
if (error == B_OK) {
|
|
if (normalize) {
|
|
char normalizedPath[B_PATH_NAME_LENGTH];
|
|
error = BPrivate::Storage::get_canonical_path(newPath, normalizedPath,
|
|
sizeof(normalizedPath));
|
|
if (error == B_OK)
|
|
error = set_path(normalizedPath);
|
|
} else
|
|
error = set_path(newPath);
|
|
}
|
|
}
|
|
// cleanup, if something went wrong
|
|
if (error != B_OK)
|
|
Unset();
|
|
fCStatus = error;
|
|
return error;
|
|
}
|
|
|
|
/*! \brief Reinitializes the object to the specified directory and relative
|
|
path combination.
|
|
\param dir Refers to the directory that provides the base component of the
|
|
pathname.
|
|
\param path the relative path name (may be \c NULL)
|
|
\param normalize boolean flag used to force normalization; normalization
|
|
may occur even if false (see \ref MustNormalize).
|
|
\return
|
|
- \c B_OK: The initialization was successful.
|
|
- \c B_BAD_VALUE: \c NULL \a dir or absolute \a path.
|
|
- \c B_NAME_TOO_LONG: The pathname is longer than \c B_PATH_NAME_LENGTH.
|
|
- other error codes.
|
|
*/
|
|
status_t
|
|
BPath::SetTo(const BDirectory *dir, const char *path, bool normalize)
|
|
{
|
|
status_t error = (dir && dir->InitCheck() == B_OK ? B_OK : B_BAD_VALUE);
|
|
// get the path of the BDirectory
|
|
BEntry entry;
|
|
if (error == B_OK)
|
|
error = dir->GetEntry(&entry);
|
|
BPath dirPath;
|
|
if (error == B_OK)
|
|
error = dirPath.SetTo(&entry);
|
|
// let the other version do the work
|
|
if (error == B_OK)
|
|
error = SetTo(dirPath.Path(), path, normalize);
|
|
if (error != B_OK)
|
|
Unset();
|
|
fCStatus = error;
|
|
return error;
|
|
}
|
|
|
|
/*! \brief Returns the object to an uninitialized state. The object frees any
|
|
resources it allocated and marks itself as uninitialized.
|
|
*/
|
|
void
|
|
BPath::Unset()
|
|
{
|
|
set_path(NULL);
|
|
fCStatus = B_NO_INIT;
|
|
}
|
|
|
|
/*! \brief Appends the given (relative) path to the end of the current path.
|
|
This call fails if the path is absolute or the object to which you're
|
|
appending is uninitialized.
|
|
\param path relative pathname to append to current path (may be \c NULL).
|
|
\param normalize boolean flag used to force normalization; normalization
|
|
may occur even if false (see \ref MustNormalize).
|
|
\return
|
|
- \c B_OK: The initialization was successful.
|
|
- \c B_BAD_VALUE: The object is not properly initialized.
|
|
- \c B_NAME_TOO_LONG: The pathname is longer than \c B_PATH_NAME_LENGTH.
|
|
- other error codes.
|
|
*/
|
|
status_t
|
|
BPath::Append(const char *path, bool normalize)
|
|
{
|
|
status_t error = (InitCheck() == B_OK ? B_OK : B_BAD_VALUE);
|
|
if (error == B_OK)
|
|
error = SetTo(Path(), path, normalize);
|
|
if (error != B_OK)
|
|
Unset();
|
|
fCStatus = error;
|
|
return error;
|
|
}
|
|
|
|
//! Returns the object's complete path name.
|
|
/*! \return
|
|
- the object's path name, or
|
|
- \c NULL, if it is not properly initialized.
|
|
*/
|
|
const char *
|
|
BPath::Path() const
|
|
{
|
|
return fName;
|
|
}
|
|
|
|
//! Returns the leaf portion of the object's path name.
|
|
/*! The leaf portion is defined as the string after the last \c '/'. For
|
|
the root path (\c "/") it is the empty string (\c "").
|
|
\return
|
|
- the leaf portion of the object's path name, or
|
|
- \c NULL, if it is not properly initialized.
|
|
*/
|
|
const char *
|
|
BPath::Leaf() const
|
|
{
|
|
const char *result = NULL;
|
|
if (InitCheck() == B_OK) {
|
|
result = fName + strlen(fName);
|
|
// There should be no need for the second condition, since we deal
|
|
// with absolute paths only and those contain at least one '/'.
|
|
// However, it doesn't harm.
|
|
while (*result != '/' && result > fName)
|
|
result--;
|
|
result++;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/*! \brief Calls the argument's SetTo() method with the name of the
|
|
object's parent directory.
|
|
No normalization is done.
|
|
\param path the BPath object to be initialized to the parent directory's
|
|
path name.
|
|
\return
|
|
- \c B_OK: Everything went fine.
|
|
- \c B_BAD_VALUE: \c NULL \a path.
|
|
- \c B_ENTRY_NOT_FOUND: The object represents \c "/".
|
|
- other error code returned by SetTo().
|
|
*/
|
|
status_t
|
|
BPath::GetParent(BPath *path) const
|
|
{
|
|
status_t error = (path ? B_OK : B_BAD_VALUE);
|
|
if (error == B_OK)
|
|
error = InitCheck();
|
|
if (error == B_OK) {
|
|
int32 len = strlen(fName);
|
|
if (len == 1) // handle "/"
|
|
error = B_ENTRY_NOT_FOUND;
|
|
else {
|
|
char parentPath[B_PATH_NAME_LENGTH];
|
|
len--;
|
|
while (fName[len] != '/' && len > 0)
|
|
len--;
|
|
if (len == 0) // parent dir is "/"
|
|
len++;
|
|
memcpy(parentPath, fName, len);
|
|
parentPath[len] = 0;
|
|
error = path->SetTo(parentPath);
|
|
}
|
|
}
|
|
return error;
|
|
}
|
|
|
|
//! Performs a simple (string-wise) comparison of paths.
|
|
/*! No normalization takes place! Uninitialized BPath objects are considered
|
|
to be equal.
|
|
\param item the BPath object to be compared with
|
|
\return \c true, if the path names are equal, \c false otherwise.
|
|
*/
|
|
bool
|
|
BPath::operator==(const BPath &item) const
|
|
{
|
|
return (*this == item.Path());
|
|
}
|
|
|
|
//! Performs a simple (string-wise) comparison of paths.
|
|
/*! No normalization takes place!
|
|
\param path the path name to be compared with
|
|
\return \c true, if the path names are equal, \c false otherwise.
|
|
*/
|
|
bool
|
|
BPath::operator==(const char *path) const
|
|
{
|
|
return (InitCheck() != B_OK && path == NULL
|
|
|| fName && path && strcmp(fName, path) == 0);
|
|
}
|
|
|
|
//! Performs a simple (string-wise) comparison of paths.
|
|
/*! No normalization takes place! Uninitialized BPath objects are considered
|
|
to be equal.
|
|
\param item the BPath object to be compared with
|
|
\return \c true, if the path names are not equal, \c false otherwise.
|
|
*/
|
|
bool
|
|
BPath::operator!=(const BPath &item) const
|
|
{
|
|
return !(*this == item);
|
|
}
|
|
|
|
//! Performs a simple (string-wise) comparison of paths.
|
|
/*! No normalization takes place!
|
|
\param path the path name to be compared with
|
|
\return \c true, if the path names are not equal, \c false otherwise.
|
|
*/
|
|
bool
|
|
BPath::operator!=(const char *path) const
|
|
{
|
|
return !(*this == path);
|
|
}
|
|
|
|
//! Initializes the object to be a copy of the argument.
|
|
/*! \param item the BPath object to be copied
|
|
\return \c *this
|
|
*/
|
|
BPath&
|
|
BPath::operator=(const BPath &item)
|
|
{
|
|
if (this != &item)
|
|
*this = item.Path();
|
|
return *this;
|
|
}
|
|
|
|
//! Initializes the object to be a copy of the argument.
|
|
/*! Has the same effect as \code SetTo(path) \endcode.
|
|
\param path the path name to be assigned to this object
|
|
\return \c *this
|
|
*/
|
|
BPath&
|
|
BPath::operator=(const char *path)
|
|
{
|
|
if (path == NULL)
|
|
Unset();
|
|
else
|
|
SetTo(path);
|
|
return *this;
|
|
}
|
|
|
|
|
|
// BFlattenable functionality
|
|
|
|
// that's the layout of a flattened entry_ref
|
|
struct flattened_entry_ref {
|
|
dev_t device;
|
|
ino_t directory;
|
|
char name[1];
|
|
};
|
|
|
|
// base size of a flattened entry ref
|
|
static const size_t flattened_entry_ref_size
|
|
= sizeof(dev_t) + sizeof(ino_t);
|
|
|
|
|
|
//! Returns \c false.
|
|
/*! Implements BFlattenable.
|
|
\return \c false
|
|
*/
|
|
bool
|
|
BPath::IsFixedSize() const
|
|
{
|
|
return false;
|
|
}
|
|
|
|
//! Returns \c B_REF_TYPE.
|
|
/*! Implements BFlattenable.
|
|
\return \c B_REF_TYPE
|
|
*/
|
|
type_code
|
|
BPath::TypeCode() const
|
|
{
|
|
return B_REF_TYPE;
|
|
}
|
|
|
|
/*! \brief Returns the size of the flattened entry_ref structure that
|
|
represents the pathname.
|
|
Implements BFlattenable.
|
|
\return the size needed for flattening.
|
|
*/
|
|
ssize_t
|
|
BPath::FlattenedSize() const
|
|
{
|
|
ssize_t size = flattened_entry_ref_size;
|
|
BEntry entry;
|
|
entry_ref ref;
|
|
if (InitCheck() == B_OK
|
|
&& entry.SetTo(Path()) == B_OK
|
|
&& entry.GetRef(&ref) == B_OK) {
|
|
size += strlen(ref.name) + 1;
|
|
}
|
|
return size;
|
|
}
|
|
|
|
/*! \brief Converts the object's pathname to an entry_ref and writes it into
|
|
buffer.
|
|
Implements BFlattenable.
|
|
\param buffer the buffer the data shall be stored in
|
|
\param size the size of \a buffer
|
|
\return
|
|
- \c B_OK: Everything went fine.
|
|
- \c B_BAD_VALUE: \c NULL buffer or the buffer is of insufficient size.
|
|
- other error codes.
|
|
\todo Reimplement for performance reasons: Don't call FlattenedSize().
|
|
*/
|
|
status_t
|
|
BPath::Flatten(void *buffer, ssize_t size) const
|
|
{
|
|
status_t error = (buffer ? B_OK : B_BAD_VALUE);
|
|
if (error == B_OK) {
|
|
ssize_t flattenedSize = FlattenedSize();
|
|
if (flattenedSize < 0)
|
|
error = flattenedSize;
|
|
if (error == B_OK && size < flattenedSize)
|
|
error = B_BAD_VALUE;
|
|
if (error == B_OK) {
|
|
// convert the path to an entry_ref
|
|
BEntry entry;
|
|
entry_ref ref;
|
|
if (InitCheck() == B_OK && entry.SetTo(Path()) == B_OK)
|
|
entry.GetRef(&ref);
|
|
// store the entry_ref in the buffer
|
|
flattened_entry_ref &fref = *(flattened_entry_ref*)buffer;
|
|
fref.device = ref.device;
|
|
fref.directory = ref.directory;
|
|
if (ref.name)
|
|
strcpy(fref.name, ref.name);
|
|
}
|
|
}
|
|
return error;
|
|
}
|
|
|
|
//! Returns \c true if code is \c B_REF_TYPE, and false otherwise.
|
|
/*! Implements BFlattenable.
|
|
\param code the type code in question
|
|
\return \c true if code is \c B_REF_TYPE, and false otherwise.
|
|
*/
|
|
bool
|
|
BPath::AllowsTypeCode(type_code code) const
|
|
{
|
|
return (code == B_REF_TYPE);
|
|
}
|
|
|
|
/*! \brief Initializes the BPath with the flattened entry_ref data that's
|
|
found in the supplied buffer.
|
|
The type code must be \c B_REF_TYPE.
|
|
Implements BFlattenable.
|
|
\param code the type code of the flattened data
|
|
\param buf a pointer to the flattened data
|
|
\param size the number of bytes contained in \a buf
|
|
\return
|
|
- \c B_OK: Everything went fine.
|
|
- \c B_BAD_VALUE: \c NULL buffer or the buffer doesn't contain an
|
|
entry_ref.
|
|
- other error codes.
|
|
*/
|
|
status_t
|
|
BPath::Unflatten(type_code code, const void *buf, ssize_t size)
|
|
{
|
|
Unset();
|
|
status_t error = B_OK;
|
|
// check params
|
|
if (!(code == B_REF_TYPE && buf
|
|
&& size >= (ssize_t)flattened_entry_ref_size)) {
|
|
error = B_BAD_VALUE;
|
|
}
|
|
if (error == B_OK) {
|
|
if (size == (ssize_t)flattened_entry_ref_size) {
|
|
// already Unset();
|
|
} else {
|
|
// reconstruct the entry_ref from the buffer
|
|
const flattened_entry_ref &fref = *(const flattened_entry_ref*)buf;
|
|
BString name(fref.name, size - flattened_entry_ref_size);
|
|
entry_ref ref(fref.device, fref.directory, name.String());
|
|
error = SetTo(&ref);
|
|
}
|
|
}
|
|
if (error != B_OK)
|
|
fCStatus = error;
|
|
return error;
|
|
}
|
|
|
|
void BPath::_WarPath1() {}
|
|
void BPath::_WarPath2() {}
|
|
void BPath::_WarPath3() {}
|
|
|
|
/*! \brief Sets the supplied path.
|
|
The path is copied. If \c NULL, the object's path is set to NULL as well.
|
|
The object's old path is deleted.
|
|
\param path the path to be set
|
|
\return
|
|
- \c B_OK: Everything went fine.
|
|
- \c B_NO_MEMORY: Insufficient memory.
|
|
*/
|
|
status_t
|
|
BPath::set_path(const char *path)
|
|
{
|
|
status_t error = B_OK;
|
|
const char *oldPath = fName;
|
|
// set the new path
|
|
if (path) {
|
|
fName = new(nothrow) char[strlen(path) + 1];
|
|
if (fName)
|
|
strcpy(fName, path);
|
|
else
|
|
error = B_NO_MEMORY;
|
|
} else
|
|
fName = NULL;
|
|
// delete the old one
|
|
delete[] oldPath;
|
|
return error;
|
|
}
|
|
|
|
|
|
|
|
/*! \brief Checks a path to see if normalization is required.
|
|
|
|
The following items require normalization:
|
|
- Relative pathnames (after concatenation; e.g. "boot/ltj")
|
|
- The presence of "." or ".." ("/boot/ltj/../ltj/./gwar")
|
|
- Redundant slashes ("/boot//ltj")
|
|
- A trailing slash ("/boot/ltj/")
|
|
|
|
\return
|
|
- \c true: \a path requires normalization
|
|
- \c false: \a path does not require normalization
|
|
|
|
\exception BPath::EBadInput : \a path is \c NULL or an empty string.
|
|
|
|
*/
|
|
bool
|
|
BPath::MustNormalize(const char *path)
|
|
{
|
|
// Check for useless input
|
|
if (path == NULL || path[0] == 0)
|
|
throw BPath::EBadInput();
|
|
|
|
int len = strlen(path);
|
|
|
|
/* Look for anything in the string that forces us to normalize:
|
|
+ No leading /
|
|
+ any occurence of /./ or /../ or //, or a trailing /. or /..
|
|
+ a trailing /
|
|
*/;
|
|
if (path[0] != '/')
|
|
return true; // not "/*"
|
|
else if (len == 1)
|
|
return false; // "/"
|
|
else if (len > 1 && path[len-1] == '/')
|
|
return true; // "*/"
|
|
else {
|
|
enum ParseState {
|
|
NoMatch,
|
|
InitialSlash,
|
|
OneDot,
|
|
TwoDots
|
|
} state = NoMatch;
|
|
|
|
for (int i = 0; path[i] != 0; i++) {
|
|
switch (state) {
|
|
case NoMatch:
|
|
if (path[i] == '/')
|
|
state = InitialSlash;
|
|
break;
|
|
|
|
case InitialSlash:
|
|
if (path[i] == '/')
|
|
return true; // "*//*"
|
|
else if (path[i] == '.')
|
|
state = OneDot;
|
|
else
|
|
state = NoMatch;
|
|
break;
|
|
|
|
case OneDot:
|
|
if (path[i] == '/')
|
|
return true; // "*/./*"
|
|
else if (path[i] == '.')
|
|
state = TwoDots;
|
|
else
|
|
state = NoMatch;
|
|
break;
|
|
|
|
case TwoDots:
|
|
if (path[i] == '/')
|
|
return true; // "*/../*"
|
|
else
|
|
state = NoMatch;
|
|
break;
|
|
}
|
|
}
|
|
// If we hit the end of the string while in either
|
|
// of these two states, there was a trailing /. or /..
|
|
if (state == OneDot || state == TwoDots)
|
|
return true;
|
|
else
|
|
return false;
|
|
}
|
|
|
|
}
|
|
|
|
/*! \class BPath::EBadInput
|
|
\brief Internal exception class thrown by BPath::MustNormalize() when given
|
|
invalid input.
|
|
*/
|
|
|
|
/*!
|
|
\var char *BPath::fName
|
|
\brief Pointer to the object's path name string.
|
|
*/
|
|
|
|
/*!
|
|
\var status_t BPath::fCStatus
|
|
\brief The object's initialization status.
|
|
*/
|
|
|
|
|
|
|