/* Copyright (C) 2002-2005 RealVNC Ltd. All Rights Reserved. * * This 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 2 of the License, or * (at your option) any later version. * * This software 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 software; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, * USA. */ // // transInitTempl.h - templates for functions to initialise lookup tables for // the translation functions. // // This file is #included after having set the following macros: // BPPOUT - 8, 16 or 32 #if !defined(BPPOUT) #error "transInitTempl.h: BPPOUT not defined" #endif namespace rfb { // CONCAT2E concatenates its arguments, expanding them if they are macros #ifndef CONCAT2E #define CONCAT2(a,b) a##b #define CONCAT2E(a,b) CONCAT2(a,b) #endif #ifndef SWAP16 #define SWAP16(n) ((((n) & 0xff) << 8) | (((n) >> 8) & 0xff)) #endif #ifndef SWAP32 #define SWAP32(n) (((n) >> 24) | (((n) & 0x00ff0000) >> 8) | \ (((n) & 0x0000ff00) << 8) | ((n) << 24)) #endif #define OUTPIXEL rdr::CONCAT2E(U,BPPOUT) #define SWAPOUT CONCAT2E(SWAP,BPPOUT) #define initSimpleCMtoTCOUT CONCAT2E(initSimpleCMtoTC,BPPOUT) #define initSimpleTCtoTCOUT CONCAT2E(initSimpleTCtoTC,BPPOUT) #define initSimpleCMtoCubeOUT CONCAT2E(initSimpleCMtoCube,BPPOUT) #define initSimpleTCtoCubeOUT CONCAT2E(initSimpleTCtoCube,BPPOUT) #define initRGBTCtoTCOUT CONCAT2E(initRGBTCtoTC,BPPOUT) #define initRGBTCtoCubeOUT CONCAT2E(initRGBTCtoCube,BPPOUT) #define initOneRGBTableOUT CONCAT2E(initOneRGBTable,BPPOUT) #define initOneRGBCubeTableOUT CONCAT2E(initOneRGBCubeTable,BPPOUT) #ifndef TRANS_INIT_TEMPL_ENDIAN_TEST #define TRANS_INIT_TEMPL_ENDIAN_TEST static rdr::U32 endianTest = 1; static bool nativeBigEndian = *(rdr::U8*)(&endianTest) != 1; #endif void initSimpleCMtoTCOUT (rdr::U8** tablep, const PixelFormat& inPF, ColourMap* cm, const PixelFormat& outPF) { if (inPF.bpp != 8 && inPF.bigEndian != nativeBigEndian) throw Exception("Internal error: inPF is not native endian"); int size = 1 << inPF.bpp; delete [] *tablep; *tablep = new rdr::U8[size * sizeof(OUTPIXEL)]; OUTPIXEL* table = (OUTPIXEL*)*tablep; for (int i = 0; i < size; i++) { int r,g,b; cm->lookup(i,&r,&g,&b); table[i] = ((((r * outPF.redMax + 32767) / 65535) << outPF.redShift) | (((g * outPF.greenMax + 32767) / 65535) << outPF.greenShift) | (((b * outPF.blueMax + 32767) / 65535) << outPF.blueShift)); #if (BPPOUT != 8) if (outPF.bigEndian != nativeBigEndian) table[i] = SWAPOUT (table[i]); #endif } } void initSimpleTCtoTCOUT (rdr::U8** tablep, const PixelFormat& inPF, const PixelFormat& outPF) { if (inPF.bpp != 8 && inPF.bigEndian != nativeBigEndian) throw Exception("Internal error: inPF is not native endian"); int size = 1 << inPF.bpp; delete [] *tablep; *tablep = new rdr::U8[size * sizeof(OUTPIXEL)]; OUTPIXEL* table = (OUTPIXEL*)*tablep; for (int i = 0; i < size; i++) { int r = (i >> inPF.redShift) & inPF.redMax; int g = (i >> inPF.greenShift) & inPF.greenMax; int b = (i >> inPF.blueShift) & inPF.blueMax; r = (r * outPF.redMax + inPF.redMax/2) / inPF.redMax; g = (g * outPF.greenMax + inPF.greenMax/2) / inPF.greenMax; b = (b * outPF.blueMax + inPF.blueMax/2) / inPF.blueMax; table[i] = ((r << outPF.redShift) | (g << outPF.greenShift) | (b << outPF.blueShift)); #if (BPPOUT != 8) if (outPF.bigEndian != nativeBigEndian) table[i] = SWAPOUT (table[i]); #endif } } void initSimpleCMtoCubeOUT (rdr::U8** tablep, const PixelFormat& inPF, ColourMap* cm, ColourCube* cube) { if (inPF.bpp != 8 && inPF.bigEndian != nativeBigEndian) throw Exception("Internal error: inPF is not native endian"); int size = 1 << inPF.bpp; delete [] *tablep; *tablep = new rdr::U8[size * sizeof(OUTPIXEL)]; OUTPIXEL* table = (OUTPIXEL*)*tablep; for (int i = 0; i < size; i++) { int r,g,b; cm->lookup(i,&r,&g,&b); r = (r * (cube->nRed-1) + 32767) / 65535; g = (g * (cube->nGreen-1) + 32767) / 65535; b = (b * (cube->nBlue-1) + 32767) / 65535; table[i] = cube->lookup(r, g, b); } } void initSimpleTCtoCubeOUT (rdr::U8** tablep, const PixelFormat& inPF, ColourCube* cube) { if (inPF.bpp != 8 && inPF.bigEndian != nativeBigEndian) throw Exception("Internal error: inPF is not native endian"); int size = 1 << inPF.bpp; delete [] *tablep; *tablep = new rdr::U8[size * sizeof(OUTPIXEL)]; OUTPIXEL* table = (OUTPIXEL*)*tablep; for (int i = 0; i < size; i++) { int r = (i >> inPF.redShift) & inPF.redMax; int g = (i >> inPF.greenShift) & inPF.greenMax; int b = (i >> inPF.blueShift) & inPF.blueMax; r = (r * (cube->nRed-1) + inPF.redMax/2) / inPF.redMax; g = (g * (cube->nGreen-1) + inPF.greenMax/2) / inPF.greenMax; b = (b * (cube->nBlue-1) + inPF.blueMax/2) / inPF.blueMax; table[i] = cube->lookup(r, g, b); } } void initOneRGBTableOUT (OUTPIXEL* table, int inMax, int outMax, int outShift, bool swap) { int size = inMax + 1; for (int i = 0; i < size; i++) { table[i] = ((i * outMax + inMax / 2) / inMax) << outShift; #if (BPPOUT != 8) if (swap) table[i] = SWAPOUT (table[i]); #endif } } void initRGBTCtoTCOUT (rdr::U8** tablep, const PixelFormat& inPF, const PixelFormat& outPF) { if (inPF.bpp != 8 && inPF.bigEndian != nativeBigEndian) throw Exception("Internal error: inPF is not native endian"); int size = inPF.redMax + inPF.greenMax + inPF.blueMax + 3; delete [] *tablep; *tablep = new rdr::U8[size * sizeof(OUTPIXEL)]; OUTPIXEL* redTable = (OUTPIXEL*)*tablep; OUTPIXEL* greenTable = redTable + inPF.redMax + 1; OUTPIXEL* blueTable = greenTable + inPF.greenMax + 1; bool swap = (outPF.bigEndian != nativeBigEndian); initOneRGBTableOUT (redTable, inPF.redMax, outPF.redMax, outPF.redShift, swap); initOneRGBTableOUT (greenTable, inPF.greenMax, outPF.greenMax, outPF.greenShift, swap); initOneRGBTableOUT (blueTable, inPF.blueMax, outPF.blueMax, outPF.blueShift, swap); } void initOneRGBCubeTableOUT (OUTPIXEL* table, int inMax, int outMax, int outMult) { int size = inMax + 1; for (int i = 0; i < size; i++) { table[i] = ((i * outMax + inMax / 2) / inMax) * outMult; } } void initRGBTCtoCubeOUT (rdr::U8** tablep, const PixelFormat& inPF, ColourCube* cube) { if (inPF.bpp != 8 && inPF.bigEndian != nativeBigEndian) throw Exception("Internal error: inPF is not native endian"); int size = inPF.redMax + inPF.greenMax + inPF.blueMax + 3 + cube->size(); delete [] *tablep; *tablep = new rdr::U8[size * sizeof(OUTPIXEL)]; OUTPIXEL* redTable = (OUTPIXEL*)*tablep; OUTPIXEL* greenTable = redTable + inPF.redMax + 1; OUTPIXEL* blueTable = greenTable + inPF.greenMax + 1; OUTPIXEL* cubeTable = blueTable + inPF.blueMax + 1; initOneRGBCubeTableOUT (redTable, inPF.redMax, cube->nRed-1, cube->redMult()); initOneRGBCubeTableOUT (greenTable, inPF.greenMax, cube->nGreen-1, cube->greenMult()); initOneRGBCubeTableOUT (blueTable, inPF.blueMax, cube->nBlue-1, cube->blueMult()); for (int i = 0; i < cube->size(); i++) { cubeTable[i] = cube->table[i]; } } #undef OUTPIXEL #undef initSimpleCMtoTCOUT #undef initSimpleTCtoTCOUT #undef initSimpleCMtoCubeOUT #undef initSimpleTCtoCubeOUT #undef initRGBTCtoTCOUT #undef initRGBTCtoCubeOUT #undef initOneRGBTableOUT #undef initOneRGBCubeTableOUT }