diff --git a/snesfilter/2xSaI/2xSaI.cpp b/snesfilter/2xSaI/2xSaI.cpp deleted file mode 100755 index d9dc61fa..00000000 --- a/snesfilter/2xSaI/2xSaI.cpp +++ /dev/null @@ -1,32 +0,0 @@ -#include -#include -using namespace nall; - -#include "implementation.cpp" - -extern "C" { - void filter_size(unsigned&, unsigned&); - void filter_render(uint32_t*, uint32_t*, unsigned, const uint16_t*, unsigned, unsigned, unsigned); -}; - -uint32_t temp[512 * 480]; - -dllexport void filter_size(unsigned &width, unsigned &height) { - width *= 2; - height *= 2; -} - -dllexport void filter_render( - uint32_t *colortable, uint32_t *output, unsigned outpitch, - const uint16_t *input, unsigned pitch, unsigned width, unsigned height -) { - for(unsigned y = 0; y < height; y++) { - const uint16_t *line_in = (const uint16_t*)(((const uint8_t*)input) + pitch * y); - uint32_t *line_out = temp + y * width; - for(unsigned x = 0; x < width; x++) { - line_out[x] = colortable[line_in[x]]; - } - } - - _2xSaI32((unsigned char*)temp, width * sizeof(uint32_t), 0, (unsigned char*)output, outpitch, width, height); -} diff --git a/snesfilter/2xSaI/Super-2xSaI.cpp b/snesfilter/2xSaI/Super-2xSaI.cpp deleted file mode 100755 index 267e11e5..00000000 --- a/snesfilter/2xSaI/Super-2xSaI.cpp +++ /dev/null @@ -1,32 +0,0 @@ -#include -#include -using namespace nall; - -#include "implementation.cpp" - -extern "C" { - void filter_size(unsigned&, unsigned&); - void filter_render(uint32_t*, uint32_t*, unsigned, const uint16_t*, unsigned, unsigned, unsigned); -}; - -uint32_t temp[512 * 480]; - -dllexport void filter_size(unsigned &width, unsigned &height) { - width *= 2; - height *= 2; -} - -dllexport void filter_render( - uint32_t *colortable, uint32_t *output, unsigned outpitch, - const uint16_t *input, unsigned pitch, unsigned width, unsigned height -) { - for(unsigned y = 0; y < height; y++) { - const uint16_t *line_in = (const uint16_t*)(((const uint8_t*)input) + pitch * y); - uint32_t *line_out = temp + y * width; - for(unsigned x = 0; x < width; x++) { - line_out[x] = colortable[line_in[x]]; - } - } - - Super2xSaI32((unsigned char*)temp, width * sizeof(uint32_t), 0, (unsigned char*)output, outpitch, width, height); -} diff --git a/snesfilter/2xSaI/Super-Eagle.cpp b/snesfilter/2xSaI/Super-Eagle.cpp deleted file mode 100755 index c548348c..00000000 --- a/snesfilter/2xSaI/Super-Eagle.cpp +++ /dev/null @@ -1,32 +0,0 @@ -#include -#include -using namespace nall; - -#include "implementation.cpp" - -extern "C" { - void filter_size(unsigned&, unsigned&); - void filter_render(uint32_t*, uint32_t*, unsigned, const uint16_t*, unsigned, unsigned, unsigned); -}; - -uint32_t temp[512 * 480]; - -dllexport void filter_size(unsigned &width, unsigned &height) { - width *= 2; - height *= 2; -} - -dllexport void filter_render( - uint32_t *colortable, uint32_t *output, unsigned outpitch, - const uint16_t *input, unsigned pitch, unsigned width, unsigned height -) { - for(unsigned y = 0; y < height; y++) { - const uint16_t *line_in = (const uint16_t*)(((const uint8_t*)input) + pitch * y); - uint32_t *line_out = temp + y * width; - for(unsigned x = 0; x < width; x++) { - line_out[x] = colortable[line_in[x]]; - } - } - - SuperEagle32((unsigned char*)temp, width * sizeof(uint32_t), 0, (unsigned char*)output, outpitch, width, height); -} diff --git a/snesfilter/2xSaI/implementation.cpp b/snesfilter/2xSaI/implementation.cpp deleted file mode 100755 index cebca7ef..00000000 --- a/snesfilter/2xSaI/implementation.cpp +++ /dev/null @@ -1,1171 +0,0 @@ -static uint32_t colorMask = 0xFEFEFE; -static uint32_t lowPixelMask = 0x010101; -static uint32_t qcolorMask = 0xFCFCFC; -static uint32_t qlowpixelMask = 0x030303; -static uint32_t redblueMask = 0xFF00FF; -static uint32_t greenMask = 0xFF00; - -uint32_t qRGB_COLOR_MASK[2] = { 0xFEFEFE, 0xFEFEFE }; - -static inline int GetResult1 (uint32_t A, uint32_t B, uint32_t C, uint32_t D, - uint32_t /* E */) -{ - int x = 0; - int y = 0; - int r = 0; - - if (A == C) - x += 1; - else if (B == C) - y += 1; - if (A == D) - x += 1; - else if (B == D) - y += 1; - if (x <= 1) - r += 1; - if (y <= 1) - r -= 1; - return r; -} - -static inline int GetResult2 (uint32_t A, uint32_t B, uint32_t C, uint32_t D, - uint32_t /* E */) -{ - int x = 0; - int y = 0; - int r = 0; - - if (A == C) - x += 1; - else if (B == C) - y += 1; - if (A == D) - x += 1; - else if (B == D) - y += 1; - if (x <= 1) - r -= 1; - if (y <= 1) - r += 1; - return r; -} - -static inline int GetResult (uint32_t A, uint32_t B, uint32_t C, uint32_t D) -{ - int x = 0; - int y = 0; - int r = 0; - - if (A == C) - x += 1; - else if (B == C) - y += 1; - if (A == D) - x += 1; - else if (B == D) - y += 1; - if (x <= 1) - r += 1; - if (y <= 1) - r -= 1; - return r; -} - -static inline uint32_t INTERPOLATE (uint32_t A, uint32_t B) -{ - if (A != B) { - return (((A & colorMask) >> 1) + ((B & colorMask) >> 1) + - (A & B & lowPixelMask)); - } else - return A; -} - -static inline uint32_t Q_INTERPOLATE (uint32_t A, uint32_t B, uint32_t C, uint32_t D) -{ - register uint32_t x = ((A & qcolorMask) >> 2) + - ((B & qcolorMask) >> 2) + - ((C & qcolorMask) >> 2) + ((D & qcolorMask) >> 2); - register uint32_t y = (A & qlowpixelMask) + - (B & qlowpixelMask) + (C & qlowpixelMask) + (D & qlowpixelMask); - - y = (y >> 2) & qlowpixelMask; - return x + y; -} - -static inline int GetResult1_32 (uint32_t A, uint32_t B, uint32_t C, uint32_t D, - uint32_t /* E */) -{ - int x = 0; - int y = 0; - int r = 0; - - if (A == C) - x += 1; - else if (B == C) - y += 1; - if (A == D) - x += 1; - else if (B == D) - y += 1; - if (x <= 1) - r += 1; - if (y <= 1) - r -= 1; - return r; -} - -static inline int GetResult2_32 (uint32_t A, uint32_t B, uint32_t C, uint32_t D, - uint32_t /* E */) -{ - int x = 0; - int y = 0; - int r = 0; - - if (A == C) - x += 1; - else if (B == C) - y += 1; - if (A == D) - x += 1; - else if (B == D) - y += 1; - if (x <= 1) - r -= 1; - if (y <= 1) - r += 1; - return r; -} - -#define BLUE_MASK565 0x001F001F -#define RED_MASK565 0xF800F800 -#define GREEN_MASK565 0x07E007E0 - -#define BLUE_MASK555 0x001F001F -#define RED_MASK555 0x7C007C00 -#define GREEN_MASK555 0x03E003E0 - -void Super2xSaI (uint8_t *srcPtr, uint32_t srcPitch, - uint8_t *deltaPtr, uint8_t *dstPtr, uint32_t dstPitch, - int width, int height) -{ - uint16_t *bP; - uint8_t *dP; - uint32_t inc_bP; - uint32_t Nextline = srcPitch >> 1; - { - inc_bP = 1; - - for (; height; height--) { - bP = (uint16_t *) srcPtr; - dP = (uint8_t *) dstPtr; - - for (uint32_t finish = width; finish; finish -= inc_bP) { - uint32_t color4, color5, color6; - uint32_t color1, color2, color3; - uint32_t colorA0, colorA1, colorA2, colorA3, - colorB0, colorB1, colorB2, colorB3, colorS1, colorS2; - uint32_t product1a, product1b, product2a, product2b; - - //--------------------------------------- B1 B2 - // 4 5 6 S2 - // 1 2 3 S1 - // A1 A2 - - colorB0 = *(bP - Nextline - 1); - colorB1 = *(bP - Nextline); - colorB2 = *(bP - Nextline + 1); - colorB3 = *(bP - Nextline + 2); - - color4 = *(bP - 1); - color5 = *(bP); - color6 = *(bP + 1); - colorS2 = *(bP + 2); - - color1 = *(bP + Nextline - 1); - color2 = *(bP + Nextline); - color3 = *(bP + Nextline + 1); - colorS1 = *(bP + Nextline + 2); - - colorA0 = *(bP + Nextline + Nextline - 1); - colorA1 = *(bP + Nextline + Nextline); - colorA2 = *(bP + Nextline + Nextline + 1); - colorA3 = *(bP + Nextline + Nextline + 2); - - //-------------------------------------- - if (color2 == color6 && color5 != color3) { - product2b = product1b = color2; - } else if (color5 == color3 && color2 != color6) { - product2b = product1b = color5; - } else if (color5 == color3 && color2 == color6) { - register int r = 0; - - r += GetResult (color6, color5, color1, colorA1); - r += GetResult (color6, color5, color4, colorB1); - r += GetResult (color6, color5, colorA2, colorS1); - r += GetResult (color6, color5, colorB2, colorS2); - - if (r > 0) - product2b = product1b = color6; - else if (r < 0) - product2b = product1b = color5; - else { - product2b = product1b = INTERPOLATE (color5, color6); - } - } else { - if (color6 == color3 && color3 == colorA1 - && color2 != colorA2 && color3 != colorA0) - product2b = - Q_INTERPOLATE (color3, color3, color3, color2); - else if (color5 == color2 && color2 == colorA2 - && colorA1 != color3 && color2 != colorA3) - product2b = - Q_INTERPOLATE (color2, color2, color2, color3); - else - product2b = INTERPOLATE (color2, color3); - - if (color6 == color3 && color6 == colorB1 - && color5 != colorB2 && color6 != colorB0) - product1b = - Q_INTERPOLATE (color6, color6, color6, color5); - else if (color5 == color2 && color5 == colorB2 - && colorB1 != color6 && color5 != colorB3) - product1b = - Q_INTERPOLATE (color6, color5, color5, color5); - else - product1b = INTERPOLATE (color5, color6); - } - - if (color5 == color3 && color2 != color6 && color4 == color5 - && color5 != colorA2) - product2a = INTERPOLATE (color2, color5); - else - if (color5 == color1 && color6 == color5 - && color4 != color2 && color5 != colorA0) - product2a = INTERPOLATE (color2, color5); - else - product2a = color2; - - if (color2 == color6 && color5 != color3 && color1 == color2 - && color2 != colorB2) - product1a = INTERPOLATE (color2, color5); - else - if (color4 == color2 && color3 == color2 - && color1 != color5 && color2 != colorB0) - product1a = INTERPOLATE (color2, color5); - else - product1a = color5; - -#ifdef WORDS_BIGENDIAN - product1a = (product1a << 16) | product1b; - product2a = (product2a << 16) | product2b; -#else - product1a = product1a | (product1b << 16); - product2a = product2a | (product2b << 16); -#endif - - *((uint32_t *) dP) = product1a; - *((uint32_t *) (dP + dstPitch)) = product2a; - - bP += inc_bP; - dP += sizeof (uint32_t); - } // end of for ( finish= width etc..) - - srcPtr += srcPitch; - dstPtr += dstPitch << 1; - deltaPtr += srcPitch; - } // endof: for (; height; height--) - } -} - -void Super2xSaI32 (uint8_t *srcPtr, uint32_t srcPitch, - uint8_t * /* deltaPtr */, uint8_t *dstPtr, uint32_t dstPitch, - int width, int height) -{ - uint32_t *bP; - uint32_t *dP; - uint32_t inc_bP; - uint32_t Nextline = srcPitch >> 2; - inc_bP = 1; - - for (; height; height--) { - bP = (uint32_t *) srcPtr; - dP = (uint32_t *) dstPtr; - - for (uint32_t finish = width; finish; finish -= inc_bP) { - uint32_t color4, color5, color6; - uint32_t color1, color2, color3; - uint32_t colorA0, colorA1, colorA2, colorA3, - colorB0, colorB1, colorB2, colorB3, colorS1, colorS2; - uint32_t product1a, product1b, product2a, product2b; - - //--------------------------------------- B1 B2 - // 4 5 6 S2 - // 1 2 3 S1 - // A1 A2 - - colorB0 = *(bP - Nextline - 1); - colorB1 = *(bP - Nextline); - colorB2 = *(bP - Nextline + 1); - colorB3 = *(bP - Nextline + 2); - - color4 = *(bP - 1); - color5 = *(bP); - color6 = *(bP + 1); - colorS2 = *(bP + 2); - - color1 = *(bP + Nextline - 1); - color2 = *(bP + Nextline); - color3 = *(bP + Nextline + 1); - colorS1 = *(bP + Nextline + 2); - - colorA0 = *(bP + Nextline + Nextline - 1); - colorA1 = *(bP + Nextline + Nextline); - colorA2 = *(bP + Nextline + Nextline + 1); - colorA3 = *(bP + Nextline + Nextline + 2); - - //-------------------------------------- - if (color2 == color6 && color5 != color3) { - product2b = product1b = color2; - } else if (color5 == color3 && color2 != color6) { - product2b = product1b = color5; - } else if (color5 == color3 && color2 == color6) { - register int r = 0; - - r += GetResult (color6, color5, color1, colorA1); - r += GetResult (color6, color5, color4, colorB1); - r += GetResult (color6, color5, colorA2, colorS1); - r += GetResult (color6, color5, colorB2, colorS2); - - if (r > 0) - product2b = product1b = color6; - else if (r < 0) - product2b = product1b = color5; - else { - product2b = product1b = INTERPOLATE (color5, color6); - } - } else { - if (color6 == color3 && color3 == colorA1 - && color2 != colorA2 && color3 != colorA0) - product2b = - Q_INTERPOLATE (color3, color3, color3, color2); - else if (color5 == color2 && color2 == colorA2 - && colorA1 != color3 && color2 != colorA3) - product2b = - Q_INTERPOLATE (color2, color2, color2, color3); - else - product2b = INTERPOLATE (color2, color3); - - if (color6 == color3 && color6 == colorB1 - && color5 != colorB2 && color6 != colorB0) - product1b = - Q_INTERPOLATE (color6, color6, color6, color5); - else if (color5 == color2 && color5 == colorB2 - && colorB1 != color6 && color5 != colorB3) - product1b = - Q_INTERPOLATE (color6, color5, color5, color5); - else - product1b = INTERPOLATE (color5, color6); - } - - if (color5 == color3 && color2 != color6 && color4 == color5 - && color5 != colorA2) - product2a = INTERPOLATE (color2, color5); - else - if (color5 == color1 && color6 == color5 - && color4 != color2 && color5 != colorA0) - product2a = INTERPOLATE (color2, color5); - else - product2a = color2; - - if (color2 == color6 && color5 != color3 && color1 == color2 - && color2 != colorB2) - product1a = INTERPOLATE (color2, color5); - else - if (color4 == color2 && color3 == color2 - && color1 != color5 && color2 != colorB0) - product1a = INTERPOLATE (color2, color5); - else - product1a = color5; - *(dP) = product1a; - *(dP+1) = product1b; - *(dP + (dstPitch >> 2)) = product2a; - *(dP + (dstPitch >> 2) + 1) = product2b; - - bP += inc_bP; - dP += 2; - } // end of for ( finish= width etc..) - - srcPtr += srcPitch; - dstPtr += dstPitch << 1; - // deltaPtr += srcPitch; - } // endof: for (; height; height--) -} - -void SuperEagle (uint8_t *srcPtr, uint32_t srcPitch, uint8_t *deltaPtr, - uint8_t *dstPtr, uint32_t dstPitch, int width, int height) -{ - uint8_t *dP; - uint16_t *bP; - uint16_t *xP; - uint32_t inc_bP; - - { - inc_bP = 1; - - uint32_t Nextline = srcPitch >> 1; - - for (; height; height--) { - bP = (uint16_t *) srcPtr; - xP = (uint16_t *) deltaPtr; - dP = dstPtr; - for (uint32_t finish = width; finish; finish -= inc_bP) { - uint32_t color4, color5, color6; - uint32_t color1, color2, color3; - uint32_t colorA1, colorA2, colorB1, colorB2, colorS1, colorS2; - uint32_t product1a, product1b, product2a, product2b; - - colorB1 = *(bP - Nextline); - colorB2 = *(bP - Nextline + 1); - - color4 = *(bP - 1); - color5 = *(bP); - color6 = *(bP + 1); - colorS2 = *(bP + 2); - - color1 = *(bP + Nextline - 1); - color2 = *(bP + Nextline); - color3 = *(bP + Nextline + 1); - colorS1 = *(bP + Nextline + 2); - - colorA1 = *(bP + Nextline + Nextline); - colorA2 = *(bP + Nextline + Nextline + 1); - - // -------------------------------------- - if (color2 == color6 && color5 != color3) { - product1b = product2a = color2; - if ((color1 == color2) || (color6 == colorB2)) { - product1a = INTERPOLATE (color2, color5); - product1a = INTERPOLATE (color2, product1a); - // product1a = color2; - } else { - product1a = INTERPOLATE (color5, color6); - } - - if ((color6 == colorS2) || (color2 == colorA1)) { - product2b = INTERPOLATE (color2, color3); - product2b = INTERPOLATE (color2, product2b); - // product2b = color2; - } else { - product2b = INTERPOLATE (color2, color3); - } - } else if (color5 == color3 && color2 != color6) { - product2b = product1a = color5; - - if ((colorB1 == color5) || (color3 == colorS1)) { - product1b = INTERPOLATE (color5, color6); - product1b = INTERPOLATE (color5, product1b); - // product1b = color5; - } else { - product1b = INTERPOLATE (color5, color6); - } - - if ((color3 == colorA2) || (color4 == color5)) { - product2a = INTERPOLATE (color5, color2); - product2a = INTERPOLATE (color5, product2a); - // product2a = color5; - } else { - product2a = INTERPOLATE (color2, color3); - } - - } else if (color5 == color3 && color2 == color6) { - register int r = 0; - - r += GetResult (color6, color5, color1, colorA1); - r += GetResult (color6, color5, color4, colorB1); - r += GetResult (color6, color5, colorA2, colorS1); - r += GetResult (color6, color5, colorB2, colorS2); - - if (r > 0) { - product1b = product2a = color2; - product1a = product2b = INTERPOLATE (color5, color6); - } else if (r < 0) { - product2b = product1a = color5; - product1b = product2a = INTERPOLATE (color5, color6); - } else { - product2b = product1a = color5; - product1b = product2a = color2; - } - } else { - product2b = product1a = INTERPOLATE (color2, color6); - product2b = - Q_INTERPOLATE (color3, color3, color3, product2b); - product1a = - Q_INTERPOLATE (color5, color5, color5, product1a); - - product2a = product1b = INTERPOLATE (color5, color3); - product2a = - Q_INTERPOLATE (color2, color2, color2, product2a); - product1b = - Q_INTERPOLATE (color6, color6, color6, product1b); - - // product1a = color5; - // product1b = color6; - // product2a = color2; - // product2b = color3; - } -#ifdef WORDS_BIGENDIAN - product1a = (product1a << 16) | product1b; - product2a = (product2a << 16) | product2b; -#else - product1a = product1a | (product1b << 16); - product2a = product2a | (product2b << 16); -#endif - - *((uint32_t *) dP) = product1a; - *((uint32_t *) (dP + dstPitch)) = product2a; - *xP = color5; - - bP += inc_bP; - xP += inc_bP; - dP += sizeof (uint32_t); - } // end of for ( finish= width etc..) - - srcPtr += srcPitch; - dstPtr += dstPitch << 1; - deltaPtr += srcPitch; - } // endof: for (height; height; height--) - } -} - -void SuperEagle32 (uint8_t *srcPtr, uint32_t srcPitch, uint8_t */*deltaPtr*/, - uint8_t *dstPtr, uint32_t dstPitch, int width, int height) -{ - uint32_t *dP; - uint32_t *bP; - //uint32_t *xP; - uint32_t inc_bP; - - inc_bP = 1; - - uint32_t Nextline = srcPitch >> 2; - - for (; height; height--) { - bP = (uint32_t *) srcPtr; - //xP = (uint32_t *) deltaPtr; - dP = (uint32_t *)dstPtr; - for (uint32_t finish = width; finish; finish -= inc_bP) { - uint32_t color4, color5, color6; - uint32_t color1, color2, color3; - uint32_t colorA1, colorA2, colorB1, colorB2, colorS1, colorS2; - uint32_t product1a, product1b, product2a, product2b; - - colorB1 = *(bP - Nextline); - colorB2 = *(bP - Nextline + 1); - - color4 = *(bP - 1); - color5 = *(bP); - color6 = *(bP + 1); - colorS2 = *(bP + 2); - - color1 = *(bP + Nextline - 1); - color2 = *(bP + Nextline); - color3 = *(bP + Nextline + 1); - colorS1 = *(bP + Nextline + 2); - - colorA1 = *(bP + Nextline + Nextline); - colorA2 = *(bP + Nextline + Nextline + 1); - - // -------------------------------------- - if (color2 == color6 && color5 != color3) { - product1b = product2a = color2; - if ((color1 == color2) || (color6 == colorB2)) { - product1a = INTERPOLATE (color2, color5); - product1a = INTERPOLATE (color2, product1a); - // product1a = color2; - } else { - product1a = INTERPOLATE (color5, color6); - } - - if ((color6 == colorS2) || (color2 == colorA1)) { - product2b = INTERPOLATE (color2, color3); - product2b = INTERPOLATE (color2, product2b); - // product2b = color2; - } else { - product2b = INTERPOLATE (color2, color3); - } - } else if (color5 == color3 && color2 != color6) { - product2b = product1a = color5; - - if ((colorB1 == color5) || (color3 == colorS1)) { - product1b = INTERPOLATE (color5, color6); - product1b = INTERPOLATE (color5, product1b); - // product1b = color5; - } else { - product1b = INTERPOLATE (color5, color6); - } - - if ((color3 == colorA2) || (color4 == color5)) { - product2a = INTERPOLATE (color5, color2); - product2a = INTERPOLATE (color5, product2a); - // product2a = color5; - } else { - product2a = INTERPOLATE (color2, color3); - } - - } else if (color5 == color3 && color2 == color6) { - register int r = 0; - - r += GetResult (color6, color5, color1, colorA1); - r += GetResult (color6, color5, color4, colorB1); - r += GetResult (color6, color5, colorA2, colorS1); - r += GetResult (color6, color5, colorB2, colorS2); - - if (r > 0) { - product1b = product2a = color2; - product1a = product2b = INTERPOLATE (color5, color6); - } else if (r < 0) { - product2b = product1a = color5; - product1b = product2a = INTERPOLATE (color5, color6); - } else { - product2b = product1a = color5; - product1b = product2a = color2; - } - } else { - product2b = product1a = INTERPOLATE (color2, color6); - product2b = - Q_INTERPOLATE (color3, color3, color3, product2b); - product1a = - Q_INTERPOLATE (color5, color5, color5, product1a); - - product2a = product1b = INTERPOLATE (color5, color3); - product2a = - Q_INTERPOLATE (color2, color2, color2, product2a); - product1b = - Q_INTERPOLATE (color6, color6, color6, product1b); - - // product1a = color5; - // product1b = color6; - // product2a = color2; - // product2b = color3; - } - *(dP) = product1a; - *(dP+1) = product1b; - *(dP + (dstPitch >> 2)) = product2a; - *(dP + (dstPitch >> 2) +1) = product2b; - //*xP = color5; - - bP += inc_bP; - //xP += inc_bP; - dP += 2; - } // end of for ( finish= width etc..) - - srcPtr += srcPitch; - dstPtr += dstPitch << 1; - //deltaPtr += srcPitch; - } // endof: for (height; height; height--) -} - -void _2xSaI (uint8_t *srcPtr, uint32_t srcPitch, uint8_t *deltaPtr, - uint8_t *dstPtr, uint32_t dstPitch, int width, int height) -{ - uint8_t *dP; - uint16_t *bP; - uint32_t inc_bP; - - { - inc_bP = 1; - - uint32_t Nextline = srcPitch >> 1; - - for (; height; height--) { - bP = (uint16_t *) srcPtr; - dP = dstPtr; - - for (uint32_t finish = width; finish; finish -= inc_bP) { - - register uint32_t colorA, colorB; - uint32_t colorC, colorD, - colorE, colorF, colorG, colorH, - colorI, colorJ, colorK, colorL, - - colorM, colorN, colorO, colorP; - uint32_t product, product1, product2; - - //--------------------------------------- - // Map of the pixels: I|E F|J - // G|A B|K - // H|C D|L - // M|N O|P - colorI = *(bP - Nextline - 1); - colorE = *(bP - Nextline); - colorF = *(bP - Nextline + 1); - colorJ = *(bP - Nextline + 2); - - colorG = *(bP - 1); - colorA = *(bP); - colorB = *(bP + 1); - colorK = *(bP + 2); - - colorH = *(bP + Nextline - 1); - colorC = *(bP + Nextline); - colorD = *(bP + Nextline + 1); - colorL = *(bP + Nextline + 2); - - colorM = *(bP + Nextline + Nextline - 1); - colorN = *(bP + Nextline + Nextline); - colorO = *(bP + Nextline + Nextline + 1); - colorP = *(bP + Nextline + Nextline + 2); - - if ((colorA == colorD) && (colorB != colorC)) { - if (((colorA == colorE) && (colorB == colorL)) || - ((colorA == colorC) && (colorA == colorF) - && (colorB != colorE) && (colorB == colorJ))) { - product = colorA; - } else { - product = INTERPOLATE (colorA, colorB); - } - - if (((colorA == colorG) && (colorC == colorO)) || - ((colorA == colorB) && (colorA == colorH) - && (colorG != colorC) && (colorC == colorM))) { - product1 = colorA; - } else { - product1 = INTERPOLATE (colorA, colorC); - } - product2 = colorA; - } else if ((colorB == colorC) && (colorA != colorD)) { - if (((colorB == colorF) && (colorA == colorH)) || - ((colorB == colorE) && (colorB == colorD) - && (colorA != colorF) && (colorA == colorI))) { - product = colorB; - } else { - product = INTERPOLATE (colorA, colorB); - } - - if (((colorC == colorH) && (colorA == colorF)) || - ((colorC == colorG) && (colorC == colorD) - && (colorA != colorH) && (colorA == colorI))) { - product1 = colorC; - } else { - product1 = INTERPOLATE (colorA, colorC); - } - product2 = colorB; - } else if ((colorA == colorD) && (colorB == colorC)) { - if (colorA == colorB) { - product = colorA; - product1 = colorA; - product2 = colorA; - } else { - register int r = 0; - - product1 = INTERPOLATE (colorA, colorC); - product = INTERPOLATE (colorA, colorB); - - r += - GetResult1 (colorA, colorB, colorG, colorE, - colorI); - r += - GetResult2 (colorB, colorA, colorK, colorF, - colorJ); - r += - GetResult2 (colorB, colorA, colorH, colorN, - colorM); - r += - GetResult1 (colorA, colorB, colorL, colorO, - colorP); - - if (r > 0) - product2 = colorA; - else if (r < 0) - product2 = colorB; - else { - product2 = - Q_INTERPOLATE (colorA, colorB, colorC, - colorD); - } - } - } else { - product2 = Q_INTERPOLATE (colorA, colorB, colorC, colorD); - - if ((colorA == colorC) && (colorA == colorF) - && (colorB != colorE) && (colorB == colorJ)) { - product = colorA; - } else if ((colorB == colorE) && (colorB == colorD) - && (colorA != colorF) && (colorA == colorI)) { - product = colorB; - } else { - product = INTERPOLATE (colorA, colorB); - } - - if ((colorA == colorB) && (colorA == colorH) - && (colorG != colorC) && (colorC == colorM)) { - product1 = colorA; - } else if ((colorC == colorG) && (colorC == colorD) - && (colorA != colorH) && (colorA == colorI)) { - product1 = colorC; - } else { - product1 = INTERPOLATE (colorA, colorC); - } - } - -#ifdef WORDS_BIGENDIAN - product = (colorA << 16) | product ; - product1 = (product1 << 16) | product2 ; -#else - product = colorA | (product << 16); - product1 = product1 | (product2 << 16); -#endif - *((int32_t *) dP) = product; - *((uint32_t *) (dP + dstPitch)) = product1; - - bP += inc_bP; - dP += sizeof (uint32_t); - } // end of for ( finish= width etc..) - - srcPtr += srcPitch; - dstPtr += dstPitch << 1; - deltaPtr += srcPitch; - } // endof: for (height; height; height--) - } -} - -void _2xSaI32 (uint8_t *srcPtr, uint32_t srcPitch, uint8_t * /* deltaPtr */, - uint8_t *dstPtr, uint32_t dstPitch, int width, int height) -{ - uint32_t *dP; - uint32_t *bP; - uint32_t inc_bP = 1; - - uint32_t Nextline = srcPitch >> 2; - - for (; height; height--) { - bP = (uint32_t *) srcPtr; - dP = (uint32_t *) dstPtr; - - for (uint32_t finish = width; finish; finish -= inc_bP) { - register uint32_t colorA, colorB; - uint32_t colorC, colorD, - colorE, colorF, colorG, colorH, - colorI, colorJ, colorK, colorL, - - colorM, colorN, colorO, colorP; - uint32_t product, product1, product2; - - //--------------------------------------- - // Map of the pixels: I|E F|J - // G|A B|K - // H|C D|L - // M|N O|P - colorI = *(bP - Nextline - 1); - colorE = *(bP - Nextline); - colorF = *(bP - Nextline + 1); - colorJ = *(bP - Nextline + 2); - - colorG = *(bP - 1); - colorA = *(bP); - colorB = *(bP + 1); - colorK = *(bP + 2); - - colorH = *(bP + Nextline - 1); - colorC = *(bP + Nextline); - colorD = *(bP + Nextline + 1); - colorL = *(bP + Nextline + 2); - - colorM = *(bP + Nextline + Nextline - 1); - colorN = *(bP + Nextline + Nextline); - colorO = *(bP + Nextline + Nextline + 1); - colorP = *(bP + Nextline + Nextline + 2); - - if ((colorA == colorD) && (colorB != colorC)) { - if (((colorA == colorE) && (colorB == colorL)) || - ((colorA == colorC) && (colorA == colorF) - && (colorB != colorE) && (colorB == colorJ))) { - product = colorA; - } else { - product = INTERPOLATE (colorA, colorB); - } - - if (((colorA == colorG) && (colorC == colorO)) || - ((colorA == colorB) && (colorA == colorH) - && (colorG != colorC) && (colorC == colorM))) { - product1 = colorA; - } else { - product1 = INTERPOLATE (colorA, colorC); - } - product2 = colorA; - } else if ((colorB == colorC) && (colorA != colorD)) { - if (((colorB == colorF) && (colorA == colorH)) || - ((colorB == colorE) && (colorB == colorD) - && (colorA != colorF) && (colorA == colorI))) { - product = colorB; - } else { - product = INTERPOLATE (colorA, colorB); - } - - if (((colorC == colorH) && (colorA == colorF)) || - ((colorC == colorG) && (colorC == colorD) - && (colorA != colorH) && (colorA == colorI))) { - product1 = colorC; - } else { - product1 = INTERPOLATE (colorA, colorC); - } - product2 = colorB; - } else if ((colorA == colorD) && (colorB == colorC)) { - if (colorA == colorB) { - product = colorA; - product1 = colorA; - product2 = colorA; - } else { - register int r = 0; - - product1 = INTERPOLATE (colorA, colorC); - product = INTERPOLATE (colorA, colorB); - - r += - GetResult1 (colorA, colorB, colorG, colorE, - colorI); - r += - GetResult2 (colorB, colorA, colorK, colorF, - colorJ); - r += - GetResult2 (colorB, colorA, colorH, colorN, - colorM); - r += - GetResult1 (colorA, colorB, colorL, colorO, - colorP); - - if (r > 0) - product2 = colorA; - else if (r < 0) - product2 = colorB; - else { - product2 = - Q_INTERPOLATE (colorA, colorB, colorC, - colorD); - } - } - } else { - product2 = Q_INTERPOLATE (colorA, colorB, colorC, colorD); - - if ((colorA == colorC) && (colorA == colorF) - && (colorB != colorE) && (colorB == colorJ)) { - product = colorA; - } else if ((colorB == colorE) && (colorB == colorD) - && (colorA != colorF) && (colorA == colorI)) { - product = colorB; - } else { - product = INTERPOLATE (colorA, colorB); - } - - if ((colorA == colorB) && (colorA == colorH) - && (colorG != colorC) && (colorC == colorM)) { - product1 = colorA; - } else if ((colorC == colorG) && (colorC == colorD) - && (colorA != colorH) && (colorA == colorI)) { - product1 = colorC; - } else { - product1 = INTERPOLATE (colorA, colorC); - } - } - *(dP) = colorA; - *(dP + 1) = product; - *(dP + (dstPitch >> 2)) = product1; - *(dP + (dstPitch >> 2) + 1) = product2; - - bP += inc_bP; - dP += 2; - } // end of for ( finish= width etc..) - - srcPtr += srcPitch; - dstPtr += dstPitch << 1; - // deltaPtr += srcPitch; - } // endof: for (height; height; height--) -} - -static uint32_t Bilinear (uint32_t A, uint32_t B, uint32_t x) -{ - unsigned long areaA, areaB; - unsigned long result; - - if (A == B) - return A; - - areaB = (x >> 11) & 0x1f; // reduce 16 bit fraction to 5 bits - areaA = 0x20 - areaB; - - A = (A & redblueMask) | ((A & greenMask) << 16); - B = (B & redblueMask) | ((B & greenMask) << 16); - - result = ((areaA * A) + (areaB * B)) >> 5; - - return (result & redblueMask) | ((result >> 16) & greenMask); -} - -static uint32_t Bilinear4 (uint32_t A, uint32_t B, uint32_t C, uint32_t D, uint32_t x, - uint32_t y) -{ - unsigned long areaA, areaB, areaC, areaD; - unsigned long result, xy; - - x = (x >> 11) & 0x1f; - y = (y >> 11) & 0x1f; - xy = (x * y) >> 5; - - A = (A & redblueMask) | ((A & greenMask) << 16); - B = (B & redblueMask) | ((B & greenMask) << 16); - C = (C & redblueMask) | ((C & greenMask) << 16); - D = (D & redblueMask) | ((D & greenMask) << 16); - - areaA = 0x20 + xy - x - y; - areaB = x - xy; - areaC = y - xy; - areaD = xy; - - result = ((areaA * A) + (areaB * B) + (areaC * C) + (areaD * D)) >> 5; - - return (result & redblueMask) | ((result >> 16) & greenMask); -} - -void Scale_2xSaI (uint8_t *srcPtr, uint32_t srcPitch, uint8_t * /* deltaPtr */, - uint8_t *dstPtr, uint32_t dstPitch, - uint32_t dstWidth, uint32_t dstHeight, int width, int height) -{ - uint8_t *dP; - uint16_t *bP; - - uint32_t w; - uint32_t h; - uint32_t dw; - uint32_t dh; - uint32_t hfinish; - uint32_t wfinish; - - uint32_t Nextline = srcPitch >> 1; - - wfinish = (width - 1) << 16; // convert to fixed point - dw = wfinish / (dstWidth - 1); - hfinish = (height - 1) << 16; // convert to fixed point - dh = hfinish / (dstHeight - 1); - - for (h = 0; h < hfinish; h += dh) { - uint32_t y1, y2; - - y1 = h & 0xffff; // fraction part of fixed point - bP = (uint16_t *) (srcPtr + ((h >> 16) * srcPitch)); - dP = dstPtr; - y2 = 0x10000 - y1; - - w = 0; - - for (; w < wfinish;) { - uint32_t A, B, C, D; - uint32_t E, F, G, H; - uint32_t I, J, K, L; - uint32_t x1, x2, a1, f1, f2; - uint32_t position, product1; - - position = w >> 16; - A = bP[position]; // current pixel - B = bP[position + 1]; // next pixel - C = bP[position + Nextline]; - D = bP[position + Nextline + 1]; - E = bP[position - Nextline]; - F = bP[position - Nextline + 1]; - G = bP[position - 1]; - H = bP[position + Nextline - 1]; - I = bP[position + 2]; - J = bP[position + Nextline + 2]; - K = bP[position + Nextline + Nextline]; - L = bP[position + Nextline + Nextline + 1]; - - x1 = w & 0xffff; // fraction part of fixed point - x2 = 0x10000 - x1; - - /*0*/ - if (A == B && C == D && A == C) - product1 = A; - else /*1*/ if (A == D && B != C) { - f1 = (x1 >> 1) + (0x10000 >> 2); - f2 = (y1 >> 1) + (0x10000 >> 2); - if (y1 <= f1 && A == J && A != E) // close to B - { - a1 = f1 - y1; - product1 = Bilinear (A, B, a1); - } else if (y1 >= f1 && A == G && A != L) // close to C - { - a1 = y1 - f1; - product1 = Bilinear (A, C, a1); - } - else if (x1 >= f2 && A == E && A != J) // close to B - { - a1 = x1 - f2; - product1 = Bilinear (A, B, a1); - } - else if (x1 <= f2 && A == L && A != G) // close to C - { - a1 = f2 - x1; - product1 = Bilinear (A, C, a1); - } - else if (y1 >= x1) // close to C - { - a1 = y1 - x1; - product1 = Bilinear (A, C, a1); - } - else if (y1 <= x1) // close to B - { - a1 = x1 - y1; - product1 = Bilinear (A, B, a1); - } - } - else - /*2*/ - if (B == C && A != D) - { - f1 = (x1 >> 1) + (0x10000 >> 2); - f2 = (y1 >> 1) + (0x10000 >> 2); - if (y2 >= f1 && B == H && B != F) // close to A - { - a1 = y2 - f1; - product1 = Bilinear (B, A, a1); - } - else if (y2 <= f1 && B == I && B != K) // close to D - { - a1 = f1 - y2; - product1 = Bilinear (B, D, a1); - } - else if (x2 >= f2 && B == F && B != H) // close to A - { - a1 = x2 - f2; - product1 = Bilinear (B, A, a1); - } - else if (x2 <= f2 && B == K && B != I) // close to D - { - a1 = f2 - x2; - product1 = Bilinear (B, D, a1); - } - else if (y2 >= x1) // close to A - { - a1 = y2 - x1; - product1 = Bilinear (B, A, a1); - } - else if (y2 <= x1) // close to D - { - a1 = x1 - y2; - product1 = Bilinear (B, D, a1); - } - } - /*3*/ - else - { - product1 = Bilinear4 (A, B, C, D, x1, y1); - } - - //end First Pixel - *(uint32_t *) dP = product1; - dP += 2; - w += dw; - } - dstPtr += dstPitch; - } -} diff --git a/snesfilter/HQ2x/HQ2x.cpp b/snesfilter/HQ2x/HQ2x.cpp index f4662b6d..89aa1e70 100755 --- a/snesfilter/HQ2x/HQ2x.cpp +++ b/snesfilter/HQ2x/HQ2x.cpp @@ -4,7 +4,7 @@ using namespace nall; extern "C" { void filter_size(unsigned&, unsigned&); - void filter_render(uint32_t*, uint32_t*, unsigned, const uint16_t*, unsigned, unsigned, unsigned); + void filter_render(uint16_t*, unsigned, const uint16_t*, unsigned, unsigned, unsigned); }; enum { @@ -147,21 +147,21 @@ dllexport void filter_size(unsigned &width, unsigned &height) { } dllexport void filter_render( - uint32_t *colortable, uint32_t *output, unsigned outpitch, - const uint16_t *input, unsigned pitch, unsigned width, unsigned height + uint16_t *output, unsigned outputPitch, + const uint16_t *input, unsigned inputPitch, + unsigned width, unsigned height ) { initialize(); - pitch >>= 1; - outpitch >>= 2; + outputPitch >>= 1, inputPitch >>= 1; -//#pragma omp parallel for + #pragma omp parallel for for(unsigned y = 0; y < height; y++) { - const uint16_t *in = input + y * pitch; - uint32_t *out0 = output + y * outpitch * 2; - uint32_t *out1 = output + y * outpitch * 2 + outpitch; + const uint16_t *in = input + y * inputPitch; + uint16_t *out0 = output + y * outputPitch * 2; + uint16_t *out1 = output + y * outputPitch * 2 + outputPitch; - int prevline = (y == 0 ? 0 : pitch); - int nextline = (y == height - 1 ? 0 : pitch); + int prevline = (y == 0 ? 0 : inputPitch); + int nextline = (y == height - 1 ? 0 : inputPitch); in++; *out0++ = 0; *out0++ = 0; @@ -189,10 +189,10 @@ dllexport void filter_render( pattern |= diff(e, H) << 6; pattern |= diff(e, I) << 7; - *(out0 + 0) = colortable[blend(hqTable[pattern], E, A, B, D, F, H)]; pattern = rotate[pattern]; - *(out0 + 1) = colortable[blend(hqTable[pattern], E, C, F, B, H, D)]; pattern = rotate[pattern]; - *(out1 + 1) = colortable[blend(hqTable[pattern], E, I, H, F, D, B)]; pattern = rotate[pattern]; - *(out1 + 0) = colortable[blend(hqTable[pattern], E, G, D, H, B, F)]; + *(out0 + 0) = blend(hqTable[pattern], E, A, B, D, F, H); pattern = rotate[pattern]; + *(out0 + 1) = blend(hqTable[pattern], E, C, F, B, H, D); pattern = rotate[pattern]; + *(out1 + 1) = blend(hqTable[pattern], E, I, H, F, D, B); pattern = rotate[pattern]; + *(out1 + 0) = blend(hqTable[pattern], E, G, D, H, B, F); in++; out0 += 2; diff --git a/snesfilter/LQ2x/LQ2x.cpp b/snesfilter/LQ2x/LQ2x.cpp index 8015aa49..9da5f593 100755 --- a/snesfilter/LQ2x/LQ2x.cpp +++ b/snesfilter/LQ2x/LQ2x.cpp @@ -4,7 +4,7 @@ using namespace nall; extern "C" { void filter_size(unsigned&, unsigned&); - void filter_render(uint32_t*, uint32_t*, unsigned, const uint16_t*, unsigned, unsigned, unsigned); + void filter_render(uint16_t*, unsigned, const uint16_t*, unsigned, unsigned, unsigned); }; dllexport void filter_size(unsigned &width, unsigned &height) { @@ -13,18 +13,20 @@ dllexport void filter_size(unsigned &width, unsigned &height) { } dllexport void filter_render( - uint32_t *colortable, uint32_t *output, unsigned outpitch, - const uint16_t *input, unsigned pitch, unsigned width, unsigned height + uint16_t *output, unsigned outputPitch, + const uint16_t *input, unsigned inputPitch, + unsigned width, unsigned height ) { - pitch >>= 1; - outpitch >>= 2; - - uint32_t *out0 = output; - uint32_t *out1 = output + outpitch; + outputPitch >>= 1, inputPitch >>= 1; + #pragma omp parallel for for(unsigned y = 0; y < height; y++) { - int prevline = (y == 0 ? 0 : pitch); - int nextline = (y == height - 1 ? 0 : pitch); + const uint16_t *in = input + y * inputPitch; + uint16_t *out0 = output + y * outputPitch * 2; + uint16_t *out1 = output + y * outputPitch * 2 + outputPitch; + + int prevline = (y == 0 ? 0 : inputPitch); + int nextline = (y == height - 1 ? 0 : inputPitch); for(unsigned x = 0; x < width; x++) { uint16_t A = *(input - prevline); @@ -32,23 +34,18 @@ dllexport void filter_render( uint16_t C = *input; uint16_t D = (x < width - 1) ? *(input + 1) : *input; uint16_t E = *(input++ + nextline); - uint32_t c = colortable[C]; if(A != E && B != D) { - *out0++ = (A == B ? colortable[C + A - ((C ^ A) & 0x0421) >> 1] : c); - *out0++ = (A == D ? colortable[C + A - ((C ^ A) & 0x0421) >> 1] : c); - *out1++ = (E == B ? colortable[C + E - ((C ^ E) & 0x0421) >> 1] : c); - *out1++ = (E == D ? colortable[C + E - ((C ^ E) & 0x0421) >> 1] : c); + *out0++ = (A == B ? C + A - ((C ^ A) & 0x0421) >> 1 : C); + *out0++ = (A == D ? C + A - ((C ^ A) & 0x0421) >> 1 : C); + *out1++ = (E == B ? C + E - ((C ^ E) & 0x0421) >> 1 : C); + *out1++ = (E == D ? C + E - ((C ^ E) & 0x0421) >> 1 : C); } else { - *out0++ = c; - *out0++ = c; - *out1++ = c; - *out1++ = c; + *out0++ = C; + *out0++ = C; + *out1++ = C; + *out1++ = C; } } - - input += pitch - width; - out0 += outpitch + outpitch - (width << 1); - out1 += outpitch + outpitch - (width << 1); } } diff --git a/snesfilter/Makefile b/snesfilter/Makefile index 49abea35..87fd2096 100755 --- a/snesfilter/Makefile +++ b/snesfilter/Makefile @@ -7,19 +7,11 @@ link := -s objects := objects += out/Pixellate2x.filter -objects += out/Scanline-Black.filter objects += out/Scanline-Dark.filter objects += out/Scanline-Light.filter objects += out/Scale2x.filter -objects += out/2xSaI.filter -objects += out/Super-2xSaI.filter -objects += out/Super-Eagle.filter objects += out/LQ2x.filter objects += out/HQ2x.filter -objects += out/NTSC-RF.filter -objects += out/NTSC-Composite.filter -objects += out/NTSC-SVideo.filter -objects += out/NTSC-RGB.filter compile = $(cpp) $(link) $(flags) -o $@ -shared $< @@ -28,26 +20,18 @@ compile = $(cpp) $(link) $(flags) -o $@ -shared $< all: build; out/Pixellate2x.filter: Pixellate2x/Pixellate2x.cpp Pixellate2x/* -out/Scanline-Black.filter: Scanline/Scanline-Black.cpp Scanline/* out/Scanline-Dark.filter: Scanline/Scanline-Dark.cpp Scanline/* out/Scanline-Light.filter: Scanline/Scanline-Light.cpp Scanline/* out/Scale2x.filter: Scale2x/Scale2x.cpp Scale2x/* -out/2xSaI.filter: 2xSaI/2xSaI.cpp 2xSaI/* -out/Super-2xSaI.filter: 2xSaI/Super-2xSaI.cpp 2xSaI/* -out/Super-Eagle.filter: 2xSaI/Super-Eagle.cpp 2xSaI/* out/LQ2x.filter: LQ2x/LQ2x.cpp LQ2x/* out/HQ2x.filter: HQ2x/HQ2x.cpp HQ2x/* -out/NTSC-RF.filter: NTSC/NTSC-RF.cpp NTSC/* -out/NTSC-Composite.filter: NTSC/NTSC-Composite.cpp NTSC/* -out/NTSC-SVideo.filter: NTSC/NTSC-SVideo.cpp NTSC/* -out/NTSC-RGB.filter: NTSC/NTSC-RGB.cpp NTSC/* build: $(objects) install: - mkdir -p ~/.config/bsnes/filters - chmod 777 ~/.config/bsnes/filters - cp out/*.filter ~/.config/bsnes/filters + mkdir -p ~/.config/batch/filters + chmod 777 ~/.config/batch/filters + cp out/*.filter ~/.config/batch/filters clean: rm out/*.filter diff --git a/snesfilter/NTSC/NTSC-Composite.cpp b/snesfilter/NTSC/NTSC-Composite.cpp deleted file mode 100755 index 0e0ba16c..00000000 --- a/snesfilter/NTSC/NTSC-Composite.cpp +++ /dev/null @@ -1,63 +0,0 @@ -#include -#include -using namespace nall; - -uint32_t *colortable; - -#include "snes_ntsc/snes_ntsc.h" -#include "snes_ntsc/snes_ntsc.c" - -extern "C" { - void filter_size(unsigned&, unsigned&); - void filter_render(uint32_t*, uint32_t*, unsigned, const uint16_t*, unsigned, unsigned, unsigned); -}; - - -struct snes_ntsc_t *ntsc; -snes_ntsc_setup_t setup; -int burst; -int burst_toggle; - -void initialize() { - static bool initialized = false; - if(initialized == true) return; - initialized = true; - - ntsc = (snes_ntsc_t*)malloc(sizeof *ntsc); - setup = snes_ntsc_composite; - setup.merge_fields = 1; - snes_ntsc_init(ntsc, &setup); - - burst = 0; - burst_toggle = (setup.merge_fields ? 0 : 1); -} - -void terminate() { - if(ntsc) free(ntsc); -} - -dllexport void filter_size(unsigned &width, unsigned &height) { - initialize(); - width = SNES_NTSC_OUT_WIDTH(256); - height = height; -} - -dllexport void filter_render( - uint32_t *colortable_, uint32_t *output, unsigned outpitch, - const uint16_t *input, unsigned pitch, unsigned width, unsigned height -) { - initialize(); - if(!ntsc) return; - colortable = colortable_; - - pitch >>= 1; - outpitch >>= 2; - - if(width <= 256) { - snes_ntsc_blit (ntsc, input, pitch, burst, width, height, output, outpitch << 2); - } else { - snes_ntsc_blit_hires(ntsc, input, pitch, burst, width, height, output, outpitch << 2); - } - - burst ^= burst_toggle; -} diff --git a/snesfilter/NTSC/NTSC-RF.cpp b/snesfilter/NTSC/NTSC-RF.cpp deleted file mode 100755 index 57929c92..00000000 --- a/snesfilter/NTSC/NTSC-RF.cpp +++ /dev/null @@ -1,63 +0,0 @@ -#include -#include -using namespace nall; - -uint32_t *colortable; - -#include "snes_ntsc/snes_ntsc.h" -#include "snes_ntsc/snes_ntsc.c" - -extern "C" { - void filter_size(unsigned&, unsigned&); - void filter_render(uint32_t*, uint32_t*, unsigned, const uint16_t*, unsigned, unsigned, unsigned); -}; - - -struct snes_ntsc_t *ntsc; -snes_ntsc_setup_t setup; -int burst; -int burst_toggle; - -void initialize() { - static bool initialized = false; - if(initialized == true) return; - initialized = true; - - ntsc = (snes_ntsc_t*)malloc(sizeof *ntsc); - setup = snes_ntsc_composite; - setup.merge_fields = 0; - snes_ntsc_init(ntsc, &setup); - - burst = 0; - burst_toggle = (setup.merge_fields ? 0 : 1); -} - -void terminate() { - if(ntsc) free(ntsc); -} - -dllexport void filter_size(unsigned &width, unsigned &height) { - initialize(); - width = SNES_NTSC_OUT_WIDTH(256); - height = height; -} - -dllexport void filter_render( - uint32_t *colortable_, uint32_t *output, unsigned outpitch, - const uint16_t *input, unsigned pitch, unsigned width, unsigned height -) { - initialize(); - if(!ntsc) return; - colortable = colortable_; - - pitch >>= 1; - outpitch >>= 2; - - if(width <= 256) { - snes_ntsc_blit (ntsc, input, pitch, burst, width, height, output, outpitch << 2); - } else { - snes_ntsc_blit_hires(ntsc, input, pitch, burst, width, height, output, outpitch << 2); - } - - burst ^= burst_toggle; -} diff --git a/snesfilter/NTSC/NTSC-RGB.cpp b/snesfilter/NTSC/NTSC-RGB.cpp deleted file mode 100755 index 9067df75..00000000 --- a/snesfilter/NTSC/NTSC-RGB.cpp +++ /dev/null @@ -1,63 +0,0 @@ -#include -#include -using namespace nall; - -uint32_t *colortable; - -#include "snes_ntsc/snes_ntsc.h" -#include "snes_ntsc/snes_ntsc.c" - -extern "C" { - void filter_size(unsigned&, unsigned&); - void filter_render(uint32_t*, uint32_t*, unsigned, const uint16_t*, unsigned, unsigned, unsigned); -}; - - -struct snes_ntsc_t *ntsc; -snes_ntsc_setup_t setup; -int burst; -int burst_toggle; - -void initialize() { - static bool initialized = false; - if(initialized == true) return; - initialized = true; - - ntsc = (snes_ntsc_t*)malloc(sizeof *ntsc); - setup = snes_ntsc_rgb; - setup.merge_fields = 1; - snes_ntsc_init(ntsc, &setup); - - burst = 0; - burst_toggle = (setup.merge_fields ? 0 : 1); -} - -void terminate() { - if(ntsc) free(ntsc); -} - -dllexport void filter_size(unsigned &width, unsigned &height) { - initialize(); - width = SNES_NTSC_OUT_WIDTH(256); - height = height; -} - -dllexport void filter_render( - uint32_t *colortable_, uint32_t *output, unsigned outpitch, - const uint16_t *input, unsigned pitch, unsigned width, unsigned height -) { - initialize(); - if(!ntsc) return; - colortable = colortable_; - - pitch >>= 1; - outpitch >>= 2; - - if(width <= 256) { - snes_ntsc_blit (ntsc, input, pitch, burst, width, height, output, outpitch << 2); - } else { - snes_ntsc_blit_hires(ntsc, input, pitch, burst, width, height, output, outpitch << 2); - } - - burst ^= burst_toggle; -} diff --git a/snesfilter/NTSC/NTSC-SVideo.cpp b/snesfilter/NTSC/NTSC-SVideo.cpp deleted file mode 100755 index 9fe8f9d3..00000000 --- a/snesfilter/NTSC/NTSC-SVideo.cpp +++ /dev/null @@ -1,63 +0,0 @@ -#include -#include -using namespace nall; - -uint32_t *colortable; - -#include "snes_ntsc/snes_ntsc.h" -#include "snes_ntsc/snes_ntsc.c" - -extern "C" { - void filter_size(unsigned&, unsigned&); - void filter_render(uint32_t*, uint32_t*, unsigned, const uint16_t*, unsigned, unsigned, unsigned); -}; - - -struct snes_ntsc_t *ntsc; -snes_ntsc_setup_t setup; -int burst; -int burst_toggle; - -void initialize() { - static bool initialized = false; - if(initialized == true) return; - initialized = true; - - ntsc = (snes_ntsc_t*)malloc(sizeof *ntsc); - setup = snes_ntsc_svideo; - setup.merge_fields = 1; - snes_ntsc_init(ntsc, &setup); - - burst = 0; - burst_toggle = (setup.merge_fields ? 0 : 1); -} - -void terminate() { - if(ntsc) free(ntsc); -} - -dllexport void filter_size(unsigned &width, unsigned &height) { - initialize(); - width = SNES_NTSC_OUT_WIDTH(256); - height = height; -} - -dllexport void filter_render( - uint32_t *colortable_, uint32_t *output, unsigned outpitch, - const uint16_t *input, unsigned pitch, unsigned width, unsigned height -) { - initialize(); - if(!ntsc) return; - colortable = colortable_; - - pitch >>= 1; - outpitch >>= 2; - - if(width <= 256) { - snes_ntsc_blit (ntsc, input, pitch, burst, width, height, output, outpitch << 2); - } else { - snes_ntsc_blit_hires(ntsc, input, pitch, burst, width, height, output, outpitch << 2); - } - - burst ^= burst_toggle; -} diff --git a/snesfilter/NTSC/snes_ntsc/snes_ntsc.c b/snesfilter/NTSC/snes_ntsc/snes_ntsc.c deleted file mode 100755 index f622baf8..00000000 --- a/snesfilter/NTSC/snes_ntsc/snes_ntsc.c +++ /dev/null @@ -1,251 +0,0 @@ -/* snes_ntsc 0.2.2. http://www.slack.net/~ant/ */ - -#include "snes_ntsc.h" - -/* Copyright (C) 2006-2007 Shay Green. This module is free software; you -can redistribute it and/or modify it under the terms of the GNU Lesser -General Public License as published by the Free Software Foundation; either -version 2.1 of the License, or (at your option) any later version. This -module 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 Lesser General Public License for more -details. You should have received a copy of the GNU Lesser General Public -License along with this module; if not, write to the Free Software Foundation, -Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */ - -snes_ntsc_setup_t const snes_ntsc_monochrome = { 0,-1, 0, 0,.2, 0,.2,-.2,-.2,-1, 1, 0, 0 }; -snes_ntsc_setup_t const snes_ntsc_composite = { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0 }; -snes_ntsc_setup_t const snes_ntsc_svideo = { 0, 0, 0, 0,.2, 0,.2, -1, -1, 0, 1, 0, 0 }; -snes_ntsc_setup_t const snes_ntsc_rgb = { 0, 0, 0, 0,.2, 0,.7, -1, -1,-1, 1, 0, 0 }; - -#define alignment_count 3 -#define burst_count 3 -#define rescale_in 8 -#define rescale_out 7 - -#define artifacts_mid 1.0f -#define fringing_mid 1.0f -#define std_decoder_hue 0 - -#define rgb_bits 7 /* half normal range to allow for doubled hires pixels */ -#define gamma_size 32 - -#include "snes_ntsc_impl.h" - -/* 3 input pixels -> 8 composite samples */ -pixel_info_t const snes_ntsc_pixels [alignment_count] = { - { PIXEL_OFFSET( -4, -9 ), { 1, 1, .6667f, 0 } }, - { PIXEL_OFFSET( -2, -7 ), { .3333f, 1, 1, .3333f } }, - { PIXEL_OFFSET( 0, -5 ), { 0, .6667f, 1, 1 } }, -}; - -static void merge_kernel_fields( snes_ntsc_rgb_t* io ) -{ - int n; - for ( n = burst_size; n; --n ) - { - snes_ntsc_rgb_t p0 = io [burst_size * 0] + rgb_bias; - snes_ntsc_rgb_t p1 = io [burst_size * 1] + rgb_bias; - snes_ntsc_rgb_t p2 = io [burst_size * 2] + rgb_bias; - /* merge colors without losing precision */ - io [burst_size * 0] = - ((p0 + p1 - ((p0 ^ p1) & snes_ntsc_rgb_builder)) >> 1) - rgb_bias; - io [burst_size * 1] = - ((p1 + p2 - ((p1 ^ p2) & snes_ntsc_rgb_builder)) >> 1) - rgb_bias; - io [burst_size * 2] = - ((p2 + p0 - ((p2 ^ p0) & snes_ntsc_rgb_builder)) >> 1) - rgb_bias; - ++io; - } -} - -static void correct_errors( snes_ntsc_rgb_t color, snes_ntsc_rgb_t* out ) -{ - int n; - for ( n = burst_count; n; --n ) - { - unsigned i; - for ( i = 0; i < rgb_kernel_size / 2; i++ ) - { - snes_ntsc_rgb_t error = color - - out [i ] - out [(i+12)%14+14] - out [(i+10)%14+28] - - out [i + 7] - out [i + 5 +14] - out [i + 3 +28]; - DISTRIBUTE_ERROR( i+3+28, i+5+14, i+7 ); - } - out += alignment_count * rgb_kernel_size; - } -} - -void snes_ntsc_init( snes_ntsc_t* ntsc, snes_ntsc_setup_t const* setup ) -{ - int merge_fields; - int entry; - init_t impl; - if ( !setup ) - setup = &snes_ntsc_composite; - init( &impl, setup ); - - merge_fields = setup->merge_fields; - if ( setup->artifacts <= -1 && setup->fringing <= -1 ) - merge_fields = 1; - - for ( entry = 0; entry < snes_ntsc_palette_size; entry++ ) - { - /* Reduce number of significant bits of source color. Clearing the - low bits of R and B were least notictable. Modifying green was too - noticeable. */ - int ir = entry >> 8 & 0x1E; - int ig = entry >> 4 & 0x1F; - int ib = entry << 1 & 0x1E; - - #if SNES_NTSC_BSNES_COLORTBL - if ( setup->bsnes_colortbl ) - { - int bgr15 = (ib << 10) | (ig << 5) | ir; - unsigned long rgb16 = setup->bsnes_colortbl [bgr15]; - ir = rgb16 >> 11 & 0x1E; - ig = rgb16 >> 6 & 0x1F; - ib = rgb16 & 0x1E; - } - #endif - - { - float rr = impl.to_float [ir]; - float gg = impl.to_float [ig]; - float bb = impl.to_float [ib]; - - float y, i, q = RGB_TO_YIQ( rr, gg, bb, y, i ); - - int r, g, b = YIQ_TO_RGB( y, i, q, impl.to_rgb, int, r, g ); - snes_ntsc_rgb_t rgb = PACK_RGB( r, g, b ); - - snes_ntsc_rgb_t* out = ntsc->table [entry]; - gen_kernel( &impl, y, i, q, out ); - if ( merge_fields ) - merge_kernel_fields( out ); - correct_errors( rgb, out ); - } - } -} - -#ifndef SNES_NTSC_NO_BLITTERS - -void snes_ntsc_blit( snes_ntsc_t const* ntsc, SNES_NTSC_IN_T const* input, long in_row_width, - int burst_phase, int in_width, int in_height, void* rgb_out, long out_pitch ) -{ - int chunk_count = (in_width - 1) / snes_ntsc_in_chunk; - for ( ; in_height; --in_height ) - { - SNES_NTSC_IN_T const* line_in = input; - SNES_NTSC_BEGIN_ROW( ntsc, burst_phase, - snes_ntsc_black, snes_ntsc_black, SNES_NTSC_ADJ_IN( *line_in ) ); - snes_ntsc_out_t* restrict line_out = (snes_ntsc_out_t*) rgb_out; - int n; - ++line_in; - - for ( n = chunk_count; n; --n ) - { - /* order of input and output pixels must not be altered */ - SNES_NTSC_COLOR_IN( 0, SNES_NTSC_ADJ_IN( line_in [0] ) ); - SNES_NTSC_RGB_OUT( 0, line_out [0], SNES_NTSC_OUT_DEPTH ); - SNES_NTSC_RGB_OUT( 1, line_out [1], SNES_NTSC_OUT_DEPTH ); - - SNES_NTSC_COLOR_IN( 1, SNES_NTSC_ADJ_IN( line_in [1] ) ); - SNES_NTSC_RGB_OUT( 2, line_out [2], SNES_NTSC_OUT_DEPTH ); - SNES_NTSC_RGB_OUT( 3, line_out [3], SNES_NTSC_OUT_DEPTH ); - - SNES_NTSC_COLOR_IN( 2, SNES_NTSC_ADJ_IN( line_in [2] ) ); - SNES_NTSC_RGB_OUT( 4, line_out [4], SNES_NTSC_OUT_DEPTH ); - SNES_NTSC_RGB_OUT( 5, line_out [5], SNES_NTSC_OUT_DEPTH ); - SNES_NTSC_RGB_OUT( 6, line_out [6], SNES_NTSC_OUT_DEPTH ); - - line_in += 3; - line_out += 7; - } - - /* finish final pixels */ - SNES_NTSC_COLOR_IN( 0, snes_ntsc_black ); - SNES_NTSC_RGB_OUT( 0, line_out [0], SNES_NTSC_OUT_DEPTH ); - SNES_NTSC_RGB_OUT( 1, line_out [1], SNES_NTSC_OUT_DEPTH ); - - SNES_NTSC_COLOR_IN( 1, snes_ntsc_black ); - SNES_NTSC_RGB_OUT( 2, line_out [2], SNES_NTSC_OUT_DEPTH ); - SNES_NTSC_RGB_OUT( 3, line_out [3], SNES_NTSC_OUT_DEPTH ); - - SNES_NTSC_COLOR_IN( 2, snes_ntsc_black ); - SNES_NTSC_RGB_OUT( 4, line_out [4], SNES_NTSC_OUT_DEPTH ); - SNES_NTSC_RGB_OUT( 5, line_out [5], SNES_NTSC_OUT_DEPTH ); - SNES_NTSC_RGB_OUT( 6, line_out [6], SNES_NTSC_OUT_DEPTH ); - - burst_phase = (burst_phase + 1) % snes_ntsc_burst_count; - input += in_row_width; - rgb_out = (char*) rgb_out + out_pitch; - } -} - -void snes_ntsc_blit_hires( snes_ntsc_t const* ntsc, SNES_NTSC_IN_T const* input, long in_row_width, - int burst_phase, int in_width, int in_height, void* rgb_out, long out_pitch ) -{ - int chunk_count = (in_width - 2) / (snes_ntsc_in_chunk * 2); - for ( ; in_height; --in_height ) - { - SNES_NTSC_IN_T const* line_in = input; - SNES_NTSC_HIRES_ROW( ntsc, burst_phase, - snes_ntsc_black, snes_ntsc_black, snes_ntsc_black, - SNES_NTSC_ADJ_IN( line_in [0] ), - SNES_NTSC_ADJ_IN( line_in [1] ) ); - snes_ntsc_out_t* restrict line_out = (snes_ntsc_out_t*) rgb_out; - int n; - line_in += 2; - - for ( n = chunk_count; n; --n ) - { - /* twice as many input pixels per chunk */ - SNES_NTSC_COLOR_IN( 0, SNES_NTSC_ADJ_IN( line_in [0] ) ); - SNES_NTSC_HIRES_OUT( 0, line_out [0], SNES_NTSC_OUT_DEPTH ); - - SNES_NTSC_COLOR_IN( 1, SNES_NTSC_ADJ_IN( line_in [1] ) ); - SNES_NTSC_HIRES_OUT( 1, line_out [1], SNES_NTSC_OUT_DEPTH ); - - SNES_NTSC_COLOR_IN( 2, SNES_NTSC_ADJ_IN( line_in [2] ) ); - SNES_NTSC_HIRES_OUT( 2, line_out [2], SNES_NTSC_OUT_DEPTH ); - - SNES_NTSC_COLOR_IN( 3, SNES_NTSC_ADJ_IN( line_in [3] ) ); - SNES_NTSC_HIRES_OUT( 3, line_out [3], SNES_NTSC_OUT_DEPTH ); - - SNES_NTSC_COLOR_IN( 4, SNES_NTSC_ADJ_IN( line_in [4] ) ); - SNES_NTSC_HIRES_OUT( 4, line_out [4], SNES_NTSC_OUT_DEPTH ); - - SNES_NTSC_COLOR_IN( 5, SNES_NTSC_ADJ_IN( line_in [5] ) ); - SNES_NTSC_HIRES_OUT( 5, line_out [5], SNES_NTSC_OUT_DEPTH ); - SNES_NTSC_HIRES_OUT( 6, line_out [6], SNES_NTSC_OUT_DEPTH ); - - line_in += 6; - line_out += 7; - } - - SNES_NTSC_COLOR_IN( 0, snes_ntsc_black ); - SNES_NTSC_HIRES_OUT( 0, line_out [0], SNES_NTSC_OUT_DEPTH ); - - SNES_NTSC_COLOR_IN( 1, snes_ntsc_black ); - SNES_NTSC_HIRES_OUT( 1, line_out [1], SNES_NTSC_OUT_DEPTH ); - - SNES_NTSC_COLOR_IN( 2, snes_ntsc_black ); - SNES_NTSC_HIRES_OUT( 2, line_out [2], SNES_NTSC_OUT_DEPTH ); - - SNES_NTSC_COLOR_IN( 3, snes_ntsc_black ); - SNES_NTSC_HIRES_OUT( 3, line_out [3], SNES_NTSC_OUT_DEPTH ); - - SNES_NTSC_COLOR_IN( 4, snes_ntsc_black ); - SNES_NTSC_HIRES_OUT( 4, line_out [4], SNES_NTSC_OUT_DEPTH ); - - SNES_NTSC_COLOR_IN( 5, snes_ntsc_black ); - SNES_NTSC_HIRES_OUT( 5, line_out [5], SNES_NTSC_OUT_DEPTH ); - SNES_NTSC_HIRES_OUT( 6, line_out [6], SNES_NTSC_OUT_DEPTH ); - - burst_phase = (burst_phase + 1) % snes_ntsc_burst_count; - input += in_row_width; - rgb_out = (char*) rgb_out + out_pitch; - } -} - -#endif diff --git a/snesfilter/NTSC/snes_ntsc/snes_ntsc.h b/snesfilter/NTSC/snes_ntsc/snes_ntsc.h deleted file mode 100755 index fff97ecd..00000000 --- a/snesfilter/NTSC/snes_ntsc/snes_ntsc.h +++ /dev/null @@ -1,228 +0,0 @@ -/* SNES NTSC video filter */ - -/* snes_ntsc 0.2.2 */ -#ifndef SNES_NTSC_H -#define SNES_NTSC_H - -#include "snes_ntsc_config.h" - -#ifdef __cplusplus - extern "C" { -#endif - -/* Image parameters, ranging from -1.0 to 1.0. Actual internal values shown -in parenthesis and should remain fairly stable in future versions. */ -typedef struct snes_ntsc_setup_t -{ - /* Basic parameters */ - double hue; /* -1 = -180 degrees +1 = +180 degrees */ - double saturation; /* -1 = grayscale (0.0) +1 = oversaturated colors (2.0) */ - double contrast; /* -1 = dark (0.5) +1 = light (1.5) */ - double brightness; /* -1 = dark (0.5) +1 = light (1.5) */ - double sharpness; /* edge contrast enhancement/blurring */ - - /* Advanced parameters */ - double gamma; /* -1 = dark (1.5) +1 = light (0.5) */ - double resolution; /* image resolution */ - double artifacts; /* artifacts caused by color changes */ - double fringing; /* color artifacts caused by brightness changes */ - double bleed; /* color bleed (color resolution reduction) */ - int merge_fields; /* if 1, merges even and odd fields together to reduce flicker */ - float const* decoder_matrix; /* optional RGB decoder matrix, 6 elements */ - - unsigned long const* bsnes_colortbl; /* undocumented; set to 0 */ -} snes_ntsc_setup_t; - -/* Video format presets */ -extern snes_ntsc_setup_t const snes_ntsc_composite; /* color bleeding + artifacts */ -extern snes_ntsc_setup_t const snes_ntsc_svideo; /* color bleeding only */ -extern snes_ntsc_setup_t const snes_ntsc_rgb; /* crisp image */ -extern snes_ntsc_setup_t const snes_ntsc_monochrome;/* desaturated + artifacts */ - -/* Initializes and adjusts parameters. Can be called multiple times on the same -snes_ntsc_t object. Can pass NULL for either parameter. */ -typedef struct snes_ntsc_t snes_ntsc_t; -void snes_ntsc_init( snes_ntsc_t* ntsc, snes_ntsc_setup_t const* setup ); - -/* Filters one or more rows of pixels. Input pixel format is set by SNES_NTSC_IN_FORMAT -and output RGB depth is set by SNES_NTSC_OUT_DEPTH. Both default to 16-bit RGB. -In_row_width is the number of pixels to get to the next input row. Out_pitch -is the number of *bytes* to get to the next output row. */ -void snes_ntsc_blit( snes_ntsc_t const* ntsc, SNES_NTSC_IN_T const* input, - long in_row_width, int burst_phase, int in_width, int in_height, - void* rgb_out, long out_pitch ); - -void snes_ntsc_blit_hires( snes_ntsc_t const* ntsc, SNES_NTSC_IN_T const* input, - long in_row_width, int burst_phase, int in_width, int in_height, - void* rgb_out, long out_pitch ); - -/* Number of output pixels written by low-res blitter for given input width. Width -might be rounded down slightly; use SNES_NTSC_IN_WIDTH() on result to find rounded -value. Guaranteed not to round 256 down at all. */ -#define SNES_NTSC_OUT_WIDTH( in_width ) \ - ((((in_width) - 1) / snes_ntsc_in_chunk + 1) * snes_ntsc_out_chunk) - -/* Number of low-res input pixels that will fit within given output width. Might be -rounded down slightly; use SNES_NTSC_OUT_WIDTH() on result to find rounded -value. */ -#define SNES_NTSC_IN_WIDTH( out_width ) \ - (((out_width) / snes_ntsc_out_chunk - 1) * snes_ntsc_in_chunk + 1) - - -/* Interface for user-defined custom blitters */ - -enum { snes_ntsc_in_chunk = 3 }; /* number of input pixels read per chunk */ -enum { snes_ntsc_out_chunk = 7 }; /* number of output pixels generated per chunk */ -enum { snes_ntsc_black = 0 }; /* palette index for black */ -enum { snes_ntsc_burst_count = 3 }; /* burst phase cycles through 0, 1, and 2 */ - -/* Begins outputting row and starts three pixels. First pixel will be cut off a bit. -Use snes_ntsc_black for unused pixels. Declares variables, so must be before first -statement in a block (unless you're using C++). */ -#define SNES_NTSC_BEGIN_ROW( ntsc, burst, pixel0, pixel1, pixel2 ) \ - char const* ktable = \ - (char const*) (ntsc)->table + burst * (snes_ntsc_burst_size * sizeof (snes_ntsc_rgb_t));\ - SNES_NTSC_BEGIN_ROW_6_( pixel0, pixel1, pixel2, SNES_NTSC_IN_FORMAT, ktable ) - -/* Begins input pixel */ -#define SNES_NTSC_COLOR_IN( index, color ) \ - SNES_NTSC_COLOR_IN_( index, color, SNES_NTSC_IN_FORMAT, ktable ) - -/* Generates output pixel. Bits can be 24, 16, 15, 14, 32 (treated as 24), or 0: -24: RRRRRRRR GGGGGGGG BBBBBBBB (8-8-8 RGB) -16: RRRRRGGG GGGBBBBB (5-6-5 RGB) -15: RRRRRGG GGGBBBBB (5-5-5 RGB) -14: BBBBBGG GGGRRRRR (5-5-5 BGR, native SNES format) - 0: xxxRRRRR RRRxxGGG GGGGGxxB BBBBBBBx (native internal format; x = junk bits) */ -#define SNES_NTSC_RGB_OUT( index, rgb_out, bits ) \ - SNES_NTSC_RGB_OUT_14_( index, rgb_out, bits, 1 ) - -/* Hires equivalents */ -#define SNES_NTSC_HIRES_ROW( ntsc, burst, pixel1, pixel2, pixel3, pixel4, pixel5 ) \ - char const* ktable = \ - (char const*) (ntsc)->table + burst * (snes_ntsc_burst_size * sizeof (snes_ntsc_rgb_t));\ - unsigned const snes_ntsc_pixel1_ = (pixel1);\ - snes_ntsc_rgb_t const* kernel1 = SNES_NTSC_IN_FORMAT( ktable, snes_ntsc_pixel1_ );\ - unsigned const snes_ntsc_pixel2_ = (pixel2);\ - snes_ntsc_rgb_t const* kernel2 = SNES_NTSC_IN_FORMAT( ktable, snes_ntsc_pixel2_ );\ - unsigned const snes_ntsc_pixel3_ = (pixel3);\ - snes_ntsc_rgb_t const* kernel3 = SNES_NTSC_IN_FORMAT( ktable, snes_ntsc_pixel3_ );\ - unsigned const snes_ntsc_pixel4_ = (pixel4);\ - snes_ntsc_rgb_t const* kernel4 = SNES_NTSC_IN_FORMAT( ktable, snes_ntsc_pixel4_ );\ - unsigned const snes_ntsc_pixel5_ = (pixel5);\ - snes_ntsc_rgb_t const* kernel5 = SNES_NTSC_IN_FORMAT( ktable, snes_ntsc_pixel5_ );\ - snes_ntsc_rgb_t const* kernel0 = kernel1;\ - snes_ntsc_rgb_t const* kernelx0;\ - snes_ntsc_rgb_t const* kernelx1 = kernel1;\ - snes_ntsc_rgb_t const* kernelx2 = kernel1;\ - snes_ntsc_rgb_t const* kernelx3 = kernel1;\ - snes_ntsc_rgb_t const* kernelx4 = kernel1;\ - snes_ntsc_rgb_t const* kernelx5 = kernel1 - -#define SNES_NTSC_HIRES_OUT( x, rgb_out, bits ) {\ - snes_ntsc_rgb_t raw_ =\ - kernel0 [ x ] + kernel2 [(x+5)%7+14] + kernel4 [(x+3)%7+28] +\ - kernelx0 [(x+7)%7+7] + kernelx2 [(x+5)%7+21] + kernelx4 [(x+3)%7+35] +\ - kernel1 [(x+6)%7 ] + kernel3 [(x+4)%7+14] + kernel5 [(x+2)%7+28] +\ - kernelx1 [(x+6)%7+7] + kernelx3 [(x+4)%7+21] + kernelx5 [(x+2)%7+35];\ - SNES_NTSC_CLAMP_( raw_, 0 );\ - SNES_NTSC_RGB_OUT_( rgb_out, (bits), 0 );\ -} - - -/* private */ -enum { snes_ntsc_entry_size = 128 }; -enum { snes_ntsc_palette_size = 0x2000 }; -typedef unsigned long snes_ntsc_rgb_t; -struct snes_ntsc_t { - snes_ntsc_rgb_t table [snes_ntsc_palette_size] [snes_ntsc_entry_size]; -}; -enum { snes_ntsc_burst_size = snes_ntsc_entry_size / snes_ntsc_burst_count }; - -#define SNES_NTSC_RGB16( ktable, n ) \ - (snes_ntsc_rgb_t const*) (ktable + ((n & 0x001E) | (n >> 1 & 0x03E0) | (n >> 2 & 0x3C00)) * \ - (snes_ntsc_entry_size / 2 * sizeof (snes_ntsc_rgb_t))) - -#define SNES_NTSC_BGR15( ktable, n ) \ - (snes_ntsc_rgb_t const*) (ktable + ((n << 9 & 0x3C00) | (n & 0x03E0) | (n >> 10 & 0x001E)) * \ - (snes_ntsc_entry_size / 2 * sizeof (snes_ntsc_rgb_t))) - -/* common 3->7 ntsc macros */ -#define SNES_NTSC_BEGIN_ROW_6_( pixel0, pixel1, pixel2, ENTRY, table ) \ - unsigned const snes_ntsc_pixel0_ = (pixel0);\ - snes_ntsc_rgb_t const* kernel0 = ENTRY( table, snes_ntsc_pixel0_ );\ - unsigned const snes_ntsc_pixel1_ = (pixel1);\ - snes_ntsc_rgb_t const* kernel1 = ENTRY( table, snes_ntsc_pixel1_ );\ - unsigned const snes_ntsc_pixel2_ = (pixel2);\ - snes_ntsc_rgb_t const* kernel2 = ENTRY( table, snes_ntsc_pixel2_ );\ - snes_ntsc_rgb_t const* kernelx0;\ - snes_ntsc_rgb_t const* kernelx1 = kernel0;\ - snes_ntsc_rgb_t const* kernelx2 = kernel0 - -#define SNES_NTSC_RGB_OUT_14_( x, rgb_out, bits, shift ) {\ - snes_ntsc_rgb_t raw_ =\ - kernel0 [x ] + kernel1 [(x+12)%7+14] + kernel2 [(x+10)%7+28] +\ - kernelx0 [(x+7)%14] + kernelx1 [(x+ 5)%7+21] + kernelx2 [(x+ 3)%7+35];\ - SNES_NTSC_CLAMP_( raw_, shift );\ - SNES_NTSC_RGB_OUT_( rgb_out, bits, shift );\ -} - -/* common ntsc macros */ -#define snes_ntsc_rgb_builder ((1L << 21) | (1 << 11) | (1 << 1)) -#define snes_ntsc_clamp_mask (snes_ntsc_rgb_builder * 3 / 2) -#define snes_ntsc_clamp_add (snes_ntsc_rgb_builder * 0x101) -#define SNES_NTSC_CLAMP_( io, shift ) {\ - snes_ntsc_rgb_t sub = (io) >> (9-(shift)) & snes_ntsc_clamp_mask;\ - snes_ntsc_rgb_t clamp = snes_ntsc_clamp_add - sub;\ - io |= clamp;\ - clamp -= sub;\ - io &= clamp;\ -} - -#define SNES_NTSC_COLOR_IN_( index, color, ENTRY, table ) {\ - unsigned color_;\ - kernelx##index = kernel##index;\ - kernel##index = (color_ = (color), ENTRY( table, color_ ));\ -} - -/* x is always zero except in snes_ntsc library */ -/* original routine */ -/* -#define SNES_NTSC_RGB_OUT_( rgb_out, bits, x ) {\ - if ( bits == 16 )\ - rgb_out = (raw_>>(13-x)& 0xF800)|(raw_>>(8-x)&0x07E0)|(raw_>>(4-x)&0x001F);\ - if ( bits == 24 || bits == 32 )\ - rgb_out = (raw_>>(5-x)&0xFF0000)|(raw_>>(3-x)&0xFF00)|(raw_>>(1-x)&0xFF);\ - if ( bits == 15 )\ - rgb_out = (raw_>>(14-x)& 0x7C00)|(raw_>>(9-x)&0x03E0)|(raw_>>(4-x)&0x001F);\ - if ( bits == 14 )\ - rgb_out = (raw_>>(24-x)& 0x001F)|(raw_>>(9-x)&0x03E0)|(raw_<<(6+x)&0x7C00);\ - if ( bits == 0 )\ - rgb_out = raw_ << x;\ -} -*/ - -/* custom bsnes routine -- hooks into bsnes colortable */ -#define SNES_NTSC_RGB_OUT_( rgb_out, bits, x ) {\ - if ( bits == 16 ) {\ - rgb_out = (raw_>>(13-x)& 0xF800)|(raw_>>(8-x)&0x07E0)|(raw_>>(4-x)&0x001F);\ - rgb_out = ((rgb_out&0xf800)>>11)|((rgb_out&0x07c0)>>1)|((rgb_out&0x001f)<<10);\ - rgb_out = colortable[rgb_out];\ - } else if ( bits == 24 || bits == 32 ) {\ - rgb_out = (raw_>>(5-x)&0xFF0000)|(raw_>>(3-x)&0xFF00)|(raw_>>(1-x)&0xFF);\ - rgb_out = ((rgb_out&0xf80000)>>19)|((rgb_out&0x00f800)>>6)|((rgb_out&0x0000f8)<<7);\ - rgb_out = colortable[rgb_out];\ - } else if ( bits == 15 ) {\ - rgb_out = (raw_>>(14-x)& 0x7C00)|(raw_>>(9-x)&0x03E0)|(raw_>>(4-x)&0x001F);\ - rgb_out = ((rgb_out&0x7c00)>>10)|((rgb_out&0x03e0))|((rgb_out&0x001f)<<10);\ - rgb_out = colortable[rgb_out];\ - } else {\ - rgb_out = raw_ << x;\ - }\ -} - -#ifdef __cplusplus - } -#endif - -#endif diff --git a/snesfilter/NTSC/snes_ntsc/snes_ntsc_config.h b/snesfilter/NTSC/snes_ntsc/snes_ntsc_config.h deleted file mode 100755 index 7ab94c2c..00000000 --- a/snesfilter/NTSC/snes_ntsc/snes_ntsc_config.h +++ /dev/null @@ -1,26 +0,0 @@ -/* Configure library by modifying this file */ - -#ifndef SNES_NTSC_CONFIG_H -#define SNES_NTSC_CONFIG_H - -/* Format of source pixels */ -/* #define SNES_NTSC_IN_FORMAT SNES_NTSC_RGB16 */ -#define SNES_NTSC_IN_FORMAT SNES_NTSC_BGR15 - -/* The following affect the built-in blitter only; a custom blitter can -handle things however it wants. */ - -/* Bits per pixel of output. Can be 15, 16, 32, or 24 (same as 32). */ -#define SNES_NTSC_OUT_DEPTH 32 - -/* Type of input pixel values */ -#define SNES_NTSC_IN_T unsigned short - -/* Each raw pixel input value is passed through this. You might want to mask -the pixel index if you use the high bits as flags, etc. */ -#define SNES_NTSC_ADJ_IN( in ) in - -/* For each pixel, this is the basic operation: -output_color = SNES_NTSC_ADJ_IN( SNES_NTSC_IN_T ) */ - -#endif diff --git a/snesfilter/NTSC/snes_ntsc/snes_ntsc_impl.h b/snesfilter/NTSC/snes_ntsc/snes_ntsc_impl.h deleted file mode 100755 index 1d7adc78..00000000 --- a/snesfilter/NTSC/snes_ntsc/snes_ntsc_impl.h +++ /dev/null @@ -1,439 +0,0 @@ -/* snes_ntsc 0.2.2. http://www.slack.net/~ant/ */ - -/* Common implementation of NTSC filters */ - -#include -#include - -/* Copyright (C) 2006 Shay Green. This module is free software; you -can redistribute it and/or modify it under the terms of the GNU Lesser -General Public License as published by the Free Software Foundation; either -version 2.1 of the License, or (at your option) any later version. This -module 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 Lesser General Public License for more -details. You should have received a copy of the GNU Lesser General Public -License along with this module; if not, write to the Free Software Foundation, -Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */ - -#define DISABLE_CORRECTION 0 - -#undef PI -#define PI 3.14159265358979323846f - -#ifndef LUMA_CUTOFF - #define LUMA_CUTOFF 0.20 -#endif -#ifndef gamma_size - #define gamma_size 1 -#endif -#ifndef rgb_bits - #define rgb_bits 8 -#endif -#ifndef artifacts_max - #define artifacts_max (artifacts_mid * 1.5f) -#endif -#ifndef fringing_max - #define fringing_max (fringing_mid * 2) -#endif -#ifndef STD_HUE_CONDITION - #define STD_HUE_CONDITION( setup ) 1 -#endif - -#define ext_decoder_hue (std_decoder_hue + 15) -#define rgb_unit (1 << rgb_bits) -#define rgb_offset (rgb_unit * 2 + 0.5f) - -enum { burst_size = snes_ntsc_entry_size / burst_count }; -enum { kernel_half = 16 }; -enum { kernel_size = kernel_half * 2 + 1 }; - -typedef struct init_t -{ - float to_rgb [burst_count * 6]; - float to_float [gamma_size]; - float contrast; - float brightness; - float artifacts; - float fringing; - float kernel [rescale_out * kernel_size * 2]; -} init_t; - -#define ROTATE_IQ( i, q, sin_b, cos_b ) {\ - float t;\ - t = i * cos_b - q * sin_b;\ - q = i * sin_b + q * cos_b;\ - i = t;\ -} - -static void init_filters( init_t* impl, snes_ntsc_setup_t const* setup ) -{ -#if rescale_out > 1 - float kernels [kernel_size * 2]; -#else - float* const kernels = impl->kernel; -#endif - - /* generate luma (y) filter using sinc kernel */ - { - /* sinc with rolloff (dsf) */ - float const rolloff = 1 + (float) setup->sharpness * (float) 0.032; - float const maxh = 32; - float const pow_a_n = (float) pow( rolloff, maxh ); - float sum; - int i; - /* quadratic mapping to reduce negative (blurring) range */ - float to_angle = (float) setup->resolution + 1; - to_angle = PI / maxh * (float) LUMA_CUTOFF * (to_angle * to_angle + 1); - - kernels [kernel_size * 3 / 2] = maxh; /* default center value */ - for ( i = 0; i < kernel_half * 2 + 1; i++ ) - { - int x = i - kernel_half; - float angle = x * to_angle; - /* instability occurs at center point with rolloff very close to 1.0 */ - if ( x || pow_a_n > (float) 1.056 || pow_a_n < (float) 0.981 ) - { - float rolloff_cos_a = rolloff * (float) cos( angle ); - float num = 1 - rolloff_cos_a - - pow_a_n * (float) cos( maxh * angle ) + - pow_a_n * rolloff * (float) cos( (maxh - 1) * angle ); - float den = 1 - rolloff_cos_a - rolloff_cos_a + rolloff * rolloff; - float dsf = num / den; - kernels [kernel_size * 3 / 2 - kernel_half + i] = dsf - (float) 0.5; - } - } - - /* apply blackman window and find sum */ - sum = 0; - for ( i = 0; i < kernel_half * 2 + 1; i++ ) - { - float x = PI * 2 / (kernel_half * 2) * i; - float blackman = 0.42f - 0.5f * (float) cos( x ) + 0.08f * (float) cos( x * 2 ); - sum += (kernels [kernel_size * 3 / 2 - kernel_half + i] *= blackman); - } - - /* normalize kernel */ - sum = 1.0f / sum; - for ( i = 0; i < kernel_half * 2 + 1; i++ ) - { - int x = kernel_size * 3 / 2 - kernel_half + i; - kernels [x] *= sum; - assert( kernels [x] == kernels [x] ); /* catch numerical instability */ - } - } - - /* generate chroma (iq) filter using gaussian kernel */ - { - float const cutoff_factor = -0.03125f; - float cutoff = (float) setup->bleed; - int i; - - if ( cutoff < 0 ) - { - /* keep extreme value accessible only near upper end of scale (1.0) */ - cutoff *= cutoff; - cutoff *= cutoff; - cutoff *= cutoff; - cutoff *= -30.0f / 0.65f; - } - cutoff = cutoff_factor - 0.65f * cutoff_factor * cutoff; - - for ( i = -kernel_half; i <= kernel_half; i++ ) - kernels [kernel_size / 2 + i] = (float) exp( i * i * cutoff ); - - /* normalize even and odd phases separately */ - for ( i = 0; i < 2; i++ ) - { - float sum = 0; - int x; - for ( x = i; x < kernel_size; x += 2 ) - sum += kernels [x]; - - sum = 1.0f / sum; - for ( x = i; x < kernel_size; x += 2 ) - { - kernels [x] *= sum; - assert( kernels [x] == kernels [x] ); /* catch numerical instability */ - } - } - } - - /* - printf( "luma:\n" ); - for ( i = kernel_size; i < kernel_size * 2; i++ ) - printf( "%f\n", kernels [i] ); - printf( "chroma:\n" ); - for ( i = 0; i < kernel_size; i++ ) - printf( "%f\n", kernels [i] ); - */ - - /* generate linear rescale kernels */ - #if rescale_out > 1 - { - float weight = 1.0f; - float* out = impl->kernel; - int n = rescale_out; - do - { - float remain = 0; - int i; - weight -= 1.0f / rescale_in; - for ( i = 0; i < kernel_size * 2; i++ ) - { - float cur = kernels [i]; - float m = cur * weight; - *out++ = m + remain; - remain = cur - m; - } - } - while ( --n ); - } - #endif -} - -static float const default_decoder [6] = - { 0.956f, 0.621f, -0.272f, -0.647f, -1.105f, 1.702f }; - -static void init( init_t* impl, snes_ntsc_setup_t const* setup ) -{ - impl->brightness = (float) setup->brightness * (0.5f * rgb_unit) + rgb_offset; - impl->contrast = (float) setup->contrast * (0.5f * rgb_unit) + rgb_unit; - #ifdef default_palette_contrast - if ( !setup->palette ) - impl->contrast *= default_palette_contrast; - #endif - - impl->artifacts = (float) setup->artifacts; - if ( impl->artifacts > 0 ) - impl->artifacts *= artifacts_max - artifacts_mid; - impl->artifacts = impl->artifacts * artifacts_mid + artifacts_mid; - - impl->fringing = (float) setup->fringing; - if ( impl->fringing > 0 ) - impl->fringing *= fringing_max - fringing_mid; - impl->fringing = impl->fringing * fringing_mid + fringing_mid; - - init_filters( impl, setup ); - - /* generate gamma table */ - if ( gamma_size > 1 ) - { - float const to_float = 1.0f / (gamma_size - (gamma_size > 1)); - float const gamma = 1.1333f - (float) setup->gamma * 0.5f; - /* match common PC's 2.2 gamma to TV's 2.65 gamma */ - int i; - for ( i = 0; i < gamma_size; i++ ) - impl->to_float [i] = - (float) pow( i * to_float, gamma ) * impl->contrast + impl->brightness; - } - - /* setup decoder matricies */ - { - float hue = (float) setup->hue * PI + PI / 180 * ext_decoder_hue; - float sat = (float) setup->saturation + 1; - float const* decoder = setup->decoder_matrix; - if ( !decoder ) - { - decoder = default_decoder; - if ( STD_HUE_CONDITION( setup ) ) - hue += PI / 180 * (std_decoder_hue - ext_decoder_hue); - } - - { - float s = (float) sin( hue ) * sat; - float c = (float) cos( hue ) * sat; - float* out = impl->to_rgb; - int n; - - n = burst_count; - do - { - float const* in = decoder; - int n = 3; - do - { - float i = *in++; - float q = *in++; - *out++ = i * c - q * s; - *out++ = i * s + q * c; - } - while ( --n ); - if ( burst_count <= 1 ) - break; - ROTATE_IQ( s, c, 0.866025f, -0.5f ); /* +120 degrees */ - } - while ( --n ); - } - } -} - -/* kernel generation */ - -#define RGB_TO_YIQ( r, g, b, y, i ) (\ - (y = (r) * 0.299f + (g) * 0.587f + (b) * 0.114f),\ - (i = (r) * 0.596f - (g) * 0.275f - (b) * 0.321f),\ - ((r) * 0.212f - (g) * 0.523f + (b) * 0.311f)\ -) - -#define YIQ_TO_RGB( y, i, q, to_rgb, type, r, g ) (\ - r = (type) (y + to_rgb [0] * i + to_rgb [1] * q),\ - g = (type) (y + to_rgb [2] * i + to_rgb [3] * q),\ - (type) (y + to_rgb [4] * i + to_rgb [5] * q)\ -) - -#define PACK_RGB( r, g, b ) ((r) << 21 | (g) << 11 | (b) << 1) - -enum { rgb_kernel_size = burst_size / alignment_count }; -enum { rgb_bias = rgb_unit * 2 * snes_ntsc_rgb_builder }; - -typedef struct pixel_info_t -{ - int offset; - float negate; - float kernel [4]; -} pixel_info_t; - -#if rescale_in > 1 - #define PIXEL_OFFSET_( ntsc, scaled ) \ - (kernel_size / 2 + ntsc + (scaled != 0) + (rescale_out - scaled) % rescale_out + \ - (kernel_size * 2 * scaled)) - - #define PIXEL_OFFSET( ntsc, scaled ) \ - PIXEL_OFFSET_( ((ntsc) - (scaled) / rescale_out * rescale_in),\ - (((scaled) + rescale_out * 10) % rescale_out) ),\ - (1.0f - (((ntsc) + 100) & 2)) -#else - #define PIXEL_OFFSET( ntsc, scaled ) \ - (kernel_size / 2 + (ntsc) - (scaled)),\ - (1.0f - (((ntsc) + 100) & 2)) -#endif - -extern pixel_info_t const snes_ntsc_pixels [alignment_count]; - -/* Generate pixel at all burst phases and column alignments */ -static void gen_kernel( init_t* impl, float y, float i, float q, snes_ntsc_rgb_t* out ) -{ - /* generate for each scanline burst phase */ - float const* to_rgb = impl->to_rgb; - int burst_remain = burst_count; - y -= rgb_offset; - do - { - /* Encode yiq into *two* composite signals (to allow control over artifacting). - Convolve these with kernels which: filter respective components, apply - sharpening, and rescale horizontally. Convert resulting yiq to rgb and pack - into integer. Based on algorithm by NewRisingSun. */ - pixel_info_t const* pixel = snes_ntsc_pixels; - int alignment_remain = alignment_count; - do - { - /* negate is -1 when composite starts at odd multiple of 2 */ - float const yy = y * impl->fringing * pixel->negate; - float const ic0 = (i + yy) * pixel->kernel [0]; - float const qc1 = (q + yy) * pixel->kernel [1]; - float const ic2 = (i - yy) * pixel->kernel [2]; - float const qc3 = (q - yy) * pixel->kernel [3]; - - float const factor = impl->artifacts * pixel->negate; - float const ii = i * factor; - float const yc0 = (y + ii) * pixel->kernel [0]; - float const yc2 = (y - ii) * pixel->kernel [2]; - - float const qq = q * factor; - float const yc1 = (y + qq) * pixel->kernel [1]; - float const yc3 = (y - qq) * pixel->kernel [3]; - - float const* k = &impl->kernel [pixel->offset]; - int n; - ++pixel; - for ( n = rgb_kernel_size; n; --n ) - { - float i = k[0]*ic0 + k[2]*ic2; - float q = k[1]*qc1 + k[3]*qc3; - float y = k[kernel_size+0]*yc0 + k[kernel_size+1]*yc1 + - k[kernel_size+2]*yc2 + k[kernel_size+3]*yc3 + rgb_offset; - if ( rescale_out <= 1 ) - k--; - else if ( k < &impl->kernel [kernel_size * 2 * (rescale_out - 1)] ) - k += kernel_size * 2 - 1; - else - k -= kernel_size * 2 * (rescale_out - 1) + 2; - { - int r, g, b = YIQ_TO_RGB( y, i, q, to_rgb, int, r, g ); - *out++ = PACK_RGB( r, g, b ) - rgb_bias; - } - } - } - while ( alignment_count > 1 && --alignment_remain ); - - if ( burst_count <= 1 ) - break; - - to_rgb += 6; - - ROTATE_IQ( i, q, -0.866025f, -0.5f ); /* -120 degrees */ - } - while ( --burst_remain ); -} - -static void correct_errors( snes_ntsc_rgb_t color, snes_ntsc_rgb_t* out ); - -#if DISABLE_CORRECTION - #define CORRECT_ERROR( a ) { out [i] += rgb_bias; } - #define DISTRIBUTE_ERROR( a, b, c ) { out [i] += rgb_bias; } -#else - #define CORRECT_ERROR( a ) { out [a] += error; } - #define DISTRIBUTE_ERROR( a, b, c ) {\ - snes_ntsc_rgb_t fourth = (error + 2 * snes_ntsc_rgb_builder) >> 2;\ - fourth &= (rgb_bias >> 1) - snes_ntsc_rgb_builder;\ - fourth -= rgb_bias >> 2;\ - out [a] += fourth;\ - out [b] += fourth;\ - out [c] += fourth;\ - out [i] += error - (fourth * 3);\ - } -#endif - -#define RGB_PALETTE_OUT( rgb, out_ )\ -{\ - unsigned char* out = (out_);\ - snes_ntsc_rgb_t clamped = (rgb);\ - SNES_NTSC_CLAMP_( clamped, (8 - rgb_bits) );\ - out [0] = (unsigned char) (clamped >> 21);\ - out [1] = (unsigned char) (clamped >> 11);\ - out [2] = (unsigned char) (clamped >> 1);\ -} - -/* blitter related */ - -#ifndef restrict - #if defined (__GNUC__) - #define restrict __restrict__ - #elif defined (_MSC_VER) && _MSC_VER > 1300 - #define restrict __restrict - #else - /* no support for restricted pointers */ - #define restrict - #endif -#endif - -#include - -#if SNES_NTSC_OUT_DEPTH <= 16 - #if USHRT_MAX == 0xFFFF - typedef unsigned short snes_ntsc_out_t; - #else - #error "Need 16-bit int type" - #endif - -#else - #if UINT_MAX == 0xFFFFFFFF - typedef unsigned int snes_ntsc_out_t; - #elif ULONG_MAX == 0xFFFFFFFF - typedef unsigned long snes_ntsc_out_t; - #else - #error "Need 32-bit int type" - #endif - -#endif diff --git a/snesfilter/Pixellate2x/Pixellate2x.cpp b/snesfilter/Pixellate2x/Pixellate2x.cpp index f57c5c39..c15a3b67 100755 --- a/snesfilter/Pixellate2x/Pixellate2x.cpp +++ b/snesfilter/Pixellate2x/Pixellate2x.cpp @@ -4,42 +4,33 @@ using namespace nall; extern "C" { void filter_size(unsigned&, unsigned&); - void filter_render(uint32_t*, uint32_t*, unsigned, const uint16_t*, unsigned, unsigned, unsigned); + void filter_render(uint16_t*, unsigned, const uint16_t*, unsigned, unsigned, unsigned); }; dllexport void filter_size(unsigned &width, unsigned &height) { - width = (width <= 256) ? width * 2 : width; - height = (height <= 240) ? height * 2 : height; + width *= 2; + height *= 2; } dllexport void filter_render( - uint32_t *colortable, uint32_t *output, unsigned outpitch, - const uint16_t *input, unsigned pitch, unsigned width, unsigned height + uint16_t *output, unsigned outputPitch, + const uint16_t *input, unsigned inputPitch, + unsigned width, unsigned height ) { - pitch >>= 1; - outpitch >>= 2; - - uint32_t *out0 = output; - uint32_t *out1 = output + outpitch; + outputPitch >>= 1, inputPitch >>= 1; + #pragma omp parallel for for(unsigned y = 0; y < height; y++) { + const uint16_t *in = input + y * inputPitch; + uint16_t *out0 = output + y * outputPitch * 2; + uint16_t *out1 = output + y * outputPitch * 2 + outputPitch; + for(unsigned x = 0; x < width; x++) { - uint32_t p = colortable[*input++]; - - *out0++ = p; - if(height <= 240) *out1++ = p; - if(width > 256) continue; - - *out0++ = p; - if(height <= 240) *out1++ = p; - } - - input += pitch - width; - if(height <= 240) { - out0 += outpitch + outpitch - 512; - out1 += outpitch + outpitch - 512; - } else { - out0 += outpitch - 512; + uint16_t pixel = *in++; + *out0++ = pixel; + *out0++ = pixel; + *out1++ = pixel; + *out1++ = pixel; } } } diff --git a/snesfilter/Scale2x/Scale2x.cpp b/snesfilter/Scale2x/Scale2x.cpp index 16f8231f..a98f08d4 100755 --- a/snesfilter/Scale2x/Scale2x.cpp +++ b/snesfilter/Scale2x/Scale2x.cpp @@ -4,7 +4,7 @@ using namespace nall; extern "C" { void filter_size(unsigned&, unsigned&); - void filter_render(uint32_t*, uint32_t*, unsigned, const uint16_t*, unsigned, unsigned, unsigned); + void filter_render(uint16_t*, unsigned, const uint16_t*, unsigned, unsigned, unsigned); }; dllexport void filter_size(unsigned &width, unsigned &height) { @@ -13,18 +13,20 @@ dllexport void filter_size(unsigned &width, unsigned &height) { } dllexport void filter_render( - uint32_t *colortable, uint32_t *output, unsigned outpitch, - const uint16_t *input, unsigned pitch, unsigned width, unsigned height + uint16_t *output, unsigned outputPitch, + const uint16_t *input, unsigned inputPitch, + unsigned width, unsigned height ) { - pitch >>= 1; - outpitch >>= 2; - - uint32_t *out0 = output; - uint32_t *out1 = output + outpitch; + outputPitch >>= 1, inputPitch >>= 1; + #pragma omp parallel for for(unsigned y = 0; y < height; y++) { - int prevline = (y == 0 ? 0 : pitch); - int nextline = (y == height - 1 ? 0 : pitch); + const uint16_t *in = input + y * inputPitch; + uint16_t *out0 = output + y * outputPitch * 2; + uint16_t *out1 = output + y * outputPitch * 2 + outputPitch; + + int prevline = (y == 0 ? 0 : inputPitch); + int nextline = (y == height - 1 ? 0 : inputPitch); for(unsigned x = 0; x < width; x++) { uint16_t A = *(input - prevline); @@ -32,23 +34,18 @@ dllexport void filter_render( uint16_t C = *input; uint16_t D = (x < width - 1) ? *(input + 1) : *input; uint16_t E = *(input++ + nextline); - uint32_t c = colortable[C]; if(A != E && B != D) { - *out0++ = (A == B ? colortable[A] : c); - *out0++ = (A == D ? colortable[A] : c); - *out1++ = (E == B ? colortable[E] : c); - *out1++ = (E == D ? colortable[E] : c); + *out0++ = (A == B ? A : C); + *out0++ = (A == D ? A : C); + *out1++ = (E == B ? E : C); + *out1++ = (E == D ? E : C); } else { - *out0++ = c; - *out0++ = c; - *out1++ = c; - *out1++ = c; + *out0++ = C; + *out0++ = C; + *out1++ = C; + *out1++ = C; } } - - input += pitch - width; - out0 += outpitch + outpitch - (width << 1); - out1 += outpitch + outpitch - (width << 1); } } diff --git a/snesfilter/Scanline/Scanline-Black.cpp b/snesfilter/Scanline/Scanline-Black.cpp deleted file mode 100755 index 7b9d6265..00000000 --- a/snesfilter/Scanline/Scanline-Black.cpp +++ /dev/null @@ -1,34 +0,0 @@ -#include -#include -using namespace nall; - -extern "C" { - void filter_size(unsigned&, unsigned&); - void filter_render(uint32_t*, uint32_t*, unsigned, const uint16_t*, unsigned, unsigned, unsigned); -}; - -dllexport void filter_size(unsigned &width, unsigned &height) { - height *= 2; -} - -dllexport void filter_render( - uint32_t *palette, uint32_t *output, unsigned outpitch, - const uint16_t *input, unsigned pitch, unsigned width, unsigned height -) { - pitch >>= 1; - outpitch >>= 2; - - uint32_t *out0 = output; - uint32_t *out1 = output + outpitch; - - for(unsigned y = 0; y < height; y++) { - const uint16_t *in = input + y * pitch; - uint32_t *out0 = output + y * outpitch * 2; - uint32_t *out1 = output + y * outpitch * 2 + outpitch; - - for(unsigned x = 0; x < width; x++) { - *out0++ = palette[*in++]; - *out1++ = 0; - } - } -} diff --git a/snesfilter/Scanline/Scanline-Dark.cpp b/snesfilter/Scanline/Scanline-Dark.cpp index 9d9c9466..68c1f72c 100755 --- a/snesfilter/Scanline/Scanline-Dark.cpp +++ b/snesfilter/Scanline/Scanline-Dark.cpp @@ -4,52 +4,29 @@ using namespace nall; extern "C" { void filter_size(unsigned&, unsigned&); - void filter_render(uint32_t*, uint32_t*, unsigned, const uint16_t*, unsigned, unsigned, unsigned); + void filter_render(uint16_t*, unsigned, const uint16_t*, unsigned, unsigned, unsigned); }; -uint16_t adjust[32768]; - -void initialize() { - static bool initialized = false; - if(initialized == true) return; - initialized = true; - - for(unsigned i = 0; i < 32768; i++) { - uint8_t r = (i >> 10) & 31; - uint8_t g = (i >> 5) & 31; - uint8_t b = (i >> 0) & 31; - r *= 0.333; - g *= 0.333; - b *= 0.333; - adjust[i] = (r << 10) + (g << 5) + (b << 0); - } -} - dllexport void filter_size(unsigned &width, unsigned &height) { - initialize(); height *= 2; } dllexport void filter_render( - uint32_t *palette, uint32_t *output, unsigned outpitch, - const uint16_t *input, unsigned pitch, unsigned width, unsigned height + uint16_t *output, unsigned outputPitch, + const uint16_t *input, unsigned inputPitch, + unsigned width, unsigned height ) { - initialize(); - pitch >>= 1; - outpitch >>= 2; - - uint32_t *out0 = output; - uint32_t *out1 = output + outpitch; + outputPitch >>= 1, inputPitch >>= 1; + #pragma omp parallel for for(unsigned y = 0; y < height; y++) { - const uint16_t *in = input + y * pitch; - uint32_t *out0 = output + y * outpitch * 2; - uint32_t *out1 = output + y * outpitch * 2 + outpitch; + const uint16_t *in = input + y * inputPitch; + uint16_t *out0 = output + y * outputPitch * 2; + uint16_t *out1 = output + y * outputPitch * 2 + outputPitch; for(unsigned x = 0; x < width; x++) { - uint16_t color = *in++; - *out0++ = palette[color]; - *out1++ = palette[adjust[color]]; + *out0++ = *in++; + *out1++ = 0; } } } diff --git a/snesfilter/Scanline/Scanline-Light.cpp b/snesfilter/Scanline/Scanline-Light.cpp index b66891e1..6662a9c7 100755 --- a/snesfilter/Scanline/Scanline-Light.cpp +++ b/snesfilter/Scanline/Scanline-Light.cpp @@ -4,52 +4,29 @@ using namespace nall; extern "C" { void filter_size(unsigned&, unsigned&); - void filter_render(uint32_t*, uint32_t*, unsigned, const uint16_t*, unsigned, unsigned, unsigned); + void filter_render(uint16_t*, unsigned, const uint16_t*, unsigned, unsigned, unsigned); }; -uint16_t adjust[32768]; - -void initialize() { - static bool initialized = false; - if(initialized == true) return; - initialized = true; - - for(unsigned i = 0; i < 32768; i++) { - uint8_t r = (i >> 10) & 31; - uint8_t g = (i >> 5) & 31; - uint8_t b = (i >> 0) & 31; - r *= 0.666; - g *= 0.666; - b *= 0.666; - adjust[i] = (r << 10) + (g << 5) + (b << 0); - } -} - dllexport void filter_size(unsigned &width, unsigned &height) { - initialize(); height *= 2; } dllexport void filter_render( - uint32_t *palette, uint32_t *output, unsigned outpitch, - const uint16_t *input, unsigned pitch, unsigned width, unsigned height + uint16_t *output, unsigned outputPitch, + const uint16_t *input, unsigned inputPitch, + unsigned width, unsigned height ) { - initialize(); - pitch >>= 1; - outpitch >>= 2; - - uint32_t *out0 = output; - uint32_t *out1 = output + outpitch; + outputPitch >>= 1, inputPitch >>= 1; + #pragma omp parallel for for(unsigned y = 0; y < height; y++) { - const uint16_t *in = input + y * pitch; - uint32_t *out0 = output + y * outpitch * 2; - uint32_t *out1 = output + y * outpitch * 2 + outpitch; + const uint16_t *in = input + y * inputPitch; + uint16_t *out0 = output + y * outputPitch * 2; + uint16_t *out1 = output + y * outputPitch * 2 + outputPitch; for(unsigned x = 0; x < width; x++) { - uint16_t color = *in++; - *out0++ = palette[color]; - *out1++ = palette[adjust[color]]; + *out0++ = *in; + *out1++ = (*in++ & 0x7bde) >> 1; } } } diff --git a/snesfilter/nall/array.hpp b/snesfilter/nall/array.hpp index 9cfe7758..e1fb1fcb 100755 --- a/snesfilter/nall/array.hpp +++ b/snesfilter/nall/array.hpp @@ -54,6 +54,10 @@ namespace nall { operator[](buffersize) = data; } + void remove() { + if(size > 0) resize(size - 1); //remove last element only + } + template void insert(unsigned index, const U list) { unsigned listsize = container_size(list); resize(buffersize + listsize); @@ -133,6 +137,12 @@ namespace nall { if(index >= buffersize) throw "array[] out of bounds"; return pool[index]; } + + //iteration + T* begin() { return &pool[0]; } + T* end() { return &pool[buffersize]; } + const T* begin() const { return &pool[0]; } + const T* end() const { return &pool[buffersize]; } }; template struct has_size> { enum { value = true }; }; diff --git a/snesfilter/nall/base64.hpp b/snesfilter/nall/base64.hpp index e41c87b7..ee59c1be 100755 --- a/snesfilter/nall/base64.hpp +++ b/snesfilter/nall/base64.hpp @@ -72,6 +72,7 @@ namespace nall { private: static char enc(uint8_t n) { + //base64 for URL encodings static char lookup_table[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_"; return lookup_table[n & 63]; } diff --git a/snesfilter/nall/bmp.hpp b/snesfilter/nall/bmp.hpp new file mode 100755 index 00000000..33cdf4dc --- /dev/null +++ b/snesfilter/nall/bmp.hpp @@ -0,0 +1,101 @@ +#ifndef NALL_BMP_HPP +#define NALL_BMP_HPP + +#include + +//BMP reader / writer +//author: byuu +//note: only 24-bit RGB and 32-bit ARGB uncompressed images supported + +namespace nall { + +struct bmp { + inline static bool read(const string &filename, uint32_t *&data, unsigned &width, unsigned &height); + inline static bool write(const string &filename, const uint32_t *data, unsigned width, unsigned height, unsigned pitch, bool alpha = false); +}; + +bool bmp::read(const string &filename, uint32_t *&data, unsigned &width, unsigned &height) { + file fp; + if(fp.open(filename, file::mode::read) == false) return false; + if(fp.size() < 0x36) return false; + + if(fp.readm(2) != 0x424d) return false; + fp.seek(0x000a); + unsigned offset = fp.readl(4); + unsigned dibsize = fp.readl(4); + if(dibsize != 40) return false; + signed headerWidth = fp.readl(4); + if(headerWidth < 0) return false; + signed headerHeight = fp.readl(4); + fp.readl(2); + unsigned bitsPerPixel = fp.readl(2); + if(bitsPerPixel != 24 && bitsPerPixel != 32) return false; + unsigned compression = fp.readl(4); + if(compression != 0) return false; + fp.seek(offset); + + bool noFlip = headerHeight < 0; + width = headerWidth, height = abs(headerHeight); + data = new uint32_t[width * height]; + + unsigned bytesPerPixel = bitsPerPixel / 8; + unsigned alignedWidth = width * bytesPerPixel; + unsigned paddingLength = 0; + while(alignedWidth % 4) alignedWidth++, paddingLength++; + + for(unsigned y = 0; y < height; y++) { + uint32_t *p = noFlip ? data + y * width : data + (height - 1 - y) * width; + for(unsigned x = 0; x < width; x++, p++) { + *p = fp.readl(bytesPerPixel); + if(bytesPerPixel == 3) *p |= 255 << 24; + } + if(paddingLength) fp.readl(paddingLength); + } + + fp.close(); + return true; +} + +bool bmp::write(const string &filename, const uint32_t *data, unsigned width, unsigned height, unsigned pitch, bool alpha) { + file fp; + if(fp.open(filename, file::mode::write) == false) return false; + + unsigned bitsPerPixel = alpha ? 32 : 24; + unsigned bytesPerPixel = bitsPerPixel / 8; + unsigned alignedWidth = width * bytesPerPixel; + unsigned paddingLength = 0; + unsigned imageSize = alignedWidth * height; + unsigned fileSize = 0x36 + imageSize; + while(alignedWidth % 4) alignedWidth++, paddingLength++; + + fp.writem(0x424d, 2); //signature + fp.writel(fileSize, 4); //file size + fp.writel(0, 2); //reserved + fp.writel(0, 2); //reserved + fp.writel(0x36, 4); //offset + + fp.writel(40, 4); //DIB size + fp.writel(width, 4); //width + fp.writel(-height, 4); //height + fp.writel(1, 2); //color planes + fp.writel(bitsPerPixel, 2); //bits per pixel + fp.writel(0, 4); //compression method (BI_RGB) + fp.writel(imageSize, 4); //image data size + fp.writel(3780, 4); //horizontal resolution + fp.writel(3780, 4); //vertical resolution + fp.writel(0, 4); //palette size + fp.writel(0, 4); //important color count + + for(unsigned y = 0; y < height; y++) { + const uint32_t *p = (const uint32_t*)((const uint8_t*)data + y * pitch); + for(unsigned x = 0; x < width; x++) fp.writel(*p++, bytesPerPixel); + if(paddingLength) fp.writel(0, paddingLength); + } + + fp.close(); + return true; +} + +} + +#endif diff --git a/snesfilter/nall/bps/delta.hpp b/snesfilter/nall/bps/delta.hpp new file mode 100755 index 00000000..a3af047c --- /dev/null +++ b/snesfilter/nall/bps/delta.hpp @@ -0,0 +1,214 @@ +#ifndef NALL_BPS_DELTA_HPP +#define NALL_BPS_DELTA_HPP + +#include +#include +#include +#include +#include + +namespace nall { + +struct bpsdelta { + inline void source(const uint8_t *data, unsigned size); + inline void target(const uint8_t *data, unsigned size); + + inline bool source(const string &filename); + inline bool target(const string &filename); + inline bool create(const string &filename, const string &metadata = ""); + +protected: + enum : unsigned { SourceRead, TargetRead, SourceCopy, TargetCopy }; + enum : unsigned { Granularity = 1 }; + + struct Node { + unsigned offset; + Node *next; + inline Node() : offset(0), next(0) {} + inline ~Node() { if(next) delete next; } + }; + + filemap sourceFile; + const uint8_t *sourceData; + unsigned sourceSize; + + filemap targetFile; + const uint8_t *targetData; + unsigned targetSize; +}; + +void bpsdelta::source(const uint8_t *data, unsigned size) { + sourceData = data; + sourceSize = size; +} + +void bpsdelta::target(const uint8_t *data, unsigned size) { + targetData = data; + targetSize = size; +} + +bool bpsdelta::source(const string &filename) { + if(sourceFile.open(filename, filemap::mode::read) == false) return false; + source(sourceFile.data(), sourceFile.size()); + return true; +} + +bool bpsdelta::target(const string &filename) { + if(targetFile.open(filename, filemap::mode::read) == false) return false; + target(targetFile.data(), targetFile.size()); + return true; +} + +bool bpsdelta::create(const string &filename, const string &metadata) { + file modifyFile; + if(modifyFile.open(filename, file::mode::write) == false) return false; + + uint32_t sourceChecksum = ~0, modifyChecksum = ~0; + unsigned sourceRelativeOffset = 0, targetRelativeOffset = 0, outputOffset = 0; + + auto write = [&](uint8_t data) { + modifyFile.write(data); + modifyChecksum = crc32_adjust(modifyChecksum, data); + }; + + auto encode = [&](uint64_t data) { + while(true) { + uint64_t x = data & 0x7f; + data >>= 7; + if(data == 0) { + write(0x80 | x); + break; + } + write(x); + data--; + } + }; + + write('B'); + write('P'); + write('S'); + write('1'); + + encode(sourceSize); + encode(targetSize); + + unsigned markupSize = metadata.length(); + encode(markupSize); + for(unsigned n = 0; n < markupSize; n++) write(metadata[n]); + + Node *sourceTree[65536], *targetTree[65536]; + for(unsigned n = 0; n < 65536; n++) sourceTree[n] = 0, targetTree[n] = 0; + + //source tree creation + for(unsigned offset = 0; offset < sourceSize; offset++) { + uint16_t symbol = sourceData[offset + 0]; + sourceChecksum = crc32_adjust(sourceChecksum, symbol); + if(offset < sourceSize - 1) symbol |= sourceData[offset + 1] << 8; + Node *node = new Node; + node->offset = offset; + node->next = sourceTree[symbol]; + sourceTree[symbol] = node; + } + + unsigned targetReadLength = 0; + + auto targetReadFlush = [&]() { + if(targetReadLength) { + encode(TargetRead | ((targetReadLength - 1) << 2)); + unsigned offset = outputOffset - targetReadLength; + while(targetReadLength) write(targetData[offset++]), targetReadLength--; + } + }; + + while(outputOffset < targetSize) { + unsigned maxLength = 0, maxOffset = 0, mode = TargetRead; + + uint16_t symbol = targetData[outputOffset + 0]; + if(outputOffset < targetSize - 1) symbol |= targetData[outputOffset + 1] << 8; + + { //source read + unsigned length = 0, offset = outputOffset; + while(offset < sourceSize && offset < targetSize && sourceData[offset] == targetData[offset]) { + length++; + offset++; + } + if(length > maxLength) maxLength = length, mode = SourceRead; + } + + { //source copy + Node *node = sourceTree[symbol]; + while(node) { + unsigned length = 0, x = node->offset, y = outputOffset; + while(x < sourceSize && y < targetSize && sourceData[x++] == targetData[y++]) length++; + if(length > maxLength) maxLength = length, maxOffset = node->offset, mode = SourceCopy; + node = node->next; + } + } + + { //target copy + Node *node = targetTree[symbol]; + while(node) { + unsigned length = 0, x = node->offset, y = outputOffset; + while(y < targetSize && targetData[x++] == targetData[y++]) length++; + if(length > maxLength) maxLength = length, maxOffset = node->offset, mode = TargetCopy; + node = node->next; + } + + //target tree append + node = new Node; + node->offset = outputOffset; + node->next = targetTree[symbol]; + targetTree[symbol] = node; + } + + { //target read + if(maxLength < 4) { + maxLength = min((unsigned)Granularity, targetSize - outputOffset); + mode = TargetRead; + } + } + + if(mode != TargetRead) targetReadFlush(); + + switch(mode) { + case SourceRead: + encode(SourceRead | ((maxLength - 1) << 2)); + break; + case TargetRead: + //delay write to group sequential TargetRead commands into one + targetReadLength += maxLength; + break; + case SourceCopy: + case TargetCopy: + encode(mode | ((maxLength - 1) << 2)); + signed relativeOffset; + if(mode == SourceCopy) { + relativeOffset = maxOffset - sourceRelativeOffset; + sourceRelativeOffset = maxOffset + maxLength; + } else { + relativeOffset = maxOffset - targetRelativeOffset; + targetRelativeOffset = maxOffset + maxLength; + } + encode((relativeOffset < 0) | (abs(relativeOffset) << 1)); + break; + } + + outputOffset += maxLength; + } + + targetReadFlush(); + + sourceChecksum = ~sourceChecksum; + for(unsigned n = 0; n < 32; n += 8) write(sourceChecksum >> n); + uint32_t targetChecksum = crc32_calculate(targetData, targetSize); + for(unsigned n = 0; n < 32; n += 8) write(targetChecksum >> n); + uint32_t outputChecksum = ~modifyChecksum; + for(unsigned n = 0; n < 32; n += 8) write(outputChecksum >> n); + + modifyFile.close(); + return true; +} + +} + +#endif diff --git a/snesfilter/nall/bps/linear.hpp b/snesfilter/nall/bps/linear.hpp new file mode 100755 index 00000000..df840283 --- /dev/null +++ b/snesfilter/nall/bps/linear.hpp @@ -0,0 +1,152 @@ +#ifndef NALL_BPS_LINEAR_HPP +#define NALL_BPS_LINEAR_HPP + +#include +#include +#include +#include +#include + +namespace nall { + +struct bpslinear { + inline void source(const uint8_t *data, unsigned size); + inline void target(const uint8_t *data, unsigned size); + + inline bool source(const string &filename); + inline bool target(const string &filename); + inline bool create(const string &filename, const string &metadata = ""); + +protected: + enum : unsigned { SourceRead, TargetRead, SourceCopy, TargetCopy }; + enum : unsigned { Granularity = 1 }; + + filemap sourceFile; + const uint8_t *sourceData; + unsigned sourceSize; + + filemap targetFile; + const uint8_t *targetData; + unsigned targetSize; +}; + +void bpslinear::source(const uint8_t *data, unsigned size) { + sourceData = data; + sourceSize = size; +} + +void bpslinear::target(const uint8_t *data, unsigned size) { + targetData = data; + targetSize = size; +} + +bool bpslinear::source(const string &filename) { + if(sourceFile.open(filename, filemap::mode::read) == false) return false; + source(sourceFile.data(), sourceFile.size()); + return true; +} + +bool bpslinear::target(const string &filename) { + if(targetFile.open(filename, filemap::mode::read) == false) return false; + target(targetFile.data(), targetFile.size()); + return true; +} + +bool bpslinear::create(const string &filename, const string &metadata) { + file modifyFile; + if(modifyFile.open(filename, file::mode::write) == false) return false; + + uint32_t modifyChecksum = ~0; + unsigned targetRelativeOffset = 0, outputOffset = 0; + + auto write = [&](uint8_t data) { + modifyFile.write(data); + modifyChecksum = crc32_adjust(modifyChecksum, data); + }; + + auto encode = [&](uint64_t data) { + while(true) { + uint64_t x = data & 0x7f; + data >>= 7; + if(data == 0) { + write(0x80 | x); + break; + } + write(x); + data--; + } + }; + + unsigned targetReadLength = 0; + + auto targetReadFlush = [&]() { + if(targetReadLength) { + encode(TargetRead | ((targetReadLength - 1) << 2)); + unsigned offset = outputOffset - targetReadLength; + while(targetReadLength) write(targetData[offset++]), targetReadLength--; + } + }; + + write('B'); + write('P'); + write('S'); + write('1'); + + encode(sourceSize); + encode(targetSize); + + unsigned markupSize = metadata.length(); + encode(markupSize); + for(unsigned n = 0; n < markupSize; n++) write(metadata[n]); + + while(outputOffset < targetSize) { + unsigned sourceLength = 0; + for(unsigned n = 0; outputOffset + n < min(sourceSize, targetSize); n++) { + if(sourceData[outputOffset + n] != targetData[outputOffset + n]) break; + sourceLength++; + } + + unsigned rleLength = 0; + for(unsigned n = 1; outputOffset + n < targetSize; n++) { + if(targetData[outputOffset] != targetData[outputOffset + n]) break; + rleLength++; + } + + if(rleLength >= 4) { + //write byte to repeat + targetReadLength++; + outputOffset++; + targetReadFlush(); + + //copy starting from repetition byte + encode(TargetCopy | ((rleLength - 1) << 2)); + unsigned relativeOffset = (outputOffset - 1) - targetRelativeOffset; + encode(relativeOffset << 1); + outputOffset += rleLength; + targetRelativeOffset = outputOffset - 1; + } else if(sourceLength >= 4) { + targetReadFlush(); + encode(SourceRead | ((sourceLength - 1) << 2)); + outputOffset += sourceLength; + } else { + targetReadLength += Granularity; + outputOffset += Granularity; + } + } + + targetReadFlush(); + + uint32_t sourceChecksum = crc32_calculate(sourceData, sourceSize); + for(unsigned n = 0; n < 32; n += 8) write(sourceChecksum >> n); + uint32_t targetChecksum = crc32_calculate(targetData, targetSize); + for(unsigned n = 0; n < 32; n += 8) write(targetChecksum >> n); + uint32_t outputChecksum = ~modifyChecksum; + for(unsigned n = 0; n < 32; n += 8) write(outputChecksum >> n); + + modifyFile.close(); + return true; +} + +} + +#endif diff --git a/snesfilter/nall/bps/metadata.hpp b/snesfilter/nall/bps/metadata.hpp new file mode 100755 index 00000000..46759e6f --- /dev/null +++ b/snesfilter/nall/bps/metadata.hpp @@ -0,0 +1,121 @@ +#ifndef NALL_BPS_METADATA_HPP +#define NALL_BPS_METADATA_HPP + +#include +#include +#include +#include +#include + +namespace nall { + +struct bpsmetadata { + inline bool load(const string &filename); + inline bool save(const string &filename, const string &metadata); + inline string metadata() const; + +protected: + file sourceFile; + string metadataString; +}; + +bool bpsmetadata::load(const string &filename) { + if(sourceFile.open(filename, file::mode::read) == false) return false; + + auto read = [&]() -> uint8_t { + return sourceFile.read(); + }; + + auto decode = [&]() -> uint64_t { + uint64_t data = 0, shift = 1; + while(true) { + uint8_t x = read(); + data += (x & 0x7f) * shift; + if(x & 0x80) break; + shift <<= 7; + data += shift; + } + return data; + }; + + if(read() != 'B') return false; + if(read() != 'P') return false; + if(read() != 'S') return false; + if(read() != '1') return false; + decode(); + decode(); + unsigned metadataSize = decode(); + char data[metadataSize + 1]; + for(unsigned n = 0; n < metadataSize; n++) data[n] = read(); + data[metadataSize] = 0; + metadataString = (const char*)data; + + return true; +} + +bool bpsmetadata::save(const string &filename, const string &metadata) { + file targetFile; + if(targetFile.open(filename, file::mode::write) == false) return false; + if(sourceFile.open() == false) return false; + sourceFile.seek(0); + + auto read = [&]() -> uint8_t { + return sourceFile.read(); + }; + + auto decode = [&]() -> uint64_t { + uint64_t data = 0, shift = 1; + while(true) { + uint8_t x = read(); + data += (x & 0x7f) * shift; + if(x & 0x80) break; + shift <<= 7; + data += shift; + } + return data; + }; + + uint32_t checksum = ~0; + + auto write = [&](uint8_t data) { + targetFile.write(data); + checksum = crc32_adjust(checksum, data); + }; + + auto encode = [&](uint64_t data) { + while(true) { + uint64_t x = data & 0x7f; + data >>= 7; + if(data == 0) { + write(0x80 | x); + break; + } + write(x); + data--; + } + }; + + for(unsigned n = 0; n < 4; n++) write(read()); + encode(decode()); + encode(decode()); + unsigned sourceLength = decode(); + unsigned targetLength = metadata.length(); + encode(targetLength); + sourceFile.seek(sourceLength, file::index::relative); + for(unsigned n = 0; n < targetLength; n++) write(metadata[n]); + unsigned length = sourceFile.size() - sourceFile.offset() - 4; + for(unsigned n = 0; n < length; n++) write(read()); + uint32_t outputChecksum = ~checksum; + for(unsigned n = 0; n < 32; n += 8) write(outputChecksum >> n); + + targetFile.close(); + return true; +} + +string bpsmetadata::metadata() const { + return metadataString; +} + +} + +#endif diff --git a/snesfilter/nall/bps/patch.hpp b/snesfilter/nall/bps/patch.hpp new file mode 100755 index 00000000..85c4dcae --- /dev/null +++ b/snesfilter/nall/bps/patch.hpp @@ -0,0 +1,219 @@ +#ifndef NALL_BPS_PATCH_HPP +#define NALL_BPS_PATCH_HPP + +#include +#include +#include +#include +#include + +namespace nall { + +struct bpspatch { + inline bool modify(const uint8_t *data, unsigned size); + inline void source(const uint8_t *data, unsigned size); + inline void target(uint8_t *data, unsigned size); + + inline bool modify(const string &filename); + inline bool source(const string &filename); + inline bool target(const string &filename); + + inline string metadata() const; + inline unsigned size() const; + + enum result : unsigned { + unknown, + success, + patch_too_small, + patch_invalid_header, + source_too_small, + target_too_small, + source_checksum_invalid, + target_checksum_invalid, + patch_checksum_invalid, + }; + + inline result apply(); + +protected: + enum : unsigned { SourceRead, TargetRead, SourceCopy, TargetCopy }; + + filemap modifyFile; + const uint8_t *modifyData; + unsigned modifySize; + + filemap sourceFile; + const uint8_t *sourceData; + unsigned sourceSize; + + filemap targetFile; + uint8_t *targetData; + unsigned targetSize; + + unsigned modifySourceSize; + unsigned modifyTargetSize; + unsigned modifyMarkupSize; + string metadataString; +}; + +bool bpspatch::modify(const uint8_t *data, unsigned size) { + if(size < 19) return false; + modifyData = data; + modifySize = size; + + unsigned offset = 4; + auto decode = [&]() -> uint64_t { + uint64_t data = 0, shift = 1; + while(true) { + uint8_t x = modifyData[offset++]; + data += (x & 0x7f) * shift; + if(x & 0x80) break; + shift <<= 7; + data += shift; + } + return data; + }; + + modifySourceSize = decode(); + modifyTargetSize = decode(); + modifyMarkupSize = decode(); + + char buffer[modifyMarkupSize + 1]; + for(unsigned n = 0; n < modifyMarkupSize; n++) buffer[n] = modifyData[offset++]; + buffer[modifyMarkupSize] = 0; + metadataString = (const char*)buffer; + + return true; +} + +void bpspatch::source(const uint8_t *data, unsigned size) { + sourceData = data; + sourceSize = size; +} + +void bpspatch::target(uint8_t *data, unsigned size) { + targetData = data; + targetSize = size; +} + +bool bpspatch::modify(const string &filename) { + if(modifyFile.open(filename, filemap::mode::read) == false) return false; + return modify(modifyFile.data(), modifyFile.size()); +} + +bool bpspatch::source(const string &filename) { + if(sourceFile.open(filename, filemap::mode::read) == false) return false; + source(sourceFile.data(), sourceFile.size()); + return true; +} + +bool bpspatch::target(const string &filename) { + file fp; + if(fp.open(filename, file::mode::write) == false) return false; + fp.truncate(modifyTargetSize); + fp.close(); + + if(targetFile.open(filename, filemap::mode::readwrite) == false) return false; + target(targetFile.data(), targetFile.size()); + return true; +} + +string bpspatch::metadata() const { + return metadataString; +} + +unsigned bpspatch::size() const { + return modifyTargetSize; +} + +bpspatch::result bpspatch::apply() { + if(modifySize < 19) return result::patch_too_small; + + uint32_t modifyChecksum = ~0, targetChecksum = ~0; + unsigned modifyOffset = 0, sourceRelativeOffset = 0, targetRelativeOffset = 0, outputOffset = 0; + + auto read = [&]() -> uint8_t { + uint8_t data = modifyData[modifyOffset++]; + modifyChecksum = crc32_adjust(modifyChecksum, data); + return data; + }; + + auto decode = [&]() -> uint64_t { + uint64_t data = 0, shift = 1; + while(true) { + uint8_t x = read(); + data += (x & 0x7f) * shift; + if(x & 0x80) break; + shift <<= 7; + data += shift; + } + return data; + }; + + auto write = [&](uint8_t data) { + targetData[outputOffset++] = data; + targetChecksum = crc32_adjust(targetChecksum, data); + }; + + if(read() != 'B') return result::patch_invalid_header; + if(read() != 'P') return result::patch_invalid_header; + if(read() != 'S') return result::patch_invalid_header; + if(read() != '1') return result::patch_invalid_header; + + modifySourceSize = decode(); + modifyTargetSize = decode(); + modifyMarkupSize = decode(); + for(unsigned n = 0; n < modifyMarkupSize; n++) read(); + + if(modifySourceSize > sourceSize) return result::source_too_small; + if(modifyTargetSize > targetSize) return result::target_too_small; + + while(modifyOffset < modifySize - 12) { + unsigned length = decode(); + unsigned mode = length & 3; + length = (length >> 2) + 1; + + switch(mode) { + case SourceRead: + while(length--) write(sourceData[outputOffset]); + break; + case TargetRead: + while(length--) write(read()); + break; + case SourceCopy: + case TargetCopy: + signed offset = decode(); + bool negative = offset & 1; + offset >>= 1; + if(negative) offset = -offset; + + if(mode == SourceCopy) { + sourceRelativeOffset += offset; + while(length--) write(sourceData[sourceRelativeOffset++]); + } else { + targetRelativeOffset += offset; + while(length--) write(targetData[targetRelativeOffset++]); + } + break; + } + } + + uint32_t modifySourceChecksum = 0, modifyTargetChecksum = 0, modifyModifyChecksum = 0; + for(unsigned n = 0; n < 32; n += 8) modifySourceChecksum |= read() << n; + for(unsigned n = 0; n < 32; n += 8) modifyTargetChecksum |= read() << n; + uint32_t checksum = ~modifyChecksum; + for(unsigned n = 0; n < 32; n += 8) modifyModifyChecksum |= read() << n; + + uint32_t sourceChecksum = crc32_calculate(sourceData, modifySourceSize); + targetChecksum = ~targetChecksum; + + if(sourceChecksum != modifySourceChecksum) return result::source_checksum_invalid; + if(targetChecksum != modifyTargetChecksum) return result::target_checksum_invalid; + if(checksum != modifyModifyChecksum) return result::patch_checksum_invalid; + + return result::success; +} + +} + +#endif diff --git a/snesfilter/nall/compositor.hpp b/snesfilter/nall/compositor.hpp new file mode 100755 index 00000000..6d5c46c9 --- /dev/null +++ b/snesfilter/nall/compositor.hpp @@ -0,0 +1,79 @@ +#ifndef NALL_COMPOSITOR_HPP +#define NALL_COMPOSITOR_HPP + +#include + +namespace nall { + +struct compositor { + inline static bool enabled(); + inline static bool enable(bool status); +}; + +#if defined(PLATFORM_X) + +bool compositor::enabled() { + FILE *fp = popen("xfconf-query -c xfwm4 -p '/general/use_compositing'", "r"); + if(fp == 0) return false; + + char buffer[512]; + if(fgets(buffer, sizeof buffer, fp) == 0) return false; + + if(!memcmp(buffer, "true", 4)) return true; + return false; +} + +bool compositor::enable(bool status) { + FILE *fp; + if(status) { + fp = popen("xfconf-query -c xfwm4 -p '/general/use_compositing' -t 'bool' -s 'true'", "r"); + } else { + fp = popen("xfconf-query -c xfwm4 -p '/general/use_compositing' -t 'bool' -s 'false'", "r"); + } + if(fp == 0) return false; + pclose(fp); + return true; +} + +#elif defined(PLATFORM_WIN) + +bool compositor::enabled() { + HMODULE module = GetModuleHandleW(L"dwmapi"); + if(module == 0) module = LoadLibraryW(L"dwmapi"); + if(module == 0) return false; + + auto pDwmIsCompositionEnabled = (HRESULT (WINAPI*)(BOOL*))GetProcAddress(module, "DwmIsCompositionEnabled"); + if(pDwmIsCompositionEnabled == 0) return false; + + BOOL result; + if(pDwmIsCompositionEnabled(&result) != S_OK) return false; + return result; +} + +bool compositor::enable(bool status) { + HMODULE module = GetModuleHandleW(L"dwmapi"); + if(module == 0) module = LoadLibraryW(L"dwmapi"); + if(module == 0) return false; + + auto pDwmEnableComposition = (HRESULT (WINAPI*)(UINT))GetProcAddress(module, "DwmEnableComposition"); + if(pDwmEnableComposition == 0) return false; + + if(pDwmEnableComposition(status) != S_OK) return false; + return true; +} + +#else + +bool compositor::enabled() { + return false; +} + +bool compositor::enable(bool) { + return false; +} + +#endif + +} + +#endif diff --git a/snesfilter/nall/config.hpp b/snesfilter/nall/config.hpp index b8381b16..99aaee08 100755 --- a/snesfilter/nall/config.hpp +++ b/snesfilter/nall/config.hpp @@ -34,11 +34,11 @@ namespace nall { string get() const { switch(type) { - case boolean_t: return string() << *(bool*)data; - case signed_t: return string() << *(signed*)data; - case unsigned_t: return string() << *(unsigned*)data; - case double_t: return string() << *(double*)data; - case string_t: return string() << "\"" << *(string*)data << "\""; + case boolean_t: return { *(bool*)data }; + case signed_t: return { *(signed*)data }; + case unsigned_t: return { *(unsigned*)data }; + case double_t: return { *(double*)data }; + case string_t: return { "\"", *(string*)data, "\"" }; } return "???"; } @@ -105,9 +105,9 @@ namespace nall { if(fp.open(filename, file::mode::write)) { for(unsigned i = 0; i < list.size(); i++) { string output; - output << list[i].name << " = " << list[i].get(); - if(list[i].desc != "") output << " # " << list[i].desc; - output << "\r\n"; + output.append(list[i].name, " = ", list[i].get()); + if(list[i].desc != "") output.append(" # ", list[i].desc); + output.append("\r\n"); fp.print(output); } diff --git a/snesfilter/nall/detect.hpp b/snesfilter/nall/detect.hpp index b4991aaf..85122fbd 100755 --- a/snesfilter/nall/detect.hpp +++ b/snesfilter/nall/detect.hpp @@ -15,7 +15,7 @@ #define PLATFORM_WIN #elif defined(__APPLE__) #define PLATFORM_OSX -#elif defined(linux) || defined(__sun__) || defined(__FreeBSD__) || defined(__NetBSD__) || defined(__OpenBSD__) +#elif defined(linux) || defined(__sun__) || defined(__FreeBSD__) || defined(__FreeBSD_kernel__) || defined(__NetBSD__) || defined(__OpenBSD__) #define PLATFORM_X #endif diff --git a/snesfilter/nall/dictionary.hpp b/snesfilter/nall/dictionary.hpp deleted file mode 100755 index dcb04151..00000000 --- a/snesfilter/nall/dictionary.hpp +++ /dev/null @@ -1,75 +0,0 @@ -#ifndef NALL_DICTIONARY_HPP -#define NALL_DICTIONARY_HPP - -#include -#include -#include - -namespace nall { - class dictionary { - public: - string operator[](const char *input) { - for(unsigned i = 0; i < index_input.size(); i++) { - if(index_input[i] == input) return index_output[i]; - } - - //no match, use input; remove input identifier, if one exists - if(strbegin(input, "{{")) { - if(auto pos = strpos(input, "}}")) { - string temp = substr(input, pos() + 2); - return temp; - } - } - - return input; - } - - bool import(const char *filename) { - string data; - if(data.readfile(filename) == false) return false; - data.ltrim<1>("\xef\xbb\xbf"); //remove UTF-8 marker, if it exists - data.replace("\r", ""); - - lstring line; - line.split("\n", data); - for(unsigned i = 0; i < line.size(); i++) { - lstring part; - //format: "Input" = "Output" - part.qsplit("=", line[i]); - if(part.size() != 2) continue; - - //remove whitespace - part[0].trim(); - part[1].trim(); - - //remove quotes - part[0].trim<1>("\""); - part[1].trim<1>("\""); - - unsigned n = index_input.size(); - index_input[n] = part[0]; - index_output[n] = part[1]; - } - - return true; - } - - void reset() { - index_input.reset(); - index_output.reset(); - } - - ~dictionary() { - reset(); - } - - dictionary& operator=(const dictionary&) = delete; - dictionary(const dictionary&) = delete; - - protected: - lstring index_input; - lstring index_output; - }; -} - -#endif diff --git a/snesfilter/nall/directory.hpp b/snesfilter/nall/directory.hpp index c4f94c9a..7fbc15f4 100755 --- a/snesfilter/nall/directory.hpp +++ b/snesfilter/nall/directory.hpp @@ -6,7 +6,7 @@ #include #if defined(_WIN32) - #include + #include #else #include #include @@ -42,20 +42,21 @@ struct directory { if(wcscmp(data.cFileName, L".") && wcscmp(data.cFileName, L"..")) { if(data.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) { string name = (const char*)utf8_t(data.cFileName); - if(wildcard(name, pattern)) list.append(string(name, "/")); + if(wildcard(name, pattern)) list.append(name); } } while(FindNextFile(handle, &data) != false) { if(wcscmp(data.cFileName, L".") && wcscmp(data.cFileName, L"..")) { if(data.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) { string name = (const char*)utf8_t(data.cFileName); - if(wildcard(name, pattern)) list.append(string(name, "/")); + if(wildcard(name, pattern)) list.append(name); } } } FindClose(handle); } if(list.size() > 0) sort(&list[0], list.size()); + foreach(name, list) name.append("/"); //must append after sorting return list; } @@ -109,14 +110,14 @@ struct directory { if(!strcmp(ep->d_name, ".")) continue; if(!strcmp(ep->d_name, "..")) continue; if(ep->d_type & DT_DIR) { - if(wildcard(ep->d_name, pattern)) list.append(string(ep->d_name, "/")); + if(wildcard(ep->d_name, pattern)) list.append(ep->d_name); } } closedir(dp); } if(list.size() > 0) sort(&list[0], list.size()); + foreach(name, list) name.append("/"); //must append after sorting return list; - } inline lstring directory::files(const string &pathname, const string &pattern) { diff --git a/snesfilter/nall/dl.hpp b/snesfilter/nall/dl.hpp index ebfa5585..c697958c 100755 --- a/snesfilter/nall/dl.hpp +++ b/snesfilter/nall/dl.hpp @@ -12,7 +12,7 @@ #include #elif defined(PLATFORM_WIN) #include - #include + #include #endif namespace nall { diff --git a/snesfilter/nall/dsp.hpp b/snesfilter/nall/dsp.hpp new file mode 100755 index 00000000..009c8b6c --- /dev/null +++ b/snesfilter/nall/dsp.hpp @@ -0,0 +1,8 @@ +#ifndef NALL_DSP_HPP +#define NALL_DSP_HPP + +#define NALL_DSP_INTERNAL_HPP +#include +#undef NALL_DSP_INTERNAL_HPP + +#endif diff --git a/snesfilter/nall/dsp/buffer.hpp b/snesfilter/nall/dsp/buffer.hpp new file mode 100755 index 00000000..4386d0e9 --- /dev/null +++ b/snesfilter/nall/dsp/buffer.hpp @@ -0,0 +1,51 @@ +#ifdef NALL_DSP_INTERNAL_HPP + +struct Buffer { + double **sample; + uint16_t rdoffset; + uint16_t wroffset; + unsigned channels; + + void setChannels(unsigned channels) { + for(unsigned c = 0; c < this->channels; c++) { + if(sample[c]) delete[] sample[c]; + } + if(sample) delete[] sample; + + this->channels = channels; + if(channels == 0) return; + + sample = new double*[channels]; + for(unsigned c = 0; c < channels; c++) { + sample[c] = new double[65536](); + } + } + + inline double& read(unsigned channel, signed offset = 0) { + return sample[channel][(uint16_t)(rdoffset + offset)]; + } + + inline double& write(unsigned channel, signed offset = 0) { + return sample[channel][(uint16_t)(wroffset + offset)]; + } + + inline void clear() { + for(unsigned c = 0; c < channels; c++) { + for(unsigned n = 0; n < 65536; n++) { + sample[c][n] = 0; + } + } + rdoffset = 0; + wroffset = 0; + } + + Buffer() { + channels = 0; + } + + ~Buffer() { + setChannels(0); + } +}; + +#endif diff --git a/snesfilter/nall/dsp/core.hpp b/snesfilter/nall/dsp/core.hpp new file mode 100755 index 00000000..a4c58c38 --- /dev/null +++ b/snesfilter/nall/dsp/core.hpp @@ -0,0 +1,162 @@ +#ifdef NALL_DSP_INTERNAL_HPP + +#include +#include + +namespace nall { + +struct DSP { + enum class Resampler : unsigned { + Point, + Linear, + Cosine, + Cubic, + Hermite, + Average, + }; + + inline void setChannels(unsigned channels); + inline void setPrecision(unsigned precision); + inline void setFrequency(double frequency); //inputFrequency + inline void setVolume(double volume); + inline void setBalance(double balance); + + inline void setResampler(Resampler resampler); + inline void setResamplerFrequency(double frequency); //outputFrequency + + inline void sample(signed channel[]); + inline bool pending(); + inline void read(signed channel[]); + + inline void clear(); + inline DSP(); + inline ~DSP(); + +protected: + struct Settings { + unsigned channels; + unsigned precision; + double frequency; + double volume; + double balance; + //internal + double intensity; + } settings; + + struct ResamplerSettings { + Resampler engine; + double frequency; + //internal + double fraction; + double step; + } resampler; + + inline void resamplerRun(); + inline void resamplerWrite(double channel[]); + + inline void resamplePoint(); + inline void resampleLinear(); + inline void resampleCosine(); + inline void resampleCubic(); + inline void resampleHermite(); + inline void resampleAverage(); + + #include "buffer.hpp" + Buffer buffer; + Buffer output; + + inline void adjustVolume(); + inline void adjustBalance(); + inline signed clamp(const unsigned bits, const signed x); +}; + +#include "settings.hpp" + +void DSP::sample(signed channel[]) { + for(unsigned c = 0; c < settings.channels; c++) { + buffer.write(c) = (double)channel[c] / settings.intensity; + } + buffer.wroffset++; + resamplerRun(); +} + +bool DSP::pending() { + return output.rdoffset != output.wroffset; +} + +void DSP::read(signed channel[]) { + adjustVolume(); + adjustBalance(); + + for(unsigned c = 0; c < settings.channels; c++) { + channel[c] = clamp(settings.precision, output.read(c) * settings.intensity); + } + output.rdoffset++; +} + +void DSP::resamplerRun() { + switch(resampler.engine) { + case Resampler::Point: return resamplePoint(); + case Resampler::Linear: return resampleLinear(); + case Resampler::Cosine: return resampleCosine(); + case Resampler::Cubic: return resampleCubic(); + case Resampler::Hermite: return resampleHermite(); + case Resampler::Average: return resampleAverage(); + } +} + +void DSP::resamplerWrite(double channel[]) { + for(unsigned c = 0; c < settings.channels; c++) { + output.write(c) = channel[c]; + } + output.wroffset++; +} + +#include "resample/point.hpp" +#include "resample/linear.hpp" +#include "resample/cosine.hpp" +#include "resample/cubic.hpp" +#include "resample/hermite.hpp" +#include "resample/average.hpp" + +void DSP::adjustVolume() { + for(unsigned c = 0; c < settings.channels; c++) { + output.read(c) *= settings.volume; + } +} + +void DSP::adjustBalance() { + if(settings.channels != 2) return; //TODO: support > 2 channels + if(settings.balance < 0.0) output.read(1) *= 1.0 + settings.balance; + if(settings.balance > 0.0) output.read(0) *= 1.0 - settings.balance; +} + +signed DSP::clamp(const unsigned bits, const signed x) { + const signed b = 1U << (bits - 1); + const signed m = (1U << (bits - 1)) - 1; + return (x > m) ? m : (x < -b) ? -b : x; +} + +void DSP::clear() { + resampler.fraction = 0.0; + buffer.clear(); + output.clear(); +} + +DSP::DSP() { + setChannels(2); + setPrecision(16); + setFrequency(44100.0); + setVolume(1.0); + setBalance(0.0); + setResampler(Resampler::Hermite); + setResamplerFrequency(44100.0); + clear(); +} + +DSP::~DSP() { +} + +} + +#endif diff --git a/snesfilter/nall/dsp/resample/average.hpp b/snesfilter/nall/dsp/resample/average.hpp new file mode 100755 index 00000000..c5cdbca3 --- /dev/null +++ b/snesfilter/nall/dsp/resample/average.hpp @@ -0,0 +1,31 @@ +#ifdef NALL_DSP_INTERNAL_HPP + +void DSP::resampleAverage() { + //can only average if input frequency >= output frequency + if(resampler.step < 1.0) return resampleHermite(); + + resampler.fraction += 1.0; + + double scalar = 1.0; + if(resampler.fraction > resampler.step) scalar = 1.0 - (resampler.fraction - resampler.step); + + for(unsigned c = 0; c < settings.channels; c++) { + output.write(c) += buffer.read(c) * scalar; + } + + if(resampler.fraction >= resampler.step) { + for(unsigned c = 0; c < settings.channels; c++) { + output.write(c) /= resampler.step; + } + output.wroffset++; + + resampler.fraction -= resampler.step; + for(unsigned c = 0; c < settings.channels; c++) { + output.write(c) = buffer.read(c) * resampler.fraction; + } + } + + buffer.rdoffset++; +} + +#endif diff --git a/snesfilter/nall/dsp/resample/cosine.hpp b/snesfilter/nall/dsp/resample/cosine.hpp new file mode 100755 index 00000000..5405b7f3 --- /dev/null +++ b/snesfilter/nall/dsp/resample/cosine.hpp @@ -0,0 +1,25 @@ +#ifdef NALL_DSP_INTERNAL_HPP + +void DSP::resampleCosine() { + while(resampler.fraction <= 1.0) { + double channel[settings.channels]; + + for(unsigned n = 0; n < settings.channels; n++) { + double a = buffer.read(n, -1); + double b = buffer.read(n, -0); + + double mu = resampler.fraction; + mu = (1.0 - cos(mu * 3.14159265)) / 2.0; + + channel[n] = a * (1.0 - mu) + b * mu; + } + + resamplerWrite(channel); + resampler.fraction += resampler.step; + } + + buffer.rdoffset++; + resampler.fraction -= 1.0; +} + +#endif diff --git a/snesfilter/nall/dsp/resample/cubic.hpp b/snesfilter/nall/dsp/resample/cubic.hpp new file mode 100755 index 00000000..71e3766f --- /dev/null +++ b/snesfilter/nall/dsp/resample/cubic.hpp @@ -0,0 +1,31 @@ +#ifdef NALL_DSP_INTERNAL_HPP + +void DSP::resampleCubic() { + while(resampler.fraction <= 1.0) { + double channel[settings.channels]; + + for(unsigned n = 0; n < settings.channels; n++) { + double a = buffer.read(n, -3); + double b = buffer.read(n, -2); + double c = buffer.read(n, -1); + double d = buffer.read(n, -0); + + double mu = resampler.fraction; + + double A = d - c - a + b; + double B = a - b - A; + double C = c - a; + double D = b; + + channel[n] = A * (mu * 3) + B * (mu * 2) + C * mu + D; + } + + resamplerWrite(channel); + resampler.fraction += resampler.step; + } + + buffer.rdoffset++; + resampler.fraction -= 1.0; +} + +#endif diff --git a/snesfilter/nall/dsp/resample/hermite.hpp b/snesfilter/nall/dsp/resample/hermite.hpp new file mode 100755 index 00000000..6eed087d --- /dev/null +++ b/snesfilter/nall/dsp/resample/hermite.hpp @@ -0,0 +1,43 @@ +#ifdef NALL_DSP_INTERNAL_HPP + +void DSP::resampleHermite() { + while(resampler.fraction <= 1.0) { + double channel[settings.channels]; + + for(unsigned n = 0; n < settings.channels; n++) { + double a = buffer.read(n, -3); + double b = buffer.read(n, -2); + double c = buffer.read(n, -1); + double d = buffer.read(n, -0); + + const double tension = 0.0; //-1 = low, 0 = normal, +1 = high + const double bias = 0.0; //-1 = left, 0 = even, +1 = right + + double mu1, mu2, mu3, m0, m1, a0, a1, a2, a3; + + mu1 = resampler.fraction; + mu2 = mu1 * mu1; + mu3 = mu2 * mu1; + + m0 = (b - a) * (1.0 + bias) * (1.0 - tension) / 2.0; + m0 += (c - b) * (1.0 - bias) * (1.0 - tension) / 2.0; + m1 = (c - b) * (1.0 + bias) * (1.0 - tension) / 2.0; + m1 += (d - c) * (1.0 - bias) * (1.0 - tension) / 2.0; + + a0 = +2 * mu3 - 3 * mu2 + 1; + a1 = mu3 - 2 * mu2 + mu1; + a2 = mu3 - mu2; + a3 = -2 * mu3 + 3 * mu2; + + channel[n] = (a0 * b) + (a1 * m0) + (a2 * m1) + (a3 * c); + } + + resamplerWrite(channel); + resampler.fraction += resampler.step; + } + + buffer.rdoffset++; + resampler.fraction -= 1.0; +} + +#endif diff --git a/snesfilter/nall/dsp/resample/linear.hpp b/snesfilter/nall/dsp/resample/linear.hpp new file mode 100755 index 00000000..3dbda6a0 --- /dev/null +++ b/snesfilter/nall/dsp/resample/linear.hpp @@ -0,0 +1,24 @@ +#ifdef NALL_DSP_INTERNAL_HPP + +void DSP::resampleLinear() { + while(resampler.fraction <= 1.0) { + double channel[settings.channels]; + + for(unsigned n = 0; n < settings.channels; n++) { + double a = buffer.read(n, -1); + double b = buffer.read(n, -0); + + double mu = resampler.fraction; + + channel[n] = a * (1.0 - mu) + b * mu; + } + + resamplerWrite(channel); + resampler.fraction += resampler.step; + } + + buffer.rdoffset++; + resampler.fraction -= 1.0; +} + +#endif diff --git a/snesfilter/nall/dsp/resample/point.hpp b/snesfilter/nall/dsp/resample/point.hpp new file mode 100755 index 00000000..b1cc7dae --- /dev/null +++ b/snesfilter/nall/dsp/resample/point.hpp @@ -0,0 +1,24 @@ +#ifdef NALL_DSP_INTERNAL_HPP + +void DSP::resamplePoint() { + while(resampler.fraction <= 1.0) { + double channel[settings.channels]; + + for(unsigned n = 0; n < settings.channels; n++) { + double a = buffer.read(n, -1); + double b = buffer.read(n, -0); + + double mu = resampler.fraction; + + channel[n] = mu < 0.5 ? a : b; + } + + resamplerWrite(channel); + resampler.fraction += resampler.step; + } + + buffer.rdoffset++; + resampler.fraction -= 1.0; +} + +#endif diff --git a/snesfilter/nall/dsp/settings.hpp b/snesfilter/nall/dsp/settings.hpp new file mode 100755 index 00000000..dc422e39 --- /dev/null +++ b/snesfilter/nall/dsp/settings.hpp @@ -0,0 +1,39 @@ +#ifdef NALL_DSP_INTERNAL_HPP + +void DSP::setChannels(unsigned channels) { + assert(channels > 0); + buffer.setChannels(channels); + output.setChannels(channels); + settings.channels = channels; +} + +void DSP::setPrecision(unsigned precision) { + settings.precision = precision; + settings.intensity = 1 << (settings.precision - 1); +} + +void DSP::setFrequency(double frequency) { + settings.frequency = frequency; + resampler.fraction = 0; + resampler.step = settings.frequency / resampler.frequency; +} + +void DSP::setVolume(double volume) { + settings.volume = volume; +} + +void DSP::setBalance(double balance) { + settings.balance = balance; +} + +void DSP::setResampler(Resampler engine) { + resampler.engine = engine; +} + +void DSP::setResamplerFrequency(double frequency) { + resampler.frequency = frequency; + resampler.fraction = 0; + resampler.step = settings.frequency / resampler.frequency; +} + +#endif diff --git a/snesfilter/nall/file.hpp b/snesfilter/nall/file.hpp index 103c7d4a..8447f3ab 100755 --- a/snesfilter/nall/file.hpp +++ b/snesfilter/nall/file.hpp @@ -1,22 +1,14 @@ #ifndef NALL_FILE_HPP #define NALL_FILE_HPP -#include -#include - -#if !defined(_WIN32) - #include -#else - #include -#endif - +#include #include #include -#include #include +#include namespace nall { - inline FILE* fopen_utf8(const char *utf8_filename, const char *mode) { + inline FILE* fopen_utf8(const string &utf8_filename, const char *mode) { #if !defined(_WIN32) return fopen(utf8_filename, mode); #else @@ -28,6 +20,30 @@ namespace nall { public: enum class mode : unsigned { read, write, readwrite, writeread }; enum class index : unsigned { absolute, relative }; + enum class time : unsigned { create, modify, access }; + + static bool read(const string &filename, uint8_t *&data, unsigned &size) { + data = 0; + file fp; + if(fp.open(filename, mode::read) == false) return false; + size = fp.size(); + data = new uint8_t[size]; + fp.read(data, size); + fp.close(); + return true; + } + + static bool read(const string &filename, const uint8_t *&data, unsigned &size) { + return file::read(filename, (uint8_t*&)data, size); + } + + static bool write(const string &filename, const uint8_t *data, unsigned size) { + file fp; + if(fp.open(filename, mode::write) == false) return false; + fp.write(data, size); + fp.close(); + return true; + } uint8_t read() { if(!fp) return 0xff; //file not open @@ -142,52 +158,60 @@ namespace nall { return file_offset >= file_size; } - static bool exists(const char *fn) { + static bool exists(const string &filename) { #if !defined(_WIN32) - FILE *fp = fopen(fn, "rb"); + struct stat64 data; + return stat64(filename, &data) == 0; #else - FILE *fp = _wfopen(utf16_t(fn), L"rb"); + struct __stat64 data; + return _wstat64(utf16_t(filename), &data) == 0; #endif - if(fp) { - fclose(fp); - return true; - } - return false; } - static unsigned size(const char *fn) { + static uintmax_t size(const string &filename) { #if !defined(_WIN32) - FILE *fp = fopen(fn, "rb"); + struct stat64 data; + stat64(filename, &data); #else - FILE *fp = _wfopen(utf16_t(fn), L"rb"); + struct __stat64 data; + _wstat64(utf16_t(filename), &data); #endif - unsigned filesize = 0; - if(fp) { - fseek(fp, 0, SEEK_END); - filesize = ftell(fp); - fclose(fp); + return S_ISREG(data.st_mode) ? data.st_size : 0u; + } + + static time_t timestamp(const string &filename, file::time mode = file::time::create) { + #if !defined(_WIN32) + struct stat64 data; + stat64(filename, &data); + #else + struct __stat64 data; + _wstat64(utf16_t(filename), &data); + #endif + switch(mode) { default: + case file::time::create: return data.st_ctime; + case file::time::modify: return data.st_mtime; + case file::time::access: return data.st_atime; } - return filesize; } bool open() { return fp; } - bool open(const char *fn, mode mode_) { + bool open(const string &filename, mode mode_) { if(fp) return false; switch(file_mode = mode_) { #if !defined(_WIN32) - case mode::read: fp = fopen(fn, "rb"); break; - case mode::write: fp = fopen(fn, "wb+"); break; //need read permission for buffering - case mode::readwrite: fp = fopen(fn, "rb+"); break; - case mode::writeread: fp = fopen(fn, "wb+"); break; + case mode::read: fp = fopen(filename, "rb" ); break; + case mode::write: fp = fopen(filename, "wb+"); break; //need read permission for buffering + case mode::readwrite: fp = fopen(filename, "rb+"); break; + case mode::writeread: fp = fopen(filename, "wb+"); break; #else - case mode::read: fp = _wfopen(utf16_t(fn), L"rb"); break; - case mode::write: fp = _wfopen(utf16_t(fn), L"wb+"); break; - case mode::readwrite: fp = _wfopen(utf16_t(fn), L"rb+"); break; - case mode::writeread: fp = _wfopen(utf16_t(fn), L"wb+"); break; + case mode::read: fp = _wfopen(utf16_t(filename), L"rb" ); break; + case mode::write: fp = _wfopen(utf16_t(filename), L"wb+"); break; + case mode::readwrite: fp = _wfopen(utf16_t(filename), L"rb+"); break; + case mode::writeread: fp = _wfopen(utf16_t(filename), L"wb+"); break; #endif } if(!fp) return false; diff --git a/snesfilter/nall/filemap.hpp b/snesfilter/nall/filemap.hpp index 5e8cc059..7eeac2b0 100755 --- a/snesfilter/nall/filemap.hpp +++ b/snesfilter/nall/filemap.hpp @@ -2,7 +2,7 @@ #define NALL_FILEMAP_HPP #include -#include +#include #include #include @@ -47,6 +47,12 @@ namespace nall { } bool p_open(const char *filename, mode mode_) { + if(file::exists(filename) && file::size(filename) == 0) { + p_handle = 0; + p_size = 0; + return true; + } + int desired_access, creation_disposition, flprotect, map_access; switch(mode_) { @@ -133,6 +139,12 @@ namespace nall { } bool p_open(const char *filename, mode mode_) { + if(file::exists(filename) && file::size(filename) == 0) { + p_handle = 0; + p_size = 0; + return true; + } + int open_flags, mmap_flags; switch(mode_) { diff --git a/snesfilter/nall/gameboy/cartridge.hpp b/snesfilter/nall/gameboy/cartridge.hpp index 0e1b28d8..af04e0bb 100755 --- a/snesfilter/nall/gameboy/cartridge.hpp +++ b/snesfilter/nall/gameboy/cartridge.hpp @@ -6,7 +6,7 @@ namespace nall { class GameBoyCartridge { public: string xml; - inline GameBoyCartridge(const uint8_t *data, unsigned size); + inline GameBoyCartridge(uint8_t *data, unsigned size); //private: struct Information { @@ -21,7 +21,7 @@ public: } info; }; -GameBoyCartridge::GameBoyCartridge(const uint8_t *romdata, unsigned romsize) { +GameBoyCartridge::GameBoyCartridge(uint8_t *romdata, unsigned romsize) { xml = "\n"; if(romsize < 0x4000) return; @@ -34,6 +34,20 @@ GameBoyCartridge::GameBoyCartridge(const uint8_t *romdata, unsigned romsize) { info.romsize = 0; info.ramsize = 0; + unsigned base = romsize - 0x8000; + if(romdata[base + 0x0104] == 0xce && romdata[base + 0x0105] == 0xed + && romdata[base + 0x0106] == 0x66 && romdata[base + 0x0107] == 0x66 + && romdata[base + 0x0108] == 0xcc && romdata[base + 0x0109] == 0x0d + && romdata[base + 0x0147] >= 0x0b && romdata[base + 0x0147] <= 0x0d + ) { + //MMM01 stores header at bottom of image + //flip this around for consistency with all other mappers + uint8_t header[0x8000]; + memcpy(header, romdata + base, 0x8000); + memmove(romdata + 0x8000, romdata, romsize - 0x8000); + memcpy(romdata, header, 0x8000); + } + switch(romdata[0x0147]) { case 0x00: info.mapper = "none"; break; case 0x01: info.mapper = "MBC1"; break; @@ -86,17 +100,17 @@ GameBoyCartridge::GameBoyCartridge(const uint8_t *romdata, unsigned romsize) { if(info.mapper == "MBC2") info.ramsize = 512; //512 x 4-bit - xml << "\n"; + xml.append("\n"); - xml << " \n"; //TODO: trust/check info.romsize? + xml.append(" \n"); //TODO: trust/check info.romsize? if(info.ramsize > 0) - xml << " \n"; + xml.append(" \n"); - xml << "\n"; + xml.append("\n"); xml.transform("'", "\""); } diff --git a/snesfilter/nall/gzip.hpp b/snesfilter/nall/gzip.hpp new file mode 100755 index 00000000..635d3277 --- /dev/null +++ b/snesfilter/nall/gzip.hpp @@ -0,0 +1,87 @@ +#ifndef NALL_GZIP_HPP +#define NALL_GZIP_HPP + +#include +#include + +namespace nall { + +struct gzip { + string filename; + uint8_t *data; + unsigned size; + + bool decompress(const string &filename); + bool decompress(const uint8_t *data, unsigned size); + + gzip(); + ~gzip(); +}; + +bool gzip::decompress(const string &filename) { + uint8_t *data; + unsigned size; + if(file::read(filename, data, size) == false) return false; + bool result = decompress(data, size); + delete[] data; + return result; +} + +bool gzip::decompress(const uint8_t *data, unsigned size) { + if(size < 18) return false; + if(data[0] != 0x1f) return false; + if(data[1] != 0x8b) return false; + unsigned cm = data[2]; + unsigned flg = data[3]; + unsigned mtime = data[4]; + mtime |= data[5] << 8; + mtime |= data[6] << 16; + mtime |= data[7] << 24; + unsigned xfl = data[8]; + unsigned os = data[9]; + unsigned p = 10; + unsigned isize = data[size - 4]; + isize |= data[size - 3] << 8; + isize |= data[size - 2] << 16; + isize |= data[size - 1] << 24; + filename = ""; + + if(flg & 0x04) { //FEXTRA + unsigned xlen = data[p + 0]; + xlen |= data[p + 1] << 8; + p += 2 + xlen; + } + + if(flg & 0x08) { //FNAME + char buffer[PATH_MAX]; + for(unsigned n = 0; n < PATH_MAX; n++, p++) { + buffer[n] = data[p]; + if(data[p] == 0) break; + } + if(data[p++]) return false; + filename = buffer; + } + + if(flg & 0x10) { //FCOMMENT + while(data[p++]); + } + + if(flg & 0x02) { //FHCRC + p += 2; + } + + this->size = isize; + this->data = new uint8_t[this->size]; + return inflate(this->data, this->size, data + p, size - p - 8); +} + +gzip::gzip() : data(0) { +} + +gzip::~gzip() { + if(data) delete[] data; +} + +} + +#endif diff --git a/snesfilter/nall/http.hpp b/snesfilter/nall/http.hpp new file mode 100755 index 00000000..1b2eab4f --- /dev/null +++ b/snesfilter/nall/http.hpp @@ -0,0 +1,176 @@ +#ifndef NALL_HTTP_HPP +#define NALL_HTTP_HPP + +#if !defined(_WIN32) + #include + #include + #include + #include +#else + #include + #include + #include +#endif + +#include +#include + +namespace nall { + +struct http { + string hostname; + addrinfo *serverinfo; + int serversocket; + string header; + + inline void download(const string &path, uint8_t *&data, unsigned &size) { + data = 0; + size = 0; + + send({ + "GET ", path, " HTTP/1.1\r\n" + "Host: ", hostname, "\r\n" + "Connection: close\r\n" + "\r\n" + }); + + header = downloadHeader(); + downloadContent(data, size); + } + + inline bool connect(string host, unsigned port) { + hostname = host; + + addrinfo hints; + memset(&hints, 0, sizeof(addrinfo)); + hints.ai_family = AF_UNSPEC; + hints.ai_socktype = SOCK_STREAM; + hints.ai_flags = AI_PASSIVE; + + int status = getaddrinfo(hostname, string(port), &hints, &serverinfo); + if(status != 0) return false; + + serversocket = socket(serverinfo->ai_family, serverinfo->ai_socktype, serverinfo->ai_protocol); + if(serversocket == -1) return false; + + int result = ::connect(serversocket, serverinfo->ai_addr, serverinfo->ai_addrlen); + if(result == -1) return false; + + return true; + } + + inline bool send(const string &data) { + return send((const uint8_t*)(const char*)data, data.length()); + } + + inline bool send(const uint8_t *data, unsigned size) { + while(size) { + int length = ::send(serversocket, (const char*)data, size, 0); + if(length == -1) return false; + data += length; + size -= length; + } + return true; + } + + inline string downloadHeader() { + string output; + do { + char buffer[2]; + int length = recv(serversocket, buffer, 1, 0); + if(length <= 0) return output; + buffer[1] = 0; + output.append(buffer); + } while(output.endswith("\r\n\r\n") == false); + return output; + } + + inline string downloadChunkLength() { + string output; + do { + char buffer[2]; + int length = recv(serversocket, buffer, 1, 0); + if(length <= 0) return output; + buffer[1] = 0; + output.append(buffer); + } while(output.endswith("\r\n") == false); + return output; + } + + inline void downloadContent(uint8_t *&data, unsigned &size) { + unsigned capacity = 0; + + if(header.iposition("\r\nTransfer-Encoding: chunked\r\n")) { + while(true) { + unsigned length = hex(downloadChunkLength()); + if(length == 0) break; + capacity += length; + data = (uint8_t*)realloc(data, capacity); + + char buffer[length]; + while(length) { + int packetlength = recv(serversocket, buffer, length, 0); + if(packetlength <= 0) break; + memcpy(data + size, buffer, packetlength); + size += packetlength; + length -= packetlength; + } + } + } else if(auto position = header.iposition("\r\nContent-Length: ")) { + unsigned length = decimal((const char*)header + position() + 16); + while(length) { + char buffer[256]; + int packetlength = recv(serversocket, buffer, min(256, length), 0); + if(packetlength <= 0) break; + capacity += packetlength; + data = (uint8_t*)realloc(data, capacity); + memcpy(data + size, buffer, packetlength); + size += packetlength; + length -= packetlength; + } + } else { + while(true) { + char buffer[256]; + int packetlength = recv(serversocket, buffer, 256, 0); + if(packetlength <= 0) break; + capacity += packetlength; + data = (uint8_t*)realloc(data, capacity); + memcpy(data + size, buffer, packetlength); + size += packetlength; + } + } + + data = (uint8_t*)realloc(data, capacity + 1); + data[capacity] = 0; + } + + inline void disconnect() { + close(serversocket); + freeaddrinfo(serverinfo); + serverinfo = 0; + serversocket = -1; + } + + #ifdef _WIN32 + inline int close(int sock) { + return closesocket(sock); + } + + inline http() { + int sock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); + if(sock == INVALID_SOCKET && WSAGetLastError() == WSANOTINITIALISED) { + WSADATA wsaData; + if(WSAStartup(MAKEWORD(2, 2), &wsaData) != 0) { + WSACleanup(); + return; + } + } else { + close(sock); + } + } + #endif +}; + +} + +#endif diff --git a/snesfilter/nall/inflate.hpp b/snesfilter/nall/inflate.hpp new file mode 100755 index 00000000..c989e3f1 --- /dev/null +++ b/snesfilter/nall/inflate.hpp @@ -0,0 +1,358 @@ +#ifndef NALL_INFLATE_HPP +#define NALL_INFLATE_HPP + +#include + +namespace nall { + +namespace puff { + inline int puff( + unsigned char *dest, unsigned long *destlen, + unsigned char *source, unsigned long *sourcelen + ); +} + +inline bool inflate( + uint8_t *target, unsigned targetLength, + const uint8_t *source, unsigned sourceLength +) { + unsigned long tl = targetLength, sl = sourceLength; + int result = puff::puff((unsigned char*)target, &tl, (unsigned char*)source, &sl); + return result == 0; +} + +namespace puff { + +//zlib/contrib/puff.c +//version 2.1* +//author: Mark Adler +//license: zlib +//ported by: byuu + +//* I have corrected a bug in fixed(), where it was accessing uninitialized +// memory: calling construct() with lencode prior to initializing lencode.count + +enum { + MAXBITS = 15, + MAXLCODES = 286, + MAXDCODES = 30, + FIXLCODES = 288, + MAXCODES = MAXLCODES + MAXDCODES, +}; + +struct state { + unsigned char *out; + unsigned long outlen; + unsigned long outcnt; + + unsigned char *in; + unsigned long inlen; + unsigned long incnt; + int bitbuf; + int bitcnt; + + jmp_buf env; +}; + +struct huffman { + short *count; + short *symbol; +}; + +inline int bits(state *s, int need) { + long val; + + val = s->bitbuf; + while(s->bitcnt < need) { + if(s->incnt == s->inlen) longjmp(s->env, 1); + val |= (long)(s->in[s->incnt++]) << s->bitcnt; + s->bitcnt += 8; + } + + s->bitbuf = (int)(val >> need); + s->bitcnt -= need; + + return (int)(val & ((1L << need) - 1)); +} + +inline int stored(state *s) { + unsigned len; + + s->bitbuf = 0; + s->bitcnt = 0; + + if(s->incnt + 4 > s->inlen) return 2; + len = s->in[s->incnt++]; + len |= s->in[s->incnt++] << 8; + if(s->in[s->incnt++] != (~len & 0xff) || + s->in[s->incnt++] != ((~len >> 8) & 0xff) + ) return 2; + + if(s->incnt + len > s->inlen) return 2; + if(s->out != 0) { + if(s->outcnt + len > s->outlen) return 1; + while(len--) s->out[s->outcnt++] = s->in[s->incnt++]; + } else { + s->outcnt += len; + s->incnt += len; + } + + return 0; +} + +inline int decode(state *s, huffman *h) { + int len, code, first, count, index, bitbuf, left; + short *next; + + bitbuf = s->bitbuf; + left = s->bitcnt; + code = first = index = 0; + len = 1; + next = h->count + 1; + while(true) { + while(left--) { + code |= bitbuf & 1; + bitbuf >>= 1; + count = *next++; + if(code - count < first) { + s->bitbuf = bitbuf; + s->bitcnt = (s->bitcnt - len) & 7; + return h->symbol[index + (code - first)]; + } + index += count; + first += count; + first <<= 1; + code <<= 1; + len++; + } + left = (MAXBITS + 1) - len; + if(left == 0) break; + if(s->incnt == s->inlen) longjmp(s->env, 1); + bitbuf = s->in[s->incnt++]; + if(left > 8) left = 8; + } + + return -10; +} + +inline int construct(huffman *h, short *length, int n) { + int symbol, len, left; + short offs[MAXBITS + 1]; + + for(len = 0; len <= MAXBITS; len++) h->count[len] = 0; + for(symbol = 0; symbol < n; symbol++) h->count[length[symbol]]++; + if(h->count[0] == n) return 0; + + left = 1; + for(len = 1; len <= MAXBITS; len++) { + left <<= 1; + left -= h->count[len]; + if(left < 0) return left; + } + + offs[1] = 0; + for(len = 1; len < MAXBITS; len++) offs[len + 1] = offs[len] + h->count[len]; + + for(symbol = 0; symbol < n; symbol++) { + if(length[symbol] != 0) h->symbol[offs[length[symbol]]++] = symbol; + } + + return left; +} + +inline int codes(state *s, huffman *lencode, huffman *distcode) { + int symbol, len; + unsigned dist; + static const short lens[29] = { + 3, 4, 5, 6, 7, 8, 9, 10, 11, 13, 15, 17, 19, 23, 27, 31, + 35, 43, 51, 59, 67, 83, 99, 115, 131, 163, 195, 227, 258 + }; + static const short lext[29] = { + 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 2, 2, 2, 2, + 3, 3, 3, 3, 4, 4, 4, 4, 5, 5, 5, 5, 0 + }; + static const short dists[30] = { + 1, 2, 3, 4, 5, 7, 9, 13, 17, 25, 33, 49, 65, 97, 129, 193, + 257, 385, 513, 769, 1025, 1537, 2049, 3073, 4097, 6145, + 8193, 12289, 16385, 24577 + }; + static const short dext[30] = { + 0, 0, 0, 0, 1, 1, 2, 2, 3, 3, 4, 4, 5, 5, 6, 6, + 7, 7, 8, 8, 9, 9, 10, 10, 11, 11, + 12, 12, 13, 13 + }; + + do { + symbol = decode(s, lencode); + if(symbol < 0) return symbol; + if(symbol < 256) { + if(s->out != 0) { + if(s->outcnt == s->outlen) return 1; + s->out[s->outcnt] = symbol; + } + s->outcnt++; + } else if(symbol > 256) { + symbol -= 257; + if(symbol >= 29) return -10; + len = lens[symbol] + bits(s, lext[symbol]); + + symbol = decode(s, distcode); + if(symbol < 0) return symbol; + dist = dists[symbol] + bits(s, dext[symbol]); +#ifndef INFLATE_ALLOW_INVALID_DISTANCE_TOO_FAR + if(dist > s->outcnt) return -11; +#endif + + if(s->out != 0) { + if(s->outcnt + len > s->outlen) return 1; + while(len--) { + s->out[s->outcnt] = +#ifdef INFLATE_ALLOW_INVALID_DISTANCE_TOO_FAR + dist > s->outcnt ? 0 : +#endif + s->out[s->outcnt - dist]; + s->outcnt++; + } + } else { + s->outcnt += len; + } + } + } while(symbol != 256); + + return 0; +} + +inline int fixed(state *s) { + static int virgin = 1; + static short lencnt[MAXBITS + 1], lensym[FIXLCODES]; + static short distcnt[MAXBITS + 1], distsym[MAXDCODES]; + static huffman lencode, distcode; + + if(virgin) { + int symbol = 0; + short lengths[FIXLCODES]; + + lencode.count = lencnt; + lencode.symbol = lensym; + distcode.count = distcnt; + distcode.symbol = distsym; + + for(; symbol < 144; symbol++) lengths[symbol] = 8; + for(; symbol < 256; symbol++) lengths[symbol] = 9; + for(; symbol < 280; symbol++) lengths[symbol] = 7; + for(; symbol < FIXLCODES; symbol++) lengths[symbol] = 8; + construct(&lencode, lengths, FIXLCODES); + + for(symbol = 0; symbol < MAXDCODES; symbol++) lengths[symbol] = 5; + construct(&distcode, lengths, MAXDCODES); + + virgin = 0; + } + + return codes(s, &lencode, &distcode); +} + +inline int dynamic(state *s) { + int nlen, ndist, ncode, index, err; + short lengths[MAXCODES]; + short lencnt[MAXBITS + 1], lensym[MAXLCODES]; + short distcnt[MAXBITS + 1], distsym[MAXDCODES]; + huffman lencode, distcode; + static const short order[19] = { + 16, 17, 18, 0, 8, 7, 9, 6, 10, 5, 11, 4, 12, 3, 13, 2, 14, 1, 15 + }; + + lencode.count = lencnt; + lencode.symbol = lensym; + distcode.count = distcnt; + distcode.symbol = distsym; + + nlen = bits(s, 5) + 257; + ndist = bits(s, 5) + 1; + ncode = bits(s, 4) + 4; + if(nlen > MAXLCODES || ndist > MAXDCODES) return -3; + + for(index = 0; index < ncode; index++) lengths[order[index]] = bits(s, 3); + for(; index < 19; index++) lengths[order[index]] = 0; + + err = construct(&lencode, lengths, 19); + if(err != 0) return -4; + + index = 0; + while(index < nlen + ndist) { + int symbol, len; + + symbol = decode(s, &lencode); + if(symbol < 16) { + lengths[index++] = symbol; + } else { + len = 0; + if(symbol == 16) { + if(index == 0) return -5; + len = lengths[index - 1]; + symbol = 3 + bits(s, 2); + } else if(symbol == 17) { + symbol = 3 + bits(s, 3); + } else { + symbol = 11 + bits(s, 7); + } + if(index + symbol > nlen + ndist) return -6; + while(symbol--) lengths[index++] = len; + } + } + + if(lengths[256] == 0) return -9; + + err = construct(&lencode, lengths, nlen); + if(err < 0 || (err > 0 && nlen - lencode.count[0] != 1)) return -7; + + err = construct(&distcode, lengths + nlen, ndist); + if(err < 0 || (err > 0 && ndist - distcode.count[0] != 1)) return -8; + + return codes(s, &lencode, &distcode); +} + +inline int puff( + unsigned char *dest, unsigned long *destlen, + unsigned char *source, unsigned long *sourcelen +) { + state s; + int last, type, err; + + s.out = dest; + s.outlen = *destlen; + s.outcnt = 0; + + s.in = source; + s.inlen = *sourcelen; + s.incnt = 0; + s.bitbuf = 0; + s.bitcnt = 0; + + if(setjmp(s.env) != 0) { + err = 2; + } else { + do { + last = bits(&s, 1); + type = bits(&s, 2); + err = type == 0 ? stored(&s) + : type == 1 ? fixed(&s) + : type == 2 ? dynamic(&s) + : -1; + if(err != 0) break; + } while(!last); + } + + if(err <= 0) { + *destlen = s.outcnt; + *sourcelen = s.incnt; + } + + return err; +} + +} + +} + +#endif diff --git a/snesfilter/nall/input.hpp b/snesfilter/nall/input.hpp index 1fd680f4..cd765393 100755 --- a/snesfilter/nall/input.hpp +++ b/snesfilter/nall/input.hpp @@ -110,7 +110,7 @@ struct Keyboard { break; } } - return string() << "KB" << ID << "::" << KeyboardScancodeName[index]; + return { "KB", ID, "::", KeyboardScancodeName[index] }; } uint16_t operator[](Scancode code) const { return Base + ID * Size + code; } @@ -207,7 +207,7 @@ struct Mouse { break; } } - return string() << "MS" << ID << "::" << MouseScancodeName[index]; + return { "MS", ID, "::", MouseScancodeName[index] }; } uint16_t operator[](Scancode code) const { return Base + ID * Size + code; } @@ -330,7 +330,7 @@ struct Joypad { index = code - (Base + Size * i); } } - return string() << "JP" << ID << "::" << JoypadScancodeName[index]; + return { "JP", ID, "::", JoypadScancodeName[index] }; } uint16_t operator[](Scancode code) const { return Base + ID * Size + code; } diff --git a/snesfilter/nall/ips.hpp b/snesfilter/nall/ips.hpp new file mode 100755 index 00000000..87c7de25 --- /dev/null +++ b/snesfilter/nall/ips.hpp @@ -0,0 +1,110 @@ +#ifndef NALL_IPS_HPP +#define NALL_IPS_HPP + +#include +#include +#include + +namespace nall { + +struct ips { + inline bool apply(); + inline void source(const uint8_t *data, unsigned size); + inline void modify(const uint8_t *data, unsigned size); + inline bool source(const string &filename); + inline bool modify(const string &filename); + inline ips(); + inline ~ips(); + + uint8_t *data; + unsigned size; + const uint8_t *sourceData; + unsigned sourceSize; + const uint8_t *modifyData; + unsigned modifySize; +}; + +bool ips::apply() { + if(modifySize < 8) return false; + if(modifyData[0] != 'P') return false; + if(modifyData[1] != 'A') return false; + if(modifyData[2] != 'T') return false; + if(modifyData[3] != 'C') return false; + if(modifyData[4] != 'H') return false; + + if(data) delete[] data; + data = new uint8_t[16 * 1024 * 1024 + 65536](); //maximum size of IPS patch + single-tag padding + size = sourceSize; + memcpy(data, sourceData, sourceSize); + unsigned offset = 5; + + while(true) { + unsigned address, length; + + if(offset > modifySize - 3) break; + address = modifyData[offset++] << 16; + address |= modifyData[offset++] << 8; + address |= modifyData[offset++] << 0; + + if(address == 0x454f46) { //EOF + if(offset == modifySize) return true; + if(offset == modifySize - 3) { + size = modifyData[offset++] << 16; + size |= modifyData[offset++] << 8; + size |= modifyData[offset++] << 0; + return true; + } + } + + if(offset > modifySize - 2) break; + length = modifyData[offset++] << 8; + length |= modifyData[offset++] << 0; + + if(length) { //Copy + if(offset > modifySize - length) break; + while(length--) data[address++] = modifyData[offset++]; + } else { //RLE + if(offset > modifySize - 3) break; + length = modifyData[offset++] << 8; + length |= modifyData[offset++] << 0; + if(length == 0) break; //illegal + while(length--) data[address++] = modifyData[offset]; + offset++; + } + + size = max(size, address); + } + + delete[] data; + data = 0; + return false; +} + +void ips::source(const uint8_t *data, unsigned size) { + sourceData = data, sourceSize = size; +} + +void ips::modify(const uint8_t *data, unsigned size) { + modifyData = data, modifySize = size; +} + +bool ips::source(const string &filename) { + return file::read(filename, sourceData, sourceSize); +} + +bool ips::modify(const string &filename) { + return file::read(filename, modifyData, modifySize); +} + +ips::ips() : data(0), sourceData(0), modifyData(0) { +} + +ips::~ips() { + if(data) delete[] data; + if(sourceData) delete[] sourceData; + if(modifyData) delete[] modifyData; +} + +} + +#endif diff --git a/snesfilter/nall/lzss.hpp b/snesfilter/nall/lzss.hpp index 202bc814..147e1e62 100755 --- a/snesfilter/nall/lzss.hpp +++ b/snesfilter/nall/lzss.hpp @@ -1,81 +1,165 @@ #ifndef NALL_LZSS_HPP #define NALL_LZSS_HPP -#include -#include +#include +#include #include +#include namespace nall { - class lzss { - public: - static bool encode(uint8_t *&output, unsigned &outlength, const uint8_t *input, unsigned inlength) { - output = new(zeromemory) uint8_t[inlength * 9 / 8 + 9]; - unsigned i = 0, o = 0; - while(i < inlength) { - unsigned flagoffset = o++; - uint8_t flag = 0x00; +//19:5 pulldown +//8:1 marker: d7-d0 +//length: { 4 - 35 }, offset: { 1 - 0x80000 } +//4-byte file size header +//little-endian encoding +struct lzss { + inline void source(const uint8_t *data, unsigned size); + inline bool source(const string &filename); + inline unsigned size() const; + inline bool compress(const string &filename); + inline bool decompress(uint8_t *targetData, unsigned targetSize); + inline bool decompress(const string &filename); - for(unsigned b = 0; b < 8 && i < inlength; b++) { - unsigned longest = 0, pointer; - for(unsigned index = 1; index < 4096; index++) { - unsigned count = 0; - while(true) { - if(count >= 15 + 3) break; //verify pattern match is not longer than max length - if(i + count >= inlength) break; //verify pattern match does not read past end of input - if(i + count < index) break; //verify read is not before start of input - if(input[i + count] != input[i + count - index]) break; //verify pattern still matches - count++; - } +protected: + struct Node { + unsigned offset; + Node *next; + inline Node() : offset(0), next(0) {} + inline ~Node() { if(next) delete next; } + } *tree[65536]; - if(count > longest) { - longest = count; - pointer = index; - } - } + filemap sourceFile; + const uint8_t *sourceData; + unsigned sourceSize; - if(longest < 3) output[o++] = input[i++]; - else { - flag |= 1 << b; - uint16_t x = ((longest - 3) << 12) + pointer; - output[o++] = x; - output[o++] = x >> 8; - i += longest; - } +public: + inline lzss() : sourceData(0), sourceSize(0) {} +}; + +void lzss::source(const uint8_t *data, unsigned size) { + sourceData = data; + sourceSize = size; +} + +bool lzss::source(const string &filename) { + if(sourceFile.open(filename, filemap::mode::read) == false) return false; + sourceData = sourceFile.data(); + sourceSize = sourceFile.size(); + return true; +} + +unsigned lzss::size() const { + unsigned size = 0; + if(sourceSize < 4) return size; + for(unsigned n = 0; n < 32; n += 8) size |= sourceData[n >> 3] << n; + return size; +} + +bool lzss::compress(const string &filename) { + file targetFile; + if(targetFile.open(filename, file::mode::write) == false) return false; + + for(unsigned n = 0; n < 32; n += 8) targetFile.write(sourceSize >> n); + for(unsigned n = 0; n < 65536; n++) tree[n] = 0; + + uint8_t buffer[25]; + unsigned sourceOffset = 0; + + while(sourceOffset < sourceSize) { + uint8_t mask = 0x00; + unsigned bufferOffset = 1; + + for(unsigned iteration = 0; iteration < 8; iteration++) { + if(sourceOffset >= sourceSize) break; + + uint16_t symbol = sourceData[sourceOffset + 0]; + if(sourceOffset < sourceSize - 1) symbol |= sourceData[sourceOffset + 1] << 8; + Node *node = tree[symbol]; + unsigned maxLength = 0, maxOffset = 0; + + while(node) { + if(node->offset < sourceOffset - 0x80000) { + //out-of-range: all subsequent nodes will also be, so free up their memory + if(node->next) { delete node->next; node->next = 0; } + break; } - output[flagoffset] = flag; + unsigned length = 0, x = sourceOffset, y = node->offset; + while(length < 35 && x < sourceSize && sourceData[x++] == sourceData[y++]) length++; + if(length > maxLength) maxLength = length, maxOffset = node->offset; + if(length == 35) break; + + node = node->next; } - outlength = o; - return true; - } + //attach current symbol to top of tree for subsequent searches + node = new Node; + node->offset = sourceOffset; + node->next = tree[symbol]; + tree[symbol] = node; - static bool decode(uint8_t *&output, const uint8_t *input, unsigned length) { - output = new(zeromemory) uint8_t[length]; - - unsigned i = 0, o = 0; - while(o < length) { - uint8_t flag = input[i++]; - - for(unsigned b = 0; b < 8 && o < length; b++) { - if(!(flag & (1 << b))) output[o++] = input[i++]; - else { - uint16_t offset = input[i++]; - offset += input[i++] << 8; - uint16_t lookuplength = (offset >> 12) + 3; - offset &= 4095; - for(unsigned index = 0; index < lookuplength && o + index < length; index++) { - output[o + index] = output[o + index - offset]; - } - o += lookuplength; - } - } + if(maxLength < 4) { + buffer[bufferOffset++] = sourceData[sourceOffset++]; + } else { + unsigned output = ((maxLength - 4) << 19) | (sourceOffset - 1 - maxOffset); + for(unsigned n = 0; n < 24; n += 8) buffer[bufferOffset++] = output >> n; + mask |= 0x80 >> iteration; + sourceOffset += maxLength; } - - return true; } - }; + + buffer[0] = mask; + targetFile.write(buffer, bufferOffset); + } + + sourceFile.close(); + targetFile.close(); + return true; +} + +bool lzss::decompress(uint8_t *targetData, unsigned targetSize) { + if(targetSize < size()) return false; + + unsigned sourceOffset = 4, targetOffset = 0; + while(sourceOffset < sourceSize) { + uint8_t mask = sourceData[sourceOffset++]; + + for(unsigned iteration = 0; iteration < 8; iteration++) { + if(sourceOffset >= sourceSize) break; + + if((mask & (0x80 >> iteration)) == 0) { + targetData[targetOffset++] = sourceData[sourceOffset++]; + } else { + unsigned code = 0; + for(unsigned n = 0; n < 24; n += 8) code |= sourceData[sourceOffset++] << n; + unsigned length = (code >> 19) + 4; + unsigned offset = targetOffset - 1 - (code & 0x7ffff); + while(length--) targetData[targetOffset++] = targetData[offset++]; + } + } + } +} + +bool lzss::decompress(const string &filename) { + if(sourceSize < 4) return false; + unsigned targetSize = size(); + + file fp; + if(fp.open(filename, file::mode::write) == false) return false; + fp.truncate(targetSize); + fp.close(); + + filemap targetFile; + if(targetFile.open(filename, filemap::mode::readwrite) == false) return false; + uint8_t *targetData = targetFile.data(); + + bool result = decompress(targetData, targetSize); + sourceFile.close(); + targetFile.close(); + return result; +} + } #endif diff --git a/snesfilter/nall/platform.hpp b/snesfilter/nall/platform.hpp index 72eeec09..539b2345 100755 --- a/snesfilter/nall/platform.hpp +++ b/snesfilter/nall/platform.hpp @@ -1,7 +1,12 @@ #ifndef NALL_PLATFORM_HPP #define NALL_PLATFORM_HPP -#include +#if defined(_WIN32) + //minimum version needed for _wstat64, etc + #undef __MSVCRT_VERSION__ + #define __MSVCRT_VERSION__ 0x0601 + #include +#endif //========================= //standard platform headers @@ -18,16 +23,19 @@ #include #include +#include +#include + #if defined(_WIN32) #include #include #include + #include #undef interface #define dllexport __declspec(dllexport) #else #include #include - #include #define dllexport #endif @@ -53,11 +61,11 @@ #if defined(_WIN32) #define getcwd _getcwd #define ftruncate _chsize - #define putenv _putenv #define mkdir(n, m) _wmkdir(nall::utf16_t(n)) + #define putenv _putenv #define rmdir _rmdir - #define vsnprintf _vsnprintf #define usleep(n) Sleep(n / 1000) + #define vsnprintf _vsnprintf #endif //================ @@ -87,6 +95,7 @@ wchar_t fn[_MAX_PATH] = L""; _wfullpath(fn, nall::utf16_t(filename), _MAX_PATH); strcpy(resolvedname, nall::utf8_t(fn)); + for(unsigned n = 0; resolvedname[n]; n++) if(resolvedname[n] == '\\') resolvedname[n] = '/'; return resolvedname; } @@ -94,6 +103,7 @@ wchar_t fp[_MAX_PATH] = L""; SHGetFolderPathW(0, CSIDL_APPDATA | CSIDL_FLAG_CREATE, 0, 0, fp); strcpy(path, nall::utf8_t(fp)); + for(unsigned n = 0; path[n]; n++) if(path[n] == '\\') path[n] = '/'; return path; } @@ -101,6 +111,7 @@ wchar_t fp[_MAX_PATH] = L""; _wgetcwd(fp, _MAX_PATH); strcpy(path, nall::utf8_t(fp)); + for(unsigned n = 0; path[n]; n++) if(path[n] == '\\') path[n] = '/'; return path; } #else @@ -110,11 +121,16 @@ *path = 0; struct passwd *userinfo = getpwuid(getuid()); if(userinfo) strcpy(path, userinfo->pw_dir); + unsigned length = strlen(path); + if(path[length] != '/') strcpy(path + length, "/"); return path; } inline char *getcwd(char *path) { - return getcwd(path, PATH_MAX); + auto unused = getcwd(path, PATH_MAX); + unsigned length = strlen(path); + if(path[length] != '/') strcpy(path + length, "/"); + return path; } #endif diff --git a/snesfilter/nall/png.hpp b/snesfilter/nall/png.hpp new file mode 100755 index 00000000..025044b2 --- /dev/null +++ b/snesfilter/nall/png.hpp @@ -0,0 +1,426 @@ +#ifndef NALL_PNG_HPP +#define NALL_PNG_HPP + +//PNG image decoder +//author: byuu + +#include +#include + +namespace nall { + +struct png { + uint32_t *data; + unsigned size; + + struct Info { + unsigned width; + unsigned height; + unsigned bitDepth; + unsigned colorType; + unsigned compressionMethod; + unsigned filterType; + unsigned interlaceMethod; + + unsigned bytesPerPixel; + unsigned pitch; + + uint8_t palette[256][3]; + } info; + + uint8_t *rawData; + unsigned rawSize; + + inline bool decode(const string &filename); + inline bool decode(const uint8_t *sourceData, unsigned sourceSize); + inline void transform(); + inline void alphaTransform(uint32_t rgb = 0xffffff); + inline png(); + inline ~png(); + +protected: + enum class FourCC : unsigned { + IHDR = 0x49484452, + PLTE = 0x504c5445, + IDAT = 0x49444154, + IEND = 0x49454e44, + }; + + unsigned bitpos; + + inline unsigned interlace(unsigned pass, unsigned index); + inline unsigned inflateSize(); + inline bool deinterlace(const uint8_t *&inputData, unsigned pass); + inline bool filter(uint8_t *outputData, const uint8_t *inputData, unsigned width, unsigned height); + inline unsigned read(const uint8_t *data, unsigned length); + inline unsigned decode(const uint8_t *&data); + inline unsigned readbits(const uint8_t *&data); + inline unsigned scale(unsigned n); +}; + +bool png::decode(const string &filename) { + uint8_t *data; + unsigned size; + if(file::read(filename, data, size) == false) return false; + bool result = decode(data, size); + delete[] data; + return result; +} + +bool png::decode(const uint8_t *sourceData, unsigned sourceSize) { + if(sourceSize < 8) return false; + if(read(sourceData + 0, 4) != 0x89504e47) return false; + if(read(sourceData + 4, 4) != 0x0d0a1a0a) return false; + + uint8_t *compressedData = 0; + unsigned compressedSize = 0; + + unsigned offset = 8; + while(offset < sourceSize) { + unsigned length = read(sourceData + offset + 0, 4); + unsigned fourCC = read(sourceData + offset + 4, 4); + unsigned checksum = read(sourceData + offset + 8 + length, 4); + + if(fourCC == (unsigned)FourCC::IHDR) { + info.width = read(sourceData + offset + 8, 4); + info.height = read(sourceData + offset + 12, 4); + info.bitDepth = read(sourceData + offset + 16, 1); + info.colorType = read(sourceData + offset + 17, 1); + info.compressionMethod = read(sourceData + offset + 18, 1); + info.filterType = read(sourceData + offset + 19, 1); + info.interlaceMethod = read(sourceData + offset + 20, 1); + + if(info.bitDepth == 0 || info.bitDepth > 16) return false; + if(info.bitDepth & (info.bitDepth - 1)) return false; //not a power of two + if(info.compressionMethod != 0) return false; + if(info.filterType != 0) return false; + if(info.interlaceMethod != 0 && info.interlaceMethod != 1) return false; + + switch(info.colorType) { + case 0: info.bytesPerPixel = info.bitDepth * 1; break; //L + case 2: info.bytesPerPixel = info.bitDepth * 3; break; //R,G,B + case 3: info.bytesPerPixel = info.bitDepth * 1; break; //P + case 4: info.bytesPerPixel = info.bitDepth * 2; break; //L,A + case 6: info.bytesPerPixel = info.bitDepth * 4; break; //R,G,B,A + default: return false; + } + + if(info.colorType == 2 || info.colorType == 4 || info.colorType == 6) + if(info.bitDepth != 8 && info.bitDepth != 16) return false; + if(info.colorType == 3 && info.bitDepth == 16) return false; + + info.bytesPerPixel = (info.bytesPerPixel + 7) / 8; + info.pitch = (int)info.width * info.bytesPerPixel; + } + + if(fourCC == (unsigned)FourCC::PLTE) { + if(length % 3) return false; + for(unsigned n = 0, p = offset + 8; n < length / 3; n++) { + info.palette[n][0] = sourceData[p++]; + info.palette[n][1] = sourceData[p++]; + info.palette[n][2] = sourceData[p++]; + } + } + + if(fourCC == (unsigned)FourCC::IDAT) { + compressedData = (uint8_t*)realloc(compressedData, compressedSize + length); + memcpy(compressedData + compressedSize, sourceData + offset + 8, length); + compressedSize += length; + } + + if(fourCC == (unsigned)FourCC::IEND) { + break; + } + + offset += 4 + 4 + length + 4; + } + + unsigned interlacedSize = inflateSize(); + uint8_t *interlacedData = new uint8_t[interlacedSize]; + + bool result = inflate(interlacedData, interlacedSize, compressedData + 2, compressedSize - 6); + delete[] compressedData; + + if(result == false) { + delete[] interlacedData; + return false; + } + + rawSize = info.width * info.height * info.bytesPerPixel; + rawData = new uint8_t[rawSize]; + + if(info.interlaceMethod == 0) { + if(filter(rawData, interlacedData, info.width, info.height) == false) { + delete[] interlacedData; + delete[] rawData; + rawData = 0; + return false; + } + } else { + const uint8_t *passData = interlacedData; + for(unsigned pass = 0; pass < 7; pass++) { + if(deinterlace(passData, pass) == false) { + delete[] interlacedData; + delete[] rawData; + rawData = 0; + return false; + } + } + } + + delete[] interlacedData; + return true; +} + +unsigned png::interlace(unsigned pass, unsigned index) { + static const unsigned data[7][4] = { + //x-distance, y-distance, x-origin, y-origin + { 8, 8, 0, 0 }, + { 8, 8, 4, 0 }, + { 4, 8, 0, 4 }, + { 4, 4, 2, 0 }, + { 2, 4, 0, 2 }, + { 2, 2, 1, 0 }, + { 1, 2, 0, 1 }, + }; + return data[pass][index]; +} + +unsigned png::inflateSize() { + if(info.interlaceMethod == 0) { + return info.width * info.height * info.bytesPerPixel + info.height; + } + + unsigned size = 0; + for(unsigned pass = 0; pass < 7; pass++) { + unsigned xd = interlace(pass, 0), yd = interlace(pass, 1); + unsigned xo = interlace(pass, 2), yo = interlace(pass, 3); + unsigned width = (info.width + (xd - xo - 1)) / xd; + unsigned height = (info.height + (yd - yo - 1)) / yd; + if(width == 0 || height == 0) continue; + size += width * height * info.bytesPerPixel + height; + } + return size; +} + +bool png::deinterlace(const uint8_t *&inputData, unsigned pass) { + unsigned xd = interlace(pass, 0), yd = interlace(pass, 1); + unsigned xo = interlace(pass, 2), yo = interlace(pass, 3); + unsigned width = (info.width + (xd - xo - 1)) / xd; + unsigned height = (info.height + (yd - yo - 1)) / yd; + if(width == 0 || height == 0) return true; + + unsigned outputSize = width * height * info.bytesPerPixel; + uint8_t *outputData = new uint8_t[outputSize]; + bool result = filter(outputData, inputData, width, height); + + const uint8_t *rd = outputData; + for(unsigned y = yo; y < info.height; y += yd) { + uint8_t *wr = rawData + y * info.pitch; + for(unsigned x = xo; x < info.width; x += xd) { + for(unsigned b = 0; b < info.bytesPerPixel; b++) { + wr[x * info.bytesPerPixel + b] = *rd++; + } + } + } + + inputData += outputSize + height; + delete[] outputData; + return result; +} + +bool png::filter(uint8_t *outputData, const uint8_t *inputData, unsigned width, unsigned height) { + uint8_t *wr = outputData; + const uint8_t *rd = inputData; + int bpp = info.bytesPerPixel, pitch = width * bpp; + for(int y = 0; y < height; y++) { + uint8_t filter = *rd++; + + switch(filter) { + case 0x00: //None + for(int x = 0; x < pitch; x++) { + wr[x] = rd[x]; + } + break; + + case 0x01: //Subtract + for(int x = 0; x < pitch; x++) { + wr[x] = rd[x] + (x - bpp < 0 ? 0 : wr[x - bpp]); + } + break; + + case 0x02: //Above + for(int x = 0; x < pitch; x++) { + wr[x] = rd[x] + (y - 1 < 0 ? 0 : wr[x - pitch]); + } + break; + + case 0x03: //Average + for(int x = 0; x < pitch; x++) { + short a = x - bpp < 0 ? 0 : wr[x - bpp]; + short b = y - 1 < 0 ? 0 : wr[x - pitch]; + + wr[x] = rd[x] + (uint8_t)((a + b) / 2); + } + break; + + case 0x04: //Paeth + for(int x = 0; x < pitch; x++) { + short a = x - bpp < 0 ? 0 : wr[x - bpp]; + short b = y - 1 < 0 ? 0 : wr[x - pitch]; + short c = x - bpp < 0 || y - 1 < 0 ? 0 : wr[x - pitch - bpp]; + + short p = a + b - c; + short pa = p > a ? p - a : a - p; + short pb = p > b ? p - b : b - p; + short pc = p > c ? p - c : c - p; + + uint8_t paeth = (uint8_t)((pa <= pb && pa <= pc) ? a : (pb <= pc) ? b : c); + + wr[x] = rd[x] + paeth; + } + break; + + default: //Invalid + return false; + } + + rd += pitch; + wr += pitch; + } + + return true; +} + +unsigned png::read(const uint8_t *data, unsigned length) { + unsigned result = 0; + while(length--) result = (result << 8) | (*data++); + return result; +} + +unsigned png::decode(const uint8_t *&data) { + unsigned p, r, g, b, a; + + switch(info.colorType) { + case 0: //L + r = g = b = scale(readbits(data)); + a = 0xff; + break; + case 2: //R,G,B + r = scale(readbits(data)); + g = scale(readbits(data)); + b = scale(readbits(data)); + a = 0xff; + break; + case 3: //P + p = readbits(data); + r = info.palette[p][0]; + g = info.palette[p][1]; + b = info.palette[p][2]; + a = 0xff; + break; + case 4: //L,A + r = g = b = scale(readbits(data)); + a = scale(readbits(data)); + break; + case 6: //R,G,B,A + r = scale(readbits(data)); + g = scale(readbits(data)); + b = scale(readbits(data)); + a = scale(readbits(data)); + break; + } + + return (a << 24) | (r << 16) | (g << 8) | (b << 0); +} + +unsigned png::readbits(const uint8_t *&data) { + unsigned result = 0; + switch(info.bitDepth) { + case 1: + result = (*data >> bitpos) & 1; + bitpos++; + if(bitpos == 8) { data++; bitpos = 0; } + break; + case 2: + result = (*data >> bitpos) & 3; + bitpos += 2; + if(bitpos == 8) { data++; bitpos = 0; } + break; + case 4: + result = (*data >> bitpos) & 15; + bitpos += 4; + if(bitpos == 8) { data++; bitpos = 0; } + break; + case 8: + result = *data++; + break; + case 16: + result = (data[0] << 8) | (data[1] << 0); + data += 2; + break; + } + return result; +} + +unsigned png::scale(unsigned n) { + switch(info.bitDepth) { + case 1: return n ? 0xff : 0x00; + case 2: return n * 0x55; + case 4: return n * 0x11; + case 8: return n; + case 16: return n >> 8; + } + return 0; +} + +void png::transform() { + if(data) delete[] data; + data = new uint32_t[info.width * info.height]; + + bitpos = 0; + const uint8_t *rd = rawData; + for(unsigned y = 0; y < info.height; y++) { + uint32_t *wr = data + y * info.width; + for(unsigned x = 0; x < info.width; x++) { + wr[x] = decode(rd); + } + } +} + +void png::alphaTransform(uint32_t rgb) { + transform(); + + uint8_t ir = rgb >> 16; + uint8_t ig = rgb >> 8; + uint8_t ib = rgb >> 0; + + uint32_t *p = data; + for(unsigned y = 0; y < info.height; y++) { + for(unsigned x = 0; x < info.width; x++) { + uint32_t pixel = *p; + uint8_t a = pixel >> 24; + uint8_t r = pixel >> 16; + uint8_t g = pixel >> 8; + uint8_t b = pixel >> 0; + + r = (r * a) + (ir * (255 - a)) >> 8; + g = (g * a) + (ig * (255 - a)) >> 8; + b = (b * a) + (ib * (255 - a)) >> 8; + + *p++ = (255 << 24) | (r << 16) | (g << 8) | (b << 0); + } + } +} + +png::png() : data(0), rawData(0) { +} + +png::~png() { + if(data) delete[] data; + if(rawData) delete[] rawData; +} + +} + +#endif diff --git a/snesfilter/nall/reference_array.hpp b/snesfilter/nall/reference_array.hpp index ac47c32b..77d06d86 100755 --- a/snesfilter/nall/reference_array.hpp +++ b/snesfilter/nall/reference_array.hpp @@ -36,10 +36,26 @@ namespace nall { buffersize = newsize; } - void append(const T data) { + bool append(const T data) { + for(unsigned index = 0; index < buffersize; index++) { + if(pool[index] == &data) return false; + } + unsigned index = buffersize++; if(index >= poolsize) resize(index + 1); pool[index] = &data; + return true; + } + + bool remove(const T data) { + for(unsigned index = 0; index < buffersize; index++) { + if(pool[index] == &data) { + for(unsigned i = index; i < buffersize - 1; i++) pool[i] = pool[i + 1]; + resize(buffersize - 1); + return true; + } + } + return false; } template reference_array(Args&... args) : pool(0), poolsize(0), buffersize(0) { diff --git a/snesfilter/nall/resource.hpp b/snesfilter/nall/resource.hpp new file mode 100755 index 00000000..f8fd5153 --- /dev/null +++ b/snesfilter/nall/resource.hpp @@ -0,0 +1,61 @@ +#ifndef NALL_RESOURCE_HPP +#define NALL_RESOURCE_HPP + +#include +#include + +namespace nall { + +struct resource { + //create resource with "zip -9 resource.zip resource" + static bool encode(const char *outputFilename, const char *inputFilename) { + file fp; + if(fp.open(inputFilename, file::mode::read) == false) return false; + unsigned size = fp.size(); + uint8_t *data = new uint8_t[size]; + fp.read(data, size); + fp.close(); + + fp.open(outputFilename, file::mode::write); + fp.print("static const uint8_t data[", size, "] = {\n"); + uint8_t *p = data; + while(size) { + fp.print(" "); + for(unsigned n = 0; n < 32 && size; n++, size--) { + fp.print((unsigned)*p++, ","); + } + fp.print("\n"); + } + fp.print("};\n"); + fp.close(); + + delete[] data; + } + + uint8_t *data; + unsigned size; + + //extract first file from ZIP archive + bool decode(const uint8_t *cdata, unsigned csize) { + if(data) delete[] data; + + zip archive; + if(archive.open(cdata, csize) == false) return false; + if(archive.file.size() == 0) return false; + bool result = archive.extract(archive.file[0], data, size); + archive.close(); + + return result; + } + + resource() : data(0), size(0) { + } + + ~resource() { + if(data) delete[] data; + } +}; + +} + +#endif diff --git a/snesfilter/nall/sha256.hpp b/snesfilter/nall/sha256.hpp index 7f41f04e..c63367a7 100755 --- a/snesfilter/nall/sha256.hpp +++ b/snesfilter/nall/sha256.hpp @@ -3,6 +3,8 @@ //author: vladitx +#include + namespace nall { #define PTR(t, a) ((t*)(a)) @@ -49,7 +51,7 @@ namespace nall { uint64_t len; }; - void sha256_init(sha256_ctx *p) { + inline void sha256_init(sha256_ctx *p) { memset(p, 0, sizeof(sha256_ctx)); memcpy(p->h, T_H, sizeof(T_H)); } @@ -90,7 +92,7 @@ namespace nall { p->inlen = 0; } - void sha256_chunk(sha256_ctx *p, const uint8_t *s, unsigned len) { + inline void sha256_chunk(sha256_ctx *p, const uint8_t *s, unsigned len) { unsigned l; p->len += len; @@ -107,7 +109,7 @@ namespace nall { } } - void sha256_final(sha256_ctx *p) { + inline void sha256_final(sha256_ctx *p) { uint64_t len; p->in[p->inlen++] = 0x80; @@ -124,7 +126,7 @@ namespace nall { sha256_block(p); } - void sha256_hash(sha256_ctx *p, uint8_t *s) { + inline void sha256_hash(sha256_ctx *p, uint8_t *s) { uint32_t *t = (uint32_t*)s; for(unsigned i = 0; i < 8; i++) ST32BE(t++, p->h[i]); } diff --git a/snesfilter/nall/snes/cartridge.hpp b/snesfilter/nall/snes/cartridge.hpp index e3c0e0c5..6847ba3a 100755 --- a/snesfilter/nall/snes/cartridge.hpp +++ b/snesfilter/nall/snes/cartridge.hpp @@ -111,422 +111,426 @@ SNESCartridge::SNESCartridge(const uint8_t *data, unsigned size) { string xml = "\n"; if(type == TypeBsx) { - xml << ""; + xml.append(""); xmlMemoryMap = xml.transform("'", "\""); return; } if(type == TypeSufamiTurbo) { - xml << ""; + xml.append(""); xmlMemoryMap = xml.transform("'", "\""); return; } if(type == TypeGameBoy) { - xml << "\n"; + xml.append("\n"); if(gameboy_ram_size(data, size) > 0) { - xml << " \n"; + xml.append(" \n"); } - xml << "\n"; + xml.append("\n"); xmlMemoryMap = xml.transform("'", "\""); return; } - xml << "\n"; + xml.append(">\n"); if(type == TypeSuperGameBoy1Bios) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); } else if(type == TypeSuperGameBoy2Bios) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + } else if(has_cx4) { + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); } else if(has_spc7110) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); if(has_spc7110rtc) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); } - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); } else if(mapper == LoROM) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); if(ram_size > 0) { - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); if((rom_size > 0x200000) || (ram_size > 32 * 1024)) { - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); } else { - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); } - xml << " \n"; + xml.append(" \n"); } } else if(mapper == HiROM) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); if(ram_size > 0) { - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); if((rom_size > 0x200000) || (ram_size > 32 * 1024)) { - xml << " \n"; + xml.append(" \n"); } else { - xml << " \n"; + xml.append(" \n"); } - xml << " \n"; + xml.append(" \n"); } } else if(mapper == ExLoROM) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); if(ram_size > 0) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); } } else if(mapper == ExHiROM) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); if(ram_size > 0) { - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); if((rom_size > 0x200000) || (ram_size > 32 * 1024)) { - xml << " \n"; + xml.append(" \n"); } else { - xml << " \n"; + xml.append(" \n"); } - xml << " \n"; + xml.append(" \n"); } } else if(mapper == SuperFXROM) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); } else if(mapper == SA1ROM) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); } else if(mapper == BSCLoROM) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); } else if(mapper == BSCHiROM) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); } else if(mapper == BSXROM) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); } else if(mapper == STROM) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); } if(has_srtc) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); } if(has_sdd1) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - } - - if(has_cx4) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); } if(has_dsp1) { - xml << " \n"; + xml.append(" \n"); if(dsp1_mapper == DSP1LoROM1MB) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); } else if(dsp1_mapper == DSP1LoROM2MB) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); } else if(dsp1_mapper == DSP1HiROM) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); } - xml << " \n"; + xml.append(" \n"); } if(has_dsp2) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); } if(has_dsp3) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); } if(has_dsp4) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); } if(has_obc1) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); } if(has_st010) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); } if(has_st011) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); } if(has_st018) { - xml << " \n"; - xml << " \n"; - xml << " \n"; - xml << " \n"; + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); + xml.append(" \n"); } - xml << "\n"; + xml.append("\n"); xmlMemoryMap = xml.transform("'", "\""); } diff --git a/snesfilter/nall/stack.hpp b/snesfilter/nall/stack.hpp new file mode 100755 index 00000000..a4aacfa7 --- /dev/null +++ b/snesfilter/nall/stack.hpp @@ -0,0 +1,29 @@ +#ifndef NALL_STACK_HPP +#define NALL_STACK_HPP + +#include +#include + +namespace nall { + template struct stack : public linear_vector { + void push(const T &value) { + linear_vector::append(value); + } + + T pull() { + if(linear_vector::size() == 0) throw; + T value = linear_vector::operator[](linear_vector::size() - 1); + linear_vector::remove(linear_vector::size() - 1); + return value; + } + + T& operator()() { + if(linear_vector::size() == 0) throw; + return linear_vector::operator[](linear_vector::size() - 1); + } + }; + + template struct has_size> { enum { value = true }; }; +} + +#endif diff --git a/snesfilter/nall/string.hpp b/snesfilter/nall/string.hpp index 9acc2e9d..91bee596 100755 --- a/snesfilter/nall/string.hpp +++ b/snesfilter/nall/string.hpp @@ -2,7 +2,9 @@ #define NALL_STRING_HPP #include +#include #include +#include #include #include diff --git a/snesfilter/nall/string/base.hpp b/snesfilter/nall/string/base.hpp index f6172c26..86c42c20 100755 --- a/snesfilter/nall/string/base.hpp +++ b/snesfilter/nall/string/base.hpp @@ -6,12 +6,14 @@ #include #include #include +#include #include -#include #include +#include namespace nall { class string; + class lstring; template inline const char* to_string(T); class string { @@ -20,13 +22,13 @@ namespace nall { template inline string& assign(Args&&... args); template inline string& append(Args&&... args); - inline string& assign_(const char*); - inline string& append_(const char*); inline bool readfile(const string&); - inline string& replace (const char*, const char*); - inline string& qreplace(const char*, const char*); + template inline string& replace(const char*, const char*); + template inline string& ireplace(const char*, const char*); + template inline string& qreplace(const char*, const char*); + template inline string& iqreplace(const char*, const char*); inline unsigned length() const; @@ -43,17 +45,18 @@ namespace nall { inline string& lower(); inline string& upper(); + inline string& qlower(); + inline string& qupper(); inline string& transform(const char *before, const char *after); template inline string& ltrim(const char *key = " "); template inline string& rtrim(const char *key = " "); - template inline string& trim (const char *key = " "); + template inline string& trim(const char *key = " ", const char *rkey = 0); inline optional position(const char *key) const; + inline optional iposition(const char *key) const; inline optional qposition(const char *key) const; - - template inline string& operator= (T value); - template inline string& operator<<(T value); + inline optional iqposition(const char *key) const; inline operator const char*() const; inline char* operator()(); @@ -74,10 +77,16 @@ namespace nall { inline string(string&&); inline ~string(); + //internal functions + inline string& assign_(const char*); + inline string& append_(const char*); + protected: char *data; unsigned size; + template inline string& ureplace(const char*, const char*); + #if defined(QSTRING_H) public: inline operator QString() const; @@ -89,36 +98,46 @@ namespace nall { template inline lstring& operator<<(T value); inline optional find(const char*) const; - template inline void split (const char*, const char*); - template inline void qsplit(const char*, const char*); + template inline lstring& split(const char*, const char*); + template inline lstring& isplit(const char*, const char*); + template inline lstring& qsplit(const char*, const char*); + template inline lstring& iqsplit(const char*, const char*); + + inline bool operator==(const lstring&) const; + inline bool operator!=(const lstring&) const; lstring(); lstring(std::initializer_list); + + protected: + template inline lstring& usplit(const char*, const char*); }; //compare.hpp inline char chrlower(char c); inline char chrupper(char c); - inline int stricmp(const char *str1, const char *str2); + inline int istrcmp(const char *str1, const char *str2); inline bool wildcard(const char *str, const char *pattern); inline bool iwildcard(const char *str, const char *pattern); - inline bool strbegin (const char *str, const char *key); - inline bool stribegin(const char *str, const char *key); - inline bool strend (const char *str, const char *key); - inline bool striend(const char *str, const char *key); + inline bool strbegin(const char *str, const char *key); + inline bool istrbegin(const char *str, const char *key); + inline bool strend(const char *str, const char *key); + inline bool istrend(const char *str, const char *key); //convert.hpp inline char* strlower(char *str); inline char* strupper(char *str); + inline char* qstrlower(char *str); + inline char* qstrupper(char *str); inline char* strtr(char *dest, const char *before, const char *after); - inline uintmax_t hex (const char *str); - inline intmax_t integer(const char *str); + inline uintmax_t hex(const char *str); + inline intmax_t integer(const char *str); inline uintmax_t decimal(const char *str); - inline uintmax_t binary (const char *str); - inline double fp (const char *str); + inline uintmax_t binary(const char *str); + inline double fp(const char *str); //math.hpp - inline bool strint (const char *str, int &result); + inline bool strint(const char *str, int &result); inline bool strmath(const char *str, int &result); //platform.hpp @@ -132,26 +151,31 @@ namespace nall { //strpos.hpp inline optional strpos(const char *str, const char *key); + inline optional istrpos(const char *str, const char *key); inline optional qstrpos(const char *str, const char *key); + inline optional iqstrpos(const char *str, const char *key); + template inline optional ustrpos(const char *str, const char *key); //trim.hpp template inline char* ltrim(char *str, const char *key = " "); template inline char* rtrim(char *str, const char *key = " "); - template inline char* trim (char *str, const char *key = " "); + template inline char* trim(char *str, const char *key = " ", const char *rkey = 0); //utility.hpp + template alwaysinline bool chrequal(char x, char y); + template alwaysinline bool quoteskip(T *&p); + template alwaysinline bool quotecopy(char *&t, T *&p); inline unsigned strlcpy(string &dest, const char *src, unsigned length); inline unsigned strlcat(string &dest, const char *src, unsigned length); - inline string substr(const char *src, unsigned start = 0, unsigned length = 0); + inline string substr(const char *src, unsigned start = 0, unsigned length = ~0u); + inline string sha256(const uint8_t *data, unsigned size); - inline string integer(intmax_t value); - template inline string linteger(intmax_t value); - template inline string rinteger(intmax_t value); - inline string decimal(uintmax_t value); - template inline string ldecimal(uintmax_t value); - template inline string rdecimal(uintmax_t value); - template inline string hex(uintmax_t value); - template inline string binary(uintmax_t value); + template inline string integer(intmax_t value); + template inline string linteger(intmax_t value); + template inline string decimal(uintmax_t value); + template inline string ldecimal(uintmax_t value); + template inline string hex(uintmax_t value); + template inline string binary(uintmax_t value); inline unsigned fp(char *str, double value); inline string fp(double value); diff --git a/snesfilter/nall/string/cast.hpp b/snesfilter/nall/string/cast.hpp index 14f005da..2d010bfa 100755 --- a/snesfilter/nall/string/cast.hpp +++ b/snesfilter/nall/string/cast.hpp @@ -6,16 +6,15 @@ namespace nall { //this is needed, as C++0x does not support explicit template specialization inside classes template<> inline const char* to_string (bool v) { return v ? "true" : "false"; } template<> inline const char* to_string (signed int v) { static char temp[256]; snprintf(temp, 255, "%+d", v); return temp; } -template<> inline const char* to_string (unsigned int v) { static char temp[256]; snprintf(temp, 255, "%u", v); return temp; } -template<> inline const char* to_string (double v) { static char temp[256]; snprintf(temp, 255, "%f", v); return temp; } +template<> inline const char* to_string (unsigned int v) { static char temp[256]; snprintf(temp, 255, "%u", v); return temp; } +template<> inline const char* to_string (intmax_t v) { static char temp[256]; snprintf(temp, 255, "%+lld", (long long)v); return temp; } +template<> inline const char* to_string (uintmax_t v) { static char temp[256]; snprintf(temp, 255, "%llu", (unsigned long long)v); return temp; } +template<> inline const char* to_string (double v) { static char temp[256]; snprintf(temp, 255, "%f", v); return temp; } template<> inline const char* to_string (char *v) { return v; } template<> inline const char* to_string (const char *v) { return v; } template<> inline const char* to_string (string v) { return v; } template<> inline const char* to_string(const string &v) { return v; } -template string& string::operator= (T value) { return assign(to_string(value)); } -template string& string::operator<<(T value) { return append(to_string(value)); } - template lstring& lstring::operator<<(T value) { operator[](size()).assign(to_string(value)); return *this; diff --git a/snesfilter/nall/string/compare.hpp b/snesfilter/nall/string/compare.hpp index bce0895b..ad311d74 100755 --- a/snesfilter/nall/string/compare.hpp +++ b/snesfilter/nall/string/compare.hpp @@ -11,7 +11,7 @@ char chrupper(char c) { return (c >= 'a' && c <= 'z') ? c - ('a' - 'A') : c; } -int stricmp(const char *str1, const char *str2) { +int istrcmp(const char *str1, const char *str2) { while(*str1) { if(chrlower(*str1) != chrlower(*str2)) break; str1++, str2++; @@ -66,7 +66,7 @@ bool strbegin(const char *str, const char *key) { return (!memcmp(str, key, ksl)); } -bool stribegin(const char *str, const char *key) { +bool istrbegin(const char *str, const char *key) { int ssl = strlen(str), ksl = strlen(key); if(ksl > ssl) return false; @@ -89,7 +89,7 @@ bool strend(const char *str, const char *key) { return (!memcmp(str + ssl - ksl, key, ksl)); } -bool striend(const char *str, const char *key) { +bool istrend(const char *str, const char *key) { int ssl = strlen(str), ksl = strlen(key); if(ksl > ssl) return false; diff --git a/snesfilter/nall/string/convert.hpp b/snesfilter/nall/string/convert.hpp index 603d2e0e..3dd487f6 100755 --- a/snesfilter/nall/string/convert.hpp +++ b/snesfilter/nall/string/convert.hpp @@ -23,6 +23,26 @@ char* strupper(char *str) { return str; } +char* qstrlower(char *s) { + if(!s) return 0; + bool quoted = false; + while(*s) { + if(*s == '\"' || *s == '\'') quoted ^= 1; + if(quoted == false && *s >= 'A' && *s <= 'Z') *s += 0x20; + s++; + } +} + +char* qstrupper(char *s) { + if(!s) return 0; + bool quoted = false; + while(*s) { + if(*s == '\"' || *s == '\'') quoted ^= 1; + if(quoted == false && *s >= 'a' && *s <= 'z') *s -= 0x20; + s++; + } +} + char* strtr(char *dest, const char *before, const char *after) { if(!dest || !before || !after) return dest; int sl = strlen(dest), bsl = strlen(before), asl = strlen(after); diff --git a/snesfilter/nall/string/core.hpp b/snesfilter/nall/string/core.hpp index 4ffda4ee..e2af4eea 100755 --- a/snesfilter/nall/string/core.hpp +++ b/snesfilter/nall/string/core.hpp @@ -66,11 +66,13 @@ bool string::operator> (const char *str) const { return strcmp(data, str) > 0; bool string::operator>=(const char *str) const { return strcmp(data, str) >= 0; } string& string::operator=(const string &value) { + if(&value == this) return *this; assign(value); return *this; } string& string::operator=(string &&source) { + if(&source == this) return *this; if(data) free(data); size = source.size; data = source.data; @@ -87,11 +89,13 @@ template string::string(Args&&... args) { } string::string(const string &value) { + if(&value == this) return; size = strlen(value); data = strdup(value); } string::string(string &&source) { + if(&source == this) return; size = source.size; data = source.data; source.data = 0; @@ -131,6 +135,19 @@ optional lstring::find(const char *key) const { return { false, 0 }; } +bool lstring::operator==(const lstring &source) const { + if(this == &source) return true; + if(size() != source.size()) return false; + for(unsigned n = 0; n < size(); n++) { + if(operator[](n) != source[n]) return false; + } + return true; +} + +bool lstring::operator!=(const lstring &source) const { + return !operator==(source); +} + inline lstring::lstring() { } diff --git a/snesfilter/nall/string/math.hpp b/snesfilter/nall/string/math.hpp index ea8b99c8..d4bc9d25 100755 --- a/snesfilter/nall/string/math.hpp +++ b/snesfilter/nall/string/math.hpp @@ -3,6 +3,8 @@ namespace nall { +static function eval_fallback; + static int eval_integer(const char *&s) { if(!*s) throw "unrecognized_integer"; int value = 0, x = *s, y = *(s + 1); @@ -58,7 +60,7 @@ static int eval_integer(const char *&s) { } static int eval(const char *&s, int depth = 0) { - while(*s == ' ' || *s == '\t') s++; //trim whitespace + while(*s == ' ' || *s == '\t') s++; //trim whitespace if(!*s) throw "unrecognized_token"; int value = 0, x = *s, y = *(s + 1); @@ -74,10 +76,12 @@ static int eval(const char *&s, int depth = 0) { else if((x >= '0' && x <= '9') || x == '\'') value = eval_integer(s); + else if(eval_fallback) value = eval_fallback(s); //optional user-defined syntax parsing + else throw "unrecognized_token"; while(true) { - while(*s == ' ' || *s == '\t') s++; //trim whitespace + while(*s == ' ' || *s == '\t') s++; //trim whitespace if(!*s) break; x = *s, y = *(s + 1); diff --git a/snesfilter/nall/string/replace.hpp b/snesfilter/nall/string/replace.hpp index db405a9b..7c7a09d4 100755 --- a/snesfilter/nall/string/replace.hpp +++ b/snesfilter/nall/string/replace.hpp @@ -3,100 +3,49 @@ namespace nall { -string& string::replace(const char *key, const char *token) { - int i, z, ksl = strlen(key), tsl = strlen(token), ssl = length(); - unsigned int replace_count = 0, size = ssl; - char *buffer; +template +string& string::ureplace(const char *key, const char *token) { + if(!key || !*key) return *this; + enum : unsigned { limit = Limit ? Limit : ~0u }; - if(ksl <= ssl) { - if(tsl > ksl) { //the new string may be longer than the old string... - for(i = 0; i <= ssl - ksl;) { //so let's find out how big of a string we'll need... - if(!memcmp(data + i, key, ksl)) { - replace_count++; - i += ksl; - } else i++; - } - size = ssl + ((tsl - ksl) * replace_count); - reserve(size); + const char *p = data; + unsigned counter = 0, keyLength = 0; + + while(*p) { + if(quoteskip(p)) continue; + for(unsigned n = 0;; n++) { + if(key[n] == 0) { counter++; p += n; keyLength = n; break; } + if(!chrequal(key[n], p[n])) { p++; break; } } - - buffer = new char[size + 1]; - for(i = z = 0; i < ssl;) { - if(i <= ssl - ksl) { - if(!memcmp(data + i, key, ksl)) { - memcpy(buffer + z, token, tsl); - z += tsl; - i += ksl; - } else buffer[z++] = data[i++]; - } else buffer[z++] = data[i++]; - } - buffer[z] = 0; - - assign(buffer); - delete[] buffer; } + if(counter == 0) return *this; + if(Limit) counter = min(counter, Limit); + + char *t = data, *base; + unsigned tokenLength = strlen(token); + if(tokenLength > keyLength) { + t = base = strdup(data); + reserve((unsigned)(p - data) + ((tokenLength - keyLength) * counter)); + } + char *o = data; + + while(*t && counter) { + if(quotecopy(o, t)) continue; + for(unsigned n = 0;; n++) { + if(key[n] == 0) { counter--; memcpy(o, token, tokenLength); t += keyLength; o += tokenLength; break; } + if(!chrequal(key[n], t[n])) { *o++ = *t++; break; } + } + } + do *o++ = *t; while(*t++); + if(tokenLength > keyLength) free(base); return *this; } -string& string::qreplace(const char *key, const char *token) { - int i, l, z, ksl = strlen(key), tsl = strlen(token), ssl = length(); - unsigned int replace_count = 0, size = ssl; - uint8_t x; - char *buffer; - - if(ksl <= ssl) { - if(tsl > ksl) { - for(i = 0; i <= ssl - ksl;) { - x = data[i]; - if(x == '\"' || x == '\'') { - l = i; - i++; - while(data[i++] != x) { - if(i == ssl) { - i = l; - break; - } - } - } - if(!memcmp(data + i, key, ksl)) { - replace_count++; - i += ksl; - } else i++; - } - size = ssl + ((tsl - ksl) * replace_count); - reserve(size); - } - - buffer = new char[size + 1]; - for(i = z = 0; i < ssl;) { - x = data[i]; - if(x == '\"' || x == '\'') { - l = i++; - while(data[i] != x && i < ssl)i++; - if(i >= ssl)i = l; - else { - memcpy(buffer + z, data + l, i - l); - z += i - l; - } - } - if(i <= ssl - ksl) { - if(!memcmp(data + i, key, ksl)) { - memcpy(buffer + z, token, tsl); - z += tsl; - i += ksl; - replace_count++; - } else buffer[z++] = data[i++]; - } else buffer[z++] = data[i++]; - } - buffer[z] = 0; - - assign(buffer); - delete[] buffer; - } - - return *this; -} +template string &string::replace(const char *key, const char *token) { return ureplace(key, token); } +template string &string::ireplace(const char *key, const char *token) { return ureplace(key, token); } +template string &string::qreplace(const char *key, const char *token) { return ureplace(key, token); } +template string &string::iqreplace(const char *key, const char *token) { return ureplace(key, token); } }; diff --git a/snesfilter/nall/string/split.hpp b/snesfilter/nall/string/split.hpp index 8d3ca877..1644401b 100755 --- a/snesfilter/nall/string/split.hpp +++ b/snesfilter/nall/string/split.hpp @@ -3,56 +3,36 @@ namespace nall { -template void lstring::split(const char *key, const char *src) { - unsigned limit = Limit; +template lstring& lstring::usplit(const char *key, const char *base) { reset(); + if(!key || !*key) return *this; - int ssl = strlen(src), ksl = strlen(key); - int lp = 0, split_count = 0; + const char *p = base; + unsigned counter = 0; - for(int i = 0; i <= ssl - ksl;) { - if(!memcmp(src + i, key, ksl)) { - strlcpy(operator[](split_count++), src + lp, i - lp + 1); - i += ksl; - lp = i; - if(!--limit) break; - } else i++; - } - - operator[](split_count++) = src + lp; -} - -template void lstring::qsplit(const char *key, const char *src) { - unsigned limit = Limit; - reset(); - - int ssl = strlen(src), ksl = strlen(key); - int lp = 0, split_count = 0; - - for(int i = 0; i <= ssl - ksl;) { - uint8_t x = src[i]; - - if(x == '\"' || x == '\'') { - int z = i++; //skip opening quote - while(i < ssl && src[i] != x) i++; - if(i >= ssl) i = z; //failed match, rewind i - else { - i++; //skip closing quote - continue; //restart in case next char is also a quote + while(*p) { + if(Limit) if(counter >= Limit) break; + if(quoteskip(p)) continue; + for(unsigned n = 0;; n++) { + if(key[n] == 0) { + strlcpy(operator[](counter++), base, (unsigned)(p - base + 1)); + p += n; + base = p; + break; } + if(!chrequal(key[n], p[n])) { p++; break; } } - - if(!memcmp(src + i, key, ksl)) { - strlcpy(operator[](split_count++), src + lp, i - lp + 1); - i += ksl; - lp = i; - if(!--limit) break; - } else i++; } - operator[](split_count++) = src + lp; + operator[](counter) = base; + return *this; } +template lstring& lstring::split(const char *key, const char *src) { return usplit(key, src); } +template lstring& lstring::isplit(const char *key, const char *src) { return usplit(key, src); } +template lstring& lstring::qsplit(const char *key, const char *src) { return usplit(key, src); } +template lstring& lstring::iqsplit(const char *key, const char *src) { return usplit(key, src); } + }; #endif diff --git a/snesfilter/nall/string/strpos.hpp b/snesfilter/nall/string/strpos.hpp index 1907a2f3..3b28923e 100755 --- a/snesfilter/nall/string/strpos.hpp +++ b/snesfilter/nall/string/strpos.hpp @@ -2,40 +2,33 @@ #define NALL_STRING_STRPOS_HPP //usage example: -//if(auto pos = strpos(str, key)) print(pos(), "\n"); -//prints position of key within str, only if it is found +//if(auto position = strpos(str, key)) print(position(), "\n"); +//prints position of key within str; but only if it is found namespace nall { -optional strpos(const char *str, const char *key) { - unsigned ssl = strlen(str), ksl = strlen(key); - if(ksl > ssl) return { false, 0 }; +template +optional ustrpos(const char *str, const char *key) { + const char *base = str; - for(unsigned i = 0; i <= ssl - ksl; i++) { - if(!memcmp(str + i, key, ksl)) return { true, i }; - } - - return { false, 0 }; -} - -optional qstrpos(const char *str, const char *key) { - unsigned ssl = strlen(str), ksl = strlen(key); - if(ksl > ssl) return { false, 0 }; - - for(unsigned i = 0; i <= ssl - ksl;) { - uint8_t x = str[i]; - if(x == '\"' || x == '\'') { - uint8_t z = i++; - while(str[i] != x && i < ssl) i++; - if(i >= ssl) i = z; + while(*str) { + if(quoteskip(str)) continue; + for(unsigned n = 0;; n++) { + if(key[n] == 0) return { true, (unsigned)(str - base) }; + if(str[n] == 0) return { false, 0 }; + if(!chrequal(str[n], key[n])) break; } - if(!memcmp(str + i, key, ksl)) return { true, i }; - i++; + str++; } return { false, 0 }; } +optional strpos(const char *str, const char *key) { return ustrpos(str, key); } +optional istrpos(const char *str, const char *key) { return ustrpos(str, key); } +optional qstrpos(const char *str, const char *key) { return ustrpos(str, key); } +optional iqstrpos(const char *str, const char *key) { return ustrpos(str, key); } + } #endif diff --git a/snesfilter/nall/string/trim.hpp b/snesfilter/nall/string/trim.hpp index f5355d7d..d1f15ee1 100755 --- a/snesfilter/nall/string/trim.hpp +++ b/snesfilter/nall/string/trim.hpp @@ -29,7 +29,8 @@ template char* rtrim(char *str, const char *key) { return str; } -template char* trim(char *str, const char *key) { +template char* trim(char *str, const char *key, const char *rkey) { + if(rkey) return ltrim(rtrim(str, rkey), key); return ltrim(rtrim(str, key), key); } diff --git a/snesfilter/nall/string/utility.hpp b/snesfilter/nall/string/utility.hpp index 8e6c1005..13faaf64 100755 --- a/snesfilter/nall/string/utility.hpp +++ b/snesfilter/nall/string/utility.hpp @@ -3,6 +3,38 @@ namespace nall { +template +bool chrequal(char x, char y) { + if(Insensitive) return chrlower(x) == chrlower(y); + return x == y; +} + +template +bool quoteskip(T *&p) { + if(Quoted == false) return false; + if(*p != '\'' && *p != '\"') return false; + + while(*p == '\'' || *p == '\"') { + char x = *p++; + while(*p && *p++ != x); + } + return true; +} + +template +bool quotecopy(char *&t, T *&p) { + if(Quoted == false) return false; + if(*p != '\'' && *p != '\"') return false; + + while(*p == '\'' || *p == '\"') { + char x = *p++; + *t++ = x; + while(*p && *p != x) *t++ = *p++; + *t++ = *p++; + } + return true; +} + unsigned strlcpy(string &dest, const char *src, unsigned length) { dest.reserve(length); return strlcpy(dest(), src, length); @@ -15,7 +47,7 @@ unsigned strlcat(string &dest, const char *src, unsigned length) { string substr(const char *src, unsigned start, unsigned length) { string dest; - if(length == 0) { + if(length == ~0u) { //copy entire string dest = src + start; } else { @@ -25,35 +57,21 @@ string substr(const char *src, unsigned start, unsigned length) { return dest; } +string sha256(const uint8_t *data, unsigned size) { + sha256_ctx sha; + uint8_t hash[32]; + sha256_init(&sha); + sha256_chunk(&sha, data, size); + sha256_final(&sha); + sha256_hash(&sha, hash); + string result; + foreach(byte, hash) result.append(hex<2>(byte)); + return result; +} + /* arithmetic <> string */ -string integer(intmax_t value) { - bool negative = value < 0; - if(negative) value = abs(value); - - char buffer[64]; - unsigned size = 0; - - do { - unsigned n = value % 10; - buffer[size++] = '0' + n; - value /= 10; - } while(value); - buffer[size++] = negative ? '-' : '+'; - buffer[size] = 0; - - char result[size + 1]; - memset(result, '0', size); - result[size] = 0; - - for(signed x = size - 1, y = 0; x >= 0 && y < size; x--, y++) { - result[x] = buffer[y]; - } - - return (const char*)result; -} - -template string linteger(intmax_t value) { +template string integer(intmax_t value) { bool negative = value < 0; if(negative) value = abs(value); @@ -70,34 +88,7 @@ template string linteger(intmax_t value) { unsigned length = (length_ == 0 ? size : length_); char result[length + 1]; - memset(result, ' ', length); - result[length] = 0; - - for(signed x = 0, y = size - 1; x < length && y >= 0; x++, y--) { - result[x] = buffer[y]; - } - - return (const char*)result; -} - -template string rinteger(intmax_t value) { - bool negative = value < 0; - if(negative) value = abs(value); - - char buffer[64]; - unsigned size = 0; - - do { - unsigned n = value % 10; - buffer[size++] = '0' + n; - value /= 10; - } while(value); - buffer[size++] = negative ? '-' : '+'; - buffer[size] = 0; - - unsigned length = (length_ == 0 ? size : length_); - char result[length + 1]; - memset(result, ' ', length); + memset(result, padding, length); result[length] = 0; for(signed x = length - 1, y = 0; x >= 0 && y < size; x--, y++) { @@ -107,29 +98,10 @@ template string rinteger(intmax_t value) { return (const char*)result; } -string decimal(uintmax_t value) { - char buffer[64]; - unsigned size = 0; - - do { - unsigned n = value % 10; - buffer[size++] = '0' + n; - value /= 10; - } while(value); - buffer[size] = 0; - - char result[size + 1]; - memset(result, '0', size); - result[size] = 0; - - for(signed x = size - 1, y = 0; x >= 0 && y < size; x--, y++) { - result[x] = buffer[y]; - } - - return (const char*)result; -} - -template string ldecimal(uintmax_t value) { +template string linteger(intmax_t value) { + bool negative = value < 0; + if(negative) value = abs(value); + char buffer[64]; unsigned size = 0; @@ -138,11 +110,12 @@ template string ldecimal(uintmax_t value) { buffer[size++] = '0' + n; value /= 10; } while(value); + buffer[size++] = negative ? '-' : '+'; buffer[size] = 0; unsigned length = (length_ == 0 ? size : length_); char result[length + 1]; - memset(result, ' ', length); + memset(result, padding, length); result[length] = 0; for(signed x = 0, y = size - 1; x < length && y >= 0; x++, y--) { @@ -152,7 +125,7 @@ template string ldecimal(uintmax_t value) { return (const char*)result; } -template string rdecimal(uintmax_t value) { +template string decimal(uintmax_t value) { char buffer[64]; unsigned size = 0; @@ -165,7 +138,7 @@ template string rdecimal(uintmax_t value) { unsigned length = (length_ == 0 ? size : length_); char result[length + 1]; - memset(result, ' ', length); + memset(result, padding, length); result[length] = 0; for(signed x = length - 1, y = 0; x >= 0 && y < size; x--, y++) { @@ -175,7 +148,30 @@ template string rdecimal(uintmax_t value) { return (const char*)result; } -template string hex(uintmax_t value) { +template string ldecimal(uintmax_t value) { + char buffer[64]; + unsigned size = 0; + + do { + unsigned n = value % 10; + buffer[size++] = '0' + n; + value /= 10; + } while(value); + buffer[size] = 0; + + unsigned length = (length_ == 0 ? size : length_); + char result[length + 1]; + memset(result, padding, length); + result[length] = 0; + + for(signed x = 0, y = size - 1; x < length && y >= 0; x++, y--) { + result[x] = buffer[y]; + } + + return (const char*)result; +} + +template string hex(uintmax_t value) { char buffer[64]; unsigned size = 0; @@ -187,7 +183,7 @@ template string hex(uintmax_t value) { unsigned length = (length_ == 0 ? size : length_); char result[length + 1]; - memset(result, '0', length); + memset(result, padding, length); result[length] = 0; for(signed x = length - 1, y = 0; x >= 0 && y < size; x--, y++) { @@ -197,7 +193,7 @@ template string hex(uintmax_t value) { return (const char*)result; } -template string binary(uintmax_t value) { +template string binary(uintmax_t value) { char buffer[256]; unsigned size = 0; @@ -209,7 +205,7 @@ template string binary(uintmax_t value) { unsigned length = (length_ == 0 ? size : length_); char result[length + 1]; - memset(result, '0', length); + memset(result, padding, length); result[length] = 0; for(signed x = length - 1, y = 0; x >= 0 && y < size; x--, y++) { diff --git a/snesfilter/nall/string/wrapper.hpp b/snesfilter/nall/string/wrapper.hpp index eadf0a10..a28c1ced 100755 --- a/snesfilter/nall/string/wrapper.hpp +++ b/snesfilter/nall/string/wrapper.hpp @@ -6,27 +6,31 @@ namespace nall { unsigned string::length() const { return strlen(data); } bool string::equals(const char *str) const { return !strcmp(data, str); } -bool string::iequals(const char *str) const { return !stricmp(data, str); } +bool string::iequals(const char *str) const { return !istrcmp(data, str); } bool string::wildcard(const char *str) const { return nall::wildcard(data, str); } bool string::iwildcard(const char *str) const { return nall::iwildcard(data, str); } bool string::beginswith(const char *str) const { return strbegin(data, str); } -bool string::ibeginswith(const char *str) const { return stribegin(data, str); } +bool string::ibeginswith(const char *str) const { return istrbegin(data, str); } bool string::endswith(const char *str) const { return strend(data, str); } -bool string::iendswith(const char *str) const { return striend(data, str); } +bool string::iendswith(const char *str) const { return istrend(data, str); } string& string::lower() { nall::strlower(data); return *this; } string& string::upper() { nall::strupper(data); return *this; } +string& string::qlower() { nall::qstrlower(data); return *this; } +string& string::qupper() { nall::qstrupper(data); return *this; } string& string::transform(const char *before, const char *after) { nall::strtr(data, before, after); return *this; } template string& string::ltrim(const char *key) { nall::ltrim(data, key); return *this; } template string& string::rtrim(const char *key) { nall::rtrim(data, key); return *this; } -template string& string::trim (const char *key) { nall::trim (data, key); return *this; } +template string& string::trim(const char *key, const char *rkey) { nall::trim (data, key, rkey); return *this; } optional string::position(const char *key) const { return strpos(data, key); } +optional string::iposition(const char *key) const { return istrpos(data, key); } optional string::qposition(const char *key) const { return qstrpos(data, key); } +optional string::iqposition(const char *key) const { return iqstrpos(data, key); } } diff --git a/snesfilter/nall/string/xml.hpp b/snesfilter/nall/string/xml.hpp index 185a89f9..47653786 100755 --- a/snesfilter/nall/string/xml.hpp +++ b/snesfilter/nall/string/xml.hpp @@ -77,7 +77,7 @@ inline string xml_element::parse() const { if(auto pos = strpos(source, "]]>")) { if(pos() - 9 > 0) { string cdata = substr(source, 9, pos() - 9); - data << cdata; + data.append(cdata); offset += strlen(cdata); } source += 9 + offset + 3; diff --git a/snesfilter/nall/test/cc.bat b/snesfilter/nall/test/cc.bat new file mode 100755 index 00000000..f6434960 --- /dev/null +++ b/snesfilter/nall/test/cc.bat @@ -0,0 +1,2 @@ +g++ -std=gnu++0x -O3 -fomit-frame-pointer -s -o test test.cpp -I../.. -lws2_32 +@pause diff --git a/snesfilter/nall/test/cc.sh b/snesfilter/nall/test/cc.sh new file mode 100755 index 00000000..d084a256 --- /dev/null +++ b/snesfilter/nall/test/cc.sh @@ -0,0 +1,2 @@ +g++-4.5 -std=gnu++0x -g -o test test.cpp -I../.. +#g++-4.5 -std=gnu++0x -O3 -fomit-frame-pointer -s -o test test.cpp -I../.. diff --git a/snesfilter/nall/test/test b/snesfilter/nall/test/test new file mode 100755 index 00000000..56b80ee4 Binary files /dev/null and b/snesfilter/nall/test/test differ diff --git a/snesfilter/nall/test/test.cpp b/snesfilter/nall/test/test.cpp new file mode 100755 index 00000000..3449661a --- /dev/null +++ b/snesfilter/nall/test/test.cpp @@ -0,0 +1,50 @@ +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +using namespace nall; + +int main(int main, char **argv) { + lstring a = { "hey", "hi" }, b = { "hey", "hi" }; + print(a == b, "\n"); + + return 0; +} diff --git a/snesfilter/nall/utility.hpp b/snesfilter/nall/utility.hpp index 60bda562..374b5469 100755 --- a/snesfilter/nall/utility.hpp +++ b/snesfilter/nall/utility.hpp @@ -26,6 +26,8 @@ namespace nall { public: inline operator bool() const { return valid; } inline const T& operator()() const { if(!valid) throw; return value; } + inline optional& operator=(const optional &source) { valid = source.valid; value = source.value; return *this; } + inline optional() : valid(false) {} inline optional(bool valid, const T &value) : valid(valid), value(value) {} }; diff --git a/snesfilter/nall/varint.hpp b/snesfilter/nall/varint.hpp index 35649896..d91ea2a5 100755 --- a/snesfilter/nall/varint.hpp +++ b/snesfilter/nall/varint.hpp @@ -30,6 +30,9 @@ namespace nall { inline uint_t() : data(0) {} inline uint_t(const unsigned i) : data(uclip(i)) {} + + template inline unsigned operator=(const uint_t &i) { return data = uclip((unsigned)i); } + template inline uint_t(const uint_t &i) : data(uclip(i)) {} }; template class int_t { diff --git a/snesfilter/nall/vector.hpp b/snesfilter/nall/vector.hpp index c6ef24f2..f98eb375 100755 --- a/snesfilter/nall/vector.hpp +++ b/snesfilter/nall/vector.hpp @@ -94,14 +94,15 @@ namespace nall { else resize(objectsize - count); } - inline T& operator[](unsigned index) { - if(index >= objectsize) resize(index + 1); - return pool[index]; + linear_vector() : pool(0), poolsize(0), objectsize(0) { } - inline const T& operator[](unsigned index) const { - if(index >= objectsize) throw "vector[] out of bounds"; - return pool[index]; + linear_vector(std::initializer_list list) : pool(0), poolsize(0), objectsize(0) { + for(const T *p = list.begin(); p != list.end(); ++p) append(*p); + } + + ~linear_vector() { + reset(); } //copy @@ -132,17 +133,22 @@ namespace nall { operator=(std::move(source)); } - //construction - linear_vector() : pool(0), poolsize(0), objectsize(0) { + //index + inline T& operator[](unsigned index) { + if(index >= objectsize) resize(index + 1); + return pool[index]; } - linear_vector(std::initializer_list list) : pool(0), poolsize(0), objectsize(0) { - for(const T *p = list.begin(); p != list.end(); ++p) append(*p); + inline const T& operator[](unsigned index) const { + if(index >= objectsize) throw "vector[] out of bounds"; + return pool[index]; } - ~linear_vector() { - reset(); - } + //iteration + T* begin() { return &pool[0]; } + T* end() { return &pool[objectsize]; } + const T* begin() const { return &pool[0]; } + const T* end() const { return &pool[objectsize]; } }; //pointer_vector @@ -222,15 +228,15 @@ namespace nall { else resize(objectsize - count); } - inline T& operator[](unsigned index) { - if(index >= objectsize) resize(index + 1); - if(!pool[index]) pool[index] = new T; - return *pool[index]; + pointer_vector() : pool(0), poolsize(0), objectsize(0) { } - inline const T& operator[](unsigned index) const { - if(index >= objectsize || !pool[index]) throw "vector[] out of bounds"; - return *pool[index]; + pointer_vector(std::initializer_list list) : pool(0), poolsize(0), objectsize(0) { + for(const T *p = list.begin(); p != list.end(); ++p) append(*p); + } + + ~pointer_vector() { + reset(); } //copy @@ -261,17 +267,31 @@ namespace nall { operator=(std::move(source)); } - //construction - pointer_vector() : pool(0), poolsize(0), objectsize(0) { + //index + inline T& operator[](unsigned index) { + if(index >= objectsize) resize(index + 1); + if(!pool[index]) pool[index] = new T; + return *pool[index]; } - pointer_vector(std::initializer_list list) : pool(0), poolsize(0), objectsize(0) { - for(const T *p = list.begin(); p != list.end(); ++p) append(*p); + inline const T& operator[](unsigned index) const { + if(index >= objectsize || !pool[index]) throw "vector[] out of bounds"; + return *pool[index]; } - ~pointer_vector() { - reset(); - } + //iteration + struct iterator { + bool operator!=(const iterator &source) const { return index != source.index; } + T& operator*() { return vector.operator[](index); } + iterator& operator++() { index++; return *this; } + iterator(pointer_vector &vector, unsigned index) : vector(vector), index(index) {} + private: + pointer_vector &vector; + unsigned index; + }; + + iterator begin() { return iterator(*this, 0); } + iterator end() { return iterator(*this, objectsize); } }; template struct has_size> { enum { value = true }; }; diff --git a/snesfilter/nall/windows/detour.hpp b/snesfilter/nall/windows/detour.hpp new file mode 100755 index 00000000..e270f318 --- /dev/null +++ b/snesfilter/nall/windows/detour.hpp @@ -0,0 +1,192 @@ +#ifndef NALL_WINDOWS_DETOUR_HPP +#define NALL_WINDOWS_DETOUR_HPP + +#include +#include +#include +#include +#include + +namespace nall { + +#define Copy 0 +#define RelNear 1 + +struct detour { + static bool insert(const string &moduleName, const string &functionName, void *&source, void *target); + static bool remove(const string &moduleName, const string &functionName, void *&source); + +protected: + static unsigned length(const uint8_t *function); + static unsigned mirror(uint8_t *target, const uint8_t *source); + + struct opcode { + uint16_t prefix; + unsigned length; + unsigned mode; + uint16_t modify; + }; + static opcode opcodes[]; +}; + +//TODO: +//* fs:, gs: should force another opcode copy +//* conditional branches within +5-byte range should fail +detour::opcode detour::opcodes[] = { + { 0x50, 1 }, //push eax + { 0x51, 1 }, //push ecx + { 0x52, 1 }, //push edx + { 0x53, 1 }, //push ebx + { 0x54, 1 }, //push esp + { 0x55, 1 }, //push ebp + { 0x56, 1 }, //push esi + { 0x57, 1 }, //push edi + { 0x58, 1 }, //pop eax + { 0x59, 1 }, //pop ecx + { 0x5a, 1 }, //pop edx + { 0x5b, 1 }, //pop ebx + { 0x5c, 1 }, //pop esp + { 0x5d, 1 }, //pop ebp + { 0x5e, 1 }, //pop esi + { 0x5f, 1 }, //pop edi + { 0x64, 1 }, //fs: + { 0x65, 1 }, //gs: + { 0x68, 5 }, //push dword + { 0x6a, 2 }, //push byte + { 0x74, 2, RelNear, 0x0f84 }, //je near -> je far + { 0x75, 2, RelNear, 0x0f85 }, //jne near -> jne far + { 0x89, 2 }, //mov reg,reg + { 0x8b, 2 }, //mov reg,reg + { 0x90, 1 }, //nop + { 0xa1, 5 }, //mov eax,[dword] + { 0xeb, 2, RelNear, 0xe9 }, //jmp near -> jmp far +}; + +bool detour::insert(const string &moduleName, const string &functionName, void *&source, void *target) { + HMODULE module = GetModuleHandleW(utf16_t(moduleName)); + if(!module) return false; + + uint8_t *sourceData = (uint8_t*)GetProcAddress(module, functionName); + if(!sourceData) return false; + + unsigned sourceLength = detour::length(sourceData); + if(sourceLength < 5) { + //unable to clone enough bytes to insert hook + #if 1 + string output = { "detour::insert(", moduleName, "::", functionName, ") failed: " }; + for(unsigned n = 0; n < 16; n++) output.append(hex<2>(sourceData[n]), " "); + output.rtrim<1>(" "); + MessageBoxA(0, output, "nall::detour", MB_OK); + #endif + return false; + } + + uint8_t *mirrorData = new uint8_t[512](); + detour::mirror(mirrorData, sourceData); + + DWORD privileges; + VirtualProtect((void*)mirrorData, 512, PAGE_EXECUTE_READWRITE, &privileges); + VirtualProtect((void*)sourceData, 256, PAGE_EXECUTE_READWRITE, &privileges); + uintmax_t address = (uintmax_t)target - ((uintmax_t)sourceData + 5); + sourceData[0] = 0xe9; //jmp target + sourceData[1] = address >> 0; + sourceData[2] = address >> 8; + sourceData[3] = address >> 16; + sourceData[4] = address >> 24; + VirtualProtect((void*)sourceData, 256, privileges, &privileges); + + source = (void*)mirrorData; + return true; +} + +bool detour::remove(const string &moduleName, const string &functionName, void *&source) { + HMODULE module = GetModuleHandleW(utf16_t(moduleName)); + if(!module) return false; + + uint8_t *sourceData = (uint8_t*)GetProcAddress(module, functionName); + if(!sourceData) return false; + + uint8_t *mirrorData = (uint8_t*)source; + if(mirrorData == sourceData) return false; //hook was never installed + + unsigned length = detour::length(256 + mirrorData); + if(length < 5) return false; + + DWORD privileges; + VirtualProtect((void*)sourceData, 256, PAGE_EXECUTE_READWRITE, &privileges); + for(unsigned n = 0; n < length; n++) sourceData[n] = mirrorData[256 + n]; + VirtualProtect((void*)sourceData, 256, privileges, &privileges); + + source = (void*)sourceData; + delete[] mirrorData; + return true; +} + +unsigned detour::length(const uint8_t *function) { + unsigned length = 0; + while(length < 5) { + detour::opcode *opcode = 0; + foreach(op, detour::opcodes) { + if(function[length] == op.prefix) { + opcode = &op; + break; + } + } + if(opcode == 0) break; + length += opcode->length; + } + return length; +} + +unsigned detour::mirror(uint8_t *target, const uint8_t *source) { + const uint8_t *entryPoint = source; + for(unsigned n = 0; n < 256; n++) target[256 + n] = source[n]; + + unsigned size = detour::length(source); + while(size) { + detour::opcode *opcode = 0; + foreach(op, detour::opcodes) { + if(*source == op.prefix) { + opcode = &op; + break; + } + } + + switch(opcode->mode) { + case Copy: + for(unsigned n = 0; n < opcode->length; n++) *target++ = *source++; + break; + case RelNear: { + source++; + uintmax_t sourceAddress = (uintmax_t)source + 1 + (int8_t)*source; + *target++ = opcode->modify; + if(opcode->modify >> 8) *target++ = opcode->modify >> 8; + uintmax_t targetAddress = (uintmax_t)target + 4; + uintmax_t address = sourceAddress - targetAddress; + *target++ = address >> 0; + *target++ = address >> 8; + *target++ = address >> 16; + *target++ = address >> 24; + source += 2; + } break; + } + + size -= opcode->length; + } + + uintmax_t address = (entryPoint + detour::length(entryPoint)) - (target + 5); + *target++ = 0xe9; //jmp entryPoint + *target++ = address >> 0; + *target++ = address >> 8; + *target++ = address >> 16; + *target++ = address >> 24; + + return source - entryPoint; +} + +#undef Implied +#undef RelNear + +} + +#endif diff --git a/snesfilter/nall/windows/launcher.hpp b/snesfilter/nall/windows/launcher.hpp new file mode 100755 index 00000000..914683ec --- /dev/null +++ b/snesfilter/nall/windows/launcher.hpp @@ -0,0 +1,94 @@ +#ifndef NALL_WINDOWS_LAUNCHER_HPP +#define NALL_WINDOWS_LAUNCHER_HPP + +namespace nall { + +//launch a new process and inject specified DLL into it + +bool launch(const char *applicationName, const char *libraryName, uint32_t entryPoint) { + //if a launcher does not send at least one message, a wait cursor will appear + PostThreadMessage(GetCurrentThreadId(), WM_USER, 0, 0); + MSG msg; + GetMessage(&msg, 0, 0, 0); + + STARTUPINFOW si; + PROCESS_INFORMATION pi; + + memset(&si, 0, sizeof(STARTUPINFOW)); + BOOL result = CreateProcessW( + utf16_t(applicationName), GetCommandLineW(), NULL, NULL, TRUE, + DEBUG_PROCESS | DEBUG_ONLY_THIS_PROCESS, //do not break if application creates its own processes + NULL, NULL, &si, &pi + ); + if(result == false) return false; + + uint8_t entryData[1024], entryHook[1024] = { + 0x68, 0x00, 0x00, 0x00, 0x00, //push libraryName + 0xb8, 0x00, 0x00, 0x00, 0x00, //mov eax,LoadLibraryW + 0xff, 0xd0, //call eax + 0xcd, 0x03, //int 3 + }; + + entryHook[1] = (uint8_t)((entryPoint + 14) >> 0); + entryHook[2] = (uint8_t)((entryPoint + 14) >> 8); + entryHook[3] = (uint8_t)((entryPoint + 14) >> 16); + entryHook[4] = (uint8_t)((entryPoint + 14) >> 24); + + uint32_t pLoadLibraryW = (uint32_t)GetProcAddress(GetModuleHandleW(L"kernel32"), "LoadLibraryW"); + entryHook[6] = pLoadLibraryW >> 0; + entryHook[7] = pLoadLibraryW >> 8; + entryHook[8] = pLoadLibraryW >> 16; + entryHook[9] = pLoadLibraryW >> 24; + + utf16_t buffer = utf16_t(libraryName); + memcpy(entryHook + 14, buffer, 2 * wcslen(buffer) + 2); + + while(true) { + DEBUG_EVENT event; + WaitForDebugEvent(&event, INFINITE); + + if(event.dwDebugEventCode == EXIT_PROCESS_DEBUG_EVENT) break; + + if(event.dwDebugEventCode == EXCEPTION_DEBUG_EVENT) { + if(event.u.Exception.ExceptionRecord.ExceptionCode == EXCEPTION_BREAKPOINT) { + if(event.u.Exception.ExceptionRecord.ExceptionAddress == (void*)(entryPoint + 14 - 1)) { + HANDLE hProcess = OpenProcess(0, FALSE, event.dwProcessId); + HANDLE hThread = OpenThread(THREAD_ALL_ACCESS, FALSE, event.dwThreadId); + + CONTEXT context; + context.ContextFlags = CONTEXT_FULL; + GetThreadContext(hThread, &context); + + WriteProcessMemory(pi.hProcess, (void*)entryPoint, (void*)&entryData, sizeof entryData, NULL); + context.Eip = entryPoint; + SetThreadContext(hThread, &context); + + CloseHandle(hThread); + CloseHandle(hProcess); + } + + ContinueDebugEvent(event.dwProcessId, event.dwThreadId, DBG_CONTINUE); + continue; + } + + ContinueDebugEvent(event.dwProcessId, event.dwThreadId, DBG_EXCEPTION_NOT_HANDLED); + continue; + } + + if(event.dwDebugEventCode == CREATE_PROCESS_DEBUG_EVENT) { + ReadProcessMemory(pi.hProcess, (void*)entryPoint, (void*)&entryData, sizeof entryData, NULL); + WriteProcessMemory(pi.hProcess, (void*)entryPoint, (void*)&entryHook, sizeof entryHook, NULL); + + ContinueDebugEvent(event.dwProcessId, event.dwThreadId, DBG_CONTINUE); + continue; + } + + ContinueDebugEvent(event.dwProcessId, event.dwThreadId, DBG_CONTINUE); + } + + return true; +} + +} + +#endif diff --git a/snesfilter/nall/utf8.hpp b/snesfilter/nall/windows/utf8.hpp similarity index 100% rename from snesfilter/nall/utf8.hpp rename to snesfilter/nall/windows/utf8.hpp diff --git a/snesfilter/nall/zip.hpp b/snesfilter/nall/zip.hpp new file mode 100755 index 00000000..ad9c7506 --- /dev/null +++ b/snesfilter/nall/zip.hpp @@ -0,0 +1,124 @@ +#ifndef NALL_UNZIP_HPP +#define NALL_UNZIP_HPP + +#include +#include +#include +#include + +namespace nall { + +struct zip { + struct File { + string name; + const uint8_t *data; + unsigned size; + unsigned csize; + unsigned cmode; //0 = uncompressed, 8 = deflate + unsigned crc32; + }; + + inline bool open(const string &filename) { + close(); + if(fm.open(filename, filemap::mode::read) == false) return false; + if(open(fm.data(), fm.size()) == false) { + fm.close(); + return false; + } + return true; + } + + inline bool open(const uint8_t *data, unsigned size) { + if(size < 22) return false; + + filedata = data; + filesize = size; + + file.reset(); + + const uint8_t *footer = data + size - 22; + const uint8_t *directory = data + read(footer + 16, 4); + + while(true) { + unsigned signature = read(directory + 0, 4); + if(signature != 0x02014b50) break; + + File file; + file.cmode = read(directory + 10, 2); + file.crc32 = read(directory + 16, 4); + file.csize = read(directory + 20, 4); + file.size = read(directory + 24, 4); + + unsigned namelength = read(directory + 28, 2); + unsigned extralength = read(directory + 30, 2); + unsigned commentlength = read(directory + 32, 2); + + char *filename = new char[namelength + 1]; + memcpy(filename, directory + 46, namelength); + filename[namelength] = 0; + file.name = filename; + delete[] filename; + + unsigned offset = read(directory + 42, 4); + unsigned offsetNL = read(data + offset + 26, 2); + unsigned offsetEL = read(data + offset + 28, 2); + file.data = data + offset + 30 + offsetNL + offsetEL; + + directory += 46 + namelength + extralength + commentlength; + + this->file.append(file); + } + + return true; + } + + inline bool extract(File &file, uint8_t *&data, unsigned &size) { + data = 0, size = 0; + + if(file.cmode == 0) { + size = file.size; + data = new uint8_t[size]; + memcpy(data, file.data, size); + return true; + } + + if(file.cmode == 8) { + size = file.size; + data = new uint8_t[size]; + if(inflate(data, size, file.data, file.csize) == false) { + delete[] data; + size = 0; + return false; + } + return true; + } + + return false; + } + + inline void close() { + if(fm.open()) fm.close(); + } + + ~zip() { + close(); + } + +protected: + filemap fm; + const uint8_t *filedata; + unsigned filesize; + + unsigned read(const uint8_t *data, unsigned size) { + unsigned result = 0, shift = 0; + while(size--) { result |= *data++ << shift; shift += 8; } + return result; + } + +public: + linear_vector file; +}; + +} + +#endif