summaryrefslogtreecommitdiffstats
path: root/src/boost/libs/endian/example
diff options
context:
space:
mode:
Diffstat (limited to 'src/boost/libs/endian/example')
-rw-r--r--src/boost/libs/endian/example/conversion_use_case.cpp51
-rw-r--r--src/boost/libs/endian/example/endian_example.cpp75
-rw-r--r--src/boost/libs/endian/example/third_party_format.hpp19
-rw-r--r--src/boost/libs/endian/example/udt_conversion_example.cpp79
-rw-r--r--src/boost/libs/endian/example/use_cases.cpp152
5 files changed, 376 insertions, 0 deletions
diff --git a/src/boost/libs/endian/example/conversion_use_case.cpp b/src/boost/libs/endian/example/conversion_use_case.cpp
new file mode 100644
index 00000000..544b0c0b
--- /dev/null
+++ b/src/boost/libs/endian/example/conversion_use_case.cpp
@@ -0,0 +1,51 @@
+// endian/example/conversion_use_case.cpp
+
+// Copyright Beman Dawes 2014
+
+// Distributed under the Boost Software License, Version 1.0.
+// See http://www.boost.org/LICENSE_1_0.txt
+
+// This program reads a binary file of fixed length records, with a format defined
+// in a header file supplied by a third-party. The records inserted into a vector,
+// and the vector is sorted. The sorted records are then written to an output file.
+
+// Full I/O error testing omitted for brevity, So don't try this at home.
+
+#include "third_party_format.hpp"
+#include <boost/endian/conversion.hpp>
+#include <vector>
+#include <fstream>
+#include <algorithm>
+#include <iostream>
+
+using third_party::record;
+
+int main()
+{
+ std::ifstream in("data.bin", std::ios::binary);
+ if (!in) { std::cout << "Could not open data.bin\n"; return 1; }
+
+ std::ofstream out("sorted-data.bin", std::ios::binary);
+ if (!out) { std::cout << "Could not open sorted-data.bin\n"; return 1; }
+
+ record rec;
+ std::vector<record> recs;
+
+ while (!in.eof()) // read each record
+ {
+ in.read((char*)&rec, sizeof(rec));
+ rec.balance = boost::endian::big_to_native(rec.balance); // reverse if needed
+ recs.push_back(rec);
+ }
+
+ std::sort(recs.begin(), recs.end(), // decending sort by balance
+ [](const record& lhs, const record& rhs) -> bool
+ { return lhs.balance > rhs.balance; });
+
+ for (auto &out_rec : recs) // write each record
+ {
+ out_rec.balance = boost::endian::native_to_big(out_rec.balance); // reverse if needed
+ out.write((const char*)&out_rec, sizeof(out_rec));
+ }
+
+}
diff --git a/src/boost/libs/endian/example/endian_example.cpp b/src/boost/libs/endian/example/endian_example.cpp
new file mode 100644
index 00000000..fcab6580
--- /dev/null
+++ b/src/boost/libs/endian/example/endian_example.cpp
@@ -0,0 +1,75 @@
+// endian_example.cpp -------------------------------------------------------//
+
+// Copyright Beman Dawes, 2006
+
+// Distributed under the Boost Software License, Version 1.0.
+// See http://www.boost.org/LICENSE_1_0.txt
+
+// See library home page at http://www.boost.org/libs/endian
+
+//----------------------------------------------------------------------------//
+
+#include <boost/endian/detail/disable_warnings.hpp>
+
+#include <iostream>
+#include <cstdio>
+#include <boost/endian/buffers.hpp>
+#include <boost/static_assert.hpp>
+
+using namespace boost::endian;
+
+namespace
+{
+ // This is an extract from a very widely used GIS file format. Why the designer
+ // decided to mix big and little endians in the same file is not known. But
+ // this is a real-world format and users wishing to write low level code
+ // manipulating these files have to deal with the mixed endianness.
+
+ struct header
+ {
+ big_int32_buf_at file_code;
+ big_int32_buf_at file_length;
+ little_int32_buf_at version;
+ little_int32_buf_at shape_type;
+ };
+
+ const char* filename = "test.dat";
+}
+
+int main(int, char* [])
+{
+ header h;
+
+ BOOST_STATIC_ASSERT(sizeof(h) == 16U); // reality check
+
+ h.file_code = 0x01020304;
+ h.file_length = sizeof(header);
+ h.version = 1;
+ h.shape_type = 0x01020304;
+
+ // Low-level I/O such as POSIX read/write or <cstdio> fread/fwrite is sometimes
+ // used for binary file operations when ultimate efficiency is important.
+ // Such I/O is often performed in some C++ wrapper class, but to drive home the
+ // point that endian integers are often used in fairly low-level code that
+ // does bulk I/O operations, <cstdio> fopen/fwrite is used for I/O in this example.
+
+ std::FILE* fi = std::fopen(filename, "wb"); // MUST BE BINARY
+
+ if (!fi)
+ {
+ std::cout << "could not open " << filename << '\n';
+ return 1;
+ }
+
+ if (std::fwrite(&h, sizeof(header), 1, fi)!= 1)
+ {
+ std::cout << "write failure for " << filename << '\n';
+ return 1;
+ }
+
+ std::fclose(fi);
+
+ std::cout << "created file " << filename << '\n';
+
+ return 0;
+}
diff --git a/src/boost/libs/endian/example/third_party_format.hpp b/src/boost/libs/endian/example/third_party_format.hpp
new file mode 100644
index 00000000..d6472837
--- /dev/null
+++ b/src/boost/libs/endian/example/third_party_format.hpp
@@ -0,0 +1,19 @@
+// endian/example/third_party_format.hpp
+
+// Copyright Beman Dawes 2014
+
+// Distributed under the Boost Software License, Version 1.0.
+// See http://www.boost.org/LICENSE_1_0.txt
+
+#include <cstdint>
+
+namespace third_party
+{
+ struct record
+ {
+ uint32_t id; // big endian
+ int32_t balance; // big endian
+
+ // ... data members whose endianness is not a concern
+ };
+}
diff --git a/src/boost/libs/endian/example/udt_conversion_example.cpp b/src/boost/libs/endian/example/udt_conversion_example.cpp
new file mode 100644
index 00000000..b0a6d088
--- /dev/null
+++ b/src/boost/libs/endian/example/udt_conversion_example.cpp
@@ -0,0 +1,79 @@
+// udt_conversion_example.cpp --------------------------------------------------------//
+
+// Copyright Beman Dawes 2013
+
+// Distributed under the Boost Software License, Version 1.0.
+// http://www.boost.org/LICENSE_1_0.txt
+
+//--------------------------------------------------------------------------------------//
+
+#include <boost/endian/detail/disable_warnings.hpp>
+
+#include <boost/endian/conversion.hpp>
+#include <iostream>
+#include <cstring>
+
+using namespace boost::endian;
+using std::cout;
+using std::endl;
+using boost::int32_t;
+using boost::int64_t;
+
+namespace user
+{
+ class UDT
+ {
+ public:
+ UDT() : id_(0), value_(0) {desc_[0] = '\0';}
+ UDT(int32_t id, int64_t value, const char* desc) : id_(id), value_(value)
+ {
+ std::strncpy(desc_, desc, sizeof(desc_)-1);
+ desc_[sizeof(desc_)-1] = '\0';
+ }
+ int32_t id() const {return id_;}
+ int64_t value() const {return value_;}
+ const char* desc() const {return desc_;}
+ void id(int32_t x) {id_ = x;}
+ void value(int64_t v) {value_ = v;}
+ void desc(const char* s)
+ {
+ std::strncpy(desc_, s, sizeof(desc_)-1);
+ desc_[sizeof(desc_-1)] = '\0';
+ }
+
+ friend void endian_reverse_inplace(UDT&);
+
+ private:
+ int32_t id_;
+ int64_t value_;
+ char desc_[56]; // '/0'
+ };
+
+ void endian_reverse_inplace(UDT& x)
+ {
+ boost::endian::endian_reverse_inplace(x.id_);
+ boost::endian::endian_reverse_inplace(x.value_);
+ }
+}
+
+int main(int, char* [])
+{
+ user::UDT x(1, 123456789012345LL, "Bingo!");
+
+ //cout << std::hex;
+ cout << "(1) " << x.id() << ' ' << x.value() << ' ' << x.desc() << endl;
+
+ user::endian_reverse_inplace(x);
+ cout << "(2) " << x.id() << ' ' << x.value() << ' ' << x.desc() << endl;
+
+ endian_reverse_inplace(x);
+ cout << "(3) " << x.id() << ' ' << x.value() << ' ' << x.desc() << endl;
+
+ conditional_reverse_inplace<order::little, order::big>(x);
+ cout << "(4) " << x.id() << ' ' << x.value() << ' ' << x.desc() << endl;
+
+ conditional_reverse_inplace(x, order::big, order::little);
+ cout << "(5) " << x.id() << ' ' << x.value() << ' ' << x.desc() << endl;
+}
+
+#include <boost/endian/detail/disable_warnings_pop.hpp>
diff --git a/src/boost/libs/endian/example/use_cases.cpp b/src/boost/libs/endian/example/use_cases.cpp
new file mode 100644
index 00000000..7bbe9b28
--- /dev/null
+++ b/src/boost/libs/endian/example/use_cases.cpp
@@ -0,0 +1,152 @@
+// endian/example/uses_cases.cpp -----------------------------------------------------//
+
+// Copyright Beman Dawes 2014
+
+// Distributed under the Boost Software License, Version 1.0.
+// See http://www.boost.org/LICENSE_1_0.txt
+
+//--------------------------------------------------------------------------------------//
+
+#ifndef _SCL_SECURE_NO_WARNINGS
+# define _SCL_SECURE_NO_WARNINGS
+#endif
+
+#ifndef _CRT_SECURE_NO_WARNINGS
+# define _CRT_SECURE_NO_WARNINGS
+#endif
+
+
+#include <boost/endian/conversion.hpp>
+#include <boost/endian/buffers.hpp>
+#include <boost/endian/arithmetic.hpp>
+#include <iostream>
+
+using namespace boost::endian;
+
+using std::cout;
+using std::endl;
+
+
+
+
+ { // Use case 2 - Endian buffer types
+
+ struct Record
+ {
+ big_ubuf32_t count; // big endian
+ big_buf32_t value; // big endian
+ };
+
+ Record rec;
+
+ read(&rec, sizeof(Record));
+
+ uint32_t count = rec.count.value();
+ int32_t value = rec.value.value();
+
+ ++count;
+ value += fee;
+
+ rec.count = count;
+ rec.value = value;
+
+ write(&rec, sizeof(Record));
+ }
+
+ { // Use case 3a - Endian arithmetic types
+
+ struct Record
+ {
+ big_uint32_t count; // big endian
+ big_int32_t value; // big endian
+ };
+
+ Record rec;
+
+ read(&rec, sizeof(Record));
+
+ ++rec.count;
+ rec.value += fee;
+
+ write(&rec, sizeof(Record));
+ }
+
+ { // Use case 3b - Endian arithmetic types
+
+ struct Record
+ {
+ big_uint32_t count; // big endian
+ big_int32_t value; // big endian
+ };
+
+ Record rec;
+
+ read(&rec, sizeof(Record));
+
+ uint32_t count = rec.count;
+ int32_t value = rec.value;
+
+ ++count;
+ value += fee;
+
+ rec.count = count;
+ rec.value = value;
+
+ write(&rec, sizeof(Record));
+ }
+
+ // Recommended approach when conversion time is not a concern
+ //
+ // Conversion time is not a concert with this application because the minimum
+ // possible number of conversions is performed and because I/O time will be
+ // much greater than conversion time.
+
+ {
+ struct Record
+ {
+ big_uint32_t count; // big endian
+ big_int32_t value; // big endian
+ };
+
+ Record rec;
+
+ read(&rec, sizeof(Record));
+
+ ++rec.count;
+ rec.value += fee;
+
+ write(&rec, sizeof(Record));
+ }
+
+// Recommended approach when conversion time is a concern
+ //
+ // Conversion time is a concert with this application because (1) any conversions
+ // performed in the loop will consume a great deal of time and because (2)
+ // computation time will be much greater than I/O time.
+
+ {
+ struct Record
+ {
+ big_uint32_t count; // big endian
+ big_int32_t value; // big endian
+ };
+
+ Record rec;
+
+ read(&rec, sizeof(Record));
+
+ uint32_t count = rec.count;
+ int32_t value = rec.value;
+
+ for (long long i = 0; i < several_gazillion; ++i) // (1)
+ {
+ ... immensely complex computation using rec variables many times // (2)
+ }
+
+ rec.count = count;
+ rec.value = value;
+
+ write(&rec, sizeof(Record));
+ }
+
+}