32 #include <unordered_map>
40 enum class Type : uint8_t
57 : stride(stride), str(str) {}
62 static std::map<Type, PropertyInfo> PropertyTable
65 { Type::UINT8, PropertyInfo(1, std::string(
"uchar")) },
66 { Type::INT16, PropertyInfo(2, std::string(
"short")) },
67 { Type::UINT16, PropertyInfo(2, std::string(
"ushort")) },
68 { Type::INT32, PropertyInfo(4, std::string(
"int")) },
69 { Type::UINT32, PropertyInfo(4, std::string(
"uint")) },
70 { Type::FLOAT32, PropertyInfo(4, std::string(
"float")) },
71 { Type::FLOAT64, PropertyInfo(8, std::string(
"double")) },
72 { Type::INVALID, PropertyInfo(0, std::string(
"INVALID"))}
77 uint8_t * alias{
nullptr };
78 struct delete_array {
void operator()(uint8_t * p) {
delete[] p; } };
83 Buffer(
const size_t size) : data(new uint8_t[size], delete_array()), size(size) { alias = data.get(); }
84 Buffer(
const uint8_t * ptr): alias(const_cast<uint8_t*>(ptr)) { }
85 uint8_t * get() {
return alias; }
86 const uint8_t * get_const() {
return const_cast<const uint8_t*
>(alias); }
87 size_t size_bytes()
const {
return size; }
101 PlyProperty(Type type, std::string & _name) : name(_name), propertyType(type) {}
102 PlyProperty(Type list_type, Type prop_type, std::string & _name,
size_t list_count)
103 : name(_name), propertyType(prop_type), isList(
true), listType(list_type), listCount(list_count) {}
105 Type propertyType{ Type::INVALID };
106 bool isList{
false };
107 Type listType{ Type::INVALID };
108 size_t listCount {0};
114 PlyElement(
const std::string & _name,
size_t count) : name(_name), size(count) {}
117 std::vector<PlyProperty> properties;
123 std::unique_ptr<PlyFileImpl> impl;
134 bool parse_header(std::istream & is);
140 void read(std::istream & is);
146 void write(std::ostream & os,
bool isBinary);
152 std::vector<PlyElement> get_elements()
const;
153 std::vector<std::string> get_info()
const;
154 std::vector<std::string> & get_comments();
155 bool is_binary_file()
const;
164 std::shared_ptr<PlyData> request_properties_from_element(
const std::string & elementKey,
165 const std::vector<std::string> propertyKeys,
const uint32_t list_size_hint = 0);
167 void add_properties_to_element(
const std::string & elementKey,
168 const std::vector<std::string> propertyKeys,
171 const uint8_t * data,
173 const size_t listCount);
184 #ifdef TINYPLY_IMPLEMENTATION
187 #include <functional>
188 #include <type_traits>
197 template<
typename T,
typename T2>
inline T2 endian_swap(
const T & v) noexcept {
return v; }
198 template<>
inline uint16_t endian_swap<uint16_t, uint16_t>(
const uint16_t & v) noexcept {
return (v << 8) | (v >> 8); }
199 template<>
inline uint32_t endian_swap<uint32_t, uint32_t>(
const uint32_t & v) noexcept {
return (v << 24) | ((v << 8) & 0x00ff0000) | ((v >> 8) & 0x0000ff00) | (v >> 24); }
200 template<>
inline uint64_t endian_swap<uint64_t, uint64_t>(
const uint64_t & v) noexcept
202 return (((v & 0x00000000000000ffLL) << 56) |
203 ((v & 0x000000000000ff00LL) << 40) |
204 ((v & 0x0000000000ff0000LL) << 24) |
205 ((v & 0x00000000ff000000LL) << 8) |
206 ((v & 0x000000ff00000000LL) >> 8) |
207 ((v & 0x0000ff0000000000LL) >> 24) |
208 ((v & 0x00ff000000000000LL) >> 40) |
209 ((v & 0xff00000000000000LL) >> 56));
211 template<>
inline int16_t endian_swap<int16_t, int16_t>(
const int16_t & v) noexcept { uint16_t r = endian_swap<uint16_t, uint16_t>(*(uint16_t*)&v);
return *(int16_t*)&r; }
212 template<>
inline int32_t endian_swap<int32_t, int32_t>(
const int32_t & v) noexcept { uint32_t r = endian_swap<uint32_t, uint32_t>(*(uint32_t*)&v);
return *(int32_t*)&r; }
213 template<>
inline int64_t endian_swap<int64_t, int64_t>(
const int64_t & v) noexcept { uint64_t r = endian_swap<uint64_t, uint64_t>(*(uint64_t*)&v);
return *(int64_t*)&r; }
214 template<>
inline float endian_swap<uint32_t, float>(
const uint32_t & v) noexcept {
union {
float f; uint32_t i; }; i = endian_swap<uint32_t, uint32_t>(v);
return f; }
215 template<>
inline double endian_swap<uint64_t, double>(
const uint64_t & v) noexcept {
union {
double d; uint64_t i; }; i = endian_swap<uint64_t, uint64_t>(v);
return d; }
217 inline uint32_t hash_fnv1a(
const std::string & str) noexcept
219 static const uint32_t fnv1aBase32 = 0x811C9DC5u;
220 static const uint32_t fnv1aPrime32 = 0x01000193u;
221 uint32_t result = fnv1aBase32;
222 for (
auto & c : str) { result ^=
static_cast<uint32_t
>(c); result *= fnv1aPrime32; }
226 inline Type property_type_from_string(
const std::string & t) noexcept
228 if (t ==
"int8" || t ==
"char")
return Type::INT8;
229 else if (t ==
"uint8" || t ==
"uchar")
return Type::UINT8;
230 else if (t ==
"int16" || t ==
"short")
return Type::INT16;
231 else if (t ==
"uint16" || t ==
"ushort")
return Type::UINT16;
232 else if (t ==
"int32" || t ==
"int")
return Type::INT32;
233 else if (t ==
"uint32" || t ==
"uint")
return Type::UINT32;
234 else if (t ==
"float32" || t ==
"float")
return Type::FLOAT32;
235 else if (t ==
"float64" || t ==
"double")
return Type::FLOAT64;
236 return Type::INVALID;
239 struct PlyFile::PlyFileImpl
243 size_t byteOffset{ 0 };
244 size_t totalSizeBytes{ 0 };
249 std::shared_ptr<PlyData> data;
250 std::shared_ptr<PlyDataCursor> cursor;
251 uint32_t list_size_hint;
254 struct PropertyLookup
256 ParsingHelper * helper{
nullptr };
258 size_t prop_stride{ 0 };
259 size_t list_stride{ 0 };
262 std::unordered_map<uint32_t, ParsingHelper> userData;
264 bool isBinary =
false;
265 bool isBigEndian =
false;
266 std::vector<PlyElement> elements;
267 std::vector<std::string> comments;
268 std::vector<std::string> objInfo;
271 void read(std::istream & is);
272 void write(std::ostream & os,
bool isBinary);
274 std::shared_ptr<PlyData> request_properties_from_element(
const std::string & elementKey,
275 const std::vector<std::string> propertyKeys,
276 const uint32_t list_size_hint);
278 void add_properties_to_element(
const std::string & elementKey,
279 const std::vector<std::string> propertyKeys,
280 const Type type,
const size_t count,
const uint8_t * data,
const Type listType,
const size_t listCount);
282 size_t read_property_binary(
const size_t & stride,
void * dest,
size_t & destOffset, std::istream & is) noexcept;
283 size_t read_property_ascii(
const Type & t,
const size_t & stride,
void * dest,
size_t & destOffset, std::istream & is);
285 std::vector<std::vector<PropertyLookup>> make_property_lookup_table();
287 bool parse_header(std::istream & is);
288 void parse_data(std::istream & is,
bool firstPass);
289 void read_header_format(std::istream & is);
290 void read_header_element(std::istream & is);
291 void read_header_property(std::istream & is);
292 void read_header_text(std::string line, std::vector<std::string> & place,
int erase = 0);
294 void write_header(std::ostream & os) noexcept;
295 void write_ascii_internal(std::ostream & os) noexcept;
296 void write_binary_internal(std::ostream & os) noexcept;
297 void write_property_ascii(Type t, std::ostream & os,
const uint8_t * src,
size_t & srcOffset);
298 void write_property_binary(std::ostream & os,
const uint8_t * src,
size_t & srcOffset,
const size_t & stride) noexcept;
301 PlyProperty::PlyProperty(std::istream & is) : isList(false)
307 std::string countType;
308 is >> countType >> type;
309 listType = property_type_from_string(countType);
312 propertyType = property_type_from_string(type);
316 PlyElement::PlyElement(std::istream & is)
321 template<
typename T>
inline T ply_read_ascii(std::istream & is)
328 template<
typename T,
typename T2>
329 inline void endian_swap_buffer(uint8_t * data_ptr,
const size_t num_bytes,
const size_t stride)
331 for (
size_t count = 0; count < num_bytes; count += stride)
333 *(
reinterpret_cast<T2 *
>(data_ptr)) = endian_swap<T, T2>(*(
reinterpret_cast<const T *
>(data_ptr)));
338 template<
typename T>
void ply_cast_ascii(
void * dest, std::istream & is)
340 *(
static_cast<T *
>(dest)) = ply_read_ascii<T>(is);
343 int64_t find_element(
const std::string & key,
const std::vector<PlyElement> & list)
345 for (
size_t i = 0; i < list.size(); i++)
if (list[i].name == key)
return i;
349 int64_t find_property(
const std::string & key,
const std::vector<PlyProperty> & list)
351 for (
size_t i = 0; i < list.size(); ++i)
if (list[i].name == key)
return i;
359 std::vector<std::vector<PlyFile::PlyFileImpl::PropertyLookup>> PlyFile::PlyFileImpl::make_property_lookup_table()
361 std::vector<std::vector<PropertyLookup>> element_property_lookup;
363 for (
auto & element : elements)
365 std::vector<PropertyLookup> lookups;
367 for (
auto & property : element.properties)
371 auto cursorIt = userData.find(hash_fnv1a(element.name + property.name));
372 if (cursorIt != userData.end()) f.helper = &cursorIt->second;
375 f.prop_stride = PropertyTable[
property.propertyType].stride;
376 if (property.isList) f.list_stride = PropertyTable[
property.listType].stride;
378 lookups.push_back(f);
381 element_property_lookup.push_back(lookups);
384 return element_property_lookup;
387 bool PlyFile::PlyFileImpl::parse_header(std::istream & is)
391 while (std::getline(is, line))
393 std::istringstream ls(line);
396 if (token ==
"ply" || token ==
"PLY" || token ==
"")
continue;
397 else if (token ==
"comment") read_header_text(line, comments, 8);
398 else if (token ==
"format") read_header_format(ls);
399 else if (token ==
"element") read_header_element(ls);
400 else if (token ==
"property") read_header_property(ls);
401 else if (token ==
"obj_info") read_header_text(line, objInfo, 9);
402 else if (token ==
"end_header")
break;
403 else success =
false;
408 void PlyFile::PlyFileImpl::read_header_text(std::string line, std::vector<std::string>& place,
int erase)
410 place.push_back((erase > 0) ? line.erase(0, erase) : line);
413 void PlyFile::PlyFileImpl::read_header_format(std::istream & is)
417 if (s ==
"binary_little_endian") isBinary =
true;
418 else if (s ==
"binary_big_endian") isBinary = isBigEndian =
true;
421 void PlyFile::PlyFileImpl::read_header_element(std::istream & is)
423 elements.emplace_back(is);
426 void PlyFile::PlyFileImpl::read_header_property(std::istream & is)
428 if (!elements.size())
throw std::runtime_error(
"no elements defined; file is malformed");
429 elements.back().properties.emplace_back(is);
432 size_t PlyFile::PlyFileImpl::read_property_binary(
const size_t & stride,
void * dest,
size_t & destOffset, std::istream & is) noexcept
434 destOffset += stride;
435 is.read((
char*)dest, stride);
439 size_t PlyFile::PlyFileImpl::read_property_ascii(
const Type & t,
const size_t & stride,
void * dest,
size_t & destOffset, std::istream & is)
441 destOffset += stride;
444 case Type::INT8: *((int8_t *)dest) =
static_cast<int8_t
>(ply_read_ascii<int32_t>(is));
break;
445 case Type::UINT8: *((uint8_t *)dest) =
static_cast<uint8_t
>(ply_read_ascii<uint32_t>(is));
break;
446 case Type::INT16: ply_cast_ascii<int16_t>(dest, is);
break;
447 case Type::UINT16: ply_cast_ascii<uint16_t>(dest, is);
break;
448 case Type::INT32: ply_cast_ascii<int32_t>(dest, is);
break;
449 case Type::UINT32: ply_cast_ascii<uint32_t>(dest, is);
break;
450 case Type::FLOAT32: ply_cast_ascii<float>(dest, is);
break;
451 case Type::FLOAT64: ply_cast_ascii<double>(dest, is);
break;
452 case Type::INVALID:
throw std::invalid_argument(
"invalid ply property");
457 void PlyFile::PlyFileImpl::write_property_ascii(Type t, std::ostream & os,
const uint8_t * src,
size_t & srcOffset)
461 case Type::INT8: os << static_cast<int32_t>(*
reinterpret_cast<const int8_t*
>(src));
break;
462 case Type::UINT8: os << static_cast<uint32_t>(*
reinterpret_cast<const uint8_t*
>(src));
break;
463 case Type::INT16: os << *reinterpret_cast<const int16_t*>(src);
break;
464 case Type::UINT16: os << *reinterpret_cast<const uint16_t*>(src);
break;
465 case Type::INT32: os << *reinterpret_cast<const int32_t*>(src);
break;
466 case Type::UINT32: os << *reinterpret_cast<const uint32_t*>(src);
break;
467 case Type::FLOAT32: os << *reinterpret_cast<const float*>(src);
break;
468 case Type::FLOAT64: os << *reinterpret_cast<const double*>(src);
break;
469 case Type::INVALID:
throw std::invalid_argument(
"invalid ply property");
472 srcOffset += PropertyTable[t].stride;
475 void PlyFile::PlyFileImpl::write_property_binary(std::ostream & os,
const uint8_t * src,
size_t & srcOffset,
const size_t & stride) noexcept
477 os.write((
char *)src, stride);
481 void PlyFile::PlyFileImpl::read(std::istream & is)
483 std::vector<std::shared_ptr<PlyData>> buffers;
484 for (
auto & entry : userData) buffers.push_back(entry.second.data);
487 uint32_t list_hints = 0;
488 for (
auto & b : buffers)
for (
auto & entry : userData) {list_hints += entry.second.list_size_hint;(void)b;}
493 parse_data(is,
true);
499 std::unordered_map<PlyData*, int32_t> unique_data_count;
500 for (
auto & ptr : buffers) unique_data_count[ptr.get()] += 1;
504 std::sort(buffers.begin(), buffers.end());
505 buffers.erase(std::unique(buffers.begin(), buffers.end()), buffers.end());
508 for (
auto & b : buffers)
510 for (
auto & entry : userData)
512 if (entry.second.data == b && b->buffer.get() ==
nullptr)
518 b->buffer = Buffer(entry.second.cursor->totalSizeBytes);
523 const size_t list_size_multiplier = (entry.second.data->isList ? entry.second.list_size_hint : 1);
524 auto bytes_per_property = entry.second.data->count * PropertyTable[entry.second.data->t].stride * list_size_multiplier;
525 bytes_per_property *= unique_data_count[b.get()];
526 b->buffer = Buffer(bytes_per_property);
534 parse_data(is,
false);
539 for (
auto & b : buffers)
541 uint8_t * data_ptr = b->buffer.get();
542 const size_t stride = PropertyTable[b->t].stride;
543 const size_t buffer_size_bytes = b->buffer.size_bytes();
547 case Type::INT16: endian_swap_buffer<int16_t, int16_t>(data_ptr, buffer_size_bytes, stride);
break;
548 case Type::UINT16: endian_swap_buffer<uint16_t, uint16_t>(data_ptr, buffer_size_bytes, stride);
break;
549 case Type::INT32: endian_swap_buffer<int32_t, int32_t>(data_ptr, buffer_size_bytes, stride);
break;
550 case Type::UINT32: endian_swap_buffer<uint32_t, uint32_t>(data_ptr, buffer_size_bytes, stride);
break;
551 case Type::FLOAT32: endian_swap_buffer<uint32_t, float>(data_ptr, buffer_size_bytes, stride);
break;
552 case Type::FLOAT64: endian_swap_buffer<uint64_t, double>(data_ptr, buffer_size_bytes, stride);
break;
559 void PlyFile::PlyFileImpl::write(std::ostream & os,
bool _isBinary)
561 for (
auto & d : userData) { d.second.cursor->byteOffset = 0; }
566 write_binary_internal(os);
572 write_ascii_internal(os);
576 void PlyFile::PlyFileImpl::write_binary_internal(std::ostream & os) noexcept
582 uint8_t listSize[4] = { 0, 0, 0, 0 };
583 size_t dummyCount = 0;
585 auto element_property_lookup = make_property_lookup_table();
587 size_t element_idx = 0;
588 for (
auto & e : elements)
590 for (
size_t i = 0; i < e.size; ++i)
592 size_t property_index = 0;
593 for (
auto & p : e.properties)
595 auto & f = element_property_lookup[element_idx][property_index];
596 auto * helper = f.helper;
597 if (f.skip || helper ==
nullptr)
continue;
601 std::memcpy(listSize, &p.listCount,
sizeof(uint32_t));
602 write_property_binary(os, listSize, dummyCount, f.list_stride);
603 write_property_binary(os, (helper->data->buffer.get_const() + helper->cursor->byteOffset), helper->cursor->byteOffset, f.prop_stride * p.listCount);
607 write_property_binary(os, (helper->data->buffer.get_const() + helper->cursor->byteOffset), helper->cursor->byteOffset, f.prop_stride);
616 void PlyFile::PlyFileImpl::write_ascii_internal(std::ostream & os) noexcept
620 auto element_property_lookup = make_property_lookup_table();
622 size_t element_idx = 0;
623 for (
auto & e : elements)
625 for (
size_t i = 0; i < e.size; ++i)
627 size_t property_index = 0;
628 for (
auto & p : e.properties)
630 auto & f = element_property_lookup[element_idx][property_index];
631 auto * helper = f.helper;
632 if (f.skip || helper ==
nullptr)
continue;
636 os << p.listCount <<
" ";
637 for (
size_t j = 0; j < p.listCount; ++j)
639 write_property_ascii(p.propertyType, os, (helper->data->buffer.get() + helper->cursor->byteOffset), helper->cursor->byteOffset);
644 write_property_ascii(p.propertyType, os, (helper->data->buffer.get() + helper->cursor->byteOffset), helper->cursor->byteOffset);
654 void PlyFile::PlyFileImpl::write_header(std::ostream & os) noexcept
656 const std::locale & fixLoc = std::locale(
"C");
660 if (isBinary) os << ((isBigEndian) ?
"format binary_big_endian 1.0" :
"format binary_little_endian 1.0") <<
"\n";
661 else os <<
"format ascii 1.0\n";
663 for (
const auto & comment : comments) os <<
"comment " << comment <<
"\n";
665 auto property_lookup = make_property_lookup_table();
667 size_t element_idx = 0;
668 for (
auto & e : elements)
670 os <<
"element " << e.name <<
" " << e.size <<
"\n";
671 size_t property_idx = 0;
672 for (
const auto & p : e.properties)
674 PropertyLookup & lookup = property_lookup[element_idx][property_idx];
680 os <<
"property list " << PropertyTable[p.listType].str <<
" "
681 << PropertyTable[p.propertyType].str <<
" " << p.name <<
"\n";
685 os <<
"property " << PropertyTable[p.propertyType].str <<
" " << p.name <<
"\n";
692 os <<
"end_header\n";
695 std::shared_ptr<PlyData> PlyFile::PlyFileImpl::request_properties_from_element(
const std::string & elementKey,
696 const std::vector<std::string> propertyKeys,
697 const uint32_t list_size_hint)
699 if (elements.empty())
throw std::runtime_error(
"header had no elements defined. malformed file?");
700 if (elementKey.empty())
throw std::invalid_argument(
"`elementKey` argument is empty");
701 if (propertyKeys.empty())
throw std::invalid_argument(
"`propertyKeys` argument is empty");
703 std::shared_ptr<PlyData> out_data = std::make_shared<PlyData>();
705 const int64_t elementIndex = find_element(elementKey, elements);
707 std::vector<std::string> keys_not_found;
710 if (elementIndex >= 0)
713 const PlyElement & element = elements[elementIndex];
722 ParsingHelper helper;
723 helper.data = out_data;
724 helper.data->count = element.size;
725 helper.data->isList =
false;
726 helper.data->t = Type::INVALID;
727 helper.cursor = std::make_shared<PlyDataCursor>();
728 helper.list_size_hint = list_size_hint;
731 for (
const auto & key : propertyKeys)
733 const int64_t propertyIndex = find_property(key, element.properties);
734 if (propertyIndex < 0) keys_not_found.push_back(key);
737 if (keys_not_found.size())
739 std::stringstream ss;
740 for (
auto & str : keys_not_found) ss << str <<
", ";
741 throw std::invalid_argument(
"the following property keys were not found in the header: " + ss.str());
744 for (
const auto & key : propertyKeys)
746 const int64_t propertyIndex = find_property(key, element.properties);
747 const PlyProperty &
property = element.properties[propertyIndex];
748 helper.data->t =
property.propertyType;
749 helper.data->isList =
property.isList;
750 auto result = userData.insert(std::pair<uint32_t, ParsingHelper>(hash_fnv1a(element.name + property.name), helper));
751 if (result.second ==
false)
753 throw std::invalid_argument(
"element-property key has already been requested: " + element.name +
" " + property.name);
758 std::vector<Type> propertyTypes;
759 for (
const auto & key : propertyKeys)
761 const int64_t propertyIndex = find_property(key, element.properties);
762 const PlyProperty &
property = element.properties[propertyIndex];
763 propertyTypes.push_back(property.propertyType);
766 if (std::adjacent_find(propertyTypes.begin(), propertyTypes.end(), std::not_equal_to<Type>()) != propertyTypes.end())
768 throw std::invalid_argument(
"all requested properties must share the same type.");
771 else throw std::invalid_argument(
"the element key was not found in the header: " + elementKey);
776 void PlyFile::PlyFileImpl::add_properties_to_element(
const std::string & elementKey,
777 const std::vector<std::string> propertyKeys,
778 const Type type,
const size_t count,
const uint8_t * data,
const Type listType,
const size_t listCount)
780 ParsingHelper helper;
781 helper.data = std::make_shared<PlyData>();
782 helper.data->count = count;
783 helper.data->t = type;
784 helper.data->buffer = Buffer(data);
785 helper.cursor = std::make_shared<PlyDataCursor>();
787 auto create_property_on_element = [&](PlyElement & e)
789 for (
auto key : propertyKeys)
791 PlyProperty newProp = (listType == Type::INVALID) ? PlyProperty(type, key) : PlyProperty(listType, type, key, listCount);
792 userData.insert(std::pair<uint32_t, ParsingHelper>(hash_fnv1a(elementKey + key), helper));
793 e.properties.push_back(newProp);
797 const int64_t idx = find_element(elementKey, elements);
800 PlyElement & e = elements[idx];
801 create_property_on_element(e);
805 PlyElement newElement = (listType == Type::INVALID) ? PlyElement(elementKey, count) : PlyElement(elementKey, count);
806 create_property_on_element(newElement);
807 elements.push_back(newElement);
811 void PlyFile::PlyFileImpl::parse_data(std::istream & is,
bool firstPass)
813 std::function<void(PropertyLookup & f,
const PlyProperty & p, uint8_t * dest,
size_t & destOffset, std::istream & is)> read;
814 std::function<size_t(PropertyLookup & f,
const PlyProperty & p, std::istream & is)> skip;
816 const auto start = is.tellg();
818 uint32_t listSize = 0;
819 size_t dummyCount = 0;
820 std::string skip_ascii_buffer;
827 auto read_list_binary = [
this](
const Type & t,
void * dst,
size_t & destOffset,
const size_t & stride, std::istream & _is) noexcept
829 destOffset += stride;
830 _is.read((
char*)dst, stride);
836 case Type::INT16: *(int16_t*)dst = endian_swap<int16_t, int16_t>(*(int16_t*)dst);
break;
837 case Type::UINT16: *(uint16_t*)dst = endian_swap<uint16_t, uint16_t>(*(uint16_t*)dst);
break;
838 case Type::INT32: *(int32_t*)dst = endian_swap<int32_t, int32_t>(*(int32_t*)dst);
break;
839 case Type::UINT32: *(uint32_t*)dst = endian_swap<uint32_t, uint32_t>(*(uint32_t*)dst);
break;
849 read = [
this, &listSize, &dummyCount, &read_list_binary](PropertyLookup & f,
const PlyProperty & p, uint8_t * dest,
size_t & destOffset, std::istream & _is) noexcept
853 return read_property_binary(f.prop_stride, dest + destOffset, destOffset, _is);
855 read_list_binary(p.listType, &listSize, dummyCount, f.list_stride, _is);
856 return read_property_binary(f.prop_stride * listSize, dest + destOffset, destOffset, _is);
858 skip = [
this, &listSize, &dummyCount, &read_list_binary](PropertyLookup & f,
const PlyProperty & p, std::istream & _is) noexcept
862 _is.read((
char*)scratch, f.prop_stride);
863 return f.prop_stride;
865 read_list_binary(p.listType, &listSize, dummyCount, f.list_stride, _is);
866 auto bytes_to_skip = f.prop_stride * listSize;
867 _is.ignore(bytes_to_skip);
868 return bytes_to_skip;
873 read = [
this, &listSize, &dummyCount](PropertyLookup & f,
const PlyProperty & p, uint8_t * dest,
size_t & destOffset, std::istream & _is) noexcept
877 read_property_ascii(p.propertyType, f.prop_stride, dest + destOffset, destOffset, _is);
881 read_property_ascii(p.listType, f.list_stride, &listSize, dummyCount, _is);
882 for (
size_t i = 0; i < listSize; ++i)
884 read_property_ascii(p.propertyType, f.prop_stride, dest + destOffset, destOffset, _is);
888 skip = [
this, &listSize, &dummyCount, &skip_ascii_buffer](PropertyLookup & f,
const PlyProperty & p, std::istream & _is) noexcept
890 skip_ascii_buffer.clear();
893 read_property_ascii(p.listType, f.list_stride, &listSize, dummyCount, _is);
894 for (
size_t i = 0; i < listSize; ++i) _is >> skip_ascii_buffer;
895 return listSize * f.prop_stride;
897 _is >> skip_ascii_buffer;
898 return f.prop_stride;
902 std::vector<std::vector<PropertyLookup>> element_property_lookup = make_property_lookup_table();
903 size_t element_idx = 0;
904 size_t property_idx = 0;
905 ParsingHelper * helper {
nullptr};
908 for (
auto & element : elements)
910 for (
size_t count = 0; count < element.size; ++count)
913 for (
auto & property : element.properties)
915 PropertyLookup & lookup = element_property_lookup[element_idx][property_idx];
919 helper = lookup.helper;
922 helper->cursor->totalSizeBytes += skip(lookup, property, is);
927 if (property.listCount == 0)
property.listCount = listSize;
928 if (property.listCount != listSize)
throw std::runtime_error(
"variable length lists are not supported yet.");
932 read(lookup, property, helper->data->buffer.get(), helper->cursor->byteOffset, is);
937 skip(lookup, property, is);
946 if (firstPass) is.seekg(start, is.beg);
951 PlyFile::PlyFile() { impl.reset(
new PlyFileImpl()); }
952 PlyFile::~PlyFile() { }
953 bool PlyFile::parse_header(std::istream & is) {
return impl->parse_header(is); }
954 void PlyFile::read(std::istream & is) {
return impl->read(is); }
955 void PlyFile::write(std::ostream & os,
bool isBinary) {
return impl->write(os, isBinary); }
956 std::vector<PlyElement> PlyFile::get_elements()
const {
return impl->elements; }
957 std::vector<std::string> & PlyFile::get_comments() {
return impl->comments; }
958 std::vector<std::string> PlyFile::get_info()
const {
return impl->objInfo; }
959 bool PlyFile::is_binary_file()
const {
return impl->isBinary; }
960 std::shared_ptr<PlyData> PlyFile::request_properties_from_element(
const std::string & elementKey,
961 const std::vector<std::string> propertyKeys,
962 const uint32_t list_size_hint)
964 return impl->request_properties_from_element(elementKey, propertyKeys, list_size_hint);
966 void PlyFile::add_properties_to_element(
const std::string & elementKey,
967 const std::vector<std::string> propertyKeys,
968 const Type type,
const size_t count,
const uint8_t * data,
const Type listType,
const size_t listCount)
970 return impl->add_properties_to_element(elementKey, propertyKeys, type, count, data, listType, listCount);
Definition: tinyply.h:112
Definition: tinyply.h:121