/* * Portions of this file are copyright Rebirth contributors and licensed as * described in COPYING.txt. * Portions of this file are copyright Parallax Software and licensed * according to the Parallax license below. * See COPYING.txt for license details. THE COMPUTER CODE CONTAINED HEREIN IS THE SOLE PROPERTY OF PARALLAX SOFTWARE CORPORATION ("PARALLAX"). PARALLAX, IN DISTRIBUTING THE CODE TO END-USERS, AND SUBJECT TO ALL OF THE TERMS AND CONDITIONS HEREIN, GRANTS A ROYALTY-FREE, PERPETUAL LICENSE TO SUCH END-USERS FOR USE BY SUCH END-USERS IN USING, DISPLAYING, AND CREATING DERIVATIVE WORKS THEREOF, SO LONG AS SUCH USE, DISPLAY OR CREATION IS FOR NON-COMMERCIAL, ROYALTY OR REVENUE FREE PURPOSES. IN NO EVENT SHALL THE END-USER USE THE COMPUTER CODE CONTAINED HEREIN FOR REVENUE-BEARING PURPOSES. THE END-USER UNDERSTANDS AND AGREES TO THE TERMS HEREIN AND ACCEPTS THE SAME BY USE OF THIS FILE. COPYRIGHT 1993-1999 PARALLAX SOFTWARE CORPORATION. ALL RIGHTS RESERVED. */ /* * * Routines for bitblt's. * */ #include #include #include #include "u_mem.h" #include "gr.h" #include "grdef.h" #include "rle.h" #include "dxxerror.h" #include "byteutil.h" #ifdef OGL #include "ogl_init.h" #endif #include "compiler-array.h" static int gr_bitblt_dest_step_shift = 0; static ubyte *gr_bitblt_fade_table=NULL; static void gr_bm_ubitblt00_rle(unsigned w, unsigned h, int dx, int dy, int sx, int sy, const grs_bitmap &src, grs_bitmap &dest); static void gr_bm_ubitblt00m_rle(unsigned w, unsigned h, int dx, int dy, int sx, int sy, const grs_bitmap &src, grs_bitmap &dest); static void gr_bm_ubitblt0x_rle(unsigned w, unsigned h, int dx, int dy, int sx, int sy, const grs_bitmap &src, grs_bitmap &dest); #define gr_linear_movsd(S,D,L) memcpy(D,S,L) static void gr_linear_rep_movsdm(const ubyte *src, ubyte *dest, int num_pixels) { register ubyte c; while (num_pixels--) if ((c=*src++)!=255) *dest++=c; else dest++; } static void gr_linear_rep_movsdm_faded(const ubyte * src, ubyte * dest, int num_pixels, ubyte fade_value ) { auto predicate = [&](ubyte s, ubyte d) { return s == 255 ? d : gr_fade_table[fade_value][s]; }; std::transform(src, src + num_pixels, dest, dest, predicate); } static void gr_ubitmap00(unsigned x, unsigned y, const grs_bitmap &bm) { int dest_rowsize; dest_rowsize=grd_curcanv->cv_bitmap.bm_rowsize << gr_bitblt_dest_step_shift; auto dest = &(grd_curcanv->cv_bitmap.get_bitmap_data()[ dest_rowsize*y+x ]); auto src = bm.get_bitmap_data(); for (int y1=0; y1 < bm.bm_h; y1++ ) { gr_linear_movsd( src, dest, bm.bm_w ); src += bm.bm_rowsize; dest+= (int)(dest_rowsize); } } static void gr_ubitmap00m(unsigned x, unsigned y, const grs_bitmap &bm) { int dest_rowsize; dest_rowsize=grd_curcanv->cv_bitmap.bm_rowsize << gr_bitblt_dest_step_shift; auto dest = &(grd_curcanv->cv_bitmap.get_bitmap_data()[ dest_rowsize*y+x ]); auto src = bm.get_bitmap_data(); if (gr_bitblt_fade_table==NULL) { for (int y1=0; y1 < bm.bm_h; y1++ ) { gr_linear_rep_movsdm( src, dest, bm.bm_w ); src += bm.bm_rowsize; dest+= (int)(dest_rowsize); } } else { for (int y1=0; y1 < bm.bm_h; y1++ ) { gr_linear_rep_movsdm_faded( src, dest, bm.bm_w, gr_bitblt_fade_table[y1+y] ); src += bm.bm_rowsize; dest+= (int)(dest_rowsize); } } } static void gr_ubitmap012(unsigned x, unsigned y, const grs_bitmap &bm) { auto src = bm.bm_data; for (int y1=y; y1 < (y+bm.bm_h); y1++ ) { for (int x1=x; x1 < (x+bm.bm_w); x1++ ) { gr_setcolor( *src++ ); gr_upixel( x1, y1 ); } } } static void gr_ubitmap012m(unsigned x, unsigned y, const grs_bitmap &bm) { auto src = bm.bm_data; for (int y1=y; y1 < (y+bm.bm_h); y1++ ) { for (int x1=x; x1 < (x+bm.bm_w); x1++ ) { if ( *src != 255 ) { gr_setcolor( *src ); gr_upixel( x1, y1 ); } src++; } } } static void gr_ubitmapGENERIC(unsigned x, unsigned y, const grs_bitmap &bm) { for (int y1=0; y1 < bm.bm_h; y1++ ) { for (int x1=0; x1 < bm.bm_w; x1++ ) { gr_setcolor( gr_gpixel(bm,x1,y1) ); gr_upixel( x+x1, y+y1 ); } } } static void gr_ubitmapGENERICm(unsigned x, unsigned y, const grs_bitmap &bm) { ubyte c; for (int y1=0; y1 < bm.bm_h; y1++ ) { for (int x1=0; x1 < bm.bm_w; x1++ ) { c = gr_gpixel(bm,x1,y1); if ( c != 255 ) { gr_setcolor( c ); gr_upixel( x+x1, y+y1 ); } } } } void gr_ubitmap(grs_bitmap &bm) { int source, dest; const unsigned x = 0; const unsigned y = 0; source = bm.bm_type; dest = TYPE; if (source==BM_LINEAR) { switch( dest ) { case BM_LINEAR: if ( bm.bm_flags & BM_FLAG_RLE ) gr_bm_ubitblt00_rle(bm.bm_w, bm.bm_h, x, y, 0, 0, bm, grd_curcanv->cv_bitmap ); else gr_ubitmap00( x, y, bm ); return; #ifdef OGL case BM_OGL: ogl_ubitmapm_cs(x,y,-1,-1,bm,-1,F1_0); return; #endif default: gr_ubitmap012( x, y, bm ); return; } } else { gr_ubitmapGENERIC(x, y, bm); } } void gr_ubitmapm(unsigned x, unsigned y, grs_bitmap &bm) { auto source = bm.bm_type; auto dest = TYPE; if (source==BM_LINEAR) { switch( dest ) { case BM_LINEAR: if ( bm.bm_flags & BM_FLAG_RLE ) gr_bm_ubitblt00m_rle(bm.bm_w, bm.bm_h, x, y, 0, 0, bm, grd_curcanv->cv_bitmap ); else gr_ubitmap00m(x, y, bm); return; #ifdef OGL case BM_OGL: ogl_ubitmapm_cs(x,y,-1,-1,bm,-1,F1_0); return; #endif default: gr_ubitmap012m( x, y, bm ); return; } } else { gr_ubitmapGENERICm(x, y, bm); } } // From Linear to Linear static void gr_bm_ubitblt00(unsigned w, unsigned h, unsigned dx, unsigned dy, unsigned sx, unsigned sy, const grs_bitmap &src, grs_bitmap &dest) { //int src_bm_rowsize_2, dest_bm_rowsize_2; auto sbits = &src.get_bitmap_data()[(src.bm_rowsize * sy) + sx]; auto dbits = &dest.get_bitmap_data()[(dest.bm_rowsize * dy) + dx]; auto dstep = dest.bm_rowsize << gr_bitblt_dest_step_shift; // No interlacing, copy the whole buffer. for (unsigned i=0; i < h; i++ ) { gr_linear_movsd( sbits, dbits, w ); //memcpy(dbits, sbits, w); sbits += src.bm_rowsize; dbits += dstep; } } // From Linear to Linear Masked static void gr_bm_ubitblt00m(unsigned w, unsigned h, unsigned dx, unsigned dy, unsigned sx, unsigned sy, const grs_bitmap &src, grs_bitmap &dest) { //int src_bm_rowsize_2, dest_bm_rowsize_2; auto sbits = &src.get_bitmap_data()[(src.bm_rowsize * sy) + sx]; auto dbits = &dest.get_bitmap_data()[(dest.bm_rowsize * dy) + dx]; // No interlacing, copy the whole buffer. if (gr_bitblt_fade_table==NULL) { for (int i=0; i < h; i++ ) { gr_linear_rep_movsdm( sbits, dbits, w ); sbits += src.bm_rowsize; dbits += dest.bm_rowsize; } } else { for (int i=0; i < h; i++ ) { gr_linear_rep_movsdm_faded( sbits, dbits, w, gr_bitblt_fade_table[dy+i] ); sbits += src.bm_rowsize; dbits += dest.bm_rowsize; } } } void gr_bm_ubitblt(unsigned w, unsigned h, int dx, int dy, int sx, int sy, const grs_bitmap &src, grs_bitmap &dest) { if ( (src.bm_type == BM_LINEAR) && (dest.bm_type == BM_LINEAR )) { if ( src.bm_flags & BM_FLAG_RLE ) gr_bm_ubitblt00_rle( w, h, dx, dy, sx, sy, src, dest ); else gr_bm_ubitblt00( w, h, dx, dy, sx, sy, src, dest ); return; } #ifdef OGL if ( (src.bm_type == BM_LINEAR) && (dest.bm_type == BM_OGL )) { ogl_ubitblt(w, h, dx, dy, sx, sy, src, dest); return; } if ( (src.bm_type == BM_OGL) && (dest.bm_type == BM_LINEAR )) { return; } if ( (src.bm_type == BM_OGL) && (dest.bm_type == BM_OGL )) { return; } #endif if ( (src.bm_flags & BM_FLAG_RLE ) && (src.bm_type == BM_LINEAR) ) { gr_bm_ubitblt0x_rle(w, h, dx, dy, sx, sy, src, dest); return; } for (int y1=0; y1 < h; y1++ ) { for (int x1=0; x1 < w; x1++ ) { gr_bm_pixel(dest, dx+x1, dy+y1, gr_gpixel(src,sx+x1,sy+y1) ); } } } // Clipped bitmap ... void gr_bitmap( int x, int y, grs_bitmap *bm ) { int dx1=x, dx2=x+bm->bm_w-1; int dy1=y, dy2=y+bm->bm_h-1; #ifndef OGL int sx=0, sy=0; #endif if ((dx1 >= grd_curcanv->cv_bitmap.bm_w ) || (dx2 < 0)) return; if ((dy1 >= grd_curcanv->cv_bitmap.bm_h) || (dy2 < 0)) return; // Draw bitmap bm[x,y] into (dx1,dy1)-(dx2,dy2) #ifdef OGL ogl_ubitmapm_cs(x, y, 0, 0, *bm, -1, F1_0); #else if ( dx1 < 0 ) { sx = -dx1; dx1 = 0; } if ( dy1 < 0 ) { sy = -dy1; dy1 = 0; } if ( dx2 >= grd_curcanv->cv_bitmap.bm_w ) { dx2 = grd_curcanv->cv_bitmap.bm_w-1; } if ( dy2 >= grd_curcanv->cv_bitmap.bm_h ) { dy2 = grd_curcanv->cv_bitmap.bm_h-1; } gr_bm_ubitblt(dx2-dx1+1,dy2-dy1+1, dx1, dy1, sx, sy, *bm, grd_curcanv->cv_bitmap ); #endif } void gr_bitmapm( int x, int y, grs_bitmap *bm ) { int dx1=x, dx2=x+bm->bm_w-1; int dy1=y, dy2=y+bm->bm_h-1; int sx=0, sy=0; if ((dx1 >= grd_curcanv->cv_bitmap.bm_w ) || (dx2 < 0)) return; if ((dy1 >= grd_curcanv->cv_bitmap.bm_h) || (dy2 < 0)) return; if ( dx1 < 0 ) { sx = -dx1; dx1 = 0; } if ( dy1 < 0 ) { sy = -dy1; dy1 = 0; } if ( dx2 >= grd_curcanv->cv_bitmap.bm_w ) { dx2 = grd_curcanv->cv_bitmap.bm_w-1; } if ( dy2 >= grd_curcanv->cv_bitmap.bm_h ) { dy2 = grd_curcanv->cv_bitmap.bm_h-1; } // Draw bitmap bm[x,y] into (dx1,dy1)-(dx2,dy2) if ( (bm->bm_type == BM_LINEAR) && (grd_curcanv->cv_bitmap.bm_type == BM_LINEAR )) { if ( bm->bm_flags & BM_FLAG_RLE ) gr_bm_ubitblt00m_rle(dx2-dx1+1,dy2-dy1+1, dx1, dy1, sx, sy, *bm, grd_curcanv->cv_bitmap ); else gr_bm_ubitblt00m(dx2-dx1+1,dy2-dy1+1, dx1, dy1, sx, sy, *bm, grd_curcanv->cv_bitmap ); return; } gr_bm_ubitbltm(dx2-dx1+1,dy2-dy1+1, dx1, dy1, sx, sy, *bm, grd_curcanv->cv_bitmap ); } void gr_bm_ubitbltm(unsigned w, unsigned h, unsigned dx, unsigned dy, unsigned sx, unsigned sy, const grs_bitmap &src, grs_bitmap &dest) { ubyte c; #ifdef OGL if ( (src.bm_type == BM_LINEAR) && (dest.bm_type == BM_OGL )) { ogl_ubitblt(w, h, dx, dy, sx, sy, src, dest); return; } if ( (src.bm_type == BM_OGL) && (dest.bm_type == BM_LINEAR )) { return; } if ( (src.bm_type == BM_OGL) && (dest.bm_type == BM_OGL )) { return; } #endif for (int y1=0; y1 < h; y1++ ) { for (int x1=0; x1 < w; x1++ ) { if ((c=gr_gpixel(src,sx+x1,sy+y1))!=255) gr_bm_pixel(dest, dx+x1, dy+y1,c ); } } } static void gr_bm_ubitblt00_rle(unsigned w, unsigned h, int dx, int dy, int sx, int sy, const grs_bitmap &src, grs_bitmap &dest) { int data_offset; data_offset = 1; if (src.bm_flags & BM_FLAG_RLE_BIG) data_offset = 2; auto sbits = &src.get_bitmap_data()[4 + (src.bm_h*data_offset)]; for (int i=0; i= ilen) { c -= ilen; goto inside; } while(--i>=0) { inside: *out++ = *in; } in++; } } static void gr_bitmap_scale_to(const grs_bitmap &src, grs_bitmap &dst) { auto s = src.get_bitmap_data(); auto d = dst.get_bitmap_data(); int h = src.bm_h; int a = dst.bm_h/h, b = dst.bm_h%h; int c = 0, i; for(int y=0; y= h) { c -= h; goto inside; } while(--i>=0) { inside: scale_line(s, d, src.bm_w, dst.bm_w); d += dst.bm_rowsize; } s += src.bm_rowsize; } } void show_fullscr(grs_bitmap *bm) { grs_bitmap * const scr = &grd_curcanv->cv_bitmap; #ifdef OGL if(bm->bm_type == BM_LINEAR && scr->bm_type == BM_OGL && bm->bm_w <= grd_curscreen->sc_w && bm->bm_h <= grd_curscreen->sc_h) // only scale with OGL if bitmap is not bigger than screen size { ogl_ubitmapm_cs(0,0,-1,-1,*bm,-1,F1_0);//use opengl to scale, faster and saves ram. -MPM return; } #endif if(scr->bm_type != BM_LINEAR) { grs_bitmap_ptr p = gr_create_bitmap(scr->bm_w, scr->bm_h); grs_bitmap *tmp = p.get(); gr_bitmap_scale_to(*bm, *tmp); gr_bitmap(0, 0, tmp); return; } gr_bitmap_scale_to(*bm, *scr); } // Find transparent area in bitmap void gr_bitblt_find_transparent_area(grs_bitmap *bm, unsigned &minx, unsigned &miny, unsigned &maxx, unsigned &maxy) { using std::advance; using std::min; using std::max; if (!(bm->bm_flags&BM_FLAG_TRANSPARENT)) return; minx = bm->bm_w - 1; maxx = 0; miny = bm->bm_h - 1; maxy = 0; unsigned i = 0, count = 0; auto check = [&](unsigned x, unsigned y, ubyte c) { if (c == TRANSPARENCY_COLOR) { // don't look for transparancy color here. count++; minx = min(x, minx); miny = min(y, miny); maxx = max(x, maxx); maxy = max(y, maxy); } }; // decode the bitmap if (bm->bm_flags & BM_FLAG_RLE){ unsigned data_offset; data_offset = 1; if (bm->bm_flags & BM_FLAG_RLE_BIG) data_offset = 2; auto sbits = &bm->bm_data[4 + (bm->bm_h * data_offset)]; for (unsigned y = 0; y < bm->bm_h; ++y) { array buf; gr_rle_decode({sbits, begin(buf)}, rle_end(*bm, buf)); advance(sbits, bm->bm_data[4+i] | (data_offset == 2 ? static_cast(bm->bm_data[5+i]) << 8 : 0)); i += data_offset; for (unsigned x = 0; x < bm->bm_w; ++x) check(x, y, buf[x]); } } else { for (unsigned y = 0; y < bm->bm_h; ++y) for (unsigned x = 0; x < bm->bm_w; ++x) check(x, y, bm->bm_data[i++]); } Assert (count); }