2015-08-31 21:42:22 +00:00
|
|
|
// Copyright (c) 2015 The Khronos Group Inc.
|
|
|
|
//
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
|
|
// copy of this software and/or associated documentation files (the
|
|
|
|
// "Materials"), to deal in the Materials without restriction, including
|
|
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
|
|
// distribute, sublicense, and/or sell copies of the Materials, and to
|
|
|
|
// permit persons to whom the Materials are furnished to do so, subject to
|
|
|
|
// the following conditions:
|
|
|
|
//
|
|
|
|
// The above copyright notice and this permission notice shall be included
|
|
|
|
// in all copies or substantial portions of the Materials.
|
|
|
|
//
|
|
|
|
// MODIFICATIONS TO THIS FILE MAY MEAN IT NO LONGER ACCURATELY REFLECTS
|
|
|
|
// KHRONOS STANDARDS. THE UNMODIFIED, NORMATIVE VERSIONS OF KHRONOS
|
|
|
|
// SPECIFICATIONS AND HEADER INFORMATION ARE LOCATED AT
|
|
|
|
// https://www.khronos.org/registry/
|
|
|
|
//
|
|
|
|
// THE MATERIALS ARE PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
|
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
|
|
|
// IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
|
|
|
|
// CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
|
|
|
|
// TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
|
|
|
|
// MATERIALS OR THE USE OR OTHER DEALINGS IN THE MATERIALS.
|
|
|
|
|
2015-09-02 18:48:33 +00:00
|
|
|
#include <cassert>
|
2015-09-11 20:34:49 +00:00
|
|
|
#include <string>
|
2015-09-02 18:48:33 +00:00
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include <gmock/gmock.h>
|
|
|
|
|
2015-08-31 21:42:22 +00:00
|
|
|
#include "TestFixture.h"
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2015-09-22 19:50:33 +00:00
|
|
|
using spvtest::MakeInstruction;
|
2015-09-21 15:36:44 +00:00
|
|
|
using spvtest::TextToBinaryTest;
|
2015-09-02 18:48:33 +00:00
|
|
|
using ::testing::ElementsAre;
|
2015-09-22 19:50:33 +00:00
|
|
|
using ::testing::Eq;
|
2015-09-11 04:43:11 +00:00
|
|
|
using ::testing::HasSubstr;
|
|
|
|
using ::testing::StrEq;
|
2015-08-31 21:42:22 +00:00
|
|
|
|
2015-09-12 00:10:54 +00:00
|
|
|
const auto kCAF = SPV_ASSEMBLY_SYNTAX_FORMAT_CANONICAL;
|
|
|
|
|
2015-08-31 21:42:22 +00:00
|
|
|
TEST_F(TextToBinaryTest, ImmediateIntOpCode) {
|
|
|
|
SetText("!0x00FF00FF");
|
2015-09-09 17:04:32 +00:00
|
|
|
ASSERT_EQ(SPV_SUCCESS,
|
|
|
|
spvTextToBinary(text.str, text.length, opcodeTable, operandTable,
|
|
|
|
extInstTable, &binary, &diagnostic));
|
2015-08-31 21:42:22 +00:00
|
|
|
EXPECT_EQ(0x00FF00FF, binary->code[5]);
|
|
|
|
if (diagnostic) {
|
|
|
|
spvDiagnosticPrint(diagnostic);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(TextToBinaryTest, ImmediateIntOperand) {
|
|
|
|
SetText("OpCapability !0x00FF00FF");
|
2015-09-09 17:04:32 +00:00
|
|
|
EXPECT_EQ(SPV_SUCCESS,
|
|
|
|
spvTextToBinary(text.str, text.length, opcodeTable, operandTable,
|
|
|
|
extInstTable, &binary, &diagnostic));
|
2015-08-31 21:42:22 +00:00
|
|
|
EXPECT_EQ(0x00FF00FF, binary->code[6]);
|
|
|
|
if (diagnostic) {
|
|
|
|
spvDiagnosticPrint(diagnostic);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-02 18:48:33 +00:00
|
|
|
using ImmediateIntTest = TextToBinaryTest;
|
|
|
|
|
|
|
|
TEST_F(ImmediateIntTest, AnyWordInSimpleStatement) {
|
2015-09-29 15:28:34 +00:00
|
|
|
EXPECT_THAT(CompiledInstructions("!0x00040018 %a %b %123", kCAF),
|
|
|
|
Eq(MakeInstruction(spv::OpTypeMatrix, {1, 2,3 })));
|
|
|
|
EXPECT_THAT(CompiledInstructions("OpTypeMatrix !1 %b %123", kCAF),
|
|
|
|
Eq(MakeInstruction(spv::OpTypeMatrix, {1, 1, 2})));
|
|
|
|
EXPECT_THAT(CompiledInstructions("OpTypeMatrix %1 !2 %123", kCAF),
|
|
|
|
Eq(MakeInstruction(spv::OpTypeMatrix, {1, 2, 2})));
|
|
|
|
EXPECT_THAT(CompiledInstructions("OpTypeMatrix %a %b !123", kCAF),
|
|
|
|
Eq(MakeInstruction(spv::OpTypeMatrix, {1, 2, 123})));
|
|
|
|
EXPECT_THAT(CompiledInstructions("!0x00040018 %1 !2 %123", kCAF),
|
|
|
|
Eq(MakeInstruction(spv::OpTypeMatrix, {1, 2, 2})));
|
|
|
|
EXPECT_THAT(CompiledInstructions("OpTypeMatrix !1 %b !123", kCAF),
|
|
|
|
Eq(MakeInstruction(spv::OpTypeMatrix, {1, 1, 123})));
|
|
|
|
EXPECT_THAT(CompiledInstructions("!0x00040018 !1 !2 !123", kCAF),
|
|
|
|
Eq(MakeInstruction(spv::OpTypeMatrix, {1, 2, 123})));
|
2015-09-02 18:48:33 +00:00
|
|
|
}
|
|
|
|
|
2015-09-11 19:03:54 +00:00
|
|
|
TEST_F(ImmediateIntTest, AnyWordAfterEqualsAndOpCode) {
|
2015-09-22 19:50:33 +00:00
|
|
|
EXPECT_THAT(CompiledInstructions("%a = OpArrayLength !2 %c 123"),
|
|
|
|
Eq(MakeInstruction(spv::OpArrayLength, {2, 1, 2, 123})));
|
|
|
|
EXPECT_THAT(CompiledInstructions("%a = OpArrayLength %b !3 123"),
|
|
|
|
Eq(MakeInstruction(spv::OpArrayLength, {1, 2, 3, 123})));
|
|
|
|
EXPECT_THAT(CompiledInstructions("%a = OpArrayLength %b %c !123"),
|
|
|
|
Eq(MakeInstruction(spv::OpArrayLength, {1, 2, 3, 123})));
|
|
|
|
EXPECT_THAT(CompiledInstructions("%a = OpArrayLength %b !3 !123"),
|
|
|
|
Eq(MakeInstruction(spv::OpArrayLength, {1, 2, 3, 123})));
|
|
|
|
EXPECT_THAT(CompiledInstructions("%a = OpArrayLength !2 !3 123"),
|
|
|
|
Eq(MakeInstruction(spv::OpArrayLength, {2, 1, 3, 123})));
|
|
|
|
EXPECT_THAT(CompiledInstructions("%a = OpArrayLength !2 !3 !123"),
|
|
|
|
Eq(MakeInstruction(spv::OpArrayLength, {2, 1, 3, 123})));
|
2015-09-11 19:03:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(ImmediateIntTest, ResultIdInAssignment) {
|
|
|
|
EXPECT_EQ("!2 not allowed before =.",
|
|
|
|
CompileFailure("!2 = OpArrayLength %12 %1 123"));
|
|
|
|
EXPECT_EQ("!2 not allowed before =.",
|
|
|
|
CompileFailure("!2 = !0x00040044 %12 %1 123"));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(ImmediateIntTest, OpCodeInAssignment) {
|
|
|
|
EXPECT_EQ("Invalid Opcode prefix '!0x00040044'.",
|
|
|
|
CompileFailure("%2 = !0x00040044 %12 %1 123"));
|
2015-09-02 18:48:33 +00:00
|
|
|
}
|
|
|
|
|
2015-09-08 15:41:36 +00:00
|
|
|
// Literal integers after !<integer> are handled correctly.
|
|
|
|
TEST_F(ImmediateIntTest, IntegerFollowingImmediate) {
|
2015-09-28 15:51:32 +00:00
|
|
|
const SpirvVector original = CompiledInstructions("OpTypeInt %1 8 1", kCAF);
|
|
|
|
EXPECT_EQ(original, CompiledInstructions("!0x00040015 1 8 1", kCAF));
|
2015-09-22 19:50:33 +00:00
|
|
|
EXPECT_EQ(original, CompiledInstructions("OpTypeInt !1 8 1", kCAF));
|
2015-09-08 15:41:36 +00:00
|
|
|
|
|
|
|
// 64-bit integer literal.
|
2015-09-29 15:28:34 +00:00
|
|
|
EXPECT_EQ(CompiledInstructions("OpTypeInt %i64 64 0\n"
|
|
|
|
"OpConstant %i64 %2 5000000000", kCAF),
|
|
|
|
CompiledInstructions("OpTypeInt %i64 64 0\n"
|
|
|
|
"OpConstant %i64 !2 5000000000", kCAF));
|
2015-09-08 15:41:36 +00:00
|
|
|
|
|
|
|
// Negative integer.
|
2015-09-29 15:28:34 +00:00
|
|
|
EXPECT_EQ(CompiledInstructions("OpTypeInt %i64 32 1\n"
|
|
|
|
"OpConstant %i64 %2 -123", kCAF),
|
|
|
|
CompiledInstructions("OpTypeInt %i64 32 1\n"
|
|
|
|
"OpConstant %i64 !2 -123", kCAF));
|
2015-09-08 18:20:12 +00:00
|
|
|
|
2015-09-29 21:07:21 +00:00
|
|
|
// TODO(deki): uncomment assertions below and make them pass.
|
2015-09-08 18:20:12 +00:00
|
|
|
// Hex value(s).
|
2015-09-12 00:10:54 +00:00
|
|
|
// EXPECT_EQ(CompileSuccessfully("OpConstant %10 %1 0x12345678", kCAF),
|
|
|
|
// CompileSuccessfully("OpConstant %10 !1 0x12345678", kCAF));
|
|
|
|
// EXPECT_EQ(
|
|
|
|
// CompileSuccessfully("OpConstant %10 %1 0x12345678 0x87654321", kCAF),
|
|
|
|
// CompileSuccessfully("OpConstant %10 !1 0x12345678 0x87654321", kCAF));
|
2015-09-08 15:41:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Literal floats after !<integer> are handled correctly.
|
|
|
|
TEST_F(ImmediateIntTest, FloatFollowingImmediate) {
|
2015-09-29 15:28:34 +00:00
|
|
|
EXPECT_EQ(CompiledInstructions("OpTypeMatrix %10 %2 0.123", kCAF),
|
|
|
|
CompiledInstructions("OpTypeMatrix %10 !2 0.123", kCAF));
|
|
|
|
EXPECT_EQ(CompiledInstructions("OpTypeMatrix %10 %2 -0.5", kCAF),
|
|
|
|
CompiledInstructions("OpTypeMatrix %10 !2 -0.5", kCAF));
|
2015-09-08 15:41:36 +00:00
|
|
|
// 64-bit float.
|
|
|
|
EXPECT_EQ(
|
2015-09-22 19:50:33 +00:00
|
|
|
CompiledInstructions(
|
2015-09-29 15:28:34 +00:00
|
|
|
"OpTypeMatrix %10 %2 9999999999999999999999999999999999999999.9", kCAF),
|
2015-09-22 19:50:33 +00:00
|
|
|
CompiledInstructions(
|
2015-09-29 15:28:34 +00:00
|
|
|
"OpTypeMatrix %10 !2 9999999999999999999999999999999999999999.9",
|
2015-09-12 00:10:54 +00:00
|
|
|
kCAF));
|
2015-09-08 15:41:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Literal strings after !<integer> are handled correctly.
|
|
|
|
TEST_F(ImmediateIntTest, StringFollowingImmediate) {
|
|
|
|
// Try a variety of strings, including empty and single-character.
|
2015-09-11 20:34:49 +00:00
|
|
|
for (std::string name : {"", "s", "longish", "really looooooooooooooooong"}) {
|
2015-09-08 16:14:07 +00:00
|
|
|
const SpirvVector original =
|
2015-09-22 19:50:33 +00:00
|
|
|
CompiledInstructions("OpMemberName %10 4 \"" + name + "\"", kCAF);
|
|
|
|
EXPECT_EQ(original, CompiledInstructions(
|
|
|
|
"OpMemberName %10 !4 \"" + name + "\"", kCAF))
|
2015-09-11 20:34:49 +00:00
|
|
|
<< name;
|
|
|
|
EXPECT_EQ(original,
|
2015-09-22 19:50:33 +00:00
|
|
|
CompiledInstructions("OpMemberName !1 !4 \"" + name + "\"", kCAF))
|
2015-09-11 20:34:49 +00:00
|
|
|
<< name;
|
|
|
|
const uint32_t wordCount = 4 + name.size() / 4;
|
|
|
|
const uint32_t firstWord = spvOpcodeMake(wordCount, spv::OpMemberName);
|
2015-09-22 19:50:33 +00:00
|
|
|
EXPECT_EQ(original, CompiledInstructions("!" + std::to_string(firstWord) +
|
|
|
|
" %10 !4 \"" + name + "\"",
|
|
|
|
kCAF))
|
2015-09-11 20:34:49 +00:00
|
|
|
<< name;
|
2015-09-08 15:41:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// IDs after !<integer> are handled correctly.
|
|
|
|
TEST_F(ImmediateIntTest, IdFollowingImmediate) {
|
2015-09-12 00:10:54 +00:00
|
|
|
EXPECT_EQ(CompileSuccessfully("OpDecorationGroup %123", kCAF),
|
|
|
|
CompileSuccessfully("!0x00020049 %123", kCAF));
|
|
|
|
EXPECT_EQ(CompileSuccessfully("OpDecorationGroup %group", kCAF),
|
|
|
|
CompileSuccessfully("!0x00020049 %group", kCAF));
|
2015-09-08 15:41:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// !<integer> after !<integer> is handled correctly.
|
|
|
|
TEST_F(ImmediateIntTest, ImmediateFollowingImmediate) {
|
2015-09-12 00:10:54 +00:00
|
|
|
const SpirvVector original =
|
2015-09-22 19:50:33 +00:00
|
|
|
CompiledInstructions("OpTypeMatrix %a %b 7", kCAF);
|
|
|
|
EXPECT_EQ(original, CompiledInstructions("OpTypeMatrix %a !2 !7", kCAF));
|
|
|
|
EXPECT_EQ(original, CompiledInstructions("!0x00040018 %a !2 !7", kCAF));
|
2015-09-08 15:41:36 +00:00
|
|
|
}
|
|
|
|
|
2015-09-02 18:48:33 +00:00
|
|
|
TEST_F(ImmediateIntTest, InvalidStatement) {
|
2015-09-10 18:00:00 +00:00
|
|
|
EXPECT_THAT(
|
2015-09-12 00:10:54 +00:00
|
|
|
Subvector(CompileSuccessfully("!4 !3 !2 !1", kCAF), kFirstInstruction),
|
2015-09-10 18:00:00 +00:00
|
|
|
ElementsAre(4, 3, 2, 1));
|
2015-09-02 18:48:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(ImmediateIntTest, InvalidStatementBetweenValidOnes) {
|
2015-09-12 00:10:54 +00:00
|
|
|
EXPECT_THAT(Subvector(CompileSuccessfully(
|
|
|
|
"OpTypeFloat %10 32 !5 !6 !7 OpEmitVertex", kCAF),
|
2015-09-10 18:00:00 +00:00
|
|
|
kFirstInstruction),
|
2015-09-22 19:50:33 +00:00
|
|
|
ElementsAre(spvOpcodeMake(3, spv::OpTypeFloat), 1, 32, 5, 6, 7,
|
2015-09-10 18:00:00 +00:00
|
|
|
spvOpcodeMake(1, spv::OpEmitVertex)));
|
2015-09-02 18:48:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(ImmediateIntTest, NextOpcodeRecognized) {
|
2015-09-12 00:10:54 +00:00
|
|
|
const SpirvVector original = CompileSuccessfully(R"(
|
2015-09-02 18:48:33 +00:00
|
|
|
OpLoad %10 %1 %2 Volatile
|
|
|
|
OpCompositeInsert %11 %4 %1 %3 0 1 2
|
2015-09-12 00:10:54 +00:00
|
|
|
)",
|
|
|
|
kCAF);
|
|
|
|
const SpirvVector alternate = CompileSuccessfully(R"(
|
2015-09-02 18:48:33 +00:00
|
|
|
OpLoad %10 %1 %2 !1
|
|
|
|
OpCompositeInsert %11 %4 %1 %3 0 1 2
|
2015-09-12 00:10:54 +00:00
|
|
|
)",
|
|
|
|
kCAF);
|
2015-09-02 18:48:33 +00:00
|
|
|
EXPECT_EQ(original, alternate);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(ImmediateIntTest, WrongLengthButNextOpcodeStillRecognized) {
|
2015-09-12 00:10:54 +00:00
|
|
|
const SpirvVector original = CompileSuccessfully(R"(
|
2015-09-02 18:48:33 +00:00
|
|
|
OpLoad %10 %1 %2 Volatile
|
|
|
|
OpCopyMemorySized %3 %4 %1
|
2015-09-12 00:10:54 +00:00
|
|
|
)",
|
|
|
|
kCAF);
|
|
|
|
const SpirvVector alternate = CompileSuccessfully(R"(
|
2015-09-02 18:48:33 +00:00
|
|
|
!0x0002003D %10 %1 %2 !1
|
|
|
|
OpCopyMemorySized %3 %4 %1
|
2015-09-12 00:10:54 +00:00
|
|
|
)",
|
|
|
|
kCAF);
|
2015-09-02 18:48:33 +00:00
|
|
|
EXPECT_EQ(0x0002003D, alternate[kFirstInstruction]);
|
|
|
|
EXPECT_EQ(Subvector(original, kFirstInstruction + 1),
|
|
|
|
Subvector(alternate, kFirstInstruction + 1));
|
|
|
|
}
|
|
|
|
|
2015-09-03 21:42:50 +00:00
|
|
|
// Like NextOpcodeRecognized, but next statement is in assignment form.
|
2015-09-11 20:34:49 +00:00
|
|
|
TEST_F(ImmediateIntTest, NextAssignmentRecognized) {
|
2015-09-08 16:14:07 +00:00
|
|
|
const SpirvVector original = CompileSuccessfully(R"(
|
2015-09-10 18:00:00 +00:00
|
|
|
%1 = OpLoad %10 %2 None
|
2015-09-22 19:50:33 +00:00
|
|
|
%4 = OpFunctionCall %10 %3 %123
|
2015-09-02 21:17:42 +00:00
|
|
|
)");
|
2015-09-08 16:14:07 +00:00
|
|
|
const SpirvVector alternate = CompileSuccessfully(R"(
|
2015-09-11 20:34:49 +00:00
|
|
|
%1 = OpLoad %10 %2 !0
|
2015-09-22 19:50:33 +00:00
|
|
|
%4 = OpFunctionCall %10 %3 %123
|
2015-09-02 21:17:42 +00:00
|
|
|
)");
|
|
|
|
EXPECT_EQ(original, alternate);
|
2015-09-02 18:48:33 +00:00
|
|
|
}
|
|
|
|
|
2015-09-03 21:42:50 +00:00
|
|
|
// Two instructions in a row each have !<integer> opcode.
|
2015-09-02 18:48:33 +00:00
|
|
|
TEST_F(ImmediateIntTest, ConsecutiveImmediateOpcodes) {
|
2015-09-08 16:14:07 +00:00
|
|
|
const SpirvVector original = CompileSuccessfully(R"(
|
2015-09-10 18:00:00 +00:00
|
|
|
%1 = OpConstantSampler %10 Clamp 78 Linear
|
|
|
|
%4 = OpFRem %11 %3 %2
|
2015-09-03 21:42:50 +00:00
|
|
|
%5 = OpIsValidEvent %12 %2
|
|
|
|
)");
|
2015-09-08 16:14:07 +00:00
|
|
|
const SpirvVector alternate = CompileSuccessfully(R"(
|
2015-09-03 21:42:50 +00:00
|
|
|
!0x0006002D %10 %1 !2 78 !1
|
|
|
|
!0x0005008C %11 %4 %3 %2
|
|
|
|
%5 = OpIsValidEvent %12 %2
|
|
|
|
)");
|
|
|
|
EXPECT_EQ(original, alternate);
|
2015-09-02 18:48:33 +00:00
|
|
|
}
|
|
|
|
|
2015-09-08 18:17:14 +00:00
|
|
|
// !<integer> followed by, eg, an enum or '=' or a random bareword.
|
2015-09-02 18:48:33 +00:00
|
|
|
TEST_F(ImmediateIntTest, ForbiddenOperands) {
|
2015-09-08 18:17:14 +00:00
|
|
|
EXPECT_THAT(CompileFailure("OpMemoryModel !0 OpenCL"), HasSubstr("OpenCL"));
|
|
|
|
EXPECT_THAT(CompileFailure("!1 %0 = !2"), HasSubstr("="));
|
2015-09-30 13:49:00 +00:00
|
|
|
EXPECT_THAT(CompileFailure("OpMemoryModel !0 random_bareword"),
|
|
|
|
HasSubstr("random_bareword"));
|
2015-09-08 18:17:14 +00:00
|
|
|
// Immediate integers longer than one 32-bit word.
|
|
|
|
EXPECT_THAT(CompileFailure("!5000000000"), HasSubstr("5000000000"));
|
2015-09-29 21:07:21 +00:00
|
|
|
EXPECT_THAT(CompileFailure("!999999999999999999"),
|
|
|
|
HasSubstr("999999999999999999"));
|
|
|
|
EXPECT_THAT(CompileFailure("!0x00020049 !5000000000"),
|
|
|
|
HasSubstr("5000000000"));
|
2015-09-30 13:49:00 +00:00
|
|
|
// Negative numbers.
|
|
|
|
EXPECT_THAT(CompileFailure("!0x00020049 !-123"), HasSubstr("-123"));
|
2015-09-02 18:48:33 +00:00
|
|
|
}
|
|
|
|
|
2015-09-11 04:43:11 +00:00
|
|
|
TEST_F(ImmediateIntTest, NotInteger) {
|
|
|
|
EXPECT_THAT(CompileFailure("!abc"),
|
|
|
|
StrEq("Invalid immediate integer '!abc'."));
|
|
|
|
EXPECT_THAT(CompileFailure("!12.3"),
|
|
|
|
StrEq("Invalid immediate integer '!12.3'."));
|
|
|
|
EXPECT_THAT(CompileFailure("!12K"),
|
|
|
|
StrEq("Invalid immediate integer '!12K'."));
|
|
|
|
}
|
|
|
|
|
2015-08-31 21:42:22 +00:00
|
|
|
} // anonymous namespace
|