bFile.cpp 37 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789
  1. /*
  2. bParse
  3. Copyright (c) 2006-2009 Charlie C & Erwin Coumans http://gamekit.googlecode.com
  4. This software is provided 'as-is', without any express or implied warranty.
  5. In no event will the authors be held liable for any damages arising from the use of this software.
  6. Permission is granted to anyone to use this software for any purpose,
  7. including commercial applications, and to alter it and redistribute it freely,
  8. subject to the following restrictions:
  9. 1. The origin of this software must not be misrepresented; you must not claim that you wrote the original software. If you use this software in a product, an acknowledgment in the product documentation would be appreciated but is not required.
  10. 2. Altered source versions must be plainly marked as such, and must not be misrepresented as being the original software.
  11. 3. This notice may not be removed or altered from any source distribution.
  12. */
  13. #include "bFile.h"
  14. #include "bCommon.h"
  15. #include "bChunk.h"
  16. #include "bDNA.h"
  17. #include <math.h>
  18. #include <string.h>
  19. #include <stdlib.h>
  20. #include "bDefines.h"
  21. #include "LinearMath/btSerializer.h"
  22. #include "LinearMath/btAlignedAllocator.h"
  23. #include "LinearMath/btMinMax.h"
  24. #define SIZEOFBLENDERHEADER 12
  25. #define MAX_ARRAY_LENGTH 512
  26. using namespace bParse;
  27. #define MAX_STRLEN 1024
  28. const char* getCleanName(const char* memName, char* buffer)
  29. {
  30. int slen = strlen(memName);
  31. assert(slen<MAX_STRLEN);
  32. slen=btMin(slen,MAX_STRLEN);
  33. for (int i=0;i<slen;i++)
  34. {
  35. if (memName[i]==']'||memName[i]=='[')
  36. {
  37. buffer[i] = 0;//'_';
  38. } else
  39. {
  40. buffer[i] = memName[i];
  41. }
  42. }
  43. buffer[slen]=0;
  44. return buffer;
  45. }
  46. int numallocs = 0;
  47. // ----------------------------------------------------- //
  48. bFile::bFile(const char *filename, const char headerString[7])
  49. : mOwnsBuffer(true),
  50. mFileBuffer(0),
  51. mFileLen(0),
  52. mVersion(0),
  53. mDataStart(0),
  54. mFileDNA(0),
  55. mMemoryDNA(0),
  56. mFlags(FD_INVALID)
  57. {
  58. for (int i=0;i<7;i++)
  59. {
  60. m_headerString[i] = headerString[i];
  61. }
  62. FILE *fp = fopen(filename, "rb");
  63. if (fp)
  64. {
  65. fseek(fp, 0L, SEEK_END);
  66. mFileLen = ftell(fp);
  67. fseek(fp, 0L, SEEK_SET);
  68. mFileBuffer = (char*)malloc(mFileLen+1);
  69. fread(mFileBuffer, mFileLen, 1, fp);
  70. fclose(fp);
  71. //
  72. parseHeader();
  73. }
  74. }
  75. // ----------------------------------------------------- //
  76. bFile::bFile( char *memoryBuffer, int len, const char headerString[7])
  77. : mOwnsBuffer(false),
  78. mFileBuffer(0),
  79. mFileLen(0),
  80. mVersion(0),
  81. mDataStart(0),
  82. mFileDNA(0),
  83. mMemoryDNA(0),
  84. mFlags(FD_INVALID)
  85. {
  86. for (int i=0;i<7;i++)
  87. {
  88. m_headerString[i] = headerString[i];
  89. }
  90. mFileBuffer = memoryBuffer;
  91. mFileLen = len;
  92. parseHeader();
  93. }
  94. // ----------------------------------------------------- //
  95. bFile::~bFile()
  96. {
  97. if (mOwnsBuffer && mFileBuffer)
  98. {
  99. free(mFileBuffer);
  100. mFileBuffer = 0;
  101. }
  102. delete mMemoryDNA;
  103. delete mFileDNA;
  104. }
  105. // ----------------------------------------------------- //
  106. void bFile::parseHeader()
  107. {
  108. if (!mFileLen || !mFileBuffer)
  109. return;
  110. char *blenderBuf = mFileBuffer;
  111. char header[SIZEOFBLENDERHEADER+1] ;
  112. memcpy(header, blenderBuf, SIZEOFBLENDERHEADER);
  113. header[SIZEOFBLENDERHEADER]='\0';
  114. if (strncmp(header, m_headerString, 6)!=0)
  115. {
  116. memcpy(header, m_headerString, SIZEOFBLENDERHEADER);
  117. return;
  118. }
  119. if (header[6] == 'd')
  120. {
  121. mFlags |= FD_DOUBLE_PRECISION;
  122. }
  123. char *ver = header+9;
  124. mVersion = atoi(ver);
  125. if (mVersion <= 241)
  126. {
  127. //printf("Warning, %d not fully tested : <= 242\n", mVersion);
  128. }
  129. int littleEndian= 1;
  130. littleEndian= ((char*)&littleEndian)[0];
  131. // swap ptr sizes...
  132. if (header[7]=='-')
  133. {
  134. mFlags |= FD_FILE_64;
  135. if (!VOID_IS_8)
  136. mFlags |= FD_BITS_VARIES;
  137. }
  138. else if (VOID_IS_8) mFlags |= FD_BITS_VARIES;
  139. // swap endian...
  140. if (header[8]=='V')
  141. {
  142. if (littleEndian ==1)
  143. mFlags |= FD_ENDIAN_SWAP;
  144. }
  145. else
  146. if (littleEndian==0)
  147. mFlags |= FD_ENDIAN_SWAP;
  148. mFlags |= FD_OK;
  149. }
  150. // ----------------------------------------------------- //
  151. bool bFile::ok()
  152. {
  153. return (mFlags &FD_OK)!=0;
  154. }
  155. void bFile::setFileDNA(int verboseMode, char* dnaBuffer, int dnaLen)
  156. {
  157. mFileDNA = new bDNA();
  158. ///mFileDNA->init will convert part of DNA file endianness to current CPU endianness if necessary
  159. mFileDNA->init((char*)dnaBuffer, dnaLen, (mFlags & FD_ENDIAN_SWAP)!=0);
  160. if (verboseMode & FD_VERBOSE_DUMP_DNA_TYPE_DEFINITIONS)
  161. mFileDNA->dumpTypeDefinitions();
  162. }
  163. // ----------------------------------------------------- //
  164. void bFile::parseInternal(int verboseMode, char* memDna,int memDnaLength)
  165. {
  166. if ( (mFlags &FD_OK) ==0)
  167. return;
  168. if (mFlags & FD_FILEDNA_IS_MEMDNA)
  169. {
  170. setFileDNA(verboseMode,memDna,memDnaLength);
  171. }
  172. if (mFileDNA==0)
  173. {
  174. char *blenderData = mFileBuffer;
  175. bChunkInd dna;
  176. dna.oldPtr = 0;
  177. char *tempBuffer = blenderData;
  178. for (int i=0; i<mFileLen; i++)
  179. {
  180. // looking for the data's starting position
  181. // and the start of SDNA decls
  182. if (!mDataStart && strncmp(tempBuffer, "REND", 4)==0)
  183. mDataStart = i;
  184. if (strncmp(tempBuffer, "DNA1", 4)==0)
  185. {
  186. // read the DNA1 block and extract SDNA
  187. if (getNextBlock(&dna, tempBuffer, mFlags) > 0)
  188. {
  189. if (strncmp((tempBuffer + ChunkUtils::getOffset(mFlags)), "SDNANAME", 8) ==0)
  190. dna.oldPtr = (tempBuffer + ChunkUtils::getOffset(mFlags));
  191. else dna.oldPtr = 0;
  192. }
  193. else dna.oldPtr = 0;
  194. }
  195. // Some Bullet files are missing the DNA1 block
  196. // In Blender it's DNA1 + ChunkUtils::getOffset() + SDNA + NAME
  197. // In Bullet tests its SDNA + NAME
  198. else if (strncmp(tempBuffer, "SDNANAME", 8) ==0)
  199. {
  200. dna.oldPtr = blenderData + i;
  201. dna.len = mFileLen-i;
  202. // Also no REND block, so exit now.
  203. if (mVersion==276) break;
  204. }
  205. if (mDataStart && dna.oldPtr) break;
  206. tempBuffer++;
  207. }
  208. if (!dna.oldPtr || !dna.len)
  209. {
  210. //printf("Failed to find DNA1+SDNA pair\n");
  211. mFlags &= ~FD_OK;
  212. return;
  213. }
  214. mFileDNA = new bDNA();
  215. ///mFileDNA->init will convert part of DNA file endianness to current CPU endianness if necessary
  216. mFileDNA->init((char*)dna.oldPtr, dna.len, (mFlags & FD_ENDIAN_SWAP)!=0);
  217. if (mVersion==276)
  218. {
  219. int i;
  220. for (i=0;i<mFileDNA->getNumNames();i++)
  221. {
  222. if (strcmp(mFileDNA->getName(i),"int")==0)
  223. {
  224. mFlags |= FD_BROKEN_DNA;
  225. }
  226. }
  227. if ((mFlags&FD_BROKEN_DNA)!=0)
  228. {
  229. //printf("warning: fixing some broken DNA version\n");
  230. }
  231. }
  232. if (verboseMode & FD_VERBOSE_DUMP_DNA_TYPE_DEFINITIONS)
  233. mFileDNA->dumpTypeDefinitions();
  234. }
  235. mMemoryDNA = new bDNA();
  236. int littleEndian= 1;
  237. littleEndian= ((char*)&littleEndian)[0];
  238. mMemoryDNA->init(memDna,memDnaLength,littleEndian==0);
  239. ///@todo we need a better version check, add version/sub version info from FileGlobal into memory DNA/header files
  240. if (mMemoryDNA->getNumNames() != mFileDNA->getNumNames())
  241. {
  242. mFlags |= FD_VERSION_VARIES;
  243. //printf ("Warning, file DNA is different than built in, performance is reduced. Best to re-export file with a matching version/platform");
  244. }
  245. // as long as it kept up to date it will be ok!!
  246. if (mMemoryDNA->lessThan(mFileDNA))
  247. {
  248. //printf ("Warning, file DNA is newer than built in.");
  249. }
  250. mFileDNA->initCmpFlags(mMemoryDNA);
  251. parseData();
  252. resolvePointers(verboseMode);
  253. updateOldPointers();
  254. }
  255. // ----------------------------------------------------- //
  256. void bFile::swap(char *head, bChunkInd& dataChunk, bool ignoreEndianFlag)
  257. {
  258. char *data = head;
  259. short *strc = mFileDNA->getStruct(dataChunk.dna_nr);
  260. const char s[] = "SoftBodyMaterialData";
  261. int szs = sizeof(s);
  262. if (strncmp((char*)&dataChunk.code,"ARAY",4)==0)
  263. {
  264. short *oldStruct = mFileDNA->getStruct(dataChunk.dna_nr);
  265. char *oldType = mFileDNA->getType(oldStruct[0]);
  266. if (strncmp(oldType,s,szs)==0)
  267. {
  268. return;
  269. }
  270. }
  271. int len = mFileDNA->getLength(strc[0]);
  272. for (int i=0; i<dataChunk.nr; i++)
  273. {
  274. swapStruct(dataChunk.dna_nr, data,ignoreEndianFlag);
  275. data+=len;
  276. }
  277. }
  278. void bFile::swapLen(char *dataPtr)
  279. {
  280. const bool VOID_IS_8 = ((sizeof(void*)==8));
  281. if (VOID_IS_8)
  282. {
  283. if (mFlags &FD_BITS_VARIES)
  284. {
  285. bChunkPtr4*c = (bChunkPtr4*) dataPtr;
  286. if ((c->code & 0xFFFF)==0)
  287. c->code >>=16;
  288. SWITCH_INT(c->len);
  289. SWITCH_INT(c->dna_nr);
  290. SWITCH_INT(c->nr);
  291. } else
  292. {
  293. bChunkPtr8* c = (bChunkPtr8*) dataPtr;
  294. if ((c->code & 0xFFFF)==0)
  295. c->code >>=16;
  296. SWITCH_INT(c->len);
  297. SWITCH_INT(c->dna_nr);
  298. SWITCH_INT(c->nr);
  299. }
  300. } else
  301. {
  302. if (mFlags &FD_BITS_VARIES)
  303. {
  304. bChunkPtr8*c = (bChunkPtr8*) dataPtr;
  305. if ((c->code & 0xFFFF)==0)
  306. c->code >>=16;
  307. SWITCH_INT(c->len);
  308. SWITCH_INT(c->dna_nr);
  309. SWITCH_INT(c->nr);
  310. } else
  311. {
  312. bChunkPtr4* c = (bChunkPtr4*) dataPtr;
  313. if ((c->code & 0xFFFF)==0)
  314. c->code >>=16;
  315. SWITCH_INT(c->len);
  316. SWITCH_INT(c->dna_nr);
  317. SWITCH_INT(c->nr);
  318. }
  319. }
  320. }
  321. void bFile::swapDNA(char* ptr)
  322. {
  323. bool swap = ((mFlags & FD_ENDIAN_SWAP)!=0);
  324. int offset = (mFlags & FD_FILE_64)? 24 : 20;
  325. char* data = &ptr[offset];
  326. // void bDNA::init(char *data, int len, bool swap)
  327. int *intPtr=0;short *shtPtr=0;
  328. char *cp = 0;int dataLen =0;long nr=0;
  329. intPtr = (int*)data;
  330. /*
  331. SDNA (4 bytes) (magic number)
  332. NAME (4 bytes)
  333. <nr> (4 bytes) amount of names (int)
  334. <string>
  335. <string>
  336. */
  337. if (strncmp(data, "SDNA", 4)==0)
  338. {
  339. // skip ++ NAME
  340. intPtr++;
  341. intPtr++;
  342. } else
  343. {
  344. if (strncmp(data+4, "SDNA", 4)==0)
  345. {
  346. // skip ++ NAME
  347. intPtr++;
  348. intPtr++;
  349. intPtr++;
  350. }
  351. }
  352. // Parse names
  353. if (swap)
  354. dataLen = ChunkUtils::swapInt(*intPtr);
  355. else
  356. dataLen = *intPtr;
  357. *intPtr = ChunkUtils::swapInt(*intPtr);
  358. intPtr++;
  359. cp = (char*)intPtr;
  360. int i;
  361. for ( i=0; i<dataLen; i++)
  362. {
  363. while (*cp)cp++;
  364. cp++;
  365. }
  366. {
  367. nr= (long)cp;
  368. //long mask=3;
  369. nr= ((nr+3)&~3)-nr;
  370. while (nr--)
  371. {
  372. cp++;
  373. }
  374. }
  375. /*
  376. TYPE (4 bytes)
  377. <nr> amount of types (int)
  378. <string>
  379. <string>
  380. */
  381. intPtr = (int*)cp;
  382. assert(strncmp(cp, "TYPE", 4)==0); intPtr++;
  383. if (swap)
  384. dataLen = ChunkUtils::swapInt(*intPtr);
  385. else
  386. dataLen = *intPtr;
  387. *intPtr = ChunkUtils::swapInt(*intPtr);
  388. intPtr++;
  389. cp = (char*)intPtr;
  390. for ( i=0; i<dataLen; i++)
  391. {
  392. while (*cp)cp++;
  393. cp++;
  394. }
  395. {
  396. nr= (long)cp;
  397. // long mask=3;
  398. nr= ((nr+3)&~3)-nr;
  399. while (nr--)
  400. {
  401. cp++;
  402. }
  403. }
  404. /*
  405. TLEN (4 bytes)
  406. <len> (short) the lengths of types
  407. <len>
  408. */
  409. // Parse type lens
  410. intPtr = (int*)cp;
  411. assert(strncmp(cp, "TLEN", 4)==0); intPtr++;
  412. shtPtr = (short*)intPtr;
  413. for ( i=0; i<dataLen; i++, shtPtr++)
  414. {
  415. //??????if (swap)
  416. shtPtr[0] = ChunkUtils::swapShort(shtPtr[0]);
  417. }
  418. if (dataLen & 1)
  419. shtPtr++;
  420. /*
  421. STRC (4 bytes)
  422. <nr> amount of structs (int)
  423. <typenr>
  424. <nr_of_elems>
  425. <typenr>
  426. <namenr>
  427. <typenr>
  428. <namenr>
  429. */
  430. intPtr = (int*)shtPtr;
  431. cp = (char*)intPtr;
  432. assert(strncmp(cp, "STRC", 4)==0);
  433. intPtr++;
  434. if (swap)
  435. dataLen = ChunkUtils::swapInt(*intPtr);
  436. else
  437. dataLen = *intPtr;
  438. *intPtr = ChunkUtils::swapInt(*intPtr);
  439. intPtr++;
  440. shtPtr = (short*)intPtr;
  441. for ( i=0; i<dataLen; i++)
  442. {
  443. //if (swap)
  444. {
  445. int len = shtPtr[1];
  446. shtPtr[0]= ChunkUtils::swapShort(shtPtr[0]);
  447. shtPtr[1]= ChunkUtils::swapShort(shtPtr[1]);
  448. shtPtr+= 2;
  449. for (int a=0; a<len; a++, shtPtr+=2)
  450. {
  451. shtPtr[0]= ChunkUtils::swapShort(shtPtr[0]);
  452. shtPtr[1]= ChunkUtils::swapShort(shtPtr[1]);
  453. }
  454. }
  455. // else
  456. // shtPtr+= (2*shtPtr[1])+2;
  457. }
  458. }
  459. void bFile::writeFile(const char* fileName)
  460. {
  461. FILE* f = fopen(fileName,"wb");
  462. fwrite(mFileBuffer,1,mFileLen,f);
  463. fclose(f);
  464. }
  465. void bFile::preSwap()
  466. {
  467. const bool brokenDNA = (mFlags&FD_BROKEN_DNA)!=0;
  468. //FD_ENDIAN_SWAP
  469. //byte 8 determines the endianness of the file, little (v) versus big (V)
  470. int littleEndian= 1;
  471. littleEndian= ((char*)&littleEndian)[0];
  472. if (mFileBuffer[8]=='V')
  473. {
  474. mFileBuffer[8]='v';
  475. }
  476. else
  477. {
  478. mFileBuffer[8]='V';
  479. }
  480. mDataStart = 12;
  481. char *dataPtr = mFileBuffer+mDataStart;
  482. bChunkInd dataChunk;
  483. dataChunk.code = 0;
  484. bool ignoreEndianFlag = true;
  485. //we always want to swap here
  486. int seek = getNextBlock(&dataChunk, dataPtr, mFlags);
  487. //dataPtr += ChunkUtils::getOffset(mFlags);
  488. char *dataPtrHead = 0;
  489. while (1)
  490. {
  491. // one behind
  492. if (dataChunk.code == SDNA || dataChunk.code==DNA1 || dataChunk.code == TYPE || dataChunk.code == TLEN || dataChunk.code==STRC)
  493. {
  494. swapDNA(dataPtr);
  495. break;
  496. } else
  497. {
  498. //if (dataChunk.code == DNA1) break;
  499. dataPtrHead = dataPtr+ChunkUtils::getOffset(mFlags);
  500. swapLen(dataPtr);
  501. if (dataChunk.dna_nr>=0)
  502. {
  503. swap(dataPtrHead, dataChunk,ignoreEndianFlag);
  504. } else
  505. {
  506. printf("unknown chunk\n");
  507. }
  508. }
  509. // next please!
  510. dataPtr += seek;
  511. seek = getNextBlock(&dataChunk, dataPtr, mFlags);
  512. if (seek < 0)
  513. break;
  514. }
  515. if (mFlags & FD_ENDIAN_SWAP)
  516. {
  517. mFlags &= ~FD_ENDIAN_SWAP;
  518. } else
  519. {
  520. mFlags |= FD_ENDIAN_SWAP;
  521. }
  522. }
  523. // ----------------------------------------------------- //
  524. char* bFile::readStruct(char *head, bChunkInd& dataChunk)
  525. {
  526. bool ignoreEndianFlag = false;
  527. if (mFlags & FD_ENDIAN_SWAP)
  528. swap(head, dataChunk, ignoreEndianFlag);
  529. if (!mFileDNA->flagEqual(dataChunk.dna_nr))
  530. {
  531. // Ouch! need to rebuild the struct
  532. short *oldStruct,*curStruct;
  533. char *oldType, *newType;
  534. int oldLen, curLen, reverseOld;
  535. oldStruct = mFileDNA->getStruct(dataChunk.dna_nr);
  536. oldType = mFileDNA->getType(oldStruct[0]);
  537. oldLen = mFileDNA->getLength(oldStruct[0]);
  538. if ((mFlags&FD_BROKEN_DNA)!=0)
  539. {
  540. if ((strcmp(oldType,"btQuantizedBvhNodeData")==0)&&oldLen==20)
  541. {
  542. return 0;
  543. }
  544. if ((strcmp(oldType,"btShortIntIndexData")==0))
  545. {
  546. int allocLen = 2;
  547. char *dataAlloc = new char[(dataChunk.nr*allocLen)+1];
  548. memset(dataAlloc, 0, (dataChunk.nr*allocLen)+1);
  549. short* dest = (short*) dataAlloc;
  550. const short* src = (short*) head;
  551. for (int i=0;i<dataChunk.nr;i++)
  552. {
  553. dest[i] = src[i];
  554. if (mFlags &FD_ENDIAN_SWAP)
  555. {
  556. SWITCH_SHORT(dest[i]);
  557. }
  558. }
  559. addDataBlock(dataAlloc);
  560. return dataAlloc;
  561. }
  562. }
  563. ///don't try to convert Link block data, just memcpy it. Other data can be converted.
  564. if (strcmp("Link",oldType)!=0)
  565. {
  566. reverseOld = mMemoryDNA->getReverseType(oldType);
  567. if ((reverseOld!=-1))
  568. {
  569. // make sure it's here
  570. //assert(reverseOld!= -1 && "getReverseType() returned -1, struct required!");
  571. //
  572. curStruct = mMemoryDNA->getStruct(reverseOld);
  573. newType = mMemoryDNA->getType(curStruct[0]);
  574. curLen = mMemoryDNA->getLength(curStruct[0]);
  575. // make sure it's the same
  576. assert((strcmp(oldType, newType)==0) && "internal error, struct mismatch!");
  577. numallocs++;
  578. // numBlocks * length
  579. int allocLen = (curLen);
  580. char *dataAlloc = new char[(dataChunk.nr*allocLen)+1];
  581. memset(dataAlloc, 0, (dataChunk.nr*allocLen));
  582. // track allocated
  583. addDataBlock(dataAlloc);
  584. char *cur = dataAlloc;
  585. char *old = head;
  586. for (int block=0; block<dataChunk.nr; block++)
  587. {
  588. bool fixupPointers = true;
  589. parseStruct(cur, old, dataChunk.dna_nr, reverseOld, fixupPointers);
  590. mLibPointers.insert(old,(bStructHandle*)cur);
  591. cur += curLen;
  592. old += oldLen;
  593. }
  594. return dataAlloc;
  595. }
  596. } else
  597. {
  598. //printf("Link found\n");
  599. }
  600. } else
  601. {
  602. //#define DEBUG_EQUAL_STRUCTS
  603. #ifdef DEBUG_EQUAL_STRUCTS
  604. short *oldStruct;
  605. char *oldType;
  606. oldStruct = mFileDNA->getStruct(dataChunk.dna_nr);
  607. oldType = mFileDNA->getType(oldStruct[0]);
  608. printf("%s equal structure, just memcpy\n",oldType);
  609. #endif //
  610. }
  611. char *dataAlloc = new char[(dataChunk.len)+1];
  612. memset(dataAlloc, 0, dataChunk.len+1);
  613. // track allocated
  614. addDataBlock(dataAlloc);
  615. memcpy(dataAlloc, head, dataChunk.len);
  616. return dataAlloc;
  617. }
  618. // ----------------------------------------------------- //
  619. void bFile::parseStruct(char *strcPtr, char *dtPtr, int old_dna, int new_dna, bool fixupPointers)
  620. {
  621. if (old_dna == -1) return;
  622. if (new_dna == -1) return;
  623. //disable this, because we need to fixup pointers/ListBase
  624. if (0)//mFileDNA->flagEqual(old_dna))
  625. {
  626. short *strc = mFileDNA->getStruct(old_dna);
  627. int len = mFileDNA->getLength(strc[0]);
  628. memcpy(strcPtr, dtPtr, len);
  629. return;
  630. }
  631. // Ok, now build the struct
  632. char *memType, *memName, *cpc, *cpo;
  633. short *fileStruct, *filePtrOld, *memoryStruct, *firstStruct;
  634. int elementLength, size, revType, old_nr, new_nr, fpLen;
  635. short firstStructType;
  636. // File to memory lookup
  637. memoryStruct = mMemoryDNA->getStruct(new_dna);
  638. fileStruct = mFileDNA->getStruct(old_dna);
  639. firstStruct = fileStruct;
  640. filePtrOld = fileStruct;
  641. firstStructType = mMemoryDNA->getStruct(0)[0];
  642. // Get number of elements
  643. elementLength = memoryStruct[1];
  644. memoryStruct+=2;
  645. cpc = strcPtr; cpo = 0;
  646. for (int ele=0; ele<elementLength; ele++, memoryStruct+=2)
  647. {
  648. memType = mMemoryDNA->getType(memoryStruct[0]);
  649. memName = mMemoryDNA->getName(memoryStruct[1]);
  650. size = mMemoryDNA->getElementSize(memoryStruct[0], memoryStruct[1]);
  651. revType = mMemoryDNA->getReverseType(memoryStruct[0]);
  652. if (revType != -1 && memoryStruct[0]>=firstStructType && memName[0] != '*')
  653. {
  654. cpo = getFileElement(firstStruct, memName, memType, dtPtr, &filePtrOld);
  655. if (cpo)
  656. {
  657. int arrayLen = mFileDNA->getArraySizeNew(filePtrOld[1]);
  658. old_nr = mFileDNA->getReverseType(memType);
  659. new_nr = revType;
  660. fpLen = mFileDNA->getElementSize(filePtrOld[0], filePtrOld[1]);
  661. if (arrayLen==1)
  662. {
  663. parseStruct(cpc, cpo, old_nr, new_nr,fixupPointers);
  664. } else
  665. {
  666. char* tmpCpc = cpc;
  667. char* tmpCpo = cpo;
  668. for (int i=0;i<arrayLen;i++)
  669. {
  670. parseStruct(tmpCpc, tmpCpo, old_nr, new_nr,fixupPointers);
  671. tmpCpc += size/arrayLen;
  672. tmpCpo += fpLen/arrayLen;
  673. }
  674. }
  675. cpc+=size;
  676. cpo+=fpLen;
  677. }
  678. else
  679. cpc+=size;
  680. }
  681. else
  682. {
  683. getMatchingFileDNA(fileStruct, memName, memType, cpc, dtPtr,fixupPointers);
  684. cpc+=size;
  685. }
  686. }
  687. }
  688. // ----------------------------------------------------- //
  689. static void getElement(int arrayLen, const char *cur, const char *old, char *oldPtr, char *curData)
  690. {
  691. #define getEle(value, current, type, cast, size, ptr)\
  692. if (strcmp(current, type)==0)\
  693. {\
  694. value = (*(cast*)ptr);\
  695. ptr += size;\
  696. }
  697. #define setEle(value, current, type, cast, size, ptr)\
  698. if (strcmp(current, type)==0)\
  699. {\
  700. (*(cast*)ptr) = (cast)value;\
  701. ptr += size;\
  702. }
  703. double value = 0.0;
  704. for (int i=0; i<arrayLen; i++)
  705. {
  706. getEle(value, old, "char", char, sizeof(char), oldPtr);
  707. setEle(value, cur, "char", char, sizeof(char), curData);
  708. getEle(value, old, "short", short, sizeof(short), oldPtr);
  709. setEle(value, cur, "short", short, sizeof(short), curData);
  710. getEle(value, old, "ushort", unsigned short, sizeof(unsigned short), oldPtr);
  711. setEle(value, cur, "ushort", unsigned short, sizeof(unsigned short), curData);
  712. getEle(value, old, "int", int, sizeof(int), oldPtr);
  713. setEle(value, cur, "int", int, sizeof(int), curData);
  714. getEle(value, old, "long", int, sizeof(int), oldPtr);
  715. setEle(value, cur, "long", int, sizeof(int), curData);
  716. getEle(value, old, "float", float, sizeof(float), oldPtr);
  717. setEle(value, cur, "float", float, sizeof(float), curData);
  718. getEle(value, old, "double", double, sizeof(double), oldPtr);
  719. setEle(value, cur, "double", double, sizeof(double), curData);
  720. }
  721. }
  722. // ----------------------------------------------------- //
  723. void bFile::swapData(char *data, short type, int arraySize,bool ignoreEndianFlag)
  724. {
  725. if (ignoreEndianFlag || (mFlags &FD_ENDIAN_SWAP))
  726. {
  727. if (type == 2 || type == 3)
  728. {
  729. short *sp = (short*)data;
  730. for (int i=0; i<arraySize; i++)
  731. {
  732. sp[0] = ChunkUtils::swapShort(sp[0]);
  733. sp++;
  734. }
  735. }
  736. if (type>3 && type <8)
  737. {
  738. char c;
  739. char *cp = data;
  740. for (int i=0; i<arraySize; i++)
  741. {
  742. c = cp[0];
  743. cp[0] = cp[3];
  744. cp[3] = c;
  745. c = cp[1];
  746. cp[1] = cp[2];
  747. cp[2] = c;
  748. cp+=4;
  749. }
  750. }
  751. }
  752. }
  753. void bFile::safeSwapPtr(char *dst, const char *src)
  754. {
  755. int ptrFile = mFileDNA->getPointerSize();
  756. int ptrMem = mMemoryDNA->getPointerSize();
  757. if (!src && !dst)
  758. return;
  759. if (ptrFile == ptrMem)
  760. {
  761. memcpy(dst, src, ptrMem);
  762. }
  763. else if (ptrMem==4 && ptrFile==8)
  764. {
  765. btPointerUid* oldPtr = (btPointerUid*)src;
  766. btPointerUid* newPtr = (btPointerUid*)dst;
  767. if (oldPtr->m_uniqueIds[0] == oldPtr->m_uniqueIds[1])
  768. {
  769. //Bullet stores the 32bit unique ID in both upper and lower part of 64bit pointers
  770. //so it can be used to distinguish between .blend and .bullet
  771. newPtr->m_uniqueIds[0] = oldPtr->m_uniqueIds[0];
  772. } else
  773. {
  774. //deal with pointers the Blender .blend style way, see
  775. //readfile.c in the Blender source tree
  776. long64 longValue = *((long64*)src);
  777. //endian swap for 64bit pointer otherwise truncation will fail due to trailing zeros
  778. if (mFlags & FD_ENDIAN_SWAP)
  779. SWITCH_LONGINT(longValue);
  780. *((int*)dst) = (int)(longValue>>3);
  781. }
  782. }
  783. else if (ptrMem==8 && ptrFile==4)
  784. {
  785. btPointerUid* oldPtr = (btPointerUid*)src;
  786. btPointerUid* newPtr = (btPointerUid*)dst;
  787. if (oldPtr->m_uniqueIds[0] == oldPtr->m_uniqueIds[1])
  788. {
  789. newPtr->m_uniqueIds[0] = oldPtr->m_uniqueIds[0];
  790. newPtr->m_uniqueIds[1] = 0;
  791. } else
  792. {
  793. *((long64*)dst)= *((int*)src);
  794. }
  795. }
  796. else
  797. {
  798. printf ("%d %d\n", ptrFile,ptrMem);
  799. assert(0 && "Invalid pointer len");
  800. }
  801. }
  802. // ----------------------------------------------------- //
  803. void bFile::getMatchingFileDNA(short* dna_addr, const char* lookupName, const char* lookupType, char *strcData, char *data, bool fixupPointers)
  804. {
  805. // find the matching memory dna data
  806. // to the file being loaded. Fill the
  807. // memory with the file data...
  808. int len = dna_addr[1];
  809. dna_addr+=2;
  810. for (int i=0; i<len; i++, dna_addr+=2)
  811. {
  812. const char* type = mFileDNA->getType(dna_addr[0]);
  813. const char* name = mFileDNA->getName(dna_addr[1]);
  814. int eleLen = mFileDNA->getElementSize(dna_addr[0], dna_addr[1]);
  815. if ((mFlags&FD_BROKEN_DNA)!=0)
  816. {
  817. if ((strcmp(type,"short")==0)&&(strcmp(name,"int")==0))
  818. {
  819. eleLen = 0;
  820. }
  821. }
  822. if (strcmp(lookupName, name)==0)
  823. {
  824. //int arrayLenold = mFileDNA->getArraySize((char*)name.c_str());
  825. int arrayLen = mFileDNA->getArraySizeNew(dna_addr[1]);
  826. //assert(arrayLenold == arrayLen);
  827. if (name[0] == '*')
  828. {
  829. // cast pointers
  830. int ptrFile = mFileDNA->getPointerSize();
  831. int ptrMem = mMemoryDNA->getPointerSize();
  832. safeSwapPtr(strcData,data);
  833. if (fixupPointers)
  834. {
  835. if (arrayLen > 1)
  836. {
  837. //void **sarray = (void**)strcData;
  838. //void **darray = (void**)data;
  839. char *cpc, *cpo;
  840. cpc = (char*)strcData;
  841. cpo = (char*)data;
  842. for (int a=0; a<arrayLen; a++)
  843. {
  844. safeSwapPtr(cpc, cpo);
  845. m_pointerFixupArray.push_back(cpc);
  846. cpc += ptrMem;
  847. cpo += ptrFile;
  848. }
  849. }
  850. else
  851. {
  852. if (name[1] == '*')
  853. m_pointerPtrFixupArray.push_back(strcData);
  854. else
  855. m_pointerFixupArray.push_back(strcData);
  856. }
  857. }
  858. else
  859. {
  860. // printf("skipped %s %s : %x\n",type.c_str(),name.c_str(),strcData);
  861. }
  862. }
  863. else if (strcmp(type, lookupType)==0)
  864. memcpy(strcData, data, eleLen);
  865. else
  866. getElement(arrayLen, lookupType, type, data, strcData);
  867. // --
  868. return;
  869. }
  870. data+=eleLen;
  871. }
  872. }
  873. // ----------------------------------------------------- //
  874. char* bFile::getFileElement(short *firstStruct, char *lookupName, char *lookupType, char *data, short **foundPos)
  875. {
  876. short *old = firstStruct;//mFileDNA->getStruct(old_nr);
  877. int elementLength = old[1];
  878. old+=2;
  879. for (int i=0; i<elementLength; i++, old+=2)
  880. {
  881. char* type = mFileDNA->getType(old[0]);
  882. char* name = mFileDNA->getName(old[1]);
  883. int len = mFileDNA->getElementSize(old[0], old[1]);
  884. if (strcmp(lookupName, name)==0)
  885. {
  886. if (strcmp(type, lookupType)==0)
  887. {
  888. if (foundPos)
  889. *foundPos = old;
  890. return data;
  891. }
  892. return 0;
  893. }
  894. data+=len;
  895. }
  896. return 0;
  897. }
  898. // ----------------------------------------------------- //
  899. void bFile::swapStruct(int dna_nr, char *data,bool ignoreEndianFlag)
  900. {
  901. if (dna_nr == -1) return;
  902. short *strc = mFileDNA->getStruct(dna_nr);
  903. //short *firstStrc = strc;
  904. int elementLen= strc[1];
  905. strc+=2;
  906. short first = mFileDNA->getStruct(0)[0];
  907. char *buf = data;
  908. for (int i=0; i<elementLen; i++, strc+=2)
  909. {
  910. char *type = mFileDNA->getType(strc[0]);
  911. char *name = mFileDNA->getName(strc[1]);
  912. int size = mFileDNA->getElementSize(strc[0], strc[1]);
  913. if (strc[0] >= first && name[0]!='*')
  914. {
  915. int old_nr = mFileDNA->getReverseType(type);
  916. int arrayLen = mFileDNA->getArraySizeNew(strc[1]);
  917. if (arrayLen==1)
  918. {
  919. swapStruct(old_nr,buf,ignoreEndianFlag);
  920. } else
  921. {
  922. char* tmpBuf = buf;
  923. for (int i=0;i<arrayLen;i++)
  924. {
  925. swapStruct(old_nr,tmpBuf,ignoreEndianFlag);
  926. tmpBuf+=size/arrayLen;
  927. }
  928. }
  929. }
  930. else
  931. {
  932. //int arrayLenOld = mFileDNA->getArraySize(name);
  933. int arrayLen = mFileDNA->getArraySizeNew(strc[1]);
  934. //assert(arrayLenOld == arrayLen);
  935. swapData(buf, strc[0], arrayLen,ignoreEndianFlag);
  936. }
  937. buf+=size;
  938. }
  939. }
  940. void bFile::resolvePointersMismatch()
  941. {
  942. // printf("resolvePointersStructMismatch\n");
  943. int i;
  944. for (i=0;i< m_pointerFixupArray.size();i++)
  945. {
  946. char* cur = m_pointerFixupArray.at(i);
  947. void** ptrptr = (void**) cur;
  948. void* ptr = *ptrptr;
  949. ptr = findLibPointer(ptr);
  950. if (ptr)
  951. {
  952. //printf("Fixup pointer!\n");
  953. *(ptrptr) = ptr;
  954. } else
  955. {
  956. // printf("pointer not found: %x\n",cur);
  957. }
  958. }
  959. for (i=0; i<m_pointerPtrFixupArray.size(); i++)
  960. {
  961. char* cur= m_pointerPtrFixupArray.at(i);
  962. void** ptrptr = (void**)cur;
  963. bChunkInd *block = m_chunkPtrPtrMap.find(*ptrptr);
  964. if (block)
  965. {
  966. int ptrMem = mMemoryDNA->getPointerSize();
  967. int ptrFile = mFileDNA->getPointerSize();
  968. int blockLen = block->len / ptrFile;
  969. void *onptr = findLibPointer(*ptrptr);
  970. if (onptr)
  971. {
  972. char *newPtr = new char[blockLen * ptrMem];
  973. addDataBlock(newPtr);
  974. memset(newPtr, 0, blockLen * ptrMem);
  975. void **onarray = (void**)onptr;
  976. char *oldPtr = (char*)onarray;
  977. int p = 0;
  978. while (blockLen-- > 0)
  979. {
  980. btPointerUid dp = {{0}};
  981. safeSwapPtr((char*)dp.m_uniqueIds, oldPtr);
  982. void **tptr = (void**)(newPtr + p * ptrMem);
  983. *tptr = findLibPointer(dp.m_ptr);
  984. oldPtr += ptrFile;
  985. ++p;
  986. }
  987. *ptrptr = newPtr;
  988. }
  989. }
  990. }
  991. }
  992. ///this loop only works fine if the Blender DNA structure of the file matches the headerfiles
  993. void bFile::resolvePointersChunk(const bChunkInd& dataChunk, int verboseMode)
  994. {
  995. bParse::bDNA* fileDna = mFileDNA ? mFileDNA : mMemoryDNA;
  996. short int* oldStruct = fileDna->getStruct(dataChunk.dna_nr);
  997. short oldLen = fileDna->getLength(oldStruct[0]);
  998. //char* structType = fileDna->getType(oldStruct[0]);
  999. char* cur = (char*)findLibPointer(dataChunk.oldPtr);
  1000. for (int block=0; block<dataChunk.nr; block++)
  1001. {
  1002. resolvePointersStructRecursive(cur,dataChunk.dna_nr, verboseMode,1);
  1003. cur += oldLen;
  1004. }
  1005. }
  1006. int bFile::resolvePointersStructRecursive(char *strcPtr, int dna_nr, int verboseMode,int recursion)
  1007. {
  1008. bParse::bDNA* fileDna = mFileDNA ? mFileDNA : mMemoryDNA;
  1009. char* memType;
  1010. char* memName;
  1011. short firstStructType = fileDna->getStruct(0)[0];
  1012. char* elemPtr= strcPtr;
  1013. short int* oldStruct = fileDna->getStruct(dna_nr);
  1014. int elementLength = oldStruct[1];
  1015. oldStruct+=2;
  1016. int totalSize = 0;
  1017. for (int ele=0; ele<elementLength; ele++, oldStruct+=2)
  1018. {
  1019. memType = fileDna->getType(oldStruct[0]);
  1020. memName = fileDna->getName(oldStruct[1]);
  1021. int arrayLen = fileDna->getArraySizeNew(oldStruct[1]);
  1022. if (memName[0] == '*')
  1023. {
  1024. if (arrayLen > 1)
  1025. {
  1026. void **array= (void**)elemPtr;
  1027. for (int a=0; a<arrayLen; a++)
  1028. {
  1029. if (verboseMode & FD_VERBOSE_EXPORT_XML)
  1030. {
  1031. for (int i=0;i<recursion;i++)
  1032. {
  1033. printf(" ");
  1034. }
  1035. //skip the *
  1036. printf("<%s type=\"pointer\"> ",&memName[1]);
  1037. printf("%d ", array[a]);
  1038. printf("</%s>\n",&memName[1]);
  1039. }
  1040. array[a] = findLibPointer(array[a]);
  1041. }
  1042. }
  1043. else
  1044. {
  1045. void** ptrptr = (void**) elemPtr;
  1046. void* ptr = *ptrptr;
  1047. if (verboseMode & FD_VERBOSE_EXPORT_XML)
  1048. {
  1049. for (int i=0;i<recursion;i++)
  1050. {
  1051. printf(" ");
  1052. }
  1053. printf("<%s type=\"pointer\"> ",&memName[1]);
  1054. printf("%d ", ptr);
  1055. printf("</%s>\n",&memName[1]);
  1056. }
  1057. ptr = findLibPointer(ptr);
  1058. if (ptr)
  1059. {
  1060. // printf("Fixup pointer at 0x%x from 0x%x to 0x%x!\n",ptrptr,*ptrptr,ptr);
  1061. *(ptrptr) = ptr;
  1062. if (memName[1] == '*' && ptrptr && *ptrptr)
  1063. {
  1064. // This will only work if the given **array is continuous
  1065. void **array= (void**)*(ptrptr);
  1066. void *np= array[0];
  1067. int n=0;
  1068. while (np)
  1069. {
  1070. np= findLibPointer(array[n]);
  1071. if (np) array[n]= np;
  1072. n++;
  1073. }
  1074. }
  1075. } else
  1076. {
  1077. // printf("Cannot fixup pointer at 0x%x from 0x%x to 0x%x!\n",ptrptr,*ptrptr,ptr);
  1078. }
  1079. }
  1080. } else
  1081. {
  1082. int revType = fileDna->getReverseType(oldStruct[0]);
  1083. if (oldStruct[0]>=firstStructType) //revType != -1 &&
  1084. {
  1085. char cleanName[MAX_STRLEN];
  1086. getCleanName(memName,cleanName);
  1087. int arrayLen = fileDna->getArraySizeNew(oldStruct[1]);
  1088. int byteOffset = 0;
  1089. if (verboseMode & FD_VERBOSE_EXPORT_XML)
  1090. {
  1091. for (int i=0;i<recursion;i++)
  1092. {
  1093. printf(" ");
  1094. }
  1095. if (arrayLen>1)
  1096. {
  1097. printf("<%s type=\"%s\" count=%d>\n",cleanName,memType, arrayLen);
  1098. } else
  1099. {
  1100. printf("<%s type=\"%s\">\n",cleanName,memType);
  1101. }
  1102. }
  1103. for (int i=0;i<arrayLen;i++)
  1104. {
  1105. byteOffset += resolvePointersStructRecursive(elemPtr+byteOffset,revType, verboseMode,recursion+1);
  1106. }
  1107. if (verboseMode & FD_VERBOSE_EXPORT_XML)
  1108. {
  1109. for (int i=0;i<recursion;i++)
  1110. {
  1111. printf(" ");
  1112. }
  1113. printf("</%s>\n",cleanName);
  1114. }
  1115. } else
  1116. {
  1117. //export a simple type
  1118. if (verboseMode & FD_VERBOSE_EXPORT_XML)
  1119. {
  1120. if (arrayLen>MAX_ARRAY_LENGTH)
  1121. {
  1122. printf("too long\n");
  1123. } else
  1124. {
  1125. //printf("%s %s\n",memType,memName);
  1126. bool isIntegerType = (strcmp(memType,"char")==0) || (strcmp(memType,"int")==0) || (strcmp(memType,"short")==0);
  1127. if (isIntegerType)
  1128. {
  1129. const char* newtype="int";
  1130. int dbarray[MAX_ARRAY_LENGTH];
  1131. int* dbPtr = 0;
  1132. char* tmp = elemPtr;
  1133. dbPtr = &dbarray[0];
  1134. if (dbPtr)
  1135. {
  1136. char cleanName[MAX_STRLEN];
  1137. getCleanName(memName,cleanName);
  1138. int i;
  1139. getElement(arrayLen, newtype,memType, tmp, (char*)dbPtr);
  1140. for (i=0;i<recursion;i++)
  1141. printf(" ");
  1142. if (arrayLen==1)
  1143. printf("<%s type=\"%s\">",cleanName,memType);
  1144. else
  1145. printf("<%s type=\"%s\" count=%d>",cleanName,memType,arrayLen);
  1146. for (i=0;i<arrayLen;i++)
  1147. printf(" %d ",dbPtr[i]);
  1148. printf("</%s>\n",cleanName);
  1149. }
  1150. } else
  1151. {
  1152. const char* newtype="double";
  1153. double dbarray[MAX_ARRAY_LENGTH];
  1154. double* dbPtr = 0;
  1155. char* tmp = elemPtr;
  1156. dbPtr = &dbarray[0];
  1157. if (dbPtr)
  1158. {
  1159. int i;
  1160. getElement(arrayLen, newtype,memType, tmp, (char*)dbPtr);
  1161. for (i=0;i<recursion;i++)
  1162. printf(" ");
  1163. char cleanName[MAX_STRLEN];
  1164. getCleanName(memName,cleanName);
  1165. if (arrayLen==1)
  1166. {
  1167. printf("<%s type=\"%s\">",memName,memType);
  1168. }
  1169. else
  1170. {
  1171. printf("<%s type=\"%s\" count=%d>",cleanName,memType,arrayLen);
  1172. }
  1173. for (i=0;i<arrayLen;i++)
  1174. printf(" %f ",dbPtr[i]);
  1175. printf("</%s>\n",cleanName);
  1176. }
  1177. }
  1178. }
  1179. }
  1180. }
  1181. }
  1182. int size = fileDna->getElementSize(oldStruct[0], oldStruct[1]);
  1183. totalSize += size;
  1184. elemPtr+=size;
  1185. }
  1186. return totalSize;
  1187. }
  1188. ///Resolve pointers replaces the original pointers in structures, and linked lists by the new in-memory structures
  1189. void bFile::resolvePointers(int verboseMode)
  1190. {
  1191. bParse::bDNA* fileDna = mFileDNA ? mFileDNA : mMemoryDNA;
  1192. //char *dataPtr = mFileBuffer+mDataStart;
  1193. if (1) //mFlags & (FD_BITS_VARIES | FD_VERSION_VARIES))
  1194. {
  1195. resolvePointersMismatch();
  1196. }
  1197. {
  1198. if (verboseMode & FD_VERBOSE_EXPORT_XML)
  1199. {
  1200. printf("<?xml version=\"1.0\" encoding=\"utf-8\"?>\n");
  1201. int numitems = m_chunks.size();
  1202. printf("<bullet_physics version=%d itemcount = %d>\n", btGetVersion(), numitems);
  1203. }
  1204. for (int i=0;i<m_chunks.size();i++)
  1205. {
  1206. const bChunkInd& dataChunk = m_chunks.at(i);
  1207. if (!mFileDNA || fileDna->flagEqual(dataChunk.dna_nr))
  1208. {
  1209. //dataChunk.len
  1210. short int* oldStruct = fileDna->getStruct(dataChunk.dna_nr);
  1211. char* oldType = fileDna->getType(oldStruct[0]);
  1212. if (verboseMode & FD_VERBOSE_EXPORT_XML)
  1213. printf(" <%s pointer=%d>\n",oldType,dataChunk.oldPtr);
  1214. resolvePointersChunk(dataChunk, verboseMode);
  1215. if (verboseMode & FD_VERBOSE_EXPORT_XML)
  1216. printf(" </%s>\n",oldType);
  1217. } else
  1218. {
  1219. //printf("skipping mStruct\n");
  1220. }
  1221. }
  1222. if (verboseMode & FD_VERBOSE_EXPORT_XML)
  1223. {
  1224. printf("</bullet_physics>\n");
  1225. }
  1226. }
  1227. }
  1228. // ----------------------------------------------------- //
  1229. void* bFile::findLibPointer(void *ptr)
  1230. {
  1231. bStructHandle** ptrptr = getLibPointers().find(ptr);
  1232. if (ptrptr)
  1233. return *ptrptr;
  1234. return 0;
  1235. }
  1236. void bFile::updateOldPointers()
  1237. {
  1238. int i;
  1239. for (i=0;i<m_chunks.size();i++)
  1240. {
  1241. bChunkInd& dataChunk = m_chunks[i];
  1242. dataChunk.oldPtr = findLibPointer(dataChunk.oldPtr);
  1243. }
  1244. }
  1245. void bFile::dumpChunks(bParse::bDNA* dna)
  1246. {
  1247. int i;
  1248. for (i=0;i<m_chunks.size();i++)
  1249. {
  1250. bChunkInd& dataChunk = m_chunks[i];
  1251. char* codeptr = (char*)&dataChunk.code;
  1252. char codestr[5] = {codeptr[0],codeptr[1],codeptr[2],codeptr[3],0};
  1253. short* newStruct = dna->getStruct(dataChunk.dna_nr);
  1254. char* typeName = dna->getType(newStruct[0]);
  1255. printf("%3d: %s ",i,typeName);
  1256. printf("code=%s ",codestr);
  1257. printf("ptr=%p ",dataChunk.oldPtr);
  1258. printf("len=%d ",dataChunk.len);
  1259. printf("nr=%d ",dataChunk.nr);
  1260. if (dataChunk.nr!=1)
  1261. {
  1262. printf("not 1\n");
  1263. }
  1264. printf("\n");
  1265. }
  1266. #if 0
  1267. IDFinderData ifd;
  1268. ifd.success = 0;
  1269. ifd.IDname = NULL;
  1270. ifd.just_print_it = 1;
  1271. for (i=0; i<bf->m_blocks.size(); ++i)
  1272. {
  1273. BlendBlock* bb = bf->m_blocks[i];
  1274. printf("tag='%s'\tptr=%p\ttype=%s\t[%4d]", bb->tag, bb,bf->types[bb->type_index].name,bb->m_array_entries_.size());
  1275. block_ID_finder(bb, bf, &ifd);
  1276. printf("\n");
  1277. }
  1278. #endif
  1279. }
  1280. void bFile::writeChunks(FILE* fp, bool fixupPointers)
  1281. {
  1282. bParse::bDNA* fileDna = mFileDNA ? mFileDNA : mMemoryDNA;
  1283. for (int i=0;i<m_chunks.size();i++)
  1284. {
  1285. bChunkInd& dataChunk = m_chunks.at(i);
  1286. // Ouch! need to rebuild the struct
  1287. short *oldStruct,*curStruct;
  1288. char *oldType, *newType;
  1289. int curLen, reverseOld;
  1290. oldStruct = fileDna->getStruct(dataChunk.dna_nr);
  1291. oldType = fileDna->getType(oldStruct[0]);
  1292. //int oldLen = fileDna->getLength(oldStruct[0]);
  1293. ///don't try to convert Link block data, just memcpy it. Other data can be converted.
  1294. reverseOld = mMemoryDNA->getReverseType(oldType);
  1295. if ((reverseOld!=-1))
  1296. {
  1297. // make sure it's here
  1298. //assert(reverseOld!= -1 && "getReverseType() returned -1, struct required!");
  1299. //
  1300. curStruct = mMemoryDNA->getStruct(reverseOld);
  1301. newType = mMemoryDNA->getType(curStruct[0]);
  1302. // make sure it's the same
  1303. assert((strcmp(oldType, newType)==0) && "internal error, struct mismatch!");
  1304. curLen = mMemoryDNA->getLength(curStruct[0]);
  1305. dataChunk.dna_nr = reverseOld;
  1306. if (strcmp("Link",oldType)!=0)
  1307. {
  1308. dataChunk.len = curLen * dataChunk.nr;
  1309. } else
  1310. {
  1311. // printf("keep length of link = %d\n",dataChunk.len);
  1312. }
  1313. //write the structure header
  1314. fwrite(&dataChunk,sizeof(bChunkInd),1,fp);
  1315. short int* curStruct1;
  1316. curStruct1 = mMemoryDNA->getStruct(dataChunk.dna_nr);
  1317. assert(curStruct1 == curStruct);
  1318. char* cur = fixupPointers ? (char*)findLibPointer(dataChunk.oldPtr) : (char*)dataChunk.oldPtr;
  1319. //write the actual contents of the structure(s)
  1320. fwrite(cur,dataChunk.len,1,fp);
  1321. } else
  1322. {
  1323. printf("serious error, struct mismatch: don't write\n");
  1324. }
  1325. }
  1326. }
  1327. // ----------------------------------------------------- //
  1328. int bFile::getNextBlock(bChunkInd *dataChunk, const char *dataPtr, const int flags)
  1329. {
  1330. bool swap = false;
  1331. bool varies = false;
  1332. if (flags &FD_ENDIAN_SWAP)
  1333. swap = true;
  1334. if (flags &FD_BITS_VARIES)
  1335. varies = true;
  1336. if (VOID_IS_8)
  1337. {
  1338. if (varies)
  1339. {
  1340. bChunkPtr4 head;
  1341. memcpy(&head, dataPtr, sizeof(bChunkPtr4));
  1342. bChunkPtr8 chunk;
  1343. chunk.code = head.code;
  1344. chunk.len = head.len;
  1345. chunk.m_uniqueInts[0] = head.m_uniqueInt;
  1346. chunk.m_uniqueInts[1] = 0;
  1347. chunk.dna_nr = head.dna_nr;
  1348. chunk.nr = head.nr;
  1349. if (swap)
  1350. {
  1351. if ((chunk.code & 0xFFFF)==0)
  1352. chunk.code >>=16;
  1353. SWITCH_INT(chunk.len);
  1354. SWITCH_INT(chunk.dna_nr);
  1355. SWITCH_INT(chunk.nr);
  1356. }
  1357. memcpy(dataChunk, &chunk, sizeof(bChunkInd));
  1358. }
  1359. else
  1360. {
  1361. bChunkPtr8 c;
  1362. memcpy(&c, dataPtr, sizeof(bChunkPtr8));
  1363. if (swap)
  1364. {
  1365. if ((c.code & 0xFFFF)==0)
  1366. c.code >>=16;
  1367. SWITCH_INT(c.len);
  1368. SWITCH_INT(c.dna_nr);
  1369. SWITCH_INT(c.nr);
  1370. }
  1371. memcpy(dataChunk, &c, sizeof(bChunkInd));
  1372. }
  1373. }
  1374. else
  1375. {
  1376. if (varies)
  1377. {
  1378. bChunkPtr8 head;
  1379. memcpy(&head, dataPtr, sizeof(bChunkPtr8));
  1380. bChunkPtr4 chunk;
  1381. chunk.code = head.code;
  1382. chunk.len = head.len;
  1383. if (head.m_uniqueInts[0]==head.m_uniqueInts[1])
  1384. {
  1385. chunk.m_uniqueInt = head.m_uniqueInts[0];
  1386. } else
  1387. {
  1388. long64 oldPtr =0;
  1389. memcpy(&oldPtr, &head.m_uniqueInts[0], 8);
  1390. if (swap)
  1391. SWITCH_LONGINT(oldPtr);
  1392. chunk.m_uniqueInt = (int)(oldPtr >> 3);
  1393. }
  1394. chunk.dna_nr = head.dna_nr;
  1395. chunk.nr = head.nr;
  1396. if (swap)
  1397. {
  1398. if ((chunk.code & 0xFFFF)==0)
  1399. chunk.code >>=16;
  1400. SWITCH_INT(chunk.len);
  1401. SWITCH_INT(chunk.dna_nr);
  1402. SWITCH_INT(chunk.nr);
  1403. }
  1404. memcpy(dataChunk, &chunk, sizeof(bChunkInd));
  1405. }
  1406. else
  1407. {
  1408. bChunkPtr4 c;
  1409. memcpy(&c, dataPtr, sizeof(bChunkPtr4));
  1410. if (swap)
  1411. {
  1412. if ((c.code & 0xFFFF)==0)
  1413. c.code >>=16;
  1414. SWITCH_INT(c.len);
  1415. SWITCH_INT(c.dna_nr);
  1416. SWITCH_INT(c.nr);
  1417. }
  1418. memcpy(dataChunk, &c, sizeof(bChunkInd));
  1419. }
  1420. }
  1421. if (dataChunk->len < 0)
  1422. return -1;
  1423. #if 0
  1424. print ("----------");
  1425. print (dataChunk->code);
  1426. print (dataChunk->len);
  1427. print (dataChunk->old);
  1428. print (dataChunk->dna_nr);
  1429. print (dataChunk->nr);
  1430. #endif
  1431. return (dataChunk->len+ChunkUtils::getOffset(flags));
  1432. }
  1433. //eof