2015-06-30 17:01:00 +00:00
|
|
|
/* -*- 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 "mozilla/UniquePtr.h"
|
|
|
|
#include "mozilla/Variant.h"
|
|
|
|
|
|
|
|
using mozilla::MakeUnique;
|
|
|
|
using mozilla::UniquePtr;
|
|
|
|
using mozilla::Variant;
|
|
|
|
|
|
|
|
struct Destroyer {
|
|
|
|
static int destroyedCount;
|
|
|
|
~Destroyer() {
|
|
|
|
destroyedCount++;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
int Destroyer::destroyedCount = 0;
|
|
|
|
|
|
|
|
static void
|
|
|
|
testSimple()
|
|
|
|
{
|
|
|
|
printf("testSimple\n");
|
|
|
|
Variant<uint32_t, uint64_t> v(uint64_t(1));
|
|
|
|
MOZ_RELEASE_ASSERT(v.is<uint64_t>());
|
|
|
|
MOZ_RELEASE_ASSERT(!v.is<uint32_t>());
|
|
|
|
MOZ_RELEASE_ASSERT(v.as<uint64_t>() == 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
testCopy()
|
|
|
|
{
|
|
|
|
printf("testCopy\n");
|
|
|
|
Variant<uint32_t, uint64_t> v1(uint64_t(1));
|
|
|
|
Variant<uint32_t, uint64_t> v2(v1);
|
|
|
|
MOZ_RELEASE_ASSERT(v2.is<uint64_t>());
|
|
|
|
MOZ_RELEASE_ASSERT(!v2.is<uint32_t>());
|
|
|
|
MOZ_RELEASE_ASSERT(v2.as<uint64_t>() == 1);
|
|
|
|
|
|
|
|
Variant<uint32_t, uint64_t> v3(uint32_t(10));
|
|
|
|
v3 = v2;
|
|
|
|
MOZ_RELEASE_ASSERT(v3.is<uint64_t>());
|
|
|
|
MOZ_RELEASE_ASSERT(v3.as<uint64_t>() == 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
testMove()
|
|
|
|
{
|
|
|
|
printf("testMove\n");
|
|
|
|
Variant<UniquePtr<int>, char> v1(MakeUnique<int>(5));
|
|
|
|
Variant<UniquePtr<int>, char> v2(Move(v1));
|
|
|
|
|
|
|
|
MOZ_RELEASE_ASSERT(v2.is<UniquePtr<int>>());
|
|
|
|
MOZ_RELEASE_ASSERT(*v2.as<UniquePtr<int>>() == 5);
|
|
|
|
|
|
|
|
MOZ_RELEASE_ASSERT(v1.is<UniquePtr<int>>());
|
|
|
|
MOZ_RELEASE_ASSERT(v1.as<UniquePtr<int>>() == nullptr);
|
|
|
|
|
|
|
|
Destroyer::destroyedCount = 0;
|
|
|
|
{
|
|
|
|
Variant<char, UniquePtr<Destroyer>> v3(MakeUnique<Destroyer>());
|
|
|
|
Variant<char, UniquePtr<Destroyer>> v4(Move(v3));
|
|
|
|
|
|
|
|
Variant<char, UniquePtr<Destroyer>> v5('a');
|
|
|
|
v5 = Move(v4);
|
|
|
|
|
|
|
|
auto ptr = v5.extract<UniquePtr<Destroyer>>();
|
|
|
|
MOZ_RELEASE_ASSERT(Destroyer::destroyedCount == 0);
|
|
|
|
}
|
|
|
|
MOZ_RELEASE_ASSERT(Destroyer::destroyedCount == 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
testDestructor()
|
|
|
|
{
|
|
|
|
printf("testDestructor\n");
|
|
|
|
Destroyer::destroyedCount = 0;
|
|
|
|
|
|
|
|
{
|
|
|
|
Destroyer d;
|
|
|
|
|
|
|
|
{
|
|
|
|
Variant<char, UniquePtr<char[]>, Destroyer> v(d);
|
|
|
|
MOZ_RELEASE_ASSERT(Destroyer::destroyedCount == 0); // None detroyed yet.
|
|
|
|
}
|
|
|
|
|
|
|
|
MOZ_RELEASE_ASSERT(Destroyer::destroyedCount == 1); // v's copy of d is destroyed.
|
|
|
|
}
|
|
|
|
|
|
|
|
MOZ_RELEASE_ASSERT(Destroyer::destroyedCount == 2); // d is destroyed.
|
|
|
|
}
|
|
|
|
|
2015-07-19 14:32:00 +00:00
|
|
|
static void
|
|
|
|
testEquality()
|
|
|
|
{
|
|
|
|
printf("testEquality\n");
|
|
|
|
using V = Variant<char, int>;
|
|
|
|
|
|
|
|
V v0('a');
|
|
|
|
V v1('b');
|
|
|
|
V v2('b');
|
|
|
|
V v3(42);
|
|
|
|
V v4(27);
|
|
|
|
V v5(27);
|
|
|
|
V v6(int('b'));
|
|
|
|
|
|
|
|
MOZ_RELEASE_ASSERT(v0 != v1);
|
|
|
|
MOZ_RELEASE_ASSERT(v1 == v2);
|
|
|
|
MOZ_RELEASE_ASSERT(v2 != v3);
|
|
|
|
MOZ_RELEASE_ASSERT(v3 != v4);
|
|
|
|
MOZ_RELEASE_ASSERT(v4 == v5);
|
|
|
|
MOZ_RELEASE_ASSERT(v1 != v6);
|
|
|
|
|
|
|
|
MOZ_RELEASE_ASSERT(v0 == v0);
|
|
|
|
MOZ_RELEASE_ASSERT(v1 == v1);
|
|
|
|
MOZ_RELEASE_ASSERT(v2 == v2);
|
|
|
|
MOZ_RELEASE_ASSERT(v3 == v3);
|
|
|
|
MOZ_RELEASE_ASSERT(v4 == v4);
|
|
|
|
MOZ_RELEASE_ASSERT(v5 == v5);
|
|
|
|
MOZ_RELEASE_ASSERT(v6 == v6);
|
|
|
|
}
|
|
|
|
|
2015-08-08 23:43:35 +00:00
|
|
|
struct Describer
|
|
|
|
{
|
|
|
|
static const char* little;
|
|
|
|
static const char* medium;
|
|
|
|
static const char* big;
|
|
|
|
|
|
|
|
using ReturnType = const char*;
|
|
|
|
|
|
|
|
const char* match(const uint8_t&) { return little; }
|
|
|
|
const char* match(const uint32_t&) { return medium; }
|
|
|
|
const char* match(const uint64_t&) { return big; }
|
|
|
|
};
|
|
|
|
|
|
|
|
const char* Describer::little = "little";
|
|
|
|
const char* Describer::medium = "medium";
|
|
|
|
const char* Describer::big = "big";
|
|
|
|
|
|
|
|
static void
|
|
|
|
testMatching()
|
|
|
|
{
|
|
|
|
printf("testMatching\n");
|
|
|
|
using V = Variant<uint8_t, uint32_t, uint64_t>;
|
|
|
|
|
|
|
|
Describer desc;
|
|
|
|
|
|
|
|
V v1(uint8_t(1));
|
|
|
|
V v2(uint32_t(2));
|
|
|
|
V v3(uint64_t(3));
|
|
|
|
|
|
|
|
MOZ_RELEASE_ASSERT(v1.match(desc) == Describer::little);
|
|
|
|
MOZ_RELEASE_ASSERT(v2.match(desc) == Describer::medium);
|
|
|
|
MOZ_RELEASE_ASSERT(v3.match(desc) == Describer::big);
|
|
|
|
|
|
|
|
const V& constRef1 = v1;
|
|
|
|
const V& constRef2 = v2;
|
|
|
|
const V& constRef3 = v3;
|
|
|
|
|
|
|
|
MOZ_RELEASE_ASSERT(constRef1.match(desc) == Describer::little);
|
|
|
|
MOZ_RELEASE_ASSERT(constRef2.match(desc) == Describer::medium);
|
|
|
|
MOZ_RELEASE_ASSERT(constRef3.match(desc) == Describer::big);
|
|
|
|
}
|
|
|
|
|
2015-06-30 17:01:00 +00:00
|
|
|
int
|
|
|
|
main()
|
|
|
|
{
|
|
|
|
testSimple();
|
|
|
|
testCopy();
|
|
|
|
testMove();
|
|
|
|
testDestructor();
|
2015-07-19 14:32:00 +00:00
|
|
|
testEquality();
|
2015-08-08 23:43:35 +00:00
|
|
|
testMatching();
|
2015-06-30 17:01:00 +00:00
|
|
|
|
|
|
|
printf("TestVariant OK!\n");
|
|
|
|
return 0;
|
|
|
|
}
|