From: Jeff Hobbs Date: Sat, 19 Oct 2002 02:44:43 +0000 (+0000) Subject: whitespace cleanup X-Git-Tag: vfs-1-2~27 X-Git-Url: https://privyetmir.co.uk/gitweb?a=commitdiff_plain;h=5fc320f52f0873dcaf3b63d313a6f5d4fc8e5a51;p=tclvfs whitespace cleanup --- diff --git a/library/mk4vfs.tcl b/library/mk4vfs.tcl index 1763534..199ff73 100644 --- a/library/mk4vfs.tcl +++ b/library/mk4vfs.tcl @@ -16,479 +16,486 @@ package require vfs # need this so init failure in interactive mode does not mess up errorInfo if {[info exists env(VFS_DEBUG)] && [info commands history] == ""} { - proc history {args} {} + proc history {args} {} } # things that can no longer really be left out (but this is the wrong spot!) # be as non-invasive as possible, using these definitions as last resort - if {![info exists auto_index(lassign)] && [info commands lassign] == ""} { +if {![info exists auto_index(lassign)] && [info commands lassign] == ""} { set auto_index(lassign) { - proc lassign {l args} { - foreach v $l a $args { uplevel 1 [list set $a $v] } - } + proc lassign {l args} { + foreach v $l a $args { uplevel 1 [list set $a $v] } + } } - } +} namespace eval vfs::mk4 { - - proc Mount {mkfile local args} { - set db [eval [list ::mk4vfs::_mount $local $mkfile] $args] - ::vfs::filesystem mount $local [list ::vfs::mk4::handler $db] - ::vfs::RegisterMount $local [list ::vfs::mk4::Unmount $db] - return $db - } - - proc Unmount {db local} { - vfs::filesystem unmount $local - ::mk4vfs::_umount $db - } - - proc handler {db cmd root relative actualpath args} { - #puts stderr "handler: $db - $cmd - $root - $relative - $actualpath - $args" - if {$cmd == "matchindirectory"} { - eval [list $cmd $db $relative $actualpath] $args - } elseif {$cmd == "fileattributes"} { - eval [list $cmd $db $root $relative] $args - } else { - eval [list $cmd $db $relative] $args + proc Mount {mkfile local args} { + set db [eval [list ::mk4vfs::_mount $local $mkfile] $args] + ::vfs::filesystem mount $local [list ::vfs::mk4::handler $db] + ::vfs::RegisterMount $local [list ::vfs::mk4::Unmount $db] + return $db } - } - proc utime {db path actime modtime} { - ::mk4vfs::stat $db $path sb - - if { $sb(type) == "file" } { - ::mk::set $sb(ino) date $modtime - } - } - - proc matchindirectory {db path actualpath pattern type} { - set newres [list] - if {![string length $pattern]} { - # check single file - set res [list $actualpath] - set actualpath "" - } else { - set res [::mk4vfs::getdir $db $path $pattern] + proc Unmount {db local} { + vfs::filesystem unmount $local + ::mk4vfs::_umount $db } - foreach p [::vfs::matchCorrectTypes $type $res $actualpath] { - lappend newres "$actualpath$p" - } - return $newres - } - - proc stat {db name} { - ::mk4vfs::stat $db $name sb - - set sb(ino) 0 - array get sb - } - - proc access {db name mode} { - # This needs implementing better. - ::mk4vfs::stat $db $name sb - } - - proc open {db file mode permissions} { - # return a list of two elements: - # 1. first element is the Tcl channel name which has been opened - # 2. second element (optional) is a command to evaluate when - # the channel is closed. - switch -glob -- $mode { - {} - - r { - ::mk4vfs::stat $db $file sb - - if { $sb(csize) != $sb(size) } { - set fd [vfs::memchan] - fconfigure $fd -translation binary - set s [mk::get $sb(ino) contents] - puts -nonewline $fd [vfs::zip -mode decompress $s] - - fconfigure $fd -translation auto - seek $fd 0 - return $fd - } elseif { $::mk4vfs::direct } { - set fd [vfs::memchan] - fconfigure $fd -translation binary - puts -nonewline $fd [mk::get $sb(ino) contents] - - fconfigure $fd -translation auto - seek $fd 0 - return $fd + + proc handler {db cmd root relative actualpath args} { + #puts stderr "handler: $db - $cmd - $root - $relative - $actualpath - $args" + if {$cmd == "matchindirectory"} { + eval [list $cmd $db $relative $actualpath] $args + } elseif {$cmd == "fileattributes"} { + eval [list $cmd $db $root $relative] $args } else { - set fd [mk::channel $sb(ino) contents r] - } - return [list $fd] - } - a { - if { [catch {::mk4vfs::stat $db $file sb }] } { - # Create file - ::mk4vfs::stat $db [file dirname $file] sb - set tail [file tail $file] - set fview $sb(ino).files - if {[info exists mk4vfs::v::fcache($fview)]} { - lappend mk4vfs::v::fcache($fview) $tail - } - set now [clock seconds] - set sb(ino) [mk::row append $fview name $tail size 0 date $now ] - - if { [string match *z* $mode] || $mk4vfs::compress } { - set sb(csize) -1 ;# HACK - force compression - } else { - set sb(csize) 0 - } + eval [list $cmd $db $relative] $args } + } - set fd [vfs::memchan] - fconfigure $fd -translation binary - set s [mk::get $sb(ino) contents] + proc utime {db path actime modtime} { + ::mk4vfs::stat $db $path sb + + if { $sb(type) == "file" } { + ::mk::set $sb(ino) date $modtime + } + } - if { $sb(csize) != $sb(size) && $sb(csize) > 0 } { - append mode z - puts -nonewline $fd [vfs::zip -mode decompress $s] + proc matchindirectory {db path actualpath pattern type} { + set newres [list] + if {![string length $pattern]} { + # check single file + set res [list $actualpath] + set actualpath "" } else { - if { $mk4vfs::compress } { append mode z } - puts -nonewline $fd $s - #set fd [mk::channel $sb(ino) contents a] + set res [::mk4vfs::getdir $db $path $pattern] } - fconfigure $fd -translation auto - seek $fd 0 end - return [list $fd [list mk4vfs::do_close $fd $mode $sb(ino)]] - } - w* { - if { [catch {::mk4vfs::stat $db $file sb }] } { - # Create file - ::mk4vfs::stat $db [file dirname $file] sb - set tail [file tail $file] - set fview $sb(ino).files - if {[info exists mk4vfs::v::fcache($fview)]} { - lappend mk4vfs::v::fcache($fview) $tail - } - set now [clock seconds] - set sb(ino) [mk::row append $fview name $tail size 0 date $now ] + foreach p [::vfs::matchCorrectTypes $type $res $actualpath] { + lappend newres "$actualpath$p" } + return $newres + } - if { [string match *z* $mode] || $mk4vfs::compress } { - append mode z - set fd [vfs::memchan] - } else { - set fd [mk::channel $sb(ino) contents w] + proc stat {db name} { + ::mk4vfs::stat $db $name sb + + set sb(ino) 0 + array get sb + } + + proc access {db name mode} { + # This needs implementing better. + ::mk4vfs::stat $db $name sb + } + + proc open {db file mode permissions} { + # return a list of two elements: + # 1. first element is the Tcl channel name which has been opened + # 2. second element (optional) is a command to evaluate when + # the channel is closed. + switch -glob -- $mode { + {} - + r { + ::mk4vfs::stat $db $file sb + + if { $sb(csize) != $sb(size) } { + set fd [vfs::memchan] + fconfigure $fd -translation binary + set s [mk::get $sb(ino) contents] + puts -nonewline $fd [vfs::zip -mode decompress $s] + + fconfigure $fd -translation auto + seek $fd 0 + return $fd + } elseif { $::mk4vfs::direct } { + set fd [vfs::memchan] + fconfigure $fd -translation binary + puts -nonewline $fd [mk::get $sb(ino) contents] + + fconfigure $fd -translation auto + seek $fd 0 + return $fd + } else { + set fd [mk::channel $sb(ino) contents r] + } + return [list $fd] + } + a { + if { [catch {::mk4vfs::stat $db $file sb }] } { + # Create file + ::mk4vfs::stat $db [file dirname $file] sb + set tail [file tail $file] + set fview $sb(ino).files + if {[info exists mk4vfs::v::fcache($fview)]} { + lappend mk4vfs::v::fcache($fview) $tail + } + set now [clock seconds] + set sb(ino) [mk::row append $fview \ + name $tail size 0 date $now ] + + if { [string match *z* $mode] || $mk4vfs::compress } { + set sb(csize) -1 ;# HACK - force compression + } else { + set sb(csize) 0 + } + } + + set fd [vfs::memchan] + fconfigure $fd -translation binary + set s [mk::get $sb(ino) contents] + + if { $sb(csize) != $sb(size) && $sb(csize) > 0 } { + append mode z + puts -nonewline $fd [vfs::zip -mode decompress $s] + } else { + if { $mk4vfs::compress } { append mode z } + puts -nonewline $fd $s + #set fd [mk::channel $sb(ino) contents a] + } + fconfigure $fd -translation auto + seek $fd 0 end + return [list $fd [list mk4vfs::do_close $fd $mode $sb(ino)]] + } + w* { + if { [catch {::mk4vfs::stat $db $file sb }] } { + # Create file + ::mk4vfs::stat $db [file dirname $file] sb + set tail [file tail $file] + set fview $sb(ino).files + if {[info exists mk4vfs::v::fcache($fview)]} { + lappend mk4vfs::v::fcache($fview) $tail + } + set now [clock seconds] + set sb(ino) [mk::row append $fview \ + name $tail size 0 date $now ] + } + + if { [string match *z* $mode] || $mk4vfs::compress } { + append mode z + set fd [vfs::memchan] + } else { + set fd [mk::channel $sb(ino) contents w] + } + return [list $fd [list mk4vfs::do_close $fd $mode $sb(ino)]] + } + default { + error "illegal access mode \"$mode\"" + } } - return [list $fd [list mk4vfs::do_close $fd $mode $sb(ino)]] - } - default { - error "illegal access mode \"$mode\"" - } } - } - - proc createdirectory {db name} { - mk4vfs::mkdir $db $name - } - - proc removedirectory {db name recursive} { - mk4vfs::delete $db $name $recursive - } - - proc deletefile {db name} { - mk4vfs::delete $db $name - } - - proc fileattributes {db root relative args} { - switch -- [llength $args] { - 0 { - # list strings - return [::vfs::listAttributes] - } - 1 { - # get value - set index [lindex $args 0] - return [::vfs::attributesGet $root $relative $index] - - } - 2 { - # set value - set index [lindex $args 0] - set val [lindex $args 1] - return [::vfs::attributesSet $root $relative $index $val] - } + + proc createdirectory {db name} { + mk4vfs::mkdir $db $name + } + + proc removedirectory {db name recursive} { + mk4vfs::delete $db $name $recursive + } + + proc deletefile {db name} { + mk4vfs::delete $db $name + } + + proc fileattributes {db root relative args} { + switch -- [llength $args] { + 0 { + # list strings + return [::vfs::listAttributes] + } + 1 { + # get value + set index [lindex $args 0] + return [::vfs::attributesGet $root $relative $index] + + } + 2 { + # set value + set index [lindex $args 0] + set val [lindex $args 1] + return [::vfs::attributesSet $root $relative $index $val] + } + } } - } } namespace eval mk4vfs { - variable compress 1 ;# HACK - needs to be part of "Super-Block" - variable flush 5000 ;# Auto-Commit frequency - variable direct 0 ;# read through a memchan, or from Mk4tcl if zero + variable compress 1 ;# HACK - needs to be part of "Super-Block" + variable flush 5000 ;# Auto-Commit frequency + variable direct 0 ;# read through a memchan, or from Mk4tcl if zero - namespace eval v { - variable seq 0 + namespace eval v { + variable seq 0 - array set cache {} - array set fcache {} - } + array set cache {} + array set fcache {} + } - namespace export mount umount + namespace export mount umount - proc init {db} { - mk::view layout $db.dirs {name:S parent:I {files {name:S size:I date:I contents:M}}} + proc init {db} { + mk::view layout $db.dirs \ + {name:S parent:I {files {name:S size:I date:I contents:M}}} - if { [mk::view size $db.dirs] == 0 } { - mk::row append $db.dirs name parent 0 - } + if { [mk::view size $db.dirs] == 0 } { + mk::row append $db.dirs name parent 0 + } - # 2001-12-13: use parent -1 for root level! - mk::set $db.dirs!0 parent -1 - } + # 2001-12-13: use parent -1 for root level! + mk::set $db.dirs!0 parent -1 + } - proc mount {local mkfile args} { - uplevel [list ::vfs::mk4::Mount $mkfile $local] $args - } + proc mount {local mkfile args} { + uplevel [list ::vfs::mk4::Mount $mkfile $local] $args + } - proc _mount {path file args} { - set db mk4vfs[incr v::seq] + proc _mount {path file args} { + set db mk4vfs[incr v::seq] - eval [list mk::file open $db $file] $args + eval [list mk::file open $db $file] $args - init $db + init $db - set flush 1 - for {set idx 0} {$idx < [llength $args]} {incr idx} { - switch -- [lindex $args $idx] { - -readonly - - -nocommit {set flush 0} - } - } - if { $flush } { - _commit $db + set flush 1 + for {set idx 0} {$idx < [llength $args]} {incr idx} { + switch -- [lindex $args $idx] { + -readonly - + -nocommit {set flush 0} + } + } + if { $flush } { + _commit $db + } + return $db } - return $db - } - - proc _commit {db} { - variable flush - after $flush [list mk4vfs::_commit $db] - mk::file commit $db - } - - proc umount {local} { - foreach {db path} [mk::file open] { - if {[string equal $local $path]} { - uplevel ::vfs::mk4::Unmount $db $local - return - } + + proc _commit {db} { + variable flush + after $flush [list mk4vfs::_commit $db] + mk::file commit $db } - tclLog "umount $local? [mk::file open]" - } - - proc _umount {db} { - after cancel [list mk4vfs::_commit $db] - array unset v::cache $db,* - array unset v::fcache $db.* - mk::file close $db - } - - proc stat {db path arr} { - upvar 1 $arr sb - - set sp [::file split $path] - set tail [lindex $sp end] - - set parent 0 - set view $db.dirs - set type directory - - foreach ele [lrange $sp 0 end-1] { - if {[info exists v::cache($db,$parent,$ele)]} { - set parent $v::cache($db,$parent,$ele) - } else { - set row [mk::select $view -count 1 parent $parent name $ele] - if { $row == "" } { - return -code error "could not read \"$path\": no such file or directory" + + proc umount {local} { + foreach {db path} [mk::file open] { + if {[string equal $local $path]} { + uplevel ::vfs::mk4::Unmount $db $local + return + } } - set v::cache($db,$parent,$ele) $row - set parent $row - } + tclLog "umount $local? [mk::file open]" } - - # Now check if final comp is a directory or a file - # CACHING is required - it can deliver a x15 speed-up! - - if { [string equal $tail "."] || [string equal $tail ":"] || - [string equal $tail ""] } { - set row $parent - - } elseif { [info exists v::cache($db,$parent,$tail)] } { - set row $v::cache($db,$parent,$tail) - } else { - # File? - set fview $view!$parent.files - # create a name cache of files in this directory - if {![info exists v::fcache($fview)]} { - # cache only a limited number of directories - if {[array size v::fcache] >= 10} { - array unset v::fcache * + + proc _umount {db} { + after cancel [list mk4vfs::_commit $db] + array unset v::cache $db,* + array unset v::fcache $db.* + mk::file close $db + } + + proc stat {db path arr} { + upvar 1 $arr sb + + set sp [::file split $path] + set tail [lindex $sp end] + + set parent 0 + set view $db.dirs + set type directory + + foreach ele [lrange $sp 0 end-1] { + if {[info exists v::cache($db,$parent,$ele)]} { + set parent $v::cache($db,$parent,$ele) + } else { + set row [mk::select $view -count 1 parent $parent name $ele] + if { $row == "" } { + return -code error "could not read \"$path\":\ + no such file or directory" + } + set v::cache($db,$parent,$ele) $row + set parent $row + } } - set v::fcache($fview) {} - mk::loop c $fview { - lappend v::fcache($fview) [mk::get $c name] + + # Now check if final comp is a directory or a file + # CACHING is required - it can deliver a x15 speed-up! + + if { [string equal $tail "."] || [string equal $tail ":"] || + [string equal $tail ""] } { + set row $parent + + } elseif { [info exists v::cache($db,$parent,$tail)] } { + set row $v::cache($db,$parent,$tail) + } else { + # File? + set fview $view!$parent.files + # create a name cache of files in this directory + if {![info exists v::fcache($fview)]} { + # cache only a limited number of directories + if {[array size v::fcache] >= 10} { + array unset v::fcache * + } + set v::fcache($fview) {} + mk::loop c $fview { + lappend v::fcache($fview) [mk::get $c name] + } + } + set row [lsearch -exact $v::fcache($fview) $tail] + #set row [mk::select $fview -count 1 name $tail] + #if {$row == ""} { set row -1 } + if { $row != -1 } { + set type file + set view $view!$parent.files + } else { + # Directory? + set row [mk::select $view -count 1 parent $parent name $tail] + if { $row != "" } { + set v::cache($db,$parent,$tail) $row + } else { + return -code error "could not read \"$path\":\ + no such file or directory" + } + } } - } - set row [lsearch -exact $v::fcache($fview) $tail] - #set row [mk::select $fview -count 1 name $tail] - #if {$row == ""} { set row -1 } - if { $row != -1 } { - set type file - set view $view!$parent.files - } else { - # Directory? - set row [mk::select $view -count 1 parent $parent name $tail] - if { $row != "" } { - set v::cache($db,$parent,$tail) $row - } else { - return -code error "could not read \"$path\": no such file or directory" + set cur $view!$row + + set sb(type) $type + set sb(view) $view + set sb(ino) $cur + + if { [string equal $type "directory"] } { + set sb(atime) 0 + set sb(ctime) 0 + set sb(gid) 0 + set sb(mode) 0777 + set sb(mtime) 0 + set sb(nlink) [expr { [mk::get $cur files] + 1 }] + set sb(size) 0 + set sb(csize) 0 + set sb(uid) 0 + } else { + set mtime [mk::get $cur date] + set sb(atime) $mtime + set sb(ctime) $mtime + set sb(gid) 0 + set sb(mode) 0777 + set sb(mtime) $mtime + set sb(nlink) 1 + set sb(size) [mk::get $cur size] + set sb(csize) [mk::get $cur -size contents] + set sb(uid) 0 } - } - } - set cur $view!$row - - set sb(type) $type - set sb(view) $view - set sb(ino) $cur - - if { [string equal $type "directory"] } { - set sb(atime) 0 - set sb(ctime) 0 - set sb(gid) 0 - set sb(mode) 0777 - set sb(mtime) 0 - set sb(nlink) [expr { [mk::get $cur files] + 1 }] - set sb(size) 0 - set sb(csize) 0 - set sb(uid) 0 - } else { - set mtime [mk::get $cur date] - set sb(atime) $mtime - set sb(ctime) $mtime - set sb(gid) 0 - set sb(mode) 0777 - set sb(mtime) $mtime - set sb(nlink) 1 - set sb(size) [mk::get $cur size] - set sb(csize) [mk::get $cur -size contents] - set sb(uid) 0 } - } - proc do_close {fd mode cur} { - if {![regexp {[aw]} $mode]} { - error "mk4vfs::do_close called with bad mode: $mode" - } + proc do_close {fd mode cur} { + if {![regexp {[aw]} $mode]} { + error "mk4vfs::do_close called with bad mode: $mode" + } - mk::set $cur size -1 date [clock seconds] - flush $fd - if { [string match *z* $mode] } { - fconfigure $fd -translation binary - seek $fd 0 - set data [read $fd] - set cdata [vfs::zip -mode compress $data] - set len [string length $data] - set clen [string length $cdata] - if { $clen < $len } { - mk::set $cur size $len contents $cdata - } else { - mk::set $cur size $len contents $data - } - } else { - mk::set $cur size [mk::get $cur -size contents] - } - # added 30-10-2000 - set db [lindex [split $cur .] 0] - mk::file autocommit $db - } - - proc mkdir {db path} { - set sp [::file split $path] - set parent 0 - set view $db.dirs - - set npath {} - foreach ele $sp { - set npath [file join $npath $ele] - - if { ![catch {stat $db $npath sb}] } { - if { $sb(type) != "directory" } { - return -code error "can't create directory \"$npath\": file already exists" + mk::set $cur size -1 date [clock seconds] + flush $fd + if { [string match *z* $mode] } { + fconfigure $fd -translation binary + seek $fd 0 + set data [read $fd] + set cdata [vfs::zip -mode compress $data] + set len [string length $data] + set clen [string length $cdata] + if { $clen < $len } { + mk::set $cur size $len contents $cdata + } else { + mk::set $cur size $len contents $data + } + } else { + mk::set $cur size [mk::get $cur -size contents] } - set parent [mk::cursor position sb(ino)] - continue - } - #set parent [mk::cursor position sb(ino)] - set cur [mk::row append $view name $ele parent $parent] - set parent [mk::cursor position cur] + # added 30-10-2000 + set db [lindex [split $cur .] 0] + mk::file autocommit $db } - } - proc getdir {db path {pat *}} { - if {[catch { stat $db $path sb }] || $sb(type) != "directory" } { - return + proc mkdir {db path} { + set sp [::file split $path] + set parent 0 + set view $db.dirs + + set npath {} + foreach ele $sp { + set npath [file join $npath $ele] + + if { ![catch {stat $db $npath sb}] } { + if { $sb(type) != "directory" } { + return -code error "can't create directory \"$npath\":\ + file already exists" + } + set parent [mk::cursor position sb(ino)] + continue + } + #set parent [mk::cursor position sb(ino)] + set cur [mk::row append $view name $ele parent $parent] + set parent [mk::cursor position cur] + } } - # Match directories - set parent [mk::cursor position sb(ino)] - foreach row [mk::select $sb(view) parent $parent -glob name $pat] { - set hits([mk::get $sb(view)!$row name]) 1 - } - # Match files - set view $sb(view)!$parent.files - foreach row [mk::select $view -glob name $pat] { - set hits([mk::get $view!$row name]) 1 - } - return [lsort [array names hits]] - } + proc getdir {db path {pat *}} { + if {[catch { stat $db $path sb }] || $sb(type) != "directory" } { + return + } - proc mtime {db path time} { - stat $db $path sb - if { $sb(type) == "file" } { - mk::set $sb(ino) date $time + # Match directories + set parent [mk::cursor position sb(ino)] + foreach row [mk::select $sb(view) parent $parent -glob name $pat] { + set hits([mk::get $sb(view)!$row name]) 1 + } + # Match files + set view $sb(view)!$parent.files + foreach row [mk::select $view -glob name $pat] { + set hits([mk::get $view!$row name]) 1 + } + return [lsort [array names hits]] } - return $time - } - - proc delete {db path {recursive 0}} { - #puts stderr "mk4delete db $db path $path recursive $recursive" - stat $db $path sb - if {$sb(type) == "file" } { - mk::row delete $sb(ino) - if {[regexp {(.*)!(\d+)} $sb(ino) - v r] && [info exists v::fcache($v)]} { - set v::fcache($v) [lreplace $v::fcache($v) $r $r] - } - } else { - # just mark dirs as deleted - set contents [getdir $db $path *] - if {$recursive} { - # We have to delete these manually, else - # they (or their cache) may conflict with - # something later - foreach f $contents { - delete $db [file join $path $f] $recursive + + proc mtime {db path time} { + stat $db $path sb + if { $sb(type) == "file" } { + mk::set $sb(ino) date $time } - } else { - if {[llength $contents]} { - return -code error "Non-empty" + return $time + } + + proc delete {db path {recursive 0}} { + #puts stderr "mk4delete db $db path $path recursive $recursive" + stat $db $path sb + if {$sb(type) == "file" } { + mk::row delete $sb(ino) + if {[regexp {(.*)!(\d+)} $sb(ino) - v r] \ + && [info exists v::fcache($v)]} { + set v::fcache($v) [lreplace $v::fcache($v) $r $r] + } + } else { + # just mark dirs as deleted + set contents [getdir $db $path *] + if {$recursive} { + # We have to delete these manually, else + # they (or their cache) may conflict with + # something later + foreach f $contents { + delete $db [file join $path $f] $recursive + } + } else { + if {[llength $contents]} { + return -code error "Non-empty" + } + } + array unset v::cache \ + "$db,[mk::get $sb(ino) parent],[file tail $path]" + + # flag with -99, because parent -1 is not reserved for the root dir + # deleted entries never get re-used, should be cleaned up one day + mk::set $sb(ino) parent -99 name "" + # get rid of file entries to release the space in the datafile + mk::view size $sb(ino).files 0 } - } - array unset v::cache "$db,[mk::get $sb(ino) parent],[file tail $path]" - - # flag with -99, because parent -1 is not reserved for the root dir - # deleted entries never get re-used, should be cleaned up one day - mk::set $sb(ino) parent -99 name "" - # get rid of file entries to release the space in the datafile - mk::view size $sb(ino).files 0 + return "" } - return "" - } } diff --git a/library/starkit.tcl b/library/starkit.tcl index cfd0763..3d9c370 100644 --- a/library/starkit.tcl +++ b/library/starkit.tcl @@ -18,78 +18,78 @@ package provide starkit 1.0 # lassign is used so widely by now, make sure it is always available if {![info exists auto_index(lassign)] && [info commands lassign] eq ""} { - set auto_index(lassign) { - proc lassign {l args} { - foreach v $l a $args { uplevel 1 [list set $a $v] } + set auto_index(lassign) { + proc lassign {l args} { + foreach v $l a $args { uplevel 1 [list set $a $v] } + } } - } } namespace eval starkit { + # called from the header of a starkit + proc header {driver args} { + if {[catch { + set self [info script] -# called from the header of a starkit - proc header {driver args} { - if {[catch { - set self [info script] + package require ${driver}vfs + eval [list ::vfs::${driver}::Mount $self $self] $args - package require ${driver}vfs - eval [list ::vfs::${driver}::Mount $self $self] $args - - uplevel [list source [file join $self main.tcl]] - }]} { - panic $::errorInfo + uplevel [list source [file join $self main.tcl]] + }]} { + panic $::errorInfo + } } - } -# called from the startup script of a starkit to init topdir and auto_path -# returns how the script was launched: starkit, starpack, unwrapped, or sourced - proc startup {} { - global argv0 - variable topdir ;# the root directory (while the starkit is mounted) + # called from the startup script of a starkit to init topdir and auto_path + # returns how the script was launched: starkit, starpack, unwrapped, or + # sourced + proc startup {} { + global argv0 + variable topdir ;# the root directory (while the starkit is mounted) - set script [file normalize [info script]] - set topdir [file dirname $script] + set script [file normalize [info script]] + set topdir [file dirname $script] - if {$topdir eq [info nameofexe]} { return starpack } + if {$topdir eq [info nameofexe]} { return starpack } - # pkgs live in the $topdir/lib/ directory - set lib [file join $topdir lib] - if {[file isdir $lib]} { autoextend $lib } + # pkgs live in the $topdir/lib/ directory + set lib [file join $topdir lib] + if {[file isdir $lib]} { autoextend $lib } - set a0 [file normalize $argv0] - if {$topdir eq $a0} { return starkit } - if {$script eq $a0} { return unwrapped } - return sourced - } + set a0 [file normalize $argv0] + if {$topdir eq $a0} { return starkit } + if {$script eq $a0} { return unwrapped } + return sourced + } -# append an entry to auto_path if it's not yet listed - proc autoextend {dir} { - global auto_path - set dir [file normalize $dir] - if {[lsearch $auto_path $dir] < 0} { - lappend auto_path $dir + # append an entry to auto_path if it's not yet listed + proc autoextend {dir} { + global auto_path + set dir [file normalize $dir] + if {[lsearch $auto_path $dir] < 0} { + lappend auto_path $dir + } } - } -# remount a starkit with different options - proc remount {args} { - variable topdir - lassign [vfs::filesystem info $topdir] drv arg - vfs::unmount $topdir - - eval [list [regsub handler $drv Mount] $topdir $topdir] $args - } + # remount a starkit with different options + proc remount {args} { + variable topdir + lassign [vfs::filesystem info $topdir] drv arg + vfs::unmount $topdir + + eval [list [regsub handler $drv Mount] $topdir $topdir] $args + } -# terminate with an error message, using most appropriate mechanism - proc panic {msg} { - if {[info commands wm] ne ""} { - wm withdraw . - tk_messageBox -icon error -message $msg -title "Fatal error" - } elseif {[info commands eventlog] ne ""} { - eventlog error $msg - } else { - puts stderr $msg + # terminate with an error message, using most appropriate mechanism + proc panic {msg} { + if {[info commands wm] ne ""} { + wm withdraw . + tk_messageBox -icon error -message $msg -title "Fatal error" + } elseif {[info commands eventlog] ne ""} { + eventlog error $msg + } else { + puts stderr $msg + } + exit } - exit - } } diff --git a/library/vfsUtils.tcl b/library/vfsUtils.tcl index aa56557..63032b6 100644 --- a/library/vfsUtils.tcl +++ b/library/vfsUtils.tcl @@ -306,84 +306,84 @@ proc vfs::posixError {name} { return $posix($name) } -set vfs::posix(EPERM) 1 ;# Operation not permitted -set vfs::posix(ENOENT) 2 ;# No such file or directory -set vfs::posix(ESRCH) 3 ;# No such process -set vfs::posix(EINTR) 4 ;# Interrupted system call -set vfs::posix(EIO) 5 ;# Input/output error -set vfs::posix(ENXIO) 6 ;# Device not configured -set vfs::posix(E)2BIG 7 ;# Argument list too long -set vfs::posix(ENOEXEC) 8 ;# Exec format error -set vfs::posix(EBADF) 9 ;# Bad file descriptor -set vfs::posix(ECHILD) 10 ;# No child processes -set vfs::posix(EDEADLK) 11 ;# Resource deadlock avoided +set vfs::posix(EPERM) 1 ;# Operation not permitted +set vfs::posix(ENOENT) 2 ;# No such file or directory +set vfs::posix(ESRCH) 3 ;# No such process +set vfs::posix(EINTR) 4 ;# Interrupted system call +set vfs::posix(EIO) 5 ;# Input/output error +set vfs::posix(ENXIO) 6 ;# Device not configured +set vfs::posix(E)2BIG 7 ;# Argument list too long +set vfs::posix(ENOEXEC) 8 ;# Exec format error +set vfs::posix(EBADF) 9 ;# Bad file descriptor +set vfs::posix(ECHILD) 10 ;# No child processes +set vfs::posix(EDEADLK) 11 ;# Resource deadlock avoided ;# 11 was EAGAIN -set vfs::posix(ENOMEM) 12 ;# Cannot allocate memory -set vfs::posix(EACCES) 13 ;# Permission denied -set vfs::posix(EFAULT) 14 ;# Bad address -set vfs::posix(ENOTBLK) 15 ;# Block device required -set vfs::posix(EBUSY) 16 ;# Device busy -set vfs::posix(EEXIST) 17 ;# File exists -set vfs::posix(EXDEV) 18 ;# Cross-device link -set vfs::posix(ENODEV) 19 ;# Operation not supported by device -set vfs::posix(ENOTDIR) 20 ;# Not a directory -set vfs::posix(EISDIR) 21 ;# Is a directory -set vfs::posix(EINVAL) 22 ;# Invalid argument -set vfs::posix(ENFILE) 23 ;# Too many open files in system -set vfs::posix(EMFILE) 24 ;# Too many open files -set vfs::posix(ENOTTY) 25 ;# Inappropriate ioctl for device -set vfs::posix(ETXTBSY) 26 ;# Text file busy -set vfs::posix(EFBIG) 27 ;# File too large -set vfs::posix(ENOSPC) 28 ;# No space left on device -set vfs::posix(ESPIPE) 29 ;# Illegal seek -set vfs::posix(EROFS) 30 ;# Read-only file system -set vfs::posix(EMLINK) 31 ;# Too many links -set vfs::posix(EPIPE) 32 ;# Broken pipe -set vfs::posix(EDOM) 33 ;# Numerical argument out of domain -set vfs::posix(ERANGE) 34 ;# Result too large -set vfs::posix(EAGAIN) 35 ;# Resource temporarily unavailable -set vfs::posix(EWOULDBLOCK) 35 ;# Operation would block -set vfs::posix(EINPROGRESS) 36 ;# Operation now in progress -set vfs::posix(EALREADY) 37 ;# Operation already in progress -set vfs::posix(ENOTSOCK) 38 ;# Socket operation on non-socket -set vfs::posix(EDESTADDRREQ) 39 ;# Destination address required -set vfs::posix(EMSGSIZE) 40 ;# Message too long -set vfs::posix(EPROTOTYPE) 41 ;# Protocol wrong type for socket -set vfs::posix(ENOPROTOOPT) 42 ;# Protocol not available -set vfs::posix(EPROTONOSUPPORT) 43 ;# Protocol not supported -set vfs::posix(ESOCKTNOSUPPORT) 44 ;# Socket type not supported -set vfs::posix(EOPNOTSUPP) 45 ;# Operation not supported on socket -set vfs::posix(EPFNOSUPPORT) 46 ;# Protocol family not supported -set vfs::posix(EAFNOSUPPORT) 47 ;# Address family not supported by protocol family -set vfs::posix(EADDRINUSE) 48 ;# Address already in use -set vfs::posix(EADDRNOTAVAIL) 49 ;# Can't assign requested address -set vfs::posix(ENETDOWN) 50 ;# Network is down -set vfs::posix(ENETUNREACH) 51 ;# Network is unreachable -set vfs::posix(ENETRESET) 52 ;# Network dropped connection on reset -set vfs::posix(ECONNABORTED) 53 ;# Software caused connection abort -set vfs::posix(ECONNRESET) 54 ;# Connection reset by peer -set vfs::posix(ENOBUFS) 55 ;# No buffer space available -set vfs::posix(EISCONN) 56 ;# Socket is already connected -set vfs::posix(ENOTCONN) 57 ;# Socket is not connected -set vfs::posix(ESHUTDOWN) 58 ;# Can't send after socket shutdown -set vfs::posix(ETOOMANYREFS) 59 ;# Too many references: can't splice -set vfs::posix(ETIMEDOUT) 60 ;# Connection timed out -set vfs::posix(ECONNREFUSED) 61 ;# Connection refused -set vfs::posix(ELOOP) 62 ;# Too many levels of symbolic links -set vfs::posix(ENAMETOOLONG) 63 ;# File name too long -set vfs::posix(EHOSTDOWN) 64 ;# Host is down -set vfs::posix(EHOSTUNREACH) 65 ;# No route to host -set vfs::posix(ENOTEMPTY) 66 ;# Directory not empty -set vfs::posix(EPROCLIM) 67 ;# Too many processes -set vfs::posix(EUSERS) 68 ;# Too many users -set vfs::posix(EDQUOT) 69 ;# Disc quota exceeded -set vfs::posix(ESTALE) 70 ;# Stale NFS file handle -set vfs::posix(EREMOTE) 71 ;# Too many levels of remote in path -set vfs::posix(EBADRPC) 72 ;# RPC struct is bad -set vfs::posix(ERPCMISMATCH) 73 ;# RPC version wrong -set vfs::posix(EPROGUNAVAIL) 74 ;# RPC prog. not avail -set vfs::posix(EPROGMISMATCH) 75 ;# Program version wrong -set vfs::posix(EPROCUNAVAIL) 76 ;# Bad procedure for program -set vfs::posix(ENOLCK) 77 ;# No locks available -set vfs::posix(ENOSYS) 78 ;# Function not implemented -set vfs::posix(EFTYPE) 79 ;# Inappropriate file type or format +set vfs::posix(ENOMEM) 12 ;# Cannot allocate memory +set vfs::posix(EACCES) 13 ;# Permission denied +set vfs::posix(EFAULT) 14 ;# Bad address +set vfs::posix(ENOTBLK) 15 ;# Block device required +set vfs::posix(EBUSY) 16 ;# Device busy +set vfs::posix(EEXIST) 17 ;# File exists +set vfs::posix(EXDEV) 18 ;# Cross-device link +set vfs::posix(ENODEV) 19 ;# Operation not supported by device +set vfs::posix(ENOTDIR) 20 ;# Not a directory +set vfs::posix(EISDIR) 21 ;# Is a directory +set vfs::posix(EINVAL) 22 ;# Invalid argument +set vfs::posix(ENFILE) 23 ;# Too many open files in system +set vfs::posix(EMFILE) 24 ;# Too many open files +set vfs::posix(ENOTTY) 25 ;# Inappropriate ioctl for device +set vfs::posix(ETXTBSY) 26 ;# Text file busy +set vfs::posix(EFBIG) 27 ;# File too large +set vfs::posix(ENOSPC) 28 ;# No space left on device +set vfs::posix(ESPIPE) 29 ;# Illegal seek +set vfs::posix(EROFS) 30 ;# Read-only file system +set vfs::posix(EMLINK) 31 ;# Too many links +set vfs::posix(EPIPE) 32 ;# Broken pipe +set vfs::posix(EDOM) 33 ;# Numerical argument out of domain +set vfs::posix(ERANGE) 34 ;# Result too large +set vfs::posix(EAGAIN) 35 ;# Resource temporarily unavailable +set vfs::posix(EWOULDBLOCK) 35 ;# Operation would block +set vfs::posix(EINPROGRESS) 36 ;# Operation now in progress +set vfs::posix(EALREADY) 37 ;# Operation already in progress +set vfs::posix(ENOTSOCK) 38 ;# Socket operation on non-socket +set vfs::posix(EDESTADDRREQ) 39 ;# Destination address required +set vfs::posix(EMSGSIZE) 40 ;# Message too long +set vfs::posix(EPROTOTYPE) 41 ;# Protocol wrong type for socket +set vfs::posix(ENOPROTOOPT) 42 ;# Protocol not available +set vfs::posix(EPROTONOSUPPORT) 43 ;# Protocol not supported +set vfs::posix(ESOCKTNOSUPPORT) 44 ;# Socket type not supported +set vfs::posix(EOPNOTSUPP) 45 ;# Operation not supported on socket +set vfs::posix(EPFNOSUPPORT) 46 ;# Protocol family not supported +set vfs::posix(EAFNOSUPPORT) 47 ;# Address family not supported by protocol family +set vfs::posix(EADDRINUSE) 48 ;# Address already in use +set vfs::posix(EADDRNOTAVAIL) 49 ;# Can't assign requested address +set vfs::posix(ENETDOWN) 50 ;# Network is down +set vfs::posix(ENETUNREACH) 51 ;# Network is unreachable +set vfs::posix(ENETRESET) 52 ;# Network dropped connection on reset +set vfs::posix(ECONNABORTED) 53 ;# Software caused connection abort +set vfs::posix(ECONNRESET) 54 ;# Connection reset by peer +set vfs::posix(ENOBUFS) 55 ;# No buffer space available +set vfs::posix(EISCONN) 56 ;# Socket is already connected +set vfs::posix(ENOTCONN) 57 ;# Socket is not connected +set vfs::posix(ESHUTDOWN) 58 ;# Can't send after socket shutdown +set vfs::posix(ETOOMANYREFS) 59 ;# Too many references: can't splice +set vfs::posix(ETIMEDOUT) 60 ;# Connection timed out +set vfs::posix(ECONNREFUSED) 61 ;# Connection refused +set vfs::posix(ELOOP) 62 ;# Too many levels of symbolic links +set vfs::posix(ENAMETOOLONG) 63 ;# File name too long +set vfs::posix(EHOSTDOWN) 64 ;# Host is down +set vfs::posix(EHOSTUNREACH) 65 ;# No route to host +set vfs::posix(ENOTEMPTY) 66 ;# Directory not empty +set vfs::posix(EPROCLIM) 67 ;# Too many processes +set vfs::posix(EUSERS) 68 ;# Too many users +set vfs::posix(EDQUOT) 69 ;# Disc quota exceeded +set vfs::posix(ESTALE) 70 ;# Stale NFS file handle +set vfs::posix(EREMOTE) 71 ;# Too many levels of remote in path +set vfs::posix(EBADRPC) 72 ;# RPC struct is bad +set vfs::posix(ERPCMISMATCH) 73 ;# RPC version wrong +set vfs::posix(EPROGUNAVAIL) 74 ;# RPC prog. not avail +set vfs::posix(EPROGMISMATCH) 75 ;# Program version wrong +set vfs::posix(EPROCUNAVAIL) 76 ;# Bad procedure for program +set vfs::posix(ENOLCK) 77 ;# No locks available +set vfs::posix(ENOSYS) 78 ;# Function not implemented +set vfs::posix(EFTYPE) 79 ;# Inappropriate file type or format diff --git a/library/vfslib.tcl b/library/vfslib.tcl index f6d440f..b1290b1 100644 --- a/library/vfslib.tcl +++ b/library/vfslib.tcl @@ -1,82 +1,83 @@ # Remnants of what used to be VFS init, this is TclKit-specific +package require Tcl 8.4; # vfs is all new for 8.4 package provide vfslib 1.3 namespace eval ::vfs { -# for backwards compatibility - proc normalize {path} { ::file normalize $path } + # for backwards compatibility + proc normalize {path} { ::file normalize $path } -# use zlib to define zip and crc if available - if {[info command zlib] != "" || ![catch {load "" zlib}]} { + # use zlib to define zip and crc if available + if {[info command zlib] != "" || ![catch {load "" zlib}]} { - proc zip {flag value args} { - switch -glob -- "$flag $value" { - {-mode d*} { set mode decompress } - {-mode c*} { set mode compress } - default { error "usage: zip -mode {compress|decompress} data" } - } - # kludge to allow "-nowrap 1" as second option, 5-9-2002 - if {[llength $args] > 2 && [lrange $args 0 1] == "-nowrap 1"} { - if {$mode == "compress"} { - set mode deflate - } else { - set mode inflate + proc zip {flag value args} { + switch -glob -- "$flag $value" { + {-mode d*} { set mode decompress } + {-mode c*} { set mode compress } + default { error "usage: zip -mode {compress|decompress} data" } + } + # kludge to allow "-nowrap 1" as second option, 5-9-2002 + if {[llength $args] > 2 && [lrange $args 0 1] == "-nowrap 1"} { + if {$mode == "compress"} { + set mode deflate + } else { + set mode inflate + } + } + return [zlib $mode [lindex $args end]] } - } - return [zlib $mode [lindex $args end]] - } - proc crc {data} { - return [zlib crc32 $data] + proc crc {data} { + return [zlib crc32 $data] + } } - } -# use rechan to define memchan if available - if {[info command rechan] != "" || ![catch {load "" rechan}]} { + # use rechan to define memchan if available + if {[info command rechan] != "" || ![catch {load "" rechan}]} { - proc memchan_handler {cmd fd args} { - upvar ::vfs::_memchan_buf($fd) buf - upvar ::vfs::_memchan_pos($fd) pos - set arg1 [lindex $args 0] - - switch -- $cmd { - seek { - switch [lindex $args 1] { - 1 - current { incr arg1 $pos } - 2 - end { incr arg1 [string length $buf]} - } - return [set pos $arg1] - } - read { - set r [string range $buf $pos [expr { $pos + $arg1 - 1 }]] - incr pos [string length $r] - return $r - } - write { - set n [string length $arg1] - if { $pos >= [string length $buf] } { - append buf $arg1 - } else { # the following doesn't work yet :( - set last [expr { $pos + $n - 1 }] - set buf [string replace $buf $pos $last $arg1] - error "vfs memchan: sorry no inline write yet" - } - incr pos $n - return $n + proc memchan_handler {cmd fd args} { + upvar ::vfs::_memchan_buf($fd) buf + upvar ::vfs::_memchan_pos($fd) pos + set arg1 [lindex $args 0] + + switch -- $cmd { + seek { + switch [lindex $args 1] { + 1 - current { incr arg1 $pos } + 2 - end { incr arg1 [string length $buf]} + } + return [set pos $arg1] + } + read { + set r [string range $buf $pos [expr { $pos + $arg1 - 1 }]] + incr pos [string length $r] + return $r + } + write { + set n [string length $arg1] + if { $pos >= [string length $buf] } { + append buf $arg1 + } else { # the following doesn't work yet :( + set last [expr { $pos + $n - 1 }] + set buf [string replace $buf $pos $last $arg1] + error "vfs memchan: sorry no inline write yet" + } + incr pos $n + return $n + } + close { + unset buf pos + } + default { error "bad cmd in memchan_handler: $cmd" } + } } - close { - unset buf pos + + proc memchan {} { + set fd [rechan ::vfs::memchan_handler 6] + set ::vfs::_memchan_buf($fd) "" + set ::vfs::_memchan_pos($fd) 0 + return $fd } - default { error "bad cmd in memchan_handler: $cmd" } - } - } - - proc memchan {} { - set fd [rechan ::vfs::memchan_handler 6] - set ::vfs::_memchan_buf($fd) "" - set ::vfs::_memchan_pos($fd) 0 - return $fd } - } } diff --git a/library/webdavvfs.tcl b/library/webdavvfs.tcl index 3b41fc0..0106def 100644 --- a/library/webdavvfs.tcl +++ b/library/webdavvfs.tcl @@ -24,7 +24,7 @@ proc vfs::webdav::Mount {dirurl local} { } if {![regexp {(([^:]*)(:([^@]*))?@)?([^/]*)(/(.*/)?([^/]*))?$} $rest \ - junk junk user junk pass host junk path file]} { + junk junk user junk pass host junk path file]} { return -code error "Sorry I didn't understand\ the url address \"$dirurl\"" } @@ -41,7 +41,8 @@ proc vfs::webdav::Mount {dirurl local} { set dirurl "http://$host/$path" - set extraHeadersList [list Authorization [list Basic [base64::encode ${user}:${pass}]]] + set extraHeadersList [list Authorization \ + [list Basic [base64::encode ${user}:${pass}]]] set token [::http::geturl $dirurl -headers $extraHeadersList -validate 1] http::cleanup $token @@ -52,7 +53,8 @@ proc vfs::webdav::Mount {dirurl local} { vfs::unmount $dirurl } ::vfs::log "http $host, $path mounted at $local" - vfs::filesystem mount $local [list vfs::webdav::handler $dirurl $extraHeadersList $path] + vfs::filesystem mount $local [list vfs::webdav::handler \ + $dirurl $extraHeadersList $path] # Register command to unmount vfs::RegisterMount $local [list ::vfs::webdav::Unmount $dirurl] return $dirurl