fuse_i.h 11.6 KB
Newer Older
Miklos Szeredi's avatar
Miklos Szeredi committed
1 2
/*
  FUSE: Filesystem in Userspace
3
  Copyright (C) 2001-2006  Miklos Szeredi <miklos@szeredi.hu>
Miklos Szeredi's avatar
Miklos Szeredi committed
4 5 6 7 8 9 10

  This program can be distributed under the terms of the GNU GPL.
  See the file COPYING.
*/

#include <linux/fuse.h>
#include <linux/fs.h>
11
#include <linux/mount.h>
Miklos Szeredi's avatar
Miklos Szeredi committed
12 13 14 15 16
#include <linux/wait.h>
#include <linux/list.h>
#include <linux/spinlock.h>
#include <linux/mm.h>
#include <linux/backing-dev.h>
17
#include <linux/mutex.h>
Miklos Szeredi's avatar
Miklos Szeredi committed
18

19 20 21
/** Max number of pages that can be used in a single read request */
#define FUSE_MAX_PAGES_PER_REQ 32

22 23 24
/** Maximum number of outstanding background requests */
#define FUSE_MAX_BACKGROUND 10

25 26 27
/** It could be as large as PATH_MAX, but would that have any uses? */
#define FUSE_NAME_MAX 1024

28 29 30
/** Number of dentries for each connection in the control filesystem */
#define FUSE_CTL_NUM_DENTRIES 3

31 32 33 34 35 36 37 38 39
/** If the FUSE_DEFAULT_PERMISSIONS flag is given, the filesystem
    module will check permissions based on the file mode.  Otherwise no
    permission checking is done in the kernel */
#define FUSE_DEFAULT_PERMISSIONS (1 << 0)

/** If the FUSE_ALLOW_OTHER flag is given, then not only the user
    doing the mount will be allowed to access the filesystem */
#define FUSE_ALLOW_OTHER         (1 << 1)

40 41 42 43 44
/** List of active connections */
extern struct list_head fuse_conn_list;

/** Global mutex protecting fuse_conn_list and the control filesystem */
extern struct mutex fuse_mutex;
Miklos Szeredi's avatar
Miklos Szeredi committed
45

Miklos Szeredi's avatar
Miklos Szeredi committed
46 47 48 49 50 51 52 53 54
/** FUSE inode */
struct fuse_inode {
	/** Inode data */
	struct inode inode;

	/** Unique ID, which identifies the inode between userspace
	 * and kernel */
	u64 nodeid;

55 56 57
	/** Number of lookups on this inode */
	u64 nlookup;

58 59 60
	/** The request used for sending the FORGET message */
	struct fuse_req *forget_req;

Miklos Szeredi's avatar
Miklos Szeredi committed
61 62 63 64
	/** Time in jiffies until the file attributes are valid */
	unsigned long i_time;
};

65 66 67
/** FUSE specific file data */
struct fuse_file {
	/** Request reserved for flush and release */
68
	struct fuse_req *reserved_req;
69 70 71 72 73

	/** File handle used by userspace */
	u64 fh;
};

74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105
/** One input argument of a request */
struct fuse_in_arg {
	unsigned size;
	const void *value;
};

/** The request input */
struct fuse_in {
	/** The request header */
	struct fuse_in_header h;

	/** True if the data for the last argument is in req->pages */
	unsigned argpages:1;

	/** Number of arguments */
	unsigned numargs;

	/** Array of arguments */
	struct fuse_in_arg args[3];
};

/** One output argument of a request */
struct fuse_arg {
	unsigned size;
	void *value;
};

/** The request output */
struct fuse_out {
	/** Header returned from userspace */
	struct fuse_out_header h;

106 107 108 109 110
	/*
	 * The following bitfields are not changed during the request
	 * processing
	 */

111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127
	/** Last argument is variable length (can be shorter than
	    arg->size) */
	unsigned argvar:1;

	/** Last argument is a list of pages to copy data to */
	unsigned argpages:1;

	/** Zero partially or not copied pages */
	unsigned page_zeroing:1;

	/** Number or arguments */
	unsigned numargs;

	/** Array of arguments */
	struct fuse_arg args[3];
};

128 129 130 131 132 133 134 135 136
/** The request state */
enum fuse_req_state {
	FUSE_REQ_INIT = 0,
	FUSE_REQ_PENDING,
	FUSE_REQ_READING,
	FUSE_REQ_SENT,
	FUSE_REQ_FINISHED
};

137 138
struct fuse_conn;

139 140 141 142
/**
 * A request to the client
 */
struct fuse_req {
143 144
	/** This can be on either pending processing or io lists in
	    fuse_conn */
145 146 147 148 149
	struct list_head list;

	/** refcount */
	atomic_t count;

150 151 152
	/*
	 * The following bitfields are either set once before the
	 * request is queued or setting/clearing them is protected by
153
	 * fuse_conn->lock
154 155
	 */

156 157 158
	/** True if the request has reply */
	unsigned isreply:1;

159 160 161
	/** Force sending of the request even if interrupted */
	unsigned force:1;

162 163 164 165 166 167 168 169 170
	/** The request was interrupted */
	unsigned interrupted:1;

	/** Request is sent in the background */
	unsigned background:1;

	/** Data is being copied to/from the request */
	unsigned locked:1;

171 172 173
	/** Request is counted as "waiting" */
	unsigned waiting:1;

174 175
	/** State of the request */
	enum fuse_req_state state;
176 177 178 179 180 181 182 183 184 185 186 187

	/** The request input */
	struct fuse_in in;

	/** The request output */
	struct fuse_out out;

	/** Used to wake up the task waiting for completion of request*/
	wait_queue_head_t waitq;

	/** Data for asynchronous requests */
	union {
188
		struct fuse_forget_in forget_in;
189
		struct fuse_release_in release_in;
190 191
		struct fuse_init_in init_in;
		struct fuse_init_out init_out;
192
		struct fuse_read_in read_in;
193
		struct fuse_lk_in lk_in;
194 195 196 197 198 199 200 201 202 203 204 205 206
	} misc;

	/** page vector */
	struct page *pages[FUSE_MAX_PAGES_PER_REQ];

	/** number of pages in vector */
	unsigned num_pages;

	/** offset of data on first page */
	unsigned page_offset;

	/** File used in the request (or NULL) */
	struct file *file;
207

208 209 210 211 212 213
	/** vfsmount used in release */
	struct vfsmount *vfsmount;

	/** dentry used in release */
	struct dentry *dentry;

214 215
	/** Request completion callback */
	void (*end)(struct fuse_conn *, struct fuse_req *);
216 217 218

	/** Request is stolen from fuse_file->reserved_req */
	struct file *stolen_file;
219 220
};

Miklos Szeredi's avatar
Miklos Szeredi committed
221 222 223 224 225 226 227 228
/**
 * A Fuse connection.
 *
 * This structure is created, when the filesystem is mounted, and is
 * destroyed, when the client device is closed and the filesystem is
 * unmounted.
 */
struct fuse_conn {
229 230 231
	/** Lock protecting accessess to  members of this structure */
	spinlock_t lock;

232 233 234
	/** Refcount */
	atomic_t count;

Miklos Szeredi's avatar
Miklos Szeredi committed
235 236 237
	/** The user id for this mount */
	uid_t user_id;

238 239 240
	/** The group id for this mount */
	gid_t group_id;

241 242 243
	/** The fuse mount flags for this mount */
	unsigned flags;

244 245 246
	/** Maximum read size */
	unsigned max_read;

Miklos Szeredi's avatar
Miklos Szeredi committed
247 248 249
	/** Maximum write size */
	unsigned max_write;

250 251 252 253 254 255 256 257 258
	/** Readers of the connection are waiting on this */
	wait_queue_head_t waitq;

	/** The list of pending requests */
	struct list_head pending;

	/** The list of requests being processed */
	struct list_head processing;

259 260 261
	/** The list of requests under I/O */
	struct list_head io;

262 263 264 265 266 267 268 269 270 271 272
	/** Number of requests currently in the background */
	unsigned num_background;

	/** Flag indicating if connection is blocked.  This will be
	    the case before the INIT reply is received, and if there
	    are too many outstading backgrounds requests */
	int blocked;

	/** waitq for blocked connection */
	wait_queue_head_t blocked_waitq;

273 274 275
	/** The next unique request id */
	u64 reqctr;

276 277
	/** Connection established, cleared on umount, connection
	    abort and device release */
278
	unsigned connected;
279

280 281 282
	/** Connection failed (version mismatch).  Cannot race with
	    setting other bitfields since it is only set once in INIT
	    reply, before any other request, and never cleared */
283 284
	unsigned conn_error : 1;

285 286 287
	/** Do readpages asynchronously?  Only set in INIT */
	unsigned async_read : 1;

288 289 290 291 292
	/*
	 * The following bitfields are only for optimization purposes
	 * and hence races in setting them will not cause malfunction
	 */

293 294 295
	/** Is fsync not implemented by fs? */
	unsigned no_fsync : 1;

296 297 298
	/** Is fsyncdir not implemented by fs? */
	unsigned no_fsyncdir : 1;

299 300 301
	/** Is flush not implemented by fs? */
	unsigned no_flush : 1;

302 303 304 305 306 307 308 309 310 311 312 313
	/** Is setxattr not implemented by fs? */
	unsigned no_setxattr : 1;

	/** Is getxattr not implemented by fs? */
	unsigned no_getxattr : 1;

	/** Is listxattr not implemented by fs? */
	unsigned no_listxattr : 1;

	/** Is removexattr not implemented by fs? */
	unsigned no_removexattr : 1;

314 315 316
	/** Are file locking primitives not implemented by fs? */
	unsigned no_lock : 1;

317 318 319
	/** Is access not implemented by fs? */
	unsigned no_access : 1;

320 321 322
	/** Is create not implemented by fs? */
	unsigned no_create : 1;

323 324 325
	/** The number of requests waiting for completion */
	atomic_t num_waiting;

326 327 328
	/** Negotiated minor version */
	unsigned minor;

Miklos Szeredi's avatar
Miklos Szeredi committed
329 330
	/** Backing dev info */
	struct backing_dev_info bdi;
331

332 333 334 335 336 337 338 339 340 341 342
	/** Entry on the fuse_conn_list */
	struct list_head entry;

	/** Unique ID */
	u64 id;

	/** Dentries in the control filesystem */
	struct dentry *ctl_dentry[FUSE_CTL_NUM_DENTRIES];

	/** number of dentries used in the above array */
	int ctl_ndents;
343 344 345

	/** O_ASYNC requests */
	struct fasync_struct *fasync;
Miklos Szeredi's avatar
Miklos Szeredi committed
346 347 348 349
};

static inline struct fuse_conn *get_fuse_conn_super(struct super_block *sb)
{
350
	return sb->s_fs_info;
Miklos Szeredi's avatar
Miklos Szeredi committed
351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367
}

static inline struct fuse_conn *get_fuse_conn(struct inode *inode)
{
	return get_fuse_conn_super(inode->i_sb);
}

static inline struct fuse_inode *get_fuse_inode(struct inode *inode)
{
	return container_of(inode, struct fuse_inode, inode);
}

static inline u64 get_node_id(struct inode *inode)
{
	return get_fuse_inode(inode)->nodeid;
}

368
/** Device operations */
369
extern const struct file_operations fuse_dev_operations;
370

371 372 373 374
/**
 * Get a filled in inode
 */
struct inode *fuse_iget(struct super_block *sb, unsigned long nodeid,
375
			int generation, struct fuse_attr *attr);
376 377 378 379 380

/**
 * Send FORGET command
 */
void fuse_send_forget(struct fuse_conn *fc, struct fuse_req *req,
381
		      unsigned long nodeid, u64 nlookup);
382

383
/**
384
 * Initialize READ or READDIR request
385
 */
386 387
void fuse_read_fill(struct fuse_req *req, struct file *file,
		    struct inode *inode, loff_t pos, size_t count, int opcode);
388 389 390 391 392 393

/**
 * Send OPEN or OPENDIR request
 */
int fuse_open_common(struct inode *inode, struct file *file, int isdir);

394 395 396 397 398
struct fuse_file *fuse_file_alloc(void);
void fuse_file_free(struct fuse_file *ff);
void fuse_finish_open(struct inode *inode, struct file *file,
		      struct fuse_file *ff, struct fuse_open_out *outarg);

399 400 401
/** */
struct fuse_req *fuse_release_fill(struct fuse_file *ff, u64 nodeid, int flags,
				   int opcode);
402 403 404 405 406
/**
 * Send RELEASE or RELEASEDIR request
 */
int fuse_release_common(struct inode *inode, struct file *file, int isdir);

407 408 409 410 411 412
/**
 * Send FSYNC or FSYNCDIR request
 */
int fuse_fsync_common(struct file *file, struct dentry *de, int datasync,
		      int isdir);

413
/**
414
 * Initialize file operations on a regular file
415 416 417
 */
void fuse_init_file_inode(struct inode *inode);

418
/**
419
 * Initialize inode operations on regular files and special files
420 421 422 423
 */
void fuse_init_common(struct inode *inode);

/**
424
 * Initialize inode and file operations on a directory
425 426 427 428
 */
void fuse_init_dir(struct inode *inode);

/**
429
 * Initialize inode operations on a symlink
430 431 432 433 434 435 436 437
 */
void fuse_init_symlink(struct inode *inode);

/**
 * Change attributes of an inode
 */
void fuse_change_attributes(struct inode *inode, struct fuse_attr *attr);

438 439 440 441 442 443 444 445 446 447
/**
 * Initialize the client device
 */
int fuse_dev_init(void);

/**
 * Cleanup the client device
 */
void fuse_dev_cleanup(void);

448 449 450
int fuse_ctl_init(void);
void fuse_ctl_cleanup(void);

451 452 453 454 455 456 457 458 459 460 461
/**
 * Allocate a request
 */
struct fuse_req *fuse_request_alloc(void);

/**
 * Free a request
 */
void fuse_request_free(struct fuse_req *req);

/**
462
 * Get a request, may fail with -ENOMEM
463
 */
464
struct fuse_req *fuse_get_req(struct fuse_conn *fc);
465

466 467 468 469 470
/**
 * Gets a requests for a file operation, always succeeds
 */
struct fuse_req *fuse_get_req_nofail(struct fuse_conn *fc, struct file *file);

471
/**
472 473
 * Decrement reference count of a request.  If count goes to zero free
 * the request.
474 475 476 477
 */
void fuse_put_request(struct fuse_conn *fc, struct fuse_req *req);

/**
478
 * Send a request (synchronous)
479 480 481 482 483 484 485 486 487 488 489 490 491
 */
void request_send(struct fuse_conn *fc, struct fuse_req *req);

/**
 * Send a request with no reply
 */
void request_send_noreply(struct fuse_conn *fc, struct fuse_req *req);

/**
 * Send a request in the background
 */
void request_send_background(struct fuse_conn *fc, struct fuse_req *req);

492
/* Abort all requests */
493 494
void fuse_abort_conn(struct fuse_conn *fc);

495 496 497 498 499 500 501 502 503
/**
 * Get the attributes of a file
 */
int fuse_do_getattr(struct inode *inode);

/**
 * Invalidate inode attributes
 */
void fuse_invalidate_attr(struct inode *inode);
504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523

/**
 * Acquire reference to fuse_conn
 */
struct fuse_conn *fuse_conn_get(struct fuse_conn *fc);

/**
 * Release reference to fuse_conn
 */
void fuse_conn_put(struct fuse_conn *fc);

/**
 * Add connection to control filesystem
 */
int fuse_ctl_add_conn(struct fuse_conn *fc);

/**
 * Remove connection from control filesystem
 */
void fuse_ctl_remove_conn(struct fuse_conn *fc);