//------------------------------------------------------------------------- /* 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. */ //------------------------------------------------------------------------- #include "ns.h" // Must come before everything else! #include #include "build.h" #include "common_game.h" #include "blood.h" #include "db.h" #include "eventq.h" #include "globals.h" #include "levels.h" #include "loadsave.h" #include "sound.h" #include "seq.h" #include "gameutil.h" #include "actor.h" #include "view.h" #include "raze_sound.h" #include "seqcb.h" BEGIN_BLD_NS static void (*seqClientCallback[])(int, int) = { FireballSeqCallback, sub_38938, NapalmSeqCallback, sub_3888C, TreeToGibCallback, DudeToGibCallback1, DudeToGibCallback2, batBiteSeqCallback, SlashSeqCallback, StompSeqCallback, eelBiteSeqCallback, BurnSeqCallback, SeqAttackCallback, cerberusBiteSeqCallback, cerberusBurnSeqCallback, cerberusBurnSeqCallback2, TommySeqCallback, TeslaSeqCallback, ShotSeqCallback, cultThrowSeqCallback, sub_68170, sub_68230, SlashFSeqCallback, ThrowFSeqCallback, ThrowSSeqCallback, BlastSSeqCallback, ghostSlashSeqCallback, ghostThrowSeqCallback, ghostBlastSeqCallback, GillBiteSeqCallback, HandJumpSeqCallback, houndBiteSeqCallback, houndBurnSeqCallback, sub_6FFA0, sub_70284, sub_6FF08, sub_6FF54, ratBiteSeqCallback, SpidBiteSeqCallback, SpidJumpSeqCallback, sub_71370, sub_71BD4, sub_720AC, sub_71A90, genDudeAttack1, punchCallback, ThrowCallback1, ThrowCallback2, HackSeqCallback, StandSeqCallback, zombfHackSeqCallback, PukeSeqCallback, ThrowSeqCallback, PlayerSurvive, PlayerKneelsOver, FireballTrapSeqCallback, MGunFireSeqCallback, MGunOpenSeqCallback, }; enum { kMaxSeqClients = 256, kMaxSequences = 1024 }; static ACTIVE activeList[kMaxSequences]; static int seqActiveCount = 0; void Seq::Preload(void) { if (memcmp(signature, "SEQ\x1a", 4) != 0) I_Error("Invalid sequence"); if ((version & 0xff00) != 0x300) I_Error("Obsolete sequence version"); for (int i = 0; i < nFrames; i++) tilePreloadTile(seqGetTile(&frames[i])); } void Seq::Precache(HitList &hits) { if (memcmp(signature, "SEQ\x1a", 4) != 0) I_Error("Invalid sequence"); if ((version & 0xff00) != 0x300) I_Error("Obsolete sequence version"); for (int i = 0; i < nFrames; i++) tilePrecacheTile(seqGetTile(&frames[i]), -1, hits); } void seqPrecacheId(int id, HitList &hits) { auto pSeq = getSequence(id); if (pSeq) pSeq->Precache(hits); } SEQINST siWall[kMaxXWalls]; SEQINST siCeiling[kMaxXSectors]; SEQINST siFloor[kMaxXSectors]; SEQINST siSprite[kMaxXSprites]; SEQINST siMasked[kMaxXWalls]; void UpdateSprite(int nXSprite, SEQFRAME *pFrame) { assert(nXSprite > 0 && nXSprite < kMaxXSprites); int nSprite = xsprite[nXSprite].reference; assert(nSprite >= 0 && nSprite < kMaxSprites); spritetype *pSprite = &sprite[nSprite]; assert(pSprite->extra == nXSprite); if (pSprite->flags & 2) { if (tilesiz[pSprite->picnum].y != tilesiz[seqGetTile(pFrame)].y || tileTopOffset(pSprite->picnum) != tileTopOffset(seqGetTile(pFrame)) || (pFrame->at3_0 && pFrame->at3_0 != pSprite->yrepeat)) pSprite->flags |= 4; } pSprite->picnum = seqGetTile(pFrame); if (pFrame->at5_0) pSprite->pal = pFrame->at5_0; pSprite->shade = pFrame->at4_0; int scale = xsprite[nXSprite].scale; // SEQ size scaling if (pFrame->at2_0) { if (scale) pSprite->xrepeat = ClipRange(mulscale8(pFrame->at2_0, scale), 0, 255); else pSprite->xrepeat = pFrame->at2_0; } if (pFrame->at3_0) { if (scale) pSprite->yrepeat = ClipRange(mulscale8(pFrame->at3_0, scale), 0, 255); else pSprite->yrepeat = pFrame->at3_0; } if (pFrame->at1_4) pSprite->cstat |= 2; else pSprite->cstat &= ~2; if (pFrame->at1_5) pSprite->cstat |= 512; else pSprite->cstat &= ~512; if (pFrame->at1_6) pSprite->cstat |= 1; else pSprite->cstat &= ~1; if (pFrame->at1_7) pSprite->cstat |= 256; else pSprite->cstat &= ~256; if (pFrame->at6_2) pSprite->cstat |= 32768; else pSprite->cstat &= (unsigned short)~32768; if (pFrame->at6_0) pSprite->cstat |= 4096; else pSprite->cstat &= ~4096; if (pFrame->at5_6) pSprite->flags |= 256; else pSprite->flags &= ~256; if (pFrame->at5_7) pSprite->flags |= 8; else pSprite->flags &= ~8; if (pFrame->at6_3) pSprite->flags |= 1024; else pSprite->flags &= ~1024; if (pFrame->at6_4) pSprite->flags |= 2048; else pSprite->flags &= ~2048; } void UpdateWall(int nXWall, SEQFRAME *pFrame) { assert(nXWall > 0 && nXWall < kMaxXWalls); int nWall = xwall[nXWall].reference; assert(nWall >= 0 && nWall < kMaxWalls); walltype *pWall = &wall[nWall]; assert(pWall->extra == nXWall); pWall->picnum = seqGetTile(pFrame); if (pFrame->at5_0) pWall->pal = pFrame->at5_0; if (pFrame->at1_4) pWall->cstat |= 128; else pWall->cstat &= ~128; if (pFrame->at1_5) pWall->cstat |= 512; else pWall->cstat &= ~512; if (pFrame->at1_6) pWall->cstat |= 1; else pWall->cstat &= ~1; if (pFrame->at1_7) pWall->cstat |= 64; else pWall->cstat &= ~64; } void UpdateMasked(int nXWall, SEQFRAME *pFrame) { assert(nXWall > 0 && nXWall < kMaxXWalls); int nWall = xwall[nXWall].reference; assert(nWall >= 0 && nWall < kMaxWalls); walltype *pWall = &wall[nWall]; assert(pWall->extra == nXWall); assert(pWall->nextwall >= 0); walltype *pWallNext = &wall[pWall->nextwall]; pWall->overpicnum = pWallNext->overpicnum = seqGetTile(pFrame); if (pFrame->at5_0) pWall->pal = pWallNext->pal = pFrame->at5_0; if (pFrame->at1_4) { pWall->cstat |= 128; pWallNext->cstat |= 128; } else { pWall->cstat &= ~128; pWallNext->cstat &= ~128; } if (pFrame->at1_5) { pWall->cstat |= 512; pWallNext->cstat |= 512; } else { pWall->cstat &= ~512; pWallNext->cstat &= ~512; } if (pFrame->at1_6) { pWall->cstat |= 1; pWallNext->cstat |= 1; } else { pWall->cstat &= ~1; pWallNext->cstat &= ~1; } if (pFrame->at1_7) { pWall->cstat |= 64; pWallNext->cstat |= 64; } else { pWall->cstat &= ~64; pWallNext->cstat &= ~64; } } void UpdateFloor(int nXSector, SEQFRAME *pFrame) { assert(nXSector > 0 && nXSector < kMaxXSectors); int nSector = xsector[nXSector].reference; assert(nSector >= 0 && nSector < kMaxSectors); sectortype *pSector = §or[nSector]; assert(pSector->extra == nXSector); pSector->floorpicnum = seqGetTile(pFrame); pSector->floorshade = pFrame->at4_0; if (pFrame->at5_0) pSector->floorpal = pFrame->at5_0; } void UpdateCeiling(int nXSector, SEQFRAME *pFrame) { assert(nXSector > 0 && nXSector < kMaxXSectors); int nSector = xsector[nXSector].reference; assert(nSector >= 0 && nSector < kMaxSectors); sectortype *pSector = §or[nSector]; assert(pSector->extra == nXSector); pSector->ceilingpicnum = seqGetTile(pFrame); pSector->ceilingshade = pFrame->at4_0; if (pFrame->at5_0) pSector->ceilingpal = pFrame->at5_0; } void SEQINST::Update(ACTIVE *pActive) { assert(frameIndex < pSequence->nFrames); switch (pActive->type) { case 0: UpdateWall(pActive->xindex, &pSequence->frames[frameIndex]); break; case 1: UpdateCeiling(pActive->xindex , &pSequence->frames[frameIndex]); break; case 2: UpdateFloor(pActive->xindex, &pSequence->frames[frameIndex]); break; case 3: { UpdateSprite(pActive->xindex, &pSequence->frames[frameIndex]); if (pSequence->frames[frameIndex].at6_1) { int sound = pSequence->ata; // by NoOne: add random sound range feature if (!VanillaMode() && pSequence->frames[frameIndex].soundRange > 0) sound += Random(((pSequence->frames[frameIndex].soundRange == 1) ? 2 : pSequence->frames[frameIndex].soundRange)); sfxPlay3DSound(&sprite[xsprite[pActive->xindex].reference], sound, -1, 0); } // by NoOne: add surfaceSound trigger feature spritetype* pSprite = &sprite[xsprite[pActive->xindex].reference]; if (!VanillaMode() && pSequence->frames[frameIndex].surfaceSound && zvel[pSprite->index] == 0 && xvel[pSprite->index] != 0) { if (gUpperLink[pSprite->sectnum] >= 0) break; // don't play surface sound for stacked sectors int surf = tileGetSurfType(pSprite->sectnum + 0x4000); if (!surf) break; static int surfSfxMove[15][4] = { /* {snd1, snd2, gameVolume, myVolume} */ {800,801,80,25}, {802,803,80,25}, {804,805,80,25}, {806,807,80,25}, {808,809,80,25}, {810,811,80,25}, {812,813,80,25}, {814,815,80,25}, {816,817,80,25}, {818,819,80,25}, {820,821,80,25}, {822,823,80,25}, {824,825,80,25}, {826,827,80,25}, {828,829,80,25}, }; int sndId = surfSfxMove[surf][Random(2)]; auto snd = soundEngine->FindSoundByResID(sndId); if (snd > 0) { auto udata = soundEngine->GetUserData(snd); int relVol = udata ? udata[2] : 255; sfxPlay3DSoundCP(pSprite, sndId, -1, 0, 0, (surfSfxMove[surf][2] != relVol) ? relVol : surfSfxMove[surf][3]); } } break; } case 4: UpdateMasked(pActive->xindex, &pSequence->frames[frameIndex]); break; } if (pSequence->frames[frameIndex].at5_5 && atc != -1) seqClientCallback[atc](pActive->type, pActive->xindex); } SEQINST * GetInstance(int a1, int a2) { switch (a1) { case 0: if (a2 > 0 && a2 < kMaxXWalls) return &siWall[a2]; break; case 1: if (a2 > 0 && a2 < kMaxXSectors) return &siCeiling[a2]; break; case 2: if (a2 > 0 && a2 < kMaxXSectors) return &siFloor[a2]; break; case 3: if (a2 > 0 && a2 < kMaxXSprites) return &siSprite[a2]; break; case 4: if (a2 > 0 && a2 < kMaxWalls) return &siMasked[a2]; break; } return NULL; } void UnlockInstance(SEQINST *pInst) { assert(pInst != NULL); assert(pInst->pSequence != NULL); pInst->pSequence = NULL; pInst->at13 = 0; } void seqSpawn(int a1, int a2, int a3, int a4) { SEQINST *pInst = GetInstance(a2, a3); if (!pInst) return; auto pSeq = getSequence(a1); if (!pSeq) I_Error("Missing sequence #%d", a1); int i = seqActiveCount; if (pInst->at13) { if (pSeq == pInst->pSequence) return; UnlockInstance(pInst); for (i = 0; i < seqActiveCount; i++) { if (activeList[i].type == a2 && activeList[i].xindex == a3) break; } assert(i < seqActiveCount); } if (memcmp(pSeq->signature, "SEQ\x1a", 4) != 0) I_Error("Invalid sequence %d", a1); if ((pSeq->version & 0xff00) != 0x300) I_Error("Sequence %d is obsolete version", a1); if ((pSeq->version & 0xff) == 0x00) { for (int i = 0; i < pSeq->nFrames; i++) pSeq->frames[i].tile2 = 0; } pInst->at13 = 1; pInst->pSequence = pSeq; pInst->at8 = a1; pInst->atc = a4; pInst->at10 = pSeq->at8; pInst->frameIndex = 0; if (i == seqActiveCount) { assert(seqActiveCount < kMaxSequences); activeList[seqActiveCount].type = a2; activeList[seqActiveCount].xindex = a3; seqActiveCount++; } pInst->Update(&activeList[i]); } void seqKill(int a1, int a2) { SEQINST *pInst = GetInstance(a1, a2); if (!pInst || !pInst->at13) return; int i; for (i = 0; i < seqActiveCount; i++) { if (activeList[i].type == a1 && activeList[i].xindex == a2) break; } assert(i < seqActiveCount); seqActiveCount--; activeList[i] = activeList[seqActiveCount]; pInst->at13 = 0; UnlockInstance(pInst); } void seqKillAll(void) { for (int i = 0; i < kMaxXWalls; i++) { if (siWall[i].at13) UnlockInstance(&siWall[i]); if (siMasked[i].at13) UnlockInstance(&siMasked[i]); } for (int i = 0; i < kMaxXSectors; i++) { if (siCeiling[i].at13) UnlockInstance(&siCeiling[i]); if (siFloor[i].at13) UnlockInstance(&siFloor[i]); } for (int i = 0; i < kMaxXSprites; i++) { if (siSprite[i].at13) UnlockInstance(&siSprite[i]); } seqActiveCount = 0; } int seqGetStatus(int a1, int a2) { SEQINST *pInst = GetInstance(a1, a2); if (pInst && pInst->at13) return pInst->frameIndex; return -1; } int seqGetID(int a1, int a2) { SEQINST *pInst = GetInstance(a1, a2); if (pInst) return pInst->at8; return -1; } void seqProcess(int a1) { for (int i = 0; i < seqActiveCount; i++) { SEQINST *pInst = GetInstance(activeList[i].type, activeList[i].xindex); Seq *pSeq = pInst->pSequence; assert(pInst->frameIndex < pSeq->nFrames); pInst->at10 -= a1; while (pInst->at10 < 0) { pInst->at10 += pSeq->at8; pInst->frameIndex++; if (pInst->frameIndex == pSeq->nFrames) { if (pSeq->atc & 1) pInst->frameIndex = 0; else { UnlockInstance(pInst); if (pSeq->atc & 2) { switch (activeList[i].type) { case 3: { int nXSprite = activeList[i].xindex; int nSprite = xsprite[nXSprite].reference; assert(nSprite >= 0 && nSprite < kMaxSprites); evKill(nSprite, 3); if ((sprite[nSprite].flags & kHitagRespawn) && sprite[nSprite].inittype >= kDudeBase && sprite[nSprite].inittype < kDudeMax) evPost(nSprite, 3, gGameOptions.nMonsterRespawnTime, kCallbackRespawn); else DeleteSprite(nSprite); break; } case 4: { int nXWall = activeList[i].xindex; int nWall = xwall[nXWall].reference; assert(nWall >= 0 && nWall < kMaxWalls); wall[nWall].cstat &= ~(8 + 16 + 32); if (wall[nWall].nextwall != -1) wall[wall[nWall].nextwall].cstat &= ~(8 + 16 + 32); break; } } } activeList[i--] = activeList[--seqActiveCount]; break; } } pInst->Update(&activeList[i]); } } } class SeqLoadSave : public LoadSave { virtual void Load(void); virtual void Save(void); }; void SeqLoadSave::Load(void) { Read(&siWall, sizeof(siWall)); Read(&siMasked, sizeof(siMasked)); Read(&siCeiling, sizeof(siCeiling)); Read(&siFloor, sizeof(siFloor)); Read(&siSprite, sizeof(siSprite)); Read(&activeList, sizeof(activeList)); Read(&seqActiveCount, sizeof(seqActiveCount)); for (int i = 0; i < kMaxXWalls; i++) { siWall[i].pSequence = NULL; siMasked[i].pSequence = NULL; } for (int i = 0; i < kMaxXSectors; i++) { siCeiling[i].pSequence = NULL; siFloor[i].pSequence = NULL; } for (int i = 0; i < kMaxXSprites; i++) { siSprite[i].pSequence = NULL; } for (int i = 0; i < seqActiveCount; i++) { SEQINST *pInst = GetInstance(activeList[i].type, activeList[i].xindex); if (pInst->at13) { int nSeq = pInst->at8; auto pSeq = getSequence(nSeq); if (!pSeq) { I_Error("Missing sequence #%d", nSeq); continue; } if (memcmp(pSeq->signature, "SEQ\x1a", 4) != 0) I_Error("Invalid sequence %d", nSeq); if ((pSeq->version & 0xff00) != 0x300) I_Error("Sequence %d is obsolete version", nSeq); pInst->pSequence = pSeq; } } } void SeqLoadSave::Save(void) { Write(&siWall, sizeof(siWall)); Write(&siMasked, sizeof(siMasked)); Write(&siCeiling, sizeof(siCeiling)); Write(&siFloor, sizeof(siFloor)); Write(&siSprite, sizeof(siSprite)); Write(&activeList, sizeof(activeList)); Write(&seqActiveCount, sizeof(seqActiveCount)); } void SeqLoadSaveConstruct(void) { new SeqLoadSave(); } static void ByteSwapSEQ(Seq* pSeq) { #if B_BIG_ENDIAN == 1 pSeq->version = LittleShort(pSeq->version); pSeq->nFrames = LittleShort(pSeq->nFrames); pSeq->at8 = LittleShort(pSeq->at8); pSeq->ata = LittleShort(pSeq->ata); pSeq->atc = LittleLong(pSeq->atc); for (int i = 0; i < pSeq->nFrames; i++) { SEQFRAME* pFrame = &pSeq->frames[i]; BitReader bitReader((char*)pFrame, sizeof(SEQFRAME)); SEQFRAME swapFrame; swapFrame.tile = bitReader.readUnsigned(12); swapFrame.at1_4 = bitReader.readBit(); swapFrame.at1_5 = bitReader.readBit(); swapFrame.at1_6 = bitReader.readBit(); swapFrame.at1_7 = bitReader.readBit(); swapFrame.at2_0 = bitReader.readUnsigned(8); swapFrame.at3_0 = bitReader.readUnsigned(8); swapFrame.at4_0 = bitReader.readSigned(8); swapFrame.at5_0 = bitReader.readUnsigned(5); swapFrame.at5_5 = bitReader.readBit(); swapFrame.at5_6 = bitReader.readBit(); swapFrame.at5_7 = bitReader.readBit(); swapFrame.at6_0 = bitReader.readBit(); swapFrame.at6_1 = bitReader.readBit(); swapFrame.at6_2 = bitReader.readBit(); swapFrame.at6_3 = bitReader.readBit(); swapFrame.at6_4 = bitReader.readBit(); swapFrame.tile2 = bitReader.readUnsigned(4); swapFrame.soundRange = bitReader.readUnsigned(4); swapFrame.surfaceSound = bitReader.readBit(); swapFrame.reserved = bitReader.readUnsigned(2); *pFrame = swapFrame; } #endif } // This is to eliminate a huge design issue in NBlood that was apparently copied verbatim from the DOS-Version. // Sequences were cached in the resource and directly returned from there in writable form, with byte swapping directly performed in the cache on Big Endian systems. // To avoid such unsafe operations this caches the read data separately in static memory that's guaranteed not to be touched by the file system. FMemArena seqcache; static TMap sequences; Seq* getSequence(int res_id) { auto p = sequences.CheckKey(res_id); if (p != nullptr) return *p; int index = fileSystem.FindResource(res_id, "SEQ"); if (index < 0) { return nullptr; } auto fr = fileSystem.OpenFileReader(index); auto seqdata = (Seq*)seqcache.Alloc(fr.GetLength()); fr.Read(seqdata, fr.GetLength()); sequences.Insert(res_id, seqdata); ByteSwapSEQ(seqdata); return seqdata; } END_BLD_NS