/* * Copyright (C) 2005-2012 MaNGOS * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include #include #include #include "DB2FileLoader.h" DB2FileLoader::DB2FileLoader() { data = NULL; fieldsOffset = NULL; } bool DB2FileLoader::Load(const char *filename, const char *fmt) { uint32 header = 48; if(data) { delete [] data; data=NULL; } FILE * f = fopen(filename, "rb"); if(!f)return false; if(fread(&header, 4, 1, f) != 1) // Signature { fclose(f); return false; } EndianConvert(header); if(header != 0x32424457) { fclose(f); return false; //'WDB2' } if(fread(&recordCount,4,1,f)!=1) // Number of records { fclose(f); return false; } EndianConvert(recordCount); if(fread(&fieldCount,4,1,f)!=1) // Number of fields { fclose(f); return false; } EndianConvert(fieldCount); if(fread(&recordSize,4,1,f)!=1) // Size of a record { fclose(f); return false; } EndianConvert(recordSize); if(fread(&stringSize,4,1,f)!=1) // String size { fclose(f); return false; } EndianConvert(stringSize); /* NEW WDB2 FIELDS*/ if(fread(&tableHash,4,1,f)!=1) // Table hash { fclose(f); return false; } EndianConvert(tableHash); if(fread(&build,4,1,f)!=1) // Build { fclose(f); return false; } EndianConvert(build); if(fread(&unk1,4,1,f)!=1) // Unknown WDB2 { fclose(f); return false; } EndianConvert(unk1); if(fread(&unk2,4,1,f)!=1) // Unknown WDB2 { fclose(f); return false; } EndianConvert(unk2); if(fread(&unk3,4,1,f)!=1) // Unknown WDB2 { fclose(f); return false; } EndianConvert(unk3); if(fread(&locale,4,1,f)!=1) // Locales { fclose(f); return false; } EndianConvert(locale); if(fread(&unk5,4,1,f)!=1) // Unknown WDB2 { fclose(f); return false; } EndianConvert(unk5); fieldsOffset = new uint32[fieldCount]; fieldsOffset[0] = 0; for(uint32 i = 1; i < fieldCount; i++) { fieldsOffset[i] = fieldsOffset[i - 1]; if (fmt[i - 1] == 'b' || fmt[i - 1] == 'X') // byte fields fieldsOffset[i] += 1; else // 4 byte fields (int32/float/strings) fieldsOffset[i] += 4; } data = new unsigned char[recordSize*recordCount+stringSize]; stringTable = data + recordSize*recordCount; if(fread(data, recordSize * recordCount + stringSize, 1, f) != 1) { fclose(f); return false; } fclose(f); return true; } DB2FileLoader::~DB2FileLoader() { if(data) delete [] data; if(fieldsOffset) delete [] fieldsOffset; } DB2FileLoader::Record DB2FileLoader::getRecord(size_t id) { assert(data); return Record(*this, data + id*recordSize); } uint32 DB2FileLoader::GetFormatRecordSize(const char * format,int32* index_pos) { uint32 recordsize = 0; int32 i = -1; for(uint32 x = 0; format[x]; ++ x) { switch(format[x]) { case FT_FLOAT: recordsize += sizeof(float); break; case FT_INT: recordsize += sizeof(uint32); break; case FT_STRING: recordsize += sizeof(char*); break; case FT_SORT: i=x; break; case FT_IND: i=x; recordsize += sizeof(uint32); break; case FT_BYTE: recordsize += sizeof(uint8); break; case FT_LOGIC: assert(false && "2 files not have logic field type"); break; case FT_NA: case FT_NA_BYTE: break; default: assert(false && "unknown format character"); break; } } if (index_pos) *index_pos = i; return recordsize; } uint32 DB2FileLoader::GetFormatStringsFields(const char * format) { uint32 stringfields = 0; for(uint32 x=0; format[x];++x) if (format[x] == FT_STRING) ++stringfields; return stringfields; } char* DB2FileLoader::AutoProduceData(const char* format, uint32& records, char**& indexTable) { /* format STRING, NA, FLOAT,NA,INT <=> struct{ char* field0, float field1, int field2 }entry; this func will generate entry[rows] data; */ typedef char * ptr; if(strlen(format)!=fieldCount) return NULL; //get struct size and index pos int32 i; uint32 recordsize=GetFormatRecordSize(format,&i); if(i>=0) { uint32 maxi=0; //find max index for(uint32 y=0;ymaxi)maxi=ind; } ++maxi; records=maxi; indexTable=new ptr[maxi]; memset(indexTable,0,maxi*sizeof(ptr)); } else { records = recordCount; indexTable = new ptr[recordCount]; } char* dataTable= new char[recordCount*recordsize]; uint32 offset=0; for(uint32 y =0; y < recordCount; ++y) { if (i >= 0) { indexTable[getRecord(y).getUInt(i)]=&dataTable[offset]; } else indexTable[y]=&dataTable[offset]; for(uint32 x = 0; x < fieldCount; ++x) { switch(format[x]) { case FT_FLOAT: *((float*)(&dataTable[offset]))=getRecord(y).getFloat(x); offset += sizeof(float); break; case FT_IND: case FT_INT: *((uint32*)(&dataTable[offset]))=getRecord(y).getUInt(x); offset += sizeof(uint32); break; case FT_BYTE: *((uint8*)(&dataTable[offset]))=getRecord(y).getUInt8(x); offset += sizeof(uint8); break; case FT_STRING: *((char**)(&dataTable[offset]))=NULL; // will be replaces non-empty or "" strings in AutoProduceStrings offset += sizeof(char*); break; case FT_LOGIC: assert(false && "DBC files not have logic field type"); break; case FT_NA: case FT_NA_BYTE: case FT_SORT: break; default: assert(false && "unknown format character"); break; } } } return dataTable; } static char const* const nullStr = ""; char* DB2FileLoader::AutoProduceStringsArrayHolders(const char* format, char* dataTable) { if(strlen(format)!=fieldCount) return NULL; // we store flat holders pool as single memory block size_t stringFields = GetFormatStringsFields(format); // each string field at load have array of string for each locale size_t stringHolderSize = sizeof(char*) * MAX_LOCALE; size_t stringHoldersRecordPoolSize = stringFields * stringHolderSize; size_t stringHoldersPoolSize = stringHoldersRecordPoolSize * recordCount; char* stringHoldersPool= new char[stringHoldersPoolSize]; // dbc strings expected to have at least empty string for(size_t i = 0; i < stringHoldersPoolSize / sizeof(char*); ++i) ((char const**)stringHoldersPool)[i] = nullStr; uint32 offset=0; // assign string holders to string field slots for(uint32 y =0;y