2016-09-01 02:07:06 +08:00
|
|
|
/* pathfinder.h: find one of multiple file names in path list
|
|
|
|
|
|
|
|
This file is part of Cygwin.
|
|
|
|
|
|
|
|
This software is a copyrighted work licensed under the terms of the
|
|
|
|
Cygwin license. Please consult the file "CYGWIN_LICENSE" for
|
|
|
|
details. */
|
|
|
|
|
|
|
|
#include "vstrlist.h"
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
|
|
|
|
/* Search a list of directory names for first occurrence of a file,
|
|
|
|
which's file name matches one out of a list of file names. */
|
|
|
|
class pathfinder
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
typedef vstrlist searchdirlist;
|
|
|
|
typedef vstrlist basenamelist;
|
|
|
|
|
|
|
|
private:
|
|
|
|
pathfinder ();
|
|
|
|
pathfinder (pathfinder const &);
|
|
|
|
pathfinder & operator = (pathfinder const &);
|
|
|
|
|
|
|
|
basenamelist basenames_;
|
|
|
|
size_t basenames_maxlen_;
|
|
|
|
|
|
|
|
/* Add to searchdirs_ with extra buffer for any basename we may search for.
|
|
|
|
This is an optimization for the loops in check_path_access method. */
|
|
|
|
searchdirlist searchdirs_;
|
|
|
|
|
|
|
|
public:
|
|
|
|
~pathfinder () {}
|
|
|
|
|
|
|
|
/* We need the basenames to search for first, to allow for optimized
|
|
|
|
memory allocation of each searchpath + longest basename combination.
|
|
|
|
The incoming list of basenames is emptied (ownership take over). */
|
|
|
|
pathfinder (allocator_interface & a, basenamelist & basenames)
|
|
|
|
: basenames_ (a)
|
|
|
|
, basenames_maxlen_ ()
|
|
|
|
, searchdirs_(a)
|
|
|
|
{
|
|
|
|
basenames_.swap(basenames);
|
|
|
|
|
|
|
|
for (basenamelist::buffer_iterator basename (basenames_.begin ());
|
|
|
|
basename != basenames_.end ();
|
|
|
|
++ basename)
|
|
|
|
{
|
|
|
|
if (basenames_maxlen_ < basename->bufferlength ())
|
|
|
|
basenames_maxlen_ = basename->bufferlength ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void add_searchdir (const char *dir, int dirlen)
|
|
|
|
{
|
|
|
|
if (dirlen < 0)
|
|
|
|
dirlen = strlen (dir);
|
|
|
|
|
|
|
|
if (!dirlen)
|
|
|
|
return;
|
|
|
|
|
|
|
|
searchdirs_.appendv (dir, dirlen, "/", 1 + basenames_maxlen_, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
void add_searchpath (const char *path)
|
|
|
|
{
|
|
|
|
while (path && *path)
|
|
|
|
{
|
|
|
|
const char *next = strchr (path, ':');
|
|
|
|
add_searchdir (path, next ? next - path : -1);
|
|
|
|
path = next ? next + 1 : next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void add_envsearchpath (const char *envpath)
|
|
|
|
{
|
|
|
|
add_searchpath (getenv (envpath));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* pathfinder::criterion_interface
|
|
|
|
Overload this test method when you need separate dir and basename. */
|
|
|
|
struct criterion_interface
|
|
|
|
{
|
|
|
|
virtual char const * name () const { return NULL; }
|
|
|
|
|
|
|
|
virtual bool test (searchdirlist::iterator dir,
|
|
|
|
basenamelist::iterator name) const = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/* pathfinder::simple_criterion_interface
|
|
|
|
Overload this test method when you need a single filename. */
|
|
|
|
class simple_criterion_interface
|
|
|
|
: public criterion_interface
|
|
|
|
{
|
|
|
|
virtual bool test (searchdirlist::iterator dir,
|
|
|
|
basenamelist::iterator name) const
|
|
|
|
{
|
|
|
|
/* Complete the filename path to search for within dir,
|
|
|
|
We have allocated enough memory above. */
|
|
|
|
searchdirlist::buffer_iterator dirbuf (dir);
|
|
|
|
memcpy (dirbuf->buffer () + dirbuf->stringlength (),
|
|
|
|
name->string (), name->stringlength () + 1);
|
|
|
|
bool ret = test (dirbuf->string ());
|
|
|
|
/* reset original dir */
|
|
|
|
dirbuf->buffer ()[dirbuf->stringlength ()] = '\0';
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
virtual bool test (const char * filename) const = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/* pathfinder::path_conv_criterion_interface
|
|
|
|
Overload this test method when you need a path_conv. */
|
|
|
|
class path_conv_criterion_interface
|
|
|
|
: public simple_criterion_interface
|
|
|
|
{
|
|
|
|
path_conv mypc_;
|
|
|
|
path_conv & pc_;
|
|
|
|
unsigned opt_;
|
|
|
|
|
|
|
|
/* simple_criterion_interface */
|
|
|
|
virtual bool test (const char * filename) const
|
|
|
|
{
|
|
|
|
pc_.check (filename, opt_);
|
|
|
|
return test (pc_);
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
path_conv_criterion_interface (unsigned opt = PC_SYM_FOLLOW)
|
|
|
|
: mypc_ ()
|
|
|
|
, pc_ (mypc_)
|
|
|
|
, opt_ (opt)
|
|
|
|
{}
|
|
|
|
|
|
|
|
path_conv_criterion_interface (path_conv & ret, unsigned opt = PC_SYM_FOLLOW)
|
|
|
|
: mypc_ ()
|
|
|
|
, pc_ (ret)
|
|
|
|
, opt_ (opt)
|
|
|
|
{}
|
|
|
|
|
|
|
|
virtual bool test (path_conv & pc) const = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/* pathfinder::exists_and_not_dir
|
|
|
|
Test if path_conv argument does exist and is not a directory. */
|
|
|
|
struct exists_and_not_dir
|
|
|
|
: public path_conv_criterion_interface
|
|
|
|
{
|
|
|
|
virtual char const * name () const { return "exists and not dir"; }
|
|
|
|
|
|
|
|
exists_and_not_dir (path_conv & pc, unsigned opt = PC_SYM_FOLLOW)
|
|
|
|
: path_conv_criterion_interface (pc, opt)
|
|
|
|
{}
|
|
|
|
|
|
|
|
/* path_conv_criterion_interface */
|
|
|
|
virtual bool test (path_conv & pc) const
|
|
|
|
{
|
|
|
|
if (pc.exists () && !pc.isdir ())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
pc.error = ENOENT;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/* Find the single dir + basename that matches criterion.
|
|
|
|
|
|
|
|
Calls criterion.test method for each registered dir + basename
|
|
|
|
until returning true:
|
|
|
|
Returns true with found_dir + found_basename set.
|
|
|
|
If criterion.test method never returns true:
|
|
|
|
Returns false, not modifying found_dir nor found_basename. */
|
|
|
|
bool find (criterion_interface const & criterion,
|
|
|
|
searchdirlist::member const ** found_dir = NULL,
|
|
|
|
basenamelist::member const ** found_basename = NULL)
|
|
|
|
{
|
|
|
|
char const * critname = criterion.name ();
|
2016-09-01 02:07:07 +08:00
|
|
|
for (searchdirlist::iterator dir(searchdirs_.begin ());
|
|
|
|
dir != searchdirs_.end ();
|
|
|
|
++dir)
|
|
|
|
for (basenamelist::iterator name = basenames_.begin ();
|
|
|
|
name != basenames_.end ();
|
|
|
|
++name)
|
2016-09-01 02:07:06 +08:00
|
|
|
if (criterion.test (dir, name))
|
|
|
|
{
|
|
|
|
debug_printf ("(%s), take %s%s", critname,
|
|
|
|
dir->string(), name->string ());
|
|
|
|
if (found_dir)
|
|
|
|
*found_dir = dir.operator -> ();
|
|
|
|
if (found_basename)
|
|
|
|
*found_basename = name.operator -> ();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
debug_printf ("not (%s), skip %s%s", critname,
|
|
|
|
dir->string(), name->string ());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
#endif /* __cplusplus */
|