1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204 |
- (* Aos, Copyright 2001, Pieter Muller, ETH Zurich *)
- MODULE UnixFiles; (** AUTHOR "gf"; PURPOSE "Unix file systems" *)
- (* derived fron (SPARCOberon) Files.Mod by J. Templ 1.12. 89/14.05.93 *)
- IMPORT S := SYSTEM, Unix, Kernel, Modules, Log := KernelLog, Files, Commands, Tr := Trace;
- CONST
- NBufs = 4; Bufsize = 4096; NoDesc = -1;
- Open = 0; Create = 1; Closed = 2; (* file states *)
-
- NoKey = -1;
- CreateFlags = Unix.rdwr + Unix.creat + Unix.trunc;
-
- TraceCollection = 0;
- Trace = {};
- VAR
- tempno: INTEGER;
- openfiles: LONGINT;
-
- searchPath: ARRAY 1024 OF CHAR;
- cwd: ARRAY 256 OF CHAR;
-
- unixFS: UnixFileSystem; (* must be unique *)
- collection: Collection; (* must be unique *)
- TYPE
- Filename = ARRAY 256 OF CHAR;
-
- NameSet = OBJECT
- VAR
- name: ARRAY 64 OF CHAR;
- left, right: NameSet;
- PROCEDURE Add( CONST filename: ARRAY OF CHAR ): BOOLEAN;
- (* add filename if it not already exists. else return false *)
- BEGIN
- IF filename = name THEN RETURN FALSE END;
- IF filename < name THEN
- IF left = NIL THEN NEW( left, filename ); RETURN TRUE
- ELSE RETURN left.Add( filename )
- END
- ELSE
- IF right = NIL THEN NEW( right, filename ); RETURN TRUE
- ELSE RETURN right.Add( filename )
- END
- END
- END Add;
- PROCEDURE & Init( CONST filename: ARRAY OF CHAR );
- BEGIN
- COPY( filename, name );
- left := NIL; right := NIL
- END Init;
- END NameSet;
-
- AliasFileSystem = OBJECT (Files.FileSystem)
- VAR
- fs: UnixFileSystem;
-
- PROCEDURE & Init*( realFS: UnixFileSystem);
- BEGIN
- SELF.fs := realFS;
- END Init;
- PROCEDURE New0( name: ARRAY OF CHAR ): Files.File;
- VAR f: Files.File;
- BEGIN
- f := fs.New0( name );
- IF f # NIL THEN f.fs := SELF END;
- RETURN f;
- END New0;
- PROCEDURE Old0( name: ARRAY OF CHAR ): Files.File;
- VAR f: Files.File;
- BEGIN
- f := fs.Old0( name );
- IF f # NIL THEN f.fs := SELF END;
- RETURN f;
- END Old0;
- PROCEDURE Delete0( name: ARRAY OF CHAR; VAR key, res: LONGINT );
- BEGIN
- fs.Delete0( name, key, res );
- END Delete0;
- PROCEDURE Rename0( old, new: ARRAY OF CHAR; fold: Files.File; VAR res: LONGINT );
- BEGIN
- fs.Rename0( old, new, fold, res );
- END Rename0;
- PROCEDURE Enumerate0( mask: ARRAY OF CHAR; flags: SET; enum: Files.Enumerator );
- BEGIN
- fs.Enumerate0( mask, flags, enum );
- END Enumerate0;
- PROCEDURE FileKey( name: ARRAY OF CHAR ): LONGINT;
- VAR
- BEGIN
- RETURN fs.FileKey( name );
- END FileKey;
- PROCEDURE CreateDirectory0( name: ARRAY OF CHAR; VAR res: LONGINT );
- BEGIN
- fs.CreateDirectory0( name, res );
- END CreateDirectory0;
- PROCEDURE RemoveDirectory0( name: ARRAY OF CHAR; force: BOOLEAN; VAR key, res: LONGINT );
- BEGIN
- fs.RemoveDirectory0( name, force, key, res );
- END RemoveDirectory0;
-
- PROCEDURE Has(CONST name: ARRAY OF CHAR; VAR fullName: ARRAY OF CHAR; VAR flags: SET): BOOLEAN;
- BEGIN
- RETURN fs.Has(name, fullName, flags);
- END Has;
- END AliasFileSystem;
-
-
- FinalizeFiles = OBJECT
- PROCEDURE EnumFile( f: ANY; VAR cont: BOOLEAN );
- VAR F: File;
- BEGIN
- F := f( File ); F.Finalize(); cont := TRUE
- END EnumFile;
- END FinalizeFiles;
-
- SearchByFstat = OBJECT
- VAR
- found: File;
- stat: Unix.Status;
-
- PROCEDURE Init( s: Unix.Status );
- BEGIN
- found := NIL;
- stat := s;
- END Init;
- PROCEDURE EnumFile( f: ANY; VAR cont: BOOLEAN );
- BEGIN
- WITH f: File DO
- IF (stat.ino = f.ino) & (stat.dev = f.dev) THEN
- (* possible different name but same file! *)
- ResetBuffers( f, stat );
- found := f; cont := FALSE;
- END;
- END;
- END EnumFile;
- END SearchByFstat;
-
-
- Collection = OBJECT (* methods in Collection shared by objects Filesystem and File *)
- VAR oldFiles, newFiles: Kernel.FinalizedCollection;
- ssearch: SearchByFstat;
-
- PROCEDURE & Init*;
- BEGIN
- NEW( oldFiles ); NEW( newFiles ); NEW( ssearch );
- END Init;
- PROCEDURE AddNew( F: File );
- BEGIN {EXCLUSIVE}
- IF TraceCollection IN Trace THEN Log.String( "Collections.AddNew: " ); Log.String( F.workName ); Log.Ln; END;
- newFiles.Add( F, FinalizeFile );
- END AddNew;
- PROCEDURE AddOld( F: File );
- BEGIN {EXCLUSIVE}
- IF TraceCollection IN Trace THEN Log.String( "Collections.AddOld: " ); Log.String( F.workName ); Log.Ln; END;
- oldFiles.Add( F, FinalizeFile );
- END AddOld;
- PROCEDURE ByStat(CONST stat: Unix.Status): File;
- BEGIN{EXCLUSIVE}
- ssearch.Init(stat);
- oldFiles.EnumerateN(ssearch.EnumFile);
- IF ssearch.found = NIL THEN
- newFiles.EnumerateN(ssearch.EnumFile)
- END;
-
- IF TraceCollection IN Trace THEN
- Log.String( "Collections.ByStatus: " ); Log.Ln;
- IF ssearch.found = NIL THEN Log.String("not found") ELSE Log.String("found") END;
- END;
-
- RETURN ssearch.found;
- END ByStat;
-
- PROCEDURE Finalize;
- VAR fin: FinalizeFiles;
- BEGIN {EXCLUSIVE}
- IF TraceCollection IN Trace THEN Log.String( "Collections.Finalize " ); Log.Ln; END;
- NEW( fin );
- newFiles.Enumerate( fin.EnumFile ); newFiles.Clear();
- oldFiles.Enumerate( fin.EnumFile ); oldFiles.Clear();
- END Finalize;
- PROCEDURE FinalizeFile( obj: ANY );
- VAR F: File;
- BEGIN
- F := obj( File );
- IF TraceCollection IN Trace THEN Log.String( "Collections.FinalizeFile " ); Log.String( F.workName ); Log.Ln; END;
- F.Finalize()
- END FinalizeFile;
- END Collection;
- UnixFileSystem* = OBJECT (Files.FileSystem)
- PROCEDURE & Init;
- BEGIN
- prefix := ""; vol := NIL; desc := "UnixFS"
- END Init;
- PROCEDURE New0*( name: ARRAY OF CHAR ): Files.File;
- VAR f: File; stat: Unix.Status;
- res,err: LONGINT;
- path,nameonly: ARRAY 512 OF CHAR;
- BEGIN {EXCLUSIVE}
- (*AwaitFinalizingDone;*)
-
- (*first check if the path actually exits first using fstat. fstat returns -1 and sets erno to ENOENT when a component of the path doesn't exist or the entire path is empty*)
- Files.SplitPath(name,path, nameonly);
- res:=Unix.stat( ADDRESSOF( path ), stat ) ;
- err:=Unix.errno();
- IF (name="") OR (path="") OR (res>=0) OR (err#Unix.ENOENT) THEN
- NEW( f, SELF );
- f.workName := ""; COPY( name, f.registerName );
- f.fd := NoDesc; f.state := Create; f.fsize := 0; f.fpos := 0;
- f.swapper := -1; (*all f.buf[i] = NIL*)
- f.key := NoKey; f.fs := SELF;
- RETURN f;
- ELSE
- Log.String( "UnixFileSystem.New0: file allocation failed. Probably a nonexistent path." ); Log.Ln;
- RETURN NIL;
- END;
- END New0;
-
-
- PROCEDURE IsDirectory( VAR stat: Unix.Status ): BOOLEAN;
- VAR mode: LONGINT;
- BEGIN
- mode := stat.mode;
- RETURN ODD( mode DIV 4000H )
- END IsDirectory;
- PROCEDURE Old0*( name: ARRAY OF CHAR ): Files.File;
- VAR f: File; stat: Unix.Status; fd, r, pos: LONGINT;
- oflags: SET; nextdir, path: Filename;
- BEGIN {EXCLUSIVE}
- IF name = "" THEN RETURN NIL END;
-
- IF IsFullName( name ) THEN
- COPY( name, path ); nextdir := "";
- ELSE
- pos := 0; ScanPath( pos, nextdir ); MakePath( nextdir, name, path );
- ScanPath( pos, nextdir )
- END;
-
- LOOP
- f := NIL;
- r := Unix.access( ADDRESSOF( path ), Unix.R_OK );
- IF r >= 0 THEN
- r := Unix.access( ADDRESSOF( path ), Unix.W_OK );
- IF r < 0 THEN oflags := Unix.rdonly ELSE oflags := Unix.rdwr END;
- fd := UnixOpen( ADDRESSOF( path ), oflags, {} );
- IF fd >= 0 THEN
- r := Unix.fstat( fd, stat );
- f := collection.ByStat(stat);
- IF f # NIL THEN
- (* use the file already cached *) r := Unix.close( fd ); EXIT
- ELSE
- NEW( f, SELF );
- f.fd := fd; f.dev := stat.dev; f.ino := stat.ino;
- f.mtime := stat.mtime.sec; f.fsize := stat.size; f.fpos := 0;
- f.state := Open; f.swapper := -1; (*all f.buf[i] = NIL*)
- COPY( path, f.workName ); f.registerName := "";
- f.tempFile := FALSE;
- IF IsDirectory( stat ) THEN
- f.flags := {Files.Directory, Files.ReadOnly}
- ELSIF oflags = Unix.rdonly THEN
- f.flags := {Files.ReadOnly}
- END;
- f.key := NoKey; f.fs := SELF;
- IncOpenFiles();
- collection.AddOld(f);
- EXIT
- END
- ELSE
- (* file exists but open failed *) f := NIL; EXIT
- END
- ELSIF nextdir # "" THEN
- MakePath( nextdir, name, path ); ScanPath( pos, nextdir );
- ELSE
- f := NIL; EXIT
- END;
- END; (* loop *)
- RETURN f
- END Old0;
- (** Return the unique non-zero key of the named file, if it exists. *)
- PROCEDURE FileKey*( name: ARRAY OF CHAR ): LONGINT;
- (* Can not be used for Unix files as LONGINT is too small.
- In the Unix filesystem a file is identified by
- - dev (64 bit (Linux), 32 bit (Solaris, Darwin)) +
- - ino (32 bit)
- *)
- BEGIN
- RETURN 0
- END FileKey;
- PROCEDURE Delete0*( name: ARRAY OF CHAR; VAR key, res: LONGINT );
- VAR r: LONGINT;
- BEGIN {EXCLUSIVE}
- r := Unix.unlink( ADDRESSOF( name ) );
- IF r = 0 THEN res := Files.Ok
- ELSE res := Unix.errno( )
- END;
- key := 0;
- END Delete0;
-
-
- (* return remaining old file, if any *)
- PROCEDURE TryRename*( old, new: ARRAY OF CHAR; f: Files.File; VAR res: LONGINT ): LONGINT;
- CONST Bufsize = 4096;
- VAR fdold, fdnew, fo, n, r: LONGINT; ostat, nstat: Unix.Status;
- buf: ARRAY Bufsize OF CHAR;
- BEGIN {EXCLUSIVE}
- fo := NoDesc;
- r:= Unix.stat( ADDRESSOF( old ), ostat );
- IF r >= 0 THEN
- r := Unix.stat( ADDRESSOF( new ), nstat );
- IF (r >= 0) & ((ostat.dev # nstat.dev) OR (ostat.ino # nstat.ino)) THEN
- r := Unix.unlink( ADDRESSOF( new ) ) (* work around stale nfs handles *);
- END;
- r := Unix.rename( ADDRESSOF( old ), ADDRESSOF( new ) );
- IF r < 0 THEN (* could not rename, try copy *)
- res := Unix.errno( );
- IF (res = Unix.EXDEV) OR (res = Unix.ETXTBSY) THEN (* cross device link, move the file / file busy frequently happens in VirtualBox *)
- fdold := UnixOpen( ADDRESSOF( old ), Unix.rdonly, {} );
- fo := fdold;
- IF fdold < 0 THEN
- res := Unix.errno( );
- RETURN NoDesc;
- END;
- fdnew := UnixOpen( ADDRESSOF( new ), Unix.rdwr + Unix.creat + Unix.trunc, Unix.rwrwr );
- IF fdnew < 0 THEN
- res := Unix.errno( );
- RETURN NoDesc;
- END;
- REPEAT
- n := UnixRead( fdold, ADDRESSOF( buf ), Bufsize );
- IF n > 0 THEN
- r := UnixWrite( fdnew, ADDRESSOF( buf ), n );
- IF r < 0 THEN
- r := Unix.close( fdold );
- r := Unix.close( fdnew );
- RETURN NoDesc;
- END;
- END
- UNTIL n = 0;
- r := Unix.unlink( ADDRESSOF( old ) );
- r := Unix.close( fdold );
- r := Unix.close( fdnew );
- res := Files.Ok
- ELSE
- RETURN NoDesc (* res is Unix.rename return code *)
- END
- END;
- res := Files.Ok
- ELSE
- res := Unix.errno();
- END;
- RETURN fo;
- END TryRename;
- PROCEDURE Rename0*( old, new: ARRAY OF CHAR; f: Files.File; VAR res: LONGINT );
- VAR of: LONGINT;
- BEGIN
- of := TryRename(old, new, f, res);
- END Rename0;
-
- PROCEDURE CreateDirectory0*( path: ARRAY OF CHAR; VAR res: LONGINT );
- VAR r: LONGINT;
- BEGIN {EXCLUSIVE}
- r := Unix.mkdir( ADDRESSOF( path ), Unix.rwxrwxrwx );
- IF r = 0 THEN res := Files.Ok
- ELSE res := Unix.errno( )
- END
- END CreateDirectory0;
-
- PROCEDURE RemoveDirectory0*( path: ARRAY OF CHAR; force: BOOLEAN; VAR key, res: LONGINT );
- VAR r: LONGINT;
- BEGIN {EXCLUSIVE}
- r := Unix.rmdir( ADDRESSOF( path ) );
- IF r = 0 THEN res := Files.Ok
- ELSE res := Unix.errno( )
- END
- END RemoveDirectory0;
- PROCEDURE Enumerate0*( mask: ARRAY OF CHAR; flags: SET; enum: Files.Enumerator );
- VAR
- path, filemask: Filename;
- isPath: BOOLEAN;
- i, j: INTEGER; dirName, fileName, fullName, xName: Filename;
- checkSet: NameSet; ent: Unix.Dirent;
-
- PROCEDURE GetEntryName;
- VAR i: INTEGER; adr: ADDRESS;
- BEGIN
- i := -1; adr := ADDRESSOF( ent.name );
- REPEAT INC( i ); S.GET( adr, fileName[i] ); INC( adr ) UNTIL fileName[i] = 0X
- END GetEntryName;
-
- PROCEDURE EnumDir( CONST dirName: ARRAY OF CHAR );
- VAR
- dir: ADDRESS;
- tm: Unix.TmPtr; date, time: LONGINT;
- stat: Unix.Status; r: LONGINT;
- BEGIN
- dir := Unix.opendir( ADDRESSOF( dirName ) );
- IF dir # 0 THEN
- ent := Unix.readdir( dir );
- WHILE ent # NIL DO
- COPY( dirName, fullName );
-
- GetEntryName; AppendName( fullName, fileName );
- IF (fileName[0] # '.') & Match( fileName, filemask, 0, 0 ) THEN
- IF checkSet.Add( fileName ) THEN (* not a covered name *)
- r := Unix.stat( ADDRESSOF( fullName ), stat );
- tm := Unix.localtime( stat.mtime );
- date := tm.year*200H + (tm.mon + 1)*20H + tm.mday;
- time := tm.hour*1000H + tm.min*40H + tm.sec;
- flags := {};
- IF IsDirectory( stat ) THEN
- flags := {Files.ReadOnly, Files.Directory}
- ELSE
- r := Unix.access( ADDRESSOF( fullName ), Unix.W_OK );
- IF r < 0 THEN flags := {Files.ReadOnly} END
- END;
- r := Unix.realpath(ADDRESSOF(fullName), ADDRESSOF(xName));
- IF (r # 0) THEN COPY(xName, fullName) END;
- enum.PutEntry( fullName, flags, time, date, stat.size );
- END
- END;
- ent := Unix.readdir( dir );
- END;
- Unix.closedir( dir )
- END;
- END EnumDir;
-
- BEGIN {EXCLUSIVE}
- Files.SplitName( mask, prefix, fullName );
- Files.SplitPath( fullName, path, filemask );
- NEW( checkSet, "M###N" );
- isPath:= path#"";
- IF isPath THEN
- CleanPath(path); (*get rid of xxx/../xxx and xxx/./xxx in the path string*)
- END;
- IF isPath & (path[0] = '/') THEN (*check for absolute path*)
- EnumDir( path);
- ELSE (*no path or relative path*)
- i := 0; j := 0;
- LOOP (*go through the search paths, every time a complete search path has been traversed, look for the element there*)
- IF (searchPath[i] = " ") OR (searchPath[i] = 0X) THEN
- dirName[j] := 0X;
- IF isPath THEN (*if relative path: add relative path to the current search path*)
- Files.JoinPath(dirName, path, dirName);
- END;
- EnumDir( dirName );
- IF searchPath[i] = 0X THEN EXIT
- ELSE INC( i ); j := 0
- END
- ELSE
- dirName[j] := searchPath[i]; INC( j ); INC( i )
- END
- END
- END;
- checkSet := NIL;
- END Enumerate0;
-
- PROCEDURE Has(CONST name: ARRAY OF CHAR; VAR fullName: ARRAY OF CHAR; VAR flags: SET): BOOLEAN;
- VAR r: LONGINT;stat: Unix.Status;
- BEGIN
- r := Unix.stat( ADDRESSOF( name ), stat );
- IF r # 0 THEN RETURN FALSE END;
- flags := {};
- IF IsDirectory( stat ) THEN
- flags := {Files.ReadOnly, Files.Directory}
- ELSE
- r := Unix.access( ADDRESSOF( name ), Unix.W_OK );
- IF r < 0 THEN flags := {Files.ReadOnly} END
- END;
- r := Unix.realpath(ADDRESSOF(name), ADDRESSOF(fullName));
- IF (r = 0) THEN COPY(name, fullName) END; (* no success *)
- RETURN TRUE;
- END Has;
- END UnixFileSystem;
-
-
- Buffer = POINTER TO RECORD (Files.Hint)
- chg: BOOLEAN;
- org, size: LONGINT;
- data: ARRAY Bufsize OF CHAR;
- END;
-
- File* = OBJECT (Files.File)
- VAR
- fd: LONGINT;
- workName, registerName: Filename;
- tempFile: BOOLEAN;
- dev: Unix.DevT;
- ino: LONGINT;
- mtime: HUGEINT;
- fsize, fpos: SIZE;
- bufs: ARRAY NBufs OF Buffer;
- swapper, state: LONGINT;
- PROCEDURE & Init( fs: Files.FileSystem );
- BEGIN
- SELF.fs := fs; flags := {};
- END Init;
- PROCEDURE CreateUnixFile;
- VAR
- stat: Unix.Status; errno: LONGINT;
- BEGIN
- IF state = Create THEN
- GetTempName( registerName, workName ); tempFile := TRUE
- ELSIF state = Closed THEN
- IF registerName # "" THEN
- workName := registerName; registerName := ""; tempFile := FALSE;
- ELSE
- RETURN;
- END;
- END;
- errno := Unix.unlink( ADDRESSOF( workName ) );
- (*unlink first to avoid stale NFS handles and to avoid reuse of inodes*)
-
- fd := UnixOpen( ADDRESSOF( workName ), CreateFlags, Unix.rwrwr );
- IF fd >= 0 THEN
- errno := Unix.fstat( fd, stat );
- dev := stat.dev; ino := stat.ino; mtime := stat.mtime.sec;
- state := Open; fpos := 0;
- IncOpenFiles();
- collection.AddNew(SELF);
- ELSE
- Halt( SELF, TRUE, "UnixFiles.File.Create: open failed" );
- END
- END CreateUnixFile;
-
-
- PROCEDURE Flush( buf: Buffer );
- VAR res: LONGINT; stat: Unix.Status;
- BEGIN
- IF buf.chg THEN
- IF fd = NoDesc THEN CreateUnixFile END;
- IF buf.org # fpos THEN
- IF Unix.lseek( fd, buf.org, 0 ) = -1 THEN
- Halt( SELF, TRUE, "UnixFiles.File.Flush: lseek failed" )
- END
- END;
- res := UnixWrite( fd, ADDRESSOF( buf.data ), buf.size );
- IF res < 0 THEN Halt( SELF, TRUE, "UnixFiles.File.Flush: write failed" ) END;
- fpos := buf.org + buf.size; buf.chg := FALSE;
- res := Unix.fstat( fd, stat ); mtime := stat.mtime.sec
- END
- END Flush;
-
-
- PROCEDURE Set*( VAR r: Files.Rider; pos: LONGINT );
- BEGIN {EXCLUSIVE}
- SetX( r, pos )
- END Set;
-
- PROCEDURE SetX( VAR r: Files.Rider; p: LONGINT );
- VAR org, offset, i, n, res: LONGINT; buf: Buffer;
- BEGIN
- IF p > fsize THEN p := LONGINT(fsize)
- ELSIF p < 0 THEN p := 0
- END;
- offset := p MOD Bufsize; org := p - offset;
- i := 0;
- WHILE (i < NBufs) & (bufs[i] # NIL) & (org # bufs[i].org) DO INC( i ) END;
- IF i < NBufs THEN
- IF bufs[i] = NIL THEN
- NEW( buf ); buf.chg := FALSE; buf.org := -1;
- bufs[i] := buf
- ELSE
- swapper := i;
- buf := bufs[swapper]; Flush( buf )
- END
- ELSE
- swapper := (swapper + 1) MOD NBufs;
- buf := bufs[swapper]; Flush( buf )
- END;
- IF buf.org # org THEN
- IF org = fsize THEN
- buf.size := 0
- ELSE
- IF fd = NoDesc THEN CreateUnixFile END;
- IF fpos # org THEN
- IF Unix.lseek( fd, org, 0 ) = -1 THEN
- Halt( SELF, TRUE, "UnixFiles.File.Set: lseek failed" )
- END
- END;
- n := UnixRead( fd, ADDRESSOF( buf.data ), Bufsize );
- IF n < 0 THEN
- IF p < fsize THEN Halt( SELF, TRUE, "UnixFiles.File.Set: read failed" )
- ELSE n := 0
- END
- END;
- fpos := org + n; buf.size := n
- END;
- buf.org := org; buf.chg := FALSE
- ELSE
- org := buf.org
- END;
- r.hint := buf; r.apos := org; r.bpos := offset;
- r.res := 0; r.eof := FALSE;
- r.file := SELF; r.fs := fs
- END SetX;
-
- PROCEDURE Pos*( VAR r: Files.Rider ): LONGINT;
- BEGIN
- RETURN r.apos + r.bpos
- END Pos;
- PROCEDURE Read*( VAR r: Files.Rider; VAR x: CHAR );
- VAR offset: LONGINT; buf: Buffer;
- BEGIN {EXCLUSIVE}
- buf := r.hint(Buffer); offset := r.bpos;
- IF r.apos # buf.org THEN
- SetX( r, r.apos + offset );
- buf := r.hint(Buffer); offset := r.bpos
- END;
- IF (offset < buf.size) THEN
- x := buf.data[offset]; r.bpos := offset + 1
- ELSIF r.apos + offset < fsize THEN
- SetX( r, r.apos + offset );
- x := r.hint(Buffer).data[0]; r.bpos := 1
- ELSE
- x := 0X; r.eof := TRUE
- END
- END Read;
- PROCEDURE ReadBytes*( VAR r: Files.Rider; VAR x: ARRAY OF CHAR; ofs, len: LONGINT );
- VAR xpos, min, restInBuf, offset: LONGINT; buf: Buffer;
- BEGIN {EXCLUSIVE}
- x[ofs] := 0X; xpos := ofs;
- buf := r.hint(Buffer); offset := r.bpos;
- WHILE len > 0 DO
- IF (r.apos # buf.org) OR (offset >= Bufsize) THEN
- SetX( r, r.apos + offset );
- buf := r.hint(Buffer); offset := r.bpos
- END;
- restInBuf := buf.size - offset;
- IF restInBuf = 0 THEN r.res := len; r.eof := TRUE; RETURN
- ELSIF len > restInBuf THEN min := restInBuf
- ELSE min := len
- END;
- S.MOVE( ADDRESSOF( buf.data ) + offset, ADDRESSOF( x ) + xpos, min );
- INC( offset, min ); r.bpos := offset;
- INC( xpos, min ); DEC( len, min )
- END;
- r.res := 0; r.eof := FALSE;
- END ReadBytes;
-
-
- PROCEDURE Write*( VAR r: Files.Rider; x: CHAR );
- VAR buf: Buffer; offset: LONGINT;
- BEGIN {EXCLUSIVE}
- buf := r.hint(Buffer); offset := r.bpos;
- IF (r.apos # buf.org) OR (offset >= Bufsize) THEN
- SetX( r, r.apos + offset );
- buf := r.hint(Buffer); offset := r.bpos
- END;
- buf.data[offset] := x; buf.chg := TRUE;
- IF offset = buf.size THEN INC( buf.size ); INC( fsize ) END;
- r.bpos := offset + 1; r.res := Files.Ok
- END Write;
- PROCEDURE WriteBytes*( VAR r: Files.Rider; CONST x: ARRAY OF CHAR; ofs, len: LONGINT );
- VAR xpos, min, restInBuf, offset: LONGINT; buf: Buffer;
- BEGIN {EXCLUSIVE}
- xpos := ofs; buf := r.hint(Buffer); offset := r.bpos;
- WHILE len > 0 DO
- IF (r.apos # buf.org) OR (offset >= Bufsize) THEN
- SetX( r, r.apos + offset );
- buf := r.hint(Buffer); offset := r.bpos
- END;
- restInBuf := Bufsize - offset;
- IF len > restInBuf THEN min := restInBuf ELSE min := len END;
- S.MOVE( ADDRESSOF( x ) + xpos, ADDRESSOF( buf.data ) + offset, min );
- INC( offset, min ); r.bpos := offset;
- IF offset > buf.size THEN
- INC( fsize, offset - buf.size ); buf.size := offset
- END;
- INC( xpos, min ); DEC( len, min ); buf.chg := TRUE
- END;
- r.res := Files.Ok
- END WriteBytes;
-
- PROCEDURE Length*( ): LONGINT;
- BEGIN
- RETURN LONGINT(fsize)
- END Length;
- PROCEDURE GetDate*( VAR t, d: LONGINT );
- VAR stat: Unix.Status; r: LONGINT; time: Unix.TmPtr;
- BEGIN {EXCLUSIVE}
- IF fd = NoDesc THEN CreateUnixFile END;
- r := Unix.fstat( fd, stat );
- time := Unix.localtime( stat.mtime );
- t := time.sec + ASH( time.min, 6 ) + ASH( time.hour, 12 );
- d := time.mday + ASH( time.mon + 1, 5 ) + ASH( time.year, 9 );
- END GetDate;
- PROCEDURE SetDate*( t, d: LONGINT );
- TYPE
- Time = RECORD actime, modtime: LONGINT END;
- VAR
- tm: Unix.Tm; buf: Time; r: LONGINT; path: Filename;
- BEGIN {EXCLUSIVE}
- IF registerName # "" THEN COPY( registerName, path )
- ELSE COPY( workName, path )
- END;
- (* get year and timezone *)
- (* fill in new date *)
- tm.isdst := -1; tm.sec := t MOD 64; tm.min := t DIV 64 MOD 64;
- tm.hour := t DIV 4096 MOD 32;
- tm.mday := d MOD 32; tm.mon := d DIV 32 MOD 16 - 1; tm.year := d DIV 512;
- tm.wday := 0; tm.yday := 0;
- buf.actime := Unix.mktime( tm ); buf.modtime := buf.actime;
- r := Unix.utime( ADDRESSOF( path ), ADDRESSOF( buf ) );
- END SetDate;
- PROCEDURE GetAttributes*( ): SET;
- BEGIN {EXCLUSIVE}
- RETURN flags
- END GetAttributes;
- PROCEDURE SetAttributes*( attr: SET );
- BEGIN {EXCLUSIVE}
- (* flags := attr *)
- END SetAttributes;
- PROCEDURE Register0*( VAR res: LONGINT );
- VAR fo, r: LONGINT;
- BEGIN {EXCLUSIVE}
- IF (state = Create) & (registerName # "") THEN
- state := Closed (* shortcut renaming *) ;
- END;
-
- FlushBuffers;
-
- IF registerName # "" THEN
- fo := unixFS.TryRename( workName, registerName, SELF, res );
- IF res # Files.Ok THEN
- Halt( SELF, FALSE, "UnixFiles.File.Register: rename failed" )
- END;
- IF fo # NoDesc THEN (* SELF still refers to old file *)
- r := Unix.close( fd ); (* VirtualBox ! Can only delete file when closed. *)
- r := Unix.unlink( ADDRESSOF( workName ) );
- fd := UnixOpen( ADDRESSOF( registerName ), Unix.rdwr, Unix.rwrwr );
- END;
- workName := registerName; registerName := ""; tempFile := FALSE;
- END;
- END Register0;
-
- PROCEDURE Update*;
- BEGIN {EXCLUSIVE}
- FlushBuffers
- END Update;
-
-
- PROCEDURE FlushBuffers;
- VAR i: LONGINT;
- BEGIN
- IF fd = NoDesc THEN CreateUnixFile END;
- FOR i := 0 TO NBufs - 1 DO
- IF bufs[i] # NIL THEN Flush( bufs[i] ) END
- END;
- END FlushBuffers;
-
-
- PROCEDURE Finalize*;
- VAR r: LONGINT;
- BEGIN {EXCLUSIVE}
- IF tempFile THEN
- IF fd # NoDesc THEN r := Unix.close( fd ) END;
- r := Unix.unlink( ADDRESSOF( workName ) );
- fd := NoDesc;
- ELSE
- FlushBuffers;
- IF fd # NoDesc THEN
- r := Unix.close( fd );
- fd := NoDesc;
- END;
- END;
- DecOpenFiles();
- state := Closed;
- END Finalize;
-
- PROCEDURE Close;
- BEGIN
- Finalize;
- collection.newFiles.Remove(SELF);
- collection.oldFiles.Remove(SELF);
- END Close;
-
- PROCEDURE GetName*( VAR name: ARRAY OF CHAR );
- BEGIN {EXCLUSIVE}
- IF registerName = "" THEN COPY( workName, name ) ;
- ELSE COPY( registerName, name )
- END;
- CleanPath( name )
- END GetName;
- END File;
-
-
-
- (*===================================================================*)
- (** Get the current directory. *)
- PROCEDURE GetWorkingDirectory*( VAR path: ARRAY OF CHAR );
- BEGIN
- COPY( cwd, path )
- END GetWorkingDirectory;
-
- (** Change to directory path. *)
- PROCEDURE ChangeDirectory*( CONST path: ARRAY OF CHAR; VAR done: BOOLEAN );
- VAR r: LONGINT; newdir: Filename;
- BEGIN
- IF path[0] # '/' THEN
- COPY( cwd, newdir ); AppendName( newdir, path );
- CleanPath( newdir )
- ELSE
- COPY( path, newdir );
- END;
- r := Unix.chdir( ADDRESSOF( newdir ) );
- IF r = 0 THEN COPY( newdir, cwd ); done := TRUE ELSE done := FALSE END
- END ChangeDirectory;
- (*===================================================================*)
-
- PROCEDURE StripPath*( CONST path: ARRAY OF CHAR; VAR name: ARRAY OF CHAR );
- VAR i, p: INTEGER; c: CHAR;
- BEGIN
- i := 0; p := 0;
- REPEAT
- IF path[i] = '/' THEN p := i + 1 END;
- INC( i )
- UNTIL path[i] = 0X;
- i := 0;
- REPEAT c := path[p]; name[i] := c; INC( i ); INC( p ) UNTIL c = 0X
- END StripPath;
-
-
- PROCEDURE CleanPath*( VAR path: ARRAY OF CHAR );
- (*
- /aaa/../bbb/./ccc/../ddd/. ==> /bbb/ddd
- ../aaa ==> CWD/../aaa ==> . . .
- *)
- VAR
- i, prevNameStart, nameStart: INTEGER;
- c1, c2, c3: CHAR;
- PROCEDURE prependCWD;
- VAR tmp: ARRAY 256 OF CHAR;
- BEGIN
- COPY( cwd, tmp ); AppendName( tmp, path ); COPY( tmp, path )
- END prependCWD;
-
- PROCEDURE restart;
- BEGIN
- IF path[0] = '/' THEN nameStart := 1 ELSE nameStart := 0 END;
- i := -1; prevNameStart := -1;
- END restart;
- PROCEDURE shift( p0, p1: INTEGER );
- VAR c: CHAR;
- BEGIN
- REPEAT c := path[p1]; path[p0] := c; INC( p0 ); INC( p1 ) UNTIL c = 0X;
- IF p0 > 1 THEN restart ELSE i := 0 END
- END shift;
- BEGIN
- restart;
- REPEAT
- INC( i );
- IF i = nameStart THEN
- c1 := path[i]; c2 := path[i + 1]; c3 := path[i + 2];
- IF c1 = '/' THEN shift( i, i + 1 ) (* // *)
- ELSIF c1 = '.' THEN
- IF c2 = 0X THEN
- IF i > 1 THEN DEC( i ) END;
- path[i] := 0X
- ELSIF c2 = '/' THEN shift( i, i + 2 ); (* ./ *)
- ELSIF (c2 = '.') & ((c3 = 0X) OR (c3 = '/')) THEN (* .. *)
- IF i = 0 THEN prependCWD; restart
- ELSIF c3 = 0X THEN DEC( i ); path[i] := 0X
- ELSIF c3 = '/' THEN (* ../ *)
- IF prevNameStart >= 0 THEN shift( prevNameStart, i + 3 ) END
- END
- END
- END
- ELSIF path[i] = '/' THEN
- IF i > 0 THEN prevNameStart := nameStart END;
- nameStart := i + 1
- END;
- UNTIL (i >= 0) & (path[i] = 0X);
- IF (i > 1) & (path[i - 1] = '/') THEN path[i - 1] := 0X END;
- IF path = "" THEN COPY( cwd, path ) END;
- END CleanPath;
- PROCEDURE Match( CONST name, pat: ARRAY OF CHAR; i, j: INTEGER ): BOOLEAN;
- BEGIN
- IF (name[i] = 0X) & (pat[j] = 0X) THEN RETURN TRUE
- ELSIF pat[j] # "*" THEN RETURN (name[i] = pat[j]) & Match( name, pat, i + 1, j + 1 )
- ELSE (* pat[j] = "*", name[i] may be 0X *)
- RETURN Match( name, pat, i, j + 1 ) OR ((name[i] # 0X) & Match( name, pat, i + 1, j ))
- END
- END Match;
-
-
- PROCEDURE Append( VAR a: Filename; CONST this: ARRAY OF CHAR );
- VAR i, j: LONGINT;
- BEGIN
- i := 0; j := 0;
- WHILE a[i] # 0X DO INC( i ) END;
- WHILE (i < LEN( a ) - 1) & (this[j] # 0X) DO a[i] := this[j]; INC( i ); INC( j ) END;
- a[i] := 0X
- END Append;
- PROCEDURE AppendName( VAR path: Filename; CONST filename: ARRAY OF CHAR );
- VAR i, j, max: LONGINT;
- BEGIN
- i := 0; j := 0; max := LEN( path ) - 1;
- WHILE path[i] # 0X DO INC( i ) END;
- IF (i > 0) & (path[i - 1] # "/") THEN path[i] := "/"; INC( i ); path[i] := 0X END;
- Append( path, filename );
- END AppendName;
-
-
- PROCEDURE AppendInt( VAR str: Filename; n: LONGINT );
- VAR i: LONGINT;
- BEGIN
- i := 0;
- WHILE str[i] # 0X DO INC(i) END;
- WHILE n > 0 DO str[i] := CHR( n MOD 10 + ORD('0') ); n := n DIV 10; INC(i) END;
- str[i] := 0X
- END AppendInt;
- PROCEDURE IsFullName( CONST name: ARRAY OF CHAR ): BOOLEAN;
- VAR i: INTEGER; ch: CHAR;
- BEGIN
- i := 0; ch := name[0];
- WHILE (ch # 0X) & (ch # "/") DO INC( i ); ch := name[i] END;
- RETURN ch = "/"
- END IsFullName;
- PROCEDURE Halt( f: File; unixError: BOOLEAN; CONST msg: ARRAY OF CHAR );
- VAR fd, errno: LONGINT;
- workName, registerName: Filename;
- BEGIN
- IF f = NIL THEN
- workName := "???"; registerName := "???"
- ELSE
- workName := f.workName; registerName := f.registerName; fd := f.fd
- END;
- IF unixError THEN errno := Unix.errno( ); Unix.Perror( msg ) END;
- HALT( 99 )
- END Halt;
- PROCEDURE ResetBuffers( f: File; VAR stat: Unix.Status );
- VAR i: INTEGER;
- BEGIN
- f.fsize := stat.size;
- IF (f.mtime # stat.mtime.sec) THEN
- FOR i := 0 TO NBufs - 1 DO
- IF f.bufs[i] # NIL THEN f.bufs[i].org := -1; f.bufs[i] := NIL END;
- END;
- f.swapper := -1; f.mtime := stat.mtime.sec
- END
- END ResetBuffers;
- PROCEDURE MakePath( CONST dir, name: ARRAY OF CHAR; VAR dest: ARRAY OF CHAR );
- VAR i, j: INTEGER;
- BEGIN
- i := 0; j := 0;
- WHILE dir[i] # 0X DO dest[i] := dir[i]; INC( i ) END;
- IF (i>0) & (dest[i - 1] # "/") THEN dest[i] := "/"; INC( i ) END;
- WHILE name[j] # 0X DO dest[i] := name[j]; INC( i ); INC( j ) END;
- dest[i] := 0X
- END MakePath;
-
-
- PROCEDURE ScanPath( VAR pos: LONGINT; VAR dir: ARRAY OF CHAR );
- VAR i: LONGINT; ch: CHAR;
- BEGIN
- i := 0; ch := searchPath[pos];
- WHILE ch = " " DO INC( pos ); ch := searchPath[pos] END;
- WHILE ch > " " DO dir[i] := ch; INC( i ); INC( pos ); ch := searchPath[pos] END;
- dir[i] := 0X
- END ScanPath;
- PROCEDURE GetTempName( CONST finalName: ARRAY OF CHAR; VAR tempName: Filename );
- VAR n, i, j, pe, pid: LONGINT;
- BEGIN
- INC(tempno); n := tempno; i := 0; j := 0; pe := 1;
- WHILE finalName[j] = ' ' DO INC(j) END; (* skip leading spaces *)
- IF finalName[j] # "/" THEN (* relative pathname *)
- WHILE cwd[i] # 0X DO tempName[i] := cwd[i]; INC(i) END;
- IF tempName[i - 1] # '/' THEN tempName[i] := '/'; INC(i) END;
- pe := i - 1
- END;
- WHILE finalName[j] # 0X DO tempName[i] := finalName[j]; INC(i); INC(j) END;
- WHILE (i > pe) & (tempName[i-1] # '/') DO DEC(i) END; (* remove filename *)
- tempName[i] := 0X;
- Append( tempName, ".tmp." );
- AppendInt( tempName, n ); Append( tempName, "." );
- pid := Unix.getpid();
- AppendInt( tempName, pid )
- END GetTempName;
- PROCEDURE Install;
- VAR aliasFS: AliasFileSystem;
- BEGIN
- NEW(collection);
- NEW( unixFS ); (* Files.Add( unixFS, "" ); *)
- NEW( aliasFS, unixFS ); Files.Add( aliasFS, "searcher" )
- END Install;
-
- PROCEDURE Initialize;
- VAR a: ADDRESS; i: INTEGER; ch: CHAR;
- BEGIN
- (* get current working directory *)
- a := Unix.getenv( ADDRESSOF( "PWD" ) );
- IF a > 0 THEN
- i := 0;
- REPEAT S.GET( a, ch ); INC( a ); cwd[i] := ch; INC( i ) UNTIL ch = 0X;
- ELSE
- (* $PWD not set *)
- a := Unix.getcwd( ADDRESSOF( cwd ), LEN( cwd ) )
- END;
- i := 0;
- WHILE cwd[i] # 0X DO INC( i ) END;
- DEC( i );
- IF (i > 0) & (cwd[i] = '/') THEN cwd[i] := 0X END;
- (* get search pathes *)
- a := Unix.getenv( ADDRESSOF( "AOSPATH" ) ); i := 0;
- IF a = 0 THEN
- Log.String( "UnixFiles.Initialize: environment variable AOSPATH not defined" ); Log.Ln;
- ELSE
- REPEAT
- S.GET( a, ch ); INC( a );
- IF ch = ":" THEN ch := " " END;
- searchPath[i] := ch; INC( i )
- UNTIL ch = 0X;
- END;
- tempno := 1; openfiles := 0;
- Modules.InstallTermHandler( Finalization )
- END Initialize;
-
-
- (*! The system calls open, read and write return -1 when they get interrupted
- by receiving a signal. Possibly through Objects.SuspendActivities() (GC).
- *)
- PROCEDURE UnixOpen( path: ADDRESS; flags, permissions: SET ): LONGINT;
- VAR
- fd, fo, errno: LONGINT; failure: BOOLEAN;
- BEGIN
- failure := FALSE;
- REPEAT
- fd := Unix.open( path, flags, permissions );
- IF fd < 0 THEN
- errno := Unix.errno();
- IF errno IN {Unix.ENFILE, Unix.EMFILE} THEN
- fo := openfiles; Kernel.GC; WaitClose( fo )
- ELSIF (errno = Unix.EINVAL) & (openfiles > 1000) THEN
- (* in Solaris open fails with EINVAL ??? *)
- fo := openfiles; Kernel.GC; WaitClose( fo )
- ELSIF errno # Unix.EINTR THEN
- failure := TRUE
- END
- END
- UNTIL (fd >= 0) OR failure;
- RETURN fd
- END UnixOpen;
-
- PROCEDURE UnixRead( fd: LONGINT; buf: ADDRESS; len: SIZE ): LONGINT;
- VAR n: LONGINT;
- BEGIN
- REPEAT
- n := Unix.read( fd, buf, len )
- UNTIL (n >= 0) OR (Unix.errno() # Unix.EINTR);
- RETURN n
- END UnixRead;
-
- PROCEDURE UnixWrite( fd: LONGINT; buf: ADDRESS; len: SIZE ): LONGINT;
- VAR n: LONGINT;
- BEGIN
- REPEAT
- n := Unix.write( fd, buf, len )
- UNTIL (n >= 0) OR (Unix.errno() # Unix.EINTR);
- RETURN n
- END UnixWrite;
-
-
- PROCEDURE AddSearchPath*(context: Commands.Context);
- VAR name: Files.FileName; i,j: LONGINT; ch : CHAR;
- BEGIN
- IF context.arg.GetString(name) THEN
- i := 0; j := 0;
- WHILE(searchPath[i] # 0X) DO
- INC(i);
- END;
- searchPath[i] := " ";INC(i);
- REPEAT
- ch := name[j];
- searchPath[i] := name[j];
- INC(j);INC(i);
- UNTIL ch = 0X;
- END;
- END AddSearchPath;
- PROCEDURE Finalization;
- VAR ft: Files.FileSystemTable; i: LONGINT;
- BEGIN
- Files.GetList( ft );
- IF ft # NIL THEN
- FOR i := 0 TO LEN( ft^ ) - 1 DO
- IF ft[i] IS AliasFileSystem THEN Files.Remove( ft[i] ) END
- END
- END;
- collection.Finalize;
- unixFS.Finalize;
- END Finalization;
-
- PROCEDURE DecOpenFiles;
- BEGIN{EXCLUSIVE}
- DEC(openfiles);
- END DecOpenFiles;
- PROCEDURE IncOpenFiles;
- BEGIN{EXCLUSIVE}
- INC(openfiles);
- END IncOpenFiles;
-
-
- PROCEDURE WaitClose(no: LONGINT);
- BEGIN{EXCLUSIVE}
- AWAIT(openfiles < no);
- END WaitClose;
-
- BEGIN
- Initialize;
- Install
- END UnixFiles.
|