/* ScopedHolder(Test) - holding and owning noncopyable objects Copyright (C) Lumiera.org 2008, Hermann Vosseler This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. * *****************************************************/ #include "lib/test/run.hpp" #include "lib/test/test-helper.hpp" #include "lib/util.hpp" #include "lib/error.hpp" #include "lib/scoped-holder.hpp" #include "testdummy.hpp" #include #include #include namespace lib { namespace test{ using ::Test; using util::isnil; using lumiera::error::LUMIERA_ERROR_LOGIC; using std::map; using std::cout; typedef ScopedHolder HolderD; typedef ScopedPtrHolder PtrHolderD; /********************************************************************************** * @test ScopedHolder and ScopedPtrHolder are initially empty and copyable. * After taking ownership, they prohibit copy operations, manage the * lifecycle of the contained object and provide smart-ptr like access. * A series of identical tests is conducted both with the ScopedPtrHolder * (the contained objects are heap allocated but managed by the holder) * and with the ScopedHolder (objects placed inline) */ class ScopedHolder_test : public Test { virtual void run (Arg) { cout << "checking ScopedHolder...\n"; checkAllocation(); checkErrorHandling(); checkCopyProtocol(); checkSTLContainer(); cout << "checking ScopedPtrHolder...\n"; checkAllocation(); checkErrorHandling(); checkCopyProtocol(); checkSTLContainer(); } void create_contained_object (HolderD& holder) { holder.create(); } void create_contained_object (PtrHolderD& holder) { holder.reset(new Dummy()); } template void checkAllocation() { ASSERT (0==checksum); { HO holder; ASSERT (!holder); ASSERT (0==checksum); create_contained_object (holder); ASSERT (holder); ASSERT (false!=holder); ASSERT (holder!=false); ASSERT (0!=checksum); ASSERT ( &(*holder)); ASSERT (holder->add(2) == checksum+2); Dummy *rawP = holder.get(); ASSERT (rawP); ASSERT (holder); ASSERT (rawP == &(*holder)); ASSERT (rawP->add(-5) == holder->add(-5)); TRACE (test, "holder at %p", &holder); TRACE (test, "object at %p", holder.get() ); TRACE (test, "size(object) = %lu", sizeof(*holder)); TRACE (test, "size(holder) = %lu", sizeof(holder)); } ASSERT (0==checksum); } template void checkErrorHandling() { ASSERT (0==checksum); { HO holder; throw_in_ctor = true; try { create_contained_object (holder); NOTREACHED (); } catch (int val) { ASSERT (0!=checksum); checksum -= val; ASSERT (0==checksum); } ASSERT (!holder); /* because the exception happens in ctor object doesn't count as "created" */ throw_in_ctor = false; } ASSERT (0==checksum); } template void checkCopyProtocol() { ASSERT (0==checksum); { HO holder; HO holder2 (holder); holder2 = holder; // copy and assignment of empty holders is tolerated // but after enclosing an object it will be copy protected... ASSERT (!holder); create_contained_object (holder); ASSERT (holder); long currSum = checksum; void* adr = holder.get(); VERIFY_ERROR(LOGIC, holder2 = holder ); ASSERT (holder); ASSERT (!holder2); ASSERT (holder.get()==adr); ASSERT (checksum==currSum); VERIFY_ERROR(LOGIC, holder = holder2 ); ASSERT (holder); ASSERT (!holder2); ASSERT (holder.get()==adr); ASSERT (checksum==currSum); create_contained_object (holder2); ASSERT (holder2); ASSERT (checksum != currSum); currSum = checksum; VERIFY_ERROR(LOGIC, holder = holder2 ); ASSERT (holder); ASSERT (holder2); ASSERT (holder.get()==adr); ASSERT (checksum==currSum); VERIFY_ERROR(LOGIC, HO holder3 (holder2) ); ASSERT (holder); ASSERT (holder2); ASSERT (checksum==currSum); } ASSERT (0==checksum); } /** @test collection of noncopyable objects * maintained within a STL map */ template void checkSTLContainer() { typedef std::map MapHO; ASSERT (0==checksum); { MapHO maph; ASSERT (isnil (maph)); for (uint i=0; i<100; ++i) { HO & contained = maph[i]; ASSERT (!contained); } // 100 holder objects created by sideeffect ASSERT (0==checksum); // ..... without creating any contained object! ASSERT (!isnil (maph)); ASSERT (100==maph.size()); for (uint i=0; i<100; ++i) { create_contained_object (maph[i]); ASSERT (maph[i]); ASSERT (0 < maph[i]->add(12)); } ASSERT (100==maph.size()); ASSERT (0!=checksum); long value55 = maph[55]->add(0); long currSum = checksum; ASSERT (1 == maph.erase(55)); ASSERT (checksum == currSum - value55); // proves object#55's dtor has been invoked ASSERT (maph.size() == 99); maph[55]; // create new empty holder by sideeffect... ASSERT (&maph[55]); ASSERT (!maph[55]); ASSERT (maph.size() == 100); } ASSERT (0==checksum); } }; LAUNCHER (ScopedHolder_test, "unit common"); }} // namespace lib::test