DiskFS.Mod 43 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429
  1. (* Aos, Copyright 2001, Pieter Muller, ETH Zurich *)
  2. MODULE DiskFS; (** AUTHOR "pjm"; PURPOSE "Aos disk file system"; *)
  3. IMPORT SYSTEM, Machine, KernelLog, Modules, Clock, Files, Kernel;
  4. CONST
  5. SkipIndexFlag = 31; (* DiskFS filesystem flag. Do not write index map back to disk when unmounting *)
  6. MinVolSize = 4;
  7. SectorFactor = 29;
  8. (* WARNING: When the maximum length of filenames is changed, volumes must be re-formatted!!! *)
  9. FileNameLength = 128; (* includes 0X *)
  10. SectorTableSize = 128;
  11. SectorSize = 4096;
  12. IndexSize = SectorSize DIV 4;
  13. DiskAdrSize = 4; (* bytes *)
  14. HeaderSize = 4 (* mark *) + FileNameLength + 4*4 (* aleng, bleng, time, date *) + (SectorTableSize+1)*DiskAdrSize;
  15. DirEntrySize = FileNameLength + 2*DiskAdrSize (* adr, p *);
  16. DirPgHeaderSize = 2*4 (* mark, m *) + DiskAdrSize (* p0 *) + 4 (* min. FillerSize *);
  17. DirPgSize = (SectorSize - DirPgHeaderSize) DIV DirEntrySize;
  18. FillerSize = (SectorSize - DirPgHeaderSize) MOD DirEntrySize + 4 (* min. FillerSize *);
  19. DirRootAdr = 1*SectorFactor;
  20. N = DirPgSize DIV 2;
  21. DirMark = LONGINT(9B1EA38DH);
  22. HeaderMark = LONGINT(9BA71D86H);
  23. MapIndexSize = (SectorSize-4) DIV 4;
  24. MapSize = SectorSize DIV SIZEOF (SET); (* {MapSize MOD SIZEOF (SET) = 0} *)
  25. MapMark = LONGINT(9C2F977FH);
  26. MaxBufs = 1024;
  27. InitHint = 200*SectorFactor;
  28. Closed = 0X; Opening = 1X; Opened = 2X; Closing = 3X;
  29. SetSize = MAX (SET) + 1;
  30. TYPE
  31. DiskSector = RECORD END; (* Oberon Sector, size SectorSize *)
  32. DiskSectorArr = ARRAY SectorSize OF CHAR;
  33. DiskAdr = LONGINT;
  34. FileName = ARRAY FileNameLength OF CHAR;
  35. SectorTable = ARRAY SectorTableSize OF DiskAdr;
  36. FileHeader = RECORD (DiskSector) (* allocated in the first page of each file on disk *)
  37. mark: LONGINT;
  38. name: FileName;
  39. aleng, bleng: LONGINT;
  40. date, time: LONGINT;
  41. sec: SectorTable;
  42. ext: DiskAdr;
  43. data: ARRAY SectorSize-HeaderSize OF CHAR
  44. END;
  45. IndexSector = RECORD (DiskSector)
  46. x: ARRAY IndexSize OF DiskAdr
  47. END;
  48. DataSector = RECORD (DiskSector)
  49. B: ARRAY SectorSize OF CHAR
  50. END;
  51. DirEntry = RECORD (*B-tree node*)
  52. name: FileName;
  53. adr: DiskAdr; (*sec no of file header*)
  54. p: DiskAdr (*sec no of descendant in directory*)
  55. END;
  56. DirPage = RECORD (DiskSector)
  57. mark: LONGINT;
  58. m: LONGINT;
  59. p0: DiskAdr; (*sec no of left descendant in directory*)
  60. fill: ARRAY FillerSize OF CHAR;
  61. e: ARRAY DirPgSize OF DirEntry
  62. END;
  63. MapIndex = RECORD (DiskSector)
  64. mark: LONGINT;
  65. index: ARRAY MapIndexSize OF DiskAdr
  66. END;
  67. MapSector = RECORD (DiskSector)
  68. map: ARRAY MapSize OF SET
  69. END;
  70. Buffer = POINTER TO RECORD (Files.Hint)
  71. apos, lim: LONGINT;
  72. mod: BOOLEAN;
  73. next: Buffer;
  74. data: DataSector
  75. END;
  76. SuperIndex = POINTER TO RECORD
  77. adr: DiskAdr;
  78. mod: BOOLEAN;
  79. sub: ARRAY IndexSize OF SubIndex
  80. END;
  81. SubIndex = POINTER TO RECORD
  82. adr: DiskAdr;
  83. mod: BOOLEAN;
  84. sec: IndexSector
  85. END;
  86. TYPE
  87. Directory = OBJECT
  88. VAR
  89. vol: Files.Volume;
  90. state: CHAR;
  91. lastSectorReserved, noCleanup: BOOLEAN;
  92. (* "exported" methods: Search, Insert, Delete *)
  93. PROCEDURE Search(VAR name: FileName; VAR A: DiskAdr);
  94. VAR i, L, R: LONGINT; dadr: DiskAdr; a: DirPage;
  95. BEGIN {EXCLUSIVE}
  96. ASSERT(state = Opened);
  97. dadr := DirRootAdr;
  98. LOOP
  99. GetSector(vol, dadr, a);
  100. ASSERT(a.mark = DirMark);
  101. L := 0; R := a.m; (*binary search*)
  102. WHILE L < R DO
  103. i := (L+R) DIV 2;
  104. IF name <= a.e[i].name THEN R := i ELSE L := i+1 END
  105. END ;
  106. IF (R < a.m) & (name = a.e[R].name) THEN
  107. A := a.e[R].adr; EXIT (*found*)
  108. END ;
  109. IF R = 0 THEN dadr := a.p0 ELSE dadr := a.e[R-1].p END ;
  110. IF dadr = 0 THEN A := 0; EXIT (*not found*) END
  111. END
  112. END Search;
  113. PROCEDURE insert(VAR name: FileName; dpg0: DiskAdr; VAR h: BOOLEAN; VAR v: DirEntry; fad: DiskAdr);
  114. (*h = "tree has become higher and v is ascending element"*)
  115. VAR ch: CHAR; i, j, L, R: LONGINT; dpg1: DiskAdr; u: DirEntry; a: DirPage;
  116. BEGIN (*~h*)
  117. ASSERT(state = Opened);
  118. GetSector(vol, dpg0, a);
  119. L := 0; R := a.m; (*binary search*)
  120. WHILE L < R DO
  121. i := (L+R) DIV 2;
  122. IF name <= a.e[i].name THEN R := i ELSE L := i+1 END
  123. END ;
  124. IF (R < a.m) & (name = a.e[R].name) THEN
  125. a.e[R].adr := fad; PutSector(vol, dpg0, a) (*replace*)
  126. ELSE (*not on this page*)
  127. IF R = 0 THEN dpg1 := a.p0 ELSE dpg1 := a.e[R-1].p END ;
  128. IF dpg1 = 0 THEN (*not in tree, insert*)
  129. u.adr := fad; u.p := 0; h := TRUE; j := 0;
  130. REPEAT ch := name[j]; u.name[j] := ch; INC(j)
  131. UNTIL ch = 0X;
  132. WHILE j < FileNameLength DO u.name[j] := 0X; INC(j) END
  133. ELSE
  134. insert(name, dpg1, h, u, fad)
  135. END ;
  136. IF h THEN (*insert u to the left of e[R]*)
  137. IF a.m < DirPgSize THEN
  138. h := FALSE; i := a.m;
  139. WHILE i > R DO DEC(i); a.e[i+1] := a.e[i] END ;
  140. a.e[R] := u; INC(a.m)
  141. ELSE (*split page and assign the middle element to v*)
  142. a.m := N; a.mark := DirMark;
  143. IF R < N THEN (*insert in left half*)
  144. v := a.e[N-1]; i := N-1;
  145. WHILE i > R DO DEC(i); a.e[i+1] := a.e[i] END ;
  146. a.e[R] := u; PutSector(vol, dpg0, a);
  147. AllocSector(vol, dpg0, dpg0); i := 0;
  148. WHILE i < N DO a.e[i] := a.e[i+N]; INC(i) END
  149. ELSE (*insert in right half*)
  150. PutSector(vol, dpg0, a);
  151. AllocSector(vol, dpg0, dpg0); DEC(R, N); i := 0;
  152. IF R = 0 THEN v := u
  153. ELSE v := a.e[N];
  154. WHILE i < R-1 DO a.e[i] := a.e[N+1+i]; INC(i) END ;
  155. a.e[i] := u; INC(i)
  156. END ;
  157. WHILE i < N DO a.e[i] := a.e[N+i]; INC(i) END
  158. END ;
  159. a.p0 := v.p; v.p := dpg0
  160. END ;
  161. PutSector(vol, dpg0, a)
  162. END
  163. END
  164. END insert;
  165. PROCEDURE Insert(VAR name: FileName; fad: DiskAdr);
  166. VAR oldroot: DiskAdr; h: BOOLEAN; U: DirEntry; a: DirPage;
  167. BEGIN {EXCLUSIVE}
  168. h := FALSE;
  169. insert(name, DirRootAdr, h, U, fad);
  170. IF h THEN (*root overflow*)
  171. GetSector(vol, DirRootAdr, a);
  172. AllocSector(vol, DirRootAdr, oldroot); PutSector(vol, oldroot, a);
  173. a.mark := DirMark; a.m := 1; a.p0 := oldroot; a.e[0] := U;
  174. PutSector(vol, DirRootAdr, a)
  175. END
  176. END Insert;
  177. PROCEDURE underflow(VAR c: DirPage; (*ancestor page*) dpg0: DiskAdr; s: LONGINT; (*insertion point in c*)
  178. VAR h: BOOLEAN); (*c undersize*)
  179. VAR i, k: LONGINT; dpg1: DiskAdr; a, b: DirPage; (*a := underflowing page, b := neighbouring page*)
  180. BEGIN
  181. GetSector(vol, dpg0, a);
  182. (*h & a.m = N-1 & dpg0 = c.e[s-1].p*)
  183. IF s < c.m THEN (*b := page to the right of a*)
  184. dpg1 := c.e[s].p; GetSector(vol, dpg1, b);
  185. k := (b.m-N+1) DIV 2; (*k = no. of items available on page b*)
  186. a.e[N-1] := c.e[s]; a.e[N-1].p := b.p0;
  187. IF k > 0 THEN
  188. (*move k-1 items from b to a, one to c*) i := 0;
  189. WHILE i < k-1 DO a.e[i+N] := b.e[i]; INC(i) END ;
  190. c.e[s] := b.e[i]; b.p0 := c.e[s].p;
  191. c.e[s].p := dpg1; DEC(b.m, k); i := 0;
  192. WHILE i < b.m DO b.e[i] := b.e[i+k]; INC(i) END ;
  193. PutSector(vol, dpg1, b); a.m := N-1+k; h := FALSE
  194. ELSE (*merge pages a and b, discard b*) i := 0;
  195. WHILE i < N DO a.e[i+N] := b.e[i]; INC(i) END ;
  196. i := s; DEC(c.m);
  197. WHILE i < c.m DO c.e[i] := c.e[i+1]; INC(i) END ;
  198. a.m := 2*N; h := c.m < N;
  199. FreeSector(vol, dpg1) (* free b *)
  200. END ;
  201. PutSector(vol, dpg0, a)
  202. ELSE (*b := page to the left of a*) DEC(s);
  203. IF s = 0 THEN dpg1 := c.p0 ELSE dpg1 := c.e[s-1].p END ;
  204. GetSector(vol, dpg1, b);
  205. k := (b.m-N+1) DIV 2; (*k = no. of items available on page b*)
  206. IF k > 0 THEN
  207. i := N-1;
  208. WHILE i > 0 DO DEC(i); a.e[i+k] := a.e[i] END ;
  209. i := k-1; a.e[i] := c.e[s]; a.e[i].p := a.p0;
  210. (*move k-1 items from b to a, one to c*) DEC(b.m, k);
  211. WHILE i > 0 DO DEC(i); a.e[i] := b.e[i+b.m+1] END ;
  212. c.e[s] := b.e[b.m]; a.p0 := c.e[s].p;
  213. c.e[s].p := dpg0; a.m := N-1+k; h := FALSE;
  214. PutSector(vol, dpg0, a)
  215. ELSE (*merge pages a and b, discard a*)
  216. c.e[s].p := a.p0; b.e[N] := c.e[s]; i := 0;
  217. WHILE i < N-1 DO b.e[i+N+1] := a.e[i]; INC(i) END ;
  218. b.m := 2*N; DEC(c.m); h := c.m < N;
  219. FreeSector(vol, dpg0) (* free a *)
  220. END ;
  221. PutSector(vol, dpg1, b)
  222. END
  223. END underflow;
  224. PROCEDURE delete(VAR name: FileName; dpg0: DiskAdr; VAR h: BOOLEAN; VAR fad: DiskAdr);
  225. (*search and delete entry with key name; if a page underflow arises,
  226. balance with adjacent page or merge; h := "page dpg0 is undersize"*)
  227. VAR i, L, R: LONGINT; dpg1: DiskAdr; a: DirPage;
  228. PROCEDURE del(dpg1: DiskAdr; VAR h: BOOLEAN);
  229. VAR dpg2: DiskAdr; (*global: a, R*) b: DirPage;
  230. BEGIN
  231. GetSector(vol, dpg1, b); dpg2 := b.e[b.m-1].p;
  232. IF dpg2 # 0 THEN del(dpg2, h);
  233. IF h THEN underflow(b, dpg2, b.m, h); PutSector(vol, dpg1, b) END
  234. ELSE
  235. b.e[b.m-1].p := a.e[R].p; a.e[R] := b.e[b.m-1];
  236. DEC(b.m); h := b.m < N; PutSector(vol, dpg1, b)
  237. END
  238. END del;
  239. BEGIN (*~h*)
  240. ASSERT(state = Opened);
  241. GetSector(vol, dpg0, a);
  242. L := 0; R := a.m; (*binary search*)
  243. WHILE L < R DO
  244. i := (L+R) DIV 2;
  245. IF name <= a.e[i].name THEN R := i ELSE L := i+1 END
  246. END ;
  247. IF R = 0 THEN dpg1 := a.p0 ELSE dpg1 := a.e[R-1].p END ;
  248. IF (R < a.m) & (name = a.e[R].name) THEN
  249. (*found, now delete*) fad := a.e[R].adr;
  250. IF dpg1 = 0 THEN (*a is a leaf page*)
  251. DEC(a.m); h := a.m < N; i := R;
  252. WHILE i < a.m DO a.e[i] := a.e[i+1]; INC(i) END
  253. ELSE del(dpg1, h);
  254. IF h THEN underflow(a, dpg1, R, h) END
  255. END ;
  256. PutSector(vol, dpg0, a)
  257. ELSIF dpg1 # 0 THEN
  258. delete(name, dpg1, h, fad);
  259. IF h THEN underflow(a, dpg1, R, h); PutSector(vol, dpg0, a) END
  260. ELSE (*not in tree*) fad := 0
  261. END
  262. END delete;
  263. PROCEDURE Delete(VAR name: FileName; VAR fad: DiskAdr);
  264. VAR h: BOOLEAN; newroot: DiskAdr; a: DirPage;
  265. BEGIN {EXCLUSIVE}
  266. h := FALSE;
  267. delete(name, DirRootAdr, h, fad);
  268. IF h THEN (*root underflow*)
  269. GetSector(vol, DirRootAdr, a);
  270. IF (a.m = 0) & (a.p0 # 0) THEN
  271. newroot := a.p0; GetSector(vol, newroot, a);
  272. PutSector(vol, DirRootAdr, a); (*discard newroot*)
  273. FreeSector(vol, newroot)
  274. END
  275. END
  276. END Delete;
  277. PROCEDURE Startup;
  278. VAR
  279. j, sec, size, q, free, thres: LONGINT; mi: MapIndex; ms: MapSector;
  280. s: ARRAY 10 OF CHAR; found: BOOLEAN;
  281. BEGIN (* only called from Init *)
  282. size := vol.size; found := FALSE;
  283. IF (vol.Available() = size) & (size # 0) THEN (* all sectors available *)
  284. GetSector(vol, size*SectorFactor, mi);
  285. IF mi.mark = MapMark THEN
  286. j := 0; (* check consistency of index *)
  287. WHILE (j # MapIndexSize) & (mi.index[j] >= 0) & (mi.index[j] MOD SectorFactor = 0) DO
  288. INC(j)
  289. END;
  290. IF j = MapIndexSize THEN
  291. found := TRUE;
  292. mi.mark := 0; PutSector(vol, size*SectorFactor, mi); (* invalidate index *)
  293. j := 0; sec := 1; q := 0;
  294. LOOP
  295. IF (j = MapIndexSize) OR (mi.index[j] = 0) THEN EXIT END;
  296. GetSector(vol, mi.index[j], ms);
  297. REPEAT
  298. IF (sec MOD SetSize) IN ms.map[sec DIV SetSize MOD MapSize] THEN
  299. MarkSector(vol, sec*SectorFactor);
  300. INC(q)
  301. END;
  302. IF sec = size THEN EXIT END;
  303. INC(sec)
  304. UNTIL sec MOD (MapSize*SetSize) = 0;
  305. INC(j)
  306. END;
  307. Machine.GetConfig("DiskGC", s);
  308. thres := 0; j := 0;
  309. WHILE s[j] # 0X DO thres := thres*10+(ORD(s[j])-48); INC(j) END;
  310. IF thres < 10 THEN thres := 10
  311. ELSIF thres > 100 THEN thres := 100
  312. END;
  313. ASSERT(q = size-vol.Available());
  314. free := vol.Available()*100 DIV size;
  315. IF (free > thres) & (vol.Available() > 100000H DIV SectorSize) THEN
  316. state := Opened
  317. ELSE (* undo *)
  318. FOR j := SectorFactor TO size*SectorFactor BY SectorFactor DO
  319. IF Marked(vol, j) THEN FreeSector(vol, j) END
  320. END;
  321. ASSERT(vol.Available() = size);
  322. KernelLog.String("DiskFS: "); KernelLog.Int(free, 1);
  323. KernelLog.String("% free, forcing disk GC on ");
  324. KernelLog.String(vol.name); KernelLog.Ln
  325. END
  326. END
  327. END;
  328. IF ~found THEN
  329. KernelLog.String("DiskFS: Index not found on ");
  330. KernelLog.String(vol.name); KernelLog.Ln
  331. END
  332. END
  333. END Startup;
  334. PROCEDURE &Init*(vol: Files.Volume);
  335. VAR k: LONGINT; A: ARRAY 2000 OF DiskAdr; files: LONGINT; bad: BOOLEAN;
  336. PROCEDURE MarkSectors;
  337. VAR L, R, i, j, n: LONGINT; x: DiskAdr; hd: FileHeader; sup, sub: IndexSector;
  338. PROCEDURE sift(L, R: LONGINT);
  339. VAR i, j: LONGINT; x: DiskAdr;
  340. BEGIN j := L; x := A[j];
  341. LOOP i := j; j := 2*j + 1;
  342. IF (j+1 < R) & (A[j] < A[j+1]) THEN INC(j) END ;
  343. IF (j >= R) OR (x > A[j]) THEN EXIT END ;
  344. A[i] := A[j]
  345. END ;
  346. A[i] := x
  347. END sift;
  348. BEGIN
  349. KernelLog.String(" marking");
  350. L := k DIV 2; R := k; (*heapsort*)
  351. WHILE L > 0 DO DEC(L); sift(L, R) END ;
  352. WHILE R > 0 DO
  353. DEC(R); x := A[0]; A[0] := A[R]; A[R] := x; sift(L, R)
  354. END;
  355. WHILE L < k DO
  356. bad := FALSE; INC(files);
  357. IF files MOD 128 = 0 THEN KernelLog.Char(".") END;
  358. GetSector(vol, A[L], hd);
  359. IF hd.aleng < SectorTableSize THEN
  360. j := hd.aleng + 1;
  361. REPEAT
  362. DEC(j);
  363. IF hd.sec[j] # 0 THEN MarkSector(vol, hd.sec[j]) ELSE hd.aleng := j-1; bad := TRUE END
  364. UNTIL j = 0
  365. ELSE
  366. j := SectorTableSize;
  367. REPEAT
  368. DEC(j);
  369. IF hd.sec[j] # 0 THEN MarkSector(vol, hd.sec[j]) ELSE hd.aleng := j-1; bad := TRUE END
  370. UNTIL j = 0;
  371. IF hd.ext = 0 THEN hd.aleng := SectorTableSize-1; bad := TRUE END;
  372. IF ~bad THEN
  373. MarkSector(vol, hd.ext); GetSector(vol, hd.ext, sup);
  374. n := (hd.aleng - SectorTableSize) DIV IndexSize; i := 0;
  375. WHILE (i <= n) & ~bad DO
  376. IF sup.x[i] # 0 THEN
  377. MarkSector(vol, sup.x[i]); GetSector(vol, sup.x[i], sub);
  378. IF i < n THEN j := IndexSize
  379. ELSE j := (hd.aleng - SectorTableSize) MOD IndexSize + 1
  380. END;
  381. REPEAT
  382. DEC(j);
  383. IF (sub.x[j] MOD SectorFactor = 0) & (sub.x[j] > 0) THEN
  384. MarkSector(vol, sub.x[j])
  385. ELSE
  386. bad := TRUE
  387. END
  388. UNTIL j = 0;
  389. INC(i)
  390. ELSE bad := TRUE
  391. END;
  392. IF bad THEN
  393. IF i = 0 THEN hd.aleng := SectorTableSize-1
  394. ELSE hd.aleng := SectorTableSize + (i-1) * IndexSize
  395. END
  396. END
  397. END
  398. END
  399. END;
  400. IF bad THEN
  401. KernelLog.Ln; KernelLog.String(hd.name); KernelLog.String(" truncated");
  402. hd.bleng := SectorSize; IF hd.aleng < 0 THEN hd.aleng := 0 (* really bad *) END;
  403. PutSector(vol, A[L], hd)
  404. END;
  405. INC(L)
  406. END
  407. END MarkSectors;
  408. PROCEDURE TraverseDir(dpg: DiskAdr);
  409. VAR i: LONGINT; a: DirPage;
  410. BEGIN
  411. GetSector(vol, dpg, a); MarkSector(vol, dpg); i := 0;
  412. WHILE i < a.m DO
  413. A[k] := a.e[i].adr;
  414. (*
  415. IF A[k] = 0DEADDEADH THEN
  416. KernelLog.Enter; KernelLog.Int(dpg DIV SectorFactor, 1); KernelLog.Char(" "); KernelLog.Int(k, 1); KernelLog.Exit
  417. END;
  418. *)
  419. INC(k); INC(i);
  420. IF k = 2000 THEN MarkSectors; k := 0 END
  421. END ;
  422. IF a.p0 # 0 THEN
  423. TraverseDir(a.p0); i := 0;
  424. WHILE i < a.m DO
  425. TraverseDir(a.e[i].p); INC(i)
  426. END
  427. END
  428. END TraverseDir;
  429. BEGIN
  430. SELF.vol := vol; lastSectorReserved := FALSE;
  431. IF ~(Files.ReadOnly IN vol.flags) THEN
  432. state := Opening; k := 0;
  433. Startup;
  434. IF state # Opened THEN
  435. files := 0; KernelLog.String("DiskFS: Scanning ");
  436. KernelLog.String(vol.name); KernelLog.String("...");
  437. TraverseDir(DirRootAdr);
  438. MarkSectors;
  439. KernelLog.Int(files, 6); KernelLog.String(" files"); KernelLog.Ln;
  440. state := Opened
  441. END;
  442. IF ~Marked(vol, vol.size*SectorFactor) THEN (* last sector still free *)
  443. MarkSector(vol, vol.size*SectorFactor); lastSectorReserved := TRUE (* allocate it *)
  444. END;
  445. KernelLog.String("DiskFS: "); KernelLog.Int(vol.Available() * (SectorSize DIV 1024), 1);
  446. KernelLog.String("K of "); KernelLog.Int(vol.size * (SectorSize DIV 1024), 1);
  447. KernelLog.String("K available on "); KernelLog.String(vol.name);
  448. KernelLog.Ln
  449. ELSE
  450. state := Opened
  451. END
  452. END Init;
  453. PROCEDURE Cleanup;
  454. VAR i, j, p, q, sec, size: LONGINT; mi: MapIndex; ms: MapSector;
  455. BEGIN {EXCLUSIVE}
  456. (*KernelLog.String("DiskFS: Cleanup "); KernelLog.String(vol.name); KernelLog.Ln;*)
  457. state := Closing;
  458. size := vol.size; i := size*SectorFactor;
  459. IF ~(Files.ReadOnly IN vol.flags) & ~noCleanup THEN
  460. IF lastSectorReserved THEN FreeSector(vol, i); lastSectorReserved := FALSE END;
  461. IF ~Marked(vol, i) THEN (* last sector is available for us *)
  462. j := 0; sec := 1; q := 0;
  463. LOOP
  464. REPEAT DEC(i, SectorFactor) UNTIL (i = 0) OR ~Marked(vol, i); (* find a free sector *)
  465. IF i = 0 THEN RETURN END; (* no more space, don't commit *)
  466. mi.index[j] := i; INC(j);
  467. FOR p := 0 TO MapSize-1 DO ms.map[p] := {} END;
  468. REPEAT
  469. IF Marked(vol, sec*SectorFactor) THEN
  470. INCL(ms.map[sec DIV SetSize MOD MapSize], sec MOD SetSize);
  471. INC(q)
  472. END;
  473. IF sec = size THEN
  474. PutSector(vol, i, ms);
  475. EXIT
  476. END;
  477. INC(sec)
  478. UNTIL sec MOD (MapSize*SetSize) = 0;
  479. PutSector(vol, i, ms)
  480. END;
  481. WHILE j # MapIndexSize DO mi.index[j] := 0; INC(j) END;
  482. mi.mark := MapMark;
  483. PutSector(vol, size*SectorFactor, mi); (* commit *)
  484. KernelLog.String("DiskFS: Map saved on ");
  485. KernelLog.String(vol.name); KernelLog.Ln
  486. (*ELSE
  487. KernelLog.String("DiskFS: sector in use "); KernelLog.Int(size, 1); KernelLog.Ln*)
  488. END
  489. (*ELSE
  490. KernelLog.String("DiskFS: Read-only"); KernelLog.Ln*)
  491. END;
  492. state := Closed; vol := NIL
  493. END Cleanup;
  494. END Directory;
  495. TYPE
  496. FileSystem = OBJECT (Files.FileSystem) (* our file system type *)
  497. VAR
  498. dir: Directory;
  499. finalizeFiles: Kernel.FinalizedCollection;
  500. openFiles: DiskAdrList;
  501. (* all files that are registered, must be stored separately of finalizeFiles because of race
  502. between Delete0/Rename0 and deferred execution of file close finalizer *)
  503. tempRegFileSec: DiskAdrList; (* temporary used for PurgeOpenedFile *)
  504. PROCEDURE &Init*;
  505. BEGIN NEW(finalizeFiles); NEW(openFiles); NEW(tempRegFileSec)
  506. END Init;
  507. PROCEDURE New0(name: ARRAY OF CHAR): Files.File;
  508. VAR i, res: LONGINT; f: File; buf: Buffer; head {UNTRACED}: POINTER {UNSAFE} TO FileHeader; namebuf: FileName;
  509. BEGIN {EXCLUSIVE}
  510. f := NIL; Check(name, namebuf, res);
  511. IF res <= 0 THEN
  512. NEW(buf); buf.apos := 0; buf.mod := TRUE; buf.lim := HeaderSize; buf.next := buf;
  513. head := ADDRESSOF(buf.data);
  514. head.mark := HeaderMark;
  515. head.aleng := 0; head.bleng := HeaderSize; head.name := namebuf;
  516. Clock.Get(head.time, head.date);
  517. NEW(f); f.fs := SELF; f.key := 0; f.aleng := 0; f.bleng := HeaderSize; f.modH := TRUE;
  518. f.time := head.time; f.date := head.date;
  519. f.firstbuf := buf; f.nofbufs := 1; f.name := namebuf; f.sechint := InitHint;
  520. f.registered := (f.name[0] = 0X);
  521. f.ext := NIL; i := 0;
  522. REPEAT f.sec[i] := 0; head.sec[i] := 0; INC(i) UNTIL i = SectorTableSize;
  523. finalizeFiles.Add(f, Collect);
  524. ELSE
  525. KernelLog.String("DiskFS: "); KernelLog.String(name); KernelLog.String(", res: "); KernelLog.Int(res, 0); KernelLog.Ln;
  526. END;
  527. RETURN f
  528. END New0;
  529. PROCEDURE Old0(name: ARRAY OF CHAR): Files.File;
  530. VAR
  531. i, k, res: LONGINT; f: File; header: DiskAdr; buf: Buffer; head {UNTRACED}: POINTER {UNSAFE} TO FileHeader;
  532. namebuf: FileName; super: SuperIndex; sub: SubIndex; sec: IndexSector;
  533. BEGIN {EXCLUSIVE}
  534. f := NIL; Check(name, namebuf, res);
  535. IF res = 0 THEN
  536. dir.Search(namebuf, header);
  537. IF header # 0 THEN
  538. NEW(buf); buf.apos := 0; buf.next := buf; buf.mod := FALSE;
  539. GetSector(vol, header, buf.data);
  540. head := ADDRESSOF(buf.data);
  541. NEW(f); f.fs := SELF; f.key := header;
  542. f.aleng := head.aleng; f.bleng := head.bleng;
  543. f.time := head.time; f.date := head.date;
  544. IF f.aleng = 0 THEN buf.lim := f.bleng ELSE buf.lim := SectorSize END;
  545. f.firstbuf := buf; f.nofbufs := 1;
  546. f.name := namebuf; f.registered := TRUE;
  547. f.sec := head.sec;
  548. k := (f.aleng + (IndexSize-SectorTableSize)) DIV IndexSize;
  549. IF k # 0 THEN
  550. NEW(super); super.adr := head.ext; super.mod := FALSE; f.ext := super;
  551. GetSector(vol, super.adr, sec); i := 0;
  552. WHILE i # k DO
  553. NEW(sub); sub.adr := sec.x[i]; sub.mod := FALSE; super.sub[i] := sub;
  554. GetSector(vol, sub.adr, sub.sec); INC(i)
  555. END;
  556. WHILE i # IndexSize DO super.sub[i] := NIL; INC(i) END
  557. ELSE
  558. f.ext := NIL
  559. END;
  560. f.sechint := header; f.modH := FALSE;
  561. finalizeFiles.Add(f, Collect); openFiles.Add(f.key)
  562. END
  563. END;
  564. RETURN f
  565. END Old0;
  566. PROCEDURE Delete0(name: ARRAY OF CHAR; VAR key, res: LONGINT);
  567. VAR adr: DiskAdr; namebuf: FileName; head: FileHeader;
  568. BEGIN {EXCLUSIVE}
  569. Check(name, namebuf, res);
  570. IF res = 0 THEN
  571. dir.Delete(namebuf, adr);
  572. key := adr;
  573. IF adr # 0 THEN
  574. IF ~openFiles.Contains(adr) THEN
  575. PurgeByAdr(adr)
  576. ELSE
  577. GetSector(vol, adr, head);
  578. head.mark := HeaderMark+1; (* invalidate mark *)
  579. PutSector(vol, adr, head)
  580. END
  581. ELSE
  582. res := 2
  583. END
  584. ELSE
  585. key := 0
  586. END
  587. END Delete0;
  588. PROCEDURE Rename0(old, new: ARRAY OF CHAR; f: Files.File; VAR res: LONGINT);
  589. VAR adr, newAdr: DiskAdr; oldbuf, newbuf: FileName; head: FileHeader;
  590. BEGIN {EXCLUSIVE}
  591. Check(old, oldbuf, res);
  592. IF res = 0 THEN
  593. Check(new, newbuf, res);
  594. IF res = 0 THEN
  595. dir.Delete(oldbuf, adr);
  596. IF adr # 0 THEN
  597. dir.Search(newbuf, newAdr); ASSERT(adr # newAdr);
  598. IF (newAdr # 0) & ~openFiles.Contains(newAdr) THEN
  599. PurgeByAdr(newAdr)
  600. END;
  601. IF f # NIL THEN (* file is open *)
  602. ASSERT(f.key = adr); (* it's key must match *)
  603. f(File).name := newbuf
  604. END;
  605. dir.Insert(newbuf, adr);
  606. GetSector(vol, adr, head);
  607. head.name := newbuf;
  608. PutSector(vol, adr, head)
  609. ELSE res := 2
  610. END
  611. END
  612. END
  613. END Rename0;
  614. PROCEDURE Enumerate0(mask: ARRAY OF CHAR; flags: SET; enum: Files.Enumerator);
  615. VAR b: BOOLEAN; fh: FileHeader; fn: ARRAY Files.PrefixLength+FileNameLength OF CHAR;
  616. BEGIN {EXCLUSIVE}
  617. b := TRUE; enumerate(SELF, mask, DirRootAdr, flags, enum, b, fh, fn)
  618. END Enumerate0;
  619. PROCEDURE FileKey(name: ARRAY OF CHAR): LONGINT;
  620. VAR res: LONGINT; namebuf: FileName; header: DiskAdr;
  621. BEGIN {EXCLUSIVE}
  622. header := 0;
  623. Check(name, namebuf, res);
  624. IF res = 0 THEN
  625. dir.Search(namebuf, header)
  626. END;
  627. RETURN header
  628. END FileKey;
  629. (* exlcusive lock must be acquired, result in tempRegFileSec *)
  630. PROCEDURE CollectRegisteredFileSectors(adr: DiskAdr);
  631. VAR hd: FileHeader; i, p, m, n: LONGINT; super, sub: IndexSector;
  632. BEGIN
  633. tempRegFileSec.Clear;
  634. GetSector(vol, adr, hd);
  635. tempRegFileSec.Add(adr);
  636. ASSERT(hd.sec[0] = adr);
  637. IF hd.aleng < SectorTableSize THEN m := hd.aleng + 1 ELSE m := SectorTableSize END; p := 1;
  638. WHILE p < m DO
  639. IF hd.sec[p] # 0 THEN tempRegFileSec.Add(hd.sec[p]) END;
  640. INC(p)
  641. END;
  642. IF (hd.aleng >= SectorTableSize) & (hd.ext # 0) THEN
  643. GetSector(vol, hd.ext, super); tempRegFileSec.Add(hd.ext);
  644. n := (hd.aleng - SectorTableSize) DIV IndexSize; i := 0;
  645. WHILE i <= n DO
  646. IF super.x[i] # 0 THEN
  647. GetSector(vol, super.x[i], sub); tempRegFileSec.Add(super.x[i]);
  648. IF i < n THEN m := IndexSize
  649. ELSE m := (hd.aleng - SectorTableSize) MOD IndexSize + 1
  650. END;
  651. p := 0;
  652. WHILE p < m DO
  653. IF sub.x[p] # 0 THEN tempRegFileSec.Add(sub.x[p]) END;
  654. INC(p)
  655. END
  656. END;
  657. INC(i)
  658. END
  659. END
  660. END CollectRegisteredFileSectors;
  661. (* exlcusive lock must be acquired! *)
  662. PROCEDURE PurgeByAdr(adr: DiskAdr);
  663. VAR hd: FileHeader; i, p, m, n: LONGINT; super, sub: IndexSector;
  664. BEGIN
  665. GetSector(vol, adr, hd);
  666. FreeSector(vol, adr);
  667. ASSERT(hd.sec[0] = adr);
  668. IF hd.aleng < SectorTableSize THEN m := hd.aleng + 1 ELSE m := SectorTableSize END; p := 1;
  669. WHILE p < m DO
  670. IF hd.sec[p] # 0 THEN FreeSector(vol, hd.sec[p]) END;
  671. INC(p)
  672. END;
  673. IF (hd.aleng >= SectorTableSize) & (hd.ext # 0) THEN
  674. GetSector(vol, hd.ext, super); FreeSector(vol, hd.ext);
  675. n := (hd.aleng - SectorTableSize) DIV IndexSize; i := 0;
  676. WHILE i <= n DO
  677. IF super.x[i] # 0 THEN
  678. GetSector(vol, super.x[i], sub); FreeSector(vol, super.x[i]);
  679. IF i < n THEN m := IndexSize
  680. ELSE m := (hd.aleng - SectorTableSize) MOD IndexSize + 1
  681. END;
  682. p := 0;
  683. WHILE p < m DO
  684. IF sub.x[p] # 0 THEN FreeSector(vol, sub.x[p]) END;
  685. INC(p)
  686. END
  687. END;
  688. INC(i)
  689. END
  690. END
  691. END PurgeByAdr;
  692. (* purge all sectors of f except the sectors in 'except', except may be NIL *)
  693. PROCEDURE PurgeOpenedFile(f: File; except: DiskAdrList);
  694. VAR i, p, m, n: LONGINT; super, sub: IndexSector;
  695. PROCEDURE FreeExcept(sec: DiskAdr);
  696. BEGIN
  697. IF (except = NIL) OR ~except.Contains(sec) THEN FreeSector(vol, sec) END
  698. END FreeExcept;
  699. BEGIN
  700. IF f.aleng < SectorTableSize THEN m := f.aleng + 1 ELSE m := SectorTableSize END; p := 0; (* include sec[0] *)
  701. WHILE p < m DO
  702. IF f.sec[p] # 0 THEN FreeExcept(f.sec[p]) END;
  703. INC(p)
  704. END;
  705. IF (f.aleng >= SectorTableSize) & (f.ext # NIL) & (f.ext.adr # 0) THEN
  706. GetSector(vol, f.ext.adr, super); FreeExcept(f.ext.adr);
  707. n := (f.aleng - SectorTableSize) DIV IndexSize; i := 0;
  708. WHILE i <= n DO
  709. IF super.x[i] # 0 THEN
  710. GetSector(vol, super.x[i], sub); FreeExcept(super.x[i]);
  711. IF i < n THEN m := IndexSize
  712. ELSE m := (f.aleng - SectorTableSize) MOD IndexSize + 1
  713. END;
  714. p := 0;
  715. WHILE p < m DO
  716. IF sub.x[p] # 0 THEN FreeExcept(sub.x[p]) END;
  717. INC(p)
  718. END
  719. END;
  720. INC(i)
  721. END
  722. END
  723. END PurgeOpenedFile;
  724. PROCEDURE Close(f: File);
  725. VAR adr: DiskAdr;
  726. BEGIN {EXCLUSIVE}
  727. IF f.key # 0 THEN
  728. ASSERT(openFiles.Contains(f.key));
  729. openFiles.Remove(f.key);
  730. dir.Search(f.name, adr);
  731. IF (adr = 0) OR (adr # f.key) THEN (* deleted or overwritten *)
  732. PurgeOpenedFile(f, NIL)
  733. ELSE
  734. CollectRegisteredFileSectors(adr);
  735. PurgeOpenedFile(f, tempRegFileSec);
  736. tempRegFileSec.Clear
  737. END
  738. ELSE
  739. PurgeOpenedFile(f, NIL)
  740. END
  741. END Close;
  742. PROCEDURE Finalize;
  743. BEGIN {EXCLUSIVE}
  744. dir.Cleanup();
  745. vol.Finalize;
  746. Finalize^ (* see note in Files *)
  747. END Finalize;
  748. END FileSystem;
  749. DiskAdrArray = POINTER TO ARRAY OF DiskAdr;
  750. (* analogous to TFClasses.List *)
  751. DiskAdrList = OBJECT
  752. VAR
  753. list : DiskAdrArray;
  754. count : LONGINT;
  755. PROCEDURE &New*;
  756. BEGIN NEW(list, 8); count := 0
  757. END New;
  758. PROCEDURE GetCount() : LONGINT;
  759. BEGIN {EXCLUSIVE}
  760. RETURN count
  761. END GetCount;
  762. PROCEDURE Grow;
  763. VAR old: DiskAdrArray; i : LONGINT;
  764. BEGIN
  765. old := list;
  766. NEW(list, LEN(list)*2);
  767. FOR i := 0 TO count-1 DO list[i] := old[i] END
  768. END Grow;
  769. PROCEDURE Add(x: DiskAdr);
  770. BEGIN {EXCLUSIVE}
  771. ASSERT(x # 0);
  772. IF count = LEN(list) THEN Grow END;
  773. list[count] := x;
  774. INC(count)
  775. END Add;
  776. PROCEDURE Remove(x: DiskAdr);
  777. VAR i : LONGINT;
  778. BEGIN {EXCLUSIVE}
  779. ASSERT(x # 0);
  780. i := 0; WHILE (i < count) & (list[i] # x) DO INC(i) END;
  781. IF i < count THEN
  782. WHILE (i < count-1) DO list[i] := list[i+1]; INC(i) END;
  783. DEC(count);
  784. list[count] := 0
  785. END
  786. END Remove;
  787. PROCEDURE Clear;
  788. VAR i : LONGINT;
  789. BEGIN {EXCLUSIVE}
  790. FOR i := 0 TO count - 1 DO list[i] := 0 END;
  791. count := 0
  792. END Clear;
  793. PROCEDURE Contains(x: DiskAdr) : BOOLEAN;
  794. VAR i: LONGINT;
  795. BEGIN {EXCLUSIVE}
  796. i := 0 ; WHILE i < count DO IF list[i] = x THEN RETURN TRUE END; INC(i) END;
  797. RETURN FALSE
  798. END Contains;
  799. END DiskAdrList;
  800. TYPE
  801. File = OBJECT (Files.File)
  802. VAR
  803. aleng, bleng: LONGINT;
  804. nofbufs: LONGINT;
  805. modH, registered: BOOLEAN;
  806. firstbuf: Buffer;
  807. sechint: DiskAdr;
  808. name: FileName;
  809. time, date: LONGINT;
  810. ext: SuperIndex;
  811. sec: SectorTable;
  812. PROCEDURE Set(VAR r: Files.Rider; pos: LONGINT);
  813. VAR a, b: LONGINT;
  814. BEGIN {EXCLUSIVE}
  815. r.eof := FALSE; r.res := 0; r.file := SELF; r.fs := fs;
  816. IF pos < 0 THEN
  817. a := 0; b := HeaderSize
  818. ELSIF pos < aleng*SectorSize + bleng - HeaderSize THEN
  819. a := (pos + HeaderSize) DIV SectorSize; b := (pos + HeaderSize) MOD SectorSize
  820. ELSE
  821. a := aleng; b := bleng
  822. END;
  823. r.apos := a; r.bpos := b; r.hint := firstbuf
  824. END Set;
  825. PROCEDURE Pos(VAR r: Files.Rider): LONGINT;
  826. BEGIN
  827. RETURN r.apos*SectorSize + r.bpos - HeaderSize
  828. END Pos;
  829. PROCEDURE Read(VAR r: Files.Rider; VAR x: CHAR);
  830. VAR buf: Buffer;
  831. BEGIN {EXCLUSIVE}
  832. buf := r.hint(Buffer);
  833. IF r.apos # buf.apos THEN buf := GetBuf(SELF, r.apos); r.hint := buf END;
  834. IF r.bpos < buf.lim THEN
  835. x := buf.data.B[r.bpos]; INC(r.bpos)
  836. ELSIF r.apos < aleng THEN
  837. INC(r.apos);
  838. buf := SearchBuf(SELF, r.apos);
  839. IF buf = NIL THEN
  840. buf := r.hint(Buffer);
  841. IF buf.mod THEN WriteBuf(SELF, buf) END ;
  842. ReadBuf(SELF, buf, r.apos)
  843. ELSE
  844. r.hint := buf
  845. END;
  846. ASSERT(buf.lim > 0);
  847. x := buf.data.B[0]; r.bpos := 1
  848. ELSE
  849. x := 0X; r.eof := TRUE
  850. END
  851. END Read;
  852. PROCEDURE ReadBytes(VAR r: Files.Rider; VAR x: ARRAY OF CHAR; ofs, len: LONGINT);
  853. VAR src: ADDRESS; m: LONGINT; buf: Buffer;
  854. BEGIN {EXCLUSIVE}
  855. IF LEN(x)-ofs < len THEN SYSTEM.HALT(19) END;
  856. IF len > 0 THEN
  857. buf := r.hint(Buffer);
  858. IF r.apos # buf.apos THEN buf := GetBuf(SELF, r.apos); r.hint := buf END;
  859. LOOP
  860. IF len <= 0 THEN EXIT END ;
  861. src := ADDRESSOF(buf.data.B[0]) + r.bpos; m := r.bpos + len;
  862. IF m <= buf.lim THEN
  863. SYSTEM.MOVE(src, ADDRESSOF(x[ofs]), len); r.bpos := m; r.res := 0; EXIT
  864. ELSIF buf.lim = SectorSize THEN
  865. m := buf.lim - r.bpos;
  866. IF m > 0 THEN SYSTEM.MOVE(src, ADDRESSOF(x[ofs]), m); INC(ofs, m); DEC(len, m) END ;
  867. IF r.apos < aleng THEN
  868. INC(r.apos); r.bpos := 0; buf := SearchBuf(SELF, r.apos);
  869. IF buf = NIL THEN
  870. buf := r.hint(Buffer);
  871. IF buf.mod THEN WriteBuf(SELF, buf) END ;
  872. ReadBuf(SELF, buf, r.apos)
  873. ELSE
  874. r.hint := buf
  875. END
  876. ELSE
  877. r.bpos := buf.lim; r.res := len; r.eof := TRUE; EXIT
  878. END
  879. ELSE
  880. m := buf.lim - r.bpos;
  881. IF m > 0 THEN SYSTEM.MOVE(src, ADDRESSOF(x[ofs]), m); r.bpos := buf.lim END ;
  882. r.res := len - m; r.eof := TRUE; EXIT
  883. END
  884. END;
  885. ELSE
  886. r.res := 0
  887. END
  888. END ReadBytes;
  889. PROCEDURE Write(VAR r: Files.Rider; x: CHAR);
  890. VAR buf: Buffer;
  891. BEGIN {EXCLUSIVE}
  892. buf := r.hint(Buffer);
  893. IF r.apos # buf.apos THEN buf := GetBuf(SELF, r.apos); r.hint := buf END;
  894. IF r.bpos >= buf.lim THEN
  895. IF r.bpos < SectorSize THEN
  896. INC(buf.lim); INC(bleng); modH := TRUE
  897. ELSE
  898. WriteBuf(SELF, buf); INC(r.apos); buf := SearchBuf(SELF, r.apos);
  899. IF buf = NIL THEN
  900. buf := r.hint(Buffer);
  901. IF r.apos <= aleng THEN
  902. ReadBuf(SELF, buf, r.apos)
  903. ELSE
  904. buf.apos := r.apos; buf.lim := 1; INC(aleng); bleng := 1; modH := TRUE;
  905. IF (aleng - SectorTableSize) MOD IndexSize = 0 THEN NewSub(SELF) END
  906. END
  907. ELSE
  908. r.hint := buf
  909. END;
  910. r.bpos := 0
  911. END
  912. END;
  913. buf.data.B[r.bpos] := x; INC(r.bpos); buf.mod := TRUE
  914. END Write;
  915. PROCEDURE WriteBytes(VAR r: Files.Rider; CONST x: ARRAY OF CHAR; ofs, len: LONGINT);
  916. VAR src, dst: ADDRESS; m: LONGINT; buf: Buffer;
  917. BEGIN {EXCLUSIVE}
  918. IF LEN(x)-ofs < len THEN SYSTEM.HALT(19) END;
  919. IF len > 0 THEN
  920. buf := r.hint(Buffer);
  921. IF r.apos # buf.apos THEN buf := GetBuf(SELF, r.apos); r.hint := buf END;
  922. LOOP
  923. IF len <= 0 THEN EXIT END;
  924. buf.mod := TRUE; dst := ADDRESSOF(buf.data.B[0]) + r.bpos; m := r.bpos + len;
  925. IF m <= buf.lim THEN
  926. SYSTEM.MOVE(ADDRESSOF(x[ofs]), dst, len); r.bpos := m; EXIT
  927. ELSIF m <= SectorSize THEN
  928. SYSTEM.MOVE(ADDRESSOF(x[ofs]), dst, len); r.bpos := m;
  929. bleng := m; buf.lim := m; modH := TRUE; EXIT
  930. ELSE
  931. m := SectorSize - r.bpos;
  932. IF m > 0 THEN SYSTEM.MOVE(ADDRESSOF(x[ofs]), dst, m); INC(ofs, m); DEC(len, m) END;
  933. WriteBuf(SELF, buf); INC(r.apos); r.bpos := 0; buf := SearchBuf(SELF, r.apos);
  934. IF buf = NIL THEN
  935. buf := r.hint(Buffer);
  936. IF r.apos <= aleng THEN ReadBuf(SELF, buf, r.apos)
  937. ELSE
  938. buf.apos := r.apos; buf.lim := 0; INC(aleng); bleng := 0; modH := TRUE;
  939. IF (aleng - SectorTableSize) MOD IndexSize = 0 THEN NewSub(SELF) END
  940. END
  941. ELSE
  942. r.hint := buf
  943. END
  944. END
  945. END
  946. END
  947. END WriteBytes;
  948. PROCEDURE Length(): LONGINT;
  949. BEGIN {EXCLUSIVE}
  950. RETURN aleng*SectorSize + bleng - HeaderSize
  951. END Length;
  952. PROCEDURE GetDate(VAR t, d: LONGINT);
  953. BEGIN {EXCLUSIVE}
  954. t := time; d := date
  955. END GetDate;
  956. PROCEDURE SetDate(t, d: LONGINT);
  957. BEGIN {EXCLUSIVE}
  958. modH := TRUE; time := t; date := d
  959. END SetDate;
  960. PROCEDURE GetName(VAR name: ARRAY OF CHAR);
  961. BEGIN {EXCLUSIVE}
  962. Files.JoinName(fs.prefix, SELF.name, name)
  963. END GetName;
  964. PROCEDURE Register0(VAR res: LONGINT);
  965. VAR oldAdr: DiskAdr; fs0: FileSystem;
  966. BEGIN {EXCLUSIVE}
  967. Unbuffer(SELF);
  968. IF ~registered & (name # "") THEN
  969. fs0 := fs(FileSystem);
  970. fs0.dir.Search(name, oldAdr);
  971. fs0.dir.Insert(name, sec[0]);
  972. registered := TRUE; key := sec[0];
  973. fs0.openFiles.Add(key);
  974. IF (oldAdr # 0) & ~fs0.openFiles.Contains(oldAdr) THEN (* overwrite not opened file *)
  975. ASSERT(oldAdr # key);
  976. fs0.PurgeByAdr(oldAdr)
  977. END;
  978. res := 0
  979. ELSE
  980. res := 1
  981. END
  982. END Register0;
  983. PROCEDURE Update;
  984. BEGIN {EXCLUSIVE}
  985. Unbuffer(SELF)
  986. END Update;
  987. END File;
  988. PROCEDURE Collect(f: ANY);
  989. VAR file: File; fs: FileSystem;
  990. BEGIN
  991. file := f(File);
  992. IF file.fs # NIL THEN
  993. fs := file.fs(FileSystem);
  994. IF (fs.vol # NIL) & ~(Files.ReadOnly IN fs.vol.flags) THEN fs.Close(file) END
  995. END
  996. END Collect;
  997. PROCEDURE GetSector(vol: Files.Volume; src: DiskAdr; VAR dest: DiskSector);
  998. BEGIN
  999. IF src MOD SectorFactor # 0 THEN SYSTEM.HALT(15) END;
  1000. vol.GetBlock(src DIV SectorFactor, SYSTEM.VAL(DiskSectorArr, dest))
  1001. END GetSector;
  1002. PROCEDURE PutSector(vol: Files.Volume; dest: DiskAdr; VAR src: DiskSector);
  1003. BEGIN
  1004. ASSERT(~(Files.ReadOnly IN vol.flags));
  1005. IF dest MOD SectorFactor # 0 THEN SYSTEM.HALT(15) END;
  1006. vol.PutBlock(dest DIV SectorFactor, SYSTEM.VAL(DiskSectorArr, src))
  1007. END PutSector;
  1008. PROCEDURE AllocSector(vol: Files.Volume; hint: DiskAdr; VAR sec: DiskAdr);
  1009. BEGIN
  1010. ASSERT(~(Files.ReadOnly IN vol.flags));
  1011. vol.AllocBlock(hint DIV SectorFactor, sec);
  1012. sec := sec * SectorFactor
  1013. END AllocSector;
  1014. PROCEDURE MarkSector(vol: Files.Volume; sec: LONGINT);
  1015. BEGIN
  1016. ASSERT(~(Files.ReadOnly IN vol.flags));
  1017. vol.MarkBlock(sec DIV SectorFactor)
  1018. END MarkSector;
  1019. PROCEDURE FreeSector(vol: Files.Volume; sec: LONGINT);
  1020. BEGIN
  1021. ASSERT(~(Files.ReadOnly IN vol.flags));
  1022. ASSERT(Marked(vol, sec));
  1023. vol.FreeBlock(sec DIV SectorFactor)
  1024. END FreeSector;
  1025. PROCEDURE Marked(vol: Files.Volume; sec: LONGINT): BOOLEAN;
  1026. BEGIN
  1027. ASSERT(~(Files.ReadOnly IN vol.flags));
  1028. RETURN vol.Marked(sec DIV SectorFactor)
  1029. END Marked;
  1030. PROCEDURE Match*(mask, name: ARRAY OF CHAR): BOOLEAN;
  1031. VAR m,n, om, on: LONGINT; f: BOOLEAN;
  1032. BEGIN
  1033. m := 0; n := 0; om := -1;
  1034. f := TRUE;
  1035. LOOP
  1036. IF (mask[m] = "*") THEN
  1037. om := m; INC(m);
  1038. WHILE (name[n] # 0X) & (name[n] # mask[m]) DO INC(n) END;
  1039. on := n
  1040. ELSIF (mask[m] = "?") THEN
  1041. IF (name[n] = 0X) THEN f := FALSE; EXIT END;
  1042. INC(m); INC(n)
  1043. ELSE
  1044. IF (mask[m] # name[n]) THEN
  1045. IF (om = -1) THEN f := FALSE; EXIT
  1046. ELSIF (name[n] # 0X) THEN (* try the next position *)
  1047. m := om; n := on + 1;
  1048. IF (name[n] = 0X) THEN f := FALSE; EXIT END
  1049. ELSE
  1050. f := FALSE; EXIT
  1051. END
  1052. ELSE INC(m); INC(n)
  1053. END
  1054. END;
  1055. IF (mask[m] = 0X) & ((name[n] = 0X) OR (om=-1)) THEN EXIT END
  1056. END;
  1057. RETURN f & (name[n] = 0X)
  1058. END Match;
  1059. PROCEDURE enumerate(fs: Files.FileSystem; VAR mask: ARRAY OF CHAR; dpg: DiskAdr; flags: SET; enum: Files.Enumerator; VAR continue: BOOLEAN; VAR fh: FileHeader; VAR fn: ARRAY OF CHAR);
  1060. VAR i, diff: LONGINT; dpg1: DiskAdr; a: DirPage; time, date, size: LONGINT;
  1061. BEGIN
  1062. GetSector(fs.vol, dpg, a); i := 0;
  1063. WHILE (i < a.m) & continue DO
  1064. (* MatchPrefix(mask, a.e[i].name, pos, diff); *)
  1065. IF i = 0 THEN dpg1 := a.p0 ELSE dpg1 := a.e[i-1].p END;
  1066. IF diff >= 0 THEN (* matching prefix *)
  1067. IF dpg1 # 0 THEN enumerate(fs, mask, dpg1, flags, enum, continue, fh, fn) END;
  1068. IF diff = 0 THEN
  1069. IF continue & ((mask = "") OR Match(mask, a.e[i].name)) THEN
  1070. time := 0; date := 0; size := 0;
  1071. IF flags * {Files.EnumTime, Files.EnumSize} # {} THEN
  1072. GetSector(fs.vol, a.e[i].adr, fh);
  1073. IF Files.EnumTime IN flags THEN
  1074. time := fh.time; date := fh.date
  1075. END;
  1076. IF Files.EnumSize IN flags THEN
  1077. size := fh.aleng*SectorSize + fh.bleng - HeaderSize
  1078. END
  1079. END;
  1080. Files.JoinName(fs.prefix, a.e[i].name, fn);
  1081. enum.PutEntry(fn, {}, time, date, size)
  1082. END
  1083. ELSE continue := FALSE
  1084. END
  1085. END;
  1086. INC(i)
  1087. END;
  1088. IF continue & (i > 0) & (a.e[i-1].p # 0) THEN
  1089. enumerate(fs, mask, a.e[i-1].p, flags, enum, continue, fh, fn)
  1090. END
  1091. END enumerate;
  1092. (* Check a file name. *)
  1093. PROCEDURE Check(VAR s: ARRAY OF CHAR; VAR name: FileName; VAR res: LONGINT);
  1094. VAR i, k: LONGINT; ch: CHAR;
  1095. BEGIN
  1096. ch := s[0]; i := 0; k := 0;
  1097. IF (ch = 0X) THEN name[0] := 0X; res := -1
  1098. ELSE
  1099. IF (ch = Files.PathDelimiter) THEN k := 1; ch := s[k] END; (* skip first path delimiter *)
  1100. LOOP
  1101. IF (ch < " ") OR (ch = ":") OR (ch = Files.PathDelimiter) THEN res := 3; EXIT END;
  1102. name[i] := ch; INC(i); INC(k); ch := s[k];
  1103. IF (ch = 0X) THEN
  1104. WHILE (i < FileNameLength) DO name[i] := 0X; INC(i) END;
  1105. res := 0; EXIT
  1106. END;
  1107. IF (i = FileNameLength-1) THEN res := 4; EXIT END
  1108. END
  1109. END
  1110. END Check;
  1111. PROCEDURE UpdateHeader(f: File; VAR h: FileHeader);
  1112. BEGIN
  1113. h.aleng := f.aleng; h.bleng := f.bleng;
  1114. h.sec := f.sec;
  1115. IF f.ext # NIL THEN h.ext := f.ext.adr ELSE h.ext := 0 END;
  1116. h.date := f.date; h.time := f.time
  1117. END UpdateHeader;
  1118. PROCEDURE ReadBuf(f: File; buf: Buffer; pos: LONGINT);
  1119. VAR sec: DiskAdr; xpos: LONGINT;
  1120. BEGIN
  1121. IF pos < SectorTableSize THEN
  1122. sec := f.sec[pos]
  1123. ELSE
  1124. xpos := pos-SectorTableSize;
  1125. sec := f.ext.sub[xpos DIV IndexSize].sec.x[xpos MOD IndexSize]
  1126. END;
  1127. GetSector(f.fs.vol, sec, buf.data);
  1128. IF pos < f.aleng THEN buf.lim := SectorSize ELSE buf.lim := f.bleng END;
  1129. buf.apos := pos; buf.mod := FALSE
  1130. END ReadBuf;
  1131. PROCEDURE NewSuper(f: File);
  1132. VAR i: LONGINT; super: SuperIndex;
  1133. BEGIN
  1134. NEW(super); super.adr := 0; super.mod := TRUE; f.modH := TRUE; f.ext := super;
  1135. FOR i := 0 TO IndexSize-1 DO super.sub[i] := NIL END
  1136. END NewSuper;
  1137. PROCEDURE WriteBuf(f: File; buf: Buffer);
  1138. VAR i, k, xpos: LONGINT; secadr: DiskAdr; super: SuperIndex; sub: SubIndex; vol: Files.Volume;
  1139. BEGIN
  1140. vol := f.fs.vol;
  1141. Clock.Get(f.time, f.date); f.modH := TRUE;
  1142. IF buf.apos < SectorTableSize THEN
  1143. secadr := f.sec[buf.apos];
  1144. IF secadr = 0 THEN
  1145. AllocSector(vol, f.sechint, secadr);
  1146. f.modH := TRUE; f.sec[buf.apos] := secadr; f.sechint := secadr
  1147. END;
  1148. IF buf.apos = 0 THEN
  1149. UpdateHeader(f, SYSTEM.VAL(FileHeader, buf.data)); f.modH := FALSE
  1150. END
  1151. ELSE
  1152. super := f.ext;
  1153. IF super = NIL THEN NewSuper(f); super := f.ext END;
  1154. xpos := buf.apos-SectorTableSize;
  1155. i := xpos DIV IndexSize; sub := super.sub[i];
  1156. IF sub = NIL THEN
  1157. NEW(sub); sub.adr := 0; sub.sec.x[0] := 0; super.sub[i] := sub; super.mod := TRUE
  1158. END;
  1159. k := xpos MOD IndexSize; secadr := sub.sec.x[k];
  1160. IF secadr = 0 THEN
  1161. AllocSector(vol, f.sechint, secadr); f.sechint := secadr;
  1162. sub.mod := TRUE; sub.sec.x[k] := secadr
  1163. END
  1164. END;
  1165. PutSector(vol, secadr, buf.data); buf.mod := FALSE
  1166. END WriteBuf;
  1167. PROCEDURE SearchBuf(f: File; pos: LONGINT): Buffer;
  1168. VAR buf: Buffer;
  1169. BEGIN
  1170. buf := f.firstbuf;
  1171. LOOP
  1172. IF buf.apos = pos THEN EXIT END;
  1173. buf := buf.next;
  1174. IF buf = f.firstbuf THEN buf := NIL; EXIT END
  1175. END;
  1176. RETURN buf
  1177. END SearchBuf;
  1178. PROCEDURE GetBuf(f: File; pos: LONGINT): Buffer;
  1179. VAR buf: Buffer;
  1180. BEGIN
  1181. buf := f.firstbuf;
  1182. LOOP
  1183. IF buf.apos = pos THEN EXIT END;
  1184. IF buf.next = f.firstbuf THEN
  1185. IF f.nofbufs < MaxBufs THEN (* allocate new buffer *)
  1186. NEW(buf); buf.next := f.firstbuf.next; f.firstbuf.next := buf;
  1187. INC(f.nofbufs)
  1188. ELSE (* take one of the buffers *)
  1189. f.firstbuf := buf;
  1190. IF buf.mod THEN WriteBuf(f, buf) END
  1191. END;
  1192. buf.apos := pos;
  1193. IF pos <= f.aleng THEN ReadBuf(f, buf, pos) END;
  1194. EXIT
  1195. END;
  1196. buf := buf.next
  1197. END;
  1198. RETURN buf
  1199. END GetBuf;
  1200. PROCEDURE Unbuffer(f: File);
  1201. VAR
  1202. i, k: LONGINT; buf: Buffer; super: SuperIndex; sub: SubIndex; head: FileHeader;
  1203. sec: IndexSector; vol: Files.Volume;
  1204. BEGIN
  1205. vol := f.fs.vol;
  1206. buf := f.firstbuf;
  1207. REPEAT
  1208. IF buf.mod THEN WriteBuf(f, buf) END;
  1209. buf := buf.next
  1210. UNTIL buf = f.firstbuf;
  1211. super := f.ext;
  1212. IF super # NIL THEN
  1213. k := (f.aleng + (IndexSize-SectorTableSize)) DIV IndexSize; i := 0;
  1214. WHILE i # k DO
  1215. sub := super.sub[i]; INC(i);
  1216. IF sub.mod THEN
  1217. IF sub.adr = 0 THEN
  1218. AllocSector(vol, f.sechint, sub.adr); f.sechint := sub.adr;
  1219. super.mod := TRUE
  1220. END;
  1221. PutSector(vol, sub.adr, sub.sec); sub.mod := FALSE
  1222. END
  1223. END;
  1224. IF super.mod THEN
  1225. IF super.adr = 0 THEN
  1226. AllocSector(vol, f.sechint, super.adr); f.sechint := super.adr;
  1227. f.modH := TRUE
  1228. END;
  1229. i := 0;
  1230. WHILE i # k DO sec.x[i] := super.sub[i].adr; INC(i) END;
  1231. WHILE i # IndexSize DO sec.x[i] := 0; INC(i) END;
  1232. PutSector(vol, super.adr, sec); super.mod := FALSE
  1233. END
  1234. END;
  1235. IF f.modH THEN
  1236. GetSector(vol, f.sec[0], head); UpdateHeader(f, head);
  1237. PutSector(vol, f.sec[0], head); f.modH := FALSE
  1238. END
  1239. END Unbuffer;
  1240. PROCEDURE NewSub(f: File);
  1241. VAR i, k: LONGINT; sub: SubIndex;
  1242. BEGIN
  1243. k := (f.aleng - SectorTableSize) DIV IndexSize;
  1244. IF k = IndexSize THEN SYSTEM.HALT(18) END;
  1245. NEW(sub); sub.adr := 0; sub.mod := TRUE;
  1246. FOR i := 0 TO IndexSize-1 DO sub.sec.x[i] := 0 END;
  1247. IF f.ext = NIL THEN NewSuper(f) END;
  1248. f.ext.sub[k] := sub
  1249. END NewSub;
  1250. (** Generate a new file system object. Files.NewVol has volume parameter, Files.Par has mount prefix. *)
  1251. PROCEDURE NewFS*(context : Files.Parameters);
  1252. VAR fs: FileSystem; fh: FileHeader; skipIndexMapWriteback: BOOLEAN; options: ARRAY 8 OF CHAR;
  1253. BEGIN
  1254. (* Get options *)
  1255. context.arg.SkipWhitespace;
  1256. REPEAT UNTIL ~context.arg.GetString(options) OR (options = '|');
  1257. IF context.arg.GetString(options) THEN
  1258. skipIndexMapWriteback := options = 'N'
  1259. END;
  1260. IF Files.This(context.prefix) = NIL THEN
  1261. IF (context.vol.blockSize = SectorSize) & (context.vol.size >= MinVolSize) THEN
  1262. GetSector(context.vol, DirRootAdr, fh);
  1263. IF fh.mark = DirMark THEN (* assume it is an Aos filesystem *)
  1264. NEW(fs); fs.vol := context.vol;
  1265. ASSERT(context.vol.size < MAX(LONGINT) DIV SectorFactor);
  1266. fs.desc := "AosFS";
  1267. NEW(fs.dir, context.vol); (* initialize directory and volume *)
  1268. ASSERT(fs.dir.state = Opened); (* will have to undo changes to vol before continuing *)
  1269. Files.Add(fs, context.prefix);
  1270. IF skipIndexMapWriteback THEN
  1271. INCL(fs.flags, SkipIndexFlag);
  1272. fs.dir.noCleanup := TRUE
  1273. END
  1274. ELSE
  1275. context.error.String("DiskFS: File system not found on ");
  1276. context.error.String(context.vol.name); context.error.Ln
  1277. END
  1278. ELSE
  1279. context.error.String("DiskFS: Bad volume size"); context.error.Ln
  1280. END
  1281. ELSE
  1282. context.error.String("DiskFS: "); context.error.String(context.prefix);
  1283. context.error.String(" already in use"); context.error.Ln
  1284. END;
  1285. END NewFS;
  1286. (* Clean up when module unloaded. *)
  1287. PROCEDURE Cleanup;
  1288. VAR ft: Files.FileSystemTable; i: LONGINT;
  1289. BEGIN
  1290. IF Modules.shutdown = Modules.None THEN
  1291. Files.GetList(ft);
  1292. IF ft # NIL THEN
  1293. FOR i := 0 TO LEN(ft^)-1 DO
  1294. IF ft[i] IS FileSystem THEN Files.Remove(ft[i]) END
  1295. END
  1296. END
  1297. END
  1298. END Cleanup;
  1299. BEGIN
  1300. ASSERT((SIZEOF(FileHeader) = SectorSize) & (SIZEOF(IndexSector) = SectorSize) & (SIZEOF(DataSector) = SectorSize) &
  1301. (SIZEOF(DirPage) = SectorSize) & (SIZEOF(MapIndex) = SectorSize) & (SIZEOF(MapSector) = SectorSize) &
  1302. (DirPgSize MOD 2 = 0));
  1303. Modules.InstallTermHandler(Cleanup);
  1304. END DiskFS.
  1305. (*
  1306. aleng * SectorSize + bleng = length (including header)
  1307. apos * SectorSize + bpos = current position
  1308. 0 <= bpos <= lim <= SectorSize
  1309. 0 <= apos <= aleng < SectorTableSize + IndexSize*IndexSize
  1310. (apos < aleng) & (lim = SectorSize) OR (apos = aleng)
  1311. Methods with {} notation are explicitly unprotected. They must be called only from a protected context.
  1312. *)
  1313. (*
  1314. 04.02.2004 lb Prevent disk space leaks during system run (disk GC)
  1315. 03.01.2006 staubesv Avoid longint overflow that caused disk gc even if not necessary
  1316. *)