serialize.cpp 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756
  1. //
  2. // Copyright (C) 2014 Christian Schüller <schuellchr@gmail.com>
  3. //
  4. // This Source Code Form is subject to the terms of the Mozilla Public License
  5. // v. 2.0. If a copy of the MPL was not distributed with this file, You can
  6. // obtain one at http://mozilla.org/MPL/2.0/.
  7. #include "serialize.h"
  8. namespace igl
  9. {
  10. template <typename T>
  11. IGL_INLINE bool serialize(const T& obj,const std::string& filename)
  12. {
  13. return serialize(obj,"obj",filename,true);
  14. }
  15. template <typename T>
  16. IGL_INLINE bool serialize(const T& obj,const std::string& objectName,const std::string& filename,bool overwrite)
  17. {
  18. bool success = false;
  19. std::vector<char> buffer;
  20. std::ios_base::openmode mode = std::ios::out | std::ios::binary;
  21. if(overwrite)
  22. mode |= std::ios::trunc;
  23. else
  24. mode |= std::ios::app;
  25. std::ofstream file(filename.c_str(),mode);
  26. if(file.is_open())
  27. {
  28. serialize(obj,objectName,buffer);
  29. file.write(&buffer[0],buffer.size());
  30. file.close();
  31. success = true;
  32. }
  33. else
  34. {
  35. std::cerr << "saving binary serialization failed!" << std::endl;
  36. }
  37. return success;
  38. }
  39. template <typename T>
  40. IGL_INLINE bool serialize(const T& obj,const std::string& objectName,std::vector<char>& buffer)
  41. {
  42. // serialize object data
  43. size_t size = serialization::getByteSize(obj);
  44. std::vector<char> tmp(size);
  45. auto it = tmp.begin();
  46. serialization::serialize(obj,tmp,it);
  47. std::string objectType(typeid(obj).name());
  48. size_t newObjectSize = tmp.size();
  49. size_t newHeaderSize = serialization::getByteSize(objectName) + serialization::getByteSize(objectType) + sizeof(size_t);
  50. size_t curSize = buffer.size();
  51. size_t newSize = curSize + newHeaderSize + newObjectSize;
  52. buffer.resize(newSize);
  53. std::vector<char>::iterator iter = buffer.begin()+curSize;
  54. // serialize object header (name/type/size)
  55. serialization::serialize(objectName,buffer,iter);
  56. serialization::serialize(objectType,buffer,iter);
  57. serialization::serialize(newObjectSize,buffer,iter);
  58. // copy serialized data to buffer
  59. iter = std::copy(tmp.begin(),tmp.end(),iter);
  60. return true;
  61. }
  62. template <typename T>
  63. IGL_INLINE bool deserialize(T& obj,const std::string& filename)
  64. {
  65. return deserialize(obj,"obj",filename);
  66. }
  67. template <typename T>
  68. IGL_INLINE bool deserialize(T& obj,const std::string& objectName,const std::string& filename)
  69. {
  70. bool success = false;
  71. std::ifstream file(filename.c_str(),std::ios::binary);
  72. if(file.is_open())
  73. {
  74. file.seekg(0,std::ios::end);
  75. std::streamoff size = file.tellg();
  76. file.seekg(0,std::ios::beg);
  77. std::vector<char> buffer(size);
  78. file.read(&buffer[0],size);
  79. deserialize(obj,objectName,buffer);
  80. file.close();
  81. success = true;
  82. }
  83. else
  84. {
  85. std::cerr << "Loading binary serialization failed!" << std::endl;
  86. }
  87. return success;
  88. }
  89. template <typename T>
  90. IGL_INLINE bool deserialize(T& obj,const std::string& objectName,const std::vector<char>& buffer)
  91. {
  92. bool success = false;
  93. // find suitable object header
  94. auto objectIter = buffer.cend();
  95. auto iter = buffer.cbegin();
  96. while(iter != buffer.end())
  97. {
  98. std::string name;
  99. std::string type;
  100. size_t size;
  101. serialization::deserialize(name,iter);
  102. serialization::deserialize(type,iter);
  103. serialization::deserialize(size,iter);
  104. if(name == objectName && type == typeid(obj).name())
  105. {
  106. objectIter = iter;
  107. //break; // find first suitable object header
  108. }
  109. iter+=size;
  110. }
  111. if(objectIter != buffer.end())
  112. {
  113. serialization::deserialize(obj,objectIter);
  114. success = true;
  115. }
  116. else
  117. {
  118. obj = T();
  119. }
  120. return success;
  121. }
  122. // Wrapper function which combines both, de- and serialization
  123. template <typename T>
  124. IGL_INLINE bool serializer(bool s,T& obj,std::string& filename)
  125. {
  126. return s ? serialize(obj,filename) : deserialize(obj,filename);
  127. }
  128. template <typename T>
  129. IGL_INLINE bool serializer(bool s,T& obj,std::string& objectName,const std::string& filename,bool overwrite)
  130. {
  131. return s ? serialize(obj,objectName,filename,overwrite) : deserialize(obj,objectName,filename);
  132. }
  133. template <typename T>
  134. IGL_INLINE bool serializer(bool s,T& obj,std::string& objectName,std::vector<char>& buffer)
  135. {
  136. return s ? serialize(obj,objectName,buffer) : deserialize(obj,objectName,buffer);
  137. }
  138. IGL_INLINE bool Serializable::PreSerialization() const
  139. {
  140. return true;
  141. }
  142. IGL_INLINE void Serializable::PostSerialization() const
  143. {
  144. }
  145. IGL_INLINE bool Serializable::PreDeserialization()
  146. {
  147. return true;
  148. }
  149. IGL_INLINE void Serializable::PostDeserialization()
  150. {
  151. }
  152. IGL_INLINE void Serializable::Serialize(std::vector<char>& buffer) const
  153. {
  154. if(this->PreSerialization())
  155. {
  156. if(initialized == false)
  157. {
  158. objects.clear();
  159. (const_cast<Serializable*>(this))->InitSerialization();
  160. initialized = true;
  161. }
  162. for(const auto& v : objects)
  163. {
  164. v->Serialize(buffer);
  165. }
  166. this->PostSerialization();
  167. }
  168. }
  169. IGL_INLINE void Serializable::Deserialize(const std::vector<char>& buffer)
  170. {
  171. if(this->PreDeserialization())
  172. {
  173. if(initialized == false)
  174. {
  175. objects.clear();
  176. (const_cast<Serializable*>(this))->InitSerialization();
  177. initialized = true;
  178. }
  179. for(auto& v : objects)
  180. {
  181. v->Deserialize(buffer);
  182. }
  183. this->PostDeserialization();
  184. }
  185. }
  186. IGL_INLINE Serializable::Serializable()
  187. {
  188. initialized = false;
  189. }
  190. IGL_INLINE Serializable::Serializable(const Serializable& obj)
  191. {
  192. initialized = false;
  193. objects.clear();
  194. }
  195. IGL_INLINE Serializable::~Serializable()
  196. {
  197. initialized = false;
  198. objects.clear();
  199. }
  200. IGL_INLINE Serializable& Serializable::operator=(const Serializable& obj)
  201. {
  202. if(this != &obj)
  203. {
  204. if(initialized)
  205. {
  206. initialized = false;
  207. objects.clear();
  208. }
  209. }
  210. return *this;
  211. }
  212. template <typename T>
  213. IGL_INLINE void Serializable::Add(T& obj,std::string name,bool binary)
  214. {
  215. auto object = new SerializationObject<T>();
  216. object->Binary = binary;
  217. object->Name = name;
  218. object->Object = std::unique_ptr<T>(&obj);
  219. objects.push_back(object);
  220. }
  221. namespace serialization
  222. {
  223. // not serializable
  224. template <typename T>
  225. IGL_INLINE typename std::enable_if<!is_serializable<T>::value,size_t>::type getByteSize(const T& obj)
  226. {
  227. return sizeof(std::vector<char>::size_type);
  228. }
  229. template <typename T>
  230. IGL_INLINE typename std::enable_if<!is_serializable<T>::value>::type serialize(const T& obj,std::vector<char>& buffer,std::vector<char>::iterator& iter)
  231. {
  232. // data
  233. std::vector<char> tmp;
  234. serialize(obj,tmp);
  235. // size
  236. size_t size = buffer.size();
  237. serialization::serialize(tmp.size(),buffer,iter);
  238. size_t cur = iter - buffer.begin();
  239. buffer.resize(size+tmp.size());
  240. iter = buffer.begin()+cur;
  241. iter = std::copy(tmp.begin(),tmp.end(),iter);
  242. }
  243. template <typename T>
  244. IGL_INLINE typename std::enable_if<!is_serializable<T>::value>::type deserialize(T& obj,std::vector<char>::const_iterator& iter)
  245. {
  246. std::vector<char>::size_type size;
  247. serialization::deserialize(size,iter);
  248. std::vector<char> tmp;
  249. tmp.resize(size);
  250. std::copy(iter,iter+size,tmp.begin());
  251. deserialize(obj,tmp);
  252. iter += size;
  253. }
  254. // fundamental types
  255. template <typename T>
  256. IGL_INLINE typename std::enable_if<std::is_fundamental<T>::value,size_t>::type getByteSize(const T& obj)
  257. {
  258. return sizeof(T);
  259. }
  260. template <typename T>
  261. IGL_INLINE typename std::enable_if<std::is_fundamental<T>::value>::type serialize(const T& obj,std::vector<char>& buffer,std::vector<char>::iterator& iter)
  262. {
  263. //serialization::updateMemoryMap(obj,sizeof(T),memoryMap);
  264. const uint8_t* ptr = reinterpret_cast<const uint8_t*>(&obj);
  265. iter = std::copy(ptr,ptr+sizeof(T),iter);
  266. }
  267. template <typename T>
  268. IGL_INLINE typename std::enable_if<std::is_fundamental<T>::value>::type deserialize(T& obj,std::vector<char>::const_iterator& iter)
  269. {
  270. uint8_t* ptr = reinterpret_cast<uint8_t*>(&obj);
  271. std::copy(iter,iter+sizeof(T),ptr);
  272. iter += sizeof(T);
  273. }
  274. // std::string
  275. IGL_INLINE size_t getByteSize(const std::string& obj)
  276. {
  277. return getByteSize(obj.length())+obj.length()*sizeof(uint8_t);
  278. }
  279. IGL_INLINE void serialize(const std::string& obj,std::vector<char>& buffer,std::vector<char>::iterator& iter)
  280. {
  281. serialization::serialize(obj.length(),buffer,iter);
  282. for(const auto& cur : obj)
  283. {
  284. serialization::serialize(cur,buffer,iter);
  285. }
  286. }
  287. IGL_INLINE void deserialize(std::string& obj,std::vector<char>::const_iterator& iter)
  288. {
  289. size_t size;
  290. serialization::deserialize(size,iter);
  291. std::string str(size,'\0');
  292. for(size_t i=0; i<size; ++i)
  293. {
  294. serialization::deserialize(str.at(i),iter);
  295. }
  296. obj = str;
  297. }
  298. // SerializableBase
  299. template <typename T>
  300. IGL_INLINE typename std::enable_if<std::is_base_of<SerializableBase,T>::value,size_t>::type getByteSize(const T& obj)
  301. {
  302. return sizeof(std::vector<char>::size_type);
  303. }
  304. template <typename T>
  305. IGL_INLINE typename std::enable_if<std::is_base_of<SerializableBase,T>::value>::type serialize(const T& obj,std::vector<char>& buffer,std::vector<char>::iterator& iter)
  306. {
  307. // data
  308. std::vector<char> tmp;
  309. obj.Serialize(tmp);
  310. // size
  311. size_t size = buffer.size();
  312. serialization::serialize(tmp.size(),buffer,iter);
  313. size_t cur = iter - buffer.begin();
  314. buffer.resize(size+tmp.size());
  315. iter = buffer.begin()+cur;
  316. iter = std::copy(tmp.begin(),tmp.end(),iter);
  317. }
  318. template <typename T>
  319. IGL_INLINE typename std::enable_if<std::is_base_of<SerializableBase,T>::value>::type deserialize(T& obj,std::vector<char>::const_iterator& iter)
  320. {
  321. std::vector<char>::size_type size;
  322. serialization::deserialize(size,iter);
  323. std::vector<char> tmp;
  324. tmp.resize(size);
  325. std::copy(iter,iter+size,tmp.begin());
  326. obj.Deserialize(tmp);
  327. iter += size;
  328. }
  329. // STL containers
  330. // std::pair
  331. template <typename T1,typename T2>
  332. IGL_INLINE size_t getByteSize(const std::pair<T1,T2>& obj)
  333. {
  334. return getByteSize(obj.first)+getByteSize(obj.second);
  335. }
  336. template <typename T1,typename T2>
  337. IGL_INLINE void serialize(const std::pair<T1,T2>& obj,std::vector<char>& buffer,std::vector<char>::iterator& iter)
  338. {
  339. serialization::serialize(obj.first,buffer,iter);
  340. serialization::serialize(obj.second,buffer,iter);
  341. }
  342. template <typename T1,typename T2>
  343. IGL_INLINE void deserialize(std::pair<T1,T2>& obj,std::vector<char>::const_iterator& iter)
  344. {
  345. serialization::deserialize(obj.first,iter);
  346. serialization::deserialize(obj.second,iter);
  347. }
  348. // std::vector
  349. template <typename T1,typename T2>
  350. IGL_INLINE size_t getByteSize(const std::vector<T1,T2>& obj)
  351. {
  352. return std::accumulate(obj.begin(),obj.end(),sizeof(size_t),[](const size_t& acc,const T1& cur) { return acc+getByteSize(cur); });
  353. }
  354. template <typename T1,typename T2>
  355. IGL_INLINE void serialize(const std::vector<T1,T2>& obj,std::vector<char>& buffer,std::vector<char>::iterator& iter)
  356. {
  357. size_t size = obj.size();
  358. serialization::serialize(size,buffer,iter);
  359. for(const auto& cur : obj)
  360. {
  361. serialization::serialize(cur,buffer,iter);
  362. }
  363. }
  364. template <typename T1,typename T2>
  365. IGL_INLINE void deserialize(std::vector<T1,T2>& obj,std::vector<char>::const_iterator& iter)
  366. {
  367. size_t size;
  368. serialization::deserialize(size,iter);
  369. obj.resize(size);
  370. for(auto& v : obj)
  371. {
  372. serialization::deserialize(v,iter);
  373. }
  374. }
  375. //std::set
  376. template <typename T>
  377. IGL_INLINE size_t getByteSize(const std::set<T>& obj)
  378. {
  379. return std::accumulate(obj.begin(),obj.end(),getByteSize(obj.size()),[](const size_t& acc,const T& cur) { return acc+getByteSize(cur); });
  380. }
  381. template <typename T>
  382. IGL_INLINE void serialize(const std::set<T>& obj,std::vector<char>& buffer,std::vector<char>::iterator& iter)
  383. {
  384. serialization::serialize(obj.size(),buffer,iter);
  385. for(const auto& cur : obj)
  386. {
  387. serialization::serialize(cur,buffer,iter);
  388. }
  389. }
  390. template <typename T>
  391. IGL_INLINE void deserialize(std::set<T>& obj,std::vector<char>::const_iterator& iter)
  392. {
  393. size_t size;
  394. serialization::deserialize(size,iter);
  395. obj.clear();
  396. for(size_t i=0; i<size; ++i)
  397. {
  398. T val;
  399. serialization::deserialize(val,iter);
  400. obj.insert(val);
  401. }
  402. }
  403. // std::map
  404. template <typename T1,typename T2>
  405. IGL_INLINE size_t getByteSize(const std::map<T1,T2>& obj)
  406. {
  407. return std::accumulate(obj.begin(),obj.end(),sizeof(size_t),[](const size_t& acc,const std::pair<T1,T2>& cur) { return acc+getByteSize(cur); });
  408. }
  409. template <typename T1,typename T2>
  410. IGL_INLINE void serialize(const std::map<T1,T2>& obj,std::vector<char>& buffer,std::vector<char>::iterator& iter)
  411. {
  412. serialization::serialize(obj.size(),buffer,iter);
  413. for(const auto& cur : obj)
  414. {
  415. serialization::serialize(cur,buffer,iter);
  416. }
  417. }
  418. template <typename T1,typename T2>
  419. IGL_INLINE void deserialize(std::map<T1,T2>& obj,std::vector<char>::const_iterator& iter)
  420. {
  421. size_t size;
  422. serialization::deserialize(size,iter);
  423. obj.clear();
  424. for(size_t i=0; i<size; ++i)
  425. {
  426. std::pair<T1,T2> pair;
  427. serialization::deserialize(pair,iter);
  428. obj.insert(pair);
  429. }
  430. }
  431. // Eigen types
  432. template<typename T,int R,int C,int P,int MR,int MC>
  433. IGL_INLINE size_t getByteSize(const Eigen::Matrix<T,R,C,P,MR,MC>& obj)
  434. {
  435. // space for numbers of rows,cols and data
  436. return 2*sizeof(typename Eigen::Matrix<T,R,C,P,MR,MC>::Index)+sizeof(T)*obj.rows()*obj.cols();
  437. }
  438. template<typename T,int R,int C,int P,int MR,int MC>
  439. IGL_INLINE void serialize(const Eigen::Matrix<T,R,C,P,MR,MC>& obj,std::vector<char>& buffer,std::vector<char>::iterator& iter)
  440. {
  441. serialization::serialize(obj.rows(),buffer,iter);
  442. serialization::serialize(obj.cols(),buffer,iter);
  443. size_t size = sizeof(T)*obj.rows()*obj.cols();
  444. auto ptr = reinterpret_cast<const uint8_t*>(obj.data());
  445. iter = std::copy(ptr,ptr+size,iter);
  446. }
  447. template<typename T,int R,int C,int P,int MR,int MC>
  448. IGL_INLINE void deserialize(Eigen::Matrix<T,R,C,P,MR,MC>& obj,std::vector<char>::const_iterator& iter)
  449. {
  450. typename Eigen::Matrix<T,R,C,P,MR,MC>::Index rows,cols;
  451. serialization::deserialize(rows,iter);
  452. serialization::deserialize(cols,iter);
  453. size_t size = sizeof(T)*rows*cols;
  454. obj.resize(rows,cols);
  455. auto ptr = reinterpret_cast<uint8_t*>(obj.data());
  456. std::copy(iter,iter+size,ptr);
  457. iter+=size;
  458. }
  459. template<typename T,int P,typename I>
  460. IGL_INLINE size_t getByteSize(const Eigen::SparseMatrix<T,P,I>& obj)
  461. {
  462. // space for numbers of rows,cols,nonZeros and tripplets with data (rowIdx,colIdx,value)
  463. size_t size = sizeof(Eigen::SparseMatrix<T,P,I>::Index);
  464. return 3*size+(sizeof(T)+2*size)*obj.nonZeros();
  465. }
  466. template<typename T,int P,typename I>
  467. IGL_INLINE void serialize(const Eigen::SparseMatrix<T,P,I>& obj,std::vector<char>& buffer,std::vector<char>::iterator& iter)
  468. {
  469. serialization::serialize(obj.rows(),buffer,iter);
  470. serialization::serialize(obj.cols(),buffer,iter);
  471. serialization::serialize(obj.nonZeros(),buffer,iter);
  472. for(int k=0;k<obj.outerSize();++k)
  473. {
  474. for(typename Eigen::SparseMatrix<T,P,I>::InnerIterator it(obj,k);it;++it)
  475. {
  476. serialization::serialize(it.row(),buffer,iter);
  477. serialization::serialize(it.col(),buffer,iter);
  478. serialization::serialize(it.value(),buffer,iter);
  479. }
  480. }
  481. }
  482. template<typename T,int P,typename I>
  483. IGL_INLINE void deserialize(Eigen::SparseMatrix<T,P,I>& obj,std::vector<char>::const_iterator& iter)
  484. {
  485. typename Eigen::SparseMatrix<T,P,I>::Index rows,cols,nonZeros;
  486. serialization::deserialize(rows,iter);
  487. serialization::deserialize(cols,iter);
  488. serialization::deserialize(nonZeros,iter);
  489. obj.resize(rows,cols);
  490. obj.setZero();
  491. std::vector<Eigen::Triplet<T,I> > triplets;
  492. for(int i=0;i<nonZeros;i++)
  493. {
  494. typename Eigen::SparseMatrix<T,P,I>::Index rowId,colId;
  495. serialization::deserialize(rowId,iter);
  496. serialization::deserialize(colId,iter);
  497. T value;
  498. serialization::deserialize(value,iter);
  499. triplets.push_back(Eigen::Triplet<T,I>(rowId,colId,value));
  500. }
  501. obj.setFromTriplets(triplets.begin(),triplets.end());
  502. }
  503. // pointers
  504. template <typename T>
  505. IGL_INLINE typename std::enable_if<std::is_pointer<T>::value,size_t>::type getByteSize(const T& obj)
  506. {
  507. size_t size = sizeof(bool);
  508. if(obj)
  509. size += getByteSize(*obj);
  510. return size;
  511. }
  512. template <typename T>
  513. IGL_INLINE typename std::enable_if<std::is_pointer<T>::value>::type serialize(const T& obj,std::vector<char>& buffer,std::vector<char>::iterator& iter)
  514. {
  515. serialization::serialize(obj == nullptr,buffer,iter);
  516. if(obj)
  517. serialization::serialize(*obj,buffer,iter);
  518. }
  519. template <typename T>
  520. IGL_INLINE typename std::enable_if<std::is_pointer<T>::value>::type deserialize(T& obj,std::vector<char>::const_iterator& iter)
  521. {
  522. bool isNullPtr;
  523. serialization::deserialize(isNullPtr,iter);
  524. if(isNullPtr)
  525. {
  526. if(obj)
  527. {
  528. std::cout << "deserialization: possible memory leak for '" << typeid(obj).name() << "'" << std::endl;
  529. obj = nullptr;
  530. }
  531. }
  532. else
  533. {
  534. if(obj)
  535. std::cout << "deserialization: possible memory leak for '" << typeid(obj).name() << "'" << std::endl;
  536. obj = new std::remove_pointer<T>::type();
  537. serialization::deserialize(*obj,iter);
  538. }
  539. }
  540. // std::shared_ptr and std::unique_ptr
  541. /*template <typename T>
  542. IGL_INLINE typename std::enable_if<serialization::is_smart_ptr<T>::value,size_t>::type getByteSize(const T& obj)
  543. {
  544. return getByteSize(obj.get());
  545. }
  546. template <typename T>
  547. IGL_INLINE typename std::enable_if<serialization::is_smart_ptr<T>::value>::type serialize(const T& obj,std::vector<char>& buffer,std::vector<char>::iterator& iter)
  548. {
  549. serialize(obj.get(),buffer,iter);
  550. }
  551. template <template<typename> class T0,typename T1>
  552. IGL_INLINE typename std::enable_if<serialization::is_smart_ptr<T0<T1> >::value>::type deserialize(T0<T1>& obj,std::vector<char>::const_iterator& iter)
  553. {
  554. bool isNullPtr;
  555. serialization::deserialize(isNullPtr,iter);
  556. if(isNullPtr)
  557. {
  558. obj.reset();
  559. }
  560. else
  561. {
  562. obj = T0<T1>(new T1());
  563. serialization::deserialize(*obj,iter);
  564. }
  565. }
  566. // std::weak_ptr
  567. template <typename T>
  568. IGL_INLINE size_t getByteSize(const std::weak_ptr<T>& obj)
  569. {
  570. return sizeof(size_t);
  571. }
  572. template <typename T>
  573. IGL_INLINE void serialize(const std::weak_ptr<T>& obj,std::vector<char>& buffer,std::vector<char>::iterator& iter)
  574. {
  575. }
  576. template <typename T>
  577. IGL_INLINE void deserialize(std::weak_ptr<T>& obj,std::vector<char>::const_iterator& iter)
  578. {
  579. }*/
  580. // functions to overload for non-intrusive serialization
  581. template <typename T>
  582. IGL_INLINE void serialize(const T& obj,std::vector<char>& buffer)
  583. {
  584. static_assert(false,"type is not serializable: derive from igl::Serializable or overload the function igl::serialization::serialize(const T& obj,std::vector<char>& buffer)");
  585. }
  586. template <typename T>
  587. IGL_INLINE void deserialize(T& obj,const std::vector<char>& buffer)
  588. {
  589. static_assert(false,"type is not serializable: derive from igl::Serializable or overload the function igl::serialization::deserialize(const T& obj,std::vector<char>& buffer)");
  590. }
  591. // helper functions
  592. template <typename T>
  593. IGL_INLINE void updateMemoryMap(T& obj,size_t size,std::map<std::uintptr_t,IndexedPointerBase*>& memoryMap)
  594. {
  595. // check if object is already serialized
  596. auto startPtr = new IndexedPointer<T>();
  597. startPtr->Object = &obj;
  598. auto startBasePtr = static_cast<IndexedPointerBase*>(startPtr);
  599. startBasePtr->Type = IndexedPointerBase::BEGIN;
  600. auto startAddress = reinterpret_cast<std::uintptr_t>(&obj);
  601. auto p = std::pair<std::uintptr_t,IndexedPointerBase*>(startAddress,startBasePtr);
  602. auto el = memoryMap.insert(p);
  603. auto iter = ++el.first; // next elememt
  604. if(el.second && (iter == memoryMap.end() || iter->second->Type != IndexedPointerBase::END))
  605. {
  606. // not yet serialized
  607. auto endPtr = new IndexedPointer<T>();
  608. auto endBasePtr = static_cast<IndexedPointerBase*>(endPtr);
  609. endBasePtr->Type = IndexedPointerBase::END;
  610. auto endAddress = reinterpret_cast<std::uintptr_t>(&obj) + size - 1;
  611. auto p = std::pair<std::uintptr_t,IndexedPointerBase*>(endAddress,endBasePtr);
  612. // insert end address
  613. memoryMap.insert(el.first,p);
  614. }
  615. else
  616. {
  617. // already serialized
  618. // remove inserted address
  619. memoryMap.erase(el.first);
  620. }
  621. }
  622. }
  623. }