<html xmlns:o="urn:schemas-microsoft-com:office:office" xmlns:w="urn:schemas-microsoft-com:office:word" xmlns:m="http://schemas.microsoft.com/office/2004/12/omml" xmlns="http://www.w3.org/TR/REC-html40">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8">
<meta name="Title" content="">
<meta name="Keywords" content="">
<meta name="Generator" content="Microsoft Word 15 (filtered medium)">
<style><!--
/* Font Definitions */
@font-face
        {font-family:"Cambria Math";
        panose-1:2 4 5 3 5 4 6 3 2 4;}
@font-face
        {font-family:Calibri;
        panose-1:2 15 5 2 2 2 4 3 2 4;}
/* Style Definitions */
p.MsoNormal, li.MsoNormal, div.MsoNormal
        {margin:0in;
        margin-bottom:.0001pt;
        font-size:12.0pt;
        font-family:"Calibri",sans-serif;}
a:link, span.MsoHyperlink
        {mso-style-priority:99;
        color:#0563C1;
        text-decoration:underline;}
a:visited, span.MsoHyperlinkFollowed
        {mso-style-priority:99;
        color:#954F72;
        text-decoration:underline;}
span.EmailStyle17
        {mso-style-type:personal-compose;
        font-family:"Calibri",sans-serif;
        color:windowtext;}
span.msoIns
        {mso-style-type:export-only;
        mso-style-name:"";
        text-decoration:underline;
        color:teal;}
.MsoChpDefault
        {mso-style-type:export-only;
        font-family:"Calibri",sans-serif;}
@page WordSection1
        {size:8.5in 11.0in;
        margin:1.0in 1.0in 1.0in 1.0in;}
div.WordSection1
        {page:WordSection1;}
--></style>
</head>
<body bgcolor="white" lang="EN-US" link="#0563C1" vlink="#954F72">
<div class="WordSection1">
<p class="MsoNormal"><span style="font-size:11.0pt">Hello,<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt"><o:p> </o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">I recently ran into an issue where I wanted to use dyn_cast with a Multiple Inheritance hierarchy. LLVM’s help page on RTTI claims that it can be done, and that Clang’s Decl and DeclContext implement it; however,
 when I try to use it I run into odd behavior. Here’s my sample code which doesn’t work:<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt"><o:p> </o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">```<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">struct Base {<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">  void *ptr;<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">  bool hasInfo;<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">};<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt"><o:p> </o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">struct Info {<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">  int size;<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt"><o:p> </o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">  static bool classof(const Base *b) { return b->hasInfo; };<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">};<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt"><o:p> </o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">struct Child : public Base {<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">  Child() {<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">    this->ptr = this;<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">    this->hasInfo = false;<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">  }<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">};<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">struct ChildWithInfo : public Base, public Info {<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">  ChildWithInfo() {<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">    this->ptr = this;<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">    this->hasInfo = true;<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">    this->size = 10;<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">  }<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">};<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt"><o:p> </o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">int main() {<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">  Base *c = new Child();<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">  Base *i = new ChildWithInfo();<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">  try {<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">    if (Info *inf = llvm::dyn_cast<Info>(c)) {<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">      throw std::string("Casted a child to an info incorrectly");<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">    }<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">    if (Info *inf = llvm::dyn_cast<Info>(i)) {<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">      if (inf->size != 10) {<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">        std::ostringstream str;<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">        str << "Object was sliced: expected 10 but got " << inf->size;<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">        throw str.str();<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">      }<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">    } else {<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">      throw std::string("Couldn't cast child with info to info");<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">    }<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">  } catch (std::string &msg) {<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">    std::cout << msg << std::endl;<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt"> }<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">  delete c;<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">  delete i;<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">}<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">```<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt"><o:p> </o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">Basically the error is that when a `Base` is cast to an `Info`, it starts reading out of the same offset where `ptr` is stored instead of `size`.<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">How can I modify my class setup so that I can use this scheme safely?<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt"><o:p> </o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">Looking at `Decl` and `DeclContext`, it is unclear to me how they set things up so that the field offsets are correct, although I have a feeling<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">It has something to do with the `LLVM_ALIGNAS` in the class declaration of `Decl`.<br>
<br>
<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">Regards,<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt">Riley Dulin<o:p></o:p></span></p>
<p class="MsoNormal"><span style="font-size:11.0pt"><o:p> </o:p></span></p>
</div>
</body>
</html>