123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432 |
- /* Copyright (C) 2001, 2009, 2010, 2011, 2012, 2013 Free Software Foundation, Inc.
- *
- * This library is free software; you can redistribute it and/or
- * modify it under the terms of the GNU Lesser General Public License
- * as published by the Free Software Foundation; either version 3 of
- * the License, or (at your option) any later version.
- *
- * This library is distributed in the hope that it will be useful, but
- * WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- * Lesser General Public License for more details.
- *
- * You should have received a copy of the GNU Lesser General Public
- * License along with this library; if not, write to the Free Software
- * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
- * 02110-1301 USA
- */
- /* This file is included in vm.c multiple times. */
- /* Virtual Machine
- This file contains two virtual machines. First, the old one -- the
- one that is currently used, and corresponds to Guile 2.0. It's a
- stack machine, meaning that most instructions pop their operands from
- the top of the stack, and push results there too.
- Following it is the new virtual machine. It's a register machine,
- meaning that intructions address their operands by index, and store
- results in indexed slots as well. Those slots are on the stack.
- It's somewhat confusing to call it a register machine, given that the
- values are on the stack. Perhaps it needs a new name.
- Anyway, things are in a transitional state. We're going to try to
- avoid munging the old VM very much while we flesh out the new one.
- We're also going to try to make them interoperable, as much as
- possible -- to have the old VM be able to call procedures for the new
- VM, and vice versa. This should ease the bootstrapping process. */
- /* The old VM. */
- static SCM VM_NAME (SCM, SCM, SCM*, int);
- /* The new VM. */
- static SCM RTL_VM_NAME (SCM, SCM, SCM*, size_t);
- #if (VM_ENGINE == SCM_VM_REGULAR_ENGINE)
- # define VM_USE_HOOKS 0 /* Various hooks */
- #elif (VM_ENGINE == SCM_VM_DEBUG_ENGINE)
- # define VM_USE_HOOKS 1
- #else
- # error unknown debug engine VM_ENGINE
- #endif
- /* Assign some registers by hand. There used to be a bigger list here,
- but it was never tested, and in the case of x86-32, was a source of
- compilation failures. It can be revived if it's useful, but my naive
- hope is that simply annotating the locals with "register" will be a
- sufficient hint to the compiler. */
- #ifdef __GNUC__
- # if defined __x86_64__
- /* GCC 4.6 chooses %rbp for IP_REG and %rbx for SP_REG, which works
- well. Tell it to keep the jump table in a r12, which is
- callee-saved. */
- # define JT_REG asm ("r12")
- # endif
- #endif
- #ifndef IP_REG
- # define IP_REG
- #endif
- #ifndef SP_REG
- # define SP_REG
- #endif
- #ifndef FP_REG
- # define FP_REG
- #endif
- #ifndef JT_REG
- # define JT_REG
- #endif
- #define VM_ASSERT(condition, handler) \
- do { \
- if (SCM_UNLIKELY (!(condition))) \
- { \
- SYNC_ALL(); \
- handler; \
- } \
- } while (0)
- #ifdef VM_ENABLE_ASSERTIONS
- # define ASSERT(condition) VM_ASSERT (condition, abort())
- #else
- # define ASSERT(condition)
- #endif
- #if VM_USE_HOOKS
- #define RUN_HOOK(h, args, n) \
- do { \
- if (SCM_UNLIKELY (vp->trace_level > 0)) \
- { \
- SYNC_REGISTER (); \
- vm_dispatch_hook (vm, h, args, n); \
- } \
- } while (0)
- #else
- #define RUN_HOOK(h, args, n)
- #endif
- #define RUN_HOOK0(h) RUN_HOOK(h, NULL, 0)
- #define APPLY_HOOK() \
- RUN_HOOK0 (SCM_VM_APPLY_HOOK)
- #define PUSH_CONTINUATION_HOOK() \
- RUN_HOOK0 (SCM_VM_PUSH_CONTINUATION_HOOK)
- #define POP_CONTINUATION_HOOK(vals, n) \
- RUN_HOOK (SCM_VM_POP_CONTINUATION_HOOK, vals, n)
- #define NEXT_HOOK() \
- RUN_HOOK0 (SCM_VM_NEXT_HOOK)
- #define ABORT_CONTINUATION_HOOK(vals, n) \
- RUN_HOOK (SCM_VM_ABORT_CONTINUATION_HOOK, vals, n)
- #define RESTORE_CONTINUATION_HOOK() \
- RUN_HOOK0 (SCM_VM_RESTORE_CONTINUATION_HOOK)
- #define VM_HANDLE_INTERRUPTS \
- SCM_ASYNC_TICK_WITH_CODE (current_thread, SYNC_REGISTER ())
- /* Cache the VM's instruction, stack, and frame pointer in local variables. */
- #define CACHE_REGISTER() \
- { \
- ip = vp->ip; \
- sp = vp->sp; \
- fp = vp->fp; \
- }
- /* Update the registers in VP, a pointer to the current VM. This must be done
- at least before any GC invocation so that `vp->sp' is up-to-date and the
- whole stack gets marked. */
- #define SYNC_REGISTER() \
- { \
- vp->ip = ip; \
- vp->sp = sp; \
- vp->fp = fp; \
- }
- /* FIXME */
- #define ASSERT_VARIABLE(x) \
- VM_ASSERT (SCM_VARIABLEP (x), abort())
- #define ASSERT_BOUND_VARIABLE(x) \
- VM_ASSERT (SCM_VARIABLEP (x) \
- && !scm_is_eq (SCM_VARIABLE_REF (x), SCM_UNDEFINED), \
- abort())
- #ifdef VM_ENABLE_PARANOID_ASSERTIONS
- #define CHECK_IP() \
- do { if (ip < bp->base || ip - bp->base > bp->len) abort (); } while (0)
- #define ASSERT_ALIGNED_PROCEDURE() \
- do { if ((scm_t_bits)bp % 8) abort (); } while (0)
- #define ASSERT_BOUND(x) \
- VM_ASSERT (!scm_is_eq ((x), SCM_UNDEFINED), abort())
- #else
- #define CHECK_IP()
- #define ASSERT_ALIGNED_PROCEDURE()
- #define ASSERT_BOUND(x)
- #endif
- /* Cache the object table and free variables. */
- #define CACHE_PROGRAM() \
- { \
- if (bp != SCM_PROGRAM_DATA (program)) { \
- bp = SCM_PROGRAM_DATA (program); \
- ASSERT_ALIGNED_PROCEDURE (); \
- if (SCM_I_IS_VECTOR (SCM_PROGRAM_OBJTABLE (program))) { \
- objects = SCM_I_VECTOR_WELTS (SCM_PROGRAM_OBJTABLE (program)); \
- } else { \
- objects = NULL; \
- } \
- } \
- }
- #define SYNC_BEFORE_GC() \
- { \
- SYNC_REGISTER (); \
- }
- #define SYNC_ALL() \
- { \
- SYNC_REGISTER (); \
- }
- /*
- * Error check
- */
- /* Accesses to a program's object table. */
- #define CHECK_OBJECT(_num)
- #define CHECK_FREE_VARIABLE(_num)
- /*
- * Stack operation
- */
- #ifdef VM_ENABLE_STACK_NULLING
- # define CHECK_STACK_LEAKN(_n) ASSERT (!sp[_n]);
- # define CHECK_STACK_LEAK() CHECK_STACK_LEAKN(1)
- # define NULLSTACK(_n) { int __x = _n; CHECK_STACK_LEAKN (_n+1); while (__x > 0) sp[__x--] = NULL; }
- /* If you have a nonlocal exit in a pre-wind proc while invoking a continuation
- inside a dynwind (phew!), the stack is fully rewound but vm_reset_stack for
- that continuation doesn't have a chance to run. It's not important on a
- semantic level, but it does mess up our stack nulling -- so this macro is to
- fix that. */
- # define NULLSTACK_FOR_NONLOCAL_EXIT() if (vp->sp > sp) NULLSTACK (vp->sp - sp);
- #else
- # define CHECK_STACK_LEAKN(_n)
- # define CHECK_STACK_LEAK()
- # define NULLSTACK(_n)
- # define NULLSTACK_FOR_NONLOCAL_EXIT()
- #endif
- /* For this check, we don't use VM_ASSERT, because that leads to a
- per-site SYNC_ALL, which is too much code growth. The real problem
- of course is having to check for overflow all the time... */
- #define CHECK_OVERFLOW() \
- do { if (SCM_UNLIKELY (sp >= stack_limit)) goto handle_overflow; } while (0)
- #ifdef VM_CHECK_UNDERFLOW
- #define PRE_CHECK_UNDERFLOW(N) \
- VM_ASSERT (sp - (N) > SCM_FRAME_UPPER_ADDRESS (fp), vm_error_stack_underflow ())
- #define CHECK_UNDERFLOW() PRE_CHECK_UNDERFLOW (0)
- #else
- #define PRE_CHECK_UNDERFLOW(N) /* nop */
- #define CHECK_UNDERFLOW() /* nop */
- #endif
- #define PUSH(x) do { sp++; CHECK_OVERFLOW (); *sp = x; } while (0)
- #define DROP() do { sp--; CHECK_UNDERFLOW (); NULLSTACK (1); } while (0)
- #define DROPN(_n) do { sp -= (_n); CHECK_UNDERFLOW (); NULLSTACK (_n); } while (0)
- #define POP(x) do { PRE_CHECK_UNDERFLOW (1); x = *sp--; NULLSTACK (1); } while (0)
- #define POP2(x,y) do { PRE_CHECK_UNDERFLOW (2); x = *sp--; y = *sp--; NULLSTACK (2); } while (0)
- #define POP3(x,y,z) do { PRE_CHECK_UNDERFLOW (3); x = *sp--; y = *sp--; z = *sp--; NULLSTACK (3); } while (0)
- /* Pop the N objects on top of the stack and push a list that contains
- them. */
- #define POP_LIST(n) \
- do \
- { \
- int i; \
- SCM l = SCM_EOL, x; \
- SYNC_BEFORE_GC (); \
- for (i = n; i; i--) \
- { \
- POP (x); \
- l = scm_cons (x, l); \
- } \
- PUSH (l); \
- } while (0)
- /* The opposite: push all of the elements in L onto the list. */
- #define PUSH_LIST(l, NILP) \
- do \
- { \
- for (; scm_is_pair (l); l = SCM_CDR (l)) \
- PUSH (SCM_CAR (l)); \
- VM_ASSERT (NILP (l), vm_error_improper_list (l)); \
- } while (0)
- /*
- * Instruction operation
- */
- #define FETCH() (*ip++)
- #define FETCH_LENGTH(len) do { len=*ip++; len<<=8; len+=*ip++; len<<=8; len+=*ip++; } while (0)
- #undef NEXT_JUMP
- #ifdef HAVE_LABELS_AS_VALUES
- # define NEXT_JUMP() goto *jump_table[FETCH () & SCM_VM_INSTRUCTION_MASK]
- #else
- # define NEXT_JUMP() goto vm_start
- #endif
- #define NEXT \
- { \
- NEXT_HOOK (); \
- CHECK_STACK_LEAK (); \
- NEXT_JUMP (); \
- }
- /* See frames.h for the layout of stack frames */
- /* When this is called, bp points to the new program data,
- and the arguments are already on the stack */
- #define DROP_FRAME() \
- { \
- sp -= 3; \
- NULLSTACK (3); \
- CHECK_UNDERFLOW (); \
- }
-
- static SCM
- VM_NAME (SCM vm, SCM program, SCM *argv, int nargs)
- {
- /* VM registers */
- register scm_t_uint8 *ip IP_REG; /* instruction pointer */
- register SCM *sp SP_REG; /* stack pointer */
- register SCM *fp FP_REG; /* frame pointer */
- struct scm_vm *vp = SCM_VM_DATA (vm);
- /* Cache variables */
- struct scm_objcode *bp = NULL; /* program base pointer */
- SCM *objects = NULL; /* constant objects */
- SCM *stack_limit = vp->stack_limit; /* stack limit address */
- scm_i_thread *current_thread = SCM_I_CURRENT_THREAD;
- /* Internal variables */
- int nvalues = 0;
- scm_i_jmp_buf registers; /* used for prompts */
- #ifdef HAVE_LABELS_AS_VALUES
- static const void **jump_table_pointer = NULL;
- #endif
- #ifdef HAVE_LABELS_AS_VALUES
- register const void **jump_table JT_REG;
- if (SCM_UNLIKELY (!jump_table_pointer))
- {
- int i;
- jump_table_pointer = malloc (SCM_VM_NUM_INSTRUCTIONS * sizeof (void*));
- for (i = 0; i < SCM_VM_NUM_INSTRUCTIONS; i++)
- jump_table_pointer[i] = &&vm_error_bad_instruction;
- #define VM_INSTRUCTION_TO_LABEL 1
- #define jump_table jump_table_pointer
- #include <libguile/vm-expand.h>
- #include <libguile/vm-i-system.i>
- #include <libguile/vm-i-scheme.i>
- #include <libguile/vm-i-loader.i>
- #undef jump_table
- #undef VM_INSTRUCTION_TO_LABEL
- }
- /* Attempt to keep JUMP_TABLE_POINTER in a register. This saves one
- load instruction at each instruction dispatch. */
- jump_table = jump_table_pointer;
- #endif
- if (SCM_I_SETJMP (registers))
- {
- /* Non-local return. Cache the VM registers back from the vp, and
- go to the handler.
- Note, at this point, we must assume that any variable local to
- vm_engine that can be assigned *has* been assigned. So we need to pull
- all our state back from the ip/fp/sp.
- */
- CACHE_REGISTER ();
- program = SCM_FRAME_PROGRAM (fp);
- CACHE_PROGRAM ();
- /* The stack contains the values returned to this continuation,
- along with a number-of-values marker -- like an MV return. */
- ABORT_CONTINUATION_HOOK (sp - SCM_I_INUM (*sp), SCM_I_INUM (*sp));
- NEXT;
- }
- CACHE_REGISTER ();
- /* Since it's possible to receive the arguments on the stack itself,
- and indeed the RTL VM invokes us that way, shuffle up the
- arguments first. */
- VM_ASSERT (sp + 8 + nargs < stack_limit, vm_error_too_many_args (nargs));
- {
- int i;
- for (i = nargs - 1; i >= 0; i--)
- sp[9 + i] = argv[i];
- }
- /* Initial frame */
- PUSH (SCM_PACK (fp)); /* dynamic link */
- PUSH (SCM_PACK (0)); /* mvra */
- PUSH (SCM_PACK (ip)); /* ra */
- PUSH (boot_continuation);
- fp = sp + 1;
- ip = SCM_C_OBJCODE_BASE (SCM_PROGRAM_DATA (boot_continuation));
- /* MV-call frame, function & arguments */
- PUSH (SCM_PACK (fp)); /* dynamic link */
- PUSH (SCM_PACK (ip + 1)); /* mvra */
- PUSH (SCM_PACK (ip)); /* ra */
- PUSH (program);
- fp = sp + 1;
- sp += nargs;
- PUSH_CONTINUATION_HOOK ();
- apply:
- program = fp[-1];
- if (!SCM_PROGRAM_P (program))
- {
- if (SCM_STRUCTP (program) && SCM_STRUCT_APPLICABLE_P (program))
- fp[-1] = SCM_STRUCT_PROCEDURE (program);
- else if (SCM_HAS_TYP7 (program, scm_tc7_rtl_program))
- {
- SCM ret;
- SYNC_ALL ();
- ret = RTL_VM_NAME (vm, program, fp, sp - fp + 1);
- NULLSTACK_FOR_NONLOCAL_EXIT ();
- if (SCM_UNLIKELY (SCM_VALUESP (ret)))
- {
- /* multiple values returned to continuation */
- ret = scm_struct_ref (ret, SCM_INUM0);
- nvalues = scm_ilength (ret);
- PUSH_LIST (ret, scm_is_null);
- goto vm_return_values;
- }
- else
- {
- PUSH (ret);
- goto vm_return;
- }
- }
- else if (SCM_HAS_TYP7 (program, scm_tc7_smob)
- && SCM_SMOB_APPLICABLE_P (program))
- {
- /* (smob arg0 ... argN) => (apply-smob smob arg0 ... argN) */
- int i;
- PUSH (SCM_BOOL_F);
- for (i = sp - fp; i >= 0; i--)
- fp[i] = fp[i - 1];
- fp[-1] = SCM_SMOB_DESCRIPTOR (program).apply_trampoline;
- }
- else
- {
- SYNC_ALL();
- vm_error_wrong_type_apply (program);
- }
- goto apply;
- }
- CACHE_PROGRAM ();
- ip = SCM_C_OBJCODE_BASE (bp);
- APPLY_HOOK ();
- /* Let's go! */
- NEXT;
- #ifndef HAVE_LABELS_AS_VALUES
- vm_start:
- switch ((*ip++) & SCM_VM_INSTRUCTION_MASK) {
- #endif
- #include "vm-expand.h"
- #include "vm-i-system.c"
- #include "vm-i-scheme.c"
- #include "vm-i-loader.c"
- #ifndef HAVE_LABELS_AS_VALUES
- default:
- goto vm_error_bad_instruction;
- }
- #endif
- abort (); /* never reached */
- vm_error_bad_instruction:
- vm_error_bad_instruction (ip[-1]);
- abort (); /* never reached */
- handle_overflow:
- SYNC_ALL ();
- vm_error_stack_overflow (vp);
- abort (); /* never reached */
- }
- #undef ALIGNED_P
- #undef CACHE_REGISTER
- #undef CHECK_OVERFLOW
- #undef FUNC2
- #undef INIT
- #undef INUM_MAX
- #undef INUM_MIN
- #undef INUM_STEP
- #undef jump_table
- #undef LOCAL_REF
- #undef LOCAL_SET
- #undef NEXT
- #undef NEXT_JUMP
- #undef REL
- #undef RETURN
- #undef RETURN_ONE_VALUE
- #undef RETURN_VALUE_LIST
- #undef SYNC_ALL
- #undef SYNC_BEFORE_GC
- #undef SYNC_IP
- #undef SYNC_REGISTER
- #undef VARIABLE_BOUNDP
- #undef VARIABLE_REF
- #undef VARIABLE_SET
- #undef VM_DEFINE_OP
- #undef VM_INSTRUCTION_TO_LABEL
- /* Virtual Machine
- This is Guile's new virtual machine. When I say "new", I mean
- relative to the current virtual machine. At some point it will
- become "the" virtual machine, and we'll delete this paragraph. As
- such, the rest of the comments speak as if there's only one VM.
- In difference from the old VM, local 0 is the procedure, and the
- first argument is local 1. At some point in the future we should
- change the fp to point to the procedure and not to local 1.
- <more overview here>
- */
- /* The VM has three state bits: the instruction pointer (IP), the frame
- pointer (FP), and the top-of-stack pointer (SP). We cache the first
- two of these in machine registers, local to the VM, because they are
- used extensively by the VM. As the SP is used more by code outside
- the VM than by the VM itself, we don't bother caching it locally.
- Since the FP changes infrequently, relative to the IP, we keep vp->fp
- in sync with the local FP. This would be a big lose for the IP,
- though, so instead of updating vp->ip all the time, we call SYNC_IP
- whenever we would need to know the IP of the top frame. In practice,
- we need to SYNC_IP whenever we call out of the VM to a function that
- would like to walk the stack, perhaps as the result of an
- exception. */
- #define SYNC_IP() \
- vp->ip = (scm_t_uint8 *) (ip)
- #define SYNC_REGISTER() \
- SYNC_IP()
- #define SYNC_BEFORE_GC() /* Only SP and FP needed to trace GC */
- #define SYNC_ALL() /* FP already saved */ \
- SYNC_IP()
- #define CHECK_OVERFLOW(sp) \
- do { \
- if (SCM_UNLIKELY ((sp) >= stack_limit)) \
- vm_error_stack_overflow (vp); \
- } while (0)
- /* Reserve stack space for a frame. Will check that there is sufficient
- stack space for N locals, including the procedure, in addition to
- 3 words to set up the next frame. Invoke after preparing the new
- frame and setting the fp and ip. */
- #define ALLOC_FRAME(n) \
- do { \
- SCM *new_sp = vp->sp = fp - 1 + n - 1; \
- CHECK_OVERFLOW (new_sp + 4); \
- } while (0)
- /* Reset the current frame to hold N locals. Used when we know that no
- stack expansion is needed. */
- #define RESET_FRAME(n) \
- do { \
- vp->sp = fp - 2 + n; \
- } while (0)
- /* Compute the number of locals in the frame. This is equal to the
- number of actual arguments when a function is first called, plus
- one for the function. */
- #define FRAME_LOCALS_COUNT() \
- (vp->sp + 1 - (fp - 1))
- /* Restore registers after returning from a frame. */
- #define RESTORE_FRAME() \
- do { \
- } while (0)
- #define CACHE_REGISTER() \
- do { \
- ip = (scm_t_uint32 *) vp->ip; \
- fp = vp->fp; \
- } while (0)
- #ifdef HAVE_LABELS_AS_VALUES
- # define BEGIN_DISPATCH_SWITCH /* */
- # define END_DISPATCH_SWITCH /* */
- # define NEXT(n) \
- do \
- { \
- ip += n; \
- NEXT_HOOK (); \
- op = *ip; \
- goto *jump_table[op & 0xff]; \
- } \
- while (0)
- # define VM_DEFINE_OP(opcode, tag, name, meta) \
- op_##tag:
- #else
- # define BEGIN_DISPATCH_SWITCH \
- vm_start: \
- NEXT_HOOK (); \
- op = *ip; \
- switch (op & 0xff) \
- {
- # define END_DISPATCH_SWITCH \
- default: \
- goto vm_error_bad_instruction; \
- }
- # define NEXT(n) \
- do \
- { \
- ip += n; \
- goto vm_start; \
- } \
- while (0)
- # define VM_DEFINE_OP(opcode, tag, name, meta) \
- op_##tag: \
- case opcode:
- #endif
- #define LOCAL_REF(i) SCM_FRAME_VARIABLE (fp, (i) - 1)
- #define LOCAL_SET(i,o) SCM_FRAME_VARIABLE (fp, (i) - 1) = o
- #define VARIABLE_REF(v) SCM_VARIABLE_REF (v)
- #define VARIABLE_SET(v,o) SCM_VARIABLE_SET (v, o)
- #define VARIABLE_BOUNDP(v) (!scm_is_eq (VARIABLE_REF (v), SCM_UNDEFINED))
- #define RETURN_ONE_VALUE(ret) \
- do { \
- SCM val = ret; \
- SCM *sp = SCM_FRAME_LOWER_ADDRESS (fp); \
- VM_HANDLE_INTERRUPTS; \
- ip = SCM_FRAME_RTL_RETURN_ADDRESS (fp); \
- fp = vp->fp = SCM_FRAME_DYNAMIC_LINK (fp); \
- /* Clear frame. */ \
- sp[0] = SCM_BOOL_F; \
- sp[1] = SCM_BOOL_F; \
- sp[2] = SCM_BOOL_F; \
- /* Leave proc. */ \
- sp[4] = val; \
- vp->sp = sp + 4; \
- POP_CONTINUATION_HOOK (sp, 1); \
- NEXT (0); \
- } while (0)
- /* While we could generate the list-unrolling code here, it's fine for
- now to just tail-call (apply values vals). */
- #define RETURN_VALUE_LIST(vals_) \
- do { \
- SCM vals = vals_; \
- VM_HANDLE_INTERRUPTS; \
- fp[-1] = rtl_apply; \
- fp[0] = rtl_values; \
- fp[1] = vals; \
- RESET_FRAME (3); \
- ip = (scm_t_uint32 *) rtl_apply_code; \
- goto op_tail_apply; \
- } while (0)
- #define BR_NARGS(rel) \
- scm_t_uint16 expected; \
- SCM_UNPACK_RTL_24 (op, expected); \
- if (FRAME_LOCALS_COUNT() rel expected) \
- { \
- scm_t_int32 offset = ip[1]; \
- offset >>= 8; /* Sign-extending shift. */ \
- NEXT (offset); \
- } \
- NEXT (2)
- #define BR_UNARY(x, exp) \
- scm_t_uint32 test; \
- SCM x; \
- SCM_UNPACK_RTL_24 (op, test); \
- x = LOCAL_REF (test); \
- if ((ip[1] & 0x1) ? !(exp) : (exp)) \
- { \
- scm_t_int32 offset = ip[1]; \
- offset >>= 8; /* Sign-extending shift. */ \
- if (offset < 0) \
- VM_HANDLE_INTERRUPTS; \
- NEXT (offset); \
- } \
- NEXT (2)
- #define BR_BINARY(x, y, exp) \
- scm_t_uint16 a, b; \
- SCM x, y; \
- SCM_UNPACK_RTL_12_12 (op, a, b); \
- x = LOCAL_REF (a); \
- y = LOCAL_REF (b); \
- if ((ip[1] & 0x1) ? !(exp) : (exp)) \
- { \
- scm_t_int32 offset = ip[1]; \
- offset >>= 8; /* Sign-extending shift. */ \
- if (offset < 0) \
- VM_HANDLE_INTERRUPTS; \
- NEXT (offset); \
- } \
- NEXT (2)
- #define BR_ARITHMETIC(crel,srel) \
- { \
- scm_t_uint16 a, b; \
- SCM x, y; \
- SCM_UNPACK_RTL_12_12 (op, a, b); \
- x = LOCAL_REF (a); \
- y = LOCAL_REF (b); \
- if (SCM_I_INUMP (x) && SCM_I_INUMP (y)) \
- { \
- scm_t_signed_bits x_bits = SCM_UNPACK (x); \
- scm_t_signed_bits y_bits = SCM_UNPACK (y); \
- if ((ip[1] & 0x1) ? !(x_bits crel y_bits) : (x_bits crel y_bits)) \
- { \
- scm_t_int32 offset = ip[1]; \
- offset >>= 8; /* Sign-extending shift. */ \
- if (offset < 0) \
- VM_HANDLE_INTERRUPTS; \
- NEXT (offset); \
- } \
- NEXT (2); \
- } \
- else \
- { \
- SCM res; \
- SYNC_IP (); \
- res = srel (x, y); \
- if ((ip[1] & 0x1) ? scm_is_false (res) : scm_is_true (res)) \
- { \
- scm_t_int32 offset = ip[1]; \
- offset >>= 8; /* Sign-extending shift. */ \
- if (offset < 0) \
- VM_HANDLE_INTERRUPTS; \
- NEXT (offset); \
- } \
- NEXT (2); \
- } \
- }
- #define ARGS1(a1) \
- scm_t_uint16 dst, src; \
- SCM a1; \
- SCM_UNPACK_RTL_12_12 (op, dst, src); \
- a1 = LOCAL_REF (src)
- #define ARGS2(a1, a2) \
- scm_t_uint8 dst, src1, src2; \
- SCM a1, a2; \
- SCM_UNPACK_RTL_8_8_8 (op, dst, src1, src2); \
- a1 = LOCAL_REF (src1); \
- a2 = LOCAL_REF (src2)
- #define RETURN(x) \
- do { LOCAL_SET (dst, x); NEXT (1); } while (0)
- /* The maximum/minimum tagged integers. */
- #define INUM_MAX \
- ((scm_t_signed_bits) SCM_UNPACK (SCM_I_MAKINUM (SCM_MOST_POSITIVE_FIXNUM)))
- #define INUM_MIN \
- ((scm_t_signed_bits) SCM_UNPACK (SCM_I_MAKINUM (SCM_MOST_NEGATIVE_FIXNUM)))
- #define INUM_STEP \
- ((scm_t_signed_bits) SCM_UNPACK (SCM_INUM1) \
- - (scm_t_signed_bits) SCM_UNPACK (SCM_INUM0))
- #define BINARY_INTEGER_OP(CFUNC,SFUNC) \
- { \
- ARGS2 (x, y); \
- if (SCM_I_INUMP (x) && SCM_I_INUMP (y)) \
- { \
- scm_t_int64 n = SCM_I_INUM (x) CFUNC SCM_I_INUM (y); \
- if (SCM_FIXABLE (n)) \
- RETURN (SCM_I_MAKINUM (n)); \
- } \
- SYNC_IP (); \
- RETURN (SFUNC (x, y)); \
- }
- #define VM_VALIDATE_PAIR(x, proc) \
- VM_ASSERT (scm_is_pair (x), vm_error_not_a_pair (proc, x))
-
- #define VM_VALIDATE_STRUCT(obj, proc) \
- VM_ASSERT (SCM_STRUCTP (obj), vm_error_not_a_pair (proc, obj))
- #define VM_VALIDATE_BYTEVECTOR(x, proc) \
- VM_ASSERT (SCM_BYTEVECTOR_P (x), vm_error_not_a_bytevector (proc, x))
- /* Return true (non-zero) if PTR has suitable alignment for TYPE. */
- #define ALIGNED_P(ptr, type) \
- ((scm_t_uintptr) (ptr) % alignof_type (type) == 0)
- static SCM
- RTL_VM_NAME (SCM vm, SCM program, SCM *argv, size_t nargs_)
- {
- /* Instruction pointer: A pointer to the opcode that is currently
- running. */
- register scm_t_uint32 *ip IP_REG;
- /* Frame pointer: A pointer into the stack, off of which we index
- arguments and local variables. Pushed at function calls, popped on
- returns. */
- register SCM *fp FP_REG;
- /* Current opcode: A cache of *ip. */
- register scm_t_uint32 op;
- /* Cached variables. */
- struct scm_vm *vp = SCM_VM_DATA (vm);
- SCM *stack_limit = vp->stack_limit; /* stack limit address */
- scm_i_thread *current_thread = SCM_I_CURRENT_THREAD;
- scm_i_jmp_buf registers; /* used for prompts */
- #ifdef HAVE_LABELS_AS_VALUES
- static const void **jump_table_pointer = NULL;
- register const void **jump_table JT_REG;
- if (SCM_UNLIKELY (!jump_table_pointer))
- {
- int i;
- jump_table_pointer = malloc (SCM_VM_NUM_INSTRUCTIONS * sizeof (void*));
- for (i = 0; i < SCM_VM_NUM_INSTRUCTIONS; i++)
- jump_table_pointer[i] = &&vm_error_bad_instruction;
- #define INIT(opcode, tag, name, meta) jump_table_pointer[opcode] = &&op_##tag;
- FOR_EACH_VM_OPERATION(INIT);
- #undef INIT
- }
- /* Attempt to keep JUMP_TABLE_POINTER in a register. This saves one
- load instruction at each instruction dispatch. */
- jump_table = jump_table_pointer;
- #endif
- if (SCM_I_SETJMP (registers))
- {
- /* Non-local return. The values are on the stack, on a new frame
- set up to call `values' to return the values to the handler.
- Cache the VM registers back from the vp, and dispatch to the
- body of `values'.
- Note, at this point, we must assume that any variable local to
- vm_engine that can be assigned *has* been assigned. So we need
- to pull all our state back from the ip/fp/sp.
- */
- CACHE_REGISTER ();
- ABORT_CONTINUATION_HOOK (fp, FRAME_LOCALS_COUNT());
- NEXT (0);
- }
- /* Load previous VM registers. */
- CACHE_REGISTER ();
- VM_HANDLE_INTERRUPTS;
- /* Initialization */
- {
- SCM *base;
- /* Check that we have enough space: 4 words for the boot
- continuation, 4 + nargs for the procedure application, and 4 for
- setting up a new frame. */
- base = vp->sp + 1;
- CHECK_OVERFLOW (vp->sp + 4 + 4 + nargs_ + 4);
- /* Since it's possible to receive the arguments on the stack itself,
- and indeed the regular VM invokes us that way, shuffle up the
- arguments first. */
- {
- int i;
- for (i = nargs_ - 1; i >= 0; i--)
- base[8 + i] = argv[i];
- }
- /* Initial frame, saving previous fp and ip, with the boot
- continuation. */
- base[0] = SCM_PACK (fp); /* dynamic link */
- base[1] = SCM_PACK (0); /* the boot continuation does not return to scheme */
- base[2] = SCM_PACK (ip); /* ra */
- base[3] = rtl_boot_continuation;
- fp = &base[4];
- ip = (scm_t_uint32 *) rtl_boot_continuation_code;
- /* MV-call frame, function & arguments */
- base[4] = SCM_PACK (fp); /* dynamic link */
- base[5] = SCM_PACK (ip); /* in RTL programs, MVRA same as RA */
- base[6] = SCM_PACK (ip); /* ra */
- base[7] = program;
- fp = vp->fp = &base[8];
- RESET_FRAME (nargs_ + 1);
- }
- apply:
- while (!SCM_RTL_PROGRAM_P (SCM_FRAME_PROGRAM (fp)))
- {
- #if 0
- SCM proc = SCM_FRAME_PROGRAM (fp);
- if (SCM_STRUCTP (proc) && SCM_STRUCT_APPLICABLE_P (proc))
- {
- fp[-1] = SCM_STRUCT_PROCEDURE (proc);
- continue;
- }
- if (SCM_HAS_TYP7 (proc, scm_tc7_smob) && SCM_SMOB_APPLICABLE_P (proc))
- {
- scm_t_uint32 n = FRAME_LOCALS_COUNT();
- /* Shuffle args up, place smob in local 0. */
- CHECK_OVERFLOW (vp->sp + 1);
- vp->sp++;
- while (n--)
- LOCAL_SET (n + 1, LOCAL_REF (n));
- fp[-1] = SCM_SMOB_DESCRIPTOR (proc).apply_trampoline;
- continue;
- }
- SYNC_IP();
- vm_error_wrong_type_apply (proc);
- #else
- SCM ret;
- SYNC_ALL ();
- ret = VM_NAME (vm, fp[-1], fp, FRAME_LOCALS_COUNT () - 1);
- if (SCM_UNLIKELY (SCM_VALUESP (ret)))
- RETURN_VALUE_LIST (scm_struct_ref (ret, SCM_INUM0));
- else
- RETURN_ONE_VALUE (ret);
- #endif
- }
- /* Let's go! */
- ip = SCM_RTL_PROGRAM_CODE (SCM_FRAME_PROGRAM (fp));
- NEXT (0);
- BEGIN_DISPATCH_SWITCH;
-
-
- /*
- * Call and return
- */
- /* halt _:24
- *
- * Bring the VM to a halt, returning all the values from the stack.
- */
- VM_DEFINE_OP (0, halt, "halt", OP1 (U8_X24))
- {
- scm_t_uint32 nvals = FRAME_LOCALS_COUNT() - 5;
- SCM ret;
- /* Boot closure in r0, empty frame in r1/r2/r3, proc in r4, values from r5. */
- if (nvals == 1)
- ret = LOCAL_REF (5);
- else
- {
- scm_t_uint32 n;
- ret = SCM_EOL;
- SYNC_BEFORE_GC();
- for (n = nvals; n > 0; n--)
- ret = scm_cons (LOCAL_REF (5 + n - 1), ret);
- ret = scm_values (ret);
- }
- vp->ip = SCM_FRAME_RETURN_ADDRESS (fp);
- vp->sp = SCM_FRAME_LOWER_ADDRESS (fp) - 1;
- vp->fp = SCM_FRAME_DYNAMIC_LINK (fp);
- return ret;
- }
- /* call proc:24 _:8 nlocals:24
- *
- * Call a procedure. PROC is the local corresponding to a procedure.
- * The three values below PROC will be overwritten by the saved call
- * frame data. The new frame will have space for NLOCALS locals: one
- * for the procedure, and the rest for the arguments which should
- * already have been pushed on.
- *
- * When the call returns, execution proceeds with the next
- * instruction. There may be any number of values on the return
- * stack; the precise number can be had by subtracting the address of
- * PROC from the post-call SP.
- */
- VM_DEFINE_OP (1, call, "call", OP2 (U8_U24, X8_U24))
- {
- scm_t_uint32 proc, nlocals;
- SCM *old_fp = fp;
- SCM_UNPACK_RTL_24 (op, proc);
- SCM_UNPACK_RTL_24 (ip[1], nlocals);
- VM_HANDLE_INTERRUPTS;
- fp = vp->fp = old_fp + proc;
- SCM_FRAME_SET_DYNAMIC_LINK (fp, old_fp);
- SCM_FRAME_SET_RTL_MV_RETURN_ADDRESS (fp, ip + 2);
- SCM_FRAME_SET_RTL_RETURN_ADDRESS (fp, ip + 2);
- RESET_FRAME (nlocals);
- PUSH_CONTINUATION_HOOK ();
- APPLY_HOOK ();
- if (SCM_UNLIKELY (!SCM_RTL_PROGRAM_P (SCM_FRAME_PROGRAM (fp))))
- goto apply;
- ip = SCM_RTL_PROGRAM_CODE (SCM_FRAME_PROGRAM (fp));
- NEXT (0);
- }
- /* tail-call nlocals:24
- *
- * Tail-call a procedure. Requires that the procedure and all of the
- * arguments have already been shuffled into position.
- */
- VM_DEFINE_OP (2, tail_call, "tail-call", OP1 (U8_U24))
- {
- scm_t_uint32 nlocals;
-
- SCM_UNPACK_RTL_24 (op, nlocals);
- VM_HANDLE_INTERRUPTS;
- RESET_FRAME (nlocals);
- APPLY_HOOK ();
- if (SCM_UNLIKELY (!SCM_RTL_PROGRAM_P (SCM_FRAME_PROGRAM (fp))))
- goto apply;
- ip = SCM_RTL_PROGRAM_CODE (SCM_FRAME_PROGRAM (fp));
- NEXT (0);
- }
- /* receive dst:12 proc:12 _:8 nlocals:24
- *
- * Receive a single return value from a call whose procedure was in
- * PROC, asserting that the call actually returned at least one
- * value. Afterwards, resets the frame to NLOCALS locals.
- */
- VM_DEFINE_OP (3, receive, "receive", OP2 (U8_U12_U12, X8_U24) | OP_DST)
- {
- scm_t_uint16 dst, proc;
- scm_t_uint32 nlocals;
- SCM_UNPACK_RTL_12_12 (op, dst, proc);
- SCM_UNPACK_RTL_24 (ip[1], nlocals);
- VM_ASSERT (FRAME_LOCALS_COUNT () > proc + 1, vm_error_no_values ());
- LOCAL_SET (dst, LOCAL_REF (proc + 1));
- RESET_FRAME (nlocals);
- NEXT (2);
- }
- /* receive-values proc:24 _:8 nvalues:24
- *
- * Receive a return of multiple values from a call whose procedure was
- * in PROC. If fewer than NVALUES values were returned, signal an
- * error. After receive-values has run, the values can be copied down
- * via `mov'.
- */
- VM_DEFINE_OP (4, receive_values, "receive-values", OP2 (U8_U24, X8_U24))
- {
- scm_t_uint32 proc, nvalues;
- SCM_UNPACK_RTL_24 (op, proc);
- SCM_UNPACK_RTL_24 (ip[1], nvalues);
- VM_ASSERT (FRAME_LOCALS_COUNT () > proc + nvalues,
- vm_error_not_enough_values ());
- NEXT (2);
- }
- /* return src:24
- *
- * Return a value.
- */
- VM_DEFINE_OP (5, return, "return", OP1 (U8_U24))
- {
- scm_t_uint32 src;
- SCM_UNPACK_RTL_24 (op, src);
- RETURN_ONE_VALUE (LOCAL_REF (src));
- }
- /* return-values _:24
- *
- * Return a number of values from a call frame. This opcode
- * corresponds to an application of `values' in tail position. As
- * with tail calls, we expect that the values have already been
- * shuffled down to a contiguous array starting at slot 1.
- * We also expect the frame has already been reset.
- */
- VM_DEFINE_OP (6, return_values, "return-values", OP1 (U8_X24))
- {
- scm_t_uint32 nvalues _GL_UNUSED = FRAME_LOCALS_COUNT();
- SCM *base = fp;
- VM_HANDLE_INTERRUPTS;
- ip = SCM_FRAME_RTL_MV_RETURN_ADDRESS (fp);
- fp = vp->fp = SCM_FRAME_DYNAMIC_LINK (fp);
- /* Clear stack frame. */
- base[-2] = SCM_BOOL_F;
- base[-3] = SCM_BOOL_F;
- base[-4] = SCM_BOOL_F;
- POP_CONTINUATION_HOOK (base, nvalues);
- NEXT (0);
- }
-
- /*
- * Specialized call stubs
- */
- /* subr-call ptr-idx:24
- *
- * Call a subr, passing all locals in this frame as arguments. Fetch
- * the foreign pointer from PTR-IDX, a free variable. Return from the
- * calling frame. This instruction is part of the trampolines
- * created in gsubr.c, and is not generated by the compiler.
- */
- VM_DEFINE_OP (7, subr_call, "subr-call", OP1 (U8_U24))
- {
- scm_t_uint32 ptr_idx;
- SCM pointer, ret;
- SCM (*subr)();
- SCM_UNPACK_RTL_24 (op, ptr_idx);
- pointer = SCM_RTL_PROGRAM_FREE_VARIABLE_REF (LOCAL_REF (0), ptr_idx);
- subr = SCM_POINTER_VALUE (pointer);
- VM_HANDLE_INTERRUPTS;
- SYNC_IP ();
- switch (FRAME_LOCALS_COUNT ())
- {
- case 0:
- ret = subr ();
- break;
- case 1:
- ret = subr (fp[0]);
- break;
- case 2:
- ret = subr (fp[0], fp[1]);
- break;
- case 3:
- ret = subr (fp[0], fp[1], fp[2]);
- break;
- case 4:
- ret = subr (fp[0], fp[1], fp[2], fp[3]);
- break;
- case 5:
- ret = subr (fp[0], fp[1], fp[2], fp[3], fp[4]);
- break;
- case 6:
- ret = subr (fp[0], fp[1], fp[2], fp[3], fp[4], fp[5]);
- break;
- case 7:
- ret = subr (fp[0], fp[1], fp[2], fp[3], fp[4], fp[5], fp[6]);
- break;
- case 8:
- ret = subr (fp[0], fp[1], fp[2], fp[3], fp[4], fp[5], fp[6], fp[7]);
- break;
- case 9:
- ret = subr (fp[0], fp[1], fp[2], fp[3], fp[4], fp[5], fp[6], fp[7], fp[8]);
- break;
- case 10:
- ret = subr (fp[0], fp[1], fp[2], fp[3], fp[4], fp[5], fp[6], fp[7], fp[8], fp[9]);
- break;
- default:
- abort ();
- }
- // NULLSTACK_FOR_NONLOCAL_EXIT ();
- if (SCM_UNLIKELY (SCM_VALUESP (ret)))
- /* multiple values returned to continuation */
- RETURN_VALUE_LIST (scm_struct_ref (ret, SCM_INUM0));
- else
- RETURN_ONE_VALUE (ret);
- }
- /* foreign-call cif-idx:12 ptr-idx:12
- *
- * Call a foreign function. Fetch the CIF and foreign pointer from
- * CIF-IDX and PTR-IDX, both free variables. Return from the calling
- * frame. Arguments are taken from the stack. This instruction is
- * part of the trampolines created by the FFI, and is not generated by
- * the compiler.
- */
- VM_DEFINE_OP (8, foreign_call, "foreign-call", OP1 (U8_U12_U12))
- {
- scm_t_uint16 cif_idx, ptr_idx;
- SCM closure, cif, pointer, ret;
- SCM_UNPACK_RTL_12_12 (op, cif_idx, ptr_idx);
- closure = LOCAL_REF (0);
- cif = SCM_RTL_PROGRAM_FREE_VARIABLE_REF (closure, cif_idx);
- pointer = SCM_RTL_PROGRAM_FREE_VARIABLE_REF (closure, ptr_idx);
- SYNC_IP ();
- VM_HANDLE_INTERRUPTS;
- // FIXME: separate args
- ret = scm_i_foreign_call (scm_cons (cif, pointer), fp);
- // NULLSTACK_FOR_NONLOCAL_EXIT ();
- if (SCM_UNLIKELY (SCM_VALUESP (ret)))
- /* multiple values returned to continuation */
- RETURN_VALUE_LIST (scm_struct_ref (ret, SCM_INUM0));
- else
- RETURN_ONE_VALUE (ret);
- }
- /* continuation-call contregs:24
- *
- * Return to a continuation, nonlocally. The arguments to the
- * continuation are taken from the stack. CONTREGS is a free variable
- * containing the reified continuation. This instruction is part of
- * the implementation of undelimited continuations, and is not
- * generated by the compiler.
- */
- VM_DEFINE_OP (9, continuation_call, "continuation-call", OP1 (U8_U24))
- {
- SCM contregs;
- scm_t_uint32 contregs_idx;
- SCM_UNPACK_RTL_24 (op, contregs_idx);
- contregs =
- SCM_RTL_PROGRAM_FREE_VARIABLE_REF (LOCAL_REF (0), contregs_idx);
- SYNC_IP ();
- scm_i_check_continuation (contregs);
- vm_return_to_continuation (scm_i_contregs_vm (contregs),
- scm_i_contregs_vm_cont (contregs),
- FRAME_LOCALS_COUNT (), fp);
- scm_i_reinstate_continuation (contregs);
- /* no NEXT */
- abort ();
- }
- /* compose-continuation cont:24
- *
- * Compose a partial continution with the current continuation. The
- * arguments to the continuation are taken from the stack. CONT is a
- * free variable containing the reified continuation. This
- * instruction is part of the implementation of partial continuations,
- * and is not generated by the compiler.
- */
- VM_DEFINE_OP (10, compose_continuation, "compose-continuation", OP1 (U8_U24))
- {
- SCM vmcont;
- scm_t_uint32 cont_idx;
- SCM_UNPACK_RTL_24 (op, cont_idx);
- vmcont = LOCAL_REF (cont_idx);
- SYNC_IP ();
- VM_ASSERT (SCM_VM_CONT_REWINDABLE_P (vmcont),
- vm_error_continuation_not_rewindable (vmcont));
- vm_reinstate_partial_continuation (vm, vmcont, FRAME_LOCALS_COUNT (), fp,
- ¤t_thread->dynstack,
- ®isters);
- CACHE_REGISTER ();
- NEXT (0);
- }
- /* tail-apply _:24
- *
- * Tail-apply the procedure in local slot 0 to the rest of the
- * arguments. This instruction is part of the implementation of
- * `apply', and is not generated by the compiler.
- */
- VM_DEFINE_OP (11, tail_apply, "tail-apply", OP1 (U8_X24))
- {
- int i, list_idx, list_len, nargs;
- SCM list;
- VM_HANDLE_INTERRUPTS;
- VM_ASSERT (FRAME_LOCALS_COUNT () >= 2, abort ());
- nargs = FRAME_LOCALS_COUNT ();
- list_idx = nargs - 1;
- list = LOCAL_REF (list_idx);
- list_len = scm_ilength (list);
- VM_ASSERT (list_len >= 0, vm_error_apply_to_non_list (list));
- nargs = nargs - 2 + list_len;
- ALLOC_FRAME (nargs);
- for (i = 0; i < list_idx; i++)
- LOCAL_SET(i - 1, LOCAL_REF (i));
- /* Null out these slots, just in case there are less than 2 elements
- in the list. */
- LOCAL_SET (list_idx - 1, SCM_UNDEFINED);
- LOCAL_SET (list_idx, SCM_UNDEFINED);
- for (i = 0; i < list_len; i++, list = SCM_CDR (list))
- LOCAL_SET (list_idx - 1 + i, SCM_CAR (list));
- APPLY_HOOK ();
- if (SCM_UNLIKELY (!SCM_RTL_PROGRAM_P (SCM_FRAME_PROGRAM (fp))))
- goto apply;
- ip = SCM_RTL_PROGRAM_CODE (SCM_FRAME_PROGRAM (fp));
- NEXT (0);
- }
- /* call/cc _:24
- *
- * Capture the current continuation, and tail-apply the procedure in
- * local slot 0 to it. This instruction is part of the implementation
- * of `call/cc', and is not generated by the compiler.
- */
- VM_DEFINE_OP (12, call_cc, "call/cc", OP1 (U8_X24))
- #if 0
- {
- SCM vm_cont, cont;
- scm_t_dynstack *dynstack;
- VM_HANDLE_INTERRUPTS;
- SYNC_IP ();
- dynstack = scm_dynstack_capture_all (¤t_thread->dynstack);
- vm_cont = scm_i_vm_capture_stack (vp->stack_base,
- SCM_FRAME_DYNAMIC_LINK (fp),
- SCM_FRAME_LOWER_ADDRESS (fp) - 1,
- SCM_FRAME_RETURN_ADDRESS (fp),
- SCM_FRAME_MV_RETURN_ADDRESS (fp),
- dynstack,
- 0);
- cont = scm_i_make_continuation (®isters, vm, vm_cont);
- fp[-1] = fp[0];
- fp[0] = cont;
- RESET_FRAME (2);
- APPLY_HOOK ();
- if (SCM_UNLIKELY (!SCM_RTL_PROGRAM_P (SCM_FRAME_PROGRAM (fp))))
- goto apply;
- ip = SCM_RTL_PROGRAM_CODE (SCM_FRAME_PROGRAM (fp));
- NEXT (0);
- }
- #else
- abort();
- #endif
-
- /*
- * Function prologues
- */
- /* br-if-nargs-ne expected:24 _:8 offset:24
- * br-if-nargs-lt expected:24 _:8 offset:24
- * br-if-nargs-gt expected:24 _:8 offset:24
- *
- * If the number of actual arguments is not equal, less than, or greater
- * than EXPECTED, respectively, add OFFSET, a signed 24-bit number, to
- * the current instruction pointer.
- */
- VM_DEFINE_OP (13, br_if_nargs_ne, "br-if-nargs-ne", OP2 (U8_U24, X8_L24))
- {
- BR_NARGS (!=);
- }
- VM_DEFINE_OP (14, br_if_nargs_lt, "br-if-nargs-lt", OP2 (U8_U24, X8_L24))
- {
- BR_NARGS (<);
- }
- VM_DEFINE_OP (15, br_if_nargs_gt, "br-if-nargs-gt", OP2 (U8_U24, X8_L24))
- {
- BR_NARGS (>);
- }
- /* assert-nargs-ee expected:24
- * assert-nargs-ge expected:24
- * assert-nargs-le expected:24
- *
- * If the number of actual arguments is not ==, >=, or <= EXPECTED,
- * respectively, signal an error.
- */
- VM_DEFINE_OP (16, assert_nargs_ee, "assert-nargs-ee", OP1 (U8_U24))
- {
- scm_t_uint32 expected;
- SCM_UNPACK_RTL_24 (op, expected);
- VM_ASSERT (FRAME_LOCALS_COUNT () == expected,
- vm_error_wrong_num_args (SCM_FRAME_PROGRAM (fp)));
- NEXT (1);
- }
- VM_DEFINE_OP (17, assert_nargs_ge, "assert-nargs-ge", OP1 (U8_U24))
- {
- scm_t_uint32 expected;
- SCM_UNPACK_RTL_24 (op, expected);
- VM_ASSERT (FRAME_LOCALS_COUNT () >= expected,
- vm_error_wrong_num_args (SCM_FRAME_PROGRAM (fp)));
- NEXT (1);
- }
- VM_DEFINE_OP (18, assert_nargs_le, "assert-nargs-le", OP1 (U8_U24))
- {
- scm_t_uint32 expected;
- SCM_UNPACK_RTL_24 (op, expected);
- VM_ASSERT (FRAME_LOCALS_COUNT () <= expected,
- vm_error_wrong_num_args (SCM_FRAME_PROGRAM (fp)));
- NEXT (1);
- }
- /* alloc-frame nlocals:24
- *
- * Ensure that there is space on the stack for NLOCALS local variables,
- * setting them all to SCM_UNDEFINED, except those nargs values that
- * were passed as arguments and procedure.
- */
- VM_DEFINE_OP (19, alloc_frame, "alloc-frame", OP1 (U8_U24))
- {
- scm_t_uint32 nlocals, nargs;
- SCM_UNPACK_RTL_24 (op, nlocals);
- nargs = FRAME_LOCALS_COUNT ();
- ALLOC_FRAME (nlocals);
- while (nlocals-- > nargs)
- LOCAL_SET (nlocals, SCM_UNDEFINED);
- NEXT (1);
- }
- /* reset-frame nlocals:24
- *
- * Like alloc-frame, but doesn't check that the stack is big enough.
- * Used to reset the frame size to something less than the size that
- * was previously set via alloc-frame.
- */
- VM_DEFINE_OP (20, reset_frame, "reset-frame", OP1 (U8_U24))
- {
- scm_t_uint32 nlocals;
- SCM_UNPACK_RTL_24 (op, nlocals);
- RESET_FRAME (nlocals);
- NEXT (1);
- }
- /* assert-nargs-ee/locals expected:12 nlocals:12
- *
- * Equivalent to a sequence of assert-nargs-ee and reserve-locals. The
- * number of locals reserved is EXPECTED + NLOCALS.
- */
- VM_DEFINE_OP (21, assert_nargs_ee_locals, "assert-nargs-ee/locals", OP1 (U8_U12_U12))
- {
- scm_t_uint16 expected, nlocals;
- SCM_UNPACK_RTL_12_12 (op, expected, nlocals);
- VM_ASSERT (FRAME_LOCALS_COUNT () == expected,
- vm_error_wrong_num_args (SCM_FRAME_PROGRAM (fp)));
- ALLOC_FRAME (expected + nlocals);
- while (nlocals--)
- LOCAL_SET (expected + nlocals, SCM_UNDEFINED);
- NEXT (1);
- }
- /* bind-kwargs nreq:24 allow-other-keys:1 has-rest:1 _:6 nreq-and-opt:24
- * _:8 ntotal:24 kw-offset:32
- *
- * Find the last positional argument, and shuffle all the rest above
- * NTOTAL. Initialize the intervening locals to SCM_UNDEFINED. Then
- * load the constant at KW-OFFSET words from the current IP, and use it
- * to bind keyword arguments. If HAS-REST, collect all shuffled
- * arguments into a list, and store it in NREQ-AND-OPT. Finally, clear
- * the arguments that we shuffled up.
- *
- * A macro-mega-instruction.
- */
- VM_DEFINE_OP (22, bind_kwargs, "bind-kwargs", OP4 (U8_U24, U8_U24, X8_U24, N32))
- {
- scm_t_uint32 nreq, nreq_and_opt, ntotal, npositional, nkw, n, nargs;
- scm_t_int32 kw_offset;
- scm_t_bits kw_bits;
- SCM kw;
- char allow_other_keys, has_rest;
- SCM_UNPACK_RTL_24 (op, nreq);
- allow_other_keys = ip[1] & 0x1;
- has_rest = ip[1] & 0x2;
- SCM_UNPACK_RTL_24 (ip[1], nreq_and_opt);
- SCM_UNPACK_RTL_24 (ip[2], ntotal);
- kw_offset = ip[3];
- kw_bits = (scm_t_bits) (ip + kw_offset);
- VM_ASSERT (!(kw_bits & 0x7), abort());
- kw = SCM_PACK (kw_bits);
- nargs = FRAME_LOCALS_COUNT ();
- /* look in optionals for first keyword or last positional */
- /* starting after the last required positional arg */
- npositional = nreq;
- while (/* while we have args */
- npositional < nargs
- /* and we still have positionals to fill */
- && npositional < nreq_and_opt
- /* and we haven't reached a keyword yet */
- && !scm_is_keyword (LOCAL_REF (npositional)))
- /* bind this optional arg (by leaving it in place) */
- npositional++;
- nkw = nargs - npositional;
- /* shuffle non-positional arguments above ntotal */
- ALLOC_FRAME (ntotal + nkw);
- n = nkw;
- while (n--)
- LOCAL_SET (ntotal + n, LOCAL_REF (npositional + n));
- /* and fill optionals & keyword args with SCM_UNDEFINED */
- n = npositional;
- while (n < ntotal)
- LOCAL_SET (n++, SCM_UNDEFINED);
- VM_ASSERT (has_rest || (nkw % 2) == 0,
- vm_error_kwargs_length_not_even (SCM_FRAME_PROGRAM (fp)));
- /* Now bind keywords, in the order given. */
- for (n = 0; n < nkw; n++)
- if (scm_is_keyword (LOCAL_REF (ntotal + n)))
- {
- SCM walk;
- for (walk = kw; scm_is_pair (walk); walk = SCM_CDR (walk))
- if (scm_is_eq (SCM_CAAR (walk), LOCAL_REF (ntotal + n)))
- {
- SCM si = SCM_CDAR (walk);
- LOCAL_SET (SCM_I_INUMP (si) ? SCM_I_INUM (si) : scm_to_uint32 (si),
- LOCAL_REF (ntotal + n + 1));
- break;
- }
- VM_ASSERT (scm_is_pair (walk) || allow_other_keys,
- vm_error_kwargs_unrecognized_keyword (SCM_FRAME_PROGRAM (fp),
- LOCAL_REF (ntotal + n)));
- n++;
- }
- else
- VM_ASSERT (has_rest, vm_error_kwargs_invalid_keyword (SCM_FRAME_PROGRAM (fp),
- LOCAL_REF (ntotal + n)));
- if (has_rest)
- {
- SCM rest = SCM_EOL;
- n = nkw;
- while (n--)
- rest = scm_cons (LOCAL_REF (ntotal + n), rest);
- LOCAL_SET (nreq_and_opt, rest);
- }
- RESET_FRAME (ntotal);
- NEXT (4);
- }
- /* bind-rest dst:24
- *
- * Collect any arguments at or above DST into a list, and store that
- * list at DST.
- */
- VM_DEFINE_OP (23, bind_rest, "bind-rest", OP1 (U8_U24) | OP_DST)
- {
- scm_t_uint32 dst, nargs;
- SCM rest = SCM_EOL;
- SCM_UNPACK_RTL_24 (op, dst);
- nargs = FRAME_LOCALS_COUNT ();
- while (nargs-- > dst)
- {
- rest = scm_cons (LOCAL_REF (nargs), rest);
- LOCAL_SET (nargs, SCM_UNDEFINED);
- }
- LOCAL_SET (dst, rest);
- RESET_FRAME (dst + 1);
- NEXT (1);
- }
-
- /*
- * Branching instructions
- */
- /* br offset:24
- *
- * Add OFFSET, a signed 24-bit number, to the current instruction
- * pointer.
- */
- VM_DEFINE_OP (24, br, "br", OP1 (U8_L24))
- {
- scm_t_int32 offset = op;
- offset >>= 8; /* Sign-extending shift. */
- NEXT (offset);
- }
- /* br-if-true test:24 invert:1 _:7 offset:24
- *
- * If the value in TEST is true for the purposes of Scheme, add
- * OFFSET, a signed 24-bit number, to the current instruction pointer.
- */
- VM_DEFINE_OP (25, br_if_true, "br-if-true", OP2 (U8_U24, B1_X7_L24))
- {
- BR_UNARY (x, scm_is_true (x));
- }
- /* br-if-null test:24 invert:1 _:7 offset:24
- *
- * If the value in TEST is the end-of-list or Lisp nil, add OFFSET, a
- * signed 24-bit number, to the current instruction pointer.
- */
- VM_DEFINE_OP (26, br_if_null, "br-if-null", OP2 (U8_U24, B1_X7_L24))
- {
- BR_UNARY (x, scm_is_null (x));
- }
- /* br-if-nil test:24 invert:1 _:7 offset:24
- *
- * If the value in TEST is false to Lisp, add OFFSET, a signed 24-bit
- * number, to the current instruction pointer.
- */
- VM_DEFINE_OP (27, br_if_nil, "br-if-nil", OP2 (U8_U24, B1_X7_L24))
- {
- BR_UNARY (x, scm_is_lisp_false (x));
- }
- /* br-if-pair test:24 invert:1 _:7 offset:24
- *
- * If the value in TEST is a pair, add OFFSET, a signed 24-bit number,
- * to the current instruction pointer.
- */
- VM_DEFINE_OP (28, br_if_pair, "br-if-pair", OP2 (U8_U24, B1_X7_L24))
- {
- BR_UNARY (x, scm_is_pair (x));
- }
- /* br-if-struct test:24 invert:1 _:7 offset:24
- *
- * If the value in TEST is a struct, add OFFSET, a signed 24-bit
- * number, to the current instruction pointer.
- */
- VM_DEFINE_OP (29, br_if_struct, "br-if-struct", OP2 (U8_U24, B1_X7_L24))
- {
- BR_UNARY (x, SCM_STRUCTP (x));
- }
- /* br-if-char test:24 invert:1 _:7 offset:24
- *
- * If the value in TEST is a char, add OFFSET, a signed 24-bit number,
- * to the current instruction pointer.
- */
- VM_DEFINE_OP (30, br_if_char, "br-if-char", OP2 (U8_U24, B1_X7_L24))
- {
- BR_UNARY (x, SCM_CHARP (x));
- }
- /* br-if-tc7 test:24 invert:1 tc7:7 offset:24
- *
- * If the value in TEST has the TC7 given in the second word, add
- * OFFSET, a signed 24-bit number, to the current instruction pointer.
- */
- VM_DEFINE_OP (31, br_if_tc7, "br-if-tc7", OP2 (U8_U24, B1_U7_L24))
- {
- BR_UNARY (x, SCM_HAS_TYP7 (x, (ip[1] >> 1) & 0x7f));
- }
- /* br-if-eq a:12 b:12 invert:1 _:7 offset:24
- *
- * If the value in A is eq? to the value in B, add OFFSET, a signed
- * 24-bit number, to the current instruction pointer.
- */
- VM_DEFINE_OP (32, br_if_eq, "br-if-eq", OP2 (U8_U12_U12, B1_X7_L24))
- {
- BR_BINARY (x, y, scm_is_eq (x, y));
- }
- /* br-if-eqv a:12 b:12 invert:1 _:7 offset:24
- *
- * If the value in A is eqv? to the value in B, add OFFSET, a signed
- * 24-bit number, to the current instruction pointer.
- */
- VM_DEFINE_OP (33, br_if_eqv, "br-if-eqv", OP2 (U8_U12_U12, B1_X7_L24))
- {
- BR_BINARY (x, y,
- scm_is_eq (x, y)
- || (SCM_NIMP (x) && SCM_NIMP (y)
- && scm_is_true (scm_eqv_p (x, y))));
- }
- // FIXME: remove, have compiler inline eqv test instead
- /* br-if-equal a:12 b:12 invert:1 _:7 offset:24
- *
- * If the value in A is equal? to the value in B, add OFFSET, a signed
- * 24-bit number, to the current instruction pointer.
- */
- // FIXME: should sync_ip before calling out?
- VM_DEFINE_OP (34, br_if_equal, "br-if-equal", OP2 (U8_U12_U12, B1_X7_L24))
- {
- BR_BINARY (x, y,
- scm_is_eq (x, y)
- || (SCM_NIMP (x) && SCM_NIMP (y)
- && scm_is_true (scm_equal_p (x, y))));
- }
- /* br-if-= a:12 b:12 invert:1 _:7 offset:24
- *
- * If the value in A is = to the value in B, add OFFSET, a signed
- * 24-bit number, to the current instruction pointer.
- */
- VM_DEFINE_OP (35, br_if_ee, "br-if-=", OP2 (U8_U12_U12, B1_X7_L24))
- {
- BR_ARITHMETIC (==, scm_num_eq_p);
- }
- /* br-if-< a:12 b:12 _:8 offset:24
- *
- * If the value in A is < to the value in B, add OFFSET, a signed
- * 24-bit number, to the current instruction pointer.
- */
- VM_DEFINE_OP (36, br_if_lt, "br-if-<", OP2 (U8_U12_U12, B1_X7_L24))
- {
- BR_ARITHMETIC (<, scm_less_p);
- }
- /* br-if-<= a:12 b:12 _:8 offset:24
- *
- * If the value in A is <= to the value in B, add OFFSET, a signed
- * 24-bit number, to the current instruction pointer.
- */
- VM_DEFINE_OP (37, br_if_le, "br-if-<=", OP2 (U8_U12_U12, B1_X7_L24))
- {
- BR_ARITHMETIC (<=, scm_leq_p);
- }
-
- /*
- * Lexical binding instructions
- */
- /* mov dst:12 src:12
- *
- * Copy a value from one local slot to another.
- */
- VM_DEFINE_OP (38, mov, "mov", OP1 (U8_U12_U12) | OP_DST)
- {
- scm_t_uint16 dst;
- scm_t_uint16 src;
- SCM_UNPACK_RTL_12_12 (op, dst, src);
- LOCAL_SET (dst, LOCAL_REF (src));
- NEXT (1);
- }
- /* long-mov dst:24 _:8 src:24
- *
- * Copy a value from one local slot to another.
- */
- VM_DEFINE_OP (39, long_mov, "long-mov", OP2 (U8_U24, X8_U24) | OP_DST)
- {
- scm_t_uint32 dst;
- scm_t_uint32 src;
- SCM_UNPACK_RTL_24 (op, dst);
- SCM_UNPACK_RTL_24 (ip[1], src);
- LOCAL_SET (dst, LOCAL_REF (src));
- NEXT (2);
- }
- /* box dst:12 src:12
- *
- * Create a new variable holding SRC, and place it in DST.
- */
- VM_DEFINE_OP (40, box, "box", OP1 (U8_U12_U12) | OP_DST)
- {
- scm_t_uint16 dst, src;
- SCM_UNPACK_RTL_12_12 (op, dst, src);
- LOCAL_SET (dst, scm_cell (scm_tc7_variable, SCM_UNPACK (LOCAL_REF (src))));
- NEXT (1);
- }
- /* box-ref dst:12 src:12
- *
- * Unpack the variable at SRC into DST, asserting that the variable is
- * actually bound.
- */
- VM_DEFINE_OP (41, box_ref, "box-ref", OP1 (U8_U12_U12) | OP_DST)
- {
- scm_t_uint16 dst, src;
- SCM var;
- SCM_UNPACK_RTL_12_12 (op, dst, src);
- var = LOCAL_REF (src);
- VM_ASSERT (SCM_VARIABLEP (var), abort ());
- VM_ASSERT (VARIABLE_BOUNDP (var),
- vm_error_unbound (SCM_FRAME_PROGRAM (fp), var));
- LOCAL_SET (dst, VARIABLE_REF (var));
- NEXT (1);
- }
- /* box-set! dst:12 src:12
- *
- * Set the contents of the variable at DST to SET.
- */
- VM_DEFINE_OP (42, box_set, "box-set!", OP1 (U8_U12_U12))
- {
- scm_t_uint16 dst, src;
- SCM var;
- SCM_UNPACK_RTL_12_12 (op, dst, src);
- var = LOCAL_REF (dst);
- VM_ASSERT (SCM_VARIABLEP (var), abort ());
- VARIABLE_SET (var, LOCAL_REF (src));
- NEXT (1);
- }
- /* make-closure dst:24 offset:32 _:8 nfree:24
- *
- * Make a new closure, and write it to DST. The code for the closure
- * will be found at OFFSET words from the current IP. OFFSET is a
- * signed 32-bit integer. Space for NFREE free variables will be
- * allocated.
- */
- VM_DEFINE_OP (43, make_closure, "make-closure", OP3 (U8_U24, L32, X8_U24) | OP_DST)
- {
- scm_t_uint32 dst, nfree, n;
- scm_t_int32 offset;
- SCM closure;
- SCM_UNPACK_RTL_24 (op, dst);
- offset = ip[1];
- SCM_UNPACK_RTL_24 (ip[2], nfree);
- // FIXME: Assert range of nfree?
- closure = scm_words (scm_tc7_rtl_program | (nfree << 16), nfree + 2);
- SCM_SET_CELL_WORD_1 (closure, ip + offset);
- // FIXME: Elide these initializations?
- for (n = 0; n < nfree; n++)
- SCM_RTL_PROGRAM_FREE_VARIABLE_SET (closure, n, SCM_BOOL_F);
- LOCAL_SET (dst, closure);
- NEXT (3);
- }
- /* free-ref dst:12 src:12 _:8 idx:24
- *
- * Load free variable IDX from the closure SRC into local slot DST.
- */
- VM_DEFINE_OP (44, free_ref, "free-ref", OP2 (U8_U12_U12, X8_U24) | OP_DST)
- {
- scm_t_uint16 dst, src;
- scm_t_uint32 idx;
- SCM_UNPACK_RTL_12_12 (op, dst, src);
- SCM_UNPACK_RTL_24 (ip[1], idx);
- /* CHECK_FREE_VARIABLE (src); */
- LOCAL_SET (dst, SCM_RTL_PROGRAM_FREE_VARIABLE_REF (LOCAL_REF (src), idx));
- NEXT (2);
- }
- /* free-set! dst:12 src:12 _8 idx:24
- *
- * Set free variable IDX from the closure DST to SRC.
- */
- VM_DEFINE_OP (45, free_set, "free-set!", OP2 (U8_U12_U12, X8_U24))
- {
- scm_t_uint16 dst, src;
- scm_t_uint32 idx;
- SCM_UNPACK_RTL_12_12 (op, dst, src);
- SCM_UNPACK_RTL_24 (ip[1], idx);
- /* CHECK_FREE_VARIABLE (src); */
- SCM_RTL_PROGRAM_FREE_VARIABLE_SET (LOCAL_REF (dst), idx, LOCAL_REF (src));
- NEXT (2);
- }
-
- /*
- * Immediates and statically allocated non-immediates
- */
- /* make-short-immediate dst:8 low-bits:16
- *
- * Make an immediate whose low bits are LOW-BITS, and whose top bits are
- * 0.
- */
- VM_DEFINE_OP (46, make_short_immediate, "make-short-immediate", OP1 (U8_U8_I16) | OP_DST)
- {
- scm_t_uint8 dst;
- scm_t_bits val;
- SCM_UNPACK_RTL_8_16 (op, dst, val);
- LOCAL_SET (dst, SCM_PACK (val));
- NEXT (1);
- }
- /* make-long-immediate dst:24 low-bits:32
- *
- * Make an immediate whose low bits are LOW-BITS, and whose top bits are
- * 0.
- */
- VM_DEFINE_OP (47, make_long_immediate, "make-long-immediate", OP2 (U8_U24, I32))
- {
- scm_t_uint8 dst;
- scm_t_bits val;
- SCM_UNPACK_RTL_24 (op, dst);
- val = ip[1];
- LOCAL_SET (dst, SCM_PACK (val));
- NEXT (2);
- }
- /* make-long-long-immediate dst:24 high-bits:32 low-bits:32
- *
- * Make an immediate with HIGH-BITS and LOW-BITS.
- */
- VM_DEFINE_OP (48, make_long_long_immediate, "make-long-long-immediate", OP3 (U8_U24, A32, B32) | OP_DST)
- {
- scm_t_uint8 dst;
- scm_t_bits val;
- SCM_UNPACK_RTL_24 (op, dst);
- #if SIZEOF_SCM_T_BITS > 4
- val = ip[1];
- val <<= 32;
- val |= ip[2];
- #else
- ASSERT (ip[1] == 0);
- val = ip[2];
- #endif
- LOCAL_SET (dst, SCM_PACK (val));
- NEXT (3);
- }
- /* make-non-immediate dst:24 offset:32
- *
- * Load a pointer to statically allocated memory into DST. The
- * object's memory is will be found OFFSET 32-bit words away from the
- * current instruction pointer. OFFSET is a signed value. The
- * intention here is that the compiler would produce an object file
- * containing the words of a non-immediate object, and this
- * instruction creates a pointer to that memory, effectively
- * resurrecting that object.
- *
- * Whether the object is mutable or immutable depends on where it was
- * allocated by the compiler, and loaded by the loader.
- */
- VM_DEFINE_OP (49, make_non_immediate, "make-non-immediate", OP2 (U8_U24, N32) | OP_DST)
- {
- scm_t_uint32 dst;
- scm_t_int32 offset;
- scm_t_uint32* loc;
- scm_t_bits unpacked;
- SCM_UNPACK_RTL_24 (op, dst);
- offset = ip[1];
- loc = ip + offset;
- unpacked = (scm_t_bits) loc;
- VM_ASSERT (!(unpacked & 0x7), abort());
- LOCAL_SET (dst, SCM_PACK (unpacked));
- NEXT (2);
- }
- /* static-ref dst:24 offset:32
- *
- * Load a SCM value into DST. The SCM value will be fetched from
- * memory, OFFSET 32-bit words away from the current instruction
- * pointer. OFFSET is a signed value.
- *
- * The intention is for this instruction to be used to load constants
- * that the compiler is unable to statically allocate, like symbols.
- * These values would be initialized when the object file loads.
- */
- VM_DEFINE_OP (50, static_ref, "static-ref", OP2 (U8_U24, S32))
- {
- scm_t_uint32 dst;
- scm_t_int32 offset;
- scm_t_uint32* loc;
- scm_t_uintptr loc_bits;
- SCM_UNPACK_RTL_24 (op, dst);
- offset = ip[1];
- loc = ip + offset;
- loc_bits = (scm_t_uintptr) loc;
- VM_ASSERT (ALIGNED_P (loc, SCM), abort());
- LOCAL_SET (dst, *((SCM *) loc_bits));
- NEXT (2);
- }
- /* static-set! src:24 offset:32
- *
- * Store a SCM value into memory, OFFSET 32-bit words away from the
- * current instruction pointer. OFFSET is a signed value.
- */
- VM_DEFINE_OP (51, static_set, "static-set!", OP2 (U8_U24, LO32))
- {
- scm_t_uint32 src;
- scm_t_int32 offset;
- scm_t_uint32* loc;
- SCM_UNPACK_RTL_24 (op, src);
- offset = ip[1];
- loc = ip + offset;
- VM_ASSERT (ALIGNED_P (loc, SCM), abort());
- *((SCM *) loc) = LOCAL_REF (src);
- NEXT (2);
- }
- /* link-procedure! src:24 offset:32
- *
- * Set the code pointer of the procedure in SRC to point OFFSET 32-bit
- * words away from the current instruction pointer. OFFSET is a
- * signed value.
- */
- VM_DEFINE_OP (52, link_procedure, "link-procedure!", OP2 (U8_U24, L32))
- {
- scm_t_uint32 src;
- scm_t_int32 offset;
- scm_t_uint32* loc;
- SCM_UNPACK_RTL_24 (op, src);
- offset = ip[1];
- loc = ip + offset;
- SCM_SET_CELL_WORD_1 (LOCAL_REF (src), (scm_t_bits) loc);
- NEXT (2);
- }
-
- /*
- * Mutable top-level bindings
- */
- /* There are three slightly different ways to resolve toplevel
- variables.
- 1. A toplevel reference outside of a function. These need to be
- looked up when the expression is evaluated -- no later, and no
- before. They are looked up relative to the module that is
- current when the expression is evaluated. For example:
- (if (foo) a b)
- The "resolve" instruction resolves the variable (box), and then
- access is via box-ref or box-set!.
- 2. A toplevel reference inside a function. These are looked up
- relative to the module that was current when the function was
- defined. Unlike code at the toplevel, which is usually run only
- once, these bindings benefit from memoized lookup, in which the
- variable resulting from the lookup is cached in the function.
- (lambda () (if (foo) a b))
- The toplevel-box instruction is equivalent to "resolve", but
- caches the resulting variable in statically allocated memory.
- 3. A reference to an identifier with respect to a particular
- module. This can happen for primitive references, and
- references residualized by macro expansions. These can always
- be cached. Use module-box for these.
- */
- /* current-module dst:24
- *
- * Store the current module in DST.
- */
- VM_DEFINE_OP (53, current_module, "current-module", OP1 (U8_U24) | OP_DST)
- {
- scm_t_uint32 dst;
- SCM_UNPACK_RTL_24 (op, dst);
- SYNC_IP ();
- LOCAL_SET (dst, scm_current_module ());
- NEXT (1);
- }
- /* resolve dst:24 bound?:1 _:7 sym:24
- *
- * Resolve SYM in the current module, and place the resulting variable
- * in DST.
- */
- VM_DEFINE_OP (54, resolve, "resolve", OP2 (U8_U24, B1_X7_U24) | OP_DST)
- {
- scm_t_uint32 dst;
- scm_t_uint32 sym;
- SCM var;
- SCM_UNPACK_RTL_24 (op, dst);
- SCM_UNPACK_RTL_24 (ip[1], sym);
- SYNC_IP ();
- var = scm_lookup (LOCAL_REF (sym));
- if (ip[1] & 0x1)
- VM_ASSERT (VARIABLE_BOUNDP (var),
- vm_error_unbound (fp[-1], LOCAL_REF (sym)));
- LOCAL_SET (dst, var);
- NEXT (2);
- }
- /* define sym:12 val:12
- *
- * Look up a binding for SYM in the current module, creating it if
- * necessary. Set its value to VAL.
- */
- VM_DEFINE_OP (55, define, "define", OP1 (U8_U12_U12))
- {
- scm_t_uint16 sym, val;
- SCM_UNPACK_RTL_12_12 (op, sym, val);
- SYNC_IP ();
- scm_define (LOCAL_REF (sym), LOCAL_REF (val));
- NEXT (1);
- }
- /* toplevel-box dst:24 var-offset:32 mod-offset:32 sym-offset:32 bound?:1 _:31
- *
- * Load a SCM value. The SCM value will be fetched from memory,
- * VAR-OFFSET 32-bit words away from the current instruction pointer.
- * VAR-OFFSET is a signed value. Up to here, toplevel-box is like
- * static-ref.
- *
- * Then, if the loaded value is a variable, it is placed in DST, and control
- * flow continues.
- *
- * Otherwise, we have to resolve the variable. In that case we load
- * the module from MOD-OFFSET, just as we loaded the variable.
- * Usually the module gets set when the closure is created. The name
- * is an offset to a symbol.
- *
- * We use the module and the symbol to resolve the variable, placing it in
- * DST, and caching the resolved variable so that we will hit the cache next
- * time.
- */
- VM_DEFINE_OP (56, toplevel_box, "toplevel-box", OP5 (U8_U24, S32, S32, N32, B1_X31) | OP_DST)
- {
- scm_t_uint32 dst;
- scm_t_int32 var_offset;
- scm_t_uint32* var_loc_u32;
- SCM *var_loc;
- SCM var;
- SCM_UNPACK_RTL_24 (op, dst);
- var_offset = ip[1];
- var_loc_u32 = ip + var_offset;
- VM_ASSERT (ALIGNED_P (var_loc_u32, SCM), abort());
- var_loc = (SCM *) var_loc_u32;
- var = *var_loc;
- if (SCM_UNLIKELY (!SCM_VARIABLEP (var)))
- {
- SCM mod, sym;
- scm_t_int32 mod_offset = ip[2]; /* signed */
- scm_t_int32 sym_offset = ip[3]; /* signed */
- scm_t_uint32 *mod_loc = ip + mod_offset;
- scm_t_uint32 *sym_loc = ip + sym_offset;
-
- SYNC_IP ();
- VM_ASSERT (ALIGNED_P (mod_loc, SCM), abort());
- VM_ASSERT (ALIGNED_P (sym_loc, SCM), abort());
- mod = *((SCM *) mod_loc);
- sym = *((SCM *) sym_loc);
- var = scm_module_lookup (mod, sym);
- if (ip[4] & 0x1)
- VM_ASSERT (VARIABLE_BOUNDP (var), vm_error_unbound (fp[-1], sym));
- *var_loc = var;
- }
- LOCAL_SET (dst, var);
- NEXT (5);
- }
- /* module-box dst:24 var-offset:32 mod-offset:32 sym-offset:32 bound?:1 _:31
- *
- * Like toplevel-box, except MOD-OFFSET points at the name of a module
- * instead of the module itself.
- */
- VM_DEFINE_OP (57, module_box, "module-box", OP5 (U8_U24, S32, N32, N32, B1_X31) | OP_DST)
- {
- scm_t_uint32 dst;
- scm_t_int32 var_offset;
- scm_t_uint32* var_loc_u32;
- SCM *var_loc;
- SCM var;
- SCM_UNPACK_RTL_24 (op, dst);
- var_offset = ip[1];
- var_loc_u32 = ip + var_offset;
- VM_ASSERT (ALIGNED_P (var_loc_u32, SCM), abort());
- var_loc = (SCM *) var_loc_u32;
- var = *var_loc;
- if (SCM_UNLIKELY (!SCM_VARIABLEP (var)))
- {
- SCM modname, sym;
- scm_t_int32 modname_offset = ip[2]; /* signed */
- scm_t_int32 sym_offset = ip[3]; /* signed */
- scm_t_uint32 *modname_words = ip + modname_offset;
- scm_t_uint32 *sym_loc = ip + sym_offset;
- SYNC_IP ();
- VM_ASSERT (!(((scm_t_uintptr) modname_words) & 0x7), abort());
- VM_ASSERT (ALIGNED_P (sym_loc, SCM), abort());
- modname = SCM_PACK ((scm_t_bits) modname_words);
- sym = *((SCM *) sym_loc);
- if (scm_is_true (SCM_CAR (modname)))
- var = scm_public_lookup (SCM_CDR (modname), sym);
- else
- var = scm_private_lookup (SCM_CDR (modname), sym);
- if (ip[4] & 0x1)
- VM_ASSERT (VARIABLE_BOUNDP (var), vm_error_unbound (fp[-1], sym));
- *var_loc = var;
- }
- LOCAL_SET (dst, var);
- NEXT (5);
- }
-
- /*
- * The dynamic environment
- */
- /* prompt tag:24 flags:8 handler-offset:24
- *
- * Push a new prompt on the dynamic stack, with a tag from TAG and a
- * handler at HANDLER-OFFSET words from the current IP. The handler
- * will expect a multiple-value return.
- */
- VM_DEFINE_OP (58, prompt, "prompt", OP2 (U8_U24, U8_L24))
- #if 0
- {
- scm_t_uint32 tag;
- scm_t_int32 offset;
- scm_t_uint8 escape_only_p;
- scm_t_dynstack_prompt_flags flags;
- SCM_UNPACK_RTL_24 (op, tag);
- escape_only_p = ip[1] & 0xff;
- offset = ip[1];
- offset >>= 8; /* Sign extension */
-
- /* Push the prompt onto the dynamic stack. */
- flags = escape_only_p ? SCM_F_DYNSTACK_PROMPT_ESCAPE_ONLY : 0;
- scm_dynstack_push_prompt (¤t_thread->dynstack, flags,
- LOCAL_REF (tag),
- fp, vp->sp, ip + offset, ®isters);
- NEXT (2);
- }
- #else
- abort();
- #endif
- /* wind winder:12 unwinder:12
- *
- * Push wind and unwind procedures onto the dynamic stack. Note that
- * neither are actually called; the compiler should emit calls to wind
- * and unwind for the normal dynamic-wind control flow. Also note that
- * the compiler should have inserted checks that they wind and unwind
- * procs are thunks, if it could not prove that to be the case.
- */
- VM_DEFINE_OP (59, wind, "wind", OP1 (U8_U12_U12))
- {
- scm_t_uint16 winder, unwinder;
- SCM_UNPACK_RTL_12_12 (op, winder, unwinder);
- scm_dynstack_push_dynwind (¤t_thread->dynstack,
- LOCAL_REF (winder), LOCAL_REF (unwinder));
- NEXT (1);
- }
- /* abort tag:24 _:8 proc:24
- *
- * Return a number of values to a prompt handler. The values are
- * expected in a frame pushed on at PROC.
- */
- VM_DEFINE_OP (60, abort, "abort", OP2 (U8_U24, X8_U24))
- #if 0
- {
- scm_t_uint32 tag, from, nvalues;
- SCM *base;
- SCM_UNPACK_RTL_24 (op, tag);
- SCM_UNPACK_RTL_24 (ip[1], from);
- base = (fp - 1) + from + 3;
- nvalues = FRAME_LOCALS_COUNT () - from - 3;
- SYNC_IP ();
- vm_abort (vm, LOCAL_REF (tag), base, nvalues, ®isters);
- /* vm_abort should not return */
- abort ();
- }
- #else
- abort();
- #endif
- /* unwind _:24
- *
- * A normal exit from the dynamic extent of an expression. Pop the top
- * entry off of the dynamic stack.
- */
- VM_DEFINE_OP (61, unwind, "unwind", OP1 (U8_X24))
- {
- scm_dynstack_pop (¤t_thread->dynstack);
- NEXT (1);
- }
- /* push-fluid fluid:12 value:12
- *
- * Dynamically bind N fluids to values. The fluids are expected to be
- * allocated in a continguous range on the stack, starting from
- * FLUID-BASE. The values do not have this restriction.
- */
- VM_DEFINE_OP (62, push_fluid, "push-fluid", OP1 (U8_U12_U12))
- {
- scm_t_uint32 fluid, value;
- SCM_UNPACK_RTL_12_12 (op, fluid, value);
- scm_dynstack_push_fluid (¤t_thread->dynstack,
- fp[fluid], fp[value],
- current_thread->dynamic_state);
- NEXT (1);
- }
- /* pop-fluid _:24
- *
- * Leave the dynamic extent of a with-fluids expression, restoring the
- * fluids to their previous values.
- */
- VM_DEFINE_OP (63, pop_fluid, "pop-fluid", OP1 (U8_X24))
- {
- /* This function must not allocate. */
- scm_dynstack_unwind_fluid (¤t_thread->dynstack,
- current_thread->dynamic_state);
- NEXT (1);
- }
- /* fluid-ref dst:12 src:12
- *
- * Reference the fluid in SRC, and place the value in DST.
- */
- VM_DEFINE_OP (64, fluid_ref, "fluid-ref", OP1 (U8_U12_U12) | OP_DST)
- {
- scm_t_uint16 dst, src;
- size_t num;
- SCM fluid, fluids;
- SCM_UNPACK_RTL_12_12 (op, dst, src);
- fluid = LOCAL_REF (src);
- fluids = SCM_I_DYNAMIC_STATE_FLUIDS (current_thread->dynamic_state);
- if (SCM_UNLIKELY (!SCM_FLUID_P (fluid))
- || ((num = SCM_I_FLUID_NUM (fluid)) >= SCM_SIMPLE_VECTOR_LENGTH (fluids)))
- {
- /* Punt dynstate expansion and error handling to the C proc. */
- SYNC_IP ();
- LOCAL_SET (dst, scm_fluid_ref (fluid));
- }
- else
- {
- SCM val = SCM_SIMPLE_VECTOR_REF (fluids, num);
- if (scm_is_eq (val, SCM_UNDEFINED))
- val = SCM_I_FLUID_DEFAULT (fluid);
- VM_ASSERT (!scm_is_eq (val, SCM_UNDEFINED),
- vm_error_unbound_fluid (program, fluid));
- LOCAL_SET (dst, val);
- }
- NEXT (1);
- }
- /* fluid-set fluid:12 val:12
- *
- * Set the value of the fluid in DST to the value in SRC.
- */
- VM_DEFINE_OP (65, fluid_set, "fluid-set", OP1 (U8_U12_U12))
- {
- scm_t_uint16 a, b;
- size_t num;
- SCM fluid, fluids;
- SCM_UNPACK_RTL_12_12 (op, a, b);
- fluid = LOCAL_REF (a);
- fluids = SCM_I_DYNAMIC_STATE_FLUIDS (current_thread->dynamic_state);
- if (SCM_UNLIKELY (!SCM_FLUID_P (fluid))
- || ((num = SCM_I_FLUID_NUM (fluid)) >= SCM_SIMPLE_VECTOR_LENGTH (fluids)))
- {
- /* Punt dynstate expansion and error handling to the C proc. */
- SYNC_IP ();
- scm_fluid_set_x (fluid, LOCAL_REF (b));
- }
- else
- SCM_SIMPLE_VECTOR_SET (fluids, num, LOCAL_REF (b));
- NEXT (1);
- }
-
- /*
- * Strings, symbols, and keywords
- */
- /* string-length dst:12 src:12
- *
- * Store the length of the string in SRC in DST.
- */
- VM_DEFINE_OP (66, string_length, "string-length", OP1 (U8_U12_U12) | OP_DST)
- {
- ARGS1 (str);
- if (SCM_LIKELY (scm_is_string (str)))
- RETURN (SCM_I_MAKINUM (scm_i_string_length (str)));
- else
- {
- SYNC_IP ();
- RETURN (scm_string_length (str));
- }
- }
- /* string-ref dst:8 src:8 idx:8
- *
- * Fetch the character at position IDX in the string in SRC, and store
- * it in DST.
- */
- VM_DEFINE_OP (67, string_ref, "string-ref", OP1 (U8_U8_U8_U8) | OP_DST)
- {
- scm_t_signed_bits i = 0;
- ARGS2 (str, idx);
- if (SCM_LIKELY (scm_is_string (str)
- && SCM_I_INUMP (idx)
- && ((i = SCM_I_INUM (idx)) >= 0)
- && i < scm_i_string_length (str)))
- RETURN (SCM_MAKE_CHAR (scm_i_string_ref (str, i)));
- else
- {
- SYNC_IP ();
- RETURN (scm_string_ref (str, idx));
- }
- }
- /* No string-set! instruction, as there is no good fast path there. */
- /* string-to-number dst:12 src:12
- *
- * Parse a string in SRC to a number, and store in DST.
- */
- VM_DEFINE_OP (68, string_to_number, "string->number", OP1 (U8_U12_U12) | OP_DST)
- {
- scm_t_uint16 dst, src;
- SCM_UNPACK_RTL_12_12 (op, dst, src);
- SYNC_IP ();
- LOCAL_SET (dst,
- scm_string_to_number (LOCAL_REF (src),
- SCM_UNDEFINED /* radix = 10 */));
- NEXT (1);
- }
- /* string-to-symbol dst:12 src:12
- *
- * Parse a string in SRC to a symbol, and store in DST.
- */
- VM_DEFINE_OP (69, string_to_symbol, "string->symbol", OP1 (U8_U12_U12) | OP_DST)
- {
- scm_t_uint16 dst, src;
- SCM_UNPACK_RTL_12_12 (op, dst, src);
- SYNC_IP ();
- LOCAL_SET (dst, scm_string_to_symbol (LOCAL_REF (src)));
- NEXT (1);
- }
- /* symbol->keyword dst:12 src:12
- *
- * Make a keyword from the symbol in SRC, and store it in DST.
- */
- VM_DEFINE_OP (70, symbol_to_keyword, "symbol->keyword", OP1 (U8_U12_U12) | OP_DST)
- {
- scm_t_uint16 dst, src;
- SCM_UNPACK_RTL_12_12 (op, dst, src);
- SYNC_IP ();
- LOCAL_SET (dst, scm_symbol_to_keyword (LOCAL_REF (src)));
- NEXT (1);
- }
-
- /*
- * Pairs
- */
- /* cons dst:8 car:8 cdr:8
- *
- * Cons CAR and CDR, and store the result in DST.
- */
- VM_DEFINE_OP (71, cons, "cons", OP1 (U8_U8_U8_U8) | OP_DST)
- {
- ARGS2 (x, y);
- RETURN (scm_cons (x, y));
- }
- /* car dst:12 src:12
- *
- * Place the car of SRC in DST.
- */
- VM_DEFINE_OP (72, car, "car", OP1 (U8_U12_U12) | OP_DST)
- {
- ARGS1 (x);
- VM_VALIDATE_PAIR (x, "car");
- RETURN (SCM_CAR (x));
- }
- /* cdr dst:12 src:12
- *
- * Place the cdr of SRC in DST.
- */
- VM_DEFINE_OP (73, cdr, "cdr", OP1 (U8_U12_U12) | OP_DST)
- {
- ARGS1 (x);
- VM_VALIDATE_PAIR (x, "cdr");
- RETURN (SCM_CDR (x));
- }
- /* set-car! pair:12 car:12
- *
- * Set the car of DST to SRC.
- */
- VM_DEFINE_OP (74, set_car, "set-car!", OP1 (U8_U12_U12))
- {
- scm_t_uint16 a, b;
- SCM x, y;
- SCM_UNPACK_RTL_12_12 (op, a, b);
- x = LOCAL_REF (a);
- y = LOCAL_REF (b);
- VM_VALIDATE_PAIR (x, "set-car!");
- SCM_SETCAR (x, y);
- NEXT (1);
- }
- /* set-cdr! pair:12 cdr:12
- *
- * Set the cdr of DST to SRC.
- */
- VM_DEFINE_OP (75, set_cdr, "set-cdr!", OP1 (U8_U12_U12))
- {
- scm_t_uint16 a, b;
- SCM x, y;
- SCM_UNPACK_RTL_12_12 (op, a, b);
- x = LOCAL_REF (a);
- y = LOCAL_REF (b);
- VM_VALIDATE_PAIR (x, "set-car!");
- SCM_SETCDR (x, y);
- NEXT (1);
- }
-
- /*
- * Numeric operations
- */
- /* add dst:8 a:8 b:8
- *
- * Add A to B, and place the result in DST.
- */
- VM_DEFINE_OP (76, add, "add", OP1 (U8_U8_U8_U8) | OP_DST)
- {
- BINARY_INTEGER_OP (+, scm_sum);
- }
- /* add1 dst:12 src:12
- *
- * Add 1 to the value in SRC, and place the result in DST.
- */
- VM_DEFINE_OP (77, add1, "add1", OP1 (U8_U12_U12) | OP_DST)
- {
- ARGS1 (x);
- /* Check for overflow. We must avoid overflow in the signed
- addition below, even if X is not an inum. */
- if (SCM_LIKELY ((scm_t_signed_bits) SCM_UNPACK (x) <= INUM_MAX - INUM_STEP))
- {
- SCM result;
- /* Add 1 to the integer without untagging. */
- result = SCM_PACK ((scm_t_signed_bits) SCM_UNPACK (x) + INUM_STEP);
- if (SCM_LIKELY (SCM_I_INUMP (result)))
- RETURN (result);
- }
- SYNC_IP ();
- RETURN (scm_sum (x, SCM_I_MAKINUM (1)));
- }
- /* sub dst:8 a:8 b:8
- *
- * Subtract B from A, and place the result in DST.
- */
- VM_DEFINE_OP (78, sub, "sub", OP1 (U8_U8_U8_U8) | OP_DST)
- {
- BINARY_INTEGER_OP (-, scm_difference);
- }
- /* sub1 dst:12 src:12
- *
- * Subtract 1 from SRC, and place the result in DST.
- */
- VM_DEFINE_OP (79, sub1, "sub1", OP1 (U8_U12_U12) | OP_DST)
- {
- ARGS1 (x);
- /* Check for overflow. We must avoid overflow in the signed
- subtraction below, even if X is not an inum. */
- if (SCM_LIKELY ((scm_t_signed_bits) SCM_UNPACK (x) >= INUM_MIN + INUM_STEP))
- {
- SCM result;
- /* Substract 1 from the integer without untagging. */
- result = SCM_PACK ((scm_t_signed_bits) SCM_UNPACK (x) - INUM_STEP);
- if (SCM_LIKELY (SCM_I_INUMP (result)))
- RETURN (result);
- }
- SYNC_IP ();
- RETURN (scm_difference (x, SCM_I_MAKINUM (1)));
- }
- /* mul dst:8 a:8 b:8
- *
- * Multiply A and B, and place the result in DST.
- */
- VM_DEFINE_OP (80, mul, "mul", OP1 (U8_U8_U8_U8) | OP_DST)
- {
- ARGS2 (x, y);
- SYNC_IP ();
- RETURN (scm_product (x, y));
- }
- /* div dst:8 a:8 b:8
- *
- * Divide A by B, and place the result in DST.
- */
- VM_DEFINE_OP (81, div, "div", OP1 (U8_U8_U8_U8) | OP_DST)
- {
- ARGS2 (x, y);
- SYNC_IP ();
- RETURN (scm_divide (x, y));
- }
- /* quo dst:8 a:8 b:8
- *
- * Divide A by B, and place the quotient in DST.
- */
- VM_DEFINE_OP (82, quo, "quo", OP1 (U8_U8_U8_U8) | OP_DST)
- {
- ARGS2 (x, y);
- SYNC_IP ();
- RETURN (scm_quotient (x, y));
- }
- /* rem dst:8 a:8 b:8
- *
- * Divide A by B, and place the remainder in DST.
- */
- VM_DEFINE_OP (83, rem, "rem", OP1 (U8_U8_U8_U8) | OP_DST)
- {
- ARGS2 (x, y);
- SYNC_IP ();
- RETURN (scm_remainder (x, y));
- }
- /* mod dst:8 a:8 b:8
- *
- * Place the modulo of A by B in DST.
- */
- VM_DEFINE_OP (84, mod, "mod", OP1 (U8_U8_U8_U8) | OP_DST)
- {
- ARGS2 (x, y);
- SYNC_IP ();
- RETURN (scm_modulo (x, y));
- }
- /* ash dst:8 a:8 b:8
- *
- * Shift A arithmetically by B bits, and place the result in DST.
- */
- VM_DEFINE_OP (85, ash, "ash", OP1 (U8_U8_U8_U8) | OP_DST)
- {
- ARGS2 (x, y);
- if (SCM_I_INUMP (x) && SCM_I_INUMP (y))
- {
- if (SCM_I_INUM (y) < 0)
- /* Right shift, will be a fixnum. */
- RETURN (SCM_I_MAKINUM
- (SCM_SRS (SCM_I_INUM (x),
- (-SCM_I_INUM (y) <= SCM_I_FIXNUM_BIT-1)
- ? -SCM_I_INUM (y) : SCM_I_FIXNUM_BIT-1)));
- else
- /* Left shift. See comments in scm_ash. */
- {
- scm_t_signed_bits nn, bits_to_shift;
- nn = SCM_I_INUM (x);
- bits_to_shift = SCM_I_INUM (y);
- if (bits_to_shift < SCM_I_FIXNUM_BIT-1
- && ((scm_t_bits)
- (SCM_SRS (nn, (SCM_I_FIXNUM_BIT-1 - bits_to_shift)) + 1)
- <= 1))
- RETURN (SCM_I_MAKINUM (nn << bits_to_shift));
- /* fall through */
- }
- /* fall through */
- }
- SYNC_IP ();
- RETURN (scm_ash (x, y));
- }
- /* logand dst:8 a:8 b:8
- *
- * Place the bitwise AND of A and B into DST.
- */
- VM_DEFINE_OP (86, logand, "logand", OP1 (U8_U8_U8_U8) | OP_DST)
- {
- ARGS2 (x, y);
- if (SCM_I_INUMP (x) && SCM_I_INUMP (y))
- /* Compute bitwise AND without untagging */
- RETURN (SCM_PACK (SCM_UNPACK (x) & SCM_UNPACK (y)));
- SYNC_IP ();
- RETURN (scm_logand (x, y));
- }
- /* logior dst:8 a:8 b:8
- *
- * Place the bitwise inclusive OR of A with B in DST.
- */
- VM_DEFINE_OP (87, logior, "logior", OP1 (U8_U8_U8_U8) | OP_DST)
- {
- ARGS2 (x, y);
- if (SCM_I_INUMP (x) && SCM_I_INUMP (y))
- /* Compute bitwise OR without untagging */
- RETURN (SCM_PACK (SCM_UNPACK (x) | SCM_UNPACK (y)));
- SYNC_IP ();
- RETURN (scm_logior (x, y));
- }
- /* logxor dst:8 a:8 b:8
- *
- * Place the bitwise exclusive OR of A with B in DST.
- */
- VM_DEFINE_OP (88, logxor, "logxor", OP1 (U8_U8_U8_U8) | OP_DST)
- {
- ARGS2 (x, y);
- if (SCM_I_INUMP (x) && SCM_I_INUMP (y))
- RETURN (SCM_I_MAKINUM (SCM_I_INUM (x) ^ SCM_I_INUM (y)));
- SYNC_IP ();
- RETURN (scm_logxor (x, y));
- }
- /* vector-length dst:12 src:12
- *
- * Store the length of the vector in SRC in DST.
- */
- VM_DEFINE_OP (89, vector_length, "vector-length", OP1 (U8_U12_U12) | OP_DST)
- {
- ARGS1 (vect);
- if (SCM_LIKELY (SCM_I_IS_VECTOR (vect)))
- RETURN (SCM_I_MAKINUM (SCM_I_VECTOR_LENGTH (vect)));
- else
- {
- SYNC_IP ();
- RETURN (scm_vector_length (vect));
- }
- }
- /* vector-ref dst:8 src:8 idx:8
- *
- * Fetch the item at position IDX in the vector in SRC, and store it
- * in DST.
- */
- VM_DEFINE_OP (90, vector_ref, "vector-ref", OP1 (U8_U8_U8_U8) | OP_DST)
- {
- scm_t_signed_bits i = 0;
- ARGS2 (vect, idx);
- if (SCM_LIKELY (SCM_I_IS_NONWEAK_VECTOR (vect)
- && SCM_I_INUMP (idx)
- && ((i = SCM_I_INUM (idx)) >= 0)
- && i < SCM_I_VECTOR_LENGTH (vect)))
- RETURN (SCM_I_VECTOR_ELTS (vect)[i]);
- else
- {
- SYNC_IP ();
- RETURN (scm_vector_ref (vect, idx));
- }
- }
- /* constant-vector-ref dst:8 src:8 idx:8
- *
- * Fill DST with the item IDX elements into the vector at SRC. Useful
- * for building data types using vectors.
- */
- VM_DEFINE_OP (91, constant_vector_ref, "constant-vector-ref", OP1 (U8_U8_U8_U8) | OP_DST)
- {
- scm_t_uint8 dst, src, idx;
- SCM v;
-
- SCM_UNPACK_RTL_8_8_8 (op, dst, src, idx);
- v = LOCAL_REF (src);
- if (SCM_LIKELY (SCM_I_IS_NONWEAK_VECTOR (v)
- && idx < SCM_I_VECTOR_LENGTH (v)))
- LOCAL_SET (dst, SCM_I_VECTOR_ELTS (LOCAL_REF (src))[idx]);
- else
- LOCAL_SET (dst, scm_c_vector_ref (v, idx));
- NEXT (1);
- }
- /* vector-set! dst:8 idx:8 src:8
- *
- * Store SRC into the vector DST at index IDX.
- */
- VM_DEFINE_OP (92, vector_set, "vector-set", OP1 (U8_U8_U8_U8))
- {
- scm_t_uint8 dst, idx_var, src;
- SCM vect, idx, val;
- scm_t_signed_bits i = 0;
- SCM_UNPACK_RTL_8_8_8 (op, dst, idx_var, src);
- vect = LOCAL_REF (dst);
- idx = LOCAL_REF (idx_var);
- val = LOCAL_REF (src);
- if (SCM_LIKELY (SCM_I_IS_NONWEAK_VECTOR (vect)
- && SCM_I_INUMP (idx)
- && ((i = SCM_I_INUM (idx)) >= 0)
- && i < SCM_I_VECTOR_LENGTH (vect)))
- SCM_I_VECTOR_WELTS (vect)[i] = val;
- else
- {
- SYNC_IP ();
- scm_vector_set_x (vect, idx, val);
- }
- NEXT (1);
- }
-
- /*
- * Structs and GOOPS
- */
- /* struct-vtable dst:12 src:12
- *
- * Store the vtable of SRC into DST.
- */
- VM_DEFINE_OP (93, struct_vtable, "struct-vtable", OP1 (U8_U12_U12) | OP_DST)
- {
- ARGS1 (obj);
- VM_VALIDATE_STRUCT (obj, "struct_vtable");
- RETURN (SCM_STRUCT_VTABLE (obj));
- }
- /* allocate-struct dst:8 vtable:8 nfields:8
- *
- * Allocate a new struct with VTABLE, and place it in DST. The struct
- * will be constructed with space for NFIELDS fields, which should
- * correspond to the field count of the VTABLE.
- */
- VM_DEFINE_OP (94, allocate_struct, "allocate-struct", OP1 (U8_U8_U8_U8) | OP_DST)
- {
- scm_t_uint8 dst, vtable, nfields;
- SCM ret;
- SCM_UNPACK_RTL_8_8_8 (op, dst, vtable, nfields);
- SYNC_IP ();
- ret = scm_allocate_struct (LOCAL_REF (vtable), SCM_I_MAKINUM (nfields));
- LOCAL_SET (dst, ret);
- NEXT (1);
- }
- /* struct-ref dst:8 src:8 idx:8
- *
- * Fetch the item at slot IDX in the struct in SRC, and store it
- * in DST.
- */
- VM_DEFINE_OP (95, struct_ref, "struct-ref", OP1 (U8_U8_U8_U8) | OP_DST)
- {
- ARGS2 (obj, pos);
- if (SCM_LIKELY (SCM_STRUCTP (obj)
- && SCM_STRUCT_VTABLE_FLAG_IS_SET (obj,
- SCM_VTABLE_FLAG_SIMPLE)
- && SCM_I_INUMP (pos)))
- {
- SCM vtable;
- scm_t_bits index, len;
- /* True, an inum is a signed value, but cast to unsigned it will
- certainly be more than the length, so we will fall through if
- index is negative. */
- index = SCM_I_INUM (pos);
- vtable = SCM_STRUCT_VTABLE (obj);
- len = SCM_STRUCT_DATA_REF (vtable, scm_vtable_index_size);
- if (SCM_LIKELY (index < len))
- {
- scm_t_bits *data = SCM_STRUCT_DATA (obj);
- RETURN (SCM_PACK (data[index]));
- }
- }
- SYNC_IP ();
- RETURN (scm_struct_ref (obj, pos));
- }
- /* struct-set! dst:8 idx:8 src:8
- *
- * Store SRC into the struct DST at slot IDX.
- */
- VM_DEFINE_OP (96, struct_set, "struct-set!", OP1 (U8_U8_U8_U8))
- {
- scm_t_uint8 dst, idx, src;
- SCM obj, pos, val;
-
- SCM_UNPACK_RTL_8_8_8 (op, dst, idx, src);
- obj = LOCAL_REF (dst);
- pos = LOCAL_REF (idx);
- val = LOCAL_REF (src);
-
- if (SCM_LIKELY (SCM_STRUCTP (obj)
- && SCM_STRUCT_VTABLE_FLAG_IS_SET (obj,
- SCM_VTABLE_FLAG_SIMPLE)
- && SCM_STRUCT_VTABLE_FLAG_IS_SET (obj,
- SCM_VTABLE_FLAG_SIMPLE_RW)
- && SCM_I_INUMP (pos)))
- {
- SCM vtable;
- scm_t_bits index, len;
- /* See above regarding index being >= 0. */
- index = SCM_I_INUM (pos);
- vtable = SCM_STRUCT_VTABLE (obj);
- len = SCM_STRUCT_DATA_REF (vtable, scm_vtable_index_size);
- if (SCM_LIKELY (index < len))
- {
- scm_t_bits *data = SCM_STRUCT_DATA (obj);
- data[index] = SCM_UNPACK (val);
- NEXT (1);
- }
- }
- SYNC_IP ();
- scm_struct_set_x (obj, pos, val);
- NEXT (1);
- }
- /* class-of dst:12 type:12
- *
- * Store the vtable of SRC into DST.
- */
- VM_DEFINE_OP (97, class_of, "class-of", OP1 (U8_U12_U12) | OP_DST)
- {
- ARGS1 (obj);
- if (SCM_INSTANCEP (obj))
- RETURN (SCM_CLASS_OF (obj));
- SYNC_IP ();
- RETURN (scm_class_of (obj));
- }
- /* slot-ref dst:8 src:8 idx:8
- *
- * Fetch the item at slot IDX in the struct in SRC, and store it in
- * DST. Unlike struct-ref, IDX is an 8-bit immediate value, not an
- * index into the stack.
- */
- VM_DEFINE_OP (98, slot_ref, "slot-ref", OP1 (U8_U8_U8_U8) | OP_DST)
- {
- scm_t_uint8 dst, src, idx;
- SCM_UNPACK_RTL_8_8_8 (op, dst, src, idx);
- LOCAL_SET (dst,
- SCM_PACK (SCM_STRUCT_DATA (LOCAL_REF (src))[idx]));
- NEXT (1);
- }
- /* slot-set! dst:8 idx:8 src:8
- *
- * Store SRC into slot IDX of the struct in DST. Unlike struct-set!,
- * IDX is an 8-bit immediate value, not an index into the stack.
- */
- VM_DEFINE_OP (99, slot_set, "slot-set!", OP1 (U8_U8_U8_U8))
- {
- scm_t_uint8 dst, idx, src;
- SCM_UNPACK_RTL_8_8_8 (op, dst, idx, src);
- SCM_STRUCT_DATA (LOCAL_REF (dst))[idx] = SCM_UNPACK (LOCAL_REF (src));
- NEXT (1);
- }
-
- /*
- * Arrays, packed uniform arrays, and bytevectors.
- */
- /* load-typed-array dst:8 type:8 shape:8 offset:32 len:32
- *
- * Load the contiguous typed array located at OFFSET 32-bit words away
- * from the instruction pointer, and store into DST. LEN is a byte
- * length. OFFSET is signed.
- */
- VM_DEFINE_OP (100, load_typed_array, "load-typed-array", OP3 (U8_U8_U8_U8, N32, U32) | OP_DST)
- {
- scm_t_uint8 dst, type, shape;
- scm_t_int32 offset;
- scm_t_uint32 len;
- SCM_UNPACK_RTL_8_8_8 (op, dst, type, shape);
- offset = ip[1];
- len = ip[2];
- SYNC_IP ();
- LOCAL_SET (dst, scm_from_contiguous_typed_array (LOCAL_REF (type),
- LOCAL_REF (shape),
- ip + offset, len));
- NEXT (3);
- }
- /* make-array dst:12 type:12 _:8 fill:12 bounds:12
- *
- * Make a new array with TYPE, FILL, and BOUNDS, storing it in DST.
- */
- VM_DEFINE_OP (101, make_array, "make-array", OP2 (U8_U12_U12, X8_U12_U12) | OP_DST)
- {
- scm_t_uint16 dst, type, fill, bounds;
- SCM_UNPACK_RTL_12_12 (op, dst, type);
- SCM_UNPACK_RTL_12_12 (ip[1], fill, bounds);
- SYNC_IP ();
- LOCAL_SET (dst, scm_make_typed_array (LOCAL_REF (type), LOCAL_REF (fill),
- LOCAL_REF (bounds)));
- NEXT (2);
- }
- /* bv-u8-ref dst:8 src:8 idx:8
- * bv-s8-ref dst:8 src:8 idx:8
- * bv-u16-ref dst:8 src:8 idx:8
- * bv-s16-ref dst:8 src:8 idx:8
- * bv-u32-ref dst:8 src:8 idx:8
- * bv-s32-ref dst:8 src:8 idx:8
- * bv-u64-ref dst:8 src:8 idx:8
- * bv-s64-ref dst:8 src:8 idx:8
- * bv-f32-ref dst:8 src:8 idx:8
- * bv-f64-ref dst:8 src:8 idx:8
- *
- * Fetch the item at byte offset IDX in the bytevector SRC, and store
- * it in DST. All accesses use native endianness.
- */
- #define BV_FIXABLE_INT_REF(stem, fn_stem, type, size) \
- do { \
- scm_t_signed_bits i; \
- const scm_t_ ## type *int_ptr; \
- ARGS2 (bv, idx); \
- \
- VM_VALIDATE_BYTEVECTOR (bv, "bv-" #stem "-ref"); \
- i = SCM_I_INUM (idx); \
- int_ptr = (scm_t_ ## type *) (SCM_BYTEVECTOR_CONTENTS (bv) + i); \
- \
- if (SCM_LIKELY (SCM_I_INUMP (idx) \
- && (i >= 0) \
- && (i + size <= SCM_BYTEVECTOR_LENGTH (bv)) \
- && (ALIGNED_P (int_ptr, scm_t_ ## type)))) \
- RETURN (SCM_I_MAKINUM (*int_ptr)); \
- else \
- { \
- SYNC_IP (); \
- RETURN (scm_bytevector_ ## fn_stem ## _ref (bv, idx)); \
- } \
- } while (0)
- #define BV_INT_REF(stem, type, size) \
- do { \
- scm_t_signed_bits i; \
- const scm_t_ ## type *int_ptr; \
- ARGS2 (bv, idx); \
- \
- VM_VALIDATE_BYTEVECTOR (bv, "bv-" #stem "-ref"); \
- i = SCM_I_INUM (idx); \
- int_ptr = (scm_t_ ## type *) (SCM_BYTEVECTOR_CONTENTS (bv) + i); \
- \
- if (SCM_LIKELY (SCM_I_INUMP (idx) \
- && (i >= 0) \
- && (i + size <= SCM_BYTEVECTOR_LENGTH (bv)) \
- && (ALIGNED_P (int_ptr, scm_t_ ## type)))) \
- { \
- scm_t_ ## type x = *int_ptr; \
- if (SCM_FIXABLE (x)) \
- RETURN (SCM_I_MAKINUM (x)); \
- else \
- { \
- SYNC_IP (); \
- RETURN (scm_from_ ## type (x)); \
- } \
- } \
- else \
- { \
- SYNC_IP (); \
- RETURN (scm_bytevector_ ## stem ## _native_ref (bv, idx)); \
- } \
- } while (0)
- #define BV_FLOAT_REF(stem, fn_stem, type, size) \
- do { \
- scm_t_signed_bits i; \
- const type *float_ptr; \
- ARGS2 (bv, idx); \
- \
- VM_VALIDATE_BYTEVECTOR (bv, "bv-" #stem "-ref"); \
- i = SCM_I_INUM (idx); \
- float_ptr = (type *) (SCM_BYTEVECTOR_CONTENTS (bv) + i); \
- \
- SYNC_IP (); \
- if (SCM_LIKELY (SCM_I_INUMP (idx) \
- && (i >= 0) \
- && (i + size <= SCM_BYTEVECTOR_LENGTH (bv)) \
- && (ALIGNED_P (float_ptr, type)))) \
- RETURN (scm_from_double (*float_ptr)); \
- else \
- RETURN (scm_bytevector_ ## fn_stem ## _native_ref (bv, idx)); \
- } while (0)
- VM_DEFINE_OP (102, bv_u8_ref, "bv-u8-ref", OP1 (U8_U8_U8_U8) | OP_DST)
- BV_FIXABLE_INT_REF (u8, u8, uint8, 1);
- VM_DEFINE_OP (103, bv_s8_ref, "bv-s8-ref", OP1 (U8_U8_U8_U8) | OP_DST)
- BV_FIXABLE_INT_REF (s8, s8, int8, 1);
- VM_DEFINE_OP (104, bv_u16_ref, "bv-u16-ref", OP1 (U8_U8_U8_U8) | OP_DST)
- BV_FIXABLE_INT_REF (u16, u16_native, uint16, 2);
- VM_DEFINE_OP (105, bv_s16_ref, "bv-s16-ref", OP1 (U8_U8_U8_U8) | OP_DST)
- BV_FIXABLE_INT_REF (s16, s16_native, int16, 2);
- VM_DEFINE_OP (106, bv_u32_ref, "bv-u32-ref", OP1 (U8_U8_U8_U8) | OP_DST)
- #if SIZEOF_VOID_P > 4
- BV_FIXABLE_INT_REF (u32, u32_native, uint32, 4);
- #else
- BV_INT_REF (u32, uint32, 4);
- #endif
- VM_DEFINE_OP (107, bv_s32_ref, "bv-s32-ref", OP1 (U8_U8_U8_U8) | OP_DST)
- #if SIZEOF_VOID_P > 4
- BV_FIXABLE_INT_REF (s32, s32_native, int32, 4);
- #else
- BV_INT_REF (s32, int32, 4);
- #endif
- VM_DEFINE_OP (108, bv_u64_ref, "bv-u64-ref", OP1 (U8_U8_U8_U8) | OP_DST)
- BV_INT_REF (u64, uint64, 8);
- VM_DEFINE_OP (109, bv_s64_ref, "bv-s64-ref", OP1 (U8_U8_U8_U8) | OP_DST)
- BV_INT_REF (s64, int64, 8);
- VM_DEFINE_OP (110, bv_f32_ref, "bv-f32-ref", OP1 (U8_U8_U8_U8) | OP_DST)
- BV_FLOAT_REF (f32, ieee_single, float, 4);
- VM_DEFINE_OP (111, bv_f64_ref, "bv-f64-ref", OP1 (U8_U8_U8_U8) | OP_DST)
- BV_FLOAT_REF (f64, ieee_double, double, 8);
- /* bv-u8-set! dst:8 idx:8 src:8
- * bv-s8-set! dst:8 idx:8 src:8
- * bv-u16-set! dst:8 idx:8 src:8
- * bv-s16-set! dst:8 idx:8 src:8
- * bv-u32-set! dst:8 idx:8 src:8
- * bv-s32-set! dst:8 idx:8 src:8
- * bv-u64-set! dst:8 idx:8 src:8
- * bv-s64-set! dst:8 idx:8 src:8
- * bv-f32-set! dst:8 idx:8 src:8
- * bv-f64-set! dst:8 idx:8 src:8
- *
- * Store SRC into the bytevector DST at byte offset IDX. Multibyte
- * values are written using native endianness.
- */
- #define BV_FIXABLE_INT_SET(stem, fn_stem, type, min, max, size) \
- do { \
- scm_t_uint8 dst, idx, src; \
- scm_t_signed_bits i, j = 0; \
- SCM bv, scm_idx, val; \
- scm_t_ ## type *int_ptr; \
- \
- SCM_UNPACK_RTL_8_8_8 (op, dst, idx, src); \
- bv = LOCAL_REF (dst); \
- scm_idx = LOCAL_REF (idx); \
- val = LOCAL_REF (src); \
- VM_VALIDATE_BYTEVECTOR (bv, "bv-" #stem "-set"); \
- i = SCM_I_INUM (scm_idx); \
- int_ptr = (scm_t_ ## type *) (SCM_BYTEVECTOR_CONTENTS (bv) + i); \
- \
- if (SCM_LIKELY (SCM_I_INUMP (scm_idx) \
- && (i >= 0) \
- && (i + size <= SCM_BYTEVECTOR_LENGTH (bv)) \
- && (ALIGNED_P (int_ptr, scm_t_ ## type)) \
- && (SCM_I_INUMP (val)) \
- && ((j = SCM_I_INUM (val)) >= min) \
- && (j <= max))) \
- *int_ptr = (scm_t_ ## type) j; \
- else \
- { \
- SYNC_IP (); \
- scm_bytevector_ ## fn_stem ## _set_x (bv, scm_idx, val); \
- } \
- NEXT (1); \
- } while (0)
- #define BV_INT_SET(stem, type, size) \
- do { \
- scm_t_uint8 dst, idx, src; \
- scm_t_signed_bits i; \
- SCM bv, scm_idx, val; \
- scm_t_ ## type *int_ptr; \
- \
- SCM_UNPACK_RTL_8_8_8 (op, dst, idx, src); \
- bv = LOCAL_REF (dst); \
- scm_idx = LOCAL_REF (idx); \
- val = LOCAL_REF (src); \
- VM_VALIDATE_BYTEVECTOR (bv, "bv-" #stem "-set"); \
- i = SCM_I_INUM (scm_idx); \
- int_ptr = (scm_t_ ## type *) (SCM_BYTEVECTOR_CONTENTS (bv) + i); \
- \
- if (SCM_LIKELY (SCM_I_INUMP (scm_idx) \
- && (i >= 0) \
- && (i + size <= SCM_BYTEVECTOR_LENGTH (bv)) \
- && (ALIGNED_P (int_ptr, scm_t_ ## type)))) \
- *int_ptr = scm_to_ ## type (val); \
- else \
- { \
- SYNC_IP (); \
- scm_bytevector_ ## stem ## _native_set_x (bv, scm_idx, val); \
- } \
- NEXT (1); \
- } while (0)
- #define BV_FLOAT_SET(stem, fn_stem, type, size) \
- do { \
- scm_t_uint8 dst, idx, src; \
- scm_t_signed_bits i; \
- SCM bv, scm_idx, val; \
- type *float_ptr; \
- \
- SCM_UNPACK_RTL_8_8_8 (op, dst, idx, src); \
- bv = LOCAL_REF (dst); \
- scm_idx = LOCAL_REF (idx); \
- val = LOCAL_REF (src); \
- VM_VALIDATE_BYTEVECTOR (bv, "bv-" #stem "-set"); \
- i = SCM_I_INUM (scm_idx); \
- float_ptr = (type *) (SCM_BYTEVECTOR_CONTENTS (bv) + i); \
- \
- if (SCM_LIKELY (SCM_I_INUMP (scm_idx) \
- && (i >= 0) \
- && (i + size <= SCM_BYTEVECTOR_LENGTH (bv)) \
- && (ALIGNED_P (float_ptr, type)))) \
- *float_ptr = scm_to_double (val); \
- else \
- { \
- SYNC_IP (); \
- scm_bytevector_ ## fn_stem ## _native_set_x (bv, scm_idx, val); \
- } \
- NEXT (1); \
- } while (0)
- VM_DEFINE_OP (112, bv_u8_set, "bv-u8-set!", OP1 (U8_U8_U8_U8))
- BV_FIXABLE_INT_SET (u8, u8, uint8, 0, SCM_T_UINT8_MAX, 1);
- VM_DEFINE_OP (113, bv_s8_set, "bv-s8-set!", OP1 (U8_U8_U8_U8))
- BV_FIXABLE_INT_SET (s8, s8, int8, SCM_T_INT8_MIN, SCM_T_INT8_MAX, 1);
- VM_DEFINE_OP (114, bv_u16_set, "bv-u16-set!", OP1 (U8_U8_U8_U8))
- BV_FIXABLE_INT_SET (u16, u16_native, uint16, 0, SCM_T_UINT16_MAX, 2);
- VM_DEFINE_OP (115, bv_s16_set, "bv-s16-set!", OP1 (U8_U8_U8_U8))
- BV_FIXABLE_INT_SET (s16, s16_native, int16, SCM_T_INT16_MIN, SCM_T_INT16_MAX, 2);
- VM_DEFINE_OP (116, bv_u32_set, "bv-u32-set!", OP1 (U8_U8_U8_U8))
- #if SIZEOF_VOID_P > 4
- BV_FIXABLE_INT_SET (u32, u32_native, uint32, 0, SCM_T_UINT32_MAX, 4);
- #else
- BV_INT_SET (u32, uint32, 4);
- #endif
- VM_DEFINE_OP (117, bv_s32_set, "bv-s32-set!", OP1 (U8_U8_U8_U8))
- #if SIZEOF_VOID_P > 4
- BV_FIXABLE_INT_SET (s32, s32_native, int32, SCM_T_INT32_MIN, SCM_T_INT32_MAX, 4);
- #else
- BV_INT_SET (s32, int32, 4);
- #endif
- VM_DEFINE_OP (118, bv_u64_set, "bv-u64-set!", OP1 (U8_U8_U8_U8))
- BV_INT_SET (u64, uint64, 8);
- VM_DEFINE_OP (119, bv_s64_set, "bv-s64-set!", OP1 (U8_U8_U8_U8))
- BV_INT_SET (s64, int64, 8);
- VM_DEFINE_OP (120, bv_f32_set, "bv-f32-set!", OP1 (U8_U8_U8_U8))
- BV_FLOAT_SET (f32, ieee_single, float, 4);
- VM_DEFINE_OP (121, bv_f64_set, "bv-f64-set!", OP1 (U8_U8_U8_U8))
- BV_FLOAT_SET (f64, ieee_double, double, 8);
- END_DISPATCH_SWITCH;
- vm_error_bad_instruction:
- vm_error_bad_instruction (op);
- abort (); /* never reached */
- }
- #undef ABORT_CONTINUATION_HOOK
- #undef ALIGNED_P
- #undef APPLY_HOOK
- #undef ARGS1
- #undef ARGS2
- #undef BEGIN_DISPATCH_SWITCH
- #undef BINARY_INTEGER_OP
- #undef BR_ARITHMETIC
- #undef BR_BINARY
- #undef BR_NARGS
- #undef BR_UNARY
- #undef BV_FIXABLE_INT_REF
- #undef BV_FIXABLE_INT_SET
- #undef BV_FLOAT_REF
- #undef BV_FLOAT_SET
- #undef BV_INT_REF
- #undef BV_INT_SET
- #undef CACHE_REGISTER
- #undef CHECK_OVERFLOW
- #undef END_DISPATCH_SWITCH
- #undef FREE_VARIABLE_REF
- #undef INIT
- #undef INUM_MAX
- #undef INUM_MIN
- #undef LOCAL_REF
- #undef LOCAL_SET
- #undef NEXT
- #undef NEXT_HOOK
- #undef NEXT_JUMP
- #undef POP_CONTINUATION_HOOK
- #undef PUSH_CONTINUATION_HOOK
- #undef RESTORE_CONTINUATION_HOOK
- #undef RETURN
- #undef RETURN_ONE_VALUE
- #undef RETURN_VALUE_LIST
- #undef RUN_HOOK
- #undef RUN_HOOK0
- #undef SYNC_ALL
- #undef SYNC_BEFORE_GC
- #undef SYNC_IP
- #undef SYNC_REGISTER
- #undef VARIABLE_BOUNDP
- #undef VARIABLE_REF
- #undef VARIABLE_SET
- #undef VM_CHECK_FREE_VARIABLE
- #undef VM_CHECK_OBJECT
- #undef VM_CHECK_UNDERFLOW
- #undef VM_DEFINE_OP
- #undef VM_INSTRUCTION_TO_LABEL
- #undef VM_USE_HOOKS
- #undef VM_VALIDATE_BYTEVECTOR
- #undef VM_VALIDATE_PAIR
- #undef VM_VALIDATE_STRUCT
- /*
- (defun renumber-ops ()
- "start from top of buffer and renumber 'VM_DEFINE_FOO (\n' sequences"
- (interactive "")
- (save-excursion
- (let ((counter -1)) (goto-char (point-min))
- (while (re-search-forward "^ *VM_DEFINE_[^ ]+ (\\([^,]+\\)," (point-max) t)
- (replace-match
- (number-to-string (setq counter (1+ counter)))
- t t nil 1)))))
- (renumber-ops)
- */
- /*
- Local Variables:
- c-file-style: "gnu"
- End:
- */
|