additional part-cell primitives
[rrq/rrqforth.git] / rrqforth.asm
index c4b63b4e62d66bc362acc3a1c1fc5a55ddcbdd19..18009d41bfba8f75f5b7afef2463f89988bf10e6 100644 (file)
@@ -2,33 +2,12 @@
        format elf64 executable
        entry main
 
-previous_word = 0      ; Used for chaining the words
-
 include 'machine.asm'
 
 ;;; ============================================================
 
-       segment readable executable
-
-;;; This is the very first word
+       segment readable writable executable
        
-       ;; FORTH is the last word of WORDLIST FORTH
-       WORD p_forth,'FORTH',dowordlist
-       ;; ( -- )
-       ;; Change to use this wordlist
-       dq last_forth_word
-
-       WORD p_system,'SYSTEM',dowordlist
-       ;; ( -- )
-       ;; Change to use this wordlist
-       dq last_system_word
-
-last_wordlists_word:
-       WORD p_wordlists,'WORDLISTS',dowordlist
-       ;; ( -- )
-       ;; Change to use this wordlist
-       dq p_wordlists_TFA
-
 ;;; ========================================
 ;;; These are the core "execution semantics" words, which are placed
 ;;; first so as to remain at the same binary address at successive
@@ -46,6 +25,8 @@ last_wordlists_word:
 ;;; rbp = return stack pointer
 ;;; 
 
+previous_word = 0      ; Used for chaining the words
+
        WORD p_dofasm,'doFASM',dovariable
        ;; Execution semantics for assembly words.
 dofasm:
@@ -92,22 +73,19 @@ dostring:
        pushpname rax
        next
 
-       WORD p_dowordlist,'doWORDLIST',dovariable
-       ;; Execution semantics for DOES>
-       ;; [cfa-8] holds the adjustment offset ("does offset")
-dowordlist:
-       pushr rsi
-       lea rsi, [rax+8]        ; rsi = the DFA of the rax word
-       add rsi,qword [rax-8]   ; adjust rsi by the "does offset'
-       next
-
+       WORD p_calltrace,'[calltrace]',dovalue
+       ;; Common call point for debugging
+       ;; rax = cfa of called word
+       ;; rsi = cell* of next forth word
+       ;; [$rsp] = from where the call was
+       ret
 
 include 'syscalls.asm'
 
 ;;; ========================================
 ;;; The stacks are placed here.
        
-       segment readable writable
+       ;segment readable writable
 
        WORD return_stack,'RETURN-STACK',dovariable
        ;; The return stack
@@ -122,15 +100,24 @@ last_system_word:
        rb 1048576              ; 1 Mb data stack
 DS_TOP:                                ; The initial rsp
 
+
 ;;; ========================================
 ;;; Core execution control words
 
-       segment readable executable
+       ;segment readable executable
+
+include 'signals.asm'
 
 ;;; At fasm compilation: reset previous_word to make a new word list
 ;;; Words above belong to the SYSTEM wordlist, and the following
 ;;; belong to the FORTH wordlist.
-previous_word = last_wordlists_word
+previous_word = 0
+
+       WORD p_system,'SYSTEM',dovariable
+       ;; ( -- dfa )
+       ;; The SYSTEM word list
+       dq last_system_word     ; tfa of last SYSTEM word
+       dq p_forth_DFA          ; dfa of successor word list
 
        WORD inline_code,'[ASM]',fasm
        ;; ( -- )
@@ -141,13 +128,17 @@ previous_word = last_wordlists_word
        jmp qword rsi
 
        WORD p_execute,'EXECUTE',fasm
-       ;; ( tfa -- )
+       ;; ( cfa -- )
        ;; Execute the word
        pop rax
-       tfa2cfa rax
        jmp qword [rax]         ; goto code of that FORTH word (64 bit jump)
        
-       WORD p_exit, 'EXIT',fasm
+       WORD p_sysexit, 'EXIT',
+       ;; ( v -- )
+       ;; Terminate RRQFORTH with error code
+       dq sys_exit
+
+       WORD p_return, 'RETURN',fasm
        ;; ( -- ) ( R: addr -- )
        ;; Returns execution to the calling definition as per the
        ;; return stack.
@@ -163,53 +154,137 @@ terminate_special:
        mov eax,60
        syscall
 
-       WORD p_branch,'BRANCH',fasm
-       ;; ( -- )
-       ;; Using subsequent inline cell as branch offset, branch
-       ;; accordingly
-       add rsi,qword [rsi]     
-       add rsi,8
-       next
-       
-       WORD p_zero_branch,'0BRANCH',fasm
-       ;; ( v -- )
-       ;; Using subsequent inline cell as branch offset, branch
-       ;; accordingly if the stacked value is zero, otherwise just
-       ;; skip over the branch offset
-       pop rax
-       cmp rax,0
-       jne p_zero_branch_SKIP
-       add rsi,qword [rsi]
-p_zero_branch_SKIP:
-       add rsi,8
-       next
-
 ;;; ========================================
 ;;; Core extension(s)
 
-       segment readable writable executable
+       ;segment readable writable executable
        
+include 'control.asm'
 include 'wordlists.asm'
 include 'memory.asm'
 include 'stack.asm'
 include 'math.asm'
 include 'logic.asm'
 include 'stdio.asm'
+include 'temp.asm'
 include 'compile.asm'
 
        WORD p_program_version,'PROGRAM_VERSION',dostring
-       STRING 'RRQ Forth version 0.1 - 2021-05-13',10
+       STRING 'RRQ Forth version 0.1 - 2021-06-05',10
 
        WORD p_stdin,'STDIN',dovalue
        ;; Initialised to hold a STREAM for fd 0
        dq 0
 
-;;; The main entry point.
+       WORD p_args,'MAIN-ARGS',dovalue
+       ;; Pointer to initial argument block
+       dq 0       ; *(int argc,(char*[argc]) argv)
+
+       WORD p_verboseQ,'VERBOSE?',dovariable
+       ;; Flags whether the running is in verbose mode ot not
+       dq 0       ; 
+
+       WORD p_lparen,'(',fasm,IMMEDIATE
+       pushr rsi
+p_lparen_loop:
+       DOFORTH  p_input, p_get, p_read_word
+       pop rax
+       pop rbx
+       cmp rax,0 ; end of stream
+       je p_lparen_exit
+       cmp rax,1
+       jne p_lparen_loop
+       push rbx
+       push p_lparen_rparen
+       push 1
+       DOFORTH p_strncmp
+       pop rax
+       cmp rax,0
+       jne p_lparen_loop
+p_lparen_exit:
+       popr rsi
+       next
+p_lparen_rparen: db ')',0
+
+;;; ******** The main entry point. ********
+;;; This could be set up as a WORD but it isn't
+
+main:  
+       ;; Initial rsp points to the arguments block of size (64 bits)
+       ;; followed by the argument pointers.
+       mov qword [p_args_DFA],rsp
+       mov rbp,RS_TOP
+       call p_setup_signals_DFA
+       call main_is_verbose
+       mov qword [p_verboseQ_DFA],rdx
+       jmp p_quit_DFA          ; QUIT
+
+       ;; Subroutine to check the command line for a "-v"
+       ;; return boolean in rdx
+       ;; implementation for that 2 byte asciiz string
+main_is_verbose_data:
+       db '-v',0
+
+main_is_verbose:
+       mov rbx,qword [p_args_DFA] ; Pointer to main arguments
+       mov r8,qword [rbx]         ; r8 = count of pointers
+       xor rdx,rdx
+       cld
+main_is_verbose_next:
+       dec r8
+       jl main_is_not_verbose
+       add rbx,8
+       mov rsi,qword [rbx]
+       mov rdi,main_is_verbose_data
+main_is_verbose_loop:
+       cmpsb
+       jne main_is_verbose_next
+       cmp byte[rsi-1],0
+       jne main_is_verbose_loop
+       not rdx
+main_is_not_verbose:
+       ret
+
+       WORD p_process_args_var,'PROCESS-ARGS-VAR',dovariable
+       ;; ( -- a )
+       ;; Two cells for iterating and load the main args
+p_process_args_ptr: dq 0
+p_process_args_end: dq 0
+       
+       WORD p_process_args,'PROCESS-ARGS',fasm
+       pushr rsi
+       mov rax,qword [p_args_DFA] ; args*
+       mov rbx,qword [rax] ; count
+       shl rbx,3
+       add rax,8
+       add rbx,rax ; end
+       mov qword [p_process_args_end],rbx
+       add rax,8
+       mov qword [p_process_args_ptr],rax
+p_process_args_loop:
+       mov rax,qword [p_process_args_ptr]
+       cmp rax,qword [p_process_args_end]
+       jge p_process_args_done
+       add qword [p_process_args_ptr],8
+       mov rax,qword [rax]
+       mov bl,[rax]
+       cmp bl,'-'
+       je p_process_args_loop
+       push rax
+       push rax
+       FORTH
+       dq p_strlen, p_load_file
+       BRANCH 0, p_quit_ERROR
+       ENDFORTH
+       jmp p_process_args_loop
+p_process_args_done:
+       popr rsi
+       next
+       
 ;;; This word is also the last word before syscalls
 last_forth_word:
        WORD p_quit,'QUIT',fasm
        ;; QUIT is the program entry point ********************
-main:
        mov rsp,DS_TOP
        mov rbp,RS_TOP
        cmp qword [p_stdin_DFA],0
@@ -219,36 +294,31 @@ main:
        push 10000
        DOFORTH p_stream
        pop qword [p_stdin_DFA] ; Assign STDIN
+       DOFORTH p_process_args
 
 p_quit_INITIALIZED:
-       ;; Initial blurb
+       ;; Setup INPUT from STDIN
        FORTH
-       dq p_program_version
-       dq p_tell
-       dq p_stdin
-       dq p_clear_stream
-       dq p_stdin
-       dq p_evaluate_stream
+       dq p_verboseQ, p_get
+       BRANCH 0,p_quit_EVAL
+       dq p_program_version, p_tell
+p_quit_EVAL:
+       dq p_stdin, p_evaluate_stream
        BRANCH 0,p_quit_ERROR
-       dq p_false
-       dq sys_exit
+       dq p_false, sys_exit
 p_quit_ERROR:
        dq p_literal_string
        STRING 10,'*** Unknown word: '
        dq p_tell
-       dq p_this_word
-       dq p_2get
-       dq p_tell
-       dq p_literal_string
-       STRING 10
-       dq p_tell
+       dq p_this_word, p_2get, p_tell
+       dq p_nl, p_emit
        ENDFORTH
        mov rbp,RS_TOP          ; reset the return stack
-       jmp main
+       jmp p_quit_INITIALIZED
 
 ;;; ========================================
 
-       segment readable writable
+       ;segment readable writable
        
 heap_start:
        rb 1048576              ; +1 Mb heap