From 98ec338285bc418d38837fe4d1ed4cb3cbea8aa7 Mon Sep 17 00:00:00 2001 From: Tim Allen Date: Thu, 22 Sep 2011 10:03:11 +1000 Subject: [PATCH] Update snesfilter to release 20110920. This was released beside bsnes v082r19. byuu didn't mention it in the v082r19 release notes, but in a previous post mentioned that a number of filters stopped working when bsnes switched to using RGB555 for all its internal data. --- snesfilter/2xSaI/2xSaI.cpp | 32 - snesfilter/2xSaI/Super-2xSaI.cpp | 32 - snesfilter/2xSaI/Super-Eagle.cpp | 32 - snesfilter/2xSaI/implementation.cpp | 1171 ------------------ snesfilter/HQ2x/HQ2x.cpp | 30 +- snesfilter/LQ2x/LQ2x.cpp | 43 +- snesfilter/Makefile | 22 +- snesfilter/NTSC/NTSC-Composite.cpp | 63 - snesfilter/NTSC/NTSC-RF.cpp | 63 - snesfilter/NTSC/NTSC-RGB.cpp | 63 - snesfilter/NTSC/NTSC-SVideo.cpp | 63 - snesfilter/NTSC/snes_ntsc/snes_ntsc.c | 251 ---- snesfilter/NTSC/snes_ntsc/snes_ntsc.h | 228 ---- snesfilter/NTSC/snes_ntsc/snes_ntsc_config.h | 26 - snesfilter/NTSC/snes_ntsc/snes_ntsc_impl.h | 439 ------- snesfilter/Pixellate2x/Pixellate2x.cpp | 43 +- snesfilter/Scale2x/Scale2x.cpp | 43 +- snesfilter/Scanline/Scanline-Black.cpp | 34 - snesfilter/Scanline/Scanline-Dark.cpp | 45 +- snesfilter/Scanline/Scanline-Light.cpp | 45 +- snesfilter/nall/array.hpp | 10 + snesfilter/nall/base64.hpp | 1 + snesfilter/nall/bmp.hpp | 101 ++ snesfilter/nall/bps/delta.hpp | 214 ++++ snesfilter/nall/bps/linear.hpp | 152 +++ snesfilter/nall/bps/metadata.hpp | 121 ++ snesfilter/nall/bps/patch.hpp | 219 ++++ snesfilter/nall/compositor.hpp | 79 ++ snesfilter/nall/config.hpp | 16 +- snesfilter/nall/detect.hpp | 2 +- snesfilter/nall/dictionary.hpp | 75 -- snesfilter/nall/directory.hpp | 11 +- snesfilter/nall/dl.hpp | 2 +- snesfilter/nall/dsp.hpp | 8 + snesfilter/nall/dsp/buffer.hpp | 51 + snesfilter/nall/dsp/core.hpp | 162 +++ snesfilter/nall/dsp/resample/average.hpp | 31 + snesfilter/nall/dsp/resample/cosine.hpp | 25 + snesfilter/nall/dsp/resample/cubic.hpp | 31 + snesfilter/nall/dsp/resample/hermite.hpp | 43 + snesfilter/nall/dsp/resample/linear.hpp | 24 + snesfilter/nall/dsp/resample/point.hpp | 24 + snesfilter/nall/dsp/settings.hpp | 39 + snesfilter/nall/file.hpp | 98 +- snesfilter/nall/filemap.hpp | 14 +- snesfilter/nall/gameboy/cartridge.hpp | 32 +- snesfilter/nall/gzip.hpp | 87 ++ snesfilter/nall/http.hpp | 176 +++ snesfilter/nall/inflate.hpp | 358 ++++++ snesfilter/nall/input.hpp | 6 +- snesfilter/nall/ips.hpp | 110 ++ snesfilter/nall/lzss.hpp | 206 ++- snesfilter/nall/platform.hpp | 26 +- snesfilter/nall/png.hpp | 426 +++++++ snesfilter/nall/reference_array.hpp | 18 +- snesfilter/nall/resource.hpp | 61 + snesfilter/nall/sha256.hpp | 10 +- snesfilter/nall/snes/cartridge.hpp | 650 +++++----- snesfilter/nall/stack.hpp | 29 + snesfilter/nall/string.hpp | 2 + snesfilter/nall/string/base.hpp | 86 +- snesfilter/nall/string/cast.hpp | 9 +- snesfilter/nall/string/compare.hpp | 6 +- snesfilter/nall/string/convert.hpp | 20 + snesfilter/nall/string/core.hpp | 17 + snesfilter/nall/string/math.hpp | 8 +- snesfilter/nall/string/replace.hpp | 123 +- snesfilter/nall/string/split.hpp | 62 +- snesfilter/nall/string/strpos.hpp | 41 +- snesfilter/nall/string/trim.hpp | 3 +- snesfilter/nall/string/utility.hpp | 168 ++- snesfilter/nall/string/wrapper.hpp | 12 +- snesfilter/nall/string/xml.hpp | 2 +- snesfilter/nall/test/cc.bat | 2 + snesfilter/nall/test/cc.sh | 2 + snesfilter/nall/test/test | Bin 0 -> 228157 bytes snesfilter/nall/test/test.cpp | 50 + snesfilter/nall/utility.hpp | 2 + snesfilter/nall/varint.hpp | 3 + snesfilter/nall/vector.hpp | 74 +- snesfilter/nall/windows/detour.hpp | 192 +++ snesfilter/nall/windows/launcher.hpp | 94 ++ snesfilter/nall/{ => windows}/utf8.hpp | 0 snesfilter/nall/zip.hpp | 124 ++ 84 files changed, 4101 insertions(+), 3517 deletions(-) delete mode 100755 snesfilter/2xSaI/2xSaI.cpp delete mode 100755 snesfilter/2xSaI/Super-2xSaI.cpp delete mode 100755 snesfilter/2xSaI/Super-Eagle.cpp delete mode 100755 snesfilter/2xSaI/implementation.cpp delete mode 100755 snesfilter/NTSC/NTSC-Composite.cpp delete mode 100755 snesfilter/NTSC/NTSC-RF.cpp delete mode 100755 snesfilter/NTSC/NTSC-RGB.cpp delete mode 100755 snesfilter/NTSC/NTSC-SVideo.cpp delete mode 100755 snesfilter/NTSC/snes_ntsc/snes_ntsc.c delete mode 100755 snesfilter/NTSC/snes_ntsc/snes_ntsc.h delete mode 100755 snesfilter/NTSC/snes_ntsc/snes_ntsc_config.h delete mode 100755 snesfilter/NTSC/snes_ntsc/snes_ntsc_impl.h delete mode 100755 snesfilter/Scanline/Scanline-Black.cpp create mode 100755 snesfilter/nall/bmp.hpp create mode 100755 snesfilter/nall/bps/delta.hpp create mode 100755 snesfilter/nall/bps/linear.hpp create mode 100755 snesfilter/nall/bps/metadata.hpp create mode 100755 snesfilter/nall/bps/patch.hpp create mode 100755 snesfilter/nall/compositor.hpp delete mode 100755 snesfilter/nall/dictionary.hpp create mode 100755 snesfilter/nall/dsp.hpp create mode 100755 snesfilter/nall/dsp/buffer.hpp create mode 100755 snesfilter/nall/dsp/core.hpp create mode 100755 snesfilter/nall/dsp/resample/average.hpp create mode 100755 snesfilter/nall/dsp/resample/cosine.hpp create mode 100755 snesfilter/nall/dsp/resample/cubic.hpp create mode 100755 snesfilter/nall/dsp/resample/hermite.hpp create mode 100755 snesfilter/nall/dsp/resample/linear.hpp create mode 100755 snesfilter/nall/dsp/resample/point.hpp create mode 100755 snesfilter/nall/dsp/settings.hpp create mode 100755 snesfilter/nall/gzip.hpp create mode 100755 snesfilter/nall/http.hpp create mode 100755 snesfilter/nall/inflate.hpp create mode 100755 snesfilter/nall/ips.hpp create mode 100755 snesfilter/nall/png.hpp create mode 100755 snesfilter/nall/resource.hpp create mode 100755 snesfilter/nall/stack.hpp create mode 100755 snesfilter/nall/test/cc.bat create mode 100755 snesfilter/nall/test/cc.sh create mode 100755 snesfilter/nall/test/test create mode 100755 snesfilter/nall/test/test.cpp create mode 100755 snesfilter/nall/windows/detour.hpp create mode 100755 snesfilter/nall/windows/launcher.hpp rename snesfilter/nall/{ => windows}/utf8.hpp (100%) create mode 100755 snesfilter/nall/zip.hpp 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 0000000000000000000000000000000000000000..56b80ee4a07f105a3a3ddcf00161650511511448 GIT binary patch literal 228157 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 z zZFiY9ujD_wUfuV%)~9!c{_Xh7j^Q_d`0tmN9)IGJo}Y&0bvl*ZzE$E^YeyulAGL6C zo2@e!E^qaI+v+FWEgrsRyV3IHC!4Q~eK_*m@uwsHlJdRfs^1SZOTFukgrDA7ki7ct zA!(O=+iX$#k}FR9Ue)IV=gj4wM7=xVvjrJnyz%qTXGjdCS8_A3Qtn zi)mjh*n0NC^a0CCQ^!5zA>jyd{_7CXDR7` zX7gvavb1wQnY=D_dqUj@?nUvo_tQG%o^HAMzsK8rG&L{m)&sAG4v+7-;JKeV&8wJr zbL#j{m!^-Ju<+RSQ!`I~JipZ!G3Q294xO@n&+~6R`R3=V-hbuh8ux3_UB2JD_x=OB z-;RFx)hW+Dy|a7xkc!<8H~Z*cz3=#Z>b?aZ|Ed4-`LC8th(8o^Mao_4`(#`@w6xXE ze>~Xw;1yql&n!9{LSNP3kM)xnh66JM@J+yF?wEOi{v1sHiNV>qfG)tl%!WPqGHV?m{Vw9ByFd@vdk*>m@LRwmfHUTT9`FKS6ux-6dmiiu?1LBc z&j73eTn#ug0rCOg0W|RC(~*g=8?a3h>;}9CFa@w%GTvecco?t*Fl0XZ8E_O}XM6!@ z8(<9Jmq=FAH3-Acw zc0d-AK<-!m4N2}4SdOP4qh463-DdQNr1O6MO*++0&WJJh1WP80jvT%2e=CF8S9KM z{rv|p1~6+m^Z*_LECB5EXZQoS7O(>FB;Yx~S$L~n=U#ZjD_{)Zt$49v3g8ET#ek7` z{ofA2?SPelT~@)q-sm^LUVvW$+5z+No~}&5Q-H;Q6Y#3k62KC`D!`Waf)8I5{4<~x zFoIsM3%CI=6R_J_j3>aC0LuWU;DyF6z$(C~KCt^f+{5#-g!0!Od z0OuBf4{#S?1inQ0Jzy_Dn-lr~^8pJ0{|2}^0`mo68Q`~oRe(PNT3Z;#Y`icy2Jir2 z3g8LA0>Ezow*q!9MEn8A0lENp14gt&e?9hOr!Q7~nR*8GyS1a{-S4ZUj6HSOR#{I?%VqcmNy**rpitfVTtY0vdlsJOCFy zjD7|j^C;quWnng;6|i6<>;mlc80-R^4p;zqFW^?dsK?t_l0k=H`{{UxifxTEK3ZF$h0bS3b{{Z{F2tL3q+h9M| zjqd?-0jK{H{sN`|((jJ!0jvc42CxpW{dUM}XBd3}hXIZPoB_B5Fc+}dOYk3XAz&Hc zYQQSMjexW$o&vN2-n0Yu04{tP_5iK`Tn+dv;AX&20LuWoyaIl})qwQ!lCJ=*fKji4 zAJ73f5AX@V)quTsB94GV0bPJLz+PDQrUDiKo+Lb=3-AbF&(}Z)I2bSj>);qbJK!Y1 z0>C+dM*!0RU4Y90Bd{J80NMd(?Lt2QrUD)TTnab|>)(BVm4HvY4mzxVF9N0jz5!SO zSPobL_#xmqz^?$Kur7TExEat5xEkwDn-bKqu3Q0F1=t&~4(r8Gzyi$U;{dIg=MDp= z^u@S%1O5T_0xSch58UIAW%*>ok{e~YqIL6#Vhdu6R{GZ-bHoUu7~M8%?0q3^nk@l= z#Ek^pa;@9_ZK#G1JsnCE#5)`R+JGNjM^n2m&$O{dbAY>3*7yJu@zmq|SRP^YH^XTagd~5)G zEbv2sAE@&$_V6bGZwJ1gj!*I6*8)Es_-q{?lFkHV-xlCs0Dh(yewv9t1bj8{V|4sV zkNj%jn_+J6t>cpfp5hc1f-wX9ejQJ7dPdMtGuc5O%koKaoGHyiquRz8p(PzC-AX4 zewDyeAJT7O9RvO)l*7ukew>gO_CRRJW}>9&qa)%t1ABs0z_1TBI(uGd=$Md}pc{>4 zkv~Kq3%ZA}NB9W(!>a4i|2@=;p8VJphJL`B-cX#$&h?Nt82G-hH_Xz)lW$9qZ{K=5dR_I%Yd)*;{UV zFZ;qw`@)*xmzLIHE%M?Y%ls7Q?!Z3}{BJsb7GnGvlTa)B83($1u-Cz5tQLn~Sr5rc z1Afj!@Kd)VOc?)`M;v_DDN)y3iL*IXi(LxxwwePS-ORVp@fm22os2fo|-h z7>8bRW|(p=0Ka;p+igQWn*A1XGX(u?3L-h(n`5oU9&4$WoGg|@_*meJ2V)$d?2#iT zSi&a({|WHn$U_s4LkKdK@N0oD8|HTRMA^j03s(rg1^C0j$Le@FUTIuyLjV37`0Ejm zuuHUY5HgkqY-ozbGqgzFdB}^!UU8x>FT~CY3EvS5d=~JDUidjCelYO6fcLjvP6d7| z@Ke3`Z!`HFz@GyCS{=S-x)yvDd2ws{#viR?P2Ea1>i$)ZQw6XU9j;wk86nD zx_*(?oVO!^{~h?jIzGc=|7_sfL;e^YukA}#06!RbfAQM{d~N{z9^l^up2mPVZWeg- zp920!0R9WWx8B^af0PH5Czk;K0Lo$0wLF-LJh;=72eDY>mVwUSejy3?RYU!+FKdCn zIRO6_;O_zdEZxtL>Fg-w!6D$uPJjBVfqxAABX$0zo;ZcIHjLK-*w-ESLx|%Voj+uS zsXrF@p3v`apPvMLXW&Qa{L4K0*8)Ezfc;y5?;Zeu2>8JP{Hq53d)V(UK4G|)jt;=z z9r&^U{IS6I4ZxoS{DT4b*8;yH0DcSb3j^Q}0q+{(zkjQN9|ryp^!SIoXZBwh0#h46 zes|yxLH=l+Uvxju1F^t=4g8HdKG)N~Nx(lAK>k|bPY2-N0=(M~p2q1R;JYCH>AL>e z7^Cu9lbUIqoB`d{e(WGQt#Rft26BpsLG2S0AxExF)J$>)gYG%dO{NA#CzTO?D)1i% z$Oi}T&tl%Z)N5XDZ|+Oh1OFlTpVH+o$_sxWtSB@jOT?Az-2=MYuX?`qIZ^y4?# zOTTJ9A^^TS@V))u$&OgyuZJCf*X>R5wwLszf$l8)MmKokrd{tp0Q_fu@T6xO@M+MK zuk$DHHHp}qpq=!b0Nqj0jnwJnd`EK51OGMfTXcLX%aNxW;##%~7RLDDf%Ad=2i;-N zT})mY=*HY|ak}-OD{Tbb9?;z|;^OkELAMKZrMkZvvVZRPT<(6t@qb|2I2yF0Jh1K~ws4~B+3P8L$XZUfyr*q1ewKhzKJ0{k=;q4yL60yyVnAr0{q47-UhmA z&|OU53D7Mc>tEk_;NJoMVsg4*;~pQ|upI7x;FJH>aC`_q8~BF;@UH-VPyqf-z#sI( zPyM|I_>Q)VkDF@H?FF5`ya~hEblY(kkAGj#-G31}@_Q=in%(Sw+&F;m7688<_}-5- z99QytC-7H+KhU~Ibf-Xa3;cZn_>+Kt$`79UX)W-jz+bFC zw}CEfd|*1V=LF~`fNmtp<~*rg@0j~udNHt^K}@GF2f zCiu5=6YyOE;P(JO%nzRPANZ-j2Z}4%8HR(Dzxk0v`0l{(3V@FV{&)a<67c^GfL{xI z%Uk@%VGHm*1K{dPN+yQ7iA8szFx-c@53~-2eFVIq;K-q2Bcj>FEx9 z)I{7<)bYu_`|dcp8YW1+ zodb05fi6(Ih;9?;I!+HvM|K?o-AvHkfwHMLL>#1(oND0z3j7uwUvJ(aIjy_md4)Ut z%jpYzEbtezdn)MezX%=aTLHQX&^0vg5`Gi#Av5aDZy}!Ry*2Hm}lpqmZ4Z$TF*?$kePL6JCyMU{(A^I@ zuk}O5m+)QwWEh_UKN4j#ue5#SNZ@}0zM=jjzh(p9d-lccSPQxg&^465WXBfZp9S8Z z9fyEF0DMF18}U~I|3BbkP&WPevpDx5d>D+SXX3`{_`%)HoAz+5KV0Uqi#u@*Df?$wBjMK;ujU({=_ zsaJ0%8E`9=KfPMVFl-~LM#%9SwuOSnYB`y_ls!e=GiCE;NS zPfGZWgg;8ytW6B7>Lg)L35Q5HPQvLD&X;hBg!f7Kh=k8dxJ$yr5}uUs8wr1uuvuHF zU&5Xe*7u*%=;pCwZ?N{7mXMR~$hj^q6+cs*VIAE6`u+p@4b9=kfxSiD59Q$ggZm8_ zEJ&{>QZVzwWFp5z%5;nr2HiScn2;a`;0S5a6S9tbP5gOD*4xT@sjSbJ_3rayxTNa$ z%X&MB-&C)@UDgLme3h*Cll8U>Vz`to>-Wm~Oj$oA>tkg7s+1Tm@xwkS*=2pd#J5Zp z_%~(!Em_Bz0)I}+dbF&!Ul_wBRUaek%3g=8t9aa7uU;(c*tYTK1zEpK)|)Si;nMT6 zzC+e~$ohw}9xLmYrNwaRC0SQrjHHKG>7%|#Ne`#er)72wL#4mFtkc7$^qJ&{VfY~b z#Rasi-z4i-I*H$rE9m*5PPDzT zOw{cXZ&@ztzsdUVvd#}@qV4w;0?!X)qVD*ASo;#Vs;ce(eJ<05%OHw^2#SJSMNtH2 z5kXOLLY#-ZG6({KfH>QrIh0veXqIL9G_5ROrj?})PS001nw6yuHu);+RW^?=|KGLt z-scX0_2d5ee7I-twb%Ntwbx#IIMY7Y$UHxEiG0EendgTrk)N5 zuakLxNK)Wp$6A>mgEaj;rSh|6R%-KirXQ=$GD*vj=YkN{}l>B{%=)Y_rFUX z`H)+sye^MX`D>N@R+VqA@_SW2L*>6#c`g5kM?R>|9Y4w=pXibA>5(7oklGx|Fys8w!CHUmYq7u>m7SEExnwpZT;(W2JUWut@ zR&~|<1#z>=3Tugvi2adiVF*xxxrJc)t@CHSWJ z8x3$QteRh5T2)dU$E+b66>DcCWXn0#6jv`Ssw~M6vozE42AH^YGQ@)o6*-$PL%h)d zgKMZ{h(ApZRI{k0w7g0z-K4Dosgge5{);OG+sTX6+dWy9J1Gl`ASZq?32qw|^zUC& zSc~szO}|4Uxa@nN3zIhMLbGMA(M^+Qmd!4&ETLF0EEZ2SMEs&5VvjpDv=pCTzN|#d z@E{^cs>Ci2;;hPQ*!{T&l~6?TUGAXQFm-|lwP@k&*(J64^OxcS>_-|dn_YtzjbD-_56p3pV!QlCR={(|EE{mZM$7nB!PqHeYMm52+^ zTFY@M>Vdz>ju}o21+(;>IfQ6`q^HNZD-cha%#AitT&Kj&SEM;IM9W()g ze4C?`M)poFEt=Rlqkn(RAqM;(gtVFX{wj9;u4sZJ=H_=zG0z%dUgx^>OzGDf)romf zS7#wOF}A1AF07ni(lw3`8R9duTt`U#$WiMPwm0%2jrUjluyU};tgV#m6Sw9`%n)l$ zKDPjJV}rZ3=MeB;n-UsX9^FKMT)11C>%r%2O7>O{xawY$cW!x=n-{vqx%DOy`_I3J z+7qZZgZFQB0?$q&j+N@bi zQN8vwieWma)g!vvmu)6moh}ghVE$Y6<{@vL39Z|ya6T=m+^ce0V>VDx)FEu|f`08@Su+#qNv54<26cryTd7x~fOWJ?~NM;bf|8>WRzq^a0O= zmppjn8B5ob9%K*r9u%?j&*(`R5pVq!`-0kqB^dc7_xs?MePTlH&t9ka-IRFcBw)TXC1<_i=dTu4gU^ zeE)}}A^f8H4yW9aK;-;dr;QB~!moAD={We$_5tmBlm_HatP!g~vgYS6;si=xZi`Z^ z@fs*KMBtmfP>R?`=}^&mHWGpD0zg9RGwFI8k=7@GbWV}}K_vg}9YFeyNl?0U1<`uq zVji|#Q>yX`iB^mYf=0Ba^{yt`ZK0rbbEg$e0d26R1zbyfNy(s1cBef9_ZV|Dt=l}} z`(`p|*SOP4$i5Alw&_~pdvGjh_q)?#^FVt_)4pN8!(`v9?zFrx&_33*ek%#DGy}As z+-Xyqffj^@myN-EYp8B*HI2RO|1=ucNY%6y_DjDs(1y9whGl^^UDLu>k>1Kg(5l>N zv>S)9QqwA!Z^wAhZg;0u^#<)BP3t;~@TNq8cF3J}02elm*EQ`|##=|h_o+MWU5J^UFciL{k8>wj_YY1;v4rnvoX=^B~Yc=iV zBI0W|6*StFgMBY)1JG@ZyEJV#i`kF3$$SJ`qW^KI`6TB4@$zBv9fs2=?_?Gxr(LNS@^PTNiKSEOml z)ZP6j4+3qGJI&GqG&~a{eQ=QZ9_$F(R!yS;LileQ4ceocc5*Gz&XC^c+-O0@lu>y5 zr&nno>?gJY#b4-R>fW{pEcZl^k$^EOGRxW;?@bac?(0Fu`z?tl0p=eio@JVcy10$|d64l- zcjEal5tNnS*{XS{d)v5=2O0a>p2g^{lZfY-=Ao`_!RUTDPgspg?B zYU-*V2jSBD(j$Rp9BSd8j+vxK9TeSE6%8 zW?eutUr#*aH4k-V8~5iR-8SkvyFHTX&&mTHtw%MMspwHxtJxy2gLKS=Ao`@bC|w{%|RGJ?geL?z=(8)0pxivtF2h{riX~N%K(mv~eE|GP2;<$gKSsCf_EWT+Kt> z)y91mKIHcLu?3zjzYaXLnuofojr(hmF+2u1y)uG@s34x(H4k-3+sUKspUGW_XGbL7 znJ1p7G!J!0ThejnNymT~ne`5a=u53SAZ;A!GX)I8J$ZDU@7u3+QbaMD%nE5z^Uga%_!ceHVz6d@=U)~JymAv{9u z$V82zq$OVXg!M%tYQ`t{ZD!CiA!>||VXTFc@CYa{Y{V#<8y$fiHTE$~20q4ivlIoH z2`I=LGL{OSau&2Ox|78C2kqE{5D-4b+f-;LOi2A3ArUpPdL$KGbV@}?6Q&!!5RnE5 zMaws0O`5F;BD$c^=1->pQHpV&&`cXlU5!VTN=j^iKA!ZPa8YcrwDu$}04vL}9 z$-oE^Mil10sD;LEvml%T!w@Zv_UJTGOC0Txqn!tp|40v7=(9K!u33-5Y+_fA5Us3{ z&EkX=q^M)V@G~PCNwiDUscg!{an34QQ(Gv1L|bWf7i}e>HdenO2*qe?GBPnlkG1wi zvTZwSJ8PV;AJ()e97%ILJg}xKKhLkB->a6XouDIS0PRR9f)X%HirL( z<|s8Rn^G~}mV@7Dll?TxIK14KhzWtMz&dmt+)EsWA!3Lj{(33(ZG%#VlBz~eQBrLT z>kXvT5>SF+vyE@S8dwGcucU+FeB;IZEeTw(tx6=nAK@oTZ)1B#DG zj7D(zhP3*Mn9hJl*zcn?f=GN6BbaYUWH6BI_s3H~6?hDLOK7);BN?pHn#%*;1D~@$Y~cp*xNzE07RWMe1ixg|r;& zoY0Tf!)`XBlV*aFe2lZ9mxjV$Tr|Wi>SC~S1{gNXcwA=lA-$WndlDKe+|yATk@j_{RFu64Jgr67#{g|@n-F!e8KR-%8FU#MX((pfCQ^T~#QVasu=X&%HCU2- z+XZ|N`&6YaV0bgQA_yT=$H!j7m3k6U>zAB0Vk#;X3-$ne9@JUwV_`<1eLBnzvL67C zEjbTW!kIm)^y^Tn(l!_!WbC0Zpw<5fxQxLiLoz%J+aACvk>Yw)TCYVG!d7aHC42?PlrVv?DX(@ zeDc%@H6kH-TH#dD(O!tqO-!CpGga8_XVKW5>>rIquSy;PkRP{Icl#HhqyR_su&)PWs(mB^EX}?dt=Q9^7>DgJ?4LEqOS|?}EwCM@eFaSK zo18UG^s|45+V{7A2ond`KSFOvPo9<~2HJB_`3%`g!R0AZG}iNvu;2zi+I$_# zgN;cTd5kVUB2C&18;_(SWg=W<8DQKEx^2LE=t;RBru+IEdx)aesHYH^egS{j@MsiF zFkYY{HVVr#G>)J3wmemIo{FeDqVLi>u6}{zQGJ%WL`AGM$he)Pf)6C9qUdlG<<4R$ zJH~UrkXOd2qFz*V6|@4d4hHyzzJw`{0rRMcQYh+!sPqf7z$si*K}9?p1Q{m`M1RJ8 zDbhd2jao{I_IMQiJHW~#$aX5?Q;8mX13=AS>NA?E>*dHqKL|XNr9Al&m6L@GsB4%A zvjW@Pi(hLITxK9-WVte!K!Ore5k1@-iB!&~1yrsnY ziowzj!#pY-xyp@&Pxw-%pxwbTShA4sTu_Q0BSjzBnWaRsP@uZ7Ji#mx$;?9AAFC#F zVd1l@l-~z4%Wla+Jb$WlIl+kH-KF z3#jo}HWc|ZI^9g;E-IUwuLtmn=&F^Z_d5gvd_apf$kMFFvO`Ih+g*yWgp7k77`Y*_ zL`wYb-aONvAkbP&D`26WodHS#$Dj`qDCMM8Gz4(UB@%5*S0FH18((ptlt@{M$rE_A z%Led9tX6p65S}N>5R~%hBG0r_u9?iTw4(-O^R;X*1T>q4aM8+GAg@LQDIw9|g*SOmmPY$Is5puBa4iCH~gh#${M&e(%PQ~O^ zNsRZc2i_}<;Qf^oT|9ZO>{fj+sJYfW7I~MN^QEc=ql8ORmUL(PuJ=GGBNS(S%~a#T zuj>1$#Cu5Mb;k(j(%GfT>MU0IsfWr_q>>sTjiw$~MMSjmweOVbY?kfN#=X)Z$fwab zRB)L6NB~ZEsxVsvLEyzp zeOD)q7d$Q@FBd?)uqcN_jfccIk|2jm3poBfJ!!Uwt0y##eb=Z03B*1)O5SW`A7FaINJ}B*E8wT_iZ4G7v!pi2@IabmU#Svsudq z+gv)EI~@g$8o1j^k8kimUP;KFJwV3&A&lv;<&1~KA$MCmedTO!>9vQ$8Ko^wF zyKuPF$kPHRinOJ$7o%iJloN$+!_xIQeGnmO%S|$5D$?|mP;s_s<%2tp*xGOfGRZQ|on;iW?D1!o8zoDr zJ4+$6%nxIhJ(A^Wcb3J>@~T{%oRTaX-C5Q%%P`r&evm8=y0h$HmbJO8D6&0Q`mj68 zL1rmWVwV1r<+M8s-}6X0fHK6XL$Xk-y10&-QKS?~Tke!B`_`z&TaA3Z#^bw4JWtEU zd#5wQpO&%*F*I<{(5JAZtV_`F0-R-K&^(Db5vKhjWxL)lzdWtLrXK)ATZckR^AJ6sJ;f`_-vs^3t`C!Si87J=S>}2HYRrN7Q zI42g7vTF>(mq=MU^Kyv;I@J;>ugb`#&x*m5hutkW$QDc)!6>^W3pKY33g5^~nKO`C zUX?7T-BC_5%Ga`0zm+Usy0iR;S-zJJi%4KwjCF2`erJ{;gIG}?$?}W4Ps1T0QWmE( z%XG;S>y8r1EP3OYwYvGx{jSMq30Mtl({Nf9Z7aRnC=HB!QCnJY+w=VbcvY#!A{Tjbnxuaq^fjcm}eO3Yk{ zh?Fh{Ydhy5@q(6cU5uGxYW06F$Np!KLQJ1L5 zc|3zre)B+iw?4{coa5r5ee#k;?<98*wMSlBMdvT|qL_&(1?|}ux=P{NMn$G9H3Fk} z#xn<|s!^DXl06c|yr8?4P+U&t$%*Pbeorv#{ZLAnS1MoAdX+XBk(G{j;#ndFCz7y~ zE-dh<5`es{rArh$PuOG>pToe3*3b2jrKbX1>t>3HlwK`Z;x;K^+P6$g)H~%iEpO&_?{V`&iEvRtC1UGFx%%M;J_Lzv~Hhq-&}D{45XIW}BQH^*aL zfF`qH;q6-epG-WL$=X?SCZ_x>PZ@`M$fhD+FI1eeGE_Rtwb@cBODA<4X5}R75@B@5 zL|tbsk->FXN>Jx#iTg;R!Lv=1H^C=U? zv$j{Hgz2GYNy25YWWUI;EtjVhfmp$y^-YN%w1jI6nPO_H^q<1ohI+`RAYZTbowBO+ z<#2edhis)O>(UfXS=k=3xIB-Debz&Emnqv&Rn_1Y-GgO)upV)(?*%v6hAZtd5mdoF zuwak}N)qz*#w)g-3|AKh{pHlQ#zVHkO}3r|CV=w&?vZT4QHerl+Zf`Z@$4T^SIJa1Lv&SQ~Gr6_f_fre;j+?^6{bLZYuNJ zvit0+cCBCe@QU6O^G69w2j7ce7RKi8>ju|l_nCL{_apo7dw2hZD~?U>_igyM({`>L zwBhv3nBAKn3OHk2v8wr@H9Zpkb#?Hh1&%|tf9^ZC;sO8nwk-H|%h99dTh3oM{LnYg z-E%bTw zPLG@QqJL@U&Jk;Zp0xLE-=^CibGwdd5z=?U#Q8xbPY2Ce)a`|1*RJVWyJ7q1hwmLV zJ9<~mjiZk37+3jx>(4iwO7Oq)tu?(uf2i3ur&+6`J6l^T`?MXsILE&KnY}$93y$eG z@}V=iV_&@citnqtm3_VPmIc+b?pqSE{NDTx!-(q5-!NQz@#b$B zF3foIH>|fe()hY@A1~#xl=J3qSU+#1VSf*dgI@d%>)~Ah4(s7v0Orwaz7uBEmSA^^ zru;#)y0UU~^{k4fNYgW13$X)6)Bh%xS7I-RrvK$Goa%dUdoT~QXVIUUj2>oN*Z6A@KSz^m-09>^y+V{SJGJj zcqu>5OZo9$%H#5cSAP?|lEz!RUdm7QQXUtzy!xB!l{DTu^-_Mim-3@aW-l0Ai$w&s z-)YL<`0~;+Zvc4xQrYxH)Pxf3`BshHwwhu*WU{cm)q>jUnzAOYLWWe&om<#M-6WP* zQd?f_#S&(M7rrC?qe_<66c#uA6iUiB{S->dH~kb!$~XNKO3F9=6iUiB{S->dH~kb! z$~XNKO3F9=6iUiB{S+#%n%L1U4O%;aR#KIyk zr18bsk_8KEn>zVH^8~tfP*m9DE;Od3YN0n3m{3;kRRavEuEO-)^eZS?m0mSJ`aibf z_QC;uEw~dMyJpTs+!;AynOeA{e7+ZG)4Y_Q?xpXuSTSN z(^n%>zGstLP|7!bH6rDkz8aD8O<#@leFIO%#(uVmqFIFI6fS6bD8Pr1Z+a+5`KE`0 zly7<{NcpCRf|PH1C`kFHhk}%EdMH3X3xk1|`UoK9n?3?a`KFHmQoiXUfRb-=04n(= z2cVK~asVp%CI_IBZ*l-C`6dUTl5cVVD)}Y{pptKL0807PCJ%s0-plw)^)miay^Oz9 zFXJ!O%lJ$6GX7G%jK5Sb<1f|A_@mA1n)WB-FRjV(*Y{v2Y-$L}D&Cs9_TA8(2i{xS_QL58*Pea-jxM902z>Uc zeb)peANZ^;_VBI6&7O??E^hCfoOuVoKbLiA_0?~_6aD4+_ven=|K<b znQ}I7J-n+&#aj$PqYIgdY8`kWb*KYq4{`Z|f{ExIZ z4}5;_3js@d{wl(5-W7Oz)aKBSo(`F_JbB<;>m8NFb;gWYpZ+keqN1p2kXiy$HRVn?(E)@ybDh@|Kh-*8?Rb>@Q)8--!B@G{7%d7uX^LA zlF@JeKDF@CF|Ur>-+zI4X2-`VPrsRU-v=Y#-gIW*+~cQq{{F*TZ_K$+KID(HRXvBT zEq(a40|oEQPxj6JPpnT`{^+2GUbrgc{q%W$c@Je-?9H0l4qX=){6fe5Wm6toQymxh z=In$2IA8c`=&$EyZhqlx=MI}++xz&gx1Wz3_}H{-LmuxoV8*jgK3w_Kk+@X{zIx)Z z!*>OL{DjY4ukTyXZ}{mi9$EbQJuMU8^&w0~+WvXWzuYmU=;xP8OIFP5UD^6G|B9+bZQlIz z!!B>!Fn!rO&0Ze)erfdw2Y>zbn?pAbIrL=Y>6`Z!e7@tYx31ZEYWwkHXZnu+?t`)4 zmOg!|bjmaLzwf_)(5T*z=DgD-st&KErKKA8cjm&^^&_(xCxQ{kxOx*+(}_f4qzD zoX2K7aOv%1Jx;#1|HiFnyT;&#)Qcm$sJNQog6? zmp3j|6b?OE*=othLEFB5J>>0)t9)~hJ?7J|FtPN3pIa5YHfDJFh<6uPWsID2YR`Mq z-#RHxp0>j<#GCzLkeZQhznu#Kk=wn4|l+q}&^ zS!%bxNstUDaRdANoj{$M#|(hs^nQUc&$WzpNNye*V!<}rRJMW2-lvr96Z~Rtoic!6ir zW5y%f(zpt(V~Mw>AN5DJmC80tZw+{Rq{>F6AN4~vN@e5Hi^d__T4g(>FUIpu_Gpz& zN&gWx+1sdW-}Kn_$i}E_X8KUPvSM$mvN`F;GLdbkvU%x``XC#pvQyJn;Z0(D2Lm70 zfsqC24_T2-FjjE3EPW83Q?e(j8r7t)NkBG9Wf!FXj!3b0HLlFY z_!UO_L_B0>?_q=>0)66F8tJc-A*lv!pXd|6%1G})4(V-d;QU%6{Sv_qGG+tLC;nC= zJpd~>dyc`IfX8n((nBdfLglv^=`ATgQss9U=`oZarSf}?^mxjTHs~uQKJf>P^k!sF zE?e&tf5b??iK?Dw(B1<+@y87@@ITaGk7vvbp}k_Css%l&4944^rX*T4Dvc1I!AA_K zL8EBbCHoQC+Rd;bceIs4BHB6t<*lr|D>l`Qdm_;(Mwq%wQ(K5ZwCi+@VM+cKv`l<` z&(e)jq-@|+9J|YYnVGp*Sb7BFWkO4Qc4nEdq_##HR$DUhNkU;syM&nbiT~Up+EY&; zMpipWv?l;!cFohPXPgfZA++;xv)~q!aW5)#HPXS8WWWkb=*y^X@MLL$)zZ>fjo=S% zd4C}KbPW)~Nk(n+8f0ukuL+Ju=*dk&>zoWT=r&i7u|5f2z6wJ@%0!!CWfS!yC{bF^%yp`21A$65I&4>TRnyi#L#s%;)WUa*JJp#B^bIT z!N@A|%qe1EFT+RgT2a=0IC>|@9gMQn zyN0Qvk&fynU5Lf)ME&OV}gg2>t?^P9>ZF4Pqr;eA^fu* zLt8=^x>^VCEA<%mS-~*u^k9YXbv*`}e1mgJz`y|zHb=X_rg|SUjL1NoFhh!yVG;a- z*_rmb4<2oda2BwnDi~uFISZ6TIJq|0xXLV0A2bLuPT|}$cwDO-rR*M)LkG-sL}T#y zRdI^pkOxDA(H~QB@F?SL6!1%&sux5Ux6u$a%D99AewXxnBzR+0Jsd4v!PQAXH}#!p z*eeuY8zkiWk!slDcO@f?8|=(CR`U%ZzHhDthT`KiJ^<1ke6hFUq5!=RZ7kRP^NF8G znxE4V#zh)#Mi~!izAcULwWe)*CK@L--@k~Dw-i?TIUQlVh;a#gKWn}(iI3k>R(zZe zG9rh7FOqhsWjpO?T=r8NB=jL{uUA%T(9sF#_8}JKTKOjzO#_3gl~2{Jq(4DzD7J4*E$QS zR&rPHmd5mCa7;U!4UQci9K?n$7l}b9ywG^kEY-AP<293JyAQl5XhqCr#usLR+1sdV zhtFN(gEpZ$+p3V|+S?M4*O&{91doCs{-{a#vd0sZ^w8k0-ciOazf-&{uE6H&w&MTz1A; zhRt+I?)Ls@G9IOWFoDHjDR;+A_=C$-+#qaQz=oJ52$yIp{l=g# zZYTj|aKa580UwhN;aF=5lIYEp387$X)}mYJJak|6O}?OMwb?SM?nDI z@Ci#nE3`5`qwovs4yAlH^4MK#QCi{z5FLvLBsgG5}hP4g)ZkDH&9^54fVO z(SV4d1c$~)Imj?V&kPtTHrV#oH4n}>Sh3q-{^o<(zi2|Sf%~8J}!i&Za<3+MA z5VFou|Gi9A_NlW>kNUk)0pm}iN3e_@_5TqEdYk#0?mp^o%t2w_dWCY--;TC5COHfF zYd(6^{~`i~i=Bl$>MIU8>hBr{jypX#+(-R~%~Cx=7aPx+G}{4G2?c7@|IjS(81>JD zqDiRE#sagJamZ`Tg+{bTL692t|BHnWOAc@{B)UkhHYR!$=uv-F2EJ=TIu|+_+(!MW zn2J-0;dWx+W^^C*^LyZIl^7m15zJBlBg6ogz3ME}qrT55e2ItTzA%B!QUBv4Ri@$w zkxESxZ4HA!OcjD7+DgB%>X*nmO^o>d6){Q~N`|Zl6+wMwF$Xn$w@C#xQT~2Vr(&w) zp#BFy!_Pt18PpvyOmNxL&N3a;A7Rij-XQt~meD~y6XTz8(WJWv^^52VM#v&9qzh$G zzlt%_NOBhPr>JyLPwR=o(au6L%ceMFPO@WulWh_4b}NAlf(8S@)~oY@r_4;4(dO9E6HYHmwvPT?jpI`=%@ULNxBBviB)v){{OVnnRRB&N5xkjp*uJ zresh<0j_B4C-{w__96t95GLM~CzUqMpnY|71VK^oYsyDO77F`rKWJy$)xi zjP$0-S0 z9tUe&O$OqkHYJTV@6+E zQtoe%Xs>QY1$WNi8h4>DI#u}5u_5+d{?q_4GN5M$r!)CzrF$PXSS27Nm#bT6^Lk%2ioQF~$ zLq)f|Z9qMBG@UeE3d0aE1A!H5rO=4BevXlVmgK*Iqb=WJuwA3B@Lv*PN8ra!ajigU zPp2e_x+p+ooHB$s(K*2ZQq&Z%b?!`p7(AjzVR_g&iK4+0pBeSZK;*itM`Kvxb41uN zV(n2zh=jHT|LQKsAo_$P&X~H;2NE-@u|f>(N=bq|h?3A`ssP zBx4lc`3UVHi{hg0MXZOW%3P-?V2DuOMbr|X5;bHpay_M$Ju^$NI0@}dm9fP4g^*#C=03JanO1H?`@ zN5c5KXO{S8Q53yl!8A5n;-i9$N??VBZl9nc@)n4P6NaIqof*PtJ97@%{Z0mF?~X9& zQ<7of#tUXC&dYWYt+^rLoabeI!NX1c6B6wG5Y&M#fXCBUD2=xM(JOGL`EhiFww7N{jgK@MA!r5r9VK z%hd53h$S+au0=C4Q57;X#9C?6&%K}6fE2Y-@sWT=qO^#AidZ6(X{$m;BC0}ShFB}@ zikSNq+_^O1l?qO^$rSR&DelZ-@Eg~SZ8e6cw9Cx!G+%|`+niP9oI$igi~!eEA2D}4u#eQ!sCeQy=H!xZvJEk^(v znZFbhKV2fzewB<&RE5k8u~zEQxm613HqA!@8i~>(ex*d(@5-T&m?8Ieg>go6be-kI zptOj8NMg{{ezu0F3WFJ9t@N}=?jeQLVYT8T0c{Q6Du{nwBGH!&7>TF~i5X(8^vwhI z7;R3;9&3*NO}ig6wOUhsMee6c-v-bajYPCQ-rF+%H>r>Ik2N`zK4yru9z+o%zkmem z3m3VeKCyC}d{HovAR^X%2cWMq^M~#Mep;+JKa;BdWq+hFB|YRm#>KMS`sxByz7) z$mg{j0cd32%{2ZliA;MmGcr*XGBd`_SFG#?3QBub0;V-krzs>Mh|RY=Sb z%bzdJeM=!N(R?JJk@$;T@n1+J+PayMXl_2r$oL}J6(5PZGjBLR)XpQDKHFOg_xYDOZeLSlwkEA3&;EieQLZUKLh zo2!s#YdHeY$h^&Ue33+^F9tC(Q57;X#9B|l##k$TK8W$@`!S3k3@UwFhN*Omjj7E* z<3bQm){eTYmo5te zmA(SR67+E&mIwuvKK#Q}`koI{(?F$f{VAO8lr5%Bpw->0ic`#G|qp3ELyH$C~dV}%-iD)lTTEsspy+j|! zVJ{I?d5Ia&M^MDb^mQE8mkNFKS8gmA-MqRQhxcQ(J?2 zy{6LFYM43%RC?Ag)=D3%VF~)Q3718IN>3u8{<`d>E^7lSeHMl#=xZ-5aWkm&B^ajC z2VR)E9#mtUqS7~Am|6!aJtY`xrBAxBM6xc66M1xXB*zv;7lTIJD8vv-V99Gl-8mlT zYvFX+)Dqt*>NPmj)|{?4VSPQ0uOs`!q4zDpzp)Z{;e3tDMwiJ%w($~v4P%Q*S8La7 z<7F;*%q-9XlKz}Y7vqf|TSH~9L;Z2SF&k)>W`gWCz91gHXeOE)XPbj(c1N7iSqv-8 zS61+ntA%kMrwF#i9~g4sZ84IZ9K5)gV6pIK|o$9fB&7M>lq? z5!Mb!@52AYkT7S(#`BY&5rKO;<7@~JCfAf^fQx1_KF|&RzF_F@{81S*Ov*<9+^vwDVKZxO+)Y-cGNC=`|qSx{C`WTuM?W|`@Nf?_j0qoBl0Pc4{jrY9DZn(4d( zJQ=7kM;G8Fbju)8;8XYb@`44aX6~8g1#=7MsH~+fvtSm)6z2g<4u!^mR7jORnsFQJFoe0$!lI3HF=fhE#eEkSi+ zi4uCc&A0C!#q^;%7Jxb8R?m7zcOYKsYL>cUQ9{qQ`DSR5kW%TgBM=>{MYUx!vAY;} zWu*&52|ZBeJ2F&>T&s?8zCx_iqT0mb-!W$UjwCvTL(C#nv5cNK^S#2d55#h_Y39(Q z=e}1EQN+;GW)-&}xl)0@r;aP2EJoX$z1Q?=%#5Bxs1()fEXeGFz_@Ra^ zp~pjf*X5}4z;dZ}8~DzGg~O8)Xww|&uyyBHG&BDWCG$UZET|H`>&@_rX;*`K(E}xA z^q7e6U6y7WRP`xg%$bb&&i5{Q$l;0^h}$g`VUVE5Nh&^piXdgmztvIrV>ca0R(^v* zpSlCfD}Uw0uH|z{u*wf*mb$4J5%)9`LQ8$DqzfQ@H8^H25`F=+?4%LY@d7wCNAo$E zgd>iC^9{wxXA|Je|4P9$Q%u#LbP~ePN*@NKud5q@K5-H7I{!J&16899RswMWnE5tJ z{ef!C&5u=unQo|o5!p`E<|gVTplY-QN?-+0mn+m@`er?vG=HortaU>T{&y!QY6}xJ zI$Q{iwo(av2-F7^Y6xAZqd;-=0;lGEe7_JE>u-Iu=xxLB~odR!~q&xz#dbYxOQ>I9S@D4^thoYZ+hba( zorRp5Sc+*Lv2qOi)IGC2zm<~RqmEb$Sc9ZwLBV`s<)_ZzQ49}Oc^w)(PdGk;dad@f z5?ulHskf_^4{W0Xid$ai3WURGE`*k7T`6n+AtaYUk_`_^q$Glt4G&6GlLH$bl&D82 zo}C?u0Mj5R6j-JWUSKm&@1PVCr}b0u5%?}Bh2(web}pB2D!XlYDY-1D2Zgp}zR2&P zunwxDYjGWHcF^irE9_A%Yq1TAC*7e&T}l`}b(@#-2I{4BRo})BvHH|O8{H(c#j3iL z(o|w=C+nGq2BuZDgU(M<6>{KwsUad5gYIUfffCU0%QbTggb-3gZ`n6*RT}!ZY3Qdk z)Z7aGk6~W{cJ~yvG4z}vmK<}B5X>qvbgYC~fmK@meKnb&qub4sM=6x8gisJeCrD-V z;9zipB?(6H%*(Tc>wFf7k|0%&1a(}C`s9IL*%CiIOqJkXt1iu3AT@ptuH9fLrW++8 zttp8H%`YaOhP2+x*=_m7e50@Re$H;rpL-Fe&7)h-(LQy1^I;}ksgH@nC>ujpFdcO3 zS8F(@H06W{@gD&2yh#bUdZK&34!V)6`C=DJ%XWdk$DJPh#e91_hi>R%VZw11lB2`c zLH9%3D)ABOXblx&h9``0(EY}diskJ^QezodcX+Z&$DANKX>8pJemOyRAT4z>WW619 zfAWmNurHQaCxG>aCzhB-H!3ZkgU;>9UJw<;*Gb83RmVUTqdT`{Y*oZRsVoO!mb}GN*)t04NyR0VMVQKJ6n%p(XJsv2 zm5st^rIi@Vq_SO5b{NWv_a(ra&1BT%6fbH)ITyIQi&RU#6- zOwFjJ=VMSQm77}neJ^KjVkzIrRb|1LlC2|9vS$u{&&iUwca_Y?-B9w_9{heHOA1tp zsygZ5s>g^Jsv`feQwWJmVnxiWR6;+F5RCNNEqhlhu74ySA7g}DQbF6E&I~fAa`M1vAmW&sPo&F zFF4R2#`zf=akv$jaP~G8dHH^8q!U7PHM#e;QPrF}p9ejiod|ESPsuX&=>R+tLA`c` zCd4WN>EVcNE@G%K+Dc=;V+j1O{a(3Ns@?_v54%^0g|yraO*iOK7@xW`S1-JQLNVRg z1aY6b*REbjOVZGRs{F*&3pXP~keB60uFj9SUINZh!1L#0hkiV6mPXU z@+vs2m0Ek1^3yYsa)c@4$uF$6yn0ZLj!h5F@TzbTJx9~FR$(kokowD@|EnFY`hDuw z%2j7rTR9i6kgLuxe$C9M?vI)I+Z0Hl0x6g^SA=yS3w-L{%rBT-St`QfW%jlF(tWs2 z6P6&eFXtC5nO!NuI?3#b`~ob_!#c~{@%(}s5yn^leCpoIFTE18R2W}M_NhCPzxb*S z(&SgwaW~8^y%9>oQlK%3>8<(aA+XMAD(r1+V9fxjj6 zG*Pbb*{Rld+$N%ee6&PWcdI&{fbTkkuksfZ(yDIB3y>WrM!3nkt__J>OMk3TI<>?g zNhm$>Lnup)exvZ1RWc0>nkmvQtFso&r@&Y2Wbvtc`I_RMLr8+2UV^KxnM;?3aZS8g zH+l~VGZ?Ok%QP(&aGC$m9FV^T4!;lTyok8WccQ2ueBG4}u9bTKfZmw<6o2K1(o*7{ zMsJl`Mxcn_dhCUSd^6GsdZq6~8!-T3=?YvpBb=ucZtM*be+E|bw`h=MCP*dW!6$&I z9L&iA^!WqmO>LE6cXfOQ08fE~`w=DNNe#Uv=U85aH|rGNYITf>#QC!)e=&`+;Wtw_ z&^6v^w7Z5S{(cn+lrZM}ZzL|9wWI#US6rL7>jbihq0KpB=n;epLe*OKC~a3k+br-^ zTJVGVR0i|0xH6KDQ^hoTtI@IqIrbYR3_N@d>nli3~HB6WKt&=TtH@I3npd3pH=~#M!%t5b<`76GG z>i7rv=~X#y2*p1Y4^%qn<+=>T^r||}zyQ1!%<8o|mf>@1J_DeuS;0kgmhOv5$X`sO z7nLph!OyX_9SQjsvU{u%0~&*#<*FcXt*pn5vPQ2#>F*lkQxl|;@Kmhr;bZ}N3cVm^ zX|_xGl@e$dddsYWAr$0rbxeuIJuvW8`r(J##W4f&IjS^Y9d8BKY%p+jDIr~&3seWa z{#Ky)cBtb;@bC2CpE|Q9K1D<%5*>4dx_xJ_K<+J&*g?mTkgs36JBS$IAbLct;z6W$ zDm?)=NeAV%LFt1EBmfBsQb~BqK_@v02RRmEeW(R{td{_L0q_~*y4x@;@$(bTdyss^ zG>|X2(S;*<;Y?+3U7jDQZ0N3_N;`DKvMpF$)VXT^O<2Gymd@%J7`7X_u6_+gVs0Ld zd{rV)omG}y^99JSE6(?{M2pIR^;DOt%0E$TsY7U@D*e!XQevkDP0b%tZAA&$ zR+w72mG~;fa<4kR2G+G;<+iFqLcaRWQK9euTdzh5^`ezciV{pw$r116<-4Dj8Zw^mP{%Td!c%!1{q<3R1%)DJ)DyTaMwDFqLv$w zWgV0-rswAu;x%7-?-t2QB1WV);b-(F;wgbzQ`q(GC{uP2llBK`h_(I;lo-0%;iv(y z2JG4?6OO@X+5tO3OY2ar8>7X4WVpPTMQODIl%LgM5Vrn#+n%`5ER;Xh^ zxd>PHIDp}J7O}lm7422W^8i|>fy`xx<7aeF&GNNk&5jr1HL&tH`UDbk0qQZ1Mivnd zDgdKt8R|$y`)G(i6x2>Y1?^SsLpRA(`w)%W=LUGxF&UCtBB(ACPQ(f&0Wqs3Y3ZRP ziN=zjkmN=*fOdSDYJg%$ZXn5tVhea0ZvbM_4L}+k^tPMUAEoR#4z!oG9W=u$JBY@1 zBy;Db^C8C*utr<4Q-?!GA>#J2HN?}nHN>Q?IR$XXR#>C0NL1FW2HJE&qj;xbLs>&K zwq`MCj;F*MaUPPpq#g7OedBfzleR++Oh*ax7_8A&^i-)-Du(1kBso#g)Ao&ffS9xgNP~m6X3_fNH%PSOK&3iE!O%p8rjhfC)B$74&{c`kl#o06L}@sxEqK`yP*i_SSvka;us7}jdhp8JP%C2 z#~G7OvKz-FCXM+7FdgGjLtV2$s)pH}g&0I=6y$W*O*JvmxQ1U*qm)Dlq4mG5?AQs( z^&}}*D2>}8*DE~9;rU-k#L$Bjj!Q zih(wEKU+xMkcZf|$n)|~?rh{XE9==%_R72~eS_5UHK$0DV5DOaoT!~QM#m=H(`@2VGLcZyh@_r*QwxV2w+TCwOeAqS z<1G4>6ML7#!6Y3+uhckbmm00CP$^vvrPH5aznnpovCa=$CG8aem8O}a1I}$xOUJb)T;d|X(hSe-C>wO|@}f6TNcYNA z6Vt^nUM0#WU{1gwZOFsQAm3zsG~@sqM6c|yLCug)Ys3s4@g#%wADKZ)Y1u4W9QBN$ z_ktXEA?$T~%vUZdhPEq78-+c+xue3KXdLzrkmMmqYKd!=rjU4m2fAsW}`A0(LsNiDHkNoIE!;#Vz6Z}})mqOs(!BsmeD&>lFYJh2m!3!h?7 z(4vh!F#!2A($5#A()g_*Hz(*&*|3-3IaK7po>L@AnMH(2PCt8WGncl z2=Oneo5r5TM+Pc)`ada1|;qZyX{H+vb zbeA@kFW=zXZivVNwsN z_&gLhf7ZRmZ@DX`xvdF`r72wF@F=`|tQ5Ygj@hYrp3p<_Smd3p+Qg=H4W}tktQFpr zDr-zP%_9!E*Z7tCit9Br#gEHkfW{yizGYKm&XB8tZ91`exhlGq+WY|+q74|CCJpgT z!x-aX$YkVQ+MGHXXI=f@h#}hF4HzPe*pQ!Lh&I65QyNkXLlO_Wd*p)#4DpP>|GOcy zs`II1Lx#7;GndMM=hX2y40*)EkTT>wVxnGPG-AkQnnpGj8*&E>(FWYnOL`=sCpMiq z;_i{38ZqSmL!fAD)W9wfFu9n3^u%*;qqJ%@tXf7^(ezI*hpGvPXgmQ80nPC(>Z9v1 zptqDf56NFhQtk`N6VNlryM!9QZ2DK@TK(aLQ04FvLkNyO(YU9fwEn7&*}X9C9Cg<| z2zi%gZEz_6+s!H)8a+-s5{6t4L$m?C`banIgdr!TA>CJ8ZZ4!+sa0~Ubv@Oq7VV+i z;d#{_zP)ip<(O&@IyIE-(Tix@9;D0h5Dd`<98-oALvk5Oy0ph|rM^oqqP_{i1IP% zb988f^F5{Fh&m?p!Q&MFa<5t}@@e%0Os{Zct$2ka9iId2iNMyF#r_1R2cFu<5`1O z6B==`9KBp!EQeXR*kwHX8ir^CzE_4M^cUh$4?}JsLwI)#x52yt@5(EI}#|h;4}8W5l1roT`So@964%Wh&G@@hBV|n47tO@kSgT4 znO!wE7(((Ck}CNWx|!}?pXhEj0N1|>o4OfoNT6m^qH#BCe?27cg;m;`jmoN>kW4<| z?jB#{(+K4+ts+Sd>==PwO(2I_KhsM!E)jroPbUJdNNFS^(G5BD0= zpyS@|4GcB1tGiA-dVGbe&p@$OcypGl)Ojf0=b?BZ@-D8fU%kJjSdK9~HPHNC zVcR=`r*Sl5(rE1ntwS8Hdn=UN)$u&gCP*~e@}zMzV$x{62yH0PG|KA=Ejvqy?GkO= zWzmR9qtOm4jxwNWl+bJ#g}7>~EV%6_A* zSsJe?F=@2LgtiB08s$2LRt&ToB^qt%(l{D1X*Aji#&HU08s(TmI}Wt75{|MqEo2%;BPNYTJI^>0t$6%Gb?A`WG6nsu23jGZQOBm8 zV${q>H1623H$aj$7SQ_p50#SVA^C{3jxhnnqcq(6Y0I_@6|hVWRO` z5tBxvVZh-4nnpRV&~^f?;x+aWZ92sxK!4=_Y6PHbyp4?jjys`9D;PgaI>C3S5N}FF z2UjaaJ4{85k8Udt)>9-WMZh{vLy=Z+vr<$HMX{&2M%~dOjr)g~w0~$fAjdC2(hYFjJiKi>L5{am zpiOI8ptP-qHv8-B3EHNpaZeDFww-o2ax4RyM%k;-&I4_sM5FDE8b>20jYhj7Iqm_P zM)|Ko%N~yF{}S!`%c2pJMx))79JCdHM(LCz{kjuq;cswF4-{U`He%9fvTPk!ZA?6Zf#Yk^ieUCz(duoOX$F z42CwXC3J-J#BpeA^Co+O?))`wJ27e7`OaPm&@{?Og_bY^>jQ~Kn@}~5Mob#*AlbGS zXd2}@g|-@KPf9cz0vbn?eV(5baO?(<1~{gG&I3rC=1S5oSd9Y_lQ!!lne{5rG|Ha} zEqf$xHA^(wJ*#mvV$x{;A+&FSrcwHjRG|p8yCs^j?s90vq|s=zE=NQH2786FNF9Ag zVeu`|ez|O56O%@xO}-p`fTmF%RcOUP>-!e_jduNNyr#sY(IN?LI?y!A*9z@8(5{nc zpI_ECd0Hu(^J)NTfTU5sg0Ji+_4vuHoiU5_~MylJUX^>@yyHJ zsE#;0P+IO&$Lujsafa(=YWvF6MyrK1dN8Sh%Mbdzi=@?UYk2_&Cw#Qx;M>-HP{_Bf z<(VJfwkBI*2y_~KG|Vy|EL_RyNVq&?>;6k;1%tSR%0ZpZ!RYLv+Nb~?(Yu->UmSB!K*N2raZg8 z3)mX7Oko!Td#S{xThv_jJmk53-Os7Z)I~x$Df8no^1Ns&?0y;88uOyUJ`U`Q65D)Y z9d5fc)#l%_L-@>6Y>pQ zk3yU2uP%Bk2d9rRhp)%k-_qv@U zmEGnEk1IU7)zkxb&Zl`O8hZj;W3E%!zT@yQOo?qq`XJ=xNJeegaCKxXu;27$0C^<8 z)j*L}@Sajs3`H+XMdlMHJG3Hq9l8>9ZAyinvrF9zY>hc`oOIA}VE1~T9b{trA}=dJ zckgqsPR*#1{6sWDVhvLTET5f(RnDUk%~;$4MP5}+P?wUIop2&6ln#m zC&(IQPry4>9*Q1zQ&jK5taIRQT${Qx!rM2Tf!hB^+m`^?Ra9x;_g?b2SqPPceQh8l z5TMiD>COUKx|1}}*+~aXg7A3V{kqekH@aWHP8bAC5S2|(Tt>LL}S(^v>xnd(Y&5Q_a^njzwIrP%`bnEn;uHt zI}gmDeppa^#uO z{WR%@&b4o-XKMxblJ5U9y64>>ee*}5dxYv|G9l7DyT{0Qbms!W-ygr=M#B z*?zf%<6er!!sm2)Y&5cTZ=x~$1F0z!k%DPT4yEq$Dd*B$t^}~@GLzm3+k)Uj zv4DDQK*JAA$^_)<@`V%-0=QoSYNLSuY69X{X?j5aeGVuY1{ymSm+_*}O__Wme@b)t zE(!pO*S*oCLaY>me;5KD1PykRo<8qU-8!3QQztkzSgU6Y4Kib2kF`6z6 ziBAA7Kev*W@vwAv@uH}IE6SMirRrIL0-o!p8UHI|Nbhw=q5CZs~ z1hndn^b!59qQ(1Z4k-Cx2a}lsryF+j!h%e8`3FchbgsO@SnFQWT`;#% zo&W1-aSrL;wcpV;^rFb(rh4?^`k|W(f&|q_K_Q4=OHf~>p#CWq)PM~tS&Byr3N0Ol z@@5JOK@44KQZ?%en%)0RG*v&bK_xq+k%BUo)XM}Pqo5GP33;hC6x7YJpw`p9F(3yu z&J06%{^G4n70uC0fcMGuAO-xVSioQAfNk$FYR(xeQX6FY3!TR06cmD3+hWqVDo5Km z{5ERN8oGBujV+Cd@ccBsg#t!^`z7FeDc}PpV15s(TZ(_;fQOm5L`^fVMRk+J&1+<} z!R23~fDpjhttN#Nt|X7cg-H*^3d9~TJLKjhCe2H{JXNTx+Bi+{;-TcCI_(6kd zaLl0L=xSOuB?=yq>sLw94~-%_kbj6H8D9cPFPEj7nr2@&XPn1ouzeBfhR!){#sIVO z^!?ZGwE?VdMfX~FncRh?O#s@`CII_U4_V_z3J3vQCjs3^0o`o^vfWsb14@oIkO&r+ zZp@4_Y&Z5<3IK}tO8^HcfZvz^_{gT=&24hcnENSG1Od^ag;`-ih=q+Y|1V)QX#+Xqf}ceK3`pRk_( zFp23IbG;3Zzr7JFxcx2@+b85Yp_LZl65n+z{oz#xH>BP}lNl)ljPonnNbb$4+3U>t z45?J=#N()~+MSwGLw#l@__)z};v`Kj(qx5@yHgj;XQ6NgnVNq3ESTlb)7ruUC8JgU zT~1##6SF(DNsCrIVDxFx9T1%i(al=)%C)Su;%lDhyCJ#|qStHD$_*^~^fyG5^JhOq z={&-J(Lzh8!y%y`7@<9>`$;O;llm!LQa7f;8)$ErNtrLc$2i}ZT0_JoMBHNtI$$Jq z#nee<5m#GOy`l>CRx^U){ zDF}Vq;S@UGG|gNzxrTHfakM3o?ZnGv>WCio*A zZv>BW1bH{y(MQlJHnhFT%#@4bv3?Y3%I&BW2Xj12aw1QdMPM13n?~QWb+S)V;&k%t zfDR|+J6b#~Vj-2gRH|tv>p1OHlAlM8r_YQSWfnrzCS5+2bLWh-WYg>kWPBI9IcE}o z;^Fjee}crBcDY+DHl!DaU^%?AZHgyKX3_o@oYh7hEvO}BAuFr2l9Wbcnt7D}l);?; zRh8t2GtIbE;})VdZ{3|D2@Z1pJZ+L%V#qtqTh~$uCcl?VJ&{WEf(VhJ=Q<*!L~yzn z`L`#M`(osc3E(^zQx-qpCTAiCzeFrKk&73t21DVZFAuGtt`f_24opO`tS5gGP4?HI z1JQ&Sj+L)Surwq-kMU$TlCRSTz zoRiSECc)CsM3=FN^jvnHt={WxannZHAXXtG(M}ZBZOTcykX~$*>U1M5n`}T+G{1Jr zy2&R=Icbi$apd@jC0%+JP3DSwNlXOZX;uHRRAb)2Rg23Sy^eB{YbidgX=^3bl?t6{ z+a}svq9Z@1q-gIBSDL6FXUXZQ8L9d#XTTLsI!$?+n?eoc4DxfO^TT}J1*y~K6tr-| z!y>H_5hu__AxP0>u>yzBCCW*(vjn$Z?B3yuHs5ZCxJg_mL}?EqrdvAf}UEr?}c`qisd@QK9*ryT#Yny!2Kdg`Omil;wHC6xO#d9B|;)n;+Xr% z95S}$kaik3lcrtHa7|tsu6Y}^4s0m;UY%gdz-v#Egl?I#RcH5`IH@#n@_8DkaO+b$ z)!F!lR3#Z1438E#1x+)TPvzpveS2+ip-$-@jkra-QKR#xaQcJ$PNe=aIE>nwRO&C( zLD3+Cib+8KQk9kS=>ML4S647^aG-x)XHRR_U|V5c>)B_|tF5e~b`$RN=%|s-o{q9& z^IAHK1IDGRvt?dKYwNt7ix%V-)Slbb*)zEF+>V|>)>c^!{pGb3`>S)E-F;n=PK5Rxr!Yh*4_bTs3z{MmwB`qlFyr3tzPxkl&KI{s zj8fa>VLHSCcZ0>wuFm2v-0tkQaxM7*8%%p=m%$XL&Yt$Je9_)^T+`X7eUnqMK(T41 zsMD`hXf5{k!^ICq>*y?Y=DRwtDfH*MItPlJtoFg4)?#OG54_~NI(qv%i`%MP`m{dvw| zs*VH2d{GqYSoIBB)3^2w>VOU-scw@(f=gNkJG+YKcJ?geG`9A)&aXlAST&h7pdm!D zvzuc!iM)1p3=~KwvlNfc9>&ZQ?lq9g7$Ax6;IOnbFiHEd)&-!ay|V-1@_-M+@66oP zV33}W%6^mnrn6f$z9So+x;34qiZUThvN$>}b6)CgT53N_rQed8L36a^>rwubUO&^n ztU7{ava7SdPS?&%*B)u*!o3W`m*T`4Cel?}FnpYEGD9Ipij`!h9!$5*Ot&4iZ^DGk z{nuCSnQ+)V4mFj1GMk<~Gkx|^`sNT9q{F11W1k{ZrKtfObO|DA zO8U%HrWBzjr)Q@!LlJ5cOYL%0niXc+vmxh+Np&h4Hk@t38zs8osp(lqa#p9Z?b$Rn z^MV#zw9q+Aj*#HlAW5EbBE4M1aT{~KAnJ_UER{I~SUZZBm~%V=Bj0l%>BTeXlF?;{ zGi$<@?1PeUy5mHwkA$K9EZ2UlII`1}W+^6GBWFiu3IEA1tuiI7J-a<~A!XmTns1nA!NgEW(UJP8s3q8bj`BxNiJlm>>XJXxhLfsR#4gO z(Er^T@_&T)hL49Y@qg(hM?N0zAv6>o`Dl0`JWs25G8`Dm_=Yy@q|(1XdmugM$VQalRjO`9AIbsp_D$Ju9|suxQ8@ou#^n(xKWTkV!L01Vn!<7#nA!E$&G? zYbzvGLwYGytVmKpKmb?7K?7h31j7rnN zw@)}Mo7sE^G3SIYR;~_t?31Cf&)qUlnt3ir?N4Vx<1pCFubI{(r3oaBBV1`Jes6BC=MiF%nU~r#=zQTQ)y(_|jYCs23scXA zGhcG@PeaYp)Qjnppd6>2W*!exFHxJB%HESTW03vXXAd0Mw{MNpfI~|&|4p)c!$-qs zPIr#~(;=~ZI!Hab_rQU#&2nxhZx^>;ked#Klsjh}Aewo5iVPr;Fr5((V+QZkRC>vb z?0`nAv*Bf1oqcfFXXYQt`#q0_ryB9u+)t#kkAx3h`f9kth)*PQ;&jhrxr3oW2>ng# zo@I-P5zF9Q&*L6QpTU-5ZpIFih~qtOg>acEsc$+-oQav4n%Zqj2K(O)*~5MI4zSax zxX`&WsR>z<#j%!|rPOnMOUH9J_1ce{7=|Q)(-)6_cF!K&`*WtHvpmq9ke zfVj?NC#N$-I@X2rjFw5H1&1hRj;Atl6o)>USR9a;IU^O$qrxb`veR*0l9TgN;ov?q z?2y5D`0Z?EHgiq-^i*b2xFvlAYnwt|a8MzI@zSHEaOqYv+z3}^X+WSOH8yy(mVF?+ zL!L#Vh)$rU>C%1#z%f;bD z<8Z>vv8nLIXN@7#XHXTP;n}kkr)M+QYzdFc>>}0gH95e(aL{0;Qfl9eS@URjP_AJy zJNPK2!4wSkg0luQ-w3D-P?FN}5V9K;m!)u&VCL3T%8V#U^IoR~=LcrKK{b=Lq_Qt& zKa`$(BpFTqKl6B@2M-*e7KQ%F^Cj!YnSq(7=?M9k9u4QcB8>_=;y}R6FUT?6dSD+l zR|j0!IO;F+0y#hGcpUDR*_|eD+zCAz?xVEpXoTG6r$b$R^Z+}%RP*gzuxF*tbZT&H zU*C)Lo*EC8APUXu?PfgC<$KX*5pRs~@M*Y5k~`4&R{D$`AJG zN3_^#`m0+FdQw^YAx(w!fNEOoP(MSd+rpQtfoh#?ctusO?9FP2Aa@tQf798%lwC5 z((1$g*$;)Y(zA5lWv-!s4`wryDl=Dx#mpIDh4g7`%jVq@G?n~t_JN^&`*c>K7FK2t z=yaQjfl=u!c;knu*|OR8i9Hh3tCY&gVdWkQD%{6Ue9l{YmWu=ITxaZz z$b2*Vpq`iE2l@{UUKZ}@2tPC^oeocLg!?+e2g7%g0($u5T9w8VrkPU6^c>6_9h&}$ z4P>&E@Ab2DD%_GSQEQ<+M%nOAn#QGWhAoD}#xjqkQ|UQJ9|&hYNdNZjC4G<4qw5Y1 zzLo#b#3#joOLHdM#(zunXM!dsW($NTa&Bfdp{IW)WtVnjb{0Q2tAqN&qojOrBaOqD zGT#blq?Vbye_#0cUTJFB9fv8w{Y7?1Z#ta>a9C2Nf`$Z>I3>*DNTkedsg!Br_w!s9 z4Gndk%^(q+W0W~LMZLyx@QLGyGBgI;O-~S7o>k(Ap$reeB!)AI$Du;p%u5WI#@Ruc zZwEB4nak=qJvb#O^D|2C#LUCjQ`p;PuKA$pws@$DBYiU8rxBu`+9S!a<`yYv)f5&_ zwohOdhwx-(5vvCc%;KmW9)iqB*RU|_zZWXv`gt&e({3`8Q!m{(Ve*yj$4E(2)JhL6DK+g#34{g8}h6CZ?x6sbs&Vci$qE`i4_a1=S> zq?O}L^YkJmO3$TavK=N-tl@{lY2cj4`6Ze2Q!klvbu_U!pX3-Bou}7x51Gn7dbi2? zV@L+4jnLp~j~VnbivvG0UkrR2=Wlp4+XtssWKO42%8{35et$Nc`5ZNYCJ`J@97~b; z3Kv}62yohP1V!d(en5fgIU+bw!V8gU92XIzae70H#xV*p8s`~Ala8|rV!|u#lnith z;iPW_z1Y_I@W{_=8}L+CD%>B=$j&o_nPu0{HyQ1?I{R%Ib#a=o(|;^rY>ywKvEjQ% zb9}H-dFI^Iv)|HP8ZzS`MQ1|Db8RW=wYg)13un`_`)muxalyXYo_40OX?Bdpj@U68 z8%xJ%>>3@Tv6p9z#+H)NOu}w}Q5Kt*#Z+M9uo#W)lA=0ZdsiZ#xrE}|VQwMAHQYqf zJdYXZaJz@aQuOq0+Lrz4vu5aGPp|S?2HS(&w)M6AR&s%P>$s@DkZN;&5Bpgrx%Wee1FcoOKV!+oU5KsZ|HLQBE7wA87vkW8|(TD^wy}c zL~oj|p*LH>%JmH`1HE13wxY8ba3Hzf_I45p+I#zl^8IZ~T6=p2iWRNf^Zj#YFSB8B z)F={-8`rcpSLcG>!6Kh6O7F+|ivcSLuH%3UU4`yKPcg_#7y-Sz5ei3_#Syb6UiNFb z%v0`AQVSjZ`L3KcUcRTeq@}mF>zs=AeAht174b5~1bHL*P(x>)Hf_ia5&hi(gge_{DCJTbIkXwe{1> zYC;`UQ%5VJaXZxMvPp)>ZEG$r6bm@Nj1teL3hbb-#xXiJc{AP7wXW{BW%5rYQMzvwInp+EY^Q-%y!Q~Yd2VH$~yc~kH z>JZBmrAQml=nB*@DX60a^!AYno0^OH)~mL1Q_D?%S8wZ8L9sVird?eY7;BrqqovRm zZJRbkzP4|$y}hozb7!HgapQXGfa-MEjW!ca6I@-zHm3A*Z)K{(m{Y$X-$!X{Yvh(~ z7z0$3_IMjC-AJU1GjY&WQfC3`nnaTA2x`Q_H)`w2MXEsMf4B(Okoz!cS>Q#BEs z6GffdLR%}PS#fQ@YZEosKiFCft|M5?(}1AgCKzEY3!1hxxk_m0TB>bMon=c$h#fa)R}WGS_m87O*ntXi%fxaamoiu0nWT;{M{RI> zA;-;lK%FhMP#DF)_lSAzcQmYnn`mjoyy1^FvNR--cEk-9^uilj@YEYbMq-iS&KS$z z7&HoQkEsMx+y}dgu@bV7Y`l}(>!?OfJD3se$wg^I0j>bKZPGjp*S<6^ zpeVG`=n-L>bS>ss43j$;7U=BJdKb3s>dAL^wnkg7Tu)(Vkrsrtb++$v%JdMZwJl%F z2W_-6jSb+HG&c<9xKl@0U6;#6R+3fEkKCHE8jaU&L(0SJT5cX`5a*5bHf`L#Ne?u2 zv=`ejn_3#VZ`w?wJ+p?+YFgO5zOlJ^-Nx#w-oDn}wn7sXIUX>2-FaucBkw7Wr7Pdi zT!X%RQ?xU;Jn{qAB_?0uy|pVV#(h4Q=&hzR+G72VSWJ=+0>1dN;KKEnRIEJv>}t?|BYR1e=aep9a7tx+3;cn`EJ zpz_~Q-0rG}=^Xq}sGZ~k_ZdbiG(Z z6ABMkp{{`cb_WB60`*2{N;uyZ=mA)FWA}~``Xn7Y?xJBmQ%GyLD~M9@#%x*8%y~FU zXUt7#R8I|;|3NaoIRMkx_@wz&en%X=c6*XE{ESBiSu{7Z9?z5b3eI4RYjZF}Znlv{WN*H)d4-#mx%azD@k3e&8G#giJ z+`*5Bwk)HR*S2?3iy_bOy7WmeNgc@so-Qpn6pueLiebd(g8Tr_iZwOq!9(LF(z2>= z1CPVxG22!TYR(hs=PDnoq<5&O-KcLUL{z;PxtHSvjrZi z7IMLL7}&5CbW+QiFFIARpufN$J~ZywVPike1}iG=-^J@*vZluEKxxfwYwOxrYIeio zGHI~moX(xx7Yr8ZLyKXe@5g7jmLVz=i&MOX=4aJj`ZDFJ&b}pjM$pgASYNp#-f{o^ z2HCcpi!EsOpUZe_zAxWOlWR2k(L>1#nzpQ6xnWgf^Kf$w+%wp5nO|;ozt4{`3-$2L zpAT(nS=&-g?+VQ9o;AGbYpQuBrfIFTlfef20Al6FWp3ZPxs{^J{ox;`12dyj&?!01(AL~HN}RQHX+K<{9GYr*EC?~8J3n6^WY zGVP$-waH#j%uRhQjjWUsUWy+!@a;XMBQ&Bj_PSFL_r;@5jfju@4K z05f6&d2oohP{m#*-vs7b^h_~5hBmmoks7tw7INLhj42pB3xizIui!?XZ{nKWR^VYr ztX#Er^$zfh88Zt!)^tW&`$Rq^AMQr&p4f}b5Kz${>=_&=wAI+SE*KbWp%k-?s59*; zQ0YZCJioiMr)kHI#>UM#{d#WGn$6z4Pniy`{HQzrOeu4b*F{D!F8fB;y9nI{88@yx zR+qS+&~yC52*ygZt>03H3Rj!5b2XmChJDM%uf)9eq0D4WS0g>!$~>+-Ikfb)?UE)^ z#_2Hq!a>Rt3bbi$OI0KPYotCo=f5-Mh|yS4e%eM{FHgK}%@{4Udr|(T-cgiF-y&*2pBo4|@Eq@4J zUB&%DcOO4nQVAnV^2402I{uT~qM^>NwpLn#gNfH^!Y`e?7xoQzIGM()iXUbCEuOI3v}6{mZR_WI|gb-dfO`NdZ^jZ zQzzEAUbeT{cNdXS0>>cM^XP};c<)3*7cNlO8PD#$vVNzF;(6KSuBeo2lcG<;cKL@y z$TwXgQlnKz47yE~DsJzLjg`$LbQe!TbC%Dq@dl{n%RsDgq4%FO$xTnc<#IX^-DLAJ zV(N5Th$O_aqs+)|p)^fpf_1SjlL59G=!S@=^rKT<{B|IBXe&;5ZmP_R6HC)8;qLsv zRk@-3fL@>DH7dyw$6_*pc~xH9+S|8_+6=$UdhanB@CJffF)v0Ln!Nv{s!7jw2lZVU9kH(1ta8H;N?-Y@ z0UBdrYQVfB8ZbLs&>A;d_C&$Rh!?7rd6H=F8XVZ} zo54L3$ChFFWmzw5q_b|6vl z=v}O3;gJPboMC4!<6}rEi~1cSb@OKB396>eS#2-Y?WS<3g3%NeCav)}>_5UR3pQS= zcbu`N7$xeXJZ;7sU@xUowW_N-^}4&{i{3gKS@aZ}np!rQS8t7trRa3CsmxpebjuMf zOCx8=vS1~>QHxJA%;yp6h*=TsI^8e^okrA_(HPZnaY7$`>QW5-xjha%8(7Pmb0%L3b=fL?l+m^fECN5i>7Q%ZW9!;ISZS#_*wMk<9|A5Wok6a*s||}bb2*x8sHP!2U1jp*ES15|e08!kV4yka9gC|4MM7hllN@i;113$p*Xc4HeX_6<(t7SF+Vs}yCF^>+ zgGvOi>WaG8RK@z@vYiJdBO0Wgr87a=$BKTvcVVPA;5qkxnjJ-DTe74g=Fac$g6kp+ zOReE_d%BoELC_1NqN_@IT}LguHBz0SuaLN!H1RCrN-l%E$RM6;{s@cePOPGrd7R({ zy8JYO88;IUH|92qV<>RuO}6>eYS`vET1ee|LZmG!7sXO%KYxg|5}#=BXKs;dM{VJu zzQwn*4Y8%NmJiHAl3aiVG+Ufu{1o9EFn-O^$Sc-ZEK;bdtA=@aFmGn{T1YaXegtak z9pvZm>$n+miqK}!n;P>OA*!F&u)uuvN1Zig2Y>n&^tSAv2c$B!VodK7Fa(c2rSVuR z5^Qx16&7Buj|X?SmMJ%`O|T_~V<#U)!CoVIw0*j~L2=zitij^e359xzUoDdDm>zJW z6uP$pk(cC@qrAe~ekQcVLGDk7UCLuqY(IvGl-x~!hjkI7f4LRu?#nKxZ>;O>Z}S&b zN8Vl8H~y5+NRj5&gBBR`_>ewd;*C|^6b#+b?yUKqwj8g-=kHngzAE6ibJk*JI7v-s zPhyOz$w(gP=8%_BCB9qYkzusmuU<^;zAF;Ld<#?9NxlZCobZDWufVtO-JMrEm!m(2 zwcp_en`QFB+N4aejaadM^S%pMRpGr&y|4 z3uazoPCUizihX1)p6STFm*@I1TK{R>xP4*}C z5?OUn_G9ugMRK%4GHTS3vm**hqZ?^XXtgtUobWGYd8E0L$ZJ~LQAK75A~#aF{PH$T z(}3aJnP@2?s!#1`S8E@A?}WZtPd|*w zD~b*+IY9X^;|{aLd%Il^XkOCO2ZDq2)w})zc7P|)WkhK5t6WnTpZWC{u8h=1+8)yP z5FgjZY{9KhGk-*DEis~P?X;_nhrhVQd|AYu6j`QcsCy&zX6xM7h`F}G?(SXIXbU9| z8aK95S>IyqhUcp9th zch?=JqSCJFJp)woy8|}A%?mSlr4^>M=9~5=;CC=GiuOLmv{>RFQZZ;zIB-yYfyhPtH!D!YPri&SNu$Or0XWzuh0I|#zj(eS^ z6R+#!BJ2x7j{^-W@vJ8w-r4tB{6f;&e8ln}AMSG(f${DzUxqg>q@kU8J4fTUF8ZDl ztwx!;Z>w;bmFm(vk8x88esbYz7#SApB_OuymI*&$cipR59}>Ols?O+4!YGZ5Zo5a3 z;in9Fp*+%jU1Wxdnhly@YTe$i_bxDNVBGL^#EgtDIvKIAx4B|VVZ^G7ZSCbwSC7c_ zHyYHZ^|jNsCw>7TRXl?l`O?I8d66bX$0*XNFYvZ#&^-yi0pq&wCPd70YGN`YRy)lX z7;iRwX052p z)G*yVEj3vd$VWH97PQl55|JNRtTjd3LYY+1!yQ=K_AE3NVJ9R_A5+q=X#|>321WZjW2J;R>Cul}&W$O((r!Axm_6%XG$4;OvA>H*ZE^O0TWHTP#XeJQ_f_ zm4|xJszq8Q7c*EM@wx9)~%(J|xv*R6C=+gRky zyGH57bn^vc=?C=%W|4oS!H<4g7%x4MnF$_L@QN%Ne$l*-(PBSJ+&Dn5igIlOy}9i) zJEZ3~y&-g@kK%>kkz8!X2H&*!hiGn+e`Mk>SI{Et679eljwvqNCFg^Gf z#!z~RlectZ#LiuhH*!4x*wbr6mJXv%@4Z5GR`$ZM)-OAIqU&p9=v&9x6HRh_G@M*j z?8jEWOz^U4Q^V``pfk2Mt%}+-TF1aEWxck{k8`3O<6fG3e?6-{T!Y4Ad~*m_uC=(6 zJ`x$y6X-mStzJNd!!%K2T)5Bk6PQV$U6m8{Ft%7=g}>vJ?r;{EDaz=3A&MV8kw-Ru z8@7Fx&&qWx9(Ud-*CyIGJN|VX*NN;C(zF7|ba!T=Visr6dl;cmE-eZkS%e8g((<*Wkq8s>o zLt2iST@e}NTU+_lYrC};yXI)BoVH--lxk{$7o6u0$deGZnUCi zh~CizgvlV9K+avbcEgH>wYgkV4jG^PdLi<%fm&d{xm$B+bIGd6YeD{ z70%)I?zB&5cC=25Hu(I#vQuEbg%X7=ZXjJR;>{*kDqb_e5AgiOqkhT43(g}M$#YvH zJaa``L^}YMnxvPv=#MGo#+LV*p+-kOV;iRQ>tfCmmU>wv-j4Vo zx=fK$v%u_V$L(39t~bz(L2^aGnBEWIyqboS8TFTl}`t#WixIv5Ip+a0HX zKWMp<0>-gN>^F%0a2(`ON0_aTSqAu6$0^`H&wyM>0sqBvN&N`>9viQr5ctE6Q@{^e zuB3o*K2y2)!|fB+-%tqrDaR?`&s(mffFHA5GX6;0U%fN}zvegvJYlA@5Aa0GCH3Qw zsekk00Q_agDd5xPM5uI7QouNis+@h`c82vg6at^=I0gK6%as%`4znt!A8xl;e?uYg z?T%Bx@3&k@0e`@9N&Dc$th+rK;14-Y0Y6~5k^8C2z<2T z6mWy(N(vY!!j;nxx0Tl4PzZd1;}r0K;Xt`wm!fnu#0p9621^i~ql@u^esVf(M zxb3n2hC<+*9H)TaVY!k5#(8(;^uz5=>u)FozRPh6_bF0soifN(vZf6PD8tw+>da(0B<8P*?-Qw5*tI0d|+=;8{DGZV|{hug*0-%tp=*>MW^$CfK8 zU>vYmPCwj!YW)p`z|S~N0iQVN;tzb1<&yOr$2Okg$pD|`I0bx>;$#O~kIQDUk zCj-3RaSHg!l8Zm^0m~)z<5>KQ&#_!mKTeiB%aZ{<$8ieyZp)Pv zFpi%rm%nh^YyAy{z#nm(0{&Oal@u_Ju`H(_ZjV@hLm}|z9jAaF+Ue{AjPota>4)3H z)*rRwf|s zE0!xMU>vhqPCwjUv;Kxci~j*}9a3q*ugJlf+z$c2YPn?jgxhPLjKy#A^a~!i4jhdm z1>;1`a`A`Tp!GKt0`GL30zO^N$>jJ0YkfH&=S@ds|UT+%*p z+w92zzrk?|_#rs~lj9GJQ!&fMA8rp@e^mYjf7)>h_>VWa_ygm(%5wVQ_P^HOPzd~g zj#I!_z0K(d#@UqR^kWP-sRI6nLg1;6Q^0+eD=FZsEtj-^zqk!}GQcIrDd5s9#G|Bu zhb))Wzf;_Hc{0G)I!*!KZ8>V61>a-2q<*;V^<;oQ;y4BTGt1F)tl(!Xm(&lppL;UE z&pA#3-#Oda2l$U_2bahYEK5Z)^Q4WkL5}V_(sbm z^~3Ego(%BY9H)TyS+1mjAGBOjKinSjWPm^EI0gJe%as)HQaLD=Qssib&InZaJA)B?X+bTv9&{5zc!uz-^9Gz#p+( zNdbS|=PdVnGQj**P<@CetS?g~o1pcMt6z~Odax>>I@P(F3 z+UFv1Yw~1(*E&uC|BK~H3K&O9my18#9m8?n+bma7z&IMaoPFTdZv72~z@3g$z)xC^>?8Pq z<%UAw7c7_T6X5oul^F_w|KvCY{HoTHg5wnMC6+5G;7cu+ z%oBti+=09#8NnIHDd4G=D=A={E?zGF2>T6Q9DuhuP625fyt$6KzXfKRYoQa{4}q!$O^&p1v2|JHIP1^he9CH2GY_nr*!3yxF3XP=3DDkK)(gaR8p{I0Zc4awP>^Yq_L;xYc6HGfEz4F@8AWmuv}6<+*W!r zz!x}90dKHeNda%PTv9)5_cPCizy}?tfKNEfwV%KzS}v&{VPEXU0eG|H6!2;9bozl$ zw_H*`+-7+)z;hg@fU7J=&m4lQEtk}fw4LOo5xBx}3iyqdD=Fa1Etk{}w<|mu;Jo7$ z@IP9vq=4UVxukxit;S0uaGm26@G{Gl6!3D(CG{ih4|s6^zQ=J2_({vrm{ITn%O&;0 z?T4NW@Q)m)fM?Ei2)}RR9ZMmd=*sk8QA@EAaDd4r1 zD=FZ0mP_hK*n7M<0N>;|1w8F+XCL4rESJ;|H|5Cy&vcvuzS43f1)R5BQa{{UJQ?7E z;}q}*El1-k!FO9OseirLZi8n-;7yKGz-{kx_5m(fE~y`3-{Zvr_`{AMwcFUC%@H;G*)DPP|=-CkXVaF-pHRrng1zu~pq<(~b zhZhIncR5Z0f6#JN{srG{xukx$-Q&psf7o#f_+KnXb1H&AZn>m>q-~v-M&ON(Q^4P~ zTuA{xVY#G!xP8x)0sevG6!6qaXMfOD= zvRqO>+}`TR0N?C51$WAAJPX>6s;}q~=wN5{9Xt|_*xJ~k8 zfTuc60e{VMB?bI-%O&;0?His9@V6YNfZtQ+;t%{@%O&-r?R(%|@JDmTf**360{)!k z=smRHM=Y1rkFek8#R2#Oj#I$rEkHa<3iy1>CH2E?i6;ZR+;Ix{Wy_Tm@c&sZsUL2C z_GExxahw9KUFhNuTxYqYe%NlmXG7q}9H)T4Zn=^I{)Xj}`Vsa5FAl(q9jAc5WVw<8 z{<7tg`r-B!PX_qwj#I$v7diU?Z?IfaKioEYGQiD_Q^5abxsn3@qUDnM;r1m@2KcLv zQ^2i@UHpODESJ;|w}K}FyxnmM_<75f6z~g{OX^47f72@iz~6S90{$<{(Hd^SKd@X< zKf->|iv#eV9H)REJkQw&_#w+B^~3EGo(%A(9H)R!Ip66AuCQEEKip3BWPoQmP62<# zawP@)Rm&yy!|iLH4DdG{r-0wQ#Kj+Yx8;)h;kL(<0lvv`3iw6Kl@#zxmP_h~+aEm{ z;Q!}11^kJnF8;s|TP~>|ZlCmIfIs6n1$^o^|w0Vc@46r+}ZaTuA}{ z+;U0%2>Wy|4#2Y=r-0vSxsn2Ym*tZB5%xhZ4#2-~oC1E{a@0Nue!+4{{RsOmFAl)( zb({juEO+(?o?yA8ez;BaWPn4*Dd7LI9JN1!e_*+!ez-m9$pAm)I0f8T@8S=9f#s6= z;dY@X1H9UC3iwx+D=FY#TP~>|ZolzlfPe2e1$>iT@H3W6>WACUJsIHV9H)R!T3H|P zkJqFCpKQ6Lez=|D$pD|`I0d}bawP@4&2mZoaJ$Tt0lva<3i#cYD=FajST3m_ZtwME zfZy*p1^k5NN(%UUmP_h~+xI;g;3plYfDd2g><>K6a!LJgJHnFzp5ZtJ++?|u0$y#o zq<*-q@nnG4J5B+=#d0MDe3Rvp`r-CgPX_pA$0^`1S&p7B1b^9bN&Rs9iYEj7b;l{- zS1d>0ZU}zWa!LJYJA%8AZzLl)<2VI8)pE4HMeyO4OX^41uX%B>xY35G(t;OTuB3p^ zvs_X?+|Ku8fR{N=0S{WPq<~A7OX`Q)kS7CtjpG#X&n-u5?gSsSTv9*Wp7msaf9W^{ zeDMXaSxEtJvRqO>+?qWZ;7c8+fN!&0Ndez(xukx$y~C3Mey8IU@Z*-F{#Wp~ESJ;| zw{Lqg!2jtu1-$S=XCL52mP_h~+hR`!c!}c_@POs0e-vD_Tv9*W20aWAC+JsIF99jAZ~Z*uklo@TkE zez+as$pFuAoC0pL9QBWaS6eQrA8uWAA`JQ?7xJ5B+=Vmay`1;1*!q<*-)=E+#R8hJ_IP6}RZIqJUzpJ%zG zez=|Q$p9~NoB|%S9Q9vV3 z3ivk5CG{ihfAZo0e81xq@KNiWeSnX)Tv9*Wj`3uGk9V8`?ywxK{}jC4a!LJg>-1!R zyBw#0|6sY20)F0dN&Rqp!IJ_0qvI6ts`bu3z>Stm>WA9}o(yo4;}q~OEmu;&zp`9X zKiq!p$pHV(aSHei8(jQ>w^%NzA8uPc8Q?cMP62<@awP@)xaE@i;r1<02KYZ5r+`n` z=;9B2qUDnM;dYWI16<)a1^jl)(H?e!Z?;@gKiqEdWPopXoC5xl0)Oe;MJB(>WAAJPX>6s;}r0}T8{Rn5d1mICH2GY5l;sA^Nv%%XKZru z2cB)Yq<*-~@nnGKI!*z<%W@?Je3#{t`r-C&PX_pXj#I$LHoN!(A7{Cwez+a)$pD|^ zI0byS<>=dX!S`4$sUL28JsIGSI8Fhdw%NrW_;kx9^}}tJCj&gkaSHgKEmu;&_gOBf zA8z-1GQbZwP61zViHkq*g_cX|hucM-4DcGqDd10AuB3oJW4WY$xP8`>0e-}B3V89Q zF8;vhSuUv`Zs&V4z{?z`fN!x}Ndez#xukx$-R8*v-{Cj~{A0_}-baFeV!5P#xc$_V z0e;4D3V6*MoPB`TS}v&{ZtFZ5;Ej$`z;|1Y>X+bqESJ;|x4oVW@JAe{fKS`v;tzbf z<&yg0Hp`O%p5r(Le23*q3iwXTCH2GYot_NvyB(*1kJ{?u4}7%clKSCxj3)zpyyFz` z`z=R%ybAt+<&yg0_CZeuc(3CW@ROFK_49%cST3m_Za?&7fPdsT1w6RT*$22}xukx$ z4S6!a*EmiA|J-sk|0(#O<&yg0_N*rZ{7c6v;EOMF@dw^yxukx$HG49^mpV=X-)Fg! z0>0mJN&Rs97f%LwpW_tp+BdrR1Fy4OQa{|*dosWmJ5B+A)N=Iwm*9IXm(&lpk9jh{ z_c=}hPruy7A9#l4lKSB`(~|){+Hnf_ddrm*@C}wr>WAB#JsIE|9jAbwupG_r3I3kt zlKSEHeNP7XNyjPR6<0X>0I#%MQa{{Qc{0EkI!*z9$8seF{9Vf>^~3E6PX_qE9H)Ta zm~-(5zT9$2{cyX&lL5{0~ez@)Q zWPty{aSHhOyweYSg5{F>;dY`Y1AL0(6!1S=j>eyY@3UM|Kiuy3WPl%VoC03c;^Ggy z)^bVxa9iie0B>}h0)E7DB?Y|Sa!LJgd(@Kw{(|Ea@Z8q=fPegcANXv`CH2GY98U&# zp5qj7yX8s>xWjTu{czjv$pByFI0gJU%hCE_!H-xjsUL3pJsIG~9H)S1w>kR&&#_!m zKitmrWPs0hoC1E6c@B0L=vs_aD?RS~@yhE=1jXdx>9jAc5YdQM9M(`7s zOX^41>%BMtU+g#qoVQ#_0k>E#ssE)3To?84g&?>v3<7uDDWCqdVR8`AwXL(gJy#62 z(H);MM(}2WbnsQ7qd(u3(?_J_=>WMemSr!?2Xu@S9r;ZD+JW_Pr&sGcIh%Yp-3Sa@ z0y>Y3PsYl31;3&jKJ#u7dHDUrTcEJrF;M5r zZFj!^szQIzL1OHxMeidw??kn;+Ud5~-`UdL6&BLtulR{yBUn~TZt`F=Pu!{(HT+`VX{FOlS`o`v!8!ri#-(Ya_x{bm0gzYg- z!AIy0C&L6^i8wvyJ01RJ-&YmgSzi}NoQl5F$0JVv={x-?;&kk8PvLU9aWQQ1o!%aC zdcb%3kBHOrzSEpNzQWaXXCr@g#OW^I=`#_hANo$OMVwB%(Now^uD`;xzTGp2C%M zEu?b^Zcvstk>5&<@z4E{mSxjNDDL%{9IyXk5PfRwMUw zxxOIRKg%^PkE5N?HQ|aQnQ+AsO1R=UBpgkIBZP3pkw3WN_#Pb9gX4B^v<{BI!S(PC zla6U}#ql#ZY6eHb;EE$$aK&*fIC^E0l&8sZ#gQhsJ|jG?I5Gs+6+29Pa4pI`uDj(P zR~&hPD~_nZ6-P4QiX#+o#gPZN;)nuVaijpQDF4{@ALScc@?-0KT(Q+XuGpF$S8Own zE#aJDH_AP(*v1vtr^FrG zuVR~2TqoXa^kRF{rE-t$NU=pIuGsDqTYF-A&Wmm_da#}5m2!{mFYlFmY)OeLwwJ8A z&G=s_S8V-=?HhZ9$Ciq?VhcoEv7I5dF8uiIMlY_|E)ZJ-VtYSq<%cczaK(0bxMF)b zTo1p)=*RU0xj#{^cgq#qiedXNT(MmjuGn4+*U5Jp{kUTLCtR_e60X?(2v=-Zge$iH z!S(ca8vVFpdm3D^T?($)`UKl}V7m%j2i|4$;fn1baD9ik) zp59`ivSCHjxjZyCcO5;0l`VsvU2W&marBjZw>>|wJ*cG9vq_)1(nFYbDHZw$=y@n| zlOs1eTRP7k1od?lgGzoH2`Y<)o%FYzZpf{-EnmzBm4)rO_Wpc#A-BDa{Nj$3wzkTt z!1UzM(p3m5TZ_H@0~7^w+0jbkIMlncqn8xZpGqB?(cCgX(yhJSdJIB(bGjs?c)pDP{si)p zqJL@ex+q<*(IO#E6do6r^?#?+&0+Kp{x_%V=`IL13sIr(+@A>!YE(^segDODz2=MJ zf0n*;f8zdEk<*3iMFbY&6vugcbImn!x^zyFIFEOr}QK{Bs-@t+&kv$tTb1ay(UbMZ$w zxL!r>uX_G>ivOMB{|52T(cyR^{z3=W5MY}^-^=3vWf8nm1mpHY`frT+-!iFQ``9wc zc!V9<|5W;i@ZtY<;@Bo)uHek6^>i1|RbK+gnL86#V7|Vc{`vmv#eco{e?De#PAmVE zxdeC7)%RaDtzNrSO)~;-qsKD;2Y=w+Prxp&uVr_a6HNQfG9Lw^xIhz}89|FHkl z;{R#!|6C&^&9x$W8{9|tzW;Cjs9sBa^G8Oo4F)&2iSR#5yzjs2V7->CI%quHILln| zH>NK5B02c}KOHt`(NBlW2EnKFU-!qb?*4lOnUiW3Jb!pzJ3fE7k&NemJp89({=;d; ze>lzfkLOq1|I;!5$BO@B#eZBre*Y}_N8KVAREYly@&82=4O_n9AMyHqEc~aBF!6u- z2;&j*tebzh$Mq%RSQ7U8rTG6P{7d-kmw)hw>tBiU{XehrTJCveJRYg<+#mP@lO{3d z5PrpU?yZOYci~P!}$N!90}=#k^P@Pod07@i_pNGqxTc7mT4Ns|C9>j|AANs9`}EV z_>bHFUzZsFE4SF7G=P6xuN41LsUZ0CUB>@cTca-U2llFA{D0wPH^vHV-2Zfn2+A(v#Mg;g!@fP{A$F_Fo(akK9RznwH~#OKV|c#&kp3Y$XxF%# zEEj&oV#A)a!uZGEc=+JUT{9HBMd|v4Q-*q*&1LyaP%O(&2;NtgpGf1R`XK(xP-};A zknfhE#sK4+C?5tGw?z3#0mdUyesX|uMwFjo#}83{YJhP;ls`N`e;?(i1?b14{1E~A z=P0iN^t(}hdVu~k%FhVUPe%Efw*QOrM+WHEqWn<-`l~2^wC#tY{4oLgmneU1fPN#& z9~YoMi1Nn=Xy>CmKTRZmnIP5(iAW|mDJe`$COCN*{*+<(iedOuhv82fhCh86J~jv< zDXxd<;PBv$GG2s7ZciozFT93(T-gGtAep~&HJ{AiMVe3MFAoPf+#6!yUZ&-b4_=o3 z7wyCj;e#1@LazPUNtpFKAJa2j{BIzsW#(kQkKYKNwjQ_`p z*YTHeF66&Npq$;lCGwa;|AjP`ejVFWgvXam_Zm9bFFbZIeT(p~2#+t5dWAoa#)YgOJBC8fkns3Y2l0GJcju1u_NXe#BR%}LEv!lB?|2PCgHKeCFCCy9$%I~{x8C>o?ovu zLw@-bqaR<2K>kg_hqY1p$Arh1ACP}pcU!HJig3; zeO?wGJ2E2viw-yX@udagQxYCKETSIWCp>m8M7ereczjWTay4_B(T|-7A-_&|d?5k( z-NIvML9`E#2#*~Bk={QG|C39NoqNP?b@Uv^`Hmg_5N@yV__6`v-YYzI)PsGV6CPhG zz&@ubqaQouA)Z@>$If<$=lg}n7YT^x4}`}~bqM#k=|(@kFhIC(5FR_vA>7-A$BuE( z^IhTbB?0tIoniDpywunYdR7XLFPo9?gTiCSHON08{3ll%J&^y2@YvxD@<-8-g7XDm z9zgy=;jv@c6Oxa&3Xd-dQ2xIoJa!<1{_K%PKXwj-{VxZ44!*pUl-oAB8c#y-$>=Xrl zQ21>t>jVCQzfbu0#P}Zx|GgML?HCg-c0z)l2H{t)is~;4e^ZRVPxvp#_#X-XYK)(D ztO*x8=-6J5(R1v{MnAsrM*Fit zczl`NEb_aB#}^xr|E%!%(gOX|^TOi`3Gj1IG2!CN2JmgdY&VW5sm6#HaE!^IREx+%Ob9{aY%0#RK(* zpnA){E5LuPUf+K)tuMi~!aw~D!w1t0{{iqn5PpK;KSjJQp9RBHv#)>8Y5C;*%wLHQ z=z07Phv$u;{5|~i@_YsHsUSHobQbXyL2{m_miW`CzF;ZU<)VL`@K~~Rw(u>&V<}ga z@YfOlVR}yg(0|u!<~-s5RphZ0XS(peB%X!}iQivE9!q(s`O?2LW|??m$<7_ZuO+^m zo-2pp2Sg8+4k2A1)bjMad{@2JoEH5L45R1Eq6bUD5YGd`W2wJTY@oaxAeOw{R`h>@lF2w&X;jt76_L+31(Ss#6*NdKWgvZi0*k`NoSkePM zUlu-IK7T{}3~F5B^VKhC-nW3kQ_d=HpHqh6Ylh)3CZ6qsrJj#UP&WyWC1$YCeZphu z-e!^ip72=0H&^)I3y-CD(4$D8T>j1_p5ue1w2-M0KHhFzCOnox!OmY39!pA5?|wr( z*RP5rP5q+ARR5+?LouD|(HGxSZ-~%b&LN)TgQbn@3>{n~JeH&(+=qq7(#@Ae{y&As zl3Oy1{yif+mTFSxp??$3G2vp#9`yGR&+-4_F(&@>n5lp77kMm6T`v5?!ei+m-Rj?W zw4Nh^D}Pk4@6peO=Zbw~eiimPn|QVl-kVG?YJ&5$JoO8rhpM^$6^LiM#q<5G!eeRS zn??T9!efc-2I0RYe7syeM?A`gi0aGB!|+opd96sYe>{$Oj%U1_pHDout4}{wul3Sn zh5l_8c`U`9B;}`Hcr58|7x~*Ezo%X^jTz(cu<%$Si+b^0;p64)x6pHk(X-5`367j+ z^6NuV9x8-iPCTazOH}7dJg*jh)on%(%Hb`my2gZyC9TkZmGD?v&hz{HyH$8B(Whdme;*YdOIN=y z{1d`sN%{dPC(nbwrCv*7JbA?Y^5y?@;-{yN3tpD>AEG{}B3`$9GxLU$^fqhxBLfwC zA9N-0oL^Y-PmfFbcdhVP`cK)UfA1rn%gOUHzIwax_X)pqR=sA>KI|u+^ZjWlZ|8~p zOT=?Iyi#~2{IpsVpLjpDQg|#CM!tMTcr2L(e^Q<3xzof4{M(5CFtrZ>*1br$pCq2s z)i33LqBML@h#oAVhMrXmjQl;4-c_RKQsUX3Sdxo!J19Js21Cz}g~yU^)YEA+;5?Pu z{~IKp6{7z%;%5l7PE>Ry$Tx|5m$Ir(IAMa1k79LB#5$+wtv)w)< z`nN~`UlVzJIx`^r{}9jhrC!G2D2InpK`j^m(}`z$#`A0QF!I}n;cpW?_;hER#OGtf z@Q(?PPhB1o`R@~7E}p*>`GdFBYf1Dw;rZp|Pb8kx6(1+p3y)7+Zju1^rp@xiAW#|Zz#FnXR9d3<_>e)%=w@k!DBqURVI2(kY7^FpQY z_+;i2lHM)gchqYQC~t$pD_18G&vu)2L%o)qC;i|C;qj>i()*cV z^n8nWZvTV-HhPYgaGw=D_+$)vGBnWDwM=+?8gr}2=ZG)Y z{w@9N5Rboch;==<~jV$j-O94HR9w^z2($LlLMZ`hPu+tj={m!rE? zk-OC^J*T$bTt`=L3vD)_zevgD2Y1ppz1@9Xg<_$tazS0~;(#4;oozemuC6*TpSiRR zc6aYWIF0L9>0mC{)Ud8GhdT}i*oC4$f6<{73j@W<*1o}-jvsjH-v`8ub|!r6lUDmI5Omy8MskQ4&Ot)_0= z6^1KvGew7jg~LNvBMgm9(d;D*QAX|(={ zdmAscexo%?E8_X`@>Od0sA*1w%aK2XG)vYVYVKM)jVrM(&_m5*-3+jrLrAh@GbMS< zp~tMPLq9}=yU_oZa;9T|UR5`WBaQB&Ma{GRSbND>4WrQM|3ff+BI?qMnLzbZm7L)RV?;*8Y686}#gEQU>^bjm086vIJ zj6>qHMz+M?;_9BaR!%Y1IC;g4b|hGrfaAk|6E}8xSNcIy7ABM zRgXV&YsR0s^T*vPk&(yUz%>^7`%!4RT`%p()h?% zfPwueBPH&U?fi}(n~~;usNf&^T)k-|V+KbfG}?K<1)q`Yn1iti5)zHHLm(W zGwS4XTwc5wHI4P*Eau0ba~dW856G(ihf(6tvjADBhP^2 zNTvV$F~;iAKvjy0x~`LdLph+3$DA)*xX0UPawN_3w3n+R<~~(=^5SK{SUqxsVH>%z zdgA1dGfjAxHDcYXt7usds=(j3hQYnQBRXEi&ppU>dO!7xw^H#mAW#$^(iU6 zaP5W_4Qq3`rW}?>)bc3_)m3c;UUSl47#L`bypptDHe7H)bK~aR=7tq(8+kky<#aes;A+Eua5T%SUlE zmMpor*h~I-eF+yd^0Apc$oh`jK3Z~9$A%l=1uIp#I(-_1!3(#OV|Tu%k*5u6y9zx6 ztaMFtv3h~cGdb9z$*h6F;+4ia6Tec>M*$bx>+3IE+q7b3uBNi4vd+e=dVb{e41a{y zxT?lW!a_MyqiY}{)3}KuOsVNEbPp7Y(J&#rM-^PKJzKtIV8z!&N zyJ_S0O*`01YxJrvDbcMZ9!$7OCG^UgCN_&1((uwLR~I2sos{sFz45}c)ur4fs?7tm zAN0zu{J;S1UYy%TJ0eTz-LfS&zp|E!N}))nwnU3``#_;^Rg^;hBK(@h4OB}wL4JBx z>S&n-TGUP>>RxU~#2e$8-cuY)wr|i4ht78zHToIP4T|Q=MU47v8POnKk{Kh_Tvrpk zL~Lr%Q@XEfK^iGiD|IWT8rKDBTEk>@J4lRi}O9rDkqN-iKp>|6QT|a3H3l~-y?~KFtojV(KmC+?^5l(2L z?uOFeLJMQcWwIT>NI2R&bP&YbrUuQSbGD2#m zUf1HT+s*1~I$DaF`P{&+o-)lMx{cUunQM}RDy%oRL@QM$5{^5sWbu_Ro6ur(*Tc)nH zt9QU3XGZFMwW;^*{e?obnWB*>4Md_`E>~udizdl*RM9wZzEome?Va%C>WL|$HBv;| zy9Nigd$AYA3-f){#J1(Q9$G``vRc-pRs``PTXK;|fgJjUY-hrmIEme*Dk9+EzpB4HUp;iw7%!tN!Z0*v9*(}5N z>S63T+EdOp$`E?RK~4D1d`oAkn%*+yT6rjw<366AcBmDLba|1QF9WDp<^TJ(&|YQc zo+M)Du~gPNHaw5ZycTQ6p|)2vgJA|gu|bjLIeR6ua9Ci|1zl!7i?WQS+M+R$@zU#R z!I8S%NdK?vkp9!Sar?$@Y9eXsj#FfM1AfFY!y@07G8%ZjQecc3d%UUZ%lD%NiwyRA z3Xw)(q?9k9wvivVy(fl9CGx{CqoYKIYGY6>Zq3IBKDoAm-rRPY521rsY?<}?iMH4v z!h9$rAFYVJJ@FPdATfF2xnVrwKl}1CzRNdm1O9()*RtFu5Cr)o$FZd3?D8R{3+2Zp z7fJ%zSy{FmJ+S@v>6v{nv%8XTN?f4|mU;Ac_dwzD=2Q62H$z@#Sq{a#0{yVk|-pZ zGWWzL)E)`$G;1QF2rQp`Y^fKtCa4b> z3-$+3faHoYilA4`2YsgNCLB&4d@#)fRB!f?Wm8r7C05q=bN-=>l=A{*R1((FD<&99 z=KQyCm?}_nI1E>-tvN;GRGZK4yw{|5hzzDopaHDdqDa)wRsH7~A2=G}*uwGpjRc;y znqhRfE}~5|cF3&l!VU|}1@|*Ej2Tp`d2-^p{Wn;GIn(SiIliCcroDeM%g< zt$y zN6xejAt#3-(ifc}A&Zeb18jJ9Qeb6=kL%)ZgfE8@7usaRgQ;KiMyG!Vz(sn&zhCr$ z6bta80oOQ*(kyy;X4I_8_AXYP2?IXYR%FAOR--Mv>gnU7dc%J&*ZAz->^`gZB8%n$ zTjU!n7G)k@-wnK<+r@VOLbZ2&jTy@YLJ}rnc_(B7D`SOy1j=^lSg&0x4}x=jK}JxQ zoopM>EC$O05z?pmnyLYhDbWnL670eoBJfjFV$y9_bF +#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