1 
2 /**
3  * \file
4  * \brief Header: Virtual File System directory structure
5  */
6 
7 
8 #ifndef MC__VFS_XDIRENTRY_H
9 #define MC__VFS_XDIRENTRY_H
10 
11 #include <stdio.h>
12 #include <sys/types.h>
13 
14 #include "lib/global.h"         /* GList */
15 #include "lib/vfs/path.h"       /* vfs_path_t */
16 
17 /*** typedefs(not structures) and defined constants **********************************************/
18 
19 #define LINK_FOLLOW 15
20 #define LINK_NO_FOLLOW -1
21 
22 /* For vfs_s_find_entry and vfs_s_find_inode */
23 #define FL_NONE 0
24 #define FL_MKDIR 1
25 #define FL_MKFILE 2
26 #define FL_DIR 4
27 
28 /* For open_super */
29 #define FL_NO_OPEN 1
30 
31 /* For vfs_s_entry_from_path */
32 #define FL_FOLLOW 1
33 #define FL_DIR 4
34 
35 #define ERRNOR(a, b) do { me->verrno = a; return b; } while (0)
36 
37 #define VFS_SUBCLASS(a) ((struct vfs_s_subclass *) (a))
38 
39 #define VFS_SUPER(a) ((struct vfs_s_super *) (a))
40 #define CONST_VFS_SUPER(a) ((const struct vfs_s_super *) (a))
41 #define VFS_ENTRY(a) ((struct vfs_s_entry *) (a))
42 #define VFS_INODE(a) ((struct vfs_s_inode *) (a))
43 
44 #define VFS_FILE_HANDLER(a) ((vfs_file_handler_t *) a)
45 #define VFS_FILE_HANDLER_SUPER(a) VFS_FILE_HANDLER (a)->ino->super
46 
47 /*** enums ***************************************************************************************/
48 
49 typedef enum
50 {
51     LS_NOT_LINEAR = 0,
52     LS_LINEAR_CLOSED = 1,
53     LS_LINEAR_OPEN = 2,
54     LS_LINEAR_PREOPEN = 3
55 } vfs_linear_state_t;
56 
57 /*** structures declarations (and typedefs of structures)*****************************************/
58 
59 /* Single connection or archive */
60 struct vfs_s_super
61 {
62     struct vfs_class *me;
63     struct vfs_s_inode *root;
64     char *name;                 /* My name, whatever it means */
65     int fd_usage;               /* Number of open files */
66     int ino_usage;              /* Usage count of this superblock */
67     gboolean want_stale;        /* If set, we do not flush cache properly */
68 #ifdef ENABLE_VFS_NET
69     vfs_path_element_t *path_element;
70 #endif                          /* ENABLE_VFS_NET */
71 };
72 
73 /*
74  * Single virtual file - directory entry.  The same inode can have many
75  * entries (i.e. hard links), but usually has only one.
76  */
77 struct vfs_s_entry
78 {
79     struct vfs_s_inode *dir;    /* Directory we are in, i.e. our parent */
80     char *name;                 /* Name of this entry */
81     struct vfs_s_inode *ino;    /* ... and its inode */
82 };
83 
84 /* Single virtual file - inode */
85 struct vfs_s_inode
86 {
87     struct vfs_s_super *super;  /* Archive the file is on */
88     struct vfs_s_entry *ent;    /* Our entry in the parent directory -
89                                    use only for directories because they
90                                    cannot be hardlinked */
91     GQueue *subdir;             /* If this is a directory, its entry. List of vfs_s_entry */
92     struct stat st;             /* Parameters of this inode */
93     char *linkname;             /* Symlink's contents */
94     char *localname;            /* Filename of local file, if we have one */
95     gint64 timestamp;           /* Subclass specific */
96     off_t data_offset;          /* Subclass specific */
97 };
98 
99 /* Data associated with an open file */
100 typedef struct
101 {
102     struct vfs_s_inode *ino;
103     off_t pos;                  /* This is for module's use */
104     int handle;                 /* This is for module's use, but if != -1, will be mc_close()d */
105     gboolean changed;           /* Did this file change? */
106     vfs_linear_state_t linear;  /* Is that file open with O_LINEAR? */
107 } vfs_file_handler_t;
108 
109 /*
110  * One of our subclasses (tar, cpio, fish, ftpfs) with data and methods.
111  * Extends vfs_class.
112  */
113 struct vfs_s_subclass
114 {
115     struct vfs_class base;      /* base class */
116 
117     GList *supers;
118     int inode_counter;
119     dev_t rdev;
120 
121     /* *INDENT-OFF* */
122     int (*init_inode) (struct vfs_class * me, struct vfs_s_inode * ino);        /* optional */
123     void (*free_inode) (struct vfs_class * me, struct vfs_s_inode * ino);       /* optional */
124     int (*init_entry) (struct vfs_class * me, struct vfs_s_entry * entry);      /* optional */
125 
126     void *(*archive_check) (const vfs_path_t * vpath);  /* optional */
127     int (*archive_same) (const vfs_path_element_t * vpath_element, struct vfs_s_super * psup,
128                          const vfs_path_t * vpath, void *cookie);
129     struct vfs_s_super *(*new_archive) (struct vfs_class * me);
130     int (*open_archive) (struct vfs_s_super * psup,
131                          const vfs_path_t * vpath, const vfs_path_element_t * vpath_element);
132     void (*free_archive) (struct vfs_class * me, struct vfs_s_super * psup);
133 
134     vfs_file_handler_t *(*fh_new) (struct vfs_s_inode * ino, gboolean changed);
135     int (*fh_open) (struct vfs_class * me, vfs_file_handler_t * fh, int flags, mode_t mode);
136     int (*fh_close) (struct vfs_class * me, vfs_file_handler_t * fh);
137     void (*fh_free) (vfs_file_handler_t * fh);
138 
139     struct vfs_s_entry *(*find_entry) (struct vfs_class * me,
140                                        struct vfs_s_inode * root,
141                                        const char *path, int follow, int flags);
142     int (*dir_load) (struct vfs_class * me, struct vfs_s_inode * ino, char *path);
143     gboolean (*dir_uptodate) (struct vfs_class * me, struct vfs_s_inode * ino);
144     int (*file_store) (struct vfs_class * me, vfs_file_handler_t * fh, char *path, char *localname);
145 
146     int (*linear_start) (struct vfs_class * me, vfs_file_handler_t * fh, off_t from);
147     ssize_t (*linear_read) (struct vfs_class * me, vfs_file_handler_t * fh, void *buf, size_t len);
148     void (*linear_close) (struct vfs_class * me, vfs_file_handler_t * fh);
149     /* *INDENT-ON* */
150 };
151 
152 /*** global variables defined in .c file *********************************************************/
153 
154 /*** declarations of public functions ************************************************************/
155 
156 /* entries and inodes */
157 struct vfs_s_inode *vfs_s_new_inode (struct vfs_class *me,
158                                      struct vfs_s_super *super, struct stat *initstat);
159 void vfs_s_free_inode (struct vfs_class *me, struct vfs_s_inode *ino);
160 
161 struct vfs_s_entry *vfs_s_new_entry (struct vfs_class *me, const char *name,
162                                      struct vfs_s_inode *inode);
163 void vfs_s_free_entry (struct vfs_class *me, struct vfs_s_entry *ent);
164 void vfs_s_insert_entry (struct vfs_class *me, struct vfs_s_inode *dir, struct vfs_s_entry *ent);
165 int vfs_s_entry_compare (const void *a, const void *b);
166 struct stat *vfs_s_default_stat (struct vfs_class *me, mode_t mode);
167 
168 struct vfs_s_entry *vfs_s_generate_entry (struct vfs_class *me, const char *name,
169                                           struct vfs_s_inode *parent, mode_t mode);
170 struct vfs_s_inode *vfs_s_find_inode (struct vfs_class *me,
171                                       const struct vfs_s_super *super,
172                                       const char *path, int follow, int flags);
173 struct vfs_s_inode *vfs_s_find_root (struct vfs_class *me, struct vfs_s_entry *entry);
174 
175 /* outside interface */
176 void vfs_init_subclass (struct vfs_s_subclass *sub, const char *name, vfs_flags_t flags,
177                         const char *prefix);
178 const char *vfs_s_get_path (const vfs_path_t * vpath, struct vfs_s_super **archive, int flags);
179 struct vfs_s_super *vfs_get_super_by_vpath (const vfs_path_t * vpath);
180 
181 void vfs_s_invalidate (struct vfs_class *me, struct vfs_s_super *super);
182 char *vfs_s_fullpath (struct vfs_class *me, struct vfs_s_inode *ino);
183 
184 void vfs_s_init_fh (vfs_file_handler_t * fh, struct vfs_s_inode *ino, gboolean changed);
185 
186 /* network filesystems support */
187 int vfs_s_select_on_two (int fd1, int fd2);
188 int vfs_s_get_line (struct vfs_class *me, int sock, char *buf, int buf_len, char term);
189 int vfs_s_get_line_interruptible (struct vfs_class *me, char *buffer, int size, int fd);
190 /* misc */
191 int vfs_s_retrieve_file (struct vfs_class *me, struct vfs_s_inode *ino);
192 
193 void vfs_s_normalize_filename_leading_spaces (struct vfs_s_inode *root_inode, size_t final_filepos);
194 
195 /*** inline functions ****************************************************************************/
196 
197 static inline void
vfs_s_store_filename_leading_spaces(struct vfs_s_entry * entry,size_t position)198 vfs_s_store_filename_leading_spaces (struct vfs_s_entry *entry, size_t position)
199 {
200     entry->ino->data_offset = (off_t) position;
201 }
202 
203 #endif
204