// { dg-do run }
|
// { dg-do run }
|
// { dg-options "-w" }
|
// { dg-options "-w" }
|
// Copyright (C) 1999, 2000 Free Software Foundation, Inc.
|
// Copyright (C) 1999, 2000 Free Software Foundation, Inc.
|
// Contributed by Nathan Sidwell 6 Jun 1999
|
// Contributed by Nathan Sidwell 6 Jun 1999
|
|
|
// We cannot catch an ambiguous base class.
|
// We cannot catch an ambiguous base class.
|
// -- public, << private, == virtual
|
// -- public, << private, == virtual
|
|
|
// D--B--A
|
// D--B--A
|
// +--C<
|
// +--C<
|
|
|
|
|
struct A { int m; virtual ~A(){}};
|
struct A { int m; virtual ~A(){}};
|
struct B : A { int m; };
|
struct B : A { int m; };
|
struct C : private A { int m; };
|
struct C : private A { int m; };
|
struct D : B, C { int m; };
|
struct D : B, C { int m; };
|
|
|
void fna(A *obj) { throw obj; }
|
void fna(A *obj) { throw obj; }
|
void fnb(B *obj) { throw obj; }
|
void fnb(B *obj) { throw obj; }
|
void fnc(C *obj) { throw obj; }
|
void fnc(C *obj) { throw obj; }
|
void fnd(D *obj) { throw obj; }
|
void fnd(D *obj) { throw obj; }
|
|
|
extern "C" void abort();
|
extern "C" void abort();
|
|
|
void check(D *d)
|
void check(D *d)
|
{
|
{
|
int caught;
|
int caught;
|
|
|
// try with whole object
|
// try with whole object
|
caught = 0;
|
caught = 0;
|
try { fnd(d); }
|
try { fnd(d); }
|
catch(A *p) { abort(); } // A is ambiguous
|
catch(A *p) { abort(); } // A is ambiguous
|
catch(D *p) { caught = 1; if (p != d) abort();}
|
catch(D *p) { caught = 1; if (p != d) abort();}
|
catch(...) { abort(); }
|
catch(...) { abort(); }
|
if (!caught) abort();
|
if (!caught) abort();
|
|
|
caught = 0;
|
caught = 0;
|
try { fnd(d); }
|
try { fnd(d); }
|
catch(A *p) { abort(); } // A is ambiguous
|
catch(A *p) { abort(); } // A is ambiguous
|
catch(B *p) { caught = 1; if (p != d) abort();}
|
catch(B *p) { caught = 1; if (p != d) abort();}
|
catch(...) { abort(); }
|
catch(...) { abort(); }
|
if (!caught) abort();
|
if (!caught) abort();
|
|
|
caught = 0;
|
caught = 0;
|
try { fnd(d); }
|
try { fnd(d); }
|
catch(A *p) { abort(); } // A is ambiguous
|
catch(A *p) { abort(); } // A is ambiguous
|
catch(C *p) { caught = 1; if (p != d) abort();}
|
catch(C *p) { caught = 1; if (p != d) abort();}
|
catch(...) { abort(); }
|
catch(...) { abort(); }
|
if (!caught) abort();
|
if (!caught) abort();
|
|
|
// try with an A object
|
// try with an A object
|
caught = 0;
|
caught = 0;
|
try { fna((B *)d); }
|
try { fna((B *)d); }
|
catch(B *p) { abort(); } // throw type is static type
|
catch(B *p) { abort(); } // throw type is static type
|
catch(A *p) { caught = 1; if (p != (B *)d) abort();}
|
catch(A *p) { caught = 1; if (p != (B *)d) abort();}
|
catch(...) { abort(); }
|
catch(...) { abort(); }
|
if (!caught) abort();
|
if (!caught) abort();
|
|
|
caught = 0;
|
caught = 0;
|
try { fna((A *)(C *)d); }
|
try { fna((A *)(C *)d); }
|
catch(C *p) { abort(); } // throw type is static type
|
catch(C *p) { abort(); } // throw type is static type
|
catch(A *p) { caught = 1; if (p != (A *)(C *)d) abort();}
|
catch(A *p) { caught = 1; if (p != (A *)(C *)d) abort();}
|
catch(...) { abort(); }
|
catch(...) { abort(); }
|
if (!caught) abort();
|
if (!caught) abort();
|
|
|
// try with B object
|
// try with B object
|
caught = 0;
|
caught = 0;
|
try { fnb((B *)d); }
|
try { fnb((B *)d); }
|
catch(A *p) { caught = 1; if (p != (B *)d) abort();}
|
catch(A *p) { caught = 1; if (p != (B *)d) abort();}
|
catch(...) { abort(); }
|
catch(...) { abort(); }
|
if (!caught) abort();
|
if (!caught) abort();
|
|
|
caught = 0;
|
caught = 0;
|
try { fnb((B *)d); }
|
try { fnb((B *)d); }
|
catch(B *p) { caught = 1; if (p != d) abort();}
|
catch(B *p) { caught = 1; if (p != d) abort();}
|
catch(...) { abort(); }
|
catch(...) { abort(); }
|
if (!caught) abort();
|
if (!caught) abort();
|
|
|
caught = 0;
|
caught = 0;
|
try { fnb((B *)d); }
|
try { fnb((B *)d); }
|
catch(C *p) { abort(); }
|
catch(C *p) { abort(); }
|
catch(D *p) { abort(); }
|
catch(D *p) { abort(); }
|
catch(...) { caught =1; }
|
catch(...) { caught =1; }
|
if (!caught) abort();
|
if (!caught) abort();
|
|
|
// try with C object
|
// try with C object
|
caught = 0;
|
caught = 0;
|
try { fnc((C *)d); }
|
try { fnc((C *)d); }
|
catch(A *p) { abort();}
|
catch(A *p) { abort();}
|
catch(C *p) { caught = 1; if (p != d) abort();}
|
catch(C *p) { caught = 1; if (p != d) abort();}
|
catch(...) { abort(); }
|
catch(...) { abort(); }
|
if (!caught) abort();
|
if (!caught) abort();
|
|
|
caught = 0;
|
caught = 0;
|
try { fnc((C *)d); }
|
try { fnc((C *)d); }
|
catch(B *p) { abort(); }
|
catch(B *p) { abort(); }
|
catch(D *p) { abort(); }
|
catch(D *p) { abort(); }
|
catch(...) { caught =1; }
|
catch(...) { caught =1; }
|
if (!caught) abort();
|
if (!caught) abort();
|
|
|
return;
|
return;
|
}
|
}
|
|
|
int main ()
|
int main ()
|
{
|
{
|
D d;
|
D d;
|
check (&d); // try with an object
|
check (&d); // try with an object
|
check ((D *)0); // try with no object
|
check ((D *)0); // try with no object
|
|
|
return 0;
|
return 0;
|
}
|
}
|
|
|