<div dir="ltr"><div class="gmail_extra"><div class="gmail_quote">On 21 March 2017 at 14:53, Bruno Cardoso Lopes <span dir="ltr"><<a href="mailto:bruno.cardoso@gmail.com" target="_blank">bruno.cardoso@gmail.com</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">(+cfe-dev & others for extra input)<br>
<br>
Hi Richard,<br>
<br>
Unlike C++/ObjC++, we don't currently merge struct (and others)<br>
definitions in C/ObjC. This is fine since we are not even talking<br>
about local submodules visibility. However, consider the following:<br>
<br>
- Consider module F, and its submodule F.Private.<br>
- The submodule is marked explicit and should only contain private<br>
headers that won't be visible unless by explicit include or @import<br>
Module.Private.<br>
- A private header NS.h defines 'struct NS {...}'<br>
- A TU x.c also defines 'struct NS {...}' but it never includes NS.h.<br>
The private module is built though, since module.private.modulemap<br>
builds any submodule of what has been previously defined in<br>
module.modulemap<br>
- BOOM -> error: redefinition of 'NS'<br>
<br>
Turns out that this example does not error in C++ because we merge<br>
this definition if one of the candidates is hidden, but only in C++.<br>
Shouldn't we behave the same in C/ObjC for the case where: (a) the<br>
definition comes from a module marked 'explicit' and (b) there's no<br>
@import / direct include in the TU at the point of re-definition?<br>
<br>
Am I missing something or is it just that we never really implemented<br>
this? What's your general opinion on this?<br></blockquote><div><br></div><div>The testcase should be made to work, one way or another. In C++, for various reasons we have a strong requirement that there is at most one definition per CXXRecordDecl, and we achieve that in this case by detecting that we're about to parse a second definition and instead skipping the tokens of the new definition and making the existing definition visible.</div><div><br></div><div>I didn't implement the same thing in C because it didn't seem in the spirit of the C linkage rules for types; C does not have a direct analogue of C++'s ODR, and instead has a form of structural typing for cross-translation-unit linking of types. My opinion is that the correct way to handle this is to treat declarations of tag types with the same name in different modules as being different entities in C, and to extend Clang's notion of "compatible type" to treat tag types from two different modules as being compatible types if they have the same name and pass the structural compatibility check described in C11 6.2.7/1. If we want to allow the definition of 'struct NS' in F.Private to be different from that in x.c, I think that's the most sensible way to approach the problem.</div><div><br></div><div>However, we could alternatively say that a rule like C++'s ODR applies to C modules, and use a strategy similar to the one we use in C++ (such as ignoring the definition of 'struct NS' in x.c and substituting the one from NS.h). I think you folks have more experience with C and ObjC modules than anyone else, so if you think something like that is viable, I'm OK with that too.</div><div> </div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">
Here's the reduced testcase:<br>
<br>
== F.framework/Headers/F.h<br>
// F.h<br>
== F.framework/Modules/module.<wbr>modulemap<br>
framework module F [extern_c] [system] {<br>
  umbrella header "F.h"<br>
  module * {<br>
      export *<br>
  }<br>
  export *<br>
}<br>
== F.framework/Modules/module.<wbr>private.modulemap<br>
explicit module F.Private [system] {<br>
  explicit module NS {<br>
      header "NS.h"<br>
      export *<br>
  }<br>
  export *<br>
}<br>
== F.framework/PrivateHeaders/NS.<wbr>h<br>
struct NS {<br>
  int a;<br>
  int b;<br>
};<br>
== x.c<br>
#include "F/F.h"<br>
//@import F.Private;<br>
struct NS {<br>
  int a;<br>
  int b;<br>
};<br>
<br>
$ clang x.c -c -fmodules -F`pwd`<br>
<span class="HOEnZb"><font color="#888888"><br>
--<br>
Bruno Cardoso Lopes<br>
<a href="http://www.brunocardoso.cc" rel="noreferrer" target="_blank">http://www.brunocardoso.cc</a><br>
</font></span></blockquote></div><br></div></div>