Demangler.cs 117 KB

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