1
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 5 years have passed since last update.

AUTOSAR CountdownAdvent Calendar 2022

Day 22

N4606 Working Draft 2016, ISO/IEC 14882, C++ standard(40)5.2.7 Dynamic cast [expr.dynamic.cast] p111

Posted at

#はじめに
N4606 Working Draft, Standard for Programming Language C++
http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2016/#mailing2016-11
http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2016/n4606.pdf

n4606は、ISO/IEC JTC1 SC22 WG21の作業原案(Working Draft)です。
公式のISO/IEC 14882原本ではありません。
ISO/IEC JTC1 SC22 WG21では、可能な限り作業文書を公開し、幅広い意見を求めています。
一連の記事はコード断片をコンパイルできる形にする方法を検討してコンパイル、リンク、実行して、規格案の原文と処理系(g++, Clang++)との違いを確認し、技術内容を検討し、ISO/IEC JTC1 SC22 WG21にフィードバックするために用います。
また、CERT C++, MISRA C++等のコーディング標準のコード断片をコンパイルする際の参考にさせていただこうと考えています。CERT C++, MISRA C++が標準化の動きとの時間的な擦れの確認が結果としてできれば幸いです。

list

N4606 Working Draft 2016, ISO/IEC 14882, C++ standard(1) coding list
https://qiita.com/kaizen_nagoya/items/df5d62c35bd6ed1c3d43/

Compiler

###clang++ --version
clang version 6.0.0 (tags/RELEASE_600/final)

###g++-7 --version
g++-7 (Homebrew GCC 7.3.0_1) 7.3.0
Copyright (C) 2017 Free Software Foundation, Inc.

#(40)5.2.7 Dynamic cast [expr.dynamic.cast]
p111

p111.cpp
// N4606 Committee Draft, Standard for Programming Language C++
// http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2016/n4606.pdf 
#define msg "p111.cpp (40)5.2.7 Dynamic cast [expr.dynamic.cast]"

#include <iostream>

struct B1 { };
struct D1 : B1 { };
void foo(D1* dp) {
B1* bp = dynamic_cast<B1*>(dp); // equivalent to B1* bp = dp;
}

class A { virtual void f(); };
class B { virtual void g(); };
class D : public virtual A, private B { };
void g() {
D d;
B* bp = (B*)&d; // cast needed to break protection
A* ap = &d; // public derivation, no cast needed
D& dr = dynamic_cast<D&>(*bp); // fails
ap = dynamic_cast<A*>(bp); // fails
bp = dynamic_cast<B*>(ap); // fails
ap = dynamic_cast<A*>(&d); // succeeds
bp = dynamic_cast<B*>(&d); // ill-formed (not a run-time check)
}
class E : public D, public B { };
class F : public E, public D { };
void h() {
F f;
A* ap = &f; // succeeds: finds unique A
D* dp = dynamic_cast<D*>(ap); // fails: yields 0
// f has two D subobjects
E* ep = (E*)ap; // ill-formed: cast from virtual base
E* ep1 = dynamic_cast<E*>(ap); // succeeds
}


int main(int argc, char *argv[], char *envp[]){
  std::cout << msg << std::endl;
  return EXIT_SUCCESS;
}
$ ./cppgl17.sh p111
$ clang++ p111.cpp
p111.cpp:24:6: error: cannot cast 'D' to its private base class 'B'
bp = dynamic_cast<B*>(&d); // ill-formed (not a run-time check)
     ^
p111.cpp:15:29: note: declared private here
class D : public virtual A, private B { };
                            ^~~~~~~~~
p111.cpp:26:21: warning: direct base 'B' is inaccessible due to ambiguity:
    class E -> class D -> class B
    class E -> class B [-Winaccessible-base]
class E : public D, public B { };
                    ^~~~~~~~
p111.cpp:27:21: warning: direct base 'D' is inaccessible due to ambiguity:
    class F -> class E -> class D
    class F -> class D [-Winaccessible-base]
class F : public E, public D { };
                    ^~~~~~~~
p111.cpp:33:9: error: cannot cast 'A *' to 'E *' via virtual base 'A'
E* ep = (E*)ap; // ill-formed: cast from virtual base
        ^~~~~~
2 warnings and 2 errors generated.

$ g++-7 p111.cpp
p111.cpp: In function 'void g()':
p111.cpp:24:25: error: 'B' is an inaccessible base of 'D'
 bp = dynamic_cast<B*>(&d); // ill-formed (not a run-time check)
                         ^
p111.cpp:24:25: error: 'B' is an inaccessible base of 'D'
p111.cpp: At global scope:
p111.cpp:26:7: warning: direct base 'B' inaccessible in 'E' due to ambiguity
 class E : public D, public B { };
       ^
p111.cpp:27:7: warning: direct base 'D' inaccessible in 'F' due to ambiguity
 class F : public E, public D { };
       ^
p111.cpp: In function 'void h()':
p111.cpp:33:13: error: cannot convert from pointer to base class 'A' to pointer to derived class 'E' because the base is virtual
 E* ep = (E*)ap; // ill-formed: cast from virtual base
             ^~

#検討事項
 コンパイルエラーにならない修正。
 内容の出力。
 g++とclang++のエラーの違いの説明。

#参考資料

コンパイル用shell script C版(clangとgcc)とC++版(clang++とg++)
https://qiita.com/kaizen_nagoya/items/74220c0577a512c2d7da
Qiitaに投稿するCのStyle例(暫定)
https://qiita.com/kaizen_nagoya/items/946df1528a6a1ef2bc0d
#文書履歴

0.10 初稿 2080415

1
1
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
1
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?