2019-09-19 22:42:45 +00:00
|
|
|
//-------------------------------------------------------------------------
|
|
|
|
/*
|
|
|
|
Copyright (C) 2010-2019 EDuke32 developers and contributors
|
|
|
|
Copyright (C) 2019 Nuke.YKT
|
|
|
|
|
|
|
|
This file is part of NBlood.
|
|
|
|
|
|
|
|
NBlood is free software; you can redistribute it and/or
|
|
|
|
modify it under the terms of the GNU General Public License version 2
|
|
|
|
as published by the Free Software Foundation.
|
|
|
|
|
|
|
|
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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*/
|
|
|
|
//-------------------------------------------------------------------------
|
2019-09-21 18:59:54 +00:00
|
|
|
|
|
|
|
#include "ns.h" // Must come before everything else!
|
|
|
|
|
2019-09-19 22:42:45 +00:00
|
|
|
#include "build.h"
|
|
|
|
#include "compat.h"
|
|
|
|
#include "common_game.h"
|
2019-09-22 21:15:46 +00:00
|
|
|
#include "crc32_.h"
|
2019-09-19 22:42:45 +00:00
|
|
|
|
2019-06-27 04:33:22 +00:00
|
|
|
//#include "actor.h"
|
|
|
|
#include "globals.h"
|
2019-09-19 22:42:45 +00:00
|
|
|
#include "db.h"
|
|
|
|
#include "iob.h"
|
2019-10-07 19:29:52 +00:00
|
|
|
#include "eventq.h"
|
2019-09-19 22:42:45 +00:00
|
|
|
|
2019-09-22 06:39:22 +00:00
|
|
|
BEGIN_BLD_NS
|
|
|
|
|
2019-09-19 22:42:45 +00:00
|
|
|
unsigned short gStatCount[kMaxStatus + 1];
|
|
|
|
|
|
|
|
XSPRITE xsprite[kMaxXSprites];
|
|
|
|
XSECTOR xsector[kMaxXSectors];
|
|
|
|
XWALL xwall[kMaxXWalls];
|
|
|
|
|
2019-06-27 04:33:22 +00:00
|
|
|
SPRITEHIT gSpriteHit[kMaxXSprites];
|
|
|
|
|
2019-09-19 22:42:45 +00:00
|
|
|
int xvel[kMaxSprites], yvel[kMaxSprites], zvel[kMaxSprites];
|
|
|
|
|
2019-06-28 12:24:46 +00:00
|
|
|
char qsprite_filler[kMaxSprites], qsector_filler[kMaxSectors];
|
|
|
|
|
2019-09-19 22:42:45 +00:00
|
|
|
int gVisibility;
|
2019-10-07 19:29:52 +00:00
|
|
|
bool gModernMap = false;
|
2019-09-19 22:42:45 +00:00
|
|
|
const char *gItemText[] = {
|
|
|
|
"Skull Key",
|
|
|
|
"Eye Key",
|
|
|
|
"Fire Key",
|
|
|
|
"Dagger Key",
|
|
|
|
"Spider Key",
|
|
|
|
"Moon Key",
|
|
|
|
"Key 7",
|
|
|
|
"Doctor's Bag",
|
|
|
|
"Medicine Pouch",
|
|
|
|
"Life Essence",
|
|
|
|
"Life Seed",
|
|
|
|
"Red Potion",
|
|
|
|
"Feather Fall",
|
|
|
|
"Limited Invisibility",
|
|
|
|
"INVULNERABILITY",
|
|
|
|
"Boots of Jumping",
|
|
|
|
"Raven Flight",
|
|
|
|
"Guns Akimbo",
|
|
|
|
"Diving Suit",
|
|
|
|
"Gas mask",
|
|
|
|
"Clone",
|
|
|
|
"Crystal Ball",
|
|
|
|
"Decoy",
|
|
|
|
"Doppleganger",
|
|
|
|
"Reflective shots",
|
|
|
|
"Beast Vision",
|
|
|
|
"ShadowCloak",
|
|
|
|
"Rage shroom",
|
|
|
|
"Delirium Shroom",
|
|
|
|
"Grow shroom",
|
|
|
|
"Shrink shroom",
|
|
|
|
"Death mask",
|
|
|
|
"Wine Goblet",
|
|
|
|
"Wine Bottle",
|
|
|
|
"Skull Grail",
|
|
|
|
"Silver Grail",
|
|
|
|
"Tome",
|
|
|
|
"Black Chest",
|
|
|
|
"Wooden Chest",
|
|
|
|
"Asbestos Armor",
|
|
|
|
"Basic Armor",
|
|
|
|
"Body Armor",
|
|
|
|
"Fire Armor",
|
|
|
|
"Spirit Armor",
|
|
|
|
"Super Armor",
|
|
|
|
"Blue Team Base",
|
|
|
|
"Red Team Base",
|
|
|
|
"Blue Flag",
|
|
|
|
"Red Flag",
|
|
|
|
"DUMMY",
|
|
|
|
"Level map",
|
|
|
|
};
|
|
|
|
|
|
|
|
const char *gAmmoText[] = {
|
|
|
|
"Spray can",
|
|
|
|
"Bundle of TNT*",
|
|
|
|
"Bundle of TNT",
|
|
|
|
"Case of TNT",
|
|
|
|
"Proximity Detonator",
|
|
|
|
"Remote Detonator",
|
|
|
|
"Trapped Soul",
|
|
|
|
"4 shotgun shells",
|
|
|
|
"Box of shotgun shells",
|
|
|
|
"A few bullets",
|
|
|
|
"Voodoo Doll",
|
|
|
|
"OBSOLETE",
|
|
|
|
"Full drum of bullets",
|
|
|
|
"Tesla Charge",
|
|
|
|
"OBSOLETE",
|
|
|
|
"OBSOLETE",
|
|
|
|
"Flares",
|
|
|
|
"OBSOLETE",
|
|
|
|
"OBSOLETE",
|
|
|
|
"Gasoline Can",
|
|
|
|
NULL,
|
|
|
|
};
|
|
|
|
|
|
|
|
const char *gWeaponText[] = {
|
|
|
|
"RANDOM",
|
|
|
|
"Sawed-off",
|
|
|
|
"Tommy Gun",
|
|
|
|
"Flare Pistol",
|
|
|
|
"Voodoo Doll",
|
|
|
|
"Tesla Cannon",
|
|
|
|
"Napalm Launcher",
|
|
|
|
"Pitchfork",
|
|
|
|
"Spray Can",
|
|
|
|
"Dynamite",
|
|
|
|
"Life Leech",
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void dbCrypt(char *pPtr, int nLength, int nKey)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < nLength; i++)
|
|
|
|
{
|
|
|
|
pPtr[i] = pPtr[i] ^ nKey;
|
|
|
|
nKey++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void InsertSpriteSect(int nSprite, int nSector)
|
|
|
|
{
|
|
|
|
dassert(nSprite >= 0 && nSprite < kMaxSprites);
|
|
|
|
dassert(nSector >= 0 && nSector < kMaxSectors);
|
|
|
|
int nOther = headspritesect[nSector];
|
|
|
|
if (nOther >= 0)
|
|
|
|
{
|
|
|
|
prevspritesect[nSprite] = prevspritesect[nOther];
|
|
|
|
nextspritesect[nSprite] = -1;
|
|
|
|
nextspritesect[prevspritesect[nOther]] = nSprite;
|
|
|
|
prevspritesect[nOther] = nSprite;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
prevspritesect[nSprite] = nSprite;
|
|
|
|
nextspritesect[nSprite] = -1;
|
|
|
|
headspritesect[nSector] = nSprite;
|
|
|
|
}
|
|
|
|
sprite[nSprite].sectnum = nSector;
|
|
|
|
}
|
|
|
|
|
|
|
|
void RemoveSpriteSect(int nSprite)
|
|
|
|
{
|
|
|
|
dassert(nSprite >= 0 && nSprite < kMaxSprites);
|
|
|
|
int nSector = sprite[nSprite].sectnum;
|
|
|
|
dassert(nSector >= 0 && nSector < kMaxSectors);
|
|
|
|
int nOther = nextspritesect[nSprite];
|
|
|
|
if (nOther < 0)
|
|
|
|
{
|
|
|
|
nOther = headspritesect[nSector];
|
|
|
|
}
|
|
|
|
prevspritesect[nOther] = prevspritesect[nSprite];
|
|
|
|
if (headspritesect[nSector] != nSprite)
|
|
|
|
{
|
|
|
|
nextspritesect[prevspritesect[nSprite]] = nextspritesect[nSprite];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
headspritesect[nSector] = nextspritesect[nSprite];
|
|
|
|
}
|
|
|
|
sprite[nSprite].sectnum = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void InsertSpriteStat(int nSprite, int nStat)
|
|
|
|
{
|
|
|
|
dassert(nSprite >= 0 && nSprite < kMaxSprites);
|
|
|
|
dassert(nStat >= 0 && nStat <= kMaxStatus);
|
|
|
|
int nOther = headspritestat[nStat];
|
|
|
|
if (nOther >= 0)
|
|
|
|
{
|
|
|
|
prevspritestat[nSprite] = prevspritestat[nOther];
|
|
|
|
nextspritestat[nSprite] = -1;
|
|
|
|
nextspritestat[prevspritestat[nOther]] = nSprite;
|
|
|
|
prevspritestat[nOther] = nSprite;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
prevspritestat[nSprite] = nSprite;
|
|
|
|
nextspritestat[nSprite] = -1;
|
|
|
|
headspritestat[nStat] = nSprite;
|
|
|
|
}
|
|
|
|
sprite[nSprite].statnum = nStat;
|
|
|
|
gStatCount[nStat]++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void RemoveSpriteStat(int nSprite)
|
|
|
|
{
|
|
|
|
dassert(nSprite >= 0 && nSprite < kMaxSprites);
|
|
|
|
int nStat = sprite[nSprite].statnum;
|
|
|
|
dassert(nStat >= 0 && nStat <= kMaxStatus);
|
|
|
|
int nOther = nextspritestat[nSprite];
|
|
|
|
if (nOther < 0)
|
|
|
|
{
|
|
|
|
nOther = headspritestat[nStat];
|
|
|
|
}
|
|
|
|
prevspritestat[nOther] = prevspritestat[nSprite];
|
|
|
|
if (headspritestat[nStat] != nSprite)
|
|
|
|
{
|
|
|
|
nextspritestat[prevspritestat[nSprite]] = nextspritestat[nSprite];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
headspritestat[nStat] = nextspritestat[nSprite];
|
|
|
|
}
|
2019-10-07 19:29:52 +00:00
|
|
|
sprite[nSprite].statnum = kStatNothing;
|
2019-09-19 22:42:45 +00:00
|
|
|
gStatCount[nStat]--;
|
|
|
|
}
|
|
|
|
|
|
|
|
void qinitspritelists(void) // Replace
|
|
|
|
{
|
|
|
|
for (short i = 0; i <= kMaxSectors; i++)
|
|
|
|
{
|
|
|
|
headspritesect[i] = -1;
|
|
|
|
}
|
|
|
|
for (short i = 0; i <= kMaxStatus; i++)
|
|
|
|
{
|
|
|
|
headspritestat[i] = -1;
|
|
|
|
}
|
2019-06-28 12:24:46 +00:00
|
|
|
int const nMaxSprites = bVanilla ? 4096 : kMaxSprites;
|
|
|
|
for (short i = 0; i < nMaxSprites; i++)
|
2019-09-19 22:42:45 +00:00
|
|
|
{
|
|
|
|
sprite[i].sectnum = -1;
|
|
|
|
sprite[i].index = -1;
|
|
|
|
InsertSpriteStat(i, kMaxStatus);
|
|
|
|
}
|
|
|
|
memset(gStatCount, 0, sizeof(gStatCount));
|
2019-06-29 12:24:23 +00:00
|
|
|
Numsprites = 0;
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int InsertSprite(int nSector, int nStat)
|
|
|
|
{
|
|
|
|
int nSprite = headspritestat[kMaxStatus];
|
|
|
|
dassert(nSprite < kMaxSprites);
|
|
|
|
if (nSprite < 0)
|
|
|
|
{
|
|
|
|
return nSprite;
|
|
|
|
}
|
|
|
|
RemoveSpriteStat(nSprite);
|
|
|
|
spritetype *pSprite = &sprite[nSprite];
|
|
|
|
memset(&sprite[nSprite], 0, sizeof(spritetype));
|
|
|
|
InsertSpriteStat(nSprite, nStat);
|
|
|
|
InsertSpriteSect(nSprite, nSector);
|
|
|
|
pSprite->cstat = 128;
|
|
|
|
pSprite->clipdist = 32;
|
|
|
|
pSprite->xrepeat = pSprite->yrepeat = 64;
|
|
|
|
pSprite->owner = -1;
|
|
|
|
pSprite->extra = -1;
|
|
|
|
pSprite->index = nSprite;
|
|
|
|
xvel[nSprite] = yvel[nSprite] = zvel[nSprite] = 0;
|
|
|
|
|
2019-06-29 12:24:23 +00:00
|
|
|
Numsprites++;
|
|
|
|
|
2019-09-19 22:42:45 +00:00
|
|
|
return nSprite;
|
|
|
|
}
|
|
|
|
|
|
|
|
int qinsertsprite(short nSector, short nStat) // Replace
|
|
|
|
{
|
|
|
|
return InsertSprite(nSector, nStat);
|
|
|
|
}
|
|
|
|
|
|
|
|
int DeleteSprite(int nSprite)
|
|
|
|
{
|
|
|
|
if (sprite[nSprite].extra > 0)
|
|
|
|
{
|
|
|
|
dbDeleteXSprite(sprite[nSprite].extra);
|
|
|
|
}
|
|
|
|
dassert(sprite[nSprite].statnum >= 0 && sprite[nSprite].statnum < kMaxStatus);
|
|
|
|
RemoveSpriteStat(nSprite);
|
|
|
|
dassert(sprite[nSprite].sectnum >= 0 && sprite[nSprite].sectnum < kMaxSectors);
|
|
|
|
RemoveSpriteSect(nSprite);
|
|
|
|
InsertSpriteStat(nSprite, kMaxStatus);
|
|
|
|
|
2019-06-29 12:24:23 +00:00
|
|
|
Numsprites--;
|
|
|
|
|
2019-09-19 22:42:45 +00:00
|
|
|
return nSprite;
|
|
|
|
}
|
|
|
|
|
|
|
|
int qdeletesprite(short nSprite) // Replace
|
|
|
|
{
|
|
|
|
return DeleteSprite(nSprite);
|
|
|
|
}
|
|
|
|
|
|
|
|
int ChangeSpriteSect(int nSprite, int nSector)
|
|
|
|
{
|
|
|
|
dassert(nSprite >= 0 && nSprite < kMaxSprites);
|
|
|
|
dassert(nSector >= 0 && nSector < kMaxSectors);
|
|
|
|
dassert(sprite[nSprite].sectnum >= 0 && sprite[nSprite].sectnum < kMaxSectors);
|
|
|
|
RemoveSpriteSect(nSprite);
|
|
|
|
InsertSpriteSect(nSprite, nSector);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int qchangespritesect(short nSprite, short nSector)
|
|
|
|
{
|
|
|
|
return ChangeSpriteSect(nSprite, nSector);
|
|
|
|
}
|
|
|
|
|
|
|
|
int ChangeSpriteStat(int nSprite, int nStatus)
|
|
|
|
{
|
|
|
|
dassert(nSprite >= 0 && nSprite < kMaxSprites);
|
|
|
|
dassert(nStatus >= 0 && nStatus < kMaxStatus);
|
|
|
|
dassert(sprite[nSprite].statnum >= 0 && sprite[nSprite].statnum < kMaxStatus);
|
|
|
|
dassert(sprite[nSprite].sectnum >= 0 && sprite[nSprite].sectnum < kMaxSectors);
|
|
|
|
RemoveSpriteStat(nSprite);
|
|
|
|
InsertSpriteStat(nSprite, nStatus);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int qchangespritestat(short nSprite, short nStatus)
|
|
|
|
{
|
|
|
|
return ChangeSpriteStat(nSprite, nStatus);
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned short nextXSprite[kMaxXSprites];
|
|
|
|
unsigned short nextXWall[kMaxXWalls];
|
|
|
|
unsigned short nextXSector[kMaxXSectors];
|
|
|
|
|
|
|
|
void InitFreeList(unsigned short *pList, int nCount)
|
|
|
|
{
|
|
|
|
for (int i = 1; i < nCount; i++)
|
|
|
|
{
|
|
|
|
pList[i] = i-1;
|
|
|
|
}
|
|
|
|
pList[0] = nCount - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void InsertFree(unsigned short *pList, int nIndex)
|
|
|
|
{
|
|
|
|
pList[nIndex] = pList[0];
|
|
|
|
pList[0] = nIndex;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned short dbInsertXSprite(int nSprite)
|
|
|
|
{
|
|
|
|
int nXSprite = nextXSprite[0];
|
|
|
|
nextXSprite[0] = nextXSprite[nXSprite];
|
|
|
|
if (nXSprite == 0)
|
|
|
|
{
|
|
|
|
ThrowError("Out of free XSprites");
|
|
|
|
}
|
|
|
|
memset(&xsprite[nXSprite], 0, sizeof(XSPRITE));
|
|
|
|
if (!bVanilla)
|
|
|
|
memset(&gSpriteHit[nXSprite], 0, sizeof(SPRITEHIT));
|
|
|
|
xsprite[nXSprite].reference = nSprite;
|
|
|
|
sprite[nSprite].extra = nXSprite;
|
|
|
|
return nXSprite;
|
|
|
|
}
|
|
|
|
|
|
|
|
void dbDeleteXSprite(int nXSprite)
|
|
|
|
{
|
|
|
|
dassert(xsprite[nXSprite].reference >= 0);
|
|
|
|
dassert(sprite[xsprite[nXSprite].reference].extra == nXSprite);
|
|
|
|
InsertFree(nextXSprite, nXSprite);
|
|
|
|
sprite[xsprite[nXSprite].reference].extra = -1;
|
|
|
|
xsprite[nXSprite].reference = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned short dbInsertXWall(int nWall)
|
|
|
|
{
|
|
|
|
int nXWall = nextXWall[0];
|
|
|
|
nextXWall[0] = nextXWall[nXWall];
|
|
|
|
if (nXWall == 0)
|
|
|
|
{
|
|
|
|
ThrowError("Out of free XWalls");
|
|
|
|
}
|
|
|
|
memset(&xwall[nXWall], 0, sizeof(XWALL));
|
|
|
|
xwall[nXWall].reference = nWall;
|
|
|
|
wall[nWall].extra = nXWall;
|
|
|
|
return nXWall;
|
|
|
|
}
|
|
|
|
|
|
|
|
void dbDeleteXWall(int nXWall)
|
|
|
|
{
|
|
|
|
dassert(xwall[nXWall].reference >= 0);
|
|
|
|
InsertFree(nextXWall, nXWall);
|
|
|
|
wall[xwall[nXWall].reference].extra = -1;
|
|
|
|
xwall[nXWall].reference = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned short dbInsertXSector(int nSector)
|
|
|
|
{
|
|
|
|
int nXSector = nextXSector[0];
|
|
|
|
nextXSector[0] = nextXSector[nXSector];
|
|
|
|
if (nXSector == 0)
|
|
|
|
{
|
|
|
|
ThrowError("Out of free XSectors");
|
|
|
|
}
|
|
|
|
memset(&xsector[nXSector], 0, sizeof(XSECTOR));
|
|
|
|
xsector[nXSector].reference = nSector;
|
|
|
|
sector[nSector].extra = nXSector;
|
|
|
|
return nXSector;
|
|
|
|
}
|
|
|
|
|
|
|
|
void dbDeleteXSector(int nXSector)
|
|
|
|
{
|
|
|
|
dassert(xsector[nXSector].reference >= 0);
|
|
|
|
InsertFree(nextXSector, nXSector);
|
|
|
|
sector[xsector[nXSector].reference].extra = -1;
|
|
|
|
xsector[nXSector].reference = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void dbXSpriteClean(void)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < kMaxSprites; i++)
|
|
|
|
{
|
|
|
|
int nXSprite = sprite[i].extra;
|
|
|
|
if (nXSprite == 0)
|
|
|
|
{
|
|
|
|
sprite[i].extra = -1;
|
|
|
|
}
|
|
|
|
if (sprite[i].statnum < kMaxStatus && nXSprite > 0)
|
|
|
|
{
|
|
|
|
dassert(nXSprite < kMaxXSprites);
|
|
|
|
if (xsprite[nXSprite].reference != i)
|
|
|
|
{
|
|
|
|
int nXSprite2 = dbInsertXSprite(i);
|
|
|
|
memcpy(&xsprite[nXSprite2], &xsprite[nXSprite], sizeof(XSPRITE));
|
|
|
|
xsprite[nXSprite2].reference = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (int i = 1; i < kMaxXSprites; i++)
|
|
|
|
{
|
|
|
|
int nSprite = xsprite[i].reference;
|
|
|
|
if (nSprite >= 0)
|
|
|
|
{
|
|
|
|
dassert(nSprite < kMaxSprites);
|
|
|
|
if (sprite[nSprite].statnum >= kMaxStatus || sprite[nSprite].extra != i)
|
|
|
|
{
|
|
|
|
InsertFree(nextXSprite, i);
|
|
|
|
xsprite[i].reference = -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void dbXWallClean(void)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < numwalls; i++)
|
|
|
|
{
|
|
|
|
int nXWall = wall[i].extra;
|
|
|
|
if (nXWall == 0)
|
|
|
|
{
|
|
|
|
wall[i].extra = -1;
|
|
|
|
}
|
|
|
|
if (nXWall > 0)
|
|
|
|
{
|
|
|
|
dassert(nXWall < kMaxXWalls);
|
|
|
|
if (xwall[nXWall].reference == -1)
|
|
|
|
{
|
|
|
|
wall[i].extra = -1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
xwall[nXWall].reference = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (int i = 0; i < numwalls; i++)
|
|
|
|
{
|
|
|
|
int nXWall = wall[i].extra;
|
|
|
|
if (nXWall > 0)
|
|
|
|
{
|
|
|
|
dassert(nXWall < kMaxXWalls);
|
|
|
|
if (xwall[nXWall].reference != i)
|
|
|
|
{
|
|
|
|
int nXWall2 = dbInsertXWall(i);
|
|
|
|
memcpy(&xwall[nXWall2], &xwall[nXWall], sizeof(XWALL));
|
|
|
|
xwall[nXWall2].reference = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (int i = 1; i < kMaxXWalls; i++)
|
|
|
|
{
|
|
|
|
int nWall = xwall[i].reference;
|
|
|
|
if (nWall >= 0)
|
|
|
|
{
|
|
|
|
dassert(nWall < kMaxWalls);
|
|
|
|
if (nWall >= numwalls || wall[nWall].extra != i)
|
|
|
|
{
|
|
|
|
InsertFree(nextXWall, i);
|
|
|
|
xwall[i].reference = -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void dbXSectorClean(void)
|
|
|
|
{
|
2019-09-21 11:02:17 +00:00
|
|
|
|
|
|
|
|
2019-09-19 22:42:45 +00:00
|
|
|
for (int i = 0; i < numsectors; i++)
|
|
|
|
{
|
|
|
|
int nXSector = sector[i].extra;
|
|
|
|
if (nXSector == 0)
|
|
|
|
{
|
|
|
|
sector[i].extra = -1;
|
|
|
|
}
|
|
|
|
if (nXSector > 0)
|
|
|
|
{
|
|
|
|
dassert(nXSector < kMaxXSectors);
|
|
|
|
if (xsector[nXSector].reference == -1)
|
|
|
|
{
|
|
|
|
sector[i].extra = -1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
xsector[nXSector].reference = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (int i = 0; i < numsectors; i++)
|
|
|
|
{
|
|
|
|
int nXSector = sector[i].extra;
|
|
|
|
if (nXSector > 0)
|
|
|
|
{
|
|
|
|
dassert(nXSector < kMaxXSectors);
|
|
|
|
if (xsector[nXSector].reference != i)
|
|
|
|
{
|
|
|
|
int nXSector2 = dbInsertXSector(i);
|
|
|
|
memcpy(&xsector[nXSector2], &xsector[nXSector], sizeof(XSECTOR));
|
|
|
|
xsector[nXSector2].reference = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (int i = 1; i < kMaxXSectors; i++)
|
|
|
|
{
|
|
|
|
int nSector = xsector[i].reference;
|
|
|
|
if (nSector >= 0)
|
|
|
|
{
|
|
|
|
dassert(nSector < kMaxSectors);
|
|
|
|
if (nSector >= numsectors || sector[nSector].extra != i)
|
|
|
|
{
|
|
|
|
InsertFree(nextXSector, i);
|
|
|
|
xsector[i].reference = -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void dbInit(void)
|
|
|
|
{
|
|
|
|
InitFreeList(nextXSprite, kMaxXSprites);
|
|
|
|
for (int i = 1; i < kMaxXSprites; i++)
|
|
|
|
{
|
|
|
|
xsprite[i].reference = -1;
|
|
|
|
}
|
|
|
|
InitFreeList(nextXWall, kMaxXWalls);
|
|
|
|
for (int i = 1; i < kMaxXWalls; i++)
|
|
|
|
{
|
|
|
|
xwall[i].reference = -1;
|
|
|
|
}
|
|
|
|
InitFreeList(nextXSector, kMaxXSectors);
|
|
|
|
for (int i = 1; i < kMaxXSectors; i++)
|
|
|
|
{
|
|
|
|
xsector[i].reference = -1;
|
|
|
|
}
|
|
|
|
initspritelists();
|
|
|
|
for (int i = 0; i < kMaxSprites; i++)
|
|
|
|
{
|
|
|
|
sprite[i].cstat = 128;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void PropagateMarkerReferences(void)
|
|
|
|
{
|
|
|
|
int nSprite, nNextSprite;
|
2019-10-11 21:59:39 +00:00
|
|
|
for (nSprite = headspritestat[kStatMarker]; nSprite != -1; nSprite = nNextSprite) {
|
|
|
|
|
2019-09-19 22:42:45 +00:00
|
|
|
nNextSprite = nextspritestat[nSprite];
|
2019-10-11 21:59:39 +00:00
|
|
|
|
|
|
|
switch (sprite[nSprite].type) {
|
|
|
|
case kMarkerOff:
|
|
|
|
case kMarkerAxis:
|
|
|
|
case kMarkerWarpDest: {
|
|
|
|
int nOwner = sprite[nSprite].owner;
|
|
|
|
if (nOwner >= 0 && nOwner < numsectors) {
|
|
|
|
int nXSector = sector[nOwner].extra;
|
|
|
|
if (nXSector > 0 && nXSector < kMaxXSectors) {
|
|
|
|
xsector[nXSector].marker0 = nSprite;
|
|
|
|
continue;
|
|
|
|
}
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2019-10-11 21:59:39 +00:00
|
|
|
case kMarkerOn: {
|
|
|
|
int nOwner = sprite[nSprite].owner;
|
|
|
|
if (nOwner >= 0 && nOwner < numsectors) {
|
|
|
|
int nXSector = sector[nOwner].extra;
|
|
|
|
if (nXSector > 0 && nXSector < kMaxXSectors) {
|
|
|
|
xsector[nXSector].marker1 = nSprite;
|
|
|
|
continue;
|
|
|
|
}
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2019-10-11 21:59:39 +00:00
|
|
|
|
2019-09-19 22:42:45 +00:00
|
|
|
DeleteSprite(nSprite);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool byte_1A76C6, byte_1A76C7, byte_1A76C8;
|
|
|
|
|
|
|
|
MAPHEADER2 byte_19AE44;
|
|
|
|
|
|
|
|
unsigned int dbReadMapCRC(const char *pPath)
|
|
|
|
{
|
2019-09-30 17:08:35 +00:00
|
|
|
char name2[BMAX_PATH];
|
2019-09-19 22:42:45 +00:00
|
|
|
byte_1A76C7 = 0;
|
|
|
|
byte_1A76C8 = 0;
|
2019-09-30 17:08:35 +00:00
|
|
|
|
|
|
|
int const bakpathsearchmode = pathsearchmode;
|
|
|
|
pathsearchmode = 1;
|
|
|
|
|
|
|
|
Bstrncpy(name2, pPath, BMAX_PATH);
|
|
|
|
Bstrupr(name2);
|
|
|
|
DICTNODE* pNode = *gSysRes.Probe(name2, "MAP");
|
|
|
|
if (pNode && pNode->flags & DICT_EXTERNAL)
|
|
|
|
{
|
|
|
|
gSysRes.RemoveNode(pNode);
|
|
|
|
}
|
|
|
|
pNode = gSysRes.Lookup(pPath, "MAP");
|
|
|
|
if (!pNode)
|
|
|
|
{
|
|
|
|
char name2[BMAX_PATH];
|
|
|
|
Bstrncpy(name2, pPath, BMAX_PATH);
|
|
|
|
ChangeExtension(name2, "");
|
|
|
|
pNode = gSysRes.Lookup(name2, "MAP");
|
|
|
|
}
|
|
|
|
|
2019-09-19 22:42:45 +00:00
|
|
|
if (!pNode)
|
|
|
|
{
|
2019-09-30 17:08:35 +00:00
|
|
|
initprintf("Error opening map file %s", pPath);
|
|
|
|
pathsearchmode = bakpathsearchmode;
|
|
|
|
return -1;
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
char *pData = (char*)gSysRes.Lock(pNode);
|
2019-09-30 17:08:35 +00:00
|
|
|
pathsearchmode = bakpathsearchmode;
|
|
|
|
|
2019-09-19 22:42:45 +00:00
|
|
|
int nSize = pNode->size;
|
|
|
|
MAPSIGNATURE header;
|
|
|
|
IOBuffer(nSize, pData).Read(&header, 6);
|
|
|
|
#if B_BIG_ENDIAN == 1
|
|
|
|
header.version = B_LITTLE16(header.version);
|
|
|
|
#endif
|
|
|
|
if (memcmp(header.signature, "BLM\x1a", 4))
|
|
|
|
{
|
|
|
|
ThrowError("Map file corrupted");
|
|
|
|
}
|
|
|
|
if ((header.version & 0xff00) == 0x600)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
else if ((header.version & 0xff00) == 0x700)
|
|
|
|
{
|
|
|
|
byte_1A76C8 = 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ThrowError("Map file is wrong version");
|
|
|
|
}
|
|
|
|
unsigned int nCRC = *(unsigned int*)(pData+nSize-4);
|
|
|
|
gSysRes.Unlock(pNode);
|
|
|
|
return nCRC;
|
|
|
|
}
|
|
|
|
|
|
|
|
int gMapRev, gSongId, gSkyCount;
|
|
|
|
//char byte_19AE44[128];
|
2019-06-29 15:37:04 +00:00
|
|
|
const int nXSectorSize = 60;
|
|
|
|
const int nXSpriteSize = 56;
|
|
|
|
const int nXWallSize = 24;
|
2019-09-19 22:42:45 +00:00
|
|
|
|
2019-06-28 16:03:47 +00:00
|
|
|
int dbLoadMap(const char *pPath, int *pX, int *pY, int *pZ, short *pAngle, short *pSector, unsigned int *pCRC)
|
2019-09-19 22:42:45 +00:00
|
|
|
{
|
2019-09-30 17:08:35 +00:00
|
|
|
char name2[BMAX_PATH];
|
2019-09-19 22:42:45 +00:00
|
|
|
int16_t tpskyoff[256];
|
|
|
|
memset(show2dsector, 0, sizeof(show2dsector));
|
|
|
|
memset(show2dwall, 0, sizeof(show2dwall));
|
|
|
|
memset(show2dsprite, 0, sizeof(show2dsprite));
|
|
|
|
#ifdef USE_OPENGL
|
|
|
|
Polymost_prepare_loadboard();
|
|
|
|
#endif
|
2019-09-22 08:16:16 +00:00
|
|
|
|
|
|
|
int const bakpathsearchmode = pathsearchmode;
|
|
|
|
pathsearchmode = 1;
|
|
|
|
|
2019-09-30 17:08:35 +00:00
|
|
|
Bstrncpy(name2, pPath, BMAX_PATH);
|
|
|
|
Bstrupr(name2);
|
|
|
|
DICTNODE* pNode = *gSysRes.Probe(name2, "MAP");
|
|
|
|
if (pNode && pNode->flags & DICT_EXTERNAL)
|
2019-09-19 22:42:45 +00:00
|
|
|
{
|
2019-09-30 17:08:35 +00:00
|
|
|
gSysRes.RemoveNode(pNode);
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
2019-09-30 17:08:35 +00:00
|
|
|
|
|
|
|
pNode = gSysRes.Lookup(pPath, "MAP");
|
2019-09-19 22:42:45 +00:00
|
|
|
if (!pNode)
|
|
|
|
{
|
2019-06-28 16:03:47 +00:00
|
|
|
char name2[BMAX_PATH];
|
|
|
|
Bstrncpy(name2, pPath, BMAX_PATH);
|
|
|
|
ChangeExtension(name2, "");
|
|
|
|
pNode = gSysRes.Lookup(name2, "MAP");
|
|
|
|
}
|
2019-09-22 08:16:16 +00:00
|
|
|
|
2019-06-28 16:03:47 +00:00
|
|
|
if (!pNode)
|
|
|
|
{
|
|
|
|
initprintf("Error opening map file %s", pPath);
|
2019-09-30 17:08:35 +00:00
|
|
|
pathsearchmode = bakpathsearchmode;
|
2019-06-28 16:03:47 +00:00
|
|
|
return -1;
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
char *pData = (char*)gSysRes.Lock(pNode);
|
2019-09-30 17:08:35 +00:00
|
|
|
pathsearchmode = bakpathsearchmode;
|
2019-09-19 22:42:45 +00:00
|
|
|
int nSize = pNode->size;
|
|
|
|
MAPSIGNATURE header;
|
|
|
|
IOBuffer IOBuffer1 = IOBuffer(nSize, pData);
|
|
|
|
IOBuffer1.Read(&header, 6);
|
|
|
|
#if B_BIG_ENDIAN == 1
|
|
|
|
header.version = B_LITTLE16(header.version);
|
|
|
|
#endif
|
|
|
|
if (memcmp(header.signature, "BLM\x1a", 4))
|
|
|
|
{
|
2019-06-28 16:03:47 +00:00
|
|
|
initprintf("Map file corrupted");
|
|
|
|
gSysRes.Unlock(pNode);
|
|
|
|
return -1;
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
byte_1A76C8 = 0;
|
2019-10-11 21:59:39 +00:00
|
|
|
if ((header.version & 0xff00) == 0x700) {
|
2019-09-19 22:42:45 +00:00
|
|
|
byte_1A76C8 = 1;
|
2019-10-11 21:59:39 +00:00
|
|
|
|
|
|
|
// by NoOne: indicate if the map requires modern features to work properly
|
|
|
|
// for maps wich created in PMAPEDIT BETA13 or higher versions. Since only minor version changed,
|
|
|
|
// the map is still can be loaded with vanilla BLOOD / MAPEDIT and should work in other ports too.
|
|
|
|
if ((header.version & 0x00ff) == 0x001) gModernMap = true;
|
|
|
|
else gModernMap = false;
|
|
|
|
|
|
|
|
} else {
|
2019-06-28 16:03:47 +00:00
|
|
|
initprintf("Map file is wrong version");
|
|
|
|
gSysRes.Unlock(pNode);
|
|
|
|
return -1;
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
2019-10-11 21:59:39 +00:00
|
|
|
|
2019-09-19 22:42:45 +00:00
|
|
|
MAPHEADER mapHeader;
|
|
|
|
IOBuffer1.Read(&mapHeader,37/* sizeof(mapHeader)*/);
|
2019-10-11 21:59:39 +00:00
|
|
|
if (mapHeader.at16 != 0 && mapHeader.at16 != 0x7474614d && mapHeader.at16 != 0x4d617474) {
|
2019-09-19 22:42:45 +00:00
|
|
|
dbCrypt((char*)&mapHeader, sizeof(mapHeader), 0x7474614d);
|
|
|
|
byte_1A76C7 = 1;
|
|
|
|
}
|
2019-10-11 21:59:39 +00:00
|
|
|
|
2019-09-19 22:42:45 +00:00
|
|
|
#if B_BIG_ENDIAN == 1
|
|
|
|
mapHeader.at0 = B_LITTLE32(mapHeader.at0);
|
|
|
|
mapHeader.at4 = B_LITTLE32(mapHeader.at4);
|
|
|
|
mapHeader.at8 = B_LITTLE32(mapHeader.at8);
|
|
|
|
mapHeader.atc = B_LITTLE16(mapHeader.atc);
|
|
|
|
mapHeader.ate = B_LITTLE16(mapHeader.ate);
|
|
|
|
mapHeader.at10 = B_LITTLE16(mapHeader.at10);
|
|
|
|
mapHeader.at12 = B_LITTLE32(mapHeader.at12);
|
|
|
|
mapHeader.at16 = B_LITTLE32(mapHeader.at16);
|
|
|
|
mapHeader.at1b = B_LITTLE32(mapHeader.at1b);
|
|
|
|
mapHeader.at1f = B_LITTLE16(mapHeader.at1f);
|
|
|
|
mapHeader.at21 = B_LITTLE16(mapHeader.at21);
|
|
|
|
mapHeader.at23 = B_LITTLE16(mapHeader.at23);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
psky_t *pSky = tileSetupSky(0);
|
|
|
|
pSky->horizfrac = 65536;
|
|
|
|
|
|
|
|
*pX = mapHeader.at0;
|
|
|
|
*pY = mapHeader.at4;
|
|
|
|
*pZ = mapHeader.at8;
|
|
|
|
*pAngle = mapHeader.atc;
|
|
|
|
*pSector = mapHeader.ate;
|
|
|
|
pSky->lognumtiles = mapHeader.at10;
|
|
|
|
gVisibility = g_visibility = mapHeader.at12;
|
|
|
|
gSongId = mapHeader.at16;
|
|
|
|
if (byte_1A76C8)
|
|
|
|
{
|
|
|
|
if (mapHeader.at16 == 0x7474614d || mapHeader.at16 == 0x4d617474)
|
|
|
|
{
|
|
|
|
byte_1A76C6 = 1;
|
|
|
|
}
|
|
|
|
else if (!mapHeader.at16)
|
|
|
|
{
|
|
|
|
byte_1A76C6 = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-06-28 16:03:47 +00:00
|
|
|
initprintf("Corrupted Map file");
|
|
|
|
gSysRes.Unlock(pNode);
|
|
|
|
return -1;
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (mapHeader.at16)
|
|
|
|
{
|
2019-06-28 16:03:47 +00:00
|
|
|
initprintf("Corrupted Map file");
|
|
|
|
gSysRes.Unlock(pNode);
|
|
|
|
return -1;
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
parallaxtype = mapHeader.at1a;
|
|
|
|
gMapRev = mapHeader.at1b;
|
|
|
|
numsectors = mapHeader.at1f;
|
|
|
|
numwalls = mapHeader.at21;
|
|
|
|
dbInit();
|
|
|
|
if (byte_1A76C8)
|
|
|
|
{
|
|
|
|
IOBuffer1.Read(&byte_19AE44, 128);
|
|
|
|
dbCrypt((char*)&byte_19AE44, 128, numwalls);
|
|
|
|
#if B_BIG_ENDIAN == 1
|
|
|
|
byte_19AE44.at40 = B_LITTLE32(byte_19AE44.at40);
|
|
|
|
byte_19AE44.at44 = B_LITTLE32(byte_19AE44.at44);
|
|
|
|
byte_19AE44.at48 = B_LITTLE32(byte_19AE44.at48);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
memset(&byte_19AE44, 0, 128);
|
|
|
|
}
|
|
|
|
gSkyCount = 1<<pSky->lognumtiles;
|
|
|
|
IOBuffer1.Read(tpskyoff, gSkyCount*sizeof(tpskyoff[0]));
|
|
|
|
if (byte_1A76C8)
|
|
|
|
{
|
|
|
|
dbCrypt((char*)tpskyoff, gSkyCount*sizeof(tpskyoff[0]), gSkyCount*2);
|
|
|
|
}
|
|
|
|
for (int i = 0; i < ClipHigh(gSkyCount, MAXPSKYTILES); i++)
|
|
|
|
{
|
|
|
|
pSky->tileofs[i] = B_LITTLE16(tpskyoff[i]);
|
|
|
|
}
|
|
|
|
for (int i = 0; i < numsectors; i++)
|
|
|
|
{
|
|
|
|
sectortype *pSector = §or[i];
|
|
|
|
IOBuffer1.Read(pSector, sizeof(sectortype));
|
|
|
|
if (byte_1A76C8)
|
|
|
|
{
|
|
|
|
dbCrypt((char*)pSector, sizeof(sectortype), gMapRev*sizeof(sectortype));
|
|
|
|
}
|
|
|
|
#if B_BIG_ENDIAN == 1
|
|
|
|
pSector->wallptr = B_LITTLE16(pSector->wallptr);
|
|
|
|
pSector->wallnum = B_LITTLE16(pSector->wallnum);
|
|
|
|
pSector->ceilingz = B_LITTLE32(pSector->ceilingz);
|
|
|
|
pSector->floorz = B_LITTLE32(pSector->floorz);
|
|
|
|
pSector->ceilingstat = B_LITTLE16(pSector->ceilingstat);
|
|
|
|
pSector->floorstat = B_LITTLE16(pSector->floorstat);
|
|
|
|
pSector->ceilingpicnum = B_LITTLE16(pSector->ceilingpicnum);
|
|
|
|
pSector->ceilingheinum = B_LITTLE16(pSector->ceilingheinum);
|
|
|
|
pSector->floorpicnum = B_LITTLE16(pSector->floorpicnum);
|
|
|
|
pSector->floorheinum = B_LITTLE16(pSector->floorheinum);
|
2019-09-30 07:18:01 +00:00
|
|
|
pSector->type = B_LITTLE16(pSector->type);
|
2019-09-19 22:42:45 +00:00
|
|
|
pSector->hitag = B_LITTLE16(pSector->hitag);
|
|
|
|
pSector->extra = B_LITTLE16(pSector->extra);
|
|
|
|
#endif
|
|
|
|
qsector_filler[i] = pSector->fogpal;
|
|
|
|
pSector->fogpal = 0;
|
|
|
|
if (sector[i].extra > 0)
|
|
|
|
{
|
|
|
|
char pBuffer[nXSectorSize];
|
|
|
|
int nXSector = dbInsertXSector(i);
|
|
|
|
XSECTOR *pXSector = &xsector[nXSector];
|
|
|
|
memset(pXSector, 0, sizeof(XSECTOR));
|
|
|
|
int nCount;
|
|
|
|
if (!byte_1A76C8)
|
|
|
|
{
|
|
|
|
nCount = nXSectorSize;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
nCount = byte_19AE44.at48;
|
|
|
|
}
|
|
|
|
dassert(nCount <= nXSectorSize);
|
|
|
|
IOBuffer1.Read(pBuffer, nCount);
|
|
|
|
BitReader bitReader(pBuffer, nCount);
|
|
|
|
pXSector->reference = bitReader.readSigned(14);
|
|
|
|
pXSector->state = bitReader.readUnsigned(1);
|
|
|
|
pXSector->busy = bitReader.readUnsigned(17);
|
|
|
|
pXSector->data = bitReader.readUnsigned(16);
|
|
|
|
pXSector->txID = bitReader.readUnsigned(10);
|
|
|
|
pXSector->at7_2 = bitReader.readUnsigned(3);
|
|
|
|
pXSector->at7_5 = bitReader.readUnsigned(3);
|
|
|
|
pXSector->rxID = bitReader.readUnsigned(10);
|
|
|
|
pXSector->command = bitReader.readUnsigned(8);
|
|
|
|
pXSector->triggerOn = bitReader.readUnsigned(1);
|
|
|
|
pXSector->triggerOff = bitReader.readUnsigned(1);
|
|
|
|
pXSector->busyTimeA = bitReader.readUnsigned(12);
|
|
|
|
pXSector->waitTimeA = bitReader.readUnsigned(12);
|
|
|
|
pXSector->atd_4 = bitReader.readUnsigned(1);
|
|
|
|
pXSector->interruptable = bitReader.readUnsigned(1);
|
|
|
|
pXSector->amplitude = bitReader.readSigned(8);
|
|
|
|
pXSector->freq = bitReader.readUnsigned(8);
|
|
|
|
pXSector->atf_6 = bitReader.readUnsigned(1);
|
|
|
|
pXSector->atf_7 = bitReader.readUnsigned(1);
|
|
|
|
pXSector->phase = bitReader.readUnsigned(8);
|
|
|
|
pXSector->wave = bitReader.readUnsigned(4);
|
|
|
|
pXSector->shadeAlways = bitReader.readUnsigned(1);
|
|
|
|
pXSector->shadeFloor = bitReader.readUnsigned(1);
|
|
|
|
pXSector->shadeCeiling = bitReader.readUnsigned(1);
|
|
|
|
pXSector->shadeWalls = bitReader.readUnsigned(1);
|
|
|
|
pXSector->shade = bitReader.readSigned(8);
|
|
|
|
pXSector->panAlways = bitReader.readUnsigned(1);
|
|
|
|
pXSector->panFloor = bitReader.readUnsigned(1);
|
|
|
|
pXSector->panCeiling = bitReader.readUnsigned(1);
|
|
|
|
pXSector->Drag = bitReader.readUnsigned(1);
|
|
|
|
pXSector->Underwater = bitReader.readUnsigned(1);
|
|
|
|
pXSector->Depth = bitReader.readUnsigned(3);
|
|
|
|
pXSector->panVel = bitReader.readUnsigned(8);
|
|
|
|
pXSector->panAngle = bitReader.readUnsigned(11);
|
|
|
|
pXSector->at16_3 = bitReader.readUnsigned(1);
|
|
|
|
pXSector->decoupled = bitReader.readUnsigned(1);
|
|
|
|
pXSector->triggerOnce = bitReader.readUnsigned(1);
|
2019-10-11 21:59:39 +00:00
|
|
|
pXSector->isTriggered = bitReader.readUnsigned(1);
|
2019-09-19 22:42:45 +00:00
|
|
|
pXSector->Key = bitReader.readUnsigned(3);
|
|
|
|
pXSector->Push = bitReader.readUnsigned(1);
|
|
|
|
pXSector->Vector = bitReader.readUnsigned(1);
|
|
|
|
pXSector->Reserved = bitReader.readUnsigned(1);
|
|
|
|
pXSector->Enter = bitReader.readUnsigned(1);
|
|
|
|
pXSector->Exit = bitReader.readUnsigned(1);
|
|
|
|
pXSector->Wallpush = bitReader.readUnsigned(1);
|
|
|
|
pXSector->color = bitReader.readUnsigned(1);
|
|
|
|
pXSector->at18_1 = bitReader.readUnsigned(1);
|
|
|
|
pXSector->busyTimeB = bitReader.readUnsigned(12);
|
|
|
|
pXSector->waitTimeB = bitReader.readUnsigned(12);
|
2019-10-11 21:59:39 +00:00
|
|
|
pXSector->stopOn = bitReader.readUnsigned(1);
|
|
|
|
pXSector->stopOff = bitReader.readUnsigned(1);
|
2019-09-19 22:42:45 +00:00
|
|
|
pXSector->ceilpal = bitReader.readUnsigned(4);
|
|
|
|
pXSector->at1c_0 = bitReader.readSigned(32);
|
|
|
|
pXSector->at20_0 = bitReader.readSigned(32);
|
|
|
|
pXSector->at24_0 = bitReader.readSigned(32);
|
|
|
|
pXSector->at28_0 = bitReader.readSigned(32);
|
2019-10-11 21:59:39 +00:00
|
|
|
pXSector->marker0 = bitReader.readUnsigned(16);
|
|
|
|
pXSector->marker1 = bitReader.readUnsigned(16);
|
2019-09-19 22:42:45 +00:00
|
|
|
pXSector->Crush = bitReader.readUnsigned(1);
|
|
|
|
pXSector->at30_1 = bitReader.readUnsigned(8);
|
|
|
|
pXSector->at31_1 = bitReader.readUnsigned(8);
|
|
|
|
pXSector->at32_1 = bitReader.readUnsigned(8);
|
|
|
|
pXSector->damageType = bitReader.readUnsigned(3);
|
|
|
|
pXSector->floorpal = bitReader.readUnsigned(4);
|
|
|
|
pXSector->at34_0 = bitReader.readUnsigned(8);
|
|
|
|
pXSector->locked = bitReader.readUnsigned(1);
|
|
|
|
pXSector->windVel = bitReader.readUnsigned(10);
|
|
|
|
pXSector->windAng = bitReader.readUnsigned(11);
|
|
|
|
pXSector->windAlways = bitReader.readUnsigned(1);
|
|
|
|
pXSector->at37_7 = bitReader.readUnsigned(1);
|
|
|
|
pXSector->bobTheta = bitReader.readUnsigned(11);
|
|
|
|
pXSector->bobZRange = bitReader.readUnsigned(5);
|
|
|
|
pXSector->bobSpeed = bitReader.readSigned(12);
|
|
|
|
pXSector->bobAlways = bitReader.readUnsigned(1);
|
|
|
|
pXSector->bobFloor = bitReader.readUnsigned(1);
|
|
|
|
pXSector->bobCeiling = bitReader.readUnsigned(1);
|
|
|
|
pXSector->bobRotate = bitReader.readUnsigned(1);
|
|
|
|
xsector[sector[i].extra].reference = i;
|
|
|
|
xsector[sector[i].extra].busy = xsector[sector[i].extra].state<<16;
|
2019-10-07 19:29:52 +00:00
|
|
|
|
|
|
|
// by NoOne: indicate if the map requires modern features to work properly
|
|
|
|
// for maps wich created in different editors (include vanilla MAPEDIT) or in PMAPEDIT version below than BETA13
|
2019-10-11 21:59:39 +00:00
|
|
|
if (pXSector->rxID == kChannelMapExtended && pXSector->rxID == pXSector->txID && pXSector->command == kCmdModernFeaturesEnable)
|
2019-10-07 19:29:52 +00:00
|
|
|
gModernMap = true;
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
for (int i = 0; i < numwalls; i++)
|
|
|
|
{
|
|
|
|
walltype *pWall = &wall[i];
|
|
|
|
IOBuffer1.Read(pWall, sizeof(walltype));
|
|
|
|
if (byte_1A76C8)
|
|
|
|
{
|
|
|
|
dbCrypt((char*)pWall, sizeof(walltype), (gMapRev*sizeof(sectortype)) | 0x7474614d);
|
|
|
|
}
|
|
|
|
#if B_BIG_ENDIAN == 1
|
|
|
|
pWall->x = B_LITTLE32(pWall->x);
|
|
|
|
pWall->y = B_LITTLE32(pWall->y);
|
|
|
|
pWall->point2 = B_LITTLE16(pWall->point2);
|
|
|
|
pWall->nextwall = B_LITTLE16(pWall->nextwall);
|
|
|
|
pWall->nextsector = B_LITTLE16(pWall->nextsector);
|
|
|
|
pWall->cstat = B_LITTLE16(pWall->cstat);
|
|
|
|
pWall->picnum = B_LITTLE16(pWall->picnum);
|
|
|
|
pWall->overpicnum = B_LITTLE16(pWall->overpicnum);
|
2019-09-30 07:18:01 +00:00
|
|
|
pWall->type = B_LITTLE16(pWall->type);
|
2019-09-19 22:42:45 +00:00
|
|
|
pWall->hitag = B_LITTLE16(pWall->hitag);
|
|
|
|
pWall->extra = B_LITTLE16(pWall->extra);
|
|
|
|
#endif
|
|
|
|
if (wall[i].extra > 0)
|
|
|
|
{
|
|
|
|
char pBuffer[nXWallSize];
|
|
|
|
int nXWall = dbInsertXWall(i);
|
|
|
|
XWALL *pXWall = &xwall[nXWall];
|
|
|
|
memset(pXWall, 0, sizeof(XWALL));
|
|
|
|
int nCount;
|
|
|
|
if (!byte_1A76C8)
|
|
|
|
{
|
|
|
|
nCount = nXWallSize;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
nCount = byte_19AE44.at44;
|
|
|
|
}
|
|
|
|
dassert(nCount <= nXWallSize);
|
|
|
|
IOBuffer1.Read(pBuffer, nCount);
|
|
|
|
BitReader bitReader(pBuffer, nCount);
|
|
|
|
pXWall->reference = bitReader.readSigned(14);
|
|
|
|
pXWall->state = bitReader.readUnsigned(1);
|
|
|
|
pXWall->busy = bitReader.readUnsigned(17);
|
|
|
|
pXWall->data = bitReader.readSigned(16);
|
|
|
|
pXWall->txID = bitReader.readUnsigned(10);
|
|
|
|
pXWall->at7_2 = bitReader.readUnsigned(6);
|
|
|
|
pXWall->rxID = bitReader.readUnsigned(10);
|
|
|
|
pXWall->command = bitReader.readUnsigned(8);
|
|
|
|
pXWall->triggerOn = bitReader.readUnsigned(1);
|
|
|
|
pXWall->triggerOff = bitReader.readUnsigned(1);
|
|
|
|
pXWall->busyTime = bitReader.readUnsigned(12);
|
|
|
|
pXWall->waitTime = bitReader.readUnsigned(12);
|
|
|
|
pXWall->restState = bitReader.readUnsigned(1);
|
|
|
|
pXWall->interruptable = bitReader.readUnsigned(1);
|
|
|
|
pXWall->panAlways = bitReader.readUnsigned(1);
|
|
|
|
pXWall->panXVel = bitReader.readSigned(8);
|
|
|
|
pXWall->panYVel = bitReader.readSigned(8);
|
|
|
|
pXWall->decoupled = bitReader.readUnsigned(1);
|
|
|
|
pXWall->triggerOnce = bitReader.readUnsigned(1);
|
|
|
|
pXWall->isTriggered = bitReader.readUnsigned(1);
|
|
|
|
pXWall->key = bitReader.readUnsigned(3);
|
|
|
|
pXWall->triggerPush = bitReader.readUnsigned(1);
|
|
|
|
pXWall->triggerVector = bitReader.readUnsigned(1);
|
2019-09-21 11:02:17 +00:00
|
|
|
pXWall->triggerTouch = bitReader.readUnsigned(1);
|
2019-09-19 22:42:45 +00:00
|
|
|
pXWall->at11_0 = bitReader.readUnsigned(2);
|
|
|
|
pXWall->xpanFrac = bitReader.readUnsigned(8);
|
|
|
|
pXWall->ypanFrac = bitReader.readUnsigned(8);
|
|
|
|
pXWall->locked = bitReader.readUnsigned(1);
|
|
|
|
pXWall->dudeLockout = bitReader.readUnsigned(1);
|
|
|
|
pXWall->at13_4 = bitReader.readUnsigned(4);
|
|
|
|
pXWall->at14_0 = bitReader.readUnsigned(32);
|
|
|
|
xwall[wall[i].extra].reference = i;
|
|
|
|
xwall[wall[i].extra].busy = xwall[wall[i].extra].state << 16;
|
2019-10-07 19:29:52 +00:00
|
|
|
|
|
|
|
// by NoOne: indicate if the map requires modern features to work properly
|
|
|
|
// for maps wich created in different editors (include vanilla MAPEDIT) or in PMAPEDIT version below than BETA13
|
2019-10-11 21:59:39 +00:00
|
|
|
if (pXWall->rxID == kChannelMapExtended && pXWall->rxID == pXWall->txID && pXWall->command == kCmdModernFeaturesEnable)
|
2019-10-07 19:29:52 +00:00
|
|
|
gModernMap = true;
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
initspritelists();
|
|
|
|
for (int i = 0; i < mapHeader.at23; i++)
|
|
|
|
{
|
|
|
|
RemoveSpriteStat(i);
|
|
|
|
spritetype *pSprite = &sprite[i];
|
|
|
|
IOBuffer1.Read(pSprite, sizeof(spritetype));
|
|
|
|
if (byte_1A76C8)
|
|
|
|
{
|
|
|
|
dbCrypt((char*)pSprite, sizeof(spritetype), (gMapRev*sizeof(spritetype)) | 0x7474614d);
|
|
|
|
}
|
|
|
|
#if B_BIG_ENDIAN == 1
|
|
|
|
pSprite->x = B_LITTLE32(pSprite->x);
|
|
|
|
pSprite->y = B_LITTLE32(pSprite->y);
|
|
|
|
pSprite->z = B_LITTLE32(pSprite->z);
|
|
|
|
pSprite->cstat = B_LITTLE16(pSprite->cstat);
|
|
|
|
pSprite->picnum = B_LITTLE16(pSprite->picnum);
|
|
|
|
pSprite->sectnum = B_LITTLE16(pSprite->sectnum);
|
|
|
|
pSprite->statnum = B_LITTLE16(pSprite->statnum);
|
|
|
|
pSprite->ang = B_LITTLE16(pSprite->ang);
|
|
|
|
pSprite->owner = B_LITTLE16(pSprite->owner);
|
2019-09-30 07:18:01 +00:00
|
|
|
pSprite->index = B_LITTLE16(pSprite->index);
|
2019-09-19 22:42:45 +00:00
|
|
|
pSprite->yvel = B_LITTLE16(pSprite->yvel);
|
2019-09-30 07:18:01 +00:00
|
|
|
pSprite->inittype = B_LITTLE16(pSprite->inittype);
|
|
|
|
pSprite->type = B_LITTLE16(pSprite->type);
|
|
|
|
pSprite->flags = B_LITTLE16(pSprite->hitag);
|
2019-09-19 22:42:45 +00:00
|
|
|
pSprite->extra = B_LITTLE16(pSprite->extra);
|
|
|
|
#endif
|
|
|
|
InsertSpriteSect(i, sprite[i].sectnum);
|
|
|
|
InsertSpriteStat(i, sprite[i].statnum);
|
2019-06-29 12:24:23 +00:00
|
|
|
Numsprites++;
|
2019-09-19 22:42:45 +00:00
|
|
|
sprite[i].index = i;
|
|
|
|
qsprite_filler[i] = pSprite->blend;
|
|
|
|
pSprite->blend = 0;
|
|
|
|
if (sprite[i].extra > 0)
|
|
|
|
{
|
|
|
|
char pBuffer[nXSpriteSize];
|
|
|
|
int nXSprite = dbInsertXSprite(i);
|
|
|
|
XSPRITE *pXSprite = &xsprite[nXSprite];
|
|
|
|
memset(pXSprite, 0, sizeof(XSPRITE));
|
|
|
|
int nCount;
|
|
|
|
if (!byte_1A76C8)
|
|
|
|
{
|
|
|
|
nCount = nXSpriteSize;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
nCount = byte_19AE44.at40;
|
|
|
|
}
|
|
|
|
dassert(nCount <= nXSpriteSize);
|
|
|
|
IOBuffer1.Read(pBuffer, nCount);
|
|
|
|
BitReader bitReader(pBuffer, nCount);
|
|
|
|
pXSprite->reference = bitReader.readSigned(14);
|
|
|
|
pXSprite->state = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->busy = bitReader.readUnsigned(17);
|
|
|
|
pXSprite->txID = bitReader.readUnsigned(10);
|
|
|
|
pXSprite->rxID = bitReader.readUnsigned(10);
|
|
|
|
pXSprite->command = bitReader.readUnsigned(8);
|
|
|
|
pXSprite->triggerOn = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->triggerOff = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->wave = bitReader.readUnsigned(2);
|
|
|
|
pXSprite->busyTime = bitReader.readUnsigned(12);
|
|
|
|
pXSprite->waitTime = bitReader.readUnsigned(12);
|
|
|
|
pXSprite->restState = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->Interrutable = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->atb_2 = bitReader.readUnsigned(2);
|
|
|
|
pXSprite->respawnPending = bitReader.readUnsigned(2);
|
|
|
|
pXSprite->atb_6 = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->lT = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->dropMsg = bitReader.readUnsigned(8);
|
|
|
|
pXSprite->Decoupled = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->triggerOnce = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->isTriggered = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->key = bitReader.readUnsigned(3);
|
|
|
|
pXSprite->Push = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->Vector = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->Impact = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->Pickup = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->Touch = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->Sight = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->Proximity = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->ate_5 = bitReader.readUnsigned(2);
|
|
|
|
pXSprite->lSkill = bitReader.readUnsigned(5);
|
|
|
|
pXSprite->lS = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->lB = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->lC = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->DudeLockout = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->data1 = bitReader.readSigned(16);
|
|
|
|
pXSprite->data2 = bitReader.readSigned(16);
|
|
|
|
pXSprite->data3 = bitReader.readSigned(16);
|
|
|
|
pXSprite->goalAng = bitReader.readUnsigned(11);
|
|
|
|
pXSprite->dodgeDir = bitReader.readSigned(2);
|
|
|
|
pXSprite->locked = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->medium = bitReader.readUnsigned(2);
|
|
|
|
pXSprite->respawn = bitReader.readUnsigned(2);
|
|
|
|
pXSprite->data4 = bitReader.readUnsigned(16);
|
|
|
|
pXSprite->at1a_2 = bitReader.readUnsigned(6);
|
|
|
|
pXSprite->lockMsg = bitReader.readUnsigned(8);
|
|
|
|
pXSprite->health = bitReader.readUnsigned(12);
|
|
|
|
pXSprite->dudeDeaf = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->dudeAmbush = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->dudeGuard = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->dudeFlag4 = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->target = bitReader.readSigned(16);
|
|
|
|
pXSprite->targetX = bitReader.readSigned(32);
|
|
|
|
pXSprite->targetY = bitReader.readSigned(32);
|
|
|
|
pXSprite->targetZ = bitReader.readSigned(32);
|
|
|
|
pXSprite->burnTime = bitReader.readUnsigned(16);
|
|
|
|
pXSprite->burnSource = bitReader.readSigned(16);
|
|
|
|
pXSprite->height = bitReader.readUnsigned(16);
|
|
|
|
pXSprite->stateTimer = bitReader.readUnsigned(16);
|
|
|
|
pXSprite->aiState = NULL;
|
|
|
|
bitReader.skipBits(32);
|
|
|
|
xsprite[sprite[i].extra].reference = i;
|
|
|
|
xsprite[sprite[i].extra].busy = xsprite[sprite[i].extra].state << 16;
|
2019-10-07 19:29:52 +00:00
|
|
|
if (!byte_1A76C8) {
|
2019-09-19 22:42:45 +00:00
|
|
|
xsprite[sprite[i].extra].lT |= xsprite[sprite[i].extra].lB;
|
|
|
|
}
|
2019-10-07 19:29:52 +00:00
|
|
|
|
|
|
|
// by NoOne: indicate if the map requires modern features to work properly
|
|
|
|
// for maps wich created in different editors (include vanilla MAPEDIT) or in PMAPEDIT version below than BETA13
|
2019-10-11 21:59:39 +00:00
|
|
|
if (pXSprite->rxID == kChannelMapExtended && pXSprite->rxID == pXSprite->txID && pXSprite->command == kCmdModernFeaturesEnable)
|
2019-10-07 19:29:52 +00:00
|
|
|
gModernMap = true;
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
if ((sprite[i].cstat & 0x30) == 0x30)
|
|
|
|
{
|
|
|
|
sprite[i].cstat &= ~0x30;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
unsigned int nCRC;
|
|
|
|
IOBuffer1.Read(&nCRC, 4);
|
|
|
|
#if B_BIG_ENDIAN == 1
|
|
|
|
nCRC = B_LITTLE32(nCRC);
|
|
|
|
#endif
|
|
|
|
if (Bcrc32(pData, nSize-4, 0) != nCRC)
|
|
|
|
{
|
2019-06-28 16:03:47 +00:00
|
|
|
initprintf("Map File does not match CRC");
|
|
|
|
gSysRes.Unlock(pNode);
|
|
|
|
return -1;
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
2019-06-28 16:03:47 +00:00
|
|
|
if (pCRC)
|
|
|
|
*pCRC = nCRC;
|
2019-09-19 22:42:45 +00:00
|
|
|
gSysRes.Unlock(pNode);
|
|
|
|
PropagateMarkerReferences();
|
|
|
|
if (byte_1A76C8)
|
|
|
|
{
|
|
|
|
if (gSongId == 0x7474614d || gSongId == 0x4d617474)
|
|
|
|
{
|
|
|
|
byte_1A76C6 = 1;
|
|
|
|
}
|
|
|
|
else if (!gSongId)
|
|
|
|
{
|
|
|
|
byte_1A76C6 = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-06-28 16:03:47 +00:00
|
|
|
initprintf("Corrupted Map file");
|
|
|
|
gSysRes.Unlock(pNode);
|
|
|
|
return -1;
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (gSongId != 0)
|
|
|
|
{
|
2019-06-28 16:03:47 +00:00
|
|
|
initprintf("Corrupted Map file");
|
|
|
|
gSysRes.Unlock(pNode);
|
|
|
|
return -1;
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef POLYMER
|
|
|
|
if (videoGetRenderMode() == REND_POLYMER)
|
|
|
|
polymer_loadboard();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if ((header.version & 0xff00) == 0x600)
|
|
|
|
{
|
|
|
|
switch (header.version&0xff)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
for (int i = 0; i < numsectors; i++)
|
|
|
|
{
|
|
|
|
sectortype *pSector = §or[i];
|
|
|
|
if (pSector->extra > 0)
|
|
|
|
{
|
|
|
|
XSECTOR *pXSector = &xsector[pSector->extra];
|
|
|
|
pXSector->busyTimeB = pXSector->busyTimeA;
|
|
|
|
if (pXSector->busyTimeA > 0)
|
|
|
|
{
|
|
|
|
if (!pXSector->atd_4)
|
|
|
|
{
|
|
|
|
pXSector->atf_6 = 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
pXSector->waitTimeB = pXSector->busyTimeA;
|
|
|
|
pXSector->waitTimeA = 0;
|
|
|
|
pXSector->atf_7 = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fallthrough__;
|
|
|
|
case 1:
|
|
|
|
for (int i = 0; i < numsectors; i++)
|
|
|
|
{
|
|
|
|
sectortype *pSector = §or[i];
|
|
|
|
if (pSector->extra > 0)
|
|
|
|
{
|
|
|
|
XSECTOR *pXSector = &xsector[pSector->extra];
|
|
|
|
pXSector->freq >>= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fallthrough__;
|
|
|
|
case 2:
|
|
|
|
for (int i = 0; i < kMaxSprites; i++)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
2019-06-26 13:20:03 +00:00
|
|
|
|
2019-06-26 14:02:01 +00:00
|
|
|
#ifdef YAX_ENABLE
|
2019-06-29 17:48:05 +00:00
|
|
|
// NUKE-TODO: Should be updated for new map format only
|
|
|
|
yax_update(0);
|
2019-06-26 14:02:01 +00:00
|
|
|
#endif
|
2019-06-28 16:03:47 +00:00
|
|
|
|
|
|
|
g_loadedMapVersion = 7;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-06-29 15:37:04 +00:00
|
|
|
int dbSaveMap(const char *pPath, int nX, int nY, int nZ, short nAngle, short nSector)
|
|
|
|
{
|
2019-09-21 11:02:17 +00:00
|
|
|
char sMapExt[BMAX_PATH];
|
2019-06-29 15:37:04 +00:00
|
|
|
int16_t tpskyoff[256];
|
|
|
|
int nSpriteNum;
|
|
|
|
psky_t *pSky = tileSetupSky(0);
|
|
|
|
gSkyCount = 1<<pSky->lognumtiles;
|
|
|
|
gMapRev++;
|
|
|
|
nSpriteNum = 0;
|
|
|
|
strcpy(sMapExt, pPath);
|
|
|
|
ChangeExtension(sMapExt, ".MAP");
|
|
|
|
int nSize = sizeof(MAPSIGNATURE)+sizeof(MAPHEADER);
|
|
|
|
if (byte_1A76C8)
|
|
|
|
{
|
|
|
|
nSize += sizeof(MAPHEADER2);
|
|
|
|
}
|
2019-06-29 16:54:00 +00:00
|
|
|
for (int i = 0; i < gSkyCount; i++)
|
2019-06-29 15:37:04 +00:00
|
|
|
tpskyoff[i] = pSky->tileofs[i];
|
|
|
|
nSize += gSkyCount*sizeof(tpskyoff[0]);
|
|
|
|
nSize += sizeof(sectortype)*numsectors;
|
|
|
|
for (int i = 0; i < numsectors; i++)
|
|
|
|
{
|
|
|
|
if (sector[i].extra > 0)
|
|
|
|
{
|
|
|
|
nSize += nXSectorSize;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
nSize += sizeof(walltype)*numwalls;
|
|
|
|
for (int i = 0; i < numwalls; i++)
|
|
|
|
{
|
|
|
|
if (wall[i].extra > 0)
|
|
|
|
{
|
|
|
|
nSize += nXWallSize;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (int i = 0; i < kMaxSprites; i++)
|
|
|
|
{
|
|
|
|
if (sprite[i].statnum < kMaxStatus)
|
|
|
|
{
|
|
|
|
nSpriteNum++;
|
|
|
|
if (sprite[i].extra > 0)
|
|
|
|
{
|
|
|
|
nSize += nXSpriteSize;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
nSize += sizeof(spritetype)*nSpriteNum;
|
|
|
|
nSize += 4;
|
|
|
|
char *pData = (char*)Xmalloc(nSize);
|
|
|
|
IOBuffer IOBuffer1 = IOBuffer(nSize, pData);
|
|
|
|
MAPSIGNATURE header;
|
|
|
|
memcpy(&header, "BLM\x1a", 4);
|
|
|
|
if (byte_1A76C8)
|
|
|
|
{
|
|
|
|
header.version = 0x700;
|
|
|
|
byte_1A76C7 = 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
header.version = 0x603;
|
|
|
|
byte_1A76C7 = 0;
|
|
|
|
}
|
|
|
|
IOBuffer1.Write(&header, sizeof(header));
|
|
|
|
MAPHEADER mapheader;
|
|
|
|
mapheader.at0 = B_LITTLE32(nX);
|
|
|
|
mapheader.at4 = B_LITTLE32(nY);
|
|
|
|
mapheader.at8 = B_LITTLE32(nZ);
|
|
|
|
mapheader.atc = B_LITTLE16(nAngle);
|
|
|
|
mapheader.ate = B_LITTLE16(nSector);
|
|
|
|
mapheader.at10 = B_LITTLE16(pSky->lognumtiles);
|
|
|
|
mapheader.at12 = B_LITTLE32(gVisibility);
|
|
|
|
if (byte_1A76C6)
|
|
|
|
{
|
|
|
|
gSongId = 0x7474614d;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gSongId = 0;
|
|
|
|
}
|
|
|
|
mapheader.at16 = B_LITTLE32(gSongId);
|
|
|
|
mapheader.at1a = parallaxtype;
|
|
|
|
mapheader.at1b = gMapRev;
|
|
|
|
mapheader.at1f = B_LITTLE16(numsectors);
|
|
|
|
mapheader.at21 = B_LITTLE16(numwalls);
|
|
|
|
mapheader.at23 = B_LITTLE16(nSpriteNum);
|
|
|
|
if (byte_1A76C7)
|
|
|
|
{
|
|
|
|
dbCrypt((char*)&mapheader, sizeof(MAPHEADER), 'ttaM');
|
|
|
|
}
|
|
|
|
IOBuffer1.Write(&mapheader, sizeof(MAPHEADER));
|
|
|
|
if (byte_1A76C8)
|
|
|
|
{
|
|
|
|
byte_19AE44.at48 = nXSectorSize;
|
|
|
|
byte_19AE44.at44 = nXWallSize;
|
|
|
|
byte_19AE44.at40 = nXSpriteSize;
|
|
|
|
dbCrypt((char*)&byte_19AE44, sizeof(MAPHEADER2), numwalls);
|
|
|
|
IOBuffer1.Write(&byte_19AE44, sizeof(MAPHEADER2));
|
|
|
|
dbCrypt((char*)&byte_19AE44, sizeof(MAPHEADER2), numwalls);
|
|
|
|
}
|
|
|
|
if (byte_1A76C8)
|
|
|
|
{
|
|
|
|
dbCrypt((char*)tpskyoff, gSkyCount*sizeof(tpskyoff[0]), gSkyCount*sizeof(tpskyoff[0]));
|
|
|
|
}
|
|
|
|
IOBuffer1.Write(tpskyoff, gSkyCount*sizeof(tpskyoff[0]));
|
|
|
|
if (byte_1A76C8)
|
|
|
|
{
|
|
|
|
dbCrypt((char*)tpskyoff, gSkyCount*sizeof(tpskyoff[0]), gSkyCount*sizeof(tpskyoff[0]));
|
|
|
|
}
|
|
|
|
for (int i = 0; i < numsectors; i++)
|
|
|
|
{
|
|
|
|
if (byte_1A76C8)
|
|
|
|
{
|
|
|
|
dbCrypt((char*)§or[i], sizeof(sectortype), gMapRev*sizeof(sectortype));
|
|
|
|
}
|
|
|
|
IOBuffer1.Write(§or[i], sizeof(sectortype));
|
|
|
|
if (byte_1A76C8)
|
|
|
|
{
|
|
|
|
dbCrypt((char*)§or[i], sizeof(sectortype), gMapRev*sizeof(sectortype));
|
|
|
|
}
|
|
|
|
if (sector[i].extra > 0)
|
|
|
|
{
|
|
|
|
char pBuffer[nXSectorSize];
|
|
|
|
BitWriter bitWriter(pBuffer, nXSectorSize);
|
|
|
|
XSECTOR* pXSector = &xsector[sector[i].extra];
|
|
|
|
bitWriter.write(pXSector->reference, 14);
|
|
|
|
bitWriter.write(pXSector->state, 1);
|
|
|
|
bitWriter.write(pXSector->busy, 17);
|
|
|
|
bitWriter.write(pXSector->data, 16);
|
|
|
|
bitWriter.write(pXSector->txID, 10);
|
|
|
|
bitWriter.write(pXSector->at7_2, 3);
|
|
|
|
bitWriter.write(pXSector->at7_5, 3);
|
|
|
|
bitWriter.write(pXSector->rxID, 10);
|
|
|
|
bitWriter.write(pXSector->command, 8);
|
|
|
|
bitWriter.write(pXSector->triggerOn, 1);
|
|
|
|
bitWriter.write(pXSector->triggerOff, 1);
|
|
|
|
bitWriter.write(pXSector->busyTimeA, 12);
|
|
|
|
bitWriter.write(pXSector->waitTimeA, 12);
|
|
|
|
bitWriter.write(pXSector->atd_4, 1);
|
|
|
|
bitWriter.write(pXSector->interruptable, 1);
|
|
|
|
bitWriter.write(pXSector->amplitude, 8);
|
|
|
|
bitWriter.write(pXSector->freq, 8);
|
|
|
|
bitWriter.write(pXSector->atf_6, 1);
|
|
|
|
bitWriter.write(pXSector->atf_7, 1);
|
|
|
|
bitWriter.write(pXSector->phase, 8);
|
|
|
|
bitWriter.write(pXSector->wave, 4);
|
|
|
|
bitWriter.write(pXSector->shadeAlways, 1);
|
|
|
|
bitWriter.write(pXSector->shadeFloor, 1);
|
|
|
|
bitWriter.write(pXSector->shadeCeiling, 1);
|
|
|
|
bitWriter.write(pXSector->shadeWalls, 1);
|
|
|
|
bitWriter.write(pXSector->shade, 8);
|
|
|
|
bitWriter.write(pXSector->panAlways, 1);
|
|
|
|
bitWriter.write(pXSector->panFloor, 1);
|
|
|
|
bitWriter.write(pXSector->panCeiling, 1);
|
|
|
|
bitWriter.write(pXSector->Drag, 1);
|
|
|
|
bitWriter.write(pXSector->Underwater, 1);
|
|
|
|
bitWriter.write(pXSector->Depth, 3);
|
|
|
|
bitWriter.write(pXSector->panVel, 8);
|
|
|
|
bitWriter.write(pXSector->panAngle, 11);
|
|
|
|
bitWriter.write(pXSector->at16_3, 1);
|
|
|
|
bitWriter.write(pXSector->decoupled, 1);
|
|
|
|
bitWriter.write(pXSector->triggerOnce, 1);
|
2019-10-11 21:59:39 +00:00
|
|
|
bitWriter.write(pXSector->isTriggered, 1);
|
2019-06-29 15:37:04 +00:00
|
|
|
bitWriter.write(pXSector->Key, 3);
|
|
|
|
bitWriter.write(pXSector->Push, 1);
|
|
|
|
bitWriter.write(pXSector->Vector, 1);
|
|
|
|
bitWriter.write(pXSector->Reserved, 1);
|
|
|
|
bitWriter.write(pXSector->Enter, 1);
|
|
|
|
bitWriter.write(pXSector->Exit, 1);
|
|
|
|
bitWriter.write(pXSector->Wallpush, 1);
|
|
|
|
bitWriter.write(pXSector->color, 1);
|
|
|
|
bitWriter.write(pXSector->at18_1, 1);
|
|
|
|
bitWriter.write(pXSector->busyTimeB, 12);
|
|
|
|
bitWriter.write(pXSector->waitTimeB, 12);
|
2019-10-11 21:59:39 +00:00
|
|
|
bitWriter.write(pXSector->stopOn, 1);
|
|
|
|
bitWriter.write(pXSector->stopOff, 1);
|
2019-06-29 15:37:04 +00:00
|
|
|
bitWriter.write(pXSector->ceilpal, 4);
|
|
|
|
bitWriter.write(pXSector->at1c_0, 32);
|
|
|
|
bitWriter.write(pXSector->at20_0, 32);
|
|
|
|
bitWriter.write(pXSector->at24_0, 32);
|
|
|
|
bitWriter.write(pXSector->at28_0, 32);
|
2019-10-11 21:59:39 +00:00
|
|
|
bitWriter.write(pXSector->marker0, 16);
|
|
|
|
bitWriter.write(pXSector->marker1, 16);
|
2019-06-29 15:37:04 +00:00
|
|
|
bitWriter.write(pXSector->Crush, 1);
|
|
|
|
bitWriter.write(pXSector->at30_1, 8);
|
|
|
|
bitWriter.write(pXSector->at31_1, 8);
|
|
|
|
bitWriter.write(pXSector->at32_1, 8);
|
|
|
|
bitWriter.write(pXSector->damageType, 3);
|
|
|
|
bitWriter.write(pXSector->floorpal, 4);
|
|
|
|
bitWriter.write(pXSector->at34_0, 8);
|
|
|
|
bitWriter.write(pXSector->locked, 1);
|
|
|
|
bitWriter.write(pXSector->windVel, 10);
|
|
|
|
bitWriter.write(pXSector->windAng, 11);
|
|
|
|
bitWriter.write(pXSector->windAlways, 1);
|
|
|
|
bitWriter.write(pXSector->at37_7, 1);
|
|
|
|
bitWriter.write(pXSector->bobTheta, 11);
|
|
|
|
bitWriter.write(pXSector->bobZRange, 5);
|
|
|
|
bitWriter.write(pXSector->bobSpeed, 12);
|
|
|
|
bitWriter.write(pXSector->bobAlways, 1);
|
|
|
|
bitWriter.write(pXSector->bobFloor, 1);
|
|
|
|
bitWriter.write(pXSector->bobCeiling, 1);
|
|
|
|
bitWriter.write(pXSector->bobRotate, 1);
|
|
|
|
IOBuffer1.Write(pBuffer, nXSectorSize);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (int i = 0; i < numwalls; i++)
|
|
|
|
{
|
|
|
|
if (byte_1A76C8)
|
|
|
|
{
|
|
|
|
dbCrypt((char*)&wall[i], sizeof(walltype), gMapRev*sizeof(sectortype) | 0x7474614d);
|
|
|
|
}
|
|
|
|
IOBuffer1.Write(&wall[i], sizeof(walltype));
|
|
|
|
if (byte_1A76C8)
|
|
|
|
{
|
|
|
|
dbCrypt((char*)&wall[i], sizeof(walltype), gMapRev*sizeof(sectortype) | 0x7474614d);
|
|
|
|
}
|
|
|
|
if (wall[i].extra > 0)
|
|
|
|
{
|
|
|
|
char pBuffer[nXWallSize];
|
|
|
|
BitWriter bitWriter(pBuffer, nXWallSize);
|
|
|
|
XWALL* pXWall = &xwall[wall[i].extra];
|
|
|
|
bitWriter.write(pXWall->reference, 14);
|
|
|
|
bitWriter.write(pXWall->state, 1);
|
|
|
|
bitWriter.write(pXWall->busy, 17);
|
|
|
|
bitWriter.write(pXWall->data, 16);
|
|
|
|
bitWriter.write(pXWall->txID, 10);
|
|
|
|
bitWriter.write(pXWall->at7_2, 6);
|
|
|
|
bitWriter.write(pXWall->rxID, 10);
|
|
|
|
bitWriter.write(pXWall->command, 8);
|
|
|
|
bitWriter.write(pXWall->triggerOn, 1);
|
|
|
|
bitWriter.write(pXWall->triggerOff, 1);
|
|
|
|
bitWriter.write(pXWall->busyTime, 12);
|
|
|
|
bitWriter.write(pXWall->waitTime, 12);
|
|
|
|
bitWriter.write(pXWall->restState, 1);
|
|
|
|
bitWriter.write(pXWall->interruptable, 1);
|
|
|
|
bitWriter.write(pXWall->panAlways, 1);
|
|
|
|
bitWriter.write(pXWall->panXVel, 8);
|
|
|
|
bitWriter.write(pXWall->panYVel, 8);
|
|
|
|
bitWriter.write(pXWall->decoupled, 1);
|
|
|
|
bitWriter.write(pXWall->triggerOnce, 1);
|
|
|
|
bitWriter.write(pXWall->isTriggered, 1);
|
|
|
|
bitWriter.write(pXWall->key, 3);
|
|
|
|
bitWriter.write(pXWall->triggerPush, 1);
|
|
|
|
bitWriter.write(pXWall->triggerVector, 1);
|
2019-09-21 11:02:17 +00:00
|
|
|
bitWriter.write(pXWall->triggerTouch, 1);
|
2019-06-29 15:37:04 +00:00
|
|
|
bitWriter.write(pXWall->at11_0, 2);
|
|
|
|
bitWriter.write(pXWall->xpanFrac, 8);
|
|
|
|
bitWriter.write(pXWall->ypanFrac, 8);
|
|
|
|
bitWriter.write(pXWall->locked, 1);
|
|
|
|
bitWriter.write(pXWall->dudeLockout, 1);
|
|
|
|
bitWriter.write(pXWall->at13_4, 4);
|
|
|
|
bitWriter.write(pXWall->at14_0, 32);
|
|
|
|
IOBuffer1.Write(pBuffer, nXWallSize);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (int i = 0; i < kMaxSprites; i++)
|
|
|
|
{
|
|
|
|
if (sprite[i].statnum < kMaxStatus)
|
|
|
|
{
|
|
|
|
if (byte_1A76C8)
|
|
|
|
{
|
|
|
|
dbCrypt((char*)&sprite[i], sizeof(spritetype), gMapRev*sizeof(spritetype) | 'ttaM');
|
|
|
|
}
|
|
|
|
IOBuffer1.Write(&sprite[i], sizeof(spritetype));
|
|
|
|
if (byte_1A76C8)
|
|
|
|
{
|
|
|
|
dbCrypt((char*)&sprite[i], sizeof(spritetype), gMapRev*sizeof(spritetype) | 'ttaM');
|
|
|
|
}
|
|
|
|
if (sprite[i].extra > 0)
|
|
|
|
{
|
|
|
|
char pBuffer[nXSpriteSize];
|
|
|
|
BitWriter bitWriter(pBuffer, nXSpriteSize);
|
|
|
|
XSPRITE* pXSprite = &xsprite[sprite[i].extra];
|
|
|
|
bitWriter.write(pXSprite->reference, 14);
|
|
|
|
bitWriter.write(pXSprite->state, 1);
|
|
|
|
bitWriter.write(pXSprite->busy, 17);
|
|
|
|
bitWriter.write(pXSprite->txID, 10);
|
|
|
|
bitWriter.write(pXSprite->rxID, 10);
|
|
|
|
bitWriter.write(pXSprite->command, 8);
|
|
|
|
bitWriter.write(pXSprite->triggerOn, 1);
|
|
|
|
bitWriter.write(pXSprite->triggerOff, 1);
|
|
|
|
bitWriter.write(pXSprite->wave, 2);
|
|
|
|
bitWriter.write(pXSprite->busyTime, 12);
|
|
|
|
bitWriter.write(pXSprite->waitTime, 12);
|
|
|
|
bitWriter.write(pXSprite->restState, 1);
|
|
|
|
bitWriter.write(pXSprite->Interrutable, 1);
|
|
|
|
bitWriter.write(pXSprite->atb_2, 2);
|
|
|
|
bitWriter.write(pXSprite->respawnPending, 2);
|
|
|
|
bitWriter.write(pXSprite->atb_6, 1);
|
|
|
|
bitWriter.write(pXSprite->lT, 1);
|
|
|
|
bitWriter.write(pXSprite->dropMsg, 8);
|
|
|
|
bitWriter.write(pXSprite->Decoupled, 1);
|
|
|
|
bitWriter.write(pXSprite->triggerOnce, 1);
|
|
|
|
bitWriter.write(pXSprite->isTriggered, 1);
|
|
|
|
bitWriter.write(pXSprite->key, 3);
|
|
|
|
bitWriter.write(pXSprite->Push, 1);
|
|
|
|
bitWriter.write(pXSprite->Vector, 1);
|
|
|
|
bitWriter.write(pXSprite->Impact, 1);
|
|
|
|
bitWriter.write(pXSprite->Pickup, 1);
|
|
|
|
bitWriter.write(pXSprite->Touch, 1);
|
|
|
|
bitWriter.write(pXSprite->Sight, 1);
|
|
|
|
bitWriter.write(pXSprite->Proximity, 1);
|
|
|
|
bitWriter.write(pXSprite->ate_5, 2);
|
|
|
|
bitWriter.write(pXSprite->lSkill, 5);
|
|
|
|
bitWriter.write(pXSprite->lS, 1);
|
|
|
|
bitWriter.write(pXSprite->lB, 1);
|
|
|
|
bitWriter.write(pXSprite->lC, 1);
|
|
|
|
bitWriter.write(pXSprite->DudeLockout, 1);
|
|
|
|
bitWriter.write(pXSprite->data1, 16);
|
|
|
|
bitWriter.write(pXSprite->data2, 16);
|
|
|
|
bitWriter.write(pXSprite->data3, 16);
|
|
|
|
bitWriter.write(pXSprite->goalAng, 11);
|
|
|
|
bitWriter.write(pXSprite->dodgeDir, 2);
|
|
|
|
bitWriter.write(pXSprite->locked, 1);
|
|
|
|
bitWriter.write(pXSprite->medium, 2);
|
|
|
|
bitWriter.write(pXSprite->respawn, 2);
|
|
|
|
bitWriter.write(pXSprite->data4, 16);
|
|
|
|
bitWriter.write(pXSprite->at1a_2, 6);
|
|
|
|
bitWriter.write(pXSprite->lockMsg, 8);
|
|
|
|
bitWriter.write(pXSprite->health, 12);
|
|
|
|
bitWriter.write(pXSprite->dudeDeaf, 1);
|
|
|
|
bitWriter.write(pXSprite->dudeAmbush, 1);
|
|
|
|
bitWriter.write(pXSprite->dudeGuard, 1);
|
|
|
|
bitWriter.write(pXSprite->dudeFlag4, 1);
|
|
|
|
bitWriter.write(pXSprite->target, 16);
|
|
|
|
bitWriter.write(pXSprite->targetX, 32);
|
|
|
|
bitWriter.write(pXSprite->targetY, 32);
|
|
|
|
bitWriter.write(pXSprite->targetZ, 32);
|
|
|
|
bitWriter.write(pXSprite->burnTime, 16);
|
|
|
|
bitWriter.write(pXSprite->burnSource, 16);
|
|
|
|
bitWriter.write(pXSprite->height, 16);
|
|
|
|
bitWriter.write(pXSprite->stateTimer, 16);
|
|
|
|
IOBuffer1.Write(pBuffer, nXSpriteSize);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-09-24 02:22:16 +00:00
|
|
|
unsigned int nCRC = Bcrc32(pData, nSize-4, 0);
|
2019-06-29 15:37:04 +00:00
|
|
|
IOBuffer1.Write(&nCRC, 4);
|
|
|
|
int nHandle = Bopen(sMapExt, BO_BINARY|BO_TRUNC|BO_CREAT|BO_WRONLY, BS_IREAD|BS_IWRITE);
|
|
|
|
if (nHandle == -1)
|
|
|
|
{
|
|
|
|
initprintf("Couldn't open \"%s\" for writing: %s\n", sMapExt, strerror(errno));
|
|
|
|
Bfree(pData);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (Bwrite(nHandle, pData, nSize) != nSize)
|
|
|
|
{
|
|
|
|
initprintf("Couldn't write to \"%s\": %s\n", sMapExt, strerror(errno));
|
|
|
|
Bclose(nHandle);
|
|
|
|
Bfree(pData);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
Bclose(nHandle);
|
|
|
|
Bfree(pData);
|
|
|
|
return 0;
|
2019-09-22 06:39:22 +00:00
|
|
|
|
2019-06-29 15:37:04 +00:00
|
|
|
}
|
|
|
|
|
2019-06-28 16:03:47 +00:00
|
|
|
int32_t qloadboard(const char* filename, char flags, vec3_t* dapos, int16_t* daang, int16_t* dacursectnum)
|
|
|
|
{
|
2019-06-29 15:37:04 +00:00
|
|
|
// NUKE-TODO: implement flags, see mapedit.cpp
|
2019-06-28 16:03:47 +00:00
|
|
|
return dbLoadMap(filename, &dapos->x, &dapos->y, &dapos->z, (short*)daang, (short*)dacursectnum, NULL);
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
2019-06-29 15:37:04 +00:00
|
|
|
|
|
|
|
int32_t qsaveboard(const char* filename, const vec3_t* dapos, int16_t daang, int16_t dacursectnum)
|
|
|
|
{
|
|
|
|
// NUKE-TODO: see mapedit.cpp
|
|
|
|
byte_1A76C6 = byte_1A76C8 = byte_1A76C7 = 1;
|
|
|
|
return dbSaveMap(filename, dapos->x, dapos->y, dapos->z, daang, dacursectnum);
|
|
|
|
}
|
2019-09-22 06:39:22 +00:00
|
|
|
|
|
|
|
END_BLD_NS
|