ops.c 141 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902490349044905490649074908490949104911491249134914491549164917491849194920492149224923492449254926492749284929493049314932493349344935493649374938493949404941494249434944494549464947494849494950495149524953495449554956495749584959496049614962496349644965496649674968496949704971497249734974497549764977497849794980498149824983498449854986498749884989499049914992499349944995499649974998499950005001500250035004500550065007500850095010501150125013501450155016501750185019502050215022502350245025502650275028502950305031503250335034503550365037503850395040504150425043504450455046504750485049505050515052505350545055505650575058505950605061506250635064506550665067506850695070507150725073507450755076507750785079508050815082508350845085508650875088508950905091509250935094509550965097509850995100510151025103510451055106510751085109511051115112511351145115511651175118511951205121512251235124512551265127512851295130513151325133513451355136513751385139514051415142514351445145514651475148514951505151515251535154515551565157515851595160516151625163516451655166516751685169517051715172517351745175517651775178517951805181518251835184518551865187518851895190519151925193519451955196519751985199520052015202520352045205520652075208520952105211521252135214521552165217521852195220522152225223522452255226522752285229523052315232523352345235523652375238523952405241524252435244524552465247524852495250525152525253525452555256525752585259526052615262526352645265526652675268526952705271527252735274527552765277527852795280528152825283528452855286528752885289529052915292529352945295529652975298529953005301530253035304530553065307530853095310531153125313531453155316531753185319532053215322532353245325532653275328532953305331533253335334533553365337533853395340534153425343534453455346534753485349535053515352535353545355535653575358535953605361536253635364536553665367536853695370537153725373537453755376537753785379538053815382538353845385538653875388538953905391539253935394539553965397539853995400540154025403540454055406540754085409541054115412541354145415541654175418541954205421542254235424542554265427542854295430543154325433543454355436
  1. /****************************************************************************
  2. * Realmode X86 Emulator Library
  3. *
  4. * Copyright (C) 2007 Freescale Semiconductor, Inc. All rights reserved.
  5. * Jason Jin <Jason.jin@freescale.com>
  6. *
  7. * Copyright (C) 1991-2004 SciTech Software, Inc.
  8. * Copyright (C) David Mosberger-Tang
  9. * Copyright (C) 1999 Egbert Eich
  10. *
  11. * ========================================================================
  12. *
  13. * Permission to use, copy, modify, distribute, and sell this software and
  14. * its documentation for any purpose is hereby granted without fee,
  15. * provided that the above copyright notice appear in all copies and that
  16. * both that copyright notice and this permission notice appear in
  17. * supporting documentation, and that the name of the authors not be used
  18. * in advertising or publicity pertaining to distribution of the software
  19. * without specific, written prior permission. The authors makes no
  20. * representations about the suitability of this software for any purpose.
  21. * It is provided "as is" without express or implied warranty.
  22. *
  23. * THE AUTHORS DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
  24. * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
  25. * EVENT SHALL THE AUTHORS BE LIABLE FOR ANY SPECIAL, INDIRECT OR
  26. * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF
  27. * USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
  28. * OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
  29. * PERFORMANCE OF THIS SOFTWARE.
  30. *
  31. * ========================================================================
  32. *
  33. * Language: ANSI C
  34. * Environment: Any
  35. * Developer: Kendall Bennett
  36. *
  37. * Description: This file includes subroutines to implement the decoding
  38. * and emulation of all the x86 processor instructions.
  39. *
  40. * There are approximately 250 subroutines in here, which correspond
  41. * to the 256 byte-"opcodes" found on the 8086. The table which
  42. * dispatches this is found in the files optab.[ch].
  43. *
  44. * Each opcode proc has a comment preceeding it which gives it's table
  45. * address. Several opcodes are missing (undefined) in the table.
  46. *
  47. * Each proc includes information for decoding (DECODE_PRINTF and
  48. * DECODE_PRINTF2), debugging (TRACE_REGS, SINGLE_STEP), and misc
  49. * functions (START_OF_INSTR, END_OF_INSTR).
  50. *
  51. * Many of the procedures are *VERY* similar in coding. This has
  52. * allowed for a very large amount of code to be generated in a fairly
  53. * short amount of time (i.e. cut, paste, and modify). The result is
  54. * that much of the code below could have been folded into subroutines
  55. * for a large reduction in size of this file. The downside would be
  56. * that there would be a penalty in execution speed. The file could
  57. * also have been *MUCH* larger by inlining certain functions which
  58. * were called. This could have resulted even faster execution. The
  59. * prime directive I used to decide whether to inline the code or to
  60. * modularize it, was basically: 1) no unnecessary subroutine calls,
  61. * 2) no routines more than about 200 lines in size, and 3) modularize
  62. * any code that I might not get right the first time. The fetch_*
  63. * subroutines fall into the latter category. The The decode_* fall
  64. * into the second category. The coding of the "switch(mod){ .... }"
  65. * in many of the subroutines below falls into the first category.
  66. * Especially, the coding of {add,and,or,sub,...}_{byte,word}
  67. * subroutines are an especially glaring case of the third guideline.
  68. * Since so much of the code is cloned from other modules (compare
  69. * opcode #00 to opcode #01), making the basic operations subroutine
  70. * calls is especially important; otherwise mistakes in coding an
  71. * "add" would represent a nightmare in maintenance.
  72. *
  73. * Jason ported this file to u-boot. place all the function pointer in
  74. * the got2 sector. Removed some opcode.
  75. *
  76. ****************************************************************************/
  77. #include "x86emu/x86emui.h"
  78. #if defined(CONFIG_BIOSEMU)
  79. /*----------------------------- Implementation ----------------------------*/
  80. /* constant arrays to do several instructions in just one function */
  81. #ifdef DEBUG
  82. static char *x86emu_GenOpName[8] = {
  83. "ADD", "OR", "ADC", "SBB", "AND", "SUB", "XOR", "CMP"};
  84. #endif
  85. /* used by several opcodes */
  86. static u8 (*genop_byte_operation[])(u8 d, u8 s) __attribute__ ((section(".got2"))) =
  87. {
  88. add_byte, /* 00 */
  89. or_byte, /* 01 */
  90. adc_byte, /* 02 */
  91. sbb_byte, /* 03 */
  92. and_byte, /* 04 */
  93. sub_byte, /* 05 */
  94. xor_byte, /* 06 */
  95. cmp_byte, /* 07 */
  96. };
  97. static u16 (*genop_word_operation[])(u16 d, u16 s) __attribute__ ((section(".got2"))) =
  98. {
  99. add_word, /*00 */
  100. or_word, /*01 */
  101. adc_word, /*02 */
  102. sbb_word, /*03 */
  103. and_word, /*04 */
  104. sub_word, /*05 */
  105. xor_word, /*06 */
  106. cmp_word, /*07 */
  107. };
  108. static u32 (*genop_long_operation[])(u32 d, u32 s) __attribute__ ((section(".got2"))) =
  109. {
  110. add_long, /*00 */
  111. or_long, /*01 */
  112. adc_long, /*02 */
  113. sbb_long, /*03 */
  114. and_long, /*04 */
  115. sub_long, /*05 */
  116. xor_long, /*06 */
  117. cmp_long, /*07 */
  118. };
  119. /* used by opcodes 80, c0, d0, and d2. */
  120. static u8(*opcD0_byte_operation[])(u8 d, u8 s) __attribute__ ((section(".got2"))) =
  121. {
  122. rol_byte,
  123. ror_byte,
  124. rcl_byte,
  125. rcr_byte,
  126. shl_byte,
  127. shr_byte,
  128. shl_byte, /* sal_byte === shl_byte by definition */
  129. sar_byte,
  130. };
  131. /* used by opcodes c1, d1, and d3. */
  132. static u16(*opcD1_word_operation[])(u16 s, u8 d) __attribute__ ((section(".got2"))) =
  133. {
  134. rol_word,
  135. ror_word,
  136. rcl_word,
  137. rcr_word,
  138. shl_word,
  139. shr_word,
  140. shl_word, /* sal_byte === shl_byte by definition */
  141. sar_word,
  142. };
  143. /* used by opcodes c1, d1, and d3. */
  144. static u32 (*opcD1_long_operation[])(u32 s, u8 d) __attribute__ ((section(".got2"))) =
  145. {
  146. rol_long,
  147. ror_long,
  148. rcl_long,
  149. rcr_long,
  150. shl_long,
  151. shr_long,
  152. shl_long, /* sal_byte === shl_byte by definition */
  153. sar_long,
  154. };
  155. #ifdef DEBUG
  156. static char *opF6_names[8] =
  157. { "TEST\t", "", "NOT\t", "NEG\t", "MUL\t", "IMUL\t", "DIV\t", "IDIV\t" };
  158. #endif
  159. /****************************************************************************
  160. PARAMETERS:
  161. op1 - Instruction op code
  162. REMARKS:
  163. Handles illegal opcodes.
  164. ****************************************************************************/
  165. void x86emuOp_illegal_op(
  166. u8 op1)
  167. {
  168. START_OF_INSTR();
  169. if (M.x86.R_SP != 0) {
  170. DECODE_PRINTF("ILLEGAL X86 OPCODE\n");
  171. TRACE_REGS();
  172. DB( printk("%04x:%04x: %02X ILLEGAL X86 OPCODE!\n",
  173. M.x86.R_CS, M.x86.R_IP-1,op1));
  174. HALT_SYS();
  175. }
  176. else {
  177. /* If we get here, it means the stack pointer is back to zero
  178. * so we are just returning from an emulator service call
  179. * so therte is no need to display an error message. We trap
  180. * the emulator with an 0xF1 opcode to finish the service
  181. * call.
  182. */
  183. X86EMU_halt_sys();
  184. }
  185. END_OF_INSTR();
  186. }
  187. /****************************************************************************
  188. REMARKS:
  189. Handles opcodes 0x00, 0x08, 0x10, 0x18, 0x20, 0x28, 0x30, 0x38
  190. ****************************************************************************/
  191. void x86emuOp_genop_byte_RM_R(u8 op1)
  192. {
  193. int mod, rl, rh;
  194. uint destoffset;
  195. u8 *destreg, *srcreg;
  196. u8 destval;
  197. op1 = (op1 >> 3) & 0x7;
  198. START_OF_INSTR();
  199. DECODE_PRINTF(x86emu_GenOpName[op1]);
  200. DECODE_PRINTF("\t");
  201. FETCH_DECODE_MODRM(mod, rh, rl);
  202. if(mod<3)
  203. { destoffset = decode_rmXX_address(mod,rl);
  204. DECODE_PRINTF(",");
  205. destval = fetch_data_byte(destoffset);
  206. srcreg = DECODE_RM_BYTE_REGISTER(rh);
  207. DECODE_PRINTF("\n");
  208. TRACE_AND_STEP();
  209. destval = genop_byte_operation[op1](destval, *srcreg);
  210. store_data_byte(destoffset, destval);
  211. }
  212. else
  213. { /* register to register */
  214. destreg = DECODE_RM_BYTE_REGISTER(rl);
  215. DECODE_PRINTF(",");
  216. srcreg = DECODE_RM_BYTE_REGISTER(rh);
  217. DECODE_PRINTF("\n");
  218. TRACE_AND_STEP();
  219. *destreg = genop_byte_operation[op1](*destreg, *srcreg);
  220. }
  221. DECODE_CLEAR_SEGOVR();
  222. END_OF_INSTR();
  223. }
  224. /****************************************************************************
  225. REMARKS:
  226. Handles opcodes 0x01, 0x09, 0x11, 0x19, 0x21, 0x29, 0x31, 0x39
  227. ****************************************************************************/
  228. void x86emuOp_genop_word_RM_R(u8 op1)
  229. {
  230. int mod, rl, rh;
  231. uint destoffset;
  232. op1 = (op1 >> 3) & 0x7;
  233. START_OF_INSTR();
  234. DECODE_PRINTF(x86emu_GenOpName[op1]);
  235. DECODE_PRINTF("\t");
  236. FETCH_DECODE_MODRM(mod, rh, rl);
  237. if(mod<3) {
  238. destoffset = decode_rmXX_address(mod,rl);
  239. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  240. u32 destval;
  241. u32 *srcreg;
  242. DECODE_PRINTF(",");
  243. destval = fetch_data_long(destoffset);
  244. srcreg = DECODE_RM_LONG_REGISTER(rh);
  245. DECODE_PRINTF("\n");
  246. TRACE_AND_STEP();
  247. destval = genop_long_operation[op1](destval, *srcreg);
  248. store_data_long(destoffset, destval);
  249. } else {
  250. u16 destval;
  251. u16 *srcreg;
  252. DECODE_PRINTF(",");
  253. destval = fetch_data_word(destoffset);
  254. srcreg = DECODE_RM_WORD_REGISTER(rh);
  255. DECODE_PRINTF("\n");
  256. TRACE_AND_STEP();
  257. destval = genop_word_operation[op1](destval, *srcreg);
  258. store_data_word(destoffset, destval);
  259. }
  260. } else { /* register to register */
  261. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  262. u32 *destreg,*srcreg;
  263. destreg = DECODE_RM_LONG_REGISTER(rl);
  264. DECODE_PRINTF(",");
  265. srcreg = DECODE_RM_LONG_REGISTER(rh);
  266. DECODE_PRINTF("\n");
  267. TRACE_AND_STEP();
  268. *destreg = genop_long_operation[op1](*destreg, *srcreg);
  269. } else {
  270. u16 *destreg,*srcreg;
  271. destreg = DECODE_RM_WORD_REGISTER(rl);
  272. DECODE_PRINTF(",");
  273. srcreg = DECODE_RM_WORD_REGISTER(rh);
  274. DECODE_PRINTF("\n");
  275. TRACE_AND_STEP();
  276. *destreg = genop_word_operation[op1](*destreg, *srcreg);
  277. }
  278. }
  279. DECODE_CLEAR_SEGOVR();
  280. END_OF_INSTR();
  281. }
  282. /****************************************************************************
  283. REMARKS:
  284. Handles opcodes 0x02, 0x0a, 0x12, 0x1a, 0x22, 0x2a, 0x32, 0x3a
  285. ****************************************************************************/
  286. void x86emuOp_genop_byte_R_RM(u8 op1)
  287. {
  288. int mod, rl, rh;
  289. u8 *destreg, *srcreg;
  290. uint srcoffset;
  291. u8 srcval;
  292. op1 = (op1 >> 3) & 0x7;
  293. START_OF_INSTR();
  294. DECODE_PRINTF(x86emu_GenOpName[op1]);
  295. DECODE_PRINTF("\t");
  296. FETCH_DECODE_MODRM(mod, rh, rl);
  297. if (mod < 3) {
  298. destreg = DECODE_RM_BYTE_REGISTER(rh);
  299. DECODE_PRINTF(",");
  300. srcoffset = decode_rmXX_address(mod,rl);
  301. srcval = fetch_data_byte(srcoffset);
  302. } else { /* register to register */
  303. destreg = DECODE_RM_BYTE_REGISTER(rh);
  304. DECODE_PRINTF(",");
  305. srcreg = DECODE_RM_BYTE_REGISTER(rl);
  306. srcval = *srcreg;
  307. }
  308. DECODE_PRINTF("\n");
  309. TRACE_AND_STEP();
  310. *destreg = genop_byte_operation[op1](*destreg, srcval);
  311. DECODE_CLEAR_SEGOVR();
  312. END_OF_INSTR();
  313. }
  314. /****************************************************************************
  315. REMARKS:
  316. Handles opcodes 0x03, 0x0b, 0x13, 0x1b, 0x23, 0x2b, 0x33, 0x3b
  317. ****************************************************************************/
  318. void x86emuOp_genop_word_R_RM(u8 op1)
  319. {
  320. int mod, rl, rh;
  321. uint srcoffset;
  322. u32 *destreg32, srcval;
  323. u16 *destreg;
  324. op1 = (op1 >> 3) & 0x7;
  325. START_OF_INSTR();
  326. DECODE_PRINTF(x86emu_GenOpName[op1]);
  327. DECODE_PRINTF("\t");
  328. FETCH_DECODE_MODRM(mod, rh, rl);
  329. if (mod < 3) {
  330. srcoffset = decode_rmXX_address(mod,rl);
  331. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  332. destreg32 = DECODE_RM_LONG_REGISTER(rh);
  333. DECODE_PRINTF(",");
  334. srcval = fetch_data_long(srcoffset);
  335. DECODE_PRINTF("\n");
  336. TRACE_AND_STEP();
  337. *destreg32 = genop_long_operation[op1](*destreg32, srcval);
  338. } else {
  339. destreg = DECODE_RM_WORD_REGISTER(rh);
  340. DECODE_PRINTF(",");
  341. srcval = fetch_data_word(srcoffset);
  342. DECODE_PRINTF("\n");
  343. TRACE_AND_STEP();
  344. *destreg = genop_word_operation[op1](*destreg, srcval);
  345. }
  346. } else { /* register to register */
  347. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  348. u32 *srcreg;
  349. destreg32 = DECODE_RM_LONG_REGISTER(rh);
  350. DECODE_PRINTF(",");
  351. srcreg = DECODE_RM_LONG_REGISTER(rl);
  352. DECODE_PRINTF("\n");
  353. TRACE_AND_STEP();
  354. *destreg32 = genop_long_operation[op1](*destreg32, *srcreg);
  355. } else {
  356. u16 *srcreg;
  357. destreg = DECODE_RM_WORD_REGISTER(rh);
  358. DECODE_PRINTF(",");
  359. srcreg = DECODE_RM_WORD_REGISTER(rl);
  360. DECODE_PRINTF("\n");
  361. TRACE_AND_STEP();
  362. *destreg = genop_word_operation[op1](*destreg, *srcreg);
  363. }
  364. }
  365. DECODE_CLEAR_SEGOVR();
  366. END_OF_INSTR();
  367. }
  368. /****************************************************************************
  369. REMARKS:
  370. Handles opcodes 0x04, 0x0c, 0x14, 0x1c, 0x24, 0x2c, 0x34, 0x3c
  371. ****************************************************************************/
  372. void x86emuOp_genop_byte_AL_IMM(u8 op1)
  373. {
  374. u8 srcval;
  375. op1 = (op1 >> 3) & 0x7;
  376. START_OF_INSTR();
  377. DECODE_PRINTF(x86emu_GenOpName[op1]);
  378. DECODE_PRINTF("\tAL,");
  379. srcval = fetch_byte_imm();
  380. DECODE_PRINTF2("%x\n", srcval);
  381. TRACE_AND_STEP();
  382. M.x86.R_AL = genop_byte_operation[op1](M.x86.R_AL, srcval);
  383. DECODE_CLEAR_SEGOVR();
  384. END_OF_INSTR();
  385. }
  386. /****************************************************************************
  387. REMARKS:
  388. Handles opcodes 0x05, 0x0d, 0x15, 0x1d, 0x25, 0x2d, 0x35, 0x3d
  389. ****************************************************************************/
  390. void x86emuOp_genop_word_AX_IMM(u8 op1)
  391. {
  392. u32 srcval;
  393. op1 = (op1 >> 3) & 0x7;
  394. START_OF_INSTR();
  395. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  396. DECODE_PRINTF(x86emu_GenOpName[op1]);
  397. DECODE_PRINTF("\tEAX,");
  398. srcval = fetch_long_imm();
  399. } else {
  400. DECODE_PRINTF(x86emu_GenOpName[op1]);
  401. DECODE_PRINTF("\tAX,");
  402. srcval = fetch_word_imm();
  403. }
  404. DECODE_PRINTF2("%x\n", srcval);
  405. TRACE_AND_STEP();
  406. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  407. M.x86.R_EAX = genop_long_operation[op1](M.x86.R_EAX, srcval);
  408. } else {
  409. M.x86.R_AX = genop_word_operation[op1](M.x86.R_AX, (u16)srcval);
  410. }
  411. DECODE_CLEAR_SEGOVR();
  412. END_OF_INSTR();
  413. }
  414. /****************************************************************************
  415. REMARKS:
  416. Handles opcode 0x06
  417. ****************************************************************************/
  418. void x86emuOp_push_ES(u8 X86EMU_UNUSED(op1))
  419. {
  420. START_OF_INSTR();
  421. DECODE_PRINTF("PUSH\tES\n");
  422. TRACE_AND_STEP();
  423. push_word(M.x86.R_ES);
  424. DECODE_CLEAR_SEGOVR();
  425. END_OF_INSTR();
  426. }
  427. /****************************************************************************
  428. REMARKS:
  429. Handles opcode 0x07
  430. ****************************************************************************/
  431. void x86emuOp_pop_ES(u8 X86EMU_UNUSED(op1))
  432. {
  433. START_OF_INSTR();
  434. DECODE_PRINTF("POP\tES\n");
  435. TRACE_AND_STEP();
  436. M.x86.R_ES = pop_word();
  437. DECODE_CLEAR_SEGOVR();
  438. END_OF_INSTR();
  439. }
  440. /****************************************************************************
  441. REMARKS:
  442. Handles opcode 0x0e
  443. ****************************************************************************/
  444. void x86emuOp_push_CS(u8 X86EMU_UNUSED(op1))
  445. {
  446. START_OF_INSTR();
  447. DECODE_PRINTF("PUSH\tCS\n");
  448. TRACE_AND_STEP();
  449. push_word(M.x86.R_CS);
  450. DECODE_CLEAR_SEGOVR();
  451. END_OF_INSTR();
  452. }
  453. /****************************************************************************
  454. REMARKS:
  455. Handles opcode 0x0f. Escape for two-byte opcode (286 or better)
  456. ****************************************************************************/
  457. void x86emuOp_two_byte(u8 X86EMU_UNUSED(op1))
  458. {
  459. u8 op2 = (*sys_rdb)(((u32)M.x86.R_CS << 4) + (M.x86.R_IP++));
  460. INC_DECODED_INST_LEN(1);
  461. (*x86emu_optab2[op2])(op2);
  462. }
  463. /****************************************************************************
  464. REMARKS:
  465. Handles opcode 0x16
  466. ****************************************************************************/
  467. void x86emuOp_push_SS(u8 X86EMU_UNUSED(op1))
  468. {
  469. START_OF_INSTR();
  470. DECODE_PRINTF("PUSH\tSS\n");
  471. TRACE_AND_STEP();
  472. push_word(M.x86.R_SS);
  473. DECODE_CLEAR_SEGOVR();
  474. END_OF_INSTR();
  475. }
  476. /****************************************************************************
  477. REMARKS:
  478. Handles opcode 0x17
  479. ****************************************************************************/
  480. void x86emuOp_pop_SS(u8 X86EMU_UNUSED(op1))
  481. {
  482. START_OF_INSTR();
  483. DECODE_PRINTF("POP\tSS\n");
  484. TRACE_AND_STEP();
  485. M.x86.R_SS = pop_word();
  486. DECODE_CLEAR_SEGOVR();
  487. END_OF_INSTR();
  488. }
  489. /****************************************************************************
  490. REMARKS:
  491. Handles opcode 0x1e
  492. ****************************************************************************/
  493. void x86emuOp_push_DS(u8 X86EMU_UNUSED(op1))
  494. {
  495. START_OF_INSTR();
  496. DECODE_PRINTF("PUSH\tDS\n");
  497. TRACE_AND_STEP();
  498. push_word(M.x86.R_DS);
  499. DECODE_CLEAR_SEGOVR();
  500. END_OF_INSTR();
  501. }
  502. /****************************************************************************
  503. REMARKS:
  504. Handles opcode 0x1f
  505. ****************************************************************************/
  506. void x86emuOp_pop_DS(u8 X86EMU_UNUSED(op1))
  507. {
  508. START_OF_INSTR();
  509. DECODE_PRINTF("POP\tDS\n");
  510. TRACE_AND_STEP();
  511. M.x86.R_DS = pop_word();
  512. DECODE_CLEAR_SEGOVR();
  513. END_OF_INSTR();
  514. }
  515. /****************************************************************************
  516. REMARKS:
  517. Handles opcode 0x26
  518. ****************************************************************************/
  519. void x86emuOp_segovr_ES(u8 X86EMU_UNUSED(op1))
  520. {
  521. START_OF_INSTR();
  522. DECODE_PRINTF("ES:\n");
  523. TRACE_AND_STEP();
  524. M.x86.mode |= SYSMODE_SEGOVR_ES;
  525. /*
  526. * note the lack of DECODE_CLEAR_SEGOVR(r) since, here is one of 4
  527. * opcode subroutines we do not want to do this.
  528. */
  529. END_OF_INSTR();
  530. }
  531. /****************************************************************************
  532. REMARKS:
  533. Handles opcode 0x27
  534. ****************************************************************************/
  535. void x86emuOp_daa(u8 X86EMU_UNUSED(op1))
  536. {
  537. START_OF_INSTR();
  538. DECODE_PRINTF("DAA\n");
  539. TRACE_AND_STEP();
  540. M.x86.R_AL = daa_byte(M.x86.R_AL);
  541. DECODE_CLEAR_SEGOVR();
  542. END_OF_INSTR();
  543. }
  544. /****************************************************************************
  545. REMARKS:
  546. Handles opcode 0x2e
  547. ****************************************************************************/
  548. void x86emuOp_segovr_CS(u8 X86EMU_UNUSED(op1))
  549. {
  550. START_OF_INSTR();
  551. DECODE_PRINTF("CS:\n");
  552. TRACE_AND_STEP();
  553. M.x86.mode |= SYSMODE_SEGOVR_CS;
  554. /* note no DECODE_CLEAR_SEGOVR here. */
  555. END_OF_INSTR();
  556. }
  557. /****************************************************************************
  558. REMARKS:
  559. Handles opcode 0x2f
  560. ****************************************************************************/
  561. void x86emuOp_das(u8 X86EMU_UNUSED(op1))
  562. {
  563. START_OF_INSTR();
  564. DECODE_PRINTF("DAS\n");
  565. TRACE_AND_STEP();
  566. M.x86.R_AL = das_byte(M.x86.R_AL);
  567. DECODE_CLEAR_SEGOVR();
  568. END_OF_INSTR();
  569. }
  570. /****************************************************************************
  571. REMARKS:
  572. Handles opcode 0x36
  573. ****************************************************************************/
  574. void x86emuOp_segovr_SS(u8 X86EMU_UNUSED(op1))
  575. {
  576. START_OF_INSTR();
  577. DECODE_PRINTF("SS:\n");
  578. TRACE_AND_STEP();
  579. M.x86.mode |= SYSMODE_SEGOVR_SS;
  580. /* no DECODE_CLEAR_SEGOVR ! */
  581. END_OF_INSTR();
  582. }
  583. /****************************************************************************
  584. REMARKS:
  585. Handles opcode 0x37
  586. ****************************************************************************/
  587. void x86emuOp_aaa(u8 X86EMU_UNUSED(op1))
  588. {
  589. START_OF_INSTR();
  590. DECODE_PRINTF("AAA\n");
  591. TRACE_AND_STEP();
  592. M.x86.R_AX = aaa_word(M.x86.R_AX);
  593. DECODE_CLEAR_SEGOVR();
  594. END_OF_INSTR();
  595. }
  596. /****************************************************************************
  597. REMARKS:
  598. Handles opcode 0x3e
  599. ****************************************************************************/
  600. void x86emuOp_segovr_DS(u8 X86EMU_UNUSED(op1))
  601. {
  602. START_OF_INSTR();
  603. DECODE_PRINTF("DS:\n");
  604. TRACE_AND_STEP();
  605. M.x86.mode |= SYSMODE_SEGOVR_DS;
  606. /* NO DECODE_CLEAR_SEGOVR! */
  607. END_OF_INSTR();
  608. }
  609. /****************************************************************************
  610. REMARKS:
  611. Handles opcode 0x3f
  612. ****************************************************************************/
  613. void x86emuOp_aas(u8 X86EMU_UNUSED(op1))
  614. {
  615. START_OF_INSTR();
  616. DECODE_PRINTF("AAS\n");
  617. TRACE_AND_STEP();
  618. M.x86.R_AX = aas_word(M.x86.R_AX);
  619. DECODE_CLEAR_SEGOVR();
  620. END_OF_INSTR();
  621. }
  622. /****************************************************************************
  623. REMARKS:
  624. Handles opcode 0x40 - 0x47
  625. ****************************************************************************/
  626. void x86emuOp_inc_register(u8 op1)
  627. {
  628. START_OF_INSTR();
  629. op1 &= 0x7;
  630. DECODE_PRINTF("INC\t");
  631. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  632. u32 *reg;
  633. reg = DECODE_RM_LONG_REGISTER(op1);
  634. DECODE_PRINTF("\n");
  635. TRACE_AND_STEP();
  636. *reg = inc_long(*reg);
  637. } else {
  638. u16 *reg;
  639. reg = DECODE_RM_WORD_REGISTER(op1);
  640. DECODE_PRINTF("\n");
  641. TRACE_AND_STEP();
  642. *reg = inc_word(*reg);
  643. }
  644. DECODE_CLEAR_SEGOVR();
  645. END_OF_INSTR();
  646. }
  647. /****************************************************************************
  648. REMARKS:
  649. Handles opcode 0x48 - 0x4F
  650. ****************************************************************************/
  651. void x86emuOp_dec_register(u8 op1)
  652. {
  653. START_OF_INSTR();
  654. op1 &= 0x7;
  655. DECODE_PRINTF("DEC\t");
  656. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  657. u32 *reg;
  658. reg = DECODE_RM_LONG_REGISTER(op1);
  659. DECODE_PRINTF("\n");
  660. TRACE_AND_STEP();
  661. *reg = dec_long(*reg);
  662. } else {
  663. u16 *reg;
  664. reg = DECODE_RM_WORD_REGISTER(op1);
  665. DECODE_PRINTF("\n");
  666. TRACE_AND_STEP();
  667. *reg = dec_word(*reg);
  668. }
  669. DECODE_CLEAR_SEGOVR();
  670. END_OF_INSTR();
  671. }
  672. /****************************************************************************
  673. REMARKS:
  674. Handles opcode 0x50 - 0x57
  675. ****************************************************************************/
  676. void x86emuOp_push_register(u8 op1)
  677. {
  678. START_OF_INSTR();
  679. op1 &= 0x7;
  680. DECODE_PRINTF("PUSH\t");
  681. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  682. u32 *reg;
  683. reg = DECODE_RM_LONG_REGISTER(op1);
  684. DECODE_PRINTF("\n");
  685. TRACE_AND_STEP();
  686. push_long(*reg);
  687. } else {
  688. u16 *reg;
  689. reg = DECODE_RM_WORD_REGISTER(op1);
  690. DECODE_PRINTF("\n");
  691. TRACE_AND_STEP();
  692. push_word(*reg);
  693. }
  694. DECODE_CLEAR_SEGOVR();
  695. END_OF_INSTR();
  696. }
  697. /****************************************************************************
  698. REMARKS:
  699. Handles opcode 0x58 - 0x5F
  700. ****************************************************************************/
  701. void x86emuOp_pop_register(u8 op1)
  702. {
  703. START_OF_INSTR();
  704. op1 &= 0x7;
  705. DECODE_PRINTF("POP\t");
  706. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  707. u32 *reg;
  708. reg = DECODE_RM_LONG_REGISTER(op1);
  709. DECODE_PRINTF("\n");
  710. TRACE_AND_STEP();
  711. *reg = pop_long();
  712. } else {
  713. u16 *reg;
  714. reg = DECODE_RM_WORD_REGISTER(op1);
  715. DECODE_PRINTF("\n");
  716. TRACE_AND_STEP();
  717. *reg = pop_word();
  718. }
  719. DECODE_CLEAR_SEGOVR();
  720. END_OF_INSTR();
  721. }
  722. /****************************************************************************
  723. REMARKS:
  724. Handles opcode 0x60
  725. ****************************************************************************/
  726. void x86emuOp_push_all(u8 X86EMU_UNUSED(op1))
  727. {
  728. START_OF_INSTR();
  729. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  730. DECODE_PRINTF("PUSHAD\n");
  731. } else {
  732. DECODE_PRINTF("PUSHA\n");
  733. }
  734. TRACE_AND_STEP();
  735. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  736. u32 old_sp = M.x86.R_ESP;
  737. push_long(M.x86.R_EAX);
  738. push_long(M.x86.R_ECX);
  739. push_long(M.x86.R_EDX);
  740. push_long(M.x86.R_EBX);
  741. push_long(old_sp);
  742. push_long(M.x86.R_EBP);
  743. push_long(M.x86.R_ESI);
  744. push_long(M.x86.R_EDI);
  745. } else {
  746. u16 old_sp = M.x86.R_SP;
  747. push_word(M.x86.R_AX);
  748. push_word(M.x86.R_CX);
  749. push_word(M.x86.R_DX);
  750. push_word(M.x86.R_BX);
  751. push_word(old_sp);
  752. push_word(M.x86.R_BP);
  753. push_word(M.x86.R_SI);
  754. push_word(M.x86.R_DI);
  755. }
  756. DECODE_CLEAR_SEGOVR();
  757. END_OF_INSTR();
  758. }
  759. /****************************************************************************
  760. REMARKS:
  761. Handles opcode 0x61
  762. ****************************************************************************/
  763. void x86emuOp_pop_all(u8 X86EMU_UNUSED(op1))
  764. {
  765. START_OF_INSTR();
  766. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  767. DECODE_PRINTF("POPAD\n");
  768. } else {
  769. DECODE_PRINTF("POPA\n");
  770. }
  771. TRACE_AND_STEP();
  772. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  773. M.x86.R_EDI = pop_long();
  774. M.x86.R_ESI = pop_long();
  775. M.x86.R_EBP = pop_long();
  776. M.x86.R_ESP += 4; /* skip ESP */
  777. M.x86.R_EBX = pop_long();
  778. M.x86.R_EDX = pop_long();
  779. M.x86.R_ECX = pop_long();
  780. M.x86.R_EAX = pop_long();
  781. } else {
  782. M.x86.R_DI = pop_word();
  783. M.x86.R_SI = pop_word();
  784. M.x86.R_BP = pop_word();
  785. M.x86.R_SP += 2; /* skip SP */
  786. M.x86.R_BX = pop_word();
  787. M.x86.R_DX = pop_word();
  788. M.x86.R_CX = pop_word();
  789. M.x86.R_AX = pop_word();
  790. }
  791. DECODE_CLEAR_SEGOVR();
  792. END_OF_INSTR();
  793. }
  794. /*opcode 0x62 ILLEGAL OP, calls x86emuOp_illegal_op() */
  795. /*opcode 0x63 ILLEGAL OP, calls x86emuOp_illegal_op() */
  796. /****************************************************************************
  797. REMARKS:
  798. Handles opcode 0x64
  799. ****************************************************************************/
  800. void x86emuOp_segovr_FS(u8 X86EMU_UNUSED(op1))
  801. {
  802. START_OF_INSTR();
  803. DECODE_PRINTF("FS:\n");
  804. TRACE_AND_STEP();
  805. M.x86.mode |= SYSMODE_SEGOVR_FS;
  806. /*
  807. * note the lack of DECODE_CLEAR_SEGOVR(r) since, here is one of 4
  808. * opcode subroutines we do not want to do this.
  809. */
  810. END_OF_INSTR();
  811. }
  812. /****************************************************************************
  813. REMARKS:
  814. Handles opcode 0x65
  815. ****************************************************************************/
  816. void x86emuOp_segovr_GS(u8 X86EMU_UNUSED(op1))
  817. {
  818. START_OF_INSTR();
  819. DECODE_PRINTF("GS:\n");
  820. TRACE_AND_STEP();
  821. M.x86.mode |= SYSMODE_SEGOVR_GS;
  822. /*
  823. * note the lack of DECODE_CLEAR_SEGOVR(r) since, here is one of 4
  824. * opcode subroutines we do not want to do this.
  825. */
  826. END_OF_INSTR();
  827. }
  828. /****************************************************************************
  829. REMARKS:
  830. Handles opcode 0x66 - prefix for 32-bit register
  831. ****************************************************************************/
  832. void x86emuOp_prefix_data(u8 X86EMU_UNUSED(op1))
  833. {
  834. START_OF_INSTR();
  835. DECODE_PRINTF("DATA:\n");
  836. TRACE_AND_STEP();
  837. M.x86.mode |= SYSMODE_PREFIX_DATA;
  838. /* note no DECODE_CLEAR_SEGOVR here. */
  839. END_OF_INSTR();
  840. }
  841. /****************************************************************************
  842. REMARKS:
  843. Handles opcode 0x67 - prefix for 32-bit address
  844. ****************************************************************************/
  845. void x86emuOp_prefix_addr(u8 X86EMU_UNUSED(op1))
  846. {
  847. START_OF_INSTR();
  848. DECODE_PRINTF("ADDR:\n");
  849. TRACE_AND_STEP();
  850. M.x86.mode |= SYSMODE_PREFIX_ADDR;
  851. /* note no DECODE_CLEAR_SEGOVR here. */
  852. END_OF_INSTR();
  853. }
  854. /****************************************************************************
  855. REMARKS:
  856. Handles opcode 0x68
  857. ****************************************************************************/
  858. void x86emuOp_push_word_IMM(u8 X86EMU_UNUSED(op1))
  859. {
  860. u32 imm;
  861. START_OF_INSTR();
  862. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  863. imm = fetch_long_imm();
  864. } else {
  865. imm = fetch_word_imm();
  866. }
  867. DECODE_PRINTF2("PUSH\t%x\n", imm);
  868. TRACE_AND_STEP();
  869. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  870. push_long(imm);
  871. } else {
  872. push_word((u16)imm);
  873. }
  874. DECODE_CLEAR_SEGOVR();
  875. END_OF_INSTR();
  876. }
  877. /****************************************************************************
  878. REMARKS:
  879. Handles opcode 0x69
  880. ****************************************************************************/
  881. void x86emuOp_imul_word_IMM(u8 X86EMU_UNUSED(op1))
  882. {
  883. int mod, rl, rh;
  884. uint srcoffset;
  885. START_OF_INSTR();
  886. DECODE_PRINTF("IMUL\t");
  887. FETCH_DECODE_MODRM(mod, rh, rl);
  888. if (mod < 3) {
  889. srcoffset = decode_rmXX_address(mod, rl);
  890. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  891. u32 *destreg;
  892. u32 srcval;
  893. u32 res_lo,res_hi;
  894. s32 imm;
  895. destreg = DECODE_RM_LONG_REGISTER(rh);
  896. DECODE_PRINTF(",");
  897. srcval = fetch_data_long(srcoffset);
  898. imm = fetch_long_imm();
  899. DECODE_PRINTF2(",%d\n", (s32)imm);
  900. TRACE_AND_STEP();
  901. imul_long_direct(&res_lo,&res_hi,(s32)srcval,(s32)imm);
  902. if ((((res_lo & 0x80000000) == 0) && (res_hi == 0x00000000)) ||
  903. (((res_lo & 0x80000000) != 0) && (res_hi == 0xFFFFFFFF))) {
  904. CLEAR_FLAG(F_CF);
  905. CLEAR_FLAG(F_OF);
  906. } else {
  907. SET_FLAG(F_CF);
  908. SET_FLAG(F_OF);
  909. }
  910. *destreg = (u32)res_lo;
  911. } else {
  912. u16 *destreg;
  913. u16 srcval;
  914. u32 res;
  915. s16 imm;
  916. destreg = DECODE_RM_WORD_REGISTER(rh);
  917. DECODE_PRINTF(",");
  918. srcval = fetch_data_word(srcoffset);
  919. imm = fetch_word_imm();
  920. DECODE_PRINTF2(",%d\n", (s32)imm);
  921. TRACE_AND_STEP();
  922. res = (s16)srcval * (s16)imm;
  923. if ((((res & 0x8000) == 0) && ((res >> 16) == 0x0000)) ||
  924. (((res & 0x8000) != 0) && ((res >> 16) == 0xFFFF))) {
  925. CLEAR_FLAG(F_CF);
  926. CLEAR_FLAG(F_OF);
  927. } else {
  928. SET_FLAG(F_CF);
  929. SET_FLAG(F_OF);
  930. }
  931. *destreg = (u16)res;
  932. }
  933. } else { /* register to register */
  934. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  935. u32 *destreg,*srcreg;
  936. u32 res_lo,res_hi;
  937. s32 imm;
  938. destreg = DECODE_RM_LONG_REGISTER(rh);
  939. DECODE_PRINTF(",");
  940. srcreg = DECODE_RM_LONG_REGISTER(rl);
  941. imm = fetch_long_imm();
  942. DECODE_PRINTF2(",%d\n", (s32)imm);
  943. TRACE_AND_STEP();
  944. imul_long_direct(&res_lo,&res_hi,(s32)*srcreg,(s32)imm);
  945. if ((((res_lo & 0x80000000) == 0) && (res_hi == 0x00000000)) ||
  946. (((res_lo & 0x80000000) != 0) && (res_hi == 0xFFFFFFFF))) {
  947. CLEAR_FLAG(F_CF);
  948. CLEAR_FLAG(F_OF);
  949. } else {
  950. SET_FLAG(F_CF);
  951. SET_FLAG(F_OF);
  952. }
  953. *destreg = (u32)res_lo;
  954. } else {
  955. u16 *destreg,*srcreg;
  956. u32 res;
  957. s16 imm;
  958. destreg = DECODE_RM_WORD_REGISTER(rh);
  959. DECODE_PRINTF(",");
  960. srcreg = DECODE_RM_WORD_REGISTER(rl);
  961. imm = fetch_word_imm();
  962. DECODE_PRINTF2(",%d\n", (s32)imm);
  963. res = (s16)*srcreg * (s16)imm;
  964. if ((((res & 0x8000) == 0) && ((res >> 16) == 0x0000)) ||
  965. (((res & 0x8000) != 0) && ((res >> 16) == 0xFFFF))) {
  966. CLEAR_FLAG(F_CF);
  967. CLEAR_FLAG(F_OF);
  968. } else {
  969. SET_FLAG(F_CF);
  970. SET_FLAG(F_OF);
  971. }
  972. *destreg = (u16)res;
  973. }
  974. }
  975. DECODE_CLEAR_SEGOVR();
  976. END_OF_INSTR();
  977. }
  978. /****************************************************************************
  979. REMARKS:
  980. Handles opcode 0x6a
  981. ****************************************************************************/
  982. void x86emuOp_push_byte_IMM(u8 X86EMU_UNUSED(op1))
  983. {
  984. s16 imm;
  985. START_OF_INSTR();
  986. imm = (s8)fetch_byte_imm();
  987. DECODE_PRINTF2("PUSH\t%d\n", imm);
  988. TRACE_AND_STEP();
  989. push_word(imm);
  990. DECODE_CLEAR_SEGOVR();
  991. END_OF_INSTR();
  992. }
  993. /****************************************************************************
  994. REMARKS:
  995. Handles opcode 0x6b
  996. ****************************************************************************/
  997. void x86emuOp_imul_byte_IMM(u8 X86EMU_UNUSED(op1))
  998. {
  999. int mod, rl, rh;
  1000. uint srcoffset;
  1001. s8 imm;
  1002. START_OF_INSTR();
  1003. DECODE_PRINTF("IMUL\t");
  1004. FETCH_DECODE_MODRM(mod, rh, rl);
  1005. if (mod < 3) {
  1006. srcoffset = decode_rmXX_address(mod, rl);
  1007. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  1008. u32 *destreg;
  1009. u32 srcval;
  1010. u32 res_lo,res_hi;
  1011. destreg = DECODE_RM_LONG_REGISTER(rh);
  1012. DECODE_PRINTF(",");
  1013. srcval = fetch_data_long(srcoffset);
  1014. imm = fetch_byte_imm();
  1015. DECODE_PRINTF2(",%d\n", (s32)imm);
  1016. TRACE_AND_STEP();
  1017. imul_long_direct(&res_lo,&res_hi,(s32)srcval,(s32)imm);
  1018. if ((((res_lo & 0x80000000) == 0) && (res_hi == 0x00000000)) ||
  1019. (((res_lo & 0x80000000) != 0) && (res_hi == 0xFFFFFFFF))) {
  1020. CLEAR_FLAG(F_CF);
  1021. CLEAR_FLAG(F_OF);
  1022. } else {
  1023. SET_FLAG(F_CF);
  1024. SET_FLAG(F_OF);
  1025. }
  1026. *destreg = (u32)res_lo;
  1027. } else {
  1028. u16 *destreg;
  1029. u16 srcval;
  1030. u32 res;
  1031. destreg = DECODE_RM_WORD_REGISTER(rh);
  1032. DECODE_PRINTF(",");
  1033. srcval = fetch_data_word(srcoffset);
  1034. imm = fetch_byte_imm();
  1035. DECODE_PRINTF2(",%d\n", (s32)imm);
  1036. TRACE_AND_STEP();
  1037. res = (s16)srcval * (s16)imm;
  1038. if ((((res & 0x8000) == 0) && ((res >> 16) == 0x0000)) ||
  1039. (((res & 0x8000) != 0) && ((res >> 16) == 0xFFFF))) {
  1040. CLEAR_FLAG(F_CF);
  1041. CLEAR_FLAG(F_OF);
  1042. } else {
  1043. SET_FLAG(F_CF);
  1044. SET_FLAG(F_OF);
  1045. }
  1046. *destreg = (u16)res;
  1047. }
  1048. } else { /* register to register */
  1049. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  1050. u32 *destreg,*srcreg;
  1051. u32 res_lo,res_hi;
  1052. destreg = DECODE_RM_LONG_REGISTER(rh);
  1053. DECODE_PRINTF(",");
  1054. srcreg = DECODE_RM_LONG_REGISTER(rl);
  1055. imm = fetch_byte_imm();
  1056. DECODE_PRINTF2(",%d\n", (s32)imm);
  1057. TRACE_AND_STEP();
  1058. imul_long_direct(&res_lo,&res_hi,(s32)*srcreg,(s32)imm);
  1059. if ((((res_lo & 0x80000000) == 0) && (res_hi == 0x00000000)) ||
  1060. (((res_lo & 0x80000000) != 0) && (res_hi == 0xFFFFFFFF))) {
  1061. CLEAR_FLAG(F_CF);
  1062. CLEAR_FLAG(F_OF);
  1063. } else {
  1064. SET_FLAG(F_CF);
  1065. SET_FLAG(F_OF);
  1066. }
  1067. *destreg = (u32)res_lo;
  1068. } else {
  1069. u16 *destreg,*srcreg;
  1070. u32 res;
  1071. destreg = DECODE_RM_WORD_REGISTER(rh);
  1072. DECODE_PRINTF(",");
  1073. srcreg = DECODE_RM_WORD_REGISTER(rl);
  1074. imm = fetch_byte_imm();
  1075. DECODE_PRINTF2(",%d\n", (s32)imm);
  1076. TRACE_AND_STEP();
  1077. res = (s16)*srcreg * (s16)imm;
  1078. if ((((res & 0x8000) == 0) && ((res >> 16) == 0x0000)) ||
  1079. (((res & 0x8000) != 0) && ((res >> 16) == 0xFFFF))) {
  1080. CLEAR_FLAG(F_CF);
  1081. CLEAR_FLAG(F_OF);
  1082. } else {
  1083. SET_FLAG(F_CF);
  1084. SET_FLAG(F_OF);
  1085. }
  1086. *destreg = (u16)res;
  1087. }
  1088. }
  1089. DECODE_CLEAR_SEGOVR();
  1090. END_OF_INSTR();
  1091. }
  1092. /****************************************************************************
  1093. REMARKS:
  1094. Handles opcode 0x6c
  1095. ****************************************************************************/
  1096. void x86emuOp_ins_byte(u8 X86EMU_UNUSED(op1))
  1097. {
  1098. START_OF_INSTR();
  1099. DECODE_PRINTF("INSB\n");
  1100. ins(1);
  1101. TRACE_AND_STEP();
  1102. DECODE_CLEAR_SEGOVR();
  1103. END_OF_INSTR();
  1104. }
  1105. /****************************************************************************
  1106. REMARKS:
  1107. Handles opcode 0x6d
  1108. ****************************************************************************/
  1109. void x86emuOp_ins_word(u8 X86EMU_UNUSED(op1))
  1110. {
  1111. START_OF_INSTR();
  1112. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  1113. DECODE_PRINTF("INSD\n");
  1114. ins(4);
  1115. } else {
  1116. DECODE_PRINTF("INSW\n");
  1117. ins(2);
  1118. }
  1119. TRACE_AND_STEP();
  1120. DECODE_CLEAR_SEGOVR();
  1121. END_OF_INSTR();
  1122. }
  1123. /****************************************************************************
  1124. REMARKS:
  1125. Handles opcode 0x6e
  1126. ****************************************************************************/
  1127. void x86emuOp_outs_byte(u8 X86EMU_UNUSED(op1))
  1128. {
  1129. START_OF_INSTR();
  1130. DECODE_PRINTF("OUTSB\n");
  1131. outs(1);
  1132. TRACE_AND_STEP();
  1133. DECODE_CLEAR_SEGOVR();
  1134. END_OF_INSTR();
  1135. }
  1136. /****************************************************************************
  1137. REMARKS:
  1138. Handles opcode 0x6f
  1139. ****************************************************************************/
  1140. void x86emuOp_outs_word(u8 X86EMU_UNUSED(op1))
  1141. {
  1142. START_OF_INSTR();
  1143. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  1144. DECODE_PRINTF("OUTSD\n");
  1145. outs(4);
  1146. } else {
  1147. DECODE_PRINTF("OUTSW\n");
  1148. outs(2);
  1149. }
  1150. TRACE_AND_STEP();
  1151. DECODE_CLEAR_SEGOVR();
  1152. END_OF_INSTR();
  1153. }
  1154. /****************************************************************************
  1155. REMARKS:
  1156. Handles opcode 0x70 - 0x7F
  1157. ****************************************************************************/
  1158. int x86emu_check_jump_condition(u8 op);
  1159. void x86emuOp_jump_near_cond(u8 op1)
  1160. {
  1161. s8 offset;
  1162. u16 target;
  1163. int cond;
  1164. /* jump to byte offset if overflow flag is set */
  1165. START_OF_INSTR();
  1166. cond = x86emu_check_jump_condition(op1 & 0xF);
  1167. offset = (s8)fetch_byte_imm();
  1168. target = (u16)(M.x86.R_IP + (s16)offset);
  1169. DECODE_PRINTF2("%x\n", target);
  1170. TRACE_AND_STEP();
  1171. if (cond)
  1172. M.x86.R_IP = target;
  1173. DECODE_CLEAR_SEGOVR();
  1174. END_OF_INSTR();
  1175. }
  1176. /****************************************************************************
  1177. REMARKS:
  1178. Handles opcode 0x80
  1179. ****************************************************************************/
  1180. void x86emuOp_opc80_byte_RM_IMM(u8 X86EMU_UNUSED(op1))
  1181. {
  1182. int mod, rl, rh;
  1183. u8 *destreg;
  1184. uint destoffset;
  1185. u8 imm;
  1186. u8 destval;
  1187. /*
  1188. * Weirdo special case instruction format. Part of the opcode
  1189. * held below in "RH". Doubly nested case would result, except
  1190. * that the decoded instruction
  1191. */
  1192. START_OF_INSTR();
  1193. FETCH_DECODE_MODRM(mod, rh, rl);
  1194. #ifdef DEBUG
  1195. if (DEBUG_DECODE()) {
  1196. /* XXX DECODE_PRINTF may be changed to something more
  1197. general, so that it is important to leave the strings
  1198. in the same format, even though the result is that the
  1199. above test is done twice. */
  1200. switch (rh) {
  1201. case 0:
  1202. DECODE_PRINTF("ADD\t");
  1203. break;
  1204. case 1:
  1205. DECODE_PRINTF("OR\t");
  1206. break;
  1207. case 2:
  1208. DECODE_PRINTF("ADC\t");
  1209. break;
  1210. case 3:
  1211. DECODE_PRINTF("SBB\t");
  1212. break;
  1213. case 4:
  1214. DECODE_PRINTF("AND\t");
  1215. break;
  1216. case 5:
  1217. DECODE_PRINTF("SUB\t");
  1218. break;
  1219. case 6:
  1220. DECODE_PRINTF("XOR\t");
  1221. break;
  1222. case 7:
  1223. DECODE_PRINTF("CMP\t");
  1224. break;
  1225. }
  1226. }
  1227. #endif
  1228. /* know operation, decode the mod byte to find the addressing
  1229. mode. */
  1230. if (mod < 3) {
  1231. DECODE_PRINTF("BYTE PTR ");
  1232. destoffset = decode_rmXX_address(mod, rl);
  1233. DECODE_PRINTF(",");
  1234. destval = fetch_data_byte(destoffset);
  1235. imm = fetch_byte_imm();
  1236. DECODE_PRINTF2("%x\n", imm);
  1237. TRACE_AND_STEP();
  1238. destval = (*genop_byte_operation[rh]) (destval, imm);
  1239. if (rh != 7)
  1240. store_data_byte(destoffset, destval);
  1241. } else { /* register to register */
  1242. destreg = DECODE_RM_BYTE_REGISTER(rl);
  1243. DECODE_PRINTF(",");
  1244. imm = fetch_byte_imm();
  1245. DECODE_PRINTF2("%x\n", imm);
  1246. TRACE_AND_STEP();
  1247. destval = (*genop_byte_operation[rh]) (*destreg, imm);
  1248. if (rh != 7)
  1249. *destreg = destval;
  1250. }
  1251. DECODE_CLEAR_SEGOVR();
  1252. END_OF_INSTR();
  1253. }
  1254. /****************************************************************************
  1255. REMARKS:
  1256. Handles opcode 0x81
  1257. ****************************************************************************/
  1258. void x86emuOp_opc81_word_RM_IMM(u8 X86EMU_UNUSED(op1))
  1259. {
  1260. int mod, rl, rh;
  1261. uint destoffset;
  1262. /*
  1263. * Weirdo special case instruction format. Part of the opcode
  1264. * held below in "RH". Doubly nested case would result, except
  1265. * that the decoded instruction
  1266. */
  1267. START_OF_INSTR();
  1268. FETCH_DECODE_MODRM(mod, rh, rl);
  1269. #ifdef DEBUG
  1270. if (DEBUG_DECODE()) {
  1271. /* XXX DECODE_PRINTF may be changed to something more
  1272. general, so that it is important to leave the strings
  1273. in the same format, even though the result is that the
  1274. above test is done twice. */
  1275. switch (rh) {
  1276. case 0:
  1277. DECODE_PRINTF("ADD\t");
  1278. break;
  1279. case 1:
  1280. DECODE_PRINTF("OR\t");
  1281. break;
  1282. case 2:
  1283. DECODE_PRINTF("ADC\t");
  1284. break;
  1285. case 3:
  1286. DECODE_PRINTF("SBB\t");
  1287. break;
  1288. case 4:
  1289. DECODE_PRINTF("AND\t");
  1290. break;
  1291. case 5:
  1292. DECODE_PRINTF("SUB\t");
  1293. break;
  1294. case 6:
  1295. DECODE_PRINTF("XOR\t");
  1296. break;
  1297. case 7:
  1298. DECODE_PRINTF("CMP\t");
  1299. break;
  1300. }
  1301. }
  1302. #endif
  1303. /*
  1304. * Know operation, decode the mod byte to find the addressing
  1305. * mode.
  1306. */
  1307. if (mod < 3) {
  1308. DECODE_PRINTF("DWORD PTR ");
  1309. destoffset = decode_rmXX_address(mod, rl);
  1310. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  1311. u32 destval,imm;
  1312. DECODE_PRINTF(",");
  1313. destval = fetch_data_long(destoffset);
  1314. imm = fetch_long_imm();
  1315. DECODE_PRINTF2("%x\n", imm);
  1316. TRACE_AND_STEP();
  1317. destval = (*genop_long_operation[rh]) (destval, imm);
  1318. if (rh != 7)
  1319. store_data_long(destoffset, destval);
  1320. } else {
  1321. u16 destval,imm;
  1322. DECODE_PRINTF(",");
  1323. destval = fetch_data_word(destoffset);
  1324. imm = fetch_word_imm();
  1325. DECODE_PRINTF2("%x\n", imm);
  1326. TRACE_AND_STEP();
  1327. destval = (*genop_word_operation[rh]) (destval, imm);
  1328. if (rh != 7)
  1329. store_data_word(destoffset, destval);
  1330. }
  1331. } else { /* register to register */
  1332. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  1333. u32 *destreg;
  1334. u32 destval,imm;
  1335. destreg = DECODE_RM_LONG_REGISTER(rl);
  1336. DECODE_PRINTF(",");
  1337. imm = fetch_long_imm();
  1338. DECODE_PRINTF2("%x\n", imm);
  1339. TRACE_AND_STEP();
  1340. destval = (*genop_long_operation[rh]) (*destreg, imm);
  1341. if (rh != 7)
  1342. *destreg = destval;
  1343. } else {
  1344. u16 *destreg;
  1345. u16 destval,imm;
  1346. destreg = DECODE_RM_WORD_REGISTER(rl);
  1347. DECODE_PRINTF(",");
  1348. imm = fetch_word_imm();
  1349. DECODE_PRINTF2("%x\n", imm);
  1350. TRACE_AND_STEP();
  1351. destval = (*genop_word_operation[rh]) (*destreg, imm);
  1352. if (rh != 7)
  1353. *destreg = destval;
  1354. }
  1355. }
  1356. DECODE_CLEAR_SEGOVR();
  1357. END_OF_INSTR();
  1358. }
  1359. /****************************************************************************
  1360. REMARKS:
  1361. Handles opcode 0x82
  1362. ****************************************************************************/
  1363. void x86emuOp_opc82_byte_RM_IMM(u8 X86EMU_UNUSED(op1))
  1364. {
  1365. int mod, rl, rh;
  1366. u8 *destreg;
  1367. uint destoffset;
  1368. u8 imm;
  1369. u8 destval;
  1370. /*
  1371. * Weirdo special case instruction format. Part of the opcode
  1372. * held below in "RH". Doubly nested case would result, except
  1373. * that the decoded instruction Similar to opcode 81, except that
  1374. * the immediate byte is sign extended to a word length.
  1375. */
  1376. START_OF_INSTR();
  1377. FETCH_DECODE_MODRM(mod, rh, rl);
  1378. #ifdef DEBUG
  1379. if (DEBUG_DECODE()) {
  1380. /* XXX DECODE_PRINTF may be changed to something more
  1381. general, so that it is important to leave the strings
  1382. in the same format, even though the result is that the
  1383. above test is done twice. */
  1384. switch (rh) {
  1385. case 0:
  1386. DECODE_PRINTF("ADD\t");
  1387. break;
  1388. case 1:
  1389. DECODE_PRINTF("OR\t");
  1390. break;
  1391. case 2:
  1392. DECODE_PRINTF("ADC\t");
  1393. break;
  1394. case 3:
  1395. DECODE_PRINTF("SBB\t");
  1396. break;
  1397. case 4:
  1398. DECODE_PRINTF("AND\t");
  1399. break;
  1400. case 5:
  1401. DECODE_PRINTF("SUB\t");
  1402. break;
  1403. case 6:
  1404. DECODE_PRINTF("XOR\t");
  1405. break;
  1406. case 7:
  1407. DECODE_PRINTF("CMP\t");
  1408. break;
  1409. }
  1410. }
  1411. #endif
  1412. /* know operation, decode the mod byte to find the addressing
  1413. mode. */
  1414. if (mod < 3) {
  1415. DECODE_PRINTF("BYTE PTR ");
  1416. destoffset = decode_rmXX_address(mod, rl);
  1417. destval = fetch_data_byte(destoffset);
  1418. imm = fetch_byte_imm();
  1419. DECODE_PRINTF2(",%x\n", imm);
  1420. TRACE_AND_STEP();
  1421. destval = (*genop_byte_operation[rh]) (destval, imm);
  1422. if (rh != 7)
  1423. store_data_byte(destoffset, destval);
  1424. } else { /* register to register */
  1425. destreg = DECODE_RM_BYTE_REGISTER(rl);
  1426. imm = fetch_byte_imm();
  1427. DECODE_PRINTF2(",%x\n", imm);
  1428. TRACE_AND_STEP();
  1429. destval = (*genop_byte_operation[rh]) (*destreg, imm);
  1430. if (rh != 7)
  1431. *destreg = destval;
  1432. }
  1433. DECODE_CLEAR_SEGOVR();
  1434. END_OF_INSTR();
  1435. }
  1436. /****************************************************************************
  1437. REMARKS:
  1438. Handles opcode 0x83
  1439. ****************************************************************************/
  1440. void x86emuOp_opc83_word_RM_IMM(u8 X86EMU_UNUSED(op1))
  1441. {
  1442. int mod, rl, rh;
  1443. uint destoffset;
  1444. /*
  1445. * Weirdo special case instruction format. Part of the opcode
  1446. * held below in "RH". Doubly nested case would result, except
  1447. * that the decoded instruction Similar to opcode 81, except that
  1448. * the immediate byte is sign extended to a word length.
  1449. */
  1450. START_OF_INSTR();
  1451. FETCH_DECODE_MODRM(mod, rh, rl);
  1452. #ifdef DEBUG
  1453. if (DEBUG_DECODE()) {
  1454. /* XXX DECODE_PRINTF may be changed to something more
  1455. general, so that it is important to leave the strings
  1456. in the same format, even though the result is that the
  1457. above test is done twice. */
  1458. switch (rh) {
  1459. case 0:
  1460. DECODE_PRINTF("ADD\t");
  1461. break;
  1462. case 1:
  1463. DECODE_PRINTF("OR\t");
  1464. break;
  1465. case 2:
  1466. DECODE_PRINTF("ADC\t");
  1467. break;
  1468. case 3:
  1469. DECODE_PRINTF("SBB\t");
  1470. break;
  1471. case 4:
  1472. DECODE_PRINTF("AND\t");
  1473. break;
  1474. case 5:
  1475. DECODE_PRINTF("SUB\t");
  1476. break;
  1477. case 6:
  1478. DECODE_PRINTF("XOR\t");
  1479. break;
  1480. case 7:
  1481. DECODE_PRINTF("CMP\t");
  1482. break;
  1483. }
  1484. }
  1485. #endif
  1486. /* know operation, decode the mod byte to find the addressing
  1487. mode. */
  1488. if (mod < 3) {
  1489. DECODE_PRINTF("DWORD PTR ");
  1490. destoffset = decode_rmXX_address(mod,rl);
  1491. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  1492. u32 destval,imm;
  1493. destval = fetch_data_long(destoffset);
  1494. imm = (s8) fetch_byte_imm();
  1495. DECODE_PRINTF2(",%x\n", imm);
  1496. TRACE_AND_STEP();
  1497. destval = (*genop_long_operation[rh]) (destval, imm);
  1498. if (rh != 7)
  1499. store_data_long(destoffset, destval);
  1500. } else {
  1501. u16 destval,imm;
  1502. destval = fetch_data_word(destoffset);
  1503. imm = (s8) fetch_byte_imm();
  1504. DECODE_PRINTF2(",%x\n", imm);
  1505. TRACE_AND_STEP();
  1506. destval = (*genop_word_operation[rh]) (destval, imm);
  1507. if (rh != 7)
  1508. store_data_word(destoffset, destval);
  1509. }
  1510. } else { /* register to register */
  1511. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  1512. u32 *destreg;
  1513. u32 destval,imm;
  1514. destreg = DECODE_RM_LONG_REGISTER(rl);
  1515. imm = (s8) fetch_byte_imm();
  1516. DECODE_PRINTF2(",%x\n", imm);
  1517. TRACE_AND_STEP();
  1518. destval = (*genop_long_operation[rh]) (*destreg, imm);
  1519. if (rh != 7)
  1520. *destreg = destval;
  1521. } else {
  1522. u16 *destreg;
  1523. u16 destval,imm;
  1524. destreg = DECODE_RM_WORD_REGISTER(rl);
  1525. imm = (s8) fetch_byte_imm();
  1526. DECODE_PRINTF2(",%x\n", imm);
  1527. TRACE_AND_STEP();
  1528. destval = (*genop_word_operation[rh]) (*destreg, imm);
  1529. if (rh != 7)
  1530. *destreg = destval;
  1531. }
  1532. }
  1533. DECODE_CLEAR_SEGOVR();
  1534. END_OF_INSTR();
  1535. }
  1536. /****************************************************************************
  1537. REMARKS:
  1538. Handles opcode 0x84
  1539. ****************************************************************************/
  1540. void x86emuOp_test_byte_RM_R(u8 X86EMU_UNUSED(op1))
  1541. {
  1542. int mod, rl, rh;
  1543. u8 *destreg, *srcreg;
  1544. uint destoffset;
  1545. u8 destval;
  1546. START_OF_INSTR();
  1547. DECODE_PRINTF("TEST\t");
  1548. FETCH_DECODE_MODRM(mod, rh, rl);
  1549. if (mod < 3) {
  1550. destoffset = decode_rmXX_address(mod, rl);
  1551. DECODE_PRINTF(",");
  1552. destval = fetch_data_byte(destoffset);
  1553. srcreg = DECODE_RM_BYTE_REGISTER(rh);
  1554. DECODE_PRINTF("\n");
  1555. TRACE_AND_STEP();
  1556. test_byte(destval, *srcreg);
  1557. } else { /* register to register */
  1558. destreg = DECODE_RM_BYTE_REGISTER(rl);
  1559. DECODE_PRINTF(",");
  1560. srcreg = DECODE_RM_BYTE_REGISTER(rh);
  1561. DECODE_PRINTF("\n");
  1562. TRACE_AND_STEP();
  1563. test_byte(*destreg, *srcreg);
  1564. }
  1565. DECODE_CLEAR_SEGOVR();
  1566. END_OF_INSTR();
  1567. }
  1568. /****************************************************************************
  1569. REMARKS:
  1570. Handles opcode 0x85
  1571. ****************************************************************************/
  1572. void x86emuOp_test_word_RM_R(u8 X86EMU_UNUSED(op1))
  1573. {
  1574. int mod, rl, rh;
  1575. uint destoffset;
  1576. START_OF_INSTR();
  1577. DECODE_PRINTF("TEST\t");
  1578. FETCH_DECODE_MODRM(mod, rh, rl);
  1579. if (mod < 3) {
  1580. destoffset = decode_rmXX_address(mod, rl);
  1581. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  1582. u32 destval;
  1583. u32 *srcreg;
  1584. DECODE_PRINTF(",");
  1585. destval = fetch_data_long(destoffset);
  1586. srcreg = DECODE_RM_LONG_REGISTER(rh);
  1587. DECODE_PRINTF("\n");
  1588. TRACE_AND_STEP();
  1589. test_long(destval, *srcreg);
  1590. } else {
  1591. u16 destval;
  1592. u16 *srcreg;
  1593. DECODE_PRINTF(",");
  1594. destval = fetch_data_word(destoffset);
  1595. srcreg = DECODE_RM_WORD_REGISTER(rh);
  1596. DECODE_PRINTF("\n");
  1597. TRACE_AND_STEP();
  1598. test_word(destval, *srcreg);
  1599. }
  1600. } else { /* register to register */
  1601. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  1602. u32 *destreg,*srcreg;
  1603. destreg = DECODE_RM_LONG_REGISTER(rl);
  1604. DECODE_PRINTF(",");
  1605. srcreg = DECODE_RM_LONG_REGISTER(rh);
  1606. DECODE_PRINTF("\n");
  1607. TRACE_AND_STEP();
  1608. test_long(*destreg, *srcreg);
  1609. } else {
  1610. u16 *destreg,*srcreg;
  1611. destreg = DECODE_RM_WORD_REGISTER(rl);
  1612. DECODE_PRINTF(",");
  1613. srcreg = DECODE_RM_WORD_REGISTER(rh);
  1614. DECODE_PRINTF("\n");
  1615. TRACE_AND_STEP();
  1616. test_word(*destreg, *srcreg);
  1617. }
  1618. }
  1619. DECODE_CLEAR_SEGOVR();
  1620. END_OF_INSTR();
  1621. }
  1622. /****************************************************************************
  1623. REMARKS:
  1624. Handles opcode 0x86
  1625. ****************************************************************************/
  1626. void x86emuOp_xchg_byte_RM_R(u8 X86EMU_UNUSED(op1))
  1627. {
  1628. int mod, rl, rh;
  1629. u8 *destreg, *srcreg;
  1630. uint destoffset;
  1631. u8 destval;
  1632. u8 tmp;
  1633. START_OF_INSTR();
  1634. DECODE_PRINTF("XCHG\t");
  1635. FETCH_DECODE_MODRM(mod, rh, rl);
  1636. if (mod < 3) {
  1637. destoffset = decode_rmXX_address(mod, rl);
  1638. DECODE_PRINTF(",");
  1639. destval = fetch_data_byte(destoffset);
  1640. srcreg = DECODE_RM_BYTE_REGISTER(rh);
  1641. DECODE_PRINTF("\n");
  1642. TRACE_AND_STEP();
  1643. tmp = *srcreg;
  1644. *srcreg = destval;
  1645. destval = tmp;
  1646. store_data_byte(destoffset, destval);
  1647. } else { /* register to register */
  1648. destreg = DECODE_RM_BYTE_REGISTER(rl);
  1649. DECODE_PRINTF(",");
  1650. srcreg = DECODE_RM_BYTE_REGISTER(rh);
  1651. DECODE_PRINTF("\n");
  1652. TRACE_AND_STEP();
  1653. tmp = *srcreg;
  1654. *srcreg = *destreg;
  1655. *destreg = tmp;
  1656. }
  1657. DECODE_CLEAR_SEGOVR();
  1658. END_OF_INSTR();
  1659. }
  1660. /****************************************************************************
  1661. REMARKS:
  1662. Handles opcode 0x87
  1663. ****************************************************************************/
  1664. void x86emuOp_xchg_word_RM_R(u8 X86EMU_UNUSED(op1))
  1665. {
  1666. int mod, rl, rh;
  1667. uint destoffset;
  1668. START_OF_INSTR();
  1669. DECODE_PRINTF("XCHG\t");
  1670. FETCH_DECODE_MODRM(mod, rh, rl);
  1671. if (mod < 3) {
  1672. destoffset = decode_rmXX_address(mod, rl);
  1673. DECODE_PRINTF(",");
  1674. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  1675. u32 *srcreg;
  1676. u32 destval,tmp;
  1677. destval = fetch_data_long(destoffset);
  1678. srcreg = DECODE_RM_LONG_REGISTER(rh);
  1679. DECODE_PRINTF("\n");
  1680. TRACE_AND_STEP();
  1681. tmp = *srcreg;
  1682. *srcreg = destval;
  1683. destval = tmp;
  1684. store_data_long(destoffset, destval);
  1685. } else {
  1686. u16 *srcreg;
  1687. u16 destval,tmp;
  1688. destval = fetch_data_word(destoffset);
  1689. srcreg = DECODE_RM_WORD_REGISTER(rh);
  1690. DECODE_PRINTF("\n");
  1691. TRACE_AND_STEP();
  1692. tmp = *srcreg;
  1693. *srcreg = destval;
  1694. destval = tmp;
  1695. store_data_word(destoffset, destval);
  1696. }
  1697. } else { /* register to register */
  1698. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  1699. u32 *destreg,*srcreg;
  1700. u32 tmp;
  1701. destreg = DECODE_RM_LONG_REGISTER(rl);
  1702. DECODE_PRINTF(",");
  1703. srcreg = DECODE_RM_LONG_REGISTER(rh);
  1704. DECODE_PRINTF("\n");
  1705. TRACE_AND_STEP();
  1706. tmp = *srcreg;
  1707. *srcreg = *destreg;
  1708. *destreg = tmp;
  1709. } else {
  1710. u16 *destreg,*srcreg;
  1711. u16 tmp;
  1712. destreg = DECODE_RM_WORD_REGISTER(rl);
  1713. DECODE_PRINTF(",");
  1714. srcreg = DECODE_RM_WORD_REGISTER(rh);
  1715. DECODE_PRINTF("\n");
  1716. TRACE_AND_STEP();
  1717. tmp = *srcreg;
  1718. *srcreg = *destreg;
  1719. *destreg = tmp;
  1720. }
  1721. }
  1722. DECODE_CLEAR_SEGOVR();
  1723. END_OF_INSTR();
  1724. }
  1725. /****************************************************************************
  1726. REMARKS:
  1727. Handles opcode 0x88
  1728. ****************************************************************************/
  1729. void x86emuOp_mov_byte_RM_R(u8 X86EMU_UNUSED(op1))
  1730. {
  1731. int mod, rl, rh;
  1732. u8 *destreg, *srcreg;
  1733. uint destoffset;
  1734. START_OF_INSTR();
  1735. DECODE_PRINTF("MOV\t");
  1736. FETCH_DECODE_MODRM(mod, rh, rl);
  1737. if (mod < 3) {
  1738. destoffset = decode_rmXX_address(mod, rl);
  1739. DECODE_PRINTF(",");
  1740. srcreg = DECODE_RM_BYTE_REGISTER(rh);
  1741. DECODE_PRINTF("\n");
  1742. TRACE_AND_STEP();
  1743. store_data_byte(destoffset, *srcreg);
  1744. } else { /* register to register */
  1745. destreg = DECODE_RM_BYTE_REGISTER(rl);
  1746. DECODE_PRINTF(",");
  1747. srcreg = DECODE_RM_BYTE_REGISTER(rh);
  1748. DECODE_PRINTF("\n");
  1749. TRACE_AND_STEP();
  1750. *destreg = *srcreg;
  1751. }
  1752. DECODE_CLEAR_SEGOVR();
  1753. END_OF_INSTR();
  1754. }
  1755. /****************************************************************************
  1756. REMARKS:
  1757. Handles opcode 0x89
  1758. ****************************************************************************/
  1759. void x86emuOp_mov_word_RM_R(u8 X86EMU_UNUSED(op1))
  1760. {
  1761. int mod, rl, rh;
  1762. uint destoffset;
  1763. START_OF_INSTR();
  1764. DECODE_PRINTF("MOV\t");
  1765. FETCH_DECODE_MODRM(mod, rh, rl);
  1766. if (mod < 3) {
  1767. destoffset = decode_rmXX_address(mod, rl);
  1768. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  1769. u32 *srcreg;
  1770. DECODE_PRINTF(",");
  1771. srcreg = DECODE_RM_LONG_REGISTER(rh);
  1772. DECODE_PRINTF("\n");
  1773. TRACE_AND_STEP();
  1774. store_data_long(destoffset, *srcreg);
  1775. } else {
  1776. u16 *srcreg;
  1777. DECODE_PRINTF(",");
  1778. srcreg = DECODE_RM_WORD_REGISTER(rh);
  1779. DECODE_PRINTF("\n");
  1780. TRACE_AND_STEP();
  1781. store_data_word(destoffset, *srcreg);
  1782. }
  1783. } else { /* register to register */
  1784. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  1785. u32 *destreg,*srcreg;
  1786. destreg = DECODE_RM_LONG_REGISTER(rl);
  1787. DECODE_PRINTF(",");
  1788. srcreg = DECODE_RM_LONG_REGISTER(rh);
  1789. DECODE_PRINTF("\n");
  1790. TRACE_AND_STEP();
  1791. *destreg = *srcreg;
  1792. } else {
  1793. u16 *destreg,*srcreg;
  1794. destreg = DECODE_RM_WORD_REGISTER(rl);
  1795. DECODE_PRINTF(",");
  1796. srcreg = DECODE_RM_WORD_REGISTER(rh);
  1797. DECODE_PRINTF("\n");
  1798. TRACE_AND_STEP();
  1799. *destreg = *srcreg;
  1800. }
  1801. }
  1802. DECODE_CLEAR_SEGOVR();
  1803. END_OF_INSTR();
  1804. }
  1805. /****************************************************************************
  1806. REMARKS:
  1807. Handles opcode 0x8a
  1808. ****************************************************************************/
  1809. void x86emuOp_mov_byte_R_RM(u8 X86EMU_UNUSED(op1))
  1810. {
  1811. int mod, rl, rh;
  1812. u8 *destreg, *srcreg;
  1813. uint srcoffset;
  1814. u8 srcval;
  1815. START_OF_INSTR();
  1816. DECODE_PRINTF("MOV\t");
  1817. FETCH_DECODE_MODRM(mod, rh, rl);
  1818. if (mod < 3) {
  1819. destreg = DECODE_RM_BYTE_REGISTER(rh);
  1820. DECODE_PRINTF(",");
  1821. srcoffset = decode_rmXX_address(mod, rl);
  1822. srcval = fetch_data_byte(srcoffset);
  1823. DECODE_PRINTF("\n");
  1824. TRACE_AND_STEP();
  1825. *destreg = srcval;
  1826. } else { /* register to register */
  1827. destreg = DECODE_RM_BYTE_REGISTER(rh);
  1828. DECODE_PRINTF(",");
  1829. srcreg = DECODE_RM_BYTE_REGISTER(rl);
  1830. DECODE_PRINTF("\n");
  1831. TRACE_AND_STEP();
  1832. *destreg = *srcreg;
  1833. }
  1834. DECODE_CLEAR_SEGOVR();
  1835. END_OF_INSTR();
  1836. }
  1837. /****************************************************************************
  1838. REMARKS:
  1839. Handles opcode 0x8b
  1840. ****************************************************************************/
  1841. void x86emuOp_mov_word_R_RM(u8 X86EMU_UNUSED(op1))
  1842. {
  1843. int mod, rl, rh;
  1844. uint srcoffset;
  1845. START_OF_INSTR();
  1846. DECODE_PRINTF("MOV\t");
  1847. FETCH_DECODE_MODRM(mod, rh, rl);
  1848. if (mod < 3) {
  1849. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  1850. u32 *destreg;
  1851. u32 srcval;
  1852. destreg = DECODE_RM_LONG_REGISTER(rh);
  1853. DECODE_PRINTF(",");
  1854. srcoffset = decode_rmXX_address(mod, rl);
  1855. srcval = fetch_data_long(srcoffset);
  1856. DECODE_PRINTF("\n");
  1857. TRACE_AND_STEP();
  1858. *destreg = srcval;
  1859. } else {
  1860. u16 *destreg;
  1861. u16 srcval;
  1862. destreg = DECODE_RM_WORD_REGISTER(rh);
  1863. DECODE_PRINTF(",");
  1864. srcoffset = decode_rmXX_address(mod, rl);
  1865. srcval = fetch_data_word(srcoffset);
  1866. DECODE_PRINTF("\n");
  1867. TRACE_AND_STEP();
  1868. *destreg = srcval;
  1869. }
  1870. } else { /* register to register */
  1871. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  1872. u32 *destreg, *srcreg;
  1873. destreg = DECODE_RM_LONG_REGISTER(rh);
  1874. DECODE_PRINTF(",");
  1875. srcreg = DECODE_RM_LONG_REGISTER(rl);
  1876. DECODE_PRINTF("\n");
  1877. TRACE_AND_STEP();
  1878. *destreg = *srcreg;
  1879. } else {
  1880. u16 *destreg, *srcreg;
  1881. destreg = DECODE_RM_WORD_REGISTER(rh);
  1882. DECODE_PRINTF(",");
  1883. srcreg = DECODE_RM_WORD_REGISTER(rl);
  1884. DECODE_PRINTF("\n");
  1885. TRACE_AND_STEP();
  1886. *destreg = *srcreg;
  1887. }
  1888. }
  1889. DECODE_CLEAR_SEGOVR();
  1890. END_OF_INSTR();
  1891. }
  1892. /****************************************************************************
  1893. REMARKS:
  1894. Handles opcode 0x8c
  1895. ****************************************************************************/
  1896. void x86emuOp_mov_word_RM_SR(u8 X86EMU_UNUSED(op1))
  1897. {
  1898. int mod, rl, rh;
  1899. u16 *destreg, *srcreg;
  1900. uint destoffset;
  1901. u16 destval;
  1902. START_OF_INSTR();
  1903. DECODE_PRINTF("MOV\t");
  1904. FETCH_DECODE_MODRM(mod, rh, rl);
  1905. if (mod < 3) {
  1906. destoffset = decode_rmXX_address(mod, rl);
  1907. DECODE_PRINTF(",");
  1908. srcreg = decode_rm_seg_register(rh);
  1909. DECODE_PRINTF("\n");
  1910. TRACE_AND_STEP();
  1911. destval = *srcreg;
  1912. store_data_word(destoffset, destval);
  1913. } else { /* register to register */
  1914. destreg = DECODE_RM_WORD_REGISTER(rl);
  1915. DECODE_PRINTF(",");
  1916. srcreg = decode_rm_seg_register(rh);
  1917. DECODE_PRINTF("\n");
  1918. TRACE_AND_STEP();
  1919. *destreg = *srcreg;
  1920. }
  1921. DECODE_CLEAR_SEGOVR();
  1922. END_OF_INSTR();
  1923. }
  1924. /****************************************************************************
  1925. REMARKS:
  1926. Handles opcode 0x8d
  1927. ****************************************************************************/
  1928. void x86emuOp_lea_word_R_M(u8 X86EMU_UNUSED(op1))
  1929. {
  1930. int mod, rl, rh;
  1931. u16 *srcreg;
  1932. uint destoffset;
  1933. /*
  1934. * TODO: Need to handle address size prefix!
  1935. *
  1936. * lea eax,[eax+ebx*2] ??
  1937. */
  1938. START_OF_INSTR();
  1939. DECODE_PRINTF("LEA\t");
  1940. FETCH_DECODE_MODRM(mod, rh, rl);
  1941. if (mod < 3) {
  1942. srcreg = DECODE_RM_WORD_REGISTER(rh);
  1943. DECODE_PRINTF(",");
  1944. destoffset = decode_rmXX_address(mod, rl);
  1945. DECODE_PRINTF("\n");
  1946. TRACE_AND_STEP();
  1947. *srcreg = (u16)destoffset;
  1948. }
  1949. /* } else { undefined. Do nothing. } */
  1950. DECODE_CLEAR_SEGOVR();
  1951. END_OF_INSTR();
  1952. }
  1953. /****************************************************************************
  1954. REMARKS:
  1955. Handles opcode 0x8e
  1956. ****************************************************************************/
  1957. void x86emuOp_mov_word_SR_RM(u8 X86EMU_UNUSED(op1))
  1958. {
  1959. int mod, rl, rh;
  1960. u16 *destreg, *srcreg;
  1961. uint srcoffset;
  1962. u16 srcval;
  1963. START_OF_INSTR();
  1964. DECODE_PRINTF("MOV\t");
  1965. FETCH_DECODE_MODRM(mod, rh, rl);
  1966. if (mod < 3) {
  1967. destreg = decode_rm_seg_register(rh);
  1968. DECODE_PRINTF(",");
  1969. srcoffset = decode_rmXX_address(mod, rl);
  1970. srcval = fetch_data_word(srcoffset);
  1971. DECODE_PRINTF("\n");
  1972. TRACE_AND_STEP();
  1973. *destreg = srcval;
  1974. } else { /* register to register */
  1975. destreg = decode_rm_seg_register(rh);
  1976. DECODE_PRINTF(",");
  1977. srcreg = DECODE_RM_WORD_REGISTER(rl);
  1978. DECODE_PRINTF("\n");
  1979. TRACE_AND_STEP();
  1980. *destreg = *srcreg;
  1981. }
  1982. /*
  1983. * Clean up, and reset all the R_xSP pointers to the correct
  1984. * locations. This is about 3x too much overhead (doing all the
  1985. * segreg ptrs when only one is needed, but this instruction
  1986. * *cannot* be that common, and this isn't too much work anyway.
  1987. */
  1988. DECODE_CLEAR_SEGOVR();
  1989. END_OF_INSTR();
  1990. }
  1991. /****************************************************************************
  1992. REMARKS:
  1993. Handles opcode 0x8f
  1994. ****************************************************************************/
  1995. void x86emuOp_pop_RM(u8 X86EMU_UNUSED(op1))
  1996. {
  1997. int mod, rl, rh;
  1998. uint destoffset;
  1999. START_OF_INSTR();
  2000. DECODE_PRINTF("POP\t");
  2001. FETCH_DECODE_MODRM(mod, rh, rl);
  2002. if (rh != 0) {
  2003. DECODE_PRINTF("ILLEGAL DECODE OF OPCODE 8F\n");
  2004. HALT_SYS();
  2005. }
  2006. if (mod < 3) {
  2007. destoffset = decode_rmXX_address(mod, rl);
  2008. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2009. u32 destval;
  2010. DECODE_PRINTF("\n");
  2011. TRACE_AND_STEP();
  2012. destval = pop_long();
  2013. store_data_long(destoffset, destval);
  2014. } else {
  2015. u16 destval;
  2016. DECODE_PRINTF("\n");
  2017. TRACE_AND_STEP();
  2018. destval = pop_word();
  2019. store_data_word(destoffset, destval);
  2020. }
  2021. } else { /* register to register */
  2022. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2023. u32 *destreg;
  2024. destreg = DECODE_RM_LONG_REGISTER(rl);
  2025. DECODE_PRINTF("\n");
  2026. TRACE_AND_STEP();
  2027. *destreg = pop_long();
  2028. } else {
  2029. u16 *destreg;
  2030. destreg = DECODE_RM_WORD_REGISTER(rl);
  2031. DECODE_PRINTF("\n");
  2032. TRACE_AND_STEP();
  2033. *destreg = pop_word();
  2034. }
  2035. }
  2036. DECODE_CLEAR_SEGOVR();
  2037. END_OF_INSTR();
  2038. }
  2039. /****************************************************************************
  2040. REMARKS:
  2041. Handles opcode 0x90
  2042. ****************************************************************************/
  2043. void x86emuOp_nop(u8 X86EMU_UNUSED(op1))
  2044. {
  2045. START_OF_INSTR();
  2046. DECODE_PRINTF("NOP\n");
  2047. TRACE_AND_STEP();
  2048. DECODE_CLEAR_SEGOVR();
  2049. END_OF_INSTR();
  2050. }
  2051. /****************************************************************************
  2052. REMARKS:
  2053. Handles opcode 0x91-0x97
  2054. ****************************************************************************/
  2055. void x86emuOp_xchg_word_AX_register(u8 X86EMU_UNUSED(op1))
  2056. {
  2057. u32 tmp;
  2058. op1 &= 0x7;
  2059. START_OF_INSTR();
  2060. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2061. u32 *reg32;
  2062. DECODE_PRINTF("XCHG\tEAX,");
  2063. reg32 = DECODE_RM_LONG_REGISTER(op1);
  2064. DECODE_PRINTF("\n");
  2065. TRACE_AND_STEP();
  2066. tmp = M.x86.R_EAX;
  2067. M.x86.R_EAX = *reg32;
  2068. *reg32 = tmp;
  2069. } else {
  2070. u16 *reg16;
  2071. DECODE_PRINTF("XCHG\tAX,");
  2072. reg16 = DECODE_RM_WORD_REGISTER(op1);
  2073. DECODE_PRINTF("\n");
  2074. TRACE_AND_STEP();
  2075. tmp = M.x86.R_AX;
  2076. M.x86.R_EAX = *reg16;
  2077. *reg16 = (u16)tmp;
  2078. }
  2079. DECODE_CLEAR_SEGOVR();
  2080. END_OF_INSTR();
  2081. }
  2082. /****************************************************************************
  2083. REMARKS:
  2084. Handles opcode 0x98
  2085. ****************************************************************************/
  2086. void x86emuOp_cbw(u8 X86EMU_UNUSED(op1))
  2087. {
  2088. START_OF_INSTR();
  2089. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2090. DECODE_PRINTF("CWDE\n");
  2091. } else {
  2092. DECODE_PRINTF("CBW\n");
  2093. }
  2094. TRACE_AND_STEP();
  2095. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2096. if (M.x86.R_AX & 0x8000) {
  2097. M.x86.R_EAX |= 0xffff0000;
  2098. } else {
  2099. M.x86.R_EAX &= 0x0000ffff;
  2100. }
  2101. } else {
  2102. if (M.x86.R_AL & 0x80) {
  2103. M.x86.R_AH = 0xff;
  2104. } else {
  2105. M.x86.R_AH = 0x0;
  2106. }
  2107. }
  2108. DECODE_CLEAR_SEGOVR();
  2109. END_OF_INSTR();
  2110. }
  2111. /****************************************************************************
  2112. REMARKS:
  2113. Handles opcode 0x99
  2114. ****************************************************************************/
  2115. void x86emuOp_cwd(u8 X86EMU_UNUSED(op1))
  2116. {
  2117. START_OF_INSTR();
  2118. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2119. DECODE_PRINTF("CDQ\n");
  2120. } else {
  2121. DECODE_PRINTF("CWD\n");
  2122. }
  2123. DECODE_PRINTF("CWD\n");
  2124. TRACE_AND_STEP();
  2125. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2126. if (M.x86.R_EAX & 0x80000000) {
  2127. M.x86.R_EDX = 0xffffffff;
  2128. } else {
  2129. M.x86.R_EDX = 0x0;
  2130. }
  2131. } else {
  2132. if (M.x86.R_AX & 0x8000) {
  2133. M.x86.R_DX = 0xffff;
  2134. } else {
  2135. M.x86.R_DX = 0x0;
  2136. }
  2137. }
  2138. DECODE_CLEAR_SEGOVR();
  2139. END_OF_INSTR();
  2140. }
  2141. /****************************************************************************
  2142. REMARKS:
  2143. Handles opcode 0x9a
  2144. ****************************************************************************/
  2145. void x86emuOp_call_far_IMM(u8 X86EMU_UNUSED(op1))
  2146. {
  2147. u16 farseg, faroff;
  2148. START_OF_INSTR();
  2149. DECODE_PRINTF("CALL\t");
  2150. faroff = fetch_word_imm();
  2151. farseg = fetch_word_imm();
  2152. DECODE_PRINTF2("%04x:", farseg);
  2153. DECODE_PRINTF2("%04x\n", faroff);
  2154. CALL_TRACE(M.x86.saved_cs, M.x86.saved_ip, farseg, faroff, "FAR ");
  2155. /* XXX
  2156. *
  2157. * Hooked interrupt vectors calling into our "BIOS" will cause
  2158. * problems unless all intersegment stuff is checked for BIOS
  2159. * access. Check needed here. For moment, let it alone.
  2160. */
  2161. TRACE_AND_STEP();
  2162. push_word(M.x86.R_CS);
  2163. M.x86.R_CS = farseg;
  2164. push_word(M.x86.R_IP);
  2165. M.x86.R_IP = faroff;
  2166. DECODE_CLEAR_SEGOVR();
  2167. END_OF_INSTR();
  2168. }
  2169. /****************************************************************************
  2170. REMARKS:
  2171. Handles opcode 0x9b
  2172. ****************************************************************************/
  2173. void x86emuOp_wait(u8 X86EMU_UNUSED(op1))
  2174. {
  2175. START_OF_INSTR();
  2176. DECODE_PRINTF("WAIT");
  2177. TRACE_AND_STEP();
  2178. /* NADA. */
  2179. DECODE_CLEAR_SEGOVR();
  2180. END_OF_INSTR();
  2181. }
  2182. /****************************************************************************
  2183. REMARKS:
  2184. Handles opcode 0x9c
  2185. ****************************************************************************/
  2186. void x86emuOp_pushf_word(u8 X86EMU_UNUSED(op1))
  2187. {
  2188. u32 flags;
  2189. START_OF_INSTR();
  2190. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2191. DECODE_PRINTF("PUSHFD\n");
  2192. } else {
  2193. DECODE_PRINTF("PUSHF\n");
  2194. }
  2195. TRACE_AND_STEP();
  2196. /* clear out *all* bits not representing flags, and turn on real bits */
  2197. flags = (M.x86.R_EFLG & F_MSK) | F_ALWAYS_ON;
  2198. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2199. push_long(flags);
  2200. } else {
  2201. push_word((u16)flags);
  2202. }
  2203. DECODE_CLEAR_SEGOVR();
  2204. END_OF_INSTR();
  2205. }
  2206. /****************************************************************************
  2207. REMARKS:
  2208. Handles opcode 0x9d
  2209. ****************************************************************************/
  2210. void x86emuOp_popf_word(u8 X86EMU_UNUSED(op1))
  2211. {
  2212. START_OF_INSTR();
  2213. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2214. DECODE_PRINTF("POPFD\n");
  2215. } else {
  2216. DECODE_PRINTF("POPF\n");
  2217. }
  2218. TRACE_AND_STEP();
  2219. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2220. M.x86.R_EFLG = pop_long();
  2221. } else {
  2222. M.x86.R_FLG = pop_word();
  2223. }
  2224. DECODE_CLEAR_SEGOVR();
  2225. END_OF_INSTR();
  2226. }
  2227. /****************************************************************************
  2228. REMARKS:
  2229. Handles opcode 0x9e
  2230. ****************************************************************************/
  2231. void x86emuOp_sahf(u8 X86EMU_UNUSED(op1))
  2232. {
  2233. START_OF_INSTR();
  2234. DECODE_PRINTF("SAHF\n");
  2235. TRACE_AND_STEP();
  2236. /* clear the lower bits of the flag register */
  2237. M.x86.R_FLG &= 0xffffff00;
  2238. /* or in the AH register into the flags register */
  2239. M.x86.R_FLG |= M.x86.R_AH;
  2240. DECODE_CLEAR_SEGOVR();
  2241. END_OF_INSTR();
  2242. }
  2243. /****************************************************************************
  2244. REMARKS:
  2245. Handles opcode 0x9f
  2246. ****************************************************************************/
  2247. void x86emuOp_lahf(u8 X86EMU_UNUSED(op1))
  2248. {
  2249. START_OF_INSTR();
  2250. DECODE_PRINTF("LAHF\n");
  2251. TRACE_AND_STEP();
  2252. M.x86.R_AH = (u8)(M.x86.R_FLG & 0xff);
  2253. /*undocumented TC++ behavior??? Nope. It's documented, but
  2254. you have too look real hard to notice it. */
  2255. M.x86.R_AH |= 0x2;
  2256. DECODE_CLEAR_SEGOVR();
  2257. END_OF_INSTR();
  2258. }
  2259. /****************************************************************************
  2260. REMARKS:
  2261. Handles opcode 0xa0
  2262. ****************************************************************************/
  2263. void x86emuOp_mov_AL_M_IMM(u8 X86EMU_UNUSED(op1))
  2264. {
  2265. u16 offset;
  2266. START_OF_INSTR();
  2267. DECODE_PRINTF("MOV\tAL,");
  2268. offset = fetch_word_imm();
  2269. DECODE_PRINTF2("[%04x]\n", offset);
  2270. TRACE_AND_STEP();
  2271. M.x86.R_AL = fetch_data_byte(offset);
  2272. DECODE_CLEAR_SEGOVR();
  2273. END_OF_INSTR();
  2274. }
  2275. /****************************************************************************
  2276. REMARKS:
  2277. Handles opcode 0xa1
  2278. ****************************************************************************/
  2279. void x86emuOp_mov_AX_M_IMM(u8 X86EMU_UNUSED(op1))
  2280. {
  2281. u16 offset;
  2282. START_OF_INSTR();
  2283. offset = fetch_word_imm();
  2284. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2285. DECODE_PRINTF2("MOV\tEAX,[%04x]\n", offset);
  2286. } else {
  2287. DECODE_PRINTF2("MOV\tAX,[%04x]\n", offset);
  2288. }
  2289. TRACE_AND_STEP();
  2290. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2291. M.x86.R_EAX = fetch_data_long(offset);
  2292. } else {
  2293. M.x86.R_AX = fetch_data_word(offset);
  2294. }
  2295. DECODE_CLEAR_SEGOVR();
  2296. END_OF_INSTR();
  2297. }
  2298. /****************************************************************************
  2299. REMARKS:
  2300. Handles opcode 0xa2
  2301. ****************************************************************************/
  2302. void x86emuOp_mov_M_AL_IMM(u8 X86EMU_UNUSED(op1))
  2303. {
  2304. u16 offset;
  2305. START_OF_INSTR();
  2306. DECODE_PRINTF("MOV\t");
  2307. offset = fetch_word_imm();
  2308. DECODE_PRINTF2("[%04x],AL\n", offset);
  2309. TRACE_AND_STEP();
  2310. store_data_byte(offset, M.x86.R_AL);
  2311. DECODE_CLEAR_SEGOVR();
  2312. END_OF_INSTR();
  2313. }
  2314. /****************************************************************************
  2315. REMARKS:
  2316. Handles opcode 0xa3
  2317. ****************************************************************************/
  2318. void x86emuOp_mov_M_AX_IMM(u8 X86EMU_UNUSED(op1))
  2319. {
  2320. u16 offset;
  2321. START_OF_INSTR();
  2322. offset = fetch_word_imm();
  2323. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2324. DECODE_PRINTF2("MOV\t[%04x],EAX\n", offset);
  2325. } else {
  2326. DECODE_PRINTF2("MOV\t[%04x],AX\n", offset);
  2327. }
  2328. TRACE_AND_STEP();
  2329. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2330. store_data_long(offset, M.x86.R_EAX);
  2331. } else {
  2332. store_data_word(offset, M.x86.R_AX);
  2333. }
  2334. DECODE_CLEAR_SEGOVR();
  2335. END_OF_INSTR();
  2336. }
  2337. /****************************************************************************
  2338. REMARKS:
  2339. Handles opcode 0xa4
  2340. ****************************************************************************/
  2341. void x86emuOp_movs_byte(u8 X86EMU_UNUSED(op1))
  2342. {
  2343. u8 val;
  2344. u32 count;
  2345. int inc;
  2346. START_OF_INSTR();
  2347. DECODE_PRINTF("MOVS\tBYTE\n");
  2348. if (ACCESS_FLAG(F_DF)) /* down */
  2349. inc = -1;
  2350. else
  2351. inc = 1;
  2352. TRACE_AND_STEP();
  2353. count = 1;
  2354. if (M.x86.mode & (SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE)) {
  2355. /* dont care whether REPE or REPNE */
  2356. /* move them until CX is ZERO. */
  2357. count = M.x86.R_CX;
  2358. M.x86.R_CX = 0;
  2359. M.x86.mode &= ~(SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE);
  2360. }
  2361. while (count--) {
  2362. val = fetch_data_byte(M.x86.R_SI);
  2363. store_data_byte_abs(M.x86.R_ES, M.x86.R_DI, val);
  2364. M.x86.R_SI += inc;
  2365. M.x86.R_DI += inc;
  2366. }
  2367. DECODE_CLEAR_SEGOVR();
  2368. END_OF_INSTR();
  2369. }
  2370. /****************************************************************************
  2371. REMARKS:
  2372. Handles opcode 0xa5
  2373. ****************************************************************************/
  2374. void x86emuOp_movs_word(u8 X86EMU_UNUSED(op1))
  2375. {
  2376. u32 val;
  2377. int inc;
  2378. u32 count;
  2379. START_OF_INSTR();
  2380. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2381. DECODE_PRINTF("MOVS\tDWORD\n");
  2382. if (ACCESS_FLAG(F_DF)) /* down */
  2383. inc = -4;
  2384. else
  2385. inc = 4;
  2386. } else {
  2387. DECODE_PRINTF("MOVS\tWORD\n");
  2388. if (ACCESS_FLAG(F_DF)) /* down */
  2389. inc = -2;
  2390. else
  2391. inc = 2;
  2392. }
  2393. TRACE_AND_STEP();
  2394. count = 1;
  2395. if (M.x86.mode & (SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE)) {
  2396. /* dont care whether REPE or REPNE */
  2397. /* move them until CX is ZERO. */
  2398. count = M.x86.R_CX;
  2399. M.x86.R_CX = 0;
  2400. M.x86.mode &= ~(SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE);
  2401. }
  2402. while (count--) {
  2403. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2404. val = fetch_data_long(M.x86.R_SI);
  2405. store_data_long_abs(M.x86.R_ES, M.x86.R_DI, val);
  2406. } else {
  2407. val = fetch_data_word(M.x86.R_SI);
  2408. store_data_word_abs(M.x86.R_ES, M.x86.R_DI, (u16)val);
  2409. }
  2410. M.x86.R_SI += inc;
  2411. M.x86.R_DI += inc;
  2412. }
  2413. DECODE_CLEAR_SEGOVR();
  2414. END_OF_INSTR();
  2415. }
  2416. /****************************************************************************
  2417. REMARKS:
  2418. Handles opcode 0xa6
  2419. ****************************************************************************/
  2420. void x86emuOp_cmps_byte(u8 X86EMU_UNUSED(op1))
  2421. {
  2422. s8 val1, val2;
  2423. int inc;
  2424. START_OF_INSTR();
  2425. DECODE_PRINTF("CMPS\tBYTE\n");
  2426. TRACE_AND_STEP();
  2427. if (ACCESS_FLAG(F_DF)) /* down */
  2428. inc = -1;
  2429. else
  2430. inc = 1;
  2431. if (M.x86.mode & (SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE)) {
  2432. /* REPE */
  2433. /* move them until CX is ZERO. */
  2434. while (M.x86.R_CX != 0) {
  2435. val1 = fetch_data_byte(M.x86.R_SI);
  2436. val2 = fetch_data_byte_abs(M.x86.R_ES, M.x86.R_DI);
  2437. cmp_byte(val1, val2);
  2438. M.x86.R_CX -= 1;
  2439. M.x86.R_SI += inc;
  2440. M.x86.R_DI += inc;
  2441. if ( (M.x86.mode & SYSMODE_PREFIX_REPE) && (ACCESS_FLAG(F_ZF) == 0) ) break;
  2442. if ( (M.x86.mode & SYSMODE_PREFIX_REPNE) && ACCESS_FLAG(F_ZF) ) break;
  2443. }
  2444. M.x86.mode &= ~(SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE);
  2445. } else {
  2446. val1 = fetch_data_byte(M.x86.R_SI);
  2447. val2 = fetch_data_byte_abs(M.x86.R_ES, M.x86.R_DI);
  2448. cmp_byte(val1, val2);
  2449. M.x86.R_SI += inc;
  2450. M.x86.R_DI += inc;
  2451. }
  2452. DECODE_CLEAR_SEGOVR();
  2453. END_OF_INSTR();
  2454. }
  2455. /****************************************************************************
  2456. REMARKS:
  2457. Handles opcode 0xa7
  2458. ****************************************************************************/
  2459. void x86emuOp_cmps_word(u8 X86EMU_UNUSED(op1))
  2460. {
  2461. u32 val1,val2;
  2462. int inc;
  2463. START_OF_INSTR();
  2464. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2465. DECODE_PRINTF("CMPS\tDWORD\n");
  2466. inc = 4;
  2467. } else {
  2468. DECODE_PRINTF("CMPS\tWORD\n");
  2469. inc = 2;
  2470. }
  2471. if (ACCESS_FLAG(F_DF)) /* down */
  2472. inc = -inc;
  2473. TRACE_AND_STEP();
  2474. if (M.x86.mode & (SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE)) {
  2475. /* REPE */
  2476. /* move them until CX is ZERO. */
  2477. while (M.x86.R_CX != 0) {
  2478. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2479. val1 = fetch_data_long(M.x86.R_SI);
  2480. val2 = fetch_data_long_abs(M.x86.R_ES, M.x86.R_DI);
  2481. cmp_long(val1, val2);
  2482. } else {
  2483. val1 = fetch_data_word(M.x86.R_SI);
  2484. val2 = fetch_data_word_abs(M.x86.R_ES, M.x86.R_DI);
  2485. cmp_word((u16)val1, (u16)val2);
  2486. }
  2487. M.x86.R_CX -= 1;
  2488. M.x86.R_SI += inc;
  2489. M.x86.R_DI += inc;
  2490. if ( (M.x86.mode & SYSMODE_PREFIX_REPE) && ACCESS_FLAG(F_ZF) == 0 ) break;
  2491. if ( (M.x86.mode & SYSMODE_PREFIX_REPNE) && ACCESS_FLAG(F_ZF) ) break;
  2492. }
  2493. M.x86.mode &= ~(SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE);
  2494. } else {
  2495. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2496. val1 = fetch_data_long(M.x86.R_SI);
  2497. val2 = fetch_data_long_abs(M.x86.R_ES, M.x86.R_DI);
  2498. cmp_long(val1, val2);
  2499. } else {
  2500. val1 = fetch_data_word(M.x86.R_SI);
  2501. val2 = fetch_data_word_abs(M.x86.R_ES, M.x86.R_DI);
  2502. cmp_word((u16)val1, (u16)val2);
  2503. }
  2504. M.x86.R_SI += inc;
  2505. M.x86.R_DI += inc;
  2506. }
  2507. DECODE_CLEAR_SEGOVR();
  2508. END_OF_INSTR();
  2509. }
  2510. /****************************************************************************
  2511. REMARKS:
  2512. Handles opcode 0xa8
  2513. ****************************************************************************/
  2514. void x86emuOp_test_AL_IMM(u8 X86EMU_UNUSED(op1))
  2515. {
  2516. int imm;
  2517. START_OF_INSTR();
  2518. DECODE_PRINTF("TEST\tAL,");
  2519. imm = fetch_byte_imm();
  2520. DECODE_PRINTF2("%04x\n", imm);
  2521. TRACE_AND_STEP();
  2522. test_byte(M.x86.R_AL, (u8)imm);
  2523. DECODE_CLEAR_SEGOVR();
  2524. END_OF_INSTR();
  2525. }
  2526. /****************************************************************************
  2527. REMARKS:
  2528. Handles opcode 0xa9
  2529. ****************************************************************************/
  2530. void x86emuOp_test_AX_IMM(u8 X86EMU_UNUSED(op1))
  2531. {
  2532. u32 srcval;
  2533. START_OF_INSTR();
  2534. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2535. DECODE_PRINTF("TEST\tEAX,");
  2536. srcval = fetch_long_imm();
  2537. } else {
  2538. DECODE_PRINTF("TEST\tAX,");
  2539. srcval = fetch_word_imm();
  2540. }
  2541. DECODE_PRINTF2("%x\n", srcval);
  2542. TRACE_AND_STEP();
  2543. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2544. test_long(M.x86.R_EAX, srcval);
  2545. } else {
  2546. test_word(M.x86.R_AX, (u16)srcval);
  2547. }
  2548. DECODE_CLEAR_SEGOVR();
  2549. END_OF_INSTR();
  2550. }
  2551. /****************************************************************************
  2552. REMARKS:
  2553. Handles opcode 0xaa
  2554. ****************************************************************************/
  2555. void x86emuOp_stos_byte(u8 X86EMU_UNUSED(op1))
  2556. {
  2557. int inc;
  2558. START_OF_INSTR();
  2559. DECODE_PRINTF("STOS\tBYTE\n");
  2560. if (ACCESS_FLAG(F_DF)) /* down */
  2561. inc = -1;
  2562. else
  2563. inc = 1;
  2564. TRACE_AND_STEP();
  2565. if (M.x86.mode & (SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE)) {
  2566. /* dont care whether REPE or REPNE */
  2567. /* move them until CX is ZERO. */
  2568. while (M.x86.R_CX != 0) {
  2569. store_data_byte_abs(M.x86.R_ES, M.x86.R_DI, M.x86.R_AL);
  2570. M.x86.R_CX -= 1;
  2571. M.x86.R_DI += inc;
  2572. }
  2573. M.x86.mode &= ~(SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE);
  2574. } else {
  2575. store_data_byte_abs(M.x86.R_ES, M.x86.R_DI, M.x86.R_AL);
  2576. M.x86.R_DI += inc;
  2577. }
  2578. DECODE_CLEAR_SEGOVR();
  2579. END_OF_INSTR();
  2580. }
  2581. /****************************************************************************
  2582. REMARKS:
  2583. Handles opcode 0xab
  2584. ****************************************************************************/
  2585. void x86emuOp_stos_word(u8 X86EMU_UNUSED(op1))
  2586. {
  2587. int inc;
  2588. u32 count;
  2589. START_OF_INSTR();
  2590. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2591. DECODE_PRINTF("STOS\tDWORD\n");
  2592. if (ACCESS_FLAG(F_DF)) /* down */
  2593. inc = -4;
  2594. else
  2595. inc = 4;
  2596. } else {
  2597. DECODE_PRINTF("STOS\tWORD\n");
  2598. if (ACCESS_FLAG(F_DF)) /* down */
  2599. inc = -2;
  2600. else
  2601. inc = 2;
  2602. }
  2603. TRACE_AND_STEP();
  2604. count = 1;
  2605. if (M.x86.mode & (SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE)) {
  2606. /* dont care whether REPE or REPNE */
  2607. /* move them until CX is ZERO. */
  2608. count = M.x86.R_CX;
  2609. M.x86.R_CX = 0;
  2610. M.x86.mode &= ~(SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE);
  2611. }
  2612. while (count--) {
  2613. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2614. store_data_long_abs(M.x86.R_ES, M.x86.R_DI, M.x86.R_EAX);
  2615. } else {
  2616. store_data_word_abs(M.x86.R_ES, M.x86.R_DI, M.x86.R_AX);
  2617. }
  2618. M.x86.R_DI += inc;
  2619. }
  2620. DECODE_CLEAR_SEGOVR();
  2621. END_OF_INSTR();
  2622. }
  2623. /****************************************************************************
  2624. REMARKS:
  2625. Handles opcode 0xac
  2626. ****************************************************************************/
  2627. void x86emuOp_lods_byte(u8 X86EMU_UNUSED(op1))
  2628. {
  2629. int inc;
  2630. START_OF_INSTR();
  2631. DECODE_PRINTF("LODS\tBYTE\n");
  2632. TRACE_AND_STEP();
  2633. if (ACCESS_FLAG(F_DF)) /* down */
  2634. inc = -1;
  2635. else
  2636. inc = 1;
  2637. if (M.x86.mode & (SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE)) {
  2638. /* dont care whether REPE or REPNE */
  2639. /* move them until CX is ZERO. */
  2640. while (M.x86.R_CX != 0) {
  2641. M.x86.R_AL = fetch_data_byte(M.x86.R_SI);
  2642. M.x86.R_CX -= 1;
  2643. M.x86.R_SI += inc;
  2644. }
  2645. M.x86.mode &= ~(SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE);
  2646. } else {
  2647. M.x86.R_AL = fetch_data_byte(M.x86.R_SI);
  2648. M.x86.R_SI += inc;
  2649. }
  2650. DECODE_CLEAR_SEGOVR();
  2651. END_OF_INSTR();
  2652. }
  2653. /****************************************************************************
  2654. REMARKS:
  2655. Handles opcode 0xad
  2656. ****************************************************************************/
  2657. void x86emuOp_lods_word(u8 X86EMU_UNUSED(op1))
  2658. {
  2659. int inc;
  2660. u32 count;
  2661. START_OF_INSTR();
  2662. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2663. DECODE_PRINTF("LODS\tDWORD\n");
  2664. if (ACCESS_FLAG(F_DF)) /* down */
  2665. inc = -4;
  2666. else
  2667. inc = 4;
  2668. } else {
  2669. DECODE_PRINTF("LODS\tWORD\n");
  2670. if (ACCESS_FLAG(F_DF)) /* down */
  2671. inc = -2;
  2672. else
  2673. inc = 2;
  2674. }
  2675. TRACE_AND_STEP();
  2676. count = 1;
  2677. if (M.x86.mode & (SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE)) {
  2678. /* dont care whether REPE or REPNE */
  2679. /* move them until CX is ZERO. */
  2680. count = M.x86.R_CX;
  2681. M.x86.R_CX = 0;
  2682. M.x86.mode &= ~(SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE);
  2683. }
  2684. while (count--) {
  2685. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2686. M.x86.R_EAX = fetch_data_long(M.x86.R_SI);
  2687. } else {
  2688. M.x86.R_AX = fetch_data_word(M.x86.R_SI);
  2689. }
  2690. M.x86.R_SI += inc;
  2691. }
  2692. DECODE_CLEAR_SEGOVR();
  2693. END_OF_INSTR();
  2694. }
  2695. /****************************************************************************
  2696. REMARKS:
  2697. Handles opcode 0xae
  2698. ****************************************************************************/
  2699. void x86emuOp_scas_byte(u8 X86EMU_UNUSED(op1))
  2700. {
  2701. s8 val2;
  2702. int inc;
  2703. START_OF_INSTR();
  2704. DECODE_PRINTF("SCAS\tBYTE\n");
  2705. TRACE_AND_STEP();
  2706. if (ACCESS_FLAG(F_DF)) /* down */
  2707. inc = -1;
  2708. else
  2709. inc = 1;
  2710. if (M.x86.mode & SYSMODE_PREFIX_REPE) {
  2711. /* REPE */
  2712. /* move them until CX is ZERO. */
  2713. while (M.x86.R_CX != 0) {
  2714. val2 = fetch_data_byte_abs(M.x86.R_ES, M.x86.R_DI);
  2715. cmp_byte(M.x86.R_AL, val2);
  2716. M.x86.R_CX -= 1;
  2717. M.x86.R_DI += inc;
  2718. if (ACCESS_FLAG(F_ZF) == 0)
  2719. break;
  2720. }
  2721. M.x86.mode &= ~SYSMODE_PREFIX_REPE;
  2722. } else if (M.x86.mode & SYSMODE_PREFIX_REPNE) {
  2723. /* REPNE */
  2724. /* move them until CX is ZERO. */
  2725. while (M.x86.R_CX != 0) {
  2726. val2 = fetch_data_byte_abs(M.x86.R_ES, M.x86.R_DI);
  2727. cmp_byte(M.x86.R_AL, val2);
  2728. M.x86.R_CX -= 1;
  2729. M.x86.R_DI += inc;
  2730. if (ACCESS_FLAG(F_ZF))
  2731. break; /* zero flag set means equal */
  2732. }
  2733. M.x86.mode &= ~SYSMODE_PREFIX_REPNE;
  2734. } else {
  2735. val2 = fetch_data_byte_abs(M.x86.R_ES, M.x86.R_DI);
  2736. cmp_byte(M.x86.R_AL, val2);
  2737. M.x86.R_DI += inc;
  2738. }
  2739. DECODE_CLEAR_SEGOVR();
  2740. END_OF_INSTR();
  2741. }
  2742. /****************************************************************************
  2743. REMARKS:
  2744. Handles opcode 0xaf
  2745. ****************************************************************************/
  2746. void x86emuOp_scas_word(u8 X86EMU_UNUSED(op1))
  2747. {
  2748. int inc;
  2749. u32 val;
  2750. START_OF_INSTR();
  2751. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2752. DECODE_PRINTF("SCAS\tDWORD\n");
  2753. if (ACCESS_FLAG(F_DF)) /* down */
  2754. inc = -4;
  2755. else
  2756. inc = 4;
  2757. } else {
  2758. DECODE_PRINTF("SCAS\tWORD\n");
  2759. if (ACCESS_FLAG(F_DF)) /* down */
  2760. inc = -2;
  2761. else
  2762. inc = 2;
  2763. }
  2764. TRACE_AND_STEP();
  2765. if (M.x86.mode & SYSMODE_PREFIX_REPE) {
  2766. /* REPE */
  2767. /* move them until CX is ZERO. */
  2768. while (M.x86.R_CX != 0) {
  2769. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2770. val = fetch_data_long_abs(M.x86.R_ES, M.x86.R_DI);
  2771. cmp_long(M.x86.R_EAX, val);
  2772. } else {
  2773. val = fetch_data_word_abs(M.x86.R_ES, M.x86.R_DI);
  2774. cmp_word(M.x86.R_AX, (u16)val);
  2775. }
  2776. M.x86.R_CX -= 1;
  2777. M.x86.R_DI += inc;
  2778. if (ACCESS_FLAG(F_ZF) == 0)
  2779. break;
  2780. }
  2781. M.x86.mode &= ~SYSMODE_PREFIX_REPE;
  2782. } else if (M.x86.mode & SYSMODE_PREFIX_REPNE) {
  2783. /* REPNE */
  2784. /* move them until CX is ZERO. */
  2785. while (M.x86.R_CX != 0) {
  2786. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2787. val = fetch_data_long_abs(M.x86.R_ES, M.x86.R_DI);
  2788. cmp_long(M.x86.R_EAX, val);
  2789. } else {
  2790. val = fetch_data_word_abs(M.x86.R_ES, M.x86.R_DI);
  2791. cmp_word(M.x86.R_AX, (u16)val);
  2792. }
  2793. M.x86.R_CX -= 1;
  2794. M.x86.R_DI += inc;
  2795. if (ACCESS_FLAG(F_ZF))
  2796. break; /* zero flag set means equal */
  2797. }
  2798. M.x86.mode &= ~SYSMODE_PREFIX_REPNE;
  2799. } else {
  2800. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2801. val = fetch_data_long_abs(M.x86.R_ES, M.x86.R_DI);
  2802. cmp_long(M.x86.R_EAX, val);
  2803. } else {
  2804. val = fetch_data_word_abs(M.x86.R_ES, M.x86.R_DI);
  2805. cmp_word(M.x86.R_AX, (u16)val);
  2806. }
  2807. M.x86.R_DI += inc;
  2808. }
  2809. DECODE_CLEAR_SEGOVR();
  2810. END_OF_INSTR();
  2811. }
  2812. /****************************************************************************
  2813. REMARKS:
  2814. Handles opcode 0xb0 - 0xb7
  2815. ****************************************************************************/
  2816. void x86emuOp_mov_byte_register_IMM(u8 op1)
  2817. {
  2818. u8 imm, *ptr;
  2819. START_OF_INSTR();
  2820. DECODE_PRINTF("MOV\t");
  2821. ptr = DECODE_RM_BYTE_REGISTER(op1 & 0x7);
  2822. DECODE_PRINTF(",");
  2823. imm = fetch_byte_imm();
  2824. DECODE_PRINTF2("%x\n", imm);
  2825. TRACE_AND_STEP();
  2826. *ptr = imm;
  2827. DECODE_CLEAR_SEGOVR();
  2828. END_OF_INSTR();
  2829. }
  2830. /****************************************************************************
  2831. REMARKS:
  2832. Handles opcode 0xb8 - 0xbf
  2833. ****************************************************************************/
  2834. void x86emuOp_mov_word_register_IMM(u8 X86EMU_UNUSED(op1))
  2835. {
  2836. u32 srcval;
  2837. op1 &= 0x7;
  2838. START_OF_INSTR();
  2839. DECODE_PRINTF("MOV\t");
  2840. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2841. u32 *reg32;
  2842. reg32 = DECODE_RM_LONG_REGISTER(op1);
  2843. srcval = fetch_long_imm();
  2844. DECODE_PRINTF2(",%x\n", srcval);
  2845. TRACE_AND_STEP();
  2846. *reg32 = srcval;
  2847. } else {
  2848. u16 *reg16;
  2849. reg16 = DECODE_RM_WORD_REGISTER(op1);
  2850. srcval = fetch_word_imm();
  2851. DECODE_PRINTF2(",%x\n", srcval);
  2852. TRACE_AND_STEP();
  2853. *reg16 = (u16)srcval;
  2854. }
  2855. DECODE_CLEAR_SEGOVR();
  2856. END_OF_INSTR();
  2857. }
  2858. /****************************************************************************
  2859. REMARKS:
  2860. Handles opcode 0xc0
  2861. ****************************************************************************/
  2862. void x86emuOp_opcC0_byte_RM_MEM(u8 X86EMU_UNUSED(op1))
  2863. {
  2864. int mod, rl, rh;
  2865. u8 *destreg;
  2866. uint destoffset;
  2867. u8 destval;
  2868. u8 amt;
  2869. /*
  2870. * Yet another weirdo special case instruction format. Part of
  2871. * the opcode held below in "RH". Doubly nested case would
  2872. * result, except that the decoded instruction
  2873. */
  2874. START_OF_INSTR();
  2875. FETCH_DECODE_MODRM(mod, rh, rl);
  2876. #ifdef DEBUG
  2877. if (DEBUG_DECODE()) {
  2878. /* XXX DECODE_PRINTF may be changed to something more
  2879. general, so that it is important to leave the strings
  2880. in the same format, even though the result is that the
  2881. above test is done twice. */
  2882. switch (rh) {
  2883. case 0:
  2884. DECODE_PRINTF("ROL\t");
  2885. break;
  2886. case 1:
  2887. DECODE_PRINTF("ROR\t");
  2888. break;
  2889. case 2:
  2890. DECODE_PRINTF("RCL\t");
  2891. break;
  2892. case 3:
  2893. DECODE_PRINTF("RCR\t");
  2894. break;
  2895. case 4:
  2896. DECODE_PRINTF("SHL\t");
  2897. break;
  2898. case 5:
  2899. DECODE_PRINTF("SHR\t");
  2900. break;
  2901. case 6:
  2902. DECODE_PRINTF("SAL\t");
  2903. break;
  2904. case 7:
  2905. DECODE_PRINTF("SAR\t");
  2906. break;
  2907. }
  2908. }
  2909. #endif
  2910. /* know operation, decode the mod byte to find the addressing
  2911. mode. */
  2912. if (mod < 3) {
  2913. DECODE_PRINTF("BYTE PTR ");
  2914. destoffset = decode_rmXX_address(mod, rl);
  2915. amt = fetch_byte_imm();
  2916. DECODE_PRINTF2(",%x\n", amt);
  2917. destval = fetch_data_byte(destoffset);
  2918. TRACE_AND_STEP();
  2919. destval = (*opcD0_byte_operation[rh]) (destval, amt);
  2920. store_data_byte(destoffset, destval);
  2921. } else { /* register to register */
  2922. destreg = DECODE_RM_BYTE_REGISTER(rl);
  2923. amt = fetch_byte_imm();
  2924. DECODE_PRINTF2(",%x\n", amt);
  2925. TRACE_AND_STEP();
  2926. destval = (*opcD0_byte_operation[rh]) (*destreg, amt);
  2927. *destreg = destval;
  2928. }
  2929. DECODE_CLEAR_SEGOVR();
  2930. END_OF_INSTR();
  2931. }
  2932. /****************************************************************************
  2933. REMARKS:
  2934. Handles opcode 0xc1
  2935. ****************************************************************************/
  2936. void x86emuOp_opcC1_word_RM_MEM(u8 X86EMU_UNUSED(op1))
  2937. {
  2938. int mod, rl, rh;
  2939. uint destoffset;
  2940. u8 amt;
  2941. /*
  2942. * Yet another weirdo special case instruction format. Part of
  2943. * the opcode held below in "RH". Doubly nested case would
  2944. * result, except that the decoded instruction
  2945. */
  2946. START_OF_INSTR();
  2947. FETCH_DECODE_MODRM(mod, rh, rl);
  2948. #ifdef DEBUG
  2949. if (DEBUG_DECODE()) {
  2950. /* XXX DECODE_PRINTF may be changed to something more
  2951. general, so that it is important to leave the strings
  2952. in the same format, even though the result is that the
  2953. above test is done twice. */
  2954. switch (rh) {
  2955. case 0:
  2956. DECODE_PRINTF("ROL\t");
  2957. break;
  2958. case 1:
  2959. DECODE_PRINTF("ROR\t");
  2960. break;
  2961. case 2:
  2962. DECODE_PRINTF("RCL\t");
  2963. break;
  2964. case 3:
  2965. DECODE_PRINTF("RCR\t");
  2966. break;
  2967. case 4:
  2968. DECODE_PRINTF("SHL\t");
  2969. break;
  2970. case 5:
  2971. DECODE_PRINTF("SHR\t");
  2972. break;
  2973. case 6:
  2974. DECODE_PRINTF("SAL\t");
  2975. break;
  2976. case 7:
  2977. DECODE_PRINTF("SAR\t");
  2978. break;
  2979. }
  2980. }
  2981. #endif
  2982. /* know operation, decode the mod byte to find the addressing
  2983. mode. */
  2984. if (mod < 3) {
  2985. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  2986. u32 destval;
  2987. DECODE_PRINTF("DWORD PTR ");
  2988. destoffset = decode_rmXX_address(mod, rl);
  2989. amt = fetch_byte_imm();
  2990. DECODE_PRINTF2(",%x\n", amt);
  2991. destval = fetch_data_long(destoffset);
  2992. TRACE_AND_STEP();
  2993. destval = (*opcD1_long_operation[rh]) (destval, amt);
  2994. store_data_long(destoffset, destval);
  2995. } else {
  2996. u16 destval;
  2997. DECODE_PRINTF("WORD PTR ");
  2998. destoffset = decode_rmXX_address(mod, rl);
  2999. amt = fetch_byte_imm();
  3000. DECODE_PRINTF2(",%x\n", amt);
  3001. destval = fetch_data_word(destoffset);
  3002. TRACE_AND_STEP();
  3003. destval = (*opcD1_word_operation[rh]) (destval, amt);
  3004. store_data_word(destoffset, destval);
  3005. }
  3006. } else { /* register to register */
  3007. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  3008. u32 *destreg;
  3009. destreg = DECODE_RM_LONG_REGISTER(rl);
  3010. amt = fetch_byte_imm();
  3011. DECODE_PRINTF2(",%x\n", amt);
  3012. TRACE_AND_STEP();
  3013. *destreg = (*opcD1_long_operation[rh]) (*destreg, amt);
  3014. } else {
  3015. u16 *destreg;
  3016. destreg = DECODE_RM_WORD_REGISTER(rl);
  3017. amt = fetch_byte_imm();
  3018. DECODE_PRINTF2(",%x\n", amt);
  3019. TRACE_AND_STEP();
  3020. *destreg = (*opcD1_word_operation[rh]) (*destreg, amt);
  3021. }
  3022. }
  3023. DECODE_CLEAR_SEGOVR();
  3024. END_OF_INSTR();
  3025. }
  3026. /****************************************************************************
  3027. REMARKS:
  3028. Handles opcode 0xc2
  3029. ****************************************************************************/
  3030. void x86emuOp_ret_near_IMM(u8 X86EMU_UNUSED(op1))
  3031. {
  3032. u16 imm;
  3033. START_OF_INSTR();
  3034. DECODE_PRINTF("RET\t");
  3035. imm = fetch_word_imm();
  3036. DECODE_PRINTF2("%x\n", imm);
  3037. RETURN_TRACE("RET",M.x86.saved_cs,M.x86.saved_ip);
  3038. TRACE_AND_STEP();
  3039. M.x86.R_IP = pop_word();
  3040. M.x86.R_SP += imm;
  3041. DECODE_CLEAR_SEGOVR();
  3042. END_OF_INSTR();
  3043. }
  3044. /****************************************************************************
  3045. REMARKS:
  3046. Handles opcode 0xc3
  3047. ****************************************************************************/
  3048. void x86emuOp_ret_near(u8 X86EMU_UNUSED(op1))
  3049. {
  3050. START_OF_INSTR();
  3051. DECODE_PRINTF("RET\n");
  3052. RETURN_TRACE("RET",M.x86.saved_cs,M.x86.saved_ip);
  3053. TRACE_AND_STEP();
  3054. M.x86.R_IP = pop_word();
  3055. DECODE_CLEAR_SEGOVR();
  3056. END_OF_INSTR();
  3057. }
  3058. /****************************************************************************
  3059. REMARKS:
  3060. Handles opcode 0xc4
  3061. ****************************************************************************/
  3062. void x86emuOp_les_R_IMM(u8 X86EMU_UNUSED(op1))
  3063. {
  3064. int mod, rh, rl;
  3065. u16 *dstreg;
  3066. uint srcoffset;
  3067. START_OF_INSTR();
  3068. DECODE_PRINTF("LES\t");
  3069. FETCH_DECODE_MODRM(mod, rh, rl);
  3070. if (mod < 3) {
  3071. dstreg = DECODE_RM_WORD_REGISTER(rh);
  3072. DECODE_PRINTF(",");
  3073. srcoffset = decode_rmXX_address(mod, rl);
  3074. DECODE_PRINTF("\n");
  3075. TRACE_AND_STEP();
  3076. *dstreg = fetch_data_word(srcoffset);
  3077. M.x86.R_ES = fetch_data_word(srcoffset + 2);
  3078. }
  3079. /* else UNDEFINED! register to register */
  3080. DECODE_CLEAR_SEGOVR();
  3081. END_OF_INSTR();
  3082. }
  3083. /****************************************************************************
  3084. REMARKS:
  3085. Handles opcode 0xc5
  3086. ****************************************************************************/
  3087. void x86emuOp_lds_R_IMM(u8 X86EMU_UNUSED(op1))
  3088. {
  3089. int mod, rh, rl;
  3090. u16 *dstreg;
  3091. uint srcoffset;
  3092. START_OF_INSTR();
  3093. DECODE_PRINTF("LDS\t");
  3094. FETCH_DECODE_MODRM(mod, rh, rl);
  3095. if (mod < 3) {
  3096. dstreg = DECODE_RM_WORD_REGISTER(rh);
  3097. DECODE_PRINTF(",");
  3098. srcoffset = decode_rmXX_address(mod, rl);
  3099. DECODE_PRINTF("\n");
  3100. TRACE_AND_STEP();
  3101. *dstreg = fetch_data_word(srcoffset);
  3102. M.x86.R_DS = fetch_data_word(srcoffset + 2);
  3103. }
  3104. /* else UNDEFINED! */
  3105. DECODE_CLEAR_SEGOVR();
  3106. END_OF_INSTR();
  3107. }
  3108. /****************************************************************************
  3109. REMARKS:
  3110. Handles opcode 0xc6
  3111. ****************************************************************************/
  3112. void x86emuOp_mov_byte_RM_IMM(u8 X86EMU_UNUSED(op1))
  3113. {
  3114. int mod, rl, rh;
  3115. u8 *destreg;
  3116. uint destoffset;
  3117. u8 imm;
  3118. START_OF_INSTR();
  3119. DECODE_PRINTF("MOV\t");
  3120. FETCH_DECODE_MODRM(mod, rh, rl);
  3121. if (rh != 0) {
  3122. DECODE_PRINTF("ILLEGAL DECODE OF OPCODE c6\n");
  3123. HALT_SYS();
  3124. }
  3125. if (mod < 3) {
  3126. DECODE_PRINTF("BYTE PTR ");
  3127. destoffset = decode_rmXX_address(mod, rl);
  3128. imm = fetch_byte_imm();
  3129. DECODE_PRINTF2(",%2x\n", imm);
  3130. TRACE_AND_STEP();
  3131. store_data_byte(destoffset, imm);
  3132. } else { /* register to register */
  3133. destreg = DECODE_RM_BYTE_REGISTER(rl);
  3134. imm = fetch_byte_imm();
  3135. DECODE_PRINTF2(",%2x\n", imm);
  3136. TRACE_AND_STEP();
  3137. *destreg = imm;
  3138. }
  3139. DECODE_CLEAR_SEGOVR();
  3140. END_OF_INSTR();
  3141. }
  3142. /****************************************************************************
  3143. REMARKS:
  3144. Handles opcode 0xc7
  3145. ****************************************************************************/
  3146. void x86emuOp_mov_word_RM_IMM(u8 X86EMU_UNUSED(op1))
  3147. {
  3148. int mod, rl, rh;
  3149. uint destoffset;
  3150. START_OF_INSTR();
  3151. DECODE_PRINTF("MOV\t");
  3152. FETCH_DECODE_MODRM(mod, rh, rl);
  3153. if (rh != 0) {
  3154. DECODE_PRINTF("ILLEGAL DECODE OF OPCODE 8F\n");
  3155. HALT_SYS();
  3156. }
  3157. if (mod < 3) {
  3158. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  3159. u32 imm;
  3160. DECODE_PRINTF("DWORD PTR ");
  3161. destoffset = decode_rmXX_address(mod, rl);
  3162. imm = fetch_long_imm();
  3163. DECODE_PRINTF2(",%x\n", imm);
  3164. TRACE_AND_STEP();
  3165. store_data_long(destoffset, imm);
  3166. } else {
  3167. u16 imm;
  3168. DECODE_PRINTF("WORD PTR ");
  3169. destoffset = decode_rmXX_address(mod, rl);
  3170. imm = fetch_word_imm();
  3171. DECODE_PRINTF2(",%x\n", imm);
  3172. TRACE_AND_STEP();
  3173. store_data_word(destoffset, imm);
  3174. }
  3175. } else { /* register to register */
  3176. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  3177. u32 *destreg;
  3178. u32 imm;
  3179. destreg = DECODE_RM_LONG_REGISTER(rl);
  3180. imm = fetch_long_imm();
  3181. DECODE_PRINTF2(",%x\n", imm);
  3182. TRACE_AND_STEP();
  3183. *destreg = imm;
  3184. } else {
  3185. u16 *destreg;
  3186. u16 imm;
  3187. destreg = DECODE_RM_WORD_REGISTER(rl);
  3188. imm = fetch_word_imm();
  3189. DECODE_PRINTF2(",%x\n", imm);
  3190. TRACE_AND_STEP();
  3191. *destreg = imm;
  3192. }
  3193. }
  3194. DECODE_CLEAR_SEGOVR();
  3195. END_OF_INSTR();
  3196. }
  3197. /****************************************************************************
  3198. REMARKS:
  3199. Handles opcode 0xc8
  3200. ****************************************************************************/
  3201. void x86emuOp_enter(u8 X86EMU_UNUSED(op1))
  3202. {
  3203. u16 local,frame_pointer;
  3204. u8 nesting;
  3205. int i;
  3206. START_OF_INSTR();
  3207. local = fetch_word_imm();
  3208. nesting = fetch_byte_imm();
  3209. DECODE_PRINTF2("ENTER %x\n", local);
  3210. DECODE_PRINTF2(",%x\n", nesting);
  3211. TRACE_AND_STEP();
  3212. push_word(M.x86.R_BP);
  3213. frame_pointer = M.x86.R_SP;
  3214. if (nesting > 0) {
  3215. for (i = 1; i < nesting; i++) {
  3216. M.x86.R_BP -= 2;
  3217. push_word(fetch_data_word_abs(M.x86.R_SS, M.x86.R_BP));
  3218. }
  3219. push_word(frame_pointer);
  3220. }
  3221. M.x86.R_BP = frame_pointer;
  3222. M.x86.R_SP = (u16)(M.x86.R_SP - local);
  3223. DECODE_CLEAR_SEGOVR();
  3224. END_OF_INSTR();
  3225. }
  3226. /****************************************************************************
  3227. REMARKS:
  3228. Handles opcode 0xc9
  3229. ****************************************************************************/
  3230. void x86emuOp_leave(u8 X86EMU_UNUSED(op1))
  3231. {
  3232. START_OF_INSTR();
  3233. DECODE_PRINTF("LEAVE\n");
  3234. TRACE_AND_STEP();
  3235. M.x86.R_SP = M.x86.R_BP;
  3236. M.x86.R_BP = pop_word();
  3237. DECODE_CLEAR_SEGOVR();
  3238. END_OF_INSTR();
  3239. }
  3240. /****************************************************************************
  3241. REMARKS:
  3242. Handles opcode 0xca
  3243. ****************************************************************************/
  3244. void x86emuOp_ret_far_IMM(u8 X86EMU_UNUSED(op1))
  3245. {
  3246. u16 imm;
  3247. START_OF_INSTR();
  3248. DECODE_PRINTF("RETF\t");
  3249. imm = fetch_word_imm();
  3250. DECODE_PRINTF2("%x\n", imm);
  3251. RETURN_TRACE("RETF",M.x86.saved_cs,M.x86.saved_ip);
  3252. TRACE_AND_STEP();
  3253. M.x86.R_IP = pop_word();
  3254. M.x86.R_CS = pop_word();
  3255. M.x86.R_SP += imm;
  3256. DECODE_CLEAR_SEGOVR();
  3257. END_OF_INSTR();
  3258. }
  3259. /****************************************************************************
  3260. REMARKS:
  3261. Handles opcode 0xcb
  3262. ****************************************************************************/
  3263. void x86emuOp_ret_far(u8 X86EMU_UNUSED(op1))
  3264. {
  3265. START_OF_INSTR();
  3266. DECODE_PRINTF("RETF\n");
  3267. RETURN_TRACE("RETF",M.x86.saved_cs,M.x86.saved_ip);
  3268. TRACE_AND_STEP();
  3269. M.x86.R_IP = pop_word();
  3270. M.x86.R_CS = pop_word();
  3271. DECODE_CLEAR_SEGOVR();
  3272. END_OF_INSTR();
  3273. }
  3274. /****************************************************************************
  3275. REMARKS:
  3276. Handles opcode 0xcc
  3277. ****************************************************************************/
  3278. void x86emuOp_int3(u8 X86EMU_UNUSED(op1))
  3279. {
  3280. u16 tmp;
  3281. START_OF_INSTR();
  3282. DECODE_PRINTF("INT 3\n");
  3283. tmp = (u16) mem_access_word(3 * 4 + 2);
  3284. /* access the segment register */
  3285. TRACE_AND_STEP();
  3286. if (_X86EMU_intrTab[3]) {
  3287. (*_X86EMU_intrTab[3])(3);
  3288. } else {
  3289. push_word((u16)M.x86.R_FLG);
  3290. CLEAR_FLAG(F_IF);
  3291. CLEAR_FLAG(F_TF);
  3292. push_word(M.x86.R_CS);
  3293. M.x86.R_CS = mem_access_word(3 * 4 + 2);
  3294. push_word(M.x86.R_IP);
  3295. M.x86.R_IP = mem_access_word(3 * 4);
  3296. }
  3297. DECODE_CLEAR_SEGOVR();
  3298. END_OF_INSTR();
  3299. }
  3300. /****************************************************************************
  3301. REMARKS:
  3302. Handles opcode 0xcd
  3303. ****************************************************************************/
  3304. void x86emuOp_int_IMM(u8 X86EMU_UNUSED(op1))
  3305. {
  3306. u16 tmp;
  3307. u8 intnum;
  3308. START_OF_INSTR();
  3309. DECODE_PRINTF("INT\t");
  3310. intnum = fetch_byte_imm();
  3311. DECODE_PRINTF2("%x\n", intnum);
  3312. tmp = mem_access_word(intnum * 4 + 2);
  3313. TRACE_AND_STEP();
  3314. if (_X86EMU_intrTab[intnum]) {
  3315. (*_X86EMU_intrTab[intnum])(intnum);
  3316. } else {
  3317. push_word((u16)M.x86.R_FLG);
  3318. CLEAR_FLAG(F_IF);
  3319. CLEAR_FLAG(F_TF);
  3320. push_word(M.x86.R_CS);
  3321. M.x86.R_CS = mem_access_word(intnum * 4 + 2);
  3322. push_word(M.x86.R_IP);
  3323. M.x86.R_IP = mem_access_word(intnum * 4);
  3324. }
  3325. DECODE_CLEAR_SEGOVR();
  3326. END_OF_INSTR();
  3327. }
  3328. /****************************************************************************
  3329. REMARKS:
  3330. Handles opcode 0xce
  3331. ****************************************************************************/
  3332. void x86emuOp_into(u8 X86EMU_UNUSED(op1))
  3333. {
  3334. u16 tmp;
  3335. START_OF_INSTR();
  3336. DECODE_PRINTF("INTO\n");
  3337. TRACE_AND_STEP();
  3338. if (ACCESS_FLAG(F_OF)) {
  3339. tmp = mem_access_word(4 * 4 + 2);
  3340. if (_X86EMU_intrTab[4]) {
  3341. (*_X86EMU_intrTab[4])(4);
  3342. } else {
  3343. push_word((u16)M.x86.R_FLG);
  3344. CLEAR_FLAG(F_IF);
  3345. CLEAR_FLAG(F_TF);
  3346. push_word(M.x86.R_CS);
  3347. M.x86.R_CS = mem_access_word(4 * 4 + 2);
  3348. push_word(M.x86.R_IP);
  3349. M.x86.R_IP = mem_access_word(4 * 4);
  3350. }
  3351. }
  3352. DECODE_CLEAR_SEGOVR();
  3353. END_OF_INSTR();
  3354. }
  3355. /****************************************************************************
  3356. REMARKS:
  3357. Handles opcode 0xcf
  3358. ****************************************************************************/
  3359. void x86emuOp_iret(u8 X86EMU_UNUSED(op1))
  3360. {
  3361. START_OF_INSTR();
  3362. DECODE_PRINTF("IRET\n");
  3363. TRACE_AND_STEP();
  3364. M.x86.R_IP = pop_word();
  3365. M.x86.R_CS = pop_word();
  3366. M.x86.R_FLG = pop_word();
  3367. DECODE_CLEAR_SEGOVR();
  3368. END_OF_INSTR();
  3369. }
  3370. /****************************************************************************
  3371. REMARKS:
  3372. Handles opcode 0xd0
  3373. ****************************************************************************/
  3374. void x86emuOp_opcD0_byte_RM_1(u8 X86EMU_UNUSED(op1))
  3375. {
  3376. int mod, rl, rh;
  3377. u8 *destreg;
  3378. uint destoffset;
  3379. u8 destval;
  3380. /*
  3381. * Yet another weirdo special case instruction format. Part of
  3382. * the opcode held below in "RH". Doubly nested case would
  3383. * result, except that the decoded instruction
  3384. */
  3385. START_OF_INSTR();
  3386. FETCH_DECODE_MODRM(mod, rh, rl);
  3387. #ifdef DEBUG
  3388. if (DEBUG_DECODE()) {
  3389. /* XXX DECODE_PRINTF may be changed to something more
  3390. general, so that it is important to leave the strings
  3391. in the same format, even though the result is that the
  3392. above test is done twice. */
  3393. switch (rh) {
  3394. case 0:
  3395. DECODE_PRINTF("ROL\t");
  3396. break;
  3397. case 1:
  3398. DECODE_PRINTF("ROR\t");
  3399. break;
  3400. case 2:
  3401. DECODE_PRINTF("RCL\t");
  3402. break;
  3403. case 3:
  3404. DECODE_PRINTF("RCR\t");
  3405. break;
  3406. case 4:
  3407. DECODE_PRINTF("SHL\t");
  3408. break;
  3409. case 5:
  3410. DECODE_PRINTF("SHR\t");
  3411. break;
  3412. case 6:
  3413. DECODE_PRINTF("SAL\t");
  3414. break;
  3415. case 7:
  3416. DECODE_PRINTF("SAR\t");
  3417. break;
  3418. }
  3419. }
  3420. #endif
  3421. /* know operation, decode the mod byte to find the addressing
  3422. mode. */
  3423. if (mod < 3) {
  3424. DECODE_PRINTF("BYTE PTR ");
  3425. destoffset = decode_rmXX_address(mod, rl);
  3426. DECODE_PRINTF(",1\n");
  3427. destval = fetch_data_byte(destoffset);
  3428. TRACE_AND_STEP();
  3429. destval = (*opcD0_byte_operation[rh]) (destval, 1);
  3430. store_data_byte(destoffset, destval);
  3431. } else { /* register to register */
  3432. destreg = DECODE_RM_BYTE_REGISTER(rl);
  3433. DECODE_PRINTF(",1\n");
  3434. TRACE_AND_STEP();
  3435. destval = (*opcD0_byte_operation[rh]) (*destreg, 1);
  3436. *destreg = destval;
  3437. }
  3438. DECODE_CLEAR_SEGOVR();
  3439. END_OF_INSTR();
  3440. }
  3441. /****************************************************************************
  3442. REMARKS:
  3443. Handles opcode 0xd1
  3444. ****************************************************************************/
  3445. void x86emuOp_opcD1_word_RM_1(u8 X86EMU_UNUSED(op1))
  3446. {
  3447. int mod, rl, rh;
  3448. uint destoffset;
  3449. /*
  3450. * Yet another weirdo special case instruction format. Part of
  3451. * the opcode held below in "RH". Doubly nested case would
  3452. * result, except that the decoded instruction
  3453. */
  3454. START_OF_INSTR();
  3455. FETCH_DECODE_MODRM(mod, rh, rl);
  3456. #ifdef DEBUG
  3457. if (DEBUG_DECODE()) {
  3458. /* XXX DECODE_PRINTF may be changed to something more
  3459. general, so that it is important to leave the strings
  3460. in the same format, even though the result is that the
  3461. above test is done twice. */
  3462. switch (rh) {
  3463. case 0:
  3464. DECODE_PRINTF("ROL\t");
  3465. break;
  3466. case 1:
  3467. DECODE_PRINTF("ROR\t");
  3468. break;
  3469. case 2:
  3470. DECODE_PRINTF("RCL\t");
  3471. break;
  3472. case 3:
  3473. DECODE_PRINTF("RCR\t");
  3474. break;
  3475. case 4:
  3476. DECODE_PRINTF("SHL\t");
  3477. break;
  3478. case 5:
  3479. DECODE_PRINTF("SHR\t");
  3480. break;
  3481. case 6:
  3482. DECODE_PRINTF("SAL\t");
  3483. break;
  3484. case 7:
  3485. DECODE_PRINTF("SAR\t");
  3486. break;
  3487. }
  3488. }
  3489. #endif
  3490. /* know operation, decode the mod byte to find the addressing
  3491. mode. */
  3492. if (mod < 3) {
  3493. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  3494. u32 destval;
  3495. DECODE_PRINTF("DWORD PTR ");
  3496. destoffset = decode_rmXX_address(mod, rl);
  3497. DECODE_PRINTF(",1\n");
  3498. destval = fetch_data_long(destoffset);
  3499. TRACE_AND_STEP();
  3500. destval = (*opcD1_long_operation[rh]) (destval, 1);
  3501. store_data_long(destoffset, destval);
  3502. } else {
  3503. u16 destval;
  3504. DECODE_PRINTF("WORD PTR ");
  3505. destoffset = decode_rmXX_address(mod, rl);
  3506. DECODE_PRINTF(",1\n");
  3507. destval = fetch_data_word(destoffset);
  3508. TRACE_AND_STEP();
  3509. destval = (*opcD1_word_operation[rh]) (destval, 1);
  3510. store_data_word(destoffset, destval);
  3511. }
  3512. } else { /* register to register */
  3513. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  3514. u32 destval;
  3515. u32 *destreg;
  3516. destreg = DECODE_RM_LONG_REGISTER(rl);
  3517. DECODE_PRINTF(",1\n");
  3518. TRACE_AND_STEP();
  3519. destval = (*opcD1_long_operation[rh]) (*destreg, 1);
  3520. *destreg = destval;
  3521. } else {
  3522. u16 destval;
  3523. u16 *destreg;
  3524. destreg = DECODE_RM_WORD_REGISTER(rl);
  3525. DECODE_PRINTF(",1\n");
  3526. TRACE_AND_STEP();
  3527. destval = (*opcD1_word_operation[rh]) (*destreg, 1);
  3528. *destreg = destval;
  3529. }
  3530. }
  3531. DECODE_CLEAR_SEGOVR();
  3532. END_OF_INSTR();
  3533. }
  3534. /****************************************************************************
  3535. REMARKS:
  3536. Handles opcode 0xd2
  3537. ****************************************************************************/
  3538. void x86emuOp_opcD2_byte_RM_CL(u8 X86EMU_UNUSED(op1))
  3539. {
  3540. int mod, rl, rh;
  3541. u8 *destreg;
  3542. uint destoffset;
  3543. u8 destval;
  3544. u8 amt;
  3545. /*
  3546. * Yet another weirdo special case instruction format. Part of
  3547. * the opcode held below in "RH". Doubly nested case would
  3548. * result, except that the decoded instruction
  3549. */
  3550. START_OF_INSTR();
  3551. FETCH_DECODE_MODRM(mod, rh, rl);
  3552. #ifdef DEBUG
  3553. if (DEBUG_DECODE()) {
  3554. /* XXX DECODE_PRINTF may be changed to something more
  3555. general, so that it is important to leave the strings
  3556. in the same format, even though the result is that the
  3557. above test is done twice. */
  3558. switch (rh) {
  3559. case 0:
  3560. DECODE_PRINTF("ROL\t");
  3561. break;
  3562. case 1:
  3563. DECODE_PRINTF("ROR\t");
  3564. break;
  3565. case 2:
  3566. DECODE_PRINTF("RCL\t");
  3567. break;
  3568. case 3:
  3569. DECODE_PRINTF("RCR\t");
  3570. break;
  3571. case 4:
  3572. DECODE_PRINTF("SHL\t");
  3573. break;
  3574. case 5:
  3575. DECODE_PRINTF("SHR\t");
  3576. break;
  3577. case 6:
  3578. DECODE_PRINTF("SAL\t");
  3579. break;
  3580. case 7:
  3581. DECODE_PRINTF("SAR\t");
  3582. break;
  3583. }
  3584. }
  3585. #endif
  3586. /* know operation, decode the mod byte to find the addressing
  3587. mode. */
  3588. amt = M.x86.R_CL;
  3589. if (mod < 3) {
  3590. DECODE_PRINTF("BYTE PTR ");
  3591. destoffset = decode_rmXX_address(mod, rl);
  3592. DECODE_PRINTF(",CL\n");
  3593. destval = fetch_data_byte(destoffset);
  3594. TRACE_AND_STEP();
  3595. destval = (*opcD0_byte_operation[rh]) (destval, amt);
  3596. store_data_byte(destoffset, destval);
  3597. } else { /* register to register */
  3598. destreg = DECODE_RM_BYTE_REGISTER(rl);
  3599. DECODE_PRINTF(",CL\n");
  3600. TRACE_AND_STEP();
  3601. destval = (*opcD0_byte_operation[rh]) (*destreg, amt);
  3602. *destreg = destval;
  3603. }
  3604. DECODE_CLEAR_SEGOVR();
  3605. END_OF_INSTR();
  3606. }
  3607. /****************************************************************************
  3608. REMARKS:
  3609. Handles opcode 0xd3
  3610. ****************************************************************************/
  3611. void x86emuOp_opcD3_word_RM_CL(u8 X86EMU_UNUSED(op1))
  3612. {
  3613. int mod, rl, rh;
  3614. uint destoffset;
  3615. u8 amt;
  3616. /*
  3617. * Yet another weirdo special case instruction format. Part of
  3618. * the opcode held below in "RH". Doubly nested case would
  3619. * result, except that the decoded instruction
  3620. */
  3621. START_OF_INSTR();
  3622. FETCH_DECODE_MODRM(mod, rh, rl);
  3623. #ifdef DEBUG
  3624. if (DEBUG_DECODE()) {
  3625. /* XXX DECODE_PRINTF may be changed to something more
  3626. general, so that it is important to leave the strings
  3627. in the same format, even though the result is that the
  3628. above test is done twice. */
  3629. switch (rh) {
  3630. case 0:
  3631. DECODE_PRINTF("ROL\t");
  3632. break;
  3633. case 1:
  3634. DECODE_PRINTF("ROR\t");
  3635. break;
  3636. case 2:
  3637. DECODE_PRINTF("RCL\t");
  3638. break;
  3639. case 3:
  3640. DECODE_PRINTF("RCR\t");
  3641. break;
  3642. case 4:
  3643. DECODE_PRINTF("SHL\t");
  3644. break;
  3645. case 5:
  3646. DECODE_PRINTF("SHR\t");
  3647. break;
  3648. case 6:
  3649. DECODE_PRINTF("SAL\t");
  3650. break;
  3651. case 7:
  3652. DECODE_PRINTF("SAR\t");
  3653. break;
  3654. }
  3655. }
  3656. #endif
  3657. /* know operation, decode the mod byte to find the addressing
  3658. mode. */
  3659. amt = M.x86.R_CL;
  3660. if (mod < 3) {
  3661. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  3662. u32 destval;
  3663. DECODE_PRINTF("DWORD PTR ");
  3664. destoffset = decode_rmXX_address(mod, rl);
  3665. DECODE_PRINTF(",CL\n");
  3666. destval = fetch_data_long(destoffset);
  3667. TRACE_AND_STEP();
  3668. destval = (*opcD1_long_operation[rh]) (destval, amt);
  3669. store_data_long(destoffset, destval);
  3670. } else {
  3671. u16 destval;
  3672. DECODE_PRINTF("WORD PTR ");
  3673. destoffset = decode_rmXX_address(mod, rl);
  3674. DECODE_PRINTF(",CL\n");
  3675. destval = fetch_data_word(destoffset);
  3676. TRACE_AND_STEP();
  3677. destval = (*opcD1_word_operation[rh]) (destval, amt);
  3678. store_data_word(destoffset, destval);
  3679. }
  3680. } else { /* register to register */
  3681. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  3682. u32 *destreg;
  3683. destreg = DECODE_RM_LONG_REGISTER(rl);
  3684. DECODE_PRINTF(",CL\n");
  3685. TRACE_AND_STEP();
  3686. *destreg = (*opcD1_long_operation[rh]) (*destreg, amt);
  3687. } else {
  3688. u16 *destreg;
  3689. destreg = DECODE_RM_WORD_REGISTER(rl);
  3690. DECODE_PRINTF(",CL\n");
  3691. TRACE_AND_STEP();
  3692. *destreg = (*opcD1_word_operation[rh]) (*destreg, amt);
  3693. }
  3694. }
  3695. DECODE_CLEAR_SEGOVR();
  3696. END_OF_INSTR();
  3697. }
  3698. /****************************************************************************
  3699. REMARKS:
  3700. Handles opcode 0xd4
  3701. ****************************************************************************/
  3702. void x86emuOp_aam(u8 X86EMU_UNUSED(op1))
  3703. {
  3704. u8 a;
  3705. START_OF_INSTR();
  3706. DECODE_PRINTF("AAM\n");
  3707. a = fetch_byte_imm(); /* this is a stupid encoding. */
  3708. if (a != 10) {
  3709. DECODE_PRINTF("ERROR DECODING AAM\n");
  3710. TRACE_REGS();
  3711. HALT_SYS();
  3712. }
  3713. TRACE_AND_STEP();
  3714. /* note the type change here --- returning AL and AH in AX. */
  3715. M.x86.R_AX = aam_word(M.x86.R_AL);
  3716. DECODE_CLEAR_SEGOVR();
  3717. END_OF_INSTR();
  3718. }
  3719. /****************************************************************************
  3720. REMARKS:
  3721. Handles opcode 0xd5
  3722. ****************************************************************************/
  3723. void x86emuOp_aad(u8 X86EMU_UNUSED(op1))
  3724. {
  3725. u8 a;
  3726. START_OF_INSTR();
  3727. DECODE_PRINTF("AAD\n");
  3728. a = fetch_byte_imm();
  3729. TRACE_AND_STEP();
  3730. M.x86.R_AX = aad_word(M.x86.R_AX);
  3731. DECODE_CLEAR_SEGOVR();
  3732. END_OF_INSTR();
  3733. }
  3734. /* opcode 0xd6 ILLEGAL OPCODE */
  3735. /****************************************************************************
  3736. REMARKS:
  3737. Handles opcode 0xd7
  3738. ****************************************************************************/
  3739. void x86emuOp_xlat(u8 X86EMU_UNUSED(op1))
  3740. {
  3741. u16 addr;
  3742. START_OF_INSTR();
  3743. DECODE_PRINTF("XLAT\n");
  3744. TRACE_AND_STEP();
  3745. addr = (u16)(M.x86.R_BX + (u8)M.x86.R_AL);
  3746. M.x86.R_AL = fetch_data_byte(addr);
  3747. DECODE_CLEAR_SEGOVR();
  3748. END_OF_INSTR();
  3749. }
  3750. /* instuctions D8 .. DF are in i87_ops.c */
  3751. /****************************************************************************
  3752. REMARKS:
  3753. Handles opcode 0xe0
  3754. ****************************************************************************/
  3755. void x86emuOp_loopne(u8 X86EMU_UNUSED(op1))
  3756. {
  3757. s16 ip;
  3758. START_OF_INSTR();
  3759. DECODE_PRINTF("LOOPNE\t");
  3760. ip = (s8) fetch_byte_imm();
  3761. ip += (s16) M.x86.R_IP;
  3762. DECODE_PRINTF2("%04x\n", ip);
  3763. TRACE_AND_STEP();
  3764. M.x86.R_CX -= 1;
  3765. if (M.x86.R_CX != 0 && !ACCESS_FLAG(F_ZF)) /* CX != 0 and !ZF */
  3766. M.x86.R_IP = ip;
  3767. DECODE_CLEAR_SEGOVR();
  3768. END_OF_INSTR();
  3769. }
  3770. /****************************************************************************
  3771. REMARKS:
  3772. Handles opcode 0xe1
  3773. ****************************************************************************/
  3774. void x86emuOp_loope(u8 X86EMU_UNUSED(op1))
  3775. {
  3776. s16 ip;
  3777. START_OF_INSTR();
  3778. DECODE_PRINTF("LOOPE\t");
  3779. ip = (s8) fetch_byte_imm();
  3780. ip += (s16) M.x86.R_IP;
  3781. DECODE_PRINTF2("%04x\n", ip);
  3782. TRACE_AND_STEP();
  3783. M.x86.R_CX -= 1;
  3784. if (M.x86.R_CX != 0 && ACCESS_FLAG(F_ZF)) /* CX != 0 and ZF */
  3785. M.x86.R_IP = ip;
  3786. DECODE_CLEAR_SEGOVR();
  3787. END_OF_INSTR();
  3788. }
  3789. /****************************************************************************
  3790. REMARKS:
  3791. Handles opcode 0xe2
  3792. ****************************************************************************/
  3793. void x86emuOp_loop(u8 X86EMU_UNUSED(op1))
  3794. {
  3795. s16 ip;
  3796. START_OF_INSTR();
  3797. DECODE_PRINTF("LOOP\t");
  3798. ip = (s8) fetch_byte_imm();
  3799. ip += (s16) M.x86.R_IP;
  3800. DECODE_PRINTF2("%04x\n", ip);
  3801. TRACE_AND_STEP();
  3802. M.x86.R_CX -= 1;
  3803. if (M.x86.R_CX != 0)
  3804. M.x86.R_IP = ip;
  3805. DECODE_CLEAR_SEGOVR();
  3806. END_OF_INSTR();
  3807. }
  3808. /****************************************************************************
  3809. REMARKS:
  3810. Handles opcode 0xe3
  3811. ****************************************************************************/
  3812. void x86emuOp_jcxz(u8 X86EMU_UNUSED(op1))
  3813. {
  3814. u16 target;
  3815. s8 offset;
  3816. /* jump to byte offset if overflow flag is set */
  3817. START_OF_INSTR();
  3818. DECODE_PRINTF("JCXZ\t");
  3819. offset = (s8)fetch_byte_imm();
  3820. target = (u16)(M.x86.R_IP + offset);
  3821. DECODE_PRINTF2("%x\n", target);
  3822. TRACE_AND_STEP();
  3823. if (M.x86.R_CX == 0)
  3824. M.x86.R_IP = target;
  3825. DECODE_CLEAR_SEGOVR();
  3826. END_OF_INSTR();
  3827. }
  3828. /****************************************************************************
  3829. REMARKS:
  3830. Handles opcode 0xe4
  3831. ****************************************************************************/
  3832. void x86emuOp_in_byte_AL_IMM(u8 X86EMU_UNUSED(op1))
  3833. {
  3834. u8 port;
  3835. START_OF_INSTR();
  3836. DECODE_PRINTF("IN\t");
  3837. port = (u8) fetch_byte_imm();
  3838. DECODE_PRINTF2("%x,AL\n", port);
  3839. TRACE_AND_STEP();
  3840. M.x86.R_AL = (*sys_inb)(port);
  3841. DECODE_CLEAR_SEGOVR();
  3842. END_OF_INSTR();
  3843. }
  3844. /****************************************************************************
  3845. REMARKS:
  3846. Handles opcode 0xe5
  3847. ****************************************************************************/
  3848. void x86emuOp_in_word_AX_IMM(u8 X86EMU_UNUSED(op1))
  3849. {
  3850. u8 port;
  3851. START_OF_INSTR();
  3852. DECODE_PRINTF("IN\t");
  3853. port = (u8) fetch_byte_imm();
  3854. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  3855. DECODE_PRINTF2("EAX,%x\n", port);
  3856. } else {
  3857. DECODE_PRINTF2("AX,%x\n", port);
  3858. }
  3859. TRACE_AND_STEP();
  3860. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  3861. M.x86.R_EAX = (*sys_inl)(port);
  3862. } else {
  3863. M.x86.R_AX = (*sys_inw)(port);
  3864. }
  3865. DECODE_CLEAR_SEGOVR();
  3866. END_OF_INSTR();
  3867. }
  3868. /****************************************************************************
  3869. REMARKS:
  3870. Handles opcode 0xe6
  3871. ****************************************************************************/
  3872. void x86emuOp_out_byte_IMM_AL(u8 X86EMU_UNUSED(op1))
  3873. {
  3874. u8 port;
  3875. START_OF_INSTR();
  3876. DECODE_PRINTF("OUT\t");
  3877. port = (u8) fetch_byte_imm();
  3878. DECODE_PRINTF2("%x,AL\n", port);
  3879. TRACE_AND_STEP();
  3880. (*sys_outb)(port, M.x86.R_AL);
  3881. DECODE_CLEAR_SEGOVR();
  3882. END_OF_INSTR();
  3883. }
  3884. /****************************************************************************
  3885. REMARKS:
  3886. Handles opcode 0xe7
  3887. ****************************************************************************/
  3888. void x86emuOp_out_word_IMM_AX(u8 X86EMU_UNUSED(op1))
  3889. {
  3890. u8 port;
  3891. START_OF_INSTR();
  3892. DECODE_PRINTF("OUT\t");
  3893. port = (u8) fetch_byte_imm();
  3894. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  3895. DECODE_PRINTF2("%x,EAX\n", port);
  3896. } else {
  3897. DECODE_PRINTF2("%x,AX\n", port);
  3898. }
  3899. TRACE_AND_STEP();
  3900. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  3901. (*sys_outl)(port, M.x86.R_EAX);
  3902. } else {
  3903. (*sys_outw)(port, M.x86.R_AX);
  3904. }
  3905. DECODE_CLEAR_SEGOVR();
  3906. END_OF_INSTR();
  3907. }
  3908. /****************************************************************************
  3909. REMARKS:
  3910. Handles opcode 0xe8
  3911. ****************************************************************************/
  3912. void x86emuOp_call_near_IMM(u8 X86EMU_UNUSED(op1))
  3913. {
  3914. s16 ip;
  3915. START_OF_INSTR();
  3916. DECODE_PRINTF("CALL\t");
  3917. ip = (s16) fetch_word_imm();
  3918. ip += (s16) M.x86.R_IP; /* CHECK SIGN */
  3919. DECODE_PRINTF2("%04x\n", ip);
  3920. CALL_TRACE(M.x86.saved_cs, M.x86.saved_ip, M.x86.R_CS, ip, "");
  3921. TRACE_AND_STEP();
  3922. push_word(M.x86.R_IP);
  3923. M.x86.R_IP = ip;
  3924. DECODE_CLEAR_SEGOVR();
  3925. END_OF_INSTR();
  3926. }
  3927. /****************************************************************************
  3928. REMARKS:
  3929. Handles opcode 0xe9
  3930. ****************************************************************************/
  3931. void x86emuOp_jump_near_IMM(u8 X86EMU_UNUSED(op1))
  3932. {
  3933. int ip;
  3934. START_OF_INSTR();
  3935. DECODE_PRINTF("JMP\t");
  3936. ip = (s16)fetch_word_imm();
  3937. ip += (s16)M.x86.R_IP;
  3938. DECODE_PRINTF2("%04x\n", ip);
  3939. TRACE_AND_STEP();
  3940. M.x86.R_IP = (u16)ip;
  3941. DECODE_CLEAR_SEGOVR();
  3942. END_OF_INSTR();
  3943. }
  3944. /****************************************************************************
  3945. REMARKS:
  3946. Handles opcode 0xea
  3947. ****************************************************************************/
  3948. void x86emuOp_jump_far_IMM(u8 X86EMU_UNUSED(op1))
  3949. {
  3950. u16 cs, ip;
  3951. START_OF_INSTR();
  3952. DECODE_PRINTF("JMP\tFAR ");
  3953. ip = fetch_word_imm();
  3954. cs = fetch_word_imm();
  3955. DECODE_PRINTF2("%04x:", cs);
  3956. DECODE_PRINTF2("%04x\n", ip);
  3957. TRACE_AND_STEP();
  3958. M.x86.R_IP = ip;
  3959. M.x86.R_CS = cs;
  3960. DECODE_CLEAR_SEGOVR();
  3961. END_OF_INSTR();
  3962. }
  3963. /****************************************************************************
  3964. REMARKS:
  3965. Handles opcode 0xeb
  3966. ****************************************************************************/
  3967. void x86emuOp_jump_byte_IMM(u8 X86EMU_UNUSED(op1))
  3968. {
  3969. u16 target;
  3970. s8 offset;
  3971. START_OF_INSTR();
  3972. DECODE_PRINTF("JMP\t");
  3973. offset = (s8)fetch_byte_imm();
  3974. target = (u16)(M.x86.R_IP + offset);
  3975. DECODE_PRINTF2("%x\n", target);
  3976. TRACE_AND_STEP();
  3977. M.x86.R_IP = target;
  3978. DECODE_CLEAR_SEGOVR();
  3979. END_OF_INSTR();
  3980. }
  3981. /****************************************************************************
  3982. REMARKS:
  3983. Handles opcode 0xec
  3984. ****************************************************************************/
  3985. void x86emuOp_in_byte_AL_DX(u8 X86EMU_UNUSED(op1))
  3986. {
  3987. START_OF_INSTR();
  3988. DECODE_PRINTF("IN\tAL,DX\n");
  3989. TRACE_AND_STEP();
  3990. M.x86.R_AL = (*sys_inb)(M.x86.R_DX);
  3991. DECODE_CLEAR_SEGOVR();
  3992. END_OF_INSTR();
  3993. }
  3994. /****************************************************************************
  3995. REMARKS:
  3996. Handles opcode 0xed
  3997. ****************************************************************************/
  3998. void x86emuOp_in_word_AX_DX(u8 X86EMU_UNUSED(op1))
  3999. {
  4000. START_OF_INSTR();
  4001. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  4002. DECODE_PRINTF("IN\tEAX,DX\n");
  4003. } else {
  4004. DECODE_PRINTF("IN\tAX,DX\n");
  4005. }
  4006. TRACE_AND_STEP();
  4007. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  4008. M.x86.R_EAX = (*sys_inl)(M.x86.R_DX);
  4009. } else {
  4010. M.x86.R_AX = (*sys_inw)(M.x86.R_DX);
  4011. }
  4012. DECODE_CLEAR_SEGOVR();
  4013. END_OF_INSTR();
  4014. }
  4015. /****************************************************************************
  4016. REMARKS:
  4017. Handles opcode 0xee
  4018. ****************************************************************************/
  4019. void x86emuOp_out_byte_DX_AL(u8 X86EMU_UNUSED(op1))
  4020. {
  4021. START_OF_INSTR();
  4022. DECODE_PRINTF("OUT\tDX,AL\n");
  4023. TRACE_AND_STEP();
  4024. (*sys_outb)(M.x86.R_DX, M.x86.R_AL);
  4025. DECODE_CLEAR_SEGOVR();
  4026. END_OF_INSTR();
  4027. }
  4028. /****************************************************************************
  4029. REMARKS:
  4030. Handles opcode 0xef
  4031. ****************************************************************************/
  4032. void x86emuOp_out_word_DX_AX(u8 X86EMU_UNUSED(op1))
  4033. {
  4034. START_OF_INSTR();
  4035. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  4036. DECODE_PRINTF("OUT\tDX,EAX\n");
  4037. } else {
  4038. DECODE_PRINTF("OUT\tDX,AX\n");
  4039. }
  4040. TRACE_AND_STEP();
  4041. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  4042. (*sys_outl)(M.x86.R_DX, M.x86.R_EAX);
  4043. } else {
  4044. (*sys_outw)(M.x86.R_DX, M.x86.R_AX);
  4045. }
  4046. DECODE_CLEAR_SEGOVR();
  4047. END_OF_INSTR();
  4048. }
  4049. /****************************************************************************
  4050. REMARKS:
  4051. Handles opcode 0xf0
  4052. ****************************************************************************/
  4053. void x86emuOp_lock(u8 X86EMU_UNUSED(op1))
  4054. {
  4055. START_OF_INSTR();
  4056. DECODE_PRINTF("LOCK:\n");
  4057. TRACE_AND_STEP();
  4058. DECODE_CLEAR_SEGOVR();
  4059. END_OF_INSTR();
  4060. }
  4061. /*opcode 0xf1 ILLEGAL OPERATION */
  4062. /****************************************************************************
  4063. REMARKS:
  4064. Handles opcode 0xf2
  4065. ****************************************************************************/
  4066. void x86emuOp_repne(u8 X86EMU_UNUSED(op1))
  4067. {
  4068. START_OF_INSTR();
  4069. DECODE_PRINTF("REPNE\n");
  4070. TRACE_AND_STEP();
  4071. M.x86.mode |= SYSMODE_PREFIX_REPNE;
  4072. DECODE_CLEAR_SEGOVR();
  4073. END_OF_INSTR();
  4074. }
  4075. /****************************************************************************
  4076. REMARKS:
  4077. Handles opcode 0xf3
  4078. ****************************************************************************/
  4079. void x86emuOp_repe(u8 X86EMU_UNUSED(op1))
  4080. {
  4081. START_OF_INSTR();
  4082. DECODE_PRINTF("REPE\n");
  4083. TRACE_AND_STEP();
  4084. M.x86.mode |= SYSMODE_PREFIX_REPE;
  4085. DECODE_CLEAR_SEGOVR();
  4086. END_OF_INSTR();
  4087. }
  4088. /****************************************************************************
  4089. REMARKS:
  4090. Handles opcode 0xf4
  4091. ****************************************************************************/
  4092. void x86emuOp_halt(u8 X86EMU_UNUSED(op1))
  4093. {
  4094. START_OF_INSTR();
  4095. DECODE_PRINTF("HALT\n");
  4096. TRACE_AND_STEP();
  4097. HALT_SYS();
  4098. DECODE_CLEAR_SEGOVR();
  4099. END_OF_INSTR();
  4100. }
  4101. /****************************************************************************
  4102. REMARKS:
  4103. Handles opcode 0xf5
  4104. ****************************************************************************/
  4105. void x86emuOp_cmc(u8 X86EMU_UNUSED(op1))
  4106. {
  4107. /* complement the carry flag. */
  4108. START_OF_INSTR();
  4109. DECODE_PRINTF("CMC\n");
  4110. TRACE_AND_STEP();
  4111. TOGGLE_FLAG(F_CF);
  4112. DECODE_CLEAR_SEGOVR();
  4113. END_OF_INSTR();
  4114. }
  4115. /****************************************************************************
  4116. REMARKS:
  4117. Handles opcode 0xf6
  4118. ****************************************************************************/
  4119. void x86emuOp_opcF6_byte_RM(u8 X86EMU_UNUSED(op1))
  4120. {
  4121. int mod, rl, rh;
  4122. u8 *destreg;
  4123. uint destoffset;
  4124. u8 destval, srcval;
  4125. /* long, drawn out code follows. Double switch for a total
  4126. of 32 cases. */
  4127. START_OF_INSTR();
  4128. FETCH_DECODE_MODRM(mod, rh, rl);
  4129. DECODE_PRINTF(opF6_names[rh]);
  4130. if (mod < 3) {
  4131. DECODE_PRINTF("BYTE PTR ");
  4132. destoffset = decode_rmXX_address(mod, rl);
  4133. destval = fetch_data_byte(destoffset);
  4134. switch (rh) {
  4135. case 0: /* test byte imm */
  4136. DECODE_PRINTF(",");
  4137. srcval = fetch_byte_imm();
  4138. DECODE_PRINTF2("%02x\n", srcval);
  4139. TRACE_AND_STEP();
  4140. test_byte(destval, srcval);
  4141. break;
  4142. case 1:
  4143. DECODE_PRINTF("ILLEGAL OP MOD=00 RH=01 OP=F6\n");
  4144. HALT_SYS();
  4145. break;
  4146. case 2:
  4147. DECODE_PRINTF("\n");
  4148. TRACE_AND_STEP();
  4149. destval = not_byte(destval);
  4150. store_data_byte(destoffset, destval);
  4151. break;
  4152. case 3:
  4153. DECODE_PRINTF("\n");
  4154. TRACE_AND_STEP();
  4155. destval = neg_byte(destval);
  4156. store_data_byte(destoffset, destval);
  4157. break;
  4158. case 4:
  4159. DECODE_PRINTF("\n");
  4160. TRACE_AND_STEP();
  4161. mul_byte(destval);
  4162. break;
  4163. case 5:
  4164. DECODE_PRINTF("\n");
  4165. TRACE_AND_STEP();
  4166. imul_byte(destval);
  4167. break;
  4168. case 6:
  4169. DECODE_PRINTF("\n");
  4170. TRACE_AND_STEP();
  4171. div_byte(destval);
  4172. break;
  4173. default:
  4174. DECODE_PRINTF("\n");
  4175. TRACE_AND_STEP();
  4176. idiv_byte(destval);
  4177. break;
  4178. }
  4179. } else { /* mod=11 */
  4180. destreg = DECODE_RM_BYTE_REGISTER(rl);
  4181. switch (rh) {
  4182. case 0: /* test byte imm */
  4183. DECODE_PRINTF(",");
  4184. srcval = fetch_byte_imm();
  4185. DECODE_PRINTF2("%02x\n", srcval);
  4186. TRACE_AND_STEP();
  4187. test_byte(*destreg, srcval);
  4188. break;
  4189. case 1:
  4190. DECODE_PRINTF("ILLEGAL OP MOD=00 RH=01 OP=F6\n");
  4191. HALT_SYS();
  4192. break;
  4193. case 2:
  4194. DECODE_PRINTF("\n");
  4195. TRACE_AND_STEP();
  4196. *destreg = not_byte(*destreg);
  4197. break;
  4198. case 3:
  4199. DECODE_PRINTF("\n");
  4200. TRACE_AND_STEP();
  4201. *destreg = neg_byte(*destreg);
  4202. break;
  4203. case 4:
  4204. DECODE_PRINTF("\n");
  4205. TRACE_AND_STEP();
  4206. mul_byte(*destreg); /*!!! */
  4207. break;
  4208. case 5:
  4209. DECODE_PRINTF("\n");
  4210. TRACE_AND_STEP();
  4211. imul_byte(*destreg);
  4212. break;
  4213. case 6:
  4214. DECODE_PRINTF("\n");
  4215. TRACE_AND_STEP();
  4216. div_byte(*destreg);
  4217. break;
  4218. default:
  4219. DECODE_PRINTF("\n");
  4220. TRACE_AND_STEP();
  4221. idiv_byte(*destreg);
  4222. break;
  4223. }
  4224. }
  4225. DECODE_CLEAR_SEGOVR();
  4226. END_OF_INSTR();
  4227. }
  4228. /****************************************************************************
  4229. REMARKS:
  4230. Handles opcode 0xf7
  4231. ****************************************************************************/
  4232. void x86emuOp_opcF7_word_RM(u8 X86EMU_UNUSED(op1))
  4233. {
  4234. int mod, rl, rh;
  4235. uint destoffset;
  4236. START_OF_INSTR();
  4237. FETCH_DECODE_MODRM(mod, rh, rl);
  4238. DECODE_PRINTF(opF6_names[rh]);
  4239. if (mod < 3) {
  4240. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  4241. u32 destval, srcval;
  4242. DECODE_PRINTF("DWORD PTR ");
  4243. destoffset = decode_rmXX_address(mod, rl);
  4244. destval = fetch_data_long(destoffset);
  4245. switch (rh) {
  4246. case 0:
  4247. DECODE_PRINTF(",");
  4248. srcval = fetch_long_imm();
  4249. DECODE_PRINTF2("%x\n", srcval);
  4250. TRACE_AND_STEP();
  4251. test_long(destval, srcval);
  4252. break;
  4253. case 1:
  4254. DECODE_PRINTF("ILLEGAL OP MOD=00 RH=01 OP=F7\n");
  4255. HALT_SYS();
  4256. break;
  4257. case 2:
  4258. DECODE_PRINTF("\n");
  4259. TRACE_AND_STEP();
  4260. destval = not_long(destval);
  4261. store_data_long(destoffset, destval);
  4262. break;
  4263. case 3:
  4264. DECODE_PRINTF("\n");
  4265. TRACE_AND_STEP();
  4266. destval = neg_long(destval);
  4267. store_data_long(destoffset, destval);
  4268. break;
  4269. case 4:
  4270. DECODE_PRINTF("\n");
  4271. TRACE_AND_STEP();
  4272. mul_long(destval);
  4273. break;
  4274. case 5:
  4275. DECODE_PRINTF("\n");
  4276. TRACE_AND_STEP();
  4277. imul_long(destval);
  4278. break;
  4279. case 6:
  4280. DECODE_PRINTF("\n");
  4281. TRACE_AND_STEP();
  4282. div_long(destval);
  4283. break;
  4284. case 7:
  4285. DECODE_PRINTF("\n");
  4286. TRACE_AND_STEP();
  4287. idiv_long(destval);
  4288. break;
  4289. }
  4290. } else {
  4291. u16 destval, srcval;
  4292. DECODE_PRINTF("WORD PTR ");
  4293. destoffset = decode_rmXX_address(mod, rl);
  4294. destval = fetch_data_word(destoffset);
  4295. switch (rh) {
  4296. case 0: /* test word imm */
  4297. DECODE_PRINTF(",");
  4298. srcval = fetch_word_imm();
  4299. DECODE_PRINTF2("%x\n", srcval);
  4300. TRACE_AND_STEP();
  4301. test_word(destval, srcval);
  4302. break;
  4303. case 1:
  4304. DECODE_PRINTF("ILLEGAL OP MOD=00 RH=01 OP=F7\n");
  4305. HALT_SYS();
  4306. break;
  4307. case 2:
  4308. DECODE_PRINTF("\n");
  4309. TRACE_AND_STEP();
  4310. destval = not_word(destval);
  4311. store_data_word(destoffset, destval);
  4312. break;
  4313. case 3:
  4314. DECODE_PRINTF("\n");
  4315. TRACE_AND_STEP();
  4316. destval = neg_word(destval);
  4317. store_data_word(destoffset, destval);
  4318. break;
  4319. case 4:
  4320. DECODE_PRINTF("\n");
  4321. TRACE_AND_STEP();
  4322. mul_word(destval);
  4323. break;
  4324. case 5:
  4325. DECODE_PRINTF("\n");
  4326. TRACE_AND_STEP();
  4327. imul_word(destval);
  4328. break;
  4329. case 6:
  4330. DECODE_PRINTF("\n");
  4331. TRACE_AND_STEP();
  4332. div_word(destval);
  4333. break;
  4334. case 7:
  4335. DECODE_PRINTF("\n");
  4336. TRACE_AND_STEP();
  4337. idiv_word(destval);
  4338. break;
  4339. }
  4340. }
  4341. } else { /* mod=11 */
  4342. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  4343. u32 *destreg;
  4344. u32 srcval;
  4345. destreg = DECODE_RM_LONG_REGISTER(rl);
  4346. switch (rh) {
  4347. case 0: /* test word imm */
  4348. DECODE_PRINTF(",");
  4349. srcval = fetch_long_imm();
  4350. DECODE_PRINTF2("%x\n", srcval);
  4351. TRACE_AND_STEP();
  4352. test_long(*destreg, srcval);
  4353. break;
  4354. case 1:
  4355. DECODE_PRINTF("ILLEGAL OP MOD=00 RH=01 OP=F6\n");
  4356. HALT_SYS();
  4357. break;
  4358. case 2:
  4359. DECODE_PRINTF("\n");
  4360. TRACE_AND_STEP();
  4361. *destreg = not_long(*destreg);
  4362. break;
  4363. case 3:
  4364. DECODE_PRINTF("\n");
  4365. TRACE_AND_STEP();
  4366. *destreg = neg_long(*destreg);
  4367. break;
  4368. case 4:
  4369. DECODE_PRINTF("\n");
  4370. TRACE_AND_STEP();
  4371. mul_long(*destreg); /*!!! */
  4372. break;
  4373. case 5:
  4374. DECODE_PRINTF("\n");
  4375. TRACE_AND_STEP();
  4376. imul_long(*destreg);
  4377. break;
  4378. case 6:
  4379. DECODE_PRINTF("\n");
  4380. TRACE_AND_STEP();
  4381. div_long(*destreg);
  4382. break;
  4383. case 7:
  4384. DECODE_PRINTF("\n");
  4385. TRACE_AND_STEP();
  4386. idiv_long(*destreg);
  4387. break;
  4388. }
  4389. } else {
  4390. u16 *destreg;
  4391. u16 srcval;
  4392. destreg = DECODE_RM_WORD_REGISTER(rl);
  4393. switch (rh) {
  4394. case 0: /* test word imm */
  4395. DECODE_PRINTF(",");
  4396. srcval = fetch_word_imm();
  4397. DECODE_PRINTF2("%x\n", srcval);
  4398. TRACE_AND_STEP();
  4399. test_word(*destreg, srcval);
  4400. break;
  4401. case 1:
  4402. DECODE_PRINTF("ILLEGAL OP MOD=00 RH=01 OP=F6\n");
  4403. HALT_SYS();
  4404. break;
  4405. case 2:
  4406. DECODE_PRINTF("\n");
  4407. TRACE_AND_STEP();
  4408. *destreg = not_word(*destreg);
  4409. break;
  4410. case 3:
  4411. DECODE_PRINTF("\n");
  4412. TRACE_AND_STEP();
  4413. *destreg = neg_word(*destreg);
  4414. break;
  4415. case 4:
  4416. DECODE_PRINTF("\n");
  4417. TRACE_AND_STEP();
  4418. mul_word(*destreg); /*!!! */
  4419. break;
  4420. case 5:
  4421. DECODE_PRINTF("\n");
  4422. TRACE_AND_STEP();
  4423. imul_word(*destreg);
  4424. break;
  4425. case 6:
  4426. DECODE_PRINTF("\n");
  4427. TRACE_AND_STEP();
  4428. div_word(*destreg);
  4429. break;
  4430. case 7:
  4431. DECODE_PRINTF("\n");
  4432. TRACE_AND_STEP();
  4433. idiv_word(*destreg);
  4434. break;
  4435. }
  4436. }
  4437. }
  4438. DECODE_CLEAR_SEGOVR();
  4439. END_OF_INSTR();
  4440. }
  4441. /****************************************************************************
  4442. REMARKS:
  4443. Handles opcode 0xf8
  4444. ****************************************************************************/
  4445. void x86emuOp_clc(u8 X86EMU_UNUSED(op1))
  4446. {
  4447. /* clear the carry flag. */
  4448. START_OF_INSTR();
  4449. DECODE_PRINTF("CLC\n");
  4450. TRACE_AND_STEP();
  4451. CLEAR_FLAG(F_CF);
  4452. DECODE_CLEAR_SEGOVR();
  4453. END_OF_INSTR();
  4454. }
  4455. /****************************************************************************
  4456. REMARKS:
  4457. Handles opcode 0xf9
  4458. ****************************************************************************/
  4459. void x86emuOp_stc(u8 X86EMU_UNUSED(op1))
  4460. {
  4461. /* set the carry flag. */
  4462. START_OF_INSTR();
  4463. DECODE_PRINTF("STC\n");
  4464. TRACE_AND_STEP();
  4465. SET_FLAG(F_CF);
  4466. DECODE_CLEAR_SEGOVR();
  4467. END_OF_INSTR();
  4468. }
  4469. /****************************************************************************
  4470. REMARKS:
  4471. Handles opcode 0xfa
  4472. ****************************************************************************/
  4473. void x86emuOp_cli(u8 X86EMU_UNUSED(op1))
  4474. {
  4475. /* clear interrupts. */
  4476. START_OF_INSTR();
  4477. DECODE_PRINTF("CLI\n");
  4478. TRACE_AND_STEP();
  4479. CLEAR_FLAG(F_IF);
  4480. DECODE_CLEAR_SEGOVR();
  4481. END_OF_INSTR();
  4482. }
  4483. /****************************************************************************
  4484. REMARKS:
  4485. Handles opcode 0xfb
  4486. ****************************************************************************/
  4487. void x86emuOp_sti(u8 X86EMU_UNUSED(op1))
  4488. {
  4489. /* enable interrupts. */
  4490. START_OF_INSTR();
  4491. DECODE_PRINTF("STI\n");
  4492. TRACE_AND_STEP();
  4493. SET_FLAG(F_IF);
  4494. DECODE_CLEAR_SEGOVR();
  4495. END_OF_INSTR();
  4496. }
  4497. /****************************************************************************
  4498. REMARKS:
  4499. Handles opcode 0xfc
  4500. ****************************************************************************/
  4501. void x86emuOp_cld(u8 X86EMU_UNUSED(op1))
  4502. {
  4503. /* clear interrupts. */
  4504. START_OF_INSTR();
  4505. DECODE_PRINTF("CLD\n");
  4506. TRACE_AND_STEP();
  4507. CLEAR_FLAG(F_DF);
  4508. DECODE_CLEAR_SEGOVR();
  4509. END_OF_INSTR();
  4510. }
  4511. /****************************************************************************
  4512. REMARKS:
  4513. Handles opcode 0xfd
  4514. ****************************************************************************/
  4515. void x86emuOp_std(u8 X86EMU_UNUSED(op1))
  4516. {
  4517. /* clear interrupts. */
  4518. START_OF_INSTR();
  4519. DECODE_PRINTF("STD\n");
  4520. TRACE_AND_STEP();
  4521. SET_FLAG(F_DF);
  4522. DECODE_CLEAR_SEGOVR();
  4523. END_OF_INSTR();
  4524. }
  4525. /****************************************************************************
  4526. REMARKS:
  4527. Handles opcode 0xfe
  4528. ****************************************************************************/
  4529. void x86emuOp_opcFE_byte_RM(u8 X86EMU_UNUSED(op1))
  4530. {
  4531. int mod, rh, rl;
  4532. u8 destval;
  4533. uint destoffset;
  4534. u8 *destreg;
  4535. /* Yet another special case instruction. */
  4536. START_OF_INSTR();
  4537. FETCH_DECODE_MODRM(mod, rh, rl);
  4538. #ifdef DEBUG
  4539. if (DEBUG_DECODE()) {
  4540. /* XXX DECODE_PRINTF may be changed to something more
  4541. general, so that it is important to leave the strings
  4542. in the same format, even though the result is that the
  4543. above test is done twice. */
  4544. switch (rh) {
  4545. case 0:
  4546. DECODE_PRINTF("INC\t");
  4547. break;
  4548. case 1:
  4549. DECODE_PRINTF("DEC\t");
  4550. break;
  4551. case 2:
  4552. case 3:
  4553. case 4:
  4554. case 5:
  4555. case 6:
  4556. case 7:
  4557. DECODE_PRINTF2("ILLEGAL OP MAJOR OP 0xFE MINOR OP %x \n", mod);
  4558. HALT_SYS();
  4559. break;
  4560. }
  4561. }
  4562. #endif
  4563. if (mod < 3) {
  4564. DECODE_PRINTF("BYTE PTR ");
  4565. destoffset = decode_rmXX_address(mod, rl);
  4566. DECODE_PRINTF("\n");
  4567. destval = fetch_data_byte(destoffset);
  4568. TRACE_AND_STEP();
  4569. if (rh == 0)
  4570. destval = inc_byte(destval);
  4571. else
  4572. destval = dec_byte(destval);
  4573. store_data_byte(destoffset, destval);
  4574. } else {
  4575. destreg = DECODE_RM_BYTE_REGISTER(rl);
  4576. DECODE_PRINTF("\n");
  4577. TRACE_AND_STEP();
  4578. if (rh == 0)
  4579. *destreg = inc_byte(*destreg);
  4580. else
  4581. *destreg = dec_byte(*destreg);
  4582. }
  4583. DECODE_CLEAR_SEGOVR();
  4584. END_OF_INSTR();
  4585. }
  4586. /****************************************************************************
  4587. REMARKS:
  4588. Handles opcode 0xff
  4589. ****************************************************************************/
  4590. void x86emuOp_opcFF_word_RM(u8 X86EMU_UNUSED(op1))
  4591. {
  4592. int mod, rh, rl;
  4593. uint destoffset = 0;
  4594. u16 *destreg;
  4595. u16 destval,destval2;
  4596. /* Yet another special case instruction. */
  4597. START_OF_INSTR();
  4598. FETCH_DECODE_MODRM(mod, rh, rl);
  4599. #ifdef DEBUG
  4600. if (DEBUG_DECODE()) {
  4601. /* XXX DECODE_PRINTF may be changed to something more
  4602. general, so that it is important to leave the strings
  4603. in the same format, even though the result is that the
  4604. above test is done twice. */
  4605. switch (rh) {
  4606. case 0:
  4607. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  4608. DECODE_PRINTF("INC\tDWORD PTR ");
  4609. } else {
  4610. DECODE_PRINTF("INC\tWORD PTR ");
  4611. }
  4612. break;
  4613. case 1:
  4614. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  4615. DECODE_PRINTF("DEC\tDWORD PTR ");
  4616. } else {
  4617. DECODE_PRINTF("DEC\tWORD PTR ");
  4618. }
  4619. break;
  4620. case 2:
  4621. DECODE_PRINTF("CALL\t ");
  4622. break;
  4623. case 3:
  4624. DECODE_PRINTF("CALL\tFAR ");
  4625. break;
  4626. case 4:
  4627. DECODE_PRINTF("JMP\t");
  4628. break;
  4629. case 5:
  4630. DECODE_PRINTF("JMP\tFAR ");
  4631. break;
  4632. case 6:
  4633. DECODE_PRINTF("PUSH\t");
  4634. break;
  4635. case 7:
  4636. DECODE_PRINTF("ILLEGAL DECODING OF OPCODE FF\t");
  4637. HALT_SYS();
  4638. break;
  4639. }
  4640. }
  4641. #endif
  4642. if (mod < 3) {
  4643. destoffset = decode_rmXX_address(mod, rl);
  4644. DECODE_PRINTF("\n");
  4645. switch (rh) {
  4646. case 0: /* inc word ptr ... */
  4647. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  4648. u32 destval;
  4649. destval = fetch_data_long(destoffset);
  4650. TRACE_AND_STEP();
  4651. destval = inc_long(destval);
  4652. store_data_long(destoffset, destval);
  4653. } else {
  4654. u16 destval;
  4655. destval = fetch_data_word(destoffset);
  4656. TRACE_AND_STEP();
  4657. destval = inc_word(destval);
  4658. store_data_word(destoffset, destval);
  4659. }
  4660. break;
  4661. case 1: /* dec word ptr ... */
  4662. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  4663. u32 destval;
  4664. destval = fetch_data_long(destoffset);
  4665. TRACE_AND_STEP();
  4666. destval = dec_long(destval);
  4667. store_data_long(destoffset, destval);
  4668. } else {
  4669. u16 destval;
  4670. destval = fetch_data_word(destoffset);
  4671. TRACE_AND_STEP();
  4672. destval = dec_word(destval);
  4673. store_data_word(destoffset, destval);
  4674. }
  4675. break;
  4676. case 2: /* call word ptr ... */
  4677. destval = fetch_data_word(destoffset);
  4678. TRACE_AND_STEP();
  4679. push_word(M.x86.R_IP);
  4680. M.x86.R_IP = destval;
  4681. break;
  4682. case 3: /* call far ptr ... */
  4683. destval = fetch_data_word(destoffset);
  4684. destval2 = fetch_data_word(destoffset + 2);
  4685. TRACE_AND_STEP();
  4686. push_word(M.x86.R_CS);
  4687. M.x86.R_CS = destval2;
  4688. push_word(M.x86.R_IP);
  4689. M.x86.R_IP = destval;
  4690. break;
  4691. case 4: /* jmp word ptr ... */
  4692. destval = fetch_data_word(destoffset);
  4693. TRACE_AND_STEP();
  4694. M.x86.R_IP = destval;
  4695. break;
  4696. case 5: /* jmp far ptr ... */
  4697. destval = fetch_data_word(destoffset);
  4698. destval2 = fetch_data_word(destoffset + 2);
  4699. TRACE_AND_STEP();
  4700. M.x86.R_IP = destval;
  4701. M.x86.R_CS = destval2;
  4702. break;
  4703. case 6: /* push word ptr ... */
  4704. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  4705. u32 destval;
  4706. destval = fetch_data_long(destoffset);
  4707. TRACE_AND_STEP();
  4708. push_long(destval);
  4709. } else {
  4710. u16 destval;
  4711. destval = fetch_data_word(destoffset);
  4712. TRACE_AND_STEP();
  4713. push_word(destval);
  4714. }
  4715. break;
  4716. }
  4717. } else {
  4718. switch (rh) {
  4719. case 0:
  4720. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  4721. u32 *destreg;
  4722. destreg = DECODE_RM_LONG_REGISTER(rl);
  4723. DECODE_PRINTF("\n");
  4724. TRACE_AND_STEP();
  4725. *destreg = inc_long(*destreg);
  4726. } else {
  4727. u16 *destreg;
  4728. destreg = DECODE_RM_WORD_REGISTER(rl);
  4729. DECODE_PRINTF("\n");
  4730. TRACE_AND_STEP();
  4731. *destreg = inc_word(*destreg);
  4732. }
  4733. break;
  4734. case 1:
  4735. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  4736. u32 *destreg;
  4737. destreg = DECODE_RM_LONG_REGISTER(rl);
  4738. DECODE_PRINTF("\n");
  4739. TRACE_AND_STEP();
  4740. *destreg = dec_long(*destreg);
  4741. } else {
  4742. u16 *destreg;
  4743. destreg = DECODE_RM_WORD_REGISTER(rl);
  4744. DECODE_PRINTF("\n");
  4745. TRACE_AND_STEP();
  4746. *destreg = dec_word(*destreg);
  4747. }
  4748. break;
  4749. case 2: /* call word ptr ... */
  4750. destreg = DECODE_RM_WORD_REGISTER(rl);
  4751. DECODE_PRINTF("\n");
  4752. TRACE_AND_STEP();
  4753. push_word(M.x86.R_IP);
  4754. M.x86.R_IP = *destreg;
  4755. break;
  4756. case 3: /* jmp far ptr ... */
  4757. DECODE_PRINTF("OPERATION UNDEFINED 0XFF \n");
  4758. TRACE_AND_STEP();
  4759. HALT_SYS();
  4760. break;
  4761. case 4: /* jmp ... */
  4762. destreg = DECODE_RM_WORD_REGISTER(rl);
  4763. DECODE_PRINTF("\n");
  4764. TRACE_AND_STEP();
  4765. M.x86.R_IP = (u16) (*destreg);
  4766. break;
  4767. case 5: /* jmp far ptr ... */
  4768. DECODE_PRINTF("OPERATION UNDEFINED 0XFF \n");
  4769. TRACE_AND_STEP();
  4770. HALT_SYS();
  4771. break;
  4772. case 6:
  4773. if (M.x86.mode & SYSMODE_PREFIX_DATA) {
  4774. u32 *destreg;
  4775. destreg = DECODE_RM_LONG_REGISTER(rl);
  4776. DECODE_PRINTF("\n");
  4777. TRACE_AND_STEP();
  4778. push_long(*destreg);
  4779. } else {
  4780. u16 *destreg;
  4781. destreg = DECODE_RM_WORD_REGISTER(rl);
  4782. DECODE_PRINTF("\n");
  4783. TRACE_AND_STEP();
  4784. push_word(*destreg);
  4785. }
  4786. break;
  4787. }
  4788. }
  4789. DECODE_CLEAR_SEGOVR();
  4790. END_OF_INSTR();
  4791. }
  4792. /***************************************************************************
  4793. * Single byte operation code table:
  4794. **************************************************************************/
  4795. void (*x86emu_optab[256])(u8) __attribute__ ((section(".got2"))) =
  4796. {
  4797. /* 0x00 */ x86emuOp_genop_byte_RM_R,
  4798. /* 0x01 */ x86emuOp_genop_word_RM_R,
  4799. /* 0x02 */ x86emuOp_genop_byte_R_RM,
  4800. /* 0x03 */ x86emuOp_genop_word_R_RM,
  4801. /* 0x04 */ x86emuOp_genop_byte_AL_IMM,
  4802. /* 0x05 */ x86emuOp_genop_word_AX_IMM,
  4803. /* 0x06 */ x86emuOp_push_ES,
  4804. /* 0x07 */ x86emuOp_pop_ES,
  4805. /* 0x08 */ x86emuOp_genop_byte_RM_R,
  4806. /* 0x09 */ x86emuOp_genop_word_RM_R,
  4807. /* 0x0a */ x86emuOp_genop_byte_R_RM,
  4808. /* 0x0b */ x86emuOp_genop_word_R_RM,
  4809. /* 0x0c */ x86emuOp_genop_byte_AL_IMM,
  4810. /* 0x0d */ x86emuOp_genop_word_AX_IMM,
  4811. /* 0x0e */ x86emuOp_push_CS,
  4812. /* 0x0f */ x86emuOp_two_byte,
  4813. /* 0x10 */ x86emuOp_genop_byte_RM_R,
  4814. /* 0x11 */ x86emuOp_genop_word_RM_R,
  4815. /* 0x12 */ x86emuOp_genop_byte_R_RM,
  4816. /* 0x13 */ x86emuOp_genop_word_R_RM,
  4817. /* 0x14 */ x86emuOp_genop_byte_AL_IMM,
  4818. /* 0x15 */ x86emuOp_genop_word_AX_IMM,
  4819. /* 0x16 */ x86emuOp_push_SS,
  4820. /* 0x17 */ x86emuOp_pop_SS,
  4821. /* 0x18 */ x86emuOp_genop_byte_RM_R,
  4822. /* 0x19 */ x86emuOp_genop_word_RM_R,
  4823. /* 0x1a */ x86emuOp_genop_byte_R_RM,
  4824. /* 0x1b */ x86emuOp_genop_word_R_RM,
  4825. /* 0x1c */ x86emuOp_genop_byte_AL_IMM,
  4826. /* 0x1d */ x86emuOp_genop_word_AX_IMM,
  4827. /* 0x1e */ x86emuOp_push_DS,
  4828. /* 0x1f */ x86emuOp_pop_DS,
  4829. /* 0x20 */ x86emuOp_genop_byte_RM_R,
  4830. /* 0x21 */ x86emuOp_genop_word_RM_R,
  4831. /* 0x22 */ x86emuOp_genop_byte_R_RM,
  4832. /* 0x23 */ x86emuOp_genop_word_R_RM,
  4833. /* 0x24 */ x86emuOp_genop_byte_AL_IMM,
  4834. /* 0x25 */ x86emuOp_genop_word_AX_IMM,
  4835. /* 0x26 */ x86emuOp_segovr_ES,
  4836. /* 0x27 */ x86emuOp_daa,
  4837. /* 0x28 */ x86emuOp_genop_byte_RM_R,
  4838. /* 0x29 */ x86emuOp_genop_word_RM_R,
  4839. /* 0x2a */ x86emuOp_genop_byte_R_RM,
  4840. /* 0x2b */ x86emuOp_genop_word_R_RM,
  4841. /* 0x2c */ x86emuOp_genop_byte_AL_IMM,
  4842. /* 0x2d */ x86emuOp_genop_word_AX_IMM,
  4843. /* 0x2e */ x86emuOp_segovr_CS,
  4844. /* 0x2f */ x86emuOp_das,
  4845. /* 0x30 */ x86emuOp_genop_byte_RM_R,
  4846. /* 0x31 */ x86emuOp_genop_word_RM_R,
  4847. /* 0x32 */ x86emuOp_genop_byte_R_RM,
  4848. /* 0x33 */ x86emuOp_genop_word_R_RM,
  4849. /* 0x34 */ x86emuOp_genop_byte_AL_IMM,
  4850. /* 0x35 */ x86emuOp_genop_word_AX_IMM,
  4851. /* 0x36 */ x86emuOp_segovr_SS,
  4852. /* 0x37 */ x86emuOp_aaa,
  4853. /* 0x38 */ x86emuOp_genop_byte_RM_R,
  4854. /* 0x39 */ x86emuOp_genop_word_RM_R,
  4855. /* 0x3a */ x86emuOp_genop_byte_R_RM,
  4856. /* 0x3b */ x86emuOp_genop_word_R_RM,
  4857. /* 0x3c */ x86emuOp_genop_byte_AL_IMM,
  4858. /* 0x3d */ x86emuOp_genop_word_AX_IMM,
  4859. /* 0x3e */ x86emuOp_segovr_DS,
  4860. /* 0x3f */ x86emuOp_aas,
  4861. /* 0x40 */ x86emuOp_inc_register,
  4862. /* 0x41 */ x86emuOp_inc_register,
  4863. /* 0x42 */ x86emuOp_inc_register,
  4864. /* 0x43 */ x86emuOp_inc_register,
  4865. /* 0x44 */ x86emuOp_inc_register,
  4866. /* 0x45 */ x86emuOp_inc_register,
  4867. /* 0x46 */ x86emuOp_inc_register,
  4868. /* 0x47 */ x86emuOp_inc_register,
  4869. /* 0x48 */ x86emuOp_dec_register,
  4870. /* 0x49 */ x86emuOp_dec_register,
  4871. /* 0x4a */ x86emuOp_dec_register,
  4872. /* 0x4b */ x86emuOp_dec_register,
  4873. /* 0x4c */ x86emuOp_dec_register,
  4874. /* 0x4d */ x86emuOp_dec_register,
  4875. /* 0x4e */ x86emuOp_dec_register,
  4876. /* 0x4f */ x86emuOp_dec_register,
  4877. /* 0x50 */ x86emuOp_push_register,
  4878. /* 0x51 */ x86emuOp_push_register,
  4879. /* 0x52 */ x86emuOp_push_register,
  4880. /* 0x53 */ x86emuOp_push_register,
  4881. /* 0x54 */ x86emuOp_push_register,
  4882. /* 0x55 */ x86emuOp_push_register,
  4883. /* 0x56 */ x86emuOp_push_register,
  4884. /* 0x57 */ x86emuOp_push_register,
  4885. /* 0x58 */ x86emuOp_pop_register,
  4886. /* 0x59 */ x86emuOp_pop_register,
  4887. /* 0x5a */ x86emuOp_pop_register,
  4888. /* 0x5b */ x86emuOp_pop_register,
  4889. /* 0x5c */ x86emuOp_pop_register,
  4890. /* 0x5d */ x86emuOp_pop_register,
  4891. /* 0x5e */ x86emuOp_pop_register,
  4892. /* 0x5f */ x86emuOp_pop_register,
  4893. /* 0x60 */ x86emuOp_push_all,
  4894. /* 0x61 */ x86emuOp_pop_all,
  4895. /* 0x62 */ x86emuOp_illegal_op, /* bound */
  4896. /* 0x63 */ x86emuOp_illegal_op, /* arpl */
  4897. /* 0x64 */ x86emuOp_segovr_FS,
  4898. /* 0x65 */ x86emuOp_segovr_GS,
  4899. /* 0x66 */ x86emuOp_prefix_data,
  4900. /* 0x67 */ x86emuOp_prefix_addr,
  4901. /* 0x68 */ x86emuOp_push_word_IMM,
  4902. /* 0x69 */ x86emuOp_imul_word_IMM,
  4903. /* 0x6a */ x86emuOp_push_byte_IMM,
  4904. /* 0x6b */ x86emuOp_imul_byte_IMM,
  4905. /* 0x6c */ x86emuOp_ins_byte,
  4906. /* 0x6d */ x86emuOp_ins_word,
  4907. /* 0x6e */ x86emuOp_outs_byte,
  4908. /* 0x6f */ x86emuOp_outs_word,
  4909. /* 0x70 */ x86emuOp_jump_near_cond,
  4910. /* 0x71 */ x86emuOp_jump_near_cond,
  4911. /* 0x72 */ x86emuOp_jump_near_cond,
  4912. /* 0x73 */ x86emuOp_jump_near_cond,
  4913. /* 0x74 */ x86emuOp_jump_near_cond,
  4914. /* 0x75 */ x86emuOp_jump_near_cond,
  4915. /* 0x76 */ x86emuOp_jump_near_cond,
  4916. /* 0x77 */ x86emuOp_jump_near_cond,
  4917. /* 0x78 */ x86emuOp_jump_near_cond,
  4918. /* 0x79 */ x86emuOp_jump_near_cond,
  4919. /* 0x7a */ x86emuOp_jump_near_cond,
  4920. /* 0x7b */ x86emuOp_jump_near_cond,
  4921. /* 0x7c */ x86emuOp_jump_near_cond,
  4922. /* 0x7d */ x86emuOp_jump_near_cond,
  4923. /* 0x7e */ x86emuOp_jump_near_cond,
  4924. /* 0x7f */ x86emuOp_jump_near_cond,
  4925. /* 0x80 */ x86emuOp_opc80_byte_RM_IMM,
  4926. /* 0x81 */ x86emuOp_opc81_word_RM_IMM,
  4927. /* 0x82 */ x86emuOp_opc82_byte_RM_IMM,
  4928. /* 0x83 */ x86emuOp_opc83_word_RM_IMM,
  4929. /* 0x84 */ x86emuOp_test_byte_RM_R,
  4930. /* 0x85 */ x86emuOp_test_word_RM_R,
  4931. /* 0x86 */ x86emuOp_xchg_byte_RM_R,
  4932. /* 0x87 */ x86emuOp_xchg_word_RM_R,
  4933. /* 0x88 */ x86emuOp_mov_byte_RM_R,
  4934. /* 0x89 */ x86emuOp_mov_word_RM_R,
  4935. /* 0x8a */ x86emuOp_mov_byte_R_RM,
  4936. /* 0x8b */ x86emuOp_mov_word_R_RM,
  4937. /* 0x8c */ x86emuOp_mov_word_RM_SR,
  4938. /* 0x8d */ x86emuOp_lea_word_R_M,
  4939. /* 0x8e */ x86emuOp_mov_word_SR_RM,
  4940. /* 0x8f */ x86emuOp_pop_RM,
  4941. /* 0x90 */ x86emuOp_nop,
  4942. /* 0x91 */ x86emuOp_xchg_word_AX_register,
  4943. /* 0x92 */ x86emuOp_xchg_word_AX_register,
  4944. /* 0x93 */ x86emuOp_xchg_word_AX_register,
  4945. /* 0x94 */ x86emuOp_xchg_word_AX_register,
  4946. /* 0x95 */ x86emuOp_xchg_word_AX_register,
  4947. /* 0x96 */ x86emuOp_xchg_word_AX_register,
  4948. /* 0x97 */ x86emuOp_xchg_word_AX_register,
  4949. /* 0x98 */ x86emuOp_cbw,
  4950. /* 0x99 */ x86emuOp_cwd,
  4951. /* 0x9a */ x86emuOp_call_far_IMM,
  4952. /* 0x9b */ x86emuOp_wait,
  4953. /* 0x9c */ x86emuOp_pushf_word,
  4954. /* 0x9d */ x86emuOp_popf_word,
  4955. /* 0x9e */ x86emuOp_sahf,
  4956. /* 0x9f */ x86emuOp_lahf,
  4957. /* 0xa0 */ x86emuOp_mov_AL_M_IMM,
  4958. /* 0xa1 */ x86emuOp_mov_AX_M_IMM,
  4959. /* 0xa2 */ x86emuOp_mov_M_AL_IMM,
  4960. /* 0xa3 */ x86emuOp_mov_M_AX_IMM,
  4961. /* 0xa4 */ x86emuOp_movs_byte,
  4962. /* 0xa5 */ x86emuOp_movs_word,
  4963. /* 0xa6 */ x86emuOp_cmps_byte,
  4964. /* 0xa7 */ x86emuOp_cmps_word,
  4965. /* 0xa8 */ x86emuOp_test_AL_IMM,
  4966. /* 0xa9 */ x86emuOp_test_AX_IMM,
  4967. /* 0xaa */ x86emuOp_stos_byte,
  4968. /* 0xab */ x86emuOp_stos_word,
  4969. /* 0xac */ x86emuOp_lods_byte,
  4970. /* 0xad */ x86emuOp_lods_word,
  4971. /* 0xac */ x86emuOp_scas_byte,
  4972. /* 0xad */ x86emuOp_scas_word,
  4973. /* 0xb0 */ x86emuOp_mov_byte_register_IMM,
  4974. /* 0xb1 */ x86emuOp_mov_byte_register_IMM,
  4975. /* 0xb2 */ x86emuOp_mov_byte_register_IMM,
  4976. /* 0xb3 */ x86emuOp_mov_byte_register_IMM,
  4977. /* 0xb4 */ x86emuOp_mov_byte_register_IMM,
  4978. /* 0xb5 */ x86emuOp_mov_byte_register_IMM,
  4979. /* 0xb6 */ x86emuOp_mov_byte_register_IMM,
  4980. /* 0xb7 */ x86emuOp_mov_byte_register_IMM,
  4981. /* 0xb8 */ x86emuOp_mov_word_register_IMM,
  4982. /* 0xb9 */ x86emuOp_mov_word_register_IMM,
  4983. /* 0xba */ x86emuOp_mov_word_register_IMM,
  4984. /* 0xbb */ x86emuOp_mov_word_register_IMM,
  4985. /* 0xbc */ x86emuOp_mov_word_register_IMM,
  4986. /* 0xbd */ x86emuOp_mov_word_register_IMM,
  4987. /* 0xbe */ x86emuOp_mov_word_register_IMM,
  4988. /* 0xbf */ x86emuOp_mov_word_register_IMM,
  4989. /* 0xc0 */ x86emuOp_opcC0_byte_RM_MEM,
  4990. /* 0xc1 */ x86emuOp_opcC1_word_RM_MEM,
  4991. /* 0xc2 */ x86emuOp_ret_near_IMM,
  4992. /* 0xc3 */ x86emuOp_ret_near,
  4993. /* 0xc4 */ x86emuOp_les_R_IMM,
  4994. /* 0xc5 */ x86emuOp_lds_R_IMM,
  4995. /* 0xc6 */ x86emuOp_mov_byte_RM_IMM,
  4996. /* 0xc7 */ x86emuOp_mov_word_RM_IMM,
  4997. /* 0xc8 */ x86emuOp_enter,
  4998. /* 0xc9 */ x86emuOp_leave,
  4999. /* 0xca */ x86emuOp_ret_far_IMM,
  5000. /* 0xcb */ x86emuOp_ret_far,
  5001. /* 0xcc */ x86emuOp_int3,
  5002. /* 0xcd */ x86emuOp_int_IMM,
  5003. /* 0xce */ x86emuOp_into,
  5004. /* 0xcf */ x86emuOp_iret,
  5005. /* 0xd0 */ x86emuOp_opcD0_byte_RM_1,
  5006. /* 0xd1 */ x86emuOp_opcD1_word_RM_1,
  5007. /* 0xd2 */ x86emuOp_opcD2_byte_RM_CL,
  5008. /* 0xd3 */ x86emuOp_opcD3_word_RM_CL,
  5009. /* 0xd4 */ x86emuOp_aam,
  5010. /* 0xd5 */ x86emuOp_aad,
  5011. /* 0xd6 */ x86emuOp_illegal_op, /* Undocumented SETALC instruction */
  5012. /* 0xd7 */ x86emuOp_xlat,
  5013. /* 0xd8 */ NULL, /*x86emuOp_esc_coprocess_d8,*/
  5014. /* 0xd9 */ NULL, /*x86emuOp_esc_coprocess_d9,*/
  5015. /* 0xda */ NULL, /*x86emuOp_esc_coprocess_da,*/
  5016. /* 0xdb */ NULL, /*x86emuOp_esc_coprocess_db,*/
  5017. /* 0xdc */ NULL, /*x86emuOp_esc_coprocess_dc,*/
  5018. /* 0xdd */ NULL, /*x86emuOp_esc_coprocess_dd,*/
  5019. /* 0xde */ NULL, /*x86emuOp_esc_coprocess_de,*/
  5020. /* 0xdf */ NULL, /*x86emuOp_esc_coprocess_df,*/
  5021. /* 0xe0 */ x86emuOp_loopne,
  5022. /* 0xe1 */ x86emuOp_loope,
  5023. /* 0xe2 */ x86emuOp_loop,
  5024. /* 0xe3 */ x86emuOp_jcxz,
  5025. /* 0xe4 */ x86emuOp_in_byte_AL_IMM,
  5026. /* 0xe5 */ x86emuOp_in_word_AX_IMM,
  5027. /* 0xe6 */ x86emuOp_out_byte_IMM_AL,
  5028. /* 0xe7 */ x86emuOp_out_word_IMM_AX,
  5029. /* 0xe8 */ x86emuOp_call_near_IMM,
  5030. /* 0xe9 */ x86emuOp_jump_near_IMM,
  5031. /* 0xea */ x86emuOp_jump_far_IMM,
  5032. /* 0xeb */ x86emuOp_jump_byte_IMM,
  5033. /* 0xec */ x86emuOp_in_byte_AL_DX,
  5034. /* 0xed */ x86emuOp_in_word_AX_DX,
  5035. /* 0xee */ x86emuOp_out_byte_DX_AL,
  5036. /* 0xef */ x86emuOp_out_word_DX_AX,
  5037. /* 0xf0 */ x86emuOp_lock,
  5038. /* 0xf1 */ x86emuOp_illegal_op,
  5039. /* 0xf2 */ x86emuOp_repne,
  5040. /* 0xf3 */ x86emuOp_repe,
  5041. /* 0xf4 */ x86emuOp_halt,
  5042. /* 0xf5 */ x86emuOp_cmc,
  5043. /* 0xf6 */ x86emuOp_opcF6_byte_RM,
  5044. /* 0xf7 */ x86emuOp_opcF7_word_RM,
  5045. /* 0xf8 */ x86emuOp_clc,
  5046. /* 0xf9 */ x86emuOp_stc,
  5047. /* 0xfa */ x86emuOp_cli,
  5048. /* 0xfb */ x86emuOp_sti,
  5049. /* 0xfc */ x86emuOp_cld,
  5050. /* 0xfd */ x86emuOp_std,
  5051. /* 0xfe */ x86emuOp_opcFE_byte_RM,
  5052. /* 0xff */ x86emuOp_opcFF_word_RM,
  5053. };
  5054. #endif