( -*- forth -*- @@PLEAC@@_NAME @@SKIP@@ Forth @@PLEAC@@_WEB @@SKIP@@ http://www.forth.org/ @@PLEAC@@_INTRO Implementations vary wildly. I'm pretty sure these are all legal ANS. @@PLEAC@@_APPENDIX Variable names tend to assume you have made a word with that name which leaves the value you want on the stack. Some Forths have these and others do not, so to be safe: ) -1 constant t 0 constant nil : not ( f -- f ) 0= ; ( @@PLEAC@@_1.0 ) : string ( -- c- u ) s" \n" ; ( two characters, \ and an n ) : string ( -- c- u ) s" John 'Maddog' Orwant" ; ( literal single quotes ---------------------------- ) : string ( -- ) cr ; ( a "newline" character, or: ) : string ( -- ) newline type ; ( a "newline" character, or: ) : string ( -- c ) 13 ; ( depending, obviously, on your platform's newline ) ( double quotes must be entered manually ) : put-quote ( c- u -- ) + 34 swap c! ; : string ( -- c- u ) s" Jon Maddog Orwant" over dup 4 put" 11 put" ( literal double quotes ) ; ( no q/ / type quoting options ---------------------------- some words to make a here document: ) : newline? ( c -- f ) dup 10 = swap 13 = or ( or whatever the newline is for your platform ) ; : get-terminator ( -- c- u ) here dup 12 chars allot 12 accept ( change 12 if you absolutely must have a larger EOF marker than 12 characters ) cr ; : get-char ( -- c ) key dup emit dup here 1 chars allot c! ; : line-in ( -- u ) 0 begin 1 + get-char 13 = until 1 - ; : <<< ( -- c- u ) \ A parsing <<< would be cute but I don't want to deal with writing one get-terminator here -rot begin line-in here over - 1 - swap 2over compare not until drop drop here over - ; <<< EOF This is a multiline here document terminated by EOF on a line by itself EOF ( @@PLEAC@@_1.1 ) : substr ( c- u u u -- c- u ) dup 0= if drop tuck - else 2dup + 2swap -rot dup -rot > if rot drop 2dup swap - nip else drop swap then then -rot + 0 -rot swap here swap 2dup chars allot drop 0 do over i + c@ over i + c! rot 1+ -rot loop nip swap ; : value string offset count substr ; : value string offset 0 substr ; : reassign ( c- u u u c- u -- c- u ) { addr1 strlen1 offset count addr2 strlen2 } strlen1 count - strlen2 + allocate throw addr1 over offset cmove addr2 over offset chars + strlen2 cmove addr1 offset chars + count chars + over offset chars + strlen2 chars + strlen1 count - offset - cmove strlen1 count - strlen2 + ; string offset count newstring reassign ( ---------------------------- get a 5-byte string, skip 3, then grab 2 8-byte strings, then the rest. ) : unpack ( c- u -- c-1 u1 c-2 u2 c-3 u3 c-4 u4 ) 2>r 5 chars allocate throw 2r@ drop over 5 cmove 5 8 chars allocate throw 2r@ drop 8 chars + over 8 cmove 8 8 chars allocate throw 2r@ drop 16 chars + over 8 cmove 8 2r> 24 - dup chars allocate throw dup 2swap >r r@ swap 24 + -rot cmove r> ; ( split at five byte boundaries ) : fivers ( c- u -- c-1 u1 c-2 u2 ... c-n un ) 5 / 0 do 5 chars allocate throw 2dup swap i 5 * chars + swap 5 cmove 5 rot loop drop ; ( chop string into individual characters ) : chop ( c- u -- c1 c2 ... cn ) 0 do dup i chars + c@ swap loop drop ; ( ---------------------------- ) : string s" This is what you have" ; : first string drop 1 ; ( "T" ) : start string drop 5 chars + 2 ; ( "is" ) : rest string 13 swap over - -rot chars + swap; ( "you have" ) : last string 1 - chars + 1 ; ( "e" ) : end string 4 - chars + 4 ; ( "have" ) : piece string 8 - chars + 3 ; ( "you" ) ( ---------------------------- ) : string s" This is what you have" ; string type ( This is what you have ) string 5 2 s" wasn't" reassign ( change "is" to "wasn't" This wasn't what you have ) string 13 12 s" ondrous" reassign ( replace last 12 characters This wasn't wondrous ) string swap 1 chars + swap 1 - ( delete first character his wasn't wondrous ) string 10 - dup -rot resize throw swap ( delete last 10 characters his wasn' ) ( ---------------------------- TODO RE's ------------------------------ exchange first and last letters in a string ) : a s" make a hat" ; : exchange ( c- u -- ) 1 - 2dup chars + c@ -rot over c@ -rot chars over + rot swap c! c! ; a exchange type ( take a ham ------------------------------ extract column ) : a "To be or not to be" ; : b ( c- u u u -- c- u ) dup chars allocate throw swap ( c- len off cou c-2 ) >r >r nip chars + 2r@ cmove r> r> ; a 6 6 ( skip 6, grab 6 ) b type ( or not ) ( @@PLEAC@@_1.2 ) ( use b if b is true, else c ) : default ( x a- -- ) dup @ not if ! else 2drop then ; c b default a ! ( set x to y unless x is already true ) : !? dup if ! else drop then ; x dup @ 0= y @ and swap !? ( @@PLEAC@@_1.3 ) : swap! ( x x -- ) 2dup @ swap @ rot ! swap ! ; var1 var2 swap! ( ---------------------------- ) variable temp a @ temp ! b @ a ! temp @ b ! ( ---------------------------- ) ( @@PLEAC@@_1.4 ) ( not an issue, really; a char is just an integer and stored as a cell on the stack ) : ord ( c -- u ) ; : chr ( u -- c ) ; ( or, to see the char ) : chr ( u -- ) emit ; ( but if it's an actual variable rather than a stack element: ) : ord ( c- -- a- ) 1 cells allocate throw swap c@ swap ! ; : chr ( a- -- c- ) 1 chars allocate throw swap @ swap c! ; ( @@PLEAC@@_1.5 ) ( with a literal word ) : process-string ( c- u -- variable stack effect ) 0 do dup i chars + c@ ( your-word ) swap loop drop ; ( or better yet, with an execution token ) : map-xt ( xt c- u -- variable stack effect ) 0 do 2dup i chars + c@ swap execute loop ; ( @@PLEAC@@_1.6 ) : 4dup-nip ( a b c d -- a b c d a b d ) 2over 2over nip ; : c[i]@ ( c- i -- c ) chars + c@ ; : c[i]! ( c- i c -- ) chars + c! ; : -cmove ( c- x c- -- c- x ) over 0 u+do i 4dup-nip 1+ - c[i]@ 2over nip rot c[i]! loop rot drop swap ; : revbytes ( c- x -- c- x ) dup allocate throw -cmove ; : gnirts s" string" revbytes ; : space? 32 = ; : ^? ( c c- a -- c t ) c[i]@ tuck space? not swap space? and ; : -^? ( c c- a -- c t ) c[i]@ tuck space? swap space? not and ; : count-words ( c- x -- x ) 0 -rot 32 swap 0 u+do over i ^? if rot 1+ -rot then loop drop drop ; : last-spaces ( c- x -- x ) 0 -rot begin 1- 2dup c[i]@ space? over 0 >= and while rot 1+ -rot repeat drop drop ; : last-word ( c- x -- x ) 0 -rot begin 1- 2dup c[i]@ space? not over 0 >= and while rot 1+ -rot repeat drop drop ; : back>front ( c- x c- x -- ) 2swap 2dup last-word -rot chars + over chars - swap rot drop rot swap cmove ; : sback>front ( c- x c- x -- ) 2swap 2dup last-spaces -rot chars + over chars - swap rot drop rot swap cmove ; : sdrow ( c- x -- c- x ) 2dup last-spaces - dup >r dup chars allocate throw dup >r over 2swap 2dup count-words 0 u+do 2dup last-word >r 2over 2over 2swap back>front r@ - rot drop dup -rot 2swap r> rot + swap 2swap 2dup last-spaces >r 2over 2over 2swap sback>front r@ - rot drop dup -rot 2swap r> rot + swap 2swap loop 2drop 2drop r> r> ; : count-spaces ( c- x -- x ) 0 -rot 0 u+do dup i c[i]@ space? if swap 1+ swap then loop drop ; : move-if-not-space ( c- c- -- f ) swap c@ dup space? if drop drop false else swap c! true then ; : squash-words ( c- x -- c- x ) 2dup count-spaces over swap - dup chars allocate throw swap >r dup >r -rot 0 u+do 2dup i chars + swap move-if-not-space if swap 1 chars + swap then loop drop drop r> r> ; : confused ( c- x -- c- x ) squash-words revbytes ; 80 constant biggest-word create buff biggest-word 2 + allot : palindrome? ( c- x -- f ) 2dup revbytes compare not ; : chomp ( c- x -- c- x ) 2dup last-spaces - ; s" /usr/share/dict/words" r/o open-file throw value dict : scan-dictionary ( c- x -- ) begin buff biggest-word dict read-line throw while dup buff swap chomp dup 5 > -rot palindrome? and if buff swap type cr else drop then repeat ;