208 lines
5.7 KiB
C++
208 lines
5.7 KiB
C++
/**
|
|
* Copyright - See the COPYRIGHT that is included with this distribution.
|
|
* pvxs is distributed subject to a Software License Agreement found
|
|
* in file LICENSE that is included with this distribution.
|
|
*/
|
|
|
|
#include <testMain.h>
|
|
|
|
#include <epicsUnitTest.h>
|
|
|
|
#include <pvxs/unittest.h>
|
|
#include <pvxs/data.h>
|
|
#include <pvxs/nt.h>
|
|
#include "utilpvt.h"
|
|
#include "pvaproto.h"
|
|
#include "dataimpl.h"
|
|
|
|
using namespace pvxs;
|
|
namespace {
|
|
|
|
template<size_t N>
|
|
testCase
|
|
testBytes(const std::vector<uint8_t>& actual, const char(&buf)[N])
|
|
{
|
|
bool ok = actual.size()==(N-1) && std::equal(actual.begin(),
|
|
actual.end(),
|
|
(const uint8_t*)buf);
|
|
|
|
testCase ret(ok);
|
|
ret<<"Expect: \""<<escape(std::string((const char*)buf, N-1))<<"\"\n"
|
|
<<"Actual: \""<<escape(std::string((const char*)actual.data(), actual.size()))<<"\"\n";
|
|
return ret;
|
|
}
|
|
|
|
template<typename Fn, size_t N>
|
|
void testToBytes(bool be, Fn&& fn, const char(&expect)[N])
|
|
{
|
|
std::vector<uint8_t> buf;
|
|
VectorOutBuf S(be, buf);
|
|
fn(S);
|
|
buf.resize(buf.size()-S.size());
|
|
testBytes(buf, expect);
|
|
}
|
|
|
|
void testSerialize1()
|
|
{
|
|
testDiag("%s", __func__);
|
|
|
|
auto val = nt::NTScalar{TypeCode::UInt32}.build().create();
|
|
|
|
testToBytes(true, [&val](Buffer& buf) {
|
|
to_wire_full(buf, val);
|
|
}, "\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00");
|
|
|
|
testToBytes(true, [&val](Buffer& buf) {
|
|
to_wire_valid(buf, val);
|
|
}, "\x00");
|
|
|
|
val["value"] = 0xdeadbeef;
|
|
|
|
testToBytes(true, [&val](Buffer& buf) {
|
|
to_wire_valid(buf, val);
|
|
}, "\x01\x02\xde\xad\xbe\xef");
|
|
|
|
val["value"].unmark();
|
|
|
|
testToBytes(true, [&val](Buffer& buf) {
|
|
to_wire_valid(buf, val);
|
|
}, "\x00");
|
|
|
|
val["timeStamp.nanoseconds"] = 0xab;
|
|
val["alarm.message"] = "hello world";
|
|
|
|
testToBytes(true, [&val](Buffer& buf) {
|
|
to_wire_valid(buf, val);
|
|
}, "\x02 \x01\x0bhello world\x00\x00\x00\xab");
|
|
}
|
|
|
|
void testSerialize2()
|
|
{
|
|
testDiag("%s", __func__);
|
|
|
|
TypeDef def(TypeCode::Struct, "simple_t", {
|
|
Member(TypeCode::Float64A, "value"),
|
|
Member(TypeCode::Struct, "timeStamp", "time_t", {
|
|
Member(TypeCode::UInt64, "secondsPastEpoch"),
|
|
Member(TypeCode::UInt32, "nanoseconds"),
|
|
}),
|
|
Member(TypeCode::Struct, "arbitrary", {
|
|
Member(TypeCode::StructA, "sarr", {
|
|
Member(TypeCode::UInt32, "value"),
|
|
}),
|
|
}),
|
|
Member(TypeCode::Any, "any"),
|
|
Member(TypeCode::AnyA, "anya"),
|
|
Member(TypeCode::Union, "choice", {
|
|
Member(TypeCode::Float32, "a"),
|
|
Member(TypeCode::String, "b"),
|
|
}),
|
|
Member(TypeCode::UnionA, "achoice", {
|
|
Member(TypeCode::String, "x"),
|
|
Member(TypeCode::String, "y"),
|
|
}),
|
|
});
|
|
|
|
{
|
|
auto val = def.create();
|
|
|
|
auto fld = val["arbitrary.sarr"];
|
|
shared_array<Value> arr(3);
|
|
arr[0] = fld.allocMember();
|
|
arr[1] = fld.allocMember();
|
|
// leave [2] as null
|
|
arr[0]["value"] = 0xdeadbeef;
|
|
arr[1]["value"] = 0x1badface;
|
|
|
|
fld.from(shared_array_static_cast<const void>(freeze(std::move(arr))));
|
|
|
|
testToBytes(true, [&val](Buffer& buf) {
|
|
to_wire_valid(buf, val);
|
|
}, "\x01@\x03\x01\xde\xad\xbe\xef\x01\x1b\xad\xfa\xce\x00");
|
|
}
|
|
|
|
{
|
|
auto val = def.create();
|
|
|
|
val["choice->b"] = "test";
|
|
val["choice"].mark(); // TODO: auto mark back through union
|
|
|
|
testToBytes(true, [&val](Buffer& buf) {
|
|
to_wire_valid(buf, val);
|
|
}, "\x02\x00\x02\x01\x04test");
|
|
}
|
|
|
|
{
|
|
auto val = def.create();
|
|
|
|
auto fld = val["achoice"];
|
|
shared_array<Value> arr(3);
|
|
arr[0] = fld.allocMember();
|
|
arr[1] = fld.allocMember();
|
|
// leave [2] as null
|
|
arr[0]["->x"] = "theX";
|
|
arr[1]["->y"] = "theY";
|
|
|
|
fld.from(shared_array_static_cast<const void>(freeze(std::move(arr))));
|
|
|
|
testToBytes(true, [&val](Buffer& buf) {
|
|
to_wire_valid(buf, val);
|
|
}, "\x02\x00\x04\x03\x01\x00\x04theX\x01\x01\x04theY\x00");
|
|
}
|
|
|
|
// Any
|
|
{
|
|
auto val = def.create();
|
|
|
|
auto v = TypeDef(TypeCode::UInt32).create();
|
|
v = 0x600df00d;
|
|
|
|
val["any"].from(v);
|
|
|
|
testToBytes(true, [&val](Buffer& buf) {
|
|
to_wire_valid(buf, val);
|
|
}, "\x01\x80\x26\x60\x0d\xf0\x0d");
|
|
}
|
|
|
|
// Any
|
|
{
|
|
auto val = def.create();
|
|
val["any"].mark();
|
|
|
|
testToBytes(true, [&val](Buffer& buf) {
|
|
to_wire_valid(buf, val);
|
|
}, "\x01\x80\xff");
|
|
}
|
|
|
|
// Any[]
|
|
{
|
|
auto val = def.create();
|
|
|
|
auto fld = val["anya"];
|
|
shared_array<Value> arr(3);
|
|
arr[0] = TypeDef(TypeCode::UInt32).create();
|
|
arr[1] = TypeDef(TypeCode::Struct, {Member(TypeCode::String, "q")}).create();
|
|
// leave [2] as null
|
|
|
|
arr[0] = 0x7b;
|
|
arr[1]["q"] = "theq";
|
|
|
|
fld.from(shared_array_static_cast<const void>(freeze(std::move(arr))));
|
|
|
|
testToBytes(true, [&val](Buffer& buf) {
|
|
to_wire_valid(buf, val);
|
|
}, "\x02\x00\x01\x03\x01\x26\x00\x00\x00\x7b\x01\x80\x00\x01\x01q\x60\x04theq\x00");
|
|
}
|
|
}
|
|
|
|
} // namespace
|
|
|
|
MAIN(testdata)
|
|
{
|
|
testPlan(11);
|
|
testSerialize1();
|
|
testSerialize2();
|
|
cleanup_for_valgrind();
|
|
return testDone();
|
|
}
|