mirror of
https://github.com/Ed94/Odin.git
synced 2026-06-18 20:02:22 -07:00
168 lines
4.2 KiB
Odin
168 lines
4.2 KiB
Odin
package os
|
|
|
|
import "base:intrinsics"
|
|
import "base:runtime"
|
|
import "core:io"
|
|
|
|
Platform_Error :: _Platform_Error
|
|
#assert(size_of(Platform_Error) <= 4)
|
|
#assert(intrinsics.type_has_nil(Platform_Error))
|
|
|
|
General_Error :: enum u32 {
|
|
None,
|
|
|
|
Permission_Denied,
|
|
Exist,
|
|
Not_Exist,
|
|
Closed,
|
|
|
|
Timeout,
|
|
|
|
Broken_Pipe,
|
|
|
|
// Indicates that an attempt to retrieve a file's size was made, but the
|
|
// file doesn't have a size.
|
|
No_Size,
|
|
|
|
Invalid_File,
|
|
Invalid_Dir,
|
|
Invalid_Path,
|
|
Invalid_Callback,
|
|
|
|
Pattern_Has_Separator,
|
|
|
|
Unsupported,
|
|
}
|
|
|
|
|
|
Errno :: Error // alias for legacy use
|
|
|
|
Error :: union #shared_nil {
|
|
General_Error,
|
|
io.Error,
|
|
runtime.Allocator_Error,
|
|
Platform_Error,
|
|
}
|
|
#assert(size_of(Error) == 8)
|
|
|
|
ERROR_NONE :: Error{}
|
|
|
|
@(require_results)
|
|
is_platform_error :: proc "contextless" (ferr: Error) -> (err: i32, ok: bool) {
|
|
v := ferr.(Platform_Error) or_else {}
|
|
return i32(v), i32(v) != 0
|
|
}
|
|
|
|
@(require_results)
|
|
error_string :: proc "contextless" (ferr: Error) -> string {
|
|
if ferr == nil {
|
|
return ""
|
|
}
|
|
switch e in ferr {
|
|
case General_Error:
|
|
switch e {
|
|
case .None: return ""
|
|
case .Permission_Denied: return "permission denied"
|
|
case .Exist: return "file already exists"
|
|
case .Not_Exist: return "file does not exist"
|
|
case .Closed: return "file already closed"
|
|
case .Timeout: return "i/o timeout"
|
|
case .Broken_Pipe: return "Broken pipe"
|
|
case .No_Size: return "file has no definite size"
|
|
case .Invalid_File: return "invalid file"
|
|
case .Invalid_Dir: return "invalid directory"
|
|
case .Invalid_Path: return "invalid path"
|
|
case .Invalid_Callback: return "invalid callback"
|
|
case .Unsupported: return "unsupported"
|
|
case .Pattern_Has_Separator: return "pattern has separator"
|
|
}
|
|
case io.Error:
|
|
switch e {
|
|
case .None: return ""
|
|
case .EOF: return "eof"
|
|
case .Unexpected_EOF: return "unexpected eof"
|
|
case .Short_Write: return "short write"
|
|
case .Invalid_Write: return "invalid write result"
|
|
case .Short_Buffer: return "short buffer"
|
|
case .No_Progress: return "multiple read calls return no data or error"
|
|
case .Invalid_Whence: return "invalid whence"
|
|
case .Invalid_Offset: return "invalid offset"
|
|
case .Invalid_Unread: return "invalid unread"
|
|
case .Negative_Read: return "negative read"
|
|
case .Negative_Write: return "negative write"
|
|
case .Negative_Count: return "negative count"
|
|
case .Buffer_Full: return "buffer full"
|
|
case .Unknown, .Empty: //
|
|
}
|
|
case runtime.Allocator_Error:
|
|
switch e {
|
|
case .None: return ""
|
|
case .Out_Of_Memory: return "out of memory"
|
|
case .Invalid_Pointer: return "invalid allocator pointer"
|
|
case .Invalid_Argument: return "invalid allocator argument"
|
|
case .Mode_Not_Implemented: return "allocator mode not implemented"
|
|
}
|
|
case Platform_Error:
|
|
return _error_string(e)
|
|
}
|
|
|
|
return "unknown error"
|
|
}
|
|
|
|
print_error :: proc(f: Handle, ferr: Error, msg: string) -> (n: int, err: Error) {
|
|
err_str := error_string(ferr)
|
|
|
|
// msg + ": " + err_str + '\n'
|
|
length := len(msg) + 2 + len(err_str) + 1
|
|
buf_ := intrinsics.alloca(length, 1)
|
|
buf := buf_[:length]
|
|
|
|
copy(buf, msg)
|
|
buf[len(msg)] = ':'
|
|
buf[len(msg) + 1] = ' '
|
|
copy(buf[len(msg) + 2:], err_str)
|
|
buf[length - 1] = '\n'
|
|
return write(f, buf)
|
|
}
|
|
|
|
|
|
@(require_results, private)
|
|
_error_string :: proc "contextless" (e: Platform_Error) -> string where intrinsics.type_is_enum(Platform_Error) {
|
|
@(require_results)
|
|
binary_search :: proc "contextless" (array: $A/[]$T, key: T) -> (index: int, found: bool) #no_bounds_check {
|
|
n := len(array)
|
|
left, right := 0, n
|
|
for left < right {
|
|
mid := int(uint(left+right) >> 1)
|
|
if array[mid] < key {
|
|
left = mid+1
|
|
} else {
|
|
// equal or greater
|
|
right = mid
|
|
}
|
|
}
|
|
return left, left < n && array[left] == key
|
|
}
|
|
|
|
|
|
if e == nil {
|
|
return ""
|
|
}
|
|
|
|
err := runtime.Type_Info_Enum_Value(e)
|
|
|
|
ti := &runtime.type_info_base(type_info_of(Platform_Error)).variant.(runtime.Type_Info_Enum)
|
|
if idx, ok := binary_search(ti.values, err); ok {
|
|
return ti.names[idx]
|
|
}
|
|
return "<unknown platform error>"
|
|
}
|
|
|
|
@(private, require_results)
|
|
error_to_io_error :: proc(ferr: Error) -> io.Error {
|
|
if ferr == nil {
|
|
return .None
|
|
}
|
|
return ferr.(io.Error) or_else .Unknown
|
|
}
|