#+private package os2 import "base:runtime" import "core:time" import win32 "core:sys/windows" @(private="file") find_data_to_file_info :: proc(base_path: string, d: ^win32.WIN32_FIND_DATAW, allocator: runtime.Allocator) -> (fi: File_Info, err: Error) { // Ignore "." and ".." if d.cFileName[0] == '.' && d.cFileName[1] == 0 { return } if d.cFileName[0] == '.' && d.cFileName[1] == '.' && d.cFileName[2] == 0 { return } temp_allocator := TEMP_ALLOCATOR_GUARD({ allocator }) path := concatenate({base_path, `\`, win32_wstring_to_utf8(cstring16(raw_data(d.cFileName[:])), temp_allocator) or_else ""}, allocator) or_return handle := win32.HANDLE(_open_internal(path, {.Read}, 0o666) or_else 0) defer win32.CloseHandle(handle) fi.fullpath = path fi.name = basename(path) fi.size = i64(d.nFileSizeHigh)<<32 + i64(d.nFileSizeLow) fi.type, fi.mode = _file_type_mode_from_file_attributes(d.dwFileAttributes, handle, d.dwReserved0) fi.creation_time = time.unix(0, win32.FILETIME_as_unix_nanoseconds(d.ftCreationTime)) fi.modification_time = time.unix(0, win32.FILETIME_as_unix_nanoseconds(d.ftLastWriteTime)) fi.access_time = time.unix(0, win32.FILETIME_as_unix_nanoseconds(d.ftLastAccessTime)) if file_id_info: win32.FILE_ID_INFO; handle != nil && win32.GetFileInformationByHandleEx(handle, .FileIdInfo, &file_id_info, size_of(file_id_info)) { #assert(size_of(fi.inode) == size_of(file_id_info.FileId)) #assert(size_of(fi.inode) == 16) runtime.mem_copy_non_overlapping(&fi.inode, &file_id_info.FileId, 16) } return } Read_Directory_Iterator_Impl :: struct { find_data: win32.WIN32_FIND_DATAW, find_handle: win32.HANDLE, path: string, prev_fi: File_Info, no_more_files: bool, } @(require_results) _read_directory_iterator :: proc(it: ^Read_Directory_Iterator) -> (fi: File_Info, index: int, ok: bool) { for !it.impl.no_more_files { err: Error file_info_delete(it.impl.prev_fi, file_allocator()) it.impl.prev_fi = {} fi, err = find_data_to_file_info(it.impl.path, &it.impl.find_data, file_allocator()) if err != nil { read_directory_iterator_set_error(it, it.impl.path, err) return } if fi.name != "" { it.impl.prev_fi = fi ok = true index = it.index it.index += 1 } if !win32.FindNextFileW(it.impl.find_handle, &it.impl.find_data) { e := _get_platform_error() if pe, _ := is_platform_error(e); pe != i32(win32.ERROR_NO_MORE_FILES) { read_directory_iterator_set_error(it, it.impl.path, e) } it.impl.no_more_files = true } if ok { return } } return } _read_directory_iterator_init :: proc(it: ^Read_Directory_Iterator, f: ^File) { it.impl.no_more_files = false if f == nil || f.impl == nil { read_directory_iterator_set_error(it, "", .Invalid_File) return } it.f = f impl := (^File_Impl)(f.impl) // NOTE: Allow calling `init` to target a new directory with the same iterator - reset idx. if it.impl.find_handle != nil { win32.FindClose(it.impl.find_handle) } if it.impl.path != "" { delete(it.impl.path, file_allocator()) } if !is_directory(impl.name) { read_directory_iterator_set_error(it, impl.name, .Invalid_Dir) return } wpath := string16(impl.wname) temp_allocator := TEMP_ALLOCATOR_GUARD({}) wpath_search := make([]u16, len(wpath)+3, temp_allocator) copy(wpath_search, wpath) wpath_search[len(wpath)+0] = '\\' wpath_search[len(wpath)+1] = '*' wpath_search[len(wpath)+2] = 0 it.impl.find_handle = win32.FindFirstFileW(cstring16(raw_data(wpath_search)), &it.impl.find_data) if it.impl.find_handle == win32.INVALID_HANDLE_VALUE { read_directory_iterator_set_error(it, impl.name, _get_platform_error()) return } defer if it.err.err != nil { win32.FindClose(it.impl.find_handle) } err: Error it.impl.path, err = _cleanpath_from_buf(wpath, file_allocator()) if err != nil { read_directory_iterator_set_error(it, impl.name, err) } return } _read_directory_iterator_destroy :: proc(it: ^Read_Directory_Iterator) { if it.f == nil { return } file_info_delete(it.impl.prev_fi, file_allocator()) delete(it.impl.path, file_allocator()) win32.FindClose(it.impl.find_handle) }