[SLOF] [PATCH] Add a global "dir" method
Benjamin Herrenschmidt
benh at kernel.crashing.org
Thu Jun 9 13:46:46 AEST 2016
This adds a method akin to "boot" and "load" which takes the subsequent
command line arguments, parses them as a device specification and
calls the dir method on said device
Signed-off-by: Benjamin Herrenschmidt <benh at kernel.crashing.org>
---
slof/fs/boot.fs | 15 ++++++++++++
slof/fs/packages/fat-files.fs | 56 ++++++++++++++++++++++++++++++++++---------
2 files changed, 60 insertions(+), 11 deletions(-)
diff --git a/slof/fs/boot.fs b/slof/fs/boot.fs
index e99a164..e436380 100644
--- a/slof/fs/boot.fs
+++ b/slof/fs/boot.fs
@@ -174,6 +174,21 @@ defer go ( -- )
\ Generic device load method:
\ *
+: do-dir ( devstr len -- )
+ cr ." Directory of: " 2dup type ." ... "
+ open-dev dup IF
+ s" dir" 2 pick ['] $call-method CATCH IF
+ ." no dir method on target !" cr
+ 3drop
+ THEN
+ close-dev cr
+ ELSE drop THEN
+;
+
+: dir ( "{devstring}" -- )
+ parse-word de-alias do-dir
+;
+
: do-load ( devstr len -- img-size ) \ Device method wrapper
use-load-watchdog? IF
\ Set watchdog timer to 10 minutes, multiply with 2 because DHCP
diff --git a/slof/fs/packages/fat-files.fs b/slof/fs/packages/fat-files.fs
index 5d578f1..ac2f141 100644
--- a/slof/fs/packages/fat-files.fs
+++ b/slof/fs/packages/fat-files.fs
@@ -33,6 +33,8 @@ INSTANCE VARIABLE root-offset
INSTANCE VARIABLE cluster-offset
INSTANCE VARIABLE #clusters
+INSTANCE VARIABLE dir?
+
: seek s" seek" $call-parent ;
: read s" read" $call-parent ;
@@ -54,7 +56,7 @@ CREATE fat-buf 8 allot
fat-buf 8 read 8 <> ABORT" fat-files read-fat: read failed"
fat-buf 8c@ bxjoin fat-type @ dup >r 2* #split drop r> #split
rot IF swap THEN drop ;
-
+
INSTANCE VARIABLE next-cluster
: read-cluster ( cluster# -- )
@@ -130,14 +132,35 @@ CREATE dos-name b allot
: find-file ( dir-cluster name len -- cluster file-len is-dir? true | false )
make-dos-name read-dir BEGIN (find-file) 0= WHILE next-cluster @ WHILE
next-cluster @ read-cluster REPEAT false ELSE true THEN ;
-: find-path ( dir-cluster name len -- cluster file-len true | false )
- dup 0= IF 3drop false ." empty name " EXIT THEN
- over c@ [char] \ = IF 1 /string RECURSE EXIT THEN
- [char] \ split 2>r find-file 0= IF 2r> 2drop false ." not found " EXIT THEN
- r@ 0<> <> IF 2drop 2r> 2drop false ." no dir<->file match " EXIT THEN
- r@ 0<> IF drop 2r> RECURSE EXIT THEN
- 2r> 2drop true ;
-
+
+: find-path ( dir-cluster name len -- cluster file-len is-dir? true | false )
+ dup 0= IF
+ \ empty name, assume directory
+ 2drop 0 true true EXIT
+ THEN
+ \ Strip leading backslashes
+ over c@ [char] \ = IF
+ 1 /string RECURSE EXIT
+ THEN
+ \ Split at backslash
+ [char] \ split
+ \ Store right side on return stack
+ 2>r
+ find-file
+ 0= IF
+ 2r> 2drop false ." not found " EXIT
+ THEN
+ \ right side (from stack) has non-0 len, must be a dir
+ dup 0= r@ 0<> and IF
+ 3drop 2r> 2drop false ." path component not a dir " EXIT
+ THEN
+ r@ 0<> IF
+ 2drop 2r> RECURSE EXIT
+ THEN
+ 2r> 2drop
+ true
+;
+
: do-super ( -- )
0 200 read-data
data @ 0b + 2c@ bwjoin bytes/sector !
@@ -204,7 +227,18 @@ INSTANCE VARIABLE pos-in-data
file-len @ read dup file-len @ <> ABORT" fat-files: failed loading file" ;
: close free-data ;
+
+: dir
+ dir? @ IF file-cluster @ .dir ELSE ." not a directory!" cr THEN
+ ;
+
: open
do-super
- 0 my-args find-path 0= IF close false EXIT THEN
- file-len ! file-cluster ! 0 0 seek 0= ;
+ 0 my-args find-path
+ 0= IF free-data false EXIT
+ THEN
+ dir? ! file-len ! file-cluster !
+ dir? @ IF
+ 0 0 seek 0=
+ ELSE true THEN
+;
More information about the SLOF
mailing list