123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902490349044905490649074908490949104911491249134914491549164917491849194920492149224923492449254926492749284929493049314932493349344935493649374938493949404941494249434944494549464947494849494950495149524953495449554956495749584959496049614962496349644965496649674968496949704971497249734974497549764977497849794980498149824983498449854986498749884989499049914992499349944995499649974998499950005001500250035004500550065007500850095010501150125013501450155016501750185019502050215022502350245025502650275028502950305031503250335034503550365037503850395040504150425043504450455046504750485049505050515052505350545055505650575058505950605061506250635064506550665067506850695070507150725073507450755076507750785079508050815082508350845085508650875088508950905091509250935094509550965097509850995100510151025103510451055106510751085109511051115112511351145115511651175118511951205121512251235124512551265127512851295130513151325133513451355136513751385139514051415142514351445145514651475148514951505151515251535154515551565157515851595160516151625163516451655166516751685169517051715172517351745175517651775178517951805181518251835184518551865187518851895190519151925193519451955196519751985199520052015202520352045205520652075208520952105211521252135214521552165217521852195220522152225223522452255226522752285229523052315232523352345235523652375238523952405241524252435244524552465247524852495250525152525253525452555256525752585259526052615262526352645265526652675268526952705271527252735274527552765277527852795280528152825283528452855286528752885289529052915292529352945295529652975298529953005301530253035304530553065307530853095310531153125313531453155316531753185319532053215322532353245325532653275328532953305331533253335334533553365337533853395340534153425343534453455346534753485349535053515352535353545355535653575358535953605361536253635364536553665367536853695370537153725373537453755376537753785379538053815382538353845385538653875388538953905391539253935394539553965397539853995400540154025403540454055406540754085409541054115412541354145415541654175418541954205421542254235424542554265427542854295430543154325433543454355436543754385439544054415442544354445445544654475448544954505451545254535454545554565457545854595460546154625463546454655466546754685469547054715472547354745475547654775478547954805481548254835484548554865487548854895490549154925493549454955496549754985499550055015502550355045505550655075508550955105511551255135514551555165517551855195520552155225523552455255526552755285529553055315532553355345535553655375538553955405541554255435544554555465547554855495550555155525553555455555556555755585559556055615562556355645565556655675568556955705571557255735574557555765577557855795580558155825583558455855586558755885589559055915592559355945595559655975598559956005601560256035604560556065607560856095610561156125613561456155616561756185619562056215622562356245625562656275628562956305631563256335634563556365637563856395640564156425643564456455646564756485649565056515652565356545655565656575658565956605661566256635664566556665667566856695670567156725673567456755676567756785679568056815682568356845685568656875688568956905691569256935694569556965697569856995700570157025703570457055706570757085709571057115712571357145715571657175718571957205721572257235724572557265727572857295730573157325733573457355736573757385739574057415742574357445745574657475748574957505751575257535754575557565757575857595760576157625763576457655766576757685769577057715772577357745775577657775778577957805781578257835784578557865787578857895790579157925793579457955796579757985799580058015802580358045805580658075808580958105811581258135814581558165817581858195820582158225823582458255826582758285829583058315832583358345835583658375838583958405841584258435844584558465847584858495850585158525853585458555856585758585859586058615862586358645865586658675868586958705871587258735874587558765877587858795880588158825883588458855886588758885889589058915892589358945895589658975898589959005901590259035904590559065907590859095910591159125913591459155916591759185919592059215922592359245925592659275928592959305931593259335934593559365937593859395940594159425943594459455946594759485949595059515952595359545955595659575958595959605961596259635964596559665967596859695970597159725973597459755976597759785979598059815982598359845985598659875988598959905991599259935994599559965997599859996000600160026003600460056006600760086009601060116012601360146015601660176018601960206021602260236024602560266027602860296030603160326033603460356036603760386039604060416042604360446045604660476048604960506051605260536054605560566057605860596060606160626063606460656066606760686069607060716072607360746075607660776078607960806081608260836084608560866087608860896090609160926093609460956096609760986099610061016102610361046105610661076108610961106111611261136114611561166117611861196120612161226123612461256126612761286129613061316132613361346135613661376138613961406141614261436144614561466147614861496150615161526153615461556156615761586159616061616162616361646165616661676168616961706171617261736174617561766177617861796180618161826183618461856186618761886189619061916192619361946195619661976198619962006201620262036204620562066207620862096210621162126213621462156216621762186219622062216222622362246225622662276228622962306231623262336234623562366237623862396240624162426243624462456246624762486249625062516252625362546255625662576258625962606261626262636264626562666267626862696270627162726273627462756276627762786279628062816282628362846285628662876288628962906291629262936294629562966297629862996300630163026303630463056306630763086309631063116312631363146315631663176318631963206321632263236324632563266327632863296330633163326333633463356336633763386339634063416342634363446345634663476348634963506351635263536354635563566357635863596360636163626363636463656366636763686369637063716372637363746375637663776378637963806381638263836384638563866387638863896390639163926393639463956396639763986399640064016402640364046405640664076408640964106411641264136414641564166417641864196420642164226423642464256426642764286429643064316432643364346435643664376438643964406441644264436444644564466447644864496450645164526453645464556456645764586459646064616462646364646465646664676468646964706471647264736474647564766477647864796480648164826483648464856486648764886489649064916492649364946495649664976498649965006501650265036504650565066507650865096510651165126513651465156516651765186519652065216522652365246525652665276528652965306531653265336534653565366537653865396540654165426543654465456546654765486549655065516552655365546555655665576558655965606561656265636564656565666567656865696570657165726573657465756576657765786579658065816582658365846585658665876588658965906591659265936594659565966597659865996600660166026603660466056606660766086609661066116612661366146615661666176618661966206621662266236624662566266627662866296630663166326633663466356636663766386639664066416642664366446645664666476648664966506651665266536654665566566657665866596660666166626663666466656666666766686669667066716672667366746675667666776678667966806681668266836684668566866687668866896690669166926693669466956696669766986699670067016702670367046705670667076708670967106711671267136714671567166717671867196720672167226723672467256726672767286729673067316732673367346735673667376738673967406741674267436744674567466747674867496750675167526753675467556756675767586759676067616762676367646765676667676768676967706771677267736774677567766777677867796780678167826783678467856786678767886789679067916792679367946795679667976798679968006801680268036804680568066807680868096810681168126813681468156816681768186819682068216822682368246825682668276828682968306831683268336834683568366837683868396840684168426843684468456846684768486849685068516852685368546855685668576858685968606861686268636864686568666867686868696870687168726873687468756876687768786879688068816882688368846885688668876888688968906891689268936894689568966897689868996900690169026903690469056906690769086909691069116912691369146915691669176918691969206921692269236924692569266927692869296930693169326933693469356936693769386939694069416942694369446945694669476948694969506951695269536954695569566957695869596960696169626963696469656966696769686969697069716972697369746975697669776978697969806981698269836984698569866987698869896990699169926993699469956996699769986999700070017002700370047005700670077008700970107011701270137014701570167017701870197020702170227023702470257026702770287029703070317032703370347035703670377038703970407041704270437044704570467047704870497050705170527053705470557056705770587059706070617062706370647065706670677068706970707071707270737074707570767077707870797080708170827083708470857086708770887089709070917092709370947095709670977098709971007101710271037104710571067107710871097110711171127113711471157116711771187119712071217122712371247125712671277128712971307131713271337134713571367137713871397140714171427143714471457146714771487149715071517152715371547155715671577158715971607161716271637164716571667167716871697170717171727173717471757176717771787179718071817182718371847185718671877188718971907191719271937194719571967197719871997200720172027203720472057206720772087209721072117212721372147215721672177218721972207221722272237224722572267227722872297230723172327233723472357236723772387239724072417242724372447245724672477248724972507251725272537254725572567257725872597260726172627263726472657266726772687269727072717272727372747275727672777278727972807281728272837284728572867287728872897290729172927293729472957296729772987299730073017302730373047305730673077308730973107311731273137314731573167317731873197320732173227323732473257326732773287329733073317332733373347335733673377338733973407341734273437344734573467347734873497350735173527353735473557356735773587359736073617362736373647365736673677368736973707371737273737374737573767377737873797380738173827383738473857386738773887389739073917392739373947395739673977398739974007401740274037404740574067407740874097410741174127413741474157416741774187419742074217422742374247425742674277428742974307431743274337434743574367437743874397440744174427443744474457446744774487449745074517452745374547455745674577458745974607461746274637464746574667467746874697470747174727473747474757476747774787479748074817482748374847485748674877488748974907491749274937494749574967497749874997500750175027503750475057506750775087509751075117512751375147515751675177518751975207521752275237524752575267527752875297530753175327533753475357536753775387539754075417542754375447545754675477548754975507551755275537554755575567557755875597560756175627563756475657566756775687569757075717572757375747575757675777578757975807581758275837584758575867587758875897590759175927593759475957596759775987599760076017602760376047605760676077608760976107611761276137614761576167617761876197620762176227623762476257626762776287629763076317632763376347635763676377638763976407641764276437644764576467647764876497650765176527653765476557656765776587659766076617662766376647665766676677668766976707671767276737674767576767677767876797680768176827683768476857686768776887689769076917692769376947695769676977698769977007701770277037704770577067707770877097710771177127713771477157716771777187719772077217722772377247725772677277728772977307731773277337734773577367737773877397740774177427743774477457746774777487749775077517752775377547755775677577758775977607761776277637764776577667767776877697770777177727773777477757776777777787779778077817782778377847785778677877788778977907791779277937794779577967797779877997800780178027803780478057806780778087809781078117812781378147815781678177818781978207821782278237824782578267827782878297830783178327833783478357836783778387839784078417842784378447845784678477848784978507851785278537854785578567857785878597860786178627863786478657866786778687869787078717872787378747875787678777878787978807881788278837884788578867887788878897890789178927893789478957896789778987899790079017902790379047905790679077908790979107911791279137914791579167917791879197920792179227923792479257926792779287929793079317932793379347935793679377938793979407941794279437944794579467947794879497950795179527953795479557956795779587959796079617962796379647965796679677968796979707971797279737974797579767977797879797980798179827983798479857986798779887989799079917992799379947995799679977998799980008001800280038004800580068007800880098010801180128013801480158016801780188019802080218022802380248025802680278028802980308031803280338034803580368037803880398040804180428043804480458046804780488049805080518052805380548055805680578058805980608061806280638064806580668067806880698070807180728073807480758076807780788079808080818082808380848085808680878088808980908091809280938094809580968097809880998100810181028103810481058106810781088109811081118112811381148115811681178118811981208121812281238124812581268127812881298130813181328133813481358136813781388139814081418142814381448145814681478148814981508151815281538154815581568157815881598160816181628163816481658166816781688169817081718172817381748175817681778178817981808181818281838184818581868187818881898190819181928193819481958196819781988199820082018202820382048205820682078208820982108211821282138214821582168217821882198220822182228223822482258226822782288229823082318232823382348235823682378238823982408241824282438244824582468247824882498250825182528253825482558256825782588259826082618262826382648265826682678268826982708271827282738274827582768277827882798280828182828283828482858286828782888289829082918292829382948295829682978298829983008301830283038304830583068307830883098310831183128313831483158316831783188319832083218322832383248325832683278328832983308331833283338334833583368337833883398340834183428343834483458346834783488349835083518352835383548355835683578358835983608361836283638364836583668367836883698370837183728373837483758376837783788379838083818382838383848385838683878388838983908391839283938394839583968397839883998400840184028403840484058406840784088409841084118412841384148415841684178418841984208421842284238424842584268427842884298430843184328433843484358436843784388439844084418442844384448445844684478448844984508451845284538454845584568457845884598460846184628463846484658466846784688469847084718472847384748475847684778478847984808481848284838484848584868487848884898490849184928493849484958496849784988499850085018502850385048505850685078508850985108511851285138514851585168517851885198520852185228523852485258526852785288529853085318532853385348535853685378538853985408541854285438544854585468547854885498550855185528553855485558556855785588559856085618562856385648565856685678568856985708571857285738574857585768577857885798580858185828583858485858586858785888589859085918592859385948595859685978598859986008601860286038604860586068607860886098610861186128613861486158616861786188619862086218622862386248625862686278628862986308631863286338634863586368637863886398640864186428643864486458646864786488649865086518652865386548655865686578658865986608661866286638664866586668667866886698670867186728673867486758676867786788679868086818682868386848685868686878688868986908691869286938694869586968697869886998700870187028703870487058706870787088709871087118712871387148715871687178718871987208721872287238724872587268727872887298730873187328733873487358736873787388739874087418742874387448745874687478748874987508751875287538754875587568757875887598760876187628763876487658766876787688769877087718772877387748775877687778778877987808781878287838784878587868787878887898790879187928793879487958796879787988799880088018802880388048805880688078808880988108811881288138814881588168817881888198820882188228823882488258826882788288829883088318832883388348835883688378838883988408841884288438844884588468847884888498850885188528853885488558856885788588859886088618862886388648865886688678868886988708871887288738874887588768877887888798880888188828883888488858886888788888889889088918892889388948895889688978898889989008901890289038904890589068907890889098910891189128913891489158916891789188919892089218922892389248925892689278928892989308931893289338934893589368937893889398940894189428943894489458946894789488949895089518952895389548955895689578958895989608961896289638964896589668967896889698970897189728973897489758976897789788979898089818982898389848985898689878988898989908991899289938994899589968997899889999000900190029003900490059006900790089009901090119012901390149015901690179018901990209021902290239024902590269027902890299030903190329033903490359036903790389039904090419042904390449045904690479048904990509051905290539054905590569057905890599060906190629063906490659066906790689069907090719072907390749075907690779078907990809081908290839084908590869087908890899090909190929093909490959096909790989099910091019102910391049105910691079108910991109111911291139114911591169117911891199120912191229123912491259126912791289129913091319132913391349135913691379138913991409141914291439144914591469147914891499150915191529153915491559156915791589159916091619162916391649165916691679168916991709171917291739174917591769177917891799180918191829183918491859186918791889189919091919192919391949195919691979198919992009201920292039204920592069207920892099210921192129213921492159216921792189219922092219222922392249225922692279228922992309231923292339234923592369237923892399240924192429243924492459246924792489249925092519252925392549255925692579258925992609261926292639264926592669267926892699270927192729273927492759276927792789279928092819282928392849285928692879288928992909291929292939294929592969297929892999300930193029303930493059306930793089309931093119312931393149315931693179318931993209321932293239324932593269327932893299330933193329333933493359336933793389339934093419342934393449345934693479348934993509351935293539354 |
- /*++
- Copyright (c) 2012 Minoca Corp.
- This file is licensed under the terms of the GNU General Public License
- version 3. Alternative licensing terms are available. Contact
- info@minocacorp.com for details. See the LICENSE file at the root of this
- project for complete licensing information.
- Module Name:
- amlopex.c
- Abstract:
- This module implements ACPI AML low level opcode support for executing
- AML statements.
- Author:
- Evan Green 13-Nov-2012
- Environment:
- Kernel
- --*/
- //
- // ------------------------------------------------------------------- Includes
- //
- #include <minoca/kernel/driver.h>
- #include "acpip.h"
- #include "amlos.h"
- #include "amlops.h"
- #include "namespce.h"
- #include "oprgn.h"
- //
- // ---------------------------------------------------------------- Definitions
- //
- #define ACPI_RESOURCE_END_TAG 0x79
- //
- // Define the longest string that can be created from converting a decimal
- // integer to a string.
- //
- #define MAX_DECIMAL_STRING_LENGTH 22
- //
- // Define the portion of the mutex sync flags that represent the sync level.
- //
- #define MUTEX_FLAG_SYNC_LEVEL_MASK 0xF
- //
- // Define the bitfields of the method flags byte.
- //
- #define METHOD_ARGUMENT_COUNT_MASK 0x7
- #define METHOD_SERIALIZED_FLAG 0x08
- #define METHOD_SYNC_LEVEL_SHIFT 4
- #define METHOD_SYNC_LEVEL_MASK (0xF << METHOD_SYNC_LEVEL_SHIFT)
- //
- // ------------------------------------------------------ Data Type Definitions
- //
- typedef enum _ACPI_MATCH_OPERATOR {
- MatchOperatorTrue = 0,
- MatchOperatorEqual = 1,
- MatchOperatorLessThanOrEqualTo = 2,
- MatchOperatorLessThan = 3,
- MatchOperatorGreaterThanOrEqualTo = 4,
- MatchOperatorGreaterThan = 5,
- MatchOperatorCount
- } ACPI_MATCH_OPERATOR, *PACPI_MATCH_OPERATOR;
- //
- // ----------------------------------------------- Internal Function Prototypes
- //
- PACPI_OBJECT
- AcpipConvertObjectTypeToInteger (
- PAML_EXECUTION_CONTEXT Context,
- PACPI_OBJECT Object
- );
- PACPI_OBJECT
- AcpipConvertObjectTypeToString (
- PAML_EXECUTION_CONTEXT Context,
- PACPI_OBJECT Object
- );
- PACPI_OBJECT
- AcpipConvertObjectTypeToBuffer (
- PAML_EXECUTION_CONTEXT Context,
- PACPI_OBJECT Object
- );
- BOOL
- AcpipEvaluateMatchComparison (
- PAML_EXECUTION_CONTEXT Context,
- PACPI_OBJECT PackageElement,
- PACPI_OBJECT Operand1,
- ACPI_MATCH_OPERATOR Operator1,
- PACPI_OBJECT Operand2,
- ACPI_MATCH_OPERATOR Operator2
- );
- //
- // -------------------------------------------------------------------- Globals
- //
- PSTR AcpiMatchOpcodeString[MatchOperatorCount] = {
- "MTR", // Always TRUE.
- "MEQ", // Equal to.
- "MLE", // Less than or Equal to.
- "MLT", // Less than.
- "MGE", // Greater than or equal to.
- "MGT" // Greater than.
- };
- //
- // ------------------------------------------------------------------ Functions
- //
- KSTATUS
- AcpipEvaluateAcquireStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates an Acquire (mutex) statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PUCHAR InstructionPointer;
- PACPI_OBJECT NewArgument;
- ULONGLONG ResultValue;
- USHORT TimeoutValue;
- //
- // Gather arguments if needed.
- //
- TimeoutValue = 0;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("Acquire (");
- }
- }
- //
- // An argument is required.
- //
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[0] = NULL;
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- if (NewArgument == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- Statement->Argument[0] = NewArgument;
- AcpipObjectAddReference(NewArgument);
- }
- Statement->ArgumentsAcquired += 1;
- //
- // The first argument should be acquired now, and the second argument
- // is a constant word representing the timeout value.
- //
- InstructionPointer = (PUCHAR)Context->AmlCode + Context->CurrentOffset;
- if (Context->CurrentOffset + sizeof(USHORT) >= Context->AmlCodeSize) {
- return STATUS_MALFORMED_DATA_STREAM;
- }
- TimeoutValue = *((PUSHORT)InstructionPointer);
- Context->CurrentOffset += sizeof(USHORT);
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint(", %d)", TimeoutValue);
- }
- }
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements != FALSE) {
- ASSERT(Statement->Argument[0]->Type == AcpiObjectMutex);
- ResultValue = AcpipAcquireMutex(Context,
- Statement->Argument[0]->U.Mutex.OsMutex,
- TimeoutValue);
- Statement->Reduction = AcpipCreateNamespaceObject(Context,
- AcpiObjectInteger,
- NULL,
- &ResultValue,
- sizeof(ULONGLONG));
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateAliasStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates the alias statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT Destination;
- ASSERT(Statement->Argument[0]->Type == AcpiObjectString);
- ASSERT(Statement->Argument[1]->Type == AcpiObjectString);
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("Alias (%s, %s)\n",
- Statement->Argument[0]->U.String.String,
- Statement->Argument[1]->U.String.String);
- }
- if (Context->ExecuteStatements != FALSE) {
- //
- // Get the destination object that the alias points to.
- //
- Destination = AcpipGetNamespaceObject(
- Statement->Argument[0]->U.String.String,
- Context->CurrentScope);
- if (Destination == NULL) {
- return STATUS_NOT_FOUND;
- }
- //
- // Create the alias.
- //
- Statement->Reduction = AcpipCreateNamespaceObject(
- Context,
- AcpiObjectAlias,
- Statement->Argument[1]->U.String.String,
- &Destination,
- sizeof(PVOID));
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateArgumentStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates the ArgX opcodes.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- STATUS_SUCCESS always.
- --*/
- {
- ULONG ArgumentNumber;
- PACPI_OBJECT ArgumentObject;
- ArgumentNumber = (ULONG)Statement->AdditionalData;
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("Arg%d", ArgumentNumber);
- }
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements != FALSE) {
- ArgumentObject = Context->CurrentMethod->Argument[ArgumentNumber];
- if (ArgumentObject != NULL) {
- Statement->Reduction = ArgumentObject;
- AcpipObjectAddReference(ArgumentObject);
- }
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateBankFieldStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a BankField (in an Operation Region) statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- STATUS_SUCCESS always.
- --*/
- {
- UCHAR AccessFlags;
- PUCHAR InstructionPointer;
- PACPI_OBJECT NewArgument;
- PACPI_OBJECT OperationRegion;
- KSTATUS Status;
- ASSERT(Statement->Argument[0]->Type == AcpiObjectString);
- ASSERT(Statement->Argument[1]->Type == AcpiObjectString);
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("BankField (%s, ",
- Statement->Argument[0]->U.String.String);
- } else {
- RtlDebugPrint(")");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- ASSERT(Statement->ArgumentsAcquired == 2);
- //
- // Perform a conversion if needed.
- //
- NewArgument = Context->PreviousStatement->Reduction;
- if (NewArgument == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- if (NewArgument->Type != AcpiObjectInteger) {
- NewArgument = AcpipConvertObjectType(Context,
- NewArgument,
- AcpiObjectInteger);
- if (NewArgument == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- } else {
- AcpipObjectAddReference(NewArgument);
- }
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- }
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- ASSERT(Statement->Argument[2]->Type == AcpiObjectInteger);
- //
- // Parse the starting flags and store them in additional data 2.
- //
- InstructionPointer = (PUCHAR)Context->AmlCode + Context->CurrentOffset;
- AccessFlags = *InstructionPointer;
- Context->CurrentOffset += 1;
- //
- // Get the operation region this Field refers to.
- //
- OperationRegion =
- AcpipGetNamespaceObject(Statement->Argument[0]->U.String.String,
- Context->CurrentScope);
- if (OperationRegion == NULL) {
- return STATUS_NOT_FOUND;
- }
- //
- // Parse the field list.
- //
- Status = AcpipParseFieldList(Context,
- Statement->Type,
- OperationRegion,
- Statement->Argument[1],
- Statement->Argument[2],
- NULL,
- NULL,
- Statement->AdditionalData,
- AccessFlags);
- return Status;
- }
- KSTATUS
- AcpipEvaluateBreakPointStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates the BreakPoint statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- STATUS_SUCCESS always.
- --*/
- {
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("BreakPoint");
- }
- if (Context->ExecuteStatements != FALSE) {
- RtlDebugBreak();
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateBufferStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a buffer declaration statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT Argument;
- PACPI_OBJECT BufferObject;
- ULONG BufferSize;
- ULONG ByteIndex;
- ULONG ByteListLength;
- PUCHAR ByteListPointer;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- if (Context->PreviousStatement == NULL) {
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("Buffer (");
- }
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- if (Context->ExecuteStatements != FALSE) {
- //
- // Convert the buffer size object to an integer if needed.
- //
- Argument = Context->PreviousStatement->Reduction;
- if (Argument == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- if (Argument->Type != AcpiObjectInteger) {
- Argument = AcpipConvertObjectType(Context,
- Argument,
- AcpiObjectInteger);
- } else {
- AcpipObjectAddReference(Argument);
- }
- Statement->Argument[0] = Argument;
- //
- // Just pretend the argument would have been there.
- //
- } else {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- }
- Statement->ArgumentsAcquired += 1;
- }
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- //
- // Collect the byte list following the buffer size argument just acquired.
- //
- ByteListLength = Statement->AdditionalData - Context->CurrentOffset;
- ByteListPointer = (PUCHAR)Context->AmlCode + Context->CurrentOffset;
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint(") {");
- for (ByteIndex = 0; ByteIndex < ByteListLength; ByteIndex += 1) {
- RtlDebugPrint("%02x ", ByteListPointer[ByteIndex]);
- }
- RtlDebugPrint("}");
- }
- if (Context->ExecuteStatements != FALSE) {
- ASSERT(Statement->Argument[0]->Type == AcpiObjectInteger);
- //
- // The buffer size comes from evaluating the argument. If the
- // initializer is bigger than the buffer size, then expand it to fit
- // the initializer.
- //
- BufferSize = Statement->Argument[0]->U.Integer.Value;
- if (BufferSize < ByteListLength) {
- BufferSize = ByteListLength;
- }
- //
- // If the buffer size is greater than the initializer, allocate and
- // initialize in two steps. Otherwise, pass the data directly.
- //
- if (BufferSize > ByteListLength) {
- BufferObject = AcpipCreateNamespaceObject(Context,
- AcpiObjectBuffer,
- NULL,
- NULL,
- BufferSize);
- if (BufferObject == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- RtlCopyMemory(BufferObject->U.Buffer.Buffer,
- ByteListPointer,
- ByteListLength);
- } else {
- BufferObject = AcpipCreateNamespaceObject(Context,
- AcpiObjectBuffer,
- NULL,
- ByteListPointer,
- ByteListLength);
- if (BufferObject == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- }
- Statement->Reduction = BufferObject;
- }
- //
- // Move the instruction pointer over the byte list.
- //
- Context->CurrentOffset = (ULONG)Statement->AdditionalData;
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateConcatenateResourceTemplatesStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a "Concatenate Resource Templates" statement, which
- concatenates two buffers that are resources templates. It automatically
- strips the end tags off the two, adds it to the concatenation, and calcuates
- the checksum.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT Argument0;
- ULONG Argument0Length;
- PACPI_OBJECT Argument1;
- ULONG Argument1Length;
- PUCHAR BytePointer;
- PACPI_OBJECT NewArgument;
- PACPI_OBJECT Result;
- UCHAR SumOfTemplate;
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("ConcatResTemplate (");
- } else if ((Statement->ArgumentsAcquired == 0) ||
- (Statement->ArgumentsAcquired == 1)) {
- RtlDebugPrint(", ");
- } else {
- RtlDebugPrint(")");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- if ((Statement->ArgumentsAcquired == 0) ||
- (Statement->ArgumentsAcquired == 1)) {
- //
- // Fail if there is no argument there.
- //
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- //
- // Only buffers are supported.
- //
- if (NewArgument->Type != AcpiObjectBuffer) {
- return STATUS_INVALID_PARAMETER;
- }
- }
- AcpipObjectAddReference(NewArgument);
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- }
- }
- //
- // Evaluate the result.
- //
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- Argument0 = Statement->Argument[0];
- Argument1 = Statement->Argument[1];
- ASSERT(Argument0->Type == AcpiObjectBuffer);
- ASSERT(Argument1->Type == AcpiObjectBuffer);
- //
- // Strip off the end tag of argument 0, if there is one.
- //
- Argument0Length = Argument0->U.Buffer.Length;
- if (Argument0Length < 2) {
- Argument0Length = 0;
- } else {
- BytePointer = (PUCHAR)Argument0->U.Buffer.Buffer +
- Argument0->U.Buffer.Length - 2;
- if (*BytePointer == ACPI_RESOURCE_END_TAG) {
- Argument0Length -= 2;
- }
- }
- //
- // Strip off argument 1's end tag.
- //
- Argument1Length = Argument1->U.Buffer.Length;
- if (Argument1Length < 2) {
- Argument1Length = 0;
- } else {
- BytePointer = (PUCHAR)Argument1->U.Buffer.Buffer +
- Argument1->U.Buffer.Length - 2;
- if (*BytePointer == ACPI_RESOURCE_END_TAG) {
- Argument1Length -= 2;
- }
- }
- //
- // Create the new buffer object with space for an end tag.
- //
- Result = AcpipCreateNamespaceObject(Context,
- AcpiObjectBuffer,
- NULL,
- NULL,
- Argument0Length + Argument1Length + 2);
- if (Result == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- //
- // Copy the templates over, back to back
- //
- RtlCopyMemory(Result->U.Buffer.Buffer,
- Argument0->U.Buffer.Buffer,
- Argument0Length);
- RtlCopyMemory(Result->U.Buffer.Buffer + Argument0Length,
- Argument1->U.Buffer.Buffer,
- Argument1Length);
- //
- // Slap a new end tag and checksum on that puppy.
- //
- BytePointer = (PUCHAR)(Result->U.Buffer.Buffer) +
- Argument0Length + Argument1Length;
- *BytePointer = ACPI_RESOURCE_END_TAG;
- BytePointer += 1;
- SumOfTemplate = AcpipChecksumData(Result->U.Buffer.Buffer,
- Argument0Length + Argument1Length + 1);
- *BytePointer = -SumOfTemplate;
- Statement->Reduction = Result;
- //
- // Store the result in the target if supplied.
- //
- if (Statement->Argument[2] != NULL) {
- return AcpipPerformStoreOperation(Context,
- Statement->Reduction,
- Statement->Argument[2]);
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateConcatenateStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a concatenate statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT Argument0;
- PACPI_OBJECT Argument1;
- PACPI_OBJECT NewArgument;
- ULONG NewLength;
- PACPI_OBJECT Result;
- ULONG String0Length;
- ULONG String1Length;
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("Concat (");
- } else if ((Statement->ArgumentsAcquired == 0) ||
- (Statement->ArgumentsAcquired == 1)) {
- RtlDebugPrint(", ");
- } else {
- RtlDebugPrint(")");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- if ((Statement->ArgumentsAcquired == 0) ||
- (Statement->ArgumentsAcquired == 1)) {
- //
- // Fail if there is no argument there.
- //
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- //
- // Only integers, strings, and buffers are supported.
- //
- if ((NewArgument->Type != AcpiObjectInteger) &&
- (NewArgument->Type != AcpiObjectString) &&
- (NewArgument->Type != AcpiObjectBuffer)) {
- return STATUS_INVALID_PARAMETER;
- }
- //
- // Perform an implicit conversion on the second argument (to the
- // type of the first argument) if needed.
- //
- if (Statement->ArgumentsAcquired == 0) {
- AcpipObjectAddReference(Statement->Argument[0]);
- } else if (Statement->ArgumentsAcquired == 1) {
- if (Statement->Argument[0]->Type != NewArgument->Type) {
- NewArgument = AcpipConvertObjectType(
- Context,
- NewArgument,
- Statement->Argument[0]->Type);
- if (NewArgument == NULL) {
- return STATUS_CONVERSION_FAILED;
- }
- //
- // No conversion is needed, just add to the reference count.
- //
- } else {
- AcpipObjectAddReference(NewArgument);
- }
- } else {
- ASSERT(Statement->ArgumentsAcquired == 2);
- AcpipObjectAddReference(NewArgument);
- }
- }
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- }
- }
- //
- // Evaluate the result.
- //
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- if (Context->ExecuteStatements == FALSE) {
- return STATUS_SUCCESS;
- }
- Argument0 = Statement->Argument[0];
- Argument1 = Statement->Argument[1];
- switch (Argument0->Type) {
- //
- // Two integers get put together and make a buffer.
- //
- case AcpiObjectInteger:
- NewLength = 2 * sizeof(ULONGLONG);
- if (Context->CurrentMethod->IntegerWidthIs32 != FALSE) {
- NewLength = 2 * sizeof(ULONG);
- }
- //
- // Create a new buffer.
- //
- Result = AcpipCreateNamespaceObject(Context,
- AcpiObjectBuffer,
- NULL,
- NULL,
- NewLength);
- if (Result == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- //
- // Copy the integers in like buffers.
- //
- RtlCopyMemory(Result->U.Buffer.Buffer,
- &(Argument0->U.Integer.Value),
- NewLength / 2);
- RtlCopyMemory(Result->U.Buffer.Buffer + NewLength / 2,
- &(Argument1->U.Integer.Value),
- NewLength / 2);
- break;
- //
- // Two buffers simply get glommed together.
- //
- case AcpiObjectBuffer:
- NewLength = Argument0->U.Buffer.Length + Argument1->U.Buffer.Length;
- Result = AcpipCreateNamespaceObject(Context,
- AcpiObjectBuffer,
- NULL,
- NULL,
- NewLength);
- if (Result == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- RtlCopyMemory(Result->U.Buffer.Buffer,
- Argument0->U.Buffer.Buffer,
- Argument0->U.Buffer.Length);
- RtlCopyMemory(Result->U.Buffer.Buffer + Argument0->U.Buffer.Length,
- Argument1->U.Buffer.Buffer,
- Argument1->U.Buffer.Length);
- break;
- //
- // Two strings get concatenated into another string.
- //
- case AcpiObjectString:
- String0Length = RtlStringLength(Argument0->U.String.String);
- String1Length = RtlStringLength(Argument1->U.String.String);
- NewLength = String0Length + String1Length + 1;
- Result = AcpipCreateNamespaceObject(Context,
- AcpiObjectString,
- NULL,
- NULL,
- NewLength);
- if (Result == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- RtlCopyMemory(Result->U.String.String,
- Argument0->U.String.String,
- String0Length);
- RtlCopyMemory(Result->U.String.String + String0Length,
- Argument1->U.String.String,
- String1Length);
- Result->U.String.String[NewLength - 1] = '\0';
- break;
- default:
- ASSERT(FALSE);
- return STATUS_CONVERSION_FAILED;
- }
- Statement->Reduction = Result;
- //
- // Store the result in the target if supplied.
- //
- if (Statement->Argument[2] != NULL) {
- return AcpipPerformStoreOperation(Context,
- Statement->Reduction,
- Statement->Argument[2]);
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateConditionalReferenceOfStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates an "Reference Of" statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT NameString;
- PACPI_OBJECT NewArgument;
- PACPI_OBJECT Reference;
- ULONGLONG ResultValue;
- //
- // Gather arguments if needed.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("CondRefOf (");
- } else if (Statement->ArgumentsAcquired == 0) {
- RtlDebugPrint(", ");
- } else {
- RtlDebugPrint(")");
- }
- }
- //
- // The argument for RefOf is a "SuperName", which is a SimpleName,
- // DebugOp, or Type6Opcode. If this is the first time through, try to
- // parse a name string.
- //
- if (Context->PreviousStatement == NULL) {
- NameString = AcpipParseNameString(Context);
- if (NameString == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- if (Context->ExecuteStatements != FALSE) {
- Statement->Argument[0] =
- AcpipGetNamespaceObject(NameString->U.String.String,
- Context->CurrentScope);
- } else {
- Statement->Argument[0] = NULL;
- }
- if (Statement->Argument[0] != NULL) {
- AcpipObjectAddReference(Statement->Argument[0]);
- }
- Statement->ArgumentsAcquired += 1;
- AcpipObjectReleaseReference(NameString);
- //
- // Increment the reference count on the object.
- //
- } else {
- if (Context->ExecuteStatements != FALSE) {
- NewArgument = Context->PreviousStatement->Reduction;
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- if (NewArgument != NULL) {
- AcpipObjectAddReference(NewArgument);
- }
- } else {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- }
- Statement->ArgumentsAcquired += 1;
- }
- if (Statement->ArgumentsAcquired != Statement->ArgumentsNeeded) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- }
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- if (Context->ExecuteStatements != FALSE) {
- //
- // The reduction of this statement is a boolean indicating whether the
- // object actually exists or not.
- //
- ResultValue = FALSE;
- if ((Statement->Argument[0] != NULL) &&
- (Statement->Argument[0]->Type != AcpiObjectUninitialized)) {
- ResultValue = TRUE;
- }
- Statement->Reduction = AcpipCreateNamespaceObject(Context,
- AcpiObjectInteger,
- NULL,
- &ResultValue,
- sizeof(ULONGLONG));
- if ((Statement->Argument[0] != NULL) &&
- (Statement->Argument[1] != NULL)) {
- Reference = AcpipCreateNamespaceObject(Context,
- AcpiObjectAlias,
- NULL,
- &(Statement->Argument[0]),
- sizeof(PACPI_OBJECT));
- if (Reference == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- AcpipPerformStoreOperation(Context,
- Reference,
- Statement->Argument[1]);
- AcpipObjectReleaseReference(Reference);
- }
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateCopyObjectStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a "Copy Object" statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT Destination;
- PACPI_OBJECT NewArgument;
- PACPI_OBJECT Source;
- KSTATUS Status;
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("CopyObject (");
- } else if (Statement->ArgumentsAcquired == 0) {
- RtlDebugPrint(", ");
- } else {
- RtlDebugPrint(")");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- ASSERT(Statement->ArgumentsAcquired == 0);
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- AcpipObjectAddReference(NewArgument);
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- }
- }
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements != FALSE) {
- Source = Statement->Argument[0];
- //
- // This needs to perform an implicit source converstion into a
- // DataRefObject (i.e. a DataObject or Reference Object).
- //
- Status = AcpipConvertToDataReferenceObject(Context, Source, &Source);
- if (!KSUCCESS(Status)) {
- goto EvaluateCopyObjectStatementEnd;
- }
- Statement->Reduction = AcpipCopyObject(Source);
- AcpipObjectReleaseReference(Source);
- //
- // If the target is supplied, replace it with the copy.
- //
- Destination = Statement->Argument[1];
- if (Destination != NULL) {
- Status = AcpipResolveStoreDestination(Context,
- Destination,
- &Destination);
- if (!KSUCCESS(Status)) {
- goto EvaluateCopyObjectStatementEnd;
- }
- Status = AcpipReplaceObjectContents(Context,
- Destination,
- Statement->Reduction);
- AcpipObjectReleaseReference(Destination);
- if (!KSUCCESS(Status)) {
- goto EvaluateCopyObjectStatementEnd;
- }
- }
- }
- Status = STATUS_SUCCESS;
- EvaluateCopyObjectStatementEnd:
- return Status;
- }
- KSTATUS
- AcpipEvaluateCreateBufferFieldStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a CreateField (from a buffer) statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- ACPI_BUFFER_FIELD_OBJECT BufferField;
- PSTR Name;
- PACPI_OBJECT NewArgument;
- ACPI_OBJECT_TYPE ObjectType;
- NewArgument = NULL;
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("CreateField (");
- } else if ((Statement->ArgumentsAcquired == 0) ||
- (Statement->ArgumentsAcquired == 1) ||
- (Statement->ArgumentsAcquired == 2)) {
- RtlDebugPrint(", ");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- if ((Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) &&
- (Statement->ArgumentsAcquired != 3)) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- //
- // Fail if there is no argument there.
- //
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- //
- // The first required argument is a buffer, the second is the
- // bit index (Integer), and the third is the bit length (Integer).
- //
- ObjectType = AcpiObjectBuffer;
- if ((Statement->ArgumentsAcquired == 1) ||
- (Statement->ArgumentsAcquired == 2)) {
- ObjectType = AcpiObjectInteger;
- }
- //
- // Perform an implicit conversion if needed.
- //
- if (NewArgument->Type != ObjectType) {
- NewArgument = AcpipConvertObjectType(Context,
- NewArgument,
- ObjectType);
- if (NewArgument == NULL) {
- return STATUS_CONVERSION_FAILED;
- }
- //
- // The object is fine, take ownership of it.
- //
- } else {
- AcpipObjectAddReference(NewArgument);
- }
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsAcquired != 3) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- }
- //
- // The fourth argument is a name string, which can be parsed now.
- //
- ASSERT(Statement->ArgumentsAcquired == 3);
- Statement->Argument[3] = AcpipParseNameString(Context);
- if (Statement->Argument[3] == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- Statement->ArgumentsAcquired += 1;
- }
- //
- // Evaluate the result.
- //
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- ASSERT(Statement->Argument[3]->Type == AcpiObjectString);
- Name = Statement->Argument[3]->U.String.String;
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("%s)", Name);
- }
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements == FALSE) {
- return STATUS_SUCCESS;
- }
- ASSERT(Statement->Argument[0]->Type == AcpiObjectBuffer);
- ASSERT(Statement->Argument[1]->Type == AcpiObjectInteger);
- ASSERT(Statement->Argument[2]->Type == AcpiObjectInteger);
- //
- // Create the buffer field object. Remember that additional data holds the
- // bit field length.
- //
- RtlZeroMemory(&BufferField, sizeof(ACPI_BUFFER_FIELD_OBJECT));
- BufferField.DestinationObject = Statement->Argument[0];
- BufferField.BitOffset = Statement->Argument[1]->U.Integer.Value;
- BufferField.BitLength = Statement->Argument[2]->U.Integer.Value;
- Statement->Reduction = AcpipCreateNamespaceObject(
- Context,
- AcpiObjectBufferField,
- Name,
- &BufferField,
- sizeof(ACPI_BUFFER_FIELD_OBJECT));
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateCreateFixedBufferFieldStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a CreateBitField, CreateByteField, CreateWordField,
- CreateDWordField, or CreateQWordField statement, which creates a Buffer
- Field object pointing at a buffer.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- ACPI_BUFFER_FIELD_OBJECT BufferField;
- PSTR Name;
- PACPI_OBJECT NewArgument;
- ACPI_OBJECT_TYPE ObjectType;
- NewArgument = NULL;
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- switch (Statement->AdditionalData) {
- case 1:
- RtlDebugPrint("CreateBitField (");
- break;
- case BITS_PER_BYTE:
- RtlDebugPrint("CreateByteField (");
- break;
- case sizeof(USHORT) * BITS_PER_BYTE:
- RtlDebugPrint("CreateWordField (");
- break;
- case sizeof(ULONG) * BITS_PER_BYTE:
- RtlDebugPrint("CreateDWordField (");
- break;
- case sizeof(ULONGLONG) * BITS_PER_BYTE:
- RtlDebugPrint("CreateQWordField (");
- break;
- default:
- ASSERT(FALSE);
- return STATUS_NOT_SUPPORTED;
- }
- } else if ((Statement->ArgumentsAcquired == 0) ||
- (Statement->ArgumentsAcquired == 1)) {
- RtlDebugPrint(", ");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- ASSERT(Statement->ArgumentsAcquired != 2);
- //
- // If not executing, then assume the argument would be there but
- // don't try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- //
- // Grab the first or second argument.
- //
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- //
- // Fail if there is no argument there.
- //
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- //
- // The first required argument is a buffer, and the second is the
- // bit index (Integer).
- //
- ObjectType = AcpiObjectBuffer;
- if (Statement->ArgumentsAcquired == 1) {
- ObjectType = AcpiObjectInteger;
- }
- //
- // Perform an implicit conversion if needed.
- //
- if (NewArgument->Type != ObjectType) {
- NewArgument = AcpipConvertObjectType(Context,
- NewArgument,
- ObjectType);
- if (NewArgument == NULL) {
- return STATUS_CONVERSION_FAILED;
- }
- //
- // The object is fine, take ownership of it.
- //
- } else {
- AcpipObjectAddReference(NewArgument);
- }
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- }
- //
- // If only the first argument has been parsed, then another one is
- // needed. If two have, then continue to parse the third.
- //
- if (Statement->ArgumentsAcquired == 1) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // The third argument is a name string, which can be parsed now.
- //
- if (Statement->ArgumentsAcquired == 2) {
- ASSERT(Statement->ArgumentsAcquired == 2);
- Statement->Argument[2] = AcpipParseNameString(Context);
- if (Statement->Argument[2] == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- Statement->ArgumentsAcquired += 1;
- }
- }
- //
- // Evaluate the result.
- //
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- ASSERT(Statement->Argument[2]->Type == AcpiObjectString);
- Name = Statement->Argument[2]->U.String.String;
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("%s)", Name);
- }
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements == FALSE) {
- return STATUS_SUCCESS;
- }
- ASSERT(Statement->Argument[0]->Type == AcpiObjectBuffer);
- ASSERT(Statement->Argument[1]->Type == AcpiObjectInteger);
- //
- // Create the buffer field object. Remember that additional data holds the
- // bit field length.
- //
- RtlZeroMemory(&BufferField, sizeof(ACPI_BUFFER_FIELD_OBJECT));
- BufferField.DestinationObject = Statement->Argument[0];
- BufferField.BitLength = Statement->AdditionalData;
- BufferField.BitOffset = Statement->Argument[1]->U.Integer.Value;
- //
- // Bitfields are specified in bits, but all other sized fields are
- // specified in bytes.
- //
- if (BufferField.BitLength > 1) {
- BufferField.BitOffset *= BITS_PER_BYTE;
- }
- Statement->Reduction = AcpipCreateNamespaceObject(
- Context,
- AcpiObjectBufferField,
- Name,
- &BufferField,
- sizeof(ACPI_BUFFER_FIELD_OBJECT));
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateDataStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates constant data coming from ByteData, WordData,
- DWordData, QWordData, and StringData.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PUCHAR DataPointer;
- DataPointer = (PUCHAR)Context->AmlCode + Statement->AdditionalData;
- //
- // For string data, just create the string from the buffer pointer. A copy
- // will be made.
- //
- if (Statement->AdditionalData2 == 0) {
- Statement->Reduction = AcpipCreateNamespaceObject(
- Context,
- AcpiObjectString,
- NULL,
- DataPointer,
- RtlStringLength((PSTR)DataPointer) + 1);
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("\"");
- RtlDebugPrint(Statement->Reduction->U.String.String);
- RtlDebugPrint("\"");
- }
- //
- // The other types are integers.
- //
- } else {
- Statement->Reduction = AcpipCreateNamespaceObject(Context,
- AcpiObjectInteger,
- NULL,
- NULL,
- 0);
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- Statement->Reduction->U.Integer.Value = 0;
- RtlCopyMemory(&(Statement->Reduction->U.Integer.Value),
- DataPointer,
- Statement->AdditionalData2);
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("0x%I64x", Statement->Reduction->U.Integer.Value);
- }
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateDelayStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates both the Sleep and Stall statements.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT NewArgument;
- ULONG Operand;
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- if (Statement->Type == AmlStatementSleep) {
- RtlDebugPrint("Sleep (");
- } else {
- ASSERT(Statement->Type == AmlStatementStall);
- RtlDebugPrint("Stall (");
- }
- } else {
- RtlDebugPrint(")");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- //
- // Fail if there is no argument there.
- //
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- //
- // Perform an implicit conversion if needed.
- //
- if (NewArgument->Type != AcpiObjectInteger) {
- NewArgument = AcpipConvertObjectType(Context,
- NewArgument,
- AcpiObjectInteger);
- if (NewArgument == NULL) {
- return STATUS_CONVERSION_FAILED;
- }
- //
- // The object is fine, take ownership of it.
- //
- } else {
- AcpipObjectAddReference(NewArgument);
- }
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- }
- }
- //
- // Evaluate the result.
- //
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements == FALSE) {
- return STATUS_SUCCESS;
- }
- //
- // For sleep statements, use the scheduler and relinquish the processor.
- //
- ASSERT(Statement->Argument[0]->Type == AcpiObjectInteger);
- Operand = (ULONG)Statement->Argument[0]->U.Integer.Value;
- if (Statement->Type == AmlStatementSleep) {
- AcpipSleep(Operand);
- //
- // For stall statements, perform a busy spin.
- //
- } else {
- ASSERT(Statement->Type == AmlStatementStall);
- AcpipBusySpin(Operand);
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateDebugStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a Debug statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("Debug");
- }
- if (Context->ExecuteStatements != FALSE) {
- //
- // Create a debug object. Simple as that.
- //
- Statement->Reduction = AcpipCreateNamespaceObject(Context,
- AcpiObjectDebug,
- NULL,
- NULL,
- 0);
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateDereferenceOfStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a "Dereference Of" statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT Source;
- KSTATUS Status;
- //
- // Gather arguments.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("DerefOf (");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- } else {
- Statement->Argument[0] = NULL;
- if (Context->ExecuteStatements != FALSE) {
- Statement->Argument[0] = Context->PreviousStatement->Reduction;
- AcpipObjectAddReference(Statement->Argument[0]);
- }
- Statement->ArgumentsAcquired += 1;
- }
- }
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint(")");
- }
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements != FALSE) {
- Source = Statement->Argument[0];
- switch (Source->Type) {
- case AcpiObjectAlias:
- Statement->Reduction = Source->U.Alias.DestinationObject;
- AcpipObjectAddReference(Statement->Reduction);
- break;
- case AcpiObjectString:
- Statement->Reduction =
- AcpipGetNamespaceObject(Source->U.String.String,
- Context->CurrentScope);
- if (Statement->Reduction == NULL) {
- return STATUS_NOT_FOUND;
- }
- AcpipObjectAddReference(Statement->Reduction);
- break;
- case AcpiObjectBufferField:
- Status = AcpipReadFromBufferField(Context,
- Source,
- &(Statement->Reduction));
- if (!KSUCCESS(Status)) {
- return Status;
- }
- break;
- default:
- ASSERT(FALSE);
- return STATUS_UNEXPECTED_TYPE;
- }
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateDeviceStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a Device declaration statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PSTR DeviceName;
- PACPI_OBJECT DeviceObject;
- if (Context->PreviousStatement == NULL) {
- Statement->SavedScope = NULL;
- ASSERT(Statement->Argument[0]->Type == AcpiObjectString);
- DeviceName = Statement->Argument[0]->U.String.String;
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("Device (%s) {", DeviceName);
- }
- if (Context->ExecuteStatements != FALSE) {
- //
- // Create the device object.
- //
- DeviceObject = AcpipCreateNamespaceObject(Context,
- AcpiObjectDevice,
- DeviceName,
- NULL,
- 0);
- if (DeviceObject == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- //
- // Make this device the current scope.
- //
- Statement->SavedScope = Context->CurrentScope;
- Context->CurrentScope = DeviceObject;
- Statement->Reduction = DeviceObject;
- }
- Context->IndentationLevel += 1;
- }
- //
- // If execution is not done with the scope, keep this statement on the
- // stack.
- //
- if (Context->CurrentOffset < Statement->AdditionalData) {
- AcpipPrintIndentedNewLine(Context);
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // Pop this puppy off the stack.
- //
- Context->CurrentScope = Statement->SavedScope;
- Context->IndentationLevel -= 1;
- if (Context->PrintStatements != FALSE) {
- AcpipPrintIndentedNewLine(Context);
- RtlDebugPrint("}");
- AcpipPrintIndentedNewLine(Context);
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateDivideStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a divide statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- ULONGLONG Dividend;
- ULONGLONG Divisor;
- PACPI_OBJECT NewArgument;
- ULONGLONG Quotient;
- ULONGLONG Remainder;
- PACPI_OBJECT RemainderObject;
- KSTATUS Status;
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("Divide (");
- } else if ((Statement->ArgumentsAcquired == 0) ||
- (Statement->ArgumentsAcquired == 1) ||
- (Statement->ArgumentsAcquired == 2)) {
- RtlDebugPrint(", ");
- } else {
- RtlDebugPrint(")");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- if ((Statement->ArgumentsAcquired == 0) ||
- (Statement->ArgumentsAcquired == 1)) {
- //
- // Fail if there is no argument there.
- //
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- //
- // Perform an implicit conversion if needed.
- //
- if (NewArgument->Type != AcpiObjectInteger) {
- NewArgument = AcpipConvertObjectType(Context,
- NewArgument,
- AcpiObjectInteger);
- if (NewArgument == NULL) {
- return STATUS_CONVERSION_FAILED;
- }
- //
- // The object is fine, take ownership of it.
- //
- } else {
- AcpipObjectAddReference(NewArgument);
- }
- } else {
- ASSERT((Statement->ArgumentsAcquired == 2) ||
- (Statement->ArgumentsAcquired == 3));
- AcpipObjectAddReference(NewArgument);
- }
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsAcquired < 3) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- }
- }
- //
- // Evaluate the result.
- //
- if (Context->ExecuteStatements == FALSE) {
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- return STATUS_SUCCESS;
- }
- //
- // The first time around, perform the divide and try to store the remainder.
- //
- if (Statement->ArgumentsAcquired == 3) {
- ASSERT(Statement->Argument[0]->Type == AcpiObjectInteger);
- ASSERT(Statement->Argument[1]->Type == AcpiObjectInteger);
- Dividend = Statement->Argument[0]->U.Integer.Value;
- Divisor = Statement->Argument[1]->U.Integer.Value;
- if (Context->CurrentMethod->IntegerWidthIs32 != FALSE) {
- Dividend &= 0xFFFFFFFF;
- Dividend &= 0xFFFFFFFF;
- }
- //
- // Fail to divide by 0, otherwise do the divide.
- //
- if (Divisor == 0) {
- return STATUS_DIVIDE_BY_ZERO;
- }
- Quotient = Dividend / Divisor;
- Remainder = Dividend % Divisor;
- if (Context->CurrentMethod->IntegerWidthIs32 != FALSE) {
- Quotient &= 0xFFFFFFFF;
- Remainder &= 0xFFFFFFFF;
- }
- Statement->Reduction = AcpipCreateNamespaceObject(Context,
- AcpiObjectInteger,
- NULL,
- &Quotient,
- sizeof(ULONGLONG));
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- //
- // Store the remainder if supplied.
- //
- if (Statement->Argument[2] != NULL) {
- RemainderObject = AcpipCreateNamespaceObject(Context,
- AcpiObjectInteger,
- NULL,
- &Remainder,
- sizeof(ULONGLONG));
- if (RemainderObject == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- Status = AcpipPerformStoreOperation(Context,
- RemainderObject,
- Statement->Argument[2]);
- AcpipObjectReleaseReference(RemainderObject);
- if (!KSUCCESS(Status)) {
- return Status;
- }
- }
- ASSERT(Statement->ArgumentsAcquired < Statement->ArgumentsNeeded);
- Status = STATUS_MORE_PROCESSING_REQUIRED;
- //
- // The second time around store the quotient.
- //
- } else {
- ASSERT(Statement->ArgumentsAcquired == 4);
- //
- // Store the quotient in the target if supplied.
- //
- if (Statement->Argument[3] != NULL) {
- return AcpipPerformStoreOperation(Context,
- Statement->Reduction,
- Statement->Argument[3]);
- }
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateElseStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates an Else statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- //
- // If this is the first time through, up the indentation level.
- //
- if (Context->PreviousStatement == NULL) {
- Context->IndentationLevel += 1;
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("Else {");
- }
- }
- if (Context->ExecuteStatements != FALSE) {
- //
- // Check the result of the last If statement. Skip over the Else if the
- // IF succeeded.
- //
- if (Context->LastIfStatementResult != FALSE) {
- Context->CurrentOffset = Statement->AdditionalData;
- }
- }
- //
- // If execution is not done with the scope, keep this statement on the
- // stack.
- //
- if (Context->CurrentOffset < Statement->AdditionalData) {
- AcpipPrintIndentedNewLine(Context);
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- Context->IndentationLevel -= 1;
- if (Context->PrintStatements != FALSE) {
- AcpipPrintIndentedNewLine(Context);
- RtlDebugPrint("}");
- }
- AcpipPrintIndentedNewLine(Context);
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateEventStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates an Event (creation) statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PSTR Name;
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- ASSERT(Statement->Argument[0]->Type == AcpiObjectString);
- Name = Statement->Argument[0]->U.String.String;
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("Event (%s)", Name);
- }
- if (Context->ExecuteStatements != FALSE) {
- Statement->Reduction = AcpipCreateNamespaceObject(Context,
- AcpiObjectEvent,
- Name,
- NULL,
- 0);
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateExecutingMethodStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates an Executing Method statement. This does not
- translate to a real ACPI opcode, but is a dummy object placed on the
- currently-executing statement stack so that return statements know how
- far to pop back up.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT Method;
- PAML_STATEMENT PreviousStatement;
- KSTATUS Status;
- ULONGLONG Zero;
- ASSERT(Context->ExecuteStatements != FALSE);
- //
- // If the method context to wait for is set, wait until the method context
- // comes back to the original routine.
- //
- if (Statement->AdditionalData2 != (UINTN)NULL) {
- if ((UINTN)Context->CurrentMethod == Statement->AdditionalData2) {
- Statement->Reduction = Context->ReturnValue;
- AcpipObjectAddReference(Statement->Reduction);
- Context->IndentationLevel -= 1;
- if (Context->PrintStatements != FALSE) {
- AcpipPrintIndentedNewLine(Context);
- RtlDebugPrint("}");
- }
- return STATUS_SUCCESS;
- }
- //
- // While not in the spec, folklore has it that an old version of the
- // Windows AML interpreter allowed for AML methods without return
- // statements (even though ACPI said it was required). The behavior
- // instead was that the function returned a constant zero integer.
- // Many BIOSes took advantage of that, so now it basically is part
- // of the spec. If the current function seems to have just finished,
- // then pop its context.
- //
- if (Context->CurrentOffset == Context->AmlCodeSize) {
- if (Context->ReturnValue == NULL) {
- Zero = 0;
- Context->ReturnValue =
- AcpipCreateNamespaceObject(Context,
- AcpiObjectInteger,
- NULL,
- &Zero,
- sizeof(ULONGLONG));
- }
- AcpipPopExecutingStatements(Context, FALSE, FALSE);
- AcpipPopCurrentMethodContext(Context);
- Context->IndentationLevel -= 1;
- if (Context->PrintStatements != FALSE) {
- AcpipPrintIndentedNewLine(Context);
- RtlDebugPrint("}");
- }
- return STATUS_SUCCESS;
- }
- AcpipPrintIndentedNewLine(Context);
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- PreviousStatement = Context->PreviousStatement;
- //
- // The evaluate executing method statement is special in that it will not
- // get called once with a previous statement of NULL. Since it is a side
- // effect of another statement spitting out a reduction of type method, this
- // statement always gets passed a previous statement. The first time it's
- // called, the previous statement should have a pointer to the method
- // object. Use that to determine the argument count. Additional data was
- // initialized to 0 to indicate the first time this statement is being
- // evaluated.
- //
- if (Statement->AdditionalData == (UINTN)NULL) {
- ASSERT((PreviousStatement != NULL) &&
- (PreviousStatement->Reduction != NULL) &&
- (PreviousStatement->Reduction->Type == AcpiObjectMethod));
- Method = PreviousStatement->Reduction;
- Statement->AdditionalData = (UINTN)Method;
- Statement->ArgumentsNeeded = Method->U.Method.ArgumentCount;
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint(" (");
- }
- if (Statement->ArgumentsNeeded != 0) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- }
- //
- // If not all arguments are acquired, wait for them to come in, and collect
- // 'em.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- if ((PreviousStatement == NULL) ||
- (PreviousStatement->Reduction == NULL)) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- Statement->Argument[Statement->ArgumentsAcquired] =
- PreviousStatement->Reduction;
- Statement->ArgumentsAcquired += 1;
- AcpipObjectAddReference(PreviousStatement->Reduction);
- }
- //
- // If all arguments are still not acquired, wait for more.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint(", ");
- }
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- Context->IndentationLevel += 1;
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint(") {");
- AcpipPrintIndentedNewLine(Context);
- }
- //
- // Store the current method context in Additional Data 2, and use that
- // to determine when to complete this statement (complete the statement
- // when this context comes back.
- //
- Statement->AdditionalData2 = (UINTN)Context->CurrentMethod;
- //
- // Push the method execution context on as the current context.
- //
- Method = (PVOID)(UINTN)Statement->AdditionalData;
- Status = AcpipPushMethodOnExecutionContext(
- Context,
- Method,
- Method->U.Method.OsMutex,
- Method->U.Method.IntegerWidthIs32,
- Method->U.Method.AmlCode,
- Method->U.Method.AmlCodeSize,
- Statement->ArgumentsNeeded,
- Statement->Argument);
- if (!KSUCCESS(Status)) {
- return Status;
- }
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- KSTATUS
- AcpipEvaluateFatalStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a fatal execution statement. This will stop the
- operating system.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PUCHAR DataPointer;
- ULONGLONG FatalArgument;
- ULONG FatalCode;
- UCHAR FatalType;
- PACPI_OBJECT NewArgument;
- DataPointer = (PUCHAR)Context->AmlCode + Statement->AdditionalData;
- FatalType = *DataPointer;
- FatalCode = *(PULONG)(DataPointer + 1);
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("Fatal (%x, %x, ", FatalType, FatalCode);
- } else {
- RtlDebugPrint(")");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- ASSERT(Statement->ArgumentsAcquired == 0);
- //
- // Convert the argument if it is there. The argument is
- // technically required, but since this is a fatal error, be a
- // bit more forgiving.
- //
- if (Context->PreviousStatement->Reduction != NULL) {
- //
- // Perform an implicit conversion if needed.
- //
- if (NewArgument->Type != AcpiObjectInteger) {
- NewArgument = AcpipConvertObjectType(Context,
- NewArgument,
- AcpiObjectInteger);
- //
- // The object is fine, take ownership of it.
- //
- } else {
- AcpipObjectAddReference(NewArgument);
- }
- }
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- }
- }
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- //
- // Die... so sad.
- //
- FatalArgument = 0;
- if ((Statement->Argument[0] != NULL) &&
- (Statement->Argument[0]->Type == AcpiObjectInteger)) {
- FatalArgument = Statement->Argument[0]->U.Integer.Value;
- }
- RtlDebugPrint("\n\n*** ACPI Fatal Error ***\n"
- "Type: 0x%x, Code: 0x%x, Argument: 0x%I64x\n"
- "Execution Context: 0x%x\n",
- FatalType,
- FatalCode,
- FatalArgument,
- Context);
- AcpipFatalError(ACPI_CRASH_FATAL_INSTRUCTION,
- FatalType,
- FatalCode,
- FatalArgument);
- //
- // Execution will never get here, but the compiler will get sassy if there's
- // just nothing.
- //
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateFieldStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a Field (in an Operation Region) statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- STATUS_SUCCESS always.
- --*/
- {
- PACPI_OBJECT OperationRegion;
- KSTATUS Status;
- OperationRegion = NULL;
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- ASSERT(Statement->Argument[0]->Type == AcpiObjectString);
- if (Context->ExecuteStatements != FALSE) {
- //
- // Get the operation region this Field refers to.
- //
- OperationRegion = AcpipGetNamespaceObject(
- Statement->Argument[0]->U.String.String,
- Context->CurrentScope);
- if (OperationRegion == NULL) {
- return STATUS_NOT_FOUND;
- }
- }
- //
- // Parse the field list.
- //
- Status = AcpipParseFieldList(Context,
- Statement->Type,
- OperationRegion,
- NULL,
- NULL,
- NULL,
- NULL,
- Statement->AdditionalData,
- (UCHAR)Statement->AdditionalData2);
- return Status;
- }
- KSTATUS
- AcpipEvaluateFindSetBitStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a find set left bit or find set right bit statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- ULONGLONG Mask;
- PACPI_OBJECT NewArgument;
- ULONGLONG Result;
- ULONGLONG Value;
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- switch (Statement->Type) {
- case AmlStatementFindSetLeftBit:
- RtlDebugPrint("FindSetLeftBit (");
- break;
- case AmlStatementFindSetRightBit:
- RtlDebugPrint("FindSetRightBit (");
- break;
- default:
- ASSERT(FALSE);
- return STATUS_NOT_SUPPORTED;
- }
- } else if (Statement->ArgumentsAcquired == 0) {
- RtlDebugPrint(", ");
- } else {
- RtlDebugPrint(")");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- if (Statement->ArgumentsAcquired == 0) {
- //
- // Fail if there is no argument there.
- //
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- //
- // Perform an implicit conversion if needed.
- //
- if (NewArgument->Type != AcpiObjectInteger) {
- NewArgument = AcpipConvertObjectType(Context,
- NewArgument,
- AcpiObjectInteger);
- if (NewArgument == NULL) {
- return STATUS_CONVERSION_FAILED;
- }
- //
- // The object is fine, take ownership of it.
- //
- } else {
- AcpipObjectAddReference(NewArgument);
- }
- } else {
- ASSERT(Statement->ArgumentsAcquired == 1);
- AcpipObjectAddReference(NewArgument);
- }
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- }
- }
- //
- // Evaluate the result.
- //
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements == FALSE) {
- return STATUS_SUCCESS;
- }
- ASSERT(Statement->Argument[0]->Type == AcpiObjectInteger);
- Value = Statement->Argument[0]->U.Integer.Value;
- if (Value != 0) {
- if (Statement->Type == AmlStatementFindSetLeftBit) {
- Result = 64;
- Mask = 0x8000000000000000ULL;
- if (Context->CurrentMethod->IntegerWidthIs32 != FALSE) {
- Result = 32;
- Mask = 0x80000000;
- ASSERT(Value <= MAX_ULONG);
- }
- while ((Value & Mask) == 0) {
- Value = Value << 1;
- Result -= 1;
- }
- } else {
- ASSERT(Statement->Type == AmlStatementFindSetRightBit);
- Mask = 1;
- Result = 1;
- while ((Value & Mask) == 0) {
- Value = Value >> 1;
- Result += 1;
- }
- }
- }
- Statement->Reduction = AcpipCreateNamespaceObject(Context,
- AcpiObjectInteger,
- NULL,
- &Result,
- sizeof(ULONGLONG));
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- //
- // Store the result in the target if supplied.
- //
- if (Statement->Argument[1] != NULL) {
- return AcpipPerformStoreOperation(Context,
- Statement->Reduction,
- Statement->Argument[1]);
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateIfStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates an If statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT NewArgument;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("If (");
- } else {
- ASSERT(Statement->ArgumentsAcquired == 0);
- RtlDebugPrint(") {");
- }
- }
- if (Context->PreviousStatement == NULL) {
- Context->IndentationLevel += 1;
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- //
- // Fail if there is no argument there.
- //
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- //
- // Perform an implicit conversion if needed.
- //
- if (NewArgument->Type != AcpiObjectInteger) {
- NewArgument = AcpipConvertObjectType(Context,
- NewArgument,
- AcpiObjectInteger);
- if (NewArgument == NULL) {
- return STATUS_CONVERSION_FAILED;
- }
- //
- // The object is fine, take ownership of it.
- //
- } else {
- AcpipObjectAddReference(NewArgument);
- }
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- }
- Statement->ArgumentsAcquired += 1;
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- if (Context->ExecuteStatements != FALSE) {
- ASSERT(Statement->Argument[0]->Type == AcpiObjectInteger);
- //
- // Evaluate the if statement by skipping the package length if it's
- // zero.
- //
- Statement->AdditionalData2 = TRUE;
- if (Statement->Argument[0]->U.Integer.Value == 0) {
- Statement->AdditionalData2 = FALSE;
- Context->CurrentOffset = Statement->AdditionalData;
- }
- }
- }
- //
- // If execution is not done with the scope, keep this statement on the
- // stack.
- //
- if (Context->CurrentOffset < Statement->AdditionalData) {
- AcpipPrintIndentedNewLine(Context);
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- Context->IndentationLevel -= 1;
- if (Context->PrintStatements != FALSE) {
- AcpipPrintIndentedNewLine(Context);
- RtlDebugPrint("}");
- }
- AcpipPrintIndentedNewLine(Context);
- //
- // Save the result of the If statement into the context so that an Else
- // can be properly evaluated if it's coming up next.
- //
- if (Statement->AdditionalData2 != FALSE) {
- Context->LastIfStatementResult = TRUE;
- } else {
- Context->LastIfStatementResult = FALSE;
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateIncrementOrDecrementStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates an Increment or Decrement statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT IntegerObject;
- PACPI_OBJECT NewArgument;
- //
- // Gather arguments if needed.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- if (Statement->Type == AmlStatementIncrement) {
- RtlDebugPrint("Increment (");
- } else {
- ASSERT(Statement->Type == AmlStatementDecrement);
- RtlDebugPrint("Decrement (");
- }
- } else {
- RtlDebugPrint(")");
- }
- }
- //
- // If there is no previous statement, wait for the argument to come in.
- //
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, assume the argument would be there, and move on.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- //
- // Increment the reference count on the object (assuming it's there).
- //
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- if (NewArgument == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- AcpipObjectAddReference(NewArgument);
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- }
- }
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- if (Context->ExecuteStatements != FALSE) {
- //
- // Perform an implicit conversion if necessary.
- //
- if (Statement->Argument[0]->Type != AcpiObjectInteger) {
- IntegerObject = AcpipConvertObjectType(Context,
- Statement->Argument[0],
- AcpiObjectInteger);
- if (IntegerObject == NULL) {
- return STATUS_CONVERSION_FAILED;
- }
- } else {
- IntegerObject = Statement->Argument[0];
- AcpipObjectAddReference(IntegerObject);
- }
- //
- // Do the increment or decrement.
- //
- if (Statement->Type == AmlStatementIncrement) {
- IntegerObject->U.Integer.Value += 1;
- } else {
- IntegerObject->U.Integer.Value -= 1;
- }
- //
- // Store the result back if this is not the argument. This also implies
- // a conversion back to the original type is necessary.
- //
- Statement->Reduction = IntegerObject;
- if (IntegerObject != Statement->Argument[0]) {
- AcpipPerformStoreOperation(Context,
- IntegerObject,
- Statement->Argument[0]);
- }
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateIndexFieldStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates an IndexField (in an Operation Region) statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- STATUS_SUCCESS always.
- --*/
- {
- PACPI_OBJECT DataRegister;
- PACPI_OBJECT IndexRegister;
- KSTATUS Status;
- IndexRegister = NULL;
- DataRegister = NULL;
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- ASSERT(Statement->Argument[0]->Type == AcpiObjectString);
- ASSERT(Statement->Argument[1]->Type == AcpiObjectString);
- if (Context->ExecuteStatements != FALSE) {
- //
- // Get the Index field object.
- //
- IndexRegister = AcpipGetNamespaceObject(
- Statement->Argument[0]->U.String.String,
- Context->CurrentScope);
- if (IndexRegister == NULL) {
- return STATUS_NOT_FOUND;
- }
- //
- // Get the Data field object.
- //
- DataRegister = AcpipGetNamespaceObject(
- Statement->Argument[1]->U.String.String,
- Context->CurrentScope);
- if (DataRegister == NULL) {
- return STATUS_NOT_FOUND;
- }
- }
- //
- // Parse the field list.
- //
- Status = AcpipParseFieldList(Context,
- Statement->Type,
- NULL,
- NULL,
- NULL,
- IndexRegister,
- DataRegister,
- Statement->AdditionalData,
- (UCHAR)Statement->AdditionalData2);
- return Status;
- }
- KSTATUS
- AcpipEvaluateIndexStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates an Index statement, which creates a reference to the
- nth object in a buffer, string, or package.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- ACPI_ALIAS_OBJECT Alias;
- ACPI_OBJECT_TYPE ArgumentType;
- ACPI_BUFFER_FIELD_OBJECT BufferField;
- PACPI_OBJECT NewArgument;
- ULONG PackageIndex;
- KSTATUS Status;
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("Index (");
- } else if ((Statement->ArgumentsAcquired == 0) ||
- (Statement->ArgumentsAcquired == 1)) {
- RtlDebugPrint(", ");
- } else {
- RtlDebugPrint(")");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- if ((Statement->ArgumentsAcquired == 0) ||
- (Statement->ArgumentsAcquired == 1)) {
- //
- // Fail if there is no argument there.
- //
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- //
- // The first argument must be a buffer, package, or string.
- //
- ArgumentType = AcpiObjectUninitialized;
- if (Statement->ArgumentsAcquired == 0) {
- if ((NewArgument->Type != AcpiObjectBuffer) &&
- (NewArgument->Type != AcpiObjectPackage) &&
- (NewArgument->Type != AcpiObjectString)) {
- ArgumentType = AcpiObjectBuffer;
- }
- //
- // The second object must evaluate to an integer.
- //
- } else if (NewArgument->Type != AcpiObjectInteger) {
- ArgumentType = AcpiObjectInteger;
- }
- //
- // Perform an implicit conversion if needed.
- //
- if (ArgumentType != AcpiObjectUninitialized) {
- NewArgument = AcpipConvertObjectType(Context,
- NewArgument,
- ArgumentType);
- if (NewArgument == NULL) {
- return STATUS_CONVERSION_FAILED;
- }
- //
- // The object is fine, take ownership of it.
- //
- } else {
- AcpipObjectAddReference(NewArgument);
- }
- } else {
- ASSERT(Statement->ArgumentsAcquired == 2);
- if (NewArgument != NULL) {
- AcpipObjectAddReference(NewArgument);
- }
- }
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- }
- }
- //
- // Evaluate the result.
- //
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements == FALSE) {
- return STATUS_SUCCESS;
- }
- ASSERT((Statement->Argument[0]->Type == AcpiObjectBuffer) ||
- (Statement->Argument[0]->Type == AcpiObjectString) ||
- (Statement->Argument[0]->Type == AcpiObjectPackage));
- ASSERT(Statement->Argument[1]->Type == AcpiObjectInteger);
- //
- // For strings, create a field that points into the string.
- //
- if ((Statement->Argument[0]->Type == AcpiObjectString) ||
- (Statement->Argument[0]->Type == AcpiObjectBuffer)) {
- RtlZeroMemory(&BufferField, sizeof(ACPI_BUFFER_FIELD_OBJECT));
- BufferField.DestinationObject = Statement->Argument[0];
- BufferField.BitOffset =
- Statement->Argument[1]->U.Integer.Value * BITS_PER_BYTE;
- BufferField.BitLength = BITS_PER_BYTE;
- Statement->Reduction = AcpipCreateNamespaceObject(
- Context,
- AcpiObjectBufferField,
- NULL,
- &BufferField,
- sizeof(ACPI_BUFFER_FIELD_OBJECT));
- } else if (Statement->Argument[0]->Type == AcpiObjectPackage) {
- RtlZeroMemory(&Alias, sizeof(ACPI_ALIAS_OBJECT));
- PackageIndex = (ULONG)Statement->Argument[1]->U.Integer.Value;
- Alias.DestinationObject = AcpipGetPackageObject(Statement->Argument[0],
- PackageIndex,
- TRUE);
- if (Alias.DestinationObject == NULL) {
- return STATUS_NOT_FOUND;
- }
- Statement->Reduction = AcpipCreateNamespaceObject(
- Context,
- AcpiObjectAlias,
- NULL,
- &Alias,
- sizeof(ACPI_ALIAS_OBJECT));
- }
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- //
- // Store the result in the target if supplied.
- //
- if (Statement->Argument[2] != NULL) {
- Status = AcpipPerformStoreOperation(Context,
- Statement->Reduction,
- Statement->Argument[2]);
- return Status;
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateIntegerArithmeticStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates simple arithmetic operations that take two operands
- and a target.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT NewArgument;
- ULONGLONG Operand1;
- ULONGLONG Operand2;
- ULONGLONG ResultValue;
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- switch (Statement->Type) {
- case AmlStatementAdd:
- RtlDebugPrint("Add (");
- break;
- case AmlStatementAnd:
- RtlDebugPrint("And (");
- break;
- case AmlStatementMod:
- RtlDebugPrint("Mod (");
- break;
- case AmlStatementMultiply:
- RtlDebugPrint("Multiply (");
- break;
- case AmlStatementNand:
- RtlDebugPrint("Nand (");
- break;
- case AmlStatementNor:
- RtlDebugPrint("Nor (");
- break;
- case AmlStatementOr:
- RtlDebugPrint("Or (");
- break;
- case AmlStatementSubtract:
- RtlDebugPrint("Subtract (");
- break;
- case AmlStatementShiftLeft:
- RtlDebugPrint("ShiftLeft (");
- break;
- case AmlStatementShiftRight:
- RtlDebugPrint("ShiftRight (");
- break;
- case AmlStatementXor:
- RtlDebugPrint("XOr (");
- break;
- default:
- ASSERT(FALSE);
- return STATUS_NOT_SUPPORTED;
- }
- } else if ((Statement->ArgumentsAcquired == 0) ||
- (Statement->ArgumentsAcquired == 1)) {
- RtlDebugPrint(", ");
- } else {
- RtlDebugPrint(")");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- if ((Statement->ArgumentsAcquired == 0) ||
- (Statement->ArgumentsAcquired == 1)) {
- //
- // Fail if there is no argument there.
- //
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- //
- // Perform an implicit conversion if needed.
- //
- if (NewArgument->Type != AcpiObjectInteger) {
- NewArgument = AcpipConvertObjectType(Context,
- NewArgument,
- AcpiObjectInteger);
- if (NewArgument == NULL) {
- return STATUS_CONVERSION_FAILED;
- }
- //
- // The object is fine, take ownership of it.
- //
- } else {
- AcpipObjectAddReference(NewArgument);
- }
- } else {
- ASSERT(Statement->ArgumentsAcquired == 2);
- if (NewArgument != NULL) {
- AcpipObjectAddReference(NewArgument);
- }
- }
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- }
- }
- //
- // Evaluate the result.
- //
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements == FALSE) {
- return STATUS_SUCCESS;
- }
- ASSERT(Statement->Argument[0]->Type == AcpiObjectInteger);
- ASSERT(Statement->Argument[1]->Type == AcpiObjectInteger);
- Operand1 = Statement->Argument[0]->U.Integer.Value;
- Operand2 = Statement->Argument[1]->U.Integer.Value;
- if (Context->CurrentMethod->IntegerWidthIs32 != FALSE) {
- Operand1 &= 0xFFFFFFFF;
- Operand2 &= 0xFFFFFFFF;
- }
- switch (Statement->Type) {
- case AmlStatementAdd:
- ResultValue = Operand1 + Operand2;
- break;
- case AmlStatementAnd:
- ResultValue = Operand1 & Operand2;
- break;
- case AmlStatementMod:
- if (Operand2 == 0) {
- return STATUS_DIVIDE_BY_ZERO;
- }
- ResultValue = Operand1 % Operand2;
- break;
- case AmlStatementMultiply:
- ResultValue = Operand1 * Operand2;
- break;
- case AmlStatementNand:
- ResultValue = ~(Operand1 & Operand2);
- break;
- case AmlStatementNor:
- ResultValue = ~(Operand1 | Operand2);
- break;
- case AmlStatementOr:
- ResultValue = Operand1 | Operand2;
- break;
- case AmlStatementSubtract:
- ResultValue = Operand1 - Operand2;
- break;
- case AmlStatementShiftLeft:
- ResultValue = Operand1 << Operand2;
- break;
- case AmlStatementShiftRight:
- ResultValue = Operand1 >> Operand2;
- break;
- case AmlStatementXor:
- ResultValue = Operand1 ^ Operand2;
- break;
- default:
- ASSERT(FALSE);
- return STATUS_NOT_SUPPORTED;
- }
- if (Context->CurrentMethod->IntegerWidthIs32 != FALSE) {
- ResultValue &= 0xFFFFFFFF;
- }
- Statement->Reduction = AcpipCreateNamespaceObject(Context,
- AcpiObjectInteger,
- NULL,
- &ResultValue,
- sizeof(ULONGLONG));
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- //
- // Store the result in the target if supplied.
- //
- if (Statement->Argument[2] != NULL) {
- return AcpipPerformStoreOperation(Context,
- Statement->Reduction,
- Statement->Argument[2]);
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateIntegerStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates statements that take no arguments and generate an
- integer output. This includes the constant statements Zero, One, and Ones,
- as well as the AML Revision and Timer statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- STATUS_SUCCESS almost always.
- STATUS_INVALID_PARAMETER if this routine was called for the wrong statement
- type (an internal error for sure).
- STATUS_UNSUCCESSFUL if a namespace object could not be created.
- --*/
- {
- BOOL PrintParentheses;
- PACPI_OBJECT Reduction;
- ULONGLONG Value;
- Reduction = NULL;
- PrintParentheses = FALSE;
- switch (Statement->Type) {
- case AmlStatementZero:
- Value = 0;
- Reduction = &AcpiZero;
- break;
- case AmlStatementOne:
- Value = 1;
- Reduction = &AcpiOne;
- break;
- case AmlStatementOnes:
- if (Context->CurrentMethod->IntegerWidthIs32 != FALSE) {
- Value = 0xFFFFFFFF;
- Reduction = &AcpiOnes32;
- } else {
- Value = 0xFFFFFFFFFFFFFFFFULL;
- Reduction = &AcpiOnes64;
- }
- break;
- case AmlStatementRevision:
- if (Context->PrintStatements != FALSE) {
- PrintParentheses = TRUE;
- RtlDebugPrint("Revision (");
- }
- Value = AML_REVISION;
- break;
- case AmlStatementTimer:
- if (Context->PrintStatements != FALSE) {
- PrintParentheses = TRUE;
- RtlDebugPrint("Timer (");
- }
- Value = AcpipGetTimerValue();
- break;
- default:
- ASSERT(FALSE);
- return STATUS_INVALID_PARAMETER;
- }
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("0x%I64x", Value);
- if (PrintParentheses != FALSE) {
- RtlDebugPrint(")");
- }
- }
- if (Context->ExecuteStatements != FALSE) {
- if (Reduction != NULL) {
- AcpipObjectAddReference(Reduction);
- } else {
- Reduction = AcpipCreateNamespaceObject(Context,
- AcpiObjectInteger,
- NULL,
- &Value,
- sizeof(ULONGLONG));
- if (Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- }
- Statement->Reduction = Reduction;
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateLoadStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a Load statement, which adds the contents of a
- memory op-region as an SSDT to the namespace.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PDESCRIPTION_HEADER Buffer;
- PACPI_OBJECT BufferObject;
- UINTN BufferSize;
- PACPI_OBJECT DdbHandle;
- PACPI_OBJECT NewArgument;
- PACPI_OPERATION_REGION_OBJECT OperationRegion;
- PACPI_OBJECT Source;
- KSTATUS Status;
- Buffer = NULL;
- BufferObject = NULL;
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("Load (");
- } else if (Statement->ArgumentsAcquired == 0) {
- RtlDebugPrint(", ");
- } else {
- RtlDebugPrint(")");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- ASSERT(Statement->ArgumentsAcquired <= 1);
- if (Context->PreviousStatement->Reduction != NULL) {
- AcpipObjectAddReference(NewArgument);
- }
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- }
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- }
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements == FALSE) {
- return STATUS_SUCCESS;
- }
- //
- // The source can either be an operation region itself or a field unit.
- //
- Source = Statement->Argument[0];
- DdbHandle = Statement->Argument[1];
- //
- // If it's an operation region, read it directly. It had better be a memory
- // region.
- //
- if (Source->Type == AcpiObjectOperationRegion) {
- OperationRegion = &(Source->U.OperationRegion);
- if ((OperationRegion->Space != OperationRegionSystemMemory) ||
- (OperationRegion->Length < sizeof(DESCRIPTION_HEADER))) {
- ASSERT(FALSE);
- return STATUS_INVALID_PARAMETER;
- }
- BufferSize = OperationRegion->Length;
- ASSERT(BufferSize == OperationRegion->Length);
- Buffer = AcpipAllocateMemory(BufferSize);
- if (Buffer == NULL) {
- return STATUS_INSUFFICIENT_RESOURCES;
- }
- Status = OperationRegion->FunctionTable->Read(
- OperationRegion->OsContext,
- 0,
- OperationRegion->Length * BITS_PER_BYTE,
- Buffer);
- ASSERT(KSUCCESS(Status));
- //
- // Convert the field unit into a buffer, which performs a read of the
- // op-region.
- //
- } else if (Source->Type == AcpiObjectFieldUnit) {
- BufferObject = AcpipConvertObjectType(Context,
- Source,
- AcpiObjectBuffer);
- if (BufferObject == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- //
- // Steal the buffer from the buffer object.
- //
- Buffer = BufferObject->U.Buffer.Buffer;
- BufferSize = BufferObject->U.Buffer.Length;
- BufferObject->U.Buffer.Buffer = NULL;
- BufferObject->U.Buffer.Length = 0;
- } else {
- RtlDebugPrint("ACPI: Load source should be an op-region or field.\n");
- ASSERT(FALSE);
- return STATUS_UNEXPECTED_TYPE;
- }
- //
- // Validate the buffer a bit.
- //
- if ((BufferSize < sizeof(DESCRIPTION_HEADER)) ||
- (BufferSize < Buffer->Length)) {
- Status = STATUS_DATA_LENGTH_MISMATCH;
- goto EvaluateLoadStatementEnd;
- }
- if (AcpipChecksumData(Buffer, Buffer->Length) != 0) {
- Status = STATUS_CHECKSUM_MISMATCH;
- goto EvaluateLoadStatementEnd;
- }
- //
- // Load the definition block synchronously.
- //
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("\nLoading Definition Block...\n");
- }
- Status = AcpiLoadDefinitionBlock(Buffer, DdbHandle);
- if (!KSUCCESS(Status)) {
- RtlDebugPrint("ACPI: Failed to execute Load: %d\n", Status);
- goto EvaluateLoadStatementEnd;
- }
- //
- // Run any _INI methods.
- //
- Status = AcpipRunInitializationMethods(NULL);
- if (!KSUCCESS(Status)) {
- goto EvaluateLoadStatementEnd;
- }
- //
- // The definition block owns the buffer now.
- //
- Buffer = NULL;
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("\nDone Loading Definition Block\n");
- }
- EvaluateLoadStatementEnd:
- if (BufferObject != NULL) {
- AcpipObjectReleaseReference(BufferObject);
- }
- if (Buffer != NULL) {
- AcpipFreeMemory(Buffer);
- }
- return Status;
- }
- KSTATUS
- AcpipEvaluateLocalStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates the LocalX opcodes.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- ULONG LocalNumber;
- PACPI_OBJECT LocalObject;
- LocalNumber = (ULONG)Statement->AdditionalData;
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("Local%d", LocalNumber);
- }
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements != FALSE) {
- LocalObject = Context->CurrentMethod->LocalVariable[LocalNumber];
- //
- // Create an uninitialized object if none exists yet.
- //
- if (LocalObject == NULL) {
- LocalObject = AcpipCreateNamespaceObject(Context,
- AcpiObjectUninitialized,
- NULL,
- NULL,
- 0);
- if (LocalObject == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- Context->CurrentMethod->LocalVariable[LocalNumber] = LocalObject;
- }
- Statement->Reduction = LocalObject;
- AcpipObjectAddReference(LocalObject);
- Context->CurrentMethod->LastLocalIndex = LocalNumber;
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateLogicalExpressionStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates logical binary operators (Logical and, equal,
- greater, less, and or).
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT NewArgument;
- ULONGLONG Operand1;
- ULONGLONG Operand2;
- ULONGLONG ResultValue;
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- switch (Statement->Type) {
- case AmlStatementLogicalAnd:
- RtlDebugPrint("LAnd (");
- break;
- case AmlStatementLogicalEqual:
- RtlDebugPrint("LEqual (");
- break;
- case AmlStatementLogicalGreater:
- RtlDebugPrint("LGreater (");
- break;
- case AmlStatementLogicalLess:
- RtlDebugPrint("LLess (");
- break;
- case AmlStatementLogicalOr:
- RtlDebugPrint("LOr (");
- break;
- default:
- ASSERT(FALSE);
- return STATUS_NOT_SUPPORTED;
- }
- } else if (Statement->ArgumentsAcquired == 0) {
- RtlDebugPrint(", ");
- } else {
- RtlDebugPrint(")");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- //
- // Fail if there is no argument there.
- //
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- //
- // Perform an implicit conversion if needed.
- //
- if (NewArgument->Type != AcpiObjectInteger) {
- NewArgument = AcpipConvertObjectType(Context,
- NewArgument,
- AcpiObjectInteger);
- if (NewArgument == NULL) {
- return STATUS_CONVERSION_FAILED;
- }
- //
- // The object is fine, take ownership of it.
- //
- } else {
- AcpipObjectAddReference(NewArgument);
- }
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- }
- }
- //
- // Evaluate the result.
- //
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements == FALSE) {
- return STATUS_SUCCESS;
- }
- ASSERT(Statement->Argument[0]->Type == AcpiObjectInteger);
- ASSERT(Statement->Argument[1]->Type == AcpiObjectInteger);
- Operand1 = Statement->Argument[0]->U.Integer.Value;
- Operand2 = Statement->Argument[1]->U.Integer.Value;
- ResultValue = FALSE;
- switch (Statement->Type) {
- case AmlStatementLogicalAnd:
- if ((Operand1 != 0) && (Operand2 != 0)) {
- ResultValue = TRUE;
- }
- break;
- case AmlStatementLogicalEqual:
- if (Operand1 == Operand2) {
- ResultValue = TRUE;
- }
- break;
- case AmlStatementLogicalGreater:
- if (Operand1 > Operand2) {
- ResultValue = TRUE;
- }
- break;
- case AmlStatementLogicalLess:
- if (Operand1 < Operand2) {
- ResultValue = TRUE;
- }
- break;
- case AmlStatementLogicalOr:
- if ((Operand1 != 0) || (Operand2 != 0)) {
- ResultValue = TRUE;
- }
- break;
- default:
- ASSERT(FALSE);
- return STATUS_NOT_SUPPORTED;
- }
- Statement->Reduction = AcpipCreateNamespaceObject(Context,
- AcpiObjectInteger,
- NULL,
- &ResultValue,
- sizeof(ULONGLONG));
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateLogicalNotStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates logical NOT operator.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT NewArgument;
- ULONGLONG ResultValue;
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("LNot (");
- } else {
- RtlDebugPrint(")");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- //
- // Fail if there is no argument there.
- //
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- //
- // Perform an implicit conversion if needed.
- //
- if (NewArgument->Type != AcpiObjectInteger) {
- NewArgument = AcpipConvertObjectType(Context,
- NewArgument,
- AcpiObjectInteger);
- if (NewArgument == NULL) {
- return STATUS_CONVERSION_FAILED;
- }
- //
- // The object is fine, take ownership of it.
- //
- } else {
- AcpipObjectAddReference(NewArgument);
- }
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- }
- }
- //
- // Evaluate the result.
- //
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements == FALSE) {
- return STATUS_SUCCESS;
- }
- ASSERT(Statement->Argument[0]->Type == AcpiObjectInteger);
- ResultValue = FALSE;
- if (Statement->Argument[0]->U.Integer.Value == 0) {
- ResultValue = TRUE;
- }
- Statement->Reduction = AcpipCreateNamespaceObject(Context,
- AcpiObjectInteger,
- NULL,
- &ResultValue,
- sizeof(ULONGLONG));
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateMatchStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a Match statement, which iterates over a package
- doing some simple comparisons.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PUCHAR InstructionPointer;
- ULONGLONG ItemCount;
- ULONGLONG ItemIndex;
- BOOL Match;
- PACPI_OBJECT NewArgument;
- PACPI_OBJECT Operand1;
- PACPI_OBJECT Operand2;
- ACPI_MATCH_OPERATOR Operator1;
- ACPI_MATCH_OPERATOR Operator2;
- PACPI_OBJECT Package;
- PACPI_OBJECT PackageElement;
- PACPI_OBJECT StartIndex;
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("Match (");
- } else if (Statement->ArgumentsAcquired < 3) {
- RtlDebugPrint(", ");
- } else {
- RtlDebugPrint(")");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- //
- // An argument is required.
- //
- if (NewArgument == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- //
- // The first object must be a package.
- //
- if (Statement->ArgumentsAcquired == 0) {
- if (NewArgument->Type != AcpiObjectPackage) {
- return STATUS_CONVERSION_FAILED;
- }
- AcpipObjectAddReference(NewArgument);
- //
- // The second and third arguments must be an integer, buffer, or
- // string. Convert that to an integer. The fourth argument
- // (StartIndex) is also an integer.
- //
- } else if ((Statement->ArgumentsAcquired == 1) ||
- (Statement->ArgumentsAcquired == 2) ||
- (Statement->ArgumentsAcquired == 3)) {
- if (NewArgument->Type != AcpiObjectInteger) {
- NewArgument = AcpipConvertObjectType(Context,
- NewArgument,
- AcpiObjectInteger);
- if (NewArgument == NULL) {
- return STATUS_CONVERSION_FAILED;
- }
- } else {
- AcpipObjectAddReference(NewArgument);
- }
- }
- //
- // Save the argument and return if not all arguments have been
- // collected yet.
- //
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- }
- //
- // After the first and second arguments come match opcodes. Pull
- // those out and store them in additional data.
- //
- if ((Statement->ArgumentsAcquired == 1) ||
- (Statement->ArgumentsAcquired == 2)) {
- InstructionPointer =
- (PUCHAR)(Context->AmlCode) + Context->CurrentOffset;
- if (Context->CurrentOffset >= Context->AmlCodeSize) {
- return STATUS_MALFORMED_DATA_STREAM;
- }
- Statement->AdditionalData = (Statement->AdditionalData << 8) |
- *InstructionPointer;
- if (*InstructionPointer >= MatchOperatorCount) {
- return STATUS_MALFORMED_DATA_STREAM;
- }
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("%s, ",
- AcpiMatchOpcodeString[*InstructionPointer]);
- }
- Context->CurrentOffset += 1;
- }
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- }
- //
- // Evaluate the result.
- //
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements == FALSE) {
- return STATUS_SUCCESS;
- }
- //
- // Assert that all the arguments are the expected types.
- //
- Package = Statement->Argument[0];
- Operand1 = Statement->Argument[1];
- Operand2 = Statement->Argument[2];
- StartIndex = Statement->Argument[3];
- Operator1 = (Statement->AdditionalData >> 8) & 0xFF;
- Operator2 = Statement->AdditionalData & 0xFF;
- ASSERT((Package->Type == AcpiObjectPackage) &&
- (StartIndex->Type == AcpiObjectInteger));
- ASSERT((Operand1->Type == AcpiObjectInteger) &&
- (Operand2->Type == AcpiObjectInteger));
- ASSERT((Operator1 < MatchOperatorCount) &&
- (Operator2 < MatchOperatorCount));
- //
- // Perform the match operation.
- //
- ItemIndex = StartIndex->U.Integer.Value;
- ItemCount = Package->U.Package.ElementCount;
- while (ItemIndex < ItemCount) {
- PackageElement = AcpipGetPackageObject(Package, ItemIndex, FALSE);
- Match = AcpipEvaluateMatchComparison(Context,
- PackageElement,
- Operand1,
- Operator1,
- Operand2,
- Operator2);
- if (Match != FALSE) {
- break;
- }
- ItemIndex += 1;
- }
- //
- // If a match was never found (as evidenced by the index being all the way
- // at the end), the ACPI says to return the constant "Ones".
- //
- if (ItemIndex == ItemCount) {
- Statement->Reduction = &AcpiOnes64;
- if (Context->CurrentMethod->IntegerWidthIs32 != FALSE) {
- Statement->Reduction = &AcpiOnes32;
- }
- AcpipObjectAddReference(Statement->Reduction);
- //
- // Otherwise, return the result value.
- //
- } else {
- Statement->Reduction = AcpipCreateNamespaceObject(Context,
- AcpiObjectInteger,
- NULL,
- &ItemIndex,
- sizeof(ULONGLONG));
- if (Statement->Reduction == NULL) {
- return STATUS_INSUFFICIENT_RESOURCES;
- }
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateMethodStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a Method declaration statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- ACPI_METHOD_OBJECT Method;
- UCHAR MethodFlags;
- PACPI_OBJECT MethodObject;
- PSTR Name;
- //
- // If the previous statement is NULL, this is the first time through.
- //
- if (Context->PreviousStatement == NULL) {
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- ASSERT(Statement->Argument[0]->Type == AcpiObjectString);
- //
- // Get the method flags out of additional data 2.
- //
- MethodFlags = (UCHAR)Statement->AdditionalData2;
- //
- // Initialize the method structure.
- //
- RtlZeroMemory(&Method, sizeof(ACPI_METHOD_OBJECT));
- Method.IntegerWidthIs32 = Context->CurrentMethod->IntegerWidthIs32;
- Method.ArgumentCount = MethodFlags & METHOD_ARGUMENT_COUNT_MASK;
- Method.Serialized = FALSE;
- if ((MethodFlags & METHOD_SERIALIZED_FLAG) != 0) {
- Method.Serialized = TRUE;
- }
- Method.SyncLevel = (MethodFlags & METHOD_SYNC_LEVEL_MASK) >>
- METHOD_SYNC_LEVEL_SHIFT;
- Method.AmlCode = (PUCHAR)Context->AmlCode + Context->CurrentOffset;
- //
- // Additional Data stored the end offset, so the size is the end offset
- // minus the current offset.
- //
- Method.AmlCodeSize = Statement->AdditionalData - Context->CurrentOffset;
- Name = Statement->Argument[0]->U.String.String;
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("Method (%s, 0x%02x)", Name, MethodFlags);
- }
- //
- // Create the object if execution is enabled.
- //
- if (Context->ExecuteStatements != FALSE) {
- MethodObject = AcpipCreateNamespaceObject(
- Context,
- AcpiObjectMethod,
- Name,
- &Method,
- sizeof(ACPI_METHOD_OBJECT));
- if (MethodObject == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- //
- // Advance the current offset to the end of the method, as it's not
- // being executed now, and complete the statement.
- //
- Context->CurrentOffset = Statement->AdditionalData;
- return STATUS_SUCCESS;
- //
- // If the context is printing but not executing, add to the indentation
- // level and delve into the function for execution.
- //
- } else {
- Context->IndentationLevel += 1;
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint(" {");
- }
- }
- }
- //
- // Wait for the end of the routine.
- //
- if (Context->CurrentOffset < Statement->AdditionalData) {
- AcpipPrintIndentedNewLine(Context);
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // Decrease the indentation level and print the closing curly brace if
- // needed.
- //
- Context->IndentationLevel -= 1;
- if (Context->PrintStatements != FALSE) {
- AcpipPrintIndentedNewLine(Context);
- RtlDebugPrint("}");
- }
- AcpipPrintIndentedNewLine(Context);
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateMidStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a mid statement, which splits a string up.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PVOID Buffer;
- ULONGLONG BufferLength;
- ULONGLONG MidIndex;
- ULONGLONG MidLength;
- PACPI_OBJECT NewArgument;
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("Mid (");
- } else if ((Statement->ArgumentsAcquired == 0) ||
- (Statement->ArgumentsAcquired == 1) ||
- (Statement->ArgumentsAcquired == 2)) {
- RtlDebugPrint(", ");
- } else {
- RtlDebugPrint(")");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- if (Statement->ArgumentsAcquired == 0) {
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- if ((NewArgument->Type != AcpiObjectString) &&
- (NewArgument->Type != AcpiObjectBuffer)) {
- //
- // Perform an implicit conversion if needed.
- //
- NewArgument = AcpipConvertObjectType(Context,
- NewArgument,
- AcpiObjectBuffer);
- if (NewArgument == NULL) {
- return STATUS_CONVERSION_FAILED;
- }
- //
- // The object is fine, take ownership of it.
- //
- } else {
- AcpipObjectAddReference(NewArgument);
- }
- } else if ((Statement->ArgumentsAcquired == 1) ||
- (Statement->ArgumentsAcquired == 2)) {
- //
- // Fail if there is no argument there.
- //
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- //
- // Perform an implicit conversion if needed.
- //
- if (NewArgument->Type != AcpiObjectInteger) {
- NewArgument = AcpipConvertObjectType(Context,
- NewArgument,
- AcpiObjectInteger);
- if (NewArgument == NULL) {
- return STATUS_CONVERSION_FAILED;
- }
- //
- // The object is fine, take ownership of it.
- //
- } else {
- AcpipObjectAddReference(NewArgument);
- }
- //
- // Parse the target argument.
- //
- } else {
- ASSERT(Statement->ArgumentsAcquired == 3);
- if (NewArgument != NULL) {
- AcpipObjectAddReference(NewArgument);
- }
- }
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- }
- }
- //
- // Evaluate the result.
- //
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements == FALSE) {
- return STATUS_SUCCESS;
- }
- ASSERT((Statement->Argument[0]->Type == AcpiObjectString) ||
- (Statement->Argument[0]->Type == AcpiObjectBuffer));
- ASSERT(Statement->Argument[1]->Type == AcpiObjectInteger);
- ASSERT(Statement->Argument[2]->Type == AcpiObjectInteger);
- //
- // The Mid statement gets a portion of a string or buffer at an offset
- // (Index) with a length. If the mid statement tries to go over, the
- // resulting buffer is clipped with the original. If the offset is beyond
- // the end of the buffer, an empty buffer is created.
- //
- MidIndex = Statement->Argument[1]->U.Integer.Value;
- MidLength = Statement->Argument[2]->U.Integer.Value;
- BufferLength = 0;
- if (Statement->Argument[0]->Type == AcpiObjectString) {
- Buffer = Statement->Argument[0]->U.String.String;
- if (Buffer != NULL) {
- BufferLength = RtlStringLength(Buffer) + 1;
- }
- } else {
- Buffer = Statement->Argument[0]->U.Buffer.Buffer;
- BufferLength = Statement->Argument[0]->U.Buffer.Length;
- }
- //
- // Cap the mid statement from going over the buffer.
- //
- if (MidIndex >= BufferLength) {
- MidIndex = 0;
- MidLength = 0;
- }
- if (MidIndex + MidLength > BufferLength) {
- MidLength = BufferLength - MidIndex;
- }
- //
- // Create the mid buffer.
- //
- Statement->Reduction = AcpipCreateNamespaceObject(Context,
- AcpiObjectInteger,
- NULL,
- Buffer + MidIndex,
- MidLength);
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- //
- // Store the result in the target if supplied.
- //
- if (Statement->Argument[3] != NULL) {
- return AcpipPerformStoreOperation(Context,
- Statement->Reduction,
- Statement->Argument[2]);
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateMutexStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a Mutex (creation) statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PSTR NameString;
- UCHAR SyncLevel;
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- ASSERT(Statement->Argument[0]->Type == AcpiObjectString);
- NameString = Statement->Argument[0]->U.String.String;
- SyncLevel = Statement->AdditionalData & MUTEX_FLAG_SYNC_LEVEL_MASK;
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("Mutex (%s, %d)", NameString, SyncLevel);
- }
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements == FALSE) {
- return STATUS_SUCCESS;
- }
- //
- // Create the mutex object.
- //
- Statement->Reduction = AcpipCreateNamespaceObject(Context,
- AcpiObjectMutex,
- NameString,
- NULL,
- 0);
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateNameStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a Name statement, which creates a new named object
- in the namespace given an existing one.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT NamedObject;
- PACPI_OBJECT NewArgument;
- PACPI_OBJECT ObjectWithContents;
- KSTATUS Status;
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- ASSERT(Statement->Argument[0]->Type == AcpiObjectString);
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("Name (%s,",
- Statement->Argument[0]->U.String.String);
- } else {
- RtlDebugPrint(")");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- //
- // Fail if there is no argument there.
- //
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- AcpipObjectAddReference(NewArgument);
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- }
- }
- //
- // The arguments should be all gathered.
- //
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements == FALSE) {
- return STATUS_SUCCESS;
- }
- ASSERT(Statement->Argument[0]->Type == AcpiObjectString);
- //
- // Create the new object without the stuff in it.
- //
- ObjectWithContents = Statement->Argument[1];
- NamedObject = AcpipCreateNamespaceObject(
- Context,
- AcpiObjectUninitialized,
- Statement->Argument[0]->U.String.String,
- NULL,
- 0);
- if (NamedObject == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- //
- // Put the stuff from the other object in this new named object.
- //
- Status = AcpipPerformStoreOperation(Context,
- ObjectWithContents,
- NamedObject);
- if (!KSUCCESS(Status)) {
- return Status;
- }
- Statement->Reduction = NamedObject;
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateNameStringStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a NameString statement, which evaluates to an
- object that is expected to exist in the namespace.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PSTR Name;
- ACPI_UNRESOLVED_NAME_OBJECT UnresolvedName;
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- ASSERT(Statement->Argument[0]->Type == AcpiObjectString);
- Name = Statement->Argument[0]->U.String.String;
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint(Name);
- }
- //
- // Get the object if the interpreter is executing statements.
- //
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements != FALSE) {
- Statement->Reduction = AcpipGetNamespaceObject(Name,
- Context->CurrentScope);
- //
- // If the object could not be found, then a name is being referenced
- // before it is defined. In certain situations this is alright, such as
- // the definition of a package object during a load operation. Create
- // an unresolved name object to remember to re-evaluate this name
- // when the object is referenced.
- //
- if (Statement->Reduction == NULL) {
- RtlZeroMemory(&UnresolvedName, sizeof(ACPI_UNRESOLVED_NAME_OBJECT));
- UnresolvedName.Name = Name;
- UnresolvedName.Scope = Context->CurrentScope;
- Statement->Reduction = AcpipCreateNamespaceObject(
- Context,
- AcpiObjectUnresolvedName,
- NULL,
- &UnresolvedName,
- sizeof(ACPI_UNRESOLVED_NAME_OBJECT));
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint(" ?");
- }
- } else {
- AcpipObjectAddReference(Statement->Reduction);
- }
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateNoOpStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a No-Op statement, which is really quite easy since
- it doesn't do anything.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- STATUS_SUCCESS always.
- --*/
- {
- //
- // Finally, an easy one!
- //
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("NoOp");
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateNotifyStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a Notify (the operating system) statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT NewArgument;
- KSTATUS Status;
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("Notify (");
- } else if (Statement->ArgumentsAcquired == 0) {
- RtlDebugPrint(", ");
- } else {
- RtlDebugPrint(")");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- //
- // Fail if there is no argument there.
- //
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- //
- // The first argument needs to be a Thermal Zone, Processor, or
- // Device.
- //
- if (Statement->ArgumentsAcquired == 0) {
- if ((NewArgument->Type != AcpiObjectProcessor) &&
- (NewArgument->Type != AcpiObjectThermalZone) &&
- (NewArgument->Type != AcpiObjectDevice)) {
- return STATUS_INVALID_PARAMETER;
- } else {
- AcpipObjectAddReference(NewArgument);
- }
- //
- // The second argument needs to come out as an integer.
- //
- } else {
- ASSERT(Statement->ArgumentsAcquired == 1);
- if (NewArgument->Type != AcpiObjectInteger) {
- NewArgument = AcpipConvertObjectType(Context,
- NewArgument,
- AcpiObjectInteger);
- if (NewArgument == NULL) {
- return STATUS_CONVERSION_FAILED;
- }
- //
- // The object is fine, take ownership of it.
- //
- } else {
- AcpipObjectAddReference(NewArgument);
- }
- }
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- }
- }
- //
- // Evaluate the result.
- //
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements == FALSE) {
- return STATUS_SUCCESS;
- }
- ASSERT((Statement->Argument[0]->Type == AcpiObjectProcessor) ||
- (Statement->Argument[0]->Type == AcpiObjectThermalZone) ||
- (Statement->Argument[0]->Type == AcpiObjectDevice));
- ASSERT(Statement->Argument[1]->Type == AcpiObjectInteger);
- //
- // Pass the notification on to the rest of the system.
- //
- Status = AcpipNotifyOperatingSystem(
- Statement->Argument[0],
- Statement->Argument[1]->U.Integer.Value);
- return Status;
- }
- KSTATUS
- AcpipEvaluateNotStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates bitwise NOT operator.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT NewArgument;
- ULONGLONG ResultValue;
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("Not (");
- } else if (Statement->ArgumentsAcquired == 0) {
- RtlDebugPrint(", ");
- } else {
- RtlDebugPrint(")");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- if (Statement->ArgumentsAcquired == 0) {
- //
- // Fail if there is no argument there.
- //
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- //
- // Perform an implicit conversion if needed.
- //
- if (NewArgument->Type != AcpiObjectInteger) {
- NewArgument = AcpipConvertObjectType(Context,
- NewArgument,
- AcpiObjectInteger);
- if (NewArgument == NULL) {
- return STATUS_CONVERSION_FAILED;
- }
- //
- // The object is fine, take ownership of it.
- //
- } else {
- AcpipObjectAddReference(NewArgument);
- }
- } else {
- if (NewArgument != NULL) {
- AcpipObjectAddReference(NewArgument);
- }
- }
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- }
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- }
- //
- // Evaluate the result.
- //
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements == FALSE) {
- return STATUS_SUCCESS;
- }
- ASSERT(Statement->Argument[0]->Type == AcpiObjectInteger);
- //
- // Seems like so much build up just for this...
- //
- ResultValue = ~(Statement->Argument[0]->U.Integer.Value);
- Statement->Reduction = AcpipCreateNamespaceObject(Context,
- AcpiObjectInteger,
- NULL,
- &ResultValue,
- sizeof(ULONGLONG));
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- if (Statement->Argument[1] != NULL) {
- return AcpipPerformStoreOperation(Context,
- Statement->Reduction,
- Statement->Argument[0]);
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateObjectTypeStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates an Object Type statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT NewArgument;
- PACPI_OBJECT Object;
- ULONGLONG ObjectType;
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("ObjectType (");
- } else {
- RtlDebugPrint(")");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- if (Statement->ArgumentsAcquired == 0) {
- //
- // Fail if there is no argument there.
- //
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- AcpipObjectAddReference(NewArgument);
- } else {
- if (NewArgument != NULL) {
- AcpipObjectAddReference(NewArgument);
- }
- }
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- }
- }
- //
- // Evaluate the result.
- //
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements == FALSE) {
- return STATUS_SUCCESS;
- }
- Object = Statement->Argument[0];
- while (Object->Type == AcpiObjectAlias) {
- Object = Object->U.Alias.DestinationObject;
- }
- ObjectType = Object->Type;
- Statement->Reduction = AcpipCreateNamespaceObject(Context,
- AcpiObjectInteger,
- NULL,
- &ObjectType,
- sizeof(ULONGLONG));
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateOperationRegionStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates an Operation Region statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PSTR Name;
- PACPI_OBJECT NewArgument;
- ULONGLONG RegionLength;
- ULONGLONG RegionOffset;
- ACPI_OPERATION_REGION_SPACE RegionSpace;
- KSTATUS Status;
- ASSERT(Statement->Argument[0]->Type == AcpiObjectString);
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("OpRegion (%s, %x, ",
- Statement->Argument[0]->U.String.String,
- Statement->AdditionalData);
- } else if (Statement->ArgumentsAcquired == 1) {
- RtlDebugPrint(", ");
- } else {
- RtlDebugPrint(")");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- //
- // Fail if there is no argument there.
- //
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- //
- // Perform an implicit conversion if needed.
- //
- if (NewArgument->Type != AcpiObjectInteger) {
- NewArgument = AcpipConvertObjectType(Context,
- NewArgument,
- AcpiObjectInteger);
- if (NewArgument == NULL) {
- return STATUS_CONVERSION_FAILED;
- }
- //
- // The object is fine, take ownership of it.
- //
- } else {
- AcpipObjectAddReference(NewArgument);
- }
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- }
- }
- //
- // Evaluate the result.
- //
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements == FALSE) {
- return STATUS_SUCCESS;
- }
- ASSERT(Statement->Argument[0]->Type == AcpiObjectString);
- ASSERT(Statement->Argument[1]->Type == AcpiObjectInteger);
- ASSERT(Statement->Argument[2]->Type == AcpiObjectInteger);
- Name = Statement->Argument[0]->U.String.String;
- RegionOffset = Statement->Argument[1]->U.Integer.Value;
- RegionLength = Statement->Argument[2]->U.Integer.Value;
- RegionSpace = (UCHAR)Statement->AdditionalData;
- Status = AcpipCreateOperationRegion(Context,
- Name,
- RegionSpace,
- RegionOffset,
- RegionLength);
- if (!KSUCCESS(Status)) {
- return Status;
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluatePackageStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a Package or Variable Package statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- ULONG BufferSize;
- PACPI_OBJECT NewArgument;
- if (Context->PreviousStatement == NULL) {
- Context->IndentationLevel += 1;
- if (Context->PrintStatements != FALSE) {
- if (Statement->Type == AmlStatementPackage) {
- RtlDebugPrint("Package (%d) {", Statement->AdditionalData2);
- AcpipPrintIndentedNewLine(Context);
- } else {
- ASSERT(Statement->Type == AmlStatementVariablePackage);
- RtlDebugPrint("VarPackage (");
- }
- }
- }
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument. This only occurs in a variable package when looking
- // for the package size.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- ASSERT(Statement->ArgumentsAcquired == 0);
- //
- // Fail if there is no argument there.
- //
- if (NewArgument == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- //
- // Perform an implicit conversion if needed.
- //
- if (NewArgument->Type != AcpiObjectInteger) {
- NewArgument = AcpipConvertObjectType(Context,
- NewArgument,
- AcpiObjectInteger);
- if (NewArgument == NULL) {
- return STATUS_CONVERSION_FAILED;
- }
- //
- // The object is fine, take ownership of it.
- //
- } else {
- AcpipObjectAddReference(NewArgument);
- }
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- }
- //
- // Finish printing the header for the variable package.
- //
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint(") {");
- AcpipPrintIndentedNewLine(Context);
- }
- //
- // Assuming the length isn't 0, wait for the first package object.
- //
- if (Context->CurrentOffset != Statement->AdditionalData) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- }
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- //
- // If not at the end, add this to the collection.
- //
- if ((Context->PrintStatements != FALSE) &&
- (Context->CurrentOffset != Statement->AdditionalData) &&
- (Context->PreviousStatement != NULL)) {
- RtlDebugPrint(", ");
- AcpipPrintIndentedNewLine(Context);
- }
- if (Context->ExecuteStatements != FALSE) {
- //
- // If the object has never been created before, create it now.
- //
- if (Statement->Reduction == NULL) {
- BufferSize = Statement->AdditionalData2 * sizeof(PACPI_OBJECT);
- Statement->Reduction = AcpipCreateNamespaceObject(Context,
- AcpiObjectPackage,
- NULL,
- NULL,
- BufferSize);
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- //
- // Additional data 2 now becomes a counter of the current
- // element.
- //
- Statement->AdditionalData2 = 0;
- }
- //
- // Add the object to the package/array.
- //
- if (Context->PreviousStatement != NULL) {
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- AcpipSetPackageObject(Statement->Reduction,
- Statement->AdditionalData2,
- Context->PreviousStatement->Reduction);
- Statement->AdditionalData2 += 1;
- }
- }
- if (Context->CurrentOffset == Statement->AdditionalData) {
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("}");
- }
- Context->IndentationLevel -= 1;
- return STATUS_SUCCESS;
- }
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- KSTATUS
- AcpipEvaluatePowerResourceStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a Power Resource declaration statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PUCHAR DataPointer;
- PSTR Name;
- ACPI_POWER_RESOURCE_OBJECT PowerResource;
- PACPI_OBJECT PowerResourceObject;
- if (Context->PreviousStatement == NULL) {
- Statement->SavedScope = NULL;
- ASSERT(Statement->Argument[0]->Type == AcpiObjectString);
- Name = Statement->Argument[0]->U.String.String;
- //
- // Get the system level and resource order.
- //
- DataPointer = Context->AmlCode + Statement->AdditionalData2;
- PowerResource.SystemLevel = *DataPointer;
- PowerResource.ResourceOrder = *(PUSHORT)(DataPointer + 1);
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("PowerResource (%s, %d, %d) {",
- Name,
- PowerResource.SystemLevel,
- PowerResource.ResourceOrder);
- }
- if (Context->ExecuteStatements != FALSE) {
- //
- // Create the power resource object.
- //
- PowerResourceObject = AcpipCreateNamespaceObject(
- Context,
- AcpiObjectPowerResource,
- Name,
- &PowerResource,
- sizeof(ACPI_POWER_RESOURCE_OBJECT));
- if (PowerResourceObject == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- //
- // Make this device the current scope.
- //
- Statement->SavedScope = Context->CurrentScope;
- Context->CurrentScope = PowerResourceObject;
- Statement->Reduction = PowerResourceObject;
- }
- Context->IndentationLevel += 1;
- }
- //
- // If execution is not done with the scope, keep this statement on the
- // stack.
- //
- if (Context->CurrentOffset < Statement->AdditionalData) {
- AcpipPrintIndentedNewLine(Context);
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // Pop this puppy off the stack.
- //
- Context->CurrentScope = Statement->SavedScope;
- Context->IndentationLevel -= 1;
- if (Context->PrintStatements != FALSE) {
- AcpipPrintIndentedNewLine(Context);
- RtlDebugPrint("}");
- }
- AcpipPrintIndentedNewLine(Context);
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateProcessorStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a Power Resource declaration statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PUCHAR DataPointer;
- PSTR DeviceName;
- ACPI_PROCESSOR_OBJECT Processor;
- PACPI_OBJECT ProcessorObject;
- if (Context->PreviousStatement == NULL) {
- Statement->SavedScope = NULL;
- ASSERT(Statement->Argument[0]->Type == AcpiObjectString);
- DeviceName = Statement->Argument[0]->U.String.String;
- //
- // Get the processor ID and processor block register information.
- //
- RtlZeroMemory(&Processor, sizeof(ACPI_PROCESSOR_OBJECT));
- DataPointer = Context->AmlCode + Statement->AdditionalData2;
- Processor.ProcessorId = *DataPointer;
- DataPointer += sizeof(BYTE);
- Processor.ProcessorBlockAddress = *(PULONG)DataPointer;
- DataPointer += sizeof(ULONG);
- Processor.ProcessorBlockLength = *DataPointer;
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("Processor (%s, %d, 0x%x, %d) {",
- DeviceName,
- Processor.ProcessorId,
- Processor.ProcessorBlockAddress,
- Processor.ProcessorBlockLength);
- }
- if (Context->ExecuteStatements != FALSE) {
- //
- // Create the processor object.
- //
- ProcessorObject = AcpipCreateNamespaceObject(
- Context,
- AcpiObjectProcessor,
- DeviceName,
- &Processor,
- sizeof(ACPI_PROCESSOR_OBJECT));
- if (ProcessorObject == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- //
- // Make this device the current scope.
- //
- Statement->SavedScope = Context->CurrentScope;
- Context->CurrentScope = ProcessorObject;
- Statement->Reduction = ProcessorObject;
- }
- Context->IndentationLevel += 1;
- }
- //
- // If execution is not done with the scope, keep this statement on the
- // stack.
- //
- if (Context->CurrentOffset < Statement->AdditionalData) {
- AcpipPrintIndentedNewLine(Context);
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // Pop this puppy off the stack.
- //
- Context->CurrentScope = Statement->SavedScope;
- Context->IndentationLevel -= 1;
- if (Context->PrintStatements != FALSE) {
- AcpipPrintIndentedNewLine(Context);
- RtlDebugPrint("}");
- }
- AcpipPrintIndentedNewLine(Context);
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateReferenceOfStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates an "Reference Of" statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- //
- // Gather arguments if needed.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("RefOf (");
- } else {
- RtlDebugPrint(")");
- }
- }
- //
- // If there is no previous statement, wait for the argument to come in.
- //
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- //
- // Increment the reference count on the object.
- //
- } else {
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[0] = NULL;
- } else {
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- Statement->Argument[0] = Context->PreviousStatement->Reduction;
- AcpipObjectAddReference(Context->PreviousStatement->Reduction);
- }
- Statement->ArgumentsAcquired += 1;
- }
- }
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- if (Context->ExecuteStatements != FALSE) {
- Statement->Reduction = AcpipCreateNamespaceObject(
- Context,
- AcpiObjectAlias,
- NULL,
- &(Statement->Argument[0]),
- sizeof(PACPI_OBJECT));
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateReturnStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates an Return statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- //
- // Gather arguments if needed.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("Return (");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- //
- // Increment the reference count on the object.
- //
- } else {
- if (Context->ExecuteStatements != FALSE) {
- if (Context->PreviousStatement->Reduction != NULL) {
- Statement->Argument[0] =
- Context->PreviousStatement->Reduction;
- AcpipObjectAddReference(Statement->Argument[0]);
- }
- } else {
- Statement->Argument[0] = NULL;
- }
- Statement->ArgumentsAcquired += 1;
- }
- }
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint(")");
- }
- //
- // Return from the function.
- //
- if (Context->ExecuteStatements != FALSE) {
- //
- // If there was an old return value there (possibly from a nested
- // function call), release it.
- //
- if (Context->ReturnValue != NULL) {
- AcpipObjectReleaseReference(Context->ReturnValue);
- }
- Context->ReturnValue = Statement->Argument[0];
- AcpipObjectAddReference(Context->ReturnValue);
- AcpipPopExecutingStatements(Context, FALSE, FALSE);
- AcpipPopCurrentMethodContext(Context);
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateScopeStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a Device declaration statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT Object;
- PSTR ScopeName;
- if (Context->PreviousStatement == NULL) {
- Statement->SavedScope = NULL;
- ASSERT(Statement->Argument[0]->Type == AcpiObjectString);
- ScopeName = Statement->Argument[0]->U.String.String;
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("Scope (%s) {", ScopeName);
- }
- if (Context->ExecuteStatements != FALSE) {
- //
- // Go find the object.
- //
- Object = AcpipGetNamespaceObject(ScopeName, Context->CurrentScope);
- //
- // Make this device the current scope.
- //
- Statement->SavedScope = Context->CurrentScope;
- Context->CurrentScope = Object;
- }
- Context->IndentationLevel += 1;
- }
- //
- // If execution is not done with the scope, keep this statement on the
- // stack.
- //
- if (Context->CurrentOffset < Statement->AdditionalData) {
- AcpipPrintIndentedNewLine(Context);
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // Pop this puppy off the stack.
- //
- Context->CurrentScope = Statement->SavedScope;
- Context->IndentationLevel -= 1;
- if (Context->PrintStatements != FALSE) {
- AcpipPrintIndentedNewLine(Context);
- RtlDebugPrint("}");
- }
- AcpipPrintIndentedNewLine(Context);
- Statement->Reduction = NULL;
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateSizeOfStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a "Size Of" statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT Object;
- ULONGLONG Size;
- //
- // Gather arguments.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("SizeOf (");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- } else {
- Statement->Argument[0] = NULL;
- if (Context->ExecuteStatements != FALSE) {
- Statement->Argument[0] = Context->PreviousStatement->Reduction;
- AcpipObjectAddReference(Statement->Argument[0]);
- }
- Statement->ArgumentsAcquired += 1;
- }
- }
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint(")");
- }
- Object = NULL;
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements != FALSE) {
- //
- // If the object is an alias, use the destination.
- //
- Object = Statement->Argument[0];
- while (Object->Type == AcpiObjectAlias) {
- Object = Statement->Argument[0]->U.Alias.DestinationObject;
- }
- Size = 0;
- switch (Object->Type) {
- case AcpiObjectString:
- if (Object->U.String.String != NULL) {
- Size = RtlStringLength(Object->U.String.String);
- }
- break;
- case AcpiObjectBuffer:
- Size = Object->U.Buffer.Length;
- break;
- case AcpiObjectPackage:
- Size = Object->U.Package.ElementCount;
- break;
- default:
- return STATUS_NOT_SUPPORTED;
- }
- //
- // Create the integer result.
- //
- Statement->Reduction = AcpipCreateNamespaceObject(Context,
- AcpiObjectInteger,
- NULL,
- &Size,
- sizeof(ULONGLONG));
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateStoreStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a Store statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT NewArgument;
- KSTATUS Status;
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("Store (");
- } else if (Statement->ArgumentsAcquired == 0) {
- RtlDebugPrint(", ");
- } else {
- RtlDebugPrint(")");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- //
- // Fail if there is no argument there.
- //
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- AcpipObjectAddReference(NewArgument);
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- }
- }
- //
- // All arguments have been acquired.
- //
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements == FALSE) {
- return STATUS_SUCCESS;
- }
- //
- // Store the first argument into the second. The reduction of the statement
- // is the second operand.
- //
- Status = AcpipPerformStoreOperation(Context,
- Statement->Argument[0],
- Statement->Argument[1]);
- if (!KSUCCESS(Status)) {
- return Status;
- }
- Statement->Reduction = Statement->Argument[1];
- AcpipObjectAddReference(Statement->Reduction);
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateSyncObjectStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a Release (mutex), Reset (event), or Signal (event)
- statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT NameString;
- PACPI_OBJECT NewArgument;
- //
- // Gather arguments if needed.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- if (Statement->Type == AmlStatementRelease) {
- RtlDebugPrint("Release (");
- } else if (Statement->Type == AmlStatementSignal) {
- RtlDebugPrint("Signal (");
- } else {
- ASSERT(Statement->Type == AmlStatementReset);
- RtlDebugPrint("Reset (");
- }
- }
- }
- //
- // The argument for Release, Reset, and Signal is a "SuperName", which
- // is a SimpleName, DebugOp, or Type6Opcode. If this is the first time
- // through, try to parse a name string.
- //
- if (Context->PreviousStatement == NULL) {
- NameString = AcpipParseNameString(Context);
- if (NameString == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint(NameString->U.String.String);
- }
- if (Context->ExecuteStatements != FALSE) {
- Statement->Argument[0] =
- AcpipGetNamespaceObject(NameString->U.String.String,
- Context->CurrentScope);
- if (Statement->Argument[0] == NULL) {
- return STATUS_NOT_FOUND;
- }
- } else {
- Statement->Argument[0] = NULL;
- }
- if (Statement->Argument[0] != NULL) {
- AcpipObjectAddReference(Statement->Argument[0]);
- }
- Statement->ArgumentsAcquired += 1;
- AcpipObjectReleaseReference(NameString);
- //
- // Increment the reference count on the object.
- //
- } else {
- if (Context->ExecuteStatements != FALSE) {
- NewArgument = Context->PreviousStatement->Reduction;
- if (NewArgument == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- AcpipObjectAddReference(NewArgument);
- }
- Statement->ArgumentsAcquired += 1;
- }
- }
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint(")");
- }
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements != FALSE) {
- if (Statement->Type == AmlStatementRelease) {
- ASSERT(Statement->Argument[0]->Type == AcpiObjectMutex);
- AcpipReleaseMutex(Context,
- Statement->Argument[0]->U.Mutex.OsMutex);
- } else if (Statement->Type == AmlStatementSignal) {
- ASSERT(Statement->Argument[0]->Type == AcpiObjectEvent);
- AcpipSignalEvent(Statement->Argument[0]->U.Event.OsEvent);
- } else {
- ASSERT(Statement->Type == AmlStatementReset);
- ASSERT(Statement->Argument[0]->Type == AcpiObjectEvent);
- AcpipResetEvent(Statement->Argument[0]->U.Event.OsEvent);
- }
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateThermalZoneStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a Thermal Zone declaration statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PSTR Name;
- PACPI_OBJECT ThermalZoneObject;
- if (Context->PreviousStatement == NULL) {
- Statement->SavedScope = NULL;
- ASSERT(Statement->Argument[0]->Type == AcpiObjectString);
- Name = Statement->Argument[0]->U.String.String;
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint("ThermalZone (%s) {", Name);
- }
- if (Context->ExecuteStatements != FALSE) {
- //
- // Create the thermal zone object.
- //
- ThermalZoneObject = AcpipCreateNamespaceObject(
- Context,
- AcpiObjectPowerResource,
- Name,
- NULL,
- 0);
- if (ThermalZoneObject == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- //
- // Make this device the current scope.
- //
- Statement->SavedScope = Context->CurrentScope;
- Context->CurrentScope = ThermalZoneObject;
- Statement->Reduction = ThermalZoneObject;
- }
- Context->IndentationLevel += 1;
- }
- //
- // If execution is not done with the scope, keep this statement on the
- // stack.
- //
- if (Context->CurrentOffset < Statement->AdditionalData) {
- AcpipPrintIndentedNewLine(Context);
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // Pop this puppy off the stack.
- //
- Context->CurrentScope = Statement->SavedScope;
- Context->IndentationLevel -= 1;
- if (Context->PrintStatements != FALSE) {
- AcpipPrintIndentedNewLine(Context);
- RtlDebugPrint("}");
- }
- AcpipPrintIndentedNewLine(Context);
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateToFormatStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates the "To" something and "From" something statements,
- including ToBCD, ToBuffer, ToDecimalString, ToHexString, ToInteger,
- ToString, ToUUID, Unicode, and FromBcd.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- ULONGLONG BcdValue;
- ULONG ByteCount;
- ULONG ByteIndex;
- PUCHAR BytePointer;
- ULONG ByteStringSize;
- ULONG Digit;
- ULONGLONG IntegerValue;
- PACPI_OBJECT NewArgument;
- ULONG Nibble;
- CHAR ResultString[MAX_DECIMAL_STRING_LENGTH];
- ULONGLONG ResultValue;
- PSTR String;
- ULONG StringSize;
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- switch (Statement->Type) {
- case AmlStatementFromBcd:
- RtlDebugPrint("FromBCD (");
- break;
- case AmlStatementToBcd:
- RtlDebugPrint("ToBCD (");
- break;
- case AmlStatementToBuffer:
- RtlDebugPrint("ToBuffer (");
- break;
- case AmlStatementToDecimalString:
- RtlDebugPrint("ToDecimalString (");
- break;
- case AmlStatementToHexString:
- RtlDebugPrint("ToHexString (");
- break;
- case AmlStatementToInteger:
- RtlDebugPrint("ToInteger (");
- break;
- case AmlStatementToString:
- RtlDebugPrint("ToString (");
- break;
- default:
- ASSERT(FALSE);
- return STATUS_NOT_SUPPORTED;
- }
- } else {
- RtlDebugPrint(")");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- if (Statement->ArgumentsAcquired == 0) {
- //
- // Fail if there is no argument there.
- //
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- //
- // If it's not an integer, buffer, or string, fail.
- //
- if ((NewArgument->Type != AcpiObjectInteger) &&
- (NewArgument->Type != AcpiObjectBuffer) &&
- (NewArgument->Type != AcpiObjectString)) {
- return STATUS_INVALID_PARAMETER;
- }
- //
- // Perform an implicit conversion if needed.
- //
- if (((Statement->Type == AmlStatementToBcd) ||
- (Statement->Type == AmlStatementFromBcd)) &&
- (NewArgument->Type != AcpiObjectInteger)) {
- NewArgument = AcpipConvertObjectType(Context,
- NewArgument,
- AcpiObjectInteger);
- if (NewArgument == NULL) {
- return STATUS_CONVERSION_FAILED;
- }
- //
- // The object is fine, take ownership of it.
- //
- } else {
- AcpipObjectAddReference(NewArgument);
- }
- } else {
- ASSERT(Statement->ArgumentsAcquired == 1);
- AcpipObjectAddReference(NewArgument);
- }
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- }
- }
- //
- // Evaluate the result.
- //
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements == FALSE) {
- return STATUS_SUCCESS;
- }
- switch (Statement->Type) {
- case AmlStatementFromBcd:
- //
- // Convert the BCD value to an integer.
- //
- ASSERT(Statement->Argument[0]->Type == AcpiObjectInteger);
- BcdValue = Statement->Argument[0]->U.Integer.Value;
- ResultValue = 0;
- for (Nibble = 0; Nibble < sizeof(ULONGLONG) * 2; Nibble += 1) {
- Digit = (BcdValue & 0xF000000000000000ULL) >> 60;
- ResultValue = (ResultValue * 10) + Digit;
- Digit = Digit << 4;
- }
- if (Context->CurrentMethod->IntegerWidthIs32 != FALSE) {
- ResultValue &= 0xFFFFFFFF;
- }
- Statement->Reduction = AcpipCreateNamespaceObject(Context,
- AcpiObjectInteger,
- NULL,
- &ResultValue,
- sizeof(ULONGLONG));
- break;
- case AmlStatementToBcd:
- //
- // Convert the integer to a BCD value.
- //
- ASSERT(Statement->Argument[0]->Type == AcpiObjectInteger);
- IntegerValue = Statement->Argument[0]->U.Integer.Value;
- ResultValue = 0;
- for (Nibble = 0; Nibble < sizeof(ULONGLONG) * 2; Nibble += 1) {
- if (IntegerValue == 0) {
- break;
- }
- Digit = IntegerValue % 10;
- ResultValue = (ResultValue << 4) | Digit;
- IntegerValue = IntegerValue / 10;
- }
- if (IntegerValue != 0) {
- return STATUS_CONVERSION_FAILED;
- }
- Statement->Reduction = AcpipCreateNamespaceObject(Context,
- AcpiObjectInteger,
- NULL,
- &ResultValue,
- sizeof(ULONGLONG));
- break;
- case AmlStatementToBuffer:
- Statement->Reduction = AcpipConvertObjectType(Context,
- Statement->Argument[0],
- AcpiObjectBuffer);
- break;
- case AmlStatementToDecimalString:
- //
- // If the result is already a string, no action is performed.
- //
- if (Statement->Argument[0]->Type == AcpiObjectString) {
- Statement->Reduction = Statement->Argument[0];
- AcpipObjectAddReference(Statement->Reduction);
- //
- // Convert the integer to a string.
- //
- } else if (Statement->Argument[0]->Type == AcpiObjectInteger) {
- RtlPrintToString(ResultString,
- MAX_DECIMAL_STRING_LENGTH,
- CharacterEncodingAscii,
- "%I64d",
- Statement->Argument[0]->U.Integer.Value);
- Statement->Reduction = AcpipCreateNamespaceObject(
- Context,
- AcpiObjectString,
- NULL,
- ResultString,
- MAX_DECIMAL_STRING_LENGTH);
- //
- // Convert the buffer to a comma delimited string of decimal integers.
- //
- } else if (Statement->Argument[0]->Type == AcpiObjectBuffer) {
- //
- // Create the result string with buffer first. The size is up to
- // three decimal digits, plus one comma per byte, minus the comma
- // at the end, plus the null delimiter.
- //
- StringSize = Statement->Argument[0]->U.Buffer.Length * 4;
- if (StringSize == 0) {
- StringSize = 1;
- }
- Statement->Reduction = AcpipCreateNamespaceObject(Context,
- AcpiObjectString,
- NULL,
- NULL,
- StringSize);
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- //
- // Print out each byte individually, except the last one.
- //
- String = Statement->Reduction->U.String.String;
- BytePointer = Statement->Argument[0]->U.Buffer.Buffer;
- ByteCount = Statement->Argument[0]->U.Buffer.Length;
- for (ByteIndex = 0; ByteIndex < ByteCount - 1; ByteIndex += 1) {
- ByteStringSize = RtlPrintToString(String,
- StringSize,
- CharacterEncodingAscii,
- "%d,",
- *BytePointer);
- if (ByteStringSize > StringSize) {
- ByteStringSize = StringSize;
- }
- BytePointer += 1;
- String += ByteStringSize;
- StringSize -= ByteStringSize;
- }
- //
- // Do the last one without a comma.
- //
- RtlPrintToString(String,
- StringSize,
- CharacterEncodingAscii,
- "%d",
- *BytePointer);
- } else {
- ASSERT(FALSE);
- return STATUS_NOT_SUPPORTED;
- }
- break;
- case AmlStatementToHexString:
- //
- // If the result is already a string, no action is performed.
- //
- if (Statement->Argument[0]->Type == AcpiObjectString) {
- Statement->Reduction = Statement->Argument[0];
- AcpipObjectAddReference(Statement->Reduction);
- //
- // Convert the integer or buffer to a string.
- //
- } else {
- Statement->Reduction = AcpipConvertObjectType(
- Context,
- Statement->Argument[0],
- AcpiObjectString);
- }
- break;
- case AmlStatementToInteger:
- if (Statement->Argument[0]->Type == AcpiObjectInteger) {
- Statement->Reduction = Statement->Argument[0];
- AcpipObjectAddReference(Statement->Reduction);
- } else if (Statement->Argument[0]->Type == AcpiObjectBuffer) {
- Statement->Reduction = AcpipConvertObjectType(
- Context,
- Statement->Argument[0],
- AcpiObjectInteger);
- } else if (Statement->Argument[0]->Type != AcpiObjectString) {
- ASSERT(FALSE);
- return STATUS_NOT_SUPPORTED;
- }
- String = Statement->Argument[0]->U.String.String;
- if (String == NULL) {
- break;
- }
- //
- // Parse the string as a decimal or a hex string depending on whether
- // there is an 0x prepending or not.
- //
- IntegerValue = 0;
- if ((*String == '0') && (*(String + 1) == 'x')) {
- while (((*String >= '0') && (*String <= '9')) ||
- ((*String >= 'a') && (*String <= 'f')) ||
- ((*String >= 'A') && (*String <= 'F'))) {
- if ((*String >= '0') && (*String <= '9')) {
- Digit = *String - '0';
- } else if (((*String >= 'a') && (*String <= 'f'))) {
- Digit = *String - 'a';
- } else {
- ASSERT((*String >= 'A') && (*String <= 'F'));
- Digit = *String - 'A';
- }
- IntegerValue = (IntegerValue << 4) | Digit;
- String += 1;
- }
- //
- // Parse it as a decimal string.
- //
- } else {
- while ((*String >= '0') && (*String <= '9')) {
- Digit = *String - '0';
- IntegerValue = (IntegerValue * 10) + Digit;
- String += 1;
- }
- }
- Statement->Reduction = AcpipCreateNamespaceObject(Context,
- AcpiObjectInteger,
- NULL,
- &IntegerValue,
- sizeof(ULONGLONG));
- break;
- case AmlStatementToString:
- Statement->Reduction = AcpipConvertObjectType(Context,
- Statement->Argument[0],
- AcpiObjectString);
- break;
- default:
- ASSERT(FALSE);
- return STATUS_NOT_SUPPORTED;
- }
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- //
- // Store the result in the target if supplied.
- //
- if (Statement->Argument[2] != NULL) {
- return AcpipPerformStoreOperation(Context,
- Statement->Reduction,
- Statement->Argument[2]);
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateUnloadStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates an Unload statement, which unloads a previously
- loaded definition block.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT DdbHandle;
- PACPI_OBJECT NewArgument;
- //
- // If not all arguments are acquired, evaluate the previous statement to get
- // the next argument.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("Unload (");
- } else if (Statement->ArgumentsAcquired == 0) {
- RtlDebugPrint(")");
- }
- }
- if (Context->PreviousStatement == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- ASSERT(Statement->ArgumentsAcquired <= 1);
- if (Context->PreviousStatement->Reduction != NULL) {
- AcpipObjectAddReference(NewArgument);
- }
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- }
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- }
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements == FALSE) {
- return STATUS_SUCCESS;
- }
- DdbHandle = Statement->Argument[0];
- ASSERT(DdbHandle != NULL);
- AcpiUnloadDefinitionBlock(DdbHandle);
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateWaitStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates an Wait (for Event) statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT IntegerObject;
- PACPI_OBJECT NameString;
- PACPI_OBJECT NewArgument;
- ULONGLONG ResultValue;
- ULONG TimeoutValue;
- //
- // Gather arguments if needed.
- //
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("Wait (");
- } else if (Statement->ArgumentsAcquired == 0) {
- RtlDebugPrint(", ");
- } else {
- RtlDebugPrint(")");
- }
- }
- //
- // The argument for Wait is a "SuperName", which is a SimpleName,
- // DebugOp, or Type6Opcode. If this is the first time through, try to
- // parse a name string.
- //
- if (Context->PreviousStatement == NULL) {
- NameString = AcpipParseNameString(Context);
- if (NameString == NULL) {
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- if (Context->PrintStatements != FALSE) {
- RtlDebugPrint(NameString->U.String.String);
- }
- if (Context->ExecuteStatements != FALSE) {
- Statement->Argument[0] =
- AcpipGetNamespaceObject(NameString->U.String.String,
- Context->CurrentScope);
- if (Statement->Argument[0] == NULL) {
- return STATUS_NOT_FOUND;
- }
- AcpipObjectAddReference(Statement->Argument[0]);
- }
- Statement->ArgumentsAcquired += 1;
- AcpipObjectReleaseReference(NameString);
- //
- // Get the argument from the previous statement.
- //
- } else {
- if (Context->ExecuteStatements != FALSE) {
- NewArgument = Context->PreviousStatement->Reduction;
- if (NewArgument == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- //
- // The first argument is the Event object.
- //
- if (Statement->ArgumentsAcquired == 0) {
- if (NewArgument->Type != AcpiObjectEvent) {
- return STATUS_INVALID_PARAMETER;
- }
- Statement->Argument[Statement->ArgumentsAcquired] =
- NewArgument;
- AcpipObjectAddReference(NewArgument);
- //
- // The second argument should evaluate to an integer specifying
- // the number of milliseconds to wait for the given event.
- //
- } else {
- if (NewArgument->Type == AcpiObjectInteger) {
- Statement->Argument[Statement->ArgumentsAcquired] =
- NewArgument;
- AcpipObjectAddReference(NewArgument);
- } else {
- IntegerObject = AcpipConvertObjectType(
- Context,
- NewArgument,
- AcpiObjectInteger);
- if (IntegerObject == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- Statement->Argument[Statement->ArgumentsAcquired] =
- IntegerObject;
- }
- }
- }
- Statement->ArgumentsAcquired += 1;
- }
- }
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- Statement->Reduction = NULL;
- if (Context->ExecuteStatements != FALSE) {
- ASSERT(Statement->Argument[0]->Type == AcpiObjectEvent);
- ASSERT(Statement->Argument[1]->Type == AcpiObjectInteger);
- TimeoutValue = Statement->Argument[1]->U.Integer.Value;
- ResultValue = AcpipWaitForEvent(Statement->Argument[0]->U.Event.OsEvent,
- TimeoutValue);
- Statement->Reduction = AcpipCreateNamespaceObject(Context,
- AcpiObjectInteger,
- NULL,
- &ResultValue,
- sizeof(ULONGLONG));
- if (Statement->Reduction == NULL) {
- return STATUS_UNSUCCESSFUL;
- }
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateWhileModifierStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates one of the statements that modifies a While loop,
- a Break or Continue.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- BOOL Continue;
- if (Context->PrintStatements != FALSE) {
- if (Statement->Type == AmlStatementBreak) {
- RtlDebugPrint("Break");
- } else {
- ASSERT(Statement->Type == AmlStatementContinue);
- RtlDebugPrint("Continue");
- }
- }
- if (Context->ExecuteStatements != FALSE) {
- if (Statement->Type == AmlStatementContinue) {
- Continue = TRUE;
- } else {
- Continue = FALSE;
- }
- AcpipPopExecutingStatements(Context, TRUE, Continue);
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipEvaluateWhileStatement (
- PAML_EXECUTION_CONTEXT Context,
- PAML_STATEMENT Statement
- )
- /*++
- Routine Description:
- This routine evaluates a While statement.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Statement - Supplies a pointer to the statement to evaluate.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT NewArgument;
- if (Statement->ArgumentsNeeded != Statement->ArgumentsAcquired) {
- //
- // Print out the piece of the statement depending on the number of
- // arguments acquired.
- //
- if (Context->PrintStatements != FALSE) {
- if (Context->PreviousStatement == NULL) {
- RtlDebugPrint("While (");
- } else {
- ASSERT(Statement->ArgumentsAcquired == 0);
- RtlDebugPrint(") {");
- }
- }
- if (Context->PreviousStatement == NULL) {
- Context->IndentationLevel += 1;
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- //
- // If not executing, then assume the argument would be there but don't
- // try to dink with it.
- //
- if (Context->ExecuteStatements == FALSE) {
- Statement->Argument[Statement->ArgumentsAcquired] = NULL;
- Statement->ArgumentsAcquired += 1;
- } else {
- NewArgument = Context->PreviousStatement->Reduction;
- //
- // Fail if there is no argument there.
- //
- if (Context->PreviousStatement->Reduction == NULL) {
- return STATUS_ARGUMENT_EXPECTED;
- }
- //
- // Perform an implicit conversion if needed.
- //
- if (NewArgument->Type != AcpiObjectInteger) {
- NewArgument = AcpipConvertObjectType(Context,
- NewArgument,
- AcpiObjectInteger);
- if (NewArgument == NULL) {
- return STATUS_CONVERSION_FAILED;
- }
- //
- // The object is fine, take ownership of it.
- //
- } else {
- AcpipObjectAddReference(NewArgument);
- }
- Statement->Argument[Statement->ArgumentsAcquired] = NewArgument;
- Statement->ArgumentsAcquired += 1;
- }
- //
- // Evaluate the predicate.
- //
- ASSERT(Statement->ArgumentsNeeded == Statement->ArgumentsAcquired);
- if (Context->ExecuteStatements != FALSE) {
- ASSERT(Statement->Argument[0]->Type == AcpiObjectInteger);
- //
- // Evaluate the while statement by skipping the package length if
- // it's zero, and completing the while statement.
- //
- if (Statement->Argument[0]->U.Integer.Value == 0) {
- Context->CurrentOffset = Statement->AdditionalData;
- Context->IndentationLevel -= 1;
- if (Context->PrintStatements != FALSE) {
- AcpipPrintIndentedNewLine(Context);
- RtlDebugPrint("}");
- }
- return STATUS_SUCCESS;
- }
- }
- }
- //
- // If execution is not done with the scope, keep this statement on the
- // stack.
- //
- if (Context->CurrentOffset < Statement->AdditionalData) {
- AcpipPrintIndentedNewLine(Context);
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- ASSERT(Context->IndentationLevel != 0);
- //
- // Move the offset back to the predicate, release the argument to pretend
- // like the predicate was never seen before, and start again.
- //
- if (Context->ExecuteStatements != FALSE) {
- Context->CurrentOffset = Statement->AdditionalData2;
- AcpipObjectReleaseReference(Statement->Argument[0]);
- Statement->Argument[0] = NULL;
- Statement->ArgumentsAcquired = 0;
- AcpipPrintIndentedNewLine(Context);
- return STATUS_MORE_PROCESSING_REQUIRED;
- }
- AcpipPrintIndentedNewLine(Context);
- return STATUS_SUCCESS;
- }
- PACPI_OBJECT
- AcpipConvertObjectType (
- PAML_EXECUTION_CONTEXT Context,
- PACPI_OBJECT Object,
- ACPI_OBJECT_TYPE NewType
- )
- /*++
- Routine Description:
- This routine performs a conversion between supported ACPI object types.
- Arguments:
- Context - Supplies a pointer to the current execution context.
- Object - Supplies a pointer to the object to convert.
- NewType - Supplies the type to convert the given object to.
- Return Value:
- Returns a pointer to a new object (unlinked to the namespace) of the
- requested type. The caller is responsible for this memory once its returned.
- NULL on failure.
- --*/
- {
- PACPI_OBJECT NewObject;
- PACPI_OBJECT ReadResult;
- KSTATUS Status;
- NewObject = NULL;
- ReadResult = NULL;
- //
- // Get the real object being pointed to here.
- //
- while (Object->Type == AcpiObjectAlias) {
- Object = Object->U.Alias.DestinationObject;
- }
- //
- // Attempting to convert from a Field Unit to something results in a read
- // from the field.
- //
- if (Object->Type == AcpiObjectFieldUnit) {
- Status = AcpipReadFromField(Context, Object, &ReadResult);
- if (!KSUCCESS(Status)) {
- return NULL;
- }
- if (ReadResult->Type == NewType) {
- return ReadResult;
- }
- //
- // The new thing to convert is the result of the field read.
- //
- Object = ReadResult;
- } else if (Object->Type == AcpiObjectBufferField) {
- Status = AcpipReadFromBufferField(Context, Object, &ReadResult);
- if (!KSUCCESS(Status)) {
- return NULL;
- }
- if (ReadResult->Type == NewType) {
- return ReadResult;
- }
- //
- // The new thing to convert is the result of the buffer field read.
- //
- Object = ReadResult;
- }
- switch (NewType) {
- case AcpiObjectInteger:
- NewObject = AcpipConvertObjectTypeToInteger(Context, Object);
- break;
- case AcpiObjectString:
- NewObject = AcpipConvertObjectTypeToString(Context, Object);
- break;
- case AcpiObjectBuffer:
- NewObject = AcpipConvertObjectTypeToBuffer(Context, Object);
- break;
- default:
- ASSERT(FALSE);
- break;
- }
- //
- // Release the intermediate read result object.
- //
- if (ReadResult != NULL) {
- AcpipObjectReleaseReference(ReadResult);
- }
- return NewObject;
- }
- KSTATUS
- AcpipResolveStoreDestination (
- PAML_EXECUTION_CONTEXT Context,
- PACPI_OBJECT Destination,
- PACPI_OBJECT *ResolvedDestination
- )
- /*++
- Routine Description:
- This routine resolves a store destination to the proper ACPI object based
- on its type and the statement type.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Destination - Supplies a pointer to the original store destination object.
- ResolvedDestination - Supplies a pointer that receives a pointer to the
- resolved destination object. This may return a pointer to the
- original destination, but with an extra reference. The caller is always
- responsible for releasing a reference on this object.
- Return Value:
- Status code.
- --*/
- {
- ULONG LastLocalIndex;
- PACPI_OBJECT *LocalVariable;
- PACPI_OBJECT ResolvedObject;
- //
- // Follow all aliases.
- //
- while (Destination->Type == AcpiObjectAlias) {
- Destination = Destination->U.Alias.DestinationObject;
- }
- //
- // If it is a local, then store is meant to release the reference on the
- // existing local variable and set the local to a new copy of the source.
- //
- ResolvedObject = Destination;
- LastLocalIndex = Context->CurrentMethod->LastLocalIndex;
- LocalVariable = Context->CurrentMethod->LocalVariable;
- if ((LastLocalIndex != AML_INVALID_LOCAL_INDEX) &&
- (LocalVariable[LastLocalIndex] == Destination)) {
- AcpipObjectReleaseReference(Destination);
- LocalVariable[LastLocalIndex] = NULL;
- ResolvedObject = AcpipCreateNamespaceObject(Context,
- AcpiObjectUninitialized,
- NULL,
- NULL,
- 0);
- if (ResolvedObject == NULL) {
- return STATUS_INSUFFICIENT_RESOURCES;
- }
- LocalVariable[LastLocalIndex] = ResolvedObject;
- }
- AcpipObjectAddReference(ResolvedObject);
- *ResolvedDestination = ResolvedObject;
- return STATUS_SUCCESS;
- }
- KSTATUS
- AcpipConvertToDataReferenceObject (
- PAML_EXECUTION_CONTEXT Context,
- PACPI_OBJECT Object,
- PACPI_OBJECT *ResultObject
- )
- /*++
- Routine Description:
- This routine performs a conversion of an object to a type in the set of
- DataRefObject types.
- Arguments:
- Context - Supplies a pointer to the current execution context.
- Object - Supplies a pointer to the object to convert.
- ResultObject - Supplies a pointer that receives a pointer to the result
- object after the conversion. If no conversion is necessary, then this
- may be a pointer to the original object. If a conversion is necessary,
- then this will be a pointer to a new object. Either way the caller is
- responsible for releasing one reference on the result object on
- success.
- Return Value:
- Status code.
- --*/
- {
- PACPI_OBJECT DataReferenceObject;
- KSTATUS Status;
- //
- // Get the real object being pointed to here.
- //
- while (Object->Type == AcpiObjectAlias) {
- Object = Object->U.Alias.DestinationObject;
- }
- DataReferenceObject = NULL;
- Status = STATUS_SUCCESS;
- switch (Object->Type) {
- //
- // Convert a field unit to an integer or buffer.
- //
- case AcpiObjectFieldUnit:
- Status = AcpipReadFromField(Context, Object, &DataReferenceObject);
- if (!KSUCCESS(Status)) {
- goto ConvertToDataReferenceObjectEnd;
- }
- ASSERT((DataReferenceObject->Type == AcpiObjectInteger) ||
- (DataReferenceObject->Type == AcpiObjectBuffer));
- break;
- //
- // Convert a buffer field into an integer or buffer.
- //
- case AcpiObjectBufferField:
- Status = AcpipReadFromBufferField(Context,
- Object,
- &DataReferenceObject);
- if (!KSUCCESS(Status)) {
- goto ConvertToDataReferenceObjectEnd;
- }
- ASSERT((DataReferenceObject->Type == AcpiObjectInteger) ||
- (DataReferenceObject->Type == AcpiObjectBuffer));
- break;
- //
- // Just add a new reference if it is already a DataReferenceObject type.
- //
- case AcpiObjectInteger:
- case AcpiObjectString:
- case AcpiObjectBuffer:
- case AcpiObjectPackage:
- case AcpiObjectDdbHandle:
- DataReferenceObject = Object;
- AcpipObjectAddReference(DataReferenceObject);
- break;
- //
- // Anything else cannot be converted and results in failure.
- //
- default:
- RtlDebugPrint("\nACPI: Unable to convert object of type %d to a "
- "DataRefObject. Context: 0x%08x, Object 0x%08x.\n",
- Object->Type,
- Context,
- Object);
- ASSERT(FALSE);
- Status = STATUS_NOT_SUPPORTED;
- break;
- }
- ConvertToDataReferenceObjectEnd:
- if (!KSUCCESS(Status)) {
- if (DataReferenceObject != NULL) {
- AcpipObjectReleaseReference(DataReferenceObject);
- DataReferenceObject = NULL;
- }
- }
- *ResultObject = DataReferenceObject;
- return Status;
- }
- //
- // --------------------------------------------------------- Internal Functions
- //
- PACPI_OBJECT
- AcpipConvertObjectTypeToInteger (
- PAML_EXECUTION_CONTEXT Context,
- PACPI_OBJECT Object
- )
- /*++
- Routine Description:
- This routine converts the given object into an Integer object.
- Arguments:
- Context - Supplies a pointer to the AML execution context.
- Object - Supplies a pointer to the object to convert.
- Return Value:
- Returns a pointer to a new object (unlinked to the namespace) of the
- requested type. The caller is responsible for this memory once its returned.
- NULL on failure.
- --*/
- {
- ULONG CopySize;
- UCHAR Digit;
- ULONGLONG IntegerValue;
- PACPI_OBJECT NewObject;
- PSTR String;
- NewObject = NULL;
- IntegerValue = 0;
- switch (Object->Type) {
- case AcpiObjectInteger:
- NewObject = AcpipCreateNamespaceObject(NULL,
- AcpiObjectInteger,
- NULL,
- &(Object->U.Integer.Value),
- sizeof(ULONGLONG));
- break;
- //
- // Convert from a buffer to an integer by basically just casting.
- //
- case AcpiObjectBuffer:
- CopySize = Object->U.Buffer.Length;
- if (CopySize > sizeof(ULONGLONG)) {
- CopySize = sizeof(ULONGLONG);
- }
- RtlCopyMemory(&IntegerValue, Object->U.Buffer.Buffer, CopySize);
- NewObject = AcpipCreateNamespaceObject(NULL,
- AcpiObjectInteger,
- NULL,
- &IntegerValue,
- sizeof(ULONGLONG));
- break;
- //
- // To convert from a string to an integer, parse hex digits 0-9, A-F
- // (and a-f) until a non-digit is found. A leading 0x is not allowed.
- //
- case AcpiObjectString:
- String = Object->U.String.String;
- if (String == NULL) {
- break;
- }
- while (((*String >= '0') && (*String <= '9')) ||
- ((*String >= 'a') && (*String <= 'f')) ||
- ((*String >= 'A') && (*String <= 'F'))) {
- if ((*String >= '0') && (*String <= '9')) {
- Digit = *String - '0';
- } else if (((*String >= 'a') && (*String <= 'f'))) {
- Digit = *String - 'a' + 10;
- } else {
- ASSERT((*String >= 'A') && (*String <= 'F'));
- Digit = *String - 'A' + 10;
- }
- IntegerValue = (IntegerValue << 4) | Digit;
- String += 1;
- }
- NewObject = AcpipCreateNamespaceObject(NULL,
- AcpiObjectInteger,
- NULL,
- &IntegerValue,
- sizeof(ULONGLONG));
- break;
- default:
- ASSERT(FALSE);
- break;
- }
- return NewObject;
- }
- PACPI_OBJECT
- AcpipConvertObjectTypeToString (
- PAML_EXECUTION_CONTEXT Context,
- PACPI_OBJECT Object
- )
- /*++
- Routine Description:
- This routine converts the given object into a String object.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Object - Supplies a pointer to the object to convert.
- Return Value:
- Returns a pointer to a new object (unlinked to the namespace) of the
- requested type. The caller is responsible for this memory once its returned.
- NULL on failure.
- --*/
- {
- ULONG BufferLength;
- ULONG ByteIndex;
- PUCHAR CurrentBufferPosition;
- PSTR CurrentPosition;
- PACPI_OBJECT NewObject;
- ULONG NewStringLength;
- switch (Object->Type) {
- //
- // To convert an integer to a string, create an 8 or 16 byte string buffer
- // depending on whether integers are 32 or 64 bits, and then write the hex
- // value in.
- //
- case AcpiObjectInteger:
- NewStringLength = 16;
- if ((Context->CurrentMethod != NULL) &&
- (Context->CurrentMethod->IntegerWidthIs32 != FALSE)) {
- NewStringLength = 8;
- }
- NewObject = AcpipCreateNamespaceObject(NULL,
- AcpiObjectString,
- NULL,
- NULL,
- NewStringLength);
- if (NewObject == NULL) {
- return NULL;
- }
- RtlPrintToString(NewObject->U.String.String,
- NewStringLength,
- CharacterEncodingAscii,
- "%I64x",
- Object->U.Integer.Value);
- break;
- case AcpiObjectString:
- NewStringLength = RtlStringLength(Object->U.String.String) + 1;
- NewObject = AcpipCreateNamespaceObject(NULL,
- AcpiObjectString,
- NULL,
- Object->U.String.String,
- NewStringLength);
- break;
- //
- // To convert from a buffer to a string, print out all characters as two
- // digit hex values, separated by spaces.
- //
- case AcpiObjectBuffer:
- //
- // The new string length is 3 times the number of bytes there are (two
- // digits plus one space for each character), minus one since the last
- // character doesn't get a space, plus one for the null terminator.
- //
- NewStringLength = (Object->U.Buffer.Length * 3);
- NewObject = AcpipCreateNamespaceObject(NULL,
- AcpiObjectString,
- NULL,
- NULL,
- NewStringLength);
- if (NewObject == NULL) {
- return NULL;
- }
- BufferLength = Object->U.Buffer.Length;
- if (BufferLength == 0) {
- NewObject->U.String.String[0] = '\0';
- } else {
- CurrentPosition = NewObject->U.String.String;
- CurrentBufferPosition = Object->U.Buffer.Buffer;
- //
- // Print out all except the last one.
- //
- for (ByteIndex = 0; ByteIndex < BufferLength - 1; ByteIndex += 1) {
- RtlPrintToString(CurrentPosition,
- NewStringLength,
- CharacterEncodingAscii,
- "%02x ",
- *CurrentBufferPosition);
- CurrentPosition += 3;
- NewStringLength -= 3;
- CurrentBufferPosition += 1;
- }
- //
- // Do the last one without a space.
- //
- RtlPrintToString(CurrentPosition,
- NewStringLength,
- CharacterEncodingAscii,
- "%02x",
- *CurrentBufferPosition);
- }
- break;
- default:
- ASSERT(FALSE);
- return NULL;
- }
- return NewObject;
- }
- PACPI_OBJECT
- AcpipConvertObjectTypeToBuffer (
- PAML_EXECUTION_CONTEXT Context,
- PACPI_OBJECT Object
- )
- /*++
- Routine Description:
- This routine converts the given object into a Buffer object.
- Arguments:
- Context - Supplies a pointer to the current AML execution context.
- Object - Supplies a pointer to the object to convert.
- Return Value:
- Returns a pointer to a new object (unlinked to the namespace) of the
- requested type. The caller is responsible for this memory once its returned.
- NULL on failure.
- --*/
- {
- ULONG BufferSize;
- PACPI_OBJECT NewObject;
- switch (Object->Type) {
- //
- // Converting from an integer to a buffer is basically a matter of casting.
- //
- case AcpiObjectInteger:
- BufferSize = sizeof(ULONGLONG);
- if ((Context->CurrentMethod != NULL) &&
- (Context->CurrentMethod->IntegerWidthIs32 != FALSE)) {
- BufferSize = sizeof(ULONG);
- }
- NewObject = AcpipCreateNamespaceObject(Context,
- AcpiObjectBuffer,
- NULL,
- &(Object->U.Integer.Value),
- BufferSize);
- break;
- case AcpiObjectString:
- BufferSize = RtlStringLength(Object->U.String.String);
- if (BufferSize != 0) {
- BufferSize += 1;
- }
- NewObject = AcpipCreateNamespaceObject(Context,
- AcpiObjectBuffer,
- NULL,
- Object->U.String.String,
- BufferSize);
- break;
- case AcpiObjectBuffer:
- BufferSize = Object->U.Buffer.Length;
- NewObject = AcpipCreateNamespaceObject(Context,
- AcpiObjectBuffer,
- NULL,
- Object->U.Buffer.Buffer,
- BufferSize);
- break;
- default:
- ASSERT(FALSE);
- return NULL;
- }
- return NewObject;
- }
- BOOL
- AcpipEvaluateMatchComparison (
- PAML_EXECUTION_CONTEXT Context,
- PACPI_OBJECT PackageElement,
- PACPI_OBJECT Operand1,
- ACPI_MATCH_OPERATOR Operator1,
- PACPI_OBJECT Operand2,
- ACPI_MATCH_OPERATOR Operator2
- )
- /*++
- Routine Description:
- This routine performs a comparison of two or more objects as defined in the
- rules for a Match statement.
- Arguments:
- Context - Supplies a pointer to the execution context.
- PackageElement - Supplies a pointer to the element indexed from the
- Match package being iterated over.
- Operand1 - Supplies a pointer to the first value to compare against. This
- must be an integer.
- Operator1 - Supplies the operator to use to compare the package element to
- Operand1.
- Operand2 - Supplies a pointer to the second value to compare against. This
- must be an integer.
- Operator2 - Supplies the operator to use to compare the package element to
- Operand2.
- Return Value:
- TRUE if the condition matches the given operators against the given
- operands.
- FALSE if the comparison does not match.
- --*/
- {
- ULONGLONG Operand;
- ULONGLONG PackageValue;
- BOOL Result;
- ASSERT((Operator1 < MatchOperatorCount) &&
- (Operator2 < MatchOperatorCount));
- ASSERT((Operator1 == MatchOperatorTrue) ||
- (Operand1->Type == AcpiObjectInteger));
- ASSERT((Operator2 == MatchOperatorTrue) ||
- (Operand2->Type == AcpiObjectInteger));
- //
- // The ACPI spec says to skip uninitialized elements.
- //
- if ((PackageElement == NULL) ||
- (PackageElement->Type == AcpiObjectUninitialized)) {
- return FALSE;
- }
- //
- // Get an object that can be evaluated as an integer. If the conversion
- // fails, the ACPI spec says to quietly skip this value.
- //
- if (PackageElement->Type == AcpiObjectInteger) {
- PackageValue = PackageElement->U.Integer.Value;
- } else {
- PackageElement = AcpipConvertObjectType(Context,
- PackageElement,
- AcpiObjectInteger);
- if (PackageElement == NULL) {
- return FALSE;
- }
- PackageValue = PackageElement->U.Integer.Value;
- AcpipObjectReleaseReference(PackageElement);
- }
- //
- // Perform the comparison on object 1.
- //
- Result = FALSE;
- Operand = 0;
- if (Operator1 != MatchOperatorTrue) {
- Operand = Operand1->U.Integer.Value;
- }
- switch (Operator1) {
- case MatchOperatorTrue:
- Result = TRUE;
- break;
- case MatchOperatorEqual:
- if (PackageValue == Operand) {
- Result = TRUE;
- }
- break;
- case MatchOperatorLessThanOrEqualTo:
- if (PackageValue <= Operand) {
- Result = TRUE;
- }
- break;
- case MatchOperatorLessThan:
- if (PackageValue < Operand) {
- Result = TRUE;
- }
- break;
- case MatchOperatorGreaterThanOrEqualTo:
- if (PackageValue >= Operand) {
- Result = TRUE;
- }
- break;
- case MatchOperatorGreaterThan:
- if (PackageValue > Operand) {
- Result = TRUE;
- }
- break;
- default:
- ASSERT(FALSE);
- break;
- }
- //
- // The function is only a match if both return TRUE. If this returned FALSE,
- // don't bother evaluating the other side.
- //
- if (Result == FALSE) {
- return Result;
- }
- //
- // Evalute operand 2.
- //
- Result = FALSE;
- if (Operator2 != MatchOperatorTrue) {
- Operand = Operand2->U.Integer.Value;
- }
- switch (Operator2) {
- case MatchOperatorTrue:
- Result = TRUE;
- break;
- case MatchOperatorEqual:
- if (PackageValue == Operand) {
- Result = TRUE;
- }
- break;
- case MatchOperatorLessThanOrEqualTo:
- if (PackageValue <= Operand) {
- Result = TRUE;
- }
- break;
- case MatchOperatorLessThan:
- if (PackageValue < Operand) {
- Result = TRUE;
- }
- break;
- case MatchOperatorGreaterThanOrEqualTo:
- if (PackageValue >= Operand) {
- Result = TRUE;
- }
- break;
- case MatchOperatorGreaterThan:
- if (PackageValue > Operand) {
- Result = TRUE;
- }
- break;
- default:
- ASSERT(FALSE);
- break;
- }
- return Result;
- }
|