/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* vim: set ts=8 sts=2 et sw=2 tw=80: */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include #include #define ANNOTATE(property) __attribute__((annotate(property))) struct Cell { int f; } ANNOTATE("GC Thing"); namespace World { namespace NS { struct Unsafe { int g; ~Unsafe() { asm(""); } } ANNOTATE("Invalidated by GC") ANNOTATE("GC Pointer or Reference"); } // namespace NS } // namespace World extern void GC() ANNOTATE("GC Call"); extern void invisible(); void GC() { // If the implementation is too trivial, the function body won't be emitted at // all. asm(""); invisible(); } struct GCOnDestruction { ~GCOnDestruction() { GC(); } }; struct NoGCOnDestruction { ~NoGCOnDestruction() { asm(""); } }; extern void usecell(Cell*); Cell* cell() { static Cell c; return &c; } template struct SimpleTemplate { int member; }; template class ANNOTATE("moz_inherit_type_annotations_from_template_args") Container { public: template void foo(V& v, W& w) { class InnerClass {}; InnerClass xxx; return; } }; Cell* f() { Container c1; Container, SimpleTemplate> c2; Container, Container> c3; Container, void>, Container>> c4; return nullptr; }; void rvalue_ref(World::NS::Unsafe&& arg1) { GC(); } void ref(const World::NS::Unsafe& arg2) { GC(); static int use = arg2.g; } // A function that consumes a parameter, but only if passed by rvalue reference. extern void eat(World::NS::Unsafe&&); extern void eat(World::NS::Unsafe&); void rvalue_ref_ok() { World::NS::Unsafe unsafe1; eat(std::move(unsafe1)); GC(); } void rvalue_ref_not_ok() { World::NS::Unsafe unsafe2; eat(unsafe2); GC(); } void rvalue_ref_arg_ok(World::NS::Unsafe&& unsafe3) { eat(std::move(unsafe3)); GC(); } void rvalue_ref_arg_not_ok(World::NS::Unsafe&& unsafe4) { eat(unsafe4); GC(); } void shared_ptr_hazard() { Cell* unsafe5 = f(); { auto p = std::make_shared(); } usecell(unsafe5); } void shared_ptr_no_hazard() { Cell* safe6 = f(); { auto p = std::make_shared(); } usecell(safe6); }