php_reflection.c 202 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902490349044905490649074908490949104911491249134914491549164917491849194920492149224923492449254926492749284929493049314932493349344935493649374938493949404941494249434944494549464947494849494950495149524953495449554956495749584959496049614962496349644965496649674968496949704971497249734974497549764977497849794980498149824983498449854986498749884989499049914992499349944995499649974998499950005001500250035004500550065007500850095010501150125013501450155016501750185019502050215022502350245025502650275028502950305031503250335034503550365037503850395040504150425043504450455046504750485049505050515052505350545055505650575058505950605061506250635064506550665067506850695070507150725073507450755076507750785079508050815082508350845085508650875088508950905091509250935094509550965097509850995100510151025103510451055106510751085109511051115112511351145115511651175118511951205121512251235124512551265127512851295130513151325133513451355136513751385139514051415142514351445145514651475148514951505151515251535154515551565157515851595160516151625163516451655166516751685169517051715172517351745175517651775178517951805181518251835184518551865187518851895190519151925193519451955196519751985199520052015202520352045205520652075208520952105211521252135214521552165217521852195220522152225223522452255226522752285229523052315232523352345235523652375238523952405241524252435244524552465247524852495250525152525253525452555256525752585259526052615262526352645265526652675268526952705271527252735274527552765277527852795280528152825283528452855286528752885289529052915292529352945295529652975298529953005301530253035304530553065307530853095310531153125313531453155316531753185319532053215322532353245325532653275328532953305331533253335334533553365337533853395340534153425343534453455346534753485349535053515352535353545355535653575358535953605361536253635364536553665367536853695370537153725373537453755376537753785379538053815382538353845385538653875388538953905391539253935394539553965397539853995400540154025403540454055406540754085409541054115412541354145415541654175418541954205421542254235424542554265427542854295430543154325433543454355436543754385439544054415442544354445445544654475448544954505451545254535454545554565457545854595460546154625463546454655466546754685469547054715472547354745475547654775478547954805481548254835484548554865487548854895490549154925493549454955496549754985499550055015502550355045505550655075508550955105511551255135514551555165517551855195520552155225523552455255526552755285529553055315532553355345535553655375538553955405541554255435544554555465547554855495550555155525553555455555556555755585559556055615562556355645565556655675568556955705571557255735574557555765577557855795580558155825583558455855586558755885589559055915592559355945595559655975598559956005601560256035604560556065607560856095610561156125613561456155616561756185619562056215622562356245625562656275628562956305631563256335634563556365637563856395640564156425643564456455646564756485649565056515652565356545655565656575658565956605661566256635664566556665667566856695670567156725673567456755676567756785679568056815682568356845685568656875688568956905691569256935694569556965697569856995700570157025703570457055706570757085709571057115712571357145715571657175718571957205721572257235724572557265727572857295730573157325733573457355736573757385739574057415742574357445745574657475748574957505751575257535754575557565757575857595760576157625763576457655766576757685769577057715772577357745775577657775778577957805781578257835784578557865787578857895790579157925793579457955796579757985799580058015802580358045805580658075808580958105811581258135814581558165817581858195820582158225823582458255826582758285829583058315832583358345835583658375838583958405841584258435844584558465847584858495850585158525853585458555856585758585859586058615862586358645865586658675868586958705871587258735874587558765877587858795880588158825883588458855886588758885889589058915892589358945895589658975898589959005901590259035904590559065907590859095910591159125913591459155916591759185919592059215922592359245925592659275928592959305931593259335934593559365937593859395940594159425943594459455946594759485949595059515952595359545955595659575958595959605961596259635964596559665967596859695970597159725973597459755976597759785979598059815982598359845985598659875988598959905991599259935994599559965997599859996000600160026003600460056006600760086009601060116012601360146015601660176018601960206021602260236024602560266027602860296030603160326033603460356036603760386039604060416042604360446045604660476048604960506051605260536054605560566057605860596060606160626063606460656066606760686069607060716072607360746075607660776078607960806081608260836084608560866087608860896090609160926093609460956096609760986099610061016102610361046105610661076108610961106111611261136114611561166117611861196120612161226123612461256126612761286129613061316132613361346135613661376138613961406141614261436144614561466147614861496150615161526153615461556156615761586159616061616162616361646165616661676168616961706171617261736174617561766177617861796180618161826183618461856186618761886189619061916192619361946195619661976198619962006201620262036204620562066207620862096210621162126213621462156216621762186219622062216222622362246225622662276228622962306231623262336234623562366237623862396240624162426243624462456246624762486249625062516252625362546255625662576258625962606261626262636264626562666267626862696270627162726273627462756276627762786279628062816282628362846285628662876288628962906291629262936294629562966297629862996300630163026303630463056306630763086309631063116312631363146315631663176318631963206321632263236324632563266327632863296330633163326333633463356336633763386339634063416342634363446345634663476348634963506351635263536354635563566357635863596360636163626363636463656366636763686369637063716372637363746375637663776378637963806381638263836384638563866387638863896390639163926393639463956396639763986399640064016402640364046405640664076408640964106411641264136414641564166417641864196420642164226423642464256426642764286429643064316432643364346435643664376438643964406441644264436444644564466447644864496450645164526453645464556456645764586459646064616462646364646465646664676468646964706471647264736474647564766477647864796480648164826483648464856486648764886489649064916492649364946495649664976498649965006501650265036504650565066507650865096510651165126513651465156516651765186519652065216522652365246525652665276528652965306531653265336534653565366537653865396540654165426543654465456546654765486549655065516552655365546555655665576558655965606561656265636564656565666567656865696570657165726573657465756576657765786579658065816582658365846585658665876588658965906591659265936594659565966597659865996600660166026603660466056606660766086609661066116612661366146615661666176618661966206621662266236624662566266627662866296630663166326633663466356636663766386639664066416642664366446645664666476648664966506651665266536654665566566657665866596660666166626663666466656666666766686669667066716672667366746675667666776678667966806681668266836684668566866687668866896690669166926693669466956696669766986699670067016702670367046705670667076708670967106711671267136714671567166717671867196720672167226723672467256726672767286729673067316732673367346735673667376738673967406741674267436744674567466747674867496750675167526753675467556756675767586759676067616762
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 7 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1997-2018 The PHP Group |
  6. +----------------------------------------------------------------------+
  7. | This source file is subject to version 3.01 of the PHP license, |
  8. | that is bundled with this package in the file LICENSE, and is |
  9. | available through the world-wide-web at the following url: |
  10. | http://www.php.net/license/3_01.txt |
  11. | If you did not receive a copy of the PHP license and are unable to |
  12. | obtain it through the world-wide-web, please send a note to |
  13. | license@php.net so we can mail you a copy immediately. |
  14. +----------------------------------------------------------------------+
  15. | Authors: Timm Friebe <thekid@thekid.de> |
  16. | George Schlossnagle <george@omniti.com> |
  17. | Andrei Zmievski <andrei@gravitonic.com> |
  18. | Marcus Boerger <helly@php.net> |
  19. | Johannes Schlueter <johannes@php.net> |
  20. +----------------------------------------------------------------------+
  21. */
  22. #ifdef HAVE_CONFIG_H
  23. #include "config.h"
  24. #endif
  25. #include "php.h"
  26. #include "php_ini.h"
  27. #include "php_reflection.h"
  28. #include "ext/standard/info.h"
  29. #include "zend.h"
  30. #include "zend_API.h"
  31. #include "zend_exceptions.h"
  32. #include "zend_operators.h"
  33. #include "zend_constants.h"
  34. #include "zend_ini.h"
  35. #include "zend_interfaces.h"
  36. #include "zend_closures.h"
  37. #include "zend_generators.h"
  38. #include "zend_extensions.h"
  39. #include "zend_builtin_functions.h"
  40. #include "zend_smart_str.h"
  41. #define reflection_update_property(object, name, value) do { \
  42. zval member; \
  43. ZVAL_STR(&member, name); \
  44. zend_std_write_property(object, &member, value, NULL); \
  45. Z_TRY_DELREF_P(value); \
  46. zval_ptr_dtor(&member); \
  47. } while (0)
  48. #define reflection_update_property_name(object, value) \
  49. reflection_update_property(object, ZSTR_KNOWN(ZEND_STR_NAME), value)
  50. #define reflection_update_property_class(object, value) \
  51. reflection_update_property(object, ZSTR_KNOWN(ZEND_STR_CLASS), value)
  52. /* Class entry pointers */
  53. PHPAPI zend_class_entry *reflector_ptr;
  54. PHPAPI zend_class_entry *reflection_exception_ptr;
  55. PHPAPI zend_class_entry *reflection_ptr;
  56. PHPAPI zend_class_entry *reflection_function_abstract_ptr;
  57. PHPAPI zend_class_entry *reflection_function_ptr;
  58. PHPAPI zend_class_entry *reflection_generator_ptr;
  59. PHPAPI zend_class_entry *reflection_parameter_ptr;
  60. PHPAPI zend_class_entry *reflection_type_ptr;
  61. PHPAPI zend_class_entry *reflection_named_type_ptr;
  62. PHPAPI zend_class_entry *reflection_class_ptr;
  63. PHPAPI zend_class_entry *reflection_object_ptr;
  64. PHPAPI zend_class_entry *reflection_method_ptr;
  65. PHPAPI zend_class_entry *reflection_property_ptr;
  66. PHPAPI zend_class_entry *reflection_class_constant_ptr;
  67. PHPAPI zend_class_entry *reflection_extension_ptr;
  68. PHPAPI zend_class_entry *reflection_zend_extension_ptr;
  69. /* Exception throwing macro */
  70. #define _DO_THROW(msg) \
  71. zend_throw_exception(reflection_exception_ptr, msg, 0); \
  72. return; \
  73. #define RETURN_ON_EXCEPTION \
  74. if (EG(exception) && EG(exception)->ce == reflection_exception_ptr) { \
  75. return; \
  76. }
  77. #define GET_REFLECTION_OBJECT() \
  78. intern = Z_REFLECTION_P(getThis()); \
  79. if (intern->ptr == NULL) { \
  80. RETURN_ON_EXCEPTION \
  81. zend_throw_error(NULL, "Internal error: Failed to retrieve the reflection object"); \
  82. return; \
  83. } \
  84. #define GET_REFLECTION_OBJECT_PTR(target) \
  85. GET_REFLECTION_OBJECT() \
  86. target = intern->ptr; \
  87. /* Class constants */
  88. #define REGISTER_REFLECTION_CLASS_CONST_LONG(class_name, const_name, value) \
  89. zend_declare_class_constant_long(reflection_ ## class_name ## _ptr, const_name, sizeof(const_name)-1, (zend_long)value);
  90. /* {{{ Object structure */
  91. /* Struct for properties */
  92. typedef struct _property_reference {
  93. zend_class_entry *ce;
  94. zend_property_info prop;
  95. zend_string *unmangled_name;
  96. } property_reference;
  97. /* Struct for parameters */
  98. typedef struct _parameter_reference {
  99. uint32_t offset;
  100. zend_bool required;
  101. struct _zend_arg_info *arg_info;
  102. zend_function *fptr;
  103. } parameter_reference;
  104. /* Struct for type hints */
  105. typedef struct _type_reference {
  106. struct _zend_arg_info *arg_info;
  107. zend_function *fptr;
  108. } type_reference;
  109. typedef enum {
  110. REF_TYPE_OTHER, /* Must be 0 */
  111. REF_TYPE_FUNCTION,
  112. REF_TYPE_GENERATOR,
  113. REF_TYPE_PARAMETER,
  114. REF_TYPE_TYPE,
  115. REF_TYPE_PROPERTY,
  116. REF_TYPE_CLASS_CONSTANT
  117. } reflection_type_t;
  118. /* Struct for reflection objects */
  119. typedef struct {
  120. zval dummy; /* holder for the second property */
  121. zval obj;
  122. void *ptr;
  123. zend_class_entry *ce;
  124. reflection_type_t ref_type;
  125. unsigned int ignore_visibility:1;
  126. zend_object zo;
  127. } reflection_object;
  128. static inline reflection_object *reflection_object_from_obj(zend_object *obj) {
  129. return (reflection_object*)((char*)(obj) - XtOffsetOf(reflection_object, zo));
  130. }
  131. #define Z_REFLECTION_P(zv) reflection_object_from_obj(Z_OBJ_P((zv)))
  132. /* }}} */
  133. static zend_object_handlers reflection_object_handlers;
  134. static zval *_default_load_name(zval *object) /* {{{ */
  135. {
  136. return zend_hash_find_ex_ind(Z_OBJPROP_P(object), ZSTR_KNOWN(ZEND_STR_NAME), 1);
  137. }
  138. /* }}} */
  139. static void _default_get_name(zval *object, zval *return_value) /* {{{ */
  140. {
  141. zval *value;
  142. if ((value = _default_load_name(object)) == NULL) {
  143. RETURN_FALSE;
  144. }
  145. ZVAL_COPY(return_value, value);
  146. }
  147. /* }}} */
  148. static zend_function *_copy_function(zend_function *fptr) /* {{{ */
  149. {
  150. if (fptr
  151. && (fptr->internal_function.fn_flags & ZEND_ACC_CALL_VIA_TRAMPOLINE))
  152. {
  153. zend_function *copy_fptr;
  154. copy_fptr = emalloc(sizeof(zend_function));
  155. memcpy(copy_fptr, fptr, sizeof(zend_function));
  156. copy_fptr->internal_function.function_name = zend_string_copy(fptr->internal_function.function_name);
  157. return copy_fptr;
  158. } else {
  159. /* no copy needed */
  160. return fptr;
  161. }
  162. }
  163. /* }}} */
  164. static void _free_function(zend_function *fptr) /* {{{ */
  165. {
  166. if (fptr
  167. && (fptr->internal_function.fn_flags & ZEND_ACC_CALL_VIA_TRAMPOLINE))
  168. {
  169. zend_string_release_ex(fptr->internal_function.function_name, 0);
  170. zend_free_trampoline(fptr);
  171. }
  172. }
  173. /* }}} */
  174. static void reflection_free_objects_storage(zend_object *object) /* {{{ */
  175. {
  176. reflection_object *intern = reflection_object_from_obj(object);
  177. parameter_reference *reference;
  178. property_reference *prop_reference;
  179. type_reference *typ_reference;
  180. if (intern->ptr) {
  181. switch (intern->ref_type) {
  182. case REF_TYPE_PARAMETER:
  183. reference = (parameter_reference*)intern->ptr;
  184. _free_function(reference->fptr);
  185. efree(intern->ptr);
  186. break;
  187. case REF_TYPE_TYPE:
  188. typ_reference = (type_reference*)intern->ptr;
  189. _free_function(typ_reference->fptr);
  190. efree(intern->ptr);
  191. break;
  192. case REF_TYPE_FUNCTION:
  193. _free_function(intern->ptr);
  194. break;
  195. case REF_TYPE_PROPERTY:
  196. prop_reference = (property_reference*)intern->ptr;
  197. zend_string_release_ex(prop_reference->unmangled_name, 0);
  198. efree(intern->ptr);
  199. break;
  200. case REF_TYPE_GENERATOR:
  201. case REF_TYPE_CLASS_CONSTANT:
  202. case REF_TYPE_OTHER:
  203. break;
  204. }
  205. }
  206. intern->ptr = NULL;
  207. zval_ptr_dtor(&intern->obj);
  208. zend_object_std_dtor(object);
  209. }
  210. /* }}} */
  211. static HashTable *reflection_get_gc(zval *obj, zval **gc_data, int *gc_data_count) /* {{{ */
  212. {
  213. reflection_object *intern = Z_REFLECTION_P(obj);
  214. *gc_data = &intern->obj;
  215. *gc_data_count = 1;
  216. return zend_std_get_properties(obj);
  217. }
  218. /* }}} */
  219. static zend_object *reflection_objects_new(zend_class_entry *class_type) /* {{{ */
  220. {
  221. reflection_object *intern = zend_object_alloc(sizeof(reflection_object), class_type);
  222. zend_object_std_init(&intern->zo, class_type);
  223. object_properties_init(&intern->zo, class_type);
  224. intern->zo.handlers = &reflection_object_handlers;
  225. return &intern->zo;
  226. }
  227. /* }}} */
  228. static zval *reflection_instantiate(zend_class_entry *pce, zval *object) /* {{{ */
  229. {
  230. object_init_ex(object, pce);
  231. return object;
  232. }
  233. /* }}} */
  234. static void _const_string(smart_str *str, char *name, zval *value, char *indent);
  235. static void _function_string(smart_str *str, zend_function *fptr, zend_class_entry *scope, char* indent);
  236. static void _property_string(smart_str *str, zend_property_info *prop, const char *prop_name, char* indent);
  237. static void _class_const_string(smart_str *str, char *name, zend_class_constant *c, char* indent);
  238. static void _class_string(smart_str *str, zend_class_entry *ce, zval *obj, char *indent);
  239. static void _extension_string(smart_str *str, zend_module_entry *module, char *indent);
  240. static void _zend_extension_string(smart_str *str, zend_extension *extension, char *indent);
  241. /* {{{ _class_string */
  242. static void _class_string(smart_str *str, zend_class_entry *ce, zval *obj, char *indent)
  243. {
  244. int count, count_static_props = 0, count_static_funcs = 0, count_shadow_props = 0;
  245. zend_string *sub_indent = strpprintf(0, "%s ", indent);
  246. /* TBD: Repair indenting of doc comment (or is this to be done in the parser?) */
  247. if (ce->type == ZEND_USER_CLASS && ce->info.user.doc_comment) {
  248. smart_str_append_printf(str, "%s%s", indent, ZSTR_VAL(ce->info.user.doc_comment));
  249. smart_str_appendc(str, '\n');
  250. }
  251. if (obj && Z_TYPE_P(obj) == IS_OBJECT) {
  252. smart_str_append_printf(str, "%sObject of class [ ", indent);
  253. } else {
  254. char *kind = "Class";
  255. if (ce->ce_flags & ZEND_ACC_INTERFACE) {
  256. kind = "Interface";
  257. } else if (ce->ce_flags & ZEND_ACC_TRAIT) {
  258. kind = "Trait";
  259. }
  260. smart_str_append_printf(str, "%s%s [ ", indent, kind);
  261. }
  262. smart_str_append_printf(str, (ce->type == ZEND_USER_CLASS) ? "<user" : "<internal");
  263. if (ce->type == ZEND_INTERNAL_CLASS && ce->info.internal.module) {
  264. smart_str_append_printf(str, ":%s", ce->info.internal.module->name);
  265. }
  266. smart_str_append_printf(str, "> ");
  267. if (ce->get_iterator != NULL) {
  268. smart_str_append_printf(str, "<iterateable> ");
  269. }
  270. if (ce->ce_flags & ZEND_ACC_INTERFACE) {
  271. smart_str_append_printf(str, "interface ");
  272. } else if (ce->ce_flags & ZEND_ACC_TRAIT) {
  273. smart_str_append_printf(str, "trait ");
  274. } else {
  275. if (ce->ce_flags & (ZEND_ACC_IMPLICIT_ABSTRACT_CLASS|ZEND_ACC_EXPLICIT_ABSTRACT_CLASS)) {
  276. smart_str_append_printf(str, "abstract ");
  277. }
  278. if (ce->ce_flags & ZEND_ACC_FINAL) {
  279. smart_str_append_printf(str, "final ");
  280. }
  281. smart_str_append_printf(str, "class ");
  282. }
  283. smart_str_append_printf(str, "%s", ZSTR_VAL(ce->name));
  284. if (ce->parent) {
  285. smart_str_append_printf(str, " extends %s", ZSTR_VAL(ce->parent->name));
  286. }
  287. if (ce->num_interfaces) {
  288. uint32_t i;
  289. if (ce->ce_flags & ZEND_ACC_INTERFACE) {
  290. smart_str_append_printf(str, " extends %s", ZSTR_VAL(ce->interfaces[0]->name));
  291. } else {
  292. smart_str_append_printf(str, " implements %s", ZSTR_VAL(ce->interfaces[0]->name));
  293. }
  294. for (i = 1; i < ce->num_interfaces; ++i) {
  295. smart_str_append_printf(str, ", %s", ZSTR_VAL(ce->interfaces[i]->name));
  296. }
  297. }
  298. smart_str_append_printf(str, " ] {\n");
  299. /* The information where a class is declared is only available for user classes */
  300. if (ce->type == ZEND_USER_CLASS) {
  301. smart_str_append_printf(str, "%s @@ %s %d-%d\n", indent, ZSTR_VAL(ce->info.user.filename),
  302. ce->info.user.line_start, ce->info.user.line_end);
  303. }
  304. /* Constants */
  305. smart_str_append_printf(str, "\n");
  306. count = zend_hash_num_elements(&ce->constants_table);
  307. smart_str_append_printf(str, "%s - Constants [%d] {\n", indent, count);
  308. if (count > 0) {
  309. zend_string *key;
  310. zend_class_constant *c;
  311. ZEND_HASH_FOREACH_STR_KEY_PTR(&ce->constants_table, key, c) {
  312. _class_const_string(str, ZSTR_VAL(key), c, ZSTR_VAL(sub_indent));
  313. if (UNEXPECTED(EG(exception))) {
  314. return;
  315. }
  316. } ZEND_HASH_FOREACH_END();
  317. }
  318. smart_str_append_printf(str, "%s }\n", indent);
  319. /* Static properties */
  320. /* counting static properties */
  321. count = zend_hash_num_elements(&ce->properties_info);
  322. if (count > 0) {
  323. zend_property_info *prop;
  324. ZEND_HASH_FOREACH_PTR(&ce->properties_info, prop) {
  325. if(prop->flags & ZEND_ACC_SHADOW) {
  326. count_shadow_props++;
  327. } else if (prop->flags & ZEND_ACC_STATIC) {
  328. count_static_props++;
  329. }
  330. } ZEND_HASH_FOREACH_END();
  331. }
  332. /* static properties */
  333. smart_str_append_printf(str, "\n%s - Static properties [%d] {\n", indent, count_static_props);
  334. if (count_static_props > 0) {
  335. zend_property_info *prop;
  336. ZEND_HASH_FOREACH_PTR(&ce->properties_info, prop) {
  337. if ((prop->flags & ZEND_ACC_STATIC) && !(prop->flags & ZEND_ACC_SHADOW)) {
  338. _property_string(str, prop, NULL, ZSTR_VAL(sub_indent));
  339. }
  340. } ZEND_HASH_FOREACH_END();
  341. }
  342. smart_str_append_printf(str, "%s }\n", indent);
  343. /* Static methods */
  344. /* counting static methods */
  345. count = zend_hash_num_elements(&ce->function_table);
  346. if (count > 0) {
  347. zend_function *mptr;
  348. ZEND_HASH_FOREACH_PTR(&ce->function_table, mptr) {
  349. if (mptr->common.fn_flags & ZEND_ACC_STATIC
  350. && ((mptr->common.fn_flags & ZEND_ACC_PRIVATE) == 0 || mptr->common.scope == ce))
  351. {
  352. count_static_funcs++;
  353. }
  354. } ZEND_HASH_FOREACH_END();
  355. }
  356. /* static methods */
  357. smart_str_append_printf(str, "\n%s - Static methods [%d] {", indent, count_static_funcs);
  358. if (count_static_funcs > 0) {
  359. zend_function *mptr;
  360. ZEND_HASH_FOREACH_PTR(&ce->function_table, mptr) {
  361. if (mptr->common.fn_flags & ZEND_ACC_STATIC
  362. && ((mptr->common.fn_flags & ZEND_ACC_PRIVATE) == 0 || mptr->common.scope == ce))
  363. {
  364. smart_str_append_printf(str, "\n");
  365. _function_string(str, mptr, ce, ZSTR_VAL(sub_indent));
  366. }
  367. } ZEND_HASH_FOREACH_END();
  368. } else {
  369. smart_str_append_printf(str, "\n");
  370. }
  371. smart_str_append_printf(str, "%s }\n", indent);
  372. /* Default/Implicit properties */
  373. count = zend_hash_num_elements(&ce->properties_info) - count_static_props - count_shadow_props;
  374. smart_str_append_printf(str, "\n%s - Properties [%d] {\n", indent, count);
  375. if (count > 0) {
  376. zend_property_info *prop;
  377. ZEND_HASH_FOREACH_PTR(&ce->properties_info, prop) {
  378. if (!(prop->flags & (ZEND_ACC_STATIC|ZEND_ACC_SHADOW))) {
  379. _property_string(str, prop, NULL, ZSTR_VAL(sub_indent));
  380. }
  381. } ZEND_HASH_FOREACH_END();
  382. }
  383. smart_str_append_printf(str, "%s }\n", indent);
  384. if (obj && Z_TYPE_P(obj) == IS_OBJECT && Z_OBJ_HT_P(obj)->get_properties) {
  385. HashTable *properties = Z_OBJ_HT_P(obj)->get_properties(obj);
  386. zend_string *prop_name;
  387. smart_str prop_str = {0};
  388. count = 0;
  389. if (properties && zend_hash_num_elements(properties)) {
  390. ZEND_HASH_FOREACH_STR_KEY(properties, prop_name) {
  391. if (prop_name && ZSTR_LEN(prop_name) && ZSTR_VAL(prop_name)[0]) { /* skip all private and protected properties */
  392. if (!zend_hash_exists(&ce->properties_info, prop_name)) {
  393. count++;
  394. _property_string(&prop_str, NULL, ZSTR_VAL(prop_name), ZSTR_VAL(sub_indent));
  395. }
  396. }
  397. } ZEND_HASH_FOREACH_END();
  398. }
  399. smart_str_append_printf(str, "\n%s - Dynamic properties [%d] {\n", indent, count);
  400. smart_str_append_smart_str(str, &prop_str);
  401. smart_str_append_printf(str, "%s }\n", indent);
  402. smart_str_free(&prop_str);
  403. }
  404. /* Non static methods */
  405. count = zend_hash_num_elements(&ce->function_table) - count_static_funcs;
  406. if (count > 0) {
  407. zend_function *mptr;
  408. zend_string *key;
  409. smart_str method_str = {0};
  410. count = 0;
  411. ZEND_HASH_FOREACH_STR_KEY_PTR(&ce->function_table, key, mptr) {
  412. if ((mptr->common.fn_flags & ZEND_ACC_STATIC) == 0
  413. && ((mptr->common.fn_flags & ZEND_ACC_PRIVATE) == 0 || mptr->common.scope == ce))
  414. {
  415. size_t len = ZSTR_LEN(mptr->common.function_name);
  416. /* Do not display old-style inherited constructors */
  417. if ((mptr->common.fn_flags & ZEND_ACC_CTOR) == 0
  418. || mptr->common.scope == ce
  419. || !key
  420. || zend_binary_strcasecmp(ZSTR_VAL(key), ZSTR_LEN(key), ZSTR_VAL(mptr->common.function_name), len) == 0)
  421. {
  422. zend_function *closure;
  423. /* see if this is a closure */
  424. if (ce == zend_ce_closure && obj && (len == sizeof(ZEND_INVOKE_FUNC_NAME)-1)
  425. && memcmp(ZSTR_VAL(mptr->common.function_name), ZEND_INVOKE_FUNC_NAME, sizeof(ZEND_INVOKE_FUNC_NAME)-1) == 0
  426. && (closure = zend_get_closure_invoke_method(Z_OBJ_P(obj))) != NULL)
  427. {
  428. mptr = closure;
  429. } else {
  430. closure = NULL;
  431. }
  432. smart_str_appendc(&method_str, '\n');
  433. _function_string(&method_str, mptr, ce, ZSTR_VAL(sub_indent));
  434. count++;
  435. _free_function(closure);
  436. }
  437. }
  438. } ZEND_HASH_FOREACH_END();
  439. smart_str_append_printf(str, "\n%s - Methods [%d] {", indent, count);
  440. smart_str_append_smart_str(str, &method_str);
  441. if (!count) {
  442. smart_str_append_printf(str, "\n");
  443. }
  444. smart_str_free(&method_str);
  445. } else {
  446. smart_str_append_printf(str, "\n%s - Methods [0] {\n", indent);
  447. }
  448. smart_str_append_printf(str, "%s }\n", indent);
  449. smart_str_append_printf(str, "%s}\n", indent);
  450. zend_string_release_ex(sub_indent, 0);
  451. }
  452. /* }}} */
  453. /* {{{ _const_string */
  454. static void _const_string(smart_str *str, char *name, zval *value, char *indent)
  455. {
  456. char *type = zend_zval_type_name(value);
  457. if (Z_TYPE_P(value) == IS_ARRAY) {
  458. smart_str_append_printf(str, "%s Constant [ %s %s ] { Array }\n",
  459. indent, type, name);
  460. } else if (Z_TYPE_P(value) == IS_STRING) {
  461. smart_str_append_printf(str, "%s Constant [ %s %s ] { %s }\n",
  462. indent, type, name, Z_STRVAL_P(value));
  463. } else {
  464. zend_string *tmp_value_str;
  465. zend_string *value_str = zval_get_tmp_string(value, &tmp_value_str);
  466. smart_str_append_printf(str, "%s Constant [ %s %s ] { %s }\n",
  467. indent, type, name, ZSTR_VAL(value_str));
  468. zend_tmp_string_release(tmp_value_str);
  469. }
  470. }
  471. /* }}} */
  472. /* {{{ _class_const_string */
  473. static void _class_const_string(smart_str *str, char *name, zend_class_constant *c, char *indent)
  474. {
  475. char *visibility = zend_visibility_string(Z_ACCESS_FLAGS(c->value));
  476. char *type;
  477. zval_update_constant_ex(&c->value, c->ce);
  478. type = zend_zval_type_name(&c->value);
  479. if (Z_TYPE(c->value) == IS_ARRAY) {
  480. smart_str_append_printf(str, "%sConstant [ %s %s %s ] { Array }\n",
  481. indent, visibility, type, name);
  482. } else {
  483. zend_string *tmp_value_str;
  484. zend_string *value_str = zval_get_tmp_string(&c->value, &tmp_value_str);
  485. smart_str_append_printf(str, "%sConstant [ %s %s %s ] { %s }\n",
  486. indent, visibility, type, name, ZSTR_VAL(value_str));
  487. zend_tmp_string_release(tmp_value_str);
  488. }
  489. }
  490. /* }}} */
  491. /* {{{ _get_recv_opcode */
  492. static zend_op* _get_recv_op(zend_op_array *op_array, uint32_t offset)
  493. {
  494. zend_op *op = op_array->opcodes;
  495. zend_op *end = op + op_array->last;
  496. ++offset;
  497. while (op < end) {
  498. if ((op->opcode == ZEND_RECV || op->opcode == ZEND_RECV_INIT
  499. || op->opcode == ZEND_RECV_VARIADIC) && op->op1.num == offset)
  500. {
  501. return op;
  502. }
  503. ++op;
  504. }
  505. return NULL;
  506. }
  507. /* }}} */
  508. /* {{{ _parameter_string */
  509. static void _parameter_string(smart_str *str, zend_function *fptr, struct _zend_arg_info *arg_info, uint32_t offset, zend_bool required, char* indent)
  510. {
  511. smart_str_append_printf(str, "Parameter #%d [ ", offset);
  512. if (!required) {
  513. smart_str_append_printf(str, "<optional> ");
  514. } else {
  515. smart_str_append_printf(str, "<required> ");
  516. }
  517. if (ZEND_TYPE_IS_CLASS(arg_info->type)) {
  518. smart_str_append_printf(str, "%s ",
  519. ZSTR_VAL(ZEND_TYPE_NAME(arg_info->type)));
  520. if (ZEND_TYPE_ALLOW_NULL(arg_info->type)) {
  521. smart_str_append_printf(str, "or NULL ");
  522. }
  523. } else if (ZEND_TYPE_IS_CODE(arg_info->type)) {
  524. smart_str_append_printf(str, "%s ", zend_get_type_by_const(ZEND_TYPE_CODE(arg_info->type)));
  525. if (ZEND_TYPE_ALLOW_NULL(arg_info->type)) {
  526. smart_str_append_printf(str, "or NULL ");
  527. }
  528. }
  529. if (arg_info->pass_by_reference) {
  530. smart_str_appendc(str, '&');
  531. }
  532. if (arg_info->is_variadic) {
  533. smart_str_appends(str, "...");
  534. }
  535. if (arg_info->name) {
  536. smart_str_append_printf(str, "$%s",
  537. (fptr->type == ZEND_INTERNAL_FUNCTION &&
  538. !(fptr->common.fn_flags & ZEND_ACC_USER_ARG_INFO)) ?
  539. ((zend_internal_arg_info*)arg_info)->name :
  540. ZSTR_VAL(arg_info->name));
  541. } else {
  542. smart_str_append_printf(str, "$param%d", offset);
  543. }
  544. if (fptr->type == ZEND_USER_FUNCTION && !required) {
  545. zend_op *precv = _get_recv_op((zend_op_array*)fptr, offset);
  546. if (precv && precv->opcode == ZEND_RECV_INIT && precv->op2_type != IS_UNUSED) {
  547. zval zv;
  548. smart_str_appends(str, " = ");
  549. ZVAL_COPY(&zv, RT_CONSTANT(precv, precv->op2));
  550. if (UNEXPECTED(zval_update_constant_ex(&zv, fptr->common.scope) == FAILURE)) {
  551. zval_ptr_dtor(&zv);
  552. return;
  553. }
  554. if (Z_TYPE(zv) == IS_TRUE) {
  555. smart_str_appends(str, "true");
  556. } else if (Z_TYPE(zv) == IS_FALSE) {
  557. smart_str_appends(str, "false");
  558. } else if (Z_TYPE(zv) == IS_NULL) {
  559. smart_str_appends(str, "NULL");
  560. } else if (Z_TYPE(zv) == IS_STRING) {
  561. smart_str_appendc(str, '\'');
  562. smart_str_appendl(str, Z_STRVAL(zv), MIN(Z_STRLEN(zv), 15));
  563. if (Z_STRLEN(zv) > 15) {
  564. smart_str_appends(str, "...");
  565. }
  566. smart_str_appendc(str, '\'');
  567. } else if (Z_TYPE(zv) == IS_ARRAY) {
  568. smart_str_appends(str, "Array");
  569. } else {
  570. zend_string *tmp_zv_str;
  571. zend_string *zv_str = zval_get_tmp_string(&zv, &tmp_zv_str);
  572. smart_str_append(str, zv_str);
  573. zend_tmp_string_release(tmp_zv_str);
  574. }
  575. zval_ptr_dtor(&zv);
  576. }
  577. }
  578. smart_str_appends(str, " ]");
  579. }
  580. /* }}} */
  581. /* {{{ _function_parameter_string */
  582. static void _function_parameter_string(smart_str *str, zend_function *fptr, char* indent)
  583. {
  584. struct _zend_arg_info *arg_info = fptr->common.arg_info;
  585. uint32_t i, num_args, num_required = fptr->common.required_num_args;
  586. if (!arg_info) {
  587. return;
  588. }
  589. num_args = fptr->common.num_args;
  590. if (fptr->common.fn_flags & ZEND_ACC_VARIADIC) {
  591. num_args++;
  592. }
  593. smart_str_appendc(str, '\n');
  594. smart_str_append_printf(str, "%s- Parameters [%d] {\n", indent, num_args);
  595. for (i = 0; i < num_args; i++) {
  596. smart_str_append_printf(str, "%s ", indent);
  597. _parameter_string(str, fptr, arg_info, i, i < num_required, indent);
  598. smart_str_appendc(str, '\n');
  599. arg_info++;
  600. }
  601. smart_str_append_printf(str, "%s}\n", indent);
  602. }
  603. /* }}} */
  604. /* {{{ _function_closure_string */
  605. static void _function_closure_string(smart_str *str, zend_function *fptr, char* indent)
  606. {
  607. uint32_t i, count;
  608. zend_string *key;
  609. HashTable *static_variables;
  610. if (fptr->type != ZEND_USER_FUNCTION || !fptr->op_array.static_variables) {
  611. return;
  612. }
  613. static_variables = fptr->op_array.static_variables;
  614. count = zend_hash_num_elements(static_variables);
  615. if (!count) {
  616. return;
  617. }
  618. smart_str_append_printf(str, "\n");
  619. smart_str_append_printf(str, "%s- Bound Variables [%d] {\n", indent, zend_hash_num_elements(static_variables));
  620. i = 0;
  621. ZEND_HASH_FOREACH_STR_KEY(static_variables, key) {
  622. smart_str_append_printf(str, "%s Variable #%d [ $%s ]\n", indent, i++, ZSTR_VAL(key));
  623. } ZEND_HASH_FOREACH_END();
  624. smart_str_append_printf(str, "%s}\n", indent);
  625. }
  626. /* }}} */
  627. /* {{{ _function_string */
  628. static void _function_string(smart_str *str, zend_function *fptr, zend_class_entry *scope, char* indent)
  629. {
  630. smart_str param_indent = {0};
  631. zend_function *overwrites;
  632. zend_string *lc_name;
  633. /* TBD: Repair indenting of doc comment (or is this to be done in the parser?)
  634. * What's "wrong" is that any whitespace before the doc comment start is
  635. * swallowed, leading to an unaligned comment.
  636. */
  637. if (fptr->type == ZEND_USER_FUNCTION && fptr->op_array.doc_comment) {
  638. smart_str_append_printf(str, "%s%s\n", indent, ZSTR_VAL(fptr->op_array.doc_comment));
  639. }
  640. smart_str_appendl(str, indent, strlen(indent));
  641. smart_str_append_printf(str, fptr->common.fn_flags & ZEND_ACC_CLOSURE ? "Closure [ " : (fptr->common.scope ? "Method [ " : "Function [ "));
  642. smart_str_append_printf(str, (fptr->type == ZEND_USER_FUNCTION) ? "<user" : "<internal");
  643. if (fptr->common.fn_flags & ZEND_ACC_DEPRECATED) {
  644. smart_str_appends(str, ", deprecated");
  645. }
  646. if (fptr->type == ZEND_INTERNAL_FUNCTION && ((zend_internal_function*)fptr)->module) {
  647. smart_str_append_printf(str, ":%s", ((zend_internal_function*)fptr)->module->name);
  648. }
  649. if (scope && fptr->common.scope) {
  650. if (fptr->common.scope != scope) {
  651. smart_str_append_printf(str, ", inherits %s", ZSTR_VAL(fptr->common.scope->name));
  652. } else if (fptr->common.scope->parent) {
  653. lc_name = zend_string_tolower(fptr->common.function_name);
  654. if ((overwrites = zend_hash_find_ptr(&fptr->common.scope->parent->function_table, lc_name)) != NULL) {
  655. if (fptr->common.scope != overwrites->common.scope) {
  656. smart_str_append_printf(str, ", overwrites %s", ZSTR_VAL(overwrites->common.scope->name));
  657. }
  658. }
  659. zend_string_release_ex(lc_name, 0);
  660. }
  661. }
  662. if (fptr->common.prototype && fptr->common.prototype->common.scope) {
  663. smart_str_append_printf(str, ", prototype %s", ZSTR_VAL(fptr->common.prototype->common.scope->name));
  664. }
  665. if (fptr->common.fn_flags & ZEND_ACC_CTOR) {
  666. smart_str_appends(str, ", ctor");
  667. }
  668. if (fptr->common.fn_flags & ZEND_ACC_DTOR) {
  669. smart_str_appends(str, ", dtor");
  670. }
  671. smart_str_appends(str, "> ");
  672. if (fptr->common.fn_flags & ZEND_ACC_ABSTRACT) {
  673. smart_str_appends(str, "abstract ");
  674. }
  675. if (fptr->common.fn_flags & ZEND_ACC_FINAL) {
  676. smart_str_appends(str, "final ");
  677. }
  678. if (fptr->common.fn_flags & ZEND_ACC_STATIC) {
  679. smart_str_appends(str, "static ");
  680. }
  681. if (fptr->common.scope) {
  682. /* These are mutually exclusive */
  683. switch (fptr->common.fn_flags & ZEND_ACC_PPP_MASK) {
  684. case ZEND_ACC_PUBLIC:
  685. smart_str_appends(str, "public ");
  686. break;
  687. case ZEND_ACC_PRIVATE:
  688. smart_str_appends(str, "private ");
  689. break;
  690. case ZEND_ACC_PROTECTED:
  691. smart_str_appends(str, "protected ");
  692. break;
  693. default:
  694. smart_str_appends(str, "<visibility error> ");
  695. break;
  696. }
  697. smart_str_appends(str, "method ");
  698. } else {
  699. smart_str_appends(str, "function ");
  700. }
  701. if (fptr->op_array.fn_flags & ZEND_ACC_RETURN_REFERENCE) {
  702. smart_str_appendc(str, '&');
  703. }
  704. smart_str_append_printf(str, "%s ] {\n", ZSTR_VAL(fptr->common.function_name));
  705. /* The information where a function is declared is only available for user classes */
  706. if (fptr->type == ZEND_USER_FUNCTION) {
  707. smart_str_append_printf(str, "%s @@ %s %d - %d\n", indent,
  708. ZSTR_VAL(fptr->op_array.filename),
  709. fptr->op_array.line_start,
  710. fptr->op_array.line_end);
  711. }
  712. smart_str_append_printf(&param_indent, "%s ", indent);
  713. smart_str_0(&param_indent);
  714. if (fptr->common.fn_flags & ZEND_ACC_CLOSURE) {
  715. _function_closure_string(str, fptr, ZSTR_VAL(param_indent.s));
  716. }
  717. _function_parameter_string(str, fptr, ZSTR_VAL(param_indent.s));
  718. smart_str_free(&param_indent);
  719. if (fptr->op_array.fn_flags & ZEND_ACC_HAS_RETURN_TYPE) {
  720. smart_str_append_printf(str, " %s- Return [ ", indent);
  721. if (ZEND_TYPE_IS_CLASS(fptr->common.arg_info[-1].type)) {
  722. smart_str_append_printf(str, "%s ",
  723. ZSTR_VAL(ZEND_TYPE_NAME(fptr->common.arg_info[-1].type)));
  724. if (ZEND_TYPE_ALLOW_NULL(fptr->common.arg_info[-1].type)) {
  725. smart_str_appends(str, "or NULL ");
  726. }
  727. } else if (ZEND_TYPE_IS_CODE(fptr->common.arg_info[-1].type)) {
  728. smart_str_append_printf(str, "%s ", zend_get_type_by_const(ZEND_TYPE_CODE(fptr->common.arg_info[-1].type)));
  729. if (ZEND_TYPE_ALLOW_NULL(fptr->common.arg_info[-1].type)) {
  730. smart_str_appends(str, "or NULL ");
  731. }
  732. }
  733. smart_str_appends(str, "]\n");
  734. }
  735. smart_str_append_printf(str, "%s}\n", indent);
  736. }
  737. /* }}} */
  738. /* {{{ _property_string */
  739. static void _property_string(smart_str *str, zend_property_info *prop, const char *prop_name, char* indent)
  740. {
  741. smart_str_append_printf(str, "%sProperty [ ", indent);
  742. if (!prop) {
  743. smart_str_append_printf(str, "<dynamic> public $%s", prop_name);
  744. } else {
  745. if (!(prop->flags & ZEND_ACC_STATIC)) {
  746. if (prop->flags & ZEND_ACC_IMPLICIT_PUBLIC) {
  747. smart_str_appends(str, "<implicit> ");
  748. } else {
  749. smart_str_appends(str, "<default> ");
  750. }
  751. }
  752. /* These are mutually exclusive */
  753. switch (prop->flags & ZEND_ACC_PPP_MASK) {
  754. case ZEND_ACC_PUBLIC:
  755. smart_str_appends(str, "public ");
  756. break;
  757. case ZEND_ACC_PRIVATE:
  758. smart_str_appends(str, "private ");
  759. break;
  760. case ZEND_ACC_PROTECTED:
  761. smart_str_appends(str, "protected ");
  762. break;
  763. }
  764. if (prop->flags & ZEND_ACC_STATIC) {
  765. smart_str_appends(str, "static ");
  766. }
  767. if (!prop_name) {
  768. const char *class_name;
  769. zend_unmangle_property_name(prop->name, &class_name, &prop_name);
  770. }
  771. smart_str_append_printf(str, "$%s", prop_name);
  772. }
  773. smart_str_appends(str, " ]\n");
  774. }
  775. /* }}} */
  776. static int _extension_ini_string(zval *el, int num_args, va_list args, zend_hash_key *hash_key) /* {{{ */
  777. {
  778. zend_ini_entry *ini_entry = (zend_ini_entry*)Z_PTR_P(el);
  779. smart_str *str = va_arg(args, smart_str *);
  780. char *indent = va_arg(args, char *);
  781. int number = va_arg(args, int);
  782. char *comma = "";
  783. if (number == ini_entry->module_number) {
  784. smart_str_append_printf(str, " %sEntry [ %s <", indent, ZSTR_VAL(ini_entry->name));
  785. if (ini_entry->modifiable == ZEND_INI_ALL) {
  786. smart_str_appends(str, "ALL");
  787. } else {
  788. if (ini_entry->modifiable & ZEND_INI_USER) {
  789. smart_str_appends(str, "USER");
  790. comma = ",";
  791. }
  792. if (ini_entry->modifiable & ZEND_INI_PERDIR) {
  793. smart_str_append_printf(str, "%sPERDIR", comma);
  794. comma = ",";
  795. }
  796. if (ini_entry->modifiable & ZEND_INI_SYSTEM) {
  797. smart_str_append_printf(str, "%sSYSTEM", comma);
  798. }
  799. }
  800. smart_str_appends(str, "> ]\n");
  801. smart_str_append_printf(str, " %s Current = '%s'\n", indent, ini_entry->value ? ZSTR_VAL(ini_entry->value) : "");
  802. if (ini_entry->modified) {
  803. smart_str_append_printf(str, " %s Default = '%s'\n", indent, ini_entry->orig_value ? ZSTR_VAL(ini_entry->orig_value) : "");
  804. }
  805. smart_str_append_printf(str, " %s}\n", indent);
  806. }
  807. return ZEND_HASH_APPLY_KEEP;
  808. }
  809. /* }}} */
  810. static int _extension_class_string(zval *el, int num_args, va_list args, zend_hash_key *hash_key) /* {{{ */
  811. {
  812. zend_class_entry *ce = (zend_class_entry*)Z_PTR_P(el);
  813. smart_str *str = va_arg(args, smart_str *);
  814. char *indent = va_arg(args, char *);
  815. struct _zend_module_entry *module = va_arg(args, struct _zend_module_entry*);
  816. int *num_classes = va_arg(args, int*);
  817. if ((ce->type == ZEND_INTERNAL_CLASS) && ce->info.internal.module && !strcasecmp(ce->info.internal.module->name, module->name)) {
  818. /* dump class if it is not an alias */
  819. if (!zend_binary_strcasecmp(ZSTR_VAL(ce->name), ZSTR_LEN(ce->name), ZSTR_VAL(hash_key->key), ZSTR_LEN(hash_key->key))) {
  820. smart_str_append_printf(str, "\n");
  821. _class_string(str, ce, NULL, indent);
  822. (*num_classes)++;
  823. }
  824. }
  825. return ZEND_HASH_APPLY_KEEP;
  826. }
  827. /* }}} */
  828. static int _extension_const_string(zval *el, int num_args, va_list args, zend_hash_key *hash_key) /* {{{ */
  829. {
  830. zend_constant *constant = (zend_constant*)Z_PTR_P(el);
  831. smart_str *str = va_arg(args, smart_str *);
  832. char *indent = va_arg(args, char *);
  833. struct _zend_module_entry *module = va_arg(args, struct _zend_module_entry*);
  834. int *num_classes = va_arg(args, int*);
  835. if (ZEND_CONSTANT_MODULE_NUMBER(constant) == module->module_number) {
  836. _const_string(str, ZSTR_VAL(constant->name), &constant->value, indent);
  837. (*num_classes)++;
  838. }
  839. return ZEND_HASH_APPLY_KEEP;
  840. }
  841. /* }}} */
  842. static void _extension_string(smart_str *str, zend_module_entry *module, char *indent) /* {{{ */
  843. {
  844. smart_str_append_printf(str, "%sExtension [ ", indent);
  845. if (module->type == MODULE_PERSISTENT) {
  846. smart_str_appends(str, "<persistent>");
  847. }
  848. if (module->type == MODULE_TEMPORARY) {
  849. smart_str_appends(str, "<temporary>" );
  850. }
  851. smart_str_append_printf(str, " extension #%d %s version %s ] {\n",
  852. module->module_number, module->name,
  853. (module->version == NO_VERSION_YET) ? "<no_version>" : module->version);
  854. if (module->deps) {
  855. const zend_module_dep* dep = module->deps;
  856. smart_str_appends(str, "\n - Dependencies {\n");
  857. while(dep->name) {
  858. smart_str_append_printf(str, "%s Dependency [ %s (", indent, dep->name);
  859. switch(dep->type) {
  860. case MODULE_DEP_REQUIRED:
  861. smart_str_appends(str, "Required");
  862. break;
  863. case MODULE_DEP_CONFLICTS:
  864. smart_str_appends(str, "Conflicts");
  865. break;
  866. case MODULE_DEP_OPTIONAL:
  867. smart_str_appends(str, "Optional");
  868. break;
  869. default:
  870. smart_str_appends(str, "Error"); /* shouldn't happen */
  871. break;
  872. }
  873. if (dep->rel) {
  874. smart_str_append_printf(str, " %s", dep->rel);
  875. }
  876. if (dep->version) {
  877. smart_str_append_printf(str, " %s", dep->version);
  878. }
  879. smart_str_appends(str, ") ]\n");
  880. dep++;
  881. }
  882. smart_str_append_printf(str, "%s }\n", indent);
  883. }
  884. {
  885. smart_str str_ini = {0};
  886. zend_hash_apply_with_arguments(EG(ini_directives), (apply_func_args_t) _extension_ini_string, 3, &str_ini, indent, module->module_number);
  887. if (smart_str_get_len(&str_ini) > 0) {
  888. smart_str_append_printf(str, "\n - INI {\n");
  889. smart_str_append_smart_str(str, &str_ini);
  890. smart_str_append_printf(str, "%s }\n", indent);
  891. }
  892. smart_str_free(&str_ini);
  893. }
  894. {
  895. smart_str str_constants = {0};
  896. int num_constants = 0;
  897. zend_hash_apply_with_arguments(EG(zend_constants), (apply_func_args_t) _extension_const_string, 4, &str_constants, indent, module, &num_constants);
  898. if (num_constants) {
  899. smart_str_append_printf(str, "\n - Constants [%d] {\n", num_constants);
  900. smart_str_append_smart_str(str, &str_constants);
  901. smart_str_append_printf(str, "%s }\n", indent);
  902. }
  903. smart_str_free(&str_constants);
  904. }
  905. {
  906. zend_function *fptr;
  907. int first = 1;
  908. ZEND_HASH_FOREACH_PTR(CG(function_table), fptr) {
  909. if (fptr->common.type==ZEND_INTERNAL_FUNCTION
  910. && fptr->internal_function.module == module) {
  911. if (first) {
  912. smart_str_append_printf(str, "\n - Functions {\n");
  913. first = 0;
  914. }
  915. _function_string(str, fptr, NULL, " ");
  916. }
  917. } ZEND_HASH_FOREACH_END();
  918. if (!first) {
  919. smart_str_append_printf(str, "%s }\n", indent);
  920. }
  921. }
  922. {
  923. zend_string *sub_indent = strpprintf(0, "%s ", indent);
  924. smart_str str_classes = {0};
  925. int num_classes = 0;
  926. zend_hash_apply_with_arguments(EG(class_table), (apply_func_args_t) _extension_class_string, 4, &str_classes, ZSTR_VAL(sub_indent), module, &num_classes);
  927. if (num_classes) {
  928. smart_str_append_printf(str, "\n - Classes [%d] {", num_classes);
  929. smart_str_append_smart_str(str, &str_classes);
  930. smart_str_append_printf(str, "%s }\n", indent);
  931. }
  932. smart_str_free(&str_classes);
  933. zend_string_release_ex(sub_indent, 0);
  934. }
  935. smart_str_append_printf(str, "%s}\n", indent);
  936. }
  937. /* }}} */
  938. static void _zend_extension_string(smart_str *str, zend_extension *extension, char *indent) /* {{{ */
  939. {
  940. smart_str_append_printf(str, "%sZend Extension [ %s ", indent, extension->name);
  941. if (extension->version) {
  942. smart_str_append_printf(str, "%s ", extension->version);
  943. }
  944. if (extension->copyright) {
  945. smart_str_append_printf(str, "%s ", extension->copyright);
  946. }
  947. if (extension->author) {
  948. smart_str_append_printf(str, "by %s ", extension->author);
  949. }
  950. if (extension->URL) {
  951. smart_str_append_printf(str, "<%s> ", extension->URL);
  952. }
  953. smart_str_appends(str, "]\n");
  954. }
  955. /* }}} */
  956. /* {{{ _function_check_flag */
  957. static void _function_check_flag(INTERNAL_FUNCTION_PARAMETERS, int mask)
  958. {
  959. reflection_object *intern;
  960. zend_function *mptr;
  961. if (zend_parse_parameters_none() == FAILURE) {
  962. return;
  963. }
  964. GET_REFLECTION_OBJECT_PTR(mptr);
  965. RETURN_BOOL(mptr->common.fn_flags & mask);
  966. }
  967. /* }}} */
  968. /* {{{ zend_reflection_class_factory */
  969. PHPAPI void zend_reflection_class_factory(zend_class_entry *ce, zval *object)
  970. {
  971. reflection_object *intern;
  972. zval name;
  973. ZVAL_STR_COPY(&name, ce->name);
  974. reflection_instantiate(reflection_class_ptr, object);
  975. intern = Z_REFLECTION_P(object);
  976. intern->ptr = ce;
  977. intern->ref_type = REF_TYPE_OTHER;
  978. intern->ce = ce;
  979. reflection_update_property_name(object, &name);
  980. }
  981. /* }}} */
  982. /* {{{ reflection_extension_factory */
  983. static void reflection_extension_factory(zval *object, const char *name_str)
  984. {
  985. reflection_object *intern;
  986. zval name;
  987. size_t name_len = strlen(name_str);
  988. zend_string *lcname;
  989. struct _zend_module_entry *module;
  990. lcname = zend_string_alloc(name_len, 0);
  991. zend_str_tolower_copy(ZSTR_VAL(lcname), name_str, name_len);
  992. module = zend_hash_find_ptr(&module_registry, lcname);
  993. zend_string_efree(lcname);
  994. if (!module) {
  995. return;
  996. }
  997. reflection_instantiate(reflection_extension_ptr, object);
  998. intern = Z_REFLECTION_P(object);
  999. ZVAL_STRINGL(&name, module->name, name_len);
  1000. intern->ptr = module;
  1001. intern->ref_type = REF_TYPE_OTHER;
  1002. intern->ce = NULL;
  1003. reflection_update_property_name(object, &name);
  1004. }
  1005. /* }}} */
  1006. /* {{{ reflection_parameter_factory */
  1007. static void reflection_parameter_factory(zend_function *fptr, zval *closure_object, struct _zend_arg_info *arg_info, uint32_t offset, zend_bool required, zval *object)
  1008. {
  1009. reflection_object *intern;
  1010. parameter_reference *reference;
  1011. zval name;
  1012. if (arg_info->name) {
  1013. if (fptr->type == ZEND_INTERNAL_FUNCTION &&
  1014. !(fptr->common.fn_flags & ZEND_ACC_USER_ARG_INFO)) {
  1015. ZVAL_STRING(&name, ((zend_internal_arg_info*)arg_info)->name);
  1016. } else {
  1017. ZVAL_STR_COPY(&name, arg_info->name);
  1018. }
  1019. } else {
  1020. ZVAL_NULL(&name);
  1021. }
  1022. reflection_instantiate(reflection_parameter_ptr, object);
  1023. intern = Z_REFLECTION_P(object);
  1024. reference = (parameter_reference*) emalloc(sizeof(parameter_reference));
  1025. reference->arg_info = arg_info;
  1026. reference->offset = offset;
  1027. reference->required = required;
  1028. reference->fptr = fptr;
  1029. intern->ptr = reference;
  1030. intern->ref_type = REF_TYPE_PARAMETER;
  1031. intern->ce = fptr->common.scope;
  1032. if (closure_object) {
  1033. Z_ADDREF_P(closure_object);
  1034. ZVAL_COPY_VALUE(&intern->obj, closure_object);
  1035. }
  1036. reflection_update_property_name(object, &name);
  1037. }
  1038. /* }}} */
  1039. /* {{{ reflection_type_factory */
  1040. static void reflection_type_factory(zend_function *fptr, zval *closure_object, struct _zend_arg_info *arg_info, zval *object)
  1041. {
  1042. reflection_object *intern;
  1043. type_reference *reference;
  1044. reflection_instantiate(reflection_named_type_ptr, object);
  1045. intern = Z_REFLECTION_P(object);
  1046. reference = (type_reference*) emalloc(sizeof(type_reference));
  1047. reference->arg_info = arg_info;
  1048. reference->fptr = fptr;
  1049. intern->ptr = reference;
  1050. intern->ref_type = REF_TYPE_TYPE;
  1051. intern->ce = fptr->common.scope;
  1052. if (closure_object) {
  1053. Z_ADDREF_P(closure_object);
  1054. ZVAL_COPY_VALUE(&intern->obj, closure_object);
  1055. }
  1056. }
  1057. /* }}} */
  1058. /* {{{ reflection_function_factory */
  1059. static void reflection_function_factory(zend_function *function, zval *closure_object, zval *object)
  1060. {
  1061. reflection_object *intern;
  1062. zval name;
  1063. ZVAL_STR_COPY(&name, function->common.function_name);
  1064. reflection_instantiate(reflection_function_ptr, object);
  1065. intern = Z_REFLECTION_P(object);
  1066. intern->ptr = function;
  1067. intern->ref_type = REF_TYPE_FUNCTION;
  1068. intern->ce = NULL;
  1069. if (closure_object) {
  1070. Z_ADDREF_P(closure_object);
  1071. ZVAL_COPY_VALUE(&intern->obj, closure_object);
  1072. }
  1073. reflection_update_property_name(object, &name);
  1074. }
  1075. /* }}} */
  1076. /* {{{ reflection_method_factory */
  1077. static void reflection_method_factory(zend_class_entry *ce, zend_function *method, zval *closure_object, zval *object)
  1078. {
  1079. reflection_object *intern;
  1080. zval name;
  1081. zval classname;
  1082. ZVAL_STR_COPY(&name, (method->common.scope && method->common.scope->trait_aliases)?
  1083. zend_resolve_method_name(ce, method) : method->common.function_name);
  1084. ZVAL_STR_COPY(&classname, method->common.scope->name);
  1085. reflection_instantiate(reflection_method_ptr, object);
  1086. intern = Z_REFLECTION_P(object);
  1087. intern->ptr = method;
  1088. intern->ref_type = REF_TYPE_FUNCTION;
  1089. intern->ce = ce;
  1090. if (closure_object) {
  1091. Z_ADDREF_P(closure_object);
  1092. ZVAL_COPY_VALUE(&intern->obj, closure_object);
  1093. }
  1094. reflection_update_property_name(object, &name);
  1095. reflection_update_property_class(object, &classname);
  1096. }
  1097. /* }}} */
  1098. /* {{{ reflection_property_factory */
  1099. static void reflection_property_factory(zend_class_entry *ce, zend_string *name, zend_property_info *prop, zval *object)
  1100. {
  1101. reflection_object *intern;
  1102. zval propname;
  1103. zval classname;
  1104. property_reference *reference;
  1105. if (!(prop->flags & ZEND_ACC_PRIVATE)) {
  1106. /* we have to search the class hierarchy for this (implicit) public or protected property */
  1107. zend_class_entry *tmp_ce = ce, *store_ce = ce;
  1108. zend_property_info *tmp_info = NULL;
  1109. while (tmp_ce && (tmp_info = zend_hash_find_ptr(&tmp_ce->properties_info, name)) == NULL) {
  1110. ce = tmp_ce;
  1111. tmp_ce = tmp_ce->parent;
  1112. }
  1113. if (tmp_info && !(tmp_info->flags & ZEND_ACC_SHADOW)) { /* found something and it's not a parent's private */
  1114. prop = tmp_info;
  1115. } else { /* not found, use initial value */
  1116. ce = store_ce;
  1117. }
  1118. }
  1119. ZVAL_STR_COPY(&propname, name);
  1120. ZVAL_STR_COPY(&classname, prop->ce->name);
  1121. reflection_instantiate(reflection_property_ptr, object);
  1122. intern = Z_REFLECTION_P(object);
  1123. reference = (property_reference*) emalloc(sizeof(property_reference));
  1124. reference->ce = ce;
  1125. reference->prop = *prop;
  1126. reference->unmangled_name = zend_string_copy(name);
  1127. intern->ptr = reference;
  1128. intern->ref_type = REF_TYPE_PROPERTY;
  1129. intern->ce = ce;
  1130. intern->ignore_visibility = 0;
  1131. reflection_update_property_name(object, &propname);
  1132. reflection_update_property_class(object, &classname);
  1133. }
  1134. /* }}} */
  1135. static void reflection_property_factory_str(zend_class_entry *ce, const char *name_str, size_t name_len, zend_property_info *prop, zval *object)
  1136. {
  1137. zend_string *name = zend_string_init(name_str, name_len, 0);
  1138. reflection_property_factory(ce, name, prop, object);
  1139. zend_string_release(name);
  1140. }
  1141. /* {{{ reflection_class_constant_factory */
  1142. static void reflection_class_constant_factory(zend_class_entry *ce, zend_string *name_str, zend_class_constant *constant, zval *object)
  1143. {
  1144. reflection_object *intern;
  1145. zval name;
  1146. zval classname;
  1147. ZVAL_STR_COPY(&name, name_str);
  1148. ZVAL_STR_COPY(&classname, ce->name);
  1149. reflection_instantiate(reflection_class_constant_ptr, object);
  1150. intern = Z_REFLECTION_P(object);
  1151. intern->ptr = constant;
  1152. intern->ref_type = REF_TYPE_CLASS_CONSTANT;
  1153. intern->ce = constant->ce;
  1154. intern->ignore_visibility = 0;
  1155. reflection_update_property_name(object, &name);
  1156. reflection_update_property_class(object, &classname);
  1157. }
  1158. /* }}} */
  1159. /* {{{ _reflection_export */
  1160. static void _reflection_export(INTERNAL_FUNCTION_PARAMETERS, zend_class_entry *ce_ptr, int ctor_argc)
  1161. {
  1162. zval reflector;
  1163. zval *argument_ptr, *argument2_ptr;
  1164. zval retval, params[2];
  1165. int result;
  1166. int return_output = 0;
  1167. zend_fcall_info fci;
  1168. zend_fcall_info_cache fcc;
  1169. if (ctor_argc == 1) {
  1170. if (zend_parse_parameters(ZEND_NUM_ARGS(), "z|b", &argument_ptr, &return_output) == FAILURE) {
  1171. return;
  1172. }
  1173. ZVAL_COPY_VALUE(&params[0], argument_ptr);
  1174. ZVAL_NULL(&params[1]);
  1175. } else {
  1176. if (zend_parse_parameters(ZEND_NUM_ARGS(), "zz|b", &argument_ptr, &argument2_ptr, &return_output) == FAILURE) {
  1177. return;
  1178. }
  1179. ZVAL_COPY_VALUE(&params[0], argument_ptr);
  1180. ZVAL_COPY_VALUE(&params[1], argument2_ptr);
  1181. }
  1182. /* Create object */
  1183. if (object_and_properties_init(&reflector, ce_ptr, NULL) == FAILURE) {
  1184. _DO_THROW("Could not create reflector");
  1185. }
  1186. /* Call __construct() */
  1187. fci.size = sizeof(fci);
  1188. ZVAL_UNDEF(&fci.function_name);
  1189. fci.object = Z_OBJ(reflector);
  1190. fci.retval = &retval;
  1191. fci.param_count = ctor_argc;
  1192. fci.params = params;
  1193. fci.no_separation = 1;
  1194. fcc.function_handler = ce_ptr->constructor;
  1195. fcc.called_scope = Z_OBJCE(reflector);
  1196. fcc.object = Z_OBJ(reflector);
  1197. result = zend_call_function(&fci, &fcc);
  1198. zval_ptr_dtor(&retval);
  1199. if (EG(exception)) {
  1200. zval_ptr_dtor(&reflector);
  1201. return;
  1202. }
  1203. if (result == FAILURE) {
  1204. zval_ptr_dtor(&reflector);
  1205. _DO_THROW("Could not create reflector");
  1206. }
  1207. /* Call static reflection::export */
  1208. ZVAL_COPY_VALUE(&params[0], &reflector);
  1209. ZVAL_BOOL(&params[1], return_output);
  1210. ZVAL_STRINGL(&fci.function_name, "reflection::export", sizeof("reflection::export") - 1);
  1211. fci.object = NULL;
  1212. fci.retval = &retval;
  1213. fci.param_count = 2;
  1214. fci.params = params;
  1215. fci.no_separation = 1;
  1216. result = zend_call_function(&fci, NULL);
  1217. zval_ptr_dtor(&fci.function_name);
  1218. if (result == FAILURE && EG(exception) == NULL) {
  1219. zval_ptr_dtor(&reflector);
  1220. zval_ptr_dtor(&retval);
  1221. _DO_THROW("Could not execute reflection::export()");
  1222. }
  1223. if (return_output) {
  1224. ZVAL_COPY_VALUE(return_value, &retval);
  1225. } else {
  1226. zval_ptr_dtor(&retval);
  1227. }
  1228. /* Destruct reflector which is no longer needed */
  1229. zval_ptr_dtor(&reflector);
  1230. }
  1231. /* }}} */
  1232. /* {{{ _reflection_param_get_default_param */
  1233. static parameter_reference *_reflection_param_get_default_param(INTERNAL_FUNCTION_PARAMETERS)
  1234. {
  1235. reflection_object *intern;
  1236. parameter_reference *param;
  1237. intern = Z_REFLECTION_P(getThis());
  1238. if (intern->ptr == NULL) {
  1239. if (EG(exception) && EG(exception)->ce == reflection_exception_ptr) {
  1240. return NULL;
  1241. }
  1242. zend_throw_error(NULL, "Internal error: Failed to retrieve the reflection object");
  1243. return NULL;
  1244. }
  1245. param = intern->ptr;
  1246. if (param->fptr->type != ZEND_USER_FUNCTION) {
  1247. zend_throw_exception_ex(reflection_exception_ptr, 0, "Cannot determine default value for internal functions");
  1248. return NULL;
  1249. }
  1250. return param;
  1251. }
  1252. /* }}} */
  1253. /* {{{ _reflection_param_get_default_precv */
  1254. static zend_op *_reflection_param_get_default_precv(INTERNAL_FUNCTION_PARAMETERS, parameter_reference *param)
  1255. {
  1256. zend_op *precv;
  1257. if (param == NULL) {
  1258. return NULL;
  1259. }
  1260. precv = _get_recv_op((zend_op_array*)param->fptr, param->offset);
  1261. if (!precv || precv->opcode != ZEND_RECV_INIT || precv->op2_type == IS_UNUSED) {
  1262. zend_throw_exception_ex(reflection_exception_ptr, 0, "Internal error: Failed to retrieve the default value");
  1263. return NULL;
  1264. }
  1265. return precv;
  1266. }
  1267. /* }}} */
  1268. /* {{{ Preventing __clone from being called */
  1269. ZEND_METHOD(reflection, __clone)
  1270. {
  1271. /* Should never be executable */
  1272. _DO_THROW("Cannot clone object using __clone()");
  1273. }
  1274. /* }}} */
  1275. /* {{{ proto public static mixed Reflection::export(Reflector r [, bool return])
  1276. Exports a reflection object. Returns the output if TRUE is specified for return, printing it otherwise. */
  1277. ZEND_METHOD(reflection, export)
  1278. {
  1279. zval *object, fname, retval;
  1280. int result;
  1281. zend_bool return_output = 0;
  1282. ZEND_PARSE_PARAMETERS_START(1, 2)
  1283. Z_PARAM_OBJECT_OF_CLASS(object, reflector_ptr)
  1284. Z_PARAM_OPTIONAL
  1285. Z_PARAM_BOOL(return_output)
  1286. ZEND_PARSE_PARAMETERS_END();
  1287. /* Invoke the __toString() method */
  1288. ZVAL_STRINGL(&fname, "__tostring", sizeof("__tostring") - 1);
  1289. result= call_user_function(NULL, object, &fname, &retval, 0, NULL);
  1290. zval_ptr_dtor_str(&fname);
  1291. if (result == FAILURE) {
  1292. _DO_THROW("Invocation of method __toString() failed");
  1293. /* Returns from this function */
  1294. }
  1295. if (Z_TYPE(retval) == IS_UNDEF) {
  1296. php_error_docref(NULL, E_WARNING, "%s::__toString() did not return anything", ZSTR_VAL(Z_OBJCE_P(object)->name));
  1297. RETURN_FALSE;
  1298. }
  1299. if (return_output) {
  1300. ZVAL_COPY_VALUE(return_value, &retval);
  1301. } else {
  1302. /* No need for _r variant, return of __toString should always be a string */
  1303. zend_print_zval(&retval, 0);
  1304. zend_printf("\n");
  1305. zval_ptr_dtor(&retval);
  1306. }
  1307. }
  1308. /* }}} */
  1309. /* {{{ proto public static array Reflection::getModifierNames(int modifiers)
  1310. Returns an array of modifier names */
  1311. ZEND_METHOD(reflection, getModifierNames)
  1312. {
  1313. zend_long modifiers;
  1314. if (zend_parse_parameters(ZEND_NUM_ARGS(), "l", &modifiers) == FAILURE) {
  1315. return;
  1316. }
  1317. array_init(return_value);
  1318. if (modifiers & (ZEND_ACC_ABSTRACT | ZEND_ACC_EXPLICIT_ABSTRACT_CLASS)) {
  1319. add_next_index_stringl(return_value, "abstract", sizeof("abstract")-1);
  1320. }
  1321. if (modifiers & ZEND_ACC_FINAL) {
  1322. add_next_index_stringl(return_value, "final", sizeof("final")-1);
  1323. }
  1324. if (modifiers & ZEND_ACC_IMPLICIT_PUBLIC) {
  1325. add_next_index_stringl(return_value, "public", sizeof("public")-1);
  1326. }
  1327. /* These are mutually exclusive */
  1328. switch (modifiers & ZEND_ACC_PPP_MASK) {
  1329. case ZEND_ACC_PUBLIC:
  1330. add_next_index_stringl(return_value, "public", sizeof("public")-1);
  1331. break;
  1332. case ZEND_ACC_PRIVATE:
  1333. add_next_index_stringl(return_value, "private", sizeof("private")-1);
  1334. break;
  1335. case ZEND_ACC_PROTECTED:
  1336. add_next_index_stringl(return_value, "protected", sizeof("protected")-1);
  1337. break;
  1338. }
  1339. if (modifiers & ZEND_ACC_STATIC) {
  1340. add_next_index_stringl(return_value, "static", sizeof("static")-1);
  1341. }
  1342. }
  1343. /* }}} */
  1344. /* {{{ proto public static mixed ReflectionFunction::export(string name [, bool return])
  1345. Exports a reflection object. Returns the output if TRUE is specified for return, printing it otherwise. */
  1346. ZEND_METHOD(reflection_function, export)
  1347. {
  1348. _reflection_export(INTERNAL_FUNCTION_PARAM_PASSTHRU, reflection_function_ptr, 1);
  1349. }
  1350. /* }}} */
  1351. /* {{{ proto public void ReflectionFunction::__construct(string name)
  1352. Constructor. Throws an Exception in case the given function does not exist */
  1353. ZEND_METHOD(reflection_function, __construct)
  1354. {
  1355. zval name;
  1356. zval *object;
  1357. zval *closure = NULL;
  1358. reflection_object *intern;
  1359. zend_function *fptr;
  1360. zend_string *fname, *lcname;
  1361. object = getThis();
  1362. intern = Z_REFLECTION_P(object);
  1363. if (zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET, ZEND_NUM_ARGS(), "O", &closure, zend_ce_closure) == SUCCESS) {
  1364. fptr = (zend_function*)zend_get_closure_method_def(closure);
  1365. Z_ADDREF_P(closure);
  1366. } else {
  1367. ALLOCA_FLAG(use_heap)
  1368. if (zend_parse_parameters_throw(ZEND_NUM_ARGS(), "S", &fname) == FAILURE) {
  1369. return;
  1370. }
  1371. if (UNEXPECTED(ZSTR_VAL(fname)[0] == '\\')) {
  1372. /* Ignore leading "\" */
  1373. ZSTR_ALLOCA_ALLOC(lcname, ZSTR_LEN(fname) - 1, use_heap);
  1374. zend_str_tolower_copy(ZSTR_VAL(lcname), ZSTR_VAL(fname) + 1, ZSTR_LEN(fname) - 1);
  1375. fptr = zend_fetch_function(lcname);
  1376. ZSTR_ALLOCA_FREE(lcname, use_heap);
  1377. } else {
  1378. lcname = zend_string_tolower(fname);
  1379. fptr = zend_fetch_function(lcname);
  1380. zend_string_release(lcname);
  1381. }
  1382. if (fptr == NULL) {
  1383. zend_throw_exception_ex(reflection_exception_ptr, 0,
  1384. "Function %s() does not exist", ZSTR_VAL(fname));
  1385. return;
  1386. }
  1387. }
  1388. ZVAL_STR_COPY(&name, fptr->common.function_name);
  1389. reflection_update_property_name(object, &name);
  1390. intern->ptr = fptr;
  1391. intern->ref_type = REF_TYPE_FUNCTION;
  1392. if (closure) {
  1393. ZVAL_COPY_VALUE(&intern->obj, closure);
  1394. } else {
  1395. ZVAL_UNDEF(&intern->obj);
  1396. }
  1397. intern->ce = NULL;
  1398. }
  1399. /* }}} */
  1400. /* {{{ proto public string ReflectionFunction::__toString()
  1401. Returns a string representation */
  1402. ZEND_METHOD(reflection_function, __toString)
  1403. {
  1404. reflection_object *intern;
  1405. zend_function *fptr;
  1406. smart_str str = {0};
  1407. if (zend_parse_parameters_none() == FAILURE) {
  1408. return;
  1409. }
  1410. GET_REFLECTION_OBJECT_PTR(fptr);
  1411. _function_string(&str, fptr, intern->ce, "");
  1412. RETURN_STR(smart_str_extract(&str));
  1413. }
  1414. /* }}} */
  1415. /* {{{ proto public string ReflectionFunction::getName()
  1416. Returns this function's name */
  1417. ZEND_METHOD(reflection_function, getName)
  1418. {
  1419. if (zend_parse_parameters_none() == FAILURE) {
  1420. return;
  1421. }
  1422. _default_get_name(getThis(), return_value);
  1423. }
  1424. /* }}} */
  1425. /* {{{ proto public bool ReflectionFunction::isClosure()
  1426. Returns whether this is a closure */
  1427. ZEND_METHOD(reflection_function, isClosure)
  1428. {
  1429. reflection_object *intern;
  1430. zend_function *fptr;
  1431. if (zend_parse_parameters_none() == FAILURE) {
  1432. return;
  1433. }
  1434. GET_REFLECTION_OBJECT_PTR(fptr);
  1435. RETURN_BOOL(fptr->common.fn_flags & ZEND_ACC_CLOSURE);
  1436. }
  1437. /* }}} */
  1438. /* {{{ proto public bool ReflectionFunction::getClosureThis()
  1439. Returns this pointer bound to closure */
  1440. ZEND_METHOD(reflection_function, getClosureThis)
  1441. {
  1442. reflection_object *intern;
  1443. zval* closure_this;
  1444. if (zend_parse_parameters_none() == FAILURE) {
  1445. return;
  1446. }
  1447. GET_REFLECTION_OBJECT();
  1448. if (!Z_ISUNDEF(intern->obj)) {
  1449. closure_this = zend_get_closure_this_ptr(&intern->obj);
  1450. if (!Z_ISUNDEF_P(closure_this)) {
  1451. ZVAL_COPY(return_value, closure_this);
  1452. }
  1453. }
  1454. }
  1455. /* }}} */
  1456. /* {{{ proto public ReflectionClass ReflectionFunction::getClosureScopeClass()
  1457. Returns the scope associated to the closure */
  1458. ZEND_METHOD(reflection_function, getClosureScopeClass)
  1459. {
  1460. reflection_object *intern;
  1461. const zend_function *closure_func;
  1462. if (zend_parse_parameters_none() == FAILURE) {
  1463. return;
  1464. }
  1465. GET_REFLECTION_OBJECT();
  1466. if (!Z_ISUNDEF(intern->obj)) {
  1467. closure_func = zend_get_closure_method_def(&intern->obj);
  1468. if (closure_func && closure_func->common.scope) {
  1469. zend_reflection_class_factory(closure_func->common.scope, return_value);
  1470. }
  1471. }
  1472. }
  1473. /* }}} */
  1474. /* {{{ proto public mixed ReflectionFunction::getClosure()
  1475. Returns a dynamically created closure for the function */
  1476. ZEND_METHOD(reflection_function, getClosure)
  1477. {
  1478. reflection_object *intern;
  1479. zend_function *fptr;
  1480. if (zend_parse_parameters_none() == FAILURE) {
  1481. return;
  1482. }
  1483. GET_REFLECTION_OBJECT_PTR(fptr);
  1484. if (!Z_ISUNDEF(intern->obj)) {
  1485. /* Closures are immutable objects */
  1486. ZVAL_COPY(return_value, &intern->obj);
  1487. } else {
  1488. zend_create_fake_closure(return_value, fptr, NULL, NULL, NULL);
  1489. }
  1490. }
  1491. /* }}} */
  1492. /* {{{ proto public bool ReflectionFunction::isInternal()
  1493. Returns whether this is an internal function */
  1494. ZEND_METHOD(reflection_function, isInternal)
  1495. {
  1496. reflection_object *intern;
  1497. zend_function *fptr;
  1498. if (zend_parse_parameters_none() == FAILURE) {
  1499. return;
  1500. }
  1501. GET_REFLECTION_OBJECT_PTR(fptr);
  1502. RETURN_BOOL(fptr->type == ZEND_INTERNAL_FUNCTION);
  1503. }
  1504. /* }}} */
  1505. /* {{{ proto public bool ReflectionFunction::isUserDefined()
  1506. Returns whether this is a user-defined function */
  1507. ZEND_METHOD(reflection_function, isUserDefined)
  1508. {
  1509. reflection_object *intern;
  1510. zend_function *fptr;
  1511. if (zend_parse_parameters_none() == FAILURE) {
  1512. return;
  1513. }
  1514. GET_REFLECTION_OBJECT_PTR(fptr);
  1515. RETURN_BOOL(fptr->type == ZEND_USER_FUNCTION);
  1516. }
  1517. /* }}} */
  1518. /* {{{ proto public bool ReflectionFunction::isDisabled()
  1519. Returns whether this function has been disabled or not */
  1520. ZEND_METHOD(reflection_function, isDisabled)
  1521. {
  1522. reflection_object *intern;
  1523. zend_function *fptr;
  1524. GET_REFLECTION_OBJECT_PTR(fptr);
  1525. RETURN_BOOL(fptr->type == ZEND_INTERNAL_FUNCTION && fptr->internal_function.handler == zif_display_disabled_function);
  1526. }
  1527. /* }}} */
  1528. /* {{{ proto public string ReflectionFunction::getFileName()
  1529. Returns the filename of the file this function was declared in */
  1530. ZEND_METHOD(reflection_function, getFileName)
  1531. {
  1532. reflection_object *intern;
  1533. zend_function *fptr;
  1534. if (zend_parse_parameters_none() == FAILURE) {
  1535. return;
  1536. }
  1537. GET_REFLECTION_OBJECT_PTR(fptr);
  1538. if (fptr->type == ZEND_USER_FUNCTION) {
  1539. RETURN_STR_COPY(fptr->op_array.filename);
  1540. }
  1541. RETURN_FALSE;
  1542. }
  1543. /* }}} */
  1544. /* {{{ proto public int ReflectionFunction::getStartLine()
  1545. Returns the line this function's declaration starts at */
  1546. ZEND_METHOD(reflection_function, getStartLine)
  1547. {
  1548. reflection_object *intern;
  1549. zend_function *fptr;
  1550. if (zend_parse_parameters_none() == FAILURE) {
  1551. return;
  1552. }
  1553. GET_REFLECTION_OBJECT_PTR(fptr);
  1554. if (fptr->type == ZEND_USER_FUNCTION) {
  1555. RETURN_LONG(fptr->op_array.line_start);
  1556. }
  1557. RETURN_FALSE;
  1558. }
  1559. /* }}} */
  1560. /* {{{ proto public int ReflectionFunction::getEndLine()
  1561. Returns the line this function's declaration ends at */
  1562. ZEND_METHOD(reflection_function, getEndLine)
  1563. {
  1564. reflection_object *intern;
  1565. zend_function *fptr;
  1566. if (zend_parse_parameters_none() == FAILURE) {
  1567. return;
  1568. }
  1569. GET_REFLECTION_OBJECT_PTR(fptr);
  1570. if (fptr->type == ZEND_USER_FUNCTION) {
  1571. RETURN_LONG(fptr->op_array.line_end);
  1572. }
  1573. RETURN_FALSE;
  1574. }
  1575. /* }}} */
  1576. /* {{{ proto public string ReflectionFunction::getDocComment()
  1577. Returns the doc comment for this function */
  1578. ZEND_METHOD(reflection_function, getDocComment)
  1579. {
  1580. reflection_object *intern;
  1581. zend_function *fptr;
  1582. if (zend_parse_parameters_none() == FAILURE) {
  1583. return;
  1584. }
  1585. GET_REFLECTION_OBJECT_PTR(fptr);
  1586. if (fptr->type == ZEND_USER_FUNCTION && fptr->op_array.doc_comment) {
  1587. RETURN_STR_COPY(fptr->op_array.doc_comment);
  1588. }
  1589. RETURN_FALSE;
  1590. }
  1591. /* }}} */
  1592. /* {{{ proto public array ReflectionFunction::getStaticVariables()
  1593. Returns an associative array containing this function's static variables and their values */
  1594. ZEND_METHOD(reflection_function, getStaticVariables)
  1595. {
  1596. reflection_object *intern;
  1597. zend_function *fptr;
  1598. zval *val;
  1599. if (zend_parse_parameters_none() == FAILURE) {
  1600. return;
  1601. }
  1602. GET_REFLECTION_OBJECT_PTR(fptr);
  1603. /* Return an empty array in case no static variables exist */
  1604. if (fptr->type == ZEND_USER_FUNCTION && fptr->op_array.static_variables != NULL) {
  1605. array_init(return_value);
  1606. if (GC_REFCOUNT(fptr->op_array.static_variables) > 1) {
  1607. if (!(GC_FLAGS(fptr->op_array.static_variables) & IS_ARRAY_IMMUTABLE)) {
  1608. GC_DELREF(fptr->op_array.static_variables);
  1609. }
  1610. fptr->op_array.static_variables = zend_array_dup(fptr->op_array.static_variables);
  1611. }
  1612. ZEND_HASH_FOREACH_VAL(fptr->op_array.static_variables, val) {
  1613. if (UNEXPECTED(zval_update_constant_ex(val, fptr->common.scope) != SUCCESS)) {
  1614. return;
  1615. }
  1616. } ZEND_HASH_FOREACH_END();
  1617. zend_hash_copy(Z_ARRVAL_P(return_value), fptr->op_array.static_variables, zval_add_ref);
  1618. } else {
  1619. ZVAL_EMPTY_ARRAY(return_value);
  1620. }
  1621. }
  1622. /* }}} */
  1623. /* {{{ proto public mixed ReflectionFunction::invoke([mixed* args])
  1624. Invokes the function */
  1625. ZEND_METHOD(reflection_function, invoke)
  1626. {
  1627. zval retval;
  1628. zval *params = NULL;
  1629. int result, num_args = 0;
  1630. zend_fcall_info fci;
  1631. zend_fcall_info_cache fcc;
  1632. reflection_object *intern;
  1633. zend_function *fptr;
  1634. GET_REFLECTION_OBJECT_PTR(fptr);
  1635. if (zend_parse_parameters(ZEND_NUM_ARGS(), "*", &params, &num_args) == FAILURE) {
  1636. return;
  1637. }
  1638. fci.size = sizeof(fci);
  1639. ZVAL_UNDEF(&fci.function_name);
  1640. fci.object = NULL;
  1641. fci.retval = &retval;
  1642. fci.param_count = num_args;
  1643. fci.params = params;
  1644. fci.no_separation = 1;
  1645. fcc.function_handler = fptr;
  1646. fcc.called_scope = NULL;
  1647. fcc.object = NULL;
  1648. if (!Z_ISUNDEF(intern->obj)) {
  1649. Z_OBJ_HT(intern->obj)->get_closure(
  1650. &intern->obj, &fcc.called_scope, &fcc.function_handler, &fcc.object);
  1651. }
  1652. result = zend_call_function(&fci, &fcc);
  1653. if (result == FAILURE) {
  1654. zend_throw_exception_ex(reflection_exception_ptr, 0,
  1655. "Invocation of function %s() failed", ZSTR_VAL(fptr->common.function_name));
  1656. return;
  1657. }
  1658. if (Z_TYPE(retval) != IS_UNDEF) {
  1659. if (Z_ISREF(retval)) {
  1660. zend_unwrap_reference(&retval);
  1661. }
  1662. ZVAL_COPY_VALUE(return_value, &retval);
  1663. }
  1664. }
  1665. /* }}} */
  1666. /* {{{ proto public mixed ReflectionFunction::invokeArgs(array args)
  1667. Invokes the function and pass its arguments as array. */
  1668. ZEND_METHOD(reflection_function, invokeArgs)
  1669. {
  1670. zval retval;
  1671. zval *params, *val;
  1672. int result;
  1673. int i, argc;
  1674. zend_fcall_info fci;
  1675. zend_fcall_info_cache fcc;
  1676. reflection_object *intern;
  1677. zend_function *fptr;
  1678. zval *param_array;
  1679. GET_REFLECTION_OBJECT_PTR(fptr);
  1680. if (zend_parse_parameters(ZEND_NUM_ARGS(), "a", &param_array) == FAILURE) {
  1681. return;
  1682. }
  1683. argc = zend_hash_num_elements(Z_ARRVAL_P(param_array));
  1684. params = safe_emalloc(sizeof(zval), argc, 0);
  1685. argc = 0;
  1686. ZEND_HASH_FOREACH_VAL(Z_ARRVAL_P(param_array), val) {
  1687. ZVAL_COPY(&params[argc], val);
  1688. argc++;
  1689. } ZEND_HASH_FOREACH_END();
  1690. fci.size = sizeof(fci);
  1691. ZVAL_UNDEF(&fci.function_name);
  1692. fci.object = NULL;
  1693. fci.retval = &retval;
  1694. fci.param_count = argc;
  1695. fci.params = params;
  1696. fci.no_separation = 1;
  1697. fcc.function_handler = fptr;
  1698. fcc.called_scope = NULL;
  1699. fcc.object = NULL;
  1700. if (!Z_ISUNDEF(intern->obj)) {
  1701. Z_OBJ_HT(intern->obj)->get_closure(
  1702. &intern->obj, &fcc.called_scope, &fcc.function_handler, &fcc.object);
  1703. }
  1704. result = zend_call_function(&fci, &fcc);
  1705. for (i = 0; i < argc; i++) {
  1706. zval_ptr_dtor(&params[i]);
  1707. }
  1708. efree(params);
  1709. if (result == FAILURE) {
  1710. zend_throw_exception_ex(reflection_exception_ptr, 0,
  1711. "Invocation of function %s() failed", ZSTR_VAL(fptr->common.function_name));
  1712. return;
  1713. }
  1714. if (Z_TYPE(retval) != IS_UNDEF) {
  1715. if (Z_ISREF(retval)) {
  1716. zend_unwrap_reference(&retval);
  1717. }
  1718. ZVAL_COPY_VALUE(return_value, &retval);
  1719. }
  1720. }
  1721. /* }}} */
  1722. /* {{{ proto public bool ReflectionFunction::returnsReference()
  1723. Gets whether this function returns a reference */
  1724. ZEND_METHOD(reflection_function, returnsReference)
  1725. {
  1726. reflection_object *intern;
  1727. zend_function *fptr;
  1728. GET_REFLECTION_OBJECT_PTR(fptr);
  1729. RETURN_BOOL((fptr->op_array.fn_flags & ZEND_ACC_RETURN_REFERENCE) != 0);
  1730. }
  1731. /* }}} */
  1732. /* {{{ proto public bool ReflectionFunction::getNumberOfParameters()
  1733. Gets the number of parameters */
  1734. ZEND_METHOD(reflection_function, getNumberOfParameters)
  1735. {
  1736. reflection_object *intern;
  1737. zend_function *fptr;
  1738. uint32_t num_args;
  1739. GET_REFLECTION_OBJECT_PTR(fptr);
  1740. num_args = fptr->common.num_args;
  1741. if (fptr->common.fn_flags & ZEND_ACC_VARIADIC) {
  1742. num_args++;
  1743. }
  1744. RETURN_LONG(num_args);
  1745. }
  1746. /* }}} */
  1747. /* {{{ proto public bool ReflectionFunction::getNumberOfRequiredParameters()
  1748. Gets the number of required parameters */
  1749. ZEND_METHOD(reflection_function, getNumberOfRequiredParameters)
  1750. {
  1751. reflection_object *intern;
  1752. zend_function *fptr;
  1753. GET_REFLECTION_OBJECT_PTR(fptr);
  1754. RETURN_LONG(fptr->common.required_num_args);
  1755. }
  1756. /* }}} */
  1757. /* {{{ proto public ReflectionParameter[] ReflectionFunction::getParameters()
  1758. Returns an array of parameter objects for this function */
  1759. ZEND_METHOD(reflection_function, getParameters)
  1760. {
  1761. reflection_object *intern;
  1762. zend_function *fptr;
  1763. uint32_t i, num_args;
  1764. struct _zend_arg_info *arg_info;
  1765. GET_REFLECTION_OBJECT_PTR(fptr);
  1766. arg_info= fptr->common.arg_info;
  1767. num_args = fptr->common.num_args;
  1768. if (fptr->common.fn_flags & ZEND_ACC_VARIADIC) {
  1769. num_args++;
  1770. }
  1771. if (!num_args) {
  1772. ZVAL_EMPTY_ARRAY(return_value);
  1773. return;
  1774. }
  1775. array_init(return_value);
  1776. for (i = 0; i < num_args; i++) {
  1777. zval parameter;
  1778. reflection_parameter_factory(
  1779. _copy_function(fptr),
  1780. Z_ISUNDEF(intern->obj) ? NULL : &intern->obj,
  1781. arg_info,
  1782. i,
  1783. i < fptr->common.required_num_args,
  1784. &parameter
  1785. );
  1786. add_next_index_zval(return_value, &parameter);
  1787. arg_info++;
  1788. }
  1789. }
  1790. /* }}} */
  1791. /* {{{ proto public ReflectionExtension|NULL ReflectionFunction::getExtension()
  1792. Returns NULL or the extension the function belongs to */
  1793. ZEND_METHOD(reflection_function, getExtension)
  1794. {
  1795. reflection_object *intern;
  1796. zend_function *fptr;
  1797. zend_internal_function *internal;
  1798. GET_REFLECTION_OBJECT_PTR(fptr);
  1799. if (fptr->type != ZEND_INTERNAL_FUNCTION) {
  1800. RETURN_NULL();
  1801. }
  1802. internal = (zend_internal_function *)fptr;
  1803. if (internal->module) {
  1804. reflection_extension_factory(return_value, internal->module->name);
  1805. } else {
  1806. RETURN_NULL();
  1807. }
  1808. }
  1809. /* }}} */
  1810. /* {{{ proto public string|false ReflectionFunction::getExtensionName()
  1811. Returns false or the name of the extension the function belongs to */
  1812. ZEND_METHOD(reflection_function, getExtensionName)
  1813. {
  1814. reflection_object *intern;
  1815. zend_function *fptr;
  1816. zend_internal_function *internal;
  1817. GET_REFLECTION_OBJECT_PTR(fptr);
  1818. if (fptr->type != ZEND_INTERNAL_FUNCTION) {
  1819. RETURN_FALSE;
  1820. }
  1821. internal = (zend_internal_function *)fptr;
  1822. if (internal->module) {
  1823. RETURN_STRING(internal->module->name);
  1824. } else {
  1825. RETURN_FALSE;
  1826. }
  1827. }
  1828. /* }}} */
  1829. /* {{{ proto public void ReflectionGenerator::__construct(object Generator) */
  1830. ZEND_METHOD(reflection_generator, __construct)
  1831. {
  1832. zval *generator, *object;
  1833. reflection_object *intern;
  1834. zend_execute_data *ex;
  1835. object = getThis();
  1836. intern = Z_REFLECTION_P(object);
  1837. if (zend_parse_parameters_throw(ZEND_NUM_ARGS(), "O", &generator, zend_ce_generator) == FAILURE) {
  1838. return;
  1839. }
  1840. ex = ((zend_generator *) Z_OBJ_P(generator))->execute_data;
  1841. if (!ex) {
  1842. _DO_THROW("Cannot create ReflectionGenerator based on a terminated Generator");
  1843. return;
  1844. }
  1845. intern->ref_type = REF_TYPE_GENERATOR;
  1846. ZVAL_COPY(&intern->obj, generator);
  1847. intern->ce = zend_ce_generator;
  1848. }
  1849. /* }}} */
  1850. #define REFLECTION_CHECK_VALID_GENERATOR(ex) \
  1851. if (!ex) { \
  1852. _DO_THROW("Cannot fetch information from a terminated Generator"); \
  1853. return; \
  1854. }
  1855. /* {{{ proto public array ReflectionGenerator::getTrace($options = DEBUG_BACKTRACE_PROVIDE_OBJECT) */
  1856. ZEND_METHOD(reflection_generator, getTrace)
  1857. {
  1858. zend_long options = DEBUG_BACKTRACE_PROVIDE_OBJECT;
  1859. zend_generator *generator = (zend_generator *) Z_OBJ(Z_REFLECTION_P(getThis())->obj);
  1860. zend_generator *root_generator;
  1861. zend_execute_data *ex_backup = EG(current_execute_data);
  1862. zend_execute_data *ex = generator->execute_data;
  1863. zend_execute_data *root_prev = NULL, *cur_prev;
  1864. if (zend_parse_parameters(ZEND_NUM_ARGS(), "|l", &options) == FAILURE) {
  1865. return;
  1866. }
  1867. REFLECTION_CHECK_VALID_GENERATOR(ex)
  1868. root_generator = zend_generator_get_current(generator);
  1869. cur_prev = generator->execute_data->prev_execute_data;
  1870. if (generator == root_generator) {
  1871. generator->execute_data->prev_execute_data = NULL;
  1872. } else {
  1873. root_prev = root_generator->execute_data->prev_execute_data;
  1874. generator->execute_fake.prev_execute_data = NULL;
  1875. root_generator->execute_data->prev_execute_data = &generator->execute_fake;
  1876. }
  1877. EG(current_execute_data) = root_generator->execute_data;
  1878. zend_fetch_debug_backtrace(return_value, 0, options, 0);
  1879. EG(current_execute_data) = ex_backup;
  1880. root_generator->execute_data->prev_execute_data = root_prev;
  1881. generator->execute_data->prev_execute_data = cur_prev;
  1882. }
  1883. /* }}} */
  1884. /* {{{ proto public int ReflectionGenerator::getExecutingLine() */
  1885. ZEND_METHOD(reflection_generator, getExecutingLine)
  1886. {
  1887. zend_generator *generator = (zend_generator *) Z_OBJ(Z_REFLECTION_P(getThis())->obj);
  1888. zend_execute_data *ex = generator->execute_data;
  1889. if (zend_parse_parameters_none() == FAILURE) {
  1890. return;
  1891. }
  1892. REFLECTION_CHECK_VALID_GENERATOR(ex)
  1893. ZVAL_LONG(return_value, ex->opline->lineno);
  1894. }
  1895. /* }}} */
  1896. /* {{{ proto public string ReflectionGenerator::getExecutingFile() */
  1897. ZEND_METHOD(reflection_generator, getExecutingFile)
  1898. {
  1899. zend_generator *generator = (zend_generator *) Z_OBJ(Z_REFLECTION_P(getThis())->obj);
  1900. zend_execute_data *ex = generator->execute_data;
  1901. if (zend_parse_parameters_none() == FAILURE) {
  1902. return;
  1903. }
  1904. REFLECTION_CHECK_VALID_GENERATOR(ex)
  1905. ZVAL_STR_COPY(return_value, ex->func->op_array.filename);
  1906. }
  1907. /* }}} */
  1908. /* {{{ proto public ReflectionFunctionAbstract ReflectionGenerator::getFunction() */
  1909. ZEND_METHOD(reflection_generator, getFunction)
  1910. {
  1911. zend_generator *generator = (zend_generator *) Z_OBJ(Z_REFLECTION_P(getThis())->obj);
  1912. zend_execute_data *ex = generator->execute_data;
  1913. if (zend_parse_parameters_none() == FAILURE) {
  1914. return;
  1915. }
  1916. REFLECTION_CHECK_VALID_GENERATOR(ex)
  1917. if (ex->func->common.fn_flags & ZEND_ACC_CLOSURE) {
  1918. zval closure;
  1919. ZVAL_OBJ(&closure, ZEND_CLOSURE_OBJECT(ex->func));
  1920. reflection_function_factory(ex->func, &closure, return_value);
  1921. } else if (ex->func->op_array.scope) {
  1922. reflection_method_factory(ex->func->op_array.scope, ex->func, NULL, return_value);
  1923. } else {
  1924. reflection_function_factory(ex->func, NULL, return_value);
  1925. }
  1926. }
  1927. /* }}} */
  1928. /* {{{ proto public object ReflectionGenerator::getThis() */
  1929. ZEND_METHOD(reflection_generator, getThis)
  1930. {
  1931. zend_generator *generator = (zend_generator *) Z_OBJ(Z_REFLECTION_P(getThis())->obj);
  1932. zend_execute_data *ex = generator->execute_data;
  1933. if (zend_parse_parameters_none() == FAILURE) {
  1934. return;
  1935. }
  1936. REFLECTION_CHECK_VALID_GENERATOR(ex)
  1937. if (Z_TYPE(ex->This) == IS_OBJECT) {
  1938. ZVAL_COPY(return_value, &ex->This);
  1939. } else {
  1940. ZVAL_NULL(return_value);
  1941. }
  1942. }
  1943. /* }}} */
  1944. /* {{{ proto public Generator ReflectionGenerator::getExecutingGenerator() */
  1945. ZEND_METHOD(reflection_generator, getExecutingGenerator)
  1946. {
  1947. zend_generator *generator = (zend_generator *) Z_OBJ(Z_REFLECTION_P(getThis())->obj);
  1948. zend_execute_data *ex = generator->execute_data;
  1949. zend_generator *current;
  1950. if (zend_parse_parameters_none() == FAILURE) {
  1951. return;
  1952. }
  1953. REFLECTION_CHECK_VALID_GENERATOR(ex)
  1954. current = zend_generator_get_current(generator);
  1955. GC_ADDREF(&current->std);
  1956. ZVAL_OBJ(return_value, (zend_object *) current);
  1957. }
  1958. /* }}} */
  1959. /* {{{ proto public static mixed ReflectionParameter::export(mixed function, mixed parameter [, bool return]) throws ReflectionException
  1960. Exports a reflection object. Returns the output if TRUE is specified for return, printing it otherwise. */
  1961. ZEND_METHOD(reflection_parameter, export)
  1962. {
  1963. _reflection_export(INTERNAL_FUNCTION_PARAM_PASSTHRU, reflection_parameter_ptr, 2);
  1964. }
  1965. /* }}} */
  1966. /* {{{ proto public void ReflectionParameter::__construct(mixed function, mixed parameter)
  1967. Constructor. Throws an Exception in case the given method does not exist */
  1968. ZEND_METHOD(reflection_parameter, __construct)
  1969. {
  1970. parameter_reference *ref;
  1971. zval *reference, *parameter;
  1972. zval *object;
  1973. zval name;
  1974. reflection_object *intern;
  1975. zend_function *fptr;
  1976. struct _zend_arg_info *arg_info;
  1977. int position;
  1978. uint32_t num_args;
  1979. zend_class_entry *ce = NULL;
  1980. zend_bool is_closure = 0;
  1981. if (zend_parse_parameters_throw(ZEND_NUM_ARGS(), "zz", &reference, &parameter) == FAILURE) {
  1982. return;
  1983. }
  1984. object = getThis();
  1985. intern = Z_REFLECTION_P(object);
  1986. /* First, find the function */
  1987. switch (Z_TYPE_P(reference)) {
  1988. case IS_STRING: {
  1989. size_t lcname_len;
  1990. char *lcname;
  1991. lcname_len = Z_STRLEN_P(reference);
  1992. lcname = zend_str_tolower_dup(Z_STRVAL_P(reference), lcname_len);
  1993. if ((fptr = zend_hash_str_find_ptr(EG(function_table), lcname, lcname_len)) == NULL) {
  1994. efree(lcname);
  1995. zend_throw_exception_ex(reflection_exception_ptr, 0,
  1996. "Function %s() does not exist", Z_STRVAL_P(reference));
  1997. return;
  1998. }
  1999. efree(lcname);
  2000. }
  2001. ce = fptr->common.scope;
  2002. break;
  2003. case IS_ARRAY: {
  2004. zval *classref;
  2005. zval *method;
  2006. size_t lcname_len;
  2007. char *lcname;
  2008. if (((classref =zend_hash_index_find(Z_ARRVAL_P(reference), 0)) == NULL)
  2009. || ((method = zend_hash_index_find(Z_ARRVAL_P(reference), 1)) == NULL))
  2010. {
  2011. _DO_THROW("Expected array($object, $method) or array($classname, $method)");
  2012. /* returns out of this function */
  2013. }
  2014. if (Z_TYPE_P(classref) == IS_OBJECT) {
  2015. ce = Z_OBJCE_P(classref);
  2016. } else {
  2017. convert_to_string_ex(classref);
  2018. if ((ce = zend_lookup_class(Z_STR_P(classref))) == NULL) {
  2019. zend_throw_exception_ex(reflection_exception_ptr, 0,
  2020. "Class %s does not exist", Z_STRVAL_P(classref));
  2021. return;
  2022. }
  2023. }
  2024. convert_to_string_ex(method);
  2025. lcname_len = Z_STRLEN_P(method);
  2026. lcname = zend_str_tolower_dup(Z_STRVAL_P(method), lcname_len);
  2027. if (ce == zend_ce_closure && Z_TYPE_P(classref) == IS_OBJECT
  2028. && (lcname_len == sizeof(ZEND_INVOKE_FUNC_NAME)-1)
  2029. && memcmp(lcname, ZEND_INVOKE_FUNC_NAME, sizeof(ZEND_INVOKE_FUNC_NAME)-1) == 0
  2030. && (fptr = zend_get_closure_invoke_method(Z_OBJ_P(classref))) != NULL)
  2031. {
  2032. /* nothing to do. don't set is_closure since is the invoke handler,
  2033. not the closure itself */
  2034. } else if ((fptr = zend_hash_str_find_ptr(&ce->function_table, lcname, lcname_len)) == NULL) {
  2035. efree(lcname);
  2036. zend_throw_exception_ex(reflection_exception_ptr, 0,
  2037. "Method %s::%s() does not exist", ZSTR_VAL(ce->name), Z_STRVAL_P(method));
  2038. return;
  2039. }
  2040. efree(lcname);
  2041. }
  2042. break;
  2043. case IS_OBJECT: {
  2044. ce = Z_OBJCE_P(reference);
  2045. if (instanceof_function(ce, zend_ce_closure)) {
  2046. fptr = (zend_function *)zend_get_closure_method_def(reference);
  2047. Z_ADDREF_P(reference);
  2048. is_closure = 1;
  2049. } else if ((fptr = zend_hash_str_find_ptr(&ce->function_table, ZEND_INVOKE_FUNC_NAME, sizeof(ZEND_INVOKE_FUNC_NAME))) == NULL) {
  2050. zend_throw_exception_ex(reflection_exception_ptr, 0,
  2051. "Method %s::%s() does not exist", ZSTR_VAL(ce->name), ZEND_INVOKE_FUNC_NAME);
  2052. return;
  2053. }
  2054. }
  2055. break;
  2056. default:
  2057. _DO_THROW("The parameter class is expected to be either a string, an array(class, method) or a callable object");
  2058. /* returns out of this function */
  2059. }
  2060. /* Now, search for the parameter */
  2061. arg_info = fptr->common.arg_info;
  2062. num_args = fptr->common.num_args;
  2063. if (fptr->common.fn_flags & ZEND_ACC_VARIADIC) {
  2064. num_args++;
  2065. }
  2066. if (Z_TYPE_P(parameter) == IS_LONG) {
  2067. position= (int)Z_LVAL_P(parameter);
  2068. if (position < 0 || (uint32_t)position >= num_args) {
  2069. if (fptr->common.fn_flags & ZEND_ACC_CALL_VIA_TRAMPOLINE) {
  2070. if (fptr->type != ZEND_OVERLOADED_FUNCTION) {
  2071. zend_string_release_ex(fptr->common.function_name, 0);
  2072. }
  2073. zend_free_trampoline(fptr);
  2074. }
  2075. if (is_closure) {
  2076. zval_ptr_dtor(reference);
  2077. }
  2078. _DO_THROW("The parameter specified by its offset could not be found");
  2079. /* returns out of this function */
  2080. }
  2081. } else {
  2082. uint32_t i;
  2083. position= -1;
  2084. convert_to_string_ex(parameter);
  2085. if (fptr->type == ZEND_INTERNAL_FUNCTION &&
  2086. !(fptr->common.fn_flags & ZEND_ACC_USER_ARG_INFO)) {
  2087. for (i = 0; i < num_args; i++) {
  2088. if (arg_info[i].name) {
  2089. if (strcmp(((zend_internal_arg_info*)arg_info)[i].name, Z_STRVAL_P(parameter)) == 0) {
  2090. position= i;
  2091. break;
  2092. }
  2093. }
  2094. }
  2095. } else {
  2096. for (i = 0; i < num_args; i++) {
  2097. if (arg_info[i].name) {
  2098. if (strcmp(ZSTR_VAL(arg_info[i].name), Z_STRVAL_P(parameter)) == 0) {
  2099. position= i;
  2100. break;
  2101. }
  2102. }
  2103. }
  2104. }
  2105. if (position == -1) {
  2106. if (fptr->common.fn_flags & ZEND_ACC_CALL_VIA_TRAMPOLINE) {
  2107. if (fptr->type != ZEND_OVERLOADED_FUNCTION) {
  2108. zend_string_release_ex(fptr->common.function_name, 0);
  2109. }
  2110. zend_free_trampoline(fptr);
  2111. }
  2112. if (is_closure) {
  2113. zval_ptr_dtor(reference);
  2114. }
  2115. _DO_THROW("The parameter specified by its name could not be found");
  2116. /* returns out of this function */
  2117. }
  2118. }
  2119. if (arg_info[position].name) {
  2120. if (fptr->type == ZEND_INTERNAL_FUNCTION &&
  2121. !(fptr->common.fn_flags & ZEND_ACC_USER_ARG_INFO)) {
  2122. ZVAL_STRING(&name, ((zend_internal_arg_info*)arg_info)[position].name);
  2123. } else {
  2124. ZVAL_STR_COPY(&name, arg_info[position].name);
  2125. }
  2126. } else {
  2127. ZVAL_NULL(&name);
  2128. }
  2129. reflection_update_property_name(object, &name);
  2130. ref = (parameter_reference*) emalloc(sizeof(parameter_reference));
  2131. ref->arg_info = &arg_info[position];
  2132. ref->offset = (uint32_t)position;
  2133. ref->required = position < fptr->common.required_num_args;
  2134. ref->fptr = fptr;
  2135. /* TODO: copy fptr */
  2136. intern->ptr = ref;
  2137. intern->ref_type = REF_TYPE_PARAMETER;
  2138. intern->ce = ce;
  2139. if (reference && is_closure) {
  2140. ZVAL_COPY_VALUE(&intern->obj, reference);
  2141. }
  2142. }
  2143. /* }}} */
  2144. /* {{{ proto public string ReflectionParameter::__toString()
  2145. Returns a string representation */
  2146. ZEND_METHOD(reflection_parameter, __toString)
  2147. {
  2148. reflection_object *intern;
  2149. parameter_reference *param;
  2150. smart_str str = {0};
  2151. if (zend_parse_parameters_none() == FAILURE) {
  2152. return;
  2153. }
  2154. GET_REFLECTION_OBJECT_PTR(param);
  2155. _parameter_string(&str, param->fptr, param->arg_info, param->offset, param->required, "");
  2156. RETURN_STR(smart_str_extract(&str));
  2157. }
  2158. /* }}} */
  2159. /* {{{ proto public string ReflectionParameter::getName()
  2160. Returns this parameters's name */
  2161. ZEND_METHOD(reflection_parameter, getName)
  2162. {
  2163. if (zend_parse_parameters_none() == FAILURE) {
  2164. return;
  2165. }
  2166. _default_get_name(getThis(), return_value);
  2167. }
  2168. /* }}} */
  2169. /* {{{ proto public ReflectionFunction ReflectionParameter::getDeclaringFunction()
  2170. Returns the ReflectionFunction for the function of this parameter */
  2171. ZEND_METHOD(reflection_parameter, getDeclaringFunction)
  2172. {
  2173. reflection_object *intern;
  2174. parameter_reference *param;
  2175. if (zend_parse_parameters_none() == FAILURE) {
  2176. return;
  2177. }
  2178. GET_REFLECTION_OBJECT_PTR(param);
  2179. if (!param->fptr->common.scope) {
  2180. reflection_function_factory(_copy_function(param->fptr), Z_ISUNDEF(intern->obj)? NULL : &intern->obj, return_value);
  2181. } else {
  2182. reflection_method_factory(param->fptr->common.scope, _copy_function(param->fptr), Z_ISUNDEF(intern->obj)? NULL : &intern->obj, return_value);
  2183. }
  2184. }
  2185. /* }}} */
  2186. /* {{{ proto public ReflectionClass|NULL ReflectionParameter::getDeclaringClass()
  2187. Returns in which class this parameter is defined (not the type of the parameter) */
  2188. ZEND_METHOD(reflection_parameter, getDeclaringClass)
  2189. {
  2190. reflection_object *intern;
  2191. parameter_reference *param;
  2192. if (zend_parse_parameters_none() == FAILURE) {
  2193. return;
  2194. }
  2195. GET_REFLECTION_OBJECT_PTR(param);
  2196. if (param->fptr->common.scope) {
  2197. zend_reflection_class_factory(param->fptr->common.scope, return_value);
  2198. }
  2199. }
  2200. /* }}} */
  2201. /* {{{ proto public ReflectionClass|NULL ReflectionParameter::getClass()
  2202. Returns this parameters's class hint or NULL if there is none */
  2203. ZEND_METHOD(reflection_parameter, getClass)
  2204. {
  2205. reflection_object *intern;
  2206. parameter_reference *param;
  2207. zend_class_entry *ce;
  2208. if (zend_parse_parameters_none() == FAILURE) {
  2209. return;
  2210. }
  2211. GET_REFLECTION_OBJECT_PTR(param);
  2212. if (ZEND_TYPE_IS_CLASS(param->arg_info->type)) {
  2213. /* Class name is stored as a string, we might also get "self" or "parent"
  2214. * - For "self", simply use the function scope. If scope is NULL then
  2215. * the function is global and thus self does not make any sense
  2216. *
  2217. * - For "parent", use the function scope's parent. If scope is NULL then
  2218. * the function is global and thus parent does not make any sense.
  2219. * If the parent is NULL then the class does not extend anything and
  2220. * thus parent does not make any sense, either.
  2221. *
  2222. * TODO: Think about moving these checks to the compiler or some sort of
  2223. * lint-mode.
  2224. */
  2225. zend_string *class_name;
  2226. class_name = ZEND_TYPE_NAME(param->arg_info->type);
  2227. if (0 == zend_binary_strcasecmp(ZSTR_VAL(class_name), ZSTR_LEN(class_name), "self", sizeof("self")- 1)) {
  2228. ce = param->fptr->common.scope;
  2229. if (!ce) {
  2230. zend_throw_exception_ex(reflection_exception_ptr, 0,
  2231. "Parameter uses 'self' as type hint but function is not a class member!");
  2232. return;
  2233. }
  2234. } else if (0 == zend_binary_strcasecmp(ZSTR_VAL(class_name), ZSTR_LEN(class_name), "parent", sizeof("parent")- 1)) {
  2235. ce = param->fptr->common.scope;
  2236. if (!ce) {
  2237. zend_throw_exception_ex(reflection_exception_ptr, 0,
  2238. "Parameter uses 'parent' as type hint but function is not a class member!");
  2239. return;
  2240. }
  2241. if (!ce->parent) {
  2242. zend_throw_exception_ex(reflection_exception_ptr, 0,
  2243. "Parameter uses 'parent' as type hint although class does not have a parent!");
  2244. return;
  2245. }
  2246. ce = ce->parent;
  2247. } else {
  2248. ce = zend_lookup_class(class_name);
  2249. if (!ce) {
  2250. zend_throw_exception_ex(reflection_exception_ptr, 0,
  2251. "Class %s does not exist", ZSTR_VAL(class_name));
  2252. return;
  2253. }
  2254. }
  2255. zend_reflection_class_factory(ce, return_value);
  2256. }
  2257. }
  2258. /* }}} */
  2259. /* {{{ proto public bool ReflectionParameter::hasType()
  2260. Returns whether parameter has a type */
  2261. ZEND_METHOD(reflection_parameter, hasType)
  2262. {
  2263. reflection_object *intern;
  2264. parameter_reference *param;
  2265. if (zend_parse_parameters_none() == FAILURE) {
  2266. return;
  2267. }
  2268. GET_REFLECTION_OBJECT_PTR(param);
  2269. RETVAL_BOOL(ZEND_TYPE_IS_SET(param->arg_info->type));
  2270. }
  2271. /* }}} */
  2272. /* {{{ proto public ReflectionType ReflectionParameter::getType()
  2273. Returns the type associated with the parameter */
  2274. ZEND_METHOD(reflection_parameter, getType)
  2275. {
  2276. reflection_object *intern;
  2277. parameter_reference *param;
  2278. if (zend_parse_parameters_none() == FAILURE) {
  2279. return;
  2280. }
  2281. GET_REFLECTION_OBJECT_PTR(param);
  2282. if (!ZEND_TYPE_IS_SET(param->arg_info->type)) {
  2283. RETURN_NULL();
  2284. }
  2285. reflection_type_factory(_copy_function(param->fptr), Z_ISUNDEF(intern->obj)? NULL : &intern->obj, param->arg_info, return_value);
  2286. }
  2287. /* }}} */
  2288. /* {{{ proto public bool ReflectionParameter::isArray()
  2289. Returns whether parameter MUST be an array */
  2290. ZEND_METHOD(reflection_parameter, isArray)
  2291. {
  2292. reflection_object *intern;
  2293. parameter_reference *param;
  2294. if (zend_parse_parameters_none() == FAILURE) {
  2295. return;
  2296. }
  2297. GET_REFLECTION_OBJECT_PTR(param);
  2298. RETVAL_BOOL(ZEND_TYPE_CODE(param->arg_info->type) == IS_ARRAY);
  2299. }
  2300. /* }}} */
  2301. /* {{{ proto public bool ReflectionParameter::isCallable()
  2302. Returns whether parameter MUST be callable */
  2303. ZEND_METHOD(reflection_parameter, isCallable)
  2304. {
  2305. reflection_object *intern;
  2306. parameter_reference *param;
  2307. if (zend_parse_parameters_none() == FAILURE) {
  2308. return;
  2309. }
  2310. GET_REFLECTION_OBJECT_PTR(param);
  2311. RETVAL_BOOL(ZEND_TYPE_CODE(param->arg_info->type) == IS_CALLABLE);
  2312. }
  2313. /* }}} */
  2314. /* {{{ proto public bool ReflectionParameter::allowsNull()
  2315. Returns whether NULL is allowed as this parameters's value */
  2316. ZEND_METHOD(reflection_parameter, allowsNull)
  2317. {
  2318. reflection_object *intern;
  2319. parameter_reference *param;
  2320. if (zend_parse_parameters_none() == FAILURE) {
  2321. return;
  2322. }
  2323. GET_REFLECTION_OBJECT_PTR(param);
  2324. RETVAL_BOOL(ZEND_TYPE_ALLOW_NULL(param->arg_info->type));
  2325. }
  2326. /* }}} */
  2327. /* {{{ proto public bool ReflectionParameter::isPassedByReference()
  2328. Returns whether this parameters is passed to by reference */
  2329. ZEND_METHOD(reflection_parameter, isPassedByReference)
  2330. {
  2331. reflection_object *intern;
  2332. parameter_reference *param;
  2333. if (zend_parse_parameters_none() == FAILURE) {
  2334. return;
  2335. }
  2336. GET_REFLECTION_OBJECT_PTR(param);
  2337. RETVAL_BOOL(param->arg_info->pass_by_reference);
  2338. }
  2339. /* }}} */
  2340. /* {{{ proto public bool ReflectionParameter::canBePassedByValue()
  2341. Returns whether this parameter can be passed by value */
  2342. ZEND_METHOD(reflection_parameter, canBePassedByValue)
  2343. {
  2344. reflection_object *intern;
  2345. parameter_reference *param;
  2346. if (zend_parse_parameters_none() == FAILURE) {
  2347. return;
  2348. }
  2349. GET_REFLECTION_OBJECT_PTR(param);
  2350. /* true if it's ZEND_SEND_BY_VAL or ZEND_SEND_PREFER_REF */
  2351. RETVAL_BOOL(param->arg_info->pass_by_reference != ZEND_SEND_BY_REF);
  2352. }
  2353. /* }}} */
  2354. /* {{{ proto public bool ReflectionParameter::getPosition()
  2355. Returns whether this parameter is an optional parameter */
  2356. ZEND_METHOD(reflection_parameter, getPosition)
  2357. {
  2358. reflection_object *intern;
  2359. parameter_reference *param;
  2360. if (zend_parse_parameters_none() == FAILURE) {
  2361. return;
  2362. }
  2363. GET_REFLECTION_OBJECT_PTR(param);
  2364. RETVAL_LONG(param->offset);
  2365. }
  2366. /* }}} */
  2367. /* {{{ proto public bool ReflectionParameter::isOptional()
  2368. Returns whether this parameter is an optional parameter */
  2369. ZEND_METHOD(reflection_parameter, isOptional)
  2370. {
  2371. reflection_object *intern;
  2372. parameter_reference *param;
  2373. if (zend_parse_parameters_none() == FAILURE) {
  2374. return;
  2375. }
  2376. GET_REFLECTION_OBJECT_PTR(param);
  2377. RETVAL_BOOL(!param->required);
  2378. }
  2379. /* }}} */
  2380. /* {{{ proto public bool ReflectionParameter::isDefaultValueAvailable()
  2381. Returns whether the default value of this parameter is available */
  2382. ZEND_METHOD(reflection_parameter, isDefaultValueAvailable)
  2383. {
  2384. reflection_object *intern;
  2385. parameter_reference *param;
  2386. zend_op *precv;
  2387. if (zend_parse_parameters_none() == FAILURE) {
  2388. return;
  2389. }
  2390. GET_REFLECTION_OBJECT_PTR(param);
  2391. if (param->fptr->type != ZEND_USER_FUNCTION)
  2392. {
  2393. RETURN_FALSE;
  2394. }
  2395. precv = _get_recv_op((zend_op_array*)param->fptr, param->offset);
  2396. if (!precv || precv->opcode != ZEND_RECV_INIT || precv->op2_type == IS_UNUSED) {
  2397. RETURN_FALSE;
  2398. }
  2399. RETURN_TRUE;
  2400. }
  2401. /* }}} */
  2402. /* {{{ proto public bool ReflectionParameter::getDefaultValue()
  2403. Returns the default value of this parameter or throws an exception */
  2404. ZEND_METHOD(reflection_parameter, getDefaultValue)
  2405. {
  2406. parameter_reference *param;
  2407. zend_op *precv;
  2408. if (zend_parse_parameters_none() == FAILURE) {
  2409. return;
  2410. }
  2411. param = _reflection_param_get_default_param(INTERNAL_FUNCTION_PARAM_PASSTHRU);
  2412. if (!param) {
  2413. return;
  2414. }
  2415. precv = _reflection_param_get_default_precv(INTERNAL_FUNCTION_PARAM_PASSTHRU, param);
  2416. if (!precv) {
  2417. return;
  2418. }
  2419. ZVAL_COPY(return_value, RT_CONSTANT(precv, precv->op2));
  2420. if (Z_TYPE_P(return_value) == IS_CONSTANT_AST) {
  2421. zval_update_constant_ex(return_value, param->fptr->common.scope);
  2422. }
  2423. }
  2424. /* }}} */
  2425. /* {{{ proto public bool ReflectionParameter::isDefaultValueConstant()
  2426. Returns whether the default value of this parameter is constant */
  2427. ZEND_METHOD(reflection_parameter, isDefaultValueConstant)
  2428. {
  2429. zend_op *precv;
  2430. parameter_reference *param;
  2431. if (zend_parse_parameters_none() == FAILURE) {
  2432. return;
  2433. }
  2434. param = _reflection_param_get_default_param(INTERNAL_FUNCTION_PARAM_PASSTHRU);
  2435. if (!param) {
  2436. RETURN_FALSE;
  2437. }
  2438. precv = _reflection_param_get_default_precv(INTERNAL_FUNCTION_PARAM_PASSTHRU, param);
  2439. if (precv && Z_TYPE_P(RT_CONSTANT(precv, precv->op2)) == IS_CONSTANT_AST) {
  2440. zend_ast *ast = Z_ASTVAL_P(RT_CONSTANT(precv, precv->op2));
  2441. if (ast->kind == ZEND_AST_CONSTANT
  2442. || ast->kind == ZEND_AST_CONSTANT_CLASS) {
  2443. RETURN_TRUE;
  2444. }
  2445. }
  2446. RETURN_FALSE;
  2447. }
  2448. /* }}} */
  2449. /* {{{ proto public mixed ReflectionParameter::getDefaultValueConstantName()
  2450. Returns the default value's constant name if default value is constant or null */
  2451. ZEND_METHOD(reflection_parameter, getDefaultValueConstantName)
  2452. {
  2453. zend_op *precv;
  2454. parameter_reference *param;
  2455. if (zend_parse_parameters_none() == FAILURE) {
  2456. return;
  2457. }
  2458. param = _reflection_param_get_default_param(INTERNAL_FUNCTION_PARAM_PASSTHRU);
  2459. if (!param) {
  2460. return;
  2461. }
  2462. precv = _reflection_param_get_default_precv(INTERNAL_FUNCTION_PARAM_PASSTHRU, param);
  2463. if (precv && Z_TYPE_P(RT_CONSTANT(precv, precv->op2)) == IS_CONSTANT_AST) {
  2464. zend_ast *ast = Z_ASTVAL_P(RT_CONSTANT(precv, precv->op2));
  2465. if (ast->kind == ZEND_AST_CONSTANT) {
  2466. RETURN_STR_COPY(zend_ast_get_constant_name(ast));
  2467. } else if (ast->kind == ZEND_AST_CONSTANT_CLASS) {
  2468. RETURN_STRINGL("__CLASS__", sizeof("__CLASS__")-1);
  2469. }
  2470. }
  2471. }
  2472. /* }}} */
  2473. /* {{{ proto public bool ReflectionParameter::isVariadic()
  2474. Returns whether this parameter is a variadic parameter */
  2475. ZEND_METHOD(reflection_parameter, isVariadic)
  2476. {
  2477. reflection_object *intern;
  2478. parameter_reference *param;
  2479. if (zend_parse_parameters_none() == FAILURE) {
  2480. return;
  2481. }
  2482. GET_REFLECTION_OBJECT_PTR(param);
  2483. RETVAL_BOOL(param->arg_info->is_variadic);
  2484. }
  2485. /* }}} */
  2486. /* {{{ proto public bool ReflectionType::allowsNull()
  2487. Returns whether parameter MAY be null */
  2488. ZEND_METHOD(reflection_type, allowsNull)
  2489. {
  2490. reflection_object *intern;
  2491. type_reference *param;
  2492. if (zend_parse_parameters_none() == FAILURE) {
  2493. return;
  2494. }
  2495. GET_REFLECTION_OBJECT_PTR(param);
  2496. RETVAL_BOOL(ZEND_TYPE_ALLOW_NULL(param->arg_info->type));
  2497. }
  2498. /* }}} */
  2499. /* {{{ proto public bool ReflectionType::isBuiltin()
  2500. Returns whether parameter is a builtin type */
  2501. ZEND_METHOD(reflection_type, isBuiltin)
  2502. {
  2503. reflection_object *intern;
  2504. type_reference *param;
  2505. if (zend_parse_parameters_none() == FAILURE) {
  2506. return;
  2507. }
  2508. GET_REFLECTION_OBJECT_PTR(param);
  2509. RETVAL_BOOL(ZEND_TYPE_IS_CODE(param->arg_info->type));
  2510. }
  2511. /* }}} */
  2512. /* {{{ reflection_type_name */
  2513. static zend_string *reflection_type_name(type_reference *param) {
  2514. if (ZEND_TYPE_IS_CLASS(param->arg_info->type)) {
  2515. return zend_string_copy(ZEND_TYPE_NAME(param->arg_info->type));
  2516. } else {
  2517. char *name = zend_get_type_by_const(ZEND_TYPE_CODE(param->arg_info->type));
  2518. return zend_string_init(name, strlen(name), 0);
  2519. }
  2520. }
  2521. /* }}} */
  2522. /* {{{ proto public string ReflectionType::__toString()
  2523. Return the text of the type hint */
  2524. ZEND_METHOD(reflection_type, __toString)
  2525. {
  2526. reflection_object *intern;
  2527. type_reference *param;
  2528. if (zend_parse_parameters_none() == FAILURE) {
  2529. return;
  2530. }
  2531. GET_REFLECTION_OBJECT_PTR(param);
  2532. RETURN_STR(reflection_type_name(param));
  2533. }
  2534. /* }}} */
  2535. /* {{{ proto public string ReflectionNamedType::getName()
  2536. Return the text of the type hint */
  2537. ZEND_METHOD(reflection_named_type, getName)
  2538. {
  2539. reflection_object *intern;
  2540. type_reference *param;
  2541. if (zend_parse_parameters_none() == FAILURE) {
  2542. return;
  2543. }
  2544. GET_REFLECTION_OBJECT_PTR(param);
  2545. RETURN_STR(reflection_type_name(param));
  2546. }
  2547. /* }}} */
  2548. /* {{{ proto public static mixed ReflectionMethod::export(mixed class, string name [, bool return]) throws ReflectionException
  2549. Exports a reflection object. Returns the output if TRUE is specified for return, printing it otherwise. */
  2550. ZEND_METHOD(reflection_method, export)
  2551. {
  2552. _reflection_export(INTERNAL_FUNCTION_PARAM_PASSTHRU, reflection_method_ptr, 2);
  2553. }
  2554. /* }}} */
  2555. /* {{{ proto public void ReflectionMethod::__construct(mixed class_or_method [, string name])
  2556. Constructor. Throws an Exception in case the given method does not exist */
  2557. ZEND_METHOD(reflection_method, __construct)
  2558. {
  2559. zval name, *classname;
  2560. zval *object, *orig_obj;
  2561. reflection_object *intern;
  2562. char *lcname;
  2563. zend_class_entry *ce;
  2564. zend_function *mptr;
  2565. char *name_str, *tmp;
  2566. size_t name_len, tmp_len;
  2567. zval ztmp;
  2568. if (zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET, ZEND_NUM_ARGS(), "zs", &classname, &name_str, &name_len) == FAILURE) {
  2569. if (zend_parse_parameters_throw(ZEND_NUM_ARGS(), "s", &name_str, &name_len) == FAILURE) {
  2570. return;
  2571. }
  2572. if ((tmp = strstr(name_str, "::")) == NULL) {
  2573. zend_throw_exception_ex(reflection_exception_ptr, 0,
  2574. "Invalid method name %s", name_str);
  2575. return;
  2576. }
  2577. classname = &ztmp;
  2578. tmp_len = tmp - name_str;
  2579. ZVAL_STRINGL(classname, name_str, tmp_len);
  2580. name_len = name_len - (tmp_len + 2);
  2581. name_str = tmp + 2;
  2582. orig_obj = NULL;
  2583. } else if (Z_TYPE_P(classname) == IS_OBJECT) {
  2584. orig_obj = classname;
  2585. } else {
  2586. orig_obj = NULL;
  2587. }
  2588. object = getThis();
  2589. intern = Z_REFLECTION_P(object);
  2590. /* Find the class entry */
  2591. switch (Z_TYPE_P(classname)) {
  2592. case IS_STRING:
  2593. if ((ce = zend_lookup_class(Z_STR_P(classname))) == NULL) {
  2594. if (!EG(exception)) {
  2595. zend_throw_exception_ex(reflection_exception_ptr, 0,
  2596. "Class %s does not exist", Z_STRVAL_P(classname));
  2597. }
  2598. if (classname == &ztmp) {
  2599. zval_ptr_dtor_str(&ztmp);
  2600. }
  2601. return;
  2602. }
  2603. break;
  2604. case IS_OBJECT:
  2605. ce = Z_OBJCE_P(classname);
  2606. break;
  2607. default:
  2608. if (classname == &ztmp) {
  2609. zval_ptr_dtor_str(&ztmp);
  2610. }
  2611. _DO_THROW("The parameter class is expected to be either a string or an object");
  2612. /* returns out of this function */
  2613. }
  2614. if (classname == &ztmp) {
  2615. zval_ptr_dtor_str(&ztmp);
  2616. }
  2617. lcname = zend_str_tolower_dup(name_str, name_len);
  2618. if (ce == zend_ce_closure && orig_obj && (name_len == sizeof(ZEND_INVOKE_FUNC_NAME)-1)
  2619. && memcmp(lcname, ZEND_INVOKE_FUNC_NAME, sizeof(ZEND_INVOKE_FUNC_NAME)-1) == 0
  2620. && (mptr = zend_get_closure_invoke_method(Z_OBJ_P(orig_obj))) != NULL)
  2621. {
  2622. /* do nothing, mptr already set */
  2623. } else if ((mptr = zend_hash_str_find_ptr(&ce->function_table, lcname, name_len)) == NULL) {
  2624. efree(lcname);
  2625. zend_throw_exception_ex(reflection_exception_ptr, 0,
  2626. "Method %s::%s() does not exist", ZSTR_VAL(ce->name), name_str);
  2627. return;
  2628. }
  2629. efree(lcname);
  2630. ZVAL_STR_COPY(&name, mptr->common.scope->name);
  2631. reflection_update_property_class(object, &name);
  2632. ZVAL_STR_COPY(&name, mptr->common.function_name);
  2633. reflection_update_property_name(object, &name);
  2634. intern->ptr = mptr;
  2635. intern->ref_type = REF_TYPE_FUNCTION;
  2636. intern->ce = ce;
  2637. }
  2638. /* }}} */
  2639. /* {{{ proto public string ReflectionMethod::__toString()
  2640. Returns a string representation */
  2641. ZEND_METHOD(reflection_method, __toString)
  2642. {
  2643. reflection_object *intern;
  2644. zend_function *mptr;
  2645. smart_str str = {0};
  2646. if (zend_parse_parameters_none() == FAILURE) {
  2647. return;
  2648. }
  2649. GET_REFLECTION_OBJECT_PTR(mptr);
  2650. _function_string(&str, mptr, intern->ce, "");
  2651. RETURN_STR(smart_str_extract(&str));
  2652. }
  2653. /* }}} */
  2654. /* {{{ proto public mixed ReflectionMethod::getClosure([mixed object])
  2655. Invokes the function */
  2656. ZEND_METHOD(reflection_method, getClosure)
  2657. {
  2658. reflection_object *intern;
  2659. zval *obj;
  2660. zend_function *mptr;
  2661. GET_REFLECTION_OBJECT_PTR(mptr);
  2662. if (mptr->common.fn_flags & ZEND_ACC_STATIC) {
  2663. zend_create_fake_closure(return_value, mptr, mptr->common.scope, mptr->common.scope, NULL);
  2664. } else {
  2665. if (zend_parse_parameters(ZEND_NUM_ARGS(), "o", &obj) == FAILURE) {
  2666. return;
  2667. }
  2668. if (!instanceof_function(Z_OBJCE_P(obj), mptr->common.scope)) {
  2669. _DO_THROW("Given object is not an instance of the class this method was declared in");
  2670. /* Returns from this function */
  2671. }
  2672. /* This is an original closure object and __invoke is to be called. */
  2673. if (Z_OBJCE_P(obj) == zend_ce_closure &&
  2674. (mptr->internal_function.fn_flags & ZEND_ACC_CALL_VIA_TRAMPOLINE))
  2675. {
  2676. ZVAL_COPY(return_value, obj);
  2677. } else {
  2678. zend_create_fake_closure(return_value, mptr, mptr->common.scope, Z_OBJCE_P(obj), obj);
  2679. }
  2680. }
  2681. }
  2682. /* }}} */
  2683. /* {{{ reflection_method_invoke */
  2684. static void reflection_method_invoke(INTERNAL_FUNCTION_PARAMETERS, int variadic)
  2685. {
  2686. zval retval;
  2687. zval *params = NULL, *val, *object;
  2688. reflection_object *intern;
  2689. zend_function *mptr;
  2690. int i, argc = 0, result;
  2691. zend_fcall_info fci;
  2692. zend_fcall_info_cache fcc;
  2693. zend_class_entry *obj_ce;
  2694. zval *param_array;
  2695. GET_REFLECTION_OBJECT_PTR(mptr);
  2696. if (mptr->common.fn_flags & ZEND_ACC_ABSTRACT) {
  2697. zend_throw_exception_ex(reflection_exception_ptr, 0,
  2698. "Trying to invoke abstract method %s::%s()",
  2699. ZSTR_VAL(mptr->common.scope->name), ZSTR_VAL(mptr->common.function_name));
  2700. return;
  2701. }
  2702. if (!(mptr->common.fn_flags & ZEND_ACC_PUBLIC) && intern->ignore_visibility == 0) {
  2703. zend_throw_exception_ex(reflection_exception_ptr, 0,
  2704. "Trying to invoke %s method %s::%s() from scope %s",
  2705. mptr->common.fn_flags & ZEND_ACC_PROTECTED ? "protected" : "private",
  2706. ZSTR_VAL(mptr->common.scope->name), ZSTR_VAL(mptr->common.function_name),
  2707. ZSTR_VAL(Z_OBJCE_P(getThis())->name));
  2708. return;
  2709. }
  2710. if (variadic) {
  2711. if (zend_parse_parameters(ZEND_NUM_ARGS(), "o!*", &object, &params, &argc) == FAILURE) {
  2712. return;
  2713. }
  2714. } else {
  2715. if (zend_parse_parameters(ZEND_NUM_ARGS(), "o!a", &object, &param_array) == FAILURE) {
  2716. return;
  2717. }
  2718. argc = zend_hash_num_elements(Z_ARRVAL_P(param_array));
  2719. params = safe_emalloc(sizeof(zval), argc, 0);
  2720. argc = 0;
  2721. ZEND_HASH_FOREACH_VAL(Z_ARRVAL_P(param_array), val) {
  2722. ZVAL_COPY(&params[argc], val);
  2723. argc++;
  2724. } ZEND_HASH_FOREACH_END();
  2725. }
  2726. /* In case this is a static method, we should'nt pass an object_ptr
  2727. * (which is used as calling context aka $this). We can thus ignore the
  2728. * first parameter.
  2729. *
  2730. * Else, we verify that the given object is an instance of the class.
  2731. */
  2732. if (mptr->common.fn_flags & ZEND_ACC_STATIC) {
  2733. object = NULL;
  2734. obj_ce = mptr->common.scope;
  2735. } else {
  2736. if (!object) {
  2737. zend_throw_exception_ex(reflection_exception_ptr, 0,
  2738. "Trying to invoke non static method %s::%s() without an object",
  2739. ZSTR_VAL(mptr->common.scope->name), ZSTR_VAL(mptr->common.function_name));
  2740. return;
  2741. }
  2742. obj_ce = Z_OBJCE_P(object);
  2743. if (!instanceof_function(obj_ce, mptr->common.scope)) {
  2744. if (!variadic) {
  2745. efree(params);
  2746. }
  2747. _DO_THROW("Given object is not an instance of the class this method was declared in");
  2748. /* Returns from this function */
  2749. }
  2750. }
  2751. fci.size = sizeof(fci);
  2752. ZVAL_UNDEF(&fci.function_name);
  2753. fci.object = object ? Z_OBJ_P(object) : NULL;
  2754. fci.retval = &retval;
  2755. fci.param_count = argc;
  2756. fci.params = params;
  2757. fci.no_separation = 1;
  2758. fcc.function_handler = mptr;
  2759. fcc.called_scope = intern->ce;
  2760. fcc.object = object ? Z_OBJ_P(object) : NULL;
  2761. /*
  2762. * Copy the zend_function when calling via handler (e.g. Closure::__invoke())
  2763. */
  2764. if ((mptr->internal_function.fn_flags & ZEND_ACC_CALL_VIA_TRAMPOLINE)) {
  2765. fcc.function_handler = _copy_function(mptr);
  2766. }
  2767. result = zend_call_function(&fci, &fcc);
  2768. if (!variadic) {
  2769. for (i = 0; i < argc; i++) {
  2770. zval_ptr_dtor(&params[i]);
  2771. }
  2772. efree(params);
  2773. }
  2774. if (result == FAILURE) {
  2775. zend_throw_exception_ex(reflection_exception_ptr, 0,
  2776. "Invocation of method %s::%s() failed", ZSTR_VAL(mptr->common.scope->name), ZSTR_VAL(mptr->common.function_name));
  2777. return;
  2778. }
  2779. if (Z_TYPE(retval) != IS_UNDEF) {
  2780. if (Z_ISREF(retval)) {
  2781. zend_unwrap_reference(&retval);
  2782. }
  2783. ZVAL_COPY_VALUE(return_value, &retval);
  2784. }
  2785. }
  2786. /* }}} */
  2787. /* {{{ proto public mixed ReflectionMethod::invoke(mixed object, mixed* args)
  2788. Invokes the method. */
  2789. ZEND_METHOD(reflection_method, invoke)
  2790. {
  2791. reflection_method_invoke(INTERNAL_FUNCTION_PARAM_PASSTHRU, 1);
  2792. }
  2793. /* }}} */
  2794. /* {{{ proto public mixed ReflectionMethod::invokeArgs(mixed object, array args)
  2795. Invokes the function and pass its arguments as array. */
  2796. ZEND_METHOD(reflection_method, invokeArgs)
  2797. {
  2798. reflection_method_invoke(INTERNAL_FUNCTION_PARAM_PASSTHRU, 0);
  2799. }
  2800. /* }}} */
  2801. /* {{{ proto public bool ReflectionMethod::isFinal()
  2802. Returns whether this method is final */
  2803. ZEND_METHOD(reflection_method, isFinal)
  2804. {
  2805. _function_check_flag(INTERNAL_FUNCTION_PARAM_PASSTHRU, ZEND_ACC_FINAL);
  2806. }
  2807. /* }}} */
  2808. /* {{{ proto public bool ReflectionMethod::isAbstract()
  2809. Returns whether this method is abstract */
  2810. ZEND_METHOD(reflection_method, isAbstract)
  2811. {
  2812. _function_check_flag(INTERNAL_FUNCTION_PARAM_PASSTHRU, ZEND_ACC_ABSTRACT);
  2813. }
  2814. /* }}} */
  2815. /* {{{ proto public bool ReflectionMethod::isPublic()
  2816. Returns whether this method is public */
  2817. ZEND_METHOD(reflection_method, isPublic)
  2818. {
  2819. _function_check_flag(INTERNAL_FUNCTION_PARAM_PASSTHRU, ZEND_ACC_PUBLIC);
  2820. }
  2821. /* }}} */
  2822. /* {{{ proto public bool ReflectionMethod::isPrivate()
  2823. Returns whether this method is private */
  2824. ZEND_METHOD(reflection_method, isPrivate)
  2825. {
  2826. _function_check_flag(INTERNAL_FUNCTION_PARAM_PASSTHRU, ZEND_ACC_PRIVATE);
  2827. }
  2828. /* }}} */
  2829. /* {{{ proto public bool ReflectionMethod::isProtected()
  2830. Returns whether this method is protected */
  2831. ZEND_METHOD(reflection_method, isProtected)
  2832. {
  2833. _function_check_flag(INTERNAL_FUNCTION_PARAM_PASSTHRU, ZEND_ACC_PROTECTED);
  2834. }
  2835. /* }}} */
  2836. /* {{{ proto public bool ReflectionMethod::isStatic()
  2837. Returns whether this method is static */
  2838. ZEND_METHOD(reflection_method, isStatic)
  2839. {
  2840. _function_check_flag(INTERNAL_FUNCTION_PARAM_PASSTHRU, ZEND_ACC_STATIC);
  2841. }
  2842. /* }}} */
  2843. /* {{{ proto public bool ReflectionFunction::isDeprecated()
  2844. Returns whether this function is deprecated */
  2845. ZEND_METHOD(reflection_function, isDeprecated)
  2846. {
  2847. _function_check_flag(INTERNAL_FUNCTION_PARAM_PASSTHRU, ZEND_ACC_DEPRECATED);
  2848. }
  2849. /* }}} */
  2850. /* {{{ proto public bool ReflectionFunction::isGenerator()
  2851. Returns whether this function is a generator */
  2852. ZEND_METHOD(reflection_function, isGenerator)
  2853. {
  2854. _function_check_flag(INTERNAL_FUNCTION_PARAM_PASSTHRU, ZEND_ACC_GENERATOR);
  2855. }
  2856. /* }}} */
  2857. /* {{{ proto public bool ReflectionFunction::isVariadic()
  2858. Returns whether this function is variadic */
  2859. ZEND_METHOD(reflection_function, isVariadic)
  2860. {
  2861. _function_check_flag(INTERNAL_FUNCTION_PARAM_PASSTHRU, ZEND_ACC_VARIADIC);
  2862. }
  2863. /* }}} */
  2864. /* {{{ proto public bool ReflectionFunction::inNamespace()
  2865. Returns whether this function is defined in namespace */
  2866. ZEND_METHOD(reflection_function, inNamespace)
  2867. {
  2868. zval *name;
  2869. const char *backslash;
  2870. if (zend_parse_parameters_none() == FAILURE) {
  2871. return;
  2872. }
  2873. if ((name = _default_load_name(getThis())) == NULL) {
  2874. RETURN_FALSE;
  2875. }
  2876. if (Z_TYPE_P(name) == IS_STRING
  2877. && (backslash = zend_memrchr(Z_STRVAL_P(name), '\\', Z_STRLEN_P(name)))
  2878. && backslash > Z_STRVAL_P(name))
  2879. {
  2880. RETURN_TRUE;
  2881. }
  2882. RETURN_FALSE;
  2883. }
  2884. /* }}} */
  2885. /* {{{ proto public string ReflectionFunction::getNamespaceName()
  2886. Returns the name of namespace where this function is defined */
  2887. ZEND_METHOD(reflection_function, getNamespaceName)
  2888. {
  2889. zval *name;
  2890. const char *backslash;
  2891. if (zend_parse_parameters_none() == FAILURE) {
  2892. return;
  2893. }
  2894. if ((name = _default_load_name(getThis())) == NULL) {
  2895. RETURN_FALSE;
  2896. }
  2897. if (Z_TYPE_P(name) == IS_STRING
  2898. && (backslash = zend_memrchr(Z_STRVAL_P(name), '\\', Z_STRLEN_P(name)))
  2899. && backslash > Z_STRVAL_P(name))
  2900. {
  2901. RETURN_STRINGL(Z_STRVAL_P(name), backslash - Z_STRVAL_P(name));
  2902. }
  2903. RETURN_EMPTY_STRING();
  2904. }
  2905. /* }}} */
  2906. /* {{{ proto public string ReflectionFunction::getShortName()
  2907. Returns the short name of the function (without namespace part) */
  2908. ZEND_METHOD(reflection_function, getShortName)
  2909. {
  2910. zval *name;
  2911. const char *backslash;
  2912. if (zend_parse_parameters_none() == FAILURE) {
  2913. return;
  2914. }
  2915. if ((name = _default_load_name(getThis())) == NULL) {
  2916. RETURN_FALSE;
  2917. }
  2918. if (Z_TYPE_P(name) == IS_STRING
  2919. && (backslash = zend_memrchr(Z_STRVAL_P(name), '\\', Z_STRLEN_P(name)))
  2920. && backslash > Z_STRVAL_P(name))
  2921. {
  2922. RETURN_STRINGL(backslash + 1, Z_STRLEN_P(name) - (backslash - Z_STRVAL_P(name) + 1));
  2923. }
  2924. ZVAL_COPY_DEREF(return_value, name);
  2925. }
  2926. /* }}} */
  2927. /* {{{ proto public bool ReflectionFunctionAbstract:hasReturnType()
  2928. Return whether the function has a return type */
  2929. ZEND_METHOD(reflection_function, hasReturnType)
  2930. {
  2931. reflection_object *intern;
  2932. zend_function *fptr;
  2933. if (zend_parse_parameters_none() == FAILURE) {
  2934. return;
  2935. }
  2936. GET_REFLECTION_OBJECT_PTR(fptr);
  2937. RETVAL_BOOL(fptr->op_array.fn_flags & ZEND_ACC_HAS_RETURN_TYPE);
  2938. }
  2939. /* }}} */
  2940. /* {{{ proto public string ReflectionFunctionAbstract::getReturnType()
  2941. Returns the return type associated with the function */
  2942. ZEND_METHOD(reflection_function, getReturnType)
  2943. {
  2944. reflection_object *intern;
  2945. zend_function *fptr;
  2946. if (zend_parse_parameters_none() == FAILURE) {
  2947. return;
  2948. }
  2949. GET_REFLECTION_OBJECT_PTR(fptr);
  2950. if (!(fptr->op_array.fn_flags & ZEND_ACC_HAS_RETURN_TYPE)) {
  2951. RETURN_NULL();
  2952. }
  2953. reflection_type_factory(_copy_function(fptr), Z_ISUNDEF(intern->obj)? NULL : &intern->obj, &fptr->common.arg_info[-1], return_value);
  2954. }
  2955. /* }}} */
  2956. /* {{{ proto public bool ReflectionMethod::isConstructor()
  2957. Returns whether this method is the constructor */
  2958. ZEND_METHOD(reflection_method, isConstructor)
  2959. {
  2960. reflection_object *intern;
  2961. zend_function *mptr;
  2962. if (zend_parse_parameters_none() == FAILURE) {
  2963. return;
  2964. }
  2965. GET_REFLECTION_OBJECT_PTR(mptr);
  2966. /* we need to check if the ctor is the ctor of the class level we we
  2967. * looking at since we might be looking at an inherited old style ctor
  2968. * defined in base class. */
  2969. RETURN_BOOL(mptr->common.fn_flags & ZEND_ACC_CTOR && intern->ce->constructor && intern->ce->constructor->common.scope == mptr->common.scope);
  2970. }
  2971. /* }}} */
  2972. /* {{{ proto public bool ReflectionMethod::isDestructor()
  2973. Returns whether this method is static */
  2974. ZEND_METHOD(reflection_method, isDestructor)
  2975. {
  2976. reflection_object *intern;
  2977. zend_function *mptr;
  2978. if (zend_parse_parameters_none() == FAILURE) {
  2979. return;
  2980. }
  2981. GET_REFLECTION_OBJECT_PTR(mptr);
  2982. RETURN_BOOL(mptr->common.fn_flags & ZEND_ACC_DTOR);
  2983. }
  2984. /* }}} */
  2985. /* {{{ proto public int ReflectionMethod::getModifiers()
  2986. Returns a bitfield of the access modifiers for this method */
  2987. ZEND_METHOD(reflection_method, getModifiers)
  2988. {
  2989. reflection_object *intern;
  2990. zend_function *mptr;
  2991. uint32_t keep_flags = ZEND_ACC_PPP_MASK | ZEND_ACC_IMPLICIT_PUBLIC
  2992. | ZEND_ACC_STATIC | ZEND_ACC_ABSTRACT | ZEND_ACC_FINAL;
  2993. if (zend_parse_parameters_none() == FAILURE) {
  2994. return;
  2995. }
  2996. GET_REFLECTION_OBJECT_PTR(mptr);
  2997. RETURN_LONG((mptr->common.fn_flags & keep_flags));
  2998. }
  2999. /* }}} */
  3000. /* {{{ proto public ReflectionClass ReflectionMethod::getDeclaringClass()
  3001. Get the declaring class */
  3002. ZEND_METHOD(reflection_method, getDeclaringClass)
  3003. {
  3004. reflection_object *intern;
  3005. zend_function *mptr;
  3006. GET_REFLECTION_OBJECT_PTR(mptr);
  3007. if (zend_parse_parameters_none() == FAILURE) {
  3008. return;
  3009. }
  3010. zend_reflection_class_factory(mptr->common.scope, return_value);
  3011. }
  3012. /* }}} */
  3013. /* {{{ proto public ReflectionClass ReflectionMethod::getPrototype()
  3014. Get the prototype */
  3015. ZEND_METHOD(reflection_method, getPrototype)
  3016. {
  3017. reflection_object *intern;
  3018. zend_function *mptr;
  3019. GET_REFLECTION_OBJECT_PTR(mptr);
  3020. if (zend_parse_parameters_none() == FAILURE) {
  3021. return;
  3022. }
  3023. if (!mptr->common.prototype) {
  3024. zend_throw_exception_ex(reflection_exception_ptr, 0,
  3025. "Method %s::%s does not have a prototype", ZSTR_VAL(intern->ce->name), ZSTR_VAL(mptr->common.function_name));
  3026. return;
  3027. }
  3028. reflection_method_factory(mptr->common.prototype->common.scope, mptr->common.prototype, NULL, return_value);
  3029. }
  3030. /* }}} */
  3031. /* {{{ proto public void ReflectionMethod::setAccessible(bool visible)
  3032. Sets whether non-public methods can be invoked */
  3033. ZEND_METHOD(reflection_method, setAccessible)
  3034. {
  3035. reflection_object *intern;
  3036. zend_bool visible;
  3037. if (zend_parse_parameters(ZEND_NUM_ARGS(), "b", &visible) == FAILURE) {
  3038. return;
  3039. }
  3040. intern = Z_REFLECTION_P(getThis());
  3041. intern->ignore_visibility = visible;
  3042. }
  3043. /* }}} */
  3044. /* {{{ proto public void ReflectionClassConstant::__construct(mixed class, string name)
  3045. Constructor. Throws an Exception in case the given class constant does not exist */
  3046. ZEND_METHOD(reflection_class_constant, __construct)
  3047. {
  3048. zval *classname, *object, name, cname;
  3049. zend_string *constname;
  3050. reflection_object *intern;
  3051. zend_class_entry *ce;
  3052. zend_class_constant *constant = NULL;
  3053. if (zend_parse_parameters_throw(ZEND_NUM_ARGS(), "zS", &classname, &constname) == FAILURE) {
  3054. return;
  3055. }
  3056. object = getThis();
  3057. intern = Z_REFLECTION_P(object);
  3058. /* Find the class entry */
  3059. switch (Z_TYPE_P(classname)) {
  3060. case IS_STRING:
  3061. if ((ce = zend_lookup_class(Z_STR_P(classname))) == NULL) {
  3062. zend_throw_exception_ex(reflection_exception_ptr, 0,
  3063. "Class %s does not exist", Z_STRVAL_P(classname));
  3064. return;
  3065. }
  3066. break;
  3067. case IS_OBJECT:
  3068. ce = Z_OBJCE_P(classname);
  3069. break;
  3070. default:
  3071. _DO_THROW("The parameter class is expected to be either a string or an object");
  3072. /* returns out of this function */
  3073. }
  3074. if ((constant = zend_hash_find_ptr(&ce->constants_table, constname)) == NULL) {
  3075. zend_throw_exception_ex(reflection_exception_ptr, 0, "Class Constant %s::%s does not exist", ZSTR_VAL(ce->name), ZSTR_VAL(constname));
  3076. return;
  3077. }
  3078. ZVAL_STR_COPY(&name, constname);
  3079. ZVAL_STR_COPY(&cname, ce->name);
  3080. intern->ptr = constant;
  3081. intern->ref_type = REF_TYPE_CLASS_CONSTANT;
  3082. intern->ce = constant->ce;
  3083. intern->ignore_visibility = 0;
  3084. reflection_update_property_name(object, &name);
  3085. reflection_update_property_class(object, &cname);
  3086. }
  3087. /* }}} */
  3088. /* {{{ proto public string ReflectionClassConstant::__toString()
  3089. Returns a string representation */
  3090. ZEND_METHOD(reflection_class_constant, __toString)
  3091. {
  3092. reflection_object *intern;
  3093. zend_class_constant *ref;
  3094. smart_str str = {0};
  3095. zval name;
  3096. if (zend_parse_parameters_none() == FAILURE) {
  3097. return;
  3098. }
  3099. GET_REFLECTION_OBJECT_PTR(ref);
  3100. _default_get_name(getThis(), &name);
  3101. _class_const_string(&str, Z_STRVAL(name), ref, "");
  3102. zval_ptr_dtor(&name);
  3103. RETURN_STR(smart_str_extract(&str));
  3104. }
  3105. /* }}} */
  3106. /* {{{ proto public string ReflectionClassConstant::getName()
  3107. Returns the constant' name */
  3108. ZEND_METHOD(reflection_class_constant, getName)
  3109. {
  3110. if (zend_parse_parameters_none() == FAILURE) {
  3111. return;
  3112. }
  3113. _default_get_name(getThis(), return_value);
  3114. }
  3115. /* }}} */
  3116. static void _class_constant_check_flag(INTERNAL_FUNCTION_PARAMETERS, int mask) /* {{{ */
  3117. {
  3118. reflection_object *intern;
  3119. zend_class_constant *ref;
  3120. if (zend_parse_parameters_none() == FAILURE) {
  3121. return;
  3122. }
  3123. GET_REFLECTION_OBJECT_PTR(ref);
  3124. RETURN_BOOL(Z_ACCESS_FLAGS(ref->value) & mask);
  3125. }
  3126. /* }}} */
  3127. /* {{{ proto public bool ReflectionClassConstant::isPublic()
  3128. Returns whether this constant is public */
  3129. ZEND_METHOD(reflection_class_constant, isPublic)
  3130. {
  3131. _class_constant_check_flag(INTERNAL_FUNCTION_PARAM_PASSTHRU, ZEND_ACC_PUBLIC | ZEND_ACC_IMPLICIT_PUBLIC);
  3132. }
  3133. /* }}} */
  3134. /* {{{ proto public bool ReflectionClassConstant::isPrivate()
  3135. Returns whether this constant is private */
  3136. ZEND_METHOD(reflection_class_constant, isPrivate)
  3137. {
  3138. _class_constant_check_flag(INTERNAL_FUNCTION_PARAM_PASSTHRU, ZEND_ACC_PRIVATE);
  3139. }
  3140. /* }}} */
  3141. /* {{{ proto public bool ReflectionClassConstant::isProtected()
  3142. Returns whether this constant is protected */
  3143. ZEND_METHOD(reflection_class_constant, isProtected)
  3144. {
  3145. _class_constant_check_flag(INTERNAL_FUNCTION_PARAM_PASSTHRU, ZEND_ACC_PROTECTED);
  3146. }
  3147. /* }}} */
  3148. /* {{{ proto public int ReflectionClassConstant::getModifiers()
  3149. Returns a bitfield of the access modifiers for this constant */
  3150. ZEND_METHOD(reflection_class_constant, getModifiers)
  3151. {
  3152. reflection_object *intern;
  3153. zend_class_constant *ref;
  3154. if (zend_parse_parameters_none() == FAILURE) {
  3155. return;
  3156. }
  3157. GET_REFLECTION_OBJECT_PTR(ref);
  3158. RETURN_LONG(Z_ACCESS_FLAGS(ref->value));
  3159. }
  3160. /* }}} */
  3161. /* {{{ proto public mixed ReflectionClassConstant::getValue()
  3162. Returns this constant's value */
  3163. ZEND_METHOD(reflection_class_constant, getValue)
  3164. {
  3165. reflection_object *intern;
  3166. zend_class_constant *ref;
  3167. if (zend_parse_parameters_none() == FAILURE) {
  3168. return;
  3169. }
  3170. GET_REFLECTION_OBJECT_PTR(ref);
  3171. ZVAL_COPY_OR_DUP(return_value, &ref->value);
  3172. if (Z_TYPE_P(return_value) == IS_CONSTANT_AST) {
  3173. zval_update_constant_ex(return_value, ref->ce);
  3174. }
  3175. }
  3176. /* }}} */
  3177. /* {{{ proto public ReflectionClass ReflectionClassConstant::getDeclaringClass()
  3178. Get the declaring class */
  3179. ZEND_METHOD(reflection_class_constant, getDeclaringClass)
  3180. {
  3181. reflection_object *intern;
  3182. zend_class_constant *ref;
  3183. if (zend_parse_parameters_none() == FAILURE) {
  3184. return;
  3185. }
  3186. GET_REFLECTION_OBJECT_PTR(ref);
  3187. zend_reflection_class_factory(ref->ce, return_value);
  3188. }
  3189. /* }}} */
  3190. /* {{{ proto public string ReflectionClassConstant::getDocComment()
  3191. Returns the doc comment for this constant */
  3192. ZEND_METHOD(reflection_class_constant, getDocComment)
  3193. {
  3194. reflection_object *intern;
  3195. zend_class_constant *ref;
  3196. if (zend_parse_parameters_none() == FAILURE) {
  3197. return;
  3198. }
  3199. GET_REFLECTION_OBJECT_PTR(ref);
  3200. if (ref->doc_comment) {
  3201. RETURN_STR_COPY(ref->doc_comment);
  3202. }
  3203. RETURN_FALSE;
  3204. }
  3205. /* }}} */
  3206. /* {{{ proto public static mixed ReflectionClass::export(mixed argument [, bool return]) throws ReflectionException
  3207. Exports a reflection object. Returns the output if TRUE is specified for return, printing it otherwise. */
  3208. ZEND_METHOD(reflection_class, export)
  3209. {
  3210. _reflection_export(INTERNAL_FUNCTION_PARAM_PASSTHRU, reflection_class_ptr, 1);
  3211. }
  3212. /* }}} */
  3213. /* {{{ reflection_class_object_ctor */
  3214. static void reflection_class_object_ctor(INTERNAL_FUNCTION_PARAMETERS, int is_object)
  3215. {
  3216. zval *argument;
  3217. zval *object;
  3218. zval classname;
  3219. reflection_object *intern;
  3220. zend_class_entry *ce;
  3221. if (is_object) {
  3222. if (zend_parse_parameters(ZEND_NUM_ARGS(), "o", &argument) == FAILURE) {
  3223. return;
  3224. }
  3225. } else {
  3226. if (zend_parse_parameters(ZEND_NUM_ARGS(), "z", &argument) == FAILURE) {
  3227. return;
  3228. }
  3229. }
  3230. object = getThis();
  3231. intern = Z_REFLECTION_P(object);
  3232. if (Z_TYPE_P(argument) == IS_OBJECT) {
  3233. ZVAL_STR_COPY(&classname, Z_OBJCE_P(argument)->name);
  3234. reflection_update_property_name(object, &classname);
  3235. intern->ptr = Z_OBJCE_P(argument);
  3236. if (is_object) {
  3237. ZVAL_COPY(&intern->obj, argument);
  3238. }
  3239. } else {
  3240. convert_to_string_ex(argument);
  3241. if ((ce = zend_lookup_class(Z_STR_P(argument))) == NULL) {
  3242. if (!EG(exception)) {
  3243. zend_throw_exception_ex(reflection_exception_ptr, -1, "Class %s does not exist", Z_STRVAL_P(argument));
  3244. }
  3245. return;
  3246. }
  3247. ZVAL_STR_COPY(&classname, ce->name);
  3248. reflection_update_property_name(object, &classname);
  3249. intern->ptr = ce;
  3250. }
  3251. intern->ref_type = REF_TYPE_OTHER;
  3252. }
  3253. /* }}} */
  3254. /* {{{ proto public void ReflectionClass::__construct(mixed argument) throws ReflectionException
  3255. Constructor. Takes a string or an instance as an argument */
  3256. ZEND_METHOD(reflection_class, __construct)
  3257. {
  3258. reflection_class_object_ctor(INTERNAL_FUNCTION_PARAM_PASSTHRU, 0);
  3259. }
  3260. /* }}} */
  3261. /* {{{ add_class_vars */
  3262. static void add_class_vars(zend_class_entry *ce, int statics, zval *return_value)
  3263. {
  3264. zend_property_info *prop_info;
  3265. zval *prop, prop_copy;
  3266. zend_string *key;
  3267. ZEND_HASH_FOREACH_STR_KEY_PTR(&ce->properties_info, key, prop_info) {
  3268. if (((prop_info->flags & ZEND_ACC_SHADOW) &&
  3269. prop_info->ce != ce) ||
  3270. ((prop_info->flags & ZEND_ACC_PROTECTED) &&
  3271. !zend_check_protected(prop_info->ce, ce)) ||
  3272. ((prop_info->flags & ZEND_ACC_PRIVATE) &&
  3273. prop_info->ce != ce)) {
  3274. continue;
  3275. }
  3276. prop = NULL;
  3277. if (statics && (prop_info->flags & ZEND_ACC_STATIC) != 0) {
  3278. prop = &ce->default_static_members_table[prop_info->offset];
  3279. ZVAL_DEINDIRECT(prop);
  3280. } else if (!statics && (prop_info->flags & ZEND_ACC_STATIC) == 0) {
  3281. prop = &ce->default_properties_table[OBJ_PROP_TO_NUM(prop_info->offset)];
  3282. }
  3283. if (!prop) {
  3284. continue;
  3285. }
  3286. /* copy: enforce read only access */
  3287. ZVAL_DEREF(prop);
  3288. ZVAL_COPY_OR_DUP(&prop_copy, prop);
  3289. /* this is necessary to make it able to work with default array
  3290. * properties, returned to user */
  3291. if (Z_TYPE(prop_copy) == IS_CONSTANT_AST) {
  3292. if (UNEXPECTED(zval_update_constant_ex(&prop_copy, NULL) != SUCCESS)) {
  3293. return;
  3294. }
  3295. }
  3296. zend_hash_update(Z_ARRVAL_P(return_value), key, &prop_copy);
  3297. } ZEND_HASH_FOREACH_END();
  3298. }
  3299. /* }}} */
  3300. /* {{{ proto public array ReflectionClass::getStaticProperties()
  3301. Returns an associative array containing all static property values of the class */
  3302. ZEND_METHOD(reflection_class, getStaticProperties)
  3303. {
  3304. reflection_object *intern;
  3305. zend_class_entry *ce;
  3306. if (zend_parse_parameters_none() == FAILURE) {
  3307. return;
  3308. }
  3309. GET_REFLECTION_OBJECT_PTR(ce);
  3310. if (UNEXPECTED(zend_update_class_constants(ce) != SUCCESS)) {
  3311. return;
  3312. }
  3313. array_init(return_value);
  3314. add_class_vars(ce, 1, return_value);
  3315. }
  3316. /* }}} */
  3317. /* {{{ proto public mixed ReflectionClass::getStaticPropertyValue(string name [, mixed default])
  3318. Returns the value of a static property */
  3319. ZEND_METHOD(reflection_class, getStaticPropertyValue)
  3320. {
  3321. reflection_object *intern;
  3322. zend_class_entry *ce;
  3323. zend_string *name;
  3324. zval *prop, *def_value = NULL;
  3325. if (zend_parse_parameters(ZEND_NUM_ARGS(), "S|z", &name, &def_value) == FAILURE) {
  3326. return;
  3327. }
  3328. GET_REFLECTION_OBJECT_PTR(ce);
  3329. if (UNEXPECTED(zend_update_class_constants(ce) != SUCCESS)) {
  3330. return;
  3331. }
  3332. prop = zend_std_get_static_property(ce, name, 1);
  3333. if (!prop) {
  3334. if (def_value) {
  3335. ZVAL_COPY(return_value, def_value);
  3336. } else {
  3337. zend_throw_exception_ex(reflection_exception_ptr, 0,
  3338. "Class %s does not have a property named %s", ZSTR_VAL(ce->name), ZSTR_VAL(name));
  3339. }
  3340. return;
  3341. } else {
  3342. ZVAL_COPY_DEREF(return_value, prop);
  3343. }
  3344. }
  3345. /* }}} */
  3346. /* {{{ proto public void ReflectionClass::setStaticPropertyValue(string $name, mixed $value)
  3347. Sets the value of a static property */
  3348. ZEND_METHOD(reflection_class, setStaticPropertyValue)
  3349. {
  3350. reflection_object *intern;
  3351. zend_class_entry *ce;
  3352. zend_string *name;
  3353. zval *variable_ptr, *value;
  3354. if (zend_parse_parameters(ZEND_NUM_ARGS(), "Sz", &name, &value) == FAILURE) {
  3355. return;
  3356. }
  3357. GET_REFLECTION_OBJECT_PTR(ce);
  3358. if (UNEXPECTED(zend_update_class_constants(ce) != SUCCESS)) {
  3359. return;
  3360. }
  3361. variable_ptr = zend_std_get_static_property(ce, name, 1);
  3362. if (!variable_ptr) {
  3363. zend_throw_exception_ex(reflection_exception_ptr, 0,
  3364. "Class %s does not have a property named %s", ZSTR_VAL(ce->name), ZSTR_VAL(name));
  3365. return;
  3366. }
  3367. ZVAL_DEREF(variable_ptr);
  3368. zval_ptr_dtor(variable_ptr);
  3369. ZVAL_COPY(variable_ptr, value);
  3370. }
  3371. /* }}} */
  3372. /* {{{ proto public array ReflectionClass::getDefaultProperties()
  3373. Returns an associative array containing copies of all default property values of the class */
  3374. ZEND_METHOD(reflection_class, getDefaultProperties)
  3375. {
  3376. reflection_object *intern;
  3377. zend_class_entry *ce;
  3378. if (zend_parse_parameters_none() == FAILURE) {
  3379. return;
  3380. }
  3381. GET_REFLECTION_OBJECT_PTR(ce);
  3382. array_init(return_value);
  3383. if (UNEXPECTED(zend_update_class_constants(ce) != SUCCESS)) {
  3384. return;
  3385. }
  3386. add_class_vars(ce, 1, return_value);
  3387. add_class_vars(ce, 0, return_value);
  3388. }
  3389. /* }}} */
  3390. /* {{{ proto public string ReflectionClass::__toString()
  3391. Returns a string representation */
  3392. ZEND_METHOD(reflection_class, __toString)
  3393. {
  3394. reflection_object *intern;
  3395. zend_class_entry *ce;
  3396. smart_str str = {0};
  3397. if (zend_parse_parameters_none() == FAILURE) {
  3398. return;
  3399. }
  3400. GET_REFLECTION_OBJECT_PTR(ce);
  3401. _class_string(&str, ce, &intern->obj, "");
  3402. RETURN_STR(smart_str_extract(&str));
  3403. }
  3404. /* }}} */
  3405. /* {{{ proto public string ReflectionClass::getName()
  3406. Returns the class' name */
  3407. ZEND_METHOD(reflection_class, getName)
  3408. {
  3409. if (zend_parse_parameters_none() == FAILURE) {
  3410. return;
  3411. }
  3412. _default_get_name(getThis(), return_value);
  3413. }
  3414. /* }}} */
  3415. /* {{{ proto public bool ReflectionClass::isInternal()
  3416. Returns whether this class is an internal class */
  3417. ZEND_METHOD(reflection_class, isInternal)
  3418. {
  3419. reflection_object *intern;
  3420. zend_class_entry *ce;
  3421. if (zend_parse_parameters_none() == FAILURE) {
  3422. return;
  3423. }
  3424. GET_REFLECTION_OBJECT_PTR(ce);
  3425. RETURN_BOOL(ce->type == ZEND_INTERNAL_CLASS);
  3426. }
  3427. /* }}} */
  3428. /* {{{ proto public bool ReflectionClass::isUserDefined()
  3429. Returns whether this class is user-defined */
  3430. ZEND_METHOD(reflection_class, isUserDefined)
  3431. {
  3432. reflection_object *intern;
  3433. zend_class_entry *ce;
  3434. if (zend_parse_parameters_none() == FAILURE) {
  3435. return;
  3436. }
  3437. GET_REFLECTION_OBJECT_PTR(ce);
  3438. RETURN_BOOL(ce->type == ZEND_USER_CLASS);
  3439. }
  3440. /* }}} */
  3441. /* {{{ proto public bool ReflectionClass::isAnonymous()
  3442. Returns whether this class is anonymous */
  3443. ZEND_METHOD(reflection_class, isAnonymous)
  3444. {
  3445. reflection_object *intern;
  3446. zend_class_entry *ce;
  3447. if (zend_parse_parameters_none() == FAILURE) {
  3448. return;
  3449. }
  3450. GET_REFLECTION_OBJECT_PTR(ce);
  3451. RETURN_BOOL(ce->ce_flags & ZEND_ACC_ANON_CLASS);
  3452. }
  3453. /* }}} */
  3454. /* {{{ proto public string ReflectionClass::getFileName()
  3455. Returns the filename of the file this class was declared in */
  3456. ZEND_METHOD(reflection_class, getFileName)
  3457. {
  3458. reflection_object *intern;
  3459. zend_class_entry *ce;
  3460. if (zend_parse_parameters_none() == FAILURE) {
  3461. return;
  3462. }
  3463. GET_REFLECTION_OBJECT_PTR(ce);
  3464. if (ce->type == ZEND_USER_CLASS) {
  3465. RETURN_STR_COPY(ce->info.user.filename);
  3466. }
  3467. RETURN_FALSE;
  3468. }
  3469. /* }}} */
  3470. /* {{{ proto public int ReflectionClass::getStartLine()
  3471. Returns the line this class' declaration starts at */
  3472. ZEND_METHOD(reflection_class, getStartLine)
  3473. {
  3474. reflection_object *intern;
  3475. zend_class_entry *ce;
  3476. if (zend_parse_parameters_none() == FAILURE) {
  3477. return;
  3478. }
  3479. GET_REFLECTION_OBJECT_PTR(ce);
  3480. if (ce->type == ZEND_USER_CLASS) {
  3481. RETURN_LONG(ce->info.user.line_start);
  3482. }
  3483. RETURN_FALSE;
  3484. }
  3485. /* }}} */
  3486. /* {{{ proto public int ReflectionClass::getEndLine()
  3487. Returns the line this class' declaration ends at */
  3488. ZEND_METHOD(reflection_class, getEndLine)
  3489. {
  3490. reflection_object *intern;
  3491. zend_class_entry *ce;
  3492. if (zend_parse_parameters_none() == FAILURE) {
  3493. return;
  3494. }
  3495. GET_REFLECTION_OBJECT_PTR(ce);
  3496. if (ce->type == ZEND_USER_CLASS) {
  3497. RETURN_LONG(ce->info.user.line_end);
  3498. }
  3499. RETURN_FALSE;
  3500. }
  3501. /* }}} */
  3502. /* {{{ proto public string ReflectionClass::getDocComment()
  3503. Returns the doc comment for this class */
  3504. ZEND_METHOD(reflection_class, getDocComment)
  3505. {
  3506. reflection_object *intern;
  3507. zend_class_entry *ce;
  3508. if (zend_parse_parameters_none() == FAILURE) {
  3509. return;
  3510. }
  3511. GET_REFLECTION_OBJECT_PTR(ce);
  3512. if (ce->type == ZEND_USER_CLASS && ce->info.user.doc_comment) {
  3513. RETURN_STR_COPY(ce->info.user.doc_comment);
  3514. }
  3515. RETURN_FALSE;
  3516. }
  3517. /* }}} */
  3518. /* {{{ proto public ReflectionMethod ReflectionClass::getConstructor()
  3519. Returns the class' constructor if there is one, NULL otherwise */
  3520. ZEND_METHOD(reflection_class, getConstructor)
  3521. {
  3522. reflection_object *intern;
  3523. zend_class_entry *ce;
  3524. if (zend_parse_parameters_none() == FAILURE) {
  3525. return;
  3526. }
  3527. GET_REFLECTION_OBJECT_PTR(ce);
  3528. if (ce->constructor) {
  3529. reflection_method_factory(ce, ce->constructor, NULL, return_value);
  3530. } else {
  3531. RETURN_NULL();
  3532. }
  3533. }
  3534. /* }}} */
  3535. /* {{{ proto public bool ReflectionClass::hasMethod(string name)
  3536. Returns whether a method exists or not */
  3537. ZEND_METHOD(reflection_class, hasMethod)
  3538. {
  3539. reflection_object *intern;
  3540. zend_class_entry *ce;
  3541. char *name, *lc_name;
  3542. size_t name_len;
  3543. if (zend_parse_parameters(ZEND_NUM_ARGS(), "s", &name, &name_len) == FAILURE) {
  3544. return;
  3545. }
  3546. GET_REFLECTION_OBJECT_PTR(ce);
  3547. lc_name = zend_str_tolower_dup(name, name_len);
  3548. if ((ce == zend_ce_closure && (name_len == sizeof(ZEND_INVOKE_FUNC_NAME)-1)
  3549. && memcmp(lc_name, ZEND_INVOKE_FUNC_NAME, sizeof(ZEND_INVOKE_FUNC_NAME)-1) == 0)
  3550. || zend_hash_str_exists(&ce->function_table, lc_name, name_len)) {
  3551. efree(lc_name);
  3552. RETURN_TRUE;
  3553. } else {
  3554. efree(lc_name);
  3555. RETURN_FALSE;
  3556. }
  3557. }
  3558. /* }}} */
  3559. /* {{{ proto public ReflectionMethod ReflectionClass::getMethod(string name) throws ReflectionException
  3560. Returns the class' method specified by its name */
  3561. ZEND_METHOD(reflection_class, getMethod)
  3562. {
  3563. reflection_object *intern;
  3564. zend_class_entry *ce;
  3565. zend_function *mptr;
  3566. zval obj_tmp;
  3567. char *name, *lc_name;
  3568. size_t name_len;
  3569. if (zend_parse_parameters(ZEND_NUM_ARGS(), "s", &name, &name_len) == FAILURE) {
  3570. return;
  3571. }
  3572. GET_REFLECTION_OBJECT_PTR(ce);
  3573. lc_name = zend_str_tolower_dup(name, name_len);
  3574. if (ce == zend_ce_closure && !Z_ISUNDEF(intern->obj) && (name_len == sizeof(ZEND_INVOKE_FUNC_NAME)-1)
  3575. && memcmp(lc_name, ZEND_INVOKE_FUNC_NAME, sizeof(ZEND_INVOKE_FUNC_NAME)-1) == 0
  3576. && (mptr = zend_get_closure_invoke_method(Z_OBJ(intern->obj))) != NULL)
  3577. {
  3578. /* don't assign closure_object since we only reflect the invoke handler
  3579. method and not the closure definition itself */
  3580. reflection_method_factory(ce, mptr, NULL, return_value);
  3581. efree(lc_name);
  3582. } else if (ce == zend_ce_closure && Z_ISUNDEF(intern->obj) && (name_len == sizeof(ZEND_INVOKE_FUNC_NAME)-1)
  3583. && memcmp(lc_name, ZEND_INVOKE_FUNC_NAME, sizeof(ZEND_INVOKE_FUNC_NAME)-1) == 0
  3584. && object_init_ex(&obj_tmp, ce) == SUCCESS && (mptr = zend_get_closure_invoke_method(Z_OBJ(obj_tmp))) != NULL) {
  3585. /* don't assign closure_object since we only reflect the invoke handler
  3586. method and not the closure definition itself */
  3587. reflection_method_factory(ce, mptr, NULL, return_value);
  3588. zval_ptr_dtor(&obj_tmp);
  3589. efree(lc_name);
  3590. } else if ((mptr = zend_hash_str_find_ptr(&ce->function_table, lc_name, name_len)) != NULL) {
  3591. reflection_method_factory(ce, mptr, NULL, return_value);
  3592. efree(lc_name);
  3593. } else {
  3594. efree(lc_name);
  3595. zend_throw_exception_ex(reflection_exception_ptr, 0,
  3596. "Method %s does not exist", name);
  3597. return;
  3598. }
  3599. }
  3600. /* }}} */
  3601. /* {{{ _addmethod */
  3602. static void _addmethod(zend_function *mptr, zend_class_entry *ce, zval *retval, zend_long filter)
  3603. {
  3604. if (mptr->common.fn_flags & filter) {
  3605. zval method;
  3606. reflection_method_factory(ce, mptr, NULL, &method);
  3607. add_next_index_zval(retval, &method);
  3608. }
  3609. }
  3610. /* }}} */
  3611. /* {{{ _addmethod */
  3612. static int _addmethod_va(zval *el, int num_args, va_list args, zend_hash_key *hash_key)
  3613. {
  3614. zend_function *mptr = (zend_function*)Z_PTR_P(el);
  3615. zend_class_entry *ce = *va_arg(args, zend_class_entry**);
  3616. zval *retval = va_arg(args, zval*);
  3617. long filter = va_arg(args, long);
  3618. _addmethod(mptr, ce, retval, filter);
  3619. return ZEND_HASH_APPLY_KEEP;
  3620. }
  3621. /* }}} */
  3622. /* {{{ proto public ReflectionMethod[] ReflectionClass::getMethods([long $filter])
  3623. Returns an array of this class' methods */
  3624. ZEND_METHOD(reflection_class, getMethods)
  3625. {
  3626. reflection_object *intern;
  3627. zend_class_entry *ce;
  3628. zend_long filter = 0;
  3629. zend_bool filter_is_null = 1;
  3630. if (zend_parse_parameters(ZEND_NUM_ARGS(), "|l!", &filter, &filter_is_null) == FAILURE) {
  3631. return;
  3632. }
  3633. if (filter_is_null) {
  3634. filter = ZEND_ACC_PPP_MASK | ZEND_ACC_ABSTRACT | ZEND_ACC_FINAL | ZEND_ACC_STATIC;
  3635. }
  3636. GET_REFLECTION_OBJECT_PTR(ce);
  3637. array_init(return_value);
  3638. zend_hash_apply_with_arguments(&ce->function_table, (apply_func_args_t) _addmethod_va, 4, &ce, return_value, filter);
  3639. if (instanceof_function(ce, zend_ce_closure)) {
  3640. zend_bool has_obj = Z_TYPE(intern->obj) != IS_UNDEF;
  3641. zval obj_tmp;
  3642. zend_object *obj;
  3643. if (!has_obj) {
  3644. object_init_ex(&obj_tmp, ce);
  3645. obj = Z_OBJ(obj_tmp);
  3646. } else {
  3647. obj = Z_OBJ(intern->obj);
  3648. }
  3649. zend_function *closure = zend_get_closure_invoke_method(obj);
  3650. if (closure) {
  3651. _addmethod(closure, ce, return_value, filter);
  3652. }
  3653. if (!has_obj) {
  3654. zval_ptr_dtor(&obj_tmp);
  3655. }
  3656. }
  3657. }
  3658. /* }}} */
  3659. /* {{{ proto public bool ReflectionClass::hasProperty(string name)
  3660. Returns whether a property exists or not */
  3661. ZEND_METHOD(reflection_class, hasProperty)
  3662. {
  3663. reflection_object *intern;
  3664. zend_property_info *property_info;
  3665. zend_class_entry *ce;
  3666. zend_string *name;
  3667. zval property;
  3668. if (zend_parse_parameters(ZEND_NUM_ARGS(), "S", &name) == FAILURE) {
  3669. return;
  3670. }
  3671. GET_REFLECTION_OBJECT_PTR(ce);
  3672. if ((property_info = zend_hash_find_ptr(&ce->properties_info, name)) != NULL) {
  3673. if (property_info->flags & ZEND_ACC_SHADOW) {
  3674. RETURN_FALSE;
  3675. }
  3676. RETURN_TRUE;
  3677. } else {
  3678. if (Z_TYPE(intern->obj) != IS_UNDEF && Z_OBJ_HANDLER(intern->obj, has_property)) {
  3679. ZVAL_STR_COPY(&property, name);
  3680. if (Z_OBJ_HANDLER(intern->obj, has_property)(&intern->obj, &property, 2, NULL)) {
  3681. zval_ptr_dtor(&property);
  3682. RETURN_TRUE;
  3683. }
  3684. zval_ptr_dtor(&property);
  3685. }
  3686. RETURN_FALSE;
  3687. }
  3688. }
  3689. /* }}} */
  3690. /* {{{ proto public ReflectionProperty ReflectionClass::getProperty(string name) throws ReflectionException
  3691. Returns the class' property specified by its name */
  3692. ZEND_METHOD(reflection_class, getProperty)
  3693. {
  3694. reflection_object *intern;
  3695. zend_class_entry *ce, *ce2;
  3696. zend_property_info *property_info;
  3697. zend_string *name, *classname;
  3698. char *tmp, *str_name;
  3699. size_t classname_len, str_name_len;
  3700. if (zend_parse_parameters(ZEND_NUM_ARGS(), "S", &name) == FAILURE) {
  3701. return;
  3702. }
  3703. GET_REFLECTION_OBJECT_PTR(ce);
  3704. if ((property_info = zend_hash_find_ptr(&ce->properties_info, name)) != NULL) {
  3705. if ((property_info->flags & ZEND_ACC_SHADOW) == 0) {
  3706. reflection_property_factory(ce, name, property_info, return_value);
  3707. return;
  3708. }
  3709. } else if (Z_TYPE(intern->obj) != IS_UNDEF) {
  3710. /* Check for dynamic properties */
  3711. if (zend_hash_exists(Z_OBJ_HT(intern->obj)->get_properties(&intern->obj), name)) {
  3712. zend_property_info property_info_tmp;
  3713. property_info_tmp.flags = ZEND_ACC_IMPLICIT_PUBLIC;
  3714. property_info_tmp.name = name;
  3715. property_info_tmp.doc_comment = NULL;
  3716. property_info_tmp.ce = ce;
  3717. reflection_property_factory(ce, name, &property_info_tmp, return_value);
  3718. return;
  3719. }
  3720. }
  3721. str_name = ZSTR_VAL(name);
  3722. if ((tmp = strstr(ZSTR_VAL(name), "::")) != NULL) {
  3723. classname_len = tmp - ZSTR_VAL(name);
  3724. classname = zend_string_alloc(classname_len, 0);
  3725. zend_str_tolower_copy(ZSTR_VAL(classname), ZSTR_VAL(name), classname_len);
  3726. ZSTR_VAL(classname)[classname_len] = '\0';
  3727. str_name_len = ZSTR_LEN(name) - (classname_len + 2);
  3728. str_name = tmp + 2;
  3729. ce2 = zend_lookup_class(classname);
  3730. if (!ce2) {
  3731. if (!EG(exception)) {
  3732. zend_throw_exception_ex(reflection_exception_ptr, -1, "Class %s does not exist", ZSTR_VAL(classname));
  3733. }
  3734. zend_string_release_ex(classname, 0);
  3735. return;
  3736. }
  3737. zend_string_release_ex(classname, 0);
  3738. if (!instanceof_function(ce, ce2)) {
  3739. zend_throw_exception_ex(reflection_exception_ptr, -1, "Fully qualified property name %s::%s does not specify a base class of %s", ZSTR_VAL(ce2->name), str_name, ZSTR_VAL(ce->name));
  3740. return;
  3741. }
  3742. ce = ce2;
  3743. if ((property_info = zend_hash_str_find_ptr(&ce->properties_info, str_name, str_name_len)) != NULL && (property_info->flags & ZEND_ACC_SHADOW) == 0) {
  3744. reflection_property_factory_str(ce, str_name, str_name_len, property_info, return_value);
  3745. return;
  3746. }
  3747. }
  3748. zend_throw_exception_ex(reflection_exception_ptr, 0,
  3749. "Property %s does not exist", str_name);
  3750. }
  3751. /* }}} */
  3752. /* {{{ _addproperty */
  3753. static int _addproperty(zval *el, int num_args, va_list args, zend_hash_key *hash_key)
  3754. {
  3755. zval property;
  3756. zend_property_info *pptr = (zend_property_info*)Z_PTR_P(el);
  3757. zend_class_entry *ce = *va_arg(args, zend_class_entry**);
  3758. zval *retval = va_arg(args, zval*);
  3759. long filter = va_arg(args, long);
  3760. if (pptr->flags & ZEND_ACC_SHADOW) {
  3761. return 0;
  3762. }
  3763. if (pptr->flags & filter) {
  3764. const char *class_name, *prop_name;
  3765. size_t prop_name_len;
  3766. zend_unmangle_property_name_ex(pptr->name, &class_name, &prop_name, &prop_name_len);
  3767. reflection_property_factory_str(ce, prop_name, prop_name_len, pptr, &property);
  3768. add_next_index_zval(retval, &property);
  3769. }
  3770. return 0;
  3771. }
  3772. /* }}} */
  3773. /* {{{ _adddynproperty */
  3774. static int _adddynproperty(zval *ptr, int num_args, va_list args, zend_hash_key *hash_key)
  3775. {
  3776. zval property;
  3777. zend_class_entry *ce = *va_arg(args, zend_class_entry**);
  3778. zval *retval = va_arg(args, zval*);
  3779. /* under some circumstances, the properties hash table may contain numeric
  3780. * properties (e.g. when casting from array). This is a WONT FIX bug, at
  3781. * least for the moment. Ignore these */
  3782. if (hash_key->key == NULL) {
  3783. return 0;
  3784. }
  3785. if (ZSTR_VAL(hash_key->key)[0] == '\0') {
  3786. return 0; /* non public cannot be dynamic */
  3787. }
  3788. if (zend_get_property_info(ce, hash_key->key, 1) == NULL) {
  3789. zend_property_info property_info;
  3790. property_info.doc_comment = NULL;
  3791. property_info.flags = ZEND_ACC_IMPLICIT_PUBLIC;
  3792. property_info.name = hash_key->key;
  3793. property_info.ce = ce;
  3794. property_info.offset = -1;
  3795. reflection_property_factory(ce, hash_key->key, &property_info, &property);
  3796. add_next_index_zval(retval, &property);
  3797. }
  3798. return 0;
  3799. }
  3800. /* }}} */
  3801. /* {{{ proto public ReflectionProperty[] ReflectionClass::getProperties([long $filter])
  3802. Returns an array of this class' properties */
  3803. ZEND_METHOD(reflection_class, getProperties)
  3804. {
  3805. reflection_object *intern;
  3806. zend_class_entry *ce;
  3807. zend_long filter = 0;
  3808. zend_bool filter_is_null = 1;
  3809. if (zend_parse_parameters(ZEND_NUM_ARGS(), "|l!", &filter, &filter_is_null) == FAILURE) {
  3810. return;
  3811. }
  3812. if (filter_is_null) {
  3813. filter = ZEND_ACC_PPP_MASK | ZEND_ACC_STATIC;
  3814. }
  3815. GET_REFLECTION_OBJECT_PTR(ce);
  3816. array_init(return_value);
  3817. zend_hash_apply_with_arguments(&ce->properties_info, (apply_func_args_t) _addproperty, 3, &ce, return_value, filter);
  3818. if (Z_TYPE(intern->obj) != IS_UNDEF && (filter & ZEND_ACC_PUBLIC) != 0 && Z_OBJ_HT(intern->obj)->get_properties) {
  3819. HashTable *properties = Z_OBJ_HT(intern->obj)->get_properties(&intern->obj);
  3820. zend_hash_apply_with_arguments(properties, (apply_func_args_t) _adddynproperty, 2, &ce, return_value);
  3821. }
  3822. }
  3823. /* }}} */
  3824. /* {{{ proto public bool ReflectionClass::hasConstant(string name)
  3825. Returns whether a constant exists or not */
  3826. ZEND_METHOD(reflection_class, hasConstant)
  3827. {
  3828. reflection_object *intern;
  3829. zend_class_entry *ce;
  3830. zend_string *name;
  3831. if (zend_parse_parameters(ZEND_NUM_ARGS(), "S", &name) == FAILURE) {
  3832. return;
  3833. }
  3834. GET_REFLECTION_OBJECT_PTR(ce);
  3835. if (zend_hash_exists(&ce->constants_table, name)) {
  3836. RETURN_TRUE;
  3837. } else {
  3838. RETURN_FALSE;
  3839. }
  3840. }
  3841. /* }}} */
  3842. /* {{{ proto public array ReflectionClass::getConstants()
  3843. Returns an associative array containing this class' constants and their values */
  3844. ZEND_METHOD(reflection_class, getConstants)
  3845. {
  3846. reflection_object *intern;
  3847. zend_class_entry *ce;
  3848. zend_string *key;
  3849. zend_class_constant *c;
  3850. zval val;
  3851. if (zend_parse_parameters_none() == FAILURE) {
  3852. return;
  3853. }
  3854. GET_REFLECTION_OBJECT_PTR(ce);
  3855. array_init(return_value);
  3856. ZEND_HASH_FOREACH_STR_KEY_PTR(&ce->constants_table, key, c) {
  3857. if (UNEXPECTED(zval_update_constant_ex(&c->value, ce) != SUCCESS)) {
  3858. zend_array_destroy(Z_ARRVAL_P(return_value));
  3859. RETURN_NULL();
  3860. }
  3861. ZVAL_COPY_OR_DUP(&val, &c->value);
  3862. zend_hash_add_new(Z_ARRVAL_P(return_value), key, &val);
  3863. } ZEND_HASH_FOREACH_END();
  3864. }
  3865. /* }}} */
  3866. /* {{{ proto public array ReflectionClass::getReflectionConstants()
  3867. Returns an associative array containing this class' constants as ReflectionClassConstant objects */
  3868. ZEND_METHOD(reflection_class, getReflectionConstants)
  3869. {
  3870. reflection_object *intern;
  3871. zend_class_entry *ce;
  3872. zend_string *name;
  3873. zend_class_constant *constant;
  3874. if (zend_parse_parameters_none() == FAILURE) {
  3875. return;
  3876. }
  3877. GET_REFLECTION_OBJECT_PTR(ce);
  3878. array_init(return_value);
  3879. ZEND_HASH_FOREACH_STR_KEY_PTR(&ce->constants_table, name, constant) {
  3880. zval class_const;
  3881. reflection_class_constant_factory(ce, name, constant, &class_const);
  3882. zend_hash_next_index_insert(Z_ARRVAL_P(return_value), &class_const);
  3883. } ZEND_HASH_FOREACH_END();
  3884. }
  3885. /* }}} */
  3886. /* {{{ proto public mixed ReflectionClass::getConstant(string name)
  3887. Returns the class' constant specified by its name */
  3888. ZEND_METHOD(reflection_class, getConstant)
  3889. {
  3890. reflection_object *intern;
  3891. zend_class_entry *ce;
  3892. zend_class_constant *c;
  3893. zend_string *name;
  3894. if (zend_parse_parameters(ZEND_NUM_ARGS(), "S", &name) == FAILURE) {
  3895. return;
  3896. }
  3897. GET_REFLECTION_OBJECT_PTR(ce);
  3898. ZEND_HASH_FOREACH_PTR(&ce->constants_table, c) {
  3899. if (UNEXPECTED(zval_update_constant_ex(&c->value, ce) != SUCCESS)) {
  3900. return;
  3901. }
  3902. } ZEND_HASH_FOREACH_END();
  3903. if ((c = zend_hash_find_ptr(&ce->constants_table, name)) == NULL) {
  3904. RETURN_FALSE;
  3905. }
  3906. ZVAL_COPY_OR_DUP(return_value, &c->value);
  3907. }
  3908. /* }}} */
  3909. /* {{{ proto public mixed ReflectionClass::getReflectionConstant(string name)
  3910. Returns the class' constant as ReflectionClassConstant objects */
  3911. ZEND_METHOD(reflection_class, getReflectionConstant)
  3912. {
  3913. reflection_object *intern;
  3914. zend_class_entry *ce;
  3915. zend_class_constant *constant;
  3916. zend_string *name;
  3917. GET_REFLECTION_OBJECT_PTR(ce);
  3918. if (zend_parse_parameters(ZEND_NUM_ARGS(), "S", &name) == FAILURE) {
  3919. return;
  3920. }
  3921. if ((constant = zend_hash_find_ptr(&ce->constants_table, name)) == NULL) {
  3922. RETURN_FALSE;
  3923. }
  3924. reflection_class_constant_factory(ce, name, constant, return_value);
  3925. }
  3926. /* }}} */
  3927. /* {{{ _class_check_flag */
  3928. static void _class_check_flag(INTERNAL_FUNCTION_PARAMETERS, int mask)
  3929. {
  3930. reflection_object *intern;
  3931. zend_class_entry *ce;
  3932. if (zend_parse_parameters_none() == FAILURE) {
  3933. return;
  3934. }
  3935. GET_REFLECTION_OBJECT_PTR(ce);
  3936. RETVAL_BOOL(ce->ce_flags & mask);
  3937. }
  3938. /* }}} */
  3939. /* {{{ proto public bool ReflectionClass::isInstantiable()
  3940. Returns whether this class is instantiable */
  3941. ZEND_METHOD(reflection_class, isInstantiable)
  3942. {
  3943. reflection_object *intern;
  3944. zend_class_entry *ce;
  3945. if (zend_parse_parameters_none() == FAILURE) {
  3946. return;
  3947. }
  3948. GET_REFLECTION_OBJECT_PTR(ce);
  3949. if (ce->ce_flags & (ZEND_ACC_INTERFACE | ZEND_ACC_TRAIT | ZEND_ACC_EXPLICIT_ABSTRACT_CLASS | ZEND_ACC_IMPLICIT_ABSTRACT_CLASS)) {
  3950. RETURN_FALSE;
  3951. }
  3952. /* Basically, the class is instantiable. Though, if there is a constructor
  3953. * and it is not publicly accessible, it isn't! */
  3954. if (!ce->constructor) {
  3955. RETURN_TRUE;
  3956. }
  3957. RETURN_BOOL(ce->constructor->common.fn_flags & ZEND_ACC_PUBLIC);
  3958. }
  3959. /* }}} */
  3960. /* {{{ proto public bool ReflectionClass::isCloneable()
  3961. Returns whether this class is cloneable */
  3962. ZEND_METHOD(reflection_class, isCloneable)
  3963. {
  3964. reflection_object *intern;
  3965. zend_class_entry *ce;
  3966. zval obj;
  3967. if (zend_parse_parameters_none() == FAILURE) {
  3968. return;
  3969. }
  3970. GET_REFLECTION_OBJECT_PTR(ce);
  3971. if (ce->ce_flags & (ZEND_ACC_INTERFACE | ZEND_ACC_TRAIT | ZEND_ACC_EXPLICIT_ABSTRACT_CLASS | ZEND_ACC_IMPLICIT_ABSTRACT_CLASS)) {
  3972. RETURN_FALSE;
  3973. }
  3974. if (!Z_ISUNDEF(intern->obj)) {
  3975. if (ce->clone) {
  3976. RETURN_BOOL(ce->clone->common.fn_flags & ZEND_ACC_PUBLIC);
  3977. } else {
  3978. RETURN_BOOL(Z_OBJ_HANDLER(intern->obj, clone_obj) != NULL);
  3979. }
  3980. } else {
  3981. if (ce->clone) {
  3982. RETURN_BOOL(ce->clone->common.fn_flags & ZEND_ACC_PUBLIC);
  3983. } else {
  3984. if (UNEXPECTED(object_init_ex(&obj, ce) != SUCCESS)) {
  3985. return;
  3986. }
  3987. /* We're not calling the constructor, so don't call the destructor either. */
  3988. zend_object_store_ctor_failed(Z_OBJ(obj));
  3989. RETVAL_BOOL(Z_OBJ_HANDLER(obj, clone_obj) != NULL);
  3990. zval_ptr_dtor(&obj);
  3991. }
  3992. }
  3993. }
  3994. /* }}} */
  3995. /* {{{ proto public bool ReflectionClass::isInterface()
  3996. Returns whether this is an interface or a class */
  3997. ZEND_METHOD(reflection_class, isInterface)
  3998. {
  3999. _class_check_flag(INTERNAL_FUNCTION_PARAM_PASSTHRU, ZEND_ACC_INTERFACE);
  4000. }
  4001. /* }}} */
  4002. /* {{{ proto public bool ReflectionClass::isTrait()
  4003. Returns whether this is a trait */
  4004. ZEND_METHOD(reflection_class, isTrait)
  4005. {
  4006. _class_check_flag(INTERNAL_FUNCTION_PARAM_PASSTHRU, ZEND_ACC_TRAIT);
  4007. }
  4008. /* }}} */
  4009. /* {{{ proto public bool ReflectionClass::isFinal()
  4010. Returns whether this class is final */
  4011. ZEND_METHOD(reflection_class, isFinal)
  4012. {
  4013. _class_check_flag(INTERNAL_FUNCTION_PARAM_PASSTHRU, ZEND_ACC_FINAL);
  4014. }
  4015. /* }}} */
  4016. /* {{{ proto public bool ReflectionClass::isAbstract()
  4017. Returns whether this class is abstract */
  4018. ZEND_METHOD(reflection_class, isAbstract)
  4019. {
  4020. _class_check_flag(INTERNAL_FUNCTION_PARAM_PASSTHRU, ZEND_ACC_IMPLICIT_ABSTRACT_CLASS|ZEND_ACC_EXPLICIT_ABSTRACT_CLASS);
  4021. }
  4022. /* }}} */
  4023. /* {{{ proto public int ReflectionClass::getModifiers()
  4024. Returns a bitfield of the access modifiers for this class */
  4025. ZEND_METHOD(reflection_class, getModifiers)
  4026. {
  4027. reflection_object *intern;
  4028. zend_class_entry *ce;
  4029. uint32_t keep_flags = ZEND_ACC_FINAL
  4030. | ZEND_ACC_EXPLICIT_ABSTRACT_CLASS | ZEND_ACC_IMPLICIT_ABSTRACT_CLASS;
  4031. if (zend_parse_parameters_none() == FAILURE) {
  4032. return;
  4033. }
  4034. GET_REFLECTION_OBJECT_PTR(ce);
  4035. RETURN_LONG((ce->ce_flags & keep_flags));
  4036. }
  4037. /* }}} */
  4038. /* {{{ proto public bool ReflectionClass::isInstance(stdclass object)
  4039. Returns whether the given object is an instance of this class */
  4040. ZEND_METHOD(reflection_class, isInstance)
  4041. {
  4042. reflection_object *intern;
  4043. zend_class_entry *ce;
  4044. zval *object;
  4045. if (zend_parse_parameters(ZEND_NUM_ARGS(), "o", &object) == FAILURE) {
  4046. return;
  4047. }
  4048. GET_REFLECTION_OBJECT_PTR(ce);
  4049. RETURN_BOOL(instanceof_function(Z_OBJCE_P(object), ce));
  4050. }
  4051. /* }}} */
  4052. /* {{{ proto public stdclass ReflectionClass::newInstance(mixed* args, ...)
  4053. Returns an instance of this class */
  4054. ZEND_METHOD(reflection_class, newInstance)
  4055. {
  4056. zval retval;
  4057. reflection_object *intern;
  4058. zend_class_entry *ce, *old_scope;
  4059. zend_function *constructor;
  4060. GET_REFLECTION_OBJECT_PTR(ce);
  4061. if (UNEXPECTED(object_init_ex(return_value, ce) != SUCCESS)) {
  4062. return;
  4063. }
  4064. old_scope = EG(fake_scope);
  4065. EG(fake_scope) = ce;
  4066. constructor = Z_OBJ_HT_P(return_value)->get_constructor(Z_OBJ_P(return_value));
  4067. EG(fake_scope) = old_scope;
  4068. /* Run the constructor if there is one */
  4069. if (constructor) {
  4070. zval *params = NULL;
  4071. int ret, i, num_args = 0;
  4072. zend_fcall_info fci;
  4073. zend_fcall_info_cache fcc;
  4074. if (!(constructor->common.fn_flags & ZEND_ACC_PUBLIC)) {
  4075. zend_throw_exception_ex(reflection_exception_ptr, 0, "Access to non-public constructor of class %s", ZSTR_VAL(ce->name));
  4076. zval_ptr_dtor(return_value);
  4077. RETURN_NULL();
  4078. }
  4079. if (zend_parse_parameters(ZEND_NUM_ARGS(), "*", &params, &num_args) == FAILURE) {
  4080. zval_ptr_dtor(return_value);
  4081. RETURN_FALSE;
  4082. }
  4083. for (i = 0; i < num_args; i++) {
  4084. Z_TRY_ADDREF(params[i]);
  4085. }
  4086. fci.size = sizeof(fci);
  4087. ZVAL_UNDEF(&fci.function_name);
  4088. fci.object = Z_OBJ_P(return_value);
  4089. fci.retval = &retval;
  4090. fci.param_count = num_args;
  4091. fci.params = params;
  4092. fci.no_separation = 1;
  4093. fcc.function_handler = constructor;
  4094. fcc.called_scope = Z_OBJCE_P(return_value);
  4095. fcc.object = Z_OBJ_P(return_value);
  4096. ret = zend_call_function(&fci, &fcc);
  4097. zval_ptr_dtor(&retval);
  4098. for (i = 0; i < num_args; i++) {
  4099. zval_ptr_dtor(&params[i]);
  4100. }
  4101. if (EG(exception)) {
  4102. zend_object_store_ctor_failed(Z_OBJ_P(return_value));
  4103. }
  4104. if (ret == FAILURE) {
  4105. php_error_docref(NULL, E_WARNING, "Invocation of %s's constructor failed", ZSTR_VAL(ce->name));
  4106. zval_ptr_dtor(return_value);
  4107. RETURN_NULL();
  4108. }
  4109. } else if (ZEND_NUM_ARGS()) {
  4110. zend_throw_exception_ex(reflection_exception_ptr, 0, "Class %s does not have a constructor, so you cannot pass any constructor arguments", ZSTR_VAL(ce->name));
  4111. }
  4112. }
  4113. /* }}} */
  4114. /* {{{ proto public stdclass ReflectionClass::newInstanceWithoutConstructor()
  4115. Returns an instance of this class without invoking its constructor */
  4116. ZEND_METHOD(reflection_class, newInstanceWithoutConstructor)
  4117. {
  4118. reflection_object *intern;
  4119. zend_class_entry *ce;
  4120. GET_REFLECTION_OBJECT_PTR(ce);
  4121. if (ce->create_object != NULL && ce->ce_flags & ZEND_ACC_FINAL) {
  4122. zend_throw_exception_ex(reflection_exception_ptr, 0, "Class %s is an internal class marked as final that cannot be instantiated without invoking its constructor", ZSTR_VAL(ce->name));
  4123. return;
  4124. }
  4125. object_init_ex(return_value, ce);
  4126. }
  4127. /* }}} */
  4128. /* {{{ proto public stdclass ReflectionClass::newInstanceArgs([array args])
  4129. Returns an instance of this class */
  4130. ZEND_METHOD(reflection_class, newInstanceArgs)
  4131. {
  4132. zval retval, *val;
  4133. reflection_object *intern;
  4134. zend_class_entry *ce, *old_scope;
  4135. int ret, i, argc = 0;
  4136. HashTable *args;
  4137. zend_function *constructor;
  4138. GET_REFLECTION_OBJECT_PTR(ce);
  4139. if (zend_parse_parameters(ZEND_NUM_ARGS(), "|h", &args) == FAILURE) {
  4140. return;
  4141. }
  4142. if (ZEND_NUM_ARGS() > 0) {
  4143. argc = args->nNumOfElements;
  4144. }
  4145. if (UNEXPECTED(object_init_ex(return_value, ce) != SUCCESS)) {
  4146. return;
  4147. }
  4148. old_scope = EG(fake_scope);
  4149. EG(fake_scope) = ce;
  4150. constructor = Z_OBJ_HT_P(return_value)->get_constructor(Z_OBJ_P(return_value));
  4151. EG(fake_scope) = old_scope;
  4152. /* Run the constructor if there is one */
  4153. if (constructor) {
  4154. zval *params = NULL;
  4155. zend_fcall_info fci;
  4156. zend_fcall_info_cache fcc;
  4157. if (!(constructor->common.fn_flags & ZEND_ACC_PUBLIC)) {
  4158. zend_throw_exception_ex(reflection_exception_ptr, 0, "Access to non-public constructor of class %s", ZSTR_VAL(ce->name));
  4159. zval_ptr_dtor(return_value);
  4160. RETURN_NULL();
  4161. }
  4162. if (argc) {
  4163. params = safe_emalloc(sizeof(zval), argc, 0);
  4164. argc = 0;
  4165. ZEND_HASH_FOREACH_VAL(args, val) {
  4166. ZVAL_COPY(&params[argc], val);
  4167. argc++;
  4168. } ZEND_HASH_FOREACH_END();
  4169. }
  4170. fci.size = sizeof(fci);
  4171. ZVAL_UNDEF(&fci.function_name);
  4172. fci.object = Z_OBJ_P(return_value);
  4173. fci.retval = &retval;
  4174. fci.param_count = argc;
  4175. fci.params = params;
  4176. fci.no_separation = 1;
  4177. fcc.function_handler = constructor;
  4178. fcc.called_scope = Z_OBJCE_P(return_value);
  4179. fcc.object = Z_OBJ_P(return_value);
  4180. ret = zend_call_function(&fci, &fcc);
  4181. zval_ptr_dtor(&retval);
  4182. if (params) {
  4183. for (i = 0; i < argc; i++) {
  4184. zval_ptr_dtor(&params[i]);
  4185. }
  4186. efree(params);
  4187. }
  4188. if (EG(exception)) {
  4189. zend_object_store_ctor_failed(Z_OBJ_P(return_value));
  4190. }
  4191. if (ret == FAILURE) {
  4192. zval_ptr_dtor(&retval);
  4193. php_error_docref(NULL, E_WARNING, "Invocation of %s's constructor failed", ZSTR_VAL(ce->name));
  4194. zval_ptr_dtor(return_value);
  4195. RETURN_NULL();
  4196. }
  4197. } else if (argc) {
  4198. zend_throw_exception_ex(reflection_exception_ptr, 0, "Class %s does not have a constructor, so you cannot pass any constructor arguments", ZSTR_VAL(ce->name));
  4199. }
  4200. }
  4201. /* }}} */
  4202. /* {{{ proto public ReflectionClass[] ReflectionClass::getInterfaces()
  4203. Returns an array of interfaces this class implements */
  4204. ZEND_METHOD(reflection_class, getInterfaces)
  4205. {
  4206. reflection_object *intern;
  4207. zend_class_entry *ce;
  4208. if (zend_parse_parameters_none() == FAILURE) {
  4209. return;
  4210. }
  4211. GET_REFLECTION_OBJECT_PTR(ce);
  4212. if (ce->num_interfaces) {
  4213. uint32_t i;
  4214. array_init(return_value);
  4215. for (i=0; i < ce->num_interfaces; i++) {
  4216. zval interface;
  4217. zend_reflection_class_factory(ce->interfaces[i], &interface);
  4218. zend_hash_update(Z_ARRVAL_P(return_value), ce->interfaces[i]->name, &interface);
  4219. }
  4220. } else {
  4221. ZVAL_EMPTY_ARRAY(return_value);
  4222. }
  4223. }
  4224. /* }}} */
  4225. /* {{{ proto public String[] ReflectionClass::getInterfaceNames()
  4226. Returns an array of names of interfaces this class implements */
  4227. ZEND_METHOD(reflection_class, getInterfaceNames)
  4228. {
  4229. reflection_object *intern;
  4230. zend_class_entry *ce;
  4231. uint32_t i;
  4232. if (zend_parse_parameters_none() == FAILURE) {
  4233. return;
  4234. }
  4235. GET_REFLECTION_OBJECT_PTR(ce);
  4236. if (!ce->num_interfaces) {
  4237. /* Return an empty array if this class implements no interfaces */
  4238. ZVAL_EMPTY_ARRAY(return_value);
  4239. return;
  4240. }
  4241. array_init(return_value);
  4242. for (i=0; i < ce->num_interfaces; i++) {
  4243. add_next_index_str(return_value, zend_string_copy(ce->interfaces[i]->name));
  4244. }
  4245. }
  4246. /* }}} */
  4247. /* {{{ proto public ReflectionClass[] ReflectionClass::getTraits()
  4248. Returns an array of traits used by this class */
  4249. ZEND_METHOD(reflection_class, getTraits)
  4250. {
  4251. reflection_object *intern;
  4252. zend_class_entry *ce;
  4253. uint32_t i;
  4254. if (zend_parse_parameters_none() == FAILURE) {
  4255. return;
  4256. }
  4257. GET_REFLECTION_OBJECT_PTR(ce);
  4258. if (!ce->num_traits) {
  4259. ZVAL_EMPTY_ARRAY(return_value);
  4260. return;
  4261. }
  4262. array_init(return_value);
  4263. for (i=0; i < ce->num_traits; i++) {
  4264. zval trait;
  4265. zend_reflection_class_factory(ce->traits[i], &trait);
  4266. zend_hash_update(Z_ARRVAL_P(return_value), ce->traits[i]->name, &trait);
  4267. }
  4268. }
  4269. /* }}} */
  4270. /* {{{ proto public String[] ReflectionClass::getTraitNames()
  4271. Returns an array of names of traits used by this class */
  4272. ZEND_METHOD(reflection_class, getTraitNames)
  4273. {
  4274. reflection_object *intern;
  4275. zend_class_entry *ce;
  4276. uint32_t i;
  4277. if (zend_parse_parameters_none() == FAILURE) {
  4278. return;
  4279. }
  4280. GET_REFLECTION_OBJECT_PTR(ce);
  4281. if (!ce->num_traits) {
  4282. ZVAL_EMPTY_ARRAY(return_value);
  4283. return;
  4284. }
  4285. array_init(return_value);
  4286. for (i=0; i < ce->num_traits; i++) {
  4287. add_next_index_str(return_value, zend_string_copy(ce->traits[i]->name));
  4288. }
  4289. }
  4290. /* }}} */
  4291. /* {{{ proto public array ReflectionClass::getTraitAliases()
  4292. Returns an array of trait aliases */
  4293. ZEND_METHOD(reflection_class, getTraitAliases)
  4294. {
  4295. reflection_object *intern;
  4296. zend_class_entry *ce;
  4297. if (zend_parse_parameters_none() == FAILURE) {
  4298. return;
  4299. }
  4300. GET_REFLECTION_OBJECT_PTR(ce);
  4301. if (ce->trait_aliases) {
  4302. uint32_t i = 0;
  4303. array_init(return_value);
  4304. while (ce->trait_aliases[i]) {
  4305. zend_string *mname;
  4306. zend_trait_method_reference *cur_ref = &ce->trait_aliases[i]->trait_method;
  4307. if (ce->trait_aliases[i]->alias) {
  4308. mname = zend_string_alloc(ZSTR_LEN(cur_ref->class_name) + ZSTR_LEN(cur_ref->method_name) + 2, 0);
  4309. snprintf(ZSTR_VAL(mname), ZSTR_LEN(mname) + 1, "%s::%s", ZSTR_VAL(cur_ref->class_name), ZSTR_VAL(cur_ref->method_name));
  4310. add_assoc_str_ex(return_value, ZSTR_VAL(ce->trait_aliases[i]->alias), ZSTR_LEN(ce->trait_aliases[i]->alias), mname);
  4311. }
  4312. i++;
  4313. }
  4314. } else {
  4315. ZVAL_EMPTY_ARRAY(return_value);
  4316. }
  4317. }
  4318. /* }}} */
  4319. /* {{{ proto public ReflectionClass ReflectionClass::getParentClass()
  4320. Returns the class' parent class, or, if none exists, FALSE */
  4321. ZEND_METHOD(reflection_class, getParentClass)
  4322. {
  4323. reflection_object *intern;
  4324. zend_class_entry *ce;
  4325. if (zend_parse_parameters_none() == FAILURE) {
  4326. return;
  4327. }
  4328. GET_REFLECTION_OBJECT_PTR(ce);
  4329. if (ce->parent) {
  4330. zend_reflection_class_factory(ce->parent, return_value);
  4331. } else {
  4332. RETURN_FALSE;
  4333. }
  4334. }
  4335. /* }}} */
  4336. /* {{{ proto public bool ReflectionClass::isSubclassOf(string|ReflectionClass class)
  4337. Returns whether this class is a subclass of another class */
  4338. ZEND_METHOD(reflection_class, isSubclassOf)
  4339. {
  4340. reflection_object *intern, *argument;
  4341. zend_class_entry *ce, *class_ce;
  4342. zval *class_name;
  4343. GET_REFLECTION_OBJECT_PTR(ce);
  4344. if (zend_parse_parameters(ZEND_NUM_ARGS(), "z", &class_name) == FAILURE) {
  4345. return;
  4346. }
  4347. switch (Z_TYPE_P(class_name)) {
  4348. case IS_STRING:
  4349. if ((class_ce = zend_lookup_class(Z_STR_P(class_name))) == NULL) {
  4350. zend_throw_exception_ex(reflection_exception_ptr, 0,
  4351. "Class %s does not exist", Z_STRVAL_P(class_name));
  4352. return;
  4353. }
  4354. break;
  4355. case IS_OBJECT:
  4356. if (instanceof_function(Z_OBJCE_P(class_name), reflection_class_ptr)) {
  4357. argument = Z_REFLECTION_P(class_name);
  4358. if (argument->ptr == NULL) {
  4359. zend_throw_error(NULL, "Internal error: Failed to retrieve the argument's reflection object");
  4360. return;
  4361. }
  4362. class_ce = argument->ptr;
  4363. break;
  4364. }
  4365. /* no break */
  4366. default:
  4367. zend_throw_exception_ex(reflection_exception_ptr, 0,
  4368. "Parameter one must either be a string or a ReflectionClass object");
  4369. return;
  4370. }
  4371. RETURN_BOOL((ce != class_ce && instanceof_function(ce, class_ce)));
  4372. }
  4373. /* }}} */
  4374. /* {{{ proto public bool ReflectionClass::implementsInterface(string|ReflectionClass interface_name)
  4375. Returns whether this class is a subclass of another class */
  4376. ZEND_METHOD(reflection_class, implementsInterface)
  4377. {
  4378. reflection_object *intern, *argument;
  4379. zend_class_entry *ce, *interface_ce;
  4380. zval *interface;
  4381. GET_REFLECTION_OBJECT_PTR(ce);
  4382. if (zend_parse_parameters(ZEND_NUM_ARGS(), "z", &interface) == FAILURE) {
  4383. return;
  4384. }
  4385. switch (Z_TYPE_P(interface)) {
  4386. case IS_STRING:
  4387. if ((interface_ce = zend_lookup_class(Z_STR_P(interface))) == NULL) {
  4388. zend_throw_exception_ex(reflection_exception_ptr, 0,
  4389. "Interface %s does not exist", Z_STRVAL_P(interface));
  4390. return;
  4391. }
  4392. break;
  4393. case IS_OBJECT:
  4394. if (instanceof_function(Z_OBJCE_P(interface), reflection_class_ptr)) {
  4395. argument = Z_REFLECTION_P(interface);
  4396. if (argument->ptr == NULL) {
  4397. zend_throw_error(NULL, "Internal error: Failed to retrieve the argument's reflection object");
  4398. return;
  4399. }
  4400. interface_ce = argument->ptr;
  4401. break;
  4402. }
  4403. /* no break */
  4404. default:
  4405. zend_throw_exception_ex(reflection_exception_ptr, 0,
  4406. "Parameter one must either be a string or a ReflectionClass object");
  4407. return;
  4408. }
  4409. if (!(interface_ce->ce_flags & ZEND_ACC_INTERFACE)) {
  4410. zend_throw_exception_ex(reflection_exception_ptr, 0,
  4411. "%s is not an interface", ZSTR_VAL(interface_ce->name));
  4412. return;
  4413. }
  4414. RETURN_BOOL(instanceof_function(ce, interface_ce));
  4415. }
  4416. /* }}} */
  4417. /* {{{ proto public bool ReflectionClass::isIterable()
  4418. Returns whether this class is iterable (can be used inside foreach) */
  4419. ZEND_METHOD(reflection_class, isIterable)
  4420. {
  4421. reflection_object *intern;
  4422. zend_class_entry *ce;
  4423. if (zend_parse_parameters_none() == FAILURE) {
  4424. return;
  4425. }
  4426. GET_REFLECTION_OBJECT_PTR(ce);
  4427. if (ce->ce_flags & (ZEND_ACC_INTERFACE | ZEND_ACC_IMPLICIT_ABSTRACT_CLASS |
  4428. ZEND_ACC_TRAIT | ZEND_ACC_EXPLICIT_ABSTRACT_CLASS)) {
  4429. RETURN_FALSE;
  4430. }
  4431. RETURN_BOOL(ce->get_iterator || instanceof_function(ce, zend_ce_traversable));
  4432. }
  4433. /* }}} */
  4434. /* {{{ proto public ReflectionExtension|NULL ReflectionClass::getExtension()
  4435. Returns NULL or the extension the class belongs to */
  4436. ZEND_METHOD(reflection_class, getExtension)
  4437. {
  4438. reflection_object *intern;
  4439. zend_class_entry *ce;
  4440. if (zend_parse_parameters_none() == FAILURE) {
  4441. return;
  4442. }
  4443. GET_REFLECTION_OBJECT_PTR(ce);
  4444. if ((ce->type == ZEND_INTERNAL_CLASS) && ce->info.internal.module) {
  4445. reflection_extension_factory(return_value, ce->info.internal.module->name);
  4446. }
  4447. }
  4448. /* }}} */
  4449. /* {{{ proto public string|false ReflectionClass::getExtensionName()
  4450. Returns false or the name of the extension the class belongs to */
  4451. ZEND_METHOD(reflection_class, getExtensionName)
  4452. {
  4453. reflection_object *intern;
  4454. zend_class_entry *ce;
  4455. if (zend_parse_parameters_none() == FAILURE) {
  4456. return;
  4457. }
  4458. GET_REFLECTION_OBJECT_PTR(ce);
  4459. if ((ce->type == ZEND_INTERNAL_CLASS) && ce->info.internal.module) {
  4460. RETURN_STRING(ce->info.internal.module->name);
  4461. } else {
  4462. RETURN_FALSE;
  4463. }
  4464. }
  4465. /* }}} */
  4466. /* {{{ proto public bool ReflectionClass::inNamespace()
  4467. Returns whether this class is defined in namespace */
  4468. ZEND_METHOD(reflection_class, inNamespace)
  4469. {
  4470. zval *name;
  4471. const char *backslash;
  4472. if (zend_parse_parameters_none() == FAILURE) {
  4473. return;
  4474. }
  4475. if ((name = _default_load_name(getThis())) == NULL) {
  4476. RETURN_FALSE;
  4477. }
  4478. if (Z_TYPE_P(name) == IS_STRING
  4479. && (backslash = zend_memrchr(Z_STRVAL_P(name), '\\', Z_STRLEN_P(name)))
  4480. && backslash > Z_STRVAL_P(name))
  4481. {
  4482. RETURN_TRUE;
  4483. }
  4484. RETURN_FALSE;
  4485. }
  4486. /* }}} */
  4487. /* {{{ proto public string ReflectionClass::getNamespaceName()
  4488. Returns the name of namespace where this class is defined */
  4489. ZEND_METHOD(reflection_class, getNamespaceName)
  4490. {
  4491. zval *name;
  4492. const char *backslash;
  4493. if (zend_parse_parameters_none() == FAILURE) {
  4494. return;
  4495. }
  4496. if ((name = _default_load_name(getThis())) == NULL) {
  4497. RETURN_FALSE;
  4498. }
  4499. if (Z_TYPE_P(name) == IS_STRING
  4500. && (backslash = zend_memrchr(Z_STRVAL_P(name), '\\', Z_STRLEN_P(name)))
  4501. && backslash > Z_STRVAL_P(name))
  4502. {
  4503. RETURN_STRINGL(Z_STRVAL_P(name), backslash - Z_STRVAL_P(name));
  4504. }
  4505. RETURN_EMPTY_STRING();
  4506. }
  4507. /* }}} */
  4508. /* {{{ proto public string ReflectionClass::getShortName()
  4509. Returns the short name of the class (without namespace part) */
  4510. ZEND_METHOD(reflection_class, getShortName)
  4511. {
  4512. zval *name;
  4513. const char *backslash;
  4514. if (zend_parse_parameters_none() == FAILURE) {
  4515. return;
  4516. }
  4517. if ((name = _default_load_name(getThis())) == NULL) {
  4518. RETURN_FALSE;
  4519. }
  4520. if (Z_TYPE_P(name) == IS_STRING
  4521. && (backslash = zend_memrchr(Z_STRVAL_P(name), '\\', Z_STRLEN_P(name)))
  4522. && backslash > Z_STRVAL_P(name))
  4523. {
  4524. RETURN_STRINGL(backslash + 1, Z_STRLEN_P(name) - (backslash - Z_STRVAL_P(name) + 1));
  4525. }
  4526. ZVAL_COPY_DEREF(return_value, name);
  4527. }
  4528. /* }}} */
  4529. /* {{{ proto public static mixed ReflectionObject::export(mixed argument [, bool return]) throws ReflectionException
  4530. Exports a reflection object. Returns the output if TRUE is specified for return, printing it otherwise. */
  4531. ZEND_METHOD(reflection_object, export)
  4532. {
  4533. _reflection_export(INTERNAL_FUNCTION_PARAM_PASSTHRU, reflection_object_ptr, 1);
  4534. }
  4535. /* }}} */
  4536. /* {{{ proto public void ReflectionObject::__construct(mixed argument) throws ReflectionException
  4537. Constructor. Takes an instance as an argument */
  4538. ZEND_METHOD(reflection_object, __construct)
  4539. {
  4540. reflection_class_object_ctor(INTERNAL_FUNCTION_PARAM_PASSTHRU, 1);
  4541. }
  4542. /* }}} */
  4543. /* {{{ proto public static mixed ReflectionProperty::export(mixed class, string name [, bool return]) throws ReflectionException
  4544. Exports a reflection object. Returns the output if TRUE is specified for return, printing it otherwise. */
  4545. ZEND_METHOD(reflection_property, export)
  4546. {
  4547. _reflection_export(INTERNAL_FUNCTION_PARAM_PASSTHRU, reflection_property_ptr, 2);
  4548. }
  4549. /* }}} */
  4550. /* {{{ proto public static mixed ReflectionClassConstant::export(mixed class, string name [, bool return]) throws ReflectionException
  4551. Exports a reflection object. Returns the output if TRUE is specified for return, printing it otherwise. */
  4552. ZEND_METHOD(reflection_class_constant, export)
  4553. {
  4554. _reflection_export(INTERNAL_FUNCTION_PARAM_PASSTHRU, reflection_class_constant_ptr, 2);
  4555. }
  4556. /* }}} */
  4557. /* {{{ proto public void ReflectionProperty::__construct(mixed class, string name)
  4558. Constructor. Throws an Exception in case the given property does not exist */
  4559. ZEND_METHOD(reflection_property, __construct)
  4560. {
  4561. zval propname, cname, *classname;
  4562. zend_string *name;
  4563. int dynam_prop = 0;
  4564. zval *object;
  4565. reflection_object *intern;
  4566. zend_class_entry *ce;
  4567. zend_property_info *property_info = NULL;
  4568. property_reference *reference;
  4569. if (zend_parse_parameters_throw(ZEND_NUM_ARGS(), "zS", &classname, &name) == FAILURE) {
  4570. return;
  4571. }
  4572. object = getThis();
  4573. intern = Z_REFLECTION_P(object);
  4574. /* Find the class entry */
  4575. switch (Z_TYPE_P(classname)) {
  4576. case IS_STRING:
  4577. if ((ce = zend_lookup_class(Z_STR_P(classname))) == NULL) {
  4578. zend_throw_exception_ex(reflection_exception_ptr, 0,
  4579. "Class %s does not exist", Z_STRVAL_P(classname));
  4580. return;
  4581. }
  4582. break;
  4583. case IS_OBJECT:
  4584. ce = Z_OBJCE_P(classname);
  4585. break;
  4586. default:
  4587. _DO_THROW("The parameter class is expected to be either a string or an object");
  4588. /* returns out of this function */
  4589. }
  4590. if ((property_info = zend_hash_find_ptr(&ce->properties_info, name)) == NULL || (property_info->flags & ZEND_ACC_SHADOW)) {
  4591. /* Check for dynamic properties */
  4592. if (property_info == NULL && Z_TYPE_P(classname) == IS_OBJECT && Z_OBJ_HT_P(classname)->get_properties) {
  4593. if (zend_hash_exists(Z_OBJ_HT_P(classname)->get_properties(classname), name)) {
  4594. dynam_prop = 1;
  4595. }
  4596. }
  4597. if (dynam_prop == 0) {
  4598. zend_throw_exception_ex(reflection_exception_ptr, 0, "Property %s::$%s does not exist", ZSTR_VAL(ce->name), ZSTR_VAL(name));
  4599. return;
  4600. }
  4601. }
  4602. if (dynam_prop == 0 && (property_info->flags & ZEND_ACC_PRIVATE) == 0) {
  4603. /* we have to search the class hierarchy for this (implicit) public or protected property */
  4604. zend_class_entry *tmp_ce = ce;
  4605. zend_property_info *tmp_info;
  4606. while (tmp_ce && (tmp_info = zend_hash_find_ptr(&tmp_ce->properties_info, name)) == NULL) {
  4607. ce = tmp_ce;
  4608. property_info = tmp_info;
  4609. tmp_ce = tmp_ce->parent;
  4610. }
  4611. }
  4612. if (dynam_prop == 0) {
  4613. ZVAL_STR_COPY(&cname, property_info->ce->name);
  4614. } else {
  4615. ZVAL_STR_COPY(&cname, ce->name);
  4616. }
  4617. reflection_update_property_class(object, &cname);
  4618. ZVAL_STR_COPY(&propname, name);
  4619. reflection_update_property_name(object, &propname);
  4620. reference = (property_reference*) emalloc(sizeof(property_reference));
  4621. if (dynam_prop) {
  4622. reference->prop.flags = ZEND_ACC_IMPLICIT_PUBLIC;
  4623. reference->prop.name = name;
  4624. reference->prop.doc_comment = NULL;
  4625. reference->prop.ce = ce;
  4626. } else {
  4627. reference->prop = *property_info;
  4628. }
  4629. reference->ce = ce;
  4630. reference->unmangled_name = zend_string_copy(name);
  4631. intern->ptr = reference;
  4632. intern->ref_type = REF_TYPE_PROPERTY;
  4633. intern->ce = ce;
  4634. intern->ignore_visibility = 0;
  4635. }
  4636. /* }}} */
  4637. /* {{{ proto public string ReflectionProperty::__toString()
  4638. Returns a string representation */
  4639. ZEND_METHOD(reflection_property, __toString)
  4640. {
  4641. reflection_object *intern;
  4642. property_reference *ref;
  4643. smart_str str = {0};
  4644. if (zend_parse_parameters_none() == FAILURE) {
  4645. return;
  4646. }
  4647. GET_REFLECTION_OBJECT_PTR(ref);
  4648. _property_string(&str, &ref->prop, ZSTR_VAL(ref->unmangled_name), "");
  4649. RETURN_STR(smart_str_extract(&str));
  4650. }
  4651. /* }}} */
  4652. /* {{{ proto public string ReflectionProperty::getName()
  4653. Returns the class' name */
  4654. ZEND_METHOD(reflection_property, getName)
  4655. {
  4656. if (zend_parse_parameters_none() == FAILURE) {
  4657. return;
  4658. }
  4659. _default_get_name(getThis(), return_value);
  4660. }
  4661. /* }}} */
  4662. static void _property_check_flag(INTERNAL_FUNCTION_PARAMETERS, int mask) /* {{{ */
  4663. {
  4664. reflection_object *intern;
  4665. property_reference *ref;
  4666. if (zend_parse_parameters_none() == FAILURE) {
  4667. return;
  4668. }
  4669. GET_REFLECTION_OBJECT_PTR(ref);
  4670. RETURN_BOOL(ref->prop.flags & mask);
  4671. }
  4672. /* }}} */
  4673. /* {{{ proto public bool ReflectionProperty::isPublic()
  4674. Returns whether this property is public */
  4675. ZEND_METHOD(reflection_property, isPublic)
  4676. {
  4677. _property_check_flag(INTERNAL_FUNCTION_PARAM_PASSTHRU, ZEND_ACC_PUBLIC | ZEND_ACC_IMPLICIT_PUBLIC);
  4678. }
  4679. /* }}} */
  4680. /* {{{ proto public bool ReflectionProperty::isPrivate()
  4681. Returns whether this property is private */
  4682. ZEND_METHOD(reflection_property, isPrivate)
  4683. {
  4684. _property_check_flag(INTERNAL_FUNCTION_PARAM_PASSTHRU, ZEND_ACC_PRIVATE);
  4685. }
  4686. /* }}} */
  4687. /* {{{ proto public bool ReflectionProperty::isProtected()
  4688. Returns whether this property is protected */
  4689. ZEND_METHOD(reflection_property, isProtected)
  4690. {
  4691. _property_check_flag(INTERNAL_FUNCTION_PARAM_PASSTHRU, ZEND_ACC_PROTECTED);
  4692. }
  4693. /* }}} */
  4694. /* {{{ proto public bool ReflectionProperty::isStatic()
  4695. Returns whether this property is static */
  4696. ZEND_METHOD(reflection_property, isStatic)
  4697. {
  4698. _property_check_flag(INTERNAL_FUNCTION_PARAM_PASSTHRU, ZEND_ACC_STATIC);
  4699. }
  4700. /* }}} */
  4701. /* {{{ proto public bool ReflectionProperty::isDefault()
  4702. Returns whether this property is default (declared at compilation time). */
  4703. ZEND_METHOD(reflection_property, isDefault)
  4704. {
  4705. _property_check_flag(INTERNAL_FUNCTION_PARAM_PASSTHRU, ~ZEND_ACC_IMPLICIT_PUBLIC);
  4706. }
  4707. /* }}} */
  4708. /* {{{ proto public int ReflectionProperty::getModifiers()
  4709. Returns a bitfield of the access modifiers for this property */
  4710. ZEND_METHOD(reflection_property, getModifiers)
  4711. {
  4712. reflection_object *intern;
  4713. property_reference *ref;
  4714. uint32_t keep_flags = ZEND_ACC_PPP_MASK | ZEND_ACC_IMPLICIT_PUBLIC | ZEND_ACC_STATIC;
  4715. if (zend_parse_parameters_none() == FAILURE) {
  4716. return;
  4717. }
  4718. GET_REFLECTION_OBJECT_PTR(ref);
  4719. RETURN_LONG((ref->prop.flags & keep_flags));
  4720. }
  4721. /* }}} */
  4722. /* {{{ proto public mixed ReflectionProperty::getValue([stdclass object])
  4723. Returns this property's value */
  4724. ZEND_METHOD(reflection_property, getValue)
  4725. {
  4726. reflection_object *intern;
  4727. property_reference *ref;
  4728. zval *object, *name;
  4729. zval *member_p = NULL;
  4730. GET_REFLECTION_OBJECT_PTR(ref);
  4731. if (!(ref->prop.flags & (ZEND_ACC_PUBLIC | ZEND_ACC_IMPLICIT_PUBLIC)) && intern->ignore_visibility == 0) {
  4732. name = _default_load_name(getThis());
  4733. zend_throw_exception_ex(reflection_exception_ptr, 0,
  4734. "Cannot access non-public member %s::$%s", ZSTR_VAL(intern->ce->name), Z_STRVAL_P(name));
  4735. return;
  4736. }
  4737. if (ref->prop.flags & ZEND_ACC_STATIC) {
  4738. member_p = zend_read_static_property_ex(ref->ce, ref->unmangled_name, 0);
  4739. if (member_p) {
  4740. ZVAL_COPY_DEREF(return_value, member_p);
  4741. }
  4742. } else {
  4743. zval rv;
  4744. if (zend_parse_parameters(ZEND_NUM_ARGS(), "o", &object) == FAILURE) {
  4745. return;
  4746. }
  4747. if (!instanceof_function(Z_OBJCE_P(object), ref->prop.ce)) {
  4748. _DO_THROW("Given object is not an instance of the class this property was declared in");
  4749. /* Returns from this function */
  4750. }
  4751. member_p = zend_read_property_ex(ref->ce, object, ref->unmangled_name, 0, &rv);
  4752. if (member_p != &rv) {
  4753. ZVAL_COPY_DEREF(return_value, member_p);
  4754. } else {
  4755. if (Z_ISREF_P(member_p)) {
  4756. zend_unwrap_reference(member_p);
  4757. }
  4758. ZVAL_COPY_VALUE(return_value, member_p);
  4759. }
  4760. }
  4761. }
  4762. /* }}} */
  4763. /* {{{ proto public void ReflectionProperty::setValue([stdclass object,] mixed value)
  4764. Sets this property's value */
  4765. ZEND_METHOD(reflection_property, setValue)
  4766. {
  4767. reflection_object *intern;
  4768. property_reference *ref;
  4769. zval *object, *name;
  4770. zval *value;
  4771. zval *tmp;
  4772. GET_REFLECTION_OBJECT_PTR(ref);
  4773. if (!(ref->prop.flags & ZEND_ACC_PUBLIC) && intern->ignore_visibility == 0) {
  4774. name = _default_load_name(getThis());
  4775. zend_throw_exception_ex(reflection_exception_ptr, 0,
  4776. "Cannot access non-public member %s::$%s", ZSTR_VAL(intern->ce->name), Z_STRVAL_P(name));
  4777. return;
  4778. }
  4779. if (ref->prop.flags & ZEND_ACC_STATIC) {
  4780. if (zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET, ZEND_NUM_ARGS(), "z", &value) == FAILURE) {
  4781. if (zend_parse_parameters(ZEND_NUM_ARGS(), "zz", &tmp, &value) == FAILURE) {
  4782. return;
  4783. }
  4784. }
  4785. zend_update_static_property_ex(ref->ce, ref->unmangled_name, value);
  4786. } else {
  4787. if (zend_parse_parameters(ZEND_NUM_ARGS(), "oz", &object, &value) == FAILURE) {
  4788. return;
  4789. }
  4790. zend_update_property_ex(ref->ce, object, ref->unmangled_name, value);
  4791. }
  4792. }
  4793. /* }}} */
  4794. /* {{{ proto public ReflectionClass ReflectionProperty::getDeclaringClass()
  4795. Get the declaring class */
  4796. ZEND_METHOD(reflection_property, getDeclaringClass)
  4797. {
  4798. reflection_object *intern;
  4799. property_reference *ref;
  4800. zend_class_entry *tmp_ce, *ce;
  4801. zend_property_info *tmp_info;
  4802. if (zend_parse_parameters_none() == FAILURE) {
  4803. return;
  4804. }
  4805. GET_REFLECTION_OBJECT_PTR(ref);
  4806. ce = tmp_ce = ref->ce;
  4807. while (tmp_ce && (tmp_info = zend_hash_find_ptr(&tmp_ce->properties_info, ref->unmangled_name)) != NULL) {
  4808. if (tmp_info->flags & ZEND_ACC_PRIVATE || tmp_info->flags & ZEND_ACC_SHADOW) {
  4809. /* it's a private property, so it can't be inherited */
  4810. break;
  4811. }
  4812. ce = tmp_ce;
  4813. if (tmp_ce == tmp_info->ce) {
  4814. /* declared in this class, done */
  4815. break;
  4816. }
  4817. tmp_ce = tmp_ce->parent;
  4818. }
  4819. zend_reflection_class_factory(ce, return_value);
  4820. }
  4821. /* }}} */
  4822. /* {{{ proto public string ReflectionProperty::getDocComment()
  4823. Returns the doc comment for this property */
  4824. ZEND_METHOD(reflection_property, getDocComment)
  4825. {
  4826. reflection_object *intern;
  4827. property_reference *ref;
  4828. if (zend_parse_parameters_none() == FAILURE) {
  4829. return;
  4830. }
  4831. GET_REFLECTION_OBJECT_PTR(ref);
  4832. if (ref->prop.doc_comment) {
  4833. RETURN_STR_COPY(ref->prop.doc_comment);
  4834. }
  4835. RETURN_FALSE;
  4836. }
  4837. /* }}} */
  4838. /* {{{ proto public int ReflectionProperty::setAccessible(bool visible)
  4839. Sets whether non-public properties can be requested */
  4840. ZEND_METHOD(reflection_property, setAccessible)
  4841. {
  4842. reflection_object *intern;
  4843. zend_bool visible;
  4844. if (zend_parse_parameters(ZEND_NUM_ARGS(), "b", &visible) == FAILURE) {
  4845. return;
  4846. }
  4847. intern = Z_REFLECTION_P(getThis());
  4848. intern->ignore_visibility = visible;
  4849. }
  4850. /* }}} */
  4851. /* {{{ proto public static mixed ReflectionExtension::export(string name [, bool return]) throws ReflectionException
  4852. Exports a reflection object. Returns the output if TRUE is specified for return, printing it otherwise. */
  4853. ZEND_METHOD(reflection_extension, export)
  4854. {
  4855. _reflection_export(INTERNAL_FUNCTION_PARAM_PASSTHRU, reflection_extension_ptr, 1);
  4856. }
  4857. /* }}} */
  4858. /* {{{ proto public void ReflectionExtension::__construct(string name)
  4859. Constructor. Throws an Exception in case the given extension does not exist */
  4860. ZEND_METHOD(reflection_extension, __construct)
  4861. {
  4862. zval name;
  4863. zval *object;
  4864. char *lcname;
  4865. reflection_object *intern;
  4866. zend_module_entry *module;
  4867. char *name_str;
  4868. size_t name_len;
  4869. ALLOCA_FLAG(use_heap)
  4870. if (zend_parse_parameters_throw(ZEND_NUM_ARGS(), "s", &name_str, &name_len) == FAILURE) {
  4871. return;
  4872. }
  4873. object = getThis();
  4874. intern = Z_REFLECTION_P(object);
  4875. lcname = do_alloca(name_len + 1, use_heap);
  4876. zend_str_tolower_copy(lcname, name_str, name_len);
  4877. if ((module = zend_hash_str_find_ptr(&module_registry, lcname, name_len)) == NULL) {
  4878. free_alloca(lcname, use_heap);
  4879. zend_throw_exception_ex(reflection_exception_ptr, 0,
  4880. "Extension %s does not exist", name_str);
  4881. return;
  4882. }
  4883. free_alloca(lcname, use_heap);
  4884. ZVAL_STRING(&name, module->name);
  4885. reflection_update_property_name(object, &name);
  4886. intern->ptr = module;
  4887. intern->ref_type = REF_TYPE_OTHER;
  4888. intern->ce = NULL;
  4889. }
  4890. /* }}} */
  4891. /* {{{ proto public string ReflectionExtension::__toString()
  4892. Returns a string representation */
  4893. ZEND_METHOD(reflection_extension, __toString)
  4894. {
  4895. reflection_object *intern;
  4896. zend_module_entry *module;
  4897. smart_str str = {0};
  4898. if (zend_parse_parameters_none() == FAILURE) {
  4899. return;
  4900. }
  4901. GET_REFLECTION_OBJECT_PTR(module);
  4902. _extension_string(&str, module, "");
  4903. RETURN_STR(smart_str_extract(&str));
  4904. }
  4905. /* }}} */
  4906. /* {{{ proto public string ReflectionExtension::getName()
  4907. Returns this extension's name */
  4908. ZEND_METHOD(reflection_extension, getName)
  4909. {
  4910. if (zend_parse_parameters_none() == FAILURE) {
  4911. return;
  4912. }
  4913. _default_get_name(getThis(), return_value);
  4914. }
  4915. /* }}} */
  4916. /* {{{ proto public string ReflectionExtension::getVersion()
  4917. Returns this extension's version */
  4918. ZEND_METHOD(reflection_extension, getVersion)
  4919. {
  4920. reflection_object *intern;
  4921. zend_module_entry *module;
  4922. if (zend_parse_parameters_none() == FAILURE) {
  4923. return;
  4924. }
  4925. GET_REFLECTION_OBJECT_PTR(module);
  4926. /* An extension does not necessarily have a version number */
  4927. if (module->version == NO_VERSION_YET) {
  4928. RETURN_NULL();
  4929. } else {
  4930. RETURN_STRING(module->version);
  4931. }
  4932. }
  4933. /* }}} */
  4934. /* {{{ proto public ReflectionFunction[] ReflectionExtension::getFunctions()
  4935. Returns an array of this extension's functions */
  4936. ZEND_METHOD(reflection_extension, getFunctions)
  4937. {
  4938. reflection_object *intern;
  4939. zend_module_entry *module;
  4940. zval function;
  4941. zend_function *fptr;
  4942. if (zend_parse_parameters_none() == FAILURE) {
  4943. return;
  4944. }
  4945. GET_REFLECTION_OBJECT_PTR(module);
  4946. array_init(return_value);
  4947. ZEND_HASH_FOREACH_PTR(CG(function_table), fptr) {
  4948. if (fptr->common.type==ZEND_INTERNAL_FUNCTION
  4949. && fptr->internal_function.module == module) {
  4950. reflection_function_factory(fptr, NULL, &function);
  4951. zend_hash_update(Z_ARRVAL_P(return_value), fptr->common.function_name, &function);
  4952. }
  4953. } ZEND_HASH_FOREACH_END();
  4954. }
  4955. /* }}} */
  4956. static int _addconstant(zval *el, int num_args, va_list args, zend_hash_key *hash_key) /* {{{ */
  4957. {
  4958. zval const_val;
  4959. zend_constant *constant = (zend_constant*)Z_PTR_P(el);
  4960. zval *retval = va_arg(args, zval*);
  4961. int number = va_arg(args, int);
  4962. if (number == ZEND_CONSTANT_MODULE_NUMBER(constant)) {
  4963. ZVAL_COPY_OR_DUP(&const_val, &constant->value);
  4964. zend_hash_update(Z_ARRVAL_P(retval), constant->name, &const_val);
  4965. }
  4966. return 0;
  4967. }
  4968. /* }}} */
  4969. /* {{{ proto public array ReflectionExtension::getConstants()
  4970. Returns an associative array containing this extension's constants and their values */
  4971. ZEND_METHOD(reflection_extension, getConstants)
  4972. {
  4973. reflection_object *intern;
  4974. zend_module_entry *module;
  4975. if (zend_parse_parameters_none() == FAILURE) {
  4976. return;
  4977. }
  4978. GET_REFLECTION_OBJECT_PTR(module);
  4979. array_init(return_value);
  4980. zend_hash_apply_with_arguments(EG(zend_constants), (apply_func_args_t) _addconstant, 2, return_value, module->module_number);
  4981. }
  4982. /* }}} */
  4983. /* {{{ _addinientry */
  4984. static int _addinientry(zval *el, int num_args, va_list args, zend_hash_key *hash_key)
  4985. {
  4986. zend_ini_entry *ini_entry = (zend_ini_entry*)Z_PTR_P(el);
  4987. zval *retval = va_arg(args, zval*);
  4988. int number = va_arg(args, int);
  4989. if (number == ini_entry->module_number) {
  4990. if (ini_entry->value) {
  4991. zval zv;
  4992. ZVAL_STR_COPY(&zv, ini_entry->value);
  4993. zend_symtable_update(Z_ARRVAL_P(retval), ini_entry->name, &zv);
  4994. } else {
  4995. zend_symtable_update(Z_ARRVAL_P(retval), ini_entry->name, &EG(uninitialized_zval));
  4996. }
  4997. }
  4998. return ZEND_HASH_APPLY_KEEP;
  4999. }
  5000. /* }}} */
  5001. /* {{{ proto public array ReflectionExtension::getINIEntries()
  5002. Returns an associative array containing this extension's INI entries and their values */
  5003. ZEND_METHOD(reflection_extension, getINIEntries)
  5004. {
  5005. reflection_object *intern;
  5006. zend_module_entry *module;
  5007. if (zend_parse_parameters_none() == FAILURE) {
  5008. return;
  5009. }
  5010. GET_REFLECTION_OBJECT_PTR(module);
  5011. array_init(return_value);
  5012. zend_hash_apply_with_arguments(EG(ini_directives), (apply_func_args_t) _addinientry, 2, return_value, module->module_number);
  5013. }
  5014. /* }}} */
  5015. /* {{{ add_extension_class */
  5016. static int add_extension_class(zval *zv, int num_args, va_list args, zend_hash_key *hash_key)
  5017. {
  5018. zend_class_entry *ce = Z_PTR_P(zv);
  5019. zval *class_array = va_arg(args, zval*), zclass;
  5020. struct _zend_module_entry *module = va_arg(args, struct _zend_module_entry*);
  5021. int add_reflection_class = va_arg(args, int);
  5022. if ((ce->type == ZEND_INTERNAL_CLASS) && ce->info.internal.module && !strcasecmp(ce->info.internal.module->name, module->name)) {
  5023. zend_string *name;
  5024. if (zend_binary_strcasecmp(ZSTR_VAL(ce->name), ZSTR_LEN(ce->name), ZSTR_VAL(hash_key->key), ZSTR_LEN(hash_key->key))) {
  5025. /* This is an class alias, use alias name */
  5026. name = hash_key->key;
  5027. } else {
  5028. /* Use class name */
  5029. name = ce->name;
  5030. }
  5031. if (add_reflection_class) {
  5032. zend_reflection_class_factory(ce, &zclass);
  5033. zend_hash_update(Z_ARRVAL_P(class_array), name, &zclass);
  5034. } else {
  5035. add_next_index_str(class_array, zend_string_copy(name));
  5036. }
  5037. }
  5038. return ZEND_HASH_APPLY_KEEP;
  5039. }
  5040. /* }}} */
  5041. /* {{{ proto public ReflectionClass[] ReflectionExtension::getClasses()
  5042. Returns an array containing ReflectionClass objects for all classes of this extension */
  5043. ZEND_METHOD(reflection_extension, getClasses)
  5044. {
  5045. reflection_object *intern;
  5046. zend_module_entry *module;
  5047. if (zend_parse_parameters_none() == FAILURE) {
  5048. return;
  5049. }
  5050. GET_REFLECTION_OBJECT_PTR(module);
  5051. array_init(return_value);
  5052. zend_hash_apply_with_arguments(EG(class_table), (apply_func_args_t) add_extension_class, 3, return_value, module, 1);
  5053. }
  5054. /* }}} */
  5055. /* {{{ proto public array ReflectionExtension::getClassNames()
  5056. Returns an array containing all names of all classes of this extension */
  5057. ZEND_METHOD(reflection_extension, getClassNames)
  5058. {
  5059. reflection_object *intern;
  5060. zend_module_entry *module;
  5061. if (zend_parse_parameters_none() == FAILURE) {
  5062. return;
  5063. }
  5064. GET_REFLECTION_OBJECT_PTR(module);
  5065. array_init(return_value);
  5066. zend_hash_apply_with_arguments(EG(class_table), (apply_func_args_t) add_extension_class, 3, return_value, module, 0);
  5067. }
  5068. /* }}} */
  5069. /* {{{ proto public array ReflectionExtension::getDependencies()
  5070. Returns an array containing all names of all extensions this extension depends on */
  5071. ZEND_METHOD(reflection_extension, getDependencies)
  5072. {
  5073. reflection_object *intern;
  5074. zend_module_entry *module;
  5075. const zend_module_dep *dep;
  5076. if (zend_parse_parameters_none() == FAILURE) {
  5077. return;
  5078. }
  5079. GET_REFLECTION_OBJECT_PTR(module);
  5080. dep = module->deps;
  5081. if (!dep)
  5082. {
  5083. ZVAL_EMPTY_ARRAY(return_value);
  5084. return;
  5085. }
  5086. array_init(return_value);
  5087. while(dep->name) {
  5088. zend_string *relation;
  5089. char *rel_type;
  5090. size_t len = 0;
  5091. switch(dep->type) {
  5092. case MODULE_DEP_REQUIRED:
  5093. rel_type = "Required";
  5094. len += sizeof("Required") - 1;
  5095. break;
  5096. case MODULE_DEP_CONFLICTS:
  5097. rel_type = "Conflicts";
  5098. len += sizeof("Conflicts") - 1;
  5099. break;
  5100. case MODULE_DEP_OPTIONAL:
  5101. rel_type = "Optional";
  5102. len += sizeof("Optional") - 1;
  5103. break;
  5104. default:
  5105. rel_type = "Error"; /* shouldn't happen */
  5106. len += sizeof("Error") - 1;
  5107. break;
  5108. }
  5109. if (dep->rel) {
  5110. len += strlen(dep->rel) + 1;
  5111. }
  5112. if (dep->version) {
  5113. len += strlen(dep->version) + 1;
  5114. }
  5115. relation = zend_string_alloc(len, 0);
  5116. snprintf(ZSTR_VAL(relation), ZSTR_LEN(relation) + 1, "%s%s%s%s%s",
  5117. rel_type,
  5118. dep->rel ? " " : "",
  5119. dep->rel ? dep->rel : "",
  5120. dep->version ? " " : "",
  5121. dep->version ? dep->version : "");
  5122. add_assoc_str(return_value, dep->name, relation);
  5123. dep++;
  5124. }
  5125. }
  5126. /* }}} */
  5127. /* {{{ proto public void ReflectionExtension::info()
  5128. Prints phpinfo block for the extension */
  5129. ZEND_METHOD(reflection_extension, info)
  5130. {
  5131. reflection_object *intern;
  5132. zend_module_entry *module;
  5133. if (zend_parse_parameters_none() == FAILURE) {
  5134. return;
  5135. }
  5136. GET_REFLECTION_OBJECT_PTR(module);
  5137. php_info_print_module(module);
  5138. }
  5139. /* }}} */
  5140. /* {{{ proto public bool ReflectionExtension::isPersistent()
  5141. Returns whether this extension is persistent */
  5142. ZEND_METHOD(reflection_extension, isPersistent)
  5143. {
  5144. reflection_object *intern;
  5145. zend_module_entry *module;
  5146. if (zend_parse_parameters_none() == FAILURE) {
  5147. return;
  5148. }
  5149. GET_REFLECTION_OBJECT_PTR(module);
  5150. RETURN_BOOL(module->type == MODULE_PERSISTENT);
  5151. }
  5152. /* }}} */
  5153. /* {{{ proto public bool ReflectionExtension::isTemporary()
  5154. Returns whether this extension is temporary */
  5155. ZEND_METHOD(reflection_extension, isTemporary)
  5156. {
  5157. reflection_object *intern;
  5158. zend_module_entry *module;
  5159. if (zend_parse_parameters_none() == FAILURE) {
  5160. return;
  5161. }
  5162. GET_REFLECTION_OBJECT_PTR(module);
  5163. RETURN_BOOL(module->type == MODULE_TEMPORARY);
  5164. }
  5165. /* }}} */
  5166. /* {{{ proto public static mixed ReflectionZendExtension::export(string name [, bool return]) throws ReflectionException
  5167. * Exports a reflection object. Returns the output if TRUE is specified for return, printing it otherwise. */
  5168. ZEND_METHOD(reflection_zend_extension, export)
  5169. {
  5170. _reflection_export(INTERNAL_FUNCTION_PARAM_PASSTHRU, reflection_zend_extension_ptr, 1);
  5171. }
  5172. /* }}} */
  5173. /* {{{ proto public void ReflectionZendExtension::__construct(string name)
  5174. Constructor. Throws an Exception in case the given Zend extension does not exist */
  5175. ZEND_METHOD(reflection_zend_extension, __construct)
  5176. {
  5177. zval name;
  5178. zval *object;
  5179. reflection_object *intern;
  5180. zend_extension *extension;
  5181. char *name_str;
  5182. size_t name_len;
  5183. if (zend_parse_parameters_throw(ZEND_NUM_ARGS(), "s", &name_str, &name_len) == FAILURE) {
  5184. return;
  5185. }
  5186. object = getThis();
  5187. intern = Z_REFLECTION_P(object);
  5188. extension = zend_get_extension(name_str);
  5189. if (!extension) {
  5190. zend_throw_exception_ex(reflection_exception_ptr, 0,
  5191. "Zend Extension %s does not exist", name_str);
  5192. return;
  5193. }
  5194. ZVAL_STRING(&name, extension->name);
  5195. reflection_update_property_name(object, &name);
  5196. intern->ptr = extension;
  5197. intern->ref_type = REF_TYPE_OTHER;
  5198. intern->ce = NULL;
  5199. }
  5200. /* }}} */
  5201. /* {{{ proto public string ReflectionZendExtension::__toString()
  5202. Returns a string representation */
  5203. ZEND_METHOD(reflection_zend_extension, __toString)
  5204. {
  5205. reflection_object *intern;
  5206. zend_extension *extension;
  5207. smart_str str = {0};
  5208. if (zend_parse_parameters_none() == FAILURE) {
  5209. return;
  5210. }
  5211. GET_REFLECTION_OBJECT_PTR(extension);
  5212. _zend_extension_string(&str, extension, "");
  5213. RETURN_STR(smart_str_extract(&str));
  5214. }
  5215. /* }}} */
  5216. /* {{{ proto public string ReflectionZendExtension::getName()
  5217. Returns the name of this Zend extension */
  5218. ZEND_METHOD(reflection_zend_extension, getName)
  5219. {
  5220. reflection_object *intern;
  5221. zend_extension *extension;
  5222. if (zend_parse_parameters_none() == FAILURE) {
  5223. return;
  5224. }
  5225. GET_REFLECTION_OBJECT_PTR(extension);
  5226. RETURN_STRING(extension->name);
  5227. }
  5228. /* }}} */
  5229. /* {{{ proto public string ReflectionZendExtension::getVersion()
  5230. Returns the version information of this Zend extension */
  5231. ZEND_METHOD(reflection_zend_extension, getVersion)
  5232. {
  5233. reflection_object *intern;
  5234. zend_extension *extension;
  5235. if (zend_parse_parameters_none() == FAILURE) {
  5236. return;
  5237. }
  5238. GET_REFLECTION_OBJECT_PTR(extension);
  5239. if (extension->version) {
  5240. RETURN_STRING(extension->version);
  5241. } else {
  5242. RETURN_EMPTY_STRING();
  5243. }
  5244. }
  5245. /* }}} */
  5246. /* {{{ proto public void ReflectionZendExtension::getAuthor()
  5247. * Returns the name of this Zend extension's author */
  5248. ZEND_METHOD(reflection_zend_extension, getAuthor)
  5249. {
  5250. reflection_object *intern;
  5251. zend_extension *extension;
  5252. if (zend_parse_parameters_none() == FAILURE) {
  5253. return;
  5254. }
  5255. GET_REFLECTION_OBJECT_PTR(extension);
  5256. if (extension->author) {
  5257. RETURN_STRING(extension->author);
  5258. } else {
  5259. RETURN_EMPTY_STRING();
  5260. }
  5261. }
  5262. /* }}} */
  5263. /* {{{ proto public void ReflectionZendExtension::getURL()
  5264. Returns this Zend extension's URL*/
  5265. ZEND_METHOD(reflection_zend_extension, getURL)
  5266. {
  5267. reflection_object *intern;
  5268. zend_extension *extension;
  5269. if (zend_parse_parameters_none() == FAILURE) {
  5270. return;
  5271. }
  5272. GET_REFLECTION_OBJECT_PTR(extension);
  5273. if (extension->URL) {
  5274. RETURN_STRING(extension->URL);
  5275. } else {
  5276. RETURN_EMPTY_STRING();
  5277. }
  5278. }
  5279. /* }}} */
  5280. /* {{{ proto public void ReflectionZendExtension::getCopyright()
  5281. Returns this Zend extension's copyright information */
  5282. ZEND_METHOD(reflection_zend_extension, getCopyright)
  5283. {
  5284. reflection_object *intern;
  5285. zend_extension *extension;
  5286. if (zend_parse_parameters_none() == FAILURE) {
  5287. return;
  5288. }
  5289. GET_REFLECTION_OBJECT_PTR(extension);
  5290. if (extension->copyright) {
  5291. RETURN_STRING(extension->copyright);
  5292. } else {
  5293. RETURN_EMPTY_STRING();
  5294. }
  5295. }
  5296. /* }}} */
  5297. /* {{{ method tables */
  5298. static const zend_function_entry reflection_exception_functions[] = {
  5299. PHP_FE_END
  5300. };
  5301. ZEND_BEGIN_ARG_INFO(arginfo_reflection__void, 0)
  5302. ZEND_END_ARG_INFO()
  5303. ZEND_BEGIN_ARG_INFO(arginfo_reflection_getModifierNames, 0)
  5304. ZEND_ARG_INFO(0, modifiers)
  5305. ZEND_END_ARG_INFO()
  5306. ZEND_BEGIN_ARG_INFO_EX(arginfo_reflection_export, 0, 0, 1)
  5307. ZEND_ARG_OBJ_INFO(0, reflector, Reflector, 0)
  5308. ZEND_ARG_INFO(0, return)
  5309. ZEND_END_ARG_INFO()
  5310. static const zend_function_entry reflection_functions[] = {
  5311. ZEND_ME(reflection, getModifierNames, arginfo_reflection_getModifierNames, ZEND_ACC_PUBLIC|ZEND_ACC_STATIC)
  5312. ZEND_ME(reflection, export, arginfo_reflection_export, ZEND_ACC_PUBLIC|ZEND_ACC_STATIC)
  5313. PHP_FE_END
  5314. };
  5315. static const zend_function_entry reflector_functions[] = {
  5316. ZEND_FENTRY(export, NULL, NULL, ZEND_ACC_STATIC|ZEND_ACC_ABSTRACT|ZEND_ACC_PUBLIC)
  5317. ZEND_ABSTRACT_ME(reflector, __toString, arginfo_reflection__void)
  5318. PHP_FE_END
  5319. };
  5320. ZEND_BEGIN_ARG_INFO_EX(arginfo_reflection_function_export, 0, 0, 1)
  5321. ZEND_ARG_INFO(0, name)
  5322. ZEND_ARG_INFO(0, return)
  5323. ZEND_END_ARG_INFO()
  5324. ZEND_BEGIN_ARG_INFO(arginfo_reflection_function___construct, 0)
  5325. ZEND_ARG_INFO(0, name)
  5326. ZEND_END_ARG_INFO()
  5327. ZEND_BEGIN_ARG_INFO_EX(arginfo_reflection_function_invoke, 0, 0, 0)
  5328. ZEND_ARG_INFO(0, args)
  5329. ZEND_END_ARG_INFO()
  5330. ZEND_BEGIN_ARG_INFO(arginfo_reflection_function_invokeArgs, 0)
  5331. ZEND_ARG_ARRAY_INFO(0, args, 0)
  5332. ZEND_END_ARG_INFO()
  5333. static const zend_function_entry reflection_function_abstract_functions[] = {
  5334. ZEND_ME(reflection, __clone, arginfo_reflection__void, ZEND_ACC_PRIVATE|ZEND_ACC_FINAL)
  5335. ZEND_ME(reflection_function, inNamespace, arginfo_reflection__void, 0)
  5336. ZEND_ME(reflection_function, isClosure, arginfo_reflection__void, 0)
  5337. ZEND_ME(reflection_function, isDeprecated, arginfo_reflection__void, 0)
  5338. ZEND_ME(reflection_function, isInternal, arginfo_reflection__void, 0)
  5339. ZEND_ME(reflection_function, isUserDefined, arginfo_reflection__void, 0)
  5340. ZEND_ME(reflection_function, isGenerator, arginfo_reflection__void, 0)
  5341. ZEND_ME(reflection_function, isVariadic, arginfo_reflection__void, 0)
  5342. ZEND_ME(reflection_function, getClosureThis, arginfo_reflection__void, 0)
  5343. ZEND_ME(reflection_function, getClosureScopeClass, arginfo_reflection__void, 0)
  5344. ZEND_ME(reflection_function, getDocComment, arginfo_reflection__void, 0)
  5345. ZEND_ME(reflection_function, getEndLine, arginfo_reflection__void, 0)
  5346. ZEND_ME(reflection_function, getExtension, arginfo_reflection__void, 0)
  5347. ZEND_ME(reflection_function, getExtensionName, arginfo_reflection__void, 0)
  5348. ZEND_ME(reflection_function, getFileName, arginfo_reflection__void, 0)
  5349. ZEND_ME(reflection_function, getName, arginfo_reflection__void, 0)
  5350. ZEND_ME(reflection_function, getNamespaceName, arginfo_reflection__void, 0)
  5351. ZEND_ME(reflection_function, getNumberOfParameters, arginfo_reflection__void, 0)
  5352. ZEND_ME(reflection_function, getNumberOfRequiredParameters, arginfo_reflection__void, 0)
  5353. ZEND_ME(reflection_function, getParameters, arginfo_reflection__void, 0)
  5354. ZEND_ME(reflection_function, getShortName, arginfo_reflection__void, 0)
  5355. ZEND_ME(reflection_function, getStartLine, arginfo_reflection__void, 0)
  5356. ZEND_ME(reflection_function, getStaticVariables, arginfo_reflection__void, 0)
  5357. ZEND_ME(reflection_function, returnsReference, arginfo_reflection__void, 0)
  5358. ZEND_ME(reflection_function, hasReturnType, arginfo_reflection__void, 0)
  5359. ZEND_ME(reflection_function, getReturnType, arginfo_reflection__void, 0)
  5360. PHP_FE_END
  5361. };
  5362. static const zend_function_entry reflection_function_functions[] = {
  5363. ZEND_ME(reflection_function, __construct, arginfo_reflection_function___construct, 0)
  5364. ZEND_ME(reflection_function, __toString, arginfo_reflection__void, 0)
  5365. ZEND_ME(reflection_function, export, arginfo_reflection_function_export, ZEND_ACC_STATIC|ZEND_ACC_PUBLIC)
  5366. ZEND_ME(reflection_function, isDisabled, arginfo_reflection__void, 0)
  5367. ZEND_ME(reflection_function, invoke, arginfo_reflection_function_invoke, 0)
  5368. ZEND_ME(reflection_function, invokeArgs, arginfo_reflection_function_invokeArgs, 0)
  5369. ZEND_ME(reflection_function, getClosure, arginfo_reflection__void, 0)
  5370. PHP_FE_END
  5371. };
  5372. ZEND_BEGIN_ARG_INFO(arginfo_reflection_generator___construct, 0)
  5373. ZEND_ARG_INFO(0, generator)
  5374. ZEND_END_ARG_INFO()
  5375. ZEND_BEGIN_ARG_INFO_EX(arginfo_reflection_generator_trace, 0, 0, 0)
  5376. ZEND_ARG_INFO(0, options)
  5377. ZEND_END_ARG_INFO()
  5378. static const zend_function_entry reflection_generator_functions[] = {
  5379. ZEND_ME(reflection_generator, __construct, arginfo_reflection_generator___construct, 0)
  5380. ZEND_ME(reflection_generator, getExecutingLine, arginfo_reflection__void, 0)
  5381. ZEND_ME(reflection_generator, getExecutingFile, arginfo_reflection__void, 0)
  5382. ZEND_ME(reflection_generator, getTrace, arginfo_reflection_generator_trace, 0)
  5383. ZEND_ME(reflection_generator, getFunction, arginfo_reflection__void, 0)
  5384. ZEND_ME(reflection_generator, getThis, arginfo_reflection__void, 0)
  5385. ZEND_ME(reflection_generator, getExecutingGenerator, arginfo_reflection__void, 0)
  5386. PHP_FE_END
  5387. };
  5388. ZEND_BEGIN_ARG_INFO_EX(arginfo_reflection_method_export, 0, 0, 2)
  5389. ZEND_ARG_INFO(0, class)
  5390. ZEND_ARG_INFO(0, name)
  5391. ZEND_ARG_INFO(0, return)
  5392. ZEND_END_ARG_INFO()
  5393. ZEND_BEGIN_ARG_INFO_EX(arginfo_reflection_method___construct, 0, 0, 1)
  5394. ZEND_ARG_INFO(0, class_or_method)
  5395. ZEND_ARG_INFO(0, name)
  5396. ZEND_END_ARG_INFO()
  5397. ZEND_BEGIN_ARG_INFO(arginfo_reflection_method_invoke, 0)
  5398. ZEND_ARG_INFO(0, object)
  5399. ZEND_ARG_INFO(0, args)
  5400. ZEND_END_ARG_INFO()
  5401. ZEND_BEGIN_ARG_INFO(arginfo_reflection_method_invokeArgs, 0)
  5402. ZEND_ARG_INFO(0, object)
  5403. ZEND_ARG_ARRAY_INFO(0, args, 0)
  5404. ZEND_END_ARG_INFO()
  5405. ZEND_BEGIN_ARG_INFO(arginfo_reflection_method_setAccessible, 0)
  5406. ZEND_ARG_INFO(0, value)
  5407. ZEND_END_ARG_INFO()
  5408. ZEND_BEGIN_ARG_INFO(arginfo_reflection_method_getClosure, 0)
  5409. ZEND_ARG_INFO(0, object)
  5410. ZEND_END_ARG_INFO()
  5411. static const zend_function_entry reflection_method_functions[] = {
  5412. ZEND_ME(reflection_method, export, arginfo_reflection_method_export, ZEND_ACC_STATIC|ZEND_ACC_PUBLIC)
  5413. ZEND_ME(reflection_method, __construct, arginfo_reflection_method___construct, 0)
  5414. ZEND_ME(reflection_method, __toString, arginfo_reflection__void, 0)
  5415. ZEND_ME(reflection_method, isPublic, arginfo_reflection__void, 0)
  5416. ZEND_ME(reflection_method, isPrivate, arginfo_reflection__void, 0)
  5417. ZEND_ME(reflection_method, isProtected, arginfo_reflection__void, 0)
  5418. ZEND_ME(reflection_method, isAbstract, arginfo_reflection__void, 0)
  5419. ZEND_ME(reflection_method, isFinal, arginfo_reflection__void, 0)
  5420. ZEND_ME(reflection_method, isStatic, arginfo_reflection__void, 0)
  5421. ZEND_ME(reflection_method, isConstructor, arginfo_reflection__void, 0)
  5422. ZEND_ME(reflection_method, isDestructor, arginfo_reflection__void, 0)
  5423. ZEND_ME(reflection_method, getClosure, arginfo_reflection_method_getClosure, 0)
  5424. ZEND_ME(reflection_method, getModifiers, arginfo_reflection__void, 0)
  5425. ZEND_ME(reflection_method, invoke, arginfo_reflection_method_invoke, 0)
  5426. ZEND_ME(reflection_method, invokeArgs, arginfo_reflection_method_invokeArgs, 0)
  5427. ZEND_ME(reflection_method, getDeclaringClass, arginfo_reflection__void, 0)
  5428. ZEND_ME(reflection_method, getPrototype, arginfo_reflection__void, 0)
  5429. ZEND_ME(reflection_method, setAccessible, arginfo_reflection_method_setAccessible, 0)
  5430. PHP_FE_END
  5431. };
  5432. ZEND_BEGIN_ARG_INFO_EX(arginfo_reflection_class_export, 0, 0, 1)
  5433. ZEND_ARG_INFO(0, argument)
  5434. ZEND_ARG_INFO(0, return)
  5435. ZEND_END_ARG_INFO()
  5436. ZEND_BEGIN_ARG_INFO(arginfo_reflection_class___construct, 0)
  5437. ZEND_ARG_INFO(0, argument)
  5438. ZEND_END_ARG_INFO()
  5439. ZEND_BEGIN_ARG_INFO_EX(arginfo_reflection_class_getStaticPropertyValue, 0, 0, 1)
  5440. ZEND_ARG_INFO(0, name)
  5441. ZEND_ARG_INFO(0, default)
  5442. ZEND_END_ARG_INFO()
  5443. ZEND_BEGIN_ARG_INFO(arginfo_reflection_class_setStaticPropertyValue, 0)
  5444. ZEND_ARG_INFO(0, name)
  5445. ZEND_ARG_INFO(0, value)
  5446. ZEND_END_ARG_INFO()
  5447. ZEND_BEGIN_ARG_INFO(arginfo_reflection_class_hasMethod, 0)
  5448. ZEND_ARG_INFO(0, name)
  5449. ZEND_END_ARG_INFO()
  5450. ZEND_BEGIN_ARG_INFO(arginfo_reflection_class_getMethod, 0)
  5451. ZEND_ARG_INFO(0, name)
  5452. ZEND_END_ARG_INFO()
  5453. ZEND_BEGIN_ARG_INFO_EX(arginfo_reflection_class_getMethods, 0, 0, 0)
  5454. ZEND_ARG_INFO(0, filter)
  5455. ZEND_END_ARG_INFO()
  5456. ZEND_BEGIN_ARG_INFO(arginfo_reflection_class_hasProperty, 0)
  5457. ZEND_ARG_INFO(0, name)
  5458. ZEND_END_ARG_INFO()
  5459. ZEND_BEGIN_ARG_INFO(arginfo_reflection_class_getProperty, 0)
  5460. ZEND_ARG_INFO(0, name)
  5461. ZEND_END_ARG_INFO()
  5462. ZEND_BEGIN_ARG_INFO_EX(arginfo_reflection_class_getProperties, 0, 0, 0)
  5463. ZEND_ARG_INFO(0, filter)
  5464. ZEND_END_ARG_INFO()
  5465. ZEND_BEGIN_ARG_INFO(arginfo_reflection_class_hasConstant, 0)
  5466. ZEND_ARG_INFO(0, name)
  5467. ZEND_END_ARG_INFO()
  5468. ZEND_BEGIN_ARG_INFO(arginfo_reflection_class_getConstant, 0)
  5469. ZEND_ARG_INFO(0, name)
  5470. ZEND_END_ARG_INFO()
  5471. ZEND_BEGIN_ARG_INFO(arginfo_reflection_class_isInstance, 0)
  5472. ZEND_ARG_INFO(0, object)
  5473. ZEND_END_ARG_INFO()
  5474. ZEND_BEGIN_ARG_INFO(arginfo_reflection_class_newInstance, 0)
  5475. ZEND_ARG_INFO(0, args)
  5476. ZEND_END_ARG_INFO()
  5477. ZEND_BEGIN_ARG_INFO(arginfo_reflection_class_newInstanceWithoutConstructor, 0)
  5478. ZEND_END_ARG_INFO()
  5479. ZEND_BEGIN_ARG_INFO_EX(arginfo_reflection_class_newInstanceArgs, 0, 0, 0)
  5480. ZEND_ARG_ARRAY_INFO(0, args, 0)
  5481. ZEND_END_ARG_INFO()
  5482. ZEND_BEGIN_ARG_INFO(arginfo_reflection_class_isSubclassOf, 0)
  5483. ZEND_ARG_INFO(0, class)
  5484. ZEND_END_ARG_INFO()
  5485. ZEND_BEGIN_ARG_INFO(arginfo_reflection_class_implementsInterface, 0)
  5486. ZEND_ARG_INFO(0, interface)
  5487. ZEND_END_ARG_INFO()
  5488. static const zend_function_entry reflection_class_functions[] = {
  5489. ZEND_ME(reflection, __clone, arginfo_reflection__void, ZEND_ACC_PRIVATE|ZEND_ACC_FINAL)
  5490. ZEND_ME(reflection_class, export, arginfo_reflection_class_export, ZEND_ACC_STATIC|ZEND_ACC_PUBLIC)
  5491. ZEND_ME(reflection_class, __construct, arginfo_reflection_class___construct, 0)
  5492. ZEND_ME(reflection_class, __toString, arginfo_reflection__void, 0)
  5493. ZEND_ME(reflection_class, getName, arginfo_reflection__void, 0)
  5494. ZEND_ME(reflection_class, isInternal, arginfo_reflection__void, 0)
  5495. ZEND_ME(reflection_class, isUserDefined, arginfo_reflection__void, 0)
  5496. ZEND_ME(reflection_class, isAnonymous, arginfo_reflection__void, 0)
  5497. ZEND_ME(reflection_class, isInstantiable, arginfo_reflection__void, 0)
  5498. ZEND_ME(reflection_class, isCloneable, arginfo_reflection__void, 0)
  5499. ZEND_ME(reflection_class, getFileName, arginfo_reflection__void, 0)
  5500. ZEND_ME(reflection_class, getStartLine, arginfo_reflection__void, 0)
  5501. ZEND_ME(reflection_class, getEndLine, arginfo_reflection__void, 0)
  5502. ZEND_ME(reflection_class, getDocComment, arginfo_reflection__void, 0)
  5503. ZEND_ME(reflection_class, getConstructor, arginfo_reflection__void, 0)
  5504. ZEND_ME(reflection_class, hasMethod, arginfo_reflection_class_hasMethod, 0)
  5505. ZEND_ME(reflection_class, getMethod, arginfo_reflection_class_getMethod, 0)
  5506. ZEND_ME(reflection_class, getMethods, arginfo_reflection_class_getMethods, 0)
  5507. ZEND_ME(reflection_class, hasProperty, arginfo_reflection_class_hasProperty, 0)
  5508. ZEND_ME(reflection_class, getProperty, arginfo_reflection_class_getProperty, 0)
  5509. ZEND_ME(reflection_class, getProperties, arginfo_reflection_class_getProperties, 0)
  5510. ZEND_ME(reflection_class, hasConstant, arginfo_reflection_class_hasConstant, 0)
  5511. ZEND_ME(reflection_class, getConstants, arginfo_reflection__void, 0)
  5512. ZEND_ME(reflection_class, getReflectionConstants, arginfo_reflection__void, 0)
  5513. ZEND_ME(reflection_class, getConstant, arginfo_reflection_class_getConstant, 0)
  5514. ZEND_ME(reflection_class, getReflectionConstant, arginfo_reflection_class_getConstant, 0)
  5515. ZEND_ME(reflection_class, getInterfaces, arginfo_reflection__void, 0)
  5516. ZEND_ME(reflection_class, getInterfaceNames, arginfo_reflection__void, 0)
  5517. ZEND_ME(reflection_class, isInterface, arginfo_reflection__void, 0)
  5518. ZEND_ME(reflection_class, getTraits, arginfo_reflection__void, 0)
  5519. ZEND_ME(reflection_class, getTraitNames, arginfo_reflection__void, 0)
  5520. ZEND_ME(reflection_class, getTraitAliases, arginfo_reflection__void, 0)
  5521. ZEND_ME(reflection_class, isTrait, arginfo_reflection__void, 0)
  5522. ZEND_ME(reflection_class, isAbstract, arginfo_reflection__void, 0)
  5523. ZEND_ME(reflection_class, isFinal, arginfo_reflection__void, 0)
  5524. ZEND_ME(reflection_class, getModifiers, arginfo_reflection__void, 0)
  5525. ZEND_ME(reflection_class, isInstance, arginfo_reflection_class_isInstance, 0)
  5526. ZEND_ME(reflection_class, newInstance, arginfo_reflection_class_newInstance, 0)
  5527. ZEND_ME(reflection_class, newInstanceWithoutConstructor, arginfo_reflection_class_newInstanceWithoutConstructor, 0)
  5528. ZEND_ME(reflection_class, newInstanceArgs, arginfo_reflection_class_newInstanceArgs, 0)
  5529. ZEND_ME(reflection_class, getParentClass, arginfo_reflection__void, 0)
  5530. ZEND_ME(reflection_class, isSubclassOf, arginfo_reflection_class_isSubclassOf, 0)
  5531. ZEND_ME(reflection_class, getStaticProperties, arginfo_reflection__void, 0)
  5532. ZEND_ME(reflection_class, getStaticPropertyValue, arginfo_reflection_class_getStaticPropertyValue, 0)
  5533. ZEND_ME(reflection_class, setStaticPropertyValue, arginfo_reflection_class_setStaticPropertyValue, 0)
  5534. ZEND_ME(reflection_class, getDefaultProperties, arginfo_reflection__void, 0)
  5535. ZEND_ME(reflection_class, isIterable, arginfo_reflection__void, 0)
  5536. ZEND_MALIAS(reflection_class, isIterateable, isIterable, arginfo_reflection__void, 0)
  5537. ZEND_ME(reflection_class, implementsInterface, arginfo_reflection_class_implementsInterface, 0)
  5538. ZEND_ME(reflection_class, getExtension, arginfo_reflection__void, 0)
  5539. ZEND_ME(reflection_class, getExtensionName, arginfo_reflection__void, 0)
  5540. ZEND_ME(reflection_class, inNamespace, arginfo_reflection__void, 0)
  5541. ZEND_ME(reflection_class, getNamespaceName, arginfo_reflection__void, 0)
  5542. ZEND_ME(reflection_class, getShortName, arginfo_reflection__void, 0)
  5543. PHP_FE_END
  5544. };
  5545. ZEND_BEGIN_ARG_INFO_EX(arginfo_reflection_object_export, 0, 0, 1)
  5546. ZEND_ARG_INFO(0, argument)
  5547. ZEND_ARG_INFO(0, return)
  5548. ZEND_END_ARG_INFO()
  5549. ZEND_BEGIN_ARG_INFO(arginfo_reflection_object___construct, 0)
  5550. ZEND_ARG_INFO(0, argument)
  5551. ZEND_END_ARG_INFO()
  5552. static const zend_function_entry reflection_object_functions[] = {
  5553. ZEND_ME(reflection_object, export, arginfo_reflection_object_export, ZEND_ACC_STATIC|ZEND_ACC_PUBLIC)
  5554. ZEND_ME(reflection_object, __construct, arginfo_reflection_object___construct, 0)
  5555. PHP_FE_END
  5556. };
  5557. ZEND_BEGIN_ARG_INFO_EX(arginfo_reflection_property_export, 0, 0, 2)
  5558. ZEND_ARG_INFO(0, class)
  5559. ZEND_ARG_INFO(0, name)
  5560. ZEND_ARG_INFO(0, return)
  5561. ZEND_END_ARG_INFO()
  5562. ZEND_BEGIN_ARG_INFO_EX(arginfo_reflection_property___construct, 0, 0, 2)
  5563. ZEND_ARG_INFO(0, class)
  5564. ZEND_ARG_INFO(0, name)
  5565. ZEND_END_ARG_INFO()
  5566. ZEND_BEGIN_ARG_INFO_EX(arginfo_reflection_property_getValue, 0, 0, 0)
  5567. ZEND_ARG_INFO(0, object)
  5568. ZEND_END_ARG_INFO()
  5569. ZEND_BEGIN_ARG_INFO_EX(arginfo_reflection_property_setValue, 0, 0, 1)
  5570. ZEND_ARG_INFO(0, object)
  5571. ZEND_ARG_INFO(0, value)
  5572. ZEND_END_ARG_INFO()
  5573. ZEND_BEGIN_ARG_INFO(arginfo_reflection_property_setAccessible, 0)
  5574. ZEND_ARG_INFO(0, visible)
  5575. ZEND_END_ARG_INFO()
  5576. static const zend_function_entry reflection_property_functions[] = {
  5577. ZEND_ME(reflection, __clone, arginfo_reflection__void, ZEND_ACC_PRIVATE|ZEND_ACC_FINAL)
  5578. ZEND_ME(reflection_property, export, arginfo_reflection_property_export, ZEND_ACC_STATIC|ZEND_ACC_PUBLIC)
  5579. ZEND_ME(reflection_property, __construct, arginfo_reflection_property___construct, 0)
  5580. ZEND_ME(reflection_property, __toString, arginfo_reflection__void, 0)
  5581. ZEND_ME(reflection_property, getName, arginfo_reflection__void, 0)
  5582. ZEND_ME(reflection_property, getValue, arginfo_reflection_property_getValue, 0)
  5583. ZEND_ME(reflection_property, setValue, arginfo_reflection_property_setValue, 0)
  5584. ZEND_ME(reflection_property, isPublic, arginfo_reflection__void, 0)
  5585. ZEND_ME(reflection_property, isPrivate, arginfo_reflection__void, 0)
  5586. ZEND_ME(reflection_property, isProtected, arginfo_reflection__void, 0)
  5587. ZEND_ME(reflection_property, isStatic, arginfo_reflection__void, 0)
  5588. ZEND_ME(reflection_property, isDefault, arginfo_reflection__void, 0)
  5589. ZEND_ME(reflection_property, getModifiers, arginfo_reflection__void, 0)
  5590. ZEND_ME(reflection_property, getDeclaringClass, arginfo_reflection__void, 0)
  5591. ZEND_ME(reflection_property, getDocComment, arginfo_reflection__void, 0)
  5592. ZEND_ME(reflection_property, setAccessible, arginfo_reflection_property_setAccessible, 0)
  5593. PHP_FE_END
  5594. };
  5595. ZEND_BEGIN_ARG_INFO_EX(arginfo_reflection_class_constant_export, 0, 0, 2)
  5596. ZEND_ARG_INFO(0, class)
  5597. ZEND_ARG_INFO(0, name)
  5598. ZEND_ARG_INFO(0, return)
  5599. ZEND_END_ARG_INFO()
  5600. ZEND_BEGIN_ARG_INFO_EX(arginfo_reflection_class_constant___construct, 0, 0, 2)
  5601. ZEND_ARG_INFO(0, class)
  5602. ZEND_ARG_INFO(0, name)
  5603. ZEND_END_ARG_INFO()
  5604. static const zend_function_entry reflection_class_constant_functions[] = {
  5605. ZEND_ME(reflection, __clone, arginfo_reflection__void, ZEND_ACC_PRIVATE|ZEND_ACC_FINAL)
  5606. ZEND_ME(reflection_class_constant, export, arginfo_reflection_class_constant_export, ZEND_ACC_STATIC|ZEND_ACC_PUBLIC)
  5607. ZEND_ME(reflection_class_constant, __construct, arginfo_reflection_class_constant___construct, 0)
  5608. ZEND_ME(reflection_class_constant, __toString, arginfo_reflection__void, 0)
  5609. ZEND_ME(reflection_class_constant, getName, arginfo_reflection__void, 0)
  5610. ZEND_ME(reflection_class_constant, getValue, arginfo_reflection__void, 0)
  5611. ZEND_ME(reflection_class_constant, isPublic, arginfo_reflection__void, 0)
  5612. ZEND_ME(reflection_class_constant, isPrivate, arginfo_reflection__void, 0)
  5613. ZEND_ME(reflection_class_constant, isProtected, arginfo_reflection__void, 0)
  5614. ZEND_ME(reflection_class_constant, getModifiers, arginfo_reflection__void, 0)
  5615. ZEND_ME(reflection_class_constant, getDeclaringClass, arginfo_reflection__void, 0)
  5616. ZEND_ME(reflection_class_constant, getDocComment, arginfo_reflection__void, 0)
  5617. PHP_FE_END
  5618. };
  5619. ZEND_BEGIN_ARG_INFO_EX(arginfo_reflection_parameter_export, 0, 0, 2)
  5620. ZEND_ARG_INFO(0, function)
  5621. ZEND_ARG_INFO(0, parameter)
  5622. ZEND_ARG_INFO(0, return)
  5623. ZEND_END_ARG_INFO()
  5624. ZEND_BEGIN_ARG_INFO(arginfo_reflection_parameter___construct, 0)
  5625. ZEND_ARG_INFO(0, function)
  5626. ZEND_ARG_INFO(0, parameter)
  5627. ZEND_END_ARG_INFO()
  5628. static const zend_function_entry reflection_parameter_functions[] = {
  5629. ZEND_ME(reflection, __clone, arginfo_reflection__void, ZEND_ACC_PRIVATE|ZEND_ACC_FINAL)
  5630. ZEND_ME(reflection_parameter, export, arginfo_reflection_parameter_export, ZEND_ACC_STATIC|ZEND_ACC_PUBLIC)
  5631. ZEND_ME(reflection_parameter, __construct, arginfo_reflection_parameter___construct, 0)
  5632. ZEND_ME(reflection_parameter, __toString, arginfo_reflection__void, 0)
  5633. ZEND_ME(reflection_parameter, getName, arginfo_reflection__void, 0)
  5634. ZEND_ME(reflection_parameter, isPassedByReference, arginfo_reflection__void, 0)
  5635. ZEND_ME(reflection_parameter, canBePassedByValue, arginfo_reflection__void, 0)
  5636. ZEND_ME(reflection_parameter, getDeclaringFunction, arginfo_reflection__void, 0)
  5637. ZEND_ME(reflection_parameter, getDeclaringClass, arginfo_reflection__void, 0)
  5638. ZEND_ME(reflection_parameter, getClass, arginfo_reflection__void, 0)
  5639. ZEND_ME(reflection_parameter, hasType, arginfo_reflection__void, 0)
  5640. ZEND_ME(reflection_parameter, getType, arginfo_reflection__void, 0)
  5641. ZEND_ME(reflection_parameter, isArray, arginfo_reflection__void, 0)
  5642. ZEND_ME(reflection_parameter, isCallable, arginfo_reflection__void, 0)
  5643. ZEND_ME(reflection_parameter, allowsNull, arginfo_reflection__void, 0)
  5644. ZEND_ME(reflection_parameter, getPosition, arginfo_reflection__void, 0)
  5645. ZEND_ME(reflection_parameter, isOptional, arginfo_reflection__void, 0)
  5646. ZEND_ME(reflection_parameter, isDefaultValueAvailable, arginfo_reflection__void, 0)
  5647. ZEND_ME(reflection_parameter, getDefaultValue, arginfo_reflection__void, 0)
  5648. ZEND_ME(reflection_parameter, isDefaultValueConstant, arginfo_reflection__void, 0)
  5649. ZEND_ME(reflection_parameter, getDefaultValueConstantName, arginfo_reflection__void, 0)
  5650. ZEND_ME(reflection_parameter, isVariadic, arginfo_reflection__void, 0)
  5651. PHP_FE_END
  5652. };
  5653. static const zend_function_entry reflection_type_functions[] = {
  5654. ZEND_ME(reflection, __clone, arginfo_reflection__void, ZEND_ACC_PRIVATE|ZEND_ACC_FINAL)
  5655. ZEND_ME(reflection_type, allowsNull, arginfo_reflection__void, 0)
  5656. ZEND_ME(reflection_type, isBuiltin, arginfo_reflection__void, 0)
  5657. /* ReflectionType::__toString() is deprecated, but we currently do not mark it as such
  5658. * due to bad interaction with the PHPUnit error handler and exceptions in __toString().
  5659. * See PR2137. */
  5660. ZEND_ME(reflection_type, __toString, arginfo_reflection__void, 0)
  5661. PHP_FE_END
  5662. };
  5663. static const zend_function_entry reflection_named_type_functions[] = {
  5664. ZEND_ME(reflection_named_type, getName, arginfo_reflection__void, 0)
  5665. PHP_FE_END
  5666. };
  5667. ZEND_BEGIN_ARG_INFO_EX(arginfo_reflection_extension_export, 0, 0, 1)
  5668. ZEND_ARG_INFO(0, name)
  5669. ZEND_ARG_INFO(0, return)
  5670. ZEND_END_ARG_INFO()
  5671. ZEND_BEGIN_ARG_INFO(arginfo_reflection_extension___construct, 0)
  5672. ZEND_ARG_INFO(0, name)
  5673. ZEND_END_ARG_INFO()
  5674. static const zend_function_entry reflection_extension_functions[] = {
  5675. ZEND_ME(reflection, __clone, arginfo_reflection__void, ZEND_ACC_PRIVATE|ZEND_ACC_FINAL)
  5676. ZEND_ME(reflection_extension, export, arginfo_reflection_extension_export, ZEND_ACC_STATIC|ZEND_ACC_PUBLIC)
  5677. ZEND_ME(reflection_extension, __construct, arginfo_reflection_extension___construct, 0)
  5678. ZEND_ME(reflection_extension, __toString, arginfo_reflection__void, 0)
  5679. ZEND_ME(reflection_extension, getName, arginfo_reflection__void, 0)
  5680. ZEND_ME(reflection_extension, getVersion, arginfo_reflection__void, 0)
  5681. ZEND_ME(reflection_extension, getFunctions, arginfo_reflection__void, 0)
  5682. ZEND_ME(reflection_extension, getConstants, arginfo_reflection__void, 0)
  5683. ZEND_ME(reflection_extension, getINIEntries, arginfo_reflection__void, 0)
  5684. ZEND_ME(reflection_extension, getClasses, arginfo_reflection__void, 0)
  5685. ZEND_ME(reflection_extension, getClassNames, arginfo_reflection__void, 0)
  5686. ZEND_ME(reflection_extension, getDependencies, arginfo_reflection__void, 0)
  5687. ZEND_ME(reflection_extension, info, arginfo_reflection__void, 0)
  5688. ZEND_ME(reflection_extension, isPersistent, arginfo_reflection__void, 0)
  5689. ZEND_ME(reflection_extension, isTemporary, arginfo_reflection__void, 0)
  5690. PHP_FE_END
  5691. };
  5692. ZEND_BEGIN_ARG_INFO(arginfo_reflection_zend_extension___construct, 0)
  5693. ZEND_ARG_INFO(0, name)
  5694. ZEND_END_ARG_INFO()
  5695. static const zend_function_entry reflection_zend_extension_functions[] = {
  5696. ZEND_ME(reflection, __clone, arginfo_reflection__void, ZEND_ACC_PRIVATE|ZEND_ACC_FINAL)
  5697. ZEND_ME(reflection_zend_extension, export, arginfo_reflection_extension_export, ZEND_ACC_STATIC|ZEND_ACC_PUBLIC)
  5698. ZEND_ME(reflection_zend_extension, __construct, arginfo_reflection_zend_extension___construct, 0)
  5699. ZEND_ME(reflection_zend_extension, __toString, arginfo_reflection__void, 0)
  5700. ZEND_ME(reflection_zend_extension, getName, arginfo_reflection__void, 0)
  5701. ZEND_ME(reflection_zend_extension, getVersion, arginfo_reflection__void, 0)
  5702. ZEND_ME(reflection_zend_extension, getAuthor, arginfo_reflection__void, 0)
  5703. ZEND_ME(reflection_zend_extension, getURL, arginfo_reflection__void, 0)
  5704. ZEND_ME(reflection_zend_extension, getCopyright, arginfo_reflection__void, 0)
  5705. PHP_FE_END
  5706. };
  5707. /* }}} */
  5708. static const zend_function_entry reflection_ext_functions[] = { /* {{{ */
  5709. PHP_FE_END
  5710. }; /* }}} */
  5711. /* {{{ _reflection_write_property */
  5712. static void _reflection_write_property(zval *object, zval *member, zval *value, void **cache_slot)
  5713. {
  5714. if ((Z_TYPE_P(member) == IS_STRING)
  5715. && zend_hash_exists(&Z_OBJCE_P(object)->properties_info, Z_STR_P(member))
  5716. && ((Z_STRLEN_P(member) == sizeof("name") - 1 && !memcmp(Z_STRVAL_P(member), "name", sizeof("name")))
  5717. || (Z_STRLEN_P(member) == sizeof("class") - 1 && !memcmp(Z_STRVAL_P(member), "class", sizeof("class")))))
  5718. {
  5719. zend_throw_exception_ex(reflection_exception_ptr, 0,
  5720. "Cannot set read-only property %s::$%s", ZSTR_VAL(Z_OBJCE_P(object)->name), Z_STRVAL_P(member));
  5721. }
  5722. else
  5723. {
  5724. zend_std_write_property(object, member, value, cache_slot);
  5725. }
  5726. }
  5727. /* }}} */
  5728. PHP_MINIT_FUNCTION(reflection) /* {{{ */
  5729. {
  5730. zend_class_entry _reflection_entry;
  5731. memcpy(&reflection_object_handlers, &std_object_handlers, sizeof(zend_object_handlers));
  5732. reflection_object_handlers.offset = XtOffsetOf(reflection_object, zo);
  5733. reflection_object_handlers.free_obj = reflection_free_objects_storage;
  5734. reflection_object_handlers.clone_obj = NULL;
  5735. reflection_object_handlers.write_property = _reflection_write_property;
  5736. reflection_object_handlers.get_gc = reflection_get_gc;
  5737. INIT_CLASS_ENTRY(_reflection_entry, "ReflectionException", reflection_exception_functions);
  5738. reflection_exception_ptr = zend_register_internal_class_ex(&_reflection_entry, zend_ce_exception);
  5739. INIT_CLASS_ENTRY(_reflection_entry, "Reflection", reflection_functions);
  5740. reflection_ptr = zend_register_internal_class(&_reflection_entry);
  5741. INIT_CLASS_ENTRY(_reflection_entry, "Reflector", reflector_functions);
  5742. reflector_ptr = zend_register_internal_interface(&_reflection_entry);
  5743. INIT_CLASS_ENTRY(_reflection_entry, "ReflectionFunctionAbstract", reflection_function_abstract_functions);
  5744. _reflection_entry.create_object = reflection_objects_new;
  5745. reflection_function_abstract_ptr = zend_register_internal_class(&_reflection_entry);
  5746. zend_class_implements(reflection_function_abstract_ptr, 1, reflector_ptr);
  5747. zend_declare_property_string(reflection_function_abstract_ptr, "name", sizeof("name")-1, "", ZEND_ACC_ABSTRACT);
  5748. INIT_CLASS_ENTRY(_reflection_entry, "ReflectionFunction", reflection_function_functions);
  5749. _reflection_entry.create_object = reflection_objects_new;
  5750. reflection_function_ptr = zend_register_internal_class_ex(&_reflection_entry, reflection_function_abstract_ptr);
  5751. zend_declare_property_string(reflection_function_ptr, "name", sizeof("name")-1, "", ZEND_ACC_PUBLIC);
  5752. REGISTER_REFLECTION_CLASS_CONST_LONG(function, "IS_DEPRECATED", ZEND_ACC_DEPRECATED);
  5753. INIT_CLASS_ENTRY(_reflection_entry, "ReflectionGenerator", reflection_generator_functions);
  5754. _reflection_entry.create_object = reflection_objects_new;
  5755. reflection_generator_ptr = zend_register_internal_class(&_reflection_entry);
  5756. INIT_CLASS_ENTRY(_reflection_entry, "ReflectionParameter", reflection_parameter_functions);
  5757. _reflection_entry.create_object = reflection_objects_new;
  5758. reflection_parameter_ptr = zend_register_internal_class(&_reflection_entry);
  5759. zend_class_implements(reflection_parameter_ptr, 1, reflector_ptr);
  5760. zend_declare_property_string(reflection_parameter_ptr, "name", sizeof("name")-1, "", ZEND_ACC_PUBLIC);
  5761. INIT_CLASS_ENTRY(_reflection_entry, "ReflectionType", reflection_type_functions);
  5762. _reflection_entry.create_object = reflection_objects_new;
  5763. reflection_type_ptr = zend_register_internal_class(&_reflection_entry);
  5764. INIT_CLASS_ENTRY(_reflection_entry, "ReflectionNamedType", reflection_named_type_functions);
  5765. _reflection_entry.create_object = reflection_objects_new;
  5766. reflection_named_type_ptr = zend_register_internal_class_ex(&_reflection_entry, reflection_type_ptr);
  5767. INIT_CLASS_ENTRY(_reflection_entry, "ReflectionMethod", reflection_method_functions);
  5768. _reflection_entry.create_object = reflection_objects_new;
  5769. reflection_method_ptr = zend_register_internal_class_ex(&_reflection_entry, reflection_function_abstract_ptr);
  5770. zend_declare_property_string(reflection_method_ptr, "name", sizeof("name")-1, "", ZEND_ACC_PUBLIC);
  5771. zend_declare_property_string(reflection_method_ptr, "class", sizeof("class")-1, "", ZEND_ACC_PUBLIC);
  5772. REGISTER_REFLECTION_CLASS_CONST_LONG(method, "IS_STATIC", ZEND_ACC_STATIC);
  5773. REGISTER_REFLECTION_CLASS_CONST_LONG(method, "IS_PUBLIC", ZEND_ACC_PUBLIC);
  5774. REGISTER_REFLECTION_CLASS_CONST_LONG(method, "IS_PROTECTED", ZEND_ACC_PROTECTED);
  5775. REGISTER_REFLECTION_CLASS_CONST_LONG(method, "IS_PRIVATE", ZEND_ACC_PRIVATE);
  5776. REGISTER_REFLECTION_CLASS_CONST_LONG(method, "IS_ABSTRACT", ZEND_ACC_ABSTRACT);
  5777. REGISTER_REFLECTION_CLASS_CONST_LONG(method, "IS_FINAL", ZEND_ACC_FINAL);
  5778. INIT_CLASS_ENTRY(_reflection_entry, "ReflectionClass", reflection_class_functions);
  5779. _reflection_entry.create_object = reflection_objects_new;
  5780. reflection_class_ptr = zend_register_internal_class(&_reflection_entry);
  5781. zend_class_implements(reflection_class_ptr, 1, reflector_ptr);
  5782. zend_declare_property_string(reflection_class_ptr, "name", sizeof("name")-1, "", ZEND_ACC_PUBLIC);
  5783. REGISTER_REFLECTION_CLASS_CONST_LONG(class, "IS_IMPLICIT_ABSTRACT", ZEND_ACC_IMPLICIT_ABSTRACT_CLASS);
  5784. REGISTER_REFLECTION_CLASS_CONST_LONG(class, "IS_EXPLICIT_ABSTRACT", ZEND_ACC_EXPLICIT_ABSTRACT_CLASS);
  5785. REGISTER_REFLECTION_CLASS_CONST_LONG(class, "IS_FINAL", ZEND_ACC_FINAL);
  5786. INIT_CLASS_ENTRY(_reflection_entry, "ReflectionObject", reflection_object_functions);
  5787. _reflection_entry.create_object = reflection_objects_new;
  5788. reflection_object_ptr = zend_register_internal_class_ex(&_reflection_entry, reflection_class_ptr);
  5789. INIT_CLASS_ENTRY(_reflection_entry, "ReflectionProperty", reflection_property_functions);
  5790. _reflection_entry.create_object = reflection_objects_new;
  5791. reflection_property_ptr = zend_register_internal_class(&_reflection_entry);
  5792. zend_class_implements(reflection_property_ptr, 1, reflector_ptr);
  5793. zend_declare_property_string(reflection_property_ptr, "name", sizeof("name")-1, "", ZEND_ACC_PUBLIC);
  5794. zend_declare_property_string(reflection_property_ptr, "class", sizeof("class")-1, "", ZEND_ACC_PUBLIC);
  5795. INIT_CLASS_ENTRY(_reflection_entry, "ReflectionClassConstant", reflection_class_constant_functions);
  5796. _reflection_entry.create_object = reflection_objects_new;
  5797. reflection_class_constant_ptr = zend_register_internal_class(&_reflection_entry);
  5798. zend_class_implements(reflection_class_constant_ptr, 1, reflector_ptr);
  5799. zend_declare_property_string(reflection_class_constant_ptr, "name", sizeof("name")-1, "", ZEND_ACC_PUBLIC);
  5800. zend_declare_property_string(reflection_class_constant_ptr, "class", sizeof("class")-1, "", ZEND_ACC_PUBLIC);
  5801. REGISTER_REFLECTION_CLASS_CONST_LONG(property, "IS_STATIC", ZEND_ACC_STATIC);
  5802. REGISTER_REFLECTION_CLASS_CONST_LONG(property, "IS_PUBLIC", ZEND_ACC_PUBLIC);
  5803. REGISTER_REFLECTION_CLASS_CONST_LONG(property, "IS_PROTECTED", ZEND_ACC_PROTECTED);
  5804. REGISTER_REFLECTION_CLASS_CONST_LONG(property, "IS_PRIVATE", ZEND_ACC_PRIVATE);
  5805. INIT_CLASS_ENTRY(_reflection_entry, "ReflectionExtension", reflection_extension_functions);
  5806. _reflection_entry.create_object = reflection_objects_new;
  5807. reflection_extension_ptr = zend_register_internal_class(&_reflection_entry);
  5808. zend_class_implements(reflection_extension_ptr, 1, reflector_ptr);
  5809. zend_declare_property_string(reflection_extension_ptr, "name", sizeof("name")-1, "", ZEND_ACC_PUBLIC);
  5810. INIT_CLASS_ENTRY(_reflection_entry, "ReflectionZendExtension", reflection_zend_extension_functions);
  5811. _reflection_entry.create_object = reflection_objects_new;
  5812. reflection_zend_extension_ptr = zend_register_internal_class(&_reflection_entry);
  5813. zend_class_implements(reflection_zend_extension_ptr, 1, reflector_ptr);
  5814. zend_declare_property_string(reflection_zend_extension_ptr, "name", sizeof("name")-1, "", ZEND_ACC_PUBLIC);
  5815. return SUCCESS;
  5816. } /* }}} */
  5817. PHP_MINFO_FUNCTION(reflection) /* {{{ */
  5818. {
  5819. php_info_print_table_start();
  5820. php_info_print_table_row(2, "Reflection", "enabled");
  5821. php_info_print_table_end();
  5822. } /* }}} */
  5823. zend_module_entry reflection_module_entry = { /* {{{ */
  5824. STANDARD_MODULE_HEADER,
  5825. "Reflection",
  5826. reflection_ext_functions,
  5827. PHP_MINIT(reflection),
  5828. NULL,
  5829. NULL,
  5830. NULL,
  5831. PHP_MINFO(reflection),
  5832. PHP_REFLECTION_VERSION,
  5833. STANDARD_MODULE_PROPERTIES
  5834. }; /* }}} */
  5835. /*
  5836. * Local variables:
  5837. * tab-width: 4
  5838. * c-basic-offset: 4
  5839. * indent-tabs-mode: t
  5840. * End:
  5841. * vim600: noet sw=4 ts=4 fdm=marker
  5842. */