| \ Gforth primitives |
\ Gforth primitives |
| |
|
| \ Copyright (C) 1995,1996,1997,1998,2000,2003 Free Software Foundation, Inc. |
\ Copyright (C) 1995,1996,1997,1998,2000,2003,2004 Free Software Foundation, Inc. |
| |
|
| \ This file is part of Gforth. |
\ This file is part of Gforth. |
| |
|
| \ throw execute, cfa and NEXT1 out? |
\ throw execute, cfa and NEXT1 out? |
| \ macroize *ip, ip++, *ip++ (pipelining)? |
\ macroize *ip, ip++, *ip++ (pipelining)? |
| |
|
| |
\ Stack caching setup |
| |
|
| |
ifdef(`M4_ENGINE_FAST', `include(cache1.vmg)', `include(cache0.vmg)') |
| |
|
| \ these m4 macros would collide with identifiers |
\ these m4 macros would collide with identifiers |
| undefine(`index') |
undefine(`index') |
| undefine(`shift') |
undefine(`shift') |
| |
|
| (docol) ( -- R:a_retaddr ) gforth-internal paren_docol |
(docol) ( -- R:a_retaddr ) gforth-internal paren_docol |
| ""run-time routine for colon definitions"" |
""run-time routine for colon definitions"" |
| |
#ifdef NO_IP |
| |
a_retaddr = next_code; |
| |
INST_TAIL; |
| |
goto **(Label *)PFA(CFA); |
| |
#else /* !defined(NO_IP) */ |
| a_retaddr = (Cell *)IP; |
a_retaddr = (Cell *)IP; |
| SET_IP((Xt *)PFA(CFA)); |
SET_IP((Xt *)PFA(CFA)); |
| |
#endif /* !defined(NO_IP) */ |
| |
|
| (docon) ( -- w ) gforth-internal paren_docon |
(docon) ( -- w ) gforth-internal paren_docon |
| ""run-time routine for constants"" |
""run-time routine for constants"" |
| w = *(Cell *)PFA(CFA); |
w = *(Cell *)PFA(CFA); |
| |
#ifdef NO_IP |
| |
INST_TAIL; |
| |
goto *next_code; |
| |
#endif /* defined(NO_IP) */ |
| |
|
| (dovar) ( -- a_body ) gforth-internal paren_dovar |
(dovar) ( -- a_body ) gforth-internal paren_dovar |
| ""run-time routine for variables and CREATEd words"" |
""run-time routine for variables and CREATEd words"" |
| a_body = PFA(CFA); |
a_body = PFA(CFA); |
| |
#ifdef NO_IP |
| |
INST_TAIL; |
| |
goto *next_code; |
| |
#endif /* defined(NO_IP) */ |
| |
|
| (douser) ( -- a_user ) gforth-internal paren_douser |
(douser) ( -- a_user ) gforth-internal paren_douser |
| ""run-time routine for constants"" |
""run-time routine for constants"" |
| a_user = (Cell *)(up+*(Cell *)PFA(CFA)); |
a_user = (Cell *)(up+*(Cell *)PFA(CFA)); |
| |
#ifdef NO_IP |
| |
INST_TAIL; |
| |
goto *next_code; |
| |
#endif /* defined(NO_IP) */ |
| |
|
| (dodefer) ( -- ) gforth-internal paren_dodefer |
(dodefer) ( -- ) gforth-internal paren_dodefer |
| ""run-time routine for deferred words"" |
""run-time routine for deferred words"" |
| |
#ifndef NO_IP |
| ip=IP; /* undo any ip updating that may have been performed by NEXT_P0 */ |
ip=IP; /* undo any ip updating that may have been performed by NEXT_P0 */ |
| |
#endif /* !defined(NO_IP) */ |
| SUPER_END; /* !! probably unnecessary and may lead to measurement errors */ |
SUPER_END; /* !! probably unnecessary and may lead to measurement errors */ |
| EXEC(*(Xt *)PFA(CFA)); |
EXEC(*(Xt *)PFA(CFA)); |
| |
|
| (dofield) ( n1 -- n2 ) gforth-internal paren_field |
(dofield) ( n1 -- n2 ) gforth-internal paren_field |
| ""run-time routine for fields"" |
""run-time routine for fields"" |
| n2 = n1 + *(Cell *)PFA(CFA); |
n2 = n1 + *(Cell *)PFA(CFA); |
| |
#ifdef NO_IP |
| |
INST_TAIL; |
| |
goto *next_code; |
| |
#endif /* defined(NO_IP) */ |
| |
|
| (dodoes) ( -- a_body R:a_retaddr ) gforth-internal paren_dodoes |
(dodoes) ( -- a_body R:a_retaddr ) gforth-internal paren_dodoes |
| ""run-time routine for @code{does>}-defined words"" |
""run-time routine for @code{does>}-defined words"" |
| |
#ifdef NO_IP |
| |
a_retaddr = next_code; |
| |
a_body = PFA(CFA); |
| |
INST_TAIL; |
| |
goto **(Label *)DOES_CODE1(CFA); |
| |
#else /* !defined(NO_IP) */ |
| a_retaddr = (Cell *)IP; |
a_retaddr = (Cell *)IP; |
| a_body = PFA(CFA); |
a_body = PFA(CFA); |
| SET_IP(DOES_CODE1(CFA)); |
SET_IP(DOES_CODE1(CFA)); |
| |
#endif /* !defined(NO_IP) */ |
| |
|
| (does-handler) ( -- ) gforth-internal paren_does_handler |
(does-handler) ( -- ) gforth-internal paren_does_handler |
| ""just a slot to have an encoding for the DOESJUMP, |
""just a slot to have an encoding for the DOESJUMP, |
| call ( #a_callee -- R:a_retaddr ) new |
call ( #a_callee -- R:a_retaddr ) new |
| ""Call callee (a variant of docol with inline argument)."" |
""Call callee (a variant of docol with inline argument)."" |
| #ifdef NO_IP |
#ifdef NO_IP |
| |
assert(0); |
| INST_TAIL; |
INST_TAIL; |
| JUMP(a_callee); |
JUMP(a_callee); |
| #else |
#else |
| #ifndef NO_IP |
#ifndef NO_IP |
| ip=IP; |
ip=IP; |
| #endif |
#endif |
| IF_spTOS(spTOS = sp[0]); |
IF_spTOS(spTOS = sp[0]); /* inst_tail would produce a NEXT_P1 */ |
| SUPER_END; |
SUPER_END; |
| EXEC(xt); |
EXEC(xt); |
| |
|
| #ifndef NO_IP |
#ifndef NO_IP |
| ip=IP; |
ip=IP; |
| #endif |
#endif |
| IF_spTOS(spTOS = sp[0]); |
IF_spTOS(spTOS = sp[0]); /* inst_tail would produce a NEXT_P1 */ |
| SUPER_END; |
SUPER_END; |
| EXEC(*(Xt *)a_addr); |
EXEC(*(Xt *)a_addr); |
| : |
: |
| JUMP(a_target); |
JUMP(a_target); |
| #else |
#else |
| SET_IP((Xt *)a_target); |
SET_IP((Xt *)a_target); |
| |
INST_TAIL; |
| |
NEXT_P2; |
| #endif |
#endif |
| |
SUPER_CONTINUE; /* we do our own control flow, so don't append NEXT etc. */ |
| : |
: |
| r> @ >r ; |
r> @ >r ; |
| |
|
| /* dependent upon two's complement arithmetic */ |
/* dependent upon two's complement arithmetic */ |
| Cell olddiff = n1-nlimit; |
Cell olddiff = n1-nlimit; |
| n2=n1+n; |
n2=n1+n; |
| ,if ((olddiff^(olddiff+n))>=0 /* the limit is not crossed */ |
,if (((olddiff^(olddiff+n)) /* the limit is not crossed */ |
| || (olddiff^n)>=0 /* it is a wrap-around effect */) { |
&(olddiff^n)) /* OR it is a wrap-around effect */ |
| |
>=0) { /* & is used to avoid having two branches for gforth-native */ |
| ,: |
,: |
| r> swap |
r> swap |
| r> r> 2dup - >r |
r> r> 2dup - >r |
| newdiff = -newdiff; |
newdiff = -newdiff; |
| } |
} |
| n2=n1+n; |
n2=n1+n; |
| ,if (diff>=0 || newdiff<0) { |
,if (((~diff)|newdiff)<0) { /* use | to avoid two branches for gforth-native */ |
| ,) |
,) |
| |
|
| \+ |
\+ |
| |
|
| rshift ( u1 n -- u2 ) core r_shift |
rshift ( u1 n -- u2 ) core r_shift |
| ""Logical shift right by @i{n} bits."" |
""Logical shift right by @i{n} bits."" |
| |
#ifdef BROKEN_SHIFT |
| |
u2 = rshift(u1, n); |
| |
#else |
| u2 = u1>>n; |
u2 = u1>>n; |
| |
#endif |
| : |
: |
| 0 ?DO 2/ MAXI and LOOP ; |
0 ?DO 2/ MAXI and LOOP ; |
| |
|
| lshift ( u1 n -- u2 ) core l_shift |
lshift ( u1 n -- u2 ) core l_shift |
| |
#ifdef BROKEN_SHIFT |
| |
u2 = lshift(u1, n); |
| |
#else |
| u2 = u1<<n; |
u2 = u1<<n; |
| |
#endif |
| : |
: |
| 0 ?DO 2* LOOP ; |
0 ?DO 2* LOOP ; |
| |
|
| return (Label *)n; |
return (Label *)n; |
| |
|
| (system) ( c_addr u -- wretval wior ) gforth paren_system |
(system) ( c_addr u -- wretval wior ) gforth paren_system |
| #ifndef MSDOS |
wretval = gforth_system(c_addr, u); |
| int old_tp=terminal_prepped; |
|
| deprep_terminal(); |
|
| #endif |
|
| wretval=system(cstr(c_addr,u,1)); /* ~ expansion on first part of string? */ |
|
| wior = IOR(wretval==-1 || (wretval==127 && errno != 0)); |
wior = IOR(wretval==-1 || (wretval==127 && errno != 0)); |
| #ifndef MSDOS |
|
| if (old_tp) |
|
| prep_terminal(); |
|
| #endif |
|
| |
|
| getenv ( c_addr1 u1 -- c_addr2 u2 ) gforth |
getenv ( c_addr1 u1 -- c_addr2 u2 ) gforth |
| ""The string @i{c-addr1 u1} specifies an environment variable. The string @i{c-addr2 u2} |
""The string @i{c-addr1 u1} specifies an environment variable. The string @i{c-addr2 u2} |
| char * pattern = cstr(c_addr2, u2, 0); |
char * pattern = cstr(c_addr2, u2, 0); |
| flag = FLAG(!fnmatch(pattern, string, 0)); |
flag = FLAG(!fnmatch(pattern, string, 0)); |
| |
|
| |
set-dir ( c_addr u -- wior ) gforth set_dir |
| |
""Change the current directory to @i{c-addr, u}. |
| |
Return an error if this is not possible"" |
| |
wior = IOR(chdir(tilde_cstr(c_addr, u, 1))); |
| |
|
| |
get-dir ( c_addr1 u1 -- c_addr2 u2 ) gforth get_dir |
| |
""Store the current directory in the buffer specified by @{c-addr1, u1}. |
| |
If the buffer size is not sufficient, return 0 0"" |
| |
c_addr2 = getcwd(c_addr1, u1); |
| |
if(c_addr2 != NULL) { |
| |
u2 = strlen(c_addr2); |
| |
} else { |
| |
u2 = 0; |
| |
} |
| |
|
| \+ |
\+ |
| |
|
| newline ( -- c_addr u ) gforth |
newline ( -- c_addr u ) gforth |
| av_double(alist, r); |
av_double(alist, r); |
| |
|
| av-longlong ( d -- ) gforth av_longlong |
av-longlong ( d -- ) gforth av_longlong |
| |
#ifdef BUGGY_LONG_LONG |
| |
av_longlong(alist, d.lo); |
| |
#else |
| av_longlong(alist, d); |
av_longlong(alist, d); |
| |
#endif |
| |
|
| av-ptr ( c_addr -- ) gforth av_ptr |
av-ptr ( c_addr -- ) gforth av_ptr |
| av_ptr(alist, void*, c_addr); |
av_ptr(alist, void*, c_addr); |
| av_double(alist, r); |
av_double(alist, r); |
| |
|
| av-longlong-r ( R:d -- ) gforth av_longlong_r |
av-longlong-r ( R:d -- ) gforth av_longlong_r |
| |
#ifdef BUGGY_LONG_LONG |
| |
av_longlong(alist, d.lo); |
| |
#else |
| av_longlong(alist, d); |
av_longlong(alist, d); |
| |
#endif |
| |
|
| av-ptr-r ( R:c_addr -- ) gforth av_ptr_r |
av-ptr-r ( R:c_addr -- ) gforth av_ptr_r |
| av_ptr(alist, void*, c_addr); |
av_ptr(alist, void*, c_addr); |
| SAVE_REGS |
SAVE_REGS |
| av_call(alist); |
av_call(alist); |
| REST_REGS |
REST_REGS |
| |
#ifdef BUGGY_LONG_LONG |
| |
d.lo = llrv; |
| |
d.hi = 0; |
| |
#else |
| d = llrv; |
d = llrv; |
| |
#endif |
| |
|
| av-call-ptr ( -- c_addr ) gforth av_call_ptr |
av-call-ptr ( -- c_addr ) gforth av_call_ptr |
| SAVE_REGS |
SAVE_REGS |
| w = va_arg_int(clist); |
w = va_arg_int(clist); |
| |
|
| va-arg-longlong ( -- d ) gforth va_arg_longlong |
va-arg-longlong ( -- d ) gforth va_arg_longlong |
| |
#ifdef BUGGY_LONG_LONG |
| |
d.lo = va_arg_longlong(clist); |
| |
d.hi = 0; |
| |
#else |
| d = va_arg_longlong(clist); |
d = va_arg_longlong(clist); |
| |
#endif |
| |
|
| va-arg-ptr ( -- c_addr ) gforth va_arg_ptr |
va-arg-ptr ( -- c_addr ) gforth va_arg_ptr |
| c_addr = (char *)va_arg_ptr(clist,char*); |
c_addr = (char *)va_arg_ptr(clist,char*); |
| return 0; |
return 0; |
| |
|
| va-return-longlong ( d -- ) gforth va_return_longlong |
va-return-longlong ( d -- ) gforth va_return_longlong |
| |
#ifdef BUGGY_LONG_LONG |
| |
va_return_longlong(clist, d.lo); |
| |
#else |
| va_return_longlong(clist, d); |
va_return_longlong(clist, d); |
| |
#endif |
| return 0; |
return 0; |
| |
|
| va-return-float ( r -- ) gforth va_return_float |
va-return-float ( r -- ) gforth va_return_float |
| finish-code ( -- ) gforth finish_code |
finish-code ( -- ) gforth finish_code |
| ""Perform delayed steps in code generation (branch resolution, I-cache |
""Perform delayed steps in code generation (branch resolution, I-cache |
| flushing)."" |
flushing)."" |
| |
IF_spTOS(sp[0]=spTOS); /* workaround for failing to save spTOS |
| |
(gcc-2.95.1, gforth-fast --enable-force-reg) */ |
| finish_code(); |
finish_code(); |
| |
IF_spTOS(spTOS=sp[0]); |
| |
|
| forget-dyncode ( c_code -- f ) gforth-internal forget_dyncode |
forget-dyncode ( c_code -- f ) gforth-internal forget_dyncode |
| f = forget_dyncode(c_code); |
f = forget_dyncode(c_code); |
| |
|
| \g static_super |
\g static_super |
| |
|
| \C #if !defined(GFORTH_DEBUGGING) && !defined(INDIRECT_THREADED) && !defined(DOUBLY_INDIRECT) && !defined(VM_PROFILING) |
ifdef(`M4_ENGINE_FAST', |
| |
`include(peeprules.vmg)') |
| include(peeprules.vmg) |
|
| |
|
| \C #endif |
|
| |
|
| \g end |
\g end |