Demangler.cs 118 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367
  1. using System;
  2. using System.Collections.Generic;
  3. using System.IO;
  4. using System.Linq;
  5. using Ryujinx.HLE.HOS.Diagnostics.Demangler.Ast;
  6. namespace Ryujinx.HLE.HOS.Diagnostics.Demangler
  7. {
  8. class Demangler
  9. {
  10. private static readonly string Base36 = "0123456789abcdefghijklmnopqrstuvwxyz";
  11. private List<BaseNode> _substitutionList = new List<BaseNode>();
  12. private List<BaseNode> _templateParamList = new List<BaseNode>();
  13. private List<ForwardTemplateReference> _forwardTemplateReferenceList = new List<ForwardTemplateReference>();
  14. public string Mangled { get; private set; }
  15. private int _position;
  16. private int _length;
  17. private bool _canForwardTemplateReference;
  18. private bool _canParseTemplateArgs;
  19. public Demangler(string mangled)
  20. {
  21. Mangled = mangled;
  22. _position = 0;
  23. _length = mangled.Length;
  24. _canParseTemplateArgs = true;
  25. }
  26. private bool ConsumeIf(string toConsume)
  27. {
  28. string mangledPart = Mangled.Substring(_position);
  29. if (mangledPart.StartsWith(toConsume))
  30. {
  31. _position += toConsume.Length;
  32. return true;
  33. }
  34. return false;
  35. }
  36. private string PeekString(int offset = 0, int length = 1)
  37. {
  38. if (_position + offset >= length)
  39. {
  40. return null;
  41. }
  42. return Mangled.Substring(_position + offset, length);
  43. }
  44. private char Peek(int offset = 0)
  45. {
  46. if (_position + offset >= _length)
  47. {
  48. return '\0';
  49. }
  50. return Mangled[_position + offset];
  51. }
  52. private char Consume()
  53. {
  54. if (_position < _length)
  55. {
  56. return Mangled[_position++];
  57. }
  58. return '\0';
  59. }
  60. private int Count()
  61. {
  62. return _length - _position;
  63. }
  64. private static int FromBase36(string encoded)
  65. {
  66. char[] reversedEncoded = encoded.ToLower().ToCharArray().Reverse().ToArray();
  67. int result = 0;
  68. for (int i = 0; i < reversedEncoded.Length; i++)
  69. {
  70. int value = Base36.IndexOf(reversedEncoded[i]);
  71. if (value == -1)
  72. {
  73. return -1;
  74. }
  75. result += value * (int)Math.Pow(36, i);
  76. }
  77. return result;
  78. }
  79. private int ParseSeqId()
  80. {
  81. string part = Mangled.Substring(_position);
  82. int seqIdLen = 0;
  83. for (; seqIdLen < part.Length; seqIdLen++)
  84. {
  85. if (!char.IsLetterOrDigit(part[seqIdLen]))
  86. {
  87. break;
  88. }
  89. }
  90. _position += seqIdLen;
  91. return FromBase36(part.Substring(0, seqIdLen));
  92. }
  93. // <substitution> ::= S <seq-id> _
  94. // ::= S_
  95. // ::= St # std::
  96. // ::= Sa # std::allocator
  97. // ::= Sb # std::basic_string
  98. // ::= Ss # std::basic_string<char, std::char_traits<char>, std::allocator<char> >
  99. // ::= Si # std::basic_istream<char, std::char_traits<char> >
  100. // ::= So # std::basic_ostream<char, std::char_traits<char> >
  101. // ::= Sd # std::basic_iostream<char, std::char_traits<char> >
  102. private BaseNode ParseSubstitution()
  103. {
  104. if (!ConsumeIf("S"))
  105. {
  106. return null;
  107. }
  108. char substitutionSecondChar = Peek();
  109. if (char.IsLower(substitutionSecondChar))
  110. {
  111. switch (substitutionSecondChar)
  112. {
  113. case 'a':
  114. _position++;
  115. return new SpecialSubstitution(SpecialSubstitution.SpecialType.Allocator);
  116. case 'b':
  117. _position++;
  118. return new SpecialSubstitution(SpecialSubstitution.SpecialType.BasicString);
  119. case 's':
  120. _position++;
  121. return new SpecialSubstitution(SpecialSubstitution.SpecialType.String);
  122. case 'i':
  123. _position++;
  124. return new SpecialSubstitution(SpecialSubstitution.SpecialType.IStream);
  125. case 'o':
  126. _position++;
  127. return new SpecialSubstitution(SpecialSubstitution.SpecialType.OStream);
  128. case 'd':
  129. _position++;
  130. return new SpecialSubstitution(SpecialSubstitution.SpecialType.IOStream);
  131. default:
  132. return null;
  133. }
  134. }
  135. // ::= S_
  136. if (ConsumeIf("_"))
  137. {
  138. if (_substitutionList.Count != 0)
  139. {
  140. return _substitutionList[0];
  141. }
  142. return null;
  143. }
  144. // ::= S <seq-id> _
  145. int seqId = ParseSeqId();
  146. if (seqId < 0)
  147. {
  148. return null;
  149. }
  150. seqId++;
  151. if (!ConsumeIf("_") || seqId >= _substitutionList.Count)
  152. {
  153. return null;
  154. }
  155. return _substitutionList[seqId];
  156. }
  157. // NOTE: thoses data aren't used in the output
  158. // <call-offset> ::= h <nv-offset> _
  159. // ::= v <v-offset> _
  160. // <nv-offset> ::= <offset number>
  161. // # non-virtual base override
  162. // <v-offset> ::= <offset number> _ <virtual offset number>
  163. // # virtual base override, with vcall offset
  164. private bool ParseCallOffset()
  165. {
  166. if (ConsumeIf("h"))
  167. {
  168. return ParseNumber(true).Length == 0 || !ConsumeIf("_");
  169. }
  170. else if (ConsumeIf("v"))
  171. {
  172. return ParseNumber(true).Length == 0 || !ConsumeIf("_") || ParseNumber(true).Length == 0 || !ConsumeIf("_");
  173. }
  174. return true;
  175. }
  176. // <class-enum-type> ::= <name> # non-dependent type name, dependent type name, or dependent typename-specifier
  177. // ::= Ts <name> # dependent elaborated type specifier using 'struct' or 'class'
  178. // ::= Tu <name> # dependent elaborated type specifier using 'union'
  179. // ::= Te <name> # dependent elaborated type specifier using 'enum'
  180. private BaseNode ParseClassEnumType()
  181. {
  182. string elaboratedType = null;
  183. if (ConsumeIf("Ts"))
  184. {
  185. elaboratedType = "struct";
  186. }
  187. else if (ConsumeIf("Tu"))
  188. {
  189. elaboratedType = "union";
  190. }
  191. else if (ConsumeIf("Te"))
  192. {
  193. elaboratedType = "enum";
  194. }
  195. BaseNode name = ParseName();
  196. if (name == null)
  197. {
  198. return null;
  199. }
  200. if (elaboratedType == null)
  201. {
  202. return name;
  203. }
  204. return new ElaboratedType(elaboratedType, name);
  205. }
  206. // <function-type> ::= [<CV-qualifiers>] [<exception-spec>] [Dx] F [Y] <bare-function-type> [<ref-qualifier>] E
  207. // <bare-function-type> ::= <signature type>+
  208. // # types are possible return type, then parameter types
  209. // <exception-spec> ::= Do # non-throwing exception-specification (e.g., noexcept, throw())
  210. // ::= DO <expression> E # computed (instantiation-dependent) noexcept
  211. // ::= Dw <type>+ E # dynamic exception specification with instantiation-dependent types
  212. private BaseNode ParseFunctionType()
  213. {
  214. Cv cvQualifiers = ParseCvQualifiers();
  215. BaseNode exceptionSpec = null;
  216. if (ConsumeIf("Do"))
  217. {
  218. exceptionSpec = new NameType("noexcept");
  219. }
  220. else if (ConsumeIf("DO"))
  221. {
  222. BaseNode expression = ParseExpression();
  223. if (expression == null || !ConsumeIf("E"))
  224. {
  225. return null;
  226. }
  227. exceptionSpec = new NoexceptSpec(expression);
  228. }
  229. else if (ConsumeIf("Dw"))
  230. {
  231. List<BaseNode> types = new List<BaseNode>();
  232. while (!ConsumeIf("E"))
  233. {
  234. BaseNode type = ParseType();
  235. if (type == null)
  236. {
  237. return null;
  238. }
  239. types.Add(type);
  240. }
  241. exceptionSpec = new DynamicExceptionSpec(new NodeArray(types));
  242. }
  243. // We don't need the transaction
  244. ConsumeIf("Dx");
  245. if (!ConsumeIf("F"))
  246. {
  247. return null;
  248. }
  249. // extern "C"
  250. ConsumeIf("Y");
  251. BaseNode returnType = ParseType();
  252. if (returnType == null)
  253. {
  254. return null;
  255. }
  256. Reference referenceQualifier = Reference.None;
  257. List<BaseNode> Params = new List<BaseNode>();
  258. while (true)
  259. {
  260. if (ConsumeIf("E"))
  261. {
  262. break;
  263. }
  264. if (ConsumeIf("v"))
  265. {
  266. continue;
  267. }
  268. if (ConsumeIf("RE"))
  269. {
  270. referenceQualifier = Reference.LValue;
  271. break;
  272. }
  273. else if (ConsumeIf("OE"))
  274. {
  275. referenceQualifier = Reference.RValue;
  276. break;
  277. }
  278. BaseNode type = ParseType();
  279. if (type == null)
  280. {
  281. return null;
  282. }
  283. Params.Add(type);
  284. }
  285. return new FunctionType(returnType, new NodeArray(Params), new CvType(cvQualifiers, null), new SimpleReferenceType(referenceQualifier, null), exceptionSpec);
  286. }
  287. // <array-type> ::= A <positive dimension number> _ <element type>
  288. // ::= A [<dimension expression>] _ <element type>
  289. private BaseNode ParseArrayType()
  290. {
  291. if (!ConsumeIf("A"))
  292. {
  293. return null;
  294. }
  295. BaseNode elementType;
  296. if (char.IsDigit(Peek()))
  297. {
  298. string dimension = ParseNumber();
  299. if (dimension.Length == 0 || !ConsumeIf("_"))
  300. {
  301. return null;
  302. }
  303. elementType = ParseType();
  304. if (elementType == null)
  305. {
  306. return null;
  307. }
  308. return new ArrayType(elementType, dimension);
  309. }
  310. if (!ConsumeIf("_"))
  311. {
  312. BaseNode dimensionExpression = ParseExpression();
  313. if (dimensionExpression == null || !ConsumeIf("_"))
  314. {
  315. return null;
  316. }
  317. elementType = ParseType();
  318. if (elementType == null)
  319. {
  320. return null;
  321. }
  322. return new ArrayType(elementType, dimensionExpression);
  323. }
  324. elementType = ParseType();
  325. if (elementType == null)
  326. {
  327. return null;
  328. }
  329. return new ArrayType(elementType);
  330. }
  331. // <type> ::= <builtin-type>
  332. // ::= <qualified-type> (PARTIAL)
  333. // ::= <function-type>
  334. // ::= <class-enum-type>
  335. // ::= <array-type> (TODO)
  336. // ::= <pointer-to-member-type> (TODO)
  337. // ::= <template-param>
  338. // ::= <template-template-param> <template-args>
  339. // ::= <decltype>
  340. // ::= P <type> # pointer
  341. // ::= R <type> # l-value reference
  342. // ::= O <type> # r-value reference (C++11)
  343. // ::= C <type> # complex pair (C99)
  344. // ::= G <type> # imaginary (C99)
  345. // ::= <substitution> # See Compression below
  346. private BaseNode ParseType(NameParserContext context = null)
  347. {
  348. // Temporary context
  349. if (context == null)
  350. {
  351. context = new NameParserContext();
  352. }
  353. BaseNode result = null;
  354. switch (Peek())
  355. {
  356. case 'r':
  357. case 'V':
  358. case 'K':
  359. int typePos = 0;
  360. if (Peek(typePos) == 'r')
  361. {
  362. typePos++;
  363. }
  364. if (Peek(typePos) == 'V')
  365. {
  366. typePos++;
  367. }
  368. if (Peek(typePos) == 'K')
  369. {
  370. typePos++;
  371. }
  372. if (Peek(typePos) == 'F' || (Peek(typePos) == 'D' && (Peek(typePos + 1) == 'o' || Peek(typePos + 1) == 'O' || Peek(typePos + 1) == 'w' || Peek(typePos + 1) == 'x')))
  373. {
  374. result = ParseFunctionType();
  375. break;
  376. }
  377. Cv cv = ParseCvQualifiers();
  378. result = ParseType(context);
  379. if (result == null)
  380. {
  381. return null;
  382. }
  383. result = new CvType(cv, result);
  384. break;
  385. case 'U':
  386. // TODO: <extended-qualifier>
  387. return null;
  388. case 'v':
  389. _position++;
  390. return new NameType("void");
  391. case 'w':
  392. _position++;
  393. return new NameType("wchar_t");
  394. case 'b':
  395. _position++;
  396. return new NameType("bool");
  397. case 'c':
  398. _position++;
  399. return new NameType("char");
  400. case 'a':
  401. _position++;
  402. return new NameType("signed char");
  403. case 'h':
  404. _position++;
  405. return new NameType("unsigned char");
  406. case 's':
  407. _position++;
  408. return new NameType("short");
  409. case 't':
  410. _position++;
  411. return new NameType("unsigned short");
  412. case 'i':
  413. _position++;
  414. return new NameType("int");
  415. case 'j':
  416. _position++;
  417. return new NameType("unsigned int");
  418. case 'l':
  419. _position++;
  420. return new NameType("long");
  421. case 'm':
  422. _position++;
  423. return new NameType("unsigned long");
  424. case 'x':
  425. _position++;
  426. return new NameType("long long");
  427. case 'y':
  428. _position++;
  429. return new NameType("unsigned long long");
  430. case 'n':
  431. _position++;
  432. return new NameType("__int128");
  433. case 'o':
  434. _position++;
  435. return new NameType("unsigned __int128");
  436. case 'f':
  437. _position++;
  438. return new NameType("float");
  439. case 'd':
  440. _position++;
  441. return new NameType("double");
  442. case 'e':
  443. _position++;
  444. return new NameType("long double");
  445. case 'g':
  446. _position++;
  447. return new NameType("__float128");
  448. case 'z':
  449. _position++;
  450. return new NameType("...");
  451. case 'u':
  452. _position++;
  453. return ParseSourceName();
  454. case 'D':
  455. switch (Peek(1))
  456. {
  457. case 'd':
  458. _position += 2;
  459. return new NameType("decimal64");
  460. case 'e':
  461. _position += 2;
  462. return new NameType("decimal128");
  463. case 'f':
  464. _position += 2;
  465. return new NameType("decimal32");
  466. case 'h':
  467. _position += 2;
  468. // FIXME: GNU c++flit returns this but that is not what is supposed to be returned.
  469. return new NameType("half");
  470. //return new NameType("decimal16");
  471. case 'i':
  472. _position += 2;
  473. return new NameType("char32_t");
  474. case 's':
  475. _position += 2;
  476. return new NameType("char16_t");
  477. case 'a':
  478. _position += 2;
  479. return new NameType("decltype(auto)");
  480. case 'n':
  481. _position += 2;
  482. // FIXME: GNU c++flit returns this but that is not what is supposed to be returned.
  483. return new NameType("decltype(nullptr)");
  484. //return new NameType("std::nullptr_t");
  485. case 't':
  486. case 'T':
  487. _position += 2;
  488. result = ParseDecltype();
  489. break;
  490. case 'o':
  491. case 'O':
  492. case 'w':
  493. case 'x':
  494. result = ParseFunctionType();
  495. break;
  496. default:
  497. return null;
  498. }
  499. break;
  500. case 'F':
  501. result = ParseFunctionType();
  502. break;
  503. case 'A':
  504. return ParseArrayType();
  505. case 'M':
  506. // TODO: <pointer-to-member-type>
  507. _position++;
  508. return null;
  509. case 'T':
  510. // might just be a class enum type
  511. if (Peek(1) == 's' || Peek(1) == 'u' || Peek(1) == 'e')
  512. {
  513. result = ParseClassEnumType();
  514. break;
  515. }
  516. result = ParseTemplateParam();
  517. if (result == null)
  518. {
  519. return null;
  520. }
  521. if (_canParseTemplateArgs && Peek() == 'I')
  522. {
  523. BaseNode templateArguments = ParseTemplateArguments();
  524. if (templateArguments == null)
  525. {
  526. return null;
  527. }
  528. result = new NameTypeWithTemplateArguments(result, templateArguments);
  529. }
  530. break;
  531. case 'P':
  532. _position++;
  533. result = ParseType(context);
  534. if (result == null)
  535. {
  536. return null;
  537. }
  538. result = new PointerType(result);
  539. break;
  540. case 'R':
  541. _position++;
  542. result = ParseType(context);
  543. if (result == null)
  544. {
  545. return null;
  546. }
  547. result = new ReferenceType("&", result);
  548. break;
  549. case 'O':
  550. _position++;
  551. result = ParseType(context);
  552. if (result == null)
  553. {
  554. return null;
  555. }
  556. result = new ReferenceType("&&", result);
  557. break;
  558. case 'C':
  559. _position++;
  560. result = ParseType(context);
  561. if (result == null)
  562. {
  563. return null;
  564. }
  565. result = new PostfixQualifiedType(" complex", result);
  566. break;
  567. case 'G':
  568. _position++;
  569. result = ParseType(context);
  570. if (result == null)
  571. {
  572. return null;
  573. }
  574. result = new PostfixQualifiedType(" imaginary", result);
  575. break;
  576. case 'S':
  577. if (Peek(1) != 't')
  578. {
  579. BaseNode substitution = ParseSubstitution();
  580. if (substitution == null)
  581. {
  582. return null;
  583. }
  584. if (_canParseTemplateArgs && Peek() == 'I')
  585. {
  586. BaseNode templateArgument = ParseTemplateArgument();
  587. if (templateArgument == null)
  588. {
  589. return null;
  590. }
  591. result = new NameTypeWithTemplateArguments(substitution, templateArgument);
  592. break;
  593. }
  594. return substitution;
  595. }
  596. else
  597. {
  598. result = ParseClassEnumType();
  599. break;
  600. }
  601. default:
  602. result = ParseClassEnumType();
  603. break;
  604. }
  605. if (result != null)
  606. {
  607. _substitutionList.Add(result);
  608. }
  609. return result;
  610. }
  611. // <special-name> ::= TV <type> # virtual table
  612. // ::= TT <type> # VTT structure (construction vtable index)
  613. // ::= TI <type> # typeinfo structure
  614. // ::= TS <type> # typeinfo name (null-terminated byte string)
  615. // ::= Tc <call-offset> <call-offset> <base encoding>
  616. // ::= TW <object name> # Thread-local wrapper
  617. // ::= TH <object name> # Thread-local initialization
  618. // ::= T <call-offset> <base encoding>
  619. // # base is the nominal target function of thunk
  620. // ::= GV <object name> # Guard variable for one-time initialization
  621. private BaseNode ParseSpecialName(NameParserContext context = null)
  622. {
  623. if (Peek() != 'T')
  624. {
  625. if (ConsumeIf("GV"))
  626. {
  627. BaseNode name = ParseName();
  628. if (name == null)
  629. {
  630. return null;
  631. }
  632. return new SpecialName("guard variable for ", name);
  633. }
  634. return null;
  635. }
  636. BaseNode node;
  637. switch (Peek(1))
  638. {
  639. // ::= TV <type> # virtual table
  640. case 'V':
  641. _position += 2;
  642. node = ParseType(context);
  643. if (node == null)
  644. {
  645. return null;
  646. }
  647. return new SpecialName("vtable for ", node);
  648. // ::= TT <type> # VTT structure (construction vtable index)
  649. case 'T':
  650. _position += 2;
  651. node = ParseType(context);
  652. if (node == null)
  653. {
  654. return null;
  655. }
  656. return new SpecialName("VTT for ", node);
  657. // ::= TI <type> # typeinfo structure
  658. case 'I':
  659. _position += 2;
  660. node = ParseType(context);
  661. if (node == null)
  662. {
  663. return null;
  664. }
  665. return new SpecialName("typeinfo for ", node);
  666. // ::= TS <type> # typeinfo name (null-terminated byte string)
  667. case 'S':
  668. _position += 2;
  669. node = ParseType(context);
  670. if (node == null)
  671. {
  672. return null;
  673. }
  674. return new SpecialName("typeinfo name for ", node);
  675. // ::= Tc <call-offset> <call-offset> <base encoding>
  676. case 'c':
  677. _position += 2;
  678. if (ParseCallOffset() || ParseCallOffset())
  679. {
  680. return null;
  681. }
  682. node = ParseEncoding();
  683. if (node == null)
  684. {
  685. return null;
  686. }
  687. return new SpecialName("covariant return thunk to ", node);
  688. // extension ::= TC <first type> <number> _ <second type>
  689. case 'C':
  690. _position += 2;
  691. BaseNode firstType = ParseType();
  692. if (firstType == null || ParseNumber(true).Length == 0 || !ConsumeIf("_"))
  693. {
  694. return null;
  695. }
  696. BaseNode secondType = ParseType();
  697. return new CtorVtableSpecialName(secondType, firstType);
  698. // ::= TH <object name> # Thread-local initialization
  699. case 'H':
  700. _position += 2;
  701. node = ParseName();
  702. if (node == null)
  703. {
  704. return null;
  705. }
  706. return new SpecialName("thread-local initialization routine for ", node);
  707. // ::= TW <object name> # Thread-local wrapper
  708. case 'W':
  709. _position += 2;
  710. node = ParseName();
  711. if (node == null)
  712. {
  713. return null;
  714. }
  715. return new SpecialName("thread-local wrapper routine for ", node);
  716. default:
  717. _position++;
  718. bool isVirtual = Peek() == 'v';
  719. if (ParseCallOffset())
  720. {
  721. return null;
  722. }
  723. node = ParseEncoding();
  724. if (node == null)
  725. {
  726. return null;
  727. }
  728. if (isVirtual)
  729. {
  730. return new SpecialName("virtual thunk to ", node);
  731. }
  732. return new SpecialName("non-virtual thunk to ", node);
  733. }
  734. }
  735. // <CV-qualifiers> ::= [r] [V] [K] # restrict (C99), volatile, const
  736. private Cv ParseCvQualifiers()
  737. {
  738. Cv qualifiers = Cv.None;
  739. if (ConsumeIf("r"))
  740. {
  741. qualifiers |= Cv.Restricted;
  742. }
  743. if (ConsumeIf("V"))
  744. {
  745. qualifiers |= Cv.Volatile;
  746. }
  747. if (ConsumeIf("K"))
  748. {
  749. qualifiers |= Cv.Const;
  750. }
  751. return qualifiers;
  752. }
  753. // <ref-qualifier> ::= R # & ref-qualifier
  754. // <ref-qualifier> ::= O # && ref-qualifier
  755. private SimpleReferenceType ParseRefQualifiers()
  756. {
  757. Reference result = Reference.None;
  758. if (ConsumeIf("O"))
  759. {
  760. result = Reference.RValue;
  761. }
  762. else if (ConsumeIf("R"))
  763. {
  764. result = Reference.LValue;
  765. }
  766. return new SimpleReferenceType(result, null);
  767. }
  768. private BaseNode CreateNameNode(BaseNode prev, BaseNode name, NameParserContext context)
  769. {
  770. BaseNode result = name;
  771. if (prev != null)
  772. {
  773. result = new NestedName(name, prev);
  774. }
  775. if (context != null)
  776. {
  777. context.FinishWithTemplateArguments = false;
  778. }
  779. return result;
  780. }
  781. private int ParsePositiveNumber()
  782. {
  783. string part = Mangled.Substring(_position);
  784. int numberLength = 0;
  785. for (; numberLength < part.Length; numberLength++)
  786. {
  787. if (!char.IsDigit(part[numberLength]))
  788. {
  789. break;
  790. }
  791. }
  792. _position += numberLength;
  793. if (numberLength == 0)
  794. {
  795. return -1;
  796. }
  797. return int.Parse(part.Substring(0, numberLength));
  798. }
  799. private string ParseNumber(bool isSigned = false)
  800. {
  801. if (isSigned)
  802. {
  803. ConsumeIf("n");
  804. }
  805. if (Count() == 0 || !char.IsDigit(Mangled[_position]))
  806. {
  807. return null;
  808. }
  809. string part = Mangled.Substring(_position);
  810. int numberLength = 0;
  811. for (; numberLength < part.Length; numberLength++)
  812. {
  813. if (!char.IsDigit(part[numberLength]))
  814. {
  815. break;
  816. }
  817. }
  818. _position += numberLength;
  819. return part.Substring(0, numberLength);
  820. }
  821. // <source-name> ::= <positive length number> <identifier>
  822. private BaseNode ParseSourceName()
  823. {
  824. int length = ParsePositiveNumber();
  825. if (Count() < length || length <= 0)
  826. {
  827. return null;
  828. }
  829. string name = Mangled.Substring(_position, length);
  830. _position += length;
  831. if (name.StartsWith("_GLOBAL__N"))
  832. {
  833. return new NameType("(anonymous namespace)");
  834. }
  835. return new NameType(name);
  836. }
  837. // <operator-name> ::= nw # new
  838. // ::= na # new[]
  839. // ::= dl # delete
  840. // ::= da # delete[]
  841. // ::= ps # + (unary)
  842. // ::= ng # - (unary)
  843. // ::= ad # & (unary)
  844. // ::= de # * (unary)
  845. // ::= co # ~
  846. // ::= pl # +
  847. // ::= mi # -
  848. // ::= ml # *
  849. // ::= dv # /
  850. // ::= rm # %
  851. // ::= an # &
  852. // ::= or # |
  853. // ::= eo # ^
  854. // ::= aS # =
  855. // ::= pL # +=
  856. // ::= mI # -=
  857. // ::= mL # *=
  858. // ::= dV # /=
  859. // ::= rM # %=
  860. // ::= aN # &=
  861. // ::= oR # |=
  862. // ::= eO # ^=
  863. // ::= ls # <<
  864. // ::= rs # >>
  865. // ::= lS # <<=
  866. // ::= rS # >>=
  867. // ::= eq # ==
  868. // ::= ne # !=
  869. // ::= lt # <
  870. // ::= gt # >
  871. // ::= le # <=
  872. // ::= ge # >=
  873. // ::= ss # <=>
  874. // ::= nt # !
  875. // ::= aa # &&
  876. // ::= oo # ||
  877. // ::= pp # ++ (postfix in <expression> context)
  878. // ::= mm # -- (postfix in <expression> context)
  879. // ::= cm # ,
  880. // ::= pm # ->*
  881. // ::= pt # ->
  882. // ::= cl # ()
  883. // ::= ix # []
  884. // ::= qu # ?
  885. // ::= cv <type> # (cast) (TODO)
  886. // ::= li <source-name> # operator ""
  887. // ::= v <digit> <source-name> # vendor extended operator (TODO)
  888. private BaseNode ParseOperatorName(NameParserContext context)
  889. {
  890. switch (Peek())
  891. {
  892. case 'a':
  893. switch (Peek(1))
  894. {
  895. case 'a':
  896. _position += 2;
  897. return new NameType("operator&&");
  898. case 'd':
  899. case 'n':
  900. _position += 2;
  901. return new NameType("operator&");
  902. case 'N':
  903. _position += 2;
  904. return new NameType("operator&=");
  905. case 'S':
  906. _position += 2;
  907. return new NameType("operator=");
  908. default:
  909. return null;
  910. }
  911. case 'c':
  912. switch (Peek(1))
  913. {
  914. case 'l':
  915. _position += 2;
  916. return new NameType("operator()");
  917. case 'm':
  918. _position += 2;
  919. return new NameType("operator,");
  920. case 'o':
  921. _position += 2;
  922. return new NameType("operator~");
  923. case 'v':
  924. _position += 2;
  925. bool canParseTemplateArgsBackup = _canParseTemplateArgs;
  926. bool canForwardTemplateReferenceBackup = _canForwardTemplateReference;
  927. _canParseTemplateArgs = false;
  928. _canForwardTemplateReference = canForwardTemplateReferenceBackup || context != null;
  929. BaseNode type = ParseType();
  930. _canParseTemplateArgs = canParseTemplateArgsBackup;
  931. _canForwardTemplateReference = canForwardTemplateReferenceBackup;
  932. if (type == null)
  933. {
  934. return null;
  935. }
  936. if (context != null)
  937. {
  938. context.CtorDtorConversion = true;
  939. }
  940. return new ConversionOperatorType(type);
  941. default:
  942. return null;
  943. }
  944. case 'd':
  945. switch (Peek(1))
  946. {
  947. case 'a':
  948. _position += 2;
  949. return new NameType("operator delete[]");
  950. case 'e':
  951. _position += 2;
  952. return new NameType("operator*");
  953. case 'l':
  954. _position += 2;
  955. return new NameType("operator delete");
  956. case 'v':
  957. _position += 2;
  958. return new NameType("operator/");
  959. case 'V':
  960. _position += 2;
  961. return new NameType("operator/=");
  962. default:
  963. return null;
  964. }
  965. case 'e':
  966. switch (Peek(1))
  967. {
  968. case 'o':
  969. _position += 2;
  970. return new NameType("operator^");
  971. case 'O':
  972. _position += 2;
  973. return new NameType("operator^=");
  974. case 'q':
  975. _position += 2;
  976. return new NameType("operator==");
  977. default:
  978. return null;
  979. }
  980. case 'g':
  981. switch (Peek(1))
  982. {
  983. case 'e':
  984. _position += 2;
  985. return new NameType("operator>=");
  986. case 't':
  987. _position += 2;
  988. return new NameType("operator>");
  989. default:
  990. return null;
  991. }
  992. case 'i':
  993. if (Peek(1) == 'x')
  994. {
  995. _position += 2;
  996. return new NameType("operator[]");
  997. }
  998. return null;
  999. case 'l':
  1000. switch (Peek(1))
  1001. {
  1002. case 'e':
  1003. _position += 2;
  1004. return new NameType("operator<=");
  1005. case 'i':
  1006. _position += 2;
  1007. BaseNode sourceName = ParseSourceName();
  1008. if (sourceName == null)
  1009. {
  1010. return null;
  1011. }
  1012. return new LiteralOperator(sourceName);
  1013. case 's':
  1014. _position += 2;
  1015. return new NameType("operator<<");
  1016. case 'S':
  1017. _position += 2;
  1018. return new NameType("operator<<=");
  1019. case 't':
  1020. _position += 2;
  1021. return new NameType("operator<");
  1022. default:
  1023. return null;
  1024. }
  1025. case 'm':
  1026. switch (Peek(1))
  1027. {
  1028. case 'i':
  1029. _position += 2;
  1030. return new NameType("operator-");
  1031. case 'I':
  1032. _position += 2;
  1033. return new NameType("operator-=");
  1034. case 'l':
  1035. _position += 2;
  1036. return new NameType("operator*");
  1037. case 'L':
  1038. _position += 2;
  1039. return new NameType("operator*=");
  1040. case 'm':
  1041. _position += 2;
  1042. return new NameType("operator--");
  1043. default:
  1044. return null;
  1045. }
  1046. case 'n':
  1047. switch (Peek(1))
  1048. {
  1049. case 'a':
  1050. _position += 2;
  1051. return new NameType("operator new[]");
  1052. case 'e':
  1053. _position += 2;
  1054. return new NameType("operator!=");
  1055. case 'g':
  1056. _position += 2;
  1057. return new NameType("operator-");
  1058. case 't':
  1059. _position += 2;
  1060. return new NameType("operator!");
  1061. case 'w':
  1062. _position += 2;
  1063. return new NameType("operator new");
  1064. default:
  1065. return null;
  1066. }
  1067. case 'o':
  1068. switch (Peek(1))
  1069. {
  1070. case 'o':
  1071. _position += 2;
  1072. return new NameType("operator||");
  1073. case 'r':
  1074. _position += 2;
  1075. return new NameType("operator|");
  1076. case 'R':
  1077. _position += 2;
  1078. return new NameType("operator|=");
  1079. default:
  1080. return null;
  1081. }
  1082. case 'p':
  1083. switch (Peek(1))
  1084. {
  1085. case 'm':
  1086. _position += 2;
  1087. return new NameType("operator->*");
  1088. case 's':
  1089. case 'l':
  1090. _position += 2;
  1091. return new NameType("operator+");
  1092. case 'L':
  1093. _position += 2;
  1094. return new NameType("operator+=");
  1095. case 'p':
  1096. _position += 2;
  1097. return new NameType("operator++");
  1098. case 't':
  1099. _position += 2;
  1100. return new NameType("operator->");
  1101. default:
  1102. return null;
  1103. }
  1104. case 'q':
  1105. if (Peek(1) == 'u')
  1106. {
  1107. _position += 2;
  1108. return new NameType("operator?");
  1109. }
  1110. return null;
  1111. case 'r':
  1112. switch (Peek(1))
  1113. {
  1114. case 'm':
  1115. _position += 2;
  1116. return new NameType("operator%");
  1117. case 'M':
  1118. _position += 2;
  1119. return new NameType("operator%=");
  1120. case 's':
  1121. _position += 2;
  1122. return new NameType("operator>>");
  1123. case 'S':
  1124. _position += 2;
  1125. return new NameType("operator>>=");
  1126. default:
  1127. return null;
  1128. }
  1129. case 's':
  1130. if (Peek(1) == 's')
  1131. {
  1132. _position += 2;
  1133. return new NameType("operator<=>");
  1134. }
  1135. return null;
  1136. case 'v':
  1137. // TODO: ::= v <digit> <source-name> # vendor extended operator
  1138. return null;
  1139. default:
  1140. return null;
  1141. }
  1142. }
  1143. // <unqualified-name> ::= <operator-name> [<abi-tags> (TODO)]
  1144. // ::= <ctor-dtor-name> (TODO)
  1145. // ::= <source-name>
  1146. // ::= <unnamed-type-name> (TODO)
  1147. // ::= DC <source-name>+ E # structured binding declaration (TODO)
  1148. private BaseNode ParseUnqualifiedName(NameParserContext context)
  1149. {
  1150. BaseNode result = null;
  1151. char c = Peek();
  1152. if (c == 'U')
  1153. {
  1154. // TODO: Unnamed Type Name
  1155. // throw new Exception("Unnamed Type Name not implemented");
  1156. }
  1157. else if (char.IsDigit(c))
  1158. {
  1159. result = ParseSourceName();
  1160. }
  1161. else if (ConsumeIf("DC"))
  1162. {
  1163. // TODO: Structured Binding Declaration
  1164. // throw new Exception("Structured Binding Declaration not implemented");
  1165. }
  1166. else
  1167. {
  1168. result = ParseOperatorName(context);
  1169. }
  1170. if (result != null)
  1171. {
  1172. // TODO: ABI Tags
  1173. //throw new Exception("ABI Tags not implemented");
  1174. }
  1175. return result;
  1176. }
  1177. // <ctor-dtor-name> ::= C1 # complete object constructor
  1178. // ::= C2 # base object constructor
  1179. // ::= C3 # complete object allocating constructor
  1180. // ::= D0 # deleting destructor
  1181. // ::= D1 # complete object destructor
  1182. // ::= D2 # base object destructor
  1183. private BaseNode ParseCtorDtorName(NameParserContext context, BaseNode prev)
  1184. {
  1185. if (prev.Type == NodeType.SpecialSubstitution && prev is SpecialSubstitution)
  1186. {
  1187. ((SpecialSubstitution)prev).SetExtended();
  1188. }
  1189. if (ConsumeIf("C"))
  1190. {
  1191. bool isInherited = ConsumeIf("I");
  1192. char ctorDtorType = Peek();
  1193. if (ctorDtorType != '1' && ctorDtorType != '2' && ctorDtorType != '3')
  1194. {
  1195. return null;
  1196. }
  1197. _position++;
  1198. if (context != null)
  1199. {
  1200. context.CtorDtorConversion = true;
  1201. }
  1202. if (isInherited && ParseName(context) == null)
  1203. {
  1204. return null;
  1205. }
  1206. return new CtorDtorNameType(prev, false);
  1207. }
  1208. if (ConsumeIf("D"))
  1209. {
  1210. char c = Peek();
  1211. if (c != '0' && c != '1' && c != '2')
  1212. {
  1213. return null;
  1214. }
  1215. _position++;
  1216. if (context != null)
  1217. {
  1218. context.CtorDtorConversion = true;
  1219. }
  1220. return new CtorDtorNameType(prev, true);
  1221. }
  1222. return null;
  1223. }
  1224. // <function-param> ::= fp <top-level CV-qualifiers> _ # L == 0, first parameter
  1225. // ::= fp <top-level CV-qualifiers> <parameter-2 non-negative number> _ # L == 0, second and later parameters
  1226. // ::= fL <L-1 non-negative number> p <top-level CV-qualifiers> _ # L > 0, first parameter
  1227. // ::= fL <L-1 non-negative number> p <top-level CV-qualifiers> <parameter-2 non-negative number> _ # L > 0, second and later parameters
  1228. private BaseNode ParseFunctionParameter()
  1229. {
  1230. if (ConsumeIf("fp"))
  1231. {
  1232. // ignored
  1233. ParseCvQualifiers();
  1234. if (!ConsumeIf("_"))
  1235. {
  1236. return null;
  1237. }
  1238. return new FunctionParameter(ParseNumber());
  1239. }
  1240. else if (ConsumeIf("fL"))
  1241. {
  1242. string l1Number = ParseNumber();
  1243. if (l1Number == null || l1Number.Length == 0)
  1244. {
  1245. return null;
  1246. }
  1247. if (!ConsumeIf("p"))
  1248. {
  1249. return null;
  1250. }
  1251. // ignored
  1252. ParseCvQualifiers();
  1253. if (!ConsumeIf("_"))
  1254. {
  1255. return null;
  1256. }
  1257. return new FunctionParameter(ParseNumber());
  1258. }
  1259. return null;
  1260. }
  1261. // <fold-expr> ::= fL <binary-operator-name> <expression> <expression>
  1262. // ::= fR <binary-operator-name> <expression> <expression>
  1263. // ::= fl <binary-operator-name> <expression>
  1264. // ::= fr <binary-operator-name> <expression>
  1265. private BaseNode ParseFoldExpression()
  1266. {
  1267. if (!ConsumeIf("f"))
  1268. {
  1269. return null;
  1270. }
  1271. char foldKind = Peek();
  1272. bool hasInitializer = foldKind == 'L' || foldKind == 'R';
  1273. bool isLeftFold = foldKind == 'l' || foldKind == 'L';
  1274. if (!isLeftFold && !(foldKind == 'r' || foldKind == 'R'))
  1275. {
  1276. return null;
  1277. }
  1278. _position++;
  1279. string operatorName = null;
  1280. switch (PeekString(0, 2))
  1281. {
  1282. case "aa":
  1283. operatorName = "&&";
  1284. break;
  1285. case "an":
  1286. operatorName = "&";
  1287. break;
  1288. case "aN":
  1289. operatorName = "&=";
  1290. break;
  1291. case "aS":
  1292. operatorName = "=";
  1293. break;
  1294. case "cm":
  1295. operatorName = ",";
  1296. break;
  1297. case "ds":
  1298. operatorName = ".*";
  1299. break;
  1300. case "dv":
  1301. operatorName = "/";
  1302. break;
  1303. case "dV":
  1304. operatorName = "/=";
  1305. break;
  1306. case "eo":
  1307. operatorName = "^";
  1308. break;
  1309. case "eO":
  1310. operatorName = "^=";
  1311. break;
  1312. case "eq":
  1313. operatorName = "==";
  1314. break;
  1315. case "ge":
  1316. operatorName = ">=";
  1317. break;
  1318. case "gt":
  1319. operatorName = ">";
  1320. break;
  1321. case "le":
  1322. operatorName = "<=";
  1323. break;
  1324. case "ls":
  1325. operatorName = "<<";
  1326. break;
  1327. case "lS":
  1328. operatorName = "<<=";
  1329. break;
  1330. case "lt":
  1331. operatorName = "<";
  1332. break;
  1333. case "mi":
  1334. operatorName = "-";
  1335. break;
  1336. case "mI":
  1337. operatorName = "-=";
  1338. break;
  1339. case "ml":
  1340. operatorName = "*";
  1341. break;
  1342. case "mL":
  1343. operatorName = "*=";
  1344. break;
  1345. case "ne":
  1346. operatorName = "!=";
  1347. break;
  1348. case "oo":
  1349. operatorName = "||";
  1350. break;
  1351. case "or":
  1352. operatorName = "|";
  1353. break;
  1354. case "oR":
  1355. operatorName = "|=";
  1356. break;
  1357. case "pl":
  1358. operatorName = "+";
  1359. break;
  1360. case "pL":
  1361. operatorName = "+=";
  1362. break;
  1363. case "rm":
  1364. operatorName = "%";
  1365. break;
  1366. case "rM":
  1367. operatorName = "%=";
  1368. break;
  1369. case "rs":
  1370. operatorName = ">>";
  1371. break;
  1372. case "rS":
  1373. operatorName = ">>=";
  1374. break;
  1375. default:
  1376. return null;
  1377. }
  1378. _position += 2;
  1379. BaseNode expression = ParseExpression();
  1380. if (expression == null)
  1381. {
  1382. return null;
  1383. }
  1384. BaseNode initializer = null;
  1385. if (hasInitializer)
  1386. {
  1387. initializer = ParseExpression();
  1388. if (initializer == null)
  1389. {
  1390. return null;
  1391. }
  1392. }
  1393. if (isLeftFold && initializer != null)
  1394. {
  1395. BaseNode temp = expression;
  1396. expression = initializer;
  1397. initializer = temp;
  1398. }
  1399. return new FoldExpression(isLeftFold, operatorName, new PackedTemplateParameterExpansion(expression), initializer);
  1400. }
  1401. // ::= cv <type> <expression> # type (expression), conversion with one argument
  1402. // ::= cv <type> _ <expression>* E # type (expr-list), conversion with other than one argument
  1403. private BaseNode ParseConversionExpression()
  1404. {
  1405. if (!ConsumeIf("cv"))
  1406. {
  1407. return null;
  1408. }
  1409. bool canParseTemplateArgsBackup = _canParseTemplateArgs;
  1410. _canParseTemplateArgs = false;
  1411. BaseNode type = ParseType();
  1412. _canParseTemplateArgs = canParseTemplateArgsBackup;
  1413. if (type == null)
  1414. {
  1415. return null;
  1416. }
  1417. List<BaseNode> expressions = new List<BaseNode>();
  1418. if (ConsumeIf("_"))
  1419. {
  1420. while (!ConsumeIf("E"))
  1421. {
  1422. BaseNode expression = ParseExpression();
  1423. if (expression == null)
  1424. {
  1425. return null;
  1426. }
  1427. expressions.Add(expression);
  1428. }
  1429. }
  1430. else
  1431. {
  1432. BaseNode expression = ParseExpression();
  1433. if (expression == null)
  1434. {
  1435. return null;
  1436. }
  1437. expressions.Add(expression);
  1438. }
  1439. return new ConversionExpression(type, new NodeArray(expressions));
  1440. }
  1441. private BaseNode ParseBinaryExpression(string name)
  1442. {
  1443. BaseNode leftPart = ParseExpression();
  1444. if (leftPart == null)
  1445. {
  1446. return null;
  1447. }
  1448. BaseNode rightPart = ParseExpression();
  1449. if (rightPart == null)
  1450. {
  1451. return null;
  1452. }
  1453. return new BinaryExpression(leftPart, name, rightPart);
  1454. }
  1455. private BaseNode ParsePrefixExpression(string name)
  1456. {
  1457. BaseNode expression = ParseExpression();
  1458. if (expression == null)
  1459. {
  1460. return null;
  1461. }
  1462. return new PrefixExpression(name, expression);
  1463. }
  1464. // <braced-expression> ::= <expression>
  1465. // ::= di <field source-name> <braced-expression> # .name = expr
  1466. // ::= dx <index expression> <braced-expression> # [expr] = expr
  1467. // ::= dX <range begin expression> <range end expression> <braced-expression>
  1468. // # [expr ... expr] = expr
  1469. private BaseNode ParseBracedExpression()
  1470. {
  1471. if (Peek() == 'd')
  1472. {
  1473. BaseNode bracedExpressionNode;
  1474. switch (Peek(1))
  1475. {
  1476. case 'i':
  1477. _position += 2;
  1478. BaseNode field = ParseSourceName();
  1479. if (field == null)
  1480. {
  1481. return null;
  1482. }
  1483. bracedExpressionNode = ParseBracedExpression();
  1484. if (bracedExpressionNode == null)
  1485. {
  1486. return null;
  1487. }
  1488. return new BracedExpression(field, bracedExpressionNode, false);
  1489. case 'x':
  1490. _position += 2;
  1491. BaseNode index = ParseExpression();
  1492. if (index == null)
  1493. {
  1494. return null;
  1495. }
  1496. bracedExpressionNode = ParseBracedExpression();
  1497. if (bracedExpressionNode == null)
  1498. {
  1499. return null;
  1500. }
  1501. return new BracedExpression(index, bracedExpressionNode, true);
  1502. case 'X':
  1503. _position += 2;
  1504. BaseNode rangeBeginExpression = ParseExpression();
  1505. if (rangeBeginExpression == null)
  1506. {
  1507. return null;
  1508. }
  1509. BaseNode rangeEndExpression = ParseExpression();
  1510. if (rangeEndExpression == null)
  1511. {
  1512. return null;
  1513. }
  1514. bracedExpressionNode = ParseBracedExpression();
  1515. if (bracedExpressionNode == null)
  1516. {
  1517. return null;
  1518. }
  1519. return new BracedRangeExpression(rangeBeginExpression, rangeEndExpression, bracedExpressionNode);
  1520. }
  1521. }
  1522. return ParseExpression();
  1523. }
  1524. // ::= [gs] nw <expression>* _ <type> E # new (expr-list) type
  1525. // ::= [gs] nw <expression>* _ <type> <initializer> # new (expr-list) type (init)
  1526. // ::= [gs] na <expression>* _ <type> E # new[] (expr-list) type
  1527. // ::= [gs] na <expression>* _ <type> <initializer> # new[] (expr-list) type (init)
  1528. //
  1529. // <initializer> ::= pi <expression>* E # parenthesized initialization
  1530. private BaseNode ParseNewExpression()
  1531. {
  1532. bool isGlobal = ConsumeIf("gs");
  1533. bool isArray = Peek(1) == 'a';
  1534. if (!ConsumeIf("nw") || !ConsumeIf("na"))
  1535. {
  1536. return null;
  1537. }
  1538. List<BaseNode> expressions = new List<BaseNode>();
  1539. List<BaseNode> initializers = new List<BaseNode>();
  1540. while (!ConsumeIf("_"))
  1541. {
  1542. BaseNode expression = ParseExpression();
  1543. if (expression == null)
  1544. {
  1545. return null;
  1546. }
  1547. expressions.Add(expression);
  1548. }
  1549. BaseNode typeNode = ParseType();
  1550. if (typeNode == null)
  1551. {
  1552. return null;
  1553. }
  1554. if (ConsumeIf("pi"))
  1555. {
  1556. while (!ConsumeIf("E"))
  1557. {
  1558. BaseNode initializer = ParseExpression();
  1559. if (initializer == null)
  1560. {
  1561. return null;
  1562. }
  1563. initializers.Add(initializer);
  1564. }
  1565. }
  1566. else if (!ConsumeIf("E"))
  1567. {
  1568. return null;
  1569. }
  1570. return new NewExpression(new NodeArray(expressions), typeNode, new NodeArray(initializers), isGlobal, isArray);
  1571. }
  1572. // <expression> ::= <unary operator-name> <expression>
  1573. // ::= <binary operator-name> <expression> <expression>
  1574. // ::= <ternary operator-name> <expression> <expression> <expression>
  1575. // ::= pp_ <expression> # prefix ++
  1576. // ::= mm_ <expression> # prefix --
  1577. // ::= cl <expression>+ E # expression (expr-list), call
  1578. // ::= cv <type> <expression> # type (expression), conversion with one argument
  1579. // ::= cv <type> _ <expression>* E # type (expr-list), conversion with other than one argument
  1580. // ::= tl <type> <braced-expression>* E # type {expr-list}, conversion with braced-init-list argument
  1581. // ::= il <braced-expression>* E # {expr-list}, braced-init-list in any other context
  1582. // ::= [gs] nw <expression>* _ <type> E # new (expr-list) type
  1583. // ::= [gs] nw <expression>* _ <type> <initializer> # new (expr-list) type (init)
  1584. // ::= [gs] na <expression>* _ <type> E # new[] (expr-list) type
  1585. // ::= [gs] na <expression>* _ <type> <initializer> # new[] (expr-list) type (init)
  1586. // ::= [gs] dl <expression> # delete expression
  1587. // ::= [gs] da <expression> # delete[] expression
  1588. // ::= dc <type> <expression> # dynamic_cast<type> (expression)
  1589. // ::= sc <type> <expression> # static_cast<type> (expression)
  1590. // ::= cc <type> <expression> # const_cast<type> (expression)
  1591. // ::= rc <type> <expression> # reinterpret_cast<type> (expression)
  1592. // ::= ti <type> # typeid (type)
  1593. // ::= te <expression> # typeid (expression)
  1594. // ::= st <type> # sizeof (type)
  1595. // ::= sz <expression> # sizeof (expression)
  1596. // ::= at <type> # alignof (type)
  1597. // ::= az <expression> # alignof (expression)
  1598. // ::= nx <expression> # noexcept (expression)
  1599. // ::= <template-param>
  1600. // ::= <function-param>
  1601. // ::= dt <expression> <unresolved-name> # expr.name
  1602. // ::= pt <expression> <unresolved-name> # expr->name
  1603. // ::= ds <expression> <expression> # expr.*expr
  1604. // ::= sZ <template-param> # sizeof...(T), size of a template parameter pack
  1605. // ::= sZ <function-param> # sizeof...(parameter), size of a function parameter pack
  1606. // ::= sP <template-arg>* E # sizeof...(T), size of a captured template parameter pack from an alias template
  1607. // ::= sp <expression> # expression..., pack expansion
  1608. // ::= tw <expression> # throw expression
  1609. // ::= tr # throw with no operand (rethrow)
  1610. // ::= <unresolved-name> # f(p), N::f(p), ::f(p),
  1611. // # freestanding dependent name (e.g., T::x),
  1612. // # objectless nonstatic member reference
  1613. // ::= <expr-primary>
  1614. private BaseNode ParseExpression()
  1615. {
  1616. bool isGlobal = ConsumeIf("gs");
  1617. BaseNode expression = null;
  1618. if (Count() < 2)
  1619. {
  1620. return null;
  1621. }
  1622. switch (Peek())
  1623. {
  1624. case 'L':
  1625. return ParseExpressionPrimary();
  1626. case 'T':
  1627. return ParseTemplateParam();
  1628. case 'f':
  1629. char c = Peek(1);
  1630. if (c == 'p' || (c == 'L' && char.IsDigit(Peek(2))))
  1631. {
  1632. return ParseFunctionParameter();
  1633. }
  1634. return ParseFoldExpression();
  1635. case 'a':
  1636. switch (Peek(1))
  1637. {
  1638. case 'a':
  1639. _position += 2;
  1640. return ParseBinaryExpression("&&");
  1641. case 'd':
  1642. case 'n':
  1643. _position += 2;
  1644. return ParseBinaryExpression("&");
  1645. case 'N':
  1646. _position += 2;
  1647. return ParseBinaryExpression("&=");
  1648. case 'S':
  1649. _position += 2;
  1650. return ParseBinaryExpression("=");
  1651. case 't':
  1652. _position += 2;
  1653. BaseNode type = ParseType();
  1654. if (type == null)
  1655. {
  1656. return null;
  1657. }
  1658. return new EnclosedExpression("alignof (", type, ")");
  1659. case 'z':
  1660. _position += 2;
  1661. expression = ParseExpression();
  1662. if (expression == null)
  1663. {
  1664. return null;
  1665. }
  1666. return new EnclosedExpression("alignof (", expression, ")");
  1667. }
  1668. return null;
  1669. case 'c':
  1670. switch (Peek(1))
  1671. {
  1672. case 'c':
  1673. _position += 2;
  1674. BaseNode to = ParseType();
  1675. if (to == null)
  1676. {
  1677. return null;
  1678. }
  1679. BaseNode from = ParseExpression();
  1680. if (from == null)
  1681. {
  1682. return null;
  1683. }
  1684. return new CastExpression("const_cast", to, from);
  1685. case 'l':
  1686. _position += 2;
  1687. BaseNode callee = ParseExpression();
  1688. if (callee == null)
  1689. {
  1690. return null;
  1691. }
  1692. List<BaseNode> names = new List<BaseNode>();
  1693. while (!ConsumeIf("E"))
  1694. {
  1695. expression = ParseExpression();
  1696. if (expression == null)
  1697. {
  1698. return null;
  1699. }
  1700. names.Add(expression);
  1701. }
  1702. return new CallExpression(callee, names);
  1703. case 'm':
  1704. _position += 2;
  1705. return ParseBinaryExpression(",");
  1706. case 'o':
  1707. _position += 2;
  1708. return ParsePrefixExpression("~");
  1709. case 'v':
  1710. return ParseConversionExpression();
  1711. }
  1712. return null;
  1713. case 'd':
  1714. BaseNode leftNode = null;
  1715. BaseNode rightNode = null;
  1716. switch (Peek(1))
  1717. {
  1718. case 'a':
  1719. _position += 2;
  1720. expression = ParseExpression();
  1721. if (expression == null)
  1722. {
  1723. return expression;
  1724. }
  1725. return new DeleteExpression(expression, isGlobal, true);
  1726. case 'c':
  1727. _position += 2;
  1728. BaseNode type = ParseType();
  1729. if (type == null)
  1730. {
  1731. return null;
  1732. }
  1733. expression = ParseExpression();
  1734. if (expression == null)
  1735. {
  1736. return expression;
  1737. }
  1738. return new CastExpression("dynamic_cast", type, expression);
  1739. case 'e':
  1740. _position += 2;
  1741. return ParsePrefixExpression("*");
  1742. case 'l':
  1743. _position += 2;
  1744. expression = ParseExpression();
  1745. if (expression == null)
  1746. {
  1747. return null;
  1748. }
  1749. return new DeleteExpression(expression, isGlobal, false);
  1750. case 'n':
  1751. return ParseUnresolvedName();
  1752. case 's':
  1753. _position += 2;
  1754. leftNode = ParseExpression();
  1755. if (leftNode == null)
  1756. {
  1757. return null;
  1758. }
  1759. rightNode = ParseExpression();
  1760. if (rightNode == null)
  1761. {
  1762. return null;
  1763. }
  1764. return new MemberExpression(leftNode, ".*", rightNode);
  1765. case 't':
  1766. _position += 2;
  1767. leftNode = ParseExpression();
  1768. if (leftNode == null)
  1769. {
  1770. return null;
  1771. }
  1772. rightNode = ParseExpression();
  1773. if (rightNode == null)
  1774. {
  1775. return null;
  1776. }
  1777. return new MemberExpression(leftNode, ".", rightNode);
  1778. case 'v':
  1779. _position += 2;
  1780. return ParseBinaryExpression("/");
  1781. case 'V':
  1782. _position += 2;
  1783. return ParseBinaryExpression("/=");
  1784. }
  1785. return null;
  1786. case 'e':
  1787. switch (Peek(1))
  1788. {
  1789. case 'o':
  1790. _position += 2;
  1791. return ParseBinaryExpression("^");
  1792. case 'O':
  1793. _position += 2;
  1794. return ParseBinaryExpression("^=");
  1795. case 'q':
  1796. _position += 2;
  1797. return ParseBinaryExpression("==");
  1798. }
  1799. return null;
  1800. case 'g':
  1801. switch (Peek(1))
  1802. {
  1803. case 'e':
  1804. _position += 2;
  1805. return ParseBinaryExpression(">=");
  1806. case 't':
  1807. _position += 2;
  1808. return ParseBinaryExpression(">");
  1809. }
  1810. return null;
  1811. case 'i':
  1812. switch (Peek(1))
  1813. {
  1814. case 'x':
  1815. _position += 2;
  1816. BaseNode Base = ParseExpression();
  1817. if (Base == null)
  1818. {
  1819. return null;
  1820. }
  1821. BaseNode subscript = ParseExpression();
  1822. if (Base == null)
  1823. {
  1824. return null;
  1825. }
  1826. return new ArraySubscriptingExpression(Base, subscript);
  1827. case 'l':
  1828. _position += 2;
  1829. List<BaseNode> bracedExpressions = new List<BaseNode>();
  1830. while (!ConsumeIf("E"))
  1831. {
  1832. expression = ParseBracedExpression();
  1833. if (expression == null)
  1834. {
  1835. return null;
  1836. }
  1837. bracedExpressions.Add(expression);
  1838. }
  1839. return new InitListExpression(null, bracedExpressions);
  1840. }
  1841. return null;
  1842. case 'l':
  1843. switch (Peek(1))
  1844. {
  1845. case 'e':
  1846. _position += 2;
  1847. return ParseBinaryExpression("<=");
  1848. case 's':
  1849. _position += 2;
  1850. return ParseBinaryExpression("<<");
  1851. case 'S':
  1852. _position += 2;
  1853. return ParseBinaryExpression("<<=");
  1854. case 't':
  1855. _position += 2;
  1856. return ParseBinaryExpression("<");
  1857. }
  1858. return null;
  1859. case 'm':
  1860. switch (Peek(1))
  1861. {
  1862. case 'i':
  1863. _position += 2;
  1864. return ParseBinaryExpression("-");
  1865. case 'I':
  1866. _position += 2;
  1867. return ParseBinaryExpression("-=");
  1868. case 'l':
  1869. _position += 2;
  1870. return ParseBinaryExpression("*");
  1871. case 'L':
  1872. _position += 2;
  1873. return ParseBinaryExpression("*=");
  1874. case 'm':
  1875. _position += 2;
  1876. if (ConsumeIf("_"))
  1877. {
  1878. return ParsePrefixExpression("--");
  1879. }
  1880. expression = ParseExpression();
  1881. if (expression == null)
  1882. {
  1883. return null;
  1884. }
  1885. return new PostfixExpression(expression, "--");
  1886. }
  1887. return null;
  1888. case 'n':
  1889. switch (Peek(1))
  1890. {
  1891. case 'a':
  1892. case 'w':
  1893. _position += 2;
  1894. return ParseNewExpression();
  1895. case 'e':
  1896. _position += 2;
  1897. return ParseBinaryExpression("!=");
  1898. case 'g':
  1899. _position += 2;
  1900. return ParsePrefixExpression("-");
  1901. case 't':
  1902. _position += 2;
  1903. return ParsePrefixExpression("!");
  1904. case 'x':
  1905. _position += 2;
  1906. expression = ParseExpression();
  1907. if (expression == null)
  1908. {
  1909. return null;
  1910. }
  1911. return new EnclosedExpression("noexcept (", expression, ")");
  1912. }
  1913. return null;
  1914. case 'o':
  1915. switch (Peek(1))
  1916. {
  1917. case 'n':
  1918. return ParseUnresolvedName();
  1919. case 'o':
  1920. _position += 2;
  1921. return ParseBinaryExpression("||");
  1922. case 'r':
  1923. _position += 2;
  1924. return ParseBinaryExpression("|");
  1925. case 'R':
  1926. _position += 2;
  1927. return ParseBinaryExpression("|=");
  1928. }
  1929. return null;
  1930. case 'p':
  1931. switch (Peek(1))
  1932. {
  1933. case 'm':
  1934. _position += 2;
  1935. return ParseBinaryExpression("->*");
  1936. case 'l':
  1937. case 's':
  1938. _position += 2;
  1939. return ParseBinaryExpression("+");
  1940. case 'L':
  1941. _position += 2;
  1942. return ParseBinaryExpression("+=");
  1943. case 'p':
  1944. _position += 2;
  1945. if (ConsumeIf("_"))
  1946. {
  1947. return ParsePrefixExpression("++");
  1948. }
  1949. expression = ParseExpression();
  1950. if (expression == null)
  1951. {
  1952. return null;
  1953. }
  1954. return new PostfixExpression(expression, "++");
  1955. case 't':
  1956. _position += 2;
  1957. leftNode = ParseExpression();
  1958. if (leftNode == null)
  1959. {
  1960. return null;
  1961. }
  1962. rightNode = ParseExpression();
  1963. if (rightNode == null)
  1964. {
  1965. return null;
  1966. }
  1967. return new MemberExpression(leftNode, "->", rightNode);
  1968. }
  1969. return null;
  1970. case 'q':
  1971. if (Peek(1) == 'u')
  1972. {
  1973. _position += 2;
  1974. BaseNode condition = ParseExpression();
  1975. if (condition == null)
  1976. {
  1977. return null;
  1978. }
  1979. leftNode = ParseExpression();
  1980. if (leftNode == null)
  1981. {
  1982. return null;
  1983. }
  1984. rightNode = ParseExpression();
  1985. if (rightNode == null)
  1986. {
  1987. return null;
  1988. }
  1989. return new ConditionalExpression(condition, leftNode, rightNode);
  1990. }
  1991. return null;
  1992. case 'r':
  1993. switch (Peek(1))
  1994. {
  1995. case 'c':
  1996. _position += 2;
  1997. BaseNode to = ParseType();
  1998. if (to == null)
  1999. {
  2000. return null;
  2001. }
  2002. BaseNode from = ParseExpression();
  2003. if (from == null)
  2004. {
  2005. return null;
  2006. }
  2007. return new CastExpression("reinterpret_cast", to, from);
  2008. case 'm':
  2009. _position += 2;
  2010. return ParseBinaryExpression("%");
  2011. case 'M':
  2012. _position += 2;
  2013. return ParseBinaryExpression("%");
  2014. case 's':
  2015. _position += 2;
  2016. return ParseBinaryExpression(">>");
  2017. case 'S':
  2018. _position += 2;
  2019. return ParseBinaryExpression(">>=");
  2020. }
  2021. return null;
  2022. case 's':
  2023. switch (Peek(1))
  2024. {
  2025. case 'c':
  2026. _position += 2;
  2027. BaseNode to = ParseType();
  2028. if (to == null)
  2029. {
  2030. return null;
  2031. }
  2032. BaseNode from = ParseExpression();
  2033. if (from == null)
  2034. {
  2035. return null;
  2036. }
  2037. return new CastExpression("static_cast", to, from);
  2038. case 'p':
  2039. _position += 2;
  2040. expression = ParseExpression();
  2041. if (expression == null)
  2042. {
  2043. return null;
  2044. }
  2045. return new PackedTemplateParameterExpansion(expression);
  2046. case 'r':
  2047. return ParseUnresolvedName();
  2048. case 't':
  2049. _position += 2;
  2050. BaseNode enclosedType = ParseType();
  2051. if (enclosedType == null)
  2052. {
  2053. return null;
  2054. }
  2055. return new EnclosedExpression("sizeof (", enclosedType, ")");
  2056. case 'z':
  2057. _position += 2;
  2058. expression = ParseExpression();
  2059. if (expression == null)
  2060. {
  2061. return null;
  2062. }
  2063. return new EnclosedExpression("sizeof (", expression, ")");
  2064. case 'Z':
  2065. _position += 2;
  2066. BaseNode sizeofParamNode = null;
  2067. switch (Peek())
  2068. {
  2069. case 'T':
  2070. // FIXME: ??? Not entire sure if it's right
  2071. sizeofParamNode = ParseFunctionParameter();
  2072. if (sizeofParamNode == null)
  2073. {
  2074. return null;
  2075. }
  2076. return new EnclosedExpression("sizeof...(", new PackedTemplateParameterExpansion(sizeofParamNode), ")");
  2077. case 'f':
  2078. sizeofParamNode = ParseFunctionParameter();
  2079. if (sizeofParamNode == null)
  2080. {
  2081. return null;
  2082. }
  2083. return new EnclosedExpression("sizeof...(", sizeofParamNode, ")");
  2084. }
  2085. return null;
  2086. case 'P':
  2087. _position += 2;
  2088. List<BaseNode> arguments = new List<BaseNode>();
  2089. while (!ConsumeIf("E"))
  2090. {
  2091. BaseNode argument = ParseTemplateArgument();
  2092. if (argument == null)
  2093. {
  2094. return null;
  2095. }
  2096. arguments.Add(argument);
  2097. }
  2098. return new EnclosedExpression("sizeof...(", new NodeArray(arguments), ")");
  2099. }
  2100. return null;
  2101. case 't':
  2102. switch (Peek(1))
  2103. {
  2104. case 'e':
  2105. expression = ParseExpression();
  2106. if (expression == null)
  2107. {
  2108. return null;
  2109. }
  2110. return new EnclosedExpression("typeid (", expression, ")");
  2111. case 't':
  2112. BaseNode enclosedType = ParseExpression();
  2113. if (enclosedType == null)
  2114. {
  2115. return null;
  2116. }
  2117. return new EnclosedExpression("typeid (", enclosedType, ")");
  2118. case 'l':
  2119. _position += 2;
  2120. BaseNode typeNode = ParseType();
  2121. if (typeNode == null)
  2122. {
  2123. return null;
  2124. }
  2125. List<BaseNode> bracedExpressions = new List<BaseNode>();
  2126. while (!ConsumeIf("E"))
  2127. {
  2128. expression = ParseBracedExpression();
  2129. if (expression == null)
  2130. {
  2131. return null;
  2132. }
  2133. bracedExpressions.Add(expression);
  2134. }
  2135. return new InitListExpression(typeNode, bracedExpressions);
  2136. case 'r':
  2137. _position += 2;
  2138. return new NameType("throw");
  2139. case 'w':
  2140. _position += 2;
  2141. expression = ParseExpression();
  2142. if (expression == null)
  2143. {
  2144. return null;
  2145. }
  2146. return new ThrowExpression(expression);
  2147. }
  2148. return null;
  2149. }
  2150. if (char.IsDigit(Peek()))
  2151. {
  2152. return ParseUnresolvedName();
  2153. }
  2154. return null;
  2155. }
  2156. private BaseNode ParseIntegerLiteral(string literalName)
  2157. {
  2158. string number = ParseNumber(true);
  2159. if (number == null || number.Length == 0 || !ConsumeIf("E"))
  2160. {
  2161. return null;
  2162. }
  2163. return new IntegerLiteral(literalName, number);
  2164. }
  2165. // <expr-primary> ::= L <type> <value number> E # integer literal
  2166. // ::= L <type> <value float> E # floating literal (TODO)
  2167. // ::= L <string type> E # string literal
  2168. // ::= L <nullptr type> E # nullptr literal (i.e., "LDnE")
  2169. // ::= L <pointer type> 0 E # null pointer template argument
  2170. // ::= L <type> <real-part float> _ <imag-part float> E # complex floating point literal (C 2000)
  2171. // ::= L _Z <encoding> E # external name
  2172. private BaseNode ParseExpressionPrimary()
  2173. {
  2174. if (!ConsumeIf("L"))
  2175. {
  2176. return null;
  2177. }
  2178. switch (Peek())
  2179. {
  2180. case 'w':
  2181. _position++;
  2182. return ParseIntegerLiteral("wchar_t");
  2183. case 'b':
  2184. if (ConsumeIf("b0E"))
  2185. {
  2186. return new NameType("false", NodeType.BooleanExpression);
  2187. }
  2188. if (ConsumeIf("b1E"))
  2189. {
  2190. return new NameType("true", NodeType.BooleanExpression);
  2191. }
  2192. return null;
  2193. case 'c':
  2194. _position++;
  2195. return ParseIntegerLiteral("char");
  2196. case 'a':
  2197. _position++;
  2198. return ParseIntegerLiteral("signed char");
  2199. case 'h':
  2200. _position++;
  2201. return ParseIntegerLiteral("unsigned char");
  2202. case 's':
  2203. _position++;
  2204. return ParseIntegerLiteral("short");
  2205. case 't':
  2206. _position++;
  2207. return ParseIntegerLiteral("unsigned short");
  2208. case 'i':
  2209. _position++;
  2210. return ParseIntegerLiteral("");
  2211. case 'j':
  2212. _position++;
  2213. return ParseIntegerLiteral("u");
  2214. case 'l':
  2215. _position++;
  2216. return ParseIntegerLiteral("l");
  2217. case 'm':
  2218. _position++;
  2219. return ParseIntegerLiteral("ul");
  2220. case 'x':
  2221. _position++;
  2222. return ParseIntegerLiteral("ll");
  2223. case 'y':
  2224. _position++;
  2225. return ParseIntegerLiteral("ull");
  2226. case 'n':
  2227. _position++;
  2228. return ParseIntegerLiteral("__int128");
  2229. case 'o':
  2230. _position++;
  2231. return ParseIntegerLiteral("unsigned __int128");
  2232. case 'd':
  2233. case 'e':
  2234. case 'f':
  2235. // TODO: floating literal
  2236. return null;
  2237. case '_':
  2238. if (ConsumeIf("_Z"))
  2239. {
  2240. BaseNode encoding = ParseEncoding();
  2241. if (encoding != null && ConsumeIf("E"))
  2242. {
  2243. return encoding;
  2244. }
  2245. }
  2246. return null;
  2247. case 'T':
  2248. return null;
  2249. default:
  2250. BaseNode type = ParseType();
  2251. if (type == null)
  2252. {
  2253. return null;
  2254. }
  2255. string number = ParseNumber();
  2256. if (number == null || number.Length == 0 || !ConsumeIf("E"))
  2257. {
  2258. return null;
  2259. }
  2260. return new IntegerCastExpression(type, number);
  2261. }
  2262. }
  2263. // <decltype> ::= Dt <expression> E # decltype of an id-expression or class member access (C++0x)
  2264. // ::= DT <expression> E # decltype of an expression (C++0x)
  2265. private BaseNode ParseDecltype()
  2266. {
  2267. if (!ConsumeIf("D") || (!ConsumeIf("t") && !ConsumeIf("T")))
  2268. {
  2269. return null;
  2270. }
  2271. BaseNode expression = ParseExpression();
  2272. if (expression == null)
  2273. {
  2274. return null;
  2275. }
  2276. if (!ConsumeIf("E"))
  2277. {
  2278. return null;
  2279. }
  2280. return new EnclosedExpression("decltype(", expression, ")");
  2281. }
  2282. // <template-param> ::= T_ # first template parameter
  2283. // ::= T <parameter-2 non-negative number> _
  2284. // <template-template-param> ::= <template-param>
  2285. // ::= <substitution>
  2286. private BaseNode ParseTemplateParam()
  2287. {
  2288. if (!ConsumeIf("T"))
  2289. {
  2290. return null;
  2291. }
  2292. int index = 0;
  2293. if (!ConsumeIf("_"))
  2294. {
  2295. index = ParsePositiveNumber();
  2296. if (index < 0)
  2297. {
  2298. return null;
  2299. }
  2300. index++;
  2301. if (!ConsumeIf("_"))
  2302. {
  2303. return null;
  2304. }
  2305. }
  2306. // 5.1.8: TODO: lambda?
  2307. // if (IsParsingLambdaParameters)
  2308. // return new NameType("auto");
  2309. if (_canForwardTemplateReference)
  2310. {
  2311. ForwardTemplateReference forwardTemplateReference = new ForwardTemplateReference(index);
  2312. _forwardTemplateReferenceList.Add(forwardTemplateReference);
  2313. return forwardTemplateReference;
  2314. }
  2315. if (index >= _templateParamList.Count)
  2316. {
  2317. return null;
  2318. }
  2319. return _templateParamList[index];
  2320. }
  2321. // <template-args> ::= I <template-arg>+ E
  2322. private BaseNode ParseTemplateArguments(bool hasContext = false)
  2323. {
  2324. if (!ConsumeIf("I"))
  2325. {
  2326. return null;
  2327. }
  2328. if (hasContext)
  2329. {
  2330. _templateParamList.Clear();
  2331. }
  2332. List<BaseNode> args = new List<BaseNode>();
  2333. while (!ConsumeIf("E"))
  2334. {
  2335. if (hasContext)
  2336. {
  2337. List<BaseNode> templateParamListTemp = new List<BaseNode>(_templateParamList);
  2338. BaseNode templateArgument = ParseTemplateArgument();
  2339. _templateParamList = templateParamListTemp;
  2340. if (templateArgument == null)
  2341. {
  2342. return null;
  2343. }
  2344. args.Add(templateArgument);
  2345. if (templateArgument.GetType().Equals(NodeType.PackedTemplateArgument))
  2346. {
  2347. templateArgument = new PackedTemplateParameter(((NodeArray)templateArgument).Nodes);
  2348. }
  2349. _templateParamList.Add(templateArgument);
  2350. }
  2351. else
  2352. {
  2353. BaseNode templateArgument = ParseTemplateArgument();
  2354. if (templateArgument == null)
  2355. {
  2356. return null;
  2357. }
  2358. args.Add(templateArgument);
  2359. }
  2360. }
  2361. return new TemplateArguments(args);
  2362. }
  2363. // <template-arg> ::= <type> # type or template
  2364. // ::= X <expression> E # expression
  2365. // ::= <expr-primary> # simple expressions
  2366. // ::= J <template-arg>* E # argument pack
  2367. private BaseNode ParseTemplateArgument()
  2368. {
  2369. switch (Peek())
  2370. {
  2371. // X <expression> E
  2372. case 'X':
  2373. _position++;
  2374. BaseNode expression = ParseExpression();
  2375. if (expression == null || !ConsumeIf("E"))
  2376. {
  2377. return null;
  2378. }
  2379. return expression;
  2380. // <expr-primary>
  2381. case 'L':
  2382. return ParseExpressionPrimary();
  2383. // J <template-arg>* E
  2384. case 'J':
  2385. _position++;
  2386. List<BaseNode> templateArguments = new List<BaseNode>();
  2387. while (!ConsumeIf("E"))
  2388. {
  2389. BaseNode templateArgument = ParseTemplateArgument();
  2390. if (templateArgument == null)
  2391. {
  2392. return null;
  2393. }
  2394. templateArguments.Add(templateArgument);
  2395. }
  2396. return new NodeArray(templateArguments, NodeType.PackedTemplateArgument);
  2397. // <type>
  2398. default:
  2399. return ParseType();
  2400. }
  2401. }
  2402. class NameParserContext
  2403. {
  2404. public CvType Cv;
  2405. public SimpleReferenceType Ref;
  2406. public bool FinishWithTemplateArguments;
  2407. public bool CtorDtorConversion;
  2408. }
  2409. // <unresolved-type> ::= <template-param> [ <template-args> ] # T:: or T<X,Y>::
  2410. // ::= <decltype> # decltype(p)::
  2411. // ::= <substitution>
  2412. private BaseNode ParseUnresolvedType()
  2413. {
  2414. if (Peek() == 'T')
  2415. {
  2416. BaseNode templateParam = ParseTemplateParam();
  2417. if (templateParam == null)
  2418. {
  2419. return null;
  2420. }
  2421. _substitutionList.Add(templateParam);
  2422. return templateParam;
  2423. }
  2424. else if (Peek() == 'D')
  2425. {
  2426. BaseNode declType = ParseDecltype();
  2427. if (declType == null)
  2428. {
  2429. return null;
  2430. }
  2431. _substitutionList.Add(declType);
  2432. return declType;
  2433. }
  2434. return ParseSubstitution();
  2435. }
  2436. // <simple-id> ::= <source-name> [ <template-args> ]
  2437. private BaseNode ParseSimpleId()
  2438. {
  2439. BaseNode sourceName = ParseSourceName();
  2440. if (sourceName == null)
  2441. {
  2442. return null;
  2443. }
  2444. if (Peek() == 'I')
  2445. {
  2446. BaseNode templateArguments = ParseTemplateArguments();
  2447. if (templateArguments == null)
  2448. {
  2449. return null;
  2450. }
  2451. return new NameTypeWithTemplateArguments(sourceName, templateArguments);
  2452. }
  2453. return sourceName;
  2454. }
  2455. // <destructor-name> ::= <unresolved-type> # e.g., ~T or ~decltype(f())
  2456. // ::= <simple-id> # e.g., ~A<2*N>
  2457. private BaseNode ParseDestructorName()
  2458. {
  2459. BaseNode node;
  2460. if (char.IsDigit(Peek()))
  2461. {
  2462. node = ParseSimpleId();
  2463. }
  2464. else
  2465. {
  2466. node = ParseUnresolvedType();
  2467. }
  2468. if (node == null)
  2469. {
  2470. return null;
  2471. }
  2472. return new DtorName(node);
  2473. }
  2474. // <base-unresolved-name> ::= <simple-id> # unresolved name
  2475. // extension ::= <operator-name> # unresolved operator-function-id
  2476. // extension ::= <operator-name> <template-args> # unresolved operator template-id
  2477. // ::= on <operator-name> # unresolved operator-function-id
  2478. // ::= on <operator-name> <template-args> # unresolved operator template-id
  2479. // ::= dn <destructor-name> # destructor or pseudo-destructor;
  2480. // # e.g. ~X or ~X<N-1>
  2481. private BaseNode ParseBaseUnresolvedName()
  2482. {
  2483. if (char.IsDigit(Peek()))
  2484. {
  2485. return ParseSimpleId();
  2486. }
  2487. else if (ConsumeIf("dn"))
  2488. {
  2489. return ParseDestructorName();
  2490. }
  2491. ConsumeIf("on");
  2492. BaseNode operatorName = ParseOperatorName(null);
  2493. if (operatorName == null)
  2494. {
  2495. return null;
  2496. }
  2497. if (Peek() == 'I')
  2498. {
  2499. BaseNode templateArguments = ParseTemplateArguments();
  2500. if (templateArguments == null)
  2501. {
  2502. return null;
  2503. }
  2504. return new NameTypeWithTemplateArguments(operatorName, templateArguments);
  2505. }
  2506. return operatorName;
  2507. }
  2508. // <unresolved-name> ::= [gs] <base-unresolved-name> # x or (with "gs") ::x
  2509. // ::= sr <unresolved-type> <base-unresolved-name> # T::x / decltype(p)::x
  2510. // ::= srN <unresolved-type> <unresolved-qualifier-level>+ E <base-unresolved-name>
  2511. // # T::N::x /decltype(p)::N::x
  2512. // ::= [gs] sr <unresolved-qualifier-level>+ E <base-unresolved-name>
  2513. // # A::x, N::y, A<T>::z; "gs" means leading "::"
  2514. private BaseNode ParseUnresolvedName(NameParserContext context = null)
  2515. {
  2516. BaseNode result = null;
  2517. if (ConsumeIf("srN"))
  2518. {
  2519. result = ParseUnresolvedType();
  2520. if (result == null)
  2521. {
  2522. return null;
  2523. }
  2524. if (Peek() == 'I')
  2525. {
  2526. BaseNode templateArguments = ParseTemplateArguments();
  2527. if (templateArguments == null)
  2528. {
  2529. return null;
  2530. }
  2531. result = new NameTypeWithTemplateArguments(result, templateArguments);
  2532. if (result == null)
  2533. {
  2534. return null;
  2535. }
  2536. }
  2537. while (!ConsumeIf("E"))
  2538. {
  2539. BaseNode simpleId = ParseSimpleId();
  2540. if (simpleId == null)
  2541. {
  2542. return null;
  2543. }
  2544. result = new QualifiedName(result, simpleId);
  2545. if (result == null)
  2546. {
  2547. return null;
  2548. }
  2549. }
  2550. BaseNode baseName = ParseBaseUnresolvedName();
  2551. if (baseName == null)
  2552. {
  2553. return null;
  2554. }
  2555. return new QualifiedName(result, baseName);
  2556. }
  2557. bool isGlobal = ConsumeIf("gs");
  2558. // ::= [gs] <base-unresolved-name> # x or (with "gs") ::x
  2559. if (!ConsumeIf("sr"))
  2560. {
  2561. result = ParseBaseUnresolvedName();
  2562. if (result == null)
  2563. {
  2564. return null;
  2565. }
  2566. if (isGlobal)
  2567. {
  2568. result = new GlobalQualifiedName(result);
  2569. }
  2570. return result;
  2571. }
  2572. // ::= [gs] sr <unresolved-qualifier-level>+ E <base-unresolved-name>
  2573. if (char.IsDigit(Peek()))
  2574. {
  2575. do
  2576. {
  2577. BaseNode qualifier = ParseSimpleId();
  2578. if (qualifier == null)
  2579. {
  2580. return null;
  2581. }
  2582. if (result != null)
  2583. {
  2584. result = new QualifiedName(result, qualifier);
  2585. }
  2586. else if (isGlobal)
  2587. {
  2588. result = new GlobalQualifiedName(qualifier);
  2589. }
  2590. else
  2591. {
  2592. result = qualifier;
  2593. }
  2594. if (result == null)
  2595. {
  2596. return null;
  2597. }
  2598. } while (!ConsumeIf("E"));
  2599. }
  2600. // ::= sr <unresolved-type> [tempate-args] <base-unresolved-name> # T::x / decltype(p)::x
  2601. else
  2602. {
  2603. result = ParseUnresolvedType();
  2604. if (result == null)
  2605. {
  2606. return null;
  2607. }
  2608. if (Peek() == 'I')
  2609. {
  2610. BaseNode templateArguments = ParseTemplateArguments();
  2611. if (templateArguments == null)
  2612. {
  2613. return null;
  2614. }
  2615. result = new NameTypeWithTemplateArguments(result, templateArguments);
  2616. if (result == null)
  2617. {
  2618. return null;
  2619. }
  2620. }
  2621. }
  2622. if (result == null)
  2623. {
  2624. return null;
  2625. }
  2626. BaseNode baseUnresolvedName = ParseBaseUnresolvedName();
  2627. if (baseUnresolvedName == null)
  2628. {
  2629. return null;
  2630. }
  2631. return new QualifiedName(result, baseUnresolvedName);
  2632. }
  2633. // <unscoped-name> ::= <unqualified-name>
  2634. // ::= St <unqualified-name> # ::std::
  2635. private BaseNode ParseUnscopedName(NameParserContext context)
  2636. {
  2637. if (ConsumeIf("St"))
  2638. {
  2639. BaseNode unresolvedName = ParseUnresolvedName(context);
  2640. if (unresolvedName == null)
  2641. {
  2642. return null;
  2643. }
  2644. return new StdQualifiedName(unresolvedName);
  2645. }
  2646. return ParseUnresolvedName(context);
  2647. }
  2648. // <nested-name> ::= N [<CV-qualifiers>] [<ref-qualifier>] <prefix (TODO)> <unqualified-name> E
  2649. // ::= N [<CV-qualifiers>] [<ref-qualifier>] <template-prefix (TODO)> <template-args (TODO)> E
  2650. private BaseNode ParseNestedName(NameParserContext context)
  2651. {
  2652. // Impossible in theory
  2653. if (Consume() != 'N')
  2654. {
  2655. return null;
  2656. }
  2657. BaseNode result = null;
  2658. CvType cv = new CvType(ParseCvQualifiers(), null);
  2659. if (context != null)
  2660. {
  2661. context.Cv = cv;
  2662. }
  2663. SimpleReferenceType Ref = ParseRefQualifiers();
  2664. if (context != null)
  2665. {
  2666. context.Ref = Ref;
  2667. }
  2668. if (ConsumeIf("St"))
  2669. {
  2670. result = new NameType("std");
  2671. }
  2672. while (!ConsumeIf("E"))
  2673. {
  2674. // <data-member-prefix> end
  2675. if (ConsumeIf("M"))
  2676. {
  2677. if (result == null)
  2678. {
  2679. return null;
  2680. }
  2681. continue;
  2682. }
  2683. char c = Peek();
  2684. // TODO: template args
  2685. if (c == 'T')
  2686. {
  2687. BaseNode templateParam = ParseTemplateParam();
  2688. if (templateParam == null)
  2689. {
  2690. return null;
  2691. }
  2692. result = CreateNameNode(result, templateParam, context);
  2693. _substitutionList.Add(result);
  2694. continue;
  2695. }
  2696. // <template-prefix> <template-args>
  2697. if (c == 'I')
  2698. {
  2699. BaseNode templateArgument = ParseTemplateArguments(context != null);
  2700. if (templateArgument == null || result == null)
  2701. {
  2702. return null;
  2703. }
  2704. result = new NameTypeWithTemplateArguments(result, templateArgument);
  2705. if (context != null)
  2706. {
  2707. context.FinishWithTemplateArguments = true;
  2708. }
  2709. _substitutionList.Add(result);
  2710. continue;
  2711. }
  2712. // <decltype>
  2713. if (c == 'D' && (Peek(1) == 't' || Peek(1) == 'T'))
  2714. {
  2715. BaseNode decltype = ParseDecltype();
  2716. if (decltype == null)
  2717. {
  2718. return null;
  2719. }
  2720. result = CreateNameNode(result, decltype, context);
  2721. _substitutionList.Add(result);
  2722. continue;
  2723. }
  2724. // <substitution>
  2725. if (c == 'S' && Peek(1) != 't')
  2726. {
  2727. BaseNode substitution = ParseSubstitution();
  2728. if (substitution == null)
  2729. {
  2730. return null;
  2731. }
  2732. result = CreateNameNode(result, substitution, context);
  2733. if (result != substitution)
  2734. {
  2735. _substitutionList.Add(substitution);
  2736. }
  2737. continue;
  2738. }
  2739. // <ctor-dtor-name> of ParseUnqualifiedName
  2740. if (c == 'C' || (c == 'D' && Peek(1) != 'C'))
  2741. {
  2742. // We cannot have nothing before this
  2743. if (result == null)
  2744. {
  2745. return null;
  2746. }
  2747. BaseNode ctOrDtorName = ParseCtorDtorName(context, result);
  2748. if (ctOrDtorName == null)
  2749. {
  2750. return null;
  2751. }
  2752. result = CreateNameNode(result, ctOrDtorName, context);
  2753. // TODO: ABI Tags (before)
  2754. if (result == null)
  2755. {
  2756. return null;
  2757. }
  2758. _substitutionList.Add(result);
  2759. continue;
  2760. }
  2761. BaseNode unqualifiedName = ParseUnqualifiedName(context);
  2762. if (unqualifiedName == null)
  2763. {
  2764. return null;
  2765. }
  2766. result = CreateNameNode(result, unqualifiedName, context);
  2767. _substitutionList.Add(result);
  2768. }
  2769. if (result == null || _substitutionList.Count == 0)
  2770. {
  2771. return null;
  2772. }
  2773. _substitutionList.RemoveAt(_substitutionList.Count - 1);
  2774. return result;
  2775. }
  2776. // <discriminator> ::= _ <non-negative number> # when number < 10
  2777. // ::= __ <non-negative number> _ # when number >= 10
  2778. private void ParseDiscriminator()
  2779. {
  2780. if (Count() == 0)
  2781. {
  2782. return;
  2783. }
  2784. // We ignore the discriminator, we don't need it.
  2785. if (ConsumeIf("_"))
  2786. {
  2787. ConsumeIf("_");
  2788. while (char.IsDigit(Peek()) && Count() != 0)
  2789. {
  2790. Consume();
  2791. }
  2792. ConsumeIf("_");
  2793. }
  2794. }
  2795. // <local-name> ::= Z <function encoding> E <entity name> [<discriminator>]
  2796. // ::= Z <function encoding> E s [<discriminator>]
  2797. // ::= Z <function encoding> Ed [ <parameter number> ] _ <entity name>
  2798. private BaseNode ParseLocalName(NameParserContext context)
  2799. {
  2800. if (!ConsumeIf("Z"))
  2801. {
  2802. return null;
  2803. }
  2804. BaseNode encoding = ParseEncoding();
  2805. if (encoding == null || !ConsumeIf("E"))
  2806. {
  2807. return null;
  2808. }
  2809. BaseNode entityName;
  2810. if (ConsumeIf("s"))
  2811. {
  2812. ParseDiscriminator();
  2813. return new LocalName(encoding, new NameType("string literal"));
  2814. }
  2815. else if (ConsumeIf("d"))
  2816. {
  2817. ParseNumber(true);
  2818. if (!ConsumeIf("_"))
  2819. {
  2820. return null;
  2821. }
  2822. entityName = ParseName(context);
  2823. if (entityName == null)
  2824. {
  2825. return null;
  2826. }
  2827. return new LocalName(encoding, entityName);
  2828. }
  2829. entityName = ParseName(context);
  2830. if (entityName == null)
  2831. {
  2832. return null;
  2833. }
  2834. ParseDiscriminator();
  2835. return new LocalName(encoding, entityName);
  2836. }
  2837. // <name> ::= <nested-name>
  2838. // ::= <unscoped-name>
  2839. // ::= <unscoped-template-name> <template-args>
  2840. // ::= <local-name> # See Scope Encoding below (TODO)
  2841. private BaseNode ParseName(NameParserContext context = null)
  2842. {
  2843. ConsumeIf("L");
  2844. if (Peek() == 'N')
  2845. {
  2846. return ParseNestedName(context);
  2847. }
  2848. if (Peek() == 'Z')
  2849. {
  2850. return ParseLocalName(context);
  2851. }
  2852. if (Peek() == 'S' && Peek(1) != 't')
  2853. {
  2854. BaseNode substitution = ParseSubstitution();
  2855. if (substitution == null)
  2856. {
  2857. return null;
  2858. }
  2859. if (Peek() != 'I')
  2860. {
  2861. return null;
  2862. }
  2863. BaseNode templateArguments = ParseTemplateArguments(context != null);
  2864. if (templateArguments == null)
  2865. {
  2866. return null;
  2867. }
  2868. if (context != null)
  2869. {
  2870. context.FinishWithTemplateArguments = true;
  2871. }
  2872. return new NameTypeWithTemplateArguments(substitution, templateArguments);
  2873. }
  2874. BaseNode result = ParseUnscopedName(context);
  2875. if (result == null)
  2876. {
  2877. return null;
  2878. }
  2879. if (Peek() == 'I')
  2880. {
  2881. _substitutionList.Add(result);
  2882. BaseNode templateArguments = ParseTemplateArguments(context != null);
  2883. if (templateArguments == null)
  2884. {
  2885. return null;
  2886. }
  2887. if (context != null)
  2888. {
  2889. context.FinishWithTemplateArguments = true;
  2890. }
  2891. return new NameTypeWithTemplateArguments(result, templateArguments);
  2892. }
  2893. return result;
  2894. }
  2895. private bool IsEncodingEnd()
  2896. {
  2897. char c = Peek();
  2898. return Count() == 0 || c == 'E' || c == '.' || c == '_';
  2899. }
  2900. // <encoding> ::= <function name> <bare-function-type>
  2901. // ::= <data name>
  2902. // ::= <special-name>
  2903. private BaseNode ParseEncoding()
  2904. {
  2905. NameParserContext context = new NameParserContext();
  2906. if (Peek() == 'T' || (Peek() == 'G' && Peek(1) == 'V'))
  2907. {
  2908. return ParseSpecialName(context);
  2909. }
  2910. BaseNode name = ParseName(context);
  2911. if (name == null)
  2912. {
  2913. return null;
  2914. }
  2915. // TODO: compute template refs here
  2916. if (IsEncodingEnd())
  2917. {
  2918. return name;
  2919. }
  2920. // TODO: Ua9enable_ifI
  2921. BaseNode returnType = null;
  2922. if (!context.CtorDtorConversion && context.FinishWithTemplateArguments)
  2923. {
  2924. returnType = ParseType();
  2925. if (returnType == null)
  2926. {
  2927. return null;
  2928. }
  2929. }
  2930. if (ConsumeIf("v"))
  2931. {
  2932. return new EncodedFunction(name, null, context.Cv, context.Ref, null, returnType);
  2933. }
  2934. List<BaseNode> Params = new List<BaseNode>();
  2935. // backup because that can be destroyed by parseType
  2936. CvType cv = context.Cv;
  2937. SimpleReferenceType Ref = context.Ref;
  2938. while (!IsEncodingEnd())
  2939. {
  2940. BaseNode param = ParseType();
  2941. if (param == null)
  2942. {
  2943. return null;
  2944. }
  2945. Params.Add(param);
  2946. }
  2947. return new EncodedFunction(name, new NodeArray(Params), cv, Ref, null, returnType);
  2948. }
  2949. // <mangled-name> ::= _Z <encoding>
  2950. // ::= <type>
  2951. private BaseNode Parse()
  2952. {
  2953. if (ConsumeIf("_Z"))
  2954. {
  2955. BaseNode encoding = ParseEncoding();
  2956. if (encoding != null && Count() == 0)
  2957. {
  2958. return encoding;
  2959. }
  2960. return null;
  2961. }
  2962. else
  2963. {
  2964. BaseNode type = ParseType();
  2965. if (type != null && Count() == 0)
  2966. {
  2967. return type;
  2968. }
  2969. return null;
  2970. }
  2971. }
  2972. public static string Parse(string originalMangled)
  2973. {
  2974. Demangler instance = new Demangler(originalMangled);
  2975. BaseNode resNode = instance.Parse();
  2976. if (resNode != null)
  2977. {
  2978. StringWriter writer = new StringWriter();
  2979. resNode.Print(writer);
  2980. return writer.ToString();
  2981. }
  2982. return originalMangled;
  2983. }
  2984. }
  2985. }