2022-05-03 08:21:01 -04:00
|
|
|
#include <qpdf/assert_test.h>
|
|
|
|
|
2008-04-29 12:55:25 +00:00
|
|
|
#include <qpdf/Pl_Buffer.hh>
|
|
|
|
#include <qpdf/Pl_Count.hh>
|
|
|
|
#include <qpdf/Pl_Discard.hh>
|
2013-02-23 21:46:21 -05:00
|
|
|
#include <qpdf/QUtil.hh>
|
2023-05-20 12:22:32 +01:00
|
|
|
#include <cstdlib>
|
2021-12-17 07:33:42 -05:00
|
|
|
#include <cstring>
|
2011-08-11 11:56:37 -04:00
|
|
|
#include <iostream>
|
2010-10-01 11:19:36 +00:00
|
|
|
#include <stdexcept>
|
2008-04-29 12:55:25 +00:00
|
|
|
|
2013-02-23 21:46:21 -05:00
|
|
|
static unsigned char*
|
|
|
|
uc(char const* s)
|
|
|
|
{
|
|
|
|
return QUtil::unsigned_char_pointer(s);
|
|
|
|
}
|
2008-04-29 12:55:25 +00:00
|
|
|
|
|
|
|
int
|
|
|
|
main()
|
|
|
|
{
|
2019-06-21 21:32:47 -04:00
|
|
|
{
|
2023-05-25 14:10:47 +01:00
|
|
|
// Test that buffers can be copied by value using Buffer::copy.
|
|
|
|
Buffer bc1(2);
|
|
|
|
unsigned char* bc1p = bc1.getBuffer();
|
|
|
|
bc1p[0] = 'Q';
|
|
|
|
bc1p[1] = 'W';
|
|
|
|
Buffer bc2(bc1.copy());
|
|
|
|
bc1p[0] = 'R';
|
|
|
|
unsigned char* bc2p = bc2.getBuffer();
|
|
|
|
assert(bc2p != bc1p);
|
|
|
|
assert(bc2p[0] == 'Q');
|
|
|
|
assert(bc2p[1] == 'W');
|
|
|
|
bc2 = bc1.copy();
|
|
|
|
bc2p = bc2.getBuffer();
|
|
|
|
assert(bc2p != bc1p);
|
|
|
|
assert(bc2p[0] == 'R');
|
|
|
|
assert(bc2p[1] == 'W');
|
2023-11-17 18:06:57 +00:00
|
|
|
|
|
|
|
// Test Buffer(std:string&&)
|
|
|
|
Buffer bc3("QW");
|
|
|
|
unsigned char* bc3p = bc3.getBuffer();
|
|
|
|
Buffer bc4(bc3.copy());
|
|
|
|
bc3p[0] = 'R';
|
|
|
|
unsigned char* bc4p = bc4.getBuffer();
|
|
|
|
assert(bc4p != bc3p);
|
|
|
|
assert(bc4p[0] == 'Q');
|
|
|
|
assert(bc4p[1] == 'W');
|
|
|
|
bc4 = bc3.copy();
|
|
|
|
bc4p = bc4.getBuffer();
|
|
|
|
assert(bc4p != bc3p);
|
|
|
|
assert(bc4p[0] == 'R');
|
|
|
|
assert(bc4p[1] == 'W');
|
2023-05-25 14:10:47 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
# pragma warning(disable : 4996)
|
|
|
|
#endif
|
|
|
|
#if (defined(__GNUC__) || defined(__clang__))
|
|
|
|
# pragma GCC diagnostic push
|
|
|
|
# pragma GCC diagnostic ignored "-Wdeprecated-declarations"
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
// Test that buffers can be copied by value using copy construction / assignment.
|
|
|
|
Buffer::setTestMode();
|
2019-06-21 21:32:47 -04:00
|
|
|
Buffer bc1(2);
|
|
|
|
unsigned char* bc1p = bc1.getBuffer();
|
|
|
|
bc1p[0] = 'Q';
|
|
|
|
bc1p[1] = 'W';
|
|
|
|
Buffer bc2(bc1);
|
|
|
|
bc1p[0] = 'R';
|
|
|
|
unsigned char* bc2p = bc2.getBuffer();
|
|
|
|
assert(bc2p != bc1p);
|
|
|
|
assert(bc2p[0] == 'Q');
|
|
|
|
assert(bc2p[1] == 'W');
|
|
|
|
bc2 = bc1;
|
|
|
|
bc2p = bc2.getBuffer();
|
|
|
|
assert(bc2p != bc1p);
|
|
|
|
assert(bc2p[0] == 'R');
|
|
|
|
assert(bc2p[1] == 'W');
|
2023-11-17 18:06:57 +00:00
|
|
|
|
|
|
|
// Test Buffer(std:string&&)
|
|
|
|
Buffer bc3("QW");
|
|
|
|
unsigned char* bc3p = bc3.getBuffer();
|
|
|
|
Buffer bc4(bc3);
|
|
|
|
bc3p[0] = 'R';
|
|
|
|
unsigned char* bc4p = bc4.getBuffer();
|
|
|
|
assert(bc4p != bc3p);
|
|
|
|
assert(bc4p[0] == 'Q');
|
|
|
|
assert(bc4p[1] == 'W');
|
|
|
|
bc4 = bc3;
|
|
|
|
bc4p = bc4.getBuffer();
|
|
|
|
assert(bc4p != bc3p);
|
|
|
|
assert(bc2p[0] == 'R');
|
|
|
|
assert(bc2p[1] == 'W');
|
|
|
|
|
|
|
|
// Test Buffer(std:string&)
|
|
|
|
std::string s{"QW"};
|
|
|
|
Buffer bc5(s);
|
|
|
|
unsigned char* bc5p = bc5.getBuffer();
|
|
|
|
Buffer bc6(bc5);
|
|
|
|
bc5p[0] = 'R';
|
|
|
|
unsigned char* bc6p = bc6.getBuffer();
|
|
|
|
assert(bc6p != bc5p);
|
|
|
|
assert(bc6p[0] == 'Q');
|
|
|
|
assert(bc6p[1] == 'W');
|
|
|
|
bc6 = bc5;
|
|
|
|
bc6p = bc6.getBuffer();
|
|
|
|
assert(bc6p != bc5p);
|
|
|
|
assert(bc2p[0] == 'R');
|
|
|
|
assert(bc2p[1] == 'W');
|
2019-06-21 21:32:47 -04:00
|
|
|
}
|
2023-05-25 14:10:47 +01:00
|
|
|
#if (defined(__GNUC__) || defined(__clang__))
|
|
|
|
# pragma GCC diagnostic pop
|
|
|
|
#endif
|
2019-06-21 21:32:47 -04:00
|
|
|
|
2022-09-04 14:48:25 -07:00
|
|
|
{
|
|
|
|
// Test that buffers can be moved.
|
|
|
|
Buffer bm1(2);
|
|
|
|
unsigned char* bm1p = bm1.getBuffer();
|
|
|
|
bm1p[0] = 'Q';
|
|
|
|
bm1p[1] = 'W';
|
|
|
|
Buffer bm2(std::move(bm1));
|
|
|
|
bm1p[0] = 'R';
|
|
|
|
unsigned char* bm2p = bm2.getBuffer();
|
|
|
|
assert(bm2p == bm1p);
|
|
|
|
assert(bm2p[0] == 'R');
|
|
|
|
|
|
|
|
Buffer bm3 = std::move(bm2);
|
|
|
|
unsigned char* bm3p = bm3.getBuffer();
|
|
|
|
assert(bm3p == bm2p);
|
2023-11-17 18:06:57 +00:00
|
|
|
|
|
|
|
// Test Buffer(dtd::string&&)
|
|
|
|
Buffer bm4("QW");
|
|
|
|
unsigned char* bm4p = bm4.getBuffer();
|
|
|
|
Buffer bm5(std::move(bm4));
|
|
|
|
bm4p[0] = 'R';
|
|
|
|
unsigned char* bm5p = bm5.getBuffer();
|
|
|
|
assert(bm5p == bm4p);
|
|
|
|
assert(bm5p[0] == 'R');
|
|
|
|
|
|
|
|
Buffer bm6 = std::move(bm5);
|
|
|
|
unsigned char* bm6p = bm6.getBuffer();
|
|
|
|
assert(bm6p == bm5p);
|
2022-09-04 14:48:25 -07:00
|
|
|
}
|
|
|
|
|
2008-04-29 12:55:25 +00:00
|
|
|
try {
|
2022-02-08 09:18:08 -05:00
|
|
|
Pl_Discard discard;
|
|
|
|
Pl_Count count("count", &discard);
|
|
|
|
Pl_Buffer bp1("bp1", &count);
|
|
|
|
bp1.write(uc("12345"), 5);
|
|
|
|
bp1.write(uc("67890"), 5);
|
|
|
|
bp1.finish();
|
|
|
|
std::cout << "count: " << count.getCount() << std::endl;
|
|
|
|
bp1.write(uc("abcde"), 5);
|
|
|
|
bp1.write(uc("fghij"), 6);
|
|
|
|
bp1.finish();
|
|
|
|
std::cout << "count: " << count.getCount() << std::endl;
|
|
|
|
Buffer* b = bp1.getBuffer();
|
|
|
|
std::cout << "size: " << b->getSize() << std::endl;
|
|
|
|
std::cout << "data: " << b->getBuffer() << std::endl;
|
|
|
|
delete b;
|
|
|
|
bp1.write(uc("qwert"), 5);
|
|
|
|
bp1.write(uc("yuiop"), 6);
|
|
|
|
bp1.finish();
|
|
|
|
std::cout << "count: " << count.getCount() << std::endl;
|
|
|
|
b = bp1.getBuffer();
|
|
|
|
std::cout << "size: " << b->getSize() << std::endl;
|
|
|
|
std::cout << "data: " << b->getBuffer() << std::endl;
|
|
|
|
delete b;
|
2008-04-29 12:55:25 +00:00
|
|
|
|
2022-02-08 09:18:08 -05:00
|
|
|
Pl_Buffer bp2("bp2");
|
|
|
|
bp2.write(uc("moo"), 3);
|
|
|
|
bp2.write(uc("quack"), 6);
|
|
|
|
try {
|
|
|
|
delete bp2.getBuffer();
|
|
|
|
} catch (std::exception& e) {
|
|
|
|
std::cout << e.what() << std::endl;
|
|
|
|
}
|
|
|
|
bp2.finish();
|
|
|
|
b = bp2.getBuffer();
|
|
|
|
std::cout << "size: " << b->getSize() << std::endl;
|
|
|
|
std::cout << "data: " << b->getBuffer() << std::endl;
|
|
|
|
delete b;
|
2010-10-01 11:19:36 +00:00
|
|
|
|
2022-02-08 09:18:08 -05:00
|
|
|
unsigned char lbuf[10];
|
|
|
|
Buffer b1(lbuf, 10);
|
|
|
|
if (!((b1.getBuffer() == lbuf) && (b1.getSize() == 10))) {
|
|
|
|
throw std::logic_error("hand-created buffer is not as expected");
|
|
|
|
}
|
2018-03-06 11:24:30 -05:00
|
|
|
|
|
|
|
Pl_Buffer bp3("bp3");
|
|
|
|
b = bp3.getBuffer();
|
|
|
|
std::cout << "size: " << b->getSize() << std::endl;
|
2018-05-05 17:43:44 -04:00
|
|
|
delete b;
|
2019-06-21 21:32:47 -04:00
|
|
|
// Should be able to call getBuffer again and get an empty buffer
|
|
|
|
b = bp3.getBuffer();
|
|
|
|
std::cout << "size: " << b->getSize() << std::endl;
|
|
|
|
delete b;
|
|
|
|
// Also can write 0 and do it.
|
|
|
|
bp3.write(uc(""), 0);
|
|
|
|
bp3.finish();
|
|
|
|
b = bp3.getBuffer();
|
|
|
|
std::cout << "size: " << b->getSize() << std::endl;
|
|
|
|
delete b;
|
2021-12-17 07:33:42 -05:00
|
|
|
|
|
|
|
// Malloc buffer should behave similarly.
|
|
|
|
Pl_Buffer bp4("bp4");
|
|
|
|
bp4.write(uc("asdf"), 4);
|
|
|
|
unsigned char* mbuf;
|
|
|
|
size_t len;
|
|
|
|
try {
|
|
|
|
bp4.getMallocBuffer(&mbuf, &len);
|
|
|
|
assert(false);
|
|
|
|
} catch (std::logic_error& e) {
|
|
|
|
std::cout << "malloc buffer logic error: " << e.what() << std::endl;
|
|
|
|
}
|
|
|
|
bp4.finish();
|
|
|
|
bp4.getMallocBuffer(&mbuf, &len);
|
|
|
|
assert(len == 4);
|
|
|
|
assert(memcmp(mbuf, uc("asdf"), 4) == 0);
|
|
|
|
free(mbuf);
|
|
|
|
bp4.write(uc(""), 0);
|
|
|
|
bp4.finish();
|
|
|
|
bp4.getMallocBuffer(&mbuf, &len);
|
|
|
|
assert(mbuf == nullptr);
|
|
|
|
assert(len == 0);
|
2008-04-29 12:55:25 +00:00
|
|
|
} catch (std::exception& e) {
|
2022-02-08 09:18:08 -05:00
|
|
|
std::cout << "unexpected exception: " << e.what() << std::endl;
|
|
|
|
exit(2);
|
2008-04-29 12:55:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::cout << "done" << std::endl;
|
|
|
|
return 0;
|
|
|
|
}
|