| /** quotaio.h |
| * |
| * Interface to the quota library. |
| * |
| * The quota library provides interface for creating and updating the quota |
| * files and the ext4 superblock fields. It supports the new VFS_V1 quota |
| * format. The quota library also provides support for keeping track of quotas |
| * in memory. |
| * |
| * Aditya Kali <adityakali@google.com> |
| * Header of IO operations for quota utilities |
| * |
| * Jan Kara <jack@suse.cz> |
| * |
| * Hyojun Kim <hyojun@google.com> - Ported to f2fs-tools |
| */ |
| |
| #ifndef GUARD_QUOTAIO_H |
| #define GUARD_QUOTAIO_H |
| |
| #include <limits.h> |
| #include <sys/types.h> |
| #include <sys/stat.h> |
| #include <arpa/inet.h> |
| |
| #include "dict.h" |
| #include "f2fs_fs.h" |
| #include "f2fs.h" |
| #include "node.h" |
| #include "fsck.h" |
| |
| #include "dqblk_v2.h" |
| |
| typedef int64_t qsize_t; /* Type in which we store size limitations */ |
| typedef int32_t f2fs_ino_t; |
| typedef int errcode_t; |
| |
| enum quota_type { |
| USRQUOTA = 0, |
| GRPQUOTA = 1, |
| PRJQUOTA = 2, |
| MAXQUOTAS = 3, |
| }; |
| |
| #if MAXQUOTAS > 32 |
| #error "cannot have more than 32 quota types to fit in qtype_bits" |
| #endif |
| |
| |
| #define QUOTA_USR_BIT (1 << USRQUOTA) |
| #define QUOTA_GRP_BIT (1 << GRPQUOTA) |
| #define QUOTA_PRJ_BIT (1 << PRJQUOTA) |
| #define QUOTA_ALL_BIT (QUOTA_USR_BIT | QUOTA_GRP_BIT | QUOTA_PRJ_BIT) |
| |
| typedef struct quota_ctx *quota_ctx_t; |
| |
| struct quota_ctx { |
| struct f2fs_sb_info *sbi; |
| struct dict_t *quota_dict[MAXQUOTAS]; |
| struct quota_handle *quota_file[MAXQUOTAS]; |
| struct dict_t linked_inode_dict; |
| }; |
| |
| /* |
| * Definitions of magics and versions of current quota files |
| */ |
| #define INITQMAGICS {\ |
| 0xd9c01f11, /* USRQUOTA */\ |
| 0xd9c01927, /* GRPQUOTA */\ |
| 0xd9c03f14 /* PRJQUOTA */\ |
| } |
| |
| /* Size of blocks in which are counted size limits in generic utility parts */ |
| #define QUOTABLOCK_BITS 10 |
| #define QUOTABLOCK_SIZE (1 << QUOTABLOCK_BITS) |
| #define toqb(x) (((x) + QUOTABLOCK_SIZE - 1) >> QUOTABLOCK_BITS) |
| |
| /* Quota format type IDs */ |
| #define QFMT_VFS_OLD 1 |
| #define QFMT_VFS_V0 2 |
| #define QFMT_VFS_V1 4 |
| |
| /* |
| * The following constants define the default amount of time given a user |
| * before the soft limits are treated as hard limits (usually resulting |
| * in an allocation failure). The timer is started when the user crosses |
| * their soft limit, it is reset when they go below their soft limit. |
| */ |
| #define MAX_IQ_TIME 604800 /* (7*24*60*60) 1 week */ |
| #define MAX_DQ_TIME 604800 /* (7*24*60*60) 1 week */ |
| |
| #define IOFL_INFODIRTY 0x01 /* Did info change? */ |
| |
| struct quotafile_ops; |
| |
| /* Generic information about quotafile */ |
| struct util_dqinfo { |
| time_t dqi_bgrace; /* Block grace time for given quotafile */ |
| time_t dqi_igrace; /* Inode grace time for given quotafile */ |
| union { |
| struct v2_mem_dqinfo v2_mdqi; |
| } u; /* Format specific info about quotafile */ |
| }; |
| |
| struct quota_file { |
| struct f2fs_sb_info *sbi; |
| f2fs_ino_t ino; |
| int64_t filesize; |
| }; |
| |
| /* Structure for one opened quota file */ |
| struct quota_handle { |
| enum quota_type qh_type; /* Type of quotafile */ |
| int qh_fmt; /* Quotafile format */ |
| int qh_file_flags; |
| int qh_io_flags; /* IO flags for file */ |
| struct quota_file qh_qf; |
| unsigned int (*read)(struct quota_file *qf, long offset, |
| void *buf, unsigned int size); |
| unsigned int (*write)(struct quota_file *qf, long offset, |
| void *buf, unsigned int size); |
| struct quotafile_ops *qh_ops; /* Operations on quotafile */ |
| struct util_dqinfo qh_info; /* Generic quotafile info */ |
| }; |
| |
| /* Utility quota block */ |
| struct util_dqblk { |
| qsize_t dqb_ihardlimit; |
| qsize_t dqb_isoftlimit; |
| qsize_t dqb_curinodes; |
| qsize_t dqb_bhardlimit; |
| qsize_t dqb_bsoftlimit; |
| qsize_t dqb_curspace; |
| time_t dqb_btime; |
| time_t dqb_itime; |
| union { |
| struct v2_mem_dqblk v2_mdqb; |
| } u; /* Format specific dquot information */ |
| }; |
| |
| /* Structure for one loaded quota */ |
| struct dquot { |
| struct dquot *dq_next; /* Pointer to next dquot in the list */ |
| qid_t dq_id; /* ID dquot belongs to */ |
| int dq_flags; /* Some flags for utils */ |
| struct quota_handle *dq_h; /* Handle of quotafile for this dquot */ |
| struct util_dqblk dq_dqb; /* Parsed data of dquot */ |
| }; |
| |
| #define DQF_SEEN 0x0001 |
| |
| /* Structure of quotafile operations */ |
| struct quotafile_ops { |
| /* Check whether quotafile is in our format */ |
| int (*check_file) (struct quota_handle *h, int type); |
| /* Open quotafile */ |
| int (*init_io) (struct quota_handle *h); |
| /* Create new quotafile */ |
| int (*new_io) (struct quota_handle *h); |
| /* Write all changes and close quotafile */ |
| int (*end_io) (struct quota_handle *h); |
| /* Write info about quotafile */ |
| int (*write_info) (struct quota_handle *h); |
| /* Read dquot into memory */ |
| struct dquot *(*read_dquot) (struct quota_handle *h, qid_t id); |
| /* Write given dquot to disk */ |
| int (*commit_dquot) (struct dquot *dquot); |
| /* Scan quotafile and call callback on every structure */ |
| int (*scan_dquots) (struct quota_handle *h, |
| int (*process_dquot) (struct dquot *dquot, |
| void *data), |
| void *data); |
| /* Function to print format specific file information */ |
| int (*report) (struct quota_handle *h, int verbose); |
| }; |
| |
| #ifdef __CHECKER__ |
| # ifndef __bitwise |
| # define __bitwise __attribute__((bitwise)) |
| # endif |
| #define __force __attribute__((force)) |
| #else |
| # ifndef __bitwise |
| # define __bitwise |
| # endif |
| #define __force |
| #endif |
| |
| #define be32_to_cpu(n) ntohl(n) |
| |
| /* Open existing quotafile of given type (and verify its format) on given |
| * filesystem. */ |
| errcode_t quota_file_open(struct f2fs_sb_info *sbi, struct quota_handle *h, |
| enum quota_type qtype, int flags); |
| |
| /* Create new quotafile of specified format on given filesystem */ |
| errcode_t quota_file_create(struct f2fs_sb_info *sbi, struct quota_handle *h, |
| enum quota_type qtype); |
| |
| /* Close quotafile */ |
| errcode_t quota_file_close(struct f2fs_sb_info *sbi, struct quota_handle *h, |
| int update_filesize); |
| |
| /* Get empty quota structure */ |
| struct dquot *get_empty_dquot(void); |
| const char *quota_type2name(enum quota_type qtype); |
| void update_grace_times(struct dquot *q); |
| |
| /* In mkquota.c */ |
| errcode_t quota_init_context(struct f2fs_sb_info *sbi); |
| void quota_data_inodes(quota_ctx_t qctx, struct f2fs_inode *inode, int adjust); |
| void quota_data_add(quota_ctx_t qctx, struct f2fs_inode *inode, qsize_t space); |
| void quota_data_sub(quota_ctx_t qctx, struct f2fs_inode *inode, qsize_t space); |
| errcode_t quota_write_inode(struct f2fs_sb_info *sbi, enum quota_type qtype); |
| void quota_add_inode_usage(quota_ctx_t qctx, f2fs_ino_t ino, |
| struct f2fs_inode* inode); |
| void quota_release_context(quota_ctx_t *qctx); |
| errcode_t quota_compare_and_update(struct f2fs_sb_info *sbi, |
| enum quota_type qtype, int *usage_inconsistent, |
| int preserve_limits); |
| |
| static inline errcode_t quota_get_mem(unsigned long size, void *ptr) |
| { |
| void *pp; |
| |
| pp = malloc(size); |
| if (!pp) |
| return -1; |
| memcpy(ptr, &pp, sizeof (pp)); |
| return 0; |
| } |
| |
| static inline errcode_t quota_get_memzero(unsigned long size, void *ptr) |
| { |
| void *pp; |
| |
| pp = malloc(size); |
| if (!pp) |
| return -1; |
| memset(pp, 0, size); |
| memcpy(ptr, &pp, sizeof(pp)); |
| return 0; |
| } |
| |
| static inline errcode_t quota_free_mem(void *ptr) |
| { |
| void *p; |
| |
| memcpy(&p, ptr, sizeof(p)); |
| free(p); |
| p = 0; |
| memcpy(ptr, &p, sizeof(p)); |
| return 0; |
| } |
| |
| #endif /* GUARD_QUOTAIO_H */ |