autotier
Automatic Tiering Fuse Filesystem
fuseOps.hpp
1 /*
2  * Copyright (C) 2019-2021 Joshua Boudreau <jboudreau@45drives.com>
3  *
4  * This file is part of autotier.
5  *
6  * autotier is free software: you can redistribute it and/or modify
7  * it under the terms of the GNU General Public License as published by
8  * the Free Software Foundation, either version 3 of the License, or
9  * (at your option) any later version.
10  *
11  * autotier is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14  * GNU General Public License for more details.
15  *
16  * You should have received a copy of the GNU General Public License
17  * along with autotier. If not, see <https://www.gnu.org/licenses/>.
18  */
19 
20 #pragma once
21 
22 #define FUSE_USE_VERSION 30
23 
24 #include <boost/filesystem.hpp>
25 #include <mutex>
26 #include <rocksdb/db.h>
27 #include <thread>
28 namespace fs = boost::filesystem;
29 
30 extern "C" {
31 #include <fuse.h>
32 }
33 
34 class Tier;
35 class TierEngine;
36 
42 class FusePriv {
43 public:
44  fs::path config_path_;
45  fs::path mount_point_;
47  std::shared_ptr<rocksdb::DB> db_;
48  std::vector<Tier *> tiers_;
49  std::thread tier_worker_;
50  std::thread adhoc_server_;
51 
57  for (std::unordered_map<int, char *>::iterator itr = fd_to_path_.begin();
58  itr != fd_to_path_.end();
59  ++itr)
60  free(itr->second);
61  }
68  void insert_fd_to_path(int fd, char *path) {
69  std::lock_guard<std::mutex> lk(fd_to_path_mt_);
70  try {
71  char *val = fd_to_path_.at(fd);
72  // already defined
73  free(val);
74  val = strdup(path);
75  } catch (const std::out_of_range &) {
76  // insert new
77  fd_to_path_.insert({fd, strdup(path)});
78  }
79  }
85  void remove_fd_to_path(int fd) {
86  std::lock_guard<std::mutex> lk(fd_to_path_mt_);
87  try {
88  char *val = fd_to_path_.at(fd);
89  free(val);
90  } catch (const std::out_of_range &) { /* do nothing */
91  }
92  fd_to_path_.erase(fd);
93  }
100  char *fd_to_path(int fd) const {
101  return fd_to_path_.at(fd);
102  }
109  void insert_size_at_open(int fd, uintmax_t size) {
110  std::lock_guard<std::mutex> lk(size_at_open_mt_);
111  size_at_open_[fd] = size;
112  }
118  void remove_size_at_open(int fd) {
119  std::lock_guard<std::mutex> lk(size_at_open_mt_);
120  size_at_open_.erase(fd);
121  }
129  uintmax_t size_at_open(int fd) const {
130  return size_at_open_.at(fd);
131  }
132 private:
137  std::mutex fd_to_path_mt_;
142  std::unordered_map<int, char *> fd_to_path_;
147  std::mutex size_at_open_mt_;
152  std::unordered_map<int, uintmax_t> size_at_open_;
153 };
154 
155 // definitions in helpers.cpp:
160 namespace l {
167  int is_directory(const fs::path &relative_path);
174  Tier *fullpath_to_tier(fs::path fullpath);
181  intmax_t file_size(int fd);
188  intmax_t file_size(const fs::path &path);
198  void update_keys_in_directory(std::string old_directory,
199  std::string new_directory,
200  std::shared_ptr<::rocksdb::DB> db);
201 } // namespace l
202 
208 namespace fuse_ops {
209  extern TierEngine *autotier_ptr;
210 
211  int getattr(const char *path, struct stat *stbuf, struct fuse_file_info *fi);
212 
213  int readlink(const char *path, char *buf, size_t size);
214 
215  int mknod(const char *path, mode_t mode, dev_t rdev);
216 
217  int mkdir(const char *path, mode_t mode);
218 
219  int unlink(const char *path);
220 
221  int rmdir(const char *path);
222 
223  int symlink(const char *from, const char *to);
224 
225  int rename(const char *from, const char *to, unsigned int flags);
226 
227  int link(const char *from, const char *to);
228 
229  int chmod(const char *path, mode_t mode, struct fuse_file_info *fi);
230 
231  int chown(const char *path, uid_t uid, gid_t gid, struct fuse_file_info *fi);
232 
233  int truncate(const char *path, off_t size, struct fuse_file_info *fi);
234 
235  int open(const char *path, struct fuse_file_info *fi);
236 
237  int read(const char *path, char *buf, size_t size, off_t offset, struct fuse_file_info *fi);
238 
239  int
240  write(const char *path, const char *buf, size_t size, off_t offset, struct fuse_file_info *fi);
241 
242  int statfs(const char *path, struct statvfs *stbuf);
243 
244  int flush(const char *path, struct fuse_file_info *fi);
245 
246  int release(const char *path, struct fuse_file_info *fi);
247 
248  int fsync(const char *path, int isdatasync, struct fuse_file_info *fi);
249 
250  int setxattr(const char *path, const char *name, const char *value, size_t size, int flags);
251 
252  int getxattr(const char *path, const char *name, char *value, size_t size);
253 
254  int listxattr(const char *path, char *list, size_t size);
255 
256  int removexattr(const char *path, const char *name);
257 
258  int opendir(const char *path, struct fuse_file_info *fi);
259 
260  int readdir(const char *path,
261  void *buf,
262  fuse_fill_dir_t filler,
263  off_t offset,
264  struct ::fuse_file_info *fi,
265  enum fuse_readdir_flags flags);
266 
267  int releasedir(const char *path, struct fuse_file_info *fi);
268 
269 #ifdef USE_FSYNCDIR
270  int fsyncdir(const char *path, int isdatasync, struct fuse_file_info *fi);
271 #endif
272 
273  void *init(struct fuse_conn_info *conn, struct fuse_config *cfg);
274 
275  void destroy(void *private_data);
276 
277  int access(const char *path, int mask);
278 
279  int create(const char *path, mode_t mode, struct fuse_file_info *fi);
280 
281 #ifdef HAVE_LIBULOCKMGR
282  int lock(const char *path, struct ::fuse_file_info *fi, int cmd, struct flock *lock);
283 #endif
284 
285  int utimens(const char *path, const struct timespec ts[2], struct fuse_file_info *fi);
286 
287  int read_buf(const char *path,
288  struct fuse_bufvec **bufp,
289  size_t size,
290  off_t offset,
291  struct fuse_file_info *fi);
292 
293  int
294  write_buf(const char *path, struct fuse_bufvec *buf, off_t offset, struct fuse_file_info *fi);
295 
296  int flock(const char *path, struct fuse_file_info *fi, int op);
297 
298  int
299  fallocate(const char *path, int mode, off_t offset, off_t length, struct fuse_file_info *fi);
300 
301  ssize_t copy_file_range(const char *path_in,
302  struct fuse_file_info *fi_in,
303  off_t offset_in,
304  const char *path_out,
305  struct fuse_file_info *fi_out,
306  off_t offset_out,
307  size_t len,
308  int flags);
309 
310  off_t lseek(const char *path, off_t off, int whence, struct fuse_file_info *fi);
311 } // namespace fuse_ops
FusePriv::tier_worker_
std::thread tier_worker_
Thread running TierEngineTiering::begin()
Definition: fuseOps.hpp:49
Tier
Class to represent each tier in the filesystem.
Definition: tier.hpp:35
FusePriv::remove_fd_to_path
void remove_fd_to_path(int fd)
Remove fd : path pair from fd_to_path_ map.
Definition: fuseOps.hpp:85
FusePriv::fd_to_path_mt_
std::mutex fd_to_path_mt_
Mutex to synchronize insertion and deletion from fd_to_path_ map.
Definition: fuseOps.hpp:137
FusePriv::~FusePriv
~FusePriv()
Destroy the Fuse Priv object, freeing all cstrings in fd_to_path_.
Definition: fuseOps.hpp:56
FusePriv::size_at_open
uintmax_t size_at_open(int fd) const
Return file size at open corresponding to fd to determine size delta at close.
Definition: fuseOps.hpp:129
FusePriv::fd_to_path
char * fd_to_path(int fd) const
Return path corresponding to file descriptor.
Definition: fuseOps.hpp:100
l::update_keys_in_directory
void update_keys_in_directory(std::string old_directory, std::string new_directory, std::shared_ptr<::rocksdb::DB > db)
Iterate through RocksDB database, updating all paths after old_directory to new_directory for when a ...
Definition: helpers.cpp:69
FusePriv::mount_point_
fs::path mount_point_
Path to autotierfs mount point.
Definition: fuseOps.hpp:45
fuse_ops
Namespace to hold fuse operations. See fuse3/fuse.h documentation for descriptions of each function.
Definition: access.cpp:34
FusePriv::config_path_
fs::path config_path_
Path to configuration file TODO: see if this is used anywhere.
Definition: fuseOps.hpp:44
l
Local namespace.
Definition: helpers.cpp:28
FusePriv::size_at_open_mt_
std::mutex size_at_open_mt_
Mutex to synchronize insertion and deletion from size_at_open_ map.
Definition: fuseOps.hpp:147
TierEngine
Main TierEngine object to construct for setting up autotier. See other components for functionality.
Definition: TierEngine.hpp:30
FusePriv::fd_to_path_
std::unordered_map< int, char * > fd_to_path_
Map relating file descriptor to file path.
Definition: fuseOps.hpp:142
FusePriv::insert_fd_to_path
void insert_fd_to_path(int fd, char *path)
Insert fd : path pair into fd_to_path_ map.
Definition: fuseOps.hpp:68
FusePriv::tiers_
std::vector< Tier * > tiers_
List of pointers to tiers from TierEngine.
Definition: fuseOps.hpp:48
FusePriv
Fuse Private data class grabbed from fuse_get_context()->private_data (void*) in fuse filesystem func...
Definition: fuseOps.hpp:42
l::file_size
intmax_t file_size(int fd)
Get size of file from file descriptor.
Definition: helpers.cpp:53
FusePriv::db_
std::shared_ptr< rocksdb::DB > db_
RocksDB database holding file metadata.
Definition: fuseOps.hpp:47
l::fullpath_to_tier
Tier * fullpath_to_tier(fs::path fullpath)
Find tier containing full path.
Definition: helpers.cpp:41
l::is_directory
int is_directory(const fs::path &relative_path)
Test if path is a directory.
Definition: helpers.cpp:29
FusePriv::adhoc_server_
std::thread adhoc_server_
Thread running TierEngineAdhoc::process_adhoc_requests()
Definition: fuseOps.hpp:50
FusePriv::remove_size_at_open
void remove_size_at_open(int fd)
Remove fd : size at open pair from size_at_open_ map.
Definition: fuseOps.hpp:118
FusePriv::autotier_
TierEngine * autotier_
Pointer to TierEngine.
Definition: fuseOps.hpp:46
FusePriv::insert_size_at_open
void insert_size_at_open(int fd, uintmax_t size)
Insert fd : size at open pair into size_at_open_ map.
Definition: fuseOps.hpp:109
FusePriv::size_at_open_
std::unordered_map< int, uintmax_t > size_at_open_
Map relating file descriptor to file size at open.
Definition: fuseOps.hpp:152