7 use POSIX qw(ENOENT ENOSYS EEXIST EPERM O_RDONLY O_RDWR O_APPEND O_CREAT);
8 use Fcntl qw(S_ISBLK S_ISCHR S_ISFIFO SEEK_SET S_ISREG S_ISFIFO S_IMODE);
9 my $can_syscall = eval {
10 require 'syscall.ph'; # for SYS_mknod and SYS_lchown
12 if (!$can_syscall && open my $fh, '<', '/usr/include/sys/syscall.h') {
13 my %sys = do { local $/ = undef;
14 <$fh> =~ m/\#define \s+ (\w+) \s+ (\d+)/gxms;
17 if ($sys{SYS_mknod} && $sys{SYS_lchown}) {
18 *SYS_mknod = sub { $sys{SYS_mknod} };
19 *SYS_lchown = sub { $sys{SYS_lchown} };
24 sub fixup { return "/tmp/fusetest-" . $ENV{LOGNAME} . shift }
27 my ($file) = fixup(shift);
28 my (@list) = lstat($file);
29 return -$! unless @list;
34 my ($dirname) = fixup(shift);
35 unless(opendir(DIRHANDLE,$dirname)) {
38 my (@files) = readdir(DIRHANDLE);
44 my ($file) = fixup(shift);
46 return -$! unless sysopen(FILE,$file,$mode);
52 my ($file,$bufsize,$off) = @_;
54 my ($handle) = new IO::File;
55 return -ENOENT() unless -e ($file = fixup($file));
56 my ($fsize) = -s $file;
57 return -ENOSYS() unless open($handle,$file);
58 if(seek($handle,$off,SEEK_SET)) {
59 read($handle,$rv,$bufsize);
65 my ($file,$buf,$off) = @_;
67 return -ENOENT() unless -e ($file = fixup($file));
68 my ($fsize) = -s $file;
69 return -ENOSYS() unless open(FILE,'+<',$file);
70 if($rv = seek(FILE,$off,SEEK_SET)) {
71 $rv = print(FILE $buf);
73 $rv = -ENOSYS() unless $rv;
78 sub err { return (-shift || -$!) }
80 sub x_readlink { return readlink(fixup(shift)); }
81 sub x_unlink { return unlink(fixup(shift)) ? 0 : -$!; }
83 sub x_symlink { print "symlink\n"; return symlink(shift,fixup(shift)) ? 0 : -$!; }
86 my ($old) = fixup(shift);
87 my ($new) = fixup(shift);
88 my ($err) = rename($old,$new) ? 0 : -ENOENT();
91 sub x_link { return link(fixup(shift),fixup(shift)) ? 0 : -$! }
93 return -ENOSYS() if ! $can_syscall;
94 my ($fn) = fixup(shift);
95 print "nonexistent $fn\n" unless -e $fn;
97 # perl's chown() does not chown symlinks, it chowns the symlink's
98 # target. it fails when the link's target doesn't exist, because
99 # the stat64() syscall fails.
100 # this causes error messages when unpacking symlinks in tarballs.
101 my ($err) = syscall(&SYS_lchown,$fn,$uid,$gid,$fn) ? -$! : 0;
105 my ($fn) = fixup(shift);
107 my ($err) = chmod($mode,$fn) ? 0 : -$!;
110 sub x_truncate { return truncate(fixup(shift),shift) ? 0 : -$! ; }
111 sub x_utime { return utime($_[1],$_[2],fixup($_[0])) ? 0:-$!; }
113 sub x_mkdir { my ($name, $perm) = @_; return 0 if mkdir(fixup($name),$perm); return -$!; }
114 sub x_rmdir { return 0 if rmdir fixup(shift); return -$!; }
117 return -ENOSYS() if ! $can_syscall;
118 # since this is called for ALL files, not just devices, I'll do some checks
119 # and possibly run the real mknod command.
120 my ($file, $modes, $dev) = @_;
121 $file = fixup($file);
123 if ($^O eq 'freebsd' || $^O eq 'darwin' || $^O eq 'netbsd') {
124 if (S_ISREG($modes)) {
125 open(FILE, '>', $file) || return -$!;
129 } elsif (S_ISFIFO($modes)) {
130 my ($rv) = POSIX::mkfifo($file, S_IMODE($modes));
131 return $rv ? 0 : -POSIX::errno();
134 syscall(&SYS_mknod,$file,$modes,$dev);
139 sub x_statfs {return 255,1000000,500000,1000000,500000,4096}
140 my ($mountpoint) = "";
141 $mountpoint = shift(@ARGV) if @ARGV;
143 mountpoint=>$mountpoint,
144 getattr =>"main::x_getattr",
145 readlink=>"main::x_readlink",
146 getdir =>"main::x_getdir",
147 mknod =>"main::x_mknod",
148 mkdir =>"main::x_mkdir",
149 unlink =>"main::x_unlink",
150 rmdir =>"main::x_rmdir",
151 symlink =>"main::x_symlink",
152 rename =>"main::x_rename",
153 link =>"main::x_link",
154 chmod =>"main::x_chmod",
155 chown =>"main::x_chown",
156 truncate=>"main::x_truncate",
157 utime =>"main::x_utime",
158 open =>"main::x_open",
159 read =>"main::x_read",
160 write =>"main::x_write",
161 statfs =>"main::x_statfs",