Unix.UnixFiles.Mod 34 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172
  1. (* Aos, Copyright 2001, Pieter Muller, ETH Zurich *)
  2. MODULE UnixFiles; (** AUTHOR "gf"; PURPOSE "Unix file systems" *)
  3. (* derived fron (SPARCOberon) Files.Mod by J. Templ 1.12. 89/14.05.93 *)
  4. IMPORT S := SYSTEM, Unix, Kernel, Modules, Log := KernelLog, Files, Commands;
  5. CONST
  6. NBufs = 4; Bufsize = 4096; NoDesc = -1;
  7. Open = 0; Create = 1; Closed = 2; (* file states *)
  8. NoKey = -1;
  9. CreateFlags = Unix.rdwr + Unix.creat + Unix.trunc;
  10. TraceCollection = 0;
  11. Trace = {};
  12. VAR
  13. tempno: INTEGER;
  14. openfiles: LONGINT;
  15. searchPath: ARRAY 1024 OF CHAR;
  16. cwd: ARRAY 256 OF CHAR;
  17. unixFS: UnixFileSystem; (* must be unique *)
  18. collection: Collection; (* must be unique *)
  19. TYPE
  20. Filename = ARRAY 256 OF CHAR;
  21. NameSet = OBJECT
  22. VAR
  23. name: ARRAY 64 OF CHAR;
  24. left, right: NameSet;
  25. PROCEDURE Add( CONST filename: ARRAY OF CHAR ): BOOLEAN;
  26. (* add filename if it not already exists. else return false *)
  27. BEGIN
  28. IF filename = name THEN RETURN FALSE END;
  29. IF filename < name THEN
  30. IF left = NIL THEN NEW( left, filename ); RETURN TRUE
  31. ELSE RETURN left.Add( filename )
  32. END
  33. ELSE
  34. IF right = NIL THEN NEW( right, filename ); RETURN TRUE
  35. ELSE RETURN right.Add( filename )
  36. END
  37. END
  38. END Add;
  39. PROCEDURE & Init( CONST filename: ARRAY OF CHAR );
  40. BEGIN
  41. COPY( filename, name );
  42. left := NIL; right := NIL
  43. END Init;
  44. END NameSet;
  45. AliasFileSystem = OBJECT (Files.FileSystem)
  46. VAR
  47. fs: UnixFileSystem;
  48. PROCEDURE & Init*( realFS: UnixFileSystem);
  49. BEGIN
  50. SELF.fs := realFS;
  51. END Init;
  52. PROCEDURE New0( name: ARRAY OF CHAR ): Files.File;
  53. VAR f: Files.File;
  54. BEGIN
  55. f := fs.New0( name );
  56. IF f # NIL THEN f.fs := SELF END;
  57. RETURN f;
  58. END New0;
  59. PROCEDURE Old0( name: ARRAY OF CHAR ): Files.File;
  60. VAR f: Files.File;
  61. BEGIN
  62. f := fs.Old0( name );
  63. IF f # NIL THEN f.fs := SELF END;
  64. RETURN f;
  65. END Old0;
  66. PROCEDURE Delete0( name: ARRAY OF CHAR; VAR key, res: LONGINT );
  67. BEGIN
  68. fs.Delete0( name, key, res );
  69. END Delete0;
  70. PROCEDURE Rename0( old, new: ARRAY OF CHAR; fold: Files.File; VAR res: LONGINT );
  71. BEGIN
  72. fs.Rename0( old, new, fold, res );
  73. END Rename0;
  74. PROCEDURE Enumerate0( mask: ARRAY OF CHAR; flags: SET; enum: Files.Enumerator );
  75. BEGIN
  76. fs.Enumerate0( mask, flags, enum );
  77. END Enumerate0;
  78. PROCEDURE FileKey( name: ARRAY OF CHAR ): LONGINT;
  79. VAR
  80. BEGIN
  81. RETURN fs.FileKey( name );
  82. END FileKey;
  83. PROCEDURE CreateDirectory0( name: ARRAY OF CHAR; VAR res: LONGINT );
  84. BEGIN
  85. fs.CreateDirectory0( name, res );
  86. END CreateDirectory0;
  87. PROCEDURE RemoveDirectory0( name: ARRAY OF CHAR; force: BOOLEAN; VAR key, res: LONGINT );
  88. BEGIN
  89. fs.RemoveDirectory0( name, force, key, res );
  90. END RemoveDirectory0;
  91. PROCEDURE Has(CONST name: ARRAY OF CHAR; VAR fullName: ARRAY OF CHAR; VAR flags: SET): BOOLEAN;
  92. BEGIN
  93. RETURN fs.Has(name, fullName, flags);
  94. END Has;
  95. END AliasFileSystem;
  96. FinalizeFiles = OBJECT
  97. PROCEDURE EnumFile( f: ANY; VAR cont: BOOLEAN );
  98. VAR F: File;
  99. BEGIN
  100. F := f( File ); F.Finalize(); cont := TRUE
  101. END EnumFile;
  102. END FinalizeFiles;
  103. SearchByFstat = OBJECT
  104. VAR
  105. found: File;
  106. stat: Unix.Status;
  107. PROCEDURE Init( s: Unix.Status );
  108. BEGIN
  109. found := NIL;
  110. stat := s;
  111. END Init;
  112. PROCEDURE EnumFile( f: ANY; VAR cont: BOOLEAN );
  113. BEGIN
  114. WITH f: File DO
  115. IF (stat.ino = f.ino) & (stat.dev = f.dev) THEN
  116. (* possible different name but same file! *)
  117. ResetBuffers( f, stat );
  118. found := f; cont := FALSE;
  119. END;
  120. END;
  121. END EnumFile;
  122. END SearchByFstat;
  123. Collection = OBJECT (* methods in Collection shared by objects Filesystem and File *)
  124. VAR oldFiles, newFiles: Kernel.FinalizedCollection;
  125. ssearch: SearchByFstat;
  126. fileKey: LONGINT;
  127. PROCEDURE & Init*;
  128. BEGIN
  129. NEW( oldFiles ); NEW( newFiles ); NEW(ssearch); fileKey := -1;
  130. END Init;
  131. PROCEDURE AddNew( F: File );
  132. BEGIN {EXCLUSIVE}
  133. IF TraceCollection IN Trace THEN Log.String( "Collections.AddNew: " ); Log.String( F.workName ); Log.Ln; END;
  134. newFiles.Add( F, FinalizeFile );
  135. END AddNew;
  136. PROCEDURE AddOld( F: File );
  137. BEGIN {EXCLUSIVE}
  138. IF TraceCollection IN Trace THEN Log.String( "Collections.AddOld: " ); Log.String( F.workName ); Log.Ln; END;
  139. oldFiles.Add( F, FinalizeFile );
  140. END AddOld;
  141. PROCEDURE ByStat(CONST stat: Unix.Status): File;
  142. BEGIN{EXCLUSIVE}
  143. ssearch.Init(stat);
  144. oldFiles.EnumerateN(ssearch.EnumFile);
  145. IF ssearch.found = NIL THEN
  146. newFiles.EnumerateN(ssearch.EnumFile)
  147. END;
  148. IF TraceCollection IN Trace THEN
  149. Log.String( "Collections.ByStatus: " ); Log.Ln;
  150. IF ssearch.found = NIL THEN Log.String("not found") ELSE Log.String("found") END;
  151. END;
  152. RETURN ssearch.found;
  153. END ByStat;
  154. PROCEDURE Finalize;
  155. VAR fin: FinalizeFiles;
  156. BEGIN {EXCLUSIVE}
  157. IF TraceCollection IN Trace THEN Log.String( "Collections.Finalize " ); Log.Ln; END;
  158. NEW( fin ); newFiles.Enumerate( fin.EnumFile ); newFiles.Clear(); oldFiles.Enumerate( fin.EnumFile ); oldFiles.Clear();
  159. END Finalize;
  160. PROCEDURE FinalizeFile( obj: ANY );
  161. VAR F: File;
  162. BEGIN
  163. F := obj( File );
  164. IF TraceCollection IN Trace THEN Log.String( "Collections.FinalizeFile " ); Log.String( F.workName ); Log.Ln; END;
  165. F.Finalize()
  166. END FinalizeFile;
  167. END Collection;
  168. UnixFileSystem* = OBJECT (Files.FileSystem)
  169. PROCEDURE & Init;
  170. BEGIN
  171. prefix := ""; vol := NIL; desc := "UnixFS"
  172. END Init;
  173. PROCEDURE New0*( name: ARRAY OF CHAR ): Files.File;
  174. VAR f: File; stat: Unix.Status;
  175. res,err: LONGINT;
  176. path,nameonly: ARRAY 512 OF CHAR;
  177. BEGIN {EXCLUSIVE}
  178. (*AwaitFinalizingDone;*)
  179. (*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*)
  180. Files.SplitPath(name,path, nameonly);
  181. res:=Unix.stat( ADDRESSOF( path ), stat ) ;
  182. err:=Unix.errno();
  183. IF (name="") OR (path="") OR (res>=0) OR (err#Unix.ENOENT) THEN
  184. NEW( f, SELF );
  185. f.workName := ""; COPY( name, f.registerName );
  186. f.fd := NoDesc; f.state := Create; f.fsize := 0; f.fpos := 0;
  187. f.swapper := -1; (*all f.buf[i] = NIL*)
  188. f.key := NoKey; f.fs := SELF;
  189. RETURN f;
  190. ELSE
  191. Log.String( "UnixFileSystem.New0: file allocation failed. Probably a nonexistent path." ); Log.Ln;
  192. RETURN NIL;
  193. END;
  194. END New0;
  195. PROCEDURE IsDirectory( VAR stat: Unix.Status ): BOOLEAN;
  196. VAR mode: LONGINT;
  197. BEGIN
  198. mode := stat.mode;
  199. RETURN ODD( mode DIV 4000H )
  200. END IsDirectory;
  201. PROCEDURE Old0*( name: ARRAY OF CHAR ): Files.File;
  202. VAR f: File; stat: Unix.Status; fd, r, errno, pos: LONGINT;
  203. oflags: SET; nextdir, path: Filename; fo: LONGINT;
  204. BEGIN {EXCLUSIVE}
  205. IF name = "" THEN RETURN NIL END;
  206. IF IsFullName( name ) THEN
  207. COPY( name, path ); nextdir := "";
  208. ELSE
  209. pos := 0; ScanPath( pos, nextdir ); MakePath( nextdir, name, path );
  210. ScanPath( pos, nextdir )
  211. END;
  212. LOOP
  213. r := Unix.access( ADDRESSOF( path ), Unix.R_OK );
  214. IF r >= 0 THEN
  215. r := Unix.access( ADDRESSOF( path ), Unix.W_OK );
  216. IF r < 0 THEN oflags := Unix.rdonly ELSE oflags := Unix.rdwr END;
  217. fd := Unix.open( ADDRESSOF( path ), oflags, {} ); errno := Unix.errno();
  218. IF ((fd < 0) & (errno IN {Unix.ENFILE, Unix.EMFILE, Unix.EINVAL})) THEN
  219. REPEAT
  220. fo := openfiles; Kernel.GC; WaitClose( fo );
  221. fd := Unix.open( ADDRESSOF( path ), oflags, {} ); errno := Unix.errno();
  222. UNTIL (fd >= 0) OR ~ (errno IN {Unix.ENFILE, Unix.EMFILE, Unix.EINVAL})
  223. END;
  224. IF fd >= 0 THEN
  225. r := Unix.fstat( fd, stat );
  226. f := collection.ByStat(stat);
  227. IF f # NIL THEN
  228. (* use the file already cached *) r := Unix.close( fd ); EXIT
  229. ELSE
  230. NEW( f, SELF );
  231. f.fd := fd; f.dev := stat.dev; f.ino := stat.ino;
  232. f.mtime := stat.mtime.sec; f.fsize := stat.size; f.fpos := 0;
  233. f.state := Open; f.swapper := -1; (*all f.buf[i] = NIL*)
  234. COPY( path, f.workName ); f.registerName := "";
  235. f.tempFile := FALSE;
  236. IF IsDirectory( stat ) THEN
  237. f.flags := {Files.Directory, Files.ReadOnly}
  238. ELSIF oflags = Unix.rdonly THEN
  239. f.flags := {Files.ReadOnly}
  240. END;
  241. f.key := NoKey; f.fs := SELF;
  242. IncOpenFiles();
  243. collection.AddOld(f);
  244. EXIT
  245. END
  246. END
  247. ELSIF nextdir # "" THEN
  248. MakePath( nextdir, name, path ); ScanPath( pos, nextdir );
  249. ELSE
  250. f := NIL; EXIT
  251. END;
  252. END; (* loop *)
  253. RETURN f
  254. END Old0;
  255. (** Return the unique non-zero key of the named file, if it exists. *)
  256. PROCEDURE FileKey*( name: ARRAY OF CHAR ): LONGINT;
  257. (* Can not be used for Unix files as LONGINT is too small.
  258. In the Unix filesystem a file is identified by
  259. - dev (64 bit (Linux), 32 bit (Solaris, Darwin)) +
  260. - ino (32 bit)
  261. *)
  262. BEGIN
  263. RETURN 0
  264. END FileKey;
  265. PROCEDURE Delete0*( name: ARRAY OF CHAR; VAR key, res: LONGINT );
  266. VAR r: LONGINT;
  267. BEGIN {EXCLUSIVE}
  268. r := Unix.unlink( ADDRESSOF( name ) );
  269. IF r = 0 THEN res := Files.Ok
  270. ELSE res := Unix.errno( )
  271. END;
  272. key := 0;
  273. END Delete0;
  274. (* return remaining old file, if any *)
  275. PROCEDURE TryRename*( old, new: ARRAY OF CHAR; f: Files.File; VAR res: LONGINT ): LONGINT;
  276. CONST Bufsize = 4096;
  277. VAR fdold, fdnew, fo, n, r: LONGINT; ostat, nstat: Unix.Status;
  278. buf: ARRAY Bufsize OF CHAR;
  279. BEGIN {EXCLUSIVE}
  280. fo := NoDesc;
  281. r:= Unix.stat( ADDRESSOF( old ), ostat );
  282. IF r >= 0 THEN
  283. r := Unix.stat( ADDRESSOF( new ), nstat );
  284. IF (r >= 0) & ((ostat.dev # nstat.dev) OR (ostat.ino # nstat.ino)) THEN
  285. r := Unix.unlink( ADDRESSOF( new ) ) (* work around stale nfs handles *);
  286. END;
  287. r := Unix.rename( ADDRESSOF( old ), ADDRESSOF( new ) );
  288. IF r < 0 THEN (* could not rename, try copy *)
  289. res := Unix.errno( );
  290. IF (res = Unix.EXDEV) OR (res = Unix.ETXTBSY) THEN (* cross device link, move the file / file busy frequently happens in VirtualBox *)
  291. fdold := Unix.open( ADDRESSOF( old ), Unix.rdonly, {} );
  292. fo := fdold;
  293. IF fdold < 0 THEN
  294. res := Unix.errno( );
  295. RETURN NoDesc;
  296. END;
  297. fdnew := Unix.open( ADDRESSOF( new ), Unix.rdwr + Unix.creat + Unix.trunc, Unix.rwrwr );
  298. IF fdnew < 0 THEN
  299. res := Unix.errno( );
  300. RETURN NoDesc;
  301. END;
  302. REPEAT
  303. n := Unix.read( fdold, ADDRESSOF( buf ), Bufsize ); (*can return -1 with EINTR without any particular reason*)
  304. IF n > 0 THEN
  305. REPEAT
  306. r := Unix.write( fdnew, ADDRESSOF( buf ), n );
  307. IF r < 0 THEN res:= Unix.errno( ) END;
  308. UNTIL (r >= 0) OR (res # Unix.EINTR); (*ignore EINTR, retry*)
  309. IF r < 0 THEN
  310. r := Unix.close( fdold );
  311. r := Unix.close( fdnew );
  312. RETURN NoDesc;
  313. END;
  314. END
  315. UNTIL n = 0;
  316. r := Unix.unlink( ADDRESSOF( old ) );
  317. r := Unix.close( fdold );
  318. r := Unix.close( fdnew );
  319. res := Files.Ok
  320. ELSE
  321. RETURN NoDesc (* res is Unix.rename return code *)
  322. END
  323. END;
  324. res := Files.Ok
  325. ELSE
  326. res := Unix.errno();
  327. END;
  328. RETURN fo;
  329. END TryRename;
  330. PROCEDURE Rename0*( old, new: ARRAY OF CHAR; f: Files.File; VAR res: LONGINT );
  331. VAR of: LONGINT;
  332. BEGIN
  333. of := TryRename(old, new, f, res);
  334. END Rename0;
  335. PROCEDURE CreateDirectory0*( path: ARRAY OF CHAR; VAR res: LONGINT );
  336. VAR r: LONGINT;
  337. BEGIN {EXCLUSIVE}
  338. r := Unix.mkdir( ADDRESSOF( path ), Unix.rwxrwxrwx );
  339. IF r = 0 THEN res := Files.Ok
  340. ELSE res := Unix.errno( )
  341. END
  342. END CreateDirectory0;
  343. PROCEDURE RemoveDirectory0*( path: ARRAY OF CHAR; force: BOOLEAN; VAR key, res: LONGINT );
  344. VAR r: LONGINT;
  345. BEGIN {EXCLUSIVE}
  346. r := Unix.rmdir( ADDRESSOF( path ) );
  347. IF r = 0 THEN res := Files.Ok
  348. ELSE res := Unix.errno( )
  349. END
  350. END RemoveDirectory0;
  351. PROCEDURE Enumerate0*( mask: ARRAY OF CHAR; flags: SET; enum: Files.Enumerator );
  352. VAR
  353. path, filemask: Filename;
  354. isPath: BOOLEAN;
  355. i, j: INTEGER; dirName, fileName, fullName, xName: Filename;
  356. checkSet: NameSet; ent: Unix.Dirent;
  357. PROCEDURE GetEntryName;
  358. VAR i: INTEGER; adr: ADDRESS;
  359. BEGIN
  360. i := -1; adr := ADDRESSOF( ent.name );
  361. REPEAT INC( i ); S.GET( adr, fileName[i] ); INC( adr ) UNTIL fileName[i] = 0X
  362. END GetEntryName;
  363. PROCEDURE EnumDir( CONST dirName: ARRAY OF CHAR );
  364. VAR
  365. dir: ADDRESS;
  366. tm: Unix.TmPtr; date, time: LONGINT;
  367. stat: Unix.Status; r: LONGINT;
  368. BEGIN
  369. dir := Unix.opendir( ADDRESSOF( dirName ) );
  370. IF dir # 0 THEN
  371. ent := Unix.readdir( dir );
  372. WHILE ent # NIL DO
  373. COPY( dirName, fullName );
  374. GetEntryName; AppendName( fullName, fileName );
  375. IF (fileName[0] # '.') & Match( fileName, filemask, 0, 0 ) THEN
  376. IF checkSet.Add( fileName ) THEN (* not a covered name *)
  377. r := Unix.stat( ADDRESSOF( fullName ), stat );
  378. tm := Unix.localtime( stat.mtime );
  379. date := tm.year*200H + (tm.mon + 1)*20H + tm.mday;
  380. time := tm.hour*1000H + tm.min*40H + tm.sec;
  381. flags := {};
  382. IF IsDirectory( stat ) THEN
  383. flags := {Files.ReadOnly, Files.Directory}
  384. ELSE
  385. r := Unix.access( ADDRESSOF( fullName ), Unix.W_OK );
  386. IF r < 0 THEN flags := {Files.ReadOnly} END
  387. END;
  388. r := Unix.realpath(ADDRESSOF(fullName), ADDRESSOF(xName));
  389. IF (r # 0) THEN COPY(xName, fullName) END;
  390. enum.PutEntry( fullName, flags, time, date, stat.size );
  391. END
  392. END;
  393. ent := Unix.readdir( dir );
  394. END;
  395. Unix.closedir( dir )
  396. END;
  397. END EnumDir;
  398. BEGIN {EXCLUSIVE}
  399. Files.SplitName( mask, prefix, fullName );
  400. Files.SplitPath( fullName, path, filemask );
  401. NEW( checkSet, "M###N" );
  402. isPath:= path#"";
  403. IF isPath THEN
  404. CleanPath(path); (*get rid of xxx/../xxx and xxx/./xxx in the path string*)
  405. END;
  406. IF isPath & (path[0] = '/') THEN (*check for absolute path*)
  407. EnumDir( path);
  408. ELSE (*no path or relative path*)
  409. i := 0; j := 0;
  410. LOOP (*go through the search paths, every time a complete search path has been traversed, look for the element there*)
  411. IF (searchPath[i] = " ") OR (searchPath[i] = 0X) THEN
  412. dirName[j] := 0X;
  413. IF isPath THEN (*if relative path: add relative path to the current search path*)
  414. Files.JoinPath(dirName, path, dirName);
  415. END;
  416. EnumDir( dirName );
  417. IF searchPath[i] = 0X THEN EXIT
  418. ELSE INC( i ); j := 0
  419. END
  420. ELSE
  421. dirName[j] := searchPath[i]; INC( j ); INC( i )
  422. END
  423. END
  424. END;
  425. checkSet := NIL;
  426. END Enumerate0;
  427. PROCEDURE Has(CONST name: ARRAY OF CHAR; VAR fullName: ARRAY OF CHAR; VAR flags: SET): BOOLEAN;
  428. VAR r: LONGINT;stat: Unix.Status;
  429. BEGIN
  430. r := Unix.stat( ADDRESSOF( name ), stat );
  431. IF r # 0 THEN RETURN FALSE END;
  432. flags := {};
  433. IF IsDirectory( stat ) THEN
  434. flags := {Files.ReadOnly, Files.Directory}
  435. ELSE
  436. r := Unix.access( ADDRESSOF( name ), Unix.W_OK );
  437. IF r < 0 THEN flags := {Files.ReadOnly} END
  438. END;
  439. r := Unix.realpath(ADDRESSOF(name), ADDRESSOF(fullName));
  440. IF (r = 0) THEN COPY(name, fullName) END; (* no success *)
  441. RETURN TRUE;
  442. END Has;
  443. END UnixFileSystem;
  444. Buffer = POINTER TO RECORD (Files.Hint)
  445. chg: BOOLEAN;
  446. org, size: LONGINT;
  447. data: ARRAY Bufsize OF CHAR;
  448. END;
  449. File* = OBJECT (Files.File)
  450. VAR
  451. fd: LONGINT;
  452. workName, registerName: Filename;
  453. tempFile: BOOLEAN;
  454. dev: Unix.DevT;
  455. ino: LONGINT;
  456. mtime: HUGEINT;
  457. fsize, fpos: SIZE;
  458. bufs: ARRAY NBufs OF Buffer;
  459. swapper, state: LONGINT;
  460. PROCEDURE & Init( fs: Files.FileSystem );
  461. BEGIN
  462. SELF.fs := fs; flags := {};
  463. END Init;
  464. PROCEDURE CreateUnixFile;
  465. VAR
  466. stat: Unix.Status; done: BOOLEAN; errno,fo: LONGINT;
  467. BEGIN
  468. IF state = Create THEN
  469. GetTempName( registerName, workName ); tempFile := TRUE
  470. ELSIF state = Closed THEN
  471. IF registerName # "" THEN
  472. workName := registerName; registerName := ""; tempFile := FALSE;
  473. ELSE
  474. RETURN;
  475. END;
  476. END;
  477. errno := Unix.unlink( ADDRESSOF( workName ) );
  478. (*unlink first to avoid stale NFS handles and to avoid reuse of inodes*)
  479. fd := Unix.open( ADDRESSOF( workName ), CreateFlags, Unix.rwrwr );
  480. done := fd >= 0; errno := Unix.errno();
  481. IF ( (~done) & (errno IN {Unix.ENFILE, Unix.EMFILE, Unix.EINVAL})) THEN
  482. REPEAT
  483. fo := openfiles; Kernel.GC; WaitClose( fo );
  484. fd := Unix.open( ADDRESSOF( workName ), CreateFlags, Unix.rwrwr); errno := Unix.errno();
  485. UNTIL (fd >= 0) OR ~ (errno IN {Unix.ENFILE, Unix.EMFILE, Unix.EINVAL});
  486. done := fd >= 0
  487. END;
  488. IF done THEN
  489. errno := Unix.fstat( fd, stat );
  490. dev := stat.dev; ino := stat.ino; mtime := stat.mtime.sec;
  491. state := Open; fpos := 0;
  492. IncOpenFiles();
  493. collection.AddNew(SELF);
  494. ELSE
  495. Halt( SELF, TRUE, "UnixFiles.File.Create: open failed" );
  496. END
  497. END CreateUnixFile;
  498. PROCEDURE Flush( buf: Buffer );
  499. VAR res,err: LONGINT; stat: Unix.Status;
  500. BEGIN
  501. IF buf.chg THEN
  502. IF fd = NoDesc THEN CreateUnixFile END;
  503. IF buf.org # fpos THEN res := Unix.lseek( fd, buf.org, 0 ) END;
  504. REPEAT
  505. res := Unix.write( fd, ADDRESSOF( buf.data ), buf.size );
  506. IF res < 0 THEN err := Unix.errno() END;
  507. UNTIL (res >= 0) OR (err # Unix.EINTR); (*ignore EINTR, retry*)
  508. IF res < 0 THEN Halt( SELF, TRUE, "UnixFiles.File.Flush: write failed" ) END;
  509. fpos := buf.org + buf.size; buf.chg := FALSE;
  510. res := Unix.fstat( fd, stat ); mtime := stat.mtime.sec
  511. END
  512. END Flush;
  513. PROCEDURE Set*( VAR r: Files.Rider; pos: LONGINT );
  514. BEGIN {EXCLUSIVE}
  515. SetX( r, pos )
  516. END Set;
  517. PROCEDURE SetX( VAR r: Files.Rider; p: LONGINT );
  518. VAR org, offset, i, n, res: LONGINT; buf: Buffer;
  519. BEGIN
  520. IF p > fsize THEN p := LONGINT(fsize)
  521. ELSIF p < 0 THEN p := 0
  522. END;
  523. offset := p MOD Bufsize; org := p - offset;
  524. i := 0;
  525. WHILE (i < NBufs) & (bufs[i] # NIL) & (org # bufs[i].org) DO INC( i ) END;
  526. IF i < NBufs THEN
  527. IF bufs[i] = NIL THEN
  528. NEW( buf ); buf.chg := FALSE; buf.org := -1;
  529. bufs[i] := buf
  530. ELSE
  531. swapper := i;
  532. buf := bufs[swapper]; Flush( buf )
  533. END
  534. ELSE
  535. swapper := (swapper + 1) MOD NBufs;
  536. buf := bufs[swapper]; Flush( buf )
  537. END;
  538. IF buf.org # org THEN
  539. IF org = fsize THEN
  540. buf.size := 0
  541. ELSE
  542. IF fd = NoDesc THEN CreateUnixFile END;
  543. IF fpos # org THEN res := Unix.lseek( fd, org, 0 ) END;
  544. IF res < 0 THEN Halt( SELF, TRUE, "UnixFiles.File.Set: lseek failed" ) END;
  545. REPEAT
  546. n := Unix.read( fd, ADDRESSOF( buf.data ), Bufsize ); (*can return -1 with EINTR without warning*)
  547. IF n < 0 THEN res:= Unix.errno( ) END;
  548. UNTIL (n >= 0) OR ((res # Unix.EINTR) & (res # Unix.EINVAL));
  549. (* Solaris returns EINVAL instead of EINTR ! ??? *)
  550. IF n < 0 THEN
  551. IF p < fsize THEN Halt( SELF, TRUE, "UnixFiles.File.Set: read failed" )
  552. ELSE n := 0
  553. END
  554. END;
  555. fpos := org + n; buf.size := n
  556. END;
  557. buf.org := org; buf.chg := FALSE
  558. ELSE
  559. org := buf.org
  560. END;
  561. r.hint := buf; r.apos := org; r.bpos := offset;
  562. r.res := 0; r.eof := FALSE;
  563. r.file := SELF; r.fs := fs
  564. END SetX;
  565. PROCEDURE Pos*( VAR r: Files.Rider ): LONGINT;
  566. BEGIN
  567. RETURN r.apos + r.bpos
  568. END Pos;
  569. PROCEDURE Read*( VAR r: Files.Rider; VAR x: CHAR );
  570. VAR offset: LONGINT; buf: Buffer;
  571. BEGIN {EXCLUSIVE}
  572. buf := r.hint(Buffer); offset := r.bpos;
  573. IF r.apos # buf.org THEN
  574. SetX( r, r.apos + offset );
  575. buf := r.hint(Buffer); offset := r.bpos
  576. END;
  577. IF (offset < buf.size) THEN
  578. x := buf.data[offset]; r.bpos := offset + 1
  579. ELSIF r.apos + offset < fsize THEN
  580. SetX( r, r.apos + offset );
  581. x := r.hint(Buffer).data[0]; r.bpos := 1
  582. ELSE
  583. x := 0X; r.eof := TRUE
  584. END
  585. END Read;
  586. PROCEDURE ReadBytes*( VAR r: Files.Rider; VAR x: ARRAY OF CHAR; ofs, len: LONGINT );
  587. VAR xpos, min, restInBuf, offset: LONGINT; buf: Buffer;
  588. BEGIN {EXCLUSIVE}
  589. x[ofs] := 0X; xpos := ofs;
  590. buf := r.hint(Buffer); offset := r.bpos;
  591. WHILE len > 0 DO
  592. IF (r.apos # buf.org) OR (offset >= Bufsize) THEN
  593. SetX( r, r.apos + offset );
  594. buf := r.hint(Buffer); offset := r.bpos
  595. END;
  596. restInBuf := buf.size - offset;
  597. IF restInBuf = 0 THEN r.res := len; r.eof := TRUE; RETURN
  598. ELSIF len > restInBuf THEN min := restInBuf
  599. ELSE min := len
  600. END;
  601. S.MOVE( ADDRESSOF( buf.data ) + offset, ADDRESSOF( x ) + xpos, min );
  602. INC( offset, min ); r.bpos := offset;
  603. INC( xpos, min ); DEC( len, min )
  604. END;
  605. r.res := 0; r.eof := FALSE;
  606. END ReadBytes;
  607. PROCEDURE Write*( VAR r: Files.Rider; x: CHAR );
  608. VAR buf: Buffer; offset: LONGINT;
  609. BEGIN {EXCLUSIVE}
  610. buf := r.hint(Buffer); offset := r.bpos;
  611. IF (r.apos # buf.org) OR (offset >= Bufsize) THEN
  612. SetX( r, r.apos + offset );
  613. buf := r.hint(Buffer); offset := r.bpos
  614. END;
  615. buf.data[offset] := x; buf.chg := TRUE;
  616. IF offset = buf.size THEN INC( buf.size ); INC( fsize ) END;
  617. r.bpos := offset + 1; r.res := Files.Ok
  618. END Write;
  619. PROCEDURE WriteBytes*( VAR r: Files.Rider; CONST x: ARRAY OF CHAR; ofs, len: LONGINT );
  620. VAR xpos, min, restInBuf, offset: LONGINT; buf: Buffer;
  621. BEGIN {EXCLUSIVE}
  622. xpos := ofs; buf := r.hint(Buffer); offset := r.bpos;
  623. WHILE len > 0 DO
  624. IF (r.apos # buf.org) OR (offset >= Bufsize) THEN
  625. SetX( r, r.apos + offset );
  626. buf := r.hint(Buffer); offset := r.bpos
  627. END;
  628. restInBuf := Bufsize - offset;
  629. IF len > restInBuf THEN min := restInBuf ELSE min := len END;
  630. S.MOVE( ADDRESSOF( x ) + xpos, ADDRESSOF( buf.data ) + offset, min );
  631. INC( offset, min ); r.bpos := offset;
  632. IF offset > buf.size THEN
  633. INC( fsize, offset - buf.size ); buf.size := offset
  634. END;
  635. INC( xpos, min ); DEC( len, min ); buf.chg := TRUE
  636. END;
  637. r.res := Files.Ok
  638. END WriteBytes;
  639. PROCEDURE Length*( ): LONGINT;
  640. BEGIN
  641. RETURN LONGINT(fsize)
  642. END Length;
  643. PROCEDURE GetDate*( VAR t, d: LONGINT );
  644. VAR stat: Unix.Status; r: LONGINT; time: Unix.TmPtr;
  645. BEGIN {EXCLUSIVE}
  646. IF fd = NoDesc THEN CreateUnixFile END;
  647. r := Unix.fstat( fd, stat );
  648. time := Unix.localtime( stat.mtime );
  649. t := time.sec + ASH( time.min, 6 ) + ASH( time.hour, 12 );
  650. d := time.mday + ASH( time.mon + 1, 5 ) + ASH( time.year, 9 );
  651. END GetDate;
  652. PROCEDURE SetDate*( t, d: LONGINT );
  653. TYPE
  654. Time = RECORD actime, modtime: LONGINT END;
  655. VAR
  656. tm: Unix.Tm; buf: Time; r: LONGINT; path: Filename;
  657. BEGIN {EXCLUSIVE}
  658. IF registerName # "" THEN COPY( registerName, path )
  659. ELSE COPY( workName, path )
  660. END;
  661. (* get year and timezone *)
  662. (* fill in new date *)
  663. tm.isdst := -1; tm.sec := t MOD 64; tm.min := t DIV 64 MOD 64;
  664. tm.hour := t DIV 4096 MOD 32;
  665. tm.mday := d MOD 32; tm.mon := d DIV 32 MOD 16 - 1; tm.year := d DIV 512;
  666. tm.wday := 0; tm.yday := 0;
  667. buf.actime := Unix.mktime( tm ); buf.modtime := buf.actime;
  668. r := Unix.utime( ADDRESSOF( path ), ADDRESSOF( buf ) );
  669. END SetDate;
  670. PROCEDURE GetAttributes*( ): SET;
  671. BEGIN {EXCLUSIVE}
  672. RETURN flags
  673. END GetAttributes;
  674. PROCEDURE SetAttributes*( attr: SET );
  675. BEGIN {EXCLUSIVE}
  676. (* flags := attr *)
  677. END SetAttributes;
  678. PROCEDURE Register0*( VAR res: LONGINT );
  679. VAR fo: LONGINT;
  680. BEGIN {EXCLUSIVE}
  681. IF (state = Create) & (registerName # "") THEN
  682. state := Closed (* shortcut renaming *) ;
  683. END;
  684. FlushBuffers;
  685. IF registerName # "" THEN
  686. fo := unixFS.TryRename( workName, registerName, SELF, res );
  687. IF res # Files.Ok THEN
  688. Halt( SELF, FALSE, "UnixFiles.File.Register: rename failed" )
  689. END;
  690. IF fo # NoDesc THEN (* SELF still refers to old file *)
  691. res := Unix.close(fd);
  692. res := Unix.unlink(ADDRESSOF(workName)); (* VirtualBox ! Can only delete file when closed. *)
  693. fd := Unix.open( ADDRESSOF( registerName ), Unix.rdwr, Unix.rwrwr );
  694. END;
  695. workName := registerName; registerName := ""; tempFile := FALSE;
  696. END;
  697. (*collection.Register(SELF);*)
  698. END Register0;
  699. PROCEDURE Update*;
  700. BEGIN {EXCLUSIVE}
  701. FlushBuffers
  702. END Update;
  703. PROCEDURE FlushBuffers;
  704. VAR i: LONGINT;
  705. BEGIN
  706. IF fd = NoDesc THEN CreateUnixFile END;
  707. FOR i := 0 TO NBufs - 1 DO
  708. IF bufs[i] # NIL THEN Flush( bufs[i] ) END
  709. END;
  710. END FlushBuffers;
  711. PROCEDURE Finalize*;
  712. VAR r: LONGINT;
  713. BEGIN {EXCLUSIVE}
  714. IF tempFile THEN
  715. IF fd # NoDesc THEN r := Unix.close(fd) END;
  716. r := Unix.unlink( ADDRESSOF( registerName ) );
  717. fd := NoDesc;
  718. ELSE
  719. FlushBuffers;
  720. IF fd # NoDesc THEN
  721. r := Unix.close( fd );
  722. fd := NoDesc;
  723. END;
  724. END;
  725. DecOpenFiles();
  726. state := Closed;
  727. END Finalize;
  728. PROCEDURE Close;
  729. BEGIN
  730. Finalize;
  731. collection.oldFiles.Remove(SELF);
  732. END Close;
  733. PROCEDURE GetName*( VAR name: ARRAY OF CHAR );
  734. BEGIN {EXCLUSIVE}
  735. IF registerName = "" THEN COPY( workName, name ) ;
  736. ELSE COPY( registerName, name )
  737. END;
  738. CleanPath( name )
  739. END GetName;
  740. END File;
  741. (*===================================================================*)
  742. (** Get the current directory. *)
  743. PROCEDURE GetWorkingDirectory*( VAR path: ARRAY OF CHAR );
  744. BEGIN
  745. COPY( cwd, path )
  746. END GetWorkingDirectory;
  747. (** Change to directory path. *)
  748. PROCEDURE ChangeDirectory*( CONST path: ARRAY OF CHAR; VAR done: BOOLEAN );
  749. VAR r: LONGINT; newdir: Filename;
  750. BEGIN
  751. IF path[0] # '/' THEN
  752. COPY( cwd, newdir ); AppendName( newdir, path );
  753. CleanPath( newdir )
  754. ELSE
  755. COPY( path, newdir );
  756. END;
  757. r := Unix.chdir( ADDRESSOF( newdir ) );
  758. IF r = 0 THEN COPY( newdir, cwd ); done := TRUE ELSE done := FALSE END
  759. END ChangeDirectory;
  760. (*===================================================================*)
  761. PROCEDURE StripPath*( CONST path: ARRAY OF CHAR; VAR name: ARRAY OF CHAR );
  762. VAR i, p: INTEGER; c: CHAR;
  763. BEGIN
  764. i := 0; p := 0;
  765. REPEAT
  766. IF path[i] = '/' THEN p := i + 1 END;
  767. INC( i )
  768. UNTIL path[i] = 0X;
  769. i := 0;
  770. REPEAT c := path[p]; name[i] := c; INC( i ); INC( p ) UNTIL c = 0X
  771. END StripPath;
  772. PROCEDURE CleanPath*( VAR path: ARRAY OF CHAR );
  773. (*
  774. /aaa/../bbb/./ccc/../ddd/. ==> /bbb/ddd
  775. ../aaa ==> CWD/../aaa ==> . . .
  776. *)
  777. VAR
  778. i, prevNameStart, nameStart: INTEGER;
  779. c1, c2, c3: CHAR;
  780. PROCEDURE prependCWD;
  781. VAR tmp: ARRAY 256 OF CHAR;
  782. BEGIN
  783. COPY( cwd, tmp ); AppendName( tmp, path ); COPY( tmp, path )
  784. END prependCWD;
  785. PROCEDURE restart;
  786. BEGIN
  787. IF path[0] = '/' THEN nameStart := 1 ELSE nameStart := 0 END;
  788. i := -1; prevNameStart := -1;
  789. END restart;
  790. PROCEDURE shift( p0, p1: INTEGER );
  791. VAR c: CHAR;
  792. BEGIN
  793. REPEAT c := path[p1]; path[p0] := c; INC( p0 ); INC( p1 ) UNTIL c = 0X;
  794. IF p0 > 1 THEN restart ELSE i := 0 END
  795. END shift;
  796. BEGIN
  797. restart;
  798. REPEAT
  799. INC( i );
  800. IF i = nameStart THEN
  801. c1 := path[i]; c2 := path[i + 1]; c3 := path[i + 2];
  802. IF c1 = '/' THEN shift( i, i + 1 ) (* // *)
  803. ELSIF c1 = '.' THEN
  804. IF c2 = 0X THEN
  805. IF i > 1 THEN DEC( i ) END;
  806. path[i] := 0X
  807. ELSIF c2 = '/' THEN shift( i, i + 2 ); (* ./ *)
  808. ELSIF (c2 = '.') & ((c3 = 0X) OR (c3 = '/')) THEN (* .. *)
  809. IF i = 0 THEN prependCWD; restart
  810. ELSIF c3 = 0X THEN DEC( i ); path[i] := 0X
  811. ELSIF c3 = '/' THEN (* ../ *)
  812. IF prevNameStart >= 0 THEN shift( prevNameStart, i + 3 ) END
  813. END
  814. END
  815. END
  816. ELSIF path[i] = '/' THEN
  817. IF i > 0 THEN prevNameStart := nameStart END;
  818. nameStart := i + 1
  819. END;
  820. UNTIL (i >= 0) & (path[i] = 0X);
  821. IF (i > 1) & (path[i - 1] = '/') THEN path[i - 1] := 0X END;
  822. IF path = "" THEN COPY( cwd, path ) END;
  823. END CleanPath;
  824. PROCEDURE Match( CONST name, pat: ARRAY OF CHAR; i, j: INTEGER ): BOOLEAN;
  825. BEGIN
  826. IF (name[i] = 0X) & (pat[j] = 0X) THEN RETURN TRUE
  827. ELSIF pat[j] # "*" THEN RETURN (name[i] = pat[j]) & Match( name, pat, i + 1, j + 1 )
  828. ELSE (* pat[j] = "*", name[i] may be 0X *)
  829. RETURN Match( name, pat, i, j + 1 ) OR ((name[i] # 0X) & Match( name, pat, i + 1, j ))
  830. END
  831. END Match;
  832. PROCEDURE Append( VAR a: Filename; CONST this: ARRAY OF CHAR );
  833. VAR i, j: LONGINT;
  834. BEGIN
  835. i := 0; j := 0;
  836. WHILE a[i] # 0X DO INC( i ) END;
  837. WHILE (i < LEN( a ) - 1) & (this[j] # 0X) DO a[i] := this[j]; INC( i ); INC( j ) END;
  838. a[i] := 0X
  839. END Append;
  840. PROCEDURE AppendName( VAR path: Filename; CONST filename: ARRAY OF CHAR );
  841. VAR i, j, max: LONGINT;
  842. BEGIN
  843. i := 0; j := 0; max := LEN( path ) - 1;
  844. WHILE path[i] # 0X DO INC( i ) END;
  845. IF (i > 0) & (path[i - 1] # "/") THEN path[i] := "/"; INC( i ); path[i] := 0X END;
  846. Append( path, filename );
  847. END AppendName;
  848. PROCEDURE AppendInt( VAR str: Filename; n: LONGINT );
  849. VAR i: LONGINT;
  850. BEGIN
  851. i := 0;
  852. WHILE str[i] # 0X DO INC(i) END;
  853. WHILE n > 0 DO str[i] := CHR( n MOD 10 + ORD('0') ); n := n DIV 10; INC(i) END;
  854. str[i] := 0X
  855. END AppendInt;
  856. PROCEDURE IsFullName( CONST name: ARRAY OF CHAR ): BOOLEAN;
  857. VAR i: INTEGER; ch: CHAR;
  858. BEGIN
  859. i := 0; ch := name[0];
  860. WHILE (ch # 0X) & (ch # "/") DO INC( i ); ch := name[i] END;
  861. RETURN ch = "/"
  862. END IsFullName;
  863. PROCEDURE Halt( f: File; unixError: BOOLEAN; CONST msg: ARRAY OF CHAR );
  864. VAR fd, errno: LONGINT;
  865. workName, registerName: Filename;
  866. BEGIN
  867. IF f = NIL THEN
  868. workName := "???"; registerName := "???"
  869. ELSE
  870. workName := f.workName; registerName := f.registerName; fd := f.fd
  871. END;
  872. IF unixError THEN errno := Unix.errno( ); Unix.Perror( msg ) END;
  873. HALT( 99 )
  874. END Halt;
  875. PROCEDURE ResetBuffers( f: File; VAR stat: Unix.Status );
  876. VAR i: INTEGER;
  877. BEGIN
  878. f.fsize := stat.size;
  879. IF (f.mtime # stat.mtime.sec) THEN
  880. FOR i := 0 TO NBufs - 1 DO
  881. IF f.bufs[i] # NIL THEN f.bufs[i].org := -1; f.bufs[i] := NIL END;
  882. END;
  883. f.swapper := -1; f.mtime := stat.mtime.sec
  884. END
  885. END ResetBuffers;
  886. PROCEDURE MakePath( CONST dir, name: ARRAY OF CHAR; VAR dest: ARRAY OF CHAR );
  887. VAR i, j: INTEGER;
  888. BEGIN
  889. i := 0; j := 0;
  890. WHILE dir[i] # 0X DO dest[i] := dir[i]; INC( i ) END;
  891. IF (i>0) & (dest[i - 1] # "/") THEN dest[i] := "/"; INC( i ) END;
  892. WHILE name[j] # 0X DO dest[i] := name[j]; INC( i ); INC( j ) END;
  893. dest[i] := 0X
  894. END MakePath;
  895. PROCEDURE ScanPath( VAR pos: LONGINT; VAR dir: ARRAY OF CHAR );
  896. VAR i: LONGINT; ch: CHAR;
  897. BEGIN
  898. i := 0; ch := searchPath[pos];
  899. WHILE ch = " " DO INC( pos ); ch := searchPath[pos] END;
  900. WHILE ch > " " DO dir[i] := ch; INC( i ); INC( pos ); ch := searchPath[pos] END;
  901. dir[i] := 0X
  902. END ScanPath;
  903. PROCEDURE GetTempName( CONST finalName: ARRAY OF CHAR; VAR tempName: Filename );
  904. VAR n, i, j, pe, pid: LONGINT;
  905. BEGIN
  906. INC(tempno); n := tempno; i := 0; j := 0; pe := 1;
  907. WHILE finalName[j] = ' ' DO INC(j) END; (* skip leading spaces *)
  908. IF finalName[j] # "/" THEN (* relative pathname *)
  909. WHILE cwd[i] # 0X DO tempName[i] := cwd[i]; INC(i) END;
  910. IF tempName[i - 1] # '/' THEN tempName[i] := '/'; INC(i) END;
  911. pe := i - 1
  912. END;
  913. WHILE finalName[j] # 0X DO tempName[i] := finalName[j]; INC(i); INC(j) END;
  914. WHILE (i > pe) & (tempName[i-1] # '/') DO DEC(i) END; (* remove filename *)
  915. tempName[i] := 0X;
  916. Append( tempName, ".tmp." );
  917. AppendInt( tempName, n ); Append( tempName, "." );
  918. pid := Unix.getpid();
  919. AppendInt( tempName, pid )
  920. END GetTempName;
  921. PROCEDURE Install;
  922. VAR aliasFS: AliasFileSystem;
  923. BEGIN
  924. NEW(collection);
  925. NEW( unixFS ); (* Files.Add( unixFS, "" ); *)
  926. NEW( aliasFS, unixFS ); Files.Add( aliasFS, "searcher" )
  927. END Install;
  928. PROCEDURE Initialize;
  929. VAR a: ADDRESS; i: INTEGER; ch: CHAR;
  930. BEGIN
  931. (* get current working directory *)
  932. a := Unix.getenv( ADDRESSOF( "PWD" ) );
  933. IF a > 0 THEN
  934. i := 0;
  935. REPEAT S.GET( a, ch ); INC( a ); cwd[i] := ch; INC( i ) UNTIL ch = 0X;
  936. ELSE
  937. (* $PWD not set *)
  938. a := Unix.getcwd( ADDRESSOF( cwd ), LEN( cwd ) )
  939. END;
  940. i := 0;
  941. WHILE cwd[i] # 0X DO INC( i ) END;
  942. DEC( i );
  943. IF (i > 0) & (cwd[i] = '/') THEN cwd[i] := 0X END;
  944. (* get search pathes *)
  945. a := Unix.getenv( ADDRESSOF( "AOSPATH" ) ); i := 0;
  946. IF a = 0 THEN
  947. Log.String( "UnixFiles.Initialize: environment variable AOSPATH not defined" ); Log.Ln;
  948. ELSE
  949. REPEAT
  950. S.GET( a, ch ); INC( a );
  951. IF ch = ":" THEN ch := " " END;
  952. searchPath[i] := ch; INC( i )
  953. UNTIL ch = 0X;
  954. END;
  955. tempno := 1; openfiles := 0;
  956. Modules.InstallTermHandler( Finalization )
  957. END Initialize;
  958. PROCEDURE AddSearchPath*(context: Commands.Context);
  959. VAR name: Files.FileName; i,j: LONGINT; ch : CHAR;
  960. BEGIN
  961. IF context.arg.GetString(name) THEN
  962. i := 0; j := 0;
  963. WHILE(searchPath[i] # 0X) DO
  964. INC(i);
  965. END;
  966. searchPath[i] := " ";INC(i);
  967. REPEAT
  968. ch := name[j];
  969. searchPath[i] := name[j];
  970. INC(j);INC(i);
  971. UNTIL ch = 0X;
  972. END;
  973. END AddSearchPath;
  974. PROCEDURE Finalization;
  975. VAR ft: Files.FileSystemTable; i: LONGINT;
  976. BEGIN
  977. Files.GetList( ft );
  978. IF ft # NIL THEN
  979. FOR i := 0 TO LEN( ft^ ) - 1 DO
  980. IF ft[i] IS AliasFileSystem THEN Files.Remove( ft[i] ) END
  981. END
  982. END;
  983. collection.Finalize;
  984. unixFS.Finalize;
  985. END Finalization;
  986. PROCEDURE DecOpenFiles;
  987. BEGIN{EXCLUSIVE}
  988. DEC(openfiles);
  989. END DecOpenFiles;
  990. PROCEDURE IncOpenFiles;
  991. BEGIN{EXCLUSIVE}
  992. INC(openfiles);
  993. END IncOpenFiles;
  994. PROCEDURE WaitClose(no: LONGINT);
  995. BEGIN{EXCLUSIVE}
  996. AWAIT(openfiles < no);
  997. END WaitClose;
  998. BEGIN
  999. Initialize;
  1000. Install
  1001. END UnixFiles.