1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616361736183619362036213622362336243625362636273628362936303631363236333634363536363637363836393640364136423643364436453646364736483649365036513652365336543655365636573658365936603661366236633664366536663667366836693670367136723673367436753676367736783679368036813682368336843685368636873688368936903691369236933694369536963697369836993700370137023703370437053706370737083709371037113712371337143715371637173718371937203721372237233724372537263727372837293730373137323733373437353736373737383739374037413742374337443745374637473748374937503751375237533754375537563757375837593760376137623763376437653766376737683769377037713772377337743775377637773778377937803781378237833784378537863787378837893790379137923793379437953796379737983799380038013802380338043805380638073808380938103811381238133814381538163817381838193820382138223823382438253826382738283829383038313832383338343835383638373838383938403841384238433844384538463847384838493850385138523853385438553856385738583859386038613862386338643865386638673868386938703871387238733874387538763877387838793880388138823883388438853886388738883889389038913892389338943895389638973898389939003901390239033904390539063907390839093910391139123913391439153916391739183919392039213922392339243925392639273928392939303931393239333934393539363937393839393940394139423943394439453946394739483949395039513952395339543955395639573958395939603961396239633964396539663967396839693970397139723973397439753976397739783979398039813982398339843985398639873988398939903991399239933994399539963997399839994000400140024003400440054006400740084009401040114012401340144015401640174018401940204021402240234024402540264027402840294030403140324033403440354036403740384039404040414042404340444045404640474048404940504051405240534054405540564057405840594060406140624063406440654066406740684069407040714072407340744075407640774078407940804081408240834084408540864087408840894090409140924093409440954096409740984099410041014102410341044105410641074108410941104111411241134114411541164117411841194120412141224123412441254126412741284129413041314132413341344135413641374138413941404141414241434144414541464147414841494150415141524153415441554156415741584159416041614162416341644165416641674168416941704171417241734174417541764177417841794180418141824183418441854186418741884189419041914192419341944195419641974198419942004201420242034204420542064207420842094210421142124213421442154216421742184219422042214222422342244225422642274228422942304231423242334234423542364237423842394240424142424243424442454246424742484249425042514252425342544255425642574258425942604261426242634264426542664267426842694270427142724273427442754276427742784279428042814282428342844285428642874288428942904291429242934294429542964297429842994300430143024303430443054306430743084309431043114312431343144315431643174318431943204321432243234324432543264327432843294330433143324333433443354336433743384339434043414342434343444345434643474348434943504351435243534354435543564357435843594360436143624363436443654366436743684369437043714372437343744375437643774378437943804381438243834384438543864387438843894390439143924393439443954396439743984399440044014402440344044405440644074408440944104411441244134414441544164417441844194420442144224423442444254426442744284429443044314432443344344435443644374438443944404441444244434444444544464447444844494450445144524453445444554456445744584459446044614462446344644465446644674468446944704471447244734474447544764477447844794480448144824483448444854486448744884489449044914492449344944495449644974498449945004501450245034504450545064507450845094510451145124513451445154516451745184519452045214522452345244525452645274528452945304531453245334534453545364537453845394540454145424543454445454546454745484549455045514552455345544555455645574558455945604561456245634564456545664567456845694570457145724573457445754576457745784579458045814582458345844585458645874588458945904591459245934594459545964597459845994600460146024603460446054606460746084609461046114612461346144615461646174618461946204621462246234624462546264627462846294630463146324633463446354636463746384639464046414642464346444645464646474648464946504651465246534654465546564657465846594660466146624663466446654666466746684669467046714672467346744675467646774678467946804681468246834684468546864687468846894690469146924693469446954696469746984699470047014702470347044705470647074708470947104711471247134714471547164717471847194720472147224723472447254726472747284729473047314732473347344735473647374738473947404741474247434744474547464747474847494750475147524753475447554756475747584759476047614762476347644765476647674768476947704771477247734774477547764777477847794780478147824783478447854786478747884789479047914792479347944795479647974798479948004801480248034804480548064807480848094810481148124813481448154816481748184819482048214822482348244825482648274828482948304831483248334834483548364837483848394840484148424843484448454846484748484849485048514852485348544855485648574858485948604861486248634864486548664867486848694870487148724873487448754876487748784879488048814882488348844885488648874888488948904891489248934894489548964897489848994900490149024903490449054906490749084909491049114912491349144915491649174918491949204921492249234924492549264927492849294930493149324933493449354936493749384939494049414942494349444945494649474948494949504951495249534954495549564957495849594960496149624963496449654966496749684969497049714972497349744975497649774978497949804981498249834984498549864987498849894990499149924993499449954996499749984999500050015002500350045005500650075008500950105011501250135014501550165017501850195020502150225023502450255026502750285029503050315032503350345035503650375038503950405041504250435044504550465047504850495050505150525053505450555056505750585059506050615062506350645065506650675068506950705071507250735074507550765077507850795080508150825083508450855086508750885089509050915092509350945095509650975098509951005101510251035104510551065107510851095110511151125113511451155116511751185119512051215122512351245125512651275128512951305131513251335134513551365137513851395140514151425143514451455146514751485149515051515152515351545155515651575158515951605161516251635164516551665167516851695170517151725173517451755176517751785179518051815182518351845185518651875188518951905191519251935194519551965197519851995200520152025203520452055206520752085209521052115212521352145215521652175218521952205221522252235224522552265227522852295230523152325233523452355236523752385239524052415242524352445245524652475248524952505251525252535254525552565257525852595260526152625263526452655266526752685269527052715272527352745275527652775278527952805281528252835284528552865287528852895290529152925293529452955296529752985299530053015302530353045305530653075308530953105311531253135314531553165317531853195320532153225323532453255326532753285329533053315332533353345335533653375338533953405341534253435344534553465347534853495350535153525353535453555356535753585359536053615362536353645365536653675368536953705371537253735374537553765377537853795380538153825383538453855386538753885389539053915392539353945395539653975398539954005401540254035404540554065407540854095410541154125413541454155416541754185419542054215422542354245425542654275428542954305431543254335434543554365437543854395440544154425443544454455446544754485449545054515452545354545455545654575458545954605461546254635464546554665467546854695470547154725473547454755476547754785479548054815482548354845485548654875488548954905491549254935494549554965497549854995500550155025503550455055506550755085509551055115512551355145515551655175518551955205521552255235524552555265527552855295530553155325533553455355536553755385539554055415542554355445545554655475548554955505551555255535554555555565557555855595560556155625563556455655566556755685569557055715572557355745575557655775578557955805581558255835584558555865587558855895590559155925593559455955596559755985599560056015602560356045605560656075608560956105611561256135614561556165617561856195620562156225623562456255626562756285629563056315632563356345635563656375638563956405641564256435644564556465647564856495650565156525653565456555656565756585659566056615662566356645665566656675668566956705671567256735674567556765677567856795680568156825683568456855686568756885689569056915692569356945695569656975698569957005701570257035704570557065707570857095710571157125713571457155716571757185719572057215722572357245725572657275728572957305731573257335734573557365737573857395740574157425743574457455746574757485749575057515752575357545755575657575758575957605761576257635764576557665767576857695770577157725773577457755776577757785779 |
- #pragma once
- #ifdef __GNUC__
- #pragma GCC diagnostic push
- #pragma GCC diagnostic ignored "-Wunused-parameter"
- #endif
- //===--- ItaniumDemangle.h -----------*- mode:c++;eval:(read-only-mode) -*-===//
- // Do not edit! See README.txt.
- // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
- // See https://llvm.org/LICENSE.txt for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- //
- //===----------------------------------------------------------------------===//
- //
- // Generic itanium demangler library.
- // There are two copies of this file in the source tree. The one under
- // libcxxabi is the original and the one under llvm is the copy. Use
- // cp-to-llvm.sh to update the copy. See README.txt for more details.
- //
- //===----------------------------------------------------------------------===//
- #ifndef DEMANGLE_ITANIUMDEMANGLE_H
- #define DEMANGLE_ITANIUMDEMANGLE_H
- // FIXME: (possibly) incomplete list of features that clang mangles that this
- // file does not yet support:
- // - C++ modules TS
- #include "DemangleConfig.h"
- #include "StringView.h"
- #include "Utility.h"
- #include <algorithm>
- #include <cassert>
- #include <cctype>
- #include <cstdio>
- #include <cstdlib>
- #include <cstring>
- #include <limits>
- #include <type_traits>
- #include <utility>
- #define FOR_EACH_NODE_KIND(X) \
- X(NodeArrayNode) \
- X(DotSuffix) \
- X(VendorExtQualType) \
- X(QualType) \
- X(ConversionOperatorType) \
- X(PostfixQualifiedType) \
- X(ElaboratedTypeSpefType) \
- X(NameType) \
- X(AbiTagAttr) \
- X(EnableIfAttr) \
- X(ObjCProtoName) \
- X(PointerType) \
- X(ReferenceType) \
- X(PointerToMemberType) \
- X(ArrayType) \
- X(FunctionType) \
- X(NoexceptSpec) \
- X(DynamicExceptionSpec) \
- X(FunctionEncoding) \
- X(LiteralOperator) \
- X(SpecialName) \
- X(CtorVtableSpecialName) \
- X(QualifiedName) \
- X(NestedName) \
- X(LocalName) \
- X(VectorType) \
- X(PixelVectorType) \
- X(BinaryFPType) \
- X(SyntheticTemplateParamName) \
- X(TypeTemplateParamDecl) \
- X(NonTypeTemplateParamDecl) \
- X(TemplateTemplateParamDecl) \
- X(TemplateParamPackDecl) \
- X(ParameterPack) \
- X(TemplateArgumentPack) \
- X(ParameterPackExpansion) \
- X(TemplateArgs) \
- X(ForwardTemplateReference) \
- X(NameWithTemplateArgs) \
- X(GlobalQualifiedName) \
- X(StdQualifiedName) \
- X(ExpandedSpecialSubstitution) \
- X(SpecialSubstitution) \
- X(CtorDtorName) \
- X(DtorName) \
- X(UnnamedTypeName) \
- X(ClosureTypeName) \
- X(StructuredBindingName) \
- X(BinaryExpr) \
- X(ArraySubscriptExpr) \
- X(PostfixExpr) \
- X(ConditionalExpr) \
- X(MemberExpr) \
- X(SubobjectExpr) \
- X(EnclosingExpr) \
- X(CastExpr) \
- X(SizeofParamPackExpr) \
- X(CallExpr) \
- X(NewExpr) \
- X(DeleteExpr) \
- X(PrefixExpr) \
- X(FunctionParam) \
- X(ConversionExpr) \
- X(PointerToMemberConversionExpr) \
- X(InitListExpr) \
- X(FoldExpr) \
- X(ThrowExpr) \
- X(BoolExpr) \
- X(StringLiteral) \
- X(LambdaExpr) \
- X(EnumLiteral) \
- X(IntegerLiteral) \
- X(FloatLiteral) \
- X(DoubleLiteral) \
- X(LongDoubleLiteral) \
- X(BracedExpr) \
- X(BracedRangeExpr)
- DEMANGLE_NAMESPACE_BEGIN
- template <class T, size_t N> class PODSmallVector {
- static_assert(std::is_trivial<T>::value,
- "T is required to be a trivial type");
- T *First = nullptr;
- T *Last = nullptr;
- T *Cap = nullptr;
- T Inline[N] = {};
- bool isInline() const { return First == Inline; }
- void clearInline() {
- First = Inline;
- Last = Inline;
- Cap = Inline + N;
- }
- void reserve(size_t NewCap) {
- size_t S = size();
- if (isInline()) {
- auto *Tmp = static_cast<T *>(std::malloc(NewCap * sizeof(T)));
- if (Tmp == nullptr)
- std::terminate();
- std::copy(First, Last, Tmp);
- First = Tmp;
- } else {
- First = static_cast<T *>(std::realloc(First, NewCap * sizeof(T)));
- if (First == nullptr)
- std::terminate();
- }
- Last = First + S;
- Cap = First + NewCap;
- }
- public:
- PODSmallVector() : First(Inline), Last(First), Cap(Inline + N) {}
- PODSmallVector(const PODSmallVector &) = delete;
- PODSmallVector &operator=(const PODSmallVector &) = delete;
- PODSmallVector(PODSmallVector &&Other) : PODSmallVector() {
- if (Other.isInline()) {
- std::copy(Other.begin(), Other.end(), First);
- Last = First + Other.size();
- Other.clear();
- return;
- }
- First = Other.First;
- Last = Other.Last;
- Cap = Other.Cap;
- Other.clearInline();
- }
- PODSmallVector &operator=(PODSmallVector &&Other) {
- if (Other.isInline()) {
- if (!isInline()) {
- std::free(First);
- clearInline();
- }
- std::copy(Other.begin(), Other.end(), First);
- Last = First + Other.size();
- Other.clear();
- return *this;
- }
- if (isInline()) {
- First = Other.First;
- Last = Other.Last;
- Cap = Other.Cap;
- Other.clearInline();
- return *this;
- }
- std::swap(First, Other.First);
- std::swap(Last, Other.Last);
- std::swap(Cap, Other.Cap);
- Other.clear();
- return *this;
- }
- // NOLINTNEXTLINE(readability-identifier-naming)
- void push_back(const T &Elem) {
- if (Last == Cap)
- reserve(size() * 2);
- *Last++ = Elem;
- }
- // NOLINTNEXTLINE(readability-identifier-naming)
- void pop_back() {
- assert(Last != First && "Popping empty vector!");
- --Last;
- }
- void dropBack(size_t Index) {
- assert(Index <= size() && "dropBack() can't expand!");
- Last = First + Index;
- }
- T *begin() { return First; }
- T *end() { return Last; }
- bool empty() const { return First == Last; }
- size_t size() const { return static_cast<size_t>(Last - First); }
- T &back() {
- assert(Last != First && "Calling back() on empty vector!");
- return *(Last - 1);
- }
- T &operator[](size_t Index) {
- assert(Index < size() && "Invalid access!");
- return *(begin() + Index);
- }
- void clear() { Last = First; }
- ~PODSmallVector() {
- if (!isInline())
- std::free(First);
- }
- };
- // Base class of all AST nodes. The AST is built by the parser, then is
- // traversed by the printLeft/Right functions to produce a demangled string.
- class Node {
- public:
- enum Kind : unsigned char {
- #define ENUMERATOR(NodeKind) K ## NodeKind,
- FOR_EACH_NODE_KIND(ENUMERATOR)
- #undef ENUMERATOR
- };
- /// Three-way bool to track a cached value. Unknown is possible if this node
- /// has an unexpanded parameter pack below it that may affect this cache.
- enum class Cache : unsigned char { Yes, No, Unknown, };
- private:
- Kind K;
- // FIXME: Make these protected.
- public:
- /// Tracks if this node has a component on its right side, in which case we
- /// need to call printRight.
- Cache RHSComponentCache;
- /// Track if this node is a (possibly qualified) array type. This can affect
- /// how we format the output string.
- Cache ArrayCache;
- /// Track if this node is a (possibly qualified) function type. This can
- /// affect how we format the output string.
- Cache FunctionCache;
- public:
- Node(Kind K_, Cache RHSComponentCache_ = Cache::No,
- Cache ArrayCache_ = Cache::No, Cache FunctionCache_ = Cache::No)
- : K(K_), RHSComponentCache(RHSComponentCache_), ArrayCache(ArrayCache_),
- FunctionCache(FunctionCache_) {}
- /// Visit the most-derived object corresponding to this object.
- template<typename Fn> void visit(Fn F) const;
- // The following function is provided by all derived classes:
- //
- // Call F with arguments that, when passed to the constructor of this node,
- // would construct an equivalent node.
- //template<typename Fn> void match(Fn F) const;
- bool hasRHSComponent(OutputBuffer &OB) const {
- if (RHSComponentCache != Cache::Unknown)
- return RHSComponentCache == Cache::Yes;
- return hasRHSComponentSlow(OB);
- }
- bool hasArray(OutputBuffer &OB) const {
- if (ArrayCache != Cache::Unknown)
- return ArrayCache == Cache::Yes;
- return hasArraySlow(OB);
- }
- bool hasFunction(OutputBuffer &OB) const {
- if (FunctionCache != Cache::Unknown)
- return FunctionCache == Cache::Yes;
- return hasFunctionSlow(OB);
- }
- Kind getKind() const { return K; }
- virtual bool hasRHSComponentSlow(OutputBuffer &) const { return false; }
- virtual bool hasArraySlow(OutputBuffer &) const { return false; }
- virtual bool hasFunctionSlow(OutputBuffer &) const { return false; }
- // Dig through "glue" nodes like ParameterPack and ForwardTemplateReference to
- // get at a node that actually represents some concrete syntax.
- virtual const Node *getSyntaxNode(OutputBuffer &) const { return this; }
- void print(OutputBuffer &OB) const {
- printLeft(OB);
- if (RHSComponentCache != Cache::No)
- printRight(OB);
- }
- // Print the "left" side of this Node into OutputBuffer.
- virtual void printLeft(OutputBuffer &) const = 0;
- // Print the "right". This distinction is necessary to represent C++ types
- // that appear on the RHS of their subtype, such as arrays or functions.
- // Since most types don't have such a component, provide a default
- // implementation.
- virtual void printRight(OutputBuffer &) const {}
- virtual StringView getBaseName() const { return StringView(); }
- // Silence compiler warnings, this dtor will never be called.
- virtual ~Node() = default;
- #ifndef NDEBUG
- DEMANGLE_DUMP_METHOD void dump() const;
- #endif
- };
- class IASTProcessor {
- public:
- virtual void Process(const Node*) = 0;
- };
- void demangleAST(const char *MangledName, IASTProcessor *Processor);
- class NodeArray {
- Node **Elements;
- size_t NumElements;
- public:
- NodeArray() : Elements(nullptr), NumElements(0) {}
- NodeArray(Node **Elements_, size_t NumElements_)
- : Elements(Elements_), NumElements(NumElements_) {}
- bool empty() const { return NumElements == 0; }
- size_t size() const { return NumElements; }
- Node **begin() const { return Elements; }
- Node **end() const { return Elements + NumElements; }
- Node *operator[](size_t Idx) const { return Elements[Idx]; }
- void printWithComma(OutputBuffer &OB) const {
- bool FirstElement = true;
- for (size_t Idx = 0; Idx != NumElements; ++Idx) {
- size_t BeforeComma = OB.getCurrentPosition();
- if (!FirstElement)
- OB += ", ";
- size_t AfterComma = OB.getCurrentPosition();
- Elements[Idx]->print(OB);
- // Elements[Idx] is an empty parameter pack expansion, we should erase the
- // comma we just printed.
- if (AfterComma == OB.getCurrentPosition()) {
- OB.setCurrentPosition(BeforeComma);
- continue;
- }
- FirstElement = false;
- }
- }
- };
- struct NodeArrayNode : Node {
- NodeArray Array;
- NodeArrayNode(NodeArray Array_) : Node(KNodeArrayNode), Array(Array_) {}
- template<typename Fn> void match(Fn F) const { F(Array); }
- void printLeft(OutputBuffer &OB) const override { Array.printWithComma(OB); }
- };
- class DotSuffix final : public Node {
- const Node *Prefix;
- const StringView Suffix;
- public:
- DotSuffix(const Node *Prefix_, StringView Suffix_)
- : Node(KDotSuffix), Prefix(Prefix_), Suffix(Suffix_) {}
- template<typename Fn> void match(Fn F) const { F(Prefix, Suffix); }
- void printLeft(OutputBuffer &OB) const override {
- Prefix->print(OB);
- OB += " (";
- OB += Suffix;
- OB += ")";
- }
- };
- class VendorExtQualType final : public Node {
- const Node *Ty;
- StringView Ext;
- const Node *TA;
- public:
- VendorExtQualType(const Node *Ty_, StringView Ext_, const Node *TA_)
- : Node(KVendorExtQualType), Ty(Ty_), Ext(Ext_), TA(TA_) {}
- template <typename Fn> void match(Fn F) const { F(Ty, Ext, TA); }
- void printLeft(OutputBuffer &OB) const override {
- Ty->print(OB);
- OB += " ";
- OB += Ext;
- if (TA != nullptr)
- TA->print(OB);
- }
- };
- enum FunctionRefQual : unsigned char {
- FrefQualNone,
- FrefQualLValue,
- FrefQualRValue,
- };
- enum Qualifiers {
- QualNone = 0,
- QualConst = 0x1,
- QualVolatile = 0x2,
- QualRestrict = 0x4,
- };
- inline Qualifiers operator|=(Qualifiers &Q1, Qualifiers Q2) {
- return Q1 = static_cast<Qualifiers>(Q1 | Q2);
- }
- class QualType final : public Node {
- protected:
- const Qualifiers Quals;
- const Node *Child;
- void printQuals(OutputBuffer &OB) const {
- if (Quals & QualConst)
- OB += " const";
- if (Quals & QualVolatile)
- OB += " volatile";
- if (Quals & QualRestrict)
- OB += " restrict";
- }
- public:
- QualType(const Node *Child_, Qualifiers Quals_)
- : Node(KQualType, Child_->RHSComponentCache,
- Child_->ArrayCache, Child_->FunctionCache),
- Quals(Quals_), Child(Child_) {}
- template<typename Fn> void match(Fn F) const { F(Child, Quals); }
- bool hasRHSComponentSlow(OutputBuffer &OB) const override {
- return Child->hasRHSComponent(OB);
- }
- bool hasArraySlow(OutputBuffer &OB) const override {
- return Child->hasArray(OB);
- }
- bool hasFunctionSlow(OutputBuffer &OB) const override {
- return Child->hasFunction(OB);
- }
- void printLeft(OutputBuffer &OB) const override {
- Child->printLeft(OB);
- printQuals(OB);
- }
- void printRight(OutputBuffer &OB) const override { Child->printRight(OB); }
- };
- class ConversionOperatorType final : public Node {
- const Node *Ty;
- public:
- ConversionOperatorType(const Node *Ty_)
- : Node(KConversionOperatorType), Ty(Ty_) {}
- template<typename Fn> void match(Fn F) const { F(Ty); }
- void printLeft(OutputBuffer &OB) const override {
- OB += "operator ";
- Ty->print(OB);
- }
- };
- class PostfixQualifiedType final : public Node {
- const Node *Ty;
- const StringView Postfix;
- public:
- PostfixQualifiedType(Node *Ty_, StringView Postfix_)
- : Node(KPostfixQualifiedType), Ty(Ty_), Postfix(Postfix_) {}
- template<typename Fn> void match(Fn F) const { F(Ty, Postfix); }
- void printLeft(OutputBuffer &OB) const override {
- Ty->printLeft(OB);
- OB += Postfix;
- }
- };
- class NameType final : public Node {
- const StringView Name;
- public:
- NameType(StringView Name_) : Node(KNameType), Name(Name_) {}
- template<typename Fn> void match(Fn F) const { F(Name); }
- StringView getName() const { return Name; }
- StringView getBaseName() const override { return Name; }
- void printLeft(OutputBuffer &OB) const override { OB += Name; }
- };
- class ElaboratedTypeSpefType : public Node {
- StringView Kind;
- Node *Child;
- public:
- ElaboratedTypeSpefType(StringView Kind_, Node *Child_)
- : Node(KElaboratedTypeSpefType), Kind(Kind_), Child(Child_) {}
- template<typename Fn> void match(Fn F) const { F(Kind, Child); }
- void printLeft(OutputBuffer &OB) const override {
- OB += Kind;
- OB += ' ';
- Child->print(OB);
- }
- };
- struct AbiTagAttr : Node {
- Node *Base;
- StringView Tag;
- AbiTagAttr(Node* Base_, StringView Tag_)
- : Node(KAbiTagAttr, Base_->RHSComponentCache,
- Base_->ArrayCache, Base_->FunctionCache),
- Base(Base_), Tag(Tag_) {}
- template<typename Fn> void match(Fn F) const { F(Base, Tag); }
- void printLeft(OutputBuffer &OB) const override {
- Base->printLeft(OB);
- OB += "[abi:";
- OB += Tag;
- OB += "]";
- }
- };
- class EnableIfAttr : public Node {
- NodeArray Conditions;
- public:
- EnableIfAttr(NodeArray Conditions_)
- : Node(KEnableIfAttr), Conditions(Conditions_) {}
- template<typename Fn> void match(Fn F) const { F(Conditions); }
- void printLeft(OutputBuffer &OB) const override {
- OB += " [enable_if:";
- Conditions.printWithComma(OB);
- OB += ']';
- }
- };
- class ObjCProtoName : public Node {
- const Node *Ty;
- StringView Protocol;
- friend class PointerType;
- public:
- ObjCProtoName(const Node *Ty_, StringView Protocol_)
- : Node(KObjCProtoName), Ty(Ty_), Protocol(Protocol_) {}
- template<typename Fn> void match(Fn F) const { F(Ty, Protocol); }
- bool isObjCObject() const {
- return Ty->getKind() == KNameType &&
- static_cast<const NameType *>(Ty)->getName() == "objc_object";
- }
- void printLeft(OutputBuffer &OB) const override {
- Ty->print(OB);
- OB += "<";
- OB += Protocol;
- OB += ">";
- }
- };
- class PointerType final : public Node {
- const Node *Pointee;
- public:
- PointerType(const Node *Pointee_)
- : Node(KPointerType, Pointee_->RHSComponentCache),
- Pointee(Pointee_) {}
- template<typename Fn> void match(Fn F) const { F(Pointee); }
- bool hasRHSComponentSlow(OutputBuffer &OB) const override {
- return Pointee->hasRHSComponent(OB);
- }
- void printLeft(OutputBuffer &OB) const override {
- // We rewrite objc_object<SomeProtocol>* into id<SomeProtocol>.
- if (Pointee->getKind() != KObjCProtoName ||
- !static_cast<const ObjCProtoName *>(Pointee)->isObjCObject()) {
- Pointee->printLeft(OB);
- if (Pointee->hasArray(OB))
- OB += " ";
- if (Pointee->hasArray(OB) || Pointee->hasFunction(OB))
- OB += "(";
- OB += "*";
- } else {
- const auto *objcProto = static_cast<const ObjCProtoName *>(Pointee);
- OB += "id<";
- OB += objcProto->Protocol;
- OB += ">";
- }
- }
- void printRight(OutputBuffer &OB) const override {
- if (Pointee->getKind() != KObjCProtoName ||
- !static_cast<const ObjCProtoName *>(Pointee)->isObjCObject()) {
- if (Pointee->hasArray(OB) || Pointee->hasFunction(OB))
- OB += ")";
- Pointee->printRight(OB);
- }
- }
- };
- enum class ReferenceKind {
- LValue,
- RValue,
- };
- // Represents either a LValue or an RValue reference type.
- class ReferenceType : public Node {
- const Node *Pointee;
- ReferenceKind RK;
- mutable bool Printing = false;
- // Dig through any refs to refs, collapsing the ReferenceTypes as we go. The
- // rule here is rvalue ref to rvalue ref collapses to a rvalue ref, and any
- // other combination collapses to a lvalue ref.
- //
- // A combination of a TemplateForwardReference and a back-ref Substitution
- // from an ill-formed string may have created a cycle; use cycle detection to
- // avoid looping forever.
- std::pair<ReferenceKind, const Node *> collapse(OutputBuffer &OB) const {
- auto SoFar = std::make_pair(RK, Pointee);
- // Track the chain of nodes for the Floyd's 'tortoise and hare'
- // cycle-detection algorithm, since getSyntaxNode(S) is impure
- PODSmallVector<const Node *, 8> Prev;
- for (;;) {
- const Node *SN = SoFar.second->getSyntaxNode(OB);
- if (SN->getKind() != KReferenceType)
- break;
- auto *RT = static_cast<const ReferenceType *>(SN);
- SoFar.second = RT->Pointee;
- SoFar.first = std::min(SoFar.first, RT->RK);
- // The middle of Prev is the 'slow' pointer moving at half speed
- Prev.push_back(SoFar.second);
- if (Prev.size() > 1 && SoFar.second == Prev[(Prev.size() - 1) / 2]) {
- // Cycle detected
- SoFar.second = nullptr;
- break;
- }
- }
- return SoFar;
- }
- public:
- ReferenceType(const Node *Pointee_, ReferenceKind RK_)
- : Node(KReferenceType, Pointee_->RHSComponentCache),
- Pointee(Pointee_), RK(RK_) {}
- template<typename Fn> void match(Fn F) const { F(Pointee, RK); }
- bool hasRHSComponentSlow(OutputBuffer &OB) const override {
- return Pointee->hasRHSComponent(OB);
- }
- void printLeft(OutputBuffer &OB) const override {
- if (Printing)
- return;
- SwapAndRestore<bool> SavePrinting(Printing, true);
- std::pair<ReferenceKind, const Node *> Collapsed = collapse(OB);
- if (!Collapsed.second)
- return;
- Collapsed.second->printLeft(OB);
- if (Collapsed.second->hasArray(OB))
- OB += " ";
- if (Collapsed.second->hasArray(OB) || Collapsed.second->hasFunction(OB))
- OB += "(";
- OB += (Collapsed.first == ReferenceKind::LValue ? "&" : "&&");
- }
- void printRight(OutputBuffer &OB) const override {
- if (Printing)
- return;
- SwapAndRestore<bool> SavePrinting(Printing, true);
- std::pair<ReferenceKind, const Node *> Collapsed = collapse(OB);
- if (!Collapsed.second)
- return;
- if (Collapsed.second->hasArray(OB) || Collapsed.second->hasFunction(OB))
- OB += ")";
- Collapsed.second->printRight(OB);
- }
- };
- class PointerToMemberType final : public Node {
- const Node *ClassType;
- const Node *MemberType;
- public:
- PointerToMemberType(const Node *ClassType_, const Node *MemberType_)
- : Node(KPointerToMemberType, MemberType_->RHSComponentCache),
- ClassType(ClassType_), MemberType(MemberType_) {}
- template<typename Fn> void match(Fn F) const { F(ClassType, MemberType); }
- bool hasRHSComponentSlow(OutputBuffer &OB) const override {
- return MemberType->hasRHSComponent(OB);
- }
- void printLeft(OutputBuffer &OB) const override {
- MemberType->printLeft(OB);
- if (MemberType->hasArray(OB) || MemberType->hasFunction(OB))
- OB += "(";
- else
- OB += " ";
- ClassType->print(OB);
- OB += "::*";
- }
- void printRight(OutputBuffer &OB) const override {
- if (MemberType->hasArray(OB) || MemberType->hasFunction(OB))
- OB += ")";
- MemberType->printRight(OB);
- }
- };
- class ArrayType final : public Node {
- const Node *Base;
- Node *Dimension;
- public:
- ArrayType(const Node *Base_, Node *Dimension_)
- : Node(KArrayType,
- /*RHSComponentCache=*/Cache::Yes,
- /*ArrayCache=*/Cache::Yes),
- Base(Base_), Dimension(Dimension_) {}
- template<typename Fn> void match(Fn F) const { F(Base, Dimension); }
- bool hasRHSComponentSlow(OutputBuffer &) const override { return true; }
- bool hasArraySlow(OutputBuffer &) const override { return true; }
- void printLeft(OutputBuffer &OB) const override { Base->printLeft(OB); }
- void printRight(OutputBuffer &OB) const override {
- if (OB.back() != ']')
- OB += " ";
- OB += "[";
- if (Dimension)
- Dimension->print(OB);
- OB += "]";
- Base->printRight(OB);
- }
- };
- class FunctionType final : public Node {
- const Node *Ret;
- NodeArray Params;
- Qualifiers CVQuals;
- FunctionRefQual RefQual;
- const Node *ExceptionSpec;
- public:
- FunctionType(const Node *Ret_, NodeArray Params_, Qualifiers CVQuals_,
- FunctionRefQual RefQual_, const Node *ExceptionSpec_)
- : Node(KFunctionType,
- /*RHSComponentCache=*/Cache::Yes, /*ArrayCache=*/Cache::No,
- /*FunctionCache=*/Cache::Yes),
- Ret(Ret_), Params(Params_), CVQuals(CVQuals_), RefQual(RefQual_),
- ExceptionSpec(ExceptionSpec_) {}
- template<typename Fn> void match(Fn F) const {
- F(Ret, Params, CVQuals, RefQual, ExceptionSpec);
- }
- bool hasRHSComponentSlow(OutputBuffer &) const override { return true; }
- bool hasFunctionSlow(OutputBuffer &) const override { return true; }
- // Handle C++'s ... quirky decl grammar by using the left & right
- // distinction. Consider:
- // int (*f(float))(char) {}
- // f is a function that takes a float and returns a pointer to a function
- // that takes a char and returns an int. If we're trying to print f, start
- // by printing out the return types's left, then print our parameters, then
- // finally print right of the return type.
- void printLeft(OutputBuffer &OB) const override {
- Ret->printLeft(OB);
- OB += " ";
- }
- void printRight(OutputBuffer &OB) const override {
- OB += "(";
- Params.printWithComma(OB);
- OB += ")";
- Ret->printRight(OB);
- if (CVQuals & QualConst)
- OB += " const";
- if (CVQuals & QualVolatile)
- OB += " volatile";
- if (CVQuals & QualRestrict)
- OB += " restrict";
- if (RefQual == FrefQualLValue)
- OB += " &";
- else if (RefQual == FrefQualRValue)
- OB += " &&";
- if (ExceptionSpec != nullptr) {
- OB += ' ';
- ExceptionSpec->print(OB);
- }
- }
- };
- class NoexceptSpec : public Node {
- const Node *E;
- public:
- NoexceptSpec(const Node *E_) : Node(KNoexceptSpec), E(E_) {}
- template<typename Fn> void match(Fn F) const { F(E); }
- void printLeft(OutputBuffer &OB) const override {
- OB += "noexcept(";
- E->print(OB);
- OB += ")";
- }
- };
- class DynamicExceptionSpec : public Node {
- NodeArray Types;
- public:
- DynamicExceptionSpec(NodeArray Types_)
- : Node(KDynamicExceptionSpec), Types(Types_) {}
- template<typename Fn> void match(Fn F) const { F(Types); }
- void printLeft(OutputBuffer &OB) const override {
- OB += "throw(";
- Types.printWithComma(OB);
- OB += ')';
- }
- };
- class FunctionEncoding final : public Node {
- const Node *Ret;
- const Node *Name;
- NodeArray Params;
- const Node *Attrs;
- Qualifiers CVQuals;
- FunctionRefQual RefQual;
- public:
- FunctionEncoding(const Node *Ret_, const Node *Name_, NodeArray Params_,
- const Node *Attrs_, Qualifiers CVQuals_,
- FunctionRefQual RefQual_)
- : Node(KFunctionEncoding,
- /*RHSComponentCache=*/Cache::Yes, /*ArrayCache=*/Cache::No,
- /*FunctionCache=*/Cache::Yes),
- Ret(Ret_), Name(Name_), Params(Params_), Attrs(Attrs_),
- CVQuals(CVQuals_), RefQual(RefQual_) {}
- template<typename Fn> void match(Fn F) const {
- F(Ret, Name, Params, Attrs, CVQuals, RefQual);
- }
- Qualifiers getCVQuals() const { return CVQuals; }
- FunctionRefQual getRefQual() const { return RefQual; }
- NodeArray getParams() const { return Params; }
- const Node *getReturnType() const { return Ret; }
- bool hasRHSComponentSlow(OutputBuffer &) const override { return true; }
- bool hasFunctionSlow(OutputBuffer &) const override { return true; }
- const Node *getName() const { return Name; }
- void printLeft(OutputBuffer &OB) const override {
- if (Ret) {
- Ret->printLeft(OB);
- if (!Ret->hasRHSComponent(OB))
- OB += " ";
- }
- Name->print(OB);
- }
- void printRight(OutputBuffer &OB) const override {
- OB += "(";
- Params.printWithComma(OB);
- OB += ")";
- if (Ret)
- Ret->printRight(OB);
- if (CVQuals & QualConst)
- OB += " const";
- if (CVQuals & QualVolatile)
- OB += " volatile";
- if (CVQuals & QualRestrict)
- OB += " restrict";
- if (RefQual == FrefQualLValue)
- OB += " &";
- else if (RefQual == FrefQualRValue)
- OB += " &&";
- if (Attrs != nullptr)
- Attrs->print(OB);
- }
- };
- class LiteralOperator : public Node {
- const Node *OpName;
- public:
- LiteralOperator(const Node *OpName_)
- : Node(KLiteralOperator), OpName(OpName_) {}
- template<typename Fn> void match(Fn F) const { F(OpName); }
- void printLeft(OutputBuffer &OB) const override {
- OB += "operator\"\" ";
- OpName->print(OB);
- }
- };
- class SpecialName final : public Node {
- const StringView Special;
- const Node *Child;
- public:
- SpecialName(StringView Special_, const Node *Child_)
- : Node(KSpecialName), Special(Special_), Child(Child_) {}
- template<typename Fn> void match(Fn F) const { F(Special, Child); }
- void printLeft(OutputBuffer &OB) const override {
- OB += Special;
- Child->print(OB);
- }
- };
- class CtorVtableSpecialName final : public Node {
- const Node *FirstType;
- const Node *SecondType;
- public:
- CtorVtableSpecialName(const Node *FirstType_, const Node *SecondType_)
- : Node(KCtorVtableSpecialName),
- FirstType(FirstType_), SecondType(SecondType_) {}
- template<typename Fn> void match(Fn F) const { F(FirstType, SecondType); }
- void printLeft(OutputBuffer &OB) const override {
- OB += "construction vtable for ";
- FirstType->print(OB);
- OB += "-in-";
- SecondType->print(OB);
- }
- };
- struct NestedName : Node {
- Node *Qual;
- Node *Name;
- NestedName(Node *Qual_, Node *Name_)
- : Node(KNestedName), Qual(Qual_), Name(Name_) {}
- template<typename Fn> void match(Fn F) const { F(Qual, Name); }
- StringView getBaseName() const override { return Name->getBaseName(); }
- void printLeft(OutputBuffer &OB) const override {
- Qual->print(OB);
- OB += "::";
- Name->print(OB);
- }
- };
- struct LocalName : Node {
- Node *Encoding;
- Node *Entity;
- LocalName(Node *Encoding_, Node *Entity_)
- : Node(KLocalName), Encoding(Encoding_), Entity(Entity_) {}
- template<typename Fn> void match(Fn F) const { F(Encoding, Entity); }
- void printLeft(OutputBuffer &OB) const override {
- Encoding->print(OB);
- OB += "::";
- Entity->print(OB);
- }
- };
- class QualifiedName final : public Node {
- // qualifier::name
- const Node *Qualifier;
- const Node *Name;
- public:
- QualifiedName(const Node *Qualifier_, const Node *Name_)
- : Node(KQualifiedName), Qualifier(Qualifier_), Name(Name_) {}
- template<typename Fn> void match(Fn F) const { F(Qualifier, Name); }
- StringView getBaseName() const override { return Name->getBaseName(); }
- void printLeft(OutputBuffer &OB) const override {
- Qualifier->print(OB);
- OB += "::";
- Name->print(OB);
- }
- };
- class VectorType final : public Node {
- const Node *BaseType;
- const Node *Dimension;
- public:
- VectorType(const Node *BaseType_, Node *Dimension_)
- : Node(KVectorType), BaseType(BaseType_),
- Dimension(Dimension_) {}
- template<typename Fn> void match(Fn F) const { F(BaseType, Dimension); }
- void printLeft(OutputBuffer &OB) const override {
- BaseType->print(OB);
- OB += " vector[";
- if (Dimension)
- Dimension->print(OB);
- OB += "]";
- }
- };
- class PixelVectorType final : public Node {
- const Node *Dimension;
- public:
- PixelVectorType(const Node *Dimension_)
- : Node(KPixelVectorType), Dimension(Dimension_) {}
- template<typename Fn> void match(Fn F) const { F(Dimension); }
- void printLeft(OutputBuffer &OB) const override {
- // FIXME: This should demangle as "vector pixel".
- OB += "pixel vector[";
- Dimension->print(OB);
- OB += "]";
- }
- };
- class BinaryFPType final : public Node {
- const Node *Dimension;
- public:
- BinaryFPType(const Node *Dimension_)
- : Node(KBinaryFPType), Dimension(Dimension_) {}
- template<typename Fn> void match(Fn F) const { F(Dimension); }
- void printLeft(OutputBuffer &OB) const override {
- OB += "_Float";
- Dimension->print(OB);
- }
- };
- enum class TemplateParamKind { Type, NonType, Template };
- /// An invented name for a template parameter for which we don't have a
- /// corresponding template argument.
- ///
- /// This node is created when parsing the <lambda-sig> for a lambda with
- /// explicit template arguments, which might be referenced in the parameter
- /// types appearing later in the <lambda-sig>.
- class SyntheticTemplateParamName final : public Node {
- TemplateParamKind Kind;
- unsigned Index;
- public:
- SyntheticTemplateParamName(TemplateParamKind Kind_, unsigned Index_)
- : Node(KSyntheticTemplateParamName), Kind(Kind_), Index(Index_) {}
- template<typename Fn> void match(Fn F) const { F(Kind, Index); }
- void printLeft(OutputBuffer &OB) const override {
- switch (Kind) {
- case TemplateParamKind::Type:
- OB += "$T";
- break;
- case TemplateParamKind::NonType:
- OB += "$N";
- break;
- case TemplateParamKind::Template:
- OB += "$TT";
- break;
- }
- if (Index > 0)
- OB << Index - 1;
- }
- };
- /// A template type parameter declaration, 'typename T'.
- class TypeTemplateParamDecl final : public Node {
- Node *Name;
- public:
- TypeTemplateParamDecl(Node *Name_)
- : Node(KTypeTemplateParamDecl, Cache::Yes), Name(Name_) {}
- template<typename Fn> void match(Fn F) const { F(Name); }
- void printLeft(OutputBuffer &OB) const override { OB += "typename "; }
- void printRight(OutputBuffer &OB) const override { Name->print(OB); }
- };
- /// A non-type template parameter declaration, 'int N'.
- class NonTypeTemplateParamDecl final : public Node {
- Node *Name;
- Node *Type;
- public:
- NonTypeTemplateParamDecl(Node *Name_, Node *Type_)
- : Node(KNonTypeTemplateParamDecl, Cache::Yes), Name(Name_), Type(Type_) {}
- template<typename Fn> void match(Fn F) const { F(Name, Type); }
- void printLeft(OutputBuffer &OB) const override {
- Type->printLeft(OB);
- if (!Type->hasRHSComponent(OB))
- OB += " ";
- }
- void printRight(OutputBuffer &OB) const override {
- Name->print(OB);
- Type->printRight(OB);
- }
- };
- /// A template template parameter declaration,
- /// 'template<typename T> typename N'.
- class TemplateTemplateParamDecl final : public Node {
- Node *Name;
- NodeArray Params;
- public:
- TemplateTemplateParamDecl(Node *Name_, NodeArray Params_)
- : Node(KTemplateTemplateParamDecl, Cache::Yes), Name(Name_),
- Params(Params_) {}
- template<typename Fn> void match(Fn F) const { F(Name, Params); }
- void printLeft(OutputBuffer &OB) const override {
- OB += "template<";
- Params.printWithComma(OB);
- OB += "> typename ";
- }
- void printRight(OutputBuffer &OB) const override { Name->print(OB); }
- };
- /// A template parameter pack declaration, 'typename ...T'.
- class TemplateParamPackDecl final : public Node {
- Node *Param;
- public:
- TemplateParamPackDecl(Node *Param_)
- : Node(KTemplateParamPackDecl, Cache::Yes), Param(Param_) {}
- template<typename Fn> void match(Fn F) const { F(Param); }
- void printLeft(OutputBuffer &OB) const override {
- Param->printLeft(OB);
- OB += "...";
- }
- void printRight(OutputBuffer &OB) const override { Param->printRight(OB); }
- };
- /// An unexpanded parameter pack (either in the expression or type context). If
- /// this AST is correct, this node will have a ParameterPackExpansion node above
- /// it.
- ///
- /// This node is created when some <template-args> are found that apply to an
- /// <encoding>, and is stored in the TemplateParams table. In order for this to
- /// appear in the final AST, it has to referenced via a <template-param> (ie,
- /// T_).
- class ParameterPack final : public Node {
- NodeArray Data;
- // Setup OutputBuffer for a pack expansion, unless we're already expanding
- // one.
- void initializePackExpansion(OutputBuffer &OB) const {
- if (OB.CurrentPackMax == std::numeric_limits<unsigned>::max()) {
- OB.CurrentPackMax = static_cast<unsigned>(Data.size());
- OB.CurrentPackIndex = 0;
- }
- }
- public:
- ParameterPack(NodeArray Data_) : Node(KParameterPack), Data(Data_) {
- ArrayCache = FunctionCache = RHSComponentCache = Cache::Unknown;
- if (std::all_of(Data.begin(), Data.end(), [](Node* P) {
- return P->ArrayCache == Cache::No;
- }))
- ArrayCache = Cache::No;
- if (std::all_of(Data.begin(), Data.end(), [](Node* P) {
- return P->FunctionCache == Cache::No;
- }))
- FunctionCache = Cache::No;
- if (std::all_of(Data.begin(), Data.end(), [](Node* P) {
- return P->RHSComponentCache == Cache::No;
- }))
- RHSComponentCache = Cache::No;
- }
- template<typename Fn> void match(Fn F) const { F(Data); }
- bool hasRHSComponentSlow(OutputBuffer &OB) const override {
- initializePackExpansion(OB);
- size_t Idx = OB.CurrentPackIndex;
- return Idx < Data.size() && Data[Idx]->hasRHSComponent(OB);
- }
- bool hasArraySlow(OutputBuffer &OB) const override {
- initializePackExpansion(OB);
- size_t Idx = OB.CurrentPackIndex;
- return Idx < Data.size() && Data[Idx]->hasArray(OB);
- }
- bool hasFunctionSlow(OutputBuffer &OB) const override {
- initializePackExpansion(OB);
- size_t Idx = OB.CurrentPackIndex;
- return Idx < Data.size() && Data[Idx]->hasFunction(OB);
- }
- const Node *getSyntaxNode(OutputBuffer &OB) const override {
- initializePackExpansion(OB);
- size_t Idx = OB.CurrentPackIndex;
- return Idx < Data.size() ? Data[Idx]->getSyntaxNode(OB) : this;
- }
- void printLeft(OutputBuffer &OB) const override {
- initializePackExpansion(OB);
- size_t Idx = OB.CurrentPackIndex;
- if (Idx < Data.size())
- Data[Idx]->printLeft(OB);
- }
- void printRight(OutputBuffer &OB) const override {
- initializePackExpansion(OB);
- size_t Idx = OB.CurrentPackIndex;
- if (Idx < Data.size())
- Data[Idx]->printRight(OB);
- }
- };
- /// A variadic template argument. This node represents an occurrence of
- /// J<something>E in some <template-args>. It isn't itself unexpanded, unless
- /// one of it's Elements is. The parser inserts a ParameterPack into the
- /// TemplateParams table if the <template-args> this pack belongs to apply to an
- /// <encoding>.
- class TemplateArgumentPack final : public Node {
- NodeArray Elements;
- public:
- TemplateArgumentPack(NodeArray Elements_)
- : Node(KTemplateArgumentPack), Elements(Elements_) {}
- template<typename Fn> void match(Fn F) const { F(Elements); }
- NodeArray getElements() const { return Elements; }
- void printLeft(OutputBuffer &OB) const override {
- Elements.printWithComma(OB);
- }
- };
- /// A pack expansion. Below this node, there are some unexpanded ParameterPacks
- /// which each have Child->ParameterPackSize elements.
- class ParameterPackExpansion final : public Node {
- const Node *Child;
- public:
- ParameterPackExpansion(const Node *Child_)
- : Node(KParameterPackExpansion), Child(Child_) {}
- template<typename Fn> void match(Fn F) const { F(Child); }
- const Node *getChild() const { return Child; }
- void printLeft(OutputBuffer &OB) const override {
- constexpr unsigned Max = std::numeric_limits<unsigned>::max();
- SwapAndRestore<unsigned> SavePackIdx(OB.CurrentPackIndex, Max);
- SwapAndRestore<unsigned> SavePackMax(OB.CurrentPackMax, Max);
- size_t StreamPos = OB.getCurrentPosition();
- // Print the first element in the pack. If Child contains a ParameterPack,
- // it will set up S.CurrentPackMax and print the first element.
- Child->print(OB);
- // No ParameterPack was found in Child. This can occur if we've found a pack
- // expansion on a <function-param>.
- if (OB.CurrentPackMax == Max) {
- OB += "...";
- return;
- }
- // We found a ParameterPack, but it has no elements. Erase whatever we may
- // of printed.
- if (OB.CurrentPackMax == 0) {
- OB.setCurrentPosition(StreamPos);
- return;
- }
- // Else, iterate through the rest of the elements in the pack.
- for (unsigned I = 1, E = OB.CurrentPackMax; I < E; ++I) {
- OB += ", ";
- OB.CurrentPackIndex = I;
- Child->print(OB);
- }
- }
- };
- class TemplateArgs final : public Node {
- NodeArray Params;
- public:
- TemplateArgs(NodeArray Params_) : Node(KTemplateArgs), Params(Params_) {}
- template<typename Fn> void match(Fn F) const { F(Params); }
- NodeArray getParams() { return Params; }
- void printLeft(OutputBuffer &OB) const override {
- OB += "<";
- Params.printWithComma(OB);
- if (OB.back() == '>')
- OB += " ";
- OB += ">";
- }
- };
- /// A forward-reference to a template argument that was not known at the point
- /// where the template parameter name was parsed in a mangling.
- ///
- /// This is created when demangling the name of a specialization of a
- /// conversion function template:
- ///
- /// \code
- /// struct A {
- /// template<typename T> operator T*();
- /// };
- /// \endcode
- ///
- /// When demangling a specialization of the conversion function template, we
- /// encounter the name of the template (including the \c T) before we reach
- /// the template argument list, so we cannot substitute the parameter name
- /// for the corresponding argument while parsing. Instead, we create a
- /// \c ForwardTemplateReference node that is resolved after we parse the
- /// template arguments.
- struct ForwardTemplateReference : Node {
- size_t Index;
- Node *Ref = nullptr;
- // If we're currently printing this node. It is possible (though invalid) for
- // a forward template reference to refer to itself via a substitution. This
- // creates a cyclic AST, which will stack overflow printing. To fix this, bail
- // out if more than one print* function is active.
- mutable bool Printing = false;
- ForwardTemplateReference(size_t Index_)
- : Node(KForwardTemplateReference, Cache::Unknown, Cache::Unknown,
- Cache::Unknown),
- Index(Index_) {}
- // We don't provide a matcher for these, because the value of the node is
- // not determined by its construction parameters, and it generally needs
- // special handling.
- template<typename Fn> void match(Fn F) const = delete;
- bool hasRHSComponentSlow(OutputBuffer &OB) const override {
- if (Printing)
- return false;
- SwapAndRestore<bool> SavePrinting(Printing, true);
- return Ref->hasRHSComponent(OB);
- }
- bool hasArraySlow(OutputBuffer &OB) const override {
- if (Printing)
- return false;
- SwapAndRestore<bool> SavePrinting(Printing, true);
- return Ref->hasArray(OB);
- }
- bool hasFunctionSlow(OutputBuffer &OB) const override {
- if (Printing)
- return false;
- SwapAndRestore<bool> SavePrinting(Printing, true);
- return Ref->hasFunction(OB);
- }
- const Node *getSyntaxNode(OutputBuffer &OB) const override {
- if (Printing)
- return this;
- SwapAndRestore<bool> SavePrinting(Printing, true);
- return Ref->getSyntaxNode(OB);
- }
- void printLeft(OutputBuffer &OB) const override {
- if (Printing)
- return;
- SwapAndRestore<bool> SavePrinting(Printing, true);
- Ref->printLeft(OB);
- }
- void printRight(OutputBuffer &OB) const override {
- if (Printing)
- return;
- SwapAndRestore<bool> SavePrinting(Printing, true);
- Ref->printRight(OB);
- }
- };
- struct NameWithTemplateArgs : Node {
- // name<template_args>
- Node *Name;
- Node *TemplateArgs;
- NameWithTemplateArgs(Node *Name_, Node *TemplateArgs_)
- : Node(KNameWithTemplateArgs), Name(Name_), TemplateArgs(TemplateArgs_) {}
- template<typename Fn> void match(Fn F) const { F(Name, TemplateArgs); }
- StringView getBaseName() const override { return Name->getBaseName(); }
- void printLeft(OutputBuffer &OB) const override {
- Name->print(OB);
- TemplateArgs->print(OB);
- }
- };
- class GlobalQualifiedName final : public Node {
- Node *Child;
- public:
- GlobalQualifiedName(Node* Child_)
- : Node(KGlobalQualifiedName), Child(Child_) {}
- template<typename Fn> void match(Fn F) const { F(Child); }
- StringView getBaseName() const override { return Child->getBaseName(); }
- void printLeft(OutputBuffer &OB) const override {
- OB += "::";
- Child->print(OB);
- }
- };
- struct StdQualifiedName : Node {
- Node *Child;
- StdQualifiedName(Node *Child_) : Node(KStdQualifiedName), Child(Child_) {}
- template<typename Fn> void match(Fn F) const { F(Child); }
- StringView getBaseName() const override { return Child->getBaseName(); }
- void printLeft(OutputBuffer &OB) const override {
- OB += "std::";
- Child->print(OB);
- }
- };
- enum class SpecialSubKind {
- allocator,
- basic_string,
- string,
- istream,
- ostream,
- iostream,
- };
- class ExpandedSpecialSubstitution final : public Node {
- SpecialSubKind SSK;
- public:
- ExpandedSpecialSubstitution(SpecialSubKind SSK_)
- : Node(KExpandedSpecialSubstitution), SSK(SSK_) {}
- template<typename Fn> void match(Fn F) const { F(SSK); }
- StringView getBaseName() const override {
- switch (SSK) {
- case SpecialSubKind::allocator:
- return StringView("allocator");
- case SpecialSubKind::basic_string:
- return StringView("basic_string");
- case SpecialSubKind::string:
- return StringView("basic_string");
- case SpecialSubKind::istream:
- return StringView("basic_istream");
- case SpecialSubKind::ostream:
- return StringView("basic_ostream");
- case SpecialSubKind::iostream:
- return StringView("basic_iostream");
- }
- DEMANGLE_UNREACHABLE;
- }
- void printLeft(OutputBuffer &OB) const override {
- switch (SSK) {
- case SpecialSubKind::allocator:
- OB += "std::allocator";
- break;
- case SpecialSubKind::basic_string:
- OB += "std::basic_string";
- break;
- case SpecialSubKind::string:
- OB += "std::basic_string<char, std::char_traits<char>, "
- "std::allocator<char> >";
- break;
- case SpecialSubKind::istream:
- OB += "std::basic_istream<char, std::char_traits<char> >";
- break;
- case SpecialSubKind::ostream:
- OB += "std::basic_ostream<char, std::char_traits<char> >";
- break;
- case SpecialSubKind::iostream:
- OB += "std::basic_iostream<char, std::char_traits<char> >";
- break;
- }
- }
- };
- class SpecialSubstitution final : public Node {
- public:
- SpecialSubKind SSK;
- SpecialSubstitution(SpecialSubKind SSK_)
- : Node(KSpecialSubstitution), SSK(SSK_) {}
- template<typename Fn> void match(Fn F) const { F(SSK); }
- StringView getBaseName() const override {
- switch (SSK) {
- case SpecialSubKind::allocator:
- return StringView("allocator");
- case SpecialSubKind::basic_string:
- return StringView("basic_string");
- case SpecialSubKind::string:
- return StringView("string");
- case SpecialSubKind::istream:
- return StringView("istream");
- case SpecialSubKind::ostream:
- return StringView("ostream");
- case SpecialSubKind::iostream:
- return StringView("iostream");
- }
- DEMANGLE_UNREACHABLE;
- }
- void printLeft(OutputBuffer &OB) const override {
- switch (SSK) {
- case SpecialSubKind::allocator:
- OB += "std::allocator";
- break;
- case SpecialSubKind::basic_string:
- OB += "std::basic_string";
- break;
- case SpecialSubKind::string:
- OB += "std::string";
- break;
- case SpecialSubKind::istream:
- OB += "std::istream";
- break;
- case SpecialSubKind::ostream:
- OB += "std::ostream";
- break;
- case SpecialSubKind::iostream:
- OB += "std::iostream";
- break;
- }
- }
- };
- class CtorDtorName final : public Node {
- const Node *Basename;
- const bool IsDtor;
- const int Variant;
- public:
- CtorDtorName(const Node *Basename_, bool IsDtor_, int Variant_)
- : Node(KCtorDtorName), Basename(Basename_), IsDtor(IsDtor_),
- Variant(Variant_) {}
- template<typename Fn> void match(Fn F) const { F(Basename, IsDtor, Variant); }
- void printLeft(OutputBuffer &OB) const override {
- if (IsDtor)
- OB += "~";
- OB += Basename->getBaseName();
- }
- };
- class DtorName : public Node {
- const Node *Base;
- public:
- DtorName(const Node *Base_) : Node(KDtorName), Base(Base_) {}
- template<typename Fn> void match(Fn F) const { F(Base); }
- void printLeft(OutputBuffer &OB) const override {
- OB += "~";
- Base->printLeft(OB);
- }
- };
- class UnnamedTypeName : public Node {
- const StringView Count;
- public:
- UnnamedTypeName(StringView Count_) : Node(KUnnamedTypeName), Count(Count_) {}
- template<typename Fn> void match(Fn F) const { F(Count); }
- void printLeft(OutputBuffer &OB) const override {
- OB += "'unnamed";
- OB += Count;
- OB += "\'";
- }
- };
- class ClosureTypeName : public Node {
- NodeArray TemplateParams;
- NodeArray Params;
- StringView Count;
- public:
- ClosureTypeName(NodeArray TemplateParams_, NodeArray Params_,
- StringView Count_)
- : Node(KClosureTypeName), TemplateParams(TemplateParams_),
- Params(Params_), Count(Count_) {}
- template<typename Fn> void match(Fn F) const {
- F(TemplateParams, Params, Count);
- }
- void printDeclarator(OutputBuffer &OB) const {
- if (!TemplateParams.empty()) {
- OB += "<";
- TemplateParams.printWithComma(OB);
- OB += ">";
- }
- OB += "(";
- Params.printWithComma(OB);
- OB += ")";
- }
- void printLeft(OutputBuffer &OB) const override {
- OB += "\'lambda";
- OB += Count;
- OB += "\'";
- printDeclarator(OB);
- }
- };
- class StructuredBindingName : public Node {
- NodeArray Bindings;
- public:
- StructuredBindingName(NodeArray Bindings_)
- : Node(KStructuredBindingName), Bindings(Bindings_) {}
- template<typename Fn> void match(Fn F) const { F(Bindings); }
- void printLeft(OutputBuffer &OB) const override {
- OB += '[';
- Bindings.printWithComma(OB);
- OB += ']';
- }
- };
- // -- Expression Nodes --
- class BinaryExpr : public Node {
- const Node *LHS;
- const StringView InfixOperator;
- const Node *RHS;
- public:
- BinaryExpr(const Node *LHS_, StringView InfixOperator_, const Node *RHS_)
- : Node(KBinaryExpr), LHS(LHS_), InfixOperator(InfixOperator_), RHS(RHS_) {
- }
- template<typename Fn> void match(Fn F) const { F(LHS, InfixOperator, RHS); }
- void printLeft(OutputBuffer &OB) const override {
- // might be a template argument expression, then we need to disambiguate
- // with parens.
- if (InfixOperator == ">")
- OB += "(";
- OB += "(";
- LHS->print(OB);
- OB += ") ";
- OB += InfixOperator;
- OB += " (";
- RHS->print(OB);
- OB += ")";
- if (InfixOperator == ">")
- OB += ")";
- }
- };
- class ArraySubscriptExpr : public Node {
- const Node *Op1;
- const Node *Op2;
- public:
- ArraySubscriptExpr(const Node *Op1_, const Node *Op2_)
- : Node(KArraySubscriptExpr), Op1(Op1_), Op2(Op2_) {}
- template<typename Fn> void match(Fn F) const { F(Op1, Op2); }
- void printLeft(OutputBuffer &OB) const override {
- OB += "(";
- Op1->print(OB);
- OB += ")[";
- Op2->print(OB);
- OB += "]";
- }
- };
- class PostfixExpr : public Node {
- const Node *Child;
- const StringView Operator;
- public:
- PostfixExpr(const Node *Child_, StringView Operator_)
- : Node(KPostfixExpr), Child(Child_), Operator(Operator_) {}
- template<typename Fn> void match(Fn F) const { F(Child, Operator); }
- void printLeft(OutputBuffer &OB) const override {
- OB += "(";
- Child->print(OB);
- OB += ")";
- OB += Operator;
- }
- };
- class ConditionalExpr : public Node {
- const Node *Cond;
- const Node *Then;
- const Node *Else;
- public:
- ConditionalExpr(const Node *Cond_, const Node *Then_, const Node *Else_)
- : Node(KConditionalExpr), Cond(Cond_), Then(Then_), Else(Else_) {}
- template<typename Fn> void match(Fn F) const { F(Cond, Then, Else); }
- void printLeft(OutputBuffer &OB) const override {
- OB += "(";
- Cond->print(OB);
- OB += ") ? (";
- Then->print(OB);
- OB += ") : (";
- Else->print(OB);
- OB += ")";
- }
- };
- class MemberExpr : public Node {
- const Node *LHS;
- const StringView Kind;
- const Node *RHS;
- public:
- MemberExpr(const Node *LHS_, StringView Kind_, const Node *RHS_)
- : Node(KMemberExpr), LHS(LHS_), Kind(Kind_), RHS(RHS_) {}
- template<typename Fn> void match(Fn F) const { F(LHS, Kind, RHS); }
- void printLeft(OutputBuffer &OB) const override {
- LHS->print(OB);
- OB += Kind;
- RHS->print(OB);
- }
- };
- class SubobjectExpr : public Node {
- const Node *Type;
- const Node *SubExpr;
- StringView Offset;
- NodeArray UnionSelectors;
- bool OnePastTheEnd;
- public:
- SubobjectExpr(const Node *Type_, const Node *SubExpr_, StringView Offset_,
- NodeArray UnionSelectors_, bool OnePastTheEnd_)
- : Node(KSubobjectExpr), Type(Type_), SubExpr(SubExpr_), Offset(Offset_),
- UnionSelectors(UnionSelectors_), OnePastTheEnd(OnePastTheEnd_) {}
- template<typename Fn> void match(Fn F) const {
- F(Type, SubExpr, Offset, UnionSelectors, OnePastTheEnd);
- }
- void printLeft(OutputBuffer &OB) const override {
- SubExpr->print(OB);
- OB += ".<";
- Type->print(OB);
- OB += " at offset ";
- if (Offset.empty()) {
- OB += "0";
- } else if (Offset[0] == 'n') {
- OB += "-";
- OB += Offset.dropFront();
- } else {
- OB += Offset;
- }
- OB += ">";
- }
- };
- class EnclosingExpr : public Node {
- const StringView Prefix;
- const Node *Infix;
- const StringView Postfix;
- public:
- EnclosingExpr(StringView Prefix_, Node *Infix_, StringView Postfix_)
- : Node(KEnclosingExpr), Prefix(Prefix_), Infix(Infix_),
- Postfix(Postfix_) {}
- template<typename Fn> void match(Fn F) const { F(Prefix, Infix, Postfix); }
- void printLeft(OutputBuffer &OB) const override {
- OB += Prefix;
- Infix->print(OB);
- OB += Postfix;
- }
- };
- class CastExpr : public Node {
- // cast_kind<to>(from)
- const StringView CastKind;
- const Node *To;
- const Node *From;
- public:
- CastExpr(StringView CastKind_, const Node *To_, const Node *From_)
- : Node(KCastExpr), CastKind(CastKind_), To(To_), From(From_) {}
- template<typename Fn> void match(Fn F) const { F(CastKind, To, From); }
- void printLeft(OutputBuffer &OB) const override {
- OB += CastKind;
- OB += "<";
- To->printLeft(OB);
- OB += ">(";
- From->printLeft(OB);
- OB += ")";
- }
- };
- class SizeofParamPackExpr : public Node {
- const Node *Pack;
- public:
- SizeofParamPackExpr(const Node *Pack_)
- : Node(KSizeofParamPackExpr), Pack(Pack_) {}
- template<typename Fn> void match(Fn F) const { F(Pack); }
- void printLeft(OutputBuffer &OB) const override {
- OB += "sizeof...(";
- ParameterPackExpansion PPE(Pack);
- PPE.printLeft(OB);
- OB += ")";
- }
- };
- class CallExpr : public Node {
- const Node *Callee;
- NodeArray Args;
- public:
- CallExpr(const Node *Callee_, NodeArray Args_)
- : Node(KCallExpr), Callee(Callee_), Args(Args_) {}
- template<typename Fn> void match(Fn F) const { F(Callee, Args); }
- void printLeft(OutputBuffer &OB) const override {
- Callee->print(OB);
- OB += "(";
- Args.printWithComma(OB);
- OB += ")";
- }
- };
- class NewExpr : public Node {
- // new (expr_list) type(init_list)
- NodeArray ExprList;
- Node *Type;
- NodeArray InitList;
- bool IsGlobal; // ::operator new ?
- bool IsArray; // new[] ?
- public:
- NewExpr(NodeArray ExprList_, Node *Type_, NodeArray InitList_, bool IsGlobal_,
- bool IsArray_)
- : Node(KNewExpr), ExprList(ExprList_), Type(Type_), InitList(InitList_),
- IsGlobal(IsGlobal_), IsArray(IsArray_) {}
- template<typename Fn> void match(Fn F) const {
- F(ExprList, Type, InitList, IsGlobal, IsArray);
- }
- void printLeft(OutputBuffer &OB) const override {
- if (IsGlobal)
- OB += "::operator ";
- OB += "new";
- if (IsArray)
- OB += "[]";
- OB += ' ';
- if (!ExprList.empty()) {
- OB += "(";
- ExprList.printWithComma(OB);
- OB += ")";
- }
- Type->print(OB);
- if (!InitList.empty()) {
- OB += "(";
- InitList.printWithComma(OB);
- OB += ")";
- }
- }
- };
- class DeleteExpr : public Node {
- Node *Op;
- bool IsGlobal;
- bool IsArray;
- public:
- DeleteExpr(Node *Op_, bool IsGlobal_, bool IsArray_)
- : Node(KDeleteExpr), Op(Op_), IsGlobal(IsGlobal_), IsArray(IsArray_) {}
- template<typename Fn> void match(Fn F) const { F(Op, IsGlobal, IsArray); }
- void printLeft(OutputBuffer &OB) const override {
- if (IsGlobal)
- OB += "::";
- OB += "delete";
- if (IsArray)
- OB += "[] ";
- Op->print(OB);
- }
- };
- class PrefixExpr : public Node {
- StringView Prefix;
- Node *Child;
- public:
- PrefixExpr(StringView Prefix_, Node *Child_)
- : Node(KPrefixExpr), Prefix(Prefix_), Child(Child_) {}
- template<typename Fn> void match(Fn F) const { F(Prefix, Child); }
- void printLeft(OutputBuffer &OB) const override {
- OB += Prefix;
- OB += "(";
- Child->print(OB);
- OB += ")";
- }
- };
- class FunctionParam : public Node {
- StringView Number;
- public:
- FunctionParam(StringView Number_) : Node(KFunctionParam), Number(Number_) {}
- template<typename Fn> void match(Fn F) const { F(Number); }
- void printLeft(OutputBuffer &OB) const override {
- OB += "fp";
- OB += Number;
- }
- };
- class ConversionExpr : public Node {
- const Node *Type;
- NodeArray Expressions;
- public:
- ConversionExpr(const Node *Type_, NodeArray Expressions_)
- : Node(KConversionExpr), Type(Type_), Expressions(Expressions_) {}
- template<typename Fn> void match(Fn F) const { F(Type, Expressions); }
- void printLeft(OutputBuffer &OB) const override {
- OB += "(";
- Type->print(OB);
- OB += ")(";
- Expressions.printWithComma(OB);
- OB += ")";
- }
- };
- class PointerToMemberConversionExpr : public Node {
- const Node *Type;
- const Node *SubExpr;
- StringView Offset;
- public:
- PointerToMemberConversionExpr(const Node *Type_, const Node *SubExpr_,
- StringView Offset_)
- : Node(KPointerToMemberConversionExpr), Type(Type_), SubExpr(SubExpr_),
- Offset(Offset_) {}
- template<typename Fn> void match(Fn F) const { F(Type, SubExpr, Offset); }
- void printLeft(OutputBuffer &OB) const override {
- OB += "(";
- Type->print(OB);
- OB += ")(";
- SubExpr->print(OB);
- OB += ")";
- }
- };
- class InitListExpr : public Node {
- const Node *Ty;
- NodeArray Inits;
- public:
- InitListExpr(const Node *Ty_, NodeArray Inits_)
- : Node(KInitListExpr), Ty(Ty_), Inits(Inits_) {}
- template<typename Fn> void match(Fn F) const { F(Ty, Inits); }
- void printLeft(OutputBuffer &OB) const override {
- if (Ty)
- Ty->print(OB);
- OB += '{';
- Inits.printWithComma(OB);
- OB += '}';
- }
- };
- class BracedExpr : public Node {
- const Node *Elem;
- const Node *Init;
- bool IsArray;
- public:
- BracedExpr(const Node *Elem_, const Node *Init_, bool IsArray_)
- : Node(KBracedExpr), Elem(Elem_), Init(Init_), IsArray(IsArray_) {}
- template<typename Fn> void match(Fn F) const { F(Elem, Init, IsArray); }
- void printLeft(OutputBuffer &OB) const override {
- if (IsArray) {
- OB += '[';
- Elem->print(OB);
- OB += ']';
- } else {
- OB += '.';
- Elem->print(OB);
- }
- if (Init->getKind() != KBracedExpr && Init->getKind() != KBracedRangeExpr)
- OB += " = ";
- Init->print(OB);
- }
- };
- class BracedRangeExpr : public Node {
- const Node *First;
- const Node *Last;
- const Node *Init;
- public:
- BracedRangeExpr(const Node *First_, const Node *Last_, const Node *Init_)
- : Node(KBracedRangeExpr), First(First_), Last(Last_), Init(Init_) {}
- template<typename Fn> void match(Fn F) const { F(First, Last, Init); }
- void printLeft(OutputBuffer &OB) const override {
- OB += '[';
- First->print(OB);
- OB += " ... ";
- Last->print(OB);
- OB += ']';
- if (Init->getKind() != KBracedExpr && Init->getKind() != KBracedRangeExpr)
- OB += " = ";
- Init->print(OB);
- }
- };
- class FoldExpr : public Node {
- const Node *Pack, *Init;
- StringView OperatorName;
- bool IsLeftFold;
- public:
- FoldExpr(bool IsLeftFold_, StringView OperatorName_, const Node *Pack_,
- const Node *Init_)
- : Node(KFoldExpr), Pack(Pack_), Init(Init_), OperatorName(OperatorName_),
- IsLeftFold(IsLeftFold_) {}
- template<typename Fn> void match(Fn F) const {
- F(IsLeftFold, OperatorName, Pack, Init);
- }
- void printLeft(OutputBuffer &OB) const override {
- auto PrintPack = [&] {
- OB += '(';
- ParameterPackExpansion(Pack).print(OB);
- OB += ')';
- };
- OB += '(';
- if (IsLeftFold) {
- // init op ... op pack
- if (Init != nullptr) {
- Init->print(OB);
- OB += ' ';
- OB += OperatorName;
- OB += ' ';
- }
- // ... op pack
- OB += "... ";
- OB += OperatorName;
- OB += ' ';
- PrintPack();
- } else { // !IsLeftFold
- // pack op ...
- PrintPack();
- OB += ' ';
- OB += OperatorName;
- OB += " ...";
- // pack op ... op init
- if (Init != nullptr) {
- OB += ' ';
- OB += OperatorName;
- OB += ' ';
- Init->print(OB);
- }
- }
- OB += ')';
- }
- };
- class ThrowExpr : public Node {
- const Node *Op;
- public:
- ThrowExpr(const Node *Op_) : Node(KThrowExpr), Op(Op_) {}
- template<typename Fn> void match(Fn F) const { F(Op); }
- void printLeft(OutputBuffer &OB) const override {
- OB += "throw ";
- Op->print(OB);
- }
- };
- class BoolExpr : public Node {
- bool Value;
- public:
- BoolExpr(bool Value_) : Node(KBoolExpr), Value(Value_) {}
- template<typename Fn> void match(Fn F) const { F(Value); }
- void printLeft(OutputBuffer &OB) const override {
- OB += Value ? StringView("true") : StringView("false");
- }
- };
- class StringLiteral : public Node {
- const Node *Type;
- public:
- StringLiteral(const Node *Type_) : Node(KStringLiteral), Type(Type_) {}
- template<typename Fn> void match(Fn F) const { F(Type); }
- void printLeft(OutputBuffer &OB) const override {
- OB += "\"<";
- Type->print(OB);
- OB += ">\"";
- }
- };
- class LambdaExpr : public Node {
- const Node *Type;
- public:
- LambdaExpr(const Node *Type_) : Node(KLambdaExpr), Type(Type_) {}
- template<typename Fn> void match(Fn F) const { F(Type); }
- void printLeft(OutputBuffer &OB) const override {
- OB += "[]";
- if (Type->getKind() == KClosureTypeName)
- static_cast<const ClosureTypeName *>(Type)->printDeclarator(OB);
- OB += "{...}";
- }
- };
- class EnumLiteral : public Node {
- // ty(integer)
- const Node *Ty;
- StringView Integer;
- public:
- EnumLiteral(const Node *Ty_, StringView Integer_)
- : Node(KEnumLiteral), Ty(Ty_), Integer(Integer_) {}
- template<typename Fn> void match(Fn F) const { F(Ty, Integer); }
- void printLeft(OutputBuffer &OB) const override {
- OB << "(";
- Ty->print(OB);
- OB << ")";
- if (Integer[0] == 'n')
- OB << "-" << Integer.dropFront(1);
- else
- OB << Integer;
- }
- };
- class IntegerLiteral : public Node {
- StringView Type;
- StringView Value;
- public:
- IntegerLiteral(StringView Type_, StringView Value_)
- : Node(KIntegerLiteral), Type(Type_), Value(Value_) {}
- template<typename Fn> void match(Fn F) const { F(Type, Value); }
- void printLeft(OutputBuffer &OB) const override {
- if (Type.size() > 3) {
- OB += "(";
- OB += Type;
- OB += ")";
- }
- if (Value[0] == 'n') {
- OB += "-";
- OB += Value.dropFront(1);
- } else
- OB += Value;
- if (Type.size() <= 3)
- OB += Type;
- }
- };
- template <class Float> struct FloatData;
- namespace float_literal_impl {
- constexpr Node::Kind getFloatLiteralKind(float *) {
- return Node::KFloatLiteral;
- }
- constexpr Node::Kind getFloatLiteralKind(double *) {
- return Node::KDoubleLiteral;
- }
- constexpr Node::Kind getFloatLiteralKind(long double *) {
- return Node::KLongDoubleLiteral;
- }
- }
- template <class Float> class FloatLiteralImpl : public Node {
- const StringView Contents;
- static constexpr Kind KindForClass =
- float_literal_impl::getFloatLiteralKind((Float *)nullptr);
- public:
- FloatLiteralImpl(StringView Contents_)
- : Node(KindForClass), Contents(Contents_) {}
- template<typename Fn> void match(Fn F) const { F(Contents); }
- void printLeft(OutputBuffer &OB) const override {
- const char *first = Contents.begin();
- const char *last = Contents.end() + 1;
- const size_t N = FloatData<Float>::mangled_size;
- if (static_cast<std::size_t>(last - first) > N) {
- last = first + N;
- union {
- Float value;
- char buf[sizeof(Float)];
- };
- const char *t = first;
- char *e = buf;
- for (; t != last; ++t, ++e) {
- unsigned d1 = isdigit(*t) ? static_cast<unsigned>(*t - '0')
- : static_cast<unsigned>(*t - 'a' + 10);
- ++t;
- unsigned d0 = isdigit(*t) ? static_cast<unsigned>(*t - '0')
- : static_cast<unsigned>(*t - 'a' + 10);
- *e = static_cast<char>((d1 << 4) + d0);
- }
- #if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__
- std::reverse(buf, e);
- #endif
- char num[FloatData<Float>::max_demangled_size] = {0};
- int n = snprintf(num, sizeof(num), FloatData<Float>::spec, value);
- OB += StringView(num, num + n);
- }
- }
- };
- using FloatLiteral = FloatLiteralImpl<float>;
- using DoubleLiteral = FloatLiteralImpl<double>;
- using LongDoubleLiteral = FloatLiteralImpl<long double>;
- /// Visit the node. Calls \c F(P), where \c P is the node cast to the
- /// appropriate derived class.
- template<typename Fn>
- void Node::visit(Fn F) const {
- switch (K) {
- #define CASE(X) case K ## X: return F(static_cast<const X*>(this));
- FOR_EACH_NODE_KIND(CASE)
- #undef CASE
- }
- assert(0 && "unknown mangling node kind");
- }
- /// Determine the kind of a node from its type.
- template<typename NodeT> struct NodeKind;
- #define SPECIALIZATION(X) \
- template<> struct NodeKind<X> { \
- static constexpr Node::Kind Kind = Node::K##X; \
- static constexpr const char *name() { return #X; } \
- };
- FOR_EACH_NODE_KIND(SPECIALIZATION)
- #undef SPECIALIZATION
- #undef FOR_EACH_NODE_KIND
- template <typename Derived, typename Alloc> struct AbstractManglingParser {
- const char *First;
- const char *Last;
- // Name stack, this is used by the parser to hold temporary names that were
- // parsed. The parser collapses multiple names into new nodes to construct
- // the AST. Once the parser is finished, names.size() == 1.
- PODSmallVector<Node *, 32> Names;
- // Substitution table. Itanium supports name substitutions as a means of
- // compression. The string "S42_" refers to the 44nd entry (base-36) in this
- // table.
- PODSmallVector<Node *, 32> Subs;
- using TemplateParamList = PODSmallVector<Node *, 8>;
- class ScopedTemplateParamList {
- AbstractManglingParser *Parser;
- size_t OldNumTemplateParamLists;
- TemplateParamList Params;
- public:
- ScopedTemplateParamList(AbstractManglingParser *TheParser)
- : Parser(TheParser),
- OldNumTemplateParamLists(TheParser->TemplateParams.size()) {
- Parser->TemplateParams.push_back(&Params);
- }
- ~ScopedTemplateParamList() {
- assert(Parser->TemplateParams.size() >= OldNumTemplateParamLists);
- Parser->TemplateParams.dropBack(OldNumTemplateParamLists);
- }
- };
- // Template parameter table. Like the above, but referenced like "T42_".
- // This has a smaller size compared to Subs and Names because it can be
- // stored on the stack.
- TemplateParamList OuterTemplateParams;
- // Lists of template parameters indexed by template parameter depth,
- // referenced like "TL2_4_". If nonempty, element 0 is always
- // OuterTemplateParams; inner elements are always template parameter lists of
- // lambda expressions. For a generic lambda with no explicit template
- // parameter list, the corresponding parameter list pointer will be null.
- PODSmallVector<TemplateParamList *, 4> TemplateParams;
- // Set of unresolved forward <template-param> references. These can occur in a
- // conversion operator's type, and are resolved in the enclosing <encoding>.
- PODSmallVector<ForwardTemplateReference *, 4> ForwardTemplateRefs;
- bool TryToParseTemplateArgs = true;
- bool PermitForwardTemplateReferences = false;
- size_t ParsingLambdaParamsAtLevel = (size_t)-1;
- unsigned NumSyntheticTemplateParameters[3] = {};
- Alloc ASTAllocator;
- AbstractManglingParser(const char *First_, const char *Last_)
- : First(First_), Last(Last_) {}
- Derived &getDerived() { return static_cast<Derived &>(*this); }
- void reset(const char *First_, const char *Last_) {
- First = First_;
- Last = Last_;
- Names.clear();
- Subs.clear();
- TemplateParams.clear();
- ParsingLambdaParamsAtLevel = (size_t)-1;
- TryToParseTemplateArgs = true;
- PermitForwardTemplateReferences = false;
- for (int I = 0; I != 3; ++I)
- NumSyntheticTemplateParameters[I] = 0;
- ASTAllocator.reset();
- }
- template <class T, class... Args> Node *make(Args &&... args) {
- return ASTAllocator.template makeNode<T>(std::forward<Args>(args)...);
- }
- template <class It> NodeArray makeNodeArray(It begin, It end) {
- size_t sz = static_cast<size_t>(end - begin);
- void *mem = ASTAllocator.allocateNodeArray(sz);
- Node **data = new (mem) Node *[sz];
- std::copy(begin, end, data);
- return NodeArray(data, sz);
- }
- NodeArray popTrailingNodeArray(size_t FromPosition) {
- assert(FromPosition <= Names.size());
- NodeArray res =
- makeNodeArray(Names.begin() + (long)FromPosition, Names.end());
- Names.dropBack(FromPosition);
- return res;
- }
- bool consumeIf(StringView S) {
- if (StringView(First, Last).startsWith(S)) {
- First += S.size();
- return true;
- }
- return false;
- }
- bool consumeIf(char C) {
- if (First != Last && *First == C) {
- ++First;
- return true;
- }
- return false;
- }
- char consume() { return First != Last ? *First++ : '\0'; }
- char look(unsigned Lookahead = 0) const {
- if (static_cast<size_t>(Last - First) <= Lookahead)
- return '\0';
- return First[Lookahead];
- }
- size_t numLeft() const { return static_cast<size_t>(Last - First); }
- StringView parseNumber(bool AllowNegative = false);
- Qualifiers parseCVQualifiers();
- bool parsePositiveInteger(size_t *Out);
- StringView parseBareSourceName();
- bool parseSeqId(size_t *Out);
- Node *parseSubstitution();
- Node *parseTemplateParam();
- Node *parseTemplateParamDecl();
- Node *parseTemplateArgs(bool TagTemplates = false);
- Node *parseTemplateArg();
- /// Parse the <expr> production.
- Node *parseExpr();
- Node *parsePrefixExpr(StringView Kind);
- Node *parseBinaryExpr(StringView Kind);
- Node *parseIntegerLiteral(StringView Lit);
- Node *parseExprPrimary();
- template <class Float> Node *parseFloatingLiteral();
- Node *parseFunctionParam();
- Node *parseNewExpr();
- Node *parseConversionExpr();
- Node *parseBracedExpr();
- Node *parseFoldExpr();
- Node *parsePointerToMemberConversionExpr();
- Node *parseSubobjectExpr();
- /// Parse the <type> production.
- Node *parseType();
- Node *parseFunctionType();
- Node *parseVectorType();
- Node *parseDecltype();
- Node *parseArrayType();
- Node *parsePointerToMemberType();
- Node *parseClassEnumType();
- Node *parseQualifiedType();
- Node *parseEncoding();
- bool parseCallOffset();
- Node *parseSpecialName();
- /// Holds some extra information about a <name> that is being parsed. This
- /// information is only pertinent if the <name> refers to an <encoding>.
- struct NameState {
- bool CtorDtorConversion = false;
- bool EndsWithTemplateArgs = false;
- Qualifiers CVQualifiers = QualNone;
- FunctionRefQual ReferenceQualifier = FrefQualNone;
- size_t ForwardTemplateRefsBegin;
- NameState(AbstractManglingParser *Enclosing)
- : ForwardTemplateRefsBegin(Enclosing->ForwardTemplateRefs.size()) {}
- };
- bool resolveForwardTemplateRefs(NameState &State) {
- size_t I = State.ForwardTemplateRefsBegin;
- size_t E = ForwardTemplateRefs.size();
- for (; I < E; ++I) {
- size_t Idx = ForwardTemplateRefs[I]->Index;
- if (TemplateParams.empty() || !TemplateParams[0] ||
- Idx >= TemplateParams[0]->size())
- return true;
- ForwardTemplateRefs[I]->Ref = (*TemplateParams[0])[Idx];
- }
- ForwardTemplateRefs.dropBack(State.ForwardTemplateRefsBegin);
- return false;
- }
- /// Parse the <name> production>
- Node *parseName(NameState *State = nullptr);
- Node *parseLocalName(NameState *State);
- Node *parseOperatorName(NameState *State);
- Node *parseUnqualifiedName(NameState *State);
- Node *parseUnnamedTypeName(NameState *State);
- Node *parseSourceName(NameState *State);
- Node *parseUnscopedName(NameState *State);
- Node *parseNestedName(NameState *State);
- Node *parseCtorDtorName(Node *&SoFar, NameState *State);
- Node *parseAbiTags(Node *N);
- /// Parse the <unresolved-name> production.
- Node *parseUnresolvedName();
- Node *parseSimpleId();
- Node *parseBaseUnresolvedName();
- Node *parseUnresolvedType();
- Node *parseDestructorName();
- /// Top-level entry point into the parser.
- Node *parse();
- };
- const char* parse_discriminator(const char* first, const char* last);
- // <name> ::= <nested-name> // N
- // ::= <local-name> # See Scope Encoding below // Z
- // ::= <unscoped-template-name> <template-args>
- // ::= <unscoped-name>
- //
- // <unscoped-template-name> ::= <unscoped-name>
- // ::= <substitution>
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseName(NameState *State) {
- consumeIf('L'); // extension
- if (look() == 'N')
- return getDerived().parseNestedName(State);
- if (look() == 'Z')
- return getDerived().parseLocalName(State);
- Node *Result = nullptr;
- bool IsSubst = look() == 'S' && look(1) != 't';
- if (IsSubst) {
- // A substitution must lead to:
- // ::= <unscoped-template-name> <template-args>
- Result = getDerived().parseSubstitution();
- } else {
- // An unscoped name can be one of:
- // ::= <unscoped-name>
- // ::= <unscoped-template-name> <template-args>
- Result = getDerived().parseUnscopedName(State);
- }
- if (Result == nullptr)
- return nullptr;
- if (look() == 'I') {
- // ::= <unscoped-template-name> <template-args>
- if (!IsSubst)
- // An unscoped-template-name is substitutable.
- Subs.push_back(Result);
- Node *TA = getDerived().parseTemplateArgs(State != nullptr);
- if (TA == nullptr)
- return nullptr;
- if (State)
- State->EndsWithTemplateArgs = true;
- Result = make<NameWithTemplateArgs>(Result, TA);
- } else if (IsSubst) {
- // The substitution case must be followed by <template-args>.
- return nullptr;
- }
- return Result;
- }
- // <local-name> := Z <function encoding> E <entity name> [<discriminator>]
- // := Z <function encoding> E s [<discriminator>]
- // := Z <function encoding> Ed [ <parameter number> ] _ <entity name>
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseLocalName(NameState *State) {
- if (!consumeIf('Z'))
- return nullptr;
- Node *Encoding = getDerived().parseEncoding();
- if (Encoding == nullptr || !consumeIf('E'))
- return nullptr;
- if (consumeIf('s')) {
- First = parse_discriminator(First, Last);
- auto *StringLitName = make<NameType>("string literal");
- if (!StringLitName)
- return nullptr;
- return make<LocalName>(Encoding, StringLitName);
- }
- if (consumeIf('d')) {
- parseNumber(true);
- if (!consumeIf('_'))
- return nullptr;
- Node *N = getDerived().parseName(State);
- if (N == nullptr)
- return nullptr;
- return make<LocalName>(Encoding, N);
- }
- Node *Entity = getDerived().parseName(State);
- if (Entity == nullptr)
- return nullptr;
- First = parse_discriminator(First, Last);
- return make<LocalName>(Encoding, Entity);
- }
- // <unscoped-name> ::= <unqualified-name>
- // ::= St <unqualified-name> # ::std::
- // extension ::= StL<unqualified-name>
- template <typename Derived, typename Alloc>
- Node *
- AbstractManglingParser<Derived, Alloc>::parseUnscopedName(NameState *State) {
- bool IsStd = consumeIf("St");
- if (IsStd)
- consumeIf('L');
- Node *Result = getDerived().parseUnqualifiedName(State);
- if (Result == nullptr)
- return nullptr;
- if (IsStd)
- Result = make<StdQualifiedName>(Result);
- return Result;
- }
- // <unqualified-name> ::= <operator-name> [abi-tags]
- // ::= <ctor-dtor-name>
- // ::= <source-name>
- // ::= <unnamed-type-name>
- // ::= DC <source-name>+ E # structured binding declaration
- template <typename Derived, typename Alloc>
- Node *
- AbstractManglingParser<Derived, Alloc>::parseUnqualifiedName(NameState *State) {
- // <ctor-dtor-name>s are special-cased in parseNestedName().
- Node *Result;
- if (look() == 'U')
- Result = getDerived().parseUnnamedTypeName(State);
- else if (look() >= '1' && look() <= '9')
- Result = getDerived().parseSourceName(State);
- else if (consumeIf("DC")) {
- size_t BindingsBegin = Names.size();
- do {
- Node *Binding = getDerived().parseSourceName(State);
- if (Binding == nullptr)
- return nullptr;
- Names.push_back(Binding);
- } while (!consumeIf('E'));
- Result = make<StructuredBindingName>(popTrailingNodeArray(BindingsBegin));
- } else
- Result = getDerived().parseOperatorName(State);
- if (Result != nullptr)
- Result = getDerived().parseAbiTags(Result);
- return Result;
- }
- // <unnamed-type-name> ::= Ut [<nonnegative number>] _
- // ::= <closure-type-name>
- //
- // <closure-type-name> ::= Ul <lambda-sig> E [ <nonnegative number> ] _
- //
- // <lambda-sig> ::= <parameter type>+ # Parameter types or "v" if the lambda has no parameters
- template <typename Derived, typename Alloc>
- Node *
- AbstractManglingParser<Derived, Alloc>::parseUnnamedTypeName(NameState *State) {
- // <template-params> refer to the innermost <template-args>. Clear out any
- // outer args that we may have inserted into TemplateParams.
- if (State != nullptr)
- TemplateParams.clear();
- if (consumeIf("Ut")) {
- StringView Count = parseNumber();
- if (!consumeIf('_'))
- return nullptr;
- return make<UnnamedTypeName>(Count);
- }
- if (consumeIf("Ul")) {
- SwapAndRestore<size_t> SwapParams(ParsingLambdaParamsAtLevel,
- TemplateParams.size());
- ScopedTemplateParamList LambdaTemplateParams(this);
- size_t ParamsBegin = Names.size();
- while (look() == 'T' &&
- StringView("yptn").find(look(1)) != StringView::npos) {
- Node *T = parseTemplateParamDecl();
- if (!T)
- return nullptr;
- Names.push_back(T);
- }
- NodeArray TempParams = popTrailingNodeArray(ParamsBegin);
- // FIXME: If TempParams is empty and none of the function parameters
- // includes 'auto', we should remove LambdaTemplateParams from the
- // TemplateParams list. Unfortunately, we don't find out whether there are
- // any 'auto' parameters until too late in an example such as:
- //
- // template<typename T> void f(
- // decltype([](decltype([]<typename T>(T v) {}),
- // auto) {})) {}
- // template<typename T> void f(
- // decltype([](decltype([]<typename T>(T w) {}),
- // int) {})) {}
- //
- // Here, the type of v is at level 2 but the type of w is at level 1. We
- // don't find this out until we encounter the type of the next parameter.
- //
- // However, compilers can't actually cope with the former example in
- // practice, and it's likely to be made ill-formed in future, so we don't
- // need to support it here.
- //
- // If we encounter an 'auto' in the function parameter types, we will
- // recreate a template parameter scope for it, but any intervening lambdas
- // will be parsed in the 'wrong' template parameter depth.
- if (TempParams.empty())
- TemplateParams.pop_back();
- if (!consumeIf("vE")) {
- do {
- Node *P = getDerived().parseType();
- if (P == nullptr)
- return nullptr;
- Names.push_back(P);
- } while (!consumeIf('E'));
- }
- NodeArray Params = popTrailingNodeArray(ParamsBegin);
- StringView Count = parseNumber();
- if (!consumeIf('_'))
- return nullptr;
- return make<ClosureTypeName>(TempParams, Params, Count);
- }
- if (consumeIf("Ub")) {
- (void)parseNumber();
- if (!consumeIf('_'))
- return nullptr;
- return make<NameType>("'block-literal'");
- }
- return nullptr;
- }
- // <source-name> ::= <positive length number> <identifier>
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseSourceName(NameState *) {
- size_t Length = 0;
- if (parsePositiveInteger(&Length))
- return nullptr;
- if (numLeft() < Length || Length == 0)
- return nullptr;
- StringView Name(First, First + Length);
- First += Length;
- if (Name.startsWith("_GLOBAL__N"))
- return make<NameType>("(anonymous namespace)");
- return make<NameType>(Name);
- }
- // <operator-name> ::= aa # &&
- // ::= ad # & (unary)
- // ::= an # &
- // ::= aN # &=
- // ::= aS # =
- // ::= cl # ()
- // ::= cm # ,
- // ::= co # ~
- // ::= cv <type> # (cast)
- // ::= da # delete[]
- // ::= de # * (unary)
- // ::= dl # delete
- // ::= dv # /
- // ::= dV # /=
- // ::= eo # ^
- // ::= eO # ^=
- // ::= eq # ==
- // ::= ge # >=
- // ::= gt # >
- // ::= ix # []
- // ::= le # <=
- // ::= li <source-name> # operator ""
- // ::= ls # <<
- // ::= lS # <<=
- // ::= lt # <
- // ::= mi # -
- // ::= mI # -=
- // ::= ml # *
- // ::= mL # *=
- // ::= mm # -- (postfix in <expression> context)
- // ::= na # new[]
- // ::= ne # !=
- // ::= ng # - (unary)
- // ::= nt # !
- // ::= nw # new
- // ::= oo # ||
- // ::= or # |
- // ::= oR # |=
- // ::= pm # ->*
- // ::= pl # +
- // ::= pL # +=
- // ::= pp # ++ (postfix in <expression> context)
- // ::= ps # + (unary)
- // ::= pt # ->
- // ::= qu # ?
- // ::= rm # %
- // ::= rM # %=
- // ::= rs # >>
- // ::= rS # >>=
- // ::= ss # <=> C++2a
- // ::= v <digit> <source-name> # vendor extended operator
- template <typename Derived, typename Alloc>
- Node *
- AbstractManglingParser<Derived, Alloc>::parseOperatorName(NameState *State) {
- switch (look()) {
- case 'a':
- switch (look(1)) {
- case 'a':
- First += 2;
- return make<NameType>("operator&&");
- case 'd':
- case 'n':
- First += 2;
- return make<NameType>("operator&");
- case 'N':
- First += 2;
- return make<NameType>("operator&=");
- case 'S':
- First += 2;
- return make<NameType>("operator=");
- }
- return nullptr;
- case 'c':
- switch (look(1)) {
- case 'l':
- First += 2;
- return make<NameType>("operator()");
- case 'm':
- First += 2;
- return make<NameType>("operator,");
- case 'o':
- First += 2;
- return make<NameType>("operator~");
- // ::= cv <type> # (cast)
- case 'v': {
- First += 2;
- SwapAndRestore<bool> SaveTemplate(TryToParseTemplateArgs, false);
- // If we're parsing an encoding, State != nullptr and the conversion
- // operators' <type> could have a <template-param> that refers to some
- // <template-arg>s further ahead in the mangled name.
- SwapAndRestore<bool> SavePermit(PermitForwardTemplateReferences,
- PermitForwardTemplateReferences ||
- State != nullptr);
- Node *Ty = getDerived().parseType();
- if (Ty == nullptr)
- return nullptr;
- if (State) State->CtorDtorConversion = true;
- return make<ConversionOperatorType>(Ty);
- }
- }
- return nullptr;
- case 'd':
- switch (look(1)) {
- case 'a':
- First += 2;
- return make<NameType>("operator delete[]");
- case 'e':
- First += 2;
- return make<NameType>("operator*");
- case 'l':
- First += 2;
- return make<NameType>("operator delete");
- case 'v':
- First += 2;
- return make<NameType>("operator/");
- case 'V':
- First += 2;
- return make<NameType>("operator/=");
- }
- return nullptr;
- case 'e':
- switch (look(1)) {
- case 'o':
- First += 2;
- return make<NameType>("operator^");
- case 'O':
- First += 2;
- return make<NameType>("operator^=");
- case 'q':
- First += 2;
- return make<NameType>("operator==");
- }
- return nullptr;
- case 'g':
- switch (look(1)) {
- case 'e':
- First += 2;
- return make<NameType>("operator>=");
- case 't':
- First += 2;
- return make<NameType>("operator>");
- }
- return nullptr;
- case 'i':
- if (look(1) == 'x') {
- First += 2;
- return make<NameType>("operator[]");
- }
- return nullptr;
- case 'l':
- switch (look(1)) {
- case 'e':
- First += 2;
- return make<NameType>("operator<=");
- // ::= li <source-name> # operator ""
- case 'i': {
- First += 2;
- Node *SN = getDerived().parseSourceName(State);
- if (SN == nullptr)
- return nullptr;
- return make<LiteralOperator>(SN);
- }
- case 's':
- First += 2;
- return make<NameType>("operator<<");
- case 'S':
- First += 2;
- return make<NameType>("operator<<=");
- case 't':
- First += 2;
- return make<NameType>("operator<");
- }
- return nullptr;
- case 'm':
- switch (look(1)) {
- case 'i':
- First += 2;
- return make<NameType>("operator-");
- case 'I':
- First += 2;
- return make<NameType>("operator-=");
- case 'l':
- First += 2;
- return make<NameType>("operator*");
- case 'L':
- First += 2;
- return make<NameType>("operator*=");
- case 'm':
- First += 2;
- return make<NameType>("operator--");
- }
- return nullptr;
- case 'n':
- switch (look(1)) {
- case 'a':
- First += 2;
- return make<NameType>("operator new[]");
- case 'e':
- First += 2;
- return make<NameType>("operator!=");
- case 'g':
- First += 2;
- return make<NameType>("operator-");
- case 't':
- First += 2;
- return make<NameType>("operator!");
- case 'w':
- First += 2;
- return make<NameType>("operator new");
- }
- return nullptr;
- case 'o':
- switch (look(1)) {
- case 'o':
- First += 2;
- return make<NameType>("operator||");
- case 'r':
- First += 2;
- return make<NameType>("operator|");
- case 'R':
- First += 2;
- return make<NameType>("operator|=");
- }
- return nullptr;
- case 'p':
- switch (look(1)) {
- case 'm':
- First += 2;
- return make<NameType>("operator->*");
- case 'l':
- First += 2;
- return make<NameType>("operator+");
- case 'L':
- First += 2;
- return make<NameType>("operator+=");
- case 'p':
- First += 2;
- return make<NameType>("operator++");
- case 's':
- First += 2;
- return make<NameType>("operator+");
- case 't':
- First += 2;
- return make<NameType>("operator->");
- }
- return nullptr;
- case 'q':
- if (look(1) == 'u') {
- First += 2;
- return make<NameType>("operator?");
- }
- return nullptr;
- case 'r':
- switch (look(1)) {
- case 'm':
- First += 2;
- return make<NameType>("operator%");
- case 'M':
- First += 2;
- return make<NameType>("operator%=");
- case 's':
- First += 2;
- return make<NameType>("operator>>");
- case 'S':
- First += 2;
- return make<NameType>("operator>>=");
- }
- return nullptr;
- case 's':
- if (look(1) == 's') {
- First += 2;
- return make<NameType>("operator<=>");
- }
- return nullptr;
- // ::= v <digit> <source-name> # vendor extended operator
- case 'v':
- if (std::isdigit(look(1))) {
- First += 2;
- Node *SN = getDerived().parseSourceName(State);
- if (SN == nullptr)
- return nullptr;
- return make<ConversionOperatorType>(SN);
- }
- return nullptr;
- }
- return nullptr;
- }
- // <ctor-dtor-name> ::= C1 # complete object constructor
- // ::= C2 # base object constructor
- // ::= C3 # complete object allocating constructor
- // extension ::= C4 # gcc old-style "[unified]" constructor
- // extension ::= C5 # the COMDAT used for ctors
- // ::= D0 # deleting destructor
- // ::= D1 # complete object destructor
- // ::= D2 # base object destructor
- // extension ::= D4 # gcc old-style "[unified]" destructor
- // extension ::= D5 # the COMDAT used for dtors
- template <typename Derived, typename Alloc>
- Node *
- AbstractManglingParser<Derived, Alloc>::parseCtorDtorName(Node *&SoFar,
- NameState *State) {
- if (SoFar->getKind() == Node::KSpecialSubstitution) {
- auto SSK = static_cast<SpecialSubstitution *>(SoFar)->SSK;
- switch (SSK) {
- case SpecialSubKind::string:
- case SpecialSubKind::istream:
- case SpecialSubKind::ostream:
- case SpecialSubKind::iostream:
- SoFar = make<ExpandedSpecialSubstitution>(SSK);
- if (!SoFar)
- return nullptr;
- break;
- default:
- break;
- }
- }
- if (consumeIf('C')) {
- bool IsInherited = consumeIf('I');
- if (look() != '1' && look() != '2' && look() != '3' && look() != '4' &&
- look() != '5')
- return nullptr;
- int Variant = look() - '0';
- ++First;
- if (State) State->CtorDtorConversion = true;
- if (IsInherited) {
- if (getDerived().parseName(State) == nullptr)
- return nullptr;
- }
- return make<CtorDtorName>(SoFar, /*IsDtor=*/false, Variant);
- }
- if (look() == 'D' && (look(1) == '0' || look(1) == '1' || look(1) == '2' ||
- look(1) == '4' || look(1) == '5')) {
- int Variant = look(1) - '0';
- First += 2;
- if (State) State->CtorDtorConversion = true;
- return make<CtorDtorName>(SoFar, /*IsDtor=*/true, Variant);
- }
- return nullptr;
- }
- // <nested-name> ::= N [<CV-Qualifiers>] [<ref-qualifier>] <prefix> <unqualified-name> E
- // ::= N [<CV-Qualifiers>] [<ref-qualifier>] <template-prefix> <template-args> E
- //
- // <prefix> ::= <prefix> <unqualified-name>
- // ::= <template-prefix> <template-args>
- // ::= <template-param>
- // ::= <decltype>
- // ::= # empty
- // ::= <substitution>
- // ::= <prefix> <data-member-prefix>
- // extension ::= L
- //
- // <data-member-prefix> := <member source-name> [<template-args>] M
- //
- // <template-prefix> ::= <prefix> <template unqualified-name>
- // ::= <template-param>
- // ::= <substitution>
- template <typename Derived, typename Alloc>
- Node *
- AbstractManglingParser<Derived, Alloc>::parseNestedName(NameState *State) {
- if (!consumeIf('N'))
- return nullptr;
- Qualifiers CVTmp = parseCVQualifiers();
- if (State) State->CVQualifiers = CVTmp;
- if (consumeIf('O')) {
- if (State) State->ReferenceQualifier = FrefQualRValue;
- } else if (consumeIf('R')) {
- if (State) State->ReferenceQualifier = FrefQualLValue;
- } else
- if (State) State->ReferenceQualifier = FrefQualNone;
- Node *SoFar = nullptr;
- auto PushComponent = [&](Node *Comp) {
- if (!Comp) return false;
- if (SoFar) SoFar = make<NestedName>(SoFar, Comp);
- else SoFar = Comp;
- if (State) State->EndsWithTemplateArgs = false;
- return SoFar != nullptr;
- };
- if (consumeIf("St")) {
- SoFar = make<NameType>("std");
- if (!SoFar)
- return nullptr;
- }
- while (!consumeIf('E')) {
- consumeIf('L'); // extension
- // <data-member-prefix> := <member source-name> [<template-args>] M
- if (consumeIf('M')) {
- if (SoFar == nullptr)
- return nullptr;
- continue;
- }
- // ::= <template-param>
- if (look() == 'T') {
- if (!PushComponent(getDerived().parseTemplateParam()))
- return nullptr;
- Subs.push_back(SoFar);
- continue;
- }
- // ::= <template-prefix> <template-args>
- if (look() == 'I') {
- Node *TA = getDerived().parseTemplateArgs(State != nullptr);
- if (TA == nullptr || SoFar == nullptr)
- return nullptr;
- SoFar = make<NameWithTemplateArgs>(SoFar, TA);
- if (!SoFar)
- return nullptr;
- if (State) State->EndsWithTemplateArgs = true;
- Subs.push_back(SoFar);
- continue;
- }
- // ::= <decltype>
- if (look() == 'D' && (look(1) == 't' || look(1) == 'T')) {
- if (!PushComponent(getDerived().parseDecltype()))
- return nullptr;
- Subs.push_back(SoFar);
- continue;
- }
- // ::= <substitution>
- if (look() == 'S' && look(1) != 't') {
- Node *S = getDerived().parseSubstitution();
- if (!PushComponent(S))
- return nullptr;
- if (SoFar != S)
- Subs.push_back(S);
- continue;
- }
- // Parse an <unqualified-name> thats actually a <ctor-dtor-name>.
- if (look() == 'C' || (look() == 'D' && look(1) != 'C')) {
- if (SoFar == nullptr)
- return nullptr;
- if (!PushComponent(getDerived().parseCtorDtorName(SoFar, State)))
- return nullptr;
- SoFar = getDerived().parseAbiTags(SoFar);
- if (SoFar == nullptr)
- return nullptr;
- Subs.push_back(SoFar);
- continue;
- }
- // ::= <prefix> <unqualified-name>
- if (!PushComponent(getDerived().parseUnqualifiedName(State)))
- return nullptr;
- Subs.push_back(SoFar);
- }
- if (SoFar == nullptr || Subs.empty())
- return nullptr;
- Subs.pop_back();
- return SoFar;
- }
- // <simple-id> ::= <source-name> [ <template-args> ]
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseSimpleId() {
- Node *SN = getDerived().parseSourceName(/*NameState=*/nullptr);
- if (SN == nullptr)
- return nullptr;
- if (look() == 'I') {
- Node *TA = getDerived().parseTemplateArgs();
- if (TA == nullptr)
- return nullptr;
- return make<NameWithTemplateArgs>(SN, TA);
- }
- return SN;
- }
- // <destructor-name> ::= <unresolved-type> # e.g., ~T or ~decltype(f())
- // ::= <simple-id> # e.g., ~A<2*N>
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseDestructorName() {
- Node *Result;
- if (std::isdigit(look()))
- Result = getDerived().parseSimpleId();
- else
- Result = getDerived().parseUnresolvedType();
- if (Result == nullptr)
- return nullptr;
- return make<DtorName>(Result);
- }
- // <unresolved-type> ::= <template-param>
- // ::= <decltype>
- // ::= <substitution>
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseUnresolvedType() {
- if (look() == 'T') {
- Node *TP = getDerived().parseTemplateParam();
- if (TP == nullptr)
- return nullptr;
- Subs.push_back(TP);
- return TP;
- }
- if (look() == 'D') {
- Node *DT = getDerived().parseDecltype();
- if (DT == nullptr)
- return nullptr;
- Subs.push_back(DT);
- return DT;
- }
- return getDerived().parseSubstitution();
- }
- // <base-unresolved-name> ::= <simple-id> # unresolved name
- // extension ::= <operator-name> # unresolved operator-function-id
- // extension ::= <operator-name> <template-args> # unresolved operator template-id
- // ::= on <operator-name> # unresolved operator-function-id
- // ::= on <operator-name> <template-args> # unresolved operator template-id
- // ::= dn <destructor-name> # destructor or pseudo-destructor;
- // # e.g. ~X or ~X<N-1>
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseBaseUnresolvedName() {
- if (std::isdigit(look()))
- return getDerived().parseSimpleId();
- if (consumeIf("dn"))
- return getDerived().parseDestructorName();
- consumeIf("on");
- Node *Oper = getDerived().parseOperatorName(/*NameState=*/nullptr);
- if (Oper == nullptr)
- return nullptr;
- if (look() == 'I') {
- Node *TA = getDerived().parseTemplateArgs();
- if (TA == nullptr)
- return nullptr;
- return make<NameWithTemplateArgs>(Oper, TA);
- }
- return Oper;
- }
- // <unresolved-name>
- // extension ::= srN <unresolved-type> [<template-args>] <unresolved-qualifier-level>* E <base-unresolved-name>
- // ::= [gs] <base-unresolved-name> # x or (with "gs") ::x
- // ::= [gs] sr <unresolved-qualifier-level>+ E <base-unresolved-name>
- // # A::x, N::y, A<T>::z; "gs" means leading "::"
- // ::= sr <unresolved-type> <base-unresolved-name> # T::x / decltype(p)::x
- // extension ::= sr <unresolved-type> <template-args> <base-unresolved-name>
- // # T::N::x /decltype(p)::N::x
- // (ignored) ::= srN <unresolved-type> <unresolved-qualifier-level>+ E <base-unresolved-name>
- //
- // <unresolved-qualifier-level> ::= <simple-id>
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseUnresolvedName() {
- Node *SoFar = nullptr;
- // srN <unresolved-type> [<template-args>] <unresolved-qualifier-level>* E <base-unresolved-name>
- // srN <unresolved-type> <unresolved-qualifier-level>+ E <base-unresolved-name>
- if (consumeIf("srN")) {
- SoFar = getDerived().parseUnresolvedType();
- if (SoFar == nullptr)
- return nullptr;
- if (look() == 'I') {
- Node *TA = getDerived().parseTemplateArgs();
- if (TA == nullptr)
- return nullptr;
- SoFar = make<NameWithTemplateArgs>(SoFar, TA);
- if (!SoFar)
- return nullptr;
- }
- while (!consumeIf('E')) {
- Node *Qual = getDerived().parseSimpleId();
- if (Qual == nullptr)
- return nullptr;
- SoFar = make<QualifiedName>(SoFar, Qual);
- if (!SoFar)
- return nullptr;
- }
- Node *Base = getDerived().parseBaseUnresolvedName();
- if (Base == nullptr)
- return nullptr;
- return make<QualifiedName>(SoFar, Base);
- }
- bool Global = consumeIf("gs");
- // [gs] <base-unresolved-name> # x or (with "gs") ::x
- if (!consumeIf("sr")) {
- SoFar = getDerived().parseBaseUnresolvedName();
- if (SoFar == nullptr)
- return nullptr;
- if (Global)
- SoFar = make<GlobalQualifiedName>(SoFar);
- return SoFar;
- }
- // [gs] sr <unresolved-qualifier-level>+ E <base-unresolved-name>
- if (std::isdigit(look())) {
- do {
- Node *Qual = getDerived().parseSimpleId();
- if (Qual == nullptr)
- return nullptr;
- if (SoFar)
- SoFar = make<QualifiedName>(SoFar, Qual);
- else if (Global)
- SoFar = make<GlobalQualifiedName>(Qual);
- else
- SoFar = Qual;
- if (!SoFar)
- return nullptr;
- } while (!consumeIf('E'));
- }
- // sr <unresolved-type> <base-unresolved-name>
- // sr <unresolved-type> <template-args> <base-unresolved-name>
- else {
- SoFar = getDerived().parseUnresolvedType();
- if (SoFar == nullptr)
- return nullptr;
- if (look() == 'I') {
- Node *TA = getDerived().parseTemplateArgs();
- if (TA == nullptr)
- return nullptr;
- SoFar = make<NameWithTemplateArgs>(SoFar, TA);
- if (!SoFar)
- return nullptr;
- }
- }
- assert(SoFar != nullptr);
- Node *Base = getDerived().parseBaseUnresolvedName();
- if (Base == nullptr)
- return nullptr;
- return make<QualifiedName>(SoFar, Base);
- }
- // <abi-tags> ::= <abi-tag> [<abi-tags>]
- // <abi-tag> ::= B <source-name>
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseAbiTags(Node *N) {
- while (consumeIf('B')) {
- StringView SN = parseBareSourceName();
- if (SN.empty())
- return nullptr;
- N = make<AbiTagAttr>(N, SN);
- if (!N)
- return nullptr;
- }
- return N;
- }
- // <number> ::= [n] <non-negative decimal integer>
- template <typename Alloc, typename Derived>
- StringView
- AbstractManglingParser<Alloc, Derived>::parseNumber(bool AllowNegative) {
- const char *Tmp = First;
- if (AllowNegative)
- consumeIf('n');
- if (numLeft() == 0 || !std::isdigit(*First))
- return StringView();
- while (numLeft() != 0 && std::isdigit(*First))
- ++First;
- return StringView(Tmp, First);
- }
- // <positive length number> ::= [0-9]*
- template <typename Alloc, typename Derived>
- bool AbstractManglingParser<Alloc, Derived>::parsePositiveInteger(size_t *Out) {
- *Out = 0;
- if (look() < '0' || look() > '9')
- return true;
- while (look() >= '0' && look() <= '9') {
- *Out *= 10;
- *Out += static_cast<size_t>(consume() - '0');
- }
- return false;
- }
- template <typename Alloc, typename Derived>
- StringView AbstractManglingParser<Alloc, Derived>::parseBareSourceName() {
- size_t Int = 0;
- if (parsePositiveInteger(&Int) || numLeft() < Int)
- return StringView();
- StringView R(First, First + Int);
- First += Int;
- return R;
- }
- // <function-type> ::= [<CV-qualifiers>] [<exception-spec>] [Dx] F [Y] <bare-function-type> [<ref-qualifier>] E
- //
- // <exception-spec> ::= Do # non-throwing exception-specification (e.g., noexcept, throw())
- // ::= DO <expression> E # computed (instantiation-dependent) noexcept
- // ::= Dw <type>+ E # dynamic exception specification with instantiation-dependent types
- //
- // <ref-qualifier> ::= R # & ref-qualifier
- // <ref-qualifier> ::= O # && ref-qualifier
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseFunctionType() {
- Qualifiers CVQuals = parseCVQualifiers();
- Node *ExceptionSpec = nullptr;
- if (consumeIf("Do")) {
- ExceptionSpec = make<NameType>("noexcept");
- if (!ExceptionSpec)
- return nullptr;
- } else if (consumeIf("DO")) {
- Node *E = getDerived().parseExpr();
- if (E == nullptr || !consumeIf('E'))
- return nullptr;
- ExceptionSpec = make<NoexceptSpec>(E);
- if (!ExceptionSpec)
- return nullptr;
- } else if (consumeIf("Dw")) {
- size_t SpecsBegin = Names.size();
- while (!consumeIf('E')) {
- Node *T = getDerived().parseType();
- if (T == nullptr)
- return nullptr;
- Names.push_back(T);
- }
- ExceptionSpec =
- make<DynamicExceptionSpec>(popTrailingNodeArray(SpecsBegin));
- if (!ExceptionSpec)
- return nullptr;
- }
- consumeIf("Dx"); // transaction safe
- if (!consumeIf('F'))
- return nullptr;
- consumeIf('Y'); // extern "C"
- Node *ReturnType = getDerived().parseType();
- if (ReturnType == nullptr)
- return nullptr;
- FunctionRefQual ReferenceQualifier = FrefQualNone;
- size_t ParamsBegin = Names.size();
- while (true) {
- if (consumeIf('E'))
- break;
- if (consumeIf('v'))
- continue;
- if (consumeIf("RE")) {
- ReferenceQualifier = FrefQualLValue;
- break;
- }
- if (consumeIf("OE")) {
- ReferenceQualifier = FrefQualRValue;
- break;
- }
- Node *T = getDerived().parseType();
- if (T == nullptr)
- return nullptr;
- Names.push_back(T);
- }
- NodeArray Params = popTrailingNodeArray(ParamsBegin);
- return make<FunctionType>(ReturnType, Params, CVQuals,
- ReferenceQualifier, ExceptionSpec);
- }
- // extension:
- // <vector-type> ::= Dv <positive dimension number> _ <extended element type>
- // ::= Dv [<dimension expression>] _ <element type>
- // <extended element type> ::= <element type>
- // ::= p # AltiVec vector pixel
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseVectorType() {
- if (!consumeIf("Dv"))
- return nullptr;
- if (look() >= '1' && look() <= '9') {
- Node *DimensionNumber = make<NameType>(parseNumber());
- if (!DimensionNumber)
- return nullptr;
- if (!consumeIf('_'))
- return nullptr;
- if (consumeIf('p'))
- return make<PixelVectorType>(DimensionNumber);
- Node *ElemType = getDerived().parseType();
- if (ElemType == nullptr)
- return nullptr;
- return make<VectorType>(ElemType, DimensionNumber);
- }
- if (!consumeIf('_')) {
- Node *DimExpr = getDerived().parseExpr();
- if (!DimExpr)
- return nullptr;
- if (!consumeIf('_'))
- return nullptr;
- Node *ElemType = getDerived().parseType();
- if (!ElemType)
- return nullptr;
- return make<VectorType>(ElemType, DimExpr);
- }
- Node *ElemType = getDerived().parseType();
- if (!ElemType)
- return nullptr;
- return make<VectorType>(ElemType, /*Dimension=*/nullptr);
- }
- // <decltype> ::= Dt <expression> E # decltype of an id-expression or class member access (C++0x)
- // ::= DT <expression> E # decltype of an expression (C++0x)
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseDecltype() {
- if (!consumeIf('D'))
- return nullptr;
- if (!consumeIf('t') && !consumeIf('T'))
- return nullptr;
- Node *E = getDerived().parseExpr();
- if (E == nullptr)
- return nullptr;
- if (!consumeIf('E'))
- return nullptr;
- return make<EnclosingExpr>("decltype(", E, ")");
- }
- // <array-type> ::= A <positive dimension number> _ <element type>
- // ::= A [<dimension expression>] _ <element type>
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseArrayType() {
- if (!consumeIf('A'))
- return nullptr;
- Node *Dimension = nullptr;
- if (std::isdigit(look())) {
- Dimension = make<NameType>(parseNumber());
- if (!Dimension)
- return nullptr;
- if (!consumeIf('_'))
- return nullptr;
- } else if (!consumeIf('_')) {
- Node *DimExpr = getDerived().parseExpr();
- if (DimExpr == nullptr)
- return nullptr;
- if (!consumeIf('_'))
- return nullptr;
- Dimension = DimExpr;
- }
- Node *Ty = getDerived().parseType();
- if (Ty == nullptr)
- return nullptr;
- return make<ArrayType>(Ty, Dimension);
- }
- // <pointer-to-member-type> ::= M <class type> <member type>
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parsePointerToMemberType() {
- if (!consumeIf('M'))
- return nullptr;
- Node *ClassType = getDerived().parseType();
- if (ClassType == nullptr)
- return nullptr;
- Node *MemberType = getDerived().parseType();
- if (MemberType == nullptr)
- return nullptr;
- return make<PointerToMemberType>(ClassType, MemberType);
- }
- // <class-enum-type> ::= <name> # non-dependent type name, dependent type name, or dependent typename-specifier
- // ::= Ts <name> # dependent elaborated type specifier using 'struct' or 'class'
- // ::= Tu <name> # dependent elaborated type specifier using 'union'
- // ::= Te <name> # dependent elaborated type specifier using 'enum'
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseClassEnumType() {
- StringView ElabSpef;
- if (consumeIf("Ts"))
- ElabSpef = "struct";
- else if (consumeIf("Tu"))
- ElabSpef = "union";
- else if (consumeIf("Te"))
- ElabSpef = "enum";
- Node *Name = getDerived().parseName();
- if (Name == nullptr)
- return nullptr;
- if (!ElabSpef.empty())
- return make<ElaboratedTypeSpefType>(ElabSpef, Name);
- return Name;
- }
- // <qualified-type> ::= <qualifiers> <type>
- // <qualifiers> ::= <extended-qualifier>* <CV-qualifiers>
- // <extended-qualifier> ::= U <source-name> [<template-args>] # vendor extended type qualifier
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseQualifiedType() {
- if (consumeIf('U')) {
- StringView Qual = parseBareSourceName();
- if (Qual.empty())
- return nullptr;
- // extension ::= U <objc-name> <objc-type> # objc-type<identifier>
- if (Qual.startsWith("objcproto")) {
- StringView ProtoSourceName = Qual.dropFront(std::strlen("objcproto"));
- StringView Proto;
- {
- SwapAndRestore<const char *> SaveFirst(First, ProtoSourceName.begin()),
- SaveLast(Last, ProtoSourceName.end());
- Proto = parseBareSourceName();
- }
- if (Proto.empty())
- return nullptr;
- Node *Child = getDerived().parseQualifiedType();
- if (Child == nullptr)
- return nullptr;
- return make<ObjCProtoName>(Child, Proto);
- }
- Node *TA = nullptr;
- if (look() == 'I') {
- TA = getDerived().parseTemplateArgs();
- if (TA == nullptr)
- return nullptr;
- }
- Node *Child = getDerived().parseQualifiedType();
- if (Child == nullptr)
- return nullptr;
- return make<VendorExtQualType>(Child, Qual, TA);
- }
- Qualifiers Quals = parseCVQualifiers();
- Node *Ty = getDerived().parseType();
- if (Ty == nullptr)
- return nullptr;
- if (Quals != QualNone)
- Ty = make<QualType>(Ty, Quals);
- return Ty;
- }
- // <type> ::= <builtin-type>
- // ::= <qualified-type>
- // ::= <function-type>
- // ::= <class-enum-type>
- // ::= <array-type>
- // ::= <pointer-to-member-type>
- // ::= <template-param>
- // ::= <template-template-param> <template-args>
- // ::= <decltype>
- // ::= P <type> # pointer
- // ::= R <type> # l-value reference
- // ::= O <type> # r-value reference (C++11)
- // ::= C <type> # complex pair (C99)
- // ::= G <type> # imaginary (C99)
- // ::= <substitution> # See Compression below
- // extension ::= U <objc-name> <objc-type> # objc-type<identifier>
- // extension ::= <vector-type> # <vector-type> starts with Dv
- //
- // <objc-name> ::= <k0 number> objcproto <k1 number> <identifier> # k0 = 9 + <number of digits in k1> + k1
- // <objc-type> ::= <source-name> # PU<11+>objcproto 11objc_object<source-name> 11objc_object -> id<source-name>
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseType() {
- Node *Result = nullptr;
- switch (look()) {
- // ::= <qualified-type>
- case 'r':
- case 'V':
- case 'K': {
- unsigned AfterQuals = 0;
- if (look(AfterQuals) == 'r') ++AfterQuals;
- if (look(AfterQuals) == 'V') ++AfterQuals;
- if (look(AfterQuals) == 'K') ++AfterQuals;
- if (look(AfterQuals) == 'F' ||
- (look(AfterQuals) == 'D' &&
- (look(AfterQuals + 1) == 'o' || look(AfterQuals + 1) == 'O' ||
- look(AfterQuals + 1) == 'w' || look(AfterQuals + 1) == 'x'))) {
- Result = getDerived().parseFunctionType();
- break;
- }
- DEMANGLE_FALLTHROUGH;
- }
- case 'U': {
- Result = getDerived().parseQualifiedType();
- break;
- }
- // <builtin-type> ::= v # void
- case 'v':
- ++First;
- return make<NameType>("void");
- // ::= w # wchar_t
- case 'w':
- ++First;
- return make<NameType>("wchar_t");
- // ::= b # bool
- case 'b':
- ++First;
- return make<NameType>("bool");
- // ::= c # char
- case 'c':
- ++First;
- return make<NameType>("char");
- // ::= a # signed char
- case 'a':
- ++First;
- return make<NameType>("signed char");
- // ::= h # unsigned char
- case 'h':
- ++First;
- return make<NameType>("unsigned char");
- // ::= s # short
- case 's':
- ++First;
- return make<NameType>("short");
- // ::= t # unsigned short
- case 't':
- ++First;
- return make<NameType>("unsigned short");
- // ::= i # int
- case 'i':
- ++First;
- return make<NameType>("int");
- // ::= j # unsigned int
- case 'j':
- ++First;
- return make<NameType>("unsigned int");
- // ::= l # long
- case 'l':
- ++First;
- return make<NameType>("long");
- // ::= m # unsigned long
- case 'm':
- ++First;
- return make<NameType>("unsigned long");
- // ::= x # long long, __int64
- case 'x':
- ++First;
- return make<NameType>("long long");
- // ::= y # unsigned long long, __int64
- case 'y':
- ++First;
- return make<NameType>("unsigned long long");
- // ::= n # __int128
- case 'n':
- ++First;
- return make<NameType>("__int128");
- // ::= o # unsigned __int128
- case 'o':
- ++First;
- return make<NameType>("unsigned __int128");
- // ::= f # float
- case 'f':
- ++First;
- return make<NameType>("float");
- // ::= d # double
- case 'd':
- ++First;
- return make<NameType>("double");
- // ::= e # long double, __float80
- case 'e':
- ++First;
- return make<NameType>("long double");
- // ::= g # __float128
- case 'g':
- ++First;
- return make<NameType>("__float128");
- // ::= z # ellipsis
- case 'z':
- ++First;
- return make<NameType>("...");
- // <builtin-type> ::= u <source-name> # vendor extended type
- case 'u': {
- ++First;
- StringView Res = parseBareSourceName();
- if (Res.empty())
- return nullptr;
- // Typically, <builtin-type>s are not considered substitution candidates,
- // but the exception to that exception is vendor extended types (Itanium C++
- // ABI 5.9.1).
- Result = make<NameType>(Res);
- break;
- }
- case 'D':
- switch (look(1)) {
- // ::= Dd # IEEE 754r decimal floating point (64 bits)
- case 'd':
- First += 2;
- return make<NameType>("decimal64");
- // ::= De # IEEE 754r decimal floating point (128 bits)
- case 'e':
- First += 2;
- return make<NameType>("decimal128");
- // ::= Df # IEEE 754r decimal floating point (32 bits)
- case 'f':
- First += 2;
- return make<NameType>("decimal32");
- // ::= Dh # IEEE 754r half-precision floating point (16 bits)
- case 'h':
- First += 2;
- return make<NameType>("half");
- // ::= DF <number> _ # ISO/IEC TS 18661 binary floating point (N bits)
- case 'F': {
- First += 2;
- Node *DimensionNumber = make<NameType>(parseNumber());
- if (!DimensionNumber)
- return nullptr;
- if (!consumeIf('_'))
- return nullptr;
- return make<BinaryFPType>(DimensionNumber);
- }
- // ::= Di # char32_t
- case 'i':
- First += 2;
- return make<NameType>("char32_t");
- // ::= Ds # char16_t
- case 's':
- First += 2;
- return make<NameType>("char16_t");
- // ::= Du # char8_t (C++2a, not yet in the Itanium spec)
- case 'u':
- First += 2;
- return make<NameType>("char8_t");
- // ::= Da # auto (in dependent new-expressions)
- case 'a':
- First += 2;
- return make<NameType>("auto");
- // ::= Dc # decltype(auto)
- case 'c':
- First += 2;
- return make<NameType>("decltype(auto)");
- // ::= Dn # std::nullptr_t (i.e., decltype(nullptr))
- case 'n':
- First += 2;
- return make<NameType>("std::nullptr_t");
- // ::= <decltype>
- case 't':
- case 'T': {
- Result = getDerived().parseDecltype();
- break;
- }
- // extension ::= <vector-type> # <vector-type> starts with Dv
- case 'v': {
- Result = getDerived().parseVectorType();
- break;
- }
- // ::= Dp <type> # pack expansion (C++0x)
- case 'p': {
- First += 2;
- Node *Child = getDerived().parseType();
- if (!Child)
- return nullptr;
- Result = make<ParameterPackExpansion>(Child);
- break;
- }
- // Exception specifier on a function type.
- case 'o':
- case 'O':
- case 'w':
- // Transaction safe function type.
- case 'x':
- Result = getDerived().parseFunctionType();
- break;
- }
- break;
- // ::= <function-type>
- case 'F': {
- Result = getDerived().parseFunctionType();
- break;
- }
- // ::= <array-type>
- case 'A': {
- Result = getDerived().parseArrayType();
- break;
- }
- // ::= <pointer-to-member-type>
- case 'M': {
- Result = getDerived().parsePointerToMemberType();
- break;
- }
- // ::= <template-param>
- case 'T': {
- // This could be an elaborate type specifier on a <class-enum-type>.
- if (look(1) == 's' || look(1) == 'u' || look(1) == 'e') {
- Result = getDerived().parseClassEnumType();
- break;
- }
- Result = getDerived().parseTemplateParam();
- if (Result == nullptr)
- return nullptr;
- // Result could be either of:
- // <type> ::= <template-param>
- // <type> ::= <template-template-param> <template-args>
- //
- // <template-template-param> ::= <template-param>
- // ::= <substitution>
- //
- // If this is followed by some <template-args>, and we're permitted to
- // parse them, take the second production.
- if (TryToParseTemplateArgs && look() == 'I') {
- Node *TA = getDerived().parseTemplateArgs();
- if (TA == nullptr)
- return nullptr;
- Result = make<NameWithTemplateArgs>(Result, TA);
- }
- break;
- }
- // ::= P <type> # pointer
- case 'P': {
- ++First;
- Node *Ptr = getDerived().parseType();
- if (Ptr == nullptr)
- return nullptr;
- Result = make<PointerType>(Ptr);
- break;
- }
- // ::= R <type> # l-value reference
- case 'R': {
- ++First;
- Node *Ref = getDerived().parseType();
- if (Ref == nullptr)
- return nullptr;
- Result = make<ReferenceType>(Ref, ReferenceKind::LValue);
- break;
- }
- // ::= O <type> # r-value reference (C++11)
- case 'O': {
- ++First;
- Node *Ref = getDerived().parseType();
- if (Ref == nullptr)
- return nullptr;
- Result = make<ReferenceType>(Ref, ReferenceKind::RValue);
- break;
- }
- // ::= C <type> # complex pair (C99)
- case 'C': {
- ++First;
- Node *P = getDerived().parseType();
- if (P == nullptr)
- return nullptr;
- Result = make<PostfixQualifiedType>(P, " complex");
- break;
- }
- // ::= G <type> # imaginary (C99)
- case 'G': {
- ++First;
- Node *P = getDerived().parseType();
- if (P == nullptr)
- return P;
- Result = make<PostfixQualifiedType>(P, " imaginary");
- break;
- }
- // ::= <substitution> # See Compression below
- case 'S': {
- if (look(1) != 't') {
- Result = getDerived().parseSubstitution();
- if (Result == nullptr)
- return nullptr;
- // Sub could be either of:
- // <type> ::= <substitution>
- // <type> ::= <template-template-param> <template-args>
- //
- // <template-template-param> ::= <template-param>
- // ::= <substitution>
- //
- // If this is followed by some <template-args>, and we're permitted to
- // parse them, take the second production.
- if (TryToParseTemplateArgs && look() == 'I') {
- Node *TA = getDerived().parseTemplateArgs();
- if (TA == nullptr)
- return nullptr;
- Result = make<NameWithTemplateArgs>(Result, TA);
- } else {
- // If all we parsed was a substitution, don't re-insert into the
- // substitution table.
- return Result;
- }
- break;
- }
- DEMANGLE_FALLTHROUGH;
- }
- // ::= <class-enum-type>
- default: {
- Result = getDerived().parseClassEnumType();
- break;
- }
- }
- // If we parsed a type, insert it into the substitution table. Note that all
- // <builtin-type>s and <substitution>s have already bailed out, because they
- // don't get substitutions.
- if (Result != nullptr)
- Subs.push_back(Result);
- return Result;
- }
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parsePrefixExpr(StringView Kind) {
- Node *E = getDerived().parseExpr();
- if (E == nullptr)
- return nullptr;
- return make<PrefixExpr>(Kind, E);
- }
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseBinaryExpr(StringView Kind) {
- Node *LHS = getDerived().parseExpr();
- if (LHS == nullptr)
- return nullptr;
- Node *RHS = getDerived().parseExpr();
- if (RHS == nullptr)
- return nullptr;
- return make<BinaryExpr>(LHS, Kind, RHS);
- }
- template <typename Derived, typename Alloc>
- Node *
- AbstractManglingParser<Derived, Alloc>::parseIntegerLiteral(StringView Lit) {
- StringView Tmp = parseNumber(true);
- if (!Tmp.empty() && consumeIf('E'))
- return make<IntegerLiteral>(Lit, Tmp);
- return nullptr;
- }
- // <CV-Qualifiers> ::= [r] [V] [K]
- template <typename Alloc, typename Derived>
- Qualifiers AbstractManglingParser<Alloc, Derived>::parseCVQualifiers() {
- Qualifiers CVR = QualNone;
- if (consumeIf('r'))
- CVR |= QualRestrict;
- if (consumeIf('V'))
- CVR |= QualVolatile;
- if (consumeIf('K'))
- CVR |= QualConst;
- return CVR;
- }
- // <function-param> ::= fp <top-level CV-Qualifiers> _ # L == 0, first parameter
- // ::= fp <top-level CV-Qualifiers> <parameter-2 non-negative number> _ # L == 0, second and later parameters
- // ::= fL <L-1 non-negative number> p <top-level CV-Qualifiers> _ # L > 0, first parameter
- // ::= fL <L-1 non-negative number> p <top-level CV-Qualifiers> <parameter-2 non-negative number> _ # L > 0, second and later parameters
- // ::= fpT # 'this' expression (not part of standard?)
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseFunctionParam() {
- if (consumeIf("fpT"))
- return make<NameType>("this");
- if (consumeIf("fp")) {
- parseCVQualifiers();
- StringView Num = parseNumber();
- if (!consumeIf('_'))
- return nullptr;
- return make<FunctionParam>(Num);
- }
- if (consumeIf("fL")) {
- if (parseNumber().empty())
- return nullptr;
- if (!consumeIf('p'))
- return nullptr;
- parseCVQualifiers();
- StringView Num = parseNumber();
- if (!consumeIf('_'))
- return nullptr;
- return make<FunctionParam>(Num);
- }
- return nullptr;
- }
- // [gs] nw <expression>* _ <type> E # new (expr-list) type
- // [gs] nw <expression>* _ <type> <initializer> # new (expr-list) type (init)
- // [gs] na <expression>* _ <type> E # new[] (expr-list) type
- // [gs] na <expression>* _ <type> <initializer> # new[] (expr-list) type (init)
- // <initializer> ::= pi <expression>* E # parenthesized initialization
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseNewExpr() {
- bool Global = consumeIf("gs");
- bool IsArray = look(1) == 'a';
- if (!consumeIf("nw") && !consumeIf("na"))
- return nullptr;
- size_t Exprs = Names.size();
- while (!consumeIf('_')) {
- Node *Ex = getDerived().parseExpr();
- if (Ex == nullptr)
- return nullptr;
- Names.push_back(Ex);
- }
- NodeArray ExprList = popTrailingNodeArray(Exprs);
- Node *Ty = getDerived().parseType();
- if (Ty == nullptr)
- return Ty;
- if (consumeIf("pi")) {
- size_t InitsBegin = Names.size();
- while (!consumeIf('E')) {
- Node *Init = getDerived().parseExpr();
- if (Init == nullptr)
- return Init;
- Names.push_back(Init);
- }
- NodeArray Inits = popTrailingNodeArray(InitsBegin);
- return make<NewExpr>(ExprList, Ty, Inits, Global, IsArray);
- } else if (!consumeIf('E'))
- return nullptr;
- return make<NewExpr>(ExprList, Ty, NodeArray(), Global, IsArray);
- }
- // cv <type> <expression> # conversion with one argument
- // cv <type> _ <expression>* E # conversion with a different number of arguments
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseConversionExpr() {
- if (!consumeIf("cv"))
- return nullptr;
- Node *Ty;
- {
- SwapAndRestore<bool> SaveTemp(TryToParseTemplateArgs, false);
- Ty = getDerived().parseType();
- }
- if (Ty == nullptr)
- return nullptr;
- if (consumeIf('_')) {
- size_t ExprsBegin = Names.size();
- while (!consumeIf('E')) {
- Node *E = getDerived().parseExpr();
- if (E == nullptr)
- return E;
- Names.push_back(E);
- }
- NodeArray Exprs = popTrailingNodeArray(ExprsBegin);
- return make<ConversionExpr>(Ty, Exprs);
- }
- Node *E[1] = {getDerived().parseExpr()};
- if (E[0] == nullptr)
- return nullptr;
- return make<ConversionExpr>(Ty, makeNodeArray(E, E + 1));
- }
- // <expr-primary> ::= L <type> <value number> E # integer literal
- // ::= L <type> <value float> E # floating literal
- // ::= L <string type> E # string literal
- // ::= L <nullptr type> E # nullptr literal (i.e., "LDnE")
- // ::= L <lambda type> E # lambda expression
- // FIXME: ::= L <type> <real-part float> _ <imag-part float> E # complex floating point literal (C 2000)
- // ::= L <mangled-name> E # external name
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseExprPrimary() {
- if (!consumeIf('L'))
- return nullptr;
- switch (look()) {
- case 'w':
- ++First;
- return getDerived().parseIntegerLiteral("wchar_t");
- case 'b':
- if (consumeIf("b0E"))
- return make<BoolExpr>(0);
- if (consumeIf("b1E"))
- return make<BoolExpr>(1);
- return nullptr;
- case 'c':
- ++First;
- return getDerived().parseIntegerLiteral("char");
- case 'a':
- ++First;
- return getDerived().parseIntegerLiteral("signed char");
- case 'h':
- ++First;
- return getDerived().parseIntegerLiteral("unsigned char");
- case 's':
- ++First;
- return getDerived().parseIntegerLiteral("short");
- case 't':
- ++First;
- return getDerived().parseIntegerLiteral("unsigned short");
- case 'i':
- ++First;
- return getDerived().parseIntegerLiteral("");
- case 'j':
- ++First;
- return getDerived().parseIntegerLiteral("u");
- case 'l':
- ++First;
- return getDerived().parseIntegerLiteral("l");
- case 'm':
- ++First;
- return getDerived().parseIntegerLiteral("ul");
- case 'x':
- ++First;
- return getDerived().parseIntegerLiteral("ll");
- case 'y':
- ++First;
- return getDerived().parseIntegerLiteral("ull");
- case 'n':
- ++First;
- return getDerived().parseIntegerLiteral("__int128");
- case 'o':
- ++First;
- return getDerived().parseIntegerLiteral("unsigned __int128");
- case 'f':
- ++First;
- return getDerived().template parseFloatingLiteral<float>();
- case 'd':
- ++First;
- return getDerived().template parseFloatingLiteral<double>();
- case 'e':
- ++First;
- #if defined(__powerpc__) || defined(__s390__)
- // Handle cases where long doubles encoded with e have the same size
- // and representation as doubles.
- return getDerived().template parseFloatingLiteral<double>();
- #else
- return getDerived().template parseFloatingLiteral<long double>();
- #endif
- case '_':
- if (consumeIf("_Z")) {
- Node *R = getDerived().parseEncoding();
- if (R != nullptr && consumeIf('E'))
- return R;
- }
- return nullptr;
- case 'A': {
- Node *T = getDerived().parseType();
- if (T == nullptr)
- return nullptr;
- // FIXME: We need to include the string contents in the mangling.
- if (consumeIf('E'))
- return make<StringLiteral>(T);
- return nullptr;
- }
- case 'D':
- if (consumeIf("DnE"))
- return make<NameType>("nullptr");
- return nullptr;
- case 'T':
- // Invalid mangled name per
- // http://sourcerytools.com/pipermail/cxx-abi-dev/2011-August/002422.html
- return nullptr;
- case 'U': {
- // FIXME: Should we support LUb... for block literals?
- if (look(1) != 'l')
- return nullptr;
- Node *T = parseUnnamedTypeName(nullptr);
- if (!T || !consumeIf('E'))
- return nullptr;
- return make<LambdaExpr>(T);
- }
- default: {
- // might be named type
- Node *T = getDerived().parseType();
- if (T == nullptr)
- return nullptr;
- StringView N = parseNumber(/*AllowNegative=*/true);
- if (N.empty())
- return nullptr;
- if (!consumeIf('E'))
- return nullptr;
- return make<EnumLiteral>(T, N);
- }
- }
- }
- // <braced-expression> ::= <expression>
- // ::= di <field source-name> <braced-expression> # .name = expr
- // ::= dx <index expression> <braced-expression> # [expr] = expr
- // ::= dX <range begin expression> <range end expression> <braced-expression>
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseBracedExpr() {
- if (look() == 'd') {
- switch (look(1)) {
- case 'i': {
- First += 2;
- Node *Field = getDerived().parseSourceName(/*NameState=*/nullptr);
- if (Field == nullptr)
- return nullptr;
- Node *Init = getDerived().parseBracedExpr();
- if (Init == nullptr)
- return nullptr;
- return make<BracedExpr>(Field, Init, /*isArray=*/false);
- }
- case 'x': {
- First += 2;
- Node *Index = getDerived().parseExpr();
- if (Index == nullptr)
- return nullptr;
- Node *Init = getDerived().parseBracedExpr();
- if (Init == nullptr)
- return nullptr;
- return make<BracedExpr>(Index, Init, /*isArray=*/true);
- }
- case 'X': {
- First += 2;
- Node *RangeBegin = getDerived().parseExpr();
- if (RangeBegin == nullptr)
- return nullptr;
- Node *RangeEnd = getDerived().parseExpr();
- if (RangeEnd == nullptr)
- return nullptr;
- Node *Init = getDerived().parseBracedExpr();
- if (Init == nullptr)
- return nullptr;
- return make<BracedRangeExpr>(RangeBegin, RangeEnd, Init);
- }
- }
- }
- return getDerived().parseExpr();
- }
- // (not yet in the spec)
- // <fold-expr> ::= fL <binary-operator-name> <expression> <expression>
- // ::= fR <binary-operator-name> <expression> <expression>
- // ::= fl <binary-operator-name> <expression>
- // ::= fr <binary-operator-name> <expression>
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseFoldExpr() {
- if (!consumeIf('f'))
- return nullptr;
- char FoldKind = look();
- bool IsLeftFold, HasInitializer;
- HasInitializer = FoldKind == 'L' || FoldKind == 'R';
- if (FoldKind == 'l' || FoldKind == 'L')
- IsLeftFold = true;
- else if (FoldKind == 'r' || FoldKind == 'R')
- IsLeftFold = false;
- else
- return nullptr;
- ++First;
- // FIXME: This map is duplicated in parseOperatorName and parseExpr.
- StringView OperatorName;
- if (consumeIf("aa")) OperatorName = "&&";
- else if (consumeIf("an")) OperatorName = "&";
- else if (consumeIf("aN")) OperatorName = "&=";
- else if (consumeIf("aS")) OperatorName = "=";
- else if (consumeIf("cm")) OperatorName = ",";
- else if (consumeIf("ds")) OperatorName = ".*";
- else if (consumeIf("dv")) OperatorName = "/";
- else if (consumeIf("dV")) OperatorName = "/=";
- else if (consumeIf("eo")) OperatorName = "^";
- else if (consumeIf("eO")) OperatorName = "^=";
- else if (consumeIf("eq")) OperatorName = "==";
- else if (consumeIf("ge")) OperatorName = ">=";
- else if (consumeIf("gt")) OperatorName = ">";
- else if (consumeIf("le")) OperatorName = "<=";
- else if (consumeIf("ls")) OperatorName = "<<";
- else if (consumeIf("lS")) OperatorName = "<<=";
- else if (consumeIf("lt")) OperatorName = "<";
- else if (consumeIf("mi")) OperatorName = "-";
- else if (consumeIf("mI")) OperatorName = "-=";
- else if (consumeIf("ml")) OperatorName = "*";
- else if (consumeIf("mL")) OperatorName = "*=";
- else if (consumeIf("ne")) OperatorName = "!=";
- else if (consumeIf("oo")) OperatorName = "||";
- else if (consumeIf("or")) OperatorName = "|";
- else if (consumeIf("oR")) OperatorName = "|=";
- else if (consumeIf("pl")) OperatorName = "+";
- else if (consumeIf("pL")) OperatorName = "+=";
- else if (consumeIf("rm")) OperatorName = "%";
- else if (consumeIf("rM")) OperatorName = "%=";
- else if (consumeIf("rs")) OperatorName = ">>";
- else if (consumeIf("rS")) OperatorName = ">>=";
- else return nullptr;
- Node *Pack = getDerived().parseExpr(), *Init = nullptr;
- if (Pack == nullptr)
- return nullptr;
- if (HasInitializer) {
- Init = getDerived().parseExpr();
- if (Init == nullptr)
- return nullptr;
- }
- if (IsLeftFold && Init)
- std::swap(Pack, Init);
- return make<FoldExpr>(IsLeftFold, OperatorName, Pack, Init);
- }
- // <expression> ::= mc <parameter type> <expr> [<offset number>] E
- //
- // Not yet in the spec: https://github.com/itanium-cxx-abi/cxx-abi/issues/47
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parsePointerToMemberConversionExpr() {
- Node *Ty = getDerived().parseType();
- if (!Ty)
- return nullptr;
- Node *Expr = getDerived().parseExpr();
- if (!Expr)
- return nullptr;
- StringView Offset = getDerived().parseNumber(true);
- if (!consumeIf('E'))
- return nullptr;
- return make<PointerToMemberConversionExpr>(Ty, Expr, Offset);
- }
- // <expression> ::= so <referent type> <expr> [<offset number>] <union-selector>* [p] E
- // <union-selector> ::= _ [<number>]
- //
- // Not yet in the spec: https://github.com/itanium-cxx-abi/cxx-abi/issues/47
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseSubobjectExpr() {
- Node *Ty = getDerived().parseType();
- if (!Ty)
- return nullptr;
- Node *Expr = getDerived().parseExpr();
- if (!Expr)
- return nullptr;
- StringView Offset = getDerived().parseNumber(true);
- size_t SelectorsBegin = Names.size();
- while (consumeIf('_')) {
- Node *Selector = make<NameType>(parseNumber());
- if (!Selector)
- return nullptr;
- Names.push_back(Selector);
- }
- bool OnePastTheEnd = consumeIf('p');
- if (!consumeIf('E'))
- return nullptr;
- return make<SubobjectExpr>(
- Ty, Expr, Offset, popTrailingNodeArray(SelectorsBegin), OnePastTheEnd);
- }
- // <expression> ::= <unary operator-name> <expression>
- // ::= <binary operator-name> <expression> <expression>
- // ::= <ternary operator-name> <expression> <expression> <expression>
- // ::= cl <expression>+ E # call
- // ::= cv <type> <expression> # conversion with one argument
- // ::= cv <type> _ <expression>* E # conversion with a different number of arguments
- // ::= [gs] nw <expression>* _ <type> E # new (expr-list) type
- // ::= [gs] nw <expression>* _ <type> <initializer> # new (expr-list) type (init)
- // ::= [gs] na <expression>* _ <type> E # new[] (expr-list) type
- // ::= [gs] na <expression>* _ <type> <initializer> # new[] (expr-list) type (init)
- // ::= [gs] dl <expression> # delete expression
- // ::= [gs] da <expression> # delete[] expression
- // ::= pp_ <expression> # prefix ++
- // ::= mm_ <expression> # prefix --
- // ::= ti <type> # typeid (type)
- // ::= te <expression> # typeid (expression)
- // ::= dc <type> <expression> # dynamic_cast<type> (expression)
- // ::= sc <type> <expression> # static_cast<type> (expression)
- // ::= cc <type> <expression> # const_cast<type> (expression)
- // ::= rc <type> <expression> # reinterpret_cast<type> (expression)
- // ::= st <type> # sizeof (a type)
- // ::= sz <expression> # sizeof (an expression)
- // ::= at <type> # alignof (a type)
- // ::= az <expression> # alignof (an expression)
- // ::= nx <expression> # noexcept (expression)
- // ::= <template-param>
- // ::= <function-param>
- // ::= dt <expression> <unresolved-name> # expr.name
- // ::= pt <expression> <unresolved-name> # expr->name
- // ::= ds <expression> <expression> # expr.*expr
- // ::= sZ <template-param> # size of a parameter pack
- // ::= sZ <function-param> # size of a function parameter pack
- // ::= sP <template-arg>* E # sizeof...(T), size of a captured template parameter pack from an alias template
- // ::= sp <expression> # pack expansion
- // ::= tw <expression> # throw expression
- // ::= tr # throw with no operand (rethrow)
- // ::= <unresolved-name> # f(p), N::f(p), ::f(p),
- // # freestanding dependent name (e.g., T::x),
- // # objectless nonstatic member reference
- // ::= fL <binary-operator-name> <expression> <expression>
- // ::= fR <binary-operator-name> <expression> <expression>
- // ::= fl <binary-operator-name> <expression>
- // ::= fr <binary-operator-name> <expression>
- // ::= <expr-primary>
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseExpr() {
- bool Global = consumeIf("gs");
- if (numLeft() < 2)
- return nullptr;
- switch (*First) {
- case 'L':
- return getDerived().parseExprPrimary();
- case 'T':
- return getDerived().parseTemplateParam();
- case 'f': {
- // Disambiguate a fold expression from a <function-param>.
- if (look(1) == 'p' || (look(1) == 'L' && std::isdigit(look(2))))
- return getDerived().parseFunctionParam();
- return getDerived().parseFoldExpr();
- }
- case 'a':
- switch (First[1]) {
- case 'a':
- First += 2;
- return getDerived().parseBinaryExpr("&&");
- case 'd':
- First += 2;
- return getDerived().parsePrefixExpr("&");
- case 'n':
- First += 2;
- return getDerived().parseBinaryExpr("&");
- case 'N':
- First += 2;
- return getDerived().parseBinaryExpr("&=");
- case 'S':
- First += 2;
- return getDerived().parseBinaryExpr("=");
- case 't': {
- First += 2;
- Node *Ty = getDerived().parseType();
- if (Ty == nullptr)
- return nullptr;
- return make<EnclosingExpr>("alignof (", Ty, ")");
- }
- case 'z': {
- First += 2;
- Node *Ty = getDerived().parseExpr();
- if (Ty == nullptr)
- return nullptr;
- return make<EnclosingExpr>("alignof (", Ty, ")");
- }
- }
- return nullptr;
- case 'c':
- switch (First[1]) {
- // cc <type> <expression> # const_cast<type>(expression)
- case 'c': {
- First += 2;
- Node *Ty = getDerived().parseType();
- if (Ty == nullptr)
- return Ty;
- Node *Ex = getDerived().parseExpr();
- if (Ex == nullptr)
- return Ex;
- return make<CastExpr>("const_cast", Ty, Ex);
- }
- // cl <expression>+ E # call
- case 'l': {
- First += 2;
- Node *Callee = getDerived().parseExpr();
- if (Callee == nullptr)
- return Callee;
- size_t ExprsBegin = Names.size();
- while (!consumeIf('E')) {
- Node *E = getDerived().parseExpr();
- if (E == nullptr)
- return E;
- Names.push_back(E);
- }
- return make<CallExpr>(Callee, popTrailingNodeArray(ExprsBegin));
- }
- case 'm':
- First += 2;
- return getDerived().parseBinaryExpr(",");
- case 'o':
- First += 2;
- return getDerived().parsePrefixExpr("~");
- case 'v':
- return getDerived().parseConversionExpr();
- }
- return nullptr;
- case 'd':
- switch (First[1]) {
- case 'a': {
- First += 2;
- Node *Ex = getDerived().parseExpr();
- if (Ex == nullptr)
- return Ex;
- return make<DeleteExpr>(Ex, Global, /*is_array=*/true);
- }
- case 'c': {
- First += 2;
- Node *T = getDerived().parseType();
- if (T == nullptr)
- return T;
- Node *Ex = getDerived().parseExpr();
- if (Ex == nullptr)
- return Ex;
- return make<CastExpr>("dynamic_cast", T, Ex);
- }
- case 'e':
- First += 2;
- return getDerived().parsePrefixExpr("*");
- case 'l': {
- First += 2;
- Node *E = getDerived().parseExpr();
- if (E == nullptr)
- return E;
- return make<DeleteExpr>(E, Global, /*is_array=*/false);
- }
- case 'n':
- return getDerived().parseUnresolvedName();
- case 's': {
- First += 2;
- Node *LHS = getDerived().parseExpr();
- if (LHS == nullptr)
- return nullptr;
- Node *RHS = getDerived().parseExpr();
- if (RHS == nullptr)
- return nullptr;
- return make<MemberExpr>(LHS, ".*", RHS);
- }
- case 't': {
- First += 2;
- Node *LHS = getDerived().parseExpr();
- if (LHS == nullptr)
- return LHS;
- Node *RHS = getDerived().parseExpr();
- if (RHS == nullptr)
- return nullptr;
- return make<MemberExpr>(LHS, ".", RHS);
- }
- case 'v':
- First += 2;
- return getDerived().parseBinaryExpr("/");
- case 'V':
- First += 2;
- return getDerived().parseBinaryExpr("/=");
- }
- return nullptr;
- case 'e':
- switch (First[1]) {
- case 'o':
- First += 2;
- return getDerived().parseBinaryExpr("^");
- case 'O':
- First += 2;
- return getDerived().parseBinaryExpr("^=");
- case 'q':
- First += 2;
- return getDerived().parseBinaryExpr("==");
- }
- return nullptr;
- case 'g':
- switch (First[1]) {
- case 'e':
- First += 2;
- return getDerived().parseBinaryExpr(">=");
- case 't':
- First += 2;
- return getDerived().parseBinaryExpr(">");
- }
- return nullptr;
- case 'i':
- switch (First[1]) {
- case 'x': {
- First += 2;
- Node *Base = getDerived().parseExpr();
- if (Base == nullptr)
- return nullptr;
- Node *Index = getDerived().parseExpr();
- if (Index == nullptr)
- return Index;
- return make<ArraySubscriptExpr>(Base, Index);
- }
- case 'l': {
- First += 2;
- size_t InitsBegin = Names.size();
- while (!consumeIf('E')) {
- Node *E = getDerived().parseBracedExpr();
- if (E == nullptr)
- return nullptr;
- Names.push_back(E);
- }
- return make<InitListExpr>(nullptr, popTrailingNodeArray(InitsBegin));
- }
- }
- return nullptr;
- case 'l':
- switch (First[1]) {
- case 'e':
- First += 2;
- return getDerived().parseBinaryExpr("<=");
- case 's':
- First += 2;
- return getDerived().parseBinaryExpr("<<");
- case 'S':
- First += 2;
- return getDerived().parseBinaryExpr("<<=");
- case 't':
- First += 2;
- return getDerived().parseBinaryExpr("<");
- }
- return nullptr;
- case 'm':
- switch (First[1]) {
- case 'c':
- First += 2;
- return parsePointerToMemberConversionExpr();
- case 'i':
- First += 2;
- return getDerived().parseBinaryExpr("-");
- case 'I':
- First += 2;
- return getDerived().parseBinaryExpr("-=");
- case 'l':
- First += 2;
- return getDerived().parseBinaryExpr("*");
- case 'L':
- First += 2;
- return getDerived().parseBinaryExpr("*=");
- case 'm':
- First += 2;
- if (consumeIf('_'))
- return getDerived().parsePrefixExpr("--");
- Node *Ex = getDerived().parseExpr();
- if (Ex == nullptr)
- return nullptr;
- return make<PostfixExpr>(Ex, "--");
- }
- return nullptr;
- case 'n':
- switch (First[1]) {
- case 'a':
- case 'w':
- return getDerived().parseNewExpr();
- case 'e':
- First += 2;
- return getDerived().parseBinaryExpr("!=");
- case 'g':
- First += 2;
- return getDerived().parsePrefixExpr("-");
- case 't':
- First += 2;
- return getDerived().parsePrefixExpr("!");
- case 'x':
- First += 2;
- Node *Ex = getDerived().parseExpr();
- if (Ex == nullptr)
- return Ex;
- return make<EnclosingExpr>("noexcept (", Ex, ")");
- }
- return nullptr;
- case 'o':
- switch (First[1]) {
- case 'n':
- return getDerived().parseUnresolvedName();
- case 'o':
- First += 2;
- return getDerived().parseBinaryExpr("||");
- case 'r':
- First += 2;
- return getDerived().parseBinaryExpr("|");
- case 'R':
- First += 2;
- return getDerived().parseBinaryExpr("|=");
- }
- return nullptr;
- case 'p':
- switch (First[1]) {
- case 'm':
- First += 2;
- return getDerived().parseBinaryExpr("->*");
- case 'l':
- First += 2;
- return getDerived().parseBinaryExpr("+");
- case 'L':
- First += 2;
- return getDerived().parseBinaryExpr("+=");
- case 'p': {
- First += 2;
- if (consumeIf('_'))
- return getDerived().parsePrefixExpr("++");
- Node *Ex = getDerived().parseExpr();
- if (Ex == nullptr)
- return Ex;
- return make<PostfixExpr>(Ex, "++");
- }
- case 's':
- First += 2;
- return getDerived().parsePrefixExpr("+");
- case 't': {
- First += 2;
- Node *L = getDerived().parseExpr();
- if (L == nullptr)
- return nullptr;
- Node *R = getDerived().parseExpr();
- if (R == nullptr)
- return nullptr;
- return make<MemberExpr>(L, "->", R);
- }
- }
- return nullptr;
- case 'q':
- if (First[1] == 'u') {
- First += 2;
- Node *Cond = getDerived().parseExpr();
- if (Cond == nullptr)
- return nullptr;
- Node *LHS = getDerived().parseExpr();
- if (LHS == nullptr)
- return nullptr;
- Node *RHS = getDerived().parseExpr();
- if (RHS == nullptr)
- return nullptr;
- return make<ConditionalExpr>(Cond, LHS, RHS);
- }
- return nullptr;
- case 'r':
- switch (First[1]) {
- case 'c': {
- First += 2;
- Node *T = getDerived().parseType();
- if (T == nullptr)
- return T;
- Node *Ex = getDerived().parseExpr();
- if (Ex == nullptr)
- return Ex;
- return make<CastExpr>("reinterpret_cast", T, Ex);
- }
- case 'm':
- First += 2;
- return getDerived().parseBinaryExpr("%");
- case 'M':
- First += 2;
- return getDerived().parseBinaryExpr("%=");
- case 's':
- First += 2;
- return getDerived().parseBinaryExpr(">>");
- case 'S':
- First += 2;
- return getDerived().parseBinaryExpr(">>=");
- }
- return nullptr;
- case 's':
- switch (First[1]) {
- case 'c': {
- First += 2;
- Node *T = getDerived().parseType();
- if (T == nullptr)
- return T;
- Node *Ex = getDerived().parseExpr();
- if (Ex == nullptr)
- return Ex;
- return make<CastExpr>("static_cast", T, Ex);
- }
- case 'o':
- First += 2;
- return parseSubobjectExpr();
- case 'p': {
- First += 2;
- Node *Child = getDerived().parseExpr();
- if (Child == nullptr)
- return nullptr;
- return make<ParameterPackExpansion>(Child);
- }
- case 'r':
- return getDerived().parseUnresolvedName();
- case 't': {
- First += 2;
- Node *Ty = getDerived().parseType();
- if (Ty == nullptr)
- return Ty;
- return make<EnclosingExpr>("sizeof (", Ty, ")");
- }
- case 'z': {
- First += 2;
- Node *Ex = getDerived().parseExpr();
- if (Ex == nullptr)
- return Ex;
- return make<EnclosingExpr>("sizeof (", Ex, ")");
- }
- case 'Z':
- First += 2;
- if (look() == 'T') {
- Node *R = getDerived().parseTemplateParam();
- if (R == nullptr)
- return nullptr;
- return make<SizeofParamPackExpr>(R);
- } else if (look() == 'f') {
- Node *FP = getDerived().parseFunctionParam();
- if (FP == nullptr)
- return nullptr;
- return make<EnclosingExpr>("sizeof... (", FP, ")");
- }
- return nullptr;
- case 'P': {
- First += 2;
- size_t ArgsBegin = Names.size();
- while (!consumeIf('E')) {
- Node *Arg = getDerived().parseTemplateArg();
- if (Arg == nullptr)
- return nullptr;
- Names.push_back(Arg);
- }
- auto *Pack = make<NodeArrayNode>(popTrailingNodeArray(ArgsBegin));
- if (!Pack)
- return nullptr;
- return make<EnclosingExpr>("sizeof... (", Pack, ")");
- }
- }
- return nullptr;
- case 't':
- switch (First[1]) {
- case 'e': {
- First += 2;
- Node *Ex = getDerived().parseExpr();
- if (Ex == nullptr)
- return Ex;
- return make<EnclosingExpr>("typeid (", Ex, ")");
- }
- case 'i': {
- First += 2;
- Node *Ty = getDerived().parseType();
- if (Ty == nullptr)
- return Ty;
- return make<EnclosingExpr>("typeid (", Ty, ")");
- }
- case 'l': {
- First += 2;
- Node *Ty = getDerived().parseType();
- if (Ty == nullptr)
- return nullptr;
- size_t InitsBegin = Names.size();
- while (!consumeIf('E')) {
- Node *E = getDerived().parseBracedExpr();
- if (E == nullptr)
- return nullptr;
- Names.push_back(E);
- }
- return make<InitListExpr>(Ty, popTrailingNodeArray(InitsBegin));
- }
- case 'r':
- First += 2;
- return make<NameType>("throw");
- case 'w': {
- First += 2;
- Node *Ex = getDerived().parseExpr();
- if (Ex == nullptr)
- return nullptr;
- return make<ThrowExpr>(Ex);
- }
- }
- return nullptr;
- case 'u': {
- ++First;
- Node *Name = getDerived().parseSourceName(/*NameState=*/nullptr);
- if (!Name)
- return nullptr;
- // Special case legacy __uuidof mangling. The 't' and 'z' appear where the
- // standard encoding expects a <template-arg>, and would be otherwise be
- // interpreted as <type> node 'short' or 'ellipsis'. However, neither
- // __uuidof(short) nor __uuidof(...) can actually appear, so there is no
- // actual conflict here.
- if (Name->getBaseName() == "__uuidof") {
- if (numLeft() < 2)
- return nullptr;
- if (*First == 't') {
- ++First;
- Node *Ty = getDerived().parseType();
- if (!Ty)
- return nullptr;
- return make<CallExpr>(Name, makeNodeArray(&Ty, &Ty + 1));
- }
- if (*First == 'z') {
- ++First;
- Node *Ex = getDerived().parseExpr();
- if (!Ex)
- return nullptr;
- return make<CallExpr>(Name, makeNodeArray(&Ex, &Ex + 1));
- }
- }
- size_t ExprsBegin = Names.size();
- while (!consumeIf('E')) {
- Node *E = getDerived().parseTemplateArg();
- if (E == nullptr)
- return E;
- Names.push_back(E);
- }
- return make<CallExpr>(Name, popTrailingNodeArray(ExprsBegin));
- }
- case '1':
- case '2':
- case '3':
- case '4':
- case '5':
- case '6':
- case '7':
- case '8':
- case '9':
- return getDerived().parseUnresolvedName();
- }
- return nullptr;
- }
- // <call-offset> ::= h <nv-offset> _
- // ::= v <v-offset> _
- //
- // <nv-offset> ::= <offset number>
- // # non-virtual base override
- //
- // <v-offset> ::= <offset number> _ <virtual offset number>
- // # virtual base override, with vcall offset
- template <typename Alloc, typename Derived>
- bool AbstractManglingParser<Alloc, Derived>::parseCallOffset() {
- // Just scan through the call offset, we never add this information into the
- // output.
- if (consumeIf('h'))
- return parseNumber(true).empty() || !consumeIf('_');
- if (consumeIf('v'))
- return parseNumber(true).empty() || !consumeIf('_') ||
- parseNumber(true).empty() || !consumeIf('_');
- return true;
- }
- // <special-name> ::= TV <type> # virtual table
- // ::= TT <type> # VTT structure (construction vtable index)
- // ::= TI <type> # typeinfo structure
- // ::= TS <type> # typeinfo name (null-terminated byte string)
- // ::= Tc <call-offset> <call-offset> <base encoding>
- // # base is the nominal target function of thunk
- // # first call-offset is 'this' adjustment
- // # second call-offset is result adjustment
- // ::= T <call-offset> <base encoding>
- // # base is the nominal target function of thunk
- // ::= GV <object name> # Guard variable for one-time initialization
- // # No <type>
- // ::= TW <object name> # Thread-local wrapper
- // ::= TH <object name> # Thread-local initialization
- // ::= GR <object name> _ # First temporary
- // ::= GR <object name> <seq-id> _ # Subsequent temporaries
- // extension ::= TC <first type> <number> _ <second type> # construction vtable for second-in-first
- // extension ::= GR <object name> # reference temporary for object
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseSpecialName() {
- switch (look()) {
- case 'T':
- switch (look(1)) {
- // TA <template-arg> # template parameter object
- //
- // Not yet in the spec: https://github.com/itanium-cxx-abi/cxx-abi/issues/63
- case 'A': {
- First += 2;
- Node *Arg = getDerived().parseTemplateArg();
- if (Arg == nullptr)
- return nullptr;
- return make<SpecialName>("template parameter object for ", Arg);
- }
- // TV <type> # virtual table
- case 'V': {
- First += 2;
- Node *Ty = getDerived().parseType();
- if (Ty == nullptr)
- return nullptr;
- return make<SpecialName>("vtable for ", Ty);
- }
- // TT <type> # VTT structure (construction vtable index)
- case 'T': {
- First += 2;
- Node *Ty = getDerived().parseType();
- if (Ty == nullptr)
- return nullptr;
- return make<SpecialName>("VTT for ", Ty);
- }
- // TI <type> # typeinfo structure
- case 'I': {
- First += 2;
- Node *Ty = getDerived().parseType();
- if (Ty == nullptr)
- return nullptr;
- return make<SpecialName>("typeinfo for ", Ty);
- }
- // TS <type> # typeinfo name (null-terminated byte string)
- case 'S': {
- First += 2;
- Node *Ty = getDerived().parseType();
- if (Ty == nullptr)
- return nullptr;
- return make<SpecialName>("typeinfo name for ", Ty);
- }
- // Tc <call-offset> <call-offset> <base encoding>
- case 'c': {
- First += 2;
- if (parseCallOffset() || parseCallOffset())
- return nullptr;
- Node *Encoding = getDerived().parseEncoding();
- if (Encoding == nullptr)
- return nullptr;
- return make<SpecialName>("covariant return thunk to ", Encoding);
- }
- // extension ::= TC <first type> <number> _ <second type>
- // # construction vtable for second-in-first
- case 'C': {
- First += 2;
- Node *FirstType = getDerived().parseType();
- if (FirstType == nullptr)
- return nullptr;
- if (parseNumber(true).empty() || !consumeIf('_'))
- return nullptr;
- Node *SecondType = getDerived().parseType();
- if (SecondType == nullptr)
- return nullptr;
- return make<CtorVtableSpecialName>(SecondType, FirstType);
- }
- // TW <object name> # Thread-local wrapper
- case 'W': {
- First += 2;
- Node *Name = getDerived().parseName();
- if (Name == nullptr)
- return nullptr;
- return make<SpecialName>("thread-local wrapper routine for ", Name);
- }
- // TH <object name> # Thread-local initialization
- case 'H': {
- First += 2;
- Node *Name = getDerived().parseName();
- if (Name == nullptr)
- return nullptr;
- return make<SpecialName>("thread-local initialization routine for ", Name);
- }
- // T <call-offset> <base encoding>
- default: {
- ++First;
- bool IsVirt = look() == 'v';
- if (parseCallOffset())
- return nullptr;
- Node *BaseEncoding = getDerived().parseEncoding();
- if (BaseEncoding == nullptr)
- return nullptr;
- if (IsVirt)
- return make<SpecialName>("virtual thunk to ", BaseEncoding);
- else
- return make<SpecialName>("non-virtual thunk to ", BaseEncoding);
- }
- }
- case 'G':
- switch (look(1)) {
- // GV <object name> # Guard variable for one-time initialization
- case 'V': {
- First += 2;
- Node *Name = getDerived().parseName();
- if (Name == nullptr)
- return nullptr;
- return make<SpecialName>("guard variable for ", Name);
- }
- // GR <object name> # reference temporary for object
- // GR <object name> _ # First temporary
- // GR <object name> <seq-id> _ # Subsequent temporaries
- case 'R': {
- First += 2;
- Node *Name = getDerived().parseName();
- if (Name == nullptr)
- return nullptr;
- size_t Count;
- bool ParsedSeqId = !parseSeqId(&Count);
- if (!consumeIf('_') && ParsedSeqId)
- return nullptr;
- return make<SpecialName>("reference temporary for ", Name);
- }
- }
- }
- return nullptr;
- }
- // <encoding> ::= <function name> <bare-function-type>
- // ::= <data name>
- // ::= <special-name>
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseEncoding() {
- // The template parameters of an encoding are unrelated to those of the
- // enclosing context.
- class SaveTemplateParams {
- AbstractManglingParser *Parser;
- decltype(TemplateParams) OldParams;
- decltype(OuterTemplateParams) OldOuterParams;
- public:
- SaveTemplateParams(AbstractManglingParser *TheParser) : Parser(TheParser) {
- OldParams = std::move(Parser->TemplateParams);
- OldOuterParams = std::move(Parser->OuterTemplateParams);
- Parser->TemplateParams.clear();
- Parser->OuterTemplateParams.clear();
- }
- ~SaveTemplateParams() {
- Parser->TemplateParams = std::move(OldParams);
- Parser->OuterTemplateParams = std::move(OldOuterParams);
- }
- } SaveTemplateParams(this);
- if (look() == 'G' || look() == 'T')
- return getDerived().parseSpecialName();
- auto IsEndOfEncoding = [&] {
- // The set of chars that can potentially follow an <encoding> (none of which
- // can start a <type>). Enumerating these allows us to avoid speculative
- // parsing.
- return numLeft() == 0 || look() == 'E' || look() == '.' || look() == '_';
- };
- NameState NameInfo(this);
- Node *Name = getDerived().parseName(&NameInfo);
- if (Name == nullptr)
- return nullptr;
- if (resolveForwardTemplateRefs(NameInfo))
- return nullptr;
- if (IsEndOfEncoding())
- return Name;
- Node *Attrs = nullptr;
- if (consumeIf("Ua9enable_ifI")) {
- size_t BeforeArgs = Names.size();
- while (!consumeIf('E')) {
- Node *Arg = getDerived().parseTemplateArg();
- if (Arg == nullptr)
- return nullptr;
- Names.push_back(Arg);
- }
- Attrs = make<EnableIfAttr>(popTrailingNodeArray(BeforeArgs));
- if (!Attrs)
- return nullptr;
- }
- Node *ReturnType = nullptr;
- if (!NameInfo.CtorDtorConversion && NameInfo.EndsWithTemplateArgs) {
- ReturnType = getDerived().parseType();
- if (ReturnType == nullptr)
- return nullptr;
- }
- if (consumeIf('v'))
- return make<FunctionEncoding>(ReturnType, Name, NodeArray(),
- Attrs, NameInfo.CVQualifiers,
- NameInfo.ReferenceQualifier);
- size_t ParamsBegin = Names.size();
- do {
- Node *Ty = getDerived().parseType();
- if (Ty == nullptr)
- return nullptr;
- Names.push_back(Ty);
- } while (!IsEndOfEncoding());
- return make<FunctionEncoding>(ReturnType, Name,
- popTrailingNodeArray(ParamsBegin),
- Attrs, NameInfo.CVQualifiers,
- NameInfo.ReferenceQualifier);
- }
- template <class Float>
- struct FloatData;
- template <>
- struct FloatData<float>
- {
- static const size_t mangled_size = 8;
- static const size_t max_demangled_size = 24;
- static constexpr const char* spec = "%af";
- };
- template <>
- struct FloatData<double>
- {
- static const size_t mangled_size = 16;
- static const size_t max_demangled_size = 32;
- static constexpr const char* spec = "%a";
- };
- template <>
- struct FloatData<long double>
- {
- #if defined(__mips__) && defined(__mips_n64) || defined(__aarch64__) || \
- defined(__wasm__)
- static const size_t mangled_size = 32;
- #elif defined(__arm__) || defined(__mips__) || defined(__hexagon__)
- static const size_t mangled_size = 16;
- #else
- static const size_t mangled_size = 20; // May need to be adjusted to 16 or 24 on other platforms
- #endif
- // `-0x1.ffffffffffffffffffffffffffffp+16383` + 'L' + '\0' == 42 bytes.
- // 28 'f's * 4 bits == 112 bits, which is the number of mantissa bits.
- // Negatives are one character longer than positives.
- // `0x1.` and `p` are constant, and exponents `+16383` and `-16382` are the
- // same length. 1 sign bit, 112 mantissa bits, and 15 exponent bits == 128.
- static const size_t max_demangled_size = 42;
- static constexpr const char *spec = "%LaL";
- };
- template <typename Alloc, typename Derived>
- template <class Float>
- Node *AbstractManglingParser<Alloc, Derived>::parseFloatingLiteral() {
- const size_t N = FloatData<Float>::mangled_size;
- if (numLeft() <= N)
- return nullptr;
- StringView Data(First, First + N);
- for (char C : Data)
- if (!std::isxdigit(C))
- return nullptr;
- First += N;
- if (!consumeIf('E'))
- return nullptr;
- return make<FloatLiteralImpl<Float>>(Data);
- }
- // <seq-id> ::= <0-9A-Z>+
- template <typename Alloc, typename Derived>
- bool AbstractManglingParser<Alloc, Derived>::parseSeqId(size_t *Out) {
- if (!(look() >= '0' && look() <= '9') &&
- !(look() >= 'A' && look() <= 'Z'))
- return true;
- size_t Id = 0;
- while (true) {
- if (look() >= '0' && look() <= '9') {
- Id *= 36;
- Id += static_cast<size_t>(look() - '0');
- } else if (look() >= 'A' && look() <= 'Z') {
- Id *= 36;
- Id += static_cast<size_t>(look() - 'A') + 10;
- } else {
- *Out = Id;
- return false;
- }
- ++First;
- }
- }
- // <substitution> ::= S <seq-id> _
- // ::= S_
- // <substitution> ::= Sa # ::std::allocator
- // <substitution> ::= Sb # ::std::basic_string
- // <substitution> ::= Ss # ::std::basic_string < char,
- // ::std::char_traits<char>,
- // ::std::allocator<char> >
- // <substitution> ::= Si # ::std::basic_istream<char, std::char_traits<char> >
- // <substitution> ::= So # ::std::basic_ostream<char, std::char_traits<char> >
- // <substitution> ::= Sd # ::std::basic_iostream<char, std::char_traits<char> >
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseSubstitution() {
- if (!consumeIf('S'))
- return nullptr;
- if (look() >= 'a' && look() <= 'z') {
- SpecialSubKind Kind;
- switch (look()) {
- case 'a':
- Kind = SpecialSubKind::allocator;
- break;
- case 'b':
- Kind = SpecialSubKind::basic_string;
- break;
- case 'd':
- Kind = SpecialSubKind::iostream;
- break;
- case 'i':
- Kind = SpecialSubKind::istream;
- break;
- case 'o':
- Kind = SpecialSubKind::ostream;
- break;
- case 's':
- Kind = SpecialSubKind::string;
- break;
- default:
- return nullptr;
- }
- ++First;
- auto *SpecialSub = make<SpecialSubstitution>(Kind);
- if (!SpecialSub)
- return nullptr;
- // Itanium C++ ABI 5.1.2: If a name that would use a built-in <substitution>
- // has ABI tags, the tags are appended to the substitution; the result is a
- // substitutable component.
- Node *WithTags = getDerived().parseAbiTags(SpecialSub);
- if (WithTags != SpecialSub) {
- Subs.push_back(WithTags);
- SpecialSub = WithTags;
- }
- return SpecialSub;
- }
- // ::= S_
- if (consumeIf('_')) {
- if (Subs.empty())
- return nullptr;
- return Subs[0];
- }
- // ::= S <seq-id> _
- size_t Index = 0;
- if (parseSeqId(&Index))
- return nullptr;
- ++Index;
- if (!consumeIf('_') || Index >= Subs.size())
- return nullptr;
- return Subs[Index];
- }
- // <template-param> ::= T_ # first template parameter
- // ::= T <parameter-2 non-negative number> _
- // ::= TL <level-1> __
- // ::= TL <level-1> _ <parameter-2 non-negative number> _
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseTemplateParam() {
- if (!consumeIf('T'))
- return nullptr;
- size_t Level = 0;
- if (consumeIf('L')) {
- if (parsePositiveInteger(&Level))
- return nullptr;
- ++Level;
- if (!consumeIf('_'))
- return nullptr;
- }
- size_t Index = 0;
- if (!consumeIf('_')) {
- if (parsePositiveInteger(&Index))
- return nullptr;
- ++Index;
- if (!consumeIf('_'))
- return nullptr;
- }
- // If we're in a context where this <template-param> refers to a
- // <template-arg> further ahead in the mangled name (currently just conversion
- // operator types), then we should only look it up in the right context.
- // This can only happen at the outermost level.
- if (PermitForwardTemplateReferences && Level == 0) {
- Node *ForwardRef = make<ForwardTemplateReference>(Index);
- if (!ForwardRef)
- return nullptr;
- assert(ForwardRef->getKind() == Node::KForwardTemplateReference);
- ForwardTemplateRefs.push_back(
- static_cast<ForwardTemplateReference *>(ForwardRef));
- return ForwardRef;
- }
- if (Level >= TemplateParams.size() || !TemplateParams[Level] ||
- Index >= TemplateParams[Level]->size()) {
- // Itanium ABI 5.1.8: In a generic lambda, uses of auto in the parameter
- // list are mangled as the corresponding artificial template type parameter.
- if (ParsingLambdaParamsAtLevel == Level && Level <= TemplateParams.size()) {
- // This will be popped by the ScopedTemplateParamList in
- // parseUnnamedTypeName.
- if (Level == TemplateParams.size())
- TemplateParams.push_back(nullptr);
- return make<NameType>("auto");
- }
- return nullptr;
- }
- return (*TemplateParams[Level])[Index];
- }
- // <template-param-decl> ::= Ty # type parameter
- // ::= Tn <type> # non-type parameter
- // ::= Tt <template-param-decl>* E # template parameter
- // ::= Tp <template-param-decl> # parameter pack
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseTemplateParamDecl() {
- auto InventTemplateParamName = [&](TemplateParamKind Kind) {
- unsigned Index = NumSyntheticTemplateParameters[(int)Kind]++;
- Node *N = make<SyntheticTemplateParamName>(Kind, Index);
- if (N) TemplateParams.back()->push_back(N);
- return N;
- };
- if (consumeIf("Ty")) {
- Node *Name = InventTemplateParamName(TemplateParamKind::Type);
- if (!Name)
- return nullptr;
- return make<TypeTemplateParamDecl>(Name);
- }
- if (consumeIf("Tn")) {
- Node *Name = InventTemplateParamName(TemplateParamKind::NonType);
- if (!Name)
- return nullptr;
- Node *Type = parseType();
- if (!Type)
- return nullptr;
- return make<NonTypeTemplateParamDecl>(Name, Type);
- }
- if (consumeIf("Tt")) {
- Node *Name = InventTemplateParamName(TemplateParamKind::Template);
- if (!Name)
- return nullptr;
- size_t ParamsBegin = Names.size();
- ScopedTemplateParamList TemplateTemplateParamParams(this);
- while (!consumeIf("E")) {
- Node *P = parseTemplateParamDecl();
- if (!P)
- return nullptr;
- Names.push_back(P);
- }
- NodeArray Params = popTrailingNodeArray(ParamsBegin);
- return make<TemplateTemplateParamDecl>(Name, Params);
- }
- if (consumeIf("Tp")) {
- Node *P = parseTemplateParamDecl();
- if (!P)
- return nullptr;
- return make<TemplateParamPackDecl>(P);
- }
- return nullptr;
- }
- // <template-arg> ::= <type> # type or template
- // ::= X <expression> E # expression
- // ::= <expr-primary> # simple expressions
- // ::= J <template-arg>* E # argument pack
- // ::= LZ <encoding> E # extension
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parseTemplateArg() {
- switch (look()) {
- case 'X': {
- ++First;
- Node *Arg = getDerived().parseExpr();
- if (Arg == nullptr || !consumeIf('E'))
- return nullptr;
- return Arg;
- }
- case 'I': // 'I' here is an old GCC mangling ABI. It is common in outside binaries.
- case 'J': {
- ++First;
- size_t ArgsBegin = Names.size();
- while (!consumeIf('E')) {
- Node *Arg = getDerived().parseTemplateArg();
- if (Arg == nullptr)
- return nullptr;
- Names.push_back(Arg);
- }
- NodeArray Args = popTrailingNodeArray(ArgsBegin);
- return make<TemplateArgumentPack>(Args);
- }
- case 'L': {
- // ::= LZ <encoding> E # extension
- if (look(1) == 'Z') {
- First += 2;
- Node *Arg = getDerived().parseEncoding();
- if (Arg == nullptr || !consumeIf('E'))
- return nullptr;
- return Arg;
- }
- // ::= <expr-primary> # simple expressions
- return getDerived().parseExprPrimary();
- }
- default:
- return getDerived().parseType();
- }
- }
- // <template-args> ::= I <template-arg>* E
- // extension, the abi says <template-arg>+
- template <typename Derived, typename Alloc>
- Node *
- AbstractManglingParser<Derived, Alloc>::parseTemplateArgs(bool TagTemplates) {
- if (!consumeIf('I'))
- return nullptr;
- // <template-params> refer to the innermost <template-args>. Clear out any
- // outer args that we may have inserted into TemplateParams.
- if (TagTemplates) {
- TemplateParams.clear();
- TemplateParams.push_back(&OuterTemplateParams);
- OuterTemplateParams.clear();
- }
- size_t ArgsBegin = Names.size();
- while (!consumeIf('E')) {
- if (TagTemplates) {
- auto OldParams = std::move(TemplateParams);
- Node *Arg = getDerived().parseTemplateArg();
- TemplateParams = std::move(OldParams);
- if (Arg == nullptr)
- return nullptr;
- Names.push_back(Arg);
- Node *TableEntry = Arg;
- if (Arg->getKind() == Node::KTemplateArgumentPack) {
- TableEntry = make<ParameterPack>(
- static_cast<TemplateArgumentPack*>(TableEntry)->getElements());
- if (!TableEntry)
- return nullptr;
- }
- TemplateParams.back()->push_back(TableEntry);
- } else {
- Node *Arg = getDerived().parseTemplateArg();
- if (Arg == nullptr)
- return nullptr;
- Names.push_back(Arg);
- }
- }
- return make<TemplateArgs>(popTrailingNodeArray(ArgsBegin));
- }
- // <mangled-name> ::= _Z <encoding>
- // ::= <type>
- // extension ::= ___Z <encoding> _block_invoke
- // extension ::= ___Z <encoding> _block_invoke<decimal-digit>+
- // extension ::= ___Z <encoding> _block_invoke_<decimal-digit>+
- template <typename Derived, typename Alloc>
- Node *AbstractManglingParser<Derived, Alloc>::parse() {
- if (consumeIf("_Z") || consumeIf("__Z")) {
- Node *Encoding = getDerived().parseEncoding();
- if (Encoding == nullptr)
- return nullptr;
- if (look() == '.') {
- Encoding = make<DotSuffix>(Encoding, StringView(First, Last));
- First = Last;
- }
- if (numLeft() != 0)
- return nullptr;
- return Encoding;
- }
- if (consumeIf("___Z") || consumeIf("____Z")) {
- Node *Encoding = getDerived().parseEncoding();
- if (Encoding == nullptr || !consumeIf("_block_invoke"))
- return nullptr;
- bool RequireNumber = consumeIf('_');
- if (parseNumber().empty() && RequireNumber)
- return nullptr;
- if (look() == '.')
- First = Last;
- if (numLeft() != 0)
- return nullptr;
- return make<SpecialName>("invocation function for block in ", Encoding);
- }
- Node *Ty = getDerived().parseType();
- if (numLeft() != 0)
- return nullptr;
- return Ty;
- }
- template <typename Alloc>
- struct ManglingParser : AbstractManglingParser<ManglingParser<Alloc>, Alloc> {
- using AbstractManglingParser<ManglingParser<Alloc>,
- Alloc>::AbstractManglingParser;
- };
- DEMANGLE_NAMESPACE_END
- #endif // DEMANGLE_ITANIUMDEMANGLE_H
- #ifdef __GNUC__
- #pragma GCC diagnostic pop
- #endif
|