cc_amd64.s 139 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788378937903791379237933794379537963797379837993800380138023803380438053806380738083809381038113812381338143815381638173818381938203821382238233824382538263827382838293830383138323833383438353836383738383839384038413842384338443845384638473848384938503851385238533854385538563857385838593860386138623863386438653866386738683869387038713872387338743875387638773878387938803881388238833884388538863887388838893890389138923893389438953896389738983899390039013902390339043905390639073908390939103911391239133914391539163917391839193920392139223923392439253926392739283929393039313932393339343935393639373938393939403941394239433944394539463947394839493950395139523953395439553956395739583959396039613962396339643965396639673968396939703971397239733974397539763977397839793980398139823983398439853986398739883989399039913992399339943995399639973998399940004001400240034004400540064007400840094010401140124013401440154016401740184019402040214022402340244025402640274028402940304031403240334034403540364037403840394040404140424043404440454046404740484049405040514052405340544055405640574058405940604061406240634064406540664067406840694070407140724073407440754076407740784079408040814082408340844085408640874088408940904091409240934094409540964097409840994100410141024103410441054106410741084109411041114112411341144115411641174118411941204121412241234124412541264127412841294130413141324133413441354136413741384139414041414142414341444145414641474148414941504151415241534154415541564157415841594160416141624163416441654166416741684169417041714172417341744175417641774178417941804181418241834184418541864187418841894190419141924193419441954196419741984199420042014202420342044205420642074208420942104211421242134214421542164217421842194220422142224223422442254226422742284229423042314232423342344235423642374238423942404241424242434244424542464247424842494250425142524253425442554256425742584259426042614262426342644265426642674268426942704271427242734274427542764277427842794280428142824283428442854286428742884289429042914292429342944295429642974298429943004301430243034304430543064307430843094310431143124313431443154316431743184319432043214322432343244325432643274328432943304331433243334334433543364337433843394340434143424343434443454346434743484349435043514352435343544355435643574358435943604361436243634364436543664367436843694370437143724373437443754376437743784379438043814382438343844385438643874388438943904391439243934394439543964397439843994400
  1. ; Copyright (C) 2016 Jeremiah Orians
  2. ; This file is part of stage0.
  3. ;
  4. ; stage0 is free software: you can redistribute it and/or modify
  5. ; it under the terms of the GNU General Public License as published by
  6. ; the Free Software Foundation, either version 3 of the License, or
  7. ; (at your option) any later version.
  8. ;
  9. ; stage0 is distributed in the hope that it will be useful,
  10. ; but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. ; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. ; GNU General Public License for more details.
  13. ;
  14. ; You should have received a copy of the GNU General Public License
  15. ; along with stage0. If not, see <http://www.gnu.org/licenses/>.
  16. ;; A Minimal C Compiler
  17. ;; type Cells are in the following form:
  18. ;; NEXT (0), SIZE (4), OFFSET (8), INDIRECT (12), MEMBERS (16), TYPE (20), NAME (24)
  19. ;; token_list Cells are in the following form:
  20. ;; NEXT (0), LOCALS/PREV (4), S (8), TYPE/FILENAME (12), ARGUMENTS/DEPTH/LINENUMBER (16)
  21. ;; Each being the length of a register [32bits]
  22. ;;
  23. ;; STACK space: End of program -> 512KB (0x80000) [Could be reduced]
  24. ;; HEAP space: 512KB -> End of Memory
  25. ;; R15 is the STACK pointer
  26. ;; R14 is the HEAP pointer
  27. :start
  28. ;; Prep TAPE_02
  29. LOADUI R0 0x1101
  30. FOPEN_WRITE
  31. ;; Prep TAPE_01
  32. LOADUI R0 0x1100
  33. FOPEN_READ
  34. :main
  35. LOADUI R0 0x1100 ; Pass Tape_01 for reading
  36. LOADR32 R14 @HEAP ; Setup Initial HEAP
  37. LOADUI R15 $STACK ; Setup Initial STACK
  38. CALLI R15 @read_all_tokens ; Read all Tokens in Tape_01
  39. CALLI R15 @reverse_list ; Fix Token Order
  40. ; CALLI R15 @debug_list ; Lets try to debug token errors
  41. MOVE R13 R0 ; Set global_token for future reading
  42. FALSE R12 ; Set struct token_list* out to NULL
  43. FALSE R11 ; Set struct token_list* list_strings to NULL
  44. FALSE R10 ; Set struct token_list* globals_list to NULL
  45. CALLI R15 @program ; Build our output
  46. LOADUI R0 $header_string1 ; Using our first header string
  47. LOADUI R1 0x1101 ; Using Tape_02
  48. CALLI R15 @file_print ; Write string
  49. MOVE R0 R12 ; using Contents of output_list
  50. CALLI R15 @recursive_output ; Recursively write
  51. LOADUI R0 $header_string2 ; Using our second header string
  52. CALLI R15 @file_print ; Write string
  53. MOVE R0 R10 ; using Contents of globals_list
  54. CALLI R15 @recursive_output ; Recursively write
  55. LOADUI R0 $header_string3 ; Using our third header string
  56. CALLI R15 @file_print ; Write string
  57. MOVE R0 R11 ; using Contents of strings_list
  58. CALLI R15 @recursive_output ; Recursively write
  59. LOADUI R0 $header_string4 ; Using our final header string
  60. CALLI R15 @file_print ; Write string
  61. HALT ; We have completed compiling our input
  62. ;; Symbol lists
  63. :global_constant_list
  64. NOP
  65. :global_symbol_list
  66. NOP
  67. :global_function_list
  68. NOP
  69. ;; Pointer to initial HEAP ADDRESS
  70. :HEAP
  71. '00080000'
  72. ;; Output strings
  73. :header_string1
  74. "
  75. # Core program
  76. "
  77. :header_string2
  78. "
  79. # Program global variables
  80. "
  81. :header_string3
  82. "
  83. # Program strings
  84. "
  85. :header_string4
  86. "
  87. :ELF_end
  88. "
  89. ;; clearWhiteSpace function
  90. ;; Receives a character in R0 and FILE* in R1 and line_num in R11
  91. ;; Returns first non-whitespace character in R0
  92. :clearWhiteSpace
  93. CMPSKIPI.NE R0 32 ; Check for a Space
  94. JUMP @clearWhiteSpace_reset ; Looks like we need to remove a space
  95. CMPSKIPI.NE R0 9 ; Check for a tab
  96. JUMP @clearWhiteSpace_reset ; Looks like we need to remove a tab
  97. CMPSKIPI.E R0 10 ; Check for a newline
  98. RET R15 ; Looks we found a non-whitespace
  99. ADDUI R11 R11 1 ; Increment line number
  100. ;; Fall through to iterate to next char
  101. :clearWhiteSpace_reset
  102. FGETC ; Get next char
  103. JUMP @clearWhiteSpace ; Iterate
  104. ;; consume_byte function
  105. ;; Receives a char in R0, FILE* in R1 and index in R13
  106. ;; Returns next char in R0
  107. :consume_byte
  108. STOREX8 R0 R14 R13 ; Put char onto HEAP
  109. ADDUI R13 R13 1 ; Increment index
  110. FGETC ; Get next char
  111. RET R15
  112. ;; consume_word function
  113. ;; Receives a char in R0, FILE* in R1, FREQUENT in R2 and index in R13
  114. ;; Returns next char in R0
  115. :consume_word
  116. PUSHR R3 R15 ; Protect R3
  117. FALSE R3 ; ESCAPE is FALSE
  118. :consume_word_reset
  119. JUMP.NZ R3 @consume_word_iter1
  120. CMPSKIPI.NE R0 92 ; If \
  121. TRUE R3 ; Looks like we are in an escape
  122. JUMP @consume_word_iter2
  123. :consume_word_iter1
  124. FALSE R3 ; Looks like we are no longer in an escape
  125. :consume_word_iter2
  126. CALLI R15 @consume_byte ; Store the char
  127. JUMP.NZ R3 @consume_word_reset ; If escape loop
  128. CMPJUMPI.NE R0 R2 @consume_word_reset ; if not matching frequent loop
  129. FGETC ; Get a new char to return
  130. POPR R3 R15 ; Restore R3
  131. RET R15
  132. ;; fixup_label function
  133. ;; Receives nothing (But uses R14 as HEAP pointer)
  134. ;; Returns 32 in R0 and no other registers altered
  135. :fixup_label
  136. PUSHR R1 R15 ; Protect R1 from change
  137. PUSHR R2 R15 ; Protect R2 from change
  138. LOADUI R0 58 ; Set HOLD to :
  139. FALSE R2 ; Set I to 0
  140. :fixup_label_reset
  141. MOVE R1 R0 ; Set PREV = HOLD
  142. LOADXU8 R0 R14 R2 ; Read hold_string[I] into HOLD
  143. STOREX8 R1 R14 R2 ; Set hold_string[I] = PREV
  144. ADDUI R2 R2 1 ; increment I
  145. JUMP.NZ R0 @fixup_label_reset ; Loop until we hit a NULL
  146. ;; clean up
  147. ADDUI R2 R2 1 ; increment I
  148. LOADUI R0 32 ; Put 32 in R0
  149. POPR R2 R15 ; Restore R2
  150. POPR R1 R15 ; Restore R1
  151. RET R15
  152. ;; in_set2 function
  153. ;; Receives a Char in R0, FILE* in R1, char* in R2 and index in R13
  154. ;; Return result in R2
  155. :in_set2
  156. PUSHR R3 R15 ; Protect R3 from changes
  157. :in_set2_reset
  158. LOADU8 R3 R2 0 ; Get char from list
  159. JUMP.Z R3 @in_set2_fail ; Stop when 0 == s[0]
  160. CMPJUMPI.E R0 R3 @in_set2_done ; We found a match
  161. ADDUI R2 R2 1 ; Increment to next char
  162. JUMP.NZ R3 @in_set2_reset ; Iterate if not NULL
  163. :in_set2_fail
  164. ;; Looks like not found
  165. FALSE R2 ; Return FALSE
  166. :in_set2_done
  167. CMPSKIPI.E R2 0 ; Provided not FALSE
  168. TRUE R2 ; The result is true
  169. POPR R3 R15 ; Restore R3
  170. RET R15
  171. ;; in_set function
  172. ;; Receives a Char in R0, char* in R1
  173. ;; Return result in R0
  174. :in_set
  175. PUSHR R2 R15 ; Protect R3 from changes
  176. :in_set_reset
  177. LOADU8 R2 R1 0 ; Get char from list
  178. JUMP.Z R2 @in_set_fail ; Stop when 0 == s[0]
  179. CMPJUMPI.E R0 R2 @in_set_done ; We found a match
  180. ADDUI R1 R1 1 ; Increment to next char
  181. JUMP.NZ R2 @in_set_reset ; Iterate if not NULL
  182. :in_set_fail
  183. ;; Looks like not found
  184. FALSE R1 ; Return FALSE
  185. :in_set_done
  186. CMPSKIPI.E R1 0 ; Provided not FALSE
  187. TRUE R2 ; The result is true
  188. MOVE R0 R2 ; Put result in correct place
  189. POPR R2 R15 ; Restore R3
  190. RET R15
  191. ;; Common in_set strings of interest
  192. ;; As Raw strings (") is forbidden and ' has some restrictions
  193. :nice_chars
  194. "
  195. !#$%&'()*+,-./0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\]^_`abcdefghijklmnopqrstuvwxyz{|}~"
  196. :keyword_chars
  197. "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789_"
  198. :variable_chars
  199. "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ_"
  200. :symbol_chars
  201. "<=>|&!-"
  202. :hex_chars
  203. "0123456789ABCDEF"
  204. :digit_chars
  205. "0123456789"
  206. :whitespace_chars
  207. "
  208. "
  209. ;; preserve_keyword function
  210. ;; Receives a Char in R0, FILE* in R1 and index in R13
  211. ;; Overwrites R2
  212. ;; Returns next CHAR
  213. :preserve_keyword
  214. LOADUI R2 $keyword_chars ; Using keyword list of chars
  215. CALLI R15 @in_set2 ; Check if in list
  216. JUMP.Z R2 @preserve_keyword_label ; if not in set, stop iterating
  217. :preserve_keyword_reset
  218. CALLI R15 @consume_byte ; Consume another byte
  219. JUMP @preserve_keyword ; Iterate
  220. :preserve_keyword_label
  221. CMPSKIPI.NE R0 58 ; Check for label (:)
  222. CALLI R15 @fixup_label ; Looks like we found one
  223. RET R15
  224. ;; preserve_symbol function
  225. ;; Receives a Char in R0, FILE* in R1 and index in R13
  226. ;; Overwrites R2
  227. ;; Returns next CHAR
  228. :preserve_symbol
  229. LOADUI R2 $symbol_chars ; Using symbol list of chars
  230. CALLI R15 @in_set2 ; Check if in list
  231. JUMP.NZ R2 @preserve_symbol_reset
  232. ;; Looks we didn't find anything we wanted to preserve
  233. RET R15
  234. :preserve_symbol_reset
  235. CALLI R15 @consume_byte ; Consume another byte
  236. JUMP @preserve_symbol ; Iterate
  237. ;; purge_macro function
  238. ;; Receives a Char in R0, FILE* in R1 and index in R13
  239. ;; Returns next CHAR via jumping to get_token_reset
  240. :purge_macro
  241. CMPSKIPI.NE R0 10 ; Check for Line Feed
  242. JUMP @get_token_reset ; Looks like we found it, call it done
  243. FGETC ; Looks like we need another CHAR
  244. JUMP @purge_macro ; Keep looping
  245. ;; get_token function
  246. ;; Receives a Char in R0, FILE* in R1, line_num in R11 and TOKEN in R10
  247. ;; sets index in R13 and current in R12
  248. ;; Overwrites R2
  249. ;; Returns next CHAR
  250. :get_token
  251. PUSHR R12 R15 ; Preserve R12
  252. PUSHR R13 R15 ; Preserve R13
  253. COPY R12 R14 ; Save CURRENT's Address
  254. ADDUI R14 R14 20 ; Update Malloc to free space for string
  255. :get_token_reset
  256. FALSE R13 ; Reset string_index to 0
  257. CALLI R15 @clearWhiteSpace ; Clear any leading whitespace
  258. CMPSKIPI.NE R0 35 ; Deal with # line macros
  259. JUMP @purge_macro ; Returns at get_token_reset
  260. ;; Check for keywords
  261. LOADUI R2 $keyword_chars ; Using keyword list
  262. CALLI R15 @in_set2 ; Check if keyword
  263. JUMP.Z R2 @get_token_symbol ; if not a keyword
  264. CALLI R15 @preserve_keyword ; Yep its a keyword
  265. JUMP @get_token_done ; Be done with token
  266. ;; Check for symbols
  267. :get_token_symbol
  268. LOADUI R2 $symbol_chars ; Using symbol list
  269. CALLI R15 @in_set2 ; Check if symbol
  270. JUMP.Z R2 @get_token_char ; If not a symbol
  271. CALLI R15 @preserve_symbol ; Yep its a symbol
  272. JUMP @get_token_done ; Be done with token
  273. ;; Check for char
  274. :get_token_char
  275. CMPSKIPI.E R0 39 ; Check if '
  276. JUMP @get_token_string ; Not a '
  277. COPY R2 R0 ; Prepare for consume_word
  278. CALLI R15 @consume_word ; Call it
  279. JUMP @get_token_done ; Be done with token
  280. ;; Check for string
  281. :get_token_string
  282. CMPSKIPI.E R0 34 ; Check if "
  283. JUMP @get_token_EOF ; Not a "
  284. COPY R2 R0 ; Prepare for consume_word
  285. CALLI R15 @consume_word ; Call it
  286. JUMP @get_token_done ; Be done with token
  287. ;; Check for EOF
  288. :get_token_EOF
  289. CMPSKIPI.L R0 0 ; If c < 0
  290. JUMP @get_token_comment ; If not EOF
  291. POPR R13 R15 ; Restore R13
  292. POPR R12 R15 ; Restore R12
  293. RET R15 ; Otherwise just return the EOF
  294. ;; Check for C comments
  295. :get_token_comment
  296. CMPSKIPI.E R0 47 ; Deal with non-comments
  297. JUMP @get_token_else ; immediately
  298. CALLI R15 @consume_byte ; Deal with another byte
  299. CMPSKIPI.NE R0 42 ; if * make it a block comment
  300. JUMP @get_token_comment_block ; and purge it all
  301. CMPSKIPI.E R0 47 ; Check if not //
  302. JUMP @get_token_done ; Finish off the token
  303. ;; Looks like it was //
  304. FGETC ; Get next char
  305. JUMP @get_token_reset ; Try again
  306. ;; Deal with the mess that is C block comments
  307. :get_token_comment_block
  308. FGETC ; Get next char
  309. :get_token_comment_block_outer
  310. CMPSKIPI.NE R0 47 ; Check for closing /
  311. JUMP @get_token_comment_block_outer_done ; Yep has closing /
  312. :get_token_comment_block_inner
  313. CMPSKIPI.NE R0 42 ; Check for preclosing *
  314. JUMP @get_token_comment_block_inner_done ; Yep has *
  315. ;; Otherwise we are just consuming
  316. FGETC ; Remove another CHAR
  317. CMPSKIPI.NE R0 10 ; Check for Line Feed
  318. ADDUI R11 R11 1 ; Found one, updating line number
  319. JUMP @get_token_comment_block_inner
  320. :get_token_comment_block_inner_done
  321. FGETC ; Remove another CHAR
  322. CMPSKIPI.NE R0 10 ; Check for Line Feed
  323. ADDUI R11 R11 1 ; Found one, updating line number
  324. JUMP @get_token_comment_block_outer
  325. :get_token_comment_block_outer_done
  326. FGETC ; Remove another CHAR
  327. JUMP @get_token_reset ; And Try again
  328. ;; Deal with default case
  329. :get_token_else
  330. CALLI R15 @consume_byte ; Consume the byte and be done
  331. :get_token_done
  332. ADDUI R13 R13 2 ; Pad with NULL the string
  333. STORE32 R14 R12 8 ; Set CURRENT->S to String
  334. ADD R14 R14 R13 ; Add string length to HEAP
  335. STORE32 R10 R12 0 ; CURRENT->NEXT = TOKEN
  336. STORE32 R10 R12 4 ; CURRENT->PREV = TOKE
  337. STORE32 R11 R12 16 ; CURRENT->LINENUM = LINE_NUM
  338. MOVE R10 R12 ; SET TOKEN to CURRENT
  339. POPR R13 R15 ; Restore R13
  340. POPR R12 R15 ; Restore R12
  341. RET R15
  342. ;; reverse_list function
  343. ;; Receives a Token_list in R0
  344. ;; Returns List in Reverse order in R0
  345. :reverse_list
  346. PUSHR R1 R15 ; Protect R1
  347. PUSHR R2 R15 ; Protect R2
  348. FALSE R1 ; Set ROOT to NULL
  349. CMPJUMPI.E R0 R1 @reverse_list_done ; ABORT if given a NULL
  350. :reverse_list_reset
  351. LOAD32 R2 R0 0 ; SET next to HEAD->NEXT
  352. STORE32 R1 R0 0 ; SET HEAD->NEXT to ROOT
  353. MOVE R1 R0 ; SET ROOT to HEAD
  354. MOVE R0 R2 ; SET HEAD to NEXT
  355. JUMP.NZ R0 @reverse_list_reset ; Iterate if HEAD not NULL
  356. :reverse_list_done
  357. MOVE R0 R1 ; SET Result to ROOT
  358. POPR R2 R15 ; Restore R2
  359. POPR R1 R15 ; Restore R1
  360. RET R15
  361. ;; read_all_tokens function
  362. ;; Receives a FILE* in R0
  363. ;; sets line_num in R11 and TOKEN in R10
  364. ;; Overwrites R2
  365. ;; Returns struct token_list* in R0
  366. :read_all_tokens
  367. PUSHR R1 R15 ; Protect R1
  368. PUSHR R2 R15 ; Protect R2
  369. PUSHR R10 R15 ; Protect R10
  370. PUSHR R11 R15 ; Protect R11
  371. MOVE R1 R0 ; Set R1 as FILE*
  372. FGETC ; Read our first CHAR
  373. LOADUI R11 1 ; Start line_num at 1
  374. FALSE R10 ; First token is NULL
  375. :read_all_tokens_reset
  376. JUMP.NP R0 @read_all_tokens_done
  377. CALLI R15 @get_token
  378. JUMP @read_all_tokens_reset
  379. :read_all_tokens_done
  380. MOVE R0 R10 ; Return the Token
  381. POPR R11 R15 ; Restore R11
  382. POPR R10 R15 ; Restore R10
  383. POPR R2 R15 ; Restore R2
  384. POPR R1 R15 ; Restore R1
  385. RET R15
  386. ;; parse_string function
  387. ;; Receives char* string in R0
  388. ;; R14 is HEAP Pointer
  389. ;; Returns char* in R0
  390. :parse_string
  391. PUSHR R1 R15 ; Protect R1
  392. COPY R1 R0 ; Make a copy of STRING
  393. CALLI R15 @weird ; Check if string is weird
  394. SWAP R0 R1
  395. JUMP.Z R1 @parse_string_regular ; Deal with regular strings
  396. ;; Looks like we have a weirdo
  397. CALLI R15 @collect_weird_string ; Create our weird string
  398. JUMP @parse_string_done ; Simply return what was created
  399. :parse_string_regular
  400. CALLI R15 @collect_regular_string
  401. :parse_string_done
  402. POPR R1 R15 ; Restore R1
  403. RET R15
  404. ;; weird function
  405. ;; Analyze string to determine if it's output would be weird for mescc-tools
  406. ;; Receives char* in R0
  407. ;; Returns BOOL in R0
  408. :weird
  409. PUSHR R1 R15 ; Protect R1
  410. PUSHR R2 R15 ; Protect R2
  411. PUSHR R3 R15 ; Protect R3
  412. PUSHR R4 R15 ; Protect R4
  413. FALSE R2 ; Assume FALSE
  414. ADDUI R3 R0 1 ; STRING = STRING + 1
  415. :weird_iter
  416. JUMP.NZ R2 @weird_done ; Stop if TRUE
  417. LOADU8 R4 R3 0 ; C = STRING[0]
  418. JUMP.Z R4 @weird_done ; Be done at NULL Termination
  419. CMPSKIPI.E R4 92 ; If not '\\'
  420. JUMP @weird_post_escape ; Looks like no escape analysis
  421. ;; Deal with the mess
  422. COPY R0 R3 ; Using STRING
  423. CALLI R15 @escape_lookup ; Get our CHAR
  424. MOVE R4 R0 ; C = ESCAPE_LOOKUP(STRING)
  425. LOADU8 R0 R3 1 ; STRING[1]
  426. CMPSKIPI.NE R0 120 ; if 'x' == STRING[1]
  427. ADDUI R3 R3 2 ; STRING = STRING + 2
  428. ADDUI R3 R3 1 ; STRING = STRING + 1
  429. :weird_post_escape
  430. LOADUI R1 $nice_chars ; using list of nice CHARS
  431. COPY R0 R4 ; using copy of C
  432. CALLI R15 @in_set ; Use in_set
  433. CMPSKIPI.NE R0 0 ; IF TRUE
  434. TRUE R2 ; Return TRUE
  435. ADDUI R3 R3 1 ; STRING = STRING + 1
  436. LOADUI R1 $whitespace_chars ; Check Whitespace Chars
  437. COPY R0 R4 ; Using copy of C
  438. CALLI R15 @in_set ; Use in_set
  439. JUMP.Z R0 @weird_iter ; If False simply loop
  440. LOADU8 R0 R3 0 ; STRING[1]
  441. CMPSKIPI.NE R0 58 ; If ':' == STRING[1]
  442. TRUE R2 ; Flip flag
  443. JUMP @weird_iter ; Keep trying to find an answer
  444. :weird_done
  445. MOVE R0 R2 ; Whatever is in R2 is the answer
  446. POPR R4 R15 ; Restore R4
  447. POPR R3 R15 ; Restore R3
  448. POPR R2 R15 ; Restore R2
  449. POPR R1 R15 ; Restore R1
  450. RET R15
  451. ;; collect_weird_string function
  452. ;; Converts weird string into a form mescc-tools can handle cleanly
  453. ;; Receives char* in R0
  454. ;; R14 is HEAP Pointer and $hex_chars as the table
  455. ;; Returns char* in R0
  456. :collect_weird_string
  457. PUSHR R1 R15 ; Protect R1
  458. PUSHR R2 R15 ; Protect R2
  459. PUSHR R3 R15 ; Protect R3
  460. PUSHR R4 R15 ; Protect R4
  461. LOADUI R4 $hex_chars ; Pointer to TABLE
  462. COPY R3 R14 ; Get HOLD
  463. MOVE R2 R0 ; Put STRING in Place
  464. LOADUI R0 39 ; Prefix with '
  465. PUSH8 R0 R3 ; HOLD[0] = '\'' && HOLD = HOLD + 1
  466. :collect_weird_string_iter
  467. ADDUI R2 R2 1 ; STRING = STRING + 1
  468. LOADUI R0 32 ; Insert ' '
  469. PUSH8 R0 R3 ; HOLD[0] = ' ' && HOLD = HOLD + 1
  470. COPY R0 R2 ; copy STRING
  471. CALLI R15 @escape_lookup ; Get char value
  472. ANDI R1 R0 0x0F ; Save Bottom out of the way
  473. SR0I R0 4 ; Isolate Top
  474. LOADXU8 R0 R4 R0 ; Using Table
  475. LOADXU8 R1 R4 R1 ; Using Table
  476. PUSH8 R0 R3 ; HOLD[0] = TABLE[(TEMP >> 4)] && HOLD = HOLD + 1
  477. PUSH8 R1 R3 ; HOLD[0] = TABLE[(TEMP & 15)] && HOLD = HOLD + 1
  478. LOADU8 R0 R2 0 ; STRING[0]
  479. JUMP.Z R0 @collect_weird_string_done ; Stop if NULL
  480. CMPSKIPI.E R0 92 ; IF STRING[0] != '\\'
  481. JUMP @collect_weird_string_check ; Deal with iteration
  482. LOADU8 R0 R2 1 ; STRING[1]
  483. CMPSKIPI.NE R0 120 ; If STRING[1] == 'x'
  484. ADDUI R2 R2 2 ; STRING = STRING + 2
  485. ADDUI R2 R2 1 ; STRING = STRING + 1
  486. :collect_weird_string_check
  487. LOADU8 R0 R2 1 ; STRING[1]
  488. JUMP.NZ R0 @collect_weird_string_iter
  489. :collect_weird_string_done
  490. LOADUI R0 32 ; Insert ' '
  491. PUSH8 R0 R3 ; HOLD[0] = ' ' && HOLD = HOLD + 1
  492. LOADUI R0 48 ; Insert '0'
  493. PUSH8 R0 R3 ; HOLD[0] = '0' && HOLD = HOLD + 1
  494. PUSH8 R0 R3 ; HOLD[0] = '0' && HOLD = HOLD + 1
  495. LOADUI R0 39 ; Insert '\''
  496. PUSH8 R0 R3 ; HOLD[0] = '\'' && HOLD = HOLD + 1
  497. LOADUI R0 10 ; Insert '\n'
  498. PUSH8 R0 R3 ; HOLD[0] = '\n' && HOLD = HOLD + 1
  499. ADDUI R3 R3 1 ; NULL Terminate
  500. SWAP R3 R14 ; CALLOC HOLD
  501. MOVE R0 R3 ; Return HOLD
  502. POPR R4 R15 ; Restore R4
  503. POPR R3 R15 ; Restore R3
  504. POPR R2 R15 ; Restore R2
  505. POPR R1 R15 ; Restore R1
  506. RET R15
  507. ;; hex function
  508. ;; Receives Char in R0
  509. ;; Return Int in R0
  510. :hex
  511. SUBUI R0 R0 48 ; First shift
  512. CMPSKIPI.GE R0 10 ; If 0-9
  513. RET R15 ; Be done
  514. ;; Deal with A-F
  515. ANDI R0 R0 0xDF ; Unset high bit
  516. SUBUI R0 R0 7 ; Shift them down
  517. CMPSKIPI.GE R0 10 ; if between 9 and A
  518. JUMP @hex_error ; Throw an error
  519. CMPSKIPI.L R0 16 ; if > F
  520. JUMP @hex_error ; Throw an error
  521. RET R15
  522. :hex_error
  523. LOADUI R0 $hex_error_message ; Our message
  524. FALSE R1 ; For human
  525. CALLI R15 @file_print ; write it
  526. CALLI R15 @line_error ; More info
  527. HALT
  528. :hex_error_message
  529. "Tried to print non-hex number
  530. "
  531. ;; escape_lookup function
  532. ;; Receives char* in R0
  533. ;; Returns char in R0
  534. :escape_lookup
  535. PUSHR R1 R15 ; Protect R1
  536. PUSHR R2 R15 ; Protect R2
  537. MOVE R1 R0 ; Put C in the right spot
  538. FALSE R2 ; Our flag for done
  539. LOADU8 R0 R1 0 ; c[0]
  540. CMPSKIPI.E R0 92 ; If C[0] != '\\'
  541. JUMP @escape_lookup_none ; Deal with none case
  542. LOADU8 R0 R1 1 ; c[1]
  543. CMPSKIPI.NE R0 120 ; if \x??
  544. JUMP @escape_lookup_hex
  545. ;; Deal with \? escapes
  546. CMPSKIPI.NE R0 110 ; If \n
  547. LOADUI R2 10 ; return \n
  548. CMPSKIPI.NE R0 116 ; If \t
  549. LOADUI R2 9 ; return \t
  550. CMPSKIPI.NE R0 92 ; If \\
  551. LOADUI R2 92 ; return \\
  552. CMPSKIPI.NE R0 39 ; If \'
  553. LOADUI R2 39 ; return \'
  554. CMPSKIPI.NE R0 34 ; If \"
  555. LOADUI R2 34 ; return \"
  556. CMPSKIPI.NE R0 114 ; If \r
  557. LOADUI R2 13 ; return \r
  558. JUMP.Z R2 @escape_lookup_error ; Looks like we got something weird
  559. JUMP @escape_lookup_done ; Otherwise just use our R2
  560. :escape_lookup_none
  561. MOVE R2 R0 ; We just return the char at C[0]
  562. JUMP @escape_lookup_done ; Be done
  563. :escape_lookup_hex
  564. LOADU8 R0 R1 2 ; c[2]
  565. CALLI R15 @hex ; Get first char
  566. SL0I R0 4 ; Shift our first nybble
  567. MOVE R2 R0 ; Protect our top nybble
  568. LOADU8 R0 R1 3 ; c[3]
  569. CALLI R15 @hex ; Get second char
  570. ADD R2 R2 R0 ; \x?? => ? << 4 + ?
  571. :escape_lookup_done
  572. MOVE R0 R2 ; R2 has our answer
  573. POPR R2 R15 ; Restore R2
  574. POPR R1 R15 ; Restore R1
  575. RET R15
  576. :escape_lookup_error
  577. MOVE R2 R0 ; Protect Char that failed
  578. LOADUI R0 $escape_lookup_string0 ; Load message
  579. FALSE R1 ; We want the User to see
  580. CALLI R15 @file_print ; Write it
  581. MOVE R0 R2 ; Our CHAR
  582. FPUTC ; Write it
  583. LOADUI R0 10 ; '\n'
  584. FPUTC ; Write it
  585. CALLI R15 @line_error ; Provide some debug information
  586. HALT
  587. :escape_lookup_string0
  588. "Received invalid escape \\"
  589. ;; collect_regular_string function
  590. ;; Converts C string into a RAW string for mescc-tools
  591. ;; Receives char* in R0
  592. ;; R14 is HEAP Pointer
  593. ;; Returns char* in R0
  594. :collect_regular_string
  595. PUSHR R1 R15 ; Protect R1
  596. PUSHR R2 R15 ; Protect R2
  597. COPY R2 R14 ; MESSAGE
  598. MOVE R1 R0 ; Put STRING in the right place
  599. :collect_regular_string_iter
  600. LOADU8 R0 R1 0 ; STRING[0]
  601. JUMP.Z R0 @collect_regular_string_done ; End at NULL
  602. CMPSKIPI.NE R0 92 ; if STRING[0] == '\\'
  603. JUMP @collect_regular_string_escape ; deal with escapes
  604. ;; Deal with vannilla chars
  605. STORE8 R0 R2 0 ; MESSAGE[0] = STRING[0]
  606. ADDUI R2 R2 1 ; MESSAGE = MESSAGE + 1
  607. ADDUI R1 R1 1 ; STRING = STRING + 1
  608. JUMP @collect_regular_string_iter ; Loop
  609. :collect_regular_string_escape
  610. COPY R0 R1 ; Prepare for call
  611. CALLI R15 @escape_lookup ; Get what weird char we need
  612. STORE8 R0 R2 0 ; MESSAGE[0] = escape_lookup(string)
  613. ADDUI R2 R2 1 ; MESSAGE = MESSAGE + 1
  614. LOADU8 R0 R1 1 ; STRING[1]
  615. CMPSKIPI.NE R0 120 ; if \x??
  616. ADDUI R1 R1 2 ; STRING = STRING + 2
  617. ADDUI R1 R1 2 ; STRING = STRING + 2
  618. JUMP @collect_regular_string_iter ; Loop
  619. :collect_regular_string_done
  620. LOADUI R0 34 ; Using "
  621. STORE8 R0 R2 0 ; MESSAGE[0] = '"'
  622. LOADUI R0 10 ; Using '\n'
  623. STORE8 R0 R2 1 ; MESSAGE[1] = "\n"
  624. ADDUI R2 R2 3 ; Add extra NULL padding
  625. SWAP R2 R14 ; Update HEAP
  626. MOVE R0 R2 ; Put MESSAGE in the right Spot
  627. POPR R2 R15 ; Restore R2
  628. POPR R1 R15 ; Restore R1
  629. RET R15
  630. ;; unary_expr_sizeof function
  631. ;; Receives nothing
  632. ;; Returns nothing
  633. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  634. :unary_expr_sizeof
  635. PUSHR R0 R15 ; Protect R0
  636. PUSHR R1 R15 ; Protect R1
  637. PUSHR R2 R15 ; Protect R2
  638. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  639. LOADUI R0 $unary_expr_sizeof_string0 ; Our first error message
  640. LOADUI R1 $open_paren ; Using "("
  641. CALLI R15 @require_match ; Ensure a match
  642. CALLI R15 @type_name ; Get type_name
  643. MOVE R2 R0 ; Protect A
  644. LOADUI R0 $unary_expr_sizeof_string1 ; Our final error message
  645. LOADUI R1 $close_paren ; Using ")"
  646. CALLI R15 @require_match ; Ensure a match
  647. LOADUI R0 $unary_expr_sizeof_string2 ; Our header
  648. CALLI R15 @emit_out ; emit it
  649. LOAD32 R0 R2 4 ; A->SIZE
  650. CALLI R15 @numerate_number ; Convert to string
  651. CALLI R15 @emit_out ; emit it
  652. LOADUI R0 $newline ; Using "\n"
  653. CALLI R15 @emit_out ; emit it
  654. POPR R2 R15 ; Restore R2
  655. POPR R1 R15 ; Restore R1
  656. POPR R0 R15 ; Restore R0
  657. RET R15
  658. :unary_expr_sizeof_string0
  659. "ERROR in unary_expr
  660. Missing (
  661. "
  662. :unary_expr_sizeof_string1
  663. "ERROR in unary_expr
  664. Missing )
  665. "
  666. :unary_expr_sizeof_string2
  667. "LOAD_IMMEDIATE_rax %"
  668. ;; constant_load function
  669. ;; Receives struct token_list* a in R0
  670. ;; Returns nothing
  671. :constant_load
  672. PUSHR R0 R15 ; Protect R0
  673. LOADUI R0 $constant_load_string0 ; Our header
  674. CALLI R15 @emit_out ; emit it
  675. POPR R0 R15 ; Restore R0
  676. LOAD32 R0 R0 16 ; A->ARGUMENTS
  677. LOAD32 R0 R0 8 ; A->ARGUMENTS->S
  678. CALLI R15 @emit_out ; emit it
  679. LOADUI R0 $newline ; Using "\n"
  680. CALLI R15 @emit_out ; emit it
  681. RET R15
  682. :constant_load_string0
  683. "LOAD_IMMEDIATE_rax %"
  684. ;; variable_load function
  685. ;; Receives struct token_list* a in R0
  686. ;; and struct token_list* current_target in R8
  687. ;; Returns Nothing
  688. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  689. :variable_load
  690. PUSHR R1 R15 ; Protect R1
  691. PUSHR R2 R15 ; Protect R2
  692. MOVE R2 R0 ; Protect A
  693. ;; Check if function call
  694. LOADUI R0 $type_function_name ; Using "FUNCTION"
  695. LOAD32 R1 R2 12 ; A->TYPE
  696. LOAD32 R1 R1 24 ; A->TYPE->NAME
  697. CALLI R15 @match ; IF "FUNCTION" == A->TYPE->NAME
  698. JUMP.Z R0 @variable_load_regular ; Nope
  699. LOADUI R0 $open_paren ; Using "("
  700. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  701. CALLI R15 @match ; IF "(" == GLOBAL_TOKEN->S
  702. JUMP.Z R0 @variable_load_regular ; Nope
  703. ;; Deal with function call
  704. LOAD32 R0 R2 16 ; A->DEPTH
  705. CALLI R15 @numerate_number ; Convert to string
  706. TRUE R1 ; Passing TRUE
  707. CALLI R15 @function_call ; DO IT
  708. JUMP @variable_load_done ; Be done
  709. :variable_load_regular
  710. LOAD32 R8 R2 12 ; CURRENT_TARGET = A->TYPE
  711. LOADUI R0 $variable_load_string0 ; Our prefix
  712. CALLI R15 @emit_out ; emit it
  713. LOAD32 R0 R2 16 ; A->DEPTH
  714. CALLI R15 @numerate_number ; Convert to string
  715. CALLI R15 @emit_out ; emit it
  716. LOADUI R0 $newline ; Using "\n"
  717. CALLI R15 @emit_out ; emit it
  718. ;; check for special case 1
  719. LOADUI R0 $equal ; Using "="
  720. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  721. CALLI R15 @match ; IF GLOBAL_TOKEN->S == "="
  722. JUMP.NZ R0 @variable_load_done ; Be done
  723. ;; deal with the general case
  724. LOADUI R0 $variable_load_string1 ; Our postfix
  725. CALLI R15 @emit_out ; emit it
  726. :variable_load_done
  727. POPR R2 R15 ; Restore R2
  728. POPR R1 R15 ; Restore R1
  729. RET R15
  730. :variable_load_string0
  731. "LOAD_BASE_ADDRESS_rax %"
  732. :variable_load_string1
  733. "LOAD_INTEGER
  734. "
  735. ;; function_load function
  736. ;; Receives struct token_list* a in R0
  737. ;; Returns nothing
  738. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  739. :function_load
  740. PUSHR R1 R15 ; Protect R1
  741. LOAD32 R0 R0 8 ; A->S
  742. PUSHR R0 R15 ; Protect A->S
  743. LOADUI R0 $open_paren ; Using "("
  744. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  745. CALLI R15 @match ; If GLOBAL_TOKEN->S == "("
  746. JUMP.Z R0 @function_load_regular ; If not do the simple thing
  747. ;; Deal iwth function call
  748. POPR R0 R15 ; Restore A->S
  749. FALSE R1 ; FALSE
  750. CALLI R15 @function_call ; Do the function call
  751. JUMP @function_load_done ; Clean up
  752. :function_load_regular
  753. LOADUI R0 $function_load_string0 ; Using our header string
  754. CALLI R15 @emit_out ; emit it
  755. POPR R0 R15 ; Restore A->S
  756. CALLI R15 @emit_out ; emit it
  757. LOADUI R0 $newline ; Using "\n"
  758. CALLI R15 @emit_out ; emit it
  759. :function_load_done
  760. POPR R1 R15 ; Restore R1
  761. RET R15
  762. :function_load_string0
  763. "LOAD_IMMEDIATE_rax &FUNCTION_"
  764. ;; global_load function
  765. ;; Receives struct token_list* a in R0
  766. ;; and struct token_list* current_target in R8
  767. ;; Returns nothing
  768. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  769. :global_load
  770. PUSHR R0 R15 ; Protect A
  771. LOAD32 R8 R0 12 ; CURRENT_TARGET = A->TYPE
  772. LOADUI R0 $global_load_string0 ; Our header string
  773. CALLI R15 @emit_out ; emit it
  774. POPR R0 R15 ; Restore A
  775. LOAD32 R0 R0 8 ; A->S
  776. CALLI R15 @emit_out ; emit it
  777. LOADUI R0 $newline ; Using "\n"
  778. CALLI R15 @emit_out ; emit it
  779. PUSHR R1 R15 ; Protect R1
  780. LOADUI R0 $equal ; Using "="
  781. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  782. CALLI R15 @match ; IF GLOBAL_TOKEN->S == "="
  783. JUMP.NZ R0 @global_load_done ; Skip the following
  784. ;; Deal with non-assignment
  785. LOADUI R0 $global_load_string1 ; Our footer string
  786. CALLI R15 @emit_out ; emit it
  787. :global_load_done
  788. POPR R1 R15 ; Restore R1
  789. RET R15
  790. :global_load_string0
  791. "LOAD_IMMEDIATE_rax &GLOBAL_"
  792. :global_load_string1
  793. "LOAD_INTEGER
  794. "
  795. ;; primary_expr_failure function
  796. ;; Fails hard and fast
  797. ;; Receives nothing
  798. ;; HALTs and will trash registers
  799. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  800. :primary_expr_failure
  801. LOADUI R0 $primary_expr_failure_string0 ; Our first string
  802. FALSE R1 ; Display to User
  803. CALLI R15 @file_print ; Print it
  804. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  805. CALLI R15 @file_print ; Print it
  806. LOADUI R0 $primary_expr_failure_string1 ; Our last string
  807. CALLI R15 @file_print ; Print it
  808. CALLI R15 @line_error ; Make it a line error message too
  809. HALT
  810. :primary_expr_failure_string0
  811. "Received "
  812. :primary_expr_failure_string1
  813. " in primary_expr
  814. "
  815. ;; primary_expr_string function
  816. ;; Receives struct token_list* global_token in R13,
  817. ;; struct token_list* out in R12,
  818. ;; struct token_list* string_list in R11
  819. ;; struct token_list* global_list in R10
  820. ;; and struct token_list* FUNC in R9
  821. ;; and struct token_list* current_target in R8
  822. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  823. ;; Returns the token_lists modified
  824. :primary_expr_string
  825. PUSHR R0 R15 ; Protect R0
  826. PUSHR R1 R15 ; Protect R1
  827. PUSHR R2 R15 ; Protect R2
  828. LOADR32 R0 @current_count ; Using CURRENT_COUNT
  829. ADDUI R1 R0 1 ; CURRENT_COUNT = CURRENT_COUNT + 1
  830. STORER32 R1 @current_count ; Update CURRENT_COUNT
  831. CALLI R15 @numerate_number ; Convert to string
  832. MOVE R2 R0 ; Put string in safe place
  833. LOADUI R0 $primary_expr_string_string0 ; Our string prefix
  834. CALLI R15 @emit_out ; emit it
  835. LOAD32 R0 R9 8 ; FUNCTION->S
  836. COPY R1 R2 ; NUMBER_STRING
  837. CALLI R15 @uniqueID_out ; Make it unique
  838. ;; The target
  839. LOADUI R0 $primary_expr_string_string1
  840. COPY R1 R11 ; Using STRINGS_LIST
  841. CALLI R15 @emit ; emit it
  842. MOVE R1 R0 ; Put STRINGS_LIST in correct place
  843. LOAD32 R0 R9 8 ; Using FUNCTION->S
  844. CALLI R15 @uniqueID ; Make it unique
  845. MOVE R11 R0 ; Update STRINGS_LIST
  846. ;; Parse the string
  847. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  848. CALLI R15 @parse_string ; Parse it
  849. COPY R1 R11 ; Using STRINGS_LIST
  850. CALLI R15 @emit ; emit it
  851. MOVE R11 R0 ; Update STRINGS_LIST
  852. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  853. POPR R2 R15 ; Restore R2
  854. POPR R1 R15 ; Restore R1
  855. POPR R0 R15 ; Restore R0
  856. RET R15
  857. :primary_expr_string_string0
  858. "LOAD_IMMEDIATE_rax &STRING_"
  859. :primary_expr_string_string1
  860. ":STRING_"
  861. ;; primary_expr_char function
  862. ;; Receives struct token_list* global_token in R13,
  863. ;; struct token_list* out in R12,
  864. ;; struct token_list* string_list in R11
  865. ;; struct token_list* global_list in R10
  866. ;; and struct token_list* FUNC in R9
  867. ;; and struct token_list* current_target in R8
  868. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  869. ;; Returns the token_lists modified
  870. :primary_expr_char
  871. PUSHR R0 R15 ; Protect R0
  872. LOADUI R0 $primary_expr_char_string0 ; Using our header string
  873. CALLI R15 @emit_out ; emit it
  874. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  875. ADDUI R0 R0 1 ; GLOBAL_TOKEN->S + 1
  876. CALLI R15 @escape_lookup ; escape_lookup value
  877. CALLI R15 @numerate_number ; Make it a string
  878. CALLI R15 @emit_out ; emit it
  879. LOADUI R0 $newline ; Using "\n"
  880. CALLI R15 @emit_out ; emit it
  881. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  882. POPR R0 R15 ; Restore R0
  883. RET R15
  884. :primary_expr_char_string0
  885. "LOAD_IMMEDIATE_rax %"
  886. ;; primary_expr_number function
  887. ;; struct token_list* out in R12,
  888. ;; struct token_list* string_list in R11
  889. ;; struct token_list* global_list in R10
  890. ;; and struct token_list* FUNC in R9
  891. ;; and struct token_list* current_target in R8
  892. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  893. ;; Returns the token_lists modified
  894. :primary_expr_number
  895. LOADUI R0 $primary_expr_number_string0 ; Our header
  896. CALLI R15 @emit_out ; emit it
  897. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  898. CALLI R15 @emit_out ; emit it
  899. LOADUI R0 $newline ; Using "\n"
  900. CALLI R15 @emit_out ; emit it
  901. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  902. RET R15
  903. :primary_expr_number_string0
  904. "LOAD_IMMEDIATE_rax %"
  905. ;; primary_expr_variable function
  906. ;; struct token_list* out in R12,
  907. ;; struct token_list* string_list in R11
  908. ;; struct token_list* global_list in R10
  909. ;; and struct token_list* FUNC in R9
  910. ;; and struct token_list* current_target in R8
  911. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  912. ;; Returns the token_lists modified
  913. :primary_expr_variable
  914. PUSHR R0 R15 ; Protect R0
  915. PUSHR R1 R15 ; Protect R1
  916. PUSHR R2 R15 ; Protect R2
  917. LOAD32 R2 R13 8 ; S = GLOBAL_TOKEN->S
  918. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  919. COPY R0 R2 ; Using S
  920. LOADR32 R1 @global_constant_list
  921. CALLI R15 @sym_lookup ; Lookup S in CONSTANTS
  922. JUMP.Z R0 @primary_expr_variable_locals ; try Locals
  923. ;; Deal with Constants
  924. CALLI R15 @constant_load ; A is in R0 already
  925. JUMP @primary_expr_variable_done ; Moving on
  926. :primary_expr_variable_locals
  927. COPY R0 R2 ; Using S
  928. LOAD32 R1 R9 4 ; Using FUNCTION->LOCALS
  929. CALLI R15 @sym_lookup ; Lookup S in Locals
  930. JUMP.Z R0 @primary_expr_variable_arguments ; try arguments
  931. ;; Deal with Locals
  932. CALLI R15 @variable_load ; A is in R0 already
  933. JUMP @primary_expr_variable_done ; Moving on
  934. :primary_expr_variable_arguments
  935. COPY R0 R2 ; Using S
  936. LOAD32 R1 R9 16 ; Using FUNCTION->ARGUMENTS
  937. CALLI R15 @sym_lookup ; Lookup S in arguments
  938. JUMP.Z R0 @primary_expr_variable_function ; try Functions
  939. ;; Deal with argument
  940. CALLI R15 @variable_load ; A is in R0 already
  941. JUMP @primary_expr_variable_done ; Moving on
  942. :primary_expr_variable_function
  943. COPY R0 R2 ; Using S
  944. LOADR32 R1 @global_function_list ; Get current GLOBAL_FUNCTION_LIST
  945. CALLI R15 @sym_lookup ; Lookup S in GLOBAL_FUNCTION_LIST
  946. JUMP.Z R0 @primary_expr_variable_global ; try Globals
  947. ;; Deal with function
  948. CALLI R15 @function_load ; Dothe work
  949. JUMP @primary_expr_variable_done ; Moving on
  950. :primary_expr_variable_global
  951. COPY R0 R2 ; Using S
  952. LOADR32 R1 @global_symbol_list ; Get current GLOBAL_SYMBOL_LIST
  953. CALLI R15 @sym_lookup ; Lookup S in GLOBAL_SYMBOL_LIST
  954. JUMP.Z R0 @primary_expr_variable_failure ; Looks like it isn't anything we know
  955. ;; Deal with a global
  956. CALLI R15 @global_load
  957. :primary_expr_variable_done
  958. POPR R2 R15 ; Restore R2
  959. POPR R1 R15 ; Restore R1
  960. POPR R0 R15 ; Restore R0
  961. RET R15
  962. :primary_expr_variable_failure
  963. MOVE R0 R2 ; Using S
  964. FALSE R1 ; We want the user to see
  965. CALLI R15 @file_print ; Print it
  966. LOADUI R0 $primary_expr_variable_string0 ; Body
  967. CALLI R15 @file_print ; Print it
  968. CALLI R15 @line_error ; Provide useful error info
  969. HALT
  970. :primary_expr_variable_string0
  971. " is not a defined symbol
  972. "
  973. ;; promote_type function
  974. ;; Receives struct type* in R0 and struct type* in R1
  975. ;; Returns first match struct type* in R0
  976. :promote_type
  977. JUMP.Z R1 @promote_type_abort0 ; If B is NULL just abort
  978. PUSHR R1 R15 ; Protect R1
  979. SWAP R0 R1 ; Give A a try
  980. JUMP.Z R1 @promote_type_abort1 ; A is NULL just short abort
  981. ;; Looks like we have a bunch of work to do
  982. PUSHR R2 R15 ; Protect R2
  983. PUSHR R3 R15 ; Protect R3
  984. PUSHR R4 R15 ; Protect R4
  985. PUSHR R5 R15 ; Protect R5
  986. MOVE R5 R1 ; Put A in the right spot
  987. MOVE R4 R0 ; Put B in the right spot
  988. LOADR32 R3 @global_types ; I = GLOBAL_TYPES
  989. :promote_type_iter
  990. LOAD32 R1 R3 24 ; I->NAME
  991. LOAD32 R0 R5 24 ; A->NAME
  992. CMPJUMPI.E R0 R1 @promote_type_done ; break
  993. LOAD32 R0 R4 24 ; B->NAME
  994. CMPJUMPI.E R0 R1 @promote_type_done ; break
  995. LOAD32 R1 R3 12 ; I->INDIRECT
  996. LOAD32 R1 R1 24 ; I->INDIRECT->NAME
  997. LOAD32 R0 R5 24 ; A->NAME
  998. CMPJUMPI.E R0 R1 @promote_type_done ; break
  999. LOAD32 R0 R4 24 ; B->NAME
  1000. CMPJUMPI.E R0 R1 @promote_type_done ; break
  1001. LOAD32 R3 R3 0 ; I = I->NEXT
  1002. JUMP.NZ R3 @promote_type_iter ; Loop if not NULL
  1003. :promote_type_done
  1004. MOVE R0 R3 ; Return I
  1005. POPR R5 R15 ; Restore R5
  1006. POPR R4 R15 ; Restore R4
  1007. POPR R3 R15 ; Restore R3
  1008. POPR R2 R15 ; Restore R2
  1009. :promote_type_abort1
  1010. POPR R1 R15 ; Restore R1
  1011. :promote_type_abort0
  1012. RET R15
  1013. ;; common_recursion function
  1014. ;; Receives FUNCTION* in R0
  1015. ;; struct token_list* out in R12,
  1016. ;; struct token_list* string_list in R11
  1017. ;; struct token_list* global_list in R10
  1018. ;; and struct token_list* FUNC in R9
  1019. ;; and struct token_list* current_target in R8
  1020. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  1021. ;; Returns the token_lists modified
  1022. :common_recursion
  1023. PUSHR R0 R15 ; Protect R0
  1024. PUSHR R1 R15 ; Protect R1
  1025. PUSHR R2 R15 ; Protect R2
  1026. MOVE R2 R0 ; Protect F
  1027. COPY R1 R8 ; LAST_TYPE = CURRENT_TARGET
  1028. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  1029. LOADUI R0 $common_recursion_string0 ; Header string
  1030. CALLI R15 @emit_out ; Our header
  1031. CALL R2 R15 ; CALL F()
  1032. COPY R0 R8 ; Using CURRENT_TARGET
  1033. CALLI R15 @promote_type ; Promote type
  1034. MOVE R8 R0 ; update CURRENT_TARGET
  1035. LOADUI R0 $common_recursion_string1 ; Footer string
  1036. CALLI R15 @emit_out ; Our footer
  1037. POPR R2 R15 ; Restore R2
  1038. POPR R1 R15 ; Restore R1
  1039. POPR R0 R15 ; Restore R0
  1040. RET R15
  1041. :common_recursion_string0
  1042. "PUSH_RAX #_common_recursion
  1043. "
  1044. :common_recursion_string1
  1045. "POP_RBX # _common_recursion
  1046. "
  1047. ;; general_recursion function
  1048. ;; Receives FUNCTION F in R0, char* s in R1, char* name in R2
  1049. ;; and FUNCTION ITERATE in R3
  1050. ;; struct token_list* out in R12,
  1051. ;; struct token_list* string_list in R11
  1052. ;; struct token_list* global_list in R10
  1053. ;; and struct token_list* FUNC in R9
  1054. ;; and struct token_list* current_target in R8
  1055. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  1056. ;; Returns nothing
  1057. :general_recursion
  1058. PUSHR R0 R15 ; Protect R0
  1059. PUSHR R1 R15 ; Protect S
  1060. PUSHR R0 R15 ; Protect F
  1061. COPY R0 R2 ; Using NAME
  1062. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  1063. CALLI R15 @match ; IF GLOBAL_TOKEN->S == NAME
  1064. JUMP.Z R0 @general_recursion_done
  1065. ;; deal with case of match
  1066. POPR R0 R15 ; Restore F
  1067. CALLI R15 @common_recursion ; Recurse
  1068. POPR R1 R15 ; Restore S
  1069. COPY R0 R1 ; Put S in correct place
  1070. CALLI R15 @emit_out ; emit it
  1071. CALL R3 R15 ; CALL ITERATE()
  1072. POPR R0 R15 ; Restore R0
  1073. RET R15 ; Don't double pop
  1074. :general_recursion_done
  1075. POPR R0 R15 ; Restore F
  1076. POPR R1 R15 ; Restore S
  1077. POPR R0 R15 ; Restore R0
  1078. RET R15
  1079. ;; ceil_log2 function
  1080. ;; Receives INT A in R0
  1081. ;; Returns LOG2(A) in R0
  1082. :ceil_log2
  1083. PUSHR R1 R15 ; Protect R1
  1084. PUSHR R2 R15 ; Protect R2
  1085. FALSE R2 ; RESULT = 0
  1086. SUBI R1 R0 1 ; A - 1
  1087. AND R1 R1 R0 ; A & (A - 1)
  1088. CMPSKIPI.NE R1 0 ; IF (A & (A - 1)) == 0
  1089. LOADI R2 -1 ; RESULT = -1
  1090. :ceil_log2_iter
  1091. JUMP.Z R0 @ceil_log2_done ; IF A > 0
  1092. ADDI R2 R2 1 ; RESULT = RESULT + 1
  1093. SARI R0 1 ; A = A >> 1
  1094. JUMP @ceil_log2_iter ; Loop
  1095. :ceil_log2_done
  1096. MOVE R0 R2 ; Use RESULT
  1097. POPR R2 R15 ; Restore R2
  1098. POPR R1 R15 ; Restore R1
  1099. RET R15
  1100. ;; postfix_expr_arrow function
  1101. ;; Receives nothing
  1102. ;; struct token_list* out in R12,
  1103. ;; struct token_list* string_list in R11
  1104. ;; struct token_list* global_list in R10
  1105. ;; and struct token_list* FUNC in R9
  1106. ;; and struct token_list* current_target in R8
  1107. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  1108. ;; Returns the token_lists modified
  1109. :postfix_expr_arrow
  1110. PUSHR R0 R15 ; Protect R0
  1111. PUSHR R1 R15 ; Protect R1
  1112. PUSHR R2 R15 ; Protect R2
  1113. LOADUI R0 $postfix_expr_arrow_string0 ; Our header string
  1114. CALLI R15 @emit_out ; Emit it
  1115. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  1116. COPY R0 R8 ; Passing CURRENT_TARGET
  1117. LOAD32 R1 R13 8 ; Using GLOBAL_TOKEN->S
  1118. CALLI R15 @lookup_member ; Look it up
  1119. LOAD32 R2 R0 4 ; Protect I->SIZE
  1120. LOAD32 R8 R0 20 ; CURRENT_TARGET = I->TYPE
  1121. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  1122. LOAD32 R1 R0 8 ; I->OFFSET
  1123. JUMP.Z R1 @postfix_expr_arrow_offset ; If no offset needed skip the work
  1124. ;; Deal with non-zero offsets
  1125. LOADUI R0 $postfix_expr_arrow_string1 ; Our first prefix
  1126. CALLI R15 @emit_out ; Emit it
  1127. LOADUI R0 $postfix_expr_arrow_string2 ; Our second prefix
  1128. CALLI R15 @emit_out ; Emit it
  1129. MOVE R0 R1 ; Put I->OFFSET in the right place
  1130. CALLI R15 @numerate_number ; Convert to string
  1131. CALLI R15 @emit_out ; Emit it
  1132. LOADUI R0 $postfix_expr_arrow_string3 ; Our postfix
  1133. CALLI R15 @emit_out ; Emit it
  1134. :postfix_expr_arrow_offset
  1135. LOADUI R0 $equal ; Using "="
  1136. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  1137. CALLI R15 @match ; IF GLOBAL_TOKEN->S == "="
  1138. JUMP.NZ R0 @postfix_expr_arrow_done
  1139. LOADUI R0 8 ; Compare against 8
  1140. CMPJUMPI.L R0 R2 @postfix_expr_arrow_done
  1141. ;; Deal with special case
  1142. LOADUI R0 $postfix_expr_arrow_string4 ; Our final string
  1143. CALLI R15 @emit_out ; Emit it
  1144. :postfix_expr_arrow_done
  1145. POPR R2 R15 ; Restore R2
  1146. POPR R1 R15 ; Restore R1
  1147. POPR R0 R15 ; Restore R0
  1148. RET R15
  1149. :postfix_expr_arrow_string0
  1150. "# looking up offset
  1151. "
  1152. :postfix_expr_arrow_string1
  1153. "# -> offset calculation
  1154. "
  1155. :postfix_expr_arrow_string2
  1156. "LOAD_IMMEDIATE_rbx %"
  1157. :postfix_expr_arrow_string3
  1158. "
  1159. ADD_rbx_to_rax
  1160. "
  1161. :postfix_expr_arrow_string4
  1162. "LOAD_INTEGER
  1163. "
  1164. ;; postfix_expr_array function
  1165. ;; Receives nothing
  1166. ;; struct token_list* out in R12,
  1167. ;; struct token_list* string_list in R11
  1168. ;; struct token_list* global_list in R10
  1169. ;; and struct token_list* FUNC in R9
  1170. ;; and struct token_list* current_target in R8
  1171. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  1172. ;; Returns the token_lists modified
  1173. :postfix_expr_array
  1174. PUSHR R0 R15 ; Protect R0
  1175. PUSHR R1 R15 ; Protect R1
  1176. PUSHR R2 R15 ; Protect R2
  1177. COPY R2 R8 ; ARRAY = CURRENT_TARGET
  1178. LOADUI R0 $expression ; Using EXPRESSION
  1179. CALLI R15 @common_recursion ; Recurse
  1180. MOVE R8 R2 ; CURRENT_TARGET = ARRAY
  1181. LOADUI R2 $postfix_expr_array_string0 ; ASSIGN = load integer
  1182. LOADUI R0 $type_char_indirect_name ; Using "char*"
  1183. LOAD32 R1 R8 24 ; CURRENT_TARGET->NAME
  1184. CALLI R15 @match ; IF CURRENT_TARGET->NAME == "char*"
  1185. CMPSKIPI.E R0 0 ; deal with Byte
  1186. LOADUI R2 $postfix_expr_array_string1 ; ASSIGN = load byte
  1187. JUMP.NZ R0 @postfix_expr_array_byte ; Skip if Byte
  1188. ;; Deal with larger than byte
  1189. LOADUI R0 $postfix_expr_array_string2 ; Our shift
  1190. CALLI R15 @emit_out ; emit it
  1191. LOAD32 R0 R8 12 ; CURRENT_TARGET->INDIRECT
  1192. LOAD32 R0 R0 4 ; CURRENT_TARGET->INDIRECT->SIZE
  1193. CALLI R15 @ceil_log2 ; LOG2(CURRENT_TARGET->INDIRECT->SIZE)
  1194. CALLI R15 @numerate_number ; Convert to string
  1195. CALLI R15 @emit_out ; emit it
  1196. LOADUI R0 $newline ; Using "\n"
  1197. CALLI R15 @emit_out ; emit it
  1198. :postfix_expr_array_byte
  1199. LOADUI R0 $postfix_expr_array_string3 ; Add the offset
  1200. CALLI R15 @emit_out ; emit it
  1201. LOADUI R0 $postfix_expr_array_string4 ; Our final error message
  1202. LOADUI R1 $close_bracket ; Using "]"
  1203. CALLI R15 @require_match ; Ensure match
  1204. LOADUI R0 $equal ; Using "="
  1205. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  1206. CALLI R15 @match ; IF GLOBAL_TOKEN->S == "="
  1207. CMPSKIPI.E R0 0 ; If match
  1208. LOADUI R2 $postfix_expr_array_string5 ; empty string
  1209. MOVE R0 R2 ; What ever string survived
  1210. CALLI R15 @emit_out ; emit it
  1211. POPR R2 R15 ; Restore R2
  1212. POPR R1 R15 ; Restore R1
  1213. POPR R0 R15 ; Restore R0
  1214. RET R15
  1215. :postfix_expr_array_string0
  1216. "LOAD_INTEGER
  1217. "
  1218. :postfix_expr_array_string1
  1219. "LOAD_BYTE
  1220. "
  1221. :postfix_expr_array_string2
  1222. "SAL_rax_Immediate8 !"
  1223. :postfix_expr_array_string3
  1224. "ADD_rbx_to_rax
  1225. "
  1226. :postfix_expr_array_string4
  1227. "ERROR in postfix_expr
  1228. Missing ]
  1229. "
  1230. :postfix_expr_array_string5
  1231. ""
  1232. ;; postfix_expr_stub function
  1233. ;; Receives nothing
  1234. ;; struct token_list* out in R12,
  1235. ;; struct token_list* string_list in R11
  1236. ;; struct token_list* global_list in R10
  1237. ;; and struct token_list* FUNC in R9
  1238. ;; and struct token_list* current_target in R8
  1239. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  1240. ;; Returns the token_lists modified
  1241. :postfix_expr_stub
  1242. PUSHR R0 R15 ; Protect R0
  1243. PUSHR R1 R15 ; Protect R1
  1244. LOADUI R0 $open_bracket ; Using "["
  1245. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  1246. CALLI R15 @match ; IF GLOBAL_TOKEN->S == "["
  1247. JUMP.Z R0 @postfix_expr_stub_next
  1248. ;; Deal with "[" case
  1249. CALLI R15 @postfix_expr_array ; process
  1250. CALLI R15 @postfix_expr_stub ; recurse
  1251. :postfix_expr_stub_next
  1252. LOADUI R0 $arrow_string ; Using "->"
  1253. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  1254. CALLI R15 @match ; IF GLOBAL_TOKEN->S == "->"
  1255. JUMP.Z R0 @postfix_expr_stub_done ; clean up
  1256. ;; Deal with "->" case
  1257. CALLI R15 @postfix_expr_arrow ; Process
  1258. CALLI R15 @postfix_expr_stub ; recurse
  1259. :postfix_expr_stub_done
  1260. POPR R1 R15 ; Restore R1
  1261. POPR R0 R15 ; Restore R0
  1262. RET R15
  1263. ;; postfix_expr function
  1264. ;; struct token_list* out in R12,
  1265. ;; struct token_list* string_list in R11
  1266. ;; struct token_list* global_list in R10
  1267. ;; and struct token_list* FUNC in R9
  1268. ;; and struct token_list* current_target in R8
  1269. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  1270. ;; Returns the token_lists modified
  1271. :postfix_expr
  1272. CALLI R15 @primary_expr ; Walk up the tree
  1273. CALLI R15 @postfix_expr_stub ; Deal with nodes on this level
  1274. RET R15
  1275. ;; additive_expr_stub function
  1276. ;; receives nothing
  1277. ;; returns nothing
  1278. ;; Updates struct token_list*
  1279. :additive_expr_stub
  1280. PUSHR R0 R15 ; Protect R0
  1281. PUSHR R1 R15 ; Protect R1
  1282. PUSHR R2 R15 ; Protect R2
  1283. PUSHR R3 R15 ; Protect R3
  1284. ;; Fixed pieces
  1285. LOADUI R0 $postfix_expr ; Set First argument
  1286. LOADUI R3 $additive_expr_stub
  1287. ;; The + bit
  1288. LOADUI R1 $additive_expr_stub_string0 ; Our first operation
  1289. LOADUI R2 $plus_string ; Using "+"
  1290. CALLI R15 @general_recursion
  1291. ;; The - bit
  1292. LOADUI R1 $additive_expr_stub_string1 ; Our second operation
  1293. LOADUI R2 $minus_string ; Using "-"
  1294. CALLI R15 @general_recursion
  1295. ;; The * bit
  1296. LOADUI R1 $additive_expr_stub_string2 ; Our third operation
  1297. LOADUI R2 $multiply_string ; Using "*"
  1298. CALLI R15 @general_recursion
  1299. ;; The / bit
  1300. LOADUI R1 $additive_expr_stub_string3 ; Our fourth operation
  1301. LOADUI R2 $divide_string ; Using "/"
  1302. CALLI R15 @general_recursion
  1303. ;; The % bit
  1304. LOADUI R1 $additive_expr_stub_string4 ; Our fifth operation
  1305. LOADUI R2 $modulus_string ; Using "%"
  1306. CALLI R15 @general_recursion
  1307. ;; The << bit
  1308. LOADUI R1 $additive_expr_stub_string5 ; Our sixth operation
  1309. LOADUI R2 $left_shift_string ; Using "<<"
  1310. CALLI R15 @general_recursion
  1311. ;; The >> bit
  1312. LOADUI R1 $additive_expr_stub_string6 ; Our final operation
  1313. LOADUI R2 $right_shift_string ; Using ">>"
  1314. CALLI R15 @general_recursion
  1315. POPR R3 R15 ; Restore R3
  1316. POPR R2 R15 ; Restore R2
  1317. POPR R1 R15 ; Restore R1
  1318. POPR R0 R15 ; Restore R0
  1319. RET R15
  1320. :additive_expr_stub_string0
  1321. "ADD_rbx_to_rax
  1322. "
  1323. :additive_expr_stub_string1
  1324. "SUBTRACT_rax_from_rbx_into_rbx
  1325. MOVE_rbx_to_rax
  1326. "
  1327. :additive_expr_stub_string2
  1328. "MULTIPLY_rax_by_rbx_into_rax
  1329. "
  1330. :additive_expr_stub_string3
  1331. "XCHG_rax_rbx
  1332. LOAD_IMMEDIATE_rdx %0
  1333. DIVIDE_rax_by_rbx_into_rax
  1334. "
  1335. :additive_expr_stub_string4
  1336. "XCHG_rax_rbx
  1337. LOAD_IMMEDIATE_rdx %0
  1338. MODULUS_rax_from_rbx_into_rbx
  1339. MOVE_rdx_to_rax
  1340. "
  1341. :additive_expr_stub_string5
  1342. "COPY_rax_to_rcx
  1343. COPY_rbx_to_rax
  1344. SAL_rax_cl
  1345. "
  1346. :additive_expr_stub_string6
  1347. "COPY_rax_to_rcx
  1348. COPY_rbx_to_rax
  1349. SAR_rax_cl
  1350. "
  1351. ;; additive_expr function
  1352. ;; Receives struct token_list* global_token in R13,
  1353. ;; struct token_list* out in R12,
  1354. ;; struct token_list* string_list in R11
  1355. ;; struct token_list* global_list in R10
  1356. ;; and struct token_list* FUNC in R9
  1357. ;; and struct token_list* current_target in R8
  1358. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  1359. ;; Returns the token_lists modified
  1360. :additive_expr
  1361. CALLI R15 @postfix_expr ; Walk up the tree
  1362. CALLI R15 @additive_expr_stub ; Deal with nodes at this level
  1363. RET R15
  1364. ;; relational_expr_stub function
  1365. ;; receives nothing
  1366. ;; returns nothing
  1367. ;; Updates struct token_list*
  1368. :relational_expr_stub
  1369. PUSHR R0 R15 ; Protect R0
  1370. PUSHR R1 R15 ; Protect R1
  1371. PUSHR R2 R15 ; Protect R2
  1372. PUSHR R3 R15 ; Protect R3
  1373. ;; Fixed pieces
  1374. LOADUI R0 $additive_expr ; Set First argument
  1375. LOADUI R3 $relational_expr_stub
  1376. ;; The < bit
  1377. LOADUI R1 $relational_expr_stub_string0 ; Our first operation
  1378. LOADUI R2 $less_than_string ; Using "<"
  1379. CALLI R15 @general_recursion
  1380. ;; The <= bit
  1381. LOADUI R1 $relational_expr_stub_string1 ; Our second operation
  1382. LOADUI R2 $less_than_equal_string ; Using "<="
  1383. CALLI R15 @general_recursion
  1384. ;; The >= bit
  1385. LOADUI R1 $relational_expr_stub_string2 ; Our third operation
  1386. LOADUI R2 $greater_than_equal_string ; Using ">="
  1387. CALLI R15 @general_recursion
  1388. ;; The > bit
  1389. LOADUI R1 $relational_expr_stub_string3 ; Our fourth operation
  1390. LOADUI R2 $greater_than_string ; Using ">"
  1391. CALLI R15 @general_recursion
  1392. ;; The == bit
  1393. LOADUI R1 $relational_expr_stub_string4 ; Our fifth operation
  1394. LOADUI R2 $equal_to_string ; Using "=="
  1395. CALLI R15 @general_recursion
  1396. ;; The != bit
  1397. LOADUI R1 $relational_expr_stub_string5 ; Our final operation
  1398. LOADUI R2 $not_equal_string ; Using "!="
  1399. CALLI R15 @general_recursion
  1400. POPR R3 R15 ; Restore R3
  1401. POPR R2 R15 ; Restore R2
  1402. POPR R1 R15 ; Restore R1
  1403. POPR R0 R15 ; Restore R0
  1404. RET R15
  1405. :relational_expr_stub_string0
  1406. "CMP
  1407. SETL
  1408. MOVEZX
  1409. "
  1410. :relational_expr_stub_string1
  1411. "CMP
  1412. SETLE
  1413. MOVEZX
  1414. "
  1415. :relational_expr_stub_string2
  1416. "CMP
  1417. SETGE
  1418. MOVEZX
  1419. "
  1420. :relational_expr_stub_string3
  1421. "CMP
  1422. SETG
  1423. MOVEZX
  1424. "
  1425. :relational_expr_stub_string4
  1426. "CMP
  1427. SETE
  1428. MOVEZX
  1429. "
  1430. :relational_expr_stub_string5
  1431. "CMP
  1432. SETNE
  1433. MOVEZX
  1434. "
  1435. ;; relational_expr function
  1436. ;; Receives struct token_list* global_token in R13,
  1437. ;; struct token_list* out in R12,
  1438. ;; struct token_list* string_list in R11
  1439. ;; struct token_list* global_list in R10
  1440. ;; and struct token_list* FUNC in R9
  1441. ;; and struct token_list* current_target in R8
  1442. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  1443. ;; Returns the token_lists modified
  1444. :relational_expr
  1445. CALLI R15 @additive_expr ; Walk up the tree
  1446. CALLI R15 @relational_expr_stub ; Deal with nodes at this level
  1447. RET R15
  1448. ;; relational_expr_stub function
  1449. ;; receives nothing
  1450. ;; returns nothing
  1451. ;; Updates struct token_list*
  1452. :bitwise_expr_stub
  1453. PUSHR R0 R15 ; Protect R0
  1454. PUSHR R1 R15 ; Protect R1
  1455. PUSHR R2 R15 ; Protect R2
  1456. PUSHR R3 R15 ; Protect R3
  1457. ;; Fixed pieces
  1458. LOADUI R0 $relational_expr ; Set First argument
  1459. LOADUI R3 $bitwise_expr_stub
  1460. ;; The & bit
  1461. LOADUI R1 $bitwise_expr_stub_string0 ; Our first operation
  1462. LOADUI R2 $bitwise_and ; Using "&"
  1463. CALLI R15 @general_recursion
  1464. ;; The && bit
  1465. LOADUI R1 $bitwise_expr_stub_string0 ; Our first operation
  1466. LOADUI R2 $logical_and ; Using "&&"
  1467. CALLI R15 @general_recursion
  1468. ;; The | bit
  1469. LOADUI R1 $bitwise_expr_stub_string1 ; Our second operation
  1470. LOADUI R2 $bitwise_or ; Using "|"
  1471. CALLI R15 @general_recursion
  1472. ;; The || bit
  1473. LOADUI R1 $bitwise_expr_stub_string1 ; Our second operation
  1474. LOADUI R2 $logical_or ; Using "||"
  1475. CALLI R15 @general_recursion
  1476. ;; The ^ bit
  1477. LOADUI R1 $bitwise_expr_stub_string2 ; Our second operation
  1478. LOADUI R2 $bitwise_xor ; Using "^"
  1479. CALLI R15 @general_recursion
  1480. POPR R3 R15 ; Restore R3
  1481. POPR R2 R15 ; Restore R2
  1482. POPR R1 R15 ; Restore R1
  1483. POPR R0 R15 ; Restore R0
  1484. RET R15
  1485. :bitwise_expr_stub_string0
  1486. "AND_rax_rbx
  1487. "
  1488. :bitwise_expr_stub_string1
  1489. "OR_rax_rbx
  1490. "
  1491. :bitwise_expr_stub_string2
  1492. "XOR_rbx_rax_into_rax
  1493. "
  1494. ;; bitwise_expr function
  1495. ;; Receives struct token_list* global_token in R13,
  1496. ;; struct token_list* out in R12,
  1497. ;; struct token_list* string_list in R11
  1498. ;; struct token_list* global_list in R10
  1499. ;; and struct token_list* FUNC in R9
  1500. ;; and struct token_list* current_target in R8
  1501. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  1502. ;; Returns the token_lists modified
  1503. :bitwise_expr
  1504. CALLI R15 @relational_expr ; Walk up the tree
  1505. CALLI R15 @bitwise_expr_stub ; Deal with nodes at this level
  1506. RET R15
  1507. ;; primary_expr function
  1508. ;; Receives struct token_list* global_token in R13,
  1509. ;; struct token_list* out in R12,
  1510. ;; struct token_list* string_list in R11
  1511. ;; struct token_list* global_list in R10
  1512. ;; and struct token_list* FUNC in R9
  1513. ;; and struct token_list* current_target in R8
  1514. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  1515. ;; Returns the token_lists modified
  1516. :primary_expr
  1517. PUSHR R0 R15 ; Protect R0
  1518. PUSHR R1 R15 ; Protect R1
  1519. LOADUI R0 $sizeof_string ; Load "sizeof"
  1520. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  1521. CALLI R15 @match ; IF GLOBAL_TOKEN->S == "sizeof"
  1522. JUMP.Z R0 @primary_expr_negate ; Guess not
  1523. ;; Deal with sizeof expression
  1524. CALLI R15 @unary_expr_sizeof ; Do real work
  1525. JUMP @primary_expr_done ; Wrap up
  1526. :primary_expr_negate
  1527. LOADU8 R0 R1 0 ; GLOBAL_TOKEN->S[0]
  1528. CMPSKIPI.E R0 45 ; IF GLOBAL_TOKEN->S[0] == '-'
  1529. JUMP @primary_expr_bang ; If not try '!'
  1530. ;; Deal with -a and -4 expressions
  1531. LOADUI R0 $primary_expr_str0 ; Load HEADER
  1532. CALLI R15 @emit_out ; emit it
  1533. LOADUI R0 $primary_expr ; Using PRIMARY_EXPR
  1534. CALLI R15 @common_recursion ; Recurse
  1535. LOADUI R0 $primary_expr_str1 ; add footer
  1536. CALLI R15 @emit_out ; emit it
  1537. JUMP @primary_expr_done ; Wrap up
  1538. :primary_expr_bang
  1539. CMPSKIPI.E R0 33 ; IF GLOBAL_TOKEN->S[0] == "!"
  1540. JUMP @primary_expr_nested ; If not try '('
  1541. ;; deal with !a expressions
  1542. LOADUI R0 $primary_expr_str2 ; Load HEADER
  1543. CALLI R15 @emit_out ; emit it
  1544. LOADUI R0 $postfix_expr ; Using POSTFIX_EXPR
  1545. CALLI R15 @common_recursion ; Recurse
  1546. LOADUI R0 $primary_expr_str3 ; add footer
  1547. CALLI R15 @emit_out ; emit it
  1548. JUMP @primary_expr_done ; Wrap up
  1549. :primary_expr_nested
  1550. CMPSKIPI.E R0 40 ; IF GLOBAL_TOKEN->S[0] == '('
  1551. JUMP @primary_expr_ch ; If not try 'char'
  1552. ;; Deal with ( expr )
  1553. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  1554. CALLI R15 @expression ; Recurse
  1555. LOADUI R0 $primary_expr_str4 ; Using error message
  1556. LOADUI R1 $close_paren ; Using ")"
  1557. CALLI R15 @require_match ; Make sure we have closing match
  1558. JUMP @primary_expr_done ; Wrap up
  1559. :primary_expr_ch
  1560. CMPSKIPI.E R0 39 ; IF GLOBAL_TOKEN->S[0] == '\''
  1561. JUMP @primary_expr_st ; If not try "string"
  1562. ;; Deal with 'char'
  1563. CALLI R15 @primary_expr_char ; Collect char
  1564. JUMP @primary_expr_done ; Wrap up
  1565. :primary_expr_st
  1566. CMPSKIPI.E R0 34 ; IF GLOBAL_TOKEN->S[0] == '"'
  1567. JUMP @primary_expr_var ; If not try variables
  1568. ;; deal with "string"
  1569. CALLI R15 @primary_expr_string ; Collect string
  1570. JUMP @primary_expr_done ; Wrap up
  1571. :primary_expr_var
  1572. LOADUI R1 $variable_chars ; Using a-z+A-Z+_
  1573. CALLI R15 @in_set ; IF GLOBAL_TOKEN->S[0] in a-z+A-Z+_
  1574. JUMP.Z R0 @primary_expr_num
  1575. ;; Deal with foo TODO
  1576. CALLI R15 @primary_expr_variable ; deal with names
  1577. JUMP @primary_expr_done ; Wrap up
  1578. :primary_expr_num
  1579. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  1580. LOADU8 R0 R0 0 ; GLOBAL_TOKEN->S[0]
  1581. LOADUI R1 $digit_chars ; Using 0-9
  1582. CALLI R15 @in_set ; IF GLOBAL_TOKEN->S[0] in 0-9
  1583. JUMP.Z R0 @primary_expr_failure ; Fail HARD
  1584. ;; Deal with 5
  1585. CALLI R15 @primary_expr_number ; deal with number
  1586. :primary_expr_done
  1587. POPR R1 R15 ; Restore R1
  1588. POPR R0 R15 ; Restore R0
  1589. RET R15
  1590. :primary_expr_str0
  1591. "LOAD_IMMEDIATE_rax %0
  1592. "
  1593. :primary_expr_str1
  1594. "SUBTRACT_rax_from_rbx_into_rbx
  1595. MOVE_rbx_to_rax
  1596. "
  1597. :primary_expr_str2
  1598. "LOAD_IMMEDIATE_rax %1
  1599. "
  1600. :primary_expr_str3
  1601. "XOR_rbx_rax_into_rax
  1602. "
  1603. :primary_expr_str4
  1604. "Error in Primary expression
  1605. Didn't get )
  1606. "
  1607. ;; expression function
  1608. ;; Receives struct token_list* global_token in R13,
  1609. ;; struct token_list* out in R12,
  1610. ;; struct token_list* string_list in R11
  1611. ;; struct token_list* global_list in R10
  1612. ;; and struct token_list* FUNC in R9
  1613. ;; and struct token_list* current_target in R8
  1614. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  1615. ;; Returns the token_lists modified
  1616. :expression
  1617. PUSHR R0 R15 ; Protect R0
  1618. PUSHR R1 R15 ; Protect R1
  1619. PUSHR R2 R15 ; Protect R2
  1620. PUSHR R3 R15 ; Protect R3
  1621. CALLI R15 @bitwise_expr ; Check for more primitives first
  1622. LOADUI R0 $equal ; Using "="
  1623. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  1624. CALLI R15 @match ; IF GLOBAL_TOKEN->S == "="
  1625. JUMP.Z R0 @expression_done ; Be done
  1626. ;; Determine store type
  1627. LOADUI R3 $expression_string1 ; Assuming the default of STORE CHAR
  1628. ;; First possible reason for INT
  1629. LOADUI R0 $close_bracket ; Using "]"
  1630. LOAD32 R1 R13 4 ; GLOBAL_TOKEN->PREV
  1631. LOAD32 R1 R1 8 ; GLOBAL_TOKEN->PREV->S
  1632. CALLI R15 @match ; IF GLOBAL_TOKEN->PREV-> == "]"
  1633. CMPSKIPI.NE R0 0 ; IF FALSE
  1634. LOADUI R3 $expression_string0 ; STORE INTEGER
  1635. ;; Second possible reason for INTeger
  1636. LOADUI R0 $type_char_indirect_name ; Using "char*"
  1637. LOAD32 R1 R8 24 ; CURRENT_TARGET->NAME
  1638. CALLI R15 @match ; IF CURRENT_TARGET->NAME == "char*"
  1639. CMPSKIPI.NE R0 0 ; IF FALSE
  1640. LOADUI R3 $expression_string0 ; STORE INTEGER
  1641. ;; Recurse to evaluate expression being stored
  1642. LOADUI R0 $expression ; Using expression
  1643. CALLI R15 @common_recursion ; Perform common recursion
  1644. ;; Put our string and clean up
  1645. MOVE R0 R3 ; Using our STORED string
  1646. CALLI R15 @emit_out ; emit it
  1647. FALSE R8 ; CURRENT_TARGET = NULL
  1648. :expression_done
  1649. POPR R3 R15 ; Restore R3
  1650. POPR R2 R15 ; Restore R2
  1651. POPR R1 R15 ; Restore R1
  1652. POPR R0 R15 ; Restore R0
  1653. RET R15
  1654. :expression_string0
  1655. "STORE_INTEGER
  1656. "
  1657. :expression_string1
  1658. "STORE_CHAR
  1659. "
  1660. ;; process_if function
  1661. ;; struct token_list* out in R12,
  1662. ;; struct token_list* string_list in R11
  1663. ;; struct token_list* global_list in R10
  1664. ;; and struct token_list* FUNC in R9
  1665. ;; and struct token_list* current_target in R8
  1666. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  1667. ;; Returns the token_lists modified
  1668. :process_if
  1669. PUSHR R0 R15 ; Protect R0
  1670. PUSHR R1 R15 ; Protect R1
  1671. PUSHR R2 R15 ; Protect R2
  1672. LOADR32 R0 @current_count ; Using CURRENT_COUNT
  1673. ADDUI R1 R0 1 ; CURRENT_COUNT = CURRENT_COUNT + 1
  1674. STORER32 R1 @current_count ; Update CURRENT_COUNT
  1675. CALLI R15 @numerate_number ; Convert CURRENT_COUNT to string
  1676. MOVE R2 R0 ; Protect our string
  1677. LOADUI R0 $process_if_string0 ; using first string
  1678. CALLI R15 @emit_out ; emit it
  1679. LOAD32 R0 R9 8 ; FUNCTION->S
  1680. COPY R1 R2 ; Using our current count string
  1681. CALLI R15 @uniqueID_out ; Add unique identifier
  1682. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  1683. LOADUI R0 $process_if_string1 ; Our first error message
  1684. LOADUI R1 $open_paren ; Using "("
  1685. CALLI R15 @require_match ; Make sure we have what we need
  1686. CALLI R15 @expression ; Recurse to get our boolean expression
  1687. LOADUI R0 $process_if_string2 ; Our test and jump
  1688. CALLI R15 @emit_out ; emit it
  1689. LOAD32 R0 R9 8 ; FUNCTION->S
  1690. COPY R1 R2 ; Using our current count string
  1691. CALLI R15 @uniqueID_out ; Add unique identifier
  1692. LOADUI R0 $process_if_string3 ; Our second error message
  1693. LOADUI R1 $close_paren ; Using ")"
  1694. CALLI R15 @require_match ; Make sure we have what we need
  1695. CALLI R15 @statement ; Collect our if statement
  1696. LOADUI R0 $process_if_string4 ; Our jump over else
  1697. CALLI R15 @emit_out ; emit it
  1698. LOAD32 R0 R9 8 ; FUNCTION->S
  1699. COPY R1 R2 ; Using our current count string
  1700. CALLI R15 @uniqueID_out ; Add unique identifier
  1701. LOADUI R0 $process_if_string5 ; Our else label
  1702. CALLI R15 @emit_out ; emit it
  1703. LOAD32 R0 R9 8 ; FUNCTION->S
  1704. CALLI R15 @uniqueID_out ; Add unique identifier
  1705. LOADUI R0 $else_string ; Using "else"
  1706. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  1707. CALLI R15 @match ; IF GLOBAL_TOKEN->S == "else"
  1708. JUMP.Z R0 @process_if_else ; Looks like no else
  1709. ;; Deal with else
  1710. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  1711. CALLI R15 @statement ; Grab else statement
  1712. :process_if_else
  1713. LOADUI R0 $process_if_string6 ; Our jump over else
  1714. CALLI R15 @emit_out ; emit it
  1715. LOAD32 R0 R9 8 ; FUNCTION->S
  1716. COPY R1 R2 ; Using our current count string
  1717. CALLI R15 @uniqueID_out ; Add unique identifier
  1718. POPR R2 R15 ; Restore R2
  1719. POPR R1 R15 ; Restore R1
  1720. POPR R0 R15 ; Restore R0
  1721. RET R15
  1722. :process_if_string0
  1723. "# IF_"
  1724. :process_if_string1
  1725. "ERROR in process_if
  1726. MISSING (
  1727. "
  1728. :process_if_string2
  1729. "TEST
  1730. JUMP_EQ %ELSE_"
  1731. :process_if_string3
  1732. "ERROR in process_if
  1733. MISSING )
  1734. "
  1735. :process_if_string4
  1736. "JUMP %_END_IF_"
  1737. :process_if_string5
  1738. ":ELSE_"
  1739. :process_if_string6
  1740. ":_END_IF_"
  1741. ;; save_break_frame microfunction
  1742. ;; Overwrites R0 and R1
  1743. ;; Saves break frame on stack
  1744. ;; Returns to caller
  1745. :save_break_frame
  1746. POPR R1 R15 ; Save return address
  1747. LOADR32 R0 @break_frame ; Obtain BREAK_FRAME
  1748. PUSHR R0 R15 ; Protect BREAK_FRAME
  1749. LOADR32 R0 @break_target_head ; obtain HEAD
  1750. PUSHR R0 R15 ; Protect HEAD
  1751. LOADR32 R0 @break_target_func ; obtain FUNC
  1752. PUSHR R0 R15 ; Protect FUNC
  1753. LOADR32 R0 @break_target_num ; obtain NUM
  1754. PUSHR R0 R15 ; Protect NUM
  1755. PUSHR R1 R15 ; Set where we are returning to
  1756. RET R15
  1757. ;; restore_break_frame microfunction
  1758. ;; Overwrites R0 and R1
  1759. ;; Restores break frame from stack
  1760. ;; Returns to caller
  1761. :restore_break_frame
  1762. POPR R1 R15 ; Save return address
  1763. POPR R0 R15 ; obtain NUM
  1764. STORER32 R0 @break_target_num ; Restore NUM
  1765. POPR R0 R15 ; obtain FUNC
  1766. STORER32 R0 @break_target_func ; Restore FUNC
  1767. POPR R0 R15 ; obtain HEAD
  1768. STORER32 R0 @break_target_head ; Restore HEAD
  1769. POPR R0 R15 ; obtain BREAK_FRAME
  1770. STORER32 R0 @break_frame ; Restore BREAK_FRAME
  1771. PUSHR R1 R15 ; Set where we are returning to
  1772. RET R15
  1773. ;; set_break_frame microfunction
  1774. ;; Receives char* num in R0, char* head in R1
  1775. ;; Overwrites R0
  1776. ;; Sets break frame using
  1777. ;; R9 holding FUNC
  1778. ;; Returns to calling function
  1779. :set_break_frame
  1780. STORER32 R1 @break_target_head ; update BREAK_TARGET_HEAD
  1781. STORER32 R0 @break_target_num ; Update BREAK_TARGET_NUM
  1782. LOAD32 R0 R9 4 ; Using FUNCTION->LOCALS
  1783. STORER32 R0 @break_frame ; update BREAK_FRAME
  1784. LOAD32 R0 R9 8 ; Using FUNCTION->S
  1785. STORER32 R0 @break_target_func ; update BREAK_TARGET_FUNC
  1786. RET R15
  1787. ;; process_for function
  1788. ;; struct token_list* out in R12,
  1789. ;; struct token_list* string_list in R11
  1790. ;; struct token_list* global_list in R10
  1791. ;; and struct token_list* FUNC in R9
  1792. ;; and struct token_list* current_target in R8
  1793. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  1794. ;; Returns the token_lists modified
  1795. :process_for
  1796. PUSHR R2 R15 ; Protect R2
  1797. PUSHR R1 R15 ; Protect R1
  1798. PUSHR R0 R15 ; Protect R0
  1799. CALLI R15 @save_break_frame ; Save break frame
  1800. LOADR32 R0 @current_count ; Using CURRENT_COUNT
  1801. ADDUI R1 R0 1 ; CURRENT_COUNT = CURRENT_COUNT + 1
  1802. STORER32 R1 @current_count ; Update CURRENT_COUNT
  1803. CALLI R15 @numerate_number ; Convert to string
  1804. COPY R2 R0 ; Protect NUMBER_STRING
  1805. LOADUI R1 $process_for_string0 ; Get new HEAD
  1806. CALLI R15 @set_break_frame ; Set the break frame values
  1807. LOADUI R0 $process_for_string1 ; Our comment header
  1808. CALLI R15 @emit_out ; emit it
  1809. COPY R1 R2 ; Using NUMBER_STRING
  1810. LOAD32 R0 R9 8 ; FUNCTION->S
  1811. CALLI R15 @uniqueID_out ; emit it
  1812. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  1813. LOADUI R0 $process_for_string2 ; Our first error message
  1814. LOADUI R1 $open_paren ; Using "("
  1815. CALLI R15 @require_match ; Verify match
  1816. LOADUI R0 $semicolon ; Using ";"
  1817. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  1818. CALLI R15 @match ; IF GLOBAL_TOKEN->S -- ";"
  1819. CMPSKIPI.NE R0 0 ; If GLOBAL_TOKEN->S != ";"
  1820. CALLI R15 @expression ; Skip that step
  1821. LOADUI R0 $process_for_string3 ; Our comment header
  1822. CALLI R15 @emit_out ; emit it
  1823. COPY R1 R2 ; Using NUMBER_STRING
  1824. LOAD32 R0 R9 8 ; FUNCTION->S
  1825. CALLI R15 @uniqueID_out ; emit it
  1826. LOADUI R0 $process_for_string4 ; Our second error message
  1827. LOADUI R1 $semicolon ; Using ";"
  1828. CALLI R15 @require_match ; Verify match
  1829. CALLI R15 @expression ; TEST logic required
  1830. LOADUI R0 $process_for_string5 ; Our comment header
  1831. CALLI R15 @emit_out ; emit it
  1832. COPY R1 R2 ; Using NUMBER_STRING
  1833. LOAD32 R0 R9 8 ; FUNCTION->S
  1834. CALLI R15 @uniqueID_out ; emit it
  1835. LOADUI R0 $process_for_string6 ; Our comment header
  1836. CALLI R15 @emit_out ; emit it
  1837. LOAD32 R0 R9 8 ; FUNCTION->S
  1838. CALLI R15 @uniqueID_out ; emit it
  1839. LOADUI R0 $process_for_string7 ; Our comment header
  1840. CALLI R15 @emit_out ; emit it
  1841. LOAD32 R0 R9 8 ; FUNCTION->S
  1842. CALLI R15 @uniqueID_out ; emit it
  1843. LOADUI R0 $process_for_string8 ; Our third error message
  1844. LOADUI R1 $semicolon ; Using ";"
  1845. CALLI R15 @require_match ; Verify match
  1846. CALLI R15 @expression ; Iterator logic
  1847. LOADUI R0 $process_for_string9 ; Our comment header
  1848. CALLI R15 @emit_out ; emit it
  1849. COPY R1 R2 ; Using NUMBER_STRING
  1850. LOAD32 R0 R9 8 ; FUNCTION->S
  1851. CALLI R15 @uniqueID_out ; emit it
  1852. LOADUI R0 $process_for_string10 ; Our comment header
  1853. CALLI R15 @emit_out ; emit it
  1854. LOAD32 R0 R9 8 ; FUNCTION->S
  1855. CALLI R15 @uniqueID_out ; emit it
  1856. LOADUI R0 $process_for_string11 ; Our final error message
  1857. LOADUI R1 $close_paren ; Using ")"
  1858. CALLI R15 @require_match ; Verify match
  1859. CALLI R15 @statement ; Main body
  1860. LOADUI R0 $process_for_string12 ; Our comment header
  1861. CALLI R15 @emit_out ; emit it
  1862. COPY R1 R2 ; Using NUMBER_STRING
  1863. LOAD32 R0 R9 8 ; FUNCTION->S
  1864. CALLI R15 @uniqueID_out ; emit it
  1865. LOADUI R0 $process_for_string13 ; Our comment header
  1866. CALLI R15 @emit_out ; emit it
  1867. LOAD32 R0 R9 8 ; FUNCTION->S
  1868. CALLI R15 @uniqueID_out ; emit it
  1869. CALLI R15 @restore_break_frame ; Restore break frame
  1870. POPR R0 R15 ; Restore R0
  1871. POPR R1 R15 ; Restore R1
  1872. POPR R2 R15 ; Restore R2
  1873. RET R15
  1874. :process_for_string0
  1875. "FOR_END_"
  1876. :process_for_string1
  1877. "# FOR_initialization_"
  1878. :process_for_string2
  1879. "ERROR in process_for
  1880. MISSING (
  1881. "
  1882. :process_for_string3
  1883. ":FOR_"
  1884. :process_for_string4
  1885. "ERROR in process_for
  1886. MISSING ;1
  1887. "
  1888. :process_for_string5
  1889. "TEST
  1890. JUMP_EQ %FOR_END_"
  1891. :process_for_string6
  1892. "JUMP %FOR_THEN_"
  1893. :process_for_string7
  1894. ":FOR_ITER_"
  1895. :process_for_string8
  1896. "ERROR in process_for
  1897. MISSING ;2
  1898. "
  1899. :process_for_string9
  1900. "JUMP %FOR_"
  1901. :process_for_string10
  1902. ":FOR_THEN_"
  1903. :process_for_string11
  1904. "ERROR in process_for
  1905. MISSING )
  1906. "
  1907. :process_for_string12
  1908. "JUMP %FOR_ITER_"
  1909. :process_for_string13
  1910. ":FOR_END_"
  1911. ;; process_do function
  1912. ;; struct token_list* out in R12,
  1913. ;; struct token_list* string_list in R11
  1914. ;; struct token_list* global_list in R10
  1915. ;; and struct token_list* FUNC in R9
  1916. ;; and struct token_list* current_target in R8
  1917. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  1918. ;; Returns the token_lists modified
  1919. :process_do
  1920. PUSHR R2 R15 ; Protect R2
  1921. PUSHR R1 R15 ; Protect R1
  1922. PUSHR R0 R15 ; Protect R0
  1923. CALLI R15 @save_break_frame ; Save break frame
  1924. LOADR32 R0 @current_count ; Using CURRENT_COUNT
  1925. ADDUI R1 R0 1 ; CURRENT_COUNT = CURRENT_COUNT + 1
  1926. STORER32 R1 @current_count ; Update CURRENT_COUNT
  1927. CALLI R15 @numerate_number ; Convert to string
  1928. COPY R2 R0 ; Protect NUMBER_STRING
  1929. LOADUI R1 $process_do_string0 ; Using our desired head
  1930. CALLI R15 @set_break_frame ; Set the break frame values
  1931. LOADUI R0 $process_do_string1 ; Our label
  1932. CALLI R15 @emit_out ; emit it
  1933. COPY R1 R2 ; Using NUMBER_STRING
  1934. LOAD32 R0 R9 8 ; FUNCTION->S
  1935. CALLI R15 @uniqueID_out ; emit it
  1936. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  1937. CALLI R15 @statement ; Collect our Do statement
  1938. LOADUI R0 $process_do_string2 ; our first error message
  1939. LOADUI R1 $while_string ; Using "while"
  1940. CALLI R15 @require_match ; Check for match
  1941. LOADUI R0 $process_do_string3 ; our second error message
  1942. LOADUI R1 $open_paren ; Using "("
  1943. CALLI R15 @require_match ; Check for match
  1944. CALLI R15 @expression ; Our logical expression
  1945. LOADUI R0 $process_do_string4 ; our third error message
  1946. LOADUI R1 $close_paren ; Using ")"
  1947. CALLI R15 @require_match ; Check for match
  1948. LOADUI R0 $process_do_string5 ; our final error message
  1949. LOADUI R1 $semicolon ; Using ";"
  1950. CALLI R15 @require_match ; Check for match
  1951. LOADUI R0 $process_do_string6 ; Our test string
  1952. CALLI R15 @emit_out ; emit it
  1953. COPY R1 R2 ; Put NUMBER_STRING in right place
  1954. LOAD32 R0 R9 8 ; FUNCTION->S
  1955. CALLI R15 @uniqueID_out ; emit it
  1956. LOADUI R0 $process_do_string7 ; Our end label string
  1957. CALLI R15 @emit_out ; emit it
  1958. LOAD32 R0 R9 8 ; FUNCTION->S
  1959. CALLI R15 @uniqueID_out ; emit it
  1960. CALLI R15 @restore_break_frame ; Restore break frame
  1961. POPR R0 R15 ; Restore R0
  1962. POPR R1 R15 ; Restore R1
  1963. POPR R2 R15 ; Restore R2
  1964. RET R15
  1965. :process_do_string0
  1966. "DO_END_"
  1967. :process_do_string1
  1968. ":DO_"
  1969. :process_do_string2
  1970. "ERROR in process_do
  1971. MISSING while
  1972. "
  1973. :process_do_string3
  1974. "ERROR in process_do
  1975. MISSING (
  1976. "
  1977. :process_do_string4
  1978. "ERROR in process_do
  1979. MISSING )
  1980. "
  1981. :process_do_string5
  1982. "ERROR in process_do
  1983. MISSING ;
  1984. "
  1985. :process_do_string6
  1986. "TEST
  1987. JUMP_NE %DO_"
  1988. :process_do_string7
  1989. ":DO_END_"
  1990. ;; process_while function
  1991. ;; struct token_list* out in R12,
  1992. ;; struct token_list* string_list in R11
  1993. ;; struct token_list* global_list in R10
  1994. ;; and struct token_list* FUNC in R9
  1995. ;; and struct token_list* current_target in R8
  1996. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  1997. ;; Returns the token_lists modified
  1998. :process_while
  1999. PUSHR R2 R15 ; Protect R2
  2000. PUSHR R1 R15 ; Protect R1
  2001. PUSHR R0 R15 ; Protect R0
  2002. CALLI R15 @save_break_frame ; Save break frame
  2003. LOADR32 R0 @current_count ; Using CURRENT_COUNT
  2004. ADDUI R1 R0 1 ; CURRENT_COUNT = CURRENT_COUNT + 1
  2005. STORER32 R1 @current_count ; Update CURRENT_COUNT
  2006. CALLI R15 @numerate_number ; Convert to string
  2007. COPY R2 R0 ; Protect NUMBER_STRING
  2008. LOADUI R1 $process_while_string0 ; Set HEAD
  2009. CALLI R15 @set_break_frame ; Set the break frame values
  2010. LOADUI R0 $process_while_string1 ; Our head label
  2011. CALLI R15 @emit_out ; emit it
  2012. COPY R1 R2 ; Using NUMBER_STRING
  2013. LOAD32 R0 R9 8 ; FUNCTION->S
  2014. CALLI R15 @uniqueID_out ; emit it
  2015. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2016. LOADUI R0 $process_while_string2 ; Our first error message
  2017. LOADUI R1 $open_paren ; Using "("
  2018. CALLI R15 @require_match ; Check for match
  2019. CALLI R15 @expression ; Collect test expression
  2020. LOADUI R0 $process_while_string3 ; Our test and jump
  2021. CALLI R15 @emit_out ; emit it
  2022. COPY R1 R2 ; Using NUMBER_STRING
  2023. LOAD32 R0 R9 8 ; FUNCTION->S
  2024. CALLI R15 @uniqueID_out ; emit it
  2025. LOADUI R0 $process_while_string4 ; Our trailing comment
  2026. CALLI R15 @emit_out ; emit it
  2027. LOAD32 R0 R9 8 ; FUNCTION->S
  2028. CALLI R15 @uniqueID_out ; emit it
  2029. LOADUI R0 $process_while_string5 ; Our first error message
  2030. LOADUI R1 $close_paren ; Using ")"
  2031. CALLI R15 @require_match ; Check for match
  2032. CALLI R15 @statement ; Collect our loop statement
  2033. LOADUI R0 $process_while_string6 ; Our test and jump
  2034. CALLI R15 @emit_out ; emit it
  2035. COPY R1 R2 ; Using NUMBER_STRING
  2036. LOAD32 R0 R9 8 ; FUNCTION->S
  2037. CALLI R15 @uniqueID_out ; emit it
  2038. LOADUI R0 $process_while_string7 ; Our trailing comment
  2039. CALLI R15 @emit_out ; emit it
  2040. LOAD32 R0 R9 8 ; FUNCTION->S
  2041. CALLI R15 @uniqueID_out ; emit it
  2042. CALLI R15 @restore_break_frame ; Restore break frame
  2043. POPR R0 R15 ; Restore R0
  2044. POPR R1 R15 ; Restore R1
  2045. POPR R2 R15 ; Restore R2
  2046. RET R15
  2047. :process_while_string0
  2048. "END_WHILE_"
  2049. :process_while_string1
  2050. ":WHILE_"
  2051. :process_while_string2
  2052. "ERROR in process_while
  2053. MISSING (
  2054. "
  2055. :process_while_string3
  2056. "TEST
  2057. JUMP_EQ %END_WHILE_"
  2058. :process_while_string4
  2059. "# THEN_while_"
  2060. :process_while_string5
  2061. "ERROR in process_while
  2062. MISSING )
  2063. "
  2064. :process_while_string6
  2065. "JUMP %WHILE_"
  2066. :process_while_string7
  2067. ":END_WHILE_"
  2068. ;; return_result function
  2069. ;; Receives nothing
  2070. ;; Returns nothing
  2071. ;; and struct token_list* FUNC in R9
  2072. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  2073. :return_result
  2074. PUSHR R0 R15 ; Protect R0
  2075. PUSHR R1 R15 ; Protect R1
  2076. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2077. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  2078. LOADU8 R0 R0 0 ; GLOBAL_TOKEN->S[0]
  2079. CMPSKIPI.E R0 59 ; IF GLOBAL_TOKEN->S[0] == ';'
  2080. CALLI R15 @expression ; Evaluate expression
  2081. LOADUI R0 $return_result_string0 ; Using or error message
  2082. LOADUI R1 $semicolon ; Using ";"
  2083. CALLI R15 @require_match ; Require a match to ";"
  2084. LOADUI R0 $return_result_string1 ; Our pop command
  2085. LOAD32 R1 R9 4 ; FUNCTION->LOCALS
  2086. :return_result_iter
  2087. JUMP.Z R1 @return_result_done ; Be done when we hit NULL
  2088. CALLI R15 @emit_out ; Put the string every iteration
  2089. LOAD32 R1 R1 0 ; I = I->NEXT
  2090. JUMP @return_result_iter ; Keep looping
  2091. :return_result_done
  2092. LOADUI R0 $return_result_string2 ; Our footer
  2093. CALLI R15 @emit_out ; emit it
  2094. POPR R1 R15 ; Restore R1
  2095. POPR R0 R15 ; Restore R0
  2096. RET R15
  2097. :return_result_string0
  2098. "ERROR in return_result
  2099. MISSING ;
  2100. "
  2101. :return_result_string1
  2102. "POP_RBX # _return_result_locals
  2103. "
  2104. :return_result_string2
  2105. "RETURN
  2106. "
  2107. ;; process_break function
  2108. ;; Receives nothing
  2109. ;; and struct token_list* FUNC in R9
  2110. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  2111. ;; Returns the token_lists modified
  2112. :process_break
  2113. PUSHR R0 R15 ; Protect R0
  2114. PUSHR R1 R15 ; Protect R1
  2115. PUSHR R2 R15 ; Protect R2
  2116. LOADR32 R0 @break_target_head ; BREAK_TARGET_HEAD
  2117. JUMP.NZ R0 @process_break_NON_NULL
  2118. ;; Deal with NULL == BREAK_TARGET_HEAD
  2119. LOADUI R0 $process_break_string0 ; Our first error message
  2120. FALSE R1 ; Write for User
  2121. CALLI R15 @file_print ; write it
  2122. CALLI R15 @line_error ; Give useful info
  2123. LOADUI R0 $newline ; Using "\n"
  2124. CALLI R15 @file_print ; Print it
  2125. HALT
  2126. :process_break_NON_NULL
  2127. LOADR32 R2 @break_frame ; BREAK_FRAME
  2128. LOAD32 R1 R9 4 ; I = FUNCTION->LOCALS
  2129. LOADUI R0 $process_break_string1 ; Our pop string
  2130. :process_break_iter
  2131. CMPJUMPI.E R1 R2 @process_break_done ; IF I == BREAK_FRAME
  2132. JUMP.Z R1 @process_break_done ; IF NULL == I break
  2133. CALLI R15 @emit_out ; emit pop
  2134. LOAD32 R1 R1 0 ; I = I->NEXT
  2135. JUMP @process_break_iter ; Loop
  2136. :process_break_done
  2137. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2138. LOADUI R0 $process_break_string2 ; Our jump string
  2139. CALLI R15 @emit_out ; emit it
  2140. LOADR32 R0 @break_target_head ; Our HEAD string
  2141. CALLI R15 @emit_out ; emit it
  2142. LOADR32 R0 @break_target_func ; Our FUNC string
  2143. CALLI R15 @emit_out ; emit it
  2144. LOADUI R0 $underline ; Using "_"
  2145. CALLI R15 @emit_out ; emit it
  2146. LOADR32 R0 @break_target_num ; Our NUM string
  2147. CALLI R15 @emit_out ; emit it
  2148. LOADUI R0 $newline ; Using "\n"
  2149. CALLI R15 @emit_out ; emit it
  2150. LOADUI R0 $process_break_string3 ; Our final error string
  2151. LOADUI R1 $semicolon ; Using ";"
  2152. CALLI R15 @require_match ; Make sure we get that match
  2153. POPR R2 R15 ; Restore R2
  2154. POPR R1 R15 ; Restore R1
  2155. POPR R0 R15 ; Restore R0
  2156. RET R15
  2157. :process_break_string0
  2158. "Not inside of a loop or case statement
  2159. "
  2160. :process_break_string1
  2161. "POP_RBX # break_cleanup_locals
  2162. "
  2163. :process_break_string2
  2164. "JUMP %"
  2165. :process_break_string3
  2166. "ERROR in break statement
  2167. Missing ;
  2168. "
  2169. :break_frame
  2170. NOP
  2171. :break_target_head
  2172. NOP
  2173. :break_target_func
  2174. NOP
  2175. :break_target_num
  2176. NOP
  2177. ;; process_asm function
  2178. ;; Receives struct token_list* global_token in R13,
  2179. ;; struct token_list* out in R12,
  2180. ;; struct token_list* string_list in R11
  2181. ;; struct token_list* global_list in R10
  2182. ;; and struct token_list* FUNC in R9
  2183. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  2184. ;; Returns the token_lists modified
  2185. :process_asm
  2186. PUSHR R0 R15 ; Protect R0
  2187. PUSHR R1 R15 ; Protect R1
  2188. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2189. ;; First required match
  2190. LOADUI R0 $process_asm_string0 ; Using our First error message
  2191. LOADUI R1 $open_paren ; Using "("
  2192. CALLI R15 @require_match ; Make sure of our required match
  2193. :process_asm_iter
  2194. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  2195. LOADU8 R0 R0 0 ; GLOBAL_TOKEN->S[0]
  2196. CMPSKIPI.E R0 34 ; IF GLOBAL_TOKEN->S[0] == '"'
  2197. JUMP @process_asm_done ; Otherwise be done
  2198. ;; Add block of assembly
  2199. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  2200. ADDUI R0 R0 1 ; GLOBAL_TOKEN->S + 1
  2201. CALLI R15 @emit_out ; emit it
  2202. LOADUI R0 $newline ; Using "\n"
  2203. CALLI R15 @emit_out ; emit it
  2204. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2205. JUMP @process_asm_iter
  2206. :process_asm_done
  2207. LOADUI R0 $process_asm_string1 ; Using our First error message
  2208. LOADUI R1 $close_paren ; Using ")"
  2209. CALLI R15 @require_match ; Make sure of our required match
  2210. LOADUI R0 $process_asm_string2 ; Using our First error message
  2211. LOADUI R1 $semicolon ; Using ";"
  2212. CALLI R15 @require_match ; Make sure of our required match
  2213. POPR R1 R15 ; Restore R1
  2214. POPR R0 R15 ; Restore R0
  2215. RET R15
  2216. :process_asm_string0
  2217. "ERROR in process_asm
  2218. MISSING (
  2219. "
  2220. :process_asm_string1
  2221. "ERROR in process_asm
  2222. MISSING )
  2223. "
  2224. :process_asm_string2
  2225. "ERROR in process_asm
  2226. MISSING ;
  2227. "
  2228. ;; recursive_statement function
  2229. ;; Receives struct token_list* global_token in R13,
  2230. ;; struct token_list* out in R12,
  2231. ;; struct token_list* string_list in R11
  2232. ;; struct token_list* global_list in R10
  2233. ;; and struct token_list* FUNC in R9
  2234. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  2235. ;; Returns the token_lists modified
  2236. :recursive_statement
  2237. PUSHR R0 R15 ; Protect R0
  2238. PUSHR R1 R15 ; Protect R1
  2239. PUSHR R2 R15 ; Protect R2
  2240. PUSHR R3 R15 ; Protect R3
  2241. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2242. LOAD32 R3 R9 4 ; FRAME = FUNCTION->LOCALS
  2243. :recursive_statement_iter
  2244. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  2245. LOADUI R0 $close_curly_brace ; '}'
  2246. CALLI R15 @match ; IF GLOBAL_TOKEN->S == "}"
  2247. JUMP.NZ R0 @recursive_statement_cleanup
  2248. ;; Lets collect those statements
  2249. CALLI R15 @statement ; Collect next statement
  2250. JUMP @recursive_statement_iter ; Iterate
  2251. :recursive_statement_cleanup
  2252. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2253. LOAD32 R1 R12 8 ; OUT->S
  2254. LOADUI R0 $recursive_statement_string0 ; "RETURN\n"
  2255. CALLI R15 @match ; IF OUT->S == "RETURN\n"
  2256. JUMP.NZ R0 @recursive_statement_done ; Save some work
  2257. ;; Lets pop them all off
  2258. LOAD32 R2 R9 4 ; FUNC->LOCALS
  2259. LOADUI R0 $recursive_statement_string1 ; Our POP string
  2260. :recursive_statement_pop
  2261. CMPJUMPI.E R2 R3 @recursive_statement_done
  2262. CALLI R15 @emit_out ; emit it
  2263. LOAD32 R2 R2 0 ; I = I->NEXT
  2264. JUMP.NZ R2 @recursive_statement_pop ; Keep looping
  2265. :recursive_statement_done
  2266. STORE32 R3 R9 4 ; FUNC->LOCALS = FRAME
  2267. POPR R3 R15 ; Restore R3
  2268. POPR R2 R15 ; Restore R2
  2269. POPR R1 R15 ; Restore R1
  2270. POPR R0 R15 ; Restore R0
  2271. RET R15
  2272. :recursive_statement_string0
  2273. "RETURN
  2274. "
  2275. :recursive_statement_string1
  2276. "POP_RBX # _recursive_statement_locals
  2277. "
  2278. ;; statement function
  2279. ;; Receives struct token_list* global_token in R13,
  2280. ;; struct token_list* out in R12,
  2281. ;; struct token_list* string_list in R11
  2282. ;; struct token_list* global_list in R10
  2283. ;; and struct token_list* FUNC in R9
  2284. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  2285. ;; Returns the token_lists modified
  2286. :statement
  2287. PUSHR R0 R15 ; Protect R0
  2288. PUSHR R1 R15 ; Protect R1
  2289. PUSHR R2 R15 ; Protect R2
  2290. LOAD32 R2 R13 8 ; GLOBAL_TOKEN->S
  2291. LOADU8 R0 R2 0 ; GLOBAL_TOKEN->S[0]
  2292. CMPSKIPI.E R0 123 ; If GLOBAL_TOKEN->S[0] != '{'
  2293. JUMP @statement_label ; Try next match
  2294. ;; Deal with { statements }
  2295. CALLI R15 @recursive_statement
  2296. JUMP @statement_done ; All done
  2297. :statement_label
  2298. CMPSKIPI.E R0 58 ; If GLOBAL_TOKEN->S[0] != ':'
  2299. JUMP @statement_collect_local ; Try next match
  2300. ;; Deal with :label
  2301. LOAD32 R0 R13 8 ; Using GLOBAL_TOKEN->S
  2302. CALLI R15 @emit_out ; emit it
  2303. LOADUI R0 $statement_string0 ; Using label string
  2304. CALLI R15 @emit_out ; emit it
  2305. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2306. JUMP @statement_done ; Move on to next thing
  2307. :statement_collect_local
  2308. LOADUI R0 $struct ; Using "struct"
  2309. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  2310. CALLI R15 @match ; IF GLOBAL_TOKEN->S == "struct"
  2311. JUMP.NZ R0 @statement_collect_local_0
  2312. ;; Otherwise check if it is a primitive
  2313. LOADUI R0 $prim_types ; Using the Primitive types list
  2314. SWAP R0 R1 ; Put in correct order
  2315. CALLI R15 @lookup_type ; Check if a primitive type
  2316. JUMP.Z R0 @statement_process_if ; If not try the next one
  2317. :statement_collect_local_0
  2318. CALLI R15 @collect_local ; Collect the local
  2319. JUMP @statement_done ; And move on
  2320. :statement_process_if
  2321. LOADUI R0 $if_string ; Using "if"
  2322. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  2323. CALLI R15 @match ; IF GLOBAL_TOKEN->S == "if"
  2324. JUMP.Z R0 @statement_process_do
  2325. CALLI R15 @process_if ; Collect that if statement
  2326. JUMP @statement_done ; Move on to next thing
  2327. :statement_process_do
  2328. LOADUI R0 $do_string ; Using "do"
  2329. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  2330. CALLI R15 @match ; IF GLOBAL_TOKEN->S == "do"
  2331. JUMP.Z R0 @statement_process_while
  2332. CALLI R15 @process_do ; Collect that do statement
  2333. JUMP @statement_done ; Move on to next thing
  2334. :statement_process_while
  2335. LOADUI R0 $while_string ; Using "while"
  2336. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  2337. CALLI R15 @match ; IF GLOBAL_TOKEN->S == "while"
  2338. JUMP.Z R0 @statement_process_for
  2339. CALLI R15 @process_while ; Collect that while statement
  2340. JUMP @statement_done ; Move on to next thing
  2341. :statement_process_for
  2342. LOADUI R0 $for_string ; Using "for"
  2343. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  2344. CALLI R15 @match ; IF GLOBAL_TOKEN->S == "for"
  2345. JUMP.Z R0 @statement_process_asm
  2346. CALLI R15 @process_for ; Collect that FOR statement
  2347. JUMP @statement_done ; Move on to next thing
  2348. :statement_process_asm
  2349. LOADUI R0 $asm_string ; Using "asm"
  2350. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  2351. CALLI R15 @match ; IF GLOBAL_TOKEN->S == "asm"
  2352. JUMP.Z R0 @statement_goto
  2353. CALLI R15 @process_asm ; Collect that ASM statement
  2354. JUMP @statement_done ; Move on to next thing
  2355. :statement_goto
  2356. LOADUI R0 $goto_string ; Using "goto"
  2357. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  2358. CALLI R15 @match ; IF GLOBAL_TOKEN->S == "goto"
  2359. JUMP.Z R0 @statement_return_result
  2360. ;; Deal with goto label:
  2361. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2362. LOADUI R0 $statement_string1 ; Using our JUMP string
  2363. CALLI R15 @emit_out ; emit it
  2364. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  2365. CALLI R15 @emit_out ; emit it
  2366. LOADUI R0 $newline ; "\n"
  2367. CALLI R15 @emit_out ; emit it
  2368. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2369. LOADUI R0 $statement_string2 ; Using our error message
  2370. LOADUI R1 $semicolon ; Using ";"
  2371. CALLI R15 @require_match ; Make sure of our required match
  2372. JUMP @statement_done ; Move on
  2373. :statement_return_result
  2374. LOADUI R0 $return_string ; Using "return"
  2375. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  2376. CALLI R15 @match ; IF GLOBAL_TOKEN->S == "return"
  2377. JUMP.Z R0 @statement_break
  2378. ;; Deal with return statements in functions
  2379. CALLI R15 @return_result ; Do all of the work
  2380. JUMP @statement_done ; Move on to next
  2381. :statement_break
  2382. LOADUI R0 $break_string ; Using "break"
  2383. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  2384. CALLI R15 @match ; IF GLOBAL_TOKEN->S == "break"
  2385. JUMP.Z R0 @statement_continue
  2386. ;; Let break function deal with updating out
  2387. CALLI R15 @process_break ; Do all the work
  2388. JUMP @statement_done ; Move on to next
  2389. :statement_continue
  2390. LOADUI R0 $continue_string ; Using "continue"
  2391. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  2392. CALLI R15 @match ; IF GLOBAL_TOKEN->S == "continue"
  2393. JUMP.Z R0 @statement_expression
  2394. ;; Simple Continue compatibility
  2395. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2396. LOADUI R0 $statement_string3 ; Using our continue comment string
  2397. CALLI R15 @emit_out ; emit it
  2398. LOADUI R0 $statement_string2 ; Using our error message
  2399. LOADUI R1 $semicolon ; Using ";"
  2400. CALLI R15 @require_match ; Make sure of our required match
  2401. JUMP @statement_done ; Move on
  2402. :statement_expression
  2403. CALLI R15 @expression ; Do expression evaluation
  2404. LOADUI R0 $statement_string2 ; Load our error message
  2405. LOADUI R1 $semicolon ; use ";"
  2406. CALLI R15 @require_match ; Make sure GLOBAL_TOKEN-> == ";"
  2407. :statement_done
  2408. POPR R2 R15 ; Restore R2
  2409. POPR R1 R15 ; Restore R1
  2410. POPR R0 R15 ; Restore R0
  2411. RET R15
  2412. :statement_string0
  2413. " #C goto label
  2414. "
  2415. :statement_string1
  2416. "JUMP %"
  2417. :statement_string2
  2418. "ERROR in statement
  2419. MISSING ;
  2420. "
  2421. :statement_string3
  2422. "
  2423. #continue statement
  2424. "
  2425. ;; collect_local function
  2426. ;; Receives struct token_list* global_token in R13,
  2427. ;; struct token_list* out in R12,
  2428. ;; struct token_list* string_list in R11
  2429. ;; struct token_list* global_list in R10
  2430. ;; and struct token_list* FUNC in R9
  2431. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  2432. ;; Returns the token_lists modified
  2433. :collect_local
  2434. PUSHR R0 R15 ; Protect R0
  2435. PUSHR R1 R15 ; Protect R1
  2436. PUSHR R2 R15 ; Protect R2
  2437. CALLI R15 @type_name ; Get it's type
  2438. MOVE R1 R0 ; Prepare for call
  2439. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  2440. LOAD32 R2 R9 4 ; FUNC->LOCALS
  2441. CALLI R15 @sym_declare ; SET A
  2442. MOVE R2 R0 ; Protect A
  2443. ;; Figure out depth
  2444. LOADUI R0 $main_string ; Using "main"
  2445. LOAD32 R1 R9 8 ; FUNC->S
  2446. CALLI R15 @match ; IF FUNC->S == "main"
  2447. JUMP.Z R0 @collect_local_0 ; Try next
  2448. LOAD32 R0 R9 4 ; FUNC->LOCALS
  2449. JUMP.NZ R0 @collect_local_0 ; Try next
  2450. LOADI R0 -40 ; The default depth for main
  2451. STORE32 R0 R2 16 ; A->DEPTH = -40
  2452. JUMP @collect_local_output ; Deal with header
  2453. :collect_local_0
  2454. LOAD32 R0 R9 16 ; FUNC->ARGS
  2455. JUMP.NZ R0 @collect_local_1 ; Try Next
  2456. LOAD32 R0 R9 4 ; FUNC->LOCALS
  2457. JUMP.NZ R0 @collect_local_1 ; Try Next
  2458. LOADI R0 -16 ; The default depth for foo()
  2459. STORE32 R0 R2 16 ; A->DEPTH = -16
  2460. JUMP @collect_local_output ; Deal with header
  2461. :collect_local_1
  2462. LOAD32 R0 R9 4 ; FUNC->LOCALS
  2463. JUMP.NZ R0 @collect_local_2 ; Try Next
  2464. LOAD32 R0 R9 16 ; FUNC->ARGS
  2465. LOAD32 R0 R0 16 ; FUNC->ARGS->DEPTH
  2466. SUBI R0 R0 16 ; DEPTH = FUNC->ARGS->DEPTH - 16
  2467. STORE32 R0 R2 16 ; A->DEPTH = DEPTH
  2468. JUMP @collect_local_output ; Deal with header
  2469. :collect_local_2
  2470. LOAD32 R0 R9 4 ; FUNC->LOCALS
  2471. LOAD32 R0 R0 16 ; FUNC->LOCALS->DEPTH
  2472. SUBI R0 R0 8 ; DEPTH = FUNC->LOCALS->DEPTH - 8
  2473. STORE32 R0 R2 16 ; A->DEPTH = DEPTH
  2474. :collect_local_output
  2475. STORE32 R2 R9 4 ; FUNC->LOCALS = A
  2476. ;; Output header
  2477. LOADUI R0 $collect_local_string0 ; Starting with the comment
  2478. CALLI R15 @emit_out ; emit it
  2479. LOAD32 R0 R2 8 ; A->S
  2480. CALLI R15 @emit_out ; emit it
  2481. LOADUI R0 $newline ; Using "\n"
  2482. CALLI R15 @emit_out ; emit it
  2483. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2484. ;; Deal with possible assignment
  2485. LOADUI R0 $equal ; Using "="
  2486. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  2487. CALLI R15 @match ; IF GLOBAL_TOKEN->S == "="
  2488. JUMP.Z R0 @collect_local_nonassign
  2489. ;; Deal with assignment of the local
  2490. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2491. CALLI R15 @expression ; Update OUT with the evaluation of the Expression
  2492. :collect_local_nonassign
  2493. LOADUI R0 $collect_local_string1 ; Our error message
  2494. LOADUI R1 $semicolon ; Using ";"
  2495. CALLI R15 @require_match ; Make sure GLOBAL_TOKEN->S == ";"
  2496. ;; Final Footer
  2497. LOADUI R0 $collect_local_string2 ; Add our PUSH statement
  2498. CALLI R15 @emit_out ; emit it
  2499. LOAD32 R0 R2 8 ; A->S
  2500. CALLI R15 @emit_out ; emit it
  2501. LOADUI R0 $newline ; Using "\n"
  2502. CALLI R15 @emit_out ; emit it
  2503. POPR R2 R15 ; Restore R2
  2504. POPR R1 R15 ; Restore R1
  2505. POPR R0 R15 ; Restore R0
  2506. RET R15
  2507. :collect_local_string0
  2508. "# Defining local "
  2509. :collect_local_string1
  2510. "ERROR in collect_local
  2511. Missing ;
  2512. "
  2513. :collect_local_string2
  2514. "PUSH_RAX #"
  2515. ;; collect_arguments function
  2516. ;; Receives struct token_list* global_token in R13,
  2517. ;; struct token_list* out in R12,
  2518. ;; struct token_list* string_list in R11
  2519. ;; struct token_list* global_list in R10
  2520. ;; and struct token_list* FUNC in R9
  2521. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  2522. ;; Returns the token_lists modified
  2523. :collect_arguments
  2524. PUSHR R0 R15 ; Protect R0
  2525. PUSHR R1 R15 ; Protect R1
  2526. PUSHR R2 R15 ; Protect R2
  2527. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2528. :collect_arguments_iter
  2529. LOADUI R0 $close_paren ; Using ")"
  2530. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  2531. CALLI R15 @match ; IF GLOBAL_TOKEN->S == ")"
  2532. JUMP.NZ R0 @collect_arguments_done ; Be done
  2533. ;; Collect the arguments
  2534. CALLI R15 @type_name ; Get what type we are working with
  2535. MOVE R1 R0 ; Put TYPE where it will be used
  2536. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  2537. LOADU8 R0 R0 0 ; GLOBAL_TOKEN->S[0]
  2538. CMPSKIPI.NE R0 41 ; IF GLOBAL_TOKEN->S[0] == ')'
  2539. JUMP @collect_arguments_iter3 ; foo(int,char,void) doesn't need anything done
  2540. ;; Check for foo(int a,...)
  2541. CMPSKIPI.NE R0 41 ; IF GLOBAL_TOKEN->S[0] == ','
  2542. JUMP @collect_arguments_iter3 ; Looks like final case
  2543. ;; Deal with foo(int a, ...)
  2544. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  2545. LOAD32 R2 R9 16 ; FUNC->ARGUMENTS
  2546. CALLI R15 @sym_declare ; Get A
  2547. MOVE R2 R0 ; Get A out of the way
  2548. :collect_arguments_func
  2549. LOAD32 R0 R9 16 ; FUNC->ARGS
  2550. CMPSKIPI.E R0 0 ; IF NULL == FUNC->ARGS
  2551. LOAD32 R0 R0 16 ; FUNC->ARGS->DEPTH
  2552. SUBI R0 R0 8 ; FUNC->ARGS->DEPTH - 8 or NULL - 8 (-8)
  2553. STORE32 R0 R2 16 ; A->DEPTH = VALUE
  2554. :collect_arguments_iter2
  2555. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2556. STORE32 R2 R9 16 ; FUNC->ARGUMENTS = A
  2557. :collect_arguments_iter3
  2558. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  2559. LOADU8 R0 R0 0 ; GLOBAL_TOKEN->S[0]
  2560. CMPSKIPI.NE R0 44 ; IF GLOBAL_TOKEN->S[0] == ','
  2561. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2562. JUMP @collect_arguments_iter ; Keep looping
  2563. :collect_arguments_done
  2564. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2565. POPR R2 R15 ; Restore R2
  2566. POPR R1 R15 ; Restore R1
  2567. POPR R0 R15 ; Restore R0
  2568. RET R15
  2569. ;; declare_function function
  2570. ;; Receives struct token_list* global_token in R13,
  2571. ;; struct token_list* out in R12,
  2572. ;; struct token_list* string_list in R11
  2573. ;; and struct token_list* global_list in R10
  2574. ;; SETS R9 to struct token_list* FUNC
  2575. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  2576. ;; Returns the token_lists modified
  2577. :declare_function
  2578. PUSHR R0 R15 ; Protect R0
  2579. PUSHR R1 R15 ; Protect R1
  2580. PUSHR R2 R15 ; Protect R2
  2581. FALSE R0 ; Using Zero
  2582. STORER32 R0 @current_count ; CURRENT_COUNT = 0
  2583. LOAD32 R0 R13 4 ; GLOBAL_TOKEN->PREV
  2584. LOAD32 R0 R0 8 ; GLOBAL_TOKEN->PREV->S
  2585. FALSE R1 ; Passing NULL
  2586. LOADR32 R2 @global_function_list ; where the global function list is located
  2587. CALLI R15 @sym_declare ; declare FUNC
  2588. STORER32 R0 @global_function_list ; GLOBAL_FUNCTION_LIST = FUNC
  2589. MOVE R9 R0 ; SETS FUNC
  2590. CALLI R15 @collect_arguments ; Collect function arguments
  2591. LOAD32 R2 R13 8 ; GLOBAL_TOKEN->S
  2592. LOADU8 R0 R2 0 ; GLOBAL_TOKEN->S[0]
  2593. CMPSKIPI.NE R0 59 ; IF GLOBAL_TOKEN->S[0] == ';'
  2594. JUMP @declare_function_prototype ; Don't waste time
  2595. ;; Looks like it is an actual function definition
  2596. LOADUI R0 $declare_function_string0 ; Using first string
  2597. CALLI R15 @emit_out ; emit it
  2598. LOAD32 R0 R9 8 ; Using FUNC->S
  2599. CALLI R15 @emit_out ; emit it
  2600. LOADUI R0 $newline ; Using "\n"
  2601. CALLI R15 @emit_out ; emit it
  2602. LOADUI R0 $declare_function_string1 ; Using second string
  2603. CALLI R15 @emit_out ; emit it
  2604. LOAD32 R0 R9 8 ; Using FUNC->S
  2605. CALLI R15 @emit_out ; emit it
  2606. LOADUI R0 $newline ; Using "\n"
  2607. CALLI R15 @emit_out ; emit it
  2608. :declare_function_nonmain
  2609. FALSE R1 ; Cleaning up before call
  2610. CALLI R15 @statement ; Collect the statement
  2611. ;; Prevent Duplicate Returns
  2612. LOAD32 R1 R12 8 ; OUT->S
  2613. LOADUI R0 $declare_function_string2 ; Our final string
  2614. CALLI R15 @match ; Check for Match
  2615. JUMP.NZ R0 @declare_function_done ; Clean up
  2616. ;; Deal with adding the return
  2617. LOADUI R0 $declare_function_string2 ; Our final string
  2618. CALLI R15 @emit_out ; emit it
  2619. :declare_function_done
  2620. POPR R2 R15 ; Restore R2
  2621. POPR R1 R15 ; Restore R1
  2622. POPR R0 R15 ; Restore R0
  2623. RET R15
  2624. :declare_function_prototype
  2625. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2626. JUMP @declare_function_done ; Clean up
  2627. :declare_function_string0
  2628. "# Defining function "
  2629. :declare_function_string1
  2630. ":FUNCTION_"
  2631. :declare_function_string2
  2632. "RETURN
  2633. "
  2634. :current_count
  2635. NOP
  2636. ;; program function
  2637. ;; Receives struct token_list* global_token in R13,
  2638. ;; struct token_list* out in R12,
  2639. ;; struct token_list* string_list in R11
  2640. ;; and struct token_list* global_list in R10
  2641. ;; and struct token_list* FUNC in R9
  2642. ;; and struct token_list* current_target in R8
  2643. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  2644. ;; Returns the token_lists modified
  2645. :program
  2646. PUSHR R0 R15 ; Protect R0
  2647. PUSHR R1 R15 ; Protect R1
  2648. PUSHR R2 R15 ; Protect R2
  2649. PUSHR R3 R15 ; Protect R3
  2650. :program_iter
  2651. JUMP.Z R13 @program_done ; Looks like we read all the tokens
  2652. LOADUI R0 $constant ; Using the constant string
  2653. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  2654. CALLI R15 @match ; Check if they match
  2655. JUMP.Z R0 @program_type ; Looks like not
  2656. ;; Deal with CONSTANT case
  2657. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2658. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  2659. FALSE R1 ; Set NULL
  2660. LOADR32 R2 @global_constant_list ; GLOBAL_CONSTANTS_LIST
  2661. CALLI R15 @sym_declare ; Declare the global constant
  2662. STORER32 R0 @global_constant_list ; Update global constant
  2663. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2664. STORE32 R13 R0 16 ; GLOBAL_CONSTANT_LIST->ARGUMENTS = GLOBAL_TOKEN
  2665. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2666. JUMP @program_iter ; Loop again
  2667. :program_type
  2668. CALLI R15 @type_name ; Get the type
  2669. JUMP.Z R0 @program_iter ; If newly defined type iterate
  2670. ;; Looks like we got a defined type
  2671. MOVE R1 R0 ; Put the type where it can be used
  2672. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  2673. LOADUI R3 $global_symbol_list ; Get address of global symbol list
  2674. LOAD32 R2 R3 0 ; GLOBAL_SYMBOLS_LIST
  2675. CALLI R15 @sym_declare ; Declare that global symbol
  2676. STORE32 R0 R3 0 ; Update global symbol list
  2677. LOAD32 R3 R13 8 ; GLOBAL_TOKEN->S
  2678. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2679. LOADUI R0 $semicolon ; Get semicolon string
  2680. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  2681. CALLI R15 @match ; Check if they match
  2682. JUMP.Z R0 @program_function ; If not a match
  2683. ;; Deal with case of TYPE NAME;
  2684. COPY R1 R10 ; Using GLOBALS_LIST
  2685. LOADUI R0 $program_string0 ; Using the GLOBAL_ prefix
  2686. CALLI R15 @emit ; emit it
  2687. MOVE R1 R0 ; Move new GLOBALS_LIST into Place
  2688. MOVE R0 R3 ; Use GLOBAL_TOKEN->PREV->S
  2689. CALLI R15 @emit ; emit it
  2690. MOVE R1 R0 ; Move new GLOBALS_LIST into Place
  2691. LOADUI R0 $program_string1 ; Using the NOP postfix
  2692. CALLI R15 @emit ; emit it
  2693. MOVE R10 R0 ; Move new GLOBALS_LIST into Place
  2694. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2695. JUMP @program_iter
  2696. :program_function
  2697. LOADUI R0 $open_paren ; Get open paren string
  2698. CALLI R15 @match ; Check if they match
  2699. JUMP.Z R0 @program_assign ; If not a match
  2700. ;; Deal with case of TYPE NAME(...)
  2701. CALLI R15 @declare_function
  2702. JUMP @program_iter
  2703. :program_assign
  2704. LOADUI R0 $equal ; Get equal string
  2705. CALLI R15 @match ; Check if they match
  2706. JUMP.Z R0 @program_error ; If not a match
  2707. COPY R1 R10 ; Using GLOBALS_LIST
  2708. LOADUI R0 $program_string0 ; Using the GLOBAL_ prefix
  2709. CALLI R15 @emit ; emit it
  2710. MOVE R1 R0 ; Move new GLOBALS_LIST into Place
  2711. MOVE R0 R3 ; Use GLOBAL_TOKEN->PREV->S
  2712. CALLI R15 @emit ; emit it
  2713. MOVE R1 R0 ; Move new GLOBALS_LIST into Place
  2714. LOADUI R0 $newline ; Using the Newline postfix
  2715. CALLI R15 @emit ; emit it
  2716. MOVE R10 R0 ; Update GLOBALS_LIST
  2717. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2718. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  2719. LOADU8 R0 R0 0 ; GLOBAL_TOKEN->S[0]
  2720. LOADUI R1 $digit_chars ; 0-9
  2721. CALLI R15 @in_set ; Figure out if in set
  2722. JUMP.Z R0 @program_assign_string ; If not in sets
  2723. ;; Looks like we have an int
  2724. COPY R1 R10 ; Using GLOBALS_LIST
  2725. LOADUI R0 $percent ; Using percent prefix
  2726. CALLI R15 @emit ; emit it
  2727. MOVE R1 R0 ; Put GLOBALS_LIST into Place
  2728. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  2729. CALLI R15 @emit ; emit it
  2730. MOVE R1 R0 ; Put GLOBALS_LIST into Place
  2731. LOADUI R0 $newline ; Using newline postfix
  2732. CALLI R15 @emit ; emit it
  2733. MOVE R10 R0 ; Update GLOBALS_LIST
  2734. JUMP @program_assign_done ; Move on
  2735. :program_assign_string
  2736. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  2737. LOADU8 R0 R0 0 ; GLOBAL_TOKEN->S[0]
  2738. CMPSKIPI.E R0 34 ; If GLOBAL_TOKEN->S[0] == '"'
  2739. JUMP @program_error ; If not we hit an error
  2740. ;; Looks like we have a string
  2741. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  2742. CALLI R15 @parse_string ; Parse it into useful form
  2743. COPY R1 R10 ; GLOBALS_LIST
  2744. CALLI R15 @emit ; emit it
  2745. MOVE R10 R0 ; Update GLOBALS_LIST
  2746. :program_assign_done
  2747. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2748. LOADUI R0 $program_string4 ; Potential error message
  2749. LOADUI R1 $semicolon ; Checking for ;
  2750. CALLI R15 @require_match ; Catch those errors
  2751. JUMP @program_iter
  2752. :program_error
  2753. LOADUI R0 $program_string2 ; message part 1
  2754. FALSE R1 ; Show to user
  2755. CALLI R15 @file_print ; write
  2756. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  2757. CALLI R15 @file_print ; write
  2758. LOADUI R0 $program_string3 ; message part 2
  2759. CALLI R15 @file_print ; write
  2760. CALLI R15 @line_error ; Provide a meaningful error message
  2761. HALT
  2762. :program_done
  2763. POPR R3 R15 ; Restore R3
  2764. POPR R2 R15 ; Restore R2
  2765. POPR R1 R15 ; Restore R1
  2766. POPR R0 R15 ; Restore R0
  2767. RET R15
  2768. :program_string0
  2769. ":GLOBAL_"
  2770. :program_string1
  2771. "
  2772. NULL
  2773. "
  2774. :program_string2
  2775. "Received "
  2776. :program_string3
  2777. " in program
  2778. "
  2779. :program_string4
  2780. "ERROR in Program
  2781. Missing ;
  2782. "
  2783. ;; sym_declare function
  2784. ;; Receives char* in R0, struct type* in R1, struct token_list* in R2
  2785. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  2786. ;; Returns struct token_list* in R0
  2787. :sym_declare
  2788. STORE32 R2 R14 0 ; A->NEXT = LIST
  2789. STORE32 R0 R14 8 ; A->S = S
  2790. STORE32 R1 R14 12 ; A->TYPE = T
  2791. ADDUI R0 R14 20 ; CALLOC struct token_list
  2792. SWAP R0 R14 ; Prepare for Return
  2793. RET R15
  2794. ;; sym_lookup function
  2795. ;; Receives char* in R0 and struct token_list in R1
  2796. ;; Returns struct token_list* or NULL in R0
  2797. :sym_lookup
  2798. PUSHR R2 R15 ; Protect R2
  2799. MOVE R2 R1 ; Protect I
  2800. MOVE R1 R0 ; Put S in proper place
  2801. :sym_lookup_iter
  2802. JUMP.Z R2 @sym_lookup_done ; Stop if NULL
  2803. LOAD32 R0 R2 8 ; I->S
  2804. CALLI R15 @match ; if I->S == S
  2805. JUMP.NZ R0 @sym_lookup_done ; Stop if match
  2806. LOAD32 R2 R2 0 ; I = I->NEXT
  2807. JUMP @sym_lookup_iter ; Keep looping
  2808. :sym_lookup_done
  2809. MOVE R0 R2 ; Using R2 as our result
  2810. POPR R2 R15 ; Restore R2
  2811. RET R15
  2812. ;; function_call function
  2813. ;; Receives CHAR* S in R0 and INT BOOL in R1
  2814. ;; struct token_list* out in R12,
  2815. ;; struct token_list* string_list in R11
  2816. ;; and struct token_list* global_list in R10
  2817. ;; and struct token_list* FUNC in R9
  2818. ;; and struct token_list* current_target in R8
  2819. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  2820. ;; Returns the token_lists modified
  2821. :function_call
  2822. PUSHR R0 R15 ; Protect R0
  2823. PUSHR R1 R15 ; Protect R1
  2824. PUSHR R2 R15 ; Protect R2
  2825. PUSHR R3 R15 ; Protect R3
  2826. PUSHR R4 R15 ; Protect R4
  2827. MOVE R2 R0 ; Protect S
  2828. MOVE R3 R1 ; Protect BOOL
  2829. LOADUI R0 $function_call_string0 ; Our first error message
  2830. LOADUI R1 $open_paren ; Using "("
  2831. CALLI R15 @require_match ; Make sure of a match
  2832. FALSE R4 ; PASSED = 0
  2833. LOADUI R0 $function_call_string1 ; Our first header
  2834. CALLI R15 @emit_out ; emit it
  2835. LOADUI R0 $function_call_string2 ; Our second header
  2836. CALLI R15 @emit_out ; emit it
  2837. LOADUI R0 $function_call_string3 ; Our third header
  2838. CALLI R15 @emit_out ; emit it
  2839. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  2840. LOADU8 R0 R0 0 ; GLOBAL_TOKEN->S[0]
  2841. CMPSKIPI.NE R0 41 ; IF GLOBAL_TOKEN->S[0] != ')'
  2842. JUMP @function_call_collect_done ; looks like function()
  2843. ;; Collect arguments
  2844. CALLI R15 @expression ; Deal with first argument
  2845. LOADUI R0 $function_call_string4 ; Push it onto stack
  2846. CALLI R15 @emit_out ; emit it
  2847. ADDUI R4 R4 1 ; PASSED = 1
  2848. :function_call_collect_iter
  2849. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  2850. LOADU8 R0 R0 0 ; GLOBAL_TOKEN->S[0]
  2851. CMPSKIPI.E R0 44 ; IF GLOBAL_TOKEN->S[0] != ','
  2852. JUMP @function_call_collect_done ; looks like we are done collecting arguments
  2853. ;; Collect another argument
  2854. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  2855. CALLI R15 @expression ; Deal with Nth argument
  2856. LOADUI R0 $function_call_string5 ; Push it onto stack
  2857. CALLI R15 @emit_out ; emit it
  2858. ADDUI R4 R4 1 ; PASSED = PASSED + 1
  2859. JUMP @function_call_collect_iter ; Keep looping
  2860. :function_call_collect_done
  2861. LOADUI R0 $function_call_string6 ; Our second error message
  2862. LOADUI R1 $close_paren ; Using ")"
  2863. CALLI R15 @require_match ; Make sure of a match
  2864. JUMP.Z R3 @function_call_call_false ; if BOOL != TRUE
  2865. ;; Deal with TRUE == BOOL
  2866. LOADUI R0 $function_call_string7 ; Our first prefix
  2867. CALLI R15 @emit_out ; emit it
  2868. MOVE R0 R2 ; Using S
  2869. CALLI R15 @emit_out ; emit it
  2870. LOADUI R0 $function_call_string8 ; Our first postfix
  2871. CALLI R15 @emit_out ; emit it
  2872. LOADUI R0 $function_call_string9 ; Our second postfix
  2873. CALLI R15 @emit_out ; emit it
  2874. LOADUI R0 $function_call_string10 ; Our last postfix
  2875. CALLI R15 @emit_out ; emit it
  2876. JUMP @function_call_call_done ; Move on
  2877. :function_call_call_false
  2878. ;; Deal with FALSE == BOOL
  2879. LOADUI R0 $function_call_string11 ; Our first prefix
  2880. CALLI R15 @emit_out ; emit it
  2881. LOADUI R0 $function_call_string12 ; Our last prefix
  2882. CALLI R15 @emit_out ; emit it
  2883. MOVE R0 R2 ; Using S
  2884. CALLI R15 @emit_out ; emit it
  2885. LOADUI R0 $newline ; Using "\n"
  2886. CALLI R15 @emit_out ; emit it
  2887. :function_call_call_done
  2888. LOADUI R0 $function_call_string13 ; Our POP
  2889. :function_call_pop_iter
  2890. JUMP.Z R4 @function_call_pop_done ; Skip POP if out of args on Stack
  2891. CALLI R15 @emit_out ; emit our POP
  2892. SUBI R4 R4 1 ; PASSED = PASSED - 1
  2893. JUMP @function_call_pop_iter ; Loop
  2894. :function_call_pop_done
  2895. LOADUI R0 $function_call_string14 ; Our first postfix
  2896. CALLI R15 @emit_out ; emit it
  2897. LOADUI R0 $function_call_string15 ; Our final postfix
  2898. CALLI R15 @emit_out ; emit it
  2899. POPR R4 R15 ; Restore R4
  2900. POPR R3 R15 ; Restore R3
  2901. POPR R2 R15 ; Restore R2
  2902. POPR R1 R15 ; Restore R1
  2903. POPR R0 R15 ; Restore R0
  2904. RET R15
  2905. :function_call_string0
  2906. "ERROR in process_expression_list
  2907. No ( was found
  2908. "
  2909. :function_call_string1
  2910. "PUSH_RDI # Prevent overwriting in recursion
  2911. "
  2912. :function_call_string2
  2913. "PUSH_RBP # Protect the old base pointer
  2914. "
  2915. :function_call_string3
  2916. "COPY_RSP_to_RDI # Copy new base pointer
  2917. "
  2918. :function_call_string4
  2919. "PUSH_RAX #_process_expression1
  2920. "
  2921. :function_call_string5
  2922. "PUSH_RAX #_process_expression2
  2923. "
  2924. :function_call_string6
  2925. "ERROR in process_expression_list
  2926. No ) was found
  2927. "
  2928. :function_call_string7
  2929. "LOAD_BASE_ADDRESS_rax %"
  2930. :function_call_string8
  2931. "
  2932. LOAD_INTEGER
  2933. "
  2934. :function_call_string9
  2935. "COPY_rdi_to_rbp
  2936. "
  2937. :function_call_string10
  2938. "CALL_rax
  2939. "
  2940. :function_call_string11
  2941. "COPY_rdi_to_rbp
  2942. "
  2943. :function_call_string12
  2944. "CALL_IMMEDIATE %FUNCTION_"
  2945. :function_call_string13
  2946. "POP_RBX # _process_expression_locals
  2947. "
  2948. :function_call_string14
  2949. "POP_RBP # Restore old base pointer
  2950. "
  2951. :function_call_string15
  2952. "POP_RDI # Prevent overwrite
  2953. "
  2954. ;; emit function
  2955. ;; Receives char* in R0, struct token_list* in R1
  2956. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  2957. ;; Returns struct token_list* in R0
  2958. :emit
  2959. PUSHR R2 R15 ; Protect R2
  2960. COPY R2 R14 ; Pointer to T
  2961. ADDUI R14 R14 20 ; CALLOC struct token_list
  2962. STORE32 R1 R2 0 ; T->NEXT = HEAD
  2963. STORE32 R0 R2 8 ; T->S = S
  2964. MOVE R0 R2 ; Put T in proper spot for return
  2965. POPR R2 R15 ; Restore R2
  2966. RET R15
  2967. ;; emit_out function
  2968. ;; Receives char* in R0
  2969. ;; struct token_list* out in R12,
  2970. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  2971. ;; Returns struct token_list* in R0
  2972. :emit_out
  2973. STORE32 R12 R14 0 ; T->NEXT = OUT
  2974. ADDUI R12 R14 20 ; Get T
  2975. SWAP R12 R14 ; CALLOC struct token_list
  2976. STORE32 R0 R12 8 ; T->S = S
  2977. RET R15
  2978. ;; uniqueID function
  2979. ;; Receives char* in R0, struct token_list* in R1 and char* in R2
  2980. ;; Calls emit repeatedly
  2981. ;; Returns struct token_list* in R0
  2982. :uniqueID
  2983. CALLI R15 @emit ; emit S
  2984. MOVE R1 R0 ; Put L in the correct place
  2985. LOADUI R0 $underline ; Using "_"
  2986. CALLI R15 @emit ; emit it
  2987. MOVE R1 R0 ; Put L in the correct place
  2988. COPY R0 R2 ; Put NUM in the correct place
  2989. CALLI R15 @emit ; emit NUM
  2990. MOVE R1 R0 ; Put L in the correct place
  2991. LOADUI R0 $newline ; Using "\n"
  2992. CALLI R15 @emit ; emit it
  2993. RET R15
  2994. ;; uniqueID_out function
  2995. ;; Receives char* in R0, char* in R1
  2996. ;; Calls emit_out repeatedly
  2997. ;; Returns nothing
  2998. :uniqueID_out
  2999. CALLI R15 @emit_out ; emit S
  3000. LOADUI R0 $underline ; Using "_"
  3001. CALLI R15 @emit_out ; emit it
  3002. COPY R0 R1 ; Put NUM in the correct place
  3003. CALLI R15 @emit_out ; emit NUM
  3004. LOADUI R0 $newline ; Using "\n"
  3005. CALLI R15 @emit_out ; emit it
  3006. RET R15
  3007. ;; file_print function
  3008. ;; Receives pointer to string in R0 and FILE* in R1
  3009. ;; Returns nothing
  3010. :file_print
  3011. PUSHR R2 R15 ; Protect R2 from Overwrite
  3012. MOVE R2 R0 ; Put string pointer into place
  3013. :file_print_read
  3014. LOAD8 R0 R2 0 ; Get a char
  3015. JUMP.Z R0 @file_print_done ; If NULL be done
  3016. FPUTC ; Write the Char
  3017. ADDUI R2 R2 1 ; Point at next CHAR
  3018. JUMP @file_print_read ; Loop again
  3019. :file_print_done
  3020. POPR R2 R15 ; Restore R2
  3021. RET R15
  3022. ;; recursive_output function
  3023. ;; Receives token_list in R0 and FILE* in R1
  3024. ;; Returns nothing and alters nothing
  3025. :recursive_output
  3026. JUMP.Z R0 @recursive_output_abort ; Abort if NULL
  3027. PUSHR R2 R15 ; Preserve R2 from recursion
  3028. MOVE R2 R0 ; Preserve R0 from recursion
  3029. LOAD32 R0 R2 0 ; Using I->NEXT
  3030. CALLI R15 @recursive_output ; Recurse
  3031. LOAD32 R0 R2 8 ; Using I->S
  3032. CALLI R15 @file_print ; Write the string
  3033. MOVE R0 R2 ; Put R0 back
  3034. POPR R2 R15 ; Restore R0
  3035. :recursive_output_abort
  3036. RET R15
  3037. ;; match function
  3038. ;; Receives a CHAR* in R0, CHAR* in R1
  3039. ;; Returns Bool in R0 indicating if strings match
  3040. :match
  3041. PUSHR R1 R15 ; Protect R1
  3042. PUSHR R2 R15 ; Protect R2
  3043. PUSHR R3 R15 ; Protect R3
  3044. PUSHR R4 R15 ; Protect R4
  3045. MOVE R2 R0 ; Put First string in place
  3046. MOVE R3 R1 ; Put Second string in place
  3047. LOADUI R4 0 ; Set initial index of 0
  3048. :match_cmpbyte
  3049. LOADXU8 R0 R2 R4 ; Get a byte of our first string
  3050. LOADXU8 R1 R3 R4 ; Get a byte of our second string
  3051. ADDUI R4 R4 1 ; Prep for next loop
  3052. CMPSKIP.NE R1 R0 ; Compare the bytes
  3053. JUMP.NZ R1 @match_cmpbyte ; Loop if bytes are equal
  3054. ;; Done
  3055. FALSE R2 ; Default answer
  3056. CMPSKIP.NE R0 R1 ; If ended loop with everything matching
  3057. TRUE R2 ; Set as TRUE
  3058. MOVE R0 R2 ; Prepare for return
  3059. POPR R4 R15 ; Restore R4
  3060. POPR R3 R15 ; Restore R3
  3061. POPR R2 R15 ; Restore R2
  3062. POPR R1 R15 ; Restore R1
  3063. RET R15
  3064. ;; lookup_type function
  3065. ;; Receives a CHAR* in R0 and struct type* in R1
  3066. ;; Returns struct type* in R0 or NULL if no match
  3067. :lookup_type
  3068. PUSHR R1 R15 ; Protect R1
  3069. PUSHR R2 R15 ; Protect R2
  3070. MOVE R2 R1 ; Put START in correct place
  3071. MOVE R1 R0 ; Put S in correct place
  3072. :lookup_type_iter
  3073. LOAD32 R0 R2 24 ; Get I->NAME
  3074. CALLI R15 @match ; Check if I->NAME == S
  3075. JUMP.NZ R0 @lookup_type_done ; If match found be done
  3076. LOAD32 R2 R2 0 ; I = I->NEXT
  3077. JUMP.NZ R2 @lookup_type_iter ; Otherwise iterate until I == NULL
  3078. :lookup_type_done
  3079. MOVE R0 R2 ; Our answer (I or NULL)
  3080. POPR R2 R15 ; Restore R2
  3081. POPR R1 R15 ; Restore R1
  3082. RET R15
  3083. ;; lookup_member function
  3084. ;; Receives struct type* parent in R0 and char* name in R1
  3085. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  3086. ;; Returns struct type* of member in R0 or aborts with error
  3087. :lookup_member
  3088. PUSHR R1 R15 ; Protect R1
  3089. PUSHR R2 R15 ; Protect R2
  3090. PUSHR R3 R15 ; Protect R3
  3091. LOAD32 R3 R0 24 ; PARENT->NAME for error
  3092. MOVE R2 R0 ; I = PARENT
  3093. :lookup_member_iter
  3094. LOAD32 R2 R2 16 ; I = I->MEMBERS
  3095. JUMP.Z R2 @lookup_member_error ; We failed hard
  3096. LOAD32 R0 R2 24 ; I->NAME
  3097. CALLI R15 @match ; IF I->NAME == NAME
  3098. JUMP.Z R0 @lookup_member_iter ; Loop again
  3099. :lookup_member_done
  3100. MOVE R0 R2 ; Put I in the correct place
  3101. POPR R3 R15 ; Restore R3
  3102. POPR R2 R15 ; Restore R2
  3103. POPR R1 R15 ; Restore R1
  3104. RET R15
  3105. :lookup_member_error
  3106. FALSE R1 ; Write to TTY
  3107. LOADUI R0 $lookup_member_string0 ; Our header string
  3108. CALLI R15 @file_print ; Print it
  3109. MOVE R0 R3 ; Using PARENT->NAME
  3110. CALLI R15 @file_print ; Print it
  3111. LOADUI R0 $arrow_string ; Using "->"
  3112. CALLI R15 @file_print ; Print it
  3113. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  3114. CALLI R15 @file_print ; Print it
  3115. LOADUI R0 $lookup_member_string1 ; Our footer string
  3116. CALLI R15 @file_print ; Print it
  3117. CALLI R15 @line_error ; Give line info
  3118. LOADUI R0 $newline ; Our final addition
  3119. CALLI R15 @file_print ; Print it
  3120. HALT
  3121. :lookup_member_string0
  3122. "ERROR in lookup_member "
  3123. :lookup_member_string1
  3124. " does not exist
  3125. "
  3126. ;; build_member function
  3127. ;; Receives a struct type* in R0, int in R1 and int in R2
  3128. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  3129. ;; Modifies R2 to current member_size
  3130. ;; Returns struct type* in R0
  3131. :build_member
  3132. PUSHR R3 R15 ; Protect R3
  3133. PUSHR R4 R15 ; Protect R4
  3134. PUSHR R5 R15 ; Protect R5
  3135. PUSHR R6 R15 ; Protect R6
  3136. MOVE R6 R1 ; Protect OFFSET
  3137. MOVE R4 R0 ; Protect LAST
  3138. CALLI R15 @type_name ; Get MEMBER_TYPE
  3139. MOVE R5 R0 ; Protect MEMBER_TYPE
  3140. ADDUI R3 R14 28 ; CALLOC struct type
  3141. SWAP R3 R14 ; SET I
  3142. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  3143. STORE32 R0 R3 24 ; I->NAME = GLOBAL_TOKEN->S
  3144. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  3145. STORE32 R4 R3 16 ; I->MEMBERS = LAST
  3146. LOADUI R0 $open_bracket ; Using "["
  3147. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  3148. CALLI R15 @match ; IF GLOBAL_TOKEN->S == "["
  3149. JUMP.Z R0 @build_member_single
  3150. ;; Deal with type name [ number ] ;
  3151. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  3152. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  3153. CALLI R15 @numerate_string ; Convert string to int NUMBER
  3154. LOAD32 R1 R5 20 ; MEMBER_TYPE->TYPE
  3155. LOAD32 R1 R1 4 ; MEMBER_TYPE->TYPE->SIZE
  3156. MULU R0 R0 R1 ; MEMBER_TYPE->TYPE->SIZE * NUMBER
  3157. STORE32 R0 R3 4 ; I->SIZE = MEMBER_TYPE->TYPE->SIZE * NUMBER
  3158. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  3159. LOADUI R0 $build_member_string0 ; Our error message
  3160. LOADUI R1 $close_bracket ; Using "]"
  3161. CALLI R15 @require_match ; Make sure it is right
  3162. JUMP @build_member_done ; Skip over single steps
  3163. :build_member_single
  3164. LOAD32 R0 R5 4 ; MEMBER_TYPE->SIZE
  3165. STORE32 R0 R3 4 ; I->SIZE = MEMBER_TYPE->SIZE
  3166. :build_member_done
  3167. LOAD32 R2 R3 4 ; MEMBER_SIZE = I->SIZE
  3168. STORE32 R5 R3 20 ; I->TYPE = MEMBER_TYPE
  3169. STORE32 R6 R3 8 ; I->OFFSET = OFFSET
  3170. MOVE R1 R6 ; Restore OFFSET
  3171. MOVE R0 R3 ; RETURN I in R0
  3172. POPR R6 R15 ; Restore R6
  3173. POPR R5 R15 ; Restore R5
  3174. POPR R4 R15 ; Restore R4
  3175. POPR R3 R15 ; Restore R3
  3176. RET R15
  3177. :build_member_string0
  3178. "Struct only supports [num] form
  3179. "
  3180. ;; build_union function
  3181. ;; Receives a struct type* in R0, int in R1 and int in R2
  3182. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  3183. ;; Modifies R2 to current member_size
  3184. ;; Returns struct type* in R0
  3185. :build_union
  3186. PUSHR R3 R15 ; Protect R3
  3187. PUSHR R4 R15 ; Protect R4
  3188. PUSHR R5 R15 ; Protect R5
  3189. MOVE R4 R0 ; Protect LAST
  3190. MOVE R3 R1 ; Protect OFFSET
  3191. FALSE R5 ; SIZE = 0
  3192. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  3193. LOADUI R0 $build_union_string0 ; ERROR MESSAGE
  3194. LOADUI R1 $open_curly_brace ; OPEN CURLY BRACE
  3195. CALLI R15 @require_match ; Ensure we have that curly brace
  3196. :build_union_iter
  3197. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  3198. LOADU8 R0 R0 0 ; GLOBAL_TOKEN->S[0]
  3199. LOADUI R1 125 ; numerical value of }
  3200. CMPJUMPI.E R0 R1 @build_union_done ; No more looping required
  3201. MOVE R0 R4 ; We are passing last to be overwritten
  3202. MOVE R1 R3 ; We are also passing OFFSET
  3203. CALLI R15 @build_member ; To build_member to get new LAST and new member_size
  3204. CMPSKIP.LE R2 R5 ; If MEMBER_SIZE > SIZE
  3205. COPY R5 R2 ; SIZE = MEMMER_SIZE
  3206. MOVE R4 R0 ; Protect LAST
  3207. MOVE R3 R1 ; Protect OFFSET
  3208. LOADUI R0 $build_union_string1 ; ERROR MESSAGE
  3209. LOADUI R1 $semicolon ; SEMICOLON
  3210. CALLI R15 @require_match ; Ensure we have that curly brace
  3211. JUMP @build_union_iter ; Loop until we get that closing curly brace
  3212. :build_union_done
  3213. MOVE R2 R5 ; Setting MEMBER_SIZE = SIZE
  3214. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  3215. MOVE R1 R3 ; Restore OFFSET
  3216. MOVE R0 R4 ; Restore LAST as we are turning that
  3217. POPR R5 R15 ; Restore R5
  3218. POPR R4 R15 ; Restore R4
  3219. POPR R3 R15 ; Restore R3
  3220. RET R15
  3221. :build_union_string0
  3222. "ERROR in build_union
  3223. Missing {
  3224. "
  3225. :build_union_string1
  3226. "ERROR in build_union
  3227. Missing ;
  3228. "
  3229. ;; create_struct function
  3230. ;; Receives Nothing
  3231. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  3232. ;; Returns Nothing
  3233. :create_struct
  3234. PUSHR R0 R15 ; Protect R0
  3235. PUSHR R1 R15 ; Protect R1
  3236. PUSHR R2 R15 ; Protect R2
  3237. PUSHR R3 R15 ; Protect R3
  3238. PUSHR R4 R15 ; Protect R4
  3239. PUSHR R5 R15 ; Protect R5
  3240. PUSHR R6 R15 ; Protect R6
  3241. FALSE R5 ; OFFSET = 0
  3242. FALSE R2 ; MEMBER_SIZE = 0
  3243. COPY R3 R14 ; SET HEAD
  3244. ADDUI R14 R14 28 ; CALLOC struct type
  3245. COPY R4 R14 ; SET I
  3246. ADDUI R14 R14 28 ; CALLOC struct type
  3247. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  3248. STORE32 R0 R3 24 ; HEAD->NAME = GLOBAL_TOKEN->S
  3249. STORE32 R0 R4 24 ; I->NAME = GLOBAL_TOKEN->S
  3250. STORE32 R4 R3 12 ; HEAD->INDIRECT = I
  3251. STORE32 R3 R4 12 ; I->INDIRECT - HEAD
  3252. LOADR32 R0 @global_types ; Get Address of GLOBAL_TYPES
  3253. STORE R0 R3 0 ; HEAD->NEXT = GLOBAL_TYPES
  3254. STORER32 R3 @global_types ; GLOBAL_TYPES = HEAD
  3255. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  3256. LOADUI R0 8 ; Standard Pointer SIZE
  3257. STORE32 R0 R4 4 ; I->SIZE = 8
  3258. LOADUI R0 $create_struct_string0 ; ERROR MESSAGE
  3259. LOADUI R1 $open_curly_brace ; OPEN CURLY BRACE
  3260. CALLI R15 @require_match ; Ensure we have that curly brace
  3261. FALSE R6 ; LAST = NULL
  3262. :create_struct_iter
  3263. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  3264. LOADU8 R0 R0 0 ; GLOBAL_TOKEN->S[0]
  3265. LOADUI R1 125 ; Numerical value of }
  3266. CMPJUMPI.E R0 R1 @create_struct_done ; Stop looping if match
  3267. LOADUI R1 $union ; Pointer to string UNION
  3268. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  3269. CALLI R15 @match ; Check if they Match
  3270. SWAP R0 R6 ; Put LAST in place
  3271. MOVE R1 R5 ; Put OFFSET in place
  3272. JUMP.NZ R6 @create_struct_union ; Deal with union case
  3273. ;; Deal with standard member case
  3274. CALLI R15 @build_member ; Sets new LAST and MEMBER_SIZE
  3275. JUMP @create_struct_iter2 ; reset for loop
  3276. :create_struct_union
  3277. CALLI R15 @build_union
  3278. :create_struct_iter2
  3279. ADD R5 R1 R2 ; OFFSET = OFFSET + MEMBER_SIZE
  3280. SWAP R0 R6 ; Put LAST in place
  3281. LOADUI R0 $create_struct_string1 ; ERROR MESSAGE
  3282. LOADUI R1 $semicolon ; SEMICOLON
  3283. CALLI R15 @require_match ; Ensure we have that semicolon
  3284. JUMP @create_struct_iter ; Keep Looping
  3285. :create_struct_done
  3286. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  3287. LOADUI R0 $create_struct_string1 ; ERROR MESSAGE
  3288. LOADUI R1 $semicolon ; SEMICOLON
  3289. CALLI R15 @require_match ; Ensure we have that semicolon
  3290. STORE32 R5 R3 4 ; HEAD->SIZE = OFFSET
  3291. STORE32 R6 R3 16 ; HEAD->MEMBERS = LAST
  3292. STORE32 R6 R4 16 ; I->MEMBERS = LAST
  3293. POPR R6 R15 ; Restore R6
  3294. POPR R5 R15 ; Restore R5
  3295. POPR R4 R15 ; Restore R4
  3296. POPR R3 R15 ; Restore R3
  3297. POPR R2 R15 ; Restore R2
  3298. POPR R1 R15 ; Restore R1
  3299. POPR R0 R15 ; Restore R0
  3300. RET R15
  3301. :create_struct_string0
  3302. "ERROR in create_struct
  3303. Missing {
  3304. "
  3305. :create_struct_string1
  3306. "ERROR in create_struct
  3307. Missing ;
  3308. "
  3309. ;; type_name function
  3310. ;; Receives Nothing
  3311. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  3312. ;; Returns struct type* in R0
  3313. :type_name
  3314. PUSHR R1 R15 ; Protect R1
  3315. PUSHR R2 R15 ; Protect R2
  3316. LOADUI R0 $struct ; String for struct for comparison
  3317. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  3318. CALLI R15 @match ; Check if they match
  3319. CMPSKIPI.E R0 0 ; If STRUCTURE
  3320. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  3321. LOAD32 R2 R13 8 ; GLOBAL_TOKEN->S
  3322. LOADR32 R1 @global_types ; Check using the GLOBAL TYPES LIST
  3323. SWAP R0 R2 ; Put GLOBAL_TOKEN->S in the right place
  3324. CALLI R15 @lookup_type ; RET = lookup_type(GLOBAL_TOKEN->S)
  3325. MOVE R1 R2 ; Put STRUCTURE in the right place
  3326. CMPSKIP.E R0 R1 ; If RET == NULL and !STRUCTURE
  3327. JUMP @type_name_struct ; Guess not
  3328. ;; Exit with useful error message
  3329. FALSE R1 ; We will want to be writing the error message for the Human
  3330. LOADUI R0 $type_name_string0 ; The first string
  3331. CALLI R15 @file_print ; Display it
  3332. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  3333. CALLI R15 @file_print ; Display it
  3334. LOADUI R0 $newline ; Terminating linefeed
  3335. CALLI R15 @file_print ; Display it
  3336. CALLI R15 @line_error ; Give useful debug info
  3337. HALT ; Just exit
  3338. :type_name_struct
  3339. JUMP.NZ R0 @type_name_iter ; If was found
  3340. CALLI R15 @create_struct ; Otherwise create it
  3341. JUMP @type_name_done ; and be done
  3342. :type_name_iter
  3343. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  3344. LOAD32 R1 R13 8 ; GLOBAL_TOKEN->S
  3345. LOADU8 R1 R1 0 ; GLOBAL_TOKEN->S[0]
  3346. CMPSKIPI.E R1 42 ; if GLOBAL_TOKEN->S[0] == '*'
  3347. JUMP @type_name_done ; Looks like Nope
  3348. LOAD32 R0 R0 12 ; RET = RET->INDIRECT
  3349. JUMP @type_name_iter ; Keep looping
  3350. :type_name_done
  3351. POPR R2 R15 ; Restore R2
  3352. POPR R1 R15 ; Restore R1
  3353. RET R15
  3354. :type_name_string0
  3355. "Unknown type "
  3356. ;; line_error function
  3357. ;; Receives Nothing
  3358. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  3359. ;; Returns nothing
  3360. :line_error
  3361. PUSHR R0 R15 ; Protect R0
  3362. PUSHR R1 R15 ; Protect R1
  3363. LOADUI R0 $line_error_string0 ; Our leading string
  3364. FALSE R1 ; We want the user to see
  3365. CALLI R15 @file_print ; Print it
  3366. LOAD32 R0 R13 16 ; GLOBAL_TOKEN->LINENUMBER
  3367. CALLI R15 @numerate_number ; Get a string pointer for number
  3368. CALLI R15 @file_print ; And print it
  3369. POPR R1 R15 ; Restore R1
  3370. POPR R0 R15 ; Restore R0
  3371. RET R15
  3372. :line_error_string0
  3373. "In file: TTY1 On line: "
  3374. ;; require_match function
  3375. ;; Receives char* in R0 and char* in R1
  3376. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  3377. ;; Returns Nothing
  3378. :require_match
  3379. PUSHR R0 R15 ; Protect R0
  3380. PUSHR R2 R15 ; Protect R2
  3381. MOVE R2 R0 ; Get MESSAGE out of the way
  3382. LOAD32 R0 R13 8 ; GLOBAL_TOKEN->S
  3383. CALLI R15 @match ; Check if GLOBAL_TOKEN->S == REQUIRED
  3384. JUMP.NZ R0 @require_match_done ; Looks like it was a match
  3385. ;; Terminate with an error
  3386. MOVE R0 R2 ; Put MESSAGE in required spot
  3387. FALSE R1 ; We want to write for user
  3388. CALLI R15 @file_print ; Write it
  3389. CALLI R15 @line_error ; And provide some debug info
  3390. HALT ; Then Stop immediately
  3391. :require_match_done
  3392. LOAD32 R13 R13 0 ; GLOBAL_TOKEN = GLOBAL_TOKEN->NEXT
  3393. POPR R2 R15 ; Restore R2
  3394. POPR R0 R15 ; Restore R0
  3395. RET R15
  3396. ;; numerate_number function
  3397. ;; Receives int in R0
  3398. ;; R13 Holds pointer to global_token, R14 is HEAP Pointer
  3399. ;; Returns pointer to string generated
  3400. :numerate_number
  3401. PUSHR R1 R15 ; Preserve R1
  3402. PUSHR R2 R15 ; Preserve R2
  3403. PUSHR R3 R15 ; Preserve R3
  3404. PUSHR R4 R15 ; Preserve R4
  3405. PUSHR R5 R15 ; Preserve R5
  3406. PUSHR R6 R15 ; Preserve R6
  3407. MOVE R3 R0 ; Move Integer out of the way
  3408. COPY R1 R14 ; Get pointer result
  3409. ADDUI R14 R14 16 ; CALLOC the 16 chars of space
  3410. FALSE R6 ; Set index to 0
  3411. JUMP.Z R3 @numerate_number_ZERO ; Deal with Special case of ZERO
  3412. JUMP.P R3 @numerate_number_Positive
  3413. LOADUI R0 45 ; Using -
  3414. STOREX8 R0 R1 R6 ; write leading -
  3415. ADDUI R6 R6 1 ; Increment by 1
  3416. NOT R3 R3 ; Flip into positive
  3417. ADDUI R3 R3 1 ; Adjust twos
  3418. :numerate_number_Positive
  3419. LOADR R2 @Max_Decimal ; Starting from the Top
  3420. LOADUI R5 10 ; We move down by 10
  3421. FALSE R4 ; Flag leading Zeros
  3422. :numerate_number_0
  3423. DIVIDE R0 R3 R3 R2 ; Break off top 10
  3424. CMPSKIPI.E R0 0 ; If Not Zero
  3425. TRUE R4 ; Flip the Flag
  3426. JUMP.Z R4 @numerate_number_1 ; Skip leading Zeros
  3427. ADDUI R0 R0 48 ; Shift into ASCII
  3428. STOREX8 R0 R1 R6 ; write digit
  3429. ADDUI R6 R6 1 ; Increment by 1
  3430. :numerate_number_1
  3431. DIV R2 R2 R5 ; Look at next 10
  3432. CMPSKIPI.E R2 0 ; If we reached the bottom STOP
  3433. JUMP @numerate_number_0 ; Otherwise keep looping
  3434. :numerate_number_done
  3435. FALSE R0 ; NULL Terminate
  3436. STOREX8 R0 R1 R6 ; write
  3437. MOVE R0 R1 ; Return pointer to our string
  3438. ;; Cleanup
  3439. POPR R6 R15 ; Restore R6
  3440. POPR R5 R15 ; Restore R5
  3441. POPR R4 R15 ; Restore R4
  3442. POPR R3 R15 ; Restore R3
  3443. POPR R2 R15 ; Restore R2
  3444. POPR R1 R15 ; Restore R1
  3445. RET R15
  3446. :numerate_number_ZERO
  3447. LOADUI R0 48 ; Using Zero
  3448. STOREX8 R0 R1 R6 ; write
  3449. ADDUI R6 R6 1 ; Increment by 1
  3450. JUMP @numerate_number_done ; Be done
  3451. :Max_Decimal
  3452. '3B9ACA00'
  3453. ;; numerate_string function
  3454. ;; Receives pointer To string in R0
  3455. ;; Returns number in R0 equal to value of string
  3456. ;; Or Zero in the event of invalid string
  3457. :numerate_string
  3458. PUSHR R1 R15 ; Protect R1
  3459. PUSHR R2 R15 ; Protect R2
  3460. PUSHR R3 R15 ; Protect R3
  3461. PUSHR R4 R15 ; Protect R4
  3462. ;; Initialize
  3463. MOVE R1 R0 ; Get Text pointer out of the way
  3464. FALSE R2 ; Set Negative flag to false
  3465. FALSE R3 ; Set current count to Zero
  3466. LOAD8 R0 R1 1 ; Get second byte
  3467. CMPSKIPI.NE R0 120 ; If the second byte is x
  3468. JUMP @numerate_string_hex ; treat string like hex
  3469. ;; Deal with Decimal input
  3470. LOADUI R4 10 ; Multiply by 10
  3471. LOAD8 R0 R1 0 ; Get a byte
  3472. CMPSKIPI.NE R0 45 ; If - toggle flag
  3473. TRUE R2 ; So that we know to negate
  3474. CMPSKIPI.E R2 0 ; If toggled
  3475. ADDUI R1 R1 1 ; Move to next
  3476. :numerate_string_dec
  3477. LOAD8 R0 R1 0 ; Get a byte
  3478. CMPSKIPI.NE R0 0 ; If NULL
  3479. JUMP @numerate_string_done ; Be done
  3480. MUL R3 R3 R4 ; Shift counter by 10
  3481. SUBI R0 R0 48 ; Convert ascii to number
  3482. CMPSKIPI.GE R0 0 ; If less than a number
  3483. JUMP @numerate_string_done ; Terminate NOW
  3484. CMPSKIPI.L R0 10 ; If more than a number
  3485. JUMP @numerate_string_done ; Terminate NOW
  3486. ADDU R3 R3 R0 ; Don't add to the count
  3487. ADDUI R1 R1 1 ; Move onto next byte
  3488. JUMP @numerate_string_dec
  3489. ;; Deal with Hex input
  3490. :numerate_string_hex
  3491. LOAD8 R0 R1 0 ; Get a byte
  3492. CMPSKIPI.E R0 48 ; All hex strings start with 0x
  3493. JUMP @numerate_string_done ; Be done if not a match
  3494. ADDUI R1 R1 2 ; Move to after leading 0x
  3495. :numerate_string_hex_0
  3496. LOAD8 R0 R1 0 ; Get a byte
  3497. CMPSKIPI.NE R0 0 ; If NULL
  3498. JUMP @numerate_string_done ; Be done
  3499. SL0I R3 4 ; Shift counter by 16
  3500. SUBI R0 R0 48 ; Convert ascii number to number
  3501. CMPSKIPI.L R0 10 ; If A-F
  3502. SUBI R0 R0 7 ; Shove into Range
  3503. CMPSKIPI.L R0 16 ; If a-f
  3504. SUBI R0 R0 32 ; Shove into Range
  3505. ADDU R3 R3 R0 ; Add to the count
  3506. ADDUI R1 R1 1 ; Get next Hex
  3507. JUMP @numerate_string_hex_0
  3508. ;; Clean up
  3509. :numerate_string_done
  3510. CMPSKIPI.E R2 0 ; If Negate flag has been set
  3511. NEG R3 R3 ; Make the number negative
  3512. MOVE R0 R3 ; Put number in R0
  3513. POPR R4 R15 ; Restore R4
  3514. POPR R3 R15 ; Restore R3
  3515. POPR R2 R15 ; Restore R2
  3516. POPR R1 R15 ; Restore R1
  3517. RET R15
  3518. ;; Keywords
  3519. :union
  3520. "union"
  3521. :struct
  3522. "struct"
  3523. :constant
  3524. "CONSTANT"
  3525. :main_string
  3526. "main"
  3527. :argc_string
  3528. "argc"
  3529. :argv_string
  3530. "argv"
  3531. :if_string
  3532. "if"
  3533. :else_string
  3534. "else"
  3535. :do_string
  3536. "do"
  3537. :while_string
  3538. "while"
  3539. :for_string
  3540. "for"
  3541. :asm_string
  3542. "asm"
  3543. :goto_string
  3544. "goto"
  3545. :return_string
  3546. "return"
  3547. :break_string
  3548. "break"
  3549. :continue_string
  3550. "continue"
  3551. :sizeof_string
  3552. "sizeof"
  3553. :plus_string
  3554. "+"
  3555. :minus_string
  3556. "-"
  3557. :multiply_string
  3558. "*"
  3559. :divide_string
  3560. "/"
  3561. :modulus_string
  3562. "%"
  3563. :left_shift_string
  3564. "<<"
  3565. :right_shift_string
  3566. ">>"
  3567. :less_than_string
  3568. "<"
  3569. :less_than_equal_string
  3570. "<="
  3571. :greater_than_equal_string
  3572. ">="
  3573. :greater_than_string
  3574. ">"
  3575. :equal_to_string
  3576. "=="
  3577. :not_equal_string
  3578. "!="
  3579. :bitwise_and
  3580. "&"
  3581. :logical_and
  3582. "&&"
  3583. :bitwise_or
  3584. "|"
  3585. :logical_or
  3586. "||"
  3587. :bitwise_xor
  3588. "^"
  3589. :arrow_string
  3590. "->"
  3591. ;; Frequently Used strings
  3592. ;; Generally used by require_match
  3593. :open_curly_brace
  3594. "{"
  3595. :close_curly_brace
  3596. "}"
  3597. :open_paren
  3598. "("
  3599. :close_paren
  3600. ")"
  3601. :open_bracket
  3602. "["
  3603. :close_bracket
  3604. "]"
  3605. :semicolon
  3606. ";"
  3607. :equal
  3608. "="
  3609. :percent
  3610. "%"
  3611. :newline
  3612. "
  3613. "
  3614. :underline
  3615. "_"
  3616. ;; Global types
  3617. ;; NEXT (0), SIZE (4), OFFSET (8), INDIRECT (12), MEMBERS (16), TYPE (20), NAME (24)
  3618. :global_types
  3619. &type_void
  3620. :prim_types
  3621. :type_void
  3622. &type_int ; NEXT
  3623. '00 00 00 08' ; SIZE
  3624. NOP ; OFFSET
  3625. &type_void ; INDIRECT
  3626. NOP ; MEMBERS
  3627. &type_void ; TYPE
  3628. &type_void_name ; NAME
  3629. :type_void_name
  3630. "void"
  3631. :type_int
  3632. &type_char ; NEXT
  3633. '00 00 00 08' ; SIZE
  3634. NOP ; OFFSET
  3635. &type_int ; INDIRECT
  3636. NOP ; MEMBERS
  3637. &type_int ; TYPE
  3638. &type_int_name ; NAME
  3639. :type_int_name
  3640. "int"
  3641. :type_char
  3642. &type_file ; NEXT
  3643. '00 00 00 01' ; SIZE
  3644. NOP ; OFFSET
  3645. &type_char_indirect ; INDIRECT
  3646. NOP ; MEMBERS
  3647. &type_char ; TYPE
  3648. &type_char_name ; NAME
  3649. :type_char_name
  3650. "char"
  3651. :type_char_indirect
  3652. &type_file ; NEXT
  3653. '00 00 00 08' ; SIZE
  3654. NOP ; OFFSET
  3655. &type_char_double_indirect ; INDIRECT
  3656. NOP ; MEMBERS
  3657. &type_char_indirect ; TYPE
  3658. &type_char_indirect_name ; NAME
  3659. :type_char_indirect_name
  3660. "char*"
  3661. :type_char_double_indirect
  3662. &type_file ; NEXT
  3663. '00 00 00 08' ; SIZE
  3664. NOP ; OFFSET
  3665. &type_char_double_indirect ; INDIRECT
  3666. NOP ; MEMBERS
  3667. &type_char_indirect ; TYPE
  3668. &type_char_double_indirect_name ; NAME
  3669. :type_char_double_indirect_name
  3670. "char**"
  3671. :type_file
  3672. &type_function ; NEXT
  3673. '00 00 00 08' ; SIZE
  3674. NOP ; OFFSET
  3675. &type_file ; INDIRECT
  3676. NOP ; MEMBERS
  3677. &type_file ; TYPE
  3678. &type_file_name ; NAME
  3679. :type_file_name
  3680. "FILE"
  3681. :type_function
  3682. &type_unsigned ; NEXT
  3683. '00 00 00 08' ; SIZE
  3684. NOP ; OFFSET
  3685. &type_function ; INDIRECT
  3686. NOP ; MEMBERS
  3687. &type_function ; TYPE
  3688. &type_function_name ; NAME
  3689. :type_function_name
  3690. "FUNCTION"
  3691. :type_unsigned
  3692. NOP ; NEXT (NULL)
  3693. '00 00 00 08' ; SIZE
  3694. NOP ; OFFSET
  3695. &type_unsigned ; INDIRECT
  3696. NOP ; MEMBERS
  3697. &type_unsigned ; TYPE
  3698. &type_unsigned_name ; NAME
  3699. :type_unsigned_name
  3700. "unsigned"
  3701. ;; debug_list function
  3702. ;; Receives struct token_list* in R0
  3703. ;; Prints contents of list and HALTS
  3704. ;; Does not return
  3705. :debug_list
  3706. MOVE R9 R0 ; Protect the list Pointer
  3707. FALSE R1 ; Write to TTY
  3708. :debug_list_iter
  3709. ;; Header
  3710. LOADUI R0 $debug_list_string0 ; Using our first string
  3711. CALLI R15 @file_print ; Print it
  3712. COPY R0 R9 ; Use address of pointer
  3713. CALLI R15 @numerate_number ; Convert it into a string
  3714. CALLI R15 @file_print ; Print it
  3715. ;; NEXT
  3716. LOADUI R0 $debug_list_string1 ; Using our second string
  3717. CALLI R15 @file_print ; Print it
  3718. LOAD32 R0 R9 0 ; Use address of pointer
  3719. CALLI R15 @numerate_number ; Convert it into a string
  3720. CALLI R15 @file_print ; Print it
  3721. ;; PREV
  3722. LOADUI R0 $debug_list_string2 ; Using our third string
  3723. CALLI R15 @file_print ; Print it
  3724. LOAD32 R0 R9 4 ; Use address of pointer
  3725. CALLI R15 @numerate_number ; Convert it into a string
  3726. CALLI R15 @file_print ; Print it
  3727. ;; S
  3728. LOADUI R0 $debug_list_string3 ; Using our fourth string
  3729. CALLI R15 @file_print ; Print it
  3730. LOAD32 R0 R9 8 ; Use address of pointer
  3731. CALLI R15 @numerate_number ; Convert it into a string
  3732. CALLI R15 @file_print ; Print it
  3733. ;; S Contents
  3734. LOADUI R0 $debug_list_string4 ; Using our Prefix string
  3735. CALLI R15 @file_print ; Print it
  3736. LOAD32 R0 R9 8 ; Use address of pointer
  3737. CMPSKIPI.NE R0 0 ; If NULL Pointer
  3738. LOADUI R0 $debug_list_string_null ; Give meaningful message instead
  3739. CALLI R15 @file_print ; Print it
  3740. ;; TYPE
  3741. LOADUI R0 $debug_list_string5 ; Using our fifth string
  3742. CALLI R15 @file_print ; Print it
  3743. LOAD32 R0 R9 12 ; Use address of pointer
  3744. CALLI R15 @numerate_number ; Convert it into a string
  3745. CALLI R15 @file_print ; Print it
  3746. ;; PREV
  3747. LOADUI R0 $debug_list_string6 ; Using our sixth string
  3748. CALLI R15 @file_print ; Print it
  3749. LOAD32 R0 R9 16 ; Use address of pointer
  3750. CALLI R15 @numerate_number ; Convert it into a string
  3751. CALLI R15 @file_print ; Print it
  3752. ;; Add some space
  3753. LOADUI R0 10 ; Using NEWLINE
  3754. FPUTC
  3755. FPUTC
  3756. ;; Iterate if next not NULL
  3757. LOAD32 R9 R9 0 ; TOKEN = TOKEN->NEXT
  3758. JUMP.NZ R9 @debug_list_iter
  3759. ;; Looks lke we are done, wrap it up
  3760. HALT
  3761. :debug_list_string0
  3762. "Token_list node at address: "
  3763. :debug_list_string1
  3764. "
  3765. NEXT address: "
  3766. :debug_list_string2
  3767. "
  3768. PREV address: "
  3769. :debug_list_string3
  3770. "
  3771. S address: "
  3772. :debug_list_string4
  3773. "
  3774. The contents of S are: "
  3775. :debug_list_string5
  3776. "
  3777. TYPE address: "
  3778. :debug_list_string6
  3779. "
  3780. ARGUMENTS address: "
  3781. :debug_list_string_null
  3782. ">::<NULL>::<"
  3783. :STACK