BfResolvedTypeUtils.cpp 166 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788378937903791379237933794379537963797379837993800380138023803380438053806380738083809381038113812381338143815381638173818381938203821382238233824382538263827382838293830383138323833383438353836383738383839384038413842384338443845384638473848384938503851385238533854385538563857385838593860386138623863386438653866386738683869387038713872387338743875387638773878387938803881388238833884388538863887388838893890389138923893389438953896389738983899390039013902390339043905390639073908390939103911391239133914391539163917391839193920392139223923392439253926392739283929393039313932393339343935393639373938393939403941394239433944394539463947394839493950395139523953395439553956395739583959396039613962396339643965396639673968396939703971397239733974397539763977397839793980398139823983398439853986398739883989399039913992399339943995399639973998399940004001400240034004400540064007400840094010401140124013401440154016401740184019402040214022402340244025402640274028402940304031403240334034403540364037403840394040404140424043404440454046404740484049405040514052405340544055405640574058405940604061406240634064406540664067406840694070407140724073407440754076407740784079408040814082408340844085408640874088408940904091409240934094409540964097409840994100410141024103410441054106410741084109411041114112411341144115411641174118411941204121412241234124412541264127412841294130413141324133413441354136413741384139414041414142414341444145414641474148414941504151415241534154415541564157415841594160416141624163416441654166416741684169417041714172417341744175417641774178417941804181418241834184418541864187418841894190419141924193419441954196419741984199420042014202420342044205420642074208420942104211421242134214421542164217421842194220422142224223422442254226422742284229423042314232423342344235423642374238423942404241424242434244424542464247424842494250425142524253425442554256425742584259426042614262426342644265426642674268426942704271427242734274427542764277427842794280428142824283428442854286428742884289429042914292429342944295429642974298429943004301430243034304430543064307430843094310431143124313431443154316431743184319432043214322432343244325432643274328432943304331433243334334433543364337433843394340434143424343434443454346434743484349435043514352435343544355435643574358435943604361436243634364436543664367436843694370437143724373437443754376437743784379438043814382438343844385438643874388438943904391439243934394439543964397439843994400440144024403440444054406440744084409441044114412441344144415441644174418441944204421442244234424442544264427442844294430443144324433443444354436443744384439444044414442444344444445444644474448444944504451445244534454445544564457445844594460446144624463446444654466446744684469447044714472447344744475447644774478447944804481448244834484448544864487448844894490449144924493449444954496449744984499450045014502450345044505450645074508450945104511451245134514451545164517451845194520452145224523452445254526452745284529453045314532453345344535453645374538453945404541454245434544454545464547454845494550455145524553455445554556455745584559456045614562456345644565456645674568456945704571457245734574457545764577457845794580458145824583458445854586458745884589459045914592459345944595459645974598459946004601460246034604460546064607460846094610461146124613461446154616461746184619462046214622462346244625462646274628462946304631463246334634463546364637463846394640464146424643464446454646464746484649465046514652465346544655465646574658465946604661466246634664466546664667466846694670467146724673467446754676467746784679468046814682468346844685468646874688468946904691469246934694469546964697469846994700470147024703470447054706470747084709471047114712471347144715471647174718471947204721472247234724472547264727472847294730473147324733473447354736473747384739474047414742474347444745474647474748474947504751475247534754475547564757475847594760476147624763476447654766476747684769477047714772477347744775477647774778477947804781478247834784478547864787478847894790479147924793479447954796479747984799480048014802480348044805480648074808480948104811481248134814481548164817481848194820482148224823482448254826482748284829483048314832483348344835483648374838483948404841484248434844484548464847484848494850485148524853485448554856485748584859486048614862486348644865486648674868486948704871487248734874487548764877487848794880488148824883488448854886488748884889489048914892489348944895489648974898489949004901490249034904490549064907490849094910491149124913491449154916491749184919492049214922492349244925492649274928492949304931493249334934493549364937493849394940494149424943494449454946494749484949495049514952495349544955495649574958495949604961496249634964496549664967496849694970497149724973497449754976497749784979498049814982498349844985498649874988498949904991499249934994499549964997499849995000500150025003500450055006500750085009501050115012501350145015501650175018501950205021502250235024502550265027502850295030503150325033503450355036503750385039504050415042504350445045504650475048504950505051505250535054505550565057505850595060506150625063506450655066506750685069507050715072507350745075507650775078507950805081508250835084508550865087508850895090509150925093509450955096509750985099510051015102510351045105510651075108510951105111511251135114511551165117511851195120512151225123512451255126512751285129513051315132513351345135513651375138513951405141514251435144514551465147514851495150515151525153515451555156515751585159516051615162516351645165516651675168516951705171517251735174517551765177517851795180518151825183518451855186518751885189519051915192519351945195519651975198519952005201520252035204520552065207520852095210521152125213521452155216521752185219522052215222522352245225522652275228522952305231523252335234523552365237523852395240524152425243524452455246524752485249525052515252525352545255525652575258525952605261526252635264526552665267526852695270527152725273527452755276527752785279528052815282528352845285528652875288528952905291529252935294529552965297529852995300530153025303530453055306530753085309531053115312531353145315531653175318531953205321532253235324532553265327532853295330533153325333533453355336533753385339534053415342534353445345534653475348534953505351535253535354535553565357535853595360536153625363536453655366536753685369537053715372537353745375537653775378537953805381538253835384538553865387538853895390539153925393539453955396539753985399540054015402540354045405540654075408540954105411541254135414541554165417541854195420542154225423542454255426542754285429543054315432543354345435543654375438543954405441544254435444544554465447544854495450545154525453545454555456545754585459546054615462546354645465546654675468546954705471547254735474547554765477547854795480548154825483548454855486548754885489549054915492549354945495549654975498549955005501550255035504550555065507550855095510551155125513551455155516551755185519552055215522552355245525552655275528552955305531553255335534553555365537553855395540554155425543554455455546554755485549555055515552555355545555555655575558555955605561556255635564556555665567556855695570557155725573557455755576557755785579558055815582558355845585558655875588558955905591559255935594559555965597559855995600560156025603560456055606560756085609561056115612561356145615561656175618561956205621562256235624562556265627562856295630563156325633563456355636563756385639564056415642564356445645564656475648564956505651565256535654565556565657565856595660566156625663566456655666566756685669567056715672567356745675567656775678567956805681568256835684568556865687568856895690569156925693569456955696569756985699570057015702570357045705570657075708570957105711571257135714571557165717571857195720572157225723572457255726572757285729573057315732573357345735573657375738573957405741574257435744574557465747574857495750575157525753575457555756575757585759576057615762576357645765576657675768576957705771577257735774577557765777577857795780578157825783578457855786578757885789579057915792579357945795579657975798579958005801580258035804580558065807580858095810581158125813581458155816581758185819582058215822582358245825582658275828582958305831583258335834583558365837583858395840584158425843
  1. #include "BeefySysLib/util/AllocDebug.h"
  2. #include "BfCompiler.h"
  3. #include "BfParser.h"
  4. #include "BfDefBuilder.h"
  5. #include "BfMangler.h"
  6. #include "BfConstResolver.h"
  7. #include "BfModule.h"
  8. #include "CeMachine.h"
  9. #include "BeefySysLib/util/BeefPerf.h"
  10. #pragma warning(disable:4996)
  11. #pragma warning(disable:4267)
  12. USING_NS_BF;
  13. //void Beefy::DbgCheckType(llvm::Type* checkType)
  14. //{
  15. //#ifdef _DEBUG
  16. // /*while (auto ptrType = llvm::dyn_cast<llvm::PointerType>(checkType))
  17. // {
  18. // checkType = ptrType->getElementType();
  19. // }
  20. //
  21. // auto structType = llvm::dyn_cast<llvm::StructType>(checkType);
  22. // if (structType != NULL)
  23. // {
  24. // auto stringRef = structType->getName();
  25. // BF_ASSERT(strncmp(stringRef.data(), "DEAD", 4) != 0);
  26. // }*/
  27. //#endif
  28. //}
  29. void BfTypedValue::DbgCheckType() const
  30. {
  31. /*if (mType != NULL)
  32. {
  33. auto structType = llvm::dyn_cast<llvm::StructType>(mType->mIRType);
  34. if (structType != NULL)
  35. {
  36. auto stringRef = structType->getName();
  37. BF_ASSERT(strncmp(stringRef.data(), "DEAD", 4) != 0);
  38. }
  39. }*/
  40. #ifdef _DEBUG
  41. /*if (mValue != NULL)
  42. {
  43. auto checkType = mValue->getType();
  44. Beefy::DbgCheckType(checkType);
  45. }*/
  46. #endif
  47. }
  48. bool BfTypedValue::IsValuelessType() const
  49. {
  50. return mType->IsValuelessType();
  51. }
  52. bool BfTypedValue::CanModify() const
  53. {
  54. return (((IsAddr()) || (mType->IsValuelessType())) && (!IsReadOnly()));
  55. }
  56. //////////////////////////////////////////////////////////////////////////
  57. BfProtection BfUsingFieldData::MemberRef::GetProtection() const
  58. {
  59. switch (mKind)
  60. {
  61. case Kind_Field:
  62. return mTypeInstance->mTypeDef->mFields[mIdx]->mProtection;
  63. case Kind_Property:
  64. return mTypeInstance->mTypeDef->mProperties[mIdx]->mProtection;
  65. case Kind_Method:
  66. return mTypeInstance->mTypeDef->mMethods[mIdx]->mProtection;
  67. }
  68. return BfProtection_Public;
  69. }
  70. BfProtection BfUsingFieldData::MemberRef::GetUsingProtection() const
  71. {
  72. switch (mKind)
  73. {
  74. case Kind_Field:
  75. return mTypeInstance->mTypeDef->mFields[mIdx]->mUsingProtection;
  76. case Kind_Property:
  77. return mTypeInstance->mTypeDef->mProperties[mIdx]->mUsingProtection;
  78. case Kind_Method:
  79. return mTypeInstance->mTypeDef->mMethods[mIdx]->mProtection;
  80. }
  81. return BfProtection_Public;
  82. }
  83. BfTypeDef* BfUsingFieldData::MemberRef::GetDeclaringType(BfModule* curModule) const
  84. {
  85. switch (mKind)
  86. {
  87. case Kind_Field:
  88. return mTypeInstance->mTypeDef->mFields[mIdx]->mDeclaringType;
  89. case Kind_Property:
  90. return mTypeInstance->mTypeDef->mProperties[mIdx]->mDeclaringType;
  91. case Kind_Method:
  92. return mTypeInstance->mTypeDef->mMethods[mIdx]->mDeclaringType;
  93. case Kind_Local:
  94. return curModule->GetActiveTypeDef();
  95. }
  96. return NULL;
  97. }
  98. String BfUsingFieldData::MemberRef::GetFullName(BfModule* curModule) const
  99. {
  100. if (mKind == Kind_Local)
  101. return curModule->mCurMethodState->mLocals[mIdx]->mName;
  102. String result = curModule->TypeToString(mTypeInstance);
  103. if (!result.IsEmpty())
  104. result += ".";
  105. switch (mKind)
  106. {
  107. case Kind_Field:
  108. result += mTypeInstance->mTypeDef->mFields[mIdx]->mName;
  109. break;
  110. case Kind_Property:
  111. result += mTypeInstance->mTypeDef->mProperties[mIdx]->mName;
  112. break;
  113. case Kind_Method:
  114. result += mTypeInstance->mTypeDef->mMethods[mIdx]->mName;
  115. break;
  116. }
  117. return result;
  118. }
  119. String BfUsingFieldData::MemberRef::GetName(BfModule* curModule) const
  120. {
  121. switch (mKind)
  122. {
  123. case Kind_Field:
  124. return mTypeInstance->mTypeDef->mFields[mIdx]->mName;
  125. case Kind_Property:
  126. return mTypeInstance->mTypeDef->mProperties[mIdx]->mName;
  127. case Kind_Method:
  128. {
  129. auto methodInstance = curModule->GetRawMethodInstance(mTypeInstance, mTypeInstance->mTypeDef->mMethods[mIdx]);
  130. return curModule->MethodToString(methodInstance, BfMethodNameFlag_OmitTypeName);
  131. }
  132. case Kind_Local:
  133. return curModule->mCurMethodState->mLocals[mIdx]->mName;
  134. }
  135. return "";
  136. }
  137. BfAstNode* BfUsingFieldData::MemberRef::GetRefNode(BfModule* curModule) const
  138. {
  139. switch (mKind)
  140. {
  141. case Kind_Field:
  142. return mTypeInstance->mTypeDef->mFields[mIdx]->GetRefNode();
  143. case Kind_Property:
  144. return mTypeInstance->mTypeDef->mProperties[mIdx]->GetRefNode();
  145. case Kind_Method:
  146. return mTypeInstance->mTypeDef->mMethods[mIdx]->GetRefNode();
  147. case Kind_Local:
  148. return curModule->mCurMethodState->mLocals[mIdx]->mNameNode;
  149. }
  150. return NULL;
  151. }
  152. bool BfUsingFieldData::MemberRef::IsStatic() const
  153. {
  154. switch (mKind)
  155. {
  156. case Kind_Field:
  157. return mTypeInstance->mTypeDef->mFields[mIdx]->mIsStatic;
  158. case Kind_Property:
  159. return mTypeInstance->mTypeDef->mProperties[mIdx]->mIsStatic;
  160. case Kind_Method:
  161. return mTypeInstance->mTypeDef->mMethods[mIdx]->mIsStatic;
  162. }
  163. return false;
  164. }
  165. //////////////////////////////////////////////////////////////////////////
  166. bool BfGenericParamInstance::IsEnum()
  167. {
  168. if ((mGenericParamFlags & BfGenericParamFlag_Enum) != 0)
  169. return true;
  170. if (mTypeConstraint != NULL)
  171. {
  172. auto module = mTypeConstraint->GetModule();
  173. if ((module != NULL) && (mTypeConstraint->IsInstanceOf(module->mCompiler->mEnumTypeDef)))
  174. return true;
  175. }
  176. return true;
  177. return false;
  178. }
  179. //////////////////////////////////////////////////////////////////////////
  180. bool BfDependencyMap::AddUsedBy(BfType* dependentType, BfDependencyMap::DependencyFlags flags)
  181. {
  182. BF_ASSERT(dependentType != NULL);
  183. BF_ASSERT(dependentType->mRevision != -1);
  184. //auto itr = mTypeSet.insert(BfDependencyMap::TypeMap::value_type(dependentType, DependencyEntry(dependentType->mRevision, flags)));
  185. //if (!itr.second)
  186. DependencyEntry* dependencyEntry = NULL;
  187. if (mTypeSet.TryAddRaw(dependentType, NULL, &dependencyEntry))
  188. {
  189. if ((flags & ~DependencyFlag_UnspecializedType) != 0)
  190. {
  191. if (auto dependentDepType = dependentType->ToDependedType())
  192. {
  193. int tryDepth = dependentDepType->mDependencyMap.mMinDependDepth + 1;
  194. if (tryDepth < mMinDependDepth)
  195. mMinDependDepth = tryDepth;
  196. }
  197. }
  198. dependencyEntry->mRevision = dependentType->mRevision;
  199. dependencyEntry->mFlags = flags;
  200. return true;
  201. }
  202. else
  203. {
  204. if (dependencyEntry->mRevision != dependentType->mRevision)
  205. {
  206. dependencyEntry->mRevision = dependentType->mRevision;
  207. dependencyEntry->mFlags = flags;
  208. return true;
  209. }
  210. else
  211. {
  212. if ((dependencyEntry->mFlags & flags) == flags)
  213. return false;
  214. dependencyEntry->mFlags = (DependencyFlags)(dependencyEntry->mFlags | flags);
  215. return true;
  216. }
  217. }
  218. }
  219. bool BfDependencyMap::IsEmpty()
  220. {
  221. return mTypeSet.size() == 0;
  222. }
  223. BfDependencyMap::TypeMap::iterator BfDependencyMap::begin()
  224. {
  225. return mTypeSet.begin();
  226. }
  227. BfDependencyMap::TypeMap::iterator BfDependencyMap::end()
  228. {
  229. return mTypeSet.end();
  230. }
  231. BfDependencyMap::TypeMap::iterator BfDependencyMap::erase(BfDependencyMap::TypeMap::iterator& itr)
  232. {
  233. return mTypeSet.Remove(itr);
  234. }
  235. //////////////////////////////////////////////////////////////////////////
  236. BfFieldDef* BfFieldInstance::GetFieldDef()
  237. {
  238. if (mFieldIdx == -1)
  239. return NULL;
  240. return mOwner->mTypeDef->mFields[mFieldIdx];
  241. }
  242. //////////////////////////////////////////////////////////////////////////
  243. BfType::BfType()
  244. {
  245. mTypeId = -1;
  246. mContext = NULL;
  247. mRevision = -1;
  248. //mLastUsedRevision = -1;
  249. mDefineState = BfTypeDefineState_Undefined;
  250. //mDICallbackVH = NULL;
  251. //mInnerDICallbackVH = NULL;
  252. mRebuildFlags = BfTypeRebuildFlag_None;
  253. mAlign = -1;
  254. mSize = -1;
  255. //mDICallbackVH = NULL;
  256. //mInnerDICallbackVH = NULL;
  257. mDirty = true;
  258. }
  259. BfModule* BfType::GetModule()
  260. {
  261. if (mContext->mCompiler->mOptions.mCompileOnDemandKind == BfCompileOnDemandKind_AlwaysInclude)
  262. return mContext->mScratchModule;
  263. else
  264. return mContext->mUnreifiedModule;
  265. }
  266. BfTypeInstance* BfType::FindUnderlyingTypeInstance()
  267. {
  268. auto typeInstance = ToTypeInstance();
  269. if (typeInstance != NULL)
  270. return typeInstance;
  271. auto underlyingType = GetUnderlyingType();
  272. while (underlyingType != NULL)
  273. {
  274. auto underlyingTypeInst = underlyingType->ToTypeInstance();
  275. if (underlyingTypeInst != NULL)
  276. return underlyingTypeInst;
  277. underlyingType = underlyingType->GetUnderlyingType();
  278. }
  279. return NULL;
  280. }
  281. void BfType::ReportMemory(MemReporter* memReporter)
  282. {
  283. memReporter->Add(sizeof(BfType));
  284. }
  285. //////////////////////////////////////////////////////////////////////////
  286. BfNonGenericMethodRef::BfNonGenericMethodRef(BfMethodInstance* methodInstance)
  287. {
  288. *this = methodInstance;
  289. }
  290. BfNonGenericMethodRef::operator BfMethodInstance* () const
  291. {
  292. if (mTypeInstance == NULL)
  293. return NULL;
  294. if (mMethodNum < 0)
  295. return NULL;
  296. auto& methodSpecializationGroup = mTypeInstance->mMethodInstanceGroups[mMethodNum];
  297. BF_ASSERT((methodSpecializationGroup.mDefault != NULL) || (mTypeInstance->mTypeFailed));
  298. return methodSpecializationGroup.mDefault;
  299. }
  300. BfMethodInstance* BfNonGenericMethodRef::operator->() const
  301. {
  302. return *this;
  303. }
  304. BfNonGenericMethodRef& BfNonGenericMethodRef::operator=(BfMethodInstance* methodInstance)
  305. {
  306. if (methodInstance == NULL)
  307. {
  308. mTypeInstance = NULL;
  309. mMethodNum = 0;
  310. }
  311. else
  312. {
  313. mTypeInstance = methodInstance->mMethodInstanceGroup->mOwner;
  314. mMethodNum = methodInstance->mMethodInstanceGroup->mMethodIdx;
  315. BF_ASSERT((methodInstance->GetNumGenericArguments() == 0) ||
  316. ((methodInstance->mIsUnspecialized) && (!methodInstance->mIsUnspecializedVariation)));
  317. mSignatureHash = (int)mTypeInstance->mTypeDef->mSignatureHash;
  318. }
  319. return *this;
  320. }
  321. bool BfNonGenericMethodRef::operator==(const BfNonGenericMethodRef& methodRef) const
  322. {
  323. bool eq = ((methodRef.mKind == mKind) &&
  324. (methodRef.mTypeInstance == mTypeInstance) &&
  325. (methodRef.mMethodNum == mMethodNum));
  326. if (eq)
  327. {
  328. BF_ASSERT((methodRef.mSignatureHash == mSignatureHash) || (methodRef.mSignatureHash == 0) || (mSignatureHash == 0));
  329. }
  330. return eq;
  331. }
  332. bool BfNonGenericMethodRef::operator==(BfMethodInstance* methodInstance) const
  333. {
  334. if (mTypeInstance != methodInstance->GetOwner())
  335. return false;
  336. return methodInstance == (BfMethodInstance*)*this;
  337. }
  338. size_t BfNonGenericMethodRef::Hash::operator()(const BfNonGenericMethodRef& val) const
  339. {
  340. return (val.mTypeInstance->mTypeId << 10) ^ val.mMethodNum;
  341. }
  342. //////////////////////////////////////////////////////////////////////////
  343. BfMethodRef::BfMethodRef(BfMethodInstance* methodInstance)
  344. {
  345. *this = methodInstance;
  346. }
  347. BfMethodRef::operator BfMethodInstance* () const
  348. {
  349. if (mTypeInstance == NULL)
  350. return NULL;
  351. if (mMethodNum < 0)
  352. return NULL;
  353. auto& methodSpecializationGroup = mTypeInstance->mMethodInstanceGroups[mMethodNum];
  354. if (mMethodGenericArguments.size() != 0)
  355. {
  356. bool isSpecialied = false;
  357. int paramIdx = 0;
  358. for (auto genericArg : mMethodGenericArguments)
  359. {
  360. if (!genericArg->IsGenericParam())
  361. {
  362. isSpecialied = true;
  363. break;
  364. }
  365. auto genericParam = (BfGenericParamType*)genericArg;
  366. if ((genericParam->mGenericParamKind != BfGenericParamKind_Method) || (genericParam->mGenericParamIdx != paramIdx))
  367. {
  368. isSpecialied = true;
  369. break;
  370. }
  371. paramIdx++;
  372. }
  373. if (isSpecialied)
  374. {
  375. BfMethodInstance** methodInstancePtr = NULL;
  376. if (methodSpecializationGroup.mMethodSpecializationMap->TryGetValue(mMethodGenericArguments, &methodInstancePtr))
  377. return *methodInstancePtr;
  378. return NULL;
  379. }
  380. }
  381. BF_ASSERT(methodSpecializationGroup.mDefault != NULL);
  382. return methodSpecializationGroup.mDefault;
  383. }
  384. BfMethodInstance* BfMethodRef::operator->() const
  385. {
  386. return *this;
  387. }
  388. BfMethodRef& BfMethodRef::operator=(BfMethodInstance* methodInstance)
  389. {
  390. if (methodInstance == NULL)
  391. {
  392. mTypeInstance = NULL;
  393. mMethodNum = 0;
  394. mMethodRefFlags = BfMethodRefFlag_None;
  395. }
  396. else
  397. {
  398. mTypeInstance = methodInstance->mMethodInstanceGroup->mOwner;
  399. mMethodNum = methodInstance->mMethodInstanceGroup->mMethodIdx;
  400. if (methodInstance->mMethodInfoEx != NULL)
  401. {
  402. mMethodGenericArguments.Clear();
  403. for (auto type : methodInstance->mMethodInfoEx->mMethodGenericArguments)
  404. mMethodGenericArguments.Add(type);
  405. }
  406. mSignatureHash = (int)mTypeInstance->mTypeDef->mSignatureHash;
  407. if (methodInstance->mAlwaysInline)
  408. mMethodRefFlags = BfMethodRefFlag_AlwaysInclude;
  409. else
  410. mMethodRefFlags = BfMethodRefFlag_None;
  411. }
  412. return *this;
  413. }
  414. bool BfMethodRef::operator==(const BfMethodRef& methodRef) const
  415. {
  416. bool eq = ((methodRef.mKind == mKind) &&
  417. (methodRef.mTypeInstance == mTypeInstance) &&
  418. (methodRef.mMethodNum == mMethodNum) &&
  419. (methodRef.mMethodGenericArguments == mMethodGenericArguments) &&
  420. (methodRef.mMethodRefFlags == mMethodRefFlags));
  421. if (eq)
  422. {
  423. BF_ASSERT((methodRef.mSignatureHash == mSignatureHash) || (methodRef.mSignatureHash == 0) || (mSignatureHash == 0));
  424. }
  425. return eq;
  426. }
  427. bool BfMethodRef::operator==(BfMethodInstance* methodInstance) const
  428. {
  429. if (mTypeInstance != methodInstance->GetOwner())
  430. return false;
  431. return methodInstance == (BfMethodInstance*)*this;
  432. }
  433. size_t BfMethodRef::Hash::operator()(const BfMethodRef& val) const
  434. {
  435. return (val.mTypeInstance->mTypeId << 10) ^ (val.mMethodNum << 1) ^ (int)(val.mMethodRefFlags);
  436. }
  437. //////////////////////////////////////////////////////////////////////////
  438. BfFieldRef::BfFieldRef(BfTypeInstance* typeInst, BfFieldDef* fieldDef)
  439. {
  440. mTypeInstance = typeInst;
  441. mFieldIdx = fieldDef->mIdx;
  442. }
  443. BfFieldRef::BfFieldRef(BfFieldInstance* fieldInstance)
  444. {
  445. mTypeInstance = fieldInstance->mOwner;
  446. mFieldIdx = fieldInstance->mFieldIdx;
  447. }
  448. BfFieldRef::operator BfFieldInstance*() const
  449. {
  450. BF_ASSERT(!mTypeInstance->IsDataIncomplete());
  451. return &mTypeInstance->mFieldInstances[mFieldIdx];
  452. }
  453. BfFieldRef::operator BfFieldDef*() const
  454. {
  455. return mTypeInstance->mTypeDef->mFields[mFieldIdx];
  456. }
  457. //////////////////////////////////////////////////////////////////////////
  458. BfPropertyRef::BfPropertyRef(BfTypeInstance* typeInst, BfPropertyDef* propDef)
  459. {
  460. mTypeInstance = typeInst;
  461. mPropIdx = propDef->mIdx;
  462. }
  463. BfPropertyRef::operator BfPropertyDef*() const
  464. {
  465. return mTypeInstance->mTypeDef->mProperties[mPropIdx];
  466. }
  467. //////////////////////////////////////////////////////////////////////////
  468. /*BfMethodInstance* BfTypeInstance::GetVTableMethodInstance(int vtableIdx)
  469. {
  470. auto& methodSpecializationGroup = mVirtualMethodTable[vtableIdx].mTypeInstance->mMethodInstanceGroups[mVirtualMethodTable[vtableIdx].mMethodNum];
  471. return &methodSpecializationGroup.mMethodSpecializationMap.begin()->second;
  472. }*/
  473. static int gDelIdx = 0;
  474. BfType::~BfType()
  475. {
  476. if (mContext != NULL)
  477. BfLogSys(mContext->mSystem, "~BfType %p\n", this);
  478. /*gDelIdx++;
  479. auto typeInst = ToTypeInstance();
  480. OutputDebugStrF("%d Deleting %08X type %s\n", gDelIdx, this, (typeInst != NULL) ? typeInst->mTypeDef->mName.c_str() : "");*/
  481. //delete mDICallbackVH;
  482. //delete mInnerDICallbackVH;
  483. }
  484. BfFieldInstance::~BfFieldInstance()
  485. {
  486. delete mCustomAttributes;
  487. }
  488. BfType* BfFieldInstance::GetResolvedType()
  489. {
  490. return mResolvedType;
  491. }
  492. void BfFieldInstance::SetResolvedType(BfType* type)
  493. {
  494. mResolvedType = type;
  495. }
  496. void BfFieldInstance::GetDataRange(int& dataIdx, int& dataCount)
  497. {
  498. int minMergedDataIdx = mMergedDataIdx;
  499. int maxMergedDataIdx = minMergedDataIdx + 1;
  500. if (mResolvedType->IsStruct())
  501. maxMergedDataIdx = minMergedDataIdx + mResolvedType->ToTypeInstance()->mMergedFieldDataCount;
  502. if (mOwner->mIsUnion)
  503. {
  504. for (auto& checkFieldInstance : mOwner->mFieldInstances)
  505. {
  506. if (&checkFieldInstance == this)
  507. continue;
  508. if (checkFieldInstance.mDataIdx == mDataIdx)
  509. {
  510. int checkMinMergedDataIdx = checkFieldInstance.mMergedDataIdx;
  511. int checkMaxMergedDataIdx = checkMinMergedDataIdx + 1;
  512. if (checkFieldInstance.GetResolvedType()->IsStruct())
  513. checkMaxMergedDataIdx = checkMinMergedDataIdx + checkFieldInstance.mResolvedType->ToTypeInstance()->mMergedFieldDataCount;
  514. minMergedDataIdx = BF_MIN(minMergedDataIdx, checkMinMergedDataIdx);
  515. maxMergedDataIdx = BF_MAX(maxMergedDataIdx, checkMaxMergedDataIdx);
  516. }
  517. }
  518. }
  519. int fieldIdx = dataIdx - 1;
  520. if (fieldIdx == -1)
  521. {
  522. dataIdx = minMergedDataIdx + 1;
  523. }
  524. else
  525. {
  526. fieldIdx += minMergedDataIdx;
  527. dataIdx = fieldIdx + 1;
  528. }
  529. dataCount = maxMergedDataIdx - minMergedDataIdx;
  530. }
  531. int BfFieldInstance::GetAlign(int packing)
  532. {
  533. int align = mResolvedType->mAlign;
  534. if (IsAppendedObject())
  535. align = mResolvedType->ToTypeInstance()->mInstAlign;
  536. if (packing > 0)
  537. align = BF_MIN(align, packing);
  538. if (mCustomAttributes != NULL)
  539. {
  540. auto module = mOwner->mModule;
  541. for (auto& attrib : mCustomAttributes->mAttributes)
  542. {
  543. if (attrib.mType->IsInstanceOf(module->mCompiler->mAlignAttributeTypeDef))
  544. {
  545. align = 16; // System conservative default
  546. if (!attrib.mCtorArgs.IsEmpty())
  547. {
  548. BfIRConstHolder* constHolder = module->mCurTypeInstance->mConstHolder;
  549. auto constant = constHolder->GetConstant(attrib.mCtorArgs[0]);
  550. if (constant != NULL)
  551. {
  552. int alignOverride = (int)BF_MAX(1, constant->mInt64);
  553. if ((alignOverride & (alignOverride - 1)) == 0)
  554. align = alignOverride;
  555. else
  556. module->Fail("Alignment must be a power of 2", attrib.GetRefNode());
  557. }
  558. }
  559. }
  560. }
  561. }
  562. return align;
  563. }
  564. bool BfFieldInstance::IsAppendedObject()
  565. {
  566. auto fieldDef = GetFieldDef();
  567. return (fieldDef != NULL) && (fieldDef->mIsAppend) && (mResolvedType->IsObject()) && (mOwner->IsObject());
  568. }
  569. //////////////////////////////////////////////////////////////////////////
  570. int64 BfDeferredMethodCallData::GenerateMethodId(BfModule* module, int64 methodId)
  571. {
  572. // The mMethodId MUST be unique within a given deferred method processor. We are even more conservative, making it
  573. // unique per module
  574. if (module->mDeferredMethodIds.Add(methodId))
  575. {
  576. return methodId;
  577. }
  578. else
  579. {
  580. // Ideally the passed in methodId just works, otherwise --
  581. // We hope to create a hash that will hopefully be globally unique. If it isn't then it just means we will end up with two
  582. // conflicting debug info definitions for the same name, which is not an error but may cause a debugger to show the
  583. // wrong one to the user. Does not affect runtime correctness.
  584. int64 checkId = Hash64(module->mModuleName.c_str(), (int)module->mModuleName.length(), module->mDeferredMethodIds.size());
  585. while (true)
  586. {
  587. if (!module->mDeferredMethodIds.Contains(checkId))
  588. break;
  589. checkId += 0x100;
  590. }
  591. module->mDeferredMethodIds.Add(checkId);
  592. return checkId;
  593. }
  594. }
  595. //////////////////////////////////////////////////////////////////////////
  596. BfMethodCustomAttributes::~BfMethodCustomAttributes()
  597. {
  598. delete mCustomAttributes;
  599. delete mReturnCustomAttributes;
  600. for (auto paramCustomAttributes : mParamCustomAttributes)
  601. delete paramCustomAttributes;
  602. }
  603. BfMethodInstance* BfMethodParam::GetDelegateParamInvoke()
  604. {
  605. if (mResolvedType->IsMethodRef())
  606. {
  607. auto methodRefType = (BfMethodRefType*)mResolvedType;
  608. return methodRefType->mMethodRef;
  609. }
  610. else if (mResolvedType->IsTuple())
  611. return NULL;
  612. BF_ASSERT(mResolvedType->IsDelegate() || mResolvedType->IsFunction());
  613. auto bfModule = BfModule::GetModuleFor(mResolvedType);
  614. BfMethodInstance* invokeMethodInstance = bfModule->GetRawMethodInstanceAtIdx(mResolvedType->ToTypeInstance(), 0, "Invoke");
  615. return invokeMethodInstance;
  616. }
  617. BfMethodInfoEx::~BfMethodInfoEx()
  618. {
  619. for (auto genericParam : mGenericParams)
  620. genericParam->Release();
  621. delete mMethodCustomAttributes;
  622. delete mClosureInstanceInfo;
  623. }
  624. BfMethodInstance::~BfMethodInstance()
  625. {
  626. Dispose(true);
  627. if (mHasMethodRefType)
  628. {
  629. auto module = GetOwner()->mModule;
  630. if (!module->mContext->mDeleting)
  631. {
  632. auto methodRefType = module->CreateMethodRefType(this);
  633. module->mContext->DeleteType(methodRefType);
  634. }
  635. }
  636. delete mMethodInfoEx;
  637. }
  638. void BfMethodInstance::Dispose(bool isDeleting)
  639. {
  640. if (mIsDisposed)
  641. return;
  642. mIsDisposed = true;
  643. if (mMethodInstanceGroup != NULL)
  644. {
  645. BfLogSys(GetOwner()->mModule->mSystem, "BfMethodInstance::~BfMethodInstance %p Local:%d InCEMachine:%d Deleting:%d\n", this, mMethodDef->mIsLocalMethod, mInCEMachine, isDeleting);
  646. }
  647. else
  648. {
  649. BF_ASSERT(!mMethodDef->mIsLocalMethod);
  650. }
  651. if (mInCEMachine)
  652. {
  653. auto module = GetOwner()->mModule;
  654. if (module->mCompiler->mCeMachine != NULL)
  655. module->mCompiler->mCeMachine->RemoveMethod(this);
  656. }
  657. if (mMethodProcessRequest != NULL)
  658. {
  659. BF_ASSERT(mMethodProcessRequest->mMethodInstance == this);
  660. mMethodProcessRequest->mMethodInstance = NULL;
  661. }
  662. if (mHotMethod != NULL)
  663. {
  664. mHotMethod->mFlags = (BfHotDepDataFlags)(mHotMethod->mFlags & ~BfHotDepDataFlag_IsBound);
  665. mHotMethod->Deref();
  666. }
  667. }
  668. void BfMethodInstance::CopyFrom(BfMethodInstance* methodInstance)
  669. {
  670. *this = *methodInstance;
  671. if (mMethodInfoEx != NULL)
  672. {
  673. mMethodInfoEx = new BfMethodInfoEx();
  674. *mMethodInfoEx = *(methodInstance->mMethodInfoEx);
  675. for (auto genericParam : mMethodInfoEx->mGenericParams)
  676. genericParam->AddRef();
  677. mMethodInfoEx->mMethodCustomAttributes = NULL;
  678. if (mMethodInfoEx->mClosureInstanceInfo != NULL)
  679. {
  680. mMethodInfoEx->mClosureInstanceInfo = new BfClosureInstanceInfo();
  681. *mMethodInfoEx->mClosureInstanceInfo = *methodInstance->mMethodInfoEx->mClosureInstanceInfo;
  682. }
  683. }
  684. mHasMethodRefType = false;
  685. mHasBeenProcessed = false;
  686. mIRFunction = BfIRValue();
  687. mMethodProcessRequest = NULL;
  688. mHotMethod = NULL;
  689. }
  690. BfImportKind BfMethodInstance::GetImportKind()
  691. {
  692. if (mMethodDef->mImportKind != BfImportKind_Import_Unknown)
  693. return mMethodDef->mImportKind;
  694. auto module = GetOwner()->mModule;
  695. auto customAttributes = GetCustomAttributes();
  696. if (customAttributes == NULL)
  697. return BfImportKind_None;
  698. BfCustomAttribute* customAttribute = customAttributes->Get(module->mCompiler->mImportAttributeTypeDef);
  699. if (customAttribute == NULL)
  700. return BfImportKind_Import_Static;
  701. BfIRConstHolder* constHolder = GetOwner()->mConstHolder;
  702. String* filePath = module->GetStringPoolString(customAttribute->mCtorArgs[0], constHolder);
  703. if (filePath == NULL)
  704. return BfImportKind_Import_Static;
  705. return BfMethodDef::GetImportKindFromPath(*filePath);
  706. }
  707. BfMethodFlags BfMethodInstance::GetMethodFlags()
  708. {
  709. BfMethodFlags methodFlags = (BfMethodFlags)0;
  710. if (mMethodDef->mProtection == BfProtection_Protected)
  711. methodFlags = (BfMethodFlags)(methodFlags | BfMethodFlags_Protected);
  712. if (mMethodDef->mProtection == BfProtection_Public)
  713. methodFlags = (BfMethodFlags)(methodFlags | BfMethodFlags_Public);
  714. if (mMethodDef->mIsStatic)
  715. methodFlags = (BfMethodFlags)(methodFlags | BfMethodFlags_Static);
  716. if ((mMethodDef->mIsVirtual) || (mVirtualTableIdx != -1))
  717. methodFlags = (BfMethodFlags)(methodFlags | BfMethodFlags_Virtual);
  718. if (mMethodDef->mCallingConvention == BfCallingConvention_Fastcall)
  719. methodFlags = (BfMethodFlags)(methodFlags | BfMethodFlags_FastCall);
  720. if (mMethodDef->mIsMutating)
  721. methodFlags = (BfMethodFlags)(methodFlags | BfMethodFlags_Mutating);
  722. if (mMethodDef->mMethodType == BfMethodType_Ctor)
  723. methodFlags = (BfMethodFlags)(methodFlags | BfMethodFlags_Constructor);
  724. if (mMethodDef->mIsReadOnly)
  725. methodFlags = (BfMethodFlags)(methodFlags | BfMethodFlags_ReadOnly);
  726. if (mMethodDef->mMethodType == BfMethodType_Mixin)
  727. methodFlags = (BfMethodFlags)(methodFlags | BfMethodFlags_Mixin);
  728. auto callingConvention = GetOwner()->mModule->GetIRCallingConvention(this);
  729. if (callingConvention == BfIRCallingConv_ThisCall)
  730. methodFlags = (BfMethodFlags)(methodFlags | BfMethodFlags_ThisCall);
  731. else if (callingConvention == BfIRCallingConv_StdCall)
  732. methodFlags = (BfMethodFlags)(methodFlags | BfMethodFlags_StdCall);
  733. else if (callingConvention == BfIRCallingConv_FastCall)
  734. methodFlags = (BfMethodFlags)(methodFlags | BfMethodFlags_FastCall);
  735. methodFlags = (BfMethodFlags)(methodFlags | (mMethodDef->mAppendKind * BfMethodFlags_AppendBit0));
  736. methodFlags = (BfMethodFlags)(methodFlags | (mMethodDef->mCheckedKind * BfMethodFlags_CheckedBit0));
  737. return methodFlags;
  738. }
  739. BfComptimeMethodFlags BfMethodInstance::GetComptimeMethodFlags()
  740. {
  741. BfComptimeMethodFlags methodFlags = (BfComptimeMethodFlags)0;
  742. if (!mMethodDef->CanReflect())
  743. methodFlags = (BfComptimeMethodFlags)(methodFlags | BfComptimeMethodFlags_NoReflect);
  744. return methodFlags;
  745. }
  746. void BfMethodInstance::UndoDeclaration(bool keepIRFunction)
  747. {
  748. if (mMethodInfoEx != NULL)
  749. {
  750. for (auto genericParam : mMethodInfoEx->mGenericParams)
  751. genericParam->Release();
  752. mMethodInfoEx->mGenericParams.Clear();
  753. delete mMethodInfoEx->mMethodCustomAttributes;
  754. mMethodInfoEx->mMethodCustomAttributes = NULL;
  755. mMethodInfoEx->mGenericTypeBindings.Clear();
  756. }
  757. mReturnType = NULL;
  758. if (!keepIRFunction)
  759. mIRFunction = BfIRValue();
  760. mParams.Clear();
  761. mDefaultValues.Clear();
  762. if (mMethodProcessRequest != NULL)
  763. {
  764. BF_ASSERT(mMethodProcessRequest->mMethodInstance == this);
  765. mMethodProcessRequest->mMethodInstance = NULL;
  766. }
  767. mHasStartedDeclaration = false;
  768. mHasBeenDeclared = false;
  769. mHasBeenProcessed = false;
  770. mIsUnspecialized = false;
  771. mIsUnspecializedVariation = false;
  772. mDisallowCalling = false;
  773. mIsIntrinsic = false;
  774. mHasFailed = false;
  775. mFailedConstraints = false;
  776. }
  777. BfTypeInstance* BfMethodInstance::GetOwner()
  778. {
  779. return mMethodInstanceGroup->mOwner;
  780. }
  781. BfModule * BfMethodInstance::GetModule()
  782. {
  783. return mMethodInstanceGroup->mOwner->mModule;
  784. }
  785. bool BfMethodInstance::ForcingThisPtr()
  786. {
  787. if (mMethodDef->mHasExplicitThis)
  788. {
  789. auto thisType = mParams[0].mResolvedType;
  790. if (thisType->IsCRepr())
  791. return true;
  792. }
  793. else if (mMethodInstanceGroup->mOwner->IsCRepr())
  794. return true;
  795. return (mCallingConvention == BfCallingConvention_Cdecl);
  796. }
  797. bool Beefy::BfMethodInstance::IsSpecializedGenericMethod()
  798. {
  799. return (mMethodInfoEx != NULL) && (mMethodInfoEx->mGenericParams.size() != 0) && (!mIsUnspecialized);
  800. }
  801. bool Beefy::BfMethodInstance::IsSpecializedGenericMethodOrType()
  802. {
  803. if ((mMethodInfoEx != NULL) && (mMethodInfoEx->mGenericParams.size() != 0) && (!mIsUnspecialized))
  804. return true;
  805. auto owner = GetOwner();
  806. if (!owner->IsGenericTypeInstance())
  807. return false;
  808. BfTypeInstance* genericTypeInstance = (BfTypeInstance*)owner;
  809. return !genericTypeInstance->mGenericTypeInfo->mIsUnspecialized;
  810. }
  811. bool BfMethodInstance::IsSpecializedByAutoCompleteMethod()
  812. {
  813. if (mMethodInstanceGroup->mOwner->IsSpecializedByAutoCompleteMethod())
  814. return true;
  815. if (mMethodInfoEx != NULL)
  816. {
  817. for (auto methodArg : mMethodInfoEx->mMethodGenericArguments)
  818. {
  819. // If we are specialized by an autocompleted method reference
  820. if (methodArg->IsMethodRef())
  821. {
  822. auto methodRefType = (BfMethodRefType*)methodArg;
  823. if (methodRefType->mIsAutoCompleteMethod)
  824. return true;
  825. }
  826. }
  827. }
  828. return false;
  829. }
  830. bool BfMethodInstance::IsOrInUnspecializedVariation()
  831. {
  832. return (mIsUnspecializedVariation) || (GetOwner()->IsUnspecializedTypeVariation());
  833. }
  834. bool BfMethodInstance::HasExternConstraints()
  835. {
  836. return (mMethodInfoEx != NULL) && (mMethodInfoEx->mGenericParams.size() > mMethodInfoEx->mMethodGenericArguments.size());
  837. }
  838. bool BfMethodInstance::HasParamsArray()
  839. {
  840. if (mParams.size() == 0)
  841. return false;
  842. return GetParamKind((int)mParams.size() - 1) == BfParamKind_Params;
  843. }
  844. int BfMethodInstance::GetStructRetIdx(bool forceStatic)
  845. {
  846. if ((mReturnType->IsComposite()) && (!mReturnType->IsValuelessType()) && (!GetLoweredReturnType(NULL, NULL, forceStatic)) && (!mIsIntrinsic))
  847. {
  848. auto returnTypeInst = mReturnType->ToTypeInstance();
  849. if ((returnTypeInst != NULL) && (returnTypeInst->mHasUnderlyingArray))
  850. return -1;
  851. auto thisType = mMethodInstanceGroup->mOwner;
  852. if (thisType->mModule->mCompiler->mOptions.mPlatformType != BfPlatformType_Windows)
  853. return 0;
  854. if ((mMethodInfoEx != NULL) && (mMethodInfoEx->mClosureInstanceInfo != NULL) && (mMethodInfoEx->mClosureInstanceInfo->mThisOverride != NULL))
  855. thisType = mMethodInfoEx->mClosureInstanceInfo->mThisOverride;
  856. if ((!HasThis()) || (forceStatic))
  857. return 0;
  858. if (!thisType->IsValueType())
  859. return 1;
  860. if ((mMethodDef->mIsMutating) || (!thisType->IsSplattable()) || ((!AllowsSplatting(-1)) && (!thisType->GetLoweredType(BfTypeUsage_Parameter))))
  861. return 1;
  862. return 0;
  863. }
  864. return -1;
  865. }
  866. bool BfMethodInstance::HasSelf()
  867. {
  868. if (mReturnType->IsSelf())
  869. return true;
  870. for (int paramIdx = 0; paramIdx < GetParamCount(); paramIdx++)
  871. if (GetParamType(paramIdx)->IsSelf())
  872. return true;
  873. return false;
  874. }
  875. bool BfMethodInstance::GetLoweredReturnType(BfTypeCode* loweredTypeCode, BfTypeCode* loweredTypeCode2, bool forceStatic)
  876. {
  877. // Win32 handler
  878. if (((mMethodDef->mIsStatic) || (forceStatic)) &&
  879. (mReturnType->IsComposite()) &&
  880. ((mReturnType->mSize == 4) || (mReturnType->mSize == 8)))
  881. {
  882. auto returnTypeInst = mReturnType->ToTypeInstance();
  883. if ((returnTypeInst != NULL) && (returnTypeInst->mIsCRepr))
  884. {
  885. auto module = GetOwner()->mModule;
  886. auto compiler = module->mCompiler;
  887. if ((compiler->mOptions.mMachineType == BfMachineType_x86) && (compiler->mOptions.mPlatformType == BfPlatformType_Windows))
  888. {
  889. if (loweredTypeCode != NULL)
  890. {
  891. *loweredTypeCode = BfTypeCode_Int32;
  892. if (mReturnType->mSize == 8)
  893. *loweredTypeCode = BfTypeCode_Int64;
  894. }
  895. return true;
  896. }
  897. }
  898. }
  899. return mReturnType->GetLoweredType((mMethodDef->mIsStatic || forceStatic) ? BfTypeUsage_Return_Static : BfTypeUsage_Return_NonStatic, loweredTypeCode, loweredTypeCode2);
  900. }
  901. bool BfMethodInstance::WantsStructsAttribByVal(BfType* paramType)
  902. {
  903. auto owner = GetOwner();
  904. if ((owner->mModule->mCompiler->mOptions.mPlatformType == BfPlatformType_Windows) &&
  905. (owner->mModule->mCompiler->mOptions.mMachineType == BfMachineType_x64))
  906. return false;
  907. if (owner->mModule->mCompiler->mOptions.mMachineType == BfMachineType_AArch64)
  908. return false;
  909. auto typeInst = paramType->ToTypeInstance();
  910. return (typeInst != NULL) && (typeInst->mIsCRepr);
  911. }
  912. bool BfMethodInstance::IsSkipCall(bool bypassVirtual)
  913. {
  914. if ((mMethodDef->mIsSkipCall) &&
  915. ((!mMethodDef->mIsVirtual) || (bypassVirtual)))
  916. return true;
  917. return false;
  918. }
  919. bool BfMethodInstance::IsVarArgs()
  920. {
  921. if (mMethodDef->mParams.IsEmpty())
  922. return false;
  923. return mMethodDef->mParams.back()->mParamKind == BfParamKind_VarArgs;
  924. }
  925. bool BfMethodInstance::AlwaysInline()
  926. {
  927. return mAlwaysInline;
  928. }
  929. BfImportCallKind BfMethodInstance::GetImportCallKind()
  930. {
  931. if (GetImportKind() != BfImportKind_Import_Dynamic)
  932. return BfImportCallKind_None;
  933. if ((mHotMethod != NULL) && ((mHotMethod->mFlags & BfHotDepDataFlag_IsOriginalBuild) == 0))
  934. return BfImportCallKind_GlobalVar_Hot;
  935. return BfImportCallKind_GlobalVar;
  936. }
  937. bool BfMethodInstance::IsTestMethod()
  938. {
  939. return (mMethodInfoEx != NULL) && (mMethodInfoEx->mMethodCustomAttributes != NULL) && (mMethodInfoEx->mMethodCustomAttributes->mCustomAttributes != NULL) &&
  940. (mMethodInfoEx->mMethodCustomAttributes->mCustomAttributes != NULL) && (mMethodInfoEx->mMethodCustomAttributes->mCustomAttributes->Contains(GetOwner()->mModule->mCompiler->mTestAttributeTypeDef));
  941. }
  942. bool BfMethodInstance::AllowsSplatting(int paramIdx)
  943. {
  944. if (paramIdx == -1)
  945. {
  946. if (mCallingConvention != BfCallingConvention_Unspecified)
  947. return false;
  948. if (ForcingThisPtr())
  949. return false;
  950. if (mMethodDef->mIsNoSplat)
  951. return false;
  952. return !mMethodDef->HasNoThisSplat();
  953. }
  954. else
  955. {
  956. if (mCallingConvention != BfCallingConvention_Unspecified)
  957. return false;
  958. if ((mMethodDef->mIsNoSplat) || (mMethodDef->mMethodType == BfMethodType_Mixin))
  959. {
  960. if (IsImplicitCapture(paramIdx))
  961. return true;
  962. return false;
  963. }
  964. return true;
  965. }
  966. }
  967. bool BfMethodInstance::HasThis()
  968. {
  969. if (mMethodDef->mIsStatic)
  970. return false;
  971. if ((mMethodInfoEx != NULL) && (mMethodInfoEx->mClosureInstanceInfo != NULL) && (mMethodInfoEx->mClosureInstanceInfo->mThisOverride != NULL))
  972. return !mMethodInfoEx->mClosureInstanceInfo->mThisOverride->IsValuelessNonOpaqueType();
  973. return (!mMethodInstanceGroup->mOwner->IsValuelessNonOpaqueType());
  974. }
  975. bool BfMethodInstance::IsVirtual()
  976. {
  977. return mMethodDef->mIsVirtual && !mIsInnerOverride;
  978. }
  979. BfType* BfMethodInstance::GetThisType()
  980. {
  981. BF_ASSERT(!mMethodDef->mIsStatic);
  982. if (mMethodDef->mHasExplicitThis)
  983. {
  984. auto thisType = mParams[0].mResolvedType;
  985. auto owner = GetOwner();
  986. if ((thisType->IsValueType()) && ((mMethodDef->mIsMutating) || (!AllowsSplatting(-1))) && (!thisType->GetLoweredType(BfTypeUsage_Parameter)))
  987. return owner->mModule->mContext->mUnreifiedModule->CreatePointerType(thisType);
  988. return thisType;
  989. }
  990. return GetParamType(-1);
  991. }
  992. int BfMethodInstance::GetThisIdx()
  993. {
  994. if (mMethodDef->mIsStatic)
  995. return -2;
  996. if (mMethodDef->mHasExplicitThis)
  997. return 0;
  998. return -1;
  999. }
  1000. bool BfMethodInstance::HasExplicitThis()
  1001. {
  1002. if (mMethodDef->mIsStatic)
  1003. return false;
  1004. return mMethodInstanceGroup->mOwner->IsFunction();
  1005. }
  1006. int BfMethodInstance::GetParamCount()
  1007. {
  1008. return (int)mParams.size();
  1009. }
  1010. int BfMethodInstance::GetImplicitParamCount()
  1011. {
  1012. if ((mMethodInfoEx != NULL) && (mMethodInfoEx->mClosureInstanceInfo != NULL) && (mMethodDef->mIsLocalMethod))
  1013. return (int)mMethodInfoEx->mClosureInstanceInfo->mCaptureEntries.size();
  1014. return 0;
  1015. }
  1016. void BfMethodInstance::GetParamName(int paramIdx, StringImpl& name, int& namePrefixCount)
  1017. {
  1018. if (paramIdx == -1)
  1019. {
  1020. BF_ASSERT(!mMethodDef->mIsStatic);
  1021. name = "this";
  1022. return;
  1023. }
  1024. if ((mMethodInfoEx != NULL) && (mMethodInfoEx->mClosureInstanceInfo != NULL) && (mMethodDef->mIsLocalMethod))
  1025. {
  1026. if (paramIdx < (int)mMethodInfoEx->mClosureInstanceInfo->mCaptureEntries.size())
  1027. {
  1028. name = mMethodInfoEx->mClosureInstanceInfo->mCaptureEntries[paramIdx].mName;
  1029. return;
  1030. }
  1031. }
  1032. BfMethodParam* methodParam = &mParams[paramIdx];
  1033. BfParameterDef* paramDef = mMethodDef->mParams[methodParam->mParamDefIdx];
  1034. if (methodParam->mDelegateParamIdx != -1)
  1035. {
  1036. if (methodParam->mResolvedType->IsTuple())
  1037. {
  1038. auto tupleType = (BfTupleType*)methodParam->mResolvedType;
  1039. auto& fieldInstance = tupleType->mFieldInstances[methodParam->mDelegateParamIdx];
  1040. if (methodParam->mDelegateParamNameCombine)
  1041. name = paramDef->mName + "__" + fieldInstance.GetFieldDef()->mName;
  1042. else
  1043. name = fieldInstance.GetFieldDef()->mName;
  1044. return;
  1045. }
  1046. BfMethodInstance* invokeMethodInstance = methodParam->GetDelegateParamInvoke();
  1047. if (methodParam->mDelegateParamNameCombine)
  1048. name = paramDef->mName + "__" + invokeMethodInstance->GetParamName(methodParam->mDelegateParamIdx);
  1049. else
  1050. invokeMethodInstance->GetParamName(methodParam->mDelegateParamIdx, name, namePrefixCount);
  1051. return;
  1052. }
  1053. name.Reference(paramDef->mName);
  1054. namePrefixCount = paramDef->mNamePrefixCount;
  1055. }
  1056. String BfMethodInstance::GetParamName(int paramIdx)
  1057. {
  1058. StringT<256> paramName;
  1059. int namePrefixCount = 0;
  1060. GetParamName(paramIdx, paramName, namePrefixCount);
  1061. return paramName;
  1062. }
  1063. String BfMethodInstance::GetParamName(int paramIdx, int& namePrefixCount)
  1064. {
  1065. StringT<256> paramName;
  1066. GetParamName(paramIdx, paramName, namePrefixCount);
  1067. return paramName;
  1068. }
  1069. BfType* BfMethodInstance::GetParamType(int paramIdx, bool returnUnderlyingParamsType)
  1070. {
  1071. if (paramIdx == -1)
  1072. {
  1073. if ((mMethodInfoEx != NULL) && (mMethodInfoEx->mClosureInstanceInfo != NULL) && (mMethodInfoEx->mClosureInstanceInfo->mThisOverride != NULL))
  1074. return mMethodInfoEx->mClosureInstanceInfo->mThisOverride;
  1075. BF_ASSERT(!mMethodDef->mIsStatic);
  1076. auto owner = mMethodInstanceGroup->mOwner;
  1077. BfType* thisType = owner;
  1078. if (owner->IsFunction())
  1079. {
  1080. BF_FATAL("Wrong 'this' index");
  1081. }
  1082. if ((thisType->IsValueType()) && ((mMethodDef->mIsMutating) || (!AllowsSplatting(paramIdx))) && (!thisType->GetLoweredType(BfTypeUsage_Parameter)))
  1083. return owner->mModule->CreatePointerType(thisType);
  1084. return thisType;
  1085. }
  1086. BfMethodParam* methodParam = &mParams[paramIdx];
  1087. if (methodParam->mDelegateParamIdx != -1)
  1088. {
  1089. if (methodParam->mResolvedType->IsTuple())
  1090. {
  1091. auto tupleType = (BfTupleType*)methodParam->mResolvedType;
  1092. return tupleType->mFieldInstances[methodParam->mDelegateParamIdx].mResolvedType;
  1093. }
  1094. BfMethodInstance* invokeMethodInstance = methodParam->GetDelegateParamInvoke();
  1095. return invokeMethodInstance->GetParamType(methodParam->mDelegateParamIdx, true);
  1096. }
  1097. if (returnUnderlyingParamsType)
  1098. {
  1099. BfParameterDef* paramDef = mMethodDef->mParams[methodParam->mParamDefIdx];
  1100. if (paramDef->mParamKind == BfParamKind_Params)
  1101. {
  1102. auto underlyingType = methodParam->mResolvedType->GetUnderlyingType();
  1103. if (underlyingType != NULL)
  1104. return underlyingType;
  1105. return methodParam->mResolvedType;
  1106. }
  1107. }
  1108. return methodParam->mResolvedType;
  1109. }
  1110. bool BfMethodInstance::GetParamIsSplat(int paramIdx)
  1111. {
  1112. if (paramIdx == -1)
  1113. {
  1114. BF_ASSERT(!mMethodDef->mIsStatic);
  1115. auto owner = mMethodInstanceGroup->mOwner;
  1116. if ((owner->IsValueType()) && (mMethodDef->mIsMutating || !AllowsSplatting(paramIdx)))
  1117. return false;
  1118. return owner->mIsSplattable;
  1119. }
  1120. BfMethodParam* methodParam = &mParams[paramIdx];
  1121. if (methodParam->mDelegateParamIdx != -1)
  1122. {
  1123. BfMethodInstance* invokeMethodInstance = methodParam->GetDelegateParamInvoke();
  1124. if (invokeMethodInstance == NULL)
  1125. return false;
  1126. return invokeMethodInstance->GetParamIsSplat(methodParam->mDelegateParamIdx);
  1127. }
  1128. return methodParam->mIsSplat;
  1129. }
  1130. BfParamKind BfMethodInstance::GetParamKind(int paramIdx)
  1131. {
  1132. if (paramIdx == -1)
  1133. return BfParamKind_Normal;
  1134. BfMethodParam* methodParam = &mParams[paramIdx];
  1135. if (methodParam->mParamDefIdx == -1)
  1136. return BfParamKind_ImplicitCapture;
  1137. BfParameterDef* paramDef = mMethodDef->mParams[methodParam->mParamDefIdx];
  1138. if (methodParam->mDelegateParamIdx != -1)
  1139. return BfParamKind_DelegateParam;
  1140. return paramDef->mParamKind;
  1141. }
  1142. bool BfMethodInstance::WasGenericParam(int paramIdx)
  1143. {
  1144. if (paramIdx == -1)
  1145. return false;
  1146. BfMethodParam* methodParam = &mParams[paramIdx];
  1147. return methodParam->mWasGenericParam;
  1148. }
  1149. bool BfMethodInstance::IsParamSkipped(int paramIdx)
  1150. {
  1151. auto resolveModule = GetModule()->mContext->mUnreifiedModule;
  1152. if (paramIdx == -1)
  1153. return false;
  1154. BfType* paramType = GetParamType(paramIdx);
  1155. if ((paramType->CanBeValuelessType()) && (paramType->IsDataIncomplete()))
  1156. resolveModule->PopulateType(paramType, BfPopulateType_Data);
  1157. if ((paramType->IsValuelessNonOpaqueType()) && (!paramType->IsMethodRef()))
  1158. return true;
  1159. return false;
  1160. }
  1161. bool BfMethodInstance::IsImplicitCapture(int paramIdx)
  1162. {
  1163. if (paramIdx == -1)
  1164. return false;
  1165. BfMethodParam* methodParam = &mParams[paramIdx];
  1166. if (methodParam->mParamDefIdx == -1)
  1167. return true;
  1168. return false;
  1169. }
  1170. BfExpression* BfMethodInstance::GetParamInitializer(int paramIdx)
  1171. {
  1172. if (paramIdx == -1)
  1173. return NULL;
  1174. BfMethodParam* methodParam = &mParams[paramIdx];
  1175. if (methodParam->mParamDefIdx == -1)
  1176. return NULL;
  1177. BfParameterDef* paramDef = mMethodDef->mParams[methodParam->mParamDefIdx];
  1178. if (paramDef->mParamDeclaration != NULL)
  1179. return paramDef->mParamDeclaration->mInitializer;
  1180. return NULL;
  1181. }
  1182. BfTypeReference* BfMethodInstance::GetParamTypeRef(int paramIdx)
  1183. {
  1184. if (paramIdx == -1)
  1185. return NULL;
  1186. BfMethodParam* methodParam = &mParams[paramIdx];
  1187. if (methodParam->mParamDefIdx == -1)
  1188. return NULL;
  1189. BfParameterDef* paramDef = mMethodDef->mParams[methodParam->mParamDefIdx];
  1190. if (paramDef->mParamDeclaration != NULL)
  1191. return paramDef->mParamDeclaration->mTypeRef;
  1192. return NULL;
  1193. }
  1194. BfIdentifierNode* BfMethodInstance::GetParamNameNode(int paramIdx)
  1195. {
  1196. if (paramIdx == -1)
  1197. return NULL;
  1198. if ((mMethodInfoEx != NULL) && (mMethodInfoEx->mClosureInstanceInfo != NULL) && (mMethodDef->mIsLocalMethod))
  1199. {
  1200. if (paramIdx < (int)mMethodInfoEx->mClosureInstanceInfo->mCaptureEntries.size())
  1201. return mMethodInfoEx->mClosureInstanceInfo->mCaptureEntries[paramIdx].mNameNode;
  1202. }
  1203. BfMethodParam* methodParam = &mParams[paramIdx];
  1204. BfParameterDef* paramDef = mMethodDef->mParams[methodParam->mParamDefIdx];
  1205. if (paramDef->mParamDeclaration != NULL)
  1206. return BfNodeDynCast<BfIdentifierNode>(paramDef->mParamDeclaration->mNameNode);
  1207. return NULL;
  1208. }
  1209. int BfMethodInstance::DbgGetVirtualMethodNum()
  1210. {
  1211. auto module = GetOwner()->mModule;
  1212. int vDataVal = -1;
  1213. if (mVirtualTableIdx != -1)
  1214. {
  1215. module->HadSlotCountDependency();
  1216. int vDataIdx = -1;
  1217. vDataIdx = module->mCompiler->GetVDataPrefixDataCount() + module->mCompiler->mMaxInterfaceSlots;
  1218. vDataIdx += module->mCompiler->GetDynCastVDataCount();
  1219. if ((module->mCompiler->mOptions.mHasVDataExtender) && (module->mCompiler->IsHotCompile()))
  1220. {
  1221. auto typeInst = mMethodInstanceGroup->mOwner;
  1222. int extMethodIdx = (mVirtualTableIdx - typeInst->GetImplBaseVTableSize()) - typeInst->GetOrigSelfVTableSize();
  1223. if (extMethodIdx >= 0)
  1224. {
  1225. // Extension?
  1226. int vExtOfs = typeInst->GetOrigImplBaseVTableSize();
  1227. vDataVal = ((vDataIdx + vExtOfs + 1) << 20) | (extMethodIdx);
  1228. }
  1229. else
  1230. {
  1231. // Map this new virtual index back to the original index
  1232. vDataIdx += (mVirtualTableIdx - typeInst->GetImplBaseVTableSize()) + typeInst->GetOrigImplBaseVTableSize();
  1233. }
  1234. }
  1235. else
  1236. {
  1237. vDataIdx += mVirtualTableIdx;
  1238. }
  1239. if (vDataVal == -1)
  1240. vDataVal = vDataIdx;
  1241. }
  1242. return vDataVal;
  1243. }
  1244. void BfMethodInstance::GetIRFunctionInfo(BfModule* module, BfIRType& returnType, SizedArrayImpl<BfIRType>& paramTypes, bool forceStatic)
  1245. {
  1246. BfModule* resolveModule = module->mContext->mUnreifiedModule;
  1247. resolveModule->PopulateType(mReturnType);
  1248. BfTypeCode loweredReturnTypeCode = BfTypeCode_None;
  1249. BfTypeCode loweredReturnTypeCode2 = BfTypeCode_None;
  1250. if ((!module->mIsComptimeModule) && (GetLoweredReturnType(&loweredReturnTypeCode, &loweredReturnTypeCode2, forceStatic)) && (loweredReturnTypeCode != BfTypeCode_None))
  1251. {
  1252. auto irReturnType = module->GetIRLoweredType(loweredReturnTypeCode, loweredReturnTypeCode2);
  1253. returnType = irReturnType;
  1254. }
  1255. else if ((mReturnType->IsValuelessType()) || (mReturnType->IsVar()))
  1256. {
  1257. auto voidType = module->GetPrimitiveType(BfTypeCode_None);
  1258. returnType = module->mBfIRBuilder->MapType(voidType);
  1259. }
  1260. else if ((!module->mIsComptimeModule) && (GetStructRetIdx(forceStatic) != -1))
  1261. {
  1262. auto voidType = module->GetPrimitiveType(BfTypeCode_None);
  1263. returnType = module->mBfIRBuilder->MapType(voidType);
  1264. auto typeInst = mReturnType->ToTypeInstance();
  1265. if (typeInst != NULL)
  1266. {
  1267. paramTypes.push_back(module->mBfIRBuilder->MapTypeInstPtr(typeInst));
  1268. }
  1269. else
  1270. {
  1271. auto ptrType = module->CreatePointerType(mReturnType);
  1272. paramTypes.push_back(module->mBfIRBuilder->MapType(ptrType));
  1273. }
  1274. }
  1275. else
  1276. {
  1277. returnType = module->mBfIRBuilder->MapType(mReturnType);
  1278. }
  1279. for (int paramIdx = -1; paramIdx < GetParamCount(); paramIdx++)
  1280. {
  1281. BfType* checkType = NULL;
  1282. if (paramIdx == -1)
  1283. {
  1284. if ((mMethodDef->mIsStatic) || (forceStatic))
  1285. continue;
  1286. if (mIsClosure)
  1287. {
  1288. checkType = module->mCurMethodState->mClosureState->mClosureType;
  1289. }
  1290. else
  1291. {
  1292. if (HasExplicitThis())
  1293. checkType = GetParamType(0);
  1294. else
  1295. checkType = GetOwner();
  1296. }
  1297. }
  1298. else
  1299. {
  1300. if ((paramIdx == 0) && (mMethodDef->mHasExplicitThis))
  1301. continue; // Skip over the explicit 'this'
  1302. checkType = GetParamType(paramIdx);
  1303. }
  1304. /*if (GetParamName(paramIdx) == "this")
  1305. {
  1306. NOP;
  1307. }*/
  1308. bool checkLowered = false;
  1309. bool doSplat = false;
  1310. if (paramIdx == -1)
  1311. {
  1312. if (ForcingThisPtr())
  1313. {
  1314. // Pass by pointer even for typed primitives
  1315. }
  1316. else if ((!mMethodDef->mIsMutating) && (checkType->IsTypedPrimitive()))
  1317. {
  1318. checkType = checkType->GetUnderlyingType();
  1319. }
  1320. else if ((!module->mIsComptimeModule) && (checkType->IsSplattable()) && (AllowsSplatting(-1)))
  1321. {
  1322. doSplat = true;
  1323. }
  1324. else if ((!module->mIsComptimeModule) && (!mMethodDef->mIsMutating) && (mCallingConvention == BfCallingConvention_Unspecified))
  1325. checkLowered = true;
  1326. }
  1327. else
  1328. {
  1329. if ((checkType->IsComposite()) && (checkType->IsIncomplete()))
  1330. resolveModule->PopulateType(checkType, BfPopulateType_Data);
  1331. if (checkType->IsMethodRef())
  1332. {
  1333. doSplat = true;
  1334. }
  1335. else if (checkType->IsTypedPrimitive())
  1336. {
  1337. checkType = checkType->GetUnderlyingType();
  1338. }
  1339. else if ((!module->mIsComptimeModule) && (checkType->IsSplattable()) && (AllowsSplatting(paramIdx)))
  1340. {
  1341. doSplat = true;
  1342. }
  1343. else if (!module->mIsComptimeModule)
  1344. checkLowered = true;
  1345. }
  1346. BfType* checkType2 = NULL;
  1347. if (checkLowered)
  1348. {
  1349. BfTypeCode loweredTypeCode = BfTypeCode_None;
  1350. BfTypeCode loweredTypeCode2 = BfTypeCode_None;
  1351. if (checkType->GetLoweredType(BfTypeUsage_Parameter, &loweredTypeCode, &loweredTypeCode2))
  1352. {
  1353. paramTypes.push_back(module->mBfIRBuilder->GetPrimitiveType(loweredTypeCode));
  1354. if (loweredTypeCode2 != BfTypeCode_None)
  1355. paramTypes.push_back(module->mBfIRBuilder->GetPrimitiveType(loweredTypeCode2));
  1356. continue;
  1357. }
  1358. }
  1359. if (checkType->CanBeValuelessType())
  1360. resolveModule->PopulateType(checkType, BfPopulateType_Data);
  1361. if ((checkType->IsValuelessNonOpaqueType()) && (!checkType->IsMethodRef()))
  1362. continue;
  1363. if ((doSplat) && (!checkType->IsMethodRef()))
  1364. {
  1365. int splatCount = checkType->GetSplatCount();
  1366. if ((int)paramTypes.size() + splatCount > module->mCompiler->mOptions.mMaxSplatRegs)
  1367. {
  1368. auto checkTypeInst = checkType->ToTypeInstance();
  1369. if ((checkTypeInst != NULL) && (checkTypeInst->mIsCRepr))
  1370. {
  1371. // CRepr splat means always splat
  1372. }
  1373. else
  1374. doSplat = false;
  1375. }
  1376. }
  1377. auto _AddType = [&](BfType* type)
  1378. {
  1379. if ((type->IsComposite()) || ((!doSplat) && (paramIdx == -1) && (type->IsTypedPrimitive())))
  1380. {
  1381. auto typeInst = type->ToTypeInstance();
  1382. if (typeInst != NULL)
  1383. paramTypes.push_back(module->mBfIRBuilder->MapTypeInstPtr(typeInst));
  1384. else
  1385. paramTypes.push_back(module->mBfIRBuilder->MapType(module->CreatePointerType(type)));
  1386. }
  1387. else
  1388. {
  1389. paramTypes.push_back(module->mBfIRBuilder->MapType(type));
  1390. }
  1391. };
  1392. if (doSplat)
  1393. {
  1394. BfTypeUtils::SplatIterate([&](BfType* checkType)
  1395. {
  1396. _AddType(checkType);
  1397. }, checkType);
  1398. }
  1399. else
  1400. _AddType(checkType);
  1401. if (checkType2 != NULL)
  1402. _AddType(checkType2);
  1403. }
  1404. if ((!module->mIsComptimeModule) && (GetStructRetIdx(forceStatic) == 1))
  1405. {
  1406. BF_SWAP(paramTypes[0], paramTypes[1]);
  1407. }
  1408. }
  1409. int BfMethodInstance::GetIRFunctionParamCount(BfModule* module)
  1410. {
  1411. //TODO: This is dumb, do this better
  1412. SizedArray<BfIRType, 8> params;
  1413. BfIRType returnType;
  1414. GetIRFunctionInfo(module, returnType, params);
  1415. return (int)params.size();
  1416. }
  1417. bool BfMethodInstance::IsExactMatch(BfMethodInstance* other, bool ignoreImplicitParams, bool checkThis)
  1418. {
  1419. if (mReturnType != other->mReturnType)
  1420. return false;
  1421. int implicitParamCountA = ignoreImplicitParams ? GetImplicitParamCount() : 0;
  1422. int implicitParamCountB = ignoreImplicitParams ? other->GetImplicitParamCount() : 0;
  1423. if (HasExplicitThis())
  1424. {
  1425. }
  1426. // if (other->HasExplicitThis())
  1427. // {
  1428. // if (!HasExplicitThis())
  1429. // return false;
  1430. // if (GetParamType(-1) != other->GetParamType(-1))
  1431. // return false;
  1432. // }
  1433. if (checkThis)
  1434. {
  1435. if (other->mMethodDef->mIsStatic != mMethodDef->mIsStatic)
  1436. return false;
  1437. // {
  1438. // // If we are static and we have to match a non-static method, allow us to do so if we have an explicitly defined 'this' param that matches
  1439. //
  1440. // if (other->mMethodDef->mIsStatic)
  1441. // return false;
  1442. //
  1443. // if ((GetParamCount() > 0) && (GetParamName(0) == "this"))
  1444. // {
  1445. // auto thisType = GetParamType(0);
  1446. // auto otherThisType = other->GetParamType(-1);
  1447. // if (thisType != otherThisType)
  1448. // return false;
  1449. //
  1450. // implicitParamCountA++;
  1451. // }
  1452. // else
  1453. // {
  1454. // // Valueless types don't actually pass a 'this' anyway
  1455. // if (!other->GetOwner()->IsValuelessType())
  1456. // return false;
  1457. // }
  1458. // }
  1459. if (!mMethodDef->mIsStatic)
  1460. {
  1461. if (GetThisType() != other->GetThisType())
  1462. {
  1463. return false;
  1464. }
  1465. }
  1466. }
  1467. if (mMethodDef->mHasExplicitThis)
  1468. implicitParamCountA++;
  1469. if (other->mMethodDef->mHasExplicitThis)
  1470. implicitParamCountB++;
  1471. if (GetParamCount() - implicitParamCountA != other->GetParamCount() - implicitParamCountB)
  1472. return false;
  1473. for (int i = 0; i < (int)GetParamCount() - implicitParamCountA; i++)
  1474. {
  1475. auto paramA = GetParamType(i + implicitParamCountA);
  1476. auto paramB = other->GetParamType(i + implicitParamCountB);
  1477. if (paramA != paramB)
  1478. return false;
  1479. }
  1480. return true;
  1481. }
  1482. bool BfMethodInstance::IsReifiedAndImplemented()
  1483. {
  1484. return mIsReified && mMethodInstanceGroup->IsImplemented();
  1485. }
  1486. BfMethodInfoEx* BfMethodInstance::GetMethodInfoEx()
  1487. {
  1488. if (mMethodInfoEx == NULL)
  1489. mMethodInfoEx = new BfMethodInfoEx();
  1490. return mMethodInfoEx;
  1491. }
  1492. void BfMethodInstance::ReportMemory(MemReporter* memReporter)
  1493. {
  1494. memReporter->BeginSection("MethodInstance");
  1495. memReporter->Add(sizeof(BfMethodInstance));
  1496. if (mMethodInfoEx != NULL)
  1497. {
  1498. memReporter->BeginSection("MethodInfoEx");
  1499. memReporter->Add(sizeof(BfMethodInfoEx));
  1500. if (!mMethodInfoEx->mGenericParams.IsEmpty())
  1501. memReporter->AddVecPtr("GenericParams", mMethodInfoEx->mGenericParams, false);
  1502. if (!mMethodInfoEx->mGenericTypeBindings.IsEmpty())
  1503. memReporter->AddMap("GenericTypeBindings", mMethodInfoEx->mGenericTypeBindings, false);
  1504. if (mMethodInfoEx->mMethodCustomAttributes != NULL)
  1505. {
  1506. if (mMethodInfoEx->mMethodCustomAttributes->mCustomAttributes != NULL)
  1507. mMethodInfoEx->mMethodCustomAttributes->mCustomAttributes->ReportMemory(memReporter);
  1508. }
  1509. if (!mMethodInfoEx->mMethodGenericArguments.IsEmpty())
  1510. memReporter->AddVec("MethodGenericArguments", mMethodInfoEx->mMethodGenericArguments, false);
  1511. if (!mMethodInfoEx->mMangledName.IsEmpty())
  1512. memReporter->AddStr("MangledName", mMethodInfoEx->mMangledName);
  1513. memReporter->EndSection();
  1514. }
  1515. memReporter->AddVec("Params", mParams, false);
  1516. if (!mDefaultValues.IsEmpty())
  1517. memReporter->AddVec("DefaultValues", mDefaultValues, false);
  1518. memReporter->EndSection();
  1519. }
  1520. void BfCustomAttributes::ReportMemory(MemReporter* memReporter)
  1521. {
  1522. memReporter->BeginSection("CustomAttributes");
  1523. memReporter->Add(sizeof(BfCustomAttributes));
  1524. memReporter->AddVec(mAttributes);
  1525. memReporter->EndSection();
  1526. }
  1527. //////////////////////////////////////////////////////////////////////////
  1528. BfModuleMethodInstance::BfModuleMethodInstance(BfMethodInstance* methodInstance)
  1529. {
  1530. mMethodInstance = methodInstance;
  1531. if (methodInstance != NULL)
  1532. mFunc = mMethodInstance->mIRFunction;
  1533. else
  1534. mFunc = BfIRValue();
  1535. // if (methodInstance->GetImportCallKind() == BfImportCallKind_Thunk)
  1536. // {
  1537. // auto declModule = methodInstance->mDeclModule;
  1538. // BfIRValue* irFuncPtr = NULL;
  1539. // if (declModule->mFuncReferences.TryGetValue(methodInstance, &irFuncPtr))
  1540. // mFunc = *irFuncPtr;
  1541. // }
  1542. }
  1543. //////////////////////////////////////////////////////////////////////////
  1544. BfMethodRefType::~BfMethodRefType()
  1545. {
  1546. if (!mContext->mDeleting)
  1547. BF_ASSERT(mMethodRef == NULL);
  1548. }
  1549. BfMethodInstanceGroup::BfMethodInstanceGroup(BfMethodInstanceGroup&& prev) noexcept
  1550. {
  1551. mOwner = prev.mOwner;
  1552. mDefault = prev.mDefault;
  1553. mMethodSpecializationMap = prev.mMethodSpecializationMap;
  1554. mDefaultCustomAttributes = prev.mDefaultCustomAttributes;
  1555. mMethodIdx = prev.mMethodIdx;
  1556. mRefCount = prev.mRefCount;
  1557. mOnDemandKind = prev.mOnDemandKind;
  1558. mExplicitlyReflected = prev.mExplicitlyReflected;
  1559. mHasEmittedReference = prev.mHasEmittedReference;
  1560. if (mDefault != NULL)
  1561. mDefault->mMethodInstanceGroup = this;
  1562. if (mMethodSpecializationMap != NULL)
  1563. {
  1564. for (auto& pair : *mMethodSpecializationMap)
  1565. pair.mValue->mMethodInstanceGroup = this;
  1566. }
  1567. prev.mDefaultCustomAttributes = NULL;
  1568. prev.mRefCount = 0;
  1569. prev.mDefault = NULL;
  1570. prev.mMethodSpecializationMap = NULL;
  1571. }
  1572. BfMethodInstanceGroup::~BfMethodInstanceGroup()
  1573. {
  1574. if (mRefCount != 0)
  1575. {
  1576. BF_ASSERT(mOwner->mModule->mContext->mDeleting);
  1577. }
  1578. delete mDefault;
  1579. if (mMethodSpecializationMap != NULL)
  1580. {
  1581. for (auto& kv : *mMethodSpecializationMap)
  1582. delete kv.mValue;
  1583. delete mMethodSpecializationMap;
  1584. }
  1585. delete mDefaultCustomAttributes;
  1586. }
  1587. //////////////////////////////////////////////////////////////////////////
  1588. BfTypeInstance::~BfTypeInstance()
  1589. {
  1590. ReleaseData();
  1591. delete mTypeInfoEx;
  1592. delete mGenericTypeInfo;
  1593. delete mCeTypeInfo;
  1594. delete mCustomAttributes;
  1595. delete mAttributeData;
  1596. for (auto methodInst : mInternalMethods)
  1597. delete methodInst;
  1598. for (auto operatorInfo : mOperatorInfo)
  1599. delete operatorInfo;
  1600. for (auto localMethod : mOwnedLocalMethods)
  1601. delete localMethod;
  1602. delete mHotTypeData;
  1603. delete mConstHolder;
  1604. if ((mTypeDef != NULL) && (mTypeDef->mEmitParent != NULL))
  1605. {
  1606. mMethodInstanceGroups.Clear();
  1607. BfLogSys(mModule->mSystem, "Type %p dtor deleting typeDef %p\n", this, mTypeDef);
  1608. delete mTypeDef;
  1609. }
  1610. }
  1611. void BfTypeInstance::ReleaseData()
  1612. {
  1613. for (auto& kv : mInternalAccessMap)
  1614. {
  1615. auto& internalAcessSet = kv.mValue;
  1616. for (auto& namespaceComposite : internalAcessSet.mNamespaces)
  1617. mModule->mSystem->ReleaseAtomComposite(namespaceComposite);
  1618. }
  1619. mInternalAccessMap.Clear();
  1620. }
  1621. void BfTypeInstance::Dispose()
  1622. {
  1623. delete mGenericTypeInfo;
  1624. mGenericTypeInfo = NULL;
  1625. mTypeDef = NULL;
  1626. }
  1627. int BfTypeInstance::GetSplatCount(bool force)
  1628. {
  1629. if (IsValuelessType())
  1630. return 0;
  1631. if ((!mIsSplattable) && (!force))
  1632. return 1;
  1633. int splatCount = 0;
  1634. BfTypeUtils::SplatIterate([&](BfType* checkType) { splatCount++; }, this);
  1635. return splatCount;
  1636. }
  1637. bool BfTypeInstance::IsCRepr()
  1638. {
  1639. return mIsCRepr;
  1640. }
  1641. bool BfTypeInstance::IsString()
  1642. {
  1643. return IsInstanceOf(mContext->mCompiler->mStringTypeDef);
  1644. }
  1645. int BfTypeInstance::GetOrigVTableSize()
  1646. {
  1647. if (!mModule->mCompiler->mOptions.mHasVDataExtender)
  1648. {
  1649. BF_ASSERT(mHotTypeData == NULL);
  1650. return mVirtualMethodTableSize;
  1651. }
  1652. if (mHotTypeData != NULL)
  1653. {
  1654. // When we have a pending data change, treat it as a fresh vtable
  1655. if ((!mHotTypeData->mPendingDataChange) && (mHotTypeData->mVTableOrigLength != -1))
  1656. return mHotTypeData->mVTableOrigLength;
  1657. }
  1658. if (mBaseType != NULL)
  1659. return mBaseType->GetOrigVTableSize() + (mVirtualMethodTableSize - mBaseType->mVirtualMethodTableSize);
  1660. return mVirtualMethodTableSize;
  1661. }
  1662. int BfTypeInstance::GetSelfVTableSize()
  1663. {
  1664. if (mBaseType != NULL)
  1665. {
  1666. BF_ASSERT(mBaseType->mVirtualMethodTableSize > 0);
  1667. return mVirtualMethodTableSize - mBaseType->mVirtualMethodTableSize;
  1668. }
  1669. return mVirtualMethodTableSize;
  1670. }
  1671. int BfTypeInstance::GetOrigSelfVTableSize()
  1672. {
  1673. if (mBaseType != NULL)
  1674. return GetOrigVTableSize() - GetOrigImplBaseVTableSize();
  1675. return GetOrigVTableSize();
  1676. }
  1677. int BfTypeInstance::GetImplBaseVTableSize()
  1678. {
  1679. auto implBase = GetImplBaseType();
  1680. if (implBase != NULL)
  1681. return implBase->mVirtualMethodTableSize;
  1682. return 0;
  1683. }
  1684. int BfTypeInstance::GetOrigImplBaseVTableSize()
  1685. {
  1686. auto implBase = GetImplBaseType();
  1687. if (implBase != NULL)
  1688. return mBaseType->GetOrigVTableSize();
  1689. return 0;
  1690. }
  1691. int BfTypeInstance::GetIFaceVMethodSize()
  1692. {
  1693. int maxIFaceIdx = 0;
  1694. auto checkTypeInstance = this;
  1695. while (checkTypeInstance != NULL)
  1696. {
  1697. for (auto&& interfaceEntry : checkTypeInstance->mInterfaces)
  1698. {
  1699. maxIFaceIdx = BF_MAX(maxIFaceIdx, interfaceEntry.mStartVirtualIdx + interfaceEntry.mInterfaceType->mVirtualMethodTableSize);
  1700. }
  1701. checkTypeInstance = checkTypeInstance->mBaseType;
  1702. }
  1703. return maxIFaceIdx;
  1704. }
  1705. BfType* BfTypeInstance::GetUnionInnerType(bool* wantSplat)
  1706. {
  1707. if (wantSplat != NULL)
  1708. *wantSplat = false;
  1709. if (!mIsUnion)
  1710. return NULL;
  1711. BfTypeState typeState(this, mContext->mCurTypeState);
  1712. typeState.mPopulateType = BfPopulateType_Data;
  1713. SetAndRestoreValue<BfTypeState*> prevTypeState(mContext->mCurTypeState, &typeState);
  1714. int unionSize = 0;
  1715. BfType* unionInnerType = NULL;
  1716. bool makeRaw = false;
  1717. for (int fieldIdx = 0; fieldIdx < (int)mFieldInstances.size(); fieldIdx++)
  1718. {
  1719. auto fieldInstance = (BfFieldInstance*)&mFieldInstances[fieldIdx];
  1720. auto fieldDef = fieldInstance->GetFieldDef();
  1721. BfType* checkInnerType = NULL;
  1722. if (fieldDef == NULL)
  1723. continue;
  1724. if ((fieldDef->mIsConst) && (fieldInstance->mIsEnumPayloadCase))
  1725. {
  1726. BF_ASSERT(mIsUnion);
  1727. checkInnerType = fieldInstance->mResolvedType;
  1728. }
  1729. if (fieldInstance->mDataIdx >= 0)
  1730. {
  1731. checkInnerType = fieldInstance->mResolvedType;
  1732. }
  1733. if (checkInnerType != NULL)
  1734. {
  1735. SetAndRestoreValue<BfFieldDef*> prevTypeRef(mContext->mCurTypeState->mCurFieldDef, fieldDef);
  1736. if (checkInnerType->IsDataIncomplete())
  1737. mModule->PopulateType(checkInnerType, checkInnerType->IsValueType() ? BfPopulateType_Data : BfPopulateType_Declaration);
  1738. if (checkInnerType->mSize > unionSize)
  1739. unionSize = checkInnerType->mSize;
  1740. if ((!checkInnerType->IsValuelessType()) && (checkInnerType != unionInnerType))
  1741. {
  1742. if (unionInnerType == NULL)
  1743. {
  1744. unionInnerType = checkInnerType;
  1745. }
  1746. else
  1747. {
  1748. if (checkInnerType->mSize > unionInnerType->mSize)
  1749. {
  1750. bool needsMemberCasting = false;
  1751. if (!mModule->AreSplatsCompatible(checkInnerType, unionInnerType, &needsMemberCasting))
  1752. {
  1753. unionInnerType = NULL;
  1754. makeRaw = true;
  1755. }
  1756. else
  1757. {
  1758. unionInnerType = checkInnerType;
  1759. }
  1760. }
  1761. else
  1762. {
  1763. bool needsMemberCasting = false;
  1764. if (!mModule->AreSplatsCompatible(unionInnerType, checkInnerType, &needsMemberCasting))
  1765. {
  1766. unionInnerType = NULL;
  1767. makeRaw = true;
  1768. }
  1769. }
  1770. }
  1771. }
  1772. }
  1773. }
  1774. BF_ASSERT(unionInnerType != this);
  1775. // Don't allow a float for the inner type -- to avoid invalid loading invalid FP bit patterns during copies
  1776. if ((unionInnerType != NULL) && (!makeRaw))
  1777. {
  1778. //TODO: How did splats make sense? It breaks CompactList
  1779. //if (wantSplat != NULL)
  1780. // *wantSplat = true;
  1781. }
  1782. else
  1783. {
  1784. switch (unionSize)
  1785. {
  1786. case 0: return mModule->CreateSizedArrayType(mModule->GetPrimitiveType(BfTypeCode_Int8), 0);
  1787. case 1: return mModule->GetPrimitiveType(BfTypeCode_Int8);
  1788. case 2: if (mInstAlign >= 2) return mModule->GetPrimitiveType(BfTypeCode_Int16);
  1789. case 4: if (mInstAlign >= 4) return mModule->GetPrimitiveType(BfTypeCode_Int32);
  1790. case 8: if (mInstAlign >= 8) return mModule->GetPrimitiveType(BfTypeCode_Int64);
  1791. }
  1792. if ((unionSize % 8 == 0) && (mInstAlign >= 8))
  1793. return mModule->CreateSizedArrayType(mModule->GetPrimitiveType(BfTypeCode_Int64), unionSize / 8);
  1794. if ((unionSize % 4 == 0) && (mInstAlign >= 4))
  1795. return mModule->CreateSizedArrayType(mModule->GetPrimitiveType(BfTypeCode_Int32), unionSize / 4);
  1796. if ((unionSize % 2 == 0) && (mInstAlign >= 2))
  1797. return mModule->CreateSizedArrayType(mModule->GetPrimitiveType(BfTypeCode_Int16), unionSize / 2);
  1798. return mModule->CreateSizedArrayType(mModule->GetPrimitiveType(BfTypeCode_Int8), unionSize);
  1799. }
  1800. return unionInnerType;
  1801. }
  1802. BfPrimitiveType* BfTypeInstance::GetDiscriminatorType(int* outDataIdx)
  1803. {
  1804. BF_ASSERT(IsPayloadEnum());
  1805. auto& fieldInstance = mFieldInstances.back();
  1806. BF_ASSERT(fieldInstance.GetFieldDef() == NULL);
  1807. if (fieldInstance.mResolvedType == NULL)
  1808. {
  1809. BF_ASSERT(IsIncomplete());
  1810. // Use Int64 as a placeholder until we determine the correct type...
  1811. return mModule->GetPrimitiveType(BfTypeCode_Int64);
  1812. }
  1813. BF_ASSERT(fieldInstance.mResolvedType != NULL);
  1814. BF_ASSERT(fieldInstance.mResolvedType->IsPrimitiveType());
  1815. if (outDataIdx != NULL)
  1816. *outDataIdx = fieldInstance.mDataIdx;
  1817. return (BfPrimitiveType*)fieldInstance.mResolvedType;
  1818. }
  1819. void BfTypeInstance::GetUnderlyingArray(BfType*& type, int& size, bool& isVector)
  1820. {
  1821. if (mCustomAttributes == NULL)
  1822. return;
  1823. auto attributes = mCustomAttributes->Get(mModule->mCompiler->mUnderlyingArrayAttributeTypeDef);
  1824. if (attributes == NULL)
  1825. return;
  1826. if (attributes->mCtorArgs.size() != 3)
  1827. return;
  1828. auto typeConstant = mConstHolder->GetConstant(attributes->mCtorArgs[0]);
  1829. auto sizeConstant = mConstHolder->GetConstant(attributes->mCtorArgs[1]);
  1830. auto isVectorConstant = mConstHolder->GetConstant(attributes->mCtorArgs[2]);
  1831. if ((typeConstant == NULL) || (sizeConstant == NULL) || (isVectorConstant == NULL))
  1832. return;
  1833. if (typeConstant->mConstType != BfConstType_TypeOf)
  1834. return;
  1835. type = (BfType*)(intptr)typeConstant->mInt64;
  1836. size = sizeConstant->mInt32;
  1837. isVector = isVectorConstant->mBool;
  1838. }
  1839. bool BfTypeInstance::GetLoweredType(BfTypeUsage typeUsage, BfTypeCode* outTypeCode, BfTypeCode* outTypeCode2)
  1840. {
  1841. if ((mTypeDef->mTypeCode != BfTypeCode_Struct) || (IsBoxed()) || (mIsSplattable))
  1842. return false;
  1843. if (mHasUnderlyingArray)
  1844. return false;
  1845. bool deepCheck = false;
  1846. if (mModule->mCompiler->mOptions.mMachineType == BfMachineType_Wasm32)
  1847. {
  1848. if (IsComposite())
  1849. {
  1850. if (GetSplatCount(true) == 1)
  1851. {
  1852. BfType* componentType = NULL;
  1853. BfTypeUtils::SplatIterate([&](BfType* checkType) { componentType = checkType; }, this);
  1854. if (componentType != NULL)
  1855. {
  1856. if (componentType->IsPrimitiveType())
  1857. {
  1858. auto primType = (BfPrimitiveType*)componentType;
  1859. if (outTypeCode != NULL)
  1860. *outTypeCode = primType->mTypeDef->mTypeCode;
  1861. return true;
  1862. }
  1863. }
  1864. }
  1865. else
  1866. return false;
  1867. }
  1868. }
  1869. if (mModule->mCompiler->mOptions.mPlatformType == BfPlatformType_Windows)
  1870. {
  1871. // Odd Windows rule: composite returns for non-static methods are always sret
  1872. if (typeUsage == BfTypeUsage_Return_NonStatic)
  1873. return false;
  1874. }
  1875. else
  1876. {
  1877. // Non-Win64 systems allow lowered splitting of composites over multiple params
  1878. if (mModule->mSystem->mPtrSize == 8)
  1879. deepCheck = true;
  1880. else
  1881. {
  1882. // We know this is correct for Linux x86 and Android armv7
  1883. if (mModule->mCompiler->mOptions.mPlatformType == BfPlatformType_Linux)
  1884. {
  1885. if ((typeUsage == BfTypeUsage_Return_NonStatic) || (typeUsage == BfTypeUsage_Return_Static))
  1886. return false;
  1887. }
  1888. }
  1889. }
  1890. int maxInstSize = 16;
  1891. if (mModule->mCompiler->mOptions.mMachineType == BfMachineType_AArch64)
  1892. maxInstSize = 32;
  1893. if (deepCheck)
  1894. {
  1895. if ((mInstSize >= 4) && (mInstSize <= maxInstSize))
  1896. {
  1897. BfTypeCode types[8] = { BfTypeCode_None };
  1898. std::function<void(BfType*, int)> _CheckType = [&](BfType* type, int offset)
  1899. {
  1900. if (auto typeInst = type->ToTypeInstance())
  1901. {
  1902. if (typeInst->IsValueType())
  1903. {
  1904. if (typeInst->mBaseType != NULL)
  1905. _CheckType(typeInst->mBaseType, offset);
  1906. for (auto& fieldInstance : typeInst->mFieldInstances)
  1907. {
  1908. if (fieldInstance.mDataOffset >= 0)
  1909. _CheckType(fieldInstance.mResolvedType, offset + fieldInstance.mDataOffset);
  1910. }
  1911. }
  1912. else
  1913. {
  1914. types[offset / 4] = BfTypeCode_Object;
  1915. }
  1916. }
  1917. else if (type->IsPrimitiveType())
  1918. {
  1919. auto primType = (BfPrimitiveType*)type;
  1920. types[offset / 4] = primType->mTypeDef->mTypeCode;
  1921. }
  1922. else if (type->IsSizedArray())
  1923. {
  1924. auto sizedArray = (BfSizedArrayType*)type;
  1925. for (int i = 0; i < sizedArray->mElementCount; i++)
  1926. _CheckType(sizedArray->mElementType, offset + i * sizedArray->mElementType->GetStride());
  1927. }
  1928. };
  1929. _CheckType(this, 0);
  1930. bool handled = false;
  1931. if (mModule->mCompiler->mOptions.mMachineType == BfMachineType_AArch64)
  1932. {
  1933. // For returns, we want to avoid sret but not actually lower
  1934. bool writeOutCode = (typeUsage != BfTypeUsage_Return_NonStatic) && (typeUsage != BfTypeUsage_Return_Static);
  1935. if ((types[0] == BfTypeCode_Float) &&
  1936. (types[1] == BfTypeCode_Float) &&
  1937. (types[2] == BfTypeCode_None))
  1938. {
  1939. if ((outTypeCode != NULL) && (writeOutCode))
  1940. *outTypeCode = BfTypeCode_FloatX2;
  1941. return true;
  1942. }
  1943. if ((types[0] == BfTypeCode_Float) &&
  1944. (types[1] == BfTypeCode_Float) &&
  1945. (types[2] == BfTypeCode_Float) &&
  1946. (types[3] == BfTypeCode_None))
  1947. {
  1948. if ((outTypeCode != NULL) && (writeOutCode))
  1949. *outTypeCode = BfTypeCode_FloatX3;
  1950. return true;
  1951. }
  1952. if ((types[0] == BfTypeCode_Float) &&
  1953. (types[1] == BfTypeCode_Float) &&
  1954. (types[2] == BfTypeCode_Float) &&
  1955. (types[3] == BfTypeCode_Float) &&
  1956. (types[4] == BfTypeCode_None))
  1957. {
  1958. if ((outTypeCode != NULL) && (writeOutCode))
  1959. *outTypeCode = BfTypeCode_FloatX4;
  1960. return true;
  1961. }
  1962. if ((types[0] == BfTypeCode_Double) &&
  1963. (types[2] == BfTypeCode_Double) &&
  1964. (types[4] == BfTypeCode_None))
  1965. {
  1966. if ((outTypeCode != NULL) && (writeOutCode))
  1967. *outTypeCode = BfTypeCode_DoubleX2;
  1968. return true;
  1969. }
  1970. if ((types[0] == BfTypeCode_Double) &&
  1971. (types[2] == BfTypeCode_Double) &&
  1972. (types[4] == BfTypeCode_Double) &&
  1973. (types[6] == BfTypeCode_None))
  1974. {
  1975. if ((outTypeCode != NULL) && (writeOutCode))
  1976. *outTypeCode = BfTypeCode_DoubleX3;
  1977. return true;
  1978. }
  1979. if ((types[0] == BfTypeCode_Double) &&
  1980. (types[2] == BfTypeCode_Double) &&
  1981. (types[4] == BfTypeCode_Double) &&
  1982. (types[6] == BfTypeCode_Double))
  1983. {
  1984. if ((outTypeCode != NULL) && (writeOutCode))
  1985. *outTypeCode = BfTypeCode_DoubleX4;
  1986. return true;
  1987. }
  1988. if (mInstSize <= 8)
  1989. {
  1990. if ((outTypeCode != NULL) && (writeOutCode))
  1991. *outTypeCode = BfTypeCode_Int64;
  1992. return true;
  1993. }
  1994. if (mInstSize <= 16)
  1995. {
  1996. if ((outTypeCode != NULL) && (writeOutCode))
  1997. *outTypeCode = BfTypeCode_Int64X2;
  1998. return true;
  1999. }
  2000. return false;
  2001. }
  2002. else if (mModule->mCompiler->mOptions.mPlatformType == BfPlatformType_Windows)
  2003. {
  2004. bool hasFloat = false;
  2005. for (int type = 0; type < 4; type++)
  2006. {
  2007. if ((types[type] == BfTypeCode_Float) ||
  2008. (types[type] == BfTypeCode_Double))
  2009. hasFloat = false;
  2010. }
  2011. if (!hasFloat)
  2012. {
  2013. if (mInstSize == 4)
  2014. {
  2015. if (outTypeCode != NULL)
  2016. *outTypeCode = BfTypeCode_Int32;
  2017. return true;
  2018. }
  2019. if (mInstSize == 8)
  2020. {
  2021. if (outTypeCode != NULL)
  2022. *outTypeCode = BfTypeCode_Int64;
  2023. return true;
  2024. }
  2025. }
  2026. }
  2027. else
  2028. {
  2029. if (mInstSize >= 8)
  2030. {
  2031. if (outTypeCode != NULL)
  2032. *outTypeCode = BfTypeCode_Int64;
  2033. }
  2034. if (mInstSize == 8)
  2035. {
  2036. handled = true;
  2037. }
  2038. if (mInstSize == 9)
  2039. {
  2040. handled = true;
  2041. if (outTypeCode2 != NULL)
  2042. *outTypeCode2 = BfTypeCode_Int8;
  2043. }
  2044. if (mInstSize == 10)
  2045. {
  2046. handled = true;
  2047. if (outTypeCode2 != NULL)
  2048. *outTypeCode2 = BfTypeCode_Int16;
  2049. }
  2050. if (mInstSize == 12)
  2051. {
  2052. handled = true;
  2053. if (outTypeCode2 != NULL)
  2054. *outTypeCode2 = BfTypeCode_Int32;
  2055. }
  2056. if (mInstSize == 16)
  2057. {
  2058. handled = true;
  2059. if (outTypeCode2 != NULL)
  2060. *outTypeCode2 = BfTypeCode_Int64;
  2061. }
  2062. if ((types[0] == BfTypeCode_Float) && (types[1] == BfTypeCode_None))
  2063. {
  2064. handled = true;
  2065. if (outTypeCode != NULL)
  2066. *outTypeCode = BfTypeCode_Float;
  2067. }
  2068. if ((types[0] == BfTypeCode_Float) && (types[1] == BfTypeCode_Float))
  2069. {
  2070. if (outTypeCode != NULL)
  2071. *outTypeCode = BfTypeCode_Float2;
  2072. }
  2073. if (types[0] == BfTypeCode_Double)
  2074. {
  2075. if (outTypeCode != NULL)
  2076. *outTypeCode = BfTypeCode_Double;
  2077. }
  2078. if ((types[2] == BfTypeCode_Float) && (mInstSize == 12))
  2079. {
  2080. if (outTypeCode2 != NULL)
  2081. *outTypeCode2 = BfTypeCode_Float;
  2082. }
  2083. if ((types[2] == BfTypeCode_Float) && (types[3] == BfTypeCode_Float))
  2084. {
  2085. if (outTypeCode2 != NULL)
  2086. *outTypeCode2 = BfTypeCode_Float2;
  2087. }
  2088. if (types[2] == BfTypeCode_Double)
  2089. {
  2090. if (outTypeCode2 != NULL)
  2091. *outTypeCode2 = BfTypeCode_Double;
  2092. }
  2093. if (handled)
  2094. return true;
  2095. }
  2096. }
  2097. }
  2098. BfTypeCode typeCode = BfTypeCode_None;
  2099. BfTypeCode pow2TypeCode = BfTypeCode_None;
  2100. switch (mInstSize)
  2101. {
  2102. case 1:
  2103. pow2TypeCode = BfTypeCode_Int8;
  2104. break;
  2105. case 2:
  2106. pow2TypeCode = BfTypeCode_Int16;
  2107. break;
  2108. case 3:
  2109. typeCode = BfTypeCode_Int24;
  2110. break;
  2111. case 4:
  2112. pow2TypeCode = BfTypeCode_Int32;
  2113. break;
  2114. case 5:
  2115. typeCode = BfTypeCode_Int40;
  2116. break;
  2117. case 6:
  2118. typeCode = BfTypeCode_Int48;
  2119. break;
  2120. case 7:
  2121. typeCode = BfTypeCode_Int56;
  2122. break;
  2123. case 8:
  2124. if (mModule->mSystem->mPtrSize == 8)
  2125. {
  2126. pow2TypeCode = BfTypeCode_Int64;
  2127. break;
  2128. }
  2129. if ((typeUsage == BfTypeUsage_Return_Static) && (mModule->mCompiler->mOptions.mPlatformType == BfPlatformType_Windows))
  2130. {
  2131. pow2TypeCode = BfTypeCode_Int64;
  2132. break;
  2133. }
  2134. break;
  2135. }
  2136. if (pow2TypeCode != BfTypeCode_None)
  2137. {
  2138. if (outTypeCode != NULL)
  2139. *outTypeCode = pow2TypeCode;
  2140. return true;
  2141. }
  2142. if ((mModule->mCompiler->mOptions.mPlatformType != BfPlatformType_Windows) && (mModule->mSystem->mPtrSize == 8))
  2143. {
  2144. if (typeCode != BfTypeCode_None)
  2145. {
  2146. if (outTypeCode != NULL)
  2147. *outTypeCode = typeCode;
  2148. return true;
  2149. }
  2150. }
  2151. return false;
  2152. }
  2153. bool BfTypeInstance::HasEquivalentLayout(BfTypeInstance* compareTo)
  2154. {
  2155. if (mFieldInstances.size() != compareTo->mFieldInstances.size())
  2156. return false;
  2157. for (int fieldIdx = 0; fieldIdx < (int)mFieldInstances.size(); fieldIdx++)
  2158. {
  2159. auto fieldInstance = &mFieldInstances[fieldIdx];
  2160. auto otherFieldInstance = &compareTo->mFieldInstances[fieldIdx];
  2161. if (fieldInstance->mResolvedType != otherFieldInstance->mResolvedType)
  2162. return false;
  2163. }
  2164. return true;
  2165. }
  2166. BfIRConstHolder* BfTypeInstance::GetOrCreateConstHolder()
  2167. {
  2168. if (mConstHolder == NULL)
  2169. mConstHolder = new BfIRConstHolder(mModule);
  2170. return mConstHolder;
  2171. }
  2172. BfIRValue BfTypeInstance::CreateConst(BfConstant* fromConst, BfIRConstHolder* fromHolder)
  2173. {
  2174. if (mConstHolder == NULL)
  2175. mConstHolder = new BfIRConstHolder(mModule);
  2176. return mConstHolder->CreateConst(fromConst, fromHolder);
  2177. }
  2178. bool BfTypeInstance::HasOverrideMethods()
  2179. {
  2180. if (mTypeDef->mHasOverrideMethods)
  2181. return true;
  2182. if (mBaseType != NULL)
  2183. return mBaseType->HasOverrideMethods();
  2184. return false;
  2185. }
  2186. bool BfTypeInstance::GetResultInfo(BfType*& valueType, int& okTagId)
  2187. {
  2188. BF_ASSERT(!IsDataIncomplete());
  2189. if (mFieldInstances.size() < 2)
  2190. return false;
  2191. for (auto& fieldInstance : mFieldInstances)
  2192. {
  2193. if (!fieldInstance.mIsEnumPayloadCase)
  2194. continue;
  2195. if ((fieldInstance.mIsEnumPayloadCase) && (fieldInstance.GetFieldDef()->mName == "Ok") && (fieldInstance.mResolvedType->IsTuple()))
  2196. {
  2197. auto tupleType = (BfTypeInstance*)fieldInstance.mResolvedType;
  2198. if (tupleType->mFieldInstances.size() == 1)
  2199. {
  2200. valueType = tupleType->mFieldInstances[0].mResolvedType;
  2201. okTagId = -fieldInstance.mDataIdx - 1;
  2202. return true;
  2203. }
  2204. }
  2205. break;
  2206. }
  2207. return false;
  2208. }
  2209. bool BfTypeInstance::IsAnonymous()
  2210. {
  2211. return (mTypeDef->mTypeDeclaration != NULL) && (mTypeDef->mTypeDeclaration->IsAnonymous());
  2212. }
  2213. bool BfTypeInstance::IsAnonymousInitializerType()
  2214. {
  2215. return (mTypeDef->mTypeDeclaration != NULL) && (mTypeDef->mTypeDeclaration->IsAnonymousInitializerType());
  2216. }
  2217. bool BfTypeInstance::HasAppendCtor()
  2218. {
  2219. return mTypeDef->mHasAppendCtor;
  2220. }
  2221. bool BfTypeInstance::BaseHasAppendCtor()
  2222. {
  2223. if (mBaseType != NULL)
  2224. {
  2225. if (mBaseType->HasAppendCtor())
  2226. return true;
  2227. return mBaseType->BaseHasAppendCtor();
  2228. }
  2229. return false;
  2230. }
  2231. bool BfTypeInstance::HasAppendedField(bool checkBase)
  2232. {
  2233. for (auto& fieldInstance : mFieldInstances)
  2234. {
  2235. if (fieldInstance.IsAppendedObject())
  2236. return true;
  2237. }
  2238. if ((checkBase) && (mBaseType != NULL))
  2239. return mBaseType->HasAppendedField(checkBase);
  2240. return false;
  2241. }
  2242. void BfTypeInstance::ReportMemory(MemReporter* memReporter)
  2243. {
  2244. if (mGenericTypeInfo != NULL)
  2245. mGenericTypeInfo->ReportMemory(memReporter);
  2246. memReporter->Add(sizeof(BfTypeInstance));
  2247. int depSize = 0;
  2248. depSize += sizeof((int)mDependencyMap.mTypeSet.mAllocSize * sizeof(BfDependencyMap::TypeMap::EntryPair));
  2249. memReporter->Add("DepMap", depSize);
  2250. memReporter->AddVec(mInterfaces, false);
  2251. memReporter->AddVec(mInterfaceMethodTable, false);
  2252. if (mCustomAttributes != NULL)
  2253. mCustomAttributes->ReportMemory(memReporter);
  2254. int methodCount = 0;
  2255. memReporter->BeginSection("MethodData");
  2256. for (auto& methodInstGroup : mMethodInstanceGroups)
  2257. {
  2258. memReporter->Add(sizeof(BfMethodInstanceGroup));
  2259. if (methodInstGroup.mDefault != NULL)
  2260. {
  2261. methodInstGroup.mDefault->ReportMemory(memReporter);
  2262. methodCount++;
  2263. }
  2264. if (methodInstGroup.mMethodSpecializationMap != NULL)
  2265. {
  2266. memReporter->Add((int)methodInstGroup.mMethodSpecializationMap->mAllocSize * sizeof(Dictionary<BfTypeVector, BfMethodInstance*>::EntryPair));
  2267. for (auto kv : *methodInstGroup.mMethodSpecializationMap)
  2268. {
  2269. methodCount++;
  2270. kv.mValue->ReportMemory(memReporter);
  2271. }
  2272. }
  2273. }
  2274. memReporter->EndSection();
  2275. memReporter->AddVec("VirtualMethodTable", mVirtualMethodTable, false);
  2276. memReporter->AddVec(mFieldInstances, false);
  2277. memReporter->AddVec(mInternalMethods, false);
  2278. memReporter->AddMap("SpecializedMethodReferences", mSpecializedMethodReferences, false);
  2279. memReporter->AddMap("LookupResults", mLookupResults, false);
  2280. if (mConstHolder != NULL)
  2281. memReporter->Add("ConstHolder", mConstHolder->mTempAlloc.GetTotalAllocSize());
  2282. if (mHotTypeData != NULL)
  2283. {
  2284. AutoMemReporter autoMemReporter(memReporter, "HotTypeData");
  2285. memReporter->Add(sizeof(BfHotTypeData));
  2286. memReporter->AddVec(mHotTypeData->mTypeVersions, false);
  2287. for (auto typeVersion : mHotTypeData->mTypeVersions)
  2288. {
  2289. memReporter->AddVec(typeVersion->mMembers, false);
  2290. memReporter->AddVec(typeVersion->mInterfaceMapping, false);
  2291. }
  2292. memReporter->AddVec(mHotTypeData->mVTableEntries, false);
  2293. for (auto& entry : mHotTypeData->mVTableEntries)
  2294. memReporter->AddStr(entry.mFuncName, false);
  2295. }
  2296. BfLog("%s\t%d\t%d\n", mContext->mScratchModule->TypeToString(this, BfTypeNameFlags_None).c_str(), IsGenericTypeInstance(), methodCount);
  2297. }
  2298. bool BfTypeInstance::IsTypeMemberAccessible(BfTypeDef* declaringTypeDef, BfTypeDef* activeTypeDef)
  2299. {
  2300. if (activeTypeDef == NULL)
  2301. return false;
  2302. if (declaringTypeDef == activeTypeDef)
  2303. return true;
  2304. return activeTypeDef->mProject->ContainsReference(declaringTypeDef->mProject);
  2305. }
  2306. bool BfTypeInstance::IsTypeMemberAccessible(BfTypeDef* declaringTypeDef, BfProject* curProject)
  2307. {
  2308. if (declaringTypeDef->mProject == curProject)
  2309. return true;
  2310. return curProject->ContainsReference(declaringTypeDef->mProject);
  2311. }
  2312. bool BfTypeInstance::IsTypeMemberAccessible(BfTypeDef* declaringTypeDef, BfProjectSet* visibleProjectSet)
  2313. {
  2314. if (visibleProjectSet == NULL)
  2315. return false;
  2316. return visibleProjectSet->Contains(declaringTypeDef->mProject);
  2317. }
  2318. bool BfTypeInstance::WantsGCMarking()
  2319. {
  2320. BF_ASSERT(mTypeDef->mTypeCode != BfTypeCode_Extension);
  2321. if (IsObjectOrInterface())
  2322. return true;
  2323. if ((IsEnum()) && (!IsPayloadEnum()))
  2324. return false;
  2325. BF_ASSERT((mDefineState >= BfTypeDefineState_Defined) || (mTypeFailed));
  2326. return mWantsGCMarking;
  2327. }
  2328. ///
  2329. BfGenericExtensionEntry::~BfGenericExtensionEntry()
  2330. {
  2331. for (auto genericParamInstance : mGenericParams)
  2332. genericParamInstance->Release();
  2333. }
  2334. ///
  2335. BfGenericTypeInfo::~BfGenericTypeInfo()
  2336. {
  2337. for (auto genericParamInstance : mGenericParams)
  2338. genericParamInstance->Release();
  2339. delete mGenericExtensionInfo;
  2340. }
  2341. BfGenericTypeInfo::GenericParamsVector* BfTypeInstance::GetGenericParamsVector(BfTypeDef* declaringTypeDef)
  2342. {
  2343. if (mGenericTypeInfo == NULL)
  2344. return NULL;
  2345. if ((declaringTypeDef == mTypeDef) ||
  2346. (declaringTypeDef->mTypeDeclaration == mTypeDef->mTypeDeclaration))
  2347. return &mGenericTypeInfo->mGenericParams;
  2348. if (mGenericTypeInfo->mGenericExtensionInfo == NULL)
  2349. return NULL;
  2350. BfGenericExtensionEntry* genericExEntry = NULL;
  2351. if (mGenericTypeInfo->mGenericExtensionInfo->mExtensionMap.TryGetValue(declaringTypeDef, &genericExEntry))
  2352. return &genericExEntry->mGenericParams;
  2353. return &mGenericTypeInfo->mGenericParams;
  2354. }
  2355. void BfTypeInstance::GenerateProjectsReferenced()
  2356. {
  2357. if (mGenericTypeInfo == NULL)
  2358. return;
  2359. BF_ASSERT(mGenericTypeInfo->mProjectsReferenced.empty());
  2360. mGenericTypeInfo->mProjectsReferenced.push_back(mTypeDef->mProject);
  2361. for (auto genericArgType : mGenericTypeInfo->mTypeGenericArguments)
  2362. BfTypeUtils::GetProjectList(genericArgType, &mGenericTypeInfo->mProjectsReferenced, 0);
  2363. }
  2364. bool BfTypeInstance::IsAlwaysInclude()
  2365. {
  2366. bool alwaysInclude = mTypeDef->mIsAlwaysInclude || mTypeDef->mProject->mAlwaysIncludeAll;
  2367. if (mTypeOptionsIdx > 0)
  2368. {
  2369. auto typeOptions = mModule->mSystem->GetTypeOptions(mTypeOptionsIdx);
  2370. typeOptions->Apply(alwaysInclude, BfOptionFlags_ReflectAlwaysIncludeType);
  2371. }
  2372. if ((mAlwaysIncludeFlags & BfAlwaysIncludeFlag_Type) != 0)
  2373. alwaysInclude = true;
  2374. return alwaysInclude;
  2375. }
  2376. bool BfTypeInstance::IsSpecializedByAutoCompleteMethod()
  2377. {
  2378. if (mGenericTypeInfo == NULL)
  2379. return false;
  2380. for (auto methodArg : mGenericTypeInfo->mTypeGenericArguments)
  2381. {
  2382. // If we are specialized by an autocompleted method reference
  2383. if (methodArg->IsMethodRef())
  2384. {
  2385. auto methodRefType = (BfMethodRefType*)methodArg;
  2386. if (methodRefType->mIsAutoCompleteMethod)
  2387. return true;
  2388. }
  2389. }
  2390. return false;
  2391. }
  2392. bool BfTypeInstance::IsNullable()
  2393. {
  2394. return IsInstanceOf(mContext->mCompiler->mNullableTypeDef);
  2395. }
  2396. bool BfTypeInstance::HasVarConstraints()
  2397. {
  2398. if (mGenericTypeInfo == NULL)
  2399. return false;
  2400. for (auto genericParam : mGenericTypeInfo->mGenericParams)
  2401. {
  2402. if (genericParam->mGenericParamFlags & BfGenericParamFlag_Var)
  2403. return true;
  2404. }
  2405. return false;
  2406. }
  2407. bool BfTypeInstance::IsTypeMemberIncluded(BfTypeDef* typeDef, BfTypeDef* activeTypeDef, BfModule* module)
  2408. {
  2409. if (mGenericTypeInfo == NULL)
  2410. return true;
  2411. if (mGenericTypeInfo->mGenericExtensionInfo == NULL)
  2412. return true;
  2413. if ((typeDef == NULL) || (typeDef == activeTypeDef))
  2414. return true;
  2415. // The combined type declaration is the root type declaration, it's implicitly included
  2416. if (typeDef->mTypeDeclaration == mTypeDef->mTypeDeclaration)
  2417. return true;
  2418. BfGenericExtensionEntry* genericExEntry = NULL;
  2419. if (!mGenericTypeInfo->mGenericExtensionInfo->mExtensionMap.TryGetValue(typeDef, &genericExEntry))
  2420. return true;
  2421. if (mGenericTypeInfo->mIsUnspecialized)
  2422. {
  2423. if (module == NULL)
  2424. return true; // During population
  2425. auto declConstraints = &genericExEntry->mGenericParams;
  2426. for (int genericIdx = 0; genericIdx < (int)declConstraints->size(); genericIdx++)
  2427. {
  2428. auto declGenericParam = (*declConstraints)[genericIdx];
  2429. BfType* genericArg;
  2430. if (genericIdx < (int)mGenericTypeInfo->mTypeGenericArguments.size())
  2431. {
  2432. genericArg = mGenericTypeInfo->mTypeGenericArguments[genericIdx];
  2433. }
  2434. else
  2435. {
  2436. genericArg = declGenericParam->mExternType;
  2437. }
  2438. if ((genericArg == NULL) || (!module->CheckGenericConstraints(BfGenericParamSource(), genericArg, NULL, declGenericParam)))
  2439. return false;
  2440. }
  2441. return true;
  2442. }
  2443. return genericExEntry->mConstraintsPassed;
  2444. }
  2445. bool BfTypeInstance::IsZeroGap()
  2446. {
  2447. BF_ASSERT(mDefineState >= BfTypeDefineState_Defined);
  2448. for (int fieldIdx = mFieldInstances.mSize - 1; fieldIdx >= 0; fieldIdx--)
  2449. {
  2450. auto fieldInstance = &mFieldInstances[fieldIdx];
  2451. auto fieldDef = fieldInstance->GetFieldDef();
  2452. if (fieldDef == NULL)
  2453. continue;
  2454. if ((!fieldDef->mIsStatic) && (fieldDef->mDeclaringType->IsExtension()) && (!fieldInstance->mResolvedType->IsValuelessType()))
  2455. return false;
  2456. }
  2457. return true;
  2458. }
  2459. void BfGenericTypeInfo::ReportMemory(MemReporter* memReporter)
  2460. {
  2461. memReporter->Add(sizeof(BfGenericTypeInfo));
  2462. memReporter->AddVec(mTypeGenericArguments, false);
  2463. memReporter->AddVec(mGenericParams, false);
  2464. memReporter->AddVec(mProjectsReferenced, false);
  2465. }
  2466. BfType* BfTypeInstance::GetUnderlyingType()
  2467. {
  2468. if (!mIsTypedPrimitive)
  2469. {
  2470. if ((mGenericTypeInfo != NULL) && (!mGenericTypeInfo->mTypeGenericArguments.IsEmpty()))
  2471. return mGenericTypeInfo->mTypeGenericArguments[0];
  2472. return NULL;
  2473. }
  2474. if (mTypeInfoEx == NULL)
  2475. mTypeInfoEx = new BfTypeInfoEx();
  2476. if (mTypeInfoEx->mUnderlyingType != NULL)
  2477. return mTypeInfoEx->mUnderlyingType;
  2478. auto checkTypeInst = this;
  2479. while (checkTypeInst != NULL)
  2480. {
  2481. if (!checkTypeInst->mFieldInstances.empty())
  2482. {
  2483. auto& lastField = checkTypeInst->mFieldInstances.back();
  2484. if (lastField.mFieldIdx == -1)
  2485. {
  2486. auto underlyingType = lastField.mResolvedType;
  2487. BF_ASSERT(underlyingType != this);
  2488. mTypeInfoEx->mUnderlyingType = underlyingType;
  2489. return mTypeInfoEx->mUnderlyingType;
  2490. }
  2491. }
  2492. checkTypeInst = checkTypeInst->mBaseType;
  2493. if ((checkTypeInst != NULL) && (checkTypeInst->IsIncomplete()))
  2494. mModule->PopulateType(checkTypeInst, BfPopulateType_Data);
  2495. }
  2496. //BF_FATAL("Failed");
  2497. return mTypeInfoEx->mUnderlyingType;
  2498. }
  2499. bool BfTypeInstance::IsValuelessType()
  2500. {
  2501. BF_ASSERT(mTypeDef->mTypeCode != BfTypeCode_Extension);
  2502. if ((mTypeDef->mTypeCode == BfTypeCode_Object) || (mTypeDef->mTypeCode == BfTypeCode_Interface))
  2503. {
  2504. return false;
  2505. }
  2506. if (mTypeDef->mIsOpaque)
  2507. {
  2508. return true;
  2509. }
  2510. BF_ASSERT((mDefineState >= BfTypeDefineState_Defined) || (mTypeFailed));
  2511. BF_ASSERT(mInstSize >= 0);
  2512. if (mInstSize == 0)
  2513. {
  2514. return true;
  2515. }
  2516. return false;
  2517. }
  2518. bool BfTypeInstance::IsIRFuncUsed(BfIRFunction func)
  2519. {
  2520. for (auto& group : mMethodInstanceGroups)
  2521. {
  2522. if (group.mDefault != NULL)
  2523. if (group.mDefault->mIRFunction == func)
  2524. return true;
  2525. if (group.mMethodSpecializationMap != NULL)
  2526. {
  2527. for (auto& methodInstPair : *group.mMethodSpecializationMap)
  2528. {
  2529. auto methodInstance = methodInstPair.mValue;
  2530. if (methodInstance->mIRFunction == func)
  2531. return true;
  2532. }
  2533. }
  2534. }
  2535. return false;
  2536. }
  2537. void BfTypeInstance::CalcHotVirtualData(Array<int>* ifaceMapping)
  2538. {
  2539. if (IsIncomplete())
  2540. {
  2541. BF_ASSERT(mHotTypeData != NULL);
  2542. return;
  2543. }
  2544. if (ifaceMapping != NULL)
  2545. {
  2546. for (auto iface : mInterfaces)
  2547. {
  2548. int slotNum = iface.mInterfaceType->mSlotNum;
  2549. if (slotNum >= 0)
  2550. {
  2551. if (slotNum >= (int)ifaceMapping->size())
  2552. ifaceMapping->Resize(slotNum + 1);
  2553. (*ifaceMapping)[slotNum] = iface.mInterfaceType->mTypeId;
  2554. }
  2555. }
  2556. if (mBaseType != NULL)
  2557. mBaseType->CalcHotVirtualData(ifaceMapping);
  2558. }
  2559. }
  2560. //////////////////////////////////////////////////////////////////////////
  2561. BfClosureType::BfClosureType(BfTypeInstance* srcDelegate, Val128 closureHash) :
  2562. mSource(srcDelegate->mTypeDef->mSystem)
  2563. {
  2564. BF_ASSERT(srcDelegate->IsDelegate());
  2565. mSrcDelegate = srcDelegate;
  2566. mTypeDef = mSrcDelegate->mTypeDef;
  2567. mCreatedTypeDef = false;
  2568. mClosureHash = closureHash;
  2569. // Hash in 72 bits of closureHash (12 characters) - low does 60 bits, high does 12 bits
  2570. mNameAdd = "_" + BfTypeUtils::HashEncode64(mClosureHash.mLow) + BfTypeUtils::HashEncode64(mClosureHash.mHigh >> 52);
  2571. mIsUnique = false;
  2572. }
  2573. BfClosureType::~BfClosureType()
  2574. {
  2575. mMethodInstanceGroups.Clear();
  2576. if (mCreatedTypeDef)
  2577. {
  2578. delete mTypeDef;
  2579. mTypeDef = NULL;
  2580. }
  2581. for (auto directAllocNode : mDirectAllocNodes)
  2582. delete directAllocNode;
  2583. }
  2584. void BfClosureType::Init(BfProject* bfProject)
  2585. {
  2586. auto srcTypeDef = mSrcDelegate->mTypeDef;
  2587. auto system = mSrcDelegate->mModule->mSystem;
  2588. mTypeDef = new BfTypeDef();
  2589. mTypeDef->mSystem = system;
  2590. mTypeDef->mSource = &mSource;
  2591. mTypeDef->mSource->mRefCount++;
  2592. mTypeDef->mProject = bfProject;
  2593. mTypeDef->mTypeCode = srcTypeDef->mTypeCode;
  2594. mTypeDef->mName = system->GetAtom(srcTypeDef->mName->mString + mNameAdd);
  2595. // Purposely leave out 'mOuterType' - this fails if the outer type is generic
  2596. //mTypeDef->mOuterType = srcTypeDef->mOuterType;
  2597. mTypeDef->mNamespace = srcTypeDef->mNamespace;
  2598. system->AddNamespaceUsage(mTypeDef->mNamespace, mTypeDef->mProject);
  2599. mTypeDef->mHash = srcTypeDef->mHash;
  2600. mTypeDef->mSignatureHash = srcTypeDef->mSignatureHash;
  2601. // mTypeDef->mFullName = srcTypeDef->mFullName;
  2602. // if (!mTypeDef->mFullName.mParts.IsEmpty())
  2603. // mTypeDef->mFullName.mParts.pop_back();
  2604. // mTypeDef->mFullName.mParts.push_back(mTypeDef->mName);
  2605. if (srcTypeDef->mFullName.mSize > 0)
  2606. mTypeDef->mFullName.Set(srcTypeDef->mFullName.mParts, srcTypeDef->mFullName.mSize - 1, &mTypeDef->mName, 1);
  2607. else
  2608. mTypeDef->mFullName.Set(&mTypeDef->mName, 1, NULL, 0);
  2609. system->TrackName(mTypeDef);
  2610. mTypeDef->mTypeCode = BfTypeCode_Object;
  2611. mTypeDef->mIsDelegate = true;
  2612. mTypeDef->mIsClosure = true;
  2613. mTypeDef->mDefState = BfTypeDef::DefState_Defined;
  2614. auto baseDirectTypeRef = BfAstNode::ZeroedAlloc<BfDirectTypeReference>();
  2615. baseDirectTypeRef->Init(mSrcDelegate);
  2616. mDirectAllocNodes.push_back(baseDirectTypeRef);
  2617. mTypeDef->mBaseTypes.push_back(baseDirectTypeRef);
  2618. //mTypeDef->mBaseTypes.push_back(BfDefBuilder::AllocTypeReference(&mSource, mSrcDelegate));
  2619. mCreatedTypeDef = true;
  2620. //mTypeDef->mBaseTypes.push_back(srcTypeDef);
  2621. }
  2622. BfFieldDef* BfClosureType::AddField(BfType* type, const StringImpl& name)
  2623. {
  2624. auto directTypeRef = BfAstNode::ZeroedAlloc<BfDirectTypeReference>();
  2625. directTypeRef->Init(type);
  2626. mDirectAllocNodes.push_back(directTypeRef);
  2627. return BfDefBuilder::AddField(mTypeDef, directTypeRef, name);
  2628. }
  2629. BfMethodDef* BfClosureType::AddDtor()
  2630. {
  2631. return BfDefBuilder::AddDtor(mTypeDef);
  2632. }
  2633. void BfClosureType::Finish()
  2634. {
  2635. HASH128_MIXIN(mTypeDef->mSignatureHash, mClosureHash);
  2636. auto bfSource = mTypeDef->mSource;
  2637. auto bfSystem = bfSource->mSystem;
  2638. BfDefBuilder bfDefBuilder(bfSystem);
  2639. bfDefBuilder.mCurTypeDef = mTypeDef;
  2640. bfDefBuilder.FinishTypeDef(false);
  2641. }
  2642. //////////////////////////////////////////////////////////////////////////
  2643. BfDelegateType::~BfDelegateType()
  2644. {
  2645. mMethodInstanceGroups.Clear();
  2646. delete mTypeDef;
  2647. mTypeDef = NULL;
  2648. }
  2649. void BfDelegateType::Dispose()
  2650. {
  2651. delete mTypeDef;
  2652. mTypeDef = NULL;
  2653. BfTypeInstance::Dispose();
  2654. }
  2655. //////////////////////////////////////////////////////////////////////////
  2656. BfTupleType::BfTupleType()
  2657. {
  2658. mCreatedTypeDef = false;
  2659. mSource = NULL;
  2660. mTypeDef = NULL;
  2661. mIsUnspecializedType = false;
  2662. mIsUnspecializedTypeVariation = false;
  2663. mGenericDepth = 0;
  2664. }
  2665. BfTupleType::~BfTupleType()
  2666. {
  2667. mMethodInstanceGroups.Clear();
  2668. if (mCreatedTypeDef)
  2669. {
  2670. delete mTypeDef;
  2671. mTypeDef = NULL;
  2672. }
  2673. delete mSource;
  2674. }
  2675. void BfTupleType::Init(BfProject* bfProject, BfTypeInstance* valueTypeInstance)
  2676. {
  2677. auto srcTypeDef = valueTypeInstance->mTypeDef;
  2678. auto system = valueTypeInstance->mModule->mSystem;
  2679. if (mTypeDef == NULL)
  2680. mTypeDef = new BfTypeDef();
  2681. for (auto field : mTypeDef->mFields)
  2682. delete field;
  2683. mTypeDef->mFields.Clear();
  2684. mTypeDef->mSystem = system;
  2685. mTypeDef->mProject = bfProject;
  2686. mTypeDef->mTypeCode = srcTypeDef->mTypeCode;
  2687. mTypeDef->mName = system->mEmptyAtom;
  2688. mTypeDef->mSystem = system;
  2689. mTypeDef->mHash = srcTypeDef->mHash;
  2690. mTypeDef->mSignatureHash = srcTypeDef->mSignatureHash;
  2691. mTypeDef->mTypeCode = BfTypeCode_Struct;
  2692. mCreatedTypeDef = true;
  2693. }
  2694. void BfTupleType::Dispose()
  2695. {
  2696. if (mCreatedTypeDef)
  2697. {
  2698. delete mTypeDef;
  2699. mTypeDef = NULL;
  2700. mCreatedTypeDef = false;
  2701. }
  2702. BfTypeInstance::Dispose();
  2703. }
  2704. BfFieldDef* BfTupleType::AddField(const StringImpl& name)
  2705. {
  2706. return BfDefBuilder::AddField(mTypeDef, NULL, name);
  2707. }
  2708. void BfTupleType::Finish()
  2709. {
  2710. BF_ASSERT(!mTypeFailed);
  2711. auto bfSystem = mTypeDef->mSystem;
  2712. mSource = new BfSource(bfSystem);
  2713. mTypeDef->mSource = mSource;
  2714. mTypeDef->mSource->mRefCount++;
  2715. BfDefBuilder bfDefBuilder(bfSystem);
  2716. bfDefBuilder.mCurTypeDef = mTypeDef;
  2717. bfDefBuilder.mCurDeclaringTypeDef = mTypeDef;
  2718. bfDefBuilder.FinishTypeDef(true);
  2719. }
  2720. //////////////////////////////////////////////////////////////////////////
  2721. BfTagType::BfTagType()
  2722. {
  2723. mCreatedTypeDef = false;
  2724. mSource = NULL;
  2725. mTypeDef = NULL;
  2726. }
  2727. BfTagType::~BfTagType()
  2728. {
  2729. mMethodInstanceGroups.Clear();
  2730. if (mCreatedTypeDef)
  2731. {
  2732. delete mTypeDef;
  2733. mTypeDef = NULL;
  2734. }
  2735. delete mSource;
  2736. }
  2737. void BfTagType::Init(BfProject* bfProject, BfTypeInstance* valueTypeInstance, const StringImpl& name)
  2738. {
  2739. auto srcTypeDef = valueTypeInstance->mTypeDef;
  2740. auto system = valueTypeInstance->mModule->mSystem;
  2741. if (mTypeDef == NULL)
  2742. mTypeDef = new BfTypeDef();
  2743. for (auto field : mTypeDef->mFields)
  2744. delete field;
  2745. mTypeDef->mFields.Clear();
  2746. mTypeDef->mSystem = system;
  2747. mTypeDef->mProject = bfProject;
  2748. mTypeDef->mTypeCode = srcTypeDef->mTypeCode;
  2749. mTypeDef->mName = system->mEmptyAtom;
  2750. mTypeDef->mSystem = system;
  2751. mTypeDef->mHash = srcTypeDef->mHash;
  2752. mTypeDef->mSignatureHash = srcTypeDef->mSignatureHash;
  2753. mTypeDef->mTypeCode = BfTypeCode_Enum;
  2754. mCreatedTypeDef = true;
  2755. auto field = BfDefBuilder::AddField(mTypeDef, NULL, name);
  2756. field->mIsStatic = true;
  2757. field->mIsConst = true;
  2758. }
  2759. void BfTagType::Dispose()
  2760. {
  2761. if (mCreatedTypeDef)
  2762. {
  2763. delete mTypeDef;
  2764. mTypeDef = NULL;
  2765. mCreatedTypeDef = false;
  2766. }
  2767. BfTypeInstance::Dispose();
  2768. }
  2769. void BfTagType::Finish()
  2770. {
  2771. BF_ASSERT(!mTypeFailed);
  2772. auto bfSystem = mTypeDef->mSystem;
  2773. mSource = new BfSource(bfSystem);
  2774. mTypeDef->mSource = mSource;
  2775. mTypeDef->mSource->mRefCount++;
  2776. BfDefBuilder bfDefBuilder(bfSystem);
  2777. bfDefBuilder.mCurTypeDef = mTypeDef;
  2778. bfDefBuilder.mCurDeclaringTypeDef = mTypeDef;
  2779. bfDefBuilder.FinishTypeDef(true);
  2780. }
  2781. //////////////////////////////////////////////////////////////////////////
  2782. BfBoxedType::~BfBoxedType()
  2783. {
  2784. //if ((mTypeDef != NULL) && (mTypeDef->mEmitParent != NULL))
  2785. mTypeDef = NULL;
  2786. }
  2787. BfType* BfBoxedType::GetModifiedElementType()
  2788. {
  2789. if ((mBoxedFlags & BoxedFlags_StructPtr) != 0)
  2790. {
  2791. auto module = mModule;
  2792. if (module == NULL)
  2793. module = mContext->mUnreifiedModule;
  2794. return module->CreatePointerType(mElementType);
  2795. }
  2796. return mElementType;
  2797. }
  2798. //////////////////////////////////////////////////////////////////////////
  2799. int BfArrayType::GetLengthBitCount()
  2800. {
  2801. if (mBaseType == NULL)
  2802. mModule->PopulateType(mBaseType, BfPopulateType_BaseType);
  2803. mModule->PopulateType(mBaseType);
  2804. if ((mBaseType->mFieldInstances.size() == 0) || (mBaseType->mFieldInstances[0].GetFieldDef()->mName != "mLength"))
  2805. {
  2806. return 0;
  2807. }
  2808. return mBaseType->mFieldInstances[0].mResolvedType->mSize * 8;
  2809. }
  2810. //////////////////////////////////////////////////////////////////////////
  2811. int BfMethodRefType::GetCaptureDataCount()
  2812. {
  2813. if (mMethodRef == NULL)
  2814. return 0;
  2815. return (int)mDataToParamIdx.size();
  2816. }
  2817. BfType* BfMethodRefType::GetCaptureType(int captureDataIdx)
  2818. {
  2819. return mMethodRef->GetParamType(mDataToParamIdx[captureDataIdx]);
  2820. }
  2821. int BfMethodRefType::GetDataIdxFromParamIdx(int paramIdx)
  2822. {
  2823. if (paramIdx == -1)
  2824. {
  2825. if (mMethodRef->HasThis())
  2826. return 0;
  2827. return -1;
  2828. }
  2829. return mParamToDataIdx[paramIdx];
  2830. }
  2831. int BfMethodRefType::GetParamIdxFromDataIdx(int dataIdx)
  2832. {
  2833. return mDataToParamIdx[dataIdx];
  2834. }
  2835. bool BfMethodRefType::WantsDataPassedAsSplat(int dataIdx)
  2836. {
  2837. if (dataIdx != -1)
  2838. return false;
  2839. return mMethodRef->GetParamIsSplat(mDataToParamIdx[dataIdx]);
  2840. }
  2841. //////////////////////////////////////////////////////////////////////////
  2842. size_t BfTypeVectorHash::operator()(const BfTypeVector& typeVec) const
  2843. {
  2844. size_t hash = typeVec.size();
  2845. BfResolvedTypeSet::LookupContext ctx;
  2846. for (auto type : typeVec)
  2847. hash = ((hash ^ BfResolvedTypeSet::Hash(type, &ctx, Beefy::BfResolvedTypeSet::BfHashFlag_None, 0)) << 5) - hash;
  2848. return hash;
  2849. }
  2850. bool BfTypeVectorEquals::operator()(const BfTypeVector& lhs, const BfTypeVector& rhs) const
  2851. {
  2852. if (lhs.size() != rhs.size())
  2853. return false;
  2854. for (int i = 0; i < (int)lhs.size(); i++)
  2855. if (lhs[i] != rhs[i])
  2856. return false;
  2857. return true;
  2858. }
  2859. //////////////////////////////////////////////////////////////////////////
  2860. bool BfCustomAttributes::Contains(BfTypeDef* typeDef)
  2861. {
  2862. for (auto& customAttr : mAttributes)
  2863. if (customAttr.mType->mTypeDef->GetDefinition() == typeDef)
  2864. return true;
  2865. return false;
  2866. }
  2867. BfCustomAttribute* BfCustomAttributes::Get(BfTypeDef * typeDef)
  2868. {
  2869. for (auto& customAttr : mAttributes)
  2870. if (customAttr.mType->mTypeDef->GetDefinition() == typeDef)
  2871. return &customAttr;
  2872. return NULL;
  2873. }
  2874. BfCustomAttribute* BfCustomAttributes::Get(BfType* type)
  2875. {
  2876. for (auto& customAttr : mAttributes)
  2877. if (customAttr.mType == type)
  2878. return &customAttr;
  2879. return NULL;
  2880. }
  2881. BfCustomAttribute* BfCustomAttributes::Get(int idx)
  2882. {
  2883. if (idx >= mAttributes.size())
  2884. return NULL;
  2885. return &mAttributes[idx];
  2886. }
  2887. //////////////////////////////////////////////////////////////////////////
  2888. BfResolvedTypeSet::~BfResolvedTypeSet()
  2889. {
  2890. }
  2891. #define HASH_MIX(origHashVal, newHashVal) ((((origHashVal) << 5) - (origHashVal)) ^ (newHashVal))
  2892. #define HASH_VAL_PTR 1
  2893. #define HASH_VAL_BOXED 2
  2894. #define HASH_VAL_REF 3
  2895. #define HASH_VAL_OUT 4
  2896. #define HASH_VAL_MUT 5
  2897. #define HASH_MODTYPE 6
  2898. #define HASH_CONCRETE_INTERFACE 7
  2899. #define HASH_SIZED_ARRAY 8
  2900. #define HASH_CONSTTYPE 9
  2901. #define HASH_VAL_TUPLE 10
  2902. #define HASH_DELEGATE 11
  2903. #define HASH_CONSTEXPR 12
  2904. #define HASH_GLOBAL 13
  2905. #define HASH_DOTDOTDOT 14
  2906. #define HASH_TAG 15
  2907. BfVariant BfResolvedTypeSet::EvaluateToVariant(LookupContext* ctx, BfExpression* expr, BfType*& outType)
  2908. {
  2909. outType = NULL;
  2910. BfMethodState methodState;
  2911. methodState.mTempKind = BfMethodState::TempKind_Static;
  2912. SetAndRestoreValue<BfMethodState*> prevMethodState;
  2913. if (ctx->mModule->mCurMethodState == NULL)
  2914. prevMethodState.Init(ctx->mModule->mCurMethodState, &methodState);
  2915. BfConstResolver constResolver(ctx->mModule);
  2916. BfVariant variant;
  2917. constResolver.mBfEvalExprFlags = BfEvalExprFlags_NoCast;
  2918. constResolver.mBfEvalExprFlags = (BfEvalExprFlags)(constResolver.mBfEvalExprFlags | BfEvalExprFlags_AllowGenericConstValue);
  2919. constResolver.mExpectingType = ctx->mModule->GetPrimitiveType(BfTypeCode_Int64);
  2920. auto result = constResolver.Resolve(expr);
  2921. if (result)
  2922. {
  2923. // Limit the types of constants to prevent duplicate values with different types - we don't want to hash a typeref with an int32
  2924. // when the constraint requirement is int64 (but we don't know that at hash time)
  2925. if ((result.mType->IsInteger()) && ((result.mType->mSize < 8) || (result.mType->GetTypeCode() == BfTypeCode_IntPtr)))
  2926. result = ctx->mModule->Cast(expr, result, ctx->mModule->GetPrimitiveType(BfTypeCode_Int64));
  2927. else if ((result.mType->IsFloat()) && (result.mType->mSize < 8))
  2928. result = ctx->mModule->Cast(expr, result, ctx->mModule->GetPrimitiveType(BfTypeCode_Double));
  2929. else if ((result.mType->IsChar()) && (result.mType->mSize < 4))
  2930. result = ctx->mModule->Cast(expr, result, ctx->mModule->GetPrimitiveType(BfTypeCode_Char32));
  2931. outType = result.mType;
  2932. if (result.mKind == BfTypedValueKind_GenericConstValue)
  2933. {
  2934. return variant;
  2935. }
  2936. else
  2937. {
  2938. variant = ctx->mModule->TypedValueToVariant(expr, result, true);
  2939. }
  2940. }
  2941. return variant;
  2942. }
  2943. int BfResolvedTypeSet::DoHash(BfType* type, LookupContext* ctx, bool allowRef, int hashSeed)
  2944. {
  2945. //BP_ZONE("BfResolvedTypeSet::Hash");
  2946. // if (type->IsTypeAlias())
  2947. // {
  2948. // auto underlyingType = type->GetUnderlyingType();
  2949. // BF_ASSERT(underlyingType != NULL);
  2950. // if (underlyingType == NULL)
  2951. // {
  2952. // ctx->mFailed = true;
  2953. // return 0;
  2954. // }
  2955. // return Hash(underlyingType, ctx, allowRef);
  2956. // }
  2957. // else
  2958. if (type->IsBoxed())
  2959. {
  2960. BfBoxedType* boxedType = (BfBoxedType*)type;
  2961. int elemHash = Hash(boxedType->mElementType, ctx, BfHashFlag_None, hashSeed) ^ HASH_VAL_BOXED;
  2962. return (elemHash << 5) - elemHash;
  2963. }
  2964. else if (type->IsArray())
  2965. {
  2966. BfArrayType* arrayType = (BfArrayType*)type;
  2967. int elemHash = Hash(arrayType->mGenericTypeInfo->mTypeGenericArguments[0], ctx, BfHashFlag_None, hashSeed) ^ (arrayType->mDimensions << 8);
  2968. return (elemHash << 5) - elemHash;
  2969. }
  2970. else if (type->IsDelegateFromTypeRef() || type->IsFunctionFromTypeRef())
  2971. {
  2972. auto typeInst = (BfTypeInstance*)type;
  2973. int hashVal = HASH_DELEGATE;
  2974. auto delegateInfo = type->GetDelegateInfo();
  2975. hashVal = HASH_MIX(hashVal, Hash(delegateInfo->mReturnType, ctx, BfHashFlag_None, hashSeed + 1));
  2976. auto methodDef = typeInst->mTypeDef->mMethods[0];
  2977. BF_ASSERT(methodDef->mName == "Invoke");
  2978. int infoParamCount = (int)delegateInfo->mParams.size();
  2979. if (delegateInfo->mHasVarArgs)
  2980. infoParamCount++;
  2981. BF_ASSERT(infoParamCount == methodDef->mParams.size());
  2982. for (int paramIdx = 0; paramIdx < delegateInfo->mParams.size(); paramIdx++)
  2983. {
  2984. // Parse attributes?
  2985. hashVal = HASH_MIX(hashVal, Hash(delegateInfo->mParams[paramIdx], ctx, BfHashFlag_None, hashSeed + 1));
  2986. String paramName = methodDef->mParams[paramIdx]->mName;
  2987. int nameHash = (int)Hash64(paramName.c_str(), (int)paramName.length());
  2988. hashVal = HASH_MIX(hashVal, nameHash);
  2989. }
  2990. if (delegateInfo->mHasVarArgs)
  2991. hashVal = HASH_MIX(hashVal, HASH_DOTDOTDOT);
  2992. return hashVal;
  2993. }
  2994. else if ((type->IsEnum()) && (type->IsOnDemand()))
  2995. {
  2996. BfTypeInstance* typeInstance = type->ToTypeInstance();
  2997. auto fieldName = typeInstance->mTypeDef->mFields[0]->mName;
  2998. int nameHash = (int)Hash64(fieldName.c_str(), (int)fieldName.length());
  2999. return nameHash ^ HASH_TAG;
  3000. }
  3001. else if (type->IsTypeInstance())
  3002. {
  3003. BfTypeInstance* typeInst = (BfTypeInstance*)type;
  3004. int hashVal;
  3005. if (typeInst->mTypeDef != NULL)
  3006. hashVal = typeInst->mTypeDef->mHash;
  3007. if (type->IsClosure())
  3008. {
  3009. auto closureType = (BfClosureType*)type;
  3010. if (closureType->mIsUnique)
  3011. return false;
  3012. hashVal = ((hashVal ^ (int)closureType->mClosureHash.mLow) << 5) - hashVal;
  3013. }
  3014. else if (type->IsTuple())
  3015. {
  3016. hashVal = HASH_VAL_TUPLE;
  3017. BfTypeInstance* tupleType = (BfTypeInstance*)type;
  3018. for (int fieldIdx = 0; fieldIdx < (int)tupleType->mFieldInstances.size(); fieldIdx++)
  3019. {
  3020. BfFieldInstance* fieldInstance = &tupleType->mFieldInstances[fieldIdx];
  3021. auto fieldType = fieldInstance->mResolvedType;
  3022. hashVal = HASH_MIX(hashVal, Hash(fieldType, ctx, BfHashFlag_None, hashSeed + 1));
  3023. BfFieldDef* fieldDef = NULL;
  3024. if (tupleType->mTypeDef != NULL)
  3025. fieldDef = fieldInstance->GetFieldDef();
  3026. int nameHash = 0;
  3027. if (fieldDef == NULL)
  3028. {
  3029. char nameStr[64];
  3030. sprintf(nameStr, "%d", fieldIdx);
  3031. nameHash = (int)Hash64(nameStr, strlen(nameStr));
  3032. }
  3033. else
  3034. {
  3035. nameHash = (int)Hash64(fieldDef->mName.c_str(), (int)fieldDef->mName.length());
  3036. }
  3037. hashVal = HASH_MIX(hashVal, nameHash);
  3038. }
  3039. }
  3040. else if (type->IsGenericTypeInstance())
  3041. {
  3042. BfTypeInstance* genericType = (BfTypeInstance*)type;
  3043. for (auto genericArg : genericType->mGenericTypeInfo->mTypeGenericArguments)
  3044. hashVal = HASH_MIX(hashVal, Hash(genericArg, ctx, BfHashFlag_None, hashSeed + 1));
  3045. }
  3046. return hashVal;
  3047. }
  3048. else if (type->IsPrimitiveType())
  3049. {
  3050. BfPrimitiveType* primType = (BfPrimitiveType*)type;
  3051. return primType->mTypeDef->mHash;
  3052. }
  3053. else if (type->IsPointer())
  3054. {
  3055. BfPointerType* pointerType = (BfPointerType*) type;
  3056. int elemHash = Hash(pointerType->mElementType, ctx, BfHashFlag_None, hashSeed) ^ HASH_VAL_PTR;
  3057. return (elemHash << 5) - elemHash;
  3058. }
  3059. else if (type->IsGenericParam())
  3060. {
  3061. auto genericParam = (BfGenericParamType*)type;
  3062. return (((int)genericParam->mGenericParamKind + 0xB00) << 8) ^ (genericParam->mGenericParamIdx + 1);
  3063. }
  3064. else if (type->IsRef())
  3065. {
  3066. auto refType = (BfRefType*)type;
  3067. int elemHash = Hash(refType->mElementType, ctx, BfHashFlag_None, hashSeed) ^ (HASH_VAL_REF + (int)refType->mRefKind);
  3068. return (elemHash << 5) - elemHash;
  3069. }
  3070. else if (type->IsModifiedTypeType())
  3071. {
  3072. auto modifiedTypeType = (BfModifiedTypeType*)type;
  3073. int elemHash = Hash(modifiedTypeType->mElementType, ctx, BfHashFlag_None, hashSeed) ^ HASH_MODTYPE + (int)modifiedTypeType->mModifiedKind;
  3074. return (elemHash << 5) - elemHash;
  3075. }
  3076. else if (type->IsConcreteInterfaceType())
  3077. {
  3078. auto concreteInterfaceType = (BfConcreteInterfaceType*)type;
  3079. int elemHash = Hash(concreteInterfaceType->mInterface, ctx, BfHashFlag_None, hashSeed) ^ HASH_CONCRETE_INTERFACE;
  3080. return (elemHash << 5) - elemHash;
  3081. }
  3082. else if (type->IsSizedArray())
  3083. {
  3084. auto sizedArray = (BfSizedArrayType*)type;
  3085. int elemHash = Hash(sizedArray->mElementType, ctx, BfHashFlag_None, hashSeed) ^ HASH_SIZED_ARRAY;
  3086. int hashVal = (elemHash << 5) - elemHash;
  3087. if (type->IsUnknownSizedArrayType())
  3088. {
  3089. auto unknownSizedArray = (BfUnknownSizedArrayType*)type;
  3090. int elemHash = Hash(unknownSizedArray->mElementCountSource, ctx, BfHashFlag_None, hashSeed);
  3091. hashVal = ((hashVal ^ elemHash) << 5) - hashVal;
  3092. }
  3093. else
  3094. hashVal = ((hashVal ^ (int)sizedArray->mElementCount) << 5) - hashVal;
  3095. return hashVal;
  3096. }
  3097. else if (type->IsMethodRef())
  3098. {
  3099. auto methodRefType = (BfMethodRefType*)type;
  3100. if (methodRefType->IsNull())
  3101. return 0;
  3102. return (int)((int)(intptr)(methodRefType->mMethodRef) << 5) ^ (int)(intptr)(methodRefType->mOwner) ^ methodRefType->mOwnerRevision;
  3103. }
  3104. else if (type->IsConstExprValue())
  3105. {
  3106. BfConstExprValueType* constExprValueType = (BfConstExprValueType*)type;
  3107. int32 dataHash = BeefHash<BfVariant>()(constExprValueType->mValue);
  3108. int hashVal = ((int)constExprValueType->mValue.mTypeCode << 17) ^ (dataHash << 3) ^ HASH_CONSTTYPE;
  3109. hashVal = ((hashVal ^ (Hash(constExprValueType->mType, ctx, BfHashFlag_AllowRef, hashSeed))) << 5) - hashVal;
  3110. return hashVal;
  3111. }
  3112. else
  3113. {
  3114. BF_FATAL("Not handled");
  3115. }
  3116. return 0;
  3117. }
  3118. int BfResolvedTypeSet::Hash(BfType* type, LookupContext* ctx, bool allowRef, int hashSeed)
  3119. {
  3120. int hashVal = DoHash(type, ctx, allowRef, hashSeed);
  3121. if (hashSeed == 0)
  3122. return hashVal;
  3123. return HASH_MIX(hashVal, hashSeed);
  3124. }
  3125. void BfResolvedTypeSet::HashGenericArguments(BfTypeReference* typeRef, LookupContext* ctx, int& hashVal, int hashSeed)
  3126. {
  3127. if (auto genericTypeRef = BfNodeDynCast<BfGenericInstanceTypeRef>(typeRef))
  3128. {
  3129. for (int genericIdx = 0; genericIdx < BF_MAX(genericTypeRef->mGenericArguments.mSize, genericTypeRef->mCommas.mSize + 1); genericIdx++)
  3130. {
  3131. bool allowUnboundGeneric = ((ctx->mResolveFlags & BfResolveTypeRefFlag_AllowUnboundGeneric) != 0) && (hashSeed == 0);
  3132. BfAstNode* genericArgTypeRef = NULL;
  3133. if (genericIdx < genericTypeRef->mGenericArguments.mSize)
  3134. genericArgTypeRef = genericTypeRef->mGenericArguments[genericIdx];
  3135. if (allowUnboundGeneric)
  3136. {
  3137. if (BfNodeIsExact<BfWildcardTypeReference>(genericArgTypeRef))
  3138. genericArgTypeRef = NULL;
  3139. }
  3140. int argHashVal = 0;
  3141. if (genericArgTypeRef != NULL)
  3142. {
  3143. argHashVal = Hash(genericArgTypeRef, ctx, BfHashFlag_AllowGenericParamConstValue, hashSeed + 1);
  3144. if ((allowUnboundGeneric) && ((ctx->mResolveFlags & BfResolveTypeRefFlag_ForceUnboundGeneric) != 0))
  3145. genericArgTypeRef = NULL;
  3146. }
  3147. if (genericArgTypeRef == NULL)
  3148. {
  3149. if (allowUnboundGeneric)
  3150. {
  3151. ctx->mIsUnboundGeneric = true;
  3152. argHashVal = (((int)BfGenericParamKind_Type + 0xB00) << 8) ^ (genericIdx + 1);
  3153. argHashVal = HASH_MIX(argHashVal, hashSeed + 1);
  3154. }
  3155. else
  3156. {
  3157. ctx->mModule->Fail("Generic argument expected", genericTypeRef->mOpenChevron);
  3158. ctx->mFailed = true;
  3159. return;
  3160. }
  3161. }
  3162. hashVal = HASH_MIX(hashVal, argHashVal);
  3163. }
  3164. }
  3165. }
  3166. static int HashNode(BfAstNode* node)
  3167. {
  3168. if (node == NULL)
  3169. return (int)Hash64(NULL, 0);
  3170. const char* nameStr = node->GetSourceData()->mSrc + node->GetSrcStart();
  3171. return (int)Hash64(nameStr, node->GetSrcLength());
  3172. }
  3173. BfResolveTypeRefFlags BfResolvedTypeSet::GetResolveFlags(BfAstNode* typeRef, LookupContext* ctx, BfHashFlags flags)
  3174. {
  3175. bool isHeadType = typeRef == ctx->mRootTypeRef;
  3176. BfResolveTypeRefFlags resolveFlags = ctx->mResolveFlags;
  3177. if ((flags & BfHashFlag_AllowGenericParamConstValue) != 0)
  3178. resolveFlags = (BfResolveTypeRefFlags)(resolveFlags | BfResolveTypeRefFlag_AllowGenericParamConstValue);
  3179. if (!isHeadType)
  3180. resolveFlags = (BfResolveTypeRefFlags)(resolveFlags & ~BfResolveTypeRefFlag_Attribute);
  3181. return resolveFlags;
  3182. }
  3183. int BfResolvedTypeSet::DirectHash(BfTypeReference* typeRef, LookupContext* ctx, BfHashFlags flags, int hashSeed)
  3184. {
  3185. auto resolvedType = ctx->mModule->ResolveTypeRef(typeRef, BfPopulateType_Identity, GetResolveFlags(typeRef, ctx, flags));
  3186. if (resolvedType == NULL)
  3187. {
  3188. ctx->mFailed = true;
  3189. return 0;
  3190. }
  3191. if (((flags & BfHashFlag_DisallowPointer) != 0) && (resolvedType->IsPointer()))
  3192. {
  3193. ShowThisPointerWarning(ctx, typeRef);
  3194. resolvedType = resolvedType->GetUnderlyingType();
  3195. }
  3196. return Hash(resolvedType, ctx, BfHashFlag_None, hashSeed);
  3197. }
  3198. void BfResolvedTypeSet::ShowThisPointerWarning(LookupContext* ctx, BfTypeReference* typeRef)
  3199. {
  3200. ctx->mModule->Warn(0, "Pointer types cannot be used as 'this'. If 'this' address is required, use 'mut' or [CRepr]", typeRef);
  3201. }
  3202. BfTypeDef* BfResolvedTypeSet::FindRootCommonOuterType(BfTypeDef* outerType, LookupContext* ctx, BfTypeInstance*& outOuterTypeInstance)
  3203. {
  3204. if (ctx->mModule->mCurTypeInstance == NULL)
  3205. return NULL;
  3206. BfTypeDef* commonOuterType = NULL;
  3207. auto checkOuterTypeInst = ctx->mModule->mCurTypeInstance;
  3208. while (checkOuterTypeInst != NULL)
  3209. {
  3210. commonOuterType = ctx->mModule->FindCommonOuterType(checkOuterTypeInst->mTypeDef, outerType);
  3211. if (commonOuterType != NULL)
  3212. {
  3213. outOuterTypeInstance = checkOuterTypeInst;
  3214. break;
  3215. }
  3216. checkOuterTypeInst = checkOuterTypeInst->mBaseType;
  3217. }
  3218. if ((commonOuterType == NULL) && (outerType != NULL))
  3219. {
  3220. auto staticSearch = ctx->mModule->GetStaticSearch();
  3221. if (staticSearch != NULL)
  3222. {
  3223. for (auto staticTypeInst : staticSearch->mStaticTypes)
  3224. {
  3225. auto foundOuterType = ctx->mModule->FindCommonOuterType(staticTypeInst->mTypeDef, outerType);
  3226. if ((foundOuterType != NULL) &&
  3227. ((commonOuterType == NULL) || (foundOuterType->mNestDepth > commonOuterType->mNestDepth)))
  3228. {
  3229. commonOuterType = foundOuterType;
  3230. outOuterTypeInstance = staticTypeInst;
  3231. }
  3232. }
  3233. }
  3234. }
  3235. if (outOuterTypeInstance != NULL)
  3236. ctx->mRootOuterTypeInstance = outOuterTypeInstance;
  3237. return commonOuterType;
  3238. }
  3239. int BfResolvedTypeSet::DoHash(BfTypeReference* typeRef, LookupContext* ctx, BfHashFlags flags, int& hashSeed)
  3240. {
  3241. if ((typeRef == ctx->mRootTypeRef) && (ctx->mRootTypeDef != NULL) &&
  3242. ((typeRef->IsNamedTypeReference()) || (BfNodeIsA<BfDirectTypeDefReference>(typeRef))))
  3243. {
  3244. BfTypeDef* typeDef = ctx->mRootTypeDef;
  3245. int hashVal = typeDef->mHash;
  3246. if (typeDef->mGenericParamDefs.size() != 0)
  3247. {
  3248. BfTypeDef* commonOuterType = NULL;
  3249. auto checkTypeInstance = ctx->mModule->mCurTypeInstance;
  3250. if (checkTypeInstance != NULL)
  3251. {
  3252. if (checkTypeInstance->IsBoxed())
  3253. checkTypeInstance = checkTypeInstance->GetUnderlyingType()->ToTypeInstance();
  3254. auto outerType = ctx->mModule->mSystem->GetOuterTypeNonPartial(typeDef);
  3255. if (typeRef == ctx->mRootTypeRef)
  3256. {
  3257. commonOuterType = FindRootCommonOuterType(outerType, ctx, checkTypeInstance);
  3258. if ((checkTypeInstance != NULL) && (checkTypeInstance->IsBoxed()))
  3259. checkTypeInstance = checkTypeInstance->GetUnderlyingType()->ToTypeInstance();
  3260. }
  3261. else
  3262. commonOuterType = ctx->mModule->FindCommonOuterType(ctx->mModule->mCurTypeInstance->mTypeDef, outerType);
  3263. if ((commonOuterType == NULL) && (outerType != NULL))
  3264. {
  3265. auto staticSearch = ctx->mModule->GetStaticSearch();
  3266. if (staticSearch != NULL)
  3267. {
  3268. for (auto staticTypeInst : staticSearch->mStaticTypes)
  3269. {
  3270. auto foundOuterType = ctx->mModule->FindCommonOuterType(staticTypeInst->mTypeDef, outerType);
  3271. if ((foundOuterType != NULL) &&
  3272. ((commonOuterType == NULL) || (foundOuterType->mNestDepth > commonOuterType->mNestDepth)))
  3273. {
  3274. commonOuterType = foundOuterType;
  3275. checkTypeInstance = staticTypeInst;
  3276. }
  3277. }
  3278. }
  3279. }
  3280. }
  3281. if ((commonOuterType == NULL) || (commonOuterType->mGenericParamDefs.size() == 0))
  3282. {
  3283. ctx->mModule->Fail("Generic arguments expected", typeRef);
  3284. ctx->mFailed = true;
  3285. return 0;
  3286. }
  3287. BF_ASSERT(checkTypeInstance->IsGenericTypeInstance());
  3288. auto curGenericTypeInst = (BfTypeInstance*)checkTypeInstance;
  3289. int numParentGenericParams = (int)commonOuterType->mGenericParamDefs.size();
  3290. for (int i = 0; i < numParentGenericParams; i++)
  3291. {
  3292. hashVal = HASH_MIX(hashVal, Hash(curGenericTypeInst->mGenericTypeInfo->mTypeGenericArguments[i], ctx, BfHashFlag_None, hashSeed + 1));
  3293. }
  3294. }
  3295. return hashVal;
  3296. }
  3297. if (typeRef->IsNamedTypeReference())
  3298. {
  3299. int hashVal = DirectHash(typeRef, ctx, flags, hashSeed);
  3300. hashSeed = 0;
  3301. return hashVal;
  3302. }
  3303. if (auto genericInstTypeRef = BfNodeDynCastExact<BfGenericInstanceTypeRef>(typeRef))
  3304. {
  3305. //BfType* type = NULL;
  3306. BfTypeDef* elementTypeDef = ctx->mModule->ResolveGenericInstanceDef(genericInstTypeRef, NULL, GetResolveFlags(genericInstTypeRef, ctx, flags));
  3307. if (elementTypeDef == NULL)
  3308. {
  3309. ctx->mFailed = true;
  3310. return 0;
  3311. }
  3312. int typeAliasHash = 0;
  3313. bool isInnerTypeAlias = false;
  3314. // Don't translate aliases for the root type, just element types
  3315. if (ctx->mRootTypeRef == typeRef)
  3316. {
  3317. BF_ASSERT((ctx->mRootTypeDef == NULL) || (ctx->mRootTypeDef == elementTypeDef));
  3318. ctx->mRootTypeDef = elementTypeDef;
  3319. }
  3320. else if (elementTypeDef->mTypeCode == BfTypeCode_TypeAlias)
  3321. {
  3322. isInnerTypeAlias = true;
  3323. }
  3324. BfTypeVector typeAliasGenericArgs;
  3325. bool fullyQualified = false;
  3326. int hashVal = elementTypeDef->mHash;
  3327. BfTypeInstance* outerType = NULL;
  3328. int checkIdx = 0;
  3329. auto checkTypeRef = genericInstTypeRef->mElementType;
  3330. while (checkTypeRef != NULL)
  3331. {
  3332. checkIdx++;
  3333. if (checkIdx >= 2)
  3334. {
  3335. fullyQualified = true;
  3336. if ((elementTypeDef->mOuterType != NULL) && (!elementTypeDef->mOuterType->mGenericParamDefs.IsEmpty()))
  3337. {
  3338. auto resolvedType = ctx->mModule->ResolveTypeRef(checkTypeRef, BfPopulateType_Identity,
  3339. (BfResolveTypeRefFlags)(GetResolveFlags(checkTypeRef, ctx, flags) | BfResolveTypeRefFlag_IgnoreLookupError));
  3340. if (resolvedType == NULL)
  3341. {
  3342. ctx->mFailed = true;
  3343. return hashVal;
  3344. }
  3345. ctx->SetCachedResolvedType(checkTypeRef, resolvedType);
  3346. outerType = resolvedType->ToTypeInstance();
  3347. }
  3348. break;
  3349. }
  3350. if (auto elementedTypeRef = BfNodeDynCast<BfElementedTypeRef>(checkTypeRef))
  3351. {
  3352. checkTypeRef = elementedTypeRef->mElementType;
  3353. continue;
  3354. }
  3355. if (auto qualifiedTypeRef = BfNodeDynCast<BfQualifiedTypeReference>(checkTypeRef))
  3356. {
  3357. checkTypeRef = qualifiedTypeRef->mLeft;
  3358. continue;
  3359. }
  3360. break;
  3361. }
  3362. if (fullyQualified)
  3363. {
  3364. if (outerType != NULL)
  3365. {
  3366. for (auto genericArg : outerType->mGenericTypeInfo->mTypeGenericArguments)
  3367. {
  3368. if (isInnerTypeAlias)
  3369. typeAliasGenericArgs.Add(genericArg);
  3370. else
  3371. hashVal = HASH_MIX(hashVal, Hash(genericArg, ctx, Beefy::BfResolvedTypeSet::BfHashFlag_None, hashSeed + 1));
  3372. }
  3373. }
  3374. }
  3375. else
  3376. {
  3377. // Do we need to add generic arguments from an in-context outer class?
  3378. if ((elementTypeDef->mOuterType != NULL) && (ctx->mModule->mCurTypeInstance != NULL))
  3379. {
  3380. BfTypeInstance* checkTypeInstance = ctx->mModule->mCurTypeInstance;
  3381. BfTypeDef* commonOuterType;
  3382. if (typeRef == ctx->mRootTypeRef)
  3383. commonOuterType = FindRootCommonOuterType(elementTypeDef->mOuterType, ctx, checkTypeInstance);
  3384. else
  3385. commonOuterType = ctx->mModule->FindCommonOuterType(ctx->mModule->mCurTypeInstance->mTypeDef, elementTypeDef->mOuterType);
  3386. if ((commonOuterType != NULL) && (checkTypeInstance->IsGenericTypeInstance()))
  3387. {
  3388. auto parentTypeInstance = checkTypeInstance;
  3389. int numParentGenericParams = (int)commonOuterType->mGenericParamDefs.size();
  3390. for (int i = 0; i < numParentGenericParams; i++)
  3391. {
  3392. if (isInnerTypeAlias)
  3393. typeAliasGenericArgs.Add(parentTypeInstance->mGenericTypeInfo->mTypeGenericArguments[i]);
  3394. else
  3395. hashVal = HASH_MIX(hashVal, Hash(parentTypeInstance->mGenericTypeInfo->mTypeGenericArguments[i], ctx, Beefy::BfResolvedTypeSet::BfHashFlag_None, hashSeed + 1));
  3396. }
  3397. }
  3398. }
  3399. }
  3400. if (isInnerTypeAlias)
  3401. {
  3402. for (auto genericArgTypeRef : genericInstTypeRef->mGenericArguments)
  3403. {
  3404. auto argType = ctx->mModule->ResolveTypeRef(genericArgTypeRef, NULL, BfPopulateType_Identity, GetResolveFlags(genericArgTypeRef, ctx, flags));
  3405. if (argType != NULL)
  3406. typeAliasGenericArgs.Add(argType);
  3407. else
  3408. ctx->mFailed = true;
  3409. }
  3410. if (!ctx->mFailed)
  3411. {
  3412. auto resolvedType = ctx->mModule->ResolveTypeDef(elementTypeDef, typeAliasGenericArgs);
  3413. if ((resolvedType != NULL) && (resolvedType->IsTypeAlias()))
  3414. {
  3415. auto underlyingType = resolvedType->GetUnderlyingType();
  3416. if (underlyingType == NULL)
  3417. {
  3418. ctx->mFailed = true;
  3419. return 0;
  3420. }
  3421. int hashVal = Hash(underlyingType, ctx, flags, hashSeed);
  3422. hashSeed = 0;
  3423. return hashVal;
  3424. }
  3425. }
  3426. }
  3427. else
  3428. HashGenericArguments(genericInstTypeRef, ctx, hashVal, hashSeed);
  3429. return hashVal;
  3430. }
  3431. else if (auto tupleTypeRef = BfNodeDynCastExact<BfTupleTypeRef>(typeRef))
  3432. {
  3433. int hashVal = HASH_VAL_TUPLE;
  3434. for (int fieldIdx = 0; fieldIdx < (int)tupleTypeRef->mFieldTypes.size(); fieldIdx++)
  3435. {
  3436. BfTypeReference* fieldType = tupleTypeRef->mFieldTypes[fieldIdx];
  3437. hashVal = HASH_MIX(hashVal, Hash(fieldType, ctx, BfHashFlag_None, hashSeed + 1));
  3438. int nameHash = 0;
  3439. BfIdentifierNode* fieldName = NULL;
  3440. if (fieldIdx < (int)tupleTypeRef->mFieldNames.size())
  3441. fieldName = tupleTypeRef->mFieldNames[fieldIdx];
  3442. if (fieldName != NULL)
  3443. {
  3444. const char* nameStr = fieldName->GetSourceData()->mSrc + fieldName->GetSrcStart();
  3445. nameHash = (int)Hash64(nameStr, fieldName->GetSrcLength());
  3446. }
  3447. else
  3448. {
  3449. char nameStr[64];
  3450. sprintf(nameStr, "%d", fieldIdx);
  3451. nameHash = (int)Hash64(nameStr, strlen(nameStr));
  3452. }
  3453. hashVal = HASH_MIX(hashVal, nameHash);
  3454. }
  3455. return hashVal;
  3456. }
  3457. else if (auto arrayType = BfNodeDynCastExact<BfArrayTypeRef>(typeRef))
  3458. {
  3459. if ((arrayType->mDimensions == 1) && (arrayType->mParams.size() != 0))
  3460. {
  3461. int rawElemHash = Hash(arrayType->mElementType, ctx, BfHashFlag_None, hashSeed);
  3462. int elemHash = rawElemHash ^ HASH_SIZED_ARRAY;
  3463. int hashVal = (elemHash << 5) - elemHash;
  3464. // Sized array
  3465. if (arrayType->mParams.size() != 1)
  3466. {
  3467. ctx->mFailed = true;
  3468. ctx->mModule->Fail("Only one size parameter expected", arrayType->mParams[1]);
  3469. return 0;
  3470. }
  3471. intptr elementCount = -1;
  3472. BfExpression* sizeExpr = BfNodeDynCast<BfExpression>(arrayType->mParams[0]);
  3473. BF_ASSERT(sizeExpr != NULL);
  3474. if (sizeExpr != NULL)
  3475. {
  3476. BfMethodState methodState;
  3477. methodState.mTempKind = BfMethodState::TempKind_Static;
  3478. SetAndRestoreValue<BfMethodState*> prevMethodState;
  3479. if (ctx->mModule->mCurMethodState == NULL)
  3480. prevMethodState.Init(ctx->mModule->mCurMethodState, &methodState);
  3481. BfConstResolver constResolver(ctx->mModule);
  3482. BfType* intType = ctx->mModule->GetPrimitiveType(BfTypeCode_IntPtr);
  3483. constResolver.mBfEvalExprFlags = (BfEvalExprFlags)(constResolver.mBfEvalExprFlags | BfEvalExprFlags_AllowGenericConstValue);
  3484. constResolver.mExpectingType = intType;
  3485. BfTypedValue typedVal = constResolver.Resolve(sizeExpr, NULL, BfConstResolveFlag_ArrayInitSize);
  3486. if (typedVal.mKind == BfTypedValueKind_GenericConstValue)
  3487. {
  3488. ctx->mResolvedValueMap[sizeExpr] = typedVal;
  3489. int elemHash = Hash(typedVal.mType, ctx, BfHashFlag_None, hashSeed);
  3490. hashVal = ((hashVal ^ elemHash) << 5) - hashVal;
  3491. return hashVal;
  3492. }
  3493. if (!typedVal)
  3494. ctx->mFailed = true;
  3495. if (typedVal)
  3496. {
  3497. SetAndRestoreValue<bool> prevIgnoreWrites(ctx->mModule->mBfIRBuilder->mIgnoreWrites, true);
  3498. typedVal = ctx->mModule->Cast(sizeExpr, typedVal, intType);
  3499. }
  3500. ctx->mResolvedValueMap[sizeExpr] = typedVal;
  3501. if (typedVal)
  3502. {
  3503. auto constant = ctx->mModule->mBfIRBuilder->GetConstant(typedVal.mValue);
  3504. if (constant == NULL)
  3505. {
  3506. ctx->mFailed = true;
  3507. ctx->mModule->Fail("Array size not a constant value", arrayType->mParams[0]);
  3508. }
  3509. else if (constant->mConstType == BfConstType_Undef)
  3510. {
  3511. elementCount = -1; // Marker for undef
  3512. if ((arrayType->IsInferredSize()) && ((ctx->mResolveFlags & BfResolveTypeRefFlag_AllowInferredSizedArray) == 0))
  3513. {
  3514. ctx->mModule->Fail("Invalid use of inferred-sized array", sizeExpr);
  3515. }
  3516. }
  3517. else if (!BfIRBuilder::IsInt(constant->mTypeCode))
  3518. {
  3519. ctx->mFailed = true;
  3520. ctx->mModule->Fail("Array size not a constant value", arrayType->mParams[0]);
  3521. }
  3522. else
  3523. {
  3524. elementCount = (intptr)constant->mInt64;
  3525. if (elementCount < 0)
  3526. {
  3527. ctx->mFailed = true;
  3528. ctx->mModule->Fail("Arrays cannot have negative sizes", arrayType->mParams[0]);
  3529. return 0;
  3530. }
  3531. }
  3532. }
  3533. }
  3534. hashVal = ((hashVal ^ (int)elementCount) << 5) - hashVal;
  3535. return hashVal;
  3536. }
  3537. else
  3538. {
  3539. if (arrayType->mDimensions != (int)arrayType->mParams.size() + 1)
  3540. {
  3541. for (auto arg : arrayType->mParams)
  3542. {
  3543. if (auto tokenNode = BfNodeDynCastExact<BfTokenNode>(arg))
  3544. {
  3545. if (tokenNode->GetToken() == BfToken_Comma)
  3546. continue;
  3547. }
  3548. ctx->mFailed = true;
  3549. ctx->mModule->Fail("Multidimensional arrays cannot have explicit sizes. Consider using a strided array (ie: int[2][2]) instead.", arg);
  3550. return 0;
  3551. }
  3552. }
  3553. int elemHash = Hash(arrayType->mElementType, ctx, BfHashFlag_None, hashSeed) ^ (arrayType->mDimensions << 8);
  3554. return (elemHash << 5) - elemHash;
  3555. }
  3556. }
  3557. else if (auto pointerType = BfNodeDynCastExact<BfPointerTypeRef>(typeRef))
  3558. {
  3559. int elemHash = Hash(pointerType->mElementType, ctx, BfHashFlag_None, hashSeed) ^ HASH_VAL_PTR;
  3560. return (elemHash << 5) - elemHash;
  3561. }
  3562. else if (auto nullableType = BfNodeDynCastExact<BfNullableTypeRef>(typeRef))
  3563. {
  3564. if (ctx->mRootTypeRef == typeRef)
  3565. ctx->mRootTypeDef = ctx->mModule->mCompiler->mNullableTypeDef;
  3566. int hashVal = ctx->mModule->mCompiler->mNullableTypeDef->mHash;
  3567. hashVal = HASH_MIX(hashVal, Hash(nullableType->mElementType, ctx, BfHashFlag_None, hashSeed + 1));
  3568. return hashVal;
  3569. }
  3570. else if (auto refType = BfNodeDynCastExact<BfRefTypeRef>(typeRef))
  3571. {
  3572. if ((flags & BfHashFlag_AllowRef) != 0)
  3573. {
  3574. auto refKind = BfRefType::RefKind_Ref;
  3575. if (refType->mRefToken == NULL)
  3576. refKind = BfRefType::RefKind_Ref;
  3577. else if (refType->mRefToken->GetToken() == BfToken_In)
  3578. refKind = BfRefType::RefKind_In;
  3579. else if (refType->mRefToken->GetToken() == BfToken_Out)
  3580. refKind = BfRefType::RefKind_Out;
  3581. else if (refType->mRefToken->GetToken() == BfToken_Mut)
  3582. refKind = BfRefType::RefKind_Mut;
  3583. int elemHash = Hash(refType->mElementType, ctx, BfHashFlag_None, hashSeed) ^ (HASH_VAL_REF + (int)refKind);
  3584. return (elemHash << 5) - elemHash;
  3585. }
  3586. else
  3587. {
  3588. ctx->mModule->ResolveTypeRef(typeRef, BfPopulateType_Identity, GetResolveFlags(typeRef, ctx, flags)); // To throw an error...
  3589. ctx->mFailed = true;
  3590. return 0;
  3591. //return Hash(refType->mElementType, ctx);
  3592. }
  3593. }
  3594. else if (auto genericParamTypeRef = BfNodeDynCastExact<BfGenericParamTypeRef>(typeRef))
  3595. {
  3596. return (((int)genericParamTypeRef->mGenericParamKind) << 8) ^ (genericParamTypeRef->mGenericParamIdx + 1);
  3597. }
  3598. else if (auto qualifiedTypeRef = BfNodeDynCastExact<BfQualifiedTypeReference>(typeRef))
  3599. {
  3600. /*auto leftType = ctx->mModule->ResolveTypeRef(qualifiedTypeRef->mLeft, BfPopulateType_Identity);
  3601. if (leftType == NULL)
  3602. {
  3603. ctx->mFailed = true;
  3604. return 0;
  3605. }
  3606. if (qualifiedTypeRef->mRight == NULL)
  3607. {
  3608. ctx->mFailed = true;
  3609. return 0;
  3610. }
  3611. auto rightType = ctx->mModule->ResolveInnerType(leftType, qualifiedTypeRef->mRight);
  3612. if (rightType == NULL)
  3613. {
  3614. ctx->mFailed = true;
  3615. return 0;
  3616. }
  3617. return Hash(rightType, ctx);*/
  3618. auto resolvedType = ctx->mModule->ResolveTypeRef(typeRef, BfPopulateType_Identity, GetResolveFlags(typeRef, ctx, flags));
  3619. if (resolvedType == NULL)
  3620. {
  3621. ctx->mFailed = true;
  3622. return 0;
  3623. }
  3624. int hashVal = Hash(resolvedType, ctx, BfHashFlag_None, hashSeed);
  3625. hashSeed = 0;
  3626. return hashVal;
  3627. }
  3628. else if (auto varType = BfNodeDynCastExact<BfVarTypeReference>(typeRef))
  3629. {
  3630. // Don't allow 'var'
  3631. ctx->mModule->Fail("Invalid use of 'var'", typeRef);
  3632. ctx->mFailed = true;
  3633. return 0;
  3634. }
  3635. else if (auto letType = BfNodeDynCastExact<BfLetTypeReference>(typeRef))
  3636. {
  3637. // Don't allow 'let'
  3638. ctx->mModule->Fail("Invalid use of 'let'", typeRef);
  3639. ctx->mFailed = true;
  3640. return 0;
  3641. }
  3642. else if (auto retTypeTypeRef = BfNodeDynCastExact<BfModifiedTypeRef>(typeRef))
  3643. {
  3644. // Don't cause infinite loop, but if we have an inner 'rettype' then try to directly resolve that --
  3645. // Only use the HAS_RETTYPE for root-level rettype insertions
  3646. if (ctx->mRootTypeRef != retTypeTypeRef)
  3647. {
  3648. auto type = ctx->mModule->ResolveTypeRef(retTypeTypeRef, BfPopulateType_Identity, GetResolveFlags(retTypeTypeRef, ctx, flags));
  3649. if (type == NULL)
  3650. {
  3651. ctx->mFailed = true;
  3652. return 0;
  3653. }
  3654. if (type->IsRef())
  3655. type = type->GetUnderlyingType();
  3656. return DoHash(type, ctx, flags, hashSeed);
  3657. }
  3658. int elemHash = Hash(retTypeTypeRef->mElementType, ctx, BfHashFlag_None, hashSeed) ^ HASH_MODTYPE + retTypeTypeRef->mRetTypeToken->mToken;
  3659. return (elemHash << 5) - elemHash;
  3660. }
  3661. else if (auto resolvedTypeRef = BfNodeDynCastExact<BfResolvedTypeReference>(typeRef))
  3662. {
  3663. return Hash(resolvedTypeRef->mType, ctx, BfHashFlag_None, hashSeed);
  3664. }
  3665. else if (auto constTypeRef = BfNodeDynCastExact<BfConstTypeRef>(typeRef))
  3666. {
  3667. // We purposely don't mix in a HASH_CONSTTYPE because there's no such thing as a const type in Beef, so we just strip it
  3668. return Hash(constTypeRef->mElementType, ctx, flags, hashSeed);
  3669. }
  3670. else if (auto delegateTypeRef = BfNodeDynCastExact<BfDelegateTypeRef>(typeRef))
  3671. {
  3672. int hashVal = HASH_DELEGATE;
  3673. if (delegateTypeRef->mReturnType != NULL)
  3674. hashVal = HASH_MIX(hashVal, Hash(delegateTypeRef->mReturnType, ctx, BfHashFlag_AllowRef, hashSeed + 1));
  3675. else
  3676. ctx->mFailed = true;
  3677. bool isFirstParam = true;
  3678. for (int paramIdx = 0; paramIdx < delegateTypeRef->mParams.size(); paramIdx++)
  3679. {
  3680. auto param = delegateTypeRef->mParams[paramIdx];
  3681. // Parse attributes?
  3682. BfTypeReference* fieldType = param->mTypeRef;
  3683. auto hashFlags = (BfHashFlags)(BfHashFlag_AllowRef);
  3684. if (isFirstParam)
  3685. {
  3686. if ((param->mNameNode != NULL) && (param->mNameNode->Equals("this")))
  3687. {
  3688. hashFlags = (BfHashFlags)(hashFlags | BfHashFlag_DisallowPointer);
  3689. if (auto refNode = BfNodeDynCast<BfRefTypeRef>(fieldType))
  3690. fieldType = refNode->mElementType;
  3691. if (auto pointerType = BfNodeDynCast<BfPointerTypeRef>(fieldType))
  3692. {
  3693. ShowThisPointerWarning(ctx, pointerType);
  3694. fieldType = pointerType->mElementType;
  3695. }
  3696. }
  3697. }
  3698. if (paramIdx == delegateTypeRef->mParams.size() - 1)
  3699. {
  3700. if (auto dotTypeRef = BfNodeDynCastExact<BfDotTypeReference>(fieldType))
  3701. {
  3702. if (dotTypeRef->mDotToken->mToken == BfToken_DotDotDot)
  3703. {
  3704. hashVal = HASH_MIX(hashVal, HASH_DOTDOTDOT);
  3705. continue;
  3706. }
  3707. }
  3708. }
  3709. if (fieldType != NULL)
  3710. hashVal = HASH_MIX(hashVal, Hash(fieldType, ctx, hashFlags, hashSeed + 1));
  3711. hashVal = HASH_MIX(hashVal, HashNode(param->mNameNode));
  3712. isFirstParam = true;
  3713. }
  3714. return hashVal;
  3715. }
  3716. else if (auto exprModTypeRef = BfNodeDynCastExact<BfExprModTypeRef>(typeRef))
  3717. {
  3718. auto cachedResolvedType = ctx->GetCachedResolvedType(typeRef);
  3719. if (cachedResolvedType == NULL)
  3720. {
  3721. if (exprModTypeRef->mTarget != NULL)
  3722. {
  3723. BfTypedValue result;
  3724. //
  3725. {
  3726. BfMethodState methodState;
  3727. SetAndRestoreValue<BfMethodState*> prevMethodState(ctx->mModule->mCurMethodState, &methodState, false);
  3728. if (ctx->mModule->mCurMethodState == NULL)
  3729. prevMethodState.Set();
  3730. methodState.mTempKind = BfMethodState::TempKind_NonStatic;
  3731. SetAndRestoreValue<bool> ignoreWrites(ctx->mModule->mBfIRBuilder->mIgnoreWrites, true);
  3732. SetAndRestoreValue<bool> allowUninitReads(ctx->mModule->mCurMethodState->mAllowUinitReads, true);
  3733. BfEvalExprFlags exprFlags = BfEvalExprFlags_None;
  3734. if ((ctx->mResolveFlags & BfResolveTypeRefFlag_DisallowComptime) != 0)
  3735. {
  3736. exprFlags = (BfEvalExprFlags)(exprFlags | BfEvalExprFlags_DisallowComptime);
  3737. }
  3738. if (exprModTypeRef->mToken->mToken == BfToken_Comptype)
  3739. {
  3740. auto typeType = ctx->mModule->ResolveTypeDef(ctx->mModule->mCompiler->mTypeTypeDef);
  3741. exprFlags = (BfEvalExprFlags)(exprFlags | BfEvalExprFlags_Comptime | BfEvalExprFlags_NoCast);
  3742. result = ctx->mModule->CreateValueFromExpression(exprModTypeRef->mTarget, typeType, exprFlags);
  3743. if ((result.mType != NULL) && (!result.mType->IsInteger()) && (result.mType != typeType) &&
  3744. (!result.mType->IsInstanceOf(ctx->mModule->mCompiler->mReflectTypeIdTypeDef)))
  3745. result = ctx->mModule->Cast(exprModTypeRef->mTarget, result, typeType);
  3746. }
  3747. else
  3748. {
  3749. result = ctx->mModule->CreateValueFromExpression(exprModTypeRef->mTarget, NULL, BfEvalExprFlags_DeclType);
  3750. }
  3751. }
  3752. if ((result) && (exprModTypeRef->mToken->mToken == BfToken_Comptype))
  3753. {
  3754. auto constant = ctx->mModule->mBfIRBuilder->GetConstant(result.mValue);
  3755. if (constant != NULL)
  3756. {
  3757. if ((constant->mConstType == BfConstType_TypeOf) || (constant->mConstType == BfConstType_TypeOf_WithData))
  3758. {
  3759. auto typeOf = (BfTypeOf_Const*)constant;
  3760. cachedResolvedType = typeOf->mType;
  3761. }
  3762. else if (constant->mConstType == BfConstType_Undef)
  3763. {
  3764. ctx->mHadVar = true;
  3765. cachedResolvedType = ctx->mModule->GetPrimitiveType(BfTypeCode_Var);
  3766. auto typeState = ctx->mModule->mContext->mCurTypeState;
  3767. if ((typeState != NULL) && (typeState->mType != NULL) && (typeState->mType->IsTypeInstance()))
  3768. {
  3769. auto typeInst = typeState->mType->ToTypeInstance();
  3770. if (typeInst->mDefineState == BfTypeDefineState_ResolvingBaseType)
  3771. {
  3772. // Make sure we regenerate this type
  3773. ctx->mModule->mContext->mFailTypes.TryAdd(typeState->mType->ToTypeInstance(), BfFailKind_Deep);
  3774. }
  3775. }
  3776. }
  3777. else if (BfIRConstHolder::IsInt(constant->mTypeCode))
  3778. {
  3779. int typeId = constant->mInt32;
  3780. BfType* type = NULL;
  3781. if ((typeId >= 0) && (typeId < ctx->mModule->mContext->mTypes.mSize))
  3782. type = ctx->mModule->mContext->mTypes[typeId];
  3783. if (type != NULL)
  3784. {
  3785. cachedResolvedType = type;
  3786. }
  3787. else
  3788. {
  3789. ctx->mModule->Fail(StrFormat("Invalid type id '%d'", typeId), exprModTypeRef->mTarget);
  3790. ctx->mHadVar = true;
  3791. cachedResolvedType = ctx->mModule->GetPrimitiveType(BfTypeCode_Var);
  3792. }
  3793. }
  3794. }
  3795. if (cachedResolvedType == NULL)
  3796. ctx->mModule->Fail("Constant System.Type value required", exprModTypeRef->mTarget);
  3797. }
  3798. else
  3799. cachedResolvedType = result.mType;
  3800. if ((cachedResolvedType != NULL) && (cachedResolvedType->IsRef()))
  3801. cachedResolvedType = cachedResolvedType->GetUnderlyingType();
  3802. if (cachedResolvedType != NULL)
  3803. ctx->SetCachedResolvedType(typeRef, cachedResolvedType);
  3804. }
  3805. }
  3806. if (cachedResolvedType == NULL)
  3807. {
  3808. ctx->mFailed = true;
  3809. return 0;
  3810. }
  3811. int hashVal = Hash(cachedResolvedType, ctx, flags, hashSeed);
  3812. hashSeed = 0;
  3813. return hashVal;
  3814. }
  3815. else if (auto constExprTypeRef = BfNodeDynCastExact<BfConstExprTypeRef>(typeRef))
  3816. {
  3817. if ((flags & BfHashFlag_AllowGenericParamConstValue) == 0)
  3818. {
  3819. ctx->mModule->Fail("Invalid use of const expression", constExprTypeRef->mConstToken);
  3820. ctx->mFailed = true;
  3821. return 0;
  3822. }
  3823. BfVariant result;
  3824. BfType* resultType = NULL;
  3825. if (constExprTypeRef->mConstExpr != NULL)
  3826. {
  3827. result = EvaluateToVariant(ctx, constExprTypeRef->mConstExpr, resultType);
  3828. if ((resultType != NULL) && (resultType->IsGenericParam()))
  3829. {
  3830. int hashVal = Hash(resultType, ctx, BfHashFlag_None, hashSeed);
  3831. hashSeed = 0;
  3832. return hashVal;
  3833. }
  3834. }
  3835. if (resultType == NULL)
  3836. {
  3837. ctx->mFailed = true;
  3838. return 0;
  3839. }
  3840. int32 dataHash = BeefHash<BfVariant>()(result);
  3841. auto hashVal = ((int)result.mTypeCode << 17) ^ (dataHash << 3) ^ HASH_CONSTTYPE;
  3842. hashVal = ((hashVal ^ (Hash(resultType, ctx, BfHashFlag_AllowRef, hashSeed))) << 5) - hashVal;
  3843. return hashVal;
  3844. }
  3845. else if (auto dotTypeRef = BfNodeDynCastExact<BfDotTypeReference>(typeRef))
  3846. {
  3847. ctx->mModule->ResolveTypeRef(dotTypeRef, BfPopulateType_Identity, GetResolveFlags(dotTypeRef, ctx, flags));
  3848. ctx->mFailed = true;
  3849. return 0;
  3850. }
  3851. else if (auto wildcardTypeRef = BfNodeDynCastExact<BfWildcardTypeReference>(typeRef))
  3852. {
  3853. ctx->mFailed = true;
  3854. return 0;
  3855. }
  3856. else if (auto tagTypeRef = BfNodeDynCastExact<BfTagTypeRef>(typeRef))
  3857. {
  3858. int nameHash = 0;
  3859. if (tagTypeRef->mNameNode != NULL)
  3860. {
  3861. auto fieldName = tagTypeRef->mNameNode;
  3862. const char* nameStr = fieldName->GetSourceData()->mSrc + fieldName->GetSrcStart();
  3863. nameHash = (int)Hash64(nameStr, fieldName->GetSrcLength());
  3864. }
  3865. return nameHash ^ HASH_TAG;
  3866. }
  3867. // else if (auto inlineTypeRef = BfNodeDynCastExact<BfInlineTypeReference>(typeRef))
  3868. // {
  3869. // String name;
  3870. // inlineTypeRef->mTypeDeclaration->GetAnonymousName(name);
  3871. // int nameHash = (int)Hash64(name.c_str(), (int)name.length());
  3872. // return nameHash ^ HASH_TAG;
  3873. // }
  3874. else
  3875. {
  3876. BF_FATAL("Not handled");
  3877. }
  3878. return 0;
  3879. }
  3880. int BfResolvedTypeSet::Hash(BfTypeReference* typeRef, LookupContext* ctx, BfHashFlags flags, int hashSeed)
  3881. {
  3882. int hashVal = DoHash(typeRef, ctx, flags, hashSeed);
  3883. if (hashSeed == 0)
  3884. return hashVal;
  3885. return HASH_MIX(hashVal, hashSeed);
  3886. }
  3887. int BfResolvedTypeSet::Hash(BfAstNode* typeRefNode, LookupContext* ctx, BfHashFlags flags, int hashSeed)
  3888. {
  3889. if (auto typeRef = BfNodeDynCast<BfTypeReference>(typeRefNode))
  3890. return Hash(typeRef, ctx, flags, hashSeed);
  3891. auto result = ctx->mModule->ResolveTypeRef(typeRefNode, NULL, BfPopulateType_Identity, (BfResolveTypeRefFlags)(ctx->mResolveFlags | BfResolveTypeRefFlag_AllowImplicitConstExpr));
  3892. if (result == NULL)
  3893. {
  3894. ctx->mFailed = true;
  3895. return 0;
  3896. }
  3897. ctx->mResolvedTypeMap[typeRefNode] = result;
  3898. return Hash(result, ctx, false, hashSeed);
  3899. }
  3900. bool BfResolvedTypeSet::Equals(BfType* lhs, BfType* rhs, LookupContext* ctx)
  3901. {
  3902. if (lhs->IsBoxed())
  3903. {
  3904. if (!rhs->IsBoxed())
  3905. return false;
  3906. BfBoxedType* lhsBoxedType = (BfBoxedType*)lhs;
  3907. BfBoxedType* rhsBoxedType = (BfBoxedType*)rhs;
  3908. if (lhsBoxedType->mBoxedFlags != rhsBoxedType->mBoxedFlags)
  3909. return false;
  3910. return lhsBoxedType->mElementType == rhsBoxedType->mElementType;
  3911. }
  3912. else if (lhs->IsArray())
  3913. {
  3914. if (!rhs->IsArray())
  3915. return false;
  3916. BfArrayType* lhsArrayType = (BfArrayType*) lhs;
  3917. BfArrayType* rhsArrayType = (BfArrayType*) rhs;
  3918. if (lhsArrayType->mDimensions != rhsArrayType->mDimensions)
  3919. return false;
  3920. return lhsArrayType->mGenericTypeInfo->mTypeGenericArguments[0] == rhsArrayType->mGenericTypeInfo->mTypeGenericArguments[0];
  3921. }
  3922. else if (lhs->IsTypeInstance())
  3923. {
  3924. if ((!rhs->IsTypeInstance()) || (rhs->IsBoxed()))
  3925. return false;
  3926. BfTypeInstance* lhsInst = (BfTypeInstance*)lhs;
  3927. BfTypeInstance* rhsInst = (BfTypeInstance*)rhs;
  3928. if (lhs->IsClosure())
  3929. {
  3930. if (!rhs->IsClosure())
  3931. return false;
  3932. auto lhsClosure = (BfClosureType*)lhs;
  3933. auto rhsClosure = (BfClosureType*)rhs;
  3934. if ((lhsClosure->mIsUnique) || (rhsClosure->mIsUnique))
  3935. return false;
  3936. if (lhsClosure->mBaseType != rhsClosure->mBaseType)
  3937. return false;
  3938. return lhsClosure->mClosureHash == rhsClosure->mClosureHash;
  3939. }
  3940. if (lhs->IsDelegateFromTypeRef() || lhs->IsFunctionFromTypeRef())
  3941. {
  3942. if (!rhs->IsDelegateFromTypeRef() && !rhs->IsFunctionFromTypeRef())
  3943. return false;
  3944. if (lhs->IsDelegate() != rhs->IsDelegate())
  3945. return false;
  3946. BfDelegateInfo* lhsDelegateInfo = lhs->GetDelegateInfo();
  3947. BfDelegateInfo* rhsDelegateInfo = rhs->GetDelegateInfo();
  3948. if (lhsInst->mTypeDef->mIsDelegate != rhsInst->mTypeDef->mIsDelegate)
  3949. return false;
  3950. if (lhsDelegateInfo->mCallingConvention != rhsDelegateInfo->mCallingConvention)
  3951. return false;
  3952. if (lhsDelegateInfo->mHasParams != rhsDelegateInfo->mHasParams)
  3953. return false;
  3954. if (lhsDelegateInfo->mHasVarArgs != rhsDelegateInfo->mHasVarArgs)
  3955. return false;
  3956. if (lhsDelegateInfo->mHasExplicitThis != rhsDelegateInfo->mHasExplicitThis)
  3957. return false;
  3958. auto lhsMethodDef = lhsInst->mTypeDef->mMethods[0];
  3959. auto rhsMethodDef = rhsInst->mTypeDef->mMethods[0];
  3960. if (lhsMethodDef->mCallingConvention != rhsMethodDef->mCallingConvention)
  3961. return false;
  3962. if (lhsMethodDef->mIsMutating != rhsMethodDef->mIsMutating)
  3963. return false;
  3964. if (lhsDelegateInfo->mReturnType != rhsDelegateInfo->mReturnType)
  3965. return false;
  3966. if (lhsDelegateInfo->mParams.size() != rhsDelegateInfo->mParams.size())
  3967. return false;
  3968. for (int paramIdx = 0; paramIdx < lhsDelegateInfo->mParams.size(); paramIdx++)
  3969. {
  3970. if (lhsDelegateInfo->mParams[paramIdx] != rhsDelegateInfo->mParams[paramIdx])
  3971. return false;
  3972. if (lhsMethodDef->mParams[paramIdx]->mName != rhsMethodDef->mParams[paramIdx]->mName)
  3973. return false;
  3974. }
  3975. return true;
  3976. }
  3977. if (lhs->IsTuple())
  3978. {
  3979. if (!rhs->IsTuple())
  3980. return false;
  3981. BfTypeInstance* lhsTupleType = (BfTypeInstance*)lhs;
  3982. BfTypeInstance* rhsTupleType = (BfTypeInstance*)rhs;
  3983. if (lhsTupleType->mFieldInstances.size() != rhsTupleType->mFieldInstances.size())
  3984. return false;
  3985. for (int fieldIdx = 0; fieldIdx < (int)lhsTupleType->mFieldInstances.size(); fieldIdx++)
  3986. {
  3987. auto lhsFieldInstance = &lhsTupleType->mFieldInstances[fieldIdx];
  3988. auto rhsFieldInstance = &rhsTupleType->mFieldInstances[fieldIdx];
  3989. if (lhsFieldInstance->mResolvedType != rhsFieldInstance->mResolvedType)
  3990. return false;
  3991. auto lhsFieldDef = lhsFieldInstance->GetFieldDef();
  3992. if (rhsTupleType->mTypeDef == NULL)
  3993. {
  3994. char c = lhsFieldDef->mName[0];
  3995. if ((c < '0') || (c > '9'))
  3996. return false;
  3997. }
  3998. else
  3999. {
  4000. auto rhsFieldDef = rhsFieldInstance->GetFieldDef();
  4001. if (lhsFieldDef->mName != rhsFieldDef->mName)
  4002. return false;
  4003. }
  4004. }
  4005. return true;
  4006. }
  4007. if (lhs->IsGenericTypeInstance())
  4008. {
  4009. if (!rhs->IsGenericTypeInstance())
  4010. return false;
  4011. BfTypeInstance* lhsGenericType = (BfTypeInstance*)lhs;
  4012. BfTypeInstance* rhsGenericType = (BfTypeInstance*)rhs;
  4013. if (lhsGenericType->mGenericTypeInfo->mTypeGenericArguments.size() != rhsGenericType->mGenericTypeInfo->mTypeGenericArguments.size())
  4014. return false;
  4015. if (lhsGenericType->mTypeDef->GetDefinition() != rhsGenericType->mTypeDef->GetDefinition())
  4016. return false;
  4017. for (int i = 0; i < (int)lhsGenericType->mGenericTypeInfo->mTypeGenericArguments.size(); i++)
  4018. {
  4019. if (lhsGenericType->mGenericTypeInfo->mTypeGenericArguments[i] != rhsGenericType->mGenericTypeInfo->mTypeGenericArguments[i])
  4020. return false;
  4021. }
  4022. }
  4023. return lhsInst->mTypeDef->GetDefinition() == rhsInst->mTypeDef->GetDefinition();
  4024. }
  4025. else if (lhs->IsPrimitiveType())
  4026. {
  4027. if (!rhs->IsPrimitiveType())
  4028. return false;
  4029. BfPrimitiveType* lhsPrimType = (BfPrimitiveType*)lhs;
  4030. BfPrimitiveType* rhsPrimType = (BfPrimitiveType*)rhs;
  4031. return lhsPrimType->mTypeDef == rhsPrimType->mTypeDef;
  4032. }
  4033. else if (lhs->IsPointer())
  4034. {
  4035. if (!rhs->IsPointer())
  4036. return false;
  4037. BfPointerType* lhsPtrType = (BfPointerType*)lhs;
  4038. BfPointerType* rhsPtrType = (BfPointerType*)rhs;
  4039. return lhsPtrType->mElementType == rhsPtrType->mElementType;
  4040. }
  4041. else if (lhs->IsGenericParam())
  4042. {
  4043. if (!rhs->IsGenericParam())
  4044. return false;
  4045. BfGenericParamType* lhsGenericParamType = (BfGenericParamType*)lhs;
  4046. BfGenericParamType* rhsGenericParamType = (BfGenericParamType*)rhs;
  4047. return (lhsGenericParamType->mGenericParamKind == rhsGenericParamType->mGenericParamKind) &&
  4048. (lhsGenericParamType->mGenericParamIdx == rhsGenericParamType->mGenericParamIdx);
  4049. }
  4050. else if (lhs->IsRef())
  4051. {
  4052. if (!rhs->IsRef())
  4053. return false;
  4054. BfRefType* lhsRefType = (BfRefType*)lhs;
  4055. BfRefType* rhsRefType = (BfRefType*)rhs;
  4056. return (lhsRefType->mElementType == rhsRefType->mElementType) && (lhsRefType->mRefKind == rhsRefType->mRefKind);
  4057. }
  4058. else if (lhs->IsModifiedTypeType())
  4059. {
  4060. if (!rhs->IsModifiedTypeType())
  4061. return false;
  4062. BfModifiedTypeType* lhsRetTypeType = (BfModifiedTypeType*)lhs;
  4063. BfModifiedTypeType* rhsRetTypeType = (BfModifiedTypeType*)rhs;
  4064. return (lhsRetTypeType->mModifiedKind == rhsRetTypeType->mModifiedKind) &&
  4065. (lhsRetTypeType->mElementType == rhsRetTypeType->mElementType);
  4066. }
  4067. else if (lhs->IsConcreteInterfaceType())
  4068. {
  4069. if (!rhs->IsConcreteInterfaceType())
  4070. return false;
  4071. BfConcreteInterfaceType* lhsConcreteInterfaceType = (BfConcreteInterfaceType*)lhs;
  4072. BfConcreteInterfaceType* rhsConcreteInterfaceType = (BfConcreteInterfaceType*)rhs;
  4073. return (lhsConcreteInterfaceType->mInterface == rhsConcreteInterfaceType->mInterface);
  4074. }
  4075. else if (lhs->IsSizedArray())
  4076. {
  4077. if (!rhs->IsSizedArray())
  4078. return false;
  4079. BfSizedArrayType* lhsSizedArrayType = (BfSizedArrayType*)lhs;
  4080. BfSizedArrayType* rhsSizedArrayType = (BfSizedArrayType*)rhs;
  4081. return (lhsSizedArrayType->mElementType == rhsSizedArrayType->mElementType) &&
  4082. (lhsSizedArrayType->mElementCount == rhsSizedArrayType->mElementCount);
  4083. }
  4084. else if (lhs->IsMethodRef())
  4085. {
  4086. if (!rhs->IsMethodRef())
  4087. return false;
  4088. BfMethodRefType* lhsMethodRefType = (BfMethodRefType*)lhs;
  4089. BfMethodRefType* rhsMethodRefType = (BfMethodRefType*)rhs;
  4090. return (lhsMethodRefType->mMethodRef == rhsMethodRefType->mMethodRef) &&
  4091. (lhsMethodRefType->mOwner == rhsMethodRefType->mOwner) &&
  4092. (lhsMethodRefType->mOwnerRevision == rhsMethodRefType->mOwnerRevision);
  4093. }
  4094. else if ((lhs->IsConstExprValue()) || (rhs->IsConstExprValue()))
  4095. {
  4096. if (!lhs->IsConstExprValue() || !rhs->IsConstExprValue())
  4097. return false;
  4098. BfConstExprValueType* lhsConstExprValueType = (BfConstExprValueType*)lhs;
  4099. BfConstExprValueType* rhsConstExprValueType = (BfConstExprValueType*)rhs;
  4100. return (lhsConstExprValueType->mType == rhsConstExprValueType->mType) &&
  4101. (lhsConstExprValueType->mValue == rhsConstExprValueType->mValue);
  4102. }
  4103. else
  4104. {
  4105. BF_FATAL("Not handled");
  4106. }
  4107. return 0;
  4108. }
  4109. bool BfResolvedTypeSet::GenericTypeEquals(BfTypeInstance* lhsGenericType, BfTypeVector* lhsTypeGenericArguments, BfTypeReference* rhs, LookupContext* ctx, int& genericParamOffset, bool skipElement)
  4110. {
  4111. //BP_ZONE("BfResolvedTypeSet::GenericTypeEquals");
  4112. if (!skipElement)
  4113. {
  4114. if (auto elementedTypeRef = BfNodeDynCast<BfElementedTypeRef>(rhs))
  4115. {
  4116. if (!GenericTypeEquals(lhsGenericType, lhsTypeGenericArguments, elementedTypeRef->mElementType, ctx, genericParamOffset))
  4117. return false;
  4118. //_GetTypeRefs(elementedTypeRef->mElementType);
  4119. }
  4120. else if (auto qualifiedTypeRef = BfNodeDynCastExact<BfQualifiedTypeReference>(rhs))
  4121. {
  4122. //_GetTypeRefs(qualifiedTypeRef->mLeft);
  4123. if (!GenericTypeEquals(lhsGenericType, lhsTypeGenericArguments, qualifiedTypeRef->mLeft, ctx, genericParamOffset))
  4124. return false;
  4125. }
  4126. }
  4127. if (auto genericTypeRef = BfNodeDynCastExact<BfGenericInstanceTypeRef>(rhs))
  4128. {
  4129. int rhsGenericArgCount = BF_MAX(genericTypeRef->mGenericArguments.mSize, genericTypeRef->mCommas.mSize + 1);
  4130. if (genericTypeRef->mGenericArguments.size() > rhsGenericArgCount + genericParamOffset)
  4131. return false;
  4132. for (int genericIdx = 0; genericIdx < BF_MAX(genericTypeRef->mGenericArguments.mSize, genericTypeRef->mCommas.mSize + 1); genericIdx++)
  4133. {
  4134. BfType* lhsArgType = (*lhsTypeGenericArguments)[genericParamOffset++];
  4135. BfAstNode* genericArgTypeRef = NULL;
  4136. if (genericIdx < genericTypeRef->mGenericArguments.mSize)
  4137. genericArgTypeRef = genericTypeRef->mGenericArguments[genericIdx];
  4138. if ((ctx->mResolveFlags & BfResolveTypeRefFlag_ForceUnboundGeneric) != 0)
  4139. {
  4140. genericArgTypeRef = NULL;
  4141. }
  4142. else if ((ctx->mResolveFlags & BfResolveTypeRefFlag_AllowUnboundGeneric) != 0)
  4143. {
  4144. if (BfNodeIsExact<BfWildcardTypeReference>(genericArgTypeRef))
  4145. genericArgTypeRef = NULL;
  4146. }
  4147. if (genericArgTypeRef == NULL)
  4148. {
  4149. if (lhsArgType->IsGenericParam())
  4150. {
  4151. auto lhsGenericArgType = (BfGenericParamType*)lhsArgType;
  4152. if ((lhsGenericArgType->mGenericParamKind != BfGenericParamKind_Type) || (lhsGenericArgType->mGenericParamIdx != genericIdx))
  4153. return false;
  4154. }
  4155. else
  4156. return false;
  4157. }
  4158. else
  4159. {
  4160. if (!Equals(lhsArgType, genericArgTypeRef, ctx))
  4161. return false;
  4162. }
  4163. }
  4164. }
  4165. return true;
  4166. }
  4167. bool BfResolvedTypeSet::GenericTypeEquals(BfTypeInstance* lhsGenericType, BfTypeVector* lhsTypeGenericArguments, BfTypeReference* rhs, BfTypeDef* rhsTypeDef, LookupContext* ctx)
  4168. {
  4169. int genericParamOffset = 0;
  4170. bool isFullyQualified = false;
  4171. BfTypeInstance* outerType = NULL;
  4172. if (auto genericInstTypeRef = BfNodeDynCast<BfGenericInstanceTypeRef>(rhs))
  4173. {
  4174. int checkIdx = 0;
  4175. auto checkTypeRef = genericInstTypeRef->mElementType;
  4176. while (checkTypeRef != NULL)
  4177. {
  4178. checkIdx++;
  4179. if (checkIdx >= 2)
  4180. {
  4181. isFullyQualified = true;
  4182. BfType* checkType = ctx->GetCachedResolvedType(checkTypeRef);
  4183. if (checkType != NULL)
  4184. outerType = checkType->ToTypeInstance();
  4185. if (outerType != NULL)
  4186. {
  4187. BfTypeInstance* lhsCheckType = lhsGenericType;
  4188. while (lhsCheckType->mTypeDef->mNestDepth > outerType->mTypeDef->mNestDepth)
  4189. {
  4190. lhsCheckType = ctx->mModule->GetOuterType(lhsCheckType);
  4191. }
  4192. if (lhsCheckType != outerType)
  4193. return false;
  4194. if (outerType->mGenericTypeInfo != NULL)
  4195. genericParamOffset = (int)outerType->mGenericTypeInfo->mTypeGenericArguments.mSize;
  4196. }
  4197. break;
  4198. }
  4199. if (auto elementedTypeRef = BfNodeDynCast<BfElementedTypeRef>(checkTypeRef))
  4200. {
  4201. checkTypeRef = elementedTypeRef->mElementType;
  4202. continue;
  4203. }
  4204. if (auto qualifiedTypeRef = BfNodeDynCast<BfQualifiedTypeReference>(checkTypeRef))
  4205. {
  4206. checkTypeRef = qualifiedTypeRef->mLeft;
  4207. continue;
  4208. }
  4209. break;
  4210. }
  4211. }
  4212. BfTypeInstance* rootOuterTypeInstance = NULL;
  4213. auto rhsGenericTypeInstRef = BfNodeDynCastExact<BfGenericInstanceTypeRef>(rhs);
  4214. if (!isFullyQualified)
  4215. {
  4216. rootOuterTypeInstance = ctx->mModule->mCurTypeInstance;
  4217. if ((rhsTypeDef == ctx->mRootTypeDef) && (ctx->mRootOuterTypeInstance != NULL))
  4218. rootOuterTypeInstance = ctx->mRootOuterTypeInstance;
  4219. if (rhsGenericTypeInstRef == NULL)
  4220. {
  4221. if (auto rhsNullableTypeRef = BfNodeDynCastExact<BfNullableTypeRef>(rhs))
  4222. {
  4223. if (rhsNullableTypeRef != NULL)
  4224. {
  4225. if (lhsGenericType->mTypeDef != ctx->mModule->mContext->mCompiler->mNullableTypeDef)
  4226. return false;
  4227. auto rhsElemType = ctx->mModule->ResolveTypeRef(rhsNullableTypeRef->mElementType, BfPopulateType_Identity, ctx->mResolveFlags);
  4228. return lhsGenericType->mGenericTypeInfo->mTypeGenericArguments[0] == rhsElemType;
  4229. }
  4230. }
  4231. if ((rhsTypeDef != NULL) && (rootOuterTypeInstance != NULL))
  4232. {
  4233. // See if we're referring to an non-generic inner type where the outer type is generic
  4234. if (lhsGenericType->mTypeDef->GetDefinition() != rhsTypeDef->GetDefinition())
  4235. return false;
  4236. BfTypeDef* commonOuterType = ctx->mModule->FindCommonOuterType(rootOuterTypeInstance->mTypeDef, rhsTypeDef->mOuterType);
  4237. if (commonOuterType != NULL)
  4238. {
  4239. BfTypeInstance* checkTypeInstance = rootOuterTypeInstance;
  4240. if (checkTypeInstance->IsBoxed())
  4241. checkTypeInstance = checkTypeInstance->GetUnderlyingType()->ToTypeInstance();
  4242. BF_ASSERT(checkTypeInstance->IsGenericTypeInstance());
  4243. int numParentGenericParams = (int)commonOuterType->mGenericParamDefs.size();
  4244. auto curTypeInstance = (BfTypeInstance*)checkTypeInstance;
  4245. if (lhsGenericType->mGenericTypeInfo->mTypeGenericArguments.size() != numParentGenericParams)
  4246. return false;
  4247. for (int i = 0; i < (int)numParentGenericParams; i++)
  4248. if ((*lhsTypeGenericArguments)[i] != curTypeInstance->mGenericTypeInfo->mTypeGenericArguments[i])
  4249. return false;
  4250. return true;
  4251. }
  4252. }
  4253. if (auto rhsQualifiedTypeRef = BfNodeDynCastExact<BfQualifiedTypeReference>(rhs))
  4254. {
  4255. auto rhsRightType = ctx->mModule->ResolveTypeRef(rhs, BfPopulateType_Identity, ctx->mResolveFlags);
  4256. return rhsRightType == lhsGenericType;
  4257. }
  4258. return false;
  4259. }
  4260. }
  4261. if (rhsGenericTypeInstRef == NULL)
  4262. return true;
  4263. BfTypeDef* elementTypeDef = ctx->mModule->ResolveGenericInstanceDef(rhsGenericTypeInstRef, NULL, ctx->mResolveFlags);
  4264. if (elementTypeDef == NULL)
  4265. return false;
  4266. if (elementTypeDef->GetDefinition() != lhsGenericType->mTypeDef->GetDefinition())
  4267. return false;
  4268. // Do we need to add generic arguments from an in-context outer class?
  4269. if ((elementTypeDef->mOuterType != NULL) && (rootOuterTypeInstance != NULL) && (rootOuterTypeInstance->IsGenericTypeInstance()))
  4270. {
  4271. BfTypeDef* commonOuterType = ctx->mModule->FindCommonOuterType(rootOuterTypeInstance->mTypeDef, elementTypeDef->mOuterType);
  4272. if (commonOuterType != NULL)
  4273. {
  4274. auto parentTypeInstance = rootOuterTypeInstance;
  4275. genericParamOffset = (int) commonOuterType->mGenericParamDefs.size();
  4276. for (int i = 0; i < genericParamOffset; i++)
  4277. for (auto genericArg : parentTypeInstance->mGenericTypeInfo->mTypeGenericArguments)
  4278. {
  4279. if (parentTypeInstance->mGenericTypeInfo->mTypeGenericArguments[i] != (*lhsTypeGenericArguments)[i])
  4280. return false;
  4281. }
  4282. }
  4283. }
  4284. if (!GenericTypeEquals(lhsGenericType, lhsTypeGenericArguments, rhs, ctx, genericParamOffset, isFullyQualified))
  4285. return false;
  4286. return genericParamOffset == (int)lhsTypeGenericArguments->size();
  4287. }
  4288. BfType* BfResolvedTypeSet::LookupContext::GetCachedResolvedType(BfTypeReference* typeReference)
  4289. {
  4290. if (typeReference == mRootTypeRef)
  4291. return mRootResolvedType;
  4292. BfType** typePtr = NULL;
  4293. if (mResolvedTypeMap.TryGetValue(typeReference, &typePtr))
  4294. return *typePtr;
  4295. return NULL;
  4296. }
  4297. void BfResolvedTypeSet::LookupContext::SetCachedResolvedType(BfTypeReference* typeReference, BfType* type)
  4298. {
  4299. if (typeReference == mRootTypeRef)
  4300. mRootResolvedType = type;
  4301. else
  4302. mResolvedTypeMap[typeReference] = type;
  4303. }
  4304. BfType* BfResolvedTypeSet::LookupContext::ResolveTypeRef(BfTypeReference* typeReference)
  4305. {
  4306. return mModule->ResolveTypeRef(typeReference, BfPopulateType_Identity, BfResolveTypeRefFlag_AllowGenericParamConstValue);
  4307. }
  4308. BfTypeDef* BfResolvedTypeSet::LookupContext::ResolveToTypeDef(BfTypeReference* typeReference, BfType** outType)
  4309. {
  4310. if (outType != NULL)
  4311. *outType = NULL;
  4312. if (typeReference == mRootTypeRef)
  4313. return mRootTypeDef;
  4314. if (auto typeDefTypeRef = BfNodeDynCast<BfDirectTypeDefReference>(typeReference))
  4315. {
  4316. return typeDefTypeRef->mTypeDef;
  4317. }
  4318. auto type = mModule->ResolveTypeRef(typeReference, BfPopulateType_Identity, BfResolveTypeRefFlag_AllowGenericParamConstValue);
  4319. if (type == NULL)
  4320. return NULL;
  4321. if (outType != NULL)
  4322. *outType = type;
  4323. if (type->IsPrimitiveType())
  4324. return ((BfPrimitiveType*)type)->mTypeDef;
  4325. auto typeInst = type->ToTypeInstance();
  4326. if (typeInst == NULL)
  4327. return NULL;
  4328. return typeInst->mTypeDef->GetDefinition();
  4329. }
  4330. bool BfResolvedTypeSet::Equals(BfType* lhs, BfTypeReference* rhs, BfTypeDef* rhsTypeDef, LookupContext* ctx)
  4331. {
  4332. auto rhsType = ctx->mModule->ResolveTypeDef(rhsTypeDef, BfPopulateType_Identity);
  4333. if (rhsType == NULL)
  4334. {
  4335. ctx->mFailed = true;
  4336. return false;
  4337. }
  4338. return BfResolvedTypeSet::Equals(lhs, rhsType, ctx);
  4339. }
  4340. bool BfResolvedTypeSet::Equals(BfType* lhs, BfTypeReference* rhs, LookupContext* ctx)
  4341. {
  4342. //BP_ZONE("BfResolvedTypeSet::Equals");
  4343. if (ctx->mRootTypeRef != rhs)
  4344. {
  4345. if (auto retTypeRef = BfNodeDynCastExact<BfModifiedTypeRef>(rhs))
  4346. {
  4347. auto resolvedType = ctx->mModule->ResolveTypeRef(rhs);
  4348. if ((resolvedType != NULL) && (resolvedType->IsRef()))
  4349. resolvedType = resolvedType->GetUnderlyingType();
  4350. return lhs == resolvedType;
  4351. }
  4352. }
  4353. if ((rhs->IsNamedTypeReference()) || (rhs->IsA<BfGenericInstanceTypeRef>()) || (rhs->IsA<BfQualifiedTypeReference>()))
  4354. {
  4355. if ((ctx->mRootTypeRef != rhs) || (ctx->mRootTypeDef == NULL))
  4356. {
  4357. auto rhsResolvedType = ctx->ResolveTypeRef(rhs);
  4358. if (rhsResolvedType == NULL)
  4359. {
  4360. ctx->mFailed = true;
  4361. return false;
  4362. }
  4363. return lhs == rhsResolvedType;
  4364. }
  4365. }
  4366. if (auto declTypeRef = BfNodeDynCastExact<BfExprModTypeRef>(rhs))
  4367. {
  4368. auto cachedResolveType = ctx->GetCachedResolvedType(rhs);
  4369. BF_ASSERT(cachedResolveType != NULL);
  4370. return lhs == cachedResolveType;
  4371. }
  4372. // Strip off 'const' - it's just an error when applied to a typeRef in Beef
  4373. auto constTypeRef = BfNodeDynCastExact<BfConstTypeRef>(rhs);
  4374. if (constTypeRef != NULL)
  4375. return Equals(lhs, constTypeRef->mElementType, ctx);
  4376. if (lhs->IsBoxed())
  4377. {
  4378. return false;
  4379. }
  4380. else if (lhs->IsArray())
  4381. {
  4382. auto rhsArrayTypeRef = BfNodeDynCastExact<BfArrayTypeRef>(rhs);
  4383. if (rhsArrayTypeRef == NULL)
  4384. return false;
  4385. // Any non-comma param means it's a sized array
  4386. for (auto param : rhsArrayTypeRef->mParams)
  4387. {
  4388. bool isComma = false;
  4389. if (auto tokenNode = BfNodeDynCast<BfTokenNode>(param))
  4390. isComma = tokenNode->mToken == BfToken_Comma;
  4391. if (!isComma)
  4392. return false;
  4393. }
  4394. BfArrayType* lhsArrayType = (BfArrayType*) lhs;
  4395. if (lhsArrayType->mDimensions != rhsArrayTypeRef->mDimensions)
  4396. return false;
  4397. return Equals(lhsArrayType->mGenericTypeInfo->mTypeGenericArguments[0], rhsArrayTypeRef->mElementType, ctx);
  4398. }
  4399. else if (lhs->IsDelegateFromTypeRef() || lhs->IsFunctionFromTypeRef())
  4400. {
  4401. auto rhsDelegateType = BfNodeDynCastExact<BfDelegateTypeRef>(rhs);
  4402. if (rhsDelegateType == NULL)
  4403. return false;
  4404. bool wantGeneric = false;
  4405. BfDelegateInfo* lhsDelegateInfo = lhs->GetDelegateInfo();
  4406. auto lhsTypeInstance = lhs->ToTypeInstance();
  4407. BfMethodDef* invokeMethodDef = lhsTypeInstance->mTypeDef->mMethods[0];
  4408. BF_ASSERT(invokeMethodDef->mName == "Invoke");
  4409. bool rhsIsDelegate = rhsDelegateType->mTypeToken->GetToken() == BfToken_Delegate;
  4410. if ((lhs->IsDelegate()) != rhsIsDelegate)
  4411. return false;
  4412. auto _CheckType = [&](BfType* type)
  4413. {
  4414. if (type->IsTypeGenericParam())
  4415. wantGeneric = true;
  4416. };
  4417. BfCallingConvention rhsCallingConvention = BfCallingConvention_Unspecified;
  4418. if (ctx->mRootTypeRef == rhsDelegateType)
  4419. rhsCallingConvention = ctx->mCallingConvention;
  4420. else
  4421. ctx->mModule->GetDelegateTypeRefAttributes(rhsDelegateType, rhsCallingConvention);
  4422. if (lhsDelegateInfo->mCallingConvention != rhsCallingConvention)
  4423. return false;
  4424. if (!Equals(lhsDelegateInfo->mReturnType, rhsDelegateType->mReturnType, ctx))
  4425. return false;
  4426. _CheckType(lhsDelegateInfo->mReturnType);
  4427. bool isMutating = true;
  4428. int paramRefOfs = 0;
  4429. if ((!rhsDelegateType->mParams.IsEmpty()) && (lhs->IsFunction()))
  4430. {
  4431. auto param0 = rhsDelegateType->mParams[0];
  4432. if ((param0->mNameNode != NULL) && (param0->mNameNode->Equals("this")))
  4433. {
  4434. if (!lhsDelegateInfo->mHasExplicitThis)
  4435. return false;
  4436. bool handled = false;
  4437. auto lhsThisType = lhsDelegateInfo->mParams[0];
  4438. bool wantsMutating = false;
  4439. if (auto refTypeRef = BfNodeDynCast<BfRefTypeRef>(param0->mTypeRef))
  4440. {
  4441. // This catches `ref Foo*` cases (which generate warnings)
  4442. if ((refTypeRef->mRefToken != NULL) && (refTypeRef->mRefToken->mToken == BfToken_Mut))
  4443. wantsMutating = true;
  4444. }
  4445. auto rhsThisType = ctx->mModule->ResolveTypeRef(param0->mTypeRef, BfPopulateType_Identity, (BfResolveTypeRefFlags)(BfResolveTypeRefFlag_NoWarnOnMut | BfResolveTypeRefFlag_AllowRef));
  4446. if (rhsThisType == NULL)
  4447. return false;
  4448. if (rhsThisType->IsRef())
  4449. {
  4450. rhsThisType = rhsThisType->GetUnderlyingType();
  4451. if (rhsThisType->IsPointer())
  4452. rhsThisType = rhsThisType->GetUnderlyingType();
  4453. if (lhsThisType != rhsThisType)
  4454. return false;
  4455. wantsMutating = (lhsThisType->IsValueType()) || (lhsThisType->IsGenericParam());
  4456. }
  4457. else
  4458. {
  4459. if (rhsThisType->IsPointer())
  4460. rhsThisType = rhsThisType->GetUnderlyingType();
  4461. if (lhsThisType != rhsThisType)
  4462. return false;
  4463. }
  4464. if (invokeMethodDef->mIsMutating != wantsMutating)
  4465. return false;
  4466. paramRefOfs = 1;
  4467. }
  4468. }
  4469. int lhsParamsCount = (int)lhsDelegateInfo->mParams.size();
  4470. if (lhsDelegateInfo->mHasVarArgs)
  4471. lhsParamsCount++;
  4472. if (lhsParamsCount != (int)rhsDelegateType->mParams.size())
  4473. return false;
  4474. bool rhsHadParams = false;
  4475. for (int paramIdx = paramRefOfs; paramIdx < lhsDelegateInfo->mParams.size(); paramIdx++)
  4476. {
  4477. auto paramTypeRef = rhsDelegateType->mParams[paramIdx]->mTypeRef;
  4478. if (!Equals(lhsDelegateInfo->mParams[paramIdx], paramTypeRef, ctx))
  4479. return false;
  4480. _CheckType(lhsDelegateInfo->mParams[paramIdx]);
  4481. StringView rhsParamName;
  4482. if (rhsDelegateType->mParams[paramIdx]->mNameNode != NULL)
  4483. rhsParamName = rhsDelegateType->mParams[paramIdx]->mNameNode->ToStringView();
  4484. if (invokeMethodDef->mParams[paramIdx]->mName != rhsParamName)
  4485. return false;
  4486. if ((rhsDelegateType->mParams[paramIdx]->mModToken != NULL) && (rhsDelegateType->mParams[paramIdx]->mModToken->mToken == BfToken_Params))
  4487. rhsHadParams = true;
  4488. }
  4489. if (rhsHadParams != lhsDelegateInfo->mHasParams)
  4490. return false;
  4491. if ((ctx->mModule->mCurTypeInstance == NULL) || (!ctx->mModule->mCurTypeInstance->IsGenericTypeInstance()))
  4492. wantGeneric = false;
  4493. //TODO:
  4494. wantGeneric = false;
  4495. if (wantGeneric != lhsTypeInstance->IsGenericTypeInstance())
  4496. return false;
  4497. return true;
  4498. }
  4499. else if ((lhs->IsEnum()) && (lhs->IsOnDemand()))
  4500. {
  4501. BfTypeInstance* typeInstance = lhs->ToTypeInstance();
  4502. auto fieldName = typeInstance->mTypeDef->mFields[0]->mName;
  4503. auto tagTypeRef = BfNodeDynCastExact<BfTagTypeRef>(rhs);
  4504. if (tagTypeRef == NULL)
  4505. return false;
  4506. return tagTypeRef->mNameNode->Equals(fieldName);
  4507. }
  4508. else if (lhs->IsTypeInstance())
  4509. {
  4510. BfTypeInstance* lhsInst = (BfTypeInstance*) lhs;
  4511. if (lhs->IsTuple())
  4512. {
  4513. if (!rhs->IsA<BfTupleTypeRef>())
  4514. return false;
  4515. BfTupleTypeRef* rhsTupleTypeRef = (BfTupleTypeRef*)rhs;
  4516. BfTypeInstance* lhsTupleType = (BfTypeInstance*)lhs;
  4517. if (lhsTupleType->mFieldInstances.size() != rhsTupleTypeRef->mFieldTypes.size())
  4518. return false;
  4519. for (int fieldIdx = 0; fieldIdx < (int)lhsTupleType->mFieldInstances.size(); fieldIdx++)
  4520. {
  4521. BfFieldInstance* fieldInstance = &lhsTupleType->mFieldInstances[fieldIdx];
  4522. auto rhsResolvedType = ctx->mModule->ResolveTypeRef(rhsTupleTypeRef->mFieldTypes[fieldIdx], BfPopulateType_Identity);
  4523. if (rhsResolvedType != fieldInstance->mResolvedType)
  4524. return false;
  4525. BfFieldDef* fieldTypeDef = fieldInstance->GetFieldDef();
  4526. BfIdentifierNode* fieldName = NULL;
  4527. if (fieldIdx < (int)rhsTupleTypeRef->mFieldNames.size())
  4528. fieldName = rhsTupleTypeRef->mFieldNames[fieldIdx];
  4529. if (fieldName != NULL)
  4530. {
  4531. if (fieldName->ToString() != fieldTypeDef->mName)
  4532. return false;
  4533. }
  4534. else
  4535. {
  4536. char nameStr[64];
  4537. sprintf(nameStr, "%d", fieldIdx);
  4538. if (fieldTypeDef->mName != nameStr)
  4539. return false;
  4540. }
  4541. }
  4542. return true;
  4543. }
  4544. else if (lhs->IsGenericTypeInstance())
  4545. {
  4546. BfType* rhsType = NULL;
  4547. auto rhsTypeDef = ctx->ResolveToTypeDef(rhs, &rhsType);
  4548. if (rhsTypeDef == NULL)
  4549. return false;
  4550. if (rhsType != NULL)
  4551. return lhs == rhsType;
  4552. BfTypeInstance* lhsGenericType = (BfTypeInstance*) lhs;
  4553. return GenericTypeEquals(lhsGenericType, &lhsGenericType->mGenericTypeInfo->mTypeGenericArguments, rhs, rhsTypeDef, ctx);
  4554. }
  4555. else
  4556. {
  4557. if (rhs->IsA<BfElementedTypeRef>())
  4558. return false;
  4559. if (!rhs->IsTypeDefTypeReference())
  4560. {
  4561. if (rhs->IsA<BfDelegateTypeRef>())
  4562. return false; // Would have caught it before
  4563. if (rhs->IsA<BfQualifiedTypeReference>())
  4564. {
  4565. //TODO: Under what circumstances was this supposed to be used? This caused an infinite loop comparing
  4566. // 'var' against a delegate type instance
  4567. auto resolvedType = ctx->mModule->ResolveTypeRef(rhs, BfPopulateType_Identity);
  4568. if (resolvedType == lhsInst)
  4569. {
  4570. return true;
  4571. }
  4572. }
  4573. return false;
  4574. }
  4575. auto rhsTypeDef = ctx->ResolveToTypeDef(rhs);
  4576. if (rhsTypeDef == NULL)
  4577. return false;
  4578. return lhsInst->IsInstanceOf(rhsTypeDef);
  4579. }
  4580. }
  4581. else if (lhs->IsPrimitiveType())
  4582. {
  4583. if (lhs->IsDot())
  4584. {
  4585. auto varTypeReference = BfNodeDynCastExact<BfDotTypeReference>(rhs);
  4586. if (varTypeReference != NULL)
  4587. return true;
  4588. }
  4589. if (lhs->IsVar())
  4590. {
  4591. auto varTypeReference = BfNodeDynCastExact<BfVarTypeReference>(rhs);
  4592. if (varTypeReference != NULL)
  4593. return true;
  4594. }
  4595. if (lhs->IsLet())
  4596. {
  4597. auto letTypeReference = BfNodeDynCastExact<BfLetTypeReference>(rhs);
  4598. if (letTypeReference != NULL)
  4599. return true;
  4600. }
  4601. BfPrimitiveType* lhsPrimType = (BfPrimitiveType*)lhs;
  4602. auto rhsTypeDef = ctx->ResolveToTypeDef(rhs);
  4603. // if (rhsTypeDef->mTypeCode == BfTypeCode_TypeAlias)
  4604. // return Equals(lhs, rhs, rhsTypeDef, ctx);
  4605. return lhsPrimType->mTypeDef == rhsTypeDef;
  4606. }
  4607. else if (lhs->IsPointer())
  4608. {
  4609. auto rhsPointerTypeRef = BfNodeDynCastExact<BfPointerTypeRef>(rhs);
  4610. if (rhsPointerTypeRef == NULL)
  4611. return false;
  4612. BfPointerType* lhsPtrType = (BfPointerType*)lhs;
  4613. return Equals(lhsPtrType->mElementType, rhsPointerTypeRef->mElementType, ctx);
  4614. }
  4615. else if (lhs->IsGenericParam())
  4616. {
  4617. auto lhsGenericParamType = (BfGenericParamType*)lhs;
  4618. auto rhsGenericParamTypeRef = BfNodeDynCastExact<BfGenericParamTypeRef>(rhs);
  4619. if (rhsGenericParamTypeRef == NULL)
  4620. {
  4621. if (auto constExprTypeRef = BfNodeDynCastExact<BfConstExprTypeRef>(rhs))
  4622. {
  4623. BfVariant result;
  4624. if (constExprTypeRef->mConstExpr == NULL)
  4625. return false;
  4626. BfType* resultType = NULL;
  4627. result = EvaluateToVariant(ctx, constExprTypeRef->mConstExpr, resultType);
  4628. return resultType == lhs;
  4629. }
  4630. return false;
  4631. }
  4632. return (lhsGenericParamType->mGenericParamKind == rhsGenericParamTypeRef->mGenericParamKind) &&
  4633. (lhsGenericParamType->mGenericParamIdx == rhsGenericParamTypeRef->mGenericParamIdx);
  4634. }
  4635. else if (lhs->IsRef())
  4636. {
  4637. auto lhsRefType = (BfRefType*)lhs;
  4638. auto rhsRefTypeRef = BfNodeDynCastExact<BfRefTypeRef>(rhs);
  4639. if (rhsRefTypeRef == NULL)
  4640. return false;
  4641. auto refKind = BfRefType::RefKind_Ref;
  4642. if (rhsRefTypeRef->mRefToken == NULL)
  4643. refKind = BfRefType::RefKind_Ref;
  4644. else if (rhsRefTypeRef->mRefToken->GetToken() == BfToken_In)
  4645. refKind = BfRefType::RefKind_In;
  4646. else if (rhsRefTypeRef->mRefToken->GetToken() == BfToken_Out)
  4647. refKind = BfRefType::RefKind_Out;
  4648. else if (rhsRefTypeRef->mRefToken->GetToken() == BfToken_Mut)
  4649. refKind = BfRefType::RefKind_Mut;
  4650. return (lhsRefType->mRefKind == refKind) &&
  4651. Equals(lhsRefType->mElementType, rhsRefTypeRef->mElementType, ctx);
  4652. }
  4653. else if (lhs->IsModifiedTypeType())
  4654. {
  4655. auto lhsRetTypeType = (BfModifiedTypeType*)lhs;
  4656. auto rhsRetTypeTypeRef = BfNodeDynCastExact<BfModifiedTypeRef>(rhs);
  4657. if (rhsRetTypeTypeRef == NULL)
  4658. return false;
  4659. if (lhsRetTypeType->mModifiedKind != rhsRetTypeTypeRef->mRetTypeToken->mToken)
  4660. return false;
  4661. return Equals(lhsRetTypeType->mElementType, rhsRetTypeTypeRef->mElementType, ctx);
  4662. }
  4663. else if (lhs->IsConcreteInterfaceType())
  4664. {
  4665. // No way to create a reference to one of these
  4666. return false;
  4667. }
  4668. else if (lhs->IsSizedArray())
  4669. {
  4670. auto rhsArrayTypeRef = BfNodeDynCastExact<BfArrayTypeRef>(rhs);
  4671. if (rhsArrayTypeRef == NULL)
  4672. return false;
  4673. if ((rhsArrayTypeRef->mDimensions != 1) && (rhsArrayTypeRef->mParams.size() != 1))
  4674. return false;
  4675. BfSizedArrayType* lhsArrayType = (BfSizedArrayType*)lhs;
  4676. if (!Equals(lhsArrayType->mElementType, rhsArrayTypeRef->mElementType, ctx))
  4677. return false;
  4678. intptr elementCount = -1;
  4679. BfExpression* sizeExpr = BfNodeDynCast<BfExpression>(rhsArrayTypeRef->mParams[0]);
  4680. BF_ASSERT(sizeExpr != NULL);
  4681. if (sizeExpr != NULL)
  4682. {
  4683. BfType* intType = ctx->mModule->GetPrimitiveType(BfTypeCode_IntPtr);
  4684. BfTypedValue typedVal;
  4685. ctx->mResolvedValueMap.TryGetValue(sizeExpr, &typedVal);
  4686. if (typedVal.mKind == BfTypedValueKind_GenericConstValue)
  4687. {
  4688. if (!lhs->IsUnknownSizedArrayType())
  4689. return false;
  4690. auto lhsUnknownSizedArray = (BfUnknownSizedArrayType*)lhs;
  4691. return lhsUnknownSizedArray->mElementCountSource = typedVal.mType;
  4692. }
  4693. if (typedVal)
  4694. typedVal = ctx->mModule->Cast(sizeExpr, typedVal, intType);
  4695. if (typedVal)
  4696. {
  4697. if (lhs->IsUnknownSizedArrayType())
  4698. return false;
  4699. auto constant = ctx->mModule->mBfIRBuilder->GetConstant(typedVal.mValue);
  4700. if ((constant->mConstType == BfConstType_Undef) || (!BfIRBuilder::IsInt(constant->mTypeCode)))
  4701. {
  4702. elementCount = -1; // Marker for undef
  4703. }
  4704. else
  4705. {
  4706. elementCount = (intptr)constant->mInt64;
  4707. BF_ASSERT(elementCount >= 0); // Should have been caught in hash
  4708. }
  4709. }
  4710. }
  4711. return lhsArrayType->mElementCount == elementCount;
  4712. }
  4713. else if (lhs->IsMethodRef())
  4714. {
  4715. // Always make these unique. The MethodInstance value will change on rebuild anyway
  4716. return false;
  4717. }
  4718. else if (lhs->IsConstExprValue())
  4719. {
  4720. auto constExprTypeRef = BfNodeDynCastExact<BfConstExprTypeRef>(rhs);
  4721. if (constExprTypeRef == NULL)
  4722. return false;
  4723. if (!lhs->IsConstExprValue())
  4724. return false;
  4725. BfConstExprValueType* lhsConstExprType = (BfConstExprValueType*)lhs;
  4726. BfVariant result;
  4727. if (constExprTypeRef->mConstExpr != NULL)
  4728. {
  4729. BfType* resultType = NULL;
  4730. result = EvaluateToVariant(ctx, constExprTypeRef->mConstExpr, resultType);
  4731. if (resultType != lhsConstExprType->mType)
  4732. return false;
  4733. }
  4734. return result == lhsConstExprType->mValue;
  4735. }
  4736. else
  4737. {
  4738. BF_FATAL("Not handled");
  4739. }
  4740. return false;
  4741. }
  4742. bool BfResolvedTypeSet::Equals(BfType* lhs, BfAstNode* rhs, LookupContext* ctx)
  4743. {
  4744. if (auto rhsTypeRef = BfNodeDynCast<BfTypeReference>(rhs))
  4745. return Equals(lhs, rhsTypeRef, ctx);
  4746. BfType* rhsResultType;
  4747. if (ctx->mResolvedTypeMap.TryGetValue(rhs, &rhsResultType))
  4748. return lhs == rhsResultType;
  4749. BF_FATAL("Invalid value in BfResolvedTypeSet::Equals");
  4750. return false;
  4751. }
  4752. void BfResolvedTypeSet::RemoveEntry(BfResolvedTypeSet::EntryRef entry)
  4753. {
  4754. int hashIdx = (entry->mHashCode & 0x7FFFFFFF) % mHashSize;
  4755. // if (entry->mPrev == NULL)
  4756. // {
  4757. // if (entry->mNext != NULL)
  4758. // entry->mNext->mPrev = NULL;
  4759. // BF_ASSERT(mHashHeads[bucket] == entry);
  4760. // mHashHeads[bucket] = entry->mNext;
  4761. // }
  4762. // else
  4763. // {
  4764. // entry->mPrev->mNext = entry->mNext;
  4765. // if (entry->mNext != NULL)
  4766. // entry->mNext->mPrev = entry->mPrev;
  4767. // }
  4768. //
  4769. // mSize--;
  4770. bool found = false;
  4771. int* srcCheckEntryPtr = &this->mHashHeads[hashIdx];
  4772. int checkEntryIdx = *srcCheckEntryPtr;
  4773. while (checkEntryIdx != -1)
  4774. {
  4775. auto checkEntry = &mEntries[checkEntryIdx];
  4776. if (checkEntryIdx == entry.mIndex)
  4777. {
  4778. *srcCheckEntryPtr = checkEntry->mNext;
  4779. found = true;
  4780. }
  4781. srcCheckEntryPtr = &checkEntry->mNext;
  4782. checkEntryIdx = checkEntry->mNext;
  4783. }
  4784. BF_ASSERT(found);
  4785. BF_ASSERT(entry->mValue == NULL);
  4786. FreeIdx(entry.mIndex);
  4787. }
  4788. // BfResolvedTypeSet::Iterator BfResolvedTypeSet::begin()
  4789. // {
  4790. // return ++Iterator(this);
  4791. // }
  4792. //
  4793. // BfResolvedTypeSet::Iterator BfResolvedTypeSet::end()
  4794. // {
  4795. // Iterator itr(this);
  4796. // itr.mCurBucket = HashSize;
  4797. // return itr;
  4798. // }
  4799. //
  4800. // BfResolvedTypeSet::Iterator BfResolvedTypeSet::erase(BfResolvedTypeSet::Iterator& itr)
  4801. // {
  4802. // auto next = itr;
  4803. // ++next;
  4804. //
  4805. // auto cur = itr.mCurEntry;
  4806. //
  4807. // auto& hashHead = itr.mTypeSet->mHashHeads[itr.mCurBucket];
  4808. //
  4809. // if (hashHead == cur)
  4810. // hashHead = cur->mNext;
  4811. // if (cur->mPrev != NULL)
  4812. // cur->mPrev->mNext = cur->mNext;
  4813. // if (cur->mNext != NULL)
  4814. // cur->mNext->mPrev = cur->mPrev;
  4815. // delete cur;
  4816. //
  4817. // //BfLogSys("Deleting node %@ from bucket %d\n", cur, itr.mCurBucket);
  4818. //
  4819. // mSize--;
  4820. // return next;
  4821. // }
  4822. //////////////////////////////////////////////////////////////////////////
  4823. BfHotTypeVersion::~BfHotTypeVersion()
  4824. {
  4825. for (auto member : mMembers)
  4826. member->Deref();
  4827. }
  4828. BfHotTypeData::~BfHotTypeData()
  4829. {
  4830. for (auto version : mTypeVersions)
  4831. {
  4832. version->Deref();
  4833. }
  4834. }
  4835. BfHotTypeVersion* BfHotTypeData::GetTypeVersion(int hotCommitedIdx)
  4836. {
  4837. for (int checkIdx = (int)mTypeVersions.size() - 1; checkIdx >= 0; checkIdx--)
  4838. {
  4839. BfHotTypeVersion* hotTypeVersion = mTypeVersions[checkIdx];
  4840. if (hotTypeVersion->mDeclHotCompileIdx <= hotCommitedIdx)
  4841. return hotTypeVersion;
  4842. }
  4843. return NULL;
  4844. }
  4845. BfHotTypeVersion* BfHotTypeData::GetLatestVersion()
  4846. {
  4847. return mTypeVersions.back();
  4848. }
  4849. BfHotTypeVersion* BfHotTypeData::GetLatestVersionHead()
  4850. {
  4851. auto lastestVersion = mTypeVersions.back();
  4852. for (int versionIdx = (int)mTypeVersions.size() - 1; versionIdx >= 0; versionIdx--)
  4853. {
  4854. auto checkVersion = mTypeVersions[versionIdx];
  4855. if (checkVersion->mDataHash != lastestVersion->mDataHash)
  4856. break;
  4857. lastestVersion = checkVersion;
  4858. }
  4859. return lastestVersion;
  4860. }
  4861. void BfHotTypeData::ClearVersionsAfter(int hotIdx)
  4862. {
  4863. while (!mTypeVersions.IsEmpty())
  4864. {
  4865. auto hotTypeVersion = mTypeVersions.back();
  4866. if (hotTypeVersion->mDeclHotCompileIdx > hotIdx)
  4867. {
  4868. hotTypeVersion->Deref();
  4869. mTypeVersions.pop_back();
  4870. }
  4871. else
  4872. break;
  4873. }
  4874. }
  4875. void BfHotDepData::Deref()
  4876. {
  4877. mRefCount--;
  4878. BF_ASSERT(mRefCount >= 0);
  4879. if (mRefCount == 0)
  4880. {
  4881. switch (mDataKind)
  4882. {
  4883. case BfHotDepDataKind_TypeVersion:
  4884. delete (BfHotTypeVersion*)this;
  4885. break;
  4886. case BfHotDepDataKind_ThisType:
  4887. delete (BfHotThisType*)this;
  4888. break;
  4889. case BfHotDepDataKind_Allocation:
  4890. delete (BfHotAllocation*)this;
  4891. break;
  4892. case BfHotDepDataKind_Method:
  4893. delete (BfHotMethod*)this;
  4894. break;
  4895. case BfHotDepDataKind_DupMethod:
  4896. delete (BfHotDupMethod*)this;
  4897. break;
  4898. case BfHotDepDataKind_DevirtualizedMethod:
  4899. delete (BfHotDevirtualizedMethod*)this;
  4900. break;
  4901. case BfHotDepDataKind_InnerMethod:
  4902. delete (BfHotInnerMethod*)this;
  4903. break;
  4904. case BfHotDepDataKind_FunctionPtr:
  4905. delete (BfHotFunctionReference*)this;
  4906. break;
  4907. case BfHotDepDataKind_VirtualDecl:
  4908. delete (BfHotVirtualDeclaration*)this;
  4909. break;
  4910. default:
  4911. BF_FATAL("Not handled");
  4912. }
  4913. }
  4914. }
  4915. void BfHotMethod::Clear(bool keepDupMethods)
  4916. {
  4917. if (mPrevVersion != NULL)
  4918. {
  4919. mPrevVersion->Deref();
  4920. mPrevVersion = NULL;
  4921. }
  4922. if (mSrcTypeVersion != NULL)
  4923. {
  4924. mSrcTypeVersion->Deref();
  4925. mSrcTypeVersion = NULL;
  4926. }
  4927. if ((keepDupMethods) && ((mFlags & BfHotDepDataFlag_HasDup) != 0))
  4928. {
  4929. int writeIdx = 0;
  4930. for (int i = 0; i < (int)mReferences.size(); i++)
  4931. {
  4932. auto depData = mReferences[i];
  4933. if (depData->mDataKind == BfHotDepDataKind_DupMethod)
  4934. {
  4935. mReferences[writeIdx++] = depData;
  4936. }
  4937. else
  4938. {
  4939. depData->Deref();
  4940. }
  4941. }
  4942. mReferences.mSize = writeIdx;
  4943. }
  4944. else
  4945. {
  4946. for (auto depData : mReferences)
  4947. {
  4948. depData->Deref();
  4949. }
  4950. mReferences.Clear();
  4951. }
  4952. }
  4953. BfHotMethod::~BfHotMethod()
  4954. {
  4955. Clear();
  4956. }
  4957. //////////////////////////////////////////////////////////////////////////
  4958. #pragma warning(disable:4146)
  4959. // Only 63 chars - skip zero
  4960. static const char cHash64bToChar[] = { 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p',
  4961. 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z', 'A', 'B', 'C', 'D', 'E', 'F',
  4962. 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V',
  4963. 'W', 'X', 'Y', 'Z', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '_' };
  4964. String BfTypeUtils::HashEncode64(uint64 val)
  4965. {
  4966. String outStr;
  4967. if ((int64)val < 0)
  4968. {
  4969. uint64 flippedNum = (uint64)-(int64)val;
  4970. // Only flip if the encoded result would actually be shorter
  4971. if (flippedNum <= 0x00FFFFFFFFFFFFFFLL)
  4972. {
  4973. val = flippedNum;
  4974. outStr.Append('_');
  4975. }
  4976. }
  4977. while (val > 0)
  4978. {
  4979. int charIdx = val % 0x3F;
  4980. val /= 0x3F;
  4981. outStr.Append(cHash64bToChar[charIdx]);
  4982. }
  4983. return outStr;
  4984. }
  4985. BfPrimitiveType* BfTypeUtils::GetPrimitiveType(BfModule* module, BfTypeCode typeCode)
  4986. {
  4987. return module->GetPrimitiveType(typeCode);
  4988. }
  4989. void BfTypeUtils::PopulateType(BfModule* module, BfType* type)
  4990. {
  4991. module->PopulateType(type);
  4992. }
  4993. String BfTypeUtils::TypeToString(BfAstNode* typeRefNode)
  4994. {
  4995. if (auto identifierNode = BfNodeDynCast<BfIdentifierNode>(typeRefNode))
  4996. return identifierNode->ToString();
  4997. auto typeRef = BfNodeDynCast<BfTypeReference>(typeRefNode);
  4998. if (typeRef == NULL)
  4999. return "";
  5000. if (auto typeDefTypeRef = BfNodeDynCast<BfDirectTypeDefReference>(typeRef))
  5001. {
  5002. if (!typeDefTypeRef->mTypeDef->mNamespace.IsEmpty())
  5003. return typeDefTypeRef->mTypeDef->mNamespace.ToString() + "." + typeDefTypeRef->mTypeDef->mName->mString;
  5004. else
  5005. return String(typeDefTypeRef->mTypeDef->mName->mString);
  5006. }
  5007. if (typeRef->IsNamedTypeReference())
  5008. {
  5009. return typeRef->ToString();
  5010. }
  5011. if (auto ptrType = BfNodeDynCast<BfPointerTypeRef>(typeRef))
  5012. return TypeToString(ptrType->mElementType) + "*";
  5013. if (auto ptrType = BfNodeDynCast<BfArrayTypeRef>(typeRef))
  5014. {
  5015. String name = TypeToString(ptrType->mElementType) + "[";
  5016. for (int i = 1; i < ptrType->mDimensions; i++)
  5017. name += ",";
  5018. name += "]";
  5019. return name;
  5020. }
  5021. if (auto genericInstanceType = BfNodeDynCast<BfGenericInstanceTypeRef>(typeRef))
  5022. {
  5023. String name = TypeToString(genericInstanceType->mElementType);
  5024. name += "<";
  5025. for (int i = 0; i < (int)genericInstanceType->mGenericArguments.size(); i++)
  5026. {
  5027. if (i > 0)
  5028. name += ", ";
  5029. name += TypeToString(genericInstanceType->mGenericArguments[i]);
  5030. }
  5031. name += ">";
  5032. return name;
  5033. }
  5034. if (auto genericParamTypeRef = BfNodeDynCast<BfGenericParamTypeRef>(typeRef))
  5035. {
  5036. if (genericParamTypeRef->mGenericParamKind == BfGenericParamKind_Method)
  5037. return StrFormat("@M%d", genericParamTypeRef->mGenericParamIdx);
  5038. return StrFormat("@T%d", genericParamTypeRef->mGenericParamIdx);
  5039. }
  5040. if (auto qualifiedTypeRef = BfNodeDynCast<BfQualifiedTypeReference>(typeRef))
  5041. {
  5042. return TypeToString(qualifiedTypeRef->mLeft) + "." + TypeToString(qualifiedTypeRef->mRight);
  5043. }
  5044. if (auto refTypeRef = BfNodeDynCast<BfRefTypeRef>(typeRef))
  5045. {
  5046. String str = BfTokenToString(refTypeRef->mRefToken->GetToken());
  5047. str += " ";
  5048. str += TypeToString(refTypeRef->mElementType);
  5049. return str;
  5050. }
  5051. if (auto directStrTypeName = BfNodeDynCast<BfDirectStrTypeReference>(typeRef))
  5052. return directStrTypeName->mTypeName;
  5053. if (auto inlineTypeRef = BfNodeDynCast<BfInlineTypeReference>(typeRef))
  5054. return inlineTypeRef->mTypeDeclaration->mAnonymousName;
  5055. if (auto tupleTypeRef = BfNodeDynCast<BfTupleTypeRef>(typeRef))
  5056. {
  5057. String name = "(";
  5058. for (int i = 0; i < tupleTypeRef->mFieldTypes.size(); i++)
  5059. {
  5060. if (i > 0)
  5061. name += ", ";
  5062. name += TypeToString(tupleTypeRef->mFieldTypes[i]);
  5063. if ((i < tupleTypeRef->mFieldNames.size()) && (tupleTypeRef->mFieldNames[i] != NULL))
  5064. {
  5065. name += " ";
  5066. name += tupleTypeRef->mFieldNames[i]->ToString();
  5067. }
  5068. }
  5069. name += ")";
  5070. return name;
  5071. }
  5072. if (auto constTypeRef = BfNodeDynCast<BfConstExprTypeRef>(typeRef))
  5073. {
  5074. String name = "const ";
  5075. name += constTypeRef->mConstExpr->ToString();
  5076. return name;
  5077. }
  5078. //BF_DBG_FATAL("Not implemented");
  5079. return typeRef->ToString();
  5080. }
  5081. bool BfTypeUtils::TypeEquals(BfType* typeA, BfType* typeB, BfTypeInstance* selfType)
  5082. {
  5083. if (typeA->IsUnspecializedTypeVariation())
  5084. typeA = selfType->mModule->ResolveSelfType(typeA, selfType);
  5085. if (typeB->IsUnspecializedTypeVariation())
  5086. typeB = selfType->mModule->ResolveSelfType(typeB, selfType);
  5087. return typeA == typeB;
  5088. }
  5089. String BfTypeUtils::TypeToString(BfTypeDef* typeDef, BfTypeNameFlags typeNameFlags)
  5090. {
  5091. String str;
  5092. TypeToString(str, typeDef, typeNameFlags);
  5093. return str;
  5094. }
  5095. bool BfTypeUtils::TypeToString(StringImpl& str, BfTypeDef* typeDef, BfTypeNameFlags typeNameFlags)
  5096. {
  5097. auto checkTypeDef = typeDef;
  5098. char needsSep = 0;
  5099. if (checkTypeDef->mOuterType != NULL)
  5100. {
  5101. if (TypeToString(str, checkTypeDef->mOuterType, typeNameFlags))
  5102. {
  5103. if ((typeNameFlags & BfTypeNameFlag_InternalName) != 0)
  5104. needsSep = '+';
  5105. else
  5106. needsSep = '.';
  5107. }
  5108. }
  5109. else
  5110. {
  5111. if (((typeNameFlags & BfTypeNameFlag_OmitNamespace) == 0) && (!typeDef->mNamespace.IsEmpty()))
  5112. {
  5113. typeDef->mNamespace.ToString(str);
  5114. needsSep = '.';
  5115. }
  5116. }
  5117. if (needsSep != 0)
  5118. str += needsSep;
  5119. if (((typeNameFlags & BfTypeNameFlag_HideGlobalName) != 0) && (typeDef->IsGlobalsContainer()))
  5120. return false;
  5121. typeDef->mName->ToString(str);
  5122. if (typeDef->mGenericParamDefs.size() != 0)
  5123. {
  5124. int prevGenericParamCount = 0;
  5125. if (checkTypeDef->mOuterType != NULL)
  5126. prevGenericParamCount = (int)typeDef->mOuterType->mGenericParamDefs.size();
  5127. if (prevGenericParamCount != (int)checkTypeDef->mGenericParamDefs.size())
  5128. {
  5129. str += "<";
  5130. for (int i = prevGenericParamCount; i < (int)checkTypeDef->mGenericParamDefs.size(); i++)
  5131. {
  5132. if ((typeNameFlags & BfTypeNameFlag_InternalName) != 0)
  5133. {
  5134. if (i > prevGenericParamCount)
  5135. str += ",";
  5136. }
  5137. else
  5138. {
  5139. if (i > prevGenericParamCount)
  5140. str += ", ";
  5141. str += checkTypeDef->mGenericParamDefs[i]->mName;
  5142. }
  5143. }
  5144. str += ">";
  5145. }
  5146. }
  5147. return true;
  5148. }
  5149. int BfTypeUtils::GetSplatCount(BfType* type)
  5150. {
  5151. int splatCount = 0;
  5152. SplatIterate([&](BfType* checkType) { splatCount++; }, type);
  5153. return splatCount;
  5154. }
  5155. BfConstExprValueType::~BfConstExprValueType()
  5156. {
  5157. // mContext->mTypeConstExprCount--;
  5158. // BF_ASSERT(mContext->mTypeConstExprCount == 0);
  5159. }