Browse Source

os2: implement the iterator based read directory

Laytan Laats 1 year ago
parent
commit
00eb702c4a
1 changed files with 57 additions and 50 deletions
  1. 57 50
      core/os/os2/dir_posix.odin

+ 57 - 50
core/os/os2/dir_posix.odin

@@ -2,70 +2,77 @@
 //+build darwin, netbsd, freebsd, openbsd
 //+build darwin, netbsd, freebsd, openbsd
 package os2
 package os2
 
 
-import "base:runtime"
-
 import "core:sys/posix"
 import "core:sys/posix"
 
 
-@(private)
-_read_directory :: proc(f: ^File, n: int, allocator: runtime.Allocator) -> (files: []File_Info, err: Error) {
-	if f == nil || f.impl == nil {
-		err = .Invalid_File
+Read_Directory_Iterator_Impl :: struct {
+	dir:      posix.DIR,
+	idx:      int,
+
+	// NOTE: could there be paths bigger than this, maybe, probably, but why does it exist then?
+	fullpath: [posix.PATH_MAX]byte,
+}
+
+@(require_results)
+_read_directory_iterator :: proc(it: ^Read_Directory_Iterator) -> (fi: File_Info, index: int, ok: bool) {
+	index = it.impl.idx
+	it.impl.idx += 1
+
+	entry := posix.readdir(it.impl.dir)
+	if entry == nil {
+		// NOTE(laytan): would be good to have an `error` field on the `Read_Directory_Iterator`
+		// There isn't a way to now know if it failed or if we are at the end.
 		return
 		return
 	}
 	}
 
 
-	n := n
-	if n == 0 {
+	cname := cstring(raw_data(entry.d_name[:]))
+
+	// NOTE: these shouldn't be given back, but how?
+	// if cname == "." || cname == ".." {
+	//     continue
+	// }
+
+	stat: posix.stat_t
+	if posix.fstatat(posix.dirfd(it.impl.dir), cname, &stat, { .SYMLINK_NOFOLLOW }) != .OK {
+		// NOTE(laytan): would be good to have an `error` field on the `Read_Directory_Iterator`
+		// There isn't a way to now know if it failed or if we are at the end.
 		return
 		return
 	}
 	}
 
 
-	impl := (^File_Impl)(f)
+	fimpl := (^File_Impl)(it.f.impl)
 
 
-	dir := posix.fdopendir(impl.fd)
-	if dir == nil {
-		err = _get_platform_error()
+	n := copy(it.impl.fullpath[:],  fimpl.name)
+	n += copy(it.impl.fullpath[n:], "/")
+	n += copy(it.impl.fullpath[n:], string(cname))
+
+	fi = internal_stat(stat, string(it.impl.fullpath[:n]))
+	ok = true
+	return
+}
+
+@(require_results)
+_read_directory_iterator_create :: proc(f: ^File) -> (iter: Read_Directory_Iterator, err: Error) {
+	if f == nil || f.impl == nil {
+		err = .Invalid_File
 		return
 		return
 	}
 	}
-	defer posix.closedir(dir)
+	iter.f = f
+	iter.impl.idx = 0
 
 
-	dfiles: [dynamic]File_Info
-	dfiles.allocator = allocator
-	defer if err != nil {
-		file_info_slice_delete(dfiles[:], allocator)
-	}
+	impl := (^File_Impl)(f.impl)
 
 
-	for {
-		posix.set_errno(.NONE)
-		entry := posix.readdir(dir)
-		if entry == nil {
-			if errno := posix.errno(); errno != .NONE {
-				err = _get_platform_error()
-				return
-			} else {
-				break
-			}
-		}
-
-		cname := cstring(raw_data(entry.d_name[:]))
-		if cname == "." || cname == ".." {
-			continue
-		}
-
-		stat: posix.stat_t
-		if posix.fstatat(posix.dirfd(dir), cname, &stat, { .SYMLINK_NOFOLLOW }) != .OK {
-			err = _get_platform_error()
-			return
-		}
-
-		fullpath := concatenate({impl.name, "/", string(cname)}, allocator) or_return
-		fi := internal_stat(stat, fullpath)
-		append(&dfiles, fi) or_return
-
-		n -= 1
-		if n == 0 {
-			break
-		}
+	iter.impl.dir = posix.fdopendir(impl.fd)
+	if iter.impl.dir == nil {
+		err = _get_platform_error()
+		return
 	}
 	}
 
 
-	files = dfiles[:]
 	return
 	return
 }
 }
+
+_read_directory_iterator_destroy :: proc(it: ^Read_Directory_Iterator) {
+	if it == nil || it.impl.dir == nil {
+		return
+	}
+
+	posix.closedir(it.impl.dir)
+}