|
|
@ -37,69 +37,174 @@ using std::string; |
|
|
|
using std::vector; |
|
|
|
using std::size_t; |
|
|
|
|
|
|
|
namespace mergerfs |
|
|
|
{ |
|
|
|
int |
|
|
|
Policy::Func::epmfs(const Category::Enum::Type type, |
|
|
|
const vector<string> &basepaths, |
|
|
|
const string &fusepath, |
|
|
|
const size_t minfreespace, |
|
|
|
Paths &paths) |
|
|
|
static |
|
|
|
inline |
|
|
|
void |
|
|
|
_calc_epmfs(const struct statvfs &fsstats, |
|
|
|
const char *basepath, |
|
|
|
fsblkcnt_t &epmfs, |
|
|
|
const char *&epmfsbasepath, |
|
|
|
fsblkcnt_t &mfs, |
|
|
|
const char *&mfsbasepath) |
|
|
|
{ |
|
|
|
fsblkcnt_t existingmfs; |
|
|
|
fsblkcnt_t generalmfs; |
|
|
|
string fullpath; |
|
|
|
string generalmfspath; |
|
|
|
string existingmfspath; |
|
|
|
vector<string>::const_iterator iter = basepaths.begin(); |
|
|
|
vector<string>::const_iterator eiter = basepaths.end(); |
|
|
|
fsblkcnt_t spaceavail; |
|
|
|
|
|
|
|
if(iter == eiter) |
|
|
|
return (errno = ENOENT,-1); |
|
|
|
spaceavail = (fsstats.f_frsize * fsstats.f_bavail); |
|
|
|
if(spaceavail > epmfs) |
|
|
|
{ |
|
|
|
epmfs = spaceavail; |
|
|
|
epmfsbasepath = basepath; |
|
|
|
} |
|
|
|
|
|
|
|
existingmfs = 0; |
|
|
|
generalmfs = 0; |
|
|
|
do |
|
|
|
if(spaceavail > mfs) |
|
|
|
{ |
|
|
|
int rv; |
|
|
|
struct statvfs fsstats; |
|
|
|
const string &mountpoint = *iter; |
|
|
|
mfs = spaceavail; |
|
|
|
mfsbasepath = basepath; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
rv = ::statvfs(mountpoint.c_str(),&fsstats); |
|
|
|
if(rv == 0) |
|
|
|
static |
|
|
|
inline |
|
|
|
void |
|
|
|
_calc_mfs(const struct statvfs &fsstats, |
|
|
|
const char *basepath, |
|
|
|
fsblkcnt_t &mfs, |
|
|
|
const char *&mfsbasepath) |
|
|
|
{ |
|
|
|
fsblkcnt_t spaceavail; |
|
|
|
struct stat st; |
|
|
|
|
|
|
|
spaceavail = (fsstats.f_frsize * fsstats.f_bavail); |
|
|
|
if(spaceavail > generalmfs) |
|
|
|
if(spaceavail > mfs) |
|
|
|
{ |
|
|
|
generalmfs = spaceavail; |
|
|
|
generalmfspath = mountpoint; |
|
|
|
mfs = spaceavail; |
|
|
|
mfsbasepath = basepath; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
fullpath = fs::make_path(mountpoint,fusepath); |
|
|
|
rv = ::lstat(fullpath.c_str(),&st); |
|
|
|
if(rv == 0) |
|
|
|
static |
|
|
|
inline |
|
|
|
int |
|
|
|
_try_statvfs(const char *basepath, |
|
|
|
const string &fullpath, |
|
|
|
fsblkcnt_t &epmfs, |
|
|
|
const char *&epmfsbasepath, |
|
|
|
fsblkcnt_t &mfs, |
|
|
|
const char *&mfsbasepath) |
|
|
|
{ |
|
|
|
if(spaceavail > existingmfs) |
|
|
|
int rv; |
|
|
|
struct statvfs fsstats; |
|
|
|
|
|
|
|
rv = ::statvfs(fullpath.c_str(),&fsstats); |
|
|
|
if(rv == 0) |
|
|
|
_calc_epmfs(fsstats,basepath,epmfs,epmfsbasepath,mfs,mfsbasepath); |
|
|
|
|
|
|
|
return rv; |
|
|
|
} |
|
|
|
|
|
|
|
static |
|
|
|
inline |
|
|
|
int |
|
|
|
_try_statvfs(const char *basepath, |
|
|
|
fsblkcnt_t &mfs, |
|
|
|
const char *&mfsbasepath) |
|
|
|
{ |
|
|
|
existingmfs = spaceavail; |
|
|
|
existingmfspath = mountpoint; |
|
|
|
int rv; |
|
|
|
struct statvfs fsstats; |
|
|
|
|
|
|
|
rv = ::statvfs(basepath,&fsstats); |
|
|
|
if(rv == 0) |
|
|
|
_calc_mfs(fsstats,basepath,mfs,mfsbasepath); |
|
|
|
|
|
|
|
return rv; |
|
|
|
} |
|
|
|
|
|
|
|
static |
|
|
|
int |
|
|
|
_epmfs_create(const vector<string> &basepaths, |
|
|
|
const string &fusepath, |
|
|
|
Paths &paths) |
|
|
|
|
|
|
|
{ |
|
|
|
fsblkcnt_t epmfs; |
|
|
|
fsblkcnt_t mfs; |
|
|
|
const char *basepath; |
|
|
|
const char *mfsbasepath; |
|
|
|
const char *epmfsbasepath; |
|
|
|
string fullpath; |
|
|
|
|
|
|
|
mfs = 0; |
|
|
|
epmfs = 0; |
|
|
|
mfsbasepath = NULL; |
|
|
|
epmfsbasepath = NULL; |
|
|
|
for(size_t i = 0, ei = basepaths.size(); i != ei; i++) |
|
|
|
{ |
|
|
|
int rv; |
|
|
|
|
|
|
|
basepath = basepaths[i].c_str(); |
|
|
|
fullpath = fs::make_path(basepath,fusepath); |
|
|
|
|
|
|
|
rv = _try_statvfs(basepath,fusepath,epmfs,epmfsbasepath,mfs,mfsbasepath); |
|
|
|
if(rv == -1) |
|
|
|
_try_statvfs(basepath,mfs,mfsbasepath); |
|
|
|
} |
|
|
|
|
|
|
|
if(epmfsbasepath == NULL) |
|
|
|
epmfsbasepath = mfsbasepath; |
|
|
|
|
|
|
|
paths.push_back(Path(epmfsbasepath, |
|
|
|
fs::make_path(epmfsbasepath,fusepath))); |
|
|
|
|
|
|
|
return 0; |
|
|
|
} |
|
|
|
|
|
|
|
++iter; |
|
|
|
static |
|
|
|
int |
|
|
|
_epmfs(const vector<string> &basepaths, |
|
|
|
const string &fusepath, |
|
|
|
Paths &paths) |
|
|
|
|
|
|
|
{ |
|
|
|
fsblkcnt_t epmfs; |
|
|
|
const char *basepath; |
|
|
|
const char *epmfsbasepath; |
|
|
|
string fullpath; |
|
|
|
|
|
|
|
epmfs = 0; |
|
|
|
epmfsbasepath = NULL; |
|
|
|
for(size_t i = 0, ei = basepaths.size(); i != ei; i++) |
|
|
|
{ |
|
|
|
int rv; |
|
|
|
struct statvfs fsstats; |
|
|
|
|
|
|
|
basepath = basepaths[i].c_str(); |
|
|
|
fullpath = fs::make_path(basepath,fusepath); |
|
|
|
|
|
|
|
rv = ::statvfs(fullpath.c_str(),&fsstats); |
|
|
|
if(rv == 0) |
|
|
|
_calc_mfs(fsstats,basepath,epmfs,epmfsbasepath); |
|
|
|
} |
|
|
|
while(iter != eiter); |
|
|
|
|
|
|
|
if(existingmfspath.empty()) |
|
|
|
existingmfspath = generalmfspath; |
|
|
|
if(epmfsbasepath == NULL) |
|
|
|
return (errno=ENOENT,-1); |
|
|
|
|
|
|
|
paths.push_back(Path(existingmfspath, |
|
|
|
fullpath)); |
|
|
|
paths.push_back(Path(epmfsbasepath, |
|
|
|
fs::make_path(epmfsbasepath,fusepath))); |
|
|
|
|
|
|
|
return 0; |
|
|
|
} |
|
|
|
|
|
|
|
namespace mergerfs |
|
|
|
{ |
|
|
|
int |
|
|
|
Policy::Func::epmfs(const Category::Enum::Type type, |
|
|
|
const vector<string> &basepaths, |
|
|
|
const string &fusepath, |
|
|
|
const size_t minfreespace, |
|
|
|
Paths &paths) |
|
|
|
{ |
|
|
|
if(type == Category::Enum::create) |
|
|
|
return _epmfs_create(basepaths,fusepath,paths); |
|
|
|
|
|
|
|
return _epmfs(basepaths,fusepath,paths); |
|
|
|
} |
|
|
|
} |