<div>I'm really frustrated at this development. I'm now going to have to completely re-work my patch for PR9279. This commit actually applies several pieces of an early version of my patch for that bug, without handling the problems that arose from those approaches, or addressing the test cases there.</div>


<div><br></div><div>Why can't we fix the underlying problem first and then do this sort of optimization work? I really like the memory savings and other changes, but we have active bugs with the diagnostics that are still not addressed. Every time we change things underneath, it makes it harder to forward-port the fix I posted.</div>


<div><br></div><div>It would have also been really useful to provide feedback on those patches. I would have been happy to work on the memory restructuring you've done here on top of them and we could have been faster and more correct a long time ago...</div>


<div><br></div><div>On to the specific problems:</div><br><div class="gmail_quote">On Wed, Jul 6, 2011 at 8:40 PM, Argyrios Kyrtzidis <span dir="ltr"><<a href="mailto:akyrtzi@gmail.com" target="_blank" class="cremed">akyrtzi@gmail.com</a>></span> wrote:<br>


<blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">t.c:5:9: error: invalid operands to binary expression ('struct S' and 'int')<br>
int y = M(/);<br>
        ^~~~<br>
t.c:3:20: note: instantiated from:<br>
\#define M(op) (foo op 3);<br>
                ~~~ ^  ~<br>
t.c:5:11: note: instantiated from:<br>
int y = M(/);<br>
          ^<br></blockquote><div><br></div><div>I think re-showing line 5 here is really unfortunate. We already have a problem with too many lines being displayed for diagnostics. Why can't we start off the trace with the macro argument, and trace the macro argument through each level of instantiation? You're already recording most of the information necessary to do this (as you have several of the key elements of my patch for PR9279 in your patch). For example, for (what seems close to) your test case with my patch applied from PR9279 Clang prints the following:</div>


<div><br></div><div><div>t.c:4:13: error: invalid operands to binary expression ('struct S' and 'int')</div><div>  int y = M(/);</div><div>          ~~^~</div><div>t.c:3:20: note: instantiated from:</div>

<div>
#define M(op) (foo op 3)</div><div>               ~~~ ^  ~</div><div>1 error generated.</div></div><div><br></div><div>I think that's a strictly better diagnostic experience. Several of the test cases I included in that PR (and split into other PRs) highlight the difference. I think the best example is:</div>


<div><br></div><div><div>% cat t5.cc </div><div>#define M0 4</div><div>#define M1(x, y, z) y</div><div>#define M2(x, y, z) M1(x, y, z)</div><div>#define M3(x, y, z) M2(x, y, z)</div><div>M3(</div><div>  1,</div><div>  M0,</div>


<div>  3);</div></div><div><br></div><div>Which with your patch gives:</div><div><div>% ./bin/clang -fsyntax-only t5.cc</div><div>t5.cc:5:1: error: expected unqualified-id</div><div>M3(</div><div>^</div><div>t5.cc:4:21: note: instantiated from:</div>


<div>#define M3(x, y, z) M2(x, y, z)</div><div>                    ^</div><div>t5.cc:3:21: note: instantiated from:</div><div>#define M2(x, y, z) M1(x, y, z)</div><div>                    ^</div><div>t5.cc:2:21: note: instantiated from:</div>


<div>#define M1(x, y, z) y</div><div>                    ^</div><div>t5.cc:5:1: note: instantiated from:</div><div>#define M0 4</div><div>           ^</div><div>1 error generated.</div></div><div><br></div><div>We don't see where the user wrote "M0" here. We also don't see where 'y' is used within each macro. My patch produces:</div>


<div><div>% ./bin/clang -fsyntax-only t5.cc</div><div>t5.cc:7:3: error: expected unqualified-id</div><div>  M0,</div><div>  ^</div><div>t5.cc:1:12: note: instantiated from:</div><div>#define M0 4</div><div>           ^</div>


<div>t5.cc:4:27: note: instantiated from:</div><div>#define M3(x, y, z) M2(x, y, z)</div><div>                          ^</div><div>t5.cc:3:27: note: instantiated from:</div><div>#define M2(x, y, z) M1(x, y, z)</div><div>


                          ^</div><div>t5.cc:2:21: note: instantiated from:</div><div>#define M1(x, y, z) y</div><div>                    ^</div><div>1 error generated.</div></div></div>