Files
klibc/src/file.asm

222 lines
4.4 KiB
NASM

%include "src/constants.asm"
extern __INTERNAL_fmt
extern strlen
extern umask_get
section .rodata
; Open flags
O_ACCMODE equ 00000003q
O_RDONLY equ 00000000q
O_WRONLY equ 00000001q
O_RDWR equ 00000002q
O_CREAT equ 00000100q ;create if not exist
O_EXCL equ 00000200q ;fail if file exists and O_EXCL+O_CREAT set
O_NOCTTY equ 00000400q
O_TRUNC equ 00001000q
O_APPEND equ 00002000q
O_NONBLOCK equ 00004000q
O_DSYNC equ 00010000q
FASYNC equ 00020000q
O_DIRECT equ 00040000q
O_LARGEFILE equ 00100000q
O_DIRECTORY equ 00200000q
O_NOFOLLOW equ 00400000q
O_NOATIME equ 01000000q
O_CLOEXEC equ 02000000q
__O_SYNC equ 04000000q
O_SYNC equ (__O_SYNC | O_DSYNC)
O_PATH equ 01000000q
__O_TMPFILE equ 02000000q
O_TMPFILE equ (__O_TMPFILE | O_DIRECTORY)
O_NDELAY equ O_NONBLOCK
; Permission flags
S_IXOTH equ 00001q ;o=x
S_IWOTH equ 00002q ;o=w
S_IROTH equ 00004q ;o=r
S_IRWXO equ 00007q ;o=rwx
S_IXGRP equ 00010q ;g=x
S_IWGRP equ 00020q ;g=w
S_IRGRP equ 00040q ;g=r
S_IRWXG equ 00070q ;g=rwx
S_IXUSR equ 00100q ;u=x
S_IWUSR equ 00200q ;u=w
S_IRUSR equ 00400q ;u=r
S_IRWXU equ 00700q ;u=rwx
S_ISVIX equ 0001000q ;sticky bit
S_ISGID equ 0002000q ;set-group-ID bit
S_ISUID equ 0004000q ;set-user-ID bit
section .bss
section .text
global fopen
global fclose
global fexist
global fwrite
;----- fopen(*file[], char mode) -----;
; Opens a file
; Return value: Pointer to opened file pointer, -errno otherwise
; Possible modes:
; 'r' Read-only, file must exist
; 'w' Write-only. Creates empty file or truncates an existing file
; 'a' Append. Creates file if it does not exist
; 'R' Read+write, file must exist
; 'W' Read+write. Creates empty file or truncates an existing file
; 'A' Read+append. Creates file if it does not exist
; <!> Invalid mode will also return 0 (EOS)
; Used registers:
; rax* (ret)
; rdi (arg) Pointer to string holding (path+)file name
; rsi Mode to open the file with (see description above)
fopen:
%macro createFile 1
push rdi
push rsi
call fexist
pop rsi
pop rdi
test rax, 1
jz %%createFile
jmp %1
%%createFile:
push rsi
push rdi
call umask_get
pop rdi
push rdi
not rax
mov rsi, 0666q
and rsi, rax
mov rax, NR_creat
syscall
pop rdi
pop rsi
%endmacro
;;;
;;; ENTRY
;;;
sub rsp, SIZE_QWORD
cmp sil, 'r'
je .setMode_r
cmp sil, 'w'
je .setMode_w
cmp sil, 'a'
je .setMode_a
cmp sil, 'R'
je .setMode_R
cmp sil, 'W'
je .setMode_W
cmp sil, 'A'
je .setMode_A
mov rax, -EINVAL
jmp .quit
.setMode_r:
mov rsi, O_RDONLY
jmp .open
.setMode_w:
createFile .open_w
.open_w:
mov rsi, O_WRONLY | O_TRUNC
jmp .open
.setMode_a:
createFile .open_a
.open_a:
mov rsi, O_WRONLY | O_APPEND
jmp .open
.setMode_R:
mov rsi, O_RDWR
jmp .open
.setMode_W:
createFile .open_W
.open_W:
mov rsi, O_RDWR | O_TRUNC
jmp .open
.setMode_A:
createFile .open_A
.open_A:
mov rsi, O_RDWR | O_APPEND
jmp .open
.open:
mov rax, NR_open
syscall
.quit:
add rsp, SIZE_QWORD
ret
;----- fclose(*filePointer) -----;
; Closes an opened file
; Return value: 0 if file closed. Negative error number if an error occured
; Used registers:
; rax* (ret)
; rdi (arg) Pointer to opened file
fclose:
sub rsp, SIZE_QWORD
cmp rdi, 3
jl .invalid_fd ;disallow closing stdin,stdout,stderr
mov rax, NR_close
syscall
jmp .quit
.invalid_fd:
mov rax, -EINVAL
.quit:
add rsp, SIZE_QWORD
ret
;----- fexist(*file[]) -----;
; Checks if given file exists (attempts to open the file)
; Return value: 1- File exists. 0- File does not exist or is a directory
; Used registers:
; rax* (ret)
; rdi (arg) Pointer to string holding (path+)file name
fexist:
sub rsp, SIZE_QWORD
mov rsi, 'r'
call fopen
test rax, rax
js .not_exist
mov rdi, rax
call fclose
mov rax, 1
jmp .quit
.not_exist:
xor rax, rax
.quit:
add rsp, SIZE_QWORD
ret
;--- fwrite(*filePointer, *data[], ...) -----;
; Writes data to given file. Allows format specifiers and variable arguments (see `__INTERNAL_fmt` for supported specifiers)
; Return value: Length of given string(data[]), 0 if nothing was written (eg invalid file or file not writable, or data[] is empty)
; Used registers:
; rax* (ret)
; rdi (arg) Pointer to file
; rsi (arg) Data to write
fwrite:
push rbp
mov rbp, rsp
cmp rdi, 2
jg .cnt
xor rax, rax
leave
ret
.cnt:
mov rax, rdi
mov rdi, rsi
mov rsi, rdx
mov rdx, rcx
mov rcx, r8
mov r8, r9
mov r9, [rbp + RBP_OFFSET_CALLER]
call __INTERNAL_fmt
leave
ret