2019-10-28 18:28:42 +00:00
|
|
|
// SONIC ROBO BLAST 2
|
|
|
|
//-----------------------------------------------------------------------------
|
2020-08-15 23:48:28 +00:00
|
|
|
// Copyright (C) 2020 by Jaime "Lactozilla" Passos.
|
2019-10-28 18:28:42 +00:00
|
|
|
//
|
|
|
|
// This program is free software distributed under the
|
|
|
|
// terms of the GNU General Public License, version 2.
|
|
|
|
// See the 'LICENSE' file for more details.
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
/// \file r_patch.c
|
|
|
|
/// \brief Patch generation.
|
|
|
|
|
2020-08-15 23:48:28 +00:00
|
|
|
#include "doomdef.h"
|
2019-10-28 18:28:42 +00:00
|
|
|
#include "r_patch.h"
|
2020-08-15 23:48:28 +00:00
|
|
|
#include "r_defs.h"
|
2019-10-28 18:28:42 +00:00
|
|
|
#include "z_zone.h"
|
|
|
|
|
|
|
|
#ifdef HWRENDER
|
|
|
|
#include "hardware/hw_glob.h"
|
|
|
|
#endif
|
|
|
|
|
2020-08-08 08:16:47 +00:00
|
|
|
//
|
|
|
|
// Creates a patch.
|
|
|
|
// Assumes a PU_PATCH zone memory tag and no user, but can always be set later
|
|
|
|
//
|
|
|
|
|
|
|
|
patch_t *Patch_Create(softwarepatch_t *source, size_t srcsize, void *dest)
|
|
|
|
{
|
|
|
|
patch_t *patch = (dest == NULL) ? Z_Calloc(sizeof(patch_t), PU_PATCH, NULL) : (patch_t *)(dest);
|
|
|
|
|
|
|
|
if (source)
|
|
|
|
{
|
|
|
|
INT32 col, colsize;
|
|
|
|
size_t size = sizeof(INT32) * source->width;
|
|
|
|
size_t offs = (sizeof(INT16) * 4) + size;
|
|
|
|
|
|
|
|
patch->width = source->width;
|
|
|
|
patch->height = source->height;
|
|
|
|
patch->leftoffset = source->leftoffset;
|
|
|
|
patch->topoffset = source->topoffset;
|
2020-10-10 21:43:26 +00:00
|
|
|
patch->columnofs = Z_Calloc(size, PU_PATCH_DATA, NULL);
|
2020-08-08 08:16:47 +00:00
|
|
|
|
|
|
|
for (col = 0; col < source->width; col++)
|
|
|
|
{
|
|
|
|
// This makes the column offsets relative to the column data itself,
|
|
|
|
// instead of the entire patch data
|
|
|
|
patch->columnofs[col] = LONG(source->columnofs[col]) - offs;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!srcsize)
|
|
|
|
I_Error("R_CreatePatch: no source size!");
|
|
|
|
|
|
|
|
colsize = (INT32)(srcsize) - (INT32)offs;
|
|
|
|
if (colsize <= 0)
|
|
|
|
I_Error("R_CreatePatch: no column data!");
|
|
|
|
|
2020-10-10 21:43:26 +00:00
|
|
|
patch->columns = Z_Calloc(colsize, PU_PATCH_DATA, NULL);
|
2020-08-08 08:16:47 +00:00
|
|
|
M_Memcpy(patch->columns, ((UINT8 *)source + LONG(source->columnofs[0])), colsize);
|
|
|
|
}
|
|
|
|
|
|
|
|
return patch;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Frees a patch from memory.
|
|
|
|
//
|
|
|
|
|
2020-09-07 05:23:07 +00:00
|
|
|
static void Patch_FreeData(patch_t *patch)
|
2020-08-08 08:16:47 +00:00
|
|
|
{
|
|
|
|
#ifdef HWRENDER
|
|
|
|
if (patch->hardware)
|
|
|
|
HWR_FreeTexture(patch);
|
|
|
|
#endif
|
|
|
|
|
2020-10-10 21:43:26 +00:00
|
|
|
#ifdef ROTSPRITE
|
|
|
|
if (patch->rotated)
|
|
|
|
{
|
|
|
|
rotsprite_t *rotsprite = patch->rotated;
|
|
|
|
INT32 i = 0;
|
|
|
|
|
|
|
|
for (; i < rotsprite->angles; i++)
|
|
|
|
{
|
|
|
|
if (rotsprite->patches[i])
|
|
|
|
Patch_Free(rotsprite->patches[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
Z_Free(rotsprite->patches);
|
|
|
|
Z_Free(rotsprite);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2020-08-08 08:16:47 +00:00
|
|
|
if (patch->columnofs)
|
|
|
|
Z_Free(patch->columnofs);
|
|
|
|
if (patch->columns)
|
|
|
|
Z_Free(patch->columns);
|
2020-09-07 05:23:07 +00:00
|
|
|
}
|
2020-08-08 08:16:47 +00:00
|
|
|
|
2020-09-07 05:23:07 +00:00
|
|
|
void Patch_Free(patch_t *patch)
|
|
|
|
{
|
|
|
|
Patch_FreeData(patch);
|
2020-08-08 08:16:47 +00:00
|
|
|
Z_Free(patch);
|
|
|
|
}
|
|
|
|
|
2020-09-07 05:23:07 +00:00
|
|
|
//
|
|
|
|
// Frees patches with a tag range.
|
|
|
|
//
|
|
|
|
|
|
|
|
static boolean Patch_FreeTagsCallback(void *mem)
|
|
|
|
{
|
|
|
|
patch_t *patch = (patch_t *)mem;
|
|
|
|
Patch_FreeData(patch);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Patch_FreeTags(INT32 lowtag, INT32 hightag)
|
|
|
|
{
|
|
|
|
Z_IterateTags(lowtag, hightag, Patch_FreeTagsCallback);
|
|
|
|
}
|
|
|
|
|
2020-08-08 08:16:47 +00:00
|
|
|
#ifdef HWRENDER
|
|
|
|
//
|
|
|
|
// Allocates a hardware patch.
|
|
|
|
//
|
|
|
|
|
|
|
|
void *Patch_AllocateHardwarePatch(patch_t *patch)
|
|
|
|
{
|
|
|
|
if (!patch->hardware)
|
|
|
|
{
|
|
|
|
GLPatch_t *grPatch = Z_Calloc(sizeof(GLPatch_t), PU_HWRPATCHINFO, &patch->hardware);
|
|
|
|
grPatch->mipmap = Z_Calloc(sizeof(GLMipmap_t), PU_HWRPATCHINFO, &grPatch->mipmap);
|
|
|
|
}
|
|
|
|
return (void *)(patch->hardware);
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Creates a hardware patch.
|
|
|
|
//
|
|
|
|
|
|
|
|
void *Patch_CreateGL(patch_t *patch)
|
|
|
|
{
|
|
|
|
GLPatch_t *grPatch = (GLPatch_t *)Patch_AllocateHardwarePatch(patch);
|
|
|
|
if (!grPatch->mipmap->data) // Run HWR_MakePatch in all cases, to recalculate some things
|
|
|
|
HWR_MakePatch(patch, grPatch, grPatch->mipmap, false);
|
|
|
|
return grPatch;
|
|
|
|
}
|
|
|
|
#endif // HWRENDER
|