Unix.UnixFiles.Mod 35 KB

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