/* * Single File fileSystem * * Copyright 1998 Pavel Machek, distribute under GPL * * This defines whole class of filesystems which contain single file * inside. It is somehow similar to extfs, except that extfs makes * whole virtual trees and we do only single virtual files. * * If you want to gunzip something, you should open it with #ugz * suffix, DON'T try to gunzip it yourself. * * Namespace: exports vfs_sfs_ops, shell (FIXME) */ #include #include #include #include #include #include #include #include "../src/mad.h" #include "../src/fs.h" #include #include "../src/util.h" #include "../src/main.h" #include "vfs.h" #include "local.h" /* This is needed, or libvfs.so will lack symbol shell. Should look up who requires it */ char *shell = "/bin/sh"; struct cachedfile { char *name, *cache; uid_t uid; struct cachedfile *next; }; static struct cachedfile *head; #define MAXFS 32 static int sfs_no = 0; static char *sfs_prefix[ MAXFS ]; static char *sfs_command[ MAXFS ]; static int sfs_flags[ MAXFS ]; #define F_1 1 #define F_2 2 #define F_NOLOCALCOPY 4 #define F_FULLMATCH 8 static int uptodate( char *name, char *cache ) { return 1; } static int vfmake( vfs *me, char *name, char *cache ) { char *inpath, *op; int w; char pad [10240]; char *s, *t = pad; int was_percent = 0; vfs_split( name, &inpath, &op ); if ((w = (*me->which)( me, op )) == -1) vfs_die( "This cannot happen... Hopefully.\n" ); if ((sfs_flags[w] & F_1) || (!strcmp( name, "/" ))) ; else return -1; /* if ((sfs_flags[w] & F_2) || (!inpath) || (!*inpath)); else return -1; */ if (!(sfs_flags[w] & F_NOLOCALCOPY)) name = mc_getlocalcopy( name ); else name = strdup( name ); s = sfs_command[w]; #define COPY_CHAR if (t-pad>10200) return -1; else *t++ = *s; #define COPY_STRING(a) if ((t-pad)+strlen(a)>10200) return -1; else { strcpy( t, a ); t+= strlen(a); } while (*s) { if (was_percent) { switch (*s) { case '1': COPY_STRING( name ); break; case '2': COPY_STRING( op + strlen( sfs_prefix[w] ) ); break; case '3': COPY_STRING( cache ); break; case '%': COPY_CHAR; break; } was_percent = 0; } else { if (*s == '%') was_percent = 1; else COPY_CHAR; } s++; } free( name ); if (my_system (EXECUTE_AS_SHELL | EXECUTE_SETUID, "/bin/sh", pad)) { return -1; } return 0; /* OK */ } static char *redirect( vfs *me, char *name ) { struct cachedfile *cur = head; uid_t uid = vfs_uid; char *cache, *xname; int handle; while (cur) { if ((!strcmp( name, cur->name )) && (uid == cur->uid) && (uptodate( cur->name, cur->cache ))) /* FIXME: when not uptodate, we might want to kill cache * file immediately, not to wait until timeout. */ { vfs_stamp( &vfs_sfs_ops, cur ); return cur->cache; } cur = cur->next; } cache = tempnam( NULL, "sfs" ); handle = open(cache, O_RDWR | O_CREAT | O_EXCL, 0600); if (handle == -1) return "/SOMEONE_PLAYING_DIRTY_TMP_TRICKS_ON_US"; close(handle); xname = strdup( name ); if (!vfmake( me, name, cache )) { cur = xmalloc( sizeof(struct cachedfile), "SFS cache" ); cur->name = xname; cur->cache = strdup(cache); cur->uid = uid; cur->next = head; head = cur; vfs_add_noncurrent_stamps (&vfs_sfs_ops, (vfsid) head, NULL); vfs_rm_parents (NULL); return cache; } else { free(xname); } return "/I_MUST_NOT_EXIST"; } #define REDIR path = redirect( me, path ); static void *sfs_open (vfs *me, char *path, int flags, int mode) { int *sfs_info; int fd; REDIR; fd = open (path, flags, mode); if (fd == -1) return 0; sfs_info = (int *) xmalloc (sizeof (int), "SF fs"); *sfs_info = fd; return sfs_info; } static int sfs_stat (vfs *me, char *path, struct stat *buf) { REDIR; return stat (path, buf); } static int sfs_lstat (vfs *me, char *path, struct stat *buf) { REDIR; #ifndef HAVE_STATLSTAT return lstat (path,buf); #else return statlstat (path, buf); #endif } static int sfs_chmod (vfs *me, char *path, int mode) { REDIR; return chmod (path, mode); } static int sfs_chown (vfs *me, char *path, int owner, int group) { REDIR; return chown (path, owner, group); } static int sfs_utime (vfs *me, char *path, struct utimbuf *times) { REDIR; return utime (path, times); } static int sfs_readlink (vfs *me, char *path, char *buf, int size) { REDIR; return readlink (path, buf, size); } #define CUR (*cur) static vfsid sfs_getid (vfs *me, char *path, struct vfs_stamping **parent) { /* FIXME: what should I do? */ vfs *v; vfsid id; struct vfs_stamping *par; struct cachedfile **cur = &head; while (CUR) { if ((!strcmp( path, CUR->name )) && (vfs_uid == CUR->uid)) break; CUR = CUR->next; } if (!CUR) vfs_die( "sfs_getid of noncached thingie?" ); *parent = NULL; { char *path2 = strdup( path ); v = vfs_split (path2, NULL, NULL); id = (*v->getid) (v, path2, &par); free( path2 ); } if (id != (vfsid)-1) { *parent = xmalloc (sizeof (struct vfs_stamping), "vfs stamping"); (*parent)->v = v; (*parent)->id = id; (*parent)->parent = par; (*parent)->next = NULL; } return (vfsid) cur; } static void sfs_free (vfsid id) { struct cachedfile *which = (struct cachedfile *) id; struct cachedfile **cur = &head; unlink( CUR->cache ); while (CUR) { if (CUR == which) break; CUR = CUR->next; } if (!CUR) vfs_die( "Free of thing which is unknown to me\n" ); *cur = CUR->next; } #undef CUR static void sfs_fill_names (vfs *me, void (*func)(char *)) { struct cachedfile *cur = head; while (cur){ (*func)(cur->name); cur = cur->next; } } static int sfs_nothingisopen (vfsid id) { /* FIXME: Investigate whether have to guard this like in the other VFSs (see fd_usage in extfs) -- Norbert */ return 1; } static char *sfs_getlocalcopy (vfs *me, char *path) { REDIR; return strdup (path); } static void sfs_ungetlocalcopy (vfs *me, char *path, char *local, int has_changed) { } static int sfs_init (vfs *me) { FILE *cfg = fopen( LIBDIR "extfs/sfs.ini", "r" ); if (!cfg) { fprintf( stderr, "Warning: " LIBDIR "extfs/sfs.ini not found\n" ); return 0; } sfs_no = 0; while ( sfs_no < MAXFS ) { char key[256]; char *c, *semi = NULL, flags = 0; int i; if (!fgets( key, 250, cfg )) break; if (*key == '#') continue; for (i=0; i