/* * Descent 3 * Copyright (C) 2024 Parallax Software * * 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 3 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, see . --- HISTORICAL COMMENTS FOLLOW --- * $Logfile: /DescentIII/Main/2dlib/memsurf.cpp $ * $Revision: 6 $ * $Date: 4/23/98 6:38p $ * $Author: Jason $ * * memory surface class * * $Log: /DescentIII/Main/2dlib/memsurf.cpp $ * * 6 4/23/98 6:38p Jason * made bitmaps use 1555 format * * 5 3/31/98 3:48p Jason * added memory lib * * 4 11/16/97 2:19p Samir * Added constructor takes a bitmap handle. * * 3 11/14/97 12:56p Samir * Took out annoying mprintf. * * 2 10/01/97 4:22p Samir * Fixed memory leak in memsurf library destruction * * 12 6/12/97 6:29p Samir * DDGR v2.0 Changes in 2d system implemented. * * 11 5/15/97 4:24p Samir * Fixed one-pixel overwrite of clear * * 10 4/10/97 6:45p Samir * Fixed confusion of variables in memory clear function. * * 9 3/28/97 3:07p Samir * Clear function now takes a color * * 8 3/27/97 11:11a Samir * Moved code from ddgr library to 2dlib because it was ANSI compliant and * should work on all systems * * 7 2/03/97 3:00p Samir * Allow for first time initialization code to properly register surface * in surface list. * * 6 1/30/97 6:01p Samir * The partition of memory and os surfaces as well as the change in * ddgr_surface structure and all related changes. * * $NoKeywords: $ */ #include #include #include "bitmap.h" #include "lib2d.h" #include "log.h" #include "mem.h" // ---------------------------------------------------------------------------- // constructor and destructor // ---------------------------------------------------------------------------- grMemorySurface::grMemorySurface() : grSurface() { m_FirstTimeInit = true; m_AllowInit = true; } grMemorySurface::grMemorySurface(int w, int h, int bpp, unsigned flags, const char *name) : grSurface(w, h, bpp, SURFTYPE_MEMORY, flags, name) { m_FirstTimeInit = false; m_AllowInit = false; } grMemorySurface::grMemorySurface(int bm) { m_FirstTimeInit = true; m_AllowInit = true; grMemorySurface::init(bm_w(bm, 0), bm_h(bm, 0), bm_bpp(bm), (char *)bm_data(bm, 0), bm_rowsize(bm, 0), SURFFLAG_COLORKEY); } grMemorySurface::~grMemorySurface() { // this should invoke the grSurface destructor // even if we just initialized the surface, the memory routines should know this and // do appropriate deinitialization. } // ---------------------------------------------------------------------------- // initialize a memory surface with these values // ---------------------------------------------------------------------------- bool grMemorySurface::init(int w, int h, int bpp, char *data, int rowsize, unsigned flags, const char *name) { ASSERT(m_AllowInit); if (m_FirstTimeInit) { // add to list ONLY ONCE! add_to_list(&ddsfObj); m_FirstTimeInit = false; } ddsfObj.w = w; ddsfObj.h = h; ddsfObj.bpp = bpp; ddsfObj.type = SURFTYPE_MEMORY; ddsfObj.flags = (uint16_t)flags; if (name) strncpy(ddsfObj.name, name, 15); if (!gr_mem_surf_Init(&ddsfObj, data, rowsize)) return false; surf_init(1); return true; } // --------------------------------------------------------------------------- // gr_mem_xxx functions for memory surfaces // // internal functions // --------------------------------------------------------------------------- bool gr_bm_mem_Blt16(ddgr_surface *dsf, int dx, int dy, ddgr_surface *ssf, int sx, int sy, int sw, int sh); bool gr_bm_mem_Blt16ck(ddgr_surface *dsf, int dx, int dy, ddgr_surface *ssf, int sx, int sy, int sw, int sh); // --------------------------------------------------------------------------- // initializes a memory surface by allocating memory based on bit depth bool gr_mem_surf_Create(ddgr_surface *sf) { mem_bitmap *bm; bm = new mem_bitmap; memset(bm, 0, sizeof(mem_bitmap)); switch (sf->bpp) { case BPP_16: bm->bpp = BPP_16; bm->rowsize = sf->w * 2; bm->alloced = 1; break; case BPP_32: bm->bpp = BPP_32; bm->rowsize = sf->w * 4; bm->alloced = 1; break; default: Int3(); } bm->data = (char *)mem_malloc(bm->rowsize * sf->h); if (!bm->data) { LOG_WARNING.printf("mem_Create malloc fail <%s>\n", sf->name); delete bm; return false; } sf->obj = (void *)bm; return true; } // --------------------------------------------------------------------------- // deallocates memory created in a mem_Create bool gr_mem_surf_Destroy(ddgr_surface *sf) { ASSERT(sf); mem_bitmap *bm = (mem_bitmap *)sf->obj; // here, we check if we initialized, or allocated this memory surface. if (bm->alloced == 2) { delete bm; return false; } else if (bm->alloced == 0) { return false; } // otherwise free all objects mem_free(bm->data); delete bm; return true; } // --------------------------------------------------------------------------- // initializes a memory surface bool gr_mem_surf_Init(ddgr_surface *sf, char *data, int rowsize) { // initialize a memory surface, with preallocated memory mem_bitmap *mbm; // create new memory bitmap object for initialized surface if (!sf->obj) { mbm = new mem_bitmap; mbm->alloced = 2; } else { mbm = (mem_bitmap *)sf->obj; mbm->alloced = 0; } mbm->data = data; mbm->rowsize = rowsize; mbm->bpp = (int16_t)sf->bpp; mbm->flag = 0; sf->obj = (void *)mbm; return true; } // --------------------------------------------------------------------------- // clears a memory surface void gr_mem_surf_Clear(ddgr_surface *dsf, ddgr_color col, int l, int t, int w, int h) { mem_bitmap *dbm = (mem_bitmap *)dsf->obj; switch (dbm->bpp) { case BPP_16: { uint16_t *sptr; uint16_t scol; scol = (uint16_t)GR_COLOR_TO_16(col); sptr = (uint16_t *)dbm->data + (t * ((dbm->rowsize) >> 1)); for (int y = t; y < (t + h); y++) { for (int x = l; x < (l + w); x++) sptr[x] = scol; sptr += ((dbm->rowsize) >> 1); } break; } case BPP_32: default: Int3(); } } // --------------------------------------------------------------------------- // blts a memory surface to another such surface bool gr_mem_surf_Blt(ddgr_surface *dsf, int dx, int dy, ddgr_surface *ssf, int sx, int sy, int sw, int sh) { // Maybe we should allow slow 16bpp to 24bpp and vice-versa blts. // for now, we wont. mem_bitmap *dbm = (mem_bitmap *)dsf->obj; ASSERT(dbm->bpp == ((mem_bitmap *)ssf->obj)->bpp); switch (dbm->bpp) { case BPP_16: if (ssf->flags & SURFFLAG_COLORKEY) gr_bm_mem_Blt16ck(dsf, dx, dy, ssf, sx, sy, sw, sh); else gr_bm_mem_Blt16(dsf, dx, dy, ssf, sx, sy, sw, sh); break; case BPP_32: default: Int3(); } return true; } // --------------------------------------------------------------------------- // just extracts info from mem_bitmap abstract object bool gr_mem_surf_Lock(ddgr_surface *sf, void **ptr, int *rowsize) { mem_bitmap *bm = (mem_bitmap *)sf->obj; *ptr = (char *)bm->data; *rowsize = bm->rowsize; return true; } // --------------------------------------------------------------------------- // invalidates sf members bool gr_mem_surf_Unlock(ddgr_surface *sf, void *ptr) { return true; } // --------------------------------------------------------------------------- // does shit. bool gr_mem_surf_AttachHandle(ddgr_surface *sf, unsigned handle) { return true; } // --------------------------------------------------------------------------- // performs 16-bit blts. bool gr_bm_mem_Blt16(ddgr_surface *dsf, int dx, int dy, ddgr_surface *ssf, int sx, int sy, int sw, int sh) { mem_bitmap *dbm = (mem_bitmap *)dsf->obj, *sbm = (mem_bitmap *)ssf->obj; uint16_t *dbits; uint16_t *sbits; int srowsize_w, drowsize_w, row, col; // set up blt. srowsize_w = sbm->rowsize >> 1; // rowsize in shorts drowsize_w = dbm->rowsize >> 1; dbits = (uint16_t *)dbm->data + (dy * drowsize_w) + dx; sbits = (uint16_t *)sbm->data + (sy * srowsize_w) + sx; for (row = 0; row < sh; row++) { for (col = 0; col < sw; col++) dbits[col] = sbits[col]; sbits += srowsize_w; dbits += drowsize_w; } return true; } bool gr_bm_mem_Blt16ck(ddgr_surface *dsf, int dx, int dy, ddgr_surface *ssf, int sx, int sy, int sw, int sh) { mem_bitmap *dbm = (mem_bitmap *)dsf->obj, *sbm = (mem_bitmap *)ssf->obj; uint16_t *dbits; uint16_t *sbits; int srowsize_w, drowsize_w, row, col; // set up blt. srowsize_w = sbm->rowsize >> 1; // rowsize in shorts drowsize_w = dbm->rowsize >> 1; dbits = (uint16_t *)dbm->data + (dy * drowsize_w) + dx; sbits = (uint16_t *)sbm->data + (sy * srowsize_w) + sx; for (row = 0; row < sh; row++) { for (col = 0; col < sw; col++) { if (sbits[col] & OPAQUE_FLAG16) dbits[col] = sbits[col]; } sbits += srowsize_w; dbits += drowsize_w; } return true; }