X-Git-Url: https://git.rrq.au/?a=blobdiff_plain;f=stdio.asm;h=191e09893e2226146f1ee1d0b92a0650d66ec06d;hb=aa6d10c30115571b859b9e7e3d6e8a9b9ced599b;hp=5767d8b8258be3c1b11fd92d67832ce6245cbfa6;hpb=3087a82c8afa225bb29f08b5b29bed2d7dcef3ca;p=rrq%2Frrqforth.git diff --git a/stdio.asm b/stdio.asm index 5767d8b..191e098 100644 --- a/stdio.asm +++ b/stdio.asm @@ -2,20 +2,29 @@ ;;; Dynamic memory management. Allocated with MALLOC and released with ;;; MUNMAP (see below) - ;; ( size -- addr ) - ;; Allocates memory (using brk) WORD p_malloc,'MALLOC',fasm + ;; ( size -- addr ) + ;; Allocates memory (using mmap) pushr rsi ; pretend it's a FORTH word since it ; ends via sys_mmap_asm pop rax push qword 0 ; address of mapping (suggestion) push rax ; length of mapping push qword 3 ; protection mode PROT_READ | PROT_WRITE - push qword 8226 ; flags PRIVATE | ANONYMOUS | LOCKED + push qword 34 ; flags PRIVATE | ANONYMOUS push qword -1 ; fd -1 push qword 0 ; offset jmp sys_mmap_asm ; exit via sys_mmap + WORD p_realloc,'REALLOC',fasm + ;; ( addr old new -- ) + ;; Try remapping a given MMAP region of old size to a new size + ;; mremap(void *old_address, size_t old_size, + ;; size_t new_size, int flags, ... /* void *new_address */); + pushr rsi + push 1 ; MREMAP_MAYMOVE + jmp sys_mmap_asm ; exit via sys_mmap + ;;; ======================================== ;;; Input stream handling. ;;; @@ -32,7 +41,7 @@ ;;; * block address ;;; * -1 ;;; * size of block -;;; * current read position (starts at 8 = after block size) +;;; * current read position WORD p_stream,'STREAM',fasm ;; ( fd size -- addr ) or ( block -1 -- addr ) @@ -77,7 +86,37 @@ p_stream_MEM: mov rbx,qword [rax+16] ; copy fill mov qword [rax+24],rbx ; into current next - + + WORD p_stream_nchars,'STREAM-NCHARS',fasm + ;; ( stream -- n ) + ;; Scan over whitespace in the stream buffer (without actually + ;; consuming) and tell how much then remains. + pushr rsi + mov rcx,qword [rsp] + mov rbx,qword [rcx+16] ; fill + sub rbx,qword [rcx+24] ; current + cmp qword [rcx+8],-1 + je p_stream_nchars_memblock + mov rsi,rcx + add rsi,32 + jmp p_stream_nchars_skipblanks +p_stream_nchars_memblock: + mov rsi,qword [rcx] +p_stream_nchars_skipblanks: + add rsi,qword [rcx+24] ; + cmp rbx,0 + je p_stream_nchars_done +p_stream_nchars_loop: + lodsb + cmp al,32 + jg p_stream_nchars_done + dec rbx + jg p_stream_nchars_loop +p_stream_nchars_done: + mov qword [rsp],rbx + popr rsi + next + ;;; ======================================== ;;; Stream reading ;;; READ-STREAM-CHAR ( stream -- ch ) @@ -86,10 +125,10 @@ p_stream_MEM: ;; ( stream -- ch ) pushr rsi mov rax,qword [rsp] - mov rbx,[rax+16] ; fill + mov rbx,qword [rax+16] ; fill p_read_stream_char.READ: - mov rcx,[rax+24] ; current + mov rcx,qword [rax+24] ; current cmp rbx,rcx jg p_read_stream_char.CHAR @@ -173,6 +212,39 @@ p_read_word_nomore: popr rsi next + WORD p_double_quote,'"',fasm ;; " (fool emacs) + ;; ( -- char* n ) + ;; Scan to double quote in stream buffer, putting the string + ;; on PAD, plus an extra NUL, then copy that into a new temp + ;; object, but exclude the NUL from the returned count, n. + pushr rsi + push p_pad_DFA + push 0 +p_double_quote_loop: + DOFORTH p_input, p_get, p_read_stream_char + pop rax + cmp rax,0 + jl p_double_quote_endstream + cmp rax,'"' ; " (fool emacs) + je p_double_quote_endquote + lea rdi,[p_pad_DFA] + add rdi,qword [rsp] + stosb + inc qword [rsp] + jmp p_double_quote_loop +p_double_quote_endquote: +p_double_quote_endstream: + lea rdi,[p_pad_DFA] + add rdi,qword [rsp] + stosb + ;; copy PAD string into new temp object + inc qword [rsp] + DOFORTH p_str2temp + dec qword [rsp] + add qword [rsp+8],8 ; adjust pointer + popr rsi + next + WORD p_tell,'TELL',fasm ;; ( chars* n -- ) ;; Write n bytes from chars* to stdout @@ -187,19 +259,17 @@ p_read_word_nomore: popr rsi next -p_emit_buffer: dq 0 - WORD p_emit,'EMIT',fasm ;; ( c -- ) ;; Write byte to stdout pushr rsi - pop rax - mov [p_emit_buffer],al + mov rax,rsp push 1 - push p_emit_buffer + push rax push 1 DOFORTH sys_write - pop rax + pop rax ; ignore return value + pop rax ; drop input data popr rsi next @@ -211,5 +281,37 @@ p_emit_buffer: dq 0 WORD p_sp,'SP',dovalue ;; ( -- c ) ;; Pushes a space character on the stack - dq 10 - + dq 32 + + WORD p_digits,'DIGITS',dovariable + db '0123456789abcdef' + + WORD p_dot,'.',fasm + ;; ( v -- ) + ;; Print TOP value as unsigned BASE integer + pushr rsi + mov rax,qword [rsp] + cmp rax,0 + jge p_dot_positive + cmp qword [p_base_DFA],10 + jne p_dot_positive + push '-' + DOFORTH p_emit + mov rax,qword [rsp] + neg rax +p_dot_positive: + xor rdx,rdx + div qword [p_base_DFA] ; rdx:rax / BASE => q=rax, r=rdx + mov qword [rsp],rdx + cmp rax,0 + je p_dot_remainder + push rax + DOFORTH p_dot +p_dot_remainder: + pop rdx + xor rax,rax + mov al,[p_digits_DFA+rdx] + push rax + DOFORTH p_emit + popr rsi + next