% Copyright 2012-2024, Alexander Shibakov % This file is part of SPLinT % % SPLinT is free software: you can redistribute it and/or modify % it under the terms of the GNU General Public License as published by % the Free Software Foundation, either version 3 of the License, or % (at your option) any later version. % % SPLinT is distributed in the hope that it will be useful, % but WITHOUT ANY WARRANTY; without even the implied warranty of % MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the % GNU General Public License for more details. % % You should have received a copy of the GNU General Public License % along with SPLinT. If not, see <http://www.gnu.org/licenses/>. % prototypes of all the macros produced by the ld parser % we will follow the convention: 12string is a string of category 12 characters and spaces, tex_string: a string % of TeX tokens; sptr is a pointer to the stash stream, fptr is a pointer to the format stream \def\yyuniontag{\ldunion} \def\parserstrictnamespace{ld-parser-strict} % expands to an error message \def\parserprototypesnamespace{ld-parser-strict:headers} % the parameter strings \def\parserdebugnamespace{ld-parser-debug} % control sequences are \let...=\relax for debugging \def\ldunion{\currentyyunionnamespace} \def\currentyyunionnamespace{ld-generic} % types returned by the lexer (* marks the types that get removed by the parser in some cases) \defp\anint#1#2#3{} % integer :: \anint{integer}{fptr}{sptr} \defp\bint#1#2#3{} % integer in a specifix radix :: \bint{integer}{fptr}{sptr} \defp\hexint#1#2#3{} % hex integer :: \hexint{integer}{fptr}{sptr} % ld commands \defp\ldor{} % item separator :: \ldor, is only needed here for debugging (in the ld-parser-debug namespace) \defp\ldregexp#1{} % name pattern :: \ldregexp{{name}{name11}{fptr}{sptr}} \defp\ldregop#1{} % name pattern :: \ldregop{name} \defp\ldspace{} % space separator :: \ldspace \defp\ldattrstring#1#2#3#4{} % attributes string :: \ldattrstring{name}{name11}{fptr}{sptr} \defp\ldattrstringneg#1#2#3#4{} % complemented attributes string :: \ldattrstring{name}{name11}{fptr}{sptr} \defp\ldfilename#1{} % file name :: \ldfilename{name} \defp\ldcommandseparator#1#2#3#4{} % command separator :: \ldcommandseparator{fptr}{sptr}{prev command}{next command} \defp\ldassignment#1#2#3{} % assignment :: \ldassignment{lhs}{op}{rhs} \defp\ldhidden#1#2{} % hidden :: \ldhidden{lhs}{rhs} \defp\ldprovide#1#2{} % provide :: \ldprovide{lhs}{rhs} \defp\ldprovidehid#1#2{} % provide hidden :: \ldprovidehid{lhs}{rhs} \defp\ldkeep#1{} % keep :: \ldkeep{list} \defp\ldentry#1{} % entry :: \ldentry{name} \defp\ldinclude#1{} % file inclusion :: \ldinclude{file name} \defp\ldmemory#1{} % memory specification :: \ldmemory{memory spcification} \defp\ldfill#1{} % fill expression :: \ldfill{expression} \defp\ldmemoryspec#1#2#3#4{} % memory item :: \ldmemoryspec{name}{attributes}{origin}{length} \defp\ldmemspecstash#1#2{} % memory spec stash :: \ldmemspecstash{fptr}{sptr} \defp\ldmemspecseparator#1#2{} % memory spec separator :: \ldmemspecseparator{fptr}{sptr} \defp\ldoriginspec#1{} % origin :: \ldoriginspec{expression} \defp\ldlengthspec#1{} % length :: \ldlengthspec{expression} \defp\ldsections#1{} % ld sections :: \ldsections{sections} \defp\ldsectionseparator#1#2{} % section separator :: \ldsectionseparator{fptr}{sptr} \defp\ldtype#1{} % section type :: \ldtype{type} \defp\ldstatement#1{} % statement :: \ldstatement{statement} \defp\ldfreestmt#1{} % statement in a section :: \ldfreestmt{\ldstatement{statement}} \defp\ldsecspec#1{} % section spec :: \ldsecspec{section spec} \defp\ldinsertcweb#1#2#3#4{} % insert accumulated \CWEB\ material :: \ldinsertcweb % {fptr}{sptr}{command}{parsed segment} : never defined \defp\ldnamedsection#1#2#3#4#5#6#7{} % named section :: \ldnamedsection{name} % {expression}{type}}{at} % {{}{}{}:alignment} % {constraint}{statements} % {{}{}{}{}:placement} \defp\ldoverlay#1#2#3#4#5#6{} % overlay section list :: \ldoverlay{expression}{crossrefs}{at}{subalign} % {sections} % {{memspec}{memspec_at}{phdr}{fill}} \defp\ldoverlaysection#1#2#3#4{} % overlay section :: \ldoverlaysection{name}{statements}{phdr}{fill} \defp\ldsectionstash#1#2{} % sections spec stash :: \ldsectionstash{fptr}{sptr} \savecslist{ld-parser-prototypes}\ldunion % these consume parameters and expand to nothing \input ldint.sty \def\ldidxdomain{L} \let\writeldidxentry\writeidxentry % mix the \ld\ index with the \let\writeldidxhentry\writeidxhentry % \bison\ index for now \def\writeldidxhentry#1{% \indxe\gindex{{\secno}{{}{\hostparsernamespace}}{\ldidxdomain}{\headeridxrank}#1}% } \def\writeldidxentry#1{% \indxe\gindex{{\secno}{{}{\hostparsernamespace}}{\ldidxdomain}{\termidxrank}#1}% } \def\bigbracedel{\delimiter"4266308} \newtoks\lddisplay \restorecslist{ld-parser-strict}\ldunion \defc\ldinclude{% \toksa={&##\cr\ttl include\ &}#1% \concat\toksa\toksc \toksc{\cr} \concat\toksa\toksc \edef\next{\lddisplay{\the\lddisplay\halign{\the\toksa}}}\next } \defc\ldmemory{% \savecslist{local-namespace}{\ldunion}% \restorecslist{ld-parser:memory-spec}\ldunion \toksa{}#1% \toksc{% \hfil##\qquad&##\hfil&\qquad##\hfil\quad&\hfil##&\qquad\hfil##\cr \ttl memory&\hfil&\ttl attributes&\ttl starts at&\ttl length\cr \noalign{\smallskip}% }% \edef\next{\lddisplay{\the\lddisplay\halign{\the\toksc\the\toksa}}}\next \restorecslist{local-namespace}{\ldunion}% } \defc\ldsections{% \savecslist{local-namespace}{\ldunion}% \restorecslist{ld-parser:sections}\ldunion \toksa{}\def\sections@header{\ttl sections}% #1% \toksc{% &##\hfil\quad\cr }% \edef\next{\lddisplay{\the\lddisplay\halign{\the\toksc\the\toksa}}}\next \restorecslist{local-namespace}{\ldunion}% } \def\ldextractname#1#2#3#4{% \edef\next{\toksc{\gidxentry{\termttstring}{#2}{}{\ntt #2}}}\next } \def\ldextractattrs#1#2#3#4{% \edef\next{\toksc{{\ntt #2}}}\next } \def\ldextractmemname#1#2#3#4{% {% \expandafter\let\expandafter\tosmallparser\csname to\stripbrackets\cwebclinknamespace parser\endcsname \let\optstrextra\optstrextraesc \def\hostparsernamespace{[none]}% \nameproc{#2}\with\parsebin \edef\next{\toksc{\gidxentry{\termvstring}{#2}{}{\let\idxfont\nx\empty\ntt\the\toksa}}}% \expandafter }\next% } % the grammar of ld scripts is very uniform so the separator form % below should be more than adequate; if a more sophisticated spacing % strategy is required, one may consult the design of % \separatorswitcheq and \separatorswitchneq in yyunion.sty \defc\ldcommandseparator{% \expandafter\ifx\csname ldstashentry[#2]\endcsname\relax \yyifsamestring{#3}{#4}{}{\appendrnx\lddisplay{\medskip}}% \else \expandafter\expandafter\expandafter\yystashlocal\expandafter\expandafter\expandafter{\csname ldstashentry[#2]\endcsname}% \edef\next{\toksc{\toksa{}\the\yystashlocal \noexpand\ldmakestashbox{}}}\next \appendrnx\lddisplay{\smallskip\noindent}% \concat\lddisplay\toksc \appendrnx\lddisplay{\smallskip}% \fi } \defc\ldcommandseparator{% new version; TODO: remove the duplicates after the macros have been tested \expandafter\ifx\csname ldstashentry[#2]\endcsname\relax \yyifsamestring{#3}{#4}{}{\appendrnx\lddisplay{\medskip}}% \else \expandafter\expandafter\expandafter\yystashlocal\expandafter\expandafter\expandafter{\csname ldstashentry[#2]\endcsname}% \appendrnx\lddisplay{\smallskip\noindent}% \appendr \lddisplay{\toksa{\the\yystashlocal}}% \appendrnx\lddisplay{\ldmakestashbox{}\smallskip}% \fi } \defc\ldstatement{\toksc{\hbox{$#1$}}\concat\lddisplay\toksc} \let\ldsecspec\ldstatement \defc\ldassignment{% #1#2#3% } \defc\ldhidden{% \mathop{\hbox{\ssf hidden}}\hbox{$\langle\,$}#1\K#2\hbox{$\,\rangle$}% } \defc\ldprovide{% \mathop{\hbox{\ssf provide}}\hbox{$\langle\,$}#1\K#2\hbox{$\,\rangle$}% } \defc\ldprovidehid{% \mathop{\hbox{\ssf provide$_{h}$}}\hbox{$\langle\,$}#1\K#2\hbox{$\,\rangle$}% } \defc\anint{% \uppercase{\ldsciinteger{#1}}% } \defc\hexint{% \ldsciinteger{#1}% } \defc\bint{% \uppercase{\ldbasedinteger{#1}}% } \defc\ldregexp{% \ldreg@xp#1% } \def\ldreg@xp#1#2#3#4{% \expandafter\ifx\csname ldvarname[#2]\endcsname\relax {% \let\termindex\writeldidxentry \let\hostparsernamespace\cwebclinknamespace% process the variable names as in \CWEB \edef\next{\toksc{\gidxentry{\termttstring}{#1}{}}}\next \hbox{\ntt@#1}% \expandafter }\the\toksc \else \yyifsamestring{#2}{.}{% special . name {% \let\termindex\writeldidxentry \let\hostparsernamespace\cwebclinknamespace% for consistency \edef\next{\toksc{\gidxentry{\termexception}{.origin&}{.}}}\next \hbox{\csname\prettynamecs\hostparsernamespace{.origin&}\endcsname{}}% \expandafter }\the\toksc }{% {% \let\termindex\writeldidxentry \let\hostparsernamespace\cwebclinknamespace% process the variable names as in \CWEB \edef\next{\toksc{\gidxentry{\termhostidstring}{#1}{}}}\next \hbox{% \expandafter\let\expandafter\tosmallparser\csname to\stripbrackets\cwebclinknamespace parser\endcsname \let\optstrextra\optstrextraesc \nameproc{#2}\with\parsebin \\{\the\toksa}% }% \expandafter }\the\toksc }% \fi } \defc\ldregop{% \ldreg@p#1% } \def\ldreg@p#1#2#3#4{% \hbox{\ntt@#1}% } \defc\ldfill{% #1% } \defc\ldentry{% this command survives till the table time \hbox{\ttl entry{\rm: }} #1% } \defc\ldkeep{% \mathop{\hbox{\ssf keep}}(#1)% } \defc\ldfilename{\ldextractname#1} \savecslist{ld-display}\ldunion % memory specifications \restorecslist{ld-parser-strict}\ldunion \defc\ldmemoryspec{% \toksb{\hfil&}% \let\termindex\writeldidxhentry \ldextractmemname#1\concat\toksb\toksc \let\termindex\eatone \appendrnx\toksb{&}% \toksc{}#2\appendrnx\toksc{&}\concat\toksb\toksc% attributes \toksc{}#3\appendrnx\toksc{&}\concat\toksb\toksc \toksc{}#4\concat\toksb\toksc \concat\toksa\toksb \toksb{\cr}\concat\toksa\toksb } \defc\ldattrstring{% \appendrnx\toksc{{\ntt@ #2}}% } \defc\ldattrstringneg{% \appendrnx\toksc{\hbox{$\neg$}{\ntt@ #2}}% } \defc\ldspace{% } \defc\ldlengthspec{% \toksc{$#1$}% } \defc\ldoriginspec{% \toksc{#1}% } \defc\ldinclude{% \toksc={\hfil&\ttl include }% \concat\toksa\toksc#1% \concat\toksa\toksc \toksc{&\hfil&\hfil&\hfil\cr}% \concat\toksa\toksc } % TODO: change the code to use linked lists \defc\ldmemspecseparator{% \expandafter\ifx\csname ldstashentry[#2]\endcsname\relax \else \expandafter\expandafter\expandafter\yystashlocal\expandafter\expandafter\expandafter{\csname ldstashentry[#2]\endcsname}% \appendr\toksa{&\nx\multispan4\toksa{\the\yystashlocal}\nx\ldmakestashbox{\nx\cdotfill}\nx\quad\cr}% \fi } \defc\ldmemspecstash{% \expandafter\ifx\csname ldstashentry[#2]\endcsname\relax \else \expandafter\expandafter\expandafter\yystashlocal\expandafter\expandafter\expandafter{\csname ldstashentry[#2]\endcsname}% \appendr\toksa{&\nx\multispan4\toksa{\the\yystashlocal}\nx\ldmakestashbox{\nx\cdotfill}\nx\quad\cr}% \fi } \restorecs{ld-display}{\ldfilename\ldentry} \toyyunion{ld-parser:memory-spec} % sections commands \restorecslist{ld-parser-strict}\ldunion \newif\ifplacementpushed \newif\ifsectioncomplete \newif\iffillextracted \defc\ldnamedsection{ % named section :: \ldnamedsection{name} % {{bind?}{expression}{block?}{expression}{type}}{at} % {{}{}{}:alignment} % {constraint}{statements} % {{}{}{}{}:placement} \tempca=\z@ % line counter \tempcb=\z@ % alignment line counter \placementpushedfalse \sectioncompletefalse \fillextractedfalse \bloop \toksb{}% \ifnum\tempca=\z@ \toksb\expandafter{\sections@header&}% section header \ldextractname#1% section name \concat\toksb\toksc \appendrnx\toksb{&}% \ldexpwithtype#2% location and type \concat\toksb\toksc \appendrnx\toksb{&}% \yystringempty{#3}{\ldpushalignment#4}{\toksc{{\ttl at }$#3$}}% alignment \concat\toksb\toksc \appendrnx\toksb{&}% \yystringempty{#5}{\ldpushplacement#7}{\toksc{{\ttl #5}}}% constraint \concat\toksb\toksc \appendrnx\toksb{&}% \ldstartpheaders#7% possible pheaders \concat\toksb\toksc \appendrnx\toksb{\cr}% \def\sections@header{}% \else \toksb\expandafter{&}% section header \advance\tempca\m@ne \ldextractitem\tempca{#6}% next statement \advance\tempca\@ne \yytoksempty\toksc{% \ldextractfill#7% \iffillextracted \ifnum\tempcb<\tw@ \else \sectioncompletetrue \fi \fi \fillextractedtrue }{% \toksc\expandafter{\expandafter\qquad\the\toksc{}}% }% \concat\toksb\toksc \appendrnx\toksb{&}% % \ldexpwithtype#2% location and type % \concat\toksa\toksc \appendrnx\toksb{&}% \ldpushalignment#4% alignment \concat\toksb\toksc \appendrnx\toksb{&}% \ldpushplacement#7% placement \concat\toksb\toksc \appendrnx\toksb{&}% \ldpushpheaders#7% possible pheaders \concat\toksb\toksc \appendrnx\toksb{\cr}% \def\sections@header{}% \fi \ifsectioncomplete \else \concat\toksa\toksb \advance\tempca\@ne \repeat } % named section :: \ldnamedsection{name} % {{bind?}{expression}{block?}{expression}{type}}{at} % {{}{}{}:alignment} % {constraint}{statements} % {{memspec}{memspec_at}{phdr}{fill}} \defc\ldoverlay{% overlay sections :: \ldoverlay{expression}{crossrefs}{at}{subalign} % {sections} % {{memspec}{memspec_at}{phdr}{fill}} \yystringempty{#1}{\tokse{}}{\tokse{ $#1$}}% \yystringempty{#2}{\toksf{}}{\toksf{{ \ttl noxrefs}}}% \yystringempty{#3}{\toksg{}\toksc{}}{\toksg{{\ttl at} $#3$}\toksc{ }}% \yystringempty{#4}{\toksh{}}{\toksh\expandafter{\expandafter{\the\tokc\ttl subalign} $#4$}}% \appendr\toksa{&\hskip-1em\noexpand\ttl overlay\the\tokse\the\toksf&&\the\toksg\the\toksh}% \ldwrapoverlay#6% \appendr\toksa{&\the\toksc&\cr}% #5% \ldoverlayfill#6% \yytoksempty\toksd{}{\appendr\toksa{&\the\toksd&&&&&\cr}}% attach the fill at the end \appendr\toksa{&\hskip-1em{\ttl overlay end}&&&&&\cr}% } \defc\ldoverlaysection{% overlay section :: \ldoverlaysection{name}{statements}{phdr}{fill} \ldnamedsection{#1}% {{}{}{}{}{}}{}% {{}{}{}}% {}{#2}% {{}{}{#3}{#4}}% } \def\ldwrapoverlay#1#2#3#4{% puts memspec... in \toksc and fill in \toksd \toksc{}% \yystringempty{#1}{% any > ? }{% \ldextractname#1% \toksd{{\ttl in }}% \concatl\toksd\toksc }% \yystringempty{#2}{% any AT > ? }{% \yytoksempty\toksc{\toksc{{\ttl as }}}{\appendrnx\toksc{ {\ttl as }}}% \toksd=\toksc \ldextractname#2% \concat\toksd\toksc \toksc=\toksd }% } \def\ldoverlayfill#1#2#3#4{% \ld@xtractfill{#4}% } \def\ldextractitem#1#2{% #1 is a counter, #2 is a list separated by \ldor \yystringempty{#2}{% \toksc{}% }{% \let\ldor\or \toksc=\ifcase#1#2\else{}\fi \let\ldor\relax }% } \def\ldpushalignment#1#2#3{% \toksd=\ifcase\tempcb{#1}\or{#2}\or{#3}\else{}\fi \yytoksempty\toksd{% \advance\tempcb\@ne \ifnum\tempcb<\tw@ \yybreak{\ldpushalignment{#1}{#2}{#3}}% \else \yybreak{\toksc{}}% \yycontinue }{% \toksc=\ifcase\tempcb{{\ttl align }}\concat\toksc\toksd\or {{\ttl align\_with\_input}}\or {{\ttl subalign }}\concat\toksc\toksd\fi \advance\tempcb\@ne }% } \def\ldpushplacement#1#2#3#4{% \ifplacementpushed \toksc{}% \else \yystringempty{#1}{% any > ? \toksc{}% }{% \ldextractname#1% \toksd{{\ttl in }}% \concatl\toksd\toksc }% \yystringempty{#2}{% any AT > ? }{% \yytoksempty\toksc{\toksc{{\ttl as }}}{\appendrnx\toksc{ {\ttl as }}}% \toksd=\toksc \ldextractname#2% \concat\toksd\toksc \toksc=\toksd }% \placementpushedtrue \fi } \def\ldstartpheaders#1#2#3#4{% \tempcc=\z@ \yystringempty{#3}{\toksc{}}{\toksc{{\ttl phdrs}}}% } \def\ldextractfill#1#2#3#4{% \toksc{}% \yystringempty{#4}{% \fillextractedtrue }{% \iffillextracted \else \ld@xtractfill{#4}% \appendr\toksc{\noexpand\qquad{\noexpand\ttl fill }$\the\toksd$}% \fi }% } \def\ld@xtractfill#1{% #1 must be \ldfill{expression} \toksf{#1}% \toksd\expandafter{#1}% } \def\ldpushpheaders#1#2#3#4{% \ldextractitem\tempcc{#3}% \advance\tempcc\@ne \yytoksempty\toksc{}{\expandafter\ldextractname\the\toksc}% } \def\ldexpwithtype#1#2#3#4#5{% TODO \yystringempty{#2}{\toksc{$}}{\toksc{$#2}}%$ \yystringempty{#5}{\toksd{{}$}}{#5\appendlnx\toksd{[}\appendrnx\toksd{]$}}% \concat\toksc\toksd } \defc\ldtype{% \toksd{\hbox{\ttl #1}}% } \defc\ldstatement{{\let\ldrlap\relax\let\ldintfont\ntt@$#1$}} % the braces form the group for a \toks assignment \defc\ldfreestmt{\toksb\expandafter{#1}\appendr\toksa{&\the\toksb\cr}} \defc\ldsecspec{{$#1$}} \def\ldboxstash#1{% \ifchecktrim\errmessage{stash contents: \the\toksa}\fi {\setbox0 \vbox{\the\toksa}\ifdim\ht0=\z@\aftergroup\toksa\else\aftergroup\eatone\fi}{}% \yytoksempty\toksa{#1}{% $\vtop{\activateinlinec\tabskip\z@\halign{\strut\ignorespaces##\hfil\cr\the\toksa\crcr}}$\hfill}} \def\ldmakestashbox#1{\cleanstash\stripstash\ldboxstash{#1}} \defc\ldsectionseparator{% \expandafter\ifx\csname ldstashentry[#2]\endcsname\relax \else \expandafter\expandafter\expandafter\yystashlocal\expandafter\expandafter\expandafter{\csname ldstashentry[#2]\endcsname}% \appendr\toksa{&\nx\multispan5\toksa{\the\yystashlocal}\nx\ldmakestashbox{\nx\cdotfill}\nx\quad\cr}% \fi } \defc\ldsectionstash{% \expandafter\ifx\csname ldstashentry[#2]\endcsname\relax \else \expandafter\expandafter\expandafter\yystashlocal\expandafter\expandafter\expandafter{\csname ldstashentry[#2]\endcsname}% \appendr\toksa{\sections@header&\nx\multispan5\toksa{\the\yystashlocal}\nx\ldmakestashbox{\nx\cdotfill}\nx\quad\cr}% \let\sections@header\empty \fi } \restorecs{ld-display}{\ldregexp\ldassignment\ldfill\ldinsertcweb\ldentry} \toyyunion{ld-parser:sections} % preprocessing macros: collecting stash and marking variables % currently these macros do not get inside expressions to extract % variable names, consequently if the variable is used before it % appears in an assignment, it will not be displayed properly % (is this even legal in linker scripts?) although the index will % show it correctly \restorecslist{ld-parser-prototypes}\ldunion \restorecs{ld-parser-strict}{\insertcweb} \defc\ldmemory{#1} % memory specification :: \ldmemory{memory spcification} \defc\ldmemspecstash{% \readstash{#2}% \setbox0 \vbox{\toksa\expandafter{\the\yystashlocal}\cleanstash\stripstash\the\toksa}% \ifdim\ht0=\z@ \expandafter\let\csname ldstashentry[#2]\endcsname\relax % this is redundant but we may change \relax to % something else later \else \expandafter\edef\csname ldstashentry[#2]\endcsname{\the\yystashlocal}% \fi } % memory spec stash :: \ldmemspecstash{fptr}{sptr} \defc\ldmemspecseparator{% \readstash{#2}% \setbox0 \vbox{\toksa\expandafter{\the\yystashlocal}\cleanstash\stripstash\the\toksa}% \ifdim\ht0=\z@ \expandafter\let\csname ldstashentry[#2]\endcsname\relax \else \expandafter\edef\csname ldstashentry[#2]\endcsname{\the\yystashlocal}% \fi } % memory spec separator :: \ldmemspecseparator{fptr}{sptr} \defc\ldsections{#1} % ld sections :: \ldsections{sections} \defc\ldoverlay{% overlay section :: \ldoverlay{expression}{crossrefs}{at}{subalign} % {sections} % {{memspec}{memspec_at}{phdr}{fill}} #3#5% } % TODO: add handling of subalign, memmspec, and memspec_at \defc\ldsectionseparator{% \readstash{#2}% \setbox0 \vbox{\toksa\expandafter{\the\yystashlocal}\cleanstash\stripstash\the\toksa}% \ifdim\ht0=\z@ \expandafter\let\csname ldstashentry[#2]\endcsname\relax \else \expandafter\edef\csname ldstashentry[#2]\endcsname{\the\yystashlocal}% \fi } % section separator :: \ldsectionseparator{fptr}{sptr} \defc\ldsectionstash{% \readstash{#2}% \setbox0 \vbox{\toksa\expandafter{\the\yystashlocal}\cleanstash\stripstash\the\toksa}% \ifdim\ht0=\z@ \expandafter\let\csname ldstashentry[#2]\endcsname\relax \else \expandafter\edef\csname ldstashentry[#2]\endcsname{\the\yystashlocal}% \fi } % sections spec stash :: \ldsectionstash{fptr}{sptr} \defc\ldcommandseparator{% \readstash{#2}% \setbox0 \vbox{\toksa\expandafter{\the\yystashlocal}\cleanstash\stripstash\the\toksa}% \ifdim\ht0=\z@ \expandafter\let\csname ldstashentry[#2]\endcsname\relax \else \expandafter\edef\csname ldstashentry[#2]\endcsname{\the\yystashlocal}% \fi } % command separator :: \ldcommandseparator{fptr}{sptr}{prev command}{next command} \defc\ldstatement{#1} \defc\ldfreestmt{#1} \defc\ldassignment{#1} \defc\ldprovide{#1} \defc\ldentry{#1} \let\ldhidden\ldprovide \let\ldprovidehid\ldprovide \defc\ldnamedsection{\let\ldor\empty#6} \defc\ldregexp{% \ldr@gexprestash#1% } \def\ldr@gexprestash#1#2#3#4{ \expandafter\def\csname ldvarname[#2]\endcsname{\errmessage{Not an \\ldregexp!, maybe change the parser namespace?}}% } \toyyunion{ld-parser:restash}