From ee40ad3d5c32100004f75372a0861efdaa285f6c Mon Sep 17 00:00:00 2001 From: DJRobX Date: Wed, 7 Nov 2007 01:27:54 +0000 Subject: [PATCH] Upgrade to 1.8 beta core git-svn-id: https://svn.code.sf.net/p/vbam/code/trunk@23 a31d4220-a93d-0410-bf67-fe4944624d44 --- VBA.vcproj | 202 +- src/2xSaI.cpp | 5 + src/AutoBuild.h | 31 + src/Cheats.cpp | 5665 +++++------ src/Cheats.h | 110 +- src/EEprom.cpp | 8 +- src/EEprom.h | 7 +- src/Flash.cpp | 522 +- src/Flash.h | 70 +- src/GBA.cpp | 1351 ++- src/GBA.h | 305 +- src/GBAinline.h | 915 +- src/Gfx.cpp | 2 +- src/Gfx.h | 3235 +++--- src/Globals.cpp | 9 +- src/Globals.h | 2 +- src/NLS.h | 126 +- src/RTC.cpp | 442 +- src/Sound.cpp | 32 +- src/Sound.h | 2 +- src/Sram.cpp | 78 +- src/Sram.h | 54 +- src/System.h | 7 + src/Util.cpp | 2217 +++-- src/Util.h | 130 +- src/arm-new.h | 2433 ++++- src/armdis.cpp | 6 +- src/bios.cpp | 2321 ++--- src/bios.h | 3 +- src/elf.cpp | 5993 ++++++------ src/gb/GB.cpp | 8659 +++++++++++------ src/gb/gb.h | 6 +- src/gb/gbCheats.cpp | 120 +- src/gb/gbCheats.h | 1 + src/gb/gbCodes.h | 151 +- src/gb/gbCodesCB.h | 11 +- src/gb/gbGfx.cpp | 217 +- src/gb/gbGlobals.cpp | 5 +- src/gb/gbGlobals.h | 25 +- src/gb/gbMemory.cpp | 802 +- src/gb/gbMemory.h | 63 +- src/gb/gbSGB.cpp | 1834 ++-- src/gb/gbSound.cpp | 1557 ++- src/gb/gbSound.h | 120 +- src/getopt.cpp | 1048 -- src/getopt.h | 5 +- src/getopt1.cpp | 180 - src/interframe.cpp | 1229 ++- src/interp.h | 5 - src/{memgzio.cpp => memgzio.c} | 63 +- src/memgzio.h | 7 +- src/remote.cpp | 33 +- src/thumb.h | 908 +- src/unzip.cpp | 2 +- src/win32/AboutDialog.cpp | 15 +- src/win32/AboutDialog.h | 3 +- src/win32/AccelEditor.cpp | 8 +- src/win32/AcceleratorManager.cpp | 10 +- src/win32/AcceleratorManager.h | 1 + src/win32/BitmapControl.h | 2 +- src/win32/BugReport.cpp | 5 +- src/win32/CmdAccelOb.cpp | 1053 +- src/win32/ColorButton.h | 2 +- src/win32/ColorControl.h | 2 +- src/win32/Commands.cpp | 511 +- src/win32/Direct3D.cpp | 1696 ++-- src/win32/DirectDraw.cpp | 197 +- src/win32/DirectInput.cpp | 1890 ++-- src/win32/DirectSound.cpp | 763 +- src/win32/Directories.cpp | 97 +- src/win32/Directories.h | 19 +- src/win32/Disassemble.cpp | 14 +- src/win32/Display.h | 12 +- src/win32/FileDlg.cpp | 23 +- src/win32/GBACheats.cpp | 14 +- src/win32/GBACheats.h | 2 +- src/win32/GBCheatsDlg.cpp | 16 +- src/win32/GBCheatsDlg.h | 2 +- src/win32/GBColorDlg.cpp | 57 +- src/win32/GBColorDlg.h | 2 +- src/win32/GBDisassemble.cpp | 8 +- src/win32/GBDisassemble.h | 2 +- src/win32/GBMapView.cpp | 4 +- src/win32/GBMapView.h | 2 +- src/win32/GBPaletteView.cpp | 4 +- src/win32/GBTileView.cpp | 4 +- src/win32/GDIDisplay.cpp | 881 +- src/win32/IOViewer.cpp | 441 +- src/win32/IOViewer.h | 3 +- src/win32/IOViewerRegs.h | 6 +- src/win32/Input.h | 11 +- src/win32/Joypad.cpp | 232 +- src/win32/Joypad.h | 342 +- src/win32/Logging.cpp | 6 +- src/win32/MainWnd.cpp | 2451 ++--- src/win32/MainWnd.h | 888 +- src/win32/MainWndFile.cpp | 1905 ++-- src/win32/MainWndHelp.cpp | 6 + src/win32/MainWndOptions.cpp | 3708 +++---- src/win32/MainWndTools.cpp | 6 +- src/win32/MapView.cpp | 70 +- src/win32/MapView.h | 2 +- src/win32/MaxScale.cpp | 4 - src/win32/MemoryViewer.cpp | 4 +- src/win32/MemoryViewer.h | 2 +- src/win32/MemoryViewerAddressSize.h | 2 +- src/win32/MemoryViewerDlg.cpp | 152 +- src/win32/ModeConfirm.cpp | 2 +- src/win32/ModeConfirm.h | 4 +- src/win32/OamView.cpp | 53 +- src/win32/OpenGL.cpp | 1221 +-- src/win32/PaletteView.cpp | 79 +- src/win32/Reg.cpp | 743 +- src/win32/Reg.h | 74 +- src/win32/ResizeDlg.cpp | 1122 +-- src/win32/RomInfo.cpp | 54 +- src/win32/TileView.cpp | 59 +- src/win32/VBA.cpp | 4831 ++++----- src/win32/VBA.h | 559 +- {res => src/win32}/VBA.rc | 584 +- src/win32/VideoMode.cpp | 760 +- src/win32/ZoomControl.h | 2 +- src/win32/dependencies/CVS/Entries | 5 + src/win32/dependencies/CVS/Repository | 1 + src/win32/dependencies/CVS/Root | 1 + src/win32/dependencies/cximage/CVS/Entries | 55 + src/win32/dependencies/cximage/CVS/Repository | 1 + src/win32/dependencies/cximage/CVS/Root | 1 + src/win32/dependencies/cximage/cximage.vcproj | 367 + src/win32/dependencies/cximage/license.txt | 48 + src/win32/dependencies/cximage/tif_xfile.cpp | 208 + src/win32/dependencies/cximage/xfile.h | 76 + src/win32/dependencies/cximage/ximabmp.cpp | 361 + src/win32/dependencies/cximage/ximabmp.h | 79 + src/win32/dependencies/cximage/ximacfg.h | 52 + src/win32/dependencies/cximage/ximadef.h | 197 + src/win32/dependencies/cximage/ximadsp.cpp | 2370 +++++ src/win32/dependencies/cximage/ximaenc.cpp | 920 ++ src/win32/dependencies/cximage/ximaexif.cpp | 873 ++ src/win32/dependencies/cximage/ximage.cpp | 498 + src/win32/dependencies/cximage/ximage.h | 648 ++ src/win32/dependencies/cximage/ximagif.cpp | 1564 +++ src/win32/dependencies/cximage/ximagif.h | 248 + src/win32/dependencies/cximage/ximahist.cpp | 649 ++ src/win32/dependencies/cximage/ximaico.cpp | 299 + src/win32/dependencies/cximage/ximaico.h | 52 + src/win32/dependencies/cximage/ximainfo.cpp | 388 + src/win32/dependencies/cximage/ximaint.cpp | 1030 ++ src/win32/dependencies/cximage/ximaiter.h | 252 + src/win32/dependencies/cximage/ximaj2k.cpp | 250 + src/win32/dependencies/cximage/ximaj2k.h | 46 + src/win32/dependencies/cximage/ximajas.cpp | 265 + src/win32/dependencies/cximage/ximajas.h | 84 + src/win32/dependencies/cximage/ximajbg.cpp | 162 + src/win32/dependencies/cximage/ximajbg.h | 44 + src/win32/dependencies/cximage/ximajpg.cpp | 454 + src/win32/dependencies/cximage/ximajpg.h | 315 + src/win32/dependencies/cximage/ximalpha.cpp | 321 + src/win32/dependencies/cximage/ximalyr.cpp | 105 + src/win32/dependencies/cximage/ximamng.cpp | 366 + src/win32/dependencies/cximage/ximamng.h | 86 + src/win32/dependencies/cximage/ximapal.cpp | 707 ++ src/win32/dependencies/cximage/ximapcx.cpp | 445 + src/win32/dependencies/cximage/ximapcx.h | 63 + src/win32/dependencies/cximage/ximapng.cpp | 507 + src/win32/dependencies/cximage/ximapng.h | 73 + src/win32/dependencies/cximage/ximasel.cpp | 469 + src/win32/dependencies/cximage/ximatga.cpp | 300 + src/win32/dependencies/cximage/ximatga.h | 60 + src/win32/dependencies/cximage/ximath.cpp | 97 + src/win32/dependencies/cximage/ximath.h | 39 + src/win32/dependencies/cximage/ximatif.cpp | 763 ++ src/win32/dependencies/cximage/ximatif.h | 60 + src/win32/dependencies/cximage/ximatran.cpp | 2439 +++++ src/win32/dependencies/cximage/ximawbmp.cpp | 85 + src/win32/dependencies/cximage/ximawbmp.h | 44 + src/win32/dependencies/cximage/ximawmf.cpp | 458 + src/win32/dependencies/cximage/ximawmf.h | 154 + src/win32/dependencies/cximage/ximawnd.cpp | 1110 +++ src/win32/dependencies/cximage/xiofile.h | 112 + src/win32/dependencies/cximage/xmemfile.cpp | 173 + src/win32/dependencies/cximage/xmemfile.h | 39 + src/win32/dependencies/info.txt | 1 + src/win32/dependencies/libpng/.cvsignore | 3 + src/win32/dependencies/libpng/CVS/Entries | 20 + src/win32/dependencies/libpng/CVS/Repository | 1 + src/win32/dependencies/libpng/CVS/Root | 1 + src/win32/dependencies/libpng/libpng.vcproj | 238 + src/win32/dependencies/libpng/png.c | 847 ++ src/win32/dependencies/libpng/png.h | 3450 +++++++ src/win32/dependencies/libpng/pngconf.h | 1472 +++ src/win32/dependencies/libpng/pngerror.c | 313 + src/win32/dependencies/libpng/pngget.c | 937 ++ src/win32/dependencies/libpng/pngmem.c | 598 ++ src/win32/dependencies/libpng/pngpread.c | 1573 +++ src/win32/dependencies/libpng/pngread.c | 1459 +++ src/win32/dependencies/libpng/pngrio.c | 164 + src/win32/dependencies/libpng/pngrtran.c | 4219 ++++++++ src/win32/dependencies/libpng/pngrutil.c | 3124 ++++++ src/win32/dependencies/libpng/pngset.c | 1265 +++ src/win32/dependencies/libpng/pngtrans.c | 652 ++ src/win32/dependencies/libpng/pngwio.c | 228 + src/win32/dependencies/libpng/pngwrite.c | 1513 +++ src/win32/dependencies/libpng/pngwtran.c | 572 ++ src/win32/dependencies/libpng/pngwutil.c | 2745 ++++++ src/win32/dependencies/sdl/CVS/Entries | 44 + src/win32/dependencies/sdl/CVS/Repository | 1 + src/win32/dependencies/sdl/CVS/Root | 1 + src/win32/dependencies/sdl/SDL.h | 94 + .../dependencies/sdl/SDL_Debug/CVS/Entries | 3 + .../dependencies/sdl/SDL_Debug/CVS/Repository | 1 + src/win32/dependencies/sdl/SDL_Debug/CVS/Root | 1 + src/win32/dependencies/sdl/SDL_Debug/SDL.lib | Bin 0 -> 1787026 bytes .../dependencies/sdl/SDL_Debug/SDLmain.lib | Bin 0 -> 19288 bytes .../dependencies/sdl/SDL_Release/CVS/Entries | 3 + .../sdl/SDL_Release/CVS/Repository | 1 + .../dependencies/sdl/SDL_Release/CVS/Root | 1 + .../dependencies/sdl/SDL_Release/SDL.lib | Bin 0 -> 664824 bytes .../dependencies/sdl/SDL_Release/SDLmain.lib | Bin 0 -> 4972 bytes src/win32/dependencies/sdl/SDL_active.h | 58 + src/win32/dependencies/sdl/SDL_audio.h | 253 + src/win32/dependencies/sdl/SDL_byteorder.h | 24 + src/win32/dependencies/sdl/SDL_cdrom.h | 171 + src/win32/dependencies/sdl/SDL_config.h | 45 + .../dependencies/sdl/SDL_config.h.default | 45 + src/win32/dependencies/sdl/SDL_config.h.in | 305 + src/win32/dependencies/sdl/SDL_config_amiga.h | 80 + .../dependencies/sdl/SDL_config_dreamcast.h | 106 + src/win32/dependencies/sdl/SDL_config_macos.h | 112 + .../dependencies/sdl/SDL_config_macosx.h | 132 + .../dependencies/sdl/SDL_config_minimal.h | 62 + src/win32/dependencies/sdl/SDL_config_os2.h | 141 + src/win32/dependencies/sdl/SDL_config_win32.h | 171 + src/win32/dependencies/sdl/SDL_copying.h | 22 + src/win32/dependencies/sdl/SDL_cpuinfo.h | 75 + src/win32/dependencies/sdl/SDL_endian.h | 192 + src/win32/dependencies/sdl/SDL_error.h | 61 + src/win32/dependencies/sdl/SDL_events.h | 337 + src/win32/dependencies/sdl/SDL_getenv.h | 24 + src/win32/dependencies/sdl/SDL_joystick.h | 167 + src/win32/dependencies/sdl/SDL_keyboard.h | 121 + src/win32/dependencies/sdl/SDL_keysym.h | 311 + src/win32/dependencies/sdl/SDL_loadso.h | 74 + src/win32/dependencies/sdl/SDL_main.h | 98 + src/win32/dependencies/sdl/SDL_mouse.h | 136 + src/win32/dependencies/sdl/SDL_mutex.h | 162 + src/win32/dependencies/sdl/SDL_name.h | 11 + src/win32/dependencies/sdl/SDL_opengl.h | 6551 +++++++++++++ src/win32/dependencies/sdl/SDL_platform.h | 104 + src/win32/dependencies/sdl/SDL_quit.h | 50 + src/win32/dependencies/sdl/SDL_rwops.h | 139 + src/win32/dependencies/sdl/SDL_stdinc.h | 584 ++ src/win32/dependencies/sdl/SDL_syswm.h | 210 + src/win32/dependencies/sdl/SDL_thread.h | 119 + src/win32/dependencies/sdl/SDL_timer.h | 115 + src/win32/dependencies/sdl/SDL_types.h | 24 + src/win32/dependencies/sdl/SDL_version.h | 85 + src/win32/dependencies/sdl/SDL_video.h | 889 ++ src/win32/dependencies/sdl/begin_code.h | 150 + src/win32/dependencies/sdl/close_code.h | 41 + src/win32/dependencies/zlib/.cvsignore | 3 + src/win32/dependencies/zlib/CVS/Entries | 25 + src/win32/dependencies/zlib/CVS/Repository | 1 + src/win32/dependencies/zlib/CVS/Root | 1 + .../win32/dependencies/zlib}/adler32.c | 2 +- .../win32/dependencies/zlib}/compress.c | 2 +- {zlib => src/win32/dependencies/zlib}/crc32.c | 2 +- {zlib => src/win32/dependencies/zlib}/crc32.h | 0 .../win32/dependencies/zlib}/deflate.c | 2 +- .../win32/dependencies/zlib}/deflate.h | 2 +- {zlib => src/win32/dependencies/zlib}/gzio.c | 2 +- .../win32/dependencies/zlib}/infback.c | 0 .../win32/dependencies/zlib}/inffast.c | 0 .../win32/dependencies/zlib}/inffast.h | 0 .../win32/dependencies/zlib}/inffixed.h | 0 .../win32/dependencies/zlib}/inflate.c | 0 .../win32/dependencies/zlib}/inflate.h | 0 .../win32/dependencies/zlib}/inftrees.c | 0 .../win32/dependencies/zlib}/inftrees.h | 0 {zlib => src/win32/dependencies/zlib}/trees.c | 2 +- {zlib => src/win32/dependencies/zlib}/trees.h | 0 .../win32/dependencies/zlib}/uncompr.c | 2 +- {zlib => src/win32/dependencies/zlib}/zconf.h | 2 +- {zlib => src/win32/dependencies/zlib}/zlib.h | 0 src/win32/dependencies/zlib/zlib.vcproj | 255 + {zlib => src/win32/dependencies/zlib}/zutil.c | 2 +- {zlib => src/win32/dependencies/zlib}/zutil.h | 2 +- src/win32/display.cpp | 123 + src/win32/gbadvance.ico | Bin 0 -> 4710 bytes src/win32/resource.h | 807 ++ src/win32/resource2.h | 1 + src/win32/skin.cpp | 588 ++ src/win32/skin.h | 162 + src/win32/skinButton.cpp | 330 + src/win32/skinButton.h | 93 + src/win32/stdafx.cpp | 18 + src/win32/stdafx.h | 90 +- src/win32/vba.rc2 | 56 + src/win32/vbavista.ico | Bin 0 -> 501830 bytes zlib/zlib.vcproj | 239 - zlib/zlib.vcproj.7.10.old | 167 - 301 files changed, 100519 insertions(+), 33545 deletions(-) create mode 100644 src/AutoBuild.h delete mode 100644 src/getopt.cpp delete mode 100644 src/getopt1.cpp rename src/{memgzio.cpp => memgzio.c} (93%) rename {res => src/win32}/VBA.rc (83%) create mode 100644 src/win32/dependencies/CVS/Entries create mode 100644 src/win32/dependencies/CVS/Repository create mode 100644 src/win32/dependencies/CVS/Root create mode 100644 src/win32/dependencies/cximage/CVS/Entries create mode 100644 src/win32/dependencies/cximage/CVS/Repository create mode 100644 src/win32/dependencies/cximage/CVS/Root create mode 100644 src/win32/dependencies/cximage/cximage.vcproj create mode 100644 src/win32/dependencies/cximage/license.txt create mode 100644 src/win32/dependencies/cximage/tif_xfile.cpp create mode 100644 src/win32/dependencies/cximage/xfile.h create mode 100644 src/win32/dependencies/cximage/ximabmp.cpp create mode 100644 src/win32/dependencies/cximage/ximabmp.h create mode 100644 src/win32/dependencies/cximage/ximacfg.h create mode 100644 src/win32/dependencies/cximage/ximadef.h create mode 100644 src/win32/dependencies/cximage/ximadsp.cpp create mode 100644 src/win32/dependencies/cximage/ximaenc.cpp create mode 100644 src/win32/dependencies/cximage/ximaexif.cpp create mode 100644 src/win32/dependencies/cximage/ximage.cpp create mode 100644 src/win32/dependencies/cximage/ximage.h create mode 100644 src/win32/dependencies/cximage/ximagif.cpp create mode 100644 src/win32/dependencies/cximage/ximagif.h create mode 100644 src/win32/dependencies/cximage/ximahist.cpp create mode 100644 src/win32/dependencies/cximage/ximaico.cpp create mode 100644 src/win32/dependencies/cximage/ximaico.h create mode 100644 src/win32/dependencies/cximage/ximainfo.cpp create mode 100644 src/win32/dependencies/cximage/ximaint.cpp create mode 100644 src/win32/dependencies/cximage/ximaiter.h create mode 100644 src/win32/dependencies/cximage/ximaj2k.cpp create mode 100644 src/win32/dependencies/cximage/ximaj2k.h create mode 100644 src/win32/dependencies/cximage/ximajas.cpp create mode 100644 src/win32/dependencies/cximage/ximajas.h create mode 100644 src/win32/dependencies/cximage/ximajbg.cpp create mode 100644 src/win32/dependencies/cximage/ximajbg.h create mode 100644 src/win32/dependencies/cximage/ximajpg.cpp create mode 100644 src/win32/dependencies/cximage/ximajpg.h create mode 100644 src/win32/dependencies/cximage/ximalpha.cpp create mode 100644 src/win32/dependencies/cximage/ximalyr.cpp create mode 100644 src/win32/dependencies/cximage/ximamng.cpp create mode 100644 src/win32/dependencies/cximage/ximamng.h create mode 100644 src/win32/dependencies/cximage/ximapal.cpp create mode 100644 src/win32/dependencies/cximage/ximapcx.cpp create mode 100644 src/win32/dependencies/cximage/ximapcx.h create mode 100644 src/win32/dependencies/cximage/ximapng.cpp create mode 100644 src/win32/dependencies/cximage/ximapng.h create mode 100644 src/win32/dependencies/cximage/ximasel.cpp create mode 100644 src/win32/dependencies/cximage/ximatga.cpp create mode 100644 src/win32/dependencies/cximage/ximatga.h create mode 100644 src/win32/dependencies/cximage/ximath.cpp create mode 100644 src/win32/dependencies/cximage/ximath.h create mode 100644 src/win32/dependencies/cximage/ximatif.cpp create mode 100644 src/win32/dependencies/cximage/ximatif.h create mode 100644 src/win32/dependencies/cximage/ximatran.cpp create mode 100644 src/win32/dependencies/cximage/ximawbmp.cpp create mode 100644 src/win32/dependencies/cximage/ximawbmp.h create mode 100644 src/win32/dependencies/cximage/ximawmf.cpp create mode 100644 src/win32/dependencies/cximage/ximawmf.h create mode 100644 src/win32/dependencies/cximage/ximawnd.cpp create mode 100644 src/win32/dependencies/cximage/xiofile.h create mode 100644 src/win32/dependencies/cximage/xmemfile.cpp create mode 100644 src/win32/dependencies/cximage/xmemfile.h create mode 100644 src/win32/dependencies/info.txt create mode 100644 src/win32/dependencies/libpng/.cvsignore create mode 100644 src/win32/dependencies/libpng/CVS/Entries create mode 100644 src/win32/dependencies/libpng/CVS/Repository create mode 100644 src/win32/dependencies/libpng/CVS/Root create mode 100644 src/win32/dependencies/libpng/libpng.vcproj create mode 100644 src/win32/dependencies/libpng/png.c create mode 100644 src/win32/dependencies/libpng/png.h create mode 100644 src/win32/dependencies/libpng/pngconf.h create mode 100644 src/win32/dependencies/libpng/pngerror.c create mode 100644 src/win32/dependencies/libpng/pngget.c create mode 100644 src/win32/dependencies/libpng/pngmem.c create mode 100644 src/win32/dependencies/libpng/pngpread.c create mode 100644 src/win32/dependencies/libpng/pngread.c create mode 100644 src/win32/dependencies/libpng/pngrio.c create mode 100644 src/win32/dependencies/libpng/pngrtran.c create mode 100644 src/win32/dependencies/libpng/pngrutil.c create mode 100644 src/win32/dependencies/libpng/pngset.c create mode 100644 src/win32/dependencies/libpng/pngtrans.c create mode 100644 src/win32/dependencies/libpng/pngwio.c create mode 100644 src/win32/dependencies/libpng/pngwrite.c create mode 100644 src/win32/dependencies/libpng/pngwtran.c create mode 100644 src/win32/dependencies/libpng/pngwutil.c create mode 100644 src/win32/dependencies/sdl/CVS/Entries create mode 100644 src/win32/dependencies/sdl/CVS/Repository create mode 100644 src/win32/dependencies/sdl/CVS/Root create mode 100644 src/win32/dependencies/sdl/SDL.h create mode 100644 src/win32/dependencies/sdl/SDL_Debug/CVS/Entries create mode 100644 src/win32/dependencies/sdl/SDL_Debug/CVS/Repository create mode 100644 src/win32/dependencies/sdl/SDL_Debug/CVS/Root create mode 100644 src/win32/dependencies/sdl/SDL_Debug/SDL.lib create mode 100644 src/win32/dependencies/sdl/SDL_Debug/SDLmain.lib create mode 100644 src/win32/dependencies/sdl/SDL_Release/CVS/Entries create mode 100644 src/win32/dependencies/sdl/SDL_Release/CVS/Repository create mode 100644 src/win32/dependencies/sdl/SDL_Release/CVS/Root create mode 100644 src/win32/dependencies/sdl/SDL_Release/SDL.lib create mode 100644 src/win32/dependencies/sdl/SDL_Release/SDLmain.lib create mode 100644 src/win32/dependencies/sdl/SDL_active.h create mode 100644 src/win32/dependencies/sdl/SDL_audio.h create mode 100644 src/win32/dependencies/sdl/SDL_byteorder.h create mode 100644 src/win32/dependencies/sdl/SDL_cdrom.h create mode 100644 src/win32/dependencies/sdl/SDL_config.h create mode 100644 src/win32/dependencies/sdl/SDL_config.h.default create mode 100644 src/win32/dependencies/sdl/SDL_config.h.in create mode 100644 src/win32/dependencies/sdl/SDL_config_amiga.h create mode 100644 src/win32/dependencies/sdl/SDL_config_dreamcast.h create mode 100644 src/win32/dependencies/sdl/SDL_config_macos.h create mode 100644 src/win32/dependencies/sdl/SDL_config_macosx.h create mode 100644 src/win32/dependencies/sdl/SDL_config_minimal.h create mode 100644 src/win32/dependencies/sdl/SDL_config_os2.h create mode 100644 src/win32/dependencies/sdl/SDL_config_win32.h create mode 100644 src/win32/dependencies/sdl/SDL_copying.h create mode 100644 src/win32/dependencies/sdl/SDL_cpuinfo.h create mode 100644 src/win32/dependencies/sdl/SDL_endian.h create mode 100644 src/win32/dependencies/sdl/SDL_error.h create mode 100644 src/win32/dependencies/sdl/SDL_events.h create mode 100644 src/win32/dependencies/sdl/SDL_getenv.h create mode 100644 src/win32/dependencies/sdl/SDL_joystick.h create mode 100644 src/win32/dependencies/sdl/SDL_keyboard.h create mode 100644 src/win32/dependencies/sdl/SDL_keysym.h create mode 100644 src/win32/dependencies/sdl/SDL_loadso.h create mode 100644 src/win32/dependencies/sdl/SDL_main.h create mode 100644 src/win32/dependencies/sdl/SDL_mouse.h create mode 100644 src/win32/dependencies/sdl/SDL_mutex.h create mode 100644 src/win32/dependencies/sdl/SDL_name.h create mode 100644 src/win32/dependencies/sdl/SDL_opengl.h create mode 100644 src/win32/dependencies/sdl/SDL_platform.h create mode 100644 src/win32/dependencies/sdl/SDL_quit.h create mode 100644 src/win32/dependencies/sdl/SDL_rwops.h create mode 100644 src/win32/dependencies/sdl/SDL_stdinc.h create mode 100644 src/win32/dependencies/sdl/SDL_syswm.h create mode 100644 src/win32/dependencies/sdl/SDL_thread.h create mode 100644 src/win32/dependencies/sdl/SDL_timer.h create mode 100644 src/win32/dependencies/sdl/SDL_types.h create mode 100644 src/win32/dependencies/sdl/SDL_version.h create mode 100644 src/win32/dependencies/sdl/SDL_video.h create mode 100644 src/win32/dependencies/sdl/begin_code.h create mode 100644 src/win32/dependencies/sdl/close_code.h create mode 100644 src/win32/dependencies/zlib/.cvsignore create mode 100644 src/win32/dependencies/zlib/CVS/Entries create mode 100644 src/win32/dependencies/zlib/CVS/Repository create mode 100644 src/win32/dependencies/zlib/CVS/Root rename {zlib => src/win32/dependencies/zlib}/adler32.c (98%) rename {zlib => src/win32/dependencies/zlib}/compress.c (97%) rename {zlib => src/win32/dependencies/zlib}/crc32.c (99%) rename {zlib => src/win32/dependencies/zlib}/crc32.h (100%) rename {zlib => src/win32/dependencies/zlib}/deflate.c (99%) rename {zlib => src/win32/dependencies/zlib}/deflate.h (99%) rename {zlib => src/win32/dependencies/zlib}/gzio.c (99%) rename {zlib => src/win32/dependencies/zlib}/infback.c (100%) rename {zlib => src/win32/dependencies/zlib}/inffast.c (100%) rename {zlib => src/win32/dependencies/zlib}/inffast.h (100%) rename {zlib => src/win32/dependencies/zlib}/inffixed.h (100%) rename {zlib => src/win32/dependencies/zlib}/inflate.c (100%) rename {zlib => src/win32/dependencies/zlib}/inflate.h (100%) rename {zlib => src/win32/dependencies/zlib}/inftrees.c (100%) rename {zlib => src/win32/dependencies/zlib}/inftrees.h (100%) rename {zlib => src/win32/dependencies/zlib}/trees.c (99%) rename {zlib => src/win32/dependencies/zlib}/trees.h (100%) rename {zlib => src/win32/dependencies/zlib}/uncompr.c (96%) rename {zlib => src/win32/dependencies/zlib}/zconf.h (99%) rename {zlib => src/win32/dependencies/zlib}/zlib.h (100%) create mode 100644 src/win32/dependencies/zlib/zlib.vcproj rename {zlib => src/win32/dependencies/zlib}/zutil.c (99%) rename {zlib => src/win32/dependencies/zlib}/zutil.h (99%) create mode 100644 src/win32/display.cpp create mode 100644 src/win32/gbadvance.ico create mode 100644 src/win32/resource.h create mode 100644 src/win32/resource2.h create mode 100644 src/win32/skin.cpp create mode 100644 src/win32/skin.h create mode 100644 src/win32/skinButton.cpp create mode 100644 src/win32/skinButton.h create mode 100644 src/win32/vba.rc2 create mode 100644 src/win32/vbavista.ico delete mode 100644 zlib/zlib.vcproj delete mode 100644 zlib/zlib.vcproj.7.10.old diff --git a/VBA.vcproj b/VBA.vcproj index 88c3a3c9..861b63ea 100644 --- a/VBA.vcproj +++ b/VBA.vcproj @@ -51,7 +51,7 @@ - - - - @@ -683,16 +675,40 @@ > + + + + + + + + + + + @@ -1002,6 +1022,14 @@ RelativePath=".\src\win32\RomInfo.cpp" > + + + + @@ -1014,10 +1042,6 @@ RelativePath=".\src\win32\TileView.cpp" > - - @@ -1106,6 +1130,10 @@ RelativePath=".\src\win32\DirectSound.cpp" > + + @@ -1155,6 +1183,130 @@ > + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + - - - - diff --git a/src/2xSaI.cpp b/src/2xSaI.cpp index 61eb116b..1aacc0a8 100644 --- a/src/2xSaI.cpp +++ b/src/2xSaI.cpp @@ -1,6 +1,8 @@ #include "System.h" #include "Port.h" +extern int RGB_LOW_BITS_MASK; + extern "C" { @@ -48,6 +50,7 @@ int Init_2xSaI(u32 BitFormat) greenMask = 0x7E0; qRGB_COLOR_MASK[0] = qRGB_COLOR_MASK[1] = 0xF7DEF7DE; hq2x_init(16); + RGB_LOW_BITS_MASK = 0x0821; } else if (BitFormat == 555) { colorMask = 0x7BDE7BDE; lowPixelMask = 0x04210421; @@ -57,6 +60,7 @@ int Init_2xSaI(u32 BitFormat) greenMask = 0x3E0; qRGB_COLOR_MASK[0] = qRGB_COLOR_MASK[1] = 0x7BDE7BDE; hq2x_init(15); + RGB_LOW_BITS_MASK = 0x0421; } else { return 0; } @@ -67,6 +71,7 @@ int Init_2xSaI(u32 BitFormat) qlowpixelMask = 0x030303; qRGB_COLOR_MASK[0] = qRGB_COLOR_MASK[1] = 0xfefefe; hq2x_init(32); + RGB_LOW_BITS_MASK = 0x010101; } else return 0; diff --git a/src/AutoBuild.h b/src/AutoBuild.h new file mode 100644 index 00000000..425c57b0 --- /dev/null +++ b/src/AutoBuild.h @@ -0,0 +1,31 @@ +// -*- C++ -*- +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#ifndef __AUTOBUILD_H__ +#define __AUTOBUILD_H__ +#ifndef VERSION +#define VERSION "1.8.0" +#endif +//change the FALSE to TRUE for autoincrement of build number +#define INCREMENT_VERSION FALSE +#define FILEVER 1,8,0,600 +#define PRODUCTVER 1,8,0,600 +#define STRFILEVER "1, 8, 0, 600\0" +#define STRPRODUCTVER "1, 8, 0, 600\0" +#endif //__AUTOBUILD_H__ diff --git a/src/Cheats.cpp b/src/Cheats.cpp index 2eeae9b1..c8290495 100644 --- a/src/Cheats.cpp +++ b/src/Cheats.cpp @@ -1,2770 +1,2895 @@ -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#include -#include -#include -#include - -#include "GBA.h" -#include "GBAinline.h" -#include "Cheats.h" -#include "Globals.h" -#include "NLS.h" -#include "Util.h" - -/** - * Gameshark code types: (based on AR v1.0) - * - * NNNNNNNN 001DC0DE - ID code for the game (game 4 character name) from ROM - * DEADFACE XXXXXXXX - changes decryption seeds // Not supported by VBA. - * 0AAAAAAA 000000YY - 8-bit constant write - * 1AAAAAAA 0000YYYY - 16-bit constant write - * 2AAAAAAA YYYYYYYY - 32-bit constant write - * 30XXAAAA YYYYYYYY - 32bit Group Write, 8/16/32bit Sub/Add (depending on the XX value). - * 6AAAAAAA Z000YYYY - 16-bit ROM Patch (address >> 1). Z selects the Rom Patching register. - * - AR v1/2 hardware only supports Z=0. - * - AR v3 hardware should support Z=0,1,2 or 3. - * 8A1AAAAA 000000YY - 8-bit button write - * 8A2AAAAA 0000YYYY - 16-bit button write - * 8A4AAAAA YYYYYYYY - 32-bit button write // BUGGY ! Only writes 00000000 on the AR v1.0. - * 80F00000 0000YYYY - button slow motion - * DAAAAAAA 00Z0YYYY - Z = 0 : if 16-bit value at address != YYYY skip next line - * - Z = 1 : if 16-bit value at address == YYYY skip next line - * - Z = 2 : if 16-bit value at address > YYYY (Unsigned) skip next line - * - Z = 3 : if 16-bit value at address < YYYY (Unsigned) skip next line - * E0CCYYYY ZAAAAAAA - Z = 0 : if 16-bit value at address != YYYY skip CC lines - * - Z = 1 : if 16-bit value at address == YYYY skip CC lines - * - Z = 2 : if 16-bit value at address > YYYY (Unsigned) skip CC lines - * - Z = 3 : if 16-bit value at address < YYYY (Unsigned) skip CC lines - * FAAAAAAA 0000YYYY - Master code function - * - * - * - * CodeBreaker codes types: (based on the CBA clone "Cheatcode S" v1.1) - * - * 0000AAAA 000Y - Game CRC (Y are flags: 8 - CRC, 2 - DI) - * 1AAAAAAA YYYY - Master Code function (store address at ((YYYY << 0x16) - * + 0x08000100)) - * 2AAAAAAA YYYY - 16-bit or - * 3AAAAAAA YYYY - 8-bit constant write - * 4AAAAAAA YYYY - Slide code - * XXXXCCCC IIII (C is count and I is address increment, X is value incr.) - * 5AAAAAAA CCCC - Super code (Write bytes to address, 2*CCCC is count) - * BBBBBBBB BBBB - * 6AAAAAAA YYYY - 16-bit and - * 7AAAAAAA YYYY - if address contains 16-bit value enable next code - * 8AAAAAAA YYYY - 16-bit constant write - * 9AAAAAAA YYYY - change decryption (when first code only?) - * AAAAAAAA YYYY - if address does not contain 16-bit value enable next code - * BAAAAAAA YYYY - if 16-bit value at address <= YYYY skip next code - * CAAAAAAA YYYY - if 16-bit value at address >= YYYY skip next code - * D00000X0 YYYY - if button keys ... enable next code (else skip next code) - * EAAAAAAA YYYY - increase 16/32bit value stored in address - * FAAAAAAA YYYY - if 16-bit value at address AND YYYY = 0 then skip next code - **/ - -#define UNKNOWN_CODE -1 -#define INT_8_BIT_WRITE 0 -#define INT_16_BIT_WRITE 1 -#define INT_32_BIT_WRITE 2 -#define GSA_16_BIT_ROM_PATCH 3 -#define GSA_8_BIT_GS_WRITE 4 -#define GSA_16_BIT_GS_WRITE 5 -#define GSA_32_BIT_GS_WRITE 6 -#define CBA_IF_KEYS_PRESSED 7 -#define CBA_IF_TRUE 8 -#define CBA_SLIDE_CODE 9 -#define CBA_IF_FALSE 10 -#define CBA_AND 11 -#define GSA_8_BIT_GS_WRITE2 12 -#define GSA_16_BIT_GS_WRITE2 13 -#define GSA_32_BIT_GS_WRITE2 14 -#define GSA_16_BIT_ROM_PATCH2C 15 -#define GSA_8_BIT_SLIDE 16 -#define GSA_16_BIT_SLIDE 17 -#define GSA_32_BIT_SLIDE 18 -#define GSA_8_BIT_IF_TRUE 19 -#define GSA_32_BIT_IF_TRUE 20 -#define GSA_8_BIT_IF_FALSE 21 -#define GSA_32_BIT_IF_FALSE 22 -#define GSA_8_BIT_FILL 23 -#define GSA_16_BIT_FILL 24 -#define GSA_8_BIT_IF_TRUE2 25 -#define GSA_16_BIT_IF_TRUE2 26 -#define GSA_32_BIT_IF_TRUE2 27 -#define GSA_8_BIT_IF_FALSE2 28 -#define GSA_16_BIT_IF_FALSE2 29 -#define GSA_32_BIT_IF_FALSE2 30 -#define GSA_SLOWDOWN 31 -#define CBA_ADD 32 -#define CBA_OR 33 -#define CBA_LT 34 -#define CBA_GT 35 -#define CBA_SUPER 36 -#define GSA_8_BIT_POINTER 37 -#define GSA_16_BIT_POINTER 38 -#define GSA_32_BIT_POINTER 39 -#define GSA_8_BIT_ADD 40 -#define GSA_16_BIT_ADD 41 -#define GSA_32_BIT_ADD 42 -#define GSA_8_BIT_IF_LOWER_U 43 -#define GSA_16_BIT_IF_LOWER_U 44 -#define GSA_32_BIT_IF_LOWER_U 45 -#define GSA_8_BIT_IF_HIGHER_U 46 -#define GSA_16_BIT_IF_HIGHER_U 47 -#define GSA_32_BIT_IF_HIGHER_U 48 -#define GSA_8_BIT_IF_AND 49 -#define GSA_16_BIT_IF_AND 50 -#define GSA_32_BIT_IF_AND 51 -#define GSA_8_BIT_IF_LOWER_U2 52 -#define GSA_16_BIT_IF_LOWER_U2 53 -#define GSA_32_BIT_IF_LOWER_U2 54 -#define GSA_8_BIT_IF_HIGHER_U2 55 -#define GSA_16_BIT_IF_HIGHER_U2 56 -#define GSA_32_BIT_IF_HIGHER_U2 57 -#define GSA_8_BIT_IF_AND2 58 -#define GSA_16_BIT_IF_AND2 59 -#define GSA_32_BIT_IF_AND2 60 -#define GSA_ALWAYS 61 -#define GSA_ALWAYS2 62 -#define GSA_8_BIT_IF_LOWER_S 63 -#define GSA_16_BIT_IF_LOWER_S 64 -#define GSA_32_BIT_IF_LOWER_S 65 -#define GSA_8_BIT_IF_HIGHER_S 66 -#define GSA_16_BIT_IF_HIGHER_S 67 -#define GSA_32_BIT_IF_HIGHER_S 68 -#define GSA_8_BIT_IF_LOWER_S2 69 -#define GSA_16_BIT_IF_LOWER_S2 70 -#define GSA_32_BIT_IF_LOWER_S2 71 -#define GSA_8_BIT_IF_HIGHER_S2 72 -#define GSA_16_BIT_IF_HIGHER_S2 73 -#define GSA_32_BIT_IF_HIGHER_S2 74 -#define GSA_16_BIT_WRITE_IOREGS 75 -#define GSA_32_BIT_WRITE_IOREGS 76 -#define GSA_CODES_ON 77 -#define GSA_8_BIT_IF_TRUE3 78 -#define GSA_16_BIT_IF_TRUE3 79 -#define GSA_32_BIT_IF_TRUE3 80 -#define GSA_8_BIT_IF_FALSE3 81 -#define GSA_16_BIT_IF_FALSE3 82 -#define GSA_32_BIT_IF_FALSE3 83 -#define GSA_8_BIT_IF_LOWER_S3 84 -#define GSA_16_BIT_IF_LOWER_S3 85 -#define GSA_32_BIT_IF_LOWER_S3 86 -#define GSA_8_BIT_IF_HIGHER_S3 87 -#define GSA_16_BIT_IF_HIGHER_S3 88 -#define GSA_32_BIT_IF_HIGHER_S3 89 -#define GSA_8_BIT_IF_LOWER_U3 90 -#define GSA_16_BIT_IF_LOWER_U3 91 -#define GSA_32_BIT_IF_LOWER_U3 92 -#define GSA_8_BIT_IF_HIGHER_U3 93 -#define GSA_16_BIT_IF_HIGHER_U3 94 -#define GSA_32_BIT_IF_HIGHER_U3 95 -#define GSA_8_BIT_IF_AND3 96 -#define GSA_16_BIT_IF_AND3 97 -#define GSA_32_BIT_IF_AND3 98 -#define GSA_ALWAYS3 99 -#define GSA_16_BIT_ROM_PATCH2D 100 -#define GSA_16_BIT_ROM_PATCH2E 101 -#define GSA_16_BIT_ROM_PATCH2F 102 -#define GSA_GROUP_WRITE 103 -#define GSA_32_BIT_ADD2 104 -#define GSA_32_BIT_SUB2 105 -#define GSA_16_BIT_IF_LOWER_OR_EQ_U 106 -#define GSA_16_BIT_IF_HIGHER_OR_EQ_U 107 -#define GSA_16_BIT_MIF_TRUE 108 -#define GSA_16_BIT_MIF_FALSE 109 -#define GSA_16_BIT_MIF_LOWER_OR_EQ_U 110 -#define GSA_16_BIT_MIF_HIGHER_OR_EQ_U 111 - -CheatsData cheatsList[100]; -int cheatsNumber = 0; -u32 rompatch2addr [4]; -u16 rompatch2val [4]; -u16 rompatch2oldval [4]; - -u8 cheatsCBASeedBuffer[0x30]; -u32 cheatsCBASeed[4]; -u32 cheatsCBATemporaryValue = 0; -u16 cheatsCBATable[256]; -bool cheatsCBATableGenerated = false; -u16 super = 0; - -u8 cheatsCBACurrentSeed[12] = { - 0x00, 0x00, 0x00, 0x00, - 0x00, 0x00, 0x00, 0x00, - 0x00, 0x00, 0x00, 0x00 -}; - -u32 seeds_v1[4]; -u32 seeds_v3[4]; - -u32 seed_gen(u8 upper, u8 seed, u8 *deadtable1, u8 *deadtable2); - -//seed tables for AR v1 -u8 v1_deadtable1[256] = { - 0x31, 0x1C, 0x23, 0xE5, 0x89, 0x8E, 0xA1, 0x37, 0x74, 0x6D, 0x67, 0xFC, 0x1F, 0xC0, 0xB1, 0x94, - 0x3B, 0x05, 0x56, 0x86, 0x00, 0x24, 0xF0, 0x17, 0x72, 0xA2, 0x3D, 0x1B, 0xE3, 0x17, 0xC5, 0x0B, - 0xB9, 0xE2, 0xBD, 0x58, 0x71, 0x1B, 0x2C, 0xFF, 0xE4, 0xC9, 0x4C, 0x5E, 0xC9, 0x55, 0x33, 0x45, - 0x7C, 0x3F, 0xB2, 0x51, 0xFE, 0x10, 0x7E, 0x75, 0x3C, 0x90, 0x8D, 0xDA, 0x94, 0x38, 0xC3, 0xE9, - 0x95, 0xEA, 0xCE, 0xA6, 0x06, 0xE0, 0x4F, 0x3F, 0x2A, 0xE3, 0x3A, 0xE4, 0x43, 0xBD, 0x7F, 0xDA, - 0x55, 0xF0, 0xEA, 0xCB, 0x2C, 0xA8, 0x47, 0x61, 0xA0, 0xEF, 0xCB, 0x13, 0x18, 0x20, 0xAF, 0x3E, - 0x4D, 0x9E, 0x1E, 0x77, 0x51, 0xC5, 0x51, 0x20, 0xCF, 0x21, 0xF9, 0x39, 0x94, 0xDE, 0xDD, 0x79, - 0x4E, 0x80, 0xC4, 0x9D, 0x94, 0xD5, 0x95, 0x01, 0x27, 0x27, 0xBD, 0x6D, 0x78, 0xB5, 0xD1, 0x31, - 0x6A, 0x65, 0x74, 0x74, 0x58, 0xB3, 0x7C, 0xC9, 0x5A, 0xED, 0x50, 0x03, 0xC4, 0xA2, 0x94, 0x4B, - 0xF0, 0x58, 0x09, 0x6F, 0x3E, 0x7D, 0xAE, 0x7D, 0x58, 0xA0, 0x2C, 0x91, 0xBB, 0xE1, 0x70, 0xEB, - 0x73, 0xA6, 0x9A, 0x44, 0x25, 0x90, 0x16, 0x62, 0x53, 0xAE, 0x08, 0xEB, 0xDC, 0xF0, 0xEE, 0x77, - 0xC2, 0xDE, 0x81, 0xE8, 0x30, 0x89, 0xDB, 0xFE, 0xBC, 0xC2, 0xDF, 0x26, 0xE9, 0x8B, 0xD6, 0x93, - 0xF0, 0xCB, 0x56, 0x90, 0xC0, 0x46, 0x68, 0x15, 0x43, 0xCB, 0xE9, 0x98, 0xE3, 0xAF, 0x31, 0x25, - 0x4D, 0x7B, 0xF3, 0xB1, 0x74, 0xE2, 0x64, 0xAC, 0xD9, 0xF6, 0xA0, 0xD5, 0x0B, 0x9B, 0x49, 0x52, - 0x69, 0x3B, 0x71, 0x00, 0x2F, 0xBB, 0xBA, 0x08, 0xB1, 0xAE, 0xBB, 0xB3, 0xE1, 0xC9, 0xA6, 0x7F, - 0x17, 0x97, 0x28, 0x72, 0x12, 0x6E, 0x91, 0xAE, 0x3A, 0xA2, 0x35, 0x46, 0x27, 0xF8, 0x12, 0x50 -}; -u8 v1_deadtable2[256] = { - 0xD8, 0x65, 0x04, 0xC2, 0x65, 0xD5, 0xB0, 0x0C, 0xDF, 0x9D, 0xF0, 0xC3, 0x9A, 0x17, 0xC9, 0xA6, - 0xE1, 0xAC, 0x0D, 0x14, 0x2F, 0x3C, 0x2C, 0x87, 0xA2, 0xBF, 0x4D, 0x5F, 0xAC, 0x2D, 0x9D, 0xE1, - 0x0C, 0x9C, 0xE7, 0x7F, 0xFC, 0xA8, 0x66, 0x59, 0xAC, 0x18, 0xD7, 0x05, 0xF0, 0xBF, 0xD1, 0x8B, - 0x35, 0x9F, 0x59, 0xB4, 0xBA, 0x55, 0xB2, 0x85, 0xFD, 0xB1, 0x72, 0x06, 0x73, 0xA4, 0xDB, 0x48, - 0x7B, 0x5F, 0x67, 0xA5, 0x95, 0xB9, 0xA5, 0x4A, 0xCF, 0xD1, 0x44, 0xF3, 0x81, 0xF5, 0x6D, 0xF6, - 0x3A, 0xC3, 0x57, 0x83, 0xFA, 0x8E, 0x15, 0x2A, 0xA2, 0x04, 0xB2, 0x9D, 0xA8, 0x0D, 0x7F, 0xB8, - 0x0F, 0xF6, 0xAC, 0xBE, 0x97, 0xCE, 0x16, 0xE6, 0x31, 0x10, 0x60, 0x16, 0xB5, 0x83, 0x45, 0xEE, - 0xD7, 0x5F, 0x2C, 0x08, 0x58, 0xB1, 0xFD, 0x7E, 0x79, 0x00, 0x34, 0xAD, 0xB5, 0x31, 0x34, 0x39, - 0xAF, 0xA8, 0xDD, 0x52, 0x6A, 0xB0, 0x60, 0x35, 0xB8, 0x1D, 0x52, 0xF5, 0xF5, 0x30, 0x00, 0x7B, - 0xF4, 0xBA, 0x03, 0xCB, 0x3A, 0x84, 0x14, 0x8A, 0x6A, 0xEF, 0x21, 0xBD, 0x01, 0xD8, 0xA0, 0xD4, - 0x43, 0xBE, 0x23, 0xE7, 0x76, 0x27, 0x2C, 0x3F, 0x4D, 0x3F, 0x43, 0x18, 0xA7, 0xC3, 0x47, 0xA5, - 0x7A, 0x1D, 0x02, 0x55, 0x09, 0xD1, 0xFF, 0x55, 0x5E, 0x17, 0xA0, 0x56, 0xF4, 0xC9, 0x6B, 0x90, - 0xB4, 0x80, 0xA5, 0x07, 0x22, 0xFB, 0x22, 0x0D, 0xD9, 0xC0, 0x5B, 0x08, 0x35, 0x05, 0xC1, 0x75, - 0x4F, 0xD0, 0x51, 0x2D, 0x2E, 0x5E, 0x69, 0xE7, 0x3B, 0xC2, 0xDA, 0xFF, 0xF6, 0xCE, 0x3E, 0x76, - 0xE8, 0x36, 0x8C, 0x39, 0xD8, 0xF3, 0xE9, 0xA6, 0x42, 0xE6, 0xC1, 0x4C, 0x05, 0xBE, 0x17, 0xF2, - 0x5C, 0x1B, 0x19, 0xDB, 0x0F, 0xF3, 0xF8, 0x49, 0xEB, 0x36, 0xF6, 0x40, 0x6F, 0xAD, 0xC1, 0x8C -}; - -//seed tables for AR v3 -u8 v3_deadtable1[256] = { - 0xD0, 0xFF, 0xBA, 0xE5, 0xC1, 0xC7, 0xDB, 0x5B, 0x16, 0xE3, 0x6E, 0x26, 0x62, 0x31, 0x2E, 0x2A, - 0xD1, 0xBB, 0x4A, 0xE6, 0xAE, 0x2F, 0x0A, 0x90, 0x29, 0x90, 0xB6, 0x67, 0x58, 0x2A, 0xB4, 0x45, - 0x7B, 0xCB, 0xF0, 0x73, 0x84, 0x30, 0x81, 0xC2, 0xD7, 0xBE, 0x89, 0xD7, 0x4E, 0x73, 0x5C, 0xC7, - 0x80, 0x1B, 0xE5, 0xE4, 0x43, 0xC7, 0x46, 0xD6, 0x6F, 0x7B, 0xBF, 0xED, 0xE5, 0x27, 0xD1, 0xB5, - 0xD0, 0xD8, 0xA3, 0xCB, 0x2B, 0x30, 0xA4, 0xF0, 0x84, 0x14, 0x72, 0x5C, 0xFF, 0xA4, 0xFB, 0x54, - 0x9D, 0x70, 0xE2, 0xFF, 0xBE, 0xE8, 0x24, 0x76, 0xE5, 0x15, 0xFB, 0x1A, 0xBC, 0x87, 0x02, 0x2A, - 0x58, 0x8F, 0x9A, 0x95, 0xBD, 0xAE, 0x8D, 0x0C, 0xA5, 0x4C, 0xF2, 0x5C, 0x7D, 0xAD, 0x51, 0xFB, - 0xB1, 0x22, 0x07, 0xE0, 0x29, 0x7C, 0xEB, 0x98, 0x14, 0xC6, 0x31, 0x97, 0xE4, 0x34, 0x8F, 0xCC, - 0x99, 0x56, 0x9F, 0x78, 0x43, 0x91, 0x85, 0x3F, 0xC2, 0xD0, 0xD1, 0x80, 0xD1, 0x77, 0xA7, 0xE2, - 0x43, 0x99, 0x1D, 0x2F, 0x8B, 0x6A, 0xE4, 0x66, 0x82, 0xF7, 0x2B, 0x0B, 0x65, 0x14, 0xC0, 0xC2, - 0x1D, 0x96, 0x78, 0x1C, 0xC4, 0xC3, 0xD2, 0xB1, 0x64, 0x07, 0xD7, 0x6F, 0x02, 0xE9, 0x44, 0x31, - 0xDB, 0x3C, 0xEB, 0x93, 0xED, 0x9A, 0x57, 0x05, 0xB9, 0x0E, 0xAF, 0x1F, 0x48, 0x11, 0xDC, 0x35, - 0x6C, 0xB8, 0xEE, 0x2A, 0x48, 0x2B, 0xBC, 0x89, 0x12, 0x59, 0xCB, 0xD1, 0x18, 0xEA, 0x72, 0x11, - 0x01, 0x75, 0x3B, 0xB5, 0x56, 0xF4, 0x8B, 0xA0, 0x41, 0x75, 0x86, 0x7B, 0x94, 0x12, 0x2D, 0x4C, - 0x0C, 0x22, 0xC9, 0x4A, 0xD8, 0xB1, 0x8D, 0xF0, 0x55, 0x2E, 0x77, 0x50, 0x1C, 0x64, 0x77, 0xAA, - 0x3E, 0xAC, 0xD3, 0x3D, 0xCE, 0x60, 0xCA, 0x5D, 0xA0, 0x92, 0x78, 0xC6, 0x51, 0xFE, 0xF9, 0x30 -}; -u8 v3_deadtable2[256] = { - 0xAA, 0xAF, 0xF0, 0x72, 0x90, 0xF7, 0x71, 0x27, 0x06, 0x11, 0xEB, 0x9C, 0x37, 0x12, 0x72, 0xAA, - 0x65, 0xBC, 0x0D, 0x4A, 0x76, 0xF6, 0x5C, 0xAA, 0xB0, 0x7A, 0x7D, 0x81, 0xC1, 0xCE, 0x2F, 0x9F, - 0x02, 0x75, 0x38, 0xC8, 0xFC, 0x66, 0x05, 0xC2, 0x2C, 0xBD, 0x91, 0xAD, 0x03, 0xB1, 0x88, 0x93, - 0x31, 0xC6, 0xAB, 0x40, 0x23, 0x43, 0x76, 0x54, 0xCA, 0xE7, 0x00, 0x96, 0x9F, 0xD8, 0x24, 0x8B, - 0xE4, 0xDC, 0xDE, 0x48, 0x2C, 0xCB, 0xF7, 0x84, 0x1D, 0x45, 0xE5, 0xF1, 0x75, 0xA0, 0xED, 0xCD, - 0x4B, 0x24, 0x8A, 0xB3, 0x98, 0x7B, 0x12, 0xB8, 0xF5, 0x63, 0x97, 0xB3, 0xA6, 0xA6, 0x0B, 0xDC, - 0xD8, 0x4C, 0xA8, 0x99, 0x27, 0x0F, 0x8F, 0x94, 0x63, 0x0F, 0xB0, 0x11, 0x94, 0xC7, 0xE9, 0x7F, - 0x3B, 0x40, 0x72, 0x4C, 0xDB, 0x84, 0x78, 0xFE, 0xB8, 0x56, 0x08, 0x80, 0xDF, 0x20, 0x2F, 0xB9, - 0x66, 0x2D, 0x60, 0x63, 0xF5, 0x18, 0x15, 0x1B, 0x86, 0x85, 0xB9, 0xB4, 0x68, 0x0E, 0xC6, 0xD1, - 0x8A, 0x81, 0x2B, 0xB3, 0xF6, 0x48, 0xF0, 0x4F, 0x9C, 0x28, 0x1C, 0xA4, 0x51, 0x2F, 0xD7, 0x4B, - 0x17, 0xE7, 0xCC, 0x50, 0x9F, 0xD0, 0xD1, 0x40, 0x0C, 0x0D, 0xCA, 0x83, 0xFA, 0x5E, 0xCA, 0xEC, - 0xBF, 0x4E, 0x7C, 0x8F, 0xF0, 0xAE, 0xC2, 0xD3, 0x28, 0x41, 0x9B, 0xC8, 0x04, 0xB9, 0x4A, 0xBA, - 0x72, 0xE2, 0xB5, 0x06, 0x2C, 0x1E, 0x0B, 0x2C, 0x7F, 0x11, 0xA9, 0x26, 0x51, 0x9D, 0x3F, 0xF8, - 0x62, 0x11, 0x2E, 0x89, 0xD2, 0x9D, 0x35, 0xB1, 0xE4, 0x0A, 0x4D, 0x93, 0x01, 0xA7, 0xD1, 0x2D, - 0x00, 0x87, 0xE2, 0x2D, 0xA4, 0xE9, 0x0A, 0x06, 0x66, 0xF8, 0x1F, 0x44, 0x75, 0xB5, 0x6B, 0x1C, - 0xFC, 0x31, 0x09, 0x48, 0xA3, 0xFF, 0x92, 0x12, 0x58, 0xE9, 0xFA, 0xAE, 0x4F, 0xE2, 0xB4, 0xCC -}; - -#define debuggerReadMemory(addr) \ - READ32LE((&map[(addr)>>24].address[(addr) & map[(addr)>>24].mask])) - -#define debuggerReadHalfWord(addr) \ - READ16LE((&map[(addr)>>24].address[(addr) & map[(addr)>>24].mask])) - -#define debuggerReadByte(addr) \ - map[(addr)>>24].address[(addr) & map[(addr)>>24].mask] - -#define debuggerWriteMemory(addr, value) \ - WRITE32LE(&map[(addr)>>24].address[(addr) & map[(addr)>>24].mask], value) - -#define debuggerWriteHalfWord(addr, value) \ - WRITE16LE(&map[(addr)>>24].address[(addr) & map[(addr)>>24].mask], value) - -#define debuggerWriteByte(addr, value) \ - map[(addr)>>24].address[(addr) & map[(addr)>>24].mask] = (value) - - -#define CHEAT_IS_HEX(a) ( ((a)>='A' && (a) <='F') || ((a) >='0' && (a) <= '9')) - -#define CHEAT_PATCH_ROM_16BIT(a,v) \ - WRITE16LE(((u16 *)&rom[(a) & 0x1ffffff]), v); - -static bool isMultilineWithData(int i) -{ - // we consider it a multiline code if it has more than one line of data - // otherwise, it can still be considered a single code - // (Only CBA codes can be true multilines !!!) - if(i < cheatsNumber && i >= 0) - switch(cheatsList[i].size) { - case INT_8_BIT_WRITE: - case INT_16_BIT_WRITE: - case INT_32_BIT_WRITE: - case GSA_16_BIT_ROM_PATCH: - case GSA_8_BIT_GS_WRITE: - case GSA_16_BIT_GS_WRITE: - case GSA_32_BIT_GS_WRITE: - case CBA_AND: - case CBA_IF_KEYS_PRESSED: - case CBA_IF_TRUE: - case CBA_IF_FALSE: - case GSA_8_BIT_IF_TRUE: - case GSA_32_BIT_IF_TRUE: - case GSA_8_BIT_IF_FALSE: - case GSA_32_BIT_IF_FALSE: - case GSA_8_BIT_FILL: - case GSA_16_BIT_FILL: - case GSA_8_BIT_IF_TRUE2: - case GSA_16_BIT_IF_TRUE2: - case GSA_32_BIT_IF_TRUE2: - case GSA_8_BIT_IF_FALSE2: - case GSA_16_BIT_IF_FALSE2: - case GSA_32_BIT_IF_FALSE2: - case GSA_SLOWDOWN: - case CBA_ADD: - case CBA_OR: - case CBA_LT: - case CBA_GT: - case GSA_8_BIT_POINTER: - case GSA_16_BIT_POINTER: - case GSA_32_BIT_POINTER: - case GSA_8_BIT_ADD: - case GSA_16_BIT_ADD: - case GSA_32_BIT_ADD: - case GSA_8_BIT_IF_LOWER_U: - case GSA_16_BIT_IF_LOWER_U: - case GSA_32_BIT_IF_LOWER_U: - case GSA_8_BIT_IF_HIGHER_U: - case GSA_16_BIT_IF_HIGHER_U: - case GSA_32_BIT_IF_HIGHER_U: - case GSA_8_BIT_IF_AND: - case GSA_16_BIT_IF_AND: - case GSA_32_BIT_IF_AND: - case GSA_8_BIT_IF_LOWER_U2: - case GSA_16_BIT_IF_LOWER_U2: - case GSA_32_BIT_IF_LOWER_U2: - case GSA_8_BIT_IF_HIGHER_U2: - case GSA_16_BIT_IF_HIGHER_U2: - case GSA_32_BIT_IF_HIGHER_U2: - case GSA_8_BIT_IF_AND2: - case GSA_16_BIT_IF_AND2: - case GSA_32_BIT_IF_AND2: - case GSA_ALWAYS: - case GSA_ALWAYS2: - case GSA_8_BIT_IF_LOWER_S: - case GSA_16_BIT_IF_LOWER_S: - case GSA_32_BIT_IF_LOWER_S: - case GSA_8_BIT_IF_HIGHER_S: - case GSA_16_BIT_IF_HIGHER_S: - case GSA_32_BIT_IF_HIGHER_S: - case GSA_8_BIT_IF_LOWER_S2: - case GSA_16_BIT_IF_LOWER_S2: - case GSA_32_BIT_IF_LOWER_S2: - case GSA_8_BIT_IF_HIGHER_S2: - case GSA_16_BIT_IF_HIGHER_S2: - case GSA_32_BIT_IF_HIGHER_S2: - case GSA_16_BIT_WRITE_IOREGS: - case GSA_32_BIT_WRITE_IOREGS: - case GSA_CODES_ON: - case GSA_8_BIT_IF_TRUE3: - case GSA_16_BIT_IF_TRUE3: - case GSA_32_BIT_IF_TRUE3: - case GSA_8_BIT_IF_FALSE3: - case GSA_16_BIT_IF_FALSE3: - case GSA_32_BIT_IF_FALSE3: - case GSA_8_BIT_IF_LOWER_S3: - case GSA_16_BIT_IF_LOWER_S3: - case GSA_32_BIT_IF_LOWER_S3: - case GSA_8_BIT_IF_HIGHER_S3: - case GSA_16_BIT_IF_HIGHER_S3: - case GSA_32_BIT_IF_HIGHER_S3: - case GSA_8_BIT_IF_LOWER_U3: - case GSA_16_BIT_IF_LOWER_U3: - case GSA_32_BIT_IF_LOWER_U3: - case GSA_8_BIT_IF_HIGHER_U3: - case GSA_16_BIT_IF_HIGHER_U3: - case GSA_32_BIT_IF_HIGHER_U3: - case GSA_8_BIT_IF_AND3: - case GSA_16_BIT_IF_AND3: - case GSA_32_BIT_IF_AND3: - case GSA_ALWAYS3: - case GSA_8_BIT_GS_WRITE2: - case GSA_16_BIT_GS_WRITE2: - case GSA_32_BIT_GS_WRITE2: - case GSA_16_BIT_ROM_PATCH2C: - case GSA_16_BIT_ROM_PATCH2D: - case GSA_16_BIT_ROM_PATCH2E: - case GSA_16_BIT_ROM_PATCH2F: - case GSA_8_BIT_SLIDE: - case GSA_16_BIT_SLIDE: - case GSA_32_BIT_SLIDE: - case GSA_GROUP_WRITE: - case GSA_32_BIT_ADD2: - case GSA_32_BIT_SUB2: - case GSA_16_BIT_IF_LOWER_OR_EQ_U: - case GSA_16_BIT_IF_HIGHER_OR_EQ_U: - case GSA_16_BIT_MIF_TRUE: - case GSA_16_BIT_MIF_FALSE: - case GSA_16_BIT_MIF_LOWER_OR_EQ_U: - case GSA_16_BIT_MIF_HIGHER_OR_EQ_U: - return false; - // the codes below have two lines of data - case CBA_SLIDE_CODE: - case CBA_SUPER: - return true; - } - return false; -} - -static int getCodeLength(int num) -{ - if(num >= cheatsNumber || num < 0) - return 1; - - // this is for all the codes that are true multiline - switch(cheatsList[num].size) { - case INT_8_BIT_WRITE: - case INT_16_BIT_WRITE: - case INT_32_BIT_WRITE: - case GSA_16_BIT_ROM_PATCH: - case GSA_8_BIT_GS_WRITE: - case GSA_16_BIT_GS_WRITE: - case GSA_32_BIT_GS_WRITE: - case CBA_AND: - case GSA_8_BIT_FILL: - case GSA_16_BIT_FILL: - case GSA_SLOWDOWN: - case CBA_ADD: - case CBA_OR: - case GSA_8_BIT_POINTER: - case GSA_16_BIT_POINTER: - case GSA_32_BIT_POINTER: - case GSA_8_BIT_ADD: - case GSA_16_BIT_ADD: - case GSA_32_BIT_ADD: - case GSA_CODES_ON: - case GSA_8_BIT_IF_TRUE3: - case GSA_16_BIT_IF_TRUE3: - case GSA_32_BIT_IF_TRUE3: - case GSA_8_BIT_IF_FALSE3: - case GSA_16_BIT_IF_FALSE3: - case GSA_32_BIT_IF_FALSE3: - case GSA_8_BIT_IF_LOWER_S3: - case GSA_16_BIT_IF_LOWER_S3: - case GSA_32_BIT_IF_LOWER_S3: - case GSA_8_BIT_IF_HIGHER_S3: - case GSA_16_BIT_IF_HIGHER_S3: - case GSA_32_BIT_IF_HIGHER_S3: - case GSA_8_BIT_IF_LOWER_U3: - case GSA_16_BIT_IF_LOWER_U3: - case GSA_32_BIT_IF_LOWER_U3: - case GSA_8_BIT_IF_HIGHER_U3: - case GSA_16_BIT_IF_HIGHER_U3: - case GSA_32_BIT_IF_HIGHER_U3: - case GSA_8_BIT_IF_AND3: - case GSA_16_BIT_IF_AND3: - case GSA_32_BIT_IF_AND3: - case GSA_8_BIT_IF_LOWER_U: - case GSA_16_BIT_IF_LOWER_U: - case GSA_32_BIT_IF_LOWER_U: - case GSA_8_BIT_IF_HIGHER_U: - case GSA_16_BIT_IF_HIGHER_U: - case GSA_32_BIT_IF_HIGHER_U: - case GSA_8_BIT_IF_AND: - case GSA_16_BIT_IF_AND: - case GSA_32_BIT_IF_AND: - case GSA_ALWAYS: - case GSA_8_BIT_IF_LOWER_S: - case GSA_16_BIT_IF_LOWER_S: - case GSA_32_BIT_IF_LOWER_S: - case GSA_8_BIT_IF_HIGHER_S: - case GSA_16_BIT_IF_HIGHER_S: - case GSA_32_BIT_IF_HIGHER_S: - case GSA_16_BIT_WRITE_IOREGS: - case GSA_32_BIT_WRITE_IOREGS: - case GSA_8_BIT_GS_WRITE2: - case GSA_16_BIT_GS_WRITE2: - case GSA_32_BIT_GS_WRITE2: - case GSA_16_BIT_ROM_PATCH2C: - case GSA_16_BIT_ROM_PATCH2D: - case GSA_16_BIT_ROM_PATCH2E: - case GSA_16_BIT_ROM_PATCH2F: - case GSA_8_BIT_SLIDE: - case GSA_16_BIT_SLIDE: - case GSA_32_BIT_SLIDE: - case GSA_8_BIT_IF_TRUE: - case GSA_32_BIT_IF_TRUE: - case GSA_8_BIT_IF_FALSE: - case GSA_32_BIT_IF_FALSE: - case CBA_LT: - case CBA_GT: - case CBA_IF_TRUE: - case CBA_IF_FALSE: - case GSA_8_BIT_IF_TRUE2: - case GSA_16_BIT_IF_TRUE2: - case GSA_32_BIT_IF_TRUE2: - case GSA_8_BIT_IF_FALSE2: - case GSA_16_BIT_IF_FALSE2: - case GSA_32_BIT_IF_FALSE2: - case GSA_8_BIT_IF_LOWER_U2: - case GSA_16_BIT_IF_LOWER_U2: - case GSA_32_BIT_IF_LOWER_U2: - case GSA_8_BIT_IF_HIGHER_U2: - case GSA_16_BIT_IF_HIGHER_U2: - case GSA_32_BIT_IF_HIGHER_U2: - case GSA_8_BIT_IF_AND2: - case GSA_16_BIT_IF_AND2: - case GSA_32_BIT_IF_AND2: - case GSA_ALWAYS2: - case GSA_8_BIT_IF_LOWER_S2: - case GSA_16_BIT_IF_LOWER_S2: - case GSA_32_BIT_IF_LOWER_S2: - case GSA_8_BIT_IF_HIGHER_S2: - case GSA_16_BIT_IF_HIGHER_S2: - case GSA_32_BIT_IF_HIGHER_S2: - case GSA_GROUP_WRITE: - case GSA_32_BIT_ADD2: - case GSA_32_BIT_SUB2: - case GSA_16_BIT_IF_LOWER_OR_EQ_U: - case GSA_16_BIT_IF_HIGHER_OR_EQ_U: - case GSA_16_BIT_MIF_TRUE: - case GSA_16_BIT_MIF_FALSE: - case GSA_16_BIT_MIF_LOWER_OR_EQ_U: - case GSA_16_BIT_MIF_HIGHER_OR_EQ_U: - return 1; - case CBA_IF_KEYS_PRESSED: - case CBA_SLIDE_CODE: - return 2; - case CBA_SUPER: - return ((((cheatsList[num].value-1) & 0xFFFF)/3) + 1); - } - return 1; -} - -int cheatsCheckKeys(u32 keys, u32 extended) -{ - bool onoff = true; - int ticks = 1; - int i; - for (i = 0; i<4; i++) - if (rompatch2addr [i] != 0) { - CHEAT_PATCH_ROM_16BIT(rompatch2addr [i],rompatch2oldval [i]); - rompatch2addr [i] = 0; - } - - for (i = 0; i < cheatsNumber; i++) { - if(!cheatsList[i].enabled) { - // make sure we skip other lines in this code - i += getCodeLength(i)-1; - continue; - } - switch(cheatsList[i].size) { - case GSA_CODES_ON: - onoff = true; - break; - case GSA_SLOWDOWN: - // check if button was pressed and released, if so toggle our state - if((cheatsList[i].status & 4) && !(extended & 4)) - cheatsList[i].status ^= 1; - if(extended & 4) - cheatsList[i].status |= 4; - else - cheatsList[i].status &= ~4; - - if(cheatsList[i].status & 1) - ticks += ((cheatsList[i].value & 0xFFFF) * 7); - break; - case GSA_8_BIT_SLIDE: - i++; - if(i < cheatsNumber) { - u32 addr = cheatsList[i-1].value; - u8 value = cheatsList[i].rawaddress; - int vinc = (cheatsList[i].value >> 24) & 255; - int count = (cheatsList[i].value >> 16) & 255; - int ainc = (cheatsList[i].value & 0xffff); - while(count > 0) { - CPUWriteByte(addr, value); - value += vinc; - addr += ainc; - count--; - } - } - i++; - break; - case GSA_16_BIT_SLIDE: - i++; - if(i < cheatsNumber) { - u32 addr = cheatsList[i-1].value; - u16 value = cheatsList[i].rawaddress; - int vinc = (cheatsList[i].value >> 24) & 255; - int count = (cheatsList[i].value >> 16) & 255; - int ainc = (cheatsList[i].value & 0xffff)*2; - while(count > 0) { - CPUWriteHalfWord(addr, value); - value += vinc; - addr += ainc; - count--; - } - } - i++; - break; - case GSA_32_BIT_SLIDE: - i++; - if(i < cheatsNumber) { - u32 addr = cheatsList[i-1].value; - u32 value = cheatsList[i].rawaddress; - int vinc = (cheatsList[i].value >> 24) & 255; - int count = (cheatsList[i].value >> 16) & 255; - int ainc = (cheatsList[i].value & 0xffff)*4; - while(count > 0) { - CPUWriteMemory(addr, value); - value += vinc; - addr += ainc; - count--; - } - } - i++; - break; - case GSA_8_BIT_GS_WRITE2: - i++; - if(i < cheatsNumber) { - if(extended & 4) { - CPUWriteByte(cheatsList[i-1].value, cheatsList[i].address); - } - } - break; - case GSA_16_BIT_GS_WRITE2: - i++; - if(i < cheatsNumber) { - if(extended & 4) { - CPUWriteHalfWord(cheatsList[i-1].value, cheatsList[i].address); - } - } - break; - case GSA_32_BIT_GS_WRITE2: - i++; - if(i < cheatsNumber) { - if(extended & 4) { - CPUWriteMemory(cheatsList[i-1].value, cheatsList[i].address); - } - } - break; - case GSA_16_BIT_ROM_PATCH2C: - i++; - if((i < cheatsNumber) && (cheatsList[i].status & 1) == 0) { - rompatch2addr [0] = ((cheatsList[i-1].value & 0x00FFFFFF) << 1) + 0x8000000; - rompatch2oldval [0] = CPUReadHalfWord(rompatch2addr [0]); - rompatch2val [0] = cheatsList[i].rawaddress & 0xFFFF; - } - i++; - break; - case GSA_16_BIT_ROM_PATCH2D: - i++; - if((i < cheatsNumber) && (cheatsList[i].status & 1) == 0) { - rompatch2addr [1] = ((cheatsList[i-1].value & 0x00FFFFFF) << 1) + 0x8000000; - rompatch2oldval [1] = CPUReadHalfWord(rompatch2addr [1]); - rompatch2val [1] = cheatsList[i].rawaddress & 0xFFFF; - } - i++; - break; - case GSA_16_BIT_ROM_PATCH2E: - i++; - if((i < cheatsNumber) && (cheatsList[i].status & 1) == 0) { - rompatch2addr [2] = ((cheatsList[i-1].value & 0x00FFFFFF) << 1) + 0x8000000; - rompatch2oldval [2] = CPUReadHalfWord(rompatch2addr [2]); - rompatch2val [2] = cheatsList[i].rawaddress & 0xFFFF; - } - i++; - break; - case GSA_16_BIT_ROM_PATCH2F: - i++; - if((i < cheatsNumber) && (cheatsList[i].status & 1) == 0) { - rompatch2addr [3] = ((cheatsList[i-1].value & 0x00FFFFFF) << 1) + 0x8000000; - rompatch2oldval [3] = CPUReadHalfWord(rompatch2addr [3]); - rompatch2val [3] = cheatsList[i].rawaddress & 0xFFFF; - } - i++; - break; - } - if (onoff) { - switch(cheatsList[i].size) { - case INT_8_BIT_WRITE: - CPUWriteByte(cheatsList[i].address, cheatsList[i].value); - break; - case INT_16_BIT_WRITE: - CPUWriteHalfWord(cheatsList[i].address, cheatsList[i].value); - break; - case INT_32_BIT_WRITE: - CPUWriteMemory(cheatsList[i].address, cheatsList[i].value); - break; - case GSA_16_BIT_ROM_PATCH: - if((cheatsList[i].status & 1) == 0) { - if(CPUReadHalfWord(cheatsList[i].address) != cheatsList[i].value) { - cheatsList[i].oldValue = CPUReadHalfWord(cheatsList[i].address); - cheatsList[i].status |= 1; - CHEAT_PATCH_ROM_16BIT(cheatsList[i].address, cheatsList[i].value); - } - } - break; - case GSA_8_BIT_GS_WRITE: - if(extended & 4) { - CPUWriteByte(cheatsList[i].address, cheatsList[i].value); - } - break; - case GSA_16_BIT_GS_WRITE: - if(extended & 4) { - CPUWriteHalfWord(cheatsList[i].address, cheatsList[i].value); - } - break; - case GSA_32_BIT_GS_WRITE: - if(extended & 4) { - CPUWriteMemory(cheatsList[i].address, cheatsList[i].value); - } - break; - case CBA_IF_KEYS_PRESSED: - { - u16 value = cheatsList[i].value; - u32 addr = cheatsList[i].address; - if((addr & 0xF0) == 0x20) { - if((keys & value) == 0) { - i++; - } - } else if((addr & 0xF0) == 0x10) { - if((keys & value) == value) { - i++; - } - } else if((addr & 0xF0) == 0x00) { - if(((~keys) & 0x3FF) == value) { - i++; - } - } - } - break; - case CBA_IF_TRUE: - if(CPUReadHalfWord(cheatsList[i].address) != cheatsList[i].value) { - i++; - } - break; - case CBA_SLIDE_CODE: - { - u32 address = cheatsList[i].address; - u16 value = cheatsList[i].value; - i++; - if(i < cheatsNumber) { - int count = ((cheatsList[i].address - 1) & 0xFFFF); - u16 vinc = (cheatsList[i].address >> 16) & 0xFFFF; - int inc = cheatsList[i].value; - for(int x = 0; x <= count ; x++) { - CPUWriteHalfWord(address, value); - address += inc; - value += vinc; - } - } - } - break; - case CBA_IF_FALSE: - if(CPUReadHalfWord(cheatsList[i].address) == cheatsList[i].value){ - i++; - } - break; - case CBA_AND: - CPUWriteHalfWord(cheatsList[i].address, - CPUReadHalfWord(cheatsList[i].address) & - cheatsList[i].value); - break; - case GSA_8_BIT_IF_TRUE: - if(CPUReadByte(cheatsList[i].address) != cheatsList[i].value) { - i++; - } - break; - case GSA_32_BIT_IF_TRUE: - if(CPUReadMemory(cheatsList[i].address) != cheatsList[i].value) { - i++; - } - break; - case GSA_8_BIT_IF_FALSE: - if(CPUReadByte(cheatsList[i].address) == cheatsList[i].value) { - i++; - } - break; - case GSA_32_BIT_IF_FALSE: - if(CPUReadMemory(cheatsList[i].address) == cheatsList[i].value) { - i++; - } - break; - case GSA_8_BIT_FILL: - { - u32 addr = cheatsList[i].address; - u8 v = cheatsList[i].value & 0xff; - u32 end = addr + (cheatsList[i].value >> 8); - do { - CPUWriteByte(addr, v); - addr++; - } while (addr <= end); - } - break; - case GSA_16_BIT_FILL: - { - u32 addr = cheatsList[i].address; - u16 v = cheatsList[i].value & 0xffff; - u32 end = addr + ((cheatsList[i].value >> 16) << 1); - do { - CPUWriteHalfWord(addr, v); - addr+=2; - } while (addr <= end); - } - break; - case GSA_8_BIT_IF_TRUE2: - if(CPUReadByte(cheatsList[i].address) != cheatsList[i].value) { - i+=2; - } - break; - case GSA_16_BIT_IF_TRUE2: - if(CPUReadHalfWord(cheatsList[i].address) != cheatsList[i].value) { - i+=2; - } - break; - case GSA_32_BIT_IF_TRUE2: - if(CPUReadMemory(cheatsList[i].address) != cheatsList[i].value) { - i+=2; - } - break; - case GSA_8_BIT_IF_FALSE2: - if(CPUReadByte(cheatsList[i].address) == cheatsList[i].value) { - i+=2; - } - break; - case GSA_16_BIT_IF_FALSE2: - if(CPUReadHalfWord(cheatsList[i].address) == cheatsList[i].value) { - i+=2; - } - break; - case GSA_32_BIT_IF_FALSE2: - if(CPUReadMemory(cheatsList[i].address) == cheatsList[i].value) { - i+=2; - } - break; - case CBA_ADD: - if ((cheatsList[i].address & 1) == 0) { - CPUWriteHalfWord(cheatsList[i].address, - CPUReadHalfWord(cheatsList[i].address) + - cheatsList[i].value); - } else { - CPUWriteMemory(cheatsList[i].address & 0x0FFFFFFE, - CPUReadMemory(cheatsList[i].address & 0x0FFFFFFE) + - cheatsList[i].value); - } - break; - case CBA_OR: - CPUWriteHalfWord(cheatsList[i].address, - CPUReadHalfWord(cheatsList[i].address) | - cheatsList[i].value); - break; - case CBA_GT: - if (!(CPUReadHalfWord(cheatsList[i].address) > cheatsList[i].value)){ - i++; - } - break; - case CBA_LT: - if (!(CPUReadHalfWord(cheatsList[i].address) < cheatsList[i].value)){ - i++; - } - break; - case CBA_SUPER: - { - int count = 2*((cheatsList[i].value -1) & 0xFFFF)+1; - u32 address = cheatsList[i].address; - for(int x = 0; x <= count; x++) { - u8 b; - int res = x % 6; - if (res==0) - i++; - if(res < 4) - b = (cheatsList[i].address >> (24-8*res)) & 0xFF; - else - b = (cheatsList[i].value >> (8 - 8*(res-4))) & 0xFF; - CPUWriteByte(address, b); - address++; - } - } - break; - case GSA_8_BIT_POINTER : - if ((CPUReadMemory(cheatsList[i].address)>=0x02000000) && (CPUReadMemory(cheatsList[i].address)<0x02040000) || - (CPUReadMemory(cheatsList[i].address)>=0x03000000) && (CPUReadMemory(cheatsList[i].address)<0x03008000)) - { - CPUWriteByte(CPUReadMemory(cheatsList[i].address)+((cheatsList[i].value & 0xFFFFFF00) >> 8), - cheatsList[i].value & 0xFF); - } - break; - case GSA_16_BIT_POINTER : - if ((CPUReadMemory(cheatsList[i].address)>=0x02000000) && (CPUReadMemory(cheatsList[i].address)<0x02040000) || - (CPUReadMemory(cheatsList[i].address)>=0x03000000) && (CPUReadMemory(cheatsList[i].address)<0x03008000)) - { - CPUWriteHalfWord(CPUReadMemory(cheatsList[i].address)+((cheatsList[i].value & 0xFFFF0000) >> 15), - cheatsList[i].value & 0xFFFF); - } - break; - case GSA_32_BIT_POINTER : - if ((CPUReadMemory(cheatsList[i].address)>=0x02000000) && (CPUReadMemory(cheatsList[i].address)<0x02040000) || - (CPUReadMemory(cheatsList[i].address)>=0x03000000) && (CPUReadMemory(cheatsList[i].address)<0x03008000)) - { - CPUWriteMemory(CPUReadMemory(cheatsList[i].address), - cheatsList[i].value); - } - break; - case GSA_8_BIT_ADD : - CPUWriteByte(cheatsList[i].address, - (cheatsList[i].value & 0xFF) + CPUReadMemory(cheatsList[i].address) & 0xFF); - break; - case GSA_16_BIT_ADD : - CPUWriteHalfWord(cheatsList[i].address, - (cheatsList[i].value & 0xFFFF) + CPUReadMemory(cheatsList[i].address) & 0xFFFF); - break; - case GSA_32_BIT_ADD : - CPUWriteMemory(cheatsList[i].address , - cheatsList[i].value + CPUReadMemory(cheatsList[i].address) & 0xFFFFFFFF); - break; - case GSA_8_BIT_IF_LOWER_U: - if (!(CPUReadByte(cheatsList[i].address) < (cheatsList[i].value & 0xFF))) { - i++; - } - break; - case GSA_16_BIT_IF_LOWER_U: - if (!(CPUReadHalfWord(cheatsList[i].address) < (cheatsList[i].value & 0xFFFF))) { - i++; - } - break; - case GSA_32_BIT_IF_LOWER_U: - if (!(CPUReadMemory(cheatsList[i].address) < cheatsList[i].value)) { - i++; - } - break; - case GSA_8_BIT_IF_HIGHER_U: - if (!(CPUReadByte(cheatsList[i].address) > (cheatsList[i].value & 0xFF))) { - i++; - } - break; - case GSA_16_BIT_IF_HIGHER_U: - if (!(CPUReadHalfWord(cheatsList[i].address) > (cheatsList[i].value & 0xFFFF))) { - i++; - } - break; - case GSA_32_BIT_IF_HIGHER_U: - if (!(CPUReadMemory(cheatsList[i].address) > cheatsList[i].value)) { - i++; - } - break; - case GSA_8_BIT_IF_AND: - if (!(CPUReadByte(cheatsList[i].address) & (cheatsList[i].value & 0xFF))) { - i++; - } - break; - case GSA_16_BIT_IF_AND: - if (!(CPUReadHalfWord(cheatsList[i].address) & (cheatsList[i].value & 0xFFFF))) { - i++; - } - break; - case GSA_32_BIT_IF_AND: - if (!(CPUReadMemory(cheatsList[i].address) & cheatsList[i].value)) { - i++; - } - break; - case GSA_8_BIT_IF_LOWER_U2: - if (!(CPUReadByte(cheatsList[i].address) < (cheatsList[i].value & 0xFF))) { - i+=2; - } - break; - case GSA_16_BIT_IF_LOWER_U2: - if (!(CPUReadHalfWord(cheatsList[i].address) < (cheatsList[i].value & 0xFFFF))) { - i+=2; - } - break; - case GSA_32_BIT_IF_LOWER_U2: - if (!(CPUReadMemory(cheatsList[i].address) < cheatsList[i].value)) { - i+=2; - } - break; - case GSA_8_BIT_IF_HIGHER_U2: - if (!(CPUReadByte(cheatsList[i].address) > (cheatsList[i].value & 0xFF))) { - i+=2; - } - break; - case GSA_16_BIT_IF_HIGHER_U2: - if (!(CPUReadHalfWord(cheatsList[i].address) > (cheatsList[i].value & 0xFFFF))) { - i+=2; - } - break; - case GSA_32_BIT_IF_HIGHER_U2: - if (!(CPUReadMemory(cheatsList[i].address) > cheatsList[i].value)) { - i+=2; - } - break; - case GSA_8_BIT_IF_AND2: - if (!(CPUReadByte(cheatsList[i].address) & (cheatsList[i].value & 0xFF))) { - i+=2; - } - break; - case GSA_16_BIT_IF_AND2: - if (!(CPUReadHalfWord(cheatsList[i].address) & (cheatsList[i].value & 0xFFFF))) { - i+=2; - } - break; - case GSA_32_BIT_IF_AND2: - if (!(CPUReadMemory(cheatsList[i].address) & cheatsList[i].value)) { - i+=2; - } - break; - case GSA_ALWAYS: - i++; - break; - case GSA_ALWAYS2: - i+=2; - break; - case GSA_8_BIT_IF_LOWER_S: - if (!((s8)CPUReadByte(cheatsList[i].address) < ((s8)cheatsList[i].value & 0xFF))) { - i++; - } - break; - case GSA_16_BIT_IF_LOWER_S: - if (!((s16)CPUReadHalfWord(cheatsList[i].address) < ((s16)cheatsList[i].value & 0xFFFF))) { - i++; - } - break; - case GSA_32_BIT_IF_LOWER_S: - if (!((s32)CPUReadMemory(cheatsList[i].address) < (s32)cheatsList[i].value)) { - i++; - } - break; - case GSA_8_BIT_IF_HIGHER_S: - if (!((s8)CPUReadByte(cheatsList[i].address) > ((s8)cheatsList[i].value & 0xFF))) { - i++; - } - break; - case GSA_16_BIT_IF_HIGHER_S: - if (!((s16)CPUReadHalfWord(cheatsList[i].address) > ((s16)cheatsList[i].value & 0xFFFF))) { - i++; - } - break; - case GSA_32_BIT_IF_HIGHER_S: - if (!((s32)CPUReadMemory(cheatsList[i].address) > (s32)cheatsList[i].value)) { - i++; - } - break; - case GSA_8_BIT_IF_LOWER_S2: - if (!((s8)CPUReadByte(cheatsList[i].address) < ((s8)cheatsList[i].value & 0xFF))) { - i+=2; - } - break; - case GSA_16_BIT_IF_LOWER_S2: - if (!((s16)CPUReadHalfWord(cheatsList[i].address) < ((s16)cheatsList[i].value & 0xFFFF))) { - i+=2; - } - break; - case GSA_32_BIT_IF_LOWER_S2: - if (!((s32)CPUReadMemory(cheatsList[i].address) < (s32)cheatsList[i].value)) { - i+=2; - } - break; - case GSA_8_BIT_IF_HIGHER_S2: - if (!((s8)CPUReadByte(cheatsList[i].address) > ((s8)cheatsList[i].value & 0xFF))) { - i+=2; - } - break; - case GSA_16_BIT_IF_HIGHER_S2: - if (!((s16)CPUReadHalfWord(cheatsList[i].address) > ((s16)cheatsList[i].value & 0xFFFF))) { - i+=2; - } - break; - case GSA_32_BIT_IF_HIGHER_S2: - if (!((s32)CPUReadMemory(cheatsList[i].address) > (s32)cheatsList[i].value)) { - i+=2; - } - break; - case GSA_16_BIT_WRITE_IOREGS: - if ((cheatsList[i].address <= 0x3FF) && (cheatsList[i].address != 0x6) && - (cheatsList[i].address != 0x130)) - ioMem[cheatsList[i].address & 0x3FE]=cheatsList[i].value & 0xFFFF; - break; - case GSA_32_BIT_WRITE_IOREGS: - if (cheatsList[i].address<=0x3FF) - { - if (((cheatsList[i].address & 0x3FC) != 0x6) && ((cheatsList[i].address & 0x3FC) != 0x130)) - ioMem[cheatsList[i].address & 0x3FC]= (cheatsList[i].value & 0xFFFF); - if ((((cheatsList[i].address & 0x3FC)+2) != 0x6) && ((cheatsList[i].address & 0x3FC) +2) != 0x130) - ioMem[(cheatsList[i].address & 0x3FC) + 2 ]= ((cheatsList[i].value>>16 ) & 0xFFFF); - } - break; - case GSA_8_BIT_IF_TRUE3: - if(CPUReadByte(cheatsList[i].address) != cheatsList[i].value) { - onoff=false; - } - break; - case GSA_16_BIT_IF_TRUE3: - if(CPUReadHalfWord(cheatsList[i].address) != cheatsList[i].value) { - onoff=false; - } - break; - case GSA_32_BIT_IF_TRUE3: - if(CPUReadMemory(cheatsList[i].address) != cheatsList[i].value) { - onoff=false; - } - break; - case GSA_8_BIT_IF_FALSE3: - if(CPUReadByte(cheatsList[i].address) == cheatsList[i].value) { - onoff=false; - } - break; - case GSA_16_BIT_IF_FALSE3: - if(CPUReadHalfWord(cheatsList[i].address) == cheatsList[i].value) { - onoff=false; - } - break; - case GSA_32_BIT_IF_FALSE3: - if(CPUReadMemory(cheatsList[i].address) == cheatsList[i].value) { - onoff=false; - } - break; - case GSA_8_BIT_IF_LOWER_S3: - if (!((s8)CPUReadByte(cheatsList[i].address) < ((s8)cheatsList[i].value & 0xFF))) { - onoff=false; - } - break; - case GSA_16_BIT_IF_LOWER_S3: - if (!((s16)CPUReadHalfWord(cheatsList[i].address) < ((s16)cheatsList[i].value & 0xFFFF))) { - onoff=false; - } - break; - case GSA_32_BIT_IF_LOWER_S3: - if (!((s32)CPUReadMemory(cheatsList[i].address) < (s32)cheatsList[i].value)) { - onoff=false; - } - break; - case GSA_8_BIT_IF_HIGHER_S3: - if (!((s8)CPUReadByte(cheatsList[i].address) > ((s8)cheatsList[i].value & 0xFF))) { - onoff=false; - } - break; - case GSA_16_BIT_IF_HIGHER_S3: - if (!((s16)CPUReadHalfWord(cheatsList[i].address) > ((s16)cheatsList[i].value & 0xFFFF))) { - onoff=false; - } - break; - case GSA_32_BIT_IF_HIGHER_S3: - if (!((s32)CPUReadMemory(cheatsList[i].address) > (s32)cheatsList[i].value)) { - onoff=false; - } - break; - case GSA_8_BIT_IF_LOWER_U3: - if (!(CPUReadByte(cheatsList[i].address) < (cheatsList[i].value & 0xFF))) { - onoff=false; - } - break; - case GSA_16_BIT_IF_LOWER_U3: - if (!(CPUReadHalfWord(cheatsList[i].address) < (cheatsList[i].value & 0xFFFF))) { - onoff=false; - } - break; - case GSA_32_BIT_IF_LOWER_U3: - if (!(CPUReadMemory(cheatsList[i].address) < cheatsList[i].value)) { - onoff=false; - } - break; - case GSA_8_BIT_IF_HIGHER_U3: - if (!(CPUReadByte(cheatsList[i].address) > (cheatsList[i].value & 0xFF))) { - onoff=false; - } - break; - case GSA_16_BIT_IF_HIGHER_U3: - if (!(CPUReadHalfWord(cheatsList[i].address) > (cheatsList[i].value & 0xFFFF))) { - onoff=false; - } - break; - case GSA_32_BIT_IF_HIGHER_U3: - if (!(CPUReadMemory(cheatsList[i].address) > cheatsList[i].value)) { - onoff=false; - } - break; - case GSA_8_BIT_IF_AND3: - if (!(CPUReadByte(cheatsList[i].address) & (cheatsList[i].value & 0xFF))) { - onoff=false; - } - break; - case GSA_16_BIT_IF_AND3: - if (!(CPUReadHalfWord(cheatsList[i].address) & (cheatsList[i].value & 0xFFFF))) { - onoff=false; - } - break; - case GSA_32_BIT_IF_AND3: - if (!(CPUReadMemory(cheatsList[i].address) & cheatsList[i].value)) { - onoff=false; - } - break; - case GSA_ALWAYS3: - if (!(CPUReadMemory(cheatsList[i].address) & cheatsList[i].value)) { - onoff=false; - } - break; - case GSA_GROUP_WRITE: - { - int count = ((cheatsList[i].address) & 0xFFFE) +1; - u32 value = cheatsList[i].value; - if (count==0) - i++; - else - for (int x = 1; x <= count; x++) { - if ((x % 2) ==0){ - if (x cheatsList[i].value) { - i++; - } - break; - case GSA_16_BIT_IF_HIGHER_OR_EQ_U: - if(CPUReadHalfWord(cheatsList[i].address) < cheatsList[i].value) { - i++; - } - break; - case GSA_16_BIT_MIF_TRUE: - if(CPUReadHalfWord(cheatsList[i].address) != cheatsList[i].value) { - i+=((cheatsList[i].rawaddress >> 0x10) & 0xFF); - } - break; - case GSA_16_BIT_MIF_FALSE: - if(CPUReadHalfWord(cheatsList[i].address) == cheatsList[i].value) { - i+=(cheatsList[i].rawaddress >> 0x10) & 0xFF; - } - break; - case GSA_16_BIT_MIF_LOWER_OR_EQ_U: - if(CPUReadHalfWord(cheatsList[i].address) > cheatsList[i].value) { - i+=(cheatsList[i].rawaddress >> 0x10) & 0xFF; - } - break; - case GSA_16_BIT_MIF_HIGHER_OR_EQ_U: - if(CPUReadHalfWord(cheatsList[i].address) < cheatsList[i].value) { - i+=(cheatsList[i].rawaddress >> 0x10) & 0xFF; - } - break; - } - } - } - for (i = 0; i<4; i++) - if (rompatch2addr [i] != 0) - CHEAT_PATCH_ROM_16BIT(rompatch2addr [i],rompatch2val [i]); - return ticks; -} - -void cheatsAdd(const char *codeStr, - const char *desc, - u32 rawaddress, - u32 address, - u32 value, - int code, - int size) -{ - if(cheatsNumber < 100) { - int x = cheatsNumber; - cheatsList[x].code = code; - cheatsList[x].size = size; - cheatsList[x].rawaddress = rawaddress; - cheatsList[x].address = address; - cheatsList[x].value = value; - strcpy(cheatsList[x].codestring, codeStr); - strcpy(cheatsList[x].desc, desc); - cheatsList[x].enabled = true; - cheatsList[x].status = 0; - - // we only store the old value for this simple codes. ROM patching - // is taken care when it actually patches the ROM - switch(cheatsList[x].size) { - case INT_8_BIT_WRITE: - cheatsList[x].oldValue = CPUReadByte(address); - break; - case INT_16_BIT_WRITE: - cheatsList[x].oldValue = CPUReadHalfWord(address); - break; - case INT_32_BIT_WRITE: - cheatsList[x].oldValue = CPUReadMemory(address); - break; - } - cheatsNumber++; - } -} - -void cheatsDelete(int number, bool restore) -{ - if(number < cheatsNumber && number >= 0) { - int x = number; - - if(restore) { - switch(cheatsList[x].size) { - case INT_8_BIT_WRITE: - CPUWriteByte(cheatsList[x].address, (u8)cheatsList[x].oldValue); - break; - case INT_16_BIT_WRITE: - CPUWriteHalfWord(cheatsList[x].address, (u16)cheatsList[x].oldValue); - break; - case INT_32_BIT_WRITE: - CPUWriteMemory(cheatsList[x].address, cheatsList[x].oldValue); - break; - case GSA_16_BIT_ROM_PATCH: - if(cheatsList[x].status & 1) { - cheatsList[x].status &= ~1; - CHEAT_PATCH_ROM_16BIT(cheatsList[x].address, - cheatsList[x].oldValue); - } - break; - case GSA_16_BIT_ROM_PATCH2C: - case GSA_16_BIT_ROM_PATCH2D: - case GSA_16_BIT_ROM_PATCH2E: - case GSA_16_BIT_ROM_PATCH2F: - if(cheatsList[x].status & 1) { - cheatsList[x].status &= ~1; - } - break; - } - } - if((x+1) < cheatsNumber) { - memcpy(&cheatsList[x], &cheatsList[x+1], sizeof(CheatsData)* - (cheatsNumber-x-1)); - } - cheatsNumber--; - } -} - -void cheatsDeleteAll(bool restore) -{ - for(int i = cheatsNumber-1; i >= 0; i--) { - cheatsDelete(i, restore); - } -} - -void cheatsEnable(int i) -{ - if(i >= 0 && i < cheatsNumber) { - cheatsList[i].enabled = true; - } -} - -void cheatsDisable(int i) -{ - if(i >= 0 && i < cheatsNumber) { - switch(cheatsList[i].size) { - case GSA_16_BIT_ROM_PATCH: - if(cheatsList[i].status & 1) { - cheatsList[i].status &= ~1; - CHEAT_PATCH_ROM_16BIT(cheatsList[i].address, - cheatsList[i].oldValue); - } - break; - case GSA_16_BIT_ROM_PATCH2C: - case GSA_16_BIT_ROM_PATCH2D: - case GSA_16_BIT_ROM_PATCH2E: - case GSA_16_BIT_ROM_PATCH2F: - if(cheatsList[i].status & 1) { - cheatsList[i].status &= ~1; - } - break; - } - cheatsList[i].enabled = false; - } -} - -bool cheatsVerifyCheatCode(const char *code, const char *desc) -{ - int len = strlen(code); - if(len != 11 && len != 13 && len != 17) { - systemMessage(MSG_INVALID_CHEAT_CODE, N_("Invalid cheat code '%s'"), code); - return false; - } - - if(code[8] != ':') { - systemMessage(MSG_INVALID_CHEAT_CODE, N_("Invalid cheat code '%s'"), code); - return false; - } - - int i; - for(i = 0; i < 8; i++) { - if(!CHEAT_IS_HEX(code[i])) { - // wrong cheat - systemMessage(MSG_INVALID_CHEAT_CODE, - N_("Invalid cheat code '%s'"), code); - return false; - } - } - for(i = 9; i < len; i++) { - if(!CHEAT_IS_HEX(code[i])) { - // wrong cheat - systemMessage(MSG_INVALID_CHEAT_CODE, - N_("Invalid cheat code '%s'"), code); - return false; - } - } - - u32 address = 0; - u32 value = 0; - - char buffer[10]; - strncpy(buffer, code, 8); - buffer[8] = 0; - sscanf(buffer, "%x", &address); - - switch(address >> 24) { - case 2: - case 3: - break; - default: - systemMessage(MSG_INVALID_CHEAT_CODE_ADDRESS, - N_("Invalid cheat code address: %08x"), - address); - return false; - } - - strncpy(buffer, &code[9], 8); - sscanf(buffer, "%x", &value); - int type = 0; - if(len == 13) - type = 1; - if(len == 17) - type = 2; - cheatsAdd(code, desc, address, address, value, type, type); - return true; -} - -void cheatsAddCheatCode(const char *code, const char *desc) -{ - cheatsVerifyCheatCode(code, desc); -} - -u16 cheatsGSAGetDeadface(bool v3) -{ - for(int i = cheatsNumber-1; i >= 0; i--) - if ((cheatsList[i].address == 0xDEADFACE) && (cheatsList[i].code == (v3 ? 257 : 256))) - return cheatsList[i].value & 0xFFFF; - return 0; -} - -void cheatsGSAChangeEncryption(u16 value, bool v3) { - int i; - u8 *deadtable1, *deadtable2; - - if (v3) { - deadtable1 = (u8*)(&v3_deadtable1); - deadtable2 = (u8*)(&v3_deadtable2); - for (i = 0; i < 4; i++) - seeds_v3[i] = seed_gen(((value & 0xFF00) >> 8), (value & 0xFF) + i, deadtable1, deadtable2); - } - else { - deadtable1 = (u8*)(&v1_deadtable1); - deadtable2 = (u8*)(&v1_deadtable2); - for (i = 0; i < 4; i++){ - seeds_v1[i] = seed_gen(((value & 0xFF00) >> 8), (value & 0xFF) + i, deadtable1, deadtable2); - } - } -} - -u32 seed_gen(u8 upper, u8 seed, u8 *deadtable1, u8 *deadtable2) { - int i; - u32 newseed = 0; - - for (i = 0; i < 4; i++) - newseed = ((newseed << 8) | ((deadtable1[(i + upper) & 0xFF] + deadtable2[seed]) & 0xFF)); - - return newseed; -} - -void cheatsDecryptGSACode(u32& address, u32& value, bool v3) -{ - u32 rollingseed = 0xC6EF3720; - u32 *seeds = v3 ? seeds_v3 : seeds_v1; - - int bitsleft = 32; - while (bitsleft > 0) { - value -= ((((address << 4) + seeds[2]) ^ (address + rollingseed)) ^ - ((address >> 5) + seeds[3])); - address -= ((((value << 4) + seeds[0]) ^ (value + rollingseed)) ^ - ((value >> 5) + seeds[1])); - rollingseed -= 0x9E3779B9; - bitsleft--; - } -} - -void cheatsAddGSACode(const char *code, const char *desc, bool v3) -{ - if(strlen(code) != 16) { - // wrong cheat - systemMessage(MSG_INVALID_GSA_CODE, - N_("Invalid GSA code. Format is XXXXXXXXYYYYYYYY")); - return; - } - - int i; - for(i = 0; i < 16; i++) { - if(!CHEAT_IS_HEX(code[i])) { - // wrong cheat - systemMessage(MSG_INVALID_GSA_CODE, - N_("Invalid GSA code. Format is XXXXXXXXYYYYYYYY")); - return; - } - } - - char buffer[10]; - strncpy(buffer, code, 8); - buffer[8] = 0; - u32 address; - sscanf(buffer, "%x", &address); - strncpy(buffer, &code[8], 8); - buffer[8] = 0; - u32 value; - sscanf(buffer, "%x", &value); - cheatsGSAChangeEncryption(cheatsGSAGetDeadface (v3), v3); - cheatsDecryptGSACode(address, value, v3); - - if(value == 0x1DC0DE) { - u32 gamecode = READ32LE(((u32 *)&rom[0xac])); - if(gamecode != address) { - char buffer[5]; - *((u32 *)buffer) = address; - buffer[4] = 0; - char buffer2[5]; - *((u32 *)buffer2) = READ32LE(((u32 *)&rom[0xac])); - buffer2[4] = 0; - systemMessage(MSG_GBA_CODE_WARNING, N_("Warning: cheats are for game %s. Current game is %s.\nCodes may not work correctly."), - buffer, buffer2); - } - cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value, v3 ? 257 : 256, - UNKNOWN_CODE); - return; - } - if(isMultilineWithData(cheatsNumber-1)) { - cheatsAdd(code, desc, address, address, value, v3 ? 257 : 256, UNKNOWN_CODE); - return; - } - if(v3) { - int type = ((address >> 25) & 127) | ((address >> 17) & 0x80); - u32 addr = (address & 0x00F00000) << 4 | (address & 0x0003FFFF); - switch(type) { - case 0x00: - if(address == 0) { - type = (value >> 25) & 127; - addr = (value & 0x00F00000) << 4 | (value & 0x0003FFFF); - switch(type) { - case 0x04: - cheatsAdd(code, desc, address, 0, value & 0x00FFFFFF, 257, GSA_SLOWDOWN); - break; - case 0x08: - cheatsAdd(code, desc, address, 0, addr, 257, GSA_8_BIT_GS_WRITE2); - break; - case 0x09: - cheatsAdd(code, desc, address, 0, addr, 257, GSA_16_BIT_GS_WRITE2); - break; - case 0x0a: - cheatsAdd(code, desc, address, 0, addr, 257, GSA_32_BIT_GS_WRITE2); - break; - case 0x0c: - cheatsAdd(code, desc, address, 0, value & 0x00FFFFFF, 257, GSA_16_BIT_ROM_PATCH2C); - break; - case 0x0d: - cheatsAdd(code, desc, address, 0, value & 0x00FFFFFF, 257, GSA_16_BIT_ROM_PATCH2D); - break; - case 0x0e: - cheatsAdd(code, desc, address, 0, value & 0x00FFFFFF, 257, GSA_16_BIT_ROM_PATCH2E); - break; - case 0x0f: - cheatsAdd(code, desc, address, 0, value & 0x00FFFFFF, 257, GSA_16_BIT_ROM_PATCH2F); - break; - case 0x20: - cheatsAdd(code, desc, address, 0, addr, 257, GSA_CODES_ON); - break; - case 0x40: - cheatsAdd(code, desc, address, 0, addr, 257, GSA_8_BIT_SLIDE); - break; - case 0x41: - cheatsAdd(code, desc, address, 0, addr, 257, GSA_16_BIT_SLIDE); - break; - case 0x42: - cheatsAdd(code, desc, address, 0, addr, 257, GSA_32_BIT_SLIDE); - break; - default: - cheatsAdd(code, desc, address, address, value, 257, UNKNOWN_CODE); - break; - } - } else - cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_FILL); - break; - case 0x01: - cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_FILL); - break; - case 0x02: - cheatsAdd(code, desc, address, addr, value, 257, INT_32_BIT_WRITE); - break; - case 0x04: - cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_TRUE); - break; - case 0x05: - cheatsAdd(code, desc, address, addr, value, 257, CBA_IF_TRUE); - break; - case 0x06: - cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_TRUE); - break; - case 0x07: - cheatsAdd(code, desc, address, addr, value, 257, GSA_ALWAYS); - break; - case 0x08: - cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_FALSE); - break; - case 0x09: - cheatsAdd(code, desc, address, addr, value, 257, CBA_IF_FALSE); - break; - case 0x0a: - cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_FALSE); - break; - case 0xc: - cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_LOWER_S); - break; - case 0xd: - cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_LOWER_S); - break; - case 0xe: - cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_LOWER_S); - break; - case 0x10: - cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_HIGHER_S); - break; - case 0x11: - cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_HIGHER_S); - break; - case 0x12: - cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_HIGHER_S); - break; - case 0x14: - cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_LOWER_U); - break; - case 0x15: - cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_LOWER_U); - break; - case 0x16: - cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_LOWER_U); - break; - case 0x18: - cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_HIGHER_U); - break; - case 0x19: - cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_HIGHER_U); - break; - case 0x1A: - cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_HIGHER_U); - break; - case 0x1C: - cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_AND); - break; - case 0x1D: - cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_AND); - break; - case 0x1E: - cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_AND); - break; - case 0x20: - cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_POINTER); - break; - case 0x21: - cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_POINTER); - break; - case 0x22: - cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_POINTER); - break; - case 0x24: - cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_TRUE2); - break; - case 0x25: - cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_TRUE2); - break; - case 0x26: - cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_TRUE2); - break; - case 0x27: - cheatsAdd(code, desc, address, addr, value, 257, GSA_ALWAYS2); - break; - case 0x28: - cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_FALSE2); - break; - case 0x29: - cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_FALSE2); - break; - case 0x2a: - cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_FALSE2); - break; - case 0x2c: - cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_LOWER_S2); - break; - case 0x2d: - cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_LOWER_S2); - break; - case 0x2e: - cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_LOWER_S2); - break; - case 0x30: - cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_HIGHER_S2); - break; - case 0x31: - cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_HIGHER_S2); - break; - case 0x32: - cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_HIGHER_S2); - break; - case 0x34: - cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_LOWER_U2); - break; - case 0x35: - cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_LOWER_U2); - break; - case 0x36: - cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_LOWER_U2); - break; - case 0x38: - cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_HIGHER_U2); - break; - case 0x39: - cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_HIGHER_U2); - break; - case 0x3A: - cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_HIGHER_U2); - break; - case 0x3C: - cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_AND2); - break; - case 0x3D: - cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_AND2); - break; - case 0x3E: - cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_AND2); - break; - case 0x40: - cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_ADD); - break; - case 0x41: - cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_ADD); - break; - case 0x42: - cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_ADD); - break; - case 0x44: - cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_TRUE3); - break; - case 0x45: - cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_TRUE3); - break; - case 0x46: - cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_TRUE3); - break; - case 0x47: - cheatsAdd(code, desc, address, addr, value, 257, GSA_ALWAYS3); - break; - case 0x48: - cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_FALSE3); - break; - case 0x49: - cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_FALSE3); - break; - case 0x4a: - cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_FALSE3); - break; - case 0x4c: - cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_LOWER_S3); - break; - case 0x4d: - cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_LOWER_S3); - break; - case 0x4e: - cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_LOWER_S3); - break; - case 0x50: - cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_HIGHER_S3); - break; - case 0x51: - cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_HIGHER_S3); - break; - case 0x52: - cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_HIGHER_S3); - break; - case 0x54: - cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_LOWER_U3); - break; - case 0x55: - cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_LOWER_U3); - break; - case 0x56: - cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_LOWER_U3); - break; - case 0x58: - cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_HIGHER_U3); - break; - case 0x59: - cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_HIGHER_U3); - break; - case 0x5a: - cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_HIGHER_U3); - break; - case 0x5c: - cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_AND3); - break; - case 0x5d: - cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_AND3); - break; - case 0x5e: - cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_AND3); - break; - case 0x63: - cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_WRITE_IOREGS); - break; - case 0xE3: - cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_WRITE_IOREGS); - break; - default: - cheatsAdd(code, desc, address, address, value, 257, UNKNOWN_CODE); - break; - } - } else { - int type = (address >> 28) & 15; - switch(type) { - case 0: - case 1: - case 2: - cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value, 256, type); - break; - case 3: - switch ((address >> 0x10) & 0xFF){ - case 0x00: - cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value, 256, GSA_GROUP_WRITE); - break; - case 0x10: - cheatsAdd(code, desc, address, value & 0x0FFFFFFF, address & 0xFF, 256, GSA_32_BIT_ADD ); - break; - case 0x20: - cheatsAdd(code, desc, address, value & 0x0FFFFFFF, (~(address & 0xFF)+1), 256, GSA_32_BIT_ADD ); - break; - case 0x30: - cheatsAdd(code, desc, address, value & 0x0FFFFFFF, address & 0xFFFF, 256, GSA_32_BIT_ADD ); - break; - case 0x40: - cheatsAdd(code, desc, address, value & 0x0FFFFFFF, (~(address & 0xFFFF)+1), 256, GSA_32_BIT_ADD ); - break; - case 0x50: - cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value, 256, GSA_32_BIT_ADD2); - break; - case 0x60: - cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value, 256, GSA_32_BIT_SUB2); - break; - default: - // unsupported code - cheatsAdd(code, desc, address, address, value, 256, - UNKNOWN_CODE); - break; - } - break; - case 6: - address <<= 1; - type = (value >> 24) & 0xFF; - if(type == 0x00) { - cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value & 0xFFFF, 256, - GSA_16_BIT_ROM_PATCH); - break; - } - // unsupported code - cheatsAdd(code, desc, address, address, value, 256, - UNKNOWN_CODE); - break; - case 8: - switch((address >> 20) & 15) { - case 1: - cheatsAdd(code, desc, address, address & 0x0F0FFFFF, value, 256, - GSA_8_BIT_GS_WRITE); - break; - case 2: - cheatsAdd(code, desc, address, address & 0x0F0FFFFF, value, 256, - GSA_16_BIT_GS_WRITE); - break; - case 4: - // This code is buggy : the value is always set to 0 ! - cheatsAdd(code, desc, address, address & 0x0F0FFFFF, 0, 256, - GSA_32_BIT_GS_WRITE); - break; - case 15: - cheatsAdd(code, desc, address, 0, value & 0xFFFF, 256, GSA_SLOWDOWN); - break; - default: - // unsupported code - cheatsAdd(code, desc, address, address, value, 256, - UNKNOWN_CODE); - break; - } - break; - case 0x0d: - if(address != 0xDEADFACE) { - switch((value >> 20) & 0xF) { - case 0: - cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value & 0xFFFF, 256, - CBA_IF_TRUE); - break; - case 1: - cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value & 0xFFFF, 256, - CBA_IF_FALSE); - break; - case 2: - cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value & 0xFFFF, 256, - GSA_16_BIT_IF_LOWER_OR_EQ_U); - break; - case 3: - cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value & 0xFFFF, 256, - GSA_16_BIT_IF_HIGHER_OR_EQ_U); - break; - default: - // unsupported code - cheatsAdd(code, desc, address, address, value, 256, - UNKNOWN_CODE); - break; - } - } else - cheatsAdd(code, desc, address, address, value, 256, - UNKNOWN_CODE); - break; - case 0x0e: - switch((value >> 28) & 0xF) { - case 0: - cheatsAdd(code, desc, address, value & 0x0FFFFFFF, address & 0xFFFF, 256, - GSA_16_BIT_MIF_TRUE); - break; - case 1: - cheatsAdd(code, desc, address, value & 0x0FFFFFFF, address & 0xFFFF, 256, - GSA_16_BIT_MIF_FALSE); - break; - case 2: - cheatsAdd(code, desc, address, value & 0x0FFFFFFF, address & 0xFFFF, 256, - GSA_16_BIT_MIF_LOWER_OR_EQ_U); - break; - case 3: - cheatsAdd(code, desc, address, value & 0x0FFFFFFF, address & 0xFFFF, 256, - GSA_16_BIT_MIF_HIGHER_OR_EQ_U); - break; - default: - // unsupported code - cheatsAdd(code, desc, address, address, value, 256, - UNKNOWN_CODE); - break; - } - break; - default: - // unsupported code - cheatsAdd(code, desc, address, address, value, 256, - UNKNOWN_CODE); - break; - } - } -} - -bool cheatsImportGSACodeFile(const char *name, int game, bool v3) -{ - FILE *f = fopen(name, "rb"); - if(!f) - return false; - - int games = 0; - int len = 0; - fseek(f, 0x1e, SEEK_CUR); - fread(&games, 1, 4, f); - bool found = false; - int g = 0; - while(games > 0) { - if(g == game) { - found = true; - break; - } - fread(&len, 1, 4, f); - fseek(f,len,SEEK_CUR); - int codes = 0; - fread(&codes, 1, 4, f); - while(codes > 0) { - fread(&len, 1, 4, f); - fseek(f, len, SEEK_CUR); - fseek(f, 8, SEEK_CUR); - fread(&len, 1, 4, f); - fseek(f, len*12, SEEK_CUR); - codes--; - } - games--; - g++; - } - if(found) { - char desc[256]; - char code[17]; - fread(&len, 1, 4, f); - fseek(f, len, SEEK_CUR); - int codes = 0; - fread(&codes, 1, 4, f); - while(codes > 0) { - fread(&len, 1, 4, f); - fread(desc, 1, len, f); - desc[len] =0; - desc[31] = 0; - fread(&len, 1, 4, f); - fseek(f, len, SEEK_CUR); - fseek(f, 4, SEEK_CUR); - fread(&len, 1, 4, f); - while(len) { - fseek(f, 4, SEEK_CUR); - fread(code, 1, 8, f); - fseek(f, 4, SEEK_CUR); - fread(&code[8], 1, 8, f); - code[16] = 0; - cheatsAddGSACode(code, desc, v3); - len -= 2; - } - codes--; - } - } - fclose(f); - return false; -} - -void cheatsCBAReverseArray(u8 *array, u8 *dest) -{ - dest[0] = array[3]; - dest[1] = array[2]; - dest[2] = array[1]; - dest[3] = array[0]; - dest[4] = array[5]; - dest[5] = array[4]; -} - -void chatsCBAScramble(u8 *array, int count, u8 b) -{ - u8 *x = array + (count >> 3); - u8 *y = array + (b >> 3); - u32 z = *x & (1 << (count & 7)); - u32 x0 = (*x & (~(1 << (count & 7)))); - if (z != 0) - z = 1; - if ((*y & (1 << (b & 7))) != 0) - x0 |= (1 << (count & 7)); - *x = x0; - u32 temp = *y & (~(1 << (b & 7))); - if (z != 0) - temp |= (1 << (b & 7)); - *y = temp; -} - -u32 cheatsCBAGetValue(u8 *array) -{ - return array[0] | array[1]<<8 | array[2] << 16 | array[3]<<24; -} - -u16 cheatsCBAGetData(u8 *array) -{ - return array[4] | array[5]<<8; -} - -void cheatsCBAArrayToValue(u8 *array, u8 *dest) -{ - dest[0] = array[3]; - dest[1] = array[2]; - dest[2] = array[1]; - dest[3] = array[0]; - dest[4] = array[5]; - dest[5] = array[4]; -} - -void cheatsCBAParseSeedCode(u32 address, u32 value, u32 *array) -{ - array[0] = 1; - array[1] = value & 0xFF; - array[2] = (address >> 0x10) & 0xFF; - array[3] = (value >> 8) & 0xFF; - array[4] = (address >> 0x18) & 0x0F; - array[5] = address & 0xFFFF; - array[6] = address; - array[7] = value; -} - -u32 cheatsCBAEncWorker() -{ - u32 x = (cheatsCBATemporaryValue * 0x41c64e6d) + 0x3039; - u32 y = (x * 0x41c64e6d) + 0x3039; - u32 z = x >> 0x10; - x = ((y >> 0x10) & 0x7fff) << 0x0f; - z = (z << 0x1e) | x; - x = (y * 0x41c64e6d) + 0x3039; - cheatsCBATemporaryValue = x; - return z | ((x >> 0x10) & 0x7fff); -} - -#define ROR(v, s) \ - (((v) >> (s)) | (((v) & ((1 << (s))-1)) << (32 - (s)))) - -u32 cheatsCBACalcIndex(u32 x, u32 y) -{ - if(y != 0) { - if(y == 1) - x = 0; - else if(x == y) - x = 0; - if(y < 1) - return x; - else if(x < y) - return x; - u32 x0 = 1; - - while(y < 0x10000000) { - if(y < x) { - y = y << 4; - x0 = x0 << 4; - } else break; - } - - while(y < 0x80000000) { - if(y < x) { - y = y << 1; - x0 = x0 << 1; - } else break; - } - - loop: - u32 z = 0; - if(x >= y) - x -= y; - if(x >= (y >> 1)) { - x -= (y >> 1); - z |= ROR(x0, 1); - } - if(x >= (y >> 2)) { - x -= (y >> 2); - z |= ROR(x0, 2); - } - if(x >= (y >> 3)) { - x -= (y >> 3); - z |= ROR(x0, 3); - } - - u32 temp = x0; - - if(x != 0) { - x0 = x0 >> 4; - if(x0 != 0) { - y = y >> 4; - goto loop; - } - } - - z = z & 0xe0000000; - - if(z != 0) { - if((temp & 7) == 0) - return x; - } else - return x; - - if((z & ROR(temp, 3)) != 0) - x += y >> 3; - if((z & ROR(temp, 2)) != 0) - x += y >> 2; - if((z & ROR(temp, 1)) != 0) - x += y >> 1; - return x; - } else { - } - // should not happen in the current code - return 0; -} - -void cheatsCBAUpdateSeedBuffer(u32 a, u8 *buffer, int count) -{ - int i; - for(i = 0; i < count; i++) - buffer[i] = i; - for(i = 0; (u32)i < a; i++) { - u32 a = cheatsCBACalcIndex(cheatsCBAEncWorker(), count); - u32 b = cheatsCBACalcIndex(cheatsCBAEncWorker(), count); - u32 t = buffer[a]; - buffer[a] = buffer[b]; - buffer[b] = t; - } -} - -void cheatsCBAChangeEncryption(u32 *seed) -{ - int i; - - cheatsCBATemporaryValue = (seed[1] ^ 0x1111); - cheatsCBAUpdateSeedBuffer(0x50, cheatsCBASeedBuffer, 0x30); - cheatsCBATemporaryValue = 0x4efad1c3; - - for(i = 0; (u32)i < seed[4]; i++) { - cheatsCBATemporaryValue = cheatsCBAEncWorker(); - } - cheatsCBASeed[2] = cheatsCBAEncWorker(); - cheatsCBASeed[3] = cheatsCBAEncWorker(); - - cheatsCBATemporaryValue = seed[3] ^ 0xf254; - - for(i = 0; (u32)i < seed[3]; i++) { - cheatsCBATemporaryValue = cheatsCBAEncWorker(); - } - - cheatsCBASeed[0] = cheatsCBAEncWorker(); - cheatsCBASeed[1] = cheatsCBAEncWorker(); - - *((u32 *)&cheatsCBACurrentSeed[0]) = seed[6]; - *((u32 *)&cheatsCBACurrentSeed[4]) = seed[7]; - *((u32 *)&cheatsCBACurrentSeed[8]) = 0; -} - -u16 cheatsCBAGenValue(u32 x, u32 y, u32 z) -{ - y <<= 0x10; - z <<= 0x10; - x <<= 0x18; - u32 x0 = (int)y >> 0x10; - z = (int)z >> 0x10; - x = (int)x >> 0x10; - for(int i = 0; i < 8; i++) { - u32 temp = z ^ x; - if ((int)temp >= 0) { - temp = z << 0x11; - } - else { - temp = z << 0x01; - temp ^= x0; - temp = temp << 0x10; - } - z = (int)temp >> 0x10; - temp = x << 0x11; - x = (int)temp >> 0x10; - } - return z & 0xffff; -} - -void cheatsCBAGenTable() { - for (int i = 0; i < 0x100; i++) { - cheatsCBATable[i] = cheatsCBAGenValue(i, 0x1021, 0); - } - cheatsCBATableGenerated = true; -} - -u16 cheatsCBACalcCRC(u8 *rom, int count) -{ - u32 crc = 0xffffffff; - - if (count & 3) { - // 0x08000EAE - } else { - count = (count >> 2) - 1; - if(count != -1) { - while(count != -1) { - crc = (((crc << 0x08) ^ cheatsCBATable[(((u32)crc << 0x10) >> 0x18) - ^ *rom++]) << 0x10) >> 0x10; - crc = (((crc << 0x08) ^ cheatsCBATable[(((u32)crc << 0x10) >> 0x18) - ^ *rom++]) << 0x10) >> 0x10; - crc = (((crc << 0x08) ^ cheatsCBATable[(((u32)crc << 0x10) >> 0x18) - ^ *rom++]) << 0x10) >> 0x10; - crc = (((crc << 0x08) ^ cheatsCBATable[(((u32)crc << 0x10) >> 0x18) - ^ *rom++]) << 0x10) >> 0x10; - count--; - } - } - } - return crc & 0xffff; -} - -void cheatsCBADecrypt(u8 *decrypt) -{ - u8 buffer[8] = { 0, 0, 0, 0, 0, 0, 0, 0 }; - u8 *array = &buffer[1]; - - cheatsCBAReverseArray(decrypt, array); - - for(int count = 0x2f; count >= 0; count--) { - chatsCBAScramble(array, count, cheatsCBASeedBuffer[count]); - } - cheatsCBAArrayToValue(array, decrypt); - *((u32 *)decrypt) = cheatsCBAGetValue(decrypt) ^ - cheatsCBASeed[0]; - *((u16 *)(decrypt+4)) = (cheatsCBAGetData(decrypt) ^ - cheatsCBASeed[1]) & 0xffff; - - cheatsCBAReverseArray(decrypt, array); - - u32 cs = cheatsCBAGetValue(cheatsCBACurrentSeed); - for(int i = 0; i <= 4; i++) { - array[i] = ((cs >> 8) ^ array[i+1]) ^ array[i] ; - } - - array[5] = (cs >> 8) ^ array[5]; - - for(int j = 5; j >=0; j--) { - array[j] = (cs ^ array[j-1]) ^ array[j]; - } - - cheatsCBAArrayToValue(array, decrypt); - - *((u32 *)decrypt) = cheatsCBAGetValue(decrypt) - ^ cheatsCBASeed[2]; - *((u16 *)(decrypt+4)) = (cheatsCBAGetData(decrypt) - ^ cheatsCBASeed[3]) & 0xffff; -} - -int cheatsCBAGetCount() -{ - int count = 0; - for(int i = 0; i < cheatsNumber; i++) { - if(cheatsList[i].code == 512) - count++; - } - return count; -} - -bool cheatsCBAShouldDecrypt() -{ - for(int i = 0; i < cheatsNumber; i++) { - if(cheatsList[i].code == 512) { - return (cheatsList[i].codestring[0] == '9'); - } - } - return false; -} - -void cheatsAddCBACode(const char *code, const char *desc) -{ - if(strlen(code) != 13) { - // wrong cheat - systemMessage(MSG_INVALID_CBA_CODE, - N_("Invalid CBA code. Format is XXXXXXXX YYYY.")); - return; - } - - int i; - for(i = 0; i < 8; i++) { - if(!CHEAT_IS_HEX(code[i])) { - // wrong cheat - systemMessage(MSG_INVALID_CBA_CODE, - N_("Invalid CBA code. Format is XXXXXXXX YYYY.")); - return; - } - } - - if(code[8] != ' ') { - systemMessage(MSG_INVALID_CBA_CODE, - N_("Invalid CBA code. Format is XXXXXXXX YYYY.")); - return; - } - - for(i = 9; i < 13; i++) { - if(!CHEAT_IS_HEX(code[i])) { - // wrong cheat - systemMessage(MSG_INVALID_CBA_CODE, - N_("Invalid CBA code. Format is XXXXXXXX YYYY.")); - return; - } - } - - char buffer[10]; - strncpy(buffer, code, 8); - buffer[8] = 0; - u32 address; - sscanf(buffer, "%x", &address); - strncpy(buffer, &code[9], 4); - buffer[4] = 0; - u32 value; - sscanf(buffer, "%x", &value); - - u8 array[8] = { - address & 255, - (address >> 8) & 255, - (address >> 16) & 255, - (address >> 24) & 255, - (value & 255), - (value >> 8) & 255, - 0, - 0 - }; - - if(cheatsCBAGetCount() == 0 && - (address >> 28) == 9) { - u32 seed[8]; - cheatsCBAParseSeedCode(address, value, seed); - cheatsCBAChangeEncryption(seed); - cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value, 512, UNKNOWN_CODE); - } else { - if(cheatsCBAShouldDecrypt()) - cheatsCBADecrypt(array); - - address = READ32LE(((u32 *)array)); - value = READ16LE(((u16 *)&array[4])); - - int type = (address >> 28) & 15; - - if(isMultilineWithData(cheatsNumber-1) || (super>0)) { - cheatsAdd(code, desc, address, address, value, 512, UNKNOWN_CODE); - if (super>0) - super-= 1; - return; - } - - switch(type) { - case 0x00: - { - if(!cheatsCBATableGenerated) - cheatsCBAGenTable(); - u32 crc = cheatsCBACalcCRC(rom, 0x10000); - if(crc != address) { - systemMessage(MSG_CBA_CODE_WARNING, - N_("Warning: Codes seem to be for a different game.\nCodes may not work correctly.")); - } - cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value, 512, - UNKNOWN_CODE); - } - break; - case 0x02: - cheatsAdd(code, desc, address, address & 0x0FFFFFFE, value, 512, - CBA_OR); - break; - case 0x03: - cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value, 512, - INT_8_BIT_WRITE); - break; - case 0x04: - cheatsAdd(code, desc, address, address & 0x0FFFFFFE, value, 512, - CBA_SLIDE_CODE); - break; - case 0x05: - cheatsAdd(code, desc, address, address & 0x0FFFFFFE, value, 512, - CBA_SUPER); - super = getCodeLength(cheatsNumber-1); - break; - case 0x06: - cheatsAdd(code, desc, address, address & 0x0FFFFFFE, value, 512, - CBA_AND); - break; - case 0x07: - cheatsAdd(code, desc, address, address & 0x0FFFFFFE, value, 512, - CBA_IF_TRUE); - break; - case 0x08: - cheatsAdd(code, desc, address, address & 0x0FFFFFFE, value, 512, - INT_16_BIT_WRITE); - break; - case 0x0a: - cheatsAdd(code, desc, address, address & 0x0FFFFFFE, value, 512, - CBA_IF_FALSE); - break; - case 0x0b: - cheatsAdd(code, desc, address, address & 0x0FFFFFFE, value, 512, - CBA_GT); - break; - case 0x0c: - cheatsAdd(code, desc, address, address & 0x0FFFFFFE, value, 512, - CBA_LT); - break; - case 0x0d: - if ((address & 0xF0)<0x30) - cheatsAdd(code, desc, address, address & 0xF0, value, 512, - CBA_IF_KEYS_PRESSED); - break; - case 0x0e: - cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value & 0x8000 ? value | 0xFFFF0000 : value, 512, - CBA_ADD); - break; - case 0x0f: - cheatsAdd(code, desc, address, address & 0x0FFFFFFE, value, 512, - GSA_16_BIT_IF_AND); - break; - default: - // unsupported code - cheatsAdd(code, desc, address, address & 0xFFFFFFFF, value, 512, - UNKNOWN_CODE); - break; - } - } -} - -void cheatsSaveGame(gzFile file) -{ - utilWriteInt(file, cheatsNumber); - - utilGzWrite(file, cheatsList, sizeof(cheatsList)); -} - -void cheatsReadGame(gzFile file) -{ - cheatsNumber = 0; - - cheatsNumber = utilReadInt(file); - - utilGzRead(file, cheatsList, sizeof(cheatsList)); - - bool firstCodeBreaker = true; - - for(int i = 0; i < cheatsNumber; i++) { - cheatsList[i].status = 0; - if(!cheatsList[i].codestring[0]) { - switch(cheatsList[i].size) { - case 0: - sprintf(cheatsList[i].codestring, "%08x:%02x", cheatsList[i].address, - cheatsList[i].value); - break; - case 1: - sprintf(cheatsList[i].codestring, "%08x:%04x", cheatsList[i].address, - cheatsList[i].value); - break; - case 2: - sprintf(cheatsList[i].codestring, "%08x:%08x", cheatsList[i].address, - cheatsList[i].value); - break; - } - } - - if(cheatsList[i].enabled) { - cheatsEnable(i); - } - - if(cheatsList[i].code == 512 && firstCodeBreaker) { - firstCodeBreaker = false; - char buffer[10]; - strncpy(buffer, cheatsList[i].codestring, 8); - buffer[8] = 0; - u32 address; - sscanf(buffer, "%x", &address); - if((address >> 28) == 9) { - strncpy(buffer, &cheatsList[i].codestring[9], 4); - buffer[4] = 0; - u32 value; - sscanf(buffer, "%x", &value); - - u32 seed[8]; - cheatsCBAParseSeedCode(address, value, seed); - cheatsCBAChangeEncryption(seed); - } - } - } -} - -void cheatsSaveCheatList(const char *file) -{ - if(cheatsNumber == 0) - return; - FILE *f = fopen(file, "wb"); - if(f == NULL) - return; - int version = 1; - fwrite(&version, 1, sizeof(version), f); - int type = 0; - fwrite(&type, 1, sizeof(type), f); - fwrite(&cheatsNumber, 1, sizeof(cheatsNumber), f); - fwrite(cheatsList, 1, sizeof(cheatsList), f); - fclose(f); -} - -bool cheatsLoadCheatList(const char *file) -{ - cheatsNumber = 0; - - int count = 0; - - FILE *f = fopen(file, "rb"); - - if(f == NULL) - return false; - - int version = 0; - - if(fread(&version, 1, sizeof(version), f) != sizeof(version)) { - fclose(f); - return false; - } - - if(version != 1) { - systemMessage(MSG_UNSUPPORTED_CHEAT_LIST_VERSION, - N_("Unsupported cheat list version %d"), version); - fclose(f); - return false; - } - - int type = 0; - if(fread(&type, 1, sizeof(type), f) != sizeof(type)) { - fclose(f); - return false; - } - - if(type != 0) { - systemMessage(MSG_UNSUPPORTED_CHEAT_LIST_TYPE, - N_("Unsupported cheat list type %d"), type); - fclose(f); - return false; - } - - if(fread(&count, 1, sizeof(count), f) != sizeof(count)) { - fclose(f); - return false; - } - - if(fread(cheatsList, 1, sizeof(cheatsList), f) != sizeof(cheatsList)) { - fclose(f); - return false; - } - - bool firstCodeBreaker = true; - - for(int i = 0; i < count; i++) { - cheatsList[i].status = 0; // remove old status as it is not used - if(!cheatsList[i].codestring[0]) { - switch(cheatsList[i].size) { - case 0: - sprintf(cheatsList[i].codestring, "%08x:%02x", cheatsList[i].address, - cheatsList[i].value); - break; - case 1: - sprintf(cheatsList[i].codestring, "%08x:%04x", cheatsList[i].address, - cheatsList[i].value); - break; - case 2: - sprintf(cheatsList[i].codestring, "%08x:%08x", cheatsList[i].address, - cheatsList[i].value); - break; - } - } - - if(cheatsList[i].code == 512 && firstCodeBreaker) { - firstCodeBreaker = false; - char buffer[10]; - strncpy(buffer, cheatsList[i].codestring, 8); - buffer[8] = 0; - u32 address; - sscanf(buffer, "%x", &address); - if((address >> 28) == 9) { - strncpy(buffer, &cheatsList[i].codestring[9], 4); - buffer[4] = 0; - u32 value; - sscanf(buffer, "%x", &value); - - u32 seed[8]; - cheatsCBAParseSeedCode(address, value, seed); - cheatsCBAChangeEncryption(seed); - } - } - } - cheatsNumber = count; - fclose(f); - return true; -} - -extern int cpuNextEvent; - -extern void debuggerBreakOnWrite(u32 , u32, u32, int, int); - -static u8 cheatsGetType(u32 address) -{ - switch(address >> 24) { - case 2: - return freezeWorkRAM[address & 0x3FFFF]; - case 3: - return freezeInternalRAM[address & 0x7FFF]; - } - return 0; -} - -void cheatsWriteMemory(u32 address, u32 value) -{ -#ifdef BKPT_SUPPORT -#ifdef SDL - if(cheatsNumber == 0) { - int type = cheatsGetType(address); - u32 oldValue = debuggerReadMemory(address); - if(type == 1 || (type == 2 && oldValue != value)) { - debuggerBreakOnWrite(address, oldValue, value, 2, type); - cpuNextEvent = 0; - } - debuggerWriteMemory(address, value); - } -#endif -#endif -} - -void cheatsWriteHalfWord(u32 address, u16 value) -{ -#ifdef BKPT_SUPPORT -#ifdef SDL - if(cheatsNumber == 0) { - int type = cheatsGetType(address); - u16 oldValue = debuggerReadHalfWord(address); - if(type == 1 || (type == 2 && oldValue != value)) { - debuggerBreakOnWrite(address, oldValue, value, 1, type); - cpuNextEvent = 0; - } - debuggerWriteHalfWord(address, value); - } -#endif -#endif -} - -#if defined BKPT_SUPPORT && defined SDL -void cheatsWriteByte(u32 address, u8 value) -#else -void cheatsWriteByte(u32, u8) -#endif -{ -#ifdef BKPT_SUPPORT -#ifdef SDL - if(cheatsNumber == 0) { - int type = cheatsGetType(address); - u8 oldValue = debuggerReadByte(address); - if(type == 1 || (type == 2 && oldValue != value)) { - debuggerBreakOnWrite(address, oldValue, value, 0, type); - cpuNextEvent = 0; - } - debuggerWriteByte(address, value); - } -#endif -#endif -} \ No newline at end of file +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2005-2006 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#include +#include +#include +#include + +#include "GBA.h" +#include "GBAinline.h" +#include "Cheats.h" +#include "Globals.h" +#include "NLS.h" +#include "Util.h" + +/** + * Gameshark code types: (based on AR v1.0) + * + * NNNNNNNN 001DC0DE - ID code for the game (game 4 character name) from ROM + * DEADFACE XXXXXXXX - changes decryption seeds // Not supported by VBA. + * 0AAAAAAA 000000YY - 8-bit constant write + * 1AAAAAAA 0000YYYY - 16-bit constant write + * 2AAAAAAA YYYYYYYY - 32-bit constant write + * 30XXAAAA YYYYYYYY - 32bit Group Write, 8/16/32bit Sub/Add (depending on the XX value). + * 6AAAAAAA Z000YYYY - 16-bit ROM Patch (address >> 1). Z selects the Rom Patching register. + * - AR v1/2 hardware only supports Z=0. + * - AR v3 hardware should support Z=0,1,2 or 3. + * 8A1AAAAA 000000YY - 8-bit button write + * 8A2AAAAA 0000YYYY - 16-bit button write + * 8A4AAAAA YYYYYYYY - 32-bit button write // BUGGY ! Only writes 00000000 on the AR v1.0. + * 80F00000 0000YYYY - button slow motion + * DAAAAAAA 00Z0YYYY - Z = 0 : if 16-bit value at address != YYYY skip next line + * - Z = 1 : if 16-bit value at address == YYYY skip next line + * - Z = 2 : if 16-bit value at address > YYYY (Unsigned) skip next line + * - Z = 3 : if 16-bit value at address < YYYY (Unsigned) skip next line + * E0CCYYYY ZAAAAAAA - Z = 0 : if 16-bit value at address != YYYY skip CC lines + * - Z = 1 : if 16-bit value at address == YYYY skip CC lines + * - Z = 2 : if 16-bit value at address > YYYY (Unsigned) skip CC lines + * - Z = 3 : if 16-bit value at address < YYYY (Unsigned) skip CC lines + * FAAAAAAA 0000YYYY - Master code function + * + * + * + * CodeBreaker codes types: (based on the CBA clone "Cheatcode S" v1.1) + * + * 0000AAAA 000Y - Game CRC (Y are flags: 8 - CRC, 2 - DI) + * 1AAAAAAA YYYY - Master Code function (store address at ((YYYY << 0x16) + * + 0x08000100)) + * 2AAAAAAA YYYY - 16-bit or + * 3AAAAAAA YYYY - 8-bit constant write + * 4AAAAAAA YYYY - Slide code + * XXXXCCCC IIII (C is count and I is address increment, X is value incr.) + * 5AAAAAAA CCCC - Super code (Write bytes to address, 2*CCCC is count) + * BBBBBBBB BBBB + * 6AAAAAAA YYYY - 16-bit and + * 7AAAAAAA YYYY - if address contains 16-bit value enable next code + * 8AAAAAAA YYYY - 16-bit constant write + * 9AAAAAAA YYYY - change decryption (when first code only?) + * AAAAAAAA YYYY - if address does not contain 16-bit value enable next code + * BAAAAAAA YYYY - if 16-bit value at address <= YYYY skip next code + * CAAAAAAA YYYY - if 16-bit value at address >= YYYY skip next code + * D00000X0 YYYY - if button keys ... enable next code (else skip next code) + * EAAAAAAA YYYY - increase 16/32bit value stored in address + * FAAAAAAA YYYY - if 16-bit value at address AND YYYY = 0 then skip next code + **/ + +#define UNKNOWN_CODE -1 +#define INT_8_BIT_WRITE 0 +#define INT_16_BIT_WRITE 1 +#define INT_32_BIT_WRITE 2 +#define GSA_16_BIT_ROM_PATCH 3 +#define GSA_8_BIT_GS_WRITE 4 +#define GSA_16_BIT_GS_WRITE 5 +#define GSA_32_BIT_GS_WRITE 6 +#define CBA_IF_KEYS_PRESSED 7 +#define CBA_IF_TRUE 8 +#define CBA_SLIDE_CODE 9 +#define CBA_IF_FALSE 10 +#define CBA_AND 11 +#define GSA_8_BIT_GS_WRITE2 12 +#define GSA_16_BIT_GS_WRITE2 13 +#define GSA_32_BIT_GS_WRITE2 14 +#define GSA_16_BIT_ROM_PATCH2C 15 +#define GSA_8_BIT_SLIDE 16 +#define GSA_16_BIT_SLIDE 17 +#define GSA_32_BIT_SLIDE 18 +#define GSA_8_BIT_IF_TRUE 19 +#define GSA_32_BIT_IF_TRUE 20 +#define GSA_8_BIT_IF_FALSE 21 +#define GSA_32_BIT_IF_FALSE 22 +#define GSA_8_BIT_FILL 23 +#define GSA_16_BIT_FILL 24 +#define GSA_8_BIT_IF_TRUE2 25 +#define GSA_16_BIT_IF_TRUE2 26 +#define GSA_32_BIT_IF_TRUE2 27 +#define GSA_8_BIT_IF_FALSE2 28 +#define GSA_16_BIT_IF_FALSE2 29 +#define GSA_32_BIT_IF_FALSE2 30 +#define GSA_SLOWDOWN 31 +#define CBA_ADD 32 +#define CBA_OR 33 +#define CBA_LT 34 +#define CBA_GT 35 +#define CBA_SUPER 36 +#define GSA_8_BIT_POINTER 37 +#define GSA_16_BIT_POINTER 38 +#define GSA_32_BIT_POINTER 39 +#define GSA_8_BIT_ADD 40 +#define GSA_16_BIT_ADD 41 +#define GSA_32_BIT_ADD 42 +#define GSA_8_BIT_IF_LOWER_U 43 +#define GSA_16_BIT_IF_LOWER_U 44 +#define GSA_32_BIT_IF_LOWER_U 45 +#define GSA_8_BIT_IF_HIGHER_U 46 +#define GSA_16_BIT_IF_HIGHER_U 47 +#define GSA_32_BIT_IF_HIGHER_U 48 +#define GSA_8_BIT_IF_AND 49 +#define GSA_16_BIT_IF_AND 50 +#define GSA_32_BIT_IF_AND 51 +#define GSA_8_BIT_IF_LOWER_U2 52 +#define GSA_16_BIT_IF_LOWER_U2 53 +#define GSA_32_BIT_IF_LOWER_U2 54 +#define GSA_8_BIT_IF_HIGHER_U2 55 +#define GSA_16_BIT_IF_HIGHER_U2 56 +#define GSA_32_BIT_IF_HIGHER_U2 57 +#define GSA_8_BIT_IF_AND2 58 +#define GSA_16_BIT_IF_AND2 59 +#define GSA_32_BIT_IF_AND2 60 +#define GSA_ALWAYS 61 +#define GSA_ALWAYS2 62 +#define GSA_8_BIT_IF_LOWER_S 63 +#define GSA_16_BIT_IF_LOWER_S 64 +#define GSA_32_BIT_IF_LOWER_S 65 +#define GSA_8_BIT_IF_HIGHER_S 66 +#define GSA_16_BIT_IF_HIGHER_S 67 +#define GSA_32_BIT_IF_HIGHER_S 68 +#define GSA_8_BIT_IF_LOWER_S2 69 +#define GSA_16_BIT_IF_LOWER_S2 70 +#define GSA_32_BIT_IF_LOWER_S2 71 +#define GSA_8_BIT_IF_HIGHER_S2 72 +#define GSA_16_BIT_IF_HIGHER_S2 73 +#define GSA_32_BIT_IF_HIGHER_S2 74 +#define GSA_16_BIT_WRITE_IOREGS 75 +#define GSA_32_BIT_WRITE_IOREGS 76 +#define GSA_CODES_ON 77 +#define GSA_8_BIT_IF_TRUE3 78 +#define GSA_16_BIT_IF_TRUE3 79 +#define GSA_32_BIT_IF_TRUE3 80 +#define GSA_8_BIT_IF_FALSE3 81 +#define GSA_16_BIT_IF_FALSE3 82 +#define GSA_32_BIT_IF_FALSE3 83 +#define GSA_8_BIT_IF_LOWER_S3 84 +#define GSA_16_BIT_IF_LOWER_S3 85 +#define GSA_32_BIT_IF_LOWER_S3 86 +#define GSA_8_BIT_IF_HIGHER_S3 87 +#define GSA_16_BIT_IF_HIGHER_S3 88 +#define GSA_32_BIT_IF_HIGHER_S3 89 +#define GSA_8_BIT_IF_LOWER_U3 90 +#define GSA_16_BIT_IF_LOWER_U3 91 +#define GSA_32_BIT_IF_LOWER_U3 92 +#define GSA_8_BIT_IF_HIGHER_U3 93 +#define GSA_16_BIT_IF_HIGHER_U3 94 +#define GSA_32_BIT_IF_HIGHER_U3 95 +#define GSA_8_BIT_IF_AND3 96 +#define GSA_16_BIT_IF_AND3 97 +#define GSA_32_BIT_IF_AND3 98 +#define GSA_ALWAYS3 99 +#define GSA_16_BIT_ROM_PATCH2D 100 +#define GSA_16_BIT_ROM_PATCH2E 101 +#define GSA_16_BIT_ROM_PATCH2F 102 +#define GSA_GROUP_WRITE 103 +#define GSA_32_BIT_ADD2 104 +#define GSA_32_BIT_SUB2 105 +#define GSA_16_BIT_IF_LOWER_OR_EQ_U 106 +#define GSA_16_BIT_IF_HIGHER_OR_EQ_U 107 +#define GSA_16_BIT_MIF_TRUE 108 +#define GSA_16_BIT_MIF_FALSE 109 +#define GSA_16_BIT_MIF_LOWER_OR_EQ_U 110 +#define GSA_16_BIT_MIF_HIGHER_OR_EQ_U 111 +#define MASTER_CODE 112 +#define CHEATS_16_BIT_WRITE 114 +#define CHEATS_32_BIT_WRITE 115 + +CheatsData cheatsList[100]; +int cheatsNumber = 0; +u32 rompatch2addr [4]; +u16 rompatch2val [4]; +u16 rompatch2oldval [4]; + +u8 cheatsCBASeedBuffer[0x30]; +u32 cheatsCBASeed[4]; +u32 cheatsCBATemporaryValue = 0; +u16 cheatsCBATable[256]; +bool cheatsCBATableGenerated = false; +u16 super = 0; +extern u32 mastercode; + +u8 cheatsCBACurrentSeed[12] = { + 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00 +}; + +u32 seeds_v1[4]; +u32 seeds_v3[4]; + +u32 seed_gen(u8 upper, u8 seed, u8 *deadtable1, u8 *deadtable2); + +//seed tables for AR v1 +u8 v1_deadtable1[256] = { + 0x31, 0x1C, 0x23, 0xE5, 0x89, 0x8E, 0xA1, 0x37, 0x74, 0x6D, 0x67, 0xFC, 0x1F, 0xC0, 0xB1, 0x94, + 0x3B, 0x05, 0x56, 0x86, 0x00, 0x24, 0xF0, 0x17, 0x72, 0xA2, 0x3D, 0x1B, 0xE3, 0x17, 0xC5, 0x0B, + 0xB9, 0xE2, 0xBD, 0x58, 0x71, 0x1B, 0x2C, 0xFF, 0xE4, 0xC9, 0x4C, 0x5E, 0xC9, 0x55, 0x33, 0x45, + 0x7C, 0x3F, 0xB2, 0x51, 0xFE, 0x10, 0x7E, 0x75, 0x3C, 0x90, 0x8D, 0xDA, 0x94, 0x38, 0xC3, 0xE9, + 0x95, 0xEA, 0xCE, 0xA6, 0x06, 0xE0, 0x4F, 0x3F, 0x2A, 0xE3, 0x3A, 0xE4, 0x43, 0xBD, 0x7F, 0xDA, + 0x55, 0xF0, 0xEA, 0xCB, 0x2C, 0xA8, 0x47, 0x61, 0xA0, 0xEF, 0xCB, 0x13, 0x18, 0x20, 0xAF, 0x3E, + 0x4D, 0x9E, 0x1E, 0x77, 0x51, 0xC5, 0x51, 0x20, 0xCF, 0x21, 0xF9, 0x39, 0x94, 0xDE, 0xDD, 0x79, + 0x4E, 0x80, 0xC4, 0x9D, 0x94, 0xD5, 0x95, 0x01, 0x27, 0x27, 0xBD, 0x6D, 0x78, 0xB5, 0xD1, 0x31, + 0x6A, 0x65, 0x74, 0x74, 0x58, 0xB3, 0x7C, 0xC9, 0x5A, 0xED, 0x50, 0x03, 0xC4, 0xA2, 0x94, 0x4B, + 0xF0, 0x58, 0x09, 0x6F, 0x3E, 0x7D, 0xAE, 0x7D, 0x58, 0xA0, 0x2C, 0x91, 0xBB, 0xE1, 0x70, 0xEB, + 0x73, 0xA6, 0x9A, 0x44, 0x25, 0x90, 0x16, 0x62, 0x53, 0xAE, 0x08, 0xEB, 0xDC, 0xF0, 0xEE, 0x77, + 0xC2, 0xDE, 0x81, 0xE8, 0x30, 0x89, 0xDB, 0xFE, 0xBC, 0xC2, 0xDF, 0x26, 0xE9, 0x8B, 0xD6, 0x93, + 0xF0, 0xCB, 0x56, 0x90, 0xC0, 0x46, 0x68, 0x15, 0x43, 0xCB, 0xE9, 0x98, 0xE3, 0xAF, 0x31, 0x25, + 0x4D, 0x7B, 0xF3, 0xB1, 0x74, 0xE2, 0x64, 0xAC, 0xD9, 0xF6, 0xA0, 0xD5, 0x0B, 0x9B, 0x49, 0x52, + 0x69, 0x3B, 0x71, 0x00, 0x2F, 0xBB, 0xBA, 0x08, 0xB1, 0xAE, 0xBB, 0xB3, 0xE1, 0xC9, 0xA6, 0x7F, + 0x17, 0x97, 0x28, 0x72, 0x12, 0x6E, 0x91, 0xAE, 0x3A, 0xA2, 0x35, 0x46, 0x27, 0xF8, 0x12, 0x50 +}; +u8 v1_deadtable2[256] = { + 0xD8, 0x65, 0x04, 0xC2, 0x65, 0xD5, 0xB0, 0x0C, 0xDF, 0x9D, 0xF0, 0xC3, 0x9A, 0x17, 0xC9, 0xA6, + 0xE1, 0xAC, 0x0D, 0x14, 0x2F, 0x3C, 0x2C, 0x87, 0xA2, 0xBF, 0x4D, 0x5F, 0xAC, 0x2D, 0x9D, 0xE1, + 0x0C, 0x9C, 0xE7, 0x7F, 0xFC, 0xA8, 0x66, 0x59, 0xAC, 0x18, 0xD7, 0x05, 0xF0, 0xBF, 0xD1, 0x8B, + 0x35, 0x9F, 0x59, 0xB4, 0xBA, 0x55, 0xB2, 0x85, 0xFD, 0xB1, 0x72, 0x06, 0x73, 0xA4, 0xDB, 0x48, + 0x7B, 0x5F, 0x67, 0xA5, 0x95, 0xB9, 0xA5, 0x4A, 0xCF, 0xD1, 0x44, 0xF3, 0x81, 0xF5, 0x6D, 0xF6, + 0x3A, 0xC3, 0x57, 0x83, 0xFA, 0x8E, 0x15, 0x2A, 0xA2, 0x04, 0xB2, 0x9D, 0xA8, 0x0D, 0x7F, 0xB8, + 0x0F, 0xF6, 0xAC, 0xBE, 0x97, 0xCE, 0x16, 0xE6, 0x31, 0x10, 0x60, 0x16, 0xB5, 0x83, 0x45, 0xEE, + 0xD7, 0x5F, 0x2C, 0x08, 0x58, 0xB1, 0xFD, 0x7E, 0x79, 0x00, 0x34, 0xAD, 0xB5, 0x31, 0x34, 0x39, + 0xAF, 0xA8, 0xDD, 0x52, 0x6A, 0xB0, 0x60, 0x35, 0xB8, 0x1D, 0x52, 0xF5, 0xF5, 0x30, 0x00, 0x7B, + 0xF4, 0xBA, 0x03, 0xCB, 0x3A, 0x84, 0x14, 0x8A, 0x6A, 0xEF, 0x21, 0xBD, 0x01, 0xD8, 0xA0, 0xD4, + 0x43, 0xBE, 0x23, 0xE7, 0x76, 0x27, 0x2C, 0x3F, 0x4D, 0x3F, 0x43, 0x18, 0xA7, 0xC3, 0x47, 0xA5, + 0x7A, 0x1D, 0x02, 0x55, 0x09, 0xD1, 0xFF, 0x55, 0x5E, 0x17, 0xA0, 0x56, 0xF4, 0xC9, 0x6B, 0x90, + 0xB4, 0x80, 0xA5, 0x07, 0x22, 0xFB, 0x22, 0x0D, 0xD9, 0xC0, 0x5B, 0x08, 0x35, 0x05, 0xC1, 0x75, + 0x4F, 0xD0, 0x51, 0x2D, 0x2E, 0x5E, 0x69, 0xE7, 0x3B, 0xC2, 0xDA, 0xFF, 0xF6, 0xCE, 0x3E, 0x76, + 0xE8, 0x36, 0x8C, 0x39, 0xD8, 0xF3, 0xE9, 0xA6, 0x42, 0xE6, 0xC1, 0x4C, 0x05, 0xBE, 0x17, 0xF2, + 0x5C, 0x1B, 0x19, 0xDB, 0x0F, 0xF3, 0xF8, 0x49, 0xEB, 0x36, 0xF6, 0x40, 0x6F, 0xAD, 0xC1, 0x8C +}; + +//seed tables for AR v3 +u8 v3_deadtable1[256] = { + 0xD0, 0xFF, 0xBA, 0xE5, 0xC1, 0xC7, 0xDB, 0x5B, 0x16, 0xE3, 0x6E, 0x26, 0x62, 0x31, 0x2E, 0x2A, + 0xD1, 0xBB, 0x4A, 0xE6, 0xAE, 0x2F, 0x0A, 0x90, 0x29, 0x90, 0xB6, 0x67, 0x58, 0x2A, 0xB4, 0x45, + 0x7B, 0xCB, 0xF0, 0x73, 0x84, 0x30, 0x81, 0xC2, 0xD7, 0xBE, 0x89, 0xD7, 0x4E, 0x73, 0x5C, 0xC7, + 0x80, 0x1B, 0xE5, 0xE4, 0x43, 0xC7, 0x46, 0xD6, 0x6F, 0x7B, 0xBF, 0xED, 0xE5, 0x27, 0xD1, 0xB5, + 0xD0, 0xD8, 0xA3, 0xCB, 0x2B, 0x30, 0xA4, 0xF0, 0x84, 0x14, 0x72, 0x5C, 0xFF, 0xA4, 0xFB, 0x54, + 0x9D, 0x70, 0xE2, 0xFF, 0xBE, 0xE8, 0x24, 0x76, 0xE5, 0x15, 0xFB, 0x1A, 0xBC, 0x87, 0x02, 0x2A, + 0x58, 0x8F, 0x9A, 0x95, 0xBD, 0xAE, 0x8D, 0x0C, 0xA5, 0x4C, 0xF2, 0x5C, 0x7D, 0xAD, 0x51, 0xFB, + 0xB1, 0x22, 0x07, 0xE0, 0x29, 0x7C, 0xEB, 0x98, 0x14, 0xC6, 0x31, 0x97, 0xE4, 0x34, 0x8F, 0xCC, + 0x99, 0x56, 0x9F, 0x78, 0x43, 0x91, 0x85, 0x3F, 0xC2, 0xD0, 0xD1, 0x80, 0xD1, 0x77, 0xA7, 0xE2, + 0x43, 0x99, 0x1D, 0x2F, 0x8B, 0x6A, 0xE4, 0x66, 0x82, 0xF7, 0x2B, 0x0B, 0x65, 0x14, 0xC0, 0xC2, + 0x1D, 0x96, 0x78, 0x1C, 0xC4, 0xC3, 0xD2, 0xB1, 0x64, 0x07, 0xD7, 0x6F, 0x02, 0xE9, 0x44, 0x31, + 0xDB, 0x3C, 0xEB, 0x93, 0xED, 0x9A, 0x57, 0x05, 0xB9, 0x0E, 0xAF, 0x1F, 0x48, 0x11, 0xDC, 0x35, + 0x6C, 0xB8, 0xEE, 0x2A, 0x48, 0x2B, 0xBC, 0x89, 0x12, 0x59, 0xCB, 0xD1, 0x18, 0xEA, 0x72, 0x11, + 0x01, 0x75, 0x3B, 0xB5, 0x56, 0xF4, 0x8B, 0xA0, 0x41, 0x75, 0x86, 0x7B, 0x94, 0x12, 0x2D, 0x4C, + 0x0C, 0x22, 0xC9, 0x4A, 0xD8, 0xB1, 0x8D, 0xF0, 0x55, 0x2E, 0x77, 0x50, 0x1C, 0x64, 0x77, 0xAA, + 0x3E, 0xAC, 0xD3, 0x3D, 0xCE, 0x60, 0xCA, 0x5D, 0xA0, 0x92, 0x78, 0xC6, 0x51, 0xFE, 0xF9, 0x30 +}; +u8 v3_deadtable2[256] = { + 0xAA, 0xAF, 0xF0, 0x72, 0x90, 0xF7, 0x71, 0x27, 0x06, 0x11, 0xEB, 0x9C, 0x37, 0x12, 0x72, 0xAA, + 0x65, 0xBC, 0x0D, 0x4A, 0x76, 0xF6, 0x5C, 0xAA, 0xB0, 0x7A, 0x7D, 0x81, 0xC1, 0xCE, 0x2F, 0x9F, + 0x02, 0x75, 0x38, 0xC8, 0xFC, 0x66, 0x05, 0xC2, 0x2C, 0xBD, 0x91, 0xAD, 0x03, 0xB1, 0x88, 0x93, + 0x31, 0xC6, 0xAB, 0x40, 0x23, 0x43, 0x76, 0x54, 0xCA, 0xE7, 0x00, 0x96, 0x9F, 0xD8, 0x24, 0x8B, + 0xE4, 0xDC, 0xDE, 0x48, 0x2C, 0xCB, 0xF7, 0x84, 0x1D, 0x45, 0xE5, 0xF1, 0x75, 0xA0, 0xED, 0xCD, + 0x4B, 0x24, 0x8A, 0xB3, 0x98, 0x7B, 0x12, 0xB8, 0xF5, 0x63, 0x97, 0xB3, 0xA6, 0xA6, 0x0B, 0xDC, + 0xD8, 0x4C, 0xA8, 0x99, 0x27, 0x0F, 0x8F, 0x94, 0x63, 0x0F, 0xB0, 0x11, 0x94, 0xC7, 0xE9, 0x7F, + 0x3B, 0x40, 0x72, 0x4C, 0xDB, 0x84, 0x78, 0xFE, 0xB8, 0x56, 0x08, 0x80, 0xDF, 0x20, 0x2F, 0xB9, + 0x66, 0x2D, 0x60, 0x63, 0xF5, 0x18, 0x15, 0x1B, 0x86, 0x85, 0xB9, 0xB4, 0x68, 0x0E, 0xC6, 0xD1, + 0x8A, 0x81, 0x2B, 0xB3, 0xF6, 0x48, 0xF0, 0x4F, 0x9C, 0x28, 0x1C, 0xA4, 0x51, 0x2F, 0xD7, 0x4B, + 0x17, 0xE7, 0xCC, 0x50, 0x9F, 0xD0, 0xD1, 0x40, 0x0C, 0x0D, 0xCA, 0x83, 0xFA, 0x5E, 0xCA, 0xEC, + 0xBF, 0x4E, 0x7C, 0x8F, 0xF0, 0xAE, 0xC2, 0xD3, 0x28, 0x41, 0x9B, 0xC8, 0x04, 0xB9, 0x4A, 0xBA, + 0x72, 0xE2, 0xB5, 0x06, 0x2C, 0x1E, 0x0B, 0x2C, 0x7F, 0x11, 0xA9, 0x26, 0x51, 0x9D, 0x3F, 0xF8, + 0x62, 0x11, 0x2E, 0x89, 0xD2, 0x9D, 0x35, 0xB1, 0xE4, 0x0A, 0x4D, 0x93, 0x01, 0xA7, 0xD1, 0x2D, + 0x00, 0x87, 0xE2, 0x2D, 0xA4, 0xE9, 0x0A, 0x06, 0x66, 0xF8, 0x1F, 0x44, 0x75, 0xB5, 0x6B, 0x1C, + 0xFC, 0x31, 0x09, 0x48, 0xA3, 0xFF, 0x92, 0x12, 0x58, 0xE9, 0xFA, 0xAE, 0x4F, 0xE2, 0xB4, 0xCC +}; + +#define debuggerReadMemory(addr) \ + READ32LE((&map[(addr)>>24].address[(addr) & map[(addr)>>24].mask])) + +#define debuggerReadHalfWord(addr) \ + READ16LE((&map[(addr)>>24].address[(addr) & map[(addr)>>24].mask])) + +#define debuggerReadByte(addr) \ + map[(addr)>>24].address[(addr) & map[(addr)>>24].mask] + +#define debuggerWriteMemory(addr, value) \ + WRITE32LE(&map[(addr)>>24].address[(addr) & map[(addr)>>24].mask], value) + +#define debuggerWriteHalfWord(addr, value) \ + WRITE16LE(&map[(addr)>>24].address[(addr) & map[(addr)>>24].mask], value) + +#define debuggerWriteByte(addr, value) \ + map[(addr)>>24].address[(addr) & map[(addr)>>24].mask] = (value) + + +#define CHEAT_IS_HEX(a) ( ((a)>='A' && (a) <='F') || ((a) >='0' && (a) <= '9')) + +#define CHEAT_PATCH_ROM_16BIT(a,v) \ + WRITE16LE(((u16 *)&rom[(a) & 0x1ffffff]), v); + +#define CHEAT_PATCH_ROM_32BIT(a,v) \ + WRITE32LE(((u32 *)&rom[(a) & 0x1ffffff]), v); + +static bool isMultilineWithData(int i) +{ + // we consider it a multiline code if it has more than one line of data + // otherwise, it can still be considered a single code + // (Only CBA codes can be true multilines !!!) + if(i < cheatsNumber && i >= 0) + switch(cheatsList[i].size) { + case INT_8_BIT_WRITE: + case INT_16_BIT_WRITE: + case INT_32_BIT_WRITE: + case GSA_16_BIT_ROM_PATCH: + case GSA_8_BIT_GS_WRITE: + case GSA_16_BIT_GS_WRITE: + case GSA_32_BIT_GS_WRITE: + case CBA_AND: + case CBA_IF_KEYS_PRESSED: + case CBA_IF_TRUE: + case CBA_IF_FALSE: + case GSA_8_BIT_IF_TRUE: + case GSA_32_BIT_IF_TRUE: + case GSA_8_BIT_IF_FALSE: + case GSA_32_BIT_IF_FALSE: + case GSA_8_BIT_FILL: + case GSA_16_BIT_FILL: + case GSA_8_BIT_IF_TRUE2: + case GSA_16_BIT_IF_TRUE2: + case GSA_32_BIT_IF_TRUE2: + case GSA_8_BIT_IF_FALSE2: + case GSA_16_BIT_IF_FALSE2: + case GSA_32_BIT_IF_FALSE2: + case GSA_SLOWDOWN: + case CBA_ADD: + case CBA_OR: + case CBA_LT: + case CBA_GT: + case GSA_8_BIT_POINTER: + case GSA_16_BIT_POINTER: + case GSA_32_BIT_POINTER: + case GSA_8_BIT_ADD: + case GSA_16_BIT_ADD: + case GSA_32_BIT_ADD: + case GSA_8_BIT_IF_LOWER_U: + case GSA_16_BIT_IF_LOWER_U: + case GSA_32_BIT_IF_LOWER_U: + case GSA_8_BIT_IF_HIGHER_U: + case GSA_16_BIT_IF_HIGHER_U: + case GSA_32_BIT_IF_HIGHER_U: + case GSA_8_BIT_IF_AND: + case GSA_16_BIT_IF_AND: + case GSA_32_BIT_IF_AND: + case GSA_8_BIT_IF_LOWER_U2: + case GSA_16_BIT_IF_LOWER_U2: + case GSA_32_BIT_IF_LOWER_U2: + case GSA_8_BIT_IF_HIGHER_U2: + case GSA_16_BIT_IF_HIGHER_U2: + case GSA_32_BIT_IF_HIGHER_U2: + case GSA_8_BIT_IF_AND2: + case GSA_16_BIT_IF_AND2: + case GSA_32_BIT_IF_AND2: + case GSA_ALWAYS: + case GSA_ALWAYS2: + case GSA_8_BIT_IF_LOWER_S: + case GSA_16_BIT_IF_LOWER_S: + case GSA_32_BIT_IF_LOWER_S: + case GSA_8_BIT_IF_HIGHER_S: + case GSA_16_BIT_IF_HIGHER_S: + case GSA_32_BIT_IF_HIGHER_S: + case GSA_8_BIT_IF_LOWER_S2: + case GSA_16_BIT_IF_LOWER_S2: + case GSA_32_BIT_IF_LOWER_S2: + case GSA_8_BIT_IF_HIGHER_S2: + case GSA_16_BIT_IF_HIGHER_S2: + case GSA_32_BIT_IF_HIGHER_S2: + case GSA_16_BIT_WRITE_IOREGS: + case GSA_32_BIT_WRITE_IOREGS: + case GSA_CODES_ON: + case GSA_8_BIT_IF_TRUE3: + case GSA_16_BIT_IF_TRUE3: + case GSA_32_BIT_IF_TRUE3: + case GSA_8_BIT_IF_FALSE3: + case GSA_16_BIT_IF_FALSE3: + case GSA_32_BIT_IF_FALSE3: + case GSA_8_BIT_IF_LOWER_S3: + case GSA_16_BIT_IF_LOWER_S3: + case GSA_32_BIT_IF_LOWER_S3: + case GSA_8_BIT_IF_HIGHER_S3: + case GSA_16_BIT_IF_HIGHER_S3: + case GSA_32_BIT_IF_HIGHER_S3: + case GSA_8_BIT_IF_LOWER_U3: + case GSA_16_BIT_IF_LOWER_U3: + case GSA_32_BIT_IF_LOWER_U3: + case GSA_8_BIT_IF_HIGHER_U3: + case GSA_16_BIT_IF_HIGHER_U3: + case GSA_32_BIT_IF_HIGHER_U3: + case GSA_8_BIT_IF_AND3: + case GSA_16_BIT_IF_AND3: + case GSA_32_BIT_IF_AND3: + case GSA_ALWAYS3: + case GSA_8_BIT_GS_WRITE2: + case GSA_16_BIT_GS_WRITE2: + case GSA_32_BIT_GS_WRITE2: + case GSA_16_BIT_ROM_PATCH2C: + case GSA_16_BIT_ROM_PATCH2D: + case GSA_16_BIT_ROM_PATCH2E: + case GSA_16_BIT_ROM_PATCH2F: + case GSA_8_BIT_SLIDE: + case GSA_16_BIT_SLIDE: + case GSA_32_BIT_SLIDE: + case GSA_GROUP_WRITE: + case GSA_32_BIT_ADD2: + case GSA_32_BIT_SUB2: + case GSA_16_BIT_IF_LOWER_OR_EQ_U: + case GSA_16_BIT_IF_HIGHER_OR_EQ_U: + case GSA_16_BIT_MIF_TRUE: + case GSA_16_BIT_MIF_FALSE: + case GSA_16_BIT_MIF_LOWER_OR_EQ_U: + case GSA_16_BIT_MIF_HIGHER_OR_EQ_U: + case MASTER_CODE: + case CHEATS_16_BIT_WRITE: + case CHEATS_32_BIT_WRITE: + return false; + // the codes below have two lines of data + case CBA_SLIDE_CODE: + case CBA_SUPER: + return true; + } + return false; +} + +static int getCodeLength(int num) +{ + if(num >= cheatsNumber || num < 0) + return 1; + + // this is for all the codes that are true multiline + switch(cheatsList[num].size) { + case INT_8_BIT_WRITE: + case INT_16_BIT_WRITE: + case INT_32_BIT_WRITE: + case GSA_16_BIT_ROM_PATCH: + case GSA_8_BIT_GS_WRITE: + case GSA_16_BIT_GS_WRITE: + case GSA_32_BIT_GS_WRITE: + case CBA_AND: + case GSA_8_BIT_FILL: + case GSA_16_BIT_FILL: + case GSA_SLOWDOWN: + case CBA_ADD: + case CBA_OR: + case GSA_8_BIT_POINTER: + case GSA_16_BIT_POINTER: + case GSA_32_BIT_POINTER: + case GSA_8_BIT_ADD: + case GSA_16_BIT_ADD: + case GSA_32_BIT_ADD: + case GSA_CODES_ON: + case GSA_8_BIT_IF_TRUE3: + case GSA_16_BIT_IF_TRUE3: + case GSA_32_BIT_IF_TRUE3: + case GSA_8_BIT_IF_FALSE3: + case GSA_16_BIT_IF_FALSE3: + case GSA_32_BIT_IF_FALSE3: + case GSA_8_BIT_IF_LOWER_S3: + case GSA_16_BIT_IF_LOWER_S3: + case GSA_32_BIT_IF_LOWER_S3: + case GSA_8_BIT_IF_HIGHER_S3: + case GSA_16_BIT_IF_HIGHER_S3: + case GSA_32_BIT_IF_HIGHER_S3: + case GSA_8_BIT_IF_LOWER_U3: + case GSA_16_BIT_IF_LOWER_U3: + case GSA_32_BIT_IF_LOWER_U3: + case GSA_8_BIT_IF_HIGHER_U3: + case GSA_16_BIT_IF_HIGHER_U3: + case GSA_32_BIT_IF_HIGHER_U3: + case GSA_8_BIT_IF_AND3: + case GSA_16_BIT_IF_AND3: + case GSA_32_BIT_IF_AND3: + case GSA_8_BIT_IF_LOWER_U: + case GSA_16_BIT_IF_LOWER_U: + case GSA_32_BIT_IF_LOWER_U: + case GSA_8_BIT_IF_HIGHER_U: + case GSA_16_BIT_IF_HIGHER_U: + case GSA_32_BIT_IF_HIGHER_U: + case GSA_8_BIT_IF_AND: + case GSA_16_BIT_IF_AND: + case GSA_32_BIT_IF_AND: + case GSA_ALWAYS: + case GSA_8_BIT_IF_LOWER_S: + case GSA_16_BIT_IF_LOWER_S: + case GSA_32_BIT_IF_LOWER_S: + case GSA_8_BIT_IF_HIGHER_S: + case GSA_16_BIT_IF_HIGHER_S: + case GSA_32_BIT_IF_HIGHER_S: + case GSA_16_BIT_WRITE_IOREGS: + case GSA_32_BIT_WRITE_IOREGS: + case GSA_8_BIT_GS_WRITE2: + case GSA_16_BIT_GS_WRITE2: + case GSA_32_BIT_GS_WRITE2: + case GSA_16_BIT_ROM_PATCH2C: + case GSA_16_BIT_ROM_PATCH2D: + case GSA_16_BIT_ROM_PATCH2E: + case GSA_16_BIT_ROM_PATCH2F: + case GSA_8_BIT_SLIDE: + case GSA_16_BIT_SLIDE: + case GSA_32_BIT_SLIDE: + case GSA_8_BIT_IF_TRUE: + case GSA_32_BIT_IF_TRUE: + case GSA_8_BIT_IF_FALSE: + case GSA_32_BIT_IF_FALSE: + case CBA_LT: + case CBA_GT: + case CBA_IF_TRUE: + case CBA_IF_FALSE: + case GSA_8_BIT_IF_TRUE2: + case GSA_16_BIT_IF_TRUE2: + case GSA_32_BIT_IF_TRUE2: + case GSA_8_BIT_IF_FALSE2: + case GSA_16_BIT_IF_FALSE2: + case GSA_32_BIT_IF_FALSE2: + case GSA_8_BIT_IF_LOWER_U2: + case GSA_16_BIT_IF_LOWER_U2: + case GSA_32_BIT_IF_LOWER_U2: + case GSA_8_BIT_IF_HIGHER_U2: + case GSA_16_BIT_IF_HIGHER_U2: + case GSA_32_BIT_IF_HIGHER_U2: + case GSA_8_BIT_IF_AND2: + case GSA_16_BIT_IF_AND2: + case GSA_32_BIT_IF_AND2: + case GSA_ALWAYS2: + case GSA_8_BIT_IF_LOWER_S2: + case GSA_16_BIT_IF_LOWER_S2: + case GSA_32_BIT_IF_LOWER_S2: + case GSA_8_BIT_IF_HIGHER_S2: + case GSA_16_BIT_IF_HIGHER_S2: + case GSA_32_BIT_IF_HIGHER_S2: + case GSA_GROUP_WRITE: + case GSA_32_BIT_ADD2: + case GSA_32_BIT_SUB2: + case GSA_16_BIT_IF_LOWER_OR_EQ_U: + case GSA_16_BIT_IF_HIGHER_OR_EQ_U: + case GSA_16_BIT_MIF_TRUE: + case GSA_16_BIT_MIF_FALSE: + case GSA_16_BIT_MIF_LOWER_OR_EQ_U: + case GSA_16_BIT_MIF_HIGHER_OR_EQ_U: + case MASTER_CODE: + case CHEATS_16_BIT_WRITE: + case CHEATS_32_BIT_WRITE: + case UNKNOWN_CODE: + return 1; + case CBA_IF_KEYS_PRESSED: + case CBA_SLIDE_CODE: + return 2; + case CBA_SUPER: + return ((((cheatsList[num].value-1) & 0xFFFF)/3) + 1); + } + return 1; +} + +int cheatsCheckKeys(u32 keys, u32 extended) +{ + bool onoff = true; + int ticks = 0; + int i; + mastercode = 0; + + for (i = 0; i<4; i++) + if (rompatch2addr [i] != 0) { + CHEAT_PATCH_ROM_16BIT(rompatch2addr [i],rompatch2oldval [i]); + rompatch2addr [i] = 0; + } + + for (i = 0; i < cheatsNumber; i++) { + if(!cheatsList[i].enabled) { + // make sure we skip other lines in this code + i += getCodeLength(i)-1; + continue; + } + switch(cheatsList[i].size) { + case GSA_CODES_ON: + onoff = true; + break; + case GSA_SLOWDOWN: + // check if button was pressed and released, if so toggle our state + if((cheatsList[i].status & 4) && !(extended & 4)) + cheatsList[i].status ^= 1; + if(extended & 4) + cheatsList[i].status |= 4; + else + cheatsList[i].status &= ~4; + + if(cheatsList[i].status & 1) + ticks += ((cheatsList[i].value & 0xFFFF) * 7); + break; + case GSA_8_BIT_SLIDE: + i++; + if(i < cheatsNumber) { + u32 addr = cheatsList[i-1].value; + u8 value = cheatsList[i].rawaddress; + int vinc = (cheatsList[i].value >> 24) & 255; + int count = (cheatsList[i].value >> 16) & 255; + int ainc = (cheatsList[i].value & 0xffff); + while(count > 0) { + CPUWriteByte(addr, value); + value += vinc; + addr += ainc; + count--; + } + } + break; + case GSA_16_BIT_SLIDE: + i++; + if(i < cheatsNumber) { + u32 addr = cheatsList[i-1].value; + u16 value = cheatsList[i].rawaddress; + int vinc = (cheatsList[i].value >> 24) & 255; + int count = (cheatsList[i].value >> 16) & 255; + int ainc = (cheatsList[i].value & 0xffff)*2; + while(count > 0) { + CPUWriteHalfWord(addr, value); + value += vinc; + addr += ainc; + count--; + } + } + break; + case GSA_32_BIT_SLIDE: + i++; + if(i < cheatsNumber) { + u32 addr = cheatsList[i-1].value; + u32 value = cheatsList[i].rawaddress; + int vinc = (cheatsList[i].value >> 24) & 255; + int count = (cheatsList[i].value >> 16) & 255; + int ainc = (cheatsList[i].value & 0xffff)*4; + while(count > 0) { + CPUWriteMemory(addr, value); + value += vinc; + addr += ainc; + count--; + } + } + break; + case GSA_8_BIT_GS_WRITE2: + i++; + if(i < cheatsNumber) { + if(extended & 4) { + CPUWriteByte(cheatsList[i-1].value, cheatsList[i].address); + } + } + break; + case GSA_16_BIT_GS_WRITE2: + i++; + if(i < cheatsNumber) { + if(extended & 4) { + CPUWriteHalfWord(cheatsList[i-1].value, cheatsList[i].address); + } + } + break; + case GSA_32_BIT_GS_WRITE2: + i++; + if(i < cheatsNumber) { + if(extended & 4) { + CPUWriteMemory(cheatsList[i-1].value, cheatsList[i].address); + } + } + break; + case GSA_16_BIT_ROM_PATCH: + if((cheatsList[i].status & 1) == 0) { + if(CPUReadHalfWord(cheatsList[i].address) != cheatsList[i].value) { + cheatsList[i].oldValue = CPUReadHalfWord(cheatsList[i].address); + cheatsList[i].status |= 1; + CHEAT_PATCH_ROM_16BIT(cheatsList[i].address, cheatsList[i].value); + } + } + break; + case GSA_16_BIT_ROM_PATCH2C: + i++; + if(i < cheatsNumber) { + rompatch2addr [0] = ((cheatsList[i-1].value & 0x00FFFFFF) << 1) + 0x8000000; + rompatch2oldval [0] = CPUReadHalfWord(rompatch2addr [0]); + rompatch2val [0] = cheatsList[i].rawaddress & 0xFFFF; + } + break; + case GSA_16_BIT_ROM_PATCH2D: + i++; + if(i < cheatsNumber) { + rompatch2addr [1] = ((cheatsList[i-1].value & 0x00FFFFFF) << 1) + 0x8000000; + rompatch2oldval [1] = CPUReadHalfWord(rompatch2addr [1]); + rompatch2val [1] = cheatsList[i].rawaddress & 0xFFFF; + } + break; + case GSA_16_BIT_ROM_PATCH2E: + i++; + if(i < cheatsNumber) { + rompatch2addr [2] = ((cheatsList[i-1].value & 0x00FFFFFF) << 1) + 0x8000000; + rompatch2oldval [2] = CPUReadHalfWord(rompatch2addr [2]); + rompatch2val [2] = cheatsList[i].rawaddress & 0xFFFF; + } + break; + case GSA_16_BIT_ROM_PATCH2F: + i++; + if(i < cheatsNumber) { + rompatch2addr [3] = ((cheatsList[i-1].value & 0x00FFFFFF) << 1) + 0x8000000; + rompatch2oldval [3] = CPUReadHalfWord(rompatch2addr [3]); + rompatch2val [3] = cheatsList[i].rawaddress & 0xFFFF; + } + break; + case MASTER_CODE: + mastercode = cheatsList[i].address; + break; + } + if (onoff) { + switch(cheatsList[i].size) { + case INT_8_BIT_WRITE: + CPUWriteByte(cheatsList[i].address, cheatsList[i].value); + break; + case INT_16_BIT_WRITE: + CPUWriteHalfWord(cheatsList[i].address, cheatsList[i].value); + break; + case INT_32_BIT_WRITE: + CPUWriteMemory(cheatsList[i].address, cheatsList[i].value); + break; + case GSA_8_BIT_GS_WRITE: + if(extended & 4) { + CPUWriteByte(cheatsList[i].address, cheatsList[i].value); + } + break; + case GSA_16_BIT_GS_WRITE: + if(extended & 4) { + CPUWriteHalfWord(cheatsList[i].address, cheatsList[i].value); + } + break; + case GSA_32_BIT_GS_WRITE: + if(extended & 4) { + CPUWriteMemory(cheatsList[i].address, cheatsList[i].value); + } + break; + case CBA_IF_KEYS_PRESSED: + { + u16 value = cheatsList[i].value; + u32 addr = cheatsList[i].address; + if((addr & 0xF0) == 0x20) { + if((keys & value) == 0) { + i++; + } + } else if((addr & 0xF0) == 0x10) { + if((keys & value) == value) { + i++; + } + } else if((addr & 0xF0) == 0x00) { + if(((~keys) & 0x3FF) == value) { + i++; + } + } + } + break; + case CBA_IF_TRUE: + if(CPUReadHalfWord(cheatsList[i].address) != cheatsList[i].value) { + i++; + } + break; + case CBA_SLIDE_CODE: + { + u32 address = cheatsList[i].address; + u16 value = cheatsList[i].value; + i++; + if(i < cheatsNumber) { + int count = ((cheatsList[i].address - 1) & 0xFFFF); + u16 vinc = (cheatsList[i].address >> 16) & 0xFFFF; + int inc = cheatsList[i].value; + for(int x = 0; x <= count ; x++) { + CPUWriteHalfWord(address, value); + address += inc; + value += vinc; + } + } + } + break; + case CBA_IF_FALSE: + if(CPUReadHalfWord(cheatsList[i].address) == cheatsList[i].value){ + i++; + } + break; + case CBA_AND: + CPUWriteHalfWord(cheatsList[i].address, + CPUReadHalfWord(cheatsList[i].address) & + cheatsList[i].value); + break; + case GSA_8_BIT_IF_TRUE: + if(CPUReadByte(cheatsList[i].address) != cheatsList[i].value) { + i++; + } + break; + case GSA_32_BIT_IF_TRUE: + if(CPUReadMemory(cheatsList[i].address) != cheatsList[i].value) { + i++; + } + break; + case GSA_8_BIT_IF_FALSE: + if(CPUReadByte(cheatsList[i].address) == cheatsList[i].value) { + i++; + } + break; + case GSA_32_BIT_IF_FALSE: + if(CPUReadMemory(cheatsList[i].address) == cheatsList[i].value) { + i++; + } + break; + case GSA_8_BIT_FILL: + { + u32 addr = cheatsList[i].address; + u8 v = cheatsList[i].value & 0xff; + u32 end = addr + (cheatsList[i].value >> 8); + do { + CPUWriteByte(addr, v); + addr++; + } while (addr <= end); + } + break; + case GSA_16_BIT_FILL: + { + u32 addr = cheatsList[i].address; + u16 v = cheatsList[i].value & 0xffff; + u32 end = addr + ((cheatsList[i].value >> 16) << 1); + do { + CPUWriteHalfWord(addr, v); + addr+=2; + } while (addr <= end); + } + break; + case GSA_8_BIT_IF_TRUE2: + if(CPUReadByte(cheatsList[i].address) != cheatsList[i].value) { + i+=2; + } + break; + case GSA_16_BIT_IF_TRUE2: + if(CPUReadHalfWord(cheatsList[i].address) != cheatsList[i].value) { + i+=2; + } + break; + case GSA_32_BIT_IF_TRUE2: + if(CPUReadMemory(cheatsList[i].address) != cheatsList[i].value) { + i+=2; + } + break; + case GSA_8_BIT_IF_FALSE2: + if(CPUReadByte(cheatsList[i].address) == cheatsList[i].value) { + i+=2; + } + break; + case GSA_16_BIT_IF_FALSE2: + if(CPUReadHalfWord(cheatsList[i].address) == cheatsList[i].value) { + i+=2; + } + break; + case GSA_32_BIT_IF_FALSE2: + if(CPUReadMemory(cheatsList[i].address) == cheatsList[i].value) { + i+=2; + } + break; + case CBA_ADD: + if ((cheatsList[i].address & 1) == 0) { + CPUWriteHalfWord(cheatsList[i].address, + CPUReadHalfWord(cheatsList[i].address) + + cheatsList[i].value); + } else { + CPUWriteMemory(cheatsList[i].address & 0x0FFFFFFE, + CPUReadMemory(cheatsList[i].address & 0x0FFFFFFE) + + cheatsList[i].value); + } + break; + case CBA_OR: + CPUWriteHalfWord(cheatsList[i].address, + CPUReadHalfWord(cheatsList[i].address) | + cheatsList[i].value); + break; + case CBA_GT: + if (!(CPUReadHalfWord(cheatsList[i].address) > cheatsList[i].value)){ + i++; + } + break; + case CBA_LT: + if (!(CPUReadHalfWord(cheatsList[i].address) < cheatsList[i].value)){ + i++; + } + break; + case CBA_SUPER: + { + int count = 2*((cheatsList[i].value -1) & 0xFFFF)+1; + u32 address = cheatsList[i].address; + for(int x = 0; x <= count; x++) { + u8 b; + int res = x % 6; + if (res==0) + i++; + if(res < 4) + b = (cheatsList[i].address >> (24-8*res)) & 0xFF; + else + b = (cheatsList[i].value >> (8 - 8*(res-4))) & 0xFF; + CPUWriteByte(address, b); + address++; + } + } + break; + case GSA_8_BIT_POINTER : + if ((CPUReadMemory(cheatsList[i].address)>=0x02000000) && (CPUReadMemory(cheatsList[i].address)<0x02040000) || + (CPUReadMemory(cheatsList[i].address)>=0x03000000) && (CPUReadMemory(cheatsList[i].address)<0x03008000)) + { + CPUWriteByte(CPUReadMemory(cheatsList[i].address)+((cheatsList[i].value & 0xFFFFFF00) >> 8), + cheatsList[i].value & 0xFF); + } + break; + case GSA_16_BIT_POINTER : + if ((CPUReadMemory(cheatsList[i].address)>=0x02000000) && (CPUReadMemory(cheatsList[i].address)<0x02040000) || + (CPUReadMemory(cheatsList[i].address)>=0x03000000) && (CPUReadMemory(cheatsList[i].address)<0x03008000)) + { + CPUWriteHalfWord(CPUReadMemory(cheatsList[i].address)+((cheatsList[i].value & 0xFFFF0000) >> 15), + cheatsList[i].value & 0xFFFF); + } + break; + case GSA_32_BIT_POINTER : + if ((CPUReadMemory(cheatsList[i].address)>=0x02000000) && (CPUReadMemory(cheatsList[i].address)<0x02040000) || + (CPUReadMemory(cheatsList[i].address)>=0x03000000) && (CPUReadMemory(cheatsList[i].address)<0x03008000)) + { + CPUWriteMemory(CPUReadMemory(cheatsList[i].address), + cheatsList[i].value); + } + break; + case GSA_8_BIT_ADD : + CPUWriteByte(cheatsList[i].address, + (cheatsList[i].value & 0xFF) + CPUReadMemory(cheatsList[i].address) & 0xFF); + break; + case GSA_16_BIT_ADD : + CPUWriteHalfWord(cheatsList[i].address, + (cheatsList[i].value & 0xFFFF) + CPUReadMemory(cheatsList[i].address) & 0xFFFF); + break; + case GSA_32_BIT_ADD : + CPUWriteMemory(cheatsList[i].address , + cheatsList[i].value + CPUReadMemory(cheatsList[i].address) & 0xFFFFFFFF); + break; + case GSA_8_BIT_IF_LOWER_U: + if (!(CPUReadByte(cheatsList[i].address) < (cheatsList[i].value & 0xFF))) { + i++; + } + break; + case GSA_16_BIT_IF_LOWER_U: + if (!(CPUReadHalfWord(cheatsList[i].address) < (cheatsList[i].value & 0xFFFF))) { + i++; + } + break; + case GSA_32_BIT_IF_LOWER_U: + if (!(CPUReadMemory(cheatsList[i].address) < cheatsList[i].value)) { + i++; + } + break; + case GSA_8_BIT_IF_HIGHER_U: + if (!(CPUReadByte(cheatsList[i].address) > (cheatsList[i].value & 0xFF))) { + i++; + } + break; + case GSA_16_BIT_IF_HIGHER_U: + if (!(CPUReadHalfWord(cheatsList[i].address) > (cheatsList[i].value & 0xFFFF))) { + i++; + } + break; + case GSA_32_BIT_IF_HIGHER_U: + if (!(CPUReadMemory(cheatsList[i].address) > cheatsList[i].value)) { + i++; + } + break; + case GSA_8_BIT_IF_AND: + if (!(CPUReadByte(cheatsList[i].address) & (cheatsList[i].value & 0xFF))) { + i++; + } + break; + case GSA_16_BIT_IF_AND: + if (!(CPUReadHalfWord(cheatsList[i].address) & (cheatsList[i].value & 0xFFFF))) { + i++; + } + break; + case GSA_32_BIT_IF_AND: + if (!(CPUReadMemory(cheatsList[i].address) & cheatsList[i].value)) { + i++; + } + break; + case GSA_8_BIT_IF_LOWER_U2: + if (!(CPUReadByte(cheatsList[i].address) < (cheatsList[i].value & 0xFF))) { + i+=2; + } + break; + case GSA_16_BIT_IF_LOWER_U2: + if (!(CPUReadHalfWord(cheatsList[i].address) < (cheatsList[i].value & 0xFFFF))) { + i+=2; + } + break; + case GSA_32_BIT_IF_LOWER_U2: + if (!(CPUReadMemory(cheatsList[i].address) < cheatsList[i].value)) { + i+=2; + } + break; + case GSA_8_BIT_IF_HIGHER_U2: + if (!(CPUReadByte(cheatsList[i].address) > (cheatsList[i].value & 0xFF))) { + i+=2; + } + break; + case GSA_16_BIT_IF_HIGHER_U2: + if (!(CPUReadHalfWord(cheatsList[i].address) > (cheatsList[i].value & 0xFFFF))) { + i+=2; + } + break; + case GSA_32_BIT_IF_HIGHER_U2: + if (!(CPUReadMemory(cheatsList[i].address) > cheatsList[i].value)) { + i+=2; + } + break; + case GSA_8_BIT_IF_AND2: + if (!(CPUReadByte(cheatsList[i].address) & (cheatsList[i].value & 0xFF))) { + i+=2; + } + break; + case GSA_16_BIT_IF_AND2: + if (!(CPUReadHalfWord(cheatsList[i].address) & (cheatsList[i].value & 0xFFFF))) { + i+=2; + } + break; + case GSA_32_BIT_IF_AND2: + if (!(CPUReadMemory(cheatsList[i].address) & cheatsList[i].value)) { + i+=2; + } + break; + case GSA_ALWAYS: + i++; + break; + case GSA_ALWAYS2: + i+=2; + break; + case GSA_8_BIT_IF_LOWER_S: + if (!((s8)CPUReadByte(cheatsList[i].address) < ((s8)cheatsList[i].value & 0xFF))) { + i++; + } + break; + case GSA_16_BIT_IF_LOWER_S: + if (!((s16)CPUReadHalfWord(cheatsList[i].address) < ((s16)cheatsList[i].value & 0xFFFF))) { + i++; + } + break; + case GSA_32_BIT_IF_LOWER_S: + if (!((s32)CPUReadMemory(cheatsList[i].address) < (s32)cheatsList[i].value)) { + i++; + } + break; + case GSA_8_BIT_IF_HIGHER_S: + if (!((s8)CPUReadByte(cheatsList[i].address) > ((s8)cheatsList[i].value & 0xFF))) { + i++; + } + break; + case GSA_16_BIT_IF_HIGHER_S: + if (!((s16)CPUReadHalfWord(cheatsList[i].address) > ((s16)cheatsList[i].value & 0xFFFF))) { + i++; + } + break; + case GSA_32_BIT_IF_HIGHER_S: + if (!((s32)CPUReadMemory(cheatsList[i].address) > (s32)cheatsList[i].value)) { + i++; + } + break; + case GSA_8_BIT_IF_LOWER_S2: + if (!((s8)CPUReadByte(cheatsList[i].address) < ((s8)cheatsList[i].value & 0xFF))) { + i+=2; + } + break; + case GSA_16_BIT_IF_LOWER_S2: + if (!((s16)CPUReadHalfWord(cheatsList[i].address) < ((s16)cheatsList[i].value & 0xFFFF))) { + i+=2; + } + break; + case GSA_32_BIT_IF_LOWER_S2: + if (!((s32)CPUReadMemory(cheatsList[i].address) < (s32)cheatsList[i].value)) { + i+=2; + } + break; + case GSA_8_BIT_IF_HIGHER_S2: + if (!((s8)CPUReadByte(cheatsList[i].address) > ((s8)cheatsList[i].value & 0xFF))) { + i+=2; + } + break; + case GSA_16_BIT_IF_HIGHER_S2: + if (!((s16)CPUReadHalfWord(cheatsList[i].address) > ((s16)cheatsList[i].value & 0xFFFF))) { + i+=2; + } + break; + case GSA_32_BIT_IF_HIGHER_S2: + if (!((s32)CPUReadMemory(cheatsList[i].address) > (s32)cheatsList[i].value)) { + i+=2; + } + break; + case GSA_16_BIT_WRITE_IOREGS: + if ((cheatsList[i].address <= 0x3FF) && (cheatsList[i].address != 0x6) && + (cheatsList[i].address != 0x130)) + ioMem[cheatsList[i].address & 0x3FE]=cheatsList[i].value & 0xFFFF; + break; + case GSA_32_BIT_WRITE_IOREGS: + if (cheatsList[i].address<=0x3FF) + { + if (((cheatsList[i].address & 0x3FC) != 0x6) && ((cheatsList[i].address & 0x3FC) != 0x130)) + ioMem[cheatsList[i].address & 0x3FC]= (cheatsList[i].value & 0xFFFF); + if ((((cheatsList[i].address & 0x3FC)+2) != 0x6) && ((cheatsList[i].address & 0x3FC) +2) != 0x130) + ioMem[(cheatsList[i].address & 0x3FC) + 2 ]= ((cheatsList[i].value>>16 ) & 0xFFFF); + } + break; + case GSA_8_BIT_IF_TRUE3: + if(CPUReadByte(cheatsList[i].address) != cheatsList[i].value) { + onoff=false; + } + break; + case GSA_16_BIT_IF_TRUE3: + if(CPUReadHalfWord(cheatsList[i].address) != cheatsList[i].value) { + onoff=false; + } + break; + case GSA_32_BIT_IF_TRUE3: + if(CPUReadMemory(cheatsList[i].address) != cheatsList[i].value) { + onoff=false; + } + break; + case GSA_8_BIT_IF_FALSE3: + if(CPUReadByte(cheatsList[i].address) == cheatsList[i].value) { + onoff=false; + } + break; + case GSA_16_BIT_IF_FALSE3: + if(CPUReadHalfWord(cheatsList[i].address) == cheatsList[i].value) { + onoff=false; + } + break; + case GSA_32_BIT_IF_FALSE3: + if(CPUReadMemory(cheatsList[i].address) == cheatsList[i].value) { + onoff=false; + } + break; + case GSA_8_BIT_IF_LOWER_S3: + if (!((s8)CPUReadByte(cheatsList[i].address) < ((s8)cheatsList[i].value & 0xFF))) { + onoff=false; + } + break; + case GSA_16_BIT_IF_LOWER_S3: + if (!((s16)CPUReadHalfWord(cheatsList[i].address) < ((s16)cheatsList[i].value & 0xFFFF))) { + onoff=false; + } + break; + case GSA_32_BIT_IF_LOWER_S3: + if (!((s32)CPUReadMemory(cheatsList[i].address) < (s32)cheatsList[i].value)) { + onoff=false; + } + break; + case GSA_8_BIT_IF_HIGHER_S3: + if (!((s8)CPUReadByte(cheatsList[i].address) > ((s8)cheatsList[i].value & 0xFF))) { + onoff=false; + } + break; + case GSA_16_BIT_IF_HIGHER_S3: + if (!((s16)CPUReadHalfWord(cheatsList[i].address) > ((s16)cheatsList[i].value & 0xFFFF))) { + onoff=false; + } + break; + case GSA_32_BIT_IF_HIGHER_S3: + if (!((s32)CPUReadMemory(cheatsList[i].address) > (s32)cheatsList[i].value)) { + onoff=false; + } + break; + case GSA_8_BIT_IF_LOWER_U3: + if (!(CPUReadByte(cheatsList[i].address) < (cheatsList[i].value & 0xFF))) { + onoff=false; + } + break; + case GSA_16_BIT_IF_LOWER_U3: + if (!(CPUReadHalfWord(cheatsList[i].address) < (cheatsList[i].value & 0xFFFF))) { + onoff=false; + } + break; + case GSA_32_BIT_IF_LOWER_U3: + if (!(CPUReadMemory(cheatsList[i].address) < cheatsList[i].value)) { + onoff=false; + } + break; + case GSA_8_BIT_IF_HIGHER_U3: + if (!(CPUReadByte(cheatsList[i].address) > (cheatsList[i].value & 0xFF))) { + onoff=false; + } + break; + case GSA_16_BIT_IF_HIGHER_U3: + if (!(CPUReadHalfWord(cheatsList[i].address) > (cheatsList[i].value & 0xFFFF))) { + onoff=false; + } + break; + case GSA_32_BIT_IF_HIGHER_U3: + if (!(CPUReadMemory(cheatsList[i].address) > cheatsList[i].value)) { + onoff=false; + } + break; + case GSA_8_BIT_IF_AND3: + if (!(CPUReadByte(cheatsList[i].address) & (cheatsList[i].value & 0xFF))) { + onoff=false; + } + break; + case GSA_16_BIT_IF_AND3: + if (!(CPUReadHalfWord(cheatsList[i].address) & (cheatsList[i].value & 0xFFFF))) { + onoff=false; + } + break; + case GSA_32_BIT_IF_AND3: + if (!(CPUReadMemory(cheatsList[i].address) & cheatsList[i].value)) { + onoff=false; + } + break; + case GSA_ALWAYS3: + if (!(CPUReadMemory(cheatsList[i].address) & cheatsList[i].value)) { + onoff=false; + } + break; + case GSA_GROUP_WRITE: + { + int count = ((cheatsList[i].address) & 0xFFFE) +1; + u32 value = cheatsList[i].value; + if (count==0) + i++; + else + for (int x = 1; x <= count; x++) { + if ((x % 2) ==0){ + if (x cheatsList[i].value) { + i++; + } + break; + case GSA_16_BIT_IF_HIGHER_OR_EQ_U: + if(CPUReadHalfWord(cheatsList[i].address) < cheatsList[i].value) { + i++; + } + break; + case GSA_16_BIT_MIF_TRUE: + if(CPUReadHalfWord(cheatsList[i].address) != cheatsList[i].value) { + i+=((cheatsList[i].rawaddress >> 0x10) & 0xFF); + } + break; + case GSA_16_BIT_MIF_FALSE: + if(CPUReadHalfWord(cheatsList[i].address) == cheatsList[i].value) { + i+=(cheatsList[i].rawaddress >> 0x10) & 0xFF; + } + break; + case GSA_16_BIT_MIF_LOWER_OR_EQ_U: + if(CPUReadHalfWord(cheatsList[i].address) > cheatsList[i].value) { + i+=(cheatsList[i].rawaddress >> 0x10) & 0xFF; + } + break; + case GSA_16_BIT_MIF_HIGHER_OR_EQ_U: + if(CPUReadHalfWord(cheatsList[i].address) < cheatsList[i].value) { + i+=(cheatsList[i].rawaddress >> 0x10) & 0xFF; + } + break; + case CHEATS_16_BIT_WRITE: + if ((cheatsList[i].address>>24)>=0x08) { + CHEAT_PATCH_ROM_16BIT(cheatsList[i].address, cheatsList[i].value); + } else { + CPUWriteHalfWord(cheatsList[i].address, cheatsList[i].value); + } + break; + case CHEATS_32_BIT_WRITE: + if ((cheatsList[i].address>>24)>=0x08) { + CHEAT_PATCH_ROM_32BIT(cheatsList[i].address, cheatsList[i].value); + } else { + CPUWriteMemory(cheatsList[i].address, cheatsList[i].value); + } + break; + } + } + } + for (i = 0; i<4; i++) + if (rompatch2addr [i] != 0) + CHEAT_PATCH_ROM_16BIT(rompatch2addr [i],rompatch2val [i]); + return ticks; +} + +void cheatsAdd(const char *codeStr, + const char *desc, + u32 rawaddress, + u32 address, + u32 value, + int code, + int size) +{ + if(cheatsNumber < 100) { + int x = cheatsNumber; + cheatsList[x].code = code; + cheatsList[x].size = size; + cheatsList[x].rawaddress = rawaddress; + cheatsList[x].address = address; + cheatsList[x].value = value; + strcpy(cheatsList[x].codestring, codeStr); + strcpy(cheatsList[x].desc, desc); + cheatsList[x].enabled = true; + cheatsList[x].status = 0; + + // we only store the old value for this simple codes. ROM patching + // is taken care when it actually patches the ROM + switch(cheatsList[x].size) { + case INT_8_BIT_WRITE: + cheatsList[x].oldValue = CPUReadByte(address); + break; + case INT_16_BIT_WRITE: + cheatsList[x].oldValue = CPUReadHalfWord(address); + break; + case INT_32_BIT_WRITE: + cheatsList[x].oldValue = CPUReadMemory(address); + break; + case CHEATS_16_BIT_WRITE: + cheatsList[x].oldValue = CPUReadHalfWord(address); + break; + case CHEATS_32_BIT_WRITE: + cheatsList[x].oldValue = CPUReadMemory(address); + break; + } + cheatsNumber++; + } +} + +void cheatsDelete(int number, bool restore) +{ + if(number < cheatsNumber && number >= 0) { + int x = number; + + if(restore) { + switch(cheatsList[x].size) { + case INT_8_BIT_WRITE: + CPUWriteByte(cheatsList[x].address, (u8)cheatsList[x].oldValue); + break; + case INT_16_BIT_WRITE: + CPUWriteHalfWord(cheatsList[x].address, (u16)cheatsList[x].oldValue); + break; + case INT_32_BIT_WRITE: + CPUWriteMemory(cheatsList[x].address, cheatsList[x].oldValue); + break; + case CHEATS_16_BIT_WRITE: + if ((cheatsList[x].address>>24)>=0x08) { + CHEAT_PATCH_ROM_16BIT(cheatsList[x].address, cheatsList[x].oldValue); + } else { + CPUWriteHalfWord(cheatsList[x].address, cheatsList[x].oldValue); + } + break; + case CHEATS_32_BIT_WRITE: + if ((cheatsList[x].address>>24)>=0x08) { + CHEAT_PATCH_ROM_32BIT(cheatsList[x].address, cheatsList[x].oldValue); + } else { + CPUWriteMemory(cheatsList[x].address, cheatsList[x].oldValue); + } + case GSA_16_BIT_ROM_PATCH: + if(cheatsList[x].status & 1) { + cheatsList[x].status &= ~1; + CHEAT_PATCH_ROM_16BIT(cheatsList[x].address, + cheatsList[x].oldValue); + } + break; + case GSA_16_BIT_ROM_PATCH2C: + case GSA_16_BIT_ROM_PATCH2D: + case GSA_16_BIT_ROM_PATCH2E: + case GSA_16_BIT_ROM_PATCH2F: + if(cheatsList[x].status & 1) { + cheatsList[x].status &= ~1; + } + break; + case MASTER_CODE: + mastercode=0; + break; + } + } + if((x+1) < cheatsNumber) { + memcpy(&cheatsList[x], &cheatsList[x+1], sizeof(CheatsData)* + (cheatsNumber-x-1)); + } + cheatsNumber--; + } +} + +void cheatsDeleteAll(bool restore) +{ + for(int i = cheatsNumber-1; i >= 0; i--) { + cheatsDelete(i, restore); + } +} + +void cheatsEnable(int i) +{ + if(i >= 0 && i < cheatsNumber) { + cheatsList[i].enabled = true; + mastercode = 0; + } +} + +void cheatsDisable(int i) +{ + if(i >= 0 && i < cheatsNumber) { + switch(cheatsList[i].size) { + case GSA_16_BIT_ROM_PATCH: + if(cheatsList[i].status & 1) { + cheatsList[i].status &= ~1; + CHEAT_PATCH_ROM_16BIT(cheatsList[i].address, + cheatsList[i].oldValue); + } + break; + case GSA_16_BIT_ROM_PATCH2C: + case GSA_16_BIT_ROM_PATCH2D: + case GSA_16_BIT_ROM_PATCH2E: + case GSA_16_BIT_ROM_PATCH2F: + if(cheatsList[i].status & 1) { + cheatsList[i].status &= ~1; + } + break; + case MASTER_CODE: + mastercode=0; + break; + } + cheatsList[i].enabled = false; + } +} + +bool cheatsVerifyCheatCode(const char *code, const char *desc) +{ + size_t len = strlen(code); + if(len != 11 && len != 13 && len != 17) { + systemMessage(MSG_INVALID_CHEAT_CODE, N_("Invalid cheat code '%s'"), code); + return false; + } + + if(code[8] != ':') { + systemMessage(MSG_INVALID_CHEAT_CODE, N_("Invalid cheat code '%s'"), code); + return false; + } + + size_t i; + for(i = 0; i < 8; i++) { + if(!CHEAT_IS_HEX(code[i])) { + // wrong cheat + systemMessage(MSG_INVALID_CHEAT_CODE, + N_("Invalid cheat code '%s'"), code); + return false; + } + } + for(i = 9; i < len; i++) { + if(!CHEAT_IS_HEX(code[i])) { + // wrong cheat + systemMessage(MSG_INVALID_CHEAT_CODE, + N_("Invalid cheat code '%s'"), code); + return false; + } + } + + u32 address = 0; + u32 value = 0; + + char buffer[10]; + strncpy(buffer, code, 8); + buffer[8] = 0; + sscanf(buffer, "%x", &address); + + switch(address >> 24) { + case 0x02: + case 0x03: + case 0x04: + case 0x05: + case 0x06: + case 0x07: + case 0x08: + case 0x09: + case 0x0A: + case 0x0B: + case 0x0C: + case 0x0D: + break; + default: + systemMessage(MSG_INVALID_CHEAT_CODE_ADDRESS, + N_("Invalid cheat code address: %08x"), + address); + return false; + } + + strncpy(buffer, &code[9], 8); + sscanf(buffer, "%x", &value); + int type = 0; + if(len == 13) + type = 114; + if(len == 17) + type = 115; + cheatsAdd(code, desc, address, address, value, type, type); + return true; +} + +void cheatsAddCheatCode(const char *code, const char *desc) +{ + cheatsVerifyCheatCode(code, desc); +} + +u16 cheatsGSAGetDeadface(bool v3) +{ + for(int i = cheatsNumber-1; i >= 0; i--) + if ((cheatsList[i].address == 0xDEADFACE) && (cheatsList[i].code == (v3 ? 257 : 256))) + return cheatsList[i].value & 0xFFFF; + return 0; +} + +void cheatsGSAChangeEncryption(u16 value, bool v3) { + int i; + u8 *deadtable1, *deadtable2; + + if (v3) { + deadtable1 = (u8*)(&v3_deadtable1); + deadtable2 = (u8*)(&v3_deadtable2); + for (i = 0; i < 4; i++) + seeds_v3[i] = seed_gen(((value & 0xFF00) >> 8), (value & 0xFF) + i, deadtable1, deadtable2); + } + else { + deadtable1 = (u8*)(&v1_deadtable1); + deadtable2 = (u8*)(&v1_deadtable2); + for (i = 0; i < 4; i++){ + seeds_v1[i] = seed_gen(((value & 0xFF00) >> 8), (value & 0xFF) + i, deadtable1, deadtable2); + } + } +} + +u32 seed_gen(u8 upper, u8 seed, u8 *deadtable1, u8 *deadtable2) { + int i; + u32 newseed = 0; + + for (i = 0; i < 4; i++) + newseed = ((newseed << 8) | ((deadtable1[(i + upper) & 0xFF] + deadtable2[seed]) & 0xFF)); + + return newseed; +} + +void cheatsDecryptGSACode(u32& address, u32& value, bool v3) +{ + u32 rollingseed = 0xC6EF3720; + u32 *seeds = v3 ? seeds_v3 : seeds_v1; + + int bitsleft = 32; + while (bitsleft > 0) { + value -= ((((address << 4) + seeds[2]) ^ (address + rollingseed)) ^ + ((address >> 5) + seeds[3])); + address -= ((((value << 4) + seeds[0]) ^ (value + rollingseed)) ^ + ((value >> 5) + seeds[1])); + rollingseed -= 0x9E3779B9; + bitsleft--; + } +} + +void cheatsAddGSACode(const char *code, const char *desc, bool v3) +{ + if(strlen(code) != 16) { + // wrong cheat + systemMessage(MSG_INVALID_GSA_CODE, + N_("Invalid GSA code. Format is XXXXXXXXYYYYYYYY")); + return; + } + + int i; + for(i = 0; i < 16; i++) { + if(!CHEAT_IS_HEX(code[i])) { + // wrong cheat + systemMessage(MSG_INVALID_GSA_CODE, + N_("Invalid GSA code. Format is XXXXXXXXYYYYYYYY")); + return; + } + } + + char buffer[10]; + strncpy(buffer, code, 8); + buffer[8] = 0; + u32 address; + sscanf(buffer, "%x", &address); + strncpy(buffer, &code[8], 8); + buffer[8] = 0; + u32 value; + sscanf(buffer, "%x", &value); + cheatsGSAChangeEncryption(cheatsGSAGetDeadface (v3), v3); + cheatsDecryptGSACode(address, value, v3); + + if(value == 0x1DC0DE) { + u32 gamecode = READ32LE(((u32 *)&rom[0xac])); + if(gamecode != address) { + char buffer[5]; + *((u32 *)buffer) = address; + buffer[4] = 0; + char buffer2[5]; + *((u32 *)buffer2) = READ32LE(((u32 *)&rom[0xac])); + buffer2[4] = 0; + systemMessage(MSG_GBA_CODE_WARNING, N_("Warning: cheats are for game %s. Current game is %s.\nCodes may not work correctly."), + buffer, buffer2); + } + cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value, v3 ? 257 : 256, + UNKNOWN_CODE); + return; + } + if(isMultilineWithData(cheatsNumber-1)) { + cheatsAdd(code, desc, address, address, value, v3 ? 257 : 256, UNKNOWN_CODE); + return; + } + if(v3) { + int type = ((address >> 25) & 127) | ((address >> 17) & 0x80); + u32 addr = (address & 0x00F00000) << 4 | (address & 0x0003FFFF); + u16 mcode = (address>>24 & 0xFF); + + if ((mcode & 0xFE) == 0xC4) + { + cheatsAdd(code, desc, address, (address & 0x1FFFFFF) | (0x08000000), + value, 257, MASTER_CODE); + mastercode = (address & 0x1FFFFFF) | (0x08000000); + } + else + switch(type) { + case 0x00: + if(address == 0) { + type = (value >> 25) & 127; + addr = (value & 0x00F00000) << 4 | (value & 0x0003FFFF); + switch(type) { + case 0x04: + cheatsAdd(code, desc, address, 0, value & 0x00FFFFFF, 257, GSA_SLOWDOWN); + break; + case 0x08: + cheatsAdd(code, desc, address, 0, addr, 257, GSA_8_BIT_GS_WRITE2); + break; + case 0x09: + cheatsAdd(code, desc, address, 0, addr, 257, GSA_16_BIT_GS_WRITE2); + break; + case 0x0a: + cheatsAdd(code, desc, address, 0, addr, 257, GSA_32_BIT_GS_WRITE2); + break; + case 0x0c: + cheatsAdd(code, desc, address, 0, value & 0x00FFFFFF, 257, GSA_16_BIT_ROM_PATCH2C); + break; + case 0x0d: + cheatsAdd(code, desc, address, 0, value & 0x00FFFFFF, 257, GSA_16_BIT_ROM_PATCH2D); + break; + case 0x0e: + cheatsAdd(code, desc, address, 0, value & 0x00FFFFFF, 257, GSA_16_BIT_ROM_PATCH2E); + break; + case 0x0f: + cheatsAdd(code, desc, address, 0, value & 0x00FFFFFF, 257, GSA_16_BIT_ROM_PATCH2F); + break; + case 0x20: + cheatsAdd(code, desc, address, 0, addr, 257, GSA_CODES_ON); + break; + case 0x40: + cheatsAdd(code, desc, address, 0, addr, 257, GSA_8_BIT_SLIDE); + break; + case 0x41: + cheatsAdd(code, desc, address, 0, addr, 257, GSA_16_BIT_SLIDE); + break; + case 0x42: + cheatsAdd(code, desc, address, 0, addr, 257, GSA_32_BIT_SLIDE); + break; + default: + cheatsAdd(code, desc, address, address, value, 257, UNKNOWN_CODE); + break; + } + } else + cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_FILL); + break; + case 0x01: + cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_FILL); + break; + case 0x02: + cheatsAdd(code, desc, address, addr, value, 257, INT_32_BIT_WRITE); + break; + case 0x04: + cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_TRUE); + break; + case 0x05: + cheatsAdd(code, desc, address, addr, value, 257, CBA_IF_TRUE); + break; + case 0x06: + cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_TRUE); + break; + case 0x07: + cheatsAdd(code, desc, address, addr, value, 257, GSA_ALWAYS); + break; + case 0x08: + cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_FALSE); + break; + case 0x09: + cheatsAdd(code, desc, address, addr, value, 257, CBA_IF_FALSE); + break; + case 0x0a: + cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_FALSE); + break; + case 0xc: + cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_LOWER_S); + break; + case 0xd: + cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_LOWER_S); + break; + case 0xe: + cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_LOWER_S); + break; + case 0x10: + cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_HIGHER_S); + break; + case 0x11: + cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_HIGHER_S); + break; + case 0x12: + cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_HIGHER_S); + break; + case 0x14: + cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_LOWER_U); + break; + case 0x15: + cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_LOWER_U); + break; + case 0x16: + cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_LOWER_U); + break; + case 0x18: + cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_HIGHER_U); + break; + case 0x19: + cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_HIGHER_U); + break; + case 0x1A: + cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_HIGHER_U); + break; + case 0x1C: + cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_AND); + break; + case 0x1D: + cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_AND); + break; + case 0x1E: + cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_AND); + break; + case 0x20: + cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_POINTER); + break; + case 0x21: + cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_POINTER); + break; + case 0x22: + cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_POINTER); + break; + case 0x24: + cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_TRUE2); + break; + case 0x25: + cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_TRUE2); + break; + case 0x26: + cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_TRUE2); + break; + case 0x27: + cheatsAdd(code, desc, address, addr, value, 257, GSA_ALWAYS2); + break; + case 0x28: + cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_FALSE2); + break; + case 0x29: + cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_FALSE2); + break; + case 0x2a: + cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_FALSE2); + break; + case 0x2c: + cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_LOWER_S2); + break; + case 0x2d: + cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_LOWER_S2); + break; + case 0x2e: + cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_LOWER_S2); + break; + case 0x30: + cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_HIGHER_S2); + break; + case 0x31: + cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_HIGHER_S2); + break; + case 0x32: + cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_HIGHER_S2); + break; + case 0x34: + cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_LOWER_U2); + break; + case 0x35: + cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_LOWER_U2); + break; + case 0x36: + cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_LOWER_U2); + break; + case 0x38: + cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_HIGHER_U2); + break; + case 0x39: + cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_HIGHER_U2); + break; + case 0x3A: + cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_HIGHER_U2); + break; + case 0x3C: + cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_AND2); + break; + case 0x3D: + cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_AND2); + break; + case 0x3E: + cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_AND2); + break; + case 0x40: + cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_ADD); + break; + case 0x41: + cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_ADD); + break; + case 0x42: + cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_ADD); + break; + case 0x44: + cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_TRUE3); + break; + case 0x45: + cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_TRUE3); + break; + case 0x46: + cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_TRUE3); + break; + case 0x47: + cheatsAdd(code, desc, address, addr, value, 257, GSA_ALWAYS3); + break; + case 0x48: + cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_FALSE3); + break; + case 0x49: + cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_FALSE3); + break; + case 0x4a: + cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_FALSE3); + break; + case 0x4c: + cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_LOWER_S3); + break; + case 0x4d: + cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_LOWER_S3); + break; + case 0x4e: + cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_LOWER_S3); + break; + case 0x50: + cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_HIGHER_S3); + break; + case 0x51: + cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_HIGHER_S3); + break; + case 0x52: + cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_HIGHER_S3); + break; + case 0x54: + cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_LOWER_U3); + break; + case 0x55: + cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_LOWER_U3); + break; + case 0x56: + cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_LOWER_U3); + break; + case 0x58: + cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_HIGHER_U3); + break; + case 0x59: + cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_HIGHER_U3); + break; + case 0x5a: + cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_HIGHER_U3); + break; + case 0x5c: + cheatsAdd(code, desc, address, addr, value, 257, GSA_8_BIT_IF_AND3); + break; + case 0x5d: + cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_IF_AND3); + break; + case 0x5e: + cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_IF_AND3); + break; + case 0x63: + cheatsAdd(code, desc, address, addr, value, 257, GSA_16_BIT_WRITE_IOREGS); + break; + case 0xE3: + cheatsAdd(code, desc, address, addr, value, 257, GSA_32_BIT_WRITE_IOREGS); + break; + default: + cheatsAdd(code, desc, address, address, value, 257, UNKNOWN_CODE); + break; + } + } else { + int type = (address >> 28) & 15; + switch(type) { + case 0: + case 1: + case 2: + cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value, 256, type); + break; + case 3: + switch ((address >> 0x10) & 0xFF){ + case 0x00: + cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value, 256, GSA_GROUP_WRITE); + break; + case 0x10: + cheatsAdd(code, desc, address, value & 0x0FFFFFFF, address & 0xFF, 256, GSA_32_BIT_ADD ); + break; + case 0x20: + cheatsAdd(code, desc, address, value & 0x0FFFFFFF, (~(address & 0xFF)+1), 256, GSA_32_BIT_ADD ); + break; + case 0x30: + cheatsAdd(code, desc, address, value & 0x0FFFFFFF, address & 0xFFFF, 256, GSA_32_BIT_ADD ); + break; + case 0x40: + cheatsAdd(code, desc, address, value & 0x0FFFFFFF, (~(address & 0xFFFF)+1), 256, GSA_32_BIT_ADD ); + break; + case 0x50: + cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value, 256, GSA_32_BIT_ADD2); + break; + case 0x60: + cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value, 256, GSA_32_BIT_SUB2); + break; + default: + // unsupported code + cheatsAdd(code, desc, address, address, value, 256, + UNKNOWN_CODE); + break; + } + break; + case 6: + address <<= 1; + type = (value >> 24) & 0xFF; + if(type == 0x00) { + cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value & 0xFFFF, 256, + GSA_16_BIT_ROM_PATCH); + break; + } + // unsupported code + cheatsAdd(code, desc, address, address, value, 256, + UNKNOWN_CODE); + break; + case 8: + switch((address >> 20) & 15) { + case 1: + cheatsAdd(code, desc, address, address & 0x0F0FFFFF, value, 256, + GSA_8_BIT_GS_WRITE); + break; + case 2: + cheatsAdd(code, desc, address, address & 0x0F0FFFFF, value, 256, + GSA_16_BIT_GS_WRITE); + break; + case 4: + // This code is buggy : the value is always set to 0 ! + cheatsAdd(code, desc, address, address & 0x0F0FFFFF, 0, 256, + GSA_32_BIT_GS_WRITE); + break; + case 15: + cheatsAdd(code, desc, address, 0, value & 0xFFFF, 256, GSA_SLOWDOWN); + break; + default: + // unsupported code + cheatsAdd(code, desc, address, address, value, 256, + UNKNOWN_CODE); + break; + } + break; + case 0x0d: + if(address != 0xDEADFACE) { + switch((value >> 20) & 0xF) { + case 0: + cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value & 0xFFFF, 256, + CBA_IF_TRUE); + break; + case 1: + cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value & 0xFFFF, 256, + CBA_IF_FALSE); + break; + case 2: + cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value & 0xFFFF, 256, + GSA_16_BIT_IF_LOWER_OR_EQ_U); + break; + case 3: + cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value & 0xFFFF, 256, + GSA_16_BIT_IF_HIGHER_OR_EQ_U); + break; + default: + // unsupported code + cheatsAdd(code, desc, address, address, value, 256, + UNKNOWN_CODE); + break; + } + } else + cheatsAdd(code, desc, address, address, value, 256, + UNKNOWN_CODE); + break; + case 0x0e: + switch((value >> 28) & 0xF) { + case 0: + cheatsAdd(code, desc, address, value & 0x0FFFFFFF, address & 0xFFFF, 256, + GSA_16_BIT_MIF_TRUE); + break; + case 1: + cheatsAdd(code, desc, address, value & 0x0FFFFFFF, address & 0xFFFF, 256, + GSA_16_BIT_MIF_FALSE); + break; + case 2: + cheatsAdd(code, desc, address, value & 0x0FFFFFFF, address & 0xFFFF, 256, + GSA_16_BIT_MIF_LOWER_OR_EQ_U); + break; + case 3: + cheatsAdd(code, desc, address, value & 0x0FFFFFFF, address & 0xFFFF, 256, + GSA_16_BIT_MIF_HIGHER_OR_EQ_U); + break; + default: + // unsupported code + cheatsAdd(code, desc, address, address, value, 256, + UNKNOWN_CODE); + break; + } + break; + case 0x0f: + cheatsAdd(code, desc, address, (address & 0xFFFFFFF), value, 256, MASTER_CODE); + mastercode = (address & 0xFFFFFFF); + break; + default: + // unsupported code + cheatsAdd(code, desc, address, address, value, 256, + UNKNOWN_CODE); + break; + } + } +} + +bool cheatsImportGSACodeFile(const char *name, int game, bool v3) +{ + FILE *f = fopen(name, "rb"); + if(!f) + return false; + + int games = 0; + int len = 0; + fseek(f, 0x1e, SEEK_CUR); + fread(&games, 1, 4, f); + bool found = false; + int g = 0; + while(games > 0) { + if(g == game) { + found = true; + break; + } + fread(&len, 1, 4, f); + fseek(f,len,SEEK_CUR); + int codes = 0; + fread(&codes, 1, 4, f); + while(codes > 0) { + fread(&len, 1, 4, f); + fseek(f, len, SEEK_CUR); + fseek(f, 8, SEEK_CUR); + fread(&len, 1, 4, f); + fseek(f, len*12, SEEK_CUR); + codes--; + } + games--; + g++; + } + if(found) { + char desc[256]; + char code[17]; + fread(&len, 1, 4, f); + fseek(f, len, SEEK_CUR); + int codes = 0; + fread(&codes, 1, 4, f); + while(codes > 0) { + fread(&len, 1, 4, f); + fread(desc, 1, len, f); + desc[len] =0; + desc[31] = 0; + fread(&len, 1, 4, f); + fseek(f, len, SEEK_CUR); + fseek(f, 4, SEEK_CUR); + fread(&len, 1, 4, f); + while(len) { + fseek(f, 4, SEEK_CUR); + fread(code, 1, 8, f); + fseek(f, 4, SEEK_CUR); + fread(&code[8], 1, 8, f); + code[16] = 0; + cheatsAddGSACode(code, desc, v3); + len -= 2; + } + codes--; + } + } + fclose(f); + return false; +} + +void cheatsCBAReverseArray(u8 *array, u8 *dest) +{ + dest[0] = array[3]; + dest[1] = array[2]; + dest[2] = array[1]; + dest[3] = array[0]; + dest[4] = array[5]; + dest[5] = array[4]; +} + +void chatsCBAScramble(u8 *array, int count, u8 b) +{ + u8 *x = array + (count >> 3); + u8 *y = array + (b >> 3); + u32 z = *x & (1 << (count & 7)); + u32 x0 = (*x & (~(1 << (count & 7)))); + if (z != 0) + z = 1; + if ((*y & (1 << (b & 7))) != 0) + x0 |= (1 << (count & 7)); + *x = x0; + u32 temp = *y & (~(1 << (b & 7))); + if (z != 0) + temp |= (1 << (b & 7)); + *y = temp; +} + +u32 cheatsCBAGetValue(u8 *array) +{ + return array[0] | array[1]<<8 | array[2] << 16 | array[3]<<24; +} + +u16 cheatsCBAGetData(u8 *array) +{ + return array[4] | array[5]<<8; +} + +void cheatsCBAArrayToValue(u8 *array, u8 *dest) +{ + dest[0] = array[3]; + dest[1] = array[2]; + dest[2] = array[1]; + dest[3] = array[0]; + dest[4] = array[5]; + dest[5] = array[4]; +} + +void cheatsCBAParseSeedCode(u32 address, u32 value, u32 *array) +{ + array[0] = 1; + array[1] = value & 0xFF; + array[2] = (address >> 0x10) & 0xFF; + array[3] = (value >> 8) & 0xFF; + array[4] = (address >> 0x18) & 0x0F; + array[5] = address & 0xFFFF; + array[6] = address; + array[7] = value; +} + +u32 cheatsCBAEncWorker() +{ + u32 x = (cheatsCBATemporaryValue * 0x41c64e6d) + 0x3039; + u32 y = (x * 0x41c64e6d) + 0x3039; + u32 z = x >> 0x10; + x = ((y >> 0x10) & 0x7fff) << 0x0f; + z = (z << 0x1e) | x; + x = (y * 0x41c64e6d) + 0x3039; + cheatsCBATemporaryValue = x; + return z | ((x >> 0x10) & 0x7fff); +} + +#define ROR(v, s) \ + (((v) >> (s)) | (((v) & ((1 << (s))-1)) << (32 - (s)))) + +u32 cheatsCBACalcIndex(u32 x, u32 y) +{ + if(y != 0) { + if(y == 1) + x = 0; + else if(x == y) + x = 0; + if(y < 1) + return x; + else if(x < y) + return x; + u32 x0 = 1; + + while(y < 0x10000000) { + if(y < x) { + y = y << 4; + x0 = x0 << 4; + } else break; + } + + while(y < 0x80000000) { + if(y < x) { + y = y << 1; + x0 = x0 << 1; + } else break; + } + + loop: + u32 z = 0; + if(x >= y) + x -= y; + if(x >= (y >> 1)) { + x -= (y >> 1); + z |= ROR(x0, 1); + } + if(x >= (y >> 2)) { + x -= (y >> 2); + z |= ROR(x0, 2); + } + if(x >= (y >> 3)) { + x -= (y >> 3); + z |= ROR(x0, 3); + } + + u32 temp = x0; + + if(x != 0) { + x0 = x0 >> 4; + if(x0 != 0) { + y = y >> 4; + goto loop; + } + } + + z = z & 0xe0000000; + + if(z != 0) { + if((temp & 7) == 0) + return x; + } else + return x; + + if((z & ROR(temp, 3)) != 0) + x += y >> 3; + if((z & ROR(temp, 2)) != 0) + x += y >> 2; + if((z & ROR(temp, 1)) != 0) + x += y >> 1; + return x; + } else { + } + // should not happen in the current code + return 0; +} + +void cheatsCBAUpdateSeedBuffer(u32 a, u8 *buffer, int count) +{ + int i; + for(i = 0; i < count; i++) + buffer[i] = i; + for(i = 0; (u32)i < a; i++) { + u32 a = cheatsCBACalcIndex(cheatsCBAEncWorker(), count); + u32 b = cheatsCBACalcIndex(cheatsCBAEncWorker(), count); + u32 t = buffer[a]; + buffer[a] = buffer[b]; + buffer[b] = t; + } +} + +void cheatsCBAChangeEncryption(u32 *seed) +{ + int i; + + cheatsCBATemporaryValue = (seed[1] ^ 0x1111); + cheatsCBAUpdateSeedBuffer(0x50, cheatsCBASeedBuffer, 0x30); + cheatsCBATemporaryValue = 0x4efad1c3; + + for(i = 0; (u32)i < seed[4]; i++) { + cheatsCBATemporaryValue = cheatsCBAEncWorker(); + } + cheatsCBASeed[2] = cheatsCBAEncWorker(); + cheatsCBASeed[3] = cheatsCBAEncWorker(); + + cheatsCBATemporaryValue = seed[3] ^ 0xf254; + + for(i = 0; (u32)i < seed[3]; i++) { + cheatsCBATemporaryValue = cheatsCBAEncWorker(); + } + + cheatsCBASeed[0] = cheatsCBAEncWorker(); + cheatsCBASeed[1] = cheatsCBAEncWorker(); + + *((u32 *)&cheatsCBACurrentSeed[0]) = seed[6]; + *((u32 *)&cheatsCBACurrentSeed[4]) = seed[7]; + *((u32 *)&cheatsCBACurrentSeed[8]) = 0; +} + +u16 cheatsCBAGenValue(u32 x, u32 y, u32 z) +{ + y <<= 0x10; + z <<= 0x10; + x <<= 0x18; + u32 x0 = (int)y >> 0x10; + z = (int)z >> 0x10; + x = (int)x >> 0x10; + for(int i = 0; i < 8; i++) { + u32 temp = z ^ x; + if ((int)temp >= 0) { + temp = z << 0x11; + } + else { + temp = z << 0x01; + temp ^= x0; + temp = temp << 0x10; + } + z = (int)temp >> 0x10; + temp = x << 0x11; + x = (int)temp >> 0x10; + } + return z & 0xffff; +} + +void cheatsCBAGenTable() { + for (int i = 0; i < 0x100; i++) { + cheatsCBATable[i] = cheatsCBAGenValue(i, 0x1021, 0); + } + cheatsCBATableGenerated = true; +} + +u16 cheatsCBACalcCRC(u8 *rom, int count) +{ + u32 crc = 0xffffffff; + + if (count & 3) { + // 0x08000EAE + } else { + count = (count >> 2) - 1; + if(count != -1) { + while(count != -1) { + crc = (((crc << 0x08) ^ cheatsCBATable[(((u32)crc << 0x10) >> 0x18) + ^ *rom++]) << 0x10) >> 0x10; + crc = (((crc << 0x08) ^ cheatsCBATable[(((u32)crc << 0x10) >> 0x18) + ^ *rom++]) << 0x10) >> 0x10; + crc = (((crc << 0x08) ^ cheatsCBATable[(((u32)crc << 0x10) >> 0x18) + ^ *rom++]) << 0x10) >> 0x10; + crc = (((crc << 0x08) ^ cheatsCBATable[(((u32)crc << 0x10) >> 0x18) + ^ *rom++]) << 0x10) >> 0x10; + count--; + } + } + } + return crc & 0xffff; +} + +void cheatsCBADecrypt(u8 *decrypt) +{ + u8 buffer[8] = { 0, 0, 0, 0, 0, 0, 0, 0 }; + u8 *array = &buffer[1]; + + cheatsCBAReverseArray(decrypt, array); + + for(int count = 0x2f; count >= 0; count--) { + chatsCBAScramble(array, count, cheatsCBASeedBuffer[count]); + } + cheatsCBAArrayToValue(array, decrypt); + *((u32 *)decrypt) = cheatsCBAGetValue(decrypt) ^ + cheatsCBASeed[0]; + *((u16 *)(decrypt+4)) = (cheatsCBAGetData(decrypt) ^ + cheatsCBASeed[1]) & 0xffff; + + cheatsCBAReverseArray(decrypt, array); + + u32 cs = cheatsCBAGetValue(cheatsCBACurrentSeed); + for(int i = 0; i <= 4; i++) { + array[i] = ((cs >> 8) ^ array[i+1]) ^ array[i] ; + } + + array[5] = (cs >> 8) ^ array[5]; + + for(int j = 5; j >=0; j--) { + array[j] = (cs ^ array[j-1]) ^ array[j]; + } + + cheatsCBAArrayToValue(array, decrypt); + + *((u32 *)decrypt) = cheatsCBAGetValue(decrypt) + ^ cheatsCBASeed[2]; + *((u16 *)(decrypt+4)) = (cheatsCBAGetData(decrypt) + ^ cheatsCBASeed[3]) & 0xffff; +} + +int cheatsCBAGetCount() +{ + int count = 0; + for(int i = 0; i < cheatsNumber; i++) { + if(cheatsList[i].code == 512) + count++; + } + return count; +} + +bool cheatsCBAShouldDecrypt() +{ + for(int i = 0; i < cheatsNumber; i++) { + if(cheatsList[i].code == 512) { + return (cheatsList[i].codestring[0] == '9'); + } + } + return false; +} + +void cheatsAddCBACode(const char *code, const char *desc) +{ + if(strlen(code) != 13) { + // wrong cheat + systemMessage(MSG_INVALID_CBA_CODE, + N_("Invalid CBA code. Format is XXXXXXXX YYYY.")); + return; + } + + int i; + for(i = 0; i < 8; i++) { + if(!CHEAT_IS_HEX(code[i])) { + // wrong cheat + systemMessage(MSG_INVALID_CBA_CODE, + N_("Invalid CBA code. Format is XXXXXXXX YYYY.")); + return; + } + } + + if(code[8] != ' ') { + systemMessage(MSG_INVALID_CBA_CODE, + N_("Invalid CBA code. Format is XXXXXXXX YYYY.")); + return; + } + + for(i = 9; i < 13; i++) { + if(!CHEAT_IS_HEX(code[i])) { + // wrong cheat + systemMessage(MSG_INVALID_CBA_CODE, + N_("Invalid CBA code. Format is XXXXXXXX YYYY.")); + return; + } + } + + char buffer[10]; + strncpy(buffer, code, 8); + buffer[8] = 0; + u32 address; + sscanf(buffer, "%x", &address); + strncpy(buffer, &code[9], 4); + buffer[4] = 0; + u32 value; + sscanf(buffer, "%x", &value); + + u8 array[8] = { + address & 255, + (address >> 8) & 255, + (address >> 16) & 255, + (address >> 24) & 255, + (value & 255), + (value >> 8) & 255, + 0, + 0 + }; + + if(cheatsCBAGetCount() == 0 && + (address >> 28) == 9) { + u32 seed[8]; + cheatsCBAParseSeedCode(address, value, seed); + cheatsCBAChangeEncryption(seed); + cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value, 512, UNKNOWN_CODE); + } else { + if(cheatsCBAShouldDecrypt()) + cheatsCBADecrypt(array); + + address = READ32LE(((u32 *)array)); + value = READ16LE(((u16 *)&array[4])); + + int type = (address >> 28) & 15; + + if(isMultilineWithData(cheatsNumber-1) || (super>0)) { + cheatsAdd(code, desc, address, address, value, 512, UNKNOWN_CODE); + if (super>0) + super-= 1; + return; + } + + switch(type) { + case 0x00: + { + if(!cheatsCBATableGenerated) + cheatsCBAGenTable(); + u32 crc = cheatsCBACalcCRC(rom, 0x10000); + if(crc != address) { + systemMessage(MSG_CBA_CODE_WARNING, + N_("Warning: Codes seem to be for a different game.\nCodes may not work correctly.")); + } + cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value, 512, + UNKNOWN_CODE); + } + break; + case 0x01: + cheatsAdd(code, desc, address, (address & 0x1FFFFFF) | 0x08000000, value, 512, MASTER_CODE); + mastercode = (address & 0x1FFFFFF) | 0x08000000; + break; + case 0x02: + cheatsAdd(code, desc, address, address & 0x0FFFFFFE, value, 512, + CBA_OR); + break; + case 0x03: + cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value, 512, + INT_8_BIT_WRITE); + break; + case 0x04: + cheatsAdd(code, desc, address, address & 0x0FFFFFFE, value, 512, + CBA_SLIDE_CODE); + break; + case 0x05: + cheatsAdd(code, desc, address, address & 0x0FFFFFFE, value, 512, + CBA_SUPER); + super = getCodeLength(cheatsNumber-1); + break; + case 0x06: + cheatsAdd(code, desc, address, address & 0x0FFFFFFE, value, 512, + CBA_AND); + break; + case 0x07: + cheatsAdd(code, desc, address, address & 0x0FFFFFFE, value, 512, + CBA_IF_TRUE); + break; + case 0x08: + cheatsAdd(code, desc, address, address & 0x0FFFFFFE, value, 512, + INT_16_BIT_WRITE); + break; + case 0x0a: + cheatsAdd(code, desc, address, address & 0x0FFFFFFE, value, 512, + CBA_IF_FALSE); + break; + case 0x0b: + cheatsAdd(code, desc, address, address & 0x0FFFFFFE, value, 512, + CBA_GT); + break; + case 0x0c: + cheatsAdd(code, desc, address, address & 0x0FFFFFFE, value, 512, + CBA_LT); + break; + case 0x0d: + if ((address & 0xF0)<0x30) + cheatsAdd(code, desc, address, address & 0xF0, value, 512, + CBA_IF_KEYS_PRESSED); + break; + case 0x0e: + cheatsAdd(code, desc, address, address & 0x0FFFFFFF, value & 0x8000 ? value | 0xFFFF0000 : value, 512, + CBA_ADD); + break; + case 0x0f: + cheatsAdd(code, desc, address, address & 0x0FFFFFFE, value, 512, + GSA_16_BIT_IF_AND); + break; + default: + // unsupported code + cheatsAdd(code, desc, address, address & 0xFFFFFFFF, value, 512, + UNKNOWN_CODE); + break; + } + } +} + +void cheatsSaveGame(gzFile file) +{ + utilWriteInt(file, cheatsNumber); + + utilGzWrite(file, cheatsList, sizeof(cheatsList)); +} + +void cheatsReadGame(gzFile file, int version) +{ + cheatsNumber = 0; + + cheatsNumber = utilReadInt(file); + + if (version > 8) + utilGzRead(file, cheatsList, sizeof(cheatsList)); + + + bool firstCodeBreaker = true; + + for(int i = 0; i < cheatsNumber; i++) { + if (version <9) + { + cheatsList[i].code = utilReadInt(file); + cheatsList[i].size = utilReadInt(file); + cheatsList[i].status = utilReadInt(file); + cheatsList[i].enabled = utilReadInt(file) ? true : false; + utilGzRead(file, &cheatsList[i].address, sizeof(u32)); + cheatsList[i].rawaddress = cheatsList[i].address; + utilGzRead(file, &cheatsList[i].value, sizeof(u32)); + utilGzRead(file, &cheatsList[i].oldValue, sizeof(u32)); + utilGzRead(file, &cheatsList[i].codestring, 20*sizeof(char)); + utilGzRead(file, &cheatsList[i].desc, 32*sizeof(char)); + } + + cheatsList[i].status = 0; + if(!cheatsList[i].codestring[0]) { + switch(cheatsList[i].size) { + case 0: + sprintf(cheatsList[i].codestring, "%08x:%02x", cheatsList[i].address, + cheatsList[i].value); + break; + case 1: + sprintf(cheatsList[i].codestring, "%08x:%04x", cheatsList[i].address, + cheatsList[i].value); + break; + case 2: + sprintf(cheatsList[i].codestring, "%08x:%08x", cheatsList[i].address, + cheatsList[i].value); + break; + } + } + + if(cheatsList[i].enabled) { + cheatsEnable(i); + } + + if(cheatsList[i].code == 512 && firstCodeBreaker) { + firstCodeBreaker = false; + char buffer[10]; + strncpy(buffer, cheatsList[i].codestring, 8); + buffer[8] = 0; + u32 address; + sscanf(buffer, "%x", &address); + if((address >> 28) == 9) { + strncpy(buffer, &cheatsList[i].codestring[9], 4); + buffer[4] = 0; + u32 value; + sscanf(buffer, "%x", &value); + + u32 seed[8]; + cheatsCBAParseSeedCode(address, value, seed); + cheatsCBAChangeEncryption(seed); + } + } + } +} + +void cheatsSaveCheatList(const char *file) +{ + if(cheatsNumber == 0) + return; + FILE *f = fopen(file, "wb"); + if(f == NULL) + return; + int version = 1; + fwrite(&version, 1, sizeof(version), f); + int type = 1; + fwrite(&type, 1, sizeof(type), f); + fwrite(&cheatsNumber, 1, sizeof(cheatsNumber), f); + fwrite(cheatsList, 1, sizeof(cheatsList), f); + fclose(f); +} + +bool cheatsLoadCheatList(const char *file) +{ + + int count = 0; + + FILE *f = fopen(file, "rb"); + + if(f == NULL) + return false; + + int version = 0; + + if(fread(&version, 1, sizeof(version), f) != sizeof(version)) { + fclose(f); + return false; + } + + if(version != 1) { + systemMessage(MSG_UNSUPPORTED_CHEAT_LIST_VERSION, + N_("Unsupported cheat list version %d"), version); + fclose(f); + return false; + } + + int type = 0; + if(fread(&type, 1, sizeof(type), f) != sizeof(type)) { + fclose(f); + return false; + } + + + if((type != 0) && (type != 1)) { + systemMessage(MSG_UNSUPPORTED_CHEAT_LIST_TYPE, + N_("Unsupported cheat list type %d"), type); + fclose(f); + return false; + } + + if(fread(&count, 1, sizeof(count), f) != sizeof(count)) { + fclose(f); + return false; + } + if (type == 1) + { + if(fread(cheatsList, 1, sizeof(cheatsList), f) != sizeof(cheatsList)) { + fclose(f); + return false; + } + } + else if (type == 0) + { + for(int i = 0; i < count; i++) { + fread(&cheatsList[i].code, 1, sizeof(int),f); + fread(&cheatsList[i].size, 1, sizeof(int),f); + fread(&cheatsList[i].status, 1, sizeof(int),f); + fread(&cheatsList[i].enabled, 1, sizeof(int),f); + cheatsList[i].enabled = cheatsList[i].enabled ? true : false; + fread(&cheatsList[i].address, 1, sizeof(u32),f); + cheatsList[i].rawaddress = cheatsList[i].address; + fread(&cheatsList[i].value, 1, sizeof(u32),f); + fread(&cheatsList[i].oldValue, 1, sizeof(u32),f); + fread(&cheatsList[i].codestring, 1, 20*sizeof(char),f); + if(fread(&cheatsList[i].desc, 1, 32*sizeof(char),f) != 32*sizeof(char)) { + fclose(f); + return false; + } + } + } + + bool firstCodeBreaker = true; + + for(int i = 0; i < count; i++) { + cheatsList[i].status = 0; // remove old status as it is not used + if(!cheatsList[i].codestring[0]) { + switch(cheatsList[i].size) { + case 0: + sprintf(cheatsList[i].codestring, "%08x:%02x", cheatsList[i].address, + cheatsList[i].value); + break; + case 1: + sprintf(cheatsList[i].codestring, "%08x:%04x", cheatsList[i].address, + cheatsList[i].value); + break; + case 2: + sprintf(cheatsList[i].codestring, "%08x:%08x", cheatsList[i].address, + cheatsList[i].value); + break; + } + } + + if(cheatsList[i].code == 512 && firstCodeBreaker) { + firstCodeBreaker = false; + char buffer[10]; + strncpy(buffer, cheatsList[i].codestring, 8); + buffer[8] = 0; + u32 address; + sscanf(buffer, "%x", &address); + if((address >> 28) == 9) { + strncpy(buffer, &cheatsList[i].codestring[9], 4); + buffer[4] = 0; + u32 value; + sscanf(buffer, "%x", &value); + + u32 seed[8]; + cheatsCBAParseSeedCode(address, value, seed); + cheatsCBAChangeEncryption(seed); + } + } + } + cheatsNumber = count; + fclose(f); + return true; +} + +extern int cpuNextEvent; + +extern void debuggerBreakOnWrite(u32 , u32, u32, int, int); + +static u8 cheatsGetType(u32 address) +{ + switch(address >> 24) { + case 2: + return freezeWorkRAM[address & 0x3FFFF]; + case 3: + return freezeInternalRAM[address & 0x7FFF]; + case 5: + return freezePRAM[address & 0x3FC]; + case 6: + if (address > 0x06010000) + return freezeVRAM[address & 0x17FFF]; + else + return freezeVRAM[address & 0x1FFFF]; + case 7: + return freezeOAM[address & 0x3FC]; + } + return 0; +} + +void cheatsWriteMemory(u32 address, u32 value) +{ +#ifdef BKPT_SUPPORT +#ifdef SDL + if(cheatsNumber == 0) { + int type = cheatsGetType(address); + u32 oldValue = debuggerReadMemory(address); + if(type == 1 || (type == 2 && oldValue != value)) { + debuggerBreakOnWrite(address, oldValue, value, 2, type); + cpuNextEvent = 0; + } + debuggerWriteMemory(address, value); + } +#endif +#endif +} + +void cheatsWriteHalfWord(u32 address, u16 value) +{ +#ifdef BKPT_SUPPORT +#ifdef SDL + if(cheatsNumber == 0) { + int type = cheatsGetType(address); + u16 oldValue = debuggerReadHalfWord(address); + if(type == 1 || (type == 2 && oldValue != value)) { + debuggerBreakOnWrite(address, oldValue, value, 1, type); + cpuNextEvent = 0; + } + debuggerWriteHalfWord(address, value); + } +#endif +#endif +} + +#if defined BKPT_SUPPORT && defined SDL +void cheatsWriteByte(u32 address, u8 value) +#else +void cheatsWriteByte(u32, u8) +#endif +{ +#ifdef BKPT_SUPPORT +#ifdef SDL + if(cheatsNumber == 0) { + int type = cheatsGetType(address); + u8 oldValue = debuggerReadByte(address); + if(type == 1 || (type == 2 && oldValue != value)) { + debuggerBreakOnWrite(address, oldValue, value, 0, type); + cpuNextEvent = 0; + } + debuggerWriteByte(address, value); + } +#endif +#endif +} diff --git a/src/Cheats.h b/src/Cheats.h index ed828ee5..ce966872 100644 --- a/src/Cheats.h +++ b/src/Cheats.h @@ -1,55 +1,55 @@ -// -*- C++ -*- -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#ifndef GBA_CHEATS_H -#define GBA_CHEATS_H - -struct CheatsData { - int code; - int size; - int status; - bool enabled; - u32 rawaddress; - u32 address; - u32 value; - u32 oldValue; - char codestring[20]; - char desc[32]; -}; - -extern void cheatsAdd(const char *,const char *,u32, u32,u32,int,int); -extern void cheatsAddCheatCode(const char *code, const char *desc); -extern void cheatsAddGSACode(const char *code, const char *desc, bool v3); -extern void cheatsAddCBACode(const char *code, const char *desc); -extern bool cheatsImportGSACodeFile(const char *name, int game, bool v3); -extern void cheatsDelete(int number, bool restore); -extern void cheatsDeleteAll(bool restore); -extern void cheatsEnable(int number); -extern void cheatsDisable(int number); -extern void cheatsSaveGame(gzFile file); -extern void cheatsReadGame(gzFile file); -extern void cheatsSaveCheatList(const char *file); -extern bool cheatsLoadCheatList(const char *file); -extern void cheatsWriteMemory(u32, u32); -extern void cheatsWriteHalfWord(u32, u16); -extern void cheatsWriteByte(u32, u8); -extern int cheatsCheckKeys(u32,u32); -extern int cheatsNumber; -extern CheatsData cheatsList[100]; -#endif // GBA_CHEATS_H +// -*- C++ -*- +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004-2006 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#ifndef GBA_CHEATS_H +#define GBA_CHEATS_H + +struct CheatsData { + int code; + int size; + int status; + bool enabled; + u32 rawaddress; + u32 address; + u32 value; + u32 oldValue; + char codestring[20]; + char desc[32]; +}; + +extern void cheatsAdd(const char *,const char *,u32, u32,u32,int,int); +extern void cheatsAddCheatCode(const char *code, const char *desc); +extern void cheatsAddGSACode(const char *code, const char *desc, bool v3); +extern void cheatsAddCBACode(const char *code, const char *desc); +extern bool cheatsImportGSACodeFile(const char *name, int game, bool v3); +extern void cheatsDelete(int number, bool restore); +extern void cheatsDeleteAll(bool restore); +extern void cheatsEnable(int number); +extern void cheatsDisable(int number); +extern void cheatsSaveGame(gzFile file); +extern void cheatsReadGame(gzFile file, int version); +extern void cheatsSaveCheatList(const char *file); +extern bool cheatsLoadCheatList(const char *file); +extern void cheatsWriteMemory(u32, u32); +extern void cheatsWriteHalfWord(u32, u16); +extern void cheatsWriteByte(u32, u8); +extern int cheatsCheckKeys(u32,u32); +extern int cheatsNumber; +extern CheatsData cheatsList[100]; +#endif // GBA_CHEATS_H diff --git a/src/EEprom.cpp b/src/EEprom.cpp index 0992ae66..0e456c08 100644 --- a/src/EEprom.cpp +++ b/src/EEprom.cpp @@ -1,6 +1,6 @@ // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -16,6 +16,7 @@ // along with this program; if not, write to the Free Software Foundation, // Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. +#include #include "GBA.h" #include "EEprom.h" #include "Util.h" @@ -42,6 +43,11 @@ variable_desc eepromSaveData[] = { { NULL, 0 } }; +void eepromInit() +{ + memset(eepromData, 255, sizeof(eepromData)); +} + void eepromReset() { eepromMode = EEPROM_IDLE; diff --git a/src/EEprom.h b/src/EEprom.h index 4bf23afc..3418ec4a 100644 --- a/src/EEprom.h +++ b/src/EEprom.h @@ -1,7 +1,7 @@ // -*- C++ -*- // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -20,10 +20,11 @@ #ifndef VBA_EEPROM_H #define VBA_EEPROM_H -extern void eepromSaveGame(gzFile gzFile); -extern void eepromReadGame(gzFile gzFile, int version); +extern void eepromSaveGame(gzFile _gzFile); +extern void eepromReadGame(gzFile _gzFile, int version); extern int eepromRead(u32 address); extern void eepromWrite(u32 address, u8 value); +extern void eepromInit(); extern void eepromReset(); extern u8 eepromData[0x2000]; extern bool eepromInUse; diff --git a/src/Flash.cpp b/src/Flash.cpp index 0e9958a6..a04adc4a 100644 --- a/src/Flash.cpp +++ b/src/Flash.cpp @@ -1,259 +1,263 @@ -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#include -#include -#include "GBA.h" -#include "Globals.h" -#include "Flash.h" -#include "Sram.h" -#include "Util.h" - -#define FLASH_READ_ARRAY 0 -#define FLASH_CMD_1 1 -#define FLASH_CMD_2 2 -#define FLASH_AUTOSELECT 3 -#define FLASH_CMD_3 4 -#define FLASH_CMD_4 5 -#define FLASH_CMD_5 6 -#define FLASH_ERASE_COMPLETE 7 -#define FLASH_PROGRAM 8 -#define FLASH_SETBANK 9 - -u8 flashSaveMemory[0x20000]; -int flashState = FLASH_READ_ARRAY; -int flashReadState = FLASH_READ_ARRAY; -int flashSize = 0x10000; -int flashDeviceID = 0x1b; -int flashManufacturerID = 0x32; -int flashBank = 0; - -static variable_desc flashSaveData[] = { - { &flashState, sizeof(int) }, - { &flashReadState, sizeof(int) }, - { &flashSaveMemory[0], 0x10000 }, - { NULL, 0 } -}; - -static variable_desc flashSaveData2[] = { - { &flashState, sizeof(int) }, - { &flashReadState, sizeof(int) }, - { &flashSize, sizeof(int) }, - { &flashSaveMemory[0], 0x20000 }, - { NULL, 0 } -}; - -static variable_desc flashSaveData3[] = { - { &flashState, sizeof(int) }, - { &flashReadState, sizeof(int) }, - { &flashSize, sizeof(int) }, - { &flashBank, sizeof(int) }, - { &flashSaveMemory[0], 0x20000 }, - { NULL, 0 } -}; - -void flashInit() -{ - memset(flashSaveMemory, 0xff, sizeof(flashSaveMemory)); -} - -void flashReset() -{ - flashState = FLASH_READ_ARRAY; - flashReadState = FLASH_READ_ARRAY; - flashBank = 0; -} - -void flashSaveGame(gzFile gzFile) -{ - utilWriteData(gzFile, flashSaveData3); -} - -void flashReadGame(gzFile gzFile, int version) -{ - if(version < SAVE_GAME_VERSION_5) - utilReadData(gzFile, flashSaveData); - else if(version < SAVE_GAME_VERSION_7) { - utilReadData(gzFile, flashSaveData2); - flashBank = 0; - flashSetSize(flashSize); - } else { - utilReadData(gzFile, flashSaveData3); - } -} - -void flashSetSize(int size) -{ - // log("Setting flash size to %d\n", size); - flashSize = size; - if(size == 0x10000) { - flashDeviceID = 0x1b; - flashManufacturerID = 0x32; - } else { - flashDeviceID = 0x13; //0x09; - flashManufacturerID = 0x62; //0xc2; - } -} - -u8 flashRead(u32 address) -{ - // log("Reading %08x from %08x\n", address, reg[15].I); - // log("Current read state is %d\n", flashReadState); - address &= 0xFFFF; - - switch(flashReadState) { - case FLASH_READ_ARRAY: - return flashSaveMemory[(flashBank << 16) + address]; - case FLASH_AUTOSELECT: - switch(address & 0xFF) { - case 0: - // manufacturer ID - return flashManufacturerID; - case 1: - // device ID - return flashDeviceID; - } - break; - case FLASH_ERASE_COMPLETE: - flashState = FLASH_READ_ARRAY; - flashReadState = FLASH_READ_ARRAY; - return 0xFF; - }; - return 0; -} - -void flashSaveDecide(u32 address, u8 byte) -{ - // log("Deciding save type %08x\n", address); - if(address == 0x0e005555) { - saveType = 2; - cpuSaveGameFunc = flashWrite; - } else { - saveType = 1; - cpuSaveGameFunc = sramWrite; - } - - (*cpuSaveGameFunc)(address, byte); -} - -void flashDelayedWrite(u32 address, u8 byte) -{ - saveType = 2; - cpuSaveGameFunc = flashWrite; - flashWrite(address, byte); -} - -void flashWrite(u32 address, u8 byte) -{ - // log("Writing %02x at %08x\n", byte, address); - // log("Current state is %d\n", flashState); - address &= 0xFFFF; - switch(flashState) { - case FLASH_READ_ARRAY: - if(address == 0x5555 && byte == 0xAA) - flashState = FLASH_CMD_1; - break; - case FLASH_CMD_1: - if(address == 0x2AAA && byte == 0x55) - flashState = FLASH_CMD_2; - else - flashState = FLASH_READ_ARRAY; - break; - case FLASH_CMD_2: - if(address == 0x5555) { - if(byte == 0x90) { - flashState = FLASH_AUTOSELECT; - flashReadState = FLASH_AUTOSELECT; - } else if(byte == 0x80) { - flashState = FLASH_CMD_3; - } else if(byte == 0xF0) { - flashState = FLASH_READ_ARRAY; - flashReadState = FLASH_READ_ARRAY; - } else if(byte == 0xA0) { - flashState = FLASH_PROGRAM; - } else if(byte == 0xB0 && flashSize == 0x20000) { - flashState = FLASH_SETBANK; - } else { - flashState = FLASH_READ_ARRAY; - flashReadState = FLASH_READ_ARRAY; - } - } else { - flashState = FLASH_READ_ARRAY; - flashReadState = FLASH_READ_ARRAY; - } - break; - case FLASH_CMD_3: - if(address == 0x5555 && byte == 0xAA) { - flashState = FLASH_CMD_4; - } else { - flashState = FLASH_READ_ARRAY; - flashReadState = FLASH_READ_ARRAY; - } - break; - case FLASH_CMD_4: - if(address == 0x2AAA && byte == 0x55) { - flashState = FLASH_CMD_5; - } else { - flashState = FLASH_READ_ARRAY; - flashReadState = FLASH_READ_ARRAY; - } - break; - case FLASH_CMD_5: - if(byte == 0x30) { - // SECTOR ERASE - memset(&flashSaveMemory[(flashBank << 16) + (address & 0xF000)], - 0, - 0x1000); - systemSaveUpdateCounter = SYSTEM_SAVE_UPDATED; - flashReadState = FLASH_ERASE_COMPLETE; - } else if(byte == 0x10) { - // CHIP ERASE - memset(flashSaveMemory, 0, flashSize); - systemSaveUpdateCounter = SYSTEM_SAVE_UPDATED; - flashReadState = FLASH_ERASE_COMPLETE; - } else { - flashState = FLASH_READ_ARRAY; - flashReadState = FLASH_READ_ARRAY; - } - break; - case FLASH_AUTOSELECT: - if(byte == 0xF0) { - flashState = FLASH_READ_ARRAY; - flashReadState = FLASH_READ_ARRAY; - } else if(address == 0x5555 && byte == 0xAA) - flashState = FLASH_CMD_1; - else { - flashState = FLASH_READ_ARRAY; - flashReadState = FLASH_READ_ARRAY; - } - break; - case FLASH_PROGRAM: - flashSaveMemory[(flashBank<<16)+address] = byte; - systemSaveUpdateCounter = SYSTEM_SAVE_UPDATED; - flashState = FLASH_READ_ARRAY; - flashReadState = FLASH_READ_ARRAY; - break; - case FLASH_SETBANK: - if(address == 0) { - flashBank = (byte & 1); - } - flashState = FLASH_READ_ARRAY; - flashReadState = FLASH_READ_ARRAY; - break; - } -} \ No newline at end of file +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004-2006 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#include +#include +#include "GBA.h" +#include "Globals.h" +#include "Flash.h" +#include "Sram.h" +#include "Util.h" + +#define FLASH_READ_ARRAY 0 +#define FLASH_CMD_1 1 +#define FLASH_CMD_2 2 +#define FLASH_AUTOSELECT 3 +#define FLASH_CMD_3 4 +#define FLASH_CMD_4 5 +#define FLASH_CMD_5 6 +#define FLASH_ERASE_COMPLETE 7 +#define FLASH_PROGRAM 8 +#define FLASH_SETBANK 9 + +u8 flashSaveMemory[0x20000]; +int flashState = FLASH_READ_ARRAY; +int flashReadState = FLASH_READ_ARRAY; +int flashSize = 0x10000; +int flashDeviceID = 0x1b; +int flashManufacturerID = 0x32; +int flashBank = 0; + +static variable_desc flashSaveData[] = { + { &flashState, sizeof(int) }, + { &flashReadState, sizeof(int) }, + { &flashSaveMemory[0], 0x10000 }, + { NULL, 0 } +}; + +static variable_desc flashSaveData2[] = { + { &flashState, sizeof(int) }, + { &flashReadState, sizeof(int) }, + { &flashSize, sizeof(int) }, + { &flashSaveMemory[0], 0x20000 }, + { NULL, 0 } +}; + +static variable_desc flashSaveData3[] = { + { &flashState, sizeof(int) }, + { &flashReadState, sizeof(int) }, + { &flashSize, sizeof(int) }, + { &flashBank, sizeof(int) }, + { &flashSaveMemory[0], 0x20000 }, + { NULL, 0 } +}; + +void flashInit() +{ + memset(flashSaveMemory, 0xff, sizeof(flashSaveMemory)); +} + +void flashReset() +{ + flashState = FLASH_READ_ARRAY; + flashReadState = FLASH_READ_ARRAY; + flashBank = 0; +} + +void flashSaveGame(gzFile gzFile) +{ + utilWriteData(gzFile, flashSaveData3); +} + +void flashReadGame(gzFile gzFile, int version) +{ + if(version < SAVE_GAME_VERSION_5) + utilReadData(gzFile, flashSaveData); + else if(version < SAVE_GAME_VERSION_7) { + utilReadData(gzFile, flashSaveData2); + flashBank = 0; + flashSetSize(flashSize); + } else { + utilReadData(gzFile, flashSaveData3); + } +} + +void flashSetSize(int size) +{ + // log("Setting flash size to %d\n", size); + if(size == 0x10000) { + flashDeviceID = 0x1b; + flashManufacturerID = 0x32; + } else { + flashDeviceID = 0x13; //0x09; + flashManufacturerID = 0x62; //0xc2; + } + // Added to make 64k saves compatible with 128k ones + // (allow wrongfuly set 64k saves to work for Pokemon games) + if ((size == 0x20000) && (flashSize == 0x10000)) + memcpy((u8 *)(flashSaveMemory+0x10000), (u8 *)(flashSaveMemory), 0x10000); + flashSize = size; +} + +u8 flashRead(u32 address) +{ + // log("Reading %08x from %08x\n", address, reg[15].I); + // log("Current read state is %d\n", flashReadState); + address &= 0xFFFF; + + switch(flashReadState) { + case FLASH_READ_ARRAY: + return flashSaveMemory[(flashBank << 16) + address]; + case FLASH_AUTOSELECT: + switch(address & 0xFF) { + case 0: + // manufacturer ID + return flashManufacturerID; + case 1: + // device ID + return flashDeviceID; + } + break; + case FLASH_ERASE_COMPLETE: + flashState = FLASH_READ_ARRAY; + flashReadState = FLASH_READ_ARRAY; + return 0xFF; + }; + return 0; +} + +void flashSaveDecide(u32 address, u8 byte) +{ + // log("Deciding save type %08x\n", address); + if(address == 0x0e005555) { + saveType = 2; + cpuSaveGameFunc = flashWrite; + } else { + saveType = 1; + cpuSaveGameFunc = sramWrite; + } + + (*cpuSaveGameFunc)(address, byte); +} + +void flashDelayedWrite(u32 address, u8 byte) +{ + saveType = 2; + cpuSaveGameFunc = flashWrite; + flashWrite(address, byte); +} + +void flashWrite(u32 address, u8 byte) +{ + // log("Writing %02x at %08x\n", byte, address); + // log("Current state is %d\n", flashState); + address &= 0xFFFF; + switch(flashState) { + case FLASH_READ_ARRAY: + if(address == 0x5555 && byte == 0xAA) + flashState = FLASH_CMD_1; + break; + case FLASH_CMD_1: + if(address == 0x2AAA && byte == 0x55) + flashState = FLASH_CMD_2; + else + flashState = FLASH_READ_ARRAY; + break; + case FLASH_CMD_2: + if(address == 0x5555) { + if(byte == 0x90) { + flashState = FLASH_AUTOSELECT; + flashReadState = FLASH_AUTOSELECT; + } else if(byte == 0x80) { + flashState = FLASH_CMD_3; + } else if(byte == 0xF0) { + flashState = FLASH_READ_ARRAY; + flashReadState = FLASH_READ_ARRAY; + } else if(byte == 0xA0) { + flashState = FLASH_PROGRAM; + } else if(byte == 0xB0 && flashSize == 0x20000) { + flashState = FLASH_SETBANK; + } else { + flashState = FLASH_READ_ARRAY; + flashReadState = FLASH_READ_ARRAY; + } + } else { + flashState = FLASH_READ_ARRAY; + flashReadState = FLASH_READ_ARRAY; + } + break; + case FLASH_CMD_3: + if(address == 0x5555 && byte == 0xAA) { + flashState = FLASH_CMD_4; + } else { + flashState = FLASH_READ_ARRAY; + flashReadState = FLASH_READ_ARRAY; + } + break; + case FLASH_CMD_4: + if(address == 0x2AAA && byte == 0x55) { + flashState = FLASH_CMD_5; + } else { + flashState = FLASH_READ_ARRAY; + flashReadState = FLASH_READ_ARRAY; + } + break; + case FLASH_CMD_5: + if(byte == 0x30) { + // SECTOR ERASE + memset(&flashSaveMemory[(flashBank << 16) + (address & 0xF000)], + 0, + 0x1000); + systemSaveUpdateCounter = SYSTEM_SAVE_UPDATED; + flashReadState = FLASH_ERASE_COMPLETE; + } else if(byte == 0x10) { + // CHIP ERASE + memset(flashSaveMemory, 0, flashSize); + systemSaveUpdateCounter = SYSTEM_SAVE_UPDATED; + flashReadState = FLASH_ERASE_COMPLETE; + } else { + flashState = FLASH_READ_ARRAY; + flashReadState = FLASH_READ_ARRAY; + } + break; + case FLASH_AUTOSELECT: + if(byte == 0xF0) { + flashState = FLASH_READ_ARRAY; + flashReadState = FLASH_READ_ARRAY; + } else if(address == 0x5555 && byte == 0xAA) + flashState = FLASH_CMD_1; + else { + flashState = FLASH_READ_ARRAY; + flashReadState = FLASH_READ_ARRAY; + } + break; + case FLASH_PROGRAM: + flashSaveMemory[(flashBank<<16)+address] = byte; + systemSaveUpdateCounter = SYSTEM_SAVE_UPDATED; + flashState = FLASH_READ_ARRAY; + flashReadState = FLASH_READ_ARRAY; + break; + case FLASH_SETBANK: + if(address == 0) { + flashBank = (byte & 1); + } + flashState = FLASH_READ_ARRAY; + flashReadState = FLASH_READ_ARRAY; + break; + } +} diff --git a/src/Flash.h b/src/Flash.h index bde63997..debda734 100644 --- a/src/Flash.h +++ b/src/Flash.h @@ -1,35 +1,35 @@ -// -*- C++ -*- -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#ifndef VBA_FLASH_H -#define VBA_FLASH_H - -extern void flashSaveGame(gzFile gzFile); -extern void flashReadGame(gzFile gzFile, int version); -extern u8 flashRead(u32 address); -extern void flashWrite(u32 address, u8 byte); -extern void flashDelayedWrite(u32 address, u8 byte); -extern u8 flashSaveMemory[0x20000]; -extern void flashSaveDecide(u32 address, u8 byte); -extern void flashReset(); -extern void flashSetSize(int size); -extern void flashInit(); - -extern int flashSize; -#endif // VBA_FLASH_H \ No newline at end of file +// -*- C++ -*- +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#ifndef VBA_FLASH_H +#define VBA_FLASH_H + +extern void flashSaveGame(gzFile _gzFile); +extern void flashReadGame(gzFile _gzFile, int version); +extern u8 flashRead(u32 address); +extern void flashWrite(u32 address, u8 byte); +extern void flashDelayedWrite(u32 address, u8 byte); +extern u8 flashSaveMemory[0x20000]; +extern void flashSaveDecide(u32 address, u8 byte); +extern void flashReset(); +extern void flashSetSize(int size); +extern void flashInit(); + +extern int flashSize; +#endif // VBA_FLASH_H diff --git a/src/GBA.cpp b/src/GBA.cpp index 3268bc11..3ca1113a 100644 --- a/src/GBA.cpp +++ b/src/GBA.cpp @@ -1,6 +1,6 @@ // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005-2006 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -42,31 +42,51 @@ #include "prof/prof.h" #endif -#define UPDATE_REG(address, value) WRITE16LE(((u16 *)&ioMem[address]),value) +#define UPDATE_REG(address, value)\ + {\ + WRITE16LE(((u16 *)&ioMem[address]),value);\ + }\ + +#define ARM_PREFETCH \ + {\ + cpuPrefetch[0] = CPUReadMemoryQuick(armNextPC);\ + cpuPrefetch[1] = CPUReadMemoryQuick(armNextPC+4);\ + } + +#define THUMB_PREFETCH \ + {\ + cpuPrefetch[0] = CPUReadHalfWordQuick(armNextPC);\ + cpuPrefetch[1] = CPUReadHalfWordQuick(armNextPC+2);\ + } + +#define ARM_PREFETCH_NEXT \ + cpuPrefetch[1] = CPUReadMemoryQuick(armNextPC+4); + +#define THUMB_PREFETCH_NEXT\ + cpuPrefetch[1] = CPUReadHalfWordQuick(armNextPC+2); #ifdef __GNUC__ #define _stricmp strcasecmp #endif -extern int emulating; -extern int linktime; -extern void StartLink(u16); -extern void StartJOYLink(u16); -extern void StartGPLink(u16); -extern void LinkSSend(u16); -extern void LinkUpdate(int); -extern int linktime2; +extern int emulating; +int SWITicks = 0; +int IRQTicks = 0; + +u32 mastercode = 0; +int layerEnableDelay = 0; +bool busPrefetch = false; +bool busPrefetchEnable = false; +u32 busPrefetchCount = 0; int cpuDmaTicksToUpdate = 0; int cpuDmaCount = 0; bool cpuDmaHack = false; -bool cpuDmaHack2 = false; u32 cpuDmaLast = 0; int dummyAddress = 0; -bool cpuBreakLoop = true; +bool cpuBreakLoop = false; int cpuNextEvent = 0; -int cpuTotalTicks = 0; int gbaSaveType = 0; // used to remember the save type on reset bool intState = false; @@ -78,29 +98,40 @@ bool cpuFlashEnabled = true; bool cpuEEPROMEnabled = true; bool cpuEEPROMSensorEnabled = false; +u32 cpuPrefetch[2]; + +int cpuTotalTicks = 0; #ifdef PROFILING int profilingTicks = 0; int profilingTicksReload = 0; -static char *profilBuffer = NULL; -static int profilSize = 0; -static u32 profilLowPC = 0; -static int profilScale = 0; +static profile_segment *profilSegment = NULL; #endif + u8 freezeWorkRAM[0x40000]; u8 freezeInternalRAM[0x8000]; -int lcdTicks = 960; +u8 freezeVRAM[0x18000]; +u8 freezePRAM[0x400]; +u8 freezeOAM[0x400]; +bool debugger_last; + +int lcdTicks = (useBios && !skipBios) ? 1008 : 208; +u8 timerOnOffDelay = 0; +u16 timer0Value = 0; bool timer0On = false; int timer0Ticks = 0; int timer0Reload = 0; int timer0ClockReload = 0; +u16 timer1Value = 0; bool timer1On = false; int timer1Ticks = 0; int timer1Reload = 0; int timer1ClockReload = 0; +u16 timer2Value = 0; bool timer2On = false; int timer2Ticks = 0; int timer2Reload = 0; int timer2ClockReload = 0; +u16 timer3Value = 0; bool timer3On = false; int timer3Ticks = 0; int timer3Reload = 0; @@ -128,62 +159,38 @@ int capturePrevious = 0; int captureNumber = 0; const int TIMER_TICKS[4] = { - 1, - 64, - 256, - 1024 + 0, + 6, + 8, + 10 }; -const int thumbCycles[] = { -// 0 1 2 3 4 5 6 7 8 9 a b c d e f - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 0 - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 1 - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 2 - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 3 - 1, 1, 1, 1, 1, 1, 1, 3, 3, 3, 3, 3, 3, 3, 3, 3, // 4 - 2, 2, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, // 5 - 2, 2, 2, 2, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 3, // 6 - 2, 2, 2, 2, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 3, // 7 - 2, 2, 2, 2, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 3, // 8 - 2, 2, 2, 2, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 3, // 9 - 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, // a - 1, 1, 1, 1, 2, 2, 1, 1, 1, 1, 1, 1, 2, 4, 1, 1, // b - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, // c - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 3, // d - 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, // e - 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2 // f -}; +const u32 objTilesAddress [3] = {0x010000, 0x014000, 0x014000}; +const u8 gamepakRamWaitState[4] = { 4, 3, 2, 8 }; +const u8 gamepakWaitState[4] = { 4, 3, 2, 8 }; +const u8 gamepakWaitState0[2] = { 2, 1 }; +const u8 gamepakWaitState1[2] = { 4, 1 }; +const u8 gamepakWaitState2[2] = { 8, 1 }; +const bool isInRom [16]= + { false, false, false, false, false, false, false, false, + true, true, true, true, true, true, false, false }; -const int gamepakRamWaitState[4] = { 4, 3, 2, 8 }; -const int gamepakWaitState[8] = { 4, 3, 2, 8, 4, 3, 2, 8 }; -const int gamepakWaitState0[8] = { 2, 2, 2, 2, 1, 1, 1, 1 }; -const int gamepakWaitState1[8] = { 4, 4, 4, 4, 1, 1, 1, 1 }; -const int gamepakWaitState2[8] = { 8, 8, 8, 8, 1, 1, 1, 1 }; - -int memoryWait[16] = +u8 memoryWait[16] = { 0, 0, 2, 0, 0, 0, 0, 0, 4, 4, 4, 4, 4, 4, 4, 0 }; -int memoryWait32[16] = - { 0, 0, 9, 0, 0, 0, 0, 0, 8, 8, 8, 8, 8, 8, 8, 0 }; -int memoryWaitSeq[16] = +u8 memoryWait32[16] = + { 0, 0, 5, 0, 0, 1, 1, 0, 7, 7, 9, 9, 13, 13, 4, 0 }; +u8 memoryWaitSeq[16] = { 0, 0, 2, 0, 0, 0, 0, 0, 2, 2, 4, 4, 8, 8, 4, 0 }; -int memoryWaitSeq32[16] = - { 2, 0, 3, 0, 0, 2, 2, 0, 4, 4, 8, 8, 16, 16, 8, 0 }; -int memoryWaitFetch[16] = - { 3, 0, 3, 0, 0, 1, 1, 0, 4, 4, 4, 4, 4, 4, 4, 0 }; -int memoryWaitFetch32[16] = - { 6, 0, 6, 0, 0, 2, 2, 0, 8, 8, 8, 8, 8, 8, 8, 0 }; +u8 memoryWaitSeq32[16] = + { 0, 0, 5, 0, 0, 1, 1, 0, 5, 5, 9, 9, 17, 17, 4, 0 }; + +// The videoMemoryWait constants are used to add some waitstates +// if the opcode access video memory data outside of vblank/hblank +// It seems to happen on only one ticks for each pixel. +// Not used for now (too problematic with current code). +//const u8 videoMemoryWait[16] = +// {0, 0, 0, 0, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -const int cpuMemoryWait[16] = { - 0, 0, 2, 0, 0, 0, 0, 0, - 2, 2, 2, 2, 2, 2, 0, 0 -}; -const int cpuMemoryWait32[16] = { - 0, 0, 3, 0, 0, 0, 0, 0, - 3, 3, 3, 3, 3, 3, 0, 0 -}; - -const bool memory32[16] = - { true, false, false, true, true, false, false, true, false, false, false, false, false, false, true, false}; u8 biosProtected[4]; @@ -489,12 +496,9 @@ variable_desc saveGameStruct[] = { static int romSize = 0x2000000; #ifdef PROFILING -void cpuProfil(char *buf, int size, u32 lowPC, int scale) +void cpuProfil(profile_segment *seg) { - profilBuffer = buf; - profilSize = size; - profilLowPC = lowPC; - profilScale = scale; + profilSegment = seg; } void cpuEnableProfiling(int hz) @@ -506,26 +510,211 @@ void cpuEnableProfiling(int hz) } #endif -inline int CPUUpdateTicksAccess32(u32 address) + +// Waitstates when accessing data +inline int dataTicksAccess16(u32 address) // DATA 8/16bits NON SEQ { - return memoryWait32[(address>>24)&15]; + int addr = (address>>24)&15; + int value = memoryWait[addr]; + + if ((addr>=0x08) || (addr < 0x02)) + { + busPrefetchCount=0; + busPrefetch=false; + } + else if (busPrefetch) + { + int waitState = value; + if (!waitState) + waitState = 1; + busPrefetchCount = ((++busPrefetchCount)<>24)&15]; + int addr = (address>>24)&15; + int value = memoryWait32[addr]; + + if ((addr>=0x08) || (addr < 0x02)) + { + busPrefetchCount=0; + busPrefetch=false; + } + else if (busPrefetch) + { + int waitState = value; + if (!waitState) + waitState = 1; + busPrefetchCount = ((++busPrefetchCount)<>24)&15]; + int addr = (address>>24)&15; + int value = memoryWaitSeq[addr]; + + if ((addr>=0x08) || (addr < 0x02)) + { + busPrefetchCount=0; + busPrefetch=false; + } + else if (busPrefetch) + { + int waitState = value; + if (!waitState) + waitState = 1; + busPrefetchCount = ((++busPrefetchCount)<>24)&15]; + int addr = (address>>24)&15; + int value = memoryWaitSeq32[addr]; + + if ((addr>=0x08) || (addr < 0x02)) + { + busPrefetchCount=0; + busPrefetch=false; + } + else if (busPrefetch) + { + int waitState = value; + if (!waitState) + waitState = 1; + busPrefetchCount = ((++busPrefetchCount)<>24)&15; + + if ((addr>=0x08) && (addr<=0x0D)) + { + if (busPrefetchCount&0x1) + { + if (busPrefetchCount&0x2) + { + busPrefetchCount = ((busPrefetchCount&0xFF)>>2) | (busPrefetchCount&0xFFFFFF00); + return 0; + } + busPrefetchCount = ((busPrefetchCount&0xFF)>>1) | (busPrefetchCount&0xFFFFFF00); + return memoryWaitSeq[addr]-1; + } + else + { + busPrefetchCount=0; + return memoryWait[addr]; + } + } + else + { + busPrefetchCount = 0; + return memoryWait[addr]; + } +} + +inline int codeTicksAccess32(u32 address) // ARM NON SEQ +{ + int addr = (address>>24)&15; + + if ((addr>=0x08) && (addr<=0x0D)) + { + if (busPrefetchCount&0x1) + { + if (busPrefetchCount&0x2) + { + busPrefetchCount = ((busPrefetchCount&0xFF)>>2) | (busPrefetchCount&0xFFFFFF00); + return 0; + } + busPrefetchCount = ((busPrefetchCount&0xFF)>>1) | (busPrefetchCount&0xFFFFFF00); + return memoryWaitSeq[addr] - 1; + } + else + { + busPrefetchCount = 0; + return memoryWait32[addr]; + } + } + else + { + busPrefetchCount = 0; + return memoryWait32[addr]; + } +} + +inline int codeTicksAccessSeq16(u32 address) // THUMB SEQ +{ + int addr = (address>>24)&15; + + if ((addr>=0x08) && (addr<=0x0D)) + { + if (busPrefetchCount&0x1) + { + busPrefetchCount = ((busPrefetchCount&0xFF)>>1) | (busPrefetchCount&0xFFFFFF00); + return 0; + } + else + if (busPrefetchCount>0xFF) + { + busPrefetchCount=0; + return memoryWait[addr]; + } + else + return memoryWaitSeq[addr]; + } + else + { + busPrefetchCount = 0; + return memoryWaitSeq[addr]; + } +} + +inline int codeTicksAccessSeq32(u32 address) // ARM SEQ +{ + int addr = (address>>24)&15; + + if ((addr>=0x08) && (addr<=0x0D)) + { + if (busPrefetchCount&0x1) + { + if (busPrefetchCount&0x2) + { + busPrefetchCount = ((busPrefetchCount&0xFF)>>2) | (busPrefetchCount&0xFFFFFF00); + return 0; + } + busPrefetchCount = ((busPrefetchCount&0xFF)>>1) | (busPrefetchCount&0xFFFFFF00); + return memoryWaitSeq[addr]; + } + else + if (busPrefetchCount>0xFF) + { + busPrefetchCount=0; + return memoryWait32[addr]; + } + else + return memoryWaitSeq32[addr]; + } + else + { + return memoryWaitSeq32[addr]; + } +} + + inline int CPUUpdateTicks() { int cpuLoopTicks = lcdTicks; @@ -533,7 +722,7 @@ inline int CPUUpdateTicks() if(soundTicks < cpuLoopTicks) cpuLoopTicks = soundTicks; - if(timer0On && !(TM0CNT & 4) && (timer0Ticks < cpuLoopTicks)) { + if(timer0On && (timer0Ticks < cpuLoopTicks)) { cpuLoopTicks = timer0Ticks; } if(timer1On && !(TM1CNT & 4) && (timer1Ticks < cpuLoopTicks)) { @@ -553,6 +742,16 @@ inline int CPUUpdateTicks() } #endif + if (SWITicks) { + if (SWITicks < cpuLoopTicks) + cpuLoopTicks = SWITicks; + } + + if (IRQTicks) { + if (IRQTicks < cpuLoopTicks) + cpuLoopTicks = IRQTicks; + } + return cpuLoopTicks; } @@ -632,7 +831,7 @@ static bool CPUWriteState(gzFile gzFile) // new to version 0.7.1 utilWriteInt(gzFile, stopState); // new to version 0.8 - utilWriteInt(gzFile, intState); + utilWriteInt(gzFile, IRQTicks); utilGzWrite(gzFile, internalRAM, 0x8000); utilGzWrite(gzFile, paletteRAM, 0x400); @@ -640,9 +839,6 @@ static bool CPUWriteState(gzFile gzFile) utilGzWrite(gzFile, vram, 0x20000); utilGzWrite(gzFile, oam, 0x400); utilGzWrite(gzFile, pix, 4*241*162); - - for (int i = NR10; i <= 0x9F; i++) ioMem[i] = soundRead(i); - utilGzWrite(gzFile, ioMem, 0x400); eepromSaveGame(gzFile); @@ -739,9 +935,21 @@ static bool CPUReadState(gzFile gzFile) stopState = utilReadInt(gzFile) ? true : false; if(version < SAVE_GAME_VERSION_4) + { + IRQTicks = 0; intState = false; + } else - intState = utilReadInt(gzFile) ? true : false; + { + IRQTicks = utilReadInt(gzFile); + if (IRQTicks>0) + intState = true; + else + { + intState = false; + IRQTicks = 0; + } + } utilGzRead(gzFile, internalRAM, 0x8000); utilGzRead(gzFile, paletteRAM, 0x400); @@ -759,7 +967,7 @@ static bool CPUReadState(gzFile gzFile) soundReadGame(gzFile, version); if(version > SAVE_GAME_VERSION_1) { - cheatsReadGame(gzFile); + cheatsReadGame(gzFile, version); } if(version > SAVE_GAME_VERSION_6) { rtcReadGame(gzFile); @@ -783,6 +991,19 @@ static bool CPUReadState(gzFile gzFile) SWAP(dma3Dest, DM3DAD_H, DM3DAD_L); } + if(version <= SAVE_GAME_VERSION_8) { + timer0ClockReload = TIMER_TICKS[TM0CNT & 3]; + timer1ClockReload = TIMER_TICKS[TM1CNT & 3]; + timer2ClockReload = TIMER_TICKS[TM2CNT & 3]; + timer3ClockReload = TIMER_TICKS[TM3CNT & 3]; + + timer0Ticks = ((0x10000 - TM0D) << timer0ClockReload) - timer0Ticks; + timer1Ticks = ((0x10000 - TM1D) << timer1ClockReload) - timer1Ticks; + timer2Ticks = ((0x10000 - TM2D) << timer2ClockReload) - timer2Ticks; + timer3Ticks = ((0x10000 - TM3D) << timer3ClockReload) - timer3Ticks; + interp_rate(); + } + // set pointers! layerEnable = layerSettings & DISPCNT; @@ -803,6 +1024,11 @@ static bool CPUReadState(gzFile gzFile) cpuSaveGameFunc = flashWrite; gbaSaveType = 2; break; + case 3: + break; + case 5: + gbaSaveType = 5; + break; default: systemMessage(MSG_UNSUPPORTED_SAVE_TYPE, N_("Unsupported save type %d"), saveType); @@ -812,6 +1038,13 @@ static bool CPUReadState(gzFile gzFile) gbaSaveType = 3; systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; + if(armState) { + ARM_PREFETCH; + } else { + THUMB_PREFETCH; + } + + CPUUpdateRegister(0x204, CPUReadHalfWordQuick(0x4000204)); return true; } @@ -881,7 +1114,7 @@ bool CPUWriteBatteryFile(const char *fileName) } } - if(gbaSaveType) { + if((gbaSaveType) && (gbaSaveType!=5)) { FILE *file = fopen(fileName, "wb"); if(!file) { @@ -998,13 +1231,13 @@ bool CPUWriteGSASnapshot(const char *fileName, fwrite("SharkPortSave", 1, 0x0d, file); utilPutDword(buffer, 0x000f0000); fwrite(buffer, 1, 4, file); // save type 0x000f0000 = GBA save - utilPutDword(buffer, strlen(title)); + utilPutDword(buffer, (u32)strlen(title)); fwrite(buffer, 1, 4, file); // title length fwrite(title, 1, strlen(title), file); - utilPutDword(buffer, strlen(desc)); + utilPutDword(buffer, (u32)strlen(desc)); fwrite(buffer, 1, 4, file); // desc length fwrite(desc, 1, strlen(desc), file); - utilPutDword(buffer, strlen(notes)); + utilPutDword(buffer, (u32)strlen(notes)); fwrite(buffer, 1, 4, file); // notes length fwrite(notes, 1, strlen(notes), file); int saveSize = 0x10000; @@ -1131,7 +1364,7 @@ bool CPUWriteBMPFile(const char *fileName) bool CPUIsZipFile(const char * file) { if(strlen(file) > 4) { - char * p = (char *)strrchr(file,'.'); + const char * p = strrchr(file,'.'); if(p != NULL) { if(_stricmp(p, ".zip") == 0) @@ -1146,7 +1379,7 @@ bool CPUIsGBAImage(const char * file) { cpuIsMultiBoot = false; if(strlen(file) > 4) { - char * p = (char *)strrchr(file,'.'); + const char * p = strrchr(file,'.'); if(p != NULL) { if(_stricmp(p, ".gba") == 0) @@ -1170,7 +1403,7 @@ bool CPUIsGBAImage(const char * file) bool CPUIsGBABios(const char * file) { if(strlen(file) > 4) { - char * p = (char *)strrchr(file,'.'); + const char * p = strrchr(file,'.'); if(p != NULL) { if(_stricmp(p, ".gba") == 0) @@ -1190,7 +1423,7 @@ bool CPUIsGBABios(const char * file) bool CPUIsELF(const char *file) { if(strlen(file) > 4) { - char * p = (char *)strrchr(file,'.'); + const char * p = strrchr(file,'.'); if(p != NULL) { if(_stricmp(p, ".elf") == 0) @@ -1263,7 +1496,6 @@ void CPUCleanUp() int CPULoadRom(const char *szFile) { romSize = 0x2000000; - if(rom != NULL) { CPUCleanUp(); } @@ -1376,12 +1608,30 @@ int CPULoadRom(const char *szFile) } flashInit(); + eepromInit(); CPUUpdateRenderBuffers(true); return romSize; } +void doMirroring (bool b) +{ + u32 mirroredRomSize = (((romSize)>>20) & 0x3F)<<20; + u32 mirroredRomAddress = romSize; + if ((mirroredRomSize <=0x800000) && (b)) + { + mirroredRomAddress = mirroredRomSize; + if (mirroredRomSize==0) + mirroredRomSize=0x100000; + while (mirroredRomAddress<0x01000000) + { + memcpy ((u16 *)(rom+mirroredRomAddress), (u16 *)(rom), mirroredRomSize); + mirroredRomAddress+=mirroredRomSize; + } + } +} + void CPUUpdateRender() { switch(DISPCNT & 7) { @@ -1473,9 +1723,8 @@ void CPUUpdateFlags(bool breakLoop) armState = (CPSR & 0x20) ? false : true; armIrqEnable = (CPSR & 0x80) ? false : true; if(breakLoop) { - if(armIrqEnable && (IF & IE) && (IME & 1)) { - cpuNextEvent = 0; - } + if (armIrqEnable && (IF & IE) && (IME & 1)) + cpuNextEvent = cpuTotalTicks; } } @@ -1629,6 +1878,7 @@ void CPUUndefinedException() armState = true; armIrqEnable = false; armNextPC = 0x04; + ARM_PREFETCH; reg[15].I += 4; } @@ -1642,6 +1892,7 @@ void CPUSoftwareInterrupt() armState = true; armIrqEnable = false; armNextPC = 0x08; + ARM_PREFETCH; reg[15].I += 4; } @@ -1681,7 +1932,7 @@ void CPUSoftwareInterrupt(int comment) #ifdef SDL if(comment == 0xf9) { emulating = 0; - cpuNextEvent = 0; + cpuNextEvent = cpuTotalTicks; cpuBreakLoop = true; return; } @@ -1708,6 +1959,7 @@ void CPUSoftwareInterrupt(int comment) switch(comment) { case 0x00: BIOS_SoftReset(); + ARM_PREFETCH; break; case 0x01: BIOS_RegisterRamReset(); @@ -1721,7 +1973,7 @@ void CPUSoftwareInterrupt(int comment) #endif holdState = true; holdType = -1; - cpuNextEvent = 0; + cpuNextEvent = cpuTotalTicks; break; case 0x03: #ifdef DEV_VERSION @@ -1733,7 +1985,7 @@ void CPUSoftwareInterrupt(int comment) holdState = true; holdType = -1; stopState = true; - cpuNextEvent = 0; + cpuNextEvent = cpuTotalTicks; break; case 0x04: #ifdef DEV_VERSION @@ -1771,11 +2023,52 @@ void CPUSoftwareInterrupt(int comment) BIOS_ArcTan2(); break; case 0x0B: + { + int len = (reg[2].I & 0x1FFFFF) >>1; + if (!(((reg[0].I & 0xe000000) == 0) || + ((reg[0].I + len) & 0xe000000) == 0)) + { + if ((reg[2].I >> 24) & 1) + { + if ((reg[2].I >> 26) & 1) + SWITicks = (7 + memoryWait32[(reg[1].I>>24) & 0xF]) * (len>>1); + else + SWITicks = (8 + memoryWait[(reg[1].I>>24) & 0xF]) * (len); + } + else + { + if ((reg[2].I >> 26) & 1) + SWITicks = (10 + memoryWait32[(reg[0].I>>24) & 0xF] + + memoryWait32[(reg[1].I>>24) & 0xF]) * (len>>1); + else + SWITicks = (11 + memoryWait[(reg[0].I>>24) & 0xF] + + memoryWait[(reg[1].I>>24) & 0xF]) * len; + } + } + } BIOS_CpuSet(); break; case 0x0C: + { + int len = (reg[2].I & 0x1FFFFF) >>5; + if (!(((reg[0].I & 0xe000000) == 0) || + ((reg[0].I + len) & 0xe000000) == 0)) + { + if ((reg[2].I >> 24) & 1) + SWITicks = (6 + memoryWait32[(reg[1].I>>24) & 0xF] + + 7 * (memoryWaitSeq32[(reg[1].I>>24) & 0xF] + 1)) * len; + else + SWITicks = (9 + memoryWait32[(reg[0].I>>24) & 0xF] + + memoryWait32[(reg[1].I>>24) & 0xF] + + 7 * (memoryWaitSeq32[(reg[0].I>>24) & 0xF] + + memoryWaitSeq32[(reg[1].I>>24) & 0xF] + 2)) * len; + } + } BIOS_CpuFastSet(); break; + case 0x0D: + BIOS_GetBiosChecksum(); + break; case 0x0E: BIOS_BgAffineSet(); break; @@ -1783,30 +2076,89 @@ void CPUSoftwareInterrupt(int comment) BIOS_ObjAffineSet(); break; case 0x10: + { + int len = CPUReadHalfWord(reg[2].I); + if (!(((reg[0].I & 0xe000000) == 0) || + ((reg[0].I + len) & 0xe000000) == 0)) + SWITicks = (32 + memoryWait[(reg[0].I>>24) & 0xF]) * len; + } BIOS_BitUnPack(); break; case 0x11: + { + u32 len = CPUReadMemory(reg[0].I) >> 8; + if(!(((reg[0].I & 0xe000000) == 0) || + ((reg[0].I + (len & 0x1fffff)) & 0xe000000) == 0)) + SWITicks = (9 + memoryWait[(reg[1].I>>24) & 0xF]) * len; + } BIOS_LZ77UnCompWram(); break; case 0x12: + { + u32 len = CPUReadMemory(reg[0].I) >> 8; + if(!(((reg[0].I & 0xe000000) == 0) || + ((reg[0].I + (len & 0x1fffff)) & 0xe000000) == 0)) + SWITicks = (19 + memoryWait[(reg[1].I>>24) & 0xF]) * len; + } BIOS_LZ77UnCompVram(); break; case 0x13: + { + u32 len = CPUReadMemory(reg[0].I) >> 8; + if(!(((reg[0].I & 0xe000000) == 0) || + ((reg[0].I + (len & 0x1fffff)) & 0xe000000) == 0)) + SWITicks = (29 + (memoryWait[(reg[0].I>>24) & 0xF]<<1)) * len; + } BIOS_HuffUnComp(); break; case 0x14: + { + u32 len = CPUReadMemory(reg[0].I) >> 8; + if(!(((reg[0].I & 0xe000000) == 0) || + ((reg[0].I + (len & 0x1fffff)) & 0xe000000) == 0)) + SWITicks = (11 + memoryWait[(reg[0].I>>24) & 0xF] + + memoryWait[(reg[1].I>>24) & 0xF]) * len; + } BIOS_RLUnCompWram(); break; case 0x15: + { + u32 len = CPUReadMemory(reg[0].I) >> 9; + if(!(((reg[0].I & 0xe000000) == 0) || + ((reg[0].I + (len & 0x1fffff)) & 0xe000000) == 0)) + SWITicks = (34 + (memoryWait[(reg[0].I>>24) & 0xF] << 1) + + memoryWait[(reg[1].I>>24) & 0xF]) * len; + } BIOS_RLUnCompVram(); break; case 0x16: + { + u32 len = CPUReadMemory(reg[0].I) >> 8; + if(!(((reg[0].I & 0xe000000) == 0) || + ((reg[0].I + (len & 0x1fffff)) & 0xe000000) == 0)) + SWITicks = (13 + memoryWait[(reg[0].I>>24) & 0xF] + + memoryWait[(reg[1].I>>24) & 0xF]) * len; + } BIOS_Diff8bitUnFilterWram(); break; case 0x17: + { + u32 len = CPUReadMemory(reg[0].I) >> 9; + if(!(((reg[0].I & 0xe000000) == 0) || + ((reg[0].I + (len & 0x1fffff)) & 0xe000000) == 0)) + SWITicks = (39 + (memoryWait[(reg[0].I>>24) & 0xF]<<1) + + memoryWait[(reg[1].I>>24) & 0xF]) * len; + } BIOS_Diff8bitUnFilterVram(); break; case 0x18: + { + u32 len = CPUReadMemory(reg[0].I) >> 9; + if(!(((reg[0].I & 0xe000000) == 0) || + ((reg[0].I + (len & 0x1fffff)) & 0xe000000) == 0)) + SWITicks = (13 + memoryWait[(reg[0].I>>24) & 0xF] + + memoryWait[(reg[1].I>>24) & 0xF]) * len; + } BIOS_Diff16bitUnFilter(); break; case 0x19: @@ -1865,17 +2217,33 @@ void CPUCompareVCOUNT() DISPSTAT &= 0xFFFB; UPDATE_REG(0x4, DISPSTAT); } + if (layerEnableDelay>0) + { + layerEnableDelay--; + if (layerEnableDelay==1) + layerEnable = layerSettings & DISPCNT; + } + } void doDMA(u32 &s, u32 &d, u32 si, u32 di, u32 c, int transfer32) { int sm = s >> 24; int dm = d >> 24; - + int sw = 0; + int dw = 0; int sc = c; cpuDmaCount = c; + // This is done to get the correct waitstates. + if (sm>15) + sm=15; + if (dm>15) + dm=15; + //if ((sm>=0x05) && (sm<=0x07) || (dm>=0x05) && (dm <=0x07)) + // blank = (((DISPSTAT | ((DISPSTAT>>1)&1))==1) ? true : false); + if(transfer32) { s &= 0xFFFFFFFC; if(s < 0x02000000 && (reg[15].I >> 24)) { @@ -1900,7 +2268,6 @@ void doDMA(u32 &s, u32 &d, u32 si, u32 di, u32 c, int transfer32) if(s < 0x02000000 && (reg[15].I >> 24)) { while(c != 0) { CPUWriteHalfWord(d, 0); - cpuDmaLast |= (cpuDmaLast<<16); d += di; c--; } @@ -1908,6 +2275,7 @@ void doDMA(u32 &s, u32 &d, u32 si, u32 di, u32 c, int transfer32) while(c != 0) { cpuDmaLast = CPUReadHalfWord(s); CPUWriteHalfWord(d, cpuDmaLast); + cpuDmaLast |= (cpuDmaLast<<16); d += di; s += si; c--; @@ -1917,33 +2285,31 @@ void doDMA(u32 &s, u32 &d, u32 si, u32 di, u32 c, int transfer32) cpuDmaCount = 0; - int sw = 1+memoryWaitSeq[sm & 15]; - int dw = 1+memoryWaitSeq[dm & 15]; - int totalTicks = 0; if(transfer32) { - if(!memory32[sm & 15]) - sw <<= 1; - if(!memory32[dm & 15]) - dw <<= 1; + sw =1+memoryWaitSeq32[sm & 15]; + dw =1+memoryWaitSeq32[dm & 15]; + totalTicks = (sw+dw)*(sc-1) + 6 + memoryWait32[sm & 15] + + memoryWaitSeq32[dm & 15]; + } + else + { + sw = 1+memoryWaitSeq[sm & 15]; + dw = 1+memoryWaitSeq[dm & 15]; + totalTicks = (sw+dw)*(sc-1) + 6 + memoryWait[sm & 15] + + memoryWaitSeq[dm & 15]; } - - totalTicks = (sw+dw)*sc; cpuDmaTicksToUpdate += totalTicks; - cpuNextEvent = 0; } -bool CPUCheckDMA(int reason, int dmamask) +void CPUCheckDMA(int reason, int dmamask) { - bool res = false; - cpuDmaHack = 0; // DMA 0 if((DM0CNT_H & 0x8000) && (dmamask & 1)) { if(((DM0CNT_H >> 12) & 3) == reason) { - res = true; u32 sourceIncrement = 4; u32 destIncrement = 4; switch((DM0CNT_H >> 7) & 3) { @@ -1980,9 +2346,11 @@ bool CPUCheckDMA(int reason, int dmamask) DM0CNT_L ? DM0CNT_L : 0x4000, DM0CNT_H & 0x0400); cpuDmaHack = true; + if(DM0CNT_H & 0x4000) { IF |= 0x0100; UPDATE_REG(0x202, IF); + cpuNextEvent = cpuTotalTicks; } if(((DM0CNT_H >> 5) & 3) == 3) { @@ -1999,7 +2367,6 @@ bool CPUCheckDMA(int reason, int dmamask) // DMA 1 if((DM1CNT_H & 0x8000) && (dmamask & 2)) { if(((DM1CNT_H >> 12) & 3) == reason) { - res = true; u32 sourceIncrement = 4; u32 destIncrement = 4; switch((DM1CNT_H >> 7) & 3) { @@ -2048,10 +2415,11 @@ bool CPUCheckDMA(int reason, int dmamask) DM1CNT_H & 0x0400); } cpuDmaHack = true; - + if(DM1CNT_H & 0x4000) { IF |= 0x0200; UPDATE_REG(0x202, IF); + cpuNextEvent = cpuTotalTicks; } if(((DM1CNT_H >> 5) & 3) == 3) { @@ -2068,7 +2436,6 @@ bool CPUCheckDMA(int reason, int dmamask) // DMA 2 if((DM2CNT_H & 0x8000) && (dmamask & 4)) { if(((DM2CNT_H >> 12) & 3) == reason) { - res = true; u32 sourceIncrement = 4; u32 destIncrement = 4; switch((DM2CNT_H >> 7) & 3) { @@ -2118,9 +2485,11 @@ bool CPUCheckDMA(int reason, int dmamask) DM2CNT_H & 0x0400); } cpuDmaHack = true; + if(DM2CNT_H & 0x4000) { IF |= 0x0400; UPDATE_REG(0x202, IF); + cpuNextEvent = cpuTotalTicks; } if(((DM2CNT_H >> 5) & 3) == 3) { @@ -2137,7 +2506,6 @@ bool CPUCheckDMA(int reason, int dmamask) // DMA 3 if((DM3CNT_H & 0x8000) && (dmamask & 8)) { if(((DM3CNT_H >> 12) & 3) == reason) { - res = true; u32 sourceIncrement = 4; u32 destIncrement = 4; switch((DM3CNT_H >> 7) & 3) { @@ -2176,6 +2544,7 @@ bool CPUCheckDMA(int reason, int dmamask) if(DM3CNT_H & 0x4000) { IF |= 0x0800; UPDATE_REG(0x202, IF); + cpuNextEvent = cpuTotalTicks; } if(((DM3CNT_H >> 5) & 3) == 3) { @@ -2188,8 +2557,6 @@ bool CPUCheckDMA(int reason, int dmamask) } } } - cpuDmaHack = false; - return res; } void CPUUpdateRegister(u32 address, u16 value) @@ -2197,15 +2564,27 @@ void CPUUpdateRegister(u32 address, u16 value) switch(address) { case 0x00: { + if ((value & 7) >5) + DISPCNT = (value &7); bool change = ((DISPCNT ^ value) & 0x80) ? true : false; bool changeBG = ((DISPCNT ^ value) & 0x0F00) ? true : false; + u16 changeBGon = (((~DISPCNT) & value) & 0x0F00); DISPCNT = (value & 0xFFF7); UPDATE_REG(0x00, DISPCNT); - layerEnable = layerSettings & value; + + if (changeBGon) + { + layerEnableDelay=4; + layerEnable = layerSettings & value & (~changeBGon); + } + else + layerEnable = layerSettings & value; + // CPUUpdateTicks(); + windowOn = (layerEnable & 0x6000) ? true : false; if(change && !((value & 0x80))) { if(!(DISPSTAT & 1)) { - lcdTicks = 960; + lcdTicks = 1008; // VCOUNT = 0; // UPDATE_REG(0x06, VCOUNT); DISPSTAT &= 0xFFFC; @@ -2218,7 +2597,6 @@ void CPUUpdateRegister(u32 address, u16 value) // we only care about changes in BG0-BG3 if(changeBG) CPUUpdateRenderBuffers(false); - // CPUUpdateTicks(); } break; case 0x04: @@ -2566,89 +2944,41 @@ void CPUUpdateRegister(u32 address, u16 value) } } break; - case 0x100: + case 0x100: timer0Reload = value; interp_rate(); - break; + break; case 0x102: - timer0Ticks = timer0ClockReload = TIMER_TICKS[value & 3]; - if(!timer0On && (value & 0x80)) { - // reload the counter - TM0D = timer0Reload; - if(timer0ClockReload == 1) - timer0Ticks = 0x10000 - TM0D; - UPDATE_REG(0x100, TM0D); - int event = cpuTotalTicks + timer0Ticks; - if(event < cpuNextEvent) - cpuNextEvent = event; - } - timer0On = value & 0x80 ? true : false; - TM0CNT = value & 0xC7; - UPDATE_REG(0x102, TM0CNT); - interp_rate(); - // CPUUpdateTicks(); + timer0Value = value; + timerOnOffDelay|=1; + cpuNextEvent = cpuTotalTicks; break; case 0x104: timer1Reload = value; interp_rate(); - break; + break; case 0x106: - timer1Ticks = timer1ClockReload = TIMER_TICKS[value & 3]; - if(!timer1On && (value & 0x80)) { - // reload the counter - TM1D = timer1Reload; - if(timer1ClockReload == 1) - timer1Ticks = 0x10000 - TM1D; - UPDATE_REG(0x104, TM1D); - int event = cpuTotalTicks + timer1Ticks; - if(event < cpuNextEvent) - cpuNextEvent = event; - } - timer1On = value & 0x80 ? true : false; - TM1CNT = value & 0xC7; - UPDATE_REG(0x106, TM1CNT); - interp_rate(); - break; + timer1Value = value; + timerOnOffDelay|=2; + cpuNextEvent = cpuTotalTicks; + break; case 0x108: timer2Reload = value; break; case 0x10A: - timer2Ticks = timer2ClockReload = TIMER_TICKS[value & 3]; - if(!timer2On && (value & 0x80)) { - // reload the counter - TM2D = timer2Reload; - if(timer2ClockReload == 1) - timer2Ticks = 0x10000 - TM2D; - UPDATE_REG(0x108, TM2D); - int event = cpuTotalTicks + timer2Ticks; - if(event < cpuNextEvent) - cpuNextEvent = event; - } - timer2On = value & 0x80 ? true : false; - TM2CNT = value & 0xC7; - UPDATE_REG(0x10A, TM2CNT); + timer2Value = value; + timerOnOffDelay|=4; + cpuNextEvent = cpuTotalTicks; break; case 0x10C: timer3Reload = value; break; case 0x10E: - timer3Ticks = timer3ClockReload = TIMER_TICKS[value & 3]; - if(!timer3On && (value & 0x80)) { - // reload the counter - TM3D = timer3Reload; - if(timer3ClockReload == 1) - timer3Ticks = 0x10000 - TM3D; - UPDATE_REG(0x10C, TM3D); - int event = cpuTotalTicks + timer3Ticks; - if(event < cpuNextEvent) - cpuNextEvent = event; - } - timer3On = value & 0x80 ? true : false; - TM3CNT = value & 0xC7; - UPDATE_REG(0x10E, TM3CNT); + timer3Value = value; + timerOnOffDelay|=8; + cpuNextEvent = cpuTotalTicks; break; case 0x128: - StartLink(value); // Link if(value & 0x80) { value &= 0xff7f; if(value & 1 && (value & 0x4000)) { @@ -2660,14 +2990,6 @@ void CPUUpdateRegister(u32 address, u16 value) } UPDATE_REG(0x128, value); break; -/* Link -------------------------------*/ - case 0x12a: - if(lspeed) - LinkSSend(value); - UPDATE_REG(0x12a, value); - break; -/* --------------------------- */ case 0x130: P1 |= (value & 0x3FF); UPDATE_REG(0x130, P1); @@ -2675,18 +2997,11 @@ void CPUUpdateRegister(u32 address, u16 value) case 0x132: UPDATE_REG(0x132, value & 0xC3FF); break; - case 0x134: - StartGPLink(value); - break; - case 0x140: - StartJOYLink(value); - break; case 0x200: IE = value & 0x3FFF; UPDATE_REG(0x200, IE); - if((IME & 1) && (IF & IE) && armIrqEnable) { - cpuNextEvent = 0; - } + if ((IME & 1) && (IF & IE) && armIrqEnable) + cpuNextEvent = cpuTotalTicks; break; case 0x202: IF ^= (value & IF); @@ -2694,54 +3009,54 @@ void CPUUpdateRegister(u32 address, u16 value) break; case 0x204: { - int i; memoryWait[0x0e] = memoryWaitSeq[0x0e] = gamepakRamWaitState[value & 3]; if(!speedHack) { - memoryWait[0x08] = memoryWait[0x09] = gamepakWaitState[(value >> 2) & 7]; + memoryWait[0x08] = memoryWait[0x09] = gamepakWaitState[(value >> 2) & 3]; memoryWaitSeq[0x08] = memoryWaitSeq[0x09] = - gamepakWaitState0[(value >> 2) & 7]; + gamepakWaitState0[(value >> 4) & 1]; - memoryWait[0x0a] = memoryWait[0x0b] = gamepakWaitState[(value >> 5) & 7]; + memoryWait[0x0a] = memoryWait[0x0b] = gamepakWaitState[(value >> 5) & 3]; memoryWaitSeq[0x0a] = memoryWaitSeq[0x0b] = - gamepakWaitState1[(value >> 5) & 7]; + gamepakWaitState1[(value >> 7) & 1]; - memoryWait[0x0c] = memoryWait[0x0d] = gamepakWaitState[(value >> 8) & 7]; + memoryWait[0x0c] = memoryWait[0x0d] = gamepakWaitState[(value >> 8) & 3]; memoryWaitSeq[0x0c] = memoryWaitSeq[0x0d] = - gamepakWaitState2[(value >> 8) & 7]; + gamepakWaitState2[(value >> 10) & 1]; } else { - memoryWait[0x08] = memoryWait[0x09] = 4; - memoryWaitSeq[0x08] = memoryWaitSeq[0x09] = 2; + memoryWait[0x08] = memoryWait[0x09] = 3; + memoryWaitSeq[0x08] = memoryWaitSeq[0x09] = 1; - memoryWait[0x0a] = memoryWait[0x0b] = 4; - memoryWaitSeq[0x0a] = memoryWaitSeq[0x0b] = 4; + memoryWait[0x0a] = memoryWait[0x0b] = 3; + memoryWaitSeq[0x0a] = memoryWaitSeq[0x0b] = 1; - memoryWait[0x0c] = memoryWait[0x0d] = 4; - memoryWaitSeq[0x0c] = memoryWaitSeq[0x0d] = 8; + memoryWait[0x0c] = memoryWait[0x0d] = 3; + memoryWaitSeq[0x0c] = memoryWaitSeq[0x0d] = 1; } - for(i = 0; i < 16; i++) { - memoryWaitFetch32[i] = memoryWait32[i] = memoryWait[i] * - (memory32[i] ? 1 : 2); - memoryWaitFetch[i] = memoryWait[i]; + + for(int i = 8; i < 15; i++) { + memoryWait32[i] = memoryWait[i] + memoryWaitSeq[i] + 1; + memoryWaitSeq32[i] = memoryWaitSeq[i]*2 + 1; } - memoryWaitFetch32[3] += 1; - memoryWaitFetch32[2] += 3; - - if(value & 0x4000) { - for(i = 8; i < 16; i++) { - memoryWaitFetch32[i] = cpuMemoryWait32[i]; - memoryWaitFetch[i] = cpuMemoryWait[i]; - } + + if((value & 0x4000) == 0x4000) { + busPrefetchEnable = true; + busPrefetch = false; + busPrefetchCount = 0; + } else { + busPrefetchEnable = false; + busPrefetch = false; + busPrefetchCount = 0; } - UPDATE_REG(0x204, value); + UPDATE_REG(0x204, value & 0x7FFF); + } break; case 0x208: IME = value & 1; UPDATE_REG(0x208, IME); - if((IME & 1) && (IF & IE) && armIrqEnable) { - cpuNextEvent = 0; - } + if ((IME & 1) && (IF & IE) && armIrqEnable) + cpuNextEvent = cpuTotalTicks; break; case 0x300: if(value != 0) @@ -2754,6 +3069,67 @@ void CPUUpdateRegister(u32 address, u16 value) } } +void applyTimer () +{ + if (timerOnOffDelay & 1) + { + timer0ClockReload = TIMER_TICKS[timer0Value & 3]; + if(!timer0On && (timer0Value & 0x80)) { + // reload the counter + TM0D = timer0Reload; + timer0Ticks = (0x10000 - TM0D) << timer0ClockReload; + UPDATE_REG(0x100, TM0D); + } + timer0On = timer0Value & 0x80 ? true : false; + TM0CNT = timer0Value & 0xC7; + interp_rate(); + UPDATE_REG(0x102, TM0CNT); + // CPUUpdateTicks(); + } + if (timerOnOffDelay & 2) + { + timer1ClockReload = TIMER_TICKS[timer1Value & 3]; + if(!timer1On && (timer1Value & 0x80)) { + // reload the counter + TM1D = timer1Reload; + timer1Ticks = (0x10000 - TM1D) << timer1ClockReload; + UPDATE_REG(0x104, TM1D); + } + timer1On = timer1Value & 0x80 ? true : false; + TM1CNT = timer1Value & 0xC7; + interp_rate(); + UPDATE_REG(0x106, TM1CNT); + } + if (timerOnOffDelay & 4) + { + timer2ClockReload = TIMER_TICKS[timer2Value & 3]; + if(!timer2On && (timer2Value & 0x80)) { + // reload the counter + TM2D = timer2Reload; + timer2Ticks = (0x10000 - TM2D) << timer2ClockReload; + UPDATE_REG(0x108, TM2D); + } + timer2On = timer2Value & 0x80 ? true : false; + TM2CNT = timer2Value & 0xC7; + UPDATE_REG(0x10A, TM2CNT); + } + if (timerOnOffDelay & 8) + { + timer3ClockReload = TIMER_TICKS[timer3Value & 3]; + if(!timer3On && (timer3Value & 0x80)) { + // reload the counter + TM3D = timer3Reload; + timer3Ticks = (0x10000 - TM3D) << timer3ClockReload; + UPDATE_REG(0x10C, TM3D); + } + timer3On = timer3Value & 0x80 ? true : false; + TM3CNT = timer3Value & 0xC7; + UPDATE_REG(0x10E, TM3CNT); + } + cpuNextEvent = CPUUpdateTicks(); + timerOnOffDelay = 0; +} + void CPUWriteHalfWord(u32 address, u16 value) { #ifdef DEV_VERSION @@ -2769,7 +3145,7 @@ void CPUWriteHalfWord(u32 address, u16 value) switch(address >> 24) { case 2: -#ifdef SDL +#ifdef BKPT_SUPPORT if(*((u16 *)&freezeWorkRAM[address & 0x3FFFE])) cheatsWriteHalfWord(address & 0x203FFFE, value); @@ -2778,7 +3154,7 @@ void CPUWriteHalfWord(u32 address, u16 value) WRITE16LE(((u16 *)&workRAM[address & 0x3FFFE]),value); break; case 3: -#ifdef SDL +#ifdef BKPT_SUPPORT if(*((u16 *)&freezeInternalRAM[address & 0x7ffe])) cheatsWriteHalfWord(address & 0x3007ffe, value); @@ -2792,15 +3168,35 @@ void CPUWriteHalfWord(u32 address, u16 value) else goto unwritable; break; case 5: +#ifdef BKPT_SUPPORT + if(*((u16 *)&freezePRAM[address & 0x03fe])) + cheatsWriteHalfWord(address & 0x70003fe, + value); + else +#endif WRITE16LE(((u16 *)&paletteRAM[address & 0x3fe]), value); break; case 6: - if(address & 0x10000) - WRITE16LE(((u16 *)&vram[address & 0x17ffe]), value); + address = (address & 0x1fffe); + if (((DISPCNT & 7) >2) && ((address & 0x1C000) == 0x18000)) + return; + if ((address & 0x18000) == 0x18000) + address &= 0x17fff; +#ifdef BKPT_SUPPORT + if(*((u16 *)&freezeVRAM[address])) + cheatsWriteHalfWord(address + 0x06000000, + value); else - WRITE16LE(((u16 *)&vram[address & 0x1fffe]), value); +#endif + WRITE16LE(((u16 *)&vram[address]), value); break; case 7: +#ifdef BKPT_SUPPORT + if(*((u16 *)&freezeOAM[address & 0x03fe])) + cheatsWriteHalfWord(address & 0x70003fe, + value); + else +#endif WRITE16LE(((u16 *)&oam[address & 0x3fe]), value); break; case 8: @@ -2840,7 +3236,7 @@ void CPUWriteByte(u32 address, u8 b) { switch(address >> 24) { case 2: -#ifdef SDL +#ifdef BKPT_SUPPORT if(freezeWorkRAM[address & 0x3FFFF]) cheatsWriteByte(address & 0x203FFFF, b); else @@ -2848,7 +3244,7 @@ void CPUWriteByte(u32 address, u8 b) workRAM[address & 0x3FFFF] = b; break; case 3: -#ifdef SDL +#ifdef BKPT_SUPPORT if(freezeInternalRAM[address & 0x7fff]) cheatsWriteByte(address & 0x3007fff, b); else @@ -2863,7 +3259,7 @@ void CPUWriteByte(u32 address, u8 b) stopState = true; holdState = 1; holdType = -1; - cpuNextEvent = 0; + cpuNextEvent = cpuTotalTicks; break; case 0x60: case 0x61: @@ -2925,12 +3321,23 @@ void CPUWriteByte(u32 address, u8 b) *((u16 *)&paletteRAM[address & 0x3FE]) = (b << 8) | b; break; case 6: + address = (address & 0x1fffe); + if (((DISPCNT & 7) >2) && ((address & 0x1C000) == 0x18000)) + return; + if ((address & 0x18000) == 0x18000) + address &= 0x17fff; + // no need to switch // byte writes to OBJ VRAM are ignored - if(!(address & 0x10000)) - // *((u16 *)&vram[address & 0x17FFE]) = (b << 8) | b; - // else - *((u16 *)&vram[address & 0xFFFE]) = (b << 8) | b; + if ((address) < objTilesAddress[((DISPCNT&7)+1)>>2]) + { +#ifdef BKPT_SUPPORT + if(freezeVRAM[address]) + cheatsWriteByte(address + 0x06000000, b); + else +#endif + *((u16 *)&vram[address]) = (b << 8) | b; + } break; case 7: // no need to switch @@ -2944,8 +3351,11 @@ void CPUWriteByte(u32 address, u8 b) } goto unwritable; case 14: - if(!eepromInUse | cpuSramEnabled | cpuFlashEnabled) { - (*cpuSaveGameFunc)(address, b); + if (!(saveType == 5) && (!eepromInUse | cpuSramEnabled | cpuFlashEnabled)) { + + //if(!cpuEEPROMEnabled && (cpuSramEnabled | cpuFlashEnabled)) { + + (*cpuSaveGameFunc)(address, b); break; } // default @@ -2962,6 +3372,7 @@ void CPUWriteByte(u32 address, u8 b) break; } } + u8 cpuBitsSet[256]; u8 cpuLowestBitSet[256]; @@ -3077,7 +3488,7 @@ void CPUReset() } } rtcReset(); - // clen registers + // clean registers memset(®[0], 0, sizeof(reg)); // clean OAM memset(oam, 0, 0x400); @@ -3092,7 +3503,7 @@ void CPUReset() DISPCNT = 0x0080; DISPSTAT = 0x0000; - VCOUNT = 0x0000; + VCOUNT = (useBios && !skipBios) ? 0 :0x007E; BG0CNT = 0x0000; BG1CNT = 0x0000; BG2CNT = 0x0000; @@ -3194,6 +3605,7 @@ void CPUReset() armState = true; C_FLAG = V_FLAG = N_FLAG = Z_FLAG = false; UPDATE_REG(0x00, DISPCNT); + UPDATE_REG(0x06, VCOUNT); UPDATE_REG(0x20, BG2PA); UPDATE_REG(0x26, BG2PD); UPDATE_REG(0x30, BG3PA); @@ -3218,7 +3630,7 @@ void CPUReset() biosProtected[2] = 0x29; biosProtected[3] = 0xe1; - lcdTicks = 960; + lcdTicks = (useBios && !skipBios) ? 1008 : 208; timer0On = false; timer0Ticks = 0; timer0Reload = 0; @@ -3308,12 +3720,14 @@ void CPUReset() cpuFlashEnabled = true; cpuEEPROMEnabled = true; cpuEEPROMSensorEnabled = false; + saveType = gbaSaveType = 0; break; case 1: // EEPROM cpuSramEnabled = false; cpuFlashEnabled = false; cpuEEPROMEnabled = true; cpuEEPROMSensorEnabled = false; + saveType = gbaSaveType = 3; // EEPROM usage is automatically detected break; case 2: // SRAM @@ -3322,6 +3736,7 @@ void CPUReset() cpuEEPROMEnabled = false; cpuEEPROMSensorEnabled = false; cpuSaveGameFunc = sramDelayedWrite; // to insure we detect the write + saveType = gbaSaveType = 1; break; case 3: // FLASH cpuSramEnabled = false; @@ -3329,6 +3744,7 @@ void CPUReset() cpuEEPROMEnabled = false; cpuEEPROMSensorEnabled = false; cpuSaveGameFunc = flashDelayedWrite; // to insure we detect the write + saveType = gbaSaveType = 2; break; case 4: // EEPROM+Sensor cpuSramEnabled = false; @@ -3336,6 +3752,7 @@ void CPUReset() cpuEEPROMEnabled = true; cpuEEPROMSensorEnabled = true; // EEPROM usage is automatically detected + saveType = gbaSaveType = 3; break; case 5: // NONE cpuSramEnabled = false; @@ -3343,14 +3760,19 @@ void CPUReset() cpuEEPROMEnabled = false; cpuEEPROMSensorEnabled = false; // no save at all + saveType = gbaSaveType = 5; break; } + ARM_PREFETCH; + systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; - cpuDmaHack2 = false; - + cpuDmaHack = false; + lastTime = systemGetClock(); + + SWITicks = 0; } void CPUInterrupt() @@ -3367,6 +3789,7 @@ void CPUInterrupt() armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; // if(!holdState) biosProtected[0] = 0x02; @@ -3399,25 +3822,30 @@ extern void winlog(const char *, ...); void CPULoop(int ticks) { int clockTicks; - int totalTicks = 0; int timerOverflow = 0; - // variables used by the CPU core - cpuNextEvent = CPUUpdateTicks(); - if(cpuNextEvent > ticks) { - cpuNextEvent = ticks; - } - - if(intState) { - cpuNextEvent = 5; - } - if(cpuDmaHack2) - cpuNextEvent = 1; + // variable used by the CPU core + cpuTotalTicks = 0; cpuBreakLoop = false; + cpuNextEvent = CPUUpdateTicks(); + if(cpuNextEvent > ticks) + cpuNextEvent = ticks; + + for(;;) { #ifndef FINAL_VERSION if(systemDebug) { if(systemDebug >= 10 && !holdState) { CPUUpdateCPSR(); +#ifdef BKPT_SUPPORT + if (debugger_last) + { + sprintf(buffer, "R00=%08x R01=%08x R02=%08x R03=%08x R04=%08x R05=%08x R06=%08x R07=%08x R08=%08x R09=%08x R10=%08x R11=%08x R12=%08x R13=%08x R14=%08x R15=%08x R16=%08x R17=%08x\n", + oldreg[0], oldreg[1], oldreg[2], oldreg[3], oldreg[4], oldreg[5], + oldreg[6], oldreg[7], oldreg[8], oldreg[9], oldreg[10], oldreg[11], + oldreg[12], oldreg[13], oldreg[14], oldreg[15], oldreg[16], + oldreg[17]); + } +#endif sprintf(buffer, "R00=%08x R01=%08x R02=%08x R03=%08x R04=%08x R05=%08x R06=%08x R07=%08x R08=%08x R09=%08x R10=%08x R11=%08x R12=%08x R13=%08x R14=%08x R15=%08x R16=%08x R17=%08x\n", reg[0].I, reg[1].I, reg[2].I, reg[3].I, reg[4].I, reg[5].I, reg[6].I, reg[7].I, reg[8].I, reg[9].I, reg[10].I, reg[11].I, @@ -3437,67 +3865,72 @@ void CPULoop(int ticks) #endif } } -#endif +#endif /* FINAL_VERSION */ + + if(!holdState && !SWITicks) { + + // Emulates the Cheat System (m) code + if((cheatsEnabled) && (mastercode) && (mastercode == armNextPC)) + { + u32 joy = 0; + if(systemReadJoypads()) + joy = systemReadJoypad(-1); + u32 ext = (joy >> 10); + cpuTotalTicks += cheatsCheckKeys(P1^0x3FF, ext); + } + + if ((armNextPC & 0x0803FFFF) == 0x08020000) + busPrefetchCount=0x100; - if(!holdState) { if(armState) { #include "arm-new.h" } else { #include "thumb.h" } - } else { - clockTicks = lcdTicks; + } else + clockTicks = CPUUpdateTicks(); - if(soundTicks < clockTicks) - clockTicks = soundTicks; - - if(timer0On && (timer0Ticks < clockTicks)) { - clockTicks = timer0Ticks; + cpuTotalTicks += clockTicks; + + + if(cpuTotalTicks >= cpuNextEvent) { + int remainingTicks = cpuTotalTicks - cpuNextEvent; + + if (SWITicks) + { + SWITicks-=clockTicks; + if (SWITicks<0) + SWITicks = 0; } - if(timer1On && !(TM1CNT & 4) && (timer1Ticks < clockTicks)) { - clockTicks = timer1Ticks; - } - if(timer2On && !(TM2CNT & 4) && (timer2Ticks < clockTicks)) { - clockTicks = timer2Ticks; - } - if(timer3On && !(TM3CNT & 4) && (timer3Ticks < clockTicks)) { - clockTicks = timer3Ticks; - } -#ifdef PROFILING - if(profilingTicksReload != 0) { - if(profilingTicks < clockTicks) { - clockTicks = profilingTicks; - } - } -#endif - // if((clockTicks+totalTicks) > cpuNextEvent) { - // clockTicks = cpuNextEvent - totalTicks; - // } - } - totalTicks += clockTicks; - cpuTotalTicks = totalTicks; - if(totalTicks >= cpuNextEvent) { - cpuDmaHack2 = false; - int remainingTicks = totalTicks - cpuNextEvent; + clockTicks = cpuNextEvent; - int oldTotalTicks = totalTicks; - cpuTotalTicks = totalTicks = 0; - + cpuTotalTicks = 0; + cpuDmaHack = false; + updateLoop: + + if (IRQTicks) + { + IRQTicks -= clockTicks; + if (IRQTicks<0) + IRQTicks = 0; + } + lcdTicks -= clockTicks; + if(lcdTicks <= 0) { if(DISPSTAT & 1) { // V-BLANK // if in V-Blank mode, keep computing... if(DISPSTAT & 2) { - lcdTicks += 960; + lcdTicks += 1008; VCOUNT++; UPDATE_REG(0x06, VCOUNT); DISPSTAT &= 0xFFFD; UPDATE_REG(0x04, DISPSTAT); CPUCompareVCOUNT(); } else { - lcdTicks += 272; + lcdTicks += 224; DISPSTAT |= 2; UPDATE_REG(0x04, DISPSTAT); if(DISPSTAT & 16) { @@ -3506,7 +3939,7 @@ void CPULoop(int ticks) } } - if(VCOUNT >= 228) { + if(VCOUNT >= 228) { //Reaching last line DISPSTAT &= 0xFFFC; UPDATE_REG(0x04, DISPSTAT); VCOUNT = 0; @@ -3522,8 +3955,8 @@ void CPULoop(int ticks) // if in H-Blank, leave it and move to drawing mode VCOUNT++; UPDATE_REG(0x06, VCOUNT); - - lcdTicks += (960); + + lcdTicks += 1008; DISPSTAT &= 0xFFFD; if(VCOUNT == 160) { count++; @@ -3571,7 +4004,8 @@ void CPULoop(int ticks) } u32 ext = (joy >> 10); - if(cheatsEnabled) + // If no (m) code is enabled, apply the cheats at each LCDline + if((cheatsEnabled) && (mastercode==0)) remainingTicks += cheatsCheckKeys(P1^0x3FF, ext); speedup = (ext & 1) ? true : false; capture = (ext & 2) ? true : false; @@ -3581,7 +4015,7 @@ void CPULoop(int ticks) systemScreenCapture(captureNumber); } capturePrevious = capture; - + DISPSTAT |= 1; DISPSTAT &= 0xFFFD; UPDATE_REG(0x04, DISPSTAT); @@ -3589,7 +4023,7 @@ void CPULoop(int ticks) IF |= 1; UPDATE_REG(0x202, IF); } - cpuDmaHack2 = CPUCheckDMA(1, 0x0f); + CPUCheckDMA(1, 0x0f); if(frameCount >= framesToSkip) { systemDrawScreen(); frameCount = 0; @@ -3600,14 +4034,14 @@ void CPULoop(int ticks) } UPDATE_REG(0x04, DISPSTAT); - - CPUCompareVCOUNT(); + CPUCompareVCOUNT(); + } else { - if(frameCount >= framesToSkip) { - (*renderLine)(); - + if(frameCount >= framesToSkip) + { + (*renderLine)(); switch(systemColorDepth) { - case 16: + case 16: { u16 *dest = (u16 *)pix + 242 * (VCOUNT+1); for(int x = 0; x < 240;) { @@ -3635,7 +4069,7 @@ void CPULoop(int ticks) *dest++ = 0; } break; - case 24: + case 24: { u8 *dest = (u8 *)pix + 240 * VCOUNT * 3; for(int x = 0; x < 240;) { @@ -3677,7 +4111,7 @@ void CPULoop(int ticks) } } break; - case 32: + case 32: { u32 *dest = (u32 *)pix + 241 * (VCOUNT+1); for(int x = 0; x < 240; ) { @@ -3685,7 +4119,7 @@ void CPULoop(int ticks) *dest++ = systemColorMap32[lineMix[x++] & 0xFFFF]; *dest++ = systemColorMap32[lineMix[x++] & 0xFFFF]; *dest++ = systemColorMap32[lineMix[x++] & 0xFFFF]; - + *dest++ = systemColorMap32[lineMix[x++] & 0xFFFF]; *dest++ = systemColorMap32[lineMix[x++] & 0xFFFF]; *dest++ = systemColorMap32[lineMix[x++] & 0xFFFF]; @@ -3708,8 +4142,8 @@ void CPULoop(int ticks) // entering H-Blank DISPSTAT |= 2; UPDATE_REG(0x04, DISPSTAT); - lcdTicks += 272; - cpuDmaHack2 = CPUCheckDMA(2, 0x0f); + lcdTicks += 224; + CPUCheckDMA(2, 0x0f); if(DISPSTAT & 16) { IF |= 2; UPDATE_REG(0x202, IF); @@ -3720,39 +4154,18 @@ void CPULoop(int ticks) if(!stopState) { if(timer0On) { - if(timer0ClockReload == 1) { - u32 tm0d = TM0D + clockTicks; - if(tm0d > 0xffff) { - tm0d += timer0Reload; - timerOverflow |= 1; - soundTimerOverflow(0); - if(TM0CNT & 0x40) { - IF |= 0x08; - UPDATE_REG(0x202, IF); - } - } - TM0D = tm0d; - if(TM0D < timer0Reload) - TM0D = timer0Reload; - timer0Ticks = 0x10000 - TM0D; - UPDATE_REG(0x100, TM0D); - } else { - timer0Ticks -= clockTicks; - if(timer0Ticks <= 0) { - timer0Ticks += timer0ClockReload; - TM0D++; - if(TM0D == 0) { - TM0D = timer0Reload; - timerOverflow |= 1; - soundTimerOverflow(0); - if(TM0CNT & 0x40) { - IF |= 0x08; - UPDATE_REG(0x202, IF); - } - } - UPDATE_REG(0x100, TM0D); + timer0Ticks -= clockTicks; + if(timer0Ticks <= 0) { + timer0Ticks += (0x10000 - timer0Reload) << timer0ClockReload; + timerOverflow |= 1; + soundTimerOverflow(0); + if(TM0CNT & 0x40) { + IF |= 0x08; + UPDATE_REG(0x202, IF); } } + TM0D = 0xFFFF - (timer0Ticks >> timer0ClockReload); + UPDATE_REG(0x100, TM0D); } if(timer1On) { @@ -3771,40 +4184,18 @@ void CPULoop(int ticks) UPDATE_REG(0x104, TM1D); } } else { - if(timer1ClockReload == 1) { - u32 tm1d = TM1D + clockTicks; - if(tm1d > 0xffff) { - tm1d += timer1Reload; - timerOverflow |= 2; - soundTimerOverflow(1); - if(TM1CNT & 0x40) { - IF |= 0x10; - UPDATE_REG(0x202, IF); - } - } - TM1D = tm1d; - if(TM1D < timer1Reload) - TM1D = timer1Reload; - timer1Ticks = 0x10000 - TM1D; - UPDATE_REG(0x104, TM1D); - } else { - timer1Ticks -= clockTicks; - if(timer1Ticks <= 0) { - timer1Ticks += timer1ClockReload; - TM1D++; - - if(TM1D == 0) { - TM1D = timer1Reload; - timerOverflow |= 2; - soundTimerOverflow(1); - if(TM1CNT & 0x40) { - IF |= 0x10; - UPDATE_REG(0x202, IF); - } - } - UPDATE_REG(0x104, TM1D); + timer1Ticks -= clockTicks; + if(timer1Ticks <= 0) { + timer1Ticks += (0x10000 - timer1Reload) << timer1ClockReload; + timerOverflow |= 2; + soundTimerOverflow(1); + if(TM1CNT & 0x40) { + IF |= 0x10; + UPDATE_REG(0x202, IF); } } + TM1D = 0xFFFF - (timer1Ticks >> timer1ClockReload); + UPDATE_REG(0x104, TM1D); } } @@ -3823,38 +4214,17 @@ void CPULoop(int ticks) UPDATE_REG(0x108, TM2D); } } else { - if(timer2ClockReload == 1) { - u32 tm2d = TM2D + clockTicks; - if(tm2d > 0xffff) { - tm2d += timer2Reload; - timerOverflow |= 4; - if(TM2CNT & 0x40) { - IF |= 0x20; - UPDATE_REG(0x202, IF); - } - } - TM2D = tm2d; - if(TM2D < timer2Reload) - TM2D = timer2Reload; - timer2Ticks = 0x10000 - TM2D; - UPDATE_REG(0x108, TM2D); - } else { - timer2Ticks -= clockTicks; - if(timer2Ticks <= 0) { - timer2Ticks += timer2ClockReload; - TM2D++; - - if(TM2D == 0) { - TM2D = timer2Reload; - timerOverflow |= 4; - if(TM2CNT & 0x40) { - IF |= 0x20; - UPDATE_REG(0x202, IF); - } - } - UPDATE_REG(0x108, TM2D); + timer2Ticks -= clockTicks; + if(timer2Ticks <= 0) { + timer2Ticks += (0x10000 - timer2Reload) << timer2ClockReload; + timerOverflow |= 4; + if(TM2CNT & 0x40) { + IF |= 0x20; + UPDATE_REG(0x202, IF); } } + TM2D = 0xFFFF - (timer2Ticks >> timer2ClockReload); + UPDATE_REG(0x108, TM2D); } } @@ -3869,43 +4239,26 @@ void CPULoop(int ticks) UPDATE_REG(0x202, IF); } } - UPDATE_REG(0x10c, TM3D); + UPDATE_REG(0x10C, TM3D); } } else { - if(timer3ClockReload == 1) { - u32 tm3d = TM3D + clockTicks; - if(tm3d > 0xffff) { - tm3d += timer3Reload; - if(TM3CNT & 0x40) { - IF |= 0x40; - UPDATE_REG(0x202, IF); - } - } - TM3D = tm3d; - if(TM3D < timer3Reload) - TM3D = timer3Reload; - timer3Ticks = 0x10000 - TM3D; - UPDATE_REG(0x10C, TM3D); - } else { - timer3Ticks -= clockTicks; - if(timer3Ticks <= 0) { - timer3Ticks += timer3ClockReload; - TM3D++; - - if(TM3D == 0) { - TM3D = timer3Reload; - if(TM3CNT & 0x40) { - IF |= 0x40; - UPDATE_REG(0x202, IF); - } - } - UPDATE_REG(0x10C, TM3D); + timer3Ticks -= clockTicks; + if(timer3Ticks <= 0) { + timer3Ticks += (0x10000 - timer3Reload) << timer3ClockReload; + if(TM3CNT & 0x40) { + IF |= 0x40; + UPDATE_REG(0x202, IF); } } + TM3D = 0xFFFF - (timer3Ticks >> timer3ClockReload); + UPDATE_REG(0x10C, TM3D); } } } - // we shouldn't be doing sound in stop state, but we lose synchronization + + timerOverflow = 0; + + // we shouldn't be doing sound in stop state, but we loose synchronization // if sound is disabled, so in stop state, soundTick will just produce // mute sound soundTicks -= clockTicks; @@ -3913,88 +4266,110 @@ void CPULoop(int ticks) psoundTickfn(); soundTicks += SOUND_CLOCK_TICKS; } - timerOverflow = 0; #ifdef PROFILING profilingTicks -= clockTicks; if(profilingTicks <= 0) { profilingTicks += profilingTicksReload; - if(profilBuffer && profilSize) { - u16 *b = (u16 *)profilBuffer; - int pc = ((reg[15].I - profilLowPC) * profilScale)/0x10000; - if(pc >= 0 && pc < profilSize) { + if(profilSegment) { + profile_segment *seg = profilSegment; + do { + u16 *b = (u16 *)seg->sbuf; + int pc = ((reg[15].I - seg->s_lowpc) * seg->s_scale)/0x10000; + if(pc >= 0 && pc < seg->ssiz) { b[pc]++; + break; } + + seg = seg->next; + } while(seg); } } #endif ticks -= clockTicks; - - -/* Link -----------------------------------*/ - LinkUpdate(clockTicks); -/* ----------------------------- */ cpuNextEvent = CPUUpdateTicks(); if(cpuDmaTicksToUpdate > 0) { - if(cpuDmaTicksToUpdate < cpuNextEvent) - cpuNextEvent = cpuDmaTicksToUpdate; - cpuDmaTicksToUpdate -= cpuNextEvent; - clockTicks = cpuNextEvent; + if(cpuDmaTicksToUpdate > cpuNextEvent) + clockTicks = cpuNextEvent; + else + clockTicks = cpuDmaTicksToUpdate; + cpuDmaTicksToUpdate -= clockTicks; if(cpuDmaTicksToUpdate < 0) cpuDmaTicksToUpdate = 0; + cpuDmaHack = true; goto updateLoop; } - - if(cpuDmaHack2) - cpuNextEvent = 1; if(IF && (IME & 1) && armIrqEnable) { int res = IF & IE; if(stopState) res &= 0x3080; if(res) { - if(intState) { - CPUInterrupt(); - intState = false; - if(holdState) { + if (intState) + { + if (!IRQTicks) + { + CPUInterrupt(); + intState = false; holdState = false; stopState = false; - } - } else { - if(!holdState) { - intState = true; - cpuNextEvent = 5; - } else { - CPUInterrupt(); - if(holdState) { - holdState = false; - stopState = false; - } + holdType = 0; } } + else + { + if (!holdState) + { + intState = true; + IRQTicks=7; + if (cpuNextEvent> IRQTicks) + cpuNextEvent = IRQTicks; + } + else + { + CPUInterrupt(); + holdState = false; + stopState = false; + holdType = 0; + } + } + + // Stops the SWI Ticks emulation if an IRQ is executed + //(to avoid problems with nested IRQ/SWI) + if (SWITicks) + SWITicks = 0; } } - + if(remainingTicks > 0) { - if(remainingTicks < cpuNextEvent) - cpuNextEvent = remainingTicks; - remainingTicks -= cpuNextEvent; - clockTicks = cpuNextEvent; + if(remainingTicks > cpuNextEvent) + clockTicks = cpuNextEvent; + else + clockTicks = remainingTicks; + remainingTicks -= clockTicks; if(remainingTicks < 0) remainingTicks = 0; goto updateLoop; } - + + if (timerOnOffDelay) + applyTimer(); + + if(cpuNextEvent > ticks) + cpuNextEvent = ticks; + if(ticks <= 0 || cpuBreakLoop) break; + } } } + + struct EmulatedSystem GBASystem = { // emuMain CPULoop, @@ -4008,7 +4383,7 @@ struct EmulatedSystem GBASystem = { CPUWriteBatteryFile, // emuReadState CPUReadState, - // emuWriteState + // emuWriteState CPUWriteState, // emuReadMemState CPUReadMemState, diff --git a/src/GBA.h b/src/GBA.h index d6b8fa39..a79603b2 100644 --- a/src/GBA.h +++ b/src/GBA.h @@ -1,147 +1,158 @@ -// -*- C++ -*- -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#ifndef VBA_GBA_H -#define VBA_GBA_H - -#include "System.h" - -#define SAVE_GAME_VERSION_1 1 -#define SAVE_GAME_VERSION_2 2 -#define SAVE_GAME_VERSION_3 3 -#define SAVE_GAME_VERSION_4 4 -#define SAVE_GAME_VERSION_5 5 -#define SAVE_GAME_VERSION_6 6 -#define SAVE_GAME_VERSION_7 7 -#define SAVE_GAME_VERSION_8 8 -#define SAVE_GAME_VERSION SAVE_GAME_VERSION_8 - -typedef struct { - u8 *address; - u32 mask; -} memoryMap; - -typedef union { - struct { -#ifdef WORDS_BIGENDIAN - u8 B3; - u8 B2; - u8 B1; - u8 B0; -#else - u8 B0; - u8 B1; - u8 B2; - u8 B3; -#endif - } B; - struct { -#ifdef WORDS_BIGENDIAN - u16 W1; - u16 W0; -#else - u16 W0; - u16 W1; -#endif - } W; -#ifdef WORDS_BIGENDIAN - volatile u32 I; -#else - u32 I; -#endif -} reg_pair; - -#ifndef NO_GBA_MAP -extern memoryMap map[256]; -#endif - -extern reg_pair reg[45]; -extern u8 biosProtected[4]; - -extern bool N_FLAG; -extern bool Z_FLAG; -extern bool C_FLAG; -extern bool V_FLAG; -extern bool armIrqEnable; -extern bool armState; -extern int armMode; -extern void (*cpuSaveGameFunc)(u32,u8); - -extern u8 freezeWorkRAM[0x40000]; -extern u8 freezeInternalRAM[0x8000]; -extern bool CPUReadGSASnapshot(const char *); -extern bool CPUWriteGSASnapshot(const char *, const char *, const char *, const char *); -extern bool CPUWriteBatteryFile(const char *); -extern bool CPUReadBatteryFile(const char *); -extern bool CPUExportEepromFile(const char *); -extern bool CPUImportEepromFile(const char *); -extern bool CPUWritePNGFile(const char *); -extern bool CPUWriteBMPFile(const char *); -extern void CPUCleanUp(); -extern void CPUUpdateRender(); -extern bool CPUReadMemState(char *, int); -extern bool CPUReadState(const char *); -extern bool CPUWriteMemState(char *, int); -extern bool CPUWriteState(const char *); -extern int CPULoadRom(const char *); -extern void CPUUpdateRegister(u32, u16); -extern void CPUWriteHalfWord(u32, u16); -extern void CPUWriteByte(u32, u8); -extern void CPUInit(const char *,bool); -extern void CPUReset(); -extern void CPULoop(int); -extern bool CPUCheckDMA(int,int); -extern bool CPUIsGBAImage(const char *); -extern bool CPUIsZipFile(const char *); -#ifdef PROFILING -extern void cpuProfil(char *buffer, int, u32, int); -extern void cpuEnableProfiling(int hz); -#endif - -extern struct EmulatedSystem GBASystem; - -#define R13_IRQ 18 -#define R14_IRQ 19 -#define SPSR_IRQ 20 -#define R13_USR 26 -#define R14_USR 27 -#define R13_SVC 28 -#define R14_SVC 29 -#define SPSR_SVC 30 -#define R13_ABT 31 -#define R14_ABT 32 -#define SPSR_ABT 33 -#define R13_UND 34 -#define R14_UND 35 -#define SPSR_UND 36 -#define R8_FIQ 37 -#define R9_FIQ 38 -#define R10_FIQ 39 -#define R11_FIQ 40 -#define R12_FIQ 41 -#define R13_FIQ 42 -#define R14_FIQ 43 -#define SPSR_FIQ 44 - -#include "Cheats.h" -#include "Globals.h" -#include "EEprom.h" -#include "Flash.h" - -#endif //VBA_GBA_H +// -*- C++ -*- +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2005 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#ifndef VBA_GBA_H +#define VBA_GBA_H + +#include "System.h" + +#define SAVE_GAME_VERSION_1 1 +#define SAVE_GAME_VERSION_2 2 +#define SAVE_GAME_VERSION_3 3 +#define SAVE_GAME_VERSION_4 4 +#define SAVE_GAME_VERSION_5 5 +#define SAVE_GAME_VERSION_6 6 +#define SAVE_GAME_VERSION_7 7 +#define SAVE_GAME_VERSION_8 8 +#define SAVE_GAME_VERSION_9 9 +#define SAVE_GAME_VERSION SAVE_GAME_VERSION_9 + +typedef struct { + u8 *address; + u32 mask; +} memoryMap; + +typedef union { + struct { +#ifdef WORDS_BIGENDIAN + u8 B3; + u8 B2; + u8 B1; + u8 B0; +#else + u8 B0; + u8 B1; + u8 B2; + u8 B3; +#endif + } B; + struct { +#ifdef WORDS_BIGENDIAN + u16 W1; + u16 W0; +#else + u16 W0; + u16 W1; +#endif + } W; +#ifdef WORDS_BIGENDIAN + volatile u32 I; +#else + u32 I; +#endif +} reg_pair; + +#ifndef NO_GBA_MAP +extern memoryMap map[256]; +#endif + +extern reg_pair reg[45]; +extern u8 biosProtected[4]; + +extern bool N_FLAG; +extern bool Z_FLAG; +extern bool C_FLAG; +extern bool V_FLAG; +extern bool armIrqEnable; +extern bool armState; +extern int armMode; +extern void (*cpuSaveGameFunc)(u32,u8); + +extern u8 freezeWorkRAM[0x40000]; +extern u8 freezeInternalRAM[0x8000]; +extern u8 freezeVRAM[0x18000]; +extern u8 freezeOAM[0x400]; +extern u8 freezePRAM[0x400]; +extern bool debugger_last; +extern int oldreg[17]; +extern char oldbuffer[10]; + +extern bool CPUReadGSASnapshot(const char *); +extern bool CPUWriteGSASnapshot(const char *, const char *, const char *, const char *); +extern bool CPUWriteBatteryFile(const char *); +extern bool CPUReadBatteryFile(const char *); +extern bool CPUExportEepromFile(const char *); +extern bool CPUImportEepromFile(const char *); +extern bool CPUWritePNGFile(const char *); +extern bool CPUWriteBMPFile(const char *); +extern void CPUCleanUp(); +extern void CPUUpdateRender(); +extern bool CPUReadMemState(char *, int); +extern bool CPUReadState(const char *); +extern bool CPUWriteMemState(char *, int); +extern bool CPUWriteState(const char *); +extern int CPULoadRom(const char *); +extern void doMirroring(bool); +extern void CPUUpdateRegister(u32, u16); +extern void applyTimer (); +extern void CPUWriteHalfWord(u32, u16); +extern void CPUWriteByte(u32, u8); +extern void CPUInit(const char *,bool); +extern void CPUReset(); +extern void CPULoop(int); +extern void CPUCheckDMA(int,int); +extern bool CPUIsGBAImage(const char *); +extern bool CPUIsZipFile(const char *); +#ifdef PROFILING +#include "prof/prof.h" +extern void cpuProfil(profile_segment *seg); +extern void cpuEnableProfiling(int hz); +#endif + +extern struct EmulatedSystem GBASystem; + +#define R13_IRQ 18 +#define R14_IRQ 19 +#define SPSR_IRQ 20 +#define R13_USR 26 +#define R14_USR 27 +#define R13_SVC 28 +#define R14_SVC 29 +#define SPSR_SVC 30 +#define R13_ABT 31 +#define R14_ABT 32 +#define SPSR_ABT 33 +#define R13_UND 34 +#define R14_UND 35 +#define SPSR_UND 36 +#define R8_FIQ 37 +#define R9_FIQ 38 +#define R10_FIQ 39 +#define R11_FIQ 40 +#define R12_FIQ 41 +#define R13_FIQ 42 +#define R14_FIQ 43 +#define SPSR_FIQ 44 + +#include "Cheats.h" +#include "Globals.h" +#include "EEprom.h" +#include "Flash.h" + +#endif //VBA_GBA_H diff --git a/src/GBAinline.h b/src/GBAinline.h index 1c482c85..7221763c 100644 --- a/src/GBAinline.h +++ b/src/GBAinline.h @@ -1,427 +1,488 @@ -// -*- C++ -*- -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#ifndef VBA_GBAinline_H -#define VBA_GBAinline_H - -#include "System.h" -#include "Port.h" -#include "RTC.h" -#include "Sound.h" - -extern bool cpuSramEnabled; -extern bool cpuFlashEnabled; -extern bool cpuEEPROMEnabled; -extern bool cpuEEPROMSensorEnabled; -extern bool cpuDmaHack; -extern bool cpuDmaHack2; -extern u32 cpuDmaLast; - -extern int lspeed; -extern void LinkSStop(void); - -#define CPUReadByteQuick(addr) \ - map[(addr)>>24].address[(addr) & map[(addr)>>24].mask] - -#define CPUReadHalfWordQuick(addr) \ - READ16LE(((u16*)&map[(addr)>>24].address[(addr) & map[(addr)>>24].mask])) - -#define CPUReadMemoryQuick(addr) \ - READ32LE(((u32*)&map[(addr)>>24].address[(addr) & map[(addr)>>24].mask])) - -inline u32 CPUReadMemory(u32 address) -{ - -#ifdef DEV_VERSION - if(address & 3) { - if(systemVerbose & VERBOSE_UNALIGNED_MEMORY) { - log("Unaligned word read: %08x at %08x\n", address, armMode ? - armNextPC - 4 : armNextPC - 2); - } - } -#endif - - u32 value; - switch(address >> 24) { - case 0: - if(reg[15].I >> 24) { - if(address < 0x4000) { -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_ILLEGAL_READ) { - log("Illegal word read: %08x at %08x\n", address, armMode ? - armNextPC - 4 : armNextPC - 2); - } -#endif - - value = READ32LE(((u32 *)&biosProtected)); - } - else goto unreadable; - } else - value = READ32LE(((u32 *)&bios[address & 0x3FFC])); - break; - case 2: - value = READ32LE(((u32 *)&workRAM[address & 0x3FFFC])); - break; - case 3: - value = READ32LE(((u32 *)&internalRAM[address & 0x7ffC])); - break; - case 4: - if((address>=0x4000120||address<=0x4000126)&&lspeed) - LinkSStop(); - if((address < 0x4000400) && ioReadable[address & 0x3fc]) { - if(ioReadable[(address & 0x3fc) + 2]) - value = soundRead32(address & 0x3fC); - else - value = soundRead16(address & 0x3fc); - } else goto unreadable; - break; - case 5: - value = READ32LE(((u32 *)&paletteRAM[address & 0x3fC])); - break; - case 6: - value = READ32LE(((u32 *)&vram[address & 0x1fffc])); - break; - case 7: - value = READ32LE(((u32 *)&oam[address & 0x3FC])); - break; - case 8: - case 9: - case 10: - case 11: - case 12: - value = READ32LE(((u32 *)&rom[address&0x1FFFFFC])); - break; - case 13: - if(cpuEEPROMEnabled) - // no need to swap this - return eepromRead(address); - goto unreadable; - case 14: - if(cpuFlashEnabled | cpuSramEnabled) - // no need to swap this - return flashRead(address); - // default - default: - unreadable: -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_ILLEGAL_READ) { - log("Illegal word read: %08x at %08x\n", address, armMode ? - armNextPC - 4 : armNextPC - 2); - } -#endif - - if(cpuDmaHack || cpuDmaHack2) { - value = cpuDmaLast; - } else { - if(armState) { - value = CPUReadMemoryQuick(reg[15].I); - } else { - value = CPUReadHalfWordQuick(reg[15].I) | - CPUReadHalfWordQuick(reg[15].I) << 16; - } - } - } - - if(address & 3) { -#ifdef C_CORE - int shift = (address & 3) << 3; - value = (value >> shift) | (value << (32 - shift)); -#else -#ifdef __GNUC__ - asm("and $3, %%ecx;" - "shl $3 ,%%ecx;" - "ror %%cl, %0" - : "=r" (value) - : "r" (value), "c" (address)); -#else - __asm { - mov ecx, address; - and ecx, 3; - shl ecx, 3; - ror [dword ptr value], cl; - } -#endif -#endif - } - return value; -} - -extern u32 myROM[]; - -inline u32 CPUReadHalfWord(u32 address) -{ -#ifdef DEV_VERSION - if(address & 1) { - if(systemVerbose & VERBOSE_UNALIGNED_MEMORY) { - log("Unaligned halfword read: %08x at %08x\n", address, armMode ? - armNextPC - 4 : armNextPC - 2); - } - } -#endif - - u32 value; - - switch(address >> 24) { - case 0: - if (reg[15].I >> 24) { - if(address < 0x4000) { -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_ILLEGAL_READ) { - log("Illegal halfword read: %08x at %08x\n", address, armMode ? - armNextPC - 4 : armNextPC - 2); - } -#endif - value = READ16LE(((u16 *)&biosProtected[address&2])); - } else goto unreadable; - } else - value = READ16LE(((u16 *)&bios[address & 0x3FFE])); - break; - case 2: - value = READ16LE(((u16 *)&workRAM[address & 0x3FFFE])); - break; - case 3: - value = READ16LE(((u16 *)&internalRAM[address & 0x7ffe])); - break; - case 4: -if((address>=0x4000120||address<=0x4000126)&&lspeed) - LinkSStop(); - if((address < 0x4000400) && ioReadable[address & 0x3fe]) - value = READ16LE(((u16 *)&ioMem[address & 0x3fe])); - else goto unreadable; - break; - case 5: - value = READ16LE(((u16 *)&paletteRAM[address & 0x3fe])); - break; - case 6: - value = READ16LE(((u16 *)&vram[address & 0x1fffe])); - break; - case 7: - value = READ16LE(((u16 *)&oam[address & 0x3fe])); - break; - case 8: - case 9: - case 10: - case 11: - case 12: - if(address == 0x80000c4 || address == 0x80000c6 || address == 0x80000c8) - value = rtcRead(address); - else - value = READ16LE(((u16 *)&rom[address & 0x1FFFFFE])); - break; - case 13: - if(cpuEEPROMEnabled) - // no need to swap this - return eepromRead(address); - goto unreadable; - case 14: - if(cpuFlashEnabled | cpuSramEnabled) - // no need to swap this - return flashRead(address); - // default - default: - unreadable: -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_ILLEGAL_READ) { - log("Illegal halfword read: %08x at %08x\n", address, armMode ? - armNextPC - 4 : armNextPC - 2); - } -#endif - if(cpuDmaHack2 || cpuDmaHack) { - value = cpuDmaLast & 0xFFFF; - } else { - if(armState) { - value = CPUReadHalfWordQuick(reg[15].I + (address & 2)); - } else { - value = CPUReadHalfWordQuick(reg[15].I); - } - } - break; - } - - if(address & 1) { - value = (value >> 8) | (value << 24); - } - - return value; -} - -inline u16 CPUReadHalfWordSigned(u32 address) -{ - u16 value = CPUReadHalfWord(address); - if((address & 1)) - value = (s8)value; - return value; -} - -inline u8 CPUReadByte(u32 address) -{ - switch(address >> 24) { - case 0: - if (reg[15].I >> 24) { - if(address < 0x4000) { -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_ILLEGAL_READ) { - log("Illegal byte read: %08x at %08x\n", address, armMode ? - armNextPC - 4 : armNextPC - 2); - } -#endif - return biosProtected[address & 3]; - } else goto unreadable; - } - return bios[address & 0x3FFF]; - case 2: - return workRAM[address & 0x3FFFF]; - case 3: - return internalRAM[address & 0x7fff]; - case 4: - if((address>=0x4000120||address<=0x4000126)&&lspeed) - LinkSStop(); - if((address < 0x4000400) && ioReadable[address & 0x3ff]) - return soundRead(address & 0x3ff); - else goto unreadable; - case 5: - return paletteRAM[address & 0x3ff]; - case 6: - return vram[address & 0x1ffff]; - case 7: - return oam[address & 0x3ff]; - case 8: - case 9: - case 10: - case 11: - case 12: - return rom[address & 0x1FFFFFF]; - case 13: - if(cpuEEPROMEnabled) - return eepromRead(address); - goto unreadable; - case 14: - if(cpuSramEnabled | cpuFlashEnabled) - return flashRead(address); - if(cpuEEPROMSensorEnabled) { - switch(address & 0x00008f00) { - case 0x8200: - return systemGetSensorX() & 255; - case 0x8300: - return (systemGetSensorX() >> 8)|0x80; - case 0x8400: - return systemGetSensorY() & 255; - case 0x8500: - return systemGetSensorY() >> 8; - } - } - // default - default: - unreadable: -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_ILLEGAL_READ) { - log("Illegal byte read: %08x at %08x\n", address, armMode ? - armNextPC - 4 : armNextPC - 2); - } -#endif - if(cpuDmaHack || cpuDmaHack2) { - return cpuDmaLast & 0xFF; - } else { - if(armState) { - return CPUReadByteQuick(reg[15].I+(address & 3)); - } else { - return CPUReadByteQuick(reg[15].I+(address & 1)); - } - } - break; - } -} - -inline void CPUWriteMemory(u32 address, u32 value) -{ -#ifdef DEV_VERSION - if(address & 3) { - if(systemVerbose & VERBOSE_UNALIGNED_MEMORY) { - log("Unaliagned word write: %08x to %08x from %08x\n", - value, - address, - armMode ? armNextPC - 4 : armNextPC - 2); - } - } -#endif - - switch(address >> 24) { - case 0x02: -#ifdef SDL - if(*((u32 *)&freezeWorkRAM[address & 0x3FFFC])) - cheatsWriteMemory(address & 0x203FFFC, - value); - else -#endif - WRITE32LE(((u32 *)&workRAM[address & 0x3FFFC]), value); - break; - case 0x03: -#ifdef SDL - if(*((u32 *)&freezeInternalRAM[address & 0x7ffc])) - cheatsWriteMemory(address & 0x3007FFC, - value); - else -#endif - WRITE32LE(((u32 *)&internalRAM[address & 0x7ffC]), value); - break; - case 0x04: - if(address < 0x4000400) { - CPUUpdateRegister((address & 0x3FC), value & 0xFFFF); - CPUUpdateRegister((address & 0x3FC) + 2, (value >> 16)); - } else goto unwritable; - break; - case 0x05: - WRITE32LE(((u32 *)&paletteRAM[address & 0x3FC]), value); - break; - case 0x06: - if(address & 0x10000) - WRITE32LE(((u32 *)&vram[address & 0x17ffc]), value); - else - WRITE32LE(((u32 *)&vram[address & 0x1fffc]), value); - break; - case 0x07: - WRITE32LE(((u32 *)&oam[address & 0x3fc]), value); - break; - case 0x0D: - if(cpuEEPROMEnabled) { - eepromWrite(address, value); - break; - } - goto unwritable; - case 0x0E: - if(!eepromInUse | cpuSramEnabled | cpuFlashEnabled) { - (*cpuSaveGameFunc)(address, (u8)value); - break; - } - // default - default: - unwritable: -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_ILLEGAL_WRITE) { - log("Illegal word write: %08x to %08x from %08x\n", - value, - address, - armMode ? armNextPC - 4 : armNextPC - 2); - } -#endif - break; - } -} - -#endif //VBA_GBAinline_H \ No newline at end of file +// -*- C++ -*- +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2005 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#ifndef VBA_GBAinline_H +#define VBA_GBAinline_H + +#include "System.h" +#include "Port.h" +#include "RTC.h" + +extern bool cpuSramEnabled; +extern bool cpuFlashEnabled; +extern bool cpuEEPROMEnabled; +extern bool cpuEEPROMSensorEnabled; +extern bool cpuDmaHack; +extern u32 cpuDmaLast; +extern bool timer0On; +extern int timer0Ticks; +extern int timer0ClockReload; +extern bool timer1On; +extern int timer1Ticks; +extern int timer1ClockReload; +extern bool timer2On; +extern int timer2Ticks; +extern int timer2ClockReload; +extern bool timer3On; +extern int timer3Ticks; +extern int timer3ClockReload; +extern int cpuTotalTicks; + +#define CPUReadByteQuick(addr) \ + map[(addr)>>24].address[(addr) & map[(addr)>>24].mask] + +#define CPUReadHalfWordQuick(addr) \ + READ16LE(((u16*)&map[(addr)>>24].address[(addr) & map[(addr)>>24].mask])) + +#define CPUReadMemoryQuick(addr) \ + READ32LE(((u32*)&map[(addr)>>24].address[(addr) & map[(addr)>>24].mask])) + +static inline u32 CPUReadMemory(u32 address) +{ + +#ifdef DEV_VERSION + if(address & 3) { + if(systemVerbose & VERBOSE_UNALIGNED_MEMORY) { + log("Unaligned word read: %08x at %08x\n", address, armMode ? + armNextPC - 4 : armNextPC - 2); + } + } +#endif + + u32 value; + switch(address >> 24) { + case 0: + if(reg[15].I >> 24) { + if(address < 0x4000) { +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_ILLEGAL_READ) { + log("Illegal word read: %08x at %08x\n", address, armMode ? + armNextPC - 4 : armNextPC - 2); + } +#endif + + value = READ32LE(((u32 *)&biosProtected)); + } + else goto unreadable; + } else + value = READ32LE(((u32 *)&bios[address & 0x3FFC])); + break; + case 2: + value = READ32LE(((u32 *)&workRAM[address & 0x3FFFC])); + break; + case 3: + value = READ32LE(((u32 *)&internalRAM[address & 0x7ffC])); + break; + case 4: + if((address < 0x4000400) && ioReadable[address & 0x3fc]) { + if(ioReadable[(address & 0x3fc) + 2]) + value = READ32LE(((u32 *)&ioMem[address & 0x3fC])); + else + value = READ16LE(((u16 *)&ioMem[address & 0x3fc])); + } else goto unreadable; + break; + case 5: + value = READ32LE(((u32 *)&paletteRAM[address & 0x3fC])); + break; + case 6: + address = (address & 0x1fffc); + if (((DISPCNT & 7) >2) && ((address & 0x1C000) == 0x18000)) + { + value = 0; + break; + } + if ((address & 0x18000) == 0x18000) + address &= 0x17fff; + value = READ32LE(((u32 *)&vram[address])); + break; + case 7: + value = READ32LE(((u32 *)&oam[address & 0x3FC])); + break; + case 8: + case 9: + case 10: + case 11: + case 12: + value = READ32LE(((u32 *)&rom[address&0x1FFFFFC])); + break; + case 13: + if(cpuEEPROMEnabled) + // no need to swap this + return eepromRead(address); + goto unreadable; + case 14: + if(cpuFlashEnabled | cpuSramEnabled) + // no need to swap this + return flashRead(address); + // default + default: + unreadable: +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_ILLEGAL_READ) { + log("Illegal word read: %08x at %08x\n", address, armMode ? + armNextPC - 4 : armNextPC - 2); + } +#endif + + if(cpuDmaHack) { + value = cpuDmaLast; + } else { + if(armState) { + value = CPUReadMemoryQuick(reg[15].I); + } else { + value = CPUReadHalfWordQuick(reg[15].I) | + CPUReadHalfWordQuick(reg[15].I) << 16; + } + } + } + + if(address & 3) { +#ifdef C_CORE + int shift = (address & 3) << 3; + value = (value >> shift) | (value << (32 - shift)); +#else +#ifdef __GNUC__ + asm("and $3, %%ecx;" + "shl $3 ,%%ecx;" + "ror %%cl, %0" + : "=r" (value) + : "r" (value), "c" (address)); +#else + __asm { + mov ecx, address; + and ecx, 3; + shl ecx, 3; + ror [dword ptr value], cl; + } +#endif +#endif + } + return value; +} + +extern u32 myROM[]; + +static inline u32 CPUReadHalfWord(u32 address) +{ +#ifdef DEV_VERSION + if(address & 1) { + if(systemVerbose & VERBOSE_UNALIGNED_MEMORY) { + log("Unaligned halfword read: %08x at %08x\n", address, armMode ? + armNextPC - 4 : armNextPC - 2); + } + } +#endif + + u32 value; + + switch(address >> 24) { + case 0: + if (reg[15].I >> 24) { + if(address < 0x4000) { +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_ILLEGAL_READ) { + log("Illegal halfword read: %08x at %08x\n", address, armMode ? + armNextPC - 4 : armNextPC - 2); + } +#endif + value = READ16LE(((u16 *)&biosProtected[address&2])); + } else goto unreadable; + } else + value = READ16LE(((u16 *)&bios[address & 0x3FFE])); + break; + case 2: + value = READ16LE(((u16 *)&workRAM[address & 0x3FFFE])); + break; + case 3: + value = READ16LE(((u16 *)&internalRAM[address & 0x7ffe])); + break; + case 4: + if((address < 0x4000400) && ioReadable[address & 0x3fe]) + { + value = READ16LE(((u16 *)&ioMem[address & 0x3fe])); + if (((address & 0x3fe)>0xFF) && ((address & 0x3fe)<0x10E)) + { + if (((address & 0x3fe) == 0x100) && timer0On) + value = 0xFFFF - ((timer0Ticks-cpuTotalTicks) >> timer0ClockReload); + else + if (((address & 0x3fe) == 0x104) && timer1On && !(TM1CNT & 4)) + value = 0xFFFF - ((timer1Ticks-cpuTotalTicks) >> timer1ClockReload); + else + if (((address & 0x3fe) == 0x108) && timer2On && !(TM2CNT & 4)) + value = 0xFFFF - ((timer2Ticks-cpuTotalTicks) >> timer2ClockReload); + else + if (((address & 0x3fe) == 0x10C) && timer3On && !(TM3CNT & 4)) + value = 0xFFFF - ((timer3Ticks-cpuTotalTicks) >> timer3ClockReload); + } + } + else goto unreadable; + break; + case 5: + value = READ16LE(((u16 *)&paletteRAM[address & 0x3fe])); + break; + case 6: + address = (address & 0x1fffe); + if (((DISPCNT & 7) >2) && ((address & 0x1C000) == 0x18000)) + { + value = 0; + break; + } + if ((address & 0x18000) == 0x18000) + address &= 0x17fff; + value = READ16LE(((u16 *)&vram[address])); + break; + case 7: + value = READ16LE(((u16 *)&oam[address & 0x3fe])); + break; + case 8: + case 9: + case 10: + case 11: + case 12: + if(address == 0x80000c4 || address == 0x80000c6 || address == 0x80000c8) + value = rtcRead(address); + else + value = READ16LE(((u16 *)&rom[address & 0x1FFFFFE])); + break; + case 13: + if(cpuEEPROMEnabled) + // no need to swap this + return eepromRead(address); + goto unreadable; + case 14: + if(cpuFlashEnabled | cpuSramEnabled) + // no need to swap this + return flashRead(address); + // default + default: + unreadable: +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_ILLEGAL_READ) { + log("Illegal halfword read: %08x at %08x\n", address, armMode ? + armNextPC - 4 : armNextPC - 2); + } +#endif + if(cpuDmaHack) { + value = cpuDmaLast & 0xFFFF; + } else { + if(armState) { + value = CPUReadHalfWordQuick(reg[15].I + (address & 2)); + } else { + value = CPUReadHalfWordQuick(reg[15].I); + } + } + break; + } + + if(address & 1) { + value = (value >> 8) | (value << 24); + } + + return value; +} + +static inline u16 CPUReadHalfWordSigned(u32 address) +{ + u16 value = CPUReadHalfWord(address); + if((address & 1)) + value = (s8)value; + return value; +} + +static inline u8 CPUReadByte(u32 address) +{ + switch(address >> 24) { + case 0: + if (reg[15].I >> 24) { + if(address < 0x4000) { +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_ILLEGAL_READ) { + log("Illegal byte read: %08x at %08x\n", address, armMode ? + armNextPC - 4 : armNextPC - 2); + } +#endif + return biosProtected[address & 3]; + } else goto unreadable; + } + return bios[address & 0x3FFF]; + case 2: + return workRAM[address & 0x3FFFF]; + case 3: + return internalRAM[address & 0x7fff]; + case 4: + if((address < 0x4000400) && ioReadable[address & 0x3ff]) + return ioMem[address & 0x3ff]; + else goto unreadable; + case 5: + return paletteRAM[address & 0x3ff]; + case 6: + address = (address & 0x1ffff); + if (((DISPCNT & 7) >2) && ((address & 0x1C000) == 0x18000)) + return 0; + if ((address & 0x18000) == 0x18000) + address &= 0x17fff; + return vram[address]; + case 7: + return oam[address & 0x3ff]; + case 8: + case 9: + case 10: + case 11: + case 12: + return rom[address & 0x1FFFFFF]; + case 13: + if(cpuEEPROMEnabled) + return eepromRead(address); + goto unreadable; + case 14: + if(cpuSramEnabled | cpuFlashEnabled) + return flashRead(address); + if(cpuEEPROMSensorEnabled) { + switch(address & 0x00008f00) { + case 0x8200: + return systemGetSensorX() & 255; + case 0x8300: + return (systemGetSensorX() >> 8)|0x80; + case 0x8400: + return systemGetSensorY() & 255; + case 0x8500: + return systemGetSensorY() >> 8; + } + } + // default + default: + unreadable: +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_ILLEGAL_READ) { + log("Illegal byte read: %08x at %08x\n", address, armMode ? + armNextPC - 4 : armNextPC - 2); + } +#endif + if(cpuDmaHack) { + return cpuDmaLast & 0xFF; + } else { + if(armState) { + return CPUReadByteQuick(reg[15].I+(address & 3)); + } else { + return CPUReadByteQuick(reg[15].I+(address & 1)); + } + } + break; + } +} + +static inline void CPUWriteMemory(u32 address, u32 value) +{ + +#ifdef DEV_VERSION + if(address & 3) { + if(systemVerbose & VERBOSE_UNALIGNED_MEMORY) { + log("Unaligned word write: %08x to %08x from %08x\n", + value, + address, + armMode ? armNextPC - 4 : armNextPC - 2); + } + } +#endif + + switch(address >> 24) { + case 0x02: +#ifdef BKPT_SUPPORT + if(*((u32 *)&freezeWorkRAM[address & 0x3FFFC])) + cheatsWriteMemory(address & 0x203FFFC, + value); + else +#endif + WRITE32LE(((u32 *)&workRAM[address & 0x3FFFC]), value); + break; + case 0x03: +#ifdef BKPT_SUPPORT + if(*((u32 *)&freezeInternalRAM[address & 0x7ffc])) + cheatsWriteMemory(address & 0x3007FFC, + value); + else +#endif + WRITE32LE(((u32 *)&internalRAM[address & 0x7ffC]), value); + break; + case 0x04: + if(address < 0x4000400) { + CPUUpdateRegister((address & 0x3FC), value & 0xFFFF); + CPUUpdateRegister((address & 0x3FC) + 2, (value >> 16)); + } else goto unwritable; + break; + case 0x05: +#ifdef BKPT_SUPPORT + if(*((u32 *)&freezePRAM[address & 0x3fc])) + cheatsWriteMemory(address & 0x70003FC, + value); + else +#endif + WRITE32LE(((u32 *)&paletteRAM[address & 0x3FC]), value); + break; + case 0x06: + address = (address & 0x1fffc); + if (((DISPCNT & 7) >2) && ((address & 0x1C000) == 0x18000)) + return; + if ((address & 0x18000) == 0x18000) + address &= 0x17fff; + +#ifdef BKPT_SUPPORT + if(*((u32 *)&freezeVRAM[address])) + cheatsWriteMemory(address + 0x06000000, value); + else +#endif + + WRITE32LE(((u32 *)&vram[address]), value); + break; + case 0x07: +#ifdef BKPT_SUPPORT + if(*((u32 *)&freezeOAM[address & 0x3fc])) + cheatsWriteMemory(address & 0x70003FC, + value); + else +#endif + WRITE32LE(((u32 *)&oam[address & 0x3fc]), value); + break; + case 0x0D: + if(cpuEEPROMEnabled) { + eepromWrite(address, value); + break; + } + goto unwritable; + case 0x0E: + if(!eepromInUse | cpuSramEnabled | cpuFlashEnabled) { + (*cpuSaveGameFunc)(address, (u8)value); + break; + } + // default + default: + unwritable: +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_ILLEGAL_WRITE) { + log("Illegal word write: %08x to %08x from %08x\n", + value, + address, + armMode ? armNextPC - 4 : armNextPC - 2); + } +#endif + break; + } +} + +#endif //VBA_GBAinline_H diff --git a/src/Gfx.cpp b/src/Gfx.cpp index 26c15691..7dc47ef4 100644 --- a/src/Gfx.cpp +++ b/src/Gfx.cpp @@ -22,7 +22,6 @@ int coeff[32] = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16}; - u32 line0[240]; u32 line1[240]; u32 line2[240]; @@ -32,6 +31,7 @@ u32 lineOBJWin[240]; u32 lineMix[240]; bool gfxInWin0[240]; bool gfxInWin1[240]; +int lineOBJpixleft[128]; int gfxBG2Changed = 0; int gfxBG3Changed = 0; diff --git a/src/Gfx.h b/src/Gfx.h index 40abe485..2e9625bc 100644 --- a/src/Gfx.h +++ b/src/Gfx.h @@ -1,1581 +1,1654 @@ -// -*- C++ -*- -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#ifndef VBA_GFX_H -#define VBA_GFX_H - -#include "GBA.h" -#include "Gfx.h" -#include "Globals.h" - -#include "Port.h" - -//#define SPRITE_DEBUG - -void gfxDrawTextScreen(u16, u16, u16, u32 *); -void gfxDrawRotScreen(u16, - u16, u16, - u16, u16, - u16, u16, - u16, u16, - int&, int&, - int, - u32*); -void gfxDrawRotScreen16Bit(u16, - u16, u16, - u16, u16, - u16, u16, - u16, u16, - int&, int&, - int, - u32*); -void gfxDrawRotScreen256(u16, - u16, u16, - u16, u16, - u16, u16, - u16, u16, - int&, int&, - int, - u32*); -void gfxDrawRotScreen16Bit160(u16, - u16, u16, - u16, u16, - u16, u16, - u16, u16, - int&, int&, - int, - u32*); -void gfxDrawSprites(u32 *); -void gfxIncreaseBrightness(u32 *line, int coeff); -void gfxDecreaseBrightness(u32 *line, int coeff); -void gfxAlphaBlend(u32 *ta, u32 *tb, int ca, int cb); - -void mode0RenderLine(); -void mode0RenderLineNoWindow(); -void mode0RenderLineAll(); - -void mode1RenderLine(); -void mode1RenderLineNoWindow(); -void mode1RenderLineAll(); - -void mode2RenderLine(); -void mode2RenderLineNoWindow(); -void mode2RenderLineAll(); - -void mode3RenderLine(); -void mode3RenderLineNoWindow(); -void mode3RenderLineAll(); - -void mode4RenderLine(); -void mode4RenderLineNoWindow(); -void mode4RenderLineAll(); - -void mode5RenderLine(); -void mode5RenderLineNoWindow(); -void mode5RenderLineAll(); - -extern int coeff[32]; -extern u32 line0[240]; -extern u32 line1[240]; -extern u32 line2[240]; -extern u32 line3[240]; -extern u32 lineOBJ[240]; -extern u32 lineOBJWin[240]; -extern u32 lineMix[240]; -extern bool gfxInWin0[240]; -extern bool gfxInWin1[240]; - -extern int gfxBG2Changed; -extern int gfxBG3Changed; - -extern int gfxBG2X; -extern int gfxBG2Y; -extern int gfxBG2LastX; -extern int gfxBG2LastY; -extern int gfxBG3X; -extern int gfxBG3Y; -extern int gfxBG3LastX; -extern int gfxBG3LastY; -extern int gfxLastVCOUNT; - -inline void gfxClearArray(u32 *array) -{ - for(int i = 0; i < 240; i++) { - *array++ = 0x80000000; - } -} - -inline void gfxDrawTextScreen(u16 control, u16 hofs, u16 vofs, - u32 *line) -{ - u16 *palette = (u16 *)paletteRAM; - u8 *charBase = &vram[((control >> 2) & 0x03) * 0x4000]; - u16 *screenBase = (u16 *)&vram[((control >> 8) & 0x1f) * 0x800]; - u32 prio = ((control & 3)<<25) + 0x1000000; - int sizeX = 256; - int sizeY = 256; - switch((control >> 14) & 3) { - case 0: - break; - case 1: - sizeX = 512; - break; - case 2: - sizeY = 512; - break; - case 3: - sizeX = 512; - sizeY = 512; - break; - } - - int maskX = sizeX-1; - int maskY = sizeY-1; - - bool mosaicOn = (control & 0x40) ? true : false; - - int xxx = hofs & maskX; - int yyy = (vofs + VCOUNT) & maskY; - int mosaicX = (MOSAIC & 0x000F)+1; - int mosaicY = ((MOSAIC & 0x00F0)>>4)+1; - - if(mosaicOn) { - if((VCOUNT % mosaicY) != 0) { - mosaicY = (VCOUNT / mosaicY) * mosaicY; - yyy = (vofs + mosaicY) & maskY; - } - } - - if(yyy > 255 && sizeY > 256) { - yyy &= 255; - screenBase += 0x400; - if(sizeX > 256) - screenBase += 0x400; - } - - int yshift = ((yyy>>3)<<5); - if((control) & 0x80) { - u16 *screenSource = screenBase + 0x400 * (xxx>>8) + ((xxx & 255)>>3) + yshift; - for(int x = 0; x < 240; x++) { - u16 data = READ16LE(screenSource); - - int tile = data & 0x3FF; - int tileX = (xxx & 7); - int tileY = yyy & 7; - - if(data & 0x0400) - tileX = 7 - tileX; - if(data & 0x0800) - tileY = 7 - tileY; - - u8 color = charBase[tile * 64 + tileY * 8 + tileX]; - - line[x] = color ? (READ16LE(&palette[color]) | prio): 0x80000000; - - if(data & 0x0400) { - if(tileX == 0) - screenSource++; - } else if(tileX == 7) - screenSource++; - xxx++; - if(xxx == 256) { - if(sizeX > 256) - screenSource = screenBase + 0x400 + yshift; - else { - screenSource = screenBase + yshift; - xxx = 0; - } - } else if(xxx >= sizeX) { - xxx = 0; - screenSource = screenBase + yshift; - } - } - } else { - u16 *screenSource = screenBase + 0x400*(xxx>>8)+((xxx&255)>>3) + - yshift; - for(int x = 0; x < 240; x++) { - u16 data = READ16LE(screenSource); - - int tile = data & 0x3FF; - int tileX = (xxx & 7); - int tileY = yyy & 7; - - if(data & 0x0400) - tileX = 7 - tileX; - if(data & 0x0800) - tileY = 7 - tileY; - - u8 color = charBase[(tile<<5) + (tileY<<2) + (tileX>>1)]; - - if(tileX & 1) { - color = (color >> 4); - } else { - color &= 0x0F; - } - - int pal = (READ16LE(screenSource)>>8) & 0xF0; - line[x] = color ? (READ16LE(&palette[pal + color])|prio): 0x80000000; - - if(data & 0x0400) { - if(tileX == 0) - screenSource++; - } else if(tileX == 7) - screenSource++; - xxx++; - if(xxx == 256) { - if(sizeX > 256) - screenSource = screenBase + 0x400 + yshift; - else { - screenSource = screenBase + yshift; - xxx = 0; - } - } else if(xxx >= sizeX) { - xxx = 0; - screenSource = screenBase + yshift; - } - } - } - if(mosaicOn) { - if(mosaicX > 1) { - int m = 1; - for(int i = 0; i < 239; i++) { - line[i+1] = line[i]; - m++; - if(m == mosaicX) { - m = 1; - i++; - } - } - } - } -} - -inline void gfxDrawRotScreen(u16 control, - u16 x_l, u16 x_h, - u16 y_l, u16 y_h, - u16 pa, u16 pb, - u16 pc, u16 pd, - int& currentX, int& currentY, - int changed, - u32 *line) -{ - u16 *palette = (u16 *)paletteRAM; - u8 *charBase = &vram[((control >> 2) & 0x03) * 0x4000]; - u8 *screenBase = (u8 *)&vram[((control >> 8) & 0x1f) * 0x800]; - int prio = ((control & 3) << 25) + 0x1000000; - - int sizeX = 128; - int sizeY = 128; - switch((control >> 14) & 3) { - case 0: - break; - case 1: - sizeX = sizeY = 256; - break; - case 2: - sizeX = sizeY = 512; - break; - case 3: - sizeX = sizeY = 1024; - break; - } - - int dx = pa & 0x7FFF; - if(pa & 0x8000) - dx |= 0xFFFF8000; - int dmx = pb & 0x7FFF; - if(pb & 0x8000) - dmx |= 0xFFFF8000; - int dy = pc & 0x7FFF; - if(pc & 0x8000) - dy |= 0xFFFF8000; - int dmy = pd & 0x7FFFF; - if(pd & 0x8000) - dmy |= 0xFFFF8000; - - if(VCOUNT == 0) - changed = 3; - - if(changed & 1) { - currentX = (x_l) | ((x_h & 0x07FF)<<16); - if(x_h & 0x0800) - currentX |= 0xF8000000; - } else { - currentX += dmx; - } - - if(changed & 2) { - currentY = (y_l) | ((y_h & 0x07FF)<<16); - if(y_h & 0x0800) - currentY |= 0xF8000000; - } else { - currentY += dmy; - } - - int realX = currentX; - int realY = currentY; - - if(control & 0x40) { - int mosaicY = ((MOSAIC & 0xF0)>>4) + 1; - int y = (VCOUNT % mosaicY); - realX -= y*dmx; - realY -= y*dmy; - } - - int xxx = (realX >> 8); - int yyy = (realY >> 8); - - if(control & 0x2000) { - xxx %= sizeX; - yyy %= sizeY; - if(xxx < 0) - xxx += sizeX; - if(yyy < 0) - yyy += sizeY; - } - - if(control & 0x80) { - for(int x = 0; x < 240; x++) { - if(xxx < 0 || - yyy < 0 || - xxx >= sizeX || - yyy >= sizeY) { - line[x] = 0x80000000; - } else { - int tile = screenBase[(xxx>>3) + (yyy>>3)*(sizeX>>3)]; - - int tileX = (xxx & 7); - int tileY = yyy & 7; - - u8 color = charBase[(tile<<6) + (tileY<<3) + tileX]; - - line[x] = color ? (READ16LE(&palette[color])|prio): 0x80000000; - } - realX += dx; - realY += dy; - - xxx = (realX >> 8); - yyy = (realY >> 8); - - if(control & 0x2000) { - xxx %= sizeX; - yyy %= sizeY; - if(xxx < 0) - xxx += sizeX; - if(yyy < 0) - yyy += sizeY; - } - } - } else { - for(int x = 0; x < 240; x++) { - if(xxx < 0 || - yyy < 0 || - xxx >= sizeX || - yyy >= sizeY) { - line[x] = 0x80000000; - } else { - int tile = screenBase[(xxx>>3) + (yyy>>3)*(sizeX>>3)]; - - int tileX = (xxx & 7); - int tileY = yyy & 7; - - u8 color = charBase[(tile<<6) + (tileY<<3) + tileX]; - - line[x] = color ? (READ16LE(&palette[color])|prio): 0x80000000; - } - realX += dx; - realY += dy; - - xxx = (realX >> 8); - yyy = (realY >> 8); - - if(control & 0x2000) { - xxx %= sizeX; - yyy %= sizeY; - if(xxx < 0) - xxx += sizeX; - if(yyy < 0) - yyy += sizeY; - } - } - } - - if(control & 0x40) { - int mosaicX = (MOSAIC & 0xF) + 1; - if(mosaicX > 1) { - int m = 1; - for(int i = 0; i < 239; i++) { - line[i+1] = line[i]; - m++; - if(m == mosaicX) { - m = 1; - i++; - } - } - } - } -} - -inline void gfxDrawRotScreen16Bit(u16 control, - u16 x_l, u16 x_h, - u16 y_l, u16 y_h, - u16 pa, u16 pb, - u16 pc, u16 pd, - int& currentX, int& currentY, - int changed, - u32 *line) -{ - u16 *screenBase = (u16 *)&vram[0]; - int prio = ((control & 3) << 25) + 0x1000000; - int sizeX = 240; - int sizeY = 160; - - int startX = (x_l) | ((x_h & 0x07FF)<<16); - if(x_h & 0x0800) - startX |= 0xF8000000; - int startY = (y_l) | ((y_h & 0x07FF)<<16); - if(y_h & 0x0800) - startY |= 0xF8000000; - - int dx = pa & 0x7FFF; - if(pa & 0x8000) - dx |= 0xFFFF8000; - int dmx = pb & 0x7FFF; - if(pb & 0x8000) - dmx |= 0xFFFF8000; - int dy = pc & 0x7FFF; - if(pc & 0x8000) - dy |= 0xFFFF8000; - int dmy = pd & 0x7FFFF; - if(pd & 0x8000) - dmy |= 0xFFFF8000; - - if(VCOUNT == 0) - changed = 3; - - if(changed & 1) { - currentX = (x_l) | ((x_h & 0x07FF)<<16); - if(x_h & 0x0800) - currentX |= 0xF8000000; - } else - currentX += dmx; - - if(changed & 2) { - currentY = (y_l) | ((y_h & 0x07FF)<<16); - if(y_h & 0x0800) - currentY |= 0xF8000000; - } else { - currentY += dmy; - } - - int realX = currentX; - int realY = currentY; - - if(control & 0x40) { - int mosaicY = ((MOSAIC & 0xF0)>>4) + 1; - int y = (VCOUNT % mosaicY); - realX -= y*dmx; - realY -= y*dmy; - } - - int xxx = (realX >> 8); - int yyy = (realY >> 8); - - for(int x = 0; x < 240; x++) { - if(xxx < 0 || - yyy < 0 || - xxx >= sizeX || - yyy >= sizeY) { - line[x] = 0x80000000; - } else { - line[x] = (READ16LE(&screenBase[yyy * sizeX + xxx]) | prio); - } - realX += dx; - realY += dy; - - xxx = (realX >> 8); - yyy = (realY >> 8); - } - - if(control & 0x40) { - int mosaicX = (MOSAIC & 0xF) + 1; - if(mosaicX > 1) { - int m = 1; - for(int i = 0; i < 239; i++) { - line[i+1] = line[i]; - m++; - if(m == mosaicX) { - m = 1; - i++; - } - } - } - } -} - -inline void gfxDrawRotScreen256(u16 control, - u16 x_l, u16 x_h, - u16 y_l, u16 y_h, - u16 pa, u16 pb, - u16 pc, u16 pd, - int ¤tX, int& currentY, - int changed, - u32 *line) -{ - u16 *palette = (u16 *)paletteRAM; - u8 *screenBase = (DISPCNT & 0x0010) ? &vram[0xA000] : &vram[0x0000]; - int prio = ((control & 3) << 25) + 0x1000000; - int sizeX = 240; - int sizeY = 160; - - int startX = (x_l) | ((x_h & 0x07FF)<<16); - if(x_h & 0x0800) - startX |= 0xF8000000; - int startY = (y_l) | ((y_h & 0x07FF)<<16); - if(y_h & 0x0800) - startY |= 0xF8000000; - - int dx = pa & 0x7FFF; - if(pa & 0x8000) - dx |= 0xFFFF8000; - int dmx = pb & 0x7FFF; - if(pb & 0x8000) - dmx |= 0xFFFF8000; - int dy = pc & 0x7FFF; - if(pc & 0x8000) - dy |= 0xFFFF8000; - int dmy = pd & 0x7FFFF; - if(pd & 0x8000) - dmy |= 0xFFFF8000; - - if(VCOUNT == 0) - changed = 3; - - if(changed & 1) { - currentX = (x_l) | ((x_h & 0x07FF)<<16); - if(x_h & 0x0800) - currentX |= 0xF8000000; - } else { - currentX += dmx; - } - - if(changed & 2) { - currentY = (y_l) | ((y_h & 0x07FF)<<16); - if(y_h & 0x0800) - currentY |= 0xF8000000; - } else { - currentY += dmy; - } - - int realX = currentX; - int realY = currentY; - - if(control & 0x40) { - int mosaicY = ((MOSAIC & 0xF0)>>4) + 1; - int y = (VCOUNT / mosaicY) * mosaicY; - realX = startX + y*dmx; - realY = startY + y*dmy; - } - - int xxx = (realX >> 8); - int yyy = (realY >> 8); - - for(int x = 0; x < 240; x++) { - if(xxx < 0 || - yyy < 0 || - xxx >= sizeX || - yyy >= sizeY) { - line[x] = 0x80000000; - } else { - u8 color = screenBase[yyy * 240 + xxx]; - - line[x] = color ? (READ16LE(&palette[color])|prio): 0x80000000; - } - realX += dx; - realY += dy; - - xxx = (realX >> 8); - yyy = (realY >> 8); - } - - if(control & 0x40) { - int mosaicX = (MOSAIC & 0xF) + 1; - if(mosaicX > 1) { - int m = 1; - for(int i = 0; i < 239; i++) { - line[i+1] = line[i]; - m++; - if(m == mosaicX) { - m = 1; - i++; - } - } - } - } -} - -inline void gfxDrawRotScreen16Bit160(u16 control, - u16 x_l, u16 x_h, - u16 y_l, u16 y_h, - u16 pa, u16 pb, - u16 pc, u16 pd, - int& currentX, int& currentY, - int changed, - u32 *line) -{ - u16 *screenBase = (DISPCNT & 0x0010) ? (u16 *)&vram[0xa000] : - (u16 *)&vram[0]; - int prio = ((control & 3) << 25) + 0x1000000; - int sizeX = 160; - int sizeY = 128; - - int startX = (x_l) | ((x_h & 0x07FF)<<16); - if(x_h & 0x0800) - startX |= 0xF8000000; - int startY = (y_l) | ((y_h & 0x07FF)<<16); - if(y_h & 0x0800) - startY |= 0xF8000000; - - int dx = pa & 0x7FFF; - if(pa & 0x8000) - dx |= 0xFFFF8000; - int dmx = pb & 0x7FFF; - if(pb & 0x8000) - dmx |= 0xFFFF8000; - int dy = pc & 0x7FFF; - if(pc & 0x8000) - dy |= 0xFFFF8000; - int dmy = pd & 0x7FFFF; - if(pd & 0x8000) - dmy |= 0xFFFF8000; - - if(VCOUNT == 0) - changed = 3; - - if(changed & 1) { - currentX = (x_l) | ((x_h & 0x07FF)<<16); - if(x_h & 0x0800) - currentX |= 0xF8000000; - } else { - currentX += dmx; - } - - if(changed & 2) { - currentY = (y_l) | ((y_h & 0x07FF)<<16); - if(y_h & 0x0800) - currentY |= 0xF8000000; - } else { - currentY += dmy; - } - - int realX = currentX; - int realY = currentY; - - if(control & 0x40) { - int mosaicY = ((MOSAIC & 0xF0)>>4) + 1; - int y = (VCOUNT / mosaicY) * mosaicY; - realX = startX + y*dmx; - realY = startY + y*dmy; - } - - int xxx = (realX >> 8); - int yyy = (realY >> 8); - - for(int x = 0; x < 240; x++) { - if(xxx < 0 || - yyy < 0 || - xxx >= sizeX || - yyy >= sizeY) { - line[x] = 0x80000000; - } else { - line[x] = (READ16LE(&screenBase[yyy * sizeX + xxx]) | prio); - } - realX += dx; - realY += dy; - - xxx = (realX >> 8); - yyy = (realY >> 8); - } - - if(control & 0x40) { - int mosaicX = (MOSAIC & 0xF) + 1; - if(mosaicX > 1) { - int m = 1; - for(int i = 0; i < 239; i++) { - line[i+1] = line[i]; - m++; - if(m == mosaicX) { - m = 1; - i++; - } - } - } - } -} - -inline void gfxDrawSprites(u32 *lineOBJ) -{ - int m=0; - gfxClearArray(lineOBJ); - if(layerEnable & 0x1000) { - u16 *sprites = (u16 *)oam; - u16 *spritePalette = &((u16 *)paletteRAM)[256]; - int mosaicY = ((MOSAIC & 0xF000)>>12) + 1; - int mosaicX = ((MOSAIC & 0xF00)>>8) + 1; - for(int x = 0; x < 128 ; x++) { - u16 a0 = READ16LE(sprites++); - u16 a1 = READ16LE(sprites++); - u16 a2 = READ16LE(sprites++); - sprites++; - - // ignore OBJ-WIN - if((a0 & 0x0c00) == 0x0800) - continue; - - int sizeY = 8; - int sizeX = 8; - - switch(((a0 >>12) & 0x0c)|(a1>>14)) { - case 0: - break; - case 1: - sizeX = sizeY = 16; - break; - case 2: - sizeX = sizeY = 32; - break; - case 3: - sizeX = sizeY = 64; - break; - case 4: - sizeX = 16; - break; - case 5: - sizeX = 32; - break; - case 6: - sizeX = 32; - sizeY = 16; - break; - case 7: - sizeX = 64; - sizeY = 32; - break; - case 8: - sizeY = 16; - break; - case 9: - sizeY = 32; - break; - case 10: - sizeX = 16; - sizeY = 32; - break; - case 11: - sizeX = 32; - sizeY = 64; - break; - default: - continue; - } - -#ifdef SPRITE_DEBUG - int maskX = sizeX-1; - int maskY = sizeY-1; -#endif - - int sy = (a0 & 255); - - if(sy > 160) - sy -= 256; - - if(a0 & 0x0100) { - int fieldX = sizeX; - int fieldY = sizeY; - if(a0 & 0x0200) { - fieldX <<= 1; - fieldY <<= 1; - } - - int t = VCOUNT - sy; - if((t >= 0) && (t < fieldY)) { - int sx = (a1 & 0x1FF); - if((sx < 240) || (((sx + fieldX) & 511) < 240)) { - // int t2 = t - (fieldY >> 1); - int rot = (a1 >> 9) & 0x1F; - u16 *OAM = (u16 *)oam; - int dx = READ16LE(&OAM[3 + (rot << 4)]); - if(dx & 0x8000) - dx |= 0xFFFF8000; - int dmx = READ16LE(&OAM[7 + (rot << 4)]); - if(dmx & 0x8000) - dmx |= 0xFFFF8000; - int dy = READ16LE(&OAM[11 + (rot << 4)]); - if(dy & 0x8000) - dy |= 0xFFFF8000; - int dmy = READ16LE(&OAM[15 + (rot << 4)]); - if(dmy & 0x8000) - dmy |= 0xFFFF8000; - - if(a0 & 0x1000) { - t -= (t % mosaicY); - } - - int realX = ((sizeX) << 7) - (fieldX >> 1)*dx - (fieldY>>1)*dmx - + t * dmx; - int realY = ((sizeY) << 7) - (fieldX >> 1)*dy - (fieldY>>1)*dmy - + t * dmy; - - u32 prio = (((a2 >> 10) & 3) << 25) | ((a0 & 0x0c00)<<6); - - if(a0 & 0x2000) { - int c = (a2 & 0x3FF); - if((DISPCNT & 7) > 2 && (c < 512)) - continue; - int inc = 32; - if(DISPCNT & 0x40) - inc = sizeX >> 2; - else - c &= 0x3FE; - for(int x = 0; x < fieldX; x++) { - int xxx = realX >> 8; - int yyy = realY >> 8; - - if(xxx < 0 || xxx >= sizeX || - yyy < 0 || yyy >= sizeY || - sx >= 240); - else { - u32 color = vram[0x10000 + ((((c + (yyy>>3) * inc)<<5) - + ((yyy & 7)<<3) + ((xxx >> 3)<<6) + - (xxx & 7))&0x7FFF)]; - if ((color==0) && (((prio >> 25)&3) < - ((lineOBJ[sx]>>25)&3))) { - lineOBJ[sx] = (lineOBJ[sx] & 0xF9FFFFFF) | prio; - if((a0 & 0x1000) && m) - lineOBJ[sx]=(lineOBJ[sx-1] & 0xF9FFFFFF) | prio; - } else if((color) && (prio < (lineOBJ[sx]&0xFF000000))) { - lineOBJ[sx] = READ16LE(&spritePalette[color]) | prio; - if((a0 & 0x1000) && m) - lineOBJ[sx]=(lineOBJ[sx-1] & 0xF9FFFFFF) | prio; - } - - if (a0 & 0x1000) { - m++; - if (m==mosaicX) - m=0; - } -#ifdef SPRITE_DEBUG - if(t == 0 || t == maskY || x == 0 || x == maskX) - lineOBJ[sx] = 0x001F; -#endif - } - sx = (sx+1)&511;; - realX += dx; - realY += dy; - } - } else { - int c = (a2 & 0x3FF); - if((DISPCNT & 7) > 2 && (c < 512)) - continue; - - int inc = 32; - if(DISPCNT & 0x40) - inc = sizeX >> 3; - int palette = (a2 >> 8) & 0xF0; - for(int x = 0; x < fieldX; x++) { - int xxx = realX >> 8; - int yyy = realY >> 8; - if(xxx < 0 || xxx >= sizeX || - yyy < 0 || yyy >= sizeY || - sx >= 240); - else { - u32 color = vram[0x10000 + ((((c + (yyy>>3) * inc)<<5) - + ((yyy & 7)<<2) + ((xxx >> 3)<<5) + - ((xxx & 7)>>1))&0x7FFF)]; - if(xxx & 1) - color >>= 4; - else - color &= 0x0F; - - if ((color==0) && (((prio >> 25)&3) < - ((lineOBJ[sx]>>25)&3))) { - lineOBJ[sx] = (lineOBJ[sx] & 0xF9FFFFFF) | prio; - if((a0 & 0x1000) && m) - lineOBJ[sx]=(lineOBJ[sx-1] & 0xF9FFFFFF) | prio; - } else if((color) && (prio < (lineOBJ[sx]&0xFF000000))) { - lineOBJ[sx] = READ16LE(&spritePalette[palette+color]) | prio; - if((a0 & 0x1000) && m) - lineOBJ[sx]=(lineOBJ[sx-1] & 0xF9FFFFFF) | prio; - } - } - if((a0 & 0x1000) && m) { - m++; - if (m==mosaicX) - m=0; - } - -#ifdef SPRITE_DEBUG - if(t == 0 || t == maskY || x == 0 || x == maskX) - lineOBJ[sx] = 0x001F; -#endif - sx = (sx+1)&511;; - realX += dx; - realY += dy; - - } - } - } - } - } else { - int t = VCOUNT - sy; - if((t >= 0) && (t < sizeY)) { - int sx = (a1 & 0x1FF); - if(((sx < 240)||(((sx+sizeX)&511)<240)) && !(a0 & 0x0200)) { - if(a0 & 0x2000) { - if(a1 & 0x2000) - t = sizeY - t - 1; - int c = (a2 & 0x3FF); - if((DISPCNT & 7) > 2 && (c < 512)) - continue; - - int inc = 32; - if(DISPCNT & 0x40) { - inc = sizeX >> 2; - } else { - c &= 0x3FE; - } - int xxx = 0; - if(a1 & 0x1000) - xxx = sizeX-1; - - if(a0 & 0x1000) { - t -= (t % mosaicY); - } - - int address = 0x10000 + ((((c+ (t>>3) * inc) << 5) - + ((t & 7) << 3) + ((xxx>>3)<<6) + (xxx & 7)) & 0x7FFF); - - if(a1 & 0x1000) - xxx = 7; - u32 prio = (((a2 >> 10) & 3) << 25) | ((a0 & 0x0c00)<<6); - - for(int xx = 0; xx < sizeX; xx++) { - if(sx < 240) { - u8 color = vram[address]; - if ((color==0) && (((prio >> 25)&3) < - ((lineOBJ[sx]>>25)&3))) { - lineOBJ[sx] = (lineOBJ[sx] & 0xF9FFFFFF) | prio; - if((a0 & 0x1000) && m) - lineOBJ[sx]=(lineOBJ[sx-1] & 0xF9FFFFFF) | prio; - } else if((color) && (prio < (lineOBJ[sx]&0xFF000000))) { - lineOBJ[sx] = READ16LE(&spritePalette[color]) | prio; - if((a0 & 0x1000) && m) - lineOBJ[sx]=(lineOBJ[sx-1] & 0xF9FFFFFF) | prio; - } - - if (a0 & 0x1000) { - m++; - if (m==mosaicX) - m=0; - } - -#ifdef SPRITE_DEBUG - if(t == 0 || t == maskY || xx == 0 || xx == maskX) - lineOBJ[sx] = 0x001F; -#endif - } - - sx = (sx+1) & 511; - if(a1 & 0x1000) { - xxx--; - address--; - if(xxx == -1) { - address -= 56; - xxx = 7; - } - if(address < 0x10000) - address += 0x8000; - } else { - xxx++; - address++; - if(xxx == 8) { - address += 56; - xxx = 0; - } - if(address > 0x17fff) - address -= 0x8000; - } - } - } else { - if(a1 & 0x2000) - t = sizeY - t - 1; - int c = (a2 & 0x3FF); - if((DISPCNT & 7) > 2 && (c < 512)) - continue; - - int inc = 32; - if(DISPCNT & 0x40) { - inc = sizeX >> 3; - } - int xxx = 0; - if(a1 & 0x1000) - xxx = sizeX - 1; - - if(a0 & 0x1000) { - t -= (t % mosaicY); - } - - int address = 0x10000 + ((((c + (t>>3) * inc)<<5) - + ((t & 7)<<2) + ((xxx>>3)<<5) + ((xxx & 7) >> 1))&0x7FFF); - u32 prio = (((a2 >> 10) & 3) << 25) | ((a0 & 0x0c00)<<6); - int palette = (a2 >> 8) & 0xF0; - if(a1 & 0x1000) { - xxx = 7; - for(int xx = sizeX - 1; xx >= 0; xx--) { - if(sx < 240) { - u8 color = vram[address]; - if(xx & 1) { - color = (color >> 4); - } else - color &= 0x0F; - - if ((color==0) && (((prio >> 25)&3) < - ((lineOBJ[sx]>>25)&3))) { - lineOBJ[sx] = (lineOBJ[sx] & 0xF9FFFFFF) | prio; - if((a0 & 0x1000) && m) - lineOBJ[sx]=(lineOBJ[sx-1] & 0xF9FFFFFF) | prio; - } else if((color) && (prio < (lineOBJ[sx]&0xFF000000))) { - lineOBJ[sx] = READ16LE(&spritePalette[palette + color]) | prio; - if((a0 & 0x1000) && m) - lineOBJ[sx]=(lineOBJ[sx-1] & 0xF9FFFFFF) | prio; - } - } - if (a0 & 0x1000) { - m++; - if (m==mosaicX) - m=0; - } -#ifdef SPRITE_DEBUG - if(t == 0 || t == maskY || xx == 0 || xx == maskX) - lineOBJ[sx] = 0x001F; -#endif - sx = (sx+1) & 511; - xxx--; - if(!(xx & 1)) - address--; - if(xxx == -1) { - xxx = 7; - address -= 28; - } - if(address < 0x10000) - address += 0x8000; - } - } else { - for(int xx = 0; xx < sizeX; xx++) { - if(sx < 240) { - u8 color = vram[address]; - if(xx & 1) { - color = (color >> 4); - } else - color &= 0x0F; - - if ((color==0) && (((prio >> 25)&3) < - ((lineOBJ[sx]>>25)&3))) { - lineOBJ[sx] = (lineOBJ[sx] & 0xF9FFFFFF) | prio; - if((a0 & 0x1000) && m) - lineOBJ[sx]=(lineOBJ[sx-1] & 0xF9FFFFFF) | prio; - } else if((color) && (prio < (lineOBJ[sx]&0xFF000000))) { - lineOBJ[sx] = READ16LE(&spritePalette[palette + color]) | prio; - if((a0 & 0x1000) && m) - lineOBJ[sx]=(lineOBJ[sx-1] & 0xF9FFFFFF) | prio; - - } - } - if (a0 & 0x1000) { - m++; - if (m==mosaicX) - m=0; - } -#ifdef SPRITE_DEBUG - if(t == 0 || t == maskY || xx == 0 || xx == maskX) - lineOBJ[sx] = 0x001F; -#endif - sx = (sx+1) & 511; - xxx++; - if(xx & 1) - address++; - if(xxx == 8) { - address += 28; - xxx = 0; - } - if(address > 0x17fff) - address -= 0x8000; - } - } - } - } - } - } - } - } -} - -inline void gfxDrawOBJWin(u32 *lineOBJWin) -{ - gfxClearArray(lineOBJWin); - if(layerEnable & 0x8000) { - u16 *sprites = (u16 *)oam; - // u16 *spritePalette = &((u16 *)paletteRAM)[256]; - for(int x = 0; x < 128 ; x++) { - u16 a0 = READ16LE(sprites++); - u16 a1 = READ16LE(sprites++); - u16 a2 = READ16LE(sprites++); - sprites++; - - // ignore non OBJ-WIN - if((a0 & 0x0c00) != 0x0800) - continue; - - int sizeY = 8; - int sizeX = 8; - - switch(((a0 >>12) & 0x0c)|(a1>>14)) { - case 0: - break; - case 1: - sizeX = sizeY = 16; - break; - case 2: - sizeX = sizeY = 32; - break; - case 3: - sizeX = sizeY = 64; - break; - case 4: - sizeX = 16; - break; - case 5: - sizeX = 32; - break; - case 6: - sizeX = 32; - sizeY = 16; - break; - case 7: - sizeX = 64; - sizeY = 32; - break; - case 8: - sizeY = 16; - break; - case 9: - sizeY = 32; - break; - case 10: - sizeX = 16; - sizeY = 32; - break; - case 11: - sizeX = 32; - sizeY = 64; - break; - default: - continue; - } - - int sy = (a0 & 255); - - if(sy > 160) - sy -= 256; - - if(a0 & 0x0100) { - int fieldX = sizeX; - int fieldY = sizeY; - if(a0 & 0x0200) { - fieldX <<= 1; - fieldY <<= 1; - } - - int t = VCOUNT - sy; - if((t >= 0) && (t < fieldY)) { - int sx = (a1 & 0x1FF); - if((sx < 240) || (((sx + fieldX) & 511) < 240)) { - // int t2 = t - (fieldY >> 1); - int rot = (a1 >> 9) & 0x1F; - u16 *OAM = (u16 *)oam; - int dx = READ16LE(&OAM[3 + (rot << 4)]); - if(dx & 0x8000) - dx |= 0xFFFF8000; - int dmx = READ16LE(&OAM[7 + (rot << 4)]); - if(dmx & 0x8000) - dmx |= 0xFFFF8000; - int dy = READ16LE(&OAM[11 + (rot << 4)]); - if(dy & 0x8000) - dy |= 0xFFFF8000; - int dmy = READ16LE(&OAM[15 + (rot << 4)]); - if(dmy & 0x8000) - dmy |= 0xFFFF8000; - - int realX = ((sizeX) << 7) - (fieldX >> 1)*dx - (fieldY>>1)*dmx - + t * dmx; - int realY = ((sizeY) << 7) - (fieldX >> 1)*dy - (fieldY>>1)*dmy - + t * dmy; - - // u32 prio = (((a2 >> 10) & 3) << 25) | ((a0 & 0x0c00)<<6); - - if(a0 & 0x2000) { - int c = (a2 & 0x3FF); - if((DISPCNT & 7) > 2 && (c < 512)) - continue; - int inc = 32; - if(DISPCNT & 0x40) - inc = sizeX >> 2; - else - c &= 0x3FE; - for(int x = 0; x < fieldX; x++) { - int xxx = realX >> 8; - int yyy = realY >> 8; - - if(xxx < 0 || xxx >= sizeX || - yyy < 0 || yyy >= sizeY || - sx >= 240) { - } else { - u32 color = vram[0x10000 + ((((c + (yyy>>3) * inc)<<5) - + ((yyy & 7)<<3) + ((xxx >> 3)<<6) + - (xxx & 7))&0x7fff)]; - if(color) { - lineOBJWin[sx] = 1; - } - } - sx = (sx+1)&511;; - realX += dx; - realY += dy; - } - } else { - int c = (a2 & 0x3FF); - if((DISPCNT & 7) > 2 && (c < 512)) - continue; - - int inc = 32; - if(DISPCNT & 0x40) - inc = sizeX >> 3; - // int palette = (a2 >> 8) & 0xF0; - for(int x = 0; x < fieldX; x++) { - int xxx = realX >> 8; - int yyy = realY >> 8; - - // if(x == 0 || x == (sizeX-1) || - // t == 0 || t == (sizeY-1)) { - // lineOBJ[sx] = 0x001F | prio; - // } else { - if(xxx < 0 || xxx >= sizeX || - yyy < 0 || yyy >= sizeY || - sx >= 240){ - } else { - u32 color = vram[0x10000 + ((((c + (yyy>>3) * inc)<<5) - + ((yyy & 7)<<2) + ((xxx >> 3)<<5) + - ((xxx & 7)>>1))&0x7fff)]; - if(xxx & 1) - color >>= 4; - else - color &= 0x0F; - - if(color) { - lineOBJWin[sx] = 1; - } - } - // } - sx = (sx+1)&511;; - realX += dx; - realY += dy; - } - } - } - } - } else { - int t = VCOUNT - sy; - if((t >= 0) && (t < sizeY)) { - int sx = (a1 & 0x1FF); - if(((sx < 240)||(((sx+sizeX)&511)<240)) && !(a0 & 0x0200)) { - if(a0 & 0x2000) { - if(a1 & 0x2000) - t = sizeY - t - 1; - int c = (a2 & 0x3FF); - if((DISPCNT & 7) > 2 && (c < 512)) - continue; - - int inc = 32; - if(DISPCNT & 0x40) { - inc = sizeX >> 2; - } else { - c &= 0x3FE; - } - int xxx = 0; - if(a1 & 0x1000) - xxx = sizeX-1; - int address = 0x10000 + ((((c+ (t>>3) * inc) << 5) - + ((t & 7) << 3) + ((xxx>>3)<<6) + (xxx & 7))&0x7fff); - if(a1 & 0x1000) - xxx = 7; - // u32 prio = (((a2 >> 10) & 3) << 25) | ((a0 & 0x0c00)<<6); - for(int xx = 0; xx < sizeX; xx++) { - if(sx < 240) { - u8 color = vram[address]; - if(color) { - lineOBJWin[sx] = 1; - } - } - - sx = (sx+1) & 511; - if(a1 & 0x1000) { - xxx--; - address--; - if(xxx == -1) { - address -= 56; - xxx = 7; - } - if(address < 0x10000) - address += 0x8000; - } else { - xxx++; - address++; - if(xxx == 8) { - address += 56; - xxx = 0; - } - if(address > 0x17fff) - address -= 0x8000; - } - } - } else { - if(a1 & 0x2000) - t = sizeY - t - 1; - int c = (a2 & 0x3FF); - if((DISPCNT & 7) > 2 && (c < 512)) - continue; - - int inc = 32; - if(DISPCNT & 0x40) { - inc = sizeX >> 3; - } - int xxx = 0; - if(a1 & 0x1000) - xxx = sizeX - 1; - int address = 0x10000 + ((((c + (t>>3) * inc)<<5) - + ((t & 7)<<2) + ((xxx>>3)<<5) + ((xxx & 7) >> 1))&0x7fff); - // u32 prio = (((a2 >> 10) & 3) << 25) | ((a0 & 0x0c00)<<6); - // int palette = (a2 >> 8) & 0xF0; - if(a1 & 0x1000) { - xxx = 7; - for(int xx = sizeX - 1; xx >= 0; xx--) { - if(sx < 240) { - u8 color = vram[address]; - if(xx & 1) { - color = (color >> 4); - } else - color &= 0x0F; - - if(color) { - lineOBJWin[sx] = 1; - } - } - sx = (sx+1) & 511; - xxx--; - if(!(xx & 1)) - address--; - if(xxx == -1) { - xxx = 7; - address -= 28; - } - if(address < 0x10000) - address += 0x8000; - } - } else { - for(int xx = 0; xx < sizeX; xx++) { - if(sx < 240) { - u8 color = vram[address]; - if(xx & 1) { - color = (color >> 4); - } else - color &= 0x0F; - - if(color) { - lineOBJWin[sx] = 1; - } - } - sx = (sx+1) & 511; - xxx++; - if(xx & 1) - address++; - if(xxx == 8) { - address += 28; - xxx = 0; - } - if(address > 0x17fff) - address -= 0x8000; - } - } - } - } - } - } - } - } -} - -inline u32 gfxIncreaseBrightness(u32 color, int coeff) -{ - int r = (color & 0x1F); - int g = ((color >> 5) & 0x1F); - int b = ((color >> 10) & 0x1F); - - r = r + (((31 - r) * coeff) >> 4); - g = g + (((31 - g) * coeff) >> 4); - b = b + (((31 - b) * coeff) >> 4); - if(r > 31) - r = 31; - if(g > 31) - g = 31; - if(b > 31) - b = 31; - color = (color & 0xFFFF0000) | (b << 10) | (g << 5) | r; - return color; -} - -inline void gfxIncreaseBrightness(u32 *line, int coeff) -{ - for(int x = 0; x < 240; x++) { - u32 color = *line; - int r = (color & 0x1F); - int g = ((color >> 5) & 0x1F); - int b = ((color >> 10) & 0x1F); - - r = r + (((31 - r) * coeff) >> 4); - g = g + (((31 - g) * coeff) >> 4); - b = b + (((31 - b) * coeff) >> 4); - if(r > 31) - r = 31; - if(g > 31) - g = 31; - if(b > 31) - b = 31; - *line++ = (color & 0xFFFF0000) | (b << 10) | (g << 5) | r; - } -} - -inline u32 gfxDecreaseBrightness(u32 color, int coeff) -{ - int r = (color & 0x1F); - int g = ((color >> 5) & 0x1F); - int b = ((color >> 10) & 0x1F); - - r = r - ((r * coeff) >> 4); - g = g - ((g * coeff) >> 4); - b = b - ((b * coeff) >> 4); - if(r < 0) - r = 0; - if(g < 0) - g = 0; - if(b < 0) - b = 0; - color = (color & 0xFFFF0000) | (b << 10) | (g << 5) | r; - - return color; -} - -inline void gfxDecreaseBrightness(u32 *line, int coeff) -{ - for(int x = 0; x < 240; x++) { - u32 color = *line; - int r = (color & 0x1F); - int g = ((color >> 5) & 0x1F); - int b = ((color >> 10) & 0x1F); - - r = r - ((r * coeff) >> 4); - g = g - ((g * coeff) >> 4); - b = b - ((b * coeff) >> 4); - if(r < 0) - r = 0; - if(g < 0) - g = 0; - if(b < 0) - b = 0; - *line++ = (color & 0xFFFF0000) | (b << 10) | (g << 5) | r; - } -} - -inline u32 gfxAlphaBlend(u32 color, u32 color2, int ca, int cb) -{ - if(color < 0x80000000) { - int r = (color & 0x1F); - int g = ((color >> 5) & 0x1F); - int b = ((color >> 10) & 0x1F); - int r0 = (color2 & 0x1F); - int g0 = ((color2 >> 5) & 0x1F); - int b0 = ((color2 >> 10) & 0x1F); - - r = ((r * ca) >> 4) + ((r0 * cb) >> 4); - g = ((g * ca) >> 4) + ((g0 * cb) >> 4); - b = ((b * ca) >> 4) + ((b0 * cb) >> 4); - - if(r > 31) - r = 31; - if(g > 31) - g = 31; - if(b > 31) - b = 31; - - return (color & 0xFFFF0000) | (b << 10) | (g << 5) | r; - } - return color; -} - -inline void gfxAlphaBlend(u32 *ta, u32 *tb, int ca, int cb) -{ - for(int x = 0; x < 240; x++) { - u32 color = *ta; - if(color < 0x80000000) { - int r = (color & 0x1F); - int g = ((color >> 5) & 0x1F); - int b = ((color >> 10) & 0x1F); - u32 color2 = (*tb++); - int r0 = (color2 & 0x1F); - int g0 = ((color2 >> 5) & 0x1F); - int b0 = ((color2 >> 10) & 0x1F); - - r = ((r * ca) >> 4) + ((r0 * cb) >> 4); - g = ((g * ca) >> 4) + ((g0 * cb) >> 4); - b = ((b * ca) >> 4) + ((b0 * cb) >> 4); - - if(r > 31) - r = 31; - if(g > 31) - g = 31; - if(b > 31) - b = 31; - - *ta++ = (color & 0xFFFF0000) | (b << 10) | (g << 5) | r; - } else { - ta++; - tb++; - } - } -} - -#endif // VBA_GFX_H \ No newline at end of file +// -*- C++ -*- +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2005 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#ifndef VBA_GFX_H +#define VBA_GFX_H + +#include "GBA.h" +#include "Gfx.h" +#include "Globals.h" + +#include "Port.h" + +//#define SPRITE_DEBUG + +static void gfxDrawTextScreen(u16, u16, u16, u32 *); +static void gfxDrawRotScreen(u16, + u16, u16, + u16, u16, + u16, u16, + u16, u16, + int&, int&, + int, + u32*); +static void gfxDrawRotScreen16Bit(u16, + u16, u16, + u16, u16, + u16, u16, + u16, u16, + int&, int&, + int, + u32*); +static void gfxDrawRotScreen256(u16, + u16, u16, + u16, u16, + u16, u16, + u16, u16, + int&, int&, + int, + u32*); +static void gfxDrawRotScreen16Bit160(u16, + u16, u16, + u16, u16, + u16, u16, + u16, u16, + int&, int&, + int, + u32*); +static void gfxDrawSprites(u32 *); +static void gfxIncreaseBrightness(u32 *line, int coeff); +static void gfxDecreaseBrightness(u32 *line, int coeff); +static void gfxAlphaBlend(u32 *ta, u32 *tb, int ca, int cb); + +void mode0RenderLine(); +void mode0RenderLineNoWindow(); +void mode0RenderLineAll(); + +void mode1RenderLine(); +void mode1RenderLineNoWindow(); +void mode1RenderLineAll(); + +void mode2RenderLine(); +void mode2RenderLineNoWindow(); +void mode2RenderLineAll(); + +void mode3RenderLine(); +void mode3RenderLineNoWindow(); +void mode3RenderLineAll(); + +void mode4RenderLine(); +void mode4RenderLineNoWindow(); +void mode4RenderLineAll(); + +void mode5RenderLine(); +void mode5RenderLineNoWindow(); +void mode5RenderLineAll(); + +extern int coeff[32]; +extern u32 line0[240]; +extern u32 line1[240]; +extern u32 line2[240]; +extern u32 line3[240]; +extern u32 lineOBJ[240]; +extern u32 lineOBJWin[240]; +extern u32 lineMix[240]; +extern bool gfxInWin0[240]; +extern bool gfxInWin1[240]; +extern int lineOBJpixleft[128]; + +extern int gfxBG2Changed; +extern int gfxBG3Changed; + +extern int gfxBG2X; +extern int gfxBG2Y; +extern int gfxBG2LastX; +extern int gfxBG2LastY; +extern int gfxBG3X; +extern int gfxBG3Y; +extern int gfxBG3LastX; +extern int gfxBG3LastY; +extern int gfxLastVCOUNT; + +static inline void gfxClearArray(u32 *array) +{ + for(int i = 0; i < 240; i++) { + *array++ = 0x80000000; + } +} + +static inline void gfxDrawTextScreen(u16 control, u16 hofs, u16 vofs, + u32 *line) +{ + u16 *palette = (u16 *)paletteRAM; + u8 *charBase = &vram[((control >> 2) & 0x03) * 0x4000]; + u16 *screenBase = (u16 *)&vram[((control >> 8) & 0x1f) * 0x800]; + u32 prio = ((control & 3)<<25) + 0x1000000; + int sizeX = 256; + int sizeY = 256; + switch((control >> 14) & 3) { + case 0: + break; + case 1: + sizeX = 512; + break; + case 2: + sizeY = 512; + break; + case 3: + sizeX = 512; + sizeY = 512; + break; + } + + int maskX = sizeX-1; + int maskY = sizeY-1; + + bool mosaicOn = (control & 0x40) ? true : false; + + int xxx = hofs & maskX; + int yyy = (vofs + VCOUNT) & maskY; + int mosaicX = (MOSAIC & 0x000F)+1; + int mosaicY = ((MOSAIC & 0x00F0)>>4)+1; + + if(mosaicOn) { + if((VCOUNT % mosaicY) != 0) { + mosaicY = (VCOUNT / mosaicY) * mosaicY; + yyy = (vofs + mosaicY) & maskY; + } + } + + if(yyy > 255 && sizeY > 256) { + yyy &= 255; + screenBase += 0x400; + if(sizeX > 256) + screenBase += 0x400; + } + + int yshift = ((yyy>>3)<<5); + if((control) & 0x80) { + u16 *screenSource = screenBase + 0x400 * (xxx>>8) + ((xxx & 255)>>3) + yshift; + for(int x = 0; x < 240; x++) { + u16 data = READ16LE(screenSource); + + int tile = data & 0x3FF; + int tileX = (xxx & 7); + int tileY = yyy & 7; + + if(data & 0x0400) + tileX = 7 - tileX; + if(data & 0x0800) + tileY = 7 - tileY; + + u8 color = charBase[tile * 64 + tileY * 8 + tileX]; + + line[x] = color ? (READ16LE(&palette[color]) | prio): 0x80000000; + + if(data & 0x0400) { + if(tileX == 0) + screenSource++; + } else if(tileX == 7) + screenSource++; + xxx++; + if(xxx == 256) { + if(sizeX > 256) + screenSource = screenBase + 0x400 + yshift; + else { + screenSource = screenBase + yshift; + xxx = 0; + } + } else if(xxx >= sizeX) { + xxx = 0; + screenSource = screenBase + yshift; + } + } + } else { + u16 *screenSource = screenBase + 0x400*(xxx>>8)+((xxx&255)>>3) + + yshift; + for(int x = 0; x < 240; x++) { + u16 data = READ16LE(screenSource); + + int tile = data & 0x3FF; + int tileX = (xxx & 7); + int tileY = yyy & 7; + + if(data & 0x0400) + tileX = 7 - tileX; + if(data & 0x0800) + tileY = 7 - tileY; + + u8 color = charBase[(tile<<5) + (tileY<<2) + (tileX>>1)]; + + if(tileX & 1) { + color = (color >> 4); + } else { + color &= 0x0F; + } + + int pal = (READ16LE(screenSource)>>8) & 0xF0; + line[x] = color ? (READ16LE(&palette[pal + color])|prio): 0x80000000; + + if(data & 0x0400) { + if(tileX == 0) + screenSource++; + } else if(tileX == 7) + screenSource++; + xxx++; + if(xxx == 256) { + if(sizeX > 256) + screenSource = screenBase + 0x400 + yshift; + else { + screenSource = screenBase + yshift; + xxx = 0; + } + } else if(xxx >= sizeX) { + xxx = 0; + screenSource = screenBase + yshift; + } + } + } + if(mosaicOn) { + if(mosaicX > 1) { + int m = 1; + for(int i = 0; i < 239; i++) { + line[i+1] = line[i]; + m++; + if(m == mosaicX) { + m = 1; + i++; + } + } + } + } +} + +static inline void gfxDrawRotScreen(u16 control, + u16 x_l, u16 x_h, + u16 y_l, u16 y_h, + u16 pa, u16 pb, + u16 pc, u16 pd, + int& currentX, int& currentY, + int changed, + u32 *line) +{ + u16 *palette = (u16 *)paletteRAM; + u8 *charBase = &vram[((control >> 2) & 0x03) * 0x4000]; + u8 *screenBase = (u8 *)&vram[((control >> 8) & 0x1f) * 0x800]; + int prio = ((control & 3) << 25) + 0x1000000; + + int sizeX = 128; + int sizeY = 128; + switch((control >> 14) & 3) { + case 0: + break; + case 1: + sizeX = sizeY = 256; + break; + case 2: + sizeX = sizeY = 512; + break; + case 3: + sizeX = sizeY = 1024; + break; + } + + int dx = pa & 0x7FFF; + if(pa & 0x8000) + dx |= 0xFFFF8000; + int dmx = pb & 0x7FFF; + if(pb & 0x8000) + dmx |= 0xFFFF8000; + int dy = pc & 0x7FFF; + if(pc & 0x8000) + dy |= 0xFFFF8000; + int dmy = pd & 0x7FFF; + if(pd & 0x8000) + dmy |= 0xFFFF8000; + + if(VCOUNT == 0) + changed = 3; + + if(changed & 1) { + currentX = (x_l) | ((x_h & 0x07FF)<<16); + if(x_h & 0x0800) + currentX |= 0xF8000000; + } else { + currentX += dmx; + } + + if(changed & 2) { + currentY = (y_l) | ((y_h & 0x07FF)<<16); + if(y_h & 0x0800) + currentY |= 0xF8000000; + } else { + currentY += dmy; + } + + int realX = currentX; + int realY = currentY; + + if(control & 0x40) { + int mosaicY = ((MOSAIC & 0xF0)>>4) + 1; + int y = (VCOUNT % mosaicY); + realX -= y*dmx; + realY -= y*dmy; + } + + int xxx = (realX >> 8); + int yyy = (realY >> 8); + + if(control & 0x2000) { + xxx %= sizeX; + yyy %= sizeY; + if(xxx < 0) + xxx += sizeX; + if(yyy < 0) + yyy += sizeY; + } + + if(control & 0x80) { + for(int x = 0; x < 240; x++) { + if(xxx < 0 || + yyy < 0 || + xxx >= sizeX || + yyy >= sizeY) { + line[x] = 0x80000000; + } else { + int tile = screenBase[(xxx>>3) + (yyy>>3)*(sizeX>>3)]; + + int tileX = (xxx & 7); + int tileY = yyy & 7; + + u8 color = charBase[(tile<<6) + (tileY<<3) + tileX]; + + line[x] = color ? (READ16LE(&palette[color])|prio): 0x80000000; + } + realX += dx; + realY += dy; + + xxx = (realX >> 8); + yyy = (realY >> 8); + + if(control & 0x2000) { + xxx %= sizeX; + yyy %= sizeY; + if(xxx < 0) + xxx += sizeX; + if(yyy < 0) + yyy += sizeY; + } + } + } else { + for(int x = 0; x < 240; x++) { + if(xxx < 0 || + yyy < 0 || + xxx >= sizeX || + yyy >= sizeY) { + line[x] = 0x80000000; + } else { + int tile = screenBase[(xxx>>3) + (yyy>>3)*(sizeX>>3)]; + + int tileX = (xxx & 7); + int tileY = yyy & 7; + + u8 color = charBase[(tile<<6) + (tileY<<3) + tileX]; + + line[x] = color ? (READ16LE(&palette[color])|prio): 0x80000000; + } + realX += dx; + realY += dy; + + xxx = (realX >> 8); + yyy = (realY >> 8); + + if(control & 0x2000) { + xxx %= sizeX; + yyy %= sizeY; + if(xxx < 0) + xxx += sizeX; + if(yyy < 0) + yyy += sizeY; + } + } + } + + if(control & 0x40) { + int mosaicX = (MOSAIC & 0xF) + 1; + if(mosaicX > 1) { + int m = 1; + for(int i = 0; i < 239; i++) { + line[i+1] = line[i]; + m++; + if(m == mosaicX) { + m = 1; + i++; + } + } + } + } +} + +static inline void gfxDrawRotScreen16Bit(u16 control, + u16 x_l, u16 x_h, + u16 y_l, u16 y_h, + u16 pa, u16 pb, + u16 pc, u16 pd, + int& currentX, int& currentY, + int changed, + u32 *line) +{ + u16 *screenBase = (u16 *)&vram[0]; + int prio = ((control & 3) << 25) + 0x1000000; + int sizeX = 240; + int sizeY = 160; + + int startX = (x_l) | ((x_h & 0x07FF)<<16); + if(x_h & 0x0800) + startX |= 0xF8000000; + int startY = (y_l) | ((y_h & 0x07FF)<<16); + if(y_h & 0x0800) + startY |= 0xF8000000; + + int dx = pa & 0x7FFF; + if(pa & 0x8000) + dx |= 0xFFFF8000; + int dmx = pb & 0x7FFF; + if(pb & 0x8000) + dmx |= 0xFFFF8000; + int dy = pc & 0x7FFF; + if(pc & 0x8000) + dy |= 0xFFFF8000; + int dmy = pd & 0x7FFF; + if(pd & 0x8000) + dmy |= 0xFFFF8000; + + if(VCOUNT == 0) + changed = 3; + + if(changed & 1) { + currentX = (x_l) | ((x_h & 0x07FF)<<16); + if(x_h & 0x0800) + currentX |= 0xF8000000; + } else + currentX += dmx; + + if(changed & 2) { + currentY = (y_l) | ((y_h & 0x07FF)<<16); + if(y_h & 0x0800) + currentY |= 0xF8000000; + } else { + currentY += dmy; + } + + int realX = currentX; + int realY = currentY; + + if(control & 0x40) { + int mosaicY = ((MOSAIC & 0xF0)>>4) + 1; + int y = (VCOUNT % mosaicY); + realX -= y*dmx; + realY -= y*dmy; + } + + int xxx = (realX >> 8); + int yyy = (realY >> 8); + + for(int x = 0; x < 240; x++) { + if(xxx < 0 || + yyy < 0 || + xxx >= sizeX || + yyy >= sizeY) { + line[x] = 0x80000000; + } else { + line[x] = (READ16LE(&screenBase[yyy * sizeX + xxx]) | prio); + } + realX += dx; + realY += dy; + + xxx = (realX >> 8); + yyy = (realY >> 8); + } + + if(control & 0x40) { + int mosaicX = (MOSAIC & 0xF) + 1; + if(mosaicX > 1) { + int m = 1; + for(int i = 0; i < 239; i++) { + line[i+1] = line[i]; + m++; + if(m == mosaicX) { + m = 1; + i++; + } + } + } + } +} + +static inline void gfxDrawRotScreen256(u16 control, + u16 x_l, u16 x_h, + u16 y_l, u16 y_h, + u16 pa, u16 pb, + u16 pc, u16 pd, + int ¤tX, int& currentY, + int changed, + u32 *line) +{ + u16 *palette = (u16 *)paletteRAM; + u8 *screenBase = (DISPCNT & 0x0010) ? &vram[0xA000] : &vram[0x0000]; + int prio = ((control & 3) << 25) + 0x1000000; + int sizeX = 240; + int sizeY = 160; + + int startX = (x_l) | ((x_h & 0x07FF)<<16); + if(x_h & 0x0800) + startX |= 0xF8000000; + int startY = (y_l) | ((y_h & 0x07FF)<<16); + if(y_h & 0x0800) + startY |= 0xF8000000; + + int dx = pa & 0x7FFF; + if(pa & 0x8000) + dx |= 0xFFFF8000; + int dmx = pb & 0x7FFF; + if(pb & 0x8000) + dmx |= 0xFFFF8000; + int dy = pc & 0x7FFF; + if(pc & 0x8000) + dy |= 0xFFFF8000; + int dmy = pd & 0x7FFF; + if(pd & 0x8000) + dmy |= 0xFFFF8000; + + if(VCOUNT == 0) + changed = 3; + + if(changed & 1) { + currentX = (x_l) | ((x_h & 0x07FF)<<16); + if(x_h & 0x0800) + currentX |= 0xF8000000; + } else { + currentX += dmx; + } + + if(changed & 2) { + currentY = (y_l) | ((y_h & 0x07FF)<<16); + if(y_h & 0x0800) + currentY |= 0xF8000000; + } else { + currentY += dmy; + } + + int realX = currentX; + int realY = currentY; + + if(control & 0x40) { + int mosaicY = ((MOSAIC & 0xF0)>>4) + 1; + int y = (VCOUNT / mosaicY) * mosaicY; + realX = startX + y*dmx; + realY = startY + y*dmy; + } + + int xxx = (realX >> 8); + int yyy = (realY >> 8); + + for(int x = 0; x < 240; x++) { + if(xxx < 0 || + yyy < 0 || + xxx >= sizeX || + yyy >= sizeY) { + line[x] = 0x80000000; + } else { + u8 color = screenBase[yyy * 240 + xxx]; + + line[x] = color ? (READ16LE(&palette[color])|prio): 0x80000000; + } + realX += dx; + realY += dy; + + xxx = (realX >> 8); + yyy = (realY >> 8); + } + + if(control & 0x40) { + int mosaicX = (MOSAIC & 0xF) + 1; + if(mosaicX > 1) { + int m = 1; + for(int i = 0; i < 239; i++) { + line[i+1] = line[i]; + m++; + if(m == mosaicX) { + m = 1; + i++; + } + } + } + } +} + +static inline void gfxDrawRotScreen16Bit160(u16 control, + u16 x_l, u16 x_h, + u16 y_l, u16 y_h, + u16 pa, u16 pb, + u16 pc, u16 pd, + int& currentX, int& currentY, + int changed, + u32 *line) +{ + u16 *screenBase = (DISPCNT & 0x0010) ? (u16 *)&vram[0xa000] : + (u16 *)&vram[0]; + int prio = ((control & 3) << 25) + 0x1000000; + int sizeX = 160; + int sizeY = 128; + + int startX = (x_l) | ((x_h & 0x07FF)<<16); + if(x_h & 0x0800) + startX |= 0xF8000000; + int startY = (y_l) | ((y_h & 0x07FF)<<16); + if(y_h & 0x0800) + startY |= 0xF8000000; + + int dx = pa & 0x7FFF; + if(pa & 0x8000) + dx |= 0xFFFF8000; + int dmx = pb & 0x7FFF; + if(pb & 0x8000) + dmx |= 0xFFFF8000; + int dy = pc & 0x7FFF; + if(pc & 0x8000) + dy |= 0xFFFF8000; + int dmy = pd & 0x7FFF; + if(pd & 0x8000) + dmy |= 0xFFFF8000; + + if(VCOUNT == 0) + changed = 3; + + if(changed & 1) { + currentX = (x_l) | ((x_h & 0x07FF)<<16); + if(x_h & 0x0800) + currentX |= 0xF8000000; + } else { + currentX += dmx; + } + + if(changed & 2) { + currentY = (y_l) | ((y_h & 0x07FF)<<16); + if(y_h & 0x0800) + currentY |= 0xF8000000; + } else { + currentY += dmy; + } + + int realX = currentX; + int realY = currentY; + + if(control & 0x40) { + int mosaicY = ((MOSAIC & 0xF0)>>4) + 1; + int y = (VCOUNT / mosaicY) * mosaicY; + realX = startX + y*dmx; + realY = startY + y*dmy; + } + + int xxx = (realX >> 8); + int yyy = (realY >> 8); + + for(int x = 0; x < 240; x++) { + if(xxx < 0 || + yyy < 0 || + xxx >= sizeX || + yyy >= sizeY) { + line[x] = 0x80000000; + } else { + line[x] = (READ16LE(&screenBase[yyy * sizeX + xxx]) | prio); + } + realX += dx; + realY += dy; + + xxx = (realX >> 8); + yyy = (realY >> 8); + } + + if(control & 0x40) { + int mosaicX = (MOSAIC & 0xF) + 1; + if(mosaicX > 1) { + int m = 1; + for(int i = 0; i < 239; i++) { + line[i+1] = line[i]; + m++; + if(m == mosaicX) { + m = 1; + i++; + } + } + } + } +} + +static inline void gfxDrawSprites(u32 *lineOBJ) +{ + // lineOBJpix is used to keep track of the drawn OBJs + // and to stop drawing them if the 'maximum number of OBJ per line' + // has been reached. + int lineOBJpix = (DISPCNT & 0x20) ? 954 : 1226; + int m=0; + gfxClearArray(lineOBJ); + if(layerEnable & 0x1000) { + u16 *sprites = (u16 *)oam; + u16 *spritePalette = &((u16 *)paletteRAM)[256]; + int mosaicY = ((MOSAIC & 0xF000)>>12) + 1; + int mosaicX = ((MOSAIC & 0xF00)>>8) + 1; + for(int x = 0; x < 128 ; x++) { + u16 a0 = READ16LE(sprites++); + u16 a1 = READ16LE(sprites++); + u16 a2 = READ16LE(sprites++); + sprites++; + + lineOBJpixleft[x]=lineOBJpix; + + lineOBJpix-=2; + if (lineOBJpix<=0) + continue; + + if ((a0 & 0x0c00) == 0x0c00) + a0 &=0xF3FF; + + if ((a0>>14) == 3) + { + a0 &= 0x3FFF; + a1 &= 0x3FFF; + } + + int sizeX = 8<<(a1>>14); + int sizeY = sizeX; + + if ((a0>>14) & 1) + { + if (sizeX<32) + sizeX<<=1; + if (sizeY>8) + sizeY>>=1; + } + else if ((a0>>14) & 2) + { + if (sizeX>8) + sizeX>>=1; + if (sizeY<32) + sizeY<<=1; + } + +#ifdef SPRITE_DEBUG + int maskX = sizeX-1; + int maskY = sizeY-1; +#endif + + int sy = (a0 & 255); + int sx = (a1 & 0x1FF); + + // computes ticks used by OBJ-WIN if OBJWIN is enabled + if (((a0 & 0x0c00) == 0x0800) && (layerEnable & 0x8000)) + { + if ((a0 & 0x0300) == 0x0300) + { + sizeX<<=1; + sizeY<<=1; + } + if((sy+sizeY) > 256) + sy -= 256; + if ((sx+sizeX)> 512) + sx-=512; + if (sx<0) + { + sizeX+=sx; + sx = 0; + } + else if ((sx+sizeX)>240) + sizeX=240-sx; + if ((VCOUNT>=sy) && (VCOUNT 256) + sy -= 256; + int t = VCOUNT - sy; + if((t >= 0) && (t < fieldY)) { + int startpix = 0; + if ((sx+fieldX)> 512) + { + startpix=512-sx; + } + if (lineOBJpix>0) + if((sx < 240) || startpix) { + lineOBJpix-=8; + // int t2 = t - (fieldY >> 1); + int rot = (a1 >> 9) & 0x1F; + u16 *OAM = (u16 *)oam; + int dx = READ16LE(&OAM[3 + (rot << 4)]); + if(dx & 0x8000) + dx |= 0xFFFF8000; + int dmx = READ16LE(&OAM[7 + (rot << 4)]); + if(dmx & 0x8000) + dmx |= 0xFFFF8000; + int dy = READ16LE(&OAM[11 + (rot << 4)]); + if(dy & 0x8000) + dy |= 0xFFFF8000; + int dmy = READ16LE(&OAM[15 + (rot << 4)]); + if(dmy & 0x8000) + dmy |= 0xFFFF8000; + + if(a0 & 0x1000) { + t -= (t % mosaicY); + } + + int realX = ((sizeX) << 7) - (fieldX >> 1)*dx - (fieldY>>1)*dmx + + t * dmx; + int realY = ((sizeY) << 7) - (fieldX >> 1)*dy - (fieldY>>1)*dmy + + t * dmy; + + u32 prio = (((a2 >> 10) & 3) << 25) | ((a0 & 0x0c00)<<6); + + if(a0 & 0x2000) { + int c = (a2 & 0x3FF); + if((DISPCNT & 7) > 2 && (c < 512)) + continue; + int inc = 32; + if(DISPCNT & 0x40) + inc = sizeX >> 2; + else + c &= 0x3FE; + for(int x = 0; x < fieldX; x++) { + if (x >= startpix) + lineOBJpix-=2; + if (lineOBJpix<0) + continue; + int xxx = realX >> 8; + int yyy = realY >> 8; + + if(xxx < 0 || xxx >= sizeX || + yyy < 0 || yyy >= sizeY || + sx >= 240); + else { + u32 color = vram[0x10000 + ((((c + (yyy>>3) * inc)<<5) + + ((yyy & 7)<<3) + ((xxx >> 3)<<6) + + (xxx & 7))&0x7FFF)]; + if ((color==0) && (((prio >> 25)&3) < + ((lineOBJ[sx]>>25)&3))) { + lineOBJ[sx] = (lineOBJ[sx] & 0xF9FFFFFF) | prio; + if((a0 & 0x1000) && m) + lineOBJ[sx]=(lineOBJ[sx-1] & 0xF9FFFFFF) | prio; + } else if((color) && (prio < (lineOBJ[sx]&0xFF000000))) { + lineOBJ[sx] = READ16LE(&spritePalette[color]) | prio; + if((a0 & 0x1000) && m) + lineOBJ[sx]=(lineOBJ[sx-1] & 0xF9FFFFFF) | prio; + } + + if (a0 & 0x1000) { + m++; + if (m==mosaicX) + m=0; + } +#ifdef SPRITE_DEBUG + if(t == 0 || t == maskY || x == 0 || x == maskX) + lineOBJ[sx] = 0x001F; +#endif + } + sx = (sx+1)&511; + realX += dx; + realY += dy; + } + } else { + int c = (a2 & 0x3FF); + if((DISPCNT & 7) > 2 && (c < 512)) + continue; + + int inc = 32; + if(DISPCNT & 0x40) + inc = sizeX >> 3; + int palette = (a2 >> 8) & 0xF0; + for(int x = 0; x < fieldX; x++) { + if (x >= startpix) + lineOBJpix-=2; + if (lineOBJpix<0) + continue; + int xxx = realX >> 8; + int yyy = realY >> 8; + if(xxx < 0 || xxx >= sizeX || + yyy < 0 || yyy >= sizeY || + sx >= 240); + else { + u32 color = vram[0x10000 + ((((c + (yyy>>3) * inc)<<5) + + ((yyy & 7)<<2) + ((xxx >> 3)<<5) + + ((xxx & 7)>>1))&0x7FFF)]; + if(xxx & 1) + color >>= 4; + else + color &= 0x0F; + + if ((color==0) && (((prio >> 25)&3) < + ((lineOBJ[sx]>>25)&3))) { + lineOBJ[sx] = (lineOBJ[sx] & 0xF9FFFFFF) | prio; + if((a0 & 0x1000) && m) + lineOBJ[sx]=(lineOBJ[sx-1] & 0xF9FFFFFF) | prio; + } else if((color) && (prio < (lineOBJ[sx]&0xFF000000))) { + lineOBJ[sx] = READ16LE(&spritePalette[palette+color]) | prio; + if((a0 & 0x1000) && m) + lineOBJ[sx]=(lineOBJ[sx-1] & 0xF9FFFFFF) | prio; + } + } + if((a0 & 0x1000) && m) { + m++; + if (m==mosaicX) + m=0; + } + +#ifdef SPRITE_DEBUG + if(t == 0 || t == maskY || x == 0 || x == maskX) + lineOBJ[sx] = 0x001F; +#endif + sx = (sx+1)&511; + realX += dx; + realY += dy; + + } + } + } + } + } else { + if(sy+sizeY > 256) + sy -= 256; + int t = VCOUNT - sy; + if((t >= 0) && (t < sizeY)) { + int startpix = 0; + if ((sx+sizeX)> 512) + { + startpix=512-sx; + } + if((sx < 240) || startpix) { + lineOBJpix+=2; + if(a0 & 0x2000) { + if(a1 & 0x2000) + t = sizeY - t - 1; + int c = (a2 & 0x3FF); + if((DISPCNT & 7) > 2 && (c < 512)) + continue; + + int inc = 32; + if(DISPCNT & 0x40) { + inc = sizeX >> 2; + } else { + c &= 0x3FE; + } + int xxx = 0; + if(a1 & 0x1000) + xxx = sizeX-1; + + if(a0 & 0x1000) { + t -= (t % mosaicY); + } + + int address = 0x10000 + ((((c+ (t>>3) * inc) << 5) + + ((t & 7) << 3) + ((xxx>>3)<<6) + (xxx & 7)) & 0x7FFF); + + if(a1 & 0x1000) + xxx = 7; + u32 prio = (((a2 >> 10) & 3) << 25) | ((a0 & 0x0c00)<<6); + + for(int xx = 0; xx < sizeX; xx++) { + if (xx >= startpix) + lineOBJpix--; + if (lineOBJpix<0) + continue; + if(sx < 240) { + u8 color = vram[address]; + if ((color==0) && (((prio >> 25)&3) < + ((lineOBJ[sx]>>25)&3))) { + lineOBJ[sx] = (lineOBJ[sx] & 0xF9FFFFFF) | prio; + if((a0 & 0x1000) && m) + lineOBJ[sx]=(lineOBJ[sx-1] & 0xF9FFFFFF) | prio; + } else if((color) && (prio < (lineOBJ[sx]&0xFF000000))) { + lineOBJ[sx] = READ16LE(&spritePalette[color]) | prio; + if((a0 & 0x1000) && m) + lineOBJ[sx]=(lineOBJ[sx-1] & 0xF9FFFFFF) | prio; + } + + if (a0 & 0x1000) { + m++; + if (m==mosaicX) + m=0; + } + +#ifdef SPRITE_DEBUG + if(t == 0 || t == maskY || xx == 0 || xx == maskX) + lineOBJ[sx] = 0x001F; +#endif + } + + sx = (sx+1) & 511; + if(a1 & 0x1000) { + xxx--; + address--; + if(xxx == -1) { + address -= 56; + xxx = 7; + } + if(address < 0x10000) + address += 0x8000; + } else { + xxx++; + address++; + if(xxx == 8) { + address += 56; + xxx = 0; + } + if(address > 0x17fff) + address -= 0x8000; + } + } + } else { + if(a1 & 0x2000) + t = sizeY - t - 1; + int c = (a2 & 0x3FF); + if((DISPCNT & 7) > 2 && (c < 512)) + continue; + + int inc = 32; + if(DISPCNT & 0x40) { + inc = sizeX >> 3; + } + int xxx = 0; + if(a1 & 0x1000) + xxx = sizeX - 1; + + if(a0 & 0x1000) { + t -= (t % mosaicY); + } + + int address = 0x10000 + ((((c + (t>>3) * inc)<<5) + + ((t & 7)<<2) + ((xxx>>3)<<5) + ((xxx & 7) >> 1))&0x7FFF); + u32 prio = (((a2 >> 10) & 3) << 25) | ((a0 & 0x0c00)<<6); + int palette = (a2 >> 8) & 0xF0; + if(a1 & 0x1000) { + xxx = 7; + for(int xx = sizeX - 1; xx >= 0; xx--) { + if (xx >= startpix) + lineOBJpix--; + if (lineOBJpix<0) + continue; + if(sx < 240) { + u8 color = vram[address]; + if(xx & 1) { + color = (color >> 4); + } else + color &= 0x0F; + + if ((color==0) && (((prio >> 25)&3) < + ((lineOBJ[sx]>>25)&3))) { + lineOBJ[sx] = (lineOBJ[sx] & 0xF9FFFFFF) | prio; + if((a0 & 0x1000) && m) + lineOBJ[sx]=(lineOBJ[sx-1] & 0xF9FFFFFF) | prio; + } else if((color) && (prio < (lineOBJ[sx]&0xFF000000))) { + lineOBJ[sx] = READ16LE(&spritePalette[palette + color]) | prio; + if((a0 & 0x1000) && m) + lineOBJ[sx]=(lineOBJ[sx-1] & 0xF9FFFFFF) | prio; + } + } + if (a0 & 0x1000) { + m++; + if (m==mosaicX) + m=0; + } +#ifdef SPRITE_DEBUG + if(t == 0 || t == maskY || xx == 0 || xx == maskX) + lineOBJ[sx] = 0x001F; +#endif + sx = (sx+1) & 511; + xxx--; + if(!(xx & 1)) + address--; + if(xxx == -1) { + xxx = 7; + address -= 28; + } + if(address < 0x10000) + address += 0x8000; + } + } else { + for(int xx = 0; xx < sizeX; xx++) { + if (xx >= startpix) + lineOBJpix--; + if (lineOBJpix<0) + continue; + if(sx < 240) { + u8 color = vram[address]; + if(xx & 1) { + color = (color >> 4); + } else + color &= 0x0F; + + if ((color==0) && (((prio >> 25)&3) < + ((lineOBJ[sx]>>25)&3))) { + lineOBJ[sx] = (lineOBJ[sx] & 0xF9FFFFFF) | prio; + if((a0 & 0x1000) && m) + lineOBJ[sx]=(lineOBJ[sx-1] & 0xF9FFFFFF) | prio; + } else if((color) && (prio < (lineOBJ[sx]&0xFF000000))) { + lineOBJ[sx] = READ16LE(&spritePalette[palette + color]) | prio; + if((a0 & 0x1000) && m) + lineOBJ[sx]=(lineOBJ[sx-1] & 0xF9FFFFFF) | prio; + + } + } + if (a0 & 0x1000) { + m++; + if (m==mosaicX) + m=0; + } +#ifdef SPRITE_DEBUG + if(t == 0 || t == maskY || xx == 0 || xx == maskX) + lineOBJ[sx] = 0x001F; +#endif + sx = (sx+1) & 511; + xxx++; + if(xx & 1) + address++; + if(xxx == 8) { + address += 28; + xxx = 0; + } + if(address > 0x17fff) + address -= 0x8000; + } + } + } + } + } + } + } + } +} + +static inline void gfxDrawOBJWin(u32 *lineOBJWin) +{ + gfxClearArray(lineOBJWin); + if((layerEnable & 0x9000) == 0x9000) { + u16 *sprites = (u16 *)oam; + // u16 *spritePalette = &((u16 *)paletteRAM)[256]; + for(int x = 0; x < 128 ; x++) { + int lineOBJpix = lineOBJpixleft[x]; + u16 a0 = READ16LE(sprites++); + u16 a1 = READ16LE(sprites++); + u16 a2 = READ16LE(sprites++); + sprites++; + + if (lineOBJpix<=0) + continue; + + // ignores non OBJ-WIN and disabled OBJ-WIN + if(((a0 & 0x0c00) != 0x0800) || ((a0 & 0x0300) == 0x0200)) + continue; + + if ((a0 & 0x0c00) == 0x0c00) + a0 &=0xF3FF; + + if ((a0>>14) == 3) + { + a0 &= 0x3FFF; + a1 &= 0x3FFF; + } + + int sizeX = 8<<(a1>>14); + int sizeY = sizeX; + + if ((a0>>14) & 1) + { + if (sizeX<32) + sizeX<<=1; + if (sizeY>8) + sizeY>>=1; + } + else if ((a0>>14) & 2) + { + if (sizeX>8) + sizeX>>=1; + if (sizeY<32) + sizeY<<=1; + } + + int sy = (a0 & 255); + + if(a0 & 0x0100) { + int fieldX = sizeX; + int fieldY = sizeY; + if(a0 & 0x0200) { + fieldX <<= 1; + fieldY <<= 1; + } + if((sy+fieldY) > 256) + sy -= 256; + int t = VCOUNT - sy; + if((t >= 0) && (t < fieldY)) { + int sx = (a1 & 0x1FF); + int startpix = 0; + if ((sx+fieldX)> 512) + { + startpix=512-sx; + } + if((sx < 240) || startpix) { + lineOBJpix-=8; + // int t2 = t - (fieldY >> 1); + int rot = (a1 >> 9) & 0x1F; + u16 *OAM = (u16 *)oam; + int dx = READ16LE(&OAM[3 + (rot << 4)]); + if(dx & 0x8000) + dx |= 0xFFFF8000; + int dmx = READ16LE(&OAM[7 + (rot << 4)]); + if(dmx & 0x8000) + dmx |= 0xFFFF8000; + int dy = READ16LE(&OAM[11 + (rot << 4)]); + if(dy & 0x8000) + dy |= 0xFFFF8000; + int dmy = READ16LE(&OAM[15 + (rot << 4)]); + if(dmy & 0x8000) + dmy |= 0xFFFF8000; + + int realX = ((sizeX) << 7) - (fieldX >> 1)*dx - (fieldY>>1)*dmx + + t * dmx; + int realY = ((sizeY) << 7) - (fieldX >> 1)*dy - (fieldY>>1)*dmy + + t * dmy; + + // u32 prio = (((a2 >> 10) & 3) << 25) | ((a0 & 0x0c00)<<6); + + if(a0 & 0x2000) { + int c = (a2 & 0x3FF); + if((DISPCNT & 7) > 2 && (c < 512)) + continue; + int inc = 32; + if(DISPCNT & 0x40) + inc = sizeX >> 2; + else + c &= 0x3FE; + for(int x = 0; x < fieldX; x++) { + if (x >= startpix) + lineOBJpix-=2; + if (lineOBJpix<0) + continue; + int xxx = realX >> 8; + int yyy = realY >> 8; + + if(xxx < 0 || xxx >= sizeX || + yyy < 0 || yyy >= sizeY || + sx >= 240) { + } else { + u32 color = vram[0x10000 + ((((c + (yyy>>3) * inc)<<5) + + ((yyy & 7)<<3) + ((xxx >> 3)<<6) + + (xxx & 7))&0x7fff)]; + if(color) { + lineOBJWin[sx] = 1; + } + } + sx = (sx+1)&511; + realX += dx; + realY += dy; + } + } else { + int c = (a2 & 0x3FF); + if((DISPCNT & 7) > 2 && (c < 512)) + continue; + + int inc = 32; + if(DISPCNT & 0x40) + inc = sizeX >> 3; + // int palette = (a2 >> 8) & 0xF0; + for(int x = 0; x < fieldX; x++) { + if (x >= startpix) + lineOBJpix-=2; + if (lineOBJpix<0) + continue; + int xxx = realX >> 8; + int yyy = realY >> 8; + + // if(x == 0 || x == (sizeX-1) || + // t == 0 || t == (sizeY-1)) { + // lineOBJ[sx] = 0x001F | prio; + // } else { + if(xxx < 0 || xxx >= sizeX || + yyy < 0 || yyy >= sizeY || + sx >= 240) { + } else { + u32 color = vram[0x10000 + ((((c + (yyy>>3) * inc)<<5) + + ((yyy & 7)<<2) + ((xxx >> 3)<<5) + + ((xxx & 7)>>1))&0x7fff)]; + if(xxx & 1) + color >>= 4; + else + color &= 0x0F; + + if(color) { + lineOBJWin[sx] = 1; + } + } + // } + sx = (sx+1)&511; + realX += dx; + realY += dy; + } + } + } + } + } else { + if((sy+sizeY) > 256) + sy -= 256; + int t = VCOUNT - sy; + if((t >= 0) && (t < sizeY)) { + int sx = (a1 & 0x1FF); + int startpix = 0; + if ((sx+sizeX)> 512) + { + startpix=512-sx; + } + if((sx < 240) || startpix) { + lineOBJpix+=2; + if(a0 & 0x2000) { + if(a1 & 0x2000) + t = sizeY - t - 1; + int c = (a2 & 0x3FF); + if((DISPCNT & 7) > 2 && (c < 512)) + continue; + + int inc = 32; + if(DISPCNT & 0x40) { + inc = sizeX >> 2; + } else { + c &= 0x3FE; + } + int xxx = 0; + if(a1 & 0x1000) + xxx = sizeX-1; + int address = 0x10000 + ((((c+ (t>>3) * inc) << 5) + + ((t & 7) << 3) + ((xxx>>3)<<6) + (xxx & 7))&0x7fff); + if(a1 & 0x1000) + xxx = 7; + // u32 prio = (((a2 >> 10) & 3) << 25) | ((a0 & 0x0c00)<<6); + for(int xx = 0; xx < sizeX; xx++) { + if (xx >= startpix) + lineOBJpix--; + if (lineOBJpix<0) + continue; + if(sx < 240) { + u8 color = vram[address]; + if(color) { + lineOBJWin[sx] = 1; + } + } + + sx = (sx+1) & 511; + if(a1 & 0x1000) { + xxx--; + address--; + if(xxx == -1) { + address -= 56; + xxx = 7; + } + if(address < 0x10000) + address += 0x8000; + } else { + xxx++; + address++; + if(xxx == 8) { + address += 56; + xxx = 0; + } + if(address > 0x17fff) + address -= 0x8000; + } + } + } else { + if(a1 & 0x2000) + t = sizeY - t - 1; + int c = (a2 & 0x3FF); + if((DISPCNT & 7) > 2 && (c < 512)) + continue; + + int inc = 32; + if(DISPCNT & 0x40) { + inc = sizeX >> 3; + } + int xxx = 0; + if(a1 & 0x1000) + xxx = sizeX - 1; + int address = 0x10000 + ((((c + (t>>3) * inc)<<5) + + ((t & 7)<<2) + ((xxx>>3)<<5) + ((xxx & 7) >> 1))&0x7fff); + // u32 prio = (((a2 >> 10) & 3) << 25) | ((a0 & 0x0c00)<<6); + // int palette = (a2 >> 8) & 0xF0; + if(a1 & 0x1000) { + xxx = 7; + for(int xx = sizeX - 1; xx >= 0; xx--) { + if (xx >= startpix) + lineOBJpix--; + if (lineOBJpix<0) + continue; + if(sx < 240) { + u8 color = vram[address]; + if(xx & 1) { + color = (color >> 4); + } else + color &= 0x0F; + + if(color) { + lineOBJWin[sx] = 1; + } + } + sx = (sx+1) & 511; + xxx--; + if(!(xx & 1)) + address--; + if(xxx == -1) { + xxx = 7; + address -= 28; + } + if(address < 0x10000) + address += 0x8000; + } + } else { + for(int xx = 0; xx < sizeX; xx++) { + if (xx >= startpix) + lineOBJpix--; + if (lineOBJpix<0) + continue; + if(sx < 240) { + u8 color = vram[address]; + if(xx & 1) { + color = (color >> 4); + } else + color &= 0x0F; + + if(color) { + lineOBJWin[sx] = 1; + } + } + sx = (sx+1) & 511; + xxx++; + if(xx & 1) + address++; + if(xxx == 8) { + address += 28; + xxx = 0; + } + if(address > 0x17fff) + address -= 0x8000; + } + } + } + } + } + } + } + } +} + +static inline u32 gfxIncreaseBrightness(u32 color, int coeff) +{ + int r = (color & 0x1F); + int g = ((color >> 5) & 0x1F); + int b = ((color >> 10) & 0x1F); + + r = r + (((31 - r) * coeff) >> 4); + g = g + (((31 - g) * coeff) >> 4); + b = b + (((31 - b) * coeff) >> 4); + if(r > 31) + r = 31; + if(g > 31) + g = 31; + if(b > 31) + b = 31; + color = (color & 0xFFFF0000) | (b << 10) | (g << 5) | r; + return color; +} + +static inline void gfxIncreaseBrightness(u32 *line, int coeff) +{ + for(int x = 0; x < 240; x++) { + u32 color = *line; + int r = (color & 0x1F); + int g = ((color >> 5) & 0x1F); + int b = ((color >> 10) & 0x1F); + + r = r + (((31 - r) * coeff) >> 4); + g = g + (((31 - g) * coeff) >> 4); + b = b + (((31 - b) * coeff) >> 4); + if(r > 31) + r = 31; + if(g > 31) + g = 31; + if(b > 31) + b = 31; + *line++ = (color & 0xFFFF0000) | (b << 10) | (g << 5) | r; + } +} + +static inline u32 gfxDecreaseBrightness(u32 color, int coeff) +{ + int r = (color & 0x1F); + int g = ((color >> 5) & 0x1F); + int b = ((color >> 10) & 0x1F); + + r = r - ((r * coeff) >> 4); + g = g - ((g * coeff) >> 4); + b = b - ((b * coeff) >> 4); + if(r < 0) + r = 0; + if(g < 0) + g = 0; + if(b < 0) + b = 0; + color = (color & 0xFFFF0000) | (b << 10) | (g << 5) | r; + + return color; +} + +static inline void gfxDecreaseBrightness(u32 *line, int coeff) +{ + for(int x = 0; x < 240; x++) { + u32 color = *line; + int r = (color & 0x1F); + int g = ((color >> 5) & 0x1F); + int b = ((color >> 10) & 0x1F); + + r = r - ((r * coeff) >> 4); + g = g - ((g * coeff) >> 4); + b = b - ((b * coeff) >> 4); + if(r < 0) + r = 0; + if(g < 0) + g = 0; + if(b < 0) + b = 0; + *line++ = (color & 0xFFFF0000) | (b << 10) | (g << 5) | r; + } +} + +static inline u32 gfxAlphaBlend(u32 color, u32 color2, int ca, int cb) +{ + if(color < 0x80000000) { + int r = (color & 0x1F); + int g = ((color >> 5) & 0x1F); + int b = ((color >> 10) & 0x1F); + int r0 = (color2 & 0x1F); + int g0 = ((color2 >> 5) & 0x1F); + int b0 = ((color2 >> 10) & 0x1F); + + r = ((r * ca) + (r0 * cb)) >> 4; + g = ((g * ca) + (g0 * cb)) >> 4; + b = ((b * ca) + (b0 * cb)) >> 4; + + if(r > 31) + r = 31; + if(g > 31) + g = 31; + if(b > 31) + b = 31; + + return (color & 0xFFFF0000) | (b << 10) | (g << 5) | r; + } + return color; +} + +static inline void gfxAlphaBlend(u32 *ta, u32 *tb, int ca, int cb) +{ + for(int x = 0; x < 240; x++) { + u32 color = *ta; + if(color < 0x80000000) { + int r = (color & 0x1F); + int g = ((color >> 5) & 0x1F); + int b = ((color >> 10) & 0x1F); + u32 color2 = (*tb++); + int r0 = (color2 & 0x1F); + int g0 = ((color2 >> 5) & 0x1F); + int b0 = ((color2 >> 10) & 0x1F); + + r = ((r * ca) + (r0 * cb)) >> 4; + g = ((g * ca) + (g0 * cb)) >> 4; + b = ((b * ca) + (b0 * cb)) >> 4; + + if(r > 31) + r = 31; + if(g > 31) + g = 31; + if(b > 31) + b = 31; + + *ta++ = (color & 0xFFFF0000) | (b << 10) | (g << 5) | r; + } else { + ta++; + tb++; + } + } +} + +#endif // VBA_GFX_H diff --git a/src/Globals.cpp b/src/Globals.cpp index cb863c32..a256ce37 100644 --- a/src/Globals.cpp +++ b/src/Globals.cpp @@ -16,7 +16,12 @@ // along with this program; if not, write to the Free Software Foundation, // Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. -#include "Globals.h" +#include "GBA.h" + +#ifdef BKPT_SUPPORT +int oldreg[17]; +char oldbuffer[10]; +#endif reg_pair reg[45]; memoryMap map[256]; @@ -43,8 +48,8 @@ int layerSettings = 0xff00; int layerEnable = 0xff00; bool speedHack = false; int cpuSaveType = 0; -bool cpuEnhancedDetection = true; bool cheatsEnabled = true; +bool mirroringEnable = false; u8 *bios = NULL; u8 *rom = NULL; diff --git a/src/Globals.h b/src/Globals.h index 2466a3f2..34d76b88 100644 --- a/src/Globals.h +++ b/src/Globals.h @@ -57,8 +57,8 @@ extern int layerSettings; extern int layerEnable; extern bool speedHack; extern int cpuSaveType; -extern bool cpuEnhancedDetection; extern bool cheatsEnabled; +extern bool mirroringEnable; extern u8 *bios; extern u8 *rom; diff --git a/src/NLS.h b/src/NLS.h index d243a863..92b24eb3 100644 --- a/src/NLS.h +++ b/src/NLS.h @@ -1,62 +1,64 @@ -// -*- C++ -*- -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#define N_(String) (String) - -#define MSG_UNSUPPORTED_VBA_SGM 1 -#define MSG_CANNOT_LOAD_SGM 2 -#define MSG_SAVE_GAME_NOT_USING_BIOS 3 -#define MSG_SAVE_GAME_USING_BIOS 4 -#define MSG_UNSUPPORTED_SAVE_TYPE 5 -#define MSG_CANNOT_OPEN_FILE 6 -#define MSG_BAD_ARCHIVE_FILE 7 -#define MSG_NO_IMAGE_ON_ARCHIVE 8 -#define MSG_ERROR_OPENING_IMAGE 9 -#define MSG_ERROR_READING_IMAGE 10 -#define MSG_UNSUPPORTED_BIOS_FUNCTION 11 -#define MSG_INVALID_BIOS_FILE_SIZE 12 -#define MSG_INVALID_CHEAT_CODE 13 -#define MSG_UNKNOWN_ARM_OPCODE 14 -#define MSG_UNKNOWN_THUMB_OPCODE 15 -#define MSG_ERROR_CREATING_FILE 16 -#define MSG_FAILED_TO_READ_SGM 17 -#define MSG_FAILED_TO_READ_RTC 18 -#define MSG_UNSUPPORTED_VB_SGM 19 -#define MSG_CANNOT_LOAD_SGM_FOR 20 -#define MSG_ERROR_OPENING_IMAGE_FROM 21 -#define MSG_ERROR_READING_IMAGE_FROM 22 -#define MSG_UNSUPPORTED_ROM_SIZE 23 -#define MSG_UNSUPPORTED_RAM_SIZE 24 -#define MSG_UNKNOWN_CARTRIDGE_TYPE 25 -#define MSG_MAXIMUM_NUMBER_OF_CHEATS 26 -#define MSG_INVALID_GAMESHARK_CODE 27 -#define MSG_INVALID_GAMEGENIE_CODE 28 -#define MSG_INVALID_CHEAT_TO_REMOVE 29 -#define MSG_INVALID_CHEAT_CODE_ADDRESS 30 -#define MSG_UNSUPPORTED_CHEAT_LIST_VERSION 31 -#define MSG_UNSUPPORTED_CHEAT_LIST_TYPE 32 -#define MSG_INVALID_GSA_CODE 33 -#define MSG_CANNOT_IMPORT_SNAPSHOT_FOR 34 -#define MSG_UNSUPPORTED_SNAPSHOT_FILE 35 -#define MSG_UNSUPPORTED_ARM_MODE 36 -#define MSG_UNSUPPORTED_CODE_FILE 37 -#define MSG_GBA_CODE_WARNING 38 -#define MSG_INVALID_CBA_CODE 39 -#define MSG_CBA_CODE_WARNING 40 -#define MSG_OUT_OF_MEMORY 41 +// -*- C++ -*- +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#define N_(String) (String) + +#define MSG_UNSUPPORTED_VBA_SGM 1 +#define MSG_CANNOT_LOAD_SGM 2 +#define MSG_SAVE_GAME_NOT_USING_BIOS 3 +#define MSG_SAVE_GAME_USING_BIOS 4 +#define MSG_UNSUPPORTED_SAVE_TYPE 5 +#define MSG_CANNOT_OPEN_FILE 6 +#define MSG_BAD_ZIP_FILE 7 +#define MSG_NO_IMAGE_ON_ZIP 8 +#define MSG_ERROR_OPENING_IMAGE 9 +#define MSG_ERROR_READING_IMAGE 10 +#define MSG_UNSUPPORTED_BIOS_FUNCTION 11 +#define MSG_INVALID_BIOS_FILE_SIZE 12 +#define MSG_INVALID_CHEAT_CODE 13 +#define MSG_UNKNOWN_ARM_OPCODE 14 +#define MSG_UNKNOWN_THUMB_OPCODE 15 +#define MSG_ERROR_CREATING_FILE 16 +#define MSG_FAILED_TO_READ_SGM 17 +#define MSG_FAILED_TO_READ_RTC 18 +#define MSG_UNSUPPORTED_VB_SGM 19 +#define MSG_CANNOT_LOAD_SGM_FOR 20 +#define MSG_ERROR_OPENING_IMAGE_FROM 21 +#define MSG_ERROR_READING_IMAGE_FROM 22 +#define MSG_UNSUPPORTED_ROM_SIZE 23 +#define MSG_UNSUPPORTED_RAM_SIZE 24 +#define MSG_UNKNOWN_CARTRIDGE_TYPE 25 +#define MSG_MAXIMUM_NUMBER_OF_CHEATS 26 +#define MSG_INVALID_GAMESHARK_CODE 27 +#define MSG_INVALID_GAMEGENIE_CODE 28 +#define MSG_INVALID_CHEAT_TO_REMOVE 29 +#define MSG_INVALID_CHEAT_CODE_ADDRESS 30 +#define MSG_UNSUPPORTED_CHEAT_LIST_VERSION 31 +#define MSG_UNSUPPORTED_CHEAT_LIST_TYPE 32 +#define MSG_INVALID_GSA_CODE 33 +#define MSG_CANNOT_IMPORT_SNAPSHOT_FOR 34 +#define MSG_UNSUPPORTED_SNAPSHOT_FILE 35 +#define MSG_UNSUPPORTED_ARM_MODE 36 +#define MSG_UNSUPPORTED_CODE_FILE 37 +#define MSG_GBA_CODE_WARNING 38 +#define MSG_INVALID_CBA_CODE 39 +#define MSG_CBA_CODE_WARNING 40 +#define MSG_OUT_OF_MEMORY 41 +#define MSG_WRONG_GAMESHARK_CODE 42 +#define MSG_UNSUPPORTED_GAMESHARK_CODE 43 diff --git a/src/RTC.cpp b/src/RTC.cpp index 9a5b3098..9a69ecc5 100644 --- a/src/RTC.cpp +++ b/src/RTC.cpp @@ -1,220 +1,222 @@ -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#include "System.h" -#include "GBA.h" -#include "Globals.h" -#include "Port.h" -#include "Util.h" -#include "NLS.h" - -#include -#include - -enum RTCSTATE { IDLE, COMMAND, DATA, READDATA }; - -typedef struct { - u8 byte0; - u8 byte1; - u8 byte2; - u8 command; - int dataLen; - int bits; - RTCSTATE state; - u8 data[12]; - // reserved variables for future - u8 reserved[12]; - bool reserved2; - u32 reserved3; -} RTCCLOCKDATA; - -static RTCCLOCKDATA rtcClockData; -static bool rtcEnabled = false; - -void rtcEnable(bool e) -{ - rtcEnabled = e; -} - -bool rtcIsEnabled() -{ - return rtcEnabled; -} - -u16 rtcRead(u32 address) -{ - if(rtcEnabled) { - if(address == 0x80000c8) - return rtcClockData.byte2; - else if(address == 0x80000c6) - return rtcClockData.byte1; - else if(address == 0x80000c4) { - return rtcClockData.byte0; - } - } - - return READ16LE((&rom[address & 0x1FFFFFE])); -} - -static u8 toBCD(u8 value) -{ - value = value % 100; - int l = value % 10; - int h = value / 10; - return h * 16 + l; -} - -bool rtcWrite(u32 address, u16 value) -{ - if(!rtcEnabled) - return false; - - if(address == 0x80000c8) { - rtcClockData.byte2 = (u8)value; // enable ? - } else if(address == 0x80000c6) { - rtcClockData.byte1 = (u8)value; // read/write - } else if(address == 0x80000c4) { - if(rtcClockData.byte2 & 1) { - if(rtcClockData.state == IDLE && rtcClockData.byte0 == 1 && value == 5) { - rtcClockData.state = COMMAND; - rtcClockData.bits = 0; - rtcClockData.command = 0; - } else if(!(rtcClockData.byte0 & 1) && (value & 1)) { // bit transfer - rtcClockData.byte0 = (u8)value; - switch(rtcClockData.state) { - case COMMAND: - rtcClockData.command |= ((value & 2) >> 1) << (7-rtcClockData.bits); - rtcClockData.bits++; - if(rtcClockData.bits == 8) { - rtcClockData.bits = 0; - switch(rtcClockData.command) { - case 0x60: - // not sure what this command does but it doesn't take parameters - // maybe it is a reset or stop - rtcClockData.state = IDLE; - rtcClockData.bits = 0; - break; - case 0x62: - // this sets the control state but not sure what those values are - rtcClockData.state = READDATA; - rtcClockData.dataLen = 1; - break; - case 0x63: - rtcClockData.dataLen = 1; - rtcClockData.data[0] = 0x40; - rtcClockData.state = DATA; - break; - case 0x65: - { - struct tm *newtime; - time_t long_time; - - time( &long_time ); /* Get time as long integer. */ - newtime = localtime( &long_time ); /* Convert to local time. */ - - rtcClockData.dataLen = 7; - rtcClockData.data[0] = toBCD(newtime->tm_year); - rtcClockData.data[1] = toBCD(newtime->tm_mon+1); - rtcClockData.data[2] = toBCD(newtime->tm_mday); - rtcClockData.data[3] = toBCD(newtime->tm_wday); - rtcClockData.data[4] = toBCD(newtime->tm_hour); - rtcClockData.data[5] = toBCD(newtime->tm_min); - rtcClockData.data[6] = toBCD(newtime->tm_sec); - rtcClockData.state = DATA; - } - break; - case 0x67: - { - struct tm *newtime; - time_t long_time; - - time( &long_time ); /* Get time as long integer. */ - newtime = localtime( &long_time ); /* Convert to local time. */ - - rtcClockData.dataLen = 3; - rtcClockData.data[0] = toBCD(newtime->tm_hour); - rtcClockData.data[1] = toBCD(newtime->tm_min); - rtcClockData.data[2] = toBCD(newtime->tm_sec); - rtcClockData.state = DATA; - } - break; - default: - systemMessage(0, N_("Unknown RTC command %02x"), rtcClockData.command); - rtcClockData.state = IDLE; - break; - } - } - break; - case DATA: - if(rtcClockData.byte1 & 2) { - } else { - rtcClockData.byte0 = (rtcClockData.byte0 & ~2) | - ((rtcClockData.data[rtcClockData.bits >> 3] >> - (rtcClockData.bits & 7)) & 1)*2; - rtcClockData.bits++; - if(rtcClockData.bits == 8*rtcClockData.dataLen) { - rtcClockData.bits = 0; - rtcClockData.state = IDLE; - } - } - break; - case READDATA: - if(!(rtcClockData.byte1 & 2)) { - } else { - rtcClockData.data[rtcClockData.bits >> 3] = - (rtcClockData.data[rtcClockData.bits >> 3] >> 1) | - ((value << 6) & 128); - rtcClockData.bits++; - if(rtcClockData.bits == 8*rtcClockData.dataLen) { - rtcClockData.bits = 0; - rtcClockData.state = IDLE; - } - } - break; - default: - break; - } - } else - rtcClockData.byte0 = (u8)value; - } - } - return true; -} - -void rtcReset() -{ - memset(&rtcClockData, 0, sizeof(rtcClockData)); - - rtcClockData.byte0 = 0; - rtcClockData.byte1 = 0; - rtcClockData.byte2 = 0; - rtcClockData.command = 0; - rtcClockData.dataLen = 0; - rtcClockData.bits = 0; - rtcClockData.state = IDLE; -} - -void rtcSaveGame(gzFile gzFile) -{ - utilGzWrite(gzFile, &rtcClockData, sizeof(rtcClockData)); -} - -void rtcReadGame(gzFile gzFile) -{ - utilGzRead(gzFile, &rtcClockData, sizeof(rtcClockData)); -} \ No newline at end of file +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2005 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#include "System.h" +#include "GBA.h" +#include "Globals.h" +#include "Port.h" +#include "Util.h" +#include "NLS.h" + +#include +#include + +enum RTCSTATE { IDLE, COMMAND, DATA, READDATA }; + +typedef struct { + u8 byte0; + u8 byte1; + u8 byte2; + u8 command; + int dataLen; + int bits; + RTCSTATE state; + u8 data[12]; + // reserved variables for future + u8 reserved[12]; + bool reserved2; + u32 reserved3; +} RTCCLOCKDATA; + +static RTCCLOCKDATA rtcClockData; +static bool rtcEnabled = false; + +void rtcEnable(bool e) +{ + rtcEnabled = e; +} + +bool rtcIsEnabled() +{ + return rtcEnabled; +} + +u16 rtcRead(u32 address) +{ + if(rtcEnabled) { + if(address == 0x80000c8) + return rtcClockData.byte2; + else if(address == 0x80000c6) + return rtcClockData.byte1; + else if(address == 0x80000c4) { + return rtcClockData.byte0; + } + } + + return READ16LE((&rom[address & 0x1FFFFFE])); +} + +static u8 toBCD(u8 value) +{ + value = value % 100; + int l = value % 10; + int h = value / 10; + return h * 16 + l; +} + +bool rtcWrite(u32 address, u16 value) +{ + if(!rtcEnabled) + return false; + + if(address == 0x80000c8) { + rtcClockData.byte2 = (u8)value; // enable ? + } else if(address == 0x80000c6) { + rtcClockData.byte1 = (u8)value; // read/write + } else if(address == 0x80000c4) { + if(rtcClockData.byte2 & 1) { + if(rtcClockData.state == IDLE && rtcClockData.byte0 == 1 && value == 5) { + rtcClockData.state = COMMAND; + rtcClockData.bits = 0; + rtcClockData.command = 0; + } else if(!(rtcClockData.byte0 & 1) && (value & 1)) { // bit transfer + rtcClockData.byte0 = (u8)value; + switch(rtcClockData.state) { + case COMMAND: + rtcClockData.command |= ((value & 2) >> 1) << (7-rtcClockData.bits); + rtcClockData.bits++; + if(rtcClockData.bits == 8) { + rtcClockData.bits = 0; + switch(rtcClockData.command) { + case 0x60: + // not sure what this command does but it doesn't take parameters + // maybe it is a reset or stop + rtcClockData.state = IDLE; + rtcClockData.bits = 0; + break; + case 0x62: + // this sets the control state but not sure what those values are + rtcClockData.state = READDATA; + rtcClockData.dataLen = 1; + break; + case 0x63: + rtcClockData.dataLen = 1; + rtcClockData.data[0] = 0x40; + rtcClockData.state = DATA; + break; + case 0x64: + break; + case 0x65: + { + struct tm *newtime; + time_t long_time; + + time( &long_time ); /* Get time as long integer. */ + newtime = localtime( &long_time ); /* Convert to local time. */ + + rtcClockData.dataLen = 7; + rtcClockData.data[0] = toBCD(newtime->tm_year); + rtcClockData.data[1] = toBCD(newtime->tm_mon+1); + rtcClockData.data[2] = toBCD(newtime->tm_mday); + rtcClockData.data[3] = toBCD(newtime->tm_wday); + rtcClockData.data[4] = toBCD(newtime->tm_hour); + rtcClockData.data[5] = toBCD(newtime->tm_min); + rtcClockData.data[6] = toBCD(newtime->tm_sec); + rtcClockData.state = DATA; + } + break; + case 0x67: + { + struct tm *newtime; + time_t long_time; + + time( &long_time ); /* Get time as long integer. */ + newtime = localtime( &long_time ); /* Convert to local time. */ + + rtcClockData.dataLen = 3; + rtcClockData.data[0] = toBCD(newtime->tm_hour); + rtcClockData.data[1] = toBCD(newtime->tm_min); + rtcClockData.data[2] = toBCD(newtime->tm_sec); + rtcClockData.state = DATA; + } + break; + default: + systemMessage(0, N_("Unknown RTC command %02x"), rtcClockData.command); + rtcClockData.state = IDLE; + break; + } + } + break; + case DATA: + if(rtcClockData.byte1 & 2) { + } else { + rtcClockData.byte0 = (rtcClockData.byte0 & ~2) | + ((rtcClockData.data[rtcClockData.bits >> 3] >> + (rtcClockData.bits & 7)) & 1)*2; + rtcClockData.bits++; + if(rtcClockData.bits == 8*rtcClockData.dataLen) { + rtcClockData.bits = 0; + rtcClockData.state = IDLE; + } + } + break; + case READDATA: + if(!(rtcClockData.byte1 & 2)) { + } else { + rtcClockData.data[rtcClockData.bits >> 3] = + (rtcClockData.data[rtcClockData.bits >> 3] >> 1) | + ((value << 6) & 128); + rtcClockData.bits++; + if(rtcClockData.bits == 8*rtcClockData.dataLen) { + rtcClockData.bits = 0; + rtcClockData.state = IDLE; + } + } + break; + default: + break; + } + } else + rtcClockData.byte0 = (u8)value; + } + } + return true; +} + +void rtcReset() +{ + memset(&rtcClockData, 0, sizeof(rtcClockData)); + + rtcClockData.byte0 = 0; + rtcClockData.byte1 = 0; + rtcClockData.byte2 = 0; + rtcClockData.command = 0; + rtcClockData.dataLen = 0; + rtcClockData.bits = 0; + rtcClockData.state = IDLE; +} + +void rtcSaveGame(gzFile gzFile) +{ + utilGzWrite(gzFile, &rtcClockData, sizeof(rtcClockData)); +} + +void rtcReadGame(gzFile gzFile) +{ + utilGzRead(gzFile, &rtcClockData, sizeof(rtcClockData)); +} diff --git a/src/Sound.cpp b/src/Sound.cpp index ec623c8a..178dc82b 100644 --- a/src/Sound.cpp +++ b/src/Sound.cpp @@ -659,40 +659,17 @@ void soundChannel4() { } -#include - inline void soundDirectSoundA() { -#ifdef ENHANCED_RATE - double cr = calc_rate(soundDSATimer); - static int cnt = 0; - static double lastcr = 0; - static FILE *fp = NULL; - - if (fp==NULL) - fp=fopen("C:\\cr.txt", "at"); - if (cr!=lastcr) - { - fprintf(fp, "%f %d\n", lastcr, cnt); - cnt=0; - lastcr=cr; - } - else - cnt++; - - directBuffer[0][soundIndex] = interp_pop(0, calc_rate(soundDSATimer)); //soundDSAValue; -#else directBuffer[0][soundIndex] = interp_pop(0); //soundDSAValue; -#endif - } void soundDirectSoundATimer() { if(soundDSAEnabled) { if(soundDSFifoACount <= 16) { - cpuDmaHack2 = CPUCheckDMA(3, 2); + CPUCheckDMA(3, 2); if(soundDSFifoACount <= 16) { soundEvent(FIFOA_L, (u16)0); soundEvent(FIFOA_H, (u16)0); @@ -714,18 +691,15 @@ void soundDirectSoundATimer() inline void soundDirectSoundB() { -#ifdef ENHANCED_RATE - directBuffer[1][soundIndex] = interp_pop(1, calc_rate(soundDSBTimer)); //soundDSBValue; -#else directBuffer[1][soundIndex] = interp_pop(1); //soundDSBValue; -#endif } void soundDirectSoundBTimer() { if(soundDSBEnabled) { if(soundDSFifoBCount <= 16) { - cpuDmaHack2 = CPUCheckDMA(3, 4); + //cpuDmaHack2 = + CPUCheckDMA(3, 4); if(soundDSFifoBCount <= 16) { soundEvent(FIFOB_L, (u16)0); soundEvent(FIFOB_H, (u16)0); diff --git a/src/Sound.h b/src/Sound.h index d02d3838..ee808a34 100644 --- a/src/Sound.h +++ b/src/Sound.h @@ -2,6 +2,7 @@ // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten // Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2004-2006 VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -95,7 +96,6 @@ extern Multi_Buffer * apu_out; extern Gb_Apu * apu; extern const BOOST::uint8_t sound_data [Gb_Apu::register_count]; - extern void interp_rate(); diff --git a/src/Sram.cpp b/src/Sram.cpp index df582240..46e7108b 100644 --- a/src/Sram.cpp +++ b/src/Sram.cpp @@ -1,39 +1,39 @@ -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#include "GBA.h" -#include "Globals.h" -#include "Flash.h" -#include "Sram.h" - -u8 sramRead(u32 address) -{ - return flashSaveMemory[address & 0xFFFF]; -} -void sramDelayedWrite(u32 address, u8 byte) -{ - saveType = 1; - cpuSaveGameFunc = sramWrite; - sramWrite(address, byte); -} - -void sramWrite(u32 address, u8 byte) -{ - flashSaveMemory[address & 0xFFFF] = byte; - systemSaveUpdateCounter = SYSTEM_SAVE_UPDATED; -} \ No newline at end of file +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#include "GBA.h" +#include "Globals.h" +#include "Flash.h" +#include "Sram.h" + +u8 sramRead(u32 address) +{ + return flashSaveMemory[address & 0xFFFF]; +} +void sramDelayedWrite(u32 address, u8 byte) +{ + saveType = 1; + cpuSaveGameFunc = sramWrite; + sramWrite(address, byte); +} + +void sramWrite(u32 address, u8 byte) +{ + flashSaveMemory[address & 0xFFFF] = byte; + systemSaveUpdateCounter = SYSTEM_SAVE_UPDATED; +} diff --git a/src/Sram.h b/src/Sram.h index c8a90192..c4c7d257 100644 --- a/src/Sram.h +++ b/src/Sram.h @@ -1,27 +1,27 @@ -// -*- C++ -*- -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#ifndef VBA_SRAM_H -#define VBA_SRAM_H - -extern u8 sramRead(u32 address); -extern void sramWrite(u32 address, u8 byte); -extern void sramDelayedWrite(u32 address, u8 byte); - -#endif // VBA_SRAM_H \ No newline at end of file +// -*- C++ -*- +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#ifndef VBA_SRAM_H +#define VBA_SRAM_H + +extern u8 sramRead(u32 address); +extern void sramWrite(u32 address, u8 byte); +extern void sramDelayedWrite(u32 address, u8 byte); + +#endif // VBA_SRAM_H diff --git a/src/System.h b/src/System.h index 2cc6d495..21dc21cc 100644 --- a/src/System.h +++ b/src/System.h @@ -106,6 +106,12 @@ extern void system10Frames(int); extern void systemFrame(); extern void systemGbBorderOn(); +extern void Sm60FPS_Init(); +extern bool Sm60FPS_CanSkipFrame(); +extern void Sm60FPS_Sleep(); +extern void DbgMsg(const char *msg, ...); +extern void winlog(const char *,...); + extern bool systemSoundOn; extern u16 systemColorMap16[0x10000]; extern u32 systemColorMap32[0x10000]; @@ -118,6 +124,7 @@ extern int systemDebug; extern int systemVerbose; extern int systemFrameSkip; extern int systemSaveUpdateCounter; +extern int systemSpeed; #define SYSTEM_SAVE_UPDATED 30 #define SYSTEM_SAVE_NOT_UPDATED 0 diff --git a/src/Util.cpp b/src/Util.cpp index 328ed142..9e3b7d91 100644 --- a/src/Util.cpp +++ b/src/Util.cpp @@ -1,1115 +1,1102 @@ -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#include -#include -#include -#include - -extern "C" { -#include -} - -#if 0 -#include "unrarlib.h" -#endif - -#include "System.h" -#include "NLS.h" -#include "Util.h" -#include "Flash.h" -#include "GBA.h" -#include "Globals.h" -#include "RTC.h" -#include "Port.h" -#include "memgzio.h" -#include "gbafilter.h" - -#ifndef _MSC_VER -#define _stricmp strcasecmp -#endif // ! _MSC_VER - -extern int systemColorDepth; -extern int systemRedShift; -extern int systemGreenShift; -extern int systemBlueShift; - -extern u16 systemColorMap16[0x10000]; -extern u32 systemColorMap32[0x10000]; - -static int (ZEXPORT *utilGzWriteFunc)(gzFile, const voidp, unsigned int) = NULL; -static int (ZEXPORT *utilGzReadFunc)(gzFile, voidp, unsigned int) = NULL; -static int (ZEXPORT *utilGzCloseFunc)(gzFile) = NULL; - -bool utilWritePNGFile(const char *fileName, int w, int h, u8 *pix) -{ - u8 writeBuffer[512 * 3]; - - FILE *fp = fopen(fileName,"wb"); - - if(!fp) { - systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), fileName); - return false; - } - - png_structp png_ptr = png_create_write_struct(PNG_LIBPNG_VER_STRING, - NULL, - NULL, - NULL); - if(!png_ptr) { - fclose(fp); - return false; - } - - png_infop info_ptr = png_create_info_struct(png_ptr); - - if(!info_ptr) { - png_destroy_write_struct(&png_ptr,NULL); - fclose(fp); - return false; - } - - if(setjmp(png_ptr->jmpbuf)) { - png_destroy_write_struct(&png_ptr,NULL); - fclose(fp); - return false; - } - - png_init_io(png_ptr,fp); - - png_set_IHDR(png_ptr, - info_ptr, - w, - h, - 8, - PNG_COLOR_TYPE_RGB, - PNG_INTERLACE_NONE, - PNG_COMPRESSION_TYPE_DEFAULT, - PNG_FILTER_TYPE_DEFAULT); - - png_write_info(png_ptr,info_ptr); - - u8 *b = writeBuffer; - - int sizeX = w; - int sizeY = h; - - switch(systemColorDepth) { - case 16: - { - u16 *p = (u16 *)(pix+(w+2)*2); // skip first black line - for(int y = 0; y < sizeY; y++) { - for(int x = 0; x < sizeX; x++) { - u16 v = *p++; - - *b++ = ((v >> systemRedShift) & 0x001f) << 3; // R - *b++ = ((v >> systemGreenShift) & 0x001f) << 3; // G - *b++ = ((v >> systemBlueShift) & 0x01f) << 3; // B - } - p++; // skip black pixel for filters - p++; // skip black pixel for filters - png_write_row(png_ptr,writeBuffer); - - b = writeBuffer; - } - } - break; - case 24: - { - u8 *pixU8 = (u8 *)pix; - for(int y = 0; y < sizeY; y++) { - for(int x = 0; x < sizeX; x++) { - if(systemRedShift < systemBlueShift) { - *b++ = *pixU8++; // R - *b++ = *pixU8++; // G - *b++ = *pixU8++; // B - } else { - int blue = *pixU8++; - int green = *pixU8++; - int red = *pixU8++; - - *b++ = red; - *b++ = green; - *b++ = blue; - } - } - png_write_row(png_ptr,writeBuffer); - - b = writeBuffer; - } - } - break; - case 32: - { - u32 *pixU32 = (u32 *)(pix+4*(w+1)); - for(int y = 0; y < sizeY; y++) { - for(int x = 0; x < sizeX; x++) { - u32 v = *pixU32++; - - *b++ = ((v >> systemRedShift) & 0x001f) << 3; // R - *b++ = ((v >> systemGreenShift) & 0x001f) << 3; // G - *b++ = ((v >> systemBlueShift) & 0x001f) << 3; // B - } - pixU32++; - - png_write_row(png_ptr,writeBuffer); - - b = writeBuffer; - } - } - break; - } - - png_write_end(png_ptr, info_ptr); - - png_destroy_write_struct(&png_ptr, &info_ptr); - - fclose(fp); - - return true; -} - -void utilPutDword(u8 *p, u32 value) -{ - *p++ = value & 255; - *p++ = (value >> 8) & 255; - *p++ = (value >> 16) & 255; - *p = (value >> 24) & 255; -} - -void utilPutWord(u8 *p, u16 value) -{ - *p++ = value & 255; - *p = (value >> 8) & 255; -} - -void utilWriteBMP(char *buf, int w, int h, u8 *pix) -{ - u8 *b = (u8 *)buf; - - int sizeX = w; - int sizeY = h; - - switch(systemColorDepth) { - case 16: - { - u16 *p = (u16 *)(pix+(w+2)*(h)*2); // skip first black line - for(int y = 0; y < sizeY; y++) { - for(int x = 0; x < sizeX; x++) { - u16 v = *p++; - - *b++ = ((v >> systemBlueShift) & 0x01f) << 3; // B - *b++ = ((v >> systemGreenShift) & 0x001f) << 3; // G - *b++ = ((v >> systemRedShift) & 0x001f) << 3; // R - } - p++; // skip black pixel for filters - p++; // skip black pixel for filters - p -= 2*(w+2); - } - } - break; - case 24: - { - u8 *pixU8 = (u8 *)pix+3*w*(h-1); - for(int y = 0; y < sizeY; y++) { - for(int x = 0; x < sizeX; x++) { - if(systemRedShift > systemBlueShift) { - *b++ = *pixU8++; // B - *b++ = *pixU8++; // G - *b++ = *pixU8++; // R - } else { - int red = *pixU8++; - int green = *pixU8++; - int blue = *pixU8++; - - *b++ = blue; - *b++ = green; - *b++ = red; - } - } - pixU8 -= 2*3*w; - } - } - break; - case 32: - { - u32 *pixU32 = (u32 *)(pix+4*(w+1)*(h)); - for(int y = 0; y < sizeY; y++) { - for(int x = 0; x < sizeX; x++) { - u32 v = *pixU32++; - - *b++ = ((v >> systemBlueShift) & 0x001f) << 3; // B - *b++ = ((v >> systemGreenShift) & 0x001f) << 3; // G - *b++ = ((v >> systemRedShift) & 0x001f) << 3; // R - } - pixU32++; - pixU32 -= 2*(w+1); - } - } - break; - } -} - -bool utilWriteBMPFile(const char *fileName, int w, int h, u8 *pix) -{ - u8 writeBuffer[512 * 3]; - - FILE *fp = fopen(fileName,"wb"); - - if(!fp) { - systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), fileName); - return false; - } - - struct { - u8 ident[2]; - u8 filesize[4]; - u8 reserved[4]; - u8 dataoffset[4]; - u8 headersize[4]; - u8 width[4]; - u8 height[4]; - u8 planes[2]; - u8 bitsperpixel[2]; - u8 compression[4]; - u8 datasize[4]; - u8 hres[4]; - u8 vres[4]; - u8 colors[4]; - u8 importantcolors[4]; - // u8 pad[2]; - } bmpheader; - memset(&bmpheader, 0, sizeof(bmpheader)); - - bmpheader.ident[0] = 'B'; - bmpheader.ident[1] = 'M'; - - u32 fsz = sizeof(bmpheader) + w*h*3; - utilPutDword(bmpheader.filesize, fsz); - utilPutDword(bmpheader.dataoffset, 0x36); - utilPutDword(bmpheader.headersize, 0x28); - utilPutDword(bmpheader.width, w); - utilPutDword(bmpheader.height, h); - utilPutDword(bmpheader.planes, 1); - utilPutDword(bmpheader.bitsperpixel, 24); - utilPutDword(bmpheader.datasize, 3*w*h); - - fwrite(&bmpheader, 1, sizeof(bmpheader), fp); - - u8 *b = writeBuffer; - - int sizeX = w; - int sizeY = h; - - switch(systemColorDepth) { - case 16: - { - u16 *p = (u16 *)(pix+(w+2)*(h)*2); // skip first black line - for(int y = 0; y < sizeY; y++) { - for(int x = 0; x < sizeX; x++) { - u16 v = *p++; - - *b++ = ((v >> systemBlueShift) & 0x01f) << 3; // B - *b++ = ((v >> systemGreenShift) & 0x001f) << 3; // G - *b++ = ((v >> systemRedShift) & 0x001f) << 3; // R - } - p++; // skip black pixel for filters - p++; // skip black pixel for filters - p -= 2*(w+2); - fwrite(writeBuffer, 1, 3*w, fp); - - b = writeBuffer; - } - } - break; - case 24: - { - u8 *pixU8 = (u8 *)pix+3*w*(h-1); - for(int y = 0; y < sizeY; y++) { - for(int x = 0; x < sizeX; x++) { - if(systemRedShift > systemBlueShift) { - *b++ = *pixU8++; // B - *b++ = *pixU8++; // G - *b++ = *pixU8++; // R - } else { - int red = *pixU8++; - int green = *pixU8++; - int blue = *pixU8++; - - *b++ = blue; - *b++ = green; - *b++ = red; - } - } - pixU8 -= 2*3*w; - fwrite(writeBuffer, 1, 3*w, fp); - - b = writeBuffer; - } - } - break; - case 32: - { - u32 *pixU32 = (u32 *)(pix+4*(w+1)*(h)); - for(int y = 0; y < sizeY; y++) { - for(int x = 0; x < sizeX; x++) { - u32 v = *pixU32++; - - *b++ = ((v >> systemBlueShift) & 0x001f) << 3; // B - *b++ = ((v >> systemGreenShift) & 0x001f) << 3; // G - *b++ = ((v >> systemRedShift) & 0x001f) << 3; // R - } - pixU32++; - pixU32 -= 2*(w+1); - - fwrite(writeBuffer, 1, 3*w, fp); - - b = writeBuffer; - } - } - break; - } - - fclose(fp); - - return true; -} - -static int utilReadInt2(FILE *f) -{ - int res = 0; - int c = fgetc(f); - if(c == EOF) - return -1; - res = c; - c = fgetc(f); - if(c == EOF) - return -1; - return c + (res<<8); -} - -static int utilReadInt3(FILE *f) -{ - int res = 0; - int c = fgetc(f); - if(c == EOF) - return -1; - res = c; - c = fgetc(f); - if(c == EOF) - return -1; - res = c + (res<<8); - c = fgetc(f); - if(c == EOF) - return -1; - return c + (res<<8); -} - -void utilApplyIPS(const char *ips, u8 **r, int *s) -{ - // from the IPS spec at http://zerosoft.zophar.net/ips.htm - FILE *f = fopen(ips, "rb"); - if(!f) - return; - u8 *rom = *r; - int size = *s; - if(fgetc(f) == 'P' && - fgetc(f) == 'A' && - fgetc(f) == 'T' && - fgetc(f) == 'C' && - fgetc(f) == 'H') { - int b; - int offset; - int len; - for(;;) { - // read offset - offset = utilReadInt3(f); - // if offset == EOF, end of patch - if(offset == 0x454f46) - break; - // read length - len = utilReadInt2(f); - if(!len) { - // len == 0, RLE block - len = utilReadInt2(f); - // byte to fill - int c = fgetc(f); - if(c == -1) - break; - b = (u8)c; - } else - b= -1; - // check if we need to reallocate our ROM - if((offset + len) >= size) { - size *= 2; - rom = (u8 *)realloc(rom, size); - *r = rom; - *s = size; - } - if(b == -1) { - // normal block, just read the data - if(fread(&rom[offset], 1, len, f) != (size_t)len) - break; - } else { - // fill the region with the given byte - while(len--) { - rom[offset++] = b; - } - } - } - } - // close the file - fclose(f); -} - -extern bool cpuIsMultiBoot; - -bool utilIsGBAImage(const char * file) -{ - cpuIsMultiBoot = false; - if(strlen(file) > 4) { - char * p = (char *)strrchr(file,'.'); - - if(p != NULL) { - if(_stricmp(p, ".gba") == 0) - return true; - if(_stricmp(p, ".agb") == 0) - return true; - if(_stricmp(p, ".bin") == 0) - return true; - if(_stricmp(p, ".elf") == 0) - return true; - if(_stricmp(p, ".mb") == 0) { - cpuIsMultiBoot = true; - return true; - } - } - } - - return false; -} - -bool utilIsGBImage(const char * file) -{ - if(strlen(file) > 4) { - char * p = (char *)strrchr(file,'.'); - - if(p != NULL) { - if(_stricmp(p, ".gb") == 0) - return true; - if(_stricmp(p, ".gbc") == 0) - return true; - if(_stricmp(p, ".cgb") == 0) - return true; - if(_stricmp(p, ".sgb") == 0) - return true; - } - } - - return false; -} - -bool utilIsZipFile(const char *file) -{ - if(strlen(file) > 4) { - char * p = (char *)strrchr(file,'.'); - - if(p != NULL) { - if(_stricmp(p, ".zip") == 0) - return true; - } - } - - return false; -} - -#if 0 -bool utilIsRarFile(const char *file) -{ - if(strlen(file) > 4) { - const char * p = strrchr(file,'.'); - - if(p != NULL) { - if(_stricmp(p, ".rar") == 0) - return true; - } - } - - return false; -} -#endif - -bool utilIs7ZipFile(const char *file) -{ - if(strlen(file) > 3) { - const char * p = strrchr(file,'.'); - - if(p != NULL) { - if(_stricmp(p, ".7z") == 0) - return true; - } - } - - return false; -} - -bool utilIsGzipFile(const char *file) -{ - if(strlen(file) > 3) { - char * p = (char *)strrchr(file,'.'); - - if(p != NULL) { - if(_stricmp(p, ".gz") == 0) - return true; - if(_stricmp(p, ".z") == 0) - return true; - } - } - - return false; -} - -void utilGetBaseName(const char *file, char *buffer) -{ - strcpy(buffer, file); - - if(utilIsGzipFile(file)) { - char *p = strrchr(buffer, '.'); - - if(p) - *p = 0; - } -} - -IMAGE_TYPE utilFindType(const char *file) -{ - char buffer[2048]; - - if(utilIsZipFile(file)) { - unzFile unz = unzOpen(file); - - if(unz == NULL) { - systemMessage(MSG_CANNOT_OPEN_FILE, N_("Cannot open file %s"), file); - return IMAGE_UNKNOWN; - } - - int r = unzGoToFirstFile(unz); - - if(r != UNZ_OK) { - unzClose(unz); - systemMessage(MSG_BAD_ARCHIVE_FILE, N_("Bad ZIP file %s"), file); - return IMAGE_UNKNOWN; - } - - IMAGE_TYPE found = IMAGE_UNKNOWN; - - unz_file_info info; - - while(true) { - r = unzGetCurrentFileInfo(unz, - &info, - buffer, - sizeof(buffer), - NULL, - 0, - NULL, - 0); - - if(r != UNZ_OK) { - unzClose(unz); - systemMessage(MSG_BAD_ARCHIVE_FILE, N_("Bad ZIP file %s"), file); - return IMAGE_UNKNOWN; - } - - if(utilIsGBAImage(buffer)) { - found = IMAGE_GBA; - break; - } - - if(utilIsGBImage(buffer)) { - found = IMAGE_GB; - break; - } - - r = unzGoToNextFile(unz); - - if(r != UNZ_OK) - break; - } - unzClose(unz); - - if(found == IMAGE_UNKNOWN) { - systemMessage(MSG_NO_IMAGE_ON_ARCHIVE, - N_("No image found on ZIP file %s"), file); - return found; - } - return found; -#if 0 - } else if(utilIsRarFile(file)) { - IMAGE_TYPE found = IMAGE_UNKNOWN; - - ArchiveList_struct *rarList = NULL; - if(urarlib_list((void *)file, (ArchiveList_struct *)&rarList)) { - ArchiveList_struct *p = rarList; - - while(p) { - if(utilIsGBAImage(p->item.Name)) { - found = IMAGE_GBA; - break; - } - - if(utilIsGBImage(p->item.Name)) { - found = IMAGE_GB; - break; - } - p = p->next; - } - - urarlib_freelist(rarList); - } - return found; -#endif - } else { - if(utilIsGzipFile(file)) - utilGetBaseName(file, buffer); - else - strcpy(buffer, file); - - if(utilIsGBAImage(buffer)) - return IMAGE_GBA; - if(utilIsGBImage(buffer)) - return IMAGE_GB; - } - return IMAGE_UNKNOWN; -} - -static int utilGetSize(int size) -{ - int res = 1; - while(res < size) - res <<= 1; - return res; -} - -static u8 *utilLoadFromZip(const char *file, - bool (*accept)(const char *), - u8 *data, - int &size) -{ - char buffer[2048]; - - unzFile unz = unzOpen(file); - - if(unz == NULL) { - systemMessage(MSG_CANNOT_OPEN_FILE, N_("Cannot open file %s"), file); - return NULL; - } - int r = unzGoToFirstFile(unz); - - if(r != UNZ_OK) { - unzClose(unz); - systemMessage(MSG_BAD_ARCHIVE_FILE, N_("Bad ZIP file %s"), file); - return NULL; - } - - bool found = false; - - unz_file_info info; - - while(true) { - r = unzGetCurrentFileInfo(unz, - &info, - buffer, - sizeof(buffer), - NULL, - 0, - NULL, - 0); - - if(r != UNZ_OK) { - unzClose(unz); - systemMessage(MSG_BAD_ARCHIVE_FILE, N_("Bad ZIP file %s"), file); - return NULL; - } - - if(accept(buffer)) { - found = true; - break; - } - - r = unzGoToNextFile(unz); - - if(r != UNZ_OK) - break; - } - - if(!found) { - unzClose(unz); - systemMessage(MSG_NO_IMAGE_ON_ARCHIVE, - N_("No image found on ZIP file %s"), file); - return NULL; - } - - int fileSize = info.uncompressed_size; - if(size == 0) - size = fileSize; - r = unzOpenCurrentFile(unz); - - if(r != UNZ_OK) { - unzClose(unz); - systemMessage(MSG_ERROR_OPENING_IMAGE, N_("Error opening image %s"), buffer); - return NULL; - } - - u8 *image = data; - - if(image == NULL) { - image = (u8 *)malloc(utilGetSize(size)); - if(image == NULL) { - unzCloseCurrentFile(unz); - unzClose(unz); - systemMessage(MSG_OUT_OF_MEMORY, N_("Failed to allocate memory for %s"), - "data"); - return NULL; - } - size = fileSize; - } - int read = fileSize <= size ? fileSize : size; - r = unzReadCurrentFile(unz, - image, - read); - - unzCloseCurrentFile(unz); - unzClose(unz); - - if(r != (int)read) { - systemMessage(MSG_ERROR_READING_IMAGE, - N_("Error reading image %s"), buffer); - if(data == NULL) - free(image); - return NULL; - } - - size = fileSize; - - return image; -} - -static u8 *utilLoadGzipFile(const char *file, - bool (*accept)(const char *), - u8 *data, - int &size) -{ - FILE *f = fopen(file, "rb"); - - if(f == NULL) { - systemMessage(MSG_ERROR_OPENING_IMAGE, N_("Error opening image %s"), file); - return NULL; - } - - fseek(f, -4, SEEK_END); - int fileSize = fgetc(f) | (fgetc(f) << 8) | (fgetc(f) << 16) | (fgetc(f) << 24); - fclose(f); - if(size == 0) - size = fileSize; - - gzFile gz = gzopen(file, "rb"); - - if(gz == NULL) { - // should not happen, but who knows? - systemMessage(MSG_ERROR_OPENING_IMAGE, N_("Error opening image %s"), file); - return NULL; - } - - u8 *image = data; - - if(image == NULL) { - image = (u8 *)malloc(utilGetSize(size)); - if(image == NULL) { - systemMessage(MSG_OUT_OF_MEMORY, N_("Failed to allocate memory for %s"), - "data"); - fclose(f); - return NULL; - } - size = fileSize; - } - int read = fileSize <= size ? fileSize : size; - int r = gzread(gz, image, read); - gzclose(gz); - - if(r != (int)read) { - systemMessage(MSG_ERROR_READING_IMAGE, - N_("Error reading image %s"), file); - if(data == NULL) - free(image); - return NULL; - } - - size = fileSize; - - return image; -} - -#if 0 -static u8 *utilLoadRarFile(const char *file, - bool (*accept)(const char *), - u8 *data, - int &size) -{ - char buffer[2048]; - - ArchiveList_struct *rarList = NULL; - if(urarlib_list((void *)file, (ArchiveList_struct *)&rarList)) { - ArchiveList_struct *p = rarList; - - bool found = false; - while(p) { - if(accept(p->item.Name)) { - strcpy(buffer, p->item.Name); - found = true; - break; - } - p = p->next; - } - if(found) { - void *memory = NULL; - unsigned long lsize = 0; - size = p->item.UnpSize; - int r = urarlib_get((void *)&memory, &lsize, buffer, (void *)file, ""); - if(!r) { - systemMessage(MSG_ERROR_READING_IMAGE, - N_("Error reading image %s"), buffer); - urarlib_freelist(rarList); - return NULL; - } - u8 *image = (u8 *)memory; - if(data != NULL) { - memcpy(image, data, size); - } - urarlib_freelist(rarList); - return image; - } - systemMessage(MSG_NO_IMAGE_ON_ZIP, - N_("No image found on RAR file %s"), file); - urarlib_freelist(rarList); - return NULL; - } - // nothing found - return NULL; -} -#endif - -u8 *utilLoad(const char *file, - bool (*accept)(const char *), - u8 *data, - int &size) -{ - if(utilIsZipFile(file)) { - return utilLoadFromZip(file, accept, data, size); - } - if(utilIsGzipFile(file)) { - return utilLoadGzipFile(file, accept, data, size); - } -#if 0 - if(utilIsRarFile(file)) { - return utilLoadRarFile(file, accept, data, size); - } -#endif - - u8 *image = data; - - FILE *f = fopen(file, "rb"); - - if(!f) { - systemMessage(MSG_ERROR_OPENING_IMAGE, N_("Error opening image %s"), file); - return NULL; - } - - fseek(f,0,SEEK_END); - int fileSize = ftell(f); - fseek(f,0,SEEK_SET); - if(size == 0) - size = fileSize; - - if(image == NULL) { - image = (u8 *)malloc(utilGetSize(size)); - if(image == NULL) { - systemMessage(MSG_OUT_OF_MEMORY, N_("Failed to allocate memory for %s"), - "data"); - fclose(f); - return NULL; - } - size = fileSize; - } - int read = fileSize <= size ? fileSize : size; - int r = fread(image, 1, read, f); - fclose(f); - - if(r != (int)read) { - systemMessage(MSG_ERROR_READING_IMAGE, - N_("Error reading image %s"), file); - if(data == NULL) - free(image); - return NULL; - } - - size = fileSize; - - return image; -} - -void utilWriteInt(gzFile gzFile, int i) -{ - utilGzWrite(gzFile, &i, sizeof(int)); -} - -int utilReadInt(gzFile gzFile) -{ - int i = 0; - utilGzRead(gzFile, &i, sizeof(int)); - return i; -} - -void utilReadData(gzFile gzFile, variable_desc* data) -{ - while(data->address) { - utilGzRead(gzFile, data->address, data->size); - data++; - } -} - -void utilWriteData(gzFile gzFile, variable_desc *data) -{ - while(data->address) { - utilGzWrite(gzFile, data->address, data->size); - data++; - } -} - -gzFile utilGzOpen(const char *file, const char *mode) -{ - utilGzWriteFunc = (int (ZEXPORT *)(void *,void * const, unsigned int))gzwrite; - utilGzReadFunc = gzread; - utilGzCloseFunc = gzclose; - - return gzopen(file, mode); -} - -gzFile utilMemGzOpen(char *memory, int available, char *mode) -{ - utilGzWriteFunc = memgzwrite; - utilGzReadFunc = memgzread; - utilGzCloseFunc = memgzclose; - - return memgzopen(memory, available, mode); -} - -int utilGzWrite(gzFile file, const voidp buffer, unsigned int len) -{ - return utilGzWriteFunc(file, buffer, len); -} - -int utilGzRead(gzFile file, voidp buffer, unsigned int len) -{ - return utilGzReadFunc(file, buffer, len); -} - -int utilGzClose(gzFile file) -{ - return utilGzCloseFunc(file); -} - -long utilGzMemTell(gzFile file) -{ - return memtell(file); -} - -void utilGBAFindSave(const u8 *data, const int size) -{ - u32 *p = (u32 *)data; - u32 *end = (u32 *)(data + size); - int saveType = 0; - int flashSize = 0x10000; - bool rtcFound = false; - - while(p < end) { - u32 d = READ32LE(p); - - if(d == 0x52504545) { - if(memcmp(p, "EEPROM_", 7) == 0) { - if(saveType == 0) - saveType = 1; - } - } else if (d == 0x4D415253) { - if(memcmp(p, "SRAM_", 5) == 0) { - if(saveType == 0) - saveType = 2; - } - } else if (d == 0x53414C46) { - if(memcmp(p, "FLASH1M_", 8) == 0) { - if(saveType == 0) { - saveType = 3; - flashSize = 0x20000; - } - } else if(memcmp(p, "FLASH", 5) == 0) { - if(saveType == 0) { - saveType = 3; - flashSize = 0x10000; - } - } - } else if (d == 0x52494953) { - if(memcmp(p, "SIIRTC_V", 8) == 0) - rtcFound = true; - } - p++; - } - // if no matches found, then set it to NONE - if(saveType == 0) { - saveType = 5; - } - rtcEnable(rtcFound); - cpuSaveType = saveType; - flashSetSize(flashSize); -} - -void utilUpdateSystemColorMaps(int lcd) -{ - switch(systemColorDepth) { - case 16: - { - for(int i = 0; i < 0x10000; i++) { - systemColorMap16[i] = ((i & 0x1f) << systemRedShift) | - (((i & 0x3e0) >> 5) << systemGreenShift) | - (((i & 0x7c00) >> 10) << systemBlueShift); - } - if (lcd == 1) gbafilter_pal(systemColorMap16, 0x10000); - } - break; - case 24: - case 32: - { - for(int i = 0; i < 0x10000; i++) { - systemColorMap32[i] = ((i & 0x1f) << systemRedShift) | - (((i & 0x3e0) >> 5) << systemGreenShift) | - (((i & 0x7c00) >> 10) << systemBlueShift); - } - if (lcd == 1) gbafilter_pal32(systemColorMap32, 0x10000); - } - break; - } -} \ No newline at end of file +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004-2006 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#include +#include +#include +#include + +extern "C" { +#include +} + +#if 0 +#include "unrarlib.h" +#endif + +#include "System.h" +#include "NLS.h" +#include "Util.h" +#include "Flash.h" +#include "GBA.h" +#include "Globals.h" +#include "RTC.h" +#include "Port.h" + + +extern "C" { +#include "memgzio.h" +} + +#ifndef _MSC_VER +#define _stricmp strcasecmp +#endif // ! _MSC_VER + +extern int systemColorDepth; +extern int systemRedShift; +extern int systemGreenShift; +extern int systemBlueShift; + +extern u16 systemColorMap16[0x10000]; +extern u32 systemColorMap32[0x10000]; + +static int (ZEXPORT *utilGzWriteFunc)(gzFile, const voidp, unsigned int) = NULL; +static int (ZEXPORT *utilGzReadFunc)(gzFile, voidp, unsigned int) = NULL; +static int (ZEXPORT *utilGzCloseFunc)(gzFile) = NULL; + +bool utilWritePNGFile(const char *fileName, int w, int h, u8 *pix) +{ + u8 writeBuffer[512 * 3]; + + FILE *fp = fopen(fileName,"wb"); + + if(!fp) { + systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), fileName); + return false; + } + + png_structp png_ptr = png_create_write_struct(PNG_LIBPNG_VER_STRING, + NULL, + NULL, + NULL); + if(!png_ptr) { + fclose(fp); + return false; + } + + png_infop info_ptr = png_create_info_struct(png_ptr); + + if(!info_ptr) { + png_destroy_write_struct(&png_ptr,NULL); + fclose(fp); + return false; + } + + if(setjmp(png_ptr->jmpbuf)) { + png_destroy_write_struct(&png_ptr,NULL); + fclose(fp); + return false; + } + + png_init_io(png_ptr,fp); + + png_set_IHDR(png_ptr, + info_ptr, + w, + h, + 8, + PNG_COLOR_TYPE_RGB, + PNG_INTERLACE_NONE, + PNG_COMPRESSION_TYPE_DEFAULT, + PNG_FILTER_TYPE_DEFAULT); + + png_write_info(png_ptr,info_ptr); + + u8 *b = writeBuffer; + + int sizeX = w; + int sizeY = h; + + switch(systemColorDepth) { + case 16: + { + u16 *p = (u16 *)(pix+(w+2)*2); // skip first black line + for(int y = 0; y < sizeY; y++) { + for(int x = 0; x < sizeX; x++) { + u16 v = *p++; + + *b++ = ((v >> systemRedShift) & 0x001f) << 3; // R + *b++ = ((v >> systemGreenShift) & 0x001f) << 3; // G + *b++ = ((v >> systemBlueShift) & 0x01f) << 3; // B + } + p++; // skip black pixel for filters + p++; // skip black pixel for filters + png_write_row(png_ptr,writeBuffer); + + b = writeBuffer; + } + } + break; + case 24: + { + u8 *pixU8 = (u8 *)pix; + for(int y = 0; y < sizeY; y++) { + for(int x = 0; x < sizeX; x++) { + if(systemRedShift < systemBlueShift) { + *b++ = *pixU8++; // R + *b++ = *pixU8++; // G + *b++ = *pixU8++; // B + } else { + int blue = *pixU8++; + int green = *pixU8++; + int red = *pixU8++; + + *b++ = red; + *b++ = green; + *b++ = blue; + } + } + png_write_row(png_ptr,writeBuffer); + + b = writeBuffer; + } + } + break; + case 32: + { + u32 *pixU32 = (u32 *)(pix+4*(w+1)); + for(int y = 0; y < sizeY; y++) { + for(int x = 0; x < sizeX; x++) { + u32 v = *pixU32++; + + *b++ = ((v >> systemRedShift) & 0x001f) << 3; // R + *b++ = ((v >> systemGreenShift) & 0x001f) << 3; // G + *b++ = ((v >> systemBlueShift) & 0x001f) << 3; // B + } + pixU32++; + + png_write_row(png_ptr,writeBuffer); + + b = writeBuffer; + } + } + break; + } + + png_write_end(png_ptr, info_ptr); + + png_destroy_write_struct(&png_ptr, &info_ptr); + + fclose(fp); + + return true; +} + +void utilPutDword(u8 *p, u32 value) +{ + *p++ = value & 255; + *p++ = (value >> 8) & 255; + *p++ = (value >> 16) & 255; + *p = (value >> 24) & 255; +} + +void utilPutWord(u8 *p, u16 value) +{ + *p++ = value & 255; + *p = (value >> 8) & 255; +} + +void utilWriteBMP(char *buf, int w, int h, u8 *pix) +{ + u8 *b = (u8 *)buf; + + int sizeX = w; + int sizeY = h; + + switch(systemColorDepth) { + case 16: + { + u16 *p = (u16 *)(pix+(w+2)*(h)*2); // skip first black line + for(int y = 0; y < sizeY; y++) { + for(int x = 0; x < sizeX; x++) { + u16 v = *p++; + + *b++ = ((v >> systemBlueShift) & 0x01f) << 3; // B + *b++ = ((v >> systemGreenShift) & 0x001f) << 3; // G + *b++ = ((v >> systemRedShift) & 0x001f) << 3; // R + } + p++; // skip black pixel for filters + p++; // skip black pixel for filters + p -= 2*(w+2); + } + } + break; + case 24: + { + u8 *pixU8 = (u8 *)pix+3*w*(h-1); + for(int y = 0; y < sizeY; y++) { + for(int x = 0; x < sizeX; x++) { + if(systemRedShift > systemBlueShift) { + *b++ = *pixU8++; // B + *b++ = *pixU8++; // G + *b++ = *pixU8++; // R + } else { + int red = *pixU8++; + int green = *pixU8++; + int blue = *pixU8++; + + *b++ = blue; + *b++ = green; + *b++ = red; + } + } + pixU8 -= 2*3*w; + } + } + break; + case 32: + { + u32 *pixU32 = (u32 *)(pix+4*(w+1)*(h)); + for(int y = 0; y < sizeY; y++) { + for(int x = 0; x < sizeX; x++) { + u32 v = *pixU32++; + + *b++ = ((v >> systemBlueShift) & 0x001f) << 3; // B + *b++ = ((v >> systemGreenShift) & 0x001f) << 3; // G + *b++ = ((v >> systemRedShift) & 0x001f) << 3; // R + } + pixU32++; + pixU32 -= 2*(w+1); + } + } + break; + } +} + +bool utilWriteBMPFile(const char *fileName, int w, int h, u8 *pix) +{ + u8 writeBuffer[512 * 3]; + + FILE *fp = fopen(fileName,"wb"); + + if(!fp) { + systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), fileName); + return false; + } + + struct { + u8 ident[2]; + u8 filesize[4]; + u8 reserved[4]; + u8 dataoffset[4]; + u8 headersize[4]; + u8 width[4]; + u8 height[4]; + u8 planes[2]; + u8 bitsperpixel[2]; + u8 compression[4]; + u8 datasize[4]; + u8 hres[4]; + u8 vres[4]; + u8 colors[4]; + u8 importantcolors[4]; + // u8 pad[2]; + } bmpheader; + memset(&bmpheader, 0, sizeof(bmpheader)); + + bmpheader.ident[0] = 'B'; + bmpheader.ident[1] = 'M'; + + u32 fsz = sizeof(bmpheader) + w*h*3; + utilPutDword(bmpheader.filesize, fsz); + utilPutDword(bmpheader.dataoffset, 0x36); + utilPutDword(bmpheader.headersize, 0x28); + utilPutDword(bmpheader.width, w); + utilPutDword(bmpheader.height, h); + utilPutDword(bmpheader.planes, 1); + utilPutDword(bmpheader.bitsperpixel, 24); + utilPutDword(bmpheader.datasize, 3*w*h); + + fwrite(&bmpheader, 1, sizeof(bmpheader), fp); + + u8 *b = writeBuffer; + + int sizeX = w; + int sizeY = h; + + switch(systemColorDepth) { + case 16: + { + u16 *p = (u16 *)(pix+(w+2)*(h)*2); // skip first black line + for(int y = 0; y < sizeY; y++) { + for(int x = 0; x < sizeX; x++) { + u16 v = *p++; + + *b++ = ((v >> systemBlueShift) & 0x01f) << 3; // B + *b++ = ((v >> systemGreenShift) & 0x001f) << 3; // G + *b++ = ((v >> systemRedShift) & 0x001f) << 3; // R + } + p++; // skip black pixel for filters + p++; // skip black pixel for filters + p -= 2*(w+2); + fwrite(writeBuffer, 1, 3*w, fp); + + b = writeBuffer; + } + } + break; + case 24: + { + u8 *pixU8 = (u8 *)pix+3*w*(h-1); + for(int y = 0; y < sizeY; y++) { + for(int x = 0; x < sizeX; x++) { + if(systemRedShift > systemBlueShift) { + *b++ = *pixU8++; // B + *b++ = *pixU8++; // G + *b++ = *pixU8++; // R + } else { + int red = *pixU8++; + int green = *pixU8++; + int blue = *pixU8++; + + *b++ = blue; + *b++ = green; + *b++ = red; + } + } + pixU8 -= 2*3*w; + fwrite(writeBuffer, 1, 3*w, fp); + + b = writeBuffer; + } + } + break; + case 32: + { + u32 *pixU32 = (u32 *)(pix+4*(w+1)*(h)); + for(int y = 0; y < sizeY; y++) { + for(int x = 0; x < sizeX; x++) { + u32 v = *pixU32++; + + *b++ = ((v >> systemBlueShift) & 0x001f) << 3; // B + *b++ = ((v >> systemGreenShift) & 0x001f) << 3; // G + *b++ = ((v >> systemRedShift) & 0x001f) << 3; // R + } + pixU32++; + pixU32 -= 2*(w+1); + + fwrite(writeBuffer, 1, 3*w, fp); + + b = writeBuffer; + } + } + break; + } + + fclose(fp); + + return true; +} + +static int utilReadInt2(FILE *f) +{ + int res = 0; + int c = fgetc(f); + if(c == EOF) + return -1; + res = c; + c = fgetc(f); + if(c == EOF) + return -1; + return c + (res<<8); +} + +static int utilReadInt3(FILE *f) +{ + int res = 0; + int c = fgetc(f); + if(c == EOF) + return -1; + res = c; + c = fgetc(f); + if(c == EOF) + return -1; + res = c + (res<<8); + c = fgetc(f); + if(c == EOF) + return -1; + return c + (res<<8); +} + +void utilApplyIPS(const char *ips, u8 **r, int *s) +{ + // from the IPS spec at http://zerosoft.zophar.net/ips.htm + FILE *f = fopen(ips, "rb"); + if(!f) + return; + u8 *rom = *r; + int size = *s; + if(fgetc(f) == 'P' && + fgetc(f) == 'A' && + fgetc(f) == 'T' && + fgetc(f) == 'C' && + fgetc(f) == 'H') { + int b; + int offset; + int len; + for(;;) { + // read offset + offset = utilReadInt3(f); + // if offset == EOF, end of patch + if(offset == 0x454f46) + break; + // read length + len = utilReadInt2(f); + if(!len) { + // len == 0, RLE block + len = utilReadInt2(f); + // byte to fill + int c = fgetc(f); + if(c == -1) + break; + b = (u8)c; + } else + b= -1; + // check if we need to reallocate our ROM + if((offset + len) >= size) { + size *= 2; + rom = (u8 *)realloc(rom, size); + *r = rom; + *s = size; + } + if(b == -1) { + // normal block, just read the data + if(fread(&rom[offset], 1, len, f) != (size_t)len) + break; + } else { + // fill the region with the given byte + while(len--) { + rom[offset++] = b; + } + } + } + } + // close the file + fclose(f); +} + +extern bool cpuIsMultiBoot; + +bool utilIsGBAImage(const char * file) +{ + cpuIsMultiBoot = false; + if(strlen(file) > 4) { + const char * p = strrchr(file,'.'); + + if(p != NULL) { + if(_stricmp(p, ".gba") == 0) + return true; + if(_stricmp(p, ".agb") == 0) + return true; + if(_stricmp(p, ".bin") == 0) + return true; + if(_stricmp(p, ".elf") == 0) + return true; + if(_stricmp(p, ".mb") == 0) { + cpuIsMultiBoot = true; + return true; + } + } + } + + return false; +} + +bool utilIsGBImage(const char * file) +{ + if(strlen(file) > 4) { + const char * p = strrchr(file,'.'); + + if(p != NULL) { + if(_stricmp(p, ".gb") == 0) + return true; + if(_stricmp(p, ".gbc") == 0) + return true; + if(_stricmp(p, ".cgb") == 0) + return true; + if(_stricmp(p, ".sgb") == 0) + return true; + } + } + + return false; +} + +bool utilIsZipFile(const char *file) +{ + if(strlen(file) > 4) { + const char * p = strrchr(file,'.'); + + if(p != NULL) { + if(_stricmp(p, ".zip") == 0) + return true; + } + } + + return false; +} + +#if 0 +bool utilIsRarFile(const char *file) +{ + if(strlen(file) > 4) { + char * p = strrchr(file,'.'); + + if(p != NULL) { + if(_stricmp(p, ".rar") == 0) + return true; + } + } + + return false; +} +#endif + +bool utilIsGzipFile(const char *file) +{ + if(strlen(file) > 3) { + const char * p = strrchr(file,'.'); + + if(p != NULL) { + if(_stricmp(p, ".gz") == 0) + return true; + if(_stricmp(p, ".z") == 0) + return true; + } + } + + return false; +} + +void utilGetBaseName(const char *file, char *buffer) +{ + strcpy(buffer, file); + + if(utilIsGzipFile(file)) { + char *p = strrchr(buffer, '.'); + + if(p) + *p = 0; + } +} + +IMAGE_TYPE utilFindType(const char *file) +{ + char buffer[2048]; + + if(utilIsZipFile(file)) { + unzFile unz = unzOpen(file); + + if(unz == NULL) { + systemMessage(MSG_CANNOT_OPEN_FILE, N_("Cannot open file %s"), file); + return IMAGE_UNKNOWN; + } + + int r = unzGoToFirstFile(unz); + + if(r != UNZ_OK) { + unzClose(unz); + systemMessage(MSG_BAD_ZIP_FILE, N_("Bad ZIP file %s"), file); + return IMAGE_UNKNOWN; + } + + IMAGE_TYPE found = IMAGE_UNKNOWN; + + unz_file_info info; + + while(true) { + r = unzGetCurrentFileInfo(unz, + &info, + buffer, + sizeof(buffer), + NULL, + 0, + NULL, + 0); + + if(r != UNZ_OK) { + unzClose(unz); + systemMessage(MSG_BAD_ZIP_FILE, N_("Bad ZIP file %s"), file); + return IMAGE_UNKNOWN; + } + + if(utilIsGBAImage(buffer)) { + found = IMAGE_GBA; + break; + } + + if(utilIsGBImage(buffer)) { + found = IMAGE_GB; + break; + } + + r = unzGoToNextFile(unz); + + if(r != UNZ_OK) + break; + } + unzClose(unz); + + if(found == IMAGE_UNKNOWN) { + systemMessage(MSG_NO_IMAGE_ON_ZIP, + N_("No image found on ZIP file %s"), file); + return found; + } + return found; +#if 0 + } else if(utilIsRarFile(file)) { + IMAGE_TYPE found = IMAGE_UNKNOWN; + + ArchiveList_struct *rarList = NULL; + if(urarlib_list((void *)file, (ArchiveList_struct *)&rarList)) { + ArchiveList_struct *p = rarList; + + while(p) { + if(utilIsGBAImage(p->item.Name)) { + found = IMAGE_GBA; + break; + } + + if(utilIsGBImage(p->item.Name)) { + found = IMAGE_GB; + break; + } + p = p->next; + } + + urarlib_freelist(rarList); + } + return found; +#endif + } else { + if(utilIsGzipFile(file)) + utilGetBaseName(file, buffer); + else + strcpy(buffer, file); + + if(utilIsGBAImage(buffer)) + return IMAGE_GBA; + if(utilIsGBImage(buffer)) + return IMAGE_GB; + } + return IMAGE_UNKNOWN; +} + +static int utilGetSize(int size) +{ + int res = 1; + while(res < size) + res <<= 1; + return res; +} + +static u8 *utilLoadFromZip(const char *file, + bool (*accept)(const char *), + u8 *data, + int &size) +{ + char buffer[2048]; + + unzFile unz = unzOpen(file); + + if(unz == NULL) { + systemMessage(MSG_CANNOT_OPEN_FILE, N_("Cannot open file %s"), file); + return NULL; + } + int r = unzGoToFirstFile(unz); + + if(r != UNZ_OK) { + unzClose(unz); + systemMessage(MSG_BAD_ZIP_FILE, N_("Bad ZIP file %s"), file); + return NULL; + } + + bool found = false; + + unz_file_info info; + + while(true) { + r = unzGetCurrentFileInfo(unz, + &info, + buffer, + sizeof(buffer), + NULL, + 0, + NULL, + 0); + + if(r != UNZ_OK) { + unzClose(unz); + systemMessage(MSG_BAD_ZIP_FILE, N_("Bad ZIP file %s"), file); + return NULL; + } + + if(accept(buffer)) { + found = true; + break; + } + + r = unzGoToNextFile(unz); + + if(r != UNZ_OK) + break; + } + + if(!found) { + unzClose(unz); + systemMessage(MSG_NO_IMAGE_ON_ZIP, + N_("No image found on ZIP file %s"), file); + return NULL; + } + + int fileSize = info.uncompressed_size; + if(size == 0) + size = fileSize; + r = unzOpenCurrentFile(unz); + + if(r != UNZ_OK) { + unzClose(unz); + systemMessage(MSG_ERROR_OPENING_IMAGE, N_("Error opening image %s"), buffer); + return NULL; + } + + u8 *image = data; + + if(image == NULL) { + image = (u8 *)malloc(utilGetSize(size)); + if(image == NULL) { + unzCloseCurrentFile(unz); + unzClose(unz); + systemMessage(MSG_OUT_OF_MEMORY, N_("Failed to allocate memory for %s"), + "data"); + return NULL; + } + size = fileSize; + } + int read = fileSize <= size ? fileSize : size; + r = unzReadCurrentFile(unz, + image, + read); + + unzCloseCurrentFile(unz); + unzClose(unz); + + if(r != (int)read) { + systemMessage(MSG_ERROR_READING_IMAGE, + N_("Error reading image %s"), buffer); + if(data == NULL) + free(image); + return NULL; + } + + size = fileSize; + + return image; +} + +static u8 *utilLoadGzipFile(const char *file, + bool (*accept)(const char *), + u8 *data, + int &size) +{ + FILE *f = fopen(file, "rb"); + + if(f == NULL) { + systemMessage(MSG_ERROR_OPENING_IMAGE, N_("Error opening image %s"), file); + return NULL; + } + + fseek(f, -4, SEEK_END); + int fileSize = fgetc(f) | (fgetc(f) << 8) | (fgetc(f) << 16) | (fgetc(f) << 24); + fclose(f); + if(size == 0) + size = fileSize; + + gzFile gz = gzopen(file, "rb"); + + if(gz == NULL) { + // should not happen, but who knows? + systemMessage(MSG_ERROR_OPENING_IMAGE, N_("Error opening image %s"), file); + return NULL; + } + + u8 *image = data; + + if(image == NULL) { + image = (u8 *)malloc(utilGetSize(size)); + if(image == NULL) { + systemMessage(MSG_OUT_OF_MEMORY, N_("Failed to allocate memory for %s"), + "data"); + fclose(f); + return NULL; + } + size = fileSize; + } + int read = fileSize <= size ? fileSize : size; + int r = gzread(gz, image, read); + gzclose(gz); + + if(r != (int)read) { + systemMessage(MSG_ERROR_READING_IMAGE, + N_("Error reading image %s"), file); + if(data == NULL) + free(image); + return NULL; + } + + size = fileSize; + + return image; +} + +#if 0 +static u8 *utilLoadRarFile(const char *file, + bool (*accept)(const char *), + u8 *data, + int &size) +{ + char buffer[2048]; + + ArchiveList_struct *rarList = NULL; + if(urarlib_list((void *)file, (ArchiveList_struct *)&rarList)) { + ArchiveList_struct *p = rarList; + + bool found = false; + while(p) { + if(accept(p->item.Name)) { + strcpy(buffer, p->item.Name); + found = true; + break; + } + p = p->next; + } + if(found) { + void *memory = NULL; + unsigned long lsize = 0; + size = p->item.UnpSize; + int r = urarlib_get((void *)&memory, &lsize, buffer, (void *)file, ""); + if(!r) { + systemMessage(MSG_ERROR_READING_IMAGE, + N_("Error reading image %s"), buffer); + urarlib_freelist(rarList); + return NULL; + } + u8 *image = (u8 *)memory; + if(data != NULL) { + memcpy(image, data, size); + } + urarlib_freelist(rarList); + return image; + } + systemMessage(MSG_NO_IMAGE_ON_ZIP, + N_("No image found on RAR file %s"), file); + urarlib_freelist(rarList); + return NULL; + } + // nothing found + return NULL; +} +#endif + +u8 *utilLoad(const char *file, + bool (*accept)(const char *), + u8 *data, + int &size) +{ + if(utilIsZipFile(file)) { + return utilLoadFromZip(file, accept, data, size); + } + if(utilIsGzipFile(file)) { + return utilLoadGzipFile(file, accept, data, size); + } +#if 0 + if(utilIsRarFile(file)) { + return utilLoadRarFile(file, accept, data, size); + } +#endif + + u8 *image = data; + + FILE *f = fopen(file, "rb"); + + if(!f) { + systemMessage(MSG_ERROR_OPENING_IMAGE, N_("Error opening image %s"), file); + return NULL; + } + + fseek(f,0,SEEK_END); + int fileSize = ftell(f); + fseek(f,0,SEEK_SET); + if(size == 0) + size = fileSize; + + if(image == NULL) { + image = (u8 *)malloc(utilGetSize(size)); + if(image == NULL) { + systemMessage(MSG_OUT_OF_MEMORY, N_("Failed to allocate memory for %s"), + "data"); + fclose(f); + return NULL; + } + size = fileSize; + } + size_t read = fileSize <= size ? fileSize : size; + size_t r = fread(image, 1, read, f); + fclose(f); + + if(r != read) { + systemMessage(MSG_ERROR_READING_IMAGE, + N_("Error reading image %s"), file); + if(data == NULL) + free(image); + return NULL; + } + + size = fileSize; + + return image; +} + +void utilWriteInt(gzFile gzFile, int i) +{ + utilGzWrite(gzFile, &i, sizeof(int)); +} + +int utilReadInt(gzFile gzFile) +{ + int i = 0; + utilGzRead(gzFile, &i, sizeof(int)); + return i; +} + +void utilReadData(gzFile gzFile, variable_desc* data) +{ + while(data->address) { + utilGzRead(gzFile, data->address, data->size); + data++; + } +} + +void utilWriteData(gzFile gzFile, variable_desc *data) +{ + while(data->address) { + utilGzWrite(gzFile, data->address, data->size); + data++; + } +} + +gzFile utilGzOpen(const char *file, const char *mode) +{ + utilGzWriteFunc = (int (ZEXPORT *)(void *,void * const, unsigned int))gzwrite; + utilGzReadFunc = gzread; + utilGzCloseFunc = gzclose; + + return gzopen(file, mode); +} + +gzFile utilMemGzOpen(char *memory, int available, char *mode) +{ + utilGzWriteFunc = memgzwrite; + utilGzReadFunc = memgzread; + utilGzCloseFunc = memgzclose; + + return memgzopen(memory, available, mode); +} + +int utilGzWrite(gzFile file, const voidp buffer, unsigned int len) +{ + return utilGzWriteFunc(file, buffer, len); +} + +int utilGzRead(gzFile file, voidp buffer, unsigned int len) +{ + return utilGzReadFunc(file, buffer, len); +} + +int utilGzClose(gzFile file) +{ + return utilGzCloseFunc(file); +} + +long utilGzMemTell(gzFile file) +{ + return memtell(file); +} + +void utilGBAFindSave(const u8 *data, const int size) +{ + u32 *p = (u32 *)data; + u32 *end = (u32 *)(data + size); + int saveType = 0; + int flashSize = 0x10000; + bool rtcFound = false; + + while(p < end) { + u32 d = READ32LE(p); + + if(d == 0x52504545) { + if(memcmp(p, "EEPROM_", 7) == 0) { + if(saveType == 0) + saveType = 3; + } + } else if (d == 0x4D415253) { + if(memcmp(p, "SRAM_", 5) == 0) { + if(saveType == 0) + saveType = 1; + } + } else if (d == 0x53414C46) { + if(memcmp(p, "FLASH1M_", 8) == 0) { + if(saveType == 0) { + saveType = 2; + flashSize = 0x20000; + } + } else if(memcmp(p, "FLASH", 5) == 0) { + if(saveType == 0) { + saveType = 2; + flashSize = 0x10000; + } + } + } else if (d == 0x52494953) { + if(memcmp(p, "SIIRTC_V", 8) == 0) + rtcFound = true; + } + p++; + } + // if no matches found, then set it to NONE + if(saveType == 0) { + saveType = 5; + } + rtcEnable(rtcFound); + cpuSaveType = saveType; + flashSetSize(flashSize); +} + +void utilUpdateSystemColorMaps() +{ + switch(systemColorDepth) { + case 16: + { + for(int i = 0; i < 0x10000; i++) { + systemColorMap16[i] = ((i & 0x1f) << systemRedShift) | + (((i & 0x3e0) >> 5) << systemGreenShift) | + (((i & 0x7c00) >> 10) << systemBlueShift); + } + } + break; + case 24: + case 32: + { + for(int i = 0; i < 0x10000; i++) { + systemColorMap32[i] = ((i & 0x1f) << systemRedShift) | + (((i & 0x3e0) >> 5) << systemGreenShift) | + (((i & 0x7c00) >> 10) << systemBlueShift); + } + } + break; + } +} diff --git a/src/Util.h b/src/Util.h index 37fa1f6a..1197dfb6 100644 --- a/src/Util.h +++ b/src/Util.h @@ -1,65 +1,65 @@ -// -*- C++ -*- -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#ifndef VBA_UTIL_H -#define VBA_UTIL_H -enum IMAGE_TYPE { - IMAGE_UNKNOWN = -1, - IMAGE_GBA = 0, - IMAGE_GB = 1 -}; - -// save game - -typedef struct { - void *address; - int size; -} variable_desc; - -extern bool utilWritePNGFile(const char *, int, int, u8 *); -extern bool utilWriteBMPFile(const char *, int, int, u8 *); -extern void utilApplyIPS(const char *ips, u8 **rom, int *size); -extern void utilWriteBMP(char *, int, int, u8 *); -extern bool utilIsGBAImage(const char *); -extern bool utilIsGBImage(const char *); -extern bool utilIsZipFile(const char *); -extern bool utilIsGzipFile(const char *); -extern bool utilIsRarFile(const char *); -extern void utilGetBaseName(const char *, char *); -extern IMAGE_TYPE utilFindType(const char *); -extern u8 *utilLoad(const char *, - bool (*)(const char*), - u8 *, - int &); - -extern void utilPutDword(u8 *, u32); -extern void utilPutWord(u8 *, u16); -extern void utilWriteData(gzFile, variable_desc *); -extern void utilReadData(gzFile, variable_desc *); -extern int utilReadInt(gzFile); -extern void utilWriteInt(gzFile, int); -extern gzFile utilGzOpen(const char *file, const char *mode); -extern gzFile utilMemGzOpen(char *memory, int available, char *mode); -extern int utilGzWrite(gzFile file, const voidp buffer, unsigned int len); -extern int utilGzRead(gzFile file, voidp buffer, unsigned int len); -extern int utilGzClose(gzFile file); -extern long utilGzMemTell(gzFile file); -extern void utilGBAFindSave(const u8 *, const int); -extern void utilUpdateSystemColorMaps(int lcd); -#endif \ No newline at end of file +// -*- C++ -*- +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#ifndef VBA_UTIL_H +#define VBA_UTIL_H +enum IMAGE_TYPE { + IMAGE_UNKNOWN = -1, + IMAGE_GBA = 0, + IMAGE_GB = 1 +}; + +// save game + +typedef struct { + void *address; + int size; +} variable_desc; + +extern bool utilWritePNGFile(const char *, int, int, u8 *); +extern bool utilWriteBMPFile(const char *, int, int, u8 *); +extern void utilApplyIPS(const char *ips, u8 **rom, int *size); +extern void utilWriteBMP(char *, int, int, u8 *); +extern bool utilIsGBAImage(const char *); +extern bool utilIsGBImage(const char *); +extern bool utilIsZipFile(const char *); +extern bool utilIsGzipFile(const char *); +extern bool utilIsRarFile(const char *); +extern void utilGetBaseName(const char *, char *); +extern IMAGE_TYPE utilFindType(const char *); +extern u8 *utilLoad(const char *, + bool (*)(const char*), + u8 *, + int &); + +extern void utilPutDword(u8 *, u32); +extern void utilPutWord(u8 *, u16); +extern void utilWriteData(gzFile, variable_desc *); +extern void utilReadData(gzFile, variable_desc *); +extern int utilReadInt(gzFile); +extern void utilWriteInt(gzFile, int); +extern gzFile utilGzOpen(const char *file, const char *mode); +extern gzFile utilMemGzOpen(char *memory, int available, char *mode); +extern int utilGzWrite(gzFile file, const voidp buffer, unsigned int len); +extern int utilGzRead(gzFile file, voidp buffer, unsigned int len); +extern int utilGzClose(gzFile file); +extern long utilGzMemTell(gzFile file); +extern void utilGBAFindSave(const u8 *, const int); +extern void utilUpdateSystemColorMaps(); +#endif diff --git a/src/arm-new.h b/src/arm-new.h index 8c87f87c..0a1daaff 100644 --- a/src/arm-new.h +++ b/src/arm-new.h @@ -1,7 +1,7 @@ // -*- C++ -*- // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005-2006 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -1113,27 +1113,34 @@ bool C_OUT = C_FLAG;\ u32 value;\ \ - clockTicks++;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift) {\ LOGICAL_LSL_REG\ } else {\ value = reg[opcode & 0x0F].I;\ }\ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -1149,7 +1156,12 @@ int dest = (opcode>>12) & 15;\ bool C_OUT = C_FLAG;\ u32 value;\ - clockTicks++;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift) {\ LOGICAL_LSR_REG\ } else {\ @@ -1158,20 +1170,22 @@ }\ \ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -1187,7 +1201,12 @@ int dest = (opcode>>12) & 15;\ bool C_OUT = C_FLAG;\ u32 value;\ - clockTicks++;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift) {\ LOGICAL_ASR_REG\ } else {\ @@ -1201,20 +1220,22 @@ }\ \ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -1230,27 +1251,34 @@ int dest = (opcode>>12) & 15;\ bool C_OUT = C_FLAG;\ u32 value;\ - clockTicks++;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift) {\ LOGICAL_ROR_REG\ } else {\ LOGICAL_RRX_REG\ }\ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -1260,12 +1288,17 @@ case BASE+1:\ {\ /* OP Rd,Rb,Rm LSL Rs */\ - clockTicks+=2;\ int base = (opcode >> 16) & 0x0F;\ int shift = reg[(opcode >> 8)&15].B.B0;\ int dest = (opcode>>12) & 15;\ bool C_OUT = C_FLAG;\ u32 value;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift) {\ if(shift == 32) {\ value = 0;\ @@ -1280,20 +1313,22 @@ value = reg[opcode & 0x0F].I;\ }\ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -1303,12 +1338,17 @@ case BASE+3:\ {\ /* OP Rd,Rb,Rm LSR Rs */ \ - clockTicks+=2;\ int base = (opcode >> 16) & 0x0F;\ int shift = reg[(opcode >> 8)&15].B.B0;\ int dest = (opcode>>12) & 15;\ bool C_OUT = C_FLAG;\ u32 value;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift) {\ if(shift == 32) {\ value = 0;\ @@ -1323,20 +1363,22 @@ value = reg[opcode & 0x0F].I;\ }\ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -1346,12 +1388,17 @@ case BASE+5:\ {\ /* OP Rd,Rb,Rm ASR Rs */ \ - clockTicks+=2;\ int base = (opcode >> 16) & 0x0F;\ int shift = reg[(opcode >> 8)&15].B.B0;\ int dest = (opcode>>12) & 15;\ bool C_OUT = C_FLAG;\ u32 value;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift < 32) {\ if(shift) {\ LOGICAL_ASR_REG\ @@ -1368,20 +1415,22 @@ }\ }\ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -1391,12 +1440,17 @@ case BASE+7:\ {\ /* OP Rd,Rb,Rm ROR Rs */\ - clockTicks+=2;\ int base = (opcode >> 16) & 0x0F;\ int shift = reg[(opcode >> 8)&15].B.B0;\ int dest = (opcode>>12) & 15;\ bool C_OUT = C_FLAG;\ u32 value;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift) {\ shift &= 0x1f;\ if(shift) {\ @@ -1410,20 +1464,22 @@ C_OUT = (value & 0x80000000 ? true : false);\ }\ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -1452,27 +1508,34 @@ int dest = (opcode >> 12) & 0x0F;\ bool C_OUT = C_FLAG;\ u32 value;\ - clockTicks++;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift) {\ LOGICAL_ROR_IMM\ } else {\ value = opcode & 0xff;\ }\ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -1489,7 +1552,12 @@ int dest = (opcode>>12) & 15;\ bool C_OUT = C_FLAG;\ u32 value;\ - clockTicks++;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ \ if(shift) {\ LOGICAL_LSL_REG\ @@ -1497,20 +1565,22 @@ value = reg[opcode & 0x0F].I;\ }\ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -1525,7 +1595,12 @@ int dest = (opcode>>12) & 15;\ bool C_OUT = C_FLAG;\ u32 value;\ - clockTicks++;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift) {\ LOGICAL_LSR_REG\ } else {\ @@ -1534,20 +1609,22 @@ }\ \ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -1562,7 +1639,12 @@ int dest = (opcode>>12) & 15;\ bool C_OUT = C_FLAG;\ u32 value;\ - clockTicks++;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift) {\ LOGICAL_ASR_REG\ } else {\ @@ -1576,20 +1658,22 @@ }\ \ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -1604,27 +1688,34 @@ int dest = (opcode>>12) & 15;\ bool C_OUT = C_FLAG;\ u32 value;\ - clockTicks++;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift) {\ LOGICAL_ROR_REG\ } else {\ LOGICAL_RRX_REG\ }\ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -1634,11 +1725,16 @@ case BASE+1:\ {\ /* OP Rd,Rb,Rm LSL Rs */\ - clockTicks+=2;\ int shift = reg[(opcode >> 8)&15].B.B0;\ int dest = (opcode>>12) & 15;\ bool C_OUT = C_FLAG;\ u32 value;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift) {\ if(shift == 32) {\ value = 0;\ @@ -1653,20 +1749,22 @@ value = reg[opcode & 0x0F].I;\ }\ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -1676,11 +1774,16 @@ case BASE+3:\ {\ /* OP Rd,Rb,Rm LSR Rs */ \ - clockTicks+=2;\ int shift = reg[(opcode >> 8)&15].B.B0;\ int dest = (opcode>>12) & 15;\ bool C_OUT = C_FLAG;\ u32 value;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift) {\ if(shift == 32) {\ value = 0;\ @@ -1695,20 +1798,22 @@ value = reg[opcode & 0x0F].I;\ }\ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -1718,11 +1823,16 @@ case BASE+5:\ {\ /* OP Rd,Rb,Rm ASR Rs */ \ - clockTicks+=2;\ int shift = reg[(opcode >> 8)&15].B.B0;\ int dest = (opcode>>12) & 15;\ bool C_OUT = C_FLAG;\ u32 value;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift < 32) {\ if(shift) {\ LOGICAL_ASR_REG\ @@ -1739,20 +1849,22 @@ }\ }\ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -1762,11 +1874,16 @@ case BASE+7:\ {\ /* OP Rd,Rb,Rm ROR Rs */\ - clockTicks+=2;\ int shift = reg[(opcode >> 8)&15].B.B0;\ int dest = (opcode>>12) & 15;\ bool C_OUT = C_FLAG;\ u32 value;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift) {\ shift &= 0x1f;\ if(shift) {\ @@ -1780,20 +1897,22 @@ C_OUT = (value & 0x80000000 ? true : false);\ }\ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -1821,27 +1940,34 @@ int dest = (opcode >> 12) & 0x0F;\ bool C_OUT = C_FLAG;\ u32 value;\ - clockTicks++;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift) {\ LOGICAL_ROR_IMM\ } else {\ value = opcode & 0xff;\ }\ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -1858,27 +1984,34 @@ int shift = (opcode >> 7) & 0x1F;\ int dest = (opcode>>12) & 15;\ u32 value;\ - clockTicks++;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift) {\ ARITHMETIC_LSL_REG\ } else {\ value = reg[opcode & 0x0F].I;\ }\ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -1893,27 +2026,34 @@ int shift = (opcode >> 7) & 0x1F;\ int dest = (opcode>>12) & 15;\ u32 value;\ - clockTicks++;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift) {\ ARITHMETIC_LSR_REG\ } else {\ value = 0;\ }\ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -1928,7 +2068,12 @@ int shift = (opcode >> 7) & 0x1F;\ int dest = (opcode>>12) & 15;\ u32 value;\ - clockTicks++;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift) {\ ARITHMETIC_ASR_REG\ } else {\ @@ -1937,20 +2082,22 @@ } else value = 0;\ }\ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -1965,27 +2112,34 @@ int shift = (opcode >> 7) & 0x1F;\ int dest = (opcode>>12) & 15;\ u32 value;\ - clockTicks++;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift) {\ ARITHMETIC_ROR_REG\ } else {\ ARITHMETIC_RRX_REG\ }\ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -1995,12 +2149,16 @@ case BASE+1:\ {\ /* OP Rd,Rb,Rm LSL Rs */\ - clockTicks++;\ int base = (opcode >> 16) & 0x0F;\ int shift = reg[(opcode >> 8)&15].B.B0;\ int dest = (opcode>>12) & 15;\ u32 value;\ - clockTicks++;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift) {\ if(shift == 32) {\ value = 0;\ @@ -2011,20 +2169,22 @@ value = reg[opcode & 0x0F].I;\ }\ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -2034,11 +2194,16 @@ case BASE+3:\ {\ /* OP Rd,Rb,Rm LSR Rs */\ - clockTicks+=2;\ int base = (opcode >> 16) & 0x0F;\ int shift = reg[(opcode >> 8)&15].B.B0;\ int dest = (opcode>>12) & 15;\ u32 value;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift) {\ if(shift == 32) {\ value = 0;\ @@ -2049,20 +2214,22 @@ value = reg[opcode & 0x0F].I;\ }\ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -2072,11 +2239,16 @@ case BASE+5:\ {\ /* OP Rd,Rb,Rm ASR Rs */\ - clockTicks+=2;\ int base = (opcode >> 16) & 0x0F;\ int shift = reg[(opcode >> 8)&15].B.B0;\ int dest = (opcode>>12) & 15;\ u32 value;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift < 32) {\ if(shift) {\ ARITHMETIC_ASR_REG\ @@ -2089,20 +2261,22 @@ } else value = 0;\ }\ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -2112,11 +2286,16 @@ case BASE+7:\ {\ /* OP Rd,Rb,Rm ROR Rs */\ - clockTicks+=2;\ int base = (opcode >> 16) & 0x0F;\ int shift = reg[(opcode >> 8)&15].B.B0;\ int dest = (opcode>>12) & 15;\ u32 value;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ if(shift) {\ shift &= 0x1f;\ if(shift) {\ @@ -2128,20 +2307,22 @@ value = reg[opcode & 0x0F].I;\ }\ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -2169,25 +2350,32 @@ int base = (opcode >> 16) & 0x0F;\ int dest = (opcode >> 12) & 0x0F;\ u32 value;\ - clockTicks++;\ + if ((dest == 15)||((opcode & 0x02000010)==0x10))\ + {\ + clockTicks = 1+codeTicksAccess32(armNextPC);\ + if ((opcode & 0x02000010)==0x10)\ + clockTicks++;\ + }\ {\ ARITHMETIC_ROR_IMM\ }\ if(dest == 15) {\ + clockTicks+=2+codeTicksAccessSeq32(armNextPC)+codeTicksAccessSeq32(armNextPC);\ OPCODE2\ /* todo */\ if(opcode & 0x00100000) {\ - clockTicks++;\ CPUSwitchMode(reg[17].I & 0x1f, false);\ }\ if(armState) {\ reg[15].I &= 0xFFFFFFFC;\ armNextPC = reg[15].I;\ reg[15].I += 4;\ + ARM_PREFETCH;\ } else {\ reg[15].I &= 0xFFFFFFFE;\ armNextPC = reg[15].I;\ reg[15].I += 2;\ + THUMB_PREFETCH;\ }\ } else {\ OPCODE \ @@ -2195,9 +2383,16 @@ }\ break; - u32 opcode = CPUReadMemoryQuick(armNextPC); + u32 opcode = cpuPrefetch[0]; + cpuPrefetch[0] = cpuPrefetch[1]; - clockTicks = memoryWaitFetch32[(armNextPC >> 24) & 15]; + busPrefetch = false; + if (busPrefetchCount & 0xFFFFFE00) + busPrefetchCount = 0x100 | (busPrefetchCount & 0xFF); + + + clockTicks = 0;//codeTicksAccessSeq32(armNextPC)+1; + int oldArmNextPC = armNextPC; #ifndef FINAL_VERSION if(armNextPC == stop) { @@ -2207,6 +2402,8 @@ armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH_NEXT; + int cond = opcode >> 28; // suggested optimization for frequent cases bool cond_res; @@ -2276,18 +2473,23 @@ if(cond_res) { // MUL Rd, Rm, Rs int dest = (opcode >> 16) & 0x0F; int mult = (opcode & 0x0F); + clockTicks = 1; u32 rs = reg[(opcode >> 8) & 0x0F].I; reg[dest].I = reg[mult].I * rs; if(((s32)rs)<0) - rs = ~rs; + rs = ~rs; if((rs & 0xFFFFFF00) == 0) - clockTicks += 2; + clockTicks += 0; else if ((rs & 0xFFFF0000) == 0) - clockTicks += 3; + clockTicks += 1; else if ((rs & 0xFF000000) == 0) - clockTicks += 4; + clockTicks += 2; else - clockTicks += 5; + clockTicks += 3; + if (!busPrefetchCount) + busPrefetchCount = ((++busPrefetchCount)<> 16) & 0x0F; int mult = (opcode & 0x0F); + clockTicks = 1; u32 rs = reg[(opcode >> 8) & 0x0F].I; reg[dest].I = reg[mult].I * rs; N_FLAG = (reg[dest].I & 0x80000000) ? true : false; @@ -2302,24 +2505,30 @@ if(cond_res) { if(((s32)rs)<0) rs = ~rs; if((rs & 0xFFFFFF00) == 0) - clockTicks += 2; + clockTicks += 0; else if ((rs & 0xFFFF0000) == 0) - clockTicks += 3; + clockTicks += 1; else if ((rs & 0xFF000000) == 0) - clockTicks += 4; + clockTicks += 2; else - clockTicks += 5; + clockTicks += 3; + if (!busPrefetchCount) + busPrefetchCount = ((++busPrefetchCount)<> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I; int offset = reg[opcode & 0x0F].I; - clockTicks += 4 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); CPUWriteHalfWord(address, reg[dest].W.W0); address -= offset; reg[base].I = address; @@ -2329,11 +2538,14 @@ if(cond_res) { case 0x06b: { // STRH Rd, [Rn], #-offset + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I; int offset = (opcode & 0x0F) | ((opcode >> 4) & 0xF0); - clockTicks += 4 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); CPUWriteHalfWord(address, reg[dest].W.W0); address -= offset; reg[base].I = address; @@ -2343,11 +2555,14 @@ if(cond_res) { case 0x0ab: { // STRH Rd, [Rn], Rm + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I; int offset = reg[opcode & 0x0F].I; - clockTicks += 4 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); CPUWriteHalfWord(address, reg[dest].W.W0); address += offset; reg[base].I = address; @@ -2357,11 +2572,14 @@ if(cond_res) { case 0x0eb: { // STRH Rd, [Rn], #offset + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I; int offset = (opcode & 0x0F) | ((opcode >> 4) & 0xF0); - clockTicks += 4 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); CPUWriteHalfWord(address, reg[dest].W.W0); address += offset; reg[base].I = address; @@ -2370,20 +2588,26 @@ if(cond_res) { case 0x10b: { // STRH Rd, [Rn, -Rm] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I - reg[opcode & 0x0F].I; - clockTicks += 4 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); CPUWriteHalfWord(address, reg[dest].W.W0); } break; case 0x12b: { // STRH Rd, [Rn, -Rm]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I - reg[opcode & 0x0F].I; - clockTicks += 4 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); CPUWriteHalfWord(address, reg[dest].W.W0); reg[base].I = address; } @@ -2391,20 +2615,26 @@ if(cond_res) { case 0x14b: { // STRH Rd, [Rn, -#offset] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I - ((opcode & 0x0F)|((opcode>>4)&0xF0)); - clockTicks += 4 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); CPUWriteHalfWord(address, reg[dest].W.W0); } break; case 0x16b: { // STRH Rd, [Rn, -#offset]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I - ((opcode & 0x0F)|((opcode>>4)&0xF0)); - clockTicks += 4 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); CPUWriteHalfWord(address, reg[dest].W.W0); reg[base].I = address; } @@ -2412,20 +2642,26 @@ if(cond_res) { case 0x18b: { // STRH Rd, [Rn, Rm] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I + reg[opcode & 0x0F].I; - clockTicks += 4 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); CPUWriteHalfWord(address, reg[dest].W.W0); } break; case 0x1ab: { // STRH Rd, [Rn, Rm]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I + reg[opcode & 0x0F].I; - clockTicks += 4 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); CPUWriteHalfWord(address, reg[dest].W.W0); reg[base].I = address; } @@ -2433,20 +2669,26 @@ if(cond_res) { case 0x1cb: { // STRH Rd, [Rn, #offset] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I + ((opcode & 0x0F)|((opcode>>4)&0xF0)); - clockTicks += 4 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); CPUWriteHalfWord(address, reg[dest].W.W0); } break; case 0x1eb: { // STRH Rd, [Rn, #offset]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I + ((opcode & 0x0F)|((opcode>>4)&0xF0)); - clockTicks += 4 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); CPUWriteHalfWord(address, reg[dest].W.W0); reg[base].I = address; } @@ -2455,456 +2697,852 @@ if(cond_res) { case 0x03b: { // LDRH Rd, [Rn], -Rm + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I; int offset = reg[opcode & 0x0F].I; - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = CPUReadHalfWord(address); if(dest != base) { address -= offset; reg[base].I = address; } + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x05b: case 0x07b: { // LDRH Rd, [Rn], #-offset + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I; int offset = (opcode & 0x0F) | ((opcode >> 4) & 0xF0); - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = CPUReadHalfWord(address); if(dest != base) { address -= offset; reg[base].I = address; } + clockTicks=0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x09b: case 0x0bb: { // LDRH Rd, [Rn], Rm + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I; int offset = reg[opcode & 0x0F].I; - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = CPUReadHalfWord(address); if(dest != base) { address += offset; reg[base].I = address; } + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x0db: case 0x0fb: { // LDRH Rd, [Rn], #offset + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I; int offset = (opcode & 0x0F) | ((opcode >> 4) & 0xF0); - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = CPUReadHalfWord(address); if(dest != base) { address += offset; reg[base].I = address; } + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x11b: { // LDRH Rd, [Rn, -Rm] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I - reg[opcode & 0x0F].I; - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = CPUReadHalfWord(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x13b: { // LDRH Rd, [Rn, -Rm]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I - reg[opcode & 0x0F].I; - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = CPUReadHalfWord(address); if(dest != base) reg[base].I = address; + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x15b: { // LDRH Rd, [Rn, -#offset] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I - ((opcode & 0x0F)|((opcode>>4)&0xF0)); - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = CPUReadHalfWord(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x17b: { // LDRH Rd, [Rn, -#offset]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I - ((opcode & 0x0F)|((opcode>>4)&0xF0)); - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = CPUReadHalfWord(address); if(dest != base) reg[base].I = address; + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x19b: { // LDRH Rd, [Rn, Rm] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I + reg[opcode & 0x0F].I; - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = CPUReadHalfWord(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x1bb: { // LDRH Rd, [Rn, Rm]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I + reg[opcode & 0x0F].I; - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = CPUReadHalfWord(address); if(dest != base) reg[base].I = address; + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x1db: { // LDRH Rd, [Rn, #offset] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I + ((opcode & 0x0F)|((opcode>>4)&0xF0)); - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = CPUReadHalfWord(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x1fb: { // LDRH Rd, [Rn, #offset]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I + ((opcode & 0x0F)|((opcode>>4)&0xF0)); - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = CPUReadHalfWord(address); if(dest != base) reg[base].I = address; + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x01d: case 0x03d: { // LDRSB Rd, [Rn], -Rm + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I; int offset = reg[opcode & 0x0F].I; - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = (s8)CPUReadByte(address); if(dest != base) { address -= offset; reg[base].I = address; } + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x05d: case 0x07d: { // LDRSB Rd, [Rn], #-offset + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I; int offset = (opcode & 0x0F) | ((opcode >> 4) & 0xF0); - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = (s8)CPUReadByte(address); if(dest != base) { address -= offset; reg[base].I = address; } + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x09d: case 0x0bd: { // LDRSB Rd, [Rn], Rm + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I; int offset = reg[opcode & 0x0F].I; - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = (s8)CPUReadByte(address); if(dest != base) { address += offset; reg[base].I = address; } + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x0dd: case 0x0fd: { // LDRSB Rd, [Rn], #offset + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I; int offset = (opcode & 0x0F) | ((opcode >> 4) & 0xF0); - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = (s8)CPUReadByte(address); if(dest != base) { address += offset; reg[base].I = address; } + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x11d: { // LDRSB Rd, [Rn, -Rm] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I - reg[opcode & 0x0F].I; - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = (s8)CPUReadByte(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x13d: { // LDRSB Rd, [Rn, -Rm]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I - reg[opcode & 0x0F].I; - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = (s8)CPUReadByte(address); if(dest != base) reg[base].I = address; + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x15d: { // LDRSB Rd, [Rn, -#offset] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I - ((opcode & 0x0F)|((opcode>>4)&0xF0)); - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = (s8)CPUReadByte(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x17d: { // LDRSB Rd, [Rn, -#offset]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I - ((opcode & 0x0F)|((opcode>>4)&0xF0)); - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = (s8)CPUReadByte(address); if(dest != base) reg[base].I = address; + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x19d: { // LDRSB Rd, [Rn, Rm] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I + reg[opcode & 0x0F].I; - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = (s8)CPUReadByte(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x1bd: { // LDRSB Rd, [Rn, Rm]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I + reg[opcode & 0x0F].I; - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = (s8)CPUReadByte(address); if(dest != base) reg[base].I = address; + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x1dd: { // LDRSB Rd, [Rn, #offset] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I + ((opcode & 0x0F)|((opcode>>4)&0xF0)); - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = (s8)CPUReadByte(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x1fd: { // LDRSB Rd, [Rn, #offset]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I + ((opcode & 0x0F)|((opcode>>4)&0xF0)); - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = (s8)CPUReadByte(address); if(dest != base) reg[base].I = address; + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x01f: case 0x03f: { // LDRSH Rd, [Rn], -Rm + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I; int offset = reg[opcode & 0x0F].I; - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = (s16)CPUReadHalfWordSigned(address); if(dest != base) { address -= offset; reg[base].I = address; } + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x05f: case 0x07f: { // LDRSH Rd, [Rn], #-offset + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I; int offset = (opcode & 0x0F) | ((opcode >> 4) & 0xF0); - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = (s16)CPUReadHalfWordSigned(address); if(dest != base) { address -= offset; reg[base].I = address; } + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x09f: case 0x0bf: { // LDRSH Rd, [Rn], Rm + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I; int offset = reg[opcode & 0x0F].I; - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = (s16)CPUReadHalfWordSigned(address); if(dest != base) { address += offset; reg[base].I = address; } + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x0df: case 0x0ff: { // LDRSH Rd, [Rn], #offset + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I; int offset = (opcode & 0x0F) | ((opcode >> 4) & 0xF0); - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = (s16)CPUReadHalfWordSigned(address); if(dest != base) { address += offset; reg[base].I = address; } + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x11f: { // LDRSH Rd, [Rn, -Rm] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I - reg[opcode & 0x0F].I; - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = (s16)CPUReadHalfWordSigned(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x13f: { // LDRSH Rd, [Rn, -Rm]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I - reg[opcode & 0x0F].I; - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = (s16)CPUReadHalfWordSigned(address); if(dest != base) reg[base].I = address; + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x15f: { // LDRSH Rd, [Rn, -#offset] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I - ((opcode & 0x0F)|((opcode>>4)&0xF0)); - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = (s16)CPUReadHalfWordSigned(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x17f: { // LDRSH Rd, [Rn, -#offset]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I - ((opcode & 0x0F)|((opcode>>4)&0xF0)); - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = (s16)CPUReadHalfWordSigned(address); if(dest != base) reg[base].I = address; + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x19f: { // LDRSH Rd, [Rn, Rm] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I + reg[opcode & 0x0F].I; - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = (s16)CPUReadHalfWordSigned(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x1bf: { // LDRSH Rd, [Rn, Rm]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I + reg[opcode & 0x0F].I; - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = (s16)CPUReadHalfWordSigned(address); if(dest != base) reg[base].I = address; + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x1df: { // LDRSH Rd, [Rn, #offset] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I + ((opcode & 0x0F)|((opcode>>4)&0xF0)); - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = (s16)CPUReadHalfWordSigned(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x1ff: { // LDRSH Rd, [Rn, #offset]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode >> 16) & 0x0F; int dest = (opcode >> 12) & 0x0F; u32 address = reg[base].I + ((opcode & 0x0F)|((opcode>>4)&0xF0)); - clockTicks += 3 + CPUUpdateTicksAccess16(address); reg[dest].I = (s16)CPUReadHalfWordSigned(address); if(dest != base) reg[base].I = address; + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; LOGICAL_DATA_OPCODE_WITHOUT_base(OP_EOR, OP_EOR, 0x020); @@ -2912,25 +3550,28 @@ if(cond_res) { case 0x029: { // MLA Rd, Rm, Rs, Rn + clockTicks = 2; int dest = (opcode >> 16) & 0x0F; int mult = (opcode & 0x0F); u32 rs = reg[(opcode >> 8) & 0x0F].I; reg[dest].I = reg[mult].I * rs + reg[(opcode>>12)&0x0f].I; - if(((s32)rs)<0) - rs = ~rs; if((rs & 0xFFFFFF00) == 0) - clockTicks += 3; + clockTicks += 0; else if ((rs & 0xFFFF0000) == 0) - clockTicks += 4; + clockTicks += 1; else if ((rs & 0xFF000000) == 0) - clockTicks += 5; + clockTicks += 2; else - clockTicks += 6; + clockTicks += 3; + if (!busPrefetchCount) + busPrefetchCount = ((++busPrefetchCount)<> 16) & 0x0F; int mult = (opcode & 0x0F); u32 rs = reg[(opcode >> 8) & 0x0F].I; @@ -2940,13 +3581,16 @@ if(cond_res) { if(((s32)rs)<0) rs = ~rs; if((rs & 0xFFFFFF00) == 0) - clockTicks += 3; + clockTicks += 0; else if ((rs & 0xFFFF0000) == 0) - clockTicks += 4; + clockTicks += 1; else if ((rs & 0xFF000000) == 0) - clockTicks += 5; + clockTicks += 2; else - clockTicks += 6; + clockTicks += 3; + if (!busPrefetchCount) + busPrefetchCount = ((++busPrefetchCount)<> 8) & 0x0F].I; int destLo = (opcode >> 12) & 0x0F; int destHi = (opcode >> 16) & 0x0F; u64 uTemp = ((u64)umult)*((u64)usource); - reg[destLo].I = (u32)(uTemp & 0xFFFFFFFF); + reg[destLo].I = (u32)uTemp; reg[destHi].I = (u32)(uTemp >> 32); if ((usource & 0xFFFFFF00) == 0) - clockTicks += 2; + clockTicks += 0; else if ((usource & 0xFFFF0000) == 0) - clockTicks += 3; + clockTicks += 1; else if ((usource & 0xFF000000) == 0) - clockTicks += 4; + clockTicks += 2; else - clockTicks += 5; + clockTicks += 3; + if (!busPrefetchCount) + busPrefetchCount = ((++busPrefetchCount)<> 8) & 0x0F].I; int destLo = (opcode >> 12) & 0x0F; @@ -2988,13 +3637,16 @@ if(cond_res) { Z_FLAG = (uTemp) ? false : true; N_FLAG = (reg[destHi].I & 0x80000000) ? true : false; if ((usource & 0xFFFFFF00) == 0) - clockTicks += 2; + clockTicks += 0; else if ((usource & 0xFFFF0000) == 0) - clockTicks += 3; + clockTicks += 1; else if ((usource & 0xFF000000) == 0) - clockTicks += 4; + clockTicks += 2; else - clockTicks += 5; + clockTicks += 3; + if (!busPrefetchCount) + busPrefetchCount = ((++busPrefetchCount)<> 8) & 0x0F].I; int destLo = (opcode >> 12) & 0x0F; @@ -3013,18 +3666,22 @@ if(cond_res) { reg[destLo].I = (u32)uTemp; reg[destHi].I = (u32)(uTemp >> 32); if ((usource & 0xFFFFFF00) == 0) - clockTicks += 3; + clockTicks += 0; else if ((usource & 0xFFFF0000) == 0) - clockTicks += 4; + clockTicks += 1; else if ((usource & 0xFF000000) == 0) - clockTicks += 5; + clockTicks += 2; else - clockTicks += 6; + clockTicks += 3; + if (!busPrefetchCount) + busPrefetchCount = ((++busPrefetchCount)<> 8) & 0x0F].I; int destLo = (opcode >> 12) & 0x0F; @@ -3038,13 +3695,16 @@ if(cond_res) { Z_FLAG = (uTemp) ? false : true; N_FLAG = (reg[destHi].I & 0x80000000) ? true : false; if ((usource & 0xFFFFFF00) == 0) - clockTicks += 3; + clockTicks += 0; else if ((usource & 0xFFFF0000) == 0) - clockTicks += 4; + clockTicks += 1; else if ((usource & 0xFF000000) == 0) - clockTicks += 5; + clockTicks += 2; else - clockTicks += 6; + clockTicks += 3; + if (!busPrefetchCount) + busPrefetchCount = ((++busPrefetchCount)<> 12) & 0x0F; int destHi = (opcode >> 16) & 0x0F; u32 rs = reg[(opcode >> 8) & 0x0F].I; @@ -3063,18 +3724,22 @@ if(cond_res) { if(((s32)rs) < 0) rs = ~rs; if((rs & 0xFFFFFF00) == 0) - clockTicks += 2; + clockTicks += 0; else if((rs & 0xFFFF0000) == 0) - clockTicks += 3; + clockTicks += 1; else if((rs & 0xFF000000) == 0) - clockTicks += 4; + clockTicks += 2; else - clockTicks += 5; + clockTicks += 3; + if (!busPrefetchCount) + busPrefetchCount = ((++busPrefetchCount)<> 12) & 0x0F; int destHi = (opcode >> 16) & 0x0F; u32 rs = reg[(opcode >> 8) & 0x0F].I; @@ -3088,13 +3753,16 @@ if(cond_res) { if(((s32)rs) < 0) rs = ~rs; if((rs & 0xFFFFFF00) == 0) - clockTicks += 2; + clockTicks += 0; else if((rs & 0xFFFF0000) == 0) - clockTicks += 3; + clockTicks += 1; else if((rs & 0xFF000000) == 0) - clockTicks += 4; + clockTicks += 2; else - clockTicks += 5; + clockTicks += 3; + if (!busPrefetchCount) + busPrefetchCount = ((++busPrefetchCount)<> 12) & 0x0F; int destHi = (opcode >> 16) & 0x0F; u32 rs = reg[(opcode >> 8) & 0x0F].I; @@ -3116,18 +3785,21 @@ if(cond_res) { if(((s32)rs) < 0) rs = ~rs; if((rs & 0xFFFFFF00) == 0) - clockTicks += 3; + clockTicks += 0; else if((rs & 0xFFFF0000) == 0) - clockTicks += 4; + clockTicks += 1; else if((rs & 0xFF000000) == 0) - clockTicks += 5; + clockTicks += 2; else - clockTicks += 6; + clockTicks += 3; + if (!busPrefetchCount) + busPrefetchCount = ((++busPrefetchCount)<> 12) & 0x0F; int destHi = (opcode >> 16) & 0x0F; u32 rs = reg[(opcode >> 8) & 0x0F].I; @@ -3144,20 +3816,21 @@ if(cond_res) { if(((s32)rs) < 0) rs = ~rs; if((rs & 0xFFFFFF00) == 0) - clockTicks += 3; + clockTicks += 0; else if((rs & 0xFFFF0000) == 0) - clockTicks += 4; + clockTicks += 1; else if((rs & 0xFF000000) == 0) - clockTicks += 5; + clockTicks += 2; else - clockTicks += 6; + clockTicks += 3; + if (!busPrefetchCount) + busPrefetchCount = ((++busPrefetchCount)<> 12) & 0x0F].I = reg[16].I; break; @@ -3168,7 +3841,8 @@ if(cond_res) { u32 temp = CPUReadMemory(address); CPUWriteMemory(address, reg[opcode&15].I); reg[(opcode >> 12) & 15].I = temp; - clockTicks++; + clockTicks = 4 + dataTicksAccess32(address) + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; LOGICAL_DATA_OPCODE(OP_TEQ, OP_TEQ, 0x130); @@ -3192,24 +3866,33 @@ if(cond_res) { CPUSwitchMode(newValue & 0x1f, false); reg[16].I = newValue; CPUUpdateFlags(); - clockTicks++; + if(!armState) { // this should not be allowed, but it seems to work + THUMB_PREFETCH; + reg[15].I = armNextPC + 2; + } } break; case 0x121: { // BX Rm // TODO: check if right instruction... - clockTicks += 3; int base = opcode & 0x0F; + busPrefetchCount=0; armState = reg[base].I & 1 ? false : true; if(armState) { reg[15].I = reg[base].I & 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks = codeTicksAccessSeq32(armNextPC) + + codeTicksAccessSeq32(armNextPC) + codeTicksAccess32(armNextPC) + 3; } else { reg[15].I = reg[base].I & 0xFFFFFFFE; armNextPC = reg[15].I; reg[15].I += 2; + THUMB_PREFETCH; + clockTicks = codeTicksAccessSeq16(armNextPC) + + codeTicksAccessSeq16(armNextPC) + codeTicksAccess16(armNextPC) + 3; } } break; @@ -3218,7 +3901,6 @@ if(cond_res) { // MRS Rd, SPSR // TODO: check if right instruction... reg[(opcode >> 12) & 0x0F].I = reg[17].I; - clockTicks++; break; case 0x149: { @@ -3227,7 +3909,8 @@ if(cond_res) { u32 temp = CPUReadByte(address); CPUWriteByte(address, reg[opcode&15].B.B0); reg[(opcode>>12)&15].I = temp; - clockTicks++; + clockTicks = 4 + dataTicksAccess32(address) + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; ARITHMETIC_DATA_OPCODE(OP_CMN, OP_CMN, 0x170); @@ -3245,7 +3928,6 @@ if(cond_res) { if(opcode & 0x00080000) reg[17].I = (reg[17].I & 0x00FFFFFF) | (value & 0xFF000000); } - clockTicks++; } break; LOGICAL_DATA_OPCODE (OP_ORR, OP_ORR, 0x180); @@ -3306,7 +3988,10 @@ if(cond_res) { CPUSwitchMode(newValue & 0x1f, false); reg[16].I = newValue; CPUUpdateFlags(); - clockTicks++; + if(!armState) { // this should not be allowed, but it seems to work + THUMB_PREFETCH; + reg[15].I = armNextPC + 2; + } } break; case 0x360: @@ -3342,7 +4027,6 @@ if(cond_res) { if(opcode & 0x00080000) reg[17].I = (reg[17].I & 0x00FFFFFF) | (value & 0xFF000000); } - clockTicks++; } break; CASE_16(0x400) @@ -3350,13 +4034,16 @@ if(cond_res) { CASE_16(0x420) { // STR Rd, [Rn], -# + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I; CPUWriteMemory(address, reg[dest].I); reg[base].I = address - offset; - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; CASE_16(0x480) @@ -3364,64 +4051,81 @@ if(cond_res) { CASE_16(0x4a0) { // STR Rd, [Rn], # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I; CPUWriteMemory(address, reg[dest].I); reg[base].I = address + offset; - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; CASE_16(0x500) { // STR Rd, [Rn, -#] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I - offset; CPUWriteMemory(address, reg[dest].I); - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; CASE_16(0x520) { // STR Rd, [Rn, -#]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I - offset; reg[base].I = address; CPUWriteMemory(address, reg[dest].I); - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; CASE_16(0x580) { // STR Rd, [Rn, #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I + offset; CPUWriteMemory(address, reg[dest].I); - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; CASE_16(0x5a0) { // STR Rd, [Rn, #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I + offset; reg[base].I = address; CPUWriteMemory(address, reg[dest].I); - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; CASE_16(0x410) { // LDR Rd, [Rn], -# + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; @@ -3429,18 +4133,23 @@ if(cond_res) { reg[dest].I = CPUReadMemory(address); if(dest != base) reg[base].I -= offset; - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; CASE_16(0x430) { // LDRT Rd, [Rn], -# + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; @@ -3448,12 +4157,23 @@ if(cond_res) { reg[dest].I = CPUReadMemory(address); if(dest != base) reg[base].I -= offset; - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; CASE_16(0x490) { // LDR Rd, [Rn], # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; @@ -3461,18 +4181,23 @@ if(cond_res) { reg[dest].I = CPUReadMemory(address); if(dest != base) reg[base].I += offset; - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; CASE_16(0x4b0) { // LDRT Rd, [Rn], # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; @@ -3480,29 +4205,45 @@ if(cond_res) { reg[dest].I = CPUReadMemory(address); if(dest != base) reg[base].I += offset; - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; CASE_16(0x510) { // LDR Rd, [Rn, -#] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I - offset; reg[dest].I = CPUReadMemory(address); - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; CASE_16(0x530) { // LDR Rd, [Rn, -#]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; @@ -3510,35 +4251,45 @@ if(cond_res) { reg[dest].I = CPUReadMemory(address); if(dest != base) reg[base].I = address; - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } - } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); + } break; CASE_16(0x590) { // LDR Rd, [Rn, #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I + offset; reg[dest].I = CPUReadMemory(address); - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; CASE_16(0x5b0) { // LDR Rd, [Rn, #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; @@ -3546,13 +4297,16 @@ if(cond_res) { reg[dest].I = CPUReadMemory(address); if(dest != base) reg[base].I = address; - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; CASE_16(0x440) @@ -3560,73 +4314,91 @@ if(cond_res) { CASE_16(0x460) { // STRB Rd, [Rn], -# + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I; CPUWriteByte(address, reg[dest].B.B0); reg[base].I = address - offset; - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; CASE_16(0x4c0) // T versions shouldn't be different on GBA CASE_16(0x4e0) - // STRB Rd, [Rn], # { + // STRB Rd, [Rn], # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I; CPUWriteByte(address, reg[dest].B.B0); reg[base].I = address + offset; - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; CASE_16(0x540) { // STRB Rd, [Rn, -#] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I - offset; CPUWriteByte(address, reg[dest].B.B0); - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; CASE_16(0x560) { // STRB Rd, [Rn, -#]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I - offset; reg[base].I = address; CPUWriteByte(address, reg[dest].B.B0); - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; CASE_16(0x5c0) { // STRB Rd, [Rn, #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I + offset; CPUWriteByte(address, reg[dest].B.B0); - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; CASE_16(0x5e0) { // STRB Rd, [Rn, #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I + offset; reg[base].I = address; CPUWriteByte(address, reg[dest].I); - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; CASE_16(0x450) @@ -3634,6 +4406,8 @@ if(cond_res) { CASE_16(0x470) { // LDRB Rd, [Rn], -# + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; @@ -3641,13 +4415,24 @@ if(cond_res) { reg[dest].I = CPUReadByte(address); if(dest != base) reg[base].I -= offset; - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; CASE_16(0x4d0) CASE_16(0x4f0) // T versions should not be different { // LDRB Rd, [Rn], # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; @@ -3655,23 +4440,45 @@ if(cond_res) { reg[dest].I = CPUReadByte(address); if(dest != base) reg[base].I += offset; - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; CASE_16(0x550) { // LDRB Rd, [Rn, -#] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I - offset; reg[dest].I = CPUReadByte(address); - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; CASE_16(0x570) { // LDRB Rd, [Rn, -#]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; @@ -3679,23 +4486,45 @@ if(cond_res) { reg[dest].I = CPUReadByte(address); if(dest != base) reg[base].I = address; - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; CASE_16(0x5d0) { // LDRB Rd, [Rn, #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I + offset; reg[dest].I = CPUReadByte(address); - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; CASE_16(0x5f0) { // LDRB Rd, [Rn, #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = opcode & 0xFFF; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; @@ -3703,7 +4532,16 @@ if(cond_res) { reg[dest].I = CPUReadByte(address); if(dest != base) reg[base].I = address; - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x600: @@ -3713,13 +4551,16 @@ if(cond_res) { case 0x628: { // STR Rd, [Rn], -Rm, LSL # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = reg[opcode & 15].I << ((opcode>>7)& 31); int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I; CPUWriteMemory(address, reg[dest].I); reg[base].I = address - offset; - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x602: @@ -3729,6 +4570,8 @@ if(cond_res) { case 0x62a: { // STR Rd, [Rn], -Rm, LSR # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset = shift ? reg[opcode & 15].I >> shift : 0; int dest = (opcode >> 12) & 15; @@ -3736,7 +4579,8 @@ if(cond_res) { u32 address = reg[base].I; CPUWriteMemory(address, reg[dest].I); reg[base].I = address - offset; - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x604: @@ -3746,6 +4590,8 @@ if(cond_res) { case 0x62c: { // STR Rd, [Rn], -Rm, ASR # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset; if(shift) @@ -3759,7 +4605,8 @@ if(cond_res) { u32 address = reg[base].I; CPUWriteMemory(address, reg[dest].I); reg[base].I = address - offset; - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x606: @@ -3769,6 +4616,8 @@ if(cond_res) { case 0x62e: { // STR Rd, [Rn], -Rm, ROR # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; u32 value = reg[opcode & 15].I; if(shift) { @@ -3781,7 +4630,8 @@ if(cond_res) { u32 address = reg[base].I; CPUWriteMemory(address, reg[dest].I); reg[base].I = address - value; - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x680: @@ -3791,13 +4641,16 @@ if(cond_res) { case 0x6a8: { // STR Rd, [Rn], Rm, LSL # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = reg[opcode & 15].I << ((opcode>>7)& 31); int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I; CPUWriteMemory(address, reg[dest].I); reg[base].I = address + offset; - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x682: @@ -3807,6 +4660,8 @@ if(cond_res) { case 0x6aa: { // STR Rd, [Rn], Rm, LSR # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset = shift ? reg[opcode & 15].I >> shift : 0; int dest = (opcode >> 12) & 15; @@ -3814,7 +4669,8 @@ if(cond_res) { u32 address = reg[base].I; CPUWriteMemory(address, reg[dest].I); reg[base].I = address + offset; - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x684: @@ -3824,6 +4680,8 @@ if(cond_res) { case 0x6ac: { // STR Rd, [Rn], Rm, ASR # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset; if(shift) @@ -3837,7 +4695,8 @@ if(cond_res) { u32 address = reg[base].I; CPUWriteMemory(address, reg[dest].I); reg[base].I = address + offset; - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x686: @@ -3847,6 +4706,8 @@ if(cond_res) { case 0x6ae: { // STR Rd, [Rn], Rm, ROR # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; u32 value = reg[opcode & 15].I; if(shift) { @@ -3859,38 +4720,47 @@ if(cond_res) { u32 address = reg[base].I; CPUWriteMemory(address, reg[dest].I); reg[base].I = address + value; - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x700: case 0x708: { // STR Rd, [Rn, -Rm, LSL #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = reg[opcode & 15].I << ((opcode>>7)& 31); int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I - offset; CPUWriteMemory(address, reg[dest].I); - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x702: case 0x70a: { // STR Rd, [Rn, -Rm, LSR #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset = shift ? reg[opcode & 15].I >> shift : 0; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I - offset; CPUWriteMemory(address, reg[dest].I); - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x704: case 0x70c: { // STR Rd, [Rn, -Rm, ASR #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset; if(shift) @@ -3903,13 +4773,16 @@ if(cond_res) { int base = (opcode >> 16) & 15; u32 address = reg[base].I - offset; CPUWriteMemory(address, reg[dest].I); - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x706: case 0x70e: { // STR Rd, [Rn, -Rm, ROR #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; u32 value = reg[opcode & 15].I; if(shift) { @@ -3921,26 +4794,32 @@ if(cond_res) { int base = (opcode >> 16) & 15; u32 address = reg[base].I - value; CPUWriteMemory(address, reg[dest].I); - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x720: case 0x728: { // STR Rd, [Rn, -Rm, LSL #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = reg[opcode & 15].I << ((opcode>>7)& 31); int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I - offset; reg[base].I = address; CPUWriteMemory(address, reg[dest].I); - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x722: case 0x72a: { // STR Rd, [Rn, -Rm, LSR #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset = shift ? reg[opcode & 15].I >> shift : 0; int dest = (opcode >> 12) & 15; @@ -3948,13 +4827,16 @@ if(cond_res) { u32 address = reg[base].I - offset; reg[base].I = address; CPUWriteMemory(address, reg[dest].I); - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x724: case 0x72c: { // STR Rd, [Rn, -Rm, ASR #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset; if(shift) @@ -3968,13 +4850,16 @@ if(cond_res) { u32 address = reg[base].I - offset; reg[base].I = address; CPUWriteMemory(address, reg[dest].I); - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x726: case 0x72e: { // STR Rd, [Rn, -Rm, ROR #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; u32 value = reg[opcode & 15].I; if(shift) { @@ -3987,38 +4872,47 @@ if(cond_res) { u32 address = reg[base].I - value; reg[base].I = address; CPUWriteMemory(address, reg[dest].I); - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x780: case 0x788: { // STR Rd, [Rn, Rm, LSL #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = reg[opcode & 15].I << ((opcode>>7)& 31); int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I + offset; CPUWriteMemory(address, reg[dest].I); - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x782: case 0x78a: { // STR Rd, [Rn, Rm, LSR #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset = shift ? reg[opcode & 15].I >> shift : 0; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I + offset; CPUWriteMemory(address, reg[dest].I); - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x784: case 0x78c: { // STR Rd, [Rn, Rm, ASR #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset; if(shift) @@ -4031,13 +4925,16 @@ if(cond_res) { int base = (opcode >> 16) & 15; u32 address = reg[base].I + offset; CPUWriteMemory(address, reg[dest].I); - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x786: case 0x78e: { // STR Rd, [Rn, Rm, ROR #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; u32 value = reg[opcode & 15].I; if(shift) { @@ -4049,26 +4946,32 @@ if(cond_res) { int base = (opcode >> 16) & 15; u32 address = reg[base].I + value; CPUWriteMemory(address, reg[dest].I); - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x7a0: case 0x7a8: { // STR Rd, [Rn, Rm, LSL #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = reg[opcode & 15].I << ((opcode>>7)& 31); int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I + offset; reg[base].I = address; CPUWriteMemory(address, reg[dest].I); - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x7a2: case 0x7aa: { // STR Rd, [Rn, Rm, LSR #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset = shift ? reg[opcode & 15].I >> shift : 0; int dest = (opcode >> 12) & 15; @@ -4076,13 +4979,16 @@ if(cond_res) { u32 address = reg[base].I + offset; reg[base].I = address; CPUWriteMemory(address, reg[dest].I); - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x7a4: case 0x7ac: { // STR Rd, [Rn, Rm, ASR #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset; if(shift) @@ -4096,13 +5002,16 @@ if(cond_res) { u32 address = reg[base].I + offset; reg[base].I = address; CPUWriteMemory(address, reg[dest].I); - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x7a6: case 0x7ae: { // STR Rd, [Rn, Rm, ROR #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; u32 value = reg[opcode & 15].I; if(shift) { @@ -4115,7 +5024,8 @@ if(cond_res) { u32 address = reg[base].I + value; reg[base].I = address; CPUWriteMemory(address, reg[dest].I); - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks = 2 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x610: @@ -4125,6 +5035,8 @@ if(cond_res) { case 0x638: { // LDR Rd, [Rn], -Rm, LSL # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = reg[opcode & 15].I << ((opcode>>7)& 31); int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; @@ -4132,13 +5044,16 @@ if(cond_res) { reg[dest].I = CPUReadMemory(address); if(dest != base) reg[base].I = address - offset; - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x612: @@ -4148,6 +5063,8 @@ if(cond_res) { case 0x63a: { // LDR Rd, [Rn], -Rm, LSR # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset = shift ? reg[opcode & 15].I >> shift : 0; int dest = (opcode >> 12) & 15; @@ -4156,13 +5073,16 @@ if(cond_res) { reg[dest].I = CPUReadMemory(address); if(dest != base) reg[base].I = address - offset; - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x614: @@ -4172,6 +5092,8 @@ if(cond_res) { case 0x63c: { // LDR Rd, [Rn], -Rm, ASR # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset; if(shift) @@ -4186,13 +5108,16 @@ if(cond_res) { reg[dest].I = CPUReadMemory(address); if(dest != base) reg[base].I = address - offset; - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x616: @@ -4202,6 +5127,8 @@ if(cond_res) { case 0x63e: { // LDR Rd, [Rn], -Rm, ROR # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; u32 value = reg[opcode & 15].I; if(shift) { @@ -4215,13 +5142,16 @@ if(cond_res) { reg[dest].I = CPUReadMemory(address); if(dest != base) reg[base].I = address - value; - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x690: @@ -4231,6 +5161,8 @@ if(cond_res) { case 0x6b8: { // LDR Rd, [Rn], Rm, LSL # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = reg[opcode & 15].I << ((opcode>>7)& 31); int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; @@ -4238,13 +5170,16 @@ if(cond_res) { reg[dest].I = CPUReadMemory(address); if(dest != base) reg[base].I = address + offset; - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x692: @@ -4254,6 +5189,8 @@ if(cond_res) { case 0x6ba: { // LDR Rd, [Rn], Rm, LSR # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset = shift ? reg[opcode & 15].I >> shift : 0; int dest = (opcode >> 12) & 15; @@ -4262,13 +5199,16 @@ if(cond_res) { reg[dest].I = CPUReadMemory(address); if(dest != base) reg[base].I = address + offset; - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x694: @@ -4278,6 +5218,8 @@ if(cond_res) { case 0x6bc: { // LDR Rd, [Rn], Rm, ASR # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset; if(shift) @@ -4292,13 +5234,16 @@ if(cond_res) { reg[dest].I = CPUReadMemory(address); if(dest != base) reg[base].I = address + offset; - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x696: @@ -4308,6 +5253,8 @@ if(cond_res) { case 0x6be: { // LDR Rd, [Rn], Rm, ROR # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; u32 value = reg[opcode & 15].I; if(shift) { @@ -4321,56 +5268,71 @@ if(cond_res) { reg[dest].I = CPUReadMemory(address); if(dest != base) reg[base].I = address + value; - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x710: case 0x718: { // LDR Rd, [Rn, -Rm, LSL #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = reg[opcode & 15].I << ((opcode>>7)& 31); int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I - offset; reg[dest].I = CPUReadMemory(address); - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x712: case 0x71a: { // LDR Rd, [Rn, -Rm, LSR #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset = shift ? reg[opcode & 15].I >> shift : 0; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I - offset; reg[dest].I = CPUReadMemory(address); - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x714: case 0x71c: { // LDR Rd, [Rn, -Rm, ASR #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset; if(shift) @@ -4383,19 +5345,24 @@ if(cond_res) { int base = (opcode >> 16) & 15; u32 address = reg[base].I - offset; reg[dest].I = CPUReadMemory(address); - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x716: case 0x71e: { // LDR Rd, [Rn, -Rm, ROR #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; u32 value = reg[opcode & 15].I; if(shift) { @@ -4407,19 +5374,24 @@ if(cond_res) { int base = (opcode >> 16) & 15; u32 address = reg[base].I - value; reg[dest].I = CPUReadMemory(address); - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x730: case 0x738: { // LDR Rd, [Rn, -Rm, LSL #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = reg[opcode & 15].I << ((opcode>>7)& 31); int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; @@ -4427,19 +5399,24 @@ if(cond_res) { reg[dest].I = CPUReadMemory(address); if(dest != base) reg[base].I = address; - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x732: case 0x73a: { // LDR Rd, [Rn, -Rm, LSR #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset = shift ? reg[opcode & 15].I >> shift : 0; int dest = (opcode >> 12) & 15; @@ -4448,19 +5425,24 @@ if(cond_res) { reg[dest].I = CPUReadMemory(address); if(dest != base) reg[base].I = address; - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x734: case 0x73c: { // LDR Rd, [Rn, -Rm, ASR #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset; if(shift) @@ -4475,19 +5457,24 @@ if(cond_res) { reg[dest].I = CPUReadMemory(address); if(dest != base) reg[base].I = address; - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x736: case 0x73e: { // LDR Rd, [Rn, -Rm, ROR #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; u32 value = reg[opcode & 15].I; if(shift) { @@ -4501,56 +5488,71 @@ if(cond_res) { reg[dest].I = CPUReadMemory(address); if(dest != base) reg[base].I = address; - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x790: case 0x798: { // LDR Rd, [Rn, Rm, LSL #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = reg[opcode & 15].I << ((opcode>>7)& 31); int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I + offset; reg[dest].I = CPUReadMemory(address); - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x792: case 0x79a: { // LDR Rd, [Rn, Rm, LSR #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset = shift ? reg[opcode & 15].I >> shift : 0; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I + offset; reg[dest].I = CPUReadMemory(address); - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x794: case 0x79c: { // LDR Rd, [Rn, Rm, ASR #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset; if(shift) @@ -4563,19 +5565,24 @@ if(cond_res) { int base = (opcode >> 16) & 15; u32 address = reg[base].I + offset; reg[dest].I = CPUReadMemory(address); - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x796: case 0x79e: { // LDR Rd, [Rn, Rm, ROR #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; u32 value = reg[opcode & 15].I; if(shift) { @@ -4587,19 +5594,24 @@ if(cond_res) { int base = (opcode >> 16) & 15; u32 address = reg[base].I + value; reg[dest].I = CPUReadMemory(address); - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x7b0: case 0x7b8: { // LDR Rd, [Rn, Rm, LSL #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = reg[opcode & 15].I << ((opcode>>7)& 31); int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; @@ -4607,19 +5619,24 @@ if(cond_res) { reg[dest].I = CPUReadMemory(address); if(dest != base) reg[base].I = address; - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x7b2: case 0x7ba: { // LDR Rd, [Rn, Rm, LSR #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset = shift ? reg[opcode & 15].I >> shift : 0; int dest = (opcode >> 12) & 15; @@ -4628,19 +5645,24 @@ if(cond_res) { reg[dest].I = CPUReadMemory(address); if(dest != base) reg[base].I = address; - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x7b4: case 0x7bc: { // LDR Rd, [Rn, Rm, ASR #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset; if(shift) @@ -4655,19 +5677,24 @@ if(cond_res) { reg[dest].I = CPUReadMemory(address); if(dest != base) reg[base].I = address; - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x7b6: case 0x7be: { // LDR Rd, [Rn, Rm, ROR #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; u32 value = reg[opcode & 15].I; if(shift) { @@ -4681,13 +5708,16 @@ if(cond_res) { reg[dest].I = CPUReadMemory(address); if(dest != base) reg[base].I = address; - clockTicks += 3 + CPUUpdateTicksAccess32(address); + clockTicks = 0; if(dest == 15) { - clockTicks += 2; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); } + clockTicks += 3 + dataTicksAccess32(address) + + codeTicksAccess32(armNextPC); } break; case 0x640: @@ -4697,13 +5727,16 @@ if(cond_res) { case 0x668: { // STRB Rd, [Rn], -Rm, LSL # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = reg[opcode & 15].I << ((opcode>>7)& 31); int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I; CPUWriteByte(address, reg[dest].B.B0); reg[base].I = address - offset; - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x642: @@ -4713,6 +5746,8 @@ if(cond_res) { case 0x66a: { // STRB Rd, [Rn], -Rm, LSR # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset = shift ? reg[opcode & 15].I >> shift : 0; int dest = (opcode >> 12) & 15; @@ -4720,7 +5755,8 @@ if(cond_res) { u32 address = reg[base].I; CPUWriteByte(address, reg[dest].B.B0); reg[base].I = address - offset; - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x644: @@ -4730,6 +5766,8 @@ if(cond_res) { case 0x66c: { // STRB Rd, [Rn], -Rm, ASR # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset; if(shift) @@ -4743,7 +5781,8 @@ if(cond_res) { u32 address = reg[base].I; CPUWriteByte(address, reg[dest].B.B0); reg[base].I = address - offset; - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x646: @@ -4753,6 +5792,8 @@ if(cond_res) { case 0x66e: { // STRB Rd, [Rn], -Rm, ROR # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; u32 value = reg[opcode & 15].I; if(shift) { @@ -4765,7 +5806,8 @@ if(cond_res) { u32 address = reg[base].I; CPUWriteByte(address, reg[dest].B.B0); reg[base].I = address - value; - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x6c0: @@ -4775,13 +5817,16 @@ if(cond_res) { case 0x6e8: { // STRB Rd, [Rn], Rm, LSL # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = reg[opcode & 15].I << ((opcode>>7)& 31); int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I; CPUWriteByte(address, reg[dest].B.B0); reg[base].I = address + offset; - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x6c2: @@ -4791,6 +5836,8 @@ if(cond_res) { case 0x6ea: { // STRB Rd, [Rn], Rm, LSR # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset = shift ? reg[opcode & 15].I >> shift : 0; int dest = (opcode >> 12) & 15; @@ -4798,7 +5845,8 @@ if(cond_res) { u32 address = reg[base].I; CPUWriteByte(address, reg[dest].B.B0); reg[base].I = address + offset; - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x6c4: @@ -4807,7 +5855,9 @@ if(cond_res) { case 0x6e4: case 0x6ec: { - // STR Rd, [Rn], Rm, ASR # + // STRB Rd, [Rn], Rm, ASR # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset; if(shift) @@ -4821,7 +5871,8 @@ if(cond_res) { u32 address = reg[base].I; CPUWriteByte(address, reg[dest].B.B0); reg[base].I = address + offset; - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x6c6: @@ -4831,6 +5882,8 @@ if(cond_res) { case 0x6ee: { // STRB Rd, [Rn], Rm, ROR # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; u32 value = reg[opcode & 15].I; if(shift) { @@ -4843,38 +5896,47 @@ if(cond_res) { u32 address = reg[base].I; CPUWriteByte(address, reg[dest].B.B0); reg[base].I = address + value; - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x740: case 0x748: { // STRB Rd, [Rn, -Rm, LSL #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = reg[opcode & 15].I << ((opcode>>7)& 31); int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I - offset; CPUWriteByte(address, reg[dest].B.B0); - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x742: case 0x74a: { // STRB Rd, [Rn, -Rm, LSR #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset = shift ? reg[opcode & 15].I >> shift : 0; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I - offset; CPUWriteByte(address, reg[dest].B.B0); - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x744: case 0x74c: { // STRB Rd, [Rn, -Rm, ASR #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset; if(shift) @@ -4887,13 +5949,16 @@ if(cond_res) { int base = (opcode >> 16) & 15; u32 address = reg[base].I - offset; CPUWriteByte(address, reg[dest].B.B0); - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x746: case 0x74e: { // STRB Rd, [Rn, -Rm, ROR #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; u32 value = reg[opcode & 15].I; if(shift) { @@ -4905,26 +5970,32 @@ if(cond_res) { int base = (opcode >> 16) & 15; u32 address = reg[base].I - value; CPUWriteByte(address, reg[dest].B.B0); - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x760: case 0x768: { // STRB Rd, [Rn, -Rm, LSL #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = reg[opcode & 15].I << ((opcode>>7)& 31); int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I - offset; reg[base].I = address; CPUWriteByte(address, reg[dest].B.B0); - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x762: case 0x76a: { // STRB Rd, [Rn, -Rm, LSR #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset = shift ? reg[opcode & 15].I >> shift : 0; int dest = (opcode >> 12) & 15; @@ -4932,13 +6003,16 @@ if(cond_res) { u32 address = reg[base].I - offset; reg[base].I = address; CPUWriteByte(address, reg[dest].B.B0); - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x764: case 0x76c: { // STRB Rd, [Rn, -Rm, ASR #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset; if(shift) @@ -4952,13 +6026,16 @@ if(cond_res) { u32 address = reg[base].I - offset; reg[base].I = address; CPUWriteByte(address, reg[dest].B.B0); - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x766: case 0x76e: { // STRB Rd, [Rn, -Rm, ROR #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; u32 value = reg[opcode & 15].I; if(shift) { @@ -4971,38 +6048,47 @@ if(cond_res) { u32 address = reg[base].I - value; reg[base].I = address; CPUWriteByte(address, reg[dest].B.B0); - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x7c0: case 0x7c8: { // STRB Rd, [Rn, Rm, LSL #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = reg[opcode & 15].I << ((opcode>>7)& 31); int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I + offset; CPUWriteByte(address, reg[dest].B.B0); - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x7c2: case 0x7ca: { // STRB Rd, [Rn, Rm, LSR #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset = shift ? reg[opcode & 15].I >> shift : 0; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I + offset; CPUWriteByte(address, reg[dest].B.B0); - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x7c4: case 0x7cc: { // STRB Rd, [Rn, Rm, ASR #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset; if(shift) @@ -5015,13 +6101,16 @@ if(cond_res) { int base = (opcode >> 16) & 15; u32 address = reg[base].I + offset; CPUWriteByte(address, reg[dest].B.B0); - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x7c6: case 0x7ce: { // STRB Rd, [Rn, Rm, ROR #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; u32 value = reg[opcode & 15].I; if(shift) { @@ -5033,26 +6122,32 @@ if(cond_res) { int base = (opcode >> 16) & 15; u32 address = reg[base].I + value; CPUWriteByte(address, reg[dest].B.B0); - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x7e0: case 0x7e8: { // STRB Rd, [Rn, Rm, LSL #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = reg[opcode & 15].I << ((opcode>>7)& 31); int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I + offset; reg[base].I = address; CPUWriteByte(address, reg[dest].B.B0); - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x7e2: case 0x7ea: { // STRB Rd, [Rn, Rm, LSR #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset = shift ? reg[opcode & 15].I >> shift : 0; int dest = (opcode >> 12) & 15; @@ -5060,13 +6155,16 @@ if(cond_res) { u32 address = reg[base].I + offset; reg[base].I = address; CPUWriteByte(address, reg[dest].B.B0); - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x7e4: case 0x7ec: { // STRB Rd, [Rn, Rm, ASR #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset; if(shift) @@ -5080,13 +6178,16 @@ if(cond_res) { u32 address = reg[base].I + offset; reg[base].I = address; CPUWriteByte(address, reg[dest].B.B0); - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x7e6: case 0x7ee: { // STRB Rd, [Rn, Rm, ROR #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; u32 value = reg[opcode & 15].I; if(shift) { @@ -5099,7 +6200,8 @@ if(cond_res) { u32 address = reg[base].I + value; reg[base].I = address; CPUWriteByte(address, reg[dest].B.B0); - clockTicks += 2 + CPUUpdateTicksAccess16(address); + clockTicks = 2 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x650: @@ -5109,6 +6211,8 @@ if(cond_res) { case 0x678: { // LDRB Rd, [Rn], -Rm, LSL # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = reg[opcode & 15].I << ((opcode>>7)& 31); int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; @@ -5116,7 +6220,16 @@ if(cond_res) { reg[dest].I = CPUReadByte(address); if(dest != base) reg[base].I = address - offset; - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x652: @@ -5126,6 +6239,8 @@ if(cond_res) { case 0x67a: { // LDRB Rd, [Rn], -Rm, LSR # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset = shift ? reg[opcode & 15].I >> shift : 0; int dest = (opcode >> 12) & 15; @@ -5134,7 +6249,16 @@ if(cond_res) { reg[dest].I = CPUReadByte(address); if(dest != base) reg[base].I = address - offset; - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x654: @@ -5144,6 +6268,8 @@ if(cond_res) { case 0x67c: { // LDRB Rd, [Rn], -Rm, ASR # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset; if(shift) @@ -5158,7 +6284,16 @@ if(cond_res) { reg[dest].I = CPUReadByte(address); if(dest != base) reg[base].I = address - offset; - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x656: @@ -5168,6 +6303,8 @@ if(cond_res) { case 0x67e: { // LDRB Rd, [Rn], -Rm, ROR # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; u32 value = reg[opcode & 15].I; if(shift) { @@ -5181,7 +6318,16 @@ if(cond_res) { reg[dest].I = CPUReadByte(address); if(dest != base) reg[base].I = address - value; - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x6d0: @@ -5191,6 +6337,8 @@ if(cond_res) { case 0x6f8: { // LDRB Rd, [Rn], Rm, LSL # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = reg[opcode & 15].I << ((opcode>>7)& 31); int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; @@ -5198,7 +6346,16 @@ if(cond_res) { reg[dest].I = CPUReadByte(address); if(dest != base) reg[base].I = address + offset; - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x6d2: @@ -5208,6 +6365,8 @@ if(cond_res) { case 0x6fa: { // LDRB Rd, [Rn], Rm, LSR # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset = shift ? reg[opcode & 15].I >> shift : 0; int dest = (opcode >> 12) & 15; @@ -5216,7 +6375,16 @@ if(cond_res) { reg[dest].I = CPUReadByte(address); if(dest != base) reg[base].I = address + offset; - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x6d4: @@ -5226,6 +6394,8 @@ if(cond_res) { case 0x6fc: { // LDRB Rd, [Rn], Rm, ASR # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset; if(shift) @@ -5240,7 +6410,16 @@ if(cond_res) { reg[dest].I = CPUReadByte(address); if(dest != base) reg[base].I = address + offset; - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x6d6: @@ -5250,6 +6429,8 @@ if(cond_res) { case 0x6fe: { // LDRB Rd, [Rn], Rm, ROR # + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; u32 value = reg[opcode & 15].I; if(shift) { @@ -5263,38 +6444,71 @@ if(cond_res) { reg[dest].I = CPUReadByte(address); if(dest != base) reg[base].I = address + value; - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x750: case 0x758: { // LDRB Rd, [Rn, -Rm, LSL #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = reg[opcode & 15].I << ((opcode>>7)& 31); int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I - offset; reg[dest].I = CPUReadByte(address); - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x752: case 0x75a: { // LDRB Rd, [Rn, -Rm, LSR #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset = shift ? reg[opcode & 15].I >> shift : 0; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I - offset; reg[dest].I = CPUReadByte(address); - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x754: case 0x75c: { // LDRB Rd, [Rn, -Rm, ASR #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset; if(shift) @@ -5307,13 +6521,24 @@ if(cond_res) { int base = (opcode >> 16) & 15; u32 address = reg[base].I - offset; reg[dest].I = CPUReadByte(address); - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x756: case 0x75e: { // LDRB Rd, [Rn, -Rm, ROR #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; u32 value = reg[opcode & 15].I; if(shift) { @@ -5325,13 +6550,24 @@ if(cond_res) { int base = (opcode >> 16) & 15; u32 address = reg[base].I - value; reg[dest].I = CPUReadByte(address); - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x770: case 0x778: { // LDRB Rd, [Rn, -Rm, LSL #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = reg[opcode & 15].I << ((opcode>>7)& 31); int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; @@ -5339,13 +6575,24 @@ if(cond_res) { reg[dest].I = CPUReadByte(address); if(dest != base) reg[base].I = address; - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x772: case 0x77a: { // LDRB Rd, [Rn, -Rm, LSR #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset = shift ? reg[opcode & 15].I >> shift : 0; int dest = (opcode >> 12) & 15; @@ -5354,13 +6601,24 @@ if(cond_res) { reg[dest].I = CPUReadByte(address); if(dest != base) reg[base].I = address; - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x774: case 0x77c: { // LDRB Rd, [Rn, -Rm, ASR #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset; if(shift) @@ -5375,13 +6633,24 @@ if(cond_res) { reg[dest].I = CPUReadByte(address); if(dest != base) reg[base].I = address; - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x776: case 0x77e: { // LDRB Rd, [Rn, -Rm, ROR #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; u32 value = reg[opcode & 15].I; if(shift) { @@ -5395,38 +6664,71 @@ if(cond_res) { reg[dest].I = CPUReadByte(address); if(dest != base) reg[base].I = address; - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x7d0: case 0x7d8: { // LDRB Rd, [Rn, Rm, LSL #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = reg[opcode & 15].I << ((opcode>>7)& 31); int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I + offset; reg[dest].I = CPUReadByte(address); - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x7d2: case 0x7da: { // LDRB Rd, [Rn, Rm, LSR #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset = shift ? reg[opcode & 15].I >> shift : 0; int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; u32 address = reg[base].I + offset; reg[dest].I = CPUReadByte(address); - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x7d4: case 0x7dc: { // LDRB Rd, [Rn, Rm, ASR #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset; if(shift) @@ -5439,13 +6741,24 @@ if(cond_res) { int base = (opcode >> 16) & 15; u32 address = reg[base].I + offset; reg[dest].I = CPUReadByte(address); - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x7d6: case 0x7de: { // LDRB Rd, [Rn, Rm, ROR #] + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; u32 value = reg[opcode & 15].I; if(shift) { @@ -5457,13 +6770,24 @@ if(cond_res) { int base = (opcode >> 16) & 15; u32 address = reg[base].I + value; reg[dest].I = CPUReadByte(address); - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x7f0: case 0x7f8: { // LDRB Rd, [Rn, Rm, LSL #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = reg[opcode & 15].I << ((opcode>>7)& 31); int dest = (opcode >> 12) & 15; int base = (opcode >> 16) & 15; @@ -5471,13 +6795,24 @@ if(cond_res) { reg[dest].I = CPUReadByte(address); if(dest != base) reg[base].I = address; - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x7f2: case 0x7fa: { // LDRB Rd, [Rn, Rm, LSR #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset = shift ? reg[opcode & 15].I >> shift : 0; int dest = (opcode >> 12) & 15; @@ -5486,13 +6821,24 @@ if(cond_res) { reg[dest].I = CPUReadByte(address); if(dest != base) reg[base].I = address; - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x7f4: case 0x7fc: { // LDRB Rd, [Rn, Rm, ASR #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; int offset; if(shift) @@ -5507,13 +6853,24 @@ if(cond_res) { reg[dest].I = CPUReadByte(address); if(dest != base) reg[base].I = address; - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; case 0x7f6: case 0x7fe: { // LDRB Rd, [Rn, Rm, ROR #]! + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int shift = (opcode >> 7) & 31; u32 value = reg[opcode & 15].I; if(shift) { @@ -5527,7 +6884,16 @@ if(cond_res) { reg[dest].I = CPUReadByte(address); if(dest != base) reg[base].I = address; - clockTicks += 3 + CPUUpdateTicksAccess16(address); + clockTicks = 0; + if(dest == 15) { + reg[15].I &= 0xFFFFFFFC; + armNextPC = reg[15].I; + reg[15].I += 4; + ARM_PREFETCH; + clockTicks += 2 + dataTicksAccessSeq32(address) + dataTicksAccessSeq32(address); + } + clockTicks += 3 + dataTicksAccess16(address) + + codeTicksAccess32(armNextPC); } break; #define STMW_REG(val,num) \ @@ -5535,10 +6901,10 @@ if(cond_res) { CPUWriteMemory(address, reg[(num)].I);\ if(!offset) {\ reg[base].I = temp;\ - clockTicks += 1 + CPUUpdateTicksAccess32(address);\ + clockTicks += 1 + dataTicksAccess32(address);\ offset = 1;\ } else {\ - clockTicks += 1 + CPUUpdateTicksAccessSeq32(address);\ + clockTicks += 1 + dataTicksAccessSeq32(address);\ }\ address += 4;\ } @@ -5546,22 +6912,23 @@ if(cond_res) { if(opcode & (val)) {\ CPUWriteMemory(address, reg[(num)].I);\ if(!offset) {\ - clockTicks += 1 + CPUUpdateTicksAccess32(address);\ + clockTicks += 1 + dataTicksAccess32(address);\ offset = 1;\ } else {\ - clockTicks += 1 + CPUUpdateTicksAccessSeq32(address);\ + clockTicks += 1 + dataTicksAccessSeq32(address);\ }\ address += 4;\ } CASE_16(0x800) - // STMDA Rn, {Rlist} { + // STMDA Rn, {Rlist} + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 temp = reg[base].I - 4 * (cpuBitsSet[opcode & 255] + cpuBitsSet[(opcode >> 8) & 255]); u32 address = (temp + 4) & 0xFFFFFFFC; - clockTicks += 2; int offset = 0; STM_REG(1, 0); STM_REG(2, 1); @@ -5581,20 +6948,22 @@ if(cond_res) { if(opcode & 32768) { CPUWriteMemory(address, reg[15].I+4); if(!offset) - clockTicks += 1 + CPUUpdateTicksAccess32(address); + clockTicks += 1 + dataTicksAccess32(address); else - clockTicks += 1 + CPUUpdateTicksAccessSeq32(address); + clockTicks += 1 + dataTicksAccessSeq32(address); } + clockTicks += 1 + codeTicksAccess32(armNextPC); } break; CASE_16(0x820) { // STMDA Rn!, {Rlist} + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 temp = reg[base].I - 4 * (cpuBitsSet[opcode & 255] + cpuBitsSet[(opcode >> 8) & 255]); u32 address = (temp+4) & 0xFFFFFFFC; - clockTicks += 2; int offset = 0; STMW_REG(1, 0); @@ -5615,21 +6984,23 @@ if(cond_res) { if(opcode & 32768) { CPUWriteMemory(address, reg[15].I+4); if(!offset) - clockTicks += 1 + CPUUpdateTicksAccess32(address); + clockTicks += 1 + dataTicksAccess32(address); else - clockTicks += 1 + CPUUpdateTicksAccessSeq32(address); + clockTicks += 1 + dataTicksAccessSeq32(address); reg[base].I = temp; } + clockTicks += 1 + codeTicksAccess32(armNextPC); } break; CASE_16(0x840) { // STMDA Rn, {Rlist}^ + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 temp = reg[base].I - 4 * (cpuBitsSet[opcode & 255] + cpuBitsSet[(opcode >> 8) & 255]); u32 address = (temp+4) & 0xFFFFFFFC; - clockTicks += 2; int offset = 0; STM_REG(1, 0); @@ -5666,20 +7037,22 @@ if(cond_res) { if(opcode & 32768) { CPUWriteMemory(address, reg[15].I+4); if(!offset) - clockTicks += 1 + CPUUpdateTicksAccess32(address); + clockTicks += 1 + dataTicksAccess32(address); else - clockTicks += 1 + CPUUpdateTicksAccessSeq32(address); + clockTicks += 1 + dataTicksAccessSeq32(address); } + clockTicks += 1 + codeTicksAccess32(armNextPC); } break; CASE_16(0x860) { // STMDA Rn!, {Rlist}^ + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 temp = reg[base].I - 4 * (cpuBitsSet[opcode & 255] + cpuBitsSet[(opcode >> 8) & 255]); u32 address = (temp+4) & 0xFFFFFFFC; - clockTicks += 2; int offset = 0; STMW_REG(1, 0); @@ -5716,20 +7089,22 @@ if(cond_res) { if(opcode & 32768) { CPUWriteMemory(address, reg[15].I+4); if(!offset) - clockTicks += 1 + CPUUpdateTicksAccess32(address); + clockTicks += 1 + dataTicksAccess32(address); else - clockTicks += 1 + CPUUpdateTicksAccessSeq32(address); + clockTicks += 1 + dataTicksAccessSeq32(address); reg[base].I = temp; } + clockTicks += 1 + codeTicksAccess32(armNextPC); } break; CASE_16(0x880) { // STMIA Rn, {Rlist} + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 address = reg[base].I & 0xFFFFFFFC; - clockTicks += 2; int offset = 0; STM_REG(1, 0); STM_REG(2, 1); @@ -5749,18 +7124,20 @@ if(cond_res) { if(opcode & 32768) { CPUWriteMemory(address, reg[15].I+4); if(!offset) - clockTicks += 1 + CPUUpdateTicksAccess32(address); + clockTicks += 1 + dataTicksAccess32(address); else - clockTicks += 1 + CPUUpdateTicksAccessSeq32(address); + clockTicks += 1 + dataTicksAccessSeq32(address); } + clockTicks += 1 + codeTicksAccess32(armNextPC); } break; CASE_16(0x8a0) { // STMIA Rn!, {Rlist} + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 address = reg[base].I & 0xFFFFFFFC; - clockTicks += 2; int offset = 0; u32 temp = reg[base].I + 4*(cpuBitsSet[opcode & 0xFF] + cpuBitsSet[(opcode >> 8) & 255]); @@ -5783,18 +7160,20 @@ if(cond_res) { CPUWriteMemory(address, reg[15].I+4); if(!offset) { reg[base].I = temp; - clockTicks += 1 + CPUUpdateTicksAccess32(address); + clockTicks += 1 + dataTicksAccess32(address); } else - clockTicks += 1 + CPUUpdateTicksAccessSeq32(address); + clockTicks += 1 + dataTicksAccessSeq32(address); } + clockTicks += 1 + codeTicksAccess32(armNextPC); } break; CASE_16(0x8c0) { // STMIA Rn, {Rlist}^ + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 address = reg[base].I & 0xFFFFFFFC; - clockTicks += 2; int offset = 0; STM_REG(1, 0); STM_REG(2, 1); @@ -5827,18 +7206,20 @@ if(cond_res) { if(opcode & 32768) { CPUWriteMemory(address, reg[15].I+4); if(!offset) - clockTicks += 1 + CPUUpdateTicksAccess32(address); + clockTicks += 1 + dataTicksAccess32(address); else - clockTicks += 1 + CPUUpdateTicksAccessSeq32(address); + clockTicks += 1 + dataTicksAccessSeq32(address); } + clockTicks += 1 + codeTicksAccess32(armNextPC); } break; CASE_16(0x8e0) { // STMIA Rn!, {Rlist}^ + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 address = reg[base].I & 0xFFFFFFFC; - clockTicks += 2; int offset = 0; u32 temp = reg[base].I + 4*(cpuBitsSet[opcode & 0xFF] + cpuBitsSet[(opcode >> 8) & 255]); @@ -5874,21 +7255,23 @@ if(cond_res) { CPUWriteMemory(address, reg[15].I+4); if(!offset) { reg[base].I = temp; - clockTicks += 1 + CPUUpdateTicksAccess32(address); + clockTicks += 1 + dataTicksAccess32(address); } else - clockTicks += 1 + CPUUpdateTicksAccessSeq32(address); + clockTicks += 1 + dataTicksAccessSeq32(address); } + clockTicks += 1 + codeTicksAccess32(armNextPC); } break; CASE_16(0x900) { // STMDB Rn, {Rlist} + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 temp = reg[base].I - 4 * (cpuBitsSet[opcode & 255] + cpuBitsSet[(opcode >> 8) & 255]); u32 address = temp & 0xFFFFFFFC; - clockTicks += 2; int offset = 0; STM_REG(1, 0); STM_REG(2, 1); @@ -5908,20 +7291,22 @@ if(cond_res) { if(opcode & 32768) { CPUWriteMemory(address, reg[15].I+4); if(!offset) - clockTicks += 1 + CPUUpdateTicksAccess32(address); + clockTicks += 1 + dataTicksAccess32(address); else - clockTicks += 1 + CPUUpdateTicksAccessSeq32(address); + clockTicks += 1 + dataTicksAccessSeq32(address); } + clockTicks += 1 + codeTicksAccess32(armNextPC); } break; CASE_16(0x920) { // STMDB Rn!, {Rlist} + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 temp = reg[base].I - 4 * (cpuBitsSet[opcode & 255] + cpuBitsSet[(opcode >> 8) & 255]); u32 address = temp & 0xFFFFFFFC; - clockTicks += 2; int offset = 0; STMW_REG(1, 0); @@ -5942,21 +7327,23 @@ if(cond_res) { if(opcode & 32768) { CPUWriteMemory(address, reg[15].I+4); if(!offset) - clockTicks += 1 + CPUUpdateTicksAccess32(address); + clockTicks += 1 + dataTicksAccess32(address); else - clockTicks += 1 + CPUUpdateTicksAccessSeq32(address); + clockTicks += 1 + dataTicksAccessSeq32(address); reg[base].I = temp; } + clockTicks += 1 + codeTicksAccess32(armNextPC); } break; CASE_16(0x940) { // STMDB Rn, {Rlist}^ + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 temp = reg[base].I - 4 * (cpuBitsSet[opcode & 255] + cpuBitsSet[(opcode >> 8) & 255]); u32 address = temp & 0xFFFFFFFC; - clockTicks += 2; int offset = 0; STM_REG(1, 0); @@ -5993,20 +7380,22 @@ if(cond_res) { if(opcode & 32768) { CPUWriteMemory(address, reg[15].I+4); if(!offset) - clockTicks += 1 + CPUUpdateTicksAccess32(address); + clockTicks += 1 + dataTicksAccess32(address); else - clockTicks += 1 + CPUUpdateTicksAccessSeq32(address); + clockTicks += 1 + dataTicksAccessSeq32(address); } + clockTicks += 1 + codeTicksAccess32(armNextPC); } break; CASE_16(0x960) { // STMDB Rn!, {Rlist}^ + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 temp = reg[base].I - 4 * (cpuBitsSet[opcode & 255] + cpuBitsSet[(opcode >> 8) & 255]); u32 address = temp & 0xFFFFFFFC; - clockTicks += 2; int offset = 0; STMW_REG(1, 0); @@ -6043,20 +7432,22 @@ if(cond_res) { if(opcode & 32768) { CPUWriteMemory(address, reg[15].I+4); if(!offset) - clockTicks += 1 + CPUUpdateTicksAccess32(address); + clockTicks += 1 + dataTicksAccess32(address); else - clockTicks += 1 + CPUUpdateTicksAccessSeq32(address); + clockTicks += 1 + dataTicksAccessSeq32(address); reg[base].I = temp; } + clockTicks += 1 + codeTicksAccess32(armNextPC); } break; CASE_16(0x980) - // STMIB Rn, {Rlist} { + // STMIB Rn, {Rlist} + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 address = (reg[base].I+4) & 0xFFFFFFFC; - clockTicks += 2; int offset = 0; STM_REG(1, 0); STM_REG(2, 1); @@ -6076,18 +7467,20 @@ if(cond_res) { if(opcode & 32768) { CPUWriteMemory(address, reg[15].I+4); if(!offset) - clockTicks += 1 + CPUUpdateTicksAccess32(address); + clockTicks += 1 + dataTicksAccess32(address); else - clockTicks += 1 + CPUUpdateTicksAccessSeq32(address); + clockTicks += 1 + dataTicksAccessSeq32(address); } + clockTicks += 1 + codeTicksAccess32(armNextPC); } break; CASE_16(0x9a0) { // STMIB Rn!, {Rlist} + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 address = (reg[base].I+4) & 0xFFFFFFFC; - clockTicks += 2; int offset = 0; u32 temp = reg[base].I + 4*(cpuBitsSet[opcode & 0xFF] + cpuBitsSet[(opcode >> 8) & 255]); @@ -6110,18 +7503,20 @@ if(cond_res) { CPUWriteMemory(address, reg[15].I+4); if(!offset) { reg[base].I = temp; - clockTicks += 1 + CPUUpdateTicksAccess32(address); + clockTicks += 1 + dataTicksAccess32(address); } else - clockTicks += 1 + CPUUpdateTicksAccessSeq32(address); + clockTicks += 1 + dataTicksAccessSeq32(address); } + clockTicks += 1 + codeTicksAccess32(armNextPC); } break; CASE_16(0x9c0) { // STMIB Rn, {Rlist}^ + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 address = (reg[base].I+4) & 0xFFFFFFFC; - clockTicks += 2; int offset = 0; STM_REG(1, 0); STM_REG(2, 1); @@ -6154,18 +7549,20 @@ if(cond_res) { if(opcode & 32768) { CPUWriteMemory(address, reg[15].I+4); if(!offset) - clockTicks += 1 + CPUUpdateTicksAccess32(address); + clockTicks += 1 + dataTicksAccess32(address); else - clockTicks += 1 + CPUUpdateTicksAccessSeq32(address); + clockTicks += 1 + dataTicksAccessSeq32(address); } + clockTicks += 1 + codeTicksAccess32(armNextPC); } break; CASE_16(0x9e0) { // STMIB Rn!, {Rlist}^ + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 address = (reg[base].I+4) & 0xFFFFFFFC; - clockTicks += 2; int offset = 0; u32 temp = reg[base].I + 4*(cpuBitsSet[opcode & 0xFF] + cpuBitsSet[(opcode >> 8) & 255]); @@ -6201,10 +7598,11 @@ if(cond_res) { CPUWriteMemory(address, reg[15].I+4); if(!offset) { reg[base].I = temp; - clockTicks += 1 + CPUUpdateTicksAccess32(address); + clockTicks += 1 + dataTicksAccess32(address); } else - clockTicks += 1 + CPUUpdateTicksAccessSeq32(address); + clockTicks += 1 + dataTicksAccessSeq32(address); } + clockTicks += 1 + codeTicksAccess32(armNextPC); } break; @@ -6212,9 +7610,9 @@ if(cond_res) { if(opcode & (val)) {\ reg[(num)].I = CPUReadMemory(address);\ if(offset)\ - clockTicks += 1 + CPUUpdateTicksAccessSeq32(address);\ + clockTicks += 1 + dataTicksAccessSeq32(address);\ else {\ - clockTicks += 1 + CPUUpdateTicksAccess32(address);\ + clockTicks += 1 + dataTicksAccess32(address);\ offset = 1;\ }\ address += 4;\ @@ -6223,11 +7621,13 @@ if(cond_res) { CASE_16(0x810) { // LDMDA Rn, {Rlist} + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 temp = reg[base].I - 4 * (cpuBitsSet[opcode & 255] + cpuBitsSet[(opcode >> 8) & 255]); u32 address = (temp + 4) & 0xFFFFFFFC; - clockTicks += 2; + clockTicks = 0; int offset = 0; LDM_REG(1, 0); LDM_REG(2, 1); @@ -6246,23 +7646,28 @@ if(cond_res) { LDM_REG(16384, 14); if(opcode & 32768) { reg[15].I = CPUReadMemory(address); - if (!offset) - clockTicks += 2 + CPUUpdateTicksAccess32(address); - else - clockTicks += 2 + CPUUpdateTicksAccessSeq32(address); armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + if (!offset) + clockTicks += 1 + dataTicksAccess32(address); + else + clockTicks += 1 + dataTicksAccessSeq32(address); + clockTicks += 1 + codeTicksAccessSeq32(armNextPC); } + clockTicks += 2 + codeTicksAccess32(armNextPC); } break; CASE_16(0x830) { // LDMDA Rn!, {Rlist} + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 temp = reg[base].I - 4 * (cpuBitsSet[opcode & 255] + cpuBitsSet[(opcode >> 8) & 255]); u32 address = (temp + 4) & 0xFFFFFFFC; - clockTicks += 2; + clockTicks = 0; int offset = 0; LDM_REG(1, 0); LDM_REG(2, 1); @@ -6281,13 +7686,17 @@ if(cond_res) { LDM_REG(16384, 14); if(opcode & 32768) { reg[15].I = CPUReadMemory(address); - if (!offset) - clockTicks += 2 + CPUUpdateTicksAccess32(address); - else - clockTicks += 2 + CPUUpdateTicksAccessSeq32(address); + armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + if (!offset) + clockTicks += 1 + dataTicksAccess32(address); + else + clockTicks += 1 + dataTicksAccessSeq32(address); + clockTicks += 1 + codeTicksAccessSeq32(armNextPC); } + clockTicks += 2 + codeTicksAccess32(armNextPC); if(!(opcode & (1 << base))) reg[base].I = temp; } @@ -6295,11 +7704,13 @@ if(cond_res) { CASE_16(0x850) { // LDMDA Rn, {Rlist}^ + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 temp = reg[base].I - 4 * (cpuBitsSet[opcode & 255] + cpuBitsSet[(opcode >> 8) & 255]); u32 address = (temp + 4) & 0xFFFFFFFC; - clockTicks += 2; + clockTicks = 0; int offset = 0; if(opcode & 0x8000) { LDM_REG(1, 0); @@ -6319,18 +7730,16 @@ if(cond_res) { LDM_REG(16384, 14); reg[15].I = CPUReadMemory(address); - if (!offset) - clockTicks += 2 + CPUUpdateTicksAccess32(address); - else - clockTicks += 2 + CPUUpdateTicksAccessSeq32(address); CPUSwitchMode(reg[17].I & 0x1f, false); if(armState) { armNextPC = reg[15].I & 0xFFFFFFFC; reg[15].I = armNextPC + 4; + ARM_PREFETCH; } else { armNextPC = reg[15].I & 0xFFFFFFFE; reg[15].I = armNextPC + 2; + THUMB_PREFETCH; } } else { LDM_REG(1, 0); @@ -6363,17 +7772,25 @@ if(cond_res) { LDM_REG(8192, 13); LDM_REG(16384, 14); } + if (!offset) + clockTicks += 1 + dataTicksAccess32(address); + else + clockTicks += 1 + dataTicksAccessSeq32(address); + clockTicks += 1 + codeTicksAccessSeq32(armNextPC); } + clockTicks += 2 + codeTicksAccess32(armNextPC); } break; CASE_16(0x870) { // LDMDA Rn!, {Rlist}^ + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 temp = reg[base].I - 4 * (cpuBitsSet[opcode & 255] + cpuBitsSet[(opcode >> 8) & 255]); u32 address = (temp + 4) & 0xFFFFFFFC; - clockTicks += 2; + clockTicks = 0; int offset = 0; if(opcode & 0x8000) { LDM_REG(1, 0); @@ -6393,10 +7810,6 @@ if(cond_res) { LDM_REG(16384, 14); reg[15].I = CPUReadMemory(address); - if(!offset) - clockTicks += 2 + CPUUpdateTicksAccess32(address); - else - clockTicks += 2 + CPUUpdateTicksAccessSeq32(address); if(!(opcode & (1 << base))) reg[base].I = temp; @@ -6405,9 +7818,11 @@ if(cond_res) { if(armState) { armNextPC = reg[15].I & 0xFFFFFFFC; reg[15].I = armNextPC + 4; + ARM_PREFETCH; } else { armNextPC = reg[15].I & 0xFFFFFFFE; reg[15].I = armNextPC + 2; + THUMB_PREFETCH; } } else { LDM_REG(1, 0); @@ -6443,16 +7858,24 @@ if(cond_res) { if(!(opcode & (1 << base))) reg[base].I = temp; + if (!offset) + clockTicks += 1 + dataTicksAccess32(address); + else + clockTicks += 1 + dataTicksAccessSeq32(address); + clockTicks += 1 + codeTicksAccessSeq32(armNextPC); } + clockTicks += 2 + codeTicksAccess32(armNextPC); } break; CASE_16(0x890) { // LDMIA Rn, {Rlist} + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 address = reg[base].I & 0xFFFFFFFC; - clockTicks += 2; + clockTicks = 0; int offset = 0; LDM_REG(1, 0); LDM_REG(2, 1); @@ -6471,23 +7894,29 @@ if(cond_res) { LDM_REG(16384, 14); if(opcode & 32768) { reg[15].I = CPUReadMemory(address); - if (!offset) - clockTicks += 2 + CPUUpdateTicksAccess32(address); - else - clockTicks += 2 + CPUUpdateTicksAccessSeq32(address); + armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + if (!offset) + clockTicks += 1 + dataTicksAccess32(address); + else + clockTicks += 1 + dataTicksAccessSeq32(address); + clockTicks += 1 + codeTicksAccessSeq32(armNextPC); } + clockTicks += 2 + codeTicksAccess32(armNextPC); } break; CASE_16(0x8b0) { // LDMIA Rn!, {Rlist} + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 temp = reg[base].I + 4 * (cpuBitsSet[opcode & 255] + cpuBitsSet[(opcode >> 8) & 255]); u32 address = reg[base].I & 0xFFFFFFFC; - clockTicks += 2; + clockTicks = 0; int offset = 0; LDM_REG(1, 0); LDM_REG(2, 1); @@ -6506,13 +7935,17 @@ if(cond_res) { LDM_REG(16384, 14); if(opcode & 32768) { reg[15].I = CPUReadMemory(address); - if (!offset) - clockTicks += 2 + CPUUpdateTicksAccess32(address); - else - clockTicks += 2 + CPUUpdateTicksAccessSeq32(address); + armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + if (!offset) + clockTicks += 1 + dataTicksAccess32(address); + else + clockTicks += 1 + dataTicksAccessSeq32(address); + clockTicks += 1 + codeTicksAccessSeq32(armNextPC); } + clockTicks += 2 + codeTicksAccess32(armNextPC); if(!(opcode & (1 << base))) reg[base].I = temp; } @@ -6520,9 +7953,11 @@ if(cond_res) { CASE_16(0x8d0) { // LDMIA Rn, {Rlist}^ + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 address = reg[base].I & 0xFFFFFFFC; - clockTicks += 2; + clockTicks = 0; int offset = 0; if(opcode & 0x8000) { LDM_REG(1, 0); @@ -6542,18 +7977,16 @@ if(cond_res) { LDM_REG(16384, 14); reg[15].I = CPUReadMemory(address); - if (!offset) - clockTicks += 2 + CPUUpdateTicksAccess32(address); - else - clockTicks += 2 + CPUUpdateTicksAccessSeq32(address); CPUSwitchMode(reg[17].I & 0x1f, false); if(armState) { armNextPC = reg[15].I & 0xFFFFFFFC; reg[15].I = armNextPC + 4; + ARM_PREFETCH; } else { armNextPC = reg[15].I & 0xFFFFFFFE; reg[15].I = armNextPC + 2; + THUMB_PREFETCH; } } else { LDM_REG(1, 0); @@ -6586,17 +8019,25 @@ if(cond_res) { LDM_REG(8192, 13); LDM_REG(16384, 14); } + if (!offset) + clockTicks += 1 + dataTicksAccess32(address); + else + clockTicks += 1 + dataTicksAccessSeq32(address); + clockTicks += 1 + codeTicksAccessSeq32(armNextPC); } + clockTicks += 2 + codeTicksAccess32(armNextPC); } break; CASE_16(0x8f0) { // LDMIA Rn!, {Rlist}^ + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 temp = reg[base].I + 4 * (cpuBitsSet[opcode & 255] + cpuBitsSet[(opcode >> 8) & 255]); u32 address = reg[base].I & 0xFFFFFFFC; - clockTicks += 2; + clockTicks = 0; int offset = 0; if(opcode & 0x8000) { LDM_REG(1, 0); @@ -6616,10 +8057,6 @@ if(cond_res) { LDM_REG(16384, 14); reg[15].I = CPUReadMemory(address); - if(!offset) - clockTicks += 2 + CPUUpdateTicksAccess32(address); - else - clockTicks += 2 + CPUUpdateTicksAccessSeq32(address); if(!(opcode & (1 << base))) reg[base].I = temp; @@ -6628,9 +8065,11 @@ if(cond_res) { if(armState) { armNextPC = reg[15].I & 0xFFFFFFFC; reg[15].I = armNextPC + 4; + ARM_PREFETCH; } else { armNextPC = reg[15].I & 0xFFFFFFFE; reg[15].I = armNextPC + 2; + THUMB_PREFETCH; } } else { LDM_REG(1, 0); @@ -6666,18 +8105,26 @@ if(cond_res) { if(!(opcode & (1 << base))) reg[base].I = temp; + if (!offset) + clockTicks += 1 + dataTicksAccess32(address); + else + clockTicks += 1 + dataTicksAccessSeq32(address); + clockTicks += 1 + codeTicksAccessSeq32(armNextPC); } + clockTicks += 2 + codeTicksAccess32(armNextPC); } break; CASE_16(0x910) { // LDMDB Rn, {Rlist} + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 temp = reg[base].I - 4 * (cpuBitsSet[opcode & 255] + cpuBitsSet[(opcode >> 8) & 255]); u32 address = temp & 0xFFFFFFFC; - clockTicks += 2; + clockTicks = 0; int offset = 0; LDM_REG(1, 0); LDM_REG(2, 1); @@ -6696,23 +8143,29 @@ if(cond_res) { LDM_REG(16384, 14); if(opcode & 32768) { reg[15].I = CPUReadMemory(address); - if (!offset) - clockTicks += 2 + CPUUpdateTicksAccess32(address); - else - clockTicks += 2 + CPUUpdateTicksAccessSeq32(address); + armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + if (!offset) + clockTicks += 1 + dataTicksAccess32(address); + else + clockTicks += 1 + dataTicksAccessSeq32(address); + clockTicks += 1 + codeTicksAccessSeq32(armNextPC); } + clockTicks += 2 + codeTicksAccess32(armNextPC); } break; CASE_16(0x930) { // LDMDB Rn!, {Rlist} + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 temp = reg[base].I - 4 * (cpuBitsSet[opcode & 255] + cpuBitsSet[(opcode >> 8) & 255]); u32 address = temp & 0xFFFFFFFC; - clockTicks += 2; + clockTicks = 0; int offset = 0; LDM_REG(1, 0); LDM_REG(2, 1); @@ -6731,13 +8184,17 @@ if(cond_res) { LDM_REG(16384, 14); if(opcode & 32768) { reg[15].I = CPUReadMemory(address); - if (!offset) - clockTicks += 2 + CPUUpdateTicksAccess32(address); - else - clockTicks += 2 + CPUUpdateTicksAccessSeq32(address); + armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + if (!offset) + clockTicks += 1 + dataTicksAccess32(address); + else + clockTicks += 1 + dataTicksAccessSeq32(address); + clockTicks += 1 + codeTicksAccessSeq32(armNextPC); } + clockTicks += 2 + codeTicksAccess32(armNextPC); if(!(opcode & (1 << base))) reg[base].I = temp; } @@ -6745,11 +8202,13 @@ if(cond_res) { CASE_16(0x950) { // LDMDB Rn, {Rlist}^ + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 temp = reg[base].I - 4 * (cpuBitsSet[opcode & 255] + cpuBitsSet[(opcode >> 8) & 255]); u32 address = temp & 0xFFFFFFFC; - clockTicks += 2; + clockTicks = 0; int offset = 0; if(opcode & 0x8000) { LDM_REG(1, 0); @@ -6769,18 +8228,16 @@ if(cond_res) { LDM_REG(16384, 14); reg[15].I = CPUReadMemory(address); - if (!offset) - clockTicks += 2 + CPUUpdateTicksAccess32(address); - else - clockTicks += 2 + CPUUpdateTicksAccessSeq32(address); CPUSwitchMode(reg[17].I & 0x1f, false); if(armState) { armNextPC = reg[15].I & 0xFFFFFFFC; reg[15].I = armNextPC + 4; + ARM_PREFETCH; } else { armNextPC = reg[15].I & 0xFFFFFFFE; reg[15].I = armNextPC + 2; + THUMB_PREFETCH; } } else { LDM_REG(1, 0); @@ -6813,17 +8270,25 @@ if(cond_res) { LDM_REG(8192, 13); LDM_REG(16384, 14); } + if (!offset) + clockTicks += 1 + dataTicksAccess32(address); + else + clockTicks += 1 + dataTicksAccessSeq32(address); + clockTicks += 1 + codeTicksAccessSeq32(armNextPC); } + clockTicks += 2 + codeTicksAccess32(armNextPC); } break; CASE_16(0x970) { // LDMDB Rn!, {Rlist}^ + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 temp = reg[base].I - 4 * (cpuBitsSet[opcode & 255] + cpuBitsSet[(opcode >> 8) & 255]); u32 address = temp & 0xFFFFFFFC; - clockTicks += 2; + clockTicks = 0; int offset = 0; if(opcode & 0x8000) { LDM_REG(1, 0); @@ -6843,10 +8308,6 @@ if(cond_res) { LDM_REG(16384, 14); reg[15].I = CPUReadMemory(address); - if(!offset) - clockTicks += 2 + CPUUpdateTicksAccess32(address); - else - clockTicks += 2 + CPUUpdateTicksAccessSeq32(address); if(!(opcode & (1 << base))) reg[base].I = temp; @@ -6855,9 +8316,11 @@ if(cond_res) { if(armState) { armNextPC = reg[15].I & 0xFFFFFFFC; reg[15].I = armNextPC + 4; + ARM_PREFETCH; } else { armNextPC = reg[15].I & 0xFFFFFFFE; reg[15].I = armNextPC + 2; + THUMB_PREFETCH; } } else { LDM_REG(1, 0); @@ -6893,16 +8356,24 @@ if(cond_res) { if(!(opcode & (1 << base))) reg[base].I = temp; + if (!offset) + clockTicks += 1 + dataTicksAccess32(address); + else + clockTicks += 1 + dataTicksAccessSeq32(address); + clockTicks += 1 + codeTicksAccessSeq32(armNextPC); } + clockTicks += 2 + codeTicksAccess32(armNextPC); } break; CASE_16(0x990) { // LDMIB Rn, {Rlist} + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 address = (reg[base].I+4) & 0xFFFFFFFC; - clockTicks += 2; + clockTicks = 0; int offset = 0; LDM_REG(1, 0); LDM_REG(2, 1); @@ -6921,23 +8392,29 @@ if(cond_res) { LDM_REG(16384, 14); if(opcode & 32768) { reg[15].I = CPUReadMemory(address); - if (!offset) - clockTicks += 2 + CPUUpdateTicksAccess32(address); - else - clockTicks += 2 + CPUUpdateTicksAccessSeq32(address); + armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + if (!offset) + clockTicks += 1 + dataTicksAccess32(address); + else + clockTicks += 1 + dataTicksAccessSeq32(address); + clockTicks += 1 + codeTicksAccessSeq32(armNextPC); } + clockTicks += 2 + codeTicksAccess32(armNextPC); } break; CASE_16(0x9b0) { // LDMIB Rn!, {Rlist} + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 temp = reg[base].I + 4 * (cpuBitsSet[opcode & 255] + cpuBitsSet[(opcode >> 8) & 255]); u32 address = (reg[base].I+4) & 0xFFFFFFFC; - clockTicks += 2; + clockTicks = 0; int offset = 0; LDM_REG(1, 0); LDM_REG(2, 1); @@ -6957,12 +8434,21 @@ if(cond_res) { if(opcode & 32768) { reg[15].I = CPUReadMemory(address); if (!offset) - clockTicks += 2 + CPUUpdateTicksAccess32(address); + clockTicks += 1 + dataTicksAccess32(address); else - clockTicks += 2 + CPUUpdateTicksAccessSeq32(address); + clockTicks += 1 + dataTicksAccessSeq32(address); + clockTicks += 1 + dataTicksAccessSeq32(address); + armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + if (!offset) + clockTicks += 1 + dataTicksAccess32(address); + else + clockTicks += 1 + dataTicksAccessSeq32(address); + clockTicks += 1 + codeTicksAccessSeq32(armNextPC); } + clockTicks += 2 + codeTicksAccess32(armNextPC); if(!(opcode & (1 << base))) reg[base].I = temp; } @@ -6970,9 +8456,11 @@ if(cond_res) { CASE_16(0x9d0) { // LDMIB Rn, {Rlist}^ + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 address = (reg[base].I+4) & 0xFFFFFFFC; - clockTicks += 2; + clockTicks = 0; int offset = 0; if(opcode & 0x8000) { LDM_REG(1, 0); @@ -6992,18 +8480,16 @@ if(cond_res) { LDM_REG(16384, 14); reg[15].I = CPUReadMemory(address); - if (!offset) - clockTicks += 2 + CPUUpdateTicksAccess32(address); - else - clockTicks += 2 + CPUUpdateTicksAccessSeq32(address); CPUSwitchMode(reg[17].I & 0x1f, false); if(armState) { armNextPC = reg[15].I & 0xFFFFFFFC; reg[15].I = armNextPC + 4; + ARM_PREFETCH; } else { armNextPC = reg[15].I & 0xFFFFFFFE; reg[15].I = armNextPC + 2; + THUMB_PREFETCH; } } else { LDM_REG(1, 0); @@ -7036,17 +8522,25 @@ if(cond_res) { LDM_REG(8192, 13); LDM_REG(16384, 14); } + if (!offset) + clockTicks += 1 + dataTicksAccess32(address); + else + clockTicks += 1 + dataTicksAccessSeq32(address); + clockTicks += 1 + codeTicksAccessSeq32(armNextPC); } + clockTicks += 2 + codeTicksAccess32(armNextPC); } break; CASE_16(0x9f0) { // LDMIB Rn!, {Rlist}^ + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int base = (opcode & 0x000F0000) >> 16; u32 temp = reg[base].I + 4 * (cpuBitsSet[opcode & 255] + cpuBitsSet[(opcode >> 8) & 255]); u32 address = (reg[base].I+4) & 0xFFFFFFFC; - clockTicks += 2; + clockTicks = 0; int offset = 0; if(opcode & 0x8000) { LDM_REG(1, 0); @@ -7066,10 +8560,6 @@ if(cond_res) { LDM_REG(16384, 14); reg[15].I = CPUReadMemory(address); - if(!offset) - clockTicks += 2 + CPUUpdateTicksAccess32(address); - else - clockTicks += 2 + CPUUpdateTicksAccessSeq32(address); if(!(opcode & (1 << base))) reg[base].I = temp; @@ -7078,9 +8568,11 @@ if(cond_res) { if(armState) { armNextPC = reg[15].I & 0xFFFFFFFC; reg[15].I = armNextPC + 4; + ARM_PREFETCH; } else { armNextPC = reg[15].I & 0xFFFFFFFE; reg[15].I = armNextPC + 2; + THUMB_PREFETCH; } } else { LDM_REG(1, 0); @@ -7116,13 +8608,18 @@ if(cond_res) { if(!(opcode & (1 << base))) reg[base].I = temp; + if (!offset) + clockTicks += 1 + dataTicksAccess32(address); + else + clockTicks += 1 + dataTicksAccessSeq32(address); + clockTicks += 1 + codeTicksAccessSeq32(armNextPC); } + clockTicks += 2 + codeTicksAccess32(armNextPC); } break; CASE_256(0xa00) { // B - clockTicks += 3; int offset = opcode & 0x00FFFFFF; if(offset & 0x00800000) { offset |= 0xFF000000; @@ -7131,12 +8628,16 @@ if(cond_res) { reg[15].I += offset; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks = codeTicksAccessSeq32(armNextPC) + 1; + clockTicks += 2 + codeTicksAccess32(armNextPC) + + codeTicksAccessSeq32(armNextPC); + busPrefetchCount=0; } break; CASE_256(0xb00) { // BL - clockTicks += 3; int offset = opcode & 0x00FFFFFF; if(offset & 0x00800000) { offset |= 0xFF000000; @@ -7146,12 +8647,21 @@ if(cond_res) { reg[15].I += offset; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks = codeTicksAccessSeq32(armNextPC) + 1; + clockTicks += 2 + codeTicksAccess32(armNextPC) + + codeTicksAccessSeq32(armNextPC); + busPrefetchCount=0; } break; CASE_256(0xf00) // SWI - clockTicks += 3; - CPUSoftwareInterrupt(opcode & 0x00FFFFFF); + clockTicks = codeTicksAccessSeq32(armNextPC) + 1; + clockTicks += 2 + codeTicksAccess32(armNextPC) + + codeTicksAccessSeq32(armNextPC); + busPrefetchCount=0; + CPUSoftwareInterrupt(opcode & 0x00FFFFFF); + break; #ifdef GP_SUPPORT case 0xe11: @@ -7186,3 +8696,6 @@ if(cond_res) { // END } } + +if (clockTicks == 0) + clockTicks = codeTicksAccessSeq32(oldArmNextPC) + 1; diff --git a/src/armdis.cpp b/src/armdis.cpp index 5162952f..e22de54b 100644 --- a/src/armdis.cpp +++ b/src/armdis.cpp @@ -1,6 +1,6 @@ // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -117,8 +117,8 @@ const Opcodes thumbOpcodes[] = { {0xfa00, 0x5800, "ldr%b %r0, [%r3, %r6]"}, // Format 8 {0xfe00, 0x5200, "strh %r0, [%r3, %r6]"}, - {0xfe00, 0x5600, "ldrh %r0, [%r3, %r6]"}, - {0xfe00, 0x5a00, "ldsb %r0, [%r3, %r6]"}, + {0xfe00, 0x5600, "ldsb %r0, [%r3, %r6]"}, + {0xfe00, 0x5a00, "ldrh %r0, [%r3, %r6]"}, {0xfe00, 0x5e00, "ldsh %r0, [%r3, %r6]"}, // Format 9 {0xe800, 0x6000, "str%B %r0, [%r3, %p]"}, diff --git a/src/bios.cpp b/src/bios.cpp index 7876493c..b817ec7e 100644 --- a/src/bios.cpp +++ b/src/bios.cpp @@ -1,1156 +1,1165 @@ -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#include -#include -#include - -#include "GBA.h" -#include "bios.h" -#include "GBAinline.h" -#include "Globals.h" - -s16 sineTable[256] = { - (s16)0x0000, (s16)0x0192, (s16)0x0323, (s16)0x04B5, (s16)0x0645, (s16)0x07D5, (s16)0x0964, (s16)0x0AF1, - (s16)0x0C7C, (s16)0x0E05, (s16)0x0F8C, (s16)0x1111, (s16)0x1294, (s16)0x1413, (s16)0x158F, (s16)0x1708, - (s16)0x187D, (s16)0x19EF, (s16)0x1B5D, (s16)0x1CC6, (s16)0x1E2B, (s16)0x1F8B, (s16)0x20E7, (s16)0x223D, - (s16)0x238E, (s16)0x24DA, (s16)0x261F, (s16)0x275F, (s16)0x2899, (s16)0x29CD, (s16)0x2AFA, (s16)0x2C21, - (s16)0x2D41, (s16)0x2E5A, (s16)0x2F6B, (s16)0x3076, (s16)0x3179, (s16)0x3274, (s16)0x3367, (s16)0x3453, - (s16)0x3536, (s16)0x3612, (s16)0x36E5, (s16)0x37AF, (s16)0x3871, (s16)0x392A, (s16)0x39DA, (s16)0x3A82, - (s16)0x3B20, (s16)0x3BB6, (s16)0x3C42, (s16)0x3CC5, (s16)0x3D3E, (s16)0x3DAE, (s16)0x3E14, (s16)0x3E71, - (s16)0x3EC5, (s16)0x3F0E, (s16)0x3F4E, (s16)0x3F84, (s16)0x3FB1, (s16)0x3FD3, (s16)0x3FEC, (s16)0x3FFB, - (s16)0x4000, (s16)0x3FFB, (s16)0x3FEC, (s16)0x3FD3, (s16)0x3FB1, (s16)0x3F84, (s16)0x3F4E, (s16)0x3F0E, - (s16)0x3EC5, (s16)0x3E71, (s16)0x3E14, (s16)0x3DAE, (s16)0x3D3E, (s16)0x3CC5, (s16)0x3C42, (s16)0x3BB6, - (s16)0x3B20, (s16)0x3A82, (s16)0x39DA, (s16)0x392A, (s16)0x3871, (s16)0x37AF, (s16)0x36E5, (s16)0x3612, - (s16)0x3536, (s16)0x3453, (s16)0x3367, (s16)0x3274, (s16)0x3179, (s16)0x3076, (s16)0x2F6B, (s16)0x2E5A, - (s16)0x2D41, (s16)0x2C21, (s16)0x2AFA, (s16)0x29CD, (s16)0x2899, (s16)0x275F, (s16)0x261F, (s16)0x24DA, - (s16)0x238E, (s16)0x223D, (s16)0x20E7, (s16)0x1F8B, (s16)0x1E2B, (s16)0x1CC6, (s16)0x1B5D, (s16)0x19EF, - (s16)0x187D, (s16)0x1708, (s16)0x158F, (s16)0x1413, (s16)0x1294, (s16)0x1111, (s16)0x0F8C, (s16)0x0E05, - (s16)0x0C7C, (s16)0x0AF1, (s16)0x0964, (s16)0x07D5, (s16)0x0645, (s16)0x04B5, (s16)0x0323, (s16)0x0192, - (s16)0x0000, (s16)0xFE6E, (s16)0xFCDD, (s16)0xFB4B, (s16)0xF9BB, (s16)0xF82B, (s16)0xF69C, (s16)0xF50F, - (s16)0xF384, (s16)0xF1FB, (s16)0xF074, (s16)0xEEEF, (s16)0xED6C, (s16)0xEBED, (s16)0xEA71, (s16)0xE8F8, - (s16)0xE783, (s16)0xE611, (s16)0xE4A3, (s16)0xE33A, (s16)0xE1D5, (s16)0xE075, (s16)0xDF19, (s16)0xDDC3, - (s16)0xDC72, (s16)0xDB26, (s16)0xD9E1, (s16)0xD8A1, (s16)0xD767, (s16)0xD633, (s16)0xD506, (s16)0xD3DF, - (s16)0xD2BF, (s16)0xD1A6, (s16)0xD095, (s16)0xCF8A, (s16)0xCE87, (s16)0xCD8C, (s16)0xCC99, (s16)0xCBAD, - (s16)0xCACA, (s16)0xC9EE, (s16)0xC91B, (s16)0xC851, (s16)0xC78F, (s16)0xC6D6, (s16)0xC626, (s16)0xC57E, - (s16)0xC4E0, (s16)0xC44A, (s16)0xC3BE, (s16)0xC33B, (s16)0xC2C2, (s16)0xC252, (s16)0xC1EC, (s16)0xC18F, - (s16)0xC13B, (s16)0xC0F2, (s16)0xC0B2, (s16)0xC07C, (s16)0xC04F, (s16)0xC02D, (s16)0xC014, (s16)0xC005, - (s16)0xC000, (s16)0xC005, (s16)0xC014, (s16)0xC02D, (s16)0xC04F, (s16)0xC07C, (s16)0xC0B2, (s16)0xC0F2, - (s16)0xC13B, (s16)0xC18F, (s16)0xC1EC, (s16)0xC252, (s16)0xC2C2, (s16)0xC33B, (s16)0xC3BE, (s16)0xC44A, - (s16)0xC4E0, (s16)0xC57E, (s16)0xC626, (s16)0xC6D6, (s16)0xC78F, (s16)0xC851, (s16)0xC91B, (s16)0xC9EE, - (s16)0xCACA, (s16)0xCBAD, (s16)0xCC99, (s16)0xCD8C, (s16)0xCE87, (s16)0xCF8A, (s16)0xD095, (s16)0xD1A6, - (s16)0xD2BF, (s16)0xD3DF, (s16)0xD506, (s16)0xD633, (s16)0xD767, (s16)0xD8A1, (s16)0xD9E1, (s16)0xDB26, - (s16)0xDC72, (s16)0xDDC3, (s16)0xDF19, (s16)0xE075, (s16)0xE1D5, (s16)0xE33A, (s16)0xE4A3, (s16)0xE611, - (s16)0xE783, (s16)0xE8F8, (s16)0xEA71, (s16)0xEBED, (s16)0xED6C, (s16)0xEEEF, (s16)0xF074, (s16)0xF1FB, - (s16)0xF384, (s16)0xF50F, (s16)0xF69C, (s16)0xF82B, (s16)0xF9BB, (s16)0xFB4B, (s16)0xFCDD, (s16)0xFE6E -}; - -void BIOS_ArcTan() -{ -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("ArcTan: %08x (VCOUNT=%2d)\n", - reg[0].I, - VCOUNT); - } -#endif - - s32 a = -(((s32)(reg[0].I*reg[0].I)) >> 14); - s32 b = ((0xA9 * a) >> 14) + 0x390; - b = ((b * a) >> 14) + 0x91C; - b = ((b * a) >> 14) + 0xFB6; - b = ((b * a) >> 14) + 0x16AA; - b = ((b * a) >> 14) + 0x2081; - b = ((b * a) >> 14) + 0x3651; - b = ((b * a) >> 14) + 0xA2F9; - a = ((s32)reg[0].I * b) >> 16; - reg[0].I = a; - -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("ArcTan: return=%08x\n", - reg[0].I); - } -#endif -} - -void BIOS_ArcTan2() -{ -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("ArcTan2: %08x,%08x (VCOUNT=%2d)\n", - reg[0].I, - reg[1].I, - VCOUNT); - } -#endif - - s16 x = reg[0].I; - s16 y = reg[1].I; - s16 res = 0; - if (y == 0) { - res = 0x8000 & x; - } else { - if (x == 0) { - res = (0x8000 & y) + 0x4000; - } else { - if (abs(x) > abs(y)) { - reg[1].I = x; - reg[0].I = y << 14; - BIOS_Div(); - BIOS_ArcTan(); - if (x < 0) - res = 0x8000 + reg[0].I; - else - res = ((y & 0x8000) << 1 ) + reg[0].I; - } else { - reg[0].I = x << 14; - BIOS_Div(); - BIOS_ArcTan(); - res = (0x4000 + (y & 0x8000)) - reg[0].I; - } - } - } - reg[0].I = ((u32)res) & 0xffff; - -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("ArcTan2: return=%08x\n", - reg[0].I); - } -#endif -} - -void BIOS_BitUnPack() -{ -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("BitUnPack: %08x,%08x,%08x (VCOUNT=%2d)\n", - reg[0].I, - reg[1].I, - reg[2].I, - VCOUNT); - } -#endif - - u32 source = reg[0].I; - u32 dest = reg[1].I; - u32 header = reg[2].I; - - int len = CPUReadHalfWord(header); - // check address - int bits = CPUReadByte(header+2); - int revbits = 8 - bits; - // u32 value = 0; - u32 base = CPUReadMemory(header+4); - bool addBase = (base & 0x80000000) ? true : false; - base &= 0x7fffffff; - int dataSize = CPUReadByte(header+3); - - int data = 0; - int bitwritecount = 0; - while(1) { - len -= 1; - if(len < 0) - break; - int mask = 0xff >> revbits; - u8 b = CPUReadByte(source); - source++; - int bitcount = 0; - while(1) { - if(bitcount >= 8) - break; - u32 d = b & mask; - u32 temp = d >> bitcount; - if(!temp && addBase) { - temp += base; - } - data |= temp << bitwritecount; - bitwritecount += dataSize; - if(bitwritecount >= 32) { - CPUWriteMemory(dest, data); - dest += 4; - data = 0; - bitwritecount = 0; - } - mask <<= bits; - bitcount += bits; - } - } -} - -void BIOS_BgAffineSet() -{ -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("BgAffineSet: %08x,%08x,%08x (VCOUNT=%2d)\n", - reg[0].I, - reg[1].I, - reg[2].I, - VCOUNT); - } -#endif - - u32 src = reg[0].I; - u32 dest = reg[1].I; - int num = reg[2].I; - - for(int i = 0; i < num; i++) { - s32 cx = CPUReadMemory(src); - src+=4; - s32 cy = CPUReadMemory(src); - src+=4; - s16 dispx = CPUReadHalfWord(src); - src+=2; - s16 dispy = CPUReadHalfWord(src); - src+=2; - s16 rx = CPUReadHalfWord(src); - src+=2; - s16 ry = CPUReadHalfWord(src); - src+=2; - u16 theta = CPUReadHalfWord(src)>>8; - src+=4; // keep structure alignment - s32 a = sineTable[(theta+0x40)&255]; - s32 b = sineTable[theta]; - - s16 dx = (rx * a)>>14; - s16 dmx = (rx * b)>>14; - s16 dy = (ry * b)>>14; - s16 dmy = (ry * a)>>14; - - CPUWriteHalfWord(dest, dx); - dest += 2; - CPUWriteHalfWord(dest, -dmx); - dest += 2; - CPUWriteHalfWord(dest, dy); - dest += 2; - CPUWriteHalfWord(dest, dmy); - dest += 2; - - s32 startx = cx - dx * dispx + dmx * dispy; - s32 starty = cy - dy * dispx - dmy * dispy; - - CPUWriteMemory(dest, startx); - dest += 4; - CPUWriteMemory(dest, starty); - dest += 4; - } -} - -void BIOS_CpuSet() -{ -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("CpuSet: 0x%08x,0x%08x,0x%08x (VCOUNT=%d)\n", reg[0].I, reg[1].I, - reg[2].I, VCOUNT); - } -#endif - - u32 source = reg[0].I; - u32 dest = reg[1].I; - u32 cnt = reg[2].I; - - if(((source & 0xe000000) == 0) || - ((source + (((cnt << 11)>>9) & 0x1fffff)) & 0xe000000) == 0) - return; - - int count = cnt & 0x1FFFFF; - - // 32-bit ? - if((cnt >> 26) & 1) { - // needed for 32-bit mode! - source &= 0xFFFFFFFC; - dest &= 0xFFFFFFFC; - // fill ? - if((cnt >> 24) & 1) { - u32 value = CPUReadMemory(source); - while(count) { - CPUWriteMemory(dest, value); - dest += 4; - count--; - } - } else { - // copy - while(count) { - CPUWriteMemory(dest, CPUReadMemory(source)); - source += 4; - dest += 4; - count--; - } - } - } else { - // 16-bit fill? - if((cnt >> 24) & 1) { - u16 value = CPUReadHalfWord(source); - while(count) { - CPUWriteHalfWord(dest, value); - dest += 2; - count--; - } - } else { - // copy - while(count) { - CPUWriteHalfWord(dest, CPUReadHalfWord(source)); - source += 2; - dest += 2; - count--; - } - } - } -} - -void BIOS_CpuFastSet() -{ -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("CpuFastSet: 0x%08x,0x%08x,0x%08x (VCOUNT=%d)\n", reg[0].I, reg[1].I, - reg[2].I, VCOUNT); - } -#endif - - u32 source = reg[0].I; - u32 dest = reg[1].I; - u32 cnt = reg[2].I; - - if(((source & 0xe000000) == 0) || - ((source + (((cnt << 11)>>9) & 0x1fffff)) & 0xe000000) == 0) - return; - - // needed for 32-bit mode! - source &= 0xFFFFFFFC; - dest &= 0xFFFFFFFC; - - int count = cnt & 0x1FFFFF; - - // fill? - if((cnt >> 24) & 1) { - while(count > 0) { - // BIOS always transfers 32 bytes at a time - u32 value = CPUReadMemory(source); - for(int i = 0; i < 8; i++) { - CPUWriteMemory(dest, value); - dest += 4; - } - count -= 8; - } - } else { - // copy - while(count > 0) { - // BIOS always transfers 32 bytes at a time - for(int i = 0; i < 8; i++) { - CPUWriteMemory(dest, CPUReadMemory(source)); - source += 4; - dest += 4; - } - count -= 8; - } - } -} - -void BIOS_Diff8bitUnFilterWram() -{ -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("Diff8bitUnFilterWram: 0x%08x,0x%08x (VCOUNT=%d)\n", reg[0].I, - reg[1].I, VCOUNT); - } -#endif - - u32 source = reg[0].I; - u32 dest = reg[1].I; - - u32 header = CPUReadMemory(source); - source += 4; - - if(((source & 0xe000000) == 0) || - ((source + ((header >> 8) & 0x1fffff) & 0xe000000) == 0)) - return; - - int len = header >> 8; - - u8 data = CPUReadByte(source++); - CPUWriteByte(dest++, data); - len--; - - while(len > 0) { - u8 diff = CPUReadByte(source++); - data += diff; - CPUWriteByte(dest++, data); - len--; - } -} - -void BIOS_Diff8bitUnFilterVram() -{ -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("Diff8bitUnFilterVram: 0x%08x,0x%08x (VCOUNT=%d)\n", reg[0].I, - reg[1].I, VCOUNT); - } -#endif - - u32 source = reg[0].I; - u32 dest = reg[1].I; - - u32 header = CPUReadMemory(source); - source += 4; - - if(((source & 0xe000000) == 0) || - ((source + ((header >> 8) & 0x1fffff)) & 0xe000000) == 0) - return; - - int len = header >> 8; - - u8 data = CPUReadByte(source++); - u16 writeData = data; - int shift = 8; - int bytes = 1; - - while(len >= 2) { - u8 diff = CPUReadByte(source++); - data += diff; - writeData |= (data << shift); - bytes++; - shift += 8; - if(bytes == 2) { - CPUWriteHalfWord(dest, writeData); - dest += 2; - len -= 2; - bytes = 0; - writeData = 0; - shift = 0; - } - } -} - -void BIOS_Diff16bitUnFilter() -{ -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("Diff16bitUnFilter: 0x%08x,0x%08x (VCOUNT=%d)\n", reg[0].I, - reg[1].I, VCOUNT); - } -#endif - - u32 source = reg[0].I; - u32 dest = reg[1].I; - - u32 header = CPUReadMemory(source); - source += 4; - - if(((source & 0xe000000) == 0) || - ((source + ((header >> 8) & 0x1fffff)) & 0xe000000) == 0) - return; - - int len = header >> 8; - - u16 data = CPUReadHalfWord(source); - source += 2; - CPUWriteHalfWord(dest, data); - dest += 2; - len -= 2; - - while(len >= 2) { - u16 diff = CPUReadHalfWord(source); - source += 2; - data += diff; - CPUWriteHalfWord(dest, data); - dest += 2; - len -= 2; - } -} - -void BIOS_Div() -{ -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("Div: 0x%08x,0x%08x (VCOUNT=%d)\n", - reg[0].I, - reg[1].I, - VCOUNT); - } -#endif - - int number = reg[0].I; - int denom = reg[1].I; - - if(denom != 0) { - reg[0].I = number / denom; - reg[1].I = number % denom; - s32 temp = (s32)reg[0].I; - reg[3].I = temp < 0 ? (u32)-temp : (u32)temp; - } -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("Div: return=0x%08x,0x%08x,0x%08x\n", - reg[0].I, - reg[1].I, - reg[3].I); - } -#endif -} - -void BIOS_DivARM() -{ -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("DivARM: 0x%08x, (VCOUNT=%d)\n", - reg[0].I, - VCOUNT); - } -#endif - - u32 temp = reg[0].I; - reg[0].I = reg[1].I; - reg[1].I = temp; - BIOS_Div(); -} - -void BIOS_HuffUnComp() -{ -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("HuffUnComp: 0x%08x,0x%08x (VCOUNT=%d)\n", - reg[0].I, - reg[1].I, - VCOUNT); - } -#endif - - u32 source = reg[0].I; - u32 dest = reg[1].I; - - u32 header = CPUReadMemory(source); - source += 4; - - if(((source & 0xe000000) == 0) || - ((source + ((header >> 8) & 0x1fffff)) & 0xe000000) == 0) - return; - - u8 treeSize = CPUReadByte(source++); - - u32 treeStart = source; - - source += ((treeSize+1)<<1)-1; // minus because we already skipped one byte - - int len = header >> 8; - - u32 mask = 0x80000000; - u32 data = CPUReadMemory(source); - source += 4; - - int pos = 0; - u8 rootNode = CPUReadByte(treeStart); - u8 currentNode = rootNode; - bool writeData = false; - int byteShift = 0; - int byteCount = 0; - u32 writeValue = 0; - - if((header & 0x0F) == 8) { - while(len > 0) { - // take left - if(pos == 0) - pos++; - else - pos += (((currentNode & 0x3F)+1)<<1); - - if(data & mask) { - // right - if(currentNode & 0x40) - writeData = true; - currentNode = CPUReadByte(treeStart+pos+1); - } else { - // left - if(currentNode & 0x80) - writeData = true; - currentNode = CPUReadByte(treeStart+pos); - } - - if(writeData) { - writeValue |= (currentNode << byteShift); - byteCount++; - byteShift += 8; - - pos = 0; - currentNode = rootNode; - writeData = false; - - if(byteCount == 4) { - byteCount = 0; - byteShift = 0; - CPUWriteMemory(dest, writeValue); - writeValue = 0; - dest += 4; - len -= 4; - } - } - mask >>= 1; - if(mask == 0) { - mask = 0x80000000; - data = CPUReadMemory(source); - source += 4; - } - } - } else { - int halfLen = 0; - int value = 0; - while(len > 0) { - // take left - if(pos == 0) - pos++; - else - pos += (((currentNode & 0x3F)+1)<<1); - - if((data & mask)) { - // right - if(currentNode & 0x40) - writeData = true; - currentNode = CPUReadByte(treeStart+pos+1); - } else { - // left - if(currentNode & 0x80) - writeData = true; - currentNode = CPUReadByte(treeStart+pos); - } - - if(writeData) { - if(halfLen == 0) - value |= currentNode; - else - value |= (currentNode<<4); - - halfLen += 4; - if(halfLen == 8) { - writeValue |= (value << byteShift); - byteCount++; - byteShift += 8; - - halfLen = 0; - value = 0; - - if(byteCount == 4) { - byteCount = 0; - byteShift = 0; - CPUWriteMemory(dest, writeValue); - dest += 4; - writeValue = 0; - len -= 4; - } - } - pos = 0; - currentNode = rootNode; - writeData = false; - } - mask >>= 1; - if(mask == 0) { - mask = 0x80000000; - data = CPUReadMemory(source); - source += 4; - } - } - } -} - -void BIOS_LZ77UnCompVram() -{ -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("LZ77UnCompVram: 0x%08x,0x%08x (VCOUNT=%d)\n", - reg[0].I, - reg[1].I, - VCOUNT); - } -#endif - - u32 source = reg[0].I; - u32 dest = reg[1].I; - - u32 header = CPUReadMemory(source); - source += 4; - - if(((source & 0xe000000) == 0) || - ((source + ((header >> 8) & 0x1fffff)) & 0xe000000) == 0) - return; - - int byteCount = 0; - int byteShift = 0; - u32 writeValue = 0; - - int len = header >> 8; - - while(len > 0) { - u8 d = CPUReadByte(source++); - - if(d) { - for(int i = 0; i < 8; i++) { - if(d & 0x80) { - u16 data = CPUReadByte(source++) << 8; - data |= CPUReadByte(source++); - int length = (data >> 12) + 3; - int offset = (data & 0x0FFF); - u32 windowOffset = dest + byteCount - offset - 1; - for(int i = 0; i < length; i++) { - writeValue |= (CPUReadByte(windowOffset++) << byteShift); - byteShift += 8; - byteCount++; - - if(byteCount == 2) { - CPUWriteHalfWord(dest, writeValue); - dest += 2; - byteCount = 0; - byteShift = 0; - writeValue = 0; - } - len--; - if(len == 0) - return; - } - } else { - writeValue |= (CPUReadByte(source++) << byteShift); - byteShift += 8; - byteCount++; - if(byteCount == 2) { - CPUWriteHalfWord(dest, writeValue); - dest += 2; - byteCount = 0; - byteShift = 0; - writeValue = 0; - } - len--; - if(len == 0) - return; - } - d <<= 1; - } - } else { - for(int i = 0; i < 8; i++) { - writeValue |= (CPUReadByte(source++) << byteShift); - byteShift += 8; - byteCount++; - if(byteCount == 2) { - CPUWriteHalfWord(dest, writeValue); - dest += 2; - byteShift = 0; - byteCount = 0; - writeValue = 0; - } - len--; - if(len == 0) - return; - } - } - } -} - -void BIOS_LZ77UnCompWram() -{ -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("LZ77UnCompWram: 0x%08x,0x%08x (VCOUNT=%d)\n", reg[0].I, reg[1].I, - VCOUNT); - } -#endif - - u32 source = reg[0].I; - u32 dest = reg[1].I; - - u32 header = CPUReadMemory(source); - source += 4; - - if(((source & 0xe000000) == 0) || - ((source + ((header >> 8) & 0x1fffff)) & 0xe000000) == 0) - return; - - int len = header >> 8; - - while(len > 0) { - u8 d = CPUReadByte(source++); - - if(d) { - for(int i = 0; i < 8; i++) { - if(d & 0x80) { - u16 data = CPUReadByte(source++) << 8; - data |= CPUReadByte(source++); - int length = (data >> 12) + 3; - int offset = (data & 0x0FFF); - u32 windowOffset = dest - offset - 1; - for(int i = 0; i < length; i++) { - CPUWriteByte(dest++, CPUReadByte(windowOffset++)); - len--; - if(len == 0) - return; - } - } else { - CPUWriteByte(dest++, CPUReadByte(source++)); - len--; - if(len == 0) - return; - } - d <<= 1; - } - } else { - for(int i = 0; i < 8; i++) { - CPUWriteByte(dest++, CPUReadByte(source++)); - len--; - if(len == 0) - return; - } - } - } -} - -void BIOS_ObjAffineSet() -{ -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("ObjAffineSet: 0x%08x,0x%08x,0x%08x,0x%08x (VCOUNT=%d)\n", - reg[0].I, - reg[1].I, - reg[2].I, - reg[3].I, - VCOUNT); - } -#endif - - u32 src = reg[0].I; - u32 dest = reg[1].I; - int num = reg[2].I; - int offset = reg[3].I; - - for(int i = 0; i < num; i++) { - s16 rx = CPUReadHalfWord(src); - src+=2; - s16 ry = CPUReadHalfWord(src); - src+=2; - u16 theta = CPUReadHalfWord(src)>>8; - src+=4; // keep structure alignment - - s32 a = (s32)sineTable[(theta+0x40)&255]; - s32 b = (s32)sineTable[theta]; - - s16 dx = ((s32)rx * a)>>14; - s16 dmx = ((s32)rx * b)>>14; - s16 dy = ((s32)ry * b)>>14; - s16 dmy = ((s32)ry * a)>>14; - - CPUWriteHalfWord(dest, dx); - dest += offset; - CPUWriteHalfWord(dest, -dmx); - dest += offset; - CPUWriteHalfWord(dest, dy); - dest += offset; - CPUWriteHalfWord(dest, dmy); - dest += offset; - } -} - -void BIOS_RegisterRamReset(u32 flags) -{ - // no need to trace here. this is only called directly from GBA.cpp - // to emulate bios initialization - - if(flags) { - if(flags & 0x01) { - // clear work RAM - memset(workRAM, 0, 0x40000); - } - if(flags & 0x02) { - // clear internal RAM - memset(internalRAM, 0, 0x7e00); // don't clear 0x7e00-0x7fff - } - if(flags & 0x04) { - // clear palette RAM - memset(paletteRAM, 0, 0x400); - } - if(flags & 0x08) { - // clear VRAM - memset(vram, 0, 0x18000); - } - if(flags & 0x10) { - // clean OAM - memset(oam, 0, 0x400); - } - - if(flags & 0x80) { - int i; - for(i = 0; i < 8; i++) - CPUUpdateRegister(0x200+i*2, 0); - - CPUUpdateRegister(0x202, 0xFFFF); - - for(i = 0; i < 8; i++) - CPUUpdateRegister(0x4+i*2, 0); - - for(i = 0; i < 16; i++) - CPUUpdateRegister(0x20+i*2, 0); - - for(i = 0; i < 24; i++) - CPUUpdateRegister(0xb0+i*2, 0); - - CPUUpdateRegister(0x130, 0); - CPUUpdateRegister(0x20, 0x100); - CPUUpdateRegister(0x30, 0x100); - CPUUpdateRegister(0x26, 0x100); - CPUUpdateRegister(0x36, 0x100); - } - - if(flags & 0x20) { - int i; - for(i = 0; i < 8; i++) - CPUUpdateRegister(0x110+i*2, 0); - CPUUpdateRegister(0x134, 0x8000); - for(i = 0; i < 7; i++) - CPUUpdateRegister(0x140+i*2, 0); - } - - if(flags & 0x40) { - int i; - CPUWriteByte(0x4000084, 0); - CPUWriteByte(0x4000084, 0x80); - CPUWriteMemory(0x4000080, 0x880e0000); - CPUUpdateRegister(0x88, CPUReadHalfWord(0x4000088)&0x3ff); - CPUWriteByte(0x4000070, 0x70); - for(i = 0; i < 8; i++) - CPUUpdateRegister(0x90+i*2, 0); - CPUWriteByte(0x4000070, 0); - for(i = 0; i < 8; i++) - CPUUpdateRegister(0x90+i*2, 0); - CPUWriteByte(0x4000084, 0); - } - } -} - -void BIOS_RegisterRamReset() -{ -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("RegisterRamReset: 0x%08x (VCOUNT=%d)\n", - reg[0].I, - VCOUNT); - } -#endif - - BIOS_RegisterRamReset(reg[0].I); -} - -void BIOS_RLUnCompVram() -{ -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("RLUnCompVram: 0x%08x,0x%08x (VCOUNT=%d)\n", - reg[0].I, - reg[1].I, - VCOUNT); - } -#endif - - u32 source = reg[0].I; - u32 dest = reg[1].I; - - u32 header = CPUReadMemory(source); - source += 4; - - if(((source & 0xe000000) == 0) || - ((source + ((header >> 8) & 0x1fffff)) & 0xe000000) == 0) - return; - - int len = header >> 8; - int byteCount = 0; - int byteShift = 0; - u32 writeValue = 0; - - while(len > 0) { - u8 d = CPUReadByte(source++); - int l = d & 0x7F; - if(d & 0x80) { - u8 data = CPUReadByte(source++); - l += 3; - for(int i = 0;i < l; i++) { - writeValue |= (data << byteShift); - byteShift += 8; - byteCount++; - - if(byteCount == 2) { - CPUWriteHalfWord(dest, writeValue); - dest += 2; - byteCount = 0; - byteShift = 0; - writeValue = 0; - } - len--; - if(len == 0) - return; - } - } else { - l++; - for(int i = 0; i < l; i++) { - writeValue |= (CPUReadByte(source++) << byteShift); - byteShift += 8; - byteCount++; - if(byteCount == 2) { - CPUWriteHalfWord(dest, writeValue); - dest += 2; - byteCount = 0; - byteShift = 0; - writeValue = 0; - } - len--; - if(len == 0) - return; - } - } - } -} - -void BIOS_RLUnCompWram() -{ -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("RLUnCompWram: 0x%08x,0x%08x (VCOUNT=%d)\n", - reg[0].I, - reg[1].I, - VCOUNT); - } -#endif - - u32 source = reg[0].I; - u32 dest = reg[1].I; - - u32 header = CPUReadMemory(source); - source += 4; - - if(((source & 0xe000000) == 0) || - ((source + ((header >> 8) & 0x1fffff)) & 0xe000000) == 0) - return; - - int len = header >> 8; - - while(len > 0) { - u8 d = CPUReadByte(source++); - int l = d & 0x7F; - if(d & 0x80) { - u8 data = CPUReadByte(source++); - l += 3; - for(int i = 0;i < l; i++) { - CPUWriteByte(dest++, data); - len--; - if(len == 0) - return; - } - } else { - l++; - for(int i = 0; i < l; i++) { - CPUWriteByte(dest++, CPUReadByte(source++)); - len--; - if(len == 0) - return; - } - } - } -} - -void BIOS_SoftReset() -{ -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("SoftReset: (VCOUNT=%d)\n", VCOUNT); - } -#endif - - armState = true; - armMode = 0x1F; - armIrqEnable = false; - C_FLAG = V_FLAG = N_FLAG = Z_FLAG = false; - reg[13].I = 0x03007F00; - reg[14].I = 0x00000000; - reg[16].I = 0x00000000; - reg[R13_IRQ].I = 0x03007FA0; - reg[R14_IRQ].I = 0x00000000; - reg[SPSR_IRQ].I = 0x00000000; - reg[R13_SVC].I = 0x03007FE0; - reg[R14_SVC].I = 0x00000000; - reg[SPSR_SVC].I = 0x00000000; - u8 b = internalRAM[0x7ffa]; - - memset(&internalRAM[0x7e00], 0, 0x200); - - if(b) { - armNextPC = 0x02000000; - reg[15].I = 0x02000004; - } else { - armNextPC = 0x08000000; - reg[15].I = 0x08000004; - } -} - -void BIOS_Sqrt() -{ -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("Sqrt: %08x (VCOUNT=%2d)\n", - reg[0].I, - VCOUNT); - } -#endif - reg[0].I = (u32)sqrt((double)reg[0].I); -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("Sqrt: return=%08x\n", - reg[0].I); - } -#endif -} - -void BIOS_MidiKey2Freq() -{ -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("MidiKey2Freq: WaveData=%08x mk=%08x fp=%08x\n", - reg[0].I, - reg[1].I, - reg[2].I); - } -#endif - int freq = CPUReadMemory(reg[0].I+4); - double tmp; - tmp = ((double)(180 - reg[1].I)) - ((double)reg[2].I / 256.f); - tmp = pow((double)2.f, tmp / 12.f); - reg[0].I = (int)((double)freq / tmp); - -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("MidiKey2Freq: return %08x\n", - reg[0].I); - } -#endif -} - -void BIOS_SndDriverJmpTableCopy() -{ -#ifdef DEV_VERSION - if(systemVerbose & VERBOSE_SWI) { - log("SndDriverJmpTableCopy: dest=%08x\n", - reg[0].I); - } -#endif - for(int i = 0; i < 0x24; i++) { - CPUWriteMemory(reg[0].I, 0x9c); - reg[0].I += 4; - } -} +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2005-2006 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#include +#include +#include + +#include "GBA.h" +#include "bios.h" +#include "GBAinline.h" +#include "Globals.h" + +s16 sineTable[256] = { + (s16)0x0000, (s16)0x0192, (s16)0x0323, (s16)0x04B5, (s16)0x0645, (s16)0x07D5, (s16)0x0964, (s16)0x0AF1, + (s16)0x0C7C, (s16)0x0E05, (s16)0x0F8C, (s16)0x1111, (s16)0x1294, (s16)0x1413, (s16)0x158F, (s16)0x1708, + (s16)0x187D, (s16)0x19EF, (s16)0x1B5D, (s16)0x1CC6, (s16)0x1E2B, (s16)0x1F8B, (s16)0x20E7, (s16)0x223D, + (s16)0x238E, (s16)0x24DA, (s16)0x261F, (s16)0x275F, (s16)0x2899, (s16)0x29CD, (s16)0x2AFA, (s16)0x2C21, + (s16)0x2D41, (s16)0x2E5A, (s16)0x2F6B, (s16)0x3076, (s16)0x3179, (s16)0x3274, (s16)0x3367, (s16)0x3453, + (s16)0x3536, (s16)0x3612, (s16)0x36E5, (s16)0x37AF, (s16)0x3871, (s16)0x392A, (s16)0x39DA, (s16)0x3A82, + (s16)0x3B20, (s16)0x3BB6, (s16)0x3C42, (s16)0x3CC5, (s16)0x3D3E, (s16)0x3DAE, (s16)0x3E14, (s16)0x3E71, + (s16)0x3EC5, (s16)0x3F0E, (s16)0x3F4E, (s16)0x3F84, (s16)0x3FB1, (s16)0x3FD3, (s16)0x3FEC, (s16)0x3FFB, + (s16)0x4000, (s16)0x3FFB, (s16)0x3FEC, (s16)0x3FD3, (s16)0x3FB1, (s16)0x3F84, (s16)0x3F4E, (s16)0x3F0E, + (s16)0x3EC5, (s16)0x3E71, (s16)0x3E14, (s16)0x3DAE, (s16)0x3D3E, (s16)0x3CC5, (s16)0x3C42, (s16)0x3BB6, + (s16)0x3B20, (s16)0x3A82, (s16)0x39DA, (s16)0x392A, (s16)0x3871, (s16)0x37AF, (s16)0x36E5, (s16)0x3612, + (s16)0x3536, (s16)0x3453, (s16)0x3367, (s16)0x3274, (s16)0x3179, (s16)0x3076, (s16)0x2F6B, (s16)0x2E5A, + (s16)0x2D41, (s16)0x2C21, (s16)0x2AFA, (s16)0x29CD, (s16)0x2899, (s16)0x275F, (s16)0x261F, (s16)0x24DA, + (s16)0x238E, (s16)0x223D, (s16)0x20E7, (s16)0x1F8B, (s16)0x1E2B, (s16)0x1CC6, (s16)0x1B5D, (s16)0x19EF, + (s16)0x187D, (s16)0x1708, (s16)0x158F, (s16)0x1413, (s16)0x1294, (s16)0x1111, (s16)0x0F8C, (s16)0x0E05, + (s16)0x0C7C, (s16)0x0AF1, (s16)0x0964, (s16)0x07D5, (s16)0x0645, (s16)0x04B5, (s16)0x0323, (s16)0x0192, + (s16)0x0000, (s16)0xFE6E, (s16)0xFCDD, (s16)0xFB4B, (s16)0xF9BB, (s16)0xF82B, (s16)0xF69C, (s16)0xF50F, + (s16)0xF384, (s16)0xF1FB, (s16)0xF074, (s16)0xEEEF, (s16)0xED6C, (s16)0xEBED, (s16)0xEA71, (s16)0xE8F8, + (s16)0xE783, (s16)0xE611, (s16)0xE4A3, (s16)0xE33A, (s16)0xE1D5, (s16)0xE075, (s16)0xDF19, (s16)0xDDC3, + (s16)0xDC72, (s16)0xDB26, (s16)0xD9E1, (s16)0xD8A1, (s16)0xD767, (s16)0xD633, (s16)0xD506, (s16)0xD3DF, + (s16)0xD2BF, (s16)0xD1A6, (s16)0xD095, (s16)0xCF8A, (s16)0xCE87, (s16)0xCD8C, (s16)0xCC99, (s16)0xCBAD, + (s16)0xCACA, (s16)0xC9EE, (s16)0xC91B, (s16)0xC851, (s16)0xC78F, (s16)0xC6D6, (s16)0xC626, (s16)0xC57E, + (s16)0xC4E0, (s16)0xC44A, (s16)0xC3BE, (s16)0xC33B, (s16)0xC2C2, (s16)0xC252, (s16)0xC1EC, (s16)0xC18F, + (s16)0xC13B, (s16)0xC0F2, (s16)0xC0B2, (s16)0xC07C, (s16)0xC04F, (s16)0xC02D, (s16)0xC014, (s16)0xC005, + (s16)0xC000, (s16)0xC005, (s16)0xC014, (s16)0xC02D, (s16)0xC04F, (s16)0xC07C, (s16)0xC0B2, (s16)0xC0F2, + (s16)0xC13B, (s16)0xC18F, (s16)0xC1EC, (s16)0xC252, (s16)0xC2C2, (s16)0xC33B, (s16)0xC3BE, (s16)0xC44A, + (s16)0xC4E0, (s16)0xC57E, (s16)0xC626, (s16)0xC6D6, (s16)0xC78F, (s16)0xC851, (s16)0xC91B, (s16)0xC9EE, + (s16)0xCACA, (s16)0xCBAD, (s16)0xCC99, (s16)0xCD8C, (s16)0xCE87, (s16)0xCF8A, (s16)0xD095, (s16)0xD1A6, + (s16)0xD2BF, (s16)0xD3DF, (s16)0xD506, (s16)0xD633, (s16)0xD767, (s16)0xD8A1, (s16)0xD9E1, (s16)0xDB26, + (s16)0xDC72, (s16)0xDDC3, (s16)0xDF19, (s16)0xE075, (s16)0xE1D5, (s16)0xE33A, (s16)0xE4A3, (s16)0xE611, + (s16)0xE783, (s16)0xE8F8, (s16)0xEA71, (s16)0xEBED, (s16)0xED6C, (s16)0xEEEF, (s16)0xF074, (s16)0xF1FB, + (s16)0xF384, (s16)0xF50F, (s16)0xF69C, (s16)0xF82B, (s16)0xF9BB, (s16)0xFB4B, (s16)0xFCDD, (s16)0xFE6E +}; + +void BIOS_ArcTan() +{ +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("ArcTan: %08x (VCOUNT=%2d)\n", + reg[0].I, + VCOUNT); + } +#endif + + s32 a = -(((s32)(reg[0].I*reg[0].I)) >> 14); + s32 b = ((0xA9 * a) >> 14) + 0x390; + b = ((b * a) >> 14) + 0x91C; + b = ((b * a) >> 14) + 0xFB6; + b = ((b * a) >> 14) + 0x16AA; + b = ((b * a) >> 14) + 0x2081; + b = ((b * a) >> 14) + 0x3651; + b = ((b * a) >> 14) + 0xA2F9; + a = ((s32)reg[0].I * b) >> 16; + reg[0].I = a; + +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("ArcTan: return=%08x\n", + reg[0].I); + } +#endif +} + +void BIOS_ArcTan2() +{ +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("ArcTan2: %08x,%08x (VCOUNT=%2d)\n", + reg[0].I, + reg[1].I, + VCOUNT); + } +#endif + + s32 x = reg[0].I; + s32 y = reg[1].I; + u32 res = 0; + if (y == 0) { + res = ((x>>16) & 0x8000); + } else { + if (x == 0) { + res = ((y>>16) & 0x8000) + 0x4000; + } else { + if ((abs(x) > abs(y)) || ((abs(x) == abs(y)) && (!((x<0) && (y<0))))) { + reg[1].I = x; + reg[0].I = y << 14; + BIOS_Div(); + BIOS_ArcTan(); + if (x < 0) + res = 0x8000 + reg[0].I; + else + res = (((y>>16) & 0x8000)<<1) + reg[0].I; + } else { + reg[0].I = x << 14; + BIOS_Div(); + BIOS_ArcTan(); + res = (0x4000 + ((y>>16) & 0x8000)) - reg[0].I; + } + } + } + reg[0].I = res; + +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("ArcTan2: return=%08x\n", + reg[0].I); + } +#endif +} + +void BIOS_BitUnPack() +{ +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("BitUnPack: %08x,%08x,%08x (VCOUNT=%2d)\n", + reg[0].I, + reg[1].I, + reg[2].I, + VCOUNT); + } +#endif + + u32 source = reg[0].I; + u32 dest = reg[1].I; + u32 header = reg[2].I; + + int len = CPUReadHalfWord(header); + // check address + if(((source & 0xe000000) == 0) || + ((source + len) & 0xe000000) == 0) + return; + + int bits = CPUReadByte(header+2); + int revbits = 8 - bits; + // u32 value = 0; + u32 base = CPUReadMemory(header+4); + bool addBase = (base & 0x80000000) ? true : false; + base &= 0x7fffffff; + int dataSize = CPUReadByte(header+3); + + int data = 0; + int bitwritecount = 0; + while(1) { + len -= 1; + if(len < 0) + break; + int mask = 0xff >> revbits; + u8 b = CPUReadByte(source); + source++; + int bitcount = 0; + while(1) { + if(bitcount >= 8) + break; + u32 d = b & mask; + u32 temp = d >> bitcount; + if(d || addBase) { + temp += base; + } + data |= temp << bitwritecount; + bitwritecount += dataSize; + if(bitwritecount >= 32) { + CPUWriteMemory(dest, data); + dest += 4; + data = 0; + bitwritecount = 0; + } + mask <<= bits; + bitcount += bits; + } + } +} + +void BIOS_GetBiosChecksum() +{ + reg[0].I=0xBAAE187F; +} + +void BIOS_BgAffineSet() +{ +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("BgAffineSet: %08x,%08x,%08x (VCOUNT=%2d)\n", + reg[0].I, + reg[1].I, + reg[2].I, + VCOUNT); + } +#endif + + u32 src = reg[0].I; + u32 dest = reg[1].I; + int num = reg[2].I; + + for(int i = 0; i < num; i++) { + s32 cx = CPUReadMemory(src); + src+=4; + s32 cy = CPUReadMemory(src); + src+=4; + s16 dispx = CPUReadHalfWord(src); + src+=2; + s16 dispy = CPUReadHalfWord(src); + src+=2; + s16 rx = CPUReadHalfWord(src); + src+=2; + s16 ry = CPUReadHalfWord(src); + src+=2; + u16 theta = CPUReadHalfWord(src)>>8; + src+=4; // keep structure alignment + s32 a = sineTable[(theta+0x40)&255]; + s32 b = sineTable[theta]; + + s16 dx = (rx * a)>>14; + s16 dmx = (rx * b)>>14; + s16 dy = (ry * b)>>14; + s16 dmy = (ry * a)>>14; + + CPUWriteHalfWord(dest, dx); + dest += 2; + CPUWriteHalfWord(dest, -dmx); + dest += 2; + CPUWriteHalfWord(dest, dy); + dest += 2; + CPUWriteHalfWord(dest, dmy); + dest += 2; + + s32 startx = cx - dx * dispx + dmx * dispy; + s32 starty = cy - dy * dispx - dmy * dispy; + + CPUWriteMemory(dest, startx); + dest += 4; + CPUWriteMemory(dest, starty); + dest += 4; + } +} + +void BIOS_CpuSet() +{ +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("CpuSet: 0x%08x,0x%08x,0x%08x (VCOUNT=%d)\n", reg[0].I, reg[1].I, + reg[2].I, VCOUNT); + } +#endif + + u32 source = reg[0].I; + u32 dest = reg[1].I; + u32 cnt = reg[2].I; + + if(((source & 0xe000000) == 0) || + ((source + (((cnt << 11)>>9) & 0x1fffff)) & 0xe000000) == 0) + return; + + int count = cnt & 0x1FFFFF; + + // 32-bit ? + if((cnt >> 26) & 1) { + // needed for 32-bit mode! + source &= 0xFFFFFFFC; + dest &= 0xFFFFFFFC; + // fill ? + if((cnt >> 24) & 1) { + u32 value = (source>0x0EFFFFFF ? 0x1CAD1CAD : CPUReadMemory(source)); + while(count) { + CPUWriteMemory(dest, value); + dest += 4; + count--; + } + } else { + // copy + while(count) { + CPUWriteMemory(dest, (source>0x0EFFFFFF ? 0x1CAD1CAD : CPUReadMemory(source))); + source += 4; + dest += 4; + count--; + } + } + } else { + // 16-bit fill? + if((cnt >> 24) & 1) { + u16 value = (source>0x0EFFFFFF ? 0x1CAD : CPUReadHalfWord(source)); + while(count) { + CPUWriteHalfWord(dest, value); + dest += 2; + count--; + } + } else { + // copy + while(count) { + CPUWriteHalfWord(dest, (source>0x0EFFFFFF ? 0x1CAD : CPUReadHalfWord(source))); + source += 2; + dest += 2; + count--; + } + } + } +} + +void BIOS_CpuFastSet() +{ +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("CpuFastSet: 0x%08x,0x%08x,0x%08x (VCOUNT=%d)\n", reg[0].I, reg[1].I, + reg[2].I, VCOUNT); + } +#endif + + u32 source = reg[0].I; + u32 dest = reg[1].I; + u32 cnt = reg[2].I; + + if(((source & 0xe000000) == 0) || + ((source + (((cnt << 11)>>9) & 0x1fffff)) & 0xe000000) == 0) + return; + + // needed for 32-bit mode! + source &= 0xFFFFFFFC; + dest &= 0xFFFFFFFC; + + int count = cnt & 0x1FFFFF; + + // fill? + if((cnt >> 24) & 1) { + while(count > 0) { + // BIOS always transfers 32 bytes at a time + u32 value = (source>0x0EFFFFFF ? 0xBAFFFFFB : CPUReadMemory(source)); + for(int i = 0; i < 8; i++) { + CPUWriteMemory(dest, value); + dest += 4; + } + count -= 8; + } + } else { + // copy + while(count > 0) { + // BIOS always transfers 32 bytes at a time + for(int i = 0; i < 8; i++) { + CPUWriteMemory(dest, (source>0x0EFFFFFF ? 0xBAFFFFFB :CPUReadMemory(source))); + source += 4; + dest += 4; + } + count -= 8; + } + } +} + +void BIOS_Diff8bitUnFilterWram() +{ +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("Diff8bitUnFilterWram: 0x%08x,0x%08x (VCOUNT=%d)\n", reg[0].I, + reg[1].I, VCOUNT); + } +#endif + + u32 source = reg[0].I; + u32 dest = reg[1].I; + + u32 header = CPUReadMemory(source); + source += 4; + + if(((source & 0xe000000) == 0) || + ((source + ((header >> 8) & 0x1fffff) & 0xe000000) == 0)) + return; + + int len = header >> 8; + + u8 data = CPUReadByte(source++); + CPUWriteByte(dest++, data); + len--; + + while(len > 0) { + u8 diff = CPUReadByte(source++); + data += diff; + CPUWriteByte(dest++, data); + len--; + } +} + +void BIOS_Diff8bitUnFilterVram() +{ +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("Diff8bitUnFilterVram: 0x%08x,0x%08x (VCOUNT=%d)\n", reg[0].I, + reg[1].I, VCOUNT); + } +#endif + + u32 source = reg[0].I; + u32 dest = reg[1].I; + + u32 header = CPUReadMemory(source); + source += 4; + + if(((source & 0xe000000) == 0) || + ((source + ((header >> 8) & 0x1fffff)) & 0xe000000) == 0) + return; + + int len = header >> 8; + + u8 data = CPUReadByte(source++); + u16 writeData = data; + int shift = 8; + int bytes = 1; + + while(len >= 2) { + u8 diff = CPUReadByte(source++); + data += diff; + writeData |= (data << shift); + bytes++; + shift += 8; + if(bytes == 2) { + CPUWriteHalfWord(dest, writeData); + dest += 2; + len -= 2; + bytes = 0; + writeData = 0; + shift = 0; + } + } +} + +void BIOS_Diff16bitUnFilter() +{ +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("Diff16bitUnFilter: 0x%08x,0x%08x (VCOUNT=%d)\n", reg[0].I, + reg[1].I, VCOUNT); + } +#endif + + u32 source = reg[0].I; + u32 dest = reg[1].I; + + u32 header = CPUReadMemory(source); + source += 4; + + if(((source & 0xe000000) == 0) || + ((source + ((header >> 8) & 0x1fffff)) & 0xe000000) == 0) + return; + + int len = header >> 8; + + u16 data = CPUReadHalfWord(source); + source += 2; + CPUWriteHalfWord(dest, data); + dest += 2; + len -= 2; + + while(len >= 2) { + u16 diff = CPUReadHalfWord(source); + source += 2; + data += diff; + CPUWriteHalfWord(dest, data); + dest += 2; + len -= 2; + } +} + +void BIOS_Div() +{ +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("Div: 0x%08x,0x%08x (VCOUNT=%d)\n", + reg[0].I, + reg[1].I, + VCOUNT); + } +#endif + + int number = reg[0].I; + int denom = reg[1].I; + + if(denom != 0) { + reg[0].I = number / denom; + reg[1].I = number % denom; + s32 temp = (s32)reg[0].I; + reg[3].I = temp < 0 ? (u32)-temp : (u32)temp; + } +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("Div: return=0x%08x,0x%08x,0x%08x\n", + reg[0].I, + reg[1].I, + reg[3].I); + } +#endif +} + +void BIOS_DivARM() +{ +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("DivARM: 0x%08x, (VCOUNT=%d)\n", + reg[0].I, + VCOUNT); + } +#endif + + u32 temp = reg[0].I; + reg[0].I = reg[1].I; + reg[1].I = temp; + BIOS_Div(); +} + +void BIOS_HuffUnComp() +{ +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("HuffUnComp: 0x%08x,0x%08x (VCOUNT=%d)\n", + reg[0].I, + reg[1].I, + VCOUNT); + } +#endif + + u32 source = reg[0].I; + u32 dest = reg[1].I; + + u32 header = CPUReadMemory(source); + source += 4; + + if(((source & 0xe000000) == 0) || + ((source + ((header >> 8) & 0x1fffff)) & 0xe000000) == 0) + return; + + u8 treeSize = CPUReadByte(source++); + + u32 treeStart = source; + + source += ((treeSize+1)<<1)-1; // minus because we already skipped one byte + + int len = header >> 8; + + u32 mask = 0x80000000; + u32 data = CPUReadMemory(source); + source += 4; + + int pos = 0; + u8 rootNode = CPUReadByte(treeStart); + u8 currentNode = rootNode; + bool writeData = false; + int byteShift = 0; + int byteCount = 0; + u32 writeValue = 0; + + if((header & 0x0F) == 8) { + while(len > 0) { + // take left + if(pos == 0) + pos++; + else + pos += (((currentNode & 0x3F)+1)<<1); + + if(data & mask) { + // right + if(currentNode & 0x40) + writeData = true; + currentNode = CPUReadByte(treeStart+pos+1); + } else { + // left + if(currentNode & 0x80) + writeData = true; + currentNode = CPUReadByte(treeStart+pos); + } + + if(writeData) { + writeValue |= (currentNode << byteShift); + byteCount++; + byteShift += 8; + + pos = 0; + currentNode = rootNode; + writeData = false; + + if(byteCount == 4) { + byteCount = 0; + byteShift = 0; + CPUWriteMemory(dest, writeValue); + writeValue = 0; + dest += 4; + len -= 4; + } + } + mask >>= 1; + if(mask == 0) { + mask = 0x80000000; + data = CPUReadMemory(source); + source += 4; + } + } + } else { + int halfLen = 0; + int value = 0; + while(len > 0) { + // take left + if(pos == 0) + pos++; + else + pos += (((currentNode & 0x3F)+1)<<1); + + if((data & mask)) { + // right + if(currentNode & 0x40) + writeData = true; + currentNode = CPUReadByte(treeStart+pos+1); + } else { + // left + if(currentNode & 0x80) + writeData = true; + currentNode = CPUReadByte(treeStart+pos); + } + + if(writeData) { + if(halfLen == 0) + value |= currentNode; + else + value |= (currentNode<<4); + + halfLen += 4; + if(halfLen == 8) { + writeValue |= (value << byteShift); + byteCount++; + byteShift += 8; + + halfLen = 0; + value = 0; + + if(byteCount == 4) { + byteCount = 0; + byteShift = 0; + CPUWriteMemory(dest, writeValue); + dest += 4; + writeValue = 0; + len -= 4; + } + } + pos = 0; + currentNode = rootNode; + writeData = false; + } + mask >>= 1; + if(mask == 0) { + mask = 0x80000000; + data = CPUReadMemory(source); + source += 4; + } + } + } +} + +void BIOS_LZ77UnCompVram() +{ +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("LZ77UnCompVram: 0x%08x,0x%08x (VCOUNT=%d)\n", + reg[0].I, + reg[1].I, + VCOUNT); + } +#endif + + u32 source = reg[0].I; + u32 dest = reg[1].I; + + u32 header = CPUReadMemory(source); + source += 4; + + if(((source & 0xe000000) == 0) || + ((source + ((header >> 8) & 0x1fffff)) & 0xe000000) == 0) + return; + + int byteCount = 0; + int byteShift = 0; + u32 writeValue = 0; + + int len = header >> 8; + + while(len > 0) { + u8 d = CPUReadByte(source++); + + if(d) { + for(int i = 0; i < 8; i++) { + if(d & 0x80) { + u16 data = CPUReadByte(source++) << 8; + data |= CPUReadByte(source++); + int length = (data >> 12) + 3; + int offset = (data & 0x0FFF); + u32 windowOffset = dest + byteCount - offset - 1; + for(int i = 0; i < length; i++) { + writeValue |= (CPUReadByte(windowOffset++) << byteShift); + byteShift += 8; + byteCount++; + + if(byteCount == 2) { + CPUWriteHalfWord(dest, writeValue); + dest += 2; + byteCount = 0; + byteShift = 0; + writeValue = 0; + } + len--; + if(len == 0) + return; + } + } else { + writeValue |= (CPUReadByte(source++) << byteShift); + byteShift += 8; + byteCount++; + if(byteCount == 2) { + CPUWriteHalfWord(dest, writeValue); + dest += 2; + byteCount = 0; + byteShift = 0; + writeValue = 0; + } + len--; + if(len == 0) + return; + } + d <<= 1; + } + } else { + for(int i = 0; i < 8; i++) { + writeValue |= (CPUReadByte(source++) << byteShift); + byteShift += 8; + byteCount++; + if(byteCount == 2) { + CPUWriteHalfWord(dest, writeValue); + dest += 2; + byteShift = 0; + byteCount = 0; + writeValue = 0; + } + len--; + if(len == 0) + return; + } + } + } +} + +void BIOS_LZ77UnCompWram() +{ +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("LZ77UnCompWram: 0x%08x,0x%08x (VCOUNT=%d)\n", reg[0].I, reg[1].I, + VCOUNT); + } +#endif + + u32 source = reg[0].I; + u32 dest = reg[1].I; + + u32 header = CPUReadMemory(source); + source += 4; + + if(((source & 0xe000000) == 0) || + ((source + ((header >> 8) & 0x1fffff)) & 0xe000000) == 0) + return; + + int len = header >> 8; + + while(len > 0) { + u8 d = CPUReadByte(source++); + + if(d) { + for(int i = 0; i < 8; i++) { + if(d & 0x80) { + u16 data = CPUReadByte(source++) << 8; + data |= CPUReadByte(source++); + int length = (data >> 12) + 3; + int offset = (data & 0x0FFF); + u32 windowOffset = dest - offset - 1; + for(int i = 0; i < length; i++) { + CPUWriteByte(dest++, CPUReadByte(windowOffset++)); + len--; + if(len == 0) + return; + } + } else { + CPUWriteByte(dest++, CPUReadByte(source++)); + len--; + if(len == 0) + return; + } + d <<= 1; + } + } else { + for(int i = 0; i < 8; i++) { + CPUWriteByte(dest++, CPUReadByte(source++)); + len--; + if(len == 0) + return; + } + } + } +} + +void BIOS_ObjAffineSet() +{ +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("ObjAffineSet: 0x%08x,0x%08x,0x%08x,0x%08x (VCOUNT=%d)\n", + reg[0].I, + reg[1].I, + reg[2].I, + reg[3].I, + VCOUNT); + } +#endif + + u32 src = reg[0].I; + u32 dest = reg[1].I; + int num = reg[2].I; + int offset = reg[3].I; + + for(int i = 0; i < num; i++) { + s16 rx = CPUReadHalfWord(src); + src+=2; + s16 ry = CPUReadHalfWord(src); + src+=2; + u16 theta = CPUReadHalfWord(src)>>8; + src+=4; // keep structure alignment + + s32 a = (s32)sineTable[(theta+0x40)&255]; + s32 b = (s32)sineTable[theta]; + + s16 dx = ((s32)rx * a)>>14; + s16 dmx = ((s32)rx * b)>>14; + s16 dy = ((s32)ry * b)>>14; + s16 dmy = ((s32)ry * a)>>14; + + CPUWriteHalfWord(dest, dx); + dest += offset; + CPUWriteHalfWord(dest, -dmx); + dest += offset; + CPUWriteHalfWord(dest, dy); + dest += offset; + CPUWriteHalfWord(dest, dmy); + dest += offset; + } +} + +void BIOS_RegisterRamReset(u32 flags) +{ + // no need to trace here. this is only called directly from GBA.cpp + // to emulate bios initialization + + CPUUpdateRegister(0x0, 0x80); + + if(flags) { + if(flags & 0x01) { + // clear work RAM + memset(workRAM, 0, 0x40000); + } + if(flags & 0x02) { + // clear internal RAM + memset(internalRAM, 0, 0x7e00); // don't clear 0x7e00-0x7fff + } + if(flags & 0x04) { + // clear palette RAM + memset(paletteRAM, 0, 0x400); + } + if(flags & 0x08) { + // clear VRAM + memset(vram, 0, 0x18000); + } + if(flags & 0x10) { + // clean OAM + memset(oam, 0, 0x400); + } + + if(flags & 0x80) { + int i; + for(i = 0; i < 0x10; i++) + CPUUpdateRegister(0x200+i*2, 0); + + for(i = 0; i < 0xF; i++) + CPUUpdateRegister(0x4+i*2, 0); + + for(i = 0; i < 0x20; i++) + CPUUpdateRegister(0x20+i*2, 0); + + for(i = 0; i < 0x18; i++) + CPUUpdateRegister(0xb0+i*2, 0); + + CPUUpdateRegister(0x130, 0); + CPUUpdateRegister(0x20, 0x100); + CPUUpdateRegister(0x30, 0x100); + CPUUpdateRegister(0x26, 0x100); + CPUUpdateRegister(0x36, 0x100); + } + + if(flags & 0x20) { + int i; + for(i = 0; i < 8; i++) + CPUUpdateRegister(0x110+i*2, 0); + CPUUpdateRegister(0x134, 0x8000); + for(i = 0; i < 7; i++) + CPUUpdateRegister(0x140+i*2, 0); + } + + if(flags & 0x40) { + int i; + CPUWriteByte(0x4000084, 0); + CPUWriteByte(0x4000084, 0x80); + CPUWriteMemory(0x4000080, 0x880e0000); + CPUUpdateRegister(0x88, CPUReadHalfWord(0x4000088)&0x3ff); + CPUWriteByte(0x4000070, 0x70); + for(i = 0; i < 8; i++) + CPUUpdateRegister(0x90+i*2, 0); + CPUWriteByte(0x4000070, 0); + for(i = 0; i < 8; i++) + CPUUpdateRegister(0x90+i*2, 0); + CPUWriteByte(0x4000084, 0); + } + } +} + +void BIOS_RegisterRamReset() +{ +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("RegisterRamReset: 0x%08x (VCOUNT=%d)\n", + reg[0].I, + VCOUNT); + } +#endif + + BIOS_RegisterRamReset(reg[0].I); +} + +void BIOS_RLUnCompVram() +{ +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("RLUnCompVram: 0x%08x,0x%08x (VCOUNT=%d)\n", + reg[0].I, + reg[1].I, + VCOUNT); + } +#endif + + u32 source = reg[0].I; + u32 dest = reg[1].I; + + u32 header = CPUReadMemory(source & 0xFFFFFFFC); + source += 4; + + if(((source & 0xe000000) == 0) || + ((source + ((header >> 8) & 0x1fffff)) & 0xe000000) == 0) + return; + + int len = header >> 8; + int byteCount = 0; + int byteShift = 0; + u32 writeValue = 0; + + while(len > 0) { + u8 d = CPUReadByte(source++); + int l = d & 0x7F; + if(d & 0x80) { + u8 data = CPUReadByte(source++); + l += 3; + for(int i = 0;i < l; i++) { + writeValue |= (data << byteShift); + byteShift += 8; + byteCount++; + + if(byteCount == 2) { + CPUWriteHalfWord(dest, writeValue); + dest += 2; + byteCount = 0; + byteShift = 0; + writeValue = 0; + } + len--; + if(len == 0) + return; + } + } else { + l++; + for(int i = 0; i < l; i++) { + writeValue |= (CPUReadByte(source++) << byteShift); + byteShift += 8; + byteCount++; + if(byteCount == 2) { + CPUWriteHalfWord(dest, writeValue); + dest += 2; + byteCount = 0; + byteShift = 0; + writeValue = 0; + } + len--; + if(len == 0) + return; + } + } + } +} + +void BIOS_RLUnCompWram() +{ +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("RLUnCompWram: 0x%08x,0x%08x (VCOUNT=%d)\n", + reg[0].I, + reg[1].I, + VCOUNT); + } +#endif + + u32 source = reg[0].I; + u32 dest = reg[1].I; + + u32 header = CPUReadMemory(source & 0xFFFFFFFC); + source += 4; + + if(((source & 0xe000000) == 0) || + ((source + ((header >> 8) & 0x1fffff)) & 0xe000000) == 0) + return; + + int len = header >> 8; + + while(len > 0) { + u8 d = CPUReadByte(source++); + int l = d & 0x7F; + if(d & 0x80) { + u8 data = CPUReadByte(source++); + l += 3; + for(int i = 0;i < l; i++) { + CPUWriteByte(dest++, data); + len--; + if(len == 0) + return; + } + } else { + l++; + for(int i = 0; i < l; i++) { + CPUWriteByte(dest++, CPUReadByte(source++)); + len--; + if(len == 0) + return; + } + } + } +} + +void BIOS_SoftReset() +{ +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("SoftReset: (VCOUNT=%d)\n", VCOUNT); + } +#endif + + armState = true; + armMode = 0x1F; + armIrqEnable = false; + C_FLAG = V_FLAG = N_FLAG = Z_FLAG = false; + reg[13].I = 0x03007F00; + reg[14].I = 0x00000000; + reg[16].I = 0x00000000; + reg[R13_IRQ].I = 0x03007FA0; + reg[R14_IRQ].I = 0x00000000; + reg[SPSR_IRQ].I = 0x00000000; + reg[R13_SVC].I = 0x03007FE0; + reg[R14_SVC].I = 0x00000000; + reg[SPSR_SVC].I = 0x00000000; + u8 b = internalRAM[0x7ffa]; + + memset(&internalRAM[0x7e00], 0, 0x200); + + if(b) { + armNextPC = 0x02000000; + reg[15].I = 0x02000004; + } else { + armNextPC = 0x08000000; + reg[15].I = 0x08000004; + } +} + +void BIOS_Sqrt() +{ +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("Sqrt: %08x (VCOUNT=%2d)\n", + reg[0].I, + VCOUNT); + } +#endif + reg[0].I = (u32)sqrt((double)reg[0].I); +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("Sqrt: return=%08x\n", + reg[0].I); + } +#endif +} + +void BIOS_MidiKey2Freq() +{ +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("MidiKey2Freq: WaveData=%08x mk=%08x fp=%08x\n", + reg[0].I, + reg[1].I, + reg[2].I); + } +#endif + int freq = CPUReadMemory(reg[0].I+4); + double tmp; + tmp = ((double)(180 - reg[1].I)) - ((double)reg[2].I / 256.f); + tmp = pow((double)2.f, tmp / 12.f); + reg[0].I = (int)((double)freq / tmp); + +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("MidiKey2Freq: return %08x\n", + reg[0].I); + } +#endif +} + +void BIOS_SndDriverJmpTableCopy() +{ +#ifdef DEV_VERSION + if(systemVerbose & VERBOSE_SWI) { + log("SndDriverJmpTableCopy: dest=%08x\n", + reg[0].I); + } +#endif + for(int i = 0; i < 0x24; i++) { + CPUWriteMemory(reg[0].I, 0x9c); + reg[0].I += 4; + } +} diff --git a/src/bios.h b/src/bios.h index 58b3417f..7e78c238 100644 --- a/src/bios.h +++ b/src/bios.h @@ -1,7 +1,7 @@ // -*- C++ -*- // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2004-2006 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -23,6 +23,7 @@ extern void BIOS_ArcTan(); extern void BIOS_ArcTan2(); extern void BIOS_BitUnPack(); +extern void BIOS_GetBiosChecksum(); extern void BIOS_BgAffineSet(); extern void BIOS_CpuSet(); extern void BIOS_CpuFastSet(); diff --git a/src/elf.cpp b/src/elf.cpp index ce6c314e..af92ed97 100644 --- a/src/elf.cpp +++ b/src/elf.cpp @@ -1,2999 +1,2994 @@ -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#include -#include -#include - -#include "GBA.h" -#include "Port.h" -#include "elf.h" -#include "NLS.h" - -#define elfReadMemory(addr) \ - READ32LE((&map[(addr)>>24].address[(addr) & map[(addr)>>24].mask])) - -#define DW_TAG_array_type 0x01 -#define DW_TAG_enumeration_type 0x04 -#define DW_TAG_formal_parameter 0x05 -#define DW_TAG_label 0x0a -#define DW_TAG_lexical_block 0x0b -#define DW_TAG_member 0x0d -#define DW_TAG_pointer_type 0x0f -#define DW_TAG_reference_type 0x10 -#define DW_TAG_compile_unit 0x11 -#define DW_TAG_structure_type 0x13 -#define DW_TAG_subroutine_type 0x15 -#define DW_TAG_typedef 0x16 -#define DW_TAG_union_type 0x17 -#define DW_TAG_unspecified_parameters 0x18 -#define DW_TAG_inheritance 0x1c -#define DW_TAG_inlined_subroutine 0x1d -#define DW_TAG_subrange_type 0x21 -#define DW_TAG_base_type 0x24 -#define DW_TAG_const_type 0x26 -#define DW_TAG_enumerator 0x28 -#define DW_TAG_subprogram 0x2e -#define DW_TAG_variable 0x34 -#define DW_TAG_volatile_type 0x35 - -#define DW_AT_sibling 0x01 -#define DW_AT_location 0x02 -#define DW_AT_name 0x03 -#define DW_AT_byte_size 0x0b -#define DW_AT_bit_offset 0x0c -#define DW_AT_bit_size 0x0d -#define DW_AT_stmt_list 0x10 -#define DW_AT_low_pc 0x11 -#define DW_AT_high_pc 0x12 -#define DW_AT_language 0x13 -#define DW_AT_compdir 0x1b -#define DW_AT_const_value 0x1c -#define DW_AT_containing_type 0x1d -#define DW_AT_inline 0x20 -#define DW_AT_producer 0x25 -#define DW_AT_prototyped 0x27 -#define DW_AT_upper_bound 0x2f -#define DW_AT_abstract_origin 0x31 -#define DW_AT_accessibility 0x32 -#define DW_AT_artificial 0x34 -#define DW_AT_data_member_location 0x38 -#define DW_AT_decl_file 0x3a -#define DW_AT_decl_line 0x3b -#define DW_AT_declaration 0x3c -#define DW_AT_encoding 0x3e -#define DW_AT_external 0x3f -#define DW_AT_frame_base 0x40 -#define DW_AT_macro_info 0x43 -#define DW_AT_specification 0x47 -#define DW_AT_type 0x49 -#define DW_AT_virtuality 0x4c -#define DW_AT_vtable_elem_location 0x4d -// DWARF 2.1/3.0 extensions -#define DW_AT_entry_pc 0x52 -#define DW_AT_ranges 0x55 -// ARM Compiler extensions -#define DW_AT_proc_body 0x2000 -#define DW_AT_save_offset 0x2001 -#define DW_AT_user_2002 0x2002 -// MIPS extensions -#define DW_AT_MIPS_linkage_name 0x2007 - -#define DW_FORM_addr 0x01 -#define DW_FORM_data2 0x05 -#define DW_FORM_data4 0x06 -#define DW_FORM_string 0x08 -#define DW_FORM_block 0x09 -#define DW_FORM_block1 0x0a -#define DW_FORM_data1 0x0b -#define DW_FORM_flag 0x0c -#define DW_FORM_sdata 0x0d -#define DW_FORM_strp 0x0e -#define DW_FORM_udata 0x0f -#define DW_FORM_ref_addr 0x10 -#define DW_FORM_ref4 0x13 -#define DW_FORM_ref_udata 0x15 -#define DW_FORM_indirect 0x16 - -#define DW_OP_addr 0x03 -#define DW_OP_plus_uconst 0x23 -#define DW_OP_reg0 0x50 -#define DW_OP_reg1 0x51 -#define DW_OP_reg2 0x52 -#define DW_OP_reg3 0x53 -#define DW_OP_reg4 0x54 -#define DW_OP_reg5 0x55 -#define DW_OP_reg6 0x56 -#define DW_OP_reg7 0x57 -#define DW_OP_reg8 0x58 -#define DW_OP_reg9 0x59 -#define DW_OP_reg10 0x5a -#define DW_OP_reg11 0x5b -#define DW_OP_reg12 0x5c -#define DW_OP_reg13 0x5d -#define DW_OP_reg14 0x5e -#define DW_OP_reg15 0x5f -#define DW_OP_fbreg 0x91 - -#define DW_LNS_extended_op 0x00 -#define DW_LNS_copy 0x01 -#define DW_LNS_advance_pc 0x02 -#define DW_LNS_advance_line 0x03 -#define DW_LNS_set_file 0x04 -#define DW_LNS_set_column 0x05 -#define DW_LNS_negate_stmt 0x06 -#define DW_LNS_set_basic_block 0x07 -#define DW_LNS_const_add_pc 0x08 -#define DW_LNS_fixed_advance_pc 0x09 - -#define DW_LNE_end_sequence 0x01 -#define DW_LNE_set_address 0x02 -#define DW_LNE_define_file 0x03 - -#define DW_CFA_advance_loc 0x01 -#define DW_CFA_offset 0x02 -#define DW_CFA_restore 0x03 -#define DW_CFA_set_loc 0x01 -#define DW_CFA_advance_loc1 0x02 -#define DW_CFA_advance_loc2 0x03 -#define DW_CFA_advance_loc4 0x04 -#define DW_CFA_offset_extended 0x05 -#define DW_CFA_restore_extended 0x06 -#define DW_CFA_undefined 0x07 -#define DW_CFA_same_value 0x08 -#define DW_CFA_register 0x09 -#define DW_CFA_remember_state 0x0a -#define DW_CFA_restore_state 0x0b -#define DW_CFA_def_cfa 0x0c -#define DW_CFA_def_cfa_register 0x0d -#define DW_CFA_def_cfa_offset 0x0e -#define DW_CFA_nop 0x00 - -#define CASE_TYPE_TAG \ - case DW_TAG_const_type:\ - case DW_TAG_volatile_type:\ - case DW_TAG_pointer_type:\ - case DW_TAG_base_type:\ - case DW_TAG_array_type:\ - case DW_TAG_structure_type:\ - case DW_TAG_union_type:\ - case DW_TAG_typedef:\ - case DW_TAG_subroutine_type:\ - case DW_TAG_enumeration_type:\ - case DW_TAG_enumerator:\ - case DW_TAG_reference_type - -struct ELFcie { - ELFcie *next; - u32 offset; - u8 *augmentation; - u32 codeAlign; - s32 dataAlign; - int returnAddress; - u8 *data; - u32 dataLen; -}; - -struct ELFfde { - ELFcie *cie; - u32 address; - u32 end; - u8 *data; - u32 dataLen; -}; - -enum ELFRegMode { - REG_NOT_SET, - REG_OFFSET, - REG_REGISTER -}; - - -struct ELFFrameStateRegister { - ELFRegMode mode; - int reg; - s32 offset; -}; - -struct ELFFrameStateRegisters { - ELFFrameStateRegister regs[16]; - ELFFrameStateRegisters *previous; -}; - -enum ELFCfaMode { - CFA_NOT_SET, - CFA_REG_OFFSET -}; - -struct ELFFrameState { - ELFFrameStateRegisters registers; - - ELFCfaMode cfaMode; - int cfaRegister; - s32 cfaOffset; - - u32 pc; - - int dataAlign; - int codeAlign; - int returnAddress; -}; - -extern bool cpuIsMultiBoot; - -Symbol *elfSymbols = NULL; -char *elfSymbolsStrTab = NULL; -int elfSymbolsCount = 0; - -ELFSectionHeader **elfSectionHeaders = NULL; -char *elfSectionHeadersStringTable = NULL; -int elfSectionHeadersCount = 0; -u8 *elfFileData = NULL; - -CompileUnit *elfCompileUnits = NULL; -DebugInfo *elfDebugInfo = NULL; -char *elfDebugStrings = NULL; - -ELFcie *elfCies = NULL; -ELFfde **elfFdes = NULL; -int elfFdeCount = 0; - -CompileUnit *elfCurrentUnit = NULL; - -u32 elfRead4Bytes(u8 *); -u16 elfRead2Bytes(u8 *); - -CompileUnit *elfGetCompileUnit(u32 addr) -{ - if(elfCompileUnits) { - CompileUnit *unit = elfCompileUnits; - while(unit) { - if(unit->lowPC) { - if(addr >= unit->lowPC && addr < unit->highPC) - return unit; - } else { - ARanges *r = unit->ranges; - if(r) { - int count = r->count; - for(int j = 0; j < count; j++) { - if(addr >= r->ranges[j].lowPC && addr < r->ranges[j].highPC) - return unit; - } - } - } - unit = unit->next; - } - } - return NULL; -} - -char *elfGetAddressSymbol(u32 addr) -{ - static char buffer[256]; - - CompileUnit *unit = elfGetCompileUnit(addr); - // found unit, need to find function - if(unit) { - Function *func = unit->functions; - while(func) { - if(addr >= func->lowPC && addr < func->highPC) { - int offset = addr - func->lowPC; - char *name = func->name; - if(!name) - name = ""; - if(offset) - sprintf(buffer, "%s+%d", name, offset); - else - strcpy(buffer, name); - return buffer; - } - func = func->next; - } - } - - if(elfSymbolsCount) { - for(int i = 0; i < elfSymbolsCount; i++) { - Symbol *s = &elfSymbols[i]; - if((addr >= s->value) && addr < (s->value+s->size)) { - int offset = addr-s->value; - char *name = s->name; - if(name == NULL) - name = ""; - if(offset) - sprintf(buffer, "%s+%d", name, addr-s->value); - else - strcpy(buffer, name); - return buffer; - } else if(addr == s->value) { - if(s->name) - strcpy(buffer, s->name); - else - strcpy(buffer, ""); - return buffer; - } - } - } - - return ""; -} - -bool elfFindLineInModule(u32 *addr, char *name, int line) -{ - CompileUnit *unit = elfCompileUnits; - - while(unit) { - if(unit->lineInfoTable) { - int i; - int count = unit->lineInfoTable->fileCount; - char *found = NULL; - for(i = 0; i < count; i++) { - if(strcmp(name, unit->lineInfoTable->files[i]) == 0) { - found = unit->lineInfoTable->files[i]; - break; - } - } - // found a matching filename... try to find line now - if(found) { - LineInfoItem *table = unit->lineInfoTable->lines; - count = unit->lineInfoTable->number; - for(i = 0; i < count; i++) { - if(table[i].file == found && table[i].line == line) { - *addr = table[i].address; - return true; - } - } - // we can only find a single match - return false; - } - } - unit = unit->next; - } - return false; -} - -int elfFindLine(CompileUnit *unit, Function * /* func */, u32 addr, char **f) -{ - int currentLine = -1; - if(unit->hasLineInfo) { - int count = unit->lineInfoTable->number; - LineInfoItem *table = unit->lineInfoTable->lines; - int i; - for(i = 0; i < count; i++) { - if(addr <= table[i].address) - break; - } - if(i == count) - i--; - *f = table[i].file; - currentLine = table[i].line; - } - return currentLine; -} - -bool elfFindLineInUnit(u32 *addr, CompileUnit *unit, int line) -{ - if(unit->hasLineInfo) { - int count = unit->lineInfoTable->number; - LineInfoItem *table = unit->lineInfoTable->lines; - int i; - for(i = 0; i < count; i++) { - if(line == table[i].line) { - *addr = table[i].address; - return true; - } - } - } - return false; -} - -bool elfGetCurrentFunction(u32 addr, Function **f, CompileUnit **u) -{ - CompileUnit *unit = elfGetCompileUnit(addr); - // found unit, need to find function - if(unit) { - Function *func = unit->functions; - while(func) { - if(addr >= func->lowPC && addr < func->highPC) { - *f = func; - *u = unit; - return true; - } - func = func->next; - } - } - return false; -} - -bool elfGetObject(char *name, Function *f, CompileUnit *u, Object **o) -{ - if(f && u) { - Object *v = f->variables; - - while(v) { - if(strcmp(name, v->name) == 0) { - *o = v; - return true; - } - v = v->next; - } - v = f->parameters; - while(v) { - if(strcmp(name, v->name) == 0) { - *o = v; - return true; - } - v = v->next; - } - v = u->variables; - while(v) { - if(strcmp(name, v->name) == 0) { - *o = v; - return true; - } - v = v->next; - } - } - - CompileUnit *c = elfCompileUnits; - - while(c) { - if(c != u) { - Object *v = c->variables; - while(v) { - if(strcmp(name, v->name) == 0) { - *o = v; - return true; - } - v = v->next; - } - } - c = c->next; - } - - return false; -} - -char *elfGetSymbol(int i, u32 *value, u32 *size, int *type) -{ - if(i < elfSymbolsCount) { - Symbol *s = &elfSymbols[i]; - *value = s->value; - *size = s->size; - *type = s->type; - return s->name; - } - return NULL; -} - -bool elfGetSymbolAddress(char *sym, u32 *addr, u32 *size, int *type) -{ - if(elfSymbolsCount) { - for(int i = 0; i < elfSymbolsCount; i++) { - Symbol *s = &elfSymbols[i]; - if(strcmp(sym, s->name) == 0) { - *addr = s->value; - *size = s->size; - *type = s->type; - return true; - } - } - } - return false; -} - -ELFfde *elfGetFde(u32 address) -{ - if(elfFdes) { - int i; - for(i = 0; i < elfFdeCount; i++) { - if(address >= elfFdes[i]->address && - address < elfFdes[i]->end) { - return elfFdes[i]; - } - } - } - - return NULL; -} - -void elfExecuteCFAInstructions(ELFFrameState *state, u8 *data, u32 len, - u32 pc) -{ - u8 *end = data + len; - int bytes; - int reg; - ELFFrameStateRegisters *fs; - - while(data < end && state->pc < pc) { - u8 op = *data++; - - switch(op >> 6) { - case DW_CFA_advance_loc: - state->pc += (op & 0x3f) * state->codeAlign; - break; - case DW_CFA_offset: - reg = op & 0x3f; - state->registers.regs[reg].mode = REG_OFFSET; - state->registers.regs[reg].offset = state->dataAlign * - (s32)elfReadLEB128(data, &bytes); - data += bytes; - break; - case DW_CFA_restore: - // we don't care much about the other possible settings, - // so just setting to unset is enough for now - state->registers.regs[op & 0x3f].mode = REG_NOT_SET; - break; - case 0: - switch(op & 0x3f) { - case DW_CFA_nop: - break; - case DW_CFA_advance_loc1: - state->pc += state->codeAlign * (*data++); - break; - case DW_CFA_advance_loc2: - state->pc += state->codeAlign * elfRead2Bytes(data); - data += 2; - break; - case DW_CFA_advance_loc4: - state->pc += state->codeAlign * elfRead4Bytes(data); - data += 4; - break; - case DW_CFA_offset_extended: - reg = elfReadLEB128(data, &bytes); - data += bytes; - state->registers.regs[reg].mode = REG_OFFSET; - state->registers.regs[reg].offset = state->dataAlign * - (s32)elfReadLEB128(data, &bytes); - data += bytes; - break; - case DW_CFA_restore_extended: - case DW_CFA_undefined: - case DW_CFA_same_value: - reg = elfReadLEB128(data, &bytes); - data += bytes; - state->registers.regs[reg].mode = REG_NOT_SET; - break; - case DW_CFA_register: - reg = elfReadLEB128(data, &bytes); - data += bytes; - state->registers.regs[reg].mode = REG_REGISTER; - state->registers.regs[reg].reg = elfReadLEB128(data, &bytes); - data += bytes; - break; - case DW_CFA_remember_state: - fs = (ELFFrameStateRegisters *)calloc(1, - sizeof(ELFFrameStateRegisters)); - memcpy(fs, &state->registers, sizeof(ELFFrameStateRegisters)); - state->registers.previous = fs; - break; - case DW_CFA_restore_state: - if(state->registers.previous == NULL) { - printf("Error: previous frame state is NULL.\n"); - return; - } - fs = state->registers.previous; - memcpy(&state->registers, fs, sizeof(ELFFrameStateRegisters)); - free(fs); - break; - case DW_CFA_def_cfa: - state->cfaRegister = elfReadLEB128(data, &bytes); - data += bytes; - state->cfaOffset = (s32)elfReadLEB128(data, &bytes); - data += bytes; - state->cfaMode = CFA_REG_OFFSET; - break; - case DW_CFA_def_cfa_register: - state->cfaRegister = elfReadLEB128(data, &bytes); - data += bytes; - state->cfaMode = CFA_REG_OFFSET; - break; - case DW_CFA_def_cfa_offset: - state->cfaOffset = (s32)elfReadLEB128(data, &bytes); - data += bytes; - state->cfaMode = CFA_REG_OFFSET; - break; - default: - printf("Unknown CFA opcode %08x\n", op); - return; - } - break; - default: - printf("Unknown CFA opcode %08x\n", op); - return; - } - } -} - -ELFFrameState *elfGetFrameState(ELFfde *fde, u32 address) -{ - ELFFrameState *state = (ELFFrameState *)calloc(1, sizeof(ELFFrameState)); - state->pc = fde->address; - state->dataAlign = fde->cie->dataAlign; - state->codeAlign = fde->cie->codeAlign; - state->returnAddress = fde->cie->returnAddress; - - elfExecuteCFAInstructions(state, - fde->cie->data, - fde->cie->dataLen, - 0xffffffff); - elfExecuteCFAInstructions(state, - fde->data, - fde->dataLen, - address); - - return state; -} - -void elfPrintCallChain(u32 address) -{ - int count = 1; - - reg_pair regs[15]; - reg_pair newRegs[15]; - - memcpy(®s[0], ®[0], sizeof(reg_pair) * 15); - - while(count < 20) { - char *addr = elfGetAddressSymbol(address); - if(*addr == 0) - addr = "???"; - - printf("%08x %s\n", address, addr); - - ELFfde *fde = elfGetFde(address); - - if(fde == NULL) { - break; - } - - ELFFrameState *state = elfGetFrameState(fde, address); - - if(!state) { - break; - } - - if(state->cfaMode == CFA_REG_OFFSET) { - memcpy(&newRegs[0], ®s[0], sizeof(reg_pair) * 15); - u32 addr = 0; - for(int i = 0; i < 15; i++) { - ELFFrameStateRegister *r = &state->registers. - regs[i]; - - switch(r->mode) { - case REG_NOT_SET: - newRegs[i].I = regs[i].I; - break; - case REG_OFFSET: - newRegs[i].I = elfReadMemory(regs[state->cfaRegister].I + - state->cfaOffset + - r->offset); - break; - case REG_REGISTER: - newRegs[i].I = regs[r->reg].I; - break; - default: - printf("Unknown register mode: %d\n", r->mode); - break; - } - } - memcpy(regs, newRegs, sizeof(reg_pair)*15); - addr = newRegs[14].I; - addr &= 0xfffffffe; - address = addr; - count++; - } else { - printf("CFA not set\n"); - break; - } - if(state->registers.previous) { - ELFFrameStateRegisters *prev = state->registers.previous; - - while(prev) { - ELFFrameStateRegisters *p = prev->previous; - free(prev); - prev = p; - } - } - free(state); - } -} - -u32 elfDecodeLocation(Function *f, ELFBlock *o, LocationType *type, u32 base) -{ - u32 framebase = 0; - if(f && f->frameBase) { - ELFBlock *b = f->frameBase; - switch(*b->data) { - case DW_OP_reg0: - case DW_OP_reg1: - case DW_OP_reg2: - case DW_OP_reg3: - case DW_OP_reg4: - case DW_OP_reg5: - case DW_OP_reg6: - case DW_OP_reg7: - case DW_OP_reg8: - case DW_OP_reg9: - case DW_OP_reg10: - case DW_OP_reg11: - case DW_OP_reg12: - case DW_OP_reg13: - case DW_OP_reg14: - case DW_OP_reg15: - framebase = reg[*b->data-0x50].I; - break; - default: - fprintf(stderr, "Unknown frameBase %02x\n", *b->data); - break; - } - } - - ELFBlock *loc = o; - u32 location = 0; - int bytes = 0; - if(loc) { - switch(*loc->data) { - case DW_OP_addr: - location = elfRead4Bytes(loc->data+1); - *type = LOCATION_memory; - break; - case DW_OP_plus_uconst: - location = base + elfReadLEB128(loc->data+1, &bytes); - *type = LOCATION_memory; - break; - case DW_OP_reg0: - case DW_OP_reg1: - case DW_OP_reg2: - case DW_OP_reg3: - case DW_OP_reg4: - case DW_OP_reg5: - case DW_OP_reg6: - case DW_OP_reg7: - case DW_OP_reg8: - case DW_OP_reg9: - case DW_OP_reg10: - case DW_OP_reg11: - case DW_OP_reg12: - case DW_OP_reg13: - case DW_OP_reg14: - case DW_OP_reg15: - location = *loc->data - 0x50; - *type = LOCATION_register; - break; - case DW_OP_fbreg: - { - int bytes; - s32 off = elfReadSignedLEB128(loc->data+1, &bytes); - location = framebase + off; - *type = LOCATION_memory; - } - break; - default: - fprintf(stderr, "Unknown location %02x\n", *loc->data); - break; - } - } - return location; -} - -u32 elfDecodeLocation(Function *f, ELFBlock *o, LocationType *type) -{ - return elfDecodeLocation(f, o, type, 0); -} - -// reading function - -u32 elfRead4Bytes(u8 *data) -{ - u32 value = *data++; - value |= (*data++ << 8); - value |= (*data++ << 16); - value |= (*data << 24); - return value; -} - -u16 elfRead2Bytes(u8 *data) -{ - u16 value = *data++; - value |= (*data << 8); - return value; -} - -char *elfReadString(u8 *data, int *bytesRead) -{ - if(*data == 0) { - *bytesRead = 1; - return NULL; - } - *bytesRead = strlen((char *)data) + 1; - return (char *)data; -} - -s32 elfReadSignedLEB128(u8 *data, int *bytesRead) -{ - s32 result = 0; - int shift = 0; - int count = 0; - - u8 byte; - do { - byte = *data++; - count++; - result |= (byte & 0x7f) << shift; - shift += 7; - } while(byte & 0x80); - if((shift < 32) && (byte & 0x40)) - result |= -(1 << shift); - *bytesRead = count; - return result; -} - -u32 elfReadLEB128(u8 *data, int *bytesRead) -{ - u32 result = 0; - int shift = 0; - int count = 0; - u8 byte; - do { - byte = *data++; - count++; - result |= (byte & 0x7f) << shift; - shift += 7; - } while(byte & 0x80); - *bytesRead = count; - return result; -} - -u8 *elfReadSection(u8 *data, ELFSectionHeader *sh) -{ - return data + READ32LE(&sh->offset); -} - -ELFSectionHeader *elfGetSectionByName(char *name) -{ - for(int i = 0; i < elfSectionHeadersCount; i++) { - if(strcmp(name, - &elfSectionHeadersStringTable[READ32LE(&elfSectionHeaders[i]-> - name)]) == 0) { - return elfSectionHeaders[i]; - } - } - return NULL; -} - -ELFSectionHeader *elfGetSectionByNumber(int number) -{ - if(number < elfSectionHeadersCount) { - return elfSectionHeaders[number]; - } - return NULL; -} - -CompileUnit *elfGetCompileUnitForData(u8 *data) -{ - u8 *end = elfCurrentUnit->top + 4 + elfCurrentUnit->length; - - if(data >= elfCurrentUnit->top && data < end) - return elfCurrentUnit; - - CompileUnit *unit = elfCompileUnits; - - while(unit) { - end = unit->top + 4 + unit->length; - - if(data >= unit->top && data < end) - return unit; - - unit = unit->next; - } - - printf("Error: cannot find reference to compile unit at offset %08x\n", - (int)(data - elfDebugInfo->infodata)); - exit(-1); -} - -u8 *elfReadAttribute(u8 *data, ELFAttr *attr) -{ - int bytes; - int form = attr->form; - start: - switch(form) { - case DW_FORM_addr: - attr->value = elfRead4Bytes(data); - data += 4; - break; - case DW_FORM_data2: - attr->value = elfRead2Bytes(data); - data += 2; - break; - case DW_FORM_data4: - attr->value = elfRead4Bytes(data); - data += 4; - break; - case DW_FORM_string: - attr->string = (char *)data; - data += strlen(attr->string)+1; - break; - case DW_FORM_strp: - attr->string = elfDebugStrings + elfRead4Bytes(data); - data += 4; - break; - case DW_FORM_block: - attr->block = (ELFBlock *)malloc(sizeof(ELFBlock)); - attr->block->length = elfReadLEB128(data, &bytes); - data += bytes; - attr->block->data = data; - data += attr->block->length; - break; - case DW_FORM_block1: - attr->block = (ELFBlock *)malloc(sizeof(ELFBlock)); - attr->block->length = *data++; - attr->block->data = data; - data += attr->block->length; - break; - case DW_FORM_data1: - attr->value = *data++; - break; - case DW_FORM_flag: - attr->flag = (*data++) ? true : false; - break; - case DW_FORM_sdata: - attr->value = elfReadSignedLEB128(data, &bytes); - data += bytes; - break; - case DW_FORM_udata: - attr->value = elfReadLEB128(data, &bytes); - data += bytes; - break; - case DW_FORM_ref_addr: - attr->value = (elfDebugInfo->infodata + elfRead4Bytes(data)) - - elfGetCompileUnitForData(data)->top; - data += 4; - break; - case DW_FORM_ref4: - attr->value = elfRead4Bytes(data); - data += 4; - break; - case DW_FORM_ref_udata: - attr->value = (elfDebugInfo->infodata + - (elfGetCompileUnitForData(data)->top - - elfDebugInfo->infodata) + - elfReadLEB128(data, &bytes)) - - elfCurrentUnit->top; - data += bytes; - break; - case DW_FORM_indirect: - form = elfReadLEB128(data, &bytes); - data += bytes; - goto start; - default: - fprintf(stderr, "Unsupported FORM %02x\n", form); - exit(-1); - } - return data; -} - -ELFAbbrev *elfGetAbbrev(ELFAbbrev **table, u32 number) -{ - int hash = number % 121; - - ELFAbbrev *abbrev = table[hash]; - - while(abbrev) { - if(abbrev->number == number) - return abbrev; - abbrev = abbrev->next; - } - return NULL; -} - -ELFAbbrev **elfReadAbbrevs(u8 *data, u32 offset) -{ - data += offset; - ELFAbbrev **abbrevs = (ELFAbbrev **)calloc(sizeof(ELFAbbrev *)*121,1); - int bytes = 0; - u32 number = elfReadLEB128(data, &bytes); - data += bytes; - while(number) { - ELFAbbrev *abbrev = (ELFAbbrev *)calloc(sizeof(ELFAbbrev),1); - - // read tag information - abbrev->number = number; - abbrev->tag = elfReadLEB128(data, &bytes); - data += bytes; - abbrev->hasChildren = *data++ ? true: false; - - // read attributes - int name = elfReadLEB128(data, &bytes); - data += bytes; - int form = elfReadLEB128(data, &bytes); - data += bytes; - - while(name) { - if((abbrev->numAttrs % 4) == 0) { - abbrev->attrs = (ELFAttr *)realloc(abbrev->attrs, - (abbrev->numAttrs + 4) * - sizeof(ELFAttr)); - } - abbrev->attrs[abbrev->numAttrs].name = name; - abbrev->attrs[abbrev->numAttrs++].form = form; - - name = elfReadLEB128(data, &bytes); - data += bytes; - form = elfReadLEB128(data, &bytes); - data += bytes; - } - - int hash = number % 121; - abbrev->next = abbrevs[hash]; - abbrevs[hash] = abbrev; - - number = elfReadLEB128(data, &bytes); - data += bytes; - - if(elfGetAbbrev(abbrevs, number) != NULL) - break; - } - - return abbrevs; -} - -void elfParseCFA(u8 *top) -{ - ELFSectionHeader *h = elfGetSectionByName(".debug_frame"); - - if(h == NULL) { - return; - } - - u8 *data = elfReadSection(top, h); - - u8 *topOffset = data; - - u8 *end = data + READ32LE(&h->size); - - ELFcie *cies = NULL; - - while(data < end) { - u32 offset = data - topOffset; - u32 len = elfRead4Bytes(data); - data += 4; - - u8 *dataEnd = data + len; - - u32 id = elfRead4Bytes(data); - data += 4; - - if(id == 0xffffffff) { - // skip version - *data++; - - ELFcie *cie = (ELFcie *)calloc(1, sizeof(ELFcie)); - - cie->next = cies; - cies = cie; - - cie->offset = offset; - - cie->augmentation = data; - while(*data) - data++; - data++; - - if(*cie->augmentation) { - fprintf(stderr, "Error: augmentation not supported\n"); - exit(-1); - } - - int bytes; - cie->codeAlign = elfReadLEB128(data, &bytes); - data += bytes; - - cie->dataAlign = elfReadSignedLEB128(data, &bytes); - data += bytes; - - cie->returnAddress = *data++; - - cie->data = data; - cie->dataLen = dataEnd - data; - } else { - ELFfde *fde = (ELFfde *)calloc(1, sizeof(ELFfde)); - - ELFcie *cie = cies; - - while(cie != NULL) { - if(cie->offset == id) - break; - cie = cie->next; - } - - if(!cie) { - fprintf(stderr, "Cannot find CIE %08x\n", id); - exit(-1); - } - - fde->cie = cie; - - fde->address = elfRead4Bytes(data); - data += 4; - - fde->end = fde->address + elfRead4Bytes(data); - data += 4; - - fde->data = data; - fde->dataLen = dataEnd - data; - - if((elfFdeCount %10) == 0) { - elfFdes = (ELFfde **)realloc(elfFdes, (elfFdeCount+10) * - sizeof(ELFfde *)); - } - elfFdes[elfFdeCount++] = fde; - } - data = dataEnd; - } - - elfCies = cies; -} - -void elfAddLine(LineInfo *l, u32 a, int file, int line, int *max) -{ - if(l->number == *max) { - *max += 1000; - l->lines = (LineInfoItem *)realloc(l->lines, *max*sizeof(LineInfoItem)); - } - LineInfoItem *li = &l->lines[l->number]; - li->file = l->files[file-1]; - li->address = a; - li->line = line; - l->number++; -} - -void elfParseLineInfo(CompileUnit *unit, u8 *top) -{ - ELFSectionHeader *h = elfGetSectionByName(".debug_line"); - if(h == NULL) { - fprintf(stderr, "No line information found\n"); - return; - } - LineInfo *l = unit->lineInfoTable = (LineInfo *)calloc(1, sizeof(LineInfo)); - l->number = 0; - int max = 1000; - l->lines = (LineInfoItem *)malloc(1000*sizeof(LineInfoItem)); - - u8 *data = elfReadSection(top, h); - data += unit->lineInfo; - u32 totalLen = elfRead4Bytes(data); - data += 4; - u8 *end = data + totalLen; - // u16 version = elfRead2Bytes(data); - data += 2; - // u32 offset = elfRead4Bytes(data); - data += 4; - int minInstrSize = *data++; - int defaultIsStmt = *data++; - int lineBase = (s8)*data++; - int lineRange = *data++; - int opcodeBase = *data++; - u8 *stdOpLen = (u8 *)malloc(opcodeBase * sizeof(u8)); - stdOpLen[0] = 1; - int i; - for(i = 1; i < opcodeBase; i++) - stdOpLen[i] = *data++; - - free(stdOpLen);// todo - int bytes = 0; - - char *s; - while((s = elfReadString(data, &bytes)) != NULL) { - data += bytes; - // fprintf(stderr, "Directory is %s\n", s); - } - data += bytes; - int count = 4; - int index = 0; - l->files = (char **)malloc(sizeof(char *)*count); - - while((s = elfReadString(data, &bytes)) != NULL) { - l->files[index++] = s; - - data += bytes; - // directory - elfReadLEB128(data, &bytes); - data += bytes; - // time - elfReadLEB128(data, &bytes); - data += bytes; - // size - elfReadLEB128(data, &bytes); - data += bytes; - // fprintf(stderr, "File is %s\n", s); - if(index == count) { - count += 4; - l->files = (char **)realloc(l->files, sizeof(char *)*count); - } - } - l->fileCount = index; - data += bytes; - - while(data < end) { - u32 address = 0; - int file = 1; - int line = 1; - int col = 0; - int isStmt = defaultIsStmt; - int basicBlock = 0; - int endSeq = 0; - - while(!endSeq) { - int op = *data++; - switch(op) { - case DW_LNS_extended_op: - { - data++; - op = *data++; - switch(op) { - case DW_LNE_end_sequence: - endSeq = 1; - break; - case DW_LNE_set_address: - address = elfRead4Bytes(data); - data += 4; - break; - default: - fprintf(stderr, "Unknown extended LINE opcode %02x\n", op); - exit(-1); - } - } - break; - case DW_LNS_copy: - // fprintf(stderr, "Address %08x line %d (%d)\n", address, line, file); - elfAddLine(l, address, file, line, &max); - basicBlock = 0; - break; - case DW_LNS_advance_pc: - address += minInstrSize * elfReadLEB128(data, &bytes); - data += bytes; - break; - case DW_LNS_advance_line: - line += elfReadSignedLEB128(data, &bytes); - data += bytes; - break; - case DW_LNS_set_file: - file = elfReadLEB128(data, &bytes); - data += bytes; - break; - case DW_LNS_set_column: - col = elfReadLEB128(data, &bytes); - data += bytes; - break; - case DW_LNS_negate_stmt: - isStmt = !isStmt; - break; - case DW_LNS_set_basic_block: - basicBlock = 1; - break; - case DW_LNS_const_add_pc: - address += (minInstrSize *((255 - opcodeBase)/lineRange)); - break; - case DW_LNS_fixed_advance_pc: - address += elfRead2Bytes(data); - data += 2; - break; - default: - op = op - opcodeBase; - address += (op / lineRange) * minInstrSize; - line += lineBase + (op % lineRange); - elfAddLine(l, address, file, line, &max); - // fprintf(stderr, "Address %08x line %d (%d)\n", address, line,file); - basicBlock = 1; - break; - } - } - } - l->lines = (LineInfoItem *)realloc(l->lines, l->number*sizeof(LineInfoItem)); -} - -u8 *elfSkipData(u8 *data, ELFAbbrev *abbrev, ELFAbbrev **abbrevs) -{ - int i; - int bytes; - - for(i = 0; i < abbrev->numAttrs; i++) { - data = elfReadAttribute(data, &abbrev->attrs[i]); - if(abbrev->attrs[i].form == DW_FORM_block1) - free(abbrev->attrs[i].block); - } - - if(abbrev->hasChildren) { - int nesting = 1; - while(nesting) { - u32 abbrevNum = elfReadLEB128(data, &bytes); - data += bytes; - - if(!abbrevNum) { - nesting--; - continue; - } - - abbrev = elfGetAbbrev(abbrevs, abbrevNum); - - for(i = 0; i < abbrev->numAttrs; i++) { - data = elfReadAttribute(data, &abbrev->attrs[i]); - if(abbrev->attrs[i].form == DW_FORM_block1) - free(abbrev->attrs[i].block); - } - - if(abbrev->hasChildren) { - nesting++; - } - } - } - return data; -} - -Type *elfParseType(CompileUnit *unit, u32); -u8 *elfParseObject(u8 *data, ELFAbbrev *abbrev, CompileUnit *unit, - Object **object); -u8 *elfParseFunction(u8 *data, ELFAbbrev *abbrev, CompileUnit *unit, - Function **function); -void elfCleanUp(Function *); - -void elfAddType(Type *type, CompileUnit *unit, u32 offset) -{ - if(type->next == NULL) { - if(unit->types != type && type->offset == 0) { - type->offset = offset; - type->next = unit->types; - unit->types = type; - } - } -} - -void elfParseType(u8 *data, u32 offset, ELFAbbrev *abbrev, CompileUnit *unit, - Type **type) -{ - switch(abbrev->tag) { - case DW_TAG_typedef: - { - u32 typeref = 0; - char *name = NULL; - for(int i = 0; i < abbrev->numAttrs; i++) { - ELFAttr *attr = &abbrev->attrs[i]; - data = elfReadAttribute(data, attr); - switch(attr->name) { - case DW_AT_name: - name = attr->string; - break; - case DW_AT_type: - typeref = attr->value; - break; - case DW_AT_decl_file: - case DW_AT_decl_line: - break; - default: - fprintf(stderr, "Unknown attribute for typedef %02x\n", attr->name); - break; - } - } - if(abbrev->hasChildren) - fprintf(stderr, "Unexpected children for typedef\n"); - *type = elfParseType(unit, typeref); - if(name) - (*type)->name = name; - return; - } - break; - case DW_TAG_union_type: - case DW_TAG_structure_type: - { - Type *t = (Type *)calloc(sizeof(Type), 1); - if(abbrev->tag == DW_TAG_structure_type) - t->type = TYPE_struct; - else - t->type = TYPE_union; - - Struct *s = (Struct *)calloc(sizeof(Struct), 1); - t->structure = s; - elfAddType(t, unit, offset); - - for(int i = 0; i < abbrev->numAttrs; i++) { - ELFAttr *attr = &abbrev->attrs[i]; - data = elfReadAttribute(data, attr); - switch(attr->name) { - case DW_AT_name: - t->name = attr->string; - break; - case DW_AT_byte_size: - t->size = attr->value; - break; - case DW_AT_decl_file: - case DW_AT_decl_line: - case DW_AT_sibling: - case DW_AT_containing_type: // todo? - case DW_AT_declaration: - case DW_AT_specification: // TODO: - break; - default: - fprintf(stderr, "Unknown attribute for struct %02x\n", attr->name); - break; - } - } - if(abbrev->hasChildren) { - int bytes; - u32 num = elfReadLEB128(data, &bytes); - data += bytes; - int index = 0; - while(num) { - ELFAbbrev *abbr = elfGetAbbrev(unit->abbrevs, num); - - switch(abbr->tag) { - case DW_TAG_member: - { - if((index % 4) == 0) - s->members = (Member *)realloc(s->members, - sizeof(Member)*(index+4)); - Member *m = &s->members[index]; - m->location = NULL; - m->bitOffset = 0; - m->bitSize = 0; - m->byteSize = 0; - for(int i = 0; i < abbr->numAttrs; i++) { - ELFAttr *attr = &abbr->attrs[i]; - data = elfReadAttribute(data, attr); - switch(attr->name) { - case DW_AT_name: - m->name = attr->string; - break; - case DW_AT_type: - m->type = elfParseType(unit, attr->value); - break; - case DW_AT_data_member_location: - m->location = attr->block; - break; - case DW_AT_byte_size: - m->byteSize = attr->value; - break; - case DW_AT_bit_offset: - m->bitOffset = attr->value; - break; - case DW_AT_bit_size: - m->bitSize = attr->value; - break; - case DW_AT_decl_file: - case DW_AT_decl_line: - case DW_AT_accessibility: - case DW_AT_artificial: // todo? - break; - default: - fprintf(stderr, "Unknown member attribute %02x\n", - attr->name); - } - } - index++; - } - break; - case DW_TAG_subprogram: - { - Function *fnc = NULL; - data = elfParseFunction(data, abbr, unit, &fnc); - if(fnc != NULL) { - if(unit->lastFunction) - unit->lastFunction->next = fnc; - else - unit->functions = fnc; - unit->lastFunction = fnc; - } - } - break; - case DW_TAG_inheritance: - // TODO: add support - data = elfSkipData(data, abbr, unit->abbrevs); - break; - CASE_TYPE_TAG: - // skip types... parsed only when used - data = elfSkipData(data, abbr, unit->abbrevs); - break; - case DW_TAG_variable: - data = elfSkipData(data, abbr, unit->abbrevs); - break; - default: - fprintf(stderr, "Unknown struct tag %02x %s\n", abbr->tag, t->name); - data = elfSkipData(data, abbr, unit->abbrevs); - break; - } - num = elfReadLEB128(data, &bytes); - data += bytes; - } - s->memberCount = index; - } - *type = t; - return; - } - break; - case DW_TAG_base_type: - { - Type *t = (Type *)calloc(sizeof(Type), 1); - - t->type = TYPE_base; - elfAddType(t, unit, offset); - for(int i = 0; i < abbrev->numAttrs; i++) { - ELFAttr *attr = &abbrev->attrs[i]; - data = elfReadAttribute(data, attr); - switch(attr->name) { - case DW_AT_name: - t->name = attr->string; - break; - case DW_AT_encoding: - t->encoding = attr->value; - break; - case DW_AT_byte_size: - t->size = attr->value; - break; - case DW_AT_bit_size: - t->bitSize = attr->value; - break; - default: - fprintf(stderr, "Unknown attribute for base type %02x\n", - attr->name); - break; - } - } - if(abbrev->hasChildren) - fprintf(stderr, "Unexpected children for base type\n"); - *type = t; - return; - } - break; - case DW_TAG_pointer_type: - { - Type *t = (Type *)calloc(sizeof(Type), 1); - - t->type = TYPE_pointer; - - elfAddType(t, unit, offset); - - for(int i = 0; i < abbrev->numAttrs; i++) { - ELFAttr *attr = &abbrev->attrs[i]; - data =elfReadAttribute(data, attr); - switch(attr->name) { - case DW_AT_type: - t->pointer = elfParseType(unit, attr->value); - break; - case DW_AT_byte_size: - t->size = attr->value; - break; - default: - fprintf(stderr, "Unknown pointer type attribute %02x\n", attr->name); - break; - } - } - if(abbrev->hasChildren) - fprintf(stderr, "Unexpected children for pointer type\n"); - *type = t; - return; - } - break; - case DW_TAG_reference_type: - { - Type *t = (Type *)calloc(sizeof(Type), 1); - - t->type = TYPE_reference; - - elfAddType(t, unit, offset); - - for(int i = 0; i < abbrev->numAttrs; i++) { - ELFAttr *attr = &abbrev->attrs[i]; - data =elfReadAttribute(data, attr); - switch(attr->name) { - case DW_AT_type: - t->pointer = elfParseType(unit, attr->value); - break; - case DW_AT_byte_size: - t->size = attr->value; - break; - default: - fprintf(stderr, "Unknown ref type attribute %02x\n", attr->name); - break; - } - } - if(abbrev->hasChildren) - fprintf(stderr, "Unexpected children for ref type\n"); - *type = t; - return; - } - break; - case DW_TAG_volatile_type: - { - u32 typeref = 0; - - for(int i = 0; i < abbrev->numAttrs; i++) { - ELFAttr *attr = &abbrev->attrs[i]; - data = elfReadAttribute(data, attr); - switch(attr->name) { - case DW_AT_type: - typeref = attr->value; - break; - default: - fprintf(stderr, "Unknown volatile attribute for type %02x\n", - attr->name); - break; - } - } - if(abbrev->hasChildren) - fprintf(stderr, "Unexpected children for volatile type\n"); - *type = elfParseType(unit, typeref); - return; - } - break; - case DW_TAG_const_type: - { - u32 typeref = 0; - - for(int i = 0; i < abbrev->numAttrs; i++) { - ELFAttr *attr = &abbrev->attrs[i]; - data = elfReadAttribute(data, attr); - switch(attr->name) { - case DW_AT_type: - typeref = attr->value; - break; - default: - fprintf(stderr, "Unknown const attribute for type %02x\n", - attr->name); - break; - } - } - if(abbrev->hasChildren) - fprintf(stderr, "Unexpected children for const type\n"); - *type = elfParseType(unit, typeref); - return; - } - break; - case DW_TAG_enumeration_type: - { - Type *t = (Type *)calloc(sizeof(Type), 1); - t->type = TYPE_enum; - Enum *e = (Enum *)calloc(sizeof(Enum), 1); - t->enumeration = e; - elfAddType(t, unit, offset); - int count = 0; - for(int i = 0; i < abbrev->numAttrs; i++) { - ELFAttr *attr = &abbrev->attrs[i]; - data = elfReadAttribute(data, attr); - switch(attr->name) { - case DW_AT_name: - t->name = attr->string; - break; - case DW_AT_byte_size: - t->size = attr->value; - break; - case DW_AT_sibling: - case DW_AT_decl_file: - case DW_AT_decl_line: - break; - default: - fprintf(stderr, "Unknown enum attribute %02x\n", attr->name); - } - } - if(abbrev->hasChildren) { - int bytes; - u32 num = elfReadLEB128(data, &bytes); - data += bytes; - while(num) { - ELFAbbrev *abbr = elfGetAbbrev(unit->abbrevs, num); - - switch(abbr->tag) { - case DW_TAG_enumerator: - { - count++; - e->members = (EnumMember *)realloc(e->members, - count*sizeof(EnumMember)); - EnumMember *m = &e->members[count-1]; - for(int i = 0; i < abbr->numAttrs; i++) { - ELFAttr *attr = &abbr->attrs[i]; - data = elfReadAttribute(data, attr); - switch(attr->name) { - case DW_AT_name: - m->name = attr->string; - break; - case DW_AT_const_value: - m->value = attr->value; - break; - default: - fprintf(stderr, "Unknown sub param attribute %02x\n", - attr->name); - } - } - } - break; - default: - fprintf(stderr, "Unknown enum tag %02x\n", abbr->tag); - data = elfSkipData(data, abbr, unit->abbrevs); - break; - } - num = elfReadLEB128(data, &bytes); - data += bytes; - } - } - e->count = count; - *type = t; - return; - } - break; - case DW_TAG_subroutine_type: - { - Type *t = (Type *)calloc(sizeof(Type), 1); - t->type = TYPE_function; - FunctionType *f = (FunctionType *)calloc(sizeof(FunctionType), 1); - t->function = f; - elfAddType(t, unit, offset); - for(int i = 0; i < abbrev->numAttrs; i++) { - ELFAttr *attr = &abbrev->attrs[i]; - data = elfReadAttribute(data, attr); - switch(attr->name) { - case DW_AT_prototyped: - case DW_AT_sibling: - break; - case DW_AT_type: - f->returnType = elfParseType(unit, attr->value); - break; - default: - fprintf(stderr, "Unknown subroutine attribute %02x\n", attr->name); - } - } - if(abbrev->hasChildren) { - int bytes; - u32 num = elfReadLEB128(data, &bytes); - data += bytes; - Object *lastVar = NULL; - while(num) { - ELFAbbrev *abbr = elfGetAbbrev(unit->abbrevs, num); - - switch(abbr->tag) { - case DW_TAG_formal_parameter: - { - Object *o; - data = elfParseObject(data, abbr, unit, &o); - if(f->args) - lastVar->next = o; - else - f->args = o; - lastVar = o; - } - break; - case DW_TAG_unspecified_parameters: - // no use in the debugger yet - data = elfSkipData(data, abbr, unit->abbrevs); - break; - CASE_TYPE_TAG: - // skip types... parsed only when used - data = elfSkipData(data, abbr, unit->abbrevs); - break; - default: - fprintf(stderr, "Unknown subroutine tag %02x\n", abbr->tag); - data = elfSkipData(data, abbr, unit->abbrevs); - break; - } - num = elfReadLEB128(data, &bytes); - data += bytes; - } - } - *type = t; - return; - } - break; - case DW_TAG_array_type: - { - u32 typeref = 0; - int i; - Array *array = (Array *)calloc(sizeof(Array), 1); - Type *t = (Type *)calloc(sizeof(Type), 1); - t->type = TYPE_array; - elfAddType(t, unit, offset); - - for(i = 0; i < abbrev->numAttrs; i++) { - ELFAttr *attr = &abbrev->attrs[i]; - data = elfReadAttribute(data, attr); - switch(attr->name) { - case DW_AT_sibling: - break; - case DW_AT_type: - typeref = attr->value; - array->type = elfParseType(unit, typeref); - break; - default: - fprintf(stderr, "Unknown array attribute %02x\n", attr->name); - } - } - if(abbrev->hasChildren) { - int bytes; - u32 num = elfReadLEB128(data, &bytes); - data += bytes; - int index = 0; - int maxBounds = 0; - while(num) { - ELFAbbrev *abbr = elfGetAbbrev(unit->abbrevs, num); - - switch(abbr->tag) { - case DW_TAG_subrange_type: - { - if(maxBounds == index) { - maxBounds += 4; - array->bounds = (int *)realloc(array->bounds, - sizeof(int)*maxBounds); - } - for(int i = 0; i < abbr->numAttrs; i++) { - ELFAttr *attr = &abbr->attrs[i]; - data = elfReadAttribute(data, attr); - switch(attr->name) { - case DW_AT_upper_bound: - array->bounds[index] = attr->value+1; - break; - case DW_AT_type: // ignore - break; - default: - fprintf(stderr, "Unknown subrange attribute %02x\n", - attr->name); - } - } - index++; - } - break; - default: - fprintf(stderr, "Unknown array tag %02x\n", abbr->tag); - data = elfSkipData(data, abbr, unit->abbrevs); - break; - } - num = elfReadLEB128(data, &bytes); - data += bytes; - } - array->maxBounds = index; - } - t->size = array->type->size; - for(i = 0; i < array->maxBounds; i++) - t->size *= array->bounds[i]; - t->array = array; - *type = t; - return; - } - break; - default: - fprintf(stderr, "Unknown type TAG %02x\n", abbrev->tag); - exit(-1); - } -} - -Type *elfParseType(CompileUnit *unit, u32 offset) -{ - Type *t = unit->types; - - while(t) { - if(t->offset == offset) - return t; - t = t->next; - } - if(offset == 0) { - Type *t = (Type *)calloc(sizeof(Type), 1); - t->type = TYPE_void; - t->offset = 0; - elfAddType(t, unit, 0); - return t; - } - u8 *data = unit->top + offset; - int bytes; - int abbrevNum = elfReadLEB128(data, &bytes); - data += bytes; - Type *type = NULL; - - ELFAbbrev *abbrev = elfGetAbbrev(unit->abbrevs, abbrevNum); - - elfParseType(data, offset, abbrev, unit, &type); - return type; -} - -void elfGetObjectAttributes(CompileUnit *unit, u32 offset, Object *o) -{ - u8 *data = unit->top + offset; - int bytes; - u32 abbrevNum = elfReadLEB128(data, &bytes); - data += bytes; - - if(!abbrevNum) { - return; - } - - ELFAbbrev *abbrev = elfGetAbbrev(unit->abbrevs, abbrevNum); - - for(int i = 0; i < abbrev->numAttrs; i++) { - ELFAttr *attr = &abbrev->attrs[i]; - data = elfReadAttribute(data, attr); - switch(attr->name) { - case DW_AT_location: - o->location = attr->block; - break; - case DW_AT_name: - if(o->name == NULL) - o->name = attr->string; - break; - case DW_AT_MIPS_linkage_name: - o->name = attr->string; - break; - case DW_AT_decl_file: - o->file = attr->value; - break; - case DW_AT_decl_line: - o->line = attr->value; - break; - case DW_AT_type: - o->type = elfParseType(unit, attr->value); - break; - case DW_AT_external: - o->external = attr->flag; - break; - case DW_AT_const_value: - case DW_AT_abstract_origin: - case DW_AT_declaration: - case DW_AT_artificial: - // todo - break; - case DW_AT_specification: - // TODO: - break; - default: - fprintf(stderr, "Unknown object attribute %02x\n", attr->name); - break; - } - } -} - -u8 *elfParseObject(u8 *data, ELFAbbrev *abbrev, CompileUnit *unit, - Object **object) -{ - Object *o = (Object *)calloc(sizeof(Object), 1); - - o->next = NULL; - - for(int i = 0; i < abbrev->numAttrs; i++) { - ELFAttr *attr = &abbrev->attrs[i]; - data = elfReadAttribute(data, attr); - switch(attr->name) { - case DW_AT_location: - o->location = attr->block; - break; - case DW_AT_name: - if(o->name == NULL) - o->name = attr->string; - break; - case DW_AT_MIPS_linkage_name: - o->name = attr->string; - break; - case DW_AT_decl_file: - o->file = attr->value; - break; - case DW_AT_decl_line: - o->line = attr->value; - break; - case DW_AT_type: - o->type = elfParseType(unit, attr->value); - break; - case DW_AT_external: - o->external = attr->flag; - break; - case DW_AT_abstract_origin: - elfGetObjectAttributes(unit, attr->value, o); - break; - case DW_AT_const_value: - case DW_AT_declaration: - case DW_AT_artificial: - break; - case DW_AT_specification: - // TODO: - break; - default: - fprintf(stderr, "Unknown object attribute %02x\n", attr->name); - break; - } - } - *object = o; - return data; -} - -u8 *elfParseBlock(u8 *data, ELFAbbrev *abbrev, CompileUnit *unit, - Function *func, Object **lastVar) -{ - int bytes; - u32 start = func->lowPC; - u32 end = func->highPC; - - for(int i = 0; i < abbrev->numAttrs; i++) { - ELFAttr *attr = &abbrev->attrs[i]; - data = elfReadAttribute(data, attr); - switch(attr->name) { - case DW_AT_sibling: - break; - case DW_AT_low_pc: - start = attr->value; - break; - case DW_AT_high_pc: - end = attr->value; - break; - case DW_AT_ranges: // ignore for now - break; - default: - fprintf(stderr, "Unknown block attribute %02x\n", attr->name); - break; - } - } - - if(abbrev->hasChildren) { - int nesting = 1; - - while(nesting) { - u32 abbrevNum = elfReadLEB128(data, &bytes); - data += bytes; - - if(!abbrevNum) { - nesting--; - continue; - } - - abbrev = elfGetAbbrev(unit->abbrevs, abbrevNum); - - switch(abbrev->tag) { - CASE_TYPE_TAG: // types only parsed when used - case DW_TAG_label: // not needed - data = elfSkipData(data, abbrev, unit->abbrevs); - break; - case DW_TAG_lexical_block: - data = elfParseBlock(data, abbrev, unit, func, lastVar); - break; - case DW_TAG_subprogram: - { - Function *f = NULL; - data = elfParseFunction(data, abbrev, unit, &f); - if(f != NULL) { - if(unit->lastFunction) - unit->lastFunction->next = f; - else - unit->functions = f; - unit->lastFunction = f; - } - } - break; - case DW_TAG_variable: - { - Object *o; - data = elfParseObject(data, abbrev, unit, &o); - if(o->startScope == 0) - o->startScope = start; - if(o->endScope == 0) - o->endScope = 0; - if(func->variables) - (*lastVar)->next = o; - else - func->variables = o; - *lastVar = o; - } - break; - case DW_TAG_inlined_subroutine: - // TODO: - data = elfSkipData(data, abbrev, unit->abbrevs); - break; - default: - { - fprintf(stderr, "Unknown block TAG %02x\n", abbrev->tag); - data = elfSkipData(data, abbrev, unit->abbrevs); - } - break; - } - } - } - return data; -} - -void elfGetFunctionAttributes(CompileUnit *unit, u32 offset, Function *func) -{ - u8 *data = unit->top + offset; - int bytes; - u32 abbrevNum = elfReadLEB128(data, &bytes); - data += bytes; - - if(!abbrevNum) { - return; - } - - ELFAbbrev *abbrev = elfGetAbbrev(unit->abbrevs, abbrevNum); - - for(int i = 0; i < abbrev->numAttrs; i++) { - ELFAttr *attr = &abbrev->attrs[i]; - data = elfReadAttribute(data, attr); - - switch(attr->name) { - case DW_AT_sibling: - break; - case DW_AT_name: - if(func->name == NULL) - func->name = attr->string; - break; - case DW_AT_MIPS_linkage_name: - func->name = attr->string; - break; - case DW_AT_low_pc: - func->lowPC = attr->value; - break; - case DW_AT_high_pc: - func->highPC = attr->value; - break; - case DW_AT_decl_file: - func->file = attr->value; - break; - case DW_AT_decl_line: - func->line = attr->value; - break; - case DW_AT_external: - func->external = attr->flag; - break; - case DW_AT_frame_base: - func->frameBase = attr->block; - break; - case DW_AT_type: - func->returnType = elfParseType(unit, attr->value); - break; - case DW_AT_inline: - case DW_AT_specification: - case DW_AT_declaration: - case DW_AT_artificial: - case DW_AT_prototyped: - case DW_AT_proc_body: - case DW_AT_save_offset: - case DW_AT_user_2002: - case DW_AT_virtuality: - case DW_AT_containing_type: - case DW_AT_accessibility: - // todo; - break; - case DW_AT_vtable_elem_location: - free(attr->block); - break; - default: - fprintf(stderr, "Unknown function attribute %02x\n", attr->name); - break; - } - } - - return; -} - -u8 *elfParseFunction(u8 *data, ELFAbbrev *abbrev, CompileUnit *unit, - Function **f) -{ - Function *func = (Function *)calloc(sizeof(Function), 1); - *f = func; - - int bytes; - bool mangled = false; - bool declaration = false; - for(int i = 0; i < abbrev->numAttrs; i++) { - ELFAttr *attr = &abbrev->attrs[i]; - data = elfReadAttribute(data, attr); - switch(attr->name) { - case DW_AT_sibling: - break; - case DW_AT_name: - if(func->name == NULL) - func->name = attr->string; - break; - case DW_AT_MIPS_linkage_name: - func->name = attr->string; - mangled = true; - break; - case DW_AT_low_pc: - func->lowPC = attr->value; - break; - case DW_AT_high_pc: - func->highPC = attr->value; - break; - case DW_AT_prototyped: - break; - case DW_AT_decl_file: - func->file = attr->value; - break; - case DW_AT_decl_line: - func->line = attr->value; - break; - case DW_AT_external: - func->external = attr->flag; - break; - case DW_AT_frame_base: - func->frameBase = attr->block; - break; - case DW_AT_type: - func->returnType = elfParseType(unit, attr->value); - break; - case DW_AT_abstract_origin: - elfGetFunctionAttributes(unit, attr->value, func); - break; - case DW_AT_declaration: - declaration = attr->flag; - break; - case DW_AT_inline: - case DW_AT_specification: - case DW_AT_artificial: - case DW_AT_proc_body: - case DW_AT_save_offset: - case DW_AT_user_2002: - case DW_AT_virtuality: - case DW_AT_containing_type: - case DW_AT_accessibility: - // todo; - break; - case DW_AT_vtable_elem_location: - free(attr->block); - break; - default: - fprintf(stderr, "Unknown function attribute %02x\n", attr->name); - break; - } - } - - if(declaration) { - elfCleanUp(func); - free(func); - *f = NULL; - - while(1) { - u32 abbrevNum = elfReadLEB128(data, &bytes); - data += bytes; - - if(!abbrevNum) { - return data; - } - - abbrev = elfGetAbbrev(unit->abbrevs, abbrevNum); - - data = elfSkipData(data, abbrev, unit->abbrevs); - } - } - - if(abbrev->hasChildren) { - int nesting = 1; - Object *lastParam = NULL; - Object *lastVar = NULL; - - while(nesting) { - u32 abbrevNum = elfReadLEB128(data, &bytes); - data += bytes; - - if(!abbrevNum) { - nesting--; - continue; - } - - abbrev = elfGetAbbrev(unit->abbrevs, abbrevNum); - - switch(abbrev->tag) { - CASE_TYPE_TAG: // no need to parse types. only parsed when used - case DW_TAG_label: // not needed - data = elfSkipData(data, abbrev, unit->abbrevs); - break; - case DW_TAG_subprogram: - { - Function *fnc=NULL; - data = elfParseFunction(data, abbrev, unit, &fnc); - if(fnc != NULL) { - if(unit->lastFunction == NULL) - unit->functions = fnc; - else - unit->lastFunction->next = fnc; - unit->lastFunction = fnc; - } - } - break; - case DW_TAG_lexical_block: - { - data = elfParseBlock(data, abbrev, unit, func, &lastVar); - } - break; - case DW_TAG_formal_parameter: - { - Object *o; - data = elfParseObject(data, abbrev, unit, &o); - if(func->parameters) - lastParam->next = o; - else - func->parameters = o; - lastParam = o; - } - break; - case DW_TAG_variable: - { - Object *o; - data = elfParseObject(data, abbrev, unit, &o); - if(func->variables) - lastVar->next = o; - else - func->variables = o; - lastVar = o; - } - break; - case DW_TAG_unspecified_parameters: - case DW_TAG_inlined_subroutine: - { - // todo - for(int i = 0; i < abbrev->numAttrs; i++) { - data = elfReadAttribute(data, &abbrev->attrs[i]); - if(abbrev->attrs[i].form == DW_FORM_block1) - free(abbrev->attrs[i].block); - } - - if(abbrev->hasChildren) - nesting++; - } - break; - default: - { - fprintf(stderr, "Unknown function TAG %02x\n", abbrev->tag); - data = elfSkipData(data, abbrev, unit->abbrevs); - } - break; - } - } - } - return data; -} - -u8 *elfParseUnknownData(u8 *data, ELFAbbrev *abbrev, ELFAbbrev **abbrevs) -{ - int i; - int bytes; - // switch(abbrev->tag) { - // default: - fprintf(stderr, "Unknown TAG %02x\n", abbrev->tag); - - for(i = 0; i < abbrev->numAttrs; i++) { - data = elfReadAttribute(data, &abbrev->attrs[i]); - if(abbrev->attrs[i].form == DW_FORM_block1) - free(abbrev->attrs[i].block); - } - - if(abbrev->hasChildren) { - int nesting = 1; - while(nesting) { - u32 abbrevNum = elfReadLEB128(data, &bytes); - data += bytes; - - if(!abbrevNum) { - nesting--; - continue; - } - - abbrev = elfGetAbbrev(abbrevs, abbrevNum); - - fprintf(stderr, "Unknown TAG %02x\n", abbrev->tag); - - for(i = 0; i < abbrev->numAttrs; i++) { - data = elfReadAttribute(data, &abbrev->attrs[i]); - if(abbrev->attrs[i].form == DW_FORM_block1) - free(abbrev->attrs[i].block); - } - - if(abbrev->hasChildren) { - nesting++; - } - } - } - // } - return data; -} - -u8 *elfParseCompileUnitChildren(u8 *data, CompileUnit *unit) -{ - int bytes; - u32 abbrevNum = elfReadLEB128(data, &bytes); - data += bytes; - Object *lastObj = NULL; - while(abbrevNum) { - ELFAbbrev *abbrev = elfGetAbbrev(unit->abbrevs, abbrevNum); - switch(abbrev->tag) { - case DW_TAG_subprogram: - { - Function *func = NULL; - data = elfParseFunction(data, abbrev, unit, &func); - if(func != NULL) { - if(unit->lastFunction) - unit->lastFunction->next = func; - else - unit->functions = func; - unit->lastFunction = func; - } - } - break; - CASE_TYPE_TAG: - data = elfSkipData(data, abbrev, unit->abbrevs); - break; - case DW_TAG_variable: - { - Object *var = NULL; - data = elfParseObject(data, abbrev, unit, &var); - if(lastObj) - lastObj->next = var; - else - unit->variables = var; - lastObj = var; - } - break; - default: - data = elfParseUnknownData(data, abbrev, unit->abbrevs); - break; - } - - abbrevNum = elfReadLEB128(data, &bytes); - data += bytes; - } - return data; -} - - -CompileUnit *elfParseCompUnit(u8 *data, u8 *abbrevData) -{ - int bytes; - u8 *top = data; - - u32 length = elfRead4Bytes(data); - data += 4; - - u16 version = elfRead2Bytes(data); - data += 2; - - u32 offset = elfRead4Bytes(data); - data += 4; - - u8 addrSize = *data++; - - if(version != 2) { - fprintf(stderr, "Unsupported debugging information version %d\n", version); - return NULL; - } - - if(addrSize != 4) { - fprintf(stderr, "Unsupported address size %d\n", addrSize); - return NULL; - } - - ELFAbbrev **abbrevs = elfReadAbbrevs(abbrevData, offset); - - u32 abbrevNum = elfReadLEB128(data, &bytes); - data += bytes; - - ELFAbbrev *abbrev = elfGetAbbrev(abbrevs, abbrevNum); - - CompileUnit *unit = (CompileUnit *)calloc(sizeof(CompileUnit), 1); - unit->top = top; - unit->length = length; - unit->abbrevs = abbrevs; - unit->next = NULL; - - elfCurrentUnit = unit; - - int i; - - for(i = 0; i < abbrev->numAttrs; i++) { - ELFAttr *attr = &abbrev->attrs[i]; - data = elfReadAttribute(data, attr); - - switch(attr->name) { - case DW_AT_name: - unit->name = attr->string; - break; - case DW_AT_stmt_list: - unit->hasLineInfo = true; - unit->lineInfo = attr->value; - break; - case DW_AT_low_pc: - unit->lowPC = attr->value; - break; - case DW_AT_high_pc: - unit->highPC = attr->value; - break; - case DW_AT_compdir: - unit->compdir = attr->string; - break; - // ignore - case DW_AT_language: - case DW_AT_producer: - case DW_AT_macro_info: - case DW_AT_entry_pc: - break; - default: - fprintf(stderr, "Unknown attribute %02x\n", attr->name); - break; - } - } - - if(abbrev->hasChildren) - elfParseCompileUnitChildren(data, unit); - - return unit; -} - -void elfParseAranges(u8 *data) -{ - ELFSectionHeader *sh = elfGetSectionByName(".debug_aranges"); - if(sh == NULL) { - fprintf(stderr, "No aranges found\n"); - return; - } - - data = elfReadSection(data, sh); - u8 *end = data + READ32LE(&sh->size); - - int max = 4; - ARanges *ranges = (ARanges *)calloc(sizeof(ARanges), 4); - - int index = 0; - - while(data < end) { - u32 len = elfRead4Bytes(data); - data += 4; - // u16 version = elfRead2Bytes(data); - data += 2; - u32 offset = elfRead4Bytes(data); - data += 4; - // u8 addrSize = *data++; - // u8 segSize = *data++; - data += 2; // remove if uncommenting above - data += 4; - ranges[index].count = (len-20)/8; - ranges[index].offset = offset; - ranges[index].ranges = (ARange *)calloc(sizeof(ARange), (len-20)/8); - int i = 0; - while(true) { - u32 addr = elfRead4Bytes(data); - data += 4; - u32 len = elfRead4Bytes(data); - data += 4; - if(addr == 0 && len == 0) - break; - ranges[index].ranges[i].lowPC = addr; - ranges[index].ranges[i].highPC = addr+len; - i++; - } - index++; - if(index == max) { - max += 4; - ranges = (ARanges *)realloc(ranges, max*sizeof(ARanges)); - } - } - elfDebugInfo->numRanges = index; - elfDebugInfo->ranges = ranges; -} - -void elfReadSymtab(u8 *data) -{ - ELFSectionHeader *sh = elfGetSectionByName(".symtab"); - int table = READ32LE(&sh->link); - - char *strtable = (char *)elfReadSection(data, elfGetSectionByNumber(table)); - - ELFSymbol *symtab = (ELFSymbol *)elfReadSection(data, sh); - - int count = READ32LE(&sh->size) / sizeof(ELFSymbol); - elfSymbolsCount = 0; - - elfSymbols = (Symbol *)malloc(sizeof(Symbol)*count); - - int i; - - for(i = 0; i < count; i++) { - ELFSymbol *s = &symtab[i]; - int type = s->info & 15; - int binding = s->info >> 4; - - if(binding) { - Symbol *sym = &elfSymbols[elfSymbolsCount]; - sym->name = &strtable[READ32LE(&s->name)]; - sym->binding = binding; - sym->type = type; - sym->value = READ32LE(&s->value); - sym->size = READ32LE(&s->size); - elfSymbolsCount++; - } - } - for(i = 0; i < count; i++) { - ELFSymbol *s = &symtab[i]; - int bind = s->info>>4; - int type = s->info & 15; - - if(!bind) { - Symbol *sym = &elfSymbols[elfSymbolsCount]; - sym->name = &strtable[READ32LE(&s->name)]; - sym->binding = (s->info >> 4); - sym->type = type; - sym->value = READ32LE(&s->value); - sym->size = READ32LE(&s->size); - elfSymbolsCount++; - } - } - elfSymbolsStrTab = strtable; - // free(symtab); -} - -bool elfReadProgram(ELFHeader *eh, u8 *data, int& size, bool parseDebug) -{ - int count = READ16LE(&eh->e_phnum); - int i; - - if(READ32LE(&eh->e_entry) == 0x2000000) - cpuIsMultiBoot = true; - - // read program headers... should probably move this code down - u8 *p = data + READ32LE(&eh->e_phoff); - size = 0; - for(i = 0; i < count; i++) { - ELFProgramHeader *ph = (ELFProgramHeader *)p; - p += sizeof(ELFProgramHeader); - if(READ16LE(&eh->e_phentsize) != sizeof(ELFProgramHeader)) { - p += READ16LE(&eh->e_phentsize) - sizeof(ELFProgramHeader); - } - - // printf("PH %d %08x %08x %08x %08x %08x %08x %08x %08x\n", - // i, ph->type, ph->offset, ph->vaddr, ph->paddr, - // ph->filesz, ph->memsz, ph->flags, ph->align); - if(cpuIsMultiBoot) { - if(READ32LE(&ph->paddr) >= 0x2000000 && - READ32LE(&ph->paddr) <= 0x203ffff) { - memcpy(&workRAM[READ32LE(&ph->paddr) & 0x3ffff], - data + READ32LE(&ph->offset), - READ32LE(&ph->filesz)); - size += READ32LE(&ph->filesz); - } - } else { - if(READ32LE(&ph->paddr) >= 0x8000000 && - READ32LE(&ph->paddr) <= 0x9ffffff) { - memcpy(&rom[READ32LE(&ph->paddr) & 0x1ffffff], - data + READ32LE(&ph->offset), - READ32LE(&ph->filesz)); - size += READ32LE(&ph->filesz); - } - } - } - - char *stringTable = NULL; - - // read section headers - p = data + READ32LE(&eh->e_shoff); - count = READ16LE(&eh->e_shnum); - - ELFSectionHeader **sh = (ELFSectionHeader **) - malloc(sizeof(ELFSectionHeader *) * count); - - for(i = 0; i < count; i++) { - sh[i] = (ELFSectionHeader *)p; - p += sizeof(ELFSectionHeader); - if(READ16LE(&eh->e_shentsize) != sizeof(ELFSectionHeader)) - p += READ16LE(&eh->e_shentsize) - sizeof(ELFSectionHeader); - } - - if(READ16LE(&eh->e_shstrndx) != 0) { - stringTable = (char *)elfReadSection(data, - sh[READ16LE(&eh->e_shstrndx)]); - } - - elfSectionHeaders = sh; - elfSectionHeadersStringTable = stringTable; - elfSectionHeadersCount = count; - - for(i = 0; i < count; i++) { - // printf("SH %d %-20s %08x %08x %08x %08x %08x %08x %08x %08x\n", - // i, &stringTable[sh[i]->name], sh[i]->name, sh[i]->type, - // sh[i]->flags, sh[i]->addr, sh[i]->offset, sh[i]->size, - // sh[i]->link, sh[i]->info); - if(READ32LE(&sh[i]->flags) & 2) { // load section - if(cpuIsMultiBoot) { - if(READ32LE(&sh[i]->addr) >= 0x2000000 && - READ32LE(&sh[i]->addr) <= 0x203ffff) { - memcpy(&workRAM[READ32LE(&sh[i]->addr) & 0x3ffff], data + - READ32LE(&sh[i]->offset), - READ32LE(&sh[i]->size)); - size += READ32LE(&sh[i]->size); - } - } else { - if(READ32LE(&sh[i]->addr) >= 0x8000000 && - READ32LE(&sh[i]->addr) <= 0x9ffffff) { - memcpy(&rom[READ32LE(&sh[i]->addr) & 0x1ffffff], - data + READ32LE(&sh[i]->offset), - READ32LE(&sh[i]->size)); - size += READ32LE(&sh[i]->size); - } - } - } - } - - if(parseDebug) { - fprintf(stderr, "Parsing debug info\n"); - - ELFSectionHeader *dbgHeader = elfGetSectionByName(".debug_info"); - if(dbgHeader == NULL) { - fprintf(stderr, "Cannot find debug information\n"); - goto end; - } - - ELFSectionHeader *h = elfGetSectionByName(".debug_abbrev"); - if(h == NULL) { - fprintf(stderr, "Cannot find abbreviation table\n"); - goto end; - } - - elfDebugInfo = (DebugInfo *)calloc(sizeof(DebugInfo), 1); - u8 *abbrevdata = elfReadSection(data, h); - - h = elfGetSectionByName(".debug_str"); - - if(h == NULL) - elfDebugStrings = NULL; - else - elfDebugStrings = (char *)elfReadSection(data, h); - - u8 *debugdata = elfReadSection(data, dbgHeader); - - elfDebugInfo->debugdata = data; - elfDebugInfo->infodata = debugdata; - - u32 total = READ32LE(&dbgHeader->size); - u8 *end = debugdata + total; - u8 *ddata = debugdata; - - CompileUnit *last = NULL; - CompileUnit *unit = NULL; - - while(ddata < end) { - unit = elfParseCompUnit(ddata, abbrevdata); - unit->offset = ddata-debugdata; - elfParseLineInfo(unit, data); - if(last == NULL) - elfCompileUnits = unit; - else - last->next = unit; - last = unit; - ddata += 4 + unit->length; - } - elfParseAranges(data); - CompileUnit *comp = elfCompileUnits; - while(comp) { - ARanges *r = elfDebugInfo->ranges; - for(int i = 0; i < elfDebugInfo->numRanges; i++) - if(r[i].offset == comp->offset) { - comp->ranges = &r[i]; - break; - } - comp = comp->next; - } - elfParseCFA(data); - elfReadSymtab(data); - } - end: - if(sh) { - free(sh); - } - - elfSectionHeaders = NULL; - elfSectionHeadersStringTable = NULL; - elfSectionHeadersCount = 0; - - return true; -} - -extern bool parseDebug; - -bool elfRead(const char *name, int& siz, FILE *f) -{ - fseek(f, 0, SEEK_END); - long size = ftell(f); - elfFileData = (u8 *)malloc(size); - fseek(f, 0, SEEK_SET); - fread(elfFileData, 1, size, f); - fclose(f); - - ELFHeader *header = (ELFHeader *)elfFileData; - - if(READ32LE(&header->magic) != 0x464C457F || - READ16LE(&header->e_machine) != 40 || - header->clazz != 1) { - systemMessage(0, N_("Not a valid ELF file %s"), name); - free(elfFileData); - elfFileData = NULL; - return false; - } - - if(!elfReadProgram(header, elfFileData, siz, parseDebug)) { - free(elfFileData); - elfFileData = NULL; - return false; - } - - return true; -} - -void elfCleanUp(Object *o) -{ - free(o->location); -} - -void elfCleanUp(Function *func) -{ - Object *o = func->parameters; - while(o) { - elfCleanUp(o); - Object *next = o->next; - free(o); - o = next; - } - - o = func->variables; - while(o) { - elfCleanUp(o); - Object *next = o->next; - free(o); - o = next; - } - free(func->frameBase); -} - -void elfCleanUp(ELFAbbrev **abbrevs) -{ - for(int i = 0; i < 121; i++) { - ELFAbbrev *abbrev = abbrevs[i]; - - while(abbrev) { - free(abbrev->attrs); - ELFAbbrev *next = abbrev->next; - free(abbrev); - - abbrev = next; - } - } -} - -void elfCleanUp(Type *t) -{ - switch(t->type) { - case TYPE_function: - if(t->function) { - Object *o = t->function->args; - while(o) { - elfCleanUp(o); - Object *next = o->next; - free(o); - o = next; - } - free(t->function); - } - break; - case TYPE_array: - if(t->array) { - free(t->array->bounds); - free(t->array); - } - break; - case TYPE_struct: - case TYPE_union: - if(t->structure) { - for(int i = 0; i < t->structure->memberCount; i++) { - free(t->structure->members[i].location); - } - free(t->structure->members); - free(t->structure); - } - break; - case TYPE_enum: - if(t->enumeration) { - free(t->enumeration->members); - free(t->enumeration); - } - break; - case TYPE_base: - case TYPE_pointer: - case TYPE_void: - case TYPE_reference: - break; // nothing to do - } -} - -void elfCleanUp(CompileUnit *comp) -{ - elfCleanUp(comp->abbrevs); - free(comp->abbrevs); - Function *func = comp->functions; - while(func) { - elfCleanUp(func); - Function *next = func->next; - free(func); - func = next; - } - Type *t = comp->types; - while(t) { - elfCleanUp(t); - Type *next = t->next; - free(t); - t = next; - } - Object *o = comp->variables; - while(o) { - elfCleanUp(o); - Object *next = o->next; - free(o); - o = next; - } - if(comp->lineInfoTable) { - free(comp->lineInfoTable->lines); - free(comp->lineInfoTable->files); - free(comp->lineInfoTable); - } -} - -void elfCleanUp() -{ - CompileUnit *comp = elfCompileUnits; - - while(comp) { - elfCleanUp(comp); - CompileUnit *next = comp->next; - free(comp); - comp = next; - } - elfCompileUnits = NULL; - free(elfSymbols); - elfSymbols = NULL; - // free(elfSymbolsStrTab); - elfSymbolsStrTab = NULL; - - elfDebugStrings = NULL; - if(elfDebugInfo) { - int num = elfDebugInfo->numRanges; - int i; - for(i = 0; i < num; i++) { - free(elfDebugInfo->ranges[i].ranges); - } - free(elfDebugInfo->ranges); - free(elfDebugInfo); - elfDebugInfo = NULL; - } - - if(elfFdes) { - if(elfFdeCount) { - for(int i = 0; i < elfFdeCount; i++) - free(elfFdes[i]); - } - free(elfFdes); - - elfFdes = NULL; - elfFdeCount = 0; - } - - ELFcie *cie = elfCies; - while(cie) { - ELFcie *next = cie->next; - free(cie); - cie = next; - } - elfCies = NULL; - - if(elfFileData) { - free(elfFileData); - elfFileData = NULL; - } -} \ No newline at end of file +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#include +#include +#include + +#include "GBA.h" +#include "Port.h" +#include "elf.h" +#include "NLS.h" + +#define elfReadMemory(addr) \ + READ32LE((&map[(addr)>>24].address[(addr) & map[(addr)>>24].mask])) + +#define DW_TAG_array_type 0x01 +#define DW_TAG_enumeration_type 0x04 +#define DW_TAG_formal_parameter 0x05 +#define DW_TAG_label 0x0a +#define DW_TAG_lexical_block 0x0b +#define DW_TAG_member 0x0d +#define DW_TAG_pointer_type 0x0f +#define DW_TAG_reference_type 0x10 +#define DW_TAG_compile_unit 0x11 +#define DW_TAG_structure_type 0x13 +#define DW_TAG_subroutine_type 0x15 +#define DW_TAG_typedef 0x16 +#define DW_TAG_union_type 0x17 +#define DW_TAG_unspecified_parameters 0x18 +#define DW_TAG_inheritance 0x1c +#define DW_TAG_inlined_subroutine 0x1d +#define DW_TAG_subrange_type 0x21 +#define DW_TAG_base_type 0x24 +#define DW_TAG_const_type 0x26 +#define DW_TAG_enumerator 0x28 +#define DW_TAG_subprogram 0x2e +#define DW_TAG_variable 0x34 +#define DW_TAG_volatile_type 0x35 + +#define DW_AT_sibling 0x01 +#define DW_AT_location 0x02 +#define DW_AT_name 0x03 +#define DW_AT_byte_size 0x0b +#define DW_AT_bit_offset 0x0c +#define DW_AT_bit_size 0x0d +#define DW_AT_stmt_list 0x10 +#define DW_AT_low_pc 0x11 +#define DW_AT_high_pc 0x12 +#define DW_AT_language 0x13 +#define DW_AT_compdir 0x1b +#define DW_AT_const_value 0x1c +#define DW_AT_containing_type 0x1d +#define DW_AT_inline 0x20 +#define DW_AT_producer 0x25 +#define DW_AT_prototyped 0x27 +#define DW_AT_upper_bound 0x2f +#define DW_AT_abstract_origin 0x31 +#define DW_AT_accessibility 0x32 +#define DW_AT_artificial 0x34 +#define DW_AT_data_member_location 0x38 +#define DW_AT_decl_file 0x3a +#define DW_AT_decl_line 0x3b +#define DW_AT_declaration 0x3c +#define DW_AT_encoding 0x3e +#define DW_AT_external 0x3f +#define DW_AT_frame_base 0x40 +#define DW_AT_macro_info 0x43 +#define DW_AT_specification 0x47 +#define DW_AT_type 0x49 +#define DW_AT_virtuality 0x4c +#define DW_AT_vtable_elem_location 0x4d +// DWARF 2.1/3.0 extensions +#define DW_AT_entry_pc 0x52 +#define DW_AT_ranges 0x55 +// ARM Compiler extensions +#define DW_AT_proc_body 0x2000 +#define DW_AT_save_offset 0x2001 +#define DW_AT_user_2002 0x2002 +// MIPS extensions +#define DW_AT_MIPS_linkage_name 0x2007 + +#define DW_FORM_addr 0x01 +#define DW_FORM_data2 0x05 +#define DW_FORM_data4 0x06 +#define DW_FORM_string 0x08 +#define DW_FORM_block 0x09 +#define DW_FORM_block1 0x0a +#define DW_FORM_data1 0x0b +#define DW_FORM_flag 0x0c +#define DW_FORM_sdata 0x0d +#define DW_FORM_strp 0x0e +#define DW_FORM_udata 0x0f +#define DW_FORM_ref_addr 0x10 +#define DW_FORM_ref4 0x13 +#define DW_FORM_ref_udata 0x15 +#define DW_FORM_indirect 0x16 + +#define DW_OP_addr 0x03 +#define DW_OP_plus_uconst 0x23 +#define DW_OP_reg0 0x50 +#define DW_OP_reg1 0x51 +#define DW_OP_reg2 0x52 +#define DW_OP_reg3 0x53 +#define DW_OP_reg4 0x54 +#define DW_OP_reg5 0x55 +#define DW_OP_reg6 0x56 +#define DW_OP_reg7 0x57 +#define DW_OP_reg8 0x58 +#define DW_OP_reg9 0x59 +#define DW_OP_reg10 0x5a +#define DW_OP_reg11 0x5b +#define DW_OP_reg12 0x5c +#define DW_OP_reg13 0x5d +#define DW_OP_reg14 0x5e +#define DW_OP_reg15 0x5f +#define DW_OP_fbreg 0x91 + +#define DW_LNS_extended_op 0x00 +#define DW_LNS_copy 0x01 +#define DW_LNS_advance_pc 0x02 +#define DW_LNS_advance_line 0x03 +#define DW_LNS_set_file 0x04 +#define DW_LNS_set_column 0x05 +#define DW_LNS_negate_stmt 0x06 +#define DW_LNS_set_basic_block 0x07 +#define DW_LNS_const_add_pc 0x08 +#define DW_LNS_fixed_advance_pc 0x09 + +#define DW_LNE_end_sequence 0x01 +#define DW_LNE_set_address 0x02 +#define DW_LNE_define_file 0x03 + +#define DW_CFA_advance_loc 0x01 +#define DW_CFA_offset 0x02 +#define DW_CFA_restore 0x03 +#define DW_CFA_set_loc 0x01 +#define DW_CFA_advance_loc1 0x02 +#define DW_CFA_advance_loc2 0x03 +#define DW_CFA_advance_loc4 0x04 +#define DW_CFA_offset_extended 0x05 +#define DW_CFA_restore_extended 0x06 +#define DW_CFA_undefined 0x07 +#define DW_CFA_same_value 0x08 +#define DW_CFA_register 0x09 +#define DW_CFA_remember_state 0x0a +#define DW_CFA_restore_state 0x0b +#define DW_CFA_def_cfa 0x0c +#define DW_CFA_def_cfa_register 0x0d +#define DW_CFA_def_cfa_offset 0x0e +#define DW_CFA_nop 0x00 + +#define CASE_TYPE_TAG \ + case DW_TAG_const_type:\ + case DW_TAG_volatile_type:\ + case DW_TAG_pointer_type:\ + case DW_TAG_base_type:\ + case DW_TAG_array_type:\ + case DW_TAG_structure_type:\ + case DW_TAG_union_type:\ + case DW_TAG_typedef:\ + case DW_TAG_subroutine_type:\ + case DW_TAG_enumeration_type:\ + case DW_TAG_enumerator:\ + case DW_TAG_reference_type + +struct ELFcie { + ELFcie *next; + u32 offset; + u8 *augmentation; + u32 codeAlign; + s32 dataAlign; + int returnAddress; + u8 *data; + u32 dataLen; +}; + +struct ELFfde { + ELFcie *cie; + u32 address; + u32 end; + u8 *data; + u32 dataLen; +}; + +enum ELFRegMode { + REG_NOT_SET, + REG_OFFSET, + REG_REGISTER +}; + + +struct ELFFrameStateRegister { + ELFRegMode mode; + int reg; + s32 offset; +}; + +struct ELFFrameStateRegisters { + ELFFrameStateRegister regs[16]; + ELFFrameStateRegisters *previous; +}; + +enum ELFCfaMode { + CFA_NOT_SET, + CFA_REG_OFFSET +}; + +struct ELFFrameState { + ELFFrameStateRegisters registers; + + ELFCfaMode cfaMode; + int cfaRegister; + s32 cfaOffset; + + u32 pc; + + int dataAlign; + int codeAlign; + int returnAddress; +}; + +extern bool cpuIsMultiBoot; + +Symbol *elfSymbols = NULL; +char *elfSymbolsStrTab = NULL; +int elfSymbolsCount = 0; + +ELFSectionHeader **elfSectionHeaders = NULL; +char *elfSectionHeadersStringTable = NULL; +int elfSectionHeadersCount = 0; +u8 *elfFileData = NULL; + +CompileUnit *elfCompileUnits = NULL; +DebugInfo *elfDebugInfo = NULL; +char *elfDebugStrings = NULL; + +ELFcie *elfCies = NULL; +ELFfde **elfFdes = NULL; +int elfFdeCount = 0; + +CompileUnit *elfCurrentUnit = NULL; + +u32 elfRead4Bytes(u8 *); +u16 elfRead2Bytes(u8 *); + +CompileUnit *elfGetCompileUnit(u32 addr) +{ + if(elfCompileUnits) { + CompileUnit *unit = elfCompileUnits; + while(unit) { + if(unit->lowPC) { + if(addr >= unit->lowPC && addr < unit->highPC) + return unit; + } else { + ARanges *r = unit->ranges; + if(r) { + int count = r->count; + for(int j = 0; j < count; j++) { + if(addr >= r->ranges[j].lowPC && addr < r->ranges[j].highPC) + return unit; + } + } + } + unit = unit->next; + } + } + return NULL; +} + +char *elfGetAddressSymbol(u32 addr) +{ + static char buffer[256]; + + CompileUnit *unit = elfGetCompileUnit(addr); + // found unit, need to find function + if(unit) { + Function *func = unit->functions; + while(func) { + if(addr >= func->lowPC && addr < func->highPC) { + int offset = addr - func->lowPC; + char *name = func->name; + if(!name) + name = ""; + if(offset) + sprintf(buffer, "%s+%d", name, offset); + else + strcpy(buffer, name); + return buffer; + } + func = func->next; + } + } + + if(elfSymbolsCount) { + for(int i = 0; i < elfSymbolsCount; i++) { + Symbol *s = &elfSymbols[i]; + if((addr >= s->value) && addr < (s->value+s->size)) { + int offset = addr-s->value; + char *name = s->name; + if(name == NULL) + name = ""; + if(offset) + sprintf(buffer, "%s+%d", name, addr-s->value); + else + strcpy(buffer, name); + return buffer; + } else if(addr == s->value) { + if(s->name) + strcpy(buffer, s->name); + else + strcpy(buffer, ""); + return buffer; + } + } + } + + return ""; +} + +bool elfFindLineInModule(u32 *addr, char *name, int line) +{ + CompileUnit *unit = elfCompileUnits; + + while(unit) { + if(unit->lineInfoTable) { + int i; + int count = unit->lineInfoTable->fileCount; + char *found = NULL; + for(i = 0; i < count; i++) { + if(strcmp(name, unit->lineInfoTable->files[i]) == 0) { + found = unit->lineInfoTable->files[i]; + break; + } + } + // found a matching filename... try to find line now + if(found) { + LineInfoItem *table = unit->lineInfoTable->lines; + count = unit->lineInfoTable->number; + for(i = 0; i < count; i++) { + if(table[i].file == found && table[i].line == line) { + *addr = table[i].address; + return true; + } + } + // we can only find a single match + return false; + } + } + unit = unit->next; + } + return false; +} + +int elfFindLine(CompileUnit *unit, Function * /* func */, u32 addr, char **f) +{ + int currentLine = -1; + if(unit->hasLineInfo) { + int count = unit->lineInfoTable->number; + LineInfoItem *table = unit->lineInfoTable->lines; + int i; + for(i = 0; i < count; i++) { + if(addr <= table[i].address) + break; + } + if(i == count) + i--; + *f = table[i].file; + currentLine = table[i].line; + } + return currentLine; +} + +bool elfFindLineInUnit(u32 *addr, CompileUnit *unit, int line) +{ + if(unit->hasLineInfo) { + int count = unit->lineInfoTable->number; + LineInfoItem *table = unit->lineInfoTable->lines; + int i; + for(i = 0; i < count; i++) { + if(line == table[i].line) { + *addr = table[i].address; + return true; + } + } + } + return false; +} + +bool elfGetCurrentFunction(u32 addr, Function **f, CompileUnit **u) +{ + CompileUnit *unit = elfGetCompileUnit(addr); + // found unit, need to find function + if(unit) { + Function *func = unit->functions; + while(func) { + if(addr >= func->lowPC && addr < func->highPC) { + *f = func; + *u = unit; + return true; + } + func = func->next; + } + } + return false; +} + +bool elfGetObject(char *name, Function *f, CompileUnit *u, Object **o) +{ + if(f && u) { + Object *v = f->variables; + + while(v) { + if(strcmp(name, v->name) == 0) { + *o = v; + return true; + } + v = v->next; + } + v = f->parameters; + while(v) { + if(strcmp(name, v->name) == 0) { + *o = v; + return true; + } + v = v->next; + } + v = u->variables; + while(v) { + if(strcmp(name, v->name) == 0) { + *o = v; + return true; + } + v = v->next; + } + } + + CompileUnit *c = elfCompileUnits; + + while(c) { + if(c != u) { + Object *v = c->variables; + while(v) { + if(strcmp(name, v->name) == 0) { + *o = v; + return true; + } + v = v->next; + } + } + c = c->next; + } + + return false; +} + +char *elfGetSymbol(int i, u32 *value, u32 *size, int *type) +{ + if(i < elfSymbolsCount) { + Symbol *s = &elfSymbols[i]; + *value = s->value; + *size = s->size; + *type = s->type; + return s->name; + } + return NULL; +} + +bool elfGetSymbolAddress(char *sym, u32 *addr, u32 *size, int *type) +{ + if(elfSymbolsCount) { + for(int i = 0; i < elfSymbolsCount; i++) { + Symbol *s = &elfSymbols[i]; + if(strcmp(sym, s->name) == 0) { + *addr = s->value; + *size = s->size; + *type = s->type; + return true; + } + } + } + return false; +} + +ELFfde *elfGetFde(u32 address) +{ + if(elfFdes) { + int i; + for(i = 0; i < elfFdeCount; i++) { + if(address >= elfFdes[i]->address && + address < elfFdes[i]->end) { + return elfFdes[i]; + } + } + } + + return NULL; +} + +void elfExecuteCFAInstructions(ELFFrameState *state, u8 *data, u32 len, + u32 pc) +{ + u8 *end = data + len; + int bytes; + int reg; + ELFFrameStateRegisters *fs; + + while(data < end && state->pc < pc) { + u8 op = *data++; + + switch(op >> 6) { + case DW_CFA_advance_loc: + state->pc += (op & 0x3f) * state->codeAlign; + break; + case DW_CFA_offset: + reg = op & 0x3f; + state->registers.regs[reg].mode = REG_OFFSET; + state->registers.regs[reg].offset = state->dataAlign * + (s32)elfReadLEB128(data, &bytes); + data += bytes; + break; + case DW_CFA_restore: + // we don't care much about the other possible settings, + // so just setting to unset is enough for now + state->registers.regs[op & 0x3f].mode = REG_NOT_SET; + break; + case 0: + switch(op & 0x3f) { + case DW_CFA_nop: + break; + case DW_CFA_advance_loc1: + state->pc += state->codeAlign * (*data++); + break; + case DW_CFA_advance_loc2: + state->pc += state->codeAlign * elfRead2Bytes(data); + data += 2; + break; + case DW_CFA_advance_loc4: + state->pc += state->codeAlign * elfRead4Bytes(data); + data += 4; + break; + case DW_CFA_offset_extended: + reg = elfReadLEB128(data, &bytes); + data += bytes; + state->registers.regs[reg].mode = REG_OFFSET; + state->registers.regs[reg].offset = state->dataAlign * + (s32)elfReadLEB128(data, &bytes); + data += bytes; + break; + case DW_CFA_restore_extended: + case DW_CFA_undefined: + case DW_CFA_same_value: + reg = elfReadLEB128(data, &bytes); + data += bytes; + state->registers.regs[reg].mode = REG_NOT_SET; + break; + case DW_CFA_register: + reg = elfReadLEB128(data, &bytes); + data += bytes; + state->registers.regs[reg].mode = REG_REGISTER; + state->registers.regs[reg].reg = elfReadLEB128(data, &bytes); + data += bytes; + break; + case DW_CFA_remember_state: + fs = (ELFFrameStateRegisters *)calloc(1, + sizeof(ELFFrameStateRegisters)); + memcpy(fs, &state->registers, sizeof(ELFFrameStateRegisters)); + state->registers.previous = fs; + break; + case DW_CFA_restore_state: + if(state->registers.previous == NULL) { + printf("Error: previous frame state is NULL.\n"); + return; + } + fs = state->registers.previous; + memcpy(&state->registers, fs, sizeof(ELFFrameStateRegisters)); + free(fs); + break; + case DW_CFA_def_cfa: + state->cfaRegister = elfReadLEB128(data, &bytes); + data += bytes; + state->cfaOffset = (s32)elfReadLEB128(data, &bytes); + data += bytes; + state->cfaMode = CFA_REG_OFFSET; + break; + case DW_CFA_def_cfa_register: + state->cfaRegister = elfReadLEB128(data, &bytes); + data += bytes; + state->cfaMode = CFA_REG_OFFSET; + break; + case DW_CFA_def_cfa_offset: + state->cfaOffset = (s32)elfReadLEB128(data, &bytes); + data += bytes; + state->cfaMode = CFA_REG_OFFSET; + break; + default: + printf("Unknown CFA opcode %08x\n", op); + return; + } + break; + default: + printf("Unknown CFA opcode %08x\n", op); + return; + } + } +} + +ELFFrameState *elfGetFrameState(ELFfde *fde, u32 address) +{ + ELFFrameState *state = (ELFFrameState *)calloc(1, sizeof(ELFFrameState)); + state->pc = fde->address; + state->dataAlign = fde->cie->dataAlign; + state->codeAlign = fde->cie->codeAlign; + state->returnAddress = fde->cie->returnAddress; + + elfExecuteCFAInstructions(state, + fde->cie->data, + fde->cie->dataLen, + 0xffffffff); + elfExecuteCFAInstructions(state, + fde->data, + fde->dataLen, + address); + + return state; +} + +void elfPrintCallChain(u32 address) +{ + int count = 1; + + reg_pair regs[15]; + reg_pair newRegs[15]; + + memcpy(®s[0], ®[0], sizeof(reg_pair) * 15); + + while(count < 20) { + char *addr = elfGetAddressSymbol(address); + if(*addr == 0) + addr = "???"; + + printf("%08x %s\n", address, addr); + + ELFfde *fde = elfGetFde(address); + + if(fde == NULL) { + break; + } + + ELFFrameState *state = elfGetFrameState(fde, address); + + if(!state) { + break; + } + + if(state->cfaMode == CFA_REG_OFFSET) { + memcpy(&newRegs[0], ®s[0], sizeof(reg_pair) * 15); + u32 addr = 0; + for(int i = 0; i < 15; i++) { + ELFFrameStateRegister *r = &state->registers. + regs[i]; + + switch(r->mode) { + case REG_NOT_SET: + newRegs[i].I = regs[i].I; + break; + case REG_OFFSET: + newRegs[i].I = elfReadMemory(regs[state->cfaRegister].I + + state->cfaOffset + + r->offset); + break; + case REG_REGISTER: + newRegs[i].I = regs[r->reg].I; + break; + default: + printf("Unknown register mode: %d\n", r->mode); + break; + } + } + memcpy(regs, newRegs, sizeof(reg_pair)*15); + addr = newRegs[14].I; + addr &= 0xfffffffe; + address = addr; + count++; + } else { + printf("CFA not set\n"); + break; + } + if(state->registers.previous) { + ELFFrameStateRegisters *prev = state->registers.previous; + + while(prev) { + ELFFrameStateRegisters *p = prev->previous; + free(prev); + prev = p; + } + } + free(state); + } +} + +u32 elfDecodeLocation(Function *f, ELFBlock *o, LocationType *type, u32 base) +{ + u32 framebase = 0; + if(f && f->frameBase) { + ELFBlock *b = f->frameBase; + switch(*b->data) { + case DW_OP_reg0: + case DW_OP_reg1: + case DW_OP_reg2: + case DW_OP_reg3: + case DW_OP_reg4: + case DW_OP_reg5: + case DW_OP_reg6: + case DW_OP_reg7: + case DW_OP_reg8: + case DW_OP_reg9: + case DW_OP_reg10: + case DW_OP_reg11: + case DW_OP_reg12: + case DW_OP_reg13: + case DW_OP_reg14: + case DW_OP_reg15: + framebase = reg[*b->data-0x50].I; + break; + default: + fprintf(stderr, "Unknown frameBase %02x\n", *b->data); + break; + } + } + + ELFBlock *loc = o; + u32 location = 0; + int bytes = 0; + if(loc) { + switch(*loc->data) { + case DW_OP_addr: + location = elfRead4Bytes(loc->data+1); + *type = LOCATION_memory; + break; + case DW_OP_plus_uconst: + location = base + elfReadLEB128(loc->data+1, &bytes); + *type = LOCATION_memory; + break; + case DW_OP_reg0: + case DW_OP_reg1: + case DW_OP_reg2: + case DW_OP_reg3: + case DW_OP_reg4: + case DW_OP_reg5: + case DW_OP_reg6: + case DW_OP_reg7: + case DW_OP_reg8: + case DW_OP_reg9: + case DW_OP_reg10: + case DW_OP_reg11: + case DW_OP_reg12: + case DW_OP_reg13: + case DW_OP_reg14: + case DW_OP_reg15: + location = *loc->data - 0x50; + *type = LOCATION_register; + break; + case DW_OP_fbreg: + { + int bytes; + s32 off = elfReadSignedLEB128(loc->data+1, &bytes); + location = framebase + off; + *type = LOCATION_memory; + } + break; + default: + fprintf(stderr, "Unknown location %02x\n", *loc->data); + break; + } + } + return location; +} + +u32 elfDecodeLocation(Function *f, ELFBlock *o, LocationType *type) +{ + return elfDecodeLocation(f, o, type, 0); +} + +// reading function + +u32 elfRead4Bytes(u8 *data) +{ + u32 value = *data++; + value |= (*data++ << 8); + value |= (*data++ << 16); + value |= (*data << 24); + return value; +} + +u16 elfRead2Bytes(u8 *data) +{ + u16 value = *data++; + value |= (*data << 8); + return value; +} + +char *elfReadString(u8 *data, int *bytesRead) +{ + if(*data == 0) { + *bytesRead = 1; + return NULL; + } + *bytesRead = (int)strlen((char *)data) + 1; + return (char *)data; +} + +s32 elfReadSignedLEB128(u8 *data, int *bytesRead) +{ + s32 result = 0; + int shift = 0; + int count = 0; + + u8 byte; + do { + byte = *data++; + count++; + result |= (byte & 0x7f) << shift; + shift += 7; + } while(byte & 0x80); + if((shift < 32) && (byte & 0x40)) + result |= -(1 << shift); + *bytesRead = count; + return result; +} + +u32 elfReadLEB128(u8 *data, int *bytesRead) +{ + u32 result = 0; + int shift = 0; + int count = 0; + u8 byte; + do { + byte = *data++; + count++; + result |= (byte & 0x7f) << shift; + shift += 7; + } while(byte & 0x80); + *bytesRead = count; + return result; +} + +u8 *elfReadSection(u8 *data, ELFSectionHeader *sh) +{ + return data + READ32LE(&sh->offset); +} + +ELFSectionHeader *elfGetSectionByName(char *name) +{ + for(int i = 0; i < elfSectionHeadersCount; i++) { + if(strcmp(name, + &elfSectionHeadersStringTable[READ32LE(&elfSectionHeaders[i]-> + name)]) == 0) { + return elfSectionHeaders[i]; + } + } + return NULL; +} + +ELFSectionHeader *elfGetSectionByNumber(int number) +{ + if(number < elfSectionHeadersCount) { + return elfSectionHeaders[number]; + } + return NULL; +} + +CompileUnit *elfGetCompileUnitForData(u8 *data) +{ + u8 *end = elfCurrentUnit->top + 4 + elfCurrentUnit->length; + + if(data >= elfCurrentUnit->top && data < end) + return elfCurrentUnit; + + CompileUnit *unit = elfCompileUnits; + + while(unit) { + end = unit->top + 4 + unit->length; + + if(data >= unit->top && data < end) + return unit; + + unit = unit->next; + } + + printf("Error: cannot find reference to compile unit at offset %08x\n", + (int)(data - elfDebugInfo->infodata)); + exit(-1); +} + +u8 *elfReadAttribute(u8 *data, ELFAttr *attr) +{ + int bytes; + int form = attr->form; + start: + switch(form) { + case DW_FORM_addr: + attr->value = elfRead4Bytes(data); + data += 4; + break; + case DW_FORM_data2: + attr->value = elfRead2Bytes(data); + data += 2; + break; + case DW_FORM_data4: + attr->value = elfRead4Bytes(data); + data += 4; + break; + case DW_FORM_string: + attr->string = (char *)data; + data += strlen(attr->string)+1; + break; + case DW_FORM_strp: + attr->string = elfDebugStrings + elfRead4Bytes(data); + data += 4; + break; + case DW_FORM_block: + attr->block = (ELFBlock *)malloc(sizeof(ELFBlock)); + attr->block->length = elfReadLEB128(data, &bytes); + data += bytes; + attr->block->data = data; + data += attr->block->length; + break; + case DW_FORM_block1: + attr->block = (ELFBlock *)malloc(sizeof(ELFBlock)); + attr->block->length = *data++; + attr->block->data = data; + data += attr->block->length; + break; + case DW_FORM_data1: + attr->value = *data++; + break; + case DW_FORM_flag: + attr->flag = (*data++) ? true : false; + break; + case DW_FORM_sdata: + attr->value = elfReadSignedLEB128(data, &bytes); + data += bytes; + break; + case DW_FORM_udata: + attr->value = elfReadLEB128(data, &bytes); + data += bytes; + break; + case DW_FORM_ref_addr: + attr->value = (u32)((elfDebugInfo->infodata + elfRead4Bytes(data)) - elfGetCompileUnitForData(data)->top); + data += 4; + break; + case DW_FORM_ref4: + attr->value = elfRead4Bytes(data); + data += 4; + break; + case DW_FORM_ref_udata: + attr->value = (u32)((elfDebugInfo->infodata + (elfGetCompileUnitForData(data)->top - elfDebugInfo->infodata) + elfReadLEB128(data, &bytes)) - elfCurrentUnit->top); + data += bytes; + break; + case DW_FORM_indirect: + form = elfReadLEB128(data, &bytes); + data += bytes; + goto start; + default: + fprintf(stderr, "Unsupported FORM %02x\n", form); + exit(-1); + } + return data; +} + +ELFAbbrev *elfGetAbbrev(ELFAbbrev **table, u32 number) +{ + int hash = number % 121; + + ELFAbbrev *abbrev = table[hash]; + + while(abbrev) { + if(abbrev->number == number) + return abbrev; + abbrev = abbrev->next; + } + return NULL; +} + +ELFAbbrev **elfReadAbbrevs(u8 *data, u32 offset) +{ + data += offset; + ELFAbbrev **abbrevs = (ELFAbbrev **)calloc(sizeof(ELFAbbrev *)*121,1); + int bytes = 0; + u32 number = elfReadLEB128(data, &bytes); + data += bytes; + while(number) { + ELFAbbrev *abbrev = (ELFAbbrev *)calloc(sizeof(ELFAbbrev),1); + + // read tag information + abbrev->number = number; + abbrev->tag = elfReadLEB128(data, &bytes); + data += bytes; + abbrev->hasChildren = *data++ ? true: false; + + // read attributes + int name = elfReadLEB128(data, &bytes); + data += bytes; + int form = elfReadLEB128(data, &bytes); + data += bytes; + + while(name) { + if((abbrev->numAttrs % 4) == 0) { + abbrev->attrs = (ELFAttr *)realloc(abbrev->attrs, + (abbrev->numAttrs + 4) * + sizeof(ELFAttr)); + } + abbrev->attrs[abbrev->numAttrs].name = name; + abbrev->attrs[abbrev->numAttrs++].form = form; + + name = elfReadLEB128(data, &bytes); + data += bytes; + form = elfReadLEB128(data, &bytes); + data += bytes; + } + + int hash = number % 121; + abbrev->next = abbrevs[hash]; + abbrevs[hash] = abbrev; + + number = elfReadLEB128(data, &bytes); + data += bytes; + + if(elfGetAbbrev(abbrevs, number) != NULL) + break; + } + + return abbrevs; +} + +void elfParseCFA(u8 *top) +{ + ELFSectionHeader *h = elfGetSectionByName(".debug_frame"); + + if(h == NULL) { + return; + } + + u8 *data = elfReadSection(top, h); + + u8 *topOffset = data; + + u8 *end = data + READ32LE(&h->size); + + ELFcie *cies = NULL; + + while(data < end) { + u32 offset = (u32)(data - topOffset); + u32 len = elfRead4Bytes(data); + data += 4; + + u8 *dataEnd = data + len; + + u32 id = elfRead4Bytes(data); + data += 4; + + if(id == 0xffffffff) { + // skip version + *data++; + + ELFcie *cie = (ELFcie *)calloc(1, sizeof(ELFcie)); + + cie->next = cies; + cies = cie; + + cie->offset = offset; + + cie->augmentation = data; + while(*data) + data++; + data++; + + if(*cie->augmentation) { + fprintf(stderr, "Error: augmentation not supported\n"); + exit(-1); + } + + int bytes; + cie->codeAlign = elfReadLEB128(data, &bytes); + data += bytes; + + cie->dataAlign = elfReadSignedLEB128(data, &bytes); + data += bytes; + + cie->returnAddress = *data++; + + cie->data = data; + cie->dataLen = (u32)(dataEnd - data); + } else { + ELFfde *fde = (ELFfde *)calloc(1, sizeof(ELFfde)); + + ELFcie *cie = cies; + + while(cie != NULL) { + if(cie->offset == id) + break; + cie = cie->next; + } + + if(!cie) { + fprintf(stderr, "Cannot find CIE %08x\n", id); + exit(-1); + } + + fde->cie = cie; + + fde->address = elfRead4Bytes(data); + data += 4; + + fde->end = fde->address + elfRead4Bytes(data); + data += 4; + + fde->data = data; + fde->dataLen = (u32)(dataEnd - data); + + if((elfFdeCount %10) == 0) { + elfFdes = (ELFfde **)realloc(elfFdes, (elfFdeCount+10) * + sizeof(ELFfde *)); + } + elfFdes[elfFdeCount++] = fde; + } + data = dataEnd; + } + + elfCies = cies; +} + +void elfAddLine(LineInfo *l, u32 a, int file, int line, int *max) +{ + if(l->number == *max) { + *max += 1000; + l->lines = (LineInfoItem *)realloc(l->lines, *max*sizeof(LineInfoItem)); + } + LineInfoItem *li = &l->lines[l->number]; + li->file = l->files[file-1]; + li->address = a; + li->line = line; + l->number++; +} + +void elfParseLineInfo(CompileUnit *unit, u8 *top) +{ + ELFSectionHeader *h = elfGetSectionByName(".debug_line"); + if(h == NULL) { + fprintf(stderr, "No line information found\n"); + return; + } + LineInfo *l = unit->lineInfoTable = (LineInfo *)calloc(1, sizeof(LineInfo)); + l->number = 0; + int max = 1000; + l->lines = (LineInfoItem *)malloc(1000*sizeof(LineInfoItem)); + + u8 *data = elfReadSection(top, h); + data += unit->lineInfo; + u32 totalLen = elfRead4Bytes(data); + data += 4; + u8 *end = data + totalLen; + // u16 version = elfRead2Bytes(data); + data += 2; + // u32 offset = elfRead4Bytes(data); + data += 4; + int minInstrSize = *data++; + int defaultIsStmt = *data++; + int lineBase = (s8)*data++; + int lineRange = *data++; + int opcodeBase = *data++; + u8 *stdOpLen = (u8 *)malloc(opcodeBase * sizeof(u8)); + stdOpLen[0] = 1; + int i; + for(i = 1; i < opcodeBase; i++) + stdOpLen[i] = *data++; + + free(stdOpLen);// todo + int bytes = 0; + + char *s; + while((s = elfReadString(data, &bytes)) != NULL) { + data += bytes; + // fprintf(stderr, "Directory is %s\n", s); + } + data += bytes; + int count = 4; + int index = 0; + l->files = (char **)malloc(sizeof(char *)*count); + + while((s = elfReadString(data, &bytes)) != NULL) { + l->files[index++] = s; + + data += bytes; + // directory + elfReadLEB128(data, &bytes); + data += bytes; + // time + elfReadLEB128(data, &bytes); + data += bytes; + // size + elfReadLEB128(data, &bytes); + data += bytes; + // fprintf(stderr, "File is %s\n", s); + if(index == count) { + count += 4; + l->files = (char **)realloc(l->files, sizeof(char *)*count); + } + } + l->fileCount = index; + data += bytes; + + while(data < end) { + u32 address = 0; + int file = 1; + int line = 1; + int col = 0; + int isStmt = defaultIsStmt; + int basicBlock = 0; + int endSeq = 0; + + while(!endSeq) { + int op = *data++; + switch(op) { + case DW_LNS_extended_op: + { + data++; + op = *data++; + switch(op) { + case DW_LNE_end_sequence: + endSeq = 1; + break; + case DW_LNE_set_address: + address = elfRead4Bytes(data); + data += 4; + break; + default: + fprintf(stderr, "Unknown extended LINE opcode %02x\n", op); + exit(-1); + } + } + break; + case DW_LNS_copy: + // fprintf(stderr, "Address %08x line %d (%d)\n", address, line, file); + elfAddLine(l, address, file, line, &max); + basicBlock = 0; + break; + case DW_LNS_advance_pc: + address += minInstrSize * elfReadLEB128(data, &bytes); + data += bytes; + break; + case DW_LNS_advance_line: + line += elfReadSignedLEB128(data, &bytes); + data += bytes; + break; + case DW_LNS_set_file: + file = elfReadLEB128(data, &bytes); + data += bytes; + break; + case DW_LNS_set_column: + col = elfReadLEB128(data, &bytes); + data += bytes; + break; + case DW_LNS_negate_stmt: + isStmt = !isStmt; + break; + case DW_LNS_set_basic_block: + basicBlock = 1; + break; + case DW_LNS_const_add_pc: + address += (minInstrSize *((255 - opcodeBase)/lineRange)); + break; + case DW_LNS_fixed_advance_pc: + address += elfRead2Bytes(data); + data += 2; + break; + default: + op = op - opcodeBase; + address += (op / lineRange) * minInstrSize; + line += lineBase + (op % lineRange); + elfAddLine(l, address, file, line, &max); + // fprintf(stderr, "Address %08x line %d (%d)\n", address, line,file); + basicBlock = 1; + break; + } + } + } + l->lines = (LineInfoItem *)realloc(l->lines, l->number*sizeof(LineInfoItem)); +} + +u8 *elfSkipData(u8 *data, ELFAbbrev *abbrev, ELFAbbrev **abbrevs) +{ + int i; + int bytes; + + for(i = 0; i < abbrev->numAttrs; i++) { + data = elfReadAttribute(data, &abbrev->attrs[i]); + if(abbrev->attrs[i].form == DW_FORM_block1) + free(abbrev->attrs[i].block); + } + + if(abbrev->hasChildren) { + int nesting = 1; + while(nesting) { + u32 abbrevNum = elfReadLEB128(data, &bytes); + data += bytes; + + if(!abbrevNum) { + nesting--; + continue; + } + + abbrev = elfGetAbbrev(abbrevs, abbrevNum); + + for(i = 0; i < abbrev->numAttrs; i++) { + data = elfReadAttribute(data, &abbrev->attrs[i]); + if(abbrev->attrs[i].form == DW_FORM_block1) + free(abbrev->attrs[i].block); + } + + if(abbrev->hasChildren) { + nesting++; + } + } + } + return data; +} + +Type *elfParseType(CompileUnit *unit, u32); +u8 *elfParseObject(u8 *data, ELFAbbrev *abbrev, CompileUnit *unit, + Object **object); +u8 *elfParseFunction(u8 *data, ELFAbbrev *abbrev, CompileUnit *unit, + Function **function); +void elfCleanUp(Function *); + +void elfAddType(Type *type, CompileUnit *unit, u32 offset) +{ + if(type->next == NULL) { + if(unit->types != type && type->offset == 0) { + type->offset = offset; + type->next = unit->types; + unit->types = type; + } + } +} + +void elfParseType(u8 *data, u32 offset, ELFAbbrev *abbrev, CompileUnit *unit, + Type **type) +{ + switch(abbrev->tag) { + case DW_TAG_typedef: + { + u32 typeref = 0; + char *name = NULL; + for(int i = 0; i < abbrev->numAttrs; i++) { + ELFAttr *attr = &abbrev->attrs[i]; + data = elfReadAttribute(data, attr); + switch(attr->name) { + case DW_AT_name: + name = attr->string; + break; + case DW_AT_type: + typeref = attr->value; + break; + case DW_AT_decl_file: + case DW_AT_decl_line: + break; + default: + fprintf(stderr, "Unknown attribute for typedef %02x\n", attr->name); + break; + } + } + if(abbrev->hasChildren) + fprintf(stderr, "Unexpected children for typedef\n"); + *type = elfParseType(unit, typeref); + if(name) + (*type)->name = name; + return; + } + break; + case DW_TAG_union_type: + case DW_TAG_structure_type: + { + Type *t = (Type *)calloc(sizeof(Type), 1); + if(abbrev->tag == DW_TAG_structure_type) + t->type = TYPE_struct; + else + t->type = TYPE_union; + + Struct *s = (Struct *)calloc(sizeof(Struct), 1); + t->structure = s; + elfAddType(t, unit, offset); + + for(int i = 0; i < abbrev->numAttrs; i++) { + ELFAttr *attr = &abbrev->attrs[i]; + data = elfReadAttribute(data, attr); + switch(attr->name) { + case DW_AT_name: + t->name = attr->string; + break; + case DW_AT_byte_size: + t->size = attr->value; + break; + case DW_AT_decl_file: + case DW_AT_decl_line: + case DW_AT_sibling: + case DW_AT_containing_type: // todo? + case DW_AT_declaration: + case DW_AT_specification: // TODO: + break; + default: + fprintf(stderr, "Unknown attribute for struct %02x\n", attr->name); + break; + } + } + if(abbrev->hasChildren) { + int bytes; + u32 num = elfReadLEB128(data, &bytes); + data += bytes; + int index = 0; + while(num) { + ELFAbbrev *abbr = elfGetAbbrev(unit->abbrevs, num); + + switch(abbr->tag) { + case DW_TAG_member: + { + if((index % 4) == 0) + s->members = (Member *)realloc(s->members, + sizeof(Member)*(index+4)); + Member *m = &s->members[index]; + m->location = NULL; + m->bitOffset = 0; + m->bitSize = 0; + m->byteSize = 0; + for(int i = 0; i < abbr->numAttrs; i++) { + ELFAttr *attr = &abbr->attrs[i]; + data = elfReadAttribute(data, attr); + switch(attr->name) { + case DW_AT_name: + m->name = attr->string; + break; + case DW_AT_type: + m->type = elfParseType(unit, attr->value); + break; + case DW_AT_data_member_location: + m->location = attr->block; + break; + case DW_AT_byte_size: + m->byteSize = attr->value; + break; + case DW_AT_bit_offset: + m->bitOffset = attr->value; + break; + case DW_AT_bit_size: + m->bitSize = attr->value; + break; + case DW_AT_decl_file: + case DW_AT_decl_line: + case DW_AT_accessibility: + case DW_AT_artificial: // todo? + break; + default: + fprintf(stderr, "Unknown member attribute %02x\n", + attr->name); + } + } + index++; + } + break; + case DW_TAG_subprogram: + { + Function *fnc = NULL; + data = elfParseFunction(data, abbr, unit, &fnc); + if(fnc != NULL) { + if(unit->lastFunction) + unit->lastFunction->next = fnc; + else + unit->functions = fnc; + unit->lastFunction = fnc; + } + } + break; + case DW_TAG_inheritance: + // TODO: add support + data = elfSkipData(data, abbr, unit->abbrevs); + break; + CASE_TYPE_TAG: + // skip types... parsed only when used + data = elfSkipData(data, abbr, unit->abbrevs); + break; + case DW_TAG_variable: + data = elfSkipData(data, abbr, unit->abbrevs); + break; + default: + fprintf(stderr, "Unknown struct tag %02x %s\n", abbr->tag, t->name); + data = elfSkipData(data, abbr, unit->abbrevs); + break; + } + num = elfReadLEB128(data, &bytes); + data += bytes; + } + s->memberCount = index; + } + *type = t; + return; + } + break; + case DW_TAG_base_type: + { + Type *t = (Type *)calloc(sizeof(Type), 1); + + t->type = TYPE_base; + elfAddType(t, unit, offset); + for(int i = 0; i < abbrev->numAttrs; i++) { + ELFAttr *attr = &abbrev->attrs[i]; + data = elfReadAttribute(data, attr); + switch(attr->name) { + case DW_AT_name: + t->name = attr->string; + break; + case DW_AT_encoding: + t->encoding = attr->value; + break; + case DW_AT_byte_size: + t->size = attr->value; + break; + case DW_AT_bit_size: + t->bitSize = attr->value; + break; + default: + fprintf(stderr, "Unknown attribute for base type %02x\n", + attr->name); + break; + } + } + if(abbrev->hasChildren) + fprintf(stderr, "Unexpected children for base type\n"); + *type = t; + return; + } + break; + case DW_TAG_pointer_type: + { + Type *t = (Type *)calloc(sizeof(Type), 1); + + t->type = TYPE_pointer; + + elfAddType(t, unit, offset); + + for(int i = 0; i < abbrev->numAttrs; i++) { + ELFAttr *attr = &abbrev->attrs[i]; + data =elfReadAttribute(data, attr); + switch(attr->name) { + case DW_AT_type: + t->pointer = elfParseType(unit, attr->value); + break; + case DW_AT_byte_size: + t->size = attr->value; + break; + default: + fprintf(stderr, "Unknown pointer type attribute %02x\n", attr->name); + break; + } + } + if(abbrev->hasChildren) + fprintf(stderr, "Unexpected children for pointer type\n"); + *type = t; + return; + } + break; + case DW_TAG_reference_type: + { + Type *t = (Type *)calloc(sizeof(Type), 1); + + t->type = TYPE_reference; + + elfAddType(t, unit, offset); + + for(int i = 0; i < abbrev->numAttrs; i++) { + ELFAttr *attr = &abbrev->attrs[i]; + data =elfReadAttribute(data, attr); + switch(attr->name) { + case DW_AT_type: + t->pointer = elfParseType(unit, attr->value); + break; + case DW_AT_byte_size: + t->size = attr->value; + break; + default: + fprintf(stderr, "Unknown ref type attribute %02x\n", attr->name); + break; + } + } + if(abbrev->hasChildren) + fprintf(stderr, "Unexpected children for ref type\n"); + *type = t; + return; + } + break; + case DW_TAG_volatile_type: + { + u32 typeref = 0; + + for(int i = 0; i < abbrev->numAttrs; i++) { + ELFAttr *attr = &abbrev->attrs[i]; + data = elfReadAttribute(data, attr); + switch(attr->name) { + case DW_AT_type: + typeref = attr->value; + break; + default: + fprintf(stderr, "Unknown volatile attribute for type %02x\n", + attr->name); + break; + } + } + if(abbrev->hasChildren) + fprintf(stderr, "Unexpected children for volatile type\n"); + *type = elfParseType(unit, typeref); + return; + } + break; + case DW_TAG_const_type: + { + u32 typeref = 0; + + for(int i = 0; i < abbrev->numAttrs; i++) { + ELFAttr *attr = &abbrev->attrs[i]; + data = elfReadAttribute(data, attr); + switch(attr->name) { + case DW_AT_type: + typeref = attr->value; + break; + default: + fprintf(stderr, "Unknown const attribute for type %02x\n", + attr->name); + break; + } + } + if(abbrev->hasChildren) + fprintf(stderr, "Unexpected children for const type\n"); + *type = elfParseType(unit, typeref); + return; + } + break; + case DW_TAG_enumeration_type: + { + Type *t = (Type *)calloc(sizeof(Type), 1); + t->type = TYPE_enum; + Enum *e = (Enum *)calloc(sizeof(Enum), 1); + t->enumeration = e; + elfAddType(t, unit, offset); + int count = 0; + for(int i = 0; i < abbrev->numAttrs; i++) { + ELFAttr *attr = &abbrev->attrs[i]; + data = elfReadAttribute(data, attr); + switch(attr->name) { + case DW_AT_name: + t->name = attr->string; + break; + case DW_AT_byte_size: + t->size = attr->value; + break; + case DW_AT_sibling: + case DW_AT_decl_file: + case DW_AT_decl_line: + break; + default: + fprintf(stderr, "Unknown enum attribute %02x\n", attr->name); + } + } + if(abbrev->hasChildren) { + int bytes; + u32 num = elfReadLEB128(data, &bytes); + data += bytes; + while(num) { + ELFAbbrev *abbr = elfGetAbbrev(unit->abbrevs, num); + + switch(abbr->tag) { + case DW_TAG_enumerator: + { + count++; + e->members = (EnumMember *)realloc(e->members, + count*sizeof(EnumMember)); + EnumMember *m = &e->members[count-1]; + for(int i = 0; i < abbr->numAttrs; i++) { + ELFAttr *attr = &abbr->attrs[i]; + data = elfReadAttribute(data, attr); + switch(attr->name) { + case DW_AT_name: + m->name = attr->string; + break; + case DW_AT_const_value: + m->value = attr->value; + break; + default: + fprintf(stderr, "Unknown sub param attribute %02x\n", + attr->name); + } + } + } + break; + default: + fprintf(stderr, "Unknown enum tag %02x\n", abbr->tag); + data = elfSkipData(data, abbr, unit->abbrevs); + break; + } + num = elfReadLEB128(data, &bytes); + data += bytes; + } + } + e->count = count; + *type = t; + return; + } + break; + case DW_TAG_subroutine_type: + { + Type *t = (Type *)calloc(sizeof(Type), 1); + t->type = TYPE_function; + FunctionType *f = (FunctionType *)calloc(sizeof(FunctionType), 1); + t->function = f; + elfAddType(t, unit, offset); + for(int i = 0; i < abbrev->numAttrs; i++) { + ELFAttr *attr = &abbrev->attrs[i]; + data = elfReadAttribute(data, attr); + switch(attr->name) { + case DW_AT_prototyped: + case DW_AT_sibling: + break; + case DW_AT_type: + f->returnType = elfParseType(unit, attr->value); + break; + default: + fprintf(stderr, "Unknown subroutine attribute %02x\n", attr->name); + } + } + if(abbrev->hasChildren) { + int bytes; + u32 num = elfReadLEB128(data, &bytes); + data += bytes; + Object *lastVar = NULL; + while(num) { + ELFAbbrev *abbr = elfGetAbbrev(unit->abbrevs, num); + + switch(abbr->tag) { + case DW_TAG_formal_parameter: + { + Object *o; + data = elfParseObject(data, abbr, unit, &o); + if(f->args) + lastVar->next = o; + else + f->args = o; + lastVar = o; + } + break; + case DW_TAG_unspecified_parameters: + // no use in the debugger yet + data = elfSkipData(data, abbr, unit->abbrevs); + break; + CASE_TYPE_TAG: + // skip types... parsed only when used + data = elfSkipData(data, abbr, unit->abbrevs); + break; + default: + fprintf(stderr, "Unknown subroutine tag %02x\n", abbr->tag); + data = elfSkipData(data, abbr, unit->abbrevs); + break; + } + num = elfReadLEB128(data, &bytes); + data += bytes; + } + } + *type = t; + return; + } + break; + case DW_TAG_array_type: + { + u32 typeref = 0; + int i; + Array *array = (Array *)calloc(sizeof(Array), 1); + Type *t = (Type *)calloc(sizeof(Type), 1); + t->type = TYPE_array; + elfAddType(t, unit, offset); + + for(i = 0; i < abbrev->numAttrs; i++) { + ELFAttr *attr = &abbrev->attrs[i]; + data = elfReadAttribute(data, attr); + switch(attr->name) { + case DW_AT_sibling: + break; + case DW_AT_type: + typeref = attr->value; + array->type = elfParseType(unit, typeref); + break; + default: + fprintf(stderr, "Unknown array attribute %02x\n", attr->name); + } + } + if(abbrev->hasChildren) { + int bytes; + u32 num = elfReadLEB128(data, &bytes); + data += bytes; + int index = 0; + int maxBounds = 0; + while(num) { + ELFAbbrev *abbr = elfGetAbbrev(unit->abbrevs, num); + + switch(abbr->tag) { + case DW_TAG_subrange_type: + { + if(maxBounds == index) { + maxBounds += 4; + array->bounds = (int *)realloc(array->bounds, + sizeof(int)*maxBounds); + } + for(int i = 0; i < abbr->numAttrs; i++) { + ELFAttr *attr = &abbr->attrs[i]; + data = elfReadAttribute(data, attr); + switch(attr->name) { + case DW_AT_upper_bound: + array->bounds[index] = attr->value+1; + break; + case DW_AT_type: // ignore + break; + default: + fprintf(stderr, "Unknown subrange attribute %02x\n", + attr->name); + } + } + index++; + } + break; + default: + fprintf(stderr, "Unknown array tag %02x\n", abbr->tag); + data = elfSkipData(data, abbr, unit->abbrevs); + break; + } + num = elfReadLEB128(data, &bytes); + data += bytes; + } + array->maxBounds = index; + } + t->size = array->type->size; + for(i = 0; i < array->maxBounds; i++) + t->size *= array->bounds[i]; + t->array = array; + *type = t; + return; + } + break; + default: + fprintf(stderr, "Unknown type TAG %02x\n", abbrev->tag); + exit(-1); + } +} + +Type *elfParseType(CompileUnit *unit, u32 offset) +{ + Type *t = unit->types; + + while(t) { + if(t->offset == offset) + return t; + t = t->next; + } + if(offset == 0) { + Type *t = (Type *)calloc(sizeof(Type), 1); + t->type = TYPE_void; + t->offset = 0; + elfAddType(t, unit, 0); + return t; + } + u8 *data = unit->top + offset; + int bytes; + int abbrevNum = elfReadLEB128(data, &bytes); + data += bytes; + Type *type = NULL; + + ELFAbbrev *abbrev = elfGetAbbrev(unit->abbrevs, abbrevNum); + + elfParseType(data, offset, abbrev, unit, &type); + return type; +} + +void elfGetObjectAttributes(CompileUnit *unit, u32 offset, Object *o) +{ + u8 *data = unit->top + offset; + int bytes; + u32 abbrevNum = elfReadLEB128(data, &bytes); + data += bytes; + + if(!abbrevNum) { + return; + } + + ELFAbbrev *abbrev = elfGetAbbrev(unit->abbrevs, abbrevNum); + + for(int i = 0; i < abbrev->numAttrs; i++) { + ELFAttr *attr = &abbrev->attrs[i]; + data = elfReadAttribute(data, attr); + switch(attr->name) { + case DW_AT_location: + o->location = attr->block; + break; + case DW_AT_name: + if(o->name == NULL) + o->name = attr->string; + break; + case DW_AT_MIPS_linkage_name: + o->name = attr->string; + break; + case DW_AT_decl_file: + o->file = attr->value; + break; + case DW_AT_decl_line: + o->line = attr->value; + break; + case DW_AT_type: + o->type = elfParseType(unit, attr->value); + break; + case DW_AT_external: + o->external = attr->flag; + break; + case DW_AT_const_value: + case DW_AT_abstract_origin: + case DW_AT_declaration: + case DW_AT_artificial: + // todo + break; + case DW_AT_specification: + // TODO: + break; + default: + fprintf(stderr, "Unknown object attribute %02x\n", attr->name); + break; + } + } +} + +u8 *elfParseObject(u8 *data, ELFAbbrev *abbrev, CompileUnit *unit, + Object **object) +{ + Object *o = (Object *)calloc(sizeof(Object), 1); + + o->next = NULL; + + for(int i = 0; i < abbrev->numAttrs; i++) { + ELFAttr *attr = &abbrev->attrs[i]; + data = elfReadAttribute(data, attr); + switch(attr->name) { + case DW_AT_location: + o->location = attr->block; + break; + case DW_AT_name: + if(o->name == NULL) + o->name = attr->string; + break; + case DW_AT_MIPS_linkage_name: + o->name = attr->string; + break; + case DW_AT_decl_file: + o->file = attr->value; + break; + case DW_AT_decl_line: + o->line = attr->value; + break; + case DW_AT_type: + o->type = elfParseType(unit, attr->value); + break; + case DW_AT_external: + o->external = attr->flag; + break; + case DW_AT_abstract_origin: + elfGetObjectAttributes(unit, attr->value, o); + break; + case DW_AT_const_value: + case DW_AT_declaration: + case DW_AT_artificial: + break; + case DW_AT_specification: + // TODO: + break; + default: + fprintf(stderr, "Unknown object attribute %02x\n", attr->name); + break; + } + } + *object = o; + return data; +} + +u8 *elfParseBlock(u8 *data, ELFAbbrev *abbrev, CompileUnit *unit, + Function *func, Object **lastVar) +{ + int bytes; + u32 start = func->lowPC; + u32 end = func->highPC; + + for(int i = 0; i < abbrev->numAttrs; i++) { + ELFAttr *attr = &abbrev->attrs[i]; + data = elfReadAttribute(data, attr); + switch(attr->name) { + case DW_AT_sibling: + break; + case DW_AT_low_pc: + start = attr->value; + break; + case DW_AT_high_pc: + end = attr->value; + break; + case DW_AT_ranges: // ignore for now + break; + default: + fprintf(stderr, "Unknown block attribute %02x\n", attr->name); + break; + } + } + + if(abbrev->hasChildren) { + int nesting = 1; + + while(nesting) { + u32 abbrevNum = elfReadLEB128(data, &bytes); + data += bytes; + + if(!abbrevNum) { + nesting--; + continue; + } + + abbrev = elfGetAbbrev(unit->abbrevs, abbrevNum); + + switch(abbrev->tag) { + CASE_TYPE_TAG: // types only parsed when used + case DW_TAG_label: // not needed + data = elfSkipData(data, abbrev, unit->abbrevs); + break; + case DW_TAG_lexical_block: + data = elfParseBlock(data, abbrev, unit, func, lastVar); + break; + case DW_TAG_subprogram: + { + Function *f = NULL; + data = elfParseFunction(data, abbrev, unit, &f); + if(f != NULL) { + if(unit->lastFunction) + unit->lastFunction->next = f; + else + unit->functions = f; + unit->lastFunction = f; + } + } + break; + case DW_TAG_variable: + { + Object *o; + data = elfParseObject(data, abbrev, unit, &o); + if(o->startScope == 0) + o->startScope = start; + if(o->endScope == 0) + o->endScope = 0; + if(func->variables) + (*lastVar)->next = o; + else + func->variables = o; + *lastVar = o; + } + break; + case DW_TAG_inlined_subroutine: + // TODO: + data = elfSkipData(data, abbrev, unit->abbrevs); + break; + default: + { + fprintf(stderr, "Unknown block TAG %02x\n", abbrev->tag); + data = elfSkipData(data, abbrev, unit->abbrevs); + } + break; + } + } + } + return data; +} + +void elfGetFunctionAttributes(CompileUnit *unit, u32 offset, Function *func) +{ + u8 *data = unit->top + offset; + int bytes; + u32 abbrevNum = elfReadLEB128(data, &bytes); + data += bytes; + + if(!abbrevNum) { + return; + } + + ELFAbbrev *abbrev = elfGetAbbrev(unit->abbrevs, abbrevNum); + + for(int i = 0; i < abbrev->numAttrs; i++) { + ELFAttr *attr = &abbrev->attrs[i]; + data = elfReadAttribute(data, attr); + + switch(attr->name) { + case DW_AT_sibling: + break; + case DW_AT_name: + if(func->name == NULL) + func->name = attr->string; + break; + case DW_AT_MIPS_linkage_name: + func->name = attr->string; + break; + case DW_AT_low_pc: + func->lowPC = attr->value; + break; + case DW_AT_high_pc: + func->highPC = attr->value; + break; + case DW_AT_decl_file: + func->file = attr->value; + break; + case DW_AT_decl_line: + func->line = attr->value; + break; + case DW_AT_external: + func->external = attr->flag; + break; + case DW_AT_frame_base: + func->frameBase = attr->block; + break; + case DW_AT_type: + func->returnType = elfParseType(unit, attr->value); + break; + case DW_AT_inline: + case DW_AT_specification: + case DW_AT_declaration: + case DW_AT_artificial: + case DW_AT_prototyped: + case DW_AT_proc_body: + case DW_AT_save_offset: + case DW_AT_user_2002: + case DW_AT_virtuality: + case DW_AT_containing_type: + case DW_AT_accessibility: + // todo; + break; + case DW_AT_vtable_elem_location: + free(attr->block); + break; + default: + fprintf(stderr, "Unknown function attribute %02x\n", attr->name); + break; + } + } + + return; +} + +u8 *elfParseFunction(u8 *data, ELFAbbrev *abbrev, CompileUnit *unit, + Function **f) +{ + Function *func = (Function *)calloc(sizeof(Function), 1); + *f = func; + + int bytes; + bool mangled = false; + bool declaration = false; + for(int i = 0; i < abbrev->numAttrs; i++) { + ELFAttr *attr = &abbrev->attrs[i]; + data = elfReadAttribute(data, attr); + switch(attr->name) { + case DW_AT_sibling: + break; + case DW_AT_name: + if(func->name == NULL) + func->name = attr->string; + break; + case DW_AT_MIPS_linkage_name: + func->name = attr->string; + mangled = true; + break; + case DW_AT_low_pc: + func->lowPC = attr->value; + break; + case DW_AT_high_pc: + func->highPC = attr->value; + break; + case DW_AT_prototyped: + break; + case DW_AT_decl_file: + func->file = attr->value; + break; + case DW_AT_decl_line: + func->line = attr->value; + break; + case DW_AT_external: + func->external = attr->flag; + break; + case DW_AT_frame_base: + func->frameBase = attr->block; + break; + case DW_AT_type: + func->returnType = elfParseType(unit, attr->value); + break; + case DW_AT_abstract_origin: + elfGetFunctionAttributes(unit, attr->value, func); + break; + case DW_AT_declaration: + declaration = attr->flag; + break; + case DW_AT_inline: + case DW_AT_specification: + case DW_AT_artificial: + case DW_AT_proc_body: + case DW_AT_save_offset: + case DW_AT_user_2002: + case DW_AT_virtuality: + case DW_AT_containing_type: + case DW_AT_accessibility: + // todo; + break; + case DW_AT_vtable_elem_location: + free(attr->block); + break; + default: + fprintf(stderr, "Unknown function attribute %02x\n", attr->name); + break; + } + } + + if(declaration) { + elfCleanUp(func); + free(func); + *f = NULL; + + while(1) { + u32 abbrevNum = elfReadLEB128(data, &bytes); + data += bytes; + + if(!abbrevNum) { + return data; + } + + abbrev = elfGetAbbrev(unit->abbrevs, abbrevNum); + + data = elfSkipData(data, abbrev, unit->abbrevs); + } + } + + if(abbrev->hasChildren) { + int nesting = 1; + Object *lastParam = NULL; + Object *lastVar = NULL; + + while(nesting) { + u32 abbrevNum = elfReadLEB128(data, &bytes); + data += bytes; + + if(!abbrevNum) { + nesting--; + continue; + } + + abbrev = elfGetAbbrev(unit->abbrevs, abbrevNum); + + switch(abbrev->tag) { + CASE_TYPE_TAG: // no need to parse types. only parsed when used + case DW_TAG_label: // not needed + data = elfSkipData(data, abbrev, unit->abbrevs); + break; + case DW_TAG_subprogram: + { + Function *fnc=NULL; + data = elfParseFunction(data, abbrev, unit, &fnc); + if(fnc != NULL) { + if(unit->lastFunction == NULL) + unit->functions = fnc; + else + unit->lastFunction->next = fnc; + unit->lastFunction = fnc; + } + } + break; + case DW_TAG_lexical_block: + { + data = elfParseBlock(data, abbrev, unit, func, &lastVar); + } + break; + case DW_TAG_formal_parameter: + { + Object *o; + data = elfParseObject(data, abbrev, unit, &o); + if(func->parameters) + lastParam->next = o; + else + func->parameters = o; + lastParam = o; + } + break; + case DW_TAG_variable: + { + Object *o; + data = elfParseObject(data, abbrev, unit, &o); + if(func->variables) + lastVar->next = o; + else + func->variables = o; + lastVar = o; + } + break; + case DW_TAG_unspecified_parameters: + case DW_TAG_inlined_subroutine: + { + // todo + for(int i = 0; i < abbrev->numAttrs; i++) { + data = elfReadAttribute(data, &abbrev->attrs[i]); + if(abbrev->attrs[i].form == DW_FORM_block1) + free(abbrev->attrs[i].block); + } + + if(abbrev->hasChildren) + nesting++; + } + break; + default: + { + fprintf(stderr, "Unknown function TAG %02x\n", abbrev->tag); + data = elfSkipData(data, abbrev, unit->abbrevs); + } + break; + } + } + } + return data; +} + +u8 *elfParseUnknownData(u8 *data, ELFAbbrev *abbrev, ELFAbbrev **abbrevs) +{ + int i; + int bytes; + // switch(abbrev->tag) { + // default: + fprintf(stderr, "Unknown TAG %02x\n", abbrev->tag); + + for(i = 0; i < abbrev->numAttrs; i++) { + data = elfReadAttribute(data, &abbrev->attrs[i]); + if(abbrev->attrs[i].form == DW_FORM_block1) + free(abbrev->attrs[i].block); + } + + if(abbrev->hasChildren) { + int nesting = 1; + while(nesting) { + u32 abbrevNum = elfReadLEB128(data, &bytes); + data += bytes; + + if(!abbrevNum) { + nesting--; + continue; + } + + abbrev = elfGetAbbrev(abbrevs, abbrevNum); + + fprintf(stderr, "Unknown TAG %02x\n", abbrev->tag); + + for(i = 0; i < abbrev->numAttrs; i++) { + data = elfReadAttribute(data, &abbrev->attrs[i]); + if(abbrev->attrs[i].form == DW_FORM_block1) + free(abbrev->attrs[i].block); + } + + if(abbrev->hasChildren) { + nesting++; + } + } + } + // } + return data; +} + +u8 *elfParseCompileUnitChildren(u8 *data, CompileUnit *unit) +{ + int bytes; + u32 abbrevNum = elfReadLEB128(data, &bytes); + data += bytes; + Object *lastObj = NULL; + while(abbrevNum) { + ELFAbbrev *abbrev = elfGetAbbrev(unit->abbrevs, abbrevNum); + switch(abbrev->tag) { + case DW_TAG_subprogram: + { + Function *func = NULL; + data = elfParseFunction(data, abbrev, unit, &func); + if(func != NULL) { + if(unit->lastFunction) + unit->lastFunction->next = func; + else + unit->functions = func; + unit->lastFunction = func; + } + } + break; + CASE_TYPE_TAG: + data = elfSkipData(data, abbrev, unit->abbrevs); + break; + case DW_TAG_variable: + { + Object *var = NULL; + data = elfParseObject(data, abbrev, unit, &var); + if(lastObj) + lastObj->next = var; + else + unit->variables = var; + lastObj = var; + } + break; + default: + data = elfParseUnknownData(data, abbrev, unit->abbrevs); + break; + } + + abbrevNum = elfReadLEB128(data, &bytes); + data += bytes; + } + return data; +} + + +CompileUnit *elfParseCompUnit(u8 *data, u8 *abbrevData) +{ + int bytes; + u8 *top = data; + + u32 length = elfRead4Bytes(data); + data += 4; + + u16 version = elfRead2Bytes(data); + data += 2; + + u32 offset = elfRead4Bytes(data); + data += 4; + + u8 addrSize = *data++; + + if(version != 2) { + fprintf(stderr, "Unsupported debugging information version %d\n", version); + return NULL; + } + + if(addrSize != 4) { + fprintf(stderr, "Unsupported address size %d\n", addrSize); + return NULL; + } + + ELFAbbrev **abbrevs = elfReadAbbrevs(abbrevData, offset); + + u32 abbrevNum = elfReadLEB128(data, &bytes); + data += bytes; + + ELFAbbrev *abbrev = elfGetAbbrev(abbrevs, abbrevNum); + + CompileUnit *unit = (CompileUnit *)calloc(sizeof(CompileUnit), 1); + unit->top = top; + unit->length = length; + unit->abbrevs = abbrevs; + unit->next = NULL; + + elfCurrentUnit = unit; + + int i; + + for(i = 0; i < abbrev->numAttrs; i++) { + ELFAttr *attr = &abbrev->attrs[i]; + data = elfReadAttribute(data, attr); + + switch(attr->name) { + case DW_AT_name: + unit->name = attr->string; + break; + case DW_AT_stmt_list: + unit->hasLineInfo = true; + unit->lineInfo = attr->value; + break; + case DW_AT_low_pc: + unit->lowPC = attr->value; + break; + case DW_AT_high_pc: + unit->highPC = attr->value; + break; + case DW_AT_compdir: + unit->compdir = attr->string; + break; + // ignore + case DW_AT_language: + case DW_AT_producer: + case DW_AT_macro_info: + case DW_AT_entry_pc: + break; + default: + fprintf(stderr, "Unknown attribute %02x\n", attr->name); + break; + } + } + + if(abbrev->hasChildren) + elfParseCompileUnitChildren(data, unit); + + return unit; +} + +void elfParseAranges(u8 *data) +{ + ELFSectionHeader *sh = elfGetSectionByName(".debug_aranges"); + if(sh == NULL) { + fprintf(stderr, "No aranges found\n"); + return; + } + + data = elfReadSection(data, sh); + u8 *end = data + READ32LE(&sh->size); + + int max = 4; + ARanges *ranges = (ARanges *)calloc(sizeof(ARanges), 4); + + int index = 0; + + while(data < end) { + u32 len = elfRead4Bytes(data); + data += 4; + // u16 version = elfRead2Bytes(data); + data += 2; + u32 offset = elfRead4Bytes(data); + data += 4; + // u8 addrSize = *data++; + // u8 segSize = *data++; + data += 2; // remove if uncommenting above + data += 4; + ranges[index].count = (len-20)/8; + ranges[index].offset = offset; + ranges[index].ranges = (ARange *)calloc(sizeof(ARange), (len-20)/8); + int i = 0; + while(true) { + u32 addr = elfRead4Bytes(data); + data += 4; + u32 len = elfRead4Bytes(data); + data += 4; + if(addr == 0 && len == 0) + break; + ranges[index].ranges[i].lowPC = addr; + ranges[index].ranges[i].highPC = addr+len; + i++; + } + index++; + if(index == max) { + max += 4; + ranges = (ARanges *)realloc(ranges, max*sizeof(ARanges)); + } + } + elfDebugInfo->numRanges = index; + elfDebugInfo->ranges = ranges; +} + +void elfReadSymtab(u8 *data) +{ + ELFSectionHeader *sh = elfGetSectionByName(".symtab"); + int table = READ32LE(&sh->link); + + char *strtable = (char *)elfReadSection(data, elfGetSectionByNumber(table)); + + ELFSymbol *symtab = (ELFSymbol *)elfReadSection(data, sh); + + int count = READ32LE(&sh->size) / sizeof(ELFSymbol); + elfSymbolsCount = 0; + + elfSymbols = (Symbol *)malloc(sizeof(Symbol)*count); + + int i; + + for(i = 0; i < count; i++) { + ELFSymbol *s = &symtab[i]; + int type = s->info & 15; + int binding = s->info >> 4; + + if(binding) { + Symbol *sym = &elfSymbols[elfSymbolsCount]; + sym->name = &strtable[READ32LE(&s->name)]; + sym->binding = binding; + sym->type = type; + sym->value = READ32LE(&s->value); + sym->size = READ32LE(&s->size); + elfSymbolsCount++; + } + } + for(i = 0; i < count; i++) { + ELFSymbol *s = &symtab[i]; + int bind = s->info>>4; + int type = s->info & 15; + + if(!bind) { + Symbol *sym = &elfSymbols[elfSymbolsCount]; + sym->name = &strtable[READ32LE(&s->name)]; + sym->binding = (s->info >> 4); + sym->type = type; + sym->value = READ32LE(&s->value); + sym->size = READ32LE(&s->size); + elfSymbolsCount++; + } + } + elfSymbolsStrTab = strtable; + // free(symtab); +} + +bool elfReadProgram(ELFHeader *eh, u8 *data, int& size, bool parseDebug) +{ + int count = READ16LE(&eh->e_phnum); + int i; + + if(READ32LE(&eh->e_entry) == 0x2000000) + cpuIsMultiBoot = true; + + // read program headers... should probably move this code down + u8 *p = data + READ32LE(&eh->e_phoff); + size = 0; + for(i = 0; i < count; i++) { + ELFProgramHeader *ph = (ELFProgramHeader *)p; + p += sizeof(ELFProgramHeader); + if(READ16LE(&eh->e_phentsize) != sizeof(ELFProgramHeader)) { + p += READ16LE(&eh->e_phentsize) - sizeof(ELFProgramHeader); + } + + // printf("PH %d %08x %08x %08x %08x %08x %08x %08x %08x\n", + // i, ph->type, ph->offset, ph->vaddr, ph->paddr, + // ph->filesz, ph->memsz, ph->flags, ph->align); + if(cpuIsMultiBoot) { + if(READ32LE(&ph->paddr) >= 0x2000000 && + READ32LE(&ph->paddr) <= 0x203ffff) { + memcpy(&workRAM[READ32LE(&ph->paddr) & 0x3ffff], + data + READ32LE(&ph->offset), + READ32LE(&ph->filesz)); + size += READ32LE(&ph->filesz); + } + } else { + if(READ32LE(&ph->paddr) >= 0x8000000 && + READ32LE(&ph->paddr) <= 0x9ffffff) { + memcpy(&rom[READ32LE(&ph->paddr) & 0x1ffffff], + data + READ32LE(&ph->offset), + READ32LE(&ph->filesz)); + size += READ32LE(&ph->filesz); + } + } + } + + char *stringTable = NULL; + + // read section headers + p = data + READ32LE(&eh->e_shoff); + count = READ16LE(&eh->e_shnum); + + ELFSectionHeader **sh = (ELFSectionHeader **) + malloc(sizeof(ELFSectionHeader *) * count); + + for(i = 0; i < count; i++) { + sh[i] = (ELFSectionHeader *)p; + p += sizeof(ELFSectionHeader); + if(READ16LE(&eh->e_shentsize) != sizeof(ELFSectionHeader)) + p += READ16LE(&eh->e_shentsize) - sizeof(ELFSectionHeader); + } + + if(READ16LE(&eh->e_shstrndx) != 0) { + stringTable = (char *)elfReadSection(data, + sh[READ16LE(&eh->e_shstrndx)]); + } + + elfSectionHeaders = sh; + elfSectionHeadersStringTable = stringTable; + elfSectionHeadersCount = count; + + for(i = 0; i < count; i++) { + // printf("SH %d %-20s %08x %08x %08x %08x %08x %08x %08x %08x\n", + // i, &stringTable[sh[i]->name], sh[i]->name, sh[i]->type, + // sh[i]->flags, sh[i]->addr, sh[i]->offset, sh[i]->size, + // sh[i]->link, sh[i]->info); + if(READ32LE(&sh[i]->flags) & 2) { // load section + if(cpuIsMultiBoot) { + if(READ32LE(&sh[i]->addr) >= 0x2000000 && + READ32LE(&sh[i]->addr) <= 0x203ffff) { + memcpy(&workRAM[READ32LE(&sh[i]->addr) & 0x3ffff], data + + READ32LE(&sh[i]->offset), + READ32LE(&sh[i]->size)); + size += READ32LE(&sh[i]->size); + } + } else { + if(READ32LE(&sh[i]->addr) >= 0x8000000 && + READ32LE(&sh[i]->addr) <= 0x9ffffff) { + memcpy(&rom[READ32LE(&sh[i]->addr) & 0x1ffffff], + data + READ32LE(&sh[i]->offset), + READ32LE(&sh[i]->size)); + size += READ32LE(&sh[i]->size); + } + } + } + } + + if(parseDebug) { + fprintf(stderr, "Parsing debug info\n"); + + ELFSectionHeader *dbgHeader = elfGetSectionByName(".debug_info"); + if(dbgHeader == NULL) { + fprintf(stderr, "Cannot find debug information\n"); + goto end; + } + + ELFSectionHeader *h = elfGetSectionByName(".debug_abbrev"); + if(h == NULL) { + fprintf(stderr, "Cannot find abbreviation table\n"); + goto end; + } + + elfDebugInfo = (DebugInfo *)calloc(sizeof(DebugInfo), 1); + u8 *abbrevdata = elfReadSection(data, h); + + h = elfGetSectionByName(".debug_str"); + + if(h == NULL) + elfDebugStrings = NULL; + else + elfDebugStrings = (char *)elfReadSection(data, h); + + u8 *debugdata = elfReadSection(data, dbgHeader); + + elfDebugInfo->debugdata = data; + elfDebugInfo->infodata = debugdata; + + u32 total = READ32LE(&dbgHeader->size); + u8 *end = debugdata + total; + u8 *ddata = debugdata; + + CompileUnit *last = NULL; + CompileUnit *unit = NULL; + + while(ddata < end) { + unit = elfParseCompUnit(ddata, abbrevdata); + unit->offset = (u32)(ddata-debugdata); + elfParseLineInfo(unit, data); + if(last == NULL) + elfCompileUnits = unit; + else + last->next = unit; + last = unit; + ddata += 4 + unit->length; + } + elfParseAranges(data); + CompileUnit *comp = elfCompileUnits; + while(comp) { + ARanges *r = elfDebugInfo->ranges; + for(int i = 0; i < elfDebugInfo->numRanges; i++) + if(r[i].offset == comp->offset) { + comp->ranges = &r[i]; + break; + } + comp = comp->next; + } + elfParseCFA(data); + elfReadSymtab(data); + } + end: + if(sh) { + free(sh); + } + + elfSectionHeaders = NULL; + elfSectionHeadersStringTable = NULL; + elfSectionHeadersCount = 0; + + return true; +} + +extern bool parseDebug; + +bool elfRead(const char *name, int& siz, FILE *f) +{ + fseek(f, 0, SEEK_END); + long size = ftell(f); + elfFileData = (u8 *)malloc(size); + fseek(f, 0, SEEK_SET); + fread(elfFileData, 1, size, f); + fclose(f); + + ELFHeader *header = (ELFHeader *)elfFileData; + + if(READ32LE(&header->magic) != 0x464C457F || + READ16LE(&header->e_machine) != 40 || + header->clazz != 1) { + systemMessage(0, N_("Not a valid ELF file %s"), name); + free(elfFileData); + elfFileData = NULL; + return false; + } + + if(!elfReadProgram(header, elfFileData, siz, parseDebug)) { + free(elfFileData); + elfFileData = NULL; + return false; + } + + return true; +} + +void elfCleanUp(Object *o) +{ + free(o->location); +} + +void elfCleanUp(Function *func) +{ + Object *o = func->parameters; + while(o) { + elfCleanUp(o); + Object *next = o->next; + free(o); + o = next; + } + + o = func->variables; + while(o) { + elfCleanUp(o); + Object *next = o->next; + free(o); + o = next; + } + free(func->frameBase); +} + +void elfCleanUp(ELFAbbrev **abbrevs) +{ + for(int i = 0; i < 121; i++) { + ELFAbbrev *abbrev = abbrevs[i]; + + while(abbrev) { + free(abbrev->attrs); + ELFAbbrev *next = abbrev->next; + free(abbrev); + + abbrev = next; + } + } +} + +void elfCleanUp(Type *t) +{ + switch(t->type) { + case TYPE_function: + if(t->function) { + Object *o = t->function->args; + while(o) { + elfCleanUp(o); + Object *next = o->next; + free(o); + o = next; + } + free(t->function); + } + break; + case TYPE_array: + if(t->array) { + free(t->array->bounds); + free(t->array); + } + break; + case TYPE_struct: + case TYPE_union: + if(t->structure) { + for(int i = 0; i < t->structure->memberCount; i++) { + free(t->structure->members[i].location); + } + free(t->structure->members); + free(t->structure); + } + break; + case TYPE_enum: + if(t->enumeration) { + free(t->enumeration->members); + free(t->enumeration); + } + break; + case TYPE_base: + case TYPE_pointer: + case TYPE_void: + case TYPE_reference: + break; // nothing to do + } +} + +void elfCleanUp(CompileUnit *comp) +{ + elfCleanUp(comp->abbrevs); + free(comp->abbrevs); + Function *func = comp->functions; + while(func) { + elfCleanUp(func); + Function *next = func->next; + free(func); + func = next; + } + Type *t = comp->types; + while(t) { + elfCleanUp(t); + Type *next = t->next; + free(t); + t = next; + } + Object *o = comp->variables; + while(o) { + elfCleanUp(o); + Object *next = o->next; + free(o); + o = next; + } + if(comp->lineInfoTable) { + free(comp->lineInfoTable->lines); + free(comp->lineInfoTable->files); + free(comp->lineInfoTable); + } +} + +void elfCleanUp() +{ + CompileUnit *comp = elfCompileUnits; + + while(comp) { + elfCleanUp(comp); + CompileUnit *next = comp->next; + free(comp); + comp = next; + } + elfCompileUnits = NULL; + free(elfSymbols); + elfSymbols = NULL; + // free(elfSymbolsStrTab); + elfSymbolsStrTab = NULL; + + elfDebugStrings = NULL; + if(elfDebugInfo) { + int num = elfDebugInfo->numRanges; + int i; + for(i = 0; i < num; i++) { + free(elfDebugInfo->ranges[i].ranges); + } + free(elfDebugInfo->ranges); + free(elfDebugInfo); + elfDebugInfo = NULL; + } + + if(elfFdes) { + if(elfFdeCount) { + for(int i = 0; i < elfFdeCount; i++) + free(elfFdes[i]); + } + free(elfFdes); + + elfFdes = NULL; + elfFdeCount = 0; + } + + ELFcie *cie = elfCies; + while(cie) { + ELFcie *next = cie->next; + free(cie); + cie = next; + } + elfCies = NULL; + + if(elfFileData) { + free(elfFileData); + elfFileData = NULL; + } +} diff --git a/src/gb/GB.cpp b/src/gb/GB.cpp index 55563e56..4536c2db 100644 --- a/src/gb/GB.cpp +++ b/src/gb/GB.cpp @@ -1,3217 +1,5442 @@ -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#include -#include -#include -#include - -#include "../System.h" -#include "../NLS.h" -#include "GB.h" -#include "gbCheats.h" -#include "gbGlobals.h" -#include "gbMemory.h" -#include "gbSGB.h" -#include "gbSound.h" -#include "../unzip.h" -#include "../Util.h" - -#ifdef __GNUC__ -#define _stricmp strcasecmp -#endif - -extern u8 *pix; -extern bool speedup; - -bool gbUpdateSizes(); - -// debugging -bool memorydebug = false; -char gbBuffer[2048]; - -extern u16 gbLineMix[160]; - -// mappers -void (*mapper)(u16,u8) = NULL; -void (*mapperRAM)(u16,u8) = NULL; -u8 (*mapperReadRAM)(u16) = NULL; - -// registers -gbRegister PC; -gbRegister SP; -gbRegister AF; -gbRegister BC; -gbRegister DE; -gbRegister HL; -u16 IFF; -// 0xff04 -u8 register_DIV = 0; -// 0xff05 -u8 register_TIMA = 0; -// 0xff06 -u8 register_TMA = 0; -// 0xff07 -u8 register_TAC = 0; -// 0xff0f -u8 register_IF = 0; -// 0xff40 -u8 register_LCDC = 0; -// 0xff41 -u8 register_STAT = 0; -// 0xff42 -u8 register_SCY = 0; -// 0xff43 -u8 register_SCX = 0; -// 0xff44 -u8 register_LY = 0; -// 0xff45 -u8 register_LYC = 0; -// 0xff46 -u8 register_DMA = 0; -// 0xff4a -u8 register_WY = 0; -// 0xff4b -u8 register_WX = 0; -// 0xff4f -u8 register_VBK = 0; -// 0xff51 -u8 register_HDMA1 = 0; -// 0xff52 -u8 register_HDMA2 = 0; -// 0xff53 -u8 register_HDMA3 = 0; -// 0xff54 -u8 register_HDMA4 = 0; -// 0xff55 -u8 register_HDMA5 = 0; -// 0xff70 -u8 register_SVBK = 0; -// 0xffff -u8 register_IE = 0; - -// ticks definition -int GBDIV_CLOCK_TICKS = 64; -int GBLCD_MODE_0_CLOCK_TICKS = 51; -int GBLCD_MODE_1_CLOCK_TICKS = 1140; -int GBLCD_MODE_2_CLOCK_TICKS = 20; -int GBLCD_MODE_3_CLOCK_TICKS = 43; -int GBLY_INCREMENT_CLOCK_TICKS = 114; -int GBTIMER_MODE_0_CLOCK_TICKS = 256; -int GBTIMER_MODE_1_CLOCK_TICKS = 4; -int GBTIMER_MODE_2_CLOCK_TICKS = 16; -int GBTIMER_MODE_3_CLOCK_TICKS = 64; -int GBSERIAL_CLOCK_TICKS = 128; -int GBSYNCHRONIZE_CLOCK_TICKS = 52920; - -// state variables - -// interrupt -int gbInterrupt = 0; -int gbInterruptWait = 0; -// serial -int gbSerialOn = 0; -int gbSerialTicks = 0; -int gbSerialBits = 0; -// timer -int gbTimerOn = 0; -int gbTimerTicks = 0; -int gbTimerClockTicks = 0; -int gbTimerMode = 0; -// lcd -int gbLcdMode = 2; -int gbLcdTicks = GBLCD_MODE_2_CLOCK_TICKS; -int gbLcdLYIncrementTicks = 0; -// div -int gbDivTicks = GBDIV_CLOCK_TICKS; -// cgb -int gbVramBank = 0; -int gbWramBank = 1; -int gbHdmaSource = 0x0000; -int gbHdmaDestination = 0x8000; -int gbHdmaBytes = 0x0000; -int gbHdmaOn = 0; -int gbSpeed = 0; -// frame counting -int gbFrameCount = 0; -int gbFrameSkip = 0; -int gbFrameSkipCount = 0; -// timing -u32 gbLastTime = 0; -u32 gbElapsedTime = 0; -u32 gbTimeNow = 0; -int gbSynchronizeTicks = GBSYNCHRONIZE_CLOCK_TICKS; -// emulator features -int gbBattery = 0; -int gbCaptureNumber = 0; -bool gbCapture = false; -bool gbCapturePrevious = false; -int gbJoymask[4] = { 0, 0, 0, 0 }; - -int gbRomSizes[] = { 0x00008000, // 32K - 0x00010000, // 64K - 0x00020000, // 128K - 0x00040000, // 256K - 0x00080000, // 512K - 0x00100000, // 1024K - 0x00200000, // 2048K - 0x00400000, // 4096K - 0x00800000 // 8192K -}; -int gbRomSizesMasks[] = { 0x00007fff, - 0x0000ffff, - 0x0001ffff, - 0x0003ffff, - 0x0007ffff, - 0x000fffff, - 0x001fffff, - 0x003fffff, - 0x007fffff -}; - -int gbRamSizes[6] = { 0x00000000, // 0K - 0x00000800, // 2K - 0x00002000, // 8K - 0x00008000, // 32K - 0x00020000, // 128K - 0x00010000 // 64K -}; - -int gbRamSizesMasks[6] = { 0x00000000, - 0x000007ff, - 0x00001fff, - 0x00007fff, - 0x0001ffff, - 0x0000ffff -}; - -int gbCycles[] = { -// 0 1 2 3 4 5 6 7 8 9 a b c d e f - 1, 3, 2, 2, 1, 1, 2, 1, 5, 2, 2, 2, 1, 1, 2, 1, // 0 - 1, 3, 2, 2, 1, 1, 2, 1, 3, 2, 2, 2, 1, 1, 2, 1, // 1 - 2, 3, 2, 2, 1, 1, 2, 1, 2, 2, 2, 2, 1, 1, 2, 1, // 2 - 2, 3, 2, 2, 3, 3, 3, 1, 2, 2, 2, 2, 1, 1, 2, 1, // 3 - 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // 4 - 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // 5 - 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // 6 - 2, 2, 2, 2, 2, 2, 1, 2, 1, 1, 1, 1, 1, 1, 2, 1, // 7 - 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // 8 - 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // 9 - 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // a - 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // b - 2, 3, 3, 4, 3, 4, 2, 4, 2, 4, 3, 2, 3, 6, 2, 4, // c - 2, 3, 3, 0, 3, 4, 2, 4, 2, 4, 3, 0, 3, 0, 2, 4, // d - 3, 3, 2, 0, 0, 4, 2, 4, 4, 1, 4, 0, 0, 0, 2, 4, // e - 3, 3, 2, 1, 0, 4, 2, 4, 3, 2, 4, 1, 0, 0, 2, 4 // f -}; - -int gbCyclesCB[] = { -// 0 1 2 3 4 5 6 7 8 9 a b c d e f - 2, 2, 2, 2, 2, 2, 4, 2, 2, 2, 2, 2, 2, 2, 4, 2, // 0 - 2, 2, 2, 2, 2, 2, 4, 2, 2, 2, 2, 2, 2, 2, 4, 2, // 1 - 2, 2, 2, 2, 2, 2, 4, 2, 2, 2, 2, 2, 2, 2, 4, 2, // 2 - 2, 2, 2, 2, 2, 2, 4, 2, 2, 2, 2, 2, 2, 2, 4, 2, // 3 - 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // 4 - 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // 5 - 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // 6 - 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // 7 - 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // 8 - 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // 9 - 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // a - 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // b - 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // c - 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // d - 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // e - 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2 // f -}; - -u16 DAATable[] = { - 0x0080,0x0100,0x0200,0x0300,0x0400,0x0500,0x0600,0x0700, - 0x0800,0x0900,0x1020,0x1120,0x1220,0x1320,0x1420,0x1520, - 0x1000,0x1100,0x1200,0x1300,0x1400,0x1500,0x1600,0x1700, - 0x1800,0x1900,0x2020,0x2120,0x2220,0x2320,0x2420,0x2520, - 0x2000,0x2100,0x2200,0x2300,0x2400,0x2500,0x2600,0x2700, - 0x2800,0x2900,0x3020,0x3120,0x3220,0x3320,0x3420,0x3520, - 0x3000,0x3100,0x3200,0x3300,0x3400,0x3500,0x3600,0x3700, - 0x3800,0x3900,0x4020,0x4120,0x4220,0x4320,0x4420,0x4520, - 0x4000,0x4100,0x4200,0x4300,0x4400,0x4500,0x4600,0x4700, - 0x4800,0x4900,0x5020,0x5120,0x5220,0x5320,0x5420,0x5520, - 0x5000,0x5100,0x5200,0x5300,0x5400,0x5500,0x5600,0x5700, - 0x5800,0x5900,0x6020,0x6120,0x6220,0x6320,0x6420,0x6520, - 0x6000,0x6100,0x6200,0x6300,0x6400,0x6500,0x6600,0x6700, - 0x6800,0x6900,0x7020,0x7120,0x7220,0x7320,0x7420,0x7520, - 0x7000,0x7100,0x7200,0x7300,0x7400,0x7500,0x7600,0x7700, - 0x7800,0x7900,0x8020,0x8120,0x8220,0x8320,0x8420,0x8520, - 0x8000,0x8100,0x8200,0x8300,0x8400,0x8500,0x8600,0x8700, - 0x8800,0x8900,0x9020,0x9120,0x9220,0x9320,0x9420,0x9520, - 0x9000,0x9100,0x9200,0x9300,0x9400,0x9500,0x9600,0x9700, - 0x9800,0x9900,0x00B0,0x0130,0x0230,0x0330,0x0430,0x0530, - 0x0090,0x0110,0x0210,0x0310,0x0410,0x0510,0x0610,0x0710, - 0x0810,0x0910,0x1030,0x1130,0x1230,0x1330,0x1430,0x1530, - 0x1010,0x1110,0x1210,0x1310,0x1410,0x1510,0x1610,0x1710, - 0x1810,0x1910,0x2030,0x2130,0x2230,0x2330,0x2430,0x2530, - 0x2010,0x2110,0x2210,0x2310,0x2410,0x2510,0x2610,0x2710, - 0x2810,0x2910,0x3030,0x3130,0x3230,0x3330,0x3430,0x3530, - 0x3010,0x3110,0x3210,0x3310,0x3410,0x3510,0x3610,0x3710, - 0x3810,0x3910,0x4030,0x4130,0x4230,0x4330,0x4430,0x4530, - 0x4010,0x4110,0x4210,0x4310,0x4410,0x4510,0x4610,0x4710, - 0x4810,0x4910,0x5030,0x5130,0x5230,0x5330,0x5430,0x5530, - 0x5010,0x5110,0x5210,0x5310,0x5410,0x5510,0x5610,0x5710, - 0x5810,0x5910,0x6030,0x6130,0x6230,0x6330,0x6430,0x6530, - 0x6010,0x6110,0x6210,0x6310,0x6410,0x6510,0x6610,0x6710, - 0x6810,0x6910,0x7030,0x7130,0x7230,0x7330,0x7430,0x7530, - 0x7010,0x7110,0x7210,0x7310,0x7410,0x7510,0x7610,0x7710, - 0x7810,0x7910,0x8030,0x8130,0x8230,0x8330,0x8430,0x8530, - 0x8010,0x8110,0x8210,0x8310,0x8410,0x8510,0x8610,0x8710, - 0x8810,0x8910,0x9030,0x9130,0x9230,0x9330,0x9430,0x9530, - 0x9010,0x9110,0x9210,0x9310,0x9410,0x9510,0x9610,0x9710, - 0x9810,0x9910,0xA030,0xA130,0xA230,0xA330,0xA430,0xA530, - 0xA010,0xA110,0xA210,0xA310,0xA410,0xA510,0xA610,0xA710, - 0xA810,0xA910,0xB030,0xB130,0xB230,0xB330,0xB430,0xB530, - 0xB010,0xB110,0xB210,0xB310,0xB410,0xB510,0xB610,0xB710, - 0xB810,0xB910,0xC030,0xC130,0xC230,0xC330,0xC430,0xC530, - 0xC010,0xC110,0xC210,0xC310,0xC410,0xC510,0xC610,0xC710, - 0xC810,0xC910,0xD030,0xD130,0xD230,0xD330,0xD430,0xD530, - 0xD010,0xD110,0xD210,0xD310,0xD410,0xD510,0xD610,0xD710, - 0xD810,0xD910,0xE030,0xE130,0xE230,0xE330,0xE430,0xE530, - 0xE010,0xE110,0xE210,0xE310,0xE410,0xE510,0xE610,0xE710, - 0xE810,0xE910,0xF030,0xF130,0xF230,0xF330,0xF430,0xF530, - 0xF010,0xF110,0xF210,0xF310,0xF410,0xF510,0xF610,0xF710, - 0xF810,0xF910,0x00B0,0x0130,0x0230,0x0330,0x0430,0x0530, - 0x0090,0x0110,0x0210,0x0310,0x0410,0x0510,0x0610,0x0710, - 0x0810,0x0910,0x1030,0x1130,0x1230,0x1330,0x1430,0x1530, - 0x1010,0x1110,0x1210,0x1310,0x1410,0x1510,0x1610,0x1710, - 0x1810,0x1910,0x2030,0x2130,0x2230,0x2330,0x2430,0x2530, - 0x2010,0x2110,0x2210,0x2310,0x2410,0x2510,0x2610,0x2710, - 0x2810,0x2910,0x3030,0x3130,0x3230,0x3330,0x3430,0x3530, - 0x3010,0x3110,0x3210,0x3310,0x3410,0x3510,0x3610,0x3710, - 0x3810,0x3910,0x4030,0x4130,0x4230,0x4330,0x4430,0x4530, - 0x4010,0x4110,0x4210,0x4310,0x4410,0x4510,0x4610,0x4710, - 0x4810,0x4910,0x5030,0x5130,0x5230,0x5330,0x5430,0x5530, - 0x5010,0x5110,0x5210,0x5310,0x5410,0x5510,0x5610,0x5710, - 0x5810,0x5910,0x6030,0x6130,0x6230,0x6330,0x6430,0x6530, - 0x0600,0x0700,0x0800,0x0900,0x0A00,0x0B00,0x0C00,0x0D00, - 0x0E00,0x0F00,0x1020,0x1120,0x1220,0x1320,0x1420,0x1520, - 0x1600,0x1700,0x1800,0x1900,0x1A00,0x1B00,0x1C00,0x1D00, - 0x1E00,0x1F00,0x2020,0x2120,0x2220,0x2320,0x2420,0x2520, - 0x2600,0x2700,0x2800,0x2900,0x2A00,0x2B00,0x2C00,0x2D00, - 0x2E00,0x2F00,0x3020,0x3120,0x3220,0x3320,0x3420,0x3520, - 0x3600,0x3700,0x3800,0x3900,0x3A00,0x3B00,0x3C00,0x3D00, - 0x3E00,0x3F00,0x4020,0x4120,0x4220,0x4320,0x4420,0x4520, - 0x4600,0x4700,0x4800,0x4900,0x4A00,0x4B00,0x4C00,0x4D00, - 0x4E00,0x4F00,0x5020,0x5120,0x5220,0x5320,0x5420,0x5520, - 0x5600,0x5700,0x5800,0x5900,0x5A00,0x5B00,0x5C00,0x5D00, - 0x5E00,0x5F00,0x6020,0x6120,0x6220,0x6320,0x6420,0x6520, - 0x6600,0x6700,0x6800,0x6900,0x6A00,0x6B00,0x6C00,0x6D00, - 0x6E00,0x6F00,0x7020,0x7120,0x7220,0x7320,0x7420,0x7520, - 0x7600,0x7700,0x7800,0x7900,0x7A00,0x7B00,0x7C00,0x7D00, - 0x7E00,0x7F00,0x8020,0x8120,0x8220,0x8320,0x8420,0x8520, - 0x8600,0x8700,0x8800,0x8900,0x8A00,0x8B00,0x8C00,0x8D00, - 0x8E00,0x8F00,0x9020,0x9120,0x9220,0x9320,0x9420,0x9520, - 0x9600,0x9700,0x9800,0x9900,0x9A00,0x9B00,0x9C00,0x9D00, - 0x9E00,0x9F00,0x00B0,0x0130,0x0230,0x0330,0x0430,0x0530, - 0x0610,0x0710,0x0810,0x0910,0x0A10,0x0B10,0x0C10,0x0D10, - 0x0E10,0x0F10,0x1030,0x1130,0x1230,0x1330,0x1430,0x1530, - 0x1610,0x1710,0x1810,0x1910,0x1A10,0x1B10,0x1C10,0x1D10, - 0x1E10,0x1F10,0x2030,0x2130,0x2230,0x2330,0x2430,0x2530, - 0x2610,0x2710,0x2810,0x2910,0x2A10,0x2B10,0x2C10,0x2D10, - 0x2E10,0x2F10,0x3030,0x3130,0x3230,0x3330,0x3430,0x3530, - 0x3610,0x3710,0x3810,0x3910,0x3A10,0x3B10,0x3C10,0x3D10, - 0x3E10,0x3F10,0x4030,0x4130,0x4230,0x4330,0x4430,0x4530, - 0x4610,0x4710,0x4810,0x4910,0x4A10,0x4B10,0x4C10,0x4D10, - 0x4E10,0x4F10,0x5030,0x5130,0x5230,0x5330,0x5430,0x5530, - 0x5610,0x5710,0x5810,0x5910,0x5A10,0x5B10,0x5C10,0x5D10, - 0x5E10,0x5F10,0x6030,0x6130,0x6230,0x6330,0x6430,0x6530, - 0x6610,0x6710,0x6810,0x6910,0x6A10,0x6B10,0x6C10,0x6D10, - 0x6E10,0x6F10,0x7030,0x7130,0x7230,0x7330,0x7430,0x7530, - 0x7610,0x7710,0x7810,0x7910,0x7A10,0x7B10,0x7C10,0x7D10, - 0x7E10,0x7F10,0x8030,0x8130,0x8230,0x8330,0x8430,0x8530, - 0x8610,0x8710,0x8810,0x8910,0x8A10,0x8B10,0x8C10,0x8D10, - 0x8E10,0x8F10,0x9030,0x9130,0x9230,0x9330,0x9430,0x9530, - 0x9610,0x9710,0x9810,0x9910,0x9A10,0x9B10,0x9C10,0x9D10, - 0x9E10,0x9F10,0xA030,0xA130,0xA230,0xA330,0xA430,0xA530, - 0xA610,0xA710,0xA810,0xA910,0xAA10,0xAB10,0xAC10,0xAD10, - 0xAE10,0xAF10,0xB030,0xB130,0xB230,0xB330,0xB430,0xB530, - 0xB610,0xB710,0xB810,0xB910,0xBA10,0xBB10,0xBC10,0xBD10, - 0xBE10,0xBF10,0xC030,0xC130,0xC230,0xC330,0xC430,0xC530, - 0xC610,0xC710,0xC810,0xC910,0xCA10,0xCB10,0xCC10,0xCD10, - 0xCE10,0xCF10,0xD030,0xD130,0xD230,0xD330,0xD430,0xD530, - 0xD610,0xD710,0xD810,0xD910,0xDA10,0xDB10,0xDC10,0xDD10, - 0xDE10,0xDF10,0xE030,0xE130,0xE230,0xE330,0xE430,0xE530, - 0xE610,0xE710,0xE810,0xE910,0xEA10,0xEB10,0xEC10,0xED10, - 0xEE10,0xEF10,0xF030,0xF130,0xF230,0xF330,0xF430,0xF530, - 0xF610,0xF710,0xF810,0xF910,0xFA10,0xFB10,0xFC10,0xFD10, - 0xFE10,0xFF10,0x00B0,0x0130,0x0230,0x0330,0x0430,0x0530, - 0x0610,0x0710,0x0810,0x0910,0x0A10,0x0B10,0x0C10,0x0D10, - 0x0E10,0x0F10,0x1030,0x1130,0x1230,0x1330,0x1430,0x1530, - 0x1610,0x1710,0x1810,0x1910,0x1A10,0x1B10,0x1C10,0x1D10, - 0x1E10,0x1F10,0x2030,0x2130,0x2230,0x2330,0x2430,0x2530, - 0x2610,0x2710,0x2810,0x2910,0x2A10,0x2B10,0x2C10,0x2D10, - 0x2E10,0x2F10,0x3030,0x3130,0x3230,0x3330,0x3430,0x3530, - 0x3610,0x3710,0x3810,0x3910,0x3A10,0x3B10,0x3C10,0x3D10, - 0x3E10,0x3F10,0x4030,0x4130,0x4230,0x4330,0x4430,0x4530, - 0x4610,0x4710,0x4810,0x4910,0x4A10,0x4B10,0x4C10,0x4D10, - 0x4E10,0x4F10,0x5030,0x5130,0x5230,0x5330,0x5430,0x5530, - 0x5610,0x5710,0x5810,0x5910,0x5A10,0x5B10,0x5C10,0x5D10, - 0x5E10,0x5F10,0x6030,0x6130,0x6230,0x6330,0x6430,0x6530, - 0x00C0,0x0140,0x0240,0x0340,0x0440,0x0540,0x0640,0x0740, - 0x0840,0x0940,0x0440,0x0540,0x0640,0x0740,0x0840,0x0940, - 0x1040,0x1140,0x1240,0x1340,0x1440,0x1540,0x1640,0x1740, - 0x1840,0x1940,0x1440,0x1540,0x1640,0x1740,0x1840,0x1940, - 0x2040,0x2140,0x2240,0x2340,0x2440,0x2540,0x2640,0x2740, - 0x2840,0x2940,0x2440,0x2540,0x2640,0x2740,0x2840,0x2940, - 0x3040,0x3140,0x3240,0x3340,0x3440,0x3540,0x3640,0x3740, - 0x3840,0x3940,0x3440,0x3540,0x3640,0x3740,0x3840,0x3940, - 0x4040,0x4140,0x4240,0x4340,0x4440,0x4540,0x4640,0x4740, - 0x4840,0x4940,0x4440,0x4540,0x4640,0x4740,0x4840,0x4940, - 0x5040,0x5140,0x5240,0x5340,0x5440,0x5540,0x5640,0x5740, - 0x5840,0x5940,0x5440,0x5540,0x5640,0x5740,0x5840,0x5940, - 0x6040,0x6140,0x6240,0x6340,0x6440,0x6540,0x6640,0x6740, - 0x6840,0x6940,0x6440,0x6540,0x6640,0x6740,0x6840,0x6940, - 0x7040,0x7140,0x7240,0x7340,0x7440,0x7540,0x7640,0x7740, - 0x7840,0x7940,0x7440,0x7540,0x7640,0x7740,0x7840,0x7940, - 0x8040,0x8140,0x8240,0x8340,0x8440,0x8540,0x8640,0x8740, - 0x8840,0x8940,0x8440,0x8540,0x8640,0x8740,0x8840,0x8940, - 0x9040,0x9140,0x9240,0x9340,0x9440,0x9540,0x9640,0x9740, - 0x9840,0x9940,0x3450,0x3550,0x3650,0x3750,0x3850,0x3950, - 0x4050,0x4150,0x4250,0x4350,0x4450,0x4550,0x4650,0x4750, - 0x4850,0x4950,0x4450,0x4550,0x4650,0x4750,0x4850,0x4950, - 0x5050,0x5150,0x5250,0x5350,0x5450,0x5550,0x5650,0x5750, - 0x5850,0x5950,0x5450,0x5550,0x5650,0x5750,0x5850,0x5950, - 0x6050,0x6150,0x6250,0x6350,0x6450,0x6550,0x6650,0x6750, - 0x6850,0x6950,0x6450,0x6550,0x6650,0x6750,0x6850,0x6950, - 0x7050,0x7150,0x7250,0x7350,0x7450,0x7550,0x7650,0x7750, - 0x7850,0x7950,0x7450,0x7550,0x7650,0x7750,0x7850,0x7950, - 0x8050,0x8150,0x8250,0x8350,0x8450,0x8550,0x8650,0x8750, - 0x8850,0x8950,0x8450,0x8550,0x8650,0x8750,0x8850,0x8950, - 0x9050,0x9150,0x9250,0x9350,0x9450,0x9550,0x9650,0x9750, - 0x9850,0x9950,0x9450,0x9550,0x9650,0x9750,0x9850,0x9950, - 0xA050,0xA150,0xA250,0xA350,0xA450,0xA550,0xA650,0xA750, - 0xA850,0xA950,0xA450,0xA550,0xA650,0xA750,0xA850,0xA950, - 0xB050,0xB150,0xB250,0xB350,0xB450,0xB550,0xB650,0xB750, - 0xB850,0xB950,0xB450,0xB550,0xB650,0xB750,0xB850,0xB950, - 0xC050,0xC150,0xC250,0xC350,0xC450,0xC550,0xC650,0xC750, - 0xC850,0xC950,0xC450,0xC550,0xC650,0xC750,0xC850,0xC950, - 0xD050,0xD150,0xD250,0xD350,0xD450,0xD550,0xD650,0xD750, - 0xD850,0xD950,0xD450,0xD550,0xD650,0xD750,0xD850,0xD950, - 0xE050,0xE150,0xE250,0xE350,0xE450,0xE550,0xE650,0xE750, - 0xE850,0xE950,0xE450,0xE550,0xE650,0xE750,0xE850,0xE950, - 0xF050,0xF150,0xF250,0xF350,0xF450,0xF550,0xF650,0xF750, - 0xF850,0xF950,0xF450,0xF550,0xF650,0xF750,0xF850,0xF950, - 0x00D0,0x0150,0x0250,0x0350,0x0450,0x0550,0x0650,0x0750, - 0x0850,0x0950,0x0450,0x0550,0x0650,0x0750,0x0850,0x0950, - 0x1050,0x1150,0x1250,0x1350,0x1450,0x1550,0x1650,0x1750, - 0x1850,0x1950,0x1450,0x1550,0x1650,0x1750,0x1850,0x1950, - 0x2050,0x2150,0x2250,0x2350,0x2450,0x2550,0x2650,0x2750, - 0x2850,0x2950,0x2450,0x2550,0x2650,0x2750,0x2850,0x2950, - 0x3050,0x3150,0x3250,0x3350,0x3450,0x3550,0x3650,0x3750, - 0x3850,0x3950,0x3450,0x3550,0x3650,0x3750,0x3850,0x3950, - 0x4050,0x4150,0x4250,0x4350,0x4450,0x4550,0x4650,0x4750, - 0x4850,0x4950,0x4450,0x4550,0x4650,0x4750,0x4850,0x4950, - 0x5050,0x5150,0x5250,0x5350,0x5450,0x5550,0x5650,0x5750, - 0x5850,0x5950,0x5450,0x5550,0x5650,0x5750,0x5850,0x5950, - 0x6050,0x6150,0x6250,0x6350,0x6450,0x6550,0x6650,0x6750, - 0x6850,0x6950,0x6450,0x6550,0x6650,0x6750,0x6850,0x6950, - 0x7050,0x7150,0x7250,0x7350,0x7450,0x7550,0x7650,0x7750, - 0x7850,0x7950,0x7450,0x7550,0x7650,0x7750,0x7850,0x7950, - 0x8050,0x8150,0x8250,0x8350,0x8450,0x8550,0x8650,0x8750, - 0x8850,0x8950,0x8450,0x8550,0x8650,0x8750,0x8850,0x8950, - 0x9050,0x9150,0x9250,0x9350,0x9450,0x9550,0x9650,0x9750, - 0x9850,0x9950,0x9450,0x9550,0x9650,0x9750,0x9850,0x9950, - 0xFA60,0xFB60,0xFC60,0xFD60,0xFE60,0xFF60,0x00C0,0x0140, - 0x0240,0x0340,0x0440,0x0540,0x0640,0x0740,0x0840,0x0940, - 0x0A60,0x0B60,0x0C60,0x0D60,0x0E60,0x0F60,0x1040,0x1140, - 0x1240,0x1340,0x1440,0x1540,0x1640,0x1740,0x1840,0x1940, - 0x1A60,0x1B60,0x1C60,0x1D60,0x1E60,0x1F60,0x2040,0x2140, - 0x2240,0x2340,0x2440,0x2540,0x2640,0x2740,0x2840,0x2940, - 0x2A60,0x2B60,0x2C60,0x2D60,0x2E60,0x2F60,0x3040,0x3140, - 0x3240,0x3340,0x3440,0x3540,0x3640,0x3740,0x3840,0x3940, - 0x3A60,0x3B60,0x3C60,0x3D60,0x3E60,0x3F60,0x4040,0x4140, - 0x4240,0x4340,0x4440,0x4540,0x4640,0x4740,0x4840,0x4940, - 0x4A60,0x4B60,0x4C60,0x4D60,0x4E60,0x4F60,0x5040,0x5140, - 0x5240,0x5340,0x5440,0x5540,0x5640,0x5740,0x5840,0x5940, - 0x5A60,0x5B60,0x5C60,0x5D60,0x5E60,0x5F60,0x6040,0x6140, - 0x6240,0x6340,0x6440,0x6540,0x6640,0x6740,0x6840,0x6940, - 0x6A60,0x6B60,0x6C60,0x6D60,0x6E60,0x6F60,0x7040,0x7140, - 0x7240,0x7340,0x7440,0x7540,0x7640,0x7740,0x7840,0x7940, - 0x7A60,0x7B60,0x7C60,0x7D60,0x7E60,0x7F60,0x8040,0x8140, - 0x8240,0x8340,0x8440,0x8540,0x8640,0x8740,0x8840,0x8940, - 0x8A60,0x8B60,0x8C60,0x8D60,0x8E60,0x8F60,0x9040,0x9140, - 0x9240,0x9340,0x3450,0x3550,0x3650,0x3750,0x3850,0x3950, - 0x3A70,0x3B70,0x3C70,0x3D70,0x3E70,0x3F70,0x4050,0x4150, - 0x4250,0x4350,0x4450,0x4550,0x4650,0x4750,0x4850,0x4950, - 0x4A70,0x4B70,0x4C70,0x4D70,0x4E70,0x4F70,0x5050,0x5150, - 0x5250,0x5350,0x5450,0x5550,0x5650,0x5750,0x5850,0x5950, - 0x5A70,0x5B70,0x5C70,0x5D70,0x5E70,0x5F70,0x6050,0x6150, - 0x6250,0x6350,0x6450,0x6550,0x6650,0x6750,0x6850,0x6950, - 0x6A70,0x6B70,0x6C70,0x6D70,0x6E70,0x6F70,0x7050,0x7150, - 0x7250,0x7350,0x7450,0x7550,0x7650,0x7750,0x7850,0x7950, - 0x7A70,0x7B70,0x7C70,0x7D70,0x7E70,0x7F70,0x8050,0x8150, - 0x8250,0x8350,0x8450,0x8550,0x8650,0x8750,0x8850,0x8950, - 0x8A70,0x8B70,0x8C70,0x8D70,0x8E70,0x8F70,0x9050,0x9150, - 0x9250,0x9350,0x9450,0x9550,0x9650,0x9750,0x9850,0x9950, - 0x9A70,0x9B70,0x9C70,0x9D70,0x9E70,0x9F70,0xA050,0xA150, - 0xA250,0xA350,0xA450,0xA550,0xA650,0xA750,0xA850,0xA950, - 0xAA70,0xAB70,0xAC70,0xAD70,0xAE70,0xAF70,0xB050,0xB150, - 0xB250,0xB350,0xB450,0xB550,0xB650,0xB750,0xB850,0xB950, - 0xBA70,0xBB70,0xBC70,0xBD70,0xBE70,0xBF70,0xC050,0xC150, - 0xC250,0xC350,0xC450,0xC550,0xC650,0xC750,0xC850,0xC950, - 0xCA70,0xCB70,0xCC70,0xCD70,0xCE70,0xCF70,0xD050,0xD150, - 0xD250,0xD350,0xD450,0xD550,0xD650,0xD750,0xD850,0xD950, - 0xDA70,0xDB70,0xDC70,0xDD70,0xDE70,0xDF70,0xE050,0xE150, - 0xE250,0xE350,0xE450,0xE550,0xE650,0xE750,0xE850,0xE950, - 0xEA70,0xEB70,0xEC70,0xED70,0xEE70,0xEF70,0xF050,0xF150, - 0xF250,0xF350,0xF450,0xF550,0xF650,0xF750,0xF850,0xF950, - 0xFA70,0xFB70,0xFC70,0xFD70,0xFE70,0xFF70,0x00D0,0x0150, - 0x0250,0x0350,0x0450,0x0550,0x0650,0x0750,0x0850,0x0950, - 0x0A70,0x0B70,0x0C70,0x0D70,0x0E70,0x0F70,0x1050,0x1150, - 0x1250,0x1350,0x1450,0x1550,0x1650,0x1750,0x1850,0x1950, - 0x1A70,0x1B70,0x1C70,0x1D70,0x1E70,0x1F70,0x2050,0x2150, - 0x2250,0x2350,0x2450,0x2550,0x2650,0x2750,0x2850,0x2950, - 0x2A70,0x2B70,0x2C70,0x2D70,0x2E70,0x2F70,0x3050,0x3150, - 0x3250,0x3350,0x3450,0x3550,0x3650,0x3750,0x3850,0x3950, - 0x3A70,0x3B70,0x3C70,0x3D70,0x3E70,0x3F70,0x4050,0x4150, - 0x4250,0x4350,0x4450,0x4550,0x4650,0x4750,0x4850,0x4950, - 0x4A70,0x4B70,0x4C70,0x4D70,0x4E70,0x4F70,0x5050,0x5150, - 0x5250,0x5350,0x5450,0x5550,0x5650,0x5750,0x5850,0x5950, - 0x5A70,0x5B70,0x5C70,0x5D70,0x5E70,0x5F70,0x6050,0x6150, - 0x6250,0x6350,0x6450,0x6550,0x6650,0x6750,0x6850,0x6950, - 0x6A70,0x6B70,0x6C70,0x6D70,0x6E70,0x6F70,0x7050,0x7150, - 0x7250,0x7350,0x7450,0x7550,0x7650,0x7750,0x7850,0x7950, - 0x7A70,0x7B70,0x7C70,0x7D70,0x7E70,0x7F70,0x8050,0x8150, - 0x8250,0x8350,0x8450,0x8550,0x8650,0x8750,0x8850,0x8950, - 0x8A70,0x8B70,0x8C70,0x8D70,0x8E70,0x8F70,0x9050,0x9150, - 0x9250,0x9350,0x9450,0x9550,0x9650,0x9750,0x9850,0x9950, -}; - -u8 ZeroTable[] = { - 0x80,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0 -}; - -#define GBSAVE_GAME_VERSION_1 1 -#define GBSAVE_GAME_VERSION_2 2 -#define GBSAVE_GAME_VERSION_3 3 -#define GBSAVE_GAME_VERSION_4 4 -#define GBSAVE_GAME_VERSION_5 5 -#define GBSAVE_GAME_VERSION_6 6 -#define GBSAVE_GAME_VERSION_7 7 -#define GBSAVE_GAME_VERSION_8 8 -#define GBSAVE_GAME_VERSION_9 9 -#define GBSAVE_GAME_VERSION_10 10 -#define GBSAVE_GAME_VERSION GBSAVE_GAME_VERSION_10 - -int inline gbGetValue(int min,int max,int v) -{ - return (int)(min+(float)(max-min)*(2.0*(v/31.0)-(v/31.0)*(v/31.0))); -} - -void gbGenFilter() -{ - for (int r=0;r<32;r++) { - for (int g=0;g<32;g++) { - for (int b=0;b<32;b++) { - int nr=gbGetValue(gbGetValue(4,14,g), - gbGetValue(24,29,g),r)-4; - int ng=gbGetValue(gbGetValue(4+gbGetValue(0,5,r), - 14+gbGetValue(0,3,r),b), - gbGetValue(24+gbGetValue(0,3,r), - 29+gbGetValue(0,1,r),b),g)-4; - int nb=gbGetValue(gbGetValue(4+gbGetValue(0,5,r), - 14+gbGetValue(0,3,r),g), - gbGetValue(24+gbGetValue(0,3,r), - 29+gbGetValue(0,1,r),g),b)-4; - gbColorFilter[(b<<10)|(g<<5)|r]=(nb<<10)|(ng<<5)|nr; - } - } - } -} - -bool gbIsGameboyRom(char * file) -{ - if(strlen(file) > 4) { - char * p = strrchr(file,'.'); - - if(p != NULL) { - if(_stricmp(p, ".gb") == 0) - return true; - if(_stricmp(p, ".gbc") == 0) - return true; - if(_stricmp(p, ".cgb") == 0) - return true; - if(_stricmp(p, ".sgb") == 0) - return true; - } - } - - return false; -} - -void gbCopyMemory(u16 d, u16 s, int count) -{ - while(count) { - gbMemoryMap[d>>12][d & 0x0fff] = gbMemoryMap[s>>12][s & 0x0fff]; - s++; - d++; - count--; - } -} - -void gbDoHdma() -{ - gbCopyMemory(gbHdmaDestination, - gbHdmaSource, - 0x10); - - gbHdmaDestination += 0x10; - gbHdmaSource += 0x10; - - register_HDMA2 += 0x10; - if(register_HDMA2 == 0x00) - register_HDMA1++; - - register_HDMA4 += 0x10; - if(register_HDMA4 == 0x00) - register_HDMA3++; - - if(gbHdmaDestination == 0x96b0) - gbHdmaBytes = gbHdmaBytes; - gbHdmaBytes -= 0x10; - register_HDMA5--; - if(register_HDMA5 == 0xff) - gbHdmaOn = 0; -} - -// fix for Harley and Lego Racers -void gbCompareLYToLYC() -{ - if(register_LY == register_LYC) { - // mark that we have a match - register_STAT |= 4; - - // check if we need an interrupt - if((register_STAT & 0x40) && (register_IE & 2)) - gbInterrupt |= 2; - } else // no match - register_STAT &= 0xfb; -} - -void gbWriteMemory(register u16 address, register u8 value) -{ - if(address < 0x8000) { -#ifndef FINAL_VERSION - if(memorydebug && (address>0x3fff || address < 0x2000)) { - log("Memory register write %04x=%02x PC=%04x\n", - address, - value, - PC.W); - } -#endif - if(mapper) - (*mapper)(address, value); - return; - } - - if(address < 0xa000) { - gbMemoryMap[address>>12][address&0x0fff] = value; - return; - } - - if(address < 0xc000) { -#ifndef FINAL_VERSION - if(memorydebug) { - log("Memory register write %04x=%02x PC=%04x\n", - address, - value, - PC.W); - } -#endif - - if(mapper) - (*mapperRAM)(address, value); - return; - } - - if(address < 0xfe00) { - gbMemoryMap[address>>12][address & 0x0fff] = value; - return; - } - - if(address < 0xff00) { - gbMemory[address] = value; - return; - } - - switch(address & 0x00ff) { - case 0x00: { - gbMemory[0xff00] = ((gbMemory[0xff00] & 0xcf) | - (value & 0x30)); - if(gbSgbMode) { - gbSgbDoBitTransfer(value); - } - - return; - } - - case 0x01: { - gbMemory[0xff01] = value; - return; - } - - // serial control - case 0x02: { - gbSerialOn = (value & 0x80); - gbMemory[0xff02] = value; - if(gbSerialOn) { - gbSerialTicks = GBSERIAL_CLOCK_TICKS; -#ifdef LINK_EMULATION - if(linkConnected) { - if(value & 1) { - linkSendByte(0x100|gbMemory[0xFF01]); - Sleep(5); - } - } -#endif - } - - gbSerialBits = 0; - return; - } - - // DIV register resets on any write - case 0x04: { - register_DIV = 0; - return; - } - case 0x05: - register_TIMA = value; - return; - - case 0x06: - register_TMA = value; - return; - - // TIMER control - case 0x07: { - register_TAC = value; - - gbTimerOn = (value & 4); - gbTimerMode = value & 3; - // register_TIMA = register_TMA; - switch(gbTimerMode) { - case 0: - gbTimerClockTicks = gbTimerTicks = GBTIMER_MODE_0_CLOCK_TICKS; - break; - case 1: - gbTimerClockTicks = gbTimerTicks = GBTIMER_MODE_1_CLOCK_TICKS; - break; - case 2: - gbTimerClockTicks = gbTimerTicks = GBTIMER_MODE_2_CLOCK_TICKS; - break; - case 3: - gbTimerClockTicks = gbTimerTicks = GBTIMER_MODE_3_CLOCK_TICKS; - break; - } - return; - } - - case 0x0f: { - register_IF = value; - gbInterrupt = value; - return; - } - - case 0x10: - case 0x11: - case 0x12: - case 0x13: - case 0x14: - case 0x15: - case 0x16: - case 0x17: - case 0x18: - case 0x19: - case 0x1a: - case 0x1b: - case 0x1c: - case 0x1d: - case 0x1e: - case 0x1f: - case 0x20: - case 0x21: - case 0x22: - case 0x23: - case 0x24: - case 0x25: - case 0x26: - case 0x30: - case 0x31: - case 0x32: - case 0x33: - case 0x34: - case 0x35: - case 0x36: - case 0x37: - case 0x38: - case 0x39: - case 0x3a: - case 0x3b: - case 0x3c: - case 0x3d: - case 0x3e: - case 0x3f: { - SOUND_EVENT(address,value); - return; - } - case 0x40: { - int lcdChange = (register_LCDC & 0x80) ^ (value & 0x80); - - if(lcdChange) { - if(value & 0x80) { - gbLcdTicks = GBLCD_MODE_1_CLOCK_TICKS; - gbLcdMode = 0; - register_STAT &= 0xfc; - register_LY = 0x00; - } else { - gbLcdTicks = 0; - gbLcdMode = 0; - register_STAT &= 0xfc; - register_LY = 0x00; - } - // compareLYToLYC(); - } - // don't draw the window if it was not enabled and not being drawn before - if(!(register_LCDC & 0x20) && (value & 0x20) && gbWindowLine == -1 && - register_LY > register_WY) - gbWindowLine = 144; - - register_LCDC = value; - - return; - } - - // STAT - case 0x41: { - //register_STAT = (register_STAT & 0x87) | - // (value & 0x7c); - register_STAT = (value & 0xf8) | (register_STAT & 0x07); // fix ? - // GB bug from Devrs FAQ - if(!gbCgbMode && (register_LCDC & 0x80) && gbLcdMode < 2) - gbInterrupt |= 2; - return; - } - - // SCY - case 0x42: { - register_SCY = value; - return; - } - - // SCX - case 0x43: { - register_SCX = value; - return; - } - - // LY - case 0x44: { - // read only - return; - } - - // LYC - case 0x45: { - register_LYC = value; - if((register_LCDC & 0x80)) { - gbCompareLYToLYC(); - } - return; - } - - // DMA! - case 0x46: { - int source = value * 0x0100; - - gbCopyMemory(0xfe00, - source, - 0xa0); - register_DMA = value; - return; - } - - // BGP - case 0x47: { - gbBgp[0] = value & 0x03; - gbBgp[1] = (value & 0x0c)>>2; - gbBgp[2] = (value & 0x30)>>4; - gbBgp[3] = (value & 0xc0)>>6; - break; - } - - // OBP0 - case 0x48: { - gbObp0[0] = value & 0x03; - gbObp0[1] = (value & 0x0c)>>2; - gbObp0[2] = (value & 0x30)>>4; - gbObp0[3] = (value & 0xc0)>>6; - break; - } - - // OBP1 - case 0x49: { - gbObp1[0] = value & 0x03; - gbObp1[1] = (value & 0x0c)>>2; - gbObp1[2] = (value & 0x30)>>4; - gbObp1[3] = (value & 0xc0)>>6; - break; - } - - case 0x4a: - register_WY = value; - return; - - case 0x4b: - register_WX = value; - return; - - // KEY1 - case 0x4d: { - if(gbCgbMode) { - gbMemory[0xff4d] = (gbMemory[0xff4d] & 0x80) | (value & 1); - return; - } - } - break; - - // VBK - case 0x4f: { - if(gbCgbMode) { - value = value & 1; - if(value == gbVramBank) - return; - - int vramAddress = value * 0x2000; - gbMemoryMap[0x08] = &gbVram[vramAddress]; - gbMemoryMap[0x09] = &gbVram[vramAddress + 0x1000]; - - gbVramBank = value; - register_VBK = value; - } - return; - } - break; - - // HDMA1 - case 0x51: { - if(gbCgbMode) { - if(value > 0x7f && value < 0xa0) - value = 0; - - gbHdmaSource = (value << 8) | (register_HDMA2 & 0xf0); - - register_HDMA1 = value; - return; - } - } - break; - - // HDMA2 - case 0x52: { - if(gbCgbMode) { - value = value & 0xf0; - - gbHdmaSource = (register_HDMA1 << 8) | (value); - - register_HDMA2 = value; - return; - } - } - break; - - // HDMA3 - case 0x53: { - if(gbCgbMode) { - value = value & 0x1f; - gbHdmaDestination = (value << 8) | (register_HDMA4 & 0xf0); - gbHdmaDestination += 0x8000; - register_HDMA3 = value; - return; - } - } - break; - - // HDMA4 - case 0x54: { - if(gbCgbMode) { - value = value & 0xf0; - gbHdmaDestination = ((register_HDMA3 & 0x1f) << 8) | value; - gbHdmaDestination += 0x8000; - register_HDMA4 = value; - return; - } - } - break; - - // HDMA5 - case 0x55: { - if(gbCgbMode) { - gbHdmaBytes = 16 + (value & 0x7f) * 16; - if(gbHdmaOn) { - if(value & 0x80) { - register_HDMA5 = (value & 0x7f); - } else { - register_HDMA5 = 0xff; - gbHdmaOn = 0; - } - } else { - if(value & 0x80) { - gbHdmaOn = 1; - register_HDMA5 = value & 0x7f; - if(gbLcdMode == 0) - gbDoHdma(); - } else { - // we need to take the time it takes to complete the transfer into - // account... according to GB DEV FAQs, the setup time is the same - // for single and double speed, but the actual transfer takes the - // same time - if(gbSpeed) - gbDmaTicks = 231 + 16 * (value & 0x7f); - else - gbDmaTicks = 231 + 8 * (value & 0x7f); - gbCopyMemory(gbHdmaDestination, - gbHdmaSource, - gbHdmaBytes); - gbHdmaDestination += gbHdmaBytes; - gbHdmaSource += gbHdmaBytes; - - register_HDMA3 = ((gbHdmaDestination - 0x8000) >> 8) & 0x1f; - register_HDMA4 = gbHdmaDestination & 0xf0; - register_HDMA1 = (gbHdmaSource >> 8) & 0xff; - register_HDMA2 = gbHdmaSource & 0xf0; - } - } - return; - } - } - break; - - // BCPS - case 0x68: { - if(gbCgbMode) { - int paletteIndex = (value & 0x3f) >> 1; - int paletteHiLo = (value & 0x01); - - gbMemory[0xff68] = value; - gbMemory[0xff69] = (paletteHiLo ? - (gbPalette[paletteIndex] >> 8) : - (gbPalette[paletteIndex] & 0x00ff)); - return; - } - } - break; - - // BCPD - case 0x69: { - if(gbCgbMode) { - int v = gbMemory[0xff68]; - int paletteIndex = (v & 0x3f) >> 1; - int paletteHiLo = (v & 0x01); - gbMemory[0xff69] = value; - gbPalette[paletteIndex] = (paletteHiLo ? - ((value << 8) | (gbPalette[paletteIndex] & 0xff)) : - ((gbPalette[paletteIndex] & 0xff00) | (value))) & 0x7fff; - - if(gbMemory[0xff68] & 0x80) { - int index = ((gbMemory[0xff68] & 0x3f) + 1) & 0x3f; - - gbMemory[0xff68] = (gbMemory[0xff68] & 0x80) | index; - - gbMemory[0xff69] = (index & 1 ? - (gbPalette[index>>1] >> 8) : - (gbPalette[index>>1] & 0x00ff)); - - } - return; - } - } - break; - - // OCPS - case 0x6a: { - if(gbCgbMode) { - int paletteIndex = (value & 0x3f) >> 1; - int paletteHiLo = (value & 0x01); - - paletteIndex += 32; - - gbMemory[0xff6a] = value; - gbMemory[0xff6b] = (paletteHiLo ? - (gbPalette[paletteIndex] >> 8) : - (gbPalette[paletteIndex] & 0x00ff)); - return; - } - } - break; - - // OCPD - case 0x6b: { - if(gbCgbMode) { - int v = gbMemory[0xff6a]; - int paletteIndex = (v & 0x3f) >> 1; - int paletteHiLo = (v & 0x01); - - paletteIndex += 32; - - gbMemory[0xff6b] = value; - gbPalette[paletteIndex] = (paletteHiLo ? - ((value << 8) | (gbPalette[paletteIndex] & 0xff)) : - ((gbPalette[paletteIndex] & 0xff00) | (value))) & 0x7fff; - if(gbMemory[0xff6a] & 0x80) { - int index = ((gbMemory[0xff6a] & 0x3f) + 1) & 0x3f; - - gbMemory[0xff6a] = (gbMemory[0xff6a] & 0x80) | index; - - gbMemory[0xff6b] = (index & 1 ? - (gbPalette[(index>>1) + 32] >> 8) : - (gbPalette[(index>>1) + 32] & 0x00ff)); - - } - return; - } - } - break; - - // SVBK - case 0x70: { - if(gbCgbMode) { - value = value & 7; - - int bank = value; - if(value == 0) - bank = 1; - - if(bank == gbWramBank) - return; - - int wramAddress = bank * 0x1000; - gbMemoryMap[0x0d] = &gbWram[wramAddress]; - - gbWramBank = bank; - register_SVBK = value; - return; - } - } - break; - - case 0xff: { - register_IE = value; - register_IF &= value; - return; - } - } - - gbMemory[address] = value; -} - -u8 gbReadOpcode(register u16 address) -{ - if(gbCheatMap[address]) - return gbCheatRead(address); - - switch(address & 0xf000) { - case 0x0a: - case 0x0b: - if(mapperReadRAM) - return mapperReadRAM(address); - break; - case 0x0f: - if(address > 0xff00) { - if (address >= 0xFF10 && address <= 0xFF3F) - { - return gbSoundRead(address); - } - switch(address & 0x00ff) { - case 0x04: - return register_DIV; - case 0x05: - return register_TIMA; - case 0x06: - return register_TMA; - case 0x07: - return (0xf8 | register_TAC); - case 0x0f: - return (0xe0 | register_IF); - case 0x40: - return register_LCDC; - case 0x41: - return (0x80 | register_STAT); - case 0x42: - return register_SCY; - case 0x43: - return register_SCX; - case 0x44: - return register_LY; - case 0x45: - return register_LYC; - case 0x46: - return register_DMA; - case 0x4a: - return register_WY; - case 0x4b: - return register_WX; - case 0x4f: - return (0xfe | register_VBK); - case 0x51: - return register_HDMA1; - case 0x52: - return register_HDMA2; - case 0x53: - return register_HDMA3; - case 0x54: - return register_HDMA4; - case 0x55: - return register_HDMA5; - case 0x70: - return (0xf8 | register_SVBK); - case 0xff: - return register_IE; - } - } - break; - } - return gbMemoryMap[address>>12][address & 0x0fff]; -} - -u8 gbReadMemory(register u16 address) -{ - if(gbCheatMap[address]) - return gbCheatRead(address); - - if(address < 0xa000) - return gbMemoryMap[address>>12][address&0x0fff]; - - if(address < 0xc000) { -#ifndef FINAL_VERSION - if(memorydebug) { - log("Memory register read %04x PC=%04x\n", - address, - PC.W); - } -#endif - - if(mapperReadRAM) - return mapperReadRAM(address); - return gbMemoryMap[address>>12][address & 0x0fff]; - } - - if(address >= 0xff00) { - - if (address >= 0xFF10 && address <= 0xFF3F) { - return gbSoundRead(address); - } - - switch(address & 0x00ff) { - case 0x00: - { - if(gbSgbMode) { - gbSgbReadingController |= 4; - gbSgbResetPacketState(); - } - - int b = gbMemory[0xff00]; - - if((b & 0x30) == 0x20) { - b &= 0xf0; - - int joy = 0; - if(gbSgbMode && gbSgbMultiplayer) { - switch(gbSgbNextController) { - case 0x0f: - joy = 0; - break; - case 0x0e: - joy = 1; - break; - case 0x0d: - joy = 2; - break; - case 0x0c: - joy = 3; - break; - default: - joy = 0; - } - } - int joystate = gbJoymask[joy]; - if(!(joystate & 128)) - b |= 0x08; - if(!(joystate & 64)) - b |= 0x04; - if(!(joystate & 32)) - b |= 0x02; - if(!(joystate & 16)) - b |= 0x01; - - gbMemory[0xff00] = b; - } else if((b & 0x30) == 0x10) { - b &= 0xf0; - - int joy = 0; - if(gbSgbMode && gbSgbMultiplayer) { - switch(gbSgbNextController) { - case 0x0f: - joy = 0; - break; - case 0x0e: - joy = 1; - break; - case 0x0d: - joy = 2; - break; - case 0x0c: - joy = 3; - break; - default: - joy = 0; - } - } - int joystate = gbJoymask[joy]; - if(!(joystate & 8)) - b |= 0x08; - if(!(joystate & 4)) - b |= 0x04; - if(!(joystate & 2)) - b |= 0x02; - if(!(joystate & 1)) - b |= 0x01; - - gbMemory[0xff00] = b; - } else { - if(gbSgbMode && gbSgbMultiplayer) { - gbMemory[0xff00] = 0xf0 | gbSgbNextController; - } else { - gbMemory[0xff00] = 0xff; - } - } - } - return gbMemory[0xff00]; - break; - case 0x01: - return gbMemory[0xff01]; - case 0x04: - return register_DIV; - case 0x05: - return register_TIMA; - case 0x06: - return register_TMA; - case 0x07: - return (0xf8 | register_TAC); - case 0x0f: - return (0xe0 | register_IF); - case 0x40: - return register_LCDC; - case 0x41: - return (0x80 | register_STAT); - case 0x42: - return register_SCY; - case 0x43: - return register_SCX; - case 0x44: - return register_LY; - case 0x45: - return register_LYC; - case 0x46: - return register_DMA; - case 0x4a: - return register_WY; - case 0x4b: - return register_WX; - case 0x4f: - return (0xfe | register_VBK); - case 0x51: - return register_HDMA1; - case 0x52: - return register_HDMA2; - case 0x53: - return register_HDMA3; - case 0x54: - return register_HDMA4; - case 0x55: - return register_HDMA5; - case 0x70: - return (0xf8 | register_SVBK); - case 0xff: - return register_IE; - } - } - - return gbMemoryMap[address>>12][address & 0x0fff]; -} - -void gbVblank_interrupt() -{ - if(IFF & 0x80) { - PC.W++; - IFF &= 0x7f; - } - gbInterrupt &= 0xfe; - - IFF &= 0x7e; - register_IF &= 0xfe; - - gbWriteMemory(--SP.W, PC.B.B1); - gbWriteMemory(--SP.W, PC.B.B0); - PC.W = 0x40; -} - -void gbLcd_interrupt() -{ - if(IFF & 0x80) { - PC.W++; - IFF &= 0x7f; - } - gbInterrupt &= 0xfd; - IFF &= 0x7e; - register_IF &= 0xfd; - - gbWriteMemory(--SP.W, PC.B.B1); - gbWriteMemory(--SP.W, PC.B.B0); - - PC.W = 0x48; -} - -void gbTimer_interrupt() -{ - if(IFF & 0x80) { - PC.W++; - IFF &= 0x7f; - } - IFF &= 0x7e; - gbInterrupt &= 0xfb; - register_IF &= 0xfb; - - gbWriteMemory(--SP.W, PC.B.B1); - gbWriteMemory(--SP.W, PC.B.B0); - - PC.W = 0x50; -} - -void gbSerial_interrupt() -{ - if(IFF & 0x80) { - PC.W++; - IFF &= 0x7f; - } - IFF &= 0x7e; - gbInterrupt &= 0xf7; - register_IF &= 0xf7; - - gbWriteMemory(--SP.W, PC.B.B1); - gbWriteMemory(--SP.W, PC.B.B0); - - PC.W = 0x58; -} - -void gbJoypad_interrupt() -{ - if(IFF & 0x80) { - PC.W++; - IFF &= 0x7f; - } - IFF &= 0x7e; - gbInterrupt &= 0xef; - register_IF &= 0xef; - - gbWriteMemory(--SP.W, PC.B.B1); - gbWriteMemory(--SP.W, PC.B.B0); - - PC.W = 0x60; -} - -void gbSpeedSwitch() -{ - if(gbSpeed == 0) { - gbSpeed = 1; - GBLCD_MODE_0_CLOCK_TICKS = 51 * 2; //127; //51 * 2; - GBLCD_MODE_1_CLOCK_TICKS = 1140 * 2; - GBLCD_MODE_2_CLOCK_TICKS = 20 * 2; //52; //20 * 2; - GBLCD_MODE_3_CLOCK_TICKS = 43 * 2; //99; //43 * 2; - GBDIV_CLOCK_TICKS = 64 * 2; - GBLY_INCREMENT_CLOCK_TICKS = 114 * 2; - GBTIMER_MODE_0_CLOCK_TICKS = 256; //256*2; - GBTIMER_MODE_1_CLOCK_TICKS = 4; //4*2; - GBTIMER_MODE_2_CLOCK_TICKS = 16; //16*2; - GBTIMER_MODE_3_CLOCK_TICKS = 64; //64*2; - GBSERIAL_CLOCK_TICKS = 128 * 2; - gbDivTicks *= 2; - gbLcdTicks *= 2; - gbLcdLYIncrementTicks *= 2; - // timerTicks *= 2; - // timerClockTicks *= 2; - gbSerialTicks *= 2; - SOUND_CLOCK_TICKS = soundQuality * 24 * 2; - soundTicks *= 2; - // synchronizeTicks *= 2; - // SYNCHRONIZE_CLOCK_TICKS *= 2; - } else { - gbSpeed = 0; - GBLCD_MODE_0_CLOCK_TICKS = 51; - GBLCD_MODE_1_CLOCK_TICKS = 1140; - GBLCD_MODE_2_CLOCK_TICKS = 20; - GBLCD_MODE_3_CLOCK_TICKS = 43; - GBDIV_CLOCK_TICKS = 64; - GBLY_INCREMENT_CLOCK_TICKS = 114; - GBTIMER_MODE_0_CLOCK_TICKS = 256; - GBTIMER_MODE_1_CLOCK_TICKS = 4; - GBTIMER_MODE_2_CLOCK_TICKS = 16; - GBTIMER_MODE_3_CLOCK_TICKS = 64; - GBSERIAL_CLOCK_TICKS = 128; - gbDivTicks /= 2; - gbLcdTicks /= 2; - gbLcdLYIncrementTicks /= 2; - // timerTicks /= 2; - // timerClockTicks /= 2; - gbSerialTicks /= 2; - SOUND_CLOCK_TICKS = soundQuality * 24; - soundTicks /= 2; - // synchronizeTicks /= 2; - // SYNCHRONIZE_CLOCK_TICKS /= 2; - } -} - -void gbReset() -{ - SP.W = 0xfffe; - AF.W = 0x01b0; - BC.W = 0x0013; - DE.W = 0x00d8; - HL.W = 0x014d; - PC.W = 0x0100; - IFF = 0; - gbInterrupt = 1; - gbInterruptWait = 0; - - register_DIV = 0; - register_TIMA = 0; - register_TMA = 0; - register_TAC = 0; - register_IF = 1; - register_LCDC = 0x91; - register_STAT = 0; - register_SCY = 0; - register_SCX = 0; - register_LY = 0; - register_LYC = 0; - register_DMA = 0; - register_WY = 0; - register_WX = 0; - register_VBK = 0; - register_HDMA1 = 0; - register_HDMA2 = 0; - register_HDMA3 = 0; - register_HDMA4 = 0; - register_HDMA5 = 0; - register_SVBK = 0; - register_IE = 0; - - if(gbCgbMode) { - if(gbSgbMode) { - if(gbEmulatorType == 5) - AF.W = 0xffb0; - else - AF.W = 0x01b0; - BC.W = 0x0013; - DE.W = 0x00d8; - HL.W = 0x014d; - for(int i = 0; i < 8; i++) - gbPalette[i] = systemGbPalette[gbPaletteOption*8+i]; - } else { - AF.W = 0x11b0; - BC.W = 0x0000; - DE.W = 0xff56; - HL.W = 0x000d; - } - if(gbEmulatorType == 4) - BC.B.B1 |= 0x01; - - register_HDMA5 = 0xff; - gbMemory[0xff68] = 0xc0; - gbMemory[0xff6a] = 0xc0; - } else { - for(int i = 0; i < 8; i++) - gbPalette[i] = systemGbPalette[gbPaletteOption*8+i]; - } - - if(gbSpeed) { - gbSpeedSwitch(); - gbMemory[0xff4d] = 0; - } - - gbDivTicks = GBDIV_CLOCK_TICKS; - gbLcdMode = 2; - gbLcdTicks = GBLCD_MODE_2_CLOCK_TICKS; - gbLcdLYIncrementTicks = 0; - gbTimerTicks = 0; - gbTimerClockTicks = 0; - gbSerialTicks = 0; - gbSerialBits = 0; - gbSerialOn = 0; - gbWindowLine = -1; - gbTimerOn = 0; - gbTimerMode = 0; - // gbSynchronizeTicks = GBSYNCHRONIZE_CLOCK_TICKS; - gbSpeed = 0; - gbJoymask[0] = gbJoymask[1] = gbJoymask[2] = gbJoymask[3] = 0; - - if(gbCgbMode) { - gbSpeed = 0; - gbHdmaOn = 0; - gbHdmaSource = 0x0000; - gbHdmaDestination = 0x8000; - gbVramBank = 0; - gbWramBank = 1; - register_LY = 0x90; - gbLcdMode = 1; - for(int i = 0; i < 64; i++) - gbPalette[i] = 0x7fff; - } - - if(gbSgbMode) { - gbSgbReset(); - } - - for(int i =0; i < 4; i++) - gbBgp[i] = gbObp0[i] = gbObp1[i] = i; - - memset(&gbDataMBC1,0, sizeof(gbDataMBC1)); - gbDataMBC1.mapperROMBank = 1; - - gbDataMBC2.mapperRAMEnable = 0; - gbDataMBC2.mapperROMBank = 1; - - memset(&gbDataMBC3,0, 6 * sizeof(int)); - gbDataMBC3.mapperROMBank = 1; - - memset(&gbDataMBC5, 0, sizeof(gbDataMBC5)); - gbDataMBC5.mapperROMBank = 1; - - memset(&gbDataHuC1, 0, sizeof(gbDataHuC1)); - gbDataHuC1.mapperROMBank = 1; - - memset(&gbDataHuC3, 0, sizeof(gbDataHuC3)); - gbDataHuC3.mapperROMBank = 1; - - gbMemoryMap[0x00] = &gbRom[0x0000]; - gbMemoryMap[0x01] = &gbRom[0x1000]; - gbMemoryMap[0x02] = &gbRom[0x2000]; - gbMemoryMap[0x03] = &gbRom[0x3000]; - gbMemoryMap[0x04] = &gbRom[0x4000]; - gbMemoryMap[0x05] = &gbRom[0x5000]; - gbMemoryMap[0x06] = &gbRom[0x6000]; - gbMemoryMap[0x07] = &gbRom[0x7000]; - if(gbCgbMode) { - gbMemoryMap[0x08] = &gbVram[0x0000]; - gbMemoryMap[0x09] = &gbVram[0x1000]; - gbMemoryMap[0x0a] = &gbMemory[0xa000]; - gbMemoryMap[0x0b] = &gbMemory[0xb000]; - gbMemoryMap[0x0c] = &gbMemory[0xc000]; - gbMemoryMap[0x0d] = &gbWram[0x1000]; - gbMemoryMap[0x0e] = &gbMemory[0xe000]; - gbMemoryMap[0x0f] = &gbMemory[0xf000]; - } else { - gbMemoryMap[0x08] = &gbMemory[0x8000]; - gbMemoryMap[0x09] = &gbMemory[0x9000]; - gbMemoryMap[0x0a] = &gbMemory[0xa000]; - gbMemoryMap[0x0b] = &gbMemory[0xb000]; - gbMemoryMap[0x0c] = &gbMemory[0xc000]; - gbMemoryMap[0x0d] = &gbMemory[0xd000]; - gbMemoryMap[0x0e] = &gbMemory[0xe000]; - gbMemoryMap[0x0f] = &gbMemory[0xf000]; - } - - if(gbRam) { - gbMemoryMap[0x0a] = &gbRam[0x0000]; - gbMemoryMap[0x0b] = &gbRam[0x1000]; - } - - gbSoundReset(); - - systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; - - gbLastTime = systemGetClock(); - gbFrameCount = 0; -} - -void gbWriteSaveMBC1(const char * name) -{ - FILE *gzFile = fopen(name,"wb"); - - if(gzFile == NULL) { - systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), name); - return; - } - - fwrite(gbRam, - 1, - gbRamSize, - gzFile); - - fclose(gzFile); -} - -void gbWriteSaveMBC2(const char * name) -{ - FILE *file = fopen(name, "wb"); - - if(file == NULL) { - systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), name); - return; - } - - fwrite(&gbMemory[0xa000], - 1, - 256, - file); - - fclose(file); -} - -void gbWriteSaveMBC3(const char * name, bool extendedSave) -{ - FILE *gzFile = fopen(name,"wb"); - - if(gzFile == NULL) { - systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), name); - return; - } - - fwrite(gbRam, - 1, - gbRamSize, - gzFile); - - if(extendedSave) - fwrite(&gbDataMBC3.mapperSeconds, - 1, - 10*sizeof(int) + sizeof(time_t), - gzFile); - - fclose(gzFile); -} - -void gbWriteSaveMBC5(const char * name) -{ - FILE *gzFile = fopen(name,"wb"); - - if(gzFile == NULL) { - systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), name); - return; - } - - fwrite(gbRam, - 1, - gbRamSize, - gzFile); - - fclose(gzFile); -} - -void gbWriteSaveMBC7(const char * name) -{ - FILE *file = fopen(name, "wb"); - - if(file == NULL) { - systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), name); - return; - } - - fwrite(&gbMemory[0xa000], - 1, - 256, - file); - - fclose(file); -} - -bool gbReadSaveMBC1(const char * name) -{ - gzFile gzFile = gzopen(name, "rb"); - - if(gzFile == NULL) { - return false; - } - - int read = gzread(gzFile, - gbRam, - gbRamSize); - - if(read != gbRamSize) { - systemMessage(MSG_FAILED_TO_READ_SGM, N_("Failed to read complete save game %s (%d)"), name, read); - gzclose(gzFile); - return false; - } - - gzclose(gzFile); - return true; -} - -bool gbReadSaveMBC2(const char * name) -{ - FILE *file = fopen(name, "rb"); - - if(file == NULL) { - return false; - } - - int read = fread(&gbMemory[0xa000], - 1, - 256, - file); - - if(read != 256) { - systemMessage(MSG_FAILED_TO_READ_SGM, - N_("Failed to read complete save game %s (%d)"), name, read); - fclose(file); - return false; - } - - fclose(file); - return true; -} - -bool gbReadSaveMBC3(const char * name) -{ - gzFile gzFile = gzopen(name, "rb"); - - if(gzFile == NULL) { - return false; - } - - int read = gzread(gzFile, - gbRam, - gbRamSize); - - bool res = true; - - if(read != gbRamSize) { - systemMessage(MSG_FAILED_TO_READ_SGM, - N_("Failed to read complete save game %s (%d)"), name, read); - } else { - read = gzread(gzFile, - &gbDataMBC3.mapperSeconds, - sizeof(int)*10 + sizeof(time_t)); - - if(read != (sizeof(int)*10 + sizeof(time_t)) && read != 0) { - systemMessage(MSG_FAILED_TO_READ_RTC, - N_("Failed to read RTC from save game %s (continuing)"), - name); - res = false; - } - } - - gzclose(gzFile); - return res; -} - -bool gbReadSaveMBC5(const char * name) -{ - gzFile gzFile = gzopen(name, "rb"); - - if(gzFile == NULL) { - return false; - } - - int read = gzread(gzFile, - gbRam, - gbRamSize); - - if(read != gbRamSize) { - systemMessage(MSG_FAILED_TO_READ_SGM, - N_("Failed to read complete save game %s (%d)"), name, read); - gzclose(gzFile); - return false; - } - - gzclose(gzFile); - return true; -} - -bool gbReadSaveMBC7(const char * name) -{ - FILE *file = fopen(name, "rb"); - - if(file == NULL) { - return false; - } - - int read = fread(&gbMemory[0xa000], - 1, - 256, - file); - - if(read != 256) { - systemMessage(MSG_FAILED_TO_READ_SGM, - N_("Failed to read complete save game %s (%d)"), name, read); - fclose(file); - return false; - } - - fclose(file); - return true; -} - -void gbInit() -{ - gbGenFilter(); - gbSgbInit(); - - gbMemory = (u8 *)malloc(65536); - memset(gbMemory,0, 65536); - - pix = (u8 *)calloc(1,4*257*226); - - gbLineBuffer = (u16 *)malloc(160 * sizeof(u16)); -} - -bool gbWriteBatteryFile(const char *file, bool extendedSave) -{ - if(gbBattery) { - int type = gbRom[0x147]; - - switch(type) { - case 0x03: - gbWriteSaveMBC1(file); - break; - case 0x06: - gbWriteSaveMBC2(file); - break; - case 0x0f: - case 0x10: - case 0x13: - gbWriteSaveMBC3(file, extendedSave); - break; - case 0x1b: - case 0x1e: - gbWriteSaveMBC5(file); - break; - case 0x22: - gbWriteSaveMBC7(file); - break; - case 0xff: - gbWriteSaveMBC1(file); - break; - } - } - return true; -} - -bool gbWriteBatteryFile(const char *file) -{ - gbWriteBatteryFile(file, true); - return true; -} - -bool gbReadBatteryFile(const char *file) -{ - bool res = false; - if(gbBattery) { - int type = gbRom[0x147]; - - switch(type) { - case 0x03: - res = gbReadSaveMBC1(file); - break; - case 0x06: - res = gbReadSaveMBC2(file); - break; - case 0x0f: - case 0x10: - case 0x13: - if(!gbReadSaveMBC3(file)) { - time(&gbDataMBC3.mapperLastTime); - struct tm *lt; - lt = localtime(&gbDataMBC3.mapperLastTime); - gbDataMBC3.mapperSeconds = lt->tm_sec; - gbDataMBC3.mapperMinutes = lt->tm_min; - gbDataMBC3.mapperHours = lt->tm_hour; - gbDataMBC3.mapperDays = lt->tm_yday & 255; - gbDataMBC3.mapperControl = (gbDataMBC3.mapperControl & 0xfe) | - (lt->tm_yday > 255 ? 1: 0); - res = false; - break; - } - res = true; - break; - case 0x1b: - case 0x1e: - res = gbReadSaveMBC5(file); - break; - case 0x22: - res = gbReadSaveMBC7(file); - case 0xff: - res = gbReadSaveMBC1(file); - break; - } - } - systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; - return res; -} - -bool gbReadGSASnapshot(const char *fileName) -{ - FILE *file = fopen(fileName, "rb"); - - if(!file) { - systemMessage(MSG_CANNOT_OPEN_FILE, N_("Cannot open file %s"), fileName); - return false; - } - - // long size = ftell(file); - fseek(file, 0x4, SEEK_SET); - char buffer[16]; - char buffer2[16]; - fread(buffer, 1, 15, file); - buffer[15] = 0; - memcpy(buffer2, &gbRom[0x134], 15); - buffer2[15] = 0; - if(memcmp(buffer, buffer2, 15)) { - systemMessage(MSG_CANNOT_IMPORT_SNAPSHOT_FOR, - N_("Cannot import snapshot for %s. Current game is %s"), - buffer, - buffer2); - fclose(file); - return false; - } - fseek(file, 0x13, SEEK_SET); - int read = 0; - int toRead = 0; - switch(gbRom[0x147]) { - case 0x03: - case 0x0f: - case 0x10: - case 0x13: - case 0x1b: - case 0x1e: - case 0xff: - read = fread(gbRam, 1, gbRamSize, file); - toRead = gbRamSize; - break; - case 0x06: - case 0x22: - read = fread(&gbMemory[0xa000],1,256,file); - toRead = 256; - break; - default: - systemMessage(MSG_UNSUPPORTED_SNAPSHOT_FILE, - N_("Unsupported snapshot file %s"), - fileName); - fclose(file); - return false; - } - fclose(file); - gbReset(); - return true; -} - -variable_desc gbSaveGameStruct[] = { - { &PC.W, sizeof(u16) }, - { &SP.W, sizeof(u16) }, - { &AF.W, sizeof(u16) }, - { &BC.W, sizeof(u16) }, - { &DE.W, sizeof(u16) }, - { &HL.W, sizeof(u16) }, - { &IFF, sizeof(u8) }, - { &GBLCD_MODE_0_CLOCK_TICKS, sizeof(int) }, - { &GBLCD_MODE_1_CLOCK_TICKS, sizeof(int) }, - { &GBLCD_MODE_2_CLOCK_TICKS, sizeof(int) }, - { &GBLCD_MODE_3_CLOCK_TICKS, sizeof(int) }, - { &GBDIV_CLOCK_TICKS, sizeof(int) }, - { &GBLY_INCREMENT_CLOCK_TICKS, sizeof(int) }, - { &GBTIMER_MODE_0_CLOCK_TICKS, sizeof(int) }, - { &GBTIMER_MODE_1_CLOCK_TICKS, sizeof(int) }, - { &GBTIMER_MODE_2_CLOCK_TICKS, sizeof(int) }, - { &GBTIMER_MODE_3_CLOCK_TICKS, sizeof(int) }, - { &GBSERIAL_CLOCK_TICKS, sizeof(int) }, - { &GBSYNCHRONIZE_CLOCK_TICKS, sizeof(int) }, - { &gbDivTicks, sizeof(int) }, - { &gbLcdMode, sizeof(int) }, - { &gbLcdTicks, sizeof(int) }, - { &gbLcdLYIncrementTicks, sizeof(int) }, - { &gbTimerTicks, sizeof(int) }, - { &gbTimerClockTicks, sizeof(int) }, - { &gbSerialTicks, sizeof(int) }, - { &gbSerialBits, sizeof(int) }, - { &gbInterrupt, sizeof(int) }, - { &gbInterruptWait, sizeof(int) }, - { &gbSynchronizeTicks, sizeof(int) }, - { &gbTimerOn, sizeof(int) }, - { &gbTimerMode, sizeof(int) }, - { &gbSerialOn, sizeof(int) }, - { &gbWindowLine, sizeof(int) }, - { &gbCgbMode, sizeof(int) }, - { &gbVramBank, sizeof(int) }, - { &gbWramBank, sizeof(int) }, - { &gbHdmaSource, sizeof(int) }, - { &gbHdmaDestination, sizeof(int) }, - { &gbHdmaBytes, sizeof(int) }, - { &gbHdmaOn, sizeof(int) }, - { &gbSpeed, sizeof(int) }, - { &gbSgbMode, sizeof(int) }, - { ®ister_DIV, sizeof(u8) }, - { ®ister_TIMA, sizeof(u8) }, - { ®ister_TMA, sizeof(u8) }, - { ®ister_TAC, sizeof(u8) }, - { ®ister_IF, sizeof(u8) }, - { ®ister_LCDC, sizeof(u8) }, - { ®ister_STAT, sizeof(u8) }, - { ®ister_SCY, sizeof(u8) }, - { ®ister_SCX, sizeof(u8) }, - { ®ister_LY, sizeof(u8) }, - { ®ister_LYC, sizeof(u8) }, - { ®ister_DMA, sizeof(u8) }, - { ®ister_WY, sizeof(u8) }, - { ®ister_WX, sizeof(u8) }, - { ®ister_VBK, sizeof(u8) }, - { ®ister_HDMA1, sizeof(u8) }, - { ®ister_HDMA2, sizeof(u8) }, - { ®ister_HDMA3, sizeof(u8) }, - { ®ister_HDMA4, sizeof(u8) }, - { ®ister_HDMA5, sizeof(u8) }, - { ®ister_SVBK, sizeof(u8) }, - { ®ister_IE , sizeof(u8) }, - { &gbBgp[0], sizeof(u8) }, - { &gbBgp[1], sizeof(u8) }, - { &gbBgp[2], sizeof(u8) }, - { &gbBgp[3], sizeof(u8) }, - { &gbObp0[0], sizeof(u8) }, - { &gbObp0[1], sizeof(u8) }, - { &gbObp0[2], sizeof(u8) }, - { &gbObp0[3], sizeof(u8) }, - { &gbObp1[0], sizeof(u8) }, - { &gbObp1[1], sizeof(u8) }, - { &gbObp1[2], sizeof(u8) }, - { &gbObp1[3], sizeof(u8) }, - { NULL, 0 } -}; - - -static bool gbWriteSaveState(gzFile gzFile) -{ - utilWriteInt(gzFile, GBSAVE_GAME_VERSION); - - utilGzWrite(gzFile, &gbRom[0x134], 15); - - utilWriteData(gzFile, gbSaveGameStruct); - - utilGzWrite(gzFile, &IFF, 2); - - if(gbSgbMode) { - gbSgbSaveGame(gzFile); - } - - utilGzWrite(gzFile, &gbDataMBC1, sizeof(gbDataMBC1)); - utilGzWrite(gzFile, &gbDataMBC2, sizeof(gbDataMBC2)); - utilGzWrite(gzFile, &gbDataMBC3, sizeof(gbDataMBC3)); - utilGzWrite(gzFile, &gbDataMBC5, sizeof(gbDataMBC5)); - utilGzWrite(gzFile, &gbDataHuC1, sizeof(gbDataHuC1)); - utilGzWrite(gzFile, &gbDataHuC3, sizeof(gbDataHuC3)); - - // not saved anymore - // gzwrite(gzFile, pix, 256*224*sizeof(u16)); - - utilGzWrite(gzFile, gbPalette, 128 * sizeof(u16)); - // todo: remove - utilGzWrite(gzFile, gbPalette, 128 * sizeof(u16)); - - for (int i = 0xFF10; i <= 0xFF3F; i++) gbMemory[i] = gbReadMemory(i); - - utilGzWrite(gzFile, &gbMemory[0x8000], 0x8000); - - if(gbRamSize && gbRam) { - utilGzWrite(gzFile, gbRam, gbRamSize); - } - - if(gbCgbMode) { - utilGzWrite(gzFile, gbVram, 0x4000); - utilGzWrite(gzFile, gbWram, 0x8000); - } - - gbSoundSaveGame(gzFile); - - gbCheatsSaveGame(gzFile); - - return true; -} - -bool gbWriteMemSaveState(char *memory, int available) -{ - gzFile gzFile = utilMemGzOpen(memory, available, "w"); - - if(gzFile == NULL) { - return false; - } - - bool res = gbWriteSaveState(gzFile); - - long pos = utilGzMemTell(gzFile)+8; - - if(pos >= (available)) - res = false; - - utilGzClose(gzFile); - - return res; -} - -bool gbWriteSaveState(const char *name) -{ - gzFile gzFile = utilGzOpen(name,"wb"); - - if(gzFile == NULL) - return false; - - bool res = gbWriteSaveState(gzFile); - - utilGzClose(gzFile); - return res; -} - -static bool gbReadSaveState(gzFile gzFile) -{ - int version = utilReadInt(gzFile); - - if(version > GBSAVE_GAME_VERSION || version < 0) { - systemMessage(MSG_UNSUPPORTED_VB_SGM, - N_("Unsupported VisualBoy save game version %d"), version); - return false; - } - - u8 romname[20]; - - utilGzRead(gzFile, romname, 15); - - if(memcmp(&gbRom[0x134], romname, 15) != 0) { - systemMessage(MSG_CANNOT_LOAD_SGM_FOR, - N_("Cannot load save game for %s. Playing %s"), - romname, &gbRom[0x134]); - return false; - } - - utilReadData(gzFile, gbSaveGameStruct); - - if(version >= GBSAVE_GAME_VERSION_7) { - utilGzRead(gzFile, &IFF, 2); - } - - if(gbSgbMode) { - gbSgbReadGame(gzFile, version); - } else { - gbSgbMask = 0; // loading a game at the wrong time causes no display - } - - utilGzRead(gzFile, &gbDataMBC1, sizeof(gbDataMBC1)); - utilGzRead(gzFile, &gbDataMBC2, sizeof(gbDataMBC2)); - if(version < GBSAVE_GAME_VERSION_4) - // prior to version 4, there was no adjustment for the time the game - // was last played, so we have less to read. This needs update if the - // structure changes again. - utilGzRead(gzFile, &gbDataMBC3, sizeof(gbDataMBC3)-sizeof(time_t)); - else - utilGzRead(gzFile, &gbDataMBC3, sizeof(gbDataMBC3)); - utilGzRead(gzFile, &gbDataMBC5, sizeof(gbDataMBC5)); - utilGzRead(gzFile, &gbDataHuC1, sizeof(gbDataHuC1)); - utilGzRead(gzFile, &gbDataHuC3, sizeof(gbDataHuC3)); - - if(version < GBSAVE_GAME_VERSION_5) { - utilGzRead(gzFile, pix, 256*224*sizeof(u16)); - } - memset(pix, 0, 257*226*sizeof(u32)); - - if(version < GBSAVE_GAME_VERSION_6) { - utilGzRead(gzFile, gbPalette, 64 * sizeof(u16)); - } else - utilGzRead(gzFile, gbPalette, 128 * sizeof(u16)); - - // todo: remove - utilGzRead(gzFile, gbPalette, 128 * sizeof(u16)); - - if(version < GBSAVE_GAME_VERSION_10) { - if(!gbCgbMode && !gbSgbMode) { - for(int i = 0; i < 8; i++) - gbPalette[i] = systemGbPalette[gbPaletteOption*8+i]; - } - } - - utilGzRead(gzFile, &gbMemory[0x8000], 0x8000); - - for (int i = 0xFF10; i <= 0xFF3F; i++) gbSoundEvent(i, gbMemory[i]); - - if(gbRamSize && gbRam) { - utilGzRead(gzFile, gbRam, gbRamSize); - } - - gbMemoryMap[0x00] = &gbRom[0x0000]; - gbMemoryMap[0x01] = &gbRom[0x1000]; - gbMemoryMap[0x02] = &gbRom[0x2000]; - gbMemoryMap[0x03] = &gbRom[0x3000]; - gbMemoryMap[0x04] = &gbRom[0x4000]; - gbMemoryMap[0x05] = &gbRom[0x5000]; - gbMemoryMap[0x06] = &gbRom[0x6000]; - gbMemoryMap[0x07] = &gbRom[0x7000]; - gbMemoryMap[0x08] = &gbMemory[0x8000]; - gbMemoryMap[0x09] = &gbMemory[0x9000]; - gbMemoryMap[0x0a] = &gbMemory[0xa000]; - gbMemoryMap[0x0b] = &gbMemory[0xb000]; - gbMemoryMap[0x0c] = &gbMemory[0xc000]; - gbMemoryMap[0x0d] = &gbMemory[0xd000]; - gbMemoryMap[0x0e] = &gbMemory[0xe000]; - gbMemoryMap[0x0f] = &gbMemory[0xf000]; - - int type = gbRom[0x147]; - - switch(type) { - case 0x00: - case 0x01: - case 0x02: - case 0x03: - // MBC 1 - memoryUpdateMapMBC1(); - break; - case 0x05: - case 0x06: - // MBC2 - memoryUpdateMapMBC2(); - break; - case 0x0f: - case 0x10: - case 0x11: - case 0x12: - case 0x13: - // MBC 3 - memoryUpdateMapMBC3(); - break; - case 0x19: - case 0x1a: - case 0x1b: - // MBC5 - memoryUpdateMapMBC5(); - break; - case 0x1c: - case 0x1d: - case 0x1e: - // MBC 5 Rumble - memoryUpdateMapMBC5(); - break; - case 0x22: - // MBC 7 - memoryUpdateMapMBC7(); - break; - case 0xfe: - // HuC3 - memoryUpdateMapHuC3(); - break; - case 0xff: - // HuC1 - memoryUpdateMapHuC1(); - break; - } - - if(gbCgbMode) { - utilGzRead(gzFile, gbVram, 0x4000); - utilGzRead(gzFile, gbWram, 0x8000); - - int value = register_SVBK; - if(value == 0) - value = 1; - - gbMemoryMap[0x08] = &gbVram[register_VBK * 0x2000]; - gbMemoryMap[0x09] = &gbVram[register_VBK * 0x2000 + 0x1000]; - gbMemoryMap[0x0d] = &gbWram[value * 0x1000]; - } - - gbSoundReadGame(version, gzFile); - - if(gbBorderOn) { - gbSgbRenderBorder(); - } - - systemDrawScreen(); - - if(version > GBSAVE_GAME_VERSION_1) - gbCheatsReadGame(gzFile, version); - - systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; - - return true; -} - -bool gbReadMemSaveState(char *memory, int available) -{ - gzFile gzFile = utilMemGzOpen(memory, available, "r"); - - bool res = gbReadSaveState(gzFile); - - utilGzClose(gzFile); - - return res; -} - -bool gbReadSaveState(const char *name) -{ - gzFile gzFile = utilGzOpen(name,"rb"); - - if(gzFile == NULL) { - return false; - } - - bool res = gbReadSaveState(gzFile); - - utilGzClose(gzFile); - - return res; -} - -bool gbWritePNGFile(const char *fileName) -{ - if(gbBorderOn) - return utilWritePNGFile(fileName, 256, 224, pix); - return utilWritePNGFile(fileName, 160, 144, pix); -} - -bool gbWriteBMPFile(const char *fileName) -{ - if(gbBorderOn) - return utilWriteBMPFile(fileName, 256, 224, pix); - return utilWriteBMPFile(fileName, 160, 144, pix); -} - -void gbCleanUp() -{ - if(gbRam != NULL) { - free(gbRam); - gbRam = NULL; - } - - if(gbRom != NULL) { - free(gbRom); - gbRom = NULL; - } - - if(gbMemory != NULL) { - free(gbMemory); - gbMemory = NULL; - } - - if(gbLineBuffer != NULL) { - free(gbLineBuffer); - gbLineBuffer = NULL; - } - - if(pix != NULL) { - free(pix); - pix = NULL; - } - - gbSgbShutdown(); - - if(gbVram != NULL) { - free(gbVram); - gbVram = NULL; - } - - if(gbWram != NULL) { - free(gbWram); - gbWram = NULL; - } - - systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; -} - -bool gbLoadRom(const char *szFile) -{ - int size = 0; - - if(gbRom != NULL) { - gbCleanUp(); - } - - systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; - - gbRom = utilLoad(szFile, - utilIsGBImage, - NULL, - size); - if(!gbRom) - return false; - - gbRomSize = size; - - return gbUpdateSizes(); -} - -bool gbUpdateSizes() -{ - if(gbRom[0x148] > 8) { - systemMessage(MSG_UNSUPPORTED_ROM_SIZE, - N_("Unsupported rom size %02x"), gbRom[0x148]); - return false; - } - - if(gbRomSize < gbRomSizes[gbRom[0x148]]) { - gbRom = (u8 *)realloc(gbRom, gbRomSizes[gbRom[0x148]]); - } - gbRomSize = gbRomSizes[gbRom[0x148]]; - gbRomSizeMask = gbRomSizesMasks[gbRom[0x148]]; - - if(gbRom[0x149] > 5) { - systemMessage(MSG_UNSUPPORTED_RAM_SIZE, - N_("Unsupported ram size %02x"), gbRom[0x149]); - return false; - } - - gbRamSize = gbRamSizes[gbRom[0x149]]; - gbRamSizeMask = gbRamSizesMasks[gbRom[0x149]]; - - if(gbRamSize) { - gbRam = (u8 *)malloc(gbRamSize); - memset(gbRam, 0xFF, gbRamSize); - } - - int type = gbRom[0x147]; - - mapperReadRAM = NULL; - - switch(type) { - case 0x00: - case 0x01: - case 0x02: - case 0x03: - // MBC 1 - mapper = mapperMBC1ROM; - mapperRAM = mapperMBC1RAM; - break; - case 0x05: - case 0x06: - // MBC2 - mapper = mapperMBC2ROM; - mapperRAM = mapperMBC2RAM; - gbRamSize = 0x200; - gbRamSizeMask = 0x1ff; - break; - case 0x0f: - case 0x10: - case 0x11: - case 0x12: - case 0x13: - // MBC 3 - mapper = mapperMBC3ROM; - mapperRAM = mapperMBC3RAM; - mapperReadRAM = mapperMBC3ReadRAM; - break; - case 0x19: - case 0x1a: - case 0x1b: - // MBC5 - mapper = mapperMBC5ROM; - mapperRAM = mapperMBC5RAM; - break; - case 0x1c: - case 0x1d: - case 0x1e: - // MBC 5 Rumble - mapper = mapperMBC5ROM; - mapperRAM = mapperMBC5RAM; - break; - case 0x22: - // MBC 7 - mapper = mapperMBC7ROM; - mapperRAM = mapperMBC7RAM; - mapperReadRAM = mapperMBC7ReadRAM; - break; - case 0xfe: - // HuC3 - mapper = mapperHuC3ROM; - mapperRAM = mapperHuC3RAM; - mapperReadRAM = mapperHuC3ReadRAM; - break; - case 0xff: - // HuC1 - mapper = mapperHuC1ROM; - mapperRAM = mapperHuC1RAM; - break; - default: - systemMessage(MSG_UNKNOWN_CARTRIDGE_TYPE, - N_("Unknown cartridge type %02x"), type); - return false; - } - - switch(type) { - case 0x03: - case 0x06: - case 0x0f: - case 0x10: - case 0x13: - case 0x1b: - case 0x1d: - case 0x1e: - case 0x22: - case 0xff: - gbBattery = 1; - break; - } - - if(gbRom[0x146] == 0x03) { - if(gbEmulatorType == 0 || - gbEmulatorType == 2 || - gbEmulatorType == 5) - gbSgbMode = 1; - else - gbSgbMode = 0; - } else - gbSgbMode = 0; - - if(gbRom[0x143] & 0x80) { - if(gbEmulatorType == 0 || - gbEmulatorType == 1 || - gbEmulatorType == 4 || - gbEmulatorType == 5) { - gbCgbMode = 1; - gbVram = (u8 *)malloc(0x4000); - gbWram = (u8 *)malloc(0x8000); - memset(gbVram,0,0x4000); - memset(gbWram,0,0x8000); - memset(gbPalette,0, 2*128); - } else { - gbCgbMode = 0; - } - } else - gbCgbMode = 0; - - gbInit(); - gbReset(); - - switch(type) { - case 0x1c: - case 0x1d: - case 0x1e: - gbDataMBC5.isRumbleCartridge = 1; - } - - return true; -} - -void gbEmulate(int ticksToStop) -{ - gbRegister tempRegister; - u8 tempValue; - s8 offset; - - int clockTicks = 0; - gbDmaTicks = 0; - - register int opcode = 0; - - while(1) { -#ifndef FINAL_VERSION - if(systemDebug) { - if(!(IFF & 0x80)) { - if(systemDebug > 1) { - sprintf(gbBuffer,"PC=%04x AF=%04x BC=%04x DE=%04x HL=%04x SP=%04x I=%04x\n", - PC.W, AF.W, BC.W, DE.W,HL.W,SP.W,IFF); - } else { - sprintf(gbBuffer,"PC=%04x I=%02x\n", PC.W, IFF); - } - log(gbBuffer); - } - } -#endif - if(IFF & 0x80) { - if(register_LCDC & 0x80) { - clockTicks = gbLcdTicks; - } else - clockTicks = 100; - - if(gbLcdMode == 1 && (gbLcdLYIncrementTicks < clockTicks)) - clockTicks = gbLcdLYIncrementTicks; - - if(gbSerialOn && (gbSerialTicks < clockTicks)) - clockTicks = gbSerialTicks; - - if(gbTimerOn && (gbTimerTicks < clockTicks)) - clockTicks = gbTimerTicks; - - if(soundTicks && (soundTicks < clockTicks)) - clockTicks = soundTicks; - } else { - opcode = gbReadOpcode(PC.W++); - - if(IFF & 0x100) { - IFF &= 0xff; - PC.W--; - } - - clockTicks = gbCycles[opcode]; - - switch(opcode) { - case 0xCB: - // extended opcode - opcode = gbReadOpcode(PC.W++); - clockTicks = gbCyclesCB[opcode]; - switch(opcode) { -#include "gbCodesCB.h" - } - break; -#include "gbCodes.h" - } - } - - if(!emulating) - return; - - if(gbDmaTicks) { - clockTicks += gbDmaTicks; - gbDmaTicks = 0; - } - - if(gbSgbMode) { - if(gbSgbPacketTimeout) { - gbSgbPacketTimeout -= clockTicks; - - if(gbSgbPacketTimeout <= 0) - gbSgbResetPacketState(); - } - } - - ticksToStop -= clockTicks; - - // DIV register emulation - gbDivTicks -= clockTicks; - while(gbDivTicks <= 0) { - register_DIV++; - gbDivTicks += GBDIV_CLOCK_TICKS; - } - - if(register_LCDC & 0x80) { - // LCD stuff - gbLcdTicks -= clockTicks; - if(gbLcdMode == 1) { - // during V-BLANK,we need to increment LY at the same rate! - gbLcdLYIncrementTicks -= clockTicks; - while(gbLcdLYIncrementTicks <= 0) { - gbLcdLYIncrementTicks += GBLY_INCREMENT_CLOCK_TICKS; - - if(register_LY < 153) { - register_LY++; - - gbCompareLYToLYC(); - - if(register_LY >= 153) - gbLcdLYIncrementTicks = 6; - } else { - register_LY = 0x00; - // reset the window line - gbWindowLine = -1; - gbLcdLYIncrementTicks = GBLY_INCREMENT_CLOCK_TICKS * 2; - gbCompareLYToLYC(); - } - } - } - - // our counter is off, see what we need to do - while(gbLcdTicks <= 0) { - int framesToSkip = systemFrameSkip; - if(speedup) - framesToSkip = 9; // try 6 FPS during speedup - switch(gbLcdMode) { - case 0: - // H-Blank - register_LY++; - - gbCompareLYToLYC(); - - // check if we reached the V-Blank period - if(register_LY == 144) { - // Yes, V-Blank - // set the LY increment counter - gbLcdLYIncrementTicks = gbLcdTicks + GBLY_INCREMENT_CLOCK_TICKS; - gbLcdTicks += GBLCD_MODE_1_CLOCK_TICKS; - gbLcdMode = 1; - if(register_LCDC & 0x80) { - gbInterrupt |= 1; // V-Blank interrupt - gbInterruptWait = 6; - if(register_STAT & 0x10) - gbInterrupt |= 2; - } - - gbFrameCount++; - - systemFrame(); - - if((gbFrameCount % 10) == 0) - system10Frames(60); - - if(gbFrameCount >= 60) { - u32 currentTime = systemGetClock(); - if(currentTime != gbLastTime) - systemShowSpeed(100000/(currentTime - gbLastTime)); - else - systemShowSpeed(0); - gbLastTime = currentTime; - gbFrameCount = 0; - } - - if(systemReadJoypads()) { - // read joystick - if(gbSgbMode && gbSgbMultiplayer) { - if(gbSgbFourPlayers) { - gbJoymask[0] = systemReadJoypad(0); - gbJoymask[1] = systemReadJoypad(1); - gbJoymask[2] = systemReadJoypad(2); - gbJoymask[3] = systemReadJoypad(3); - } else { - gbJoymask[0] = systemReadJoypad(0); - gbJoymask[1] = systemReadJoypad(1); - } - } else { - gbJoymask[0] = systemReadJoypad(-1); - } - } - int newmask = gbJoymask[0] & 255; - - if(gbRom[0x147] == 0x22) { - systemUpdateMotionSensor(); - } - - if(newmask) { - gbInterrupt |= 16; - } - - newmask = (gbJoymask[0] >> 10); - - speedup = (newmask & 1) ? true : false; - gbCapture = (newmask & 2) ? true : false; - - if(gbCapture && !gbCapturePrevious) { - gbCaptureNumber++; - systemScreenCapture(gbCaptureNumber); - } - gbCapturePrevious = gbCapture; - - if(gbFrameSkipCount >= framesToSkip) { - systemDrawScreen(); - gbFrameSkipCount = 0; - } else - gbFrameSkipCount++; - } else { - // go the the OAM being accessed mode - gbLcdTicks += GBLCD_MODE_2_CLOCK_TICKS; - gbLcdMode = 2; - - // only one LCD interrupt per line. may need to generalize... - if(!(register_STAT & 0x40) || - (register_LY != register_LYC)) { - if((register_STAT & 0x28) == 0x20) - gbInterrupt |= 2; - } - } - break; - case 1: - // V-Blank - // next mode is OAM being accessed mode - gbLcdTicks += GBLCD_MODE_2_CLOCK_TICKS; - gbLcdMode = 2; - if(!(register_STAT & 0x40) || - (register_LY != register_LYC)) { - if((register_STAT & 0x28) == 0x20) - gbInterrupt |= 2; - } - break; - case 2: - // OAM being accessed mode - - // next mode is OAM and VRAM in use - gbLcdTicks += GBLCD_MODE_3_CLOCK_TICKS; - gbLcdMode = 3; - break; - case 3: - // OAM and VRAM in use - // next mode is H-Blank - if(register_LY < 144) { - if(!gbSgbMask) { - if(gbFrameSkipCount >= framesToSkip) { - gbRenderLine(); - gbDrawSprites(); - - switch(systemColorDepth) { - case 16: - { - u16 * dest = (u16 *)pix + - (gbBorderLineSkip+2) * (register_LY + gbBorderRowSkip+1) - + gbBorderColumnSkip; - for(int x = 0; x < 160; ) { - *dest++ = systemColorMap16[gbLineMix[x++]]; - *dest++ = systemColorMap16[gbLineMix[x++]]; - *dest++ = systemColorMap16[gbLineMix[x++]]; - *dest++ = systemColorMap16[gbLineMix[x++]]; - - *dest++ = systemColorMap16[gbLineMix[x++]]; - *dest++ = systemColorMap16[gbLineMix[x++]]; - *dest++ = systemColorMap16[gbLineMix[x++]]; - *dest++ = systemColorMap16[gbLineMix[x++]]; - - *dest++ = systemColorMap16[gbLineMix[x++]]; - *dest++ = systemColorMap16[gbLineMix[x++]]; - *dest++ = systemColorMap16[gbLineMix[x++]]; - *dest++ = systemColorMap16[gbLineMix[x++]]; - - *dest++ = systemColorMap16[gbLineMix[x++]]; - *dest++ = systemColorMap16[gbLineMix[x++]]; - *dest++ = systemColorMap16[gbLineMix[x++]]; - *dest++ = systemColorMap16[gbLineMix[x++]]; - } - if(gbBorderOn) - dest += gbBorderColumnSkip; - *dest++ = 0; // for filters that read one pixel more - } - break; - case 24: - { - u8 *dest = (u8 *)pix + - 3*(gbBorderLineSkip * (register_LY + gbBorderRowSkip) + - gbBorderColumnSkip); - for(int x = 0; x < 160;) { - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; - dest+= 3; - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; - dest+= 3; - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; - dest+= 3; - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; - dest+= 3; - - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; - dest+= 3; - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; - dest+= 3; - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; - dest+= 3; - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; - dest+= 3; - - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; - dest+= 3; - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; - dest+= 3; - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; - dest+= 3; - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; - dest+= 3; - - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; - dest+= 3; - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; - dest+= 3; - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; - dest+= 3; - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; - dest+= 3; - } - } - break; - case 32: - { - u32 * dest = (u32 *)pix + - (gbBorderLineSkip+1) * (register_LY + gbBorderRowSkip+1) - + gbBorderColumnSkip; - for(int x = 0; x < 160;) { - *dest++ = systemColorMap32[gbLineMix[x++]]; - *dest++ = systemColorMap32[gbLineMix[x++]]; - *dest++ = systemColorMap32[gbLineMix[x++]]; - *dest++ = systemColorMap32[gbLineMix[x++]]; - - *dest++ = systemColorMap32[gbLineMix[x++]]; - *dest++ = systemColorMap32[gbLineMix[x++]]; - *dest++ = systemColorMap32[gbLineMix[x++]]; - *dest++ = systemColorMap32[gbLineMix[x++]]; - - *dest++ = systemColorMap32[gbLineMix[x++]]; - *dest++ = systemColorMap32[gbLineMix[x++]]; - *dest++ = systemColorMap32[gbLineMix[x++]]; - *dest++ = systemColorMap32[gbLineMix[x++]]; - - *dest++ = systemColorMap32[gbLineMix[x++]]; - *dest++ = systemColorMap32[gbLineMix[x++]]; - *dest++ = systemColorMap32[gbLineMix[x++]]; - *dest++ = systemColorMap32[gbLineMix[x++]]; - } - } - break; - } - } - } - } - gbLcdTicks += GBLCD_MODE_0_CLOCK_TICKS; - gbLcdMode = 0; - // only one LCD interrupt per line. may need to generalize... - if(!(register_STAT & 0x40) || - (register_LY != register_LYC)) { - if(register_STAT & 0x08) - gbInterrupt |= 2; - } - if(gbHdmaOn) { - gbDoHdma(); - } - break; - } - // mark the correct lcd mode on STAT register - register_STAT = (register_STAT & 0xfc) | gbLcdMode; - } - } - - // serial emulation - if(gbSerialOn) { -#ifdef LINK_EMULATION - if(linkConnected) { - gbSerialTicks -= clockTicks; - - while(gbSerialTicks <= 0) { - // increment number of shifted bits - gbSerialBits++; - linkProc(); - if(gbSerialOn && (gbMemory[0xff02] & 1)) { - if(gbSerialBits == 8) { - gbSerialBits = 0; - gbMemory[0xff01] = 0xff; - gbMemory[0xff02] &= 0x7f; - gbSerialOn = 0; - gbInterrupt |= 8; - gbSerialTicks = 0; - } - } - gbSerialTicks += GBSERIAL_CLOCK_TICKS; - } - } else { -#endif - if(gbMemory[0xff02] & 1) { - gbSerialTicks -= clockTicks; - - // overflow - while(gbSerialTicks <= 0) { - // shift serial byte to right and put a 1 bit in its place - // gbMemory[0xff01] = 0x80 | (gbMemory[0xff01]>>1); - // increment number of shifted bits - gbSerialBits++; - if(gbSerialBits == 8) { - // end of transmission - if(gbSerialFunction) // external device - gbMemory[0xff01] = gbSerialFunction(gbMemory[0xff01]); - else - gbMemory[0xff01] = 0xff; - gbSerialTicks = 0; - gbMemory[0xff02] &= 0x7f; - gbSerialOn = 0; - gbInterrupt |= 8; - gbSerialBits = 0; - } else - gbSerialTicks += GBSERIAL_CLOCK_TICKS; - } - } -#ifdef LINK_EMULATION - } -#endif - } - - // timer emulation - if(gbTimerOn) { - gbTimerTicks -= clockTicks; - - while(gbTimerTicks <= 0) { - register_TIMA++; - - if(register_TIMA == 0) { - // timer overflow! - - // reload timer modulo - register_TIMA = register_TMA; - - // flag interrupt - gbInterrupt |= 4; - } - - gbTimerTicks += gbTimerClockTicks; - } - } - - /* - if(soundOffFlag) { - if(synchronize && !speedup) { - synchronizeTicks -= clockTicks; - - while(synchronizeTicks < 0) { - synchronizeTicks += SYNCHRONIZE_CLOCK_TICKS; - - DWORD now = timeGetTime(); - gbElapsedTime += (now - timeNow); - - if(gbElapsedTime < 50) { - DWORD diff = 50 - gbElapsedTime; - Sleep(diff); - timeNow = timeGetTime(); - elapsedTime = timeNow - now - diff; - if((int)elapsedTime < 0) - elapsedTime = 0; - } else { - timeNow = timeGetTime(); - elapsedTime = 0; - } - } - } - } - */ - - soundTicks -= clockTicks; - - while(soundTicks < 0) { - soundTicks += SOUND_CLOCK_TICKS; - - gbSoundTick(); - } - - register_IF = gbInterrupt; - - if(IFF & 0x20) { - IFF &= 0xdf; - IFF |= 0x01; - gbInterruptWait = 0; - } else if(gbInterrupt) { - if(gbInterruptWait == 0) { - // gbInterruptWait = 0; - - if(IFF & 0x01) { - if((gbInterrupt & 1) && (register_IE & 1)) { - gbVblank_interrupt(); - continue; - } - - if((gbInterrupt & 2) && (register_IE & 2)) { - gbLcd_interrupt(); - continue; - } - - if((gbInterrupt & 4) && (register_IE & 4)) { - gbTimer_interrupt(); - continue; - } - - if((gbInterrupt & 8) && (register_IE & 8)) { - gbSerial_interrupt(); - continue; - } - - if((gbInterrupt & 16) && (register_IE & 16)) { - gbJoypad_interrupt(); - continue; - } - } - } else { - gbInterruptWait -= clockTicks; - if(gbInterruptWait < 0) - gbInterruptWait = 0; - } - } - - if(ticksToStop <= 0) { - if(!(register_LCDC & 0x80)) { - if(systemReadJoypads()) { - // read joystick - if(gbSgbMode && gbSgbMultiplayer) { - if(gbSgbFourPlayers) { - gbJoymask[0] = systemReadJoypad(0); - gbJoymask[1] = systemReadJoypad(1); - gbJoymask[2] = systemReadJoypad(2); - gbJoymask[3] = systemReadJoypad(3); - } else { - gbJoymask[0] = systemReadJoypad(0); - gbJoymask[1] = systemReadJoypad(1); - } - } else { - gbJoymask[0] = systemReadJoypad(-1); - } - } - } - return; - } - } -} - -struct EmulatedSystem GBSystem = { - // emuMain - gbEmulate, - // emuReset - gbReset, - // emuCleanUp - gbCleanUp, - // emuReadBattery - gbReadBatteryFile, - // emuWriteBattery - gbWriteBatteryFile, - // emuReadState - gbReadSaveState, - // emuWriteState - gbWriteSaveState, - // emuReadMemState - gbReadMemSaveState, - // emuWriteMemState - gbWriteMemSaveState, - // emuWritePNG - gbWritePNGFile, - // emuWriteBMP - gbWriteBMPFile, - // emuUpdateCPSR - NULL, - // emuHasDebugger - false, - // emuCount -#ifdef FINAL_VERSION - 70000/4, -#else - 1000, -#endif -}; +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2005-2006 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#include +#include +#include +#include + +#include "../System.h" +#include "../NLS.h" +#include "GB.h" +#include "gbCheats.h" +#include "gbGlobals.h" +#include "gbMemory.h" +#include "gbSGB.h" +#include "gbSound.h" +#include "../unzip.h" +#include "../Util.h" + +#ifdef __GNUC__ +#define _stricmp strcasecmp +#endif + +extern u8 *pix; +extern bool speedup; + +bool gbUpdateSizes(); +bool inBios = false; + +// debugging +bool memorydebug = false; +char gbBuffer[2048]; + +extern u16 gbLineMix[160]; + +// mappers +void (*mapper)(u16,u8) = NULL; +void (*mapperRAM)(u16,u8) = NULL; +u8 (*mapperReadRAM)(u16) = NULL; +void (*mapperUpdateClock)() = NULL; + +// registers +gbRegister PC; +gbRegister SP; +gbRegister AF; +gbRegister BC; +gbRegister DE; +gbRegister HL; +u16 IFF = 0; +// 0xff04 +u8 register_DIV = 0; +// 0xff05 +u8 register_TIMA = 0; +// 0xff06 +u8 register_TMA = 0; +// 0xff07 +u8 register_TAC = 0; +// 0xff0f +u8 register_IF = 0; +// 0xff40 +u8 register_LCDC = 0; +// 0xff41 +u8 register_STAT = 0; +// 0xff42 +u8 register_SCY = 0; +// 0xff43 +u8 register_SCX = 0; +// 0xff44 +u8 register_LY = 0; +// 0xff45 +u8 register_LYC = 0; +// 0xff46 +u8 register_DMA = 0; +// 0xff4a +u8 register_WY = 0; +// 0xff4b +u8 register_WX = 0; +// 0xff4f +u8 register_VBK = 0; +// 0xff51 +u8 register_HDMA1 = 0; +// 0xff52 +u8 register_HDMA2 = 0; +// 0xff53 +u8 register_HDMA3 = 0; +// 0xff54 +u8 register_HDMA4 = 0; +// 0xff55 +u8 register_HDMA5 = 0; +// 0xff70 +u8 register_SVBK = 0; +// 0xffff +u8 register_IE = 0; + +// ticks definition +int GBDIV_CLOCK_TICKS = 64; +int GBLCD_MODE_0_CLOCK_TICKS = 51; +int GBLCD_MODE_1_CLOCK_TICKS = 1140; +int GBLCD_MODE_2_CLOCK_TICKS = 20; +int GBLCD_MODE_3_CLOCK_TICKS = 43; +int GBLY_INCREMENT_CLOCK_TICKS = 114; +int GBTIMER_MODE_0_CLOCK_TICKS = 256; +int GBTIMER_MODE_1_CLOCK_TICKS = 4; +int GBTIMER_MODE_2_CLOCK_TICKS = 16; +int GBTIMER_MODE_3_CLOCK_TICKS = 64; +int GBSERIAL_CLOCK_TICKS = 128; +int GBSYNCHRONIZE_CLOCK_TICKS = 52920; + +// state variables + +// general +int clockTicks = 0; +bool gbSystemMessage = false; +int gbGBCColorType = 0; +int gbHardware = 0; +int gbRomType = 0; +int gbRemainingClockTicks = 0; +int gbOldClockTicks = 0; +int gbIntBreak = 0; +int gbInterruptLaunched = 0; +u8 gbCheatingDevice = 0; // 1 = GS, 2 = GG +// breakpoint +bool breakpoint = false; +// interrupt +int gbInt48Signal = 0; +int gbInterruptWait = 0; +// serial +int gbSerialOn = 0; +int gbSerialTicks = 0; +int gbSerialBits = 0; +// timer +bool gbTimerOn = false; +int gbTimerTicks = GBTIMER_MODE_0_CLOCK_TICKS; +int gbTimerClockTicks = GBTIMER_MODE_0_CLOCK_TICKS; +int gbTimerMode = 0; +bool gbIncreased = false; +// The internal timer is always active, and it is +// not reset by writing to register_TIMA/TMA, but by +// writing to register_DIV... +int gbInternalTimer = 0x55; +const u8 gbTimerMask [4] = {0xff, 0x3, 0xf, 0x3f}; +const u8 gbTimerBug [8] = {0x80, 0x80, 0x02, 0x02, 0x0, 0xff, 0x0, 0xff}; +bool gbTimerModeChange = false; +bool gbTimerOnChange = false; +// lcd +bool gbScreenOn = true; +int gbLcdMode = 2; +int gbLcdModeDelayed = 2; +int gbLcdTicks = GBLCD_MODE_2_CLOCK_TICKS-1; +int gbLcdTicksDelayed = GBLCD_MODE_2_CLOCK_TICKS; +int gbLcdLYIncrementTicks = 114; +int gbLcdLYIncrementTicksDelayed = 115; +int gbScreenTicks = 0; +u8 gbSCYLine[300]; +u8 gbSCXLine[300]; +u8 gbBgpLine[300]; +u8 gbObp0Line [300]; +u8 gbObp1Line [300]; +u8 gbSpritesTicks [300]; +u8 oldRegister_WY; +bool gbLYChangeHappened = false; +bool gbLCDChangeHappened = false; +int gbLine99Ticks = 1; +int gbRegisterLYLCDCOffOn = 0; +int inUseRegister_WY = 0; + +// Used to keep track of the line that ellapse +// when screen is off +int gbWhiteScreen = 0; +bool gbBlackScreen = false; +int register_LCDCBusy = 0; + +// div +int gbDivTicks = GBDIV_CLOCK_TICKS; +// cgb +int gbVramBank = 0; +int gbWramBank = 1; +//sgb +bool gbSgbResetFlag = false; +// gbHdmaDestination is 0x99d0 on startup (tested on HW) +// but I'm not sure what gbHdmaSource is... +int gbHdmaSource = 0x99d0; +int gbHdmaDestination = 0x99d0; +int gbHdmaBytes = 0x0000; +int gbHdmaOn = 0; +int gbSpeed = 0; +// frame counting +int gbFrameCount = 0; +int gbFrameSkip = 0; +int gbFrameSkipCount = 0; +// timing +u32 gbLastTime = 0; +u32 gbElapsedTime = 0; +u32 gbTimeNow = 0; +int gbSynchronizeTicks = GBSYNCHRONIZE_CLOCK_TICKS; +// emulator features +int gbBattery = 0; +bool gbBatteryError = false; +int gbCaptureNumber = 0; +bool gbCapture = false; +bool gbCapturePrevious = false; +int gbJoymask[4] = { 0, 0, 0, 0 }; + +int gbRomSizes[] = { 0x00008000, // 32K + 0x00010000, // 64K + 0x00020000, // 128K + 0x00040000, // 256K + 0x00080000, // 512K + 0x00100000, // 1024K + 0x00200000, // 2048K + 0x00400000, // 4096K + 0x00800000 // 8192K +}; +int gbRomSizesMasks[] = { 0x00007fff, + 0x0000ffff, + 0x0001ffff, + 0x0003ffff, + 0x0007ffff, + 0x000fffff, + 0x001fffff, + 0x003fffff, + 0x007fffff +}; + +int gbRamSizes[6] = { 0x00000000, // 0K + 0x00002000, // 2K // Changed to 2000 to avoid problems with gbMemoryMap... + 0x00002000, // 8K + 0x00008000, // 32K + 0x00020000, // 128K + 0x00010000 // 64K +}; + +int gbRamSizesMasks[6] = { 0x00000000, + 0x000007ff, + 0x00001fff, + 0x00007fff, + 0x0001ffff, + 0x0000ffff +}; + +int gbCycles[] = { +// 0 1 2 3 4 5 6 7 8 9 a b c d e f + 1, 3, 2, 2, 1, 1, 2, 1, 5, 2, 2, 2, 1, 1, 2, 1, // 0 + 1, 3, 2, 2, 1, 1, 2, 1, 3, 2, 2, 2, 1, 1, 2, 1, // 1 + 2, 3, 2, 2, 1, 1, 2, 1, 2, 2, 2, 2, 1, 1, 2, 1, // 2 + 2, 3, 2, 2, 3, 3, 3, 1, 2, 2, 2, 2, 1, 1, 2, 1, // 3 + 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // 4 + 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // 5 + 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // 6 + 2, 2, 2, 2, 2, 2, 1, 2, 1, 1, 1, 1, 1, 1, 2, 1, // 7 + 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // 8 + 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // 9 + 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // a + 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // b + 2, 3, 3, 4, 3, 4, 2, 4, 2, 4, 3, 2, 3, 6, 2, 4, // c + 2, 3, 3, 1, 3, 4, 2, 4, 2, 4, 3, 1, 3, 1, 2, 4, // d + 3, 3, 2, 1, 1, 4, 2, 4, 4, 1, 4, 1, 1, 1, 2, 4, // e + 3, 3, 2, 1, 1, 4, 2, 4, 3, 2, 4, 1, 0, 1, 2, 4 // f +}; + +int gbCyclesCB[] = { +// 0 1 2 3 4 5 6 7 8 9 a b c d e f + 2, 2, 2, 2, 2, 2, 4, 2, 2, 2, 2, 2, 2, 2, 4, 2, // 0 + 2, 2, 2, 2, 2, 2, 4, 2, 2, 2, 2, 2, 2, 2, 4, 2, // 1 + 2, 2, 2, 2, 2, 2, 4, 2, 2, 2, 2, 2, 2, 2, 4, 2, // 2 + 2, 2, 2, 2, 2, 2, 4, 2, 2, 2, 2, 2, 2, 2, 4, 2, // 3 + 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // 4 + 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // 5 + 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // 6 + 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // 7 + 2, 2, 2, 2, 2, 2, 4, 2, 2, 2, 2, 2, 2, 2, 4, 2, // 8 + 2, 2, 2, 2, 2, 2, 4, 2, 2, 2, 2, 2, 2, 2, 4, 2, // 9 + 2, 2, 2, 2, 2, 2, 4, 2, 2, 2, 2, 2, 2, 2, 4, 2, // a + 2, 2, 2, 2, 2, 2, 4, 2, 2, 2, 2, 2, 2, 2, 4, 2, // b + 2, 2, 2, 2, 2, 2, 4, 2, 2, 2, 2, 2, 2, 2, 4, 2, // c + 2, 2, 2, 2, 2, 2, 4, 2, 2, 2, 2, 2, 2, 2, 4, 2, // d + 2, 2, 2, 2, 2, 2, 4, 2, 2, 2, 2, 2, 2, 2, 4, 2, // e + 2, 2, 2, 2, 2, 2, 4, 2, 2, 2, 2, 2, 2, 2, 4, 2 // f +}; + +u16 DAATable[] = { + 0x0080,0x0100,0x0200,0x0300,0x0400,0x0500,0x0600,0x0700, + 0x0800,0x0900,0x1020,0x1120,0x1220,0x1320,0x1420,0x1520, + 0x1000,0x1100,0x1200,0x1300,0x1400,0x1500,0x1600,0x1700, + 0x1800,0x1900,0x2020,0x2120,0x2220,0x2320,0x2420,0x2520, + 0x2000,0x2100,0x2200,0x2300,0x2400,0x2500,0x2600,0x2700, + 0x2800,0x2900,0x3020,0x3120,0x3220,0x3320,0x3420,0x3520, + 0x3000,0x3100,0x3200,0x3300,0x3400,0x3500,0x3600,0x3700, + 0x3800,0x3900,0x4020,0x4120,0x4220,0x4320,0x4420,0x4520, + 0x4000,0x4100,0x4200,0x4300,0x4400,0x4500,0x4600,0x4700, + 0x4800,0x4900,0x5020,0x5120,0x5220,0x5320,0x5420,0x5520, + 0x5000,0x5100,0x5200,0x5300,0x5400,0x5500,0x5600,0x5700, + 0x5800,0x5900,0x6020,0x6120,0x6220,0x6320,0x6420,0x6520, + 0x6000,0x6100,0x6200,0x6300,0x6400,0x6500,0x6600,0x6700, + 0x6800,0x6900,0x7020,0x7120,0x7220,0x7320,0x7420,0x7520, + 0x7000,0x7100,0x7200,0x7300,0x7400,0x7500,0x7600,0x7700, + 0x7800,0x7900,0x8020,0x8120,0x8220,0x8320,0x8420,0x8520, + 0x8000,0x8100,0x8200,0x8300,0x8400,0x8500,0x8600,0x8700, + 0x8800,0x8900,0x9020,0x9120,0x9220,0x9320,0x9420,0x9520, + 0x9000,0x9100,0x9200,0x9300,0x9400,0x9500,0x9600,0x9700, + 0x9800,0x9900,0x00B0,0x0130,0x0230,0x0330,0x0430,0x0530, + 0x0090,0x0110,0x0210,0x0310,0x0410,0x0510,0x0610,0x0710, + 0x0810,0x0910,0x1030,0x1130,0x1230,0x1330,0x1430,0x1530, + 0x1010,0x1110,0x1210,0x1310,0x1410,0x1510,0x1610,0x1710, + 0x1810,0x1910,0x2030,0x2130,0x2230,0x2330,0x2430,0x2530, + 0x2010,0x2110,0x2210,0x2310,0x2410,0x2510,0x2610,0x2710, + 0x2810,0x2910,0x3030,0x3130,0x3230,0x3330,0x3430,0x3530, + 0x3010,0x3110,0x3210,0x3310,0x3410,0x3510,0x3610,0x3710, + 0x3810,0x3910,0x4030,0x4130,0x4230,0x4330,0x4430,0x4530, + 0x4010,0x4110,0x4210,0x4310,0x4410,0x4510,0x4610,0x4710, + 0x4810,0x4910,0x5030,0x5130,0x5230,0x5330,0x5430,0x5530, + 0x5010,0x5110,0x5210,0x5310,0x5410,0x5510,0x5610,0x5710, + 0x5810,0x5910,0x6030,0x6130,0x6230,0x6330,0x6430,0x6530, + 0x6010,0x6110,0x6210,0x6310,0x6410,0x6510,0x6610,0x6710, + 0x6810,0x6910,0x7030,0x7130,0x7230,0x7330,0x7430,0x7530, + 0x7010,0x7110,0x7210,0x7310,0x7410,0x7510,0x7610,0x7710, + 0x7810,0x7910,0x8030,0x8130,0x8230,0x8330,0x8430,0x8530, + 0x8010,0x8110,0x8210,0x8310,0x8410,0x8510,0x8610,0x8710, + 0x8810,0x8910,0x9030,0x9130,0x9230,0x9330,0x9430,0x9530, + 0x9010,0x9110,0x9210,0x9310,0x9410,0x9510,0x9610,0x9710, + 0x9810,0x9910,0xA030,0xA130,0xA230,0xA330,0xA430,0xA530, + 0xA010,0xA110,0xA210,0xA310,0xA410,0xA510,0xA610,0xA710, + 0xA810,0xA910,0xB030,0xB130,0xB230,0xB330,0xB430,0xB530, + 0xB010,0xB110,0xB210,0xB310,0xB410,0xB510,0xB610,0xB710, + 0xB810,0xB910,0xC030,0xC130,0xC230,0xC330,0xC430,0xC530, + 0xC010,0xC110,0xC210,0xC310,0xC410,0xC510,0xC610,0xC710, + 0xC810,0xC910,0xD030,0xD130,0xD230,0xD330,0xD430,0xD530, + 0xD010,0xD110,0xD210,0xD310,0xD410,0xD510,0xD610,0xD710, + 0xD810,0xD910,0xE030,0xE130,0xE230,0xE330,0xE430,0xE530, + 0xE010,0xE110,0xE210,0xE310,0xE410,0xE510,0xE610,0xE710, + 0xE810,0xE910,0xF030,0xF130,0xF230,0xF330,0xF430,0xF530, + 0xF010,0xF110,0xF210,0xF310,0xF410,0xF510,0xF610,0xF710, + 0xF810,0xF910,0x00B0,0x0130,0x0230,0x0330,0x0430,0x0530, + 0x0090,0x0110,0x0210,0x0310,0x0410,0x0510,0x0610,0x0710, + 0x0810,0x0910,0x1030,0x1130,0x1230,0x1330,0x1430,0x1530, + 0x1010,0x1110,0x1210,0x1310,0x1410,0x1510,0x1610,0x1710, + 0x1810,0x1910,0x2030,0x2130,0x2230,0x2330,0x2430,0x2530, + 0x2010,0x2110,0x2210,0x2310,0x2410,0x2510,0x2610,0x2710, + 0x2810,0x2910,0x3030,0x3130,0x3230,0x3330,0x3430,0x3530, + 0x3010,0x3110,0x3210,0x3310,0x3410,0x3510,0x3610,0x3710, + 0x3810,0x3910,0x4030,0x4130,0x4230,0x4330,0x4430,0x4530, + 0x4010,0x4110,0x4210,0x4310,0x4410,0x4510,0x4610,0x4710, + 0x4810,0x4910,0x5030,0x5130,0x5230,0x5330,0x5430,0x5530, + 0x5010,0x5110,0x5210,0x5310,0x5410,0x5510,0x5610,0x5710, + 0x5810,0x5910,0x6030,0x6130,0x6230,0x6330,0x6430,0x6530, + 0x0600,0x0700,0x0800,0x0900,0x0A00,0x0B00,0x0C00,0x0D00, + 0x0E00,0x0F00,0x1020,0x1120,0x1220,0x1320,0x1420,0x1520, + 0x1600,0x1700,0x1800,0x1900,0x1A00,0x1B00,0x1C00,0x1D00, + 0x1E00,0x1F00,0x2020,0x2120,0x2220,0x2320,0x2420,0x2520, + 0x2600,0x2700,0x2800,0x2900,0x2A00,0x2B00,0x2C00,0x2D00, + 0x2E00,0x2F00,0x3020,0x3120,0x3220,0x3320,0x3420,0x3520, + 0x3600,0x3700,0x3800,0x3900,0x3A00,0x3B00,0x3C00,0x3D00, + 0x3E00,0x3F00,0x4020,0x4120,0x4220,0x4320,0x4420,0x4520, + 0x4600,0x4700,0x4800,0x4900,0x4A00,0x4B00,0x4C00,0x4D00, + 0x4E00,0x4F00,0x5020,0x5120,0x5220,0x5320,0x5420,0x5520, + 0x5600,0x5700,0x5800,0x5900,0x5A00,0x5B00,0x5C00,0x5D00, + 0x5E00,0x5F00,0x6020,0x6120,0x6220,0x6320,0x6420,0x6520, + 0x6600,0x6700,0x6800,0x6900,0x6A00,0x6B00,0x6C00,0x6D00, + 0x6E00,0x6F00,0x7020,0x7120,0x7220,0x7320,0x7420,0x7520, + 0x7600,0x7700,0x7800,0x7900,0x7A00,0x7B00,0x7C00,0x7D00, + 0x7E00,0x7F00,0x8020,0x8120,0x8220,0x8320,0x8420,0x8520, + 0x8600,0x8700,0x8800,0x8900,0x8A00,0x8B00,0x8C00,0x8D00, + 0x8E00,0x8F00,0x9020,0x9120,0x9220,0x9320,0x9420,0x9520, + 0x9600,0x9700,0x9800,0x9900,0x9A00,0x9B00,0x9C00,0x9D00, + 0x9E00,0x9F00,0x00B0,0x0130,0x0230,0x0330,0x0430,0x0530, + 0x0610,0x0710,0x0810,0x0910,0x0A10,0x0B10,0x0C10,0x0D10, + 0x0E10,0x0F10,0x1030,0x1130,0x1230,0x1330,0x1430,0x1530, + 0x1610,0x1710,0x1810,0x1910,0x1A10,0x1B10,0x1C10,0x1D10, + 0x1E10,0x1F10,0x2030,0x2130,0x2230,0x2330,0x2430,0x2530, + 0x2610,0x2710,0x2810,0x2910,0x2A10,0x2B10,0x2C10,0x2D10, + 0x2E10,0x2F10,0x3030,0x3130,0x3230,0x3330,0x3430,0x3530, + 0x3610,0x3710,0x3810,0x3910,0x3A10,0x3B10,0x3C10,0x3D10, + 0x3E10,0x3F10,0x4030,0x4130,0x4230,0x4330,0x4430,0x4530, + 0x4610,0x4710,0x4810,0x4910,0x4A10,0x4B10,0x4C10,0x4D10, + 0x4E10,0x4F10,0x5030,0x5130,0x5230,0x5330,0x5430,0x5530, + 0x5610,0x5710,0x5810,0x5910,0x5A10,0x5B10,0x5C10,0x5D10, + 0x5E10,0x5F10,0x6030,0x6130,0x6230,0x6330,0x6430,0x6530, + 0x6610,0x6710,0x6810,0x6910,0x6A10,0x6B10,0x6C10,0x6D10, + 0x6E10,0x6F10,0x7030,0x7130,0x7230,0x7330,0x7430,0x7530, + 0x7610,0x7710,0x7810,0x7910,0x7A10,0x7B10,0x7C10,0x7D10, + 0x7E10,0x7F10,0x8030,0x8130,0x8230,0x8330,0x8430,0x8530, + 0x8610,0x8710,0x8810,0x8910,0x8A10,0x8B10,0x8C10,0x8D10, + 0x8E10,0x8F10,0x9030,0x9130,0x9230,0x9330,0x9430,0x9530, + 0x9610,0x9710,0x9810,0x9910,0x9A10,0x9B10,0x9C10,0x9D10, + 0x9E10,0x9F10,0xA030,0xA130,0xA230,0xA330,0xA430,0xA530, + 0xA610,0xA710,0xA810,0xA910,0xAA10,0xAB10,0xAC10,0xAD10, + 0xAE10,0xAF10,0xB030,0xB130,0xB230,0xB330,0xB430,0xB530, + 0xB610,0xB710,0xB810,0xB910,0xBA10,0xBB10,0xBC10,0xBD10, + 0xBE10,0xBF10,0xC030,0xC130,0xC230,0xC330,0xC430,0xC530, + 0xC610,0xC710,0xC810,0xC910,0xCA10,0xCB10,0xCC10,0xCD10, + 0xCE10,0xCF10,0xD030,0xD130,0xD230,0xD330,0xD430,0xD530, + 0xD610,0xD710,0xD810,0xD910,0xDA10,0xDB10,0xDC10,0xDD10, + 0xDE10,0xDF10,0xE030,0xE130,0xE230,0xE330,0xE430,0xE530, + 0xE610,0xE710,0xE810,0xE910,0xEA10,0xEB10,0xEC10,0xED10, + 0xEE10,0xEF10,0xF030,0xF130,0xF230,0xF330,0xF430,0xF530, + 0xF610,0xF710,0xF810,0xF910,0xFA10,0xFB10,0xFC10,0xFD10, + 0xFE10,0xFF10,0x00B0,0x0130,0x0230,0x0330,0x0430,0x0530, + 0x0610,0x0710,0x0810,0x0910,0x0A10,0x0B10,0x0C10,0x0D10, + 0x0E10,0x0F10,0x1030,0x1130,0x1230,0x1330,0x1430,0x1530, + 0x1610,0x1710,0x1810,0x1910,0x1A10,0x1B10,0x1C10,0x1D10, + 0x1E10,0x1F10,0x2030,0x2130,0x2230,0x2330,0x2430,0x2530, + 0x2610,0x2710,0x2810,0x2910,0x2A10,0x2B10,0x2C10,0x2D10, + 0x2E10,0x2F10,0x3030,0x3130,0x3230,0x3330,0x3430,0x3530, + 0x3610,0x3710,0x3810,0x3910,0x3A10,0x3B10,0x3C10,0x3D10, + 0x3E10,0x3F10,0x4030,0x4130,0x4230,0x4330,0x4430,0x4530, + 0x4610,0x4710,0x4810,0x4910,0x4A10,0x4B10,0x4C10,0x4D10, + 0x4E10,0x4F10,0x5030,0x5130,0x5230,0x5330,0x5430,0x5530, + 0x5610,0x5710,0x5810,0x5910,0x5A10,0x5B10,0x5C10,0x5D10, + 0x5E10,0x5F10,0x6030,0x6130,0x6230,0x6330,0x6430,0x6530, + 0x00C0,0x0140,0x0240,0x0340,0x0440,0x0540,0x0640,0x0740, + 0x0840,0x0940,0x0440,0x0540,0x0640,0x0740,0x0840,0x0940, + 0x1040,0x1140,0x1240,0x1340,0x1440,0x1540,0x1640,0x1740, + 0x1840,0x1940,0x1440,0x1540,0x1640,0x1740,0x1840,0x1940, + 0x2040,0x2140,0x2240,0x2340,0x2440,0x2540,0x2640,0x2740, + 0x2840,0x2940,0x2440,0x2540,0x2640,0x2740,0x2840,0x2940, + 0x3040,0x3140,0x3240,0x3340,0x3440,0x3540,0x3640,0x3740, + 0x3840,0x3940,0x3440,0x3540,0x3640,0x3740,0x3840,0x3940, + 0x4040,0x4140,0x4240,0x4340,0x4440,0x4540,0x4640,0x4740, + 0x4840,0x4940,0x4440,0x4540,0x4640,0x4740,0x4840,0x4940, + 0x5040,0x5140,0x5240,0x5340,0x5440,0x5540,0x5640,0x5740, + 0x5840,0x5940,0x5440,0x5540,0x5640,0x5740,0x5840,0x5940, + 0x6040,0x6140,0x6240,0x6340,0x6440,0x6540,0x6640,0x6740, + 0x6840,0x6940,0x6440,0x6540,0x6640,0x6740,0x6840,0x6940, + 0x7040,0x7140,0x7240,0x7340,0x7440,0x7540,0x7640,0x7740, + 0x7840,0x7940,0x7440,0x7540,0x7640,0x7740,0x7840,0x7940, + 0x8040,0x8140,0x8240,0x8340,0x8440,0x8540,0x8640,0x8740, + 0x8840,0x8940,0x8440,0x8540,0x8640,0x8740,0x8840,0x8940, + 0x9040,0x9140,0x9240,0x9340,0x9440,0x9540,0x9640,0x9740, + 0x9840,0x9940,0x3450,0x3550,0x3650,0x3750,0x3850,0x3950, + 0x4050,0x4150,0x4250,0x4350,0x4450,0x4550,0x4650,0x4750, + 0x4850,0x4950,0x4450,0x4550,0x4650,0x4750,0x4850,0x4950, + 0x5050,0x5150,0x5250,0x5350,0x5450,0x5550,0x5650,0x5750, + 0x5850,0x5950,0x5450,0x5550,0x5650,0x5750,0x5850,0x5950, + 0x6050,0x6150,0x6250,0x6350,0x6450,0x6550,0x6650,0x6750, + 0x6850,0x6950,0x6450,0x6550,0x6650,0x6750,0x6850,0x6950, + 0x7050,0x7150,0x7250,0x7350,0x7450,0x7550,0x7650,0x7750, + 0x7850,0x7950,0x7450,0x7550,0x7650,0x7750,0x7850,0x7950, + 0x8050,0x8150,0x8250,0x8350,0x8450,0x8550,0x8650,0x8750, + 0x8850,0x8950,0x8450,0x8550,0x8650,0x8750,0x8850,0x8950, + 0x9050,0x9150,0x9250,0x9350,0x9450,0x9550,0x9650,0x9750, + 0x9850,0x9950,0x9450,0x9550,0x9650,0x9750,0x9850,0x9950, + 0xA050,0xA150,0xA250,0xA350,0xA450,0xA550,0xA650,0xA750, + 0xA850,0xA950,0xA450,0xA550,0xA650,0xA750,0xA850,0xA950, + 0xB050,0xB150,0xB250,0xB350,0xB450,0xB550,0xB650,0xB750, + 0xB850,0xB950,0xB450,0xB550,0xB650,0xB750,0xB850,0xB950, + 0xC050,0xC150,0xC250,0xC350,0xC450,0xC550,0xC650,0xC750, + 0xC850,0xC950,0xC450,0xC550,0xC650,0xC750,0xC850,0xC950, + 0xD050,0xD150,0xD250,0xD350,0xD450,0xD550,0xD650,0xD750, + 0xD850,0xD950,0xD450,0xD550,0xD650,0xD750,0xD850,0xD950, + 0xE050,0xE150,0xE250,0xE350,0xE450,0xE550,0xE650,0xE750, + 0xE850,0xE950,0xE450,0xE550,0xE650,0xE750,0xE850,0xE950, + 0xF050,0xF150,0xF250,0xF350,0xF450,0xF550,0xF650,0xF750, + 0xF850,0xF950,0xF450,0xF550,0xF650,0xF750,0xF850,0xF950, + 0x00D0,0x0150,0x0250,0x0350,0x0450,0x0550,0x0650,0x0750, + 0x0850,0x0950,0x0450,0x0550,0x0650,0x0750,0x0850,0x0950, + 0x1050,0x1150,0x1250,0x1350,0x1450,0x1550,0x1650,0x1750, + 0x1850,0x1950,0x1450,0x1550,0x1650,0x1750,0x1850,0x1950, + 0x2050,0x2150,0x2250,0x2350,0x2450,0x2550,0x2650,0x2750, + 0x2850,0x2950,0x2450,0x2550,0x2650,0x2750,0x2850,0x2950, + 0x3050,0x3150,0x3250,0x3350,0x3450,0x3550,0x3650,0x3750, + 0x3850,0x3950,0x3450,0x3550,0x3650,0x3750,0x3850,0x3950, + 0x4050,0x4150,0x4250,0x4350,0x4450,0x4550,0x4650,0x4750, + 0x4850,0x4950,0x4450,0x4550,0x4650,0x4750,0x4850,0x4950, + 0x5050,0x5150,0x5250,0x5350,0x5450,0x5550,0x5650,0x5750, + 0x5850,0x5950,0x5450,0x5550,0x5650,0x5750,0x5850,0x5950, + 0x6050,0x6150,0x6250,0x6350,0x6450,0x6550,0x6650,0x6750, + 0x6850,0x6950,0x6450,0x6550,0x6650,0x6750,0x6850,0x6950, + 0x7050,0x7150,0x7250,0x7350,0x7450,0x7550,0x7650,0x7750, + 0x7850,0x7950,0x7450,0x7550,0x7650,0x7750,0x7850,0x7950, + 0x8050,0x8150,0x8250,0x8350,0x8450,0x8550,0x8650,0x8750, + 0x8850,0x8950,0x8450,0x8550,0x8650,0x8750,0x8850,0x8950, + 0x9050,0x9150,0x9250,0x9350,0x9450,0x9550,0x9650,0x9750, + 0x9850,0x9950,0x9450,0x9550,0x9650,0x9750,0x9850,0x9950, + 0xFA60,0xFB60,0xFC60,0xFD60,0xFE60,0xFF60,0x00C0,0x0140, + 0x0240,0x0340,0x0440,0x0540,0x0640,0x0740,0x0840,0x0940, + 0x0A60,0x0B60,0x0C60,0x0D60,0x0E60,0x0F60,0x1040,0x1140, + 0x1240,0x1340,0x1440,0x1540,0x1640,0x1740,0x1840,0x1940, + 0x1A60,0x1B60,0x1C60,0x1D60,0x1E60,0x1F60,0x2040,0x2140, + 0x2240,0x2340,0x2440,0x2540,0x2640,0x2740,0x2840,0x2940, + 0x2A60,0x2B60,0x2C60,0x2D60,0x2E60,0x2F60,0x3040,0x3140, + 0x3240,0x3340,0x3440,0x3540,0x3640,0x3740,0x3840,0x3940, + 0x3A60,0x3B60,0x3C60,0x3D60,0x3E60,0x3F60,0x4040,0x4140, + 0x4240,0x4340,0x4440,0x4540,0x4640,0x4740,0x4840,0x4940, + 0x4A60,0x4B60,0x4C60,0x4D60,0x4E60,0x4F60,0x5040,0x5140, + 0x5240,0x5340,0x5440,0x5540,0x5640,0x5740,0x5840,0x5940, + 0x5A60,0x5B60,0x5C60,0x5D60,0x5E60,0x5F60,0x6040,0x6140, + 0x6240,0x6340,0x6440,0x6540,0x6640,0x6740,0x6840,0x6940, + 0x6A60,0x6B60,0x6C60,0x6D60,0x6E60,0x6F60,0x7040,0x7140, + 0x7240,0x7340,0x7440,0x7540,0x7640,0x7740,0x7840,0x7940, + 0x7A60,0x7B60,0x7C60,0x7D60,0x7E60,0x7F60,0x8040,0x8140, + 0x8240,0x8340,0x8440,0x8540,0x8640,0x8740,0x8840,0x8940, + 0x8A60,0x8B60,0x8C60,0x8D60,0x8E60,0x8F60,0x9040,0x9140, + 0x9240,0x9340,0x3450,0x3550,0x3650,0x3750,0x3850,0x3950, + 0x3A70,0x3B70,0x3C70,0x3D70,0x3E70,0x3F70,0x4050,0x4150, + 0x4250,0x4350,0x4450,0x4550,0x4650,0x4750,0x4850,0x4950, + 0x4A70,0x4B70,0x4C70,0x4D70,0x4E70,0x4F70,0x5050,0x5150, + 0x5250,0x5350,0x5450,0x5550,0x5650,0x5750,0x5850,0x5950, + 0x5A70,0x5B70,0x5C70,0x5D70,0x5E70,0x5F70,0x6050,0x6150, + 0x6250,0x6350,0x6450,0x6550,0x6650,0x6750,0x6850,0x6950, + 0x6A70,0x6B70,0x6C70,0x6D70,0x6E70,0x6F70,0x7050,0x7150, + 0x7250,0x7350,0x7450,0x7550,0x7650,0x7750,0x7850,0x7950, + 0x7A70,0x7B70,0x7C70,0x7D70,0x7E70,0x7F70,0x8050,0x8150, + 0x8250,0x8350,0x8450,0x8550,0x8650,0x8750,0x8850,0x8950, + 0x8A70,0x8B70,0x8C70,0x8D70,0x8E70,0x8F70,0x9050,0x9150, + 0x9250,0x9350,0x9450,0x9550,0x9650,0x9750,0x9850,0x9950, + 0x9A70,0x9B70,0x9C70,0x9D70,0x9E70,0x9F70,0xA050,0xA150, + 0xA250,0xA350,0xA450,0xA550,0xA650,0xA750,0xA850,0xA950, + 0xAA70,0xAB70,0xAC70,0xAD70,0xAE70,0xAF70,0xB050,0xB150, + 0xB250,0xB350,0xB450,0xB550,0xB650,0xB750,0xB850,0xB950, + 0xBA70,0xBB70,0xBC70,0xBD70,0xBE70,0xBF70,0xC050,0xC150, + 0xC250,0xC350,0xC450,0xC550,0xC650,0xC750,0xC850,0xC950, + 0xCA70,0xCB70,0xCC70,0xCD70,0xCE70,0xCF70,0xD050,0xD150, + 0xD250,0xD350,0xD450,0xD550,0xD650,0xD750,0xD850,0xD950, + 0xDA70,0xDB70,0xDC70,0xDD70,0xDE70,0xDF70,0xE050,0xE150, + 0xE250,0xE350,0xE450,0xE550,0xE650,0xE750,0xE850,0xE950, + 0xEA70,0xEB70,0xEC70,0xED70,0xEE70,0xEF70,0xF050,0xF150, + 0xF250,0xF350,0xF450,0xF550,0xF650,0xF750,0xF850,0xF950, + 0xFA70,0xFB70,0xFC70,0xFD70,0xFE70,0xFF70,0x00D0,0x0150, + 0x0250,0x0350,0x0450,0x0550,0x0650,0x0750,0x0850,0x0950, + 0x0A70,0x0B70,0x0C70,0x0D70,0x0E70,0x0F70,0x1050,0x1150, + 0x1250,0x1350,0x1450,0x1550,0x1650,0x1750,0x1850,0x1950, + 0x1A70,0x1B70,0x1C70,0x1D70,0x1E70,0x1F70,0x2050,0x2150, + 0x2250,0x2350,0x2450,0x2550,0x2650,0x2750,0x2850,0x2950, + 0x2A70,0x2B70,0x2C70,0x2D70,0x2E70,0x2F70,0x3050,0x3150, + 0x3250,0x3350,0x3450,0x3550,0x3650,0x3750,0x3850,0x3950, + 0x3A70,0x3B70,0x3C70,0x3D70,0x3E70,0x3F70,0x4050,0x4150, + 0x4250,0x4350,0x4450,0x4550,0x4650,0x4750,0x4850,0x4950, + 0x4A70,0x4B70,0x4C70,0x4D70,0x4E70,0x4F70,0x5050,0x5150, + 0x5250,0x5350,0x5450,0x5550,0x5650,0x5750,0x5850,0x5950, + 0x5A70,0x5B70,0x5C70,0x5D70,0x5E70,0x5F70,0x6050,0x6150, + 0x6250,0x6350,0x6450,0x6550,0x6650,0x6750,0x6850,0x6950, + 0x6A70,0x6B70,0x6C70,0x6D70,0x6E70,0x6F70,0x7050,0x7150, + 0x7250,0x7350,0x7450,0x7550,0x7650,0x7750,0x7850,0x7950, + 0x7A70,0x7B70,0x7C70,0x7D70,0x7E70,0x7F70,0x8050,0x8150, + 0x8250,0x8350,0x8450,0x8550,0x8650,0x8750,0x8850,0x8950, + 0x8A70,0x8B70,0x8C70,0x8D70,0x8E70,0x8F70,0x9050,0x9150, + 0x9250,0x9350,0x9450,0x9550,0x9650,0x9750,0x9850,0x9950, +}; + +u8 ZeroTable[256] = { + 0x80,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0 +}; + +#define GBSAVE_GAME_VERSION_1 1 +#define GBSAVE_GAME_VERSION_2 2 +#define GBSAVE_GAME_VERSION_3 3 +#define GBSAVE_GAME_VERSION_4 4 +#define GBSAVE_GAME_VERSION_5 5 +#define GBSAVE_GAME_VERSION_6 6 +#define GBSAVE_GAME_VERSION_7 7 +#define GBSAVE_GAME_VERSION_8 8 +#define GBSAVE_GAME_VERSION_9 9 +#define GBSAVE_GAME_VERSION_10 10 +#define GBSAVE_GAME_VERSION_11 11 +#define GBSAVE_GAME_VERSION GBSAVE_GAME_VERSION_11 + +int inline gbGetValue(int min,int max,int v) +{ + return (int)(min+(float)(max-min)*(2.0*(v/31.0)-(v/31.0)*(v/31.0))); +} + +void gbGenFilter() +{ + for (int r=0;r<32;r++) { + for (int g=0;g<32;g++) { + for (int b=0;b<32;b++) { + int nr=gbGetValue(gbGetValue(4,14,g), + gbGetValue(24,29,g),r)-4; + int ng=gbGetValue(gbGetValue(4+gbGetValue(0,5,r), + 14+gbGetValue(0,3,r),b), + gbGetValue(24+gbGetValue(0,3,r), + 29+gbGetValue(0,1,r),b),g)-4; + int nb=gbGetValue(gbGetValue(4+gbGetValue(0,5,r), + 14+gbGetValue(0,3,r),g), + gbGetValue(24+gbGetValue(0,3,r), + 29+gbGetValue(0,1,r),g),b)-4; + gbColorFilter[(b<<10)|(g<<5)|r]=(nb<<10)|(ng<<5)|nr; + } + } + } +} + +bool gbIsGameboyRom(char * file) +{ + if(strlen(file) > 4) { + char * p = strrchr(file,'.'); + + if(p != NULL) { + if(_stricmp(p, ".gb") == 0) + return true; + if(_stricmp(p, ".gbc") == 0) + return true; + if(_stricmp(p, ".cgb") == 0) + return true; + if(_stricmp(p, ".sgb") == 0) + return true; + } + } + + return false; +} + +void gbCopyMemory(u16 d, u16 s, int count) +{ + while(count) { + gbMemoryMap[d>>12][d & 0x0fff] = gbMemoryMap[s>>12][s & 0x0fff]; + s++; + d++; + count--; + } +} + +void gbDoHdma() +{ + + gbCopyMemory((gbHdmaDestination & 0x1ff0) | 0x8000, + gbHdmaSource & 0xfff0, + 0x10); + + gbHdmaDestination += 0x10; + if (gbHdmaDestination == 0xa000) + gbHdmaDestination = 0x8000; + + gbHdmaSource += 0x10; + if (gbHdmaSource == 0x8000) + gbHdmaSource = 0xa000; + + register_HDMA2 = gbHdmaSource & 0xff; + register_HDMA1 = gbHdmaSource>>8; + + register_HDMA4 = gbHdmaDestination & 0xff; + register_HDMA3 = gbHdmaDestination>>8; + + gbHdmaBytes -= 0x10; + gbMemory[0xff55] = --register_HDMA5; + if(register_HDMA5 == 0xff) + gbHdmaOn = 0; + +// We need to add the dmaClockticks for HDMA ! + if(gbSpeed) + gbDmaTicks = 17; + else + gbDmaTicks = 9; + + if (IFF & 0x80) + gbDmaTicks++; + +} + +// fix for Harley and Lego Racers +void gbCompareLYToLYC() +{ + if(register_LCDC & 0x80) { + if(register_LY == register_LYC) { + + // mark that we have a match + register_STAT |= 4; + + // check if we need an interrupt + if (register_STAT & 0x40) + { + // send LCD interrupt only if no interrupt 48h signal... + if (!gbInt48Signal) + { + register_IF |=2; + } + gbInt48Signal |= 8; + } + } + else // no match + { + register_STAT &= 0xfb; + gbInt48Signal &=~8; + } + } +} + +void gbWriteMemory(register u16 address, register u8 value) +{ + + if(address < 0x8000) { +#ifndef FINAL_VERSION + if(memorydebug && (address>0x3fff || address < 0x2000)) { + log("Memory register write %04x=%02x PC=%04x\n", + address, + value, + PC.W); + } + +#endif + if(mapper) + (*mapper)(address, value); + return; + + } + + if(address < 0xa000) { + // No access to Vram during mode 3 + // (used to emulate the gfx differences between GB & GBC-GBA/SP in Stunt Racer) + if ((gbLcdModeDelayed !=3) || + // This part is used to emulate a small difference between hardwares + // (check 8-in-1's arrow on GBA/GBC to verify it) + ((register_LY == 0) && ((gbHardware & 0xa) && (gbScreenOn==false) && + (register_LCDC & 0x80)) && + (gbLcdLYIncrementTicksDelayed ==(GBLY_INCREMENT_CLOCK_TICKS-GBLCD_MODE_2_CLOCK_TICKS)))) + gbMemoryMap[address>>12][address&0x0fff] = value; + return; + } + + // Used for the mirroring of 0xC000 in 0xE000 + if ((address >= 0xe000) && (address < 0xfe00)) + address &= ~0x2000; + + if(address < 0xc000) { +#ifndef FINAL_VERSION + if(memorydebug) { + log("Memory register write %04x=%02x PC=%04x\n", + address, + value, + PC.W); + } +#endif + + // Is that a correct fix ??? (it used to be 'if (mapper)')... + if(mapperRAM) + (*mapperRAM)(address, value); + return; + } + + + if(address < 0xfe00) { + gbMemoryMap[address>>12][address & 0x0fff] = value; + return; + } + + // OAM not accessible during mode 2 & 3. + if(address < 0xfea0) + { + if (((gbHardware & 0xa) && ((gbLcdMode | gbLcdModeDelayed) &2)) || + ((gbHardware & 5) && (((gbLcdModeDelayed == 2) && + (gbLcdTicksDelayed<=GBLCD_MODE_2_CLOCK_TICKS)) || + (gbLcdModeDelayed == 3)))) + return; + else + { + gbMemory[address] = value; + return; + } + } + + + + if((address > 0xfea0) && (address < 0xff00)){ // GBC allows reading/writing to that area + gbMemory[address] = value; + return; + } + + switch(address & 0x00ff) { + + case 0x00: { + gbMemory[0xff00] = ((gbMemory[0xff00] & 0xcf) | + (value & 0x30) | 0xc0); + if(gbSgbMode) { + gbSgbDoBitTransfer(value); + } + return; + } + + case 0x01: { + gbMemory[0xff01] = value; + return; + } + + // serial control + case 0x02: { + gbSerialOn = (value & 0x80); + gbMemory[0xff02] = value; + if(gbSerialOn) { + gbSerialTicks = GBSERIAL_CLOCK_TICKS; +#ifdef LINK_EMULATION + if(linkConnected) { + if(value & 1) { + linkSendByte(0x100|gbMemory[0xFF01]); + Sleep(5); + } + } +#endif + } + + gbSerialBits = 0; + return; + } + + case 0x04: { + // DIV register resets on any write + // (not totally perfect, but better than nothing) + gbMemory[0xff04] = register_DIV = 0; + gbDivTicks = GBDIV_CLOCK_TICKS; + // Another weird timer 'bug' : + // Writing to DIV register resets the internal timer, + // and can also increase TIMA/trigger an interrupt + // in some cases... + if (gbTimerOn && !(gbInternalTimer & (gbTimerClockTicks>>1))) + { + gbMemory[0xff05] = ++register_TIMA; + if(register_TIMA == 0) { + // timer overflow! + + // reload timer modulo + gbMemory[0xff05] = register_TIMA = register_TMA; + + // flag interrupt + gbMemory[0xff0f] = register_IF |= 4; + } + } + gbInternalTimer = 0xff; + return; + } + case 0x05: + gbMemory[0xff05] = register_TIMA = value; + return; + + case 0x06: + gbMemory[0xff06] = register_TMA = value; + return; + + // TIMER control + case 0x07: { + + gbTimerModeChange = (((value & 3) != (register_TAC&3)) && (value & register_TAC & 4)) ? true : false; + gbTimerOnChange = (((value ^ register_TAC) & 4) == 4) ? true : false; + + gbTimerOn = (value & 4) ? true : false; + + if (gbTimerOnChange || gbTimerModeChange) + { + gbTimerMode = value & 3; + + switch(gbTimerMode) { + case 0: + gbTimerClockTicks = GBTIMER_MODE_0_CLOCK_TICKS; + break; + case 1: + gbTimerClockTicks = GBTIMER_MODE_1_CLOCK_TICKS; + break; + case 2: + gbTimerClockTicks = GBTIMER_MODE_2_CLOCK_TICKS; + break; + case 3: + gbTimerClockTicks = GBTIMER_MODE_3_CLOCK_TICKS; + break; + } + } + + + // This looks weird, but this emulates a bug in which register_TIMA + // is increased when writing to register_TAC + // (This fixes Korodice's long-delay between menus bug). + + if (gbTimerOnChange || gbTimerModeChange) + { + bool temp = false; + + if ((gbTimerOn && !gbTimerModeChange) && (gbTimerMode & 2) && + !(gbInternalTimer & 0x80) && (gbInternalTimer & (gbTimerClockTicks>>1)) && + !(gbInternalTimer & (gbTimerClockTicks>>5))) + temp = true; + else if ((!gbTimerOn && !gbTimerModeChange && gbTimerOnChange ) && ((gbTimerTicks-1) < (gbTimerClockTicks>>1))) + temp = true; + else if (gbTimerOn && gbTimerModeChange && !gbTimerOnChange) + { + switch(gbTimerMode & 3) + { + case 0x00: + temp = false; + break; + case 0x01: + if (((gbInternalTimer & 0x82) == 2) && (gbTimerTicks>(clockTicks+1))) + temp = true; + break; + case 0x02: + if (((gbInternalTimer & 0x88) == 0x8) && (gbTimerTicks>(clockTicks+1))) + temp = true; + break; + case 0x03: + if (((gbInternalTimer & 0xA0) == 0x20) && (gbTimerTicks>(clockTicks+1))) + temp = true; + break; + } + } + + if (temp) + { + gbMemory[0xff05] = ++register_TIMA; + if((register_TIMA & 0xff) == 0) { + // timer overflow! + + // reload timer modulo + gbMemory[0xff05] = register_TIMA = register_TMA; + + // flag interrupt + gbMemory[0xff0f] = register_IF |= 4; + } + } + } + gbMemory[0xff07] = register_TAC = value; + return; + } + + case 0x0f: { + gbMemory[0xff0f] = register_IF = value; + //gbMemory[0xff0f] = register_IE |= value; + return; + } + + case 0x10: + case 0x11: + case 0x12: + case 0x13: + case 0x14: + case 0x16: + case 0x17: + case 0x18: + case 0x19: + case 0x1a: + case 0x1b: + case 0x1c: + case 0x1d: + case 0x1e: + case 0x20: + case 0x21: + case 0x22: + case 0x23: + case 0x24: + case 0x25: { + if (gbMemory[NR52] & 0x80) { + SOUND_EVENT(address,value); + return; + } + } + + case 0x26: { + SOUND_EVENT(address,value); + return; + } + + case 0x30: + case 0x31: + case 0x32: + case 0x33: + case 0x34: + case 0x35: + case 0x36: + case 0x37: + case 0x38: + case 0x39: + case 0x3a: + case 0x3b: + case 0x3c: + case 0x3d: + case 0x3e: + case 0x3f: { + gbMemory[address] = value; + return; + } + + case 0x40: { + int lcdChange = (register_LCDC & 0x80) ^ (value & 0x80); + + // don't draw the window if it was not enabled and not being drawn before + if(!(register_LCDC & 0x20) && (value & 0x20) && gbWindowLine == -1 && + register_LY > register_WY) + gbWindowLine = 146; + // 007 fix : don't draw the first window's 1st line if it's enable 'too late' + // (ie. if register_LY == register_WY when enabling it) + // and move it to the next line + else if (!(register_LCDC & 0x20) && (value & 0x20) && (register_LY == register_WY)) + gbWindowLine = -2; + + + gbMemory[0xff40] = register_LCDC = value; + + + if(lcdChange) { + if((value & 0x80) && (!register_LCDCBusy)) { + + // if (!gbWhiteScreen && !gbSgbMask) + + // systemDrawScreen(); + + + + gbRegisterLYLCDCOffOn = (register_LY + 144) % 154; + + gbLcdTicks = GBLCD_MODE_2_CLOCK_TICKS - (gbSpeed ? 2 : 1); + gbLcdTicksDelayed = GBLCD_MODE_2_CLOCK_TICKS - (gbSpeed ? 1 : 0); + gbLcdLYIncrementTicks = GBLY_INCREMENT_CLOCK_TICKS - (gbSpeed ? 2 : 1); + gbLcdLYIncrementTicksDelayed = GBLY_INCREMENT_CLOCK_TICKS - (gbSpeed ? 1 : 0); + gbLcdMode = 2; + gbLcdModeDelayed = 2; + gbMemory[0xff41] = register_STAT = (register_STAT & 0xfc) | 2; + gbMemory[0xff44] = register_LY = 0x00; + gbInt48Signal = 0; + gbLYChangeHappened = false; + gbLCDChangeHappened = false; + gbWindowLine = 146; + oldRegister_WY = 146; + + // Fix for Namco Gallery Vol.2 + // (along with updating register_LCDC at the start of 'case 0x40') : + if(register_STAT & 0x20) + { + // send LCD interrupt only if no interrupt 48h signal... + if (!gbInt48Signal) + { + gbMemory[0xff0f] = register_IF |= 2; + } + gbInt48Signal |= 4; + } + gbCompareLYToLYC(); + + } else { + + register_LCDCBusy = clockTicks+6; + + //used to update the screen with white lines when it's off. + //(it looks strange, but it's pretty accurate) + + gbWhiteScreen = 0; + + gbScreenTicks = ((150-register_LY)*GBLY_INCREMENT_CLOCK_TICKS + + (49<<(gbSpeed ? 1 : 0))); + + // disable the screen rendering + gbScreenOn = false; + gbLcdTicks = 0; + gbLcdMode = 0; + gbLcdModeDelayed = 0; + gbMemory[0xff41] = register_STAT &= 0xfc; + gbInt48Signal = 0; + } + } + return; + } + + // STAT + case 0x41: { + //register_STAT = (register_STAT & 0x87) | + // (value & 0x7c); + gbMemory[0xff41] = register_STAT = (value & 0xf8) | (register_STAT & 0x07); // fix ? + // GB bug from Devrs FAQ + // Corrected : it happens if Lcd Mode<2, but also if LY == LYC whatever + // Lcd Mode is, and if !gbInt48Signal in all cases. The screen being off + // doesn't matter (the bug will still happen). + // That fixes 'Satoru Nakajima - F-1 Hero' crash bug. + + if((gbHardware & 5) && (((!gbInt48Signal) && (gbLcdMode<2) && (register_LCDC & 0x80)) || + (register_LY == register_LYC))) + { + gbMemory[0xff0f] = register_IF |=2; + } + + gbInt48Signal &= ((register_STAT>>3) & 0xF); + + if((register_LCDC & 0x80)) { + if ((register_STAT & 0x08) && (gbLcdMode == 0)) + { + if (!gbInt48Signal) + { + gbMemory[0xff0f] = register_IF |=2; + } + gbInt48Signal |= 1; + } + if ((register_STAT & 0x10) && (gbLcdMode == 1)) + { + if (!gbInt48Signal) + { + gbMemory[0xff0f] = register_IF |=2; + } + gbInt48Signal |= 2; + } + if ((register_STAT & 0x20) && (gbLcdMode == 2)) + { + if (!gbInt48Signal) + { + gbMemory[0xff0f] = register_IF |=2; + } + gbInt48Signal |= 4; + } + gbCompareLYToLYC(); + + gbMemory[0xff0f] = register_IF; + gbMemory[0xff41] = register_STAT; + + } + return; + } + + // SCY + case 0x42: { + int temp = -1; + + if ((gbLcdMode == 3) || (gbLcdModeDelayed == 3)) + temp = ((GBLY_INCREMENT_CLOCK_TICKS-GBLCD_MODE_2_CLOCK_TICKS) - + gbLcdLYIncrementTicks); + + if (temp >=0) + { + for (int i=temp<<(gbSpeed ? 1 : 2);i<300;i++) + if (temp < 300) + gbSCYLine[i] = value; + } + + else + memset(gbSCYLine, value, sizeof(gbSCYLine)); + + gbMemory[0xff42] = register_SCY = value; + return; + } + + // SCX + case 0x43: { + int temp = -1; + + if (gbLcdModeDelayed == 3) + temp = ((GBLY_INCREMENT_CLOCK_TICKS-GBLCD_MODE_2_CLOCK_TICKS) - + gbLcdLYIncrementTicksDelayed); + + if (temp >=0) + { + for (int i=temp<<(gbSpeed ? 1 : 2);i<300;i++) + if (temp < 300) + gbSCXLine[i] = value; + } + + else + memset(gbSCXLine, value, sizeof(gbSCXLine)); + + gbMemory[0xff43] = register_SCX = value; + return; + } + + // LY + case 0x44: { + // read only + return; + } + + // LYC + case 0x45: { + if (register_LYC != value) + { + gbMemory[0xff45] = register_LYC = value; + if(register_LCDC & 0x80) { + gbCompareLYToLYC(); + } + } + return; + } + + // DMA! + case 0x46: { + int source = value * 0x0100; + + gbCopyMemory(0xfe00, + source, + 0xa0); + gbMemory[0xff46] = register_DMA = value; + return; + } + + // BGP + case 0x47: { + + int temp = -1; + + gbMemory[0xff47] = value; + + if (gbLcdModeDelayed == 3) + temp = ((GBLY_INCREMENT_CLOCK_TICKS-GBLCD_MODE_2_CLOCK_TICKS) - + gbLcdLYIncrementTicksDelayed); + + if (temp >=0) + { + for (int i=temp<<(gbSpeed ? 1 : 2);i<300;i++) + if (temp < 300) + gbBgpLine[i] = value; + } + else + memset(gbBgpLine,value,sizeof(gbBgpLine)); + + gbBgp[0] = value & 0x03; + gbBgp[1] = (value & 0x0c)>>2; + gbBgp[2] = (value & 0x30)>>4; + gbBgp[3] = (value & 0xc0)>>6; + break; + } + + // OBP0 + case 0x48: { + int temp = -1; + + gbMemory[0xff48] = value; + + if (gbLcdModeDelayed == 3) + temp = ((GBLY_INCREMENT_CLOCK_TICKS-GBLCD_MODE_2_CLOCK_TICKS) - + gbLcdLYIncrementTicksDelayed); + + if (temp >=0) + { + for (int i=temp<<(gbSpeed ? 1 : 2);i<300;i++) + if (temp < 300) + gbObp0Line[i] = value; + } + else + memset(gbObp0Line,value,sizeof(gbObp0Line)); + + gbObp0[0] = value & 0x03; + gbObp0[1] = (value & 0x0c)>>2; + gbObp0[2] = (value & 0x30)>>4; + gbObp0[3] = (value & 0xc0)>>6; + break; + } + + // OBP1 + case 0x49: { + int temp = -1; + + gbMemory[0xff49] = value; + + if (gbLcdModeDelayed == 3) + temp = ((GBLY_INCREMENT_CLOCK_TICKS-GBLCD_MODE_2_CLOCK_TICKS) - + gbLcdLYIncrementTicksDelayed); + + if (temp >=0) + { + for (int i=temp<<(gbSpeed ? 1 : 2);i<300;i++) + if (temp < 300) + gbObp1Line[i] = value; + } + else + memset(gbObp1Line,value,sizeof(gbObp1Line)); + + gbObp1[0] = value & 0x03; + gbObp1[1] = (value & 0x0c)>>2; + gbObp1[2] = (value & 0x30)>>4; + gbObp1[3] = (value & 0xc0)>>6; + break; + } + + // WY + case 0x4a: + gbMemory[0xff4a] = register_WY = value; + if ((register_LY <= register_WY) && ((gbWindowLine < 0) || (gbWindowLine>=144))) + { + gbWindowLine = -1; + oldRegister_WY = register_WY; + } + return; + + // WX + case 0x4b: + gbMemory[0xff4b] = register_WX = value; + return; + + // KEY1 + case 0x4d: { + if(gbCgbMode) { + gbMemory[0xff4d] = (gbMemory[0xff4d] & 0x80) | (value & 1) | 0x7e; + return; + } + } + break; + + // VBK + case 0x4f: { + if(gbCgbMode) { + value = value & 1; + if(value == gbVramBank) + return; + + int vramAddress = value * 0x2000; + gbMemoryMap[0x08] = &gbVram[vramAddress]; + gbMemoryMap[0x09] = &gbVram[vramAddress + 0x1000]; + + gbVramBank = value; + register_VBK = value; + } + return; + } + break; + + // BOOTROM disable register (also gbCgbMode enabler if value & 0x10 ?) + case 0x50 : + { + if (useBios && inBios && !skipBios && (value & 1)) + { + gbMemoryMap[0x00] = &gbRom[0x0000]; + memcpy ((u8 *)(gbRom+0x100), (u8 *)(gbMemory + 0x100), 0xF00); + inBios = false; + } + } + + // HDMA1 + case 0x51: { + if(gbCgbMode) { + if(value > 0x7f && value < 0xa0) + value = 0; + + gbHdmaSource = (value << 8) | (gbHdmaSource & 0xf0); + + register_HDMA1 = value; + return; + } + } + break; + + // HDMA2 + case 0x52: { + if(gbCgbMode) { + value = value & 0xf0; + + gbHdmaSource = (gbHdmaSource & 0xff00) | (value); + + register_HDMA2 = value; + return; + } + } + break; + + // HDMA3 + case 0x53: { + if(gbCgbMode) { + value = value & 0x1f; + gbHdmaDestination = (value << 8) | (gbHdmaDestination & 0xf0); + gbHdmaDestination |= 0x8000; + register_HDMA3 = value; + return; + } + } + break; + + // HDMA4 + case 0x54: { + if(gbCgbMode) { + value = value & 0xf0; + gbHdmaDestination = (gbHdmaDestination & 0x1f00) | value; + gbHdmaDestination |= 0x8000; + register_HDMA4 = value; + return; + } + } + break; + + // HDMA5 + case 0x55: { + + if(gbCgbMode) { + gbHdmaBytes = 16 + (value & 0x7f) * 16; + if(gbHdmaOn) { + if(value & 0x80) { + gbMemory[0xff55] = register_HDMA5 = (value & 0x7f); + } else { + register_HDMA5 = 0xff; + gbHdmaOn = 0; + } + } else { + if(value & 0x80) { + gbHdmaOn = 1; + gbMemory[0xff55] = register_HDMA5 = value & 0x7f; + if(gbLcdModeDelayed == 0) + gbDoHdma(); + } else { + // we need to take the time it takes to complete the transfer into + // account... according to GB DEV FAQs, the setup time is the same + // for single and double speed, but the actual transfer takes the + // same time + if(gbSpeed) + gbDmaTicks = 2+16 * ((value & 0x7f) +1); + else + gbDmaTicks = 1+8 * ((value & 0x7f)+1); + + gbCopyMemory((gbHdmaDestination & 0x1ff0) | 0x8000, + gbHdmaSource & 0xfff0, + gbHdmaBytes); + gbHdmaDestination += gbHdmaBytes; + gbHdmaSource += gbHdmaBytes; + + gbMemory[0xff51] = register_HDMA1 = 0xff;// = (gbHdmaSource >> 8) & 0xff; + gbMemory[0xff52] = register_HDMA2 = 0xff;// = gbHdmaSource & 0xf0; + gbMemory[0xff53] = register_HDMA3 = 0xff;// = ((gbHdmaDestination - 0x8000) >> 8) & 0x1f; + gbMemory[0xff54] = register_HDMA4 = 0xff;// = gbHdmaDestination & 0xf0; + gbMemory[0xff55] = register_HDMA5 = 0xff; + } + } + return; + } + } + break; + + // BCPS + case 0x68: { + if(gbCgbMode) { + int paletteIndex = (value & 0x3f) >> 1; + int paletteHiLo = (value & 0x01); + + gbMemory[0xff68] = value; + + gbMemory[0xff69] = (paletteHiLo ? + (gbPalette[paletteIndex] >> 8) : + (gbPalette[paletteIndex] & 0x00ff)); + return; + } + } + break; + + // BCPD + case 0x69: { + if(gbCgbMode) { + int v = gbMemory[0xff68]; + int paletteIndex = (v & 0x3f) >> 1; + int paletteHiLo = (v & 0x01); + + // No access to gbPalette during mode 3 (Color Panel Demo) + if (((gbLcdModeDelayed != 3) && (!((gbLcdMode == 0) && (gbLcdTicks>=(GBLCD_MODE_0_CLOCK_TICKS-gbSpritesTicks[299]-1)))) && (!gbSpeed)) || + (gbSpeed && ((gbLcdMode == 1) || (gbLcdMode == 2) || + ((gbLcdMode == 3) && (gbLcdTicks>(GBLCD_MODE_3_CLOCK_TICKS-2))) || + ((gbLcdMode == 0) && (gbLcdTicks<=(GBLCD_MODE_0_CLOCK_TICKS-gbSpritesTicks[299]-2)))))) + { + gbMemory[0xff69] = value; + gbPalette[paletteIndex] = (paletteHiLo ? + ((value << 8) | (gbPalette[paletteIndex] & 0xff)) : + ((gbPalette[paletteIndex] & 0xff00) | (value))) & 0x7fff; + } + + + if(gbMemory[0xff68] & 0x80) { + int index = ((gbMemory[0xff68] & 0x3f) + 1) & 0x3f; + + gbMemory[0xff68] = (gbMemory[0xff68] & 0x80) | index; + gbMemory[0xff69] = (index & 1 ? + (gbPalette[index>>1] >> 8) : + (gbPalette[index>>1] & 0x00ff)); + } + return; + } + } + break; + + // OCPS + case 0x6a: { + if(gbCgbMode) { + int paletteIndex = (value & 0x3f) >> 1; + int paletteHiLo = (value & 0x01); + + paletteIndex += 32; + + gbMemory[0xff6a] = value; + + gbMemory[0xff6b] = (paletteHiLo ? + (gbPalette[paletteIndex] >> 8) : + (gbPalette[paletteIndex] & 0x00ff)); + return; + } + } + break; + + // OCPD + case 0x6b: { + + if(gbCgbMode) { + int v = gbMemory[0xff6a]; + int paletteIndex = (v & 0x3f) >> 1; + int paletteHiLo = (v & 0x01); + + paletteIndex += 32; + + // No access to gbPalette during mode 3 (Color Panel Demo) + if (((gbLcdModeDelayed != 3) && (!((gbLcdMode == 0) && (gbLcdTicks>=(GBLCD_MODE_0_CLOCK_TICKS-gbSpritesTicks[299]-1)))) && (!gbSpeed)) || + (gbSpeed && ((gbLcdMode == 1) || (gbLcdMode == 2) || + ((gbLcdMode == 3) && (gbLcdTicks>(GBLCD_MODE_3_CLOCK_TICKS-2))) || + ((gbLcdMode == 0) && (gbLcdTicks<=(GBLCD_MODE_0_CLOCK_TICKS-gbSpritesTicks[299]-2)))))) + { + gbMemory[0xff6b] = value; + gbPalette[paletteIndex] = (paletteHiLo ? + ((value << 8) | (gbPalette[paletteIndex] & 0xff)) : + ((gbPalette[paletteIndex] & 0xff00) | (value))) & 0x7fff; + } + + if(gbMemory[0xff6a] & 0x80) { + int index = ((gbMemory[0xff6a] & 0x3f) + 1) & 0x3f; + + gbMemory[0xff6a] = (gbMemory[0xff6a] & 0x80) | index; + + gbMemory[0xff6b] = (index & 1 ? + (gbPalette[(index>>1) + 32] >> 8) : + (gbPalette[(index>>1) + 32] & 0x00ff)); + } + return; + } + } + break; + + case 0x6c: { + gbMemory[0xff6c] = 0xfe | value; + return; + } + + + // SVBK + case 0x70: { + if(gbCgbMode) { + value = value & 7; + + int bank = value; + if(value == 0) + bank = 1; + + if(bank == gbWramBank) + return; + + int wramAddress = bank * 0x1000; + gbMemoryMap[0x0d] = &gbWram[wramAddress]; + + gbWramBank = bank; + gbMemory[0xff70] = register_SVBK = value; + return; + } + } + + case 0x75:{ + gbMemory[0xff75] = 0x8f | value; + return; + } + + case 0xff: { + gbMemory[0xffff] = register_IE = value; + return; + } + } + + if(address < 0xff80) + { + gbMemory[address] = value; + return; + } + + gbMemory[address] = value; +} + +u8 gbReadOpcode(register u16 address) +{ + if(gbCheatMap[address]) + return gbCheatRead(address); + + if(address < 0x8000) + return gbMemoryMap[address>>12][address&0x0fff]; + + if(address < 0xa000) + { + // A lot of 'ugly' checks... But only way to emulate this particular behaviour... + if (((gbHardware & 0xa) && ((gbLcdModeDelayed !=3) || ((register_LY == 0) && + (gbScreenOn==false) && (register_LCDC & 0x80)) && + (gbLcdLYIncrementTicksDelayed ==(GBLY_INCREMENT_CLOCK_TICKS-GBLCD_MODE_2_CLOCK_TICKS)))) || + ((gbHardware & 0x5) && (gbLcdModeDelayed !=3) && + ((gbLcdMode !=3) || ((register_LY == 0) && ((gbScreenOn==false) && + (register_LCDC & 0x80)) && + (gbLcdLYIncrementTicks ==(GBLY_INCREMENT_CLOCK_TICKS-GBLCD_MODE_2_CLOCK_TICKS)))))) + return gbMemoryMap[address>>12][address&0x0fff]; + return 0xff; + } + + // Used for the mirroring of 0xC000 in 0xE000 + if ((address >= 0xe000) && (address < 0xfe00)) + address &= ~0x2000; + + switch(address & 0xf000) { + case 0x0a: + case 0x0b: + if(mapperReadRAM) + return mapperReadRAM(address); + break; + case 0x0f: + if(address > 0xff00) { + switch(address & 0x00ff) { + case 0x02: + return (gbMemory[0xff02]); + case 0x03: + return (0xff); + case 0x04: + return register_DIV; + case 0x05: + return register_TIMA; + case 0x06: + return register_TMA; + case 0x07: + return (0xf8 | register_TAC); + case 0x08: + case 0x09: + case 0x0a: + case 0x0b: + case 0x0c: + case 0x0d: + case 0x0e: + return (0xff); + case 0x0f: + return (0xe0 | gbMemory[0xff0f]); + case 0x40: + return register_LCDC; + case 0x41: + // This is a GB/C only bug (ie. not GBA/SP). + if ((gbHardware & 7) && (gbLcdMode == 2) && (gbLcdModeDelayed == 1) && (!gbSpeed)) + return (0x80 | gbMemory[0xff41] & 0xFC); + else + return (0x80 | gbMemory[0xff41]); + case 0x42: + return register_SCY; + case 0x43: + return register_SCX; + case 0x44: + if (((gbHardware & 7) && ((gbLcdMode == 1) && (gbLcdTicks == 0x71))) || + (!(register_LCDC && 0x80))) + return 0; + else + return register_LY; + case 0x45: + return register_LYC; + case 0x46: + return register_DMA; + case 0x4a: + return register_WY; + case 0x4b: + return register_WX; + case 0x4c: + return 0xff; + case 0x4f: + return (0xfe | register_VBK); + case 0x51: + return register_HDMA1; + case 0x52: + return register_HDMA2; + case 0x53: + return register_HDMA3; + case 0x54: + return register_HDMA4; + case 0x55: + return register_HDMA5; + case 0x68: + case 0x6a: + if (gbCgbMode) + return (0x40 | gbMemory[address]); + else + return 0xc0; + case 0x69: + case 0x6b: + if (gbCgbMode) + { + // No access to gbPalette during mode 3 (Color Panel Demo) + if (((gbLcdModeDelayed != 3) && (!((gbLcdMode == 0) && (gbLcdTicks>=(GBLCD_MODE_0_CLOCK_TICKS-gbSpritesTicks[299]-1)))) && (!gbSpeed)) || + (gbSpeed && ((gbLcdMode == 1) || (gbLcdMode == 2) || + ((gbLcdMode == 3) && (gbLcdTicks>(GBLCD_MODE_3_CLOCK_TICKS-2))) || + ((gbLcdMode == 0) && (gbLcdTicks<=(GBLCD_MODE_0_CLOCK_TICKS-gbSpritesTicks[299]-2)))))) + return (gbMemory[address]); + else + return 0xff; + } + else + return 0xff; + case 0x70: + if (gbCgbMode) + return (0xf8 | register_SVBK); + else + return 0xff; + case 0xff: + return register_IE; + } + } + // OAM not accessible during mode 2 & 3. + if(((address >= 0xfe00) && (address<0xfea0)) && + ((gbLcdMode | gbLcdModeDelayed) &2)) + return 0xff; + break; + } + + if ((address >= 0xfea0) && (address < 0xff00)) + { + if (gbHardware & 1) + return ((((address + ((address >> 4) - 0xfea)) >> 2) & 1) ? 0x00 : 0xff ); + else if (gbHardware & 2) + return gbMemoryMap[address>>12][address & 0x0fff]; + else if (gbHardware & 4) + return ((((address + ((address >> 4) - 0xfea)) >> 2) & 1) ? 0xff : 0x00 ); + else if (gbHardware & 8) + return ((address & 0xf0) |((address & 0xf0)>>4)); + } + + return gbMemoryMap[address>>12][address & 0x0fff]; +} + +u8 gbReadMemory(register u16 address) +{ + if(gbCheatMap[address]) + return gbCheatRead(address); + + + if(address < 0x8000) + return gbMemoryMap[address>>12][address&0x0fff]; + + if(address < 0xa000) + { + // A lot of 'ugly' checks... But only way to emulate this particular behaviour... + if (((gbHardware & 0xa) && ((gbLcdModeDelayed !=3) || ((register_LY == 0) && + (gbScreenOn==false) && (register_LCDC & 0x80)) && + (gbLcdLYIncrementTicksDelayed ==(GBLY_INCREMENT_CLOCK_TICKS-GBLCD_MODE_2_CLOCK_TICKS)))) || + ((gbHardware & 0x5) && (gbLcdModeDelayed !=3) && + ((gbLcdMode !=3) || ((register_LY == 0) && ((gbScreenOn==false) && + (register_LCDC & 0x80)) && + (gbLcdLYIncrementTicks ==(GBLY_INCREMENT_CLOCK_TICKS-GBLCD_MODE_2_CLOCK_TICKS)))))) + return gbMemoryMap[address>>12][address&0x0fff]; + return 0xff; + } + + if ((address >= 0xe000) && (address < 0xfe00)) + address &= ~0x2000; + + if(address < 0xc000) { +#ifndef FINAL_VERSION + if(memorydebug) { + log("Memory register read %04x PC=%04x\n", + address, + PC.W); + } +#endif + + // for the 2kb ram limit (fixes crash in shawu's story + // but now its sram test fails, as the it expects 8kb and not 2kb... + // So use the 'genericflashcard' option to fix it). + if (address<=(0xa000+gbRamSizeMask)) + { + if(mapperReadRAM) + return mapperReadRAM(address); + return gbMemoryMap[address>>12][address & 0x0fff]; + } + return 0xff; + } + + if(address >= 0xff00) { + switch(address & 0x00ff) { + case 0x00: + { + if(gbSgbMode) { + gbSgbReadingController |= 4; + gbSgbResetPacketState(); + } + + int b = gbMemory[0xff00]; + + if((b & 0x30) == 0x20) { + b &= 0xf0; + + int joy = 0; + if(gbSgbMode && gbSgbMultiplayer) { + switch(gbSgbNextController) { + case 0x0f: + joy = 0; + break; + case 0x0e: + joy = 1; + break; + case 0x0d: + joy = 2; + break; + case 0x0c: + joy = 3; + break; + default: + joy = 0; + } + } + int joystate = gbJoymask[joy]; + if(!(joystate & 128)) + b |= 0x08; + if(!(joystate & 64)) + b |= 0x04; + if(!(joystate & 32)) + b |= 0x02; + if(!(joystate & 16)) + b |= 0x01; + + gbMemory[0xff00] = b; + } else if((b & 0x30) == 0x10) { + b &= 0xf0; + + int joy = 0; + if(gbSgbMode && gbSgbMultiplayer) { + switch(gbSgbNextController) { + case 0x0f: + joy = 0; + break; + case 0x0e: + joy = 1; + break; + case 0x0d: + joy = 2; + break; + case 0x0c: + joy = 3; + break; + default: + joy = 0; + } + } + int joystate = gbJoymask[joy]; + if(!(joystate & 8)) + b |= 0x08; + if(!(joystate & 4)) + b |= 0x04; + if(!(joystate & 2)) + b |= 0x02; + if(!(joystate & 1)) + b |= 0x01; + + gbMemory[0xff00] = b; + } else { + if(gbSgbMode && gbSgbMultiplayer) { + gbMemory[0xff00] = 0xf0 | gbSgbNextController; + } else { + gbMemory[0xff00] = 0xff; + } + } + } + return gbMemory[0xff00]; + break; + case 0x01: + return gbMemory[0xff01]; + case 0x02: + return (gbMemory[0xff02]); + case 0x04: + return register_DIV; + case 0x05: + return register_TIMA; + case 0x06: + return register_TMA; + case 0x07: + return (0xf8 | register_TAC); + case 0x0f: + return (0xe0 | gbMemory[0xff0f]); + case 0x30: + case 0x31: + case 0x32: + case 0x33: + case 0x34: + case 0x35: + case 0x36: + case 0x37: + case 0x38: + case 0x39: + case 0x3A: + case 0x3B: + case 0x3C: + case 0x3D: + case 0x3E: + case 0x3F: + if ((gbMemory[NR30] & 0x80) && (gbMemory[NR34] & 0x80)) + return 0xFF; + else + return gbMemoryMap[address>>12][address & 0x0fff]; + case 0x40: + return register_LCDC; + case 0x41: + // This is a GB/C only bug (ie. not GBA/SP). + if ((gbHardware & 7) && (gbLcdMode == 2) && (gbLcdModeDelayed == 1) && (!gbSpeed)) + return (0x80 | gbMemory[0xff41] & 0xFC); + else + return (0x80 | gbMemory[0xff41]); + case 0x42: + return register_SCY; + case 0x43: + return register_SCX; + case 0x44: + if (((gbHardware & 7) && ((gbLcdMode == 1) && (gbLcdTicks == 0x71))) || + (!(register_LCDC && 0x80))) + return (0); + else + return register_LY; + case 0x45: + return register_LYC; + case 0x46: + return register_DMA; + case 0x4a: + return register_WY; + case 0x4b: + return register_WX; + case 0x4f: + return (0xfe | register_VBK); + case 0x51: + return register_HDMA1; + case 0x52: + return register_HDMA2; + case 0x53: + return register_HDMA3; + case 0x54: + return register_HDMA4; + case 0x55: + return register_HDMA5; + case 0x68: + case 0x6a: + if (gbCgbMode) + return (0x40 | gbMemory[address]); + else + return 0xc0; + case 0x69: + case 0x6b: + if (gbCgbMode) + { + // No access to gbPalette during mode 3 (Color Panel Demo) + if (((gbLcdModeDelayed != 3) && (!((gbLcdMode == 0) && (gbLcdTicks>=(GBLCD_MODE_0_CLOCK_TICKS-gbSpritesTicks[299]-1)))) && (!gbSpeed)) || + (gbSpeed && ((gbLcdMode == 1) || (gbLcdMode == 2) || + ((gbLcdMode == 3) && (gbLcdTicks>(GBLCD_MODE_3_CLOCK_TICKS-2))) || + ((gbLcdMode == 0) && (gbLcdTicks<=(GBLCD_MODE_0_CLOCK_TICKS-gbSpritesTicks[299]-2)))))) + return (gbMemory[address]); + else + return 0xff; + } + else + return 0xff; + case 0x70: + if (gbCgbMode) + return (0xf8 | register_SVBK); + else + return 0xff; + case 0xff: + return register_IE; + } + } + // OAM not accessible during mode 2 & 3. + if(((address >= 0xfe00) && (address<0xfea0)) && + (((gbLcdMode | gbLcdModeDelayed) &2) && + (!(gbSpeed && (gbHardware & 0x2) && !(gbLcdModeDelayed & 2) && (gbLcdMode == 2))) || + (gbSpeed && (gbHardware & 0x2) && (gbLcdModeDelayed == 0) && (gbLcdTicksDelayed == (GBLCD_MODE_0_CLOCK_TICKS-gbSpritesTicks[299]))))) + return 0xff; + + if ((address >= 0xfea0) && (address < 0xff00)) + { + if (gbHardware & 1) + return ((((address + ((address >> 4) - 0xfea)) >> 2) & 1) ? 0x00 : 0xff ); + else if (gbHardware & 2) + return gbMemoryMap[address>>12][address & 0x0fff]; + else if (gbHardware & 4) + return ((((address + ((address >> 4) - 0xfea)) >> 2) & 1) ? 0xff : 0x00 ); + else if (gbHardware & 8) + return ((address & 0xf0) |((address & 0xf0)>>4)); + } + + return gbMemoryMap[address>>12][address & 0x0fff]; +} + +void gbVblank_interrupt() +{ + gbCheatWrite(false); // Emulates GS codes. + gbMemory[0xff0f] = register_IF &= 0xfe; + gbWriteMemory(--SP.W, PC.B.B1); + gbWriteMemory(--SP.W, PC.B.B0); + PC.W = 0x40; +} + +void gbLcd_interrupt() +{ + gbMemory[0xff0f] = register_IF &= 0xfd; + gbWriteMemory(--SP.W, PC.B.B1); + gbWriteMemory(--SP.W, PC.B.B0); + PC.W = 0x48; +} + +void gbTimer_interrupt() +{ + gbMemory[0xff0f] = register_IF &= 0xfb; + gbWriteMemory(--SP.W, PC.B.B1); + gbWriteMemory(--SP.W, PC.B.B0); + PC.W = 0x50; +} + +void gbSerial_interrupt() +{ + gbMemory[0xff0f] = register_IF &= 0xf7; + gbWriteMemory(--SP.W, PC.B.B1); + gbWriteMemory(--SP.W, PC.B.B0); + PC.W = 0x58; +} + +void gbJoypad_interrupt() +{ + gbMemory[0xff0f] = register_IF &= 0xef; + gbWriteMemory(--SP.W, PC.B.B1); + gbWriteMemory(--SP.W, PC.B.B0); + PC.W = 0x60; +} + +void gbSpeedSwitch() +{ + gbBlackScreen = true; + if(gbSpeed == 0) { + gbSpeed = 1; + GBLCD_MODE_0_CLOCK_TICKS = 51 * 2; + GBLCD_MODE_1_CLOCK_TICKS = 1140 * 2; + GBLCD_MODE_2_CLOCK_TICKS = 20 * 2; + GBLCD_MODE_3_CLOCK_TICKS = 43 * 2; + GBLY_INCREMENT_CLOCK_TICKS = 114 * 2; + GBDIV_CLOCK_TICKS = 64; + GBTIMER_MODE_0_CLOCK_TICKS = 256; + GBTIMER_MODE_1_CLOCK_TICKS = 4; + GBTIMER_MODE_2_CLOCK_TICKS = 16; + GBTIMER_MODE_3_CLOCK_TICKS = 64; + GBSERIAL_CLOCK_TICKS = 128 * 2; + gbLcdTicks *= 2; + gbLcdTicksDelayed *=2; + gbLcdTicksDelayed--; + gbLcdLYIncrementTicks *= 2; + gbLcdLYIncrementTicksDelayed *= 2; + gbLcdLYIncrementTicksDelayed--; + gbSerialTicks *= 2; + SOUND_CLOCK_TICKS = soundQuality * 24 * 2; + soundTicks *= 2; + gbLine99Ticks = 3; + } else { + gbSpeed = 0; + GBLCD_MODE_0_CLOCK_TICKS = 51; + GBLCD_MODE_1_CLOCK_TICKS = 1140; + GBLCD_MODE_2_CLOCK_TICKS = 20; + GBLCD_MODE_3_CLOCK_TICKS = 43; + GBLY_INCREMENT_CLOCK_TICKS = 114; + GBDIV_CLOCK_TICKS = 64; + GBTIMER_MODE_0_CLOCK_TICKS = 256; + GBTIMER_MODE_1_CLOCK_TICKS = 4; + GBTIMER_MODE_2_CLOCK_TICKS = 16; + GBTIMER_MODE_3_CLOCK_TICKS = 64; + GBSERIAL_CLOCK_TICKS = 128; + gbLcdTicks >>= 1; + gbLcdTicksDelayed++; + gbLcdTicksDelayed >>=1; + gbLcdLYIncrementTicks >>= 1; + gbLcdLYIncrementTicksDelayed++; + gbLcdLYIncrementTicksDelayed >>= 1; + gbSerialTicks /= 2; + SOUND_CLOCK_TICKS = soundQuality * 24; + soundTicks /= 2; + gbLine99Ticks = 1; + if (gbHardware & 8) + gbLine99Ticks++; + } + gbDmaTicks += (134)*GBLY_INCREMENT_CLOCK_TICKS + (37<<(gbSpeed ? 1 : 0)); +} + +bool CPUIsGBBios(const char * file) +{ + if(strlen(file) > 4) { + const char * p = strrchr(file,'.'); + + if(p != NULL) { + if(_stricmp(p, ".gb") == 0) + return true; + if(_stricmp(p, ".bin") == 0) + return true; + if(_stricmp(p, ".bios") == 0) + return true; + } + } + + return false; +} + +void gbCPUInit(const char *biosFileName, bool useBiosFile) +{ + useBios = false; + if (useBiosFile) + { + int size = 0x100; + if(utilLoad(biosFileName, + CPUIsGBBios, + bios, + size)) { + if(size == 0x100) + useBios = true; + else + systemMessage(MSG_INVALID_BIOS_FILE_SIZE, N_("Invalid BOOTROM file size")); + } + } +} + +void gbGetHardwareType() +{ + gbCgbMode = 0; + gbSgbMode = 0; + if(gbRom[0x143] & 0x80) { + if((gbEmulatorType == 0) || + gbEmulatorType == 1 || + gbEmulatorType == 4) { + gbCgbMode = 1; + } + } + + if((gbCgbMode == 0 ) && (gbRom[0x146] == 0x03)) { + if(gbEmulatorType == 0 || + gbEmulatorType == 2 || + gbEmulatorType == 5) + gbSgbMode = 1; + } + + gbHardware = 1; // GB + if (((gbCgbMode == 1) && (gbEmulatorType == 0)) || (gbEmulatorType == 1)) + gbHardware = 2; // GBC + else if (((gbSgbMode == 1) && (gbEmulatorType == 0)) || (gbEmulatorType == 2) || (gbEmulatorType == 5)) + gbHardware = 4; // SGB(2) + else if (gbEmulatorType == 4) + gbHardware = 8; // GBA + + gbGBCColorType = 0; + if (gbHardware & 8) // If GBA is selected, choose the GBA default settings. + gbGBCColorType = 2; // (0 = GBC, 1 = GBA, 2 = GBASP) +} + +void gbReset() +{ + gbGetHardwareType(); + + oldRegister_WY = 146; + gbInterruptLaunched = 0; + + if(gbCgbMode == 1) { + if (gbVram == NULL) + gbVram = (u8 *)malloc(0x4000); + if (gbWram == NULL) + gbWram = (u8 *)malloc(0x8000); + memset(gbVram,0,0x4000); + memset(gbPalette,0, 2*128); + } + else + { + if(gbVram != NULL) { + free(gbVram); + gbVram = NULL; + } + if(gbWram != NULL) { + free(gbWram); + gbWram = NULL; + } + } + + gbLYChangeHappened = false; + gbLCDChangeHappened = false; + gbBlackScreen = false; + gbInterruptWait = 0; + gbDmaTicks = 0; + clockTicks = 0; + + if(gbSpeed) { + gbSpeedSwitch(); + gbMemory[0xff4d] = 0; + } + + // clean Wram + // This kinda emulates the startup state of Wram on GB/C (not very accurate, + // but way closer to the reality than filling it with 00es or FFes). + // On GBA/GBASP, it's kinda filled with random data. + // In all cases, most of the 2nd bank is filled with 00s. + // The starting data are important for some 'buggy' games, like Buster Brothers or + // Karamuchou ha Oosawagi!. + if (gbMemory != NULL) + { + memset(gbMemory,0xff, 65536); + for (int temp = 0xC000; temp < 0xE000; temp++) + if ((temp & 0x8) ^((temp & 0x800)>>8)) + { + if ((gbHardware & 0x02) && (gbGBCColorType == 0)) + gbMemory[temp] = 0x0; + else + gbMemory[temp] = 0x0f; + } + + else + gbMemory[temp] = 0xff; + } + + + + // clean LineBuffer + if (gbLineBuffer != NULL) + memset(gbLineBuffer, 0, sizeof(gbLineBuffer)); + // clean Pix + if (pix != NULL) + memset(pix, 0, sizeof(pix)); + // clean Vram + if (gbVram != NULL) + memset(gbVram, 0, 0x4000); + // clean Wram 2 + // This kinda emulates the startup state of Wram on GBC (not very accurate, + // but way closer to the reality than filling it with 00es or FFes). + // On GBA/GBASP, it's kinda filled with random data. + // In all cases, most of the 2nd bank is filled with 00s. + // The starting data are important for some 'buggy' games, like Buster Brothers or + // Karamuchou ha Oosawagi! + if (gbWram != NULL) + { + for (int i = 0; i<8; i++) + if (i != 2) + memcpy ((u16 *)(gbWram+i*0x1000), (u16 *)(gbMemory+0xC000), 0x1000); + } + + memset(gbSCYLine,0,sizeof(gbSCYLine)); + memset(gbSCXLine,0,sizeof(gbSCXLine)); + memset(gbBgpLine,0xfc,sizeof(gbBgpLine)); + if (gbHardware & 5) + { + memset(gbObp0Line,0xff,sizeof(gbObp0Line)); + memset(gbObp1Line,0xff,sizeof(gbObp1Line)); + } + else + { + memset(gbObp0Line,0x0,sizeof(gbObp0Line)); + memset(gbObp1Line,0x0,sizeof(gbObp1Line)); + } + memset(gbSpritesTicks,0x0,sizeof(gbSpritesTicks)); + + SP.W = 0xfffe; + AF.W = 0x01b0; + BC.W = 0x0013; + DE.W = 0x00d8; + HL.W = 0x014d; + PC.W = 0x0100; + IFF = 0; + gbInt48Signal = 0; + + register_TIMA = 0; + register_TMA = 0; + register_TAC = 0; + gbMemory[0xff0f] = register_IF = 1; + gbMemory[0xff40] = register_LCDC = 0x91; + gbMemory[0xff47] = 0xfc; + + if (gbCgbMode) + gbMemory[0xff4d] = 0x7e; + else + gbMemory[0xff4d] = 0xff; + + if (!gbCgbMode) + gbMemory[0xff70] = gbMemory[0xff74] = 0xff; + + if (gbCgbMode) + gbMemory[0xff56] = 0x3e; + else + gbMemory[0xff56] = 0xff; + + register_SCY = 0; + register_SCX = 0; + register_LYC = 0; + register_DMA = 0xff; + register_WY = 0; + register_WX = 0; + register_VBK = 0; + register_HDMA1 = 0xff; + register_HDMA2 = 0xff; + register_HDMA3 = 0xff; + register_HDMA4 = 0xff; + register_HDMA5 = 0xff; + register_SVBK = 0; + register_IE = 0; + + if (gbCgbMode) + gbMemory[0xff02] = 0x7c; + else + gbMemory[0xff02] = 0x7e; + + gbMemory[0xff03] = 0xff; + int i; + for (i = 0x8; i<0xf; i++) + gbMemory[0xff00+i] = 0xff; + + gbMemory[0xff13] = 0xff; + gbMemory[0xff15] = 0xff; + gbMemory[0xff18] = 0xff; + gbMemory[0xff1d] = 0xff; + gbMemory[0xff1f] = 0xff; + + for (i = 0x27; i<0x30; i++) + gbMemory[0xff00+i] = 0xff; + + gbMemory[0xff4c] = 0xff; + gbMemory[0xff4e] = 0xff; + gbMemory[0xff50] = 0xff; + + for (i = 0x57; i<0x68; i++) + gbMemory[0xff00+i] = 0xff; + + for (i = 0x5d; i<0x70; i++) + gbMemory[0xff00+i] = 0xff; + + gbMemory[0xff71] = 0xff; + + for (i = 0x78; i<0x80; i++) + gbMemory[0xff00+i] = 0xff; + + if (gbHardware & 0xa) + { + + if (gbHardware & 2) + { + AF.W = 0x1180; + BC.W = 0x0000; + } + else + { + AF.W = 0x1100; + BC.W = 0x0100; // GBA/SP have B = 0x01 (which means GBC & GBA/SP bootrom are different !) + } + + gbMemory[0xff26] = 0xf1; + if (gbCgbMode) + { + + gbMemory[0xff31] = 0xff; + gbMemory[0xff33] = 0xff; + gbMemory[0xff35] = 0xff; + gbMemory[0xff37] = 0xff; + gbMemory[0xff39] = 0xff; + gbMemory[0xff3b] = 0xff; + gbMemory[0xff3d] = 0xff; + + gbMemory[0xff44] = register_LY = 0x90; + gbDivTicks = 0x19 + ((gbHardware & 2) >> 1); + gbInternalTimer = 0x58 + ((gbHardware & 2) >> 1); + gbLcdTicks = GBLCD_MODE_1_CLOCK_TICKS - + (register_LY-0x8F)*GBLY_INCREMENT_CLOCK_TICKS + 72 + ((gbHardware & 2) >> 1); + gbLcdLYIncrementTicks = 72 + ((gbHardware & 2) >> 1); + gbMemory[0xff04] = register_DIV = 0x1E; + } + else + { + gbMemory[0xff44] = register_LY = 0x94; + gbDivTicks = 0x22 + ((gbHardware & 2) >> 1); + gbInternalTimer = 0x61 + ((gbHardware & 2) >> 1); + gbLcdTicks = GBLCD_MODE_1_CLOCK_TICKS - + (register_LY-0x8F)*GBLY_INCREMENT_CLOCK_TICKS + 25 + ((gbHardware & 2) >> 1); + gbLcdLYIncrementTicks = 25 + ((gbHardware & 2) >> 1); + gbMemory[0xff04] = register_DIV = 0x26; + } + + + DE.W = 0xff56; + HL.W = 0x000d; + + register_HDMA5 = 0xff; + gbMemory[0xff68] = 0xc0; + gbMemory[0xff6a] = 0xc0; + + + gbMemory[0xff41] = register_STAT = 0x81; + gbLcdMode = 1; + } + else + { + if (gbHardware & 4) + { + if(gbEmulatorType == 5) + AF.W = 0xffb0; + else + AF.W = 0x01b0; + BC.W = 0x0013; + DE.W = 0x00d8; + HL.W = 0x014d; + } + gbDivTicks = 14; + gbInternalTimer = gbDivTicks--; + gbMemory[0xff04] = register_DIV = 0xAB; + gbMemory[0xff41] = register_STAT = 0x85; + gbMemory[0xff44] = register_LY = 0x00; + gbLcdTicks = 15; + gbLcdLYIncrementTicks = 114+gbLcdTicks; + gbLcdMode = 1; + + // used for the handling of the gb Boot Rom + if ((gbHardware & 5) && (bios != NULL) && useBios && !skipBios) + { + memcpy ((u8 *)(gbMemory), (u8 *)(gbRom), 0x1000); + memcpy ((u8 *)(gbMemory), (u8 *)(bios), 0x100); + gbWhiteScreen = 0; + + gbInternalTimer = 0x3e; + gbDivTicks = 0x3f; + gbMemory[0xff04] = register_DIV = 0x00; + PC.W = 0x0000; + register_LCDC = 0x11; + gbScreenOn = false; + gbLcdTicks = 0; + gbLcdMode = 0; + gbLcdModeDelayed = 0; + gbMemory[0xff41] = register_STAT &= 0xfc; + gbInt48Signal = 0; + gbLcdLYIncrementTicks = GBLY_INCREMENT_CLOCK_TICKS; + } + } + + gbLine99Ticks = 1; + if (gbHardware & 8) + gbLine99Ticks++; + + gbLcdModeDelayed = gbLcdMode; + gbLcdTicksDelayed = gbLcdTicks+1; + gbLcdLYIncrementTicksDelayed = gbLcdLYIncrementTicks+1; + + + gbTimerModeChange = false; + gbTimerOnChange = false; + gbTimerOn = false; + + if(gbCgbMode) { + for (int i = 0; i<0x20; i++) + gbPalette[i] = 0x7fff; + + // This is just to show that the starting values of the OBJ palettes are different + // between the 3 consoles, and that they 'kinda' stay the same at each reset + // (they can slightly change, somehow (randomly?)). + // You can check the effects of gbGBCColorType on the "Vila Caldan Color" gbc demo. + // Note that you could also check the Div register to check on which system the game + // is running (GB,GBC and GBA(SP) have different startup values). + // Unfortunatly, I don't have any SGB system, so I can't get their starting values. + + if (gbGBCColorType == 0) // GBC Hardware + { + gbPalette[0x20] = 0x0600; + gbPalette[0x21] = 0xfdf3; + gbPalette[0x22] = 0x041c; + gbPalette[0x23] = 0xf5db; + gbPalette[0x24] = 0x4419; + gbPalette[0x25] = 0x57ea; + gbPalette[0x26] = 0x2808; + gbPalette[0x27] = 0x9b75; + gbPalette[0x28] = 0x129b; + gbPalette[0x29] = 0xfce0; + gbPalette[0x2a] = 0x22da; + gbPalette[0x2b] = 0x4ac5; + gbPalette[0x2c] = 0x2d71; + gbPalette[0x2d] = 0xf0c2; + gbPalette[0x2e] = 0x5137; + gbPalette[0x2f] = 0x2d41; + gbPalette[0x30] = 0x6b2d; + gbPalette[0x31] = 0x2215; + gbPalette[0x32] = 0xbe0a; + gbPalette[0x33] = 0xc053; + gbPalette[0x34] = 0xfe5f; + gbPalette[0x35] = 0xe000; + gbPalette[0x36] = 0xbe10; + gbPalette[0x37] = 0x914d; + gbPalette[0x38] = 0x7f91; + gbPalette[0x39] = 0x02b5; + gbPalette[0x3a] = 0x77ac; + gbPalette[0x3b] = 0x14e5; + gbPalette[0x3c] = 0xcf89; + gbPalette[0x3d] = 0xa03d; + gbPalette[0x3e] = 0xfd50; + gbPalette[0x3f] = 0x91ff; + } + else if (gbGBCColorType == 1) // GBA Hardware + { + gbPalette[0x20] = 0xbe00; + gbPalette[0x21] = 0xfdfd; + gbPalette[0x22] = 0xbd69; + gbPalette[0x23] = 0x7baf; + gbPalette[0x24] = 0xf5ff; + gbPalette[0x25] = 0x3f8f; + gbPalette[0x26] = 0xcee5; + gbPalette[0x27] = 0x5bf7; + gbPalette[0x28] = 0xb35b; + gbPalette[0x29] = 0xef97; + gbPalette[0x2a] = 0xef9f; + gbPalette[0x2b] = 0x97f7; + gbPalette[0x2c] = 0x82bf; + gbPalette[0x2d] = 0x9f3d; + gbPalette[0x2e] = 0xddde; + gbPalette[0x2f] = 0xbad5; + gbPalette[0x30] = 0x3cba; + gbPalette[0x31] = 0xdfd7; + gbPalette[0x32] = 0xedea; + gbPalette[0x33] = 0xfeda; + gbPalette[0x34] = 0xf7f9; + gbPalette[0x35] = 0xfdee; + gbPalette[0x36] = 0x6d2f; + gbPalette[0x37] = 0xf0e6; + gbPalette[0x38] = 0xf7f0; + gbPalette[0x39] = 0xf296; + gbPalette[0x3a] = 0x3bf1; + gbPalette[0x3b] = 0xe211; + gbPalette[0x3c] = 0x69ba; + gbPalette[0x3d] = 0x3d0d; + gbPalette[0x3e] = 0xdfd3; + gbPalette[0x3f] = 0xa6ba; + } + else if (gbGBCColorType == 2) // GBASP Hardware + { + gbPalette[0x20] = 0x9c00; + gbPalette[0x21] = 0x6340; + gbPalette[0x22] = 0x10c6; + gbPalette[0x23] = 0xdb97; + gbPalette[0x24] = 0x7622; + gbPalette[0x25] = 0x3e57; + gbPalette[0x26] = 0x2e12; + gbPalette[0x27] = 0x95c3; + gbPalette[0x28] = 0x1095; + gbPalette[0x29] = 0x488c; + gbPalette[0x2a] = 0x8241; + gbPalette[0x2b] = 0xde8c; + gbPalette[0x2c] = 0xfabc; + gbPalette[0x2d] = 0x0e81; + gbPalette[0x2e] = 0x7675; + gbPalette[0x2f] = 0xfdec; + gbPalette[0x30] = 0xddfd; + gbPalette[0x31] = 0x5995; + gbPalette[0x32] = 0x066a; + gbPalette[0x33] = 0xed1e; + gbPalette[0x34] = 0x1e84; + gbPalette[0x35] = 0x1d14; + gbPalette[0x36] = 0x11c3; + gbPalette[0x37] = 0x2749; + gbPalette[0x38] = 0xa727; + gbPalette[0x39] = 0x6266; + gbPalette[0x3a] = 0xe27b; + gbPalette[0x3b] = 0xe3fc; + gbPalette[0x3c] = 0x1f76; + gbPalette[0x3d] = 0xf158; + gbPalette[0x3e] = 0x468e; + gbPalette[0x3f] = 0xa540; + } + } else { + if(gbSgbMode) { + for(int i = 0; i < 8; i++) + gbPalette[i] = systemGbPalette[gbPaletteOption*8+i]; + + } + for(int i = 0; i < 8; i++) + gbPalette[i] = systemGbPalette[gbPaletteOption*8+i]; + } + + GBTIMER_MODE_0_CLOCK_TICKS = 256; + GBTIMER_MODE_1_CLOCK_TICKS = 4; + GBTIMER_MODE_2_CLOCK_TICKS = 16; + GBTIMER_MODE_3_CLOCK_TICKS = 64; + + GBLY_INCREMENT_CLOCK_TICKS = 114; + gbTimerTicks = GBTIMER_MODE_0_CLOCK_TICKS; + gbTimerClockTicks = GBTIMER_MODE_0_CLOCK_TICKS; + gbSerialTicks = 0; + gbSerialBits = 0; + gbSerialOn = 0; + gbWindowLine = -1; + gbTimerOn = false; + gbTimerMode = 0; + gbSpeed = 0; + gbJoymask[0] = gbJoymask[1] = gbJoymask[2] = gbJoymask[3] = 0; + + if(gbCgbMode) { + gbSpeed = 0; + gbHdmaOn = 0; + gbHdmaSource = 0x99d0; + gbHdmaDestination = 0x99d0; + gbVramBank = 0; + gbWramBank = 1; + + } + + // used to clean the borders + if (gbSgbMode) + { + gbSgbResetFlag = true; + gbSgbReset(); + if (gbBorderOn) + gbSgbRenderBorder(); + gbSgbResetFlag = false; + } + + for(i = 0; i < 4; i++) + gbBgp[i] = gbObp0[i] = gbObp1[i] = i; + + memset(&gbDataMBC1,0, sizeof(gbDataMBC1)); + gbDataMBC1.mapperROMBank = 1; + + gbDataMBC2.mapperRAMEnable = 0; + gbDataMBC2.mapperROMBank = 1; + + memset(&gbDataMBC3,0, 6 * sizeof(int)); + gbDataMBC3.mapperROMBank = 1; + + memset(&gbDataMBC5, 0, sizeof(gbDataMBC5)); + gbDataMBC5.mapperROMBank = 1; + + memset(&gbDataHuC1, 0, sizeof(gbDataHuC1)); + gbDataHuC1.mapperROMBank = 1; + + memset(&gbDataHuC3, 0, sizeof(gbDataHuC3)); + gbDataHuC3.mapperROMBank = 1; + + memset(&gbDataTAMA5,0, 26*sizeof(int)); + gbDataTAMA5.mapperROMBank = 1; + + memset(&gbDataMMM01,0, sizeof(gbDataMMM01)); + gbDataMMM01.mapperROMBank = 1; + + if (useBios && !skipBios && (gbHardware & 5)) + { + gbMemoryMap[0x00] = &gbMemory[0x0000]; + inBios = true; + } + else + { + gbMemoryMap[0x00] = &gbRom[0x0000]; + inBios = false; + } + + gbMemoryMap[0x01] = &gbRom[0x1000]; + gbMemoryMap[0x02] = &gbRom[0x2000]; + gbMemoryMap[0x03] = &gbRom[0x3000]; + gbMemoryMap[0x04] = &gbRom[0x4000]; + gbMemoryMap[0x05] = &gbRom[0x5000]; + gbMemoryMap[0x06] = &gbRom[0x6000]; + gbMemoryMap[0x07] = &gbRom[0x7000]; + if(gbCgbMode) { + gbMemoryMap[0x08] = &gbVram[0x0000]; + gbMemoryMap[0x09] = &gbVram[0x1000]; + gbMemoryMap[0x0a] = &gbMemory[0xa000]; + gbMemoryMap[0x0b] = &gbMemory[0xb000]; + gbMemoryMap[0x0c] = &gbMemory[0xc000]; + gbMemoryMap[0x0d] = &gbWram[0x1000]; + gbMemoryMap[0x0e] = &gbMemory[0xe000]; + gbMemoryMap[0x0f] = &gbMemory[0xf000]; + } else { + gbMemoryMap[0x08] = &gbMemory[0x8000]; + gbMemoryMap[0x09] = &gbMemory[0x9000]; + gbMemoryMap[0x0a] = &gbMemory[0xa000]; + gbMemoryMap[0x0b] = &gbMemory[0xb000]; + gbMemoryMap[0x0c] = &gbMemory[0xc000]; + gbMemoryMap[0x0d] = &gbMemory[0xd000]; + gbMemoryMap[0x0e] = &gbMemory[0xe000]; + gbMemoryMap[0x0f] = &gbMemory[0xf000]; + } + + if(gbRam) { + gbMemoryMap[0x0a] = &gbRam[0x0000]; + gbMemoryMap[0x0b] = &gbRam[0x1000]; + } + + gbSoundReset(); + + systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; + + gbLastTime = systemGetClock(); + gbFrameCount = 0; + + gbScreenOn = true; + gbSystemMessage = false; + + gbCheatWrite(true); // Emulates GS codes. + +} + +void gbWriteSaveMBC1(const char * name) +{ + if (gbRam) + { + FILE *gzFile = fopen(name,"wb"); + + if(gzFile == NULL) { + systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), name); + return; + } + + fwrite(gbRam, + 1, + (gbRamSizeMask+1), + gzFile); + + fclose(gzFile); + } +} + +void gbWriteSaveMBC2(const char * name) +{ + if (gbRam) + { + FILE *file = fopen(name, "wb"); + + if(file == NULL) { + systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), name); + return; + } + + fwrite(&gbMemory[0xa000], + 1, + 256, + file); + + fclose(file); + } +} + +void gbWriteSaveMBC3(const char * name, bool extendedSave) +{ + if (gbRam || extendedSave) + { + FILE *gzFile = fopen(name,"wb"); + if (gbRam) + { + + if(gzFile == NULL) { + systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), name); + return; + } + + fwrite(gbRam, + 1, + (gbRamSizeMask+1), + gzFile); + } + + if(extendedSave) + fwrite(&gbDataMBC3.mapperSeconds, + 1, + 10*sizeof(int) + sizeof(time_t), + gzFile); + + fclose(gzFile); + } +} + +void gbWriteSaveMBC5(const char * name) +{ + if (gbRam) + { + FILE *gzFile = fopen(name,"wb"); + + if(gzFile == NULL) { + systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), name); + return; + } + + fwrite(gbRam, + 1, + (gbRamSizeMask+1), + gzFile); + + fclose(gzFile); + } +} + +void gbWriteSaveMBC7(const char * name) +{ + if (gbRam) + { + FILE *file = fopen(name, "wb"); + + if(file == NULL) { + systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), name); + return; + } + + fwrite(&gbMemory[0xa000], + 1, + 256, + file); + + fclose(file); + } +} + +void gbWriteSaveTAMA5(const char * name, bool extendedSave) +{ + FILE *gzFile = fopen(name,"wb"); + + if(gzFile == NULL) { + systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), name); + return; + } + if (gbRam) + fwrite(gbRam, + 1, + (gbRamSizeMask+1), + gzFile); + + fwrite(gbTAMA5ram, + 1, + (gbTAMA5ramSize), + gzFile); + + if(extendedSave) + fwrite(&gbDataTAMA5.mapperSeconds, + 1, + 14*sizeof(int) + sizeof(time_t), + gzFile); + + fclose(gzFile); +} + +void gbWriteSaveMMM01(const char * name) +{ + if (gbRam) + { + FILE *gzFile = fopen(name,"wb"); + + if(gzFile == NULL) { + systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), name); + return; + } + + fwrite(gbRam, + 1, + (gbRamSizeMask+1), + gzFile); + + fclose(gzFile); + } +} + + +bool gbReadSaveMBC1(const char * name) +{ + if (gbRam) + { + gzFile gzFile = gzopen(name, "rb"); + + if(gzFile == NULL) { + return false; + } + + int read = gzread(gzFile, + gbRam, + (gbRamSizeMask+1)); + + if(read != (gbRamSizeMask+1)) { + systemMessage(MSG_FAILED_TO_READ_SGM, + N_("Battery file's size incompatible with the rom settings %s (%d).\nWarning : save of the battery file is now disabled !"), name, read); + gzclose(gzFile); + gbBatteryError = true; + return false; + } + + // Also checks if the battery file it bigger than gbRamSizeMask+1 ! + u8 data[1]; + data[0] = 0; + + read = gzread(gzFile, + data, + 1); + if(read >0) { + systemMessage(MSG_FAILED_TO_READ_SGM, + N_("Battery file's size incompatible with the rom settings %s (%d).\nWarning : save of the battery file is now disabled !"), name, read); + gzclose(gzFile); + gbBatteryError = true; + return false; + } + + gzclose(gzFile); + return true; + } + else + return false; +} + + +bool gbReadSaveMBC2(const char * name) +{ + if (gbRam) + { + FILE *file = fopen(name, "rb"); + + if(file == NULL) { + return false; + } + + size_t read = fread(&gbMemory[0xa000], + 1, + 256, + file); + + if(read != 256) { + systemMessage(MSG_FAILED_TO_READ_SGM, + N_("Battery file's size incompatible with the rom settings %s (%d).\nWarning : save of the battery file is now disabled !"), name, read); + fclose(file); + gbBatteryError = true; + return false; + } + + // Also checks if the battery file it bigger than gbRamSizeMask+1 ! + u8 data[1]; + data[0] = 0; + + read = fread(&data[0], + 1, + 1, + file); + if(read > 0) { + systemMessage(MSG_FAILED_TO_READ_SGM, + N_("Battery file's size incompatible with the rom settings %s (%d).\nWarning : save of the battery file is now disabled !"), name, read); + fclose(file); + gbBatteryError = true; + return false; + } + + fclose(file); + return true; + } + else + return false; +} + +bool gbReadSaveMBC3(const char * name) +{ + gzFile gzFile = gzopen(name, "rb"); + + if(gzFile == NULL) { + return false; + } + + int read = 0; + + if (gbRam) + read = gzread(gzFile, + gbRam, + (gbRamSizeMask+1)); + else + read = (gbRamSizeMask+1); + + + bool res = true; + + if(read != (gbRamSizeMask+1)) { + systemMessage(MSG_FAILED_TO_READ_SGM, + N_("Battery file's size incompatible with the rom settings %s (%d).\nWarning : save of the battery file is now disabled !"), name, read); + gbBatteryError = true; + res = false; + } else if ((gbRomType == 0xf) || (gbRomType == 0x10)){ + read = gzread(gzFile, + &gbDataMBC3.mapperSeconds, + sizeof(int)*10 + sizeof(time_t)); + + if(read != (sizeof(int)*10 + sizeof(time_t)) && read != 0) { + systemMessage(MSG_FAILED_TO_READ_RTC,N_("Failed to read RTC from save game %s (continuing)"), + name); + res = false; + } + else if (read == 0) + { + systemMessage(MSG_FAILED_TO_READ_RTC,N_("Failed to read RTC from save game %s (continuing)"), + name); + res = false; + } + else + { + // Also checks if the battery file it bigger than gbRamSizeMask+1+RTC ! + u8 data[1]; + data[0] = 0; + + read = gzread(gzFile, + data, + 1); + if(read >0) { + systemMessage(MSG_FAILED_TO_READ_SGM, + N_("Battery file's size incompatible with the rom settings %s (%d).\nWarning : save of the battery file is now disabled !"), name, read); + gbBatteryError = true; + res = false; + } + } + } + + gzclose(gzFile); + return res; +} + +bool gbReadSaveMBC5(const char * name) +{ + if (gbRam) + { + gzFile gzFile = gzopen(name, "rb"); + + if(gzFile == NULL) { + return false; + } + + int read = gzread(gzFile, + gbRam, + (gbRamSizeMask+1)); + + if(read != (gbRamSizeMask+1)) { + systemMessage(MSG_FAILED_TO_READ_SGM, + N_("Battery file's size incompatible with the rom settings %s (%d).\nWarning : save of the battery file is now disabled !"), name, read); + gzclose(gzFile); + gbBatteryError = true; + return false; + } + + + // Also checks if the battery file it bigger than gbRamSizeMask+1 ! + u8 data[1]; + data[0] = 0; + + read = gzread(gzFile, + data, + 1); + if(read >0) { + systemMessage(MSG_FAILED_TO_READ_SGM, + N_("Battery file's size incompatible with the rom settings %s (%d).\nWarning : save of the battery file is now disabled !"), name, read); + gzclose(gzFile); + gbBatteryError = true; + return false; + } + + gzclose(gzFile); + return true; + } + else + return false; +} + +bool gbReadSaveMBC7(const char * name) +{ + if (gbRam) + { + FILE *file = fopen(name, "rb"); + + if(file == NULL) { + return false; + } + + size_t read = fread(&gbMemory[0xa000], + 1, + 256, + file); + + if(read != 256) { + systemMessage(MSG_FAILED_TO_READ_SGM, + N_("Battery file's size incompatible with the rom settings %s (%d).\nWarning : save of the battery file is now disabled !"), name, read); + fclose(file); + gbBatteryError = true; + return false; + } + + // Also checks if the battery file it bigger than gbRamSizeMask+1 ! + u8 data[1]; + data[0] = 0; + + read = fread(&data[0], + 1, + 1, + file); + if(read > 0) { + systemMessage(MSG_FAILED_TO_READ_SGM, + N_("Battery file's size incompatible with the rom settings %s (%d).\nWarning : save of the battery file is now disabled !"), name, read); + fclose(file); + gbBatteryError = true; + return false; + } + + fclose(file); + return true; + } + else + return false; +} + +bool gbReadSaveTAMA5(const char * name) +{ + gzFile gzFile = gzopen(name, "rb"); + + if(gzFile == NULL) { + return false; + } + + int read = 0; + + if (gbRam) + read = gzread(gzFile, + gbRam, + (gbRamSizeMask+1)); + else + read = gbRamSizeMask; + + read += gzread(gzFile, + gbTAMA5ram, + gbTAMA5ramSize); + + bool res = true; + + if(read != (gbRamSizeMask+gbTAMA5ramSize+1)) { + systemMessage(MSG_FAILED_TO_READ_SGM, + N_("Battery file's size incompatible with the rom settings %s (%d).\nWarning : save of the battery file is now disabled !"), name, read); + gbBatteryError = true; + res = false; + } else { + read = gzread(gzFile, + &gbDataTAMA5.mapperSeconds, + sizeof(int)*14 + sizeof(time_t)); + + if(read != (sizeof(int)*14 + sizeof(time_t)) && read != 0) { + systemMessage(MSG_FAILED_TO_READ_RTC,N_("Failed to read RTC from save game %s (continuing)"), + name); + res = false; + } + else if (read == 0) + { + systemMessage(MSG_FAILED_TO_READ_RTC,N_("Failed to read RTC from save game %s (continuing)"), + name); + res = false; + } + else + { + // Also checks if the battery file it bigger than gbRamSizeMask+1+RTC ! + u8 data[1]; + data[0] = 0; + + read = gzread(gzFile, + data, + 1); + if(read >0) { + systemMessage(MSG_FAILED_TO_READ_SGM, + N_("Battery file's size incompatible with the rom settings %s (%d).\nWarning : save of the battery file is now disabled !"), name, read); + gbBatteryError = true; + res = false; + } + } + } + + gzclose(gzFile); + return res; +} + + +bool gbReadSaveMMM01(const char * name) +{ + if (gbRam) + { + gzFile gzFile = gzopen(name, "rb"); + + if(gzFile == NULL) { + return false; + } + + int read = gzread(gzFile, + gbRam, + (gbRamSizeMask+1)); + + if(read != (gbRamSizeMask+1)) { + systemMessage(MSG_FAILED_TO_READ_SGM, + N_("Battery file's size incompatible with the rom settings %s (%d).\nWarning : save of the battery file is now disabled !"), name, read); + gzclose(gzFile); + gbBatteryError = true; + return false; + } + + // Also checks if the battery file it bigger than gbRamSizeMask+1 ! + u8 data[1]; + data[0] = 0; + + read = gzread(gzFile, + data, + 1); + if(read >0) { + systemMessage(MSG_FAILED_TO_READ_SGM, + N_("Battery file's size incompatible with the rom settings %s (%d).\nWarning : save of the battery file is now disabled !"), name, read); + gzclose(gzFile); + gbBatteryError = true; + return false; + } + + gzclose(gzFile); + return true; + } + else + return false; +} + +void gbInit() +{ + gbGenFilter(); + gbSgbInit(); + + gbMemory = (u8 *)malloc(65536); + + pix = (u8 *)calloc(1,4*257*226); + + gbLineBuffer = (u16 *)malloc(160 * sizeof(u16)); +} + +bool gbWriteBatteryFile(const char *file, bool extendedSave) +{ + if(gbBattery) { + switch(gbRomType) { + case 0x03: + gbWriteSaveMBC1(file); + break; + case 0x06: + gbWriteSaveMBC2(file); + break; + case 0x0d: + gbWriteSaveMMM01(file); + break; + case 0x0f: + case 0x10: + gbWriteSaveMBC3(file, extendedSave); + break; + case 0x13: + case 0xfc: + gbWriteSaveMBC3(file, false); + case 0x1b: + case 0x1e: + gbWriteSaveMBC5(file); + break; + case 0x22: + gbWriteSaveMBC7(file); + break; + case 0xfd: + gbWriteSaveTAMA5(file, extendedSave); + break; + case 0xff: + gbWriteSaveMBC1(file); + break; + } + } + return true; +} + +bool gbWriteBatteryFile(const char *file) +{ + if (!gbBatteryError) + { + gbWriteBatteryFile(file, true); + return true; + } + else return false; +} + +bool gbReadBatteryFile(const char *file) +{ + bool res = false; + if(gbBattery) { + switch(gbRomType) { + case 0x03: + res = gbReadSaveMBC1(file); + break; + case 0x06: + res = gbReadSaveMBC2(file); + break; + case 0x0d: + res = gbReadSaveMMM01(file); + break; + case 0x0f: + case 0x10: + if(!gbReadSaveMBC3(file)) { + time(&gbDataMBC3.mapperLastTime); + struct tm *lt; + lt = localtime(&gbDataMBC3.mapperLastTime); + gbDataMBC3.mapperSeconds = lt->tm_sec; + gbDataMBC3.mapperMinutes = lt->tm_min; + gbDataMBC3.mapperHours = lt->tm_hour; + gbDataMBC3.mapperDays = lt->tm_yday & 255; + gbDataMBC3.mapperControl = (gbDataMBC3.mapperControl & 0xfe) | + (lt->tm_yday > 255 ? 1: 0); + res = false; + break; + } + res = true; + break; + case 0x13: + case 0xfc: + res = gbReadSaveMBC3(file); + case 0x1b: + case 0x1e: + res = gbReadSaveMBC5(file); + break; + case 0x22: + res = gbReadSaveMBC7(file); + case 0xfd: + if(!gbReadSaveTAMA5(file)) { + u8 gbDaysinMonth [12] = { 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31}; + time(&gbDataTAMA5.mapperLastTime); + struct tm *lt; + lt = localtime(&gbDataTAMA5.mapperLastTime); + gbDataTAMA5.mapperSeconds = lt->tm_sec; + gbDataTAMA5.mapperMinutes = lt->tm_min; + gbDataTAMA5.mapperHours = lt->tm_hour; + gbDataTAMA5.mapperDays = 1; + gbDataTAMA5.mapperMonths = 1; + gbDataTAMA5.mapperYears = 1970; + int days = lt->tm_yday+365*3; + while (days) + { + gbDataTAMA5.mapperDays++; + days--; + if (gbDataTAMA5.mapperDays>gbDaysinMonth[gbDataTAMA5.mapperMonths-1]) + { + gbDataTAMA5.mapperDays = 1; + gbDataTAMA5.mapperMonths++; + if (gbDataTAMA5.mapperMonths>12) + { + gbDataTAMA5.mapperMonths = 1; + gbDataTAMA5.mapperYears++; + if ((gbDataTAMA5.mapperYears & 3) == 0) + gbDaysinMonth[1] = 29; + else + gbDaysinMonth[1] = 28; + } + } + } + gbDataTAMA5.mapperControl = (gbDataTAMA5.mapperControl & 0xfe) | + (lt->tm_yday > 255 ? 1: 0); + res = false; + break; + } + res = true; + break; + case 0xff: + res = gbReadSaveMBC1(file); + break; + } + } + systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; + return res; +} + +bool gbReadGSASnapshot(const char *fileName) +{ + FILE *file = fopen(fileName, "rb"); + + if(!file) { + systemMessage(MSG_CANNOT_OPEN_FILE, N_("Cannot open file %s"), fileName); + return false; + } + + fseek(file, 0x4, SEEK_SET); + char buffer[16]; + char buffer2[16]; + fread(buffer, 1, 15, file); + buffer[15] = 0; + memcpy(buffer2, &gbRom[0x134], 15); + buffer2[15] = 0; + if(memcmp(buffer, buffer2, 15)) { + systemMessage(MSG_CANNOT_IMPORT_SNAPSHOT_FOR, + N_("Cannot import snapshot for %s. Current game is %s"), + buffer, + buffer2); + fclose(file); + return false; + } + fseek(file, 0x13, SEEK_SET); + size_t read = 0; + int toRead = 0; + switch(gbRomType) { + case 0x03: + case 0x0f: + case 0x10: + case 0x13: + case 0x1b: + case 0x1e: + case 0xff: + read = fread(gbRam, 1, (gbRamSizeMask+1), file); + toRead = (gbRamSizeMask+1); + break; + case 0x06: + case 0x22: + read = fread(&gbMemory[0xa000],1,256,file); + toRead = 256; + break; + default: + systemMessage(MSG_UNSUPPORTED_SNAPSHOT_FILE, + N_("Unsupported snapshot file %s"), + fileName); + fclose(file); + return false; + } + fclose(file); + gbReset(); + return true; +} + +variable_desc gbSaveGameStruct[] = { + { &PC.W, sizeof(u16) }, + { &SP.W, sizeof(u16) }, + { &AF.W, sizeof(u16) }, + { &BC.W, sizeof(u16) }, + { &DE.W, sizeof(u16) }, + { &HL.W, sizeof(u16) }, + { &IFF, sizeof(u8) }, + { &GBLCD_MODE_0_CLOCK_TICKS, sizeof(int) }, + { &GBLCD_MODE_1_CLOCK_TICKS, sizeof(int) }, + { &GBLCD_MODE_2_CLOCK_TICKS, sizeof(int) }, + { &GBLCD_MODE_3_CLOCK_TICKS, sizeof(int) }, + { &GBDIV_CLOCK_TICKS, sizeof(int) }, + { &GBLY_INCREMENT_CLOCK_TICKS, sizeof(int) }, + { &GBTIMER_MODE_0_CLOCK_TICKS, sizeof(int) }, + { &GBTIMER_MODE_1_CLOCK_TICKS, sizeof(int) }, + { &GBTIMER_MODE_2_CLOCK_TICKS, sizeof(int) }, + { &GBTIMER_MODE_3_CLOCK_TICKS, sizeof(int) }, + { &GBSERIAL_CLOCK_TICKS, sizeof(int) }, + { &GBSYNCHRONIZE_CLOCK_TICKS, sizeof(int) }, + { &gbDivTicks, sizeof(int) }, + { &gbLcdMode, sizeof(int) }, + { &gbLcdTicks, sizeof(int) }, + { &gbLcdLYIncrementTicks, sizeof(int) }, + { &gbTimerTicks, sizeof(int) }, + { &gbTimerClockTicks, sizeof(int) }, + { &gbSerialTicks, sizeof(int) }, + { &gbSerialBits, sizeof(int) }, + { &gbInt48Signal, sizeof(int) }, + { &gbInterruptWait, sizeof(int) }, + { &gbSynchronizeTicks, sizeof(int) }, + { &gbTimerOn, sizeof(int) }, + { &gbTimerMode, sizeof(int) }, + { &gbSerialOn, sizeof(int) }, + { &gbWindowLine, sizeof(int) }, + { &gbCgbMode, sizeof(int) }, + { &gbVramBank, sizeof(int) }, + { &gbWramBank, sizeof(int) }, + { &gbHdmaSource, sizeof(int) }, + { &gbHdmaDestination, sizeof(int) }, + { &gbHdmaBytes, sizeof(int) }, + { &gbHdmaOn, sizeof(int) }, + { &gbSpeed, sizeof(int) }, + { &gbSgbMode, sizeof(int) }, + { ®ister_DIV, sizeof(u8) }, + { ®ister_TIMA, sizeof(u8) }, + { ®ister_TMA, sizeof(u8) }, + { ®ister_TAC, sizeof(u8) }, + { ®ister_IF, sizeof(u8) }, + { ®ister_LCDC, sizeof(u8) }, + { ®ister_STAT, sizeof(u8) }, + { ®ister_SCY, sizeof(u8) }, + { ®ister_SCX, sizeof(u8) }, + { ®ister_LY, sizeof(u8) }, + { ®ister_LYC, sizeof(u8) }, + { ®ister_DMA, sizeof(u8) }, + { ®ister_WY, sizeof(u8) }, + { ®ister_WX, sizeof(u8) }, + { ®ister_VBK, sizeof(u8) }, + { ®ister_HDMA1, sizeof(u8) }, + { ®ister_HDMA2, sizeof(u8) }, + { ®ister_HDMA3, sizeof(u8) }, + { ®ister_HDMA4, sizeof(u8) }, + { ®ister_HDMA5, sizeof(u8) }, + { ®ister_SVBK, sizeof(u8) }, + { ®ister_IE , sizeof(u8) }, + { &gbBgp[0], sizeof(u8) }, + { &gbBgp[1], sizeof(u8) }, + { &gbBgp[2], sizeof(u8) }, + { &gbBgp[3], sizeof(u8) }, + { &gbObp0[0], sizeof(u8) }, + { &gbObp0[1], sizeof(u8) }, + { &gbObp0[2], sizeof(u8) }, + { &gbObp0[3], sizeof(u8) }, + { &gbObp1[0], sizeof(u8) }, + { &gbObp1[1], sizeof(u8) }, + { &gbObp1[2], sizeof(u8) }, + { &gbObp1[3], sizeof(u8) }, + { NULL, 0 } +}; + + +static bool gbWriteSaveState(gzFile gzFile) +{ + + utilWriteInt(gzFile, GBSAVE_GAME_VERSION); + + utilGzWrite(gzFile, &gbRom[0x134], 15); + + utilWriteInt(gzFile, useBios); + utilWriteInt(gzFile, inBios); + + utilWriteData(gzFile, gbSaveGameStruct); + + utilGzWrite(gzFile, &IFF, 2); + + if(gbSgbMode) { + gbSgbSaveGame(gzFile); + } + + utilGzWrite(gzFile, &gbDataMBC1, sizeof(gbDataMBC1)); + utilGzWrite(gzFile, &gbDataMBC2, sizeof(gbDataMBC2)); + utilGzWrite(gzFile, &gbDataMBC3, sizeof(gbDataMBC3)); + utilGzWrite(gzFile, &gbDataMBC5, sizeof(gbDataMBC5)); + utilGzWrite(gzFile, &gbDataHuC1, sizeof(gbDataHuC1)); + utilGzWrite(gzFile, &gbDataHuC3, sizeof(gbDataHuC3)); + utilGzWrite(gzFile, &gbDataTAMA5, sizeof(gbDataTAMA5)); + if (gbTAMA5ram != NULL) + utilGzWrite(gzFile, gbTAMA5ram, gbTAMA5ramSize); + utilGzWrite(gzFile, &gbDataMMM01, sizeof(gbDataMMM01)); + + utilGzWrite(gzFile, gbPalette, 128 * sizeof(u16)); + + utilGzWrite(gzFile, &gbMemory[0x8000], 0x8000); + + if(gbRamSize && gbRam) { + utilWriteInt(gzFile, gbRamSize); + utilGzWrite(gzFile, gbRam, gbRamSize); + } + + if(gbCgbMode) { + utilGzWrite(gzFile, gbVram, 0x4000); + utilGzWrite(gzFile, gbWram, 0x8000); + } + + gbSoundSaveGame(gzFile); + + gbCheatsSaveGame(gzFile); + + utilWriteInt(gzFile, gbLcdModeDelayed); + utilWriteInt(gzFile, gbLcdTicksDelayed); + utilWriteInt(gzFile, gbLcdLYIncrementTicksDelayed); + utilWriteInt(gzFile, gbSpritesTicks[299]); + utilWriteInt(gzFile, gbTimerModeChange); + utilWriteInt(gzFile, gbTimerOnChange); + utilWriteInt(gzFile, gbHardware); + utilWriteInt(gzFile, gbBlackScreen); + utilWriteInt(gzFile, oldRegister_WY); + utilWriteInt(gzFile, gbWindowLine); + utilWriteInt(gzFile, inUseRegister_WY); + utilWriteInt(gzFile, gbScreenOn); + return true; +} + +bool gbWriteMemSaveState(char *memory, int available) +{ + gzFile gzFile = utilMemGzOpen(memory, available, "w"); + + if(gzFile == NULL) { + return false; + } + + bool res = gbWriteSaveState(gzFile); + + long pos = utilGzMemTell(gzFile)+8; + + if(pos >= (available)) + res = false; + + utilGzClose(gzFile); + + return res; +} + +bool gbWriteSaveState(const char *name) +{ + gzFile gzFile = utilGzOpen(name,"wb"); + + if(gzFile == NULL) + return false; + + bool res = gbWriteSaveState(gzFile); + + utilGzClose(gzFile); + return res; +} + +static bool gbReadSaveState(gzFile gzFile) +{ + int version = utilReadInt(gzFile); + + if(version > GBSAVE_GAME_VERSION || version < 0) { + systemMessage(MSG_UNSUPPORTED_VB_SGM, + N_("Unsupported VisualBoy save game version %d"), version); + return false; + } + + u8 romname[20]; + + utilGzRead(gzFile, romname, 15); + + if(memcmp(&gbRom[0x134], romname, 15) != 0) { + systemMessage(MSG_CANNOT_LOAD_SGM_FOR, + N_("Cannot load save game for %s. Playing %s"), + romname, &gbRom[0x134]); + return false; + } + + + bool ub = false; + bool ib = false; + + if (version >= 11) + { + ub = utilReadInt(gzFile) ? true : false; + ib = utilReadInt(gzFile) ? true : false; + + if((ub != useBios) && (ib)) { + if(useBios) + systemMessage(MSG_SAVE_GAME_NOT_USING_BIOS, + N_("Save game is not using the BIOS files")); + else + systemMessage(MSG_SAVE_GAME_USING_BIOS, + N_("Save game is using the BIOS file")); + return false; + } + } + + gbReset(); + + inBios = ib; + + utilReadData(gzFile, gbSaveGameStruct); + + + // Correct crash when loading color gameboy save in regular gameboy type. + if (!gbCgbMode) + { + if(gbVram != NULL) { + free(gbVram); + gbVram = NULL; + } + if(gbWram != NULL) { + free(gbWram); + gbWram = NULL; + } + } + else + { + if(gbVram == NULL) + gbVram = (u8 *)malloc(0x4000); + if(gbWram == NULL) + gbWram = (u8 *)malloc(0x8000); + memset(gbVram,0,0x4000); + memset(gbPalette,0, 2*128); + } + + + + if(version >= GBSAVE_GAME_VERSION_7) { + utilGzRead(gzFile, &IFF, 2); + } + + if(gbSgbMode) { + gbSgbReadGame(gzFile, version); + } else { + gbSgbMask = 0; // loading a game at the wrong time causes no display + } + if (version<11) + utilGzRead(gzFile, &gbDataMBC1, sizeof(gbDataMBC1) - sizeof(int)); + else + utilGzRead(gzFile, &gbDataMBC1, sizeof(gbDataMBC1)); + utilGzRead(gzFile, &gbDataMBC2, sizeof(gbDataMBC2)); + if(version < GBSAVE_GAME_VERSION_4) + // prior to version 4, there was no adjustment for the time the game + // was last played, so we have less to read. This needs update if the + // structure changes again. + utilGzRead(gzFile, &gbDataMBC3, sizeof(gbDataMBC3)-sizeof(time_t)); + else + utilGzRead(gzFile, &gbDataMBC3, sizeof(gbDataMBC3)); + utilGzRead(gzFile, &gbDataMBC5, sizeof(gbDataMBC5)); + utilGzRead(gzFile, &gbDataHuC1, sizeof(gbDataHuC1)); + utilGzRead(gzFile, &gbDataHuC3, sizeof(gbDataHuC3)); + if (version>=11) + { + utilGzRead(gzFile, &gbDataTAMA5, sizeof(gbDataTAMA5)); + if (gbTAMA5ram != NULL) + utilGzRead(gzFile, gbTAMA5ram, gbTAMA5ramSize); + utilGzRead(gzFile, &gbDataMMM01, sizeof(gbDataMMM01)); + } + + if(version < GBSAVE_GAME_VERSION_5) { + utilGzRead(gzFile, pix, 256*224*sizeof(u16)); + } + memset(pix, 0, 257*226*sizeof(u32)); + + if(version < GBSAVE_GAME_VERSION_6) { + utilGzRead(gzFile, gbPalette, 64 * sizeof(u16)); + } else + utilGzRead(gzFile, gbPalette, 128 * sizeof(u16)); + + if (version < 11) + utilGzRead(gzFile, gbPalette, 128 * sizeof(u16)); + + if(version < GBSAVE_GAME_VERSION_10) { + if(!gbCgbMode && !gbSgbMode) { + for(int i = 0; i < 8; i++) + gbPalette[i] = systemGbPalette[gbPaletteOption*8+i]; + } + } + + utilGzRead(gzFile, &gbMemory[0x8000], 0x8000); + + if(gbRamSize && gbRam) { + if (version < 11) + utilGzRead(gzFile, gbRam, gbRamSize); + else + { + int ramSize = utilReadInt(gzFile); + utilGzRead(gzFile, gbRam, (gbRamSize>ramSize) ? ramSize : gbRamSize); + if (ramSize>gbRamSize) + gzseek(gzFile,ramSize-gbRamSize,SEEK_CUR); + } + } + + memset(gbSCYLine, register_SCY, sizeof(gbSCYLine)); + memset(gbSCXLine, register_SCX, sizeof(gbSCXLine)); + memset(gbBgpLine, (gbBgp[0] | (gbBgp[1]<<2) | (gbBgp[2]<<4) | + (gbBgp[3]<<6)), sizeof(gbBgpLine)); + memset(gbObp0Line, (gbObp0[0] | (gbObp0[1]<<2) | (gbObp0[2]<<4) | + (gbObp0[3]<<6)), sizeof(gbObp0Line)); + memset(gbObp1Line, (gbObp1[0] | (gbObp1[1]<<2) | (gbObp1[2]<<4) | + (gbObp1[3]<<6)), sizeof(gbObp1Line)); + memset(gbSpritesTicks, 0x0, sizeof(gbSpritesTicks)); + + if (inBios) + { + gbMemoryMap[0x00] = &gbMemory[0x0000]; + memcpy ((u8 *)(gbMemory), (u8 *)(gbRom), 0x1000); + memcpy ((u8 *)(gbMemory), (u8 *)(bios), 0x100); + } + else + gbMemoryMap[0x00] = &gbRom[0x0000]; + gbMemoryMap[0x01] = &gbRom[0x1000]; + gbMemoryMap[0x02] = &gbRom[0x2000]; + gbMemoryMap[0x03] = &gbRom[0x3000]; + gbMemoryMap[0x04] = &gbRom[0x4000]; + gbMemoryMap[0x05] = &gbRom[0x5000]; + gbMemoryMap[0x06] = &gbRom[0x6000]; + gbMemoryMap[0x07] = &gbRom[0x7000]; + gbMemoryMap[0x08] = &gbMemory[0x8000]; + gbMemoryMap[0x09] = &gbMemory[0x9000]; + gbMemoryMap[0x0a] = &gbMemory[0xa000]; + gbMemoryMap[0x0b] = &gbMemory[0xb000]; + gbMemoryMap[0x0c] = &gbMemory[0xc000]; + gbMemoryMap[0x0d] = &gbMemory[0xd000]; + gbMemoryMap[0x0e] = &gbMemory[0xe000]; + gbMemoryMap[0x0f] = &gbMemory[0xf000]; + + switch(gbRomType) { + case 0x00: + case 0x01: + case 0x02: + case 0x03: + // MBC 1 + memoryUpdateMapMBC1(); + break; + case 0x05: + case 0x06: + // MBC2 + memoryUpdateMapMBC2(); + break; + case 0x0b: + case 0x0c: + case 0x0d: + // MMM01 + memoryUpdateMapMMM01(); + break; + case 0x0f: + case 0x10: + case 0x11: + case 0x12: + case 0x13: + // MBC 3 + memoryUpdateMapMBC3(); + break; + case 0x19: + case 0x1a: + case 0x1b: + // MBC5 + memoryUpdateMapMBC5(); + break; + case 0x1c: + case 0x1d: + case 0x1e: + // MBC 5 Rumble + memoryUpdateMapMBC5(); + break; + case 0x22: + // MBC 7 + memoryUpdateMapMBC7(); + break; + case 0x56: + // GS3 + memoryUpdateMapGS3(); + break; + case 0xfd: + // TAMA5 + memoryUpdateMapTAMA5(); + break; + case 0xfe: + // HuC3 + memoryUpdateMapHuC3(); + break; + case 0xff: + // HuC1 + memoryUpdateMapHuC1(); + break; + } + + if(gbCgbMode) { + utilGzRead(gzFile, gbVram, 0x4000); + utilGzRead(gzFile, gbWram, 0x8000); + + int value = register_SVBK; + if(value == 0) + value = 1; + + gbMemoryMap[0x08] = &gbVram[register_VBK * 0x2000]; + gbMemoryMap[0x09] = &gbVram[register_VBK * 0x2000 + 0x1000]; + gbMemoryMap[0x0d] = &gbWram[value * 0x1000]; + } + + gbSoundReadGame(version, gzFile); + + if (gbCgbMode && gbSgbMode) { + gbSgbMode = 0; + } + + if(gbBorderOn && !gbSgbMask) { + gbSgbRenderBorder(); + } + + systemDrawScreen(); + + if(version > GBSAVE_GAME_VERSION_1) + gbCheatsReadGame(gzFile, version); + + if (version<11) + { + gbWriteMemory(0xff00, 0); + gbMemory[0xff04] = register_DIV; + gbMemory[0xff05] = register_TIMA; + gbMemory[0xff06] = register_TMA; + gbMemory[0xff07] = register_TAC; + gbMemory[0xff40] = register_LCDC; + gbMemory[0xff42] = register_SCY; + gbMemory[0xff43] = register_SCX; + gbMemory[0xff44] = register_LY; + gbMemory[0xff45] = register_LYC; + gbMemory[0xff46] = register_DMA; + gbMemory[0xff4a] = register_WY; + gbMemory[0xff4b] = register_WX; + gbMemory[0xff4f] = register_VBK; + gbMemory[0xff51] = register_HDMA1; + gbMemory[0xff52] = register_HDMA2; + gbMemory[0xff53] = register_HDMA3; + gbMemory[0xff54] = register_HDMA4; + gbMemory[0xff55] = register_HDMA5; + gbMemory[0xff70] = register_SVBK; + gbMemory[0xffff] = register_IE; + GBDIV_CLOCK_TICKS = 64; + + if (gbSpeed) + gbDivTicks /=2; + + if ((gbLcdMode == 0) && (register_STAT & 8)) + gbInt48Signal |= 1; + if ((gbLcdMode == 1) && (register_STAT & 0x10)) + gbInt48Signal |= 2; + if ((gbLcdMode == 2) && (register_STAT & 0x20)) + gbInt48Signal |= 4; + if ((register_LY==register_LYC) && (register_STAT & 0x40)) + gbInt48Signal |= 8; + + gbLcdLYIncrementTicks = GBLY_INCREMENT_CLOCK_TICKS; + + if (gbLcdMode == 2) + gbLcdLYIncrementTicks-=GBLCD_MODE_2_CLOCK_TICKS-gbLcdTicks; + else if (gbLcdMode == 3) + gbLcdLYIncrementTicks -=GBLCD_MODE_2_CLOCK_TICKS+GBLCD_MODE_3_CLOCK_TICKS-gbLcdTicks; + else if (gbLcdMode == 0) + gbLcdLYIncrementTicks =gbLcdTicks; + else if (gbLcdMode == 1) + { + gbLcdLYIncrementTicks = gbLcdTicks % GBLY_INCREMENT_CLOCK_TICKS; + if (register_LY == 0x99) + gbLcdLYIncrementTicks =gbLine99Ticks; + else if (register_LY == 0) + gbLcdLYIncrementTicks += GBLY_INCREMENT_CLOCK_TICKS; + } + + gbLcdModeDelayed = gbLcdMode; + gbLcdTicksDelayed = gbLcdTicks--; + gbLcdLYIncrementTicksDelayed = gbLcdLYIncrementTicks--; + gbInterruptWait = 0; + memset(gbSpritesTicks,0,sizeof(gbSpritesTicks)); + } + else + { + gbLcdModeDelayed = utilReadInt(gzFile); + gbLcdTicksDelayed = utilReadInt(gzFile); + gbLcdLYIncrementTicksDelayed = utilReadInt(gzFile); + gbSpritesTicks[299] = utilReadInt(gzFile) & 0xff; + gbTimerModeChange = (utilReadInt(gzFile) ? true : false); + gbTimerOnChange = (utilReadInt(gzFile) ? true : false); + gbHardware = utilReadInt(gzFile); + gbBlackScreen = (utilReadInt(gzFile) ? true : false); + oldRegister_WY = utilReadInt(gzFile); + gbWindowLine = utilReadInt(gzFile); + inUseRegister_WY = utilReadInt(gzFile); + gbScreenOn = (utilReadInt(gzFile) ? true : false); + } + + if (gbSpeed) + gbLine99Ticks *= 2; + + systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; + + return true; +} + +bool gbReadMemSaveState(char *memory, int available) +{ + gzFile gzFile = utilMemGzOpen(memory, available, "r"); + + bool res = gbReadSaveState(gzFile); + + utilGzClose(gzFile); + + return res; +} + +bool gbReadSaveState(const char *name) +{ + gzFile gzFile = utilGzOpen(name,"rb"); + + if(gzFile == NULL) { + return false; + } + + bool res = gbReadSaveState(gzFile); + + utilGzClose(gzFile); + + return res; +} + +bool gbWritePNGFile(const char *fileName) +{ + if(gbBorderOn) + return utilWritePNGFile(fileName, 256, 224, pix); + return utilWritePNGFile(fileName, 160, 144, pix); +} + +bool gbWriteBMPFile(const char *fileName) +{ + if(gbBorderOn) + return utilWriteBMPFile(fileName, 256, 224, pix); + return utilWriteBMPFile(fileName, 160, 144, pix); +} + +void gbCleanUp() +{ + if(gbRam != NULL) { + free(gbRam); + gbRam = NULL; + } + + if(gbRom != NULL) { + free(gbRom); + gbRom = NULL; + } + + if(gbMemory != NULL) { + free(gbMemory); + gbMemory = NULL; + } + + if(gbLineBuffer != NULL) { + free(gbLineBuffer); + gbLineBuffer = NULL; + } + + if(pix != NULL) { + free(pix); + pix = NULL; + } + + gbSgbShutdown(); + + if(gbVram != NULL) { + free(gbVram); + gbVram = NULL; + } + + if(gbWram != NULL) { + free(gbWram); + gbWram = NULL; + } + + if(gbTAMA5ram != NULL) { + free(gbTAMA5ram); + gbTAMA5ram = NULL; + } + + systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; +} + +bool gbLoadRom(const char *szFile) +{ + int size = 0; + + if(gbRom != NULL) { + gbCleanUp(); + } + + systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; + + gbRom = utilLoad(szFile, + utilIsGBImage, + NULL, + size); + if(!gbRom) + return false; + + gbRomSize = size; + + gbBatteryError = false; + + if(bios != NULL) { + free(bios); + bios = NULL; + } + bios = (u8 *)calloc(1,0x100); + + return gbUpdateSizes(); +} + +bool gbUpdateSizes() +{ + if(gbRom[0x148] > 8) { + systemMessage(MSG_UNSUPPORTED_ROM_SIZE, + N_("Unsupported rom size %02x"), gbRom[0x148]); + return false; + } + + if(gbRomSize < gbRomSizes[gbRom[0x148]]) { + gbRom = (u8 *)realloc(gbRom, gbRomSizes[gbRom[0x148]]); + for (int i = gbRomSize; igbRomSizes[gbRom[0x148]]) && (genericflashcardEnable)) + { + gbRomSize = gbRomSize>>16; + gbRom[0x148] = 0; + if (gbRomSize) + { + while (!((gbRomSize & 1) || (gbRom[0x148] == 7))) + { + gbRom[0x148]++; + gbRomSize>>=1; + } + gbRom[0x148]++; + } + gbRom = (u8 *)realloc(gbRom, gbRomSizes[gbRom[0x148]]); + } + gbRomSize = gbRomSizes[gbRom[0x148]]; + gbRomSizeMask = gbRomSizesMasks[gbRom[0x148]]; + + + // The 'genericflashcard' option allows some PD to work. + // However, the setting is dangerous (if you let in enabled + // and play a normal game, it might just break everything). + // That's why it is not saved in the emulator options. + // Also I added some checks in VBA to make sure your saves will not be + // overwritten if you wrongly enable this option for a game + // you already played (and vice-versa, ie. if you forgot to + // enable the option for a game you played with it enabled, like Shawu Story). + u8 ramsize = genericflashcardEnable ? 5 : gbRom[0x149]; + gbRom[0x149] = ramsize; + + if ((gbRom[2] == 0x6D) && (gbRom[5] == 0x47) && (gbRom[6] == 0x65) && (gbRom[7] == 0x6E) && + (gbRom[8] == 0x69) && (gbRom[9] == 0x65) && (gbRom[0xA] == 0x28) && (gbRom[0xB] == 0x54)) + { + gbCheatingDevice = 1; // GameGenie + for (int i = 0; i < 0x20; i++) // Cleans GG hardware registers + gbRom[0x4000+i] = 0; + } + else if (((gbRom[0x104] == 0x44) && (gbRom[0x156] == 0xEA) && (gbRom[0x158] == 0x7F) && + (gbRom[0x159] == 0xEA) && (gbRom[0x15B] == 0x7F)) || ((gbRom[0x165] == 0x3E) && + (gbRom[0x166] == 0xD9) && (gbRom[0x16D] == 0xE1) && (gbRom[0x16E] == 0x7F))) + gbCheatingDevice = 2; // GameShark + else gbCheatingDevice = 0; + + if(ramsize > 5) { + systemMessage(MSG_UNSUPPORTED_RAM_SIZE, + N_("Unsupported ram size %02x"), gbRom[0x149]); + return false; + } + + gbRamSize = gbRamSizes[ramsize]; + gbRamSizeMask = gbRamSizesMasks[ramsize]; + + if(gbRamSize) { + gbRam = (u8 *)malloc(gbRamSize); + memset(gbRam, 0xff, gbRamSize); + } + + + gbRomType = gbRom[0x147]; + if (genericflashcardEnable) + { + /*if (gbRomType<2) + gbRomType =3; + else if ((gbRomType == 0xc) || (gbRomType == 0xf) || (gbRomType == 0x12) || + (gbRomType == 0x16) || (gbRomType == 0x1a) || (gbRomType == 0x1d)) + gbRomType++; + else if ((gbRomType == 0xb) || (gbRomType == 0x11) || (gbRomType == 0x15) || + (gbRomType == 0x19) || (gbRomType == 0x1c)) + gbRomType+=2; + else if ((gbRomType == 0x5) || (gbRomType == 0x6)) + gbRomType = 0x1a;*/ + gbRomType = 0x1b; + } + else if (gbCheatingDevice == 1) + gbRomType = 0x55; + else if (gbCheatingDevice == 2) + gbRomType = 0x56; + + gbRom[0x147] = gbRomType; + + mapperReadRAM = NULL; + + switch(gbRomType) { + case 0x00: + case 0x01: + case 0x02: + case 0x03: + case 0x08: + case 0x09: + // MBC 1 + mapper = mapperMBC1ROM; + mapperRAM = mapperMBC1RAM; + mapperReadRAM = mapperMBC1ReadRAM; + break; + case 0x05: + case 0x06: + // MBC2 + mapper = mapperMBC2ROM; + mapperRAM = mapperMBC2RAM; + gbRamSize = 0x200; + gbRamSizeMask = 0x1ff; + break; + case 0x0b: + case 0x0c: + case 0x0d: + // MMM01 + mapper = mapperMMM01ROM; + mapperRAM = mapperMMM01RAM; + break; + case 0x0f: + case 0x10: + case 0x11: + case 0x12: + case 0x13: + case 0xfc: + // MBC 3 + mapper = mapperMBC3ROM; + mapperRAM = mapperMBC3RAM; + mapperReadRAM = mapperMBC3ReadRAM; + break; + case 0x19: + case 0x1a: + case 0x1b: + // MBC5 + mapper = mapperMBC5ROM; + mapperRAM = mapperMBC5RAM; + mapperReadRAM = mapperMBC5ReadRAM; + break; + case 0x1c: + case 0x1d: + case 0x1e: + // MBC 5 Rumble + mapper = mapperMBC5ROM; + mapperRAM = mapperMBC5RAM; + mapperReadRAM = mapperMBC5ReadRAM; + break; + case 0x22: + // MBC 7 + mapper = mapperMBC7ROM; + mapperRAM = mapperMBC7RAM; + mapperReadRAM = mapperMBC7ReadRAM; + break; + // GG (GameGenie) + case 0x55: + mapper = mapperGGROM; + break; + case 0x56: + // GS (GameShark) + mapper = mapperGS3ROM; + break; + case 0xfd: + // TAMA5 + if (gbRam!= NULL) + { + free(gbRam); + gbRam = NULL; + } + + ramsize = 3; + gbRamSize = gbRamSizes[3]; + gbRamSizeMask = gbRamSizesMasks[3]; + gbRam = (u8 *)malloc(gbRamSize); + memset(gbRam, 0x0, gbRamSize); + + gbTAMA5ramSize = 0x100; + + if (gbTAMA5ram == NULL) + gbTAMA5ram = (u8 *)malloc(gbTAMA5ramSize); + memset(gbTAMA5ram, 0x0, gbTAMA5ramSize); + + mapperRAM = mapperTAMA5RAM; + mapperReadRAM = mapperTAMA5ReadRAM; + mapperUpdateClock = memoryUpdateTAMA5Clock; + break; + case 0xfe: + // HuC3 + mapper = mapperHuC3ROM; + mapperRAM = mapperHuC3RAM; + mapperReadRAM = mapperHuC3ReadRAM; + break; + case 0xff: + // HuC1 + mapper = mapperHuC1ROM; + mapperRAM = mapperHuC1RAM; + break; + default: + systemMessage(MSG_UNKNOWN_CARTRIDGE_TYPE, + N_("Unknown cartridge type %02x"), gbRomType); + return false; + } + + switch(gbRomType) { + case 0x03: + case 0x06: + case 0x0f: + case 0x10: + case 0x13: + case 0x1b: + case 0x1d: + case 0x1e: + case 0x22: + case 0xfd: + case 0xff: + gbBattery = 1; + break; + } + + gbInit(); + + //gbReset(); + + switch(gbRomType) { + case 0x1c: + case 0x1d: + case 0x1e: + gbDataMBC5.isRumbleCartridge = 1; + } + + return true; +} + +int gbGetNextEvent (int clockTicks) +{ + if (register_LCDC & 0x80) + { + if(gbLcdTicks < clockTicks) + clockTicks = gbLcdTicks; + + if(gbLcdTicksDelayed < clockTicks) + clockTicks = gbLcdTicksDelayed; + + if(gbLcdLYIncrementTicksDelayed < clockTicks) + clockTicks = gbLcdLYIncrementTicksDelayed; + } + + if(gbLcdLYIncrementTicks < clockTicks) + clockTicks = gbLcdLYIncrementTicks; + + if(gbSerialOn && (gbSerialTicks < clockTicks)) + clockTicks = gbSerialTicks; + + if(gbTimerOn && (((gbInternalTimer) & gbTimerMask[gbTimerMode])+1 < clockTicks)) + clockTicks = ((gbInternalTimer) & gbTimerMask[gbTimerMode])+1; + + if(soundTicks && (soundTicks < clockTicks)) + clockTicks = soundTicks; + + if ((clockTicks<=0) || (gbInterruptWait)) + clockTicks = 1; + + return clockTicks; +} + +void gbDrawLine() +{ + switch(systemColorDepth) { + case 16: + { + u16 * dest = (u16 *)pix + + (gbBorderLineSkip+2) * (register_LY + gbBorderRowSkip+1) + + gbBorderColumnSkip; + for(int x = 0; x < 160; ) { + *dest++ = systemColorMap16[gbLineMix[x++]]; + *dest++ = systemColorMap16[gbLineMix[x++]]; + *dest++ = systemColorMap16[gbLineMix[x++]]; + *dest++ = systemColorMap16[gbLineMix[x++]]; + + *dest++ = systemColorMap16[gbLineMix[x++]]; + *dest++ = systemColorMap16[gbLineMix[x++]]; + *dest++ = systemColorMap16[gbLineMix[x++]]; + *dest++ = systemColorMap16[gbLineMix[x++]]; + + *dest++ = systemColorMap16[gbLineMix[x++]]; + *dest++ = systemColorMap16[gbLineMix[x++]]; + *dest++ = systemColorMap16[gbLineMix[x++]]; + *dest++ = systemColorMap16[gbLineMix[x++]]; + + *dest++ = systemColorMap16[gbLineMix[x++]]; + *dest++ = systemColorMap16[gbLineMix[x++]]; + *dest++ = systemColorMap16[gbLineMix[x++]]; + *dest++ = systemColorMap16[gbLineMix[x++]]; + } + if(gbBorderOn) + dest += gbBorderColumnSkip; + *dest++ = 0; // for filters that read one pixel more + } + break; + + case 24: + { + u8 *dest = (u8 *)pix + + 3*(gbBorderLineSkip * (register_LY + gbBorderRowSkip) + + gbBorderColumnSkip); + for(int x = 0; x < 160;) { + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; + dest+= 3; + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; + dest+= 3; + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; + dest+= 3; + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; + dest+= 3; + + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; + dest+= 3; + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; + dest+= 3; + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; + dest+= 3; + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; + dest+= 3; + + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; + dest+= 3; + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; + dest+= 3; + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; + dest+= 3; + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; + dest+= 3; + + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; + dest+= 3; + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; + dest+= 3; + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; + dest+= 3; + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; + dest+= 3; + } + } + break; + + case 32: + { + u32 * dest = (u32 *)pix + + (gbBorderLineSkip+1) * (register_LY + gbBorderRowSkip+1) + + gbBorderColumnSkip; + for(int x = 0; x < 160;) { + *dest++ = systemColorMap32[gbLineMix[x++]]; + *dest++ = systemColorMap32[gbLineMix[x++]]; + *dest++ = systemColorMap32[gbLineMix[x++]]; + *dest++ = systemColorMap32[gbLineMix[x++]]; + + *dest++ = systemColorMap32[gbLineMix[x++]]; + *dest++ = systemColorMap32[gbLineMix[x++]]; + *dest++ = systemColorMap32[gbLineMix[x++]]; + *dest++ = systemColorMap32[gbLineMix[x++]]; + + *dest++ = systemColorMap32[gbLineMix[x++]]; + *dest++ = systemColorMap32[gbLineMix[x++]]; + *dest++ = systemColorMap32[gbLineMix[x++]]; + *dest++ = systemColorMap32[gbLineMix[x++]]; + + *dest++ = systemColorMap32[gbLineMix[x++]]; + *dest++ = systemColorMap32[gbLineMix[x++]]; + *dest++ = systemColorMap32[gbLineMix[x++]]; + *dest++ = systemColorMap32[gbLineMix[x++]]; + } + } + break; + } +} + +void gbEmulate(int ticksToStop) +{ + gbRegister tempRegister; + u8 tempValue; + s8 offset; + + clockTicks = 0; + gbDmaTicks = 0; + + register int opcode = 0; + + int opcode1 = 0; + int opcode2 = 0; + bool execute = false; + + while(1) { +#ifndef FINAL_VERSION + if(systemDebug) { + if(!(IFF & 0x80)) { + if(systemDebug > 1) { + sprintf(gbBuffer,"PC=%04x AF=%04x BC=%04x DE=%04x HL=%04x SP=%04x I=%04x\n", + PC.W, AF.W, BC.W, DE.W,HL.W,SP.W,IFF); + } else { + sprintf(gbBuffer,"PC=%04x I=%02x\n", PC.W, IFF); + } + log(gbBuffer); + } + } +#endif + + u16 oldPCW = PC.W; + + if(IFF & 0x80) { + if(register_LCDC & 0x80) { + clockTicks = gbLcdTicks; + } else + clockTicks = 1000; + + clockTicks = gbGetNextEvent(clockTicks); + + /*if(gbLcdTicksDelayed < clockTicks) + clockTicks = gbLcdTicksDelayed; + + if(gbLcdLYIncrementTicksDelayed < clockTicks) + clockTicks = gbLcdLYIncrementTicksDelayed; + + if(gbLcdLYIncrementTicks < clockTicks) + clockTicks = gbLcdLYIncrementTicks; + + if(gbSerialOn && (gbSerialTicks < clockTicks)) + clockTicks = gbSerialTicks; + + if(gbTimerOn && (((gbInternalTimer) & gbTimerMask[gbTimerMode])+1 < clockTicks)) + clockTicks = ((gbInternalTimer) & gbTimerMask[gbTimerMode])+1; + + if(soundTicks && (soundTicks < clockTicks)) + clockTicks = soundTicks; + + if ((clockTicks<=0) || (gbInterruptWait)) + clockTicks = 1;*/ + + } else { + + // First we apply the clockTicks, then we execute the opcodes. + opcode1 = 0; + opcode2 = 0; + execute = true; + + opcode2 = opcode1 = opcode = gbReadOpcode(PC.W++); + + // If HALT state was launched while IME = 0 and (register_IF & register_IE & 0x1F), + // PC.W is not incremented for the first byte of the next instruction. + if (IFF & 2) + { + PC.W--; + IFF &= ~2; + } + + clockTicks = gbCycles[opcode]; + + switch(opcode) { + case 0xCB: + // extended opcode + opcode2 = opcode = gbReadOpcode(PC.W++); + clockTicks = gbCyclesCB[opcode]; + break; + } + gbOldClockTicks = clockTicks-1; + gbIntBreak = 1; + } + + + if(!emulating) + return; + + // For 'breakpoint' support (opcode 0xFC is considered as a breakpoint) + if ((clockTicks==0) && execute) + { + PC.W = oldPCW; + return; + } + + + if (!(IFF & 0x80)) + clockTicks = 1; + + gbRedoLoop: + + + + if (gbInterruptWait) + gbInterruptWait = 0; + else + gbInterruptLaunched = 0; + + + // Used for the EI/DI instruction's delay. + if (IFF & 0x38) + { + int tempIFF = (IFF >> 4) & 3; + + if (tempIFF <=clockTicks) + { + tempIFF = 0; + IFF |=1; + } + else + tempIFF -= clockTicks; + IFF = (IFF & 0xCF) | (tempIFF <<4); + + if (IFF & 0x08) + IFF &= 0x82; + } + + + if (register_LCDCBusy) + { + register_LCDCBusy-=clockTicks; + if (register_LCDCBusy<0) + register_LCDCBusy = 0; + } + + + if(gbSgbMode) { + if(gbSgbPacketTimeout) { + gbSgbPacketTimeout -= clockTicks; + + if(gbSgbPacketTimeout <= 0) + gbSgbResetPacketState(); + } + } + + ticksToStop -= clockTicks; + + // DIV register emulation + gbDivTicks -= clockTicks; + while(gbDivTicks <= 0) { + gbMemory[0xff04] = ++register_DIV; + gbDivTicks += GBDIV_CLOCK_TICKS; + } + + if(register_LCDC & 0x80) { + // LCD stuff + + gbLcdTicks -= clockTicks; + gbLcdTicksDelayed -= clockTicks; + gbLcdLYIncrementTicks -= clockTicks; + gbLcdLYIncrementTicksDelayed -= clockTicks; + + + // our counters are off, see what we need to do + + // This looks (and kinda is) complicated, however this + // is the only way I found to emulate properly the way + // the real hardware operates... + while(((gbLcdTicks <= 0) && (gbLCDChangeHappened == false)) || + ((gbLcdTicksDelayed <= 0) && (gbLCDChangeHappened == true)) || + ((gbLcdLYIncrementTicks <= 0) && (gbLYChangeHappened == false)) || + ((gbLcdLYIncrementTicksDelayed<=0) && (gbLYChangeHappened == true))) + { + + if ((gbLcdLYIncrementTicks <= 0) && (!gbLYChangeHappened)) + { + gbLYChangeHappened = true; + gbMemory[0xff44] = register_LY = (register_LY + 1) % 154; + + if (register_LY == 0x91) + { + /* if (IFF & 0x80) + gbScreenOn = !gbScreenOn; + else*/ if (register_LCDC & 0x80) + gbScreenOn = true; + } + + gbLcdLYIncrementTicks += GBLY_INCREMENT_CLOCK_TICKS; + + if (gbLcdMode == 1) + { + + if(register_LY == 153) + gbLcdLYIncrementTicks -= GBLY_INCREMENT_CLOCK_TICKS - gbLine99Ticks; + else if(register_LY == 0) + gbLcdLYIncrementTicks += GBLY_INCREMENT_CLOCK_TICKS - gbLine99Ticks; + } + + // GB only 'bug' : Halt state is broken one tick before LY==LYC interrupt + // is reflected in the registers. + if ((gbHardware & 5) && (IFF & 0x80) && (register_LY == register_LYC) + && (register_STAT & 0x40) && (register_LY != 0)) + { + if (!((gbLcdModeDelayed != 1) && (register_LY==0))) + { + gbInt48Signal &= ~9; + gbCompareLYToLYC(); + gbLYChangeHappened = false; + gbMemory[0xff41] = register_STAT; + gbMemory[0xff0f] = register_IF; + } + + gbLcdLYIncrementTicksDelayed += GBLY_INCREMENT_CLOCK_TICKS - gbLine99Ticks+1; + } + } + + + if ((gbLcdTicks <= 0) && (!gbLCDChangeHappened)) + { + gbLCDChangeHappened = true; + + switch(gbLcdMode) + { + case 0: + { + // H-Blank + // check if we reached the V-Blank period + if(register_LY == 144) { + // Yes, V-Blank + // set the LY increment counter + if (gbHardware & 0x5) + { + register_IF |= 1; // V-Blank interrupt + } + + gbInt48Signal &= ~6; + if(register_STAT & 0x10) + { + // send LCD interrupt only if no interrupt 48h signal... + if ((!(gbInt48Signal & 1)) && ((!(gbInt48Signal & 8)) || (gbHardware & 0x0a))) + { + register_IF |=2; + gbInterruptLaunched |= 2; + if (gbHardware & 0xa) + gbInterruptWait = 1; + } + gbInt48Signal |= 2; + } + gbInt48Signal &= ~1; + + gbLcdTicks += GBLCD_MODE_1_CLOCK_TICKS; + gbLcdMode = 1; + + } else { + // go the the OAM being accessed mode + gbLcdTicks += GBLCD_MODE_2_CLOCK_TICKS; + gbLcdMode = 2; + + gbInt48Signal &= ~6; + if(register_STAT & 0x20) + { + // send LCD interrupt only if no interrupt 48h signal... + if (!gbInt48Signal) + { + register_IF |= 2; + gbInterruptLaunched |= 2; + } + gbInt48Signal |= 4; + } + gbInt48Signal &= ~1; + } + } + break; + case 1: + { + // V-Blank + // next mode is OAM being accessed mode + gbInt48Signal &= ~5; + if(register_STAT & 0x20) + { + // send LCD interrupt only if no interrupt 48h signal... + if (!gbInt48Signal) + { + register_IF |= 2; + gbInterruptLaunched |= 2; + if ((gbHardware & 0xa) && (IFF & 0x80)) + gbInterruptWait = 1; + } + gbInt48Signal |= 4; + } + gbInt48Signal &= ~2; + + gbLcdTicks += GBLCD_MODE_2_CLOCK_TICKS; + + gbLcdMode = 2; + register_LY = 0x00; + + } + break; + case 2: + { + + // OAM being accessed mode + // next mode is OAM and VRAM in use + if ((gbScreenOn) && (register_LCDC & 0x80)) + { + gbDrawSprites(false); + // Used to add a one tick delay when a window line is drawn. + //(fixes a part of Carmaggedon problem) + if((register_LCDC & 0x01 || gbCgbMode) && (register_LCDC & 0x20) && + (gbWindowLine != -2)) { + + int inUseRegister_WY = 0; + int tempgbWindowLine = gbWindowLine; + + if ((tempgbWindowLine == -1) || (tempgbWindowLine>144)) + { + inUseRegister_WY = oldRegister_WY; + if (register_LY>oldRegister_WY) + tempgbWindowLine = 146; + } + + int wy = inUseRegister_WY; + + if(register_LY >= inUseRegister_WY) { + + if (tempgbWindowLine == -1) + tempgbWindowLine = 0; + + int wx = register_WX; + wx -= 7; + if (wx<0) + wx = 0; + + if((wx <= 159) && (tempgbWindowLine <= 143)) + for (int i = wx; i<300; i++) + if (gbSpeed) + gbSpritesTicks[i]+=3; + else + gbSpritesTicks[i]+=1; + } + } + } + + gbInt48Signal &= ~7; + + gbLcdTicks += GBLCD_MODE_3_CLOCK_TICKS+gbSpritesTicks[299]; + gbLcdMode = 3; + } + break; + case 3: + { + // OAM and VRAM in use + // next mode is H-Blank + + + gbInt48Signal &= ~7; + if(register_STAT & 0x08) + { + // send LCD interrupt only if no interrupt 48h signal... + if (!(gbInt48Signal & 8)) + { + register_IF |= 2; + if ((gbHardware & 0xa) && (IFF & 0x80)) + gbInterruptWait = 1; + } + gbInt48Signal |= 1; + } + + gbLcdTicks += GBLCD_MODE_0_CLOCK_TICKS-gbSpritesTicks[299]; + + gbLcdMode = 0; + + // No HDMA during HALT ! + if(gbHdmaOn && (!(IFF & 0x80) || (register_IE & register_IF & 0x1f))) { + gbDoHdma(); + } + + } + break; + } + } + + + if ((gbLcdTicksDelayed <= 0) && (gbLCDChangeHappened)) { + int framesToSkip = systemFrameSkip; + if(speedup) + framesToSkip = 9; // try 6 FPS during speedup + //gbLcdTicksDelayed = gbLcdTicks+1; + gbLCDChangeHappened = false; + switch(gbLcdModeDelayed) { + case 0: + { + // H-Blank + + memset(gbSCYLine,gbSCYLine[299],sizeof(gbSCYLine)); + memset(gbSCXLine,gbSCXLine[299],sizeof(gbSCXLine)); + memset(gbBgpLine,gbBgpLine[299],sizeof(gbBgpLine)); + memset(gbObp0Line,gbObp0Line[299],sizeof(gbObp0Line)); + memset(gbObp1Line,gbObp1Line[299],sizeof(gbObp1Line)); + memset(gbSpritesTicks,gbSpritesTicks[299],sizeof(gbSpritesTicks)); + + if (gbWindowLine <0) + oldRegister_WY = register_WY; + // check if we reached the V-Blank period + if(register_LY == 144) { + // Yes, V-Blank + // set the LY increment counter + + if(register_LCDC & 0x80) { + if (gbHardware & 0xa) + { + + register_IF |= 1; // V-Blank interrupt + gbInterruptLaunched |=1; + } + + + } + + gbLcdTicksDelayed += GBLCD_MODE_1_CLOCK_TICKS; + gbLcdModeDelayed = 1; + + gbFrameCount++; + systemFrame(); + + if((gbFrameCount % 10) == 0) + system10Frames(60); + + if(gbFrameCount >= 60) { + u32 currentTime = systemGetClock(); + if(currentTime != gbLastTime) + systemShowSpeed(100000/(currentTime - gbLastTime)); + else + systemShowSpeed(0); + gbLastTime = currentTime; + gbFrameCount = 0; + } + + if(systemReadJoypads()) { + // read joystick + if(gbSgbMode && gbSgbMultiplayer) { + if(gbSgbFourPlayers) { + gbJoymask[0] = systemReadJoypad(0); + gbJoymask[1] = systemReadJoypad(1); + gbJoymask[2] = systemReadJoypad(2); + gbJoymask[3] = systemReadJoypad(3); + } else { + gbJoymask[0] = systemReadJoypad(0); + gbJoymask[1] = systemReadJoypad(1); + } + } else { + gbJoymask[0] = systemReadJoypad(-1); + } + } + int newmask = gbJoymask[0] & 255; + + if(gbRomType == 0x22) { + systemUpdateMotionSensor(); + } + + if(newmask) + { + gbMemory[0xff0f] |= 16; + } + + + newmask = (gbJoymask[0] >> 10); + + speedup = (newmask & 1) ? true : false; + gbCapture = (newmask & 2) ? true : false; + + if(gbCapture && !gbCapturePrevious) { + gbCaptureNumber++; + systemScreenCapture(gbCaptureNumber); + } + gbCapturePrevious = gbCapture; + + if(gbFrameSkipCount >= framesToSkip) { + + if(!gbSgbMask) + { + if (gbBorderOn) + gbSgbRenderBorder(); + //if (gbScreenOn) + systemDrawScreen(); + } + gbFrameSkipCount = 0; + } else + gbFrameSkipCount++; + + } else { + // go the the OAM being accessed mode + gbLcdTicksDelayed += GBLCD_MODE_2_CLOCK_TICKS; + gbLcdModeDelayed = 2; + gbInt48Signal &= ~3; + } + } + break; + case 1: + { + // V-Blank + // next mode is OAM being accessed mode + + // gbScreenOn = true; + + oldRegister_WY = register_WY; + + gbLcdTicksDelayed += GBLCD_MODE_2_CLOCK_TICKS; + gbLcdModeDelayed = 2; + + // reset the window line + gbWindowLine = -1; + } + break; + case 2: + { + // OAM being accessed mode + // next mode is OAM and VRAM in use + gbLcdTicksDelayed += GBLCD_MODE_3_CLOCK_TICKS+gbSpritesTicks[299]; + gbLcdModeDelayed = 3; + } + break; + case 3: + { + + // OAM and VRAM in use + // next mode is H-Blank + if((register_LY < 144) && (register_LCDC & 0x80) && gbScreenOn) { + if(!gbSgbMask) { + if(gbFrameSkipCount >= framesToSkip) { + if (!gbBlackScreen) + { + gbRenderLine(); + gbDrawSprites(true); + } + else if (gbBlackScreen) + { + u16 color = gbColorOption ? gbColorFilter[0] : 0; + if (!gbCgbMode) + color = gbColorOption ? gbColorFilter[gbPalette[3] & 0x7FFF] : + gbPalette[3] & 0x7FFF; + for(int i = 0; i < 160; i++) + { + gbLineMix[i] = color; + gbLineBuffer[i] = 0; + } + } + gbDrawLine(); + } + } + } + gbLcdTicksDelayed += GBLCD_MODE_0_CLOCK_TICKS-gbSpritesTicks[299]; + gbLcdModeDelayed = 0; + } + break; + } + } + + if ((gbLcdLYIncrementTicksDelayed <= 0) && (gbLYChangeHappened == true)) + { + + gbLYChangeHappened = false; + + if (!((gbLcdMode != 1) && (register_LY==0))) + { + { + gbInt48Signal &= ~8; + gbCompareLYToLYC(); + if ((gbInt48Signal == 8) && (!((register_LY == 0) && (gbHardware & 1)))) + gbInterruptLaunched |= 2; + if ((gbHardware & (gbSpeed ? 8 : 2)) && (register_LY==0) && ((register_STAT & 0x44) == 0x44) && (gbLcdLYIncrementTicksDelayed==0)) + { + gbInterruptWait = 1; + + } + } + } + gbLcdLYIncrementTicksDelayed += GBLY_INCREMENT_CLOCK_TICKS; + + if (gbLcdModeDelayed == 1) + { + + if(register_LY == 153) + gbLcdLYIncrementTicksDelayed -= GBLY_INCREMENT_CLOCK_TICKS - gbLine99Ticks; + else if(register_LY == 0) + gbLcdLYIncrementTicksDelayed += GBLY_INCREMENT_CLOCK_TICKS - gbLine99Ticks; + } + gbMemory[0xff0f] = register_IF; + gbMemory[0xff41] = register_STAT; + } + } + gbMemory[0xff0f] = register_IF; + gbMemory[0xff41] = register_STAT = (register_STAT & 0xfc) | gbLcdModeDelayed; + } + else + { + + // Used to update the screen with white lines when it's off. + // (it looks strange, but it's kinda accurate :p) + // You can try the Mario Demo Vx.x for exemple + // (check the bottom 2 lines while moving) + if (!gbWhiteScreen) + { + gbScreenTicks -= clockTicks; + gbLcdLYIncrementTicks -= clockTicks; + while (gbLcdLYIncrementTicks <=0) + { + register_LY = ((register_LY+1)%154); + gbLcdLYIncrementTicks+=GBLY_INCREMENT_CLOCK_TICKS; + } + if (gbScreenTicks <= 0) + { + gbWhiteScreen = 1; + u8 register_LYLcdOff = ((register_LY+154)%154); + for (register_LY=0;register_LY <= 0x90;register_LY++) + { + u16 color = gbColorOption ? gbColorFilter[0x7FFF] : + 0x7FFF; + if (!gbCgbMode) + color = gbColorOption ? gbColorFilter[gbPalette[0] & 0x7FFF] : + gbPalette[0] & 0x7FFF; + for(int i = 0; i < 160; i++) + { + gbLineMix[i] = color; + gbLineBuffer[i] = 0; + } + gbDrawLine(); + } + register_LY = register_LYLcdOff; + } + } + + if (gbWhiteScreen) + { + gbLcdLYIncrementTicks -= clockTicks; + + while (gbLcdLYIncrementTicks <=0) + { + register_LY = ((register_LY+1)%154); + gbLcdLYIncrementTicks+=GBLY_INCREMENT_CLOCK_TICKS; + if (register_LY<144) + { + + u16 color = gbColorOption ? gbColorFilter[0x7FFF] : + 0x7FFF; + if (!gbCgbMode) + color = gbColorOption ? gbColorFilter[gbPalette[0] & 0x7FFF] : + gbPalette[0] & 0x7FFF; + for(int i = 0; i < 160; i++) + { + gbLineMix[i] = color; + gbLineBuffer[i] = 0; + } + gbDrawLine(); + } + else if ((register_LY==144) && (!systemFrameSkip)) + { + int framesToSkip = systemFrameSkip; + if(speedup) + framesToSkip = 9; // try 6 FPS during speedup + if((gbFrameSkipCount >= framesToSkip) || (gbWhiteScreen == 1)) { + gbWhiteScreen = 2; + + if(!gbSgbMask) + { + if (gbBorderOn) + gbSgbRenderBorder(); + //if (gbScreenOn) + systemDrawScreen(); + } + } + if(systemReadJoypads()) { + // read joystick + if(gbSgbMode && gbSgbMultiplayer) { + if(gbSgbFourPlayers) { + gbJoymask[0] = systemReadJoypad(0); + gbJoymask[1] = systemReadJoypad(1); + gbJoymask[2] = systemReadJoypad(2); + gbJoymask[3] = systemReadJoypad(3); + } else { + gbJoymask[0] = systemReadJoypad(0); + gbJoymask[1] = systemReadJoypad(1); + } + } else { + gbJoymask[0] = systemReadJoypad(-1); + } + } + gbFrameCount++; + + systemFrame(); + + if((gbFrameCount % 10) == 0) + system10Frames(60); + + if(gbFrameCount >= 60) { + u32 currentTime = systemGetClock(); + if(currentTime != gbLastTime) + systemShowSpeed(100000/(currentTime - gbLastTime)); + else + systemShowSpeed(0); + gbLastTime = currentTime; + gbFrameCount = 0; + } + } + } + } + } + + gbMemory[0xff41] = register_STAT; + + // serial emulation + if(gbSerialOn) { +#ifdef LINK_EMULATION + if(linkConnected) { + gbSerialTicks -= clockTicks; + + while(gbSerialTicks <= 0) { + // increment number of shifted bits + gbSerialBits++; + linkProc(); + if(gbSerialOn && (gbMemory[0xff02] & 1)) { + if(gbSerialBits == 8) { + gbSerialBits = 0; + gbMemory[0xff01] = 0xff; + gbMemory[0xff02] &= 0x7f; + gbSerialOn = 0; + gbMemory[0xff0f] = register_IF |= 8; + gbSerialTicks = 0; + } + } + gbSerialTicks += GBSERIAL_CLOCK_TICKS; + } + } else { +#endif + if(gbMemory[0xff02] & 1) { + gbSerialTicks -= clockTicks; + + // overflow + while(gbSerialTicks <= 0) { + // shift serial byte to right and put a 1 bit in its place + // gbMemory[0xff01] = 0x80 | (gbMemory[0xff01]>>1); + // increment number of shifted bits + gbSerialBits++; + if(gbSerialBits == 8) { + // end of transmission + if(gbSerialFunction) // external device + gbMemory[0xff01] = gbSerialFunction(gbMemory[0xff01]); + else + gbMemory[0xff01] = 0xff; + gbSerialTicks = 0; + gbMemory[0xff02] &= 0x7f; + gbSerialOn = 0; + gbMemory[0xff0f] = register_IF |= 8; + gbSerialBits = 0; + } else + gbSerialTicks += GBSERIAL_CLOCK_TICKS; + } + } +#ifdef LINK_EMULATION + } +#endif + } + + + soundTicks -= clockTicks; + + while(soundTicks < 0) { + soundTicks += SOUND_CLOCK_TICKS; + + gbSoundTick(); + } + + + // timer emulation + + if(gbTimerOn) { + gbTimerTicks= ((gbInternalTimer) & gbTimerMask[gbTimerMode])+1-clockTicks; + + while(gbTimerTicks <= 0) { + register_TIMA++; + // timer overflow! + if((register_TIMA & 0xff) == 0) { + // reload timer modulo + register_TIMA = register_TMA; + // flag interrupt + gbMemory[0xff0f] = register_IF |= 4; + } + gbTimerTicks += gbTimerClockTicks; + } + gbTimerOnChange = false; + gbTimerModeChange = false; + + gbMemory[0xff05] = register_TIMA; + + } + + gbInternalTimer -= clockTicks; + while (gbInternalTimer<0) + gbInternalTimer+=0x100; + + /* + if(soundOffFlag) { + if(synchronize && !speedup) { + synchronizeTicks -= clockTicks; + + while(synchronizeTicks < 0) { + synchronizeTicks += SYNCHRONIZE_CLOCK_TICKS; + + DWORD now = timeGetTime(); + gbElapsedTime += (now - timeNow); + + if(gbElapsedTime < 50) { + DWORD diff = 50 - gbElapsedTime; + Sleep(diff); + timeNow = timeGetTime(); + elapsedTime = timeNow - now - diff; + if((int)elapsedTime < 0) + elapsedTime = 0; + } else { + timeNow = timeGetTime(); + elapsedTime = 0; + } + } + } + } + */ + + clockTicks = 0; + + if (gbIntBreak == 1) + { + gbIntBreak = 0; + if ((register_IE & register_IF & gbInterruptLaunched & 0x3) && + ((IFF & 0x81) == 1) && (!gbInterruptWait) && (execute)) + { + gbIntBreak = 2; + PC.W = oldPCW; + execute = false; + gbOldClockTicks = 0; + } + if (gbOldClockTicks) + { + clockTicks = gbOldClockTicks; + gbOldClockTicks = 0; + goto gbRedoLoop; + } + } + + // Executes the opcode(s), and apply the instruction's remaining clockTicks (if any). + if (execute) + { + switch(opcode1) { + case 0xCB: + // extended opcode + switch(opcode2) { +#include "gbCodesCB.h" + } + break; +#include "gbCodes.h" + } + execute = false; + + if (clockTicks) + { + gbDmaTicks += clockTicks; + clockTicks = 0; + } + } + + if (gbDmaTicks) + { + clockTicks = gbGetNextEvent(gbDmaTicks); + + if (clockTicks<=gbDmaTicks) + gbDmaTicks -= clockTicks; + else + { + clockTicks = gbDmaTicks; + gbDmaTicks = 0; + } + + goto gbRedoLoop; + } + + + // Remove the 'if an IE is pending' flag if IE has finished being executed. + if ((IFF & 0x40) && !(IFF & 0x30)) + IFF &= 0x81; + + + + if ((register_IE & register_IF & 0x1f) && (IFF & 0x81) && (!gbInterruptWait)) + { + + if (IFF & 1) + { + // Add 5 ticks for the interrupt execution time + gbDmaTicks += 5; + + if (gbIntBreak == 2) + { + gbDmaTicks--; + gbIntBreak = 0; + } + + + if(register_IF & register_IE & 1) + gbVblank_interrupt(); + else if(register_IF & register_IE & 2) + gbLcd_interrupt(); + else if(register_IF & register_IE & 4) + gbTimer_interrupt(); + else if(register_IF & register_IE & 8) + gbSerial_interrupt(); + else if(register_IF & register_IE & 16) + gbJoypad_interrupt(); + } + + IFF &= ~0x81; + } + + if (IFF & 0x08) + IFF &=~0x79; + + // Used to apply the interrupt's execution time. + if (gbDmaTicks) + { + clockTicks = gbGetNextEvent(gbDmaTicks); + + if (clockTicks<=gbDmaTicks) + gbDmaTicks -= clockTicks; + else + { + clockTicks = gbDmaTicks; + gbDmaTicks = 0; + } + goto gbRedoLoop; + } + + + gbBlackScreen = false; + + if((ticksToStop <= 0)) { + if(!(register_LCDC & 0x80)) { + if(systemReadJoypads()) { + // read joystick + if(gbSgbMode && gbSgbMultiplayer) { + if(gbSgbFourPlayers) { + gbJoymask[0] = systemReadJoypad(0); + gbJoymask[1] = systemReadJoypad(1); + gbJoymask[2] = systemReadJoypad(2); + gbJoymask[3] = systemReadJoypad(3); + } else { + gbJoymask[0] = systemReadJoypad(0); + gbJoymask[1] = systemReadJoypad(1); + } + } else { + gbJoymask[0] = systemReadJoypad(-1); + } + } + } + return; + } + } +} + +struct EmulatedSystem GBSystem = { + // emuMain + gbEmulate, + // emuReset + gbReset, + // emuCleanUp + gbCleanUp, + // emuReadBattery + gbReadBatteryFile, + // emuWriteBattery + gbWriteBatteryFile, + // emuReadState + gbReadSaveState, + // emuWriteState + gbWriteSaveState, + // emuReadMemState + gbReadMemSaveState, + // emuWriteMemState + gbWriteMemSaveState, + // emuWritePNG + gbWritePNGFile, + // emuWriteBMP + gbWriteBMPFile, + // emuUpdateCPSR + NULL, + // emuHasDebugger + false, + // emuCount +#ifdef FINAL_VERSION + 70000/4, +#else + 1000, +#endif +}; diff --git a/src/gb/gb.h b/src/gb/gb.h index 1527c77a..26099f14 100644 --- a/src/gb/gb.h +++ b/src/gb/gb.h @@ -1,7 +1,7 @@ // -*- C++ -*- // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005-2006 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -38,11 +38,15 @@ typedef union { extern bool gbLoadRom(const char *); extern void gbEmulate(int); +extern void gbWriteMemory(register u16, register u8); +extern void gbDrawLine(); extern bool gbIsGameboyRom(const char *); extern void gbSoundReset(); extern void gbSoundSetQuality(int); +extern void gbGetHardwareType(); extern void gbReset(); extern void gbCleanUp(); +extern void gbCPUInit(const char *,bool); extern bool gbWriteBatteryFile(const char *); extern bool gbWriteBatteryFile(const char *, bool); extern bool gbReadBatteryFile(const char *); diff --git a/src/gb/gbCheats.cpp b/src/gb/gbCheats.cpp index 941bb166..62a7ccb3 100644 --- a/src/gb/gbCheats.cpp +++ b/src/gb/gbCheats.cpp @@ -1,6 +1,6 @@ // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -27,9 +27,11 @@ #include "gbCheats.h" #include "gbGlobals.h" +#include "GB.h" gbCheat gbCheatList[100]; int gbCheatNumber = 0; +int gbNextCheat = 0; bool gbCheatMap[0x10000]; extern bool cheatsEnabled; @@ -50,7 +52,7 @@ void gbCheatUpdateMap() void gbCheatsSaveGame(gzFile gzFile) { utilWriteInt(gzFile, gbCheatNumber); - if(gbCheatNumber) + if(gbCheatNumber>0) utilGzWrite(gzFile, &gbCheatList[0], sizeof(gbCheat)*gbCheatNumber); } @@ -81,7 +83,7 @@ void gbCheatsReadGame(gzFile gzFile, int version) } else { gbCheatNumber = utilReadInt(gzFile); - if(gbCheatNumber) { + if(gbCheatNumber>0) { utilGzRead(gzFile, &gbCheatList[0], sizeof(gbCheat)*gbCheatNumber); } } @@ -163,7 +165,7 @@ bool gbCheatsLoadCheatList(const char *file) bool gbVerifyGsCode(const char *code) { - int len = strlen(code); + size_t len = strlen(code); if(len == 0) return true; @@ -180,10 +182,6 @@ bool gbVerifyGsCode(const char *code) GBCHEAT_HEX_VALUE(code[4]) << 4 | GBCHEAT_HEX_VALUE(code[5]); - if(address < 0xa000 || - address > 0xdfff) - return false; - return true; } @@ -220,15 +218,23 @@ void gbAddGsCheat(const char *code, const char *desc) gbCheatList[i].compare = 0; gbCheatList[i].enabled = true; - - gbCheatMap[gbCheatList[i].address] = true; - + + int gsCode = gbCheatList[i].code; + + if ((gsCode !=1) && ((gsCode & 0xF0) !=0x80) && ((gsCode & 0xF0) !=0x90) && + ((gsCode & 0xF0) !=0xA0) && ((gsCode) !=0xF0) && ((gsCode) !=0xF1)) + systemMessage(MSG_WRONG_GAMESHARK_CODE, + N_("Wrong GameShark code type : %s"), code); + else if (((gsCode & 0xF0) ==0xA0) || ((gsCode) ==0xF0) || ((gsCode) ==0xF1)) + systemMessage(MSG_UNSUPPORTED_GAMESHARK_CODE, + N_("Unsupported GameShark code type : %s"), code); + gbCheatNumber++; } bool gbVerifyGgCode(const char *code) { - int len = strlen(code); + size_t len = strlen(code); if(len != 11 && len != 7 && @@ -313,12 +319,12 @@ void gbAddGgCheat(const char *code, const char *desc) int i = gbCheatNumber; - int len = strlen(code); + size_t len = strlen(code); strcpy(gbCheatList[i].cheatCode, code); strcpy(gbCheatList[i].cheatDesc, desc); - gbCheatList[i].code = 1; + gbCheatList[i].code = 0x101; gbCheatList[i].value = (GBCHEAT_HEX_VALUE(code[0]) << 4) + GBCHEAT_HEX_VALUE(code[1]); @@ -338,9 +344,12 @@ void gbAddGgCheat(const char *code, const char *desc) compare ^= 0x45; gbCheatList[i].compare = compare; - gbCheatList[i].code = 0; + //gbCheatList[i].code = 0; + gbCheatList[i].code = 0x100; // fix for compare value + } + gbCheatList[i].enabled = true; gbCheatMap[gbCheatList[i].address] = true; @@ -425,6 +434,7 @@ bool gbCheatReadGSCodeFile(const char *fileName) return true; } +// Used to emulated GG codes u8 gbCheatRead(u16 address) { if(!cheatsEnabled) @@ -437,26 +447,72 @@ u8 gbCheatRead(u16 address) if(gbMemoryMap[address>>12][address&0xFFF] == gbCheatList[i].compare) return gbCheatList[i].value; break; - case 0x00: - case 0x01: - case 0x80: - return gbCheatList[i].value; - case 0x90: - case 0x91: - case 0x92: - case 0x93: - case 0x94: - case 0x95: - case 0x96: - case 0x97: - if(address >= 0xd000 && address < 0xe000) { - if(((gbMemoryMap[0x0d] - gbWram)/0x1000) == - (gbCheatList[i].code - 0x90)) - return gbCheatList[i].value; - } else + case 0x101: // GameGenie 6 digits code support return gbCheatList[i].value; + break; } } } return gbMemoryMap[address>>12][address&0xFFF]; } + + +// Used to emulate GS codes. +void gbCheatWrite(bool reboot) +{ + if(cheatsEnabled) + { + u16 address = 0; + + if (gbNextCheat >= gbCheatNumber) + gbNextCheat = 0; + + for(int i = gbNextCheat; i < gbCheatNumber; i++) { + if(gbCheatList[i].enabled) { + address = gbCheatList[i].address; + if ((!reboot) && (address >= 0x8000) && !((address>=0xA000) && (address<0xC000))) + { // These codes are executed one per one, at each Vblank + switch(gbCheatList[i].code) { + case 0x01: + gbWriteMemory(address, gbCheatList[i].value); + gbNextCheat = i+1; + return; + case 0x90: + case 0x91: + case 0x92: + case 0x93: + case 0x94: + case 0x95: + case 0x96: + case 0x97: + case 0x98: + case 0x99: + case 0x9A: + case 0x9B: + case 0x9C: + case 0x9D: + case 0x9E: + case 0x9F: + int oldbank = gbMemory[0xff70]; + gbWriteMemory(0xff70, gbCheatList[i].code & 0xf); + gbWriteMemory(address, gbCheatList[i].value); + gbWriteMemory(0xff70, oldbank); + gbNextCheat = i+1; + return; + } + } + else // These codes are only executed when the game is booted + { + switch(gbCheatList[i].code & 0xF0) { + case 0x80: + gbWriteMemory(0x0000, 0x0A); + gbWriteMemory(0x4000, gbCheatList[i].value & 0xF); + gbWriteMemory(address, gbCheatList[i].value); + gbNextCheat = i+1; + return; + } + } + } + } + } +} \ No newline at end of file diff --git a/src/gb/gbCheats.h b/src/gb/gbCheats.h index 3845a53c..418b2862 100644 --- a/src/gb/gbCheats.h +++ b/src/gb/gbCheats.h @@ -50,6 +50,7 @@ extern void gbCheatRemoveAll(); extern void gbCheatEnable(int); extern void gbCheatDisable(int); extern u8 gbCheatRead(u16); +extern void gbCheatWrite(bool); extern int gbCheatNumber; extern gbCheat gbCheatList[100]; diff --git a/src/gb/gbCodes.h b/src/gb/gbCodes.h index 43c45199..73dd088b 100644 --- a/src/gb/gbCodes.h +++ b/src/gb/gbCodes.h @@ -1,7 +1,7 @@ // -*- C++ -*- // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005-2006 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -22,8 +22,8 @@ break; case 0x01: // LD BC, NNNN - BC.B.B0=gbReadMemory(PC.W++); - BC.B.B1=gbReadMemory(PC.W++); + BC.B.B0=gbReadOpcode(PC.W++); + BC.B.B1=gbReadOpcode(PC.W++); break; case 0x02: // LD (BC),A @@ -102,7 +102,9 @@ opcode = gbReadOpcode(PC.W++); if(gbCgbMode) { if(gbMemory[0xff4d] & 1) { + gbSpeedSwitch(); + //clockTicks += 228*144-(gbSpeed ? 62 : 63); if(gbSpeed == 0) gbMemory[0xff4d] = 0x00; @@ -113,8 +115,8 @@ break; case 0x11: // LD DE, NNNN - DE.B.B0=gbReadMemory(PC.W++); - DE.B.B1=gbReadMemory(PC.W++); + DE.B.B0=gbReadOpcode(PC.W++); + DE.B.B1=gbReadOpcode(PC.W++); break; case 0x12: // LD (DE),A @@ -147,7 +149,7 @@ break; case 0x18: // JR NN - PC.W+=(s8)gbReadMemory(PC.W)+1; + PC.W+=(s8)gbReadOpcode(PC.W)+1; break; case 0x19: // ADD HL,DE @@ -190,14 +192,14 @@ if(AF.B.B0&Z_FLAG) PC.W++; else { - PC.W+=(s8)gbReadMemory(PC.W)+1; + PC.W+=(s8)gbReadOpcode(PC.W)+1; clockTicks++; } break; case 0x21: // LD HL,NNNN - HL.B.B0=gbReadMemory(PC.W++); - HL.B.B1=gbReadMemory(PC.W++); + HL.B.B0=gbReadOpcode(PC.W++); + HL.B.B1=gbReadOpcode(PC.W++); break; case 0x22: // LDI (HL),A @@ -233,7 +235,7 @@ case 0x28: // JR Z,NN if(AF.B.B0&Z_FLAG) { - PC.W+=(s8)gbReadMemory(PC.W)+1; + PC.W+=(s8)gbReadOpcode(PC.W)+1; clockTicks++; } else PC.W++; @@ -278,14 +280,14 @@ if(AF.B.B0&C_FLAG) PC.W++; else { - PC.W+=(s8)gbReadMemory(PC.W)+1; + PC.W+=(s8)gbReadOpcode(PC.W)+1; clockTicks++; } break; case 0x31: // LD SP,NNNN - SP.B.B0=gbReadMemory(PC.W++); - SP.B.B1=gbReadMemory(PC.W++); + SP.B.B0=gbReadOpcode(PC.W++); + SP.B.B1=gbReadOpcode(PC.W++); break; case 0x32: // LDD (HL),A @@ -318,7 +320,7 @@ case 0x38: // JR C,NN if(AF.B.B0&C_FLAG) { - PC.W+=(s8)gbReadMemory(PC.W)+1; + PC.W+=(s8)gbReadOpcode(PC.W)+1; clockTicks ++; } else PC.W++; @@ -575,16 +577,24 @@ case 0x38: break; case 0x76: // HALT - if(IFF & 1) { + // If an EI is pending, the interrupts are triggered before Halt state !! + // Fix Torpedo Range's intro. + if (IFF & 0x40) + { + IFF &= ~0x70; + IFF |=1; PC.W--; - IFF |= 0x80; - } else { - if((register_IE & register_IF) > 0) - IFF |= 0x100; - else { - PC.W--; - IFF |= 0x81; + } + else + { + // if (IE & IF) and interrupts are disabeld, + // Halt is cancelled. + if ((register_IE & register_IF & 0x1f) && !(IFF & 1)) + { + IFF|=2; } + else + IFF |= 0x80; } break; case 0x77: @@ -1036,16 +1046,16 @@ case 0x38: if(AF.B.B0&Z_FLAG) PC.W+=2; else { - tempRegister.B.B0=gbReadMemory(PC.W++); - tempRegister.B.B1=gbReadMemory(PC.W); + tempRegister.B.B0=gbReadOpcode(PC.W++); + tempRegister.B.B1=gbReadOpcode(PC.W); PC.W=tempRegister.W; clockTicks++; } break; case 0xc3: // JP NNNN - tempRegister.B.B0=gbReadMemory(PC.W++); - tempRegister.B.B1=gbReadMemory(PC.W); + tempRegister.B.B0=gbReadOpcode(PC.W++); + tempRegister.B.B1=gbReadOpcode(PC.W); PC.W=tempRegister.W; break; case 0xc4: @@ -1053,8 +1063,8 @@ case 0x38: if(AF.B.B0&Z_FLAG) PC.W+=2; else { - tempRegister.B.B0=gbReadMemory(PC.W++); - tempRegister.B.B1=gbReadMemory(PC.W++); + tempRegister.B.B0=gbReadOpcode(PC.W++); + tempRegister.B.B1=gbReadOpcode(PC.W++); gbWriteMemory(--SP.W,PC.B.B1); gbWriteMemory(--SP.W,PC.B.B0); PC.W=tempRegister.W; @@ -1096,8 +1106,8 @@ case 0x38: case 0xca: // JP Z,NNNN if(AF.B.B0&Z_FLAG) { - tempRegister.B.B0=gbReadMemory(PC.W++); - tempRegister.B.B1=gbReadMemory(PC.W); + tempRegister.B.B0=gbReadOpcode(PC.W++); + tempRegister.B.B1=gbReadOpcode(PC.W); PC.W=tempRegister.W; clockTicks++; } else @@ -1107,8 +1117,8 @@ case 0x38: case 0xcc: // CALL Z,NNNN if(AF.B.B0&Z_FLAG) { - tempRegister.B.B0=gbReadMemory(PC.W++); - tempRegister.B.B1=gbReadMemory(PC.W++); + tempRegister.B.B0=gbReadOpcode(PC.W++); + tempRegister.B.B1=gbReadOpcode(PC.W++); gbWriteMemory(--SP.W,PC.B.B1); gbWriteMemory(--SP.W,PC.B.B0); PC.W=tempRegister.W; @@ -1118,8 +1128,8 @@ case 0x38: break; case 0xcd: // CALL NNNN - tempRegister.B.B0=gbReadMemory(PC.W++); - tempRegister.B.B1=gbReadMemory(PC.W++); + tempRegister.B.B0=gbReadOpcode(PC.W++); + tempRegister.B.B1=gbReadOpcode(PC.W++); gbWriteMemory(--SP.W,PC.B.B1); gbWriteMemory(--SP.W,PC.B.B0); PC.W=tempRegister.W; @@ -1156,20 +1166,24 @@ case 0x38: if(AF.B.B0&C_FLAG) PC.W+=2; else { - tempRegister.B.B0=gbReadMemory(PC.W++); - tempRegister.B.B1=gbReadMemory(PC.W); + tempRegister.B.B0=gbReadOpcode(PC.W++); + tempRegister.B.B1=gbReadOpcode(PC.W); PC.W=tempRegister.W; clockTicks++; } break; // D3 illegal + case 0xd3: + PC.W--; + IFF = 0; + break; case 0xd4: // CALL NC,NNNN if(AF.B.B0&C_FLAG) PC.W+=2; else { - tempRegister.B.B0=gbReadMemory(PC.W++); - tempRegister.B.B1=gbReadMemory(PC.W++); + tempRegister.B.B0=gbReadOpcode(PC.W++); + tempRegister.B.B1=gbReadOpcode(PC.W++); gbWriteMemory(--SP.W,PC.B.B1); gbWriteMemory(--SP.W,PC.B.B0); PC.W=tempRegister.W; @@ -1200,7 +1214,7 @@ case 0x38: if(AF.B.B0&C_FLAG) { PC.B.B0=gbReadMemory(SP.W++); PC.B.B1=gbReadMemory(SP.W++); - clockTicks += 4; + clockTicks += 3; } break; case 0xd9: @@ -1212,19 +1226,23 @@ case 0x38: case 0xda: // JP C,NNNN if(AF.B.B0&C_FLAG) { - tempRegister.B.B0=gbReadMemory(PC.W++); - tempRegister.B.B1=gbReadMemory(PC.W); + tempRegister.B.B0=gbReadOpcode(PC.W++); + tempRegister.B.B1=gbReadOpcode(PC.W); PC.W=tempRegister.W; clockTicks++; } else PC.W+=2; break; // DB illegal + case 0xdb: + PC.W--; + IFF = 0; + break; case 0xdc: // CALL C,NNNN if(AF.B.B0&C_FLAG) { - tempRegister.B.B0=gbReadMemory(PC.W++); - tempRegister.B.B1=gbReadMemory(PC.W++); + tempRegister.B.B0=gbReadOpcode(PC.W++); + tempRegister.B.B1=gbReadOpcode(PC.W++); gbWriteMemory(--SP.W,PC.B.B1); gbWriteMemory(--SP.W,PC.B.B0); PC.W=tempRegister.W; @@ -1233,6 +1251,10 @@ case 0x38: PC.W+=2; break; // DD illegal + case 0xdd: + PC.W--; + IFF = 0; + break; case 0xde: // SBC NN tempValue=gbReadOpcode(PC.W++); @@ -1262,6 +1284,11 @@ case 0x38: break; // E3 illegal // E4 illegal + case 0xe3: + case 0xe4: + PC.W--; + IFF = 0; + break; case 0xe5: // PUSH HL gbWriteMemory(--SP.W,HL.B.B1); @@ -1308,6 +1335,12 @@ case 0x38: // EB illegal // EC illegal // ED illegal + case 0xeb: + case 0xec: + case 0xed: + PC.W--; + IFF = 0; + break; case 0xee: // XOR NN tempValue=gbReadOpcode(PC.W++); @@ -1336,9 +1369,13 @@ case 0x38: case 0xf3: // DI // IFF&=0xFE; - IFF&=(~0x21); + IFF|=0x08; break; // F4 illegal + case 0xf4: + PC.W--; + IFF = 0; + break; case 0xf5: // PUSH AF gbWriteMemory(--SP.W,AF.B.B1); @@ -1383,10 +1420,23 @@ case 0x38: break; case 0xfb: // EI - IFF|=0x20; + if (!(IFF & 0x30)) + // If an EI is executed right before HALT, + // the interrupts are triggered before the Halt state !! + // Fix Torpedo Range Intro. + // IFF |= 0x10 : 1 ticks before the EI enables the interrupts + // IFF |= 0x40 : marks that an EI is being executed. + IFF|=0x50; break; - // FC illegal + // FC illegal (FC = breakpoint) + case 0xfc: + breakpoint = true; + break; // FD illegal + case 0xfd: + PC.W--; + IFF = 0; + break; case 0xfe: // CP NN tempValue=gbReadOpcode(PC.W++); @@ -1401,7 +1451,10 @@ case 0x38: PC.W=0x0038; break; default: - systemMessage(0, N_("Unknown opcode %02x at %04x"), - gbReadOpcode(PC.W-1),PC.W-1); - emulating = false; + if (gbSystemMessage == false) + { + systemMessage(0, N_("Unknown opcode %02x at %04x"), + gbReadOpcode(PC.W-1),PC.W-1); + gbSystemMessage =true; + } return; diff --git a/src/gb/gbCodesCB.h b/src/gb/gbCodesCB.h index 05dce6e6..5a09d8d5 100644 --- a/src/gb/gbCodesCB.h +++ b/src/gb/gbCodesCB.h @@ -1,7 +1,7 @@ // -*- C++ -*- // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -1282,7 +1282,10 @@ AF.B.B1|=1<<7; break; default: - systemMessage(0, N_("Unknown opcode %02x at %04x"), - gbReadOpcode(PC.W-1),PC.W-1); - emulating = false; + if (gbSystemMessage == false) + { + systemMessage(0, N_("Unknown opcode %02x at %04x"), + gbReadOpcode(PC.W-1),PC.W-1); + gbSystemMessage =true; + } return; diff --git a/src/gb/gbGfx.cpp b/src/gb/gbGfx.cpp index aa639664..6558b4eb 100644 --- a/src/gb/gbGfx.cpp +++ b/src/gb/gbGfx.cpp @@ -1,6 +1,6 @@ // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005-2006 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -58,9 +58,12 @@ u8 gbInvertTab[256] = { }; u16 gbLineMix[160]; +u16 gbWindowColor[160]; +extern int inUseRegister_WY; void gbRenderLine() { + memset(gbLineMix, 0, sizeof(gbLineMix)); u8 * bank0; u8 * bank1; if(gbCgbMode) { @@ -86,10 +89,9 @@ void gbRenderLine() if(y >= 144) return; - // int yLine = (y + gbBorderRowSkip) * gbBorderLineSkip; - - int sx = register_SCX; - int sy = register_SCY; + int SpritesTicks = gbSpritesTicks[x]*(gbSpeed ? 2 : 4); + int sx = gbSCXLine[(gbSpeed ? 0 : 4)+SpritesTicks]; + int sy = gbSCYLine[(gbSpeed ? 11 : 5)+SpritesTicks]; sy+=y; @@ -113,16 +115,17 @@ void gbRenderLine() tile_map_address++; - if((register_LCDC & 16) == 0) { - if(tile < 128) tile += 128; - else tile -= 128; - } + if(!(register_LCDC & 0x10)) + tile ^= 0x80; int tile_pattern_address = tile_pattern + tile * 16 + by*2; if(register_LCDC & 0x80) { if((register_LCDC & 0x01 || gbCgbMode) && (layerSettings & 0x0100)) { while(x < 160) { + + + u8 tile_a = 0; u8 tile_b = 0; @@ -155,7 +158,7 @@ void gbRenderLine() if(gbCgbMode) { c = c + (attrs & 7)*4; } else { - c = gbBgp[c]; + c = (gbBgpLine[x+(gbSpeed ? 5 : 11)+SpritesTicks]>>(c<<1)) &3; if(gbSgbMode && !gbCgbMode) { int dx = x >> 3; int dy = y >> 3; @@ -168,42 +171,92 @@ void gbRenderLine() c = c + 4*palette; } } - gbLineMix[x] = gbColorOption ? gbColorFilter[gbPalette[c]] : - gbPalette[c]; + gbLineMix[x] = gbColorOption ? gbColorFilter[gbPalette[c] & 0x7FFF] : + gbPalette[c] & 0x7FFF; x++; if(x >= 160) break; bx >>= 1; } - tx++; - if(tx == 32) - tx = 0; + bx = 128; - + + SpritesTicks = gbSpritesTicks[x]*(gbSpeed ? 2 : 4); + + sx = gbSCXLine[x+(gbSpeed ? 0 : 4)+SpritesTicks]; + + sy = gbSCYLine[x+(gbSpeed ? 11 : 5)+SpritesTicks]; + + + tx = ((sx+x)>>3) & 0x1f; + + sy+=y; + + sy &= 255; + + ty = sy >> 3; + + by = sy & 7; + + tile_pattern_address = tile_pattern + tile * 16 + by * 2; + + tile_map_line_y = tile_map + ty * 32; + + tile_map_address = tile_map_line_y + tx; + if(bank1) attrs = bank1[tile_map_line_y + tx]; tile = bank0[tile_map_line_y + tx]; - - if((register_LCDC & 16) == 0) { - if(tile < 128) tile += 128; - else tile -= 128; - } + + if(!(register_LCDC & 0x10)) + tile ^= 0x80; + tile_pattern_address = tile_pattern + tile * 16 + by * 2; } } else { - for(int i = 0; i < 160; i++) { - gbLineMix[i] = gbPalette[0]; + // Use gbBgp[0] instead of 0 (?) + // (this fixes white flashes on Last Bible II) + // Also added the gbColorOption (fixes Dracula Densetsu II color problems) + for(int i = 0; i < 160; i++) + { + u16 color = gbColorOption ? gbColorFilter[0x7FFF] : + 0x7FFF; + if (!gbCgbMode) + color = gbColorOption ? gbColorFilter[gbPalette[gbBgpLine[i+(gbSpeed ? 5 : 11)+gbSpritesTicks[i]*(gbSpeed ? 2 : 4)]&3] & 0x7FFF] : + gbPalette[gbBgpLine[i+(gbSpeed ? 5 : 11)+gbSpritesTicks[i]*(gbSpeed ? 2 : 4)]&3] & 0x7FFF; + gbLineMix[i] = color; gbLineBuffer[i] = 0; } } // do the window display - if((register_LCDC & 0x20) && (layerSettings & 0x2000)) { - int wy = register_WY; + // LCDC.0 also enables/disables the window in !gbCgbMode ?!?! + // (tested on real hardware) + // This fixes Last Bible II & Zankurou Musouken + if((register_LCDC & 0x01 || gbCgbMode) && (register_LCDC & 0x20) && + (layerSettings & 0x2000) && (gbWindowLine != -2)) { + int i = 0; + // Fix (accurate emulation) for most of the window display problems + // (ie. Zen - Intergalactic Ninja, Urusei Yatsura...). + if ((gbWindowLine == -1) || (gbWindowLine>144)) + { + inUseRegister_WY = oldRegister_WY; + if (register_LY>oldRegister_WY) + gbWindowLine = 146; + // for (i = 0; i<160; i++) + // gbWindowColor[i] = gbLineMix[i]; + } + + int wy = inUseRegister_WY; - if(y >= wy) { + if(y >= inUseRegister_WY) { + + if (gbWindowLine == -1) + gbWindowLine = 0; + int wx = register_WX; + int swx = 0; wx -= 7; if( wx <= 159 && gbWindowLine <= 143) { @@ -212,10 +265,7 @@ void gbRenderLine() if((register_LCDC & 0x40) != 0) tile_map = 0x1c00; - - if(gbWindowLine == -1) { - gbWindowLine = 0; - } + tx = 0; ty = gbWindowLine >> 3; @@ -223,6 +273,25 @@ void gbRenderLine() bx = 128; by = gbWindowLine & 7; + // Tries to emulate the 'window scrolling bug' when wx == 0 (ie. wx-7 == -7). + // Nothing close to perfect, but good enought for now... + if (wx == -7) + { + swx = 7-((gbSCXLine[0]-1) & 7); + bx >>= ((gbSCXLine[0]+((swx != 1) ? 1 : 0)) & 7); + if (swx == 1) + swx = 2; + + //bx >>= ((gbSCXLine[0]+(((swx>1) && (swx != 7)) ? 1 : 0)) & 7); + + if ((swx == 7)) + { + //wx = 0; + if ((gbWindowLine>0) || (wy == 0)) + swx = 0; + } + } + else if(wx < 0) { bx >>= (-wx); wx = 0; @@ -247,6 +316,10 @@ void gbRenderLine() tile_pattern_address = tile_pattern + tile * 16 + by*2; + if (wx) + for (i = 0; i=0) + { if(attrs & 0x80) gbLineBuffer[x] = 0x300 + c; else @@ -280,8 +355,8 @@ void gbRenderLine() if(gbCgbMode) { c = c + (attrs & 7) * 4; } else { - c = gbBgp[c]; - if(gbSgbMode && ! gbCgbMode) { + c = (gbBgpLine[x+(gbSpeed ? 5 : 11)+gbSpritesTicks[x]*(gbSpeed ? 2 : 4)]>>(c<<1)) &3; + if(gbSgbMode && !gbCgbMode) { int dx = x >> 3; int dy = y >> 3; @@ -293,8 +368,9 @@ void gbRenderLine() c = c + 4*palette; } } - gbLineMix[x] = gbColorOption ? gbColorFilter[gbPalette[c]] : - gbPalette[c]; + gbLineMix[x] = gbColorOption ? gbColorFilter[gbPalette[c] & 0x7FFF] : + gbPalette[c] & 0x7FFF; + } x++; if(x >= 160) break; @@ -314,13 +390,30 @@ void gbRenderLine() } tile_pattern_address = tile_pattern + tile * 16 + by * 2; } + + //for (i = swx; i<160; i++) + // gbLineMix[i] = gbWindowColor[i]; gbWindowLine++; } } } + else if (gbWindowLine == -2) + { + inUseRegister_WY = oldRegister_WY; + if (register_LY>oldRegister_WY) + gbWindowLine = 146; + else + gbWindowLine = 0; + } } else { - for(int i = 0; i < 160; i++) { - gbLineMix[i] = gbPalette[0]; + u16 color = gbColorOption ? gbColorFilter[0x7FFF] : + 0x7FFF; + if (!gbCgbMode) + color = gbColorOption ? gbColorFilter[gbPalette[0] & 0x7FFF] : + gbPalette[0] & 0x7FFF; + for(int i = 0; i < 160; i++) + { + gbLineMix[i] = color; gbLineBuffer[i] = 0; } } @@ -346,8 +439,11 @@ void gbDrawSpriteTile(int tile, int x,int y,int t, int flags, int init = 0x0000; - // int yLine = (y+gbBorderRowSkip) * gbBorderLineSkip; - + for (int i = 0; i<4; i++) + { + gbObp0[i] = (gbObp0Line[x+11+gbSpritesTicks[x]*(gbSpeed ? 2 : 4)]>>(i<<1)) & 3; + gbObp1[i] = (gbObp1Line[x+11+gbSpritesTicks[x]*(gbSpeed ? 2 : 4)]>>(i<<1)) & 3; + } u8 *pal = gbObp0; int flipx = (flags & 0x20); @@ -366,7 +462,7 @@ void gbDrawSpriteTile(int tile, int x,int y,int t, int flags, int a = 0; int b = 0; - if(gbCgbMode && flags & 0x08) { + if(gbCgbMode && (flags & 0x08)) { a = bank1[address++]; b = bank1[address++]; } else { @@ -392,12 +488,14 @@ void gbDrawSpriteTile(int tile, int x,int y,int t, int flags, continue; u16 color = gbLineBuffer[xxx]; - - if(prio) { + + // Fixes OAM-BG priority + if(prio && (register_LCDC & 1)) { if(color < 0x200 && ((color & 0xFF) != 0)) continue; } - if(color >= 0x300 && color != 0x300) + // Fixes OAM-BG priority for Moorhuhn 2 + if(color >= 0x300 && color != 0x300 && (register_LCDC & 1)) continue; else if(color >= 0x200 && color < 0x300) { int sprite = color & 0xff; @@ -412,7 +510,9 @@ void gbDrawSpriteTile(int tile, int x,int y,int t, int flags, if(sprite < spriteNumber) continue; } else { - if(spriteX < x+8) + // Fixes GB sprites priorities (was '< x + 8' before) + // ('A boy and his blob...' sprites' emulation is now correct) + if(spriteX < x) continue; } } @@ -442,12 +542,12 @@ void gbDrawSpriteTile(int tile, int x,int y,int t, int flags, } } - gbLineMix[xxx] = gbColorOption ? gbColorFilter[gbPalette[c]] : - gbPalette[c]; + gbLineMix[xxx] = gbColorOption ? gbColorFilter[gbPalette[c] & 0x7FFF] : + gbPalette[c] & 0x7FFF; } } -void gbDrawSprites() +void gbDrawSprites(bool draw) { int x = 0; int y = 0; @@ -455,6 +555,9 @@ void gbDrawSprites() int size = (register_LCDC & 4); + if (!draw) + memset (gbSpritesTicks, 0, sizeof(gbSpritesTicks)); + if(!(register_LCDC & 0x80)) return; @@ -473,11 +576,19 @@ void gbDrawSprites() if(x > 0 && y > 0 && x < 168 && y < 160) { // check if sprite intersects current line int t = yc -y + 16; - if(size && t >=0 && t < 16) { - gbDrawSpriteTile(tile,x-8,yc,t,flags,size,i); - count++; - } else if(!size && t >= 0 && t < 8) { - gbDrawSpriteTile(tile, x-8, yc, t, flags,size,i); + if((size && t >=0 && t < 16) || (!size && t >= 0 && t < 8)) { + if (draw) + gbDrawSpriteTile(tile,x-8,yc,t,flags,size,i); + else + { + for (int j = x-8; j<300; j++) + if (j>=0) + if (gbSpeed) + gbSpritesTicks[j] += 5; + else + gbSpritesTicks[j] += 2+(count&1); + + } count++; } } @@ -486,5 +597,5 @@ void gbDrawSprites() break; } } -} - + return; +} \ No newline at end of file diff --git a/src/gb/gbGlobals.cpp b/src/gb/gbGlobals.cpp index 90b1af15..877d28f4 100644 --- a/src/gb/gbGlobals.cpp +++ b/src/gb/gbGlobals.cpp @@ -1,6 +1,6 @@ // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005-2006 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -24,6 +24,7 @@ int gbRomSizeMask = 0; int gbRomSize = 0; int gbRamSizeMask = 0; int gbRamSize = 0; +int gbTAMA5ramSize = 0; u8 *gbMemory = NULL; u8 *gbVram = NULL; @@ -31,6 +32,7 @@ u8 *gbRom = NULL; u8 *gbRam = NULL; u8 *gbWram = NULL; u16 *gbLineBuffer = NULL; +u8 *gbTAMA5ram = NULL; u16 gbPalette[128]; u8 gbBgp[4] = { 0, 1, 2, 3}; @@ -38,6 +40,7 @@ u8 gbObp0[4] = { 0, 1, 2, 3}; u8 gbObp1[4] = { 0, 1, 2, 3}; int gbWindowLine = -1; +bool genericflashcardEnable = false; int gbCgbMode = 0; u16 gbColorFilter[32768]; diff --git a/src/gb/gbGlobals.h b/src/gb/gbGlobals.h index 380f9331..d214d0d3 100644 --- a/src/gb/gbGlobals.h +++ b/src/gb/gbGlobals.h @@ -1,7 +1,7 @@ // -*- C++ -*- // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005-2006 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -21,6 +21,11 @@ extern int gbRomSizeMask; extern int gbRomSize; extern int gbRamSize; extern int gbRamSizeMask; +extern int gbTAMA5ramSize; + +extern bool useBios; +extern bool skipBios; +extern u8 *bios; extern u8 *gbRom; extern u8 *gbRam; @@ -28,6 +33,7 @@ extern u8 *gbVram; extern u8 *gbWram; extern u8 *gbMemory; extern u16 *gbLineBuffer; +extern u8 *gbTAMA5ram; extern u8 *gbMemoryMap[16]; @@ -46,6 +52,19 @@ extern u8 gbBgp[4]; extern u8 gbObp0[4]; extern u8 gbObp1[4]; extern u16 gbPalette[128]; +extern bool gbScreenOn; +extern bool gbDrawWindow; +extern u8 gbSCYLine[300]; +// gbSCXLine is used for the emulation (bug) of the SX change +// found in the Artic Zone game. +extern u8 gbSCXLine[300]; +// gbBgpLine is used for the emulation of the +// Prehistorik Man's title screen scroller. +extern u8 gbBgpLine[300]; +extern u8 gbObp0Line [300]; +extern u8 gbObp1Line [300]; +// gbSpritesTicks is used for the emulation of Parodius' Laser Beam. +extern u8 gbSpritesTicks[300]; extern u8 register_LCDC; extern u8 register_LY; @@ -54,8 +73,10 @@ extern u8 register_SCX; extern u8 register_WY; extern u8 register_WX; extern u8 register_VBK; +extern u8 oldRegister_WY; extern int emulating; +extern bool genericflashcardEnable; extern int gbBorderLineSkip; extern int gbBorderRowSkip; @@ -63,6 +84,6 @@ extern int gbBorderColumnSkip; extern int gbDmaTicks; extern void gbRenderLine(); -extern void gbDrawSprites(); +extern void gbDrawSprites(bool); extern u8 (*gbSerialFunction)(u8); diff --git a/src/gb/gbMemory.cpp b/src/gb/gbMemory.cpp index be379d3e..fd57aee1 100644 --- a/src/gb/gbMemory.cpp +++ b/src/gb/gbMemory.cpp @@ -1,6 +1,6 @@ // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005-2006 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -20,6 +20,12 @@ #include "../Port.h" #include "gbGlobals.h" #include "gbMemory.h" +#include "GB.h" +u8 gbDaysinMonth [12] = { 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31}; +const u8 gbDisabledRam [8] = {0x80, 0xff, 0xf0, 0x00, 0x30, 0xbf, 0xbf, 0xbf}; +extern int gbHardware; +extern int gbGBCColorType; +extern gbRegister PC; mapperMBC1 gbDataMBC1 = { 0, // RAM enable @@ -27,7 +33,8 @@ mapperMBC1 gbDataMBC1 = { 0, // RAM bank 0, // memory model 0, // ROM high address - 0 // RAM address + 0, // RAM address + 0 // Rom Bank 0 remapping }; // MBC1 ROM write registers @@ -41,17 +48,25 @@ void mapperMBC1ROM(u16 address, u8 value) break; case 0x2000: // ROM bank select // value = value & 0x1f; - if(value == 0) - value = 1; + if ((value == 1) && (address == 0x2100)) + gbDataMBC1.mapperRomBank0Remapping = 1; + + if((value & 0x1f) == 0) + value += 1; if(value == gbDataMBC1.mapperROMBank) break; tmpAddress = value << 14; // check current model + if (gbDataMBC1.mapperRomBank0Remapping == 3) { + tmpAddress = (value & 0xf) << 14; + tmpAddress |= (gbDataMBC1.mapperROMHighAddress & 3) << 18; + } + else if(gbDataMBC1.mapperMemoryModel == 0) { // model is 16/8, so we have a high address in use - tmpAddress |= (gbDataMBC1.mapperROMHighAddress) << 19; + tmpAddress |= (gbDataMBC1.mapperROMHighAddress & 3) << 19; } tmpAddress &= gbRomSizeMask; @@ -63,16 +78,39 @@ void mapperMBC1ROM(u16 address, u8 value) break; case 0x4000: // RAM bank select if(gbDataMBC1.mapperMemoryModel == 1) { + if (!gbRamSize) + { + if (gbDataMBC1.mapperRomBank0Remapping == 3) + { + gbDataMBC1.mapperROMHighAddress = value & 0x03; + tmpAddress = (gbDataMBC1.mapperROMHighAddress) << 18; + tmpAddress &= gbRomSizeMask; + gbMemoryMap[0x00] = &gbRom[tmpAddress]; + gbMemoryMap[0x01] = &gbRom[tmpAddress + 0x1000]; + gbMemoryMap[0x02] = &gbRom[tmpAddress + 0x2000]; + gbMemoryMap[0x03] = &gbRom[tmpAddress + 0x3000]; + gbMemoryMap[0x04] = &gbRom[tmpAddress + 0x4000]; + gbMemoryMap[0x05] = &gbRom[tmpAddress + 0x5000]; + gbMemoryMap[0x06] = &gbRom[tmpAddress + 0x6000]; + gbMemoryMap[0x07] = &gbRom[tmpAddress + 0x7000]; + } + else gbDataMBC1.mapperRomBank0Remapping = 0; + } // 4/32 model, RAM bank switching provided value = value & 0x03; if(value == gbDataMBC1.mapperRAMBank) break; tmpAddress = value << 13; tmpAddress &= gbRamSizeMask; - gbMemoryMap[0x0a] = &gbRam[tmpAddress]; - gbMemoryMap[0x0b] = &gbRam[tmpAddress + 0x1000]; + if(gbRamSize) { + gbMemoryMap[0x0a] = &gbRam[tmpAddress]; + gbMemoryMap[0x0b] = &gbRam[tmpAddress + 0x1000]; + } gbDataMBC1.mapperRAMBank = value; gbDataMBC1.mapperRAMAddress = tmpAddress; + + if (gbDataMBC1.mapperRomBank0Remapping != 3) + gbDataMBC1.mapperROMHighAddress = 0; } else { // 16/8, set the high address gbDataMBC1.mapperROMHighAddress = value & 0x03; @@ -83,10 +121,57 @@ void mapperMBC1ROM(u16 address, u8 value) gbMemoryMap[0x05] = &gbRom[tmpAddress + 0x1000]; gbMemoryMap[0x06] = &gbRom[tmpAddress + 0x2000]; gbMemoryMap[0x07] = &gbRom[tmpAddress + 0x3000]; + if(gbRamSize) { + gbMemoryMap[0x0a] = &gbRam[0]; + gbMemoryMap[0x0b] = &gbRam[0x1000]; + } + + gbDataMBC1.mapperRAMBank = 0; } break; case 0x6000: // memory model select gbDataMBC1.mapperMemoryModel = value & 1; + + if(gbDataMBC1.mapperMemoryModel == 1) { + // 4/32 model, RAM bank switching provided + + value = gbDataMBC1.mapperRAMBank & 0x03; + tmpAddress = value << 13; + tmpAddress &= gbRamSizeMask; + if(gbRamSize) { + gbMemoryMap[0x0a] = &gbRam[gbDataMBC1.mapperRAMAddress]; + gbMemoryMap[0x0b] = &gbRam[gbDataMBC1.mapperRAMAddress + 0x1000]; + gbDataMBC1.mapperRomBank0Remapping = 0; + } + else gbDataMBC1.mapperRomBank0Remapping |=2; + + gbDataMBC1.mapperRAMBank = value; + gbDataMBC1.mapperRAMAddress = tmpAddress; + + tmpAddress = gbDataMBC1.mapperROMBank << 14; + + + tmpAddress &= gbRomSizeMask; + gbMemoryMap[0x04] = &gbRom[tmpAddress]; + gbMemoryMap[0x05] = &gbRom[tmpAddress + 0x1000]; + gbMemoryMap[0x06] = &gbRom[tmpAddress + 0x2000]; + gbMemoryMap[0x07] = &gbRom[tmpAddress + 0x3000]; + + } else { + // 16/8, set the high address + + tmpAddress = gbDataMBC1.mapperROMBank << 14; + tmpAddress |= (gbDataMBC1.mapperROMHighAddress) << 19; + tmpAddress &= gbRomSizeMask; + gbMemoryMap[0x04] = &gbRom[tmpAddress]; + gbMemoryMap[0x05] = &gbRom[tmpAddress + 0x1000]; + gbMemoryMap[0x06] = &gbRom[tmpAddress + 0x2000]; + gbMemoryMap[0x07] = &gbRom[tmpAddress + 0x3000]; + if(gbRamSize) { + gbMemoryMap[0x0a] = &gbRam[0]; + gbMemoryMap[0x0b] = &gbRam[0x1000]; + } + } break; } } @@ -102,23 +187,72 @@ void mapperMBC1RAM(u16 address, u8 value) } } +// MBC1 read RAM +u8 mapperMBC1ReadRAM(u16 address) +{ + + if(gbDataMBC1.mapperRAMEnable) + return gbMemoryMap[address>>12][address & 0x0fff]; + + if (!genericflashcardEnable) + return 0xff; + else + if ((address & 0x1000) >= 0x1000) + { + // The value returned when reading RAM while it's disabled + // is constant, exept for the GBASP hardware. + // (actually, is the address that read is out of the ROM, the returned value if 0xff...) + if (PC.W>=0xff80) + return 0xff; + else + if ((gbHardware & 0x08) && (gbGBCColorType == 2)) + { + if (address & 1) + return 0xfb; + else + return 0x7a; + } + else + return 0x0a; + } + else + return gbDisabledRam[address & 7]; +} + void memoryUpdateMapMBC1() { int tmpAddress = gbDataMBC1.mapperROMBank << 14; // check current model - if(gbDataMBC1.mapperMemoryModel == 1) { - // model is 16/8, so we have a high address in use - tmpAddress |= (gbDataMBC1.mapperROMHighAddress) << 19; + if (gbDataMBC1.mapperRomBank0Remapping == 3) { + tmpAddress = (gbDataMBC1.mapperROMHighAddress & 3) << 18; + tmpAddress &= gbRomSizeMask; + gbMemoryMap[0x00] = &gbRom[tmpAddress]; + gbMemoryMap[0x01] = &gbRom[tmpAddress + 0x1000]; + gbMemoryMap[0x02] = &gbRom[tmpAddress + 0x2000]; + gbMemoryMap[0x03] = &gbRom[tmpAddress + 0x3000]; + + tmpAddress |= (gbDataMBC1.mapperROMBank & 0xf) << 14; + gbMemoryMap[0x04] = &gbRom[tmpAddress]; + gbMemoryMap[0x05] = &gbRom[tmpAddress + 0x1000]; + gbMemoryMap[0x06] = &gbRom[tmpAddress + 0x2000]; + gbMemoryMap[0x07] = &gbRom[tmpAddress + 0x3000]; + } + else + { + if(gbDataMBC1.mapperMemoryModel == 0) { + // model is 16/8, so we have a high address in use + tmpAddress |= (gbDataMBC1.mapperROMHighAddress & 3) << 19; + } + + tmpAddress &= gbRomSizeMask; + gbMemoryMap[0x04] = &gbRom[tmpAddress]; + gbMemoryMap[0x05] = &gbRom[tmpAddress + 0x1000]; + gbMemoryMap[0x06] = &gbRom[tmpAddress + 0x2000]; + gbMemoryMap[0x07] = &gbRom[tmpAddress + 0x3000]; } - tmpAddress &= gbRomSizeMask; - gbMemoryMap[0x04] = &gbRom[tmpAddress]; - gbMemoryMap[0x05] = &gbRom[tmpAddress + 0x1000]; - gbMemoryMap[0x06] = &gbRom[tmpAddress + 0x2000]; - gbMemoryMap[0x07] = &gbRom[tmpAddress + 0x3000]; - - if(gbRamSize) { + if((gbRamSize) && (gbDataMBC1.mapperMemoryModel == 1)){ gbMemoryMap[0x0a] = &gbRam[gbDataMBC1.mapperRAMAddress]; gbMemoryMap[0x0b] = &gbRam[gbDataMBC1.mapperRAMAddress + 0x1000]; } @@ -210,7 +344,7 @@ void memoryUpdateMBC3Clock() time_t diff = now - gbDataMBC3.mapperLastTime; if(diff > 0) { // update the clock according to the last update time - gbDataMBC3.mapperSeconds += diff % 60; + gbDataMBC3.mapperSeconds += (int)(diff % 60); if(gbDataMBC3.mapperSeconds > 59) { gbDataMBC3.mapperSeconds -= 60; gbDataMBC3.mapperMinutes++; @@ -218,22 +352,22 @@ void memoryUpdateMBC3Clock() diff /= 60; - gbDataMBC3.mapperMinutes += diff % 60; - if(gbDataMBC3.mapperMinutes > 60) { + gbDataMBC3.mapperMinutes += (int)(diff % 60); + if(gbDataMBC3.mapperMinutes > 59) { gbDataMBC3.mapperMinutes -= 60; gbDataMBC3.mapperHours++; } diff /= 60; - gbDataMBC3.mapperHours += diff % 24; - if(gbDataMBC3.mapperHours > 24) { + gbDataMBC3.mapperHours += (int)(diff % 24); + if(gbDataMBC3.mapperHours > 23) { gbDataMBC3.mapperHours -= 24; gbDataMBC3.mapperDays++; } diff /= 24; - gbDataMBC3.mapperDays += diff; + gbDataMBC3.mapperDays += (int)(diff & 0xffffffff); if(gbDataMBC3.mapperDays > 255) { if(gbDataMBC3.mapperDays > 511) { gbDataMBC3.mapperDays %= 512; @@ -365,7 +499,30 @@ u8 mapperMBC3ReadRAM(u16 address) return gbDataMBC3.mapperLControl; } } - return 0; + + if (!genericflashcardEnable) + return 0xff; + else + if ((address & 0x1000) >= 0x1000) + { + // The value returned when reading RAM while it's disabled + // is constant, exept for the GBASP hardware. + // (actually, is the address that read is out of the ROM, the returned value if 0xff...) + if (PC.W>=0xff80) + return 0xff; + else + if ((gbHardware & 0x08) && (gbGBCColorType == 2)) + { + if (address & 1) + return 0xfb; + else + return 0x7a; + } + else + return 0x0a; + } + else + return gbDisabledRam[address & 7]; } void memoryUpdateMapMBC3() @@ -406,6 +563,7 @@ void mapperMBC5ROM(u16 address, u8 value) gbDataMBC5.mapperRAMEnable = ( ( value & 0x0a) == 0x0a ? 1 : 0); break; case 0x2000: // ROM bank select + if(address < 0x3000) { value = value & 0xff; if(value == gbDataMBC5.mapperROMBank) @@ -466,6 +624,38 @@ void mapperMBC5RAM(u16 address, u8 value) } } +// MBC5 read RAM +u8 mapperMBC5ReadRAM(u16 address) +{ + + if(gbDataMBC5.mapperRAMEnable) + return gbMemoryMap[address>>12][address & 0x0fff]; + + if (!genericflashcardEnable) + return 0xff; + else + if ((address & 0x1000) >= 0x1000) + { + // The value returned when reading RAM while it's disabled + // is constant, exept for the GBASP hardware. + // (actually, is the address that read is out of the ROM, the returned value if 0xff...) + if (PC.W>=0xff80) + return 0xff; + else + if ((gbHardware & 0x08) && (gbGBCColorType == 2)) + { + if (address & 1) + return 0xfb; + else + return 0x7a; + } + else + return 0x0a; + } + else + return gbDisabledRam[address & 7]; +} + void memoryUpdateMapMBC5() { int tmpAddress = (gbDataMBC5.mapperROMBank << 14) | @@ -568,7 +758,30 @@ u8 mapperMBC7ReadRAM(u16 address) case 0xa080: return gbDataMBC7.value; } - return 0xff; + + if (!genericflashcardEnable) + return 0xff; + else + if ((address & 0x1000) >= 0x1000) + { + // The value returned when reading RAM while it's disabled + // is constant, exept for the GBASP hardware. + // (actually, is the address that read is out of the ROM, the returned value if 0xff...) + if (PC.W>=0xff80) + return 0xff; + else + if ((gbHardware & 0x08) && (gbGBCColorType == 2)) + { + if (address & 1) + return 0xfb; + else + return 0x7a; + } + else + return 0x0a; + } + else + return gbDisabledRam[address & 7]; } // MBC7 RAM write @@ -717,7 +930,7 @@ void mapperMBC7RAM(u16 address, u8 value) void memoryUpdateMapMBC7() { - int tmpAddress = (gbDataMBC5.mapperROMBank << 14); + int tmpAddress = (gbDataMBC7.mapperROMBank << 14); tmpAddress &= gbRomSizeMask; gbMemoryMap[0x04] = &gbRom[tmpAddress]; @@ -965,3 +1178,540 @@ void memoryUpdateMapHuC3() gbMemoryMap[0x0b] = &gbRam[tmpAddress + 0x1000]; } } + +// TAMA5 (for Tamagotchi 3 (gb)). +// Very basic (and ugly :p) support, only rom bank switching is actually working... +mapperTAMA5 gbDataTAMA5 = { + 1, // RAM enable + 1, // ROM bank + 0, // RAM bank + 0, // RAM address + 0, // RAM Byte select + 0, // mapper command number + 0, // mapper last command; + 0, // commands 0x0 + 0, // commands 0x1 + 0, // commands 0x2 + 0, // commands 0x3 + 0, // commands 0x4 + 0, // commands 0x5 + 0, // commands 0x6 + 0, // commands 0x7 + 0, // commands 0x8 + 0, // commands 0x9 + 0, // commands 0xa + 0, // commands 0xb + 0, // commands 0xc + 0, // commands 0xd + 0, // commands 0xe + 0, // commands 0xf + 0, // register + 0, // timer clock latch + 0, // timer clock register + 0, // timer seconds + 0, // timer minutes + 0, // timer hours + 0, // timer days + 0, // timer months + 0, // timer years + 0, // timer control + 0, // timer latched seconds + 0, // timer latched minutes + 0, // timer latched hours + 0, // timer latched days + 0, // timer latched months + 0, // timer latched years + 0, // timer latched control + (time_t)-1 // last time +}; + + +void memoryUpdateTAMA5Clock() +{ + if ((gbDataTAMA5.mapperYears & 3) == 0) + gbDaysinMonth[1] = 29; + else + gbDaysinMonth[1] = 28; + + time_t now = time(NULL); + time_t diff = now - gbDataTAMA5.mapperLastTime; + if(diff > 0) { + // update the clock according to the last update time + gbDataTAMA5.mapperSeconds += (int)(diff % 60); + if(gbDataTAMA5.mapperSeconds > 59) { + gbDataTAMA5.mapperSeconds -= 60; + gbDataTAMA5.mapperMinutes++; + } + + diff /= 60; + + gbDataTAMA5.mapperMinutes += (int)(diff % 60); + if(gbDataTAMA5.mapperMinutes > 59) { + gbDataTAMA5.mapperMinutes -= 60; + gbDataTAMA5.mapperHours++; + } + + diff /= 60; + + gbDataTAMA5.mapperHours += (int)(diff % 24); + diff /= 24; + if(gbDataTAMA5.mapperHours > 23) { + gbDataTAMA5.mapperHours -= 24; + diff++; + + } + + time_t days = diff; + while (days) + { + gbDataTAMA5.mapperDays++; + days--; + if (gbDataTAMA5.mapperDays>gbDaysinMonth[gbDataTAMA5.mapperMonths-1]) + { + gbDataTAMA5.mapperDays = 1; + gbDataTAMA5.mapperMonths++; + if (gbDataTAMA5.mapperMonths>12) + { + gbDataTAMA5.mapperMonths = 1; + gbDataTAMA5.mapperYears++; + if ((gbDataTAMA5.mapperYears & 3) == 0) + gbDaysinMonth[1] = 29; + else + gbDaysinMonth[1] = 28; + } + } + } + } + gbDataTAMA5.mapperLastTime = now; + +} + + + +// TAMA5 RAM write +void mapperTAMA5RAM(u16 address, u8 value) +{ + if ((address & 0xffff) <= 0xa001) + { + switch (address & 1) + { + case 0: // 'Values' Register + { + value &= 0xf; + gbDataTAMA5.mapperCommands[gbDataTAMA5.mapperCommandNumber] = value; + gbMemoryMap[0xa][0] = value; + + int test = gbDataTAMA5.mapperCommands[gbDataTAMA5.mapperCommandNumber & 0x0e] | + (gbDataTAMA5.mapperCommands[(gbDataTAMA5.mapperCommandNumber & 0x0e) +1]<<4); + + if ((gbDataTAMA5.mapperCommandNumber & 0xe) == 0) // Read Command !!! + { + gbDataTAMA5.mapperROMBank = gbDataTAMA5.mapperCommands[0] | + (gbDataTAMA5.mapperCommands[1]<<4); + + int tmpAddress = (gbDataTAMA5.mapperROMBank << 14); + + tmpAddress &= gbRomSizeMask; + gbMemoryMap[0x04] = &gbRom[tmpAddress]; + gbMemoryMap[0x05] = &gbRom[tmpAddress + 0x1000]; + gbMemoryMap[0x06] = &gbRom[tmpAddress + 0x2000]; + gbMemoryMap[0x07] = &gbRom[tmpAddress + 0x3000]; + + gbDataTAMA5.mapperCommands[0x0f] = 0; + } + else if ((gbDataTAMA5.mapperCommandNumber & 0xe) == 4) + { + gbDataTAMA5.mapperCommands[0x0f] = 1; + if (gbDataTAMA5.mapperCommandNumber == 4) + gbDataTAMA5.mapperCommands[5] =0; // correct ? + } + else if ((gbDataTAMA5.mapperCommandNumber & 0xe) == 6) + { + gbDataTAMA5.mapperRamByteSelect = (gbDataTAMA5.mapperCommands[7]<<4) | + (gbDataTAMA5.mapperCommands[6]&0x0f); + + // Write Commands !!! + if (gbDataTAMA5.mapperCommands[0x0f] && (gbDataTAMA5.mapperCommandNumber == 7)) + { + int data = gbDataTAMA5.mapperCommands[0x04] & 0x0f | + (gbDataTAMA5.mapperCommands[0x05] <<4); + + // Not sure when the write command should reset... + // but it doesn't seem to matter. + // gbDataTAMA5.mapperCommands[0x0f] = 0; + + if (gbDataTAMA5.mapperRamByteSelect == 0x8) // Timer stuff + { + switch (data & 0xf) + { + case 0x7: + gbDataTAMA5.mapperDays = ((gbDataTAMA5.mapperDays)/10)*10 + (data >> 4); + break; + case 0x8: + gbDataTAMA5.mapperDays = (gbDataTAMA5.mapperDays%10) + (data >>4)*10; + break; + case 0x9: + gbDataTAMA5.mapperMonths = ((gbDataTAMA5.mapperMonths)/10)*10 + (data >> 4); + break; + case 0xa: + gbDataTAMA5.mapperMonths = (gbDataTAMA5.mapperMonths%10) + (data >>4)*10; + break; + case 0xb: + gbDataTAMA5.mapperYears = ((gbDataTAMA5.mapperYears)%1000) + (data >> 4)*1000; + break; + case 0xc: + gbDataTAMA5.mapperYears = (gbDataTAMA5.mapperYears%100) + (gbDataTAMA5.mapperYears/1000)*1000 + + (data >>4)*100; + break; + default : + break; + } + } + else if (gbDataTAMA5.mapperRamByteSelect == 0x18) // Timer stuff again + { + memoryUpdateTAMA5Clock(); + gbDataTAMA5.mapperLSeconds = gbDataTAMA5.mapperSeconds; + gbDataTAMA5.mapperLMinutes = gbDataTAMA5.mapperMinutes; + gbDataTAMA5.mapperLHours = gbDataTAMA5.mapperHours; + gbDataTAMA5.mapperLDays = gbDataTAMA5.mapperDays; + gbDataTAMA5.mapperLMonths = gbDataTAMA5.mapperMonths; + gbDataTAMA5.mapperLYears = gbDataTAMA5.mapperYears; + gbDataTAMA5.mapperLControl = gbDataTAMA5.mapperControl; + + int seconds = (gbDataTAMA5.mapperLSeconds / 10)*16 + gbDataTAMA5.mapperLSeconds %10; + int secondsL = (gbDataTAMA5.mapperLSeconds % 10); + int secondsH = (gbDataTAMA5.mapperLSeconds / 10); + int minutes = (gbDataTAMA5.mapperLMinutes / 10)*16 + gbDataTAMA5.mapperLMinutes %10; + int hours = (gbDataTAMA5.mapperLHours / 10)*16 + gbDataTAMA5.mapperLHours %10; + int DaysL = gbDataTAMA5.mapperLDays % 10; + int DaysH = gbDataTAMA5.mapperLDays /10; + int MonthsL = gbDataTAMA5.mapperLMonths % 10; + int MonthsH = gbDataTAMA5.mapperLMonths / 10; + int Years3 = (gbDataTAMA5.mapperLYears / 100) % 10; + int Years4 = (gbDataTAMA5.mapperLYears / 1000); + + switch (data & 0x0f) + { + // I guess cases 0 and 1 are used for secondsL and secondsH + // so the game would update the timer values on screen when + // the seconds reset to 0... ? + case 0x0: + gbTAMA5ram[gbDataTAMA5.mapperRamByteSelect] = secondsL; + break; + case 0x1: + gbTAMA5ram[gbDataTAMA5.mapperRamByteSelect] = secondsH; + break; + case 0x7: + gbTAMA5ram[gbDataTAMA5.mapperRamByteSelect] = DaysL; // days low + break; + case 0x8: + gbTAMA5ram[gbDataTAMA5.mapperRamByteSelect] = DaysH; // days high + break; + case 0x9: + gbTAMA5ram[gbDataTAMA5.mapperRamByteSelect] = MonthsL; // month low + break; + case 0xa: + gbTAMA5ram[gbDataTAMA5.mapperRamByteSelect] = MonthsH; // month high + break; + case 0xb: + gbTAMA5ram[gbDataTAMA5.mapperRamByteSelect] = Years4; // years 4th digit + break; + case 0xc: + gbTAMA5ram[gbDataTAMA5.mapperRamByteSelect] = Years3; // years 3rd digit + break; + default : + break; + } + + gbTAMA5ram[0x54] = seconds; // incorrect ? (not used by the game) ? + gbTAMA5ram[0x64] = minutes; + gbTAMA5ram[0x74] = hours; + gbTAMA5ram[0x84] = DaysH*16+DaysL; // incorrect ? (not used by the game) ? + gbTAMA5ram[0x94] = MonthsH*16+MonthsL; // incorrect ? (not used by the game) ? + + time(&gbDataTAMA5.mapperLastTime); + + gbMemoryMap[0xa][0] = 1; + } + else if (gbDataTAMA5.mapperRamByteSelect == 0x28) // Timer stuff again + { + if ((data & 0xf) == 0xb) + gbDataTAMA5.mapperYears = ((gbDataTAMA5.mapperYears>>2)<<2) + (data & 3); + } + else if (gbDataTAMA5.mapperRamByteSelect == 0x44) + { + gbDataTAMA5.mapperMinutes = (data/16)*10 + data%16; + } + else if (gbDataTAMA5.mapperRamByteSelect == 0x54) + { + gbDataTAMA5.mapperHours = (data/16)*10 + data%16; + } + else + { + gbTAMA5ram[gbDataTAMA5.mapperRamByteSelect] = data; + } + } + } + } + break; + case 1: // 'Commands' Register + { + gbMemoryMap[0xa][1] = gbDataTAMA5.mapperCommandNumber = value; + + // This should be only a 'is the flashrom ready ?' command. + // However as I couldn't find any 'copy' command + // (that seems to be needed for the saving system to work) + // I put it there... + if (value == 0x0a) + { + for (int i = 0; i<0x10; i++) + for (int j = 0; j<0x10; j++) + if (!(j&2)) + gbTAMA5ram[(i*0x10)+j | 2] = gbTAMA5ram[(i*0x10)+j]; + // Enable this to see the content of the flashrom in 0xe000 + /*for (int k = 0; k<0x100; k++) + gbMemoryMap[0xe][k] = gbTAMA5ram[k];*/ + + gbMemoryMap[0xa][0] = gbDataTAMA5.mapperRAMEnable = 1; + } + else + { + if ((value & 0x0e) == 0x0c) + { + gbDataTAMA5.mapperRamByteSelect = gbDataTAMA5.mapperCommands[6] | + (gbDataTAMA5.mapperCommands[7]<<4); + + u8 byte = gbTAMA5ram[gbDataTAMA5.mapperRamByteSelect]; + + gbMemoryMap[0xa][0] = (value & 1) ? byte >> 4 : byte & 0x0f; + + gbDataTAMA5.mapperCommands[0x0f] = 0; + } + } + break; + } + } + } + else + { + if(gbDataTAMA5.mapperRAMEnable) { + if(gbDataTAMA5.mapperRAMBank != -1) { + if(gbRamSize) { + gbMemoryMap[address>>12][address & 0x0fff] = value; + systemSaveUpdateCounter = SYSTEM_SAVE_UPDATED; + } + } + } + } +} + + +// TAMA5 read RAM +u8 mapperTAMA5ReadRAM(u16 address) +{ + return gbMemoryMap[address>>12][address & 0xfff]; +} + + +void memoryUpdateMapTAMA5() +{ + int tmpAddress = (gbDataTAMA5.mapperROMBank << 14); + + tmpAddress &= gbRomSizeMask; + gbMemoryMap[0x04] = &gbRom[tmpAddress]; + gbMemoryMap[0x05] = &gbRom[tmpAddress + 0x1000]; + gbMemoryMap[0x06] = &gbRom[tmpAddress + 0x2000]; + gbMemoryMap[0x07] = &gbRom[tmpAddress + 0x3000]; + + if(gbRamSize) { + tmpAddress = 0 << 13; + tmpAddress &= gbRamSizeMask; + gbMemoryMap[0x0a] = &gbRam[tmpAddress]; + gbMemoryMap[0x0b] = &gbRam[tmpAddress + 0x1000]; + } +} + +// MMM01 Used in Momotarou collection (however the rom is corrupted) +mapperMMM01 gbDataMMM01 ={ + 0, // RAM enable + 1, // ROM bank + 0, // RAM bank + 0, // memory model + 0, // ROM high address + 0, // RAM address + 0 // Rom Bank 0 remapping +}; + +// MMM01 ROM write registers +void mapperMMM01ROM(u16 address, u8 value) +{ + int tmpAddress = 0; + + switch(address & 0x6000) { + case 0x0000: // RAM enable register + gbDataMMM01.mapperRAMEnable = ( ( value & 0x0a) == 0x0a ? 1 : 0); + break; + case 0x2000: // ROM bank select + // value = value & 0x1f; + if(value == 0) + value = 1; + if(value == gbDataMMM01.mapperROMBank) + break; + + tmpAddress = value << 14; + + // check current model + if(gbDataMMM01.mapperMemoryModel == 0) { + // model is 16/8, so we have a high address in use + tmpAddress |= (gbDataMMM01.mapperROMHighAddress) << 19; + } + else + tmpAddress |= gbDataMMM01.mapperRomBank0Remapping << 18; + + tmpAddress &= gbRomSizeMask; + gbDataMMM01.mapperROMBank = value; + gbMemoryMap[0x04] = &gbRom[tmpAddress]; + gbMemoryMap[0x05] = &gbRom[tmpAddress + 0x1000]; + gbMemoryMap[0x06] = &gbRom[tmpAddress + 0x2000]; + gbMemoryMap[0x07] = &gbRom[tmpAddress + 0x3000]; + break; + case 0x4000: // RAM bank select + if(gbDataMMM01.mapperMemoryModel == 1) { + // 4/32 model, RAM bank switching provided + value = value & 0x03; + if(value == gbDataMBC1.mapperRAMBank) + break; + tmpAddress = value << 13; + tmpAddress &= gbRamSizeMask; + gbMemoryMap[0x0a] = &gbRam[tmpAddress]; + gbMemoryMap[0x0b] = &gbRam[tmpAddress + 0x1000]; + gbDataMMM01.mapperRAMBank = value; + gbDataMMM01.mapperRAMAddress = tmpAddress; + } else { + // 16/8, set the high address + gbDataMMM01.mapperROMHighAddress = value & 0x03; + tmpAddress = gbDataMMM01.mapperROMBank << 14; + tmpAddress |= (gbDataMMM01.mapperROMHighAddress) << 19; + tmpAddress &= gbRomSizeMask; + gbMemoryMap[0x04] = &gbRom[tmpAddress]; + gbMemoryMap[0x05] = &gbRom[tmpAddress + 0x1000]; + gbMemoryMap[0x06] = &gbRom[tmpAddress + 0x2000]; + gbMemoryMap[0x07] = &gbRom[tmpAddress + 0x3000]; + + gbDataMMM01.mapperRomBank0Remapping = ((value<<1) | (value & 0x40 ? 1 : 0)) & 0xff; + tmpAddress = gbDataMMM01.mapperRomBank0Remapping << 18; + tmpAddress &= gbRomSizeMask; + gbMemoryMap[0x00] = &gbRom[tmpAddress]; + gbMemoryMap[0x01] = &gbRom[tmpAddress + 0x1000]; + gbMemoryMap[0x02] = &gbRom[tmpAddress + 0x2000]; + gbMemoryMap[0x03] = &gbRom[tmpAddress + 0x3000]; + } + break; + case 0x6000: // memory model select + gbDataMMM01.mapperMemoryModel = value & 1; + break; + } +} + +// MMM01 RAM write +void mapperMMM01RAM(u16 address, u8 value) +{ + if(gbDataMMM01.mapperRAMEnable) { + if(gbRamSize) { + gbMemoryMap[address >> 12][address & 0x0fff] = value; + systemSaveUpdateCounter = SYSTEM_SAVE_UPDATED; + } + } +} + +void memoryUpdateMapMMM01() +{ + int tmpAddress = gbDataMMM01.mapperROMBank << 14; + + // check current model + if(gbDataMMM01.mapperMemoryModel == 1) { + // model is 16/8, so we have a high address in use + tmpAddress |= (gbDataMMM01.mapperROMHighAddress) << 19; + } + + tmpAddress &= gbRomSizeMask; + gbMemoryMap[0x04] = &gbRom[tmpAddress]; + gbMemoryMap[0x05] = &gbRom[tmpAddress + 0x1000]; + gbMemoryMap[0x06] = &gbRom[tmpAddress + 0x2000]; + gbMemoryMap[0x07] = &gbRom[tmpAddress + 0x3000]; + + tmpAddress = gbDataMMM01.mapperRomBank0Remapping << 18; + tmpAddress &= gbRomSizeMask; + gbMemoryMap[0x00] = &gbRom[tmpAddress]; + gbMemoryMap[0x01] = &gbRom[tmpAddress + 0x1000]; + gbMemoryMap[0x02] = &gbRom[tmpAddress + 0x2000]; + gbMemoryMap[0x03] = &gbRom[tmpAddress + 0x3000]; + + if(gbRamSize) { + gbMemoryMap[0x0a] = &gbRam[gbDataMMM01.mapperRAMAddress]; + gbMemoryMap[0x0b] = &gbRam[gbDataMMM01.mapperRAMAddress + 0x1000]; + } +} + +// GameGenie ROM write registers +void mapperGGROM(u16 address, u8 value) +{ + int tmpAddress = 0; + + switch(address & 0x6000) { + case 0x0000: // RAM enable register + break; + case 0x2000: // GameGenie has only a half bank + break; + case 0x4000: // GameGenie has no RAM + if ((address >=0x4001) && (address <= 0x4020)) // GG Hardware Registers + gbMemoryMap[address >> 12][address & 0x0fff] = value; + break; + case 0x6000: // GameGenie has only a half bank + break; + } +} + + +// GS3 Used to emulate the GS V3.0 rom bank switching +mapperGS3 gbDataGS3 = { 1 }; // ROM bank + +void mapperGS3ROM(u16 address, u8 value) +{ + int tmpAddress = 0; + + switch(address & 0x6000) { + case 0x0000: // GS has no ram + break; + case 0x2000: // GS has no 'classic' ROM bank select + break; + case 0x4000: // GS has no ram + break; + case 0x6000: // 0x6000 area is RW, and used for GS hardware registers + + if (address == 0x7FE1) // This is the (half) ROM bank select register + { + if(value == gbDataGS3.mapperROMBank) + break; + tmpAddress = value << 13; + + tmpAddress &= gbRomSizeMask; + gbDataGS3.mapperROMBank = value; + gbMemoryMap[0x04] = &gbRom[tmpAddress]; + gbMemoryMap[0x05] = &gbRom[tmpAddress + 0x1000]; + } + else + gbMemoryMap[address>>12][address & 0x0fff] = value; + break; + } +} + +void memoryUpdateMapGS3() +{ + int tmpAddress = gbDataGS3.mapperROMBank << 13; + + tmpAddress &= gbRomSizeMask; + // GS can only change a half ROM bank + gbMemoryMap[0x04] = &gbRom[tmpAddress]; + gbMemoryMap[0x05] = &gbRom[tmpAddress + 0x1000]; +} \ No newline at end of file diff --git a/src/gb/gbMemory.h b/src/gb/gbMemory.h index 9ce19edd..ef714d19 100644 --- a/src/gb/gbMemory.h +++ b/src/gb/gbMemory.h @@ -1,7 +1,7 @@ // -*- C++ -*- // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2004-2006 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -26,6 +26,7 @@ struct mapperMBC1 { int mapperMemoryModel; int mapperROMHighAddress; int mapperRAMAddress; + int mapperRomBank0Remapping; }; struct mapperMBC2 { @@ -106,15 +107,62 @@ struct mapperHuC3 { int mapperRegister8; }; +struct mapperTAMA5 { + int mapperRAMEnable; + int mapperROMBank; + int mapperRAMBank; + int mapperRAMAddress; + int mapperRamByteSelect; + int mapperCommandNumber; + int mapperLastCommandNumber; + int mapperCommands[0x10]; + int mapperRegister; + int mapperClockLatch; + int mapperClockRegister; + int mapperSeconds; + int mapperMinutes; + int mapperHours; + int mapperDays; + int mapperMonths; + int mapperYears; + int mapperControl; + int mapperLSeconds; + int mapperLMinutes; + int mapperLHours; + int mapperLDays; + int mapperLMonths; + int mapperLYears; + int mapperLControl; + time_t mapperLastTime; +}; + +struct mapperMMM01 { + int mapperRAMEnable; + int mapperROMBank; + int mapperRAMBank; + int mapperMemoryModel; + int mapperROMHighAddress; + int mapperRAMAddress; + int mapperRomBank0Remapping; +}; + +struct mapperGS3 { + int mapperROMBank; +}; + extern mapperMBC1 gbDataMBC1; extern mapperMBC2 gbDataMBC2; extern mapperMBC3 gbDataMBC3; extern mapperMBC5 gbDataMBC5; extern mapperHuC1 gbDataHuC1; extern mapperHuC3 gbDataHuC3; +extern mapperTAMA5 gbDataTAMA5; +extern mapperMMM01 gbDataMMM01; +extern mapperGS3 gbDataGS3; void mapperMBC1ROM(u16,u8); void mapperMBC1RAM(u16,u8); +u8 mapperMBC1ReadRAM(u16); void mapperMBC2ROM(u16,u8); void mapperMBC2RAM(u16,u8); void mapperMBC3ROM(u16,u8); @@ -122,6 +170,7 @@ void mapperMBC3RAM(u16,u8); u8 mapperMBC3ReadRAM(u16); void mapperMBC5ROM(u16,u8); void mapperMBC5RAM(u16,u8); +u8 mapperMBC5ReadRAM(u16); void mapperMBC7ROM(u16,u8); void mapperMBC7RAM(u16,u8); u8 mapperMBC7ReadRAM(u16); @@ -130,7 +179,13 @@ void mapperHuC1RAM(u16,u8); void mapperHuC3ROM(u16,u8); void mapperHuC3RAM(u16,u8); u8 mapperHuC3ReadRAM(u16); - +void mapperTAMA5RAM(u16,u8); +u8 mapperTAMA5ReadRAM(u16); +void memoryUpdateTAMA5Clock(); +void mapperMMM01ROM(u16,u8); +void mapperMMM01RAM(u16,u8); +void mapperGGROM(u16,u8); +void mapperGS3ROM(u16,u8); //extern void (*mapper)(u16,u8); //extern void (*mapperRAM)(u16,u8); //extern u8 (*mapperReadRAM)(u16); @@ -142,7 +197,9 @@ extern void memoryUpdateMapMBC5(); extern void memoryUpdateMapMBC7(); extern void memoryUpdateMapHuC1(); extern void memoryUpdateMapHuC3(); - +extern void memoryUpdateMapTAMA5(); +extern void memoryUpdateMapMMM01(); +extern void memoryUpdateMapGS3(); diff --git a/src/gb/gbSGB.cpp b/src/gb/gbSGB.cpp index dc7a1d80..9c8469fc 100644 --- a/src/gb/gbSGB.cpp +++ b/src/gb/gbSGB.cpp @@ -1,917 +1,917 @@ -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#include -#include - -#include "../win32/stdafx.h" -#include "../win32/VBA.h" -#include "../System.h" -#include "../Port.h" -#include "../Util.h" -#include "GB.h" -#include "gbGlobals.h" - -extern u8 *pix; -extern bool speedup; - -#define GBSGB_NONE 0 -#define GBSGB_RESET 1 -#define GBSGB_PACKET_TRANSMIT 2 - -u8 *gbSgbBorderChar = NULL; -u8 *gbSgbBorder = NULL; - -int gbSgbCGBSupport = 0; -int gbSgbMask = 0; -int gbSgbMode = 0; -int gbSgbPacketState = GBSGB_NONE; -int gbSgbBit = 0; -int gbSgbPacketTimeout = 0; -int GBSGB_PACKET_TIMEOUT = 66666; -u8 gbSgbPacket[16*7]; -int gbSgbPacketNBits = 0; -int gbSgbPacketByte = 0; -int gbSgbPacketNumber = 0; -int gbSgbMultiplayer = 0; -int gbSgbFourPlayers = 0; -u8 gbSgbNextController = 0x0f; -u8 gbSgbReadingController = 0; -u16 gbSgbSCPPalette[4*512]; -u8 gbSgbATF[20 * 18]; -u8 gbSgbATFList[45 * 20 * 18]; -u8 gbSgbScreenBuffer[4160]; - -inline void gbSgbDraw24Bit(u8 *p, u16 v) -{ - *((u32*) p) = systemColorMap32[v]; -} - -inline void gbSgbDraw32Bit(u32 *p, u16 v) -{ - *p = systemColorMap32[v]; -} - -inline void gbSgbDraw16Bit(u16 *p, u16 v) -{ - *p = systemColorMap16[v]; -} - -void gbSgbReset() -{ - gbSgbPacketTimeout = 0; - gbSgbCGBSupport = 0; - gbSgbMask = 0; - gbSgbPacketState = GBSGB_NONE; - gbSgbBit = 0; - gbSgbPacketNBits = 0; - gbSgbPacketNumber = 0; - gbSgbMultiplayer = 0; - gbSgbFourPlayers = 0; - gbSgbNextController = 0x0f; - gbSgbReadingController = 0; - - memset(gbSgbSCPPalette, 0, 512*4); - memset(gbSgbATF, 0, 20*18); - memset(gbSgbATFList, 0, 45 * 20 * 18); - memset(gbSgbPacket, 0, 16 * 7); - memset(gbSgbBorderChar, 0, 32*256); - memset(gbSgbBorder, 0, 2048); - - int i; - for(i = 1; i < 2048; i+=2) { - gbSgbBorder[i] = 1 << 2; - } - - for(i = 0; i < 4; i++) { - gbPalette[i*4] = (0x1f) | (0x1f << 5) | (0x1f << 10); - gbPalette[i*4+1] = (0x15) | (0x15 << 5) | (0x15 << 10); - gbPalette[i*4+2] = (0x0c) | (0x0c << 5) | (0x0c << 10); - gbPalette[i*4+3] = 0; - } -} - -void gbSgbInit() -{ - gbSgbBorderChar = (u8 *)malloc(32 * 256); - gbSgbBorder = (u8 *)malloc(2048); - - gbSgbReset(); -} - -void gbSgbShutdown() -{ - if(gbSgbBorderChar != NULL) { - free(gbSgbBorderChar); - gbSgbBorderChar = NULL; - } - - if(gbSgbBorder != NULL) { - free(gbSgbBorder); - gbSgbBorder = NULL; - } -} - -void gbSgbFillScreen(u16 color) -{ - switch(systemColorDepth) { - case 16: - { - for(int y = 0; y < 144; y++) { - int yLine = (y+gbBorderRowSkip+1)*(gbBorderLineSkip+2) + - gbBorderColumnSkip; - u16 *dest = (u16*)pix + yLine; - for(register int x = 0; x < 160; x++) - gbSgbDraw16Bit(dest++, color); - } - } - break; - case 24: - { - for(int y = 0; y < 144; y++) { - int yLine = (y+gbBorderRowSkip)*gbBorderLineSkip + gbBorderColumnSkip; - u8 *dest = (u8 *)pix + yLine*3; - for(register int x = 0; x < 160; x++) { - gbSgbDraw24Bit(dest, color); - dest += 3; - } - } - } - break; - case 32: - { - for(int y = 0; y < 144; y++) { - int yLine = (y+gbBorderRowSkip+1)*(gbBorderLineSkip+1) + gbBorderColumnSkip; - u32 *dest = (u32 *)pix + yLine; - for(register int x = 0; x < 160; x++) { - gbSgbDraw32Bit(dest++, color); - } - } - } - break; - } -} - -#define getmem(x) gbMemoryMap[(x) >> 12][(x) & 0xfff] - -void gbSgbRenderScreenToBuffer() -{ - u16 mapAddress = 0x9800; - - if(register_LCDC & 0x08) - mapAddress = 0x9c00; - - u16 patternAddress = 0x8800; - - int flag = 1; - - if(register_LCDC & 0x10) { - patternAddress = 0x8000; - flag = 0; - } - - u8 *toAddress = gbSgbScreenBuffer; - - for(int i = 0; i < 13; i++) { - for(int j = 0; j < 20; j++) { - int tile = getmem(mapAddress); - mapAddress++; - - if(flag) { - if(tile > 127) - tile -= 128; - else - tile += 128; - } - for(int k = 0; k < 16; k++) - *toAddress++ = getmem(patternAddress + tile*16 + k); - } - mapAddress += 12; - } -} - -void gbSgbDrawBorderTile(int x, int y, int tile, int attr) -{ - u16 *dest = (u16*)pix + ((y+1) * (256+2)) + x; - u8 *dest8 = (u8*)pix + ((y*256)+x)*3; - u32 *dest32 = (u32*)pix + ((y+1)*257) + x; - - u8 *tileAddress = &gbSgbBorderChar[tile * 32]; - u8 *tileAddress2 = &gbSgbBorderChar[tile * 32 + 16]; - - u8 l = 8; - - u8 palette = ((attr >> 2 ) & 7); - - if(palette < 4) - palette += 4; - - palette *= 16; - - u8 xx = 0; - u8 yy = 0; - - int flipX = attr & 0x40; - int flipY = attr & 0x80; - - while(l > 0) { - u8 mask = 0x80; - u8 a = *tileAddress++; - u8 b = *tileAddress++; - u8 c = *tileAddress2++; - u8 d = *tileAddress2++; - - while(mask > 0) { - - u8 color = 0; - if(a & mask) - color++; - if(b & mask) - color+=2; - if(c & mask) - color+=4; - if(d & mask) - color+=8; - - u8 xxx = xx; - u8 yyy = yy; - - if(flipX) - xxx = 7 - xx; - if(flipY) - yyy = 7 - yy; - - u16 c = gbPalette[palette + color]; - if(!color) - c = gbPalette[0]; - if((yy < 40 || yy >= 184) || (xx < 48 || xx >= 208)) { - switch(systemColorDepth) { - case 16: - gbSgbDraw16Bit(dest + yyy*(256+2) + xxx, c); - break; - case 24: - gbSgbDraw24Bit(dest8 + (yyy*256+xxx)*3, c); - break; - case 32: - gbSgbDraw32Bit(dest32 + yyy*(256+1)+xxx, c); - break; - } - } - - mask >>= 1; - - xx++; - } - yy++; - xx = 0; - l--; - mask = 0x80; - } -} - -void gbSgbRenderBorder() -{ - if(gbBorderOn) { - if (theApp.filterLCD) utilUpdateSystemColorMaps(0); - u8 *fromAddress = gbSgbBorder; - - for(u8 y = 0; y < 28; y++) { - for(u8 x = 0; x< 32; x++) { - u8 tile = *fromAddress++; - u8 attr = *fromAddress++; - - gbSgbDrawBorderTile(x*8,y*8,tile,attr); - } - } - if (theApp.filterLCD) utilUpdateSystemColorMaps(1); - } -} - -void gbSgbPicture() -{ - gbSgbRenderScreenToBuffer(); - - memcpy(gbSgbBorder, gbSgbScreenBuffer, 2048); - - u16 *paletteAddr = (u16 *)&gbSgbScreenBuffer[2048]; - - for(int i = 64; i < 128; i++) { - gbPalette[i] = READ16LE(paletteAddr++); - } - - gbSgbCGBSupport |= 4; - - if(gbBorderAutomatic && !gbBorderOn && gbSgbCGBSupport > 4) { - gbBorderOn = 1; - systemGbBorderOn(); - } - - if(gbBorderOn && !gbSgbMask) - gbSgbRenderBorder(); - - if(gbSgbMode && gbCgbMode && gbSgbCGBSupport > 4) { - gbSgbCGBSupport = 0; - gbSgbMode = 0; - gbSgbMask = 0; - gbSgbRenderBorder(); - gbReset(); - } - - if(gbSgbCGBSupport > 4) - gbSgbCGBSupport = 0; -} - -void gbSgbSetPalette(int a,int b,u16 *p) -{ - u16 bit00 = READ16LE(p++); - int i; - - for(i = 1; i < 4; i++) { - gbPalette[a*4+i] = READ16LE(p++); - } - - for(i = 1; i < 4; i++) { - gbPalette[b*4+i] = READ16LE(p++); - } - - gbPalette[0] = gbPalette[4] = gbPalette[8] = gbPalette[12] = bit00; - if(gbBorderOn && !gbSgbMask) - gbSgbRenderBorder(); -} - -void gbSgbScpPalette() -{ - gbSgbRenderScreenToBuffer(); - - u16 *fromAddress = (u16 *)gbSgbScreenBuffer; - - for(int i = 0; i < 512*4; i++) { - gbSgbSCPPalette[i] = READ16LE(fromAddress++); - } -} - -void gbSgbSetATF(int n) -{ - if(n < 0) - n = 0; - if(n > 44) - n = 44; - memcpy(gbSgbATF,&gbSgbATFList[n * 20 * 18], 20 * 18); - - if(gbSgbPacket[1] & 0x40) { - gbSgbMask = 0; - if(gbBorderOn) - gbSgbRenderBorder(); - } -} - -void gbSgbSetPalette() -{ - u16 pal = READ16LE((((u16 *)&gbSgbPacket[1])))&511; - memcpy(&gbPalette[0], &gbSgbSCPPalette[pal*4], 4 * sizeof(u16)); - - pal = READ16LE((((u16 *)&gbSgbPacket[3])))&511; - memcpy(&gbPalette[4], &gbSgbSCPPalette[pal*4], 4 * sizeof(u16)); - - pal = READ16LE((((u16 *)&gbSgbPacket[5])))&511; - memcpy(&gbPalette[8], &gbSgbSCPPalette[pal*4], 4 * sizeof(u16)); - - pal = READ16LE((((u16 *)&gbSgbPacket[7])))&511; - memcpy(&gbPalette[12], &gbSgbSCPPalette[pal*4], 4 * sizeof(u16)); - - u8 atf = gbSgbPacket[9]; - - if(atf & 0x80) { - gbSgbSetATF(atf & 0x3f); - } - - if(atf & 0x40) { - gbSgbMask = 0; - if(gbBorderOn) - gbSgbRenderBorder(); - } -} - -void gbSgbAttributeBlock() -{ - u8 *fromAddress = &gbSgbPacket[1]; - - u8 nDataSet = *fromAddress++; - if(nDataSet > 12) - nDataSet = 12; - if(nDataSet == 0) - nDataSet = 1; - - while(nDataSet) { - u8 controlCode = (*fromAddress++) & 7; - u8 paletteDesignation = (*fromAddress++) & 0x3f; - u8 startH = (*fromAddress++) & 0x1f; - u8 startV = (*fromAddress++) & 0x1f; - u8 endH = (*fromAddress++) & 0x1f; - u8 endV = (*fromAddress++) & 0x1f; - - u8 * toAddress = gbSgbATF; - - for(u8 y = 0; y < 18; y++) { - for(u8 x = 0; x < 20; x++) { - if(x < startH || y < startV || - x > endH || y > endV) { - // outside - if(controlCode & 0x04) - *toAddress = (paletteDesignation >> 4) & 0x03; - } else if(x > startH && x < endH && - y > startV && y < endV) { - // inside - if(controlCode & 0x01) - *toAddress = paletteDesignation & 0x03; - } else { - // surrounding line - if(controlCode & 0x02) - *toAddress = (paletteDesignation>>2) & 0x03; - else if(controlCode == 0x01) - *toAddress = paletteDesignation & 0x03; - } - toAddress++; - } - } - nDataSet--; - } -} - -void gbSgbSetColumnPalette(u8 col, u8 p) -{ - // if(col < 0) - // col = 0; - if(col > 19) - col = 19; - - p &= 3; - - u8 *toAddress = &gbSgbATF[col]; - - for(u8 y = 0; y < 18; y++) { - *toAddress = p; - toAddress += 20; - } -} - -void gbSgbSetRowPalette(u8 row, u8 p) -{ - // if(row < 0) - // row = 0; - if(row > 17) - row = 17; - - p &= 3; - - u8 *toAddress = &gbSgbATF[row*20]; - - for(u8 x = 0; x < 20; x++) { - *toAddress++ = p; - } -} - -void gbSgbAttributeDivide() -{ - u8 control = gbSgbPacket[1]; - u8 coord = gbSgbPacket[2]; - u8 colorBR = control & 3; - u8 colorAL = (control >> 2) & 3; - u8 colorOL = (control >> 4) & 3; - - if(control & 0x40) { - if(coord > 17) - coord = 17; - - for(u8 i = 0; i < 18; i++) { - if(i < coord) - gbSgbSetRowPalette(i, colorAL); - else if ( i > coord) - gbSgbSetRowPalette(i, colorBR); - else - gbSgbSetRowPalette(i, colorOL); - } - } else { - if(coord > 19) - coord = 19; - - for(u8 i = 0; i < 20; i++) { - if(i < coord) - gbSgbSetColumnPalette(i, colorAL); - else if ( i > coord) - gbSgbSetColumnPalette(i, colorBR); - else - gbSgbSetColumnPalette(i, colorOL); - } - } -} - -void gbSgbAttributeLine() -{ - u8 *fromAddress = &gbSgbPacket[1]; - - u8 nDataSet = *fromAddress++; - - if(nDataSet > 0x6e) - nDataSet = 0x6e; - - while(nDataSet) { - u8 line = *fromAddress++; - u8 num = line & 0x1f; - u8 pal = (line >> 5) & 0x03; - if(line & 0x80) { - if(num > 17) - num = 17; - gbSgbSetRowPalette(num,pal); - } else { - if(num > 19) - num = 19; - gbSgbSetColumnPalette(num,pal); - } - nDataSet--; - } -} - -void gbSgbAttributeCharacter() -{ - u8 startH = gbSgbPacket[1] & 0x1f; - u8 startV = gbSgbPacket[2] & 0x1f; - int nDataSet = READ16LE(((u16 *)&gbSgbPacket[3])); - int style = gbSgbPacket[5] & 1; - if(startH > 19) - startH = 19; - if(startV > 17) - startV = 17; - - u8 s = 6; - u8 *fromAddress = &gbSgbPacket[6]; - u8 v = *fromAddress++; - - if(style) { - while(nDataSet) { - u8 p = (v >> s) & 3; - gbSgbATF[startV * 20 + startH] = p; - startV++; - if(startV == 18) { - startV = 0; - startH++; - if(startH == 20) - break; - } - - if(s) - s -= 2; - else { - s = 6; - v = *fromAddress++; - nDataSet--; - } - } - } else { - while(nDataSet) { - u8 p = (v >> s) & 3; - gbSgbATF[startV * 20 + startH] = p; - startH++; - if(startH == 20) { - startH = 0; - startV++; - if(startV == 18) - break; - } - - if(s) - s -= 2; - else { - s = 6; - v = *fromAddress++; - nDataSet--; - } - } - } -} - -void gbSgbSetATFList() -{ - gbSgbRenderScreenToBuffer(); - - u8 *fromAddress = gbSgbScreenBuffer; - u8 *toAddress = gbSgbATFList; - - for(int i = 0; i < 45; i++) { - for(int j = 0; j < 90; j++) { - u8 v = *fromAddress++; - u8 s = 6; - if(i == 2) - s = 6; - for(int k = 0; k < 4; k++) { - *toAddress++ = (v >> s) & 0x03; - s -= 2; - } - } - } -} - -void gbSgbMaskEnable() -{ - int gbSgbMaskFlag = gbSgbPacket[1] & 3; - - gbSgbMask = gbSgbMaskFlag; - - switch(gbSgbMaskFlag) { - case 1: - break; - case 2: - gbSgbFillScreen(0x0000); - // memset(&gbPalette[0], 0, 128*sizeof(u16)); - break; - case 3: - gbSgbFillScreen(gbPalette[0]); - break; - } - if(!gbSgbMask) { - if(gbBorderOn) - gbSgbRenderBorder(); - } -} - -void gbSgbChrTransfer() -{ - gbSgbRenderScreenToBuffer(); - - int address = (gbSgbPacket[1] & 1) * (128*32); - - if(gbSgbPacket[1] & 1) - gbSgbCGBSupport |= 2; - else - gbSgbCGBSupport |= 1; - - memcpy(&gbSgbBorderChar[address], gbSgbScreenBuffer, 128 * 32); - - if(gbBorderAutomatic && !gbBorderOn && gbSgbCGBSupport > 4) { - gbBorderOn = 1; - systemGbBorderOn(); - } - - if(gbBorderOn && !gbSgbMask) - gbSgbRenderBorder(); - - if(gbSgbMode && gbCgbMode && gbSgbCGBSupport == 7) { - gbSgbCGBSupport = 0; - gbSgbMode = 0; - gbSgbMask = 0; - gbSgbRenderBorder(); - gbReset(); - } - - if(gbSgbCGBSupport > 4) - gbSgbCGBSupport = 0; -} - -void gbSgbMultiRequest() -{ - if(gbSgbPacket[1] & 1) { - gbSgbMultiplayer = 1; - if(gbSgbPacket[1] & 2) - gbSgbFourPlayers = 1; - else - gbSgbFourPlayers = 0; - gbSgbNextController = 0x0e; - } else { - gbSgbFourPlayers = 0; - gbSgbMultiplayer = 0; - gbSgbNextController = 0x0f; - } -} - -void gbSgbCommand() -{ - int command = gbSgbPacket[0] >> 3; - // int nPacket = gbSgbPacket[0] & 7; - - switch(command) { - case 0x00: - gbSgbSetPalette(0,1,(u16 *)&gbSgbPacket[1]); - break; - case 0x01: - gbSgbSetPalette(2,3,(u16 *)&gbSgbPacket[1]); - break; - case 0x02: - gbSgbSetPalette(0,3,(u16 *)&gbSgbPacket[1]); - break; - case 0x03: - gbSgbSetPalette(1,2,(u16 *)&gbSgbPacket[1]); - break; - case 0x04: - gbSgbAttributeBlock(); - break; - case 0x05: - gbSgbAttributeLine(); - break; - case 0x06: - gbSgbAttributeDivide(); - break; - case 0x07: - gbSgbAttributeCharacter(); - break; - case 0x0a: - gbSgbSetPalette(); - break; - case 0x0b: - gbSgbScpPalette(); - break; - case 0x11: - gbSgbMultiRequest(); - break; - case 0x13: - gbSgbChrTransfer(); - break; - case 0x14: - gbSgbPicture(); - break; - case 0x15: - gbSgbSetATFList(); - break; - case 0x16: - gbSgbSetATF(gbSgbPacket[1] & 0x3f); - break; - case 0x17: - gbSgbMaskEnable(); - break; - } -} - -void gbSgbResetPacketState() -{ - gbSgbPacketState = GBSGB_NONE; - gbSgbPacketTimeout = 0; -} - -void gbSgbDoBitTransfer(u8 value) -{ - value = value & 0x30; - switch(gbSgbPacketState) { - case GBSGB_NONE: - if(value == 0) { - gbSgbPacketState = GBSGB_RESET; - gbSgbPacketTimeout = GBSGB_PACKET_TIMEOUT; - } else if (value == 0x30) { - if(gbSgbMultiplayer) { - if((gbSgbReadingController & 7) == 7) { - gbSgbReadingController = 0; - if(gbSgbMultiplayer) { - gbSgbNextController--; - if(gbSgbFourPlayers) { - if(gbSgbNextController == 0x0b) - gbSgbNextController = 0x0f; - } else { - if(gbSgbNextController == 0x0d) - gbSgbNextController = 0x0f; - } - } - } else { - gbSgbReadingController &= 3; - } - } - gbSgbPacketTimeout = 0; - } else { - if(value == 0x10) - gbSgbReadingController |= 0x2; - else if(value == 0x20) - gbSgbReadingController |= 0x01; - gbSgbPacketTimeout = 0; - } - gbSgbPacketTimeout = 0; - break; - case GBSGB_RESET: - if(value == 0x30) { - gbSgbPacketState = GBSGB_PACKET_TRANSMIT; - gbSgbPacketByte = 0; - gbSgbPacketNBits = 0; - gbSgbPacketTimeout = GBSGB_PACKET_TIMEOUT; - } else if(value == 0x00) { - gbSgbPacketTimeout = GBSGB_PACKET_TIMEOUT; - gbSgbPacketState = GBSGB_RESET; - } else { - gbSgbPacketState = GBSGB_NONE; - gbSgbPacketTimeout = 0; - } - break; - case GBSGB_PACKET_TRANSMIT: - if(value == 0) { - gbSgbPacketState = GBSGB_RESET; - gbSgbPacketTimeout = 0; - } else if (value == 0x30){ - if(gbSgbPacketNBits == 128) { - gbSgbPacketNBits = 0; - gbSgbPacketByte = 0; - gbSgbPacketNumber++; - gbSgbPacketTimeout = 0; - if(gbSgbPacketNumber == (gbSgbPacket[0] & 7)) { - gbSgbCommand(); - gbSgbPacketNumber = 0; - gbSgbPacketState = GBSGB_NONE; - gbSgbPacketTimeout = 0; - } - } else { - if(gbSgbPacketNBits < 128) { - gbSgbPacket[gbSgbPacketNumber * 16 + gbSgbPacketByte] >>= 1; - gbSgbPacket[gbSgbPacketNumber * 16 + gbSgbPacketByte] |= gbSgbBit; - gbSgbPacketNBits++; - if(!(gbSgbPacketNBits & 7)) { - gbSgbPacketByte++; - } - gbSgbPacketTimeout = GBSGB_PACKET_TIMEOUT; - } - } - } else { - if(value == 0x20) - gbSgbBit = 0x00; - else - gbSgbBit = 0x80; - gbSgbPacketTimeout = GBSGB_PACKET_TIMEOUT; - } - gbSgbReadingController = 0; - break; - default: - gbSgbPacketState = GBSGB_NONE; - gbSgbPacketTimeout = 0; - break; - } -} - -variable_desc gbSgbSaveStruct[] = { - { &gbSgbMask, sizeof(int) }, - { &gbSgbPacketState, sizeof(int) }, - { &gbSgbBit, sizeof(int) }, - { &gbSgbPacketNBits, sizeof(int) }, - { &gbSgbPacketByte, sizeof(int) }, - { &gbSgbPacketNumber, sizeof(int) }, - { &gbSgbMultiplayer, sizeof(int) }, - { &gbSgbNextController, sizeof(u8) }, - { &gbSgbReadingController, sizeof(u8) }, - { NULL, 0 } -}; - -variable_desc gbSgbSaveStructV3[] = { - { &gbSgbMask, sizeof(int) }, - { &gbSgbPacketState, sizeof(int) }, - { &gbSgbBit, sizeof(int) }, - { &gbSgbPacketNBits, sizeof(int) }, - { &gbSgbPacketByte, sizeof(int) }, - { &gbSgbPacketNumber, sizeof(int) }, - { &gbSgbMultiplayer, sizeof(int) }, - { &gbSgbNextController, sizeof(u8) }, - { &gbSgbReadingController, sizeof(u8) }, - { &gbSgbFourPlayers, sizeof(int) }, - { NULL, 0 } -}; - -void gbSgbSaveGame(gzFile gzFile) -{ - utilWriteData(gzFile, gbSgbSaveStructV3); - - utilGzWrite(gzFile, gbSgbBorder, 2048); - utilGzWrite(gzFile, gbSgbBorderChar, 32*256); - - utilGzWrite(gzFile, gbSgbPacket, 16*7); - - utilGzWrite(gzFile, gbSgbSCPPalette, 4 * 512 * sizeof(u16)); - utilGzWrite(gzFile, gbSgbATF, 20 * 18); - utilGzWrite(gzFile, gbSgbATFList, 45 * 20 * 18); -} - -void gbSgbReadGame(gzFile gzFile, int version) -{ - if(version >= 3) - utilReadData(gzFile, gbSgbSaveStructV3); - else { - utilReadData(gzFile, gbSgbSaveStruct); - gbSgbFourPlayers = 0; - } - - if(version >= 8) { - utilGzRead(gzFile, gbSgbBorder, 2048); - utilGzRead(gzFile, gbSgbBorderChar, 32*256); - } - - utilGzRead(gzFile, gbSgbPacket, 16*7); - - utilGzRead(gzFile, gbSgbSCPPalette, 4 * 512 * sizeof(u16)); - utilGzRead(gzFile, gbSgbATF, 20 * 18); - utilGzRead(gzFile, gbSgbATFList, 45 * 20 * 18); -} +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2005-2006 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#include +#include + +#include "../System.h" +#include "../Port.h" +#include "../Util.h" +#include "GB.h" +#include "gbGlobals.h" + +extern u8 *pix; +extern bool speedup; +extern bool gbSgbResetFlag; + +#define GBSGB_NONE 0 +#define GBSGB_RESET 1 +#define GBSGB_PACKET_TRANSMIT 2 + +u8 *gbSgbBorderChar = NULL; +u8 *gbSgbBorder = NULL; + +int gbSgbCGBSupport = 0; +int gbSgbMask = 0; +int gbSgbMode = 0; +int gbSgbPacketState = GBSGB_NONE; +int gbSgbBit = 0; +int gbSgbPacketTimeout = 0; +int GBSGB_PACKET_TIMEOUT = 66666; +u8 gbSgbPacket[16*7]; +int gbSgbPacketNBits = 0; +int gbSgbPacketByte = 0; +int gbSgbPacketNumber = 0; +int gbSgbMultiplayer = 0; +int gbSgbFourPlayers = 0; +u8 gbSgbNextController = 0x0f; +u8 gbSgbReadingController = 0; +u16 gbSgbSCPPalette[4*512]; +u8 gbSgbATF[20 * 18]; +u8 gbSgbATFList[45 * 20 * 18]; +u8 gbSgbScreenBuffer[4160]; + +inline void gbSgbDraw24Bit(u8 *p, u16 v) +{ + *((u32*) p) = systemColorMap32[v]; +} + +inline void gbSgbDraw32Bit(u32 *p, u16 v) +{ + *p = systemColorMap32[v]; +} + +inline void gbSgbDraw16Bit(u16 *p, u16 v) +{ + *p = systemColorMap16[v]; +} + +void gbSgbReset() +{ + gbSgbPacketTimeout = 0; + gbSgbCGBSupport = 0; + gbSgbMask = 0; + gbSgbPacketState = GBSGB_NONE; + gbSgbBit = 0; + gbSgbPacketNBits = 0; + gbSgbPacketNumber = 0; + gbSgbMultiplayer = 0; + gbSgbFourPlayers = 0; + gbSgbNextController = 0x0f; + gbSgbReadingController = 0; + + memset(gbSgbSCPPalette, 0, 512*4); + memset(gbSgbATF, 0, 20*18); + memset(gbSgbATFList, 0, 45 * 20 * 18); + memset(gbSgbPacket, 0, 16 * 7); + memset(gbSgbBorderChar, 0, 32*256); + memset(gbSgbBorder, 0, 2048); + + int i; + for(i = 1; i < 2048; i+=2) { + gbSgbBorder[i] = 1 << 2; + } + + for(i = 0; i < 32; i++) { + gbPalette[i*4] = (0x1f) | (0x1f << 5) | (0x1f << 10); + gbPalette[i*4+1] = (0x15) | (0x15 << 5) | (0x15 << 10); + gbPalette[i*4+2] = (0x0c) | (0x0c << 5) | (0x0c << 10); + gbPalette[i*4+3] = 0; + } +} + +void gbSgbInit() +{ + gbSgbBorderChar = (u8 *)malloc(32 * 256); + gbSgbBorder = (u8 *)malloc(2048); + + gbSgbReset(); +} + +void gbSgbShutdown() +{ + if(gbSgbBorderChar != NULL) { + free(gbSgbBorderChar); + gbSgbBorderChar = NULL; + } + + if(gbSgbBorder != NULL) { + free(gbSgbBorder); + gbSgbBorder = NULL; + } +} + +void gbSgbFillScreen(u16 color) +{ + switch(systemColorDepth) { + case 16: + { + for(int y = 0; y < 144; y++) { + int yLine = (y+gbBorderRowSkip+1)*(gbBorderLineSkip+2) + + gbBorderColumnSkip; + u16 *dest = (u16*)pix + yLine; + for(register int x = 0; x < 160; x++) + gbSgbDraw16Bit(dest++, color); + } + } + break; + case 24: + { + for(int y = 0; y < 144; y++) { + int yLine = (y+gbBorderRowSkip)*gbBorderLineSkip + gbBorderColumnSkip; + u8 *dest = (u8 *)pix + yLine*3; + for(register int x = 0; x < 160; x++) { + gbSgbDraw24Bit(dest, color); + dest += 3; + } + } + } + break; + case 32: + { + for(int y = 0; y < 144; y++) { + int yLine = (y+gbBorderRowSkip+1)*(gbBorderLineSkip+1) + gbBorderColumnSkip; + u32 *dest = (u32 *)pix + yLine; + for(register int x = 0; x < 160; x++) { + gbSgbDraw32Bit(dest++, color); + } + } + } + break; + } +} + +#define getmem(x) gbMemoryMap[(x) >> 12][(x) & 0xfff] + +void gbSgbRenderScreenToBuffer() +{ + u16 mapAddress = 0x9800; + + if(register_LCDC & 0x08) + mapAddress = 0x9c00; + + u16 patternAddress = 0x8800; + + int flag = 1; + + if(register_LCDC & 0x10) { + patternAddress = 0x8000; + flag = 0; + } + + u8 *toAddress = gbSgbScreenBuffer; + + for(int i = 0; i < 13; i++) { + for(int j = 0; j < 20; j++) { + int tile = getmem(mapAddress); + mapAddress++; + + if(flag) { + if(tile > 127) + tile -= 128; + else + tile += 128; + } + for(int k = 0; k < 16; k++) + *toAddress++ = getmem(patternAddress + tile*16 + k); + } + mapAddress += 12; + } +} + +void gbSgbDrawBorderTile(int x, int y, int tile, int attr) +{ + u16 *dest = (u16*)pix + ((y+1) * (256+2)) + x; + u8 *dest8 = (u8*)pix + ((y*256)+x)*3; + u32 *dest32 = (u32*)pix + ((y+1)*257) + x; + + u8 *tileAddress = &gbSgbBorderChar[tile * 32]; + u8 *tileAddress2 = &gbSgbBorderChar[tile * 32 + 16]; + + u8 l = 8; + + u8 palette = ((attr >> 2 ) & 7); + + if(palette < 4) + palette += 4; + + palette *= 16; + + u8 xx = 0; + u8 yy = 0; + + int flipX = attr & 0x40; + int flipY = attr & 0x80; + + while(l > 0) { + u8 mask = 0x80; + u8 a = *tileAddress++; + u8 b = *tileAddress++; + u8 c = *tileAddress2++; + u8 d = *tileAddress2++; + + while(mask > 0) { + + u8 color = 0; + if(a & mask) + color++; + if(b & mask) + color+=2; + if(c & mask) + color+=4; + if(d & mask) + color+=8; + + u8 xxx = xx; + u8 yyy = yy; + + if(flipX) + xxx = 7 - xx; + if(flipY) + yyy = 7 - yy; + + u16 c = gbPalette[palette + color]; + + // Fix for Super Snaky ??? + // (it allows SGB borders to not redraw on the GB screen) + //if(!color) + // c = gbPalette[0]; + if(((yy < 40 || yy >= 184) || (xx < 48 || xx >= 208)) && (color || (gbSgbResetFlag == true))) { + switch(systemColorDepth) { + case 16: + gbSgbDraw16Bit(dest + yyy*(256+2) + xxx, c); + break; + case 24: + gbSgbDraw24Bit(dest8 + (yyy*256+xxx)*3, c); + break; + case 32: + gbSgbDraw32Bit(dest32 + yyy*(256+1)+xxx, c); + break; + } + } + + mask >>= 1; + + xx++; + } + yy++; + xx = 0; + l--; + mask = 0x80; + } +} + +void gbSgbRenderBorder() +{ + if(gbBorderOn) { + u8 *fromAddress = gbSgbBorder; + + for(u8 y = 0; y < 28; y++) { + for(u8 x = 0; x< 32; x++) { + u8 tile = *fromAddress++; + u8 attr = *fromAddress++; + + gbSgbDrawBorderTile(x*8,y*8,tile,attr); + } + } + } +} + +void gbSgbPicture() +{ + gbSgbRenderScreenToBuffer(); + + memcpy(gbSgbBorder, gbSgbScreenBuffer, 2048); + + u16 *paletteAddr = (u16 *)&gbSgbScreenBuffer[2048]; + + for(int i = 64; i < 128; i++) { + gbPalette[i] = READ16LE(paletteAddr++); + } + + gbSgbCGBSupport |= 4; + + if(gbBorderAutomatic && !gbBorderOn && gbSgbCGBSupport > 4) { + gbBorderOn = 1; + systemGbBorderOn(); + } + + if(gbBorderOn && !gbSgbMask) + gbSgbRenderBorder(); + + if(gbSgbMode && gbCgbMode && gbSgbCGBSupport > 4) { + gbSgbCGBSupport = 0; + gbSgbMode = 0; + gbSgbMask = 0; + gbSgbRenderBorder(); + gbReset(); + } + + if(gbSgbCGBSupport > 4) + gbSgbCGBSupport = 0; +} + +void gbSgbSetPalette(int a,int b,u16 *p) +{ + u16 bit00 = READ16LE(p++); + int i; + + for(i = 1; i < 4; i++) { + gbPalette[a*4+i] = READ16LE(p++); + } + + for(i = 1; i < 4; i++) { + gbPalette[b*4+i] = READ16LE(p++); + } + + gbPalette[0] = gbPalette[4] = gbPalette[8] = gbPalette[12] = bit00; + if(gbBorderOn && !gbSgbMask) + gbSgbRenderBorder(); +} + +void gbSgbScpPalette() +{ + gbSgbRenderScreenToBuffer(); + + u16 *fromAddress = (u16 *)gbSgbScreenBuffer; + + for(int i = 0; i < 512*4; i++) { + gbSgbSCPPalette[i] = READ16LE(fromAddress++); + } +} + +void gbSgbSetATF(int n) +{ + if(n < 0) + n = 0; + if(n > 44) + n = 44; + memcpy(gbSgbATF,&gbSgbATFList[n * 20 * 18], 20 * 18); + + if(gbSgbPacket[1] & 0x40) { + gbSgbMask = 0; + if(gbBorderOn) + gbSgbRenderBorder(); + } +} + +void gbSgbSetPalette() +{ + u16 pal = READ16LE((((u16 *)&gbSgbPacket[1])))&511; + memcpy(&gbPalette[0], &gbSgbSCPPalette[pal*4], 4 * sizeof(u16)); + + pal = READ16LE((((u16 *)&gbSgbPacket[3])))&511; + memcpy(&gbPalette[4], &gbSgbSCPPalette[pal*4], 4 * sizeof(u16)); + + pal = READ16LE((((u16 *)&gbSgbPacket[5])))&511; + memcpy(&gbPalette[8], &gbSgbSCPPalette[pal*4], 4 * sizeof(u16)); + + pal = READ16LE((((u16 *)&gbSgbPacket[7])))&511; + memcpy(&gbPalette[12], &gbSgbSCPPalette[pal*4], 4 * sizeof(u16)); + + u8 atf = gbSgbPacket[9]; + + if(atf & 0x80) { + gbSgbSetATF(atf & 0x3f); + } + + if(atf & 0x40) { + gbSgbMask = 0; + if(gbBorderOn) + gbSgbRenderBorder(); + } +} + +void gbSgbAttributeBlock() +{ + u8 *fromAddress = &gbSgbPacket[1]; + + u8 nDataSet = *fromAddress++; + if(nDataSet > 12) + nDataSet = 12; + if(nDataSet == 0) + nDataSet = 1; + + while(nDataSet) { + u8 controlCode = (*fromAddress++) & 7; + u8 paletteDesignation = (*fromAddress++) & 0x3f; + u8 startH = (*fromAddress++) & 0x1f; + u8 startV = (*fromAddress++) & 0x1f; + u8 endH = (*fromAddress++) & 0x1f; + u8 endV = (*fromAddress++) & 0x1f; + + u8 * toAddress = gbSgbATF; + + for(u8 y = 0; y < 18; y++) { + for(u8 x = 0; x < 20; x++) { + if(x < startH || y < startV || + x > endH || y > endV) { + // outside + if(controlCode & 0x04) + *toAddress = (paletteDesignation >> 4) & 0x03; + } else if(x > startH && x < endH && + y > startV && y < endV) { + // inside + if(controlCode & 0x01) + *toAddress = paletteDesignation & 0x03; + } else { + // surrounding line + if(controlCode & 0x02) + *toAddress = (paletteDesignation>>2) & 0x03; + else if(controlCode == 0x01) + *toAddress = paletteDesignation & 0x03; + } + toAddress++; + } + } + nDataSet--; + } +} + +void gbSgbSetColumnPalette(u8 col, u8 p) +{ + // if(col < 0) + // col = 0; + if(col > 19) + col = 19; + + p &= 3; + + u8 *toAddress = &gbSgbATF[col]; + + for(u8 y = 0; y < 18; y++) { + *toAddress = p; + toAddress += 20; + } +} + +void gbSgbSetRowPalette(u8 row, u8 p) +{ + // if(row < 0) + // row = 0; + if(row > 17) + row = 17; + + p &= 3; + + u8 *toAddress = &gbSgbATF[row*20]; + + for(u8 x = 0; x < 20; x++) { + *toAddress++ = p; + } +} + +void gbSgbAttributeDivide() +{ + u8 control = gbSgbPacket[1]; + u8 coord = gbSgbPacket[2]; + u8 colorBR = control & 3; + u8 colorAL = (control >> 2) & 3; + u8 colorOL = (control >> 4) & 3; + + if(control & 0x40) { + if(coord > 17) + coord = 17; + + for(u8 i = 0; i < 18; i++) { + if(i < coord) + gbSgbSetRowPalette(i, colorAL); + else if ( i > coord) + gbSgbSetRowPalette(i, colorBR); + else + gbSgbSetRowPalette(i, colorOL); + } + } else { + if(coord > 19) + coord = 19; + + for(u8 i = 0; i < 20; i++) { + if(i < coord) + gbSgbSetColumnPalette(i, colorAL); + else if ( i > coord) + gbSgbSetColumnPalette(i, colorBR); + else + gbSgbSetColumnPalette(i, colorOL); + } + } +} + +void gbSgbAttributeLine() +{ + u8 *fromAddress = &gbSgbPacket[1]; + + u8 nDataSet = *fromAddress++; + + if(nDataSet > 0x6e) + nDataSet = 0x6e; + + while(nDataSet) { + u8 line = *fromAddress++; + u8 num = line & 0x1f; + u8 pal = (line >> 5) & 0x03; + if(line & 0x80) { + if(num > 17) + num = 17; + gbSgbSetRowPalette(num,pal); + } else { + if(num > 19) + num = 19; + gbSgbSetColumnPalette(num,pal); + } + nDataSet--; + } +} + +void gbSgbAttributeCharacter() +{ + u8 startH = gbSgbPacket[1] & 0x1f; + u8 startV = gbSgbPacket[2] & 0x1f; + int nDataSet = READ16LE(((u16 *)&gbSgbPacket[3])); + int style = gbSgbPacket[5] & 1; + if(startH > 19) + startH = 19; + if(startV > 17) + startV = 17; + + u8 s = 6; + u8 *fromAddress = &gbSgbPacket[6]; + u8 v = *fromAddress++; + + if(style) { + while(nDataSet) { + u8 p = (v >> s) & 3; + gbSgbATF[startV * 20 + startH] = p; + startV++; + if(startV == 18) { + startV = 0; + startH++; + if(startH == 20) + break; + } + + if(s) + s -= 2; + else { + s = 6; + v = *fromAddress++; + nDataSet--; + } + } + } else { + while(nDataSet) { + u8 p = (v >> s) & 3; + gbSgbATF[startV * 20 + startH] = p; + startH++; + if(startH == 20) { + startH = 0; + startV++; + if(startV == 18) + break; + } + + if(s) + s -= 2; + else { + s = 6; + v = *fromAddress++; + nDataSet--; + } + } + } +} + +void gbSgbSetATFList() +{ + gbSgbRenderScreenToBuffer(); + + u8 *fromAddress = gbSgbScreenBuffer; + u8 *toAddress = gbSgbATFList; + + for(int i = 0; i < 45; i++) { + for(int j = 0; j < 90; j++) { + u8 v = *fromAddress++; + u8 s = 6; + if(i == 2) + s = 6; + for(int k = 0; k < 4; k++) { + *toAddress++ = (v >> s) & 0x03; + s -= 2; + } + } + } +} + +void gbSgbMaskEnable() +{ + int gbSgbMaskFlag = gbSgbPacket[1] & 3; + + gbSgbMask = gbSgbMaskFlag; + + switch(gbSgbMaskFlag) { + case 1: + break; + case 2: + gbSgbFillScreen(0x0000); + // memset(&gbPalette[0], 0, 128*sizeof(u16)); + break; + case 3: + gbSgbFillScreen(gbPalette[0]); + break; + } + if(!gbSgbMask) { + if(gbBorderOn) + gbSgbRenderBorder(); + } +} + +void gbSgbChrTransfer() +{ + gbSgbRenderScreenToBuffer(); + + int address = (gbSgbPacket[1] & 1) * (128*32); + + if(gbSgbPacket[1] & 1) + gbSgbCGBSupport |= 2; + else + gbSgbCGBSupport |= 1; + + memcpy(&gbSgbBorderChar[address], gbSgbScreenBuffer, 128 * 32); + + if(gbBorderAutomatic && !gbBorderOn && gbSgbCGBSupport > 4) { + gbBorderOn = 1; + systemGbBorderOn(); + } + + if(gbBorderOn && !gbSgbMask) + gbSgbRenderBorder(); + + if(gbSgbMode && gbCgbMode && gbSgbCGBSupport == 7) { + gbSgbCGBSupport = 0; + gbSgbMode = 0; + gbSgbMask = 0; + gbSgbRenderBorder(); + gbReset(); + } + + if(gbSgbCGBSupport > 4) + gbSgbCGBSupport = 0; +} + +void gbSgbMultiRequest() +{ + if(gbSgbPacket[1] & 1) { + gbSgbMultiplayer = 1; + if(gbSgbPacket[1] & 2) + gbSgbFourPlayers = 1; + else + gbSgbFourPlayers = 0; + gbSgbNextController = 0x0e; + } else { + gbSgbFourPlayers = 0; + gbSgbMultiplayer = 0; + gbSgbNextController = 0x0f; + } +} + +void gbSgbCommand() +{ + int command = gbSgbPacket[0] >> 3; + // int nPacket = gbSgbPacket[0] & 7; + + switch(command) { + case 0x00: + gbSgbSetPalette(0,1,(u16 *)&gbSgbPacket[1]); + break; + case 0x01: + gbSgbSetPalette(2,3,(u16 *)&gbSgbPacket[1]); + break; + case 0x02: + gbSgbSetPalette(0,3,(u16 *)&gbSgbPacket[1]); + break; + case 0x03: + gbSgbSetPalette(1,2,(u16 *)&gbSgbPacket[1]); + break; + case 0x04: + gbSgbAttributeBlock(); + break; + case 0x05: + gbSgbAttributeLine(); + break; + case 0x06: + gbSgbAttributeDivide(); + break; + case 0x07: + gbSgbAttributeCharacter(); + break; + case 0x0a: + gbSgbSetPalette(); + break; + case 0x0b: + gbSgbScpPalette(); + break; + case 0x11: + gbSgbMultiRequest(); + break; + case 0x13: + gbSgbChrTransfer(); + break; + case 0x14: + gbSgbPicture(); + break; + case 0x15: + gbSgbSetATFList(); + break; + case 0x16: + gbSgbSetATF(gbSgbPacket[1] & 0x3f); + break; + case 0x17: + gbSgbMaskEnable(); + break; + } +} + +void gbSgbResetPacketState() +{ + gbSgbPacketState = GBSGB_NONE; + gbSgbPacketTimeout = 0; +} + +void gbSgbDoBitTransfer(u8 value) +{ + value = value & 0x30; + switch(gbSgbPacketState) { + case GBSGB_NONE: + if(value == 0) { + gbSgbPacketState = GBSGB_RESET; + gbSgbPacketTimeout = GBSGB_PACKET_TIMEOUT; + } else if (value == 0x30) { + if(gbSgbMultiplayer) { + if((gbSgbReadingController & 7) == 7) { + gbSgbReadingController = 0; + if(gbSgbMultiplayer) { + gbSgbNextController--; + if(gbSgbFourPlayers) { + if(gbSgbNextController == 0x0b) + gbSgbNextController = 0x0f; + } else { + if(gbSgbNextController == 0x0d) + gbSgbNextController = 0x0f; + } + } + } else { + gbSgbReadingController &= 3; + } + } + gbSgbPacketTimeout = 0; + } else { + if(value == 0x10) + gbSgbReadingController |= 0x2; + else if(value == 0x20) + gbSgbReadingController |= 0x01; + gbSgbPacketTimeout = 0; + } + gbSgbPacketTimeout = 0; + break; + case GBSGB_RESET: + if(value == 0x30) { + gbSgbPacketState = GBSGB_PACKET_TRANSMIT; + gbSgbPacketByte = 0; + gbSgbPacketNBits = 0; + gbSgbPacketTimeout = GBSGB_PACKET_TIMEOUT; + } else if(value == 0x00) { + gbSgbPacketTimeout = GBSGB_PACKET_TIMEOUT; + gbSgbPacketState = GBSGB_RESET; + } else { + gbSgbPacketState = GBSGB_NONE; + gbSgbPacketTimeout = 0; + } + break; + case GBSGB_PACKET_TRANSMIT: + if(value == 0) { + gbSgbPacketState = GBSGB_RESET; + gbSgbPacketTimeout = 0; + } else if (value == 0x30){ + if(gbSgbPacketNBits == 128) { + gbSgbPacketNBits = 0; + gbSgbPacketByte = 0; + gbSgbPacketNumber++; + gbSgbPacketTimeout = 0; + if(gbSgbPacketNumber == (gbSgbPacket[0] & 7)) { + gbSgbCommand(); + gbSgbPacketNumber = 0; + gbSgbPacketState = GBSGB_NONE; + gbSgbPacketTimeout = 0; + } + } else { + if(gbSgbPacketNBits < 128) { + gbSgbPacket[gbSgbPacketNumber * 16 + gbSgbPacketByte] >>= 1; + gbSgbPacket[gbSgbPacketNumber * 16 + gbSgbPacketByte] |= gbSgbBit; + gbSgbPacketNBits++; + if(!(gbSgbPacketNBits & 7)) { + gbSgbPacketByte++; + } + gbSgbPacketTimeout = GBSGB_PACKET_TIMEOUT; + } + } + } else { + if(value == 0x20) + gbSgbBit = 0x00; + else + gbSgbBit = 0x80; + gbSgbPacketTimeout = GBSGB_PACKET_TIMEOUT; + } + gbSgbReadingController = 0; + break; + default: + gbSgbPacketState = GBSGB_NONE; + gbSgbPacketTimeout = 0; + break; + } +} + +variable_desc gbSgbSaveStruct[] = { + { &gbSgbMask, sizeof(int) }, + { &gbSgbPacketState, sizeof(int) }, + { &gbSgbBit, sizeof(int) }, + { &gbSgbPacketNBits, sizeof(int) }, + { &gbSgbPacketByte, sizeof(int) }, + { &gbSgbPacketNumber, sizeof(int) }, + { &gbSgbMultiplayer, sizeof(int) }, + { &gbSgbNextController, sizeof(u8) }, + { &gbSgbReadingController, sizeof(u8) }, + { NULL, 0 } +}; + +variable_desc gbSgbSaveStructV3[] = { + { &gbSgbMask, sizeof(int) }, + { &gbSgbPacketState, sizeof(int) }, + { &gbSgbBit, sizeof(int) }, + { &gbSgbPacketNBits, sizeof(int) }, + { &gbSgbPacketByte, sizeof(int) }, + { &gbSgbPacketNumber, sizeof(int) }, + { &gbSgbMultiplayer, sizeof(int) }, + { &gbSgbNextController, sizeof(u8) }, + { &gbSgbReadingController, sizeof(u8) }, + { &gbSgbFourPlayers, sizeof(int) }, + { NULL, 0 } +}; + +void gbSgbSaveGame(gzFile gzFile) +{ + utilWriteData(gzFile, gbSgbSaveStructV3); + + utilGzWrite(gzFile, gbSgbBorder, 2048); + utilGzWrite(gzFile, gbSgbBorderChar, 32*256); + + utilGzWrite(gzFile, gbSgbPacket, 16*7); + + utilGzWrite(gzFile, gbSgbSCPPalette, 4 * 512 * sizeof(u16)); + utilGzWrite(gzFile, gbSgbATF, 20 * 18); + utilGzWrite(gzFile, gbSgbATFList, 45 * 20 * 18); +} + +void gbSgbReadGame(gzFile gzFile, int version) +{ + if(version >= 3) + utilReadData(gzFile, gbSgbSaveStructV3); + else { + utilReadData(gzFile, gbSgbSaveStruct); + gbSgbFourPlayers = 0; + } + + if(version >= 8) { + utilGzRead(gzFile, gbSgbBorder, 2048); + utilGzRead(gzFile, gbSgbBorderChar, 32*256); + } + + utilGzRead(gzFile, gbSgbPacket, 16*7); + + utilGzRead(gzFile, gbSgbSCPPalette, 4 * 512 * sizeof(u16)); + utilGzRead(gzFile, gbSgbATF, 20 * 18); + utilGzRead(gzFile, gbSgbATFList, 45 * 20 * 18); +} diff --git a/src/gb/gbSound.cpp b/src/gb/gbSound.cpp index 4d018e44..0a0685b4 100644 --- a/src/gb/gbSound.cpp +++ b/src/gb/gbSound.cpp @@ -1,525 +1,1032 @@ -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#include - -#include "../System.h" -#include "../Util.h" -#include "gbGlobals.h" -#include "gbSound.h" - -#include "../Gb_Apu/Multi_Buffer.h" -#include "../Gb_Apu/Gb_Apu.h" - -extern u8 soundBuffer[6][735]; -extern u16 soundFinalWave[1470]; -extern int soundVolume; - -#define SOUND_MAGIC 0x60000000 -#define SOUND_MAGIC_2 0x30000000 -#define NOISE_MAGIC 5 - -extern int speed; - -extern void soundResume(); - -extern u8 soundWavePattern[4][32]; - -extern int soundBufferLen; -extern int soundBufferTotalLen; -extern int soundQuality; -extern bool soundPaused; -extern int soundPlay; -extern int soundTicks; -extern int SOUND_CLOCK_TICKS; -extern u32 soundNextPosition; - -extern int soundLevel1; -extern int soundLevel2; -extern int soundBalance; -extern int soundMasterOn; -extern int soundIndex; -extern int soundBufferIndex; -int soundVIN = 0; -extern int soundDebug; - -extern Multi_Buffer * apu_out; -extern Gb_Apu * apu; - -extern int sound1On; -extern int sound1ATL; -extern int sound1Skip; -extern int sound1Index; -extern int sound1Continue; -extern int sound1EnvelopeVolume; -extern int sound1EnvelopeATL; -extern int sound1EnvelopeUpDown; -extern int sound1EnvelopeATLReload; -extern int sound1SweepATL; -extern int sound1SweepATLReload; -extern int sound1SweepSteps; -extern int sound1SweepUpDown; -extern int sound1SweepStep; -extern u8 *sound1Wave; - -extern int sound2On; -extern int sound2ATL; -extern int sound2Skip; -extern int sound2Index; -extern int sound2Continue; -extern int sound2EnvelopeVolume; -extern int sound2EnvelopeATL; -extern int sound2EnvelopeUpDown; -extern int sound2EnvelopeATLReload; -extern u8 *sound2Wave; - -extern int sound3On; -extern int sound3ATL; -extern int sound3Skip; -extern int sound3Index; -extern int sound3Continue; -extern int sound3OutputLevel; -extern int sound3Last; - -extern int sound4On; -extern int sound4Clock; -extern int sound4ATL; -extern int sound4Skip; -extern int sound4Index; -extern int sound4ShiftRight; -extern int sound4ShiftSkip; -extern int sound4ShiftIndex; -extern int sound4NSteps; -extern int sound4CountDown; -extern int sound4Continue; -extern int sound4EnvelopeVolume; -extern int sound4EnvelopeATL; -extern int sound4EnvelopeUpDown; -extern int sound4EnvelopeATLReload; - -extern int soundEnableFlag; - -extern int soundFreqRatio[8]; -extern int soundShiftClock[16]; - -extern s16 soundFilter[4000]; -extern s16 soundLeft[5]; -extern s16 soundRight[5]; -extern int soundEchoIndex; -extern bool soundEcho; -extern bool soundLowPass; -extern bool soundReverse; -extern bool soundOffFlag; - -u8 gbSoundRead(u16 address) -{ - if (address < NR10 || address > 0xFF3F || !apu) return gbMemory[address]; - if (address == NR51) return soundBalance; - - int clock = (SOUND_CLOCK_TICKS - soundTicks) * 95 / (24 * (gbSpeed ? 2 : 1)); - - int ret = apu->read_register(clock, address); - - switch ( address ) - { - case NR10: - ret |= 0x80; break; - - case NR11: - case NR21: - ret |= 0x3F; break; - - case NR13: - case NR23: - case NR31: - case NR33: - ret = 0xFF; break; - - case NR14: - case NR24: - case NR34: - case NR44: - ret |= 0xBF; break; - - case NR30: - ret |= 0x7F; break; - - case NR32: - ret |= 0x9F; break; - } - - return ret; -} - -void gbSoundEvent(register u16 address, register int data) -{ - int freq = 0; - - gbMemory[address] = data; - -#ifndef FINAL_VERSION - if(soundDebug) { - // don't translate. debug only - log("Sound event: %08lx %02x\n", address, data); - } -#endif - if (apu && address >= NR10 && address <= 0xFF3F) - { - int clock = (SOUND_CLOCK_TICKS - soundTicks) * 95 / (24 * (gbSpeed ? 2 : 1)); - if (address == NR50) - { - apu->write_register(clock, address, data); - } - else if (address == NR51) - { - soundBalance = data; - apu->write_register(clock, address, data & soundEnableFlag); - } - else - apu->write_register(clock, address, data); - } -} - - void gbSoundChannel1() -{ -} - -void gbSoundChannel2() -{ -} - -void gbSoundChannel3() -{ -} - -void gbSoundChannel4() -{ -} - -void gbSoundMix() -{ - int res = 0; - - blip_sample_t out[2] = {0, 0}; - - if ( ! apu_out ) return; - - while (!apu_out->read_samples(&out[0], 2)) - { - int ticks = SOUND_CLOCK_TICKS * 95 / (24 * (gbSpeed ? 2 : 1)); - bool was_stereo = apu->end_frame( ticks ); - apu_out->end_frame( ticks, was_stereo ); - } - - res = out[0]; - - //res = (res * 7 * 60) >> 8; - - - if(soundEcho) { - res *= 2; - res += soundFilter[soundEchoIndex]; - res /= 2; - soundFilter[soundEchoIndex++] = res; - } - - if(soundLowPass) { - soundLeft[4] = soundLeft[3]; - soundLeft[3] = soundLeft[2]; - soundLeft[2] = soundLeft[1]; - soundLeft[1] = soundLeft[0]; - soundLeft[0] = res; - res = (soundLeft[4] + 2*soundLeft[3] + 8*soundLeft[2] + 2*soundLeft[1] + - soundLeft[0])/14; - } - - switch(soundVolume) { - case 0: - case 1: - case 2: - case 3: - res *= (soundVolume+1); - break; - case 4: - res >>= 2; - break; - case 5: - res >>= 1; - break; - } - - if(res > 32767) - res = 32767; - if(res < -32768) - res = -32768; - - if(soundReverse) - soundFinalWave[++soundBufferIndex] = res; - else - soundFinalWave[soundBufferIndex++] = res; - - res = out[1]; - - if(soundEcho) { - res *= 2; - res += soundFilter[soundEchoIndex]; - res /= 2; - soundFilter[soundEchoIndex++] = res; - - if(soundEchoIndex >= 4000) - soundEchoIndex = 0; - } - - if(soundLowPass) { - soundRight[4] = soundRight[3]; - soundRight[3] = soundRight[2]; - soundRight[2] = soundRight[1]; - soundRight[1] = soundRight[0]; - soundRight[0] = res; - res = (soundRight[4] + 2*soundRight[3] + 8*soundRight[2] + 2*soundRight[1] + - soundRight[0])/14; - } - - switch(soundVolume) { - case 0: - case 1: - case 2: - case 3: - res *= (soundVolume+1); - break; - case 4: - res >>= 2; - break; - case 5: - res >>= 1; - break; - } - - if(res > 32767) - res = 32767; - if(res < -32768) - res = -32768; - - if(soundReverse) - soundFinalWave[-1+soundBufferIndex++] = res; - else - soundFinalWave[soundBufferIndex++] = res; -} - -void gbSoundTick() -{ - if(systemSoundOn) { - if(soundMasterOn) { - /*gbSoundChannel1(); - gbSoundChannel2(); - gbSoundChannel3(); - gbSoundChannel4();*/ - - gbSoundMix(); - } else { - soundFinalWave[soundBufferIndex++] = 0; - soundFinalWave[soundBufferIndex++] = 0; - } - - soundIndex++; - - if(2*soundBufferIndex >= soundBufferLen) { - if(systemSoundOn) { - if(soundPaused) { - soundResume(); - } - - systemWriteDataToSoundBuffer(); - } - soundIndex = 0; - soundBufferIndex = 0; - } - } -} - -void gbSoundReset() -{ - soundPaused = 1; - soundPlay = 0; - SOUND_CLOCK_TICKS = soundQuality * 24; - soundTicks = SOUND_CLOCK_TICKS; - soundNextPosition = 0; - soundMasterOn = 1; - soundIndex = 0; - soundBufferIndex = 0; - soundLevel1 = 7; - soundLevel2 = 7; - soundVIN = 0; - - - // don't translate - if(soundDebug) { - log("*** Sound Init ***\n"); - } - - - if (apu_out) - { - apu_out->clear(); - apu->reset(false); - - extern const BOOST::uint8_t sound_data[Gb_Apu::end_addr - Gb_Apu::start_addr + 1]; - - int addr = 0; - - while (addr < 0x30) { - apu->write_register( 0, 0xFF10 + addr, sound_data [ addr ] ); - addr++; - } - } - // don't translate - if(soundDebug) { - log("*** Sound Init Complete ***\n"); - } - - if (apu) - { - int addr = 0xff30; - - while(addr < 0xff40) { - /*gbMemory[addr++] = 0x00; - gbMemory[addr++] = 0xff;*/ - gbSoundEvent(addr++, 0x00); - gbSoundEvent(addr++, 0xFF); - } - } - - memset(soundFinalWave, 0x00, soundBufferLen); - - - memset(soundFilter, 0, sizeof(soundFilter)); - soundEchoIndex = 0; -} - -extern bool soundInit(bool gba = true); -extern void soundShutdown(); - -void gbSoundSetQuality(int quality) -{ - if(soundQuality != quality && systemCanChangeSoundQuality()) { - if(!soundOffFlag) - soundShutdown(); - soundQuality = quality; - soundNextPosition = 0; - if(!soundOffFlag) - soundInit(false); - SOUND_CLOCK_TICKS = (gbSpeed ? 2 : 1) * 24 * soundQuality; - soundIndex = 0; - soundBufferIndex = 0; - } else { - soundNextPosition = 0; - SOUND_CLOCK_TICKS = (gbSpeed ? 2 : 1) * 24 * soundQuality; - soundIndex = 0; - soundBufferIndex = 0; - } -} - -variable_desc gbSoundSaveStruct[] = { - { &soundPaused, sizeof(int) }, - { &soundPlay, sizeof(int) }, - { &soundTicks, sizeof(int) }, - { &SOUND_CLOCK_TICKS, sizeof(int) }, - { &soundLevel1, sizeof(int) }, - { &soundLevel2, sizeof(int) }, - { &soundBalance, sizeof(int) }, - { &soundMasterOn, sizeof(int) }, - { &soundIndex, sizeof(int) }, - { &soundVIN, sizeof(int) }, - { &sound1On, sizeof(int) }, - { &sound1ATL, sizeof(int) }, - { &sound1Skip, sizeof(int) }, - { &sound1Index, sizeof(int) }, - { &sound1Continue, sizeof(int) }, - { &sound1EnvelopeVolume, sizeof(int) }, - { &sound1EnvelopeATL, sizeof(int) }, - { &sound1EnvelopeATLReload, sizeof(int) }, - { &sound1EnvelopeUpDown, sizeof(int) }, - { &sound1SweepATL, sizeof(int) }, - { &sound1SweepATLReload, sizeof(int) }, - { &sound1SweepSteps, sizeof(int) }, - { &sound1SweepUpDown, sizeof(int) }, - { &sound1SweepStep, sizeof(int) }, - { &sound2On, sizeof(int) }, - { &sound2ATL, sizeof(int) }, - { &sound2Skip, sizeof(int) }, - { &sound2Index, sizeof(int) }, - { &sound2Continue, sizeof(int) }, - { &sound2EnvelopeVolume, sizeof(int) }, - { &sound2EnvelopeATL, sizeof(int) }, - { &sound2EnvelopeATLReload, sizeof(int) }, - { &sound2EnvelopeUpDown, sizeof(int) }, - { &sound3On, sizeof(int) }, - { &sound3ATL, sizeof(int) }, - { &sound3Skip, sizeof(int) }, - { &sound3Index, sizeof(int) }, - { &sound3Continue, sizeof(int) }, - { &sound3OutputLevel, sizeof(int) }, - { &sound4On, sizeof(int) }, - { &sound4ATL, sizeof(int) }, - { &sound4Skip, sizeof(int) }, - { &sound4Index, sizeof(int) }, - { &sound4Clock, sizeof(int) }, - { &sound4ShiftRight, sizeof(int) }, - { &sound4ShiftSkip, sizeof(int) }, - { &sound4ShiftIndex, sizeof(int) }, - { &sound4NSteps, sizeof(int) }, - { &sound4CountDown, sizeof(int) }, - { &sound4Continue, sizeof(int) }, - { &sound4EnvelopeVolume, sizeof(int) }, - { &sound4EnvelopeATL, sizeof(int) }, - { &sound4EnvelopeATLReload, sizeof(int) }, - { &sound4EnvelopeUpDown, sizeof(int) }, - { &soundEnableFlag, sizeof(int) }, - { NULL, 0 } -}; - -void gbSoundSaveGame(gzFile gzFile) -{ - utilWriteData(gzFile, gbSoundSaveStruct); - - utilGzWrite(gzFile, soundBuffer, 4*735); - utilGzWrite(gzFile, soundFinalWave, 2*735); - utilGzWrite(gzFile, &soundQuality, sizeof(int)); -} - -void gbSoundReadGame(int version,gzFile gzFile) -{ - utilReadData(gzFile, gbSoundSaveStruct); - - soundBufferIndex = soundIndex * 2; - - utilGzRead(gzFile, soundBuffer, 4*735); - utilGzRead(gzFile, soundFinalWave, 2*735); - - if(version >=7) { - int quality = 1; - utilGzRead(gzFile, &quality, sizeof(int)); - gbSoundSetQuality(quality); - } else { - soundQuality = -1; - gbSoundSetQuality(1); - } - - sound1Wave = soundWavePattern[gbMemory[NR11] >> 6]; - sound2Wave = soundWavePattern[gbMemory[NR21] >> 6]; -} +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2005-2006 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#include + +#include "../System.h" +#include "../Util.h" +#include "gbGlobals.h" +#include "gbSound.h" + +extern u8 soundBuffer[6][735]; +extern u16 soundFinalWave[1470]; +extern int soundVolume; + +#define SOUND_MAGIC 0x60000000 +#define SOUND_MAGIC_2 0x30000000 +#define NOISE_MAGIC 5 + +extern int speed; +extern int gbHardware; + +extern void soundResume(); + +extern u8 soundWavePattern[4][32]; + +extern int soundBufferLen; +extern int soundBufferTotalLen; +extern int soundQuality; +extern bool soundPaused; +extern int soundPlay; +extern int soundTicks; +extern int SOUND_CLOCK_TICKS; +extern u32 soundNextPosition; + +extern int soundLevel1; +extern int soundLevel2; +extern int soundBalance; +extern int soundMasterOn; +extern int soundIndex; +extern int soundBufferIndex; +int soundVIN = 0; +extern int soundDebug; + +extern int sound1On; +extern int sound1ATL; +int sound1ATLreload; +int freq1low; +int freq1high; +extern int sound1Skip; +extern int sound1Index; +extern int sound1Continue; +extern int sound1EnvelopeVolume; +extern int sound1EnvelopeATL; +extern int sound1EnvelopeUpDown; +extern int sound1EnvelopeATLReload; +extern int sound1SweepATL; +extern int sound1SweepATLReload; +extern int sound1SweepSteps; +extern int sound1SweepUpDown; +extern int sound1SweepStep; +extern u8 *sound1Wave; + +extern int sound2On; +extern int sound2ATL; +int sound2ATLreload; +int freq2low; +int freq2high; +extern int sound2Skip; +extern int sound2Index; +extern int sound2Continue; +extern int sound2EnvelopeVolume; +extern int sound2EnvelopeATL; +extern int sound2EnvelopeUpDown; +extern int sound2EnvelopeATLReload; +extern u8 *sound2Wave; + +extern int sound3On; +extern int sound3ATL; +int sound3ATLreload; +int freq3low; +int freq3high; +extern int sound3Skip; +extern int sound3Index; +extern int sound3Continue; +extern int sound3OutputLevel; +extern int sound3Last; + +extern int sound4On; +extern int sound4Clock; +extern int sound4ATL; +int sound4ATLreload; +int freq4; +extern int sound4Skip; +extern int sound4Index; +extern int sound4ShiftRight; +extern int sound4ShiftSkip; +extern int sound4ShiftIndex; +extern int sound4NSteps; +extern int sound4CountDown; +extern int sound4Continue; +extern int sound4EnvelopeVolume; +extern int sound4EnvelopeATL; +extern int sound4EnvelopeUpDown; +extern int sound4EnvelopeATLReload; + +extern int soundEnableFlag; + +extern int soundFreqRatio[8]; +extern int soundShiftClock[16]; + +extern s16 soundFilter[4000]; +extern s16 soundLeft[5]; +extern s16 soundRight[5]; +extern int soundEchoIndex; +extern bool soundEcho; +extern bool soundLowPass; +extern bool soundReverse; +extern bool soundOffFlag; + +bool gbDigitalSound = false; + +void gbSoundEvent(register u16 address, register int data) +{ + int freq = 0; + + gbMemory[address] = data; + +#ifndef FINAL_VERSION + if(soundDebug) { + // don't translate. debug only + log("Sound event: %08lx %02x\n", address, data); + } +#endif + switch(address) { + case NR10: + gbMemory[address] = data | 0x80; + sound1SweepATL = sound1SweepATLReload = 344 * ((data >> 4) & 7); + sound1SweepSteps = data & 7; + sound1SweepUpDown = data & 0x08; + sound1SweepStep = 0; + break; + case NR11: + gbMemory[address] = data | 0x3f; + sound1Wave = soundWavePattern[data >> 6]; + sound1ATL = sound1ATLreload = 172 * (64 - (data & 0x3f)); + break; + case NR12: + sound1EnvelopeVolume = data >> 4; + sound1EnvelopeUpDown = data & 0x08; + sound1EnvelopeATLReload = sound1EnvelopeATL = 689 * (data & 7); + break; + case NR13: + gbMemory[address] = 0xff; + freq1low = data; + freq = ((((int)(freq1high & 7)) << 8) | freq1low); + sound1ATL = sound1ATLreload; + freq = 2048 - freq; + if(freq) { + sound1Skip = SOUND_MAGIC / freq; + } else + sound1Skip = 0; + break; + case NR14: + gbMemory[address] = data | 0xbf; + freq1high = data; + freq = ((((int)(freq1high & 7)) << 8) | freq1low); + freq = 2048 - freq; + sound1ATL = sound1ATLreload; + sound1Continue = data & 0x40; + if(freq) { + sound1Skip = SOUND_MAGIC / freq; + } else + sound1Skip = 0; + if(data & 0x80) { + gbMemory[NR52] |= 1; + sound1EnvelopeVolume = gbMemory[NR12] >> 4; + sound1EnvelopeUpDown = gbMemory[NR12] & 0x08; + sound1EnvelopeATLReload = sound1EnvelopeATL = 689 * (gbMemory[NR12] & 7); + sound1SweepATL = sound1SweepATLReload = 344 * ((gbMemory[NR10] >> 4) & 7); + sound1SweepSteps = gbMemory[NR10] & 7; + sound1SweepUpDown = gbMemory[NR10] & 0x08; + sound1SweepStep = 0; + + sound1Index = 0; + sound1On = 1; + } + break; + case NR21: + gbMemory[address] = data | 0x3f; + sound2Wave = soundWavePattern[data >> 6]; + sound2ATL = sound2ATLreload = 172 * (64 - (data & 0x3f)); + break; + case NR22: + sound2EnvelopeVolume = data >> 4; + sound2EnvelopeUpDown = data & 0x08; + sound2EnvelopeATLReload = sound2EnvelopeATL = 689 * (data & 7); + break; + case NR23: + gbMemory[address] = 0xff; + freq2low = data; + freq = (((int)(freq2high & 7)) << 8) | freq2low; + sound2ATL = sound2ATLreload; + freq = 2048 - freq; + if(freq) { + sound2Skip = SOUND_MAGIC / freq; + } else + sound2Skip = 0; + break; + case NR24: + gbMemory[address] = data | 0xbf; + freq2high = data; + freq = (((int)(freq2high & 7)) << 8) | freq2low; + freq = 2048 - freq; + sound2ATL = sound2ATLreload; + sound2Continue = data & 0x40; + if(freq) { + sound2Skip = SOUND_MAGIC / freq; + } else + sound2Skip = 0; + if(data & 0x80) { + gbMemory[NR52] |= 2; + sound2EnvelopeVolume = gbMemory[NR22] >> 4; + sound2EnvelopeUpDown = gbMemory[NR22] & 0x08; + sound2ATL = sound2ATLreload; + sound2EnvelopeATLReload = sound2EnvelopeATL = 689 * (gbMemory[NR22] & 7); + + sound2Index = 0; + sound2On = 1; + } + break; + case NR30: + gbMemory[address] = data | 0x7f; + if(!(data & 0x80)) { + gbMemory[NR52] &= 0xfb; + sound3On = 0; + } + break; + case NR31: + gbMemory[address] = 0xff; + sound3ATL = sound3ATLreload = 172 * (256-data); + break; + case NR32: + gbMemory[address] = data | 0x9f; + sound3OutputLevel = (data >> 5) & 3; + break; + case NR33: + gbMemory[address] = 0xff; + freq3low = data; + freq = 2048 - (((int)(freq3high&7) << 8) | freq3low); + if(freq) { + sound3Skip = SOUND_MAGIC_2 / freq; + } else + sound3Skip = 0; + break; + case NR34: + gbMemory[address] = data | 0xbf; + freq3high = data; + freq = 2048 - (((int)(freq3high&7) << 8) | freq3low); + if(freq) { + sound3Skip = SOUND_MAGIC_2 / freq; + } else { + sound3Skip = 0; + } + sound3Continue = data & 0x40; + if((data & 0x80) && (gbMemory[NR30] & 0x80)) { + gbMemory[NR52] |= 4; + sound3ATL = sound3ATLreload; + sound3Index = 0; + sound3On = 1; + } + break; + case NR41: + sound4ATL = sound4ATLreload = 172 * (64 - (data & 0x3f)); + break; + case NR42: + sound4EnvelopeVolume = data >> 4; + sound4EnvelopeUpDown = data & 0x08; + sound4EnvelopeATLReload = sound4EnvelopeATL = 689 * (data & 7); + break; + case NR43: + freq = freq4 = soundFreqRatio[data & 7]; + sound4NSteps = data & 0x08; + + sound4Skip = (freq << 8) / NOISE_MAGIC; + + sound4Clock = data >> 4; + + freq = freq / soundShiftClock[sound4Clock]; + + sound4ShiftSkip = (freq << 8) / NOISE_MAGIC; + + break; + case NR44: + gbMemory[address] = data | 0xbf; + sound4Continue = data & 0x40; + if(data & 0x80) { + gbMemory[NR52] |= 8; + sound4EnvelopeVolume = gbMemory[NR42] >> 4; + sound4EnvelopeUpDown = gbMemory[NR42] & 0x08; + sound4ATL = sound4ATLreload; + sound4EnvelopeATLReload = sound4EnvelopeATL = 689 * (gbMemory[NR42] & 7); + + sound4On = 1; + + sound4Index = 0; + sound4ShiftIndex = 0; + + if(sound4NSteps) + sound4ShiftRight = 0x7fff; + else + sound4ShiftRight = 0x7f; + } + break; + case NR50: + soundVIN = data & 0x88; + soundLevel1 = data & 7; + soundLevel2 = (data >> 4) & 7; + break; + case NR51: + soundBalance = (data & soundEnableFlag); + break; + case NR52: + soundMasterOn = data & 0x80; + if(!(data & 0x80)) { + sound1On = 0; + sound2On = 0; + sound3On = 0; + sound4On = 0; + gbMemory[address] &= 0xf0; + } + gbMemory[address] = data & 0x80 | 0x70 | (gbMemory[address] & 0xf); + break; + } + + gbDigitalSound = true; + + if(sound1On && sound1EnvelopeVolume != 0) + gbDigitalSound = false; + if(sound2On && sound2EnvelopeVolume != 0) + gbDigitalSound = false; + if(sound3On && sound3OutputLevel != 0) + gbDigitalSound = false; + if(sound4On && sound4EnvelopeVolume != 0) + gbDigitalSound = false; +} + + void gbSoundChannel1() +{ + int vol = sound1EnvelopeVolume; + + int freq = 0; + + int value = 0; + + if(sound1On && (sound1ATL || !sound1Continue)) { + sound1Index += soundQuality*sound1Skip; + sound1Index &= 0x1fffffff; + + value = ((s8)sound1Wave[sound1Index>>24]) * vol; + } + + soundBuffer[0][soundIndex] = value; + + + if(sound1On) { + if(sound1ATL) { + sound1ATL-=soundQuality; + + if(sound1ATL <=0 && sound1Continue) { + gbMemory[NR52] &= 0xfe; + sound1On = 0; + } + } + + if(sound1EnvelopeATL) { + sound1EnvelopeATL-=soundQuality; + + if(sound1EnvelopeATL<=0) { + if(sound1EnvelopeUpDown) { + if(sound1EnvelopeVolume < 15) + sound1EnvelopeVolume++; + } else { + if(sound1EnvelopeVolume) + sound1EnvelopeVolume--; + } + + sound1EnvelopeATL += sound1EnvelopeATLReload; + } + } + + if(sound1SweepATL) { + sound1SweepATL-=soundQuality; + + if(sound1SweepATL<=0) { + freq = (((int)(freq1high & 7)) << 8) | freq1low; + + int updown = 1; + + if(sound1SweepUpDown) + updown = -1; + + int newfreq = 0; + if(sound1SweepSteps) { + newfreq = freq + updown * freq / (1 << sound1SweepSteps); + if(newfreq == freq) + newfreq = 0; + } else + newfreq = freq; + + if(newfreq < 0) { + sound1SweepATL += sound1SweepATLReload; + } else if(newfreq > 2047) { + sound1SweepATL = 0; + sound1On = 0; + gbMemory[NR52] &= 0xfe; + } else { + sound1SweepATL += sound1SweepATLReload; + sound1Skip = SOUND_MAGIC/(2048 - newfreq); + + freq1low = newfreq & 0xff; + freq1high = (freq1high & 0xf8) |((newfreq >> 8) & 7); + } + } + } + } +} + +void gbSoundChannel2() +{ + // int freq = 0; + int vol = sound2EnvelopeVolume; + + int value = 0; + + if(sound2On && (sound2ATL || !sound2Continue)) { + sound2Index += soundQuality*sound2Skip; + sound2Index &= 0x1fffffff; + + value = ((s8)sound2Wave[sound2Index>>24]) * vol; + } + + soundBuffer[1][soundIndex] = value; + + if(sound2On) { + if(sound2ATL) { + sound2ATL-=soundQuality; + + if(sound2ATL <= 0 && sound2Continue) { + gbMemory[NR52] &= 0xfd; + sound2On = 0; + } + } + + if(sound2EnvelopeATL) { + sound2EnvelopeATL-=soundQuality; + + if(sound2EnvelopeATL <= 0) { + if(sound2EnvelopeUpDown) { + if(sound2EnvelopeVolume < 15) + sound2EnvelopeVolume++; + } else { + if(sound2EnvelopeVolume) + sound2EnvelopeVolume--; + } + sound2EnvelopeATL += sound2EnvelopeATLReload; + } + } + } +} + +void gbSoundChannel3() +{ + int value = 0; + + if(sound3On && (sound3ATL || !sound3Continue)) { + + value = sound3Last; + + sound3Index += soundQuality*sound3Skip; + sound3Index &= 0x1fffffff; + + value = gbMemory[0xff30 + (sound3Index>>25)]; + + if( (sound3Index & 0x01000000)) { + value &= 0x0f; + } else { + value >>= 4; + } + + value -= 8; + + switch(sound3OutputLevel) { + case 0: + value = 0; + break; + case 1: + break; + case 2: + value = (value >> 1); + break; + case 3: + value = (value >> 2); + break; + } + sound3Last = value; + } + + soundBuffer[2][soundIndex] = value; + + if(sound3On) { + if(sound3ATL) { + sound3ATL-=soundQuality; + + if(sound3ATL <= 0 && sound3Continue) { + gbMemory[NR52] &= 0xfb; + sound3On = 0; + } + } + } +} + +void gbSoundChannel4() +{ + int vol = sound4EnvelopeVolume; + + int value = 0; + + if(sound4Clock <= 0x0c) { + if(sound4On && (sound4ATL || !sound4Continue)) { + sound4Index += soundQuality*sound4Skip; + sound4ShiftIndex += soundQuality*sound4ShiftSkip; + + if(sound4NSteps) { + while(sound4ShiftIndex > 0x1fffff) { + sound4ShiftRight = (((sound4ShiftRight << 6) ^ + (sound4ShiftRight << 5)) & 0x40) | + (sound4ShiftRight >> 1); + sound4ShiftIndex -= 0x200000; + } + } else { + while(sound4ShiftIndex > 0x1fffff) { + sound4ShiftRight = (((sound4ShiftRight << 14) ^ + (sound4ShiftRight << 13)) & 0x4000) | + (sound4ShiftRight >> 1); + + sound4ShiftIndex -= 0x200000; + } + } + + sound4Index &= 0x1fffff; + sound4ShiftIndex &= 0x1fffff; + + value = ((sound4ShiftRight & 1)*2-1) * vol; + } else { + value = 0; + } + } + + soundBuffer[3][soundIndex] = value; + + if(sound4On) { + if(sound4ATL) { + sound4ATL-=soundQuality; + + if(sound4ATL <= 0 && sound4Continue) { + gbMemory[NR52] &= 0xf7; + sound4On = 0; + } + } + + if(sound4EnvelopeATL) { + sound4EnvelopeATL-=soundQuality; + + if(sound4EnvelopeATL <= 0) { + if(sound4EnvelopeUpDown) { + if(sound4EnvelopeVolume < 15) + sound4EnvelopeVolume++; + } else { + if(sound4EnvelopeVolume) + sound4EnvelopeVolume--; + } + sound4EnvelopeATL += sound4EnvelopeATLReload; + } + } + } +} + +void gbSoundMix() +{ + int res = 0; + + if(soundBalance & 16) { + res += ((s8)soundBuffer[0][soundIndex]); + } + if(soundBalance & 32) { + res += ((s8)soundBuffer[1][soundIndex]); + } + if(soundBalance & 64) { + res += ((s8)soundBuffer[2][soundIndex]); + } + if(soundBalance & 128) { + res += ((s8)soundBuffer[3][soundIndex]); + } + + if(gbDigitalSound) + res *= soundLevel1*256; + else + res *= soundLevel1*60; + + if(soundEcho) { + res *= 2; + res += soundFilter[soundEchoIndex]; + res /= 2; + soundFilter[soundEchoIndex++] = res; + } + + if(soundLowPass) { + soundLeft[4] = soundLeft[3]; + soundLeft[3] = soundLeft[2]; + soundLeft[2] = soundLeft[1]; + soundLeft[1] = soundLeft[0]; + soundLeft[0] = res; + res = (soundLeft[4] + 2*soundLeft[3] + 8*soundLeft[2] + 2*soundLeft[1] + + soundLeft[0])/14; + } + + switch(soundVolume) { + case 0: + case 1: + case 2: + case 3: + res *= (soundVolume+1); + break; + case 4: + res >>= 2; + break; + case 5: + res >>= 1; + break; + } + + if(res > 32767) + res = 32767; + if(res < -32768) + res = -32768; + + if(soundReverse) + soundFinalWave[++soundBufferIndex] = res; + else + soundFinalWave[soundBufferIndex++] = res; + + res = 0; + + if(soundBalance & 1) { + res += ((s8)soundBuffer[0][soundIndex]); + } + if(soundBalance & 2) { + res += ((s8)soundBuffer[1][soundIndex]); + } + if(soundBalance & 4) { + res += ((s8)soundBuffer[2][soundIndex]); + } + if(soundBalance & 8) { + res += ((s8)soundBuffer[3][soundIndex]); + } + + if(gbDigitalSound) + res *= soundLevel2*256; + else + res *= soundLevel2*60; + + if(soundEcho) { + res *= 2; + res += soundFilter[soundEchoIndex]; + res /= 2; + soundFilter[soundEchoIndex++] = res; + + if(soundEchoIndex >= 4000) + soundEchoIndex = 0; + } + + if(soundLowPass) { + soundRight[4] = soundRight[3]; + soundRight[3] = soundRight[2]; + soundRight[2] = soundRight[1]; + soundRight[1] = soundRight[0]; + soundRight[0] = res; + res = (soundRight[4] + 2*soundRight[3] + 8*soundRight[2] + 2*soundRight[1] + + soundRight[0])/14; + } + + switch(soundVolume) { + case 0: + case 1: + case 2: + case 3: + res *= (soundVolume+1); + break; + case 4: + res >>= 2; + break; + case 5: + res >>= 1; + break; + } + + if(res > 32767) + res = 32767; + if(res < -32768) + res = -32768; + + if(soundReverse) + soundFinalWave[-1+soundBufferIndex++] = res; + else + soundFinalWave[soundBufferIndex++] = res; +} + +void gbSoundTick() +{ + if(systemSoundOn) { + if(soundMasterOn) { + gbSoundChannel1(); + gbSoundChannel2(); + gbSoundChannel3(); + gbSoundChannel4(); + + gbSoundMix(); + } else { + soundFinalWave[soundBufferIndex++] = 0; + soundFinalWave[soundBufferIndex++] = 0; + } + + soundIndex++; + + if(2*soundBufferIndex >= soundBufferLen) { + if(systemSoundOn) { + if(soundPaused) { + soundResume(); + } + + systemWriteDataToSoundBuffer(); + } + soundIndex = 0; + soundBufferIndex = 0; + } + } +} + +void gbSoundReset() +{ + soundPaused = true; + soundPlay = 0; + SOUND_CLOCK_TICKS = soundQuality * 24; + soundTicks = SOUND_CLOCK_TICKS; + soundNextPosition = 0; + soundMasterOn = 1; + soundIndex = 0; + soundBufferIndex = 0; + soundLevel1 = 7; + soundLevel2 = 7; + soundVIN = 0; + + sound1On = 0; + sound1ATL = 0; + sound1Skip = 0; + sound1Index = 0; + sound1Continue = 0; + sound1EnvelopeVolume = 0; + sound1EnvelopeATL = 0; + sound1EnvelopeUpDown = 0; + sound1EnvelopeATLReload = 0; + sound1SweepATL = 0; + sound1SweepATLReload = 0; + sound1SweepSteps = 0; + sound1SweepUpDown = 0; + sound1SweepStep = 0; + sound1Wave = soundWavePattern[2]; + + sound2On = 0; + sound2ATL = 0; + sound2Skip = 0; + sound2Index = 0; + sound2Continue = 0; + sound2EnvelopeVolume = 0; + sound2EnvelopeATL = 0; + sound2EnvelopeUpDown = 0; + sound2EnvelopeATLReload = 0; + sound2Wave = soundWavePattern[2]; + + sound3On = 0; + sound3ATL = 0; + sound3Skip = 0; + sound3Index = 0; + sound3Continue = 0; + sound3OutputLevel = 0; + + sound4On = 0; + sound4Clock = 0; + sound4ATL = 0; + sound4Skip = 0; + sound4Index = 0; + sound4ShiftRight = 0x7f; + sound4NSteps = 0; + sound4CountDown = 0; + sound4Continue = 0; + sound4EnvelopeVolume = 0; + sound4EnvelopeATL = 0; + sound4EnvelopeUpDown = 0; + sound4EnvelopeATLReload = 0; + + // don't translate + if(soundDebug) { + log("*** Sound Init ***\n"); + } + + gbSoundEvent(0xff10, 0x80); + gbSoundEvent(0xff11, 0xbf); + gbSoundEvent(0xff12, 0xf3); + gbSoundEvent(0xff14, 0xbf); + gbSoundEvent(0xff16, 0x3f); + gbSoundEvent(0xff17, 0x00); + gbSoundEvent(0xff19, 0xbf); + + gbSoundEvent(0xff1a, 0x7f); + gbSoundEvent(0xff1b, 0xff); + gbSoundEvent(0xff1c, 0xbf); + gbSoundEvent(0xff1e, 0xbf); + + gbSoundEvent(0xff20, 0xff); + gbSoundEvent(0xff21, 0x00); + gbSoundEvent(0xff22, 0x00); + gbSoundEvent(0xff23, 0xbf); + gbSoundEvent(0xff24, 0x77); + gbSoundEvent(0xff25, 0xf3); + + if (gbHardware & 0x4) + gbSoundEvent(0xff26, 0xf0); + else + gbSoundEvent(0xff26, 0xf1); + + // don't translate + if(soundDebug) { + log("*** Sound Init Complete ***\n"); + } + + sound1On = 0; + sound2On = 0; + sound3On = 0; + sound4On = 0; + + int addr = 0xff30; + + while(addr < 0xff40) { + gbMemory[addr++] = 0x00; + gbMemory[addr++] = 0xff; + } + + memset(soundFinalWave, 0x00, soundBufferLen); + + + memset(soundFilter, 0, sizeof(soundFilter)); + soundEchoIndex = 0; +} + +extern bool soundInit(bool gb); +extern void soundShutdown(); + +void gbSoundSetQuality(int quality) +{ + if(soundQuality != quality && systemCanChangeSoundQuality()) { + if(!soundOffFlag) + soundShutdown(); + soundQuality = quality; + soundNextPosition = 0; + if(!soundOffFlag) + soundInit(false); + SOUND_CLOCK_TICKS = (gbSpeed ? 2 : 1) * 24 * soundQuality; + soundIndex = 0; + soundBufferIndex = 0; + } else { + soundNextPosition = 0; + SOUND_CLOCK_TICKS = (gbSpeed ? 2 : 1) * 24 * soundQuality; + soundIndex = 0; + soundBufferIndex = 0; + } +} + +variable_desc gbSoundSaveStruct[] = { + { &soundPaused, sizeof(int) }, + { &soundPlay, sizeof(int) }, + { &soundTicks, sizeof(int) }, + { &SOUND_CLOCK_TICKS, sizeof(int) }, + { &soundLevel1, sizeof(int) }, + { &soundLevel2, sizeof(int) }, + { &soundBalance, sizeof(int) }, + { &soundMasterOn, sizeof(int) }, + { &soundIndex, sizeof(int) }, + { &soundVIN, sizeof(int) }, + { &sound1On, sizeof(int) }, + { &sound1ATL, sizeof(int) }, + { &sound1Skip, sizeof(int) }, + { &sound1Index, sizeof(int) }, + { &sound1Continue, sizeof(int) }, + { &sound1EnvelopeVolume, sizeof(int) }, + { &sound1EnvelopeATL, sizeof(int) }, + { &sound1EnvelopeATLReload, sizeof(int) }, + { &sound1EnvelopeUpDown, sizeof(int) }, + { &sound1SweepATL, sizeof(int) }, + { &sound1SweepATLReload, sizeof(int) }, + { &sound1SweepSteps, sizeof(int) }, + { &sound1SweepUpDown, sizeof(int) }, + { &sound1SweepStep, sizeof(int) }, + { &sound2On, sizeof(int) }, + { &sound2ATL, sizeof(int) }, + { &sound2Skip, sizeof(int) }, + { &sound2Index, sizeof(int) }, + { &sound2Continue, sizeof(int) }, + { &sound2EnvelopeVolume, sizeof(int) }, + { &sound2EnvelopeATL, sizeof(int) }, + { &sound2EnvelopeATLReload, sizeof(int) }, + { &sound2EnvelopeUpDown, sizeof(int) }, + { &sound3On, sizeof(int) }, + { &sound3ATL, sizeof(int) }, + { &sound3Skip, sizeof(int) }, + { &sound3Index, sizeof(int) }, + { &sound3Continue, sizeof(int) }, + { &sound3OutputLevel, sizeof(int) }, + { &sound4On, sizeof(int) }, + { &sound4ATL, sizeof(int) }, + { &sound4Skip, sizeof(int) }, + { &sound4Index, sizeof(int) }, + { &sound4Clock, sizeof(int) }, + { &sound4ShiftRight, sizeof(int) }, + { &sound4ShiftSkip, sizeof(int) }, + { &sound4ShiftIndex, sizeof(int) }, + { &sound4NSteps, sizeof(int) }, + { &sound4CountDown, sizeof(int) }, + { &sound4Continue, sizeof(int) }, + { &sound4EnvelopeVolume, sizeof(int) }, + { &sound4EnvelopeATL, sizeof(int) }, + { &sound4EnvelopeATLReload, sizeof(int) }, + { &sound4EnvelopeUpDown, sizeof(int) }, + { &soundEnableFlag, sizeof(int) }, + { NULL, 0 } +}; + +void gbSoundSaveGame(gzFile gzFile) +{ + utilWriteData(gzFile, gbSoundSaveStruct); + + utilWriteInt(gzFile, sound1ATLreload); + utilWriteInt(gzFile, freq1low); + utilWriteInt(gzFile, freq1high); + utilWriteInt(gzFile, sound2ATLreload); + utilWriteInt(gzFile, freq2low); + utilWriteInt(gzFile, freq2high); + utilWriteInt(gzFile, sound3ATLreload); + utilWriteInt(gzFile, freq3low); + utilWriteInt(gzFile, freq3high); + utilWriteInt(gzFile, sound4ATLreload); + utilWriteInt(gzFile, freq4); + + utilGzWrite(gzFile, soundBuffer, 4*735); + utilGzWrite(gzFile, soundFinalWave, 2*735); + utilGzWrite(gzFile, &soundQuality, sizeof(int)); +} + +void gbSoundReadGame(int version,gzFile gzFile) +{ + utilReadData(gzFile, gbSoundSaveStruct); + + if (version<11) + { + sound1ATLreload = 172 * (64 - (gbMemory[NR11] & 0x3f)); + freq1low = gbMemory[NR13]; + freq1high = gbMemory[NR14] & 7; + sound2ATLreload = 172 * (64 - (gbMemory[NR21] & 0x3f)); + freq2low = gbMemory[NR23]; + freq2high = gbMemory[NR24] & 7; + sound3ATLreload = 172 * (256 - gbMemory[NR31]); + freq3low = gbMemory[NR33]; + freq3high = gbMemory[NR34] & 7; + sound4ATLreload = 172 * (64 - (gbMemory[NR41] & 0x3f)); + freq4 = soundFreqRatio[gbMemory[NR43] & 7]; + } + else + { + sound1ATLreload = utilReadInt(gzFile); + freq1low = utilReadInt(gzFile); + freq1high = utilReadInt(gzFile); + sound2ATLreload = utilReadInt(gzFile); + freq2low = utilReadInt(gzFile); + freq2high = utilReadInt(gzFile); + sound3ATLreload = utilReadInt(gzFile); + freq3low = utilReadInt(gzFile); + freq3high = utilReadInt(gzFile); + sound4ATLreload = utilReadInt(gzFile); + freq4 = utilReadInt(gzFile); + } + + soundBufferIndex = soundIndex * 2; + + utilGzRead(gzFile, soundBuffer, 4*735); + utilGzRead(gzFile, soundFinalWave, 2*735); + + if(version >=7) { + int quality = 1; + utilGzRead(gzFile, &quality, sizeof(int)); + gbSoundSetQuality(quality); + } else { + soundQuality = -1; + gbSoundSetQuality(1); + } + + sound1Wave = soundWavePattern[gbMemory[NR11] >> 6]; + sound2Wave = soundWavePattern[gbMemory[NR21] >> 6]; +} diff --git a/src/gb/gbSound.h b/src/gb/gbSound.h index 8509a107..50658b24 100644 --- a/src/gb/gbSound.h +++ b/src/gb/gbSound.h @@ -1,61 +1,59 @@ -// -*- C++ -*- -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#define NR10 0xff10 -#define NR11 0xff11 -#define NR12 0xff12 -#define NR13 0xff13 -#define NR14 0xff14 -#define NR21 0xff16 -#define NR22 0xff17 -#define NR23 0xff18 -#define NR24 0xff19 -#define NR30 0xff1a -#define NR31 0xff1b -#define NR32 0xff1c -#define NR33 0xff1d -#define NR34 0xff1e -#define NR41 0xff20 -#define NR42 0xff21 -#define NR43 0xff22 -#define NR44 0xff23 -#define NR50 0xff24 -#define NR51 0xff25 -#define NR52 0xff26 - -#define SOUND_EVENT(address,value) \ - gbSoundEvent(address,value) - -extern void gbSoundTick(); -extern void gbSoundPause(); -extern void gbSoundResume(); -extern void gbSoundEnable(int); -extern void gbSoundDisable(int); -extern int gbSoundGetEnable(); -extern void gbSoundReset(); -extern void gbSoundSaveGame(gzFile); -extern void gbSoundReadGame(int,gzFile); -extern void gbSoundEvent(register u16, register int); -extern void gbSoundSetQuality(int); - -extern u8 gbSoundRead(u16 address); - -extern int soundTicks; -extern int soundQuality; -extern int SOUND_CLOCK_TICKS; +// -*- C++ -*- +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#define NR10 0xff10 +#define NR11 0xff11 +#define NR12 0xff12 +#define NR13 0xff13 +#define NR14 0xff14 +#define NR21 0xff16 +#define NR22 0xff17 +#define NR23 0xff18 +#define NR24 0xff19 +#define NR30 0xff1a +#define NR31 0xff1b +#define NR32 0xff1c +#define NR33 0xff1d +#define NR34 0xff1e +#define NR41 0xff20 +#define NR42 0xff21 +#define NR43 0xff22 +#define NR44 0xff23 +#define NR50 0xff24 +#define NR51 0xff25 +#define NR52 0xff26 + +#define SOUND_EVENT(address,value) \ + gbSoundEvent(address,value) + +extern void gbSoundTick(); +extern void gbSoundPause(); +extern void gbSoundResume(); +extern void gbSoundEnable(int); +extern void gbSoundDisable(int); +extern int gbSoundGetEnable(); +extern void gbSoundReset(); +extern void gbSoundSaveGame(gzFile); +extern void gbSoundReadGame(int,gzFile); +extern void gbSoundEvent(register u16, register int); +extern void gbSoundSetQuality(int); + +extern int soundTicks; +extern int soundQuality; +extern int SOUND_CLOCK_TICKS; diff --git a/src/getopt.cpp b/src/getopt.cpp deleted file mode 100644 index 9a127443..00000000 --- a/src/getopt.cpp +++ /dev/null @@ -1,1048 +0,0 @@ -#include - -/* Getopt for GNU. - NOTE: getopt is now part of the C library, so if you don't know what - "Keep this file name-space clean" means, talk to drepper@gnu.org - before changing it! - - Copyright (C) 1987, 88, 89, 90, 91, 92, 93, 94, 95, 96, 97, 98 - Free Software Foundation, Inc. - - NOTE: This source is derived from an old version taken from the GNU C - Library (glibc). - - This program is free software; you can redistribute it and/or modify it - under the terms of the GNU General Public License as published by the - Free Software Foundation; either version 2, or (at your option) any - later version. - - This program is distributed in the hope that it will be useful, - but WITHOUT ANY WARRANTY; without even the implied warranty of - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - GNU General Public License for more details. - - You should have received a copy of the GNU General Public License - along with this program; if not, write to the Free Software - Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, - USA. */ - -/* This tells Alpha OSF/1 not to define a getopt prototype in . - Ditto for AIX 3.2 and . */ -#ifndef _NO_PROTO -# define _NO_PROTO -#endif - -#ifdef HAVE_CONFIG_H -# include -#endif - -#if !defined __STDC__ || !__STDC__ -/* This is a separate conditional since some stdc systems - reject `defined (const)'. */ -# ifndef const -# define const -# endif -#endif - -#include - -/* Comment out all this code if we are using the GNU C Library, and are not - actually compiling the library itself. This code is part of the GNU C - Library, but also included in many other GNU distributions. Compiling - and linking in this code is a waste when using the GNU C library - (especially if it is a shared library). Rather than having every GNU - program understand `configure --with-gnu-libc' and omit the object files, - it is simpler to just do this in the source for each such file. */ - -#define GETOPT_INTERFACE_VERSION 2 -#if !defined _LIBC && defined __GLIBC__ && __GLIBC__ >= 2 -# include -# if _GNU_GETOPT_INTERFACE_VERSION == GETOPT_INTERFACE_VERSION -# define ELIDE_CODE -# endif -#endif - -#ifndef ELIDE_CODE - - -/* This needs to come after some library #include - to get __GNU_LIBRARY__ defined. */ -#ifdef __GNU_LIBRARY__ -/* Don't include stdlib.h for non-GNU C libraries because some of them - contain conflicting prototypes for getopt. */ -# include -# include -#endif /* GNU C library. */ - -#ifdef VMS -# include -# if HAVE_STRING_H - 0 -# include -# endif -#endif - -#ifndef _ -/* This is for other GNU distributions with internationalized messages. - When compiling libc, the _ macro is predefined. */ -# ifdef HAVE_LIBINTL_H -# include -# define _(msgid) gettext (msgid) -# else -# define _(msgid) (msgid) -# endif -#endif - -#ifdef _MSC_VER -#include -#include -#endif - -/* This version of `getopt' appears to the caller like standard Unix `getopt' - but it behaves differently for the user, since it allows the user - to intersperse the options with the other arguments. - - As `getopt' works, it permutes the elements of ARGV so that, - when it is done, all the options precede everything else. Thus - all application programs are extended to handle flexible argument order. - - Setting the environment variable POSIXLY_CORRECT disables permutation. - Then the behavior is completely standard. - - GNU application programs can use a third alternative mode in which - they can distinguish the relative order of options and other arguments. */ - -#include "getopt.h" - -/* For communication from `getopt' to the caller. - When `getopt' finds an option that takes an argument, - the argument value is returned here. - Also, when `ordering' is RETURN_IN_ORDER, - each non-option ARGV-element is returned here. */ - -char *optarg = NULL; - -/* Index in ARGV of the next element to be scanned. - This is used for communication to and from the caller - and for communication between successive calls to `getopt'. - - On entry to `getopt', zero means this is the first call; initialize. - - When `getopt' returns -1, this is the index of the first of the - non-option elements that the caller should itself scan. - - Otherwise, `optind' communicates from one call to the next - how much of ARGV has been scanned so far. */ - -/* 1003.2 says this must be 1 before any call. */ -int optind = 1; - -/* Formerly, initialization of getopt depended on optind==0, which - causes problems with re-calling getopt as programs generally don't - know that. */ - -int __getopt_initialized = 0; - -/* The next char to be scanned in the option-element - in which the last option character we returned was found. - This allows us to pick up the scan where we left off. - - If this is zero, or a null string, it means resume the scan - by advancing to the next ARGV-element. */ - -static char *nextchar; - -/* Callers store zero here to inhibit the error message - for unrecognized options. */ - -int opterr = 1; - -/* Set to an option character which was unrecognized. - This must be initialized on some systems to avoid linking in the - system's own getopt implementation. */ - -int optopt = '?'; - -/* Describe how to deal with options that follow non-option ARGV-elements. - - If the caller did not specify anything, - the default is REQUIRE_ORDER if the environment variable - POSIXLY_CORRECT is defined, PERMUTE otherwise. - - REQUIRE_ORDER means don't recognize them as options; - stop option processing when the first non-option is seen. - This is what Unix does. - This mode of operation is selected by either setting the environment - variable POSIXLY_CORRECT, or using `+' as the first character - of the list of option characters. - - PERMUTE is the default. We permute the contents of ARGV as we scan, - so that eventually all the non-options are at the end. This allows options - to be given in any order, even with programs that were not written to - expect this. - - RETURN_IN_ORDER is an option available to programs that were written - to expect options and other ARGV-elements in any order and that care about - the ordering of the two. We describe each non-option ARGV-element - as if it were the argument of an option with character code 1. - Using `-' as the first character of the list of option characters - selects this mode of operation. - - The special argument `--' forces an end of option-scanning regardless - of the value of `ordering'. In the case of RETURN_IN_ORDER, only - `--' can cause `getopt' to return -1 with `optind' != ARGC. */ - -static enum -{ - REQUIRE_ORDER, PERMUTE, RETURN_IN_ORDER -} ordering; - -/* Value of POSIXLY_CORRECT environment variable. */ -static char *posixly_correct; - -#ifdef __GNU_LIBRARY__ -/* We want to avoid inclusion of string.h with non-GNU libraries - because there are many ways it can cause trouble. - On some systems, it contains special magic macros that don't work - in GCC. */ -# include -# define my_index strchr -#else - -# if HAVE_STRING_H -# include -# else -# if HAVE_STRINGS_H -# include -# endif -# endif - -/* Avoid depending on library functions or files - whose names are inconsistent. */ - -#ifndef getenv -extern char *getenv (); -#endif - -static char * -my_index (const char *str, int chr) -{ - while (*str) - { - if (*str == chr) - return (char *) str; - str++; - } - return 0; -} - -/* If using GCC, we can safely declare strlen this way. - If not using GCC, it is ok not to declare it. */ -#ifdef __GNUC__ -/* Note that Motorola Delta 68k R3V7 comes with GCC but not stddef.h. - That was relevant to code that was here before. */ -# if (!defined __STDC__ || !__STDC__) && !defined strlen -/* gcc with -traditional declares the built-in strlen to return int, - and has done so at least since version 2.4.5. -- rms. */ -extern int strlen (const char *); -# endif /* not __STDC__ */ -#endif /* __GNUC__ */ - -#endif /* not __GNU_LIBRARY__ */ - -/* Handle permutation of arguments. */ - -/* Describe the part of ARGV that contains non-options that have - been skipped. `first_nonopt' is the index in ARGV of the first of them; - `last_nonopt' is the index after the last of them. */ - -static int first_nonopt; -static int last_nonopt; - -#ifdef _LIBC -/* Bash 2.0 gives us an environment variable containing flags - indicating ARGV elements that should not be considered arguments. */ - -/* Defined in getopt_init.c */ -extern char *__getopt_nonoption_flags; - -static int nonoption_flags_max_len; -static int nonoption_flags_len; - -static int original_argc; -static char *const *original_argv; - -/* Make sure the environment variable bash 2.0 puts in the environment - is valid for the getopt call we must make sure that the ARGV passed - to getopt is that one passed to the process. */ -static void -__attribute__ ((unused)) -store_args_and_env (int argc, char *const *argv) -{ - /* XXX This is no good solution. We should rather copy the args so - that we can compare them later. But we must not use malloc(3). */ - original_argc = argc; - original_argv = argv; -} -# ifdef text_set_element -text_set_element (__libc_subinit, store_args_and_env); -# endif /* text_set_element */ - -# define SWAP_FLAGS(ch1, ch2) \ - if (nonoption_flags_len > 0) \ - { \ - char __tmp = __getopt_nonoption_flags[ch1]; \ - __getopt_nonoption_flags[ch1] = __getopt_nonoption_flags[ch2]; \ - __getopt_nonoption_flags[ch2] = __tmp; \ - } -#else /* !_LIBC */ -# define SWAP_FLAGS(ch1, ch2) -#endif /* _LIBC */ - -/* Exchange two adjacent subsequences of ARGV. - One subsequence is elements [first_nonopt,last_nonopt) - which contains all the non-options that have been skipped so far. - The other is elements [last_nonopt,optind), which contains all - the options processed since those non-options were skipped. - - `first_nonopt' and `last_nonopt' are relocated so that they describe - the new indices of the non-options in ARGV after they are moved. */ - -#if defined __STDC__ && __STDC__ -static void exchange (char **); -#endif - -static void -exchange (char **argv) -{ - int bottom = first_nonopt; - int middle = last_nonopt; - int top = optind; - char *tem; - - /* Exchange the shorter segment with the far end of the longer segment. - That puts the shorter segment into the right place. - It leaves the longer segment in the right place overall, - but it consists of two parts that need to be swapped next. */ - -#ifdef _LIBC - /* First make sure the handling of the `__getopt_nonoption_flags' - string can work normally. Our top argument must be in the range - of the string. */ - if (nonoption_flags_len > 0 && top >= nonoption_flags_max_len) - { - /* We must extend the array. The user plays games with us and - presents new arguments. */ - char *new_str = malloc (top + 1); - if (new_str == NULL) - nonoption_flags_len = nonoption_flags_max_len = 0; - else - { - memset (__mempcpy (new_str, __getopt_nonoption_flags, - nonoption_flags_max_len), - '\0', top + 1 - nonoption_flags_max_len); - nonoption_flags_max_len = top + 1; - __getopt_nonoption_flags = new_str; - } - } -#endif - - while (top > middle && middle > bottom) - { - if (top - middle > middle - bottom) - { - /* Bottom segment is the short one. */ - int len = middle - bottom; - register int i; - - /* Swap it with the top part of the top segment. */ - for (i = 0; i < len; i++) - { - tem = argv[bottom + i]; - argv[bottom + i] = argv[top - (middle - bottom) + i]; - argv[top - (middle - bottom) + i] = tem; - SWAP_FLAGS (bottom + i, top - (middle - bottom) + i); - } - /* Exclude the moved bottom segment from further swapping. */ - top -= len; - } - else - { - /* Top segment is the short one. */ - int len = top - middle; - register int i; - - /* Swap it with the bottom part of the bottom segment. */ - for (i = 0; i < len; i++) - { - tem = argv[bottom + i]; - argv[bottom + i] = argv[middle + i]; - argv[middle + i] = tem; - SWAP_FLAGS (bottom + i, middle + i); - } - /* Exclude the moved top segment from further swapping. */ - bottom += len; - } - } - - /* Update records for the slots the non-options now occupy. */ - - first_nonopt += (optind - last_nonopt); - last_nonopt = optind; -} - -/* Initialize the internal data when the first call is made. */ - -#if defined __STDC__ && __STDC__ -static const char *_getopt_initialize (int, char *const *, const char *); -#endif -static const char * -_getopt_initialize (int argc, char *const *argv, const char *optstring) -{ - /* Start processing options with ARGV-element 1 (since ARGV-element 0 - is the program name); the sequence of previously skipped - non-option ARGV-elements is empty. */ - - first_nonopt = last_nonopt = optind; - - nextchar = NULL; - - posixly_correct = getenv ("POSIXLY_CORRECT"); - - /* Determine how to handle the ordering of options and nonoptions. */ - - if (optstring[0] == '-') - { - ordering = RETURN_IN_ORDER; - ++optstring; - } - else if (optstring[0] == '+') - { - ordering = REQUIRE_ORDER; - ++optstring; - } - else if (posixly_correct != NULL) - ordering = REQUIRE_ORDER; - else - ordering = PERMUTE; - -#ifdef _LIBC - if (posixly_correct == NULL - && argc == original_argc && argv == original_argv) - { - if (nonoption_flags_max_len == 0) - { - if (__getopt_nonoption_flags == NULL - || __getopt_nonoption_flags[0] == '\0') - nonoption_flags_max_len = -1; - else - { - const char *orig_str = __getopt_nonoption_flags; - int len = nonoption_flags_max_len = strlen (orig_str); - if (nonoption_flags_max_len < argc) - nonoption_flags_max_len = argc; - __getopt_nonoption_flags = - (char *) malloc (nonoption_flags_max_len); - if (__getopt_nonoption_flags == NULL) - nonoption_flags_max_len = -1; - else - memset (__mempcpy (__getopt_nonoption_flags, orig_str, len), - '\0', nonoption_flags_max_len - len); - } - } - nonoption_flags_len = nonoption_flags_max_len; - } - else - nonoption_flags_len = 0; -#endif - - return optstring; -} - -/* Scan elements of ARGV (whose length is ARGC) for option characters - given in OPTSTRING. - - If an element of ARGV starts with '-', and is not exactly "-" or "--", - then it is an option element. The characters of this element - (aside from the initial '-') are option characters. If `getopt' - is called repeatedly, it returns successively each of the option characters - from each of the option elements. - - If `getopt' finds another option character, it returns that character, - updating `optind' and `nextchar' so that the next call to `getopt' can - resume the scan with the following option character or ARGV-element. - - If there are no more option characters, `getopt' returns -1. - Then `optind' is the index in ARGV of the first ARGV-element - that is not an option. (The ARGV-elements have been permuted - so that those that are not options now come last.) - - OPTSTRING is a string containing the legitimate option characters. - If an option character is seen that is not listed in OPTSTRING, - return '?' after printing an error message. If you set `opterr' to - zero, the error message is suppressed but we still return '?'. - - If a char in OPTSTRING is followed by a colon, that means it wants an arg, - so the following text in the same ARGV-element, or the text of the following - ARGV-element, is returned in `optarg'. Two colons mean an option that - wants an optional arg; if there is text in the current ARGV-element, - it is returned in `optarg', otherwise `optarg' is set to zero. - - If OPTSTRING starts with `-' or `+', it requests different methods of - handling the non-option ARGV-elements. - See the comments about RETURN_IN_ORDER and REQUIRE_ORDER, above. - - Long-named options begin with `--' instead of `-'. - Their names may be abbreviated as long as the abbreviation is unique - or is an exact match for some defined option. If they have an - argument, it follows the option name in the same ARGV-element, separated - from the option name by a `=', or else the in next ARGV-element. - When `getopt' finds a long-named option, it returns 0 if that option's - `flag' field is nonzero, the value of the option's `val' field - if the `flag' field is zero. - - The elements of ARGV aren't really const, because we permute them. - But we pretend they're const in the prototype to be compatible - with other systems. - - LONGOPTS is a vector of `struct option' terminated by an - element containing a name which is zero. - - LONGIND returns the index in LONGOPT of the long-named option found. - It is only valid when a long-named option has been found by the most - recent call. - - If LONG_ONLY is nonzero, '-' as well as '--' can introduce - long-named options. */ - -int -_getopt_internal (int argc, char *const *argv, const char *optstring, const struct option *longopts, int *longind, int long_only) -{ - optarg = NULL; - - if (optind == 0 || !__getopt_initialized) - { - if (optind == 0) - optind = 1; /* Don't scan ARGV[0], the program name. */ - optstring = _getopt_initialize (argc, argv, optstring); - __getopt_initialized = 1; - } - - /* Test whether ARGV[optind] points to a non-option argument. - Either it does not have option syntax, or there is an environment flag - from the shell indicating it is not an option. The later information - is only used when the used in the GNU libc. */ -#ifdef _LIBC -# define NONOPTION_P (argv[optind][0] != '-' || argv[optind][1] == '\0' \ - || (optind < nonoption_flags_len \ - && __getopt_nonoption_flags[optind] == '1')) -#else -# define NONOPTION_P (argv[optind][0] != '-' || argv[optind][1] == '\0') -#endif - - if (nextchar == NULL || *nextchar == '\0') - { - /* Advance to the next ARGV-element. */ - - /* Give FIRST_NONOPT & LAST_NONOPT rational values if OPTIND has been - moved back by the user (who may also have changed the arguments). */ - if (last_nonopt > optind) - last_nonopt = optind; - if (first_nonopt > optind) - first_nonopt = optind; - - if (ordering == PERMUTE) - { - /* If we have just processed some options following some non-options, - exchange them so that the options come first. */ - - if (first_nonopt != last_nonopt && last_nonopt != optind) - exchange ((char **) argv); - else if (last_nonopt != optind) - first_nonopt = optind; - - /* Skip any additional non-options - and extend the range of non-options previously skipped. */ - - while (optind < argc && NONOPTION_P) - optind++; - last_nonopt = optind; - } - - /* The special ARGV-element `--' means premature end of options. - Skip it like a null option, - then exchange with previous non-options as if it were an option, - then skip everything else like a non-option. */ - - if (optind != argc && !strcmp (argv[optind], "--")) - { - optind++; - - if (first_nonopt != last_nonopt && last_nonopt != optind) - exchange ((char **) argv); - else if (first_nonopt == last_nonopt) - first_nonopt = optind; - last_nonopt = argc; - - optind = argc; - } - - /* If we have done all the ARGV-elements, stop the scan - and back over any non-options that we skipped and permuted. */ - - if (optind == argc) - { - /* Set the next-arg-index to point at the non-options - that we previously skipped, so the caller will digest them. */ - if (first_nonopt != last_nonopt) - optind = first_nonopt; - return -1; - } - - /* If we have come to a non-option and did not permute it, - either stop the scan or describe it to the caller and pass it by. */ - - if (NONOPTION_P) - { - if (ordering == REQUIRE_ORDER) - return -1; - optarg = argv[optind++]; - return 1; - } - - /* We have found another option-ARGV-element. - Skip the initial punctuation. */ - - nextchar = (argv[optind] + 1 - + (longopts != NULL && argv[optind][1] == '-')); - } - - /* Decode the current option-ARGV-element. */ - - /* Check whether the ARGV-element is a long option. - - If long_only and the ARGV-element has the form "-f", where f is - a valid short option, don't consider it an abbreviated form of - a long option that starts with f. Otherwise there would be no - way to give the -f short option. - - On the other hand, if there's a long option "fubar" and - the ARGV-element is "-fu", do consider that an abbreviation of - the long option, just like "--fu", and not "-f" with arg "u". - - This distinction seems to be the most useful approach. */ - - if (longopts != NULL - && (argv[optind][1] == '-' - || (long_only && (argv[optind][2] || !my_index (optstring, argv[optind][1]))))) - { - char *nameend; - const struct option *p; - const struct option *pfound = NULL; - int exact = 0; - int ambig = 0; - int indfound = -1; - int option_index; - - for (nameend = nextchar; *nameend && *nameend != '='; nameend++) - /* Do nothing. */ ; - - /* Test all long options for either exact match - or abbreviated matches. */ - for (p = longopts, option_index = 0; p->name; p++, option_index++) - if (!strncmp (p->name, nextchar, nameend - nextchar)) - { - if ((unsigned int) (nameend - nextchar) - == (unsigned int) strlen (p->name)) - { - /* Exact match found. */ - pfound = p; - indfound = option_index; - exact = 1; - break; - } - else if (pfound == NULL) - { - /* First nonexact match found. */ - pfound = p; - indfound = option_index; - } - else - /* Second or later nonexact match found. */ - ambig = 1; - } - - if (ambig && !exact) - { - if (opterr) - fprintf (stderr, _("%s: option `%s' is ambiguous\n"), - argv[0], argv[optind]); - nextchar += strlen (nextchar); - optind++; - optopt = 0; - return '?'; - } - - if (pfound != NULL) - { - option_index = indfound; - optind++; - if (*nameend) - { - /* Don't test has_arg with >, because some C compilers don't - allow it to be used on enums. */ - if (pfound->has_arg) - optarg = nameend + 1; - else - { - if (opterr) - { - if (argv[optind - 1][1] == '-') - /* --option */ - fprintf (stderr, - _("%s: option `--%s' doesn't allow an argument\n"), - argv[0], pfound->name); - else - /* +option or -option */ - fprintf (stderr, - _("%s: option `%c%s' doesn't allow an argument\n"), - argv[0], argv[optind - 1][0], pfound->name); - - nextchar += strlen (nextchar); - - optopt = pfound->val; - return '?'; - } - } - } - else if (pfound->has_arg == 1) - { - if (optind < argc) - optarg = argv[optind++]; - else - { - if (opterr) - fprintf (stderr, - _("%s: option `%s' requires an argument\n"), - argv[0], argv[optind - 1]); - nextchar += strlen (nextchar); - optopt = pfound->val; - return optstring[0] == ':' ? ':' : '?'; - } - } - nextchar += strlen (nextchar); - if (longind != NULL) - *longind = option_index; - if (pfound->flag) - { - *(pfound->flag) = pfound->val; - return 0; - } - return pfound->val; - } - - /* Can't find it as a long option. If this is not getopt_long_only, - or the option starts with '--' or is not a valid short - option, then it's an error. - Otherwise interpret it as a short option. */ - if (!long_only || argv[optind][1] == '-' - || my_index (optstring, *nextchar) == NULL) - { - if (opterr) - { - if (argv[optind][1] == '-') - /* --option */ - fprintf (stderr, _("%s: unrecognized option `--%s'\n"), - argv[0], nextchar); - else - /* +option or -option */ - fprintf (stderr, _("%s: unrecognized option `%c%s'\n"), - argv[0], argv[optind][0], nextchar); - } - nextchar = (char *) ""; - optind++; - optopt = 0; - return '?'; - } - } - - /* Look at and handle the next short option-character. */ - - { - char c = *nextchar++; - char *temp = my_index (optstring, c); - - /* Increment `optind' when we start to process its last character. */ - if (*nextchar == '\0') - ++optind; - - if (temp == NULL || c == ':') - { - if (opterr) - { - if (posixly_correct) - /* 1003.2 specifies the format of this message. */ - fprintf (stderr, _("%s: illegal option -- %c\n"), - argv[0], c); - else - fprintf (stderr, _("%s: invalid option -- %c\n"), - argv[0], c); - } - optopt = c; - return '?'; - } - /* Convenience. Treat POSIX -W foo same as long option --foo */ - if (temp[0] == 'W' && temp[1] == ';') - { - char *nameend; - const struct option *p; - const struct option *pfound = NULL; - int exact = 0; - int ambig = 0; - int indfound = 0; - int option_index; - - /* This is an option that requires an argument. */ - if (*nextchar != '\0') - { - optarg = nextchar; - /* If we end this ARGV-element by taking the rest as an arg, - we must advance to the next element now. */ - optind++; - } - else if (optind == argc) - { - if (opterr) - { - /* 1003.2 specifies the format of this message. */ - fprintf (stderr, _("%s: option requires an argument -- %c\n"), - argv[0], c); - } - optopt = c; - if (optstring[0] == ':') - c = ':'; - else - c = '?'; - return c; - } - else - /* We already incremented `optind' once; - increment it again when taking next ARGV-elt as argument. */ - optarg = argv[optind++]; - - /* optarg is now the argument, see if it's in the - table of longopts. */ - - for (nextchar = nameend = optarg; *nameend && *nameend != '='; nameend++) - /* Do nothing. */ ; - - /* Test all long options for either exact match - or abbreviated matches. */ - for (p = longopts, option_index = 0; p->name; p++, option_index++) - if (!strncmp (p->name, nextchar, nameend - nextchar)) - { - if ((unsigned int) (nameend - nextchar) == strlen (p->name)) - { - /* Exact match found. */ - pfound = p; - indfound = option_index; - exact = 1; - break; - } - else if (pfound == NULL) - { - /* First nonexact match found. */ - pfound = p; - indfound = option_index; - } - else - /* Second or later nonexact match found. */ - ambig = 1; - } - if (ambig && !exact) - { - if (opterr) - fprintf (stderr, _("%s: option `-W %s' is ambiguous\n"), - argv[0], argv[optind]); - nextchar += strlen (nextchar); - optind++; - return '?'; - } - if (pfound != NULL) - { - option_index = indfound; - if (*nameend) - { - /* Don't test has_arg with >, because some C compilers don't - allow it to be used on enums. */ - if (pfound->has_arg) - optarg = nameend + 1; - else - { - if (opterr) - fprintf (stderr, _("\ -%s: option `-W %s' doesn't allow an argument\n"), - argv[0], pfound->name); - - nextchar += strlen (nextchar); - return '?'; - } - } - else if (pfound->has_arg == 1) - { - if (optind < argc) - optarg = argv[optind++]; - else - { - if (opterr) - fprintf (stderr, - _("%s: option `%s' requires an argument\n"), - argv[0], argv[optind - 1]); - nextchar += strlen (nextchar); - return optstring[0] == ':' ? ':' : '?'; - } - } - nextchar += strlen (nextchar); - if (longind != NULL) - *longind = option_index; - if (pfound->flag) - { - *(pfound->flag) = pfound->val; - return 0; - } - return pfound->val; - } - nextchar = NULL; - return 'W'; /* Let the application handle it. */ - } - if (temp[1] == ':') - { - if (temp[2] == ':') - { - /* This is an option that accepts an argument optionally. */ - if (*nextchar != '\0') - { - optarg = nextchar; - optind++; - } - else - optarg = NULL; - nextchar = NULL; - } - else - { - /* This is an option that requires an argument. */ - if (*nextchar != '\0') - { - optarg = nextchar; - /* If we end this ARGV-element by taking the rest as an arg, - we must advance to the next element now. */ - optind++; - } - else if (optind == argc) - { - if (opterr) - { - /* 1003.2 specifies the format of this message. */ - fprintf (stderr, - _("%s: option requires an argument -- %c\n"), - argv[0], c); - } - optopt = c; - if (optstring[0] == ':') - c = ':'; - else - c = '?'; - } - else - /* We already incremented `optind' once; - increment it again when taking next ARGV-elt as argument. */ - optarg = argv[optind++]; - nextchar = NULL; - } - } - return c; - } -} - -int -getopt (int argc, char *const *argv, const char *optstring) -{ - return _getopt_internal (argc, argv, optstring, - (const struct option *) 0, - (int *) 0, - 0); -} - -#endif /* Not ELIDE_CODE. */ - -#ifdef TEST - -/* Compile with -DTEST to make an executable for use in testing - the above definition of `getopt'. */ - -int -main (argc, argv) - int argc; - char **argv; -{ - int c; - int digit_optind = 0; - - while (1) - { - int this_option_optind = optind ? optind : 1; - - c = getopt (argc, argv, "abc:d:0123456789"); - if (c == -1) - break; - - switch (c) - { - case '0': - case '1': - case '2': - case '3': - case '4': - case '5': - case '6': - case '7': - case '8': - case '9': - if (digit_optind != 0 && digit_optind != this_option_optind) - printf ("digits occur in two different argv-elements.\n"); - digit_optind = this_option_optind; - printf ("option %c\n", c); - break; - - case 'a': - printf ("option a\n"); - break; - - case 'b': - printf ("option b\n"); - break; - - case 'c': - printf ("option c with value `%s'\n", optarg); - break; - - case '?': - break; - - default: - printf ("?? getopt returned character code 0%o ??\n", c); - } - } - - if (optind < argc) - { - printf ("non-option ARGV-elements: "); - while (optind < argc) - printf ("%s ", argv[optind++]); - printf ("\n"); - } - - exit (0); -} - -#endif /* TEST */ diff --git a/src/getopt.h b/src/getopt.h index 9268121e..cb5feba1 100644 --- a/src/getopt.h +++ b/src/getopt.h @@ -131,10 +131,7 @@ extern int getopt (); extern int getopt_long (); extern int getopt_long_only (); -extern int _getopt_internal (int argc, char *const *argv, - const char *shortopts, - const struct option *longopts, int *longind, - int long_only); +extern int _getopt_internal (); #endif /* __STDC__ */ #ifdef __cplusplus diff --git a/src/getopt1.cpp b/src/getopt1.cpp deleted file mode 100644 index b32d85bc..00000000 --- a/src/getopt1.cpp +++ /dev/null @@ -1,180 +0,0 @@ -/* getopt_long and getopt_long_only entry points for GNU getopt. - Copyright (C) 1987,88,89,90,91,92,93,94,96,97,98 - Free Software Foundation, Inc. - - NOTE: This source is derived from an old version taken from the GNU C - Library (glibc). - - This program is free software; you can redistribute it and/or modify it - under the terms of the GNU General Public License as published by the - Free Software Foundation; either version 2, or (at your option) any - later version. - - This program is distributed in the hope that it will be useful, - but WITHOUT ANY WARRANTY; without even the implied warranty of - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - GNU General Public License for more details. - - You should have received a copy of the GNU General Public License - along with this program; if not, write to the Free Software - Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, - USA. */ - -#ifdef HAVE_CONFIG_H -#include -#endif - -#include "getopt.h" - -#if !defined __STDC__ || !__STDC__ -/* This is a separate conditional since some stdc systems - reject `defined (const)'. */ -#ifndef const -#define const -#endif -#endif - -#include - -/* Comment out all this code if we are using the GNU C Library, and are not - actually compiling the library itself. This code is part of the GNU C - Library, but also included in many other GNU distributions. Compiling - and linking in this code is a waste when using the GNU C library - (especially if it is a shared library). Rather than having every GNU - program understand `configure --with-gnu-libc' and omit the object files, - it is simpler to just do this in the source for each such file. */ - -#define GETOPT_INTERFACE_VERSION 2 -#if !defined _LIBC && defined __GLIBC__ && __GLIBC__ >= 2 -#include -#if _GNU_GETOPT_INTERFACE_VERSION == GETOPT_INTERFACE_VERSION -#define ELIDE_CODE -#endif -#endif - -#ifndef ELIDE_CODE - - -/* This needs to come after some library #include - to get __GNU_LIBRARY__ defined. */ -#ifdef __GNU_LIBRARY__ -#include -#endif - -#ifndef NULL -#define NULL 0 -#endif - -int -getopt_long (int argc, char *const *argv, const char *options, const struct option *long_options, int *opt_index) -{ - return _getopt_internal (argc, argv, options, long_options, opt_index, 0); -} - -/* Like getopt_long, but '-' as well as '--' can indicate a long option. - If an option that starts with '-' (not '--') doesn't match a long option, - but does match a short option, it is parsed as a short option - instead. */ - -int -getopt_long_only (int argc, char *const *argv, const char *options, const struct option *long_options, int *opt_index) -{ - return _getopt_internal (argc, argv, options, long_options, opt_index, 1); -} - - -#endif /* Not ELIDE_CODE. */ - -#ifdef TEST - -#include - -int -main (argc, argv) - int argc; - char **argv; -{ - int c; - int digit_optind = 0; - - while (1) - { - int this_option_optind = optind ? optind : 1; - int option_index = 0; - static struct option long_options[] = - { - {"add", 1, 0, 0}, - {"append", 0, 0, 0}, - {"delete", 1, 0, 0}, - {"verbose", 0, 0, 0}, - {"create", 0, 0, 0}, - {"file", 1, 0, 0}, - {0, 0, 0, 0} - }; - - c = getopt_long (argc, argv, "abc:d:0123456789", - long_options, &option_index); - if (c == -1) - break; - - switch (c) - { - case 0: - printf ("option %s", long_options[option_index].name); - if (optarg) - printf (" with arg %s", optarg); - printf ("\n"); - break; - - case '0': - case '1': - case '2': - case '3': - case '4': - case '5': - case '6': - case '7': - case '8': - case '9': - if (digit_optind != 0 && digit_optind != this_option_optind) - printf ("digits occur in two different argv-elements.\n"); - digit_optind = this_option_optind; - printf ("option %c\n", c); - break; - - case 'a': - printf ("option a\n"); - break; - - case 'b': - printf ("option b\n"); - break; - - case 'c': - printf ("option c with value `%s'\n", optarg); - break; - - case 'd': - printf ("option d with value `%s'\n", optarg); - break; - - case '?': - break; - - default: - printf ("?? getopt returned character code 0%o ??\n", c); - } - } - - if (optind < argc) - { - printf ("non-option ARGV-elements: "); - while (optind < argc) - printf ("%s ", argv[optind++]); - printf ("\n"); - } - - exit (0); -} - -#endif /* TEST */ diff --git a/src/interframe.cpp b/src/interframe.cpp index 78ea76fd..e4a17227 100644 --- a/src/interframe.cpp +++ b/src/interframe.cpp @@ -1,648 +1,581 @@ -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#include "System.h" -#include -#include - -#ifdef MMX -extern "C" bool cpu_mmx; -#endif - -/* - * Thanks to Kawaks' Mr. K for the code - - Incorporated into vba by Anthony Di Franco -*/ - -static u8 *frm1 = NULL; -static u8 *frm2 = NULL; -static u8 *frm3 = NULL; - -extern int RGB_LOW_BITS_MASK; -extern u32 qRGB_COLOR_MASK[2]; - -static void Init() -{ - frm1 = (u8 *)calloc(322*242,4); - // 1 frame ago - frm2 = (u8 *)calloc(322*242,4); - // 2 frames ago - frm3 = (u8 *)calloc(322*242,4); - // 3 frames ago -} - -void InterframeCleanup() -{ - if(frm1) - free(frm1); - if(frm2) - free(frm2); - if(frm3) - free(frm3); - frm1 = frm2 = frm3 = NULL; -} - -#ifdef MMX -static void SmartIB_MMX(u8 *srcPtr, u32 srcPitch, int width, int height) -{ - u16 *src0 = (u16 *)srcPtr; - u16 *src1 = (u16 *)frm1; - u16 *src2 = (u16 *)frm2; - u16 *src3 = (u16 *)frm3; - - int sPitch = srcPitch >> 1; - if (width > sPitch) width = sPitch; - sPitch -= width; - int count = width >> 2; - - for(int i = 0; i < height; i++) { -#ifdef __GNUC__ - asm volatile ( - "push %4\n" - "movq 0(%5), %%mm7\n" // colorMask - "0:\n" - "movq 0(%0), %%mm0\n" // src0 - "movq 0(%1), %%mm1\n" // src1 - "movq 0(%2), %%mm2\n" // src2 - "movq 0(%3), %%mm3\n" // src3 - "movq %%mm0, 0(%3)\n" // src3 = src0 - "movq %%mm0, %%mm4\n" - "movq %%mm1, %%mm5\n" - "pcmpeqw %%mm2, %%mm5\n" // src1 == src2 (A) - "pcmpeqw %%mm3, %%mm4\n" // src3 == src0 (B) - "por %%mm5, %%mm4\n" // A | B - "movq %%mm2, %%mm5\n" - "pcmpeqw %%mm0, %%mm5\n" // src0 == src2 (C) - "pcmpeqw %%mm1, %%mm3\n" // src1 == src3 (D) - "por %%mm3, %%mm5\n" // C|D - "pandn %%mm5, %%mm4\n" // (!(A|B))&(C|D) - "movq %%mm0, %%mm2\n" - "pand %%mm7, %%mm2\n" // color & colorMask - "pand %%mm7, %%mm1\n" // src1 & colorMask - "psrlw $1, %%mm2\n" // (color & colorMask) >> 1 (E) - "psrlw $1, %%mm1\n" // (src & colorMask) >> 1 (F) - "paddw %%mm2, %%mm1\n" // E+F - "pand %%mm4, %%mm1\n" // (E+F) & res - "pandn %%mm0, %%mm4\n" // color& !res - - "por %%mm1, %%mm4\n" - "movq %%mm4, 0(%0)\n" // src0 = res - - "addl $8, %0\n" - "addl $8, %1\n" - "addl $8, %2\n" - "addl $8, %3\n" - - "decl %4\n" - "jnz 0b\n" - "pop %4\n" - "emms\n" - : "+r" (src0), "+r" (src1), "+r" (src2), "+r" (src3) - : "r" (count), "r" (qRGB_COLOR_MASK) - ); -#else - __asm { - movq mm7, qword ptr [qRGB_COLOR_MASK]; - mov eax, src0; - mov ebx, src1; - mov ecx, src2; - mov edx, src3; - mov edi, count; - label0: - movq mm0, qword ptr [eax]; // src0 - movq mm1, qword ptr [ebx]; // src1 - movq mm2, qword ptr [ecx]; // src2 - movq mm3, qword ptr [edx]; // src3 - movq qword ptr [edx], mm0; // src3 = src0 - movq mm4, mm0; - movq mm5, mm1; - pcmpeqw mm5, mm2; // src1 == src2 (A) - pcmpeqw mm4, mm3; // src3 == src0 (B) - por mm4, mm5; // A | B - movq mm5, mm2; - pcmpeqw mm5, mm0; // src0 == src2 (C) - pcmpeqw mm3, mm1; // src1 == src3 (D) - por mm5, mm3; // C|D - pandn mm4, mm5; // (!(A|B))&(C|D) - movq mm2, mm0; - pand mm2, mm7; // color & colorMask - pand mm1, mm7; // src1 & colorMask - psrlw mm2, 1; // (color & colorMask) >> 1 (E) - psrlw mm1, 1; // (src & colorMask) >> 1 (F) - paddw mm1, mm2; // E+F - pand mm1, mm4; // (E+F) & res - pandn mm4, mm0; // color & !res - - por mm4, mm1; - movq qword ptr [eax], mm4; // src0 = res - - add eax, 8; - add ebx, 8; - add ecx, 8; - add edx, 8; - - dec edi; - jnz label0; - mov src0, eax; - mov src1, ebx; - mov src2, ecx; - mov src3, edx; - emms; - } -#endif - src0+= sPitch; - src1+= sPitch; - src2+= sPitch; - src3+= sPitch; - } - - /* Swap buffers around */ - u8 *temp = frm1; - frm1 = frm3; - frm3 = frm2; - frm2 = temp; -} -#endif - -void SmartIB(u8 *srcPtr, u32 srcPitch, int width, int height) -{ - if(frm1 == NULL) { - Init(); - } -#ifdef MMX - if(cpu_mmx) { - SmartIB_MMX(srcPtr, srcPitch, width, height); - return; - } -#endif - - u16 colorMask = ~RGB_LOW_BITS_MASK; - - u16 *src0 = (u16 *)srcPtr; - u16 *src1 = (u16 *)frm1; - u16 *src2 = (u16 *)frm2; - u16 *src3 = (u16 *)frm3; - - int sPitch = srcPitch >> 1; - if (width > sPitch) width = sPitch; - sPitch -= width; - - int pos = 0; - for (int j = 0; j < height; j++) { - for (int i = 0; i < width; i++) { - u16 color = src0[pos]; - src0[pos] = - (src1[pos] != src2[pos]) && - (src3[pos] != color) && - ((color == src2[pos]) || (src1[pos] == src3[pos])) - ? (((color & colorMask) >> 1) + ((src1[pos] & colorMask) >> 1)) : - color; - src3[pos] = color; /* oldest buffer now holds newest frame */ - pos++; - } - pos += sPitch; - } - - /* Swap buffers around */ - u8 *temp = frm1; - frm1 = frm3; - frm3 = frm2; - frm2 = temp; -} - -#ifdef MMX -static void SmartIB32_MMX(u8 *srcPtr, u32 srcPitch, int width, int height) -{ - u32 *src0 = (u32 *)srcPtr; - u32 *src1 = (u32 *)frm1; - u32 *src2 = (u32 *)frm2; - u32 *src3 = (u32 *)frm3; - - int sPitch = srcPitch >> 2; - if (width > sPitch) width = sPitch; - sPitch -= width; - int count = width >> 1; - - for(int i = 0; i < height; i++) { -#ifdef __GNUC__ - asm volatile ( - "push %4\n" - "movq 0(%5), %%mm7\n" // colorMask - "0:\n" - "movq 0(%0), %%mm0\n" // src0 - "movq 0(%1), %%mm1\n" // src1 - "movq 0(%2), %%mm2\n" // src2 - "movq 0(%3), %%mm3\n" // src3 - "movq %%mm0, 0(%3)\n" // src3 = src0 - "movq %%mm0, %%mm4\n" - "movq %%mm1, %%mm5\n" - "pcmpeqd %%mm2, %%mm5\n" // src1 == src2 (A) - "pcmpeqd %%mm3, %%mm4\n" // src3 == src0 (B) - "por %%mm5, %%mm4\n" // A | B - "movq %%mm2, %%mm5\n" - "pcmpeqd %%mm0, %%mm5\n" // src0 == src2 (C) - "pcmpeqd %%mm1, %%mm3\n" // src1 == src3 (D) - "por %%mm3, %%mm5\n" // C|D - "pandn %%mm5, %%mm4\n" // (!(A|B))&(C|D) - "movq %%mm0, %%mm2\n" - "pand %%mm7, %%mm2\n" // color & colorMask - "pand %%mm7, %%mm1\n" // src1 & colorMask - "psrld $1, %%mm2\n" // (color & colorMask) >> 1 (E) - "psrld $1, %%mm1\n" // (src & colorMask) >> 1 (F) - "paddd %%mm2, %%mm1\n" // E+F - "pand %%mm4, %%mm1\n" // (E+F) & res - "pandn %%mm0, %%mm4\n" // color& !res - - "por %%mm1, %%mm4\n" - "movq %%mm4, 0(%0)\n" // src0 = res - - "addl $8, %0\n" - "addl $8, %1\n" - "addl $8, %2\n" - "addl $8, %3\n" - - "decl %4\n" - "jnz 0b\n" - "pop %4\n" - "emms\n" - : "+r" (src0), "+r" (src1), "+r" (src2), "+r" (src3) - : "r" (count), "r" (qRGB_COLOR_MASK) - ); -#else - __asm { - movq mm7, qword ptr [qRGB_COLOR_MASK]; - mov eax, src0; - mov ebx, src1; - mov ecx, src2; - mov edx, src3; - mov edi, count; - label0: - movq mm0, qword ptr [eax]; // src0 - movq mm1, qword ptr [ebx]; // src1 - movq mm2, qword ptr [ecx]; // src2 - movq mm3, qword ptr [edx]; // src3 - movq qword ptr [edx], mm0; // src3 = src0 - movq mm4, mm0; - movq mm5, mm1; - pcmpeqd mm5, mm2; // src1 == src2 (A) - pcmpeqd mm4, mm3; // src3 == src0 (B) - por mm4, mm5; // A | B - movq mm5, mm2; - pcmpeqd mm5, mm0; // src0 == src2 (C) - pcmpeqd mm3, mm1; // src1 == src3 (D) - por mm5, mm3; // C|D - pandn mm4, mm5; // (!(A|B))&(C|D) - movq mm2, mm0; - pand mm2, mm7; // color & colorMask - pand mm1, mm7; // src1 & colorMask - psrld mm2, 1; // (color & colorMask) >> 1 (E) - psrld mm1, 1; // (src & colorMask) >> 1 (F) - paddd mm1, mm2; // E+F - pand mm1, mm4; // (E+F) & res - pandn mm4, mm0; // color & !res - - por mm4, mm1; - movq qword ptr [eax], mm4; // src0 = res - - add eax, 8; - add ebx, 8; - add ecx, 8; - add edx, 8; - - dec edi; - jnz label0; - mov src0, eax; - mov src1, ebx; - mov src2, ecx; - mov src3, edx; - emms; - } -#endif - - src0 += sPitch; - src1 += sPitch; - src2 += sPitch; - src3 += sPitch; - } - /* Swap buffers around */ - u8 *temp = frm1; - frm1 = frm3; - frm3 = frm2; - frm2 = temp; -} -#endif - -void SmartIB32(u8 *srcPtr, u32 srcPitch, int width, int height) -{ - if(frm1 == NULL) { - Init(); - } -#ifdef MMX - if(cpu_mmx) { - SmartIB32_MMX(srcPtr, srcPitch, width, height); - return; - } -#endif - - u32 *src0 = (u32 *)srcPtr; - u32 *src1 = (u32 *)frm1; - u32 *src2 = (u32 *)frm2; - u32 *src3 = (u32 *)frm3; - - u32 colorMask = 0xfefefe; - - int sPitch = srcPitch >> 2; - if (width > sPitch) width = sPitch; - sPitch -= width; - int pos = 0; - - for (int j = 0; j < height; j++) { - for (int i = 0; i < width; i++) { - u32 color = src0[pos]; - src0[pos] = - (src1[pos] != src2[pos]) && - (src3[pos] != color) && - ((color == src2[pos]) || (src1[pos] == src3[pos])) - ? (((color & colorMask) >> 1) + ((src1[pos] & colorMask) >> 1)) : - color; - src3[pos] = color; /* oldest buffer now holds newest frame */ - pos++; - } - pos += sPitch; - } - - /* Swap buffers around */ - u8 *temp = frm1; - frm1 = frm3; - frm3 = frm2; - frm2 = temp; -} - -#ifdef MMX -static void MotionBlurIB_MMX(u8 *srcPtr, u32 srcPitch, int width, int height) -{ - u16 *src0 = (u16 *)srcPtr; - u16 *src1 = (u16 *)frm1; - - int sPitch = srcPitch >> 1; - if (width > sPitch) width = sPitch; - sPitch -= width; - int count = width >> 2; - - for(int i = 0; i < height; i++) { -#ifdef __GNUC__ - asm volatile ( - "push %2\n" - "movq 0(%3), %%mm7\n" // colorMask - "0:\n" - "movq 0(%0), %%mm0\n" // src0 - "movq 0(%1), %%mm1\n" // src1 - "movq %%mm0, 0(%1)\n" // src1 = src0 - "pand %%mm7, %%mm0\n" // color & colorMask - "pand %%mm7, %%mm1\n" // src1 & colorMask - "psrlw $1, %%mm0\n" // (color & colorMask) >> 1 (E) - "psrlw $1, %%mm1\n" // (src & colorMask) >> 1 (F) - "paddw %%mm1, %%mm0\n" // E+F - - "movq %%mm0, 0(%0)\n" // src0 = res - - "addl $8, %0\n" - "addl $8, %1\n" - - "decl %2\n" - "jnz 0b\n" - "pop %2\n" - "emms\n" - : "+r" (src0), "+r" (src1) - : "r" (count), "r" (qRGB_COLOR_MASK) - ); -#else - __asm { - movq mm7, qword ptr [qRGB_COLOR_MASK]; - mov eax, src0; - mov ebx, src1; - mov edi, count; - label0: - movq mm0, qword ptr [eax]; // src0 - movq mm1, qword ptr [ebx]; // src1 - movq qword ptr [ebx], mm0; // src1 = src0 - pand mm0, mm7; // color & colorMask - pand mm1, mm7; // src1 & colorMask - psrlw mm0, 1; // (color & colorMask) >> 1 (E) - psrlw mm1, 1; // (src & colorMask) >> 1 (F) - paddw mm0, mm1; // E+F - - movq qword ptr [eax], mm0; // src0 = res - - add eax, 8; - add ebx, 8; - - dec edi; - jnz label0; - mov src0, eax; - mov src1, ebx; - emms; - } -#endif - src0 += sPitch; - src1 += sPitch; - } -} -#endif - -void MotionBlurIB(u8 *srcPtr, u32 srcPitch, int width, int height) -{ - if(frm1 == NULL) { - Init(); - } - -#ifdef MMX - if(cpu_mmx) { - MotionBlurIB_MMX(srcPtr, srcPitch, width, height); - return; - } -#endif - - u16 colorMask = ~RGB_LOW_BITS_MASK; - - u16 *src0 = (u16 *)srcPtr; - u16 *src1 = (u16 *)frm1; - - int sPitch = srcPitch >> 1; - if (width > sPitch) width = sPitch; - sPitch -= width; - - int pos = 0; - for (int j = 0; j < height; j++) { - for (int i = 0; i < width; i++) { - u16 color = src0[pos]; - src0[pos] = - (((color & colorMask) >> 1) + ((src1[pos] & colorMask) >> 1)); - src1[pos] = color; - pos++; - } - pos += sPitch; - } -} - -#ifdef MMX -static void MotionBlurIB32_MMX(u8 *srcPtr, u32 srcPitch, int width, int height) -{ - u32 *src0 = (u32 *)srcPtr; - u32 *src1 = (u32 *)frm1; - - int sPitch = srcPitch >> 2; - if (width > sPitch) width = sPitch; - sPitch -= width; - int count = width >> 1; - - for(int i = 0; i < height; i++) { -#ifdef __GNUC__ - asm volatile ( - "push %2\n" - "movq 0(%3), %%mm7\n" // colorMask - "0:\n" - "movq 0(%0), %%mm0\n" // src0 - "movq 0(%1), %%mm1\n" // src1 - "movq %%mm0, 0(%1)\n" // src1 = src0 - "pand %%mm7, %%mm0\n" // color & colorMask - "pand %%mm7, %%mm1\n" // src1 & colorMask - "psrld $1, %%mm0\n" // (color & colorMask) >> 1 (E) - "psrld $1, %%mm1\n" // (src & colorMask) >> 1 (F) - "paddd %%mm1, %%mm0\n" // E+F - - "movq %%mm0, 0(%0)\n" // src0 = res - - "addl $8, %0\n" - "addl $8, %1\n" - - "decl %2\n" - "jnz 0b\n" - "pop %2\n" - "emms\n" - : "+r" (src0), "+r" (src1) - : "r" (count), "r" (qRGB_COLOR_MASK) - ); -#else - __asm { - movq mm7, qword ptr [qRGB_COLOR_MASK]; - mov eax, src0; - mov ebx, src1; - mov edi, count; - label0: - movq mm0, qword ptr [eax]; // src0 - movq mm1, qword ptr [ebx]; // src1 - movq qword ptr [ebx], mm0; // src1 = src0 - pand mm0, mm7; // color & colorMask - pand mm1, mm7; // src1 & colorMask - psrld mm0, 1; // (color & colorMask) >> 1 (E) - psrld mm1, 1; // (src & colorMask) >> 1 (F) - paddd mm0, mm1; // E+F - - movq qword ptr [eax], mm0; // src0 = res - - add eax, 8; - add ebx, 8; - - dec edi; - jnz label0; - mov src0, eax; - mov src1, ebx; - emms; - } -#endif - src0 += sPitch; - src1 += sPitch; - } -} -#endif - -void MotionBlurIB32(u8 *srcPtr, u32 srcPitch, int width, int height) -{ - if(frm1 == NULL) { - Init(); - } - -#ifdef MMX - if(cpu_mmx) { - MotionBlurIB32_MMX(srcPtr, srcPitch, width, height); - return; - } -#endif - - u32 *src0 = (u32 *)srcPtr; - u32 *src1 = (u32 *)frm1; - - u32 colorMask = 0xfefefe; - - int sPitch = srcPitch >> 2; - if (width > sPitch) width = sPitch; - sPitch -= width; - int pos = 0; - - for (int j = 0; j < height; j++) { - for (int i = 0; i < width; i++) { - u32 color = src0[pos]; - src0[pos] = (((color & colorMask) >> 1) + - ((src1[pos] & colorMask) >> 1)); - src1[pos] = color; - pos++; - } - pos += sPitch; - } -} - -static int count = 0; - -void InterlaceIB(u8 *srcPtr, u32 srcPitch, int width, int height) -{ - if(frm1 == NULL) { - Init(); - } - - u16 colorMask = ~RGB_LOW_BITS_MASK; - - u16 *src0 = (u16 *)srcPtr; - u16 *src1 = (u16 *)frm1; - - int sPitch = srcPitch >> 1; - - int pos = 0; - for (int j = 0; j < height; j++) { - bool render = count ? (j & 1) != 0 : (j & 1) == 0; - if(render) { - for (int i = 0; i < sPitch; i++) { - u16 color = src0[pos]; - src0[pos] = - (((color & colorMask) >> 1) + ((((src1[pos] & colorMask) >> 1) & colorMask) >> 1)); - src1[pos] = color; - pos++; - } - } else { - for (int i = 0; i < sPitch; i++) { - u16 color = src0[pos]; - src0[pos] = - (((((color & colorMask) >> 1) & colorMask) >> 1) + ((src1[pos] & colorMask) >> 1)); - src1[pos] = color; - pos++; - } - } - } - count = count ^ 1; -} +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#include "System.h" +#include +#include + +#ifdef MMX +extern "C" bool cpu_mmx; +#endif + +/* + * Thanks to Kawaks' Mr. K for the code + + Incorporated into vba by Anthony Di Franco +*/ + +static u8 *frm1 = NULL; +static u8 *frm2 = NULL; +static u8 *frm3 = NULL; + +extern int RGB_LOW_BITS_MASK; +extern u32 qRGB_COLOR_MASK[2]; + +static void Init() +{ + frm1 = (u8 *)calloc(322*242,4); + // 1 frame ago + frm2 = (u8 *)calloc(322*242,4); + // 2 frames ago + frm3 = (u8 *)calloc(322*242,4); + // 3 frames ago +} + +void InterframeCleanup() +{ + if(frm1) + free(frm1); + if(frm2) + free(frm2); + if(frm3) + free(frm3); + frm1 = frm2 = frm3 = NULL; +} + +#ifdef MMX +static void SmartIB_MMX(u8 *srcPtr, u32 srcPitch, int width, int height) +{ + u16 *src0 = (u16 *)srcPtr; + u16 *src1 = (u16 *)frm1; + u16 *src2 = (u16 *)frm2; + u16 *src3 = (u16 *)frm3; + + int count = width >> 2; + + for(int i = 0; i < height; i++) { +#ifdef __GNUC__ + asm volatile ( + "push %4\n" + "movq 0(%5), %%mm7\n" // colorMask + "0:\n" + "movq 0(%0), %%mm0\n" // src0 + "movq 0(%1), %%mm1\n" // src1 + "movq 0(%2), %%mm2\n" // src2 + "movq 0(%3), %%mm3\n" // src3 + "movq %%mm0, 0(%3)\n" // src3 = src0 + "movq %%mm0, %%mm4\n" + "movq %%mm1, %%mm5\n" + "pcmpeqw %%mm2, %%mm5\n" // src1 == src2 (A) + "pcmpeqw %%mm3, %%mm4\n" // src3 == src0 (B) + "por %%mm5, %%mm4\n" // A | B + "movq %%mm2, %%mm5\n" + "pcmpeqw %%mm0, %%mm5\n" // src0 == src2 (C) + "pcmpeqw %%mm1, %%mm3\n" // src1 == src3 (D) + "por %%mm3, %%mm5\n" // C|D + "pandn %%mm5, %%mm4\n" // (!(A|B))&(C|D) + "movq %%mm0, %%mm2\n" + "pand %%mm7, %%mm2\n" // color & colorMask + "pand %%mm7, %%mm1\n" // src1 & colorMask + "psrlw $1, %%mm2\n" // (color & colorMask) >> 1 (E) + "psrlw $1, %%mm1\n" // (src & colorMask) >> 1 (F) + "paddw %%mm2, %%mm1\n" // E+F + "pand %%mm4, %%mm1\n" // (E+F) & res + "pandn %%mm0, %%mm4\n" // color& !res + + "por %%mm1, %%mm4\n" + "movq %%mm4, 0(%0)\n" // src0 = res + + "addl $8, %0\n" + "addl $8, %1\n" + "addl $8, %2\n" + "addl $8, %3\n" + + "decl %4\n" + "jnz 0b\n" + "pop %4\n" + "emms\n" + : "+r" (src0), "+r" (src1), "+r" (src2), "+r" (src3) + : "r" (count), "r" (qRGB_COLOR_MASK) + ); +#else + __asm { + movq mm7, qword ptr [qRGB_COLOR_MASK]; + mov eax, src0; + mov ebx, src1; + mov ecx, src2; + mov edx, src3; + mov edi, count; + label0: + movq mm0, qword ptr [eax]; // src0 + movq mm1, qword ptr [ebx]; // src1 + movq mm2, qword ptr [ecx]; // src2 + movq mm3, qword ptr [edx]; // src3 + movq qword ptr [edx], mm0; // src3 = src0 + movq mm4, mm0; + movq mm5, mm1; + pcmpeqw mm5, mm2; // src1 == src2 (A) + pcmpeqw mm4, mm3; // src3 == src0 (B) + por mm4, mm5; // A | B + movq mm5, mm2; + pcmpeqw mm5, mm0; // src0 == src2 (C) + pcmpeqw mm3, mm1; // src1 == src3 (D) + por mm5, mm3; // C|D + pandn mm4, mm5; // (!(A|B))&(C|D) + movq mm2, mm0; + pand mm2, mm7; // color & colorMask + pand mm1, mm7; // src1 & colorMask + psrlw mm2, 1; // (color & colorMask) >> 1 (E) + psrlw mm1, 1; // (src & colorMask) >> 1 (F) + paddw mm1, mm2; // E+F + pand mm1, mm4; // (E+F) & res + pandn mm4, mm0; // color & !res + + por mm4, mm1; + movq qword ptr [eax], mm4; // src0 = res + + add eax, 8; + add ebx, 8; + add ecx, 8; + add edx, 8; + + dec edi; + jnz label0; + mov src0, eax; + mov src1, ebx; + mov src2, ecx; + mov src3, edx; + emms; + } +#endif + src0+=2; + src1+=2; + src2+=2; + src3+=2; + } + + /* Swap buffers around */ + u8 *temp = frm1; + frm1 = frm3; + frm3 = frm2; + frm2 = temp; +} +#endif + +void SmartIB(u8 *srcPtr, u32 srcPitch, int width, int height) +{ + if(frm1 == NULL) { + Init(); + } +#ifdef MMX + if(cpu_mmx) { + SmartIB_MMX(srcPtr, srcPitch, width, height); + return; + } +#endif + + u16 colorMask = ~RGB_LOW_BITS_MASK; + + u16 *src0 = (u16 *)srcPtr; + u16 *src1 = (u16 *)frm1; + u16 *src2 = (u16 *)frm2; + u16 *src3 = (u16 *)frm3; + + int sPitch = srcPitch >> 1; + + int pos = 0; + for (int j = 0; j < height; j++) + for (int i = 0; i < sPitch; i++) { + u16 color = src0[pos]; + src0[pos] = + (src1[pos] != src2[pos]) && + (src3[pos] != color) && + ((color == src2[pos]) || (src1[pos] == src3[pos])) + ? (((color & colorMask) >> 1) + ((src1[pos] & colorMask) >> 1)) : + color; + src3[pos] = color; /* oldest buffer now holds newest frame */ + pos++; + } + + /* Swap buffers around */ + u8 *temp = frm1; + frm1 = frm3; + frm3 = frm2; + frm2 = temp; +} + +#ifdef MMX +static void SmartIB32_MMX(u8 *srcPtr, u32 srcPitch, int width, int height) +{ + u32 *src0 = (u32 *)srcPtr; + u32 *src1 = (u32 *)frm1; + u32 *src2 = (u32 *)frm2; + u32 *src3 = (u32 *)frm3; + + int count = width >> 1; + + for(int i = 0; i < height; i++) { +#ifdef __GNUC__ + asm volatile ( + "push %4\n" + "movq 0(%5), %%mm7\n" // colorMask + "0:\n" + "movq 0(%0), %%mm0\n" // src0 + "movq 0(%1), %%mm1\n" // src1 + "movq 0(%2), %%mm2\n" // src2 + "movq 0(%3), %%mm3\n" // src3 + "movq %%mm0, 0(%3)\n" // src3 = src0 + "movq %%mm0, %%mm4\n" + "movq %%mm1, %%mm5\n" + "pcmpeqd %%mm2, %%mm5\n" // src1 == src2 (A) + "pcmpeqd %%mm3, %%mm4\n" // src3 == src0 (B) + "por %%mm5, %%mm4\n" // A | B + "movq %%mm2, %%mm5\n" + "pcmpeqd %%mm0, %%mm5\n" // src0 == src2 (C) + "pcmpeqd %%mm1, %%mm3\n" // src1 == src3 (D) + "por %%mm3, %%mm5\n" // C|D + "pandn %%mm5, %%mm4\n" // (!(A|B))&(C|D) + "movq %%mm0, %%mm2\n" + "pand %%mm7, %%mm2\n" // color & colorMask + "pand %%mm7, %%mm1\n" // src1 & colorMask + "psrld $1, %%mm2\n" // (color & colorMask) >> 1 (E) + "psrld $1, %%mm1\n" // (src & colorMask) >> 1 (F) + "paddd %%mm2, %%mm1\n" // E+F + "pand %%mm4, %%mm1\n" // (E+F) & res + "pandn %%mm0, %%mm4\n" // color& !res + + "por %%mm1, %%mm4\n" + "movq %%mm4, 0(%0)\n" // src0 = res + + "addl $8, %0\n" + "addl $8, %1\n" + "addl $8, %2\n" + "addl $8, %3\n" + + "decl %4\n" + "jnz 0b\n" + "pop %4\n" + "emms\n" + : "+r" (src0), "+r" (src1), "+r" (src2), "+r" (src3) + : "r" (count), "r" (qRGB_COLOR_MASK) + ); +#else + __asm { + movq mm7, qword ptr [qRGB_COLOR_MASK]; + mov eax, src0; + mov ebx, src1; + mov ecx, src2; + mov edx, src3; + mov edi, count; + label0: + movq mm0, qword ptr [eax]; // src0 + movq mm1, qword ptr [ebx]; // src1 + movq mm2, qword ptr [ecx]; // src2 + movq mm3, qword ptr [edx]; // src3 + movq qword ptr [edx], mm0; // src3 = src0 + movq mm4, mm0; + movq mm5, mm1; + pcmpeqd mm5, mm2; // src1 == src2 (A) + pcmpeqd mm4, mm3; // src3 == src0 (B) + por mm4, mm5; // A | B + movq mm5, mm2; + pcmpeqd mm5, mm0; // src0 == src2 (C) + pcmpeqd mm3, mm1; // src1 == src3 (D) + por mm5, mm3; // C|D + pandn mm4, mm5; // (!(A|B))&(C|D) + movq mm2, mm0; + pand mm2, mm7; // color & colorMask + pand mm1, mm7; // src1 & colorMask + psrld mm2, 1; // (color & colorMask) >> 1 (E) + psrld mm1, 1; // (src & colorMask) >> 1 (F) + paddd mm1, mm2; // E+F + pand mm1, mm4; // (E+F) & res + pandn mm4, mm0; // color & !res + + por mm4, mm1; + movq qword ptr [eax], mm4; // src0 = res + + add eax, 8; + add ebx, 8; + add ecx, 8; + add edx, 8; + + dec edi; + jnz label0; + mov src0, eax; + mov src1, ebx; + mov src2, ecx; + mov src3, edx; + emms; + } +#endif + + src0++; + src1++; + src2++; + src3++; + } + /* Swap buffers around */ + u8 *temp = frm1; + frm1 = frm3; + frm3 = frm2; + frm2 = temp; +} +#endif + +void SmartIB32(u8 *srcPtr, u32 srcPitch, int width, int height) +{ + if(frm1 == NULL) { + Init(); + } +#ifdef MMX + if(cpu_mmx) { + SmartIB32_MMX(srcPtr, srcPitch, width, height); + return; + } +#endif + + u32 *src0 = (u32 *)srcPtr; + u32 *src1 = (u32 *)frm1; + u32 *src2 = (u32 *)frm2; + u32 *src3 = (u32 *)frm3; + + u32 colorMask = 0xfefefe; + + int sPitch = srcPitch >> 2; + int pos = 0; + + for (int j = 0; j < height; j++) + for (int i = 0; i < sPitch; i++) { + u32 color = src0[pos]; + src0[pos] = + (src1[pos] != src2[pos]) && + (src3[pos] != color) && + ((color == src2[pos]) || (src1[pos] == src3[pos])) + ? (((color & colorMask) >> 1) + ((src1[pos] & colorMask) >> 1)) : + color; + src3[pos] = color; /* oldest buffer now holds newest frame */ + pos++; + } + + /* Swap buffers around */ + u8 *temp = frm1; + frm1 = frm3; + frm3 = frm2; + frm2 = temp; +} + +#ifdef MMX +static void MotionBlurIB_MMX(u8 *srcPtr, u32 srcPitch, int width, int height) +{ + u16 *src0 = (u16 *)srcPtr; + u16 *src1 = (u16 *)frm1; + + int count = width >> 2; + + for(int i = 0; i < height; i++) { +#ifdef __GNUC__ + asm volatile ( + "push %2\n" + "movq 0(%3), %%mm7\n" // colorMask + "0:\n" + "movq 0(%0), %%mm0\n" // src0 + "movq 0(%1), %%mm1\n" // src1 + "movq %%mm0, 0(%1)\n" // src1 = src0 + "pand %%mm7, %%mm0\n" // color & colorMask + "pand %%mm7, %%mm1\n" // src1 & colorMask + "psrlw $1, %%mm0\n" // (color & colorMask) >> 1 (E) + "psrlw $1, %%mm1\n" // (src & colorMask) >> 1 (F) + "paddw %%mm1, %%mm0\n" // E+F + + "movq %%mm0, 0(%0)\n" // src0 = res + + "addl $8, %0\n" + "addl $8, %1\n" + + "decl %2\n" + "jnz 0b\n" + "pop %2\n" + "emms\n" + : "+r" (src0), "+r" (src1) + : "r" (count), "r" (qRGB_COLOR_MASK) + ); +#else + __asm { + movq mm7, qword ptr [qRGB_COLOR_MASK]; + mov eax, src0; + mov ebx, src1; + mov edi, count; + label0: + movq mm0, qword ptr [eax]; // src0 + movq mm1, qword ptr [ebx]; // src1 + movq qword ptr [ebx], mm0; // src1 = src0 + pand mm0, mm7; // color & colorMask + pand mm1, mm7; // src1 & colorMask + psrlw mm0, 1; // (color & colorMask) >> 1 (E) + psrlw mm1, 1; // (src & colorMask) >> 1 (F) + paddw mm0, mm1; // E+F + + movq qword ptr [eax], mm0; // src0 = res + + add eax, 8; + add ebx, 8; + + dec edi; + jnz label0; + mov src0, eax; + mov src1, ebx; + emms; + } +#endif + src0+=2; + src1+=2; + } +} +#endif + +void MotionBlurIB(u8 *srcPtr, u32 srcPitch, int width, int height) +{ + if(frm1 == NULL) { + Init(); + } + +#ifdef MMX + if(cpu_mmx) { + MotionBlurIB_MMX(srcPtr, srcPitch, width, height); + return; + } +#endif + + u16 colorMask = ~RGB_LOW_BITS_MASK; + + u16 *src0 = (u16 *)srcPtr; + u16 *src1 = (u16 *)frm1; + + int sPitch = srcPitch >> 1; + + int pos = 0; + for (int j = 0; j < height; j++) + for (int i = 0; i < sPitch; i++) { + u16 color = src0[pos]; + src0[pos] = + (((color & colorMask) >> 1) + ((src1[pos] & colorMask) >> 1)); + src1[pos] = color; + pos++; + } +} + +#ifdef MMX +static void MotionBlurIB32_MMX(u8 *srcPtr, u32 srcPitch, int width, int height) +{ + u32 *src0 = (u32 *)srcPtr; + u32 *src1 = (u32 *)frm1; + + int count = width >> 1; + + for(int i = 0; i < height; i++) { +#ifdef __GNUC__ + asm volatile ( + "push %2\n" + "movq 0(%3), %%mm7\n" // colorMask + "0:\n" + "movq 0(%0), %%mm0\n" // src0 + "movq 0(%1), %%mm1\n" // src1 + "movq %%mm0, 0(%1)\n" // src1 = src0 + "pand %%mm7, %%mm0\n" // color & colorMask + "pand %%mm7, %%mm1\n" // src1 & colorMask + "psrld $1, %%mm0\n" // (color & colorMask) >> 1 (E) + "psrld $1, %%mm1\n" // (src & colorMask) >> 1 (F) + "paddd %%mm1, %%mm0\n" // E+F + + "movq %%mm0, 0(%0)\n" // src0 = res + + "addl $8, %0\n" + "addl $8, %1\n" + + "decl %2\n" + "jnz 0b\n" + "pop %2\n" + "emms\n" + : "+r" (src0), "+r" (src1) + : "r" (count), "r" (qRGB_COLOR_MASK) + ); +#else + __asm { + movq mm7, qword ptr [qRGB_COLOR_MASK]; + mov eax, src0; + mov ebx, src1; + mov edi, count; + label0: + movq mm0, qword ptr [eax]; // src0 + movq mm1, qword ptr [ebx]; // src1 + movq qword ptr [ebx], mm0; // src1 = src0 + pand mm0, mm7; // color & colorMask + pand mm1, mm7; // src1 & colorMask + psrld mm0, 1; // (color & colorMask) >> 1 (E) + psrld mm1, 1; // (src & colorMask) >> 1 (F) + paddd mm0, mm1; // E+F + + movq qword ptr [eax], mm0; // src0 = res + + add eax, 8; + add ebx, 8; + + dec edi; + jnz label0; + mov src0, eax; + mov src1, ebx; + emms; + } +#endif + src0++; + src1++; + } +} +#endif + +void MotionBlurIB32(u8 *srcPtr, u32 srcPitch, int width, int height) +{ + if(frm1 == NULL) { + Init(); + } + +#ifdef MMX + if(cpu_mmx) { + MotionBlurIB32_MMX(srcPtr, srcPitch, width, height); + return; + } +#endif + + u32 *src0 = (u32 *)srcPtr; + u32 *src1 = (u32 *)frm1; + + u32 colorMask = 0xfefefe; + + int sPitch = srcPitch >> 2; + int pos = 0; + + for (int j = 0; j < height; j++) + for (int i = 0; i < sPitch; i++) { + u32 color = src0[pos]; + src0[pos] = (((color & colorMask) >> 1) + + ((src1[pos] & colorMask) >> 1)); + src1[pos] = color; + pos++; + } +} diff --git a/src/interp.h b/src/interp.h index b899ea73..9df8856a 100644 --- a/src/interp.h +++ b/src/interp.h @@ -40,11 +40,6 @@ static unsigned interp_mask[2]; static unsigned interp_bits_per_pixel; -typedef unsigned short interp_uint16; -typedef unsigned int interp_uint32; - -#define restrict - #define INTERP_16_MASK_1(v) (v & interp_mask[0]) #define INTERP_16_MASK_2(v) (v & interp_mask[1]) diff --git a/src/memgzio.cpp b/src/memgzio.c similarity index 93% rename from src/memgzio.cpp rename to src/memgzio.c index 2d1d2040..528539c3 100644 --- a/src/memgzio.cpp +++ b/src/memgzio.c @@ -9,7 +9,7 @@ * Adapted from original gzio.c from zlib library by Forgotten */ -/* @(#) $Id: memgzio.c,v 1.3 2004/01/17 23:07:32 kxu Exp $ */ +/* @(#) $Id: memgzio.c,v 1.5 2006/06/06 21:04:20 spacy51 Exp $ */ #include #include @@ -125,7 +125,7 @@ local size_t memWrite(const void *buffer, size_t size, size_t count, total = file->available; } memcpy(file->next, buffer, total); - file->available -= total; + file->available -= (int)total; file->next += total; return total; } @@ -147,7 +147,7 @@ local size_t memRead(void *buffer, size_t size, size_t count, total = file->available; } memcpy(buffer, file->next, total); - file->available -= total; + file->available -= (int)total; file->next += total; return total; } @@ -170,7 +170,7 @@ local int memPutc(int c, MEMFILE *file) local long memTell(MEMFILE *f) { - return (f->next - f->memory) - 8; + return (long)(f->next - f->memory) - 8; } local int memError(MEMFILE *f) @@ -197,7 +197,7 @@ local int memPrintf(MEMFILE *f, const char *format, ...) len = vsprintf(buffer, format, list); va_end(list); - return memWrite(buffer, 1, len, f); + return (int)memWrite(buffer, 1, len, f); } /* =========================================================================== @@ -209,7 +209,10 @@ local int memPrintf(MEMFILE *f, const char *format, ...) can be checked to distinguish the two cases (if errno is zero, the zlib error is Z_MEM_ERROR). */ -local gzFile gz_open (char *memory, const int available, const char *mode) +local gzFile gz_open (memory, available, mode) + char *memory; + const int available; + const char *mode; { int err; int level = Z_DEFAULT_COMPRESSION; /* compression level */ @@ -309,7 +312,10 @@ local gzFile gz_open (char *memory, const int available, const char *mode) /* =========================================================================== Opens a gzip (.gz) file for reading or writing. */ -gzFile ZEXPORT memgzopen (char *memory, int available, const char *mode) +gzFile ZEXPORT memgzopen (memory, available, mode) + char *memory; + int available; + const char *mode; { return gz_open (memory, available, mode); } @@ -319,12 +325,13 @@ gzFile ZEXPORT memgzopen (char *memory, int available, const char *mode) for end of file. IN assertion: the stream s has been sucessfully opened for reading. */ -local int get_byte(mem_stream *s) +local int get_byte(s) + mem_stream *s; { if (s->z_eof) return EOF; if (s->stream.avail_in == 0) { errno = 0; - s->stream.avail_in = memRead(s->inbuf, 1, Z_BUFSIZE, s->file); + s->stream.avail_in = (uInt)memRead(s->inbuf, 1, Z_BUFSIZE, s->file); if (s->stream.avail_in == 0) { s->z_eof = 1; if (memError(s->file)) s->z_err = Z_ERRNO; @@ -345,7 +352,8 @@ local int get_byte(mem_stream *s) s->stream.avail_in is zero for the first time, but may be non-zero for concatenated .gz files. */ -local void check_header(mem_stream *s) +local void check_header(s) + mem_stream *s; { int method; /* method byte */ int flags; /* flags byte */ @@ -397,7 +405,8 @@ local void check_header(mem_stream *s) * Cleanup then free the given mem_stream. Return a zlib error code. Try freeing in the reverse order of allocations. */ -local int destroy (mem_stream *s) +local int destroy (s) + mem_stream *s; { int err = Z_OK; @@ -434,7 +443,10 @@ local int destroy (mem_stream *s) Reads the given number of uncompressed bytes from the compressed file. gzread returns the number of bytes actually read (0 for end of file). */ -int ZEXPORT memgzread (gzFile file, voidp buf, unsigned len) +int ZEXPORT memgzread (file, buf, len) + gzFile file; + voidp buf; + unsigned len; { mem_stream *s = (mem_stream*)file; Bytef *start = (Bytef*)buf; /* starting point for crc computation */ @@ -464,8 +476,7 @@ int ZEXPORT memgzread (gzFile file, voidp buf, unsigned len) s->stream.avail_in -= n; } if (s->stream.avail_out > 0) { - s->stream.avail_out -= memRead(next_out, 1, s->stream.avail_out, - s->file); + s->stream.avail_out -= (uInt)memRead(next_out, 1, s->stream.avail_out, s->file); } len -= s->stream.avail_out; s->stream.total_in += (uLong)len; @@ -476,7 +487,7 @@ int ZEXPORT memgzread (gzFile file, voidp buf, unsigned len) if (s->stream.avail_in == 0 && !s->z_eof) { errno = 0; - s->stream.avail_in = memRead(s->inbuf, 1, Z_BUFSIZE, s->file); + s->stream.avail_in = (uInt)memRead(s->inbuf, 1, Z_BUFSIZE, s->file); if (s->stream.avail_in == 0) { s->z_eof = 1; if (memError(s->file)) { @@ -526,7 +537,10 @@ int ZEXPORT memgzread (gzFile file, voidp buf, unsigned len) Writes the given number of uncompressed bytes into the compressed file. gzwrite returns the number of bytes actually written (0 in case of error). */ -int ZEXPORT memgzwrite (gzFile file, const voidp buf, unsigned len) +int ZEXPORT memgzwrite (file, buf, len) + gzFile file; + const voidp buf; + unsigned len; { mem_stream *s = (mem_stream*)file; @@ -558,7 +572,9 @@ int ZEXPORT memgzwrite (gzFile file, const voidp buf, unsigned len) Flushes all pending output into the compressed file. The parameter flush is as in the deflate() function. */ -local int do_flush (gzFile file, int flush) +local int do_flush (file, flush) + gzFile file; + int flush; { uInt len; int done = 0; @@ -598,7 +614,9 @@ local int do_flush (gzFile file, int flush) /* =========================================================================== Outputs a long in LSB order to the given file */ -local void putLong (MEMFILE *file, uLong x) +local void putLong (file, x) + MEMFILE *file; + uLong x; { int n; for (n = 0; n < 4; n++) { @@ -611,7 +629,8 @@ local void putLong (MEMFILE *file, uLong x) Reads a long in LSB order from the given mem_stream. Sets z_err in case of error. */ -local uLong getLong (mem_stream *s) +local uLong getLong (s) + mem_stream *s; { uLong x = (uLong)get_byte(s); int c; @@ -628,7 +647,8 @@ local uLong getLong (mem_stream *s) Flushes all pending output if necessary, closes the compressed file and deallocates all the (de)compression state. */ -int ZEXPORT memgzclose (gzFile file) +int ZEXPORT memgzclose (file) + gzFile file; { int err; mem_stream *s = (mem_stream*)file; @@ -649,7 +669,8 @@ int ZEXPORT memgzclose (gzFile file) return destroy((mem_stream*)file); } -long ZEXPORT memtell(gzFile file) +long ZEXPORT memtell(file) + gzFile file; { mem_stream *s = (mem_stream*)file; diff --git a/src/memgzio.h b/src/memgzio.h index 3f761411..d4b4ad77 100644 --- a/src/memgzio.h +++ b/src/memgzio.h @@ -8,7 +8,12 @@ /* memgzio.c - IO on .gz files in memory * Adapted from original gzio.c from zlib library by Forgotten */ -#include + +#if defined(HAVE_ZUTIL_H) || defined(_WIN32) +# include +#else +# include "../win32/dependencies/zlib/zutil.h" +#endif gzFile ZEXPORT memgzopen(char *memory, int, const char *); int ZEXPORT memgzread(gzFile, voidp, unsigned); diff --git a/src/remote.cpp b/src/remote.cpp index 40fdb3df..d673a553 100644 --- a/src/remote.cpp +++ b/src/remote.cpp @@ -20,7 +20,7 @@ #include #include -#ifndef WIN32 +#ifndef _WIN32 # include # include # include @@ -32,14 +32,15 @@ # else // ! HAVE_ARPA_INET_H # define socklen_t int # endif // ! HAVE_ARPA_INET_H -#else // WIN32 +# define SOCKET int +#else // _WIN32 # include # include # define socklen_t int # define close closesocket # define read _read # define write _write -#endif // WIN32 +#endif // _WIN32 #include "GBA.h" @@ -54,8 +55,8 @@ extern void debuggerSignal(int,int); int remotePort = 55555; int remoteSignal = 5; -int remoteSocket = -1; -int remoteListenSocket = -1; +SOCKET remoteSocket = -1; +SOCKET remoteListenSocket = -1; bool remoteConnected = false; bool remoteResumed = false; @@ -85,11 +86,11 @@ int remoteTcpRecv(char *data, int len) bool remoteTcpInit() { if(remoteSocket == -1) { -#ifdef WIN32 +#ifdef _WIN32 WSADATA wsaData; int error = WSAStartup(MAKEWORD(1,1),&wsaData); -#endif // WIN32 - int s = socket(PF_INET, SOCK_STREAM, 0); +#endif // _WIN32 + SOCKET s = socket(PF_INET, SOCK_STREAM, 0); remoteListenSocket = s; @@ -131,19 +132,19 @@ bool remoteTcpInit() } socklen_t len = sizeof(addr); -#ifdef WIN32 +#ifdef _WIN32 int flag = 0; ioctlsocket(s, FIONBIO, (unsigned long *)&flag); -#endif // WIN32 - int s2 = accept(s, (sockaddr *)&addr, &len); +#endif // _WIN32 + SOCKET s2 = accept(s, (sockaddr *)&addr, &len); if(s2 > 0) { fprintf(stderr, "Got a connection from %s %d\n", inet_ntoa((in_addr)addr.sin_addr), ntohs(addr.sin_port)); } else { -#ifdef WIN32 +#ifdef _WIN32 int error = WSAGetLastError(); -#endif // WIN32 +#endif // _WIN32 } char dummy; recv(s2, &dummy, 1, 0); @@ -230,14 +231,14 @@ void remotePutPacket(char *packet) char *hex = "0123456789abcdef"; char buffer[1024]; - int count = strlen(packet); + size_t count = strlen(packet); unsigned char csum = 0; char *p = buffer; *p++ = '$'; - for(int i = 0 ;i < count; i++) { + for(size_t i = 0 ;i < count; i++) { csum += packet[i]; *p++ = packet[i]; } @@ -246,7 +247,7 @@ void remotePutPacket(char *packet) *p++ = hex[csum & 15]; *p++ = 0; // printf("Sending %s\n", buffer); - remoteSendFnc(buffer, count + 4); + remoteSendFnc(buffer, (int)count + 4); char c = 0; remoteRecvFnc(&c, 1); diff --git a/src/thumb.h b/src/thumb.h index 51502583..ebf036a6 100644 --- a/src/thumb.h +++ b/src/thumb.h @@ -1,7 +1,7 @@ // -*- C++ -*- // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005-2006 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -17,6 +17,17 @@ // along with this program; if not, write to the Free Software Foundation, // Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. +// The following define updates oldreg when debugger_last is activated + +#define UPDATE_OLD_REG \ + if (debugger_last) { \ + sprintf(oldbuffer,"%08x", armState ? reg[15].I - 4 : reg[15].I - 4); \ + for (xxx=0; xxx<18; xxx++){ \ + oldreg[xxx]=reg[xxx].I; \ + } \ + } + + #ifdef C_CORE #define NEG(i) ((i) >> 31) #define POS(i) ((~(i)) >> 31) @@ -763,16 +774,29 @@ #endif #endif -u32 opcode = CPUReadHalfWordQuick(armNextPC); -clockTicks = thumbCycles[opcode >> 8] + memoryWaitFetch[(armNextPC >> 24) & 15]; + +#ifdef BKPT_SUPPORT +u8 xxx; +#endif + +u32 opcode = cpuPrefetch[0]; +cpuPrefetch[0] = cpuPrefetch[1]; + +busPrefetch = false; + if (busPrefetchCount & 0xFFFFFF00) + busPrefetchCount = 0x100 | (busPrefetchCount & 0xFF); + +clockTicks = 0; +u32 oldArmNextPC = armNextPC; #ifndef FINAL_VERSION if(armNextPC == stop) { - armNextPC = armNextPC++; + armNextPC++; } #endif armNextPC = reg[15].I; reg[15].I += 2; +THUMB_PREFETCH_NEXT; switch(opcode >> 8) { case 0x00: @@ -901,54 +925,22 @@ switch(opcode >> 8) { } break; case 0x20: - // MOV R0, #Offset8 - reg[0].I = opcode & 255; - N_FLAG = false; - Z_FLAG = (reg[0].I ? false : true); - break; case 0x21: - // MOV R1, #Offset8 - reg[1].I = opcode & 255; - N_FLAG = false; - Z_FLAG = (reg[1].I ? false : true); - break; case 0x22: - // MOV R2, #Offset8 - reg[2].I = opcode & 255; - N_FLAG = false; - Z_FLAG = (reg[2].I ? false : true); - break; case 0x23: - // MOV R3, #Offset8 - reg[3].I = opcode & 255; - N_FLAG = false; - Z_FLAG = (reg[3].I ? false : true); - break; case 0x24: - // MOV R4, #Offset8 - reg[4].I = opcode & 255; - N_FLAG = false; - Z_FLAG = (reg[4].I ? false : true); - break; case 0x25: - // MOV R5, #Offset8 - reg[5].I = opcode & 255; - N_FLAG = false; - Z_FLAG = (reg[5].I ? false : true); - break; case 0x26: - // MOV R6, #Offset8 - reg[6].I = opcode & 255; - N_FLAG = false; - Z_FLAG = (reg[6].I ? false : true); - break; case 0x27: - // MOV R7, #Offset8 - reg[7].I = opcode & 255; + { + u8 regist = (opcode >> 8) & 7; + // MOV R0~R7, #Offset8 + reg[regist].I = opcode & 255; N_FLAG = false; - Z_FLAG = (reg[7].I ? false : true); + Z_FLAG = (reg[regist].I ? false : true); + } break; - case 0x28: +case 0x28: // CMP R0, #Offset8 CMP_RN_O8(0); break; @@ -1044,6 +1036,7 @@ switch(opcode >> 8) { // SUB R7,#Offset8 SUB_RN_O8(7); break; + case 0x40: switch((opcode >> 6) & 3) { case 0x00: @@ -1093,7 +1086,7 @@ switch(opcode >> 8) { } N_FLAG = reg[dest].I & 0x80000000 ? true : false; Z_FLAG = reg[dest].I ? false : true; - clockTicks++; + clockTicks = codeTicksAccess16(armNextPC)+2; } break; case 0x03: @@ -1115,7 +1108,7 @@ switch(opcode >> 8) { } N_FLAG = reg[dest].I & 0x80000000 ? true : false; Z_FLAG = reg[dest].I ? false : true; - clockTicks++; + clockTicks = codeTicksAccess16(armNextPC)+2; } break; } @@ -1144,7 +1137,7 @@ switch(opcode >> 8) { } N_FLAG = reg[dest].I & 0x80000000 ? true : false; Z_FLAG = reg[dest].I ? false : true; - clockTicks++; + clockTicks = codeTicksAccess16(armNextPC)+2; } break; case 0x01: @@ -1181,7 +1174,7 @@ switch(opcode >> 8) { reg[dest].I = value; } } - clockTicks++; + clockTicks = codeTicksAccess16(armNextPC)+2; N_FLAG = reg[dest].I & 0x80000000 ? true : false; Z_FLAG = reg[dest].I ? false : true; } @@ -1239,19 +1232,22 @@ switch(opcode >> 8) { case 0x01: { // MUL Rd, Rs + clockTicks = 1; int dest = opcode & 7; - u32 rm = reg[(opcode >> 3) & 7].I; - reg[dest].I = reg[dest].I * rm; + u32 rm = reg[dest].I; + reg[dest].I = reg[(opcode >> 3) & 7].I * rm; if (((s32)rm) < 0) rm = ~rm; if ((rm & 0xFFFFFF00) == 0) - clockTicks += 1; + clockTicks += 0; else if ((rm & 0xFFFF0000) == 0) - clockTicks += 2; + clockTicks += 1; else if ((rm & 0xFF000000) == 0) - clockTicks += 3; + clockTicks += 2; else - clockTicks += 4; + clockTicks += 3; + busPrefetchCount = (busPrefetchCount<>(8-clockTicks)); + clockTicks += codeTicksAccess16(armNextPC) + 1; Z_FLAG = reg[dest].I ? false : true; N_FLAG = reg[dest].I & 0x80000000 ? true : false; } @@ -1294,7 +1290,9 @@ switch(opcode >> 8) { reg[15].I &= 0xFFFFFFFE; armNextPC = reg[15].I; reg[15].I += 2; - clockTicks++; + THUMB_PREFETCH; + clockTicks = codeTicksAccessSeq16(armNextPC)+1; + clockTicks += clockTicks+codeTicksAccess16(armNextPC)+1; } break; case 3: @@ -1304,7 +1302,9 @@ switch(opcode >> 8) { reg[15].I &= 0xFFFFFFFE; armNextPC = reg[15].I; reg[15].I += 2; - clockTicks++; + THUMB_PREFETCH; + clockTicks = codeTicksAccessSeq16(armNextPC)+1; + clockTicks += clockTicks+codeTicksAccess16(armNextPC)+1; } break; } @@ -1359,20 +1359,33 @@ switch(opcode >> 8) { // MOV Hd, Rs reg[dest+8].I = reg[base].I; if(dest == 7) { +#ifdef BKPT_SUPPORT + UPDATE_OLD_REG +#endif + reg[15].I &= 0xFFFFFFFE; armNextPC = reg[15].I; reg[15].I += 2; - clockTicks++; + THUMB_PREFETCH; + clockTicks = codeTicksAccessSeq16(armNextPC)+1; + clockTicks += clockTicks+codeTicksAccess16(armNextPC)+1; } break; case 3: // MOV Hd, Hs reg[dest+8].I = reg[base+8].I; if(dest == 7) { + +#ifdef BKPT_SUPPORT + UPDATE_OLD_REG +#endif + reg[15].I &= 0xFFFFFFFE; armNextPC = reg[15].I; reg[15].I += 2; - clockTicks++; + THUMB_PREFETCH; + clockTicks = codeTicksAccessSeq16(armNextPC)+1; + clockTicks += clockTicks+codeTicksAccess16(armNextPC)+1; } break; } @@ -1381,33 +1394,54 @@ switch(opcode >> 8) { case 0x47: { int base = (opcode >> 3) & 7; + busPrefetchCount=0; switch((opcode >>6) & 3) { case 0: // BX Rs +#ifdef BKPT_SUPPORT + UPDATE_OLD_REG +#endif reg[15].I = (reg[base].I) & 0xFFFFFFFE; if(reg[base].I & 1) { armState = false; armNextPC = reg[15].I; reg[15].I += 2; + THUMB_PREFETCH; + clockTicks = codeTicksAccessSeq16(armNextPC) + + codeTicksAccessSeq16(armNextPC) + codeTicksAccess16(armNextPC) + 3; } else { armState = true; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks = codeTicksAccessSeq32(armNextPC) + + codeTicksAccessSeq32(armNextPC) + codeTicksAccess32(armNextPC) + 3; } break; case 1: // BX Hs + +#ifdef BKPT_SUPPORT + UPDATE_OLD_REG +#endif + reg[15].I = (reg[8+base].I) & 0xFFFFFFFE; if(reg[8+base].I & 1) { armState = false; armNextPC = reg[15].I; reg[15].I += 2; + THUMB_PREFETCH; + clockTicks = dataTicksAccess16(armNextPC) + dataTicksAccess16(armNextPC) + + codeTicksAccess16(armNextPC) + 3; } else { armState = true; reg[15].I &= 0xFFFFFFFC; armNextPC = reg[15].I; reg[15].I += 4; + ARM_PREFETCH; + clockTicks = dataTicksAccess32(armNextPC) + dataTicksAccess32(armNextPC) + + codeTicksAccess32(armNextPC) + 3; } break; default: @@ -1416,143 +1450,119 @@ switch(opcode >> 8) { } break; case 0x48: - // LDR R0,[PC, #Imm] - { - u32 address = (reg[15].I & 0xFFFFFFFC) + ((opcode & 0xFF) << 2); - reg[0].I = CPUReadMemoryQuick(address); - clockTicks += CPUUpdateTicksAccess32(address); - } - break; case 0x49: - // LDR R1,[PC, #Imm] - { - u32 address = (reg[15].I & 0xFFFFFFFC) + ((opcode & 0xFF) << 2); - reg[1].I = CPUReadMemoryQuick(address); - clockTicks += CPUUpdateTicksAccess32(address); - } - break; case 0x4a: - // LDR R2,[PC, #Imm] - { - u32 address = (reg[15].I & 0xFFFFFFFC) + ((opcode & 0xFF) << 2); - reg[2].I = CPUReadMemoryQuick(address); - clockTicks += CPUUpdateTicksAccess32(address); - } - break; case 0x4b: - // LDR R3,[PC, #Imm] - { - u32 address = (reg[15].I & 0xFFFFFFFC) + ((opcode & 0xFF) << 2); - reg[3].I = CPUReadMemoryQuick(address); - clockTicks += CPUUpdateTicksAccess32(address); - } - break; case 0x4c: - // LDR R4,[PC, #Imm] - { - u32 address = (reg[15].I & 0xFFFFFFFC) + ((opcode & 0xFF) << 2); - reg[4].I = CPUReadMemoryQuick(address); - clockTicks += CPUUpdateTicksAccess32(address); - } - break; case 0x4d: - // LDR R5,[PC, #Imm] - { - u32 address = (reg[15].I & 0xFFFFFFFC) + ((opcode & 0xFF) << 2); - reg[5].I = CPUReadMemoryQuick(address); - clockTicks += CPUUpdateTicksAccess32(address); - } - break; case 0x4e: - // LDR R6,[PC, #Imm] - { - u32 address = (reg[15].I & 0xFFFFFFFC) + ((opcode & 0xFF) << 2); - reg[6].I = CPUReadMemoryQuick(address); - clockTicks += CPUUpdateTicksAccess32(address); - } - break; case 0x4f: - // LDR R7,[PC, #Imm] + // LDR R0~R7,[PC, #Imm] { - u32 address = (reg[15].I & 0xFFFFFFFC) + ((opcode & 0xFF) << 2); - reg[7].I = CPUReadMemoryQuick(address); - clockTicks += CPUUpdateTicksAccess32(address); + u8 regist = (opcode >> 8) & 7; + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; + u32 address = (reg[15].I & 0xFFFFFFFC) + ((opcode & 0xFF) << 2); + reg[regist].I = CPUReadMemoryQuick(address); + busPrefetchCount=0; + clockTicks = 3 + dataTicksAccess32(address) + + codeTicksAccess16(armNextPC); } break; case 0x50: case 0x51: // STR Rd, [Rs, Rn] { - u32 - address = reg[(opcode>>3)&7].I + reg[(opcode>>6)&7].I; + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; + u32 address = reg[(opcode>>3)&7].I + reg[(opcode>>6)&7].I; CPUWriteMemory(address, reg[opcode & 7].I); - clockTicks += CPUUpdateTicksAccess32(address); + clockTicks = dataTicksAccess32(address) + codeTicksAccess16(armNextPC) + 2; } break; case 0x52: case 0x53: // STRH Rd, [Rs, Rn] { + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; u32 address = reg[(opcode>>3)&7].I + reg[(opcode>>6)&7].I; CPUWriteHalfWord(address, reg[opcode&7].W.W0); - clockTicks += CPUUpdateTicksAccess16(address); + clockTicks = dataTicksAccess16(address) + codeTicksAccess16(armNextPC) + 2; } break; case 0x54: case 0x55: // STRB Rd, [Rs, Rn] { + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; u32 address = reg[(opcode>>3)&7].I + reg[(opcode >>6)&7].I; CPUWriteByte(address, reg[opcode & 7].B.B0); - clockTicks += CPUUpdateTicksAccess16(address); + clockTicks = dataTicksAccess16(address) + codeTicksAccess16(armNextPC) + 2; } break; case 0x56: case 0x57: // LDSB Rd, [Rs, Rn] { + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; u32 address = reg[(opcode>>3)&7].I + reg[(opcode>>6)&7].I; reg[opcode&7].I = (s8)CPUReadByte(address); - clockTicks += CPUUpdateTicksAccess16(address); + clockTicks = 3 + dataTicksAccess16(address) + + codeTicksAccess16(armNextPC); } break; case 0x58: case 0x59: // LDR Rd, [Rs, Rn] { + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; u32 address = reg[(opcode>>3)&7].I + reg[(opcode>>6)&7].I; reg[opcode&7].I = CPUReadMemory(address); - clockTicks += CPUUpdateTicksAccess32(address); + clockTicks = 3 + dataTicksAccess32(address) + + codeTicksAccess16(armNextPC); } break; case 0x5a: case 0x5b: // LDRH Rd, [Rs, Rn] { + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; u32 address = reg[(opcode>>3)&7].I + reg[(opcode>>6)&7].I; reg[opcode&7].I = CPUReadHalfWord(address); - clockTicks += CPUUpdateTicksAccess16(address); + clockTicks = 3 + dataTicksAccess32(address) + + codeTicksAccess16(armNextPC); } break; case 0x5c: case 0x5d: // LDRB Rd, [Rs, Rn] { + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; u32 address = reg[(opcode>>3)&7].I + reg[(opcode>>6)&7].I; reg[opcode&7].I = CPUReadByte(address); - clockTicks += CPUUpdateTicksAccess16(address); + clockTicks = 3 + dataTicksAccess16(address) + + codeTicksAccess16(armNextPC); } break; case 0x5e: case 0x5f: // LDSH Rd, [Rs, Rn] { + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; u32 address = reg[(opcode>>3)&7].I + reg[(opcode>>6)&7].I; reg[opcode&7].I = (s16)CPUReadHalfWordSigned(address); - clockTicks += CPUUpdateTicksAccess16(address); + clockTicks = 3 + dataTicksAccess16(address) + + codeTicksAccess16(armNextPC); } break; case 0x60: @@ -1565,10 +1575,12 @@ switch(opcode >> 8) { case 0x67: // STR Rd, [Rs, #Imm] { + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; u32 address = reg[(opcode>>3)&7].I + (((opcode>>6)&31)<<2); CPUWriteMemory(address, reg[opcode&7].I); - clockTicks += CPUUpdateTicksAccess32(address); + clockTicks = dataTicksAccess32(address) + codeTicksAccess16(armNextPC) + 2; } break; case 0x68: @@ -1581,9 +1593,12 @@ switch(opcode >> 8) { case 0x6f: // LDR Rd, [Rs, #Imm] { + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; u32 address = reg[(opcode>>3)&7].I + (((opcode>>6)&31)<<2); reg[opcode&7].I = CPUReadMemory(address); - clockTicks += CPUUpdateTicksAccess32(address); + clockTicks = 3 + dataTicksAccess32(address) + + codeTicksAccess16(armNextPC); } break; case 0x70: @@ -1596,10 +1611,12 @@ switch(opcode >> 8) { case 0x77: // STRB Rd, [Rs, #Imm] { + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; u32 address = reg[(opcode>>3)&7].I + (((opcode>>6)&31)); CPUWriteByte(address, reg[opcode&7].B.B0); - clockTicks += CPUUpdateTicksAccess16(address); + clockTicks = dataTicksAccess16(address) + codeTicksAccess16(armNextPC) + 2; } break; case 0x78: @@ -1612,9 +1629,12 @@ switch(opcode >> 8) { case 0x7f: // LDRB Rd, [Rs, #Imm] { + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; u32 address = reg[(opcode>>3)&7].I + (((opcode>>6)&31)); reg[opcode&7].I = CPUReadByte(address); - clockTicks += CPUUpdateTicksAccess16(address); + clockTicks = 3 + dataTicksAccess16(address) + + codeTicksAccess16(armNextPC); } break; case 0x80: @@ -1627,10 +1647,12 @@ switch(opcode >> 8) { case 0x87: // STRH Rd, [Rs, #Imm] { + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; u32 address = reg[(opcode>>3)&7].I + (((opcode>>6)&31)<<1); CPUWriteHalfWord(address, reg[opcode&7].W.W0); - clockTicks += CPUUpdateTicksAccess16(address); + clockTicks = dataTicksAccess16(address) + codeTicksAccess16(armNextPC) + 2; } break; case 0x88: @@ -1643,203 +1665,79 @@ switch(opcode >> 8) { case 0x8f: // LDRH Rd, [Rs, #Imm] { + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; u32 address = reg[(opcode>>3)&7].I + (((opcode>>6)&31)<<1); reg[opcode&7].I = CPUReadHalfWord(address); - clockTicks += CPUUpdateTicksAccess16(address); + clockTicks = 3 + dataTicksAccess16(address) + + codeTicksAccess16(armNextPC); } break; case 0x90: - // STR R0, [SP, #Imm] - { - u32 address = reg[13].I + ((opcode&255)<<2); - CPUWriteMemory(address, reg[0].I); - clockTicks += CPUUpdateTicksAccess32(address); - } - break; case 0x91: - // STR R1, [SP, #Imm] - { - u32 address = reg[13].I + ((opcode&255)<<2); - CPUWriteMemory(address, reg[1].I); - clockTicks += CPUUpdateTicksAccess32(address); - } - break; case 0x92: - // STR R2, [SP, #Imm] - { - u32 address = reg[13].I + ((opcode&255)<<2); - CPUWriteMemory(address, reg[2].I); - clockTicks += CPUUpdateTicksAccess32(address); - } - break; case 0x93: - // STR R3, [SP, #Imm] - { - u32 address = reg[13].I + ((opcode&255)<<2); - CPUWriteMemory(address, reg[3].I); - clockTicks += CPUUpdateTicksAccess32(address); - } - break; case 0x94: - // STR R4, [SP, #Imm] - { - u32 address = reg[13].I + ((opcode&255)<<2); - CPUWriteMemory(address, reg[4].I); - clockTicks += CPUUpdateTicksAccess32(address); - } - break; case 0x95: - // STR R5, [SP, #Imm] - { - u32 address = reg[13].I + ((opcode&255)<<2); - CPUWriteMemory(address, reg[5].I); - clockTicks += CPUUpdateTicksAccess32(address); - } - break; case 0x96: - // STR R6, [SP, #Imm] + case 0x97: + // STR R0~R7, [SP, #Imm] { - u32 address = reg[13].I + ((opcode&255)<<2); - CPUWriteMemory(address, reg[6].I); - clockTicks += CPUUpdateTicksAccess32(address); + u8 regist = (opcode >> 8) & 7; + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; + u32 address = reg[13].I + ((opcode&255)<<2); + CPUWriteMemory(address, reg[regist].I); + clockTicks = dataTicksAccess32(address) + codeTicksAccess16(armNextPC) + 2; } break; - case 0x97: - // STR R7, [SP, #Imm] - { - u32 address = reg[13].I + ((opcode&255)<<2); - CPUWriteMemory(address, reg[7].I); - clockTicks += CPUUpdateTicksAccess32(address); - } - break; case 0x98: - // LDR R0, [SP, #Imm] - { - u32 address = reg[13].I + ((opcode&255)<<2); - reg[0].I = CPUReadMemoryQuick(address); - clockTicks += CPUUpdateTicksAccess32(address); - } - break; case 0x99: - // LDR R1, [SP, #Imm] - { - u32 address = reg[13].I + ((opcode&255)<<2); - reg[1].I = CPUReadMemoryQuick(address); - clockTicks += CPUUpdateTicksAccess32(address); - } - break; case 0x9a: - // LDR R2, [SP, #Imm] - { - u32 address = reg[13].I + ((opcode&255)<<2); - reg[2].I = CPUReadMemoryQuick(address); - clockTicks += CPUUpdateTicksAccess32(address); - } - break; case 0x9b: - // LDR R3, [SP, #Imm] - { - u32 address = reg[13].I + ((opcode&255)<<2); - reg[3].I = CPUReadMemoryQuick(address); - clockTicks += CPUUpdateTicksAccess32(address); - } - break; case 0x9c: - // LDR R4, [SP, #Imm] - { - u32 address = reg[13].I + ((opcode&255)<<2); - reg[4].I = CPUReadMemoryQuick(address); - clockTicks += CPUUpdateTicksAccess32(address); - } - break; case 0x9d: - // LDR R5, [SP, #Imm] - { - u32 address = reg[13].I + ((opcode&255)<<2); - reg[5].I = CPUReadMemoryQuick(address); - clockTicks += CPUUpdateTicksAccess32(address); - } - break; case 0x9e: - // LDR R6, [SP, #Imm] - { - u32 address = reg[13].I + ((opcode&255)<<2); - reg[6].I = CPUReadMemoryQuick(address); - clockTicks += CPUUpdateTicksAccess32(address); - } - break; case 0x9f: - // LDR R7, [SP, #Imm] + // LDR R0~R7, [SP, #Imm] { + u8 regist = (opcode >> 8) & 7; + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; u32 address = reg[13].I + ((opcode&255)<<2); - reg[7].I = CPUReadMemoryQuick(address); - clockTicks += CPUUpdateTicksAccess32(address); + reg[regist].I = CPUReadMemoryQuick(address); + clockTicks = 3 + dataTicksAccess32(address) + + codeTicksAccess16(armNextPC); } break; case 0xa0: - // ADD R0, PC, Imm - reg[0].I = (reg[15].I & 0xFFFFFFFC) + ((opcode&255)<<2); - break; case 0xa1: - // ADD R1, PC, Imm - reg[1].I = (reg[15].I & 0xFFFFFFFC) + ((opcode&255)<<2); - break; case 0xa2: - // ADD R2, PC, Imm - reg[2].I = (reg[15].I & 0xFFFFFFFC) + ((opcode&255)<<2); - break; case 0xa3: - // ADD R3, PC, Imm - reg[3].I = (reg[15].I & 0xFFFFFFFC) + ((opcode&255)<<2); - break; case 0xa4: - // ADD R4, PC, Imm - reg[4].I = (reg[15].I & 0xFFFFFFFC) + ((opcode&255)<<2); - break; case 0xa5: - // ADD R5, PC, Imm - reg[5].I = (reg[15].I & 0xFFFFFFFC) + ((opcode&255)<<2); - break; case 0xa6: - // ADD R6, PC, Imm - reg[6].I = (reg[15].I & 0xFFFFFFFC) + ((opcode&255)<<2); - break; case 0xa7: - // ADD R7, PC, Imm - reg[7].I = (reg[15].I & 0xFFFFFFFC) + ((opcode&255)<<2); + { + // ADD R0~R7, PC, Imm + u8 regist = (opcode >> 8) & 7; + reg[regist].I = (reg[15].I & 0xFFFFFFFC) + ((opcode&255)<<2); + } break; case 0xa8: - // ADD R0, SP, Imm - reg[0].I = reg[13].I + ((opcode&255)<<2); - break; case 0xa9: - // ADD R1, SP, Imm - reg[1].I = reg[13].I + ((opcode&255)<<2); - break; case 0xaa: - // ADD R2, SP, Imm - reg[2].I = reg[13].I + ((opcode&255)<<2); - break; case 0xab: - // ADD R3, SP, Imm - reg[3].I = reg[13].I + ((opcode&255)<<2); - break; case 0xac: - // ADD R4, SP, Imm - reg[4].I = reg[13].I + ((opcode&255)<<2); - break; case 0xad: - // ADD R5, SP, Imm - reg[5].I = reg[13].I + ((opcode&255)<<2); - break; case 0xae: - // ADD R6, SP, Imm - reg[6].I = reg[13].I + ((opcode&255)<<2); - break; case 0xaf: - // ADD R7, SP, Imm - reg[7].I = reg[13].I + ((opcode&255)<<2); - break; + { + // ADD R0~R7, SP, Imm + u8 regist = (opcode >> 8) & 7; + reg[regist].I = reg[13].I + ((opcode&255)<<2); + } + break; case 0xb0: { // ADD SP, Imm @@ -1853,15 +1751,17 @@ switch(opcode >> 8) { if(opcode & (val)) {\ CPUWriteMemory(address, reg[(r)].I);\ if(offset)\ - clockTicks += 1 + CPUUpdateTicksAccessSeq32(address);\ + clockTicks += 1 + dataTicksAccessSeq32(address);\ else\ - clockTicks += 1 + CPUUpdateTicksAccess32(address);\ + clockTicks += 1 + dataTicksAccess32(address);\ offset = 1;\ address += 4;\ } case 0xb4: // PUSH {Rlist} { + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = 0; u32 temp = reg[13].I - 4 * cpuBitsSet[opcode & 0xff]; u32 address = temp & 0xFFFFFFFC; @@ -1873,12 +1773,15 @@ switch(opcode >> 8) { PUSH_REG(32, 5); PUSH_REG(64, 6); PUSH_REG(128, 7); + clockTicks += codeTicksAccess16(armNextPC)+1; reg[13].I = temp; } break; case 0xb5: // PUSH {Rlist, LR} { + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = 0; u32 temp = reg[13].I - 4 - 4 * cpuBitsSet[opcode & 0xff]; u32 address = temp & 0xFFFFFFFC; @@ -1891,6 +1794,7 @@ switch(opcode >> 8) { PUSH_REG(64, 6); PUSH_REG(128, 7); PUSH_REG(256, 14); + clockTicks += codeTicksAccess16(armNextPC)+1; reg[13].I = temp; } break; @@ -1898,18 +1802,21 @@ switch(opcode >> 8) { if(opcode & (val)) {\ reg[(r)].I = CPUReadMemory(address);\ if(offset)\ - clockTicks += 2 + CPUUpdateTicksAccessSeq32(address);\ + clockTicks += 1 + dataTicksAccessSeq32(address);\ else\ - clockTicks += 2 + CPUUpdateTicksAccess32(address);\ + clockTicks += 1 + dataTicksAccess32(address);\ offset = 1;\ address += 4;\ } case 0xbc: // POP {Rlist} { + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = 0; u32 address = reg[13].I & 0xFFFFFFFC; u32 temp = reg[13].I + 4*cpuBitsSet[opcode & 0xFF]; + clockTicks = 0; POP_REG(1, 0); POP_REG(2, 1); POP_REG(4, 2); @@ -1919,14 +1826,18 @@ switch(opcode >> 8) { POP_REG(64, 6); POP_REG(128, 7); reg[13].I = temp; + clockTicks += codeTicksAccess16(armNextPC)+2; } break; case 0xbd: // POP {Rlist, PC} { + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; int offset = 0; u32 address = reg[13].I & 0xFFFFFFFC; u32 temp = reg[13].I + 4 + 4*cpuBitsSet[opcode & 0xFF]; + clockTicks = 0; POP_REG(1, 0); POP_REG(2, 1); POP_REG(4, 2); @@ -1937,12 +1848,15 @@ switch(opcode >> 8) { POP_REG(128, 7); reg[15].I = (CPUReadMemory(address) & 0xFFFFFFFE); if(offset) - clockTicks += CPUUpdateTicksAccessSeq32(address); + clockTicks += 1 + dataTicksAccessSeq32(address); else - clockTicks += CPUUpdateTicksAccess32(address); + clockTicks += 1 + dataTicksAccess32(address); armNextPC = reg[15].I; reg[15].I += 2; reg[13].I = temp; + THUMB_PREFETCH; + busPrefetchCount=0; + clockTicks += codeTicksAccess16(armNextPC) + codeTicksAccess16(armNextPC) + 3; } break; #define THUMB_STM_REG(val,r,b) \ @@ -1950,297 +1864,67 @@ switch(opcode >> 8) { CPUWriteMemory(address, reg[(r)].I);\ if(!offset) {\ reg[(b)].I = temp;\ - clockTicks += 1 + CPUUpdateTicksAccess32(address);\ + clockTicks += 1 + dataTicksAccess32(address);\ } else \ - clockTicks += 1 + CPUUpdateTicksAccessSeq32(address);\ + clockTicks += 1 + dataTicksAccessSeq32(address);\ offset = 1;\ address += 4;\ } case 0xc0: - { - // STM R0!, {Rlist} - u32 address = reg[0].I & 0xFFFFFFFC; - u32 temp = reg[0].I + 4*cpuBitsSet[opcode & 0xff]; - int offset = 0; - // store - THUMB_STM_REG(1, 0, 0); - THUMB_STM_REG(2, 1, 0); - THUMB_STM_REG(4, 2, 0); - THUMB_STM_REG(8, 3, 0); - THUMB_STM_REG(16, 4, 0); - THUMB_STM_REG(32, 5, 0); - THUMB_STM_REG(64, 6, 0); - THUMB_STM_REG(128, 7, 0); - } - break; case 0xc1: - { - // STM R1!, {Rlist} - u32 address = reg[1].I & 0xFFFFFFFC; - u32 temp = reg[1].I + 4*cpuBitsSet[opcode & 0xff]; - int offset = 0; - // store - THUMB_STM_REG(1, 0, 1); - THUMB_STM_REG(2, 1, 1); - THUMB_STM_REG(4, 2, 1); - THUMB_STM_REG(8, 3, 1); - THUMB_STM_REG(16, 4, 1); - THUMB_STM_REG(32, 5, 1); - THUMB_STM_REG(64, 6, 1); - THUMB_STM_REG(128, 7, 1); - } - break; case 0xc2: - { - // STM R2!, {Rlist} - u32 address = reg[2].I & 0xFFFFFFFC; - u32 temp = reg[2].I + 4*cpuBitsSet[opcode & 0xff]; - int offset = 0; - // store - THUMB_STM_REG(1, 0, 2); - THUMB_STM_REG(2, 1, 2); - THUMB_STM_REG(4, 2, 2); - THUMB_STM_REG(8, 3, 2); - THUMB_STM_REG(16, 4, 2); - THUMB_STM_REG(32, 5, 2); - THUMB_STM_REG(64, 6, 2); - THUMB_STM_REG(128, 7, 2); - } - break; case 0xc3: - { - // STM R3!, {Rlist} - u32 address = reg[3].I & 0xFFFFFFFC; - u32 temp = reg[3].I + 4*cpuBitsSet[opcode & 0xff]; - int offset = 0; - // store - THUMB_STM_REG(1, 0, 3); - THUMB_STM_REG(2, 1, 3); - THUMB_STM_REG(4, 2, 3); - THUMB_STM_REG(8, 3, 3); - THUMB_STM_REG(16, 4, 3); - THUMB_STM_REG(32, 5, 3); - THUMB_STM_REG(64, 6, 3); - THUMB_STM_REG(128, 7, 3); - } - break; case 0xc4: - { - // STM R4!, {Rlist} - u32 address = reg[4].I & 0xFFFFFFFC; - u32 temp = reg[4].I + 4*cpuBitsSet[opcode & 0xff]; - int offset = 0; - // store - THUMB_STM_REG(1, 0, 4); - THUMB_STM_REG(2, 1, 4); - THUMB_STM_REG(4, 2, 4); - THUMB_STM_REG(8, 3, 4); - THUMB_STM_REG(16, 4, 4); - THUMB_STM_REG(32, 5, 4); - THUMB_STM_REG(64, 6, 4); - THUMB_STM_REG(128, 7, 4); - } - break; case 0xc5: - { - // STM R5!, {Rlist} - u32 address = reg[5].I & 0xFFFFFFFC; - u32 temp = reg[5].I + 4*cpuBitsSet[opcode & 0xff]; - int offset = 0; - // store - THUMB_STM_REG(1, 0, 5); - THUMB_STM_REG(2, 1, 5); - THUMB_STM_REG(4, 2, 5); - THUMB_STM_REG(8, 3, 5); - THUMB_STM_REG(16, 4, 5); - THUMB_STM_REG(32, 5, 5); - THUMB_STM_REG(64, 6, 5); - THUMB_STM_REG(128, 7, 5); - } - break; case 0xc6: - { - // STM R6!, {Rlist} - u32 address = reg[6].I & 0xFFFFFFFC; - u32 temp = reg[6].I + 4*cpuBitsSet[opcode & 0xff]; - int offset = 0; - // store - THUMB_STM_REG(1, 0, 6); - THUMB_STM_REG(2, 1, 6); - THUMB_STM_REG(4, 2, 6); - THUMB_STM_REG(8, 3, 6); - THUMB_STM_REG(16, 4, 6); - THUMB_STM_REG(32, 5, 6); - THUMB_STM_REG(64, 6, 6); - THUMB_STM_REG(128, 7, 6); - } - break; case 0xc7: { - // STM R7!, {Rlist} - u32 address = reg[7].I & 0xFFFFFFFC; - u32 temp = reg[7].I + 4*cpuBitsSet[opcode & 0xff]; + // STM R0~7!, {Rlist} + u8 regist = (opcode >> 8) & 7; + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; + u32 address = reg[regist].I & 0xFFFFFFFC; + u32 temp = reg[regist].I + 4*cpuBitsSet[opcode & 0xff]; int offset = 0; // store - THUMB_STM_REG(1, 0, 7); - THUMB_STM_REG(2, 1, 7); - THUMB_STM_REG(4, 2, 7); - THUMB_STM_REG(8, 3, 7); - THUMB_STM_REG(16, 4, 7); - THUMB_STM_REG(32, 5, 7); - THUMB_STM_REG(64, 6, 7); - THUMB_STM_REG(128, 7, 7); + THUMB_STM_REG(1, 0, regist); + THUMB_STM_REG(2, 1, regist); + THUMB_STM_REG(4, 2, regist); + THUMB_STM_REG(8, 3, regist); + THUMB_STM_REG(16, 4, regist); + THUMB_STM_REG(32, 5, regist); + THUMB_STM_REG(64, 6, regist); + THUMB_STM_REG(128, 7, regist); + clockTicks = codeTicksAccess16(armNextPC)+1; } - break; + break; #define THUMB_LDM_REG(val,r) \ if(opcode & (val)) {\ reg[(r)].I = CPUReadMemory(address);\ if(offset)\ - clockTicks += 2 + CPUUpdateTicksAccessSeq32(address);\ - else\ - clockTicks += 2 + CPUUpdateTicksAccess32(address);\ + clockTicks += 1 + dataTicksAccessSeq32(address);\ + else \ + clockTicks += 1 + dataTicksAccess32(address);\ offset = 1;\ address += 4;\ } case 0xc8: - { - // LDM R0!, {Rlist} - u32 address = reg[0].I & 0xFFFFFFFC; - u32 temp = reg[0].I + 4*cpuBitsSet[opcode & 0xFF]; - int offset = 0; - // load - THUMB_LDM_REG(1, 0); - THUMB_LDM_REG(2, 1); - THUMB_LDM_REG(4, 2); - THUMB_LDM_REG(8, 3); - THUMB_LDM_REG(16, 4); - THUMB_LDM_REG(32, 5); - THUMB_LDM_REG(64, 6); - THUMB_LDM_REG(128, 7); - if(!(opcode & 1)) - reg[0].I = temp; - } - break; case 0xc9: - { - // LDM R1!, {Rlist} - u32 address = reg[1].I & 0xFFFFFFFC; - u32 temp = reg[1].I + 4*cpuBitsSet[opcode & 0xFF]; - int offset = 0; - // load - THUMB_LDM_REG(1, 0); - THUMB_LDM_REG(2, 1); - THUMB_LDM_REG(4, 2); - THUMB_LDM_REG(8, 3); - THUMB_LDM_REG(16, 4); - THUMB_LDM_REG(32, 5); - THUMB_LDM_REG(64, 6); - THUMB_LDM_REG(128, 7); - if(!(opcode & 2)) - reg[1].I = temp; - } - break; case 0xca: - { - // LDM R2!, {Rlist} - u32 address = reg[2].I & 0xFFFFFFFC; - u32 temp = reg[2].I + 4*cpuBitsSet[opcode & 0xFF]; - int offset = 0; - // load - THUMB_LDM_REG(1, 0); - THUMB_LDM_REG(2, 1); - THUMB_LDM_REG(4, 2); - THUMB_LDM_REG(8, 3); - THUMB_LDM_REG(16, 4); - THUMB_LDM_REG(32, 5); - THUMB_LDM_REG(64, 6); - THUMB_LDM_REG(128, 7); - if(!(opcode & 4)) - reg[2].I = temp; - } - break; case 0xcb: - { - // LDM R3!, {Rlist} - u32 address = reg[3].I & 0xFFFFFFFC; - u32 temp = reg[3].I + 4*cpuBitsSet[opcode & 0xFF]; - int offset = 0; - // load - THUMB_LDM_REG(1, 0); - THUMB_LDM_REG(2, 1); - THUMB_LDM_REG(4, 2); - THUMB_LDM_REG(8, 3); - THUMB_LDM_REG(16, 4); - THUMB_LDM_REG(32, 5); - THUMB_LDM_REG(64, 6); - THUMB_LDM_REG(128, 7); - if(!(opcode & 8)) - reg[3].I = temp; - } - break; case 0xcc: - { - // LDM R4!, {Rlist} - u32 address = reg[4].I & 0xFFFFFFFC; - u32 temp = reg[4].I + 4*cpuBitsSet[opcode & 0xFF]; - int offset = 0; - // load - THUMB_LDM_REG(1, 0); - THUMB_LDM_REG(2, 1); - THUMB_LDM_REG(4, 2); - THUMB_LDM_REG(8, 3); - THUMB_LDM_REG(16, 4); - THUMB_LDM_REG(32, 5); - THUMB_LDM_REG(64, 6); - THUMB_LDM_REG(128, 7); - if(!(opcode & 16)) - reg[4].I = temp; - } - break; case 0xcd: - { - // LDM R5!, {Rlist} - u32 address = reg[5].I & 0xFFFFFFFC; - u32 temp = reg[5].I + 4*cpuBitsSet[opcode & 0xFF]; - int offset = 0; - // load - THUMB_LDM_REG(1, 0); - THUMB_LDM_REG(2, 1); - THUMB_LDM_REG(4, 2); - THUMB_LDM_REG(8, 3); - THUMB_LDM_REG(16, 4); - THUMB_LDM_REG(32, 5); - THUMB_LDM_REG(64, 6); - THUMB_LDM_REG(128, 7); - if(!(opcode & 32)) - reg[5].I = temp; - } - break; case 0xce: - { - // LDM R6!, {Rlist} - u32 address = reg[6].I & 0xFFFFFFFC; - u32 temp = reg[6].I + 4*cpuBitsSet[opcode & 0xFF]; - int offset = 0; - // load - THUMB_LDM_REG(1, 0); - THUMB_LDM_REG(2, 1); - THUMB_LDM_REG(4, 2); - THUMB_LDM_REG(8, 3); - THUMB_LDM_REG(16, 4); - THUMB_LDM_REG(32, 5); - THUMB_LDM_REG(64, 6); - THUMB_LDM_REG(128, 7); - if(!(opcode & 64)) - reg[6].I = temp; - } - break; case 0xcf: { - // LDM R7!, {Rlist} - u32 address = reg[7].I & 0xFFFFFFFC; - u32 temp = reg[7].I + 4*cpuBitsSet[opcode & 0xFF]; + // LDM R0~R7!, {Rlist} + u8 regist = (opcode >> 8) & 7; + if (!busPrefetchCount) + busPrefetch = busPrefetchEnable; + u32 address = reg[regist].I & 0xFFFFFFFC; + u32 temp = reg[regist].I + 4*cpuBitsSet[opcode & 0xFF]; int offset = 0; + clockTicks = 0; // load THUMB_LDM_REG(1, 0); THUMB_LDM_REG(2, 1); @@ -2249,141 +1933,232 @@ switch(opcode >> 8) { THUMB_LDM_REG(16, 4); THUMB_LDM_REG(32, 5); THUMB_LDM_REG(64, 6); - THUMB_LDM_REG(128, 7); - if(!(opcode & 128)) - reg[7].I = temp; + THUMB_LDM_REG(128, 7); + clockTicks += codeTicksAccess16(armNextPC)+2; + if(!(opcode & (1<> 8) { reg[15].I += offset; armNextPC = reg[15].I; reg[15].I += 2; + THUMB_PREFETCH; + clockTicks = codeTicksAccessSeq16(armNextPC) + codeTicksAccessSeq16(armNextPC) + + codeTicksAccess16(armNextPC) + 3; + busPrefetchCount=0; } break; case 0xf0: @@ -2410,6 +2189,7 @@ switch(opcode >> 8) { // BLL #offset int offset = (opcode & 0x7FF); reg[14].I = reg[15].I + (offset << 12); + clockTicks = codeTicksAccessSeq16(armNextPC) + 1; } break; case 0xf4: @@ -2420,6 +2200,7 @@ switch(opcode >> 8) { // BLL #offset int offset = (opcode & 0x7FF); reg[14].I = reg[15].I + ((offset << 12) | 0xFF800000); + clockTicks = codeTicksAccessSeq16(armNextPC) + 1; } break; case 0xf8: @@ -2438,6 +2219,10 @@ switch(opcode >> 8) { armNextPC = reg[15].I; reg[15].I += 2; reg[14].I = temp|1; + THUMB_PREFETCH; + clockTicks = codeTicksAccessSeq16(armNextPC) + + codeTicksAccess16(armNextPC) + codeTicksAccessSeq16(armNextPC) + 3; + busPrefetchCount = 0; } break; #ifdef BKPT_SUPPORT @@ -2472,3 +2257,6 @@ switch(opcode >> 8) { CPUUndefinedException(); break; } + +if (clockTicks==0) +clockTicks = codeTicksAccessSeq16(oldArmNextPC) + 1; diff --git a/src/unzip.cpp b/src/unzip.cpp index 98c50e94..876691a5 100644 --- a/src/unzip.cpp +++ b/src/unzip.cpp @@ -154,7 +154,7 @@ typedef struct local int unzlocal_getByte(FILE *fin,int *pi) { unsigned char c; - int err = fread(&c, 1, 1, fin); + size_t err = fread(&c, 1, 1, fin); if (err==1) { *pi = (int)c; diff --git a/src/win32/AboutDialog.cpp b/src/win32/AboutDialog.cpp index 0d634e2b..f558fd14 100644 --- a/src/win32/AboutDialog.cpp +++ b/src/win32/AboutDialog.cpp @@ -19,8 +19,9 @@ // AboutDialog.cpp : implementation file // +#include "stdafx.h" #include "AboutDialog.h" -#include "..\..\res\resource.h" +#include "../AutoBuild.h" #ifdef _DEBUG #define new DEBUG_NEW @@ -63,15 +64,13 @@ BOOL AboutDialog::OnInitDialog() { CDialog::OnInitDialog(); + CWnd *p = GetDlgItem(IDC_TRANSLATOR_URL); + if(p) { + m_translator.SubclassDlgItem(IDC_TRANSLATOR_URL, this); + } + m_link.SetWindowText("http://vba.ngemu.com"); return TRUE; // return TRUE unless you set the focus to a control // EXCEPTION: OCX Property Pages should return FALSE } - -void AboutDialog::OnOK() -{ - // TODO: Add extra validation here - - CDialog::OnOK(); -} diff --git a/src/win32/AboutDialog.h b/src/win32/AboutDialog.h index fdb48d3d..aacc74d8 100644 --- a/src/win32/AboutDialog.h +++ b/src/win32/AboutDialog.h @@ -27,7 +27,7 @@ // #include "stdafx.h" #include "Hyperlink.h" -#include "..\..\res\resource.h" +#include "resource.h" ///////////////////////////////////////////////////////////////////////////// // AboutDialog dialog @@ -60,7 +60,6 @@ class AboutDialog : public CDialog // Generated message map functions //{{AFX_MSG(AboutDialog) virtual BOOL OnInitDialog(); - virtual void OnOK(); //}}AFX_MSG diff --git a/src/win32/AccelEditor.cpp b/src/win32/AccelEditor.cpp index 0747e26e..b632914d 100644 --- a/src/win32/AccelEditor.cpp +++ b/src/win32/AccelEditor.cpp @@ -156,7 +156,7 @@ void AccelEditor::OnSelchangeCommands() pAccel->GetString(szBuffer); index = m_currents.AddString(szBuffer); // and a pointer to the accel object. - m_currents.SetItemData(index, (DWORD)pAccel); + m_currents.SetItemData(index, (DWORD_PTR)pAccel); } } // Init the key editor @@ -180,12 +180,12 @@ void AccelEditor::OnAssign() WORD wKey; bool bCtrl, bAlt, bShift; - int index; - + if (!m_key.GetAccelKey(wKey, bCtrl, bAlt, bShift)) return; // no valid key, abort int count = m_commands.GetCount(); + int index; for (index = 0; index < count; index++) { wIDCommand = LOWORD(m_commands.GetItemData(index)); @@ -235,7 +235,7 @@ void AccelEditor::OnAssign() pAccel->GetString(szBuffer); index = m_currents.AddString(szBuffer); - m_currents.SetItemData(index, (DWORD)pAccel); + m_currents.SetItemData(index, (DWORD_PTR)pAccel); // Reset the key editor. m_key.ResetKey(); diff --git a/src/win32/AcceleratorManager.cpp b/src/win32/AcceleratorManager.cpp index ad4d4cfb..ca3f8fc6 100644 --- a/src/win32/AcceleratorManager.cpp +++ b/src/win32/AcceleratorManager.cpp @@ -27,7 +27,7 @@ //////////////////////////////////////////////////////////////////////////////// #include "stdafx.h" -//#include "..\..\res\resource.h" +//#include "resource.h" #include "../System.h" #include "AcceleratorManager.h" @@ -247,7 +247,7 @@ bool CAcceleratorManager::UpdateWndTable() } } - int nAccel = arrayACCEL.GetSize(); + INT_PTR nAccel = arrayACCEL.GetSize(); LPACCEL lpAccel = (LPACCEL)LocalAlloc(LPTR, nAccel * sizeof(ACCEL)); if (!lpAccel) { for (iLoop = 0; iLoop < nAccel; iLoop++) @@ -268,7 +268,7 @@ bool CAcceleratorManager::UpdateWndTable() } arrayACCEL.RemoveAll(); - HACCEL hNewTable = CreateAcceleratorTable(lpAccel, nAccel); + HACCEL hNewTable = CreateAcceleratorTable(lpAccel, (int)nAccel); if (!hNewTable) { ::LocalFree(lpAccel); return false; @@ -709,14 +709,14 @@ bool CAcceleratorManager::Write() } // AccelsDatasArray.InsertAt(0, MAKELONG(65535, iCount)); - int count = AccelsDatasArray.GetSize(); + INT_PTR count = AccelsDatasArray.GetSize(); DWORD *data = (DWORD *)malloc(count * sizeof(DWORD)); ASSERT(data != NULL); for(int index = 0; index < count; index++) data[index] = AccelsDatasArray[index]; - regSetBinaryValue("keyboard", (char *)data, count*sizeof(DWORD)); + regSetBinaryValue("keyboard", (char *)data, (int)(count*sizeof(DWORD))); AccelsDatasArray.RemoveAll(); CmdDatasArray.RemoveAll(); diff --git a/src/win32/AcceleratorManager.h b/src/win32/AcceleratorManager.h index c0b12885..1999c57c 100644 --- a/src/win32/AcceleratorManager.h +++ b/src/win32/AcceleratorManager.h @@ -33,6 +33,7 @@ #endif // _MSC_VER >= 1000 + #include "CmdAccelOb.h" diff --git a/src/win32/BitmapControl.h b/src/win32/BitmapControl.h index f2f0c4d8..8a847553 100644 --- a/src/win32/BitmapControl.h +++ b/src/win32/BitmapControl.h @@ -20,7 +20,7 @@ #if !defined(AFX_BITMAPCONTROL_H__2434AADB_B6A5_4E43_AA16_7B65B6F7FA26__INCLUDED_) #define AFX_BITMAPCONTROL_H__2434AADB_B6A5_4E43_AA16_7B65B6F7FA26__INCLUDED_ -#include "..\System.h" // Added by ClassView +#include "../System.h" // Added by ClassView #if _MSC_VER > 1000 #pragma once #endif // _MSC_VER > 1000 diff --git a/src/win32/BugReport.cpp b/src/win32/BugReport.cpp index ad7c31f9..ce4dcd4d 100644 --- a/src/win32/BugReport.cpp +++ b/src/win32/BugReport.cpp @@ -1,6 +1,6 @@ // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -24,6 +24,7 @@ #include "BugReport.h" #include "../agbprint.h" +#include "../AutoBuild.h" #include "../GBA.h" #include "../Globals.h" #include "../Port.h" @@ -215,7 +216,6 @@ CString BugReport::createReport() AppendFormat(report, "Using BIOS : %d\r\n", theApp.useBiosFile); AppendFormat(report, "Skip BIOS : %d\r\n", theApp.skipBiosFile); AppendFormat(report, "Disable SFX : %d\r\n", cpuDisableSfx); - AppendFormat(report, "Skip intro : %d\r\n", theApp.removeIntros); AppendFormat(report, "Throttle : %d\r\n", theApp.throttle); AppendFormat(report, "Rewind : %d\r\n", theApp.rewindTimer); AppendFormat(report, "Auto frame : %d\r\n", theApp.autoFrameSkip); @@ -226,6 +226,7 @@ CString BugReport::createReport() AppendFormat(report, "Green shift : %08x\r\n", systemGreenShift); AppendFormat(report, "Blue shift : %08x\r\n", systemBlueShift); AppendFormat(report, "Layer setting: %04X\r\n", layerSettings); + AppendFormat(report, "Mirroring : %d\r\n", mirroringEnable); AppendFormat(report, "Save type : %d (%d)\r\n", theApp.winSaveType, cpuSaveType); AppendFormat(report, "Flash size : %08X (%08x)\r\n", diff --git a/src/win32/CmdAccelOb.cpp b/src/win32/CmdAccelOb.cpp index 4172b726..a911b9b3 100644 --- a/src/win32/CmdAccelOb.cpp +++ b/src/win32/CmdAccelOb.cpp @@ -1,526 +1,527 @@ -//////////////////////////////////////////////////////////////////////////////// -// Copyright (C) 1998 by Thierry Maurel -// All rights reserved -// -// Distribute freely, except: don't remove my name from the source or -// documentation (don't take credit for my work), mark your changes (don't -// get me blamed for your possible bugs), don't alter or remove this -// notice. -// No warrantee of any kind, express or implied, is included with this -// software; use at your own risk, responsibility for damages (if any) to -// anyone resulting from the use of this software rests entirely with the -// user. -// -// Send bug reports, bug fixes, enhancements, requests, flames, etc., and -// I'll try to keep a version up to date. I can be reached as follows: -// tmaurel@caramail.com (or tmaurel@hol.fr) -// -//////////////////////////////////////////////////////////////////////////////// -// File : CmdAccelOb.cpp -// Project : AccelsEditor -//////////////////////////////////////////////////////////////////////////////// -// Version : 1.0 * Author : T.Maurel -// Date : 17.08.98 -// -// Remarks : -// -//////////////////////////////////////////////////////////////////////////////// - -#include "stdafx.h" -#include "CmdAccelOb.h" - -//////////////////////////////////////////////////////////////////////// -// -// -MAPVIRTKEYS mapVirtKeys[] = { - {VK_LBUTTON, "VK_LBUTTON"}, - {VK_RBUTTON, "VK_RBUTTON"}, - {VK_CANCEL, "VK_CANCEL"}, - {VK_MBUTTON, "VK_MBUTTON"}, - {VK_BACK, "BACK"}, - {VK_TAB, "TAB"}, - {VK_CLEAR, "VK_CLEAR"}, - {VK_RETURN, "RETURN"}, - {VK_SHIFT, "SHIFT"}, - {VK_CONTROL, "CONTROL"}, - {VK_MENU, "MENU"}, - {VK_PAUSE, "PAUSE"}, - {VK_CAPITAL, "CAPITAL"}, - {VK_ESCAPE, "ESCAPE"}, - {VK_SPACE, "SPACE"}, - {VK_PRIOR, "PRIOR"}, - {VK_NEXT, "NEXT"}, - {VK_END, "END"}, - {VK_HOME, "HOME"}, - {VK_LEFT, "LEFT"}, - {VK_UP, "UP"}, - {VK_RIGHT, "RIGHT"}, - {VK_DOWN, "DOWN"}, - {VK_SELECT, "VK_SELECT"}, - {VK_PRINT, "PRINT"}, - {VK_EXECUTE, "EXECUTE"}, - {VK_SNAPSHOT, "SNAPSHOT"}, - {VK_INSERT, "INSERT"}, - {VK_DELETE, "DELETE"}, - {VK_HELP, "VK_HELP"}, - {WORD('0'), "0"}, - {WORD('1'), "1"}, - {WORD('2'), "2"}, - {WORD('3'), "3"}, - {WORD('4'), "4"}, - {WORD('5'), "5"}, - {WORD('6'), "6"}, - {WORD('7'), "7"}, - {WORD('8'), "8"}, - {WORD('9'), "9"}, - {WORD('A'), "A"}, - {WORD('B'), "B"}, - {WORD('C'), "C"}, - {WORD('D'), "D"}, - {WORD('E'), "E"}, - {WORD('F'), "F"}, - {WORD('G'), "G"}, - {WORD('H'), "H"}, - {WORD('I'), "I"}, - {WORD('J'), "J"}, - {WORD('K'), "K"}, - {WORD('L'), "L"}, - {WORD('M'), "M"}, - {WORD('N'), "N"}, - {WORD('O'), "O"}, - {WORD('P'), "P"}, - {WORD('Q'), "Q"}, - {WORD('R'), "R"}, - {WORD('S'), "S"}, - {WORD('T'), "T"}, - {WORD('U'), "U"}, - {WORD('V'), "V"}, - {WORD('W'), "W"}, - {WORD('X'), "X"}, - {WORD('Y'), "Y"}, - {WORD('Z'), "Z"}, - {VK_LWIN, "VK_LWIN"}, - {VK_RWIN, "VK_RWIN"}, - {VK_APPS, "VK_APPS"}, - {VK_NUMPAD0, "NUMPAD0"}, - {VK_NUMPAD1, "NUMPAD1"}, - {VK_NUMPAD2, "NUMPAD2"}, - {VK_NUMPAD3, "NUMPAD3"}, - {VK_NUMPAD4, "NUMPAD4"}, - {VK_NUMPAD5, "NUMPAD5"}, - {VK_NUMPAD6, "NUMPAD6"}, - {VK_NUMPAD7, "NUMPAD7"}, - {VK_NUMPAD8, "NUMPAD8"}, - {VK_NUMPAD9, "NUMPAD9"}, - {VK_MULTIPLY, "MULTIPLY"}, - {VK_ADD, "ADD"}, - {VK_SEPARATOR, "SEPARATOR"}, - {VK_SUBTRACT, "SUBTRACT"}, - {VK_DECIMAL, "DECIMAL"}, - {VK_DIVIDE, "DIVIDE"}, - {VK_F1, "F1"}, - {VK_F2, "F2"}, - {VK_F3, "F3"}, - {VK_F4, "F4"}, - {VK_F5, "F5"}, - {VK_F6, "F6"}, - {VK_F7, "F7"}, - {VK_F8, "F8"}, - {VK_F9, "F9"}, - {VK_F10, "F10"}, - {VK_F11, "F11"}, - {VK_F12, "F12"}, - {VK_F13, "F13"}, - {VK_F14, "F14"}, - {VK_F15, "F15"}, - {VK_F16, "F16"}, - {VK_F17, "F17"}, - {VK_F18, "F18"}, - {VK_F19, "F19"}, - {VK_F20, "F20"}, - {VK_F21, "F21"}, - {VK_F22, "F22"}, - {VK_F23, "F23"}, - {VK_F24, "F24"}, - {VK_NUMLOCK, "NUMLOCK"}, - {VK_SCROLL, "VK_SCROLL"}, - {VK_ATTN, "VK_ATTN"}, - {VK_CRSEL, "VK_CRSEL"}, - {VK_EXSEL, "VK_EXSEL"}, - {VK_EREOF, "VK_EREOF"}, - {VK_PLAY, "VK_PLAY"}, - {VK_ZOOM, "VK_ZOOM"}, - {VK_NONAME, "VK_NONAME"}, - {VK_PA1, "VK_PA1"}, - {VK_OEM_CLEAR, "VK_OEM_CLEAR"}, -}; - - -//////////////////////////////////////////////////////////////////////// -// -// -MAPVIRTKEYS mapVirtSysKeys[] = { - {FCONTROL, "Ctrl"}, - {FALT, "Alt"}, - {FSHIFT, "Shift"}, -}; - - -//////////////////////////////////////////////////////////////////////// -// helper fct for external access -//////////////////////////////////////////////////////////////////////// -// -// -TCHAR* mapVirtKeysStringFromWORD(WORD wKey) -{ - for (int index = 0; index < sizeof(mapVirtKeys)/sizeof(mapVirtKeys[0]); index++) { - if (mapVirtKeys[index].wKey == wKey) - return mapVirtKeys[index].szKey; - } - return NULL; -} - - - -//////////////////////////////////////////////////////////////////////// -// -#define DEFAULT_ACCEL 0x01 -#define USER_ACCEL 0x02 - - -//////////////////////////////////////////////////////////////////////// -// -//////////////////////////////////////////////////////////////////////// -// -// -CAccelsOb::CAccelsOb() -{ - m_cVirt = 0; - m_wKey = 0; - m_bLocked = false; -} - - -//////////////////////////////////////////////////////////////////////// -// -// -CAccelsOb::CAccelsOb(CAccelsOb* pFrom) -{ - ASSERT(pFrom != NULL); - - m_cVirt = pFrom->m_cVirt; - m_wKey = pFrom->m_wKey; - m_bLocked = pFrom->m_bLocked; -} - - -//////////////////////////////////////////////////////////////////////// -// -// -CAccelsOb::CAccelsOb(BYTE cVirt, WORD wKey, bool bLocked) -{ - m_cVirt = cVirt; - m_wKey = wKey; - m_bLocked = bLocked; -} - - -//////////////////////////////////////////////////////////////////////// -// -// -CAccelsOb::CAccelsOb(LPACCEL pACCEL) -{ - ASSERT(pACCEL != NULL); - - m_cVirt = pACCEL->fVirt; - m_wKey = pACCEL->key; - m_bLocked = false; -} - - -//////////////////////////////////////////////////////////////////////// -// -// -CAccelsOb& CAccelsOb::operator=(const CAccelsOb& from) -{ - m_cVirt = from.m_cVirt; - m_wKey = from.m_wKey; - m_bLocked = from.m_bLocked; - - return *this; -} - - -//////////////////////////////////////////////////////////////////////// -// -//////////////////////////////////////////////////////////////////////// -// -// -void CAccelsOb::GetString(CString& szBuffer) -{ - szBuffer = ""; - // in case of the object is not assigned, we avoid error messages - if (m_wKey == 0) - return; - - // modifiers part - for (int i = 0; i < sizetable(mapVirtSysKeys); i++) { - if (m_cVirt & mapVirtSysKeys[i].wKey) { - szBuffer += mapVirtSysKeys[i].szKey; - szBuffer += "+"; - } - } - // and virtual key part - if (1) for (int i = 0; i < sizetable(mapVirtKeys); i++) { - if (m_wKey == mapVirtKeys[i].wKey) { - szBuffer += mapVirtKeys[i].szKey; - return; - } - } - AfxMessageBox("Internal error : (CAccelsOb::GetString) m_wKey invalid"); -} - - -//////////////////////////////////////////////////////////////////////// -// -// -bool CAccelsOb::IsEqual(WORD wKey, bool bCtrl, bool bAlt, bool bShift) -{ - // CString szTemp; - // GetString(szTemp); - - - bool m_bCtrl = (m_cVirt & FCONTROL) ? true : false; - bool bRet = (bCtrl == m_bCtrl); - - bool m_bAlt = (m_cVirt & FALT) ? true : false; - bRet &= (bAlt == m_bAlt); - - bool m_bShift = (m_cVirt & FSHIFT) ? true : false; - bRet &= (bShift == m_bShift); - - bRet &= static_cast(m_wKey == wKey); - - return bRet; -} - - -//////////////////////////////////////////////////////////////////////// -// -// -DWORD CAccelsOb::GetData() -{ - BYTE cLocalCodes = 0; - if (m_bLocked) - cLocalCodes = DEFAULT_ACCEL; - else - cLocalCodes = USER_ACCEL; - - WORD bCodes = MAKEWORD(m_cVirt, cLocalCodes); - return MAKELONG(m_wKey, bCodes); -} - - -//////////////////////////////////////////////////////////////////////// -// -// -bool CAccelsOb::SetData(DWORD dwDatas) -{ - m_wKey = LOWORD(dwDatas); - - WORD bCodes = HIWORD(dwDatas); - m_cVirt = LOBYTE(bCodes); - - BYTE cLocalCodes = HIBYTE(bCodes); - m_bLocked = static_cast(cLocalCodes == DEFAULT_ACCEL); - return true; -} - -//////////////////////////////////////////////////////////////////////// -// -#ifdef _DEBUG -//////////////////////////////////////////////////////////////////////// -// -// -void CAccelsOb::AssertValid() const -{ - CObject::AssertValid(); -} - -//////////////////////////////////////////////////////////////////////// -// -// -void CAccelsOb::Dump(CDumpContext& dc) const -{ - dc << "\t\t"; - CObject::Dump(dc); - dc << "\t\tlocked=" << m_bLocked << ", cVirt=" << m_cVirt << ", wKey=" << m_wKey << "\n\n"; - -} -#endif - -//////////////////////////////////////////////////////////////////////// -// -//////////////////////////////////////////////////////////////////////// -// -// -CCmdAccelOb::CCmdAccelOb() -{ -} - - -//////////////////////////////////////////////////////////////////////// -// -// -CCmdAccelOb::CCmdAccelOb(WORD wIDCommand, LPCTSTR szCommand) -{ - ASSERT(szCommand != NULL); - - m_wIDCommand = wIDCommand; - m_szCommand = szCommand; -} - - -//////////////////////////////////////////////////////////////////////// -// -// -CCmdAccelOb::CCmdAccelOb(BYTE cVirt, WORD wIDCommand, WORD wKey, LPCTSTR szCommand, bool bLocked) -{ - ASSERT(szCommand != NULL); - - m_wIDCommand = wIDCommand; - m_szCommand = szCommand; - - CAccelsOb* pAccel = DEBUG_NEW CAccelsOb(cVirt, wKey, bLocked); - ASSERT(pAccel != NULL); - m_Accels.AddTail(pAccel); -} - - -//////////////////////////////////////////////////////////////////////// -// -// -CCmdAccelOb::~CCmdAccelOb() -{ - POSITION pos = m_Accels.GetHeadPosition(); - while (pos != NULL) - delete m_Accels.GetNext(pos); - m_Accels.RemoveAll(); -} - - -//////////////////////////////////////////////////////////////////////// -// -//////////////////////////////////////////////////////////////////////// -// -// -void CCmdAccelOb::Add(BYTE cVirt, WORD wKey, bool bLocked) -{ - CAccelsOb* pAccel = DEBUG_NEW CAccelsOb(cVirt, wKey, bLocked); - ASSERT(pAccel != NULL); - m_Accels.AddTail(pAccel); -} - - -//////////////////////////////////////////////////////////////////////// -// -// -void CCmdAccelOb::Add(CAccelsOb* pAccel) -{ - ASSERT(pAccel != NULL); - m_Accels.AddTail(pAccel); -} - - -//////////////////////////////////////////////////////////////////////// -// -// -CCmdAccelOb& CCmdAccelOb::operator=(const CCmdAccelOb& from) -{ - Reset(); - - m_wIDCommand = from.m_wIDCommand; - m_szCommand = from.m_szCommand; - - CAccelsOb* pAccel; - POSITION pos = from.m_Accels.GetHeadPosition(); - while (pos != NULL) { - pAccel = DEBUG_NEW CAccelsOb(from.m_Accels.GetNext(pos)); - ASSERT(pAccel != NULL); - m_Accels.AddTail(pAccel); - } - return *this; -} - - -//////////////////////////////////////////////////////////////////////// -// -// -void CCmdAccelOb::DeleteUserAccels() -{ - CAccelsOb* pAccel; - POSITION prevPos; - POSITION pos = m_Accels.GetHeadPosition(); - while (pos != NULL) { - prevPos = pos; - pAccel = m_Accels.GetNext(pos); - if (!pAccel->m_bLocked) { - delete pAccel; - m_Accels.RemoveAt(prevPos); - } - } -} - - -//////////////////////////////////////////////////////////////////////// -// -// -void CCmdAccelOb::Reset() -{ - m_wIDCommand = 0; - m_szCommand = "Empty command"; - - CAccelsOb* pAccel; - POSITION pos = m_Accels.GetHeadPosition(); - while (pos != NULL) { - pAccel = m_Accels.GetNext(pos); - delete pAccel; - } -} - -//////////////////////////////////////////////////////////////////////// -// -#ifdef _DEBUG -//////////////////////////////////////////////////////////////////////// -// -// -void CCmdAccelOb::AssertValid() const -{ - // call base class function first - CObject::AssertValid(); -} - - -//////////////////////////////////////////////////////////////////////// -// -// -void CCmdAccelOb::Dump( CDumpContext& dc ) const -{ - // call base class function first - dc << "\t"; - CObject::Dump( dc ); - - // now do the stuff for our specific class - dc << "\tIDCommand = " << m_wIDCommand; - dc << "\n\tszCommand = " << m_szCommand; - dc << "\n\tAccelerators = {\n"; - - CAccelsOb* pAccel; - POSITION pos = m_Accels.GetHeadPosition(); - while (pos != NULL) { - pAccel = m_Accels.GetNext(pos); - dc << pAccel; - } - dc << "\t}\n"; -} -#endif +//////////////////////////////////////////////////////////////////////////////// +// Copyright (C) 1998 by Thierry Maurel +// All rights reserved +// +// Distribute freely, except: don't remove my name from the source or +// documentation (don't take credit for my work), mark your changes (don't +// get me blamed for your possible bugs), don't alter or remove this +// notice. +// No warrantee of any kind, express or implied, is included with this +// software; use at your own risk, responsibility for damages (if any) to +// anyone resulting from the use of this software rests entirely with the +// user. +// +// Send bug reports, bug fixes, enhancements, requests, flames, etc., and +// I'll try to keep a version up to date. I can be reached as follows: +// tmaurel@caramail.com (or tmaurel@hol.fr) +// +//////////////////////////////////////////////////////////////////////////////// +// File : CmdAccelOb.cpp +// Project : AccelsEditor +//////////////////////////////////////////////////////////////////////////////// +// Version : 1.0 * Author : T.Maurel +// Date : 17.08.98 +// +// Remarks : +// +//////////////////////////////////////////////////////////////////////////////// + +#include "stdafx.h" +#include "CmdAccelOb.h" + +//////////////////////////////////////////////////////////////////////// +// +// +MAPVIRTKEYS mapVirtKeys[] = { + {VK_LBUTTON, "VK_LBUTTON"}, + {VK_RBUTTON, "VK_RBUTTON"}, + {VK_CANCEL, "VK_CANCEL"}, + {VK_MBUTTON, "VK_MBUTTON"}, + {VK_BACK, "BACK"}, + {VK_TAB, "TAB"}, + {VK_CLEAR, "VK_CLEAR"}, + {VK_RETURN, "RETURN"}, + {VK_SHIFT, "SHIFT"}, + {VK_CONTROL, "CONTROL"}, + {VK_MENU, "MENU"}, + {VK_PAUSE, "PAUSE"}, + {VK_CAPITAL, "CAPITAL"}, + {VK_ESCAPE, "ESCAPE"}, + {VK_SPACE, "SPACE"}, + {VK_PRIOR, "PRIOR"}, + {VK_NEXT, "NEXT"}, + {VK_END, "END"}, + {VK_HOME, "HOME"}, + {VK_LEFT, "LEFT"}, + {VK_UP, "UP"}, + {VK_RIGHT, "RIGHT"}, + {VK_DOWN, "DOWN"}, + {VK_SELECT, "VK_SELECT"}, + {VK_PRINT, "PRINT"}, + {VK_EXECUTE, "EXECUTE"}, + {VK_SNAPSHOT, "SNAPSHOT"}, + {VK_INSERT, "INSERT"}, + {VK_DELETE, "DELETE"}, + {VK_HELP, "VK_HELP"}, + {WORD('0'), "0"}, + {WORD('1'), "1"}, + {WORD('2'), "2"}, + {WORD('3'), "3"}, + {WORD('4'), "4"}, + {WORD('5'), "5"}, + {WORD('6'), "6"}, + {WORD('7'), "7"}, + {WORD('8'), "8"}, + {WORD('9'), "9"}, + {WORD('A'), "A"}, + {WORD('B'), "B"}, + {WORD('C'), "C"}, + {WORD('D'), "D"}, + {WORD('E'), "E"}, + {WORD('F'), "F"}, + {WORD('G'), "G"}, + {WORD('H'), "H"}, + {WORD('I'), "I"}, + {WORD('J'), "J"}, + {WORD('K'), "K"}, + {WORD('L'), "L"}, + {WORD('M'), "M"}, + {WORD('N'), "N"}, + {WORD('O'), "O"}, + {WORD('P'), "P"}, + {WORD('Q'), "Q"}, + {WORD('R'), "R"}, + {WORD('S'), "S"}, + {WORD('T'), "T"}, + {WORD('U'), "U"}, + {WORD('V'), "V"}, + {WORD('W'), "W"}, + {WORD('X'), "X"}, + {WORD('Y'), "Y"}, + {WORD('Z'), "Z"}, + {VK_LWIN, "VK_LWIN"}, + {VK_RWIN, "VK_RWIN"}, + {VK_APPS, "VK_APPS"}, + {VK_NUMPAD0, "NUMPAD0"}, + {VK_NUMPAD1, "NUMPAD1"}, + {VK_NUMPAD2, "NUMPAD2"}, + {VK_NUMPAD3, "NUMPAD3"}, + {VK_NUMPAD4, "NUMPAD4"}, + {VK_NUMPAD5, "NUMPAD5"}, + {VK_NUMPAD6, "NUMPAD6"}, + {VK_NUMPAD7, "NUMPAD7"}, + {VK_NUMPAD8, "NUMPAD8"}, + {VK_NUMPAD9, "NUMPAD9"}, + {VK_MULTIPLY, "MULTIPLY"}, + {VK_ADD, "ADD"}, + {VK_SEPARATOR, "SEPARATOR"}, + {VK_SUBTRACT, "SUBTRACT"}, + {VK_DECIMAL, "DECIMAL"}, + {VK_DIVIDE, "DIVIDE"}, + {VK_F1, "F1"}, + {VK_F2, "F2"}, + {VK_F3, "F3"}, + {VK_F4, "F4"}, + {VK_F5, "F5"}, + {VK_F6, "F6"}, + {VK_F7, "F7"}, + {VK_F8, "F8"}, + {VK_F9, "F9"}, + {VK_F10, "F10"}, + {VK_F11, "F11"}, + {VK_F12, "F12"}, + {VK_F13, "F13"}, + {VK_F14, "F14"}, + {VK_F15, "F15"}, + {VK_F16, "F16"}, + {VK_F17, "F17"}, + {VK_F18, "F18"}, + {VK_F19, "F19"}, + {VK_F20, "F20"}, + {VK_F21, "F21"}, + {VK_F22, "F22"}, + {VK_F23, "F23"}, + {VK_F24, "F24"}, + {VK_NUMLOCK, "NUMLOCK"}, + {VK_SCROLL, "VK_SCROLL"}, + {VK_ATTN, "VK_ATTN"}, + {VK_CRSEL, "VK_CRSEL"}, + {VK_EXSEL, "VK_EXSEL"}, + {VK_EREOF, "VK_EREOF"}, + {VK_PLAY, "VK_PLAY"}, + {VK_ZOOM, "VK_ZOOM"}, + {VK_NONAME, "VK_NONAME"}, + {VK_PA1, "VK_PA1"}, + {VK_OEM_CLEAR, "VK_OEM_CLEAR"}, +}; + + +//////////////////////////////////////////////////////////////////////// +// +// +MAPVIRTKEYS mapVirtSysKeys[] = { + {FCONTROL, "Ctrl"}, + {FALT, "Alt"}, + {FSHIFT, "Shift"}, +}; + + +//////////////////////////////////////////////////////////////////////// +// helper fct for external access +//////////////////////////////////////////////////////////////////////// +// +// +TCHAR* mapVirtKeysStringFromWORD(WORD wKey) +{ + for (int index = 0; index < sizeof(mapVirtKeys)/sizeof(mapVirtKeys[0]); index++) { + if (mapVirtKeys[index].wKey == wKey) + return mapVirtKeys[index].szKey; + } + return NULL; +} + + + +//////////////////////////////////////////////////////////////////////// +// +#define DEFAULT_ACCEL 0x01 +#define USER_ACCEL 0x02 + + +//////////////////////////////////////////////////////////////////////// +// +//////////////////////////////////////////////////////////////////////// +// +// +CAccelsOb::CAccelsOb() +{ + m_cVirt = 0; + m_wKey = 0; + m_bLocked = false; +} + + +//////////////////////////////////////////////////////////////////////// +// +// +CAccelsOb::CAccelsOb(CAccelsOb* pFrom) +{ + ASSERT(pFrom != NULL); + + m_cVirt = pFrom->m_cVirt; + m_wKey = pFrom->m_wKey; + m_bLocked = pFrom->m_bLocked; +} + + +//////////////////////////////////////////////////////////////////////// +// +// +CAccelsOb::CAccelsOb(BYTE cVirt, WORD wKey, bool bLocked) +{ + m_cVirt = cVirt; + m_wKey = wKey; + m_bLocked = bLocked; +} + + +//////////////////////////////////////////////////////////////////////// +// +// +CAccelsOb::CAccelsOb(LPACCEL pACCEL) +{ + ASSERT(pACCEL != NULL); + + m_cVirt = pACCEL->fVirt; + m_wKey = pACCEL->key; + m_bLocked = false; +} + + +//////////////////////////////////////////////////////////////////////// +// +// +CAccelsOb& CAccelsOb::operator=(const CAccelsOb& from) +{ + m_cVirt = from.m_cVirt; + m_wKey = from.m_wKey; + m_bLocked = from.m_bLocked; + + return *this; +} + + +//////////////////////////////////////////////////////////////////////// +// +//////////////////////////////////////////////////////////////////////// +// +// +void CAccelsOb::GetString(CString& szBuffer) +{ + szBuffer = ""; + // in case of the object is not assigned, we avoid error messages + if (m_wKey == 0) + return; + + // modifiers part + int i; + for (i = 0; i < sizetable(mapVirtSysKeys); i++) { + if (m_cVirt & mapVirtSysKeys[i].wKey) { + szBuffer += mapVirtSysKeys[i].szKey; + szBuffer += "+"; + } + } + // and virtual key part + for (i = 0; i < sizetable(mapVirtKeys); i++) { + if (m_wKey == mapVirtKeys[i].wKey) { + szBuffer += mapVirtKeys[i].szKey; + return; + } + } + AfxMessageBox("Internal error : (CAccelsOb::GetString) m_wKey invalid"); +} + + +//////////////////////////////////////////////////////////////////////// +// +// +bool CAccelsOb::IsEqual(WORD wKey, bool bCtrl, bool bAlt, bool bShift) +{ + // CString szTemp; + // GetString(szTemp); + + + bool m_bCtrl = (m_cVirt & FCONTROL) ? true : false; + bool bRet = (bCtrl == m_bCtrl); + + bool m_bAlt = (m_cVirt & FALT) ? true : false; + bRet &= (bAlt == m_bAlt); + + bool m_bShift = (m_cVirt & FSHIFT) ? true : false; + bRet &= (bShift == m_bShift); + + bRet &= static_cast(m_wKey == wKey); + + return bRet; +} + + +//////////////////////////////////////////////////////////////////////// +// +// +DWORD CAccelsOb::GetData() +{ + BYTE cLocalCodes = 0; + if (m_bLocked) + cLocalCodes = DEFAULT_ACCEL; + else + cLocalCodes = USER_ACCEL; + + WORD bCodes = MAKEWORD(m_cVirt, cLocalCodes); + return MAKELONG(m_wKey, bCodes); +} + + +//////////////////////////////////////////////////////////////////////// +// +// +bool CAccelsOb::SetData(DWORD dwDatas) +{ + m_wKey = LOWORD(dwDatas); + + WORD bCodes = HIWORD(dwDatas); + m_cVirt = LOBYTE(bCodes); + + BYTE cLocalCodes = HIBYTE(bCodes); + m_bLocked = static_cast(cLocalCodes == DEFAULT_ACCEL); + return true; +} + +//////////////////////////////////////////////////////////////////////// +// +#ifdef _DEBUG +//////////////////////////////////////////////////////////////////////// +// +// +void CAccelsOb::AssertValid() const +{ + CObject::AssertValid(); +} + +//////////////////////////////////////////////////////////////////////// +// +// +void CAccelsOb::Dump(CDumpContext& dc) const +{ + dc << "\t\t"; + CObject::Dump(dc); + dc << "\t\tlocked=" << m_bLocked << ", cVirt=" << m_cVirt << ", wKey=" << m_wKey << "\n\n"; + +} +#endif + +//////////////////////////////////////////////////////////////////////// +// +//////////////////////////////////////////////////////////////////////// +// +// +CCmdAccelOb::CCmdAccelOb() +{ +} + + +//////////////////////////////////////////////////////////////////////// +// +// +CCmdAccelOb::CCmdAccelOb(WORD wIDCommand, LPCTSTR szCommand) +{ + ASSERT(szCommand != NULL); + + m_wIDCommand = wIDCommand; + m_szCommand = szCommand; +} + + +//////////////////////////////////////////////////////////////////////// +// +// +CCmdAccelOb::CCmdAccelOb(BYTE cVirt, WORD wIDCommand, WORD wKey, LPCTSTR szCommand, bool bLocked) +{ + ASSERT(szCommand != NULL); + + m_wIDCommand = wIDCommand; + m_szCommand = szCommand; + + CAccelsOb* pAccel = DEBUG_NEW CAccelsOb(cVirt, wKey, bLocked); + ASSERT(pAccel != NULL); + m_Accels.AddTail(pAccel); +} + + +//////////////////////////////////////////////////////////////////////// +// +// +CCmdAccelOb::~CCmdAccelOb() +{ + POSITION pos = m_Accels.GetHeadPosition(); + while (pos != NULL) + delete m_Accels.GetNext(pos); + m_Accels.RemoveAll(); +} + + +//////////////////////////////////////////////////////////////////////// +// +//////////////////////////////////////////////////////////////////////// +// +// +void CCmdAccelOb::Add(BYTE cVirt, WORD wKey, bool bLocked) +{ + CAccelsOb* pAccel = DEBUG_NEW CAccelsOb(cVirt, wKey, bLocked); + ASSERT(pAccel != NULL); + m_Accels.AddTail(pAccel); +} + + +//////////////////////////////////////////////////////////////////////// +// +// +void CCmdAccelOb::Add(CAccelsOb* pAccel) +{ + ASSERT(pAccel != NULL); + m_Accels.AddTail(pAccel); +} + + +//////////////////////////////////////////////////////////////////////// +// +// +CCmdAccelOb& CCmdAccelOb::operator=(const CCmdAccelOb& from) +{ + Reset(); + + m_wIDCommand = from.m_wIDCommand; + m_szCommand = from.m_szCommand; + + CAccelsOb* pAccel; + POSITION pos = from.m_Accels.GetHeadPosition(); + while (pos != NULL) { + pAccel = DEBUG_NEW CAccelsOb(from.m_Accels.GetNext(pos)); + ASSERT(pAccel != NULL); + m_Accels.AddTail(pAccel); + } + return *this; +} + + +//////////////////////////////////////////////////////////////////////// +// +// +void CCmdAccelOb::DeleteUserAccels() +{ + CAccelsOb* pAccel; + POSITION prevPos; + POSITION pos = m_Accels.GetHeadPosition(); + while (pos != NULL) { + prevPos = pos; + pAccel = m_Accels.GetNext(pos); + if (!pAccel->m_bLocked) { + delete pAccel; + m_Accels.RemoveAt(prevPos); + } + } +} + + +//////////////////////////////////////////////////////////////////////// +// +// +void CCmdAccelOb::Reset() +{ + m_wIDCommand = 0; + m_szCommand = "Empty command"; + + CAccelsOb* pAccel; + POSITION pos = m_Accels.GetHeadPosition(); + while (pos != NULL) { + pAccel = m_Accels.GetNext(pos); + delete pAccel; + } +} + +//////////////////////////////////////////////////////////////////////// +// +#ifdef _DEBUG +//////////////////////////////////////////////////////////////////////// +// +// +void CCmdAccelOb::AssertValid() const +{ + // call base class function first + CObject::AssertValid(); +} + + +//////////////////////////////////////////////////////////////////////// +// +// +void CCmdAccelOb::Dump( CDumpContext& dc ) const +{ + // call base class function first + dc << "\t"; + CObject::Dump( dc ); + + // now do the stuff for our specific class + dc << "\tIDCommand = " << m_wIDCommand; + dc << "\n\tszCommand = " << m_szCommand; + dc << "\n\tAccelerators = {\n"; + + CAccelsOb* pAccel; + POSITION pos = m_Accels.GetHeadPosition(); + while (pos != NULL) { + pAccel = m_Accels.GetNext(pos); + dc << pAccel; + } + dc << "\t}\n"; +} +#endif diff --git a/src/win32/ColorButton.h b/src/win32/ColorButton.h index 3d12f74b..340db218 100644 --- a/src/win32/ColorButton.h +++ b/src/win32/ColorButton.h @@ -20,7 +20,7 @@ #if !defined(AFX_COLORBUTTON_H__DF02109B_B91C_49FD_954F_74A48B83C314__INCLUDED_) #define AFX_COLORBUTTON_H__DF02109B_B91C_49FD_954F_74A48B83C314__INCLUDED_ -#include "..\System.h" // Added by ClassView +#include "../System.h" // Added by ClassView #if _MSC_VER > 1000 #pragma once #endif // _MSC_VER > 1000 diff --git a/src/win32/ColorControl.h b/src/win32/ColorControl.h index e66832c6..9ebaf03d 100644 --- a/src/win32/ColorControl.h +++ b/src/win32/ColorControl.h @@ -20,7 +20,7 @@ #if !defined(AFX_COLORCONTROL_H__747E1E47_DDFA_4D67_B337_A473F2BACB86__INCLUDED_) #define AFX_COLORCONTROL_H__747E1E47_DDFA_4D67_B337_A473F2BACB86__INCLUDED_ -#include "..\System.h" // Added by ClassView +#include "../System.h" // Added by ClassView #if _MSC_VER > 1000 #pragma once #endif // _MSC_VER > 1000 diff --git a/src/win32/Commands.cpp b/src/win32/Commands.cpp index 3e9ae39d..163b1acc 100644 --- a/src/win32/Commands.cpp +++ b/src/win32/Commands.cpp @@ -1,253 +1,258 @@ -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#include "stdafx.h" -#include "AcceleratorManager.h" -#include "..\..\res\resource.h" -#include - -#include // MFC Templates extension -#ifndef CMapStringToWord -typedef CMap< CString, LPCSTR, WORD, WORD& > CMapStringToWord; -#endif - -static CMapStringToWord winAccelStrings; -static bool initialized = false; - -struct { - const char *command; - WORD id; -} winAccelCommands[] = { - { "FileOpen", ID_FILE_OPEN }, - { "FileOpenGameboy", ID_FILE_OPENGAMEBOY }, - { "FileLoad", ID_FILE_LOAD }, - { "FileSave", ID_FILE_SAVE }, - { "FileLoadGame01", ID_FILE_LOADGAME_SLOT1 }, - { "FileLoadGame02", ID_FILE_LOADGAME_SLOT2 }, - { "FileLoadGame03", ID_FILE_LOADGAME_SLOT3 }, - { "FileLoadGame04", ID_FILE_LOADGAME_SLOT4 }, - { "FileLoadGame05", ID_FILE_LOADGAME_SLOT5 }, - { "FileLoadGame06", ID_FILE_LOADGAME_SLOT6 }, - { "FileLoadGame07", ID_FILE_LOADGAME_SLOT7 }, - { "FileLoadGame08", ID_FILE_LOADGAME_SLOT8 }, - { "FileLoadGame09", ID_FILE_LOADGAME_SLOT9 }, - { "FileLoadGame10", ID_FILE_LOADGAME_SLOT10 }, - { "FileLoadGameAutoLoad", ID_FILE_LOADGAME_AUTOLOADMOSTRECENT }, - { "FileLoadGameRecent", ID_FILE_LOADGAME_MOSTRECENT }, - { "FileSaveGame01", ID_FILE_SAVEGAME_SLOT1 }, - { "FileSaveGame02", ID_FILE_SAVEGAME_SLOT2 }, - { "FileSaveGame03", ID_FILE_SAVEGAME_SLOT3 }, - { "FileSaveGame04", ID_FILE_SAVEGAME_SLOT4 }, - { "FileSaveGame05", ID_FILE_SAVEGAME_SLOT5 }, - { "FileSaveGame06", ID_FILE_SAVEGAME_SLOT6 }, - { "FileSaveGame07", ID_FILE_SAVEGAME_SLOT7 }, - { "FileSaveGame08", ID_FILE_SAVEGAME_SLOT8 }, - { "FileSaveGame09", ID_FILE_SAVEGAME_SLOT9 }, - { "FileSaveGame10", ID_FILE_SAVEGAME_SLOT10 }, - { "FileSaveGameOldest", ID_FILE_SAVEGAME_OLDESTSLOT }, - { "FileRecentReset", ID_FILE_RECENT_RESET }, - { "FileRecentFreeze", ID_FILE_RECENT_FREEZE }, - { "FileRecent01", ID_FILE_MRU_FILE1 }, - { "FileRecent02", ID_FILE_MRU_FILE2 }, - { "FileRecent03", ID_FILE_MRU_FILE3 }, - { "FileRecent04", ID_FILE_MRU_FILE4 }, - { "FileRecent05", ID_FILE_MRU_FILE5 }, - { "FileRecent06", ID_FILE_MRU_FILE6 }, - { "FileRecent07", ID_FILE_MRU_FILE7 }, - { "FileRecent08", ID_FILE_MRU_FILE8 }, - { "FileRecent09", ID_FILE_MRU_FILE9 }, - { "FileRecent10", ID_FILE_MRU_FILE10 }, - { "FilePause", ID_FILE_PAUSE }, - { "FileReset", ID_FILE_RESET }, - { "FileImportBatteryFile", ID_FILE_IMPORT_BATTERYFILE }, - { "FileImportGamesharkCodeFile", ID_FILE_IMPORT_GAMESHARKCODEFILE }, - { "FileImportGamesharkSnapshot", ID_FILE_IMPORT_GAMESHARKSNAPSHOT }, - { "FileExportBatteryFile", ID_FILE_EXPORT_BATTERYFILE }, - { "FileExportGamesharkSnapshot", ID_FILE_EXPORT_GAMESHARKSNAPSHOT }, - { "FileScreenCapture", ID_FILE_SCREENCAPTURE }, - { "FileRomInformation", ID_FILE_ROMINFORMATION }, - { "FileToggleMenu", ID_FILE_TOGGLEMENU }, - { "FileClose", ID_FILE_CLOSE }, - { "FileExit", ID_FILE_EXIT }, - { "OptionsFrameSkip0", ID_OPTIONS_VIDEO_FRAMESKIP_0 }, - { "OptionsFrameSkip1", ID_OPTIONS_VIDEO_FRAMESKIP_1 }, - { "OptionsFrameSkip2", ID_OPTIONS_VIDEO_FRAMESKIP_2 }, - { "OptionsFrameSkip3", ID_OPTIONS_VIDEO_FRAMESKIP_3 }, - { "OptionsFrameSkip4", ID_OPTIONS_VIDEO_FRAMESKIP_4 }, - { "OptionsFrameSkip5", ID_OPTIONS_VIDEO_FRAMESKIP_5 }, - { "OptionsFrameSkip6", ID_OPTIONS_VIDEO_FRAMESKIP_6 }, - { "OptionsFrameSkip7", ID_OPTIONS_VIDEO_FRAMESKIP_7 }, - { "OptionsFrameSkip8", ID_OPTIONS_VIDEO_FRAMESKIP_8 }, - { "OptionsFrameSkip9", ID_OPTIONS_VIDEO_FRAMESKIP_9 }, - { "OptionsThrottleNone", ID_OPTIONS_FRAMESKIP_THROTTLE_NOTHROTTLE }, - { "OptionsThrottle025%", ID_OPTIONS_FRAMESKIP_THROTTLE_25 }, - { "OptionsThrottle050%", ID_OPTIONS_FRAMESKIP_THROTTLE_50 }, - { "OptionsThrottle100%", ID_OPTIONS_FRAMESKIP_THROTTLE_100 }, - { "OptionsThrottle150%", ID_OPTIONS_FRAMESKIP_THROTTLE_150 }, - { "OptionsThrottle200%", ID_OPTIONS_FRAMESKIP_THROTTLE_200 }, - { "OptionsThrottleOther", ID_OPTIONS_FRAMESKIP_THROTTLE_OTHER }, - { "OptionsVideoRenderGDI", ID_OPTIONS_VIDEO_RENDERMETHOD_GDI }, - { "OptionsVideoRenderDDRAW", ID_OPTIONS_VIDEO_RENDERMETHOD_DIRECTDRAW }, - { "OptionsVideoRenderD3D", ID_OPTIONS_VIDEO_RENDERMETHOD_DIRECT3D }, - { "OptionsVideoRenderOGL", ID_OPTIONS_VIDEO_RENDERMETHOD_OPENGL }, - { "OptionsVideoVsync", ID_OPTIONS_VIDEO_VSYNC }, - { "OptionsVideoX1", ID_OPTIONS_VIDEO_X1 }, - { "OptionsVideoX2", ID_OPTIONS_VIDEO_X2 }, - { "OptionsVideoX3", ID_OPTIONS_VIDEO_X3 }, - { "OptionsVideoX4", ID_OPTIONS_VIDEO_X4 }, - { "OptionsVideo320x240", ID_OPTIONS_VIDEO_FULLSCREEN320X240 }, - { "OptionsVideo640x480", ID_OPTIONS_VIDEO_FULLSCREEN640X480 }, - { "OptionsVideo800x600", ID_OPTIONS_VIDEO_FULLSCREEN800X600 }, - { "OptionsVideoFullscreen", ID_OPTIONS_VIDEO_FULLSCREEN }, - { "OptionsVideoFullscreenMaxScale", ID_OPTIONS_VIDEO_FULLSCREENMAXSCALE }, - { "OptionsVideoLayersBg0", ID_OPTIONS_VIDEO_LAYERS_BG0 }, - { "OptionsVideoLayersBg1", ID_OPTIONS_VIDEO_LAYERS_BG1 }, - { "OptionsVideoLayersBg2", ID_OPTIONS_VIDEO_LAYERS_BG2 }, - { "OptionsVideoLayersBg3", ID_OPTIONS_VIDEO_LAYERS_BG3 }, - { "OptionsVideoLayersOBJ", ID_OPTIONS_VIDEO_LAYERS_OBJ }, - { "OptionsVideoLayersWIN0", ID_OPTIONS_VIDEO_LAYERS_WIN0 }, - { "OptionsVideoLayersWIN1", ID_OPTIONS_VIDEO_LAYERS_WIN1 }, - { "OptionsVideoLayersOBJWIN", ID_OPTIONS_VIDEO_LAYERS_OBJWIN }, - { "OptionsEmulatorAssociate", ID_OPTIONS_EMULATOR_ASSOCIATE }, - { "OptionsEmulatorDirectories", ID_OPTIONS_EMULATOR_DIRECTORIES }, - { "OptionsEmulatorSelectBIOS", ID_OPTIONS_EMULATOR_SELECTBIOSFILE }, - { "OptionsEmulatorUseBIOS", ID_OPTIONS_EMULATOR_USEBIOSFILE }, - { "OptionsEmulatorSkipBIOS", ID_OPTIONS_EMULATOR_SKIPBIOS }, - { "OptionsEmulatorShowSpeedNone", ID_OPTIONS_EMULATOR_SHOWSPEED_NONE }, - { "OptionsEmulatorShowSpeedPercentage", ID_OPTIONS_EMULATOR_SHOWSPEED_PERCENTAGE }, - { "OptionsEmulatorShowSpeedDetailed", ID_OPTIONS_EMULATOR_SHOWSPEED_DETAILED }, - { "OptionsEmulatorShowSpeedTransparent", ID_OPTIONS_EMULATOR_SHOWSPEED_TRANSPARENT }, - { "OptionsEmulatorSpeedupToggle", ID_OPTIONS_EMULATOR_SPEEDUPTOGGLE }, - { "OptionsEmulatorRemoveIntros", ID_OPTIONS_EMULATOR_REMOVEINTROSGBA }, - { "OptionsEmulatorAutoHideMenu", ID_OPTIONS_EMULATOR_AUTOHIDEMENU }, - { "OptionsEmulatorSaveAuto", ID_OPTIONS_EMULATOR_SAVETYPE_AUTOMATIC }, - { "OptionsEmulatorSaveEEPROM", ID_OPTIONS_EMULATOR_SAVETYPE_EEPROM }, - { "OptionsEmulatorSaveSRAM", ID_OPTIONS_EMULATOR_SAVETYPE_SRAM }, - { "OptionsEmulatorSaveFLASH", ID_OPTIONS_EMULATOR_SAVETYPE_FLASH }, - { "OptionsEmulatorSaveEEPROMSensor", ID_OPTIONS_EMULATOR_SAVETYPE_EEPROMSENSOR }, - { "OptionsEmulatorSaveFlash64K", ID_OPTIONS_EMULATOR_SAVETYPE_FLASH512K }, - { "OptionsEmulatorSaveFlash128K", ID_OPTIONS_EMULATOR_SAVETYPE_FLASH1M }, - { "OptionsEmulatorAutoIPSPatch", ID_OPTIONS_EMULATOR_AUTOMATICALLYIPSPATCH }, - { "OptionsEmulatorAGBPrint", ID_OPTIONS_EMULATOR_AGBPRINT }, - { "OptionsEmulatorRTC", ID_OPTIONS_EMULATOR_REALTIMECLOCK }, - { "OptionsEmulatorRewindInterval", ID_OPTIONS_EMULATOR_REWINDINTERVAL }, - { "OptionsSoundOff", ID_OPTIONS_SOUND_OFF }, - { "OptionsSoundMute", ID_OPTIONS_SOUND_OFF }, /* mute hax */ - { "OptionsSoundOn", ID_OPTIONS_SOUND_ON }, - { "OptionsSoundChannel1", ID_OPTIONS_SOUND_CHANNEL1 }, - { "OptionsSoundChannel2", ID_OPTIONS_SOUND_CHANNEL2 }, - { "OptionsSoundChannel3", ID_OPTIONS_SOUND_CHANNEL3 }, - { "OptionsSoundChannel4", ID_OPTIONS_SOUND_CHANNEL4 }, - { "OptionsSoundDirectSoundA", ID_OPTIONS_SOUND_DIRECTSOUNDA }, - { "OptionsSoundDirectSoundB", ID_OPTIONS_SOUND_DIRECTSOUNDB }, - { "OptionsSound11Khz", ID_OPTIONS_SOUND_11KHZ }, - { "OptionsSound22Khz", ID_OPTIONS_SOUND_22KHZ }, - { "OptionsSound44Khz", ID_OPTIONS_SOUND_44KHZ }, - { "OptionsSoundEcho", ID_OPTIONS_SOUND_ECHO }, - { "OptionsSoundLowPassFilter", ID_OPTIONS_SOUND_LOWPASSFILTER }, - { "OptionsSoundReverseStereo", ID_OPTIONS_SOUND_REVERSESTEREO }, - { "OptionsSoundVolume1x", ID_OPTIONS_SOUND_VOLUME_1X }, - { "OptionsSoundVolume2x", ID_OPTIONS_SOUND_VOLUME_2X }, - { "OptionsSoundVolume3x", ID_OPTIONS_SOUND_VOLUME_3X }, - { "OptionsSoundVolume4x", ID_OPTIONS_SOUND_VOLUME_4X }, - { "OptionsGameboyBorder", ID_OPTIONS_GAMEBOY_BORDER }, - { "OptionsGameboyBorderAutomatic", ID_OPTIONS_GAMEBOY_BORDERAUTOMATIC }, - { "OptionsGameboyColors", ID_OPTIONS_GAMEBOY_COLORS }, - { "OptionsFilterNormal", ID_OPTIONS_FILTER_NORMAL }, - { "OptionsFilterTVMode", ID_OPTIONS_FILTER_TVMODE }, - { "OptionsFilter2xSaI", ID_OPTIONS_FILTER_2XSAI }, - { "OptionsFilterSuper2xSaI", ID_OPTIONS_FILTER_SUPER2XSAI }, - { "OptionsFilterSuperEagle", ID_OPTIONS_FILTER_SUPEREAGLE }, - { "OptionsFilterPixelate", ID_OPTIONS_FILTER16BIT_PIXELATEEXPERIMENTAL }, - { "OptionsFilterAdMameScale2x", ID_OPTIONS_FILTER16BIT_ADVANCEMAMESCALE2X }, - { "OptionsFilterSimple2x", ID_OPTIONS_FILTER16BIT_SIMPLE2X }, - { "OptionsFilterBilinear", ID_OPTIONS_FILTER_BILINEAR }, - { "OptionsFilterBilinearPlus", ID_OPTIONS_FILTER_BILINEARPLUS }, - { "OptionsFilterScanlines", ID_OPTIONS_FILTER_SCANLINES }, - { "OptionsFilterHq2x", ID_OPTIONS_FILTER_HQ2X }, - { "OptionsFilterLq2x", ID_OPTIONS_FILTER_LQ2X }, - { "OptionsFilterIFBNone", ID_OPTIONS_FILTER_INTERFRAMEBLENDING_NONE }, - { "OptionsFilterIFBMotionBlur", ID_OPTIONS_FILTER_INTERFRAMEBLENDING_MOTIONBLUR }, - { "OptionsFilterIFBSmart", ID_OPTIONS_FILTER_INTERFRAMEBLENDING_SMART }, - { "OptionsFilterDisableMMX", ID_OPTIONS_FILTER_DISABLEMMX }, - { "OptionsJoypadConfigure1", ID_OPTIONS_JOYPAD_CONFIGURE_1 }, - { "OptionsJoypadConfigure2", ID_OPTIONS_JOYPAD_CONFIGURE_2 }, - { "OptionsJoypadConfigure3", ID_OPTIONS_JOYPAD_CONFIGURE_3 }, - { "OptionsJoypadConfigure4", ID_OPTIONS_JOYPAD_CONFIGURE_4 }, - { "OptionsJoypadMotionConfigure", ID_OPTIONS_JOYPAD_MOTIONCONFIGURE }, - { "OptionsJoypadAutofireA", ID_OPTIONS_JOYPAD_AUTOFIRE_A }, - { "OptionsJoypadAutofireB", ID_OPTIONS_JOYPAD_AUTOFIRE_B }, - { "OptionsJoypadAutofireL", ID_OPTIONS_JOYPAD_AUTOFIRE_L }, - { "OptionsJoypadAutofireR", ID_OPTIONS_JOYPAD_AUTOFIRE_R }, - { "CheatsSearch", ID_CHEATS_SEARCHFORCHEATS }, - { "CheatsList", ID_CHEATS_CHEATLIST }, - { "CheatsLoad", ID_CHEATS_LOADCHEATLIST }, - { "CheatsSave", ID_CHEATS_SAVECHEATLIST }, - { "CheatsDisable", ID_CHEATS_DISABLECHEATS }, - { "ToolsDebugGDB", ID_TOOLS_DEBUG_GDB }, - { "ToolsDebugGDBLoad", ID_TOOLS_DEBUG_LOADANDWAIT }, - { "ToolsDebugGDBBreak", ID_TOOLS_DEBUG_BREAK }, - { "ToolsDebugGDBDisconnect", ID_TOOLS_DEBUG_DISCONNECT }, - { "ToolsDisassemble", ID_TOOLS_DISASSEMBLE }, - { "ToolsIOViewer", ID_TOOLS_IOVIEWER }, - { "ToolsLogging", ID_TOOLS_LOGGING }, - { "ToolsMapViewer", ID_TOOLS_MAPVIEW }, - { "ToolsMemoryViewer", ID_TOOLS_MEMORYVIEWER }, - { "ToolsOAMViewer", ID_TOOLS_OAMVIEWER }, - { "ToolsPaletteViewer", ID_TOOLS_PALETTEVIEW }, - { "ToolsTileViewer", ID_TOOLS_TILEVIEWER }, - { "ToolsNextFrame", ID_DEBUG_NEXTFRAME }, - { "ToolsRecordSoundStartRecording", ID_OPTIONS_SOUND_STARTRECORDING }, - { "ToolsRecordSoundStopRecording", ID_OPTIONS_SOUND_STOPRECORDING }, - { "ToolsRecordAVIStartRecording", ID_TOOLS_RECORD_STARTAVIRECORDING }, - { "ToolsRecordAVIStopRecording", ID_TOOLS_RECORD_STOPAVIRECORDING }, - { "ToolsRecordMovieStartRecording", ID_TOOLS_RECORD_STARTMOVIERECORDING }, - { "ToolsRecordMovieStopRecording", ID_TOOLS_RECORD_STOPMOVIERECORDING }, - { "ToolsPlayMovieStartPlaying", ID_TOOLS_PLAY_STARTMOVIEPLAYING }, - { "ToolsPlayMovieStopPlaying", ID_TOOLS_PLAY_STOPMOVIEPLAYING }, - { "ToolsRewind", ID_TOOLS_REWIND }, - { "ToolsCustomize", ID_TOOLS_CUSTOMIZE }, - { "HelpBugReport", ID_HELP_BUGREPORT }, - { "HelpFAQ", ID_HELP_FAQ }, - { "HelpAbout", ID_HELP_ABOUT } -}; - -bool winAccelGetID(const char *command, WORD& id) -{ - if(!initialized) { - int count = sizeof(winAccelCommands)/sizeof(winAccelCommands[0]); - - for(int i = 0; i < count; i++) { - winAccelStrings.SetAt(winAccelCommands[i].command, winAccelCommands[i].id); - } - initialized = true; - } - - return winAccelStrings.Lookup(command, id) ? true : false; -} - -void winAccelAddCommands(CAcceleratorManager& mgr) -{ - int count = sizeof(winAccelCommands)/sizeof(winAccelCommands[0]); - - for(int i = 0; i < count; i++) { - if(!mgr.AddCommandAccel(winAccelCommands[i].id, winAccelCommands[i].command, false)) - mgr.CreateEntry(winAccelCommands[i].id, winAccelCommands[i].command); - } - -} +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2005 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#include "stdafx.h" +#include "AcceleratorManager.h" +#include "resource.h" +#include + +#include // MFC Templates extension +#ifndef CMapStringToWord +typedef CMap< CString, LPCSTR, WORD, WORD& > CMapStringToWord; +#endif + +static CMapStringToWord winAccelStrings; +static bool initialized = false; + +struct { + const char *command; + WORD id; +} winAccelCommands[] = { + { "FileOpen", ID_FILE_OPEN }, + { "FileOpenGameboy", ID_FILE_OPENGAMEBOY }, + { "FileLoad", ID_FILE_LOAD }, + { "FileSave", ID_FILE_SAVE }, + { "FileLoadGame01", ID_FILE_LOADGAME_SLOT1 }, + { "FileLoadGame02", ID_FILE_LOADGAME_SLOT2 }, + { "FileLoadGame03", ID_FILE_LOADGAME_SLOT3 }, + { "FileLoadGame04", ID_FILE_LOADGAME_SLOT4 }, + { "FileLoadGame05", ID_FILE_LOADGAME_SLOT5 }, + { "FileLoadGame06", ID_FILE_LOADGAME_SLOT6 }, + { "FileLoadGame07", ID_FILE_LOADGAME_SLOT7 }, + { "FileLoadGame08", ID_FILE_LOADGAME_SLOT8 }, + { "FileLoadGame09", ID_FILE_LOADGAME_SLOT9 }, + { "FileLoadGame10", ID_FILE_LOADGAME_SLOT10 }, + { "FileLoadGameAutoLoad", ID_FILE_LOADGAME_AUTOLOADMOSTRECENT }, + { "FileLoadGameRecent", ID_FILE_LOADGAME_MOSTRECENT }, + { "FileSaveGame01", ID_FILE_SAVEGAME_SLOT1 }, + { "FileSaveGame02", ID_FILE_SAVEGAME_SLOT2 }, + { "FileSaveGame03", ID_FILE_SAVEGAME_SLOT3 }, + { "FileSaveGame04", ID_FILE_SAVEGAME_SLOT4 }, + { "FileSaveGame05", ID_FILE_SAVEGAME_SLOT5 }, + { "FileSaveGame06", ID_FILE_SAVEGAME_SLOT6 }, + { "FileSaveGame07", ID_FILE_SAVEGAME_SLOT7 }, + { "FileSaveGame08", ID_FILE_SAVEGAME_SLOT8 }, + { "FileSaveGame09", ID_FILE_SAVEGAME_SLOT9 }, + { "FileSaveGame10", ID_FILE_SAVEGAME_SLOT10 }, + { "FileSaveGameOldest", ID_FILE_SAVEGAME_OLDESTSLOT }, + { "FileRecentReset", ID_FILE_RECENT_RESET }, + { "FileRecentFreeze", ID_FILE_RECENT_FREEZE }, + { "FileRecent01", ID_FILE_MRU_FILE1 }, + { "FileRecent02", ID_FILE_MRU_FILE2 }, + { "FileRecent03", ID_FILE_MRU_FILE3 }, + { "FileRecent04", ID_FILE_MRU_FILE4 }, + { "FileRecent05", ID_FILE_MRU_FILE5 }, + { "FileRecent06", ID_FILE_MRU_FILE6 }, + { "FileRecent07", ID_FILE_MRU_FILE7 }, + { "FileRecent08", ID_FILE_MRU_FILE8 }, + { "FileRecent09", ID_FILE_MRU_FILE9 }, + { "FileRecent10", ID_FILE_MRU_FILE10 }, + { "FilePause", ID_FILE_PAUSE }, + { "FileReset", ID_FILE_RESET }, + { "FileImportBatteryFile", ID_FILE_IMPORT_BATTERYFILE }, + { "FileImportGamesharkCodeFile", ID_FILE_IMPORT_GAMESHARKCODEFILE }, + { "FileImportGamesharkSnapshot", ID_FILE_IMPORT_GAMESHARKSNAPSHOT }, + { "FileExportBatteryFile", ID_FILE_EXPORT_BATTERYFILE }, + { "FileExportGamesharkSnapshot", ID_FILE_EXPORT_GAMESHARKSNAPSHOT }, + { "FileScreenCapture", ID_FILE_SCREENCAPTURE }, + { "FileRomInformation", ID_FILE_ROMINFORMATION }, + { "FileToggleMenu", ID_FILE_TOGGLEMENU }, + { "FileClose", ID_FILE_CLOSE }, + { "FileExit", ID_FILE_EXIT }, + { "OptionsFrameSkip0", ID_OPTIONS_VIDEO_FRAMESKIP_0 }, + { "OptionsFrameSkip1", ID_OPTIONS_VIDEO_FRAMESKIP_1 }, + { "OptionsFrameSkip2", ID_OPTIONS_VIDEO_FRAMESKIP_2 }, + { "OptionsFrameSkip3", ID_OPTIONS_VIDEO_FRAMESKIP_3 }, + { "OptionsFrameSkip4", ID_OPTIONS_VIDEO_FRAMESKIP_4 }, + { "OptionsFrameSkip5", ID_OPTIONS_VIDEO_FRAMESKIP_5 }, + { "OptionsFrameSkip6", ID_OPTIONS_VIDEO_FRAMESKIP_6 }, + { "OptionsFrameSkip7", ID_OPTIONS_VIDEO_FRAMESKIP_7 }, + { "OptionsFrameSkip8", ID_OPTIONS_VIDEO_FRAMESKIP_8 }, + { "OptionsFrameSkip9", ID_OPTIONS_VIDEO_FRAMESKIP_9 }, + { "OptionsThrottleNone", ID_OPTIONS_FRAMESKIP_THROTTLE_NOTHROTTLE }, + { "OptionsThrottle025%", ID_OPTIONS_FRAMESKIP_THROTTLE_25 }, + { "OptionsThrottle050%", ID_OPTIONS_FRAMESKIP_THROTTLE_50 }, + { "OptionsThrottle100%", ID_OPTIONS_FRAMESKIP_THROTTLE_100 }, + { "OptionsThrottle150%", ID_OPTIONS_FRAMESKIP_THROTTLE_150 }, + { "OptionsThrottle200%", ID_OPTIONS_FRAMESKIP_THROTTLE_200 }, + { "OptionsThrottleOther", ID_OPTIONS_FRAMESKIP_THROTTLE_OTHER }, + { "OptionsVideoRenderGDI", ID_OPTIONS_VIDEO_RENDERMETHOD_GDI }, + { "OptionsVideoRenderDDRAW", ID_OPTIONS_VIDEO_RENDERMETHOD_DIRECTDRAW }, + { "OptionsVideoRenderD3D", ID_OPTIONS_VIDEO_RENDERMETHOD_DIRECT3D }, + { "OptionsVideoRenderOGL", ID_OPTIONS_VIDEO_RENDERMETHOD_OPENGL }, + { "OptionsVideoRenderSelectSkin", ID_OPTIONS_VIDEO_RENDEROPTIONS_SELECTSKIN }, + { "OptionsVideoRenderSkin", ID_OPTIONS_VIDEO_RENDEROPTIONS_SKIN }, + { "OptionsVideoVsync", ID_OPTIONS_VIDEO_VSYNC }, + { "OptionsVideoX1", ID_OPTIONS_VIDEO_X1 }, + { "OptionsVideoX2", ID_OPTIONS_VIDEO_X2 }, + { "OptionsVideoX3", ID_OPTIONS_VIDEO_X3 }, + { "OptionsVideoX4", ID_OPTIONS_VIDEO_X4 }, + { "OptionsVideo320x240", ID_OPTIONS_VIDEO_FULLSCREEN320X240 }, + { "OptionsVideo640x480", ID_OPTIONS_VIDEO_FULLSCREEN640X480 }, + { "OptionsVideo800x600", ID_OPTIONS_VIDEO_FULLSCREEN800X600 }, + { "OptionsVideoFullscreen", ID_OPTIONS_VIDEO_FULLSCREEN }, + { "OptionsVideoFullscreenMaxScale", ID_OPTIONS_VIDEO_FULLSCREENMAXSCALE }, + { "OptionsVideoLayersBg0", ID_OPTIONS_VIDEO_LAYERS_BG0 }, + { "OptionsVideoLayersBg1", ID_OPTIONS_VIDEO_LAYERS_BG1 }, + { "OptionsVideoLayersBg2", ID_OPTIONS_VIDEO_LAYERS_BG2 }, + { "OptionsVideoLayersBg3", ID_OPTIONS_VIDEO_LAYERS_BG3 }, + { "OptionsVideoLayersOBJ", ID_OPTIONS_VIDEO_LAYERS_OBJ }, + { "OptionsVideoLayersWIN0", ID_OPTIONS_VIDEO_LAYERS_WIN0 }, + { "OptionsVideoLayersWIN1", ID_OPTIONS_VIDEO_LAYERS_WIN1 }, + { "OptionsVideoLayersOBJWIN", ID_OPTIONS_VIDEO_LAYERS_OBJWIN }, + { "OptionsEmulatorAssociate", ID_OPTIONS_EMULATOR_ASSOCIATE }, + { "OptionsEmulatorDirectories", ID_OPTIONS_EMULATOR_DIRECTORIES }, + { "OptionsEmulatorSelectBIOS", ID_OPTIONS_EMULATOR_SELECTBIOSFILE }, + { "OptionsEmulatorUseBIOS", ID_OPTIONS_EMULATOR_USEBIOSFILE }, + { "OptionsEmulatorGameOverrides", ID_OPTIONS_EMULATOR_GAMEOVERRIDES }, + { "OptionsEmulatorSkipBIOS", ID_OPTIONS_EMULATOR_SKIPBIOS }, + { "OptionsEmulatorShowSpeedNone", ID_OPTIONS_EMULATOR_SHOWSPEED_NONE }, + { "OptionsEmulatorShowSpeedPercentage", ID_OPTIONS_EMULATOR_SHOWSPEED_PERCENTAGE }, + { "OptionsEmulatorShowSpeedDetailed", ID_OPTIONS_EMULATOR_SHOWSPEED_DETAILED }, + { "OptionsEmulatorShowSpeedTransparent", ID_OPTIONS_EMULATOR_SHOWSPEED_TRANSPARENT }, + { "OptionsEmulatorSpeedupToggle", ID_OPTIONS_EMULATOR_SPEEDUPTOGGLE }, + { "OptionsEmulatorAutoHideMenu", ID_OPTIONS_EMULATOR_AUTOHIDEMENU }, + { "OptionsEmulatorSaveAuto", ID_OPTIONS_EMULATOR_SAVETYPE_AUTOMATIC }, + { "OptionsEmulatorSaveEEPROM", ID_OPTIONS_EMULATOR_SAVETYPE_EEPROM }, + { "OptionsEmulatorSaveSRAM", ID_OPTIONS_EMULATOR_SAVETYPE_SRAM }, + { "OptionsEmulatorSaveFLASH", ID_OPTIONS_EMULATOR_SAVETYPE_FLASH }, + { "OptionsEmulatorSaveEEPROMSensor", ID_OPTIONS_EMULATOR_SAVETYPE_EEPROMSENSOR }, + { "OptionsEmulatorSaveFlash64K", ID_OPTIONS_EMULATOR_SAVETYPE_FLASH512K }, + { "OptionsEmulatorSaveFlash128K", ID_OPTIONS_EMULATOR_SAVETYPE_FLASH1M }, + { "OptionsEmulatorAutoIPSPatch", ID_OPTIONS_EMULATOR_AUTOMATICALLYIPSPATCH }, + { "OptionsEmulatorAGBPrint", ID_OPTIONS_EMULATOR_AGBPRINT }, + { "OptionsEmulatorRTC", ID_OPTIONS_EMULATOR_REALTIMECLOCK }, + { "OptionsEmulatorGenericflashcard", ID_OPTIONS_EMULATOR_GENERICFLASHCARD }, + { "OptionsEmulatorRewindInterval", ID_OPTIONS_EMULATOR_REWINDINTERVAL }, + { "OptionsSoundOff", ID_OPTIONS_SOUND_OFF }, + { "OptionsSoundMute", ID_OPTIONS_SOUND_MUTE }, + { "OptionsSoundOn", ID_OPTIONS_SOUND_ON }, + { "OptionsSoundChannel1", ID_OPTIONS_SOUND_CHANNEL1 }, + { "OptionsSoundChannel2", ID_OPTIONS_SOUND_CHANNEL2 }, + { "OptionsSoundChannel3", ID_OPTIONS_SOUND_CHANNEL3 }, + { "OptionsSoundChannel4", ID_OPTIONS_SOUND_CHANNEL4 }, + { "OptionsSoundDirectSoundA", ID_OPTIONS_SOUND_DIRECTSOUNDA }, + { "OptionsSoundDirectSoundB", ID_OPTIONS_SOUND_DIRECTSOUNDB }, + { "OptionsSound11Khz", ID_OPTIONS_SOUND_11KHZ }, + { "OptionsSound22Khz", ID_OPTIONS_SOUND_22KHZ }, + { "OptionsSound44Khz", ID_OPTIONS_SOUND_44KHZ }, + { "OptionsSoundEcho", ID_OPTIONS_SOUND_ECHO }, + { "OptionsSoundLowPassFilter", ID_OPTIONS_SOUND_LOWPASSFILTER }, + { "OptionsSoundReverseStereo", ID_OPTIONS_SOUND_REVERSESTEREO }, + { "OptionsSoundVolume1x", ID_OPTIONS_SOUND_VOLUME_1X }, + { "OptionsSoundVolume2x", ID_OPTIONS_SOUND_VOLUME_2X }, + { "OptionsSoundVolume3x", ID_OPTIONS_SOUND_VOLUME_3X }, + { "OptionsSoundVolume4x", ID_OPTIONS_SOUND_VOLUME_4X }, + { "OptionsGameboyBorder", ID_OPTIONS_GAMEBOY_BORDER }, + { "OptionsGameboyBorderAutomatic", ID_OPTIONS_GAMEBOY_BORDERAUTOMATIC }, + { "OptionsGameboyColors", ID_OPTIONS_GAMEBOY_COLORS }, + { "OptionsFilterNormal", ID_OPTIONS_FILTER_NORMAL }, + { "OptionsFilterTVMode", ID_OPTIONS_FILTER_TVMODE }, + { "OptionsFilter2xSaI", ID_OPTIONS_FILTER_2XSAI }, + { "OptionsFilterSuper2xSaI", ID_OPTIONS_FILTER_SUPER2XSAI }, + { "OptionsFilterSuperEagle", ID_OPTIONS_FILTER_SUPEREAGLE }, + { "OptionsFilterPixelate", ID_OPTIONS_FILTER16BIT_PIXELATEEXPERIMENTAL }, + { "OptionsFilterMotionBlur", ID_OPTIONS_FILTER16BIT_MOTIONBLUREXPERIMENTAL }, + { "OptionsFilterAdMameScale2x", ID_OPTIONS_FILTER16BIT_ADVANCEMAMESCALE2X }, + { "OptionsFilterSimple2x", ID_OPTIONS_FILTER16BIT_SIMPLE2X }, + { "OptionsFilterBilinear", ID_OPTIONS_FILTER_BILINEAR }, + { "OptionsFilterBilinearPlus", ID_OPTIONS_FILTER_BILINEARPLUS }, + { "OptionsFilterScanlines", ID_OPTIONS_FILTER_SCANLINES }, + { "OptionsFilterHq2x", ID_OPTIONS_FILTER_HQ2X }, + { "OptionsFilterLq2x", ID_OPTIONS_FILTER_LQ2X }, + { "OptionsFilterIFBNone", ID_OPTIONS_FILTER_INTERFRAMEBLENDING_NONE }, + { "OptionsFilterIFBMotionBlur", ID_OPTIONS_FILTER_INTERFRAMEBLENDING_MOTIONBLUR }, + { "OptionsFilterIFBSmart", ID_OPTIONS_FILTER_INTERFRAMEBLENDING_SMART }, + { "OptionsFilterDisableMMX", ID_OPTIONS_FILTER_DISABLEMMX }, + { "OptionsJoypadConfigure1", ID_OPTIONS_JOYPAD_CONFIGURE_1 }, + { "OptionsJoypadConfigure2", ID_OPTIONS_JOYPAD_CONFIGURE_2 }, + { "OptionsJoypadConfigure3", ID_OPTIONS_JOYPAD_CONFIGURE_3 }, + { "OptionsJoypadConfigure4", ID_OPTIONS_JOYPAD_CONFIGURE_4 }, + { "OptionsJoypadMotionConfigure", ID_OPTIONS_JOYPAD_MOTIONCONFIGURE }, + { "OptionsJoypadAutofireA", ID_OPTIONS_JOYPAD_AUTOFIRE_A }, + { "OptionsJoypadAutofireB", ID_OPTIONS_JOYPAD_AUTOFIRE_B }, + { "OptionsJoypadAutofireL", ID_OPTIONS_JOYPAD_AUTOFIRE_L }, + { "OptionsJoypadAutofireR", ID_OPTIONS_JOYPAD_AUTOFIRE_R }, + { "CheatsSearch", ID_CHEATS_SEARCHFORCHEATS }, + { "CheatsList", ID_CHEATS_CHEATLIST }, + { "CheatsLoad", ID_CHEATS_LOADCHEATLIST }, + { "CheatsSave", ID_CHEATS_SAVECHEATLIST }, + { "CheatsDisable", ID_CHEATS_DISABLECHEATS }, + { "ToolsDebugGDB", ID_TOOLS_DEBUG_GDB }, + { "ToolsDebugGDBLoad", ID_TOOLS_DEBUG_LOADANDWAIT }, + { "ToolsDebugGDBBreak", ID_TOOLS_DEBUG_BREAK }, + { "ToolsDebugGDBDisconnect", ID_TOOLS_DEBUG_DISCONNECT }, + { "ToolsDisassemble", ID_TOOLS_DISASSEMBLE }, + { "ToolsIOViewer", ID_TOOLS_IOVIEWER }, + { "ToolsLogging", ID_TOOLS_LOGGING }, + { "ToolsMapViewer", ID_TOOLS_MAPVIEW }, + { "ToolsMemoryViewer", ID_TOOLS_MEMORYVIEWER }, + { "ToolsOAMViewer", ID_TOOLS_OAMVIEWER }, + { "ToolsPaletteViewer", ID_TOOLS_PALETTEVIEW }, + { "ToolsTileViewer", ID_TOOLS_TILEVIEWER }, + { "ToolsNextFrame", ID_DEBUG_NEXTFRAME }, + { "ToolsRecordSoundStartRecording", ID_OPTIONS_SOUND_STARTRECORDING }, + { "ToolsRecordSoundStopRecording", ID_OPTIONS_SOUND_STOPRECORDING }, + { "ToolsRecordAVIStartRecording", ID_TOOLS_RECORD_STARTAVIRECORDING }, + { "ToolsRecordAVIStopRecording", ID_TOOLS_RECORD_STOPAVIRECORDING }, + { "ToolsRecordMovieStartRecording", ID_TOOLS_RECORD_STARTMOVIERECORDING }, + { "ToolsRecordMovieStopRecording", ID_TOOLS_RECORD_STOPMOVIERECORDING }, + { "ToolsPlayMovieStartPlaying", ID_TOOLS_PLAY_STARTMOVIEPLAYING }, + { "ToolsPlayMovieStopPlaying", ID_TOOLS_PLAY_STOPMOVIEPLAYING }, + { "ToolsRewind", ID_TOOLS_REWIND }, + { "ToolsCustomize", ID_TOOLS_CUSTOMIZE }, + { "HelpBugReport", ID_HELP_BUGREPORT }, + { "HelpFAQ", ID_HELP_FAQ }, + { "HelpAbout", ID_HELP_ABOUT }, + { "SystemMinimize", ID_SYSTEM_MINIMIZE } +}; + +bool winAccelGetID(const char *command, WORD& id) +{ + if(!initialized) { + int count = sizeof(winAccelCommands)/sizeof(winAccelCommands[0]); + + for(int i = 0; i < count; i++) { + winAccelStrings.SetAt(winAccelCommands[i].command, winAccelCommands[i].id); + } + initialized = true; + } + + return winAccelStrings.Lookup(command, id) ? true : false; +} + +void winAccelAddCommands(CAcceleratorManager& mgr) +{ + int count = sizeof(winAccelCommands)/sizeof(winAccelCommands[0]); + + for(int i = 0; i < count; i++) { + if(!mgr.AddCommandAccel(winAccelCommands[i].id, winAccelCommands[i].command, false)) + mgr.CreateEntry(winAccelCommands[i].id, winAccelCommands[i].command); + } + +} diff --git a/src/win32/Direct3D.cpp b/src/win32/Direct3D.cpp index 6b22ce38..82de2d17 100644 --- a/src/win32/Direct3D.cpp +++ b/src/win32/Direct3D.cpp @@ -1,1060 +1,636 @@ -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2005 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#include "stdafx.h" -#include -#include "VBA.H" -#include "MainWnd.h" -#include "UniVideoModeDlg.h" -#include "../Util.h" -#include "../Globals.h" -#include "../Util.h" -#include "../gb/gbGlobals.h" -// Link with Direct3D9 -#pragma comment(lib, "D3d9.lib") -#pragma comment(lib, "D3dx9.lib") -#define DIRECT3D_VERSION 0x0900 -#include -#include - -#include "../gbafilter.h" - -#ifdef _DEBUG -#define new DEBUG_NEW -#undef THIS_FILE -static char THIS_FILE[] = __FILE__; -#endif - -#ifdef MMX -extern "C" bool cpu_mmx; -extern bool detectMMX(); -#endif - -extern int Init_2xSaI(u32); -extern void winlog(const char *,...); -extern int systemSpeed; - - -// Vertex format declarations -const DWORD D3DFVF_TEXTBOXVERTEX = D3DFVF_XYZRHW | D3DFVF_DIFFUSE; -struct TEXTBOXVERTEX { - FLOAT x, y, z, rhw; - D3DCOLOR color; -}; -const DWORD D3DFVF_IMAGEVERTEX = D3DFVF_XYZ | D3DFVF_DIFFUSE | D3DFVF_TEX1; -struct IMAGEVERTEX { - FLOAT x, y, z; - D3DCOLOR color; - FLOAT u, v; -}; - - - -class Direct3DDisplay : public IDisplay -{ -public: // Class - Direct3DDisplay(); - virtual ~Direct3DDisplay(); - - -public: // Interface - virtual bool initialize(); - virtual void cleanup(); - virtual void render(); - virtual void renderMenu(); - virtual void clear(); - virtual bool changeRenderSize(int w, int h); - virtual void resize(int w, int h); - virtual DISPLAY_TYPE getType() { return DIRECT_3D; }; - virtual void setOption(const char *, int); - virtual int selectFullScreenMode(GUID **); - virtual int selectFullScreenMode2(); - - -private: // Functions - void restoreDeviceObjects(void); - void invalidateDeviceObjects(); - void setPresentationType(); - bool initializeOffscreen(unsigned int w, unsigned int h); - void updateFiltering(int); - bool resetDevice(); - void initializeMatrices(); - - -private: // Variables - int SelectedFreq, SelectedAdapter; - bool initSucessful; - bool doNotRender; - bool filterDisabled; - bool lockableBuffer; - LPDIRECT3D9 pD3D; - LPDIRECT3DDEVICE9 pDevice; - LPDIRECT3DTEXTURE9 pTexture; - LPD3DXFONT pFont; - D3DPRESENT_PARAMETERS dpp; - D3DFORMAT screenFormat; - D3DDISPLAYMODE mode; - - bool fullscreen; - int width, height; // Size of the source image to display - IMAGEVERTEX verts[4]; // The coordinates for our image texture - TEXTBOXVERTEX msgBox[4]; - int textureWidth; // Size of the texture, - int textureHeight; // where the source image is copied to - bool keepAspectRatio; -}; - -Direct3DDisplay::Direct3DDisplay() -{ - initSucessful = false; - doNotRender = true; - pD3D = NULL; - pDevice = NULL; - pTexture = NULL; - pFont = NULL; - screenFormat = D3DFMT_UNKNOWN; - width = 0; - height = 0; - filterDisabled = false; - keepAspectRatio = false; // theApp.d3dKeepAspectRatio; - lockableBuffer = false; -} - -Direct3DDisplay::~Direct3DDisplay() -{ - cleanup(); -} - -void Direct3DDisplay::setPresentationType() -{ - // Change display mode - memset(&dpp, 0, sizeof(dpp)); - dpp.Windowed = !fullscreen; - if (fullscreen) - dpp.BackBufferFormat = - (theApp.fsColorDepth == 32) ? D3DFMT_X8R8G8B8 : D3DFMT_R5G6B5; - else - dpp.BackBufferFormat = mode.Format; - dpp.BackBufferCount = 3; - dpp.MultiSampleType = D3DMULTISAMPLE_NONE; - dpp.SwapEffect = D3DSWAPEFFECT_DISCARD; - dpp.BackBufferWidth = fullscreen ? theApp.fsWidth : theApp.surfaceSizeX; - dpp.BackBufferHeight = fullscreen ? theApp.fsHeight : theApp.surfaceSizeY; - dpp.hDeviceWindow = theApp.m_pMainWnd->GetSafeHwnd(); - dpp.FullScreen_RefreshRateInHz = fullscreen ? theApp.fsFrequency : 0; -// dpp.Flags = D3DPRESENTFLAG_LOCKABLE_BACKBUFFER; - dpp.Flags = theApp.menuToggle ? D3DPRESENTFLAG_LOCKABLE_BACKBUFFER : 0; - if (theApp.vsync) - dpp.PresentationInterval = D3DPRESENT_INTERVAL_ONE; // VSync - else - dpp.PresentationInterval = D3DPRESENT_INTERVAL_IMMEDIATE; // No Sync -} - -void Direct3DDisplay::cleanup() -{ // interface funtion - if(pD3D != NULL) { - if(pFont) { - pFont->Release(); - pFont = NULL; - } - - if(pTexture) - { - pTexture->Release(); - pTexture = NULL; - } - - if(pDevice) { - pDevice->Release(); - pDevice = NULL; - } - - pD3D->Release(); - pD3D = NULL; - } - - initSucessful = false; - doNotRender = true; -} - -bool Direct3DDisplay::initialize() -{ // interface function - initSucessful = false; - doNotRender = true; - - // Get emulated image's dimensions - switch(theApp.cartridgeType) - { - case IMAGE_GBA: - theApp.sizeX = 240; - theApp.sizeY = 160; - break; - case IMAGE_GB: - if (gbBorderOn) - { - theApp.sizeX = 256; - theApp.sizeY = 224; - } - else - { - theApp.sizeX = 160; - theApp.sizeY = 144; - } - break; - } - - theApp.rect.left = 0; - theApp.rect.top = 0; - theApp.rect.right = theApp.sizeX; - theApp.rect.bottom = theApp.sizeY; - - - switch(theApp.videoOption) - { - case VIDEO_1X: - theApp.surfaceSizeX = theApp.sizeX; - theApp.surfaceSizeY = theApp.sizeY; - fullscreen = false; - break; - case VIDEO_2X: - theApp.surfaceSizeX = theApp.sizeX * 2; - theApp.surfaceSizeY = theApp.sizeY * 2; - fullscreen = false; - break; - case VIDEO_3X: - theApp.surfaceSizeX = theApp.sizeX * 3; - theApp.surfaceSizeY = theApp.sizeY * 3; - fullscreen = false; - break; - case VIDEO_4X: - theApp.surfaceSizeX = theApp.sizeX * 4; - theApp.surfaceSizeY = theApp.sizeY * 4; - fullscreen = false; - break; - case VIDEO_320x240: - case VIDEO_640x480: - case VIDEO_800x600: - case VIDEO_1024x768: - case VIDEO_1280x1024: - case VIDEO_OTHER: - float scaleX = ((float)theApp.fsWidth / theApp.sizeX); - float scaleY = ((float)theApp.fsHeight / theApp.sizeY); - float min = scaleX < scaleY ? scaleX : scaleY; - if(theApp.fsMaxScale) - min = min > theApp.fsMaxScale ? theApp.fsMaxScale : min; - if(theApp.fullScreenStretch) - { - theApp.surfaceSizeX = theApp.fsWidth; - theApp.surfaceSizeY = theApp.fsHeight; - } - else - { - theApp.surfaceSizeX = (int)(theApp.sizeX * min); - theApp.surfaceSizeY = (int)(theApp.sizeY * min); - } - fullscreen = true; - break; - } - - - theApp.dest.left = 0; - theApp.dest.top = 0; - theApp.dest.right = theApp.surfaceSizeX; - theApp.dest.bottom = theApp.surfaceSizeY; - - DWORD style = WS_POPUP | WS_VISIBLE; - DWORD styleEx = 0; - - if(theApp.videoOption <= VIDEO_4X) - style |= WS_OVERLAPPEDWINDOW; - else - styleEx = 0; - - if(theApp.videoOption <= VIDEO_4X) - AdjustWindowRectEx(&theApp.dest, style, TRUE, styleEx); - else - AdjustWindowRectEx(&theApp.dest, style, FALSE, styleEx); - - int winSizeX = theApp.dest.right-theApp.dest.left; - int winSizeY = theApp.dest.bottom-theApp.dest.top; - - if(theApp.videoOption > VIDEO_4X) { - winSizeX = theApp.fsWidth; - winSizeY = theApp.fsHeight; - } - int x = 0; - int y = 0; - - if(theApp.videoOption <= VIDEO_4X) { - x = theApp.windowPositionX; - y = theApp.windowPositionY; - } - - // Create a window - MainWnd *pWnd = new MainWnd; - theApp.m_pMainWnd = pWnd; - - pWnd->CreateEx( - styleEx, - theApp.wndClass, - "VisualBoyAdvance", - style, - x, y, - winSizeX, winSizeY, - NULL, - 0); - - if (!(HWND)*pWnd) - { - winlog("Error creating Window %08x\n", GetLastError()); - return FALSE; - } - - theApp.updateMenuBar(); - - theApp.adjustDestRect(); - - - // Create an IDirect3D9 object - pD3D = Direct3DCreate9(D3D_SDK_VERSION); - if(pD3D == NULL) - { - winlog("Error creating Direct3D object\n"); - return FALSE; - } - - - - - // Display resolution - pD3D->GetAdapterDisplayMode(D3DADAPTER_DEFAULT, &mode); - - switch(mode.Format) { - case D3DFMT_R8G8B8: - systemColorDepth = 24; - systemRedShift = 19; - systemGreenShift = 11; - systemBlueShift = 3; - Init_2xSaI(32); - break; - case D3DFMT_X8R8G8B8: - systemColorDepth = 32; - systemRedShift = 19; - systemGreenShift = 11; - systemBlueShift = 3; - Init_2xSaI(32); - break; - case D3DFMT_R5G6B5: - systemColorDepth = 16; - systemRedShift = 11; - systemGreenShift = 6; - systemBlueShift = 0; - Init_2xSaI(565); - break; - case D3DFMT_X1R5G5B5: - systemColorDepth = 16; - systemRedShift = 10; - systemGreenShift = 5; - systemBlueShift = 0; - Init_2xSaI(555); - break; - default: - systemMessage(0,"Unsupported D3D format %d", mode.Format); - return false; - } - theApp.fsColorDepth = systemColorDepth; - - // Check the available pre-defined fullscreen modes and enable menu items - unsigned int nModes, i; - D3DDISPLAYMODE dm; - - theApp.mode320Available = false; - theApp.mode640Available = false; - theApp.mode800Available = false; - theApp.mode1024Available = false; - theApp.mode1280Available = false; - - nModes = pD3D->GetAdapterModeCount(theApp.fsAdapter, mode.Format); - for (i = 0; iEnumAdapterModes(theApp.fsAdapter, mode.Format, i, &dm) ) - { - if ( (dm.Width == 320) && (dm.Height == 240) ) - theApp.mode320Available = true; - if ( (dm.Width == 640) && (dm.Height == 480) ) - theApp.mode640Available = true; - if ( (dm.Width == 800) && (dm.Height == 600) ) - theApp.mode800Available = true; - if ( (dm.Width == 1024) && (dm.Height == 768) ) - theApp.mode1024Available = true; - if ( (dm.Width == 1280) && (dm.Height == 1024) ) - theApp.mode1280Available = true; - } - } - - -#ifdef MMX - if (!theApp.disableMMX) - cpu_mmx = theApp.detectMMX(); - else - cpu_mmx = 0; -#endif - - screenFormat = mode.Format; - - setPresentationType(); - - - DWORD BehaviorFlags; - D3DCAPS9 caps; - if (D3D_OK == pD3D->GetDeviceCaps(theApp.fsAdapter, D3DDEVTYPE_HAL, &caps)) { - if (caps.DevCaps & D3DDEVCAPS_HWTRANSFORMANDLIGHT) { - BehaviorFlags = D3DCREATE_HARDWARE_VERTEXPROCESSING; - } else { - BehaviorFlags = D3DCREATE_SOFTWARE_VERTEXPROCESSING; - } - if (caps.DevCaps & D3DDEVCAPS_PUREDEVICE) { - BehaviorFlags |= D3DCREATE_PUREDEVICE; - } - } else { - winlog("Error retrieving device's D3D capabilities\n"); - return false; - } - - - HRESULT hret = pD3D->CreateDevice(theApp.fsAdapter, - caps.DeviceType, - pWnd->GetSafeHwnd(), - BehaviorFlags, - &dpp, - &pDevice); -#ifdef _DEBUG - switch(hret) - { - case D3DERR_DEVICELOST: - winlog("Error creating Direct3DDevice (D3DERR_DEVICELOST)\n"); - return false; - break; - case D3DERR_INVALIDCALL: - winlog("Error creating Direct3DDevice (D3DERR_INVALIDCALL)\n"); - return false; - break; - case D3DERR_NOTAVAILABLE: - winlog("Error creating Direct3DDevice (D3DERR_NOTAVAILABLE)\n"); - return false; - break; - case D3DERR_OUTOFVIDEOMEMORY: - winlog("Error creating Direct3DDevice (D3DERR_OUTOFVIDEOMEMORY)\n"); - return false; - break; - } -#endif - - restoreDeviceObjects(); - - // Set the status message's background vertex information, that does not need to be changed in realtime - msgBox[0].z = 0.5f; - msgBox[0].rhw = 1.0f; - msgBox[0].color = 0x7fffffff; - msgBox[1].z = 0.5f; - msgBox[1].rhw = 1.0f; - msgBox[1].color = 0x7f7f7f7f; - msgBox[2].z = 0.5f; - msgBox[2].rhw = 1.0f; - msgBox[2].color = 0x7f7f7fff; - msgBox[3].z = 0.5f; - msgBox[3].rhw = 1.0f; - msgBox[3].color = 0x7f7f7f7f; - - // Set up the vertices of the texture - verts[0].z = verts[1].z = verts[2].z = verts[3].z = 1.0f; - verts[0].color = verts[1].color = verts[2].color = verts[3].color = D3DCOLOR_ARGB(0xff, 0xff, 0xff, 0xff); - verts[1].u = verts[2].u = 1.0f; - verts[0].u = verts[3].u = 0.0f; - verts[0].v = verts[1].v = 0.0f; - verts[2].v = verts[3].v = 1.0f; - verts[0].x = verts[3].x = 0.0f; - verts[1].x = verts[2].x = 1.0f; - verts[0].y = verts[1].y = 0.0f; - verts[2].y = verts[3].y = 1.0f; - - - - utilUpdateSystemColorMaps(theApp.filterLCD ); - theApp.updateFilter(); - theApp.updateIFB(); - - pWnd->DragAcceptFiles(TRUE); - - initSucessful = true; - doNotRender = false; - return TRUE; -} - -bool Direct3DDisplay::initializeOffscreen(unsigned int w, unsigned int h) -{ - D3DFORMAT format = screenFormat; - - unsigned int correctedWidth=w, correctedHeight=h; - - // This function corrects the texture size automaticly - if(D3D_OK == D3DXCheckTextureRequirements( - pDevice, - &correctedWidth, - &correctedHeight, - NULL, - 0, - &format, - D3DPOOL_MANAGED)) - { - if( (correctedWidth < w) || (correctedHeight < h) ) - { - if(theApp.filterFunction) - { - filterDisabled = true; - theApp.filterFunction = NULL; - systemMessage(0, "3D card cannot support needed texture size for filter function. Disabling it"); - } - else - systemMessage(0, "Graphics card doesn't support needed texture size for emulation."); - } - else filterDisabled = false; - - if(D3D_OK == D3DXCreateTexture( - pDevice, - correctedWidth, - correctedHeight, - 1, - D3DUSAGE_DYNAMIC, - format, - D3DPOOL_DEFAULT, - &pTexture) ) - { - width = w; - height = h; - textureWidth = correctedWidth; - textureHeight = correctedHeight; - return true; - } - else systemMessage(0, "Texture creation failed"); - } - return false; -} - - -void Direct3DDisplay::updateFiltering(int filter) -{ //TODO: use GetSampletState before changing - if(!pDevice) { - return; - } - - HRESULT res; - - switch(filter) - { - default: - case 0: - // point filtering - res = pDevice->SetSamplerState( 0, D3DSAMP_MINFILTER, D3DTEXF_POINT ); - if (res != D3D_OK) { - systemMessage(0, "Could not set point filtering mode: %d", res); - return; - } - res = pDevice->SetSamplerState( 0, D3DSAMP_MAGFILTER, D3DTEXF_POINT ); - if (res != D3D_OK) { - systemMessage(0, "Could not set point filtering mode: %d", res); - return; - } - break; - case 1: - // bilinear - res = pDevice->SetSamplerState( 0, D3DSAMP_MINFILTER, D3DTEXF_LINEAR ); - if (res != D3D_OK) { - systemMessage(0, "Could not set bilinear filtering mode: %d", res); - return; - } - res = pDevice->SetSamplerState( 0, D3DSAMP_MAGFILTER, D3DTEXF_LINEAR ); - if (res != D3D_OK) { - systemMessage(0, "Could not set bilinear filtering mode: %d", res); - return; - } - // Don't wrap textures .. otherwise bottom blurs top to bottom - pDevice->SetSamplerState(0, D3DSAMP_ADDRESSU, D3DTADDRESS_CLAMP); - pDevice->SetSamplerState(0, D3DSAMP_ADDRESSV, D3DTADDRESS_CLAMP); - break; - } - return; -} - -void Direct3DDisplay::clear() -{ // interface function - if(!initSucessful) return; - - if (pDevice) - pDevice->Clear( 0L, NULL, D3DCLEAR_TARGET, -#ifdef _DEBUG - 0xffff00ff //pink -#else - 0xff000000 //black -#endif - , 1.0f, 0L ); -} - -void Direct3DDisplay::renderMenu() -{ // interface function - if(!initSucessful) return; - - if(theApp.m_pMainWnd) - theApp.m_pMainWnd->DrawMenuBar(); -} - -void Direct3DDisplay::render() -{ // interface function - if(!pDevice) return; - if(!initSucessful) return; - if(doNotRender) return; - - unsigned int nBytesPerPixel = systemColorDepth >> 3; // This is the byte count of a Pixel - unsigned int pitch = (theApp.filterWidth * nBytesPerPixel) + 4; // The size of a scanline in bytes - - // Test the cooperative level to see if it's okay to render - HRESULT hr; - hr = pDevice->TestCooperativeLevel(); - switch(hr) - { - case D3DERR_DEVICENOTRESET: - resetDevice(); - break; - case D3DERR_DEVICELOST: - winlog("Render: D3DERR_DEVICELOST\n"); - return; - break; - case D3DERR_DRIVERINTERNALERROR: - winlog("Render: D3DERR_DRIVERINTERNALERROR\n"); - cleanup(); - if(initialize()) { - return; - } else { // reinitialize device failed - AfxPostQuitMessage(D3DERR_DRIVERINTERNALERROR); - } - break; - } - - // Clear the screen - if (pDevice) - pDevice->Clear( 0L, NULL, D3DCLEAR_TARGET, -#ifdef _DEBUG - 0xffff00ff //pink -#else - 0xff000000 //black -#endif - , 1.0f, 0L ); - - - if(SUCCEEDED(pDevice->BeginScene())) - { - D3DLOCKED_RECT locked; - - if( D3D_OK == pTexture->LockRect(0, &locked, NULL, D3DLOCK_DISCARD) ) - { - if(theApp.filterFunction) - { - theApp.filterFunction( - pix + pitch, - pitch, - (u8*)theApp.delta, - (u8*)locked.pBits, - locked.Pitch, - theApp.filterWidth, - theApp.filterHeight); - } - else - { - // Copy the image at [pix] to the locked Direct3D texture - __asm - { - mov eax, theApp.sizeX ; Initialize - mov ebx, theApp.sizeY ; - mov edi, locked.pBits ; - mov edx, locked.Pitch ; - - cmp systemColorDepth, 16 ; Check systemColorDepth==16bit - jnz gbaOtherColor ; - sub edx, eax ; - sub edx, eax ; - mov esi, pix ; - lea esi,[esi+2*eax+4] ; - shr eax, 1 ; -gbaLoop16bit: - mov ecx, eax ; - rep movsd ; - add esi, 4 ; - add edi, edx ; - dec ebx ; - jnz gbaLoop16bit ; - jmp gbaLoopEnd ; -gbaOtherColor: - cmp systemColorDepth, 32 ; Check systemColorDepth==32bit - jnz gbaOtherColor2 ; - - lea esi, [eax*4] ; - sub edx, esi ; - mov esi, pix ; - lea esi, [esi+4*eax+4] ; -gbaLoop32bit: - mov ecx, eax ; - rep movsd ; ECX times: Move DWORD at [ESI] to [EDI] | ESI++ EDI++ - add esi, 4 ; - add edi, edx ; - dec ebx ; - jnz gbaLoop32bit ; - jmp gbaLoopEnd ; -gbaOtherColor2: - lea eax, [eax+2*eax] ; Work like systemColorDepth==24bit - sub edx, eax ; -gbaLoop24bit: - mov ecx, eax ; - shr ecx, 2 ; - rep movsd ; - add edi, edx ; - dec ebx ; - jnz gbaLoop24bit ; -gbaLoopEnd: - } - - //C Version of the code above - //unsigned int i; - //int x, y, srcPitch = (theApp.sizeX+1) * nBytesPerPixel; - //unsigned char * src = ((unsigned char*)pix)+srcPitch; - //unsigned char * dst = (unsigned char*)locked.pBits; - //for (y=0;yUnlockRect(0); - - pDevice->SetFVF( D3DFVF_IMAGEVERTEX ); - pDevice->SetTexture( 0, pTexture ); - pDevice->DrawPrimitiveUP( D3DPT_TRIANGLEFAN, 2, verts, sizeof( IMAGEVERTEX ) ); - - } // SUCCEEDED(pTexture->LockRect... - else - { - systemMessage(0,"Rendering error"); - } - - - // Draw the screen message - if(theApp.screenMessage) - { - if( ((GetTickCount() - theApp.screenMessageTime) < 3000) && - !theApp.disableStatusMessage && pFont ) - { - CRect msgRect( - 64, - dpp.BackBufferHeight - 32, - dpp.BackBufferWidth - 64, - dpp.BackBufferHeight); - - msgBox[0].x = (FLOAT)msgRect.left; - msgBox[0].y = (FLOAT)msgRect.top; - msgBox[1].x = (FLOAT)msgRect.right; - msgBox[1].y = (FLOAT)msgRect.top; - msgBox[2].x = (FLOAT)msgRect.right; - msgBox[2].y = (FLOAT)msgRect.bottom; - msgBox[3].x = (FLOAT)msgRect.left; - msgBox[3].y = (FLOAT)msgRect.bottom; - - pDevice->SetFVF( D3DFVF_TEXTBOXVERTEX ); - pDevice->SetTexture( 0, NULL ); - pDevice->DrawPrimitiveUP( D3DPT_TRIANGLEFAN, 2, msgBox, sizeof(TEXTBOXVERTEX)); - - pFont->DrawText(NULL, theApp.screenMessageBuffer, -1, msgRect, DT_CENTER | DT_VCENTER, 0x7fff0000); - } - else theApp.screenMessage = false; - } - - - // Draw the speed - if( (theApp.videoOption > VIDEO_4X) && theApp.showSpeed ) - { - // Create the string text - char buffer[30]; - if(theApp.showSpeed == TRUE) - sprintf(buffer, "%3d%%", systemSpeed); - else - sprintf(buffer, "%3d%%(%d, %d fps)", systemSpeed, - systemFrameSkip, - theApp.showRenderedFrames); - - //Draw the string - D3DCOLOR speedColor; - theApp.showSpeedTransparent==TRUE ? speedColor = 0x7fffffff : speedColor = 0xffffffff; - - CRect speedRect( - 64, - 0, - dpp.BackBufferWidth - 64, - 32); - - pFont->DrawText(NULL, buffer, -1, speedRect, DT_CENTER | DT_VCENTER, speedColor); - } - - pDevice->EndScene(); - - pDevice->Present( NULL, NULL, NULL, NULL ); //Draw everything to the screen - } -} - - -void Direct3DDisplay::invalidateDeviceObjects() -{ - - if(pFont) { - pFont->Release(); - pFont = NULL; - } -} - -void Direct3DDisplay::restoreDeviceObjects() -{ - // Create the font - D3DXCreateFont( pDevice, 24, 0, FW_BOLD, 1, FALSE, ANSI_CHARSET, OUT_DEFAULT_PRECIS, ANTIALIASED_QUALITY, FF_DONTCARE, "Arial", &pFont ); - - // Set texture filter - updateFiltering(theApp.d3dFilter); - - // Set device settings - pDevice->SetRenderState( D3DRS_ZENABLE, D3DZB_FALSE ); - pDevice->SetRenderState( D3DRS_ALPHABLENDENABLE, TRUE ); - pDevice->SetRenderState( D3DRS_SRCBLEND, D3DBLEND_SRCALPHA ); - pDevice->SetRenderState( D3DRS_DESTBLEND, D3DBLEND_INVSRCALPHA ); - pDevice->SetTextureStageState(0, D3DTSS_ALPHAOP, D3DTOP_MODULATE); - if (theApp.menuToggle) - pDevice->SetDialogBoxMode( TRUE ); - pDevice->SetRenderState( D3DRS_LIGHTING, FALSE ); - - // Set matrices - initializeMatrices(); -} - - -void Direct3DDisplay::resize(int w, int h) -{ // interface function - if(!initSucessful) return; - - if ( (w>0) && (h>0) ) { - if(pDevice) - { - dpp.BackBufferWidth = w; - dpp.BackBufferHeight = h; - setPresentationType(); - if (resetDevice()) { - doNotRender = false; - } - } - } else { - doNotRender = true; - } - -} - - -bool Direct3DDisplay::changeRenderSize(int w, int h) -{ // interface function - if(!initSucessful) return false; - - // w and h is the size of the filtered image (So this could be 3xGBASize) - if(w != width || h != height) - { - if(pTexture) { - pTexture->Release(); - pTexture = NULL; - } - - if(!initializeOffscreen(w, h)) { - return false; - } - } - - if(filterDisabled && theApp.filterFunction) - theApp.filterFunction = NULL; - - // Set up 2D matrices - initializeMatrices(); - - return true; -} - -void Direct3DDisplay::setOption(const char *option, int value) -{ // interface function - if(!initSucessful) return; - - if(!strcmp(option, "d3dFilter")) - updateFiltering(theApp.d3dFilter); - - if(!strcmp(option, "vsync")) - { - if (pDevice) - { - if (theApp.vsync) - dpp.PresentationInterval = D3DPRESENT_INTERVAL_ONE; // VSync - else - dpp.PresentationInterval = D3DPRESENT_INTERVAL_IMMEDIATE; // No Sync - resetDevice(); - } - } - - if(!strcmp(option, "triplebuffering")) - { - if (theApp.tripleBuffering) - dpp.BackBufferCount = 3; - else - dpp.BackBufferCount = 2; - resetDevice(); - } - - if(!strcmp(option, "d3dKeepAspectRatio")) - keepAspectRatio = true; //theApp.d3dKeepAspectRatio; -} - -int Direct3DDisplay::selectFullScreenMode(GUID **) -{ // interface function - int w, h, b; - UniVideoModeDlg dlg(0, &w, &h, &b, &SelectedFreq, &SelectedAdapter); - - if (0 == dlg.DoModal()) - { - return (b<<24) + (w<<12) + h; - // Bits<<24 | Width<<12 | Height - } - else - { - return -1; - } -} - -int Direct3DDisplay::selectFullScreenMode2() -{ - return (SelectedAdapter<<16) + SelectedFreq; -} - -// Reset Device and Resources -bool Direct3DDisplay::resetDevice() -{ - invalidateDeviceObjects(); - if(pTexture) { - pDevice->SetTexture( 0, NULL); - pTexture->Release(); - pTexture = NULL; - } - if (!theApp.menuToggle) - pDevice->SetDialogBoxMode( FALSE ); - - HRESULT hr = pDevice->Reset(&dpp); - if (hr == D3D_OK) - { - restoreDeviceObjects(); - if(!initializeOffscreen(width, height)) - return false; - - return true; - } - else - { - switch(hr) - { - case D3DERR_DEVICELOST: - winlog("Render_DeviceLost: D3DERR_DEVICELOST\n"); - break; - case D3DERR_DRIVERINTERNALERROR: - winlog("Render_DeviceLost: D3DERR_DRIVERINTERNALERROR\n"); - break; - case D3DERR_INVALIDCALL: - winlog("Render_DeviceLost: D3DERR_INVALIDCALL\n"); - break; - case D3DERR_OUTOFVIDEOMEMORY: - winlog("Render_DeviceLost: D3DERR_OUTOFVIDEOMEMORY\n"); - break; - case E_OUTOFMEMORY: - winlog("Render_DeviceLost: E_OUTOFMEMORY\n"); - break; - } - winlog("Failed to reset device: %08x\n", hr); - return false; - } -} - -void Direct3DDisplay::initializeMatrices() -{ // Configure matrices to use standard orthogonal projection (2D) - D3DXMATRIX Ortho2D; - D3DXMATRIX Identity; - D3DXMATRIX temp1, temp2; - - // Initialize an orthographic matrix which automaticly compensates the difference between image size and texture size - if (!keepAspectRatio) { - D3DXMatrixOrthoOffCenterLH( - &Ortho2D, - 0.0f, // left - 1.0f * (FLOAT)width / (FLOAT)textureWidth, // right - 1.0f * (FLOAT)height / (FLOAT)textureHeight, // bottom - 0.0f, // top - 0.0f, 1.0f); // z - } else { - FLOAT l=0.0f, r=0.0f, b=0.0f, t=0.0f; - FLOAT srcAspectRatio = (FLOAT)theApp.sizeX / (FLOAT)theApp.sizeY; - FLOAT aspectRatio = (FLOAT)dpp.BackBufferWidth / (FLOAT)dpp.BackBufferHeight; - FLOAT textureImageDiffX = (FLOAT)width / (FLOAT)textureWidth; - FLOAT textureImageDiffY = (FLOAT)height / (FLOAT)textureHeight; - aspectRatio /= srcAspectRatio; - - if(aspectRatio > 1.0f) { - r = 1.0f * textureImageDiffX * aspectRatio; - b = 1.0f * textureImageDiffY; - } else { - r = 1.0f * textureImageDiffX; - b = 1.0f * textureImageDiffY * (1.0f / aspectRatio); - } - - D3DXMatrixOrthoOffCenterLH( - &temp1, - l, - r, - b, - t, - 0.0f, 1.0f); // z - D3DXMatrixTranslation( // translate matrix > move image - &temp2, - (aspectRatio>1.0)?( (aspectRatio - 1.0f) * 0.5f * textureImageDiffX ):0.0f, - (aspectRatio<1.0)?( ((1.0f/aspectRatio) - 1.0f) * 0.5f * textureImageDiffY ):0.0f, - 0.0f); - D3DXMatrixMultiply(&Ortho2D, &temp2, &temp1); - } - - D3DXMatrixIdentity(&Identity); // Identity = Do not change anything - - pDevice->SetTransform(D3DTS_PROJECTION, &Ortho2D); - pDevice->SetTransform(D3DTS_WORLD, &Identity); - pDevice->SetTransform(D3DTS_VIEW, &Identity); -} - -IDisplay *newDirect3DDisplay() -{ - return new Direct3DDisplay(); -} \ No newline at end of file +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005-2006 VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#include "stdafx.h" + +#include "Display.h" + +#include "MainWnd.h" + +#include "../System.h" +#include "../GBA.h" +#include "../Globals.h" +#include "../Text.h" +#include "../Util.h" +#include "../gb/gbGlobals.h" + +// Direct3D +#define DIRECT3D_VERSION 0x0900 +#include // main include file +#include // required for font rednering +#include // contains debug functions + +extern int Init_2xSaI(u32); // initializes all pixel filters +extern int systemSpeed; + +#ifdef _DEBUG +#define new DEBUG_NEW +#undef THIS_FILE +static char THIS_FILE[] = __FILE__; +#endif + +#ifdef MMX +extern "C" bool cpu_mmx; +extern bool detectMMX(); +#endif + + +class Direct3DDisplay : public IDisplay { +private: + LPDIRECT3D9 pD3D; + LPDIRECT3DDEVICE9 pDevice; + D3DPRESENT_PARAMETERS dpp; + D3DFORMAT screenFormat; + LPDIRECT3DSURFACE9 emulatedImage; + D3DTEXTUREFILTERTYPE filter; + int width; + int height; + RECT destRect; + bool failed; + ID3DXFont *pFont; + + void createFont(); + void destroyFont(); + void createSurface(); + void destroySurface(); + void calculateDestRect(); + bool resetDevice(); + +public: + Direct3DDisplay(); + virtual ~Direct3DDisplay(); + virtual DISPLAY_TYPE getType() { return DIRECT_3D; }; + + virtual bool initialize(); + virtual void cleanup(); + virtual void clear(); + virtual void render(); + + virtual void renderMenu(); + virtual bool changeRenderSize( int w, int h ); + virtual void resize( int w, int h ); + virtual void setOption( const char *option, int value ); + virtual int selectFullScreenMode( GUID ** ); +}; + + +Direct3DDisplay::Direct3DDisplay() +{ + pD3D = NULL; + pDevice = NULL; + screenFormat = D3DFMT_X8R8G8B8; + width = 0; + height = 0; + failed = false; + pFont = NULL; + emulatedImage = NULL; + filter = D3DTEXF_POINT; +} + + +Direct3DDisplay::~Direct3DDisplay() +{ + cleanup(); +} + + +void Direct3DDisplay::cleanup() +{ + destroyFont(); + destroySurface(); + + if( pDevice ) { + pDevice->Release(); + pDevice = NULL; + } + + if( pD3D ) { + pD3D->Release(); + pD3D = NULL; + } +} + + +bool Direct3DDisplay::initialize() +{ + switch(theApp.cartridgeType) + { + case IMAGE_GBA: + theApp.sizeX = 240; + theApp.sizeY = 160; + break; + case IMAGE_GB: + if (gbBorderOn) + { + theApp.sizeX = 256; + theApp.sizeY = 224; + } + else + { + theApp.sizeX = 160; + theApp.sizeY = 144; + } + break; + } + + + switch(theApp.videoOption) + { + case VIDEO_1X: + theApp.surfaceSizeX = theApp.sizeX; + theApp.surfaceSizeY = theApp.sizeY; + break; + case VIDEO_2X: + theApp.surfaceSizeX = theApp.sizeX * 2; + theApp.surfaceSizeY = theApp.sizeY * 2; + break; + case VIDEO_3X: + theApp.surfaceSizeX = theApp.sizeX * 3; + theApp.surfaceSizeY = theApp.sizeY * 3; + break; + case VIDEO_4X: + theApp.surfaceSizeX = theApp.sizeX * 4; + theApp.surfaceSizeY = theApp.sizeY * 4; + break; + case VIDEO_320x240: + case VIDEO_640x480: + case VIDEO_800x600: + case VIDEO_1024x768: + case VIDEO_1280x1024: + case VIDEO_OTHER: + float scaleX = ((float)theApp.fsWidth / theApp.sizeX); + float scaleY = ((float)theApp.fsHeight / theApp.sizeY); + float min = (scaleX < scaleY) ? scaleX : scaleY; + if(theApp.fullScreenStretch) { + theApp.surfaceSizeX = theApp.fsWidth; + theApp.surfaceSizeY = theApp.fsHeight; + } else { + theApp.surfaceSizeX = (int)(theApp.sizeX * min); + theApp.surfaceSizeY = (int)(theApp.sizeY * min); + } + break; + } + + theApp.rect.left = 0; + theApp.rect.top = 0; + theApp.rect.right = theApp.sizeX; + theApp.rect.bottom = theApp.sizeY; + + theApp.dest.left = 0; + theApp.dest.top = 0; + theApp.dest.right = theApp.surfaceSizeX; + theApp.dest.bottom = theApp.surfaceSizeY; + + + DWORD style = WS_POPUP | WS_VISIBLE; + DWORD styleEx = 0; + + if(theApp.videoOption <= VIDEO_4X) { + style |= WS_OVERLAPPEDWINDOW; + } else { + styleEx = 0; + } + + if(theApp.videoOption <= VIDEO_4X) { + AdjustWindowRectEx(&theApp.dest, style, TRUE, styleEx); + } else { + AdjustWindowRectEx(&theApp.dest, style, FALSE, styleEx); + } + + int winSizeX = theApp.dest.right-theApp.dest.left; + int winSizeY = theApp.dest.bottom-theApp.dest.top; + + if(theApp.videoOption > VIDEO_4X) { + winSizeX = theApp.fsWidth; + winSizeY = theApp.fsHeight; + } + + int x = 0, y = 0; + + if(theApp.videoOption <= VIDEO_4X) { + x = theApp.windowPositionX; + y = theApp.windowPositionY; + } + + + // Create a window + MainWnd *pWnd = new MainWnd; + theApp.m_pMainWnd = pWnd; + + pWnd->CreateEx(styleEx, + theApp.wndClass, + _T("VisualBoyAdvance"), + style, + x,y,winSizeX,winSizeY, + NULL, + 0); + + if (!(HWND)*pWnd) { + DXTRACE_ERR_MSGBOX( _T("Error creating window"), 0 ); + return FALSE; + } + pWnd->DragAcceptFiles(TRUE); + theApp.updateMenuBar(); + theApp.adjustDestRect(); + + + // load Direct3D v9 + pD3D = Direct3DCreate9( D3D_SDK_VERSION ); + + if(pD3D == NULL) { + DXTRACE_ERR_MSGBOX( _T("Error creating Direct3D object"), 0 ); + return FALSE; + } + + theApp.mode320Available = FALSE; + theApp.mode640Available = FALSE; + theApp.mode800Available = FALSE; + + D3DDISPLAYMODE mode; + pD3D->GetAdapterDisplayMode(D3DADAPTER_DEFAULT, &mode); + screenFormat = mode.Format; + + switch(mode.Format) { + case D3DFMT_R8G8B8: + systemColorDepth = 24; + systemRedShift = 19; + systemGreenShift = 11; + systemBlueShift = 3; + break; + case D3DFMT_X8R8G8B8: + systemColorDepth = 32; + systemRedShift = 19; + systemGreenShift = 11; + systemBlueShift = 3; + Init_2xSaI(32); + break; + case D3DFMT_R5G6B5: + systemColorDepth = 16; + systemRedShift = 11; + systemGreenShift = 6; + systemBlueShift = 0; + Init_2xSaI(565); + break; + case D3DFMT_X1R5G5B5: + systemColorDepth = 16; + systemRedShift = 10; + systemGreenShift = 5; + systemBlueShift = 0; + Init_2xSaI(555); + break; + default: + DXTRACE_ERR_MSGBOX( _T("Unsupport D3D format"), 0 ); + return false; + } + theApp.fsColorDepth = systemColorDepth; + utilUpdateSystemColorMaps(); + + +#ifdef MMX + if(!theApp.disableMMX) { + cpu_mmx = theApp.detectMMX(); + } else { + cpu_mmx = 0; + } +#endif + + + theApp.updateFilter(); + theApp.updateIFB(); + + + // create device + ZeroMemory(&dpp, sizeof(dpp)); + dpp.Windowed = TRUE; + dpp.BackBufferFormat = mode.Format; + dpp.BackBufferCount = theApp.tripleBuffering ? 2 : 1; + dpp.SwapEffect = D3DSWAPEFFECT_DISCARD; + dpp.BackBufferWidth = 0; // use width of hDeviceWindow + dpp.BackBufferHeight = 0; // use height of hDeviceWindow + dpp.Flags = D3DPRESENTFLAG_LOCKABLE_BACKBUFFER; + dpp.PresentationInterval = theApp.vsync ? D3DPRESENT_INTERVAL_ONE : D3DPRESENT_INTERVAL_IMMEDIATE; + + HRESULT hret = pD3D->CreateDevice( + D3DADAPTER_DEFAULT, + D3DDEVTYPE_HAL, + pWnd->GetSafeHwnd(), + D3DCREATE_SOFTWARE_VERTEXPROCESSING, + &dpp, + &pDevice); + if( FAILED( hret ) ) { + DXTRACE_ERR_MSGBOX( _T("Error creating Direct3D device"), hret ); + return false; + } + + createFont(); + createSurface(); + calculateDestRect(); + + setOption( _T("d3dFilter"), theApp.d3dFilter ); + + if(failed) return false; + + return TRUE; +} + + +void Direct3DDisplay::renderMenu() +{ + checkFullScreen(); + if(theApp.m_pMainWnd) { + theApp.m_pMainWnd->DrawMenuBar(); + } +} + + +void Direct3DDisplay::clear() +{ + if( pDevice ) { + pDevice->Clear( 0, NULL, D3DCLEAR_TARGET, D3DCOLOR_XRGB(0x00,0x00,0x00), 0.0f, 0 ); + } +} + + +void Direct3DDisplay::render() +{ + if( failed ) return; + if(!pDevice) return; + if( FAILED( pDevice->TestCooperativeLevel() ) ) return; + + clear(); + + pDevice->BeginScene(); + + // copy pix to emulatedImage and apply pixel filter if selected + HRESULT hr; + D3DLOCKED_RECT lr; + if( FAILED( hr = emulatedImage->LockRect( &lr, NULL, D3DLOCK_DISCARD ) ) ) { + DXTRACE_ERR_MSGBOX( _T("Can not lock back buffer"), hr ); + return; + } else { + if( !theApp.filterFunction ) { + copyImage( pix, lr.pBits, theApp.sizeX, theApp.sizeY, lr.Pitch, systemColorDepth ); + } else { + u32 pitch = theApp.filterWidth * (systemColorDepth>>3) + 4; + theApp.filterFunction( pix + pitch, + pitch, + (u8*)theApp.delta, + (u8*)lr.pBits, + lr.Pitch, + theApp.filterWidth, + theApp.filterHeight); + } + emulatedImage->UnlockRect(); + } + + // copy emulatedImage to pBackBuffer and scale with or without aspect ratio + LPDIRECT3DSURFACE9 pBackBuffer; + pDevice->GetBackBuffer( 0, 0, D3DBACKBUFFER_TYPE_MONO, &pBackBuffer ); + if( theApp.fullScreenStretch ) { + pDevice->StretchRect( emulatedImage, NULL, pBackBuffer, NULL, filter ); + } else { + pDevice->StretchRect( emulatedImage, NULL, pBackBuffer, &destRect, filter ); + } + pBackBuffer->Release(); + pBackBuffer = NULL; + + D3DCOLOR color; + RECT r; + r.left = 4; + r.right = dpp.BackBufferWidth - 4; + + if( theApp.screenMessage ) { + color = theApp.showSpeedTransparent ? D3DCOLOR_ARGB(0x7F, 0xFF, 0x00, 0x00) : D3DCOLOR_ARGB(0xFF, 0xFF, 0x00, 0x00); + if( ( ( GetTickCount() - theApp.screenMessageTime ) < 3000 ) && !theApp.disableStatusMessage && pFont ) { + r.top = dpp.BackBufferHeight - 20; + r.bottom = dpp.BackBufferHeight - 4; + pFont->DrawText( NULL, theApp.screenMessageBuffer, -1, &r, 0, color ); + } else { + theApp.screenMessage = false; + } + } + + if( theApp.showSpeed && ( theApp.videoOption > VIDEO_4X ) ) { + color = theApp.showSpeedTransparent ? D3DCOLOR_ARGB(0x7F, 0x00, 0x00, 0xFF) : D3DCOLOR_ARGB(0xFF, 0x00, 0x00, 0xFF); + char buffer[30]; + if( theApp.showSpeed == 1 ) { + sprintf( buffer, "%3d%%", systemSpeed ); + } else { + sprintf( buffer, "%3d%%(%d, %d fps)", systemSpeed, systemFrameSkip, theApp.showRenderedFrames ); + } + + r.top = 4; + r.bottom = 20; + pFont->DrawText( NULL, buffer, -1, &r, 0, color ); + } + + pDevice->EndScene(); + + pDevice->Present( NULL, NULL, NULL, NULL ); + + return; +} + + +bool Direct3DDisplay::changeRenderSize( int w, int h ) +{ + if( (w != width) || (h != height) ) { + width = w; height = h; + if( pDevice ) { + destroySurface(); + createSurface(); + calculateDestRect(); + } + } + return true; +} + + +void Direct3DDisplay::resize( int w, int h ) +{ + if( (w != dpp.BackBufferWidth) || (h != dpp.BackBufferHeight) ) { + dpp.BackBufferWidth = (UINT)w; + dpp.BackBufferHeight = (UINT)h; + resetDevice(); + calculateDestRect(); + } +} + + +int Direct3DDisplay::selectFullScreenMode( GUID ** ) +{ + HRESULT hr; + D3DDISPLAYMODE dm; + if( FAILED( hr = pDevice->GetDisplayMode( 0, &dm ) ) ) { + DXTRACE_ERR_MSGBOX( _T("pDevice->GetDisplayMode failed"), hr ); + return false; + } + + UINT bitsPerPixel; + switch( dm.Format ) + { + case D3DFMT_A2R10G10B10: + case D3DFMT_X8R8G8B8: + bitsPerPixel = 32; + break; + case D3DFMT_X1R5G5B5: + case D3DFMT_R5G6B5: + bitsPerPixel = 16; + break; + } + + return (bitsPerPixel << 24) | (dm.Width << 12) | dm.Height; +} + + +void Direct3DDisplay::createFont() +{ + if( !pFont ) { + HRESULT hr = D3DXCreateFont( + pDevice, + 14, + 0, + FW_BOLD, + 1, + FALSE, + DEFAULT_CHARSET, + OUT_DEFAULT_PRECIS, + DEFAULT_QUALITY, + DEFAULT_PITCH || FF_DONTCARE, + _T("Arial"), + &pFont ); + if( FAILED( hr ) ) { + DXTRACE_ERR_MSGBOX( _T("createFont failed"), hr ); + } + } +} + + +void Direct3DDisplay::destroyFont() +{ + if( pFont ) { + pFont->Release(); + pFont = NULL; + } +} + + +void Direct3DDisplay::createSurface() +{ + if( !emulatedImage ) { + HRESULT hr = pDevice->CreateOffscreenPlainSurface( + width, height, + dpp.BackBufferFormat, + D3DPOOL_DEFAULT, + &emulatedImage, + NULL ); + if( FAILED( hr ) ) { + DXTRACE_ERR_MSGBOX( _T("createSurface failed"), hr ); + } + } +} + + +void Direct3DDisplay::destroySurface() +{ + if( emulatedImage ) { + emulatedImage->Release(); + emulatedImage = NULL; + } +} + + +void Direct3DDisplay::calculateDestRect() +{ + float scaleX = (float)dpp.BackBufferWidth / (float)width; + float scaleY = (float)dpp.BackBufferHeight / (float)height; + float min = (scaleX < scaleY) ? scaleX : scaleY; + if( theApp.fsMaxScale && (min > theApp.fsMaxScale) ) { + min = (float)theApp.fsMaxScale; + } + destRect.left = 0; + destRect.top = 0; + destRect.right = (LONG)(width * min); + destRect.bottom = (LONG)(height * min); + if( destRect.right != dpp.BackBufferWidth ) { + LONG diff = (dpp.BackBufferWidth - destRect.right) / 2; + destRect.left += diff; + destRect.right += diff; + } + if( destRect.bottom != dpp.BackBufferHeight ) { + LONG diff = (dpp.BackBufferHeight - destRect.bottom) / 2; + destRect.top += diff; + destRect.bottom += diff; + } +} + + +void Direct3DDisplay::setOption( const char *option, int value ) +{ + if( !_tcscmp( option, _T("vsync") ) ) { + dpp.PresentationInterval = value ? D3DPRESENT_INTERVAL_ONE : D3DPRESENT_INTERVAL_IMMEDIATE; + resetDevice(); + } + + if( !_tcscmp( option, _T("tripleBuffering") ) ) { + dpp.BackBufferCount = value ? 2 : 1; + resetDevice(); + } + + if( !_tcscmp( option, _T("d3dFilter") ) ) { + switch( value ) + { + case 0: //point + filter = D3DTEXF_POINT; + break; + case 1: //linear + filter = D3DTEXF_LINEAR; + break; + } + } + + if( !_tcscmp( option, _T("maxScale") ) ) { + calculateDestRect(); + } +} + + +bool Direct3DDisplay::resetDevice() +{ + if( !pDevice ) return false; + + HRESULT hr; + destroyFont(); + destroySurface(); + if( FAILED( hr = pDevice->Reset( &dpp ) ) ) { + //DXTRACE_ERR_MSGBOX( _T("pDevice->Reset failed"), hr ); + failed = true; + return false; + } + createFont(); + createSurface(); + failed = false; + return true; +} + + +IDisplay *newDirect3DDisplay() +{ + return new Direct3DDisplay(); +} diff --git a/src/win32/DirectDraw.cpp b/src/win32/DirectDraw.cpp index 67d4f48e..0a4797e5 100644 --- a/src/win32/DirectDraw.cpp +++ b/src/win32/DirectDraw.cpp @@ -21,8 +21,6 @@ #define DIRECTDRAW_VERSION 0x0700 #include -#include - #include "../System.h" #include "../gb/gbGlobals.h" #include "../GBA.h" @@ -33,7 +31,9 @@ #include "VBA.h" #include "MainWnd.h" #include "Reg.h" -#include "..\..\res\resource.h" +#include "resource.h" + +#include "Display.h" #ifdef _DEBUG #define new DEBUG_NEW @@ -58,18 +58,7 @@ private: int height; bool failed; - volatile unsigned wait_lastscanline; - volatile unsigned wait_screenheight; - volatile unsigned wait_maxheight; - volatile unsigned wait_firstline; - HANDLE wait_event; - UINT wait_timerres; - UINT wait_timerid; - bool initializeOffscreen(int w, int h); - bool StartTimer(); - void StopTimer(); - static void CALLBACK g_timer_proc( UINT id, UINT msg, DWORD_PTR dwUser, DWORD_PTR dw1, DWORD_PTR dw2 ); public: DirectDrawDisplay(); virtual ~DirectDrawDisplay(); @@ -83,9 +72,8 @@ public: virtual bool changeRenderSize(int w, int h); virtual DISPLAY_TYPE getType() { return DIRECT_DRAW; }; virtual void setOption(const char *, int) {} + virtual bool isSkinSupported() { return true; } virtual int selectFullScreenMode(GUID **); - - void timer_proc( UINT id, UINT msg, DWORD_PTR dw1, DWORD_PTR dw2 ); }; static HRESULT WINAPI checkModesAvailable(LPDDSURFACEDESC2 surf, LPVOID lpContext) @@ -144,10 +132,6 @@ DirectDrawDisplay::DirectDrawDisplay() width = 0; height = 0; failed = false; - wait_screenheight = 0; - wait_event = 0; - wait_timerres = 0; - wait_timerid = 0; } DirectDrawDisplay::~DirectDrawDisplay() @@ -158,15 +142,6 @@ DirectDrawDisplay::~DirectDrawDisplay() void DirectDrawDisplay::cleanup() { if(pDirectDraw != NULL) { - if ( wait_timerid ) { - StopTimer(); - } - - if ( wait_event ) { - CloseHandle( wait_event ); - wait_event = 0; - } - if(ddsClipper != NULL) { ddsClipper->Release(); ddsClipper = NULL; @@ -192,7 +167,11 @@ void DirectDrawDisplay::cleanup() } if(ddrawDLL != NULL) { - FreeLibrary(ddrawDLL); +#ifdef _AFXDLL + AfxFreeLibrary( ddrawDLL ); +#else + FreeLibrary( ddrawDLL ); +#endif ddrawDLL = NULL; } width = 0; @@ -228,13 +207,13 @@ bool DirectDrawDisplay::initialize() case VIDEO_1280x1024: case VIDEO_OTHER: { - float scaleX = ((float)theApp.fsWidth / (float)theApp.sizeX); - float scaleY = ((float)theApp.fsHeight / (float)theApp.sizeY); - float min = scaleX < scaleY ? scaleX : scaleY; + int scaleX = (theApp.fsWidth / theApp.sizeX); + int scaleY = (theApp.fsHeight / theApp.sizeY); + int min = scaleX < scaleY ? scaleX : scaleY; if(theApp.fsMaxScale) min = min > theApp.fsMaxScale ? theApp.fsMaxScale : min; - theApp.surfaceSizeX = (int)(theApp.sizeX * min); - theApp.surfaceSizeY = (int)(theApp.sizeY * min); + theApp.surfaceSizeX = theApp.sizeX * min; + theApp.surfaceSizeY = theApp.sizeY * min; if(theApp.fullScreenStretch) { theApp.surfaceSizeX = theApp.fsWidth; theApp.surfaceSizeY = theApp.fsHeight; @@ -242,7 +221,7 @@ bool DirectDrawDisplay::initialize() } break; } - + theApp.rect.left = 0; theApp.rect.top = 0; theApp.rect.right = theApp.sizeX; @@ -306,7 +285,12 @@ bool DirectDrawDisplay::initialize() if(theApp.pVideoDriverGUID) guid = theApp.pVideoDriverGUID; - ddrawDLL = LoadLibrary("DDRAW.DLL"); +#ifdef _AFXDLL + ddrawDLL = AfxLoadLibrary("ddraw.dll"); +#else + ddrawDLL = LoadLibrary( _T("ddraw.dll") ); +#endif + HRESULT (WINAPI *DDrawCreateEx)(GUID *,LPVOID *,REFIID,IUnknown *); if(ddrawDLL != NULL) { DDrawCreateEx = (HRESULT (WINAPI *)(GUID *,LPVOID *,REFIID,IUnknown *)) @@ -459,12 +443,6 @@ bool DirectDrawDisplay::initialize() } // } - wait_event = CreateEvent( NULL, FALSE, FALSE, NULL ); - - if ( ! StartTimer() ) { - return FALSE; - } - DDPIXELFORMAT px; px.dwSize = sizeof(px); @@ -633,7 +611,7 @@ bool DirectDrawDisplay::initializeOffscreen(int w, int h) winlog("B shift: %d\n", systemBlueShift); } - utilUpdateSystemColorMaps(theApp.filterLCD); + utilUpdateSystemColorMaps(); width = w; height = h; return true; @@ -671,6 +649,7 @@ void DirectDrawDisplay::checkFullScreen() void DirectDrawDisplay::render() { HRESULT hret; + unsigned int nBytesPerPixel = systemColorDepth>>3; if(pDirectDraw == NULL || ddsOffscreen == NULL || @@ -743,62 +722,9 @@ void DirectDrawDisplay::render() copyY = 144; } } - // MMX doesn't seem to be faster to copy the data - __asm { - mov eax, copyX; - mov ebx, copyY; - - mov esi, pix; - mov edi, ddsDesc.lpSurface; - mov edx, ddsDesc.lPitch; - cmp systemColorDepth, 16; - jnz gbaOtherColor; - sub edx, eax; - sub edx, eax; - lea esi,[esi+2*eax+4]; - shr eax, 1; - gbaLoop16bit: - mov ecx, eax; - repz movsd; - inc esi; - inc esi; - inc esi; - inc esi; - add edi, edx; - dec ebx; - jnz gbaLoop16bit; - jmp gbaLoopEnd; - gbaOtherColor: - cmp systemColorDepth, 32; - jnz gbaOtherColor2; - - sub edx, eax; - sub edx, eax; - sub edx, eax; - sub edx, eax; - lea esi, [esi+4*eax+4]; - gbaLoop32bit: - mov ecx, eax; - repz movsd; - add esi, 4; - add edi, edx; - dec ebx; - jnz gbaLoop32bit; - jmp gbaLoopEnd; - gbaOtherColor2: - lea eax, [eax+2*eax]; - sub edx, eax; - gbaLoop24bit: - mov ecx, eax; - shr ecx, 2; - repz movsd; - add edi, edx; - dec ebx; - jnz gbaLoop24bit; - gbaLoopEnd: - } + copyImage( pix, ddsDesc.lpSurface, copyX, copyY, ddsDesc.lPitch, systemColorDepth ); } - if(theApp.showSpeed && theApp.videoOption > VIDEO_4X) { + if(theApp.showSpeed && (theApp.videoOption > VIDEO_4X || theApp.skin != NULL)) { char buffer[30]; if(theApp.showSpeed == 1) sprintf(buffer, "%3d%%", systemSpeed); @@ -826,8 +752,7 @@ void DirectDrawDisplay::render() if(hret == DD_OK) { if(theApp.vsync && !(theApp.tripleBuffering && theApp.videoOption > VIDEO_4X) && !speedup) { // isn't the Flip() call synced unless a certain flag is passed to it? - //hret = pDirectDraw->WaitForVerticalBlank(DDWAITVB_BLOCKBEGIN, 0); - WaitForSingleObject( wait_event, 100 ); + hret = pDirectDraw->WaitForVerticalBlank(DDWAITVB_BLOCKBEGIN, 0); } ddsOffscreen->PageLock(0); if(theApp.tripleBuffering && theApp.videoOption > VIDEO_4X) { @@ -865,7 +790,7 @@ void DirectDrawDisplay::render() SetTextColor(hdc, RGB(255,0,0)); SetBkMode(hdc,TRANSPARENT); TextOut(hdc, theApp.dest.left+10, theApp.dest.bottom - 20, theApp.screenMessageBuffer, - strlen(theApp.screenMessageBuffer)); + (int)_tcslen(theApp.screenMessageBuffer)); ddsPrimary->ReleaseDC(hdc); } else { theApp.screenMessage = false; @@ -883,74 +808,8 @@ int DirectDrawDisplay::selectFullScreenMode(GUID **pGUID) return winVideoModeSelect(theApp.m_pMainWnd, pGUID); } -bool DirectDrawDisplay::StartTimer() -{ - MMRESULT result; - - TIMECAPS tc; - - if ( TIMERR_NOERROR == timeGetDevCaps( & tc, sizeof( TIMECAPS ) ) ) { - wait_timerres = min( max( tc.wPeriodMin, 1 ), tc.wPeriodMax ); - timeBeginPeriod( wait_timerres ); - } else { - return false; - } - - result = timeSetEvent( wait_timerres, wait_timerres, & DirectDrawDisplay::g_timer_proc, ( DWORD_PTR ) this, TIME_PERIODIC ); - - if (NULL != result) { - wait_timerid = (UINT)result; - return true; - } - - return false; -} - -void DirectDrawDisplay::StopTimer() -{ - MMRESULT result; - - result = timeKillEvent( wait_timerid ); - if ( TIMERR_NOERROR == result ) { - wait_timerid = 0; - } - - if ( 0 != wait_timerres ) { - timeEndPeriod( wait_timerres ); - wait_timerres = 0; - } -} - -void CALLBACK DirectDrawDisplay::g_timer_proc( UINT id, UINT msg, DWORD_PTR dwUser, DWORD_PTR dw1, DWORD_PTR dw2 ) -{ - DirectDrawDisplay * p_this = reinterpret_cast< DirectDrawDisplay * >( dwUser ); - if ( p_this ) { - p_this->timer_proc( id, msg, dw1, dw2 ); - } -} - -void DirectDrawDisplay::timer_proc( UINT id, UINT msg, DWORD_PTR dw1, DWORD_PTR dw2 ) -{ - DWORD scanline; - if ( pDirectDraw->GetScanLine( & scanline ) == DD_OK ) { - unsigned height = GetSystemMetrics( SM_CYSCREEN ); - if ( wait_screenheight != height ) { - wait_screenheight = height; - wait_maxheight = height; - } - if ( scanline >= wait_maxheight ) wait_maxheight = scanline + 1; - - scanline = ( scanline + wait_maxheight - min( theApp.dest.bottom, wait_screenheight ) ) % wait_maxheight; - - if ( scanline < wait_lastscanline ) { - PulseEvent( wait_event ); - } - - wait_lastscanline = scanline; - } -} - IDisplay *newDirectDrawDisplay() { return new DirectDrawDisplay(); } + diff --git a/src/win32/DirectInput.cpp b/src/win32/DirectInput.cpp index c280ced5..074559c2 100644 --- a/src/win32/DirectInput.cpp +++ b/src/win32/DirectInput.cpp @@ -1,594 +1,598 @@ -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#include "stdafx.h" -#include "Reg.h" -#include "WinResUtil.h" -#include "Input.h" - -#define DIRECTINPUT_VERSION 0x0500 -#include - -#ifdef _DEBUG -#define new DEBUG_NEW -#undef THIS_FILE -static char THIS_FILE[] = __FILE__; -#endif - -extern void directXMessage(const char *); -extern void winlog(const char *msg,...); - -#define POV_UP 1 -#define POV_DOWN 2 -#define POV_RIGHT 4 -#define POV_LEFT 8 - -class DirectInput : public Input { -private: - HINSTANCE dinputDLL; - -public: - virtual void checkDevices(); - DirectInput(); - virtual ~DirectInput(); - - virtual bool initialize(); - virtual bool readDevices(); - virtual u32 readDevice(int which); - virtual CString getKeyName(int key); - virtual void checkKeys(); - virtual void checkMotionKeys(); - virtual void activate(); - virtual void loadSettings(); - virtual void saveSettings(); -}; - -struct deviceInfo { - LPDIRECTINPUTDEVICE device; - BOOL isPolled; - int nButtons; - int nAxes; - int nPovs; - BOOL first; - struct { - DWORD offset; - LONG center; - LONG negative; - LONG positive; - } axis[8]; - int needed; - union { - UCHAR data[256]; - DIJOYSTATE state; - }; -}; - -static deviceInfo *currentDevice = NULL; -static int numDevices = 1; -static deviceInfo *pDevices = NULL; -static LPDIRECTINPUT pDirectInput = NULL; -static int joyDebug = 0; -static int axisNumber = 0; - - -//KeyList joypad[JOYPADS * KEYS_PER_PAD + MOTION_KEYS]; - - -USHORT defvalues[JOYPADS * KEYS_PER_PAD + MOTION_KEYS] = - { - DIK_LEFT, DIK_RIGHT, - DIK_UP, DIK_DOWN, - DIK_Z, DIK_X, - DIK_RETURN,DIK_BACK, - DIK_A, DIK_S, - DIK_SPACE, DIK_F12, - DIK_C, - 0,0,0,0,0,0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0,0,0,0,0,0, - 0,0,0,0,0,0,0,0,0,0,0,0,0, - DIK_NUMPAD4, DIK_NUMPAD6, DIK_NUMPAD8, DIK_NUMPAD2 -}; - - -void winReadKey(const char *name, KeyList& Keys) -{ - CString TxtKeyList = regQueryStringValue(name, ""); - int curPos= 0; - - CString resToken=TxtKeyList.Tokenize(",",curPos); - while (resToken != "") - { - Keys.AddTail(atoi(resToken)); - resToken= TxtKeyList.Tokenize(",",curPos); - }; -} - -void winReadKey(const char *name, int num, KeyList& Keys) -{ - char buffer[80]; - - sprintf(buffer, "Joy%d_%s", num, name); - winReadKey(buffer, Keys); -} - - -void winReadKeys() -{ - - for(int i = 0; i < JOYPADS; i++) { - winReadKey("Left", i, joypad[JOYPAD(i,KEY_LEFT)]); - winReadKey("Right", i, joypad[JOYPAD(i, KEY_RIGHT)]); - winReadKey("Up", i, joypad[JOYPAD(i,KEY_UP)]); - winReadKey("Down", i, joypad[JOYPAD(i,KEY_DOWN)]); - winReadKey("A", i, joypad[JOYPAD(i,KEY_BUTTON_A)]); - winReadKey("B", i, joypad[JOYPAD(i,KEY_BUTTON_B)]); - winReadKey("L", i, joypad[JOYPAD(i,KEY_BUTTON_L)]); - winReadKey("R", i, joypad[JOYPAD(i,KEY_BUTTON_R)]); - winReadKey("Start", i, joypad[JOYPAD(i,KEY_BUTTON_START)]); - winReadKey("Select", i, joypad[JOYPAD(i,KEY_BUTTON_SELECT)]); - winReadKey("Speed", i, joypad[JOYPAD(i,KEY_BUTTON_SPEED)]); - winReadKey("Capture", i, joypad[JOYPAD(i,KEY_BUTTON_CAPTURE)]); - winReadKey("GS", i, joypad[JOYPAD(i,KEY_BUTTON_GS)]); - } - winReadKey("Motion_Left", joypad[MOTION(KEY_LEFT)]); - winReadKey("Motion_Right", joypad[MOTION(KEY_RIGHT)]); - winReadKey("Motion_Up", joypad[MOTION(KEY_UP)]); - winReadKey("Motion_Down", joypad[MOTION(KEY_DOWN)]); -} - -void winSaveKey(char *name, KeyList& value) -{ - CString txtKeys; - - POSITION p = value.GetHeadPosition(); - while(p!=NULL) - { - CString tmp; - tmp.Format("%d", value.GetNext(p)); - txtKeys+=tmp; - if (p!=NULL) - txtKeys+=","; - } - regSetStringValue(name, txtKeys); -} - -static void winSaveKey(char *name, int num, KeyList& value) -{ - char buffer[80]; - - sprintf(buffer, "Joy%d_%s", num, name); - winSaveKey(buffer, value); -} - -void winSaveKeys() -{ - for(int i = 0; i < JOYPADS; i++) { - winSaveKey("Left", i, joypad[JOYPAD(i,KEY_LEFT)]); - winSaveKey("Right", i, joypad[JOYPAD(i,KEY_RIGHT)]); - winSaveKey("Up", i, joypad[JOYPAD(i,KEY_UP)]); - winSaveKey("Speed", i, joypad[JOYPAD(i,KEY_BUTTON_SPEED)]); - winSaveKey("Capture", i, joypad[JOYPAD(i,KEY_BUTTON_CAPTURE)]); - winSaveKey("GS", i, joypad[JOYPAD(i,KEY_BUTTON_GS)]); - winSaveKey("Down", i, joypad[JOYPAD(i,KEY_DOWN)]); - winSaveKey("A", i, joypad[JOYPAD(i,KEY_BUTTON_A)]); - winSaveKey("B", i, joypad[JOYPAD(i,KEY_BUTTON_B)]); - winSaveKey("L", i, joypad[JOYPAD(i,KEY_BUTTON_L)]); - winSaveKey("R", i, joypad[JOYPAD(i,KEY_BUTTON_R)]); - winSaveKey("Start", i, joypad[JOYPAD(i,KEY_BUTTON_START)]); - winSaveKey("Select", i, joypad[JOYPAD(i,KEY_BUTTON_SELECT)]); - } - regSetDwordValue("joyVersion", 1); - - winSaveKey("Motion_Left", - joypad[MOTION(KEY_LEFT)]); - winSaveKey("Motion_Right", - joypad[MOTION(KEY_RIGHT)]); - winSaveKey("Motion_Up", - joypad[MOTION(KEY_UP)]); - winSaveKey("Motion_Down", - joypad[MOTION(KEY_DOWN)]); -} - -static BOOL CALLBACK EnumAxesCallback( const DIDEVICEOBJECTINSTANCE* pdidoi, - VOID* pContext ) -{ - DIPROPRANGE diprg; - diprg.diph.dwSize = sizeof(DIPROPRANGE); - diprg.diph.dwHeaderSize = sizeof(DIPROPHEADER); - diprg.diph.dwHow = DIPH_BYOFFSET; - diprg.diph.dwObj = pdidoi->dwOfs; // Specify the enumerated axis - - diprg.lMin = -32768; - diprg.lMax = 32767; - // try to set the range - if(FAILED(currentDevice->device->SetProperty(DIPROP_RANGE, &diprg.diph))) { - // Get the range for the axis - if( FAILED(currentDevice->device-> - GetProperty( DIPROP_RANGE, &diprg.diph ) ) ) { - return DIENUM_STOP; - } - } - - DIPROPDWORD didz; - - didz.diph.dwSize = sizeof(didz); - didz.diph.dwHeaderSize = sizeof(DIPROPHEADER); - didz.diph.dwHow = DIPH_BYOFFSET; - didz.diph.dwObj = pdidoi->dwOfs; - - didz.dwData = 5000; - - currentDevice->device->SetProperty(DIPROP_DEADZONE, &didz.diph); - - LONG center = (diprg.lMin + diprg.lMax)/2; - LONG threshold = (diprg.lMax - center)/2; - - // only 8 axis supported - if(axisNumber < 8) { - currentDevice->axis[axisNumber].center = center; - currentDevice->axis[axisNumber].negative = center - threshold; - currentDevice->axis[axisNumber].positive = center + threshold; - currentDevice->axis[axisNumber].offset = pdidoi->dwOfs; - } - axisNumber++; - return DIENUM_CONTINUE; -} - -static BOOL CALLBACK EnumPovsCallback( const DIDEVICEOBJECTINSTANCE* pdidoi, - VOID* pContext ) -{ - return DIENUM_CONTINUE; -} - -static BOOL CALLBACK DIEnumDevicesCallback(LPCDIDEVICEINSTANCE pInst, - LPVOID lpvContext) -{ - ZeroMemory(&pDevices[numDevices],sizeof(deviceInfo)); - - HRESULT hRet = pDirectInput->CreateDevice(pInst->guidInstance, - &pDevices[numDevices].device, - NULL); - - if(hRet != DI_OK) - return DIENUM_STOP; - - DIDEVCAPS caps; - caps.dwSize=sizeof(DIDEVCAPS); - - hRet = pDevices[numDevices].device->GetCapabilities(&caps); - - if(hRet == DI_OK) { - if(caps.dwFlags & DIDC_POLLEDDATAFORMAT || - caps.dwFlags & DIDC_POLLEDDEVICE) - pDevices[numDevices].isPolled = TRUE; - - pDevices[numDevices].nButtons = caps.dwButtons; - pDevices[numDevices].nAxes = caps.dwAxes; - pDevices[numDevices].nPovs = caps.dwPOVs; - - for(int i = 0; i < 6; i++) { - pDevices[numDevices].axis[i].center = 0x8000; - pDevices[numDevices].axis[i].negative = 0x4000; - pDevices[numDevices].axis[i].positive = 0xc000; - } - } else if(joyDebug) - winlog("Failed to get device capabilities %08x\n", hRet); - - if(joyDebug) { - // don't translate. debug only - winlog("******************************\n"); - winlog("Joystick %2d name : %s\n", numDevices, pInst->tszProductName); - } - - numDevices++; - - - return DIENUM_CONTINUE; -} - -BOOL CALLBACK DIEnumDevicesCallback2(LPCDIDEVICEINSTANCE pInst, - LPVOID lpvContext) -{ - numDevices++; - - return DIENUM_CONTINUE; -} - -static int getPovState(DWORD value) -{ - int state = 0; - if(LOWORD(value) != 0xFFFF) { - if(value < 9000 || value > 27000) - state |= POV_UP; - if(value > 0 && value < 18000) - state |= POV_RIGHT; - if(value > 9000 && value < 27000) - state |= POV_DOWN; - if(value > 18000) - state |= POV_LEFT; - } - return state; -} - -static void checkKeys() -{ - int dev = 0; - int i; - - for(i = 0; i < (sizeof(joypad) / sizeof(joypad[0])); i++) - { - if (joypad[i].IsEmpty() && defvalues[i]) - joypad[i].AddTail(defvalues[i]); - POSITION p = joypad[i].GetHeadPosition(); - while(p!=NULL) - { - USHORT k = joypad[i].GetNext(p); - if (k > 0 && DEVICEOF(k) < numDevices) - pDevices[DEVICEOF(k)].needed = true; - } - } -} - -#define KEYDOWN(buffer,key) (buffer[key] & 0x80) - -static bool readKeyboard() -{ - if(pDevices[0].needed) { - if (!theApp.dinputKeyFocus) { - memset(pDevices[0].data, 0, 256); - return true; - } - HRESULT hret = pDevices[0].device-> - GetDeviceState(256, - (LPVOID)pDevices[0].data); - - if(hret == DIERR_INPUTLOST || hret == DIERR_NOTACQUIRED) { - hret = pDevices[0].device->Acquire(); - if(hret != DI_OK) - return false; - hret = pDevices[0].device->GetDeviceState(256,(LPVOID)pDevices[0].data); - } - - return hret == DI_OK; - } - return true; -} - -static bool readJoystick(int joy) -{ - if(pDevices[joy].needed) { - if(pDevices[joy].isPolled) - ((LPDIRECTINPUTDEVICE2)pDevices[joy].device)->Poll(); - - HRESULT hret = pDevices[joy].device-> - GetDeviceState(sizeof(DIJOYSTATE), - (LPVOID)&pDevices[joy].state); - - if(hret == DIERR_INPUTLOST || hret == DIERR_NOTACQUIRED) { - hret = pDevices[joy].device->Acquire(); - - if(hret == DI_OK) { - - if(pDevices[joy].isPolled) - ((LPDIRECTINPUTDEVICE2)pDevices[joy].device)->Poll(); - - hret = pDevices[joy].device-> - GetDeviceState(sizeof(DIJOYSTATE), - (LPVOID)&pDevices[joy].state); - } - } - - return hret == DI_OK; - } - - return true; -} - -static void checkKeyboard() -{ - HRESULT hret = pDevices[0].device->Acquire(); - hret = pDevices[0].device-> - GetDeviceState(256, - (LPVOID)pDevices[0].data); - - if(hret == DIERR_INPUTLOST || hret == DIERR_NOTACQUIRED) { - return; - } - - if(hret == DI_OK) { - for(int i = 0; i < 256; i++) { - if(KEYDOWN(pDevices[0].data, i)) { - SendMessage(GetFocus(), JOYCONFIG_MESSAGE,0,i); - break; - } - } - } -} - -static void checkJoypads() -{ - DIDEVICEOBJECTINSTANCE di; - - ZeroMemory(&di,sizeof(DIDEVICEOBJECTINSTANCE)); - - di.dwSize = sizeof(DIDEVICEOBJECTINSTANCE); - - int i =0; - - DIJOYSTATE joystick; - - for(i = 1; i < numDevices; i++) { - HRESULT hret = pDevices[i].device->Acquire(); - - - if(pDevices[i].isPolled) - ((LPDIRECTINPUTDEVICE2)pDevices[i].device)->Poll(); - - hret = pDevices[i].device->GetDeviceState(sizeof(joystick), &joystick); - - int j; - - if(pDevices[i].first) { - memcpy(&pDevices[i].state, &joystick, sizeof(joystick)); - pDevices[i].first = FALSE; - continue; - } - - for(j = 0; j < pDevices[i].nButtons; j++) { - if(((pDevices[i].state.rgbButtons[j] ^ joystick.rgbButtons[j]) - & joystick.rgbButtons[j]) & 0x80) { - HWND focus = GetFocus(); - - SendMessage(focus, JOYCONFIG_MESSAGE, i,j+128); - } - } - - for(j = 0; j < pDevices[i].nAxes && j < 8; j++) { - LONG value = pDevices[i].axis[j].center; - LONG old = 0; - switch(pDevices[i].axis[j].offset) { - case DIJOFS_X: - value = joystick.lX; - old = pDevices[i].state.lX; - break; - case DIJOFS_Y: - value = joystick.lY; - old = pDevices[i].state.lY; - break; - case DIJOFS_Z: - value = joystick.lZ; - old = pDevices[i].state.lZ; - break; - case DIJOFS_RX: - value = joystick.lRx; - old = pDevices[i].state.lRx; - break; - case DIJOFS_RY: - value = joystick.lRy; - old = pDevices[i].state.lRy; - break; - case DIJOFS_RZ: - value = joystick.lRz; - old = pDevices[i].state.lRz; - break; - case DIJOFS_SLIDER(0): - value = joystick.rglSlider[0]; - old = pDevices[i].state.rglSlider[0]; - break; - case DIJOFS_SLIDER(1): - value = joystick.rglSlider[1]; - old = pDevices[i].state.rglSlider[1]; - break; - } - if(value != old) { - if(value < pDevices[i].axis[j].negative) - SendMessage(GetFocus(), JOYCONFIG_MESSAGE, i, (j<<1)); - else if (value > pDevices[i].axis[j].positive) - SendMessage(GetFocus(), JOYCONFIG_MESSAGE, i, (j<<1)+1); - } - } - - for(j = 0;j < 4 && j < pDevices[i].nPovs; j++) { - if(LOWORD(pDevices[i].state.rgdwPOV[j]) != LOWORD(joystick.rgdwPOV[j])) { - int state = getPovState(joystick.rgdwPOV[j]); - - if(state & POV_UP) - SendMessage(GetFocus(), JOYCONFIG_MESSAGE, i, (j<<2)+0x20); - else if(state & POV_DOWN) - SendMessage(GetFocus(), JOYCONFIG_MESSAGE, i, (j<<2)+0x21); - else if(state & POV_RIGHT) - SendMessage(GetFocus(), JOYCONFIG_MESSAGE, i, (j<<2)+0x22); - else if(state & POV_LEFT) - SendMessage(GetFocus(), JOYCONFIG_MESSAGE, i, (j<<2)+0x23); - } - } - - memcpy(&pDevices[i].state, &joystick, sizeof(joystick)); - } -} - -BOOL checkKey(int key) -{ - int dev = DEVICEOF(key); - int k = KEYOF(key); - - if(dev == 0) { - return KEYDOWN(pDevices[0].data,k); - } else { - if(k < 16) { - int axis = k >> 1; - LONG value = pDevices[dev].axis[axis].center; - switch(pDevices[dev].axis[axis].offset) { - case DIJOFS_X: - value = pDevices[dev].state.lX; - break; - case DIJOFS_Y: - value = pDevices[dev].state.lY; - break; - case DIJOFS_Z: - value = pDevices[dev].state.lZ; - break; - case DIJOFS_RX: - value = pDevices[dev].state.lRx; - break; - case DIJOFS_RY: - value = pDevices[dev].state.lRy; - break; - case DIJOFS_RZ: - value = pDevices[dev].state.lRz; - break; - case DIJOFS_SLIDER(0): - value = pDevices[dev].state.rglSlider[0]; - break; - case DIJOFS_SLIDER(1): - value = pDevices[dev].state.rglSlider[1]; - break; - } - - if(k & 1) - return value > pDevices[dev].axis[axis].positive; - return value < pDevices[dev].axis[axis].negative; - } else if(k < 48) { - int hat = (k >> 2) & 3; - int state = getPovState(pDevices[dev].state.rgdwPOV[hat]); - BOOL res = FALSE; - switch(k & 3) { - case 0: - res = state & POV_UP; - break; - case 1: - res = state & POV_DOWN; - break; - case 2: - res = state & POV_RIGHT; - break; - case 3: - res = state & POV_LEFT; - break; - } - return res; - } else if(k >= 128) { - return pDevices[dev].state.rgbButtons[k-128] & 0x80; - } - } - - return FALSE; -} + +/* VisualBoyAdvance S - GB & GBA emulator + Copyright (C) 2006 Spacy + + Original VBA Credits: + Copyright (C) 1999-2003 Forgotten + Copyright (C) 2004-2006 VBA development team + + This program is free software; you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation; either version 2 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License along + with this program; if not, write to the Free Software Foundation, Inc., + 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. +*/ + +#include "stdafx.h" +#include "VBA.h" +#include "Input.h" +#include "Reg.h" +#include "WinResUtil.h" + +#define DIRECTINPUT_VERSION 0x0800 +#include +#pragma comment(lib, "Dinput8") + + +#ifdef _DEBUG +#define new DEBUG_NEW +#undef THIS_FILE +static char THIS_FILE[] = __FILE__; +#endif + +extern void directXMessage(const char *); +extern void winlog(const char *msg,...); + +#define POV_UP 1 +#define POV_DOWN 2 +#define POV_RIGHT 4 +#define POV_LEFT 8 + +class DirectInput : public Input { +public: + virtual void checkDevices(); + DirectInput(); + virtual ~DirectInput(); + + virtual bool initialize(); + virtual bool readDevices(); + virtual u32 readDevice(int which); + virtual CString getKeyName(LONG_PTR key); + virtual void checkKeys(); + virtual void checkMotionKeys(); + virtual void activate(); + virtual void loadSettings(); + virtual void saveSettings(); +}; + +struct deviceInfo { + LPDIRECTINPUTDEVICE8 device; + BOOL isPolled; + int nButtons; + int nAxes; + int nPovs; + BOOL first; + struct { + DWORD offset; + LONG center; + LONG negative; + LONG positive; + } axis[8]; + int needed; + union { + UCHAR data[256]; + DIJOYSTATE state; + }; +}; + +static deviceInfo *currentDevice = NULL; +static int numDevices = 1; +static deviceInfo *pDevices = NULL; +static LPDIRECTINPUT8 pDirectInput = NULL; +static int axisNumber = 0; + + + + +LONG_PTR defvalues[JOYPADS * KEYS_PER_PAD + MOTION_KEYS] = + { + DIK_LEFT, DIK_RIGHT, + DIK_UP, DIK_DOWN, + DIK_Z, DIK_X, + DIK_RETURN,DIK_BACK, + DIK_A, DIK_S, + DIK_SPACE, DIK_F12, + DIK_C, + 0,0,0,0,0,0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0,0,0,0,0,0, + 0,0,0,0,0,0,0,0,0,0,0,0,0, + DIK_NUMPAD4, DIK_NUMPAD6, DIK_NUMPAD8, DIK_NUMPAD2 +}; + + +void winReadKey(const char *name, KeyList& Keys) +{ + CString TxtKeyList = regQueryStringValue(name, ""); + int curPos= 0; + + CString resToken=TxtKeyList.Tokenize(",",curPos); + while (resToken != "") + { + Keys.AddTail(atoi(resToken)); + resToken= TxtKeyList.Tokenize(",",curPos); + }; +} + +void winReadKey(const char *name, int num, KeyList& Keys) +{ + char buffer[80]; + + sprintf(buffer, "Joy%d_%s", num, name); + winReadKey(buffer, Keys); +} + + +void winReadKeys() +{ + + for(int i = 0; i < JOYPADS; i++) { + winReadKey("Left", i, theApp.input->joypaddata[JOYPAD(i,KEY_LEFT)]); + winReadKey("Right", i, theApp.input->joypaddata[JOYPAD(i, KEY_RIGHT)]); + winReadKey("Up", i, theApp.input->joypaddata[JOYPAD(i,KEY_UP)]); + winReadKey("Down", i, theApp.input->joypaddata[JOYPAD(i,KEY_DOWN)]); + winReadKey("A", i, theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_A)]); + winReadKey("B", i, theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_B)]); + winReadKey("L", i, theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_L)]); + winReadKey("R", i, theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_R)]); + winReadKey("Start", i, theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_START)]); + winReadKey("Select", i, theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_SELECT)]); + winReadKey("Speed", i, theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_SPEED)]); + winReadKey("Capture", i, theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_CAPTURE)]); + winReadKey("GS", i, theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_GS)]); + } + winReadKey("Motion_Left", theApp.input->joypaddata[MOTION(KEY_LEFT)]); + winReadKey("Motion_Right", theApp.input->joypaddata[MOTION(KEY_RIGHT)]); + winReadKey("Motion_Up", theApp.input->joypaddata[MOTION(KEY_UP)]); + winReadKey("Motion_Down", theApp.input->joypaddata[MOTION(KEY_DOWN)]); +} + +void winSaveKey(char *name, KeyList& value) +{ + CString txtKeys; + + POSITION p = value.GetHeadPosition(); + while(p!=NULL) + { + CString tmp; + tmp.Format("%d", value.GetNext(p)); + txtKeys+=tmp; + if (p!=NULL) + txtKeys+=","; + } + regSetStringValue(name, txtKeys); +} + +static void winSaveKey(char *name, int num, KeyList& value) +{ + char buffer[80]; + + sprintf(buffer, "Joy%d_%s", num, name); + winSaveKey(buffer, value); +} + +void winSaveKeys() +{ + for(int i = 0; i < JOYPADS; i++) { + winSaveKey("Left", i, theApp.input->joypaddata[JOYPAD(i,KEY_LEFT)]); + winSaveKey("Right", i, theApp.input->joypaddata[JOYPAD(i,KEY_RIGHT)]); + winSaveKey("Up", i, theApp.input->joypaddata[JOYPAD(i,KEY_UP)]); + winSaveKey("Speed", i, theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_SPEED)]); + winSaveKey("Capture", i, theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_CAPTURE)]); + winSaveKey("GS", i, theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_GS)]); + winSaveKey("Down", i, theApp.input->joypaddata[JOYPAD(i,KEY_DOWN)]); + winSaveKey("A", i, theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_A)]); + winSaveKey("B", i, theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_B)]); + winSaveKey("L", i, theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_L)]); + winSaveKey("R", i, theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_R)]); + winSaveKey("Start", i, theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_START)]); + winSaveKey("Select", i, theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_SELECT)]); + } + regSetDwordValue("joyVersion", 1); + + winSaveKey("Motion_Left", + theApp.input->joypaddata[MOTION(KEY_LEFT)]); + winSaveKey("Motion_Right", + theApp.input->joypaddata[MOTION(KEY_RIGHT)]); + winSaveKey("Motion_Up", + theApp.input->joypaddata[MOTION(KEY_UP)]); + winSaveKey("Motion_Down", + theApp.input->joypaddata[MOTION(KEY_DOWN)]); +} + +static BOOL CALLBACK EnumAxesCallback( const DIDEVICEOBJECTINSTANCE* pdidoi, + VOID* pContext ) +{ + DIPROPRANGE diprg; + diprg.diph.dwSize = sizeof(DIPROPRANGE); + diprg.diph.dwHeaderSize = sizeof(DIPROPHEADER); + diprg.diph.dwHow = DIPH_BYOFFSET; + diprg.diph.dwObj = pdidoi->dwOfs; // Specify the enumerated axis + + diprg.lMin = -32768; + diprg.lMax = 32767; + // try to set the range + if (FAILED(currentDevice->device->SetProperty(DIPROP_RANGE, &diprg.diph))) { + // Get the range for the axis + if ( FAILED(currentDevice->device-> + GetProperty( DIPROP_RANGE, &diprg.diph ) ) ) { + return DIENUM_STOP; + } + } + + DIPROPDWORD didz; + + didz.diph.dwSize = sizeof(didz); + didz.diph.dwHeaderSize = sizeof(DIPROPHEADER); + didz.diph.dwHow = DIPH_BYOFFSET; + didz.diph.dwObj = pdidoi->dwOfs; + + didz.dwData = 5000; + + currentDevice->device->SetProperty(DIPROP_DEADZONE, &didz.diph); + + LONG center = (diprg.lMin + diprg.lMax)/2; + LONG threshold = (diprg.lMax - center)/2; + + // only 8 axis supported + if (axisNumber < 8) { + currentDevice->axis[axisNumber].center = center; + currentDevice->axis[axisNumber].negative = center - threshold; + currentDevice->axis[axisNumber].positive = center + threshold; + currentDevice->axis[axisNumber].offset = pdidoi->dwOfs; + } + axisNumber++; + return DIENUM_CONTINUE; +} + +static BOOL CALLBACK EnumPovsCallback( const DIDEVICEOBJECTINSTANCE* pdidoi, + VOID* pContext ) +{ + return DIENUM_CONTINUE; +} + +static BOOL CALLBACK DIEnumDevicesCallback(LPCDIDEVICEINSTANCE pInst, + LPVOID lpvContext) +{ + ZeroMemory(&pDevices[numDevices],sizeof(deviceInfo)); + + HRESULT hRet = pDirectInput->CreateDevice(pInst->guidInstance, + &pDevices[numDevices].device, + NULL); + + if (hRet != DI_OK) + return DIENUM_STOP; + + DIDEVCAPS caps; + caps.dwSize=sizeof(DIDEVCAPS); + + hRet = pDevices[numDevices].device->GetCapabilities(&caps); + + if (hRet == DI_OK) { + if (caps.dwFlags & DIDC_POLLEDDATAFORMAT || + caps.dwFlags & DIDC_POLLEDDEVICE) + pDevices[numDevices].isPolled = TRUE; + + pDevices[numDevices].nButtons = caps.dwButtons; + pDevices[numDevices].nAxes = caps.dwAxes; + pDevices[numDevices].nPovs = caps.dwPOVs; + + for (int i = 0; i < 6; i++) { + pDevices[numDevices].axis[i].center = 0x8000; + pDevices[numDevices].axis[i].negative = 0x4000; + pDevices[numDevices].axis[i].positive = 0xc000; + } + } + + + numDevices++; + + + return DIENUM_CONTINUE; +} + +BOOL CALLBACK DIEnumDevicesCallback2(LPCDIDEVICEINSTANCE pInst, + LPVOID lpvContext) +{ + numDevices++; + + return DIENUM_CONTINUE; +} + +static int getPovState(DWORD value) +{ + int state = 0; + if (LOWORD(value) != 0xFFFF) { + if (value < 9000 || value > 27000) + state |= POV_UP; + if (value > 0 && value < 18000) + state |= POV_RIGHT; + if (value > 9000 && value < 27000) + state |= POV_DOWN; + if (value > 18000) + state |= POV_LEFT; + } + return state; +} + +static void checkKeys() +{ + LONG_PTR dev = 0; + int i; + + for(i = 0; i < (sizeof(theApp.input->joypaddata) / sizeof(theApp.input->joypaddata[0])); i++) + { + if (theApp.input->joypaddata[i].IsEmpty() && defvalues[i]) + theApp.input->joypaddata[i].AddTail(defvalues[i]); + POSITION p = theApp.input->joypaddata[i].GetHeadPosition(); + while(p!=NULL) + { + LONG_PTR k = theApp.input->joypaddata[i].GetNext(p); + if (k > 0 && DEVICEOF(k) < numDevices) + pDevices[DEVICEOF(k)].needed = true; + } + } +} + +#define KEYDOWN(buffer,key) (buffer[key] & 0x80) + +static bool readKeyboard() +{ + if (pDevices[0].needed) { + HRESULT hret = pDevices[0].device-> + GetDeviceState(256, + (LPVOID)pDevices[0].data); + + if (hret == DIERR_INPUTLOST || hret == DIERR_NOTACQUIRED) { + hret = pDevices[0].device->Acquire(); + if (hret != DI_OK) + return false; + hret = pDevices[0].device->GetDeviceState(256,(LPVOID)pDevices[0].data); + } + + return hret == DI_OK; + } + return true; +} + +static bool readJoystick(int joy) +{ + if (pDevices[joy].needed) { + if (pDevices[joy].isPolled) + ((LPDIRECTINPUTDEVICE2)pDevices[joy].device)->Poll(); + + HRESULT hret = pDevices[joy].device-> + GetDeviceState(sizeof(DIJOYSTATE), + (LPVOID)&pDevices[joy].state); + + if (hret == DIERR_INPUTLOST || hret == DIERR_NOTACQUIRED) { + hret = pDevices[joy].device->Acquire(); + + if (hret == DI_OK) { + + if (pDevices[joy].isPolled) + ((LPDIRECTINPUTDEVICE2)pDevices[joy].device)->Poll(); + + hret = pDevices[joy].device-> + GetDeviceState(sizeof(DIJOYSTATE), + (LPVOID)&pDevices[joy].state); + } + } + + return hret == DI_OK; + } + + return true; +} + +static void checkKeyboard() +{ + // mham fix. Patch #1378104 + UCHAR keystate[256]; + HRESULT hret = pDevices[0].device->Acquire(); + + if (pDevices[0].first) { + pDevices[0].device->GetDeviceState(256, (LPVOID)pDevices[0].data); + pDevices[0].first = FALSE; + return; + } + + hret = pDevices[0].device-> + GetDeviceState(256, (LPVOID)keystate); + + if (hret == DIERR_INPUTLOST || hret == DIERR_NOTACQUIRED) { + return; + } + + if (hret == DI_OK) { + for (int i = 0; i < 256; i++) { + if (keystate[i] == pDevices[0].data[i]) continue; + if (KEYDOWN(keystate, i)) { + SendMessage(GetFocus(), JOYCONFIG_MESSAGE,0,i); + break; + } + } + } + memcpy(pDevices[0].data, keystate, sizeof(UCHAR) * 256); +} + +static void checkJoypads() +{ + DIDEVICEOBJECTINSTANCE di; + + ZeroMemory(&di,sizeof(DIDEVICEOBJECTINSTANCE)); + + di.dwSize = sizeof(DIDEVICEOBJECTINSTANCE); + + int i =0; + + DIJOYSTATE joystick; + + for (i = 1; i < numDevices; i++) { + HRESULT hret = pDevices[i].device->Acquire(); + + + if (pDevices[i].isPolled) + ((LPDIRECTINPUTDEVICE2)pDevices[i].device)->Poll(); + + hret = pDevices[i].device->GetDeviceState(sizeof(joystick), &joystick); + + int j; + + if (pDevices[i].first) { + memcpy(&pDevices[i].state, &joystick, sizeof(joystick)); + pDevices[i].first = FALSE; + continue; + } + + for (j = 0; j < pDevices[i].nButtons; j++) { + if (((pDevices[i].state.rgbButtons[j] ^ joystick.rgbButtons[j]) + & joystick.rgbButtons[j]) & 0x80) { + HWND focus = GetFocus(); + + SendMessage(focus, JOYCONFIG_MESSAGE, i,j+128); + } + } + + for (j = 0; j < pDevices[i].nAxes && j < 8; j++) { + LONG value = pDevices[i].axis[j].center; + LONG old = 0; + switch (pDevices[i].axis[j].offset) { + case DIJOFS_X: + value = joystick.lX; + old = pDevices[i].state.lX; + break; + case DIJOFS_Y: + value = joystick.lY; + old = pDevices[i].state.lY; + break; + case DIJOFS_Z: + value = joystick.lZ; + old = pDevices[i].state.lZ; + break; + case DIJOFS_RX: + value = joystick.lRx; + old = pDevices[i].state.lRx; + break; + case DIJOFS_RY: + value = joystick.lRy; + old = pDevices[i].state.lRy; + break; + case DIJOFS_RZ: + value = joystick.lRz; + old = pDevices[i].state.lRz; + break; + case DIJOFS_SLIDER(0): + value = joystick.rglSlider[0]; + old = pDevices[i].state.rglSlider[0]; + break; + case DIJOFS_SLIDER(1): + value = joystick.rglSlider[1]; + old = pDevices[i].state.rglSlider[1]; + break; + } + if (value != old) { + if (value < pDevices[i].axis[j].negative) + SendMessage(GetFocus(), JOYCONFIG_MESSAGE, i, (j<<1)); + else if (value > pDevices[i].axis[j].positive) + SendMessage(GetFocus(), JOYCONFIG_MESSAGE, i, (j<<1)+1); + } + } + + for (j = 0;j < 4 && j < pDevices[i].nPovs; j++) { + if (LOWORD(pDevices[i].state.rgdwPOV[j]) != LOWORD(joystick.rgdwPOV[j])) { + int state = getPovState(joystick.rgdwPOV[j]); + + if (state & POV_UP) + SendMessage(GetFocus(), JOYCONFIG_MESSAGE, i, (j<<2)+0x20); + else if (state & POV_DOWN) + SendMessage(GetFocus(), JOYCONFIG_MESSAGE, i, (j<<2)+0x21); + else if (state & POV_RIGHT) + SendMessage(GetFocus(), JOYCONFIG_MESSAGE, i, (j<<2)+0x22); + else if (state & POV_LEFT) + SendMessage(GetFocus(), JOYCONFIG_MESSAGE, i, (j<<2)+0x23); + } + } + + memcpy(&pDevices[i].state, &joystick, sizeof(joystick)); + } +} + +BOOL checkKey(LONG_PTR key) +{ + LONG_PTR dev = (key >> 8); + + LONG_PTR k = (key & 255); + + if (dev == 0) { + return KEYDOWN(pDevices[0].data,k); + } else { + if (k < 16) { + LONG_PTR axis = k >> 1; + LONG value = pDevices[dev].axis[axis].center; + switch (pDevices[dev].axis[axis].offset) { + case DIJOFS_X: + value = pDevices[dev].state.lX; + break; + case DIJOFS_Y: + value = pDevices[dev].state.lY; + break; + case DIJOFS_Z: + value = pDevices[dev].state.lZ; + break; + case DIJOFS_RX: + value = pDevices[dev].state.lRx; + break; + case DIJOFS_RY: + value = pDevices[dev].state.lRy; + break; + case DIJOFS_RZ: + value = pDevices[dev].state.lRz; + break; + case DIJOFS_SLIDER(0): + value = pDevices[dev].state.rglSlider[0]; + break; + case DIJOFS_SLIDER(1): + value = pDevices[dev].state.rglSlider[1]; + break; + } + + if (k & 1) + return value > pDevices[dev].axis[axis].positive; + return value < pDevices[dev].axis[axis].negative; + } else if (k < 48) { + LONG_PTR hat = (k >> 2) & 3; + int state = getPovState(pDevices[dev].state.rgdwPOV[hat]); + BOOL res = FALSE; + switch (k & 3) { + case 0: + res = state & POV_UP; + break; + case 1: + res = state & POV_DOWN; + break; + case 2: + res = state & POV_RIGHT; + break; + case 3: + res = state & POV_LEFT; + break; + } + return res; + } else if (k >= 128) { + return pDevices[dev].state.rgbButtons[k-128] & 0x80; + } + } + + return FALSE; +} BOOL checkKey(KeyList &k) { @@ -599,373 +603,337 @@ BOOL checkKey(KeyList &k) return TRUE; } return FALSE; -} - -DirectInput::DirectInput() -{ - dinputDLL = NULL; -} - -DirectInput::~DirectInput() -{ - saveSettings(); - if(pDirectInput != NULL) { - if(pDevices) { - for(int i = 0; i < numDevices ; i++) { - if(pDevices[i].device) { - pDevices[i].device->Unacquire(); - pDevices[i].device->Release(); - pDevices[i].device = NULL; - } - } - free(pDevices); - pDevices = NULL; - } - - pDirectInput->Release(); - pDirectInput = NULL; - } - - if(dinputDLL) { - FreeLibrary(dinputDLL); - dinputDLL = NULL; - } -} - -bool DirectInput::initialize() -{ - joyDebug = GetPrivateProfileInt("config", - "joyDebug", - 0, - "VBA.ini"); - dinputDLL = LoadLibrary("DINPUT.DLL"); - HRESULT (WINAPI *DInputCreate)(HINSTANCE,DWORD,LPDIRECTINPUT *,IUnknown *); - if(dinputDLL != NULL) { - DInputCreate = (HRESULT (WINAPI *)(HINSTANCE,DWORD,LPDIRECTINPUT *,IUnknown *)) - GetProcAddress(dinputDLL, "DirectInputCreateA"); - - if(DInputCreate == NULL) { - directXMessage("DirectInputCreateA"); - return false; - } - } else { - directXMessage("DINPUT.DLL"); - return false; - } - - HRESULT hret = DInputCreate(AfxGetInstanceHandle(), - DIRECTINPUT_VERSION, - &pDirectInput, - NULL); - if(hret != DI_OK) { - // errorMessage(myLoadString(IDS_ERROR_DISP_CREATE), hret); - return false; - } - - hret = pDirectInput->EnumDevices(DIDEVTYPE_JOYSTICK, - DIEnumDevicesCallback2, - NULL, - DIEDFL_ATTACHEDONLY); - - - - pDevices = (deviceInfo *)calloc(numDevices, sizeof(deviceInfo)); - - hret = pDirectInput->CreateDevice(GUID_SysKeyboard,&pDevices[0].device,NULL); - pDevices[0].isPolled = false; - pDevices[0].needed = true; - - if(hret != DI_OK) { - // errorMessage(myLoadString(IDS_ERROR_DISP_CREATEDEVICE), hret); - return false; - } - - - numDevices = 1; - - hret = pDirectInput->EnumDevices(DIDEVTYPE_JOYSTICK, - DIEnumDevicesCallback, - NULL, - DIEDFL_ATTACHEDONLY); - - // hret = pDevices[0].device->SetCooperativeLevel(hWindow, - // DISCL_FOREGROUND| - // DISCL_NONEXCLUSIVE); - - if(hret != DI_OK) { - // errorMessage(myLoadString(IDS_ERROR_DISP_LEVEL), hret); - return false; - } - - hret = pDevices[0].device->SetDataFormat(&c_dfDIKeyboard); - - if(hret != DI_OK) { - // errorMessage(myLoadString(IDS_ERROR_DISP_DATAFORMAT), hret); - return false; - } - - for(int i = 1; i < numDevices; i++) { - pDevices[i].device->SetDataFormat(&c_dfDIJoystick); - pDevices[i].needed = false; - currentDevice = &pDevices[i]; - axisNumber = 0; - currentDevice->device->EnumObjects(EnumAxesCallback, NULL, DIDFT_AXIS); - currentDevice->device->EnumObjects(EnumPovsCallback, NULL, DIDFT_POV); - if(joyDebug) { - // don't translate. debug only - winlog("Joystick %2d polled : %d\n", i, currentDevice->isPolled); - winlog("Joystick %2d buttons : %d\n", i, currentDevice->nButtons); - winlog("Joystick %2d povs : %d\n", i, currentDevice->nPovs); - winlog("Joystick %2d axes : %d\n", i, currentDevice->nAxes); - for(int j = 0; j < currentDevice->nAxes; j++) { - winlog("Axis %2d offset : %08lx\n", j, currentDevice->axis[j]. - offset); - winlog("Axis %2d center : %08lx\n", j, currentDevice->axis[j]. - center); - winlog("Axis %2d negative : %08lx\n", j, currentDevice->axis[j]. - negative); - winlog("Axis %2d positive : %08lx\n", j, currentDevice->axis[j]. - positive); - } - } - - currentDevice = NULL; - } - - if (1) for(int i = 0; i < numDevices; i++) - pDevices[i].device->Acquire(); - - return true; -} - -bool DirectInput::readDevices() -{ - bool ok = true; - for(int i = 0; i < numDevices; i++) { - if(pDevices[i].needed) { - if(i) { - ok = readJoystick(i); - } else - ok = readKeyboard(); - } - } - return ok; -} - -u32 DirectInput::readDevice(int which) -{ - u32 res = 0; - int i = theApp.joypadDefault; - if(which >= 0 && which <= 3) - i = which; - - if(checkKey(joypad[JOYPAD(i,KEY_BUTTON_A)])) - res |= 1; - if(checkKey(joypad[JOYPAD(i,KEY_BUTTON_B)])) - res |= 2; - if(checkKey(joypad[JOYPAD(i,KEY_BUTTON_SELECT)])) - res |= 4; - if(checkKey(joypad[JOYPAD(i,KEY_BUTTON_START)])) - res |= 8; - if(checkKey(joypad[JOYPAD(i,KEY_RIGHT)])) - res |= 16; - if(checkKey(joypad[JOYPAD(i,KEY_LEFT)])) - res |= 32; - if(checkKey(joypad[JOYPAD(i,KEY_UP)])) - res |= 64; - if(checkKey(joypad[JOYPAD(i,KEY_DOWN)])) - res |= 128; - if(checkKey(joypad[JOYPAD(i,KEY_BUTTON_R)])) - res |= 256; - if(checkKey(joypad[JOYPAD(i,KEY_BUTTON_L)])) - res |= 512; - - if(checkKey(joypad[JOYPAD(i,KEY_BUTTON_GS)])) - res |= 4096; - - if(theApp.autoFire) { - res &= (~theApp.autoFire); - if(theApp.autoFireToggle) - res |= theApp.autoFire; - theApp.autoFireToggle = !theApp.autoFireToggle; - } - - // disallow L+R or U+D of being pressed at the same time - if((res & 48) == 48) - res &= ~16; - if((res & 192) == 192) - res &= ~128; - - if(theApp.movieRecording) { - if(i == theApp.joypadDefault) { - if(res != theApp.movieLastJoypad) { - fwrite(&theApp.movieFrame, 1, sizeof(theApp.movieFrame), theApp.movieFile); - fwrite(&res, 1, sizeof(res), theApp.movieFile); - theApp.movieLastJoypad = res; - } - } - } - if(theApp.moviePlaying) { - if(theApp.movieFrame == theApp.moviePlayFrame) { - theApp.movieLastJoypad = theApp.movieNextJoypad; - theApp.movieReadNext(); - } - res = theApp.movieLastJoypad; - } - // we don't record speed up or screen capture buttons - if(checkKey(joypad[JOYPAD(i,KEY_BUTTON_SPEED)]) || theApp.speedupToggle) - res |= 1024; - if(checkKey(joypad[JOYPAD(i,KEY_BUTTON_CAPTURE)])) - res |= 2048; - - return res; -} - -CString DirectInput::getKeyName(int key) -{ - int d = (key >> 8); - int k = key & 255; - - DIDEVICEOBJECTINSTANCE di; - - ZeroMemory(&di,sizeof(DIDEVICEOBJECTINSTANCE)); - - di.dwSize = sizeof(DIDEVICEOBJECTINSTANCE); - - CString winBuffer = winResLoadString(IDS_ERROR); - - if(d == 0) { - pDevices[0].device->GetObjectInfo(&di,key,DIPH_BYOFFSET); - winBuffer = di.tszName; - } else { - if(k < 16) { - if(k < 4) { - switch(k) { - case 0: - winBuffer.Format(winResLoadString(IDS_JOY_LEFT), d); - break; - case 1: - winBuffer.Format(winResLoadString(IDS_JOY_RIGHT), d); - break; - case 2: - winBuffer.Format(winResLoadString(IDS_JOY_UP), d); - break; - case 3: - winBuffer.Format(winResLoadString(IDS_JOY_DOWN), d); - break; - } - } else { - pDevices[d].device->GetObjectInfo(&di, - pDevices[d].axis[k>>1].offset, - DIPH_BYOFFSET); - if(k & 1) - winBuffer.Format("Joy %d %s +", d, di.tszName); - else - winBuffer.Format("Joy %d %s -", d, di.tszName); - } - } else if(k < 48) { - int hat = (k >> 2) & 3; - pDevices[d].device->GetObjectInfo(&di, - DIJOFS_POV(hat), - DIPH_BYOFFSET); - char *dir = "up"; - int dd = k & 3; - if(dd == 1) - dir = "down"; - else if(dd == 2) - dir = "right"; - else if(dd == 3) - dir = "left"; - winBuffer.Format("Joy %d %s %s", d, di.tszName, dir); - } else { - pDevices[d].device->GetObjectInfo(&di, - DIJOFS_BUTTON(k-128), - DIPH_BYOFFSET); - winBuffer.Format(winResLoadString(IDS_JOY_BUTTON),d,di.tszName); - } - } - - return winBuffer; -} - -void DirectInput::checkKeys() -{ - ::checkKeys(); -} - -void DirectInput::checkMotionKeys() -{ - if(checkKey(joypad[MOTION(KEY_LEFT)])) { - theApp.sensorX += 3; - if(theApp.sensorX > 2197) - theApp.sensorX = 2197; - if(theApp.sensorX < 2047) - theApp.sensorX = 2057; - } else if(checkKey(joypad[MOTION(KEY_RIGHT)])) { - theApp.sensorX -= 3; - if(theApp.sensorX < 1897) - theApp.sensorX = 1897; - if(theApp.sensorX > 2047) - theApp.sensorX = 2037; - } else if(theApp.sensorX > 2047) { - theApp.sensorX -= 2; - if(theApp.sensorX < 2047) - theApp.sensorX = 2047; - } else { - theApp.sensorX += 2; - if(theApp.sensorX > 2047) - theApp.sensorX = 2047; - } - - if(checkKey(joypad[MOTION(KEY_UP)])) { - theApp.sensorY += 3; - if(theApp.sensorY > 2197) - theApp.sensorY = 2197; - if(theApp.sensorY < 2047) - theApp.sensorY = 2057; - } else if(checkKey(joypad[MOTION(KEY_DOWN)])) { - theApp.sensorY -= 3; - if(theApp.sensorY < 1897) - theApp.sensorY = 1897; - if(theApp.sensorY > 2047) - theApp.sensorY = 2037; - } else if(theApp.sensorY > 2047) { - theApp.sensorY -= 2; - if(theApp.sensorY < 2047) - theApp.sensorY = 2047; - } else { - theApp.sensorY += 2; - if(theApp.sensorY > 2047) - theApp.sensorY = 2047; - } -} - -Input *newDirectInput() -{ - return new DirectInput; -} - - -void DirectInput::checkDevices() -{ - checkJoypads(); - checkKeyboard(); -} - -void DirectInput::activate() -{ - for(int i = 0; i < numDevices; i++) { - if(pDevices != NULL && pDevices[i].device != NULL) - pDevices[i].device->Acquire(); - } -} - -void DirectInput::loadSettings() -{ - winReadKeys(); -} - -void DirectInput::saveSettings() -{ - winSaveKeys(); -} +} + +DirectInput::DirectInput() +{} + +DirectInput::~DirectInput() +{ + saveSettings(); + if (pDirectInput != NULL) { + if (pDevices) { + for (int i = 0; i < numDevices ; i++) { + if (pDevices[i].device) { + pDevices[i].device->Unacquire(); + pDevices[i].device->Release(); + pDevices[i].device = NULL; + } + } + free(pDevices); + pDevices = NULL; + } + + pDirectInput->Release(); + pDirectInput = NULL; + } +} + + +bool DirectInput::initialize() +{ + + HRESULT hr; + + hr = DirectInput8Create( + AfxGetInstanceHandle(), + DIRECTINPUT_VERSION, + IID_IDirectInput8, + (LPVOID*)&pDirectInput, + NULL ); + + if ( hr != DI_OK ) { + return false; + } + + + + + hr = pDirectInput->EnumDevices(DI8DEVCLASS_GAMECTRL, + DIEnumDevicesCallback2, + NULL, + DIEDFL_ATTACHEDONLY); + + + + pDevices = (deviceInfo *)calloc(numDevices, sizeof(deviceInfo)); + + hr = pDirectInput->CreateDevice(GUID_SysKeyboard,&pDevices[0].device,NULL); + pDevices[0].isPolled = false; + pDevices[0].needed = true; + pDevices[0].first = true; + + if (hr != DI_OK) { + return false; + } + + + numDevices = 1; + + hr = pDirectInput->EnumDevices(DI8DEVCLASS_GAMECTRL, + DIEnumDevicesCallback, + NULL, + DIEDFL_ATTACHEDONLY); + + + if (hr != DI_OK) { + return false; + } + + hr = pDevices[0].device->SetDataFormat(&c_dfDIKeyboard); + + if (hr != DI_OK) { + return false; + } + + int i; + for (i = 1; i < numDevices; i++) { + pDevices[i].device->SetDataFormat(&c_dfDIJoystick); + pDevices[i].needed = false; + pDevices[i].first = true; + currentDevice = &pDevices[i]; + axisNumber = 0; + currentDevice->device->EnumObjects(EnumAxesCallback, NULL, DIDFT_AXIS); + currentDevice->device->EnumObjects(EnumPovsCallback, NULL, DIDFT_POV); + + + currentDevice = NULL; + } + + for (i = 0; i < numDevices; i++) + pDevices[i].device->Acquire(); + + return true; +} + +bool DirectInput::readDevices() +{ + bool ok = true; + for (int i = 0; i < numDevices; i++) { + if (pDevices[i].needed) { + if (i) { + ok = readJoystick(i); + } else + ok = readKeyboard(); + } + } + return ok; +} + +u32 DirectInput::readDevice(int which) +{ + u32 res = 0; + int i = theApp.joypadDefault; + if(which >= 0 && which <= 3) + i = which; + + if(checkKey(theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_A)])) + res |= 1; + if(checkKey(theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_B)])) + res |= 2; + if(checkKey(theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_SELECT)])) + res |= 4; + if(checkKey(theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_START)])) + res |= 8; + if(checkKey(theApp.input->joypaddata[JOYPAD(i,KEY_RIGHT)])) + res |= 16; + if(checkKey(theApp.input->joypaddata[JOYPAD(i,KEY_LEFT)])) + res |= 32; + if(checkKey(theApp.input->joypaddata[JOYPAD(i,KEY_UP)])) + res |= 64; + if(checkKey(theApp.input->joypaddata[JOYPAD(i,KEY_DOWN)])) + res |= 128; + if(checkKey(theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_R)])) + res |= 256; + if(checkKey(theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_L)])) + res |= 512; + + if(checkKey(theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_GS)])) + res |= 4096; + + if(theApp.autoFire) { + res &= (~theApp.autoFire); + if(theApp.autoFireToggle) + res |= theApp.autoFire; + theApp.autoFireToggle = !theApp.autoFireToggle; + } + + // disallow L+R or U+D of being pressed at the same time + if((res & 48) == 48) + res &= ~16; + if((res & 192) == 192) + res &= ~128; + + if(theApp.movieRecording) { + if(i == theApp.joypadDefault) { + if(res != theApp.movieLastJoypad) { + fwrite(&theApp.movieFrame, 1, sizeof(theApp.movieFrame), theApp.movieFile); + fwrite(&res, 1, sizeof(res), theApp.movieFile); + theApp.movieLastJoypad = res; + } + } + } + if(theApp.moviePlaying) { + if(theApp.movieFrame == theApp.moviePlayFrame) { + theApp.movieLastJoypad = theApp.movieNextJoypad; + theApp.movieReadNext(); + } + res = theApp.movieLastJoypad; + } + // we don't record speed up or screen capture buttons + if(checkKey(theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_SPEED)]) || theApp.speedupToggle) + res |= 1024; + if(checkKey(theApp.input->joypaddata[JOYPAD(i,KEY_BUTTON_CAPTURE)])) + res |= 2048; + + return res; +} + +CString DirectInput::getKeyName(LONG_PTR key) +{ + LONG_PTR d = (key >> 8); + LONG_PTR k = key & 255; + + DIDEVICEOBJECTINSTANCE di; + + ZeroMemory(&di,sizeof(DIDEVICEOBJECTINSTANCE)); + + di.dwSize = sizeof(DIDEVICEOBJECTINSTANCE); + + CString winBuffer = winResLoadString(IDS_ERROR); + + if (d == 0) { + pDevices[0].device->GetObjectInfo( &di, (DWORD)key, DIPH_BYOFFSET ); + winBuffer = di.tszName; + } else { + if (k < 16) { + if (k < 4) { + switch (k) { + case 0: + winBuffer.Format(winResLoadString(IDS_JOY_LEFT), d); + break; + case 1: + winBuffer.Format(winResLoadString(IDS_JOY_RIGHT), d); + break; + case 2: + winBuffer.Format(winResLoadString(IDS_JOY_UP), d); + break; + case 3: + winBuffer.Format(winResLoadString(IDS_JOY_DOWN), d); + break; + } + } else { + pDevices[d].device->GetObjectInfo(&di, + pDevices[d].axis[k>>1].offset, + DIPH_BYOFFSET); + if (k & 1) + winBuffer.Format("Joy %d %s +", d, di.tszName); + else + winBuffer.Format("Joy %d %s -", d, di.tszName); + } + } else if (k < 48) { + LONG_PTR hat = (k >> 2) & 3; + pDevices[d].device->GetObjectInfo(&di, + (DWORD)DIJOFS_POV(hat), + DIPH_BYOFFSET); + char *dir = "up"; + LONG_PTR dd = k & 3; + if (dd == 1) + dir = "down"; + else if (dd == 2) + dir = "right"; + else if (dd == 3) + dir = "left"; + winBuffer.Format("Joy %d %s %s", d, di.tszName, dir); + } else { + pDevices[d].device->GetObjectInfo(&di, + (DWORD)DIJOFS_BUTTON(k-128), + DIPH_BYOFFSET); + winBuffer.Format(winResLoadString(IDS_JOY_BUTTON),d,di.tszName); + } + } + + return winBuffer; +} + +void DirectInput::checkKeys() +{ + ::checkKeys(); +} + +void DirectInput::checkMotionKeys() +{ + if(checkKey(theApp.input->joypaddata[MOTION(KEY_LEFT)])) { + theApp.sensorX += 3; + if(theApp.sensorX > 2197) + theApp.sensorX = 2197; + if(theApp.sensorX < 2047) + theApp.sensorX = 2057; + } else if(checkKey(theApp.input->joypaddata[MOTION(KEY_RIGHT)])) { + theApp.sensorX -= 3; + if(theApp.sensorX < 1897) + theApp.sensorX = 1897; + if(theApp.sensorX > 2047) + theApp.sensorX = 2037; + } else if(theApp.sensorX > 2047) { + theApp.sensorX -= 2; + if(theApp.sensorX < 2047) + theApp.sensorX = 2047; + } else { + theApp.sensorX += 2; + if(theApp.sensorX > 2047) + theApp.sensorX = 2047; + } + + if(checkKey(theApp.input->joypaddata[MOTION(KEY_UP)])) { + theApp.sensorY += 3; + if(theApp.sensorY > 2197) + theApp.sensorY = 2197; + if(theApp.sensorY < 2047) + theApp.sensorY = 2057; + } else if(checkKey(theApp.input->joypaddata[MOTION(KEY_DOWN)])) { + theApp.sensorY -= 3; + if(theApp.sensorY < 1897) + theApp.sensorY = 1897; + if(theApp.sensorY > 2047) + theApp.sensorY = 2037; + } else if(theApp.sensorY > 2047) { + theApp.sensorY -= 2; + if(theApp.sensorY < 2047) + theApp.sensorY = 2047; + } else { + theApp.sensorY += 2; + if(theApp.sensorY > 2047) + theApp.sensorY = 2047; + } +} + +Input *newDirectInput() +{ + return new DirectInput; +} + + +void DirectInput::checkDevices() +{ + checkJoypads(); + checkKeyboard(); +} + +void DirectInput::activate() +{ + for (int i = 0; i < numDevices; i++) { + if (pDevices != NULL && pDevices[i].device != NULL) + pDevices[i].device->Acquire(); + } +} + +void DirectInput::loadSettings() +{ + winReadKeys(); +} + +void DirectInput::saveSettings() +{ + winSaveKeys(); +} diff --git a/src/win32/DirectSound.cpp b/src/win32/DirectSound.cpp index 24f0fb15..4c897be1 100644 --- a/src/win32/DirectSound.cpp +++ b/src/win32/DirectSound.cpp @@ -1,380 +1,383 @@ -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004-2005 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#include "stdafx.h" -#include "VBA.h" -#include "AVIWrite.h" -#include "Sound.h" -#include "WavWriter.h" - -#include "../System.h" -#include "../GBA.h" -#include "../Globals.h" -#include "../Sound.h" - -#include -#include //DirectSound - -extern bool soundBufferLow; - -class DirectSound : public ISound -{ -private: - HINSTANCE dsoundDLL; - LPDIRECTSOUND pDirectSound; - LPDIRECTSOUNDBUFFER dsbPrimary; - LPDIRECTSOUNDBUFFER dsbSecondary; - LPDIRECTSOUNDNOTIFY dsbNotify; - HANDLE dsbEvent; - WAVEFORMATEX wfx; - -public: - DirectSound(); - virtual ~DirectSound(); - - bool init(); - void pause(); - void reset(); - void resume(); - void write(); -}; - -DirectSound::DirectSound() -{ - dsoundDLL = NULL; - pDirectSound = NULL; - dsbPrimary = NULL; - dsbSecondary = NULL; - dsbNotify = NULL; - dsbEvent = NULL; -} - -DirectSound::~DirectSound() -{ - if(theApp.aviRecorder != NULL) { - delete theApp.aviRecorder; - theApp.aviRecorder = NULL; - theApp.aviFrameNumber = 0; - } - - if(theApp.soundRecording) { - if(theApp.soundRecorder != NULL) { - delete theApp.soundRecorder; - theApp.soundRecorder = NULL; - } - theApp.soundRecording = false; - } - - if(dsbNotify != NULL) { - dsbNotify->Release(); - dsbNotify = NULL; - } - - if(dsbEvent != NULL) { - CloseHandle(dsbEvent); - dsbEvent = NULL; - } - - if(pDirectSound != NULL) { - if(dsbPrimary != NULL) { - dsbPrimary->Release(); - dsbPrimary = NULL; - } - - if(dsbSecondary != NULL) { - dsbSecondary->Release(); - dsbSecondary = NULL; - } - - pDirectSound->Release(); - pDirectSound = NULL; - } - - if(dsoundDLL != NULL) { - FreeLibrary(dsoundDLL); - dsoundDLL = NULL; - } -} - -bool DirectSound::init() -{ - HRESULT hr; - - dsoundDLL = LoadLibrary("dsound.dll"); - HRESULT (WINAPI *DSoundCreate)(LPCGUID,LPDIRECTSOUND *,IUnknown *); - if(dsoundDLL != NULL) { - DSoundCreate = (HRESULT (WINAPI *)(LPCGUID,LPDIRECTSOUND *,IUnknown *)) - GetProcAddress(dsoundDLL, "DirectSoundCreate8"); - - if(DSoundCreate == NULL) { - theApp.directXMessage("DirectSoundCreate8"); - return false; - } - } else { - theApp.directXMessage("dsound.dll"); - return false; - } - - if((hr = DSoundCreate(NULL,&pDirectSound,NULL) != DS_OK)) { - // errorMessage(myLoadString(IDS_ERROR_SOUND_CREATE), hr); - systemMessage(IDS_CANNOT_CREATE_DIRECTSOUND, - "Cannot create DirectSound %08x", hr); - pDirectSound = NULL; - dsbSecondary = NULL; - return false; - } - - if((hr=pDirectSound->SetCooperativeLevel((HWND)*theApp.m_pMainWnd, - DSSCL_EXCLUSIVE)) != DS_OK) { - // errorMessage(myLoadString(IDS_ERROR_SOUND_LEVEL), hr); - systemMessage(IDS_CANNOT_SETCOOPERATIVELEVEL, - "Cannot SetCooperativeLevel %08x", hr); - return false; - } - - DSBUFFERDESC dsbdesc; - ZeroMemory(&dsbdesc,sizeof(DSBUFFERDESC)); - dsbdesc.dwSize=sizeof(DSBUFFERDESC); - dsbdesc.dwFlags = DSBCAPS_PRIMARYBUFFER; - - if((hr=pDirectSound->CreateSoundBuffer(&dsbdesc, - &dsbPrimary, - NULL) != DS_OK)) { - // errorMessage(myLoadString(IDS_ERROR_SOUND_BUFFER),hr); - systemMessage(IDS_CANNOT_CREATESOUNDBUFFER, - "Cannot CreateSoundBuffer %08x", hr); - return false; - } - - // Set primary buffer format - - memset(&wfx, 0, sizeof(WAVEFORMATEX)); - wfx.wFormatTag = WAVE_FORMAT_PCM; - wfx.nChannels = 2; - switch(soundQuality) { - case 2: - wfx.nSamplesPerSec = 22050; - soundBufferLen = 736*2; - soundBufferTotalLen = 7360*2; - break; - case 4: - wfx.nSamplesPerSec = 11025; - soundBufferLen = 368*2; - soundBufferTotalLen = 3680*2; - break; - default: - soundQuality = 1; - wfx.nSamplesPerSec = 44100; - soundBufferLen = 1470*2; - soundBufferTotalLen = 14700*2; - } - wfx.wBitsPerSample = 16; - wfx.nBlockAlign = (wfx.wBitsPerSample / 8) * wfx.nChannels; - wfx.nAvgBytesPerSec = wfx.nSamplesPerSec * wfx.nBlockAlign; - - if((hr = dsbPrimary->SetFormat(&wfx)) != DS_OK) { - // errorMessage(myLoadString(IDS_ERROR_SOUND_PRIMARY),hr); - systemMessage(IDS_CANNOT_SETFORMAT_PRIMARY, - "Cannot SetFormat for primary %08x", hr); - return false; - } - - ZeroMemory(&dsbdesc,sizeof(DSBUFFERDESC)); - dsbdesc.dwSize = sizeof(DSBUFFERDESC); - dsbdesc.dwFlags = DSBCAPS_GETCURRENTPOSITION2|DSBCAPS_CTRLPOSITIONNOTIFY; - dsbdesc.dwBufferBytes = soundBufferTotalLen; - dsbdesc.lpwfxFormat = &wfx; - - if((hr = pDirectSound->CreateSoundBuffer(&dsbdesc, &dsbSecondary, NULL)) - != DS_OK) { - dsbdesc.dwFlags = DSBCAPS_GETCURRENTPOSITION2; - if((hr = pDirectSound->CreateSoundBuffer(&dsbdesc, &dsbSecondary, NULL)) - != DS_OK) { - systemMessage(IDS_CANNOT_CREATESOUNDBUFFER_SEC, - "Cannot CreateSoundBuffer secondary %08x", hr); - return false; - } - } - - dsbSecondary->SetCurrentPosition(0); - - if(!theApp.useOldSync) { - hr = dsbSecondary->QueryInterface(IID_IDirectSoundNotify, - (void **)&dsbNotify); - if(!FAILED(hr)) { - dsbEvent = CreateEvent(NULL, FALSE, FALSE, NULL); - - DSBPOSITIONNOTIFY notify[10]; - - for(int i = 0; i < 10; i++) { - notify[i].dwOffset = i*soundBufferLen; - notify[i].hEventNotify = dsbEvent; - } - if(FAILED(dsbNotify->SetNotificationPositions(10, notify))) { - dsbNotify->Release(); - dsbNotify = NULL; - CloseHandle(dsbEvent); - dsbEvent = NULL; - } - } - } - - hr = dsbPrimary->Play(0,0,DSBPLAY_LOOPING); - - if(hr != DS_OK) { - // errorMessage(myLoadString(IDS_ERROR_SOUND_PLAYPRIM), hr); - systemMessage(IDS_CANNOT_PLAY_PRIMARY, "Cannot Play primary %08x", hr); - return false; - } - - setsystemSoundOn(true); - return true; -} - -void DirectSound::pause() -{ - if(dsbSecondary != NULL) { - DWORD status = 0; - dsbSecondary->GetStatus(&status); - - if(status & DSBSTATUS_PLAYING) { - dsbSecondary->Stop(); - } - } -} - -void DirectSound::reset() -{ - if(dsbSecondary) { - dsbSecondary->Stop(); - dsbSecondary->SetCurrentPosition(0); - } -} - -void DirectSound::resume() -{ - if(dsbSecondary != NULL) { - dsbSecondary->Play(0,0,DSBPLAY_LOOPING); - } -} - -void DirectSound::write() -{ - int len = soundBufferLen; - LPVOID lpvPtr1; - DWORD dwBytes1; - LPVOID lpvPtr2; - DWORD dwBytes2; - - if(!pDirectSound) - return; - - if(theApp.soundRecording) { - if(dsbSecondary) { - if(theApp.soundRecorder == NULL) { - theApp.soundRecorder = new WavWriter; - WAVEFORMATEX format; - dsbSecondary->GetFormat(&format, sizeof(format), NULL); - if(theApp.soundRecorder->Open(theApp.soundRecordName)) - theApp.soundRecorder->SetFormat(&format); - } - } - - if(theApp.soundRecorder) { - theApp.soundRecorder->AddSound((u8 *)soundFinalWave, len); - } - } - - if(theApp.aviRecording) { - if(theApp.aviRecorder) { - if(dsbSecondary) { - if(!theApp.aviRecorder->IsSoundAdded()) { - WAVEFORMATEX format; - dsbSecondary->GetFormat(&format, sizeof(format), NULL); - theApp.aviRecorder->SetSoundFormat(&format); - } - } - - theApp.aviRecorder->AddSound((const char *)soundFinalWave, len); - } - } - - HRESULT hr; - - if(!speedup && synchronize && !theApp.throttle) { - DWORD status=0; - hr = dsbSecondary->GetStatus(&status); - if(status && DSBSTATUS_PLAYING) { - if(!soundPaused) { - DWORD play; - while(true) { - dsbSecondary->GetCurrentPosition(&play, NULL); - int BufferLeft = ((soundNextPosition <= play) ? - play - soundNextPosition : - soundBufferTotalLen - soundNextPosition + play); - - if(BufferLeft > soundBufferLen) - { - if (BufferLeft > soundBufferTotalLen - (soundBufferLen * 3)) - soundBufferLow = true; - break; - } - soundBufferLow = false; - - if(dsbEvent) { - WaitForSingleObject(dsbEvent, 50); - } - } - } - } else { - setsoundPaused(true); - } - } - // Obtain memory address of write block. This will be in two parts - // if the block wraps around. - hr = dsbSecondary->Lock(soundNextPosition, soundBufferLen, &lpvPtr1, - &dwBytes1, &lpvPtr2, &dwBytes2, - 0); - - // If DSERR_BUFFERLOST is returned, restore and retry lock. - if (DSERR_BUFFERLOST == hr) { - dsbSecondary->Restore(); - hr = dsbSecondary->Lock(soundNextPosition, soundBufferLen,&lpvPtr1, - &dwBytes1, &lpvPtr2, &dwBytes2, - 0); - } - - soundNextPosition += soundBufferLen; - soundNextPosition = soundNextPosition % soundBufferTotalLen; - - if SUCCEEDED(hr) { - // Write to pointers. - CopyMemory(lpvPtr1, soundFinalWave, dwBytes1); - if (NULL != lpvPtr2) { - CopyMemory(lpvPtr2, soundFinalWave+dwBytes1, dwBytes2); - } - // Release the data back to DirectSound. - hr = dsbSecondary->Unlock(lpvPtr1, dwBytes1, lpvPtr2, - dwBytes2); - } -} - -ISound *newDirectSound() -{ - return new DirectSound(); -} +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2004-2006 VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +// MFC +#include "stdafx.h" + +// Tools +#include "AVIWrite.h" +#include "WavWriter.h" + +// Internals +#include "../System.h" +#include "../GBA.h" +#include "../Globals.h" +#include "../Sound.h" + +// DirectSound8 +#include +#pragma comment( lib, "Dsound" ) +#pragma comment( lib, "Dxguid" ) + +extern bool soundBufferLow; + +class DirectSound : public ISound +{ +private: + LPDIRECTSOUND8 pDirectSound; // DirectSound interface + LPDIRECTSOUNDBUFFER dsbPrimary; // Primary DirectSound buffer + LPDIRECTSOUNDBUFFER dsbSecondary; // Secondary DirectSound buffer + LPDIRECTSOUNDNOTIFY8 dsbNotify; + HANDLE dsbEvent; + WAVEFORMATEX wfx; // Primary buffer wave format + +public: + DirectSound(); + virtual ~DirectSound(); + + bool init(); // initialize the primary and secondary sound buffer + void pause(); // pause the secondary sound buffer + void reset(); // stop and reset the secondary sound buffer + void resume(); // resume the secondary sound buffer + void write(); // write the emulated sound to the secondary sound buffer +}; + + +DirectSound::DirectSound() +{ + CoInitialize( NULL ); + + pDirectSound = NULL; + dsbPrimary = NULL; + dsbSecondary = NULL; + dsbNotify = NULL; + dsbEvent = NULL; +} + + +DirectSound::~DirectSound() +{ + if(theApp.aviRecorder) { + delete theApp.aviRecorder; + theApp.aviRecorder = NULL; + theApp.aviFrameNumber = 0; + } + + if(theApp.soundRecording) { + if(theApp.soundRecorder) { + delete theApp.soundRecorder; + theApp.soundRecorder = NULL; + } + theApp.soundRecording = false; + } + + if(dsbNotify) { + dsbNotify->Release(); + dsbNotify = NULL; + } + + if(dsbEvent) { + CloseHandle(dsbEvent); + dsbEvent = NULL; + } + + if(pDirectSound) { + if(dsbPrimary) { + dsbPrimary->Release(); + dsbPrimary = NULL; + } + + if(dsbSecondary) { + dsbSecondary->Release(); + dsbSecondary = NULL; + } + + pDirectSound->Release(); + pDirectSound = NULL; + } + + CoUninitialize(); +} + + +bool DirectSound::init() +{ + HRESULT hr; + DWORD freq; + DSBUFFERDESC dsbdesc; + int i; + + + // Initialize DirectSound + if( FAILED( hr = DirectSoundCreate8( &DSDEVID_DefaultPlayback, &pDirectSound, NULL ) ) ) { + systemMessage( IDS_CANNOT_CREATE_DIRECTSOUND, _T("Cannot create DirectSound %08x"), hr ); + pDirectSound = NULL; + return false; + } + + + if( FAILED( hr = pDirectSound->SetCooperativeLevel( theApp.m_pMainWnd->GetSafeHwnd(), DSSCL_EXCLUSIVE ) ) ) { + systemMessage( IDS_CANNOT_SETCOOPERATIVELEVEL, _T("Cannot SetCooperativeLevel %08x"), hr ); + return false; + } + + + // Create primary sound buffer + ZeroMemory( &dsbdesc, sizeof(DSBUFFERDESC) ); + dsbdesc.dwSize = sizeof(DSBUFFERDESC); + dsbdesc.dwFlags = DSBCAPS_PRIMARYBUFFER; + if( theApp.dsoundDisableHardwareAcceleration ) { + dsbdesc.dwFlags |= DSBCAPS_LOCSOFTWARE; + } + + if( FAILED( hr = pDirectSound->CreateSoundBuffer( &dsbdesc, &dsbPrimary, NULL ) ) ) { + systemMessage(IDS_CANNOT_CREATESOUNDBUFFER, _T("Cannot CreateSoundBuffer %08x"), hr); + return false; + } + + switch(soundQuality) + { + case 4: + freq = 11025; + break; + case 2: + freq = 22050; + break; + default: + soundQuality = 1; + case 1: + freq = 44100; + break; + } + soundBufferLen = freq*2/30; + soundBufferTotalLen = soundBufferLen * 10; + + ZeroMemory( &wfx, sizeof(WAVEFORMATEX) ); + wfx.wFormatTag = WAVE_FORMAT_PCM; + wfx.nChannels = 2; + wfx.nSamplesPerSec = freq; + wfx.wBitsPerSample = 16; + wfx.nBlockAlign = wfx.nChannels * wfx.wBitsPerSample / 8; + wfx.nAvgBytesPerSec = wfx.nSamplesPerSec * wfx.nBlockAlign; + + if( FAILED( hr = dsbPrimary->SetFormat( &wfx ) ) ) { + systemMessage( IDS_CANNOT_SETFORMAT_PRIMARY, _T("CreateSoundBuffer(primary) failed %08x"), hr ); + return false; + } + + + // Create secondary sound buffer + ZeroMemory( &dsbdesc, sizeof(DSBUFFERDESC) ); + dsbdesc.dwSize = sizeof(DSBUFFERDESC); + dsbdesc.dwFlags = DSBCAPS_GETCURRENTPOSITION2 | DSBCAPS_CTRLPOSITIONNOTIFY | DSBCAPS_GLOBALFOCUS; + if( theApp.dsoundDisableHardwareAcceleration ) { + dsbdesc.dwFlags |= DSBCAPS_LOCSOFTWARE; + } + dsbdesc.dwBufferBytes = soundBufferTotalLen; + dsbdesc.lpwfxFormat = &wfx; + + if( FAILED( hr = pDirectSound->CreateSoundBuffer( &dsbdesc, &dsbSecondary, NULL ) ) ) { + systemMessage( IDS_CANNOT_CREATESOUNDBUFFER, _T("CreateSoundBuffer(secondary) failed %08x"), hr ); + return false; + } + + if( FAILED( hr = dsbSecondary->SetCurrentPosition( 0 ) ) ) { + systemMessage( 0, _T("dsbSecondary->SetCurrentPosition failed %08x"), hr ); + return false; + } + + + if( !theApp.useOldSync ) { + if( FAILED( hr = dsbSecondary->QueryInterface( IID_IDirectSoundNotify8, (LPVOID*)&dsbNotify ) ) ) { + dsbEvent = CreateEvent( NULL, FALSE, FALSE, NULL ); + DSBPOSITIONNOTIFY notify[10]; + for( i = 0; i < 10; i++ ) { + notify[i].dwOffset = i * soundBufferLen; + notify[i].hEventNotify = dsbEvent; + } + + if( FAILED( dsbNotify->SetNotificationPositions( 10, notify ) ) ) { + dsbNotify->Release(); + dsbNotify = NULL; + CloseHandle(dsbEvent); + dsbEvent = NULL; + } + } + } + + + // Play primary buffer + if( FAILED( hr = dsbPrimary->Play( 0, 0, DSBPLAY_LOOPING ) ) ) { + systemMessage( IDS_CANNOT_PLAY_PRIMARY, _T("Cannot Play primary %08x"), hr ); + return false; + } + + systemSoundOn = true; + + return true; +} + + +void DirectSound::pause() +{ + if( dsbSecondary == NULL ) return; + + DWORD status; + + dsbSecondary->GetStatus( &status ); + + if( status & DSBSTATUS_PLAYING ) dsbSecondary->Stop(); +} + + +void DirectSound::reset() +{ + if( dsbSecondary == NULL ) return; + + dsbSecondary->Stop(); + + dsbSecondary->SetCurrentPosition( 0 ); +} + + +void DirectSound::resume() +{ + if( dsbSecondary == NULL ) return; + + dsbSecondary->Play( 0, 0, DSBPLAY_LOOPING ); +} + + +void DirectSound::write() +{ + if(!pDirectSound) return; + + + HRESULT hr; + DWORD status = 0; + DWORD play = 0; + WAVEFORMATEX format; + LPVOID lpvPtr1; + DWORD dwBytes1 = 0; + LPVOID lpvPtr2; + DWORD dwBytes2 = 0; + + + if( theApp.soundRecording ) { + if( dsbSecondary ) { + if( theApp.soundRecorder ) { + theApp.soundRecorder->AddSound( (u8 *)soundFinalWave, soundBufferLen ); + } else { + theApp.soundRecorder = new WavWriter; + dsbSecondary->GetFormat( &format, sizeof(format), NULL ); + if( theApp.soundRecorder->Open( theApp.soundRecordName ) ) { + theApp.soundRecorder->SetFormat( &format ); + } + } + } + } + + + if( theApp.aviRecording ) { + if( theApp.aviRecorder ) { + if( dsbSecondary ) { + if( !theApp.aviRecorder->IsSoundAdded() ) { + dsbSecondary->GetFormat( &format, sizeof(format), NULL ); + theApp.aviRecorder->SetSoundFormat( &format ); + } + } + theApp.aviRecorder->AddSound( (const char *)soundFinalWave, soundBufferLen ); + } + } + + + if( !speedup && synchronize && !theApp.throttle ) { + hr = dsbSecondary->GetStatus(&status); + if( status & DSBSTATUS_PLAYING ) { + if( !soundPaused ) { + while( true ) { + dsbSecondary->GetCurrentPosition(&play, NULL); + int BufferLeft = ((soundNextPosition <= play) ? + play - soundNextPosition : + soundBufferTotalLen - soundNextPosition + play); + + if(BufferLeft > soundBufferLen) + { + if (BufferLeft > soundBufferTotalLen - (soundBufferLen * 3)) + soundBufferLow = true; + break; + } + soundBufferLow = false; + + if(dsbEvent) { + WaitForSingleObject(dsbEvent, 50); + } + } + } + } else { + setsoundPaused(true); + } + } + + + // Obtain memory address of write block. + // This will be in two parts if the block wraps around. + if( DSERR_BUFFERLOST == ( hr = dsbSecondary->Lock( + soundNextPosition, + soundBufferLen, + &lpvPtr1, + &dwBytes1, + &lpvPtr2, + &dwBytes2, + 0 ) ) ) { + // If DSERR_BUFFERLOST is returned, restore and retry lock. + dsbSecondary->Restore(); + hr = dsbSecondary->Lock( + soundNextPosition, + soundBufferLen, + &lpvPtr1, + &dwBytes1, + &lpvPtr2, + &dwBytes2, + 0 ); + } + + soundNextPosition += soundBufferLen; + soundNextPosition = soundNextPosition % soundBufferTotalLen; + + if( SUCCEEDED( hr ) ) { + // Write to pointers. + CopyMemory( lpvPtr1, soundFinalWave, dwBytes1 ); + if ( lpvPtr2 ) { + CopyMemory( lpvPtr2, soundFinalWave + dwBytes1, dwBytes2 ); + } + + // Release the data back to DirectSound. + hr = dsbSecondary->Unlock( lpvPtr1, dwBytes1, lpvPtr2, dwBytes2 ); + } else { + systemMessage( 0, _T("dsbSecondary->Lock() failed: %08x"), hr ); + return; + } +} + + +ISound *newDirectSound() +{ + return new DirectSound(); +} diff --git a/src/win32/Directories.cpp b/src/win32/Directories.cpp index 4a1bec69..d4585228 100644 --- a/src/win32/Directories.cpp +++ b/src/win32/Directories.cpp @@ -26,6 +26,7 @@ #include "WinResUtil.h" #include +#include #ifdef _DEBUG #define new DEBUG_NEW @@ -54,27 +55,21 @@ static int CALLBACK browseCallbackProc(HWND hWnd, UINT msg, Directories::Directories(CWnd* pParent /*=NULL*/) : CDialog(Directories::IDD, pParent) { - //{{AFX_DATA_INIT(Directories) - // NOTE: the ClassWizard will add member initialization here - //}}AFX_DATA_INIT } void Directories::DoDataExchange(CDataExchange* pDX) { CDialog::DoDataExchange(pDX); - //{{AFX_DATA_MAP(Directories) DDX_Control(pDX, IDC_SAVE_PATH, m_savePath); DDX_Control(pDX, IDC_ROM_PATH, m_romPath); DDX_Control(pDX, IDC_GBROM_PATH, m_gbromPath); DDX_Control(pDX, IDC_CAPTURE_PATH, m_capturePath); DDX_Control(pDX, IDC_BATTERY_PATH, m_batteryPath); - //}}AFX_DATA_MAP } BEGIN_MESSAGE_MAP(Directories, CDialog) - //{{AFX_MSG_MAP(Directories) ON_BN_CLICKED(IDC_BATTERY_DIR, OnBatteryDir) ON_BN_CLICKED(IDC_BATTERY_DIR_RESET, OnBatteryDirReset) ON_BN_CLICKED(IDC_CAPTURE_DIR, OnCaptureDir) @@ -85,8 +80,7 @@ BEGIN_MESSAGE_MAP(Directories, CDialog) ON_BN_CLICKED(IDC_ROM_DIR_RESET, OnRomDirReset) ON_BN_CLICKED(IDC_SAVE_DIR, OnSaveDir) ON_BN_CLICKED(IDC_SAVE_DIR_RESET, OnSaveDirReset) - //}}AFX_MSG_MAP - END_MESSAGE_MAP() +END_MESSAGE_MAP() ///////////////////////////////////////////////////////////////////////////// // Directories message handlers @@ -206,23 +200,76 @@ void Directories::OnCancel() void Directories::OnOK() { - CString buffer; - m_romPath.GetWindowText(buffer); - if(!buffer.IsEmpty()) - regSetStringValue("romdir", buffer); - m_gbromPath.GetWindowText(buffer); - if(!buffer.IsEmpty()) - regSetStringValue("gbromdir", buffer); - m_batteryPath.GetWindowText(buffer); - if(!buffer.IsEmpty()) - regSetStringValue("batteryDir", buffer); - m_savePath.GetWindowText(buffer); - if(!buffer.IsEmpty()) - regSetStringValue("saveDir", buffer); - m_capturePath.GetWindowText(buffer); - if(!buffer.IsEmpty()) - regSetStringValue("captureDir", buffer); - EndDialog(TRUE); + char baseDir[MAX_PATH+1]; + char temp[MAX_PATH+1]; + GetModuleFileName( NULL, baseDir, MAX_PATH ); + baseDir[MAX_PATH] = '\0'; // for security reasons + PathRemoveFileSpec( baseDir ); // removes the trailing file name and backslash + + + CString buffer; + + m_romPath.GetWindowText(buffer); + if( !buffer.IsEmpty() ) + regSetStringValue( "romdir", buffer ); + if( buffer[0] == '.' ) { + strcpy( temp, baseDir ); + strcat( temp, "\\" ); + strcat( temp, buffer ); + buffer = temp; + } + if( !directoryDoesExist( buffer ) ) + SHCreateDirectoryEx( NULL, buffer, NULL ); + + m_gbromPath.GetWindowText(buffer); + if( !buffer.IsEmpty() ) + regSetStringValue( "gbromdir", buffer ); + if( buffer[0] == '.' ) { + strcpy( temp, baseDir ); + strcat( temp, "\\" ); + strcat( temp, buffer ); + buffer = temp; + } + if( !directoryDoesExist( buffer ) ) + SHCreateDirectoryEx( NULL, buffer, NULL ); + + m_batteryPath.GetWindowText(buffer); + if( !buffer.IsEmpty() ) + regSetStringValue( "batteryDir", buffer ); + if( buffer[0] == '.' ) { + strcpy( temp, baseDir ); + strcat( temp, "\\" ); + strcat( temp, buffer ); + buffer = temp; + } + if( !directoryDoesExist( buffer ) ) + SHCreateDirectoryEx( NULL, buffer, NULL ); + + m_savePath.GetWindowText(buffer); + if( !buffer.IsEmpty() ) + regSetStringValue( "saveDir", buffer ); + if( buffer[0] == '.' ) { + strcpy( temp, baseDir ); + strcat( temp, "\\" ); + strcat( temp, buffer ); + buffer = temp; + } + if( !directoryDoesExist( buffer ) ) + SHCreateDirectoryEx( NULL, buffer, NULL ); + + m_capturePath.GetWindowText(buffer); + if( !buffer.IsEmpty() ) + regSetStringValue( "captureDir", buffer ); + if( buffer[0] == '.' ) { + strcpy( temp, baseDir ); + strcat( temp, "\\" ); + strcat( temp, buffer ); + buffer = temp; + } + if( !directoryDoesExist( buffer ) ) + SHCreateDirectoryEx( NULL, buffer, NULL ); + + EndDialog(TRUE); } CString Directories::browseForDir(CString title) diff --git a/src/win32/Directories.h b/src/win32/Directories.h index 50087355..8479694e 100644 --- a/src/win32/Directories.h +++ b/src/win32/Directories.h @@ -75,7 +75,24 @@ class Directories : public CDialog virtual void OnOK(); //}}AFX_MSG DECLARE_MESSAGE_MAP() - }; +private: + + bool directoryDoesExist(const char *directory) + { // returns true if the directory does exist + HANDLE hDir; + hDir = CreateFile( + directory, + GENERIC_READ, + FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE, + NULL, + OPEN_EXISTING, + FILE_FLAG_BACKUP_SEMANTICS, + NULL ); + bool retval = (hDir == INVALID_HANDLE_VALUE) ? false : true; + CloseHandle( hDir ); + return retval; + } +}; //{{AFX_INSERT_LOCATION}} // Microsoft Visual C++ will insert additional declarations immediately before the previous line. diff --git a/src/win32/Disassemble.cpp b/src/win32/Disassemble.cpp index fab2d533..3c4eaaa0 100644 --- a/src/win32/Disassemble.cpp +++ b/src/win32/Disassemble.cpp @@ -1,6 +1,6 @@ // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -118,6 +118,7 @@ void Disassemble::OnAutomatic() void Disassemble::OnArm() { mode = 1; + address&=0xfffffffC; refresh(); } @@ -133,21 +134,30 @@ void Disassemble::OnGo() CString buffer; m_address.GetWindowText(buffer); sscanf(buffer, "%x", &address); + if (mode==1) + address&=0xfffffffc; + else if (mode==2) + address&=0xfffffffe; refresh(); } void Disassemble::OnGopc() { + if(rom != NULL) + { if(armState) address = armNextPC - 16; else address = armNextPC - 8; refresh(); + } } void Disassemble::OnNext() { + if(rom != NULL) + { CPULoop(1); if(armState) { u32 total = address+count*4; @@ -163,6 +173,7 @@ void Disassemble::OnNext() } } refresh(); + } } void Disassemble::OnRefresh() @@ -173,6 +184,7 @@ void Disassemble::OnRefresh() void Disassemble::OnThumb() { mode = 2; + address&=0xfffffffe; refresh(); } diff --git a/src/win32/Display.h b/src/win32/Display.h index b63005e9..a44885ab 100644 --- a/src/win32/Display.h +++ b/src/win32/Display.h @@ -17,6 +17,8 @@ // along with this program; if not, write to the Free Software Foundation, // Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. +#pragma once + enum DISPLAY_TYPE { GDI = 0, DIRECT_DRAW = 1, @@ -32,13 +34,15 @@ class IDisplay { virtual bool initialize() = 0; virtual void cleanup() = 0; virtual void render() = 0; - virtual void checkFullScreen() { }; - virtual void renderMenu() { }; + virtual void checkFullScreen() {}; + virtual void renderMenu() {}; virtual void clear() = 0; virtual bool changeRenderSize(int w, int h) { return true; }; virtual void resize(int w, int h) {}; - virtual void setOption(const char *option, int value) = 0; + virtual void setOption(const char *option, int value) {}; virtual DISPLAY_TYPE getType() = 0; + virtual bool isSkinSupported() { return false; } virtual int selectFullScreenMode(GUID **) = 0; - virtual int selectFullScreenMode2() { return 0; }; }; + +void copyImage( void *source, void *destination, unsigned int width, unsigned int height, unsigned int destinationPitch, unsigned int colorDepth ); diff --git a/src/win32/FileDlg.cpp b/src/win32/FileDlg.cpp index 9cee23e1..14e30c9a 100644 --- a/src/win32/FileDlg.cpp +++ b/src/win32/FileDlg.cpp @@ -1,6 +1,6 @@ // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -18,7 +18,7 @@ // FileDlg.cpp: implementation of the FileDlg class. // -////////////////////////////////////////////////////////////////////// + #include "stdafx.h" #include #include @@ -26,7 +26,7 @@ #include "VBA.h" #include "FileDlg.h" #include "../System.h" -#include "..\..\res\resource.h" +#include "resource.h" #ifdef _DEBUG #define new DEBUG_NEW @@ -159,20 +159,23 @@ void FileDlg::OnTypeChange(HWND hwnd) ASSERT(typeControl != NULL); - int sel = ::SendMessage(typeControl, CB_GETCURSEL, 0, 0); + LRESULT sel = ::SendMessage(typeControl, CB_GETCURSEL, 0, 0); ASSERT(sel != -1); LPCTSTR typeName = extensions[sel]; if(filename.GetLength() == 0) { - filename.Format("*%s", typeName); + if(strlen(typeName) != 0) + filename.Format("*%s", typeName); } else { - int index = filename.Find('.'); - if (index == -1) { - filename = filename + typeName; - } else { - filename = filename.Left(index) + typeName; + if(strlen(typeName) != 0) { + int index = filename.Find('.'); + if (index == -1) { + filename = filename + typeName; + } else { + filename = filename.Left(index) + typeName; + } } } SetWindowText(fileNameControl, filename); diff --git a/src/win32/GBACheats.cpp b/src/win32/GBACheats.cpp index f36b3101..d7ee5866 100644 --- a/src/win32/GBACheats.cpp +++ b/src/win32/GBACheats.cpp @@ -683,7 +683,7 @@ bool AddCheat::addCheat() code.Format("%08x:%08x", address, value); break; } - + cheatsAdd(code, buffer, address ,address, value,-1, sizeType); return true; } @@ -765,16 +765,16 @@ void GBACheatList::OnEnable() if(mark != -1) { LVITEM item; for(int i = 0; i < count; i++) { - memset(&item,0, sizeof(item)); + memset(&item, 0, sizeof(item)); item.mask = LVIF_PARAM|LVIF_STATE; item.stateMask = LVIS_SELECTED; item.iItem = i; if(m_list.GetItem(&item)) { if(item.state & LVIS_SELECTED) { if(cheatsList[item.lParam].enabled) - cheatsDisable(item.lParam); + cheatsDisable((int)(item.lParam & 0xFFFFFFFF)); else - cheatsEnable(item.lParam); + cheatsEnable((int)(item.lParam & 0xFFFFFFFF)); } } } @@ -796,7 +796,7 @@ void GBACheatList::OnRemove() item.stateMask = LVIS_SELECTED; if(m_list.GetItem(&item)) { if(item.state & LVIS_SELECTED) { - cheatsDelete(item.lParam, restoreValues); + cheatsDelete((int)(item.lParam & 0xFFFFFFFF), restoreValues); } } } @@ -838,9 +838,9 @@ void GBACheatList::OnItemchangedCheatList(NMHDR* pNMHDR, LRESULT* pResult) if(((l->uOldState & LVIS_STATEIMAGEMASK)>>12) != (((l->uNewState & LVIS_STATEIMAGEMASK)>>12))) { if(m_list.GetCheck(l->iItem)) - cheatsEnable(l->lParam); + cheatsEnable((int)(l->lParam & 0xFFFFFFFF)); else - cheatsDisable(l->lParam); + cheatsDisable((int)(l->lParam & 0xFFFFFFFF)); refresh(); } } diff --git a/src/win32/GBACheats.h b/src/win32/GBACheats.h index 79a6f9ae..3066d871 100644 --- a/src/win32/GBACheats.h +++ b/src/win32/GBACheats.h @@ -20,7 +20,7 @@ #if !defined(AFX_GBACHEATS_H__FC31D47D_52C8_42B2_95C7_7C3FD09316A4__INCLUDED_) #define AFX_GBACHEATS_H__FC31D47D_52C8_42B2_95C7_7C3FD09316A4__INCLUDED_ -#include "..\System.h" // Added by ClassView +#include "../System.h" // Added by ClassView #if _MSC_VER > 1000 #pragma once #endif // _MSC_VER > 1000 diff --git a/src/win32/GBCheatsDlg.cpp b/src/win32/GBCheatsDlg.cpp index dc8c164f..6362eeaa 100644 --- a/src/win32/GBCheatsDlg.cpp +++ b/src/win32/GBCheatsDlg.cpp @@ -637,7 +637,7 @@ bool AddGBCheat::addCheat() m_desc.GetWindowText(buffer); - int bank = (address >> 16); + LONG_PTR bank = (address >> 16); address &= 0xFFFF; if(address >= 0xd000) @@ -689,9 +689,7 @@ BOOL AddGBCheat::OnInitDialog() buffer.Format("%02x:%08x", (address>>16), address&0xFFFF); m_address.SetWindowText(buffer); m_address.EnableWindow(FALSE); - ::SetWindowLong(m_address, - GWL_USERDATA, - address); + ::SetWindowLongPtr( m_address.GetSafeHwnd(), GWLP_USERDATA, address); numberType = regQueryDwordValue("gbCheatsNumberType", 2); if(numberType < 0 || numberType > 2) @@ -826,9 +824,9 @@ void GBCheatList::OnEnable() item.iItem = mark; if(m_list.GetItem(&item)) { if(gbCheatList[item.lParam].enabled) - gbCheatDisable(item.lParam); + gbCheatDisable((int)item.lParam); else - gbCheatEnable(item.lParam); + gbCheatEnable((int)item.lParam); refresh(); } } @@ -844,7 +842,7 @@ void GBCheatList::OnRemove() item.mask = LVIF_PARAM; item.iItem = mark; if(m_list.GetItem(&item)) { - gbCheatRemove(item.lParam); + gbCheatRemove((int)item.lParam); refresh(); } } @@ -872,9 +870,9 @@ void GBCheatList::OnItemchangedCheatList(NMHDR* pNMHDR, LRESULT* pResult) if(((l->uOldState & LVIS_STATEIMAGEMASK)>>12) != (((l->uNewState & LVIS_STATEIMAGEMASK)>>12))) { if(m_list.GetCheck(l->iItem)) - gbCheatEnable(l->lParam); + gbCheatEnable((int)l->lParam); else - gbCheatDisable(l->lParam); + gbCheatDisable((int)l->lParam); refresh(); } } diff --git a/src/win32/GBCheatsDlg.h b/src/win32/GBCheatsDlg.h index 0f3f1d43..68b9f71e 100644 --- a/src/win32/GBCheatsDlg.h +++ b/src/win32/GBCheatsDlg.h @@ -121,7 +121,7 @@ class AddGBCheat : public CDialog // Implementation protected: - u32 address; + LONG_PTR address; // Generated message map functions //{{AFX_MSG(AddGBCheat) diff --git a/src/win32/GBColorDlg.cpp b/src/win32/GBColorDlg.cpp index a2527aa6..6096e9d8 100644 --- a/src/win32/GBColorDlg.cpp +++ b/src/win32/GBColorDlg.cpp @@ -1,6 +1,6 @@ // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2004-2006 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -32,28 +32,31 @@ extern u16 gbPalette[128]; static u16 defaultPalettes[][24] = { { - 0x7FFF, 0x56B5, 0x318C, 0x0000, 0x7FFF, 0x56B5, 0x318C, 0x0000, + 0x7FFF, 0x56B5, 0x318C, 0x0000, 0x7FFF, 0x56B5, 0x318C, 0x0000, }, { - 0x6200, 0x7E10, 0x7C10, 0x5000, 0x6200, 0x7E10, 0x7C10, 0x5000, + 0x6200, 0x7E10, 0x7C10, 0x5000, 0x6200, 0x7E10, 0x7C10, 0x5000, }, { - 0x4008, 0x4000, 0x2000, 0x2008, 0x4008, 0x4000, 0x2000, 0x2008, + 0x4008, 0x4000, 0x2000, 0x2008, 0x4008, 0x4000, 0x2000, 0x2008, }, { - 0x43F0, 0x03E0, 0x4200, 0x2200, 0x43F0, 0x03E0, 0x4200, 0x2200, + 0x43F0, 0x03E0, 0x4200, 0x2200, 0x43F0, 0x03E0, 0x4200, 0x2200, }, { - 0x43FF, 0x03FF, 0x221F, 0x021F, 0x43FF, 0x03FF, 0x221F, 0x021F, + 0x43FF, 0x03FF, 0x221F, 0x021F, 0x43FF, 0x03FF, 0x221F, 0x021F, }, { - 0x621F, 0x7E1F, 0x7C1F, 0x2010, 0x621F, 0x7E1F, 0x7C1F, 0x2010, + 0x621F, 0x7E1F, 0x7C1F, 0x2010, 0x621F, 0x7E1F, 0x7C1F, 0x2010, }, { - 0x621F, 0x401F, 0x001F, 0x2010, 0x621F, 0x401F, 0x001F, 0x2010, + 0x621F, 0x401F, 0x001F, 0x2010, 0x621F, 0x401F, 0x001F, 0x2010, }, { - 0x421F, 0x03E0, 0x7C00, 0x401F, 0x021F, 0x2200, 0x4008, 0x2010, + 0x1B8E, 0x02C0, 0x0DA0, 0x1140, 0x1B8E, 0x02C0, 0x0DA0, 0x1140, + }, + { + 0x7BDE, /*0x23F0*/ 0x5778, /*0x5DC0*/ 0x5640, 0x0000, 0x7BDE, /*0x3678*/ 0x529C, /*0x0980*/ 0x2990, 0x0000, } }; @@ -70,20 +73,15 @@ static char THIS_FILE[] = __FILE__; GBColorDlg::GBColorDlg(CWnd* pParent /*=NULL*/) : CDialog(GBColorDlg::IDD, pParent) { - //{{AFX_DATA_INIT(GBColorDlg) - which = -1; - //}}AFX_DATA_INIT which = gbPaletteOption; } void GBColorDlg::DoDataExchange(CDataExchange* pDX) { - CDialog::DoDataExchange(pDX); - //{{AFX_DATA_MAP(GBColorDlg) + CDialog::DoDataExchange(pDX); DDX_Control(pDX, IDC_PREDEFINED, m_predefined); - DDX_Radio(pDX, IDC_DEFAULT, which); - //}}AFX_DATA_MAP + DDX_Radio(pDX, IDC_DEFAULT, which); } @@ -95,8 +93,8 @@ BEGIN_MESSAGE_MAP(GBColorDlg, CDialog) ON_BN_CLICKED(IDC_USER2, OnUser2) ON_BN_CLICKED(ID_OK, OnOk) ON_BN_CLICKED(ID_CANCEL, OnCancel) - ON_CBN_SELCHANGE(IDC_PREDEFINED, OnSelchangePredefined) - //}}AFX_MSG_MAP + ON_CBN_SELCHANGE(IDC_PREDEFINED, OnSelchangePredefined) + //}}AFX_MSG_MAP ON_CONTROL_RANGE(BN_CLICKED, IDC_COLOR_BG0, IDC_COLOR_OB3, OnColorClicked) END_MESSAGE_MAP() @@ -167,10 +165,12 @@ BOOL GBColorDlg::OnInitDialog() "Green Forest", "Hot Desert", "Pink Dreams", - "Weird Colors" + "Weird Colors", + "Real GB Colors", + "Real 'GB on GBASP' Colors" }; - for(int j = 0; j < 7; j++) { + for(int j = 0; j < 9; j++) { int index = m_predefined.AddString(names[j]); m_predefined.SetItemData(index, j); } @@ -222,15 +222,20 @@ void GBColorDlg::OnColorClicked(UINT id) { id -= IDC_COLOR_BG0; - u16 color = colors[id]; + u16 color = colors[which*8+id]; - CColorDialog dlg(RGB(color & 0x1f, (color >> 5) & 0x1f, (color >> 10) & 0x1f), + COLORREF colorInit = + RGB((color & 0x1f) << 3, ((color >> 5) & 0x1f) << 3, ((color >> 10) & 0x1f) << 3); + + CColorDialog dlg(colorInit, CC_FULLOPEN | CC_ANYCOLOR, this); - if(dlg.DoModal()) { + + if(IDOK == dlg.DoModal()) + { COLORREF c = dlg.GetColor(); - colors[which*8+id] = (u16)((c >> 3) & 0x1f | ((c >> 11) & 0x1f) << 5 | - ((c >> 19) & 0x1f) << 10); + colors[which*8+id] = (u16)((c >> 3) & 0x1f | ((c >> 11) & 0x1f) << 5 | ((c >> 19) & 0x1f) << 10); + colorControls[id].setColor(colors[which*8+id]); } } @@ -246,7 +251,7 @@ void GBColorDlg::OnSelchangePredefined() int sel = m_predefined.GetCurSel(); if(sel != -1) { - int data = m_predefined.GetItemData(sel); + DWORD_PTR data = m_predefined.GetItemData(sel); for(int i = 0; i < 8; i++) { colorControls[i].setColor(defaultPalettes[data][i]); colors[which*8+i] = defaultPalettes[data][i]; diff --git a/src/win32/GBColorDlg.h b/src/win32/GBColorDlg.h index d19c3082..f94576a2 100644 --- a/src/win32/GBColorDlg.h +++ b/src/win32/GBColorDlg.h @@ -21,7 +21,7 @@ #define AFX_GBCOLORDLG_H__8D6126EF_06BB_48CF_ABB3_2CC4B1B60358__INCLUDED_ #include "ColorButton.h" // Added by ClassView -#include "..\System.h" // Added by ClassView +#include "../System.h" // Added by ClassView #if _MSC_VER > 1000 #pragma once #endif // _MSC_VER > 1000 diff --git a/src/win32/GBDisassemble.cpp b/src/win32/GBDisassemble.cpp index 6eee4ec7..4a00d827 100644 --- a/src/win32/GBDisassemble.cpp +++ b/src/win32/GBDisassemble.cpp @@ -1,6 +1,6 @@ // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -39,6 +39,7 @@ extern gbRegister DE; extern gbRegister HL; extern gbRegister SP; extern gbRegister PC; +extern u8 register_LY; extern u16 IFF; extern int gbDis(char *, u16); @@ -114,8 +115,9 @@ void GBDisassemble::OnNext() void GBDisassemble::OnGo() { CString buffer; + m_address.GetWindowText(buffer); - sscanf(buffer, "%x", &address); + sscanf(buffer, "%hx", &address); refresh(); } @@ -242,6 +244,8 @@ void GBDisassemble::refresh() GetDlgItem(IDC_R5)->SetWindowText(buffer); sprintf(buffer, "%04x", IFF); GetDlgItem(IDC_R6)->SetWindowText(buffer); + sprintf(buffer, "%04x", register_LY); + GetDlgItem(IDC_LY)->SetWindowText(buffer); m_z = (AF.B.B0 & 0x80) != 0; m_n = (AF.B.B0 & 0x40) != 0; diff --git a/src/win32/GBDisassemble.h b/src/win32/GBDisassemble.h index 0a37a6b0..31fb6963 100644 --- a/src/win32/GBDisassemble.h +++ b/src/win32/GBDisassemble.h @@ -20,7 +20,7 @@ #if !defined(AFX_GBDISASSEMBLE_H__3EFD5B47_6DBF_4F63_8F91_A9511EC590EB__INCLUDED_) #define AFX_GBDISASSEMBLE_H__3EFD5B47_6DBF_4F63_8F91_A9511EC590EB__INCLUDED_ -#include "..\System.h" // Added by ClassView +#include "../System.h" // Added by ClassView #if _MSC_VER > 1000 #pragma once #endif // _MSC_VER > 1000 diff --git a/src/win32/GBMapView.cpp b/src/win32/GBMapView.cpp index c3cd7149..47ee202e 100644 --- a/src/win32/GBMapView.cpp +++ b/src/win32/GBMapView.cpp @@ -499,8 +499,8 @@ LRESULT GBMapView::OnMapInfo(WPARAM wParam, LPARAM lParam) u8 *colors = (u8 *)lParam; mapViewZoom.setColors(colors); - int x = wParam & 0xffff; - int y = (wParam >> 16); + int x = (int)(wParam & 0xffff); + int y = (int)(wParam >> 16); CString buffer; buffer.Format("(%d,%d)", x, y); diff --git a/src/win32/GBMapView.h b/src/win32/GBMapView.h index 2b2b006a..c429a4dc 100644 --- a/src/win32/GBMapView.h +++ b/src/win32/GBMapView.h @@ -31,7 +31,7 @@ #include "ZoomControl.h" #include "ResizeDlg.h" #include "IUpdate.h" -#include "..\System.h" // Added by ClassView +#include "../System.h" // Added by ClassView ///////////////////////////////////////////////////////////////////////////// // GBMapView dialog diff --git a/src/win32/GBPaletteView.cpp b/src/win32/GBPaletteView.cpp index fa9e57f2..1c6a4719 100644 --- a/src/win32/GBPaletteView.cpp +++ b/src/win32/GBPaletteView.cpp @@ -135,8 +135,8 @@ void GBPaletteView::save(int which) if(dlg.DoModal() == IDCANCEL) { return; } - - captureBuffer = dlg.GetPathName(); + + captureBuffer = dlg.GetPathName(); PaletteViewControl *p = NULL; diff --git a/src/win32/GBTileView.cpp b/src/win32/GBTileView.cpp index 69bbe326..23c44800 100644 --- a/src/win32/GBTileView.cpp +++ b/src/win32/GBTileView.cpp @@ -460,8 +460,8 @@ LRESULT GBTileView::OnMapInfo(WPARAM wParam, LPARAM lParam) u8 *colors = (u8 *)lParam; zoom.setColors(colors); - int x = (wParam & 0xFFFF)/8; - int y = ((wParam >> 16) & 0xFFFF)/8; + int x = (int)((wParam & 0xffff)/8); + int y = (int)(((wParam >> 16) & 0xFFFF)/8); int tiles = 0x0000; if(charBase) diff --git a/src/win32/GDIDisplay.cpp b/src/win32/GDIDisplay.cpp index dc179804..22431c33 100644 --- a/src/win32/GDIDisplay.cpp +++ b/src/win32/GDIDisplay.cpp @@ -1,479 +1,402 @@ -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2005 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#include "stdafx.h" -#include - -#include "../System.h" -#include "../GBA.h" -#include "../Globals.h" -#include "..\gb\gbGlobals.h" -#include "../Text.h" -#include "../Util.h" -#include "UniVideoModeDlg.h" - -#include "VBA.h" -#include "MainWnd.h" -#include "Reg.h" -#include "..\..\res\resource.h" - -#include "../gbafilter.h" - -#ifdef _DEBUG -#define new DEBUG_NEW -#undef THIS_FILE -static char THIS_FILE[] = __FILE__; -#endif - -extern void winlog(const char *,...); -extern int Init_2xSaI(u32); -extern int systemSpeed; -extern int winVideoModeSelect(CWnd *, GUID **); - - -class GDIDisplay : public IDisplay -{ -private: - u8 *filterData; - u8 info[sizeof(BITMAPINFOHEADER)+256*sizeof(RGBQUAD)]; - int SelectedFreq, SelectedAdapter; -public: - GDIDisplay(); - virtual ~GDIDisplay(); - - virtual bool changeRenderSize(int w, int h); - virtual bool initialize(); - virtual void cleanup(); - virtual void render(); - virtual void checkFullScreen(); - virtual void renderMenu(); - virtual void clear(); - virtual DISPLAY_TYPE getType() { return GDI; }; - virtual void setOption(const char *, int) {} - virtual int selectFullScreenMode(GUID **); - virtual int selectFullScreenMode2(); -}; - - -static int calculateShift(u32 mask) -{ - int m = 0; - - while(mask) { - m++; - mask >>= 1; - } - - return m-5; -} - -GDIDisplay::GDIDisplay() -{ - filterData = NULL; -} - -GDIDisplay::~GDIDisplay() -{ - cleanup(); -} - -void GDIDisplay::cleanup() -{ - if(filterData) - { - delete [] filterData; - filterData = NULL; - } -} - -bool GDIDisplay::initialize() -{ - switch(theApp.cartridgeType) - { - case 0: - theApp.sizeX = 240; - theApp.sizeY = 160; - break; - case 1: - if(gbBorderOn) - { - theApp.sizeX = 256; - theApp.sizeY = 224; - } - else - { - theApp.sizeX = 160; - theApp.sizeY = 144; - } - break; - } - - switch(theApp.videoOption) - { - case VIDEO_1X: - theApp.surfaceSizeX = theApp.sizeX; - theApp.surfaceSizeY = theApp.sizeY; - break; - case VIDEO_2X: - theApp.surfaceSizeX = theApp.sizeX * 2; - theApp.surfaceSizeY = theApp.sizeY * 2; - break; - case VIDEO_3X: - theApp.surfaceSizeX = theApp.sizeX * 3; - theApp.surfaceSizeY = theApp.sizeY * 3; - break; - case VIDEO_4X: - theApp.surfaceSizeX = theApp.sizeX * 4; - theApp.surfaceSizeY = theApp.sizeY * 4; - break; - case VIDEO_320x240: - case VIDEO_640x480: - case VIDEO_800x600: - case VIDEO_1024x768: - case VIDEO_1280x1024: - case VIDEO_OTHER: - float scaleX = ((float)theApp.fsWidth / theApp.sizeX); - float scaleY = ((float)theApp.fsHeight / theApp.sizeY); - float min = scaleX < scaleY ? scaleX : scaleY; - if(theApp.fsMaxScale) - min = min > theApp.fsMaxScale ? theApp.fsMaxScale : min; - if(theApp.fullScreenStretch) - { - theApp.surfaceSizeX = theApp.fsWidth; - theApp.surfaceSizeY = theApp.fsHeight; - } - else - { - theApp.surfaceSizeX = (int)(theApp.sizeX * min); - theApp.surfaceSizeY = (int)(theApp.sizeY * min); - } - break; - } - - theApp.rect.left = 0; - theApp.rect.top = 0; - theApp.rect.right = theApp.sizeX; - theApp.rect.bottom = theApp.sizeY; - - theApp.dest.left = 0; - theApp.dest.top = 0; - theApp.dest.right = theApp.surfaceSizeX; - theApp.dest.bottom = theApp.surfaceSizeY; - - DWORD style = WS_POPUP | WS_VISIBLE; - DWORD styleEx = 0; - - if(theApp.videoOption <= VIDEO_4X) - style |= WS_OVERLAPPEDWINDOW; - else - styleEx = 0; - - if(theApp.videoOption <= VIDEO_4X) - AdjustWindowRectEx(&theApp.dest, style, TRUE, styleEx); - else - AdjustWindowRectEx(&theApp.dest, style, FALSE, styleEx); - - int winSizeX = theApp.dest.right-theApp.dest.left; - int winSizeY = theApp.dest.bottom-theApp.dest.top; - - if(theApp.videoOption > VIDEO_4X) { - winSizeX = theApp.fsWidth; - winSizeY = theApp.fsHeight; - } - - int x = 0; - int y = 0; - - if(theApp.videoOption <= VIDEO_4X) { - x = theApp.windowPositionX; - y = theApp.windowPositionY; - } - - // Create a window - MainWnd *pWnd = new MainWnd; - theApp.m_pMainWnd = pWnd; - - pWnd->CreateEx(styleEx, - theApp.wndClass, - "VisualBoyAdvance", - style, - x,y,winSizeX,winSizeY, - NULL, - 0); - - if (!(HWND)*pWnd) { - winlog("Error creating Window %08x\n", GetLastError()); - return FALSE; - } - - theApp.updateMenuBar(); - - theApp.adjustDestRect(); - - // Enumerate available display modes - theApp.mode320Available = false; - theApp.mode640Available = false; - theApp.mode800Available = false; - theApp.mode1024Available = false; - theApp.mode1280Available = false; - DISPLAY_DEVICE dev; - dev.cb = sizeof(DISPLAY_DEVICE); - EnumDisplayDevices(NULL, 0, &dev, 0); - DEVMODE mode; - for (DWORD iMode = 0; - TRUE == EnumDisplaySettings(dev.DeviceName, iMode, &mode); - iMode++) - { - if ( (mode.dmBitsPerPel == 16) && - (mode.dmPelsWidth==320) && (mode.dmPelsHeight==240)) - theApp.mode320Available = true; - - if ( (mode.dmBitsPerPel == 16) && - (mode.dmPelsWidth==640) && (mode.dmPelsHeight==480)) - theApp.mode640Available = true; - - if ( (mode.dmBitsPerPel == 16) && - (mode.dmPelsWidth==800) && (mode.dmPelsHeight==600)) - theApp.mode800Available = true; - - if ( (mode.dmBitsPerPel == 32) && - (mode.dmPelsWidth==1024) && (mode.dmPelsHeight==768)) - theApp.mode1024Available = true; - - if ( (mode.dmBitsPerPel == 32) && - (mode.dmPelsWidth==1280) && (mode.dmPelsHeight==1024)) - theApp.mode1280Available = true; - } - - // Go into fullscreen - if(theApp.videoOption >= VIDEO_320x240) - { - mode.dmBitsPerPel = theApp.fsColorDepth; - mode.dmPelsWidth = theApp.fsWidth; - mode.dmPelsHeight = theApp.fsHeight; - mode.dmDisplayFrequency = theApp.fsFrequency; - mode.dmFields = DM_BITSPERPEL | DM_PELSWIDTH | DM_PELSHEIGHT | DM_DISPLAYFREQUENCY; - DISPLAY_DEVICE dd; - dd.cb = sizeof(DISPLAY_DEVICE); - EnumDisplayDevices(NULL, theApp.fsAdapter, &dd, 0); - ChangeDisplaySettingsEx(dd.DeviceName, &mode, NULL, CDS_FULLSCREEN, NULL); - } - else // Reset from fullscreen - { - ChangeDisplaySettings(NULL, 0); - } - - - // Initialize 2xSaI - HDC dc = GetDC(NULL); - HBITMAP hbm = CreateCompatibleBitmap(dc, 1, 1); - BITMAPINFO *bi = (BITMAPINFO *)info; - ZeroMemory(bi, sizeof(info)); - bi->bmiHeader.biSize = sizeof(BITMAPINFOHEADER); - GetDIBits(dc, hbm, 0, 1, NULL, (LPBITMAPINFO)info, DIB_RGB_COLORS); - GetDIBits(dc, hbm, 0, 1, NULL, (LPBITMAPINFO)info, DIB_RGB_COLORS); - DeleteObject(hbm); - ReleaseDC(NULL, dc); - - if(bi->bmiHeader.biCompression == BI_BITFIELDS) { - systemColorDepth = bi->bmiHeader.biBitCount; - if(systemColorDepth == 15) - systemColorDepth = 16; - systemRedShift = calculateShift(*((DWORD *)&bi->bmiColors[0])); - systemGreenShift = calculateShift(*((DWORD *)&bi->bmiColors[1])); - systemBlueShift = calculateShift(*((DWORD *)&bi->bmiColors[2])); - if(systemColorDepth == 16) { - if(systemGreenShift == 6) { - Init_2xSaI(565); - } else { - Init_2xSaI(555); - } - } else if(systemColorDepth == 32) - Init_2xSaI(32); - } else { - systemColorDepth = 32; - systemRedShift = 19; - systemGreenShift = 11; - systemBlueShift = 3; - - Init_2xSaI(32); - } - - - // Setup system color depth - theApp.fsColorDepth = systemColorDepth; - if(systemColorDepth == 24) - theApp.filterFunction = NULL; -#ifdef MMX - if(!theApp.disableMMX) - cpu_mmx = theApp.detectMMX(); - else - cpu_mmx = 0; -#endif - - utilUpdateSystemColorMaps(theApp.filterLCD ); - theApp.updateFilter(); - theApp.updateIFB(); - - pWnd->DragAcceptFiles(TRUE); - - return TRUE; -} - - -void GDIDisplay::clear() -{ - CDC *dc = theApp.m_pMainWnd->GetDC(); - CBrush brush(RGB(0x00, 0x00, 0x00)); - dc->FillRect(CRect(0, 0, theApp.fsWidth, theApp.fsHeight), &brush); - theApp.m_pMainWnd->ReleaseDC(dc); -} - -void GDIDisplay::renderMenu() -{ - checkFullScreen(); - theApp.m_pMainWnd->DrawMenuBar(); -} - -void GDIDisplay::checkFullScreen() -{ -} - -void GDIDisplay::render() -{ - unsigned int pitch = theApp.filterWidth * (systemColorDepth / 8) + 4; - - BITMAPINFO *bi = (BITMAPINFO *)info; - bi->bmiHeader.biWidth = theApp.filterWidth + 1; - bi->bmiHeader.biHeight = -theApp.filterHeight; - - if(theApp.filterFunction) - { - bi->bmiHeader.biWidth = theApp.rect.right; - bi->bmiHeader.biHeight = -(int)theApp.rect.bottom; - - (*theApp.filterFunction)( - pix + pitch, - pitch, - (u8*)theApp.delta, - (u8*)filterData, - theApp.rect.right * (systemColorDepth / 8), - theApp.filterWidth, - theApp.filterHeight); - } - - POINT p1, p2; - p1.x = theApp.dest.left; - p1.y = theApp.dest.top; - p2.x = theApp.dest.right; - p2.y = theApp.dest.bottom; - theApp.m_pMainWnd->ScreenToClient(&p1); - theApp.m_pMainWnd->ScreenToClient(&p2); - - CDC *dc = theApp.m_pMainWnd->GetDC(); - - // Draw bitmap to device - StretchDIBits( - dc->GetSafeHdc(), - p1.x, p1.y, - p2.x - p1.x, - p2.y - p1.y, - theApp.rect.left, theApp.rect.top, - theApp.rect.right - theApp.rect.left, - theApp.rect.bottom - theApp.rect.top, - theApp.filterFunction ? filterData : pix + pitch, - bi, - DIB_RGB_COLORS, - SRCCOPY); - - // Draw frame counter - if (theApp.showSpeed && (theApp.videoOption >= VIDEO_320x240)) - { - CString speedText; - if (theApp.showSpeed == 1) - speedText.AppendFormat("%3d%%", systemSpeed); - else - speedText.AppendFormat("%3d%%(%d, %d fps)", - systemSpeed, systemFrameSkip, theApp.showRenderedFrames); - - dc->SetTextColor(RGB(0xFF, 0x3F, 0x3F)); - if (theApp.showSpeedTransparent) - dc->SetBkMode(TRANSPARENT); - else - dc->SetBkMode(OPAQUE); - dc->SetBkColor(RGB(0xFF, 0xFF, 0xFF)); - dc->TextOut(p1.x + 16, p1.y + 16, speedText); - } - - // Draw screen message - if (theApp.screenMessage) - { - if ( ((GetTickCount() - theApp.screenMessageTime) < 3000) && !theApp.disableStatusMessage ) - { - dc->SetTextColor(RGB(0x3F, 0x3F, 0xFF)); - if (theApp.showSpeedTransparent) - dc->SetBkMode(TRANSPARENT); - else - dc->SetBkMode(OPAQUE); - dc->SetBkColor(RGB(0xFF, 0xFF, 0xFF)); - dc->TextOut(p1.x + 16, p2.y - 16, theApp.screenMessageBuffer); - } - else - theApp.screenMessage = false; - } - - theApp.m_pMainWnd->ReleaseDC(dc); -} - -int GDIDisplay::selectFullScreenMode(GUID **pGUID) -{ - int w, h, b; - UniVideoModeDlg dlg(0, &w, &h, &b, &SelectedFreq, &SelectedAdapter); - - if (0 == dlg.DoModal()) - { - return (b<<24) + (w<<12) + h; - // Bits<<24 | Width<<12 | Height - } - else - { - return -1; - } -} - -int GDIDisplay::selectFullScreenMode2() -{ - return (SelectedAdapter<<16) + SelectedFreq; -} - -bool GDIDisplay::changeRenderSize(int w, int h) -{ - if (filterData) - { - delete [] filterData; - filterData = NULL; - } - filterData = new u8[w*h*(systemColorDepth>>3)]; - return true; -} - - -IDisplay *newGDIDisplay() -{ - return new GDIDisplay(); -} \ No newline at end of file +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005-2006 VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#include "stdafx.h" + +#include "Display.h" + +#include "../System.h" +#include "../GBA.h" +#include "../Globals.h" +#include "../Text.h" +#include "../Util.h" + +#include "VBA.h" +#include "MainWnd.h" +#include "Reg.h" +#include "resource.h" + +#ifdef _DEBUG +#define new DEBUG_NEW +#undef THIS_FILE +static char THIS_FILE[] = __FILE__; +#endif + +extern void winlog(const char *,...); +extern int Init_2xSaI(u32); +extern int systemSpeed; + +class GDIDisplay : public IDisplay { +private: + u8 *filterData; + u8 info[sizeof(BITMAPINFOHEADER)+256*sizeof(RGBQUAD)]; + +public: + GDIDisplay(); + virtual ~GDIDisplay(); + + virtual bool initialize(); + virtual void cleanup(); + virtual void render(); + virtual void checkFullScreen(); + virtual void renderMenu(); + virtual void clear(); + virtual DISPLAY_TYPE getType() { return GDI; }; + virtual void setOption(const char *, int) {} + virtual bool isSkinSupported() { return true; } + virtual int selectFullScreenMode(GUID **); +}; + +static int calculateShift(u32 mask) +{ + int m = 0; + + while(mask) { + m++; + mask >>= 1; + } + + return m-5; +} + +GDIDisplay::GDIDisplay() +{ + filterData = (u8 *)malloc(4*4*256*240); +} + +GDIDisplay::~GDIDisplay() +{ + cleanup(); +} + +void GDIDisplay::cleanup() +{ + if(filterData) { + free(filterData); + filterData = NULL; + } +} + +bool GDIDisplay::initialize() +{ + theApp.sizeX = 240; + theApp.sizeY = 160; + switch(theApp.videoOption) { + case VIDEO_1X: + theApp.surfaceSizeX = theApp.sizeX; + theApp.surfaceSizeY = theApp.sizeY; + break; + case VIDEO_2X: + theApp.surfaceSizeX = theApp.sizeX * 2; + theApp.surfaceSizeY = theApp.sizeY * 2; + break; + case VIDEO_3X: + theApp.surfaceSizeX = theApp.sizeX * 3; + theApp.surfaceSizeY = theApp.sizeY * 3; + break; + case VIDEO_4X: + theApp.surfaceSizeX = theApp.sizeX * 4; + theApp.surfaceSizeY = theApp.sizeY * 4; + break; + case VIDEO_320x240: + case VIDEO_640x480: + case VIDEO_800x600: + case VIDEO_OTHER: + { + int scaleX = (theApp.fsWidth / theApp.sizeX); + int scaleY = (theApp.fsHeight / theApp.sizeY); + int min = scaleX < scaleY ? scaleX : scaleY; + if(theApp.fsMaxScale) + min = min > theApp.fsMaxScale ? theApp.fsMaxScale : min; + theApp.surfaceSizeX = theApp.sizeX * min; + theApp.surfaceSizeY = theApp.sizeY * min; + if(theApp.fullScreenStretch) { + theApp.surfaceSizeX = theApp.fsWidth; + theApp.surfaceSizeY = theApp.fsHeight; + } + } + break; + } + + theApp.rect.left = 0; + theApp.rect.top = 0; + theApp.rect.right = theApp.sizeX; + theApp.rect.bottom = theApp.sizeY; + + theApp.dest.left = 0; + theApp.dest.top = 0; + theApp.dest.right = theApp.surfaceSizeX; + theApp.dest.bottom = theApp.surfaceSizeY; + + DWORD style = WS_POPUP | WS_VISIBLE; + DWORD styleEx = 0; + + if(theApp.videoOption <= VIDEO_4X) + style |= WS_OVERLAPPEDWINDOW; + else + styleEx = 0; + + if(theApp.videoOption <= VIDEO_4X) + AdjustWindowRectEx(&theApp.dest, style, TRUE, styleEx); + else + AdjustWindowRectEx(&theApp.dest, style, FALSE, styleEx); + + int winSizeX = theApp.dest.right-theApp.dest.left; + int winSizeY = theApp.dest.bottom-theApp.dest.top; + + if(theApp.videoOption > VIDEO_4X) { + winSizeX = theApp.fsWidth; + winSizeY = theApp.fsHeight; + } + + int x = 0; + int y = 0; + + if(theApp.videoOption <= VIDEO_4X) { + x = theApp.windowPositionX; + y = theApp.windowPositionY; + } + + // Create a window + MainWnd *pWnd = new MainWnd; + theApp.m_pMainWnd = pWnd; + + pWnd->CreateEx(styleEx, + theApp.wndClass, + "VisualBoyAdvance", + style, + x,y,winSizeX,winSizeY, + NULL, + 0); + + if (!(HWND)*pWnd) { + winlog("Error creating Window %08x\n", GetLastError()); + return FALSE; + } + + theApp.updateMenuBar(); + + theApp.adjustDestRect(); + + theApp.mode320Available = false; + theApp.mode640Available = false; + theApp.mode800Available = false; + + HDC dc = GetDC(NULL); + HBITMAP hbm = CreateCompatibleBitmap(dc, 1, 1); + BITMAPINFO *bi = (BITMAPINFO *)info; + ZeroMemory(bi, sizeof(info)); + bi->bmiHeader.biSize = sizeof(BITMAPINFOHEADER); + GetDIBits(dc, hbm, 0, 1, NULL, (LPBITMAPINFO)info, DIB_RGB_COLORS); + GetDIBits(dc, hbm, 0, 1, NULL, (LPBITMAPINFO)info, DIB_RGB_COLORS); + DeleteObject(hbm); + ReleaseDC(NULL, dc); + + if(bi->bmiHeader.biCompression == BI_BITFIELDS) { + systemColorDepth = bi->bmiHeader.biBitCount; + if(systemColorDepth == 15) + systemColorDepth = 16; + systemRedShift = calculateShift(*((DWORD *)&bi->bmiColors[0])); + systemGreenShift = calculateShift(*((DWORD *)&bi->bmiColors[1])); + systemBlueShift = calculateShift(*((DWORD *)&bi->bmiColors[2])); + if(systemColorDepth == 16) { + if(systemGreenShift == 6) { + Init_2xSaI(565); + } else { + Init_2xSaI(555); + } + } else if(systemColorDepth == 32) + Init_2xSaI(32); + } else { + systemColorDepth = 32; + systemRedShift = 19; + systemGreenShift = 11; + systemBlueShift = 3; + + Init_2xSaI(32); + } + theApp.fsColorDepth = systemColorDepth; + if(systemColorDepth == 24) + theApp.filterFunction = NULL; +#ifdef MMX + if(!theApp.disableMMX) + cpu_mmx = theApp.detectMMX(); + else + cpu_mmx = 0; +#endif + + utilUpdateSystemColorMaps(); + theApp.updateFilter(); + theApp.updateIFB(); + + pWnd->DragAcceptFiles(TRUE); + + return TRUE; +} + +void GDIDisplay::clear() +{ +} + +void GDIDisplay::renderMenu() +{ + checkFullScreen(); + theApp.m_pMainWnd->DrawMenuBar(); +} + +void GDIDisplay::checkFullScreen() +{ +} + +void GDIDisplay::render() +{ + BITMAPINFO *bi = (BITMAPINFO *)info; + bi->bmiHeader.biWidth = theApp.filterWidth+1; + bi->bmiHeader.biHeight = -theApp.filterHeight; + + int pitch = theApp.filterWidth * 2 + 4; + if(systemColorDepth == 24) + pitch = theApp.filterWidth * 3; + else if(systemColorDepth == 32) + pitch = theApp.filterWidth * 4 + 4; + + if(theApp.filterFunction) { + bi->bmiHeader.biWidth = theApp.filterWidth * 2; + bi->bmiHeader.biHeight = -theApp.filterHeight * 2; + + if(systemColorDepth == 16) + (*theApp.filterFunction)(pix+pitch, + pitch, + (u8*)theApp.delta, + (u8*)filterData, + theApp.filterWidth*2*2, + theApp.filterWidth, + theApp.filterHeight); + else + (*theApp.filterFunction)(pix+pitch, + pitch, + (u8*)theApp.delta, + (u8*)filterData, + theApp.filterWidth*4*2, + theApp.filterWidth, + theApp.filterHeight); + } + + if(theApp.showSpeed && (theApp.videoOption > VIDEO_4X || theApp.skin != NULL)) { + char buffer[30]; + if(theApp.showSpeed == 1) + sprintf(buffer, "%3d%%", systemSpeed); + else + sprintf(buffer, "%3d%%(%d, %d fps)", systemSpeed, + systemFrameSkip, + theApp.showRenderedFrames); + + if(theApp.filterFunction) { + int p = theApp.filterWidth * 4; + if(systemColorDepth == 24) + p = theApp.filterWidth * 6; + else if(systemColorDepth == 32) + p = theApp.filterWidth * 8; + if(theApp.showSpeedTransparent) + drawTextTransp((u8*)filterData, + p, + 10, + theApp.filterHeight*2-10, + buffer); + else + drawText((u8*)filterData, + p, + 10, + theApp.filterHeight*2-10, + buffer); + } else { + if(theApp.showSpeedTransparent) + drawTextTransp((u8*)pix, + pitch, + 10, + theApp.filterHeight-10, + buffer); + else + drawText((u8*)pix, + pitch, + 10, + theApp.filterHeight-10, + buffer); + } + } + + POINT p; + p.x = theApp.dest.left; + p.y = theApp.dest.top; + CWnd *pWnd = theApp.m_pMainWnd; + pWnd->ScreenToClient(&p); + POINT p2; + p2.x = theApp.dest.right; + p2.y = theApp.dest.bottom; + pWnd->ScreenToClient(&p2); + + CDC *dc = pWnd->GetDC(); + + StretchDIBits((HDC)*dc, + p.x, + p.y, + p2.x - p.x, + p2.y - p.y, + 0, + 0, + theApp.rect.right, + theApp.rect.bottom, + theApp.filterFunction ? filterData : pix+pitch, + bi, + DIB_RGB_COLORS, + SRCCOPY); + + if(theApp.screenMessage) { + if(((GetTickCount() - theApp.screenMessageTime) < 3000) && + !theApp.disableStatusMessage) { + dc->SetTextColor(RGB(255,0,0)); + dc->SetBkMode(TRANSPARENT); + dc->TextOut(p.x+10, p2.y - 20, theApp.screenMessageBuffer); + } else { + theApp.screenMessage = false; + } + } + + pWnd->ReleaseDC(dc); +} + +int GDIDisplay::selectFullScreenMode(GUID **) +{ + HWND wnd = GetDesktopWindow(); + RECT r; + GetWindowRect(wnd, &r); + int w = (r.right - r.left) & 4095; + int h = (r.bottom - r.top) & 4095; + HDC dc = GetDC(wnd); + int c = GetDeviceCaps(dc, BITSPIXEL); + ReleaseDC(wnd, dc); + + return (c << 24) | (w << 12) | h; +} + +IDisplay *newGDIDisplay() +{ + return new GDIDisplay(); +} + diff --git a/src/win32/IOViewer.cpp b/src/win32/IOViewer.cpp index 491000e9..bcc5afb5 100644 --- a/src/win32/IOViewer.cpp +++ b/src/win32/IOViewer.cpp @@ -1,202 +1,239 @@ -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -// IOViewer.cpp : implementation file -// - -#include "stdafx.h" -#include "vba.h" -#include "IOViewer.h" - -#include "../System.h" -#include "../GBA.h" -#include "../Globals.h" -#include "../Sound.h" - -#include "IOViewerRegs.h" - -#ifdef _DEBUG -#define new DEBUG_NEW -#undef THIS_FILE -static char THIS_FILE[] = __FILE__; -#endif - -///////////////////////////////////////////////////////////////////////////// -// IOViewer dialog - - -IOViewer::IOViewer(CWnd* pParent /*=NULL*/) - : ResizeDlg(IOViewer::IDD, pParent) -{ - //{{AFX_DATA_INIT(IOViewer) - // NOTE: the ClassWizard will add member initialization here - //}}AFX_DATA_INIT - selected = 0; - autoUpdate = false; -} - - -void IOViewer::DoDataExchange(CDataExchange* pDX) -{ - CDialog::DoDataExchange(pDX); - //{{AFX_DATA_MAP(IOViewer) - DDX_Control(pDX, IDC_VALUE, m_value); - DDX_Control(pDX, IDC_ADDRESSES, m_address); - //}}AFX_DATA_MAP -} - - -BEGIN_MESSAGE_MAP(IOViewer, CDialog) - //{{AFX_MSG_MAP(IOViewer) - ON_BN_CLICKED(IDC_CLOSE, OnClose) - ON_BN_CLICKED(IDC_REFRESH, OnRefresh) - ON_BN_CLICKED(IDC_AUTO_UPDATE, OnAutoUpdate) - ON_CBN_SELCHANGE(IDC_ADDRESSES, OnSelchangeAddresses) - ON_BN_CLICKED(IDC_APPLY, OnApply) - //}}AFX_MSG_MAP - END_MESSAGE_MAP() - - ///////////////////////////////////////////////////////////////////////////// -// IOViewer message handlers - -void IOViewer::OnClose() -{ - theApp.winRemoveUpdateListener(this); - - DestroyWindow(); -} - -void IOViewer::OnRefresh() -{ - // TODO: Add your control notification handler code here - -} - -void IOViewer::OnAutoUpdate() -{ - autoUpdate = !autoUpdate; - if(autoUpdate) { - theApp.winAddUpdateListener(this); - } else { - theApp.winRemoveUpdateListener(this); - } -} - -void IOViewer::OnSelchangeAddresses() -{ - selected = m_address.GetCurSel(); - - update(); -} - -void IOViewer::PostNcDestroy() -{ - delete this; -} - -BOOL IOViewer::OnInitDialog() -{ - CDialog::OnInitDialog(); - - // winCenterWindow(getHandle()); - DIALOG_SIZER_START( sz ) - DIALOG_SIZER_END() - SetData(sz, - TRUE, - HKEY_CURRENT_USER, - "Software\\Emulators\\VisualBoyAdvance\\Viewer\\IOView", - NULL); - - CFont *font = CFont::FromHandle((HFONT)GetStockObject(SYSTEM_FIXED_FONT)); - int i; - for(i = 0; i < sizeof(ioViewRegisters)/sizeof(IOData); i++) { - m_address.AddString(ioViewRegisters[i].name); - } - m_address.SetFont(font); - for(i = 0; i < 16; i++) { - GetDlgItem(IDC_BIT_0+i)->SetFont(font); - } - - RECT cbSize; - int Height; - - m_address.GetClientRect(&cbSize); - Height = m_address.GetItemHeight(0); - Height += m_address.GetItemHeight(0) * (10); - - // Note: The use of SM_CYEDGE assumes that we're using Windows '95 - // Now add on the height of the border of the edit box - Height += GetSystemMetrics(SM_CYEDGE) * 2; // top & bottom edges - - // The height of the border of the drop-down box - Height += GetSystemMetrics(SM_CYEDGE) * 2; // top & bottom edges - - // now set the size of the window - m_address.SetWindowPos(NULL, - 0, 0, - cbSize.right, Height, - SWP_NOMOVE | SWP_NOZORDER); - - m_address.SetCurSel(0); - update(); - - return TRUE; // return TRUE unless you set the focus to a control - // EXCEPTION: OCX Property Pages should return FALSE -} - -void IOViewer::update() -{ - CString buffer; - - const IOData *sel = &ioViewRegisters[selected]; - u16 data = sel->address ? *sel->address : - (ioMem ? soundRead16(sel->offset) : 0); - - for(int i = 0; i < 16; i++) { - CButton *pWnd = (CButton *)GetDlgItem(IDC_BIT_0 + i); - - if(pWnd) { - if(!(sel->write & (1 << i))) - pWnd->EnableWindow(FALSE); - else - pWnd->EnableWindow(TRUE); - pWnd->SetCheck(((data & (1 << i)) >> i)); - buffer.Format("%2d %s", i, sel->bits[i]); - pWnd->SetWindowText(buffer); - } - } - - buffer.Format("%04X", data); - m_value.SetWindowText(buffer); -} - -void IOViewer::OnApply() -{ - const IOData *sel = &ioViewRegisters[selected]; - u16 res = 0; - for(int i = 0; i < 16; i++) { - CButton *pWnd = (CButton *)GetDlgItem(IDC_BIT_0 + i); - - if(pWnd) { - if(pWnd->GetCheck()) - res |= (1 << i); - } - } - CPUWriteHalfWord(0x4000000+sel->offset, res); - update(); -} +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2005 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +// IOViewer.cpp : implementation file +// + +#include "stdafx.h" +#include "vba.h" +#include "IOViewer.h" + +#include "../System.h" +#include "../GBA.h" +#include "../Globals.h" + +#include "IOViewerRegs.h" + +#ifdef _DEBUG +#define new DEBUG_NEW +#undef THIS_FILE +static char THIS_FILE[] = __FILE__; +#endif + +///////////////////////////////////////////////////////////////////////////// +// IOViewer dialog + + +IOViewer::IOViewer(CWnd* pParent /*=NULL*/) + : ResizeDlg(IOViewer::IDD, pParent) +{ + //{{AFX_DATA_INIT(IOViewer) + // NOTE: the ClassWizard will add member initialization here + //}}AFX_DATA_INIT + selected = 0; + autoUpdate = false; +} + + +void IOViewer::DoDataExchange(CDataExchange* pDX) +{ + CDialog::DoDataExchange(pDX); + //{{AFX_DATA_MAP(IOViewer) + DDX_Control(pDX, IDC_VALUE, m_value); + DDX_Control(pDX, IDC_ADDRESSES, m_address); + //}}AFX_DATA_MAP +} + + +BEGIN_MESSAGE_MAP(IOViewer, CDialog) + //{{AFX_MSG_MAP(IOViewer) + ON_BN_CLICKED(IDC_CLOSE, OnClose) + ON_BN_CLICKED(IDC_REFRESH, OnRefresh) + ON_BN_CLICKED(IDC_AUTO_UPDATE, OnAutoUpdate) + ON_CBN_SELCHANGE(IDC_ADDRESSES, OnSelchangeAddresses) + ON_BN_CLICKED(IDC_APPLY, OnApply) + ON_BN_CLICKED(IDC_BIT_0, bitChange) + ON_BN_CLICKED(IDC_BIT_1, bitChange) + ON_BN_CLICKED(IDC_BIT_2, bitChange) + ON_BN_CLICKED(IDC_BIT_3, bitChange) + ON_BN_CLICKED(IDC_BIT_4, bitChange) + ON_BN_CLICKED(IDC_BIT_5, bitChange) + ON_BN_CLICKED(IDC_BIT_6, bitChange) + ON_BN_CLICKED(IDC_BIT_7, bitChange) + ON_BN_CLICKED(IDC_BIT_8, bitChange) + ON_BN_CLICKED(IDC_BIT_9, bitChange) + ON_BN_CLICKED(IDC_BIT_10, bitChange) + ON_BN_CLICKED(IDC_BIT_11, bitChange) + ON_BN_CLICKED(IDC_BIT_12, bitChange) + ON_BN_CLICKED(IDC_BIT_13, bitChange) + ON_BN_CLICKED(IDC_BIT_14, bitChange) + ON_BN_CLICKED(IDC_BIT_15, bitChange) + //}}AFX_MSG_MAP + END_MESSAGE_MAP() + + ///////////////////////////////////////////////////////////////////////////// +// IOViewer message handlers + +void IOViewer::OnClose() +{ + theApp.winRemoveUpdateListener(this); + + DestroyWindow(); +} + +void IOViewer::bitChange() +{ + CString buffer; + u16 data = 0; + + for(int i = 0; i < 16; i++) { + CButton *pWnd = (CButton *)GetDlgItem(IDC_BIT_0 + i); + + if(pWnd) { + if(pWnd->GetCheck()) + data |= (1 << i); + } + } + + buffer.Format("%04X", data); + m_value.SetWindowText(buffer); +} + +void IOViewer::OnRefresh() +{ + // TODO: Add your control notification handler code here + + update(); +} + +void IOViewer::OnAutoUpdate() +{ + autoUpdate = !autoUpdate; + if(autoUpdate) { + theApp.winAddUpdateListener(this); + } else { + theApp.winRemoveUpdateListener(this); + } +} + +void IOViewer::OnSelchangeAddresses() +{ + selected = m_address.GetCurSel(); + + update(); +} + +void IOViewer::PostNcDestroy() +{ + delete this; +} + +BOOL IOViewer::OnInitDialog() +{ + CDialog::OnInitDialog(); + + // winCenterWindow(getHandle()); + DIALOG_SIZER_START( sz ) + DIALOG_SIZER_END() + SetData(sz, + TRUE, + HKEY_CURRENT_USER, + "Software\\Emulators\\VisualBoyAdvance\\Viewer\\IOView", + NULL); + + CFont *font = CFont::FromHandle((HFONT)GetStockObject(SYSTEM_FIXED_FONT)); + int i; + for(i = 0; i < sizeof(ioViewRegisters)/sizeof(IOData); i++) { + m_address.AddString(ioViewRegisters[i].name); + } + m_address.SetFont(font); + for(i = 0; i < 16; i++) { + GetDlgItem(IDC_BIT_0+i)->SetFont(font); + } + + RECT cbSize; + int Height; + + m_address.GetClientRect(&cbSize); + Height = m_address.GetItemHeight(0); + Height += m_address.GetItemHeight(0) * (10); + + // Note: The use of SM_CYEDGE assumes that we're using Windows '95 + // Now add on the height of the border of the edit box + Height += GetSystemMetrics(SM_CYEDGE) * 2; // top & bottom edges + + // The height of the border of the drop-down box + Height += GetSystemMetrics(SM_CYEDGE) * 2; // top & bottom edges + + // now set the size of the window + m_address.SetWindowPos(NULL, + 0, 0, + cbSize.right, Height, + SWP_NOMOVE | SWP_NOZORDER); + + m_address.SetCurSel(0); + update(); + + return TRUE; // return TRUE unless you set the focus to a control + // EXCEPTION: OCX Property Pages should return FALSE +} + +void IOViewer::update() +{ + CString buffer; + + const IOData *sel = &ioViewRegisters[selected]; + u16 data = sel->address ? *sel->address : + (ioMem ? *((u16 *)&ioMem[sel->offset]) : 0); + + for(int i = 0; i < 16; i++) { + CButton *pWnd = (CButton *)GetDlgItem(IDC_BIT_0 + i); + + if(pWnd) { + if(!(sel->write & (1 << i))) + pWnd->EnableWindow(FALSE); + else + pWnd->EnableWindow(TRUE); + pWnd->SetCheck(((data & (1 << i)) >> i)); + buffer.Format("%2d %s", i, sel->bits[i]); + pWnd->SetWindowText(buffer); + } + } + + buffer.Format("%04X", data); + m_value.SetWindowText(buffer); +} + +void IOViewer::OnApply() +{ + if(rom != NULL) + { + const IOData *sel = &ioViewRegisters[selected]; + u16 res = 0; + for(int i = 0; i < 16; i++) { + CButton *pWnd = (CButton *)GetDlgItem(IDC_BIT_0 + i); + + if(pWnd) { + if(pWnd->GetCheck()) + res |= (1 << i); + } + } + CPUWriteHalfWord(0x4000000+sel->offset, res); + update(); + } +} diff --git a/src/win32/IOViewer.h b/src/win32/IOViewer.h index d768b018..e262f9ec 100644 --- a/src/win32/IOViewer.h +++ b/src/win32/IOViewer.h @@ -1,7 +1,7 @@ // -*- C++ -*- // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -37,6 +37,7 @@ class IOViewer : public ResizeDlg, IUpdateListener // Construction public: void update(); + void bitChange(); bool autoUpdate; int selected; IOViewer(CWnd* pParent = NULL); // standard constructor diff --git a/src/win32/IOViewerRegs.h b/src/win32/IOViewerRegs.h index 9d28a6e2..4491308e 100644 --- a/src/win32/IOViewerRegs.h +++ b/src/win32/IOViewerRegs.h @@ -1,7 +1,7 @@ // -*- C++ -*- // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -258,7 +258,7 @@ const IOData ioViewRegisters[] = { } }, { - &BG2HOFS, 0x18, "0x4000018-BG8HOFS", 0x01FF, + &BG2HOFS, 0x18, "0x4000018-BG2HOFS", 0x01FF, { "", "", @@ -1119,7 +1119,7 @@ const IOData ioViewRegisters[] = { } }, { - NULL, 0x82, "0x4000082-SGCNT0_H", 0xFF1F, + NULL, 0x82, "0x4000082-SGCNT0_H", 0xFF0F, { "", "Sound 1-4 Volume (2 bits)", diff --git a/src/win32/Input.h b/src/win32/Input.h index 345e2a61..93813252 100644 --- a/src/win32/Input.h +++ b/src/win32/Input.h @@ -22,10 +22,6 @@ #include "../System.h" #define JOYCONFIG_MESSAGE (WM_USER + 1000) - -typedef CList KeyList; -//typedef CList KeyList; - #define JOYPADS 4 #define MOTION_KEYS 4 #define KEYS_PER_PAD 13 @@ -35,6 +31,8 @@ typedef CList KeyList; #define DEVICEOF(key) (key >> 8) #define KEYOF(key) (key & 255) +typedef CList KeyList; + enum { KEY_LEFT, KEY_RIGHT, KEY_UP, KEY_DOWN, @@ -46,6 +44,7 @@ enum { }; class Input { + public: KeyList joypaddata[JOYPADS * KEYS_PER_PAD + MOTION_KEYS]; @@ -56,7 +55,7 @@ class Input { virtual bool readDevices() = 0; virtual u32 readDevice(int which) = 0; - virtual CString getKeyName(int key) = 0; + virtual CString getKeyName(LONG_PTR key) = 0; virtual void checkKeys() = 0; virtual void checkMotionKeys() = 0; virtual void checkDevices() = 0; @@ -65,6 +64,4 @@ class Input { virtual void saveSettings() = 0; }; -#define joypad theApp.input->joypaddata - #endif diff --git a/src/win32/Joypad.cpp b/src/win32/Joypad.cpp index 9b88fd7d..bd8594da 100644 --- a/src/win32/Joypad.cpp +++ b/src/win32/Joypad.cpp @@ -43,25 +43,25 @@ void AssignKey(KeyList &Key, int Out) CString GetKeyListName(KeyList& Keys) { - CString txtKeys; - - POSITION p = Keys.GetHeadPosition(); - while(p!=NULL) - { - txtKeys+=theApp.input->getKeyName(Keys.GetNext(p)); - if (p!=NULL) - txtKeys+=", "; - } + CString txtKeys; + + POSITION p = Keys.GetHeadPosition(); + while(p!=NULL) + { + txtKeys+=theApp.input->getKeyName(Keys.GetNext(p)); + if (p!=NULL) + txtKeys+=", "; + } return txtKeys; } void CopyKeys(KeyList &Out, KeyList &In) { Out.RemoveAll(); - POSITION p = In.GetHeadPosition(); - while(p!=NULL) - Out.AddTail(In.GetNext(p)); -} + POSITION p = In.GetHeadPosition(); + while(p!=NULL) + Out.AddTail(In.GetNext(p)); +} #define AssignKeys(in, out) CopyKeys(out, in); @@ -100,6 +100,7 @@ LRESULT JoypadEditControl::OnJoyConfig(WPARAM wParam, LPARAM lParam) return TRUE; } + BOOL JoypadEditControl::PreTranslateMessage(MSG *pMsg) { if(pMsg->message == WM_KEYDOWN && (pMsg->wParam == VK_ESCAPE || pMsg->wParam == VK_RETURN)) @@ -125,35 +126,33 @@ JoypadConfig::JoypadConfig(int w, CWnd* pParent /*=NULL*/) void JoypadConfig::DoDataExchange(CDataExchange* pDX) { - CDialog::DoDataExchange(pDX); - //{{AFX_DATA_MAP(JoypadConfig) - DDX_Control(pDX, IDC_EDIT_UP, up); - DDX_Control(pDX, IDC_EDIT_SPEED, speed); - DDX_Control(pDX, IDC_EDIT_RIGHT, right); - DDX_Control(pDX, IDC_EDIT_LEFT, left); - DDX_Control(pDX, IDC_EDIT_DOWN, down); - DDX_Control(pDX, IDC_EDIT_CAPTURE, capture); - DDX_Control(pDX, IDC_EDIT_BUTTON_START, buttonStart); - DDX_Control(pDX, IDC_EDIT_BUTTON_SELECT, buttonSelect); - DDX_Control(pDX, IDC_EDIT_BUTTON_R, buttonR); - DDX_Control(pDX, IDC_EDIT_BUTTON_L, buttonL); - DDX_Control(pDX, IDC_EDIT_BUTTON_GS, buttonGS); - DDX_Control(pDX, IDC_EDIT_BUTTON_B, buttonB); - DDX_Control(pDX, IDC_EDIT_BUTTON_A, buttonA); - //}}AFX_DATA_MAP + CDialog::DoDataExchange(pDX); + //{{AFX_DATA_MAP(JoypadConfig) + DDX_Control(pDX, IDC_EDIT_UP, up); + DDX_Control(pDX, IDC_EDIT_SPEED, speed); + DDX_Control(pDX, IDC_EDIT_RIGHT, right); + DDX_Control(pDX, IDC_EDIT_LEFT, left); + DDX_Control(pDX, IDC_EDIT_DOWN, down); + DDX_Control(pDX, IDC_EDIT_CAPTURE, capture); + DDX_Control(pDX, IDC_EDIT_BUTTON_START, buttonStart); + DDX_Control(pDX, IDC_EDIT_BUTTON_SELECT, buttonSelect); + DDX_Control(pDX, IDC_EDIT_BUTTON_R, buttonR); + DDX_Control(pDX, IDC_EDIT_BUTTON_L, buttonL); + DDX_Control(pDX, IDC_EDIT_BUTTON_GS, buttonGS); + DDX_Control(pDX, IDC_EDIT_BUTTON_B, buttonB); + DDX_Control(pDX, IDC_EDIT_BUTTON_A, buttonA); + //}}AFX_DATA_MAP } BEGIN_MESSAGE_MAP(JoypadConfig, CDialog) - //{{AFX_MSG_MAP(JoypadConfig) ON_BN_CLICKED(ID_CANCEL, OnCancel) ON_BN_CLICKED(ID_OK, OnOk) ON_WM_CHAR() ON_WM_DESTROY() ON_WM_TIMER() ON_WM_KEYDOWN() - //}}AFX_MSG_MAP - ON_BN_CLICKED(IDC_APPENDMODE, &JoypadConfig::OnBnClickedAppendmode) + ON_BN_CLICKED(IDC_APPENDMODE, &JoypadConfig::OnBnClickedAppendmode) END_MESSAGE_MAP() ///////////////////////////////////////////////////////////////////////////// @@ -166,19 +165,19 @@ void JoypadConfig::OnCancel() void JoypadConfig::OnOk() { - AssignKeys(up.m_Keys,joypad[JOYPAD(which,KEY_UP)]); - AssignKeys(speed.m_Keys,joypad[JOYPAD(which,KEY_BUTTON_SPEED)]); - AssignKeys(right.m_Keys,joypad[JOYPAD(which,KEY_RIGHT)]); - AssignKeys(left.m_Keys,joypad[JOYPAD(which,KEY_LEFT)]); - AssignKeys(down.m_Keys,joypad[JOYPAD(which,KEY_DOWN)]); - AssignKeys(capture.m_Keys,joypad[JOYPAD(which,KEY_BUTTON_CAPTURE)]); - AssignKeys(buttonStart.m_Keys,joypad[JOYPAD(which,KEY_BUTTON_START)]); - AssignKeys(buttonSelect.m_Keys,joypad[JOYPAD(which,KEY_BUTTON_SELECT)]); - AssignKeys(buttonR.m_Keys,joypad[JOYPAD(which,KEY_BUTTON_R)]); - AssignKeys(buttonL.m_Keys,joypad[JOYPAD(which,KEY_BUTTON_L)]); - AssignKeys(buttonGS.m_Keys,joypad[JOYPAD(which,KEY_BUTTON_GS)]); - AssignKeys(buttonB.m_Keys,joypad[JOYPAD(which,KEY_BUTTON_B)]); - AssignKeys(buttonA.m_Keys,joypad[JOYPAD(which,KEY_BUTTON_A)]); + AssignKeys(up.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_UP)]); + AssignKeys(speed.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_SPEED)]); + AssignKeys(right.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_RIGHT)]); + AssignKeys(left.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_LEFT)]); + AssignKeys(down.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_DOWN)]); + AssignKeys(capture.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_CAPTURE)]); + AssignKeys(buttonStart.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_START)]); + AssignKeys(buttonSelect.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_SELECT)]); + AssignKeys(buttonR.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_R)]); + AssignKeys(buttonL.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_L)]); + AssignKeys(buttonGS.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_GS)]); + AssignKeys(buttonB.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_B)]); + AssignKeys(buttonA.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_A)]); theApp.input->checkKeys(); EndDialog(TRUE); @@ -195,7 +194,7 @@ void JoypadConfig::OnDestroy() KillTimer(timerId); } -void JoypadConfig::OnTimer(UINT nIDEvent) +void JoypadConfig::OnTimer(UINT_PTR nIDEvent) { theApp.input->checkDevices(); @@ -212,35 +211,35 @@ BOOL JoypadConfig::OnInitDialog() bAppendMode = FALSE; - timerId = SetTimer(0,200,NULL); + timerId = SetTimer(0,50,NULL); - CopyKeys(up.m_Keys,joypad[JOYPAD(which,KEY_UP)]); - CopyKeys(speed.m_Keys,joypad[JOYPAD(which,KEY_BUTTON_SPEED)]); - CopyKeys(right.m_Keys,joypad[JOYPAD(which,KEY_RIGHT)]); - CopyKeys(left.m_Keys,joypad[JOYPAD(which,KEY_LEFT)]); - CopyKeys(down.m_Keys,joypad[JOYPAD(which,KEY_DOWN)]); - CopyKeys(capture.m_Keys,joypad[JOYPAD(which,KEY_BUTTON_CAPTURE)]); - CopyKeys(buttonStart.m_Keys,joypad[JOYPAD(which,KEY_BUTTON_START)]); - CopyKeys(buttonSelect.m_Keys,joypad[JOYPAD(which,KEY_BUTTON_SELECT)]); - CopyKeys(buttonR.m_Keys,joypad[JOYPAD(which,KEY_BUTTON_R)]); - CopyKeys(buttonL.m_Keys,joypad[JOYPAD(which,KEY_BUTTON_L)]); - CopyKeys(buttonGS.m_Keys,joypad[JOYPAD(which,KEY_BUTTON_GS)]); - CopyKeys(buttonB.m_Keys,joypad[JOYPAD(which,KEY_BUTTON_B)]); - CopyKeys(buttonA.m_Keys,joypad[JOYPAD(which,KEY_BUTTON_A)]); + CopyKeys(up.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_UP)]); + CopyKeys(speed.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_SPEED)]); + CopyKeys(right.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_RIGHT)]); + CopyKeys(left.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_LEFT)]); + CopyKeys(down.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_DOWN)]); + CopyKeys(capture.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_CAPTURE)]); + CopyKeys(buttonStart.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_START)]); + CopyKeys(buttonSelect.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_SELECT)]); + CopyKeys(buttonR.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_R)]); + CopyKeys(buttonL.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_L)]); + CopyKeys(buttonGS.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_GS)]); + CopyKeys(buttonB.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_B)]); + CopyKeys(buttonA.m_Keys,theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_A)]); - up.SetWindowText(GetKeyListName(joypad[JOYPAD(which,KEY_UP)])); - down.SetWindowText(GetKeyListName(joypad[JOYPAD(which,KEY_DOWN)])); - left.SetWindowText(GetKeyListName(joypad[JOYPAD(which,KEY_LEFT)])); - right.SetWindowText(GetKeyListName(joypad[JOYPAD(which,KEY_RIGHT)])); - buttonA.SetWindowText(GetKeyListName(joypad[JOYPAD(which,KEY_BUTTON_A)])); - buttonB.SetWindowText(GetKeyListName(joypad[JOYPAD(which,KEY_BUTTON_B)])); - buttonL.SetWindowText(GetKeyListName(joypad[JOYPAD(which,KEY_BUTTON_L)])); - buttonR.SetWindowText(GetKeyListName(joypad[JOYPAD(which,KEY_BUTTON_R)])); - buttonSelect.SetWindowText(GetKeyListName(joypad[JOYPAD(which,KEY_BUTTON_SELECT)])); - buttonStart.SetWindowText(GetKeyListName(joypad[JOYPAD(which,KEY_BUTTON_START)])); - speed.SetWindowText(GetKeyListName(joypad[JOYPAD(which,KEY_BUTTON_SPEED)])); - capture.SetWindowText(GetKeyListName(joypad[JOYPAD(which,KEY_BUTTON_CAPTURE)])); - buttonGS.SetWindowText(GetKeyListName(joypad[JOYPAD(which,KEY_BUTTON_GS)])); + up.SetWindowText(GetKeyListName(theApp.input->joypaddata[JOYPAD(which,KEY_UP)])); + down.SetWindowText(GetKeyListName(theApp.input->joypaddata[JOYPAD(which,KEY_DOWN)])); + left.SetWindowText(GetKeyListName(theApp.input->joypaddata[JOYPAD(which,KEY_LEFT)])); + right.SetWindowText(GetKeyListName(theApp.input->joypaddata[JOYPAD(which,KEY_RIGHT)])); + buttonA.SetWindowText(GetKeyListName(theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_A)])); + buttonB.SetWindowText(GetKeyListName(theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_B)])); + buttonL.SetWindowText(GetKeyListName(theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_L)])); + buttonR.SetWindowText(GetKeyListName(theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_R)])); + buttonSelect.SetWindowText(GetKeyListName(theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_SELECT)])); + buttonStart.SetWindowText(GetKeyListName(theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_START)])); + speed.SetWindowText(GetKeyListName(theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_SPEED)])); + capture.SetWindowText(GetKeyListName(theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_CAPTURE)])); + buttonGS.SetWindowText(GetKeyListName(theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_GS)])); CenterWindow(); @@ -248,47 +247,47 @@ BOOL JoypadConfig::OnInitDialog() // EXCEPTION: OCX Property Pages should return FALSE } -void JoypadConfig::assignKey(int id, int key) +void JoypadConfig::assignKey(int id, LONG_PTR key) { switch(id) { case IDC_EDIT_LEFT: - AssignKey(joypad[JOYPAD(which,KEY_LEFT)],key); + AssignKey(theApp.input->joypaddata[JOYPAD(which,KEY_LEFT)],key); break; case IDC_EDIT_RIGHT: - AssignKey(joypad[JOYPAD(which,KEY_RIGHT)],key); + AssignKey(theApp.input->joypaddata[JOYPAD(which,KEY_RIGHT)],key); break; case IDC_EDIT_UP: - AssignKey(joypad[JOYPAD(which,KEY_UP)],key); + AssignKey(theApp.input->joypaddata[JOYPAD(which,KEY_UP)],key); break; case IDC_EDIT_SPEED: - AssignKey(joypad[JOYPAD(which,KEY_BUTTON_SPEED)],key); + AssignKey(theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_SPEED)],key); break; case IDC_EDIT_CAPTURE: - AssignKey(joypad[JOYPAD(which,KEY_BUTTON_CAPTURE)],key); + AssignKey(theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_CAPTURE)],key); break; case IDC_EDIT_DOWN: - AssignKey(joypad[JOYPAD(which,KEY_DOWN)],key); + AssignKey(theApp.input->joypaddata[JOYPAD(which,KEY_DOWN)],key); break; case IDC_EDIT_BUTTON_A: - AssignKey(joypad[JOYPAD(which,KEY_BUTTON_A)],key); + AssignKey(theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_A)],key); break; case IDC_EDIT_BUTTON_B: - AssignKey(joypad[JOYPAD(which,KEY_BUTTON_B)],key); + AssignKey(theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_B)],key); break; case IDC_EDIT_BUTTON_L: - AssignKey(joypad[JOYPAD(which,KEY_BUTTON_L)],key); + AssignKey(theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_L)],key); break; case IDC_EDIT_BUTTON_R: - AssignKey(joypad[JOYPAD(which,KEY_BUTTON_R)],key); + AssignKey(theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_R)],key); break; case IDC_EDIT_BUTTON_START: - AssignKey(joypad[JOYPAD(which,KEY_BUTTON_START)],key); + AssignKey(theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_START)],key); break; case IDC_EDIT_BUTTON_SELECT: - AssignKey(joypad[JOYPAD(which,KEY_BUTTON_SELECT)],key); + AssignKey(theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_SELECT)],key); break; case IDC_EDIT_BUTTON_GS: - AssignKey(joypad[JOYPAD(which,KEY_BUTTON_GS)],key); + AssignKey(theApp.input->joypaddata[JOYPAD(which,KEY_BUTTON_GS)],key); break; } } @@ -322,15 +321,12 @@ void MotionConfig::DoDataExchange(CDataExchange* pDX) BEGIN_MESSAGE_MAP(MotionConfig, CDialog) - //{{AFX_MSG_MAP(MotionConfig) ON_BN_CLICKED(ID_CANCEL, OnCancel) ON_BN_CLICKED(ID_OK, OnOk) - ON_WM_CHAR() ON_WM_DESTROY() ON_WM_KEYDOWN() ON_WM_TIMER() - //}}AFX_MSG_MAP - ON_BN_CLICKED(IDC_APPENDMODE, &MotionConfig::OnBnClickedAppendmode) + ON_BN_CLICKED(IDC_APPENDMODE, &MotionConfig::OnBnClickedAppendmode) END_MESSAGE_MAP() ///////////////////////////////////////////////////////////////////////////// @@ -343,16 +339,11 @@ void MotionConfig::OnCancel() void MotionConfig::OnOk() { + assignKeys(); theApp.input->checkKeys(); EndDialog( TRUE); } -void MotionConfig::OnChar(UINT nChar, UINT nRepCnt, UINT nFlags) -{ -} - - - void MotionConfig::OnDestroy() { CDialog::OnDestroy(); @@ -360,24 +351,23 @@ void MotionConfig::OnDestroy() KillTimer(timerId); } - BOOL MotionConfig::OnInitDialog() { CDialog::OnInitDialog(); - timerId = SetTimer(0,200,NULL); + timerId = SetTimer(0,50,NULL); - CopyKeys(up.m_Keys, joypad[MOTION(KEY_UP)]); - up.SetWindowText(GetKeyListName(joypad[MOTION(KEY_UP)])); + CopyKeys(up.m_Keys, theApp.input->joypaddata[MOTION(KEY_UP)]); + up.SetWindowText(GetKeyListName(theApp.input->joypaddata[MOTION(KEY_UP)])); - CopyKeys(down.m_Keys, joypad[MOTION(KEY_DOWN)]); - down.SetWindowText(GetKeyListName(joypad[MOTION(KEY_DOWN)])); + CopyKeys(down.m_Keys, theApp.input->joypaddata[MOTION(KEY_DOWN)]); + down.SetWindowText(GetKeyListName(theApp.input->joypaddata[MOTION(KEY_DOWN)])); - CopyKeys(left.m_Keys, joypad[MOTION(KEY_LEFT)]); - left.SetWindowText(GetKeyListName(joypad[MOTION(KEY_LEFT)])); + CopyKeys(left.m_Keys, theApp.input->joypaddata[MOTION(KEY_LEFT)]); + left.SetWindowText(GetKeyListName(theApp.input->joypaddata[MOTION(KEY_LEFT)])); - CopyKeys(right.m_Keys, joypad[MOTION(KEY_RIGHT)]); - right.SetWindowText(GetKeyListName(joypad[MOTION(KEY_RIGHT)])); + CopyKeys(right.m_Keys, theApp.input->joypaddata[MOTION(KEY_RIGHT)]); + right.SetWindowText(GetKeyListName(theApp.input->joypaddata[MOTION(KEY_RIGHT)])); CenterWindow(); @@ -400,16 +390,16 @@ void MotionConfig::assignKey(int id, int key) { switch(id) { case IDC_EDIT_LEFT: - AssignKey(joypad[MOTION(KEY_LEFT)],key); + AssignKey(theApp.input->joypaddata[MOTION(KEY_LEFT)],key); break; case IDC_EDIT_RIGHT: - AssignKey(joypad[MOTION(KEY_RIGHT)],key); + AssignKey(theApp.input->joypaddata[MOTION(KEY_RIGHT)],key); break; case IDC_EDIT_UP: - AssignKey(joypad[MOTION(KEY_UP)],key); + AssignKey(theApp.input->joypaddata[MOTION(KEY_UP)],key); break; case IDC_EDIT_DOWN: - AssignKey(joypad[MOTION(KEY_DOWN)],key); + AssignKey(theApp.input->joypaddata[MOTION(KEY_DOWN)],key); break; } } @@ -430,13 +420,13 @@ void MotionConfig::assignKeys() id = IDC_EDIT_RIGHT; assignKey(id, GetWindowLong(right, GWL_USERDATA)); } - -void JoypadConfig::OnBnClickedAppendmode() -{ - bAppendMode = (::SendMessage(GetDlgItem(IDC_APPENDMODE)->GetSafeHwnd(), BM_GETCHECK, 0, 0L) != 0); -} - -void MotionConfig::OnBnClickedAppendmode() -{ - bAppendMode = (::SendMessage(GetDlgItem(IDC_APPENDMODE)->GetSafeHwnd(), BM_GETCHECK, 0, 0L) != 0); -} + +void JoypadConfig::OnBnClickedAppendmode() +{ + bAppendMode = (::SendMessage(GetDlgItem(IDC_APPENDMODE)->GetSafeHwnd(), BM_GETCHECK, 0, 0L) != 0); +} + +void MotionConfig::OnBnClickedAppendmode() +{ + bAppendMode = (::SendMessage(GetDlgItem(IDC_APPENDMODE)->GetSafeHwnd(), BM_GETCHECK, 0, 0L) != 0); +} diff --git a/src/win32/Joypad.h b/src/win32/Joypad.h index 8bff067d..30400956 100644 --- a/src/win32/Joypad.h +++ b/src/win32/Joypad.h @@ -1,176 +1,176 @@ -// -*- C++ -*- -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - +// -*- C++ -*- +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + #include "afxwin.h" -#if !defined(AFX_JOYPAD_H__FFFB2470_9EEC_4D2D_A5F0_3BF31579999A__INCLUDED_) -#define AFX_JOYPAD_H__FFFB2470_9EEC_4D2D_A5F0_3BF31579999A__INCLUDED_ - -#if _MSC_VER > 1000 -#pragma once -#endif // _MSC_VER > 1000 -// Joypad.h : header file -// - -///////////////////////////////////////////////////////////////////////////// -// JoypadEditControl window - -class JoypadEditControl : public CEdit -{ - // Construction -public: - - JoypadEditControl(); - - KeyList m_Keys; - - // Attributes - public: - - // Operations - public: - - // Overrides - // ClassWizard generated virtual function overrides - //{{AFX_VIRTUAL(JoypadEditControl) - //}}AFX_VIRTUAL - - // Implementation - public: - virtual BOOL PreTranslateMessage(MSG *pMsg); - afx_msg LRESULT OnJoyConfig(WPARAM wParam, LPARAM lParam); - virtual ~JoypadEditControl(); - - // Generated message map functions - protected: - //{{AFX_MSG(JoypadEditControl) - afx_msg void OnChar(UINT nChar, UINT nRepCnt, UINT nFlags); - //}}AFX_MSG - - DECLARE_MESSAGE_MAP() - }; - - ///////////////////////////////////////////////////////////////////////////// - -///////////////////////////////////////////////////////////////////////////// -// JoypadConfig dialog - -class JoypadConfig : public CDialog -{ - // Construction - public: - void assignKey(int id, int key); - JoypadConfig(int w, CWnd* pParent = NULL); // standard constructor - - // Dialog Data - //{{AFX_DATA(JoypadConfig) - enum { IDD = IDD_CONFIG }; - JoypadEditControl up; - JoypadEditControl speed; - JoypadEditControl right; - JoypadEditControl left; - JoypadEditControl down; - JoypadEditControl capture; - JoypadEditControl buttonStart; - JoypadEditControl buttonSelect; - JoypadEditControl buttonR; - JoypadEditControl buttonL; - JoypadEditControl buttonGS; - JoypadEditControl buttonB; - JoypadEditControl buttonA; - - //}}AFX_DATA - - - // Overrides - // ClassWizard generated virtual function overrides - //{{AFX_VIRTUAL(JoypadConfig) - protected: - virtual void DoDataExchange(CDataExchange* pDX); // DDX/DDV support - //}}AFX_VIRTUAL - - // Implementation - protected: - UINT timerId; - int which; - - // Generated message map functions - //{{AFX_MSG(JoypadConfig) - afx_msg void OnCancel(); - afx_msg void OnOk(); - afx_msg void OnChar(UINT nChar, UINT nRepCnt, UINT nFlags); - afx_msg void OnDestroy(); - afx_msg void OnTimer(UINT nIDEvent); - afx_msg void OnKeyDown(UINT nChar, UINT nRepCnt, UINT nFlags); - virtual BOOL OnInitDialog(); - //}}AFX_MSG - DECLARE_MESSAGE_MAP() +#if !defined(AFX_JOYPAD_H__FFFB2470_9EEC_4D2D_A5F0_3BF31579999A__INCLUDED_) +#define AFX_JOYPAD_H__FFFB2470_9EEC_4D2D_A5F0_3BF31579999A__INCLUDED_ + +#if _MSC_VER > 1000 +#pragma once +#endif // _MSC_VER > 1000 +// Joypad.h : header file +// + +///////////////////////////////////////////////////////////////////////////// +// JoypadEditControl window + +class JoypadEditControl : public CEdit +{ + // Construction +public: + + JoypadEditControl(); + + KeyList m_Keys; + + // Attributes + public: + + // Operations + public: + + // Overrides + // ClassWizard generated virtual function overrides + //{{AFX_VIRTUAL(JoypadEditControl) + //}}AFX_VIRTUAL + + // Implementation + public: + virtual BOOL PreTranslateMessage(MSG *pMsg); + afx_msg LRESULT OnJoyConfig(WPARAM wParam, LPARAM lParam); + virtual ~JoypadEditControl(); + + // Generated message map functions + protected: + //{{AFX_MSG(JoypadEditControl) + afx_msg void OnChar(UINT nChar, UINT nRepCnt, UINT nFlags); + //}}AFX_MSG + + DECLARE_MESSAGE_MAP() + }; + + ///////////////////////////////////////////////////////////////////////////// + +///////////////////////////////////////////////////////////////////////////// +// JoypadConfig dialog + +class JoypadConfig : public CDialog +{ + // Construction + public: + void assignKey(int id, LONG_PTR key); + JoypadConfig(int w, CWnd* pParent = NULL); // standard constructor + + // Dialog Data + //{{AFX_DATA(JoypadConfig) + enum { IDD = IDD_CONFIG }; + JoypadEditControl up; + JoypadEditControl speed; + JoypadEditControl right; + JoypadEditControl left; + JoypadEditControl down; + JoypadEditControl capture; + JoypadEditControl buttonStart; + JoypadEditControl buttonSelect; + JoypadEditControl buttonR; + JoypadEditControl buttonL; + JoypadEditControl buttonGS; + JoypadEditControl buttonB; + JoypadEditControl buttonA; + + //}}AFX_DATA + + + // Overrides + // ClassWizard generated virtual function overrides + //{{AFX_VIRTUAL(JoypadConfig) + protected: + virtual void DoDataExchange(CDataExchange* pDX); // DDX/DDV support + //}}AFX_VIRTUAL + + // Implementation + protected: + UINT timerId; + int which; + + // Generated message map functions + //{{AFX_MSG(JoypadConfig) + afx_msg void OnCancel(); + afx_msg void OnOk(); + afx_msg void OnChar(UINT nChar, UINT nRepCnt, UINT nFlags); + afx_msg void OnDestroy(); + afx_msg void OnTimer(UINT nIDEvent); + afx_msg void OnKeyDown(UINT nChar, UINT nRepCnt, UINT nFlags); + virtual BOOL OnInitDialog(); + //}}AFX_MSG + DECLARE_MESSAGE_MAP() public: afx_msg void OnBnClickedAppendmode(); -}; - ///////////////////////////////////////////////////////////////////////////// -// MotionConfig dialog - -class MotionConfig : public CDialog -{ - // Construction - public: - void assignKeys(); - void assignKey(int id, int key); - MotionConfig(CWnd* pParent = NULL); // standard constructor - - // Dialog Data - //{{AFX_DATA(MotionConfig) - enum { IDD = IDD_MOTION_CONFIG }; - JoypadEditControl up; - JoypadEditControl right; - JoypadEditControl left; - JoypadEditControl down; - //}}AFX_DATA - - - // Overrides - // ClassWizard generated virtual function overrides - //{{AFX_VIRTUAL(MotionConfig) - protected: - virtual void DoDataExchange(CDataExchange* pDX); // DDX/DDV support - //}}AFX_VIRTUAL - - // Implementation - protected: - - // Generated message map functions - //{{AFX_MSG(MotionConfig) - afx_msg void OnCancel(); - afx_msg void OnOk(); - afx_msg void OnChar(UINT nChar, UINT nRepCnt, UINT nFlags); - afx_msg void OnDestroy(); - virtual BOOL OnInitDialog(); - afx_msg void OnKeyDown(UINT nChar, UINT nRepCnt, UINT nFlags); - afx_msg void OnTimer(UINT nIDEvent); - //}}AFX_MSG - DECLARE_MESSAGE_MAP() - private: - UINT timerId; +}; + ///////////////////////////////////////////////////////////////////////////// +// MotionConfig dialog + +class MotionConfig : public CDialog +{ + // Construction + public: + void assignKeys(); + void assignKey(int id, int key); + MotionConfig(CWnd* pParent = NULL); // standard constructor + + // Dialog Data + //{{AFX_DATA(MotionConfig) + enum { IDD = IDD_MOTION_CONFIG }; + JoypadEditControl up; + JoypadEditControl right; + JoypadEditControl left; + JoypadEditControl down; + //}}AFX_DATA + + + // Overrides + // ClassWizard generated virtual function overrides + //{{AFX_VIRTUAL(MotionConfig) + protected: + virtual void DoDataExchange(CDataExchange* pDX); // DDX/DDV support + //}}AFX_VIRTUAL + + // Implementation + protected: + + // Generated message map functions + //{{AFX_MSG(MotionConfig) + afx_msg void OnCancel(); + afx_msg void OnOk(); + afx_msg void OnChar(UINT nChar, UINT nRepCnt, UINT nFlags); + afx_msg void OnDestroy(); + virtual BOOL OnInitDialog(); + afx_msg void OnKeyDown(UINT nChar, UINT nRepCnt, UINT nFlags); + afx_msg void OnTimer(UINT nIDEvent); + //}}AFX_MSG + DECLARE_MESSAGE_MAP() + private: + UINT timerId; public: afx_msg void OnBnClickedAppendmode(); -}; -//{{AFX_INSERT_LOCATION}} -// Microsoft Visual C++ will insert additional declarations immediately before the previous line. - -#endif // !defined(AFX_JOYPAD_H__FFFB2470_9EEC_4D2D_A5F0_3BF31579999A__INCLUDED_) +}; +//{{AFX_INSERT_LOCATION}} +// Microsoft Visual C++ will insert additional declarations immediately before the previous line. + +#endif // !defined(AFX_JOYPAD_H__FFFB2470_9EEC_4D2D_A5F0_3BF31579999A__INCLUDED_) diff --git a/src/win32/Logging.cpp b/src/win32/Logging.cpp index 697b6d42..2739d573 100644 --- a/src/win32/Logging.cpp +++ b/src/win32/Logging.cpp @@ -1,6 +1,6 @@ // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -225,7 +225,7 @@ BOOL Logging::OnInitDialog() m_dma2 = (systemVerbose & 64) != 0; m_dma3 = (systemVerbose & 128) != 0; m_undefined = (systemVerbose & 256) != 0; - m_agbprint = (systemVerbose & 256) != 0; + m_agbprint = (systemVerbose & 512) != 0; UpdateData(FALSE); m_log.LimitText(-1); @@ -237,7 +237,7 @@ BOOL Logging::OnInitDialog() void Logging::log(const char *s) { - int size = ::SendMessage(m_log, WM_GETTEXTLENGTH, 0, 0); + DWORD size = (DWORD)::SendMessage(m_log, WM_GETTEXTLENGTH, 0, 0); m_log.SetSel(size, size); m_log.ReplaceSel(s); } diff --git a/src/win32/MainWnd.cpp b/src/win32/MainWnd.cpp index 227576f9..195c2622 100644 --- a/src/win32/MainWnd.cpp +++ b/src/win32/MainWnd.cpp @@ -1,1137 +1,1314 @@ -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -// MainWnd.cpp : implementation file -// - -#include "stdafx.h" -#include "VBA.h" -#include "MainWnd.h" - -#include - -#include "FileDlg.h" -#include "Reg.h" -#include "WinResUtil.h" - -#include "../System.h" -#include "../cheatSearch.h" -#include "../GBA.h" -#include "../Globals.h" -#include "../Flash.h" -#include "../Globals.h" -#include "../gb/GB.h" -#include "../gb/gbCheats.h" -#include "../gb/gbGlobals.h" -#include "../RTC.h" -#include "../Sound.h" -#include "../Util.h" -#include ".\mainwnd.h" - -#include "../Link.h" // Link - -#ifdef _DEBUG -#define new DEBUG_NEW -#undef THIS_FILE -static char THIS_FILE[] = __FILE__; -#endif - -#define VBA_CONFIRM_MODE WM_APP + 100 - -extern void remoteCleanUp(); - -///////////////////////////////////////////////////////////////////////////// -// MainWnd - -MainWnd::MainWnd() -{ - m_hAccelTable = NULL; - arrow = LoadCursor(NULL, IDC_ARROW); -} - -MainWnd::~MainWnd() -{ -} - - -BEGIN_MESSAGE_MAP(MainWnd, CWnd) - //{{AFX_MSG_MAP(MainWnd) - ON_WM_CLOSE() - ON_COMMAND(ID_HELP_ABOUT, OnHelpAbout) - ON_COMMAND(ID_HELP_FAQ, OnHelpFaq) - ON_COMMAND(ID_FILE_OPEN, OnFileOpen) - ON_WM_INITMENUPOPUP() - ON_COMMAND(ID_FILE_PAUSE, OnFilePause) - ON_UPDATE_COMMAND_UI(ID_FILE_PAUSE, OnUpdateFilePause) - ON_COMMAND(ID_FILE_RESET, OnFileReset) - ON_UPDATE_COMMAND_UI(ID_FILE_RESET, OnUpdateFileReset) - ON_UPDATE_COMMAND_UI(ID_FILE_RECENT_FREEZE, OnUpdateFileRecentFreeze) - ON_COMMAND(ID_FILE_RECENT_RESET, OnFileRecentReset) - ON_COMMAND(ID_FILE_RECENT_FREEZE, OnFileRecentFreeze) - ON_COMMAND(ID_FILE_EXIT, OnFileExit) - ON_COMMAND(ID_FILE_CLOSE, OnFileClose) - ON_UPDATE_COMMAND_UI(ID_FILE_CLOSE, OnUpdateFileClose) - ON_COMMAND(ID_FILE_OPENGAMEBOY, OnFileOpengameboy) - ON_COMMAND(ID_FILE_LOAD, OnFileLoad) - ON_UPDATE_COMMAND_UI(ID_FILE_LOAD, OnUpdateFileLoad) - ON_COMMAND(ID_FILE_SAVE, OnFileSave) - ON_UPDATE_COMMAND_UI(ID_FILE_SAVE, OnUpdateFileSave) - ON_COMMAND(ID_FILE_IMPORT_BATTERYFILE, OnFileImportBatteryfile) - ON_UPDATE_COMMAND_UI(ID_FILE_IMPORT_BATTERYFILE, OnUpdateFileImportBatteryfile) - ON_COMMAND(ID_FILE_IMPORT_GAMESHARKCODEFILE, OnFileImportGamesharkcodefile) - ON_UPDATE_COMMAND_UI(ID_FILE_IMPORT_GAMESHARKCODEFILE, OnUpdateFileImportGamesharkcodefile) - ON_COMMAND(ID_FILE_IMPORT_GAMESHARKSNAPSHOT, OnFileImportGamesharksnapshot) - ON_UPDATE_COMMAND_UI(ID_FILE_IMPORT_GAMESHARKSNAPSHOT, OnUpdateFileImportGamesharksnapshot) - ON_COMMAND(ID_FILE_EXPORT_BATTERYFILE, OnFileExportBatteryfile) - ON_UPDATE_COMMAND_UI(ID_FILE_EXPORT_BATTERYFILE, OnUpdateFileExportBatteryfile) - ON_COMMAND(ID_FILE_EXPORT_GAMESHARKSNAPSHOT, OnFileExportGamesharksnapshot) - ON_UPDATE_COMMAND_UI(ID_FILE_EXPORT_GAMESHARKSNAPSHOT, OnUpdateFileExportGamesharksnapshot) - ON_COMMAND(ID_FILE_SCREENCAPTURE, OnFileScreencapture) - ON_UPDATE_COMMAND_UI(ID_FILE_SCREENCAPTURE, OnUpdateFileScreencapture) - ON_COMMAND(ID_FILE_ROMINFORMATION, OnFileRominformation) - ON_UPDATE_COMMAND_UI(ID_FILE_ROMINFORMATION, OnUpdateFileRominformation) - ON_COMMAND(ID_FILE_TOGGLEMENU, OnFileTogglemenu) - ON_UPDATE_COMMAND_UI(ID_FILE_TOGGLEMENU, OnUpdateFileTogglemenu) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_FRAMESKIP_THROTTLE_NOTHROTTLE, OnUpdateOptionsFrameskipThrottleNothrottle) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_FRAMESKIP_THROTTLE_25, OnUpdateOptionsFrameskipThrottle25) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_FRAMESKIP_THROTTLE_50, OnUpdateOptionsFrameskipThrottle50) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_FRAMESKIP_THROTTLE_100, OnUpdateOptionsFrameskipThrottle100) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_FRAMESKIP_THROTTLE_150, OnUpdateOptionsFrameskipThrottle150) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_FRAMESKIP_THROTTLE_200, OnUpdateOptionsFrameskipThrottle200) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_FRAMESKIP_THROTTLE_OTHER, OnUpdateOptionsFrameskipThrottleOther) - ON_COMMAND(ID_OPTIONS_FRAMESKIP_THROTTLE_NOTHROTTLE, OnOptionsFrameskipThrottleNothrottle) - ON_COMMAND(ID_OPTIONS_FRAMESKIP_THROTTLE_25, OnOptionsFrameskipThrottle25) - ON_COMMAND(ID_OPTIONS_FRAMESKIP_THROTTLE_50, OnOptionsFrameskipThrottle50) - ON_COMMAND(ID_OPTIONS_FRAMESKIP_THROTTLE_100, OnOptionsFrameskipThrottle100) - ON_COMMAND(ID_OPTIONS_FRAMESKIP_THROTTLE_150, OnOptionsFrameskipThrottle150) - ON_COMMAND(ID_OPTIONS_FRAMESKIP_THROTTLE_200, OnOptionsFrameskipThrottle200) - ON_COMMAND(ID_OPTIONS_FRAMESKIP_THROTTLE_OTHER, OnOptionsFrameskipThrottleOther) - ON_COMMAND(ID_OPTIONS_FRAMESKIP_AUTOMATIC, OnOptionsFrameskipAutomatic) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_FRAMESKIP_AUTOMATIC, OnUpdateOptionsFrameskipAutomatic) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FRAMESKIP_0, OnUpdateOptionsVideoFrameskip0) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FRAMESKIP_1, OnUpdateOptionsVideoFrameskip1) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FRAMESKIP_2, OnUpdateOptionsVideoFrameskip2) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FRAMESKIP_3, OnUpdateOptionsVideoFrameskip3) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FRAMESKIP_4, OnUpdateOptionsVideoFrameskip4) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FRAMESKIP_5, OnUpdateOptionsVideoFrameskip5) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FRAMESKIP_6, OnUpdateOptionsVideoFrameskip6) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FRAMESKIP_7, OnUpdateOptionsVideoFrameskip7) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FRAMESKIP_8, OnUpdateOptionsVideoFrameskip8) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FRAMESKIP_9, OnUpdateOptionsVideoFrameskip9) - ON_COMMAND(ID_OPTIONS_VIDEO_VSYNC, OnOptionsVideoVsync) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_VSYNC, OnUpdateOptionsVideoVsync) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_X1, OnUpdateOptionsVideoX1) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_X2, OnUpdateOptionsVideoX2) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_X3, OnUpdateOptionsVideoX3) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_X4, OnUpdateOptionsVideoX4) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FULLSCREEN320X240, OnUpdateOptionsVideoFullscreen320x240) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FULLSCREEN640X480, OnUpdateOptionsVideoFullscreen640x480) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FULLSCREEN800X600, OnUpdateOptionsVideoFullscreen800x600) - ON_COMMAND(ID_OPTIONS_VIDEO_FULLSCREEN320X240, OnOptionsVideoFullscreen320x240) - ON_COMMAND(ID_OPTIONS_VIDEO_FULLSCREEN640X480, OnOptionsVideoFullscreen640x480) - ON_COMMAND(ID_OPTIONS_VIDEO_FULLSCREEN800X600, OnOptionsVideoFullscreen800x600) - ON_COMMAND(ID_OPTIONS_VIDEO_FULLSCREEN, OnOptionsVideoFullscreen) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FULLSCREEN, OnUpdateOptionsVideoFullscreen) - ON_WM_MOVE() - ON_WM_SIZE() - ON_COMMAND(ID_OPTIONS_VIDEO_DISABLESFX, OnOptionsVideoDisablesfx) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_DISABLESFX, OnUpdateOptionsVideoDisablesfx) - ON_COMMAND(ID_OPTIONS_VIDEO_FULLSCREENSTRETCHTOFIT, OnOptionsVideoFullscreenstretchtofit) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FULLSCREENSTRETCHTOFIT, OnUpdateOptionsVideoFullscreenstretchtofit) - ON_COMMAND(ID_OPTIONS_VIDEO_RENDERMETHOD_GDI, OnOptionsVideoRendermethodGdi) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_RENDERMETHOD_GDI, OnUpdateOptionsVideoRendermethodGdi) - ON_COMMAND(ID_OPTIONS_VIDEO_RENDERMETHOD_DIRECTDRAW, OnOptionsVideoRendermethodDirectdraw) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_RENDERMETHOD_DIRECTDRAW, OnUpdateOptionsVideoRendermethodDirectdraw) - ON_COMMAND(ID_OPTIONS_VIDEO_RENDERMETHOD_DIRECT3D, OnOptionsVideoRendermethodDirect3d) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_RENDERMETHOD_DIRECT3D, OnUpdateOptionsVideoRendermethodDirect3d) - ON_COMMAND(ID_OPTIONS_VIDEO_RENDERMETHOD_OPENGL, OnOptionsVideoRendermethodOpengl) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_RENDERMETHOD_OPENGL, OnUpdateOptionsVideoRendermethodOpengl) - ON_COMMAND(ID_OPTIONS_VIDEO_TRIPLEBUFFERING, OnOptionsVideoTriplebuffering) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_TRIPLEBUFFERING, OnUpdateOptionsVideoTriplebuffering) - ON_COMMAND(ID_OPTIONS_VIDEO_DDRAWEMULATIONONLY, OnOptionsVideoDdrawemulationonly) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_DDRAWEMULATIONONLY, OnUpdateOptionsVideoDdrawemulationonly) - ON_COMMAND(ID_OPTIONS_VIDEO_DDRAWUSEVIDEOMEMORY, OnOptionsVideoDdrawusevideomemory) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_DDRAWUSEVIDEOMEMORY, OnUpdateOptionsVideoDdrawusevideomemory) - ON_COMMAND(ID_OPTIONS_VIDEO_RENDEROPTIONS_D3DNOFILTER, OnOptionsVideoRenderoptionsD3dnofilter) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_RENDEROPTIONS_D3DNOFILTER, OnUpdateOptionsVideoRenderoptionsD3dnofilter) - ON_COMMAND(ID_OPTIONS_VIDEO_RENDEROPTIONS_D3DBILINEAR, OnOptionsVideoRenderoptionsD3dbilinear) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_RENDEROPTIONS_D3DBILINEAR, OnUpdateOptionsVideoRenderoptionsD3dbilinear) - ON_COMMAND(ID_OPTIONS_VIDEO_RENDEROPTIONS_GLNEAREST, OnOptionsVideoRenderoptionsGlnearest) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_RENDEROPTIONS_GLNEAREST, OnUpdateOptionsVideoRenderoptionsGlnearest) - ON_COMMAND(ID_OPTIONS_VIDEO_RENDEROPTIONS_GLBILINEAR, OnOptionsVideoRenderoptionsGlbilinear) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_RENDEROPTIONS_GLBILINEAR, OnUpdateOptionsVideoRenderoptionsGlbilinear) - ON_COMMAND(ID_OPTIONS_VIDEO_RENDEROPTIONS_GLTRIANGLE, OnOptionsVideoRenderoptionsGltriangle) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_RENDEROPTIONS_GLTRIANGLE, OnUpdateOptionsVideoRenderoptionsGltriangle) - ON_COMMAND(ID_OPTIONS_VIDEO_RENDEROPTIONS_GLQUADS, OnOptionsVideoRenderoptionsGlquads) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_RENDEROPTIONS_GLQUADS, OnUpdateOptionsVideoRenderoptionsGlquads) - ON_WM_CONTEXTMENU() - ON_COMMAND(ID_OPTIONS_EMULATOR_ASSOCIATE, OnOptionsEmulatorAssociate) - ON_COMMAND(ID_OPTIONS_EMULATOR_DIRECTORIES, OnOptionsEmulatorDirectories) - ON_COMMAND(ID_OPTIONS_EMULATOR_DISABLESTATUSMESSAGES, OnOptionsEmulatorDisablestatusmessages) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_DISABLESTATUSMESSAGES, OnUpdateOptionsEmulatorDisablestatusmessages) - ON_COMMAND(ID_OPTIONS_EMULATOR_SYNCHRONIZE, OnOptionsEmulatorSynchronize) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_SYNCHRONIZE, OnUpdateOptionsEmulatorSynchronize) - ON_COMMAND(ID_OPTIONS_EMULATOR_PAUSEWHENINACTIVE, OnOptionsEmulatorPausewheninactive) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_PAUSEWHENINACTIVE, OnUpdateOptionsEmulatorPausewheninactive) - ON_COMMAND(ID_OPTIONS_EMULATOR_SPEEDUPTOGGLE, OnOptionsEmulatorSpeeduptoggle) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_SPEEDUPTOGGLE, OnUpdateOptionsEmulatorSpeeduptoggle) - ON_COMMAND(ID_OPTIONS_EMULATOR_REMOVEINTROSGBA, OnOptionsEmulatorRemoveintrosgba) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_REMOVEINTROSGBA, OnUpdateOptionsEmulatorRemoveintrosgba) - ON_COMMAND(ID_OPTIONS_EMULATOR_AUTOMATICALLYIPSPATCH, OnOptionsEmulatorAutomaticallyipspatch) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_AUTOMATICALLYIPSPATCH, OnUpdateOptionsEmulatorAutomaticallyipspatch) - ON_COMMAND(ID_OPTIONS_EMULATOR_AGBPRINT, OnOptionsEmulatorAgbprint) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_AGBPRINT, OnUpdateOptionsEmulatorAgbprint) - ON_COMMAND(ID_OPTIONS_EMULATOR_REALTIMECLOCK, OnOptionsEmulatorRealtimeclock) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_REALTIMECLOCK, OnUpdateOptionsEmulatorRealtimeclock) - ON_COMMAND(ID_OPTIONS_EMULATOR_AUTOHIDEMENU, OnOptionsEmulatorAutohidemenu) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_AUTOHIDEMENU, OnUpdateOptionsEmulatorAutohidemenu) - ON_COMMAND(ID_OPTIONS_EMULATOR_REWINDINTERVAL, OnOptionsEmulatorRewindinterval) - ON_COMMAND(ID_OPTIONS_EMULATOR_SAVETYPE_AUTOMATIC, OnOptionsEmulatorSavetypeAutomatic) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_SAVETYPE_AUTOMATIC, OnUpdateOptionsEmulatorSavetypeAutomatic) - ON_COMMAND(ID_OPTIONS_EMULATOR_SAVETYPE_EEPROM, OnOptionsEmulatorSavetypeEeprom) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_SAVETYPE_EEPROM, OnUpdateOptionsEmulatorSavetypeEeprom) - ON_COMMAND(ID_OPTIONS_EMULATOR_SAVETYPE_SRAM, OnOptionsEmulatorSavetypeSram) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_SAVETYPE_SRAM, OnUpdateOptionsEmulatorSavetypeSram) - ON_COMMAND(ID_OPTIONS_EMULATOR_SAVETYPE_FLASH, OnOptionsEmulatorSavetypeFlash) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_SAVETYPE_FLASH, OnUpdateOptionsEmulatorSavetypeFlash) - ON_COMMAND(ID_OPTIONS_EMULATOR_SAVETYPE_EEPROMSENSOR, OnOptionsEmulatorSavetypeEepromsensor) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_SAVETYPE_EEPROMSENSOR, OnUpdateOptionsEmulatorSavetypeEepromsensor) - ON_COMMAND(ID_OPTIONS_EMULATOR_SAVETYPE_NONE, OnOptionsEmulatorSavetypeNone) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_SAVETYPE_NONE, OnUpdateOptionsEmulatorSavetypeNone) - ON_COMMAND(ID_OPTIONS_EMULATOR_SAVETYPE_FLASH512K, OnOptionsEmulatorSavetypeFlash512k) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_SAVETYPE_FLASH512K, OnUpdateOptionsEmulatorSavetypeFlash512k) - ON_COMMAND(ID_OPTIONS_EMULATOR_SAVETYPE_FLASH1M, OnOptionsEmulatorSavetypeFlash1m) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_SAVETYPE_FLASH1M, OnUpdateOptionsEmulatorSavetypeFlash1m) - ON_COMMAND(ID_OPTIONS_EMULATOR_USEBIOSFILE, OnOptionsEmulatorUsebiosfile) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_USEBIOSFILE, OnUpdateOptionsEmulatorUsebiosfile) - ON_COMMAND(ID_OPTIONS_EMULATOR_SKIPBIOS, OnOptionsEmulatorSkipbios) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_SKIPBIOS, OnUpdateOptionsEmulatorSkipbios) - ON_COMMAND(ID_OPTIONS_EMULATOR_SELECTBIOSFILE, OnOptionsEmulatorSelectbiosfile) - ON_COMMAND(ID_OPTIONS_EMULATOR_PNGFORMAT, OnOptionsEmulatorPngformat) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_PNGFORMAT, OnUpdateOptionsEmulatorPngformat) - ON_COMMAND(ID_OPTIONS_EMULATOR_BMPFORMAT, OnOptionsEmulatorBmpformat) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_BMPFORMAT, OnUpdateOptionsEmulatorBmpformat) -// ON_COMMAND(ID_OPTIONS_SOUND_OFF, OnOptionsSoundOff) /* mute hax */ -// ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_OFF, OnUpdateOptionsSoundOff) /* mute hax */ - ON_COMMAND(ID_OPTIONS_SOUND_OFF, OnOptionsSoundMute) /* mute hax */ - ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_OFF, OnUpdateOptionsSoundMute) /* mute hax */ - ON_COMMAND(ID_OPTIONS_SOUND_ON, OnOptionsSoundOn) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_ON, OnUpdateOptionsSoundOn) - ON_COMMAND(ID_OPTIONS_SOUND_USEOLDSYNCHRONIZATION, OnOptionsSoundUseoldsynchronization) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_USEOLDSYNCHRONIZATION, OnUpdateOptionsSoundUseoldsynchronization) - ON_COMMAND(ID_OPTIONS_SOUND_ECHO, OnOptionsSoundEcho) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_ECHO, OnUpdateOptionsSoundEcho) - ON_COMMAND(ID_OPTIONS_SOUND_LOWPASSFILTER, OnOptionsSoundLowpassfilter) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_LOWPASSFILTER, OnUpdateOptionsSoundLowpassfilter) - ON_COMMAND(ID_OPTIONS_SOUND_REVERSESTEREO, OnOptionsSoundReversestereo) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_REVERSESTEREO, OnUpdateOptionsSoundReversestereo) - ON_COMMAND(ID_OPTIONS_SOUND_11KHZ, OnOptionsSound11khz) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_11KHZ, OnUpdateOptionsSound11khz) - ON_COMMAND(ID_OPTIONS_SOUND_22KHZ, OnOptionsSound22khz) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_22KHZ, OnUpdateOptionsSound22khz) - ON_COMMAND(ID_OPTIONS_SOUND_44KHZ, OnOptionsSound44khz) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_44KHZ, OnUpdateOptionsSound44khz) - ON_COMMAND(ID_OPTIONS_SOUND_CHANNEL1, OnOptionsSoundChannel1) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_CHANNEL1, OnUpdateOptionsSoundChannel1) - ON_COMMAND(ID_OPTIONS_SOUND_CHANNEL2, OnOptionsSoundChannel2) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_CHANNEL2, OnUpdateOptionsSoundChannel2) - ON_COMMAND(ID_OPTIONS_SOUND_CHANNEL3, OnOptionsSoundChannel3) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_CHANNEL3, OnUpdateOptionsSoundChannel3) - ON_COMMAND(ID_OPTIONS_SOUND_CHANNEL4, OnOptionsSoundChannel4) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_CHANNEL4, OnUpdateOptionsSoundChannel4) - ON_COMMAND(ID_OPTIONS_SOUND_DIRECTSOUNDA, OnOptionsSoundDirectsounda) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_DIRECTSOUNDA, OnUpdateOptionsSoundDirectsounda) - ON_COMMAND(ID_OPTIONS_SOUND_DIRECTSOUNDB, OnOptionsSoundDirectsoundb) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_DIRECTSOUNDB, OnUpdateOptionsSoundDirectsoundb) - ON_COMMAND(ID_OPTIONS_GAMEBOY_BORDER, OnOptionsGameboyBorder) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_GAMEBOY_BORDER, OnUpdateOptionsGameboyBorder) - ON_COMMAND(ID_OPTIONS_GAMEBOY_PRINTER, OnOptionsGameboyPrinter) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_GAMEBOY_PRINTER, OnUpdateOptionsGameboyPrinter) - ON_COMMAND(ID_OPTIONS_GAMEBOY_BORDERAUTOMATIC, OnOptionsGameboyBorderAutomatic) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_GAMEBOY_BORDERAUTOMATIC, OnUpdateOptionsGameboyBorderAutomatic) - ON_COMMAND(ID_OPTIONS_GAMEBOY_AUTOMATIC, OnOptionsGameboyAutomatic) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_GAMEBOY_AUTOMATIC, OnUpdateOptionsGameboyAutomatic) - ON_COMMAND(ID_OPTIONS_GAMEBOY_GBA, OnOptionsGameboyGba) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_GAMEBOY_GBA, OnUpdateOptionsGameboyGba) - ON_COMMAND(ID_OPTIONS_GAMEBOY_CGB, OnOptionsGameboyCgb) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_GAMEBOY_CGB, OnUpdateOptionsGameboyCgb) - ON_COMMAND(ID_OPTIONS_GAMEBOY_SGB, OnOptionsGameboySgb) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_GAMEBOY_SGB, OnUpdateOptionsGameboySgb) - ON_COMMAND(ID_OPTIONS_GAMEBOY_SGB2, OnOptionsGameboySgb2) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_GAMEBOY_SGB2, OnUpdateOptionsGameboySgb2) - ON_COMMAND(ID_OPTIONS_GAMEBOY_GB, OnOptionsGameboyGb) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_GAMEBOY_GB, OnUpdateOptionsGameboyGb) - ON_COMMAND(ID_OPTIONS_GAMEBOY_REALCOLORS, OnOptionsGameboyRealcolors) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_GAMEBOY_REALCOLORS, OnUpdateOptionsGameboyRealcolors) - ON_COMMAND(ID_OPTIONS_GAMEBOY_GAMEBOYCOLORS, OnOptionsGameboyGameboycolors) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_GAMEBOY_GAMEBOYCOLORS, OnUpdateOptionsGameboyGameboycolors) - ON_COMMAND(ID_OPTIONS_GAMEBOY_COLORS, OnOptionsGameboyColors) - ON_COMMAND(ID_OPTIONS_FILTER_DISABLEMMX, OnOptionsFilterDisablemmx) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_FILTER_DISABLEMMX, OnUpdateOptionsFilterDisablemmx) - ON_COMMAND(ID_OPTIONS_LANGUAGE_SYSTEM, OnOptionsLanguageSystem) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_LANGUAGE_SYSTEM, OnUpdateOptionsLanguageSystem) - ON_COMMAND(ID_OPTIONS_LANGUAGE_ENGLISH, OnOptionsLanguageEnglish) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_LANGUAGE_ENGLISH, OnUpdateOptionsLanguageEnglish) - ON_COMMAND(ID_OPTIONS_LANGUAGE_OTHER, OnOptionsLanguageOther) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_LANGUAGE_OTHER, OnUpdateOptionsLanguageOther) - ON_COMMAND(ID_OPTIONS_JOYPAD_CONFIGURE_1, OnOptionsJoypadConfigure1) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_JOYPAD_CONFIGURE_1, OnUpdateOptionsJoypadConfigure1) - ON_COMMAND(ID_OPTIONS_JOYPAD_CONFIGURE_2, OnOptionsJoypadConfigure2) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_JOYPAD_CONFIGURE_2, OnUpdateOptionsJoypadConfigure2) - ON_COMMAND(ID_OPTIONS_JOYPAD_CONFIGURE_3, OnOptionsJoypadConfigure3) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_JOYPAD_CONFIGURE_3, OnUpdateOptionsJoypadConfigure3) - ON_COMMAND(ID_OPTIONS_JOYPAD_CONFIGURE_4, OnOptionsJoypadConfigure4) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_JOYPAD_CONFIGURE_4, OnUpdateOptionsJoypadConfigure4) - ON_COMMAND(ID_OPTIONS_JOYPAD_MOTIONCONFIGURE, OnOptionsJoypadMotionconfigure) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_JOYPAD_MOTIONCONFIGURE, OnUpdateOptionsJoypadMotionconfigure) - ON_COMMAND(ID_CHEATS_SEARCHFORCHEATS, OnCheatsSearchforcheats) - ON_UPDATE_COMMAND_UI(ID_CHEATS_SEARCHFORCHEATS, OnUpdateCheatsSearchforcheats) - ON_COMMAND(ID_CHEATS_CHEATLIST, OnCheatsCheatlist) - ON_UPDATE_COMMAND_UI(ID_CHEATS_CHEATLIST, OnUpdateCheatsCheatlist) - ON_COMMAND(ID_CHEATS_AUTOMATICSAVELOADCHEATS, OnCheatsAutomaticsaveloadcheats) - ON_COMMAND(ID_CHEATS_LOADCHEATLIST, OnCheatsLoadcheatlist) - ON_UPDATE_COMMAND_UI(ID_CHEATS_LOADCHEATLIST, OnUpdateCheatsLoadcheatlist) - ON_COMMAND(ID_CHEATS_SAVECHEATLIST, OnCheatsSavecheatlist) - ON_UPDATE_COMMAND_UI(ID_CHEATS_SAVECHEATLIST, OnUpdateCheatsSavecheatlist) - ON_COMMAND(ID_TOOLS_DISASSEMBLE, OnToolsDisassemble) - ON_UPDATE_COMMAND_UI(ID_TOOLS_DISASSEMBLE, OnUpdateToolsDisassemble) - ON_COMMAND(ID_TOOLS_LOGGING, OnToolsLogging) - ON_UPDATE_COMMAND_UI(ID_TOOLS_LOGGING, OnUpdateToolsLogging) - ON_COMMAND(ID_TOOLS_IOVIEWER, OnToolsIoviewer) - ON_UPDATE_COMMAND_UI(ID_TOOLS_IOVIEWER, OnUpdateToolsIoviewer) - ON_COMMAND(ID_TOOLS_MAPVIEW, OnToolsMapview) - ON_UPDATE_COMMAND_UI(ID_TOOLS_MAPVIEW, OnUpdateToolsMapview) - ON_COMMAND(ID_TOOLS_MEMORYVIEWER, OnToolsMemoryviewer) - ON_UPDATE_COMMAND_UI(ID_TOOLS_MEMORYVIEWER, OnUpdateToolsMemoryviewer) - ON_COMMAND(ID_TOOLS_OAMVIEWER, OnToolsOamviewer) - ON_UPDATE_COMMAND_UI(ID_TOOLS_OAMVIEWER, OnUpdateToolsOamviewer) - ON_COMMAND(ID_TOOLS_PALETTEVIEW, OnToolsPaletteview) - ON_UPDATE_COMMAND_UI(ID_TOOLS_PALETTEVIEW, OnUpdateToolsPaletteview) - ON_COMMAND(ID_TOOLS_TILEVIEWER, OnToolsTileviewer) - ON_UPDATE_COMMAND_UI(ID_TOOLS_TILEVIEWER, OnUpdateToolsTileviewer) - ON_COMMAND(ID_DEBUG_NEXTFRAME, OnDebugNextframe) - ON_UPDATE_COMMAND_UI(ID_CHEATS_AUTOMATICSAVELOADCHEATS, OnUpdateCheatsAutomaticsaveloadcheats) - ON_COMMAND(ID_TOOLS_DEBUG_GDB, OnToolsDebugGdb) - ON_UPDATE_COMMAND_UI(ID_TOOLS_DEBUG_GDB, OnUpdateToolsDebugGdb) - ON_COMMAND(ID_TOOLS_DEBUG_LOADANDWAIT, OnToolsDebugLoadandwait) - ON_UPDATE_COMMAND_UI(ID_TOOLS_DEBUG_LOADANDWAIT, OnUpdateToolsDebugLoadandwait) - ON_COMMAND(ID_TOOLS_DEBUG_BREAK, OnToolsDebugBreak) - ON_UPDATE_COMMAND_UI(ID_TOOLS_DEBUG_BREAK, OnUpdateToolsDebugBreak) - ON_COMMAND(ID_TOOLS_DEBUG_DISCONNECT, OnToolsDebugDisconnect) - ON_UPDATE_COMMAND_UI(ID_TOOLS_DEBUG_DISCONNECT, OnUpdateToolsDebugDisconnect) - ON_COMMAND(ID_OPTIONS_SOUND_STARTRECORDING, OnOptionsSoundStartrecording) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_STARTRECORDING, OnUpdateOptionsSoundStartrecording) - ON_COMMAND(ID_OPTIONS_SOUND_STOPRECORDING, OnOptionsSoundStoprecording) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_STOPRECORDING, OnUpdateOptionsSoundStoprecording) - ON_COMMAND(ID_TOOLS_RECORD_STARTAVIRECORDING, OnToolsRecordStartavirecording) - ON_UPDATE_COMMAND_UI(ID_TOOLS_RECORD_STARTAVIRECORDING, OnUpdateToolsRecordStartavirecording) - ON_COMMAND(ID_TOOLS_RECORD_STOPAVIRECORDING, OnToolsRecordStopavirecording) - ON_UPDATE_COMMAND_UI(ID_TOOLS_RECORD_STOPAVIRECORDING, OnUpdateToolsRecordStopavirecording) - ON_WM_PAINT() - ON_COMMAND(ID_TOOLS_RECORD_STARTMOVIERECORDING, OnToolsRecordStartmovierecording) - ON_UPDATE_COMMAND_UI(ID_TOOLS_RECORD_STARTMOVIERECORDING, OnUpdateToolsRecordStartmovierecording) - ON_COMMAND(ID_TOOLS_RECORD_STOPMOVIERECORDING, OnToolsRecordStopmovierecording) - ON_UPDATE_COMMAND_UI(ID_TOOLS_RECORD_STOPMOVIERECORDING, OnUpdateToolsRecordStopmovierecording) - ON_COMMAND(ID_TOOLS_PLAY_STARTMOVIEPLAYING, OnToolsPlayStartmovieplaying) - ON_UPDATE_COMMAND_UI(ID_TOOLS_PLAY_STARTMOVIEPLAYING, OnUpdateToolsPlayStartmovieplaying) - ON_COMMAND(ID_TOOLS_PLAY_STOPMOVIEPLAYING, OnToolsPlayStopmovieplaying) - ON_UPDATE_COMMAND_UI(ID_TOOLS_PLAY_STOPMOVIEPLAYING, OnUpdateToolsPlayStopmovieplaying) - ON_COMMAND(ID_TOOLS_REWIND, OnToolsRewind) - ON_UPDATE_COMMAND_UI(ID_TOOLS_REWIND, OnUpdateToolsRewind) - ON_COMMAND(ID_TOOLS_CUSTOMIZE, OnToolsCustomize) - ON_UPDATE_COMMAND_UI(ID_TOOLS_CUSTOMIZE, OnUpdateToolsCustomize) - ON_COMMAND(ID_HELP_BUGREPORT, OnHelpBugreport) - ON_WM_MOUSEMOVE() - ON_WM_INITMENU() - ON_WM_ACTIVATE() - ON_WM_ACTIVATEAPP() - ON_WM_DROPFILES() - ON_COMMAND(ID_FILE_SAVEGAME_OLDESTSLOT, OnFileSavegameOldestslot) - ON_UPDATE_COMMAND_UI(ID_FILE_SAVEGAME_OLDESTSLOT, OnUpdateFileSavegameOldestslot) - ON_COMMAND(ID_FILE_LOADGAME_MOSTRECENT, OnFileLoadgameMostrecent) - ON_UPDATE_COMMAND_UI(ID_FILE_LOADGAME_MOSTRECENT, OnUpdateFileLoadgameMostrecent) - ON_COMMAND(ID_FILE_LOADGAME_AUTOLOADMOSTRECENT, OnFileLoadgameAutoloadmostrecent) - ON_UPDATE_COMMAND_UI(ID_FILE_LOADGAME_AUTOLOADMOSTRECENT, OnUpdateFileLoadgameAutoloadmostrecent) - ON_COMMAND(ID_OPTIONS_SOUND_VOLUME_25X, OnOptionsSoundVolume25x) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_VOLUME_25X, OnUpdateOptionsSoundVolume25x) - ON_COMMAND(ID_OPTIONS_SOUND_VOLUME_5X, OnOptionsSoundVolume5x) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_VOLUME_5X, OnUpdateOptionsSoundVolume5x) - ON_COMMAND(ID_CHEATS_DISABLECHEATS, OnCheatsDisablecheats) - ON_UPDATE_COMMAND_UI(ID_CHEATS_DISABLECHEATS, OnUpdateCheatsDisablecheats) - ON_COMMAND(ID_OPTIONS_VIDEO_FULLSCREENMAXSCALE, OnOptionsVideoFullscreenmaxscale) - - ON_COMMAND(ID_OPTIONS_LINK_OPTIONS, OnLinkOptions) - ON_COMMAND(ID_OPTIONS_LINK_LOG, OnOptionsLinkLog) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_LINK_LOG, OnUpdateOptionsLinkLog) - ON_COMMAND(ID_OPTIONS_LINK_WIRELESSADAPTER, OnOptionsLinkRFU) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_LINK_WIRELESSADAPTER, OnUpdateOptionsLinkRFU) - //}}AFX_MSG_MAP - ON_COMMAND_EX_RANGE(ID_FILE_MRU_FILE1, ID_FILE_MRU_FILE10, OnFileRecentFile) - ON_COMMAND_EX_RANGE(ID_FILE_LOADGAME_SLOT1, ID_FILE_LOADGAME_SLOT10, OnFileLoadSlot) - ON_COMMAND_EX_RANGE(ID_FILE_SAVEGAME_SLOT1, ID_FILE_SAVEGAME_SLOT10, OnFileSaveSlot) - ON_UPDATE_COMMAND_UI_RANGE(ID_FILE_LOADGAME_SLOT1, ID_FILE_LOADGAME_SLOT10, OnUpdateFileLoadGameSlot) - ON_UPDATE_COMMAND_UI_RANGE(ID_FILE_SAVEGAME_SLOT1, ID_FILE_SAVEGAME_SLOT10, OnUpdateFileSaveGameSlot) - ON_COMMAND_EX_RANGE(ID_OPTIONS_VIDEO_FRAMESKIP_0, ID_OPTIONS_VIDEO_FRAMESKIP_5, OnOptionsFrameskip) - ON_COMMAND_EX_RANGE(ID_OPTIONS_VIDEO_FRAMESKIP_6, ID_OPTIONS_VIDEO_FRAMESKIP_9, OnOptionsFrameskip) - ON_COMMAND_EX_RANGE(ID_OPTIONS_VIDEO_X1, ID_OPTIONS_VIDEO_X4, OnOptionVideoSize) - ON_COMMAND_EX_RANGE(ID_OPTIONS_VIDEO_LAYERS_BG0, ID_OPTIONS_VIDEO_LAYERS_OBJWIN, OnVideoLayer) - ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_VIDEO_LAYERS_BG0, ID_OPTIONS_VIDEO_LAYERS_OBJWIN, OnUpdateVideoLayer) - ON_COMMAND_EX_RANGE(ID_OPTIONS_EMULATOR_SHOWSPEED_NONE, ID_OPTIONS_EMULATOR_SHOWSPEED_TRANSPARENT, OnOptionsEmulatorShowSpeed) - ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_EMULATOR_SHOWSPEED_NONE, ID_OPTIONS_EMULATOR_SHOWSPEED_TRANSPARENT, OnUpdateOptionsEmulatorShowSpeed) - ON_COMMAND_EX_RANGE(ID_OPTIONS_SOUND_VOLUME_1X, ID_OPTIONS_SOUND_VOLUME_4X, OnOptionsSoundVolume) - ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_SOUND_VOLUME_1X, ID_OPTIONS_SOUND_VOLUME_4X, OnUpdateOptionsSoundVolume) - ON_COMMAND_EX_RANGE(ID_OPTIONS_PRIORITY_HIGHEST, ID_OPTIONS_PRIORITY_BELOWNORMAL, OnOptionsPriority) - ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_PRIORITY_HIGHEST, ID_OPTIONS_PRIORITY_BELOWNORMAL, OnUpdateOptionsPriority) - ON_COMMAND_EX_RANGE(ID_OPTIONS_FILTER_NORMAL, ID_OPTIONS_FILTER_TVMODE, OnOptionsFilter) - ON_COMMAND_EX(ID_OPTIONS_FILTER16BIT_PIXELATEEXPERIMENTAL, OnOptionsFilter) - ON_COMMAND_EX_RANGE(ID_OPTIONS_FILTER16BIT_ADVANCEMAMESCALE2X, ID_OPTIONS_FILTER16BIT_SIMPLE2X, OnOptionsFilter) - ON_COMMAND_EX_RANGE(ID_OPTIONS_FILTER_BILINEAR, ID_OPTIONS_FILTER_BILINEARPLUS, OnOptionsFilter) - ON_COMMAND_EX_RANGE(ID_OPTIONS_FILTER_SCANLINES, ID_OPTIONS_FILTER_SCANLINES, OnOptionsFilter) - ON_COMMAND_EX_RANGE(ID_OPTIONS_FILTER_HQ2X, ID_OPTIONS_FILTER_LQ2X, OnOptionsFilter) - ON_COMMAND_EX(ID_OPTIONS_FILTER_HQ3X, OnOptionsFilter) - ON_COMMAND_EX(ID_OPTIONS_FILTER_HQ4X, OnOptionsFilter) - ON_COMMAND_EX(ID_OPTIONS_FILTER_SIMPLE3X, OnOptionsFilter) - ON_COMMAND_EX(ID_OPTIONS_FILTER_SIMPLE4X, OnOptionsFilter) - ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_FILTER_NORMAL, ID_OPTIONS_FILTER_TVMODE, OnUpdateOptionsFilter) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_FILTER16BIT_PIXELATEEXPERIMENTAL, OnUpdateOptionsFilter) - ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_FILTER16BIT_ADVANCEMAMESCALE2X, ID_OPTIONS_FILTER16BIT_SIMPLE2X, OnUpdateOptionsFilter) - ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_FILTER_BILINEAR, ID_OPTIONS_FILTER_BILINEARPLUS, OnUpdateOptionsFilter) - ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_FILTER_SCANLINES, ID_OPTIONS_FILTER_SCANLINES, OnUpdateOptionsFilter) - ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_FILTER_HQ2X, ID_OPTIONS_FILTER_LQ2X, OnUpdateOptionsFilter) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_FILTER_SIMPLE3X, OnUpdateOptionsFilter) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_FILTER_SIMPLE4X, OnUpdateOptionsFilter) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_FILTER_HQ3X, OnUpdateOptionsFilter) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_FILTER_HQ4X, OnUpdateOptionsFilter) - ON_COMMAND_EX_RANGE(ID_OPTIONS_FILTER_INTERFRAMEBLENDING_NONE, ID_OPTIONS_FILTER_INTERFRAMEBLENDING_SMART, OnOptionsFilterIFB) - ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_FILTER_INTERFRAMEBLENDING_NONE, ID_OPTIONS_FILTER_INTERFRAMEBLENDING_SMART, OnUpdateOptionsFilterIFB) - ON_COMMAND_EX_RANGE(ID_OPTIONS_JOYPAD_DEFAULTJOYPAD_1, ID_OPTIONS_JOYPAD_DEFAULTJOYPAD_4, OnOptionsJoypadDefault) - ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_JOYPAD_DEFAULTJOYPAD_1, ID_OPTIONS_JOYPAD_DEFAULTJOYPAD_4, OnUpdateOptionsJoypadDefault) - ON_COMMAND_EX_RANGE(ID_OPTIONS_JOYPAD_AUTOFIRE_A, ID_OPTIONS_JOYPAD_AUTOFIRE_R, OnOptionsJoypadAutofire) - ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_JOYPAD_AUTOFIRE_A, ID_OPTIONS_JOYPAD_AUTOFIRE_R, OnUpdateOptionsJoypadAutofire) - ON_MESSAGE(VBA_CONFIRM_MODE, OnConfirmMode) - ON_MESSAGE(WM_SYSCOMMAND, OnMySysCommand) - ON_COMMAND(ID_OPTIONS_VIDEO_FULLSCREEN1280X1024, OnOptionsVideoFullscreen1280x1024) - ON_COMMAND(ID_OPTIONS_VIDEO_FULLSCREEN1024X768, OnOptionsVideoFullscreen1024x768) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FULLSCREEN1024X768, OnUpdateOptionsVideoFullscreen1024x768) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FULLSCREEN1280X1024, OnUpdateOptionsVideoFullscreen1280x1024) - ON_COMMAND(ID_OPTIONS_FILTER_LCDCOLORS, OnOptionsFilterLcdcolors) - ON_UPDATE_COMMAND_UI(ID_OPTIONS_FILTER_LCDCOLORS, OnUpdateOptionsFilterLcdcolors) - ON_COMMAND_EX_RANGE(ID_OPTIONS_SOUND_PCMINTERPOLATION_NONE, ID_OPTIONS_SOUND_PCMINTERPOLATION_LIBRESAMPLE, OnOptionsSoundPcminterpolation) - ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_SOUND_PCMINTERPOLATION_NONE, ID_OPTIONS_SOUND_PCMINTERPOLATION_LIBRESAMPLE, OnUpdateOptionsSoundPcminterpolation) - ON_WM_SETFOCUS() - ON_WM_KILLFOCUS() - END_MESSAGE_MAP() - - - ///////////////////////////////////////////////////////////////////////////// -// MainWnd message handlers - -void MainWnd::OnClose() -{ - CWnd::OnClose(); - - delete this; -} - -bool MainWnd::FileRun() -{ - // save battery file before we change the filename... - if(rom != NULL || gbRom != NULL) { - if(theApp.autoSaveLoadCheatList) - winSaveCheatListDefault(); - writeBatteryFile(); - cheatSearchCleanup(&cheatSearchData); - theApp.emulator.emuCleanUp(); - remoteCleanUp(); - emulating = false; -#ifdef APU_LOGGER_H - end_apu_log(); -#endif - } - char tempName[2048]; - char file[2048]; - - utilGetBaseName(theApp.szFile, tempName); - - _fullpath(file, tempName, 1024); - theApp.filename = file; - - int index = theApp.filename.ReverseFind('.'); - if(index != -1) - theApp.filename = theApp.filename.Left(index); - - CString ipsname; - ipsname.Format("%s.ips", theApp.filename); - - if(!theApp.dir.GetLength()) { - int index = theApp.filename.ReverseFind('\\'); - if(index != -1) { - theApp.dir = theApp.filename.Left(index-1); - } - } - - IMAGE_TYPE type = utilFindType(theApp.szFile); - - if(type == IMAGE_UNKNOWN) { - systemMessage(IDS_UNSUPPORTED_FILE_TYPE, - "Unsupported file type: %s", theApp.szFile); - return false; - } - systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; - theApp.cartridgeType = (int)type; - if(type == IMAGE_GB) { - if(!gbLoadRom(theApp.szFile)) - return false; - theApp.emulator = GBSystem; - gbBorderOn = theApp.winGbBorderOn; - theApp.romSize = gbRomSize; - if(theApp.autoIPS) { - int size = gbRomSize; - utilApplyIPS(ipsname, &gbRom, &size); - if(size != gbRomSize) { - extern bool gbUpdateSizes(); - gbUpdateSizes(); - gbReset(); - theApp.romSize = size; - } - } - } else { - int size = CPULoadRom(theApp.szFile); - if(!size) - return false; - - theApp.romSize = size; - - flashSetSize(theApp.winFlashSize); - rtcEnable(theApp.winRtcEnable); - cpuSaveType = theApp.winSaveType; - - // if(cpuEnhancedDetection && winSaveType == 0) { - // utilGBAFindSave(rom, size); - // } - GetModuleFileName(NULL, tempName, 2048); - - char *p = strrchr(tempName, '\\'); - if(p) - *p = 0; - - char buffer[5]; - strncpy(buffer, (const char *)&rom[0xac], 4); - buffer[4] = 0; - - strcat(tempName, "\\vba-over.ini"); - - UINT i = GetPrivateProfileInt(buffer, - "rtcEnabled", - -1, - tempName); - if(i != (UINT)-1) - rtcEnable(i == 0 ? false : true); - - i = GetPrivateProfileInt(buffer, - "flashSize", - -1, - tempName); - if(i != (UINT)-1 && (i == 0x10000 || i == 0x20000)) - flashSetSize((int)i); - - i = GetPrivateProfileInt(buffer, - "saveType", - -1, - tempName); - if(i != (UINT)-1 && (i <= 5)) - cpuSaveType = (int)i; - - theApp.emulator = GBASystem; - /* disabled due to problems - if(theApp.removeIntros && rom != NULL) { - *((u32 *)rom)= 0xea00002e; - } - */ - - if(theApp.autoIPS) { - int size = 0x2000000; - utilApplyIPS(ipsname, &rom, &size); - if(size != 0x2000000) { - CPUReset(); - } - } - } - - if(theApp.soundInitialized) { - if(theApp.cartridgeType == 1) - gbSoundReset(); - else - soundReset(); - } else { - //if(!soundOffFlag) /* mute hax */ - soundInit(!theApp.cartridgeType); - if ( soundOffFlag ) - soundDisable(0x30f); - theApp.soundInitialized = true; - } - -#ifdef APU_LOGGER_H - begin_apu_log("apu_log.txt"); -#endif - - if(type == IMAGE_GBA) { - skipBios = theApp.skipBiosFile ? true : false; - CPUInit((char *)(LPCTSTR)theApp.biosFileName, theApp.useBiosFile ? true : false); - CPUReset(); - } - - readBatteryFile(); - - if(theApp.autoSaveLoadCheatList) - winLoadCheatListDefault(); - - theApp.addRecentFile(theApp.szFile); - - theApp.updateWindowSize(theApp.videoOption); - - theApp.updateFrameSkip(); - - if(theApp.autoHideMenu && theApp.videoOption > VIDEO_4X && theApp.menuToggle) - OnFileTogglemenu(); - - emulating = true; - - if(theApp.autoLoadMostRecent) - OnFileLoadgameMostrecent(); - - theApp.frameskipadjust = 0; - theApp.renderedFrames = 0; - theApp.autoFrameSkipLastTime = theApp.throttleLastTime = systemGetClock(); - - theApp.rewindCount = 0; - theApp.rewindCounter = 0; - theApp.rewindSaveNeeded = false; - - return true; -} - -void MainWnd::OnInitMenuPopup(CMenu* pMenu, UINT nIndex, BOOL bSysMenu) -{ - ASSERT(pMenu != NULL); - - CCmdUI state; - state.m_pMenu = pMenu; - ASSERT(state.m_pOther == NULL); - ASSERT(state.m_pParentMenu == NULL); - - // determine if menu is popup in top-level menu and set m_pOther to - // it if so (m_pParentMenu == NULL indicates that it is secondary popup) - HMENU hParentMenu; - if (AfxGetThreadState()->m_hTrackingMenu == pMenu->m_hMenu) - state.m_pParentMenu = pMenu; // parent == child for tracking popup - else if ((hParentMenu = ::GetMenu(m_hWnd)) != NULL) { - CWnd* pParent = GetTopLevelParent(); - // child windows don't have menus -- need to go to the top! - if (pParent != NULL && - (hParentMenu = ::GetMenu(pParent->m_hWnd)) != NULL) { - int nIndexMax = ::GetMenuItemCount(hParentMenu); - for (int nIndex = 0; nIndex < nIndexMax; nIndex++) { - if (::GetSubMenu(hParentMenu, nIndex) == pMenu->m_hMenu) { - // when popup is found, m_pParentMenu is containing menu - state.m_pParentMenu = CMenu::FromHandle(hParentMenu); - break; - } - } - } - } - - state.m_nIndexMax = pMenu->GetMenuItemCount(); - for (state.m_nIndex = 0; state.m_nIndex < state.m_nIndexMax; - state.m_nIndex++) { - state.m_nID = pMenu->GetMenuItemID(state.m_nIndex); - if (state.m_nID == 0) - continue; // menu separator or invalid cmd - ignore it - - ASSERT(state.m_pOther == NULL); - ASSERT(state.m_pMenu != NULL); - if (state.m_nID == (UINT)-1) { - // possibly a popup menu, route to first item of that popup - state.m_pSubMenu = pMenu->GetSubMenu(state.m_nIndex); - if (state.m_pSubMenu == NULL || - (state.m_nID = state.m_pSubMenu->GetMenuItemID(0)) == 0 || - state.m_nID == (UINT)-1) { - continue; // first item of popup can't be routed to - } - state.DoUpdate(this, FALSE); // popups are never auto disabled - } else { - // normal menu item - // Auto enable/disable if frame window has 'm_bAutoMenuEnable' - // set and command is _not_ a system command. - state.m_pSubMenu = NULL; - state.DoUpdate(this, state.m_nID < 0xF000); - } - - // adjust for menu deletions and additions - UINT nCount = pMenu->GetMenuItemCount(); - if (nCount < state.m_nIndexMax) { - state.m_nIndex -= (state.m_nIndexMax - nCount); - while (state.m_nIndex < nCount && - pMenu->GetMenuItemID(state.m_nIndex) == state.m_nID) { - state.m_nIndex++; - } - } - state.m_nIndexMax = nCount; - } -} - -void MainWnd::OnMove(int x, int y) -{ - CWnd::OnMove(x, y); - - if(!theApp.changingVideoSize) { - if(this) { - if(!IsIconic()) { - RECT r; - - GetWindowRect(&r); - theApp.windowPositionX = r.left; - theApp.windowPositionY = r.top; - theApp.adjustDestRect(); - regSetDwordValue("windowX", theApp.windowPositionX); - regSetDwordValue("windowY", theApp.windowPositionY); - } - } - } -} - -void MainWnd::OnSize(UINT nType, int cx, int cy) -{ - CWnd::OnSize(nType, cx, cy); - - if(!theApp.changingVideoSize) { - if(this) { - if(!IsIconic()) { - if(theApp.iconic) { - if(emulating) { - soundResume(); - theApp.paused = false; - } - } - if(theApp.videoOption <= VIDEO_4X) { - theApp.surfaceSizeX = cx; - theApp.surfaceSizeY = cy; - theApp.adjustDestRect(); - if(theApp.display) - theApp.display->resize(theApp.dest.right-theApp.dest.left, theApp.dest.bottom-theApp.dest.top); - } - } else { - if(emulating) { - if(!theApp.paused) { - theApp.paused = true; - soundPause(); - } - } - theApp.iconic = true; - } - } - } -} - -void MainWnd::winSaveCheatListDefault() -{ - CString name; - CString filename; - - int index = theApp.filename.ReverseFind('\\'); - - if(index != -1) - name = theApp.filename.Right(theApp.filename.GetLength()-index-1); - else - name = theApp.filename; - CString dir = regQueryStringValue("saveDir", NULL); - - if(!dir.GetLength()) - dir = getDirFromFile(filename); - - if(isDriveRoot(dir)) - filename.Format("%s%s.clt", dir, name); - else - filename.Format("%s\\%s.clt", dir, name); - - winSaveCheatList(filename); -} - -void MainWnd::winSaveCheatList(const char *name) -{ - if(theApp.cartridgeType == 0) - cheatsSaveCheatList(name); - else - gbCheatsSaveCheatList(name); -} - -void MainWnd::winLoadCheatListDefault() -{ - CString name; - CString filename; - - int index = theApp.filename.ReverseFind('\\'); - - if(index != -1) - name = theApp.filename.Right(theApp.filename.GetLength()-index-1); - else - name = theApp.filename; - CString dir = regQueryStringValue("saveDir", NULL); - - if(!dir.GetLength()) - dir = getDirFromFile(filename); - - if(isDriveRoot(dir)) - filename.Format("%s%s.clt", dir, name); - else - filename.Format("%s\\%s.clt", dir, name); - - winLoadCheatList(filename); -} - -void MainWnd::winLoadCheatList(const char *name) -{ - bool res = false; - - if(theApp.cartridgeType == 0) - res = cheatsLoadCheatList(name); - else - res = gbCheatsLoadCheatList(name); - - if(res) - systemScreenMessage(winResLoadString(IDS_LOADED_CHEATS)); -} - -CString MainWnd::getDirFromFile(CString& file) -{ - CString temp = file; - int index = temp.ReverseFind('\\'); - - if(index != -1) { - temp = temp.Left(index); - if(temp.GetLength() == 2 && temp[1] == ':') - temp += "\\"; - } else { - temp = ""; - } - return temp; -} - -bool MainWnd::isDriveRoot(CString& file) -{ - if(file.GetLength() == 3) { - if(file[1] == ':' && file[2] == '\\') - return true; - } - return false; -} - -void MainWnd::writeBatteryFile() -{ - CString buffer; - CString filename; - - int index = theApp.filename.ReverseFind('\\'); - - if(index != -1) - buffer = theApp.filename.Right(theApp.filename.GetLength()-index-1); - else - buffer = theApp.filename; - - CString saveDir = regQueryStringValue("batteryDir", NULL); - - if(saveDir.IsEmpty()) - saveDir = getDirFromFile(theApp.filename); - - if(isDriveRoot(saveDir)) - filename.Format("%s%s.sav", saveDir, buffer); - else - filename.Format("%s\\%s.sav", saveDir, buffer); - - if(theApp.emulator.emuWriteBattery) - theApp.emulator.emuWriteBattery(filename); -} - - -void MainWnd::readBatteryFile() -{ - CString buffer; - CString filename; - - int index = theApp.filename.ReverseFind('\\'); - - if(index != -1) - buffer = theApp.filename.Right(theApp.filename.GetLength()-index-1); - else - buffer = theApp.filename; - - CString saveDir = regQueryStringValue("batteryDir", NULL); - - if(saveDir.IsEmpty()) - saveDir = getDirFromFile(theApp.filename); - - if(isDriveRoot(saveDir)) - filename.Format("%s%s.sav", saveDir, buffer); - else - filename.Format("%s\\%s.sav", saveDir, buffer); - - bool res = false; - - if(theApp.emulator.emuReadBattery) - res = theApp.emulator.emuReadBattery(filename); - - if(res) - systemScreenMessage(winResLoadString(IDS_LOADED_BATTERY)); -} - -CString MainWnd::winLoadFilter(UINT id) -{ - CString res = winResLoadString(id); - res.Replace('_','|'); - - return res; -} - -bool MainWnd::loadSaveGame(const char *name) -{ - if(theApp.emulator.emuReadState) - return theApp.emulator.emuReadState(name); - return false; -} - -bool MainWnd::writeSaveGame(const char *name) -{ - if(theApp.emulator.emuWriteState) - return theApp.emulator.emuWriteState(name); - return false; -} - -void MainWnd::OnContextMenu(CWnd* pWnd, CPoint point) -{ - winMouseOn(); -} - -bool MainWnd::fileOpenSelect() -{ - theApp.dir = ""; - CString initialDir = regQueryStringValue("romdir","."); - if(!initialDir.IsEmpty()) - theApp.dir = initialDir; - - int selectedFilter = regQueryDwordValue("selectedFilter", 0); - if(selectedFilter < 0 || selectedFilter > 2) - selectedFilter = 0; - - theApp.szFile = ""; - - LPCTSTR exts[] = { "" }; - CString filter = winLoadFilter(IDS_FILTER_ROM); - CString title = winResLoadString(IDS_SELECT_ROM); - - FileDlg dlg(this, "", filter, selectedFilter, "", exts, theApp.dir, title, false); - - if(dlg.DoModal() == IDOK) { - regSetDwordValue("selectedFilter", dlg.m_ofn.nFilterIndex); - theApp.szFile = dlg.GetPathName(); - theApp.dir = theApp.szFile.Left(dlg.m_ofn.nFileOffset); - if(theApp.dir.GetLength() > 3 && theApp.dir[theApp.dir.GetLength()-1] == '\\') - theApp.dir = theApp.dir.Left(theApp.dir.GetLength()-1); - regSetStringValue("romdir", theApp.dir); - return true; - } - return false; -} - -void MainWnd::OnPaint() -{ - CPaintDC dc(this); // device context for painting - - if(emulating) { - theApp.painting = true; - systemDrawScreen(); - theApp.painting = false; - theApp.renderedFrames--; - } -} - -BOOL MainWnd::PreTranslateMessage(MSG* pMsg) -{ - if (CWnd::PreTranslateMessage(pMsg)) - return TRUE; - - if(pMsg->message >= WM_KEYFIRST && pMsg->message <= WM_KEYLAST) { - return theApp.hAccel != NULL && ::TranslateAccelerator(m_hWnd, theApp.hAccel, pMsg); - } - - return FALSE; -} - -void MainWnd::screenCapture(int captureNumber) -{ - CString buffer; - - CString captureDir = regQueryStringValue("captureDir", ""); - int index = theApp.filename.ReverseFind('\\'); - - CString name; - if(index != -1) - name = theApp.filename.Right(theApp.filename.GetLength()-index-1); - else - name = theApp.filename; - - if(captureDir.IsEmpty()) - captureDir = getDirFromFile(theApp.filename); - - LPCTSTR ext = "png"; - if(theApp.captureFormat != 0) - ext = "bmp"; - - if(isDriveRoot(captureDir)) - buffer.Format("%s%s_%02d.%s", - captureDir, - name, - captureNumber, - ext); - else - buffer.Format("%s\\%s_%02d.%s", - captureDir, - name, - captureNumber, - ext); - - if(theApp.captureFormat == 0) - theApp.emulator.emuWritePNG(buffer); - else - theApp.emulator.emuWriteBMP(buffer); - - CString msg = winResLoadString(IDS_SCREEN_CAPTURE); - systemScreenMessage(msg); -} - -void MainWnd::winMouseOn() -{ - SetCursor(arrow); - if(theApp.videoOption > VIDEO_4X) { - theApp.mouseCounter = 10; - } else - theApp.mouseCounter = 0; -} - -void MainWnd::OnMouseMove(UINT nFlags, CPoint point) -{ - winMouseOn(); - - CWnd::OnMouseMove(nFlags, point); -} - -void MainWnd::OnInitMenu(CMenu* pMenu) -{ - CWnd::OnInitMenu(pMenu); - - soundPause(); -} - -void MainWnd::OnActivate(UINT nState, CWnd* pWndOther, BOOL bMinimized) -{ - CWnd::OnActivate(nState, pWndOther, bMinimized); - - bool a = (nState == WA_ACTIVE) || (nState == WA_CLICKACTIVE); - - if(a && theApp.input) { - theApp.active = a; - theApp.input->activate(); - if(!theApp.paused) { - if(emulating) { - soundResume(); - } - } - } else { - theApp.wasPaused = true; - if(theApp.pauseWhenInactive) { - if(emulating) { - soundPause(); - } - theApp.active = a; - } - - memset(theApp.delta,255,sizeof(theApp.delta)); - } - - if(theApp.paused && emulating) - { - theApp.painting = true; - systemDrawScreen(); - theApp.painting = false; - theApp.renderedFrames--; - } -} - -#if _MSC_VER <= 1200 -void MainWnd::OnActivateApp(BOOL bActive, HTASK hTask) -#else -void MainWnd::OnActivateApp(BOOL bActive, DWORD hTask) -#endif -{ - CWnd::OnActivateApp(bActive, hTask); - - if(theApp.tripleBuffering && theApp.videoOption > VIDEO_4X) { - if(bActive) { - if(theApp.display) - theApp.display->clear(); - } - } -} - -void MainWnd::OnDropFiles(HDROP hDropInfo) -{ - char szFile[1024]; - - if(DragQueryFile(hDropInfo, - 0, - szFile, - 1024)) { - theApp.szFile = szFile; - if(FileRun()) { - SetForegroundWindow(); - emulating = TRUE; - } else { - emulating = FALSE; - soundPause(); - } - } - DragFinish(hDropInfo); -} - -LRESULT MainWnd::OnMySysCommand(WPARAM wParam, LPARAM lParam) -{ - if(emulating && !theApp.paused) { - if((wParam&0xFFF0) == SC_SCREENSAVE || (wParam&0xFFF0) == SC_MONITORPOWER) - return 0; - } - return Default(); -} - -void MainWnd::OnSetFocus(CWnd * pOldWnd) -{ - theApp.dinputKeyFocus = true; -} - -void MainWnd::OnKillFocus(CWnd * pNewWnd) -{ - theApp.dinputKeyFocus = false; -} +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2005-2006 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +// MainWnd.cpp : implementation file +// + +#include "stdafx.h" +#include "VBA.h" +#include "MainWnd.h" + +#include +#include + +#include "FileDlg.h" +#include "Reg.h" +#include "WinResUtil.h" + +#include "../System.h" +#include "../AutoBuild.h" +#include "../cheatSearch.h" +#include "../GBA.h" +#include "../Globals.h" +#include "../Flash.h" +#include "../Globals.h" +#include "../gb/GB.h" +#include "../gb/gbCheats.h" +#include "../gb/gbGlobals.h" +#include "../RTC.h" +#include "../Sound.h" +#include "../Util.h" + +#ifdef _DEBUG +#define new DEBUG_NEW +#undef THIS_FILE +static char THIS_FILE[] = __FILE__; +#endif + +#define VBA_CONFIRM_MODE WM_APP + 100 + +extern void remoteCleanUp(); +extern int gbHardware; + +///////////////////////////////////////////////////////////////////////////// +// MainWnd + +MainWnd::MainWnd() +{ + m_hAccelTable = NULL; + arrow = LoadCursor(NULL, IDC_ARROW); +} + +MainWnd::~MainWnd() +{ +} + + +BEGIN_MESSAGE_MAP(MainWnd, CWnd) + //{{AFX_MSG_MAP(MainWnd) + ON_WM_CLOSE() + ON_COMMAND(ID_HELP_ABOUT, OnHelpAbout) + ON_COMMAND(ID_HELP_FAQ, OnHelpFaq) + ON_COMMAND(ID_FILE_OPEN, OnFileOpen) + ON_WM_INITMENUPOPUP() + ON_COMMAND(ID_FILE_PAUSE, OnFilePause) + ON_UPDATE_COMMAND_UI(ID_FILE_PAUSE, OnUpdateFilePause) + ON_COMMAND(ID_FILE_RESET, OnFileReset) + ON_UPDATE_COMMAND_UI(ID_FILE_RESET, OnUpdateFileReset) + ON_UPDATE_COMMAND_UI(ID_FILE_RECENT_FREEZE, OnUpdateFileRecentFreeze) + ON_COMMAND(ID_FILE_RECENT_RESET, OnFileRecentReset) + ON_COMMAND(ID_FILE_RECENT_FREEZE, OnFileRecentFreeze) + ON_COMMAND(ID_FILE_EXIT, OnFileExit) + ON_COMMAND(ID_FILE_CLOSE, OnFileClose) + ON_UPDATE_COMMAND_UI(ID_FILE_CLOSE, OnUpdateFileClose) + ON_COMMAND(ID_FILE_OPENGAMEBOY, OnFileOpengameboy) + ON_COMMAND(ID_FILE_LOAD, OnFileLoad) + ON_UPDATE_COMMAND_UI(ID_FILE_LOAD, OnUpdateFileLoad) + ON_COMMAND(ID_FILE_SAVE, OnFileSave) + ON_UPDATE_COMMAND_UI(ID_FILE_SAVE, OnUpdateFileSave) + ON_COMMAND(ID_FILE_IMPORT_BATTERYFILE, OnFileImportBatteryfile) + ON_UPDATE_COMMAND_UI(ID_FILE_IMPORT_BATTERYFILE, OnUpdateFileImportBatteryfile) + ON_COMMAND(ID_FILE_IMPORT_GAMESHARKCODEFILE, OnFileImportGamesharkcodefile) + ON_UPDATE_COMMAND_UI(ID_FILE_IMPORT_GAMESHARKCODEFILE, OnUpdateFileImportGamesharkcodefile) + ON_COMMAND(ID_FILE_IMPORT_GAMESHARKSNAPSHOT, OnFileImportGamesharksnapshot) + ON_UPDATE_COMMAND_UI(ID_FILE_IMPORT_GAMESHARKSNAPSHOT, OnUpdateFileImportGamesharksnapshot) + ON_COMMAND(ID_FILE_EXPORT_BATTERYFILE, OnFileExportBatteryfile) + ON_UPDATE_COMMAND_UI(ID_FILE_EXPORT_BATTERYFILE, OnUpdateFileExportBatteryfile) + ON_COMMAND(ID_FILE_EXPORT_GAMESHARKSNAPSHOT, OnFileExportGamesharksnapshot) + ON_UPDATE_COMMAND_UI(ID_FILE_EXPORT_GAMESHARKSNAPSHOT, OnUpdateFileExportGamesharksnapshot) + ON_COMMAND(ID_FILE_SCREENCAPTURE, OnFileScreencapture) + ON_UPDATE_COMMAND_UI(ID_FILE_SCREENCAPTURE, OnUpdateFileScreencapture) + ON_COMMAND(ID_FILE_ROMINFORMATION, OnFileRominformation) + ON_UPDATE_COMMAND_UI(ID_FILE_ROMINFORMATION, OnUpdateFileRominformation) + ON_COMMAND(ID_FILE_TOGGLEMENU, OnFileTogglemenu) + ON_UPDATE_COMMAND_UI(ID_FILE_TOGGLEMENU, OnUpdateFileTogglemenu) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_FRAMESKIP_THROTTLE_NOTHROTTLE, OnUpdateOptionsFrameskipThrottleNothrottle) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_FRAMESKIP_THROTTLE_25, OnUpdateOptionsFrameskipThrottle25) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_FRAMESKIP_THROTTLE_50, OnUpdateOptionsFrameskipThrottle50) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_FRAMESKIP_THROTTLE_100, OnUpdateOptionsFrameskipThrottle100) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_FRAMESKIP_THROTTLE_150, OnUpdateOptionsFrameskipThrottle150) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_FRAMESKIP_THROTTLE_200, OnUpdateOptionsFrameskipThrottle200) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_FRAMESKIP_THROTTLE_OTHER, OnUpdateOptionsFrameskipThrottleOther) + ON_COMMAND(ID_OPTIONS_FRAMESKIP_THROTTLE_NOTHROTTLE, OnOptionsFrameskipThrottleNothrottle) + ON_COMMAND(ID_OPTIONS_FRAMESKIP_THROTTLE_25, OnOptionsFrameskipThrottle25) + ON_COMMAND(ID_OPTIONS_FRAMESKIP_THROTTLE_50, OnOptionsFrameskipThrottle50) + ON_COMMAND(ID_OPTIONS_FRAMESKIP_THROTTLE_100, OnOptionsFrameskipThrottle100) + ON_COMMAND(ID_OPTIONS_FRAMESKIP_THROTTLE_150, OnOptionsFrameskipThrottle150) + ON_COMMAND(ID_OPTIONS_FRAMESKIP_THROTTLE_200, OnOptionsFrameskipThrottle200) + ON_COMMAND(ID_OPTIONS_FRAMESKIP_THROTTLE_OTHER, OnOptionsFrameskipThrottleOther) + ON_COMMAND(ID_OPTIONS_FRAMESKIP_AUTOMATIC, OnOptionsFrameskipAutomatic) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_FRAMESKIP_AUTOMATIC, OnUpdateOptionsFrameskipAutomatic) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FRAMESKIP_0, OnUpdateOptionsVideoFrameskip0) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FRAMESKIP_1, OnUpdateOptionsVideoFrameskip1) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FRAMESKIP_2, OnUpdateOptionsVideoFrameskip2) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FRAMESKIP_3, OnUpdateOptionsVideoFrameskip3) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FRAMESKIP_4, OnUpdateOptionsVideoFrameskip4) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FRAMESKIP_5, OnUpdateOptionsVideoFrameskip5) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FRAMESKIP_6, OnUpdateOptionsVideoFrameskip6) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FRAMESKIP_7, OnUpdateOptionsVideoFrameskip7) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FRAMESKIP_8, OnUpdateOptionsVideoFrameskip8) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FRAMESKIP_9, OnUpdateOptionsVideoFrameskip9) + ON_COMMAND(ID_OPTIONS_VIDEO_VSYNC, OnOptionsVideoVsync) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_VSYNC, OnUpdateOptionsVideoVsync) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_X1, OnUpdateOptionsVideoX1) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_X2, OnUpdateOptionsVideoX2) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_X3, OnUpdateOptionsVideoX3) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_X4, OnUpdateOptionsVideoX4) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FULLSCREEN320X240, OnUpdateOptionsVideoFullscreen320x240) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FULLSCREEN640X480, OnUpdateOptionsVideoFullscreen640x480) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FULLSCREEN800X600, OnUpdateOptionsVideoFullscreen800x600) + ON_COMMAND(ID_OPTIONS_VIDEO_FULLSCREEN320X240, OnOptionsVideoFullscreen320x240) + ON_COMMAND(ID_OPTIONS_VIDEO_FULLSCREEN640X480, OnOptionsVideoFullscreen640x480) + ON_COMMAND(ID_OPTIONS_VIDEO_FULLSCREEN800X600, OnOptionsVideoFullscreen800x600) + ON_COMMAND(ID_OPTIONS_VIDEO_FULLSCREEN, OnOptionsVideoFullscreen) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FULLSCREEN, OnUpdateOptionsVideoFullscreen) + ON_WM_MOVE() + ON_WM_SIZE() + ON_COMMAND(ID_OPTIONS_VIDEO_DISABLESFX, OnOptionsVideoDisablesfx) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_DISABLESFX, OnUpdateOptionsVideoDisablesfx) + ON_COMMAND(ID_OPTIONS_VIDEO_FULLSCREENSTRETCHTOFIT, OnOptionsVideoFullscreenstretchtofit) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FULLSCREENSTRETCHTOFIT, OnUpdateOptionsVideoFullscreenstretchtofit) + ON_COMMAND(ID_OPTIONS_VIDEO_RENDERMETHOD_GDI, OnOptionsVideoRendermethodGdi) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_RENDERMETHOD_GDI, OnUpdateOptionsVideoRendermethodGdi) + ON_COMMAND(ID_OPTIONS_VIDEO_RENDERMETHOD_DIRECTDRAW, OnOptionsVideoRendermethodDirectdraw) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_RENDERMETHOD_DIRECTDRAW, OnUpdateOptionsVideoRendermethodDirectdraw) + ON_COMMAND(ID_OPTIONS_VIDEO_RENDERMETHOD_DIRECT3D, OnOptionsVideoRendermethodDirect3d) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_RENDERMETHOD_DIRECT3D, OnUpdateOptionsVideoRendermethodDirect3d) + ON_COMMAND(ID_OPTIONS_VIDEO_RENDERMETHOD_OPENGL, OnOptionsVideoRendermethodOpengl) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_RENDERMETHOD_OPENGL, OnUpdateOptionsVideoRendermethodOpengl) + ON_COMMAND(ID_OPTIONS_VIDEO_TRIPLEBUFFERING, OnOptionsVideoTriplebuffering) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_TRIPLEBUFFERING, OnUpdateOptionsVideoTriplebuffering) + ON_COMMAND(ID_OPTIONS_VIDEO_DDRAWEMULATIONONLY, OnOptionsVideoDdrawemulationonly) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_DDRAWEMULATIONONLY, OnUpdateOptionsVideoDdrawemulationonly) + ON_COMMAND(ID_OPTIONS_VIDEO_DDRAWUSEVIDEOMEMORY, OnOptionsVideoDdrawusevideomemory) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_DDRAWUSEVIDEOMEMORY, OnUpdateOptionsVideoDdrawusevideomemory) + ON_COMMAND(ID_OPTIONS_VIDEO_RENDEROPTIONS_D3DNOFILTER, OnOptionsVideoRenderoptionsD3dnofilter) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_RENDEROPTIONS_D3DNOFILTER, OnUpdateOptionsVideoRenderoptionsD3dnofilter) + ON_COMMAND(ID_OPTIONS_VIDEO_RENDEROPTIONS_D3DBILINEAR, OnOptionsVideoRenderoptionsD3dbilinear) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_RENDEROPTIONS_D3DBILINEAR, OnUpdateOptionsVideoRenderoptionsD3dbilinear) + ON_COMMAND(ID_OPTIONS_VIDEO_RENDEROPTIONS_GLNEAREST, OnOptionsVideoRenderoptionsGlnearest) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_RENDEROPTIONS_GLNEAREST, OnUpdateOptionsVideoRenderoptionsGlnearest) + ON_COMMAND(ID_OPTIONS_VIDEO_RENDEROPTIONS_GLBILINEAR, OnOptionsVideoRenderoptionsGlbilinear) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_RENDEROPTIONS_GLBILINEAR, OnUpdateOptionsVideoRenderoptionsGlbilinear) + ON_COMMAND(ID_OPTIONS_VIDEO_RENDEROPTIONS_GLTRIANGLE, OnOptionsVideoRenderoptionsGltriangle) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_RENDEROPTIONS_GLTRIANGLE, OnUpdateOptionsVideoRenderoptionsGltriangle) + ON_COMMAND(ID_OPTIONS_VIDEO_RENDEROPTIONS_GLQUADS, OnOptionsVideoRenderoptionsGlquads) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_RENDEROPTIONS_GLQUADS, OnUpdateOptionsVideoRenderoptionsGlquads) + ON_COMMAND(ID_OPTIONS_VIDEO_RENDEROPTIONS_SELECTSKIN, OnOptionsVideoRenderoptionsSelectskin) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_RENDEROPTIONS_SELECTSKIN, OnUpdateOptionsVideoRenderoptionsSelectskin) + ON_COMMAND(ID_OPTIONS_VIDEO_RENDEROPTIONS_SKIN, OnOptionsVideoRenderoptionsSkin) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_RENDEROPTIONS_SKIN, OnUpdateOptionsVideoRenderoptionsSkin) + ON_WM_CONTEXTMENU() + ON_COMMAND(ID_OPTIONS_EMULATOR_ASSOCIATE, OnOptionsEmulatorAssociate) + ON_COMMAND(ID_OPTIONS_EMULATOR_DIRECTORIES, OnOptionsEmulatorDirectories) + ON_COMMAND(ID_OPTIONS_EMULATOR_DISABLESTATUSMESSAGES, OnOptionsEmulatorDisablestatusmessages) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_DISABLESTATUSMESSAGES, OnUpdateOptionsEmulatorDisablestatusmessages) + ON_COMMAND(ID_OPTIONS_EMULATOR_SYNCHRONIZE, OnOptionsEmulatorSynchronize) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_SYNCHRONIZE, OnUpdateOptionsEmulatorSynchronize) + ON_COMMAND(ID_OPTIONS_EMULATOR_PAUSEWHENINACTIVE, OnOptionsEmulatorPausewheninactive) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_PAUSEWHENINACTIVE, OnUpdateOptionsEmulatorPausewheninactive) + ON_COMMAND(ID_OPTIONS_EMULATOR_SPEEDUPTOGGLE, OnOptionsEmulatorSpeeduptoggle) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_SPEEDUPTOGGLE, OnUpdateOptionsEmulatorSpeeduptoggle) + ON_COMMAND(ID_OPTIONS_EMULATOR_AUTOMATICALLYIPSPATCH, OnOptionsEmulatorAutomaticallyipspatch) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_AUTOMATICALLYIPSPATCH, OnUpdateOptionsEmulatorAutomaticallyipspatch) + ON_COMMAND(ID_OPTIONS_EMULATOR_AGBPRINT, OnOptionsEmulatorAgbprint) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_AGBPRINT, OnUpdateOptionsEmulatorAgbprint) + ON_COMMAND(ID_OPTIONS_EMULATOR_REALTIMECLOCK, OnOptionsEmulatorRealtimeclock) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_REALTIMECLOCK, OnUpdateOptionsEmulatorRealtimeclock) + ON_COMMAND(ID_OPTIONS_EMULATOR_GENERICFLASHCARD, OnOptionsEmulatorGenericflashcard) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_GENERICFLASHCARD, OnUpdateOptionsEmulatorGenericflashcard) + ON_COMMAND(ID_OPTIONS_EMULATOR_AUTOHIDEMENU, OnOptionsEmulatorAutohidemenu) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_AUTOHIDEMENU, OnUpdateOptionsEmulatorAutohidemenu) + ON_COMMAND(ID_OPTIONS_EMULATOR_REWINDINTERVAL, OnOptionsEmulatorRewindinterval) + ON_COMMAND(ID_OPTIONS_EMULATOR_SAVETYPE_AUTOMATIC, OnOptionsEmulatorSavetypeAutomatic) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_SAVETYPE_AUTOMATIC, OnUpdateOptionsEmulatorSavetypeAutomatic) + ON_COMMAND(ID_OPTIONS_EMULATOR_SAVETYPE_EEPROM, OnOptionsEmulatorSavetypeEeprom) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_SAVETYPE_EEPROM, OnUpdateOptionsEmulatorSavetypeEeprom) + ON_COMMAND(ID_OPTIONS_EMULATOR_SAVETYPE_SRAM, OnOptionsEmulatorSavetypeSram) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_SAVETYPE_SRAM, OnUpdateOptionsEmulatorSavetypeSram) + ON_COMMAND(ID_OPTIONS_EMULATOR_SAVETYPE_FLASH, OnOptionsEmulatorSavetypeFlash) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_SAVETYPE_FLASH, OnUpdateOptionsEmulatorSavetypeFlash) + ON_COMMAND(ID_OPTIONS_EMULATOR_SAVETYPE_EEPROMSENSOR, OnOptionsEmulatorSavetypeEepromsensor) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_SAVETYPE_EEPROMSENSOR, OnUpdateOptionsEmulatorSavetypeEepromsensor) + ON_COMMAND(ID_OPTIONS_EMULATOR_SAVETYPE_NONE, OnOptionsEmulatorSavetypeNone) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_SAVETYPE_NONE, OnUpdateOptionsEmulatorSavetypeNone) + ON_COMMAND(ID_OPTIONS_EMULATOR_SAVETYPE_FLASH512K, OnOptionsEmulatorSavetypeFlash512k) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_SAVETYPE_FLASH512K, OnUpdateOptionsEmulatorSavetypeFlash512k) + ON_COMMAND(ID_OPTIONS_EMULATOR_SAVETYPE_FLASH1M, OnOptionsEmulatorSavetypeFlash1m) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_SAVETYPE_FLASH1M, OnUpdateOptionsEmulatorSavetypeFlash1m) + ON_COMMAND(ID_OPTIONS_EMULATOR_USEBIOSFILE, OnOptionsEmulatorUsebiosfile) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_USEBIOSFILE, OnUpdateOptionsEmulatorUsebiosfile) + ON_COMMAND(ID_OPTIONS_EMULATOR_SKIPBIOS, OnOptionsEmulatorSkipbios) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_SKIPBIOS, OnUpdateOptionsEmulatorSkipbios) + ON_COMMAND(ID_OPTIONS_EMULATOR_SELECTBIOSFILE, OnOptionsEmulatorSelectbiosfile) + ON_COMMAND(ID_OPTIONS_EMULATOR_PNGFORMAT, OnOptionsEmulatorPngformat) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_PNGFORMAT, OnUpdateOptionsEmulatorPngformat) + ON_COMMAND(ID_OPTIONS_EMULATOR_BMPFORMAT, OnOptionsEmulatorBmpformat) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_BMPFORMAT, OnUpdateOptionsEmulatorBmpformat) + ON_COMMAND(ID_OPTIONS_SOUND_OFF, OnOptionsSoundOff) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_OFF, OnUpdateOptionsSoundOff) + ON_COMMAND(ID_OPTIONS_SOUND_MUTE, OnOptionsSoundMute) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_MUTE, OnUpdateOptionsSoundMute) + ON_COMMAND(ID_OPTIONS_SOUND_ON, OnOptionsSoundOn) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_ON, OnUpdateOptionsSoundOn) + ON_COMMAND(ID_OPTIONS_SOUND_USEOLDSYNCHRONIZATION, OnOptionsSoundUseoldsynchronization) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_USEOLDSYNCHRONIZATION, OnUpdateOptionsSoundUseoldsynchronization) + ON_COMMAND(ID_OPTIONS_SOUND_ECHO, OnOptionsSoundEcho) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_ECHO, OnUpdateOptionsSoundEcho) + ON_COMMAND(ID_OPTIONS_SOUND_LOWPASSFILTER, OnOptionsSoundLowpassfilter) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_LOWPASSFILTER, OnUpdateOptionsSoundLowpassfilter) + ON_COMMAND(ID_OPTIONS_SOUND_REVERSESTEREO, OnOptionsSoundReversestereo) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_REVERSESTEREO, OnUpdateOptionsSoundReversestereo) + ON_COMMAND(ID_OPTIONS_SOUND_11KHZ, OnOptionsSound11khz) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_11KHZ, OnUpdateOptionsSound11khz) + ON_COMMAND(ID_OPTIONS_SOUND_22KHZ, OnOptionsSound22khz) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_22KHZ, OnUpdateOptionsSound22khz) + ON_COMMAND(ID_OPTIONS_SOUND_44KHZ, OnOptionsSound44khz) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_44KHZ, OnUpdateOptionsSound44khz) + ON_COMMAND(ID_OPTIONS_SOUND_CHANNEL1, OnOptionsSoundChannel1) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_CHANNEL1, OnUpdateOptionsSoundChannel1) + ON_COMMAND(ID_OPTIONS_SOUND_CHANNEL2, OnOptionsSoundChannel2) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_CHANNEL2, OnUpdateOptionsSoundChannel2) + ON_COMMAND(ID_OPTIONS_SOUND_CHANNEL3, OnOptionsSoundChannel3) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_CHANNEL3, OnUpdateOptionsSoundChannel3) + ON_COMMAND(ID_OPTIONS_SOUND_CHANNEL4, OnOptionsSoundChannel4) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_CHANNEL4, OnUpdateOptionsSoundChannel4) + ON_COMMAND(ID_OPTIONS_SOUND_DIRECTSOUNDA, OnOptionsSoundDirectsounda) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_DIRECTSOUNDA, OnUpdateOptionsSoundDirectsounda) + ON_COMMAND(ID_OPTIONS_SOUND_DIRECTSOUNDB, OnOptionsSoundDirectsoundb) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_DIRECTSOUNDB, OnUpdateOptionsSoundDirectsoundb) + ON_COMMAND(ID_OPTIONS_GAMEBOY_BORDER, OnOptionsGameboyBorder) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_GAMEBOY_BORDER, OnUpdateOptionsGameboyBorder) + ON_COMMAND(ID_OPTIONS_GAMEBOY_PRINTER, OnOptionsGameboyPrinter) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_GAMEBOY_PRINTER, OnUpdateOptionsGameboyPrinter) + ON_COMMAND(ID_OPTIONS_GAMEBOY_BORDERAUTOMATIC, OnOptionsGameboyBorderAutomatic) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_GAMEBOY_BORDERAUTOMATIC, OnUpdateOptionsGameboyBorderAutomatic) + ON_COMMAND(ID_OPTIONS_GAMEBOY_AUTOMATIC, OnOptionsGameboyAutomatic) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_GAMEBOY_AUTOMATIC, OnUpdateOptionsGameboyAutomatic) + ON_COMMAND(ID_OPTIONS_GAMEBOY_GBA, OnOptionsGameboyGba) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_GAMEBOY_GBA, OnUpdateOptionsGameboyGba) + ON_COMMAND(ID_OPTIONS_GAMEBOY_CGB, OnOptionsGameboyCgb) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_GAMEBOY_CGB, OnUpdateOptionsGameboyCgb) + ON_COMMAND(ID_OPTIONS_GAMEBOY_SGB, OnOptionsGameboySgb) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_GAMEBOY_SGB, OnUpdateOptionsGameboySgb) + ON_COMMAND(ID_OPTIONS_GAMEBOY_SGB2, OnOptionsGameboySgb2) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_GAMEBOY_SGB2, OnUpdateOptionsGameboySgb2) + ON_COMMAND(ID_OPTIONS_GAMEBOY_GB, OnOptionsGameboyGb) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_GAMEBOY_GB, OnUpdateOptionsGameboyGb) + ON_COMMAND(ID_OPTIONS_GAMEBOY_REALCOLORS, OnOptionsGameboyRealcolors) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_GAMEBOY_REALCOLORS, OnUpdateOptionsGameboyRealcolors) + ON_COMMAND(ID_OPTIONS_GAMEBOY_GAMEBOYCOLORS, OnOptionsGameboyGameboycolors) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_GAMEBOY_GAMEBOYCOLORS, OnUpdateOptionsGameboyGameboycolors) + ON_COMMAND(ID_OPTIONS_GAMEBOY_COLORS, OnOptionsGameboyColors) + ON_COMMAND(ID_OPTIONS_FILTER_DISABLEMMX, OnOptionsFilterDisablemmx) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_FILTER_DISABLEMMX, OnUpdateOptionsFilterDisablemmx) + ON_COMMAND(ID_OPTIONS_LANGUAGE_SYSTEM, OnOptionsLanguageSystem) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_LANGUAGE_SYSTEM, OnUpdateOptionsLanguageSystem) + ON_COMMAND(ID_OPTIONS_LANGUAGE_ENGLISH, OnOptionsLanguageEnglish) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_LANGUAGE_ENGLISH, OnUpdateOptionsLanguageEnglish) + ON_COMMAND(ID_OPTIONS_LANGUAGE_OTHER, OnOptionsLanguageOther) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_LANGUAGE_OTHER, OnUpdateOptionsLanguageOther) + ON_COMMAND(ID_OPTIONS_JOYPAD_CONFIGURE_1, OnOptionsJoypadConfigure1) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_JOYPAD_CONFIGURE_1, OnUpdateOptionsJoypadConfigure1) + ON_COMMAND(ID_OPTIONS_JOYPAD_CONFIGURE_2, OnOptionsJoypadConfigure2) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_JOYPAD_CONFIGURE_2, OnUpdateOptionsJoypadConfigure2) + ON_COMMAND(ID_OPTIONS_JOYPAD_CONFIGURE_3, OnOptionsJoypadConfigure3) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_JOYPAD_CONFIGURE_3, OnUpdateOptionsJoypadConfigure3) + ON_COMMAND(ID_OPTIONS_JOYPAD_CONFIGURE_4, OnOptionsJoypadConfigure4) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_JOYPAD_CONFIGURE_4, OnUpdateOptionsJoypadConfigure4) + ON_COMMAND(ID_OPTIONS_JOYPAD_MOTIONCONFIGURE, OnOptionsJoypadMotionconfigure) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_JOYPAD_MOTIONCONFIGURE, OnUpdateOptionsJoypadMotionconfigure) + ON_COMMAND(ID_CHEATS_SEARCHFORCHEATS, OnCheatsSearchforcheats) + ON_UPDATE_COMMAND_UI(ID_CHEATS_SEARCHFORCHEATS, OnUpdateCheatsSearchforcheats) + ON_COMMAND(ID_CHEATS_CHEATLIST, OnCheatsCheatlist) + ON_UPDATE_COMMAND_UI(ID_CHEATS_CHEATLIST, OnUpdateCheatsCheatlist) + ON_COMMAND(ID_CHEATS_AUTOMATICSAVELOADCHEATS, OnCheatsAutomaticsaveloadcheats) + ON_COMMAND(ID_CHEATS_LOADCHEATLIST, OnCheatsLoadcheatlist) + ON_UPDATE_COMMAND_UI(ID_CHEATS_LOADCHEATLIST, OnUpdateCheatsLoadcheatlist) + ON_COMMAND(ID_CHEATS_SAVECHEATLIST, OnCheatsSavecheatlist) + ON_UPDATE_COMMAND_UI(ID_CHEATS_SAVECHEATLIST, OnUpdateCheatsSavecheatlist) + ON_COMMAND(ID_TOOLS_DISASSEMBLE, OnToolsDisassemble) + ON_UPDATE_COMMAND_UI(ID_TOOLS_DISASSEMBLE, OnUpdateToolsDisassemble) + ON_COMMAND(ID_TOOLS_LOGGING, OnToolsLogging) + ON_UPDATE_COMMAND_UI(ID_TOOLS_LOGGING, OnUpdateToolsLogging) + ON_COMMAND(ID_TOOLS_IOVIEWER, OnToolsIoviewer) + ON_UPDATE_COMMAND_UI(ID_TOOLS_IOVIEWER, OnUpdateToolsIoviewer) + ON_COMMAND(ID_TOOLS_MAPVIEW, OnToolsMapview) + ON_UPDATE_COMMAND_UI(ID_TOOLS_MAPVIEW, OnUpdateToolsMapview) + ON_COMMAND(ID_TOOLS_MEMORYVIEWER, OnToolsMemoryviewer) + ON_UPDATE_COMMAND_UI(ID_TOOLS_MEMORYVIEWER, OnUpdateToolsMemoryviewer) + ON_COMMAND(ID_TOOLS_OAMVIEWER, OnToolsOamviewer) + ON_UPDATE_COMMAND_UI(ID_TOOLS_OAMVIEWER, OnUpdateToolsOamviewer) + ON_COMMAND(ID_TOOLS_PALETTEVIEW, OnToolsPaletteview) + ON_UPDATE_COMMAND_UI(ID_TOOLS_PALETTEVIEW, OnUpdateToolsPaletteview) + ON_COMMAND(ID_TOOLS_TILEVIEWER, OnToolsTileviewer) + ON_UPDATE_COMMAND_UI(ID_TOOLS_TILEVIEWER, OnUpdateToolsTileviewer) + ON_COMMAND(ID_DEBUG_NEXTFRAME, OnDebugNextframe) + ON_UPDATE_COMMAND_UI(ID_CHEATS_AUTOMATICSAVELOADCHEATS, OnUpdateCheatsAutomaticsaveloadcheats) + ON_COMMAND(ID_TOOLS_DEBUG_GDB, OnToolsDebugGdb) + ON_UPDATE_COMMAND_UI(ID_TOOLS_DEBUG_GDB, OnUpdateToolsDebugGdb) + ON_COMMAND(ID_TOOLS_DEBUG_LOADANDWAIT, OnToolsDebugLoadandwait) + ON_UPDATE_COMMAND_UI(ID_TOOLS_DEBUG_LOADANDWAIT, OnUpdateToolsDebugLoadandwait) + ON_COMMAND(ID_TOOLS_DEBUG_BREAK, OnToolsDebugBreak) + ON_UPDATE_COMMAND_UI(ID_TOOLS_DEBUG_BREAK, OnUpdateToolsDebugBreak) + ON_COMMAND(ID_TOOLS_DEBUG_DISCONNECT, OnToolsDebugDisconnect) + ON_UPDATE_COMMAND_UI(ID_TOOLS_DEBUG_DISCONNECT, OnUpdateToolsDebugDisconnect) + ON_COMMAND(ID_OPTIONS_SOUND_STARTRECORDING, OnOptionsSoundStartrecording) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_STARTRECORDING, OnUpdateOptionsSoundStartrecording) + ON_COMMAND(ID_OPTIONS_SOUND_STOPRECORDING, OnOptionsSoundStoprecording) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_STOPRECORDING, OnUpdateOptionsSoundStoprecording) + ON_COMMAND(ID_TOOLS_RECORD_STARTAVIRECORDING, OnToolsRecordStartavirecording) + ON_UPDATE_COMMAND_UI(ID_TOOLS_RECORD_STARTAVIRECORDING, OnUpdateToolsRecordStartavirecording) + ON_COMMAND(ID_TOOLS_RECORD_STOPAVIRECORDING, OnToolsRecordStopavirecording) + ON_UPDATE_COMMAND_UI(ID_TOOLS_RECORD_STOPAVIRECORDING, OnUpdateToolsRecordStopavirecording) + ON_WM_PAINT() + ON_COMMAND(ID_TOOLS_RECORD_STARTMOVIERECORDING, OnToolsRecordStartmovierecording) + ON_UPDATE_COMMAND_UI(ID_TOOLS_RECORD_STARTMOVIERECORDING, OnUpdateToolsRecordStartmovierecording) + ON_COMMAND(ID_TOOLS_RECORD_STOPMOVIERECORDING, OnToolsRecordStopmovierecording) + ON_UPDATE_COMMAND_UI(ID_TOOLS_RECORD_STOPMOVIERECORDING, OnUpdateToolsRecordStopmovierecording) + ON_COMMAND(ID_TOOLS_PLAY_STARTMOVIEPLAYING, OnToolsPlayStartmovieplaying) + ON_UPDATE_COMMAND_UI(ID_TOOLS_PLAY_STARTMOVIEPLAYING, OnUpdateToolsPlayStartmovieplaying) + ON_COMMAND(ID_TOOLS_PLAY_STOPMOVIEPLAYING, OnToolsPlayStopmovieplaying) + ON_UPDATE_COMMAND_UI(ID_TOOLS_PLAY_STOPMOVIEPLAYING, OnUpdateToolsPlayStopmovieplaying) + ON_COMMAND(ID_TOOLS_REWIND, OnToolsRewind) + ON_UPDATE_COMMAND_UI(ID_TOOLS_REWIND, OnUpdateToolsRewind) + ON_COMMAND(ID_TOOLS_CUSTOMIZE, OnToolsCustomize) + ON_UPDATE_COMMAND_UI(ID_TOOLS_CUSTOMIZE, OnUpdateToolsCustomize) + ON_COMMAND(ID_HELP_BUGREPORT, OnHelpBugreport) + ON_WM_MOUSEMOVE() + ON_WM_INITMENU() + ON_WM_ACTIVATE() + ON_WM_ACTIVATEAPP() + ON_WM_DROPFILES() + ON_COMMAND(ID_FILE_SAVEGAME_OLDESTSLOT, OnFileSavegameOldestslot) + ON_UPDATE_COMMAND_UI(ID_FILE_SAVEGAME_OLDESTSLOT, OnUpdateFileSavegameOldestslot) + ON_COMMAND(ID_FILE_LOADGAME_MOSTRECENT, OnFileLoadgameMostrecent) + ON_UPDATE_COMMAND_UI(ID_FILE_LOADGAME_MOSTRECENT, OnUpdateFileLoadgameMostrecent) + ON_COMMAND(ID_FILE_LOADGAME_AUTOLOADMOSTRECENT, OnFileLoadgameAutoloadmostrecent) + ON_UPDATE_COMMAND_UI(ID_FILE_LOADGAME_AUTOLOADMOSTRECENT, OnUpdateFileLoadgameAutoloadmostrecent) + ON_COMMAND(ID_OPTIONS_SOUND_VOLUME_25X, OnOptionsSoundVolume25x) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_VOLUME_25X, OnUpdateOptionsSoundVolume25x) + ON_COMMAND(ID_OPTIONS_SOUND_VOLUME_5X, OnOptionsSoundVolume5x) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_VOLUME_5X, OnUpdateOptionsSoundVolume5x) + ON_COMMAND(ID_CHEATS_DISABLECHEATS, OnCheatsDisablecheats) + ON_UPDATE_COMMAND_UI(ID_CHEATS_DISABLECHEATS, OnUpdateCheatsDisablecheats) + ON_COMMAND(ID_OPTIONS_VIDEO_FULLSCREENMAXSCALE, OnOptionsVideoFullscreenmaxscale) + ON_COMMAND(ID_OPTIONS_EMULATOR_GAMEOVERRIDES, OnOptionsEmulatorGameoverrides) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_EMULATOR_GAMEOVERRIDES, OnUpdateOptionsEmulatorGameoverrides) + ON_COMMAND(ID_HELP_GNUPUBLICLICENSE, OnHelpGnupubliclicense) + ON_COMMAND(ID_OPTIONS_LINK_OPTIONS, OnLinkOptions) + ON_COMMAND(ID_OPTIONS_LINK_LOG, OnOptionsLinkLog) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_LINK_LOG, OnUpdateOptionsLinkLog) + ON_COMMAND(ID_OPTIONS_LINK_WIRELESSADAPTER, OnOptionsLinkRFU) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_LINK_WIRELESSADAPTER, OnUpdateOptionsLinkRFU) + //}}AFX_MSG_MAP + ON_COMMAND_EX_RANGE(ID_FILE_MRU_FILE1, ID_FILE_MRU_FILE10, OnFileRecentFile) + ON_COMMAND_EX_RANGE(ID_FILE_LOADGAME_SLOT1, ID_FILE_LOADGAME_SLOT10, OnFileLoadSlot) + ON_COMMAND_EX_RANGE(ID_FILE_SAVEGAME_SLOT1, ID_FILE_SAVEGAME_SLOT10, OnFileSaveSlot) + ON_UPDATE_COMMAND_UI_RANGE(ID_FILE_LOADGAME_SLOT1, ID_FILE_LOADGAME_SLOT10, OnUpdateFileLoadGameSlot) + ON_UPDATE_COMMAND_UI_RANGE(ID_FILE_SAVEGAME_SLOT1, ID_FILE_SAVEGAME_SLOT10, OnUpdateFileSaveGameSlot) + ON_COMMAND_EX_RANGE(ID_OPTIONS_VIDEO_FRAMESKIP_0, ID_OPTIONS_VIDEO_FRAMESKIP_5, OnOptionsFrameskip) + ON_COMMAND_EX_RANGE(ID_OPTIONS_VIDEO_FRAMESKIP_6, ID_OPTIONS_VIDEO_FRAMESKIP_9, OnOptionsFrameskip) + ON_COMMAND_EX_RANGE(ID_OPTIONS_VIDEO_X1, ID_OPTIONS_VIDEO_X4, OnOptionVideoSize) + ON_COMMAND_EX_RANGE(ID_OPTIONS_VIDEO_LAYERS_BG0, ID_OPTIONS_VIDEO_LAYERS_OBJWIN, OnVideoLayer) + ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_VIDEO_LAYERS_BG0, ID_OPTIONS_VIDEO_LAYERS_OBJWIN, OnUpdateVideoLayer) + ON_COMMAND(ID_SYSTEM_MINIMIZE, OnSystemMinimize) + ON_COMMAND_EX_RANGE(ID_OPTIONS_EMULATOR_SHOWSPEED_NONE, ID_OPTIONS_EMULATOR_SHOWSPEED_TRANSPARENT, OnOptionsEmulatorShowSpeed) + ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_EMULATOR_SHOWSPEED_NONE, ID_OPTIONS_EMULATOR_SHOWSPEED_TRANSPARENT, OnUpdateOptionsEmulatorShowSpeed) + ON_COMMAND_EX_RANGE(ID_OPTIONS_SOUND_VOLUME_1X, ID_OPTIONS_SOUND_VOLUME_4X, OnOptionsSoundVolume) + ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_SOUND_VOLUME_1X, ID_OPTIONS_SOUND_VOLUME_4X, OnUpdateOptionsSoundVolume) + ON_COMMAND_EX_RANGE(ID_OPTIONS_PRIORITY_HIGHEST, ID_OPTIONS_PRIORITY_BELOWNORMAL, OnOptionsPriority) + ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_PRIORITY_HIGHEST, ID_OPTIONS_PRIORITY_BELOWNORMAL, OnUpdateOptionsPriority) + ON_COMMAND_EX_RANGE(ID_OPTIONS_FILTER_NORMAL, ID_OPTIONS_FILTER_TVMODE, OnOptionsFilter) + ON_COMMAND_EX_RANGE(ID_OPTIONS_FILTER16BIT_PIXELATEEXPERIMENTAL, ID_OPTIONS_FILTER16BIT_MOTIONBLUREXPERIMENTAL, OnOptionsFilter) + ON_COMMAND_EX_RANGE(ID_OPTIONS_FILTER16BIT_ADVANCEMAMESCALE2X, ID_OPTIONS_FILTER16BIT_SIMPLE2X, OnOptionsFilter) + ON_COMMAND_EX_RANGE(ID_OPTIONS_FILTER_BILINEAR, ID_OPTIONS_FILTER_BILINEARPLUS, OnOptionsFilter) + ON_COMMAND_EX_RANGE(ID_OPTIONS_FILTER_SCANLINES, ID_OPTIONS_FILTER_SCANLINES, OnOptionsFilter) + ON_COMMAND_EX_RANGE(ID_OPTIONS_FILTER_HQ2X, ID_OPTIONS_FILTER_LQ2X, OnOptionsFilter) + ON_COMMAND_EX(ID_OPTIONS_FILTER_HQ3X, OnOptionsFilter) + ON_COMMAND_EX(ID_OPTIONS_FILTER_HQ4X, OnOptionsFilter) + ON_COMMAND_EX(ID_OPTIONS_FILTER_SIMPLE3X, OnOptionsFilter) + ON_COMMAND_EX(ID_OPTIONS_FILTER_SIMPLE4X, OnOptionsFilter) + ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_FILTER_NORMAL, ID_OPTIONS_FILTER_TVMODE, OnUpdateOptionsFilter) + ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_FILTER16BIT_PIXELATEEXPERIMENTAL, ID_OPTIONS_FILTER16BIT_MOTIONBLUREXPERIMENTAL, OnUpdateOptionsFilter) + ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_FILTER16BIT_ADVANCEMAMESCALE2X, ID_OPTIONS_FILTER16BIT_SIMPLE2X, OnUpdateOptionsFilter) + ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_FILTER_BILINEAR, ID_OPTIONS_FILTER_BILINEARPLUS, OnUpdateOptionsFilter) + ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_FILTER_SCANLINES, ID_OPTIONS_FILTER_SCANLINES, OnUpdateOptionsFilter) + ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_FILTER_HQ2X, ID_OPTIONS_FILTER_LQ2X, OnUpdateOptionsFilter) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_FILTER_SIMPLE3X, OnUpdateOptionsFilter) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_FILTER_SIMPLE4X, OnUpdateOptionsFilter) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_FILTER_HQ3X, OnUpdateOptionsFilter) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_FILTER_HQ4X, OnUpdateOptionsFilter) + ON_COMMAND_EX_RANGE(ID_OPTIONS_FILTER_INTERFRAMEBLENDING_NONE, ID_OPTIONS_FILTER_INTERFRAMEBLENDING_SMART, OnOptionsFilterIFB) + ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_FILTER_INTERFRAMEBLENDING_NONE, ID_OPTIONS_FILTER_INTERFRAMEBLENDING_SMART, OnUpdateOptionsFilterIFB) + ON_COMMAND_EX_RANGE(ID_OPTIONS_JOYPAD_DEFAULTJOYPAD_1, ID_OPTIONS_JOYPAD_DEFAULTJOYPAD_4, OnOptionsJoypadDefault) + ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_JOYPAD_DEFAULTJOYPAD_1, ID_OPTIONS_JOYPAD_DEFAULTJOYPAD_4, OnUpdateOptionsJoypadDefault) + ON_COMMAND_EX_RANGE(ID_OPTIONS_JOYPAD_AUTOFIRE_A, ID_OPTIONS_JOYPAD_AUTOFIRE_R, OnOptionsJoypadAutofire) + ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_JOYPAD_AUTOFIRE_A, ID_OPTIONS_JOYPAD_AUTOFIRE_R, OnUpdateOptionsJoypadAutofire) + ON_MESSAGE(VBA_CONFIRM_MODE, OnConfirmMode) + ON_MESSAGE(WM_SYSCOMMAND, OnMySysCommand) + ON_COMMAND(ID_OPTIONS_SOUND_HARDWAREACCELERATION, &MainWnd::OnOptionsSoundHardwareacceleration) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_SOUND_HARDWAREACCELERATION, &MainWnd::OnUpdateOptionsSoundHardwareacceleration) + ON_COMMAND(ID_OPTIONS_VIDEO_FULLSCREEN1280X1024, OnOptionsVideoFullscreen1280x1024) + ON_COMMAND(ID_OPTIONS_VIDEO_FULLSCREEN1024X768, OnOptionsVideoFullscreen1024x768) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FULLSCREEN1024X768, OnUpdateOptionsVideoFullscreen1024x768) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_VIDEO_FULLSCREEN1280X1024, OnUpdateOptionsVideoFullscreen1280x1024) + ON_COMMAND(ID_OPTIONS_FILTER_LCDCOLORS, OnOptionsFilterLcdcolors) + ON_UPDATE_COMMAND_UI(ID_OPTIONS_FILTER_LCDCOLORS, OnUpdateOptionsFilterLcdcolors) + ON_COMMAND_EX_RANGE(ID_OPTIONS_SOUND_PCMINTERPOLATION_NONE, ID_OPTIONS_SOUND_PCMINTERPOLATION_LIBRESAMPLE, OnOptionsSoundPcminterpolation) + ON_UPDATE_COMMAND_UI_RANGE(ID_OPTIONS_SOUND_PCMINTERPOLATION_NONE, ID_OPTIONS_SOUND_PCMINTERPOLATION_LIBRESAMPLE, OnUpdateOptionsSoundPcminterpolation) + ON_WM_SETFOCUS() + ON_WM_KILLFOCUS() + END_MESSAGE_MAP() + + + ///////////////////////////////////////////////////////////////////////////// +// MainWnd message handlers + +void MainWnd::OnClose() +{ + emulating = false; + CWnd::OnClose(); + + delete this; +} + +bool MainWnd::FileRun() +{ + // save battery file before we change the filename... + if(rom != NULL || gbRom != NULL) { + if(theApp.autoSaveLoadCheatList) + winSaveCheatListDefault(); + writeBatteryFile(); + cheatSearchCleanup(&cheatSearchData); + theApp.emulator.emuCleanUp(); + remoteCleanUp(); + emulating = false; +#ifdef APU_LOGGER_H + end_apu_log(); +#endif + } + char tempName[2048]; + char file[2048]; + + utilGetBaseName(theApp.szFile, tempName); + + _fullpath(file, tempName, 1024); + theApp.filename = file; + + int index = theApp.filename.ReverseFind('.'); + if(index != -1) + theApp.filename = theApp.filename.Left(index); + + CString ipsname; + ipsname.Format("%s.ips", theApp.filename); + + if(!theApp.dir.GetLength()) { + int index = theApp.filename.ReverseFind('\\'); + if(index != -1) { + theApp.dir = theApp.filename.Left(index-1); + } + } + + IMAGE_TYPE type = utilFindType(theApp.szFile); + + if(type == IMAGE_UNKNOWN) { + systemMessage(IDS_UNSUPPORTED_FILE_TYPE, + "Unsupported file type: %s", theApp.szFile); + return false; + } + systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; + theApp.cartridgeType = type; + if(type == IMAGE_GB) { + genericflashcardEnable = theApp.winGenericflashcardEnable; + + + if(!gbLoadRom(theApp.szFile)) + return false; + + gbGetHardwareType(); + + // used for the handling of the gb Boot Rom + if (gbHardware & 5) + { + char tempName[2048]; + GetModuleFileName(NULL, tempName, 2048); + + char *p = strrchr(tempName, '\\'); + if(p) + *p = 0; + + strcat(tempName, "\\DMG_ROM.bin"); + + skipBios = theApp.skipBiosFile ? true : false; + gbCPUInit(tempName, theApp.useBiosFile ? true : false); + } + + + + gbReset(); + theApp.emulator = GBSystem; + gbBorderOn = theApp.winGbBorderOn; + theApp.romSize = gbRomSize; + + + if(theApp.autoIPS) { + int size = gbRomSize; + utilApplyIPS(ipsname, &gbRom, &size); + if(size != gbRomSize) { + extern bool gbUpdateSizes(); + gbUpdateSizes(); + gbReset(); + theApp.romSize = size; + } + } + } else { + int size = CPULoadRom(theApp.szFile); + if(!size) + return false; + + theApp.romSize = size; + + flashSetSize(theApp.winFlashSize); + rtcEnable(theApp.winRtcEnable); + cpuSaveType = theApp.winSaveType; + + GetModuleFileName(NULL, tempName, 2048); + + char *p = strrchr(tempName, '\\'); + if(p) + *p = 0; + + char buffer[5]; + strncpy(buffer, (const char *)&rom[0xac], 4); + buffer[4] = 0; + + strcat(tempName, "\\vba-over.ini"); + + UINT i = GetPrivateProfileInt(buffer, + "rtcEnabled", + -1, + tempName); + if(i != (UINT)-1) + rtcEnable(i == 0 ? false : true); + + i = GetPrivateProfileInt(buffer, + "flashSize", + -1, + tempName); + if(i != (UINT)-1 && (i == 0x10000 || i == 0x20000)) + flashSetSize((int)i); + + i = GetPrivateProfileInt(buffer, + "saveType", + -1, + tempName); + if(i != (UINT)-1 && (i <= 5)) + cpuSaveType = (int)i; + i = GetPrivateProfileInt(buffer, + "mirroringEnabled", + -1, + tempName); + if(i != (UINT)-1) + doMirroring (i == 0 ? false : true); + + theApp.emulator = GBASystem; + /* disabled due to problems + if(theApp.removeIntros && rom != NULL) { + *((u32 *)rom)= 0xea00002e; + } + */ + + if(theApp.autoIPS) { + int size = 0x2000000; + utilApplyIPS(ipsname, &rom, &size); + if(size != 0x2000000) { + CPUReset(); + } + } + } + + if(theApp.soundInitialized) { + if(theApp.cartridgeType == 1) + gbSoundReset(); + else + soundReset(); + } else { + if(!soundOffFlag) + soundInit(!(theApp.cartridgeType==1)); + theApp.soundInitialized = true; + } + +#ifdef APU_LOGGER_H + begin_apu_log("apu_log.txt"); +#endif + + if(type == IMAGE_GBA) { + skipBios = theApp.skipBiosFile ? true : false; + CPUInit((char *)(LPCTSTR)theApp.biosFileName, theApp.useBiosFile ? true : false); + CPUReset(); + } + + readBatteryFile(); + + if(theApp.autoSaveLoadCheatList) + winLoadCheatListDefault(); + + theApp.addRecentFile(theApp.szFile); + + theApp.updateWindowSize(theApp.videoOption); + + theApp.updateFrameSkip(); + + if(theApp.autoHideMenu && theApp.videoOption > VIDEO_4X && theApp.menuToggle) + OnFileTogglemenu(); + + emulating = true; + + if(theApp.autoLoadMostRecent) + OnFileLoadgameMostrecent(); + + theApp.frameskipadjust = 0; + theApp.renderedFrames = 0; + theApp.autoFrameSkipLastTime = theApp.throttleLastTime = systemGetClock(); + + theApp.rewindCount = 0; + theApp.rewindCounter = 0; + theApp.rewindSaveNeeded = false; + + return true; +} + +void MainWnd::OnInitMenuPopup(CMenu* pMenu, UINT nIndex, BOOL bSysMenu) +{ + ASSERT(pMenu != NULL); + + CCmdUI state; + state.m_pMenu = pMenu; + ASSERT(state.m_pOther == NULL); + ASSERT(state.m_pParentMenu == NULL); + + // determine if menu is popup in top-level menu and set m_pOther to + // it if so (m_pParentMenu == NULL indicates that it is secondary popup) + HMENU hParentMenu; + if (AfxGetThreadState()->m_hTrackingMenu == pMenu->m_hMenu) + state.m_pParentMenu = pMenu; // parent == child for tracking popup + else if ((hParentMenu = ::GetMenu(m_hWnd)) != NULL) { + CWnd* pParent = GetTopLevelParent(); + // child windows don't have menus -- need to go to the top! + if (pParent != NULL && + (hParentMenu = ::GetMenu(pParent->m_hWnd)) != NULL) { + int nIndexMax = ::GetMenuItemCount(hParentMenu); + for (int nIndex = 0; nIndex < nIndexMax; nIndex++) { + if (::GetSubMenu(hParentMenu, nIndex) == pMenu->m_hMenu) { + // when popup is found, m_pParentMenu is containing menu + state.m_pParentMenu = CMenu::FromHandle(hParentMenu); + break; + } + } + } + } + + state.m_nIndexMax = pMenu->GetMenuItemCount(); + for (state.m_nIndex = 0; state.m_nIndex < state.m_nIndexMax; + state.m_nIndex++) { + state.m_nID = pMenu->GetMenuItemID(state.m_nIndex); + if (state.m_nID == 0) + continue; // menu separator or invalid cmd - ignore it + + ASSERT(state.m_pOther == NULL); + ASSERT(state.m_pMenu != NULL); + if (state.m_nID == (UINT)-1) { + // possibly a popup menu, route to first item of that popup + state.m_pSubMenu = pMenu->GetSubMenu(state.m_nIndex); + if (state.m_pSubMenu == NULL || + (state.m_nID = state.m_pSubMenu->GetMenuItemID(0)) == 0 || + state.m_nID == (UINT)-1) { + continue; // first item of popup can't be routed to + } + state.DoUpdate(this, FALSE); // popups are never auto disabled + } else { + // normal menu item + // Auto enable/disable if frame window has 'm_bAutoMenuEnable' + // set and command is _not_ a system command. + state.m_pSubMenu = NULL; + state.DoUpdate(this, state.m_nID < 0xF000); + } + + // adjust for menu deletions and additions + UINT nCount = pMenu->GetMenuItemCount(); + if (nCount < state.m_nIndexMax) { + state.m_nIndex -= (state.m_nIndexMax - nCount); + while (state.m_nIndex < nCount && + pMenu->GetMenuItemID(state.m_nIndex) == state.m_nID) { + state.m_nIndex++; + } + } + state.m_nIndexMax = nCount; + } +} + +void MainWnd::OnMove(int x, int y) +{ + CWnd::OnMove(x, y); + + if(!theApp.changingVideoSize) { + if(this) { + if(!IsIconic()) { + RECT r; + + GetWindowRect(&r); + theApp.windowPositionX = r.left; + theApp.windowPositionY = r.top; + theApp.adjustDestRect(); + regSetDwordValue("windowX", theApp.windowPositionX); + regSetDwordValue("windowY", theApp.windowPositionY); + } + } + } +} + +void MainWnd::OnSize(UINT nType, int cx, int cy) +{ + CWnd::OnSize(nType, cx, cy); + + if(!theApp.changingVideoSize) { + if(this) { + if(!IsIconic()) { + if(theApp.iconic) { + if(emulating) { + soundResume(); + theApp.paused = false; + } + } + if(theApp.videoOption <= VIDEO_4X) { + theApp.surfaceSizeX = cx; + theApp.surfaceSizeY = cy; + theApp.adjustDestRect(); + if(theApp.display) + theApp.display->resize(theApp.dest.right-theApp.dest.left, theApp.dest.bottom-theApp.dest.top); + } + } else { + if(emulating) { + if(!theApp.paused) { + theApp.paused = true; + soundPause(); + } + } + theApp.iconic = true; + } + } + } +} + +void MainWnd::winSaveCheatListDefault() +{ + CString name; + CString filename; + + int index = theApp.filename.ReverseFind('\\'); + + if(index != -1) + name = theApp.filename.Right(theApp.filename.GetLength()-index-1); + else + name = theApp.filename; + CString dir = regQueryStringValue("saveDir", NULL); + if( dir[0] == '.' ) { + // handle as relative path + char baseDir[MAX_PATH+1]; + GetModuleFileName( NULL, baseDir, MAX_PATH ); + baseDir[MAX_PATH] = '\0'; // for security reasons + PathRemoveFileSpec( baseDir ); // removes the trailing file name and backslash + strcat( baseDir, "\\" ); + strcat( baseDir, dir ); + dir = baseDir; + } + + if(!dir.GetLength()) + dir = getDirFromFile(filename); + + if(isDriveRoot(dir)) + filename.Format("%s%s.clt", dir, name); + else + filename.Format("%s\\%s.clt", dir, name); + + winSaveCheatList(filename); +} + +void MainWnd::winSaveCheatList(const char *name) +{ + if(theApp.cartridgeType == 0) + cheatsSaveCheatList(name); + else + gbCheatsSaveCheatList(name); +} + +void MainWnd::winLoadCheatListDefault() +{ + CString name; + CString filename; + + int index = theApp.filename.ReverseFind('\\'); + + if(index != -1) + name = theApp.filename.Right(theApp.filename.GetLength()-index-1); + else + name = theApp.filename; + CString dir = regQueryStringValue("saveDir", NULL); + if( dir[0] == '.' ) { + // handle as relative path + char baseDir[MAX_PATH+1]; + GetModuleFileName( NULL, baseDir, MAX_PATH ); + baseDir[MAX_PATH] = '\0'; // for security reasons + PathRemoveFileSpec( baseDir ); // removes the trailing file name and backslash + strcat( baseDir, "\\" ); + strcat( baseDir, dir ); + dir = baseDir; + } + + if(!dir.GetLength()) + dir = getDirFromFile(filename); + + if(isDriveRoot(dir)) + filename.Format("%s%s.clt", dir, name); + else + filename.Format("%s\\%s.clt", dir, name); + + winLoadCheatList(filename); +} + +void MainWnd::winLoadCheatList(const char *name) +{ + bool res = false; + + if(theApp.cartridgeType == 0) + res = cheatsLoadCheatList(name); + else + res = gbCheatsLoadCheatList(name); + + if(res) + systemScreenMessage(winResLoadString(IDS_LOADED_CHEATS)); +} + +CString MainWnd::getDirFromFile(CString& file) +{ + CString temp = file; + int index = temp.ReverseFind('\\'); + + if(index != -1) { + temp = temp.Left(index); + if(temp.GetLength() == 2 && temp[1] == ':') + temp += "\\"; + } else { + temp = ""; + } + return temp; +} + +bool MainWnd::isDriveRoot(CString& file) +{ + if(file.GetLength() == 3) { + if(file[1] == ':' && file[2] == '\\') + return true; + } + return false; +} + +void MainWnd::writeBatteryFile() +{ + CString buffer; + CString filename; + + int index = theApp.filename.ReverseFind('\\'); + + if(index != -1) + buffer = theApp.filename.Right(theApp.filename.GetLength()-index-1); + else + buffer = theApp.filename; + + CString saveDir = regQueryStringValue("batteryDir", NULL); + if( saveDir[0] == '.' ) { + // handle as relative path + char baseDir[MAX_PATH+1]; + GetModuleFileName( NULL, baseDir, MAX_PATH ); + baseDir[MAX_PATH] = '\0'; // for security reasons + PathRemoveFileSpec( baseDir ); // removes the trailing file name and backslash + strcat( baseDir, "\\" ); + strcat( baseDir, saveDir ); + saveDir = baseDir; + } + + if(saveDir.IsEmpty()) + saveDir = getDirFromFile(theApp.filename); + + if(isDriveRoot(saveDir)) + filename.Format("%s%s.sav", saveDir, buffer); + else + filename.Format("%s\\%s.sav", saveDir, buffer); + + if(theApp.emulator.emuWriteBattery) + theApp.emulator.emuWriteBattery(filename); +} + + +void MainWnd::readBatteryFile() +{ + CString buffer; + CString filename; + + int index = theApp.filename.ReverseFind('\\'); + + if(index != -1) + buffer = theApp.filename.Right(theApp.filename.GetLength()-index-1); + else + buffer = theApp.filename; + + CString saveDir = regQueryStringValue("batteryDir", NULL); + if( saveDir[0] == '.' ) { + // handle as relative path + char baseDir[MAX_PATH+1]; + GetModuleFileName( NULL, baseDir, MAX_PATH ); + baseDir[MAX_PATH] = '\0'; // for security reasons + PathRemoveFileSpec( baseDir ); // removes the trailing file name and backslash + strcat( baseDir, "\\" ); + strcat( baseDir, saveDir ); + saveDir = baseDir; + } + + if(saveDir.IsEmpty()) + saveDir = getDirFromFile(theApp.filename); + + if(isDriveRoot(saveDir)) + filename.Format("%s%s.sav", saveDir, buffer); + else + filename.Format("%s\\%s.sav", saveDir, buffer); + + bool res = false; + + if(theApp.emulator.emuReadBattery) + res = theApp.emulator.emuReadBattery(filename); + + if(res) + systemScreenMessage(winResLoadString(IDS_LOADED_BATTERY)); +} + +CString MainWnd::winLoadFilter(UINT id) +{ + CString res = winResLoadString(id); + res.Replace('_','|'); + + return res; +} + +bool MainWnd::loadSaveGame(const char *name) +{ + if(theApp.emulator.emuReadState) + return theApp.emulator.emuReadState(name); + return false; +} + +bool MainWnd::writeSaveGame(const char *name) +{ + if(theApp.emulator.emuWriteState) + return theApp.emulator.emuWriteState(name); + return false; +} + +void MainWnd::OnContextMenu(CWnd* pWnd, CPoint point) +{ + winMouseOn(); + if(theApp.skin) { + if(theApp.popup == NULL) { + theApp.winAccelMgr.UpdateMenu(theApp.menu); + theApp.popup = CreatePopupMenu(); + if(theApp.menu != NULL) { + int count = GetMenuItemCount(theApp.menu); + OSVERSIONINFO info; + info.dwOSVersionInfoSize = sizeof(info); + GetVersionEx(&info); + + if(info.dwPlatformId == VER_PLATFORM_WIN32_WINDOWS) { + for(int i = 0; i < count; i++) { + char buffer[256]; + MENUITEMINFO info; + ZeroMemory(&info, sizeof(info)); + info.cbSize = sizeof(info) - sizeof(HBITMAP); + info.fMask = MIIM_STRING | MIIM_SUBMENU; + info.dwTypeData = buffer; + info.cch = 256; + if(!GetMenuItemInfo(theApp.menu, i, MF_BYPOSITION, &info)) { + } + if(!AppendMenu(theApp.popup, MF_POPUP|MF_STRING, (UINT_PTR)info.hSubMenu, buffer)) { + } + } + } else { + for(int i = 0; i < count; i++) { + wchar_t buffer[256]; + MENUITEMINFOW info; + ZeroMemory(&info, sizeof(info)); + info.cbSize = sizeof(info) - sizeof(HBITMAP); + info.fMask = MIIM_STRING | MIIM_SUBMENU; + info.dwTypeData = buffer; + info.cch = 256; + if(!GetMenuItemInfoW(theApp.menu, i, MF_BYPOSITION, &info)) { + } + if(!AppendMenuW(theApp.popup, MF_POPUP|MF_STRING, (UINT_PTR)info.hSubMenu, buffer)) { + } + } + } + } + } + int x = point.x; + int y = point.y; + if(x == -1 && y == -1) { + x = (theApp.dest.left + theApp.dest.right) / 2; + y = (theApp.dest.top + theApp.dest.bottom) / 2; + } + if(!TrackPopupMenu(theApp.popup, 0, x, y, 0, m_hWnd, NULL)) { + } + } +} + +void MainWnd::OnSystemMinimize() +{ + ShowWindow(SW_SHOWMINIMIZED); +} + + +bool MainWnd::fileOpenSelect( bool gb ) +{ + theApp.dir = _T(""); + CString initialDir; + if( gb ) { + initialDir = regQueryStringValue( _T("gbromdir"), _T(".") ); + } else { + initialDir = regQueryStringValue( _T("romdir"), _T(".") ); + } + + if( initialDir[0] == '.' ) { + // handle as relative path + char baseDir[MAX_PATH+1]; + GetModuleFileName( NULL, baseDir, MAX_PATH ); + baseDir[MAX_PATH] = '\0'; // for security reasons + PathRemoveFileSpec( baseDir ); // removes the trailing file name and backslash + strcat( baseDir, "\\" ); + strcat( baseDir, initialDir ); + initialDir = baseDir; + } + + if( !initialDir.IsEmpty() ) { + theApp.dir = initialDir; + } + + int selectedFilter = 0; + if( !gb ) { + selectedFilter = regQueryDwordValue( _T("selectedFilter"), 0); + if( (selectedFilter < 0) || (selectedFilter > 2) ) { + selectedFilter = 0; + } + } + + theApp.szFile = _T(""); + + LPCTSTR exts[] = { _T(""), _T(""), _T(""), _T("") }; + CString filter; + CString title; + if( gb ) { + filter = winLoadFilter( IDS_FILTER_GBROM ); + title = winResLoadString( IDS_SELECT_ROM ); + } else { + filter = winLoadFilter( IDS_FILTER_ROM ); + title = winResLoadString( IDS_SELECT_ROM ); + } + + FileDlg dlg( this, _T(""), filter, selectedFilter, _T(""), exts, theApp.dir, title, false); + + if( dlg.DoModal() == IDOK ) { + if( !gb ) { + regSetDwordValue( _T("selectedFilter"), dlg.m_ofn.nFilterIndex ); + } + theApp.szFile = dlg.GetPathName(); + theApp.dir = theApp.szFile.Left( dlg.m_ofn.nFileOffset ); + if( (theApp.dir.GetLength() > 3) && (theApp.dir[theApp.dir.GetLength()-1] == _T('\\')) ) { + theApp.dir = theApp.dir.Left( theApp.dir.GetLength() - 1 ); + } + SetCurrentDirectory( theApp.dir ); + return true; + } + return false; +} + + +void MainWnd::OnPaint() +{ + CPaintDC dc(this); // device context for painting + + if(emulating) { + theApp.painting = true; + systemDrawScreen(); + theApp.painting = false; + theApp.renderedFrames--; + } +} + +BOOL MainWnd::PreTranslateMessage(MSG* pMsg) +{ + if (CWnd::PreTranslateMessage(pMsg)) + return TRUE; + + if(pMsg->message >= WM_KEYFIRST && pMsg->message <= WM_KEYLAST) { + return theApp.hAccel != NULL && ::TranslateAccelerator(m_hWnd, theApp.hAccel, pMsg); + } + + return FALSE; +} + +void MainWnd::screenCapture(int captureNumber) +{ + CString buffer; + + CString captureDir = regQueryStringValue("captureDir", ""); + if( captureDir[0] == '.' ) { + // handle as relative path + char baseDir[MAX_PATH+1]; + GetModuleFileName( NULL, baseDir, MAX_PATH ); + baseDir[MAX_PATH] = '\0'; // for security reasons + PathRemoveFileSpec( baseDir ); // removes the trailing file name and backslash + strcat( baseDir, "\\" ); + strcat( baseDir, captureDir ); + captureDir = baseDir; + } + int index = theApp.filename.ReverseFind('\\'); + + CString name; + if(index != -1) + name = theApp.filename.Right(theApp.filename.GetLength()-index-1); + else + name = theApp.filename; + + if(captureDir.IsEmpty()) + captureDir = getDirFromFile(theApp.filename); + + LPCTSTR ext = "png"; + if(theApp.captureFormat != 0) + ext = "bmp"; + + if(isDriveRoot(captureDir)) + buffer.Format("%s%s_%02d.%s", + captureDir, + name, + captureNumber, + ext); + else + buffer.Format("%s\\%s_%02d.%s", + captureDir, + name, + captureNumber, + ext); + + // check if file exists + DWORD dwAttr = GetFileAttributes( buffer ); + if( dwAttr != INVALID_FILE_ATTRIBUTES ) { + // screenshot file already exists + screenCapture(++captureNumber); + // this will recursively use the first non-existent screenshot number + return; + } + + if(theApp.captureFormat == 0) + theApp.emulator.emuWritePNG(buffer); + else + theApp.emulator.emuWriteBMP(buffer); + + CString msg = winResLoadString(IDS_SCREEN_CAPTURE); + systemScreenMessage(msg); +} + +void MainWnd::winMouseOn() +{ + SetCursor(arrow); + if(theApp.videoOption > VIDEO_4X) { + theApp.mouseCounter = 10; + } else + theApp.mouseCounter = 0; +} + +void MainWnd::OnMouseMove(UINT nFlags, CPoint point) +{ + winMouseOn(); + + CWnd::OnMouseMove(nFlags, point); +} + +void MainWnd::OnInitMenu(CMenu* pMenu) +{ + CWnd::OnInitMenu(pMenu); + + soundPause(); +} + +void MainWnd::OnActivate(UINT nState, CWnd* pWndOther, BOOL bMinimized) +{ + CWnd::OnActivate(nState, pWndOther, bMinimized); + + bool a = (nState == WA_ACTIVE) || (nState == WA_CLICKACTIVE); + + if(a && theApp.input) { + theApp.active = a; + theApp.input->activate(); + if(!theApp.paused) { + if(emulating) { + soundResume(); + } + } + } else { + theApp.wasPaused = true; + if(theApp.pauseWhenInactive) { + if(emulating) { + soundPause(); + } + theApp.active = a; + } + + memset(theApp.delta,255,sizeof(theApp.delta)); + } + + if(theApp.paused && emulating) + { + theApp.painting = true; + systemDrawScreen(); + theApp.painting = false; + theApp.renderedFrames--; + } +} + +#if _MSC_VER <= 1200 +void MainWnd::OnActivateApp(BOOL bActive, HTASK hTask) +#else +void MainWnd::OnActivateApp(BOOL bActive, DWORD hTask) +#endif +{ + CWnd::OnActivateApp(bActive, hTask); + + if(theApp.tripleBuffering && theApp.videoOption > VIDEO_4X) { + if(bActive) { + if(theApp.display) + theApp.display->clear(); + } + } +} + +void MainWnd::OnDropFiles(HDROP hDropInfo) +{ + char szFile[1024]; + + if(DragQueryFile(hDropInfo, + 0, + szFile, + 1024)) { + theApp.szFile = szFile; + if(FileRun()) { + SetForegroundWindow(); + emulating = TRUE; + } else { + emulating = FALSE; + soundPause(); + } + } + DragFinish(hDropInfo); +} + +LRESULT MainWnd::OnMySysCommand(WPARAM wParam, LPARAM lParam) +{ + if(emulating && !theApp.paused) { + if((wParam&0xFFF0) == SC_SCREENSAVE || (wParam&0xFFF0) == SC_MONITORPOWER) + return 0; + } + return Default(); +} + diff --git a/src/win32/MainWnd.h b/src/win32/MainWnd.h index 6407b415..14c3404f 100644 --- a/src/win32/MainWnd.h +++ b/src/win32/MainWnd.h @@ -1,442 +1,446 @@ -// -*- C++ -*- -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#if !defined(AFX_MAINWND_H__E8AD28B9_C9FB_4EC2_A2DC_DD1BBA55A275__INCLUDED_) -#define AFX_MAINWND_H__E8AD28B9_C9FB_4EC2_A2DC_DD1BBA55A275__INCLUDED_ - -#if _MSC_VER > 1000 -#pragma once -#endif // _MSC_VER > 1000 -// MainWnd.h : header file -// - -///////////////////////////////////////////////////////////////////////////// -// MainWnd window - -class MainWnd : public CWnd -{ - // Construction - public: - MainWnd(); - - // Attributes - public: - - // Operations - public: - bool FileRun(); - - // Overrides - // ClassWizard generated virtual function overrides - //{{AFX_VIRTUAL(MainWnd) - public: - virtual BOOL PreTranslateMessage(MSG* pMsg); - //}}AFX_VIRTUAL - - // Implementation - public: - HCURSOR arrow; - void winMouseOn(); - void screenCapture(int captureNumber); - HACCEL m_hAccelTable; - bool fileOpenSelect(); - afx_msg LRESULT OnConfirmMode(WPARAM, LPARAM); - afx_msg LRESULT OnMySysCommand(WPARAM, LPARAM); - afx_msg void OnUpdateFileLoadGameSlot(CCmdUI *pCmdUI); - afx_msg void OnUpdateFileSaveGameSlot(CCmdUI *pCmdUI); - afx_msg void OnUpdateOptionsJoypadAutofire(CCmdUI *pCmdUI); - afx_msg BOOL OnOptionsJoypadAutofire(UINT nID); - afx_msg void OnUpdateOptionsJoypadDefault(CCmdUI *pCmdUI); - afx_msg BOOL OnOptionsJoypadDefault(UINT nID); - afx_msg void OnUpdateOptionsFilterIFB(CCmdUI *pCmdUI); - afx_msg BOOL OnOptionsFilterIFB(UINT nID); - afx_msg void OnUpdateOptionsFilter(CCmdUI *pCmdUI); - afx_msg BOOL OnOptionsFilter(UINT nID); - afx_msg void OnUpdateOptionsPriority(CCmdUI *pCmdUI); - afx_msg BOOL OnOptionsPriority(UINT nID); - afx_msg void OnSetFocus(CWnd * pOldWnd); - afx_msg void OnKillFocus(CWnd * pNewWnd); - - void updateSoundChannels(UINT nID); - afx_msg void OnUpdateOptionsSoundVolume(CCmdUI *pCmdUI); - afx_msg BOOL OnOptionsSoundVolume(UINT nID); - afx_msg void OnUpdateOptionsEmulatorShowSpeed(CCmdUI *pCmdUI); - afx_msg BOOL OnOptionsEmulatorShowSpeed(UINT nID); - afx_msg void OnSystemMinimize(); - afx_msg void OnUpdateVideoLayer(CCmdUI* pCmdUI); - afx_msg BOOL OnVideoLayer(UINT nID); - void winConfirmMode(); - afx_msg BOOL OnOptionVideoSize(UINT nID); - afx_msg BOOL OnOptionsFrameskip(UINT nID); - bool fileImportGSACodeFile(CString& fileName); - bool writeSaveGame(const char *name); - bool loadSaveGame(const char *name); - CString winLoadFilter(UINT id); - void winLoadCheatList(const char *name); - void winLoadCheatListDefault(); - void readBatteryFile(); - void writeBatteryFile(); - bool isDriveRoot(CString& file); - CString getDirFromFile(CString& file); - void winSaveCheatList(const char *name); - void winSaveCheatListDefault(); - virtual ~MainWnd(); - - // Generated message map functions - protected: - //{{AFX_MSG(MainWnd) - afx_msg void OnClose(); - afx_msg void OnHelpAbout(); - afx_msg void OnHelpFaq(); - afx_msg void OnFileOpen(); - afx_msg void OnInitMenuPopup(CMenu* pPopupMenu, UINT nIndex, BOOL bSysMenu); - afx_msg void OnFilePause(); - afx_msg void OnUpdateFilePause(CCmdUI* pCmdUI); - afx_msg void OnFileReset(); - afx_msg void OnUpdateFileReset(CCmdUI* pCmdUI); - afx_msg void OnUpdateFileRecentFreeze(CCmdUI* pCmdUI); - afx_msg void OnFileRecentReset(); - afx_msg void OnFileRecentFreeze(); - afx_msg void OnFileExit(); - afx_msg void OnFileClose(); - afx_msg void OnUpdateFileClose(CCmdUI* pCmdUI); - afx_msg void OnFileOpengameboy(); - afx_msg void OnFileLoad(); - afx_msg void OnUpdateFileLoad(CCmdUI* pCmdUI); - afx_msg void OnFileSave(); - afx_msg void OnUpdateFileSave(CCmdUI* pCmdUI); - afx_msg void OnFileImportBatteryfile(); - afx_msg void OnUpdateFileImportBatteryfile(CCmdUI* pCmdUI); - afx_msg void OnFileImportGamesharkcodefile(); - afx_msg void OnUpdateFileImportGamesharkcodefile(CCmdUI* pCmdUI); - afx_msg void OnFileImportGamesharksnapshot(); - afx_msg void OnUpdateFileImportGamesharksnapshot(CCmdUI* pCmdUI); - afx_msg void OnFileExportBatteryfile(); - afx_msg void OnUpdateFileExportBatteryfile(CCmdUI* pCmdUI); - afx_msg void OnFileExportGamesharksnapshot(); - afx_msg void OnUpdateFileExportGamesharksnapshot(CCmdUI* pCmdUI); - afx_msg void OnFileScreencapture(); - afx_msg void OnUpdateFileScreencapture(CCmdUI* pCmdUI); - afx_msg void OnFileRominformation(); - afx_msg void OnUpdateFileRominformation(CCmdUI* pCmdUI); - afx_msg void OnFileTogglemenu(); - afx_msg void OnUpdateFileTogglemenu(CCmdUI* pCmdUI); - afx_msg void OnUpdateOptionsFrameskipThrottleNothrottle(CCmdUI* pCmdUI); - afx_msg void OnUpdateOptionsFrameskipThrottle25(CCmdUI* pCmdUI); - afx_msg void OnUpdateOptionsFrameskipThrottle50(CCmdUI* pCmdUI); - afx_msg void OnUpdateOptionsFrameskipThrottle100(CCmdUI* pCmdUI); - afx_msg void OnUpdateOptionsFrameskipThrottle150(CCmdUI* pCmdUI); - afx_msg void OnUpdateOptionsFrameskipThrottle200(CCmdUI* pCmdUI); - afx_msg void OnUpdateOptionsFrameskipThrottleOther(CCmdUI* pCmdUI); - afx_msg void OnOptionsFrameskipThrottleNothrottle(); - afx_msg void OnOptionsFrameskipThrottle25(); - afx_msg void OnOptionsFrameskipThrottle50(); - afx_msg void OnOptionsFrameskipThrottle100(); - afx_msg void OnOptionsFrameskipThrottle150(); - afx_msg void OnOptionsFrameskipThrottle200(); - afx_msg void OnOptionsFrameskipThrottleOther(); - afx_msg void OnOptionsFrameskipAutomatic(); - afx_msg void OnUpdateOptionsFrameskipAutomatic(CCmdUI* pCmdUI); - afx_msg void OnUpdateOptionsVideoFrameskip0(CCmdUI* pCmdUI); - afx_msg void OnUpdateOptionsVideoFrameskip1(CCmdUI* pCmdUI); - afx_msg void OnUpdateOptionsVideoFrameskip2(CCmdUI* pCmdUI); - afx_msg void OnUpdateOptionsVideoFrameskip3(CCmdUI* pCmdUI); - afx_msg void OnUpdateOptionsVideoFrameskip4(CCmdUI* pCmdUI); - afx_msg void OnUpdateOptionsVideoFrameskip5(CCmdUI* pCmdUI); - afx_msg void OnUpdateOptionsVideoFrameskip6(CCmdUI* pCmdUI); - afx_msg void OnUpdateOptionsVideoFrameskip7(CCmdUI* pCmdUI); - afx_msg void OnUpdateOptionsVideoFrameskip8(CCmdUI* pCmdUI); - afx_msg void OnUpdateOptionsVideoFrameskip9(CCmdUI* pCmdUI); - afx_msg void OnOptionsVideoVsync(); - afx_msg void OnUpdateOptionsVideoVsync(CCmdUI* pCmdUI); - afx_msg void OnUpdateOptionsVideoX1(CCmdUI* pCmdUI); - afx_msg void OnUpdateOptionsVideoX2(CCmdUI* pCmdUI); - afx_msg void OnUpdateOptionsVideoX3(CCmdUI* pCmdUI); - afx_msg void OnUpdateOptionsVideoX4(CCmdUI* pCmdUI); - afx_msg void OnUpdateOptionsVideoFullscreen320x240(CCmdUI* pCmdUI); - afx_msg void OnUpdateOptionsVideoFullscreen640x480(CCmdUI* pCmdUI); - afx_msg void OnUpdateOptionsVideoFullscreen800x600(CCmdUI* pCmdUI); - afx_msg void OnOptionsVideoFullscreen320x240(); - afx_msg void OnOptionsVideoFullscreen640x480(); - afx_msg void OnOptionsVideoFullscreen800x600(); - afx_msg void OnOptionsVideoFullscreen(); - afx_msg void OnUpdateOptionsVideoFullscreen(CCmdUI* pCmdUI); - afx_msg void OnMove(int x, int y); - afx_msg void OnSize(UINT nType, int cx, int cy); - afx_msg void OnOptionsVideoDisablesfx(); - afx_msg void OnUpdateOptionsVideoDisablesfx(CCmdUI* pCmdUI); - afx_msg void OnOptionsVideoFullscreenstretchtofit(); - afx_msg void OnUpdateOptionsVideoFullscreenstretchtofit(CCmdUI* pCmdUI); - afx_msg void OnOptionsVideoRendermethodGdi(); - afx_msg void OnUpdateOptionsVideoRendermethodGdi(CCmdUI* pCmdUI); - afx_msg void OnOptionsVideoRendermethodDirectdraw(); - afx_msg void OnUpdateOptionsVideoRendermethodDirectdraw(CCmdUI* pCmdUI); - afx_msg void OnOptionsVideoRendermethodDirect3d(); - afx_msg void OnUpdateOptionsVideoRendermethodDirect3d(CCmdUI* pCmdUI); - afx_msg void OnOptionsVideoRendermethodOpengl(); - afx_msg void OnUpdateOptionsVideoRendermethodOpengl(CCmdUI* pCmdUI); - afx_msg void OnOptionsVideoTriplebuffering(); - afx_msg void OnUpdateOptionsVideoTriplebuffering(CCmdUI* pCmdUI); - afx_msg void OnOptionsVideoDdrawemulationonly(); - afx_msg void OnUpdateOptionsVideoDdrawemulationonly(CCmdUI* pCmdUI); - afx_msg void OnOptionsVideoDdrawusevideomemory(); - afx_msg void OnUpdateOptionsVideoDdrawusevideomemory(CCmdUI* pCmdUI); - afx_msg void OnOptionsVideoRenderoptionsD3dnofilter(); - afx_msg void OnUpdateOptionsVideoRenderoptionsD3dnofilter(CCmdUI* pCmdUI); - afx_msg void OnOptionsVideoRenderoptionsD3dbilinear(); - afx_msg void OnUpdateOptionsVideoRenderoptionsD3dbilinear(CCmdUI* pCmdUI); - afx_msg void OnOptionsVideoRenderoptionsGlnearest(); - afx_msg void OnUpdateOptionsVideoRenderoptionsGlnearest(CCmdUI* pCmdUI); - afx_msg void OnOptionsVideoRenderoptionsGlbilinear(); - afx_msg void OnUpdateOptionsVideoRenderoptionsGlbilinear(CCmdUI* pCmdUI); - afx_msg void OnOptionsVideoRenderoptionsGltriangle(); - afx_msg void OnUpdateOptionsVideoRenderoptionsGltriangle(CCmdUI* pCmdUI); - afx_msg void OnOptionsVideoRenderoptionsGlquads(); - afx_msg void OnUpdateOptionsVideoRenderoptionsGlquads(CCmdUI* pCmdUI); - afx_msg void OnContextMenu(CWnd* pWnd, CPoint point); - afx_msg void OnOptionsEmulatorAssociate(); - afx_msg void OnOptionsEmulatorDirectories(); - afx_msg void OnOptionsEmulatorDisablestatusmessages(); - afx_msg void OnUpdateOptionsEmulatorDisablestatusmessages(CCmdUI* pCmdUI); - afx_msg void OnOptionsEmulatorSynchronize(); - afx_msg void OnUpdateOptionsEmulatorSynchronize(CCmdUI* pCmdUI); - afx_msg void OnOptionsEmulatorPausewheninactive(); - afx_msg void OnUpdateOptionsEmulatorPausewheninactive(CCmdUI* pCmdUI); - afx_msg void OnOptionsEmulatorSpeeduptoggle(); - afx_msg void OnUpdateOptionsEmulatorSpeeduptoggle(CCmdUI* pCmdUI); - afx_msg void OnOptionsEmulatorRemoveintrosgba(); - afx_msg void OnUpdateOptionsEmulatorRemoveintrosgba(CCmdUI* pCmdUI); - afx_msg void OnOptionsEmulatorAutomaticallyipspatch(); - afx_msg void OnUpdateOptionsEmulatorAutomaticallyipspatch(CCmdUI* pCmdUI); - afx_msg void OnOptionsEmulatorAgbprint(); - afx_msg void OnUpdateOptionsEmulatorAgbprint(CCmdUI* pCmdUI); - afx_msg void OnOptionsEmulatorRealtimeclock(); - afx_msg void OnUpdateOptionsEmulatorRealtimeclock(CCmdUI* pCmdUI); - afx_msg void OnOptionsEmulatorAutohidemenu(); - afx_msg void OnUpdateOptionsEmulatorAutohidemenu(CCmdUI* pCmdUI); - afx_msg void OnOptionsEmulatorRewindinterval(); - afx_msg void OnOptionsEmulatorSavetypeAutomatic(); - afx_msg void OnUpdateOptionsEmulatorSavetypeAutomatic(CCmdUI* pCmdUI); - afx_msg void OnOptionsEmulatorSavetypeEeprom(); - afx_msg void OnUpdateOptionsEmulatorSavetypeEeprom(CCmdUI* pCmdUI); - afx_msg void OnOptionsEmulatorSavetypeSram(); - afx_msg void OnUpdateOptionsEmulatorSavetypeSram(CCmdUI* pCmdUI); - afx_msg void OnOptionsEmulatorSavetypeFlash(); - afx_msg void OnUpdateOptionsEmulatorSavetypeFlash(CCmdUI* pCmdUI); - afx_msg void OnOptionsEmulatorSavetypeEepromsensor(); - afx_msg void OnUpdateOptionsEmulatorSavetypeEepromsensor(CCmdUI* pCmdUI); - afx_msg void OnOptionsEmulatorSavetypeNone(); - afx_msg void OnUpdateOptionsEmulatorSavetypeNone(CCmdUI* pCmdUI); - afx_msg void OnOptionsEmulatorSavetypeFlash512k(); - afx_msg void OnUpdateOptionsEmulatorSavetypeFlash512k(CCmdUI* pCmdUI); - afx_msg void OnOptionsEmulatorSavetypeFlash1m(); - afx_msg void OnUpdateOptionsEmulatorSavetypeFlash1m(CCmdUI* pCmdUI); - afx_msg void OnOptionsEmulatorUsebiosfile(); - afx_msg void OnUpdateOptionsEmulatorUsebiosfile(CCmdUI* pCmdUI); - afx_msg void OnOptionsEmulatorSkipbios(); - afx_msg void OnUpdateOptionsEmulatorSkipbios(CCmdUI* pCmdUI); - afx_msg void OnOptionsEmulatorSelectbiosfile(); - afx_msg void OnOptionsEmulatorPngformat(); - afx_msg void OnUpdateOptionsEmulatorPngformat(CCmdUI* pCmdUI); - afx_msg void OnOptionsEmulatorBmpformat(); - afx_msg void OnUpdateOptionsEmulatorBmpformat(CCmdUI* pCmdUI); - afx_msg void OnOptionsSoundOff(); /* mute hax */ - afx_msg void OnUpdateOptionsSoundOff(CCmdUI* pCmdUI); /* mute hax */ - afx_msg void OnOptionsSoundMute(); - afx_msg void OnUpdateOptionsSoundMute(CCmdUI* pCmdUI); - afx_msg void OnOptionsSoundOn(); - afx_msg void OnUpdateOptionsSoundOn(CCmdUI* pCmdUI); - afx_msg void OnOptionsSoundUseoldsynchronization(); - afx_msg void OnUpdateOptionsSoundUseoldsynchronization(CCmdUI* pCmdUI); - afx_msg void OnOptionsSoundEcho(); - afx_msg void OnUpdateOptionsSoundEcho(CCmdUI* pCmdUI); - afx_msg void OnOptionsSoundLowpassfilter(); - afx_msg void OnUpdateOptionsSoundLowpassfilter(CCmdUI* pCmdUI); - afx_msg void OnOptionsSoundReversestereo(); - afx_msg void OnUpdateOptionsSoundReversestereo(CCmdUI* pCmdUI); - afx_msg void OnOptionsSound11khz(); - afx_msg void OnUpdateOptionsSound11khz(CCmdUI* pCmdUI); - afx_msg void OnOptionsSound22khz(); - afx_msg void OnUpdateOptionsSound22khz(CCmdUI* pCmdUI); - afx_msg void OnOptionsSound44khz(); - afx_msg void OnUpdateOptionsSound44khz(CCmdUI* pCmdUI); - afx_msg void OnOptionsSoundChannel1(); - afx_msg void OnUpdateOptionsSoundChannel1(CCmdUI* pCmdUI); - afx_msg void OnOptionsSoundChannel2(); - afx_msg void OnUpdateOptionsSoundChannel2(CCmdUI* pCmdUI); - afx_msg void OnOptionsSoundChannel3(); - afx_msg void OnUpdateOptionsSoundChannel3(CCmdUI* pCmdUI); - afx_msg void OnOptionsSoundChannel4(); - afx_msg void OnUpdateOptionsSoundChannel4(CCmdUI* pCmdUI); - afx_msg void OnOptionsSoundDirectsounda(); - afx_msg void OnUpdateOptionsSoundDirectsounda(CCmdUI* pCmdUI); - afx_msg void OnOptionsSoundDirectsoundb(); - afx_msg void OnUpdateOptionsSoundDirectsoundb(CCmdUI* pCmdUI); - afx_msg void OnOptionsGameboyBorder(); - afx_msg void OnUpdateOptionsGameboyBorder(CCmdUI* pCmdUI); - afx_msg void OnOptionsGameboyPrinter(); - afx_msg void OnUpdateOptionsGameboyPrinter(CCmdUI* pCmdUI); - afx_msg void OnOptionsGameboyBorderAutomatic(); - afx_msg void OnUpdateOptionsGameboyBorderAutomatic(CCmdUI* pCmdUI); - afx_msg void OnOptionsGameboyAutomatic(); - afx_msg void OnUpdateOptionsGameboyAutomatic(CCmdUI* pCmdUI); - afx_msg void OnOptionsGameboyGba(); - afx_msg void OnUpdateOptionsGameboyGba(CCmdUI* pCmdUI); - afx_msg void OnOptionsGameboyCgb(); - afx_msg void OnUpdateOptionsGameboyCgb(CCmdUI* pCmdUI); - afx_msg void OnOptionsGameboySgb(); - afx_msg void OnUpdateOptionsGameboySgb(CCmdUI* pCmdUI); - afx_msg void OnOptionsGameboySgb2(); - afx_msg void OnUpdateOptionsGameboySgb2(CCmdUI* pCmdUI); - afx_msg void OnOptionsGameboyGb(); - afx_msg void OnUpdateOptionsGameboyGb(CCmdUI* pCmdUI); - afx_msg void OnOptionsGameboyRealcolors(); - afx_msg void OnUpdateOptionsGameboyRealcolors(CCmdUI* pCmdUI); - afx_msg void OnOptionsGameboyGameboycolors(); - afx_msg void OnUpdateOptionsGameboyGameboycolors(CCmdUI* pCmdUI); - afx_msg void OnOptionsGameboyColors(); - afx_msg void OnOptionsFilterDisablemmx(); - afx_msg void OnUpdateOptionsFilterDisablemmx(CCmdUI* pCmdUI); - afx_msg void OnOptionsLanguageSystem(); - afx_msg void OnUpdateOptionsLanguageSystem(CCmdUI* pCmdUI); - afx_msg void OnOptionsLanguageEnglish(); - afx_msg void OnUpdateOptionsLanguageEnglish(CCmdUI* pCmdUI); - afx_msg void OnOptionsLanguageOther(); - afx_msg void OnUpdateOptionsLanguageOther(CCmdUI* pCmdUI); - afx_msg void OnOptionsJoypadConfigure1(); - afx_msg void OnUpdateOptionsJoypadConfigure1(CCmdUI* pCmdUI); - afx_msg void OnOptionsJoypadConfigure2(); - afx_msg void OnUpdateOptionsJoypadConfigure2(CCmdUI* pCmdUI); - afx_msg void OnOptionsJoypadConfigure3(); - afx_msg void OnUpdateOptionsJoypadConfigure3(CCmdUI* pCmdUI); - afx_msg void OnOptionsJoypadConfigure4(); - afx_msg void OnUpdateOptionsJoypadConfigure4(CCmdUI* pCmdUI); - afx_msg void OnOptionsJoypadMotionconfigure(); - afx_msg void OnUpdateOptionsJoypadMotionconfigure(CCmdUI* pCmdUI); - afx_msg void OnCheatsSearchforcheats(); - afx_msg void OnUpdateCheatsSearchforcheats(CCmdUI* pCmdUI); - afx_msg void OnCheatsCheatlist(); - afx_msg void OnUpdateCheatsCheatlist(CCmdUI* pCmdUI); - afx_msg void OnCheatsAutomaticsaveloadcheats(); - afx_msg void OnCheatsLoadcheatlist(); - afx_msg void OnUpdateCheatsLoadcheatlist(CCmdUI* pCmdUI); - afx_msg void OnCheatsSavecheatlist(); - afx_msg void OnUpdateCheatsSavecheatlist(CCmdUI* pCmdUI); - afx_msg void OnToolsDisassemble(); - afx_msg void OnUpdateToolsDisassemble(CCmdUI* pCmdUI); - afx_msg void OnToolsLogging(); - afx_msg void OnUpdateToolsLogging(CCmdUI* pCmdUI); - afx_msg void OnToolsIoviewer(); - afx_msg void OnUpdateToolsIoviewer(CCmdUI* pCmdUI); - afx_msg void OnToolsMapview(); - afx_msg void OnUpdateToolsMapview(CCmdUI* pCmdUI); - afx_msg void OnToolsMemoryviewer(); - afx_msg void OnUpdateToolsMemoryviewer(CCmdUI* pCmdUI); - afx_msg void OnToolsOamviewer(); - afx_msg void OnUpdateToolsOamviewer(CCmdUI* pCmdUI); - afx_msg void OnToolsPaletteview(); - afx_msg void OnUpdateToolsPaletteview(CCmdUI* pCmdUI); - afx_msg void OnToolsTileviewer(); - afx_msg void OnUpdateToolsTileviewer(CCmdUI* pCmdUI); - afx_msg void OnDebugNextframe(); - afx_msg void OnUpdateCheatsAutomaticsaveloadcheats(CCmdUI* pCmdUI); - afx_msg void OnToolsDebugGdb(); - afx_msg void OnUpdateToolsDebugGdb(CCmdUI* pCmdUI); - afx_msg void OnToolsDebugLoadandwait(); - afx_msg void OnUpdateToolsDebugLoadandwait(CCmdUI* pCmdUI); - afx_msg void OnToolsDebugBreak(); - afx_msg void OnUpdateToolsDebugBreak(CCmdUI* pCmdUI); - afx_msg void OnToolsDebugDisconnect(); - afx_msg void OnUpdateToolsDebugDisconnect(CCmdUI* pCmdUI); - afx_msg void OnOptionsSoundStartrecording(); - afx_msg void OnUpdateOptionsSoundStartrecording(CCmdUI* pCmdUI); - afx_msg void OnOptionsSoundStoprecording(); - afx_msg void OnUpdateOptionsSoundStoprecording(CCmdUI* pCmdUI); - afx_msg void OnToolsRecordStartavirecording(); - afx_msg void OnUpdateToolsRecordStartavirecording(CCmdUI* pCmdUI); - afx_msg void OnToolsRecordStopavirecording(); - afx_msg void OnUpdateToolsRecordStopavirecording(CCmdUI* pCmdUI); - afx_msg void OnPaint(); - afx_msg void OnToolsRecordStartmovierecording(); - afx_msg void OnUpdateToolsRecordStartmovierecording(CCmdUI* pCmdUI); - afx_msg void OnToolsRecordStopmovierecording(); - afx_msg void OnUpdateToolsRecordStopmovierecording(CCmdUI* pCmdUI); - afx_msg void OnToolsPlayStartmovieplaying(); - afx_msg void OnUpdateToolsPlayStartmovieplaying(CCmdUI* pCmdUI); - afx_msg void OnToolsPlayStopmovieplaying(); - afx_msg void OnUpdateToolsPlayStopmovieplaying(CCmdUI* pCmdUI); - afx_msg void OnToolsRewind(); - afx_msg void OnUpdateToolsRewind(CCmdUI* pCmdUI); - afx_msg void OnToolsCustomize(); - afx_msg void OnUpdateToolsCustomize(CCmdUI* pCmdUI); - afx_msg void OnHelpBugreport(); - afx_msg void OnMouseMove(UINT nFlags, CPoint point); - afx_msg void OnInitMenu(CMenu* pMenu); - afx_msg void OnActivate(UINT nState, CWnd* pWndOther, BOOL bMinimized); -#if _MSC_VER <= 1200 - afx_msg void OnActivateApp(BOOL bActive, HTASK hTask); -#else - afx_msg void OnActivateApp(BOOL bActive, DWORD hTask); -#endif - afx_msg void OnDropFiles(HDROP hDropInfo); - afx_msg void OnFileSavegameOldestslot(); - afx_msg void OnUpdateFileSavegameOldestslot(CCmdUI* pCmdUI); - afx_msg void OnFileLoadgameMostrecent(); - afx_msg void OnUpdateFileLoadgameMostrecent(CCmdUI* pCmdUI); - afx_msg void OnFileLoadgameAutoloadmostrecent(); - afx_msg void OnUpdateFileLoadgameAutoloadmostrecent(CCmdUI* pCmdUI); - afx_msg void OnOptionsSoundVolume25x(); - afx_msg void OnUpdateOptionsSoundVolume25x(CCmdUI* pCmdUI); - afx_msg void OnOptionsSoundVolume5x(); - afx_msg void OnUpdateOptionsSoundVolume5x(CCmdUI* pCmdUI); - afx_msg void OnCheatsDisablecheats(); - afx_msg void OnUpdateCheatsDisablecheats(CCmdUI* pCmdUI); - afx_msg void OnOptionsVideoFullscreenmaxscale(); - //}}AFX_MSG - DECLARE_MESSAGE_MAP() - - afx_msg BOOL OnFileRecentFile(UINT nID); - afx_msg BOOL OnFileLoadSlot(UINT nID); - afx_msg BOOL OnFileSaveSlot(UINT nID); - afx_msg void OnOptionsFilterLcdcolors(); - afx_msg void OnUpdateOptionsFilterLcdcolors(CCmdUI *pCmdUI); - - afx_msg BOOL OnOptionsSoundPcminterpolation(UINT nID); - afx_msg void OnUpdateOptionsSoundPcminterpolation(CCmdUI *pCmdUI); -public: - afx_msg void OnOptionsVideoFullscreen1280x1024(); - afx_msg void OnOptionsVideoFullscreen1024x768(); - afx_msg void OnUpdateOptionsVideoFullscreen1024x768(CCmdUI *pCmdUI); - afx_msg void OnUpdateOptionsVideoFullscreen1280x1024(CCmdUI *pCmdUI); - void OnLinkOptions(); - void OnOptionsLinkLog() ; - void OnUpdateOptionsLinkLog(CCmdUI* pCmdUI) ; - void OnOptionsLinkRFU() ; - void OnUpdateOptionsLinkRFU(CCmdUI* pCmdUI) ; - - - -}; - -///////////////////////////////////////////////////////////////////////////// - -//{{AFX_INSERT_LOCATION}} -// Microsoft Visual C++ will insert additional declarations immediately before the previous line. - -#endif // !defined(AFX_MAINWND_H__E8AD28B9_C9FB_4EC2_A2DC_DD1BBA55A275__INCLUDED_) +// -*- C++ -*- +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2005 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#if !defined(AFX_MAINWND_H__E8AD28B9_C9FB_4EC2_A2DC_DD1BBA55A275__INCLUDED_) +#define AFX_MAINWND_H__E8AD28B9_C9FB_4EC2_A2DC_DD1BBA55A275__INCLUDED_ + +#if _MSC_VER > 1000 +#pragma once +#endif // _MSC_VER > 1000 +// MainWnd.h : header file +// + +///////////////////////////////////////////////////////////////////////////// +// MainWnd window + +class MainWnd : public CWnd +{ + // Construction + public: + MainWnd(); + + // Attributes + public: + + // Operations + public: + bool FileRun(); + + // Overrides + // ClassWizard generated virtual function overrides + //{{AFX_VIRTUAL(MainWnd) + public: + virtual BOOL PreTranslateMessage(MSG* pMsg); + //}}AFX_VIRTUAL + + // Implementation + public: + HCURSOR arrow; + void winMouseOn(); + void screenCapture(int captureNumber); + HACCEL m_hAccelTable; + bool fileOpenSelect( bool gb = false ); + afx_msg LRESULT OnConfirmMode(WPARAM, LPARAM); + afx_msg LRESULT OnMySysCommand(WPARAM, LPARAM); + afx_msg void OnUpdateFileLoadGameSlot(CCmdUI *pCmdUI); + afx_msg void OnUpdateFileSaveGameSlot(CCmdUI *pCmdUI); + afx_msg void OnUpdateOptionsJoypadAutofire(CCmdUI *pCmdUI); + afx_msg BOOL OnOptionsJoypadAutofire(UINT nID); + afx_msg void OnUpdateOptionsJoypadDefault(CCmdUI *pCmdUI); + afx_msg BOOL OnOptionsJoypadDefault(UINT nID); + afx_msg void OnUpdateOptionsFilterIFB(CCmdUI *pCmdUI); + afx_msg BOOL OnOptionsFilterIFB(UINT nID); + afx_msg void OnUpdateOptionsFilter(CCmdUI *pCmdUI); + afx_msg BOOL OnOptionsFilter(UINT nID); + afx_msg void OnUpdateOptionsPriority(CCmdUI *pCmdUI); + afx_msg BOOL OnOptionsPriority(UINT nID); + void updateSoundChannels(UINT nID); + afx_msg void OnUpdateOptionsSoundVolume(CCmdUI *pCmdUI); + afx_msg BOOL OnOptionsSoundVolume(UINT nID); + afx_msg void OnUpdateOptionsEmulatorShowSpeed(CCmdUI *pCmdUI); + afx_msg BOOL OnOptionsEmulatorShowSpeed(UINT nID); + afx_msg void OnSystemMinimize(); + afx_msg void OnUpdateVideoLayer(CCmdUI* pCmdUI); + afx_msg BOOL OnVideoLayer(UINT nID); + void winConfirmMode(); + afx_msg BOOL OnOptionVideoSize(UINT nID); + afx_msg BOOL OnOptionsFrameskip(UINT nID); + bool fileImportGSACodeFile(CString& fileName); + bool writeSaveGame(const char *name); + bool loadSaveGame(const char *name); + CString winLoadFilter(UINT id); + void winLoadCheatList(const char *name); + void winLoadCheatListDefault(); + void readBatteryFile(); + void writeBatteryFile(); + bool isDriveRoot(CString& file); + CString getDirFromFile(CString& file); + void winSaveCheatList(const char *name); + void winSaveCheatListDefault(); + virtual ~MainWnd(); + + // Generated message map functions + protected: + //{{AFX_MSG(MainWnd) + afx_msg void OnClose(); + afx_msg void OnHelpAbout(); + afx_msg void OnHelpFaq(); + afx_msg void OnFileOpen(); + afx_msg void OnInitMenuPopup(CMenu* pPopupMenu, UINT nIndex, BOOL bSysMenu); + afx_msg void OnFilePause(); + afx_msg void OnUpdateFilePause(CCmdUI* pCmdUI); + afx_msg void OnFileReset(); + afx_msg void OnUpdateFileReset(CCmdUI* pCmdUI); + afx_msg void OnUpdateFileRecentFreeze(CCmdUI* pCmdUI); + afx_msg void OnFileRecentReset(); + afx_msg void OnFileRecentFreeze(); + afx_msg void OnFileExit(); + afx_msg void OnFileClose(); + afx_msg void OnUpdateFileClose(CCmdUI* pCmdUI); + afx_msg void OnFileOpengameboy(); + afx_msg void OnFileLoad(); + afx_msg void OnUpdateFileLoad(CCmdUI* pCmdUI); + afx_msg void OnFileSave(); + afx_msg void OnUpdateFileSave(CCmdUI* pCmdUI); + afx_msg void OnFileImportBatteryfile(); + afx_msg void OnUpdateFileImportBatteryfile(CCmdUI* pCmdUI); + afx_msg void OnFileImportGamesharkcodefile(); + afx_msg void OnUpdateFileImportGamesharkcodefile(CCmdUI* pCmdUI); + afx_msg void OnFileImportGamesharksnapshot(); + afx_msg void OnUpdateFileImportGamesharksnapshot(CCmdUI* pCmdUI); + afx_msg void OnFileExportBatteryfile(); + afx_msg void OnUpdateFileExportBatteryfile(CCmdUI* pCmdUI); + afx_msg void OnFileExportGamesharksnapshot(); + afx_msg void OnUpdateFileExportGamesharksnapshot(CCmdUI* pCmdUI); + afx_msg void OnFileScreencapture(); + afx_msg void OnUpdateFileScreencapture(CCmdUI* pCmdUI); + afx_msg void OnFileRominformation(); + afx_msg void OnUpdateFileRominformation(CCmdUI* pCmdUI); + afx_msg void OnFileTogglemenu(); + afx_msg void OnUpdateFileTogglemenu(CCmdUI* pCmdUI); + afx_msg void OnUpdateOptionsFrameskipThrottleNothrottle(CCmdUI* pCmdUI); + afx_msg void OnUpdateOptionsFrameskipThrottle25(CCmdUI* pCmdUI); + afx_msg void OnUpdateOptionsFrameskipThrottle50(CCmdUI* pCmdUI); + afx_msg void OnUpdateOptionsFrameskipThrottle100(CCmdUI* pCmdUI); + afx_msg void OnUpdateOptionsFrameskipThrottle150(CCmdUI* pCmdUI); + afx_msg void OnUpdateOptionsFrameskipThrottle200(CCmdUI* pCmdUI); + afx_msg void OnUpdateOptionsFrameskipThrottleOther(CCmdUI* pCmdUI); + afx_msg void OnOptionsFrameskipThrottleNothrottle(); + afx_msg void OnOptionsFrameskipThrottle25(); + afx_msg void OnOptionsFrameskipThrottle50(); + afx_msg void OnOptionsFrameskipThrottle100(); + afx_msg void OnOptionsFrameskipThrottle150(); + afx_msg void OnOptionsFrameskipThrottle200(); + afx_msg void OnOptionsFrameskipThrottleOther(); + afx_msg void OnOptionsFrameskipAutomatic(); + afx_msg void OnUpdateOptionsFrameskipAutomatic(CCmdUI* pCmdUI); + afx_msg void OnUpdateOptionsVideoFrameskip0(CCmdUI* pCmdUI); + afx_msg void OnUpdateOptionsVideoFrameskip1(CCmdUI* pCmdUI); + afx_msg void OnUpdateOptionsVideoFrameskip2(CCmdUI* pCmdUI); + afx_msg void OnUpdateOptionsVideoFrameskip3(CCmdUI* pCmdUI); + afx_msg void OnUpdateOptionsVideoFrameskip4(CCmdUI* pCmdUI); + afx_msg void OnUpdateOptionsVideoFrameskip5(CCmdUI* pCmdUI); + afx_msg void OnUpdateOptionsVideoFrameskip6(CCmdUI* pCmdUI); + afx_msg void OnUpdateOptionsVideoFrameskip7(CCmdUI* pCmdUI); + afx_msg void OnUpdateOptionsVideoFrameskip8(CCmdUI* pCmdUI); + afx_msg void OnUpdateOptionsVideoFrameskip9(CCmdUI* pCmdUI); + afx_msg void OnOptionsVideoVsync(); + afx_msg void OnUpdateOptionsVideoVsync(CCmdUI* pCmdUI); + afx_msg void OnUpdateOptionsVideoX1(CCmdUI* pCmdUI); + afx_msg void OnUpdateOptionsVideoX2(CCmdUI* pCmdUI); + afx_msg void OnUpdateOptionsVideoX3(CCmdUI* pCmdUI); + afx_msg void OnUpdateOptionsVideoX4(CCmdUI* pCmdUI); + afx_msg void OnUpdateOptionsVideoFullscreen320x240(CCmdUI* pCmdUI); + afx_msg void OnUpdateOptionsVideoFullscreen640x480(CCmdUI* pCmdUI); + afx_msg void OnUpdateOptionsVideoFullscreen800x600(CCmdUI* pCmdUI); + afx_msg void OnOptionsVideoFullscreen320x240(); + afx_msg void OnOptionsVideoFullscreen640x480(); + afx_msg void OnOptionsVideoFullscreen800x600(); + afx_msg void OnOptionsVideoFullscreen(); + afx_msg void OnUpdateOptionsVideoFullscreen(CCmdUI* pCmdUI); + afx_msg void OnMove(int x, int y); + afx_msg void OnSize(UINT nType, int cx, int cy); + afx_msg void OnOptionsVideoDisablesfx(); + afx_msg void OnUpdateOptionsVideoDisablesfx(CCmdUI* pCmdUI); + afx_msg void OnOptionsVideoFullscreenstretchtofit(); + afx_msg void OnUpdateOptionsVideoFullscreenstretchtofit(CCmdUI* pCmdUI); + afx_msg void OnOptionsVideoRendermethodGdi(); + afx_msg void OnUpdateOptionsVideoRendermethodGdi(CCmdUI* pCmdUI); + afx_msg void OnOptionsVideoRendermethodDirectdraw(); + afx_msg void OnUpdateOptionsVideoRendermethodDirectdraw(CCmdUI* pCmdUI); + afx_msg void OnOptionsVideoRendermethodDirect3d(); + afx_msg void OnUpdateOptionsVideoRendermethodDirect3d(CCmdUI* pCmdUI); + afx_msg void OnOptionsVideoRendermethodOpengl(); + afx_msg void OnUpdateOptionsVideoRendermethodOpengl(CCmdUI* pCmdUI); + afx_msg void OnOptionsVideoTriplebuffering(); + afx_msg void OnUpdateOptionsVideoTriplebuffering(CCmdUI* pCmdUI); + afx_msg void OnOptionsVideoDdrawemulationonly(); + afx_msg void OnUpdateOptionsVideoDdrawemulationonly(CCmdUI* pCmdUI); + afx_msg void OnOptionsVideoDdrawusevideomemory(); + afx_msg void OnUpdateOptionsVideoDdrawusevideomemory(CCmdUI* pCmdUI); + afx_msg void OnOptionsVideoRenderoptionsD3dnofilter(); + afx_msg void OnUpdateOptionsVideoRenderoptionsD3dnofilter(CCmdUI* pCmdUI); + afx_msg void OnOptionsVideoRenderoptionsD3dbilinear(); + afx_msg void OnUpdateOptionsVideoRenderoptionsD3dbilinear(CCmdUI* pCmdUI); + afx_msg void OnOptionsVideoRenderoptionsGlnearest(); + afx_msg void OnUpdateOptionsVideoRenderoptionsGlnearest(CCmdUI* pCmdUI); + afx_msg void OnOptionsVideoRenderoptionsGlbilinear(); + afx_msg void OnUpdateOptionsVideoRenderoptionsGlbilinear(CCmdUI* pCmdUI); + afx_msg void OnOptionsVideoRenderoptionsGltriangle(); + afx_msg void OnUpdateOptionsVideoRenderoptionsGltriangle(CCmdUI* pCmdUI); + afx_msg void OnOptionsVideoRenderoptionsGlquads(); + afx_msg void OnUpdateOptionsVideoRenderoptionsGlquads(CCmdUI* pCmdUI); + afx_msg void OnOptionsVideoRenderoptionsSelectskin(); + afx_msg void OnUpdateOptionsVideoRenderoptionsSelectskin(CCmdUI* pCmdUI); + afx_msg void OnOptionsVideoRenderoptionsSkin(); + afx_msg void OnUpdateOptionsVideoRenderoptionsSkin(CCmdUI* pCmdUI); + afx_msg void OnContextMenu(CWnd* pWnd, CPoint point); + afx_msg void OnOptionsEmulatorAssociate(); + afx_msg void OnOptionsEmulatorDirectories(); + afx_msg void OnOptionsEmulatorDisablestatusmessages(); + afx_msg void OnUpdateOptionsEmulatorDisablestatusmessages(CCmdUI* pCmdUI); + afx_msg void OnOptionsEmulatorSynchronize(); + afx_msg void OnUpdateOptionsEmulatorSynchronize(CCmdUI* pCmdUI); + afx_msg void OnOptionsEmulatorPausewheninactive(); + afx_msg void OnUpdateOptionsEmulatorPausewheninactive(CCmdUI* pCmdUI); + afx_msg void OnOptionsEmulatorSpeeduptoggle(); + afx_msg void OnUpdateOptionsEmulatorSpeeduptoggle(CCmdUI* pCmdUI); + afx_msg void OnOptionsEmulatorAutomaticallyipspatch(); + afx_msg void OnUpdateOptionsEmulatorAutomaticallyipspatch(CCmdUI* pCmdUI); + afx_msg void OnOptionsEmulatorAgbprint(); + afx_msg void OnUpdateOptionsEmulatorAgbprint(CCmdUI* pCmdUI); + afx_msg void OnOptionsEmulatorRealtimeclock(); + afx_msg void OnUpdateOptionsEmulatorRealtimeclock(CCmdUI* pCmdUI); + afx_msg void OnOptionsEmulatorGenericflashcard(); + afx_msg void OnUpdateOptionsEmulatorGenericflashcard(CCmdUI* pCmdUI); + afx_msg void OnOptionsEmulatorAutohidemenu(); + afx_msg void OnUpdateOptionsEmulatorAutohidemenu(CCmdUI* pCmdUI); + afx_msg void OnOptionsEmulatorRewindinterval(); + afx_msg void OnOptionsEmulatorSavetypeAutomatic(); + afx_msg void OnUpdateOptionsEmulatorSavetypeAutomatic(CCmdUI* pCmdUI); + afx_msg void OnOptionsEmulatorSavetypeEeprom(); + afx_msg void OnUpdateOptionsEmulatorSavetypeEeprom(CCmdUI* pCmdUI); + afx_msg void OnOptionsEmulatorSavetypeSram(); + afx_msg void OnUpdateOptionsEmulatorSavetypeSram(CCmdUI* pCmdUI); + afx_msg void OnOptionsEmulatorSavetypeFlash(); + afx_msg void OnUpdateOptionsEmulatorSavetypeFlash(CCmdUI* pCmdUI); + afx_msg void OnOptionsEmulatorSavetypeEepromsensor(); + afx_msg void OnUpdateOptionsEmulatorSavetypeEepromsensor(CCmdUI* pCmdUI); + afx_msg void OnOptionsEmulatorSavetypeNone(); + afx_msg void OnUpdateOptionsEmulatorSavetypeNone(CCmdUI* pCmdUI); + afx_msg void OnOptionsEmulatorSavetypeFlash512k(); + afx_msg void OnUpdateOptionsEmulatorSavetypeFlash512k(CCmdUI* pCmdUI); + afx_msg void OnOptionsEmulatorSavetypeFlash1m(); + afx_msg void OnUpdateOptionsEmulatorSavetypeFlash1m(CCmdUI* pCmdUI); + afx_msg void OnOptionsEmulatorUsebiosfile(); + afx_msg void OnUpdateOptionsEmulatorUsebiosfile(CCmdUI* pCmdUI); + afx_msg void OnOptionsEmulatorSkipbios(); + afx_msg void OnUpdateOptionsEmulatorSkipbios(CCmdUI* pCmdUI); + afx_msg void OnOptionsEmulatorSelectbiosfile(); + afx_msg void OnOptionsEmulatorPngformat(); + afx_msg void OnUpdateOptionsEmulatorPngformat(CCmdUI* pCmdUI); + afx_msg void OnOptionsEmulatorBmpformat(); + afx_msg void OnUpdateOptionsEmulatorBmpformat(CCmdUI* pCmdUI); + afx_msg void OnOptionsSoundOff(); + afx_msg void OnUpdateOptionsSoundOff(CCmdUI* pCmdUI); + afx_msg void OnOptionsSoundMute(); + afx_msg void OnUpdateOptionsSoundMute(CCmdUI* pCmdUI); + afx_msg void OnOptionsSoundOn(); + afx_msg void OnUpdateOptionsSoundOn(CCmdUI* pCmdUI); + afx_msg void OnOptionsSoundUseoldsynchronization(); + afx_msg void OnUpdateOptionsSoundUseoldsynchronization(CCmdUI* pCmdUI); + afx_msg void OnOptionsSoundEcho(); + afx_msg void OnUpdateOptionsSoundEcho(CCmdUI* pCmdUI); + afx_msg void OnOptionsSoundLowpassfilter(); + afx_msg void OnUpdateOptionsSoundLowpassfilter(CCmdUI* pCmdUI); + afx_msg void OnOptionsSoundReversestereo(); + afx_msg void OnUpdateOptionsSoundReversestereo(CCmdUI* pCmdUI); + afx_msg void OnOptionsSound11khz(); + afx_msg void OnUpdateOptionsSound11khz(CCmdUI* pCmdUI); + afx_msg void OnOptionsSound22khz(); + afx_msg void OnUpdateOptionsSound22khz(CCmdUI* pCmdUI); + afx_msg void OnOptionsSound44khz(); + afx_msg void OnUpdateOptionsSound44khz(CCmdUI* pCmdUI); + afx_msg void OnOptionsSoundChannel1(); + afx_msg void OnUpdateOptionsSoundChannel1(CCmdUI* pCmdUI); + afx_msg void OnOptionsSoundChannel2(); + afx_msg void OnUpdateOptionsSoundChannel2(CCmdUI* pCmdUI); + afx_msg void OnOptionsSoundChannel3(); + afx_msg void OnUpdateOptionsSoundChannel3(CCmdUI* pCmdUI); + afx_msg void OnOptionsSoundChannel4(); + afx_msg void OnUpdateOptionsSoundChannel4(CCmdUI* pCmdUI); + afx_msg void OnOptionsSoundDirectsounda(); + afx_msg void OnUpdateOptionsSoundDirectsounda(CCmdUI* pCmdUI); + afx_msg void OnOptionsSoundDirectsoundb(); + afx_msg void OnUpdateOptionsSoundDirectsoundb(CCmdUI* pCmdUI); + afx_msg void OnOptionsGameboyBorder(); + afx_msg void OnUpdateOptionsGameboyBorder(CCmdUI* pCmdUI); + afx_msg void OnOptionsGameboyPrinter(); + afx_msg void OnUpdateOptionsGameboyPrinter(CCmdUI* pCmdUI); + afx_msg void OnOptionsGameboyBorderAutomatic(); + afx_msg void OnUpdateOptionsGameboyBorderAutomatic(CCmdUI* pCmdUI); + afx_msg void OnOptionsGameboyAutomatic(); + afx_msg void OnUpdateOptionsGameboyAutomatic(CCmdUI* pCmdUI); + afx_msg void OnOptionsGameboyGba(); + afx_msg void OnUpdateOptionsGameboyGba(CCmdUI* pCmdUI); + afx_msg void OnOptionsGameboyCgb(); + afx_msg void OnUpdateOptionsGameboyCgb(CCmdUI* pCmdUI); + afx_msg void OnOptionsGameboySgb(); + afx_msg void OnUpdateOptionsGameboySgb(CCmdUI* pCmdUI); + afx_msg void OnOptionsGameboySgb2(); + afx_msg void OnUpdateOptionsGameboySgb2(CCmdUI* pCmdUI); + afx_msg void OnOptionsGameboyGb(); + afx_msg void OnUpdateOptionsGameboyGb(CCmdUI* pCmdUI); + afx_msg void OnOptionsGameboyRealcolors(); + afx_msg void OnUpdateOptionsGameboyRealcolors(CCmdUI* pCmdUI); + afx_msg void OnOptionsGameboyGameboycolors(); + afx_msg void OnUpdateOptionsGameboyGameboycolors(CCmdUI* pCmdUI); + afx_msg void OnOptionsGameboyColors(); + afx_msg void OnOptionsFilterDisablemmx(); + afx_msg void OnUpdateOptionsFilterDisablemmx(CCmdUI* pCmdUI); + afx_msg void OnOptionsLanguageSystem(); + afx_msg void OnUpdateOptionsLanguageSystem(CCmdUI* pCmdUI); + afx_msg void OnOptionsLanguageEnglish(); + afx_msg void OnUpdateOptionsLanguageEnglish(CCmdUI* pCmdUI); + afx_msg void OnOptionsLanguageOther(); + afx_msg void OnUpdateOptionsLanguageOther(CCmdUI* pCmdUI); + afx_msg void OnOptionsJoypadConfigure1(); + afx_msg void OnUpdateOptionsJoypadConfigure1(CCmdUI* pCmdUI); + afx_msg void OnOptionsJoypadConfigure2(); + afx_msg void OnUpdateOptionsJoypadConfigure2(CCmdUI* pCmdUI); + afx_msg void OnOptionsJoypadConfigure3(); + afx_msg void OnUpdateOptionsJoypadConfigure3(CCmdUI* pCmdUI); + afx_msg void OnOptionsJoypadConfigure4(); + afx_msg void OnUpdateOptionsJoypadConfigure4(CCmdUI* pCmdUI); + afx_msg void OnOptionsJoypadMotionconfigure(); + afx_msg void OnUpdateOptionsJoypadMotionconfigure(CCmdUI* pCmdUI); + afx_msg void OnCheatsSearchforcheats(); + afx_msg void OnUpdateCheatsSearchforcheats(CCmdUI* pCmdUI); + afx_msg void OnCheatsCheatlist(); + afx_msg void OnUpdateCheatsCheatlist(CCmdUI* pCmdUI); + afx_msg void OnCheatsAutomaticsaveloadcheats(); + afx_msg void OnCheatsLoadcheatlist(); + afx_msg void OnUpdateCheatsLoadcheatlist(CCmdUI* pCmdUI); + afx_msg void OnCheatsSavecheatlist(); + afx_msg void OnUpdateCheatsSavecheatlist(CCmdUI* pCmdUI); + afx_msg void OnToolsDisassemble(); + afx_msg void OnUpdateToolsDisassemble(CCmdUI* pCmdUI); + afx_msg void OnToolsLogging(); + afx_msg void OnUpdateToolsLogging(CCmdUI* pCmdUI); + afx_msg void OnToolsIoviewer(); + afx_msg void OnUpdateToolsIoviewer(CCmdUI* pCmdUI); + afx_msg void OnToolsMapview(); + afx_msg void OnUpdateToolsMapview(CCmdUI* pCmdUI); + afx_msg void OnToolsMemoryviewer(); + afx_msg void OnUpdateToolsMemoryviewer(CCmdUI* pCmdUI); + afx_msg void OnToolsOamviewer(); + afx_msg void OnUpdateToolsOamviewer(CCmdUI* pCmdUI); + afx_msg void OnToolsPaletteview(); + afx_msg void OnUpdateToolsPaletteview(CCmdUI* pCmdUI); + afx_msg void OnToolsTileviewer(); + afx_msg void OnUpdateToolsTileviewer(CCmdUI* pCmdUI); + afx_msg void OnDebugNextframe(); + afx_msg void OnUpdateCheatsAutomaticsaveloadcheats(CCmdUI* pCmdUI); + afx_msg void OnToolsDebugGdb(); + afx_msg void OnUpdateToolsDebugGdb(CCmdUI* pCmdUI); + afx_msg void OnToolsDebugLoadandwait(); + afx_msg void OnUpdateToolsDebugLoadandwait(CCmdUI* pCmdUI); + afx_msg void OnToolsDebugBreak(); + afx_msg void OnUpdateToolsDebugBreak(CCmdUI* pCmdUI); + afx_msg void OnToolsDebugDisconnect(); + afx_msg void OnUpdateToolsDebugDisconnect(CCmdUI* pCmdUI); + afx_msg void OnOptionsSoundStartrecording(); + afx_msg void OnUpdateOptionsSoundStartrecording(CCmdUI* pCmdUI); + afx_msg void OnOptionsSoundStoprecording(); + afx_msg void OnUpdateOptionsSoundStoprecording(CCmdUI* pCmdUI); + afx_msg void OnToolsRecordStartavirecording(); + afx_msg void OnUpdateToolsRecordStartavirecording(CCmdUI* pCmdUI); + afx_msg void OnToolsRecordStopavirecording(); + afx_msg void OnUpdateToolsRecordStopavirecording(CCmdUI* pCmdUI); + afx_msg void OnPaint(); + afx_msg void OnToolsRecordStartmovierecording(); + afx_msg void OnUpdateToolsRecordStartmovierecording(CCmdUI* pCmdUI); + afx_msg void OnToolsRecordStopmovierecording(); + afx_msg void OnUpdateToolsRecordStopmovierecording(CCmdUI* pCmdUI); + afx_msg void OnToolsPlayStartmovieplaying(); + afx_msg void OnUpdateToolsPlayStartmovieplaying(CCmdUI* pCmdUI); + afx_msg void OnToolsPlayStopmovieplaying(); + afx_msg void OnUpdateToolsPlayStopmovieplaying(CCmdUI* pCmdUI); + afx_msg void OnToolsRewind(); + afx_msg void OnUpdateToolsRewind(CCmdUI* pCmdUI); + afx_msg void OnToolsCustomize(); + afx_msg void OnUpdateToolsCustomize(CCmdUI* pCmdUI); + afx_msg void OnHelpBugreport(); + afx_msg void OnMouseMove(UINT nFlags, CPoint point); + afx_msg void OnInitMenu(CMenu* pMenu); + afx_msg void OnActivate(UINT nState, CWnd* pWndOther, BOOL bMinimized); +#if _MSC_VER <= 1200 + afx_msg void OnActivateApp(BOOL bActive, HTASK hTask); +#else + afx_msg void OnActivateApp(BOOL bActive, DWORD hTask); +#endif + afx_msg void OnDropFiles(HDROP hDropInfo); + afx_msg void OnFileSavegameOldestslot(); + afx_msg void OnUpdateFileSavegameOldestslot(CCmdUI* pCmdUI); + afx_msg void OnFileLoadgameMostrecent(); + afx_msg void OnUpdateFileLoadgameMostrecent(CCmdUI* pCmdUI); + afx_msg void OnFileLoadgameAutoloadmostrecent(); + afx_msg void OnUpdateFileLoadgameAutoloadmostrecent(CCmdUI* pCmdUI); + afx_msg void OnOptionsSoundVolume25x(); + afx_msg void OnUpdateOptionsSoundVolume25x(CCmdUI* pCmdUI); + afx_msg void OnOptionsSoundVolume5x(); + afx_msg void OnUpdateOptionsSoundVolume5x(CCmdUI* pCmdUI); + afx_msg void OnCheatsDisablecheats(); + afx_msg void OnUpdateCheatsDisablecheats(CCmdUI* pCmdUI); + afx_msg void OnOptionsVideoFullscreenmaxscale(); + afx_msg void OnOptionsEmulatorGameoverrides(); + afx_msg void OnUpdateOptionsEmulatorGameoverrides(CCmdUI* pCmdUI); + afx_msg void OnHelpGnupubliclicense(); + //}}AFX_MSG + DECLARE_MESSAGE_MAP() + + afx_msg BOOL OnFileRecentFile(UINT nID); + afx_msg BOOL OnFileLoadSlot(UINT nID); + afx_msg BOOL OnFileSaveSlot(UINT nID); + afx_msg void OnOptionsFilterLcdcolors(); + afx_msg void OnUpdateOptionsFilterLcdcolors(CCmdUI *pCmdUI); + + afx_msg BOOL OnOptionsSoundPcminterpolation(UINT nID); + afx_msg void OnUpdateOptionsSoundPcminterpolation(CCmdUI *pCmdUI); +public: + afx_msg void OnOptionsSoundHardwareacceleration(); + afx_msg void OnUpdateOptionsSoundHardwareacceleration(CCmdUI *pCmdUI); + afx_msg void OnOptionsVideoFullscreen1280x1024(); + afx_msg void OnOptionsVideoFullscreen1024x768(); + afx_msg void OnUpdateOptionsVideoFullscreen1024x768(CCmdUI *pCmdUI); + afx_msg void OnUpdateOptionsVideoFullscreen1280x1024(CCmdUI *pCmdUI); + void OnLinkOptions(); + void OnOptionsLinkLog() ; + void OnUpdateOptionsLinkLog(CCmdUI* pCmdUI) ; + void OnOptionsLinkRFU() ; + void OnUpdateOptionsLinkRFU(CCmdUI* pCmdUI) ; + +}; + +///////////////////////////////////////////////////////////////////////////// + +//{{AFX_INSERT_LOCATION}} +// Microsoft Visual C++ will insert additional declarations immediately before the previous line. + +#endif // !defined(AFX_MAINWND_H__E8AD28B9_C9FB_4EC2_A2DC_DD1BBA55A275__INCLUDED_) diff --git a/src/win32/MainWndFile.cpp b/src/win32/MainWndFile.cpp index 1322ee5b..a5c0e6e4 100644 --- a/src/win32/MainWndFile.cpp +++ b/src/win32/MainWndFile.cpp @@ -1,900 +1,1005 @@ -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#include "stdafx.h" -#include "MainWnd.h" - -#include "ExportGSASnapshot.h" -#include "FileDlg.h" -#include "GSACodeSelect.h" -#include "RomInfo.h" -#include "Reg.h" -#include "WinResUtil.h" - -#include "../GBA.h" -#include "../Globals.h" -#include "../NLS.h" -#include "../Sound.h" -#include "../gb/GB.h" -#include "../gb/gbCheats.h" -#include "../gb/gbGlobals.h" -#include "../Link.h" - -extern int emulating; - -extern void remoteCleanUp(); - -void MainWnd::OnFileOpen() -{ - theApp.winCheckFullscreen(); - if(fileOpenSelect()) { - FileRun(); - } -} - -void MainWnd::OnFilePause() -{ - theApp.paused = !theApp.paused; - if(emulating) { - if(theApp.paused) { - theApp.wasPaused = true; - soundPause(); - } else { - soundResume(); - } - } -} - -void MainWnd::OnUpdateFilePause(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.paused); -} - -void MainWnd::OnFileReset() -{ - if(emulating) { - theApp.emulator.emuReset(); - systemScreenMessage(winResLoadString(IDS_RESET)); - } -} - -void MainWnd::OnUpdateFileReset(CCmdUI* pCmdUI) -{ - pCmdUI->Enable(emulating); -} - -void MainWnd::OnUpdateFileRecentFreeze(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.recentFreeze); - - if(pCmdUI->m_pMenu == NULL) - return; - - CMenu *pMenu = pCmdUI->m_pMenu; - - int i; - for(i = 0; i < 10; i++) { - if(!pMenu->RemoveMenu(ID_FILE_MRU_FILE1+i, MF_BYCOMMAND)) - break; - } - - for(i = 0; i < 10; i++) { - CString p = theApp.recentFiles[i]; - if(p.GetLength() == 0) - break; - int index = p.ReverseFind('\\'); - - if(index != -1) - p = p.Right(p.GetLength()-index-1); - - pMenu->AppendMenu(MF_STRING, ID_FILE_MRU_FILE1+i, p); - } - theApp.winAccelMgr.UpdateMenu((HMENU)*pMenu); -} - -BOOL MainWnd::OnFileRecentFile(UINT nID) -{ - if(theApp.recentFiles[(nID&0xFFFF)-ID_FILE_MRU_FILE1].GetLength()) { - theApp.szFile = theApp.recentFiles[(nID&0xFFFF)-ID_FILE_MRU_FILE1]; - if(FileRun()) - emulating = true; - else { - emulating = false; - soundPause(); - } - } - return TRUE; -} - -void MainWnd::OnFileRecentReset() -{ - int i = 0; - for(i = 0; i < 10; i++) - theApp.recentFiles[i] = ""; -} - -void MainWnd::OnFileRecentFreeze() -{ - theApp.recentFreeze = !theApp.recentFreeze; -} - -void MainWnd::OnFileExit() -{ - SendMessage(WM_CLOSE); -} - -void MainWnd::OnFileClose() -{ - // save battery file before we change the filename... - if(rom != NULL || gbRom != NULL) { - if(theApp.autoSaveLoadCheatList) - winSaveCheatListDefault(); - writeBatteryFile(); - soundPause(); - theApp.emulator.emuCleanUp(); - remoteCleanUp(); - } - emulating = 0; - RedrawWindow(NULL,NULL,RDW_INVALIDATE|RDW_ERASE|RDW_ALLCHILDREN); - systemSetTitle("VisualBoyAdvance"); -} - -void MainWnd::OnUpdateFileClose(CCmdUI* pCmdUI) -{ - pCmdUI->Enable(emulating); -} - -void MainWnd::OnFileOpengameboy() -{ - theApp.winCheckFullscreen(); - theApp.dir = ""; - CString initialDir = regQueryStringValue("gbromdir","."); - if(!initialDir.IsEmpty()) - theApp.dir = initialDir; - - theApp.szFile = ""; - - LPCTSTR exts[] = { "" }; - CString filter = winLoadFilter(IDS_FILTER_GBROM); - CString title = winResLoadString(IDS_SELECT_ROM); - - FileDlg dlg(this, "", filter, 0, "", exts, initialDir, title, false); - - if(dlg.DoModal() == IDOK) { - theApp.szFile = dlg.GetPathName(); - theApp.dir = theApp.szFile.Left(dlg.m_ofn.nFileOffset); - if(theApp.dir.GetLength() > 3 && theApp.dir[theApp.dir.GetLength()-1] == '\\') - theApp.dir = theApp.dir.Left(theApp.dir.GetLength()-1); - regSetStringValue("gbromdir", theApp.dir); - FileRun(); - } -} - -void MainWnd::OnFileLoad() -{ - theApp.winCheckFullscreen(); - CString buffer; - CString filename; - - int index = theApp.filename.ReverseFind('\\'); - - if(index != -1) - buffer = theApp.filename.Right(theApp.filename.GetLength()-index-1); - else - buffer = theApp.filename; - - CString saveDir = regQueryStringValue("saveDir", NULL); - - if(saveDir.IsEmpty()) - saveDir = getDirFromFile(theApp.filename); - - if(isDriveRoot(saveDir)) - filename.Format("%s%s.sgm", saveDir, buffer); - else - filename.Format("%s\\%s.sgm", saveDir, buffer); - - LPCTSTR exts[] = { ".sgm" }; - CString filter = winLoadFilter(IDS_FILTER_SGM); - CString title = winResLoadString(IDS_SELECT_SAVE_GAME_NAME); - - FileDlg dlg(this, filename, filter, 0, "", exts, saveDir, title, false); - - if(dlg.DoModal() == IDOK) { - bool res = loadSaveGame(dlg.GetPathName()); - - theApp.rewindCount = 0; - theApp.rewindCounter = 0; - theApp.rewindSaveNeeded = false; - - if(res) - systemScreenMessage(winResLoadString(IDS_LOADED_STATE)); - } -} - -void MainWnd::OnUpdateFileLoad(CCmdUI* pCmdUI) -{ - pCmdUI->Enable(emulating); -} - -BOOL MainWnd::OnFileLoadSlot(UINT nID) -{ - nID = nID + 1 - ID_FILE_LOADGAME_SLOT1; - - CString buffer; - CString filename; - - int index = theApp.filename.ReverseFind('\\'); - - if(index != -1) - buffer = theApp.filename.Right(theApp.filename.GetLength()-index-1); - else - buffer = theApp.filename; - - CString saveDir = regQueryStringValue("saveDir", NULL); - - if(saveDir.IsEmpty()) - saveDir = getDirFromFile(theApp.filename); - - if(isDriveRoot(saveDir)) - filename.Format("%s%s%d.sgm", saveDir, buffer, nID); - else - filename.Format("%s\\%s%d.sgm", saveDir, buffer, nID); - - bool res = loadSaveGame(filename); - - theApp.rewindCount = 0; - theApp.rewindCounter = 0; - theApp.rewindSaveNeeded = false; - - CString format = winResLoadString(IDS_LOADED_STATE_N); - buffer.Format(format, nID); - - systemScreenMessage(buffer); - - return res; -} - -void MainWnd::OnFileSave() -{ - theApp.winCheckFullscreen(); - CString buffer; - CString filename; - - int index = theApp.filename.ReverseFind('\\'); - - if(index != -1) - buffer = theApp.filename.Right(theApp.filename.GetLength()-index-1); - else - buffer = theApp.filename; - - CString saveDir = regQueryStringValue("saveDir", NULL); - - if(saveDir.IsEmpty()) - saveDir = getDirFromFile(theApp.filename); - - if(isDriveRoot(saveDir)) - filename.Format("%s%s.sgm", saveDir, buffer); - else - filename.Format("%s\\%s.sgm", saveDir, buffer); - - LPCTSTR exts[] = { ".sgm" }; - CString filter = winLoadFilter(IDS_FILTER_SGM); - CString title = winResLoadString(IDS_SELECT_SAVE_GAME_NAME); - - FileDlg dlg(this, filename, filter, 0, "", exts, saveDir, title, true); - - if(dlg.DoModal() == IDOK) { - bool res = writeSaveGame(dlg.GetPathName()); - if(res) - systemScreenMessage(winResLoadString(IDS_WROTE_STATE)); - } -} - -void MainWnd::OnUpdateFileSave(CCmdUI* pCmdUI) -{ - pCmdUI->Enable(emulating); -} - -BOOL MainWnd::OnFileSaveSlot(UINT nID) -{ - nID = nID + 1 - ID_FILE_SAVEGAME_SLOT1; - - CString buffer; - CString filename; - - int index = theApp.filename.ReverseFind('\\'); - - if(index != -1) - buffer = theApp.filename.Right(theApp.filename.GetLength()-index-1); - else - buffer = theApp.filename; - - CString saveDir = regQueryStringValue("saveDir", NULL); - - if(saveDir.IsEmpty()) - saveDir = getDirFromFile(theApp.filename); - - if(isDriveRoot(saveDir)) - filename.Format("%s%s%d.sgm", saveDir, buffer, nID); - else - filename.Format("%s\\%s%d.sgm", saveDir, buffer, nID); - - bool res = writeSaveGame(filename); - - CString format = winResLoadString(IDS_WROTE_STATE_N); - buffer.Format(format, nID); - - systemScreenMessage(buffer); - - return res; -} - -void MainWnd::OnFileImportBatteryfile() -{ - theApp.winCheckFullscreen(); - LPCTSTR exts[] = { ".sav" }; - CString filter = winLoadFilter(IDS_FILTER_SAV); - CString title = winResLoadString(IDS_SELECT_BATTERY_FILE); - - CString saveDir = regQueryStringValue("batteryDir", NULL); - - if(saveDir.IsEmpty()) - saveDir = getDirFromFile(theApp.filename); - - FileDlg dlg(this, "", filter, 0, "", exts, saveDir, title, false); - - if(dlg.DoModal() == IDCANCEL) - return; - - CString str1 = winResLoadString(IDS_SAVE_WILL_BE_LOST); - CString str2 = winResLoadString(IDS_CONFIRM_ACTION); - - if(MessageBox(str1, - str2, - MB_OKCANCEL) == IDCANCEL) - return; - - bool res = false; - - res = theApp.emulator.emuReadBattery(dlg.GetPathName()); - - if(!res) - systemMessage(MSG_CANNOT_OPEN_FILE, "Cannot open file %s", dlg.GetPathName()); - else { - theApp.emulator.emuReset(); - } -} - -void MainWnd::OnUpdateFileImportBatteryfile(CCmdUI* pCmdUI) -{ - pCmdUI->Enable(emulating); -} - -void MainWnd::OnFileImportGamesharkcodefile() -{ - theApp.winCheckFullscreen(); - LPCTSTR exts[] = { "" }; - CString filter = theApp.cartridgeType == 0 ? winLoadFilter(IDS_FILTER_SPC) : winLoadFilter(IDS_FILTER_GCF); - CString title = winResLoadString(IDS_SELECT_CODE_FILE); - - FileDlg dlg(this, "", filter, 0, "", exts, "", title, false); - - if(dlg.DoModal() == IDCANCEL) - return; - - CString str1 = winResLoadString(IDS_CODES_WILL_BE_LOST); - CString str2 = winResLoadString(IDS_CONFIRM_ACTION); - - if(MessageBox(str1, - str2, - MB_OKCANCEL) == IDCANCEL) - return; - - bool res = false; - CString file = dlg.GetPathName(); - if(theApp.cartridgeType == 1) - res = gbCheatReadGSCodeFile(file); - else { - res = fileImportGSACodeFile(file); - } -} - -void MainWnd::OnUpdateFileImportGamesharkcodefile(CCmdUI* pCmdUI) -{ - pCmdUI->Enable(emulating); -} - -void MainWnd::OnFileImportGamesharksnapshot() -{ - theApp.winCheckFullscreen(); - LPCTSTR exts[] = { "" }; - CString filter = theApp.cartridgeType == 1 ? winLoadFilter(IDS_FILTER_GBS) : winLoadFilter(IDS_FILTER_SPS); - CString title = winResLoadString(IDS_SELECT_SNAPSHOT_FILE); - - FileDlg dlg(this, "", filter, 0, "", exts, "", title, false); - - if(dlg.DoModal() == IDCANCEL) - return; - - CString str1 = winResLoadString(IDS_SAVE_WILL_BE_LOST); - CString str2 = winResLoadString(IDS_CONFIRM_ACTION); - - if(MessageBox(str1, - str2, - MB_OKCANCEL) == IDCANCEL) - return; - - if(theApp.cartridgeType == 1) - gbReadGSASnapshot(dlg.GetPathName()); - else - CPUReadGSASnapshot(dlg.GetPathName()); -} - -void MainWnd::OnUpdateFileImportGamesharksnapshot(CCmdUI* pCmdUI) -{ - pCmdUI->Enable(emulating); -} - -void MainWnd::OnFileExportBatteryfile() -{ - theApp.winCheckFullscreen(); - CString name; - - int index = theApp.filename.ReverseFind('\\'); - - if(index != -1) - name = theApp.filename.Right(theApp.filename.GetLength()-index-1); - else - name = theApp.filename; - - LPCTSTR exts[] = {".sav", ".dat" }; - - CString filter = winLoadFilter(IDS_FILTER_SAV); - CString title = winResLoadString(IDS_SELECT_BATTERY_FILE); - - CString saveDir = regQueryStringValue("batteryDir", NULL); - - if(saveDir.IsEmpty()) - saveDir = getDirFromFile(theApp.filename); - - FileDlg dlg(this, - name, - filter, - 1, - "SAV", - exts, - saveDir, - title, - true); - - if(dlg.DoModal() == IDCANCEL) { - return; - } - - bool result = false; - - if(theApp.cartridgeType == 1) { - result = gbWriteBatteryFile(dlg.GetPathName(), false); - } else - result = theApp.emulator.emuWriteBattery(dlg.GetPathName()); - - if(!result) - systemMessage(MSG_ERROR_CREATING_FILE, "Error creating file %s", - dlg.GetPathName()); -} - -void MainWnd::OnUpdateFileExportBatteryfile(CCmdUI* pCmdUI) -{ - pCmdUI->Enable(emulating); -} - -void MainWnd::OnFileExportGamesharksnapshot() -{ - theApp.winCheckFullscreen(); - if(eepromInUse) { - systemMessage(IDS_EEPROM_NOT_SUPPORTED, "EEPROM saves cannot be exported"); - return; - } - - CString name; - - int index = theApp.filename.ReverseFind('\\'); - - if(index != -1) - name = theApp.filename.Right(theApp.filename.GetLength()-index-1); - else - name = theApp.filename; - - LPCTSTR exts[] = {".sps" }; - - CString filter = winLoadFilter(IDS_FILTER_SPS); - CString title = winResLoadString(IDS_SELECT_SNAPSHOT_FILE); - - FileDlg dlg(this, - name, - filter, - 1, - "SPS", - exts, - "", - title, - true); - - if(dlg.DoModal() == IDCANCEL) - return; - - char buffer[16]; - strncpy(buffer, (const char *)&rom[0xa0], 12); - buffer[12] = 0; - - ExportGSASnapshot dlg2(dlg.GetPathName(), buffer, this); - dlg2.DoModal(); -} - -void MainWnd::OnUpdateFileExportGamesharksnapshot(CCmdUI* pCmdUI) -{ - pCmdUI->Enable(emulating && theApp.cartridgeType == 0); -} - -void MainWnd::OnFileScreencapture() -{ - theApp.winCheckFullscreen(); - CString name; - CString filename; - - int index = theApp.filename.ReverseFind('\\'); - - if(index != -1) - name = theApp.filename.Right(theApp.filename.GetLength()-index-1); - else - name = theApp.filename; - - CString capdir = regQueryStringValue("captureDir", ""); - if(capdir.IsEmpty()) - capdir = getDirFromFile(name); - - CString ext = "png"; - - if(theApp.captureFormat != 0) - ext = "bmp"; - - if(isDriveRoot(capdir)) - filename.Format("%s%s.%s", capdir, name, ext); - else - filename.Format("%s\\%s.%s", capdir, name, ext); - - LPCTSTR exts[] = {".png", ".bmp" }; - - CString filter = winLoadFilter(IDS_FILTER_PNG); - CString title = winResLoadString(IDS_SELECT_CAPTURE_NAME); - - FileDlg dlg(this, - filename, - filter, - theApp.captureFormat ? 2 : 1, - theApp.captureFormat ? "BMP" : "PNG", - exts, - capdir, - title, - true); - - if(dlg.DoModal() == IDCANCEL) - return; - - if(dlg.getFilterIndex() == 2) - theApp.emulator.emuWriteBMP(dlg.GetPathName()); - else - theApp.emulator.emuWritePNG(dlg.GetPathName()); - - systemScreenMessage(winResLoadString(IDS_SCREEN_CAPTURE)); -} - -void MainWnd::OnUpdateFileScreencapture(CCmdUI* pCmdUI) -{ - pCmdUI->Enable(emulating); -} - -void MainWnd::OnFileRominformation() -{ - theApp.winCheckFullscreen(); - if(theApp.cartridgeType == 0) { - RomInfoGBA dlg(rom); - dlg.DoModal(); - } else { - RomInfoGB dlg(gbRom); - dlg.DoModal(); - } -} - -void MainWnd::OnUpdateFileRominformation(CCmdUI* pCmdUI) -{ - pCmdUI->Enable(emulating); -} - -void MainWnd::OnFileTogglemenu() -{ - if(theApp.videoOption <= VIDEO_4X) - return; - - theApp.menuToggle = !theApp.menuToggle; - - if(theApp.menuToggle) { - theApp.updateMenuBar(); - if(theApp.tripleBuffering) { - if(theApp.display) - theApp.display->checkFullScreen(); - DrawMenuBar(); - } - } else { - SetMenu(NULL); - DestroyMenu(theApp.menu); - } - - theApp.adjustDestRect(); - if(theApp.display) - theApp.display->resize(theApp.dest.right-theApp.dest.left, theApp.dest.bottom-theApp.dest.top); - -} - -void MainWnd::OnUpdateFileTogglemenu(CCmdUI* pCmdUI) -{ - pCmdUI->Enable(theApp.videoOption > VIDEO_4X); -} - -bool MainWnd::fileImportGSACodeFile(CString& fileName) -{ - FILE *f = fopen(fileName, "rb"); - - if(f == NULL) { - systemMessage(MSG_CANNOT_OPEN_FILE, "Cannot open file %s", fileName); - return false; - } - - u32 len; - fread(&len, 1, 4, f); - if(len != 14) { - fclose(f); - systemMessage(MSG_UNSUPPORTED_CODE_FILE, "Unsupported code file %s", - fileName); - return false; - } - char buffer[16]; - fread(buffer, 1, 14, f); - buffer[14] = 0; - if(memcmp(buffer, "SharkPortCODES", 14)) { - fclose(f); - systemMessage(MSG_UNSUPPORTED_CODE_FILE, "Unsupported code file %s", - fileName); - return false; - } - fseek(f, 0x1e, SEEK_SET); - fread(&len, 1, 4, f); - int game = 0; - if(len > 1) { - GSACodeSelect dlg(f); - game = dlg.DoModal(); - } - fclose(f); - - bool v3 = false; - - int index = fileName.ReverseFind('.'); - - if(index != -1) { - if(fileName.Right(3).CompareNoCase("XPC") == 0) - v3 = true; - } - - if(game != -1) { - return cheatsImportGSACodeFile(fileName, game, v3); - } - - return true; -} - -void MainWnd::OnFileSavegameOldestslot() -{ - if(!emulating) - return; - - CString filename; - - int index = theApp.filename.ReverseFind('\\'); - - if(index != -1) - filename = theApp.filename.Right(theApp.filename.GetLength()-index-1); - else - filename = theApp.filename; - - CString saveDir = regQueryStringValue("saveDir", NULL); - - if(saveDir.IsEmpty()) - saveDir = getDirFromFile(theApp.filename); - - if(!isDriveRoot(saveDir)) - saveDir += "\\"; - - CString name; - CFileStatus status; - CString str; - unsigned long time = (unsigned long)-1; - int found = 0; - - for(int i = 0; i < 10; i++) { - name.Format("%s%s%d.sgm", saveDir, filename, i+1); - - if(emulating && CFile::GetStatus(name, status)) { - if((unsigned long)status.m_mtime.GetTime() < time) { - time = (time_t)status.m_mtime.GetTime(); - found = i; - } - } else { - found = i; - break; - } - } - OnFileSaveSlot(ID_FILE_SAVEGAME_SLOT1+found); -} - -void MainWnd::OnUpdateFileSavegameOldestslot(CCmdUI* pCmdUI) -{ - pCmdUI->Enable(emulating); - if(pCmdUI->m_pSubMenu != NULL) { - CMenu *pMenu = pCmdUI->m_pSubMenu; - CString filename; - - int index = theApp.filename.ReverseFind('\\'); - - if(index != -1) - filename = theApp.filename.Right(theApp.filename.GetLength()-index-1); - else - filename = theApp.filename; - - CString saveDir = regQueryStringValue("saveDir", NULL); - - if(saveDir.IsEmpty()) - saveDir = getDirFromFile(theApp.filename); - - if(!isDriveRoot(saveDir)) - saveDir += "\\"; - - CString name; - CFileStatus status; - CString str; - - for(int i = 0; i < 10; i++) { - name.Format("%s%s%d.sgm", saveDir, filename, i+1); - - if(emulating && CFile::GetStatus(name, status)) { - CString timestamp = status.m_mtime.Format("%Y/%m/%d %H:%M:%S"); - str.Format("%d %s", i+1, timestamp); - } else { - str.Format("%d ----/--/-- --:--:--", i+1); - } - pMenu->ModifyMenu(ID_FILE_SAVEGAME_SLOT1+i, MF_STRING|MF_BYCOMMAND, ID_FILE_SAVEGAME_SLOT1+i, str); - } - - theApp.winAccelMgr.UpdateMenu(pMenu->GetSafeHmenu()); - } -} - -void MainWnd::OnFileLoadgameMostrecent() -{ - if(!emulating) - return; - - CString filename; - - int index = theApp.filename.ReverseFind('\\'); - - if(index != -1) - filename = theApp.filename.Right(theApp.filename.GetLength()-index-1); - else - filename = theApp.filename; - - CString saveDir = regQueryStringValue("saveDir", NULL); - - if(saveDir.IsEmpty()) - saveDir = getDirFromFile(theApp.filename); - - if(!isDriveRoot(saveDir)) - saveDir += "\\"; - - CString name; - CFileStatus status; - CString str; - unsigned long time = 0; - int found = -1; - - for(int i = 0; i < 10; i++) { - name.Format("%s%s%d.sgm", saveDir, filename, i+1); - - if(emulating && CFile::GetStatus(name, status)) { - if((unsigned long)status.m_mtime.GetTime() > time) { - time = (time_t)status.m_mtime.GetTime(); - found = i; - } - } - } - if(found != -1) { - OnFileLoadSlot(ID_FILE_LOADGAME_SLOT1+found); - } -} - -void MainWnd::OnUpdateFileLoadgameMostrecent(CCmdUI* pCmdUI) -{ - pCmdUI->Enable(emulating); - - if(pCmdUI->m_pSubMenu != NULL) { - CMenu *pMenu = pCmdUI->m_pSubMenu; - CString filename; - - int index = theApp.filename.ReverseFind('\\'); - - if(index != -1) - filename = theApp.filename.Right(theApp.filename.GetLength()-index-1); - else - filename = theApp.filename; - - CString saveDir = regQueryStringValue("saveDir", NULL); - - if(saveDir.IsEmpty()) - saveDir = getDirFromFile(theApp.filename); - - if(!isDriveRoot(saveDir)) - saveDir += "\\"; - - CString name; - CFileStatus status; - CString str; - - for(int i = 0; i < 10; i++) { - name.Format("%s%s%d.sgm", saveDir, filename, i+1); - - if(emulating && CFile::GetStatus(name, status)) { - CString timestamp = status.m_mtime.Format("%Y/%m/%d %H:%M:%S"); - str.Format("%d %s", i+1, timestamp); - } else { - str.Format("%d ----/--/-- --:--:--", i+1); - } - pMenu->ModifyMenu(ID_FILE_LOADGAME_SLOT1+i, MF_STRING|MF_BYCOMMAND, ID_FILE_LOADGAME_SLOT1+i, str); - } - - theApp.winAccelMgr.UpdateMenu(pMenu->GetSafeHmenu()); - } -} - -void MainWnd::OnUpdateFileLoadGameSlot(CCmdUI *pCmdUI) -{ - pCmdUI->Enable(emulating); -} - -void MainWnd::OnUpdateFileSaveGameSlot(CCmdUI *pCmdUI) -{ - pCmdUI->Enable(emulating); -} - -void MainWnd::OnFileLoadgameAutoloadmostrecent() -{ - theApp.autoLoadMostRecent = !theApp.autoLoadMostRecent; -} - -void MainWnd::OnUpdateFileLoadgameAutoloadmostrecent(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.autoLoadMostRecent); -} +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2005-2006 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#include "stdafx.h" +#include "MainWnd.h" + +#include + +#include "ExportGSASnapshot.h" +#include "FileDlg.h" +#include "GSACodeSelect.h" +#include "RomInfo.h" +#include "Reg.h" +#include "WinResUtil.h" + +#include "../GBA.h" +#include "../Globals.h" +#include "../NLS.h" +#include "../Sound.h" +#include "../gb/GB.h" +#include "../gb/gbCheats.h" +#include "../gb/gbGlobals.h" + +extern int emulating; + +extern void remoteCleanUp(); +extern void InterframeCleanup(); + + +void MainWnd::OnFileOpen() +{ + theApp.winCheckFullscreen(); + if( fileOpenSelect( false ) ) { + FileRun(); + } +} + + +void MainWnd::OnFilePause() +{ + theApp.paused = !theApp.paused; + if(emulating) { + if(theApp.paused) { + theApp.wasPaused = true; + soundPause(); + } else { + soundResume(); + } + } +} + +void MainWnd::OnUpdateFilePause(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.paused); +} + +void MainWnd::OnFileReset() +{ + if(emulating) { + theApp.emulator.emuReset(); + systemScreenMessage(winResLoadString(IDS_RESET)); + } +} + +void MainWnd::OnUpdateFileReset(CCmdUI* pCmdUI) +{ + pCmdUI->Enable(emulating); +} + +void MainWnd::OnUpdateFileRecentFreeze(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.recentFreeze); + + if(pCmdUI->m_pMenu == NULL) + return; + + CMenu *pMenu = pCmdUI->m_pMenu; + + int i; + for(i = 0; i < 10; i++) { + if(!pMenu->RemoveMenu(ID_FILE_MRU_FILE1+i, MF_BYCOMMAND)) + break; + } + + for(i = 0; i < 10; i++) { + CString p = theApp.recentFiles[i]; + if(p.GetLength() == 0) + break; + int index = p.ReverseFind('\\'); + + if(index != -1) + p = p.Right(p.GetLength()-index-1); + + pMenu->AppendMenu(MF_STRING, ID_FILE_MRU_FILE1+i, p); + } + theApp.winAccelMgr.UpdateMenu((HMENU)*pMenu); +} + +BOOL MainWnd::OnFileRecentFile(UINT nID) +{ + if(theApp.recentFiles[(nID&0xFFFF)-ID_FILE_MRU_FILE1].GetLength()) { + theApp.szFile = theApp.recentFiles[(nID&0xFFFF)-ID_FILE_MRU_FILE1]; + if(FileRun()) + emulating = true; + else { + emulating = false; + soundPause(); + } + } + return TRUE; +} + +void MainWnd::OnFileRecentReset() +{ + int i = 0; + for(i = 0; i < 10; i++) + theApp.recentFiles[i] = ""; +} + +void MainWnd::OnFileRecentFreeze() +{ + theApp.recentFreeze = !theApp.recentFreeze; +} + +void MainWnd::OnFileExit() +{ + SendMessage(WM_CLOSE); +} + +void MainWnd::OnFileClose() +{ + // save battery file before we change the filename... + if(rom != NULL || gbRom != NULL) { + if(theApp.autoSaveLoadCheatList) + winSaveCheatListDefault(); + writeBatteryFile(); + soundPause(); + theApp.emulator.emuCleanUp(); + remoteCleanUp(); + } + emulating = 0; + RedrawWindow(NULL,NULL,RDW_INVALIDATE|RDW_ERASE|RDW_ALLCHILDREN); + systemSetTitle("VisualBoyAdvance"); +} + +void MainWnd::OnUpdateFileClose(CCmdUI* pCmdUI) +{ + pCmdUI->Enable(emulating); +} + + +void MainWnd::OnFileOpengameboy() +{ + theApp.winCheckFullscreen(); + if( fileOpenSelect( true ) ) { + FileRun(); + } +} + + +void MainWnd::OnFileLoad() +{ + theApp.winCheckFullscreen(); + CString buffer; + CString filename; + + int index = theApp.filename.ReverseFind('\\'); + + if(index != -1) + buffer = theApp.filename.Right(theApp.filename.GetLength()-index-1); + else + buffer = theApp.filename; + + CString saveDir = regQueryStringValue("saveDir", NULL); + if( saveDir[0] == '.' ) { + // handle as relative path + char baseDir[MAX_PATH+1]; + GetModuleFileName( NULL, baseDir, MAX_PATH ); + baseDir[MAX_PATH] = '\0'; // for security reasons + PathRemoveFileSpec( baseDir ); // removes the trailing file name and backslash + strcat( baseDir, "\\" ); + strcat( baseDir, saveDir ); + saveDir = baseDir; + } + + if(saveDir.IsEmpty()) + saveDir = getDirFromFile(theApp.filename); + + if(isDriveRoot(saveDir)) + filename.Format("%s%s.sgm", saveDir, buffer); + else + filename.Format("%s\\%s.sgm", saveDir, buffer); + + LPCTSTR exts[] = { ".sgm" }; + CString filter = winLoadFilter(IDS_FILTER_SGM); + CString title = winResLoadString(IDS_SELECT_SAVE_GAME_NAME); + + FileDlg dlg(this, filename, filter, 0, "", exts, saveDir, title, false); + + if(dlg.DoModal() == IDOK) { + bool res = loadSaveGame(dlg.GetPathName()); + + theApp.rewindCount = 0; + theApp.rewindCounter = 0; + theApp.rewindSaveNeeded = false; + + if(res) + systemScreenMessage(winResLoadString(IDS_LOADED_STATE)); + } +} + +void MainWnd::OnUpdateFileLoad(CCmdUI* pCmdUI) +{ + pCmdUI->Enable(emulating); +} + +BOOL MainWnd::OnFileLoadSlot(UINT nID) +{ + nID = nID + 1 - ID_FILE_LOADGAME_SLOT1; + + CString buffer; + CString filename; + + int index = theApp.filename.ReverseFind('\\'); + + if(index != -1) + buffer = theApp.filename.Right(theApp.filename.GetLength()-index-1); + else + buffer = theApp.filename; + + CString saveDir = regQueryStringValue("saveDir", NULL); + if( saveDir[0] == '.' ) { + // handle as relative path + char baseDir[MAX_PATH+1]; + GetModuleFileName( NULL, baseDir, MAX_PATH ); + baseDir[MAX_PATH] = '\0'; // for security reasons + PathRemoveFileSpec( baseDir ); // removes the trailing file name and backslash + strcat( baseDir, "\\" ); + strcat( baseDir, saveDir ); + saveDir = baseDir; + } + + if(saveDir.IsEmpty()) + saveDir = getDirFromFile(theApp.filename); + + if(isDriveRoot(saveDir)) + filename.Format("%s%s%d.sgm", saveDir, buffer, nID); + else + filename.Format("%s\\%s%d.sgm", saveDir, buffer, nID); + + CString format = winResLoadString(IDS_LOADED_STATE_N); + buffer.Format(format, nID); + + bool res = loadSaveGame(filename); + + if (theApp.paused) + InterframeCleanup(); + + systemScreenMessage(buffer); + + systemDrawScreen(); + + //theApp.rewindCount = 0; + //theApp.rewindCounter = 0; + //theApp.rewindSaveNeeded = false; + + return res; +} + +void MainWnd::OnFileSave() +{ + theApp.winCheckFullscreen(); + CString buffer; + CString filename; + + int index = theApp.filename.ReverseFind('\\'); + + if(index != -1) + buffer = theApp.filename.Right(theApp.filename.GetLength()-index-1); + else + buffer = theApp.filename; + + CString saveDir = regQueryStringValue("saveDir", NULL); + if( saveDir[0] == '.' ) { + // handle as relative path + char baseDir[MAX_PATH+1]; + GetModuleFileName( NULL, baseDir, MAX_PATH ); + baseDir[MAX_PATH] = '\0'; // for security reasons + PathRemoveFileSpec( baseDir ); // removes the trailing file name and backslash + strcat( baseDir, "\\" ); + strcat( baseDir, saveDir ); + saveDir = baseDir; + } + + if(saveDir.IsEmpty()) + saveDir = getDirFromFile(theApp.filename); + + if(isDriveRoot(saveDir)) + filename.Format("%s%s.sgm", saveDir, buffer); + else + filename.Format("%s\\%s.sgm", saveDir, buffer); + + LPCTSTR exts[] = { ".sgm" }; + CString filter = winLoadFilter(IDS_FILTER_SGM); + CString title = winResLoadString(IDS_SELECT_SAVE_GAME_NAME); + + FileDlg dlg(this, filename, filter, 0, "", exts, saveDir, title, true); + + if(dlg.DoModal() == IDOK) { + bool res = writeSaveGame(dlg.GetPathName()); + if(res) + systemScreenMessage(winResLoadString(IDS_WROTE_STATE)); + } +} + +void MainWnd::OnUpdateFileSave(CCmdUI* pCmdUI) +{ + pCmdUI->Enable(emulating); +} + +BOOL MainWnd::OnFileSaveSlot(UINT nID) +{ + nID = nID + 1 - ID_FILE_SAVEGAME_SLOT1; + + CString buffer; + CString filename; + + int index = theApp.filename.ReverseFind('\\'); + + if(index != -1) + buffer = theApp.filename.Right(theApp.filename.GetLength()-index-1); + else + buffer = theApp.filename; + + CString saveDir = regQueryStringValue("saveDir", NULL); + if( saveDir[0] == '.' ) { + // handle as relative path + char baseDir[MAX_PATH+1]; + GetModuleFileName( NULL, baseDir, MAX_PATH ); + baseDir[MAX_PATH] = '\0'; // for security reasons + PathRemoveFileSpec( baseDir ); // removes the trailing file name and backslash + strcat( baseDir, "\\" ); + strcat( baseDir, saveDir ); + saveDir = baseDir; + } + + if(saveDir.IsEmpty()) + saveDir = getDirFromFile(theApp.filename); + + if(isDriveRoot(saveDir)) + filename.Format("%s%s%d.sgm", saveDir, buffer, nID); + else + filename.Format("%s\\%s%d.sgm", saveDir, buffer, nID); + + bool res = writeSaveGame(filename); + + CString format = winResLoadString(IDS_WROTE_STATE_N); + buffer.Format(format, nID); + + systemScreenMessage(buffer); + + systemDrawScreen(); + + return res; +} + +void MainWnd::OnFileImportBatteryfile() +{ + theApp.winCheckFullscreen(); + LPCTSTR exts[] = { ".sav" }; + CString filter = winLoadFilter(IDS_FILTER_SAV); + CString title = winResLoadString(IDS_SELECT_BATTERY_FILE); + + CString saveDir = regQueryStringValue("batteryDir", NULL); + if( saveDir[0] == '.' ) { + // handle as relative path + char baseDir[MAX_PATH+1]; + GetModuleFileName( NULL, baseDir, MAX_PATH ); + baseDir[MAX_PATH] = '\0'; // for security reasons + PathRemoveFileSpec( baseDir ); // removes the trailing file name and backslash + strcat( baseDir, "\\" ); + strcat( baseDir, saveDir ); + saveDir = baseDir; + } + + if(saveDir.IsEmpty()) + saveDir = getDirFromFile(theApp.filename); + + FileDlg dlg(this, "", filter, 0, "", exts, saveDir, title, false); + + if(dlg.DoModal() == IDCANCEL) + return; + + CString str1 = winResLoadString(IDS_SAVE_WILL_BE_LOST); + CString str2 = winResLoadString(IDS_CONFIRM_ACTION); + + if(MessageBox(str1, + str2, + MB_OKCANCEL) == IDCANCEL) + return; + + bool res = false; + + res = theApp.emulator.emuReadBattery(dlg.GetPathName()); + + if(!res) + systemMessage(MSG_CANNOT_OPEN_FILE, "Cannot open file %s", dlg.GetPathName()); + else { + //Removed the reset to allow loading a battery file 'within' a save state. + //theApp.emulator.emuReset(); + } +} + +void MainWnd::OnUpdateFileImportBatteryfile(CCmdUI* pCmdUI) +{ + pCmdUI->Enable(emulating); +} + +void MainWnd::OnFileImportGamesharkcodefile() +{ + theApp.winCheckFullscreen(); + LPCTSTR exts[] = { "" }; + CString filter = theApp.cartridgeType == 0 ? winLoadFilter(IDS_FILTER_SPC) : winLoadFilter(IDS_FILTER_GCF); + CString title = winResLoadString(IDS_SELECT_CODE_FILE); + + FileDlg dlg(this, "", filter, 0, "", exts, "", title, false); + + if(dlg.DoModal() == IDCANCEL) + return; + + CString str1 = winResLoadString(IDS_CODES_WILL_BE_LOST); + CString str2 = winResLoadString(IDS_CONFIRM_ACTION); + + if(MessageBox(str1, + str2, + MB_OKCANCEL) == IDCANCEL) + return; + + bool res = false; + CString file = dlg.GetPathName(); + if(theApp.cartridgeType == 1) + res = gbCheatReadGSCodeFile(file); + else { + res = fileImportGSACodeFile(file); + } +} + +void MainWnd::OnUpdateFileImportGamesharkcodefile(CCmdUI* pCmdUI) +{ + pCmdUI->Enable(emulating); +} + +void MainWnd::OnFileImportGamesharksnapshot() +{ + theApp.winCheckFullscreen(); + LPCTSTR exts[] = { ".gbs" }; + CString filter = theApp.cartridgeType == 1 ? winLoadFilter(IDS_FILTER_GBS) : winLoadFilter(IDS_FILTER_SPS); + CString title = winResLoadString(IDS_SELECT_SNAPSHOT_FILE); + + FileDlg dlg(this, "", filter, 0, "", exts, "", title, false); + + if(dlg.DoModal() == IDCANCEL) + return; + + CString str1 = winResLoadString(IDS_SAVE_WILL_BE_LOST); + CString str2 = winResLoadString(IDS_CONFIRM_ACTION); + + if(MessageBox(str1, + str2, + MB_OKCANCEL) == IDCANCEL) + return; + + if(theApp.cartridgeType == 1) + gbReadGSASnapshot(dlg.GetPathName()); + else + CPUReadGSASnapshot(dlg.GetPathName()); +} + +void MainWnd::OnUpdateFileImportGamesharksnapshot(CCmdUI* pCmdUI) +{ + pCmdUI->Enable(emulating); +} + +void MainWnd::OnFileExportBatteryfile() +{ + theApp.winCheckFullscreen(); + CString name; + + int index = theApp.filename.ReverseFind('\\'); + + if(index != -1) + name = theApp.filename.Right(theApp.filename.GetLength()-index-1); + else + name = theApp.filename; + + LPCTSTR exts[] = {".sav", ".dat" }; + + CString filter = winLoadFilter(IDS_FILTER_SAV); + CString title = winResLoadString(IDS_SELECT_BATTERY_FILE); + + CString saveDir = regQueryStringValue("batteryDir", NULL); + if( saveDir[0] == '.' ) { + // handle as relative path + char baseDir[MAX_PATH+1]; + GetModuleFileName( NULL, baseDir, MAX_PATH ); + baseDir[MAX_PATH] = '\0'; // for security reasons + PathRemoveFileSpec( baseDir ); // removes the trailing file name and backslash + strcat( baseDir, "\\" ); + strcat( baseDir, saveDir ); + saveDir = baseDir; + } + + if(saveDir.IsEmpty()) + saveDir = getDirFromFile(theApp.filename); + + FileDlg dlg(this, + name, + filter, + 1, + "SAV", + exts, + saveDir, + title, + true); + + if(dlg.DoModal() == IDCANCEL) { + return; + } + + bool result = false; + + if(theApp.cartridgeType == 1) { + result = gbWriteBatteryFile(dlg.GetPathName(), false); + } else + result = theApp.emulator.emuWriteBattery(dlg.GetPathName()); + + if(!result) + systemMessage(MSG_ERROR_CREATING_FILE, "Error creating file %s", + dlg.GetPathName()); +} + +void MainWnd::OnUpdateFileExportBatteryfile(CCmdUI* pCmdUI) +{ + pCmdUI->Enable(emulating); +} + +void MainWnd::OnFileExportGamesharksnapshot() +{ + theApp.winCheckFullscreen(); + if(eepromInUse) { + systemMessage(IDS_EEPROM_NOT_SUPPORTED, "EEPROM saves cannot be exported"); + return; + } + + CString name; + + int index = theApp.filename.ReverseFind('\\'); + + if(index != -1) + name = theApp.filename.Right(theApp.filename.GetLength()-index-1); + else + name = theApp.filename; + + LPCTSTR exts[] = {".sps" }; + + CString filter = winLoadFilter(IDS_FILTER_SPS); + CString title = winResLoadString(IDS_SELECT_SNAPSHOT_FILE); + + FileDlg dlg(this, + name, + filter, + 1, + "SPS", + exts, + "", + title, + true); + + if(dlg.DoModal() == IDCANCEL) + return; + + char buffer[16]; + strncpy(buffer, (const char *)&rom[0xa0], 12); + buffer[12] = 0; + + ExportGSASnapshot dlg2(dlg.GetPathName(), buffer, this); + dlg2.DoModal(); +} + +void MainWnd::OnUpdateFileExportGamesharksnapshot(CCmdUI* pCmdUI) +{ + pCmdUI->Enable(emulating && theApp.cartridgeType == 0); +} + +void MainWnd::OnFileScreencapture() +{ + theApp.winCheckFullscreen(); + CString name; + CString filename; + + int index = theApp.filename.ReverseFind('\\'); + + if(index != -1) + name = theApp.filename.Right(theApp.filename.GetLength()-index-1); + else + name = theApp.filename; + + CString capdir = regQueryStringValue("captureDir", ""); + if( capdir[0] == '.' ) { + // handle as relative path + char baseDir[MAX_PATH+1]; + GetModuleFileName( NULL, baseDir, MAX_PATH ); + baseDir[MAX_PATH] = '\0'; // for security reasons + PathRemoveFileSpec( baseDir ); // removes the trailing file name and backslash + strcat( baseDir, "\\" ); + strcat( baseDir, capdir ); + capdir = baseDir; + } + if(capdir.IsEmpty()) + capdir = getDirFromFile(theApp.filename); + + CString ext = "png"; + + if(theApp.captureFormat != 0) + ext = "bmp"; + + if(isDriveRoot(capdir)) + filename.Format("%s%s.%s", capdir, name, ext); + else + filename.Format("%s\\%s.%s", capdir, name, ext); + + LPCTSTR exts[] = {".png", ".bmp" }; + + CString filter = winLoadFilter(IDS_FILTER_PNG); + CString title = winResLoadString(IDS_SELECT_CAPTURE_NAME); + + FileDlg dlg(this, + filename, + filter, + theApp.captureFormat ? 2 : 1, + theApp.captureFormat ? "BMP" : "PNG", + exts, + capdir, + title, + true); + + if(dlg.DoModal() == IDCANCEL) + return; + + if(dlg.getFilterIndex() == 2) + theApp.emulator.emuWriteBMP(dlg.GetPathName()); + else + theApp.emulator.emuWritePNG(dlg.GetPathName()); + + systemScreenMessage(winResLoadString(IDS_SCREEN_CAPTURE)); +} + +void MainWnd::OnUpdateFileScreencapture(CCmdUI* pCmdUI) +{ + pCmdUI->Enable(emulating); +} + +void MainWnd::OnFileRominformation() +{ + theApp.winCheckFullscreen(); + if(theApp.cartridgeType == 0) { + RomInfoGBA dlg(rom); + dlg.DoModal(); + } else { + RomInfoGB dlg(gbRom); + dlg.DoModal(); + } +} + +void MainWnd::OnUpdateFileRominformation(CCmdUI* pCmdUI) +{ + pCmdUI->Enable(emulating); +} + +void MainWnd::OnFileTogglemenu() +{ + if(theApp.videoOption <= VIDEO_4X) + return; + + theApp.menuToggle = !theApp.menuToggle; + + if(theApp.menuToggle) { + theApp.updateMenuBar(); + if(theApp.tripleBuffering) { + if(theApp.display) + theApp.display->checkFullScreen(); + DrawMenuBar(); + } + } else { + SetMenu(NULL); + DestroyMenu(theApp.menu); + } + + theApp.adjustDestRect(); + if(theApp.display) + theApp.display->resize(theApp.dest.right-theApp.dest.left, theApp.dest.bottom-theApp.dest.top); +} + +void MainWnd::OnUpdateFileTogglemenu(CCmdUI* pCmdUI) +{ + pCmdUI->Enable(theApp.videoOption > VIDEO_4X); +} + +bool MainWnd::fileImportGSACodeFile(CString& fileName) +{ + FILE *f = fopen(fileName, "rb"); + + if(f == NULL) { + systemMessage(MSG_CANNOT_OPEN_FILE, "Cannot open file %s", fileName); + return false; + } + + u32 len; + fread(&len, 1, 4, f); + if(len != 14) { + fclose(f); + systemMessage(MSG_UNSUPPORTED_CODE_FILE, "Unsupported code file %s", + fileName); + return false; + } + char buffer[16]; + fread(buffer, 1, 14, f); + buffer[14] = 0; + if(memcmp(buffer, "SharkPortCODES", 14)) { + fclose(f); + systemMessage(MSG_UNSUPPORTED_CODE_FILE, "Unsupported code file %s", + fileName); + return false; + } + fseek(f, 0x1e, SEEK_SET); + fread(&len, 1, 4, f); + INT_PTR game = 0; + if(len > 1) { + GSACodeSelect dlg(f); + game = dlg.DoModal(); + } + fclose(f); + + bool v3 = false; + + int index = fileName.ReverseFind('.'); + + if(index != -1) { + if(fileName.Right(3).CompareNoCase("XPC") == 0) + v3 = true; + } + + if(game != -1) { + return cheatsImportGSACodeFile(fileName, (int)game, v3); + } + + return true; +} + +void MainWnd::OnFileSavegameOldestslot() +{ + if(!emulating) + return; + + CString filename; + + int index = theApp.filename.ReverseFind('\\'); + + if(index != -1) + filename = theApp.filename.Right(theApp.filename.GetLength()-index-1); + else + filename = theApp.filename; + + CString saveDir = regQueryStringValue("saveDir", NULL); + if( saveDir[0] == '.' ) { + // handle as relative path + char baseDir[MAX_PATH+1]; + GetModuleFileName( NULL, baseDir, MAX_PATH ); + baseDir[MAX_PATH] = '\0'; // for security reasons + PathRemoveFileSpec( baseDir ); // removes the trailing file name and backslash + strcat( baseDir, "\\" ); + strcat( baseDir, saveDir ); + saveDir = baseDir; + } + + if(saveDir.IsEmpty()) + saveDir = getDirFromFile(theApp.filename); + + if(!isDriveRoot(saveDir)) + saveDir += "\\"; + + CString name; + CFileStatus status; + CString str; + time_t time = (time_t)-1; + int found = 0; + + for(int i = 0; i < 10; i++) { + name.Format("%s%s%d.sgm", saveDir, filename, i+1); + + if(emulating && CFile::GetStatus(name, status)) { + if(status.m_mtime.GetTime() < time) { + time = status.m_mtime.GetTime(); + found = i; + } + } else { + found = i; + break; + } + } + OnFileSaveSlot(ID_FILE_SAVEGAME_SLOT1+found); +} + +void MainWnd::OnUpdateFileSavegameOldestslot(CCmdUI* pCmdUI) +{ + pCmdUI->Enable(emulating); + if(pCmdUI->m_pSubMenu != NULL) { + CMenu *pMenu = pCmdUI->m_pSubMenu; + CString filename; + + int index = theApp.filename.ReverseFind('\\'); + + if(index != -1) + filename = theApp.filename.Right(theApp.filename.GetLength()-index-1); + else + filename = theApp.filename; + + CString saveDir = regQueryStringValue("saveDir", NULL); + if( saveDir[0] == '.' ) { + // handle as relative path + char baseDir[MAX_PATH+1]; + GetModuleFileName( NULL, baseDir, MAX_PATH ); + baseDir[MAX_PATH] = '\0'; // for security reasons + PathRemoveFileSpec( baseDir ); // removes the trailing file name and backslash + strcat( baseDir, "\\" ); + strcat( baseDir, saveDir ); + saveDir = baseDir; + } + + if(saveDir.IsEmpty()) + saveDir = getDirFromFile(theApp.filename); + + if(!isDriveRoot(saveDir)) + saveDir += "\\"; + + CString name; + CFileStatus status; + CString str; + + for(int i = 0; i < 10; i++) { + name.Format("%s%s%d.sgm", saveDir, filename, i+1); + + if(emulating && CFile::GetStatus(name, status)) { + CString timestamp = status.m_mtime.Format("%Y/%m/%d %H:%M:%S"); + str.Format("%d %s", i+1, timestamp); + } else { + str.Format("%d ----/--/-- --:--:--", i+1); + } + pMenu->ModifyMenu(ID_FILE_SAVEGAME_SLOT1+i, MF_STRING|MF_BYCOMMAND, ID_FILE_SAVEGAME_SLOT1+i, str); + } + + theApp.winAccelMgr.UpdateMenu(pMenu->GetSafeHmenu()); + } +} + +void MainWnd::OnFileLoadgameMostrecent() +{ + if(!emulating) + return; + + CString filename; + + int index = theApp.filename.ReverseFind('\\'); + + if(index != -1) + filename = theApp.filename.Right(theApp.filename.GetLength()-index-1); + else + filename = theApp.filename; + + CString saveDir = regQueryStringValue("saveDir", NULL); + if( saveDir[0] == '.' ) { + // handle as relative path + char baseDir[MAX_PATH+1]; + GetModuleFileName( NULL, baseDir, MAX_PATH ); + baseDir[MAX_PATH] = '\0'; // for security reasons + PathRemoveFileSpec( baseDir ); // removes the trailing file name and backslash + strcat( baseDir, "\\" ); + strcat( baseDir, saveDir ); + saveDir = baseDir; + } + + if(saveDir.IsEmpty()) + saveDir = getDirFromFile(theApp.filename); + + if(!isDriveRoot(saveDir)) + saveDir += "\\"; + + CString name; + CFileStatus status; + CString str; + time_t time = 0; + int found = -1; + + for(int i = 0; i < 10; i++) { + name.Format("%s%s%d.sgm", saveDir, filename, i+1); + + if(emulating && CFile::GetStatus(name, status)) { +if(status.m_mtime.GetTime() < time) { + time = status.m_mtime.GetTime(); + found = i; + } + } + } + if(found != -1) { + OnFileLoadSlot(ID_FILE_LOADGAME_SLOT1+found); + } +} + +void MainWnd::OnUpdateFileLoadgameMostrecent(CCmdUI* pCmdUI) +{ + pCmdUI->Enable(emulating); + + if(pCmdUI->m_pSubMenu != NULL) { + CMenu *pMenu = pCmdUI->m_pSubMenu; + CString filename; + + int index = theApp.filename.ReverseFind('\\'); + + if(index != -1) + filename = theApp.filename.Right(theApp.filename.GetLength()-index-1); + else + filename = theApp.filename; + + CString saveDir = regQueryStringValue("saveDir", NULL); + if( saveDir[0] == '.' ) { + // handle as relative path + char baseDir[MAX_PATH+1]; + GetModuleFileName( NULL, baseDir, MAX_PATH ); + baseDir[MAX_PATH] = '\0'; // for security reasons + PathRemoveFileSpec( baseDir ); // removes the trailing file name and backslash + strcat( baseDir, "\\" ); + strcat( baseDir, saveDir ); + saveDir = baseDir; + } + + if(saveDir.IsEmpty()) + saveDir = getDirFromFile(theApp.filename); + + if(!isDriveRoot(saveDir)) + saveDir += "\\"; + + CString name; + CFileStatus status; + CString str; + + for(int i = 0; i < 10; i++) { + name.Format("%s%s%d.sgm", saveDir, filename, i+1); + + if(emulating && CFile::GetStatus(name, status)) { + CString timestamp = status.m_mtime.Format("%Y/%m/%d %H:%M:%S"); + str.Format("%d %s", i+1, timestamp); + } else { + str.Format("%d ----/--/-- --:--:--", i+1); + } + pMenu->ModifyMenu(ID_FILE_LOADGAME_SLOT1+i, MF_STRING|MF_BYCOMMAND, ID_FILE_LOADGAME_SLOT1+i, str); + } + + theApp.winAccelMgr.UpdateMenu(pMenu->GetSafeHmenu()); + } +} + +void MainWnd::OnUpdateFileLoadGameSlot(CCmdUI *pCmdUI) +{ + pCmdUI->Enable(emulating); +} + +void MainWnd::OnUpdateFileSaveGameSlot(CCmdUI *pCmdUI) +{ + pCmdUI->Enable(emulating); +} + +void MainWnd::OnFileLoadgameAutoloadmostrecent() +{ + theApp.autoLoadMostRecent = !theApp.autoLoadMostRecent; +} + +void MainWnd::OnUpdateFileLoadgameAutoloadmostrecent(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.autoLoadMostRecent); +} diff --git a/src/win32/MainWndHelp.cpp b/src/win32/MainWndHelp.cpp index bbecc0ba..5d04917a 100644 --- a/src/win32/MainWndHelp.cpp +++ b/src/win32/MainWndHelp.cpp @@ -44,3 +44,9 @@ void MainWnd::OnHelpBugreport() dlg.DoModal(); } + +void MainWnd::OnHelpGnupubliclicense() +{ + ::ShellExecute(0, _T("open"), "http://www.gnu.org/licenses/gpl.html", + 0, 0, SW_SHOWNORMAL); +} diff --git a/src/win32/MainWndOptions.cpp b/src/win32/MainWndOptions.cpp index 541190c2..413bc984 100644 --- a/src/win32/MainWndOptions.cpp +++ b/src/win32/MainWndOptions.cpp @@ -1,1770 +1,1938 @@ -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#include "stdafx.h" -#include "MainWnd.h" - -#include "Associate.h" -#include "Directories.h" -#include "FileDlg.h" -#include "GBColorDlg.h" -#include "Joypad.h" -#include "MaxScale.h" -#include "ModeConfirm.h" -#include "Reg.h" -#include "RewindInterval.h" -#include "Throttle.h" -#include "WinResUtil.h" -#include "LinkOptions.h" - -#include "../System.h" -#include "../agbprint.h" -#include "../GBA.h" -#include "../Globals.h" -#include "../Sound.h" -#include "../Util.h" -#include "../gb/GB.h" -#include "../gb/gbGlobals.h" -#include "../gb/gbPrinter.h" -#include "../Link.h" - -extern int emulating; - -#define VBA_CONFIRM_MODE WM_APP + 100 - -extern void CPUUpdateRenderBuffers(bool force); - -void MainWnd::OnOptionsFrameskipThrottleNothrottle() -{ - theApp.throttle = 0; -} - -void MainWnd::OnUpdateOptionsFrameskipThrottleNothrottle(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.throttle == 0); -} - -void MainWnd::OnOptionsFrameskipThrottle25() -{ - theApp.throttle = 25; -} - -void MainWnd::OnUpdateOptionsFrameskipThrottle25(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.throttle == 25); -} - -void MainWnd::OnOptionsFrameskipThrottle50() -{ - theApp.throttle = 50; -} - -void MainWnd::OnUpdateOptionsFrameskipThrottle50(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.throttle == 50); -} - -void MainWnd::OnOptionsFrameskipThrottle100() -{ - theApp.throttle = 100; -} - -void MainWnd::OnUpdateOptionsFrameskipThrottle100(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.throttle == 100); -} - -void MainWnd::OnOptionsFrameskipThrottle150() -{ - theApp.throttle = 150; -} - -void MainWnd::OnUpdateOptionsFrameskipThrottle150(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.throttle == 150); -} - -void MainWnd::OnOptionsFrameskipThrottle200() -{ - theApp.throttle = 200; -} - -void MainWnd::OnUpdateOptionsFrameskipThrottle200(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.throttle == 200); -} - -void MainWnd::OnOptionsFrameskipThrottleOther() -{ - Throttle dlg; - int v = dlg.DoModal(); - if(v) - theApp.throttle = v; -} - -void MainWnd::OnUpdateOptionsFrameskipThrottleOther(CCmdUI* pCmdUI) -{ - int throttle = theApp.throttle; - pCmdUI->SetCheck(throttle != 0 && throttle != 25 && - throttle != 50 && throttle != 100 && - throttle != 150 && throttle != 200); -} - -void MainWnd::OnOptionsFrameskipAutomatic() -{ - theApp.autoFrameSkip = !theApp.autoFrameSkip; - if(!theApp.autoFrameSkip && emulating) - theApp.updateFrameSkip(); -} - -void MainWnd::OnUpdateOptionsFrameskipAutomatic(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.autoFrameSkip); -} - -BOOL MainWnd::OnOptionsFrameskip(UINT nID) -{ - switch(nID) { - case ID_OPTIONS_VIDEO_FRAMESKIP_0: - case ID_OPTIONS_VIDEO_FRAMESKIP_1: - case ID_OPTIONS_VIDEO_FRAMESKIP_2: - case ID_OPTIONS_VIDEO_FRAMESKIP_3: - case ID_OPTIONS_VIDEO_FRAMESKIP_4: - case ID_OPTIONS_VIDEO_FRAMESKIP_5: - if(theApp.cartridgeType == 0) { - frameSkip = nID - ID_OPTIONS_VIDEO_FRAMESKIP_0; - } else { - gbFrameSkip = nID - ID_OPTIONS_VIDEO_FRAMESKIP_0; - } - if(emulating) - theApp.updateFrameSkip(); - return TRUE; - break; - case ID_OPTIONS_VIDEO_FRAMESKIP_6: - case ID_OPTIONS_VIDEO_FRAMESKIP_7: - case ID_OPTIONS_VIDEO_FRAMESKIP_8: - case ID_OPTIONS_VIDEO_FRAMESKIP_9: - if(theApp.cartridgeType == 0) { - frameSkip = 6 + nID - ID_OPTIONS_VIDEO_FRAMESKIP_6; - } else { - gbFrameSkip = 6 + nID - ID_OPTIONS_VIDEO_FRAMESKIP_6; - } - if(emulating) - theApp.updateFrameSkip(); - return TRUE; - break; - } - return FALSE; -} - -void MainWnd::OnUpdateOptionsVideoFrameskip0(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.cartridgeType == 0 ? frameSkip == 0 : gbFrameSkip == 0); -} - -void MainWnd::OnUpdateOptionsVideoFrameskip1(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.cartridgeType == 0 ? frameSkip == 1 : gbFrameSkip == 1); -} - -void MainWnd::OnUpdateOptionsVideoFrameskip2(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.cartridgeType == 0 ? frameSkip == 2 : gbFrameSkip == 2); -} - -void MainWnd::OnUpdateOptionsVideoFrameskip3(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.cartridgeType == 0 ? frameSkip == 3 : gbFrameSkip == 3); -} - -void MainWnd::OnUpdateOptionsVideoFrameskip4(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.cartridgeType == 0 ? frameSkip == 4 : gbFrameSkip == 4); -} - -void MainWnd::OnUpdateOptionsVideoFrameskip5(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.cartridgeType == 0 ? frameSkip == 5 : gbFrameSkip == 5); -} - -void MainWnd::OnUpdateOptionsVideoFrameskip6(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.cartridgeType == 0 ? frameSkip == 6 : gbFrameSkip == 6); -} - -void MainWnd::OnUpdateOptionsVideoFrameskip7(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.cartridgeType == 0 ? frameSkip == 7 : gbFrameSkip == 7); -} - -void MainWnd::OnUpdateOptionsVideoFrameskip8(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.cartridgeType == 0 ? frameSkip == 8 : gbFrameSkip == 8); -} - -void MainWnd::OnUpdateOptionsVideoFrameskip9(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.cartridgeType == 0 ? frameSkip == 9 : gbFrameSkip == 9); -} - -void MainWnd::OnOptionsVideoVsync() -{ - theApp.vsync = !theApp.vsync; - theApp.updateRenderMethod(true); -} - -void MainWnd::OnUpdateOptionsVideoVsync(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.vsync); -} - -void MainWnd::OnUpdateOptionsVideoX1(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.videoOption == VIDEO_1X); -} - -void MainWnd::OnUpdateOptionsVideoX2(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.videoOption == VIDEO_2X); -} - -void MainWnd::OnUpdateOptionsVideoX3(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.videoOption == VIDEO_3X); -} - -void MainWnd::OnUpdateOptionsVideoX4(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.videoOption == VIDEO_4X); -} - -void MainWnd::OnUpdateOptionsVideoFullscreen320x240(CCmdUI* pCmdUI) -{ - pCmdUI->Enable(theApp.mode320Available); - pCmdUI->SetCheck(theApp.videoOption == VIDEO_320x240); -} - -void MainWnd::OnUpdateOptionsVideoFullscreen640x480(CCmdUI* pCmdUI) -{ - pCmdUI->Enable(theApp.mode640Available); - pCmdUI->SetCheck(theApp.videoOption == VIDEO_640x480); -} - -void MainWnd::OnUpdateOptionsVideoFullscreen800x600(CCmdUI* pCmdUI) -{ - pCmdUI->Enable(theApp.mode800Available); - pCmdUI->SetCheck(theApp.videoOption == VIDEO_800x600); -} - -void MainWnd::OnUpdateOptionsVideoFullscreen1024x768(CCmdUI *pCmdUI) -{ - pCmdUI->Enable(theApp.mode1024Available); - pCmdUI->SetCheck(theApp.videoOption == VIDEO_1024x768); -} - -void MainWnd::OnUpdateOptionsVideoFullscreen1280x1024(CCmdUI *pCmdUI) -{ - pCmdUI->Enable(theApp.mode1280Available); - pCmdUI->SetCheck(theApp.videoOption == VIDEO_1280x1024); -} - -BOOL MainWnd::OnOptionVideoSize(UINT nID) -{ - theApp.updateVideoSize(nID); - theApp.m_pMainWnd->PostMessage(VBA_CONFIRM_MODE); - return TRUE; -} - -void MainWnd::OnOptionsVideoFullscreen320x240() -{ - OnOptionVideoSize(ID_OPTIONS_VIDEO_FULLSCREEN320X240); -} - -void MainWnd::OnOptionsVideoFullscreen640x480() -{ - OnOptionVideoSize(ID_OPTIONS_VIDEO_FULLSCREEN640X480); -} - -void MainWnd::OnOptionsVideoFullscreen800x600() -{ - OnOptionVideoSize(ID_OPTIONS_VIDEO_FULLSCREEN800X600); -} - -void MainWnd::OnOptionsVideoFullscreen1024x768() -{ - OnOptionVideoSize(ID_OPTIONS_VIDEO_FULLSCREEN1024X768); -} - -void MainWnd::OnOptionsVideoFullscreen1280x1024() -{ - OnOptionVideoSize(ID_OPTIONS_VIDEO_FULLSCREEN1280X1024); -} - -void MainWnd::OnOptionsVideoFullscreen() -{ - theApp.winCheckFullscreen(); - GUID *pGUID = NULL; - int size = theApp.display->selectFullScreenMode(&pGUID); - if(size != -1) { - int width = (size >> 12) & 4095; - int height = (size & 4095); - int colorDepth = (size >> 24); - if(width != theApp.fsWidth || - height != theApp.fsHeight || - colorDepth != theApp.fsColorDepth || - pGUID != theApp.pVideoDriverGUID || - theApp.videoOption != VIDEO_OTHER) { - theApp.fsForceChange = true; - theApp.fsWidth = width; - theApp.fsHeight = height; - theApp.fsFrequency = (theApp.display->selectFullScreenMode2() & 0xFFFF); - theApp.fsAdapter = theApp.display->selectFullScreenMode2(); - theApp.fsAdapter = (theApp.fsAdapter & 0xFFFF0000) >> 16; - theApp.fsColorDepth = colorDepth; - theApp.pVideoDriverGUID = pGUID; - if(pGUID) { - theApp.videoDriverGUID = *pGUID; - regSetDwordValue("defaultVideoDriver", FALSE); - regSetBinaryValue("videoDriverGUID", - (char *)pGUID, sizeof(GUID)); - } else { - regSetDwordValue("defaultVideoDriver", TRUE); - } - theApp.updateVideoSize(ID_OPTIONS_VIDEO_FULLSCREEN); - theApp.m_pMainWnd->PostMessage(VBA_CONFIRM_MODE); - } - } - theApp.winAccelMgr.UpdateMenu(theApp.menu); -} - -void MainWnd::OnUpdateOptionsVideoFullscreen(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.videoOption == VIDEO_OTHER); -} - -void MainWnd::OnOptionsVideoDisablesfx() -{ - cpuDisableSfx = !cpuDisableSfx; - if(emulating && theApp.cartridgeType == 0) - CPUUpdateRender(); -} - -void MainWnd::OnUpdateOptionsVideoDisablesfx(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(!cpuDisableSfx); -} - -void MainWnd::OnOptionsVideoFullscreenstretchtofit() -{ - theApp.fullScreenStretch = !theApp.fullScreenStretch; - theApp.updateWindowSize(theApp.videoOption); - if(theApp.display) - theApp.display->clear(); -} - -void MainWnd::OnUpdateOptionsVideoFullscreenstretchtofit(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.fullScreenStretch); -} - -BOOL MainWnd::OnVideoLayer(UINT nID) -{ - layerSettings ^= 0x0100 << - ((nID & 0xFFFF) - ID_OPTIONS_VIDEO_LAYERS_BG0); - layerEnable = DISPCNT & layerSettings; - CPUUpdateRenderBuffers(false); - return TRUE; -} - -void MainWnd::OnUpdateVideoLayer(CCmdUI *pCmdUI) -{ - pCmdUI->SetCheck((layerSettings >> (8 + pCmdUI->m_nID - ID_OPTIONS_VIDEO_LAYERS_BG0)) & 1); - switch(pCmdUI->m_nID) { - case ID_OPTIONS_VIDEO_LAYERS_BG1: - case ID_OPTIONS_VIDEO_LAYERS_BG2: - case ID_OPTIONS_VIDEO_LAYERS_BG3: - case ID_OPTIONS_VIDEO_LAYERS_WIN1: - case ID_OPTIONS_VIDEO_LAYERS_OBJWIN: - pCmdUI->Enable(theApp.cartridgeType == 0); - break; - } -} - -void MainWnd::OnOptionsVideoRendermethodGdi() -{ - theApp.renderMethod = GDI; - theApp.updateRenderMethod(false); - theApp.winAccelMgr.UpdateMenu(theApp.menu); -} - -void MainWnd::OnUpdateOptionsVideoRendermethodGdi(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.renderMethod == GDI); -} - -void MainWnd::OnOptionsVideoRendermethodDirectdraw() -{ - theApp.renderMethod = DIRECT_DRAW; - theApp.updateRenderMethod(false); - theApp.winAccelMgr.UpdateMenu(theApp.menu); -} - -void MainWnd::OnUpdateOptionsVideoRendermethodDirectdraw(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.renderMethod == DIRECT_DRAW); -} - -void MainWnd::OnOptionsVideoRendermethodDirect3d() -{ - theApp.renderMethod = DIRECT_3D; - theApp.updateRenderMethod(false); - theApp.winAccelMgr.UpdateMenu(theApp.menu); -} - -void MainWnd::OnUpdateOptionsVideoRendermethodDirect3d(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.renderMethod == DIRECT_3D); -} - -void MainWnd::OnOptionsVideoRendermethodOpengl() -{ - theApp.renderMethod = OPENGL; - theApp.updateRenderMethod(false); - theApp.winAccelMgr.UpdateMenu(theApp.menu); -} - -void MainWnd::OnUpdateOptionsVideoRendermethodOpengl(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.renderMethod == OPENGL); -} - -void MainWnd::OnOptionsVideoTriplebuffering() -{ - theApp.tripleBuffering = !theApp.tripleBuffering; - theApp.updateRenderMethod(true); -} - -void MainWnd::OnUpdateOptionsVideoTriplebuffering(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.tripleBuffering); -} - -void MainWnd::OnOptionsVideoDdrawemulationonly() -{ - theApp.ddrawEmulationOnly = !theApp.ddrawEmulationOnly; -} - -void MainWnd::OnUpdateOptionsVideoDdrawemulationonly(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.ddrawEmulationOnly); -} - -void MainWnd::OnOptionsVideoDdrawusevideomemory() -{ - theApp.ddrawUseVideoMemory = !theApp.ddrawUseVideoMemory; -} - -void MainWnd::OnUpdateOptionsVideoDdrawusevideomemory(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.ddrawUseVideoMemory); -} - -void MainWnd::OnOptionsVideoRenderoptionsD3dnofilter() -{ - theApp.d3dFilter = 0; - if(theApp.display) - theApp.display->setOption("d3dFilter", 0); -} - -void MainWnd::OnUpdateOptionsVideoRenderoptionsD3dnofilter(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.d3dFilter == 0); -} - -void MainWnd::OnOptionsVideoRenderoptionsD3dbilinear() -{ - theApp.d3dFilter = 1; - if(theApp.display) - theApp.display->setOption("d3dFilter", 1); -} - -void MainWnd::OnUpdateOptionsVideoRenderoptionsD3dbilinear(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.d3dFilter == 1); -} - -void MainWnd::OnOptionsVideoRenderoptionsGlnearest() -{ - theApp.glFilter = 0; - if(theApp.display) - theApp.display->setOption("glFilter", 0); -} - -void MainWnd::OnUpdateOptionsVideoRenderoptionsGlnearest(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.glFilter == 0); -} - -void MainWnd::OnOptionsVideoRenderoptionsGlbilinear() -{ - theApp.glFilter = 1; - if(theApp.display) - theApp.display->setOption("glFilter", 1); -} - -void MainWnd::OnUpdateOptionsVideoRenderoptionsGlbilinear(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.glFilter == 1); -} - -void MainWnd::OnOptionsVideoRenderoptionsGltriangle() -{ - theApp.glType = 0; - if(theApp.display) - theApp.display->setOption("glType", 0); -} - -void MainWnd::OnUpdateOptionsVideoRenderoptionsGltriangle(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.glType == 0); -} - -void MainWnd::OnOptionsVideoRenderoptionsGlquads() -{ - theApp.glType = 1; - if(theApp.display) - theApp.display->setOption("glType", 1); -} - -void MainWnd::OnUpdateOptionsVideoRenderoptionsGlquads(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.glType == 1); -} - -void MainWnd::OnOptionsEmulatorAssociate() -{ - theApp.winCheckFullscreen(); - Associate dlg; - dlg.DoModal(); -} - -void MainWnd::OnOptionsEmulatorDirectories() -{ - theApp.winCheckFullscreen(); - Directories dlg; - dlg.DoModal(); -} - -void MainWnd::OnOptionsEmulatorDisablestatusmessages() -{ - theApp.disableStatusMessage = !theApp.disableStatusMessage; -} - -void MainWnd::OnUpdateOptionsEmulatorDisablestatusmessages(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.disableStatusMessage); -} - -void MainWnd::OnOptionsEmulatorSynchronize() -{ - synchronize = !synchronize; -} - -void MainWnd::OnUpdateOptionsEmulatorSynchronize(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(synchronize); -} - -void MainWnd::OnOptionsEmulatorPausewheninactive() -{ - theApp.pauseWhenInactive = !theApp.pauseWhenInactive; -} - -void MainWnd::OnUpdateOptionsEmulatorPausewheninactive(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.pauseWhenInactive); -} - -void MainWnd::OnOptionsEmulatorSpeeduptoggle() -{ - theApp.speedupToggle = !theApp.speedupToggle; -} - -void MainWnd::OnUpdateOptionsEmulatorSpeeduptoggle(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.speedupToggle); -} - -void MainWnd::OnOptionsEmulatorRemoveintrosgba() -{ - // theApp.removeIntros = !theApp.removeIntros; -} - -void MainWnd::OnUpdateOptionsEmulatorRemoveintrosgba(CCmdUI* pCmdUI) -{ - pCmdUI->Enable(false); - // pCmdUI->SetCheck(theApp.removeIntros); -} - -void MainWnd::OnOptionsEmulatorAutomaticallyipspatch() -{ - theApp.autoIPS = !theApp.autoIPS; -} - -void MainWnd::OnUpdateOptionsEmulatorAutomaticallyipspatch(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.autoIPS); -} - -void MainWnd::OnOptionsEmulatorAgbprint() -{ - agbPrintEnable(!agbPrintIsEnabled()); -} - -void MainWnd::OnUpdateOptionsEmulatorAgbprint(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(agbPrintIsEnabled()); -} - -void MainWnd::OnOptionsEmulatorRealtimeclock() -{ - theApp.winRtcEnable = !theApp.winRtcEnable; -} - -void MainWnd::OnUpdateOptionsEmulatorRealtimeclock(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.winRtcEnable); -} - -void MainWnd::OnOptionsEmulatorAutohidemenu() -{ - theApp.autoHideMenu = !theApp.autoHideMenu; -} - -void MainWnd::OnUpdateOptionsEmulatorAutohidemenu(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.autoHideMenu); -} - -void MainWnd::OnOptionsEmulatorRewindinterval() -{ - RewindInterval dlg(theApp.rewindTimer/6); - int v = dlg.DoModal(); - - if(v >= 0) { - theApp.rewindTimer = v*6; // convert to a multiple of 10 frames - regSetDwordValue("rewindTimer", v); - if(v == 0) { - if(theApp.rewindMemory) - free(theApp.rewindMemory); - theApp.rewindMemory = NULL; - theApp.rewindCount = 0; - theApp.rewindCounter = 0; - theApp.rewindSaveNeeded = false; - } else { - if(theApp.rewindMemory == NULL) - theApp.rewindMemory = (char *)malloc(8*REWIND_SIZE); - } - } -} - -BOOL MainWnd::OnOptionsEmulatorShowSpeed(UINT nID) -{ - switch(nID) { - case ID_OPTIONS_EMULATOR_SHOWSPEED_NONE: - theApp.showSpeed = 0; - systemSetTitle("VisualBoyAdvance"); - break; - case ID_OPTIONS_EMULATOR_SHOWSPEED_PERCENTAGE: - theApp.showSpeed = 1; - break; - case ID_OPTIONS_EMULATOR_SHOWSPEED_DETAILED: - theApp.showSpeed = 2; - break; - case ID_OPTIONS_EMULATOR_SHOWSPEED_TRANSPARENT: - theApp.showSpeedTransparent = !theApp.showSpeedTransparent; - break; - default: - return FALSE; - } - return TRUE; -} - -void MainWnd::OnUpdateOptionsEmulatorShowSpeed(CCmdUI *pCmdUI) -{ - switch(pCmdUI->m_nID) { - case ID_OPTIONS_EMULATOR_SHOWSPEED_NONE: - pCmdUI->SetCheck(theApp.showSpeed == 0); - break; - case ID_OPTIONS_EMULATOR_SHOWSPEED_PERCENTAGE: - pCmdUI->SetCheck(theApp.showSpeed == 1); - break; - case ID_OPTIONS_EMULATOR_SHOWSPEED_DETAILED: - pCmdUI->SetCheck(theApp.showSpeed == 2); - break; - case ID_OPTIONS_EMULATOR_SHOWSPEED_TRANSPARENT: - pCmdUI->SetCheck(theApp.showSpeedTransparent); - break; - } -} - -void MainWnd::OnOptionsEmulatorSavetypeAutomatic() -{ - theApp.winSaveType = 0; -} - -void MainWnd::OnUpdateOptionsEmulatorSavetypeAutomatic(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.winSaveType == 0); -} - -void MainWnd::OnOptionsEmulatorSavetypeEeprom() -{ - theApp.winSaveType = 1; -} - -void MainWnd::OnUpdateOptionsEmulatorSavetypeEeprom(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.winSaveType == 1); -} - -void MainWnd::OnOptionsEmulatorSavetypeSram() -{ - theApp.winSaveType = 2; -} - -void MainWnd::OnUpdateOptionsEmulatorSavetypeSram(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.winSaveType == 2); -} - -void MainWnd::OnOptionsEmulatorSavetypeFlash() -{ - theApp.winSaveType = 3; -} - -void MainWnd::OnUpdateOptionsEmulatorSavetypeFlash(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.winSaveType == 3); -} - -void MainWnd::OnOptionsEmulatorSavetypeEepromsensor() -{ - theApp.winSaveType = 4; -} - -void MainWnd::OnUpdateOptionsEmulatorSavetypeEepromsensor(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.winSaveType == 4); -} - -void MainWnd::OnOptionsEmulatorSavetypeNone() -{ - theApp.winSaveType = 5; -} - -void MainWnd::OnUpdateOptionsEmulatorSavetypeNone(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.winSaveType == 5); -} - -void MainWnd::OnOptionsEmulatorSavetypeFlash512k() -{ - flashSetSize(0x10000); - theApp.winFlashSize = 0x10000; -} - -void MainWnd::OnUpdateOptionsEmulatorSavetypeFlash512k(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.winFlashSize == 0x10000); -} - -void MainWnd::OnOptionsEmulatorSavetypeFlash1m() -{ - flashSetSize(0x20000); - theApp.winFlashSize = 0x20000; -} - -void MainWnd::OnUpdateOptionsEmulatorSavetypeFlash1m(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.winFlashSize == 0x20000); -} - -void MainWnd::OnOptionsEmulatorUsebiosfile() -{ - if(!theApp.biosFileName.IsEmpty()) - theApp.useBiosFile = !theApp.useBiosFile; -} - -void MainWnd::OnUpdateOptionsEmulatorUsebiosfile(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.useBiosFile); - pCmdUI->Enable(!theApp.biosFileName.IsEmpty()); -} - -void MainWnd::OnOptionsEmulatorSkipbios() -{ - theApp.skipBiosFile = !theApp.skipBiosFile; -} - -void MainWnd::OnUpdateOptionsEmulatorSkipbios(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.skipBiosFile); -} - -void MainWnd::OnOptionsEmulatorSelectbiosfile() -{ - theApp.winCheckFullscreen(); - LPCTSTR exts[] = { "" }; - CString filter = winLoadFilter(IDS_FILTER_BIOS); - CString title = winResLoadString(IDS_SELECT_BIOS_FILE); - - FileDlg dlg(this, - theApp.biosFileName, - filter, - 0, - "BIOS", - exts, - "", - title, - false); - - if(dlg.DoModal() == IDOK) { - theApp.biosFileName = dlg.GetPathName(); - } -} - -void MainWnd::OnOptionsEmulatorPngformat() -{ - theApp.captureFormat = 0; -} - -void MainWnd::OnUpdateOptionsEmulatorPngformat(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.captureFormat == 0); -} - -void MainWnd::OnOptionsEmulatorBmpformat() -{ - theApp.captureFormat = 1; -} - -void MainWnd::OnUpdateOptionsEmulatorBmpformat(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.captureFormat == 1); -} - -void MainWnd::OnOptionsSoundOff() -{ - soundOffFlag = true; - soundShutdown(); -} - -void MainWnd::OnUpdateOptionsSoundOff(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(soundOffFlag); -} - -void MainWnd::OnOptionsSoundMute() -{ - soundDisable(0x30f); -} - -void MainWnd::OnUpdateOptionsSoundMute(CCmdUI* pCmdUI) -{ - int active = soundGetEnable() & 0x30f; - pCmdUI->SetCheck(active == 0); -} - -void MainWnd::OnOptionsSoundOn() -{ - //if(soundOffFlag) { - soundOffFlag = false; - //soundInit(!theApp.cartridgeType); - //} - soundEnable(0x30f); -} - -void MainWnd::OnUpdateOptionsSoundOn(CCmdUI* pCmdUI) -{ - //int active = soundGetEnable() & 0x30f; - pCmdUI->SetCheck(/*active != 0 &&*/ !soundOffFlag); -} - -void MainWnd::OnOptionsSoundUseoldsynchronization() -{ - theApp.useOldSync = !theApp.useOldSync; - systemMessage(IDS_SETTING_WILL_BE_EFFECTIVE, - "Setting will be effective the next time you start the emulator"); -} - -void MainWnd::OnUpdateOptionsSoundUseoldsynchronization(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.useOldSync); -} - -void MainWnd::OnOptionsSoundEcho() -{ - soundEcho = !soundEcho; -} - -void MainWnd::OnUpdateOptionsSoundEcho(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(soundEcho); -} - -void MainWnd::OnOptionsSoundLowpassfilter() -{ - soundLowPass = !soundLowPass; -} - -void MainWnd::OnUpdateOptionsSoundLowpassfilter(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(soundLowPass); -} - -void MainWnd::OnOptionsSoundReversestereo() -{ - soundReverse = !soundReverse; -} - -void MainWnd::OnUpdateOptionsSoundReversestereo(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(soundReverse); -} - -void MainWnd::OnOptionsSound11khz() -{ - if(theApp.cartridgeType == 0) - soundSetQuality(4); - else - gbSoundSetQuality(4); -} - -void MainWnd::OnUpdateOptionsSound11khz(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(soundQuality == 4); -} - -void MainWnd::OnOptionsSound22khz() -{ - if(theApp.cartridgeType == 0) - soundSetQuality(2); - else - gbSoundSetQuality(2); -} - -void MainWnd::OnUpdateOptionsSound22khz(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(soundQuality == 2); -} - -void MainWnd::OnOptionsSound44khz() -{ - if(theApp.cartridgeType == 0) - soundSetQuality(1); - else - gbSoundSetQuality(1); -} - -void MainWnd::OnUpdateOptionsSound44khz(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(soundQuality == 1); -} - -BOOL MainWnd::OnOptionsSoundVolume(UINT nID) -{ - soundVolume = nID - ID_OPTIONS_SOUND_VOLUME_1X; - return TRUE; -} - -void MainWnd::OnUpdateOptionsSoundVolume(CCmdUI *pCmdUI) -{ - pCmdUI->SetCheck(soundVolume == (int)(pCmdUI->m_nID - ID_OPTIONS_SOUND_VOLUME_1X)); -} - - -void MainWnd::OnOptionsSoundVolume25x() -{ - soundVolume = 4; -} - -void MainWnd::OnUpdateOptionsSoundVolume25x(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(soundVolume == 4); -} - -void MainWnd::OnOptionsSoundVolume5x() -{ - soundVolume = 5; -} - -void MainWnd::OnUpdateOptionsSoundVolume5x(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(soundVolume == 5); -} - -void MainWnd::updateSoundChannels(UINT id) -{ - int flag = 0; - - if ( soundOffFlag ) return; // mute hax - - if(id == ID_OPTIONS_SOUND_CHANNEL1) - flag = 1; - - if(id == ID_OPTIONS_SOUND_CHANNEL2) - flag = 2; - - if(id == ID_OPTIONS_SOUND_CHANNEL3) - flag = 4; - - if(id == ID_OPTIONS_SOUND_CHANNEL4) - flag = 8; - - if(id == ID_OPTIONS_SOUND_DIRECTSOUNDA) - flag = 256; - - if(id == ID_OPTIONS_SOUND_DIRECTSOUNDB) - flag = 512; - - int active = soundGetEnable() & 0x30f; - - if(active & flag) - active &= (~flag); - else - active |= flag; - - soundEnable(active); - soundDisable((~active)&0x30f); -} - -void MainWnd::OnOptionsSoundChannel1() -{ - updateSoundChannels(ID_OPTIONS_SOUND_CHANNEL1); -} - -void MainWnd::OnUpdateOptionsSoundChannel1(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(soundGetEnable() & 1); -} - -void MainWnd::OnOptionsSoundChannel2() -{ - updateSoundChannels(ID_OPTIONS_SOUND_CHANNEL2); -} - -void MainWnd::OnUpdateOptionsSoundChannel2(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(soundGetEnable() & 2); -} - -void MainWnd::OnOptionsSoundChannel3() -{ - updateSoundChannels(ID_OPTIONS_SOUND_CHANNEL3); -} - -void MainWnd::OnUpdateOptionsSoundChannel3(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(soundGetEnable() & 4); -} - -void MainWnd::OnOptionsSoundChannel4() -{ - updateSoundChannels(ID_OPTIONS_SOUND_CHANNEL4); -} - -void MainWnd::OnUpdateOptionsSoundChannel4(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(soundGetEnable() & 8); -} - -void MainWnd::OnOptionsSoundDirectsounda() -{ - updateSoundChannels(ID_OPTIONS_SOUND_DIRECTSOUNDA); -} - -void MainWnd::OnUpdateOptionsSoundDirectsounda(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(soundGetEnable() & 256); - pCmdUI->Enable(theApp.cartridgeType == 0); -} - -void MainWnd::OnOptionsSoundDirectsoundb() -{ - updateSoundChannels(ID_OPTIONS_SOUND_DIRECTSOUNDB); -} - -void MainWnd::OnUpdateOptionsSoundDirectsoundb(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(soundGetEnable() & 512); - pCmdUI->Enable(theApp.cartridgeType == 0); -} - -BOOL MainWnd::OnOptionsSoundPcminterpolation(UINT nID) -{ - switch (nID) - { - case ID_OPTIONS_SOUND_PCMINTERPOLATION_NONE: - soundInterpolation = 0; - break; - case ID_OPTIONS_SOUND_PCMINTERPOLATION_LINEAR: - soundInterpolation = 1; - break; - case ID_OPTIONS_SOUND_PCMINTERPOLATION_CUBIC: - soundInterpolation = 2; - break; - case ID_OPTIONS_SOUND_PCMINTERPOLATION_FIR: - soundInterpolation = 3; - break; - case ID_OPTIONS_SOUND_PCMINTERPOLATION_LIBRESAMPLE: - soundInterpolation = 4; - break; - - default: - return FALSE; - } - - return TRUE; -} - -void MainWnd::OnUpdateOptionsSoundPcminterpolation(CCmdUI *pCmdUI) -{ - switch (pCmdUI->m_nID) - { - case ID_OPTIONS_SOUND_PCMINTERPOLATION_NONE: - pCmdUI->SetCheck(soundInterpolation == 0); - break; - case ID_OPTIONS_SOUND_PCMINTERPOLATION_LINEAR: - pCmdUI->SetCheck(soundInterpolation == 1); - break; - case ID_OPTIONS_SOUND_PCMINTERPOLATION_CUBIC: - pCmdUI->SetCheck(soundInterpolation == 2); - break; - case ID_OPTIONS_SOUND_PCMINTERPOLATION_FIR: - pCmdUI->SetCheck(soundInterpolation == 3); - break; - case ID_OPTIONS_SOUND_PCMINTERPOLATION_LIBRESAMPLE: - pCmdUI->SetCheck(soundInterpolation == 4); - break; - - default: - return; - } - pCmdUI->Enable(theApp.cartridgeType == 0); -} - -void MainWnd::OnOptionsGameboyBorder() -{ - theApp.winGbBorderOn = !theApp.winGbBorderOn; - gbBorderOn = theApp.winGbBorderOn; - if(emulating && theApp.cartridgeType == 1 && gbBorderOn) { - gbSgbRenderBorder(); - } - theApp.updateWindowSize(theApp.videoOption); -} - -void MainWnd::OnUpdateOptionsGameboyBorder(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.winGbBorderOn); -} - -void MainWnd::OnOptionsGameboyPrinter() -{ - theApp.winGbPrinterEnabled = !theApp.winGbPrinterEnabled; - if(theApp.winGbPrinterEnabled) - gbSerialFunction = gbPrinterSend; - else - gbSerialFunction = NULL; -} - -void MainWnd::OnUpdateOptionsGameboyPrinter(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(gbSerialFunction == gbPrinterSend); -} - -void MainWnd::OnOptionsGameboyBorderAutomatic() -{ - gbBorderAutomatic = !gbBorderAutomatic; - if(emulating && theApp.cartridgeType == 1 && gbBorderOn) { - gbSgbRenderBorder(); - theApp.updateWindowSize(theApp.videoOption); - } -} - -void MainWnd::OnUpdateOptionsGameboyBorderAutomatic(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(gbBorderAutomatic); -} - -void MainWnd::OnOptionsGameboyAutomatic() -{ - gbEmulatorType = 0; -} - -void MainWnd::OnUpdateOptionsGameboyAutomatic(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(gbEmulatorType == 0); -} - -void MainWnd::OnOptionsGameboyGba() -{ - gbEmulatorType = 4; -} - -void MainWnd::OnUpdateOptionsGameboyGba(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(gbEmulatorType == 4); -} - -void MainWnd::OnOptionsGameboyCgb() -{ - gbEmulatorType = 1; -} - -void MainWnd::OnUpdateOptionsGameboyCgb(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(gbEmulatorType == 1); -} - -void MainWnd::OnOptionsGameboySgb() -{ - gbEmulatorType = 2; -} - -void MainWnd::OnUpdateOptionsGameboySgb(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(gbEmulatorType == 2); -} - -void MainWnd::OnOptionsGameboySgb2() -{ - gbEmulatorType = 5; -} - -void MainWnd::OnUpdateOptionsGameboySgb2(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(gbEmulatorType == 5); -} - -void MainWnd::OnOptionsGameboyGb() -{ - gbEmulatorType = 3; -} - -void MainWnd::OnUpdateOptionsGameboyGb(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(gbEmulatorType == 3); -} - -void MainWnd::OnOptionsGameboyRealcolors() -{ - gbColorOption = 0; -} - -void MainWnd::OnUpdateOptionsGameboyRealcolors(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(gbColorOption == 0); -} - -void MainWnd::OnOptionsGameboyGameboycolors() -{ - gbColorOption = 1; -} - -void MainWnd::OnUpdateOptionsGameboyGameboycolors(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(gbColorOption == 1); -} - - -void MainWnd::OnOptionsGameboyColors() -{ - theApp.winCheckFullscreen(); - GBColorDlg dlg; - if(dlg.DoModal()) { - gbPaletteOption = dlg.getWhich(); - memcpy(systemGbPalette, dlg.getColors(), 24*sizeof(u16)); - if(emulating && theApp.cartridgeType == 1) { - memcpy(gbPalette, &systemGbPalette[dlg.getWhich()*8], 8*sizeof(u16)); - } - } -} - -BOOL MainWnd::OnOptionsPriority(UINT nID) -{ - switch(nID) { - case ID_OPTIONS_PRIORITY_HIGHEST: - theApp.threadPriority = 0; - break; - case ID_OPTIONS_PRIORITY_ABOVENORMAL: - theApp.threadPriority = 1; - break; - case ID_OPTIONS_PRIORITY_NORMAL: - theApp.threadPriority = 2; - break; - case ID_OPTIONS_PRIORITY_BELOWNORMAL: - theApp.threadPriority = 3; - break; - default: - return FALSE; - } - theApp.updatePriority(); - - return TRUE; -} - -void MainWnd::OnUpdateOptionsPriority(CCmdUI *pCmdUI) -{ - switch(pCmdUI->m_nID) { - case ID_OPTIONS_PRIORITY_HIGHEST: - pCmdUI->SetCheck(theApp.threadPriority == 0); - break; - case ID_OPTIONS_PRIORITY_ABOVENORMAL: - pCmdUI->SetCheck(theApp.threadPriority == 1); - break; - case ID_OPTIONS_PRIORITY_NORMAL: - pCmdUI->SetCheck(theApp.threadPriority == 2); - break; - case ID_OPTIONS_PRIORITY_BELOWNORMAL: - pCmdUI->SetCheck(theApp.threadPriority == 3); - break; - } -} - -BOOL MainWnd::OnOptionsFilter(UINT nID) -{ - switch(nID) - { - case ID_OPTIONS_FILTER_NORMAL: - theApp.filterType = FILTER_NONE; - break; - case ID_OPTIONS_FILTER_TVMODE: - theApp.filterType = FILTER_TVMODE; - break; - case ID_OPTIONS_FILTER_2XSAI: - theApp.filterType = FILTER_2XSAI; - break; - case ID_OPTIONS_FILTER_SUPER2XSAI: - theApp.filterType = FILTER_SUPER2XSAI; - break; - case ID_OPTIONS_FILTER_SUPEREAGLE: - theApp.filterType = FILTER_SUPEREAGLE; - break; - case ID_OPTIONS_FILTER16BIT_PIXELATEEXPERIMENTAL: - theApp.filterType = FILTER_PIXELATE; - break; - case ID_OPTIONS_FILTER16BIT_ADVANCEMAMESCALE2X: - theApp.filterType = FILTER_MAMESCALE2X; - break; - case ID_OPTIONS_FILTER16BIT_SIMPLE2X: - theApp.filterType = FILTER_SIMPLE2X; - break; - case ID_OPTIONS_FILTER_BILINEAR: - theApp.filterType = FILTER_BILINEAR; - break; - case ID_OPTIONS_FILTER_BILINEARPLUS: - theApp.filterType = FILTER_BILINEARPLUS; - break; - case ID_OPTIONS_FILTER_SCANLINES: - theApp.filterType = FILTER_SCANLINES; - break; - case ID_OPTIONS_FILTER_HQ2X: - theApp.filterType = FILTER_HQ2X; - break; - case ID_OPTIONS_FILTER_LQ2X: - theApp.filterType = FILTER_LQ2X; - break; - case ID_OPTIONS_FILTER_SIMPLE3X: - theApp.filterType = FILTER_SIMPLE3X; - break; - case ID_OPTIONS_FILTER_SIMPLE4X: - theApp.filterType = FILTER_SIMPLE4X; - break; - case ID_OPTIONS_FILTER_HQ3X: - theApp.filterType = FILTER_HQ3X; - break; - case ID_OPTIONS_FILTER_HQ4X: - theApp.filterType = FILTER_HQ4X; - break; - default: - return FALSE; - } - theApp.updateFilter(); - return TRUE; -} - -void MainWnd::OnUpdateOptionsFilter(CCmdUI *pCmdUI) -{ - pCmdUI->Enable( systemColorDepth == 16 || systemColorDepth == 32 ); - - switch(pCmdUI->m_nID) { - case ID_OPTIONS_FILTER_NORMAL: - pCmdUI->SetCheck(theApp.filterType == FILTER_NONE); - break; - case ID_OPTIONS_FILTER_TVMODE: - pCmdUI->SetCheck(theApp.filterType == FILTER_TVMODE); - break; - case ID_OPTIONS_FILTER_2XSAI: - pCmdUI->SetCheck(theApp.filterType == FILTER_2XSAI); - break; - case ID_OPTIONS_FILTER_SUPER2XSAI: - pCmdUI->SetCheck(theApp.filterType == FILTER_SUPER2XSAI); - break; - case ID_OPTIONS_FILTER_SUPEREAGLE: - pCmdUI->SetCheck(theApp.filterType == FILTER_SUPEREAGLE); - break; - case ID_OPTIONS_FILTER16BIT_PIXELATEEXPERIMENTAL: - pCmdUI->SetCheck(theApp.filterType == FILTER_PIXELATE); - break; - case ID_OPTIONS_FILTER16BIT_ADVANCEMAMESCALE2X: - pCmdUI->SetCheck(theApp.filterType == FILTER_MAMESCALE2X); - break; - case ID_OPTIONS_FILTER16BIT_SIMPLE2X: - pCmdUI->SetCheck(theApp.filterType == FILTER_SIMPLE2X); - break; - case ID_OPTIONS_FILTER_BILINEAR: - pCmdUI->SetCheck(theApp.filterType == FILTER_BILINEAR); - break; - case ID_OPTIONS_FILTER_BILINEARPLUS: - pCmdUI->SetCheck(theApp.filterType == FILTER_BILINEARPLUS); - break; - case ID_OPTIONS_FILTER_SCANLINES: - pCmdUI->SetCheck(theApp.filterType == FILTER_SCANLINES); - break; - case ID_OPTIONS_FILTER_HQ2X: - pCmdUI->SetCheck(theApp.filterType == FILTER_HQ2X); - break; - case ID_OPTIONS_FILTER_LQ2X: - pCmdUI->SetCheck(theApp.filterType == FILTER_LQ2X); - break; - case ID_OPTIONS_FILTER_SIMPLE3X: - pCmdUI->SetCheck(theApp.filterType == FILTER_SIMPLE3X); - break; - case ID_OPTIONS_FILTER_SIMPLE4X: - pCmdUI->SetCheck(theApp.filterType == FILTER_SIMPLE4X); - break; - case ID_OPTIONS_FILTER_HQ3X: - pCmdUI->SetCheck(theApp.filterType == FILTER_HQ3X); - break; - case ID_OPTIONS_FILTER_HQ4X: - pCmdUI->SetCheck(theApp.filterType == FILTER_HQ4X); - break; - } -} - -BOOL MainWnd::OnOptionsFilterIFB(UINT nID) -{ - switch(nID) { - case ID_OPTIONS_FILTER_INTERFRAMEBLENDING_NONE: - theApp.ifbType = 0; - break; - case ID_OPTIONS_FILTER_INTERFRAMEBLENDING_MOTIONBLUR: - theApp.ifbType = 1; - break; - case ID_OPTIONS_FILTER_INTERFRAMEBLENDING_SMART: - theApp.ifbType = 2; - break; - default: - return FALSE; - } - theApp.updateIFB(); - return TRUE; -} - -void MainWnd::OnUpdateOptionsFilterIFB(CCmdUI *pCmdUI) -{ - switch(pCmdUI->m_nID) { - case ID_OPTIONS_FILTER_INTERFRAMEBLENDING_NONE: - pCmdUI->SetCheck(theApp.ifbType == 0); - break; - case ID_OPTIONS_FILTER_INTERFRAMEBLENDING_MOTIONBLUR: - pCmdUI->SetCheck(theApp.ifbType == 1); - break; - case ID_OPTIONS_FILTER_INTERFRAMEBLENDING_SMART: - pCmdUI->SetCheck(theApp.ifbType == 2); - break; - } -} - -void MainWnd::OnOptionsFilterDisablemmx() -{ -#ifdef MMX - theApp.disableMMX = !theApp.disableMMX; - if(!theApp.disableMMX) - cpu_mmx = theApp.detectMMX(); - else - cpu_mmx = 0; -#endif -} - -void MainWnd::OnUpdateOptionsFilterDisablemmx(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.disableMMX); -} - -void MainWnd::OnOptionsFilterLcdcolors() -{ - theApp.filterLCD = !theApp.filterLCD; - utilUpdateSystemColorMaps(theApp.filterLCD); -} - -void MainWnd::OnUpdateOptionsFilterLcdcolors(CCmdUI *pCmdUI) -{ - pCmdUI->SetCheck(theApp.filterLCD); -} - -void MainWnd::OnOptionsLanguageSystem() -{ - theApp.winSetLanguageOption(0, false); - theApp.winAccelMgr.UpdateMenu(theApp.menu); -} - -void MainWnd::OnUpdateOptionsLanguageSystem(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.languageOption == 0); -} - -void MainWnd::OnOptionsLanguageEnglish() -{ - theApp.winSetLanguageOption(1, false); - theApp.winAccelMgr.UpdateMenu(theApp.menu); -} - -void MainWnd::OnUpdateOptionsLanguageEnglish(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.languageOption == 1); -} - -void MainWnd::OnOptionsLanguageOther() -{ - theApp.winCheckFullscreen(); - theApp.winSetLanguageOption(2, false); - theApp.winAccelMgr.UpdateMenu(theApp.menu); -} - -void MainWnd::OnUpdateOptionsLanguageOther(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(theApp.languageOption == 2); -} - - -void MainWnd::OnOptionsJoypadConfigure1() -{ - theApp.winCheckFullscreen(); - JoypadConfig dlg(0); - dlg.DoModal(); -} - -void MainWnd::OnUpdateOptionsJoypadConfigure1(CCmdUI* pCmdUI) -{ - pCmdUI->Enable(theApp.videoOption != VIDEO_320x240); -} - -void MainWnd::OnOptionsJoypadConfigure2() -{ - theApp.winCheckFullscreen(); - JoypadConfig dlg(1); - dlg.DoModal(); -} - -void MainWnd::OnUpdateOptionsJoypadConfigure2(CCmdUI* pCmdUI) -{ - pCmdUI->Enable(theApp.videoOption != VIDEO_320x240); -} - -void MainWnd::OnOptionsJoypadConfigure3() -{ - theApp.winCheckFullscreen(); - JoypadConfig dlg(2); - dlg.DoModal(); -} - -void MainWnd::OnUpdateOptionsJoypadConfigure3(CCmdUI* pCmdUI) -{ - pCmdUI->Enable(theApp.videoOption != VIDEO_320x240); -} - -void MainWnd::OnOptionsJoypadConfigure4() -{ - theApp.winCheckFullscreen(); - JoypadConfig dlg(3); - dlg.DoModal(); -} - -void MainWnd::OnUpdateOptionsJoypadConfigure4(CCmdUI* pCmdUI) -{ - pCmdUI->Enable(theApp.videoOption != VIDEO_320x240); -} - -BOOL MainWnd::OnOptionsJoypadDefault(UINT nID) -{ - theApp.joypadDefault = nID - ID_OPTIONS_JOYPAD_DEFAULTJOYPAD_1; - return TRUE; -} - -void MainWnd::OnUpdateOptionsJoypadDefault(CCmdUI *pCmdUI) -{ - pCmdUI->SetCheck(theApp.joypadDefault == (int)(pCmdUI->m_nID - ID_OPTIONS_JOYPAD_DEFAULTJOYPAD_1)); -} - -void MainWnd::OnOptionsJoypadMotionconfigure() -{ - theApp.winCheckFullscreen(); - MotionConfig dlg; - dlg.DoModal(); -} - -void MainWnd::OnUpdateOptionsJoypadMotionconfigure(CCmdUI* pCmdUI) -{ - pCmdUI->Enable(theApp.videoOption != VIDEO_320x240); -} - -BOOL MainWnd::OnOptionsJoypadAutofire(UINT nID) -{ - switch(nID) { - case ID_OPTIONS_JOYPAD_AUTOFIRE_A: - if(theApp.autoFire & 1) { - theApp.autoFire &= ~1; - systemScreenMessage(winResLoadString(IDS_AUTOFIRE_A_DISABLED)); - } else { - theApp.autoFire |= 1; - systemScreenMessage(winResLoadString(IDS_AUTOFIRE_A)); - } - break; - case ID_OPTIONS_JOYPAD_AUTOFIRE_B: - if(theApp.autoFire & 2) { - theApp.autoFire &= ~2; - systemScreenMessage(winResLoadString(IDS_AUTOFIRE_B_DISABLED)); - } else { - theApp.autoFire |= 2; - systemScreenMessage(winResLoadString(IDS_AUTOFIRE_B)); - } - break; - case ID_OPTIONS_JOYPAD_AUTOFIRE_L: - if(theApp.autoFire & 512) { - theApp.autoFire &= ~512; - systemScreenMessage(winResLoadString(IDS_AUTOFIRE_L_DISABLED)); - } else { - theApp.autoFire |= 512; - systemScreenMessage(winResLoadString(IDS_AUTOFIRE_L)); - } - break; - case ID_OPTIONS_JOYPAD_AUTOFIRE_R: - if(theApp.autoFire & 256) { - theApp.autoFire &= ~256; - systemScreenMessage(winResLoadString(IDS_AUTOFIRE_R_DISABLED)); - } else { - theApp.autoFire |= 256; - systemScreenMessage(winResLoadString(IDS_AUTOFIRE_R)); - } - break; - default: - return FALSE; - } - return TRUE; -} - -void MainWnd::OnUpdateOptionsJoypadAutofire(CCmdUI *pCmdUI) -{ - bool check = true; - switch(pCmdUI->m_nID) { - case ID_OPTIONS_JOYPAD_AUTOFIRE_A: - check = (theApp.autoFire & 1) != 0; - break; - case ID_OPTIONS_JOYPAD_AUTOFIRE_B: - check = (theApp.autoFire & 2) != 0; - break; - case ID_OPTIONS_JOYPAD_AUTOFIRE_L: - check = (theApp.autoFire & 512) != 0; - break; - case ID_OPTIONS_JOYPAD_AUTOFIRE_R: - check = (theApp.autoFire & 256) != 0; - break; - } - pCmdUI->SetCheck(check); -} - -LRESULT MainWnd::OnConfirmMode(WPARAM, LPARAM) -{ - // we need to do this separately or the window will not have the right - // parent. must be related to the way MFC does modal dialogs - winConfirmMode(); - return 0; -} - -void MainWnd::winConfirmMode() -{ - if(theApp.renderMethod == DIRECT_DRAW && theApp.videoOption > VIDEO_4X) { - theApp.winCheckFullscreen(); - ModeConfirm dlg(theApp.m_pMainWnd); - - if(!dlg.DoModal()) { - theApp.updateVideoSize(ID_OPTIONS_VIDEO_X2); - } - } - theApp.winAccelMgr.UpdateMenu(theApp.menu); -} - -void MainWnd::OnOptionsVideoFullscreenmaxscale() -{ - MaxScale dlg; - - theApp.winCheckFullscreen(); - - dlg.DoModal(); -} - - -void MainWnd::OnLinkOptions() -{ - LinkOptions dlg; - - dlg.DoModal(); -} - -void MainWnd::OnOptionsLinkLog() -{ - if(linklog){ - if(linklogfile!=NULL) fclose(linklogfile); - linklog = 0; - linklogfile = NULL; - } else { - linklog=1; - openLinkLog(); - } -} - -void MainWnd::OnUpdateOptionsLinkLog(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(linklog); -} - -void MainWnd::OnOptionsLinkRFU() -{ - if(adapter) adapter = false; - else { - adapter = true; - MessageBox("Please note this is the first version\nof RFU emulation code and it's not 100% bug free.\nAlso only 2 players single computer are supported at this time.", "Warning", MB_OK); - } -} - -void MainWnd::OnUpdateOptionsLinkRFU(CCmdUI* pCmdUI) -{ - pCmdUI->SetCheck(adapter); -} \ No newline at end of file +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#include "stdafx.h" +#include "MainWnd.h" + +#include "Associate.h" +#include "Directories.h" +#include "FileDlg.h" +#include "GameOverrides.h" +#include "LinkOptions.h" +#include "GBColorDlg.h" +#include "Joypad.h" +#include "MaxScale.h" +#include "ModeConfirm.h" +#include "Reg.h" +#include "RewindInterval.h" +#include "skin.h" +#include "Throttle.h" +#include "WinResUtil.h" + +#include "../System.h" +#include "../agbprint.h" +#include "../GBA.h" +#include "../Globals.h" +#include "../Sound.h" +#include "../gb/GB.h" +#include "../gb/gbGlobals.h" +#include "../gb/gbPrinter.h" +#include "../Link.h" +#include + +extern int emulating; + +#define VBA_CONFIRM_MODE WM_APP + 100 + +extern void CPUUpdateRenderBuffers(bool force); + +void MainWnd::OnOptionsFrameskipThrottleNothrottle() +{ + theApp.throttle = 0; + theApp.autoFrameSkip = false; +} + +void MainWnd::OnUpdateOptionsFrameskipThrottleNothrottle(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.throttle == 0); +} + +void MainWnd::OnOptionsFrameskipThrottle25() +{ + theApp.throttle = 25; + theApp.autoFrameSkip = false; +} + +void MainWnd::OnUpdateOptionsFrameskipThrottle25(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.throttle == 25); +} + +void MainWnd::OnOptionsFrameskipThrottle50() +{ + theApp.throttle = 50; + theApp.autoFrameSkip = false; +} + +void MainWnd::OnUpdateOptionsFrameskipThrottle50(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.throttle == 50); +} + +void MainWnd::OnOptionsFrameskipThrottle100() +{ + theApp.throttle = 100; + theApp.autoFrameSkip = false; +} + +void MainWnd::OnUpdateOptionsFrameskipThrottle100(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.throttle == 100); +} + +void MainWnd::OnOptionsFrameskipThrottle150() +{ + theApp.throttle = 150; + theApp.autoFrameSkip = false; +} + +void MainWnd::OnUpdateOptionsFrameskipThrottle150(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.throttle == 150); +} + +void MainWnd::OnOptionsFrameskipThrottle200() +{ + theApp.throttle = 200; + theApp.autoFrameSkip = false; +} + +void MainWnd::OnUpdateOptionsFrameskipThrottle200(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.throttle == 200); +} + + +void MainWnd::OnOptionsFrameskipThrottleOther() +{ + Throttle dlg; + int v = (int)dlg.DoModal(); + + if( v ) { + theApp.throttle = v; + theApp.autoFrameSkip = false; + } +} + + +void MainWnd::OnUpdateOptionsFrameskipThrottleOther(CCmdUI* pCmdUI) +{ + int throttle = theApp.throttle; + pCmdUI->SetCheck(throttle != 0 && throttle != 25 && + throttle != 50 && throttle != 100 && + throttle != 150 && throttle != 200); +} + +void MainWnd::OnOptionsFrameskipAutomatic() +{ + theApp.autoFrameSkip = !theApp.autoFrameSkip; + if(!theApp.autoFrameSkip && emulating) + theApp.updateFrameSkip(); + else + { + theApp.throttle = false; + frameSkip = 0; + systemFrameSkip = 0; + } +} + +void MainWnd::OnUpdateOptionsFrameskipAutomatic(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.autoFrameSkip); +} + +BOOL MainWnd::OnOptionsFrameskip(UINT nID) +{ + switch(nID) { + case ID_OPTIONS_VIDEO_FRAMESKIP_0: + case ID_OPTIONS_VIDEO_FRAMESKIP_1: + case ID_OPTIONS_VIDEO_FRAMESKIP_2: + case ID_OPTIONS_VIDEO_FRAMESKIP_3: + case ID_OPTIONS_VIDEO_FRAMESKIP_4: + case ID_OPTIONS_VIDEO_FRAMESKIP_5: + if(theApp.cartridgeType == IMAGE_GBA) { + frameSkip = nID - ID_OPTIONS_VIDEO_FRAMESKIP_0; + } else { + gbFrameSkip = nID - ID_OPTIONS_VIDEO_FRAMESKIP_0; + } + if(emulating) + theApp.updateFrameSkip(); + theApp.autoFrameSkip = false; + return TRUE; + break; + case ID_OPTIONS_VIDEO_FRAMESKIP_6: + case ID_OPTIONS_VIDEO_FRAMESKIP_7: + case ID_OPTIONS_VIDEO_FRAMESKIP_8: + case ID_OPTIONS_VIDEO_FRAMESKIP_9: + if(theApp.cartridgeType == IMAGE_GBA) { + frameSkip = 6 + nID - ID_OPTIONS_VIDEO_FRAMESKIP_6; + } else { + gbFrameSkip = 6 + nID - ID_OPTIONS_VIDEO_FRAMESKIP_6; + } + if(emulating) + theApp.updateFrameSkip(); + theApp.autoFrameSkip = false; + return TRUE; + break; + } + return FALSE; +} + +void MainWnd::OnUpdateOptionsVideoFrameskip0(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.cartridgeType == IMAGE_GBA ? frameSkip == 0 : gbFrameSkip == 0); +} + +void MainWnd::OnUpdateOptionsVideoFrameskip1(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.cartridgeType == IMAGE_GBA ? frameSkip == 1 : gbFrameSkip == 1); +} + +void MainWnd::OnUpdateOptionsVideoFrameskip2(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.cartridgeType == IMAGE_GBA ? frameSkip == 2 : gbFrameSkip == 2); +} + +void MainWnd::OnUpdateOptionsVideoFrameskip3(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.cartridgeType == IMAGE_GBA ? frameSkip == 3 : gbFrameSkip == 3); +} + +void MainWnd::OnUpdateOptionsVideoFrameskip4(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.cartridgeType == IMAGE_GBA ? frameSkip == 4 : gbFrameSkip == 4); +} + +void MainWnd::OnUpdateOptionsVideoFrameskip5(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.cartridgeType == IMAGE_GBA ? frameSkip == 5 : gbFrameSkip == 5); +} + +void MainWnd::OnUpdateOptionsVideoFrameskip6(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.cartridgeType == IMAGE_GBA ? frameSkip == 6 : gbFrameSkip == 6); +} + +void MainWnd::OnUpdateOptionsVideoFrameskip7(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.cartridgeType == IMAGE_GBA ? frameSkip == 7 : gbFrameSkip == 7); +} + +void MainWnd::OnUpdateOptionsVideoFrameskip8(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.cartridgeType == IMAGE_GBA ? frameSkip == 8 : gbFrameSkip == 8); +} + +void MainWnd::OnUpdateOptionsVideoFrameskip9(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.cartridgeType == IMAGE_GBA ? frameSkip == 9 : gbFrameSkip == 9); +} + + +void MainWnd::OnOptionsVideoVsync() +{ + theApp.vsync = !theApp.vsync; + if( theApp.display ) { + theApp.display->setOption( _T("vsync"), theApp.vsync ); + } +} + + +void MainWnd::OnUpdateOptionsVideoVsync(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.vsync); +} + +void MainWnd::OnUpdateOptionsVideoX1(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.videoOption == VIDEO_1X); +} + +void MainWnd::OnUpdateOptionsVideoX2(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.videoOption == VIDEO_2X); +} + +void MainWnd::OnUpdateOptionsVideoX3(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.videoOption == VIDEO_3X); +} + +void MainWnd::OnUpdateOptionsVideoX4(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.videoOption == VIDEO_4X); +} + +void MainWnd::OnUpdateOptionsVideoFullscreen320x240(CCmdUI* pCmdUI) +{ + pCmdUI->Enable(theApp.mode320Available); + pCmdUI->SetCheck(theApp.videoOption == VIDEO_320x240); +} + +void MainWnd::OnUpdateOptionsVideoFullscreen640x480(CCmdUI* pCmdUI) +{ + pCmdUI->Enable(theApp.mode640Available); + pCmdUI->SetCheck(theApp.videoOption == VIDEO_640x480); +} + +void MainWnd::OnUpdateOptionsVideoFullscreen800x600(CCmdUI* pCmdUI) +{ + pCmdUI->Enable(theApp.mode800Available); + pCmdUI->SetCheck(theApp.videoOption == VIDEO_800x600); +} + +void MainWnd::OnUpdateOptionsVideoFullscreen1024x768(CCmdUI *pCmdUI) +{ + pCmdUI->Enable(theApp.mode1024Available); + pCmdUI->SetCheck(theApp.videoOption == VIDEO_1024x768); +} + +void MainWnd::OnUpdateOptionsVideoFullscreen1280x1024(CCmdUI *pCmdUI) +{ + pCmdUI->Enable(theApp.mode1280Available); + pCmdUI->SetCheck(theApp.videoOption == VIDEO_1280x1024); +} + +BOOL MainWnd::OnOptionVideoSize(UINT nID) +{ + theApp.updateVideoSize(nID); + theApp.m_pMainWnd->PostMessage(VBA_CONFIRM_MODE); + return TRUE; +} + + +void MainWnd::OnOptionsVideoFullscreen320x240() +{ + OnOptionVideoSize(ID_OPTIONS_VIDEO_FULLSCREEN320X240); +} + +void MainWnd::OnOptionsVideoFullscreen640x480() +{ + OnOptionVideoSize(ID_OPTIONS_VIDEO_FULLSCREEN640X480); +} + +void MainWnd::OnOptionsVideoFullscreen800x600() +{ + OnOptionVideoSize(ID_OPTIONS_VIDEO_FULLSCREEN800X600); +} + +void MainWnd::OnOptionsVideoFullscreen1024x768() +{ + OnOptionVideoSize(ID_OPTIONS_VIDEO_FULLSCREEN1024X768); +} + +void MainWnd::OnOptionsVideoFullscreen1280x1024() +{ + OnOptionVideoSize(ID_OPTIONS_VIDEO_FULLSCREEN1280X1024); +} + +void MainWnd::OnOptionsVideoFullscreen() +{ + theApp.winCheckFullscreen(); + GUID *pGUID = NULL; + int size = theApp.display->selectFullScreenMode(&pGUID); + if(size != -1) { + int width = (size >> 12) & 4095; + int height = (size & 4095); + int colorDepth = (size >> 24); + if(width != theApp.fsWidth || + height != theApp.fsHeight || + colorDepth != theApp.fsColorDepth || + pGUID != theApp.pVideoDriverGUID || + theApp.videoOption != VIDEO_OTHER) { + theApp.fsForceChange = true; + theApp.fsWidth = width; + theApp.fsHeight = height; + theApp.fsFrequency = 60; + theApp.fsColorDepth = colorDepth; + theApp.pVideoDriverGUID = pGUID; + if(pGUID) { + theApp.videoDriverGUID = *pGUID; + regSetDwordValue("defaultVideoDriver", FALSE); + regSetBinaryValue("videoDriverGUID", + (char *)pGUID, sizeof(GUID)); + } else { + regSetDwordValue("defaultVideoDriver", TRUE); + } + theApp.updateVideoSize(ID_OPTIONS_VIDEO_FULLSCREEN); + theApp.m_pMainWnd->PostMessage(VBA_CONFIRM_MODE); + } + } + theApp.winAccelMgr.UpdateMenu(theApp.menu); +} + +void MainWnd::OnUpdateOptionsVideoFullscreen(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.videoOption == VIDEO_OTHER); +} + +void MainWnd::OnOptionsVideoDisablesfx() +{ + cpuDisableSfx = !cpuDisableSfx; + if(emulating && theApp.cartridgeType == IMAGE_GBA) + CPUUpdateRender(); +} + +void MainWnd::OnUpdateOptionsVideoDisablesfx(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(cpuDisableSfx); +} + + +void MainWnd::OnOptionsVideoFullscreenstretchtofit() +{ + theApp.fullScreenStretch = !theApp.fullScreenStretch; + theApp.updateWindowSize( theApp.videoOption ); + if( theApp.display ) { + if( emulating ) { + theApp.display->clear( ); + } + theApp.display->setOption( _T("fullScreenStretch"), theApp.fullScreenStretch ); + } +} + + +void MainWnd::OnUpdateOptionsVideoFullscreenstretchtofit(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.fullScreenStretch); +} + +BOOL MainWnd::OnVideoLayer(UINT nID) +{ + layerSettings ^= 0x0100 << + ((nID & 0xFFFF) - ID_OPTIONS_VIDEO_LAYERS_BG0); + layerEnable = DISPCNT & layerSettings; + CPUUpdateRenderBuffers(false); + return TRUE; +} + +void MainWnd::OnUpdateVideoLayer(CCmdUI *pCmdUI) +{ + pCmdUI->SetCheck((layerSettings >> (8 + pCmdUI->m_nID - ID_OPTIONS_VIDEO_LAYERS_BG0)) & 1); + switch(pCmdUI->m_nID) { + case ID_OPTIONS_VIDEO_LAYERS_BG1: + case ID_OPTIONS_VIDEO_LAYERS_BG2: + case ID_OPTIONS_VIDEO_LAYERS_BG3: + case ID_OPTIONS_VIDEO_LAYERS_WIN1: + case ID_OPTIONS_VIDEO_LAYERS_OBJWIN: + pCmdUI->Enable(theApp.cartridgeType == 0); + break; + } +} + +void MainWnd::OnOptionsVideoRendermethodGdi() +{ + theApp.renderMethod = GDI; + theApp.updateRenderMethod(false); + theApp.winAccelMgr.UpdateMenu(theApp.menu); +} + +void MainWnd::OnUpdateOptionsVideoRendermethodGdi(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.renderMethod == GDI); +} + +void MainWnd::OnOptionsVideoRendermethodDirectdraw() +{ + theApp.renderMethod = DIRECT_DRAW; + theApp.updateRenderMethod(false); + theApp.winAccelMgr.UpdateMenu(theApp.menu); +} + +void MainWnd::OnUpdateOptionsVideoRendermethodDirectdraw(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.renderMethod == DIRECT_DRAW); +} + +void MainWnd::OnOptionsVideoRendermethodDirect3d() +{ + theApp.renderMethod = DIRECT_3D; + theApp.updateRenderMethod(false); + theApp.winAccelMgr.UpdateMenu(theApp.menu); +} + +void MainWnd::OnUpdateOptionsVideoRendermethodDirect3d(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.renderMethod == DIRECT_3D); +} + +void MainWnd::OnOptionsVideoRendermethodOpengl() +{ + theApp.renderMethod = OPENGL; + theApp.updateRenderMethod(false); + theApp.winAccelMgr.UpdateMenu(theApp.menu); +} + +void MainWnd::OnUpdateOptionsVideoRendermethodOpengl(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.renderMethod == OPENGL); +} + + +void MainWnd::OnOptionsVideoTriplebuffering() +{ + theApp.tripleBuffering = !theApp.tripleBuffering; + if( theApp.display ) { + theApp.display->setOption( _T("tripleBuffering"), theApp.tripleBuffering ); + } +} + + +void MainWnd::OnUpdateOptionsVideoTriplebuffering(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.tripleBuffering); +} + +void MainWnd::OnOptionsVideoDdrawemulationonly() +{ + theApp.ddrawEmulationOnly = !theApp.ddrawEmulationOnly; + if( theApp.display ) { + theApp.display->setOption( _T("ddrawEmulationOnly"), theApp.ddrawEmulationOnly ); + } +} + + +void MainWnd::OnUpdateOptionsVideoDdrawemulationonly(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.ddrawEmulationOnly); +} + + +void MainWnd::OnOptionsVideoDdrawusevideomemory() +{ + theApp.ddrawUseVideoMemory = !theApp.ddrawUseVideoMemory; + if( theApp.display ) { + theApp.display->setOption( _T("ddrawUseVideoMemory"), theApp.ddrawUseVideoMemory ); + } +} + + +void MainWnd::OnUpdateOptionsVideoDdrawusevideomemory(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.ddrawUseVideoMemory); +} + +void MainWnd::OnOptionsVideoRenderoptionsD3dnofilter() +{ + theApp.d3dFilter = 0; + if( theApp.display ) { + theApp.display->setOption( _T("d3dFilter"), theApp.d3dFilter ); + } +} + +void MainWnd::OnUpdateOptionsVideoRenderoptionsD3dnofilter(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.d3dFilter == 0); +} + + +void MainWnd::OnOptionsVideoRenderoptionsD3dbilinear() +{ + theApp.d3dFilter = 1; + if( theApp.display ) { + theApp.display->setOption( _T("d3dFilter"), theApp.d3dFilter ); + } +} + + +void MainWnd::OnUpdateOptionsVideoRenderoptionsD3dbilinear(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.d3dFilter == 1); +} + + +void MainWnd::OnOptionsVideoRenderoptionsGlnearest() +{ + theApp.glFilter = 0; + if( theApp.display ) { + theApp.display->setOption( _T("glFilter"), theApp.glFilter ); + } +} + + +void MainWnd::OnUpdateOptionsVideoRenderoptionsGlnearest(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.glFilter == 0); +} + + +void MainWnd::OnOptionsVideoRenderoptionsGlbilinear() +{ + theApp.glFilter = 1; + if( theApp.display ) { + theApp.display->setOption( _T("glFilter"), theApp.glFilter ); + } +} + + +void MainWnd::OnUpdateOptionsVideoRenderoptionsGlbilinear(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.glFilter == 1); +} + + +void MainWnd::OnOptionsVideoRenderoptionsGltriangle() +{ + theApp.glType = 0; + if( theApp.display ) { + theApp.display->setOption( _T("glType"), theApp.glType ); + } +} + + +void MainWnd::OnUpdateOptionsVideoRenderoptionsGltriangle(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.glType == 0); +} + + +void MainWnd::OnOptionsVideoRenderoptionsGlquads() +{ + theApp.glType = 1; + if( theApp.display ) { + theApp.display->setOption( _T("glType"), theApp.glType ); + } +} + + +void MainWnd::OnUpdateOptionsVideoRenderoptionsGlquads(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.glType == 1); +} + +void MainWnd::OnOptionsVideoRenderoptionsSelectskin() +{ +#ifndef NOSKINS + LPCTSTR exts[] = {".ini" }; + CString filter = winLoadFilter(IDS_FILTER_INI); + CString title = winResLoadString(IDS_SELECT_SKIN_FILE); + + FileDlg dlg(this, + theApp.skinName, + filter, + 0, + "INI", + exts, + "", + title, + false); + + if(dlg.DoModal() == IDCANCEL) { + return; + } + + bool result = false; + if(!theApp.skinEnabled) { + theApp.skinEnabled = !theApp.skinEnabled; + regSetDwordValue("skinEnabled", theApp.skinEnabled); + } + + if(theApp.skin && theApp.skinEnabled) { + delete theApp.skin; + theApp.skin = NULL; + } + + theApp.skinName = dlg.GetPathName(); + + theApp.winUpdateSkin(); + theApp.winAccelMgr.UpdateMenu(theApp.menu); +#else + systemMessage( 0, _T("This build of VBA does not support skins!") ); +#endif +} + +void MainWnd::OnUpdateOptionsVideoRenderoptionsSelectskin(CCmdUI* pCmdUI) +{ + pCmdUI->Enable(theApp.display && theApp.display->isSkinSupported() && + theApp.videoOption <= VIDEO_4X); +} + +void MainWnd::OnOptionsVideoRenderoptionsSkin() +{ +#ifndef NOSKINS + theApp.skinEnabled = !theApp.skinEnabled; + theApp.updateRenderMethod(true); + theApp.winAccelMgr.UpdateMenu(theApp.menu); +#else + systemMessage( 0, _T("This build of VBA does not support skins!") ); +#endif +} + +void MainWnd::OnUpdateOptionsVideoRenderoptionsSkin(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.skinEnabled); + pCmdUI->Enable(theApp.display && theApp.display->isSkinSupported() && theApp.videoOption <= VIDEO_4X); +} + +void MainWnd::OnOptionsEmulatorAssociate() +{ + theApp.winCheckFullscreen(); + Associate dlg; + dlg.DoModal(); +} + +void MainWnd::OnOptionsEmulatorDirectories() +{ + theApp.winCheckFullscreen(); + Directories dlg; + dlg.DoModal(); +} + +void MainWnd::OnOptionsEmulatorDisablestatusmessages() +{ + theApp.disableStatusMessage = !theApp.disableStatusMessage; +} + +void MainWnd::OnUpdateOptionsEmulatorDisablestatusmessages(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.disableStatusMessage); +} + +void MainWnd::OnOptionsEmulatorSynchronize() +{ + synchronize = !synchronize; + if (synchronize) + theApp.throttle = false; +} + +void MainWnd::OnUpdateOptionsEmulatorSynchronize(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(synchronize); +} + +void MainWnd::OnOptionsEmulatorPausewheninactive() +{ + theApp.pauseWhenInactive = !theApp.pauseWhenInactive; +} + +void MainWnd::OnUpdateOptionsEmulatorPausewheninactive(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.pauseWhenInactive); +} + +void MainWnd::OnOptionsEmulatorSpeeduptoggle() +{ + theApp.speedupToggle = !theApp.speedupToggle; +} + +void MainWnd::OnUpdateOptionsEmulatorSpeeduptoggle(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.speedupToggle); +} + +void MainWnd::OnOptionsEmulatorAutomaticallyipspatch() +{ + theApp.autoIPS = !theApp.autoIPS; +} + +void MainWnd::OnUpdateOptionsEmulatorAutomaticallyipspatch(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.autoIPS); +} + +void MainWnd::OnOptionsEmulatorAgbprint() +{ + agbPrintEnable(!agbPrintIsEnabled()); +} + +void MainWnd::OnUpdateOptionsEmulatorAgbprint(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(agbPrintIsEnabled()); +} + +void MainWnd::OnOptionsEmulatorRealtimeclock() +{ + theApp.winRtcEnable = !theApp.winRtcEnable; +} + +void MainWnd::OnUpdateOptionsEmulatorRealtimeclock(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.winRtcEnable); +} + +void MainWnd::OnOptionsEmulatorAutohidemenu() +{ + theApp.autoHideMenu = !theApp.autoHideMenu; +} + +void MainWnd::OnOptionsEmulatorGenericflashcard() +{ + if(emulating && theApp.cartridgeType == IMAGE_GB) + theApp.winGenericflashcardEnable = !theApp.winGenericflashcardEnable; +} + +void MainWnd::OnUpdateOptionsEmulatorGenericflashcard(CCmdUI* pCmdUI) +{ + if(emulating && theApp.cartridgeType == IMAGE_GB) + pCmdUI->SetCheck(theApp.winGenericflashcardEnable); + else + pCmdUI->SetCheck(false); + + pCmdUI->Enable(emulating && theApp.cartridgeType == IMAGE_GB); +} + +void MainWnd::OnUpdateOptionsEmulatorAutohidemenu(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.autoHideMenu); +} + + +void MainWnd::OnOptionsEmulatorRewindinterval() +{ + RewindInterval dlg(theApp.rewindTimer/6); + int v = (int)dlg.DoModal(); + + if( v >= 0 ) { + theApp.rewindTimer = v*6; // convert to a multiple of 10 frames + regSetDwordValue("rewindTimer", v); + if(v == 0) { + if(theApp.rewindMemory) { + free(theApp.rewindMemory); + theApp.rewindMemory = NULL; + } + theApp.rewindCount = 0; + theApp.rewindCounter = 0; + theApp.rewindSaveNeeded = false; + } else { + if(theApp.rewindMemory == NULL) { + theApp.rewindMemory = (char *)malloc(8*REWIND_SIZE); + } + } + } +} + + +BOOL MainWnd::OnOptionsEmulatorShowSpeed(UINT nID) +{ + switch(nID) { + case ID_OPTIONS_EMULATOR_SHOWSPEED_NONE: + theApp.showSpeed = 0; + systemSetTitle("VisualBoyAdvance"); + break; + case ID_OPTIONS_EMULATOR_SHOWSPEED_PERCENTAGE: + theApp.showSpeed = 1; + break; + case ID_OPTIONS_EMULATOR_SHOWSPEED_DETAILED: + theApp.showSpeed = 2; + break; + case ID_OPTIONS_EMULATOR_SHOWSPEED_TRANSPARENT: + theApp.showSpeedTransparent = !theApp.showSpeedTransparent; + break; + default: + return FALSE; + } + return TRUE; +} + +void MainWnd::OnUpdateOptionsEmulatorShowSpeed(CCmdUI *pCmdUI) +{ + switch(pCmdUI->m_nID) { + case ID_OPTIONS_EMULATOR_SHOWSPEED_NONE: + pCmdUI->SetCheck(theApp.showSpeed == 0); + break; + case ID_OPTIONS_EMULATOR_SHOWSPEED_PERCENTAGE: + pCmdUI->SetCheck(theApp.showSpeed == 1); + break; + case ID_OPTIONS_EMULATOR_SHOWSPEED_DETAILED: + pCmdUI->SetCheck(theApp.showSpeed == 2); + break; + case ID_OPTIONS_EMULATOR_SHOWSPEED_TRANSPARENT: + pCmdUI->SetCheck(theApp.showSpeedTransparent); + break; + } +} + +void MainWnd::OnOptionsEmulatorSavetypeAutomatic() +{ + theApp.winSaveType = 0; +} + +void MainWnd::OnUpdateOptionsEmulatorSavetypeAutomatic(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.winSaveType == 0); +} + +void MainWnd::OnOptionsEmulatorSavetypeEeprom() +{ + theApp.winSaveType = 1; +} + +void MainWnd::OnUpdateOptionsEmulatorSavetypeEeprom(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.winSaveType == 1); +} + +void MainWnd::OnOptionsEmulatorSavetypeSram() +{ + theApp.winSaveType = 2; +} + +void MainWnd::OnUpdateOptionsEmulatorSavetypeSram(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.winSaveType == 2); +} + +void MainWnd::OnOptionsEmulatorSavetypeFlash() +{ + theApp.winSaveType = 3; +} + +void MainWnd::OnUpdateOptionsEmulatorSavetypeFlash(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.winSaveType == 3); +} + +void MainWnd::OnOptionsEmulatorSavetypeEepromsensor() +{ + theApp.winSaveType = 4; +} + +void MainWnd::OnUpdateOptionsEmulatorSavetypeEepromsensor(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.winSaveType == 4); +} + +void MainWnd::OnOptionsEmulatorSavetypeNone() +{ + theApp.winSaveType = 5; +} + +void MainWnd::OnUpdateOptionsEmulatorSavetypeNone(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.winSaveType == 5); +} + +void MainWnd::OnOptionsEmulatorSavetypeFlash512k() +{ + flashSetSize(0x10000); + theApp.winFlashSize = 0x10000; +} + +void MainWnd::OnUpdateOptionsEmulatorSavetypeFlash512k(CCmdUI* pCmdUI) +{ + // changed theApp.winFlashSize to flashSize to reflect the actual + // flashsize value used by the emu (it can change upon battery loading) + pCmdUI->SetCheck(flashSize == 0x10000); +} + +void MainWnd::OnOptionsEmulatorSavetypeFlash1m() +{ + flashSetSize(0x20000); + theApp.winFlashSize = 0x20000; +} + +void MainWnd::OnUpdateOptionsEmulatorSavetypeFlash1m(CCmdUI* pCmdUI) +{ + // changed theApp.winFlashSize to flashSize to reflect the actual + // flashsize value used by the emu (it can change upon battery loading) + pCmdUI->SetCheck(flashSize == 0x20000); +} + +void MainWnd::OnOptionsEmulatorUsebiosfile() +{ + if(!theApp.biosFileName.IsEmpty()) + theApp.useBiosFile = !theApp.useBiosFile; +} + +void MainWnd::OnUpdateOptionsEmulatorUsebiosfile(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.useBiosFile); + pCmdUI->Enable(!theApp.biosFileName.IsEmpty()); +} + +void MainWnd::OnOptionsEmulatorSkipbios() +{ + theApp.skipBiosFile = !theApp.skipBiosFile; +} + +void MainWnd::OnUpdateOptionsEmulatorSkipbios(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.skipBiosFile); +} + +void MainWnd::OnOptionsEmulatorSelectbiosfile() +{ + theApp.winCheckFullscreen(); + LPCTSTR exts[] = { "" }; + CString filter = winLoadFilter(IDS_FILTER_BIOS); + CString title = winResLoadString(IDS_SELECT_BIOS_FILE); + + FileDlg dlg(this, + theApp.biosFileName, + filter, + 0, + "BIOS", + exts, + "", + title, + false); + + if(dlg.DoModal() == IDOK) { + theApp.biosFileName = dlg.GetPathName(); + } +} + +void MainWnd::OnOptionsEmulatorPngformat() +{ + theApp.captureFormat = 0; +} + +void MainWnd::OnUpdateOptionsEmulatorPngformat(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.captureFormat == 0); +} + +void MainWnd::OnOptionsEmulatorBmpformat() +{ + theApp.captureFormat = 1; +} + +void MainWnd::OnUpdateOptionsEmulatorBmpformat(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.captureFormat == 1); +} + +void MainWnd::OnOptionsSoundOff() +{ + soundOffFlag = true; + soundShutdown(); + theApp.soundInitialized = false; +} + +void MainWnd::OnUpdateOptionsSoundOff(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(soundOffFlag); +} + +void MainWnd::OnOptionsSoundMute() +{ + soundDisable(0x30f); +} + +void MainWnd::OnUpdateOptionsSoundMute(CCmdUI* pCmdUI) +{ + int active = soundGetEnable() & 0x30f; + pCmdUI->SetCheck(active == 0); +} + +void MainWnd::OnOptionsSoundOn() +{ + if(soundOffFlag) { + soundOffFlag = false; + // soundInit(); + } + soundEnable(0x30f); +} + +void MainWnd::OnUpdateOptionsSoundOn(CCmdUI* pCmdUI) +{ + int active = soundGetEnable() & 0x30f; + pCmdUI->SetCheck(active != 0 && !soundOffFlag); +} + +void MainWnd::OnOptionsSoundUseoldsynchronization() +{ + theApp.useOldSync = !theApp.useOldSync; + systemMessage(IDS_SETTING_WILL_BE_EFFECTIVE, + "Setting will be effective the next time you start the emulator"); +} + +void MainWnd::OnUpdateOptionsSoundUseoldsynchronization(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.useOldSync); +} + +void MainWnd::OnOptionsSoundEcho() +{ + soundEcho = !soundEcho; +} + +void MainWnd::OnUpdateOptionsSoundEcho(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(soundEcho); +} + +void MainWnd::OnOptionsSoundLowpassfilter() +{ + soundLowPass = !soundLowPass; +} + +void MainWnd::OnUpdateOptionsSoundLowpassfilter(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(soundLowPass); +} + +void MainWnd::OnOptionsSoundReversestereo() +{ + soundReverse = !soundReverse; +} + +void MainWnd::OnUpdateOptionsSoundReversestereo(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(soundReverse); +} + +void MainWnd::OnOptionsSound11khz() +{ + if(theApp.cartridgeType == 0) + soundSetQuality(4); + else + gbSoundSetQuality(4); +} + +void MainWnd::OnUpdateOptionsSound11khz(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(soundQuality == 4); +} + +void MainWnd::OnOptionsSound22khz() +{ + if(theApp.cartridgeType == 0) + soundSetQuality(2); + else + gbSoundSetQuality(2); +} + +void MainWnd::OnUpdateOptionsSound22khz(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(soundQuality == 2); +} + +void MainWnd::OnOptionsSound44khz() +{ + if(theApp.cartridgeType == 0) + soundSetQuality(1); + else + gbSoundSetQuality(1); +} + +void MainWnd::OnUpdateOptionsSound44khz(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(soundQuality == 1); +} + +BOOL MainWnd::OnOptionsSoundVolume(UINT nID) +{ + soundVolume = nID - ID_OPTIONS_SOUND_VOLUME_1X; + return TRUE; +} + +void MainWnd::OnUpdateOptionsSoundVolume(CCmdUI *pCmdUI) +{ + pCmdUI->SetCheck(soundVolume == (int)(pCmdUI->m_nID - ID_OPTIONS_SOUND_VOLUME_1X)); +} + + +void MainWnd::OnOptionsSoundVolume25x() +{ + soundVolume = 4; +} + +void MainWnd::OnUpdateOptionsSoundVolume25x(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(soundVolume == 4); +} + +void MainWnd::OnOptionsSoundVolume5x() +{ + soundVolume = 5; +} + +void MainWnd::OnUpdateOptionsSoundVolume5x(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(soundVolume == 5); +} + +void MainWnd::updateSoundChannels(UINT id) +{ + int flag = 0; + + if(id == ID_OPTIONS_SOUND_CHANNEL1) + flag = 1; + + if(id == ID_OPTIONS_SOUND_CHANNEL2) + flag = 2; + + if(id == ID_OPTIONS_SOUND_CHANNEL3) + flag = 4; + + if(id == ID_OPTIONS_SOUND_CHANNEL4) + flag = 8; + + if(id == ID_OPTIONS_SOUND_DIRECTSOUNDA) + flag = 256; + + if(id == ID_OPTIONS_SOUND_DIRECTSOUNDB) + flag = 512; + + int active = soundGetEnable() & 0x30f; + + if(active & flag) + active &= (~flag); + else + active |= flag; + + soundEnable(active); + soundDisable((~active)&0x30f); +} + +void MainWnd::OnOptionsSoundChannel1() +{ + updateSoundChannels(ID_OPTIONS_SOUND_CHANNEL1); +} + +void MainWnd::OnUpdateOptionsSoundChannel1(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(soundGetEnable() & 1); +} + +void MainWnd::OnOptionsSoundChannel2() +{ + updateSoundChannels(ID_OPTIONS_SOUND_CHANNEL2); +} + +void MainWnd::OnUpdateOptionsSoundChannel2(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(soundGetEnable() & 2); +} + +void MainWnd::OnOptionsSoundChannel3() +{ + updateSoundChannels(ID_OPTIONS_SOUND_CHANNEL3); +} + +void MainWnd::OnUpdateOptionsSoundChannel3(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(soundGetEnable() & 4); +} + +void MainWnd::OnOptionsSoundChannel4() +{ + updateSoundChannels(ID_OPTIONS_SOUND_CHANNEL4); +} + +void MainWnd::OnUpdateOptionsSoundChannel4(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(soundGetEnable() & 8); +} + +void MainWnd::OnOptionsSoundDirectsounda() +{ + updateSoundChannels(ID_OPTIONS_SOUND_DIRECTSOUNDA); +} + +void MainWnd::OnUpdateOptionsSoundDirectsounda(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(soundGetEnable() & 256); + pCmdUI->Enable(theApp.cartridgeType == 0); +} + +void MainWnd::OnOptionsSoundDirectsoundb() +{ + updateSoundChannels(ID_OPTIONS_SOUND_DIRECTSOUNDB); +} + +void MainWnd::OnUpdateOptionsSoundDirectsoundb(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(soundGetEnable() & 512); + pCmdUI->Enable(theApp.cartridgeType == 0); +} + +BOOL MainWnd::OnOptionsSoundPcminterpolation(UINT nID) +{ + switch (nID) + { + case ID_OPTIONS_SOUND_PCMINTERPOLATION_NONE: + soundInterpolation = 0; + break; + case ID_OPTIONS_SOUND_PCMINTERPOLATION_LINEAR: + soundInterpolation = 1; + break; + case ID_OPTIONS_SOUND_PCMINTERPOLATION_CUBIC: + soundInterpolation = 2; + break; + case ID_OPTIONS_SOUND_PCMINTERPOLATION_FIR: + soundInterpolation = 3; + break; + case ID_OPTIONS_SOUND_PCMINTERPOLATION_LIBRESAMPLE: + soundInterpolation = 4; + break; + + default: + return FALSE; + } + + return TRUE; +} + +void MainWnd::OnUpdateOptionsSoundPcminterpolation(CCmdUI *pCmdUI) +{ + switch (pCmdUI->m_nID) + { + case ID_OPTIONS_SOUND_PCMINTERPOLATION_NONE: + pCmdUI->SetCheck(soundInterpolation == 0); + break; + case ID_OPTIONS_SOUND_PCMINTERPOLATION_LINEAR: + pCmdUI->SetCheck(soundInterpolation == 1); + break; + case ID_OPTIONS_SOUND_PCMINTERPOLATION_CUBIC: + pCmdUI->SetCheck(soundInterpolation == 2); + break; + case ID_OPTIONS_SOUND_PCMINTERPOLATION_FIR: + pCmdUI->SetCheck(soundInterpolation == 3); + break; + case ID_OPTIONS_SOUND_PCMINTERPOLATION_LIBRESAMPLE: + pCmdUI->SetCheck(soundInterpolation == 4); + break; + + default: + return; + } + pCmdUI->Enable(theApp.cartridgeType == 0); +} + +void MainWnd::OnOptionsGameboyBorder() +{ + theApp.winGbBorderOn = !theApp.winGbBorderOn; + gbBorderOn = theApp.winGbBorderOn; + if(emulating && theApp.cartridgeType == 1 && gbBorderOn) { + gbSgbRenderBorder(); + } + theApp.updateWindowSize(theApp.videoOption); +} + +void MainWnd::OnUpdateOptionsGameboyBorder(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.winGbBorderOn); +} + +void MainWnd::OnOptionsGameboyPrinter() +{ + theApp.winGbPrinterEnabled = !theApp.winGbPrinterEnabled; + if(theApp.winGbPrinterEnabled) + gbSerialFunction = gbPrinterSend; + else + gbSerialFunction = NULL; +} + +void MainWnd::OnUpdateOptionsGameboyPrinter(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(gbSerialFunction == gbPrinterSend); +} + +void MainWnd::OnOptionsGameboyBorderAutomatic() +{ + gbBorderAutomatic = !gbBorderAutomatic; + if(emulating && theApp.cartridgeType == 1 && gbBorderOn) { + gbSgbRenderBorder(); + theApp.updateWindowSize(theApp.videoOption); + } +} + +void MainWnd::OnUpdateOptionsGameboyBorderAutomatic(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(gbBorderAutomatic); +} + +void MainWnd::OnOptionsGameboyAutomatic() +{ + gbEmulatorType = 0; +} + +void MainWnd::OnUpdateOptionsGameboyAutomatic(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(gbEmulatorType == 0); +} + +void MainWnd::OnOptionsGameboyGba() +{ + gbEmulatorType = 4; +} + +void MainWnd::OnUpdateOptionsGameboyGba(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(gbEmulatorType == 4); +} + +void MainWnd::OnOptionsGameboyCgb() +{ + gbEmulatorType = 1; +} + +void MainWnd::OnUpdateOptionsGameboyCgb(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(gbEmulatorType == 1); +} + +void MainWnd::OnOptionsGameboySgb() +{ + gbEmulatorType = 2; +} + +void MainWnd::OnUpdateOptionsGameboySgb(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(gbEmulatorType == 2); +} + +void MainWnd::OnOptionsGameboySgb2() +{ + gbEmulatorType = 5; +} + +void MainWnd::OnUpdateOptionsGameboySgb2(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(gbEmulatorType == 5); +} + +void MainWnd::OnOptionsGameboyGb() +{ + gbEmulatorType = 3; +} + +void MainWnd::OnUpdateOptionsGameboyGb(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(gbEmulatorType == 3); +} + +void MainWnd::OnOptionsGameboyRealcolors() +{ + gbColorOption = 0; +} + +void MainWnd::OnUpdateOptionsGameboyRealcolors(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(gbColorOption == 0); +} + +void MainWnd::OnOptionsGameboyGameboycolors() +{ + gbColorOption = 1; +} + +void MainWnd::OnUpdateOptionsGameboyGameboycolors(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(gbColorOption == 1); +} + + +void MainWnd::OnOptionsGameboyColors() +{ + theApp.winCheckFullscreen(); + GBColorDlg dlg; + if(dlg.DoModal()) { + gbPaletteOption = dlg.getWhich(); + memcpy(systemGbPalette, dlg.getColors(), 24*sizeof(u16)); + if(emulating && theApp.cartridgeType == 1) { + memcpy(gbPalette, &systemGbPalette[dlg.getWhich()*8], 8*sizeof(u16)); + } + } +} + +BOOL MainWnd::OnOptionsPriority(UINT nID) +{ + switch(nID) { + case ID_OPTIONS_PRIORITY_HIGHEST: + theApp.threadPriority = 0; + break; + case ID_OPTIONS_PRIORITY_ABOVENORMAL: + theApp.threadPriority = 1; + break; + case ID_OPTIONS_PRIORITY_NORMAL: + theApp.threadPriority = 2; + break; + case ID_OPTIONS_PRIORITY_BELOWNORMAL: + theApp.threadPriority = 3; + break; + default: + return FALSE; + } + theApp.updatePriority(); + + return TRUE; +} + +void MainWnd::OnUpdateOptionsPriority(CCmdUI *pCmdUI) +{ + switch(pCmdUI->m_nID) { + case ID_OPTIONS_PRIORITY_HIGHEST: + pCmdUI->SetCheck(theApp.threadPriority == 0); + break; + case ID_OPTIONS_PRIORITY_ABOVENORMAL: + pCmdUI->SetCheck(theApp.threadPriority == 1); + break; + case ID_OPTIONS_PRIORITY_NORMAL: + pCmdUI->SetCheck(theApp.threadPriority == 2); + break; + case ID_OPTIONS_PRIORITY_BELOWNORMAL: + pCmdUI->SetCheck(theApp.threadPriority == 3); + break; + } +} + +BOOL MainWnd::OnOptionsFilter(UINT nID) +{ + switch(nID) + { + case ID_OPTIONS_FILTER_NORMAL: + theApp.filterType = FILTER_NONE; + break; + case ID_OPTIONS_FILTER_TVMODE: + theApp.filterType = FILTER_TVMODE; + break; + case ID_OPTIONS_FILTER_2XSAI: + theApp.filterType = FILTER_2XSAI; + break; + case ID_OPTIONS_FILTER_SUPER2XSAI: + theApp.filterType = FILTER_SUPER2XSAI; + break; + case ID_OPTIONS_FILTER_SUPEREAGLE: + theApp.filterType = FILTER_SUPEREAGLE; + break; + case ID_OPTIONS_FILTER16BIT_PIXELATEEXPERIMENTAL: + theApp.filterType = FILTER_PIXELATE; + break; + case ID_OPTIONS_FILTER16BIT_ADVANCEMAMESCALE2X: + theApp.filterType = FILTER_MAMESCALE2X; + break; + case ID_OPTIONS_FILTER16BIT_SIMPLE2X: + theApp.filterType = FILTER_SIMPLE2X; + break; + case ID_OPTIONS_FILTER_BILINEAR: + theApp.filterType = FILTER_BILINEAR; + break; + case ID_OPTIONS_FILTER_BILINEARPLUS: + theApp.filterType = FILTER_BILINEARPLUS; + break; + case ID_OPTIONS_FILTER_SCANLINES: + theApp.filterType = FILTER_SCANLINES; + break; + case ID_OPTIONS_FILTER_HQ2X: + theApp.filterType = FILTER_HQ2X; + break; + case ID_OPTIONS_FILTER_LQ2X: + theApp.filterType = FILTER_LQ2X; + break; + case ID_OPTIONS_FILTER_SIMPLE3X: + theApp.filterType = FILTER_SIMPLE3X; + break; + case ID_OPTIONS_FILTER_SIMPLE4X: + theApp.filterType = FILTER_SIMPLE4X; + break; + case ID_OPTIONS_FILTER_HQ3X: + theApp.filterType = FILTER_HQ3X; + break; + case ID_OPTIONS_FILTER_HQ4X: + theApp.filterType = FILTER_HQ4X; + break; + default: + return FALSE; + } + theApp.updateFilter(); + return TRUE; +} + +void MainWnd::OnUpdateOptionsFilter(CCmdUI *pCmdUI) +{ + pCmdUI->Enable( systemColorDepth == 16 || systemColorDepth == 32 ); + + switch(pCmdUI->m_nID) { + case ID_OPTIONS_FILTER_NORMAL: + pCmdUI->SetCheck(theApp.filterType == FILTER_NONE); + break; + case ID_OPTIONS_FILTER_TVMODE: + pCmdUI->SetCheck(theApp.filterType == FILTER_TVMODE); + break; + case ID_OPTIONS_FILTER_2XSAI: + pCmdUI->SetCheck(theApp.filterType == FILTER_2XSAI); + break; + case ID_OPTIONS_FILTER_SUPER2XSAI: + pCmdUI->SetCheck(theApp.filterType == FILTER_SUPER2XSAI); + break; + case ID_OPTIONS_FILTER_SUPEREAGLE: + pCmdUI->SetCheck(theApp.filterType == FILTER_SUPEREAGLE); + break; + case ID_OPTIONS_FILTER16BIT_PIXELATEEXPERIMENTAL: + pCmdUI->SetCheck(theApp.filterType == FILTER_PIXELATE); + break; + case ID_OPTIONS_FILTER16BIT_ADVANCEMAMESCALE2X: + pCmdUI->SetCheck(theApp.filterType == FILTER_MAMESCALE2X); + break; + case ID_OPTIONS_FILTER16BIT_SIMPLE2X: + pCmdUI->SetCheck(theApp.filterType == FILTER_SIMPLE2X); + break; + case ID_OPTIONS_FILTER_BILINEAR: + pCmdUI->SetCheck(theApp.filterType == FILTER_BILINEAR); + break; + case ID_OPTIONS_FILTER_BILINEARPLUS: + pCmdUI->SetCheck(theApp.filterType == FILTER_BILINEARPLUS); + break; + case ID_OPTIONS_FILTER_SCANLINES: + pCmdUI->SetCheck(theApp.filterType == FILTER_SCANLINES); + break; + case ID_OPTIONS_FILTER_HQ2X: + pCmdUI->SetCheck(theApp.filterType == FILTER_HQ2X); + break; + case ID_OPTIONS_FILTER_LQ2X: + pCmdUI->SetCheck(theApp.filterType == FILTER_LQ2X); + break; + case ID_OPTIONS_FILTER_SIMPLE3X: + pCmdUI->SetCheck(theApp.filterType == FILTER_SIMPLE3X); + break; + case ID_OPTIONS_FILTER_SIMPLE4X: + pCmdUI->SetCheck(theApp.filterType == FILTER_SIMPLE4X); + break; + case ID_OPTIONS_FILTER_HQ3X: + pCmdUI->SetCheck(theApp.filterType == FILTER_HQ3X); + break; + case ID_OPTIONS_FILTER_HQ4X: + pCmdUI->SetCheck(theApp.filterType == FILTER_HQ4X); + break; + } +} + +BOOL MainWnd::OnOptionsFilterIFB(UINT nID) +{ + switch(nID) { + case ID_OPTIONS_FILTER_INTERFRAMEBLENDING_NONE: + theApp.ifbType = 0; + break; + case ID_OPTIONS_FILTER_INTERFRAMEBLENDING_MOTIONBLUR: + theApp.ifbType = 1; + break; + case ID_OPTIONS_FILTER_INTERFRAMEBLENDING_SMART: + theApp.ifbType = 2; + break; + default: + return FALSE; + } + theApp.updateIFB(); + return TRUE; +} + +void MainWnd::OnUpdateOptionsFilterIFB(CCmdUI *pCmdUI) +{ + switch(pCmdUI->m_nID) { + case ID_OPTIONS_FILTER_INTERFRAMEBLENDING_NONE: + pCmdUI->SetCheck(theApp.ifbType == 0); + break; + case ID_OPTIONS_FILTER_INTERFRAMEBLENDING_MOTIONBLUR: + pCmdUI->SetCheck(theApp.ifbType == 1); + break; + case ID_OPTIONS_FILTER_INTERFRAMEBLENDING_SMART: + pCmdUI->SetCheck(theApp.ifbType == 2); + break; + } +} + +void MainWnd::OnOptionsFilterDisablemmx() +{ +#ifdef MMX + theApp.disableMMX = !theApp.disableMMX; + if(!theApp.disableMMX) + cpu_mmx = theApp.detectMMX(); + else + cpu_mmx = 0; +#endif +} + +void MainWnd::OnUpdateOptionsFilterDisablemmx(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.disableMMX); +} + +void MainWnd::OnOptionsFilterLcdcolors() +{ +// todo: depreciated + theApp.filterLCD = !theApp.filterLCD; + utilUpdateSystemColorMaps(); +} + +void MainWnd::OnUpdateOptionsFilterLcdcolors(CCmdUI *pCmdUI) +{ + pCmdUI->SetCheck(theApp.filterLCD); +} + +void MainWnd::OnOptionsLanguageSystem() +{ + theApp.winSetLanguageOption(0, false); + theApp.winAccelMgr.UpdateMenu(theApp.menu); +} + +void MainWnd::OnUpdateOptionsLanguageSystem(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.languageOption == 0); +} + +void MainWnd::OnOptionsLanguageEnglish() +{ + theApp.winSetLanguageOption(1, false); + theApp.winAccelMgr.UpdateMenu(theApp.menu); +} + +void MainWnd::OnUpdateOptionsLanguageEnglish(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.languageOption == 1); +} + +void MainWnd::OnOptionsLanguageOther() +{ + theApp.winCheckFullscreen(); + theApp.winSetLanguageOption(2, false); + theApp.winAccelMgr.UpdateMenu(theApp.menu); +} + +void MainWnd::OnUpdateOptionsLanguageOther(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(theApp.languageOption == 2); +} + + +void MainWnd::OnOptionsJoypadConfigure1() +{ + theApp.winCheckFullscreen(); + JoypadConfig dlg(0); + dlg.DoModal(); +} + +void MainWnd::OnUpdateOptionsJoypadConfigure1(CCmdUI* pCmdUI) +{ + pCmdUI->Enable(theApp.videoOption != VIDEO_320x240); +} + +void MainWnd::OnOptionsJoypadConfigure2() +{ + theApp.winCheckFullscreen(); + JoypadConfig dlg(1); + dlg.DoModal(); +} + +void MainWnd::OnUpdateOptionsJoypadConfigure2(CCmdUI* pCmdUI) +{ + pCmdUI->Enable(theApp.videoOption != VIDEO_320x240); +} + +void MainWnd::OnOptionsJoypadConfigure3() +{ + theApp.winCheckFullscreen(); + JoypadConfig dlg(2); + dlg.DoModal(); +} + +void MainWnd::OnUpdateOptionsJoypadConfigure3(CCmdUI* pCmdUI) +{ + pCmdUI->Enable(theApp.videoOption != VIDEO_320x240); +} + +void MainWnd::OnOptionsJoypadConfigure4() +{ + theApp.winCheckFullscreen(); + JoypadConfig dlg(3); + dlg.DoModal(); +} + +void MainWnd::OnUpdateOptionsJoypadConfigure4(CCmdUI* pCmdUI) +{ + pCmdUI->Enable(theApp.videoOption != VIDEO_320x240); +} + +BOOL MainWnd::OnOptionsJoypadDefault(UINT nID) +{ + theApp.joypadDefault = nID - ID_OPTIONS_JOYPAD_DEFAULTJOYPAD_1; + return TRUE; +} + +void MainWnd::OnUpdateOptionsJoypadDefault(CCmdUI *pCmdUI) +{ + pCmdUI->SetCheck(theApp.joypadDefault == (int)(pCmdUI->m_nID - ID_OPTIONS_JOYPAD_DEFAULTJOYPAD_1)); +} + +void MainWnd::OnOptionsJoypadMotionconfigure() +{ + theApp.winCheckFullscreen(); + MotionConfig dlg; + dlg.DoModal(); +} + +void MainWnd::OnUpdateOptionsJoypadMotionconfigure(CCmdUI* pCmdUI) +{ + pCmdUI->Enable(theApp.videoOption != VIDEO_320x240); +} + +BOOL MainWnd::OnOptionsJoypadAutofire(UINT nID) +{ + switch(nID) { + case ID_OPTIONS_JOYPAD_AUTOFIRE_A: + if(theApp.autoFire & 1) { + theApp.autoFire &= ~1; + systemScreenMessage(winResLoadString(IDS_AUTOFIRE_A_DISABLED)); + } else { + theApp.autoFire |= 1; + systemScreenMessage(winResLoadString(IDS_AUTOFIRE_A)); + } + break; + case ID_OPTIONS_JOYPAD_AUTOFIRE_B: + if(theApp.autoFire & 2) { + theApp.autoFire &= ~2; + systemScreenMessage(winResLoadString(IDS_AUTOFIRE_B_DISABLED)); + } else { + theApp.autoFire |= 2; + systemScreenMessage(winResLoadString(IDS_AUTOFIRE_B)); + } + break; + case ID_OPTIONS_JOYPAD_AUTOFIRE_L: + if(theApp.autoFire & 512) { + theApp.autoFire &= ~512; + systemScreenMessage(winResLoadString(IDS_AUTOFIRE_L_DISABLED)); + } else { + theApp.autoFire |= 512; + systemScreenMessage(winResLoadString(IDS_AUTOFIRE_L)); + } + break; + case ID_OPTIONS_JOYPAD_AUTOFIRE_R: + if(theApp.autoFire & 256) { + theApp.autoFire &= ~256; + systemScreenMessage(winResLoadString(IDS_AUTOFIRE_R_DISABLED)); + } else { + theApp.autoFire |= 256; + systemScreenMessage(winResLoadString(IDS_AUTOFIRE_R)); + } + break; + default: + return FALSE; + } + return TRUE; +} + +void MainWnd::OnUpdateOptionsJoypadAutofire(CCmdUI *pCmdUI) +{ + bool check = true; + switch(pCmdUI->m_nID) { + case ID_OPTIONS_JOYPAD_AUTOFIRE_A: + check = (theApp.autoFire & 1) != 0; + break; + case ID_OPTIONS_JOYPAD_AUTOFIRE_B: + check = (theApp.autoFire & 2) != 0; + break; + case ID_OPTIONS_JOYPAD_AUTOFIRE_L: + check = (theApp.autoFire & 512) != 0; + break; + case ID_OPTIONS_JOYPAD_AUTOFIRE_R: + check = (theApp.autoFire & 256) != 0; + break; + } + pCmdUI->SetCheck(check); +} + +LRESULT MainWnd::OnConfirmMode(WPARAM, LPARAM) +{ + // we need to do this separately or the window will not have the right + // parent. must be related to the way MFC does modal dialogs + winConfirmMode(); + return 0; +} + +void MainWnd::winConfirmMode() +{ + if( theApp.videoOption > VIDEO_4X ) { + theApp.winCheckFullscreen(); + ModeConfirm dlg(theApp.m_pMainWnd); + + if(!dlg.DoModal()) { + theApp.updateVideoSize(ID_OPTIONS_VIDEO_X2); + } + } + theApp.winAccelMgr.UpdateMenu(theApp.menu); +} + +void MainWnd::OnOptionsVideoFullscreenmaxscale() +{ + MaxScale dlg; + + theApp.winCheckFullscreen(); + + dlg.DoModal(); + + if( theApp.display ) { + theApp.display->setOption( _T("maxScale"), theApp.fsMaxScale ); + } +} + + +void MainWnd::OnLinkOptions() +{ + LinkOptions dlg; + + dlg.DoModal(); +} + +void MainWnd::OnOptionsLinkLog() +{ + if(linklog){ + if(linklogfile!=NULL) fclose(linklogfile); + linklog = 0; + linklogfile = NULL; + } else { + linklog=1; + openLinkLog(); + } +} + +void MainWnd::OnUpdateOptionsLinkLog(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(linklog); +} + +void MainWnd::OnOptionsLinkRFU() +{ + if(adapter) adapter = false; + else { + adapter = true; + MessageBox("Please note this is the first version\nof RFU emulation code and it's not 100% bug free.\nAlso only 2 players single computer are supported at this time.", "Warning", MB_OK); + } +} + +void MainWnd::OnUpdateOptionsLinkRFU(CCmdUI* pCmdUI) +{ + pCmdUI->SetCheck(adapter); +} + + +void MainWnd::OnOptionsEmulatorGameoverrides() +{ + if(emulating && theApp.cartridgeType == 0) { + GameOverrides dlg(this); + dlg.DoModal(); + } +} + +void MainWnd::OnUpdateOptionsEmulatorGameoverrides(CCmdUI* pCmdUI) +{ + pCmdUI->Enable(emulating && (theApp.cartridgeType == 0)); +} + +void MainWnd::OnOptionsSoundHardwareacceleration() +{ + theApp.dsoundDisableHardwareAcceleration = !theApp.dsoundDisableHardwareAcceleration; + systemSoundShutdown(); + systemSoundInit(); +} + +void MainWnd::OnUpdateOptionsSoundHardwareacceleration(CCmdUI *pCmdUI) +{ + pCmdUI->SetCheck(!theApp.dsoundDisableHardwareAcceleration); +} diff --git a/src/win32/MainWndTools.cpp b/src/win32/MainWndTools.cpp index 1441a148..b75e0792 100644 --- a/src/win32/MainWndTools.cpp +++ b/src/win32/MainWndTools.cpp @@ -51,7 +51,7 @@ static char THIS_FILE[] = __FILE__; extern bool debugger; extern int emulating; -extern int remoteSocket; +extern SOCKET remoteSocket; extern void remoteCleanUp(); extern void remoteSetSockets(SOCKET, SOCKET); @@ -205,7 +205,7 @@ void MainWnd::OnToolsDebugGdb() remoteSetSockets(wait.getListenSocket(), wait.getSocket()); debugger = true; emulating = 1; - theApp.cartridgeType = 0; + theApp.cartridgeType = IMAGE_GBA; theApp.filename = "\\gnu_stub"; rom = (u8 *)malloc(0x2000000); workRAM = (u8 *)calloc(1, 0x40000); @@ -214,7 +214,7 @@ void MainWnd::OnToolsDebugGdb() paletteRAM = (u8 *)calloc(1,0x400); vram = (u8 *)calloc(1, 0x20000); oam = (u8 *)calloc(1, 0x400); - pix = (u8 *)calloc(1, 4 * 240 * 160); + pix = (u8 *)calloc(1, 4 * 241 * 162); ioMem = (u8 *)calloc(1, 0x400); theApp.emulator = GBASystem; diff --git a/src/win32/MapView.cpp b/src/win32/MapView.cpp index 92e7c725..a93bcfda 100644 --- a/src/win32/MapView.cpp +++ b/src/win32/MapView.cpp @@ -1,6 +1,6 @@ // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -465,6 +465,7 @@ void MapView::renderMode5() bg = 2; } + void MapView::OnRefresh() { paint(); @@ -511,6 +512,12 @@ void MapView::paint() case 5: renderMode5(); break; + case 6: + renderMode5(); + break; + case 7: + renderMode5(); + break; } enableButtons(mode); SIZE s; @@ -759,7 +766,7 @@ u32 MapView::GetClickAddress(int x, int y) u32 base = ((control >> 8) & 0x1f) * 0x800 + 0x6000000; // all text bgs (16 bits) - if(mode == 0 ||(mode < 3 && bg < 2)) { + if(mode == 0 ||(mode < 3 && bg < 2) || mode == 6 || mode == 7) { return GetTextClickAddress(base, x, y); } // rot bgs (8 bits) @@ -779,8 +786,8 @@ LRESULT MapView::OnMapInfo(WPARAM wParam, LPARAM lParam) u8 *colors = (u8 *)lParam; mapViewZoom.setColors(colors); - int x = wParam & 0xffff; - int y = (wParam >> 16); + int x = (int)(wParam & 0xffff); + int y = (int)(wParam >> 16); CString buffer; buffer.Format("(%d,%d)", x, y); @@ -791,7 +798,7 @@ LRESULT MapView::OnMapInfo(WPARAM wParam, LPARAM lParam) GetDlgItem(IDC_ADDRESS)->SetWindowText(buffer); int mode = DISPCNT & 7; - if(mode >= 3) { + if(mode >= 3 && mode <=5) { // bitmap modes GetDlgItem(IDC_TILE_NUM)->SetWindowText("---"); GetDlgItem(IDC_FLIP)->SetWindowText("--"); @@ -991,36 +998,39 @@ void MapView::savePNG(const char *name) fclose(fp); } -void MapView::OnSave() +void MapView::OnSave() { - CString filename; + if(rom != NULL) + { + CString filename; - if(theApp.captureFormat == 0) - filename = "map.png"; - else - filename = "map.bmp"; + if(theApp.captureFormat == 0) + filename = "map.png"; + else + filename = "map.bmp"; - LPCTSTR exts[] = {".png", ".bmp" }; + LPCTSTR exts[] = {".png", ".bmp" }; - CString filter = theApp.winLoadFilter(IDS_FILTER_PNG); - CString title = winResLoadString(IDS_SELECT_CAPTURE_NAME); + CString filter = theApp.winLoadFilter(IDS_FILTER_PNG); + CString title = winResLoadString(IDS_SELECT_CAPTURE_NAME); - FileDlg dlg(this, - filename, - filter, - theApp.captureFormat ? 2 : 1, - theApp.captureFormat ? "BMP" : "PNG", - exts, - "", - title, - true); + FileDlg dlg(this, + filename, + filter, + theApp.captureFormat ? 2 : 1, + theApp.captureFormat ? "BMP" : "PNG", + exts, + "", + title, + true); - if(dlg.DoModal() == IDCANCEL) { - return; + if(dlg.DoModal() == IDCANCEL) { + return; + } + + if(dlg.getFilterIndex() == 2) + saveBMP(dlg.GetPathName()); + else + savePNG(dlg.GetPathName()); } - - if(dlg.getFilterIndex() == 2) - saveBMP(dlg.GetPathName()); - else - savePNG(dlg.GetPathName()); } diff --git a/src/win32/MapView.h b/src/win32/MapView.h index db0a575d..48d068f7 100644 --- a/src/win32/MapView.h +++ b/src/win32/MapView.h @@ -31,7 +31,7 @@ #include "ZoomControl.h" #include "ResizeDlg.h" #include "IUpdate.h" -#include "..\System.h" // Added by ClassView +#include "../System.h" // Added by ClassView ///////////////////////////////////////////////////////////////////////////// // MapView dialog diff --git a/src/win32/MaxScale.cpp b/src/win32/MaxScale.cpp index 1972f887..2daf5d7f 100644 --- a/src/win32/MaxScale.cpp +++ b/src/win32/MaxScale.cpp @@ -71,10 +71,6 @@ void MaxScale::OnOk() CString tmp; m_value.GetWindowText(tmp); theApp.fsMaxScale = atoi(tmp); - theApp.updateWindowSize(theApp.videoOption); - if(theApp.display) - theApp.display->clear(); - this->SetFocus(); EndDialog(TRUE); } diff --git a/src/win32/MemoryViewer.cpp b/src/win32/MemoryViewer.cpp index ff9c8796..27443ae2 100644 --- a/src/win32/MemoryViewer.cpp +++ b/src/win32/MemoryViewer.cpp @@ -535,8 +535,8 @@ void MemoryViewer::moveAddress(s32 offset, int nibbleOff) } LRESULT MemoryViewer::OnWMChar(WPARAM wParam, LPARAM LPARAM) -{ - if(OnEditInput(wParam)) +{ // The WM_CHAR message uses Unicode Transformation Format (UTF)-16. + if(OnEditInput((UINT)(wParam & 0xFFFF))) return 0; return 1; } diff --git a/src/win32/MemoryViewer.h b/src/win32/MemoryViewer.h index 2db7189e..e1a82e39 100644 --- a/src/win32/MemoryViewer.h +++ b/src/win32/MemoryViewer.h @@ -20,7 +20,7 @@ #if !defined(AFX_MEMORYVIEWER_H__52C50474_5399_4D0B_A3E4_4C52C4E0EAA0__INCLUDED_) #define AFX_MEMORYVIEWER_H__52C50474_5399_4D0B_A3E4_4C52C4E0EAA0__INCLUDED_ -#include "..\System.h" // Added by ClassView +#include "../System.h" // Added by ClassView #if _MSC_VER > 1000 #pragma once #endif // _MSC_VER > 1000 diff --git a/src/win32/MemoryViewerAddressSize.h b/src/win32/MemoryViewerAddressSize.h index 71d3b5ee..248c9420 100644 --- a/src/win32/MemoryViewerAddressSize.h +++ b/src/win32/MemoryViewerAddressSize.h @@ -20,7 +20,7 @@ #if !defined(AFX_MEMORYVIEWERADDRESSSIZE_H__04605262_2B1D_4EED_A467_B6C56AC2CACD__INCLUDED_) #define AFX_MEMORYVIEWERADDRESSSIZE_H__04605262_2B1D_4EED_A467_B6C56AC2CACD__INCLUDED_ -#include "..\System.h" // Added by ClassView +#include "../System.h" // Added by ClassView #if _MSC_VER > 1000 #pragma once #endif // _MSC_VER > 1000 diff --git a/src/win32/MemoryViewerDlg.cpp b/src/win32/MemoryViewerDlg.cpp index 09449de4..3078102f 100644 --- a/src/win32/MemoryViewerDlg.cpp +++ b/src/win32/MemoryViewerDlg.cpp @@ -320,14 +320,59 @@ void MemoryViewerDlg::setCurrentAddress(u32 address) void MemoryViewerDlg::OnSave() { - MemoryViewerAddressSize dlg; - CString buffer; + if(rom != NULL) + { + MemoryViewerAddressSize dlg; + CString buffer; - dlg.setAddress(m_viewer.getCurrentAddress()); + dlg.setAddress(m_viewer.getCurrentAddress()); - LPCTSTR exts[] = { ".dmp" }; + LPCTSTR exts[] = { ".dmp" }; - if(dlg.DoModal() == IDOK) { + if(dlg.DoModal() == IDOK) { + CString filter = theApp.winLoadFilter(IDS_FILTER_DUMP); + CString title = winResLoadString(IDS_SELECT_DUMP_FILE); + + FileDlg file(this, + buffer, + filter, + 0, + "DMP", + exts, + "", + title, + true); + if(file.DoModal() == IDOK) { + buffer = file.GetPathName(); + + FILE *f = fopen(buffer, "wb"); + + if(f == NULL) { + systemMessage(IDS_ERROR_CREATING_FILE, buffer); + return; + } + + int size = dlg.getSize(); + u32 addr = dlg.getAddress(); + + for(int i = 0; i < size; i++) { + fputc(CPUReadByteQuick(addr), f); + addr++; + } + + fclose(f); + } + } + } +} + +void MemoryViewerDlg::OnLoad() +{ + if(rom != NULL) + { + CString buffer; + LPCTSTR exts[] = { ".dmp" }; + CString filter = theApp.winLoadFilter(IDS_FILTER_DUMP); CString title = winResLoadString(IDS_SELECT_DUMP_FILE); @@ -338,86 +383,47 @@ void MemoryViewerDlg::OnSave() "DMP", exts, "", - title, - true); + title, + false); + if(file.DoModal() == IDOK) { buffer = file.GetPathName(); - - FILE *f = fopen(buffer, "wb"); - + FILE *f = fopen(buffer, "rb"); if(f == NULL) { - systemMessage(IDS_ERROR_CREATING_FILE, buffer); + systemMessage(IDS_CANNOT_OPEN_FILE, + "Cannot open file %s", + buffer); return; } + + MemoryViewerAddressSize dlg; - int size = dlg.getSize(); - u32 addr = dlg.getAddress(); + fseek(f, 0, SEEK_END); + int size = ftell(f); - for(int i = 0; i < size; i++) { - fputc(CPUReadByteQuick(addr), f); - addr++; + fseek(f, 0, SEEK_SET); + + dlg.setAddress(m_viewer.getCurrentAddress()); + dlg.setSize(size); + + if(dlg.DoModal() == IDOK) { + int size = dlg.getSize(); + u32 addr = dlg.getAddress(); + + for(int i = 0; i < size; i++) { + int c = fgetc(f); + if(c == -1) + break; + CPUWriteByteQuick(addr, c); + addr++; + } + OnRefresh(); } - - fclose(f); + fclose(f); } } } -void MemoryViewerDlg::OnLoad() -{ - CString buffer; - LPCTSTR exts[] = { ".dmp" }; - - CString filter = theApp.winLoadFilter(IDS_FILTER_DUMP); - CString title = winResLoadString(IDS_SELECT_DUMP_FILE); - - FileDlg file(this, - buffer, - filter, - 0, - "DMP", - exts, - "", - title, - false); - - if(file.DoModal() == IDOK) { - buffer = file.GetPathName(); - FILE *f = fopen(buffer, "rb"); - if(f == NULL) { - systemMessage(IDS_CANNOT_OPEN_FILE, - "Cannot open file %s", - buffer); - return; - } - - MemoryViewerAddressSize dlg; - - fseek(f, 0, SEEK_END); - int size = ftell(f); - - fseek(f, 0, SEEK_SET); - - dlg.setAddress(m_viewer.getCurrentAddress()); - dlg.setSize(size); - - if(dlg.DoModal() == IDOK) { - int size = dlg.getSize(); - u32 addr = dlg.getAddress(); - - for(int i = 0; i < size; i++) { - int c = fgetc(f); - if(c == -1) - break; - CPUWriteByteQuick(addr, c); - addr++; - } - OnRefresh(); - } - fclose(f); - } -} - void MemoryViewerDlg::PostNcDestroy() { delete this; diff --git a/src/win32/ModeConfirm.cpp b/src/win32/ModeConfirm.cpp index 6ac88032..6ac68bf9 100644 --- a/src/win32/ModeConfirm.cpp +++ b/src/win32/ModeConfirm.cpp @@ -100,7 +100,7 @@ BOOL ModeConfirm::OnInitDialog() // EXCEPTION: OCX Property Pages should return FALSE } -void ModeConfirm::OnTimer(UINT nIDEvent) +void ModeConfirm::OnTimer(UINT_PTR nIDEvent) { CString buffer; count--; diff --git a/src/win32/ModeConfirm.h b/src/win32/ModeConfirm.h index 0e5a1838..4f8787ec 100644 --- a/src/win32/ModeConfirm.h +++ b/src/win32/ModeConfirm.h @@ -34,7 +34,7 @@ class ModeConfirm : public CDialog // Construction public: int count; - UINT timer; + UINT_PTR timer; ModeConfirm(CWnd* pParent); // standard constructor // Dialog Data @@ -60,7 +60,7 @@ class ModeConfirm : public CDialog afx_msg void OnOk(); afx_msg void OnDestroy(); virtual BOOL OnInitDialog(); - afx_msg void OnTimer(UINT nIDEvent); + afx_msg void OnTimer(UINT_PTR nIDEvent); //}}AFX_MSG DECLARE_MESSAGE_MAP() }; diff --git a/src/win32/OamView.cpp b/src/win32/OamView.cpp index 13c233ac..c389c4d4 100644 --- a/src/win32/OamView.cpp +++ b/src/win32/OamView.cpp @@ -463,37 +463,40 @@ void OamView::savePNG(const char *name) void OamView::OnSave() { - CString captureBuffer; + if(rom != NULL) + { + CString captureBuffer; - if(theApp.captureFormat == 0) - captureBuffer = "oam.png"; - else - captureBuffer = "oam.bmp"; + if(theApp.captureFormat == 0) + captureBuffer = "oam.png"; + else + captureBuffer = "oam.bmp"; - LPCTSTR exts[] = {".png", ".bmp" }; + LPCTSTR exts[] = {".png", ".bmp" }; - CString filter = theApp.winLoadFilter(IDS_FILTER_PNG); - CString title = winResLoadString(IDS_SELECT_CAPTURE_NAME); + CString filter = theApp.winLoadFilter(IDS_FILTER_PNG); + CString title = winResLoadString(IDS_SELECT_CAPTURE_NAME); - FileDlg dlg(this, - captureBuffer, - filter, - theApp.captureFormat ? 2 : 1, - theApp.captureFormat ? "BMP" : "PNG", - exts, - "", - title, - true); + FileDlg dlg(this, + captureBuffer, + filter, + theApp.captureFormat ? 2 : 1, + theApp.captureFormat ? "BMP" : "PNG", + exts, + "", + title, + true); - if(dlg.DoModal() == IDCANCEL) { - return; + if(dlg.DoModal() == IDCANCEL) { + return; + } + captureBuffer = dlg.GetPathName(); + + if(dlg.getFilterIndex() == 2) + saveBMP(captureBuffer); + else + savePNG(captureBuffer); } - captureBuffer = dlg.GetPathName(); - - if(dlg.getFilterIndex() == 2) - saveBMP(captureBuffer); - else - savePNG(captureBuffer); } BOOL OamView::OnInitDialog() diff --git a/src/win32/OpenGL.cpp b/src/win32/OpenGL.cpp index e64b302d..2cf71e84 100644 --- a/src/win32/OpenGL.cpp +++ b/src/win32/OpenGL.cpp @@ -1,552 +1,669 @@ -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#include "stdafx.h" -#include "MainWnd.h" -#include - -#include "../System.h" -#include "../GBA.h" -#include "../Globals.h" -#include "../Text.h" -#include "../Util.h" - -#include "Reg.h" -#include "..\..\res\resource.h" - -#include "../gbafilter.h" - -#ifdef _DEBUG -#define new DEBUG_NEW -#undef THIS_FILE -static char THIS_FILE[] = __FILE__; -#endif - -#ifdef MMX -extern "C" bool cpu_mmx; - -extern bool detectMMX(); -#endif - -extern int Init_2xSaI(u32); -extern void winlog(const char *,...); -extern int systemSpeed; - -class OpenGLDisplay : public IDisplay { -private: - HDC hDC; - HGLRC hglrc; - GLuint texture; - int width; - int height; - float size; - u8 *filterData; - bool failed; - - bool initializeTexture(int w, int h); - void updateFiltering(int); -public: - OpenGLDisplay(); - virtual ~OpenGLDisplay(); - - virtual bool initialize(); - virtual void cleanup(); - virtual void render(); - virtual void checkFullScreen(); - virtual void renderMenu(); - virtual void clear(); - virtual bool changeRenderSize(int w, int h); - virtual void resize(int w, int h); - virtual DISPLAY_TYPE getType() { return OPENGL; }; - virtual void setOption(const char *, int); - virtual int selectFullScreenMode(GUID **); -}; - -OpenGLDisplay::OpenGLDisplay() -{ - hDC = NULL; - hglrc = NULL; - texture = 0; - width = 0; - height = 0; - size = 0.0f; - filterData = (u8 *)malloc(4*4*4*256*240); - failed = false; -} - -OpenGLDisplay::~OpenGLDisplay() -{ - cleanup(); -} - -void OpenGLDisplay::cleanup() -{ - if(texture != 0) { - glDeleteTextures(1, &texture); - texture = 0; - } - if(hglrc != NULL) { - wglDeleteContext(hglrc); - wglMakeCurrent(NULL, NULL); - hglrc = NULL; - } - if(hDC != NULL) { - ReleaseDC(*theApp.m_pMainWnd, hDC); - hDC = NULL; - } - if(filterData) { - free(filterData); - filterData = NULL; - } - width = 0; - height = 0; - size = 0.0f; -} - -bool OpenGLDisplay::initialize() -{ - theApp.sizeX = 240; - theApp.sizeY = 160; - - switch(theApp.videoOption) { - case VIDEO_1X: - theApp.surfaceSizeX = theApp.sizeX; - theApp.surfaceSizeY = theApp.sizeY; - break; - case VIDEO_2X: - theApp.surfaceSizeX = theApp.sizeX * 2; - theApp.surfaceSizeY = theApp.sizeY * 2; - break; - case VIDEO_3X: - theApp.surfaceSizeX = theApp.sizeX * 3; - theApp.surfaceSizeY = theApp.sizeY * 3; - break; - case VIDEO_4X: - theApp.surfaceSizeX = theApp.sizeX * 4; - theApp.surfaceSizeY = theApp.sizeY * 4; - break; - case VIDEO_320x240: - case VIDEO_640x480: - case VIDEO_800x600: - case VIDEO_1024x768: - case VIDEO_1280x1024: - case VIDEO_OTHER: - { - RECT r; - ::GetWindowRect(GetDesktopWindow(), &r); - theApp.fsWidth = r.right - r.left; - theApp.fsHeight = r.bottom - r.top; - - /* Need to fix this code later. For now, Fullscreen takes the whole - screen. - int scaleX = (fsWidth / sizeX); - int scaleY = (fsHeight / sizeY); - int min = scaleX < scaleY ? scaleX : scaleY; - surfaceSizeX = sizeX * min; - surfaceSizeY = sizeY * min; - if(fullScreenStretch) { - */ - theApp.surfaceSizeX = theApp.fsWidth; - theApp.surfaceSizeY = theApp.fsHeight; - // } - } - break; - } - - theApp.rect.left = 0; - theApp.rect.top = 0; - theApp.rect.right = theApp.sizeX; - theApp.rect.bottom = theApp.sizeY; - - theApp.dest.left = 0; - theApp.dest.top = 0; - theApp.dest.right = theApp.surfaceSizeX; - theApp.dest.bottom = theApp.surfaceSizeY; - - DWORD style = WS_POPUP | WS_VISIBLE; - DWORD styleEx = 0; - - if(theApp.videoOption <= VIDEO_4X) - style |= WS_OVERLAPPEDWINDOW; - else - styleEx = 0; - - if(theApp.videoOption <= VIDEO_4X) - AdjustWindowRectEx(&theApp.dest, style, TRUE, styleEx); - else - AdjustWindowRectEx(&theApp.dest, style, FALSE, styleEx); - - int winSizeX = theApp.dest.right-theApp.dest.left; - int winSizeY = theApp.dest.bottom-theApp.dest.top; - - if(theApp.videoOption > VIDEO_4X) { - winSizeX = theApp.fsWidth; - winSizeY = theApp.fsHeight; - } - - int x = 0; - int y = 0; - - if(theApp.videoOption <= VIDEO_4X) { - x = theApp.windowPositionX; - y = theApp.windowPositionY; - } - - // Create a window - MainWnd *pWnd = new MainWnd; - theApp.m_pMainWnd = pWnd; - - pWnd->CreateEx(styleEx, - theApp.wndClass, - "VisualBoyAdvance", - style, - x,y,winSizeX,winSizeY, - NULL, - 0); - - if (!(HWND)*pWnd) { - winlog("Error creating Window %08x\n", GetLastError()); - return FALSE; - } - - theApp.updateMenuBar(); - - theApp.adjustDestRect(); - - theApp.mode320Available = FALSE; - theApp.mode640Available = FALSE; - theApp.mode800Available = FALSE; - - CDC *dc = pWnd->GetDC(); - HDC hDC = dc->GetSafeHdc(); - - PIXELFORMATDESCRIPTOR pfd = { - sizeof(PIXELFORMATDESCRIPTOR), // size of this pfd - 1, // version number - PFD_DRAW_TO_WINDOW | // support window - PFD_SUPPORT_OPENGL | // support OpenGL - PFD_DOUBLEBUFFER, // double buffered - PFD_TYPE_RGBA, // RGBA type - 16, // 16-bit color depth - 0, 0, 0, 0, 0, 0, // color bits ignored - 0, // no alpha buffer - 0, // shift bit ignored - 0, // no accumulation buffer - 0, 0, 0, 0, // accum bits ignored - 32, // 32-bit z-buffer - 0, // no stencil buffer - 0, // no auxiliary buffer - PFD_MAIN_PLANE, // main layer - 0, // reserved - 0, 0, 0 // layer masks ignored - }; - int iPixelFormat; - - if(!(iPixelFormat = ChoosePixelFormat(hDC, &pfd))) { - winlog("Failed ChoosePixelFormat\n"); - return false; - } - - // obtain detailed information about - // the device context's first pixel format - if(!(DescribePixelFormat(hDC, iPixelFormat, - sizeof(PIXELFORMATDESCRIPTOR), &pfd))) { - winlog("Failed DescribePixelFormat\n"); - return false; - } - - if(!SetPixelFormat(hDC, iPixelFormat, &pfd)) { - winlog("Failed SetPixelFormat\n"); - return false; - } - - if(!(hglrc = wglCreateContext(hDC))) { - winlog("Failed wglCreateContext\n"); - return false; - } - - if(!wglMakeCurrent(hDC, hglrc)) { - winlog("Failed wglMakeCurrent\n"); - return false; - } - pWnd->ReleaseDC(dc); - - // setup 2D gl environment - glPushAttrib(GL_ENABLE_BIT); - glDisable(GL_DEPTH_TEST); - glDisable(GL_CULL_FACE); - glEnable(GL_TEXTURE_2D); - - glViewport(0, 0, theApp.surfaceSizeX, theApp.surfaceSizeY); - - glMatrixMode(GL_PROJECTION); - glLoadIdentity(); - - glOrtho(0.0, (GLdouble)(theApp.surfaceSizeX), (GLdouble)(theApp.surfaceSizeY), - 0.0, 0.0,1.0); - glMatrixMode(GL_MODELVIEW); - glLoadIdentity(); - - systemRedShift = 3; - systemGreenShift = 11; - systemBlueShift = 19; - systemColorDepth = 32; - theApp.fsColorDepth = 32; - - Init_2xSaI(32); -#ifdef MMX - if(!theApp.disableMMX) - cpu_mmx = theApp.detectMMX(); - else - cpu_mmx = 0; -#endif - - if(theApp.ddrawDebug) { - winlog("R shift: %d\n", systemRedShift); - winlog("G shift: %d\n", systemGreenShift); - winlog("B shift: %d\n", systemBlueShift); - } - - utilUpdateSystemColorMaps(theApp.filterLCD); - theApp.updateFilter(); - theApp.updateIFB(); - - if(failed) - return false; - - pWnd->DragAcceptFiles(TRUE); - - return TRUE; -} - -void OpenGLDisplay::clear() -{ -} - -void OpenGLDisplay::renderMenu() -{ - checkFullScreen(); - if(theApp.m_pMainWnd) - theApp.m_pMainWnd->DrawMenuBar(); -} - -void OpenGLDisplay::checkFullScreen() -{ - // if(tripleBuffering) - // pOpenGL->FlipToGDISurface(); -} - -void OpenGLDisplay::render() -{ - int pitch = theApp.filterWidth * 4 + 4; - u8 *data = pix + (theApp.sizeX+1)*4; - - if(theApp.filterFunction) { - data = filterData; - theApp.filterFunction(pix+pitch, - pitch, - (u8*)theApp.delta, - (u8*)filterData, - theApp.rect.right * (systemColorDepth / 8), - theApp.filterWidth, - theApp.filterHeight); - } - - if(theApp.videoOption > VIDEO_4X && theApp.showSpeed) { - char buffer[30]; - if(theApp.showSpeed == 1) - sprintf(buffer, "%3d%%", systemSpeed); - else - sprintf(buffer, "%3d%%(%d, %d fps)", systemSpeed, - systemFrameSkip, - theApp.showRenderedFrames); - - if(theApp.filterFunction) { - int p = theApp.rect.right * 2; - if(systemColorDepth == 24) - p = theApp.rect.right * 3; - else if(systemColorDepth == 32) - p = theApp.rect.right * 4; - if(theApp.showSpeedTransparent) - drawTextTransp((u8*)filterData, - p, - 10, - theApp.rect.bottom-10, - buffer); - else - drawText((u8*)filterData, - p, - 10, - theApp.rect.bottom-10, - buffer); - } else { - if(theApp.showSpeedTransparent) - drawTextTransp((u8*)pix, - pitch, - 10, - theApp.filterHeight-10, - buffer); - else - drawText((u8*)pix, - pitch, - 10, - theApp.filterHeight-10, - buffer); - } - } - - // Texturemap complete texture to surface so we have free scaling - // and antialiasing - if(theApp.filterFunction) { - glPixelStorei(GL_UNPACK_ROW_LENGTH, theApp.rect.right); - } else { - glPixelStorei(GL_UNPACK_ROW_LENGTH, theApp.sizeX+1); - } - - glTexSubImage2D( GL_TEXTURE_2D,0, - 0,0, theApp.rect.right,theApp.rect.bottom, - GL_RGBA,GL_UNSIGNED_BYTE,data); - - if(theApp.glType == 0) { - glBegin(GL_TRIANGLE_STRIP); - glTexCoord2f(0.0, 0.0); glVertex3i(0, 0, 0); - glTexCoord2f(theApp.rect.right/size, 0.0); glVertex3i(theApp.surfaceSizeX, 0, 0); - glTexCoord2f(0.0, theApp.rect.bottom/size); glVertex3i(0, theApp.surfaceSizeY, 0); - glTexCoord2f(theApp.rect.right/size, theApp.rect.bottom/size); glVertex3i(theApp.surfaceSizeX, theApp.surfaceSizeY, 0); - glEnd(); - } else { - glBegin(GL_QUADS); - glTexCoord2f(0.0, 0.0); glVertex3i(0, 0, 0); - glTexCoord2f(theApp.rect.right/size, 0.0); glVertex3i(theApp.surfaceSizeX, 0, 0); - glTexCoord2f(theApp.rect.right/size, theApp.rect.bottom/size); glVertex3i(theApp.surfaceSizeX, theApp.surfaceSizeY, 0); - glTexCoord2f(0.0, theApp.rect.bottom/size); glVertex3i(0, theApp.surfaceSizeY, 0); - glEnd(); - } - - CDC *dc = theApp.m_pMainWnd->GetDC(); - - if(theApp.screenMessage) { - if(((GetTickCount() - theApp.screenMessageTime) < 3000) && - !theApp.disableStatusMessage) { - dc->SetTextColor(RGB(255,0,0)); - dc->SetBkMode(TRANSPARENT); - dc->TextOut(10, theApp.surfaceSizeY - 20, theApp.screenMessageBuffer); - } else { - theApp.screenMessage = false; - } - } - - SwapBuffers(dc->GetSafeHdc()); - - theApp.m_pMainWnd->ReleaseDC(dc); -} - -void OpenGLDisplay::resize(int w, int h) -{ - glViewport(0, 0, w, h); - glMatrixMode(GL_PROJECTION); - glLoadIdentity(); - - glOrtho(0.0, (GLdouble)(w), (GLdouble)(h), - 0.0, 0.0,1.0); - glMatrixMode(GL_MODELVIEW); - glLoadIdentity(); -} - -void OpenGLDisplay::updateFiltering(int value) -{ - switch(value) { - case 0: - glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST); - glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST); - break; - case 1: - glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); - glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR); - break; - } -} - -bool OpenGLDisplay::initializeTexture(int w, int h) -{ - int mySize = 256; - size = 256.0f; - if(w > 255 || h > 255) { - size = 512.0f; - mySize = 512; - } - if(w > 512 || h > 512) { - size = 1024.0f; - mySize = 1024; - } - glGenTextures(1, &texture); - glBindTexture(GL_TEXTURE_2D, texture); - - int filter = regQueryDwordValue("glFilter", 0); - if(filter < 0 || filter > 1) - filter = 0; - updateFiltering(filter); - - glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, mySize, mySize, 0, GL_RGBA, - GL_UNSIGNED_BYTE, NULL ); - width = w; - height = h; - - return true; -} - -bool OpenGLDisplay::changeRenderSize(int w, int h) -{ - if(width != w || height != h) { - if(texture != 0) { - glDeleteTextures(1, &texture); - texture = 0; - } - if(!initializeTexture(w, h)) { - failed = true; - return false; - } - } - return true; -} - -void OpenGLDisplay::setOption(const char *option, int value) -{ - if(!strcmp(option, "glFilter")) - updateFiltering(value); -} - -int OpenGLDisplay::selectFullScreenMode(GUID **) -{ - HWND wnd = GetDesktopWindow(); - RECT r; - GetWindowRect(wnd, &r); - int w = (r.right - r.left) & 4095; - int h = (r.bottom - r.top) & 4095; - HDC dc = GetDC(wnd); - int c = GetDeviceCaps(dc, BITSPIXEL); - ReleaseDC(wnd, dc); - - return (c << 24) | (w << 12) | h; -} - -IDisplay *newOpenGLDisplay() -{ - return new OpenGLDisplay(); -} - +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004 Forgotten and the VBA development team +// Copyright (C) 2005-2006 VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#include "stdafx.h" + +#include "MainWnd.h" + +#include "../System.h" +#include "../GBA.h" +#include "../Globals.h" +#include "../Text.h" +#include "../Util.h" +#include "../gb/gbGlobals.h" + +#include + +// OpenGL +#include // main include file +typedef BOOL (APIENTRY *PFNWGLSWAPINTERVALFARPROC)( int ); + +extern int Init_2xSaI(u32); +extern void winlog(const char *,...); +extern int systemSpeed; + +#ifdef _DEBUG +#define new DEBUG_NEW +#undef THIS_FILE +static char THIS_FILE[] = __FILE__; +#endif + +#ifdef MMX +extern "C" bool cpu_mmx; +extern bool detectMMX(); +#endif + + +class OpenGLDisplay : public IDisplay { +private: + HDC hDC; + HGLRC hglrc; + GLuint texture; + int width; + int height; + float size; + u8 *filterData; + RECT destRect; + bool failed; + + void initializeMatrices( int w, int h ); + bool initializeTexture( int w, int h ); + void updateFiltering( int value ); + void setVSync( int interval = 1 ); + void calculateDestRect( int w, int h ); + +public: + OpenGLDisplay(); + virtual ~OpenGLDisplay(); + virtual DISPLAY_TYPE getType() { return OPENGL; }; + + virtual bool initialize(); + virtual void cleanup(); + virtual void render(); + virtual void renderMenu(); + virtual void clear(); + virtual bool changeRenderSize( int w, int h ); + virtual void resize( int w, int h ); + virtual void setOption( const char *, int ); + virtual int selectFullScreenMode( GUID ** ); +}; + + +OpenGLDisplay::OpenGLDisplay() +{ + hDC = NULL; + hglrc = NULL; + texture = 0; + width = 0; + height = 0; + size = 0.0f; + filterData = (u8 *)malloc(4*4*256*240); + failed = false; +} + + +OpenGLDisplay::~OpenGLDisplay() +{ + cleanup(); +} + + +void OpenGLDisplay::cleanup() +{ + if(texture != 0) { + glDeleteTextures(1, &texture); + texture = 0; + } + + if(hglrc != NULL) { + wglDeleteContext(hglrc); + wglMakeCurrent(NULL, NULL); + hglrc = NULL; + } + + if(hDC != NULL) { + ReleaseDC(*theApp.m_pMainWnd, hDC); + hDC = NULL; + } + + if(filterData) { + free(filterData); + filterData = NULL; + } + + width = 0; + height = 0; + size = 0.0f; +} + + +bool OpenGLDisplay::initialize() +{ + switch( theApp.cartridgeType ) + { + case IMAGE_GBA: + theApp.sizeX = 240; + theApp.sizeY = 160; + break; + case IMAGE_GB: + if ( gbBorderOn ) + { + theApp.sizeX = 256; + theApp.sizeY = 224; + } + else + { + theApp.sizeX = 160; + theApp.sizeY = 144; + } + break; + } + + + switch(theApp.videoOption) + { + case VIDEO_1X: + theApp.surfaceSizeX = theApp.sizeX; + theApp.surfaceSizeY = theApp.sizeY; + break; + case VIDEO_2X: + theApp.surfaceSizeX = theApp.sizeX * 2; + theApp.surfaceSizeY = theApp.sizeY * 2; + break; + case VIDEO_3X: + theApp.surfaceSizeX = theApp.sizeX * 3; + theApp.surfaceSizeY = theApp.sizeY * 3; + break; + case VIDEO_4X: + theApp.surfaceSizeX = theApp.sizeX * 4; + theApp.surfaceSizeY = theApp.sizeY * 4; + break; + case VIDEO_320x240: + case VIDEO_640x480: + case VIDEO_800x600: + case VIDEO_OTHER: + { + if( theApp.fullScreenStretch ) { + theApp.surfaceSizeX = theApp.fsWidth; + theApp.surfaceSizeY = theApp.fsHeight; + } else { + float scaleX = (float)theApp.fsWidth / (float)theApp.sizeX; + float scaleY = (float)theApp.fsHeight / (float)theApp.sizeY; + float min = ( scaleX < scaleY ) ? scaleX : scaleY; + if( theApp.fsMaxScale ) + min = ( min > (float)theApp.fsMaxScale ) ? (float)theApp.fsMaxScale : min; + theApp.surfaceSizeX = (int)((float)theApp.sizeX * min); + theApp.surfaceSizeY = (int)((float)theApp.sizeY * min); + } + } + break; + } + + theApp.rect.left = 0; + theApp.rect.top = 0; + theApp.rect.right = theApp.sizeX; + theApp.rect.bottom = theApp.sizeY; + + theApp.dest.left = 0; + theApp.dest.top = 0; + theApp.dest.right = theApp.surfaceSizeX; + theApp.dest.bottom = theApp.surfaceSizeY; + + DWORD style = WS_POPUP | WS_VISIBLE; + DWORD styleEx = 0; + + if( theApp.videoOption <= VIDEO_4X ) + style |= WS_OVERLAPPEDWINDOW; + else + styleEx = 0; + + if( theApp.videoOption <= VIDEO_4X ) + AdjustWindowRectEx( &theApp.dest, style, TRUE, styleEx ); + else + AdjustWindowRectEx( &theApp.dest, style, FALSE, styleEx ); + + int winSizeX = theApp.dest.right - theApp.dest.left; + int winSizeY = theApp.dest.bottom - theApp.dest.top; + int x = 0, y = 0; + + if( theApp.videoOption <= VIDEO_4X ) { + x = theApp.windowPositionX; + y = theApp.windowPositionY; + } else { + winSizeX = theApp.fsWidth; + winSizeY = theApp.fsHeight; + } + + // Create a window + MainWnd *pWnd = new MainWnd; + theApp.m_pMainWnd = pWnd; + + pWnd->CreateEx( + styleEx, + theApp.wndClass, + "VisualBoyAdvance", + style, + x,y,winSizeX,winSizeY, + NULL, + 0 ); + + if (!(HWND)*pWnd) { + winlog("Error creating Window %08x\n", GetLastError()); + return FALSE; + } + + theApp.updateMenuBar(); + + theApp.adjustDestRect(); + + theApp.mode320Available = FALSE; + theApp.mode640Available = FALSE; + theApp.mode800Available = FALSE; + + CDC *dc = pWnd->GetDC(); + HDC hDC = dc->GetSafeHdc(); + + PIXELFORMATDESCRIPTOR pfd = { + sizeof(PIXELFORMATDESCRIPTOR), + 1, // version number + PFD_DRAW_TO_WINDOW | // support window + PFD_SUPPORT_OPENGL | // support OpenGL + PFD_DOUBLEBUFFER, // double buffered + PFD_TYPE_RGBA, // RGBA type + 24, // 24-bit color depth + 0, 0, 0, 0, 0, 0, // color bits ignored + 0, // no alpha buffer + 0, // shift bit ignored + 0, // no accumulation buffer + 0, 0, 0, 0, // accum bits ignored + 32, // 32-bit z-buffer + 0, // no stencil buffer + 0, // no auxiliary buffer + PFD_MAIN_PLANE, // main layer + 0, // reserved + 0, 0, 0 // layer masks ignored + }; + + int iPixelFormat; + if( !(iPixelFormat = ChoosePixelFormat( hDC, &pfd )) ) { + winlog( "Failed ChoosePixelFormat\n" ); + return false; + } + + // obtain detailed information about + // the device context's first pixel format + if( !( DescribePixelFormat( + hDC, + iPixelFormat, + sizeof(PIXELFORMATDESCRIPTOR), + &pfd ) ) ) + { + winlog( "Failed DescribePixelFormat\n" ); + return false; + } + + if( !SetPixelFormat( hDC, iPixelFormat, &pfd ) ) { + winlog( "Failed SetPixelFormat\n" ); + return false; + } + + if( !( hglrc = wglCreateContext( hDC ) ) ) { + winlog( "Failed wglCreateContext\n" ); + return false; + } + + if( !wglMakeCurrent(hDC, hglrc) ) { + winlog( "Failed wglMakeCurrent\n" ); + return false; + } + + pWnd->ReleaseDC( dc ); + + // setup 2D gl environment + glPushAttrib( GL_ENABLE_BIT ); + glDisable( GL_DEPTH_TEST ); + glDisable( GL_CULL_FACE ); + glEnable( GL_TEXTURE_2D ); + + initializeMatrices( theApp.surfaceSizeX, theApp.surfaceSizeY ); + + setVSync( theApp.vsync ); + +#ifdef MMX + if(!theApp.disableMMX) + cpu_mmx = theApp.detectMMX(); + else + cpu_mmx = 0; +#endif + + systemRedShift = 3; + systemGreenShift = 11; + systemBlueShift = 19; + systemColorDepth = 32; + theApp.fsColorDepth = 32; + + Init_2xSaI(32); + + utilUpdateSystemColorMaps(); + theApp.updateFilter(); + theApp.updateIFB(); + + if(failed) + return false; + + pWnd->DragAcceptFiles(TRUE); + + return TRUE; +} + + +void OpenGLDisplay::clear() +{ + glClear( GL_COLOR_BUFFER_BIT ); +} + + +void OpenGLDisplay::renderMenu() +{ + checkFullScreen(); + if( theApp.m_pMainWnd ) + theApp.m_pMainWnd->DrawMenuBar(); +} + + +void OpenGLDisplay::render() +{ + clear(); + + int pitch = theApp.filterWidth * 4 + 4; + u8 *data = pix + ( theApp.sizeX + 1 ) * 4; + + // apply pixel filter + if(theApp.filterFunction) { + data = filterData; + theApp.filterFunction( + pix + pitch, + pitch, + (u8*)theApp.delta, + (u8*)filterData, + theApp.filterWidth * 4 * 2, + theApp.filterWidth, + theApp.filterHeight); + } + + // Texturemap complete texture to surface + // so we have free scaling and antialiasing + int mult; + if( theApp.filterFunction ) { + glPixelStorei( GL_UNPACK_ROW_LENGTH, theApp.sizeX << 1 ); + mult = 2; + } else { + glPixelStorei( GL_UNPACK_ROW_LENGTH, theApp.sizeX + 1 ); + mult = 1; + } + + glTexSubImage2D( + GL_TEXTURE_2D, + 0, + 0, + 0, + mult * theApp.sizeX, + mult * theApp.sizeY, + GL_RGBA, + GL_UNSIGNED_BYTE, + data ); + + if( theApp.glType == 0 ) { + glBegin( GL_TRIANGLE_STRIP ); + + glTexCoord2f( 0.0f, 0.0f ); + glVertex3i( 0, 0, 0 ); + + glTexCoord2f( (float)(mult * theApp.sizeX) / size, 0.0f ); + glVertex3i( theApp.surfaceSizeX, 0, 0 ); + + glTexCoord2f( 0.0f, (float)(mult * theApp.sizeY) / size ); + glVertex3i( 0, theApp.surfaceSizeY, 0 ); + + glTexCoord2f( (float)(mult * theApp.sizeX) / size, (float)(mult * theApp.sizeY) / size ); + glVertex3i( theApp.surfaceSizeX, theApp.surfaceSizeY, 0 ); + + glEnd(); + } else { + glBegin( GL_QUADS ); + + glTexCoord2f( 0.0f, 0.0f ); + glVertex3i( 0, 0, 0 ); + + glTexCoord2f( (float)(mult * theApp.sizeX) / size, 0.0f ); + glVertex3i( theApp.surfaceSizeX, 0, 0 ); + + glTexCoord2f( (float)(mult * theApp.sizeX) / size, (float)(mult * theApp.sizeY) / size ); + glVertex3i( theApp.surfaceSizeX, theApp.surfaceSizeY, 0 ); + + glTexCoord2f( 0.0f, (float)(mult * theApp.sizeY) / size ); + glVertex3i( 0, theApp.surfaceSizeY, 0 ); + + glEnd(); + } + + CDC *dc = theApp.m_pMainWnd->GetDC(); + + SwapBuffers( dc->GetSafeHdc() ); + // since OpenGL draws on the back buffer, + // we have to swap it to the front buffer to see it + + // draw informations with GDI on the front buffer + dc->SetBkMode( theApp.showSpeedTransparent ? TRANSPARENT : OPAQUE ); + if( theApp.showSpeed && ( theApp.videoOption > VIDEO_4X ) ) { + char buffer[30]; + if( theApp.showSpeed == 1 ) { + sprintf( buffer, "%3d%%", systemSpeed ); + } else { + sprintf( buffer, "%3d%%(%d, %d fps)", systemSpeed, systemFrameSkip, theApp.showRenderedFrames ); + } + dc->SetTextColor( RGB(0x00, 0x00, 0xFF) ); + dc->TextOut( 10, 20, buffer ); + } + if( theApp.screenMessage ) { + if( ( ( GetTickCount() - theApp.screenMessageTime ) < 3000 ) && !theApp.disableStatusMessage ) { + dc->SetTextColor( RGB(0xFF, 0x00, 0x00) ); + dc->TextOut( 10, theApp.surfaceSizeY - 20, theApp.screenMessageBuffer ); + } else { + theApp.screenMessage = false; + } + } + + theApp.m_pMainWnd->ReleaseDC( dc ); +} + + +void OpenGLDisplay::resize( int w, int h ) +{ + initializeMatrices( w, h ); +} + + +void OpenGLDisplay::updateFiltering( int value ) +{ + switch( value ) + { + case 0: + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST); + break; + case 1: + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR); + break; + } + + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP ); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP ); +} + + +void OpenGLDisplay::initializeMatrices( int w, int h ) +{ + if( theApp.fullScreenStretch ) { + glViewport( 0, 0, w, h ); + } else { + calculateDestRect( w, h ); + glViewport( + destRect.left, + destRect.top, + destRect.right - destRect.left, + destRect.bottom - destRect.top ); + } + + glMatrixMode( GL_PROJECTION ); + glLoadIdentity(); + glOrtho( + /* left */ 1.0f, + /* right */ (GLdouble)(w - 1), + /* bottom */ (GLdouble)(h - 1), + /* top */ 1.0f, + 0.0f, + 1.0f ); + + glMatrixMode(GL_MODELVIEW); + glLoadIdentity(); +} + + +bool OpenGLDisplay::initializeTexture( int w, int h ) +{ + // size = 2^n + // w = 24 > size = 256 = 2^8 + // w = 255 > size = 256 = 2^8 + // w = 256 > size = 512 = 2^9 + // w = 300 > size = 512 = 2^9 + // OpenGL textures have to be square and a power of 2 + + float n1 = log10( (float)w ) / log10( 2.0f ); + float n2 = log10( (float)h ) / log10( 2.0f ); + float n = ( n1 > n2 ) ? n1 : n2; + + if( ((float)((int)n)) != n ) { + // round up + n = ((float)((int)n)) + 1.0f; + } + + size = pow( 2.0f, n ); + + glGenTextures( 1, &texture ); + glBindTexture( GL_TEXTURE_2D, texture ); + updateFiltering( theApp.glFilter ); + + glTexImage2D( + GL_TEXTURE_2D, + 0, + GL_RGBA, + (GLsizei)size, + (GLsizei)size, + 0, + GL_RGBA, + GL_UNSIGNED_BYTE, + NULL ); + + width = w; + height = h; + + return ( glGetError() == GL_NO_ERROR) ? true : false; +} + + +void OpenGLDisplay::setVSync( int interval ) +{ + const char *extensions = (const char *)glGetString( GL_EXTENSIONS ); + + if( strstr( extensions, "WGL_EXT_swap_control" ) == 0 ) { + winlog( "Error: WGL_EXT_swap_control extension not supported on your computer.\n" ); + return; + } else { + PFNWGLSWAPINTERVALFARPROC wglSwapIntervalEXT = NULL; + wglSwapIntervalEXT = (PFNWGLSWAPINTERVALFARPROC)wglGetProcAddress( "wglSwapIntervalEXT" ); + if( wglSwapIntervalEXT ) { + wglSwapIntervalEXT( interval ); + } + } +} + + +bool OpenGLDisplay::changeRenderSize( int w, int h ) +{ + if( (width != w) || (height != h) ) { + if( texture != 0 ) { + glDeleteTextures( 1, &texture ); + texture = 0; + } + + if( !initializeTexture( w, h ) ) { + failed = true; + return false; + } + } + + return true; +} + + +void OpenGLDisplay::calculateDestRect( int w, int h ) +{ + float scaleX = (float)w / (float)width; + float scaleY = (float)h / (float)height; + float min = (scaleX < scaleY) ? scaleX : scaleY; + if( theApp.fsMaxScale && (min > theApp.fsMaxScale) ) { + min = (float)theApp.fsMaxScale; + } + destRect.left = 0; + destRect.top = 0; + destRect.right = (LONG)(width * min); + destRect.bottom = (LONG)(height * min); + if( destRect.right != w ) { + LONG diff = (w - destRect.right) / 2; + destRect.left += diff; + destRect.right += diff; + } + if( destRect.bottom != h ) { + LONG diff = (h - destRect.bottom) / 2; + destRect.top += diff; + destRect.bottom += diff; + } +} + + +void OpenGLDisplay::setOption( const char *option, int value ) +{ + if( !_tcscmp( option, _T("vsync") ) ) { + setVSync( value ); + } + + if( !_tcscmp( option, _T("glFilter") ) ) { + updateFiltering( value ); + } + + if( !_tcscmp( option, _T("maxScale") ) ) { + initializeMatrices( theApp.dest.right, theApp.dest.bottom ); + } + + if( !_tcscmp( option, _T("fullScreenStretch") ) ) { + initializeMatrices( theApp.dest.right, theApp.dest.bottom ); + } +} + + +int OpenGLDisplay::selectFullScreenMode( GUID ** ) +{ + HWND wnd = GetDesktopWindow(); + RECT r; + GetWindowRect( wnd, &r ); + int w = ( r.right - r.left ) & 0xFFF; + int h = ( r.bottom - r.top ) & 0xFFF; + HDC dc = GetDC( wnd ); + int c = GetDeviceCaps( dc, BITSPIXEL ); + ReleaseDC( wnd, dc ); + return (c << 24) | (w << 12) | h; +} + + +IDisplay *newOpenGLDisplay() +{ + return new OpenGLDisplay(); +} diff --git a/src/win32/PaletteView.cpp b/src/win32/PaletteView.cpp index bd4e72f0..5f164fe8 100644 --- a/src/win32/PaletteView.cpp +++ b/src/win32/PaletteView.cpp @@ -109,51 +109,54 @@ BOOL PaletteView::OnInitDialog() void PaletteView::save(int which) { - CString captureBuffer; + if(rom != NULL) + { + CString captureBuffer; - if(which == 0) - captureBuffer = "bg.pal"; - else - captureBuffer = "obj.pal"; + if(which == 0) + captureBuffer = "bg.pal"; + else + captureBuffer = "obj.pal"; - LPCTSTR exts[] = {".pal", ".pal", ".act" }; + LPCTSTR exts[] = {".pal", ".pal", ".act" }; - CString filter = theApp.winLoadFilter(IDS_FILTER_PAL); - CString title = winResLoadString(IDS_SELECT_PALETTE_NAME); - FileDlg dlg(this, - captureBuffer, - filter, - 1, - "PAL", - exts, - "", - title, - true); + CString filter = theApp.winLoadFilter(IDS_FILTER_PAL); + CString title = winResLoadString(IDS_SELECT_PALETTE_NAME); + FileDlg dlg(this, + captureBuffer, + filter, + 1, + "PAL", + exts, + "", + title, + true); - if(dlg.DoModal() == IDCANCEL) { - return; - } - - captureBuffer = dlg.GetPathName(); + if(dlg.DoModal() == IDCANCEL) { + return; + } - PaletteViewControl *p = NULL; + captureBuffer = dlg.GetPathName(); - if(which == 0) - p = &paletteView; - else - p = &paletteViewOBJ; + PaletteViewControl *p = NULL; + + if(which == 0) + p = &paletteView; + else + p = &paletteViewOBJ; - switch(dlg.getFilterIndex()) { - case 0: - case 1: - p->saveMSPAL(captureBuffer); - break; - case 2: - p->saveJASCPAL(captureBuffer); - break; - case 3: - p->saveAdobe(captureBuffer); - break; + switch(dlg.getFilterIndex()) { + case 0: + case 1: + p->saveMSPAL(captureBuffer); + break; + case 2: + p->saveJASCPAL(captureBuffer); + break; + case 3: + p->saveAdobe(captureBuffer); + break; + } } } diff --git a/src/win32/Reg.cpp b/src/win32/Reg.cpp index ac931c0a..90730671 100644 --- a/src/win32/Reg.cpp +++ b/src/win32/Reg.cpp @@ -1,369 +1,374 @@ -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#include "stdafx.h" - -static char buffer[2048]; -static HKEY vbKey = NULL; -static CString regVbaPath; - -#define VBA_PREF "preferences" - -bool regEnabled = true; - -void regInit(const char *path) -{ - DWORD disp = 0; - LONG res = RegCreateKeyEx(HKEY_CURRENT_USER, - "Software\\Emulators\\VisualBoyAdvance", - 0, - "", - REG_OPTION_NON_VOLATILE, - KEY_ALL_ACCESS, - NULL, - &vbKey, - &disp); - regVbaPath.Format("%s\\vba.ini", path); -} - -void regShutdown() -{ - LONG res = RegCloseKey(vbKey); -} - -const char *regGetINIPath() -{ - return regVbaPath; -} - -char *regQueryStringValue(const char * key, char *def) -{ - if(regEnabled) { - DWORD type = 0; - DWORD size = 2048; - - LONG res = RegQueryValueEx(vbKey, - key, - NULL, - &type, - (UCHAR *)buffer, - &size); - - if(res == ERROR_SUCCESS && type == REG_SZ) - return buffer; - - return def; - } - - DWORD res = GetPrivateProfileString(VBA_PREF, - key, - def, - (LPTSTR)buffer, - 2048, - regVbaPath); - - if(res) - return buffer; - - return def; -} - -DWORD regQueryDwordValue(const char * key, DWORD def, bool force) -{ - if(regEnabled || force) { - DWORD type = 0; - DWORD size = sizeof(DWORD); - DWORD result = 0; - - LONG res = RegQueryValueEx(vbKey, - key, - NULL, - &type, - (UCHAR *)&result, - &size); - - if(res == ERROR_SUCCESS && type == REG_DWORD) - return result; - - return def; - } - - return GetPrivateProfileInt(VBA_PREF, - key, - def, - regVbaPath); -} - -BOOL regQueryBinaryValue(const char * key, char *value, int count) -{ - if(regEnabled) { - DWORD type = 0; - DWORD size = count; - DWORD result = 0; - - - LONG res = RegQueryValueEx(vbKey, - key, - NULL, - &type, - (UCHAR *)value, - &size); - - if(res == ERROR_SUCCESS && type == REG_BINARY) - return TRUE; - - return FALSE; - } - CString k = key; - k += "Count"; - int size = GetPrivateProfileInt(VBA_PREF, - k, - -1, - regVbaPath); - if(size >= 0 && size < count) - count = size; - return GetPrivateProfileStruct(VBA_PREF, - key, - value, - count, - regVbaPath); -} - -void regSetStringValue(const char * key, const char * value) -{ - if(regEnabled) { - LONG res = RegSetValueEx(vbKey, - key, - NULL, - REG_SZ, - (const UCHAR *)value, - strlen(value)+1); - } else { - WritePrivateProfileString(VBA_PREF, - key, - value, - regVbaPath); - } -} - -void regSetDwordValue(const char * key, DWORD value, bool force) -{ - if(regEnabled || force) { - LONG res = RegSetValueEx(vbKey, - key, - NULL, - REG_DWORD, - (const UCHAR *)&value, - sizeof(DWORD)); - } else { - wsprintf(buffer, "%u", value); - WritePrivateProfileString(VBA_PREF, - key, - buffer, - regVbaPath); - } -} - -void regSetBinaryValue(const char *key, char *value, int count) -{ - if(regEnabled) { - LONG res = RegSetValueEx(vbKey, - key, - NULL, - REG_BINARY, - (const UCHAR *)value, - count); - } else { - CString k = key; - k += "Count"; - wsprintf(buffer, "%u", count); - - WritePrivateProfileString(VBA_PREF, - k, - buffer, - regVbaPath); - - WritePrivateProfileStruct(VBA_PREF, - key, - value, - count, - regVbaPath); - } -} - -void regDeleteValue(char *key) -{ - if(regEnabled) { - LONG res = RegDeleteValue(vbKey, - key); - } else { - WritePrivateProfileString(VBA_PREF, - key, - NULL, - regVbaPath); - } -} - -bool regCreateFileType(const char *ext, const char *type) -{ - DWORD disp = 0; - HKEY key; - LONG res = RegCreateKeyEx(HKEY_CLASSES_ROOT, - ext, - 0, - "", - REG_OPTION_NON_VOLATILE, - KEY_ALL_ACCESS, - NULL, - &key, - &disp); - if(res == ERROR_SUCCESS) { - res = RegSetValueEx(key, - "", - 0, - REG_SZ, - (const UCHAR *)type, - strlen(type)+1); - RegCloseKey(key); - return true; - } - return false; -} - -bool regAssociateType(const char *type, const char *desc, const char *application) -{ - DWORD disp = 0; - HKEY key; - LONG res = RegCreateKeyEx(HKEY_CLASSES_ROOT, - type, - 0, - "", - REG_OPTION_NON_VOLATILE, - KEY_ALL_ACCESS, - NULL, - &key, - &disp); - if(res == ERROR_SUCCESS) { - res = RegSetValueEx(key, - "", - 0, - REG_SZ, - (const UCHAR *)desc, - strlen(desc)+1); - HKEY key2; - res = RegCreateKeyEx(key, - "Shell\\Open\\Command", - 0, - "", - REG_OPTION_NON_VOLATILE, - KEY_ALL_ACCESS, - NULL, - &key2, - &disp); - if(res == ERROR_SUCCESS) { - res = RegSetValueEx(key2, - "", - 0, - REG_SZ, - (const UCHAR *)application, - strlen(application)+1); - RegCloseKey(key2); - RegCloseKey(key); - return true; - } - - RegCloseKey(key); - } - return false; -} - -static void regExportSettingsToINI(HKEY key, const char *section) -{ - char valueName[256]; - int index = 0; - while(1) { - DWORD nameSize = 256; - DWORD size = 2048; - DWORD type; - LONG res = RegEnumValue(key, - index, - valueName, - &nameSize, - NULL, - &type, - (LPBYTE)buffer, - &size); - - if(res == ERROR_SUCCESS) { - switch(type) { - case REG_DWORD: - { - char temp[256]; - wsprintf(temp, "%u", *((DWORD *)buffer)); - WritePrivateProfileString(section, - valueName, - temp, - regVbaPath); - } - break; - case REG_SZ: - WritePrivateProfileString(section, - valueName, - buffer, - regVbaPath); - break; - case REG_BINARY: - { - char temp[256]; - - wsprintf(temp, "%u", size); - CString k = valueName; - k += "Count"; - WritePrivateProfileString(section, - k, - temp, - regVbaPath); - WritePrivateProfileStruct(section, - valueName, - buffer, - size, - regVbaPath); - } - break; - } - index++; - } else - break; - } -} - -void regExportSettingsToINI() -{ - if(vbKey != NULL) { - regExportSettingsToINI(vbKey, VBA_PREF); - } - - HKEY key; - - if(RegOpenKey(HKEY_CURRENT_USER, - "Software\\Emulators\\VisualBoyAdvance\\Viewer", &key) == - ERROR_SUCCESS) { - regExportSettingsToINI(key, "Viewer"); - RegCloseKey(key); - } -} +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#include "stdafx.h" + +static char buffer[2048]; +static HKEY vbKey = NULL; +static CString *regVbaPath = NULL; + +#define VBA_PREF "preferences" + +bool regEnabled = true; + +void regInit(const char *path) +{ + DWORD disp = 0; + LONG res = RegCreateKeyEx(HKEY_CURRENT_USER, + "Software\\Emulators\\VisualBoyAdvance", + 0, + "", + REG_OPTION_NON_VOLATILE, + KEY_ALL_ACCESS, + NULL, + &vbKey, + &disp); + if( regVbaPath != NULL ) { + delete regVbaPath; + regVbaPath = NULL; + } + regVbaPath = new CString(); + regVbaPath->Format("%s\\vba.ini", path); +} + +void regShutdown() +{ + LONG res = RegCloseKey(vbKey); +} + +const char *regGetINIPath() +{ + return *regVbaPath; +} + +char *regQueryStringValue(const char * key, char *def) +{ + if(regEnabled) { + DWORD type = 0; + DWORD size = 2048; + + LONG res = RegQueryValueEx(vbKey, + key, + NULL, + &type, + (UCHAR *)buffer, + &size); + + if(res == ERROR_SUCCESS && type == REG_SZ) + return buffer; + + return def; + } + + DWORD res = GetPrivateProfileString(VBA_PREF, + key, + def, + (LPTSTR)buffer, + 2048, + *regVbaPath); + + if(res) + return buffer; + + return def; +} + +DWORD regQueryDwordValue(const char * key, DWORD def, bool force) +{ + if(regEnabled || force) { + DWORD type = 0; + DWORD size = sizeof(DWORD); + DWORD result = 0; + + LONG res = RegQueryValueEx(vbKey, + key, + NULL, + &type, + (UCHAR *)&result, + &size); + + if(res == ERROR_SUCCESS && type == REG_DWORD) + return result; + + return def; + } + + return GetPrivateProfileInt(VBA_PREF, + key, + def, + *regVbaPath); +} + +BOOL regQueryBinaryValue(const char * key, char *value, int count) +{ + if(regEnabled) { + DWORD type = 0; + DWORD size = count; + DWORD result = 0; + + + LONG res = RegQueryValueEx(vbKey, + key, + NULL, + &type, + (UCHAR *)value, + &size); + + if(res == ERROR_SUCCESS && type == REG_BINARY) + return TRUE; + + return FALSE; + } + CString k = key; + k += "Count"; + int size = GetPrivateProfileInt(VBA_PREF, + k, + -1, + *regVbaPath); + if(size >= 0 && size < count) + count = size; + return GetPrivateProfileStruct(VBA_PREF, + key, + value, + count, + *regVbaPath); +} + +void regSetStringValue(const char * key, const char * value) +{ + if(regEnabled) { + LONG res = RegSetValueEx(vbKey, + key, + NULL, + REG_SZ, + (const UCHAR *)value, + lstrlen(value)+1); + } else { + WritePrivateProfileString(VBA_PREF, + key, + value, + *regVbaPath); + } +} + +void regSetDwordValue(const char * key, DWORD value, bool force) +{ + if(regEnabled || force) { + LONG res = RegSetValueEx(vbKey, + key, + NULL, + REG_DWORD, + (const UCHAR *)&value, + sizeof(DWORD)); + } else { + wsprintf(buffer, "%u", value); + WritePrivateProfileString(VBA_PREF, + key, + buffer, + *regVbaPath); + } +} + +void regSetBinaryValue(const char *key, char *value, int count) +{ + if(regEnabled) { + LONG res = RegSetValueEx(vbKey, + key, + NULL, + REG_BINARY, + (const UCHAR *)value, + count); + } else { + CString k = key; + k += "Count"; + wsprintf(buffer, "%u", count); + + WritePrivateProfileString(VBA_PREF, + k, + buffer, + *regVbaPath); + + WritePrivateProfileStruct(VBA_PREF, + key, + value, + count, + *regVbaPath); + } +} + +void regDeleteValue(char *key) +{ + if(regEnabled) { + LONG res = RegDeleteValue(vbKey, + key); + } else { + WritePrivateProfileString(VBA_PREF, + key, + NULL, + *regVbaPath); + } +} + +bool regCreateFileType(const char *ext, const char *type) +{ + DWORD disp = 0; + HKEY key; + LONG res = RegCreateKeyEx(HKEY_CLASSES_ROOT, + ext, + 0, + "", + REG_OPTION_NON_VOLATILE, + KEY_ALL_ACCESS, + NULL, + &key, + &disp); + if(res == ERROR_SUCCESS) { + res = RegSetValueEx(key, + "", + 0, + REG_SZ, + (const UCHAR *)type, + lstrlen(type)+1); + RegCloseKey(key); + return true; + } + return false; +} + +bool regAssociateType(const char *type, const char *desc, const char *application) +{ + DWORD disp = 0; + HKEY key; + LONG res = RegCreateKeyEx(HKEY_CLASSES_ROOT, + type, + 0, + "", + REG_OPTION_NON_VOLATILE, + KEY_ALL_ACCESS, + NULL, + &key, + &disp); + if(res == ERROR_SUCCESS) { + res = RegSetValueEx(key, + "", + 0, + REG_SZ, + (const UCHAR *)desc, + lstrlen(desc)+1); + HKEY key2; + res = RegCreateKeyEx(key, + "Shell\\Open\\Command", + 0, + "", + REG_OPTION_NON_VOLATILE, + KEY_ALL_ACCESS, + NULL, + &key2, + &disp); + if(res == ERROR_SUCCESS) { + res = RegSetValueEx(key2, + "", + 0, + REG_SZ, + (const UCHAR *)application, + lstrlen(application)+1); + RegCloseKey(key2); + RegCloseKey(key); + return true; + } + + RegCloseKey(key); + } + return false; +} + +static void regExportSettingsToINI(HKEY key, const char *section) +{ + char valueName[256]; + int index = 0; + while(1) { + DWORD nameSize = 256; + DWORD size = 2048; + DWORD type; + LONG res = RegEnumValue(key, + index, + valueName, + &nameSize, + NULL, + &type, + (LPBYTE)buffer, + &size); + + if(res == ERROR_SUCCESS) { + switch(type) { + case REG_DWORD: + { + char temp[256]; + wsprintf(temp, "%u", *((DWORD *)buffer)); + WritePrivateProfileString(section, + valueName, + temp, + *regVbaPath); + } + break; + case REG_SZ: + WritePrivateProfileString(section, + valueName, + buffer, + *regVbaPath); + break; + case REG_BINARY: + { + char temp[256]; + + wsprintf(temp, "%u", size); + CString k = valueName; + k += "Count"; + WritePrivateProfileString(section, + k, + temp, + *regVbaPath); + WritePrivateProfileStruct(section, + valueName, + buffer, + size, + *regVbaPath); + } + break; + } + index++; + } else + break; + } +} + +void regExportSettingsToINI() +{ + if(vbKey != NULL) { + regExportSettingsToINI(vbKey, VBA_PREF); + } + + HKEY key; + + if(RegOpenKey(HKEY_CURRENT_USER, + "Software\\Emulators\\VisualBoyAdvance\\Viewer", &key) == + ERROR_SUCCESS) { + regExportSettingsToINI(key, "Viewer"); + RegCloseKey(key); + } +} diff --git a/src/win32/Reg.h b/src/win32/Reg.h index 1e658711..7d3350e9 100644 --- a/src/win32/Reg.h +++ b/src/win32/Reg.h @@ -1,37 +1,37 @@ -// -*- C++ -*- -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -#ifndef VBA_REG_H -#define VBA_REG_H - -extern bool regEnabled; - -char *regQueryStringValue(const char *key, char *def); -DWORD regQueryDwordValue(const char *key, DWORD def, bool force=false); -BOOL regQueryBinaryValue(const char *key, char *value, int count); -void regSetStringValue(const char *key,const char *value); -void regSetDwordValue(const char *key,DWORD value,bool force=false); -void regSetBinaryValue(const char *key, char *value, int count); -void regDeleteValue(char *key); -void regInit(const char *); -void regShutdown(); -bool regCreateFileType(const char *ext, const char *type); -bool regAssociateType(const char *type, const char *desc, const char *application); -void regExportSettingsToINI(); -#endif // VBA_REG_H +// -*- C++ -*- +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#ifndef VBA_REG_H +#define VBA_REG_H + +extern bool regEnabled; + +char *regQueryStringValue(const char *key, char *def); +DWORD regQueryDwordValue(const char *key, DWORD def, bool force=false); +BOOL regQueryBinaryValue(const char *key, char *value, int count); +void regSetStringValue(const char *key,const char *value); +void regSetDwordValue(const char *key,DWORD value,bool force=false); +void regSetBinaryValue(const char *key, char *value, int count); +void regDeleteValue(char *key); +void regInit(const char *); +void regShutdown(); +bool regCreateFileType(const char *ext, const char *type); +bool regAssociateType(const char *type, const char *desc, const char *application); +void regExportSettingsToINI(); +#endif // VBA_REG_H diff --git a/src/win32/ResizeDlg.cpp b/src/win32/ResizeDlg.cpp index af843dbe..93fb4735 100644 --- a/src/win32/ResizeDlg.cpp +++ b/src/win32/ResizeDlg.cpp @@ -1,561 +1,561 @@ -/*---------------------------------------------------------------------- - Copyright (c) Gipsysoft. All Rights Reserved. - File: DialogSizer_Set.cpp - Web site: http://gipsysoft.com - - This software is provided 'as-is', without any express or implied warranty. - - In no event will the author be held liable for any damages arising from the - use of this software. - - Permission is granted to anyone to use this software for any purpose, including - commercial applications, and to alter it and redistribute it freely, subject - to the following restrictions: - - 1) The origin of this software must not be misrepresented; you must not claim - that you wrote the original software. If you use this software in a product, - an acknowledgment in the product documentation is requested but not required. - 2) Altered source versions must be plainly marked as such, and must not be - misrepresented as being the original software. Altered source is encouraged - to be submitted back to the original author so it can be shared with the - community. Please share your changes. - 3) This notice may not be removed or altered from any source distribution. - - Owner: russf@gipsysoft.com - Purpose: Main functionality for sizeable dialogs - - Store a local copy of the user settings - Subclass the window - Respond to various messages withinn the subclassed window. - - ----------------------------------------------------------------------*/ -#include "stdafx.h" -#include "VBA.h" -#include "ResizeDlg.h" -#undef ASSERT -#include "WinHelper.h" - -// moved functions to this file to reduce number of files - -struct RegistryData -{ - WINDOWPLACEMENT m_wpl; -}; - - -struct DialogData // dd -{ - HKEY hkRootSave; - LPCTSTR pcszName; - - // - // The number of items contained in the psd member. - // Used in the DeferWindowPos structure and in allocating memory - int nItemCount; - DialogSizerSizingItem *psd; - - // - // We need the smallest to respond to the WM_GETMINMAXINFO message - POINT m_ptSmallest; - - // - // We don't strictly speaking need to say how big the biggest can be but - POINT m_ptLargest; - bool m_bLargestSet; - - // - // we need this to decide how much the window has changed size when we get a WM_SIZE message - SIZE m_sizeClient; - - // - // Draw the sizing grip...or not - bool m_bMaximised; - BOOL m_bShowSizingGrip; - - WinHelper::CRect m_rcGrip; -}; - -//extern bool regEnabled; -extern const char *regGetINIPath(); - -void AssertFailed(char *file, int line, char *exp) -{ - char buffer[1024]; - - sprintf(buffer, "File %s\nLine %d\nExpression %s\nPress Retry to debug", - file, line, exp); - int res = MessageBox(*theApp.m_pMainWnd, buffer, "Assertion failed!", - MB_ICONHAND | MB_SETFOREGROUND | MB_TASKMODAL | - MB_ABORTRETRYIGNORE); - - if(res == IDRETRY) { - __asm int 3; - } else if(res == IDABORT) - SendMessage(*theApp.m_pMainWnd, WM_QUIT, 0, 0); -} - -void ApiFailure(char *pcszFilename, int nLine, char *pcszExpression ) -{ - const DWORD dwLastError = ::GetLastError(); - LPCTSTR lpMsgBuf; - (void)::FormatMessage( FORMAT_MESSAGE_ALLOCATE_BUFFER | - FORMAT_MESSAGE_FROM_SYSTEM | - FORMAT_MESSAGE_IGNORE_INSERTS, - NULL, dwLastError, - MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), - (LPTSTR) &lpMsgBuf, 0, NULL ); - - char szExeName[ MAX_PATH ]; - - if( !GetModuleFileName( NULL, szExeName, countof( szExeName ) ) ) - strcpy( szExeName, "" ); - - - char szMessage[ 1024 ]; - _snprintf( szMessage, countof( szMessage ) - , "API VERIFY Failure!" - "\nProgram: %s" - "\n" - "\nFile %s" - "\nLine %d" - "\n" - "\nExpression %s" - "\n" - "\nLast Error %d" - "\n %s" - "\n\nPress Retry to debug the application" - , szExeName - , pcszFilename - , nLine - , pcszExpression - , dwLastError - , lpMsgBuf - ); - - (void)LocalFree( (LPVOID)lpMsgBuf ); - HWND hwndParent = ::GetActiveWindow(); - hwndParent = ::GetLastActivePopup( hwndParent ); - int nCode = ::MessageBoxA( hwndParent, - szMessage, - "Debug Helper", - MB_TASKMODAL | MB_ICONHAND | MB_ABORTRETRYIGNORE | - MB_SETFOREGROUND ); - if(nCode == IDABORT) { - ::SendMessage(*theApp.m_pMainWnd, WM_QUIT, 0, 0); - } else if(nCode == IDRETRY) - __asm int 3; -} - -long FASTCALL RegQueryValueExRecursive( HKEY hKey, LPCTSTR lpValueName, LPDWORD lpReserved, LPDWORD lpType, LPBYTE lpData, LPDWORD lpcbData ) -{ - TCHAR szBuffer[ 256 ]; - R_ASSERT( lstrlen( lpValueName ) < countof( szBuffer ) ); - (void)lstrcpy( szBuffer, lpValueName ); - - LPTSTR pszBuffer = szBuffer; - LPTSTR pszLast = szBuffer; - while( *pszBuffer ) - { - if( *pszBuffer == _T('\\') || *pszBuffer == _T('/') ) - { - pszLast = pszBuffer; - lpValueName = pszLast + 1; - } - pszBuffer++; - } - - /*if(!regEnabled) {*/ - if(GetPrivateProfileStruct("Viewer", - lpValueName, - lpData, - *lpcbData, - regGetINIPath())) { - *lpType = REG_BINARY; - return ERROR_SUCCESS; - } - return -1; - /*} - - bool m_bNeedToCloseKey = false; - if( pszLast != szBuffer ) - { - *pszLast = _T('\000'); - HKEY hkeyTemp; - long lRet = RegOpenKey( hKey, szBuffer, &hkeyTemp ); - if( lRet != ERROR_SUCCESS ) - { - return lRet; - } - hKey = hkeyTemp; - m_bNeedToCloseKey = true; - } - - long lRet = RegQueryValueEx( hKey, lpValueName, lpReserved, lpType, lpData, lpcbData ); - if( m_bNeedToCloseKey ) - { - R_VERIFY( RegCloseKey( hKey ) == ERROR_SUCCESS ); - } - return lRet;*/ -} - -long FASTCALL RegSetValueExRecursive( HKEY hKey, LPCTSTR lpValueName, DWORD Reserved, DWORD dwType, CONST BYTE* lpData, DWORD cbData ) -{ - TCHAR szBuffer[ 256 ]; - R_ASSERT( lstrlen( lpValueName ) < countof( szBuffer ) ); - (void)lstrcpy( szBuffer, lpValueName ); - - LPTSTR pszBuffer = szBuffer; - LPTSTR pszLast = szBuffer; - while( *pszBuffer ) - { - if( *pszBuffer == _T('\\') || *pszBuffer == _T('/') ) - { - pszLast = pszBuffer; - lpValueName = pszLast + 1; - } - pszBuffer++; - } - - /*if(!regEnabled) {*/ - if(WritePrivateProfileStruct("Viewer", - lpValueName, - (LPVOID)lpData, - cbData, - regGetINIPath())) { - return ERROR_SUCCESS; - } - return -1; - /*} - - bool m_bNeedToCloseKey = false; - if( pszLast != szBuffer ) - { - *pszLast = _T('\000'); - HKEY hkeyTemp; - long lRet = RegOpenKey( hKey, szBuffer, &hkeyTemp ); - if( lRet != ERROR_SUCCESS ) - { - lRet = RegCreateKey( hKey, szBuffer, &hkeyTemp ); - if( lRet != ERROR_SUCCESS ) - return lRet; - } - hKey = hkeyTemp; - m_bNeedToCloseKey = true; - } - - long lRet = RegSetValueEx( hKey, lpValueName, Reserved, dwType, lpData, cbData ); - if( m_bNeedToCloseKey ) - { - R_VERIFY( RegCloseKey( hKey ) == ERROR_SUCCESS ); - } - return lRet;*/ -} - - -int ResizeDlgGetItemCount(const DialogSizerSizingItem *psd) -{ - R_ASSERT( psd ); - int nCount = 0; - while( psd->uSizeInfo != 0xFFFFFFFF ) - { - nCount++; - psd++; - } - return nCount; -} - -void ResizeDlgUpdateGripperRect( const int cx, const int cy, WinHelper::CRect &rcGrip ) -{ - const int nGripWidth = GetSystemMetrics( SM_CYVSCROLL ); - const int nGripHeight = GetSystemMetrics( SM_CXVSCROLL ); - rcGrip.left = cx - nGripWidth; - rcGrip.top = cy - nGripHeight; - rcGrip.right = cx; - rcGrip.bottom = cy; -} - -void ResizeDlgUpdateGripper( HWND hwnd, DialogData *pdd ) -{ - if( pdd->m_bShowSizingGrip ) - { - WinHelper::CRect rcOld( pdd->m_rcGrip ); - - ResizeDlgUpdateGripperRect( pdd->m_sizeClient.cx, pdd->m_sizeClient.cy, pdd->m_rcGrip ); - - // - // We also need to invalidate the combined area of the old and new rectangles - // otherwise we would have trail of grippers when we sized the dialog larger - // in any axis - (void)UnionRect( &rcOld, &rcOld, &pdd->m_rcGrip ); - (void)InvalidateRect( hwnd, &rcOld, TRUE ); - } -} - -void ResizeDlgCopyItems( DialogSizerSizingItem *psdDest, const DialogSizerSizingItem *psdSource ) - // - // Will copy all of the items in psdSource into psdDest. -{ - // - // Loop til we reach the end - while( psdSource->uSizeInfo != 0xFFFFFFFF ) - { - *psdDest = *psdSource; - psdDest++; - psdSource++; - } - // And when we do copy the last item - *psdDest = *psdSource; -} - - -ResizeDlg::ResizeDlg(UINT id, CWnd *parent) - : CDialog(id, parent) -{ - dd = NULL; -} - -void *ResizeDlg::AddDialogData() - // - // Firstly determine if the data already exists, if it does then return that, if not then we will - // create and initialise a brand new structure. -{ - DialogData *pdd = (DialogData*)dd; - if( !pdd ) { - pdd = (DialogData *)calloc(1, sizeof(DialogData)); - } - - if( pdd ) { - // - // Store some sizes etc. for later. - CRect rc; - GetWindowRect( rc ); - pdd->m_ptSmallest.x = rc.Width(); - pdd->m_ptSmallest.y = rc.Height(); - - - GetClientRect( rc ); - pdd->m_sizeClient = rc.Size(); - dd = pdd; - ResizeDlgUpdateGripperRect( pdd->m_sizeClient.cx, pdd->m_sizeClient.cy, pdd->m_rcGrip ); - } - return pdd; -} - -BOOL ResizeDlg::SetData(const DialogSizerSizingItem *psd, - BOOL bShowSizingGrip, - HKEY hkRootSave, - LPCTSTR pcszName, - SIZE *psizeMax) - // - // Setting a dialog sizeable involves subclassing the window and handling it's - // WM_SIZE messages, if we have a hkRootSave and pcszName then we will also be loading/saving - // the size and position of the window from the registry. We load from the registry when we - // subclass the window and we save to the registry when we get a WM_DESTROY. - // - // It will return non-zero for success and zero if it fails -{ - R_ASSERT( psd ); - R_ASSERT( ( hkRootSave != NULL && pcszName != NULL ) - || ( hkRootSave == NULL && pcszName == NULL ) ); - // - // Make sure all of the parameters are valid. - if( ::IsWindow( *this ) - && psd - && ( ( hkRootSave != NULL && pcszName != NULL && - !IsBadStringPtr( pcszName, 0xFFFF ) ) || - ( hkRootSave == NULL && pcszName == NULL ) ) - && ( psizeMax == NULL || !IsBadReadPtr( psizeMax, sizeof( SIZE ) ) ) - ) { - DialogData *pdd = (DialogData *)AddDialogData(); - if( pdd ) { - pdd->hkRootSave = hkRootSave; - pdd->pcszName = pcszName; - pdd->m_bShowSizingGrip = bShowSizingGrip; - pdd->nItemCount = ResizeDlgGetItemCount( psd ) + 1; - pdd->psd = (DialogSizerSizingItem *) - calloc(pdd->nItemCount, - sizeof(DialogSizerSizingItem )); - if( pdd->psd ) { - // - // Copy all of the user controls etc. for later, this way the user can quite happily - // let the structure go out of scope. - ResizeDlgCopyItems( pdd->psd, psd ); - if( psizeMax ) { - pdd->m_ptLargest.x = psizeMax->cx; - pdd->m_ptLargest.y = psizeMax->cy; - pdd->m_bLargestSet = true; - } - - // - // If the there was save info passed in then we need to make damn good use of it - // by attempting to load the RegistryData structure - if( hkRootSave && pcszName ) { - RegistryData rd; - DWORD dwSize = sizeof( RegistryData ); - DWORD dwType = REG_BINARY; - if( RegQueryValueExRecursive( hkRootSave, pcszName, NULL, &dwType, reinterpret_cast( &rd ), &dwSize ) == ERROR_SUCCESS && dwSize == sizeof( rd ) ) { - if( !(GetWindowLong( *this, GWL_STYLE ) & WS_VISIBLE) ) - rd.m_wpl.showCmd = SW_HIDE; - - VAPI( SetWindowPlacement( &rd.m_wpl ) ); - } - } - return TRUE; - } else { - free(pdd); - } - } - } - return FALSE; -} - -void ResizeDlg::UpdateWindowSize(const int cx, const int cy, HWND hwnd) -{ - DialogData *pdd = (DialogData*)dd; - if( pdd ) { - const int nDeltaX = cx - pdd->m_sizeClient.cx; - const int nDeltaY = cy - pdd->m_sizeClient.cy; - WinHelper::CDeferWindowPos def( pdd->nItemCount ); - WinHelper::CRect rc; - const DialogSizerSizingItem *psd = pdd->psd; - while( psd->uSizeInfo != 0xFFFFFFFF ) { - HWND hwndChild = ::GetDlgItem( *this, psd->uControlID ); - if( ::IsWindow( hwndChild ) ) { - VAPI( ::GetWindowRect( hwndChild, rc ) ); - (void)::MapWindowPoints( ::GetDesktopWindow(), hwnd, - (LPPOINT)&rc, 2 ); - - // - // Adjust the window horizontally - if( psd->uSizeInfo & DS_MoveX ) { - rc.left += nDeltaX; - rc.right += nDeltaX; - } - - // - // Adjust the window vertically - if( psd->uSizeInfo & DS_MoveY ) { - rc.top += nDeltaY; - rc.bottom += nDeltaY; - } - - // - // Size the window horizontally - if( psd->uSizeInfo & DS_SizeX ) { - rc.right += nDeltaX; - } - - // - // Size the window vertically - if( psd->uSizeInfo & DS_SizeY ) { - rc.bottom += nDeltaY; - } - - (void)def.DeferWindowPos( hwndChild, NULL, rc, - SWP_NOACTIVATE | SWP_NOZORDER ); - } - psd++; - } - - pdd->m_sizeClient.cx = cx; - pdd->m_sizeClient.cy = cy; - - // - // If we have a sizing grip enabled then adjust it's position - ResizeDlgUpdateGripper( hwnd, pdd ); - } -} - -BOOL ResizeDlg::OnWndMsg(UINT msg, WPARAM wParam, LPARAM lParam, LRESULT *res ) - // Actual window procedure that will handle saving window size/position and moving - // the controls whilst the window sizes. -{ - if(dd == NULL) { - return CDialog::OnWndMsg(msg, wParam, lParam, res); - } - switch( msg ) { - case WM_ERASEBKGND: - { - BOOL r = CDialog::OnWndMsg(msg, wParam, lParam, res); - DialogData *pdd = (DialogData*)dd; - if( pdd && pdd->m_bShowSizingGrip && !pdd->m_bMaximised ) { - VAPI( ::DrawFrameControl( reinterpret_cast( wParam ), - pdd->m_rcGrip, - DFC_SCROLL, DFCS_SCROLLSIZEGRIP ) ); - } - return r; - } - case WM_SIZE: - { - DialogData *pdd = (DialogData*)dd; - if( pdd && wParam != SIZE_MINIMIZED ) { - pdd->m_bMaximised = ( wParam == SIZE_MAXIMIZED ? true : false ); - UpdateWindowSize( LOWORD( lParam ), HIWORD( lParam ), *this); - } - } - break; - case WM_NCHITTEST: - { - // - // If the gripper is enabled then perform a simple hit test on our gripper area. - DialogData *pdd = (DialogData*)dd; - if( pdd && pdd->m_bShowSizingGrip ) { - POINT pt = { LOWORD(lParam), HIWORD(lParam) }; - (void)ScreenToClient( &pt ); - if( PtInRect( pdd->m_rcGrip, pt ) ) - return (BOOL)HTBOTTOMRIGHT; - } - } - break; - case WM_GETMINMAXINFO: - { - // - // Our opportunity to say that we do not want the dialog to grow or shrink any more. - DialogData *pdd = (DialogData*)dd; - LPMINMAXINFO lpmmi = reinterpret_cast( lParam ); - lpmmi->ptMinTrackSize = pdd->m_ptSmallest; - if( pdd->m_bLargestSet ) { - lpmmi->ptMaxTrackSize = pdd->m_ptLargest; - } - } - return (BOOL)0; - case WM_NOTIFY: - { - if( reinterpret_cast(lParam)->code == PSN_SETACTIVE ) { - CRect rc; - VAPI( ::GetClientRect( *GetParent( ), &rc ) ); - UpdateWindowSize( rc.Width(), rc.Height(), *GetParent( ) ); - } - } - break; - case WM_DESTROY: - { - // - // Our opportunty for cleanup. - // Simply acquire all of our objects, free the appropriate memory and remove the - // properties from the window. If we do not remove the properties then they will constitute - // a resource leak. - DialogData *pdd = (DialogData*)dd; - if( pdd ) { - RegistryData rd; - rd.m_wpl.length = sizeof( rd.m_wpl ); - VAPI( GetWindowPlacement( &rd.m_wpl ) ); - - if( pdd->hkRootSave && pdd->pcszName ) { - (void)RegSetValueExRecursive( pdd->hkRootSave, pdd->pcszName, - NULL, REG_BINARY, - reinterpret_cast( &rd ), - sizeof( rd ) ); - } - - if( pdd->psd ) { - free(pdd->psd); - } - free(pdd); - } - - } - break; - } - return CDialog::OnWndMsg(msg, wParam, lParam, res); -} +/*---------------------------------------------------------------------- + Copyright (c) Gipsysoft. All Rights Reserved. + File: DialogSizer_Set.cpp + Web site: http://gipsysoft.com + + This software is provided 'as-is', without any express or implied warranty. + + In no event will the author be held liable for any damages arising from the + use of this software. + + Permission is granted to anyone to use this software for any purpose, including + commercial applications, and to alter it and redistribute it freely, subject + to the following restrictions: + + 1) The origin of this software must not be misrepresented; you must not claim + that you wrote the original software. If you use this software in a product, + an acknowledgment in the product documentation is requested but not required. + 2) Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. Altered source is encouraged + to be submitted back to the original author so it can be shared with the + community. Please share your changes. + 3) This notice may not be removed or altered from any source distribution. + + Owner: russf@gipsysoft.com + Purpose: Main functionality for sizeable dialogs + + Store a local copy of the user settings + Subclass the window + Respond to various messages withinn the subclassed window. + + ----------------------------------------------------------------------*/ +#include "stdafx.h" +#include "VBA.h" +#include "ResizeDlg.h" +#undef ASSERT +#include "WinHelper.h" + +// moved functions to this file to reduce number of files + +struct RegistryData +{ + WINDOWPLACEMENT m_wpl; +}; + + +struct DialogData // dd +{ + HKEY hkRootSave; + LPCTSTR pcszName; + + // + // The number of items contained in the psd member. + // Used in the DeferWindowPos structure and in allocating memory + int nItemCount; + DialogSizerSizingItem *psd; + + // + // We need the smallest to respond to the WM_GETMINMAXINFO message + POINT m_ptSmallest; + + // + // We don't strictly speaking need to say how big the biggest can be but + POINT m_ptLargest; + bool m_bLargestSet; + + // + // we need this to decide how much the window has changed size when we get a WM_SIZE message + SIZE m_sizeClient; + + // + // Draw the sizing grip...or not + bool m_bMaximised; + BOOL m_bShowSizingGrip; + + WinHelper::CRect m_rcGrip; +}; + +extern bool regEnabled; +extern const char *regGetINIPath(); + +void AssertFailed(char *file, int line, char *exp) +{ + char buffer[1024]; + + sprintf(buffer, "File %s\nLine %d\nExpression %s\nPress Retry to debug", + file, line, exp); + int res = MessageBox(*theApp.m_pMainWnd, buffer, "Assertion failed!", + MB_ICONHAND | MB_SETFOREGROUND | MB_TASKMODAL | + MB_ABORTRETRYIGNORE); + + if(res == IDRETRY) { + DebugBreak(); + } else if(res == IDABORT) + SendMessage(*theApp.m_pMainWnd, WM_QUIT, 0, 0); +} + +void ApiFailure(char *pcszFilename, int nLine, char *pcszExpression ) +{ + const DWORD dwLastError = ::GetLastError(); + LPCTSTR lpMsgBuf; + (void)::FormatMessage( FORMAT_MESSAGE_ALLOCATE_BUFFER | + FORMAT_MESSAGE_FROM_SYSTEM | + FORMAT_MESSAGE_IGNORE_INSERTS, + NULL, dwLastError, + MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), + (LPTSTR) &lpMsgBuf, 0, NULL ); + + char szExeName[ MAX_PATH ]; + + if( !GetModuleFileName( NULL, szExeName, countof( szExeName ) ) ) + strcpy( szExeName, "" ); + + + char szMessage[ 1024 ]; + _snprintf( szMessage, countof( szMessage ) + , "API VERIFY Failure!" + "\nProgram: %s" + "\n" + "\nFile %s" + "\nLine %d" + "\n" + "\nExpression %s" + "\n" + "\nLast Error %d" + "\n %s" + "\n\nPress Retry to debug the application" + , szExeName + , pcszFilename + , nLine + , pcszExpression + , dwLastError + , lpMsgBuf + ); + + (void)LocalFree( (LPVOID)lpMsgBuf ); + HWND hwndParent = ::GetActiveWindow(); + hwndParent = ::GetLastActivePopup( hwndParent ); + int nCode = ::MessageBoxA( hwndParent, + szMessage, + "Debug Helper", + MB_TASKMODAL | MB_ICONHAND | MB_ABORTRETRYIGNORE | + MB_SETFOREGROUND ); + if(nCode == IDABORT) { + ::SendMessage(*theApp.m_pMainWnd, WM_QUIT, 0, 0); + } else if(nCode == IDRETRY) + DebugBreak(); +} + +long FASTCALL RegQueryValueExRecursive( HKEY hKey, LPCTSTR lpValueName, LPDWORD lpReserved, LPDWORD lpType, LPBYTE lpData, LPDWORD lpcbData ) +{ + TCHAR szBuffer[ 256 ]; + R_ASSERT( lstrlen( lpValueName ) < countof( szBuffer ) ); + (void)lstrcpy( szBuffer, lpValueName ); + + LPTSTR pszBuffer = szBuffer; + LPTSTR pszLast = szBuffer; + while( *pszBuffer ) + { + if( *pszBuffer == _T('\\') || *pszBuffer == _T('/') ) + { + pszLast = pszBuffer; + lpValueName = pszLast + 1; + } + pszBuffer++; + } + + if(!regEnabled) { + if(GetPrivateProfileStruct("Viewer", + lpValueName, + lpData, + *lpcbData, + regGetINIPath())) { + *lpType = REG_BINARY; + return ERROR_SUCCESS; + } + return -1; + } + + bool m_bNeedToCloseKey = false; + if( pszLast != szBuffer ) + { + *pszLast = _T('\000'); + HKEY hkeyTemp; + long lRet = RegOpenKey( hKey, szBuffer, &hkeyTemp ); + if( lRet != ERROR_SUCCESS ) + { + return lRet; + } + hKey = hkeyTemp; + m_bNeedToCloseKey = true; + } + + long lRet = RegQueryValueEx( hKey, lpValueName, lpReserved, lpType, lpData, lpcbData ); + if( m_bNeedToCloseKey ) + { + R_VERIFY( RegCloseKey( hKey ) == ERROR_SUCCESS ); + } + return lRet; +} + +long FASTCALL RegSetValueExRecursive( HKEY hKey, LPCTSTR lpValueName, DWORD Reserved, DWORD dwType, CONST BYTE* lpData, DWORD cbData ) +{ + TCHAR szBuffer[ 256 ]; + R_ASSERT( lstrlen( lpValueName ) < countof( szBuffer ) ); + (void)lstrcpy( szBuffer, lpValueName ); + + LPTSTR pszBuffer = szBuffer; + LPTSTR pszLast = szBuffer; + while( *pszBuffer ) + { + if( *pszBuffer == _T('\\') || *pszBuffer == _T('/') ) + { + pszLast = pszBuffer; + lpValueName = pszLast + 1; + } + pszBuffer++; + } + + if(!regEnabled) { + if(WritePrivateProfileStruct("Viewer", + lpValueName, + (LPVOID)lpData, + cbData, + regGetINIPath())) { + return ERROR_SUCCESS; + } + return -1; + } + + bool m_bNeedToCloseKey = false; + if( pszLast != szBuffer ) + { + *pszLast = _T('\000'); + HKEY hkeyTemp; + long lRet = RegOpenKey( hKey, szBuffer, &hkeyTemp ); + if( lRet != ERROR_SUCCESS ) + { + lRet = RegCreateKey( hKey, szBuffer, &hkeyTemp ); + if( lRet != ERROR_SUCCESS ) + return lRet; + } + hKey = hkeyTemp; + m_bNeedToCloseKey = true; + } + + long lRet = RegSetValueEx( hKey, lpValueName, Reserved, dwType, lpData, cbData ); + if( m_bNeedToCloseKey ) + { + R_VERIFY( RegCloseKey( hKey ) == ERROR_SUCCESS ); + } + return lRet; +} + + +int ResizeDlgGetItemCount(const DialogSizerSizingItem *psd) +{ + R_ASSERT( psd ); + int nCount = 0; + while( psd->uSizeInfo != 0xFFFFFFFF ) + { + nCount++; + psd++; + } + return nCount; +} + +void ResizeDlgUpdateGripperRect( const int cx, const int cy, WinHelper::CRect &rcGrip ) +{ + const int nGripWidth = GetSystemMetrics( SM_CYVSCROLL ); + const int nGripHeight = GetSystemMetrics( SM_CXVSCROLL ); + rcGrip.left = cx - nGripWidth; + rcGrip.top = cy - nGripHeight; + rcGrip.right = cx; + rcGrip.bottom = cy; +} + +void ResizeDlgUpdateGripper( HWND hwnd, DialogData *pdd ) +{ + if( pdd->m_bShowSizingGrip ) + { + WinHelper::CRect rcOld( pdd->m_rcGrip ); + + ResizeDlgUpdateGripperRect( pdd->m_sizeClient.cx, pdd->m_sizeClient.cy, pdd->m_rcGrip ); + + // + // We also need to invalidate the combined area of the old and new rectangles + // otherwise we would have trail of grippers when we sized the dialog larger + // in any axis + (void)UnionRect( &rcOld, &rcOld, &pdd->m_rcGrip ); + (void)InvalidateRect( hwnd, &rcOld, TRUE ); + } +} + +void ResizeDlgCopyItems( DialogSizerSizingItem *psdDest, const DialogSizerSizingItem *psdSource ) + // + // Will copy all of the items in psdSource into psdDest. +{ + // + // Loop til we reach the end + while( psdSource->uSizeInfo != 0xFFFFFFFF ) + { + *psdDest = *psdSource; + psdDest++; + psdSource++; + } + // And when we do copy the last item + *psdDest = *psdSource; +} + + +ResizeDlg::ResizeDlg(UINT id, CWnd *parent) + : CDialog(id, parent) +{ + dd = NULL; +} + +void *ResizeDlg::AddDialogData() + // + // Firstly determine if the data already exists, if it does then return that, if not then we will + // create and initialise a brand new structure. +{ + DialogData *pdd = (DialogData*)dd; + if( !pdd ) { + pdd = (DialogData *)calloc(1, sizeof(DialogData)); + } + + if( pdd ) { + // + // Store some sizes etc. for later. + CRect rc; + GetWindowRect( rc ); + pdd->m_ptSmallest.x = rc.Width(); + pdd->m_ptSmallest.y = rc.Height(); + + + GetClientRect( rc ); + pdd->m_sizeClient = rc.Size(); + dd = pdd; + ResizeDlgUpdateGripperRect( pdd->m_sizeClient.cx, pdd->m_sizeClient.cy, pdd->m_rcGrip ); + } + return pdd; +} + +BOOL ResizeDlg::SetData(const DialogSizerSizingItem *psd, + BOOL bShowSizingGrip, + HKEY hkRootSave, + LPCTSTR pcszName, + SIZE *psizeMax) + // + // Setting a dialog sizeable involves subclassing the window and handling it's + // WM_SIZE messages, if we have a hkRootSave and pcszName then we will also be loading/saving + // the size and position of the window from the registry. We load from the registry when we + // subclass the window and we save to the registry when we get a WM_DESTROY. + // + // It will return non-zero for success and zero if it fails +{ + R_ASSERT( psd ); + R_ASSERT( ( hkRootSave != NULL && pcszName != NULL ) + || ( hkRootSave == NULL && pcszName == NULL ) ); + // + // Make sure all of the parameters are valid. + if( ::IsWindow( *this ) + && psd + && ( ( hkRootSave != NULL && pcszName != NULL && + !IsBadStringPtr( pcszName, 0xFFFF ) ) || + ( hkRootSave == NULL && pcszName == NULL ) ) + && ( psizeMax == NULL || !IsBadReadPtr( psizeMax, sizeof( SIZE ) ) ) + ) { + DialogData *pdd = (DialogData *)AddDialogData(); + if( pdd ) { + pdd->hkRootSave = hkRootSave; + pdd->pcszName = pcszName; + pdd->m_bShowSizingGrip = bShowSizingGrip; + pdd->nItemCount = ResizeDlgGetItemCount( psd ) + 1; + pdd->psd = (DialogSizerSizingItem *) + calloc(pdd->nItemCount, + sizeof(DialogSizerSizingItem )); + if( pdd->psd ) { + // + // Copy all of the user controls etc. for later, this way the user can quite happily + // let the structure go out of scope. + ResizeDlgCopyItems( pdd->psd, psd ); + if( psizeMax ) { + pdd->m_ptLargest.x = psizeMax->cx; + pdd->m_ptLargest.y = psizeMax->cy; + pdd->m_bLargestSet = true; + } + + // + // If the there was save info passed in then we need to make damn good use of it + // by attempting to load the RegistryData structure + if( hkRootSave && pcszName ) { + RegistryData rd; + DWORD dwSize = sizeof( RegistryData ); + DWORD dwType = REG_BINARY; + if( RegQueryValueExRecursive( hkRootSave, pcszName, NULL, &dwType, reinterpret_cast( &rd ), &dwSize ) == ERROR_SUCCESS && dwSize == sizeof( rd ) ) { + if( !(GetWindowLong( *this, GWL_STYLE ) & WS_VISIBLE) ) + rd.m_wpl.showCmd = SW_HIDE; + + VAPI( SetWindowPlacement( &rd.m_wpl ) ); + } + } + return TRUE; + } else { + free(pdd); + } + } + } + return FALSE; +} + +void ResizeDlg::UpdateWindowSize(const int cx, const int cy, HWND hwnd) +{ + DialogData *pdd = (DialogData*)dd; + if( pdd ) { + const int nDeltaX = cx - pdd->m_sizeClient.cx; + const int nDeltaY = cy - pdd->m_sizeClient.cy; + WinHelper::CDeferWindowPos def( pdd->nItemCount ); + WinHelper::CRect rc; + const DialogSizerSizingItem *psd = pdd->psd; + while( psd->uSizeInfo != 0xFFFFFFFF ) { + HWND hwndChild = ::GetDlgItem( *this, psd->uControlID ); + if( ::IsWindow( hwndChild ) ) { + VAPI( ::GetWindowRect( hwndChild, rc ) ); + (void)::MapWindowPoints( ::GetDesktopWindow(), hwnd, + (LPPOINT)&rc, 2 ); + + // + // Adjust the window horizontally + if( psd->uSizeInfo & DS_MoveX ) { + rc.left += nDeltaX; + rc.right += nDeltaX; + } + + // + // Adjust the window vertically + if( psd->uSizeInfo & DS_MoveY ) { + rc.top += nDeltaY; + rc.bottom += nDeltaY; + } + + // + // Size the window horizontally + if( psd->uSizeInfo & DS_SizeX ) { + rc.right += nDeltaX; + } + + // + // Size the window vertically + if( psd->uSizeInfo & DS_SizeY ) { + rc.bottom += nDeltaY; + } + + (void)def.DeferWindowPos( hwndChild, NULL, rc, + SWP_NOACTIVATE | SWP_NOZORDER ); + } + psd++; + } + + pdd->m_sizeClient.cx = cx; + pdd->m_sizeClient.cy = cy; + + // + // If we have a sizing grip enabled then adjust it's position + ResizeDlgUpdateGripper( hwnd, pdd ); + } +} + +BOOL ResizeDlg::OnWndMsg(UINT msg, WPARAM wParam, LPARAM lParam, LRESULT *res ) + // Actual window procedure that will handle saving window size/position and moving + // the controls whilst the window sizes. +{ + if(dd == NULL) { + return CDialog::OnWndMsg(msg, wParam, lParam, res); + } + switch( msg ) { + case WM_ERASEBKGND: + { + BOOL r = CDialog::OnWndMsg(msg, wParam, lParam, res); + DialogData *pdd = (DialogData*)dd; + if( pdd && pdd->m_bShowSizingGrip && !pdd->m_bMaximised ) { + VAPI( ::DrawFrameControl( reinterpret_cast( wParam ), + pdd->m_rcGrip, + DFC_SCROLL, DFCS_SCROLLSIZEGRIP ) ); + } + return r; + } + case WM_SIZE: + { + DialogData *pdd = (DialogData*)dd; + if( pdd && wParam != SIZE_MINIMIZED ) { + pdd->m_bMaximised = ( wParam == SIZE_MAXIMIZED ? true : false ); + UpdateWindowSize( LOWORD( lParam ), HIWORD( lParam ), *this); + } + } + break; + case WM_NCHITTEST: + { + // + // If the gripper is enabled then perform a simple hit test on our gripper area. + DialogData *pdd = (DialogData*)dd; + if( pdd && pdd->m_bShowSizingGrip ) { + POINT pt = { LOWORD(lParam), HIWORD(lParam) }; + (void)ScreenToClient( &pt ); + if( PtInRect( pdd->m_rcGrip, pt ) ) + return (BOOL)HTBOTTOMRIGHT; + } + } + break; + case WM_GETMINMAXINFO: + { + // + // Our opportunity to say that we do not want the dialog to grow or shrink any more. + DialogData *pdd = (DialogData*)dd; + LPMINMAXINFO lpmmi = reinterpret_cast( lParam ); + lpmmi->ptMinTrackSize = pdd->m_ptSmallest; + if( pdd->m_bLargestSet ) { + lpmmi->ptMaxTrackSize = pdd->m_ptLargest; + } + } + return (BOOL)0; + case WM_NOTIFY: + { + if( reinterpret_cast(lParam)->code == PSN_SETACTIVE ) { + CRect rc; + VAPI( ::GetClientRect( *GetParent( ), &rc ) ); + UpdateWindowSize( rc.Width(), rc.Height(), *GetParent( ) ); + } + } + break; + case WM_DESTROY: + { + // + // Our opportunty for cleanup. + // Simply acquire all of our objects, free the appropriate memory and remove the + // properties from the window. If we do not remove the properties then they will constitute + // a resource leak. + DialogData *pdd = (DialogData*)dd; + if( pdd ) { + RegistryData rd; + rd.m_wpl.length = sizeof( rd.m_wpl ); + VAPI( GetWindowPlacement( &rd.m_wpl ) ); + + if( pdd->hkRootSave && pdd->pcszName ) { + (void)RegSetValueExRecursive( pdd->hkRootSave, pdd->pcszName, + NULL, REG_BINARY, + reinterpret_cast( &rd ), + sizeof( rd ) ); + } + + if( pdd->psd ) { + free(pdd->psd); + } + free(pdd); + } + + } + break; + } + return CDialog::OnWndMsg(msg, wParam, lParam, res); +} diff --git a/src/win32/RomInfo.cpp b/src/win32/RomInfo.cpp index 9892038d..1569d72f 100644 --- a/src/win32/RomInfo.cpp +++ b/src/win32/RomInfo.cpp @@ -1,6 +1,6 @@ // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. // Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004-2005 Forgotten and the VBA development team +// Copyright (C) 2004-2006 Forgotten and the VBA development team // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by @@ -356,6 +356,15 @@ BOOL RomInfoGB::OnInitDialog() case 0x06: type = "ROM+MBC2+BATT"; break; + case 0x0b: + type = "ROM+MMM01"; + break; + case 0x0c: + type = "ROM+MMM01+RAM"; + break; + case 0x0d: + type = "ROM+MMM01+RAM+BATT"; + break; case 0x0f: type = "ROM+MBC3+TIMER+BATT"; break; @@ -392,6 +401,18 @@ BOOL RomInfoGB::OnInitDialog() case 0x22: type = "ROM+MBC7+BATT"; break; + case 0x55: + type = "GameGenie"; + break; + case 0x56: + type = "GameShark V3.0"; + break; + case 0xfc: + type = "ROM+POCKET CAMERA"; + break; + case 0xfd: + type = "ROM+BANDAI TAMA5"; + break; case 0xfe: type = "ROM+HuC-3"; break; @@ -533,51 +554,44 @@ void RomInfoGBA::OnOk() BOOL RomInfoGBA::OnInitDialog() { CDialog::OnInitDialog(); - - char buffer[32]; - // Game Title + char buffer[13]; + strncpy(buffer, (const char *)&rom[0xa0], 12); buffer[12] = 0; GetDlgItem(IDC_ROM_TITLE)->SetWindowText(buffer); - // Game Code strncpy(buffer, (const char *)&rom[0xac], 4); buffer[4] = 0; GetDlgItem(IDC_ROM_GAME_CODE)->SetWindowText(buffer); - // Maker Code strncpy(buffer, (const char *)&rom[0xb0],2); buffer[2] = 0; GetDlgItem(IDC_ROM_MAKER_CODE)->SetWindowText(buffer); - // Maker Code -> Maker Name GetDlgItem(IDC_ROM_MAKER_NAME)->SetWindowText(winGBARomInfoFindMakerCode(buffer)); - // Main Unit Code sprintf(buffer, "%02x", rom[0xb3]); GetDlgItem(IDC_ROM_UNIT_CODE)->SetWindowText(buffer); - // Device Type sprintf(buffer, "%02x", rom[0xb4]); + if( rom[0xb4] & 0x80 ) { + strcat(buffer, " (DACS)"); + } GetDlgItem(IDC_ROM_DEVICE_TYPE)->SetWindowText(buffer); - // ROM Version Number sprintf(buffer, "%02x", rom[0xbc]); GetDlgItem(IDC_ROM_VERSION)->SetWindowText(buffer); - // Complement Check - int x, crc = 0x19; - for(int i = 0xa0; i < 0xbd; i++) - { - x = rom[i]; - crc += x; + u8 crc = 0x19; + for(int i = 0xa0; i < 0xbd; i++) { + crc += rom[i]; } - crc = (-crc) & 0xFF; - char same = (rom[0xbd] == crc) ? '=' : '/'; // Compare - sprintf(buffer, "ROM: %02X %c EMU: %02X", rom[0xbd], same, crc); - GetDlgItem(IDC_ROM_CRC)->SetWindowText(buffer); + crc = (-crc) & 255; + + sprintf(buffer, "%02x (%02x)", crc, rom[0xbd]); + GetDlgItem(IDC_ROM_CRC)->SetWindowText(buffer); CenterWindow(); return TRUE; // return TRUE unless you set the focus to a control diff --git a/src/win32/TileView.cpp b/src/win32/TileView.cpp index 24b3a4c7..40cd0f64 100644 --- a/src/win32/TileView.cpp +++ b/src/win32/TileView.cpp @@ -265,38 +265,41 @@ void TileView::savePNG(const char *name) void TileView::OnSave() { - CString captureBuffer; + if(rom != NULL) + { + CString captureBuffer; - if(theApp.captureFormat == 0) - captureBuffer = "tiles.png"; - else - captureBuffer = "tiles.bmp"; + if(theApp.captureFormat == 0) + captureBuffer = "tiles.png"; + else + captureBuffer = "tiles.bmp"; - LPCTSTR exts[] = {".png", ".bmp" }; + LPCTSTR exts[] = {".png", ".bmp" }; - CString filter = theApp.winLoadFilter(IDS_FILTER_PNG); - CString title = winResLoadString(IDS_SELECT_CAPTURE_NAME); + CString filter = theApp.winLoadFilter(IDS_FILTER_PNG); + CString title = winResLoadString(IDS_SELECT_CAPTURE_NAME); - FileDlg dlg(this, - captureBuffer, - filter, - theApp.captureFormat ? 2 : 1, - theApp.captureFormat ? "BMP" : "PNG", - exts, - "", - title, - true); + FileDlg dlg(this, + captureBuffer, + filter, + theApp.captureFormat ? 2 : 1, + theApp.captureFormat ? "BMP" : "PNG", + exts, + "", + title, + true); - if(dlg.DoModal() == IDCANCEL) { - return; + if(dlg.DoModal() == IDCANCEL) { + return; + } + + captureBuffer = dlg.GetPathName(); + + if(dlg.getFilterIndex() == 2) + saveBMP(captureBuffer); + else + savePNG(captureBuffer); } - - captureBuffer = dlg.GetPathName(); - - if(dlg.getFilterIndex() == 2) - saveBMP(captureBuffer); - else - savePNG(captureBuffer); } void TileView::renderTile256(int tile, int x, int y, u8 *charBase, u16 *palette) @@ -526,8 +529,8 @@ LRESULT TileView::OnMapInfo(WPARAM wParam, LPARAM lParam) u8 *colors = (u8 *)lParam; zoom.setColors(colors); - int x = (wParam & 0xFFFF)/8; - int y = ((wParam >> 16) & 0xFFFF)/8; + int x = (int)((wParam & 0xFFFF) / 8); + int y = (int)(((wParam >> 16) & 0xFFFF) / 8); u32 address = 0x6000000 + 0x4000 * charBase; int tile = 32 * y + x; diff --git a/src/win32/VBA.cpp b/src/win32/VBA.cpp index 682d3f96..3dcf1381 100644 --- a/src/win32/VBA.cpp +++ b/src/win32/VBA.cpp @@ -1,2313 +1,2518 @@ -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -// VBA.cpp : Defines the class behaviors for the application. -// -#include "stdafx.h" -#include - -#include "AVIWrite.h" -#include "LangSelect.h" -#include "MainWnd.h" -#include "Reg.h" -#include "..\..\res\resource.h" -#include "WavWriter.h" -#include "WinResUtil.h" - -#include "../System.h" -#include "../agbprint.h" -#include "../cheatSearch.h" -#include "../GBA.h" -#include "../Globals.h" -#include "../RTC.h" -#include "../Sound.h" -#include "../Util.h" -#include "../gb/gbGlobals.h" -#include "../gb/gbPrinter.h" - -/* Link ----------------------*/ -#include "../Link.h" -/* ---------------- */ - -#include "../gbafilter.h" - -#ifdef SDL -#pragma comment( lib, "SDL" ) -#pragma comment( lib, "SDLmain" ) -#endif - -extern void Pixelate(u8*,u32,u8*,u8*,u32,int,int); -extern void Pixelate32(u8*,u32,u8*,u8*,u32,int,int); -extern void MotionBlur(u8*,u32,u8*,u8*,u32,int,int); -extern void MotionBlur32(u8*,u32,u8*,u8*,u32,int,int); -extern void _2xSaI(u8*,u32,u8*,u8*,u32,int,int); -extern void _2xSaI32(u8*,u32,u8*,u8*,u32,int,int); -extern void Super2xSaI(u8*,u32,u8*,u8*,u32,int,int); -extern void Super2xSaI32(u8*,u32,u8*,u8*,u32,int,int); -extern void SuperEagle(u8*,u32,u8*,u8*,u32,int,int); -extern void SuperEagle32(u8*,u32,u8*,u8*,u32,int,int); -extern void AdMame2x(u8*,u32,u8*,u8*,u32,int,int); -extern void AdMame2x32(u8*,u32,u8*,u8*,u32,int,int); -extern void Bilinear(u8*,u32,u8*,u8*,u32,int,int); -extern void Bilinear32(u8*,u32,u8*,u8*,u32,int,int); -extern void BilinearPlus(u8*,u32,u8*,u8*,u32,int,int); -extern void BilinearPlus32(u8*,u32,u8*,u8*,u32,int,int); -extern void Scanlines(u8*,u32,u8*,u8*,u32,int,int); -extern void Scanlines32(u8*,u32,u8*,u8*,u32,int,int); -extern void ScanlinesTV(u8*,u32,u8*,u8*,u32,int,int); -extern void ScanlinesTV32(u8*,u32,u8*,u8*,u32,int,int); -extern void hq2x(u8*,u32,u8*,u8*,u32,int,int); -extern void hq2x32(u8*,u32,u8*,u8*,u32,int,int); -extern void lq2x(u8*,u32,u8*,u8*,u32,int,int); -extern void lq2x32(u8*,u32,u8*,u8*,u32,int,int); -extern void Simple2x16(u8*,u32,u8*,u8*,u32,int,int); -extern void Simple2x32(u8*,u32,u8*,u8*,u32,int,int); -extern void Simple3x16(u8*,u32,u8*,u8*,u32,int,int); -extern void Simple3x32(u8*,u32,u8*,u8*,u32,int,int); -extern void Simple4x16(u8*,u32,u8*,u8*,u32,int,int); -extern void Simple4x32(u8*,u32,u8*,u8*,u32,int,int); - -extern void hq3x32(u8*,u32,u8*,u8*,u32,int,int); -extern void hq3x16(u8*,u32,u8*,u8*,u32,int,int); -extern void hq4x32(u8*,u32,u8*,u8*,u32,int,int); -extern void hq4x16(u8*,u32,u8*,u8*,u32,int,int); - -extern void SmartIB(u8*,u32,int,int); -extern void SmartIB32(u8*,u32,int,int); -extern void MotionBlurIB(u8*,u32,int,int); -extern void InterlaceIB(u8*,u32,int,int); -extern void MotionBlurIB32(u8*,u32,int,int); - -extern void toolsLog(const char *); - -extern IDisplay *newGDIDisplay(); -extern IDisplay *newDirectDrawDisplay(); -extern IDisplay *newDirect3DDisplay(); -extern IDisplay *newOpenGLDisplay(); - -extern Input *newDirectInput(); - -extern ISound *newDirectSound(); - -extern void remoteStubSignal(int, int); -extern void remoteOutput(char *, u32); -extern void remoteStubMain(); -extern void remoteSetProtocol(int); -extern void remoteCleanUp(); -extern int remoteSocket; - -extern void InterframeCleanup(); - -void winlog(const char *msg, ...); - -/* Link ----------------------*/ -extern int InitLink(void); -extern void CloseLink(void); -//extern int linkid; -extern char inifile[]; -extern FILE *linklogfile; -/* ------------------- */ -#ifdef _DEBUG -#define new DEBUG_NEW -#undef THIS_FILE -static char THIS_FILE[] = __FILE__; -#endif - -int emulating = 0; -bool debugger = false; -int RGB_LOW_BITS_MASK = 0; -bool b16to32Video = false; -int systemFrameSkip = 0; -int systemSpeed = 0; -bool systemSoundOn = false; -u32 systemColorMap32[0x10000]; -u16 systemColorMap16[0x10000]; -u16 systemGbPalette[24]; -int systemRedShift = 0; -int systemBlueShift = 0; -int systemGreenShift = 0; -int systemColorDepth = 16; -int systemVerbose = 0; -int systemDebug = 0; -int systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; -bool soundBufferLow = 0; -void winSignal(int,int); -void winOutput(char *, u32); - -void (*dbgSignal)(int,int) = winSignal; -void (*dbgOutput)(char *, u32) = winOutput; - -#ifdef MMX -extern "C" bool cpu_mmx; -#endif - -void directXMessage(const char *msg) -{ - systemMessage(IDS_DIRECTX_7_REQUIRED, - "DirectX 7.0 or greater is required to run.\nDownload at http://www.microsoft.com/directx.\n\nError found at: %s", - msg); -} - -///////////////////////////////////////////////////////////////////////////// -// VBA - -BEGIN_MESSAGE_MAP(VBA, CWinApp) - //{{AFX_MSG_MAP(VBA) - // NOTE - the ClassWizard will add and remove mapping macros here. - // DO NOT EDIT what you see in these blocks of generated code! - //}}AFX_MSG_MAP - END_MESSAGE_MAP() - - ///////////////////////////////////////////////////////////////////////////// -// VBA construction - -VBA::VBA() -{ - mode320Available = false; - mode640Available = false; - mode800Available = false; - mode1024Available = false; - mode1280Available = false; - windowPositionX = 0; - windowPositionY = 0; - filterFunction = NULL; - ifbFunction = NULL; - ifbType = 0; - filterType = FILTER_NONE; - filterWidth = 0; - filterHeight = 0; - fsWidth = 0; - fsHeight = 0; - fsColorDepth = 0; - fsForceChange = false; - surfaceSizeX = 0; - surfaceSizeY = 0; - sizeX = 0; - sizeY = 0; - videoOption = VIDEO_3X; - fullScreenStretch = false; - disableStatusMessage = false; - showSpeed = 0; - showSpeedTransparent = true; - showRenderedFrames = 0; - screenMessage = false; - screenMessageTime = 0; - menuToggle = true; - display = NULL; - menu = NULL; - popup = NULL; - cartridgeType = 0; - soundInitialized = false; - useBiosFile = false; - skipBiosFile = false; - active = true; - paused = false; - recentFreeze = false; - autoSaveLoadCheatList = false; - winout = NULL; - removeIntros = false; - autoIPS = false; - winGbBorderOn = 0; - winFlashSize = 0x20000; - winRtcEnable = false; - winSaveType = 0; - rewindMemory = NULL; - rewindPos = 0; - rewindTopPos = 0; - rewindCounter = 0; - rewindCount = 0; - rewindSaveNeeded = false; - rewindTimer = 0; - captureFormat = 0; - tripleBuffering = true; - autoHideMenu = false; - throttle = 0; - throttleLastTime = 0; - autoFrameSkipLastTime = 0; - autoFrameSkip = false; - vsync = false; - changingVideoSize = false; - pVideoDriverGUID = NULL; - renderMethod = DIRECT_DRAW; - iconic = false; - ddrawEmulationOnly = false; - ddrawUsingEmulationOnly = false; - ddrawDebug = false; - ddrawUseVideoMemory = false; - d3dFilter = 0; - glFilter = 0; - glType = 0; - regEnabled = false; - pauseWhenInactive = true; - speedupToggle = false; - useOldSync = false; - winGbPrinterEnabled = false; - threadPriority = 2; - disableMMX = false; - languageOption = 0; - languageModule = NULL; - languageName = ""; - renderedFrames = 0; - input = NULL; - joypadDefault = 0; - autoFire = 0; - autoFireToggle = false; - winPauseNextFrame = false; - soundRecording = false; - soundRecorder = NULL; - sound = NULL; - aviRecording = false; - aviRecorder = NULL; - aviFrameNumber = 0; - painting = false; - movieRecording = false; - moviePlaying = false; - movieFrame = 0; - moviePlayFrame = 0; - movieFile = NULL; - movieLastJoypad = 0; - movieNextJoypad = 0; - sensorX = 2047; - sensorY = 2047; - mouseCounter = 0; - wasPaused = false; - frameskipadjust = 0; - autoLoadMostRecent = false; - fsMaxScale = 0; - romSize = 0; - - updateCount = 0; - - systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; - - ZeroMemory(&emulator, sizeof(emulator)); - - hAccel = NULL; - - for(int i = 0; i < 24;) { - systemGbPalette[i++] = (0x1f) | (0x1f << 5) | (0x1f << 10); - systemGbPalette[i++] = (0x15) | (0x15 << 5) | (0x15 << 10); - systemGbPalette[i++] = (0x0c) | (0x0c << 5) | (0x0c << 10); - systemGbPalette[i++] = 0; - } -} - -VBA::~VBA() -{ - InterframeCleanup(); - - saveSettings(); - - if(moviePlaying) { - if(movieFile != NULL) { - fclose(movieFile); - movieFile = NULL; - } - moviePlaying = false; - movieLastJoypad = 0; - } - - if(movieRecording) { - if(movieFile != NULL) { - // record the last joypad change so that the correct time can be - // recorded - fwrite(&movieFrame, 1, sizeof(int), movieFile); - fwrite(&movieLastJoypad, 1, sizeof(u32), movieFile); - fclose(movieFile); - movieFile = NULL; - } - movieRecording = false; - moviePlaying = false; - movieLastJoypad = 0; - } - - if(aviRecorder) { - delete aviRecorder; - aviRecording = false; - } - - if(soundRecorder) { - delete soundRecorder; - soundRecorder = NULL; - } - soundRecording = false; - soundPause(); - soundShutdown(); - - if(gbRom != NULL || rom != NULL) { - if(autoSaveLoadCheatList) - ((MainWnd *)m_pMainWnd)->winSaveCheatListDefault(); - ((MainWnd *)m_pMainWnd)->writeBatteryFile(); - cheatSearchCleanup(&cheatSearchData); - emulator.emuCleanUp(); - } - - if(input) - delete input; - - shutdownDisplay(); - - if(rewindMemory) - free(rewindMemory); -} - -///////////////////////////////////////////////////////////////////////////// -// The one and only VBA object - -VBA theApp; -//#include -///////////////////////////////////////////////////////////////////////////// -// VBA initialization - -// code from SDL_main.c for Windows -/* Parse a command line buffer into arguments */ -static int parseCommandLine(char *cmdline, char **argv) -{ - char *bufp; - int argc; - - argc = 0; - for ( bufp = cmdline; *bufp; ) { - /* Skip leading whitespace */ - while ( isspace(*bufp) ) { - ++bufp; - } - /* Skip over argument */ - if ( *bufp == '"' ) { - ++bufp; - if ( *bufp ) { - if ( argv ) { - argv[argc] = bufp; - } - ++argc; - } - /* Skip over word */ - while ( *bufp && (*bufp != '"') ) { - ++bufp; - } - } else { - if ( *bufp ) { - if ( argv ) { - argv[argc] = bufp; - } - ++argc; - } - /* Skip over word */ - while ( *bufp && ! isspace(*bufp) ) { - ++bufp; - } - } - if ( *bufp ) { - if ( argv ) { - *bufp = '\0'; - } - ++bufp; - } - } - if ( argv ) { - argv[argc] = NULL; - } - return(argc); -} - -BOOL VBA::InitInstance() -{ - AfxEnableControlContainer(); - // Standard initialization - // If you are not using these features and wish to reduce the size - // of your final executable, you should remove from the following - // the specific initialization routines you do not need. - - SetRegistryKey(_T("VBA")); - - remoteSetProtocol(0); - - systemVerbose = GetPrivateProfileInt("config", - "verbose", - 0, - "VBA.ini"); - - systemDebug = GetPrivateProfileInt("config", - "debug", - 0, - "VBA.ini"); - ddrawDebug = GetPrivateProfileInt("config", - "ddrawDebug", - 0, - "VBA.ini") ? true : false; - - wndClass = AfxRegisterWndClass(0, LoadCursor(IDC_ARROW), (HBRUSH)GetStockObject(BLACK_BRUSH), LoadIcon(IDI_ICON)); - - char winBuffer[2048]; - - GetModuleFileName(NULL, winBuffer, 2048); - char *p = strrchr(winBuffer, '\\'); - if(p) - *p = 0; - - if(!InitLink()) - return FALSE;; - - regInit(winBuffer); - - loadSettings(); - - if(!openLinkLog()) - return FALSE; - - if(!initInput()) - return FALSE; - - if(!initDisplay()) { - if(videoOption >= VIDEO_320x240) { - regSetDwordValue("video", VIDEO_1X); - if(pVideoDriverGUID) - regSetDwordValue("defaultVideoDriver", TRUE); - } - return FALSE; - } - - hAccel = LoadAccelerators(AfxGetInstanceHandle(), MAKEINTRESOURCE(IDR_ACCELERATOR)); - - winAccelMgr.Connect((MainWnd *)m_pMainWnd); - - winAccelMgr.SetRegKey(HKEY_CURRENT_USER, "Software\\Emulators\\VisualBoyAdvance"); - - extern void winAccelAddCommands(CAcceleratorManager&); - - winAccelAddCommands(winAccelMgr); - - winAccelMgr.CreateDefaultTable(); - - winAccelMgr.Load(); - - winAccelMgr.UpdateWndTable(); - - winAccelMgr.UpdateMenu(menu); - - if (m_lpCmdLine[0]) - { - int argc = parseCommandLine(m_lpCmdLine, NULL); - char **argv = (char **)malloc((argc+1)*sizeof(char *)); - parseCommandLine(m_lpCmdLine, argv); - if(argc > 0) { - szFile = argv[0]; - filename = szFile; - } - int index = filename.ReverseFind('.'); - - if(index != -1) - filename = filename.Left(index); - - if(((MainWnd*)m_pMainWnd)->FileRun()) - emulating = true; - else - emulating = false; - free(argv); - } - - return TRUE; -} - -void VBA::adjustDestRect() -{ - POINT point; - point.x = 0; - point.y = 0; - - m_pMainWnd->ClientToScreen(&point); - dest.top = point.y; - dest.left = point.x; - - point.x = surfaceSizeX; - point.y = surfaceSizeY; - - m_pMainWnd->ClientToScreen(&point); - dest.bottom = point.y; - dest.right = point.x; - - // make sure that dest rect lies in the monitor - if(videoOption >= VIDEO_320x240) { - dest.top -= windowPositionY; - dest.left -= windowPositionX; - dest.bottom-= windowPositionY; - dest.right -= windowPositionX; - } - - int menuSkip = 0; - - if(videoOption >= VIDEO_320x240 && menuToggle) { - int m = GetSystemMetrics(SM_CYMENU); - menuSkip = m; - dest.bottom -=m; - } - - if(videoOption > VIDEO_4X) { - int top = (fsHeight - surfaceSizeY) / 2; - int left = (fsWidth - surfaceSizeX) / 2; - dest.top += top; - dest.bottom += top; - dest.left += left; - dest.right += left; - if(fullScreenStretch) { - dest.top = 0+menuSkip; - dest.left = 0; - dest.right = fsWidth; - dest.bottom = fsHeight; - } - } -} - -void VBA::updateIFB() -{ - if(systemColorDepth == 16) { - switch(ifbType) { - case 0: - default: - ifbFunction = NULL; - break; - case 1: - ifbFunction = MotionBlurIB; - break; - case 2: - ifbFunction = SmartIB; - break; - } - } else if(systemColorDepth == 32) { - switch(ifbType) { - case 0: - default: - ifbFunction = NULL; - break; - case 1: - ifbFunction = MotionBlurIB32; - break; - case 2: - ifbFunction = SmartIB32; - break; - } - } else - ifbFunction = NULL; -} - -void VBA::updateFilter() -{ - // BEGIN hacky ugly code - - // HQ3X asm wants 16 bit input. When we switch - // away from 16 bits we need to restore the driver values - - if ( b16to32Video ) - { - b16to32Video = false; - systemColorDepth = 32; - systemRedShift = 19; - systemGreenShift = 11; - systemBlueShift = 3; - utilUpdateSystemColorMaps(theApp.filterLCD); - } - // END hacky ugly code - - filterWidth = sizeX; - filterHeight = sizeY; - filterMagnification = 1; - - - if ( videoOption == VIDEO_1X || videoOption == VIDEO_320x240 ) - { - filterFunction = NULL; - filterMagnification = 1; - } - else - { - if ( systemColorDepth == 16 ) - { - switch(filterType) - { - default: - case FILTER_NONE: - filterFunction = NULL; - filterMagnification = 1; - break; - case FILTER_TVMODE: - filterFunction = ScanlinesTV; - filterMagnification = 2; - break; - case FILTER_2XSAI: - filterFunction = _2xSaI; - filterMagnification = 2; - break; - case FILTER_SUPER2XSAI: - filterFunction = Super2xSaI; - filterMagnification = 2; - break; - case FILTER_SUPEREAGLE: - filterFunction = SuperEagle; - filterMagnification = 2; - break; - case FILTER_PIXELATE: - filterFunction = Pixelate; - filterMagnification = 2; - break; - case FILTER_MAMESCALE2X: - filterFunction = AdMame2x; - filterMagnification = 2; - break; - case FILTER_SIMPLE2X: - filterFunction = Simple2x16; - filterMagnification = 2; - break; - case FILTER_BILINEAR: - filterFunction = Bilinear; - filterMagnification = 2; - break; - case FILTER_BILINEARPLUS: - filterFunction = BilinearPlus; - filterMagnification = 2; - break; - case FILTER_SCANLINES: - filterFunction = Scanlines; - filterMagnification = 2; - break; - case FILTER_HQ2X: - filterFunction = hq2x; - filterMagnification = 2; - break; - case FILTER_LQ2X: - filterFunction = lq2x; - filterMagnification = 2; - break; - case FILTER_SIMPLE3X: - filterFunction = Simple3x16; - filterMagnification = 3; - break; - case FILTER_SIMPLE4X: - filterFunction = Simple4x16; - filterMagnification = 4; - break; - case FILTER_HQ3X: - filterFunction = hq3x16; - filterMagnification = 3; - break; - case FILTER_HQ4X: - filterFunction = hq4x16; - filterMagnification = 4; - break; - } - } - - if ( systemColorDepth == 32 ) - { - switch(filterType) - { - default: - case FILTER_NONE: - filterFunction = NULL; - filterMagnification = 1; - break; - case FILTER_TVMODE: - filterFunction = ScanlinesTV32; - filterMagnification = 2; - break; - case FILTER_2XSAI: - filterFunction = _2xSaI32; - filterMagnification = 2; - break; - case FILTER_SUPER2XSAI: - filterFunction = Super2xSaI32; - filterMagnification = 2; - break; - case FILTER_SUPEREAGLE: - filterFunction = SuperEagle32; - filterMagnification = 2; - break; - case FILTER_PIXELATE: - filterFunction = Pixelate32; - filterMagnification = 2; - break; - case FILTER_MAMESCALE2X: - filterFunction = AdMame2x32; - filterMagnification = 2; - break; - case FILTER_SIMPLE2X: - filterFunction = Simple2x32; - filterMagnification = 2; - break; - case FILTER_BILINEAR: - filterFunction = Bilinear32; - filterMagnification = 2; - break; - case FILTER_BILINEARPLUS: - filterFunction = BilinearPlus32; - filterMagnification = 2; - break; - case FILTER_SCANLINES: - filterFunction = Scanlines32; - filterMagnification = 2; - break; - case FILTER_HQ2X: - filterFunction = hq2x32; - filterMagnification = 2; - break; - case FILTER_LQ2X: - filterFunction = lq2x32; - filterMagnification = 2; - break; - case FILTER_SIMPLE3X: - filterFunction = Simple3x32; - filterMagnification = 3; - break; - case FILTER_SIMPLE4X: - filterFunction = Simple4x32; - filterMagnification = 4; - break; - case FILTER_HQ3X: - filterFunction = hq3x32; - filterMagnification = 3; - b16to32Video=true; - break; - case FILTER_HQ4X: - filterFunction = hq4x32; - filterMagnification = 4; - b16to32Video=true; - break; - } - } - } - - rect.right = sizeX * filterMagnification; - rect.bottom = sizeY * filterMagnification; - - if( filterType != FILTER_NONE ) - memset(delta, 0xFF, sizeof(delta)); - - if( display ) - display->changeRenderSize(rect.right, rect.bottom); - - if (b16to32Video) - { - systemColorDepth = 16; - systemRedShift = 11; - systemGreenShift = 6; - systemBlueShift = 0; - utilUpdateSystemColorMaps(theApp.filterLCD); - } -} - - -void VBA::updateMenuBar() -{ - if(menu != NULL) { - if(m_pMainWnd) - m_pMainWnd->SetMenu(NULL); - m_menu.Detach(); - DestroyMenu(menu); - } - - if(popup != NULL) { - // force popup recreation if language changed - DestroyMenu(popup); - popup = NULL; - } - - m_menu.Attach(winResLoadMenu(MAKEINTRESOURCE(IDR_MENU))); - menu = (HMENU)m_menu; - - if(m_pMainWnd) - m_pMainWnd->SetMenu(&m_menu); - - if (display) - display->clear(); -} - -void winlog(const char *msg, ...) -{ - CString buffer; - va_list valist; - - va_start(valist, msg); - buffer.FormatV(msg, valist); - - if(theApp.winout == NULL) { - theApp.winout = fopen("vba-trace.log","w"); - } - - fputs(buffer, theApp.winout); - - va_end(valist); -} - -void log(const char *msg, ...) -{ - CString buffer; - va_list valist; - - va_start(valist, msg); - buffer.FormatV(msg, valist); - - toolsLog(buffer); - - va_end(valist); -} - -bool systemReadJoypads() -{ - if(theApp.input) - return theApp.input->readDevices(); - return false; -} - -u32 systemReadJoypad(int which) -{ - if(theApp.input) - return theApp.input->readDevice(which); - return 0; -} - -void systemDrawScreen() -{ - if(theApp.display == NULL) - return; - - theApp.renderedFrames++; - - if(theApp.updateCount) { - POSITION pos = theApp.updateList.GetHeadPosition(); - while(pos) { - IUpdateListener *up = theApp.updateList.GetNext(pos); - up->update(); - } - } - - if(theApp.aviRecording && !theApp.painting) { - int width = 240; - int height = 160; - switch(theApp.cartridgeType) { - case 0: - width = 240; - height = 160; - break; - case 1: - if(gbBorderOn) { - width = 256; - height = 224; - } else { - width = 160; - height = 144; - } - break; - } - - if(theApp.aviRecorder == NULL) { - theApp.aviRecorder = new AVIWrite(); - theApp.aviFrameNumber = 0; - - theApp.aviRecorder->SetFPS(60); - - BITMAPINFOHEADER bi; - memset(&bi, 0, sizeof(bi)); - bi.biSize = 0x28; - bi.biPlanes = 1; - bi.biBitCount = 24; - bi.biWidth = width; - bi.biHeight = height; - bi.biSizeImage = 3*width*height; - theApp.aviRecorder->SetVideoFormat(&bi); - theApp.aviRecorder->Open(theApp.aviRecordName); - } - - char *bmp = new char[width*height*3]; - - utilWriteBMP(bmp, width, height, pix); - theApp.aviRecorder->AddFrame(theApp.aviFrameNumber, bmp); - - delete bmp; - } - - if(theApp.ifbFunction) { - if(systemColorDepth == 16) - theApp.ifbFunction(pix+theApp.filterWidth*2+4, theApp.filterWidth*2+4, - theApp.filterWidth, theApp.filterHeight); - else - theApp.ifbFunction(pix+theApp.filterWidth*4+4, theApp.filterWidth*4+4, - theApp.filterWidth, theApp.filterHeight); - } - - if(!soundBufferLow) - theApp.display->render(); - else - soundBufferLow = false; -} - -void systemScreenCapture(int captureNumber) -{ - if(theApp.m_pMainWnd) - ((MainWnd *)theApp.m_pMainWnd)->screenCapture(captureNumber); -} - -u32 systemGetClock() -{ - return GetTickCount(); -} - -void systemMessage(int number, const char *defaultMsg, ...) -{ - CString buffer; - va_list valist; - CString msg = defaultMsg; - if(number) - msg = winResLoadString(number); - - va_start(valist, defaultMsg); - buffer.FormatV(msg, valist); - - theApp.winCheckFullscreen(); - - AfxGetApp()->m_pMainWnd->MessageBox(buffer, winResLoadString(IDS_ERROR), MB_OK|MB_ICONERROR); - - va_end(valist); -} - -void systemSetTitle(const char *title) -{ - if(theApp.m_pMainWnd != NULL) { - AfxGetApp()->m_pMainWnd->SetWindowText(title); - } -} - -void systemShowSpeed(int speed) -{ - systemSpeed = speed; - theApp.showRenderedFrames = theApp.renderedFrames; - theApp.renderedFrames = 0; - if(theApp.videoOption <= VIDEO_4X && theApp.showSpeed) { - CString buffer; - if(theApp.showSpeed == 1) - buffer.Format("VisualBoyAdvance-%3d%%", systemSpeed); - else - buffer.Format("VisualBoyAdvance-%3d%%(%d, %d fps)", systemSpeed, - systemFrameSkip, - theApp.showRenderedFrames); - - systemSetTitle(buffer); - } -} - -void systemFrame() -{ - if(theApp.aviRecording) - theApp.aviFrameNumber++; - if(theApp.movieRecording || theApp.moviePlaying) - theApp.movieFrame++; -} - -void system10Frames(int rate) -{ - u32 time = systemGetClock(); - if(!theApp.wasPaused && theApp.autoFrameSkip && !theApp.throttle) { - u32 diff = time - theApp.autoFrameSkipLastTime; - int speed = 100; - - if(diff) - speed = (1000000/rate)/diff; - - if(speed >= 98) { - theApp.frameskipadjust++; - - if(theApp.frameskipadjust >= 3) { - theApp.frameskipadjust=0; - if(systemFrameSkip > 0) - systemFrameSkip--; - } - } else { - if(speed < 80) - theApp.frameskipadjust -= (90 - speed)/5; - else if(systemFrameSkip < 9) - theApp.frameskipadjust--; - - if(theApp.frameskipadjust <= -2) { - theApp.frameskipadjust += 2; - if(systemFrameSkip < 9) - systemFrameSkip++; - } - } - } - if(!theApp.wasPaused && theApp.throttle) { - if(!speedup) { - u32 diff = time - theApp.throttleLastTime; - - int target = (1000000/(rate*theApp.throttle)); - int d = (target - diff); - - if(d > 0) { - Sleep(d); - } - } - theApp.throttleLastTime = systemGetClock(); - } - if(theApp.rewindMemory) { - if(++theApp.rewindCounter >= (theApp.rewindTimer)) { - theApp.rewindSaveNeeded = true; - theApp.rewindCounter = 0; - } - } - if(systemSaveUpdateCounter) { - if(--systemSaveUpdateCounter <= SYSTEM_SAVE_NOT_UPDATED) { - ((MainWnd *)theApp.m_pMainWnd)->writeBatteryFile(); - systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; - } - } - - theApp.wasPaused = false; - theApp.autoFrameSkipLastTime = time; -} - -void systemScreenMessage(const char *msg) -{ - theApp.screenMessage = true; - theApp.screenMessageTime = GetTickCount(); - theApp.screenMessageBuffer = msg; - - if(theApp.screenMessageBuffer.GetLength() > 40) - theApp.screenMessageBuffer = theApp.screenMessageBuffer.Left(40); -} - -void systemUpdateMotionSensor() -{ - if(theApp.input) - theApp.input->checkMotionKeys(); -} - -int systemGetSensorX() -{ - return theApp.sensorX; -} - -int systemGetSensorY() -{ - return theApp.sensorY; -} - -bool systemSoundInit() -{ - if(theApp.sound) - delete theApp.sound; - - theApp.sound = newDirectSound(); - return theApp.sound->init(); -} - - -void systemSoundShutdown() -{ - if(theApp.sound) - delete theApp.sound; - theApp.sound = NULL; -} - -void systemSoundPause() -{ - if(theApp.sound) - theApp.sound->pause(); -} - -void systemSoundReset() -{ - if(theApp.sound) - theApp.sound->reset(); -} - -void systemSoundResume() -{ - if(theApp.sound) - theApp.sound->resume(); -} - -void systemWriteDataToSoundBuffer() -{ - if(theApp.sound) - theApp.sound->write(); -} - -bool systemCanChangeSoundQuality() -{ - return true; -} - -bool systemPauseOnFrame() -{ - if(theApp.winPauseNextFrame) { - theApp.paused = true; - theApp.winPauseNextFrame = false; - return true; - } - return false; -} - -void systemGbBorderOn() -{ - if(emulating && theApp.cartridgeType == 1 && gbBorderOn) { - theApp.updateWindowSize(theApp.videoOption); - } -} - -BOOL VBA::OnIdle(LONG lCount) -{ - if(emulating && debugger) { - MSG msg; - remoteStubMain(); - if(debugger) - return TRUE; // continue loop - return !::PeekMessage(&msg, NULL, NULL, NULL, PM_NOREMOVE); - } else if(emulating && active && !paused) { - for(int i = 0; i < 2; i++) { - emulator.emuMain(emulator.emuCount); - if(lanlink.connected&&linkid&&lc.numtransfers==0) lc.CheckConn(); - - if(rewindSaveNeeded && rewindMemory && emulator.emuWriteMemState) { - rewindCount++; - if(rewindCount > 8) - rewindCount = 8; - if(emulator.emuWriteMemState(&rewindMemory[rewindPos*REWIND_SIZE], - REWIND_SIZE)) { - rewindPos = ++rewindPos & 7; - if(rewindCount == 8) - rewindTopPos = ++rewindTopPos & 7; - } - } - - rewindSaveNeeded = false; - } - - if(mouseCounter) { - if(--mouseCounter == 0) { - SetCursor(NULL); - } - } - return TRUE; - } - return FALSE; - - // return CWinApp::OnIdle(lCount); -} - -void VBA::addRecentFile(CString file) -{ - // Do not change recent list if frozen - if(recentFreeze) - return; - int i = 0; - for(i = 0; i < 10; i++) { - if(recentFiles[i].GetLength() == 0) - break; - - if(recentFiles[i].Compare(file) == 0) { - if(i == 0) - return; - CString p = recentFiles[i]; - for(int j = i; j > 0; j--) { - recentFiles[j] = recentFiles[j-1]; - } - recentFiles[0] = p; - return; - } - } - int num = 0; - for(i = 0; i < 10; i++) { - if(recentFiles[i].GetLength() != 0) - num++; - } - if(num == 10) { - num--; - } - - for(i = num; i >= 1; i--) { - recentFiles[i] = recentFiles[i-1]; - } - recentFiles[0] = file; -} - -void VBA::loadSettings() -{ - CString buffer; - - languageOption = regQueryDwordValue("language", 1); - if(languageOption < 0 || languageOption > 2) - languageOption = 1; - - buffer = regQueryStringValue("languageName", ""); - if(!buffer.IsEmpty()) { - languageName = buffer.Left(3); - } else - languageName = ""; - - winSetLanguageOption(languageOption, true); - - frameSkip = regQueryDwordValue("frameSkip", 0); - if(frameSkip < 0 || frameSkip > 9) - frameSkip = 0; - - gbFrameSkip = regQueryDwordValue("gbFrameSkip", 0); - if(gbFrameSkip < 0 || gbFrameSkip > 9) - gbFrameSkip = 0; - - autoFrameSkip = regQueryDwordValue("autoFrameSkip", FALSE) ? TRUE : FALSE; - - vsync = regQueryDwordValue("vsync", false) ? true : false ; - synchronize = regQueryDwordValue("synchronize", TRUE) ? true : false; - fullScreenStretch = regQueryDwordValue("stretch", FALSE) ? true : false; - - videoOption = regQueryDwordValue("video", VIDEO_3X); - - if(videoOption < VIDEO_1X || videoOption > VIDEO_OTHER) - videoOption = VIDEO_3X; - - bool defaultVideoDriver = regQueryDwordValue("defaultVideoDriver", true) ? - true : false; - - if(!regQueryBinaryValue("videoDriverGUID", (char *)&videoDriverGUID, - sizeof(GUID))) { - defaultVideoDriver = TRUE; - } - - if(defaultVideoDriver) - pVideoDriverGUID = NULL; - else - pVideoDriverGUID = &videoDriverGUID; - - fsWidth = regQueryDwordValue("fsWidth", 0); - fsHeight = regQueryDwordValue("fsHeight", 0); - fsColorDepth = regQueryDwordValue("fsColorDepth", 0); - fsFrequency = regQueryDwordValue("fsFrequency", 60); - fsAdapter = regQueryDwordValue("fsAdapter", 0); - - if(videoOption == VIDEO_OTHER) - { - if(fsWidth < 0 || fsWidth > 4095 || fsHeight < 0 || fsHeight > 4095) - videoOption = 0; - if(fsColorDepth != 16 && fsColorDepth != 24 && fsColorDepth != 32) - videoOption = 0; - } - - renderMethod = (DISPLAY_TYPE)regQueryDwordValue("renderMethod", DIRECT_3D); - - if(renderMethod < GDI || renderMethod > OPENGL) - renderMethod = DIRECT_3D; - - windowPositionX = regQueryDwordValue("windowX", 0); - if(windowPositionX < 0) - windowPositionX = 0; - windowPositionY = regQueryDwordValue("windowY", 0); - if(windowPositionY < 0) - windowPositionY = 0; - - useBiosFile = regQueryDwordValue("useBios", 0) ? true: false; - - skipBiosFile = regQueryDwordValue("skipBios", 0) ? true : false; - - buffer = regQueryStringValue("biosFile", ""); - - if(!buffer.IsEmpty()) { - biosFileName = buffer; - } - - int res = regQueryDwordValue("soundEnable", 0x30f); - - soundEnable(res); - soundDisable(~res); - - soundOffFlag = (regQueryDwordValue("soundOff", 0)) ? true : false; - - soundQuality = regQueryDwordValue("soundQuality", 1); - - soundEcho = regQueryDwordValue("soundEcho", 0) ? true : false; - - soundLowPass = regQueryDwordValue("soundLowPass", 0) ? true : false; - - soundReverse = regQueryDwordValue("soundReverse", 0) ? true : false; - - soundVolume = regQueryDwordValue("soundVolume", 0); - if(soundVolume < 0 || soundVolume > 5) - soundVolume = 0; - - soundInterpolation = regQueryDwordValue("soundInterpolation", 0); - if(soundInterpolation < 0 || soundInterpolation > 4) - soundInterpolation = 0; - - ddrawEmulationOnly = regQueryDwordValue("ddrawEmulationOnly", false) ? true : false; - ddrawUseVideoMemory = regQueryDwordValue("ddrawUseVideoMemory", false) ? true : false; - tripleBuffering = regQueryDwordValue("tripleBuffering", true) ? true : false; - - d3dFilter = regQueryDwordValue("d3dFilter", 0); - if(d3dFilter < 0 || d3dFilter > 1) - d3dFilter = 0; - glFilter = regQueryDwordValue("glFilter", 0); - if(glFilter < 0 || glFilter > 1) - glFilter = 0; - glType = regQueryDwordValue("glType", 0); - if(glType < 0 || glType > 1) - glType = 0; - - filterType = regQueryDwordValue("filter", 0); - if(filterType < 0 || filterType > 16) - filterType = 0; - - disableMMX = regQueryDwordValue("disableMMX", 0) ? true: false; - - disableStatusMessage = regQueryDwordValue("disableStatus", 0) ? true : false; - - showSpeed = regQueryDwordValue("showSpeed", 0); - if(showSpeed < 0 || showSpeed > 2) - showSpeed = 0; - - showSpeedTransparent = regQueryDwordValue("showSpeedTransparent", TRUE) ? - TRUE : FALSE; - - winGbPrinterEnabled = regQueryDwordValue("gbPrinter", false) ? true : false; - - if(winGbPrinterEnabled) - gbSerialFunction = gbPrinterSend; - else - gbSerialFunction = NULL; - - pauseWhenInactive = regQueryDwordValue("pauseWhenInactive", 1) ? - true : false; - - useOldSync = regQueryDwordValue("useOldSync", 0) ? - TRUE : FALSE; - - captureFormat = regQueryDwordValue("captureFormat", 0); - - removeIntros = regQueryDwordValue("removeIntros", false) ? true : false; - - recentFreeze = regQueryDwordValue("recentFreeze", false) ? true : false; - - autoIPS = regQueryDwordValue("autoIPS", false) ? true : false; - - cpuDisableSfx = regQueryDwordValue("disableSfx", 0) ? true : false; - - winSaveType = regQueryDwordValue("saveType", 0); - if(winSaveType < 0 || winSaveType > 5) - winSaveType = 0; - - cpuEnhancedDetection = regQueryDwordValue("enhancedDetection", 1) ? true : - false; - - ifbType = regQueryDwordValue("ifbType", 0); - if(ifbType < 0 || ifbType > 2) - ifbType = 0; - - winFlashSize = regQueryDwordValue("flashSize", 0x20000); - if(winFlashSize != 0x10000 && winFlashSize != 0x20000) - winFlashSize = 0x20000; - - agbPrintEnable(regQueryDwordValue("agbPrint", 0) ? true : false); - - winRtcEnable = regQueryDwordValue("rtcEnabled", 1) ? true : false; - rtcEnable(winRtcEnable); - - autoHideMenu = regQueryDwordValue("autoHideMenu", 0) ? true : false; - - switch(videoOption) { - case VIDEO_320x240: - fsWidth = 320; - fsHeight = 240; - fsColorDepth = 16; - fsFrequency = 60; - break; - case VIDEO_640x480: - fsWidth = 640; - fsHeight = 480; - fsColorDepth = 16; - fsFrequency = 60; - break; - case VIDEO_800x600: - fsWidth = 800; - fsHeight = 600; - fsColorDepth = 16; - fsFrequency = 60; - break; - case VIDEO_1024x768: - fsWidth = 1024; - fsHeight = 768; - fsColorDepth = 16; - fsFrequency = 60; - break; - case VIDEO_1280x1024: - fsWidth = 1280; - fsHeight = 1024; - fsColorDepth = 16; - fsFrequency = 60; - break; - } - - winGbBorderOn = regQueryDwordValue("borderOn", 0); - gbBorderAutomatic = regQueryDwordValue("borderAutomatic", 0); - gbEmulatorType = regQueryDwordValue("emulatorType", 4); - if(gbEmulatorType < 0 || gbEmulatorType > 5) - gbEmulatorType = 4; - gbColorOption = regQueryDwordValue("colorOption", 0); - - threadPriority = regQueryDwordValue("priority", 2); - - if(threadPriority < 0 || threadPriority >3) - threadPriority = 2; - updatePriority(); - - autoSaveLoadCheatList = regQueryDwordValue("autoSaveCheatList", 0) ? - true : false; - - gbPaletteOption = regQueryDwordValue("gbPaletteOption", 0); - if(gbPaletteOption < 0) - gbPaletteOption = 0; - if(gbPaletteOption > 2) - gbPaletteOption = 2; - - regQueryBinaryValue("gbPalette", (char *)systemGbPalette, - 24*sizeof(u16)); - - rewindTimer = regQueryDwordValue("rewindTimer", 0); - - if(rewindTimer < 0 || rewindTimer > 600) - rewindTimer = 0; - - rewindTimer *= 6; // convert to 10 frames multiple - - if(rewindTimer != 0) - rewindMemory = (char *)malloc(8*REWIND_SIZE); - - for(int i = 0; i < 10; i++) { - buffer.Format("recent%d", i); - char *s = regQueryStringValue(buffer, NULL); - if(s == NULL) - break; - recentFiles[i] = s; - } - - joypadDefault = regQueryDwordValue("joypadDefault", 0); - if(joypadDefault < 0 || joypadDefault > 3) - joypadDefault = 0; - - autoLoadMostRecent = regQueryDwordValue("autoLoadMostRecent", false) ? true : - false; - - cheatsEnabled = regQueryDwordValue("cheatsEnabled", true) ? true : false; - - fsMaxScale = regQueryDwordValue("fsMaxScale", 0); - - throttle = regQueryDwordValue("throttle", 0); - if(throttle < 5 || throttle > 1000) - throttle = 0; - - linktimeout = regQueryDwordValue("LinkTimeout", 1000); - - linklog = regQueryDwordValue("Linklog", false) ? true : false; - if(linklog) - openLinkLog(); - - adapter = regQueryDwordValue("RFU", false) ? true : false; - - lanlink.active = regQueryDwordValue("LAN", 0) ? true : false; -} - -void VBA::updateFrameSkip() -{ - switch(cartridgeType) { - case 0: - systemFrameSkip = frameSkip; - break; - case 1: - systemFrameSkip = gbFrameSkip; - break; - } -} - -void VBA::updateVideoSize(UINT id) -{ - int value = 0; - - switch(id) { - case ID_OPTIONS_VIDEO_X1: - value = VIDEO_1X; - break; - case ID_OPTIONS_VIDEO_X2: - value = VIDEO_2X; - break; - case ID_OPTIONS_VIDEO_X3: - value = VIDEO_3X; - break; - case ID_OPTIONS_VIDEO_X4: - value = VIDEO_4X; - break; - case ID_OPTIONS_VIDEO_FULLSCREEN320X240: - value = VIDEO_320x240; - fsWidth = 320; - fsHeight = 240; - fsColorDepth = 16; - break; - case ID_OPTIONS_VIDEO_FULLSCREEN640X480: - value = VIDEO_640x480; - fsWidth = 640; - fsHeight = 480; - fsColorDepth = 16; - break; - case ID_OPTIONS_VIDEO_FULLSCREEN800X600: - value = VIDEO_800x600; - fsWidth = 800; - fsHeight = 600; - fsColorDepth = 16; - break; - case ID_OPTIONS_VIDEO_FULLSCREEN1024X768: - value = VIDEO_1024x768; - fsWidth = 1024; - fsHeight = 768; - fsColorDepth = 32; - break; - case ID_OPTIONS_VIDEO_FULLSCREEN1280X1024: - value = VIDEO_1280x1024; - fsWidth = 1280; - fsHeight = 1024; - fsColorDepth = 32; - break; - case ID_OPTIONS_VIDEO_FULLSCREEN: - value = VIDEO_OTHER; - break; - } - - if(videoOption == value && value != VIDEO_OTHER) - return; - - updateWindowSize(value); -} - -typedef BOOL (WINAPI *GETMENUBARINFO)(HWND, LONG, LONG, PMENUBARINFO); - -static void winCheckMenuBarInfo(int& winSizeX, int& winSizeY) -{ - HINSTANCE hinstDll; - DWORD dwVersion = 0; - - hinstDll = LoadLibrary("USER32.DLL"); - - if(hinstDll) { - GETMENUBARINFO func = (GETMENUBARINFO)GetProcAddress(hinstDll, - "GetMenuBarInfo"); - - if(func) { - MENUBARINFO info; - info.cbSize = sizeof(info); - - func(AfxGetMainWnd()->GetSafeHwnd(), OBJID_MENU, 0, &info); - - int menuHeight = GetSystemMetrics(SM_CYMENU); - - if((info.rcBar.bottom - info.rcBar.top) > menuHeight) { - winSizeY += (info.rcBar.bottom - info.rcBar.top) - menuHeight + 1; - theApp.m_pMainWnd->SetWindowPos( - 0, //HWND_TOPMOST, - theApp.windowPositionX, - theApp.windowPositionY, - winSizeX, - winSizeY, - SWP_NOMOVE | SWP_SHOWWINDOW); - } - } - FreeLibrary(hinstDll); - } -} - -void VBA::updateWindowSize(int value) -{ - regSetDwordValue("video", value); - - if(value == VIDEO_OTHER) { - regSetDwordValue("fsWidth", fsWidth); - regSetDwordValue("fsHeight", fsHeight); - regSetDwordValue("fsColorDepth", fsColorDepth); - } - - if(((value >= VIDEO_320x240) && - (videoOption != value)) || - (videoOption >= VIDEO_320x240 && - value <= VIDEO_4X) || - fsForceChange) { - fsForceChange = false; - changingVideoSize = true; - shutdownDisplay(); - if(input) { - delete input; - input = NULL; - } - m_pMainWnd->DragAcceptFiles(FALSE); - CWnd *pWnd = m_pMainWnd; - m_pMainWnd = NULL; - pWnd->DestroyWindow(); - delete pWnd; - videoOption = value; - if(!initDisplay()) { - if(videoOption == VIDEO_320x240 || - videoOption == VIDEO_640x480 || - videoOption == VIDEO_800x600 || - videoOption == VIDEO_1024x768 || - videoOption == VIDEO_1280x1024 || - videoOption == VIDEO_OTHER) { - regSetDwordValue("video", VIDEO_1X); - if(pVideoDriverGUID) - regSetDwordValue("defaultVideoDriver", TRUE); - } - changingVideoSize = false; - AfxPostQuitMessage(0); - return; - } - if(!initInput()) { - changingVideoSize = false; - AfxPostQuitMessage(0); - return; - } - input->checkKeys(); - updateMenuBar(); - changingVideoSize = FALSE; - updateWindowSize(videoOption); - return; - } - - sizeX = 240; - sizeY = 160; - - videoOption = value; - - if(cartridgeType == 1) { - if(gbBorderOn) { - sizeX = 256; - sizeY = 224; - gbBorderLineSkip = 256; - gbBorderColumnSkip = 48; - gbBorderRowSkip = 40; - } else { - sizeX = 160; - sizeY = 144; - gbBorderLineSkip = 160; - gbBorderColumnSkip = 0; - gbBorderRowSkip = 0; - } - } - - surfaceSizeX = sizeX; - surfaceSizeY = sizeY; - - switch(videoOption) { - case VIDEO_1X: - surfaceSizeX = sizeX; - surfaceSizeY = sizeY; - break; - case VIDEO_2X: - surfaceSizeX = sizeX * 2; - surfaceSizeY = sizeY * 2; - break; - case VIDEO_3X: - surfaceSizeX = sizeX * 3; - surfaceSizeY = sizeY * 3; - break; - case VIDEO_4X: - surfaceSizeX = sizeX * 4; - surfaceSizeY = sizeY * 4; - break; - case VIDEO_320x240: - case VIDEO_640x480: - case VIDEO_800x600: - case VIDEO_1024x768: - case VIDEO_1280x1024: - case VIDEO_OTHER: - { - float scaleX = 1; - float scaleY = 1; - scaleX = ((float)fsWidth / sizeX); - scaleY = ((float)fsHeight / sizeY); - float min = scaleX < scaleY ? scaleX : scaleY; - if(fsMaxScale) - min = min > fsMaxScale ? fsMaxScale : min; - surfaceSizeX = (int)(min * sizeX); - surfaceSizeY = (int)(min * sizeY); - - if(fullScreenStretch) - { - surfaceSizeX = fsWidth; - surfaceSizeY = fsHeight; - } - } - break; - } - - rect.right = sizeX; - rect.bottom = sizeY; - - int winSizeX = sizeX; - int winSizeY = sizeY; - - if(videoOption <= VIDEO_4X) { - dest.left = 0; - dest.top = 0; - dest.right = surfaceSizeX; - dest.bottom = surfaceSizeY; - - DWORD style = WS_POPUP | WS_VISIBLE; - - style |= WS_OVERLAPPEDWINDOW; - - menuToggle = TRUE; - AdjustWindowRectEx(&dest, style, TRUE, 0); //WS_EX_TOPMOST); - - winSizeX = dest.right-dest.left; - winSizeY = dest.bottom-dest.top; - - m_pMainWnd->SetWindowPos(0, //HWND_TOPMOST, - windowPositionX, - windowPositionY, - winSizeX, - winSizeY, - SWP_NOMOVE | SWP_SHOWWINDOW); - - winCheckMenuBarInfo(winSizeX, winSizeY); - } - - adjustDestRect(); - - updateIFB(); - updateFilter(); - - m_pMainWnd->RedrawWindow(NULL,NULL,RDW_INVALIDATE|RDW_ERASE|RDW_ALLCHILDREN); -} - -bool VBA::initDisplay() -{ - return updateRenderMethod(false); -} - -bool VBA::updateRenderMethod(bool force) -{ - bool res = updateRenderMethod0(force); - - while(!res && renderMethod > 0) { - if(renderMethod == OPENGL) - renderMethod = DIRECT_3D; - else if(renderMethod == DIRECT_3D) - renderMethod = DIRECT_DRAW; - else if(renderMethod == DIRECT_DRAW) { - if(videoOption > VIDEO_4X) { - videoOption = VIDEO_2X; - force = true; - } else - renderMethod = GDI; - } - - res = updateRenderMethod(force); - } - return res; -} - -bool VBA::updateRenderMethod0(bool force) -{ - bool initInput = false; - b16to32Video = false; - - if(display) { - if(display->getType() != renderMethod || force) { - initInput = true; - changingVideoSize = true; - shutdownDisplay(); - if(input) { - delete input; - input = NULL; - } - CWnd *pWnd = m_pMainWnd; - - m_pMainWnd = NULL; - pWnd->DragAcceptFiles(FALSE); - pWnd->DestroyWindow(); - delete pWnd; - - display = NULL; - regSetDwordValue("renderMethod", renderMethod); - } - } - if(display == NULL) { - switch(renderMethod) { - case GDI: - display = newGDIDisplay(); - break; - case DIRECT_DRAW: - display = newDirectDrawDisplay(); - break; - case DIRECT_3D: - display = newDirect3DDisplay(); - break; - case OPENGL: - display = newOpenGLDisplay(); - break; - } - - if(display->initialize()) { - if(initInput) { - if(!this->initInput()) { - changingVideoSize = false; - AfxPostQuitMessage(0); - return false; - } - input->checkKeys(); - updateMenuBar(); - changingVideoSize = false; - updateWindowSize(videoOption); - - m_pMainWnd->ShowWindow(SW_SHOW); - m_pMainWnd->UpdateWindow(); - m_pMainWnd->SetFocus(); - - return true; - } else { - changingVideoSize = false; - return true; - } - } - changingVideoSize = false; - } - return true; -} - -void VBA::winCheckFullscreen() -{ - if(videoOption > VIDEO_4X && tripleBuffering) { - if(display) - display->checkFullScreen(); - } -} - -void VBA::shutdownDisplay() -{ - if(display != NULL) { - display->cleanup(); - delete display; - display = NULL; - } -} - -void VBA::directXMessage(const char *msg) -{ - systemMessage(IDS_DIRECTX_7_REQUIRED, - "DirectX 7.0 or greater is required to run.\nDownload at http://www.microsoft.com/directx.\n\nError found at: %s", - msg); -} - -void VBA::updatePriority() -{ - switch(threadPriority) { - case 0: - SetThreadPriority(THREAD_PRIORITY_HIGHEST); - break; - case 1: - SetThreadPriority(THREAD_PRIORITY_ABOVE_NORMAL); - break; - case 3: - SetThreadPriority(THREAD_PRIORITY_BELOW_NORMAL); - break; - default: - SetThreadPriority(THREAD_PRIORITY_NORMAL); - } -} - -#ifdef MMX -bool VBA::detectMMX() -{ - bool support = false; - char brand[13]; - - // check for Intel chip - __try { - __asm { - mov eax, 0; - cpuid; - mov [dword ptr brand+0], ebx; - mov [dword ptr brand+4], edx; - mov [dword ptr brand+8], ecx; - } - } - __except(EXCEPTION_EXECUTE_HANDLER) { - if(_exception_code() == STATUS_ILLEGAL_INSTRUCTION) { - return false; - } - return false; - } - // Check for Intel or AMD CPUs - if(strncmp(brand, "GenuineIntel", 12)) { - if(strncmp(brand, "AuthenticAMD", 12)) { - return false; - } - } - - __asm { - mov eax, 1; - cpuid; - test edx, 00800000h; - jz NotFound; - mov [support], 1; - NotFound: - } - return support; -} -#endif - -void VBA::winSetLanguageOption(int option, bool force) -{ - if(((option == languageOption) && option != 2) && !force) - return; - switch(option) { - case 0: - { - char lbuffer[10]; - - if(GetLocaleInfo(LOCALE_SYSTEM_DEFAULT, LOCALE_SABBREVLANGNAME, - lbuffer, 10)) { - HINSTANCE l = winLoadLanguage(lbuffer); - if(l == NULL) { - LCID locIdBase = MAKELCID( MAKELANGID( PRIMARYLANGID( GetSystemDefaultLangID() ), SUBLANG_NEUTRAL ), SORT_DEFAULT ); - if(GetLocaleInfo(locIdBase, LOCALE_SABBREVLANGNAME, - lbuffer, 10)) { - l = winLoadLanguage(lbuffer); - if(l == NULL) { - systemMessage(IDS_FAILED_TO_LOAD_LIBRARY, - "Failed to load library %s", - lbuffer); - return; - } - } - } - AfxSetResourceHandle(l); - if(languageModule != NULL) - FreeLibrary(languageModule); - languageModule = l; - } else { - systemMessage(IDS_FAILED_TO_GET_LOCINFO, - "Failed to get locale information"); - return; - } - } - break; - case 1: - if(languageModule != NULL) - FreeLibrary(languageModule); - languageModule = NULL; - AfxSetResourceHandle(AfxGetInstanceHandle()); - break; - case 2: - { - if(!force) { - LangSelect dlg; - if(dlg.DoModal()) { - HINSTANCE l = winLoadLanguage(languageName); - if(l == NULL) { - systemMessage(IDS_FAILED_TO_LOAD_LIBRARY, - "Failed to load library %s", - languageName); - return; - } - AfxSetResourceHandle(l); - if(languageModule != NULL) - FreeLibrary(languageModule); - languageModule = l; - } - } else { - if(languageName.IsEmpty()) - return; - HINSTANCE l = winLoadLanguage(languageName); - if(l == NULL) { - systemMessage(IDS_FAILED_TO_LOAD_LIBRARY, - "Failed to load library %s", - languageName); - return; - } - AfxSetResourceHandle(l); - if(languageModule != NULL) - FreeLibrary(languageModule); - languageModule = l; - } - } - break; - } - languageOption = option; - updateMenuBar(); -} - -HINSTANCE VBA::winLoadLanguage(const char *name) -{ - CString buffer; - - buffer.Format("vba_%s.dll", name); - - HINSTANCE l = LoadLibrary(buffer); - - if(l == NULL) { - if(strlen(name) == 3) { - char buffer2[3]; - buffer2[0] = name[0]; - buffer2[1] = name[1]; - buffer2[2] = 0; - buffer.Format("vba_%s.dll", buffer2); - - return LoadLibrary(buffer); - } - } - return l; -} - - -bool VBA::initInput() -{ - if(input) - delete input; - input = newDirectInput(); - if(input->initialize()) { - input->loadSettings(); - input->checkKeys(); - return true; - } - delete input; - return false; -} - -void VBA::winAddUpdateListener(IUpdateListener *l) -{ - updateList.AddTail(l); - updateCount++; -} - -void VBA::winRemoveUpdateListener(IUpdateListener *l) -{ - POSITION pos = updateList.Find(l); - if(pos) { - updateList.RemoveAt(pos); - updateCount--; - if(updateCount < 0) - updateCount = 0; - } -} - -CString VBA::winLoadFilter(UINT id) -{ - CString res = winResLoadString(id); - res.Replace('_','|'); - - return res; -} - -void VBA::movieReadNext() -{ - if(movieFile) { - bool movieEnd = false; - - if(fread(&moviePlayFrame, 1, sizeof(int), movieFile) == sizeof(int)) { - if(fread(&movieNextJoypad, 1, sizeof(u32), movieFile) == sizeof(int)) { - // make sure we don't have spurious entries on the movie that can - // cause us to play it forever - if(moviePlayFrame <= movieFrame) - movieEnd = true; - } else - movieEnd = true; - } else - movieEnd = true; - if(movieEnd) { - CString string = winResLoadString(IDS_END_OF_MOVIE); - systemScreenMessage(string); - moviePlaying = false; - fclose(movieFile); - movieFile = NULL; - return; - } - } else - moviePlaying = false; -} - -void VBA::saveSettings() -{ - regSetDwordValue("language", languageOption); - - regSetStringValue("languageName", languageName); - - regSetDwordValue("frameSkip", frameSkip); - - regSetDwordValue("gbFrameSkip", gbFrameSkip); - - regSetDwordValue("autoFrameSkip", autoFrameSkip); - - regSetDwordValue("vsync", vsync); - regSetDwordValue("synchronize", synchronize); - regSetDwordValue("stretch", fullScreenStretch); - - regSetDwordValue("video", videoOption); - - regSetDwordValue("defaultVideoDriver", pVideoDriverGUID == NULL); - - if(pVideoDriverGUID) { - regSetBinaryValue("videoDriverGUID", (char *)&videoDriverGUID, - sizeof(GUID)); - } - - - regSetDwordValue("fsWidth", fsWidth); - regSetDwordValue("fsHeight", fsHeight); - regSetDwordValue("fsColorDepth", fsColorDepth); - regSetDwordValue("fsFrequency", fsFrequency); - regSetDwordValue("fsAdapter", fsAdapter); - - regSetDwordValue("renderMethod", renderMethod); - - regSetDwordValue("windowX", windowPositionX); - regSetDwordValue("windowY", windowPositionY); - - regSetDwordValue("useBios", useBiosFile); - - regSetDwordValue("skipBios", skipBiosFile); - - if(!biosFileName.IsEmpty()) - regSetStringValue("biosFile", biosFileName); - - regSetDwordValue("soundEnable", soundGetEnable() & 0x30f); - - regSetDwordValue("soundOff", soundOffFlag); - - regSetDwordValue("soundQuality", soundQuality); - - regSetDwordValue("soundEcho", soundEcho); - - regSetDwordValue("soundLowPass", soundLowPass); - - regSetDwordValue("soundReverse", soundReverse); - - regSetDwordValue("soundVolume", soundVolume); - - regSetDwordValue("soundInterpolation", soundInterpolation); - - regSetDwordValue("ddrawEmulationOnly", ddrawEmulationOnly); - regSetDwordValue("ddrawUseVideoMemory", ddrawUseVideoMemory); - regSetDwordValue("tripleBuffering", tripleBuffering); - - regSetDwordValue("d3dFilter", d3dFilter); - regSetDwordValue("glFilter", glFilter); - regSetDwordValue("glType", glType); - - regSetDwordValue("filter", filterType); - - regSetDwordValue("LCDFilter", filterLCD); - - regSetDwordValue("disableMMX", disableMMX); - - regSetDwordValue("disableStatus", disableStatusMessage); - - regSetDwordValue("showSpeed", showSpeed); - - regSetDwordValue("showSpeedTransparent", showSpeedTransparent); - - regSetDwordValue("gbPrinter", winGbPrinterEnabled); - - regSetDwordValue("pauseWhenInactive", pauseWhenInactive); - - regSetDwordValue("useOldSync", useOldSync); - - regSetDwordValue("captureFormat", captureFormat); - - regSetDwordValue("removeIntros", removeIntros); - - regSetDwordValue("recentFreeze", recentFreeze); - - regSetDwordValue("autoIPS", autoIPS); - - regSetDwordValue("disableSfx", cpuDisableSfx); - - regSetDwordValue("saveType", winSaveType); - - regSetDwordValue("enhancedDetection", cpuEnhancedDetection); - - regSetDwordValue("ifbType", ifbType); - - regSetDwordValue("flashSize", winFlashSize); - - regSetDwordValue("agbPrint", agbPrintIsEnabled()); - - regSetDwordValue("rtcEnabled", winRtcEnable); - - regSetDwordValue("autoHideMenu", autoHideMenu); - - regSetDwordValue("borderOn", winGbBorderOn); - regSetDwordValue("borderAutomatic", gbBorderAutomatic); - regSetDwordValue("emulatorType", gbEmulatorType); - regSetDwordValue("colorOption", gbColorOption); - - regSetDwordValue("priority", threadPriority); - - regSetDwordValue("autoSaveCheatList", autoSaveLoadCheatList); - - regSetDwordValue("gbPaletteOption", gbPaletteOption); - - regSetBinaryValue("gbPalette", (char *)systemGbPalette, - 24*sizeof(u16)); - - regSetDwordValue("rewindTimer", rewindTimer/6); - - CString buffer; - for(int i = 0; i < 10; i++) { - buffer.Format("recent%d", i); - regSetStringValue(buffer, recentFiles[i]); - } - - regSetDwordValue("joypadDefault", joypadDefault); - regSetDwordValue("autoLoadMostRecent", autoLoadMostRecent); - regSetDwordValue("cheatsEnabled", cheatsEnabled); - regSetDwordValue("fsMaxScale", fsMaxScale); - regSetDwordValue("throttle", throttle); - - - regSetDwordValue("LinkTimeout", linktimeout); - regSetDwordValue("Linklog", linklog); - regSetDwordValue("RFU", adapter); -} - -void winSignal(int, int) -{ -} - -#define CPUReadByteQuick(addr) \ - map[(addr)>>24].address[(addr) & map[(addr)>>24].mask] - -void winOutput(char *s, u32 addr) -{ - if(s) { - toolsLog(s); - } else { - CString str; - char c; - - c = CPUReadByteQuick(addr); - addr++; - while(c) { - str += c; - c = CPUReadByteQuick(addr); - addr++; - } - toolsLog(str); - } -} +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2005 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +// VBA.cpp : Defines the class behaviors for the application. +// +#include "stdafx.h" + +#include "AVIWrite.h" +#include "LangSelect.h" +#include "MainWnd.h" +#include "Reg.h" +#include "resource.h" +#include "resource2.h" +#include "skin.h" +#include "WavWriter.h" +#include "WinResUtil.h" + +#include "../System.h" +#include "../agbprint.h" +#include "../cheatSearch.h" +#include "../GBA.h" +#include "../Globals.h" +#include "../RTC.h" +#include "../Sound.h" +#include "../Util.h" +#include "../gb/gbGlobals.h" +#include "../gb/gbPrinter.h" + +/* Link +---------------------*/ +#include "../Link.h" +/* ---------------- */ + +#include "../gbafilter.h" + +#ifdef SDL +#pragma comment( lib, "SDL" ) +#pragma comment( lib, "SDLmain" ) +#endif + +extern void Pixelate(u8*,u32,u8*,u8*,u32,int,int); +extern void Pixelate32(u8*,u32,u8*,u8*,u32,int,int); +extern void MotionBlur(u8*,u32,u8*,u8*,u32,int,int); +extern void MotionBlur32(u8*,u32,u8*,u8*,u32,int,int); +extern void _2xSaI(u8*,u32,u8*,u8*,u32,int,int); +extern void _2xSaI32(u8*,u32,u8*,u8*,u32,int,int); +extern void Super2xSaI(u8*,u32,u8*,u8*,u32,int,int); +extern void Super2xSaI32(u8*,u32,u8*,u8*,u32,int,int); +extern void SuperEagle(u8*,u32,u8*,u8*,u32,int,int); +extern void SuperEagle32(u8*,u32,u8*,u8*,u32,int,int); +extern void AdMame2x(u8*,u32,u8*,u8*,u32,int,int); +extern void AdMame2x32(u8*,u32,u8*,u8*,u32,int,int); +extern void Bilinear(u8*,u32,u8*,u8*,u32,int,int); +extern void Bilinear32(u8*,u32,u8*,u8*,u32,int,int); +extern void BilinearPlus(u8*,u32,u8*,u8*,u32,int,int); +extern void BilinearPlus32(u8*,u32,u8*,u8*,u32,int,int); +extern void Scanlines(u8*,u32,u8*,u8*,u32,int,int); +extern void Scanlines32(u8*,u32,u8*,u8*,u32,int,int); +extern void ScanlinesTV(u8*,u32,u8*,u8*,u32,int,int); +extern void ScanlinesTV32(u8*,u32,u8*,u8*,u32,int,int); +extern void hq2x(u8*,u32,u8*,u8*,u32,int,int); +extern void hq2x32(u8*,u32,u8*,u8*,u32,int,int); +extern void lq2x(u8*,u32,u8*,u8*,u32,int,int); +extern void lq2x32(u8*,u32,u8*,u8*,u32,int,int); +extern void Simple2x16(u8*,u32,u8*,u8*,u32,int,int); +extern void Simple2x32(u8*,u32,u8*,u8*,u32,int,int); +extern void Simple3x16(u8*,u32,u8*,u8*,u32,int,int); +extern void Simple3x32(u8*,u32,u8*,u8*,u32,int,int); +extern void Simple4x16(u8*,u32,u8*,u8*,u32,int,int); +extern void Simple4x32(u8*,u32,u8*,u8*,u32,int,int); + +extern void hq3x32(u8*,u32,u8*,u8*,u32,int,int); +extern void hq3x16(u8*,u32,u8*,u8*,u32,int,int); +extern void hq4x32(u8*,u32,u8*,u8*,u32,int,int); +extern void hq4x16(u8*,u32,u8*,u8*,u32,int,int); + +extern void SmartIB(u8*,u32,int,int); +extern void SmartIB32(u8*,u32,int,int); +extern void MotionBlurIB(u8*,u32,int,int); +extern void InterlaceIB(u8*,u32,int,int); +extern void MotionBlurIB32(u8*,u32,int,int); + +extern void toolsLog(const char *); + +extern IDisplay *newGDIDisplay(); +extern IDisplay *newDirectDrawDisplay(); +extern IDisplay *newDirect3DDisplay(); +extern IDisplay *newOpenGLDisplay(); + +extern Input *newDirectInput(); + +extern ISound *newDirectSound(); + +extern void remoteStubSignal(int, int); +extern void remoteOutput(char *, u32); +extern void remoteStubMain(); +extern void remoteSetProtocol(int); +extern void remoteCleanUp(); +extern int remoteSocket; + +extern void InterframeCleanup(); + +void winlog(const char *msg, ...); + +/* Link +---------------------*/ +extern int InitLink(void); +extern void CloseLink(void); +//extern int linkid; +extern char inifile[]; +extern FILE *linklogfile; +/* ------------------- */ +#ifdef _DEBUG +#define new DEBUG_NEW +#undef THIS_FILE +static char THIS_FILE[] = __FILE__; +#endif + +int emulating = 0; +bool debugger = false; +int RGB_LOW_BITS_MASK = 0; +bool b16to32Video = false; +int systemFrameSkip = 0; +int systemSpeed = 0; +bool systemSoundOn = false; +u32 systemColorMap32[0x10000]; +u16 systemColorMap16[0x10000]; +u16 systemGbPalette[24]; +int systemRedShift = 0; +int systemBlueShift = 0; +int systemGreenShift = 0; +int systemColorDepth = 16; +int systemVerbose = 0; +int systemDebug = 0; +int systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; +bool soundBufferLow = 0; +void winSignal(int,int); +void winOutput(char *, u32); + +void (*dbgSignal)(int,int) = winSignal; +void (*dbgOutput)(char *, u32) = winOutput; + +#ifdef MMX +extern "C" bool cpu_mmx; +#endif + +namespace Sm60FPS +{ + float K_fCpuSpeed = 98.0f; + float K_fTargetFps = 60.0f * K_fCpuSpeed / 100; + float K_fDT = 1000.0f / K_fTargetFps; + + u32 dwTimeElapse; + u32 dwTime0; + u32 dwTime1; + u32 nFrameCnt; + float fWantFPS; + float fCurFPS; + bool bLastSkip; + int nCurSpeed; + int bSaveMoreCPU; +}; + +void directXMessage(const char *msg) +{ + systemMessage(IDS_DIRECTX_7_REQUIRED, + "DirectX 7.0 or greater is required to run.\nDownload at http://www.microsoft.com/directx.\n\nError found at: %s", + msg); +} + +///////////////////////////////////////////////////////////////////////////// +// VBA + +BEGIN_MESSAGE_MAP(VBA, CWinApp) + //{{AFX_MSG_MAP(VBA) + // NOTE - the ClassWizard will add and remove mapping macros here. + // DO NOT EDIT what you see in these blocks of generated code! + //}}AFX_MSG_MAP + END_MESSAGE_MAP() + + ///////////////////////////////////////////////////////////////////////////// +// VBA construction + +VBA::VBA() +{ + mode320Available = false; + mode640Available = false; + mode800Available = false; + mode1024Available = false; + mode1280Available = false; + windowPositionX = 0; + windowPositionY = 0; + filterFunction = NULL; + ifbFunction = NULL; + ifbType = 0; + filterType = FILTER_NONE; + filterWidth = 0; + filterHeight = 0; + fsAdapter = 0; + fsWidth = 0; + fsHeight = 0; + fsColorDepth = 0; + fsFrequency = 0; + fsForceChange = false; + surfaceSizeX = 0; + surfaceSizeY = 0; + sizeX = 0; + sizeY = 0; + videoOption = 0; + fullScreenStretch = false; + disableStatusMessage = false; + showSpeed = 0; + showSpeedTransparent = true; + showRenderedFrames = 0; + screenMessage = false; + screenMessageTime = 0; + menuToggle = true; + display = NULL; + menu = NULL; + popup = NULL; + cartridgeType = IMAGE_GBA; + soundInitialized = false; + useBiosFile = false; + skipBiosFile = false; + active = true; + paused = false; + recentFreeze = false; + autoSaveLoadCheatList = false; + winout = NULL; + removeIntros = false; + autoIPS = true; + winGbBorderOn = 0; + winFlashSize = 0x20000; + winRtcEnable = false; + winGenericflashcardEnable = false; + winSaveType = 0; + rewindMemory = NULL; + rewindPos = 0; + rewindTopPos = 0; + rewindCounter = 0; + rewindCount = 0; + rewindSaveNeeded = false; + rewindTimer = 0; + captureFormat = 0; + tripleBuffering = true; + autoHideMenu = false; + throttle = 0; + throttleLastTime = 0; + autoFrameSkipLastTime = 0; + autoFrameSkip = false; + vsync = false; + changingVideoSize = false; + pVideoDriverGUID = NULL; + renderMethod = DIRECT_DRAW; + iconic = false; + ddrawEmulationOnly = false; + ddrawUsingEmulationOnly = false; + ddrawDebug = false; + ddrawUseVideoMemory = false; + d3dFilter = 0; + glFilter = 0; + glType = 0; + skin = NULL; + skinName = ""; + skinEnabled = false; + skinButtons = 0; + regEnabled = false; + pauseWhenInactive = true; + speedupToggle = false; + useOldSync = false; + winGbPrinterEnabled = false; + threadPriority = 2; + disableMMX = false; + languageOption = 0; + languageModule = NULL; + languageName = ""; + renderedFrames = 0; + input = NULL; + joypadDefault = 0; + autoFire = 0; + autoFireToggle = false; + winPauseNextFrame = false; + soundRecording = false; + soundRecorder = NULL; + dsoundDisableHardwareAcceleration = true; + sound = NULL; + aviRecording = false; + aviRecorder = NULL; + aviFrameNumber = 0; + painting = false; + movieRecording = false; + moviePlaying = false; + movieFrame = 0; + moviePlayFrame = 0; + movieFile = NULL; + movieLastJoypad = 0; + movieNextJoypad = 0; + sensorX = 2047; + sensorY = 2047; + mouseCounter = 0; + wasPaused = false; + frameskipadjust = 0; + autoLoadMostRecent = false; + fsMaxScale = 0; + romSize = 0; + + updateCount = 0; + + systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; + + ZeroMemory(&emulator, sizeof(emulator)); + + hAccel = NULL; + + for(int i = 0; i < 24;) { + systemGbPalette[i++] = (0x1f) | (0x1f << 5) | (0x1f << 10); + systemGbPalette[i++] = (0x15) | (0x15 << 5) | (0x15 << 10); + systemGbPalette[i++] = (0x0c) | (0x0c << 5) | (0x0c << 10); + systemGbPalette[i++] = 0; + } +} + +VBA::~VBA() +{ + InterframeCleanup(); + + char winBuffer[2048]; + + GetModuleFileName(NULL, winBuffer, 2048); + char *p = strrchr(winBuffer, '\\'); + if(p) + *p = 0; + + regInit(winBuffer); + + saveSettings(); + + if(moviePlaying) { + if(movieFile != NULL) { + fclose(movieFile); + movieFile = NULL; + } + moviePlaying = false; + movieLastJoypad = 0; + } + + if(movieRecording) { + if(movieFile != NULL) { + // record the last joypad change so that the correct time can be + // recorded + fwrite(&movieFrame, 1, sizeof(int), movieFile); + fwrite(&movieLastJoypad, 1, sizeof(u32), movieFile); + fclose(movieFile); + movieFile = NULL; + } + movieRecording = false; + moviePlaying = false; + movieLastJoypad = 0; + } + + if(aviRecorder) { + delete aviRecorder; + aviRecording = false; + } + + if(soundRecorder) { + delete soundRecorder; + soundRecorder = NULL; + } + soundRecording = false; + soundPause(); + soundShutdown(); + + if(gbRom != NULL || rom != NULL) { + if(autoSaveLoadCheatList) + ((MainWnd *)m_pMainWnd)->winSaveCheatListDefault(); + ((MainWnd *)m_pMainWnd)->writeBatteryFile(); + cheatSearchCleanup(&cheatSearchData); + emulator.emuCleanUp(); + } + + if(input) + delete input; + + shutdownDisplay(); + + if(skin) { + delete skin; + } + + if(rewindMemory) + free(rewindMemory); +} + +///////////////////////////////////////////////////////////////////////////// +// The one and only VBA object + +VBA theApp; +///////////////////////////////////////////////////////////////////////////// +// VBA initialization + +// code from SDL_main.c for Windows +/* Parse a command line buffer into arguments */ +static int parseCommandLine(char *cmdline, char **argv) +{ + char *bufp; + int argc; + + argc = 0; + for ( bufp = cmdline; *bufp; ) { + /* Skip leading whitespace */ + while ( isspace(*bufp) ) { + ++bufp; + } + /* Skip over argument */ + if ( *bufp == '"' ) { + ++bufp; + if ( *bufp ) { + if ( argv ) { + argv[argc] = bufp; + } + ++argc; + } + /* Skip over word */ + while ( *bufp && (*bufp != '"') ) { + ++bufp; + } + } else { + if ( *bufp ) { + if ( argv ) { + argv[argc] = bufp; + } + ++argc; + } + /* Skip over word */ + while ( *bufp && ! isspace(*bufp) ) { + ++bufp; + } + } + if ( *bufp ) { + if ( argv ) { + *bufp = '\0'; + } + ++bufp; + } + } + if ( argv ) { + argv[argc] = NULL; + } + return(argc); +} + +BOOL VBA::InitInstance() +{ +#if _MSC_VER < 1400 +#ifdef _AFXDLL + Enable3dControls(); // Call this when using MFC in a shared DLL +#else + Enable3dControlsStatic(); // Call this when linking to MFC statically +#endif +#endif + + SetRegistryKey(_T("VBA")); + + remoteSetProtocol(0); + + systemVerbose = GetPrivateProfileInt("config", + "verbose", + 0, + "VBA.ini"); + + systemDebug = GetPrivateProfileInt("config", + "debug", + 0, + "VBA.ini"); + ddrawDebug = GetPrivateProfileInt("config", + "ddrawDebug", + 0, + "VBA.ini") ? true : false; + + wndClass = AfxRegisterWndClass(0, LoadCursor(IDC_ARROW), (HBRUSH)GetStockObject(BLACK_BRUSH), LoadIcon(IDI_ICON)); + + char winBuffer[2048]; + + GetModuleFileName(NULL, winBuffer, 2048); + char *p = strrchr(winBuffer, '\\'); + if(p) + *p = 0; + + if(!InitLink()) + return FALSE;; + + regInit(winBuffer); + + loadSettings(); + + if(!openLinkLog()) + return FALSE; + + if(!initInput()) + return FALSE; + + if(!initDisplay()) { + if(videoOption >= VIDEO_320x240) { + regSetDwordValue("video", VIDEO_1X); + if(pVideoDriverGUID) + regSetDwordValue("defaultVideoDriver", TRUE); + } + return FALSE; + } + + hAccel = LoadAccelerators(AfxGetInstanceHandle(), MAKEINTRESOURCE(IDR_ACCELERATOR)); + + winAccelMgr.Connect((MainWnd *)m_pMainWnd); + + winAccelMgr.SetRegKey(HKEY_CURRENT_USER, "Software\\Emulators\\VisualBoyAdvance"); + + extern void winAccelAddCommands(CAcceleratorManager&); + + winAccelAddCommands(winAccelMgr); + + winAccelMgr.CreateDefaultTable(); + + winAccelMgr.Load(); + + winAccelMgr.UpdateWndTable(); + + winAccelMgr.UpdateMenu(menu); + + if (m_lpCmdLine[0]) + { + int argc = parseCommandLine(m_lpCmdLine, NULL); + char **argv = (char **)malloc((argc+1)*sizeof(char *)); + parseCommandLine(m_lpCmdLine, argv); + if(argc > 0) { + szFile = argv[0]; + filename = szFile; + } + int index = filename.ReverseFind('.'); + + if(index != -1) + filename = filename.Left(index); + + if(((MainWnd*)m_pMainWnd)->FileRun()) + emulating = true; + else + emulating = false; + free(argv); + } + + return TRUE; +} + +void VBA::adjustDestRect() +{ + POINT point; + RECT skinRect; + if(skin) + skinRect = skin->GetBlitRect(); + + point.x = 0; + point.y = 0; + + if(skin) { + point.x = skinRect.left; + point.y = skinRect.top; + } + + m_pMainWnd->ClientToScreen(&point); + dest.top = point.y; + dest.left = point.x; + + point.x = surfaceSizeX; + point.y = surfaceSizeY; + + if(skin) { + point.x = skinRect.right; + point.y = skinRect.bottom; + } + + m_pMainWnd->ClientToScreen(&point); + dest.bottom = point.y; + dest.right = point.x; + + // make sure that dest rect lies in the monitor + if(videoOption >= VIDEO_320x240) { + dest.top -= windowPositionY; + dest.left -= windowPositionX; + dest.bottom-= windowPositionY; + dest.right -= windowPositionX; + } + + if(skin) + return; + + int menuSkip = 0; + + if(videoOption >= VIDEO_320x240 && menuToggle) { + int m = GetSystemMetrics(SM_CYMENU); + menuSkip = m; + dest.bottom -=m; + } + + if(videoOption > VIDEO_4X) { + int top = (fsHeight - surfaceSizeY) / 2; + int left = (fsWidth - surfaceSizeX) / 2; + dest.top += top; + dest.bottom += top; + dest.left += left; + dest.right += left; + if(fullScreenStretch) { + dest.top = 0+menuSkip; + dest.left = 0; + dest.right = fsWidth; + dest.bottom = fsHeight; + } + } +} + + +void VBA::updateIFB() +{ + if(systemColorDepth == 16) { + switch(ifbType) { + case 0: + default: + ifbFunction = NULL; + break; + case 1: + ifbFunction = MotionBlurIB; + break; + case 2: + ifbFunction = SmartIB; + break; + } + } else if(systemColorDepth == 32) { + switch(ifbType) { + case 0: + default: + ifbFunction = NULL; + break; + case 1: + ifbFunction = MotionBlurIB32; + break; + case 2: + ifbFunction = SmartIB32; + break; + } + } else + ifbFunction = NULL; +} + +void VBA::updateFilter() +{ + // BEGIN hacky ugly code + + // HQ3X asm wants 16 bit input. When we switch + // away from 16 bits we need to restore the driver values + + if ( b16to32Video ) + { + b16to32Video = false; + systemColorDepth = 32; + systemRedShift = 19; + systemGreenShift = 11; + systemBlueShift = 3; + utilUpdateSystemColorMaps(); + } + // END hacky ugly code + + filterWidth = sizeX; + filterHeight = sizeY; + filterMagnification = 1; + + + if ( videoOption == VIDEO_1X || videoOption == VIDEO_320x240 ) + { + filterFunction = NULL; + filterMagnification = 1; + } + else + { + if ( systemColorDepth == 16 ) + { + switch(filterType) + { + default: + case FILTER_NONE: + filterFunction = NULL; + filterMagnification = 1; + break; + case FILTER_TVMODE: + filterFunction = ScanlinesTV; + filterMagnification = 2; + break; + case FILTER_2XSAI: + filterFunction = _2xSaI; + filterMagnification = 2; + break; + case FILTER_SUPER2XSAI: + filterFunction = Super2xSaI; + filterMagnification = 2; + break; + case FILTER_SUPEREAGLE: + filterFunction = SuperEagle; + filterMagnification = 2; + break; + case FILTER_PIXELATE: + filterFunction = Pixelate; + filterMagnification = 2; + break; + case FILTER_MAMESCALE2X: + filterFunction = AdMame2x; + filterMagnification = 2; + break; + case FILTER_SIMPLE2X: + filterFunction = Simple2x16; + filterMagnification = 2; + break; + case FILTER_BILINEAR: + filterFunction = Bilinear; + filterMagnification = 2; + break; + case FILTER_BILINEARPLUS: + filterFunction = BilinearPlus; + filterMagnification = 2; + break; + case FILTER_SCANLINES: + filterFunction = Scanlines; + filterMagnification = 2; + break; + case FILTER_HQ2X: + filterFunction = hq2x; + filterMagnification = 2; + break; + case FILTER_LQ2X: + filterFunction = lq2x; + filterMagnification = 2; + break; + case FILTER_SIMPLE3X: + filterFunction = Simple3x16; + filterMagnification = 3; + break; + case FILTER_SIMPLE4X: + filterFunction = Simple4x16; + filterMagnification = 4; + break; + case FILTER_HQ3X: + filterFunction = hq3x16; + filterMagnification = 3; + break; + case FILTER_HQ4X: + filterFunction = hq4x16; + filterMagnification = 4; + break; + } + } + + if ( systemColorDepth == 32 ) + { + switch(filterType) + { + default: + case FILTER_NONE: + filterFunction = NULL; + filterMagnification = 1; + break; + case FILTER_TVMODE: + filterFunction = ScanlinesTV32; + filterMagnification = 2; + break; + case FILTER_2XSAI: + filterFunction = _2xSaI32; + filterMagnification = 2; + break; + case FILTER_SUPER2XSAI: + filterFunction = Super2xSaI32; + filterMagnification = 2; + break; + case FILTER_SUPEREAGLE: + filterFunction = SuperEagle32; + filterMagnification = 2; + break; + case FILTER_PIXELATE: + filterFunction = Pixelate32; + filterMagnification = 2; + break; + case FILTER_MAMESCALE2X: + filterFunction = AdMame2x32; + filterMagnification = 2; + break; + case FILTER_SIMPLE2X: + filterFunction = Simple2x32; + filterMagnification = 2; + break; + case FILTER_BILINEAR: + filterFunction = Bilinear32; + filterMagnification = 2; + break; + case FILTER_BILINEARPLUS: + filterFunction = BilinearPlus32; + filterMagnification = 2; + break; + case FILTER_SCANLINES: + filterFunction = Scanlines32; + filterMagnification = 2; + break; + case FILTER_HQ2X: + filterFunction = hq2x32; + filterMagnification = 2; + break; + case FILTER_LQ2X: + filterFunction = lq2x32; + filterMagnification = 2; + break; + case FILTER_SIMPLE3X: + filterFunction = Simple3x32; + filterMagnification = 3; + break; + case FILTER_SIMPLE4X: + filterFunction = Simple4x32; + filterMagnification = 4; + break; + case FILTER_HQ3X: + filterFunction = hq3x32; + filterMagnification = 3; + b16to32Video=true; + break; + case FILTER_HQ4X: + filterFunction = hq4x32; + filterMagnification = 4; + b16to32Video=true; + break; + } + } + } + + rect.right = sizeX * filterMagnification; + rect.bottom = sizeY * filterMagnification; + + if( filterType != FILTER_NONE ) + memset(delta, 0xFF, sizeof(delta)); + + if( display ) + display->changeRenderSize(rect.right, rect.bottom); + + if (b16to32Video) + { + systemColorDepth = 16; + systemRedShift = 11; + systemGreenShift = 6; + systemBlueShift = 0; + utilUpdateSystemColorMaps(); + } +} + + +void VBA::updateMenuBar() +{ + if(menu != NULL) { + if(m_pMainWnd) + m_pMainWnd->SetMenu(NULL); + m_menu.Detach(); + DestroyMenu(menu); + } + + if(popup != NULL) { + // force popup recreation if language changed + DestroyMenu(popup); + popup = NULL; + } + + m_menu.Attach(winResLoadMenu(MAKEINTRESOURCE(IDR_MENU))); + menu = (HMENU)m_menu; + + // don't set a menu if skin is active + if(skin == NULL) + if(m_pMainWnd) + m_pMainWnd->SetMenu(&m_menu); +} + +void winlog(const char *msg, ...) +{ + CString buffer; + va_list valist; + + va_start(valist, msg); + buffer.FormatV(msg, valist); + + if(theApp.winout == NULL) { + theApp.winout = fopen("vba-trace.log","w"); + } + + fputs(buffer, theApp.winout); + + va_end(valist); +} + +void log(const char *msg, ...) +{ + CString buffer; + va_list valist; + + va_start(valist, msg); + buffer.FormatV(msg, valist); + + toolsLog(buffer); + + va_end(valist); +} + +bool systemReadJoypads() +{ + if(theApp.input) + return theApp.input->readDevices(); + return false; +} + +u32 systemReadJoypad(int which) +{ + if(theApp.input) + return theApp.input->readDevice(which); + return 0; +} + +void systemDrawScreen() +{ + if(theApp.display == NULL) + return; + + theApp.renderedFrames++; + + if(theApp.updateCount) { + POSITION pos = theApp.updateList.GetHeadPosition(); + while(pos) { + IUpdateListener *up = theApp.updateList.GetNext(pos); + up->update(); + } + } + + if (Sm60FPS_CanSkipFrame()) + return; + + if(theApp.aviRecording && !theApp.painting) { + int width = 240; + int height = 160; + switch(theApp.cartridgeType) { + case 0: + width = 240; + height = 160; + break; + case 1: + if(gbBorderOn) { + width = 256; + height = 224; + } else { + width = 160; + height = 144; + } + break; + } + + if(theApp.aviRecorder == NULL) { + theApp.aviRecorder = new AVIWrite(); + theApp.aviFrameNumber = 0; + + theApp.aviRecorder->SetFPS(60); + + BITMAPINFOHEADER bi; + memset(&bi, 0, sizeof(bi)); + bi.biSize = 0x28; + bi.biPlanes = 1; + bi.biBitCount = 24; + bi.biWidth = width; + bi.biHeight = height; + bi.biSizeImage = 3*width*height; + theApp.aviRecorder->SetVideoFormat(&bi); + theApp.aviRecorder->Open(theApp.aviRecordName); + } + + char *bmp = new char[width*height*3]; + + utilWriteBMP(bmp, width, height, pix); + theApp.aviRecorder->AddFrame(theApp.aviFrameNumber, bmp); + + delete bmp; + } + + if( theApp.ifbFunction ) { + theApp.ifbFunction( pix + (theApp.filterWidth * (systemColorDepth>>3)) + 4, + (theApp.filterWidth * (systemColorDepth>>3)) + 4, + theApp.filterWidth, theApp.filterHeight ); + } + + if(!soundBufferLow) + theApp.display->render(); + else + soundBufferLow = false; + + Sm60FPS_Sleep(); +} + +void systemScreenCapture(int captureNumber) +{ + if(theApp.m_pMainWnd) + ((MainWnd *)theApp.m_pMainWnd)->screenCapture(captureNumber); +} + +u32 systemGetClock() +{ + return GetTickCount(); +} + +void systemMessage(int number, const char *defaultMsg, ...) +{ + CString buffer; + va_list valist; + CString msg = defaultMsg; + if(number) + msg = winResLoadString(number); + + va_start(valist, defaultMsg); + buffer.FormatV(msg, valist); + + theApp.winCheckFullscreen(); + + AfxGetApp()->m_pMainWnd->MessageBox(buffer, winResLoadString(IDS_ERROR), MB_OK|MB_ICONERROR); + + va_end(valist); +} + +void systemSetTitle(const char *title) +{ + if(theApp.m_pMainWnd != NULL) { + AfxGetApp()->m_pMainWnd->SetWindowText(title); + } +} + +void systemShowSpeed(int speed) +{ + systemSpeed = speed; + theApp.showRenderedFrames = theApp.renderedFrames; + theApp.renderedFrames = 0; + if(theApp.videoOption <= VIDEO_4X && theApp.showSpeed) { + CString buffer; + if(theApp.showSpeed == 1) + buffer.Format("VisualBoyAdvance-%3d%%", systemSpeed); + else + buffer.Format("VisualBoyAdvance-%3d%%(%d, %d fps)", systemSpeed, + systemFrameSkip, + theApp.showRenderedFrames); + + systemSetTitle(buffer); + } +} + +void systemFrame() +{ + if(theApp.aviRecording) + theApp.aviFrameNumber++; + if(theApp.movieRecording || theApp.moviePlaying) + theApp.movieFrame++; +} + +void system10Frames(int rate) +{ + u32 time = systemGetClock(); + + if (theApp.autoFrameSkip) + { + u32 diff = time - theApp.autoFrameSkipLastTime; + Sm60FPS::nCurSpeed = 100; + + if (diff) + Sm60FPS::nCurSpeed = (1000000/rate)/diff; + } + + + + if(!theApp.wasPaused && theApp.throttle) { + if(!speedup) { + u32 diff = time - theApp.throttleLastTime; + + int target = (1000000/(rate*theApp.throttle)); + int d = (target - diff); + + if(d > 0) { + Sleep(d); + } + } + theApp.throttleLastTime = systemGetClock(); + } + if(theApp.rewindMemory) { + if(++theApp.rewindCounter >= (theApp.rewindTimer)) { + theApp.rewindSaveNeeded = true; + theApp.rewindCounter = 0; + } + } + if(systemSaveUpdateCounter) { + if(--systemSaveUpdateCounter <= SYSTEM_SAVE_NOT_UPDATED) { + ((MainWnd *)theApp.m_pMainWnd)->writeBatteryFile(); + systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; + } + } + + theApp.wasPaused = false; + theApp.autoFrameSkipLastTime = time; +} + +void systemScreenMessage(const char *msg) +{ + theApp.screenMessage = true; + theApp.screenMessageTime = GetTickCount(); + theApp.screenMessageBuffer = msg; + + if(theApp.screenMessageBuffer.GetLength() > 40) + theApp.screenMessageBuffer = theApp.screenMessageBuffer.Left(40); +} + +void systemUpdateMotionSensor() +{ + if(theApp.input) + theApp.input->checkMotionKeys(); +} + +int systemGetSensorX() +{ + return theApp.sensorX; +} + +int systemGetSensorY() +{ + return theApp.sensorY; +} + +bool systemSoundInit() +{ + if(theApp.sound) + delete theApp.sound; + + theApp.sound = newDirectSound(); + return theApp.sound->init(); +} + + +void systemSoundShutdown() +{ + if(theApp.sound) + delete theApp.sound; + theApp.sound = NULL; +} + +void systemSoundPause() +{ + if(theApp.sound) + theApp.sound->pause(); +} + +void systemSoundReset() +{ + if(theApp.sound) + theApp.sound->reset(); +} + +void systemSoundResume() +{ + if(theApp.sound) + theApp.sound->resume(); +} + +void systemWriteDataToSoundBuffer() +{ + if(theApp.sound) + theApp.sound->write(); +} + +bool systemCanChangeSoundQuality() +{ + return true; +} + +bool systemPauseOnFrame() +{ + if(theApp.winPauseNextFrame) { + theApp.paused = true; + theApp.winPauseNextFrame = false; + return true; + } + return false; +} + +void systemGbBorderOn() +{ + if(emulating && theApp.cartridgeType == IMAGE_GB && gbBorderOn) { + theApp.updateWindowSize(theApp.videoOption); + } +} + +BOOL VBA::OnIdle(LONG lCount) +{ + if(emulating && debugger) { + MSG msg; + remoteStubMain(); + if(debugger) + return TRUE; // continue loop + return !::PeekMessage(&msg, NULL, NULL, NULL, PM_NOREMOVE); + } else if(emulating && active && !paused) { + for(int i = 0; i < 2; i++) { + emulator.emuMain(emulator.emuCount); + if(lanlink.connected&&linkid&&lc.numtransfers==0) lc.CheckConn(); + + if(rewindSaveNeeded && rewindMemory && emulator.emuWriteMemState) { + rewindCount++; + if(rewindCount > 8) + rewindCount = 8; + if(emulator.emuWriteMemState(&rewindMemory[rewindPos*REWIND_SIZE], + REWIND_SIZE)) { + rewindPos = ++rewindPos & 7; + if(rewindCount == 8) + rewindTopPos = ++rewindTopPos & 7; + } + } + + rewindSaveNeeded = false; + } + + if(mouseCounter) { + if(--mouseCounter == 0) { + SetCursor(NULL); + } + } + return TRUE; + } + return FALSE; + + // return CWinApp::OnIdle(lCount); +} + +void VBA::addRecentFile(CString file) +{ + // Do not change recent list if frozen + if(recentFreeze) + return; + int i = 0; + for(i = 0; i < 10; i++) { + if(recentFiles[i].GetLength() == 0) + break; + + if(recentFiles[i].Compare(file) == 0) { + if(i == 0) + return; + CString p = recentFiles[i]; + for(int j = i; j > 0; j--) { + recentFiles[j] = recentFiles[j-1]; + } + recentFiles[0] = p; + return; + } + } + int num = 0; + for(i = 0; i < 10; i++) { + if(recentFiles[i].GetLength() != 0) + num++; + } + if(num == 10) { + num--; + } + + for(i = num; i >= 1; i--) { + recentFiles[i] = recentFiles[i-1]; + } + recentFiles[0] = file; +} + +void VBA::loadSettings() +{ + CString buffer; + + languageOption = regQueryDwordValue("language", 1); + if(languageOption < 0 || languageOption > 2) + languageOption = 1; + + buffer = regQueryStringValue("languageName", ""); + if(!buffer.IsEmpty()) { + languageName = buffer.Left(3); + } else + languageName = ""; + + winSetLanguageOption(languageOption, true); + + frameSkip = regQueryDwordValue("frameSkip", 0); + if(frameSkip < 0 || frameSkip > 9) + frameSkip = 0; + + gbFrameSkip = regQueryDwordValue("gbFrameSkip", 0); + if(gbFrameSkip < 0 || gbFrameSkip > 9) + gbFrameSkip = 0; + + autoFrameSkip = regQueryDwordValue("autoFrameSkip", FALSE) ? TRUE : FALSE; + + vsync = regQueryDwordValue("vsync", false) ? true : false ; + synchronize = regQueryDwordValue("synchronize", 1) ? true : false; + fullScreenStretch = regQueryDwordValue("stretch", 0) ? true : false; + + videoOption = regQueryDwordValue("video", VIDEO_3X); + + if(videoOption < VIDEO_1X || videoOption > VIDEO_OTHER) + videoOption = VIDEO_3X; + + bool defaultVideoDriver = regQueryDwordValue("defaultVideoDriver", true) ? + true : false; + + if(!regQueryBinaryValue("videoDriverGUID", (char *)&videoDriverGUID, + sizeof(GUID))) { + defaultVideoDriver = TRUE; + } + + if(defaultVideoDriver) + pVideoDriverGUID = NULL; + else + pVideoDriverGUID = &videoDriverGUID; + + fsAdapter = regQueryDwordValue("fsAdapter", 0); + fsWidth = regQueryDwordValue("fsWidth", 0); + fsHeight = regQueryDwordValue("fsHeight", 0); + fsColorDepth = regQueryDwordValue("fsColorDepth", 0); + fsFrequency = regQueryDwordValue("fsFrequency", 60); + fsAdapter = regQueryDwordValue("fsAdapter", 0); + + if(videoOption == VIDEO_OTHER) { + if(fsWidth < 0 || fsWidth > 4095 || fsHeight < 0 || fsHeight > 4095) + videoOption = 0; + if(fsColorDepth != 16 && fsColorDepth != 24 && fsColorDepth != 32) + videoOption = 0; + } + + renderMethod = (DISPLAY_TYPE)regQueryDwordValue("renderMethod", DIRECT_DRAW); + if(renderMethod < GDI || renderMethod > OPENGL) + renderMethod = DIRECT_DRAW; + + windowPositionX = regQueryDwordValue("windowX", 0); + if(windowPositionX < 0) + windowPositionX = 0; + windowPositionY = regQueryDwordValue("windowY", 0); + if(windowPositionY < 0) + windowPositionY = 0; + + useBiosFile = regQueryDwordValue("useBios", 0) ? true: false; + + skipBiosFile = regQueryDwordValue("skipBios", 0) ? true : false; + + buffer = regQueryStringValue("biosFile", ""); + + if(!buffer.IsEmpty()) { + biosFileName = buffer; + } + + int res = regQueryDwordValue("soundEnable", 0x30f); + + soundEnable(res); + soundDisable(~res); + + soundOffFlag = (regQueryDwordValue("soundOff", 0)) ? true : false; + + soundQuality = regQueryDwordValue("soundQuality", 1); + + soundEcho = regQueryDwordValue("soundEcho", 0) ? true : false; + + soundLowPass = regQueryDwordValue("soundLowPass", 0) ? true : false; + + soundReverse = regQueryDwordValue("soundReverse", 0) ? true : false; + + soundVolume = regQueryDwordValue("soundVolume", 0); + if(soundVolume < 0 || soundVolume > 5) + soundVolume = 0; + + soundInterpolation = regQueryDwordValue("soundInterpolation", 0); + if(soundInterpolation < 0 || soundInterpolation > 4) + soundInterpolation = 0; + + ddrawEmulationOnly = regQueryDwordValue("ddrawEmulationOnly", false) ? true : false; + ddrawUseVideoMemory = regQueryDwordValue("ddrawUseVideoMemory", true) ? true : false; + tripleBuffering = regQueryDwordValue("tripleBuffering", false) ? true : false; + + d3dFilter = regQueryDwordValue("d3dFilter", 1); + if(d3dFilter < 0 || d3dFilter > 1) + d3dFilter = 1; + + glFilter = regQueryDwordValue("glFilter", 1); + if(glFilter < 0 || glFilter > 1) + glFilter = 1; + + glType = regQueryDwordValue("glType", 0); + if(glType < 0 || glType > 1) + glType = 0; + + filterType = regQueryDwordValue("filter", 0); + if(filterType < 0 || filterType > 16) + filterType = 0; + + disableMMX = regQueryDwordValue("disableMMX", false) ? true: false; + + disableStatusMessage = regQueryDwordValue("disableStatus", 0) ? true : false; + + showSpeed = regQueryDwordValue("showSpeed", 0); + if(showSpeed < 0 || showSpeed > 2) + showSpeed = 0; + + showSpeedTransparent = regQueryDwordValue("showSpeedTransparent", TRUE) ? + TRUE : FALSE; + + winGbPrinterEnabled = regQueryDwordValue("gbPrinter", false) ? true : false; + + if(winGbPrinterEnabled) + gbSerialFunction = gbPrinterSend; + else + gbSerialFunction = NULL; + + pauseWhenInactive = regQueryDwordValue("pauseWhenInactive", 1) ? + true : false; + + useOldSync = regQueryDwordValue("useOldSync", 0) ? + TRUE : FALSE; + + captureFormat = regQueryDwordValue("captureFormat", 0); + + removeIntros = regQueryDwordValue("removeIntros", false) ? true : false; + + recentFreeze = regQueryDwordValue("recentFreeze", false) ? true : false; + + autoIPS = regQueryDwordValue("autoIPS", true) ? true : false; + + cpuDisableSfx = regQueryDwordValue("disableSfx", 0) ? true : false; + + winSaveType = regQueryDwordValue("saveType", 0); + if(winSaveType < 0 || winSaveType > 5) + winSaveType = 0; + + ifbType = regQueryDwordValue("ifbType", 0); + if(ifbType < 0 || ifbType > 2) + ifbType = 0; + + winFlashSize = regQueryDwordValue("flashSize", 0x10000); + if(winFlashSize != 0x10000 && winFlashSize != 0x20000) + winFlashSize = 0x10000; + flashSize = winFlashSize; + + agbPrintEnable(regQueryDwordValue("agbPrint", 0) ? true : false); + + winRtcEnable = regQueryDwordValue("rtcEnabled", 0) ? true : false; + rtcEnable(winRtcEnable); + + autoHideMenu = regQueryDwordValue("autoHideMenu", 0) ? true : false; + + skinEnabled = regQueryDwordValue("skinEnabled", 0) ? true : false; + + skinName = regQueryStringValue("skinName", ""); + + switch(videoOption) { + case VIDEO_320x240: + fsWidth = 320; + fsHeight = 240; + fsColorDepth = 16; + fsFrequency = 60; + break; + case VIDEO_640x480: + fsWidth = 640; + fsHeight = 480; + fsColorDepth = 16; + fsFrequency = 60; + break; + case VIDEO_800x600: + fsWidth = 800; + fsHeight = 600; + fsColorDepth = 16; + fsFrequency = 60; + break; + case VIDEO_1024x768: + fsWidth = 1024; + fsHeight = 768; + fsColorDepth = 16; + fsFrequency = 60; + break; + case VIDEO_1280x1024: + fsWidth = 1280; + fsHeight = 1024; + fsColorDepth = 16; + fsFrequency = 60; + break; + } + + winGbBorderOn = regQueryDwordValue("borderOn", 0); + gbBorderAutomatic = regQueryDwordValue("borderAutomatic", 0); + gbEmulatorType = regQueryDwordValue("emulatorType", 1); + if(gbEmulatorType < 0 || gbEmulatorType > 5) + gbEmulatorType = 1; + gbColorOption = regQueryDwordValue("colorOption", 0); + + threadPriority = regQueryDwordValue("priority", 2); + + if(threadPriority < 0 || threadPriority >3) + threadPriority = 2; + updatePriority(); + + autoSaveLoadCheatList = regQueryDwordValue("autoSaveCheatList", 0) ? + true : false; + + gbPaletteOption = regQueryDwordValue("gbPaletteOption", 0); + if(gbPaletteOption < 0) + gbPaletteOption = 0; + if(gbPaletteOption > 2) + gbPaletteOption = 2; + + regQueryBinaryValue("gbPalette", (char *)systemGbPalette, + 24*sizeof(u16)); + + rewindTimer = regQueryDwordValue("rewindTimer", 0); + + if(rewindTimer < 0 || rewindTimer > 600) + rewindTimer = 0; + + rewindTimer *= 6; // convert to 10 frames multiple + + if(rewindTimer != 0) + rewindMemory = (char *)malloc(8*REWIND_SIZE); + + for(int i = 0; i < 10; i++) { + buffer.Format("recent%d", i); + char *s = regQueryStringValue(buffer, NULL); + if(s == NULL) + break; + recentFiles[i] = s; + } + + joypadDefault = regQueryDwordValue("joypadDefault", 0); + if(joypadDefault < 0 || joypadDefault > 3) + joypadDefault = 0; + + autoLoadMostRecent = regQueryDwordValue("autoLoadMostRecent", false) ? true : + false; + + cheatsEnabled = regQueryDwordValue("cheatsEnabled", true) ? true : false; + + fsMaxScale = regQueryDwordValue("fsMaxScale", 0); + + throttle = regQueryDwordValue("throttle", 0); + if(throttle < 5 || throttle > 1000) + throttle = 0; + + linktimeout = regQueryDwordValue("LinkTimeout", 1000); + + linklog = regQueryDwordValue("Linklog", false) ? true : false; + if(linklog) + openLinkLog(); + + adapter = regQueryDwordValue("RFU", false) ? true : false; + + lanlink.active = regQueryDwordValue("LAN", 0) ? true : false; + if (autoFrameSkip) + { + throttle = 0; + frameSkip = 0; + systemFrameSkip = 0; + } + + Sm60FPS::bSaveMoreCPU = regQueryDwordValue("saveMoreCPU", 0); +} + +void VBA::updateFrameSkip() +{ + switch(cartridgeType) { + case 0: + systemFrameSkip = frameSkip; + break; + case 1: + systemFrameSkip = gbFrameSkip; + break; + } +} + +void VBA::updateVideoSize(UINT id) +{ + int value = 0; + + switch(id) { + case ID_OPTIONS_VIDEO_X1: + value = VIDEO_1X; + break; + case ID_OPTIONS_VIDEO_X2: + value = VIDEO_2X; + break; + case ID_OPTIONS_VIDEO_X3: + value = VIDEO_3X; + break; + case ID_OPTIONS_VIDEO_X4: + value = VIDEO_4X; + break; + case ID_OPTIONS_VIDEO_FULLSCREEN320X240: + value = VIDEO_320x240; + fsWidth = 320; + fsHeight = 240; + fsColorDepth = 16; + break; + case ID_OPTIONS_VIDEO_FULLSCREEN640X480: + value = VIDEO_640x480; + fsWidth = 640; + fsHeight = 480; + fsColorDepth = 16; + break; + case ID_OPTIONS_VIDEO_FULLSCREEN800X600: + value = VIDEO_800x600; + fsWidth = 800; + fsHeight = 600; + fsColorDepth = 16; + break; + case ID_OPTIONS_VIDEO_FULLSCREEN1024X768: + value = VIDEO_1024x768; + fsWidth = 1024; + fsHeight = 768; + fsColorDepth = 32; + break; + case ID_OPTIONS_VIDEO_FULLSCREEN1280X1024: + value = VIDEO_1280x1024; + fsWidth = 1280; + fsHeight = 1024; + fsColorDepth = 32; + break; + case ID_OPTIONS_VIDEO_FULLSCREEN: + value = VIDEO_OTHER; + break; + } + + updateWindowSize(value); +} + +typedef BOOL (WINAPI *GETMENUBARINFO)(HWND, LONG, LONG, PMENUBARINFO); + +static void winCheckMenuBarInfo(int& winSizeX, int& winSizeY) +{ + HINSTANCE hinstDll; + DWORD dwVersion = 0; + +#ifdef _AFXDLL + hinstDll = AfxLoadLibrary("user32.dll"); +#else + hinstDll = LoadLibrary( _T("user32.dll") ); +#endif + + if(hinstDll) { + GETMENUBARINFO func = (GETMENUBARINFO)GetProcAddress(hinstDll, "GetMenuBarInfo"); + + if(func) { + MENUBARINFO info; + info.cbSize = sizeof(MENUBARINFO); + + func(AfxGetMainWnd()->GetSafeHwnd(), OBJID_MENU, 0, &info); + + int menuHeight = GetSystemMetrics(SM_CYMENU); + + if((info.rcBar.bottom - info.rcBar.top) > menuHeight) { + winSizeY += (info.rcBar.bottom - info.rcBar.top) - menuHeight + 1; + theApp.m_pMainWnd->SetWindowPos( + 0, //HWND_TOPMOST, + theApp.windowPositionX, + theApp.windowPositionY, + winSizeX, + winSizeY, + SWP_NOMOVE | SWP_SHOWWINDOW); + } + } +#ifdef _AFXDLL + AfxFreeLibrary( hinstDll ); +#else + FreeLibrary( hinstDll ); +#endif + } +} + +void VBA::updateWindowSize(int value) +{ + regSetDwordValue("video", value); + + if(value == VIDEO_OTHER) { + regSetDwordValue("fsWidth", fsWidth); + regSetDwordValue("fsHeight", fsHeight); + regSetDwordValue("fsColorDepth", fsColorDepth); + } + + if(((value >= VIDEO_320x240) && + (videoOption != value)) || + (videoOption >= VIDEO_320x240 && + value <= VIDEO_4X) || + fsForceChange) { + fsForceChange = false; + changingVideoSize = true; + shutdownDisplay(); + if(input) { + delete input; + input = NULL; + } + m_pMainWnd->DragAcceptFiles(FALSE); + CWnd *pWnd = m_pMainWnd; + m_pMainWnd = NULL; + pWnd->DestroyWindow(); + delete pWnd; + videoOption = value; + if(!initDisplay()) { + if(videoOption == VIDEO_320x240 || + videoOption == VIDEO_640x480 || + videoOption == VIDEO_800x600 || + videoOption == VIDEO_1024x768 || + videoOption == VIDEO_1280x1024 || + videoOption == VIDEO_OTHER) { + regSetDwordValue("video", VIDEO_1X); + if(pVideoDriverGUID) + regSetDwordValue("defaultVideoDriver", TRUE); + } + changingVideoSize = false; + AfxPostQuitMessage(0); + return; + } + if(!initInput()) { + changingVideoSize = false; + AfxPostQuitMessage(0); + return; + } + input->checkKeys(); + updateMenuBar(); + changingVideoSize = FALSE; + updateWindowSize(videoOption); + return; + } + + sizeX = 240; + sizeY = 160; + + videoOption = value; + + if(cartridgeType == IMAGE_GB) { + if(gbBorderOn) { + sizeX = 256; + sizeY = 224; + gbBorderLineSkip = 256; + gbBorderColumnSkip = 48; + gbBorderRowSkip = 40; + } else { + sizeX = 160; + sizeY = 144; + gbBorderLineSkip = 160; + gbBorderColumnSkip = 0; + gbBorderRowSkip = 0; + } + } + + surfaceSizeX = sizeX; + surfaceSizeY = sizeY; + + switch(videoOption) { + case VIDEO_1X: + surfaceSizeX = sizeX; + surfaceSizeY = sizeY; + break; + case VIDEO_2X: + surfaceSizeX = sizeX * 2; + surfaceSizeY = sizeY * 2; + break; + case VIDEO_3X: + surfaceSizeX = sizeX * 3; + surfaceSizeY = sizeY * 3; + break; + case VIDEO_4X: + surfaceSizeX = sizeX * 4; + surfaceSizeY = sizeY * 4; + break; + case VIDEO_320x240: + case VIDEO_640x480: + case VIDEO_800x600: + case VIDEO_1024x768: + case VIDEO_1280x1024: + case VIDEO_OTHER: + { + int scaleX = 1; + int scaleY = 1; + scaleX = (fsWidth / sizeX); + scaleY = (fsHeight / sizeY); + int min = scaleX < scaleY ? scaleX : scaleY; + if(fsMaxScale) + min = min > fsMaxScale ? fsMaxScale : min; + surfaceSizeX = min * sizeX; + surfaceSizeY = min * sizeY; + if((fullScreenStretch && (display != NULL && + (display->getType() != DIRECT_3D))) + || (display != NULL && display->getType() >= DIRECT_3D)) { + surfaceSizeX = fsWidth; + surfaceSizeY = fsHeight; + } + } + break; + } + + rect.right = sizeX; + rect.bottom = sizeY; + + int winSizeX = sizeX; + int winSizeY = sizeY; + + if(videoOption <= VIDEO_4X) { + dest.left = 0; + dest.top = 0; + dest.right = surfaceSizeX; + dest.bottom = surfaceSizeY; + + DWORD style = WS_POPUP | WS_VISIBLE; + + style |= WS_OVERLAPPEDWINDOW; + + menuToggle = TRUE; + AdjustWindowRectEx(&dest, style, TRUE, 0); //WS_EX_TOPMOST); + + winSizeX = dest.right-dest.left; + winSizeY = dest.bottom-dest.top; + + if(skin == NULL) { + m_pMainWnd->SetWindowPos(0, //HWND_TOPMOST, + windowPositionX, + windowPositionY, + winSizeX, + winSizeY, + SWP_NOMOVE | SWP_SHOWWINDOW); + + winCheckMenuBarInfo(winSizeX, winSizeY); + } + } + + adjustDestRect(); + + updateIFB(); + updateFilter(); + + if(display) + display->resize(theApp.dest.right-theApp.dest.left, theApp.dest.bottom-theApp.dest.top); + + m_pMainWnd->RedrawWindow(NULL,NULL,RDW_INVALIDATE|RDW_ERASE|RDW_ALLCHILDREN); +} + +bool VBA::initDisplay() +{ + return updateRenderMethod(false); +} + +bool VBA::updateRenderMethod(bool force) +{ + Sm60FPS_Init(); + bool res = updateRenderMethod0(force); + + while(!res && renderMethod > 0) { + if( fsAdapter > 0 ) { + fsAdapter = 0; + } else { + if( videoOption > VIDEO_4X ) { + videoOption = VIDEO_1X; + force = true; + } else { + if(renderMethod == OPENGL) { + renderMethod = DIRECT_3D; + } else { + if(renderMethod == DIRECT_3D) { + renderMethod = DIRECT_DRAW; + } else { + if(renderMethod == DIRECT_DRAW) { + renderMethod = GDI; + } + } + } + } + } + res = updateRenderMethod(force); + } + + return res; +} + + +bool VBA::updateRenderMethod0(bool force) +{ + bool initInput = false; + b16to32Video = false; + + if(display) { + if(display->getType() != renderMethod || force) { + if(skin) { + delete skin; + skin = NULL; + } + initInput = true; + changingVideoSize = true; + shutdownDisplay(); + if(input) { + delete input; + input = NULL; + } + CWnd *pWnd = m_pMainWnd; + + m_pMainWnd = NULL; + pWnd->DragAcceptFiles(FALSE); + pWnd->DestroyWindow(); + delete pWnd; + + display = NULL; + regSetDwordValue("renderMethod", renderMethod); + } + } + if(display == NULL) { + switch(renderMethod) { + case GDI: + display = newGDIDisplay(); + break; + case DIRECT_DRAW: + pVideoDriverGUID = NULL; + ZeroMemory( &videoDriverGUID, sizeof( GUID ) ); + display = newDirectDrawDisplay(); + break; + case DIRECT_3D: + display = newDirect3DDisplay(); + break; + case OPENGL: + display = newOpenGLDisplay(); + break; + } + + if(display->initialize()) { + winUpdateSkin(); + if(initInput) { + if(!this->initInput()) { + changingVideoSize = false; + AfxPostQuitMessage(0); + return false; + } + input->checkKeys(); + updateMenuBar(); + changingVideoSize = false; + updateWindowSize(videoOption); + + m_pMainWnd->ShowWindow(SW_SHOW); + m_pMainWnd->UpdateWindow(); + m_pMainWnd->SetFocus(); + + return true; + } else { + changingVideoSize = false; + return true; + } + } + changingVideoSize = false; + } + return true; +} + + +void VBA::winCheckFullscreen() +{ + if(videoOption > VIDEO_4X && tripleBuffering) { + if(display) { + display->checkFullScreen(); + } + } +} + + +void VBA::shutdownDisplay() +{ + if(display != NULL) { + display->cleanup(); + delete display; + display = NULL; + } +} + +void VBA::directXMessage(const char *msg) +{ + systemMessage(IDS_DIRECTX_7_REQUIRED, + "DirectX 7.0 or greater is required to run.\nDownload at http://www.microsoft.com/directx.\n\nError found at: %s", + msg); +} + +void VBA::winUpdateSkin() +{ +#ifndef NOSKINS + skinButtons = 0; + if(skin) { + delete skin; + skin = NULL; + } + + if(!skinName.IsEmpty() && skinEnabled && display->isSkinSupported()) { + skin = new CSkin(); + if(skin->Initialize(skinName)) { + skin->Hook(m_pMainWnd); + skin->Enable(true); + } else { + delete skin; + skin = NULL; + } + } + + if(!skin) { + adjustDestRect(); + updateMenuBar(); + } +#endif +} + +void VBA::updatePriority() +{ + switch(threadPriority) { + case 0: + SetThreadPriority(THREAD_PRIORITY_HIGHEST); + break; + case 1: + SetThreadPriority(THREAD_PRIORITY_ABOVE_NORMAL); + break; + case 3: + SetThreadPriority(THREAD_PRIORITY_BELOW_NORMAL); + break; + default: + SetThreadPriority(THREAD_PRIORITY_NORMAL); + } +} + +#ifdef MMX +bool VBA::detectMMX() +{ + bool support = false; + char brand[13]; + + // check for Intel chip + __try { + __asm { + mov eax, 0; + cpuid; + mov [dword ptr brand+0], ebx; + mov [dword ptr brand+4], edx; + mov [dword ptr brand+8], ecx; + } + } + __except(EXCEPTION_EXECUTE_HANDLER) { + if(_exception_code() == STATUS_ILLEGAL_INSTRUCTION) { + return false; + } + return false; + } + // Check for Intel or AMD CPUs + if(strncmp(brand, "GenuineIntel", 12)) { + if(strncmp(brand, "AuthenticAMD", 12)) { + return false; + } + } + + __asm { + mov eax, 1; + cpuid; + test edx, 00800000h; + jz NotFound; + mov [support], 1; + NotFound: + } + return support; +} +#endif + +void VBA::winSetLanguageOption(int option, bool force) +{ + if(((option == languageOption) && option != 2) && !force) + return; + switch(option) { + case 0: + { + char lbuffer[10]; + + if(GetLocaleInfo(LOCALE_SYSTEM_DEFAULT, LOCALE_SABBREVLANGNAME, + lbuffer, 10)) { + HINSTANCE l = winLoadLanguage(lbuffer); + if(l == NULL) { + LCID locIdBase = MAKELCID( MAKELANGID( PRIMARYLANGID( GetSystemDefaultLangID() ), SUBLANG_NEUTRAL ), SORT_DEFAULT ); + if(GetLocaleInfo(locIdBase, LOCALE_SABBREVLANGNAME, + lbuffer, 10)) { + l = winLoadLanguage(lbuffer); + if(l == NULL) { + systemMessage(IDS_FAILED_TO_LOAD_LIBRARY, + "Failed to load library %s", + lbuffer); + return; + } + } + } + AfxSetResourceHandle(l); + if(languageModule != NULL) +#ifdef _AFXDLL + AfxFreeLibrary( languageModule ); +#else + FreeLibrary( languageModule ); +#endif + languageModule = l; + } else { + systemMessage(IDS_FAILED_TO_GET_LOCINFO, + "Failed to get locale information"); + return; + } + } + break; + case 1: + if(languageModule != NULL) +#ifdef _AFXDLL + AfxFreeLibrary( languageModule ); +#else + FreeLibrary( languageModule ); +#endif + languageModule = NULL; + AfxSetResourceHandle(AfxGetInstanceHandle()); + break; + case 2: + { + if(!force) { + LangSelect dlg; + if(dlg.DoModal()) { + HINSTANCE l = winLoadLanguage(languageName); + if(l == NULL) { + systemMessage(IDS_FAILED_TO_LOAD_LIBRARY, + "Failed to load library %s", + languageName); + return; + } + AfxSetResourceHandle(l); + if(languageModule != NULL) + { +#ifdef _AFXDLL + AfxFreeLibrary( languageModule ); +#else + FreeLibrary( languageModule ); +#endif + } + languageModule = l; + } + } else { + if(languageName.IsEmpty()) + return; + HINSTANCE l = winLoadLanguage(languageName); + if(l == NULL) { + systemMessage(IDS_FAILED_TO_LOAD_LIBRARY, + "Failed to load library %s", + languageName); + return; + } + AfxSetResourceHandle(l); + if(languageModule != NULL) + FreeLibrary(languageModule); + languageModule = l; + } + } + break; + } + languageOption = option; + updateMenuBar(); +} + +HINSTANCE VBA::winLoadLanguage(const char *name) +{ + CString buffer; + + buffer.Format( _T("vba_%s.dll"), name); + +#ifdef _AFXDLL + HINSTANCE l = AfxLoadLibrary( buffer ); +#else + HMODULE l = LoadLibrary( buffer ); +#endif + + if(l == NULL) { + if(strlen(name) == 3) { + char buffer2[3]; + buffer2[0] = name[0]; + buffer2[1] = name[1]; + buffer2[2] = 0; + buffer.Format("vba_%s.dll", buffer2); + +#ifdef _AFXDLL + return AfxLoadLibrary( buffer ); +#else + return LoadLibrary( buffer ); +#endif + } + } + return l; +} + + +bool VBA::initInput() +{ + if(input) + delete input; + input = newDirectInput(); + if(input->initialize()) { + input->loadSettings(); + input->checkKeys(); + return true; + } + delete input; + return false; +} + +void VBA::winAddUpdateListener(IUpdateListener *l) +{ + updateList.AddTail(l); + updateCount++; +} + +void VBA::winRemoveUpdateListener(IUpdateListener *l) +{ + POSITION pos = updateList.Find(l); + if(pos) { + updateList.RemoveAt(pos); + updateCount--; + if(updateCount < 0) + updateCount = 0; + } +} + +CString VBA::winLoadFilter(UINT id) +{ + CString res = winResLoadString(id); + res.Replace('_','|'); + + return res; +} + +void VBA::movieReadNext() +{ + if(movieFile) { + bool movieEnd = false; + + if(fread(&moviePlayFrame, 1, sizeof(int), movieFile) == sizeof(int)) { + if(fread(&movieNextJoypad, 1, sizeof(u32), movieFile) == sizeof(int)) { + // make sure we don't have spurious entries on the movie that can + // cause us to play it forever + if(moviePlayFrame <= movieFrame) + movieEnd = true; + } else + movieEnd = true; + } else + movieEnd = true; + if(movieEnd) { + CString string = winResLoadString(IDS_END_OF_MOVIE); + systemScreenMessage(string); + moviePlaying = false; + fclose(movieFile); + movieFile = NULL; + return; + } + } else + moviePlaying = false; +} + +void VBA::saveSettings() +{ + regSetDwordValue("language", languageOption); + + regSetStringValue("languageName", languageName); + + regSetDwordValue("frameSkip", frameSkip); + + regSetDwordValue("gbFrameSkip", gbFrameSkip); + + regSetDwordValue("autoFrameSkip", autoFrameSkip); + + regSetDwordValue("vsync", vsync); + regSetDwordValue("synchronize", synchronize); + regSetDwordValue("stretch", fullScreenStretch); + + regSetDwordValue("video", videoOption); + + regSetDwordValue("defaultVideoDriver", pVideoDriverGUID == NULL); + + if(pVideoDriverGUID) { + regSetBinaryValue("videoDriverGUID", (char *)&videoDriverGUID, + sizeof(GUID)); + } + + + regSetDwordValue("fsAdapter", fsAdapter); + regSetDwordValue("fsWidth", fsWidth); + regSetDwordValue("fsHeight", fsHeight); + regSetDwordValue("fsColorDepth", fsColorDepth); + regSetDwordValue("fsFrequency", fsFrequency); + + regSetDwordValue("renderMethod", renderMethod); + + regSetDwordValue("windowX", windowPositionX); + regSetDwordValue("windowY", windowPositionY); + + regSetDwordValue("useBios", useBiosFile); + + regSetDwordValue("skipBios", skipBiosFile); + + if(!biosFileName.IsEmpty()) + regSetStringValue("biosFile", biosFileName); + + regSetDwordValue("soundEnable", soundGetEnable() & 0x30f); + + regSetDwordValue("soundOff", soundOffFlag); + + regSetDwordValue("soundQuality", soundQuality); + + regSetDwordValue("soundEcho", soundEcho); + + regSetDwordValue("soundLowPass", soundLowPass); + + regSetDwordValue("soundReverse", soundReverse); + + regSetDwordValue("soundVolume", soundVolume); + + regSetDwordValue("soundInterpolation", soundInterpolation); + + regSetDwordValue("ddrawEmulationOnly", ddrawEmulationOnly); + regSetDwordValue("ddrawUseVideoMemory", ddrawUseVideoMemory); + regSetDwordValue("tripleBuffering", tripleBuffering); + + regSetDwordValue("d3dFilter", d3dFilter); + regSetDwordValue("glFilter", glFilter); + regSetDwordValue("glType", glType); + + regSetDwordValue("filter", filterType); + + regSetDwordValue("LCDFilter", filterLCD); + + regSetDwordValue("disableMMX", disableMMX); + + regSetDwordValue("disableStatus", disableStatusMessage); + + regSetDwordValue("showSpeed", showSpeed); + + regSetDwordValue("showSpeedTransparent", showSpeedTransparent); + + regSetDwordValue("gbPrinter", winGbPrinterEnabled); + + regSetDwordValue("pauseWhenInactive", pauseWhenInactive); + + regSetDwordValue("useOldSync", useOldSync); + + regSetDwordValue("captureFormat", captureFormat); + + regSetDwordValue("removeIntros", removeIntros); + + regSetDwordValue("recentFreeze", recentFreeze); + + regSetDwordValue("autoIPS", autoIPS); + + regSetDwordValue("disableSfx", cpuDisableSfx); + + regSetDwordValue("saveType", winSaveType); + + regSetDwordValue("ifbType", ifbType); + + regSetDwordValue("flashSize", winFlashSize); + + regSetDwordValue("agbPrint", agbPrintIsEnabled()); + + regSetDwordValue("rtcEnabled", winRtcEnable); + + regSetDwordValue("autoHideMenu", autoHideMenu); + + regSetDwordValue("skinEnabled", skinEnabled); + + regSetStringValue("skinName", skinName); + + regSetDwordValue("borderOn", winGbBorderOn); + regSetDwordValue("borderAutomatic", gbBorderAutomatic); + regSetDwordValue("emulatorType", gbEmulatorType); + regSetDwordValue("colorOption", gbColorOption); + + regSetDwordValue("priority", threadPriority); + + regSetDwordValue("autoSaveCheatList", autoSaveLoadCheatList); + + regSetDwordValue("gbPaletteOption", gbPaletteOption); + + regSetBinaryValue("gbPalette", (char *)systemGbPalette, + 24*sizeof(u16)); + + regSetDwordValue("rewindTimer", rewindTimer/6); + + CString buffer; + for(int i = 0; i < 10; i++) { + buffer.Format("recent%d", i); + regSetStringValue(buffer, recentFiles[i]); + } + + regSetDwordValue("joypadDefault", joypadDefault); + regSetDwordValue("autoLoadMostRecent", autoLoadMostRecent); + regSetDwordValue("cheatsEnabled", cheatsEnabled); + regSetDwordValue("fsMaxScale", fsMaxScale); + regSetDwordValue("throttle", throttle); + + regSetDwordValue("saveMoreCPU", Sm60FPS::bSaveMoreCPU); + regSetDwordValue("LinkTimeout", linktimeout); + regSetDwordValue("Linklog", linklog); + regSetDwordValue("RFU", adapter); +} + +void winSignal(int, int) +{ +} + +#define CPUReadByteQuick(addr) \ + map[(addr)>>24].address[(addr) & map[(addr)>>24].mask] + +void winOutput(char *s, u32 addr) +{ + if(s) { + toolsLog(s); + } else { + CString str; + char c; + + c = CPUReadByteQuick(addr); + addr++; + while(c) { + str += c; + c = CPUReadByteQuick(addr); + addr++; + } + toolsLog(str); + } +} + + +void Sm60FPS_Init() +{ + Sm60FPS::dwTimeElapse = 0; + Sm60FPS::fWantFPS = 60.f; + Sm60FPS::fCurFPS = 0.f; + Sm60FPS::nFrameCnt = 0; + Sm60FPS::bLastSkip = false; + Sm60FPS::nCurSpeed = 100; +} + + +bool Sm60FPS_CanSkipFrame() +{ + if( theApp.autoFrameSkip ) { + if( Sm60FPS::nFrameCnt == 0 ) { + Sm60FPS::nFrameCnt = 0; + Sm60FPS::dwTimeElapse = 0; + Sm60FPS::dwTime0 = timeGetTime(); + } else { + if( Sm60FPS::nFrameCnt >= 10 ) { + Sm60FPS::nFrameCnt = 0; + Sm60FPS::dwTimeElapse = 0; + + if( Sm60FPS::nCurSpeed > Sm60FPS::K_fCpuSpeed ) { + Sm60FPS::fWantFPS += 1; + if( Sm60FPS::fWantFPS > Sm60FPS::K_fTargetFps ){ + Sm60FPS::fWantFPS = Sm60FPS::K_fTargetFps; + } + } else { + if( Sm60FPS::nCurSpeed < (Sm60FPS::K_fCpuSpeed - 5) ) { + Sm60FPS::fWantFPS -= 1; + if( Sm60FPS::fWantFPS < 30.f ) { + Sm60FPS::fWantFPS = 30.f; + } + } + } + } else { // between frame 1-10 + Sm60FPS::dwTime1 = timeGetTime(); + Sm60FPS::dwTimeElapse += (Sm60FPS::dwTime1 - Sm60FPS::dwTime0); + Sm60FPS::dwTime0 = Sm60FPS::dwTime1; + if( !Sm60FPS::bLastSkip && + ( (Sm60FPS::fWantFPS < Sm60FPS::K_fTargetFps) || Sm60FPS::bSaveMoreCPU) ) { + Sm60FPS::fCurFPS = (float)Sm60FPS::nFrameCnt * 1000 / Sm60FPS::dwTimeElapse; + if( (Sm60FPS::fCurFPS < Sm60FPS::K_fTargetFps) || Sm60FPS::bSaveMoreCPU ) { + Sm60FPS::bLastSkip = true; + Sm60FPS::nFrameCnt++; + return true; + } + } + } + } + Sm60FPS::bLastSkip = false; + Sm60FPS::nFrameCnt++; + } + return false; +} + + +void Sm60FPS_Sleep() +{ + if( theApp.autoFrameSkip ) { + u32 dwTimePass = Sm60FPS::dwTimeElapse + (timeGetTime() - Sm60FPS::dwTime0); + u32 dwTimeShould = (u32)(Sm60FPS::nFrameCnt * Sm60FPS::K_fDT); + if( dwTimeShould > dwTimePass ) { + Sleep(dwTimeShould - dwTimePass); + } + } +} diff --git a/src/win32/VBA.h b/src/win32/VBA.h index ee1916ca..1a7ba1fd 100644 --- a/src/win32/VBA.h +++ b/src/win32/VBA.h @@ -1,280 +1,279 @@ -// -*- C++ -*- -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -// VBA.h : main header file for the VBA application -// - -#if !defined(AFX_VBA_H__57514A10_49F9_4B83_A928_0D8A4A7306A3__INCLUDED_) -#define AFX_VBA_H__57514A10_49F9_4B83_A928_0D8A4A7306A3__INCLUDED_ - -#if _MSC_VER > 1000 -#pragma once -#endif // _MSC_VER > 1000 - -#ifndef __AFXWIN_H__ -#error include 'stdafx.h' before including this file for PCH -#endif - -#include - -#include "AcceleratorManager.h" -#include "..\..\res\resource.h" // main symbols -#include "Display.h" -#include "Input.h" -#include "IUpdate.h" -#include "Sound.h" -#include "../System.h" - -///////////////////////////////////////////////////////////////////////////// -// VBA: -// See VBA.cpp for the implementation of this class -// - -enum { - VIDEO_1X, VIDEO_2X, VIDEO_3X, VIDEO_4X, - VIDEO_320x240, VIDEO_640x480, VIDEO_800x600, VIDEO_1024x768, VIDEO_1280x1024, - VIDEO_OTHER -}; - -enum pixelFilterType -{ - FILTER_NONE, - - FILTER_SIMPLE2X, FILTER_PIXELATE, FILTER_TVMODE, FILTER_SCANLINES, - FILTER_BILINEAR, FILTER_BILINEARPLUS, FILTER_MAMESCALE2X, - FILTER_2XSAI, FILTER_SUPER2XSAI, FILTER_SUPEREAGLE, FILTER_LQ2X, FILTER_HQ2X, - - FILTER_SIMPLE3X, FILTER_HQ3X, - - FILTER_SIMPLE4X, FILTER_HQ4X -}; - -#define REWIND_SIZE 400000 - -class AVIWrite; -class WavWriter; - -class VBA : public CWinApp -{ - public: - CMenu m_menu; - HMENU menu; - HMENU popup; - bool mode320Available; - bool mode640Available; - bool mode800Available; - bool mode1024Available; - bool mode1280Available; - int windowPositionX; - int windowPositionY; - void (*filterFunction)(u8*,u32,u8*,u8*,u32,int,int); - void (*ifbFunction)(u8*,u32,int,int); - int ifbType; - int filterType; - int filterWidth; - int filterHeight; - int filterMagnification; - int filterLCD; - int fsWidth; - int fsHeight; - int fsColorDepth; - int fsFrequency; - int fsAdapter; - bool fsForceChange; - int sizeX; - int sizeY; - int surfaceSizeX; - int surfaceSizeY; - int videoOption; - bool fullScreenStretch; - bool disableStatusMessage; - int showSpeed; - BOOL showSpeedTransparent; - int showRenderedFrames; - bool screenMessage; - CString screenMessageBuffer; - DWORD screenMessageTime; - u8 *delta[257*244*4]; - bool menuToggle; - IDisplay *display; - int cartridgeType; - bool soundInitialized; - bool useBiosFile; - bool skipBiosFile; - CString biosFileName; - bool active; - bool paused; - CString recentFiles[10]; - bool recentFreeze; - bool autoSaveLoadCheatList; - FILE *winout; - bool removeIntros; - bool autoIPS; - int winGbBorderOn; - int winFlashSize; - bool winRtcEnable; - int winSaveType; - char *rewindMemory; - int rewindPos; - int rewindTopPos; - int rewindCounter; - int rewindCount; - bool rewindSaveNeeded; - int rewindTimer; - int captureFormat; - bool tripleBuffering; - bool autoHideMenu; - int throttle; - u32 throttleLastTime; - u32 autoFrameSkipLastTime; - bool autoFrameSkip; - bool vsync; - bool changingVideoSize; - GUID videoDriverGUID; - GUID *pVideoDriverGUID; - DISPLAY_TYPE renderMethod; - bool iconic; - bool ddrawEmulationOnly; - bool ddrawUsingEmulationOnly; - bool ddrawDebug; - bool ddrawUseVideoMemory; - int d3dFilter; - int glFilter; - int glType; - bool dinputKeyFocus; - bool pauseWhenInactive; - bool speedupToggle; - bool useOldSync; - bool winGbPrinterEnabled; - int threadPriority; - bool disableMMX; - int languageOption; - CString languageName; - HINSTANCE languageModule; - int renderedFrames; - Input *input; - int joypadDefault; - int autoFire; - bool autoFireToggle; - bool winPauseNextFrame; - bool soundRecording; - WavWriter *soundRecorder; - CString soundRecordName; - ISound *sound; - bool aviRecording; - AVIWrite *aviRecorder; - CString aviRecordName; - int aviFrameNumber; - bool painting; - bool movieRecording; - bool moviePlaying; - int movieFrame; - int moviePlayFrame; - FILE *movieFile; - u32 movieLastJoypad; - u32 movieNextJoypad; - int sensorX; - int sensorY; - int mouseCounter; - bool wasPaused; - int frameskipadjust; - bool autoLoadMostRecent; - int fsMaxScale; - int romSize; - - CList updateList; - int updateCount; - - CAcceleratorManager winAccelMgr; - HACCEL hAccel; - - RECT rect; - RECT dest; - - struct EmulatedSystem emulator; - - CString szFile; - CString filename; - CString dir; - - CString wndClass; - - public: - VBA(); - ~VBA(); - - void adjustDestRect(); - void updateIFB(); - void updateFilter(); - void updateMenuBar(); - void winAddUpdateListener(IUpdateListener *l); - void winRemoveUpdateListener(IUpdateListener *l); - CString winLoadFilter(UINT id); - - // Overrides - // ClassWizard generated virtual function overrides - //{{AFX_VIRTUAL(VBA) - public: - virtual BOOL InitInstance(); - virtual BOOL OnIdle(LONG lCount); - //}}AFX_VIRTUAL - - // Implementation - - public: - void saveSettings(); - void movieReadNext(); - bool initInput(); - HINSTANCE winLoadLanguage(const char *name); - void winSetLanguageOption(int option, bool force); - bool detectMMX(); -#ifdef MMX -#endif - void updatePriority(); - void directXMessage(const char *msg); - void shutdownDisplay(); - void winCheckFullscreen(); - bool updateRenderMethod0(bool force); - bool updateRenderMethod(bool force); - bool initDisplay(); - void updateWindowSize(int value); - void updateVideoSize(UINT id); - void updateFrameSkip(); - void loadSettings(); - void addRecentFile(CString file); - //{{AFX_MSG(VBA) - afx_msg void OnAppAbout(); - // NOTE - the ClassWizard will add and remove member functions here. - // DO NOT EDIT what you see in these blocks of generated code ! - //}}AFX_MSG - DECLARE_MESSAGE_MAP() - }; - - extern VBA theApp; - -#ifdef MMX - extern "C" bool cpu_mmx; -#endif - - ///////////////////////////////////////////////////////////////////////////// - - //{{AFX_INSERT_LOCATION}} - // Microsoft Visual C++ will insert additional declarations immediately before the previous line. - -#endif // !defined(AFX_VBA_H__57514A10_49F9_4B83_A928_0D8A4A7306A3__INCLUDED_) +// -*- C++ -*- +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2005 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +// VBA.h : main header file for the VBA application +// + +#pragma once + +#ifndef __AFXWIN_H__ +#error include 'stdafx.h' before including this file for PCH +#endif + +#include "stdafx.h" +#include "resource.h" +//#include + +#include "AcceleratorManager.h" +#include "Display.h" +#include "Input.h" +#include "IUpdate.h" +#include "Sound.h" +#include "../System.h" +#include "../Util.h" + +///////////////////////////////////////////////////////////////////////////// +// VBA: +// See VBA.cpp for the implementation of this class +// + +enum { + VIDEO_1X, VIDEO_2X, VIDEO_3X, VIDEO_4X, + VIDEO_320x240, VIDEO_640x480, VIDEO_800x600, VIDEO_1024x768, VIDEO_1280x1024, + VIDEO_OTHER +}; + +enum pixelFilterType +{ + FILTER_NONE, + + FILTER_SIMPLE2X, FILTER_PIXELATE, FILTER_TVMODE, FILTER_SCANLINES, + FILTER_BILINEAR, FILTER_BILINEARPLUS, FILTER_MAMESCALE2X, + FILTER_2XSAI, FILTER_SUPER2XSAI, FILTER_SUPEREAGLE, FILTER_LQ2X, FILTER_HQ2X, + + FILTER_SIMPLE3X, FILTER_HQ3X, + + FILTER_SIMPLE4X, FILTER_HQ4X +}; + +#define REWIND_SIZE 400000 + +class CSkin; +class AVIWrite; +class WavWriter; + +class VBA : public CWinApp +{ + public: + CMenu m_menu; + HMENU menu; + HMENU popup; + bool mode320Available; + bool mode640Available; + bool mode800Available; + bool mode1024Available; + bool mode1280Available; + int windowPositionX; + int windowPositionY; + void (*filterFunction)(u8*,u32,u8*,u8*,u32,int,int); + void (*ifbFunction)(u8*,u32,int,int); + int ifbType; + int filterType; + int filterWidth; + int filterHeight; + int filterMagnification; + int filterLCD; + int fsWidth; + int fsHeight; + int fsColorDepth; + int fsFrequency; + int fsAdapter; + bool fsForceChange; + int sizeX; + int sizeY; + int surfaceSizeX; + int surfaceSizeY; + int videoOption; + bool fullScreenStretch; + bool disableStatusMessage; + int showSpeed; + BOOL showSpeedTransparent; + int showRenderedFrames; + bool screenMessage; + CString screenMessageBuffer; + DWORD screenMessageTime; + u8 *delta[257*244*4]; + bool menuToggle; + IDisplay *display; + IMAGE_TYPE cartridgeType; + bool soundInitialized; + bool useBiosFile; + bool skipBiosFile; + CString biosFileName; + bool active; + bool paused; + CString recentFiles[10]; + bool recentFreeze; + bool autoSaveLoadCheatList; + FILE *winout; + bool removeIntros; + bool autoIPS; + int winGbBorderOn; + int winFlashSize; + bool winRtcEnable; + bool winGenericflashcardEnable; + int winSaveType; + char *rewindMemory; + int rewindPos; + int rewindTopPos; + int rewindCounter; + int rewindCount; + bool rewindSaveNeeded; + int rewindTimer; + int captureFormat; + bool tripleBuffering; + bool autoHideMenu; + int throttle; + u32 throttleLastTime; + u32 autoFrameSkipLastTime; + bool autoFrameSkip; + bool vsync; + bool changingVideoSize; + GUID videoDriverGUID; + GUID *pVideoDriverGUID; + DISPLAY_TYPE renderMethod; + bool iconic; + bool ddrawEmulationOnly; + bool ddrawUsingEmulationOnly; + bool ddrawDebug; + bool ddrawUseVideoMemory; + int d3dFilter; + int glFilter; + int glType; + bool dinputKeyFocus; + CSkin *skin; + CString skinName; + bool skinEnabled; + int skinButtons; + bool pauseWhenInactive; + bool speedupToggle; + bool useOldSync; + bool winGbPrinterEnabled; + int threadPriority; + bool disableMMX; + int languageOption; + CString languageName; + HMODULE languageModule; + int renderedFrames; + Input *input; + int joypadDefault; + int autoFire; + bool autoFireToggle; + bool winPauseNextFrame; + bool soundRecording; + WavWriter *soundRecorder; + CString soundRecordName; + bool dsoundDisableHardwareAcceleration; + ISound *sound; + bool aviRecording; + AVIWrite *aviRecorder; + CString aviRecordName; + int aviFrameNumber; + bool painting; + bool movieRecording; + bool moviePlaying; + int movieFrame; + int moviePlayFrame; + FILE *movieFile; + u32 movieLastJoypad; + u32 movieNextJoypad; + int sensorX; + int sensorY; + int mouseCounter; + bool wasPaused; + int frameskipadjust; + bool autoLoadMostRecent; + int fsMaxScale; + int romSize; + + CList updateList; + int updateCount; + + CAcceleratorManager winAccelMgr; + HACCEL hAccel; + + RECT rect; + RECT dest; + + struct EmulatedSystem emulator; + + CString szFile; + CString filename; + CString dir; + + CString wndClass; + + public: + VBA(); + ~VBA(); + + void adjustDestRect(); + void updateIFB(); + void updateFilter(); + void updateMenuBar(); + void winAddUpdateListener(IUpdateListener *l); + void winRemoveUpdateListener(IUpdateListener *l); + CString winLoadFilter(UINT id); + + // Overrides + // ClassWizard generated virtual function overrides + //{{AFX_VIRTUAL(VBA) + public: + virtual BOOL InitInstance(); + virtual BOOL OnIdle(LONG lCount); + //}}AFX_VIRTUAL + + // Implementation + + public: + void saveSettings(); + void movieReadNext(); + bool initInput(); + HMODULE winLoadLanguage(const char *name); + void winSetLanguageOption(int option, bool force); +#ifdef MMX + bool detectMMX(); +#endif + void updatePriority(); + void winUpdateSkin(); + void directXMessage(const char *msg); + void shutdownDisplay(); + void winCheckFullscreen(); + bool updateRenderMethod0(bool force); + bool updateRenderMethod(bool force); + bool initDisplay(); + void updateWindowSize(int value); + void updateVideoSize(UINT id); + void updateFrameSkip(); + void loadSettings(); + void addRecentFile(CString file); + //{{AFX_MSG(VBA) + afx_msg void OnAppAbout(); + // NOTE - the ClassWizard will add and remove member functions here. + // DO NOT EDIT what you see in these blocks of generated code ! + //}}AFX_MSG + DECLARE_MESSAGE_MAP() + }; + + extern VBA theApp; + extern int emulating; + +#ifdef MMX + extern "C" bool cpu_mmx; +#endif \ No newline at end of file diff --git a/res/VBA.rc b/src/win32/VBA.rc similarity index 83% rename from res/VBA.rc rename to src/win32/VBA.rc index d7b97664..f5cded1b 100644 --- a/res/VBA.rc +++ b/src/win32/VBA.rc @@ -8,30 +8,10 @@ // Generated from the TEXTINCLUDE 2 resource. // #include "afxres.h" +#include "resource2.h" ///////////////////////////////////////////////////////////////////////////// #undef APSTUDIO_READONLY_SYMBOLS -///////////////////////////////////////////////////////////////////////////// -// German (Germany) resources - -#if !defined(AFX_RESOURCE_DLL) || defined(AFX_TARG_DEU) -#ifdef _WIN32 -LANGUAGE LANG_GERMAN, SUBLANG_GERMAN -#pragma code_page(1252) -#endif //_WIN32 - -///////////////////////////////////////////////////////////////////////////// -// -// Icon -// - -// Icon with lowest ID value placed first to ensure application icon -// remains consistent on all systems. -IDI_ICON ICON "VBA.ico" -#endif // German (Germany) resources -///////////////////////////////////////////////////////////////////////////// - - ///////////////////////////////////////////////////////////////////////////// // English (U.S.) resources @@ -54,13 +34,13 @@ END 2 TEXTINCLUDE BEGIN - "#include ""afxres.h""\0" + "#include ""afxres.h""\r\n" + "#include ""resource2.h""\0" END 3 TEXTINCLUDE BEGIN - "\r\n" - "\0" + "#include ""vba.rc2""\0" END #endif // APSTUDIO_INVOKED @@ -153,72 +133,73 @@ BEGIN PUSHBUTTON "Cancel",IDCANCEL,90,102,50,14,WS_GROUP END -IDD_ABOUT DIALOGEX 0, 0, 172, 102 -STYLE DS_SETFONT | DS_MODALFRAME | DS_3DLOOK | WS_POPUP | WS_CAPTION | WS_SYSMENU +IDD_ABOUT DIALOGEX 0, 0, 157, 78 +STYLE DS_SETFONT | DS_MODALFRAME | WS_POPUP | WS_CAPTION | WS_SYSMENU +EXSTYLE WS_EX_TOOLWINDOW CAPTION "About" FONT 8, "MS Sans Serif", 0, 0, 0x0 BEGIN - DEFPUSHBUTTON "OK",IDOK,6,84,162,12 - ICON IDI_ICON,IDC_STATIC,12,6,21,20,SS_CENTERIMAGE,WS_EX_TRANSPARENT - CTEXT "VisualBoyAdvance Emulator",IDC_STATIC,42,6,126,8 - CTEXT "Copyright © 2005 Forgotten and the VBA team",IDC_STATIC,6,30,162,8 - CTEXT "http://vba.ngemu.com",IDC_URL,90,42,76,8 - CTEXT "Contribution by Costis",IDC_STATIC,6,42,78,8 - CTEXT "Version:",IDC_STATIC,42,18,42,8 - LTEXT "",IDC_VERSION,90,18,78,8 - CTEXT "Changes by Spacy:",IDC_URL2,6,60,162,8 - CTEXT "http://www.spacyhacks.de.vu",IDC_URL3,6,72,162,8 + ICON 101,IDC_STATIC,6,6,20,20 + CTEXT "VisualBoyAdvance Emulator",IDC_STATIC,30,6,120,8 + CTEXT "Copyright © 2006 VBA development team",IDC_STATIC,6,36,144,8 + CTEXT "http://vba.ngemu.com",IDC_URL,6,66,144,8 + CTEXT "Contribution by Costis",IDC_STATIC,6,48,144,8 + CTEXT "Version",IDC_STATIC,30,18,60,8 + CTEXT "",IDC_VERSION,90,18,60,8,SS_NOPREFIX END -IDD_DIRECTORIES DIALOG 0, 0, 284, 129 +IDD_DIRECTORIES DIALOGEX 0, 0, 220, 301 STYLE DS_SETFONT | DS_MODALFRAME | WS_POPUP | WS_CAPTION | WS_SYSMENU +EXSTYLE WS_EX_TOOLWINDOW CAPTION "Directories" -FONT 8, "MS Sans Serif" +FONT 8, "MS Sans Serif", 0, 0, 0x0 BEGIN - PUSHBUTTON "Browse...",IDC_ROM_DIR,174,7,50,14 - PUSHBUTTON "Browse...",IDC_GBROM_DIR,174,26,50,14 - PUSHBUTTON "Browse...",IDC_BATTERY_DIR,174,45,50,14 - PUSHBUTTON "Browse...",IDC_SAVE_DIR,174,64,50,14 - PUSHBUTTON "Browse...",IDC_CAPTURE_DIR,174,83,50,14 - DEFPUSHBUTTON "OK",IDOK,88,108,50,14 - PUSHBUTTON "Cancel",IDCANCEL,145,108,50,14 - LTEXT "ROM:",IDC_STATIC,7,9,20,8 - LTEXT "Battery:",IDC_STATIC,7,47,25,8 - LTEXT "Save Game:",IDC_STATIC,7,66,40,8 - LTEXT "Capture:",IDC_STATIC,7,85,28,8 - EDITTEXT IDC_ROM_PATH,49,7,121,14,ES_AUTOHSCROLL | WS_DISABLED - EDITTEXT IDC_BATTERY_PATH,49,45,121,14,ES_AUTOHSCROLL | WS_DISABLED - EDITTEXT IDC_SAVE_PATH,49,64,121,14,ES_AUTOHSCROLL | WS_DISABLED - EDITTEXT IDC_CAPTURE_PATH,49,83,121,14,ES_AUTOHSCROLL | WS_DISABLED - LTEXT "GB ROM:",IDC_STATIC,7,28,32,8 - EDITTEXT IDC_GBROM_PATH,49,26,121,14,ES_AUTOHSCROLL | WS_DISABLED - PUSHBUTTON "Reset",IDC_ROM_DIR_RESET,228,6,50,14 - PUSHBUTTON "Reset",IDC_GBROM_DIR_RESET,228,25,50,14 - PUSHBUTTON "Reset",IDC_BATTERY_DIR_RESET,228,45,50,14 - PUSHBUTTON "Reset",IDC_SAVE_DIR_RESET,228,64,50,14 - PUSHBUTTON "Reset",IDC_CAPTURE_DIR_RESET,228,83,50,14 + PUSHBUTTON "Browse...",IDC_ROM_DIR,12,66,90,12 + PUSHBUTTON "Browse...",IDC_GBROM_DIR,12,114,90,12 + PUSHBUTTON "Browse...",IDC_BATTERY_DIR,12,162,90,12 + PUSHBUTTON "Browse...",IDC_SAVE_DIR,12,210,90,12 + PUSHBUTTON "Browse...",IDC_CAPTURE_DIR,12,258,90,12 + DEFPUSHBUTTON "OK",IDOK,120,282,42,12 + PUSHBUTTON "Cancel",IDCANCEL,168,282,42,12 + EDITTEXT IDC_ROM_PATH,12,54,198,12,ES_AUTOHSCROLL + EDITTEXT IDC_BATTERY_PATH,12,150,198,12,ES_AUTOHSCROLL + EDITTEXT IDC_SAVE_PATH,12,198,198,12,ES_AUTOHSCROLL + EDITTEXT IDC_CAPTURE_PATH,12,246,198,12,ES_AUTOHSCROLL + EDITTEXT IDC_GBROM_PATH,12,102,198,12,ES_AUTOHSCROLL + PUSHBUTTON "Reset",IDC_ROM_DIR_RESET,120,66,90,12 + PUSHBUTTON "Reset",IDC_GBROM_DIR_RESET,120,114,84,12 + PUSHBUTTON "Reset",IDC_BATTERY_DIR_RESET,120,162,90,12 + PUSHBUTTON "Reset",IDC_SAVE_DIR_RESET,120,210,90,12 + PUSHBUTTON "Reset",IDC_CAPTURE_DIR_RESET,120,258,90,12 + CONTROL "You can either select or enter absolute paths to directories,\nor you can use relative paths by beginning with a . (point).\nExamples: c:\\emulation\\roms | .\\battery | ..\\snapshots\\gba",IDC_STATIC, + "Static",SS_LEFTNOWORDWRAP | WS_GROUP,6,6,210,30,WS_EX_STATICEDGE + GROUPBOX "Game Boy Advance ROM Images",IDC_STATIC,6,42,210,42 + GROUPBOX "Game Boy / Game Boy Color ROM Images",IDC_STATIC,6,90,210,42 + GROUPBOX "Game Boy Internal Battery Saves",IDC_STATIC,6,138,210,42 + GROUPBOX "Visual Boy Advance Save Games",IDC_STATIC,6,186,210,42 + GROUPBOX "Screenshots",IDC_STATIC,6,234,210,42 END -IDD_CONFIG DIALOGEX 0, 0, 209, 230 +IDD_CONFIG DIALOGEX 0, 0, 221, 241 STYLE DS_SYSMODAL | DS_SETFONT | DS_MODALFRAME | WS_POPUP | WS_CAPTION | WS_SYSMENU CAPTION "Joypad configuration" FONT 8, "MS Sans Serif", 0, 0, 0x0 BEGIN - EDITTEXT IDC_EDIT_UP,45,5,157,12,ES_AUTOHSCROLL - EDITTEXT IDC_EDIT_DOWN,45,19,157,12,ES_AUTOHSCROLL - EDITTEXT IDC_EDIT_LEFT,45,33,157,12,ES_AUTOHSCROLL - EDITTEXT IDC_EDIT_RIGHT,45,47,157,12,ES_AUTOHSCROLL - EDITTEXT IDC_EDIT_BUTTON_A,45,61,157,12,ES_AUTOHSCROLL - EDITTEXT IDC_EDIT_BUTTON_B,45,75,157,12,ES_AUTOHSCROLL - EDITTEXT IDC_EDIT_BUTTON_L,45,89,157,12,ES_AUTOHSCROLL - EDITTEXT IDC_EDIT_BUTTON_R,45,103,157,12,ES_AUTOHSCROLL - EDITTEXT IDC_EDIT_BUTTON_SELECT,45,117,157,12,ES_AUTOHSCROLL - EDITTEXT IDC_EDIT_BUTTON_START,45,131,157,12,ES_AUTOHSCROLL - EDITTEXT IDC_EDIT_SPEED,45,145,157,12,ES_AUTOHSCROLL - EDITTEXT IDC_EDIT_CAPTURE,45,159,157,12,ES_AUTOHSCROLL - EDITTEXT IDC_EDIT_BUTTON_GS,45,173,157,12,ES_AUTOHSCROLL - PUSHBUTTON "OK",ID_OK,63,209,40,14 - PUSHBUTTON "Cancel",ID_CANCEL,113,209,40,14 + EDITTEXT IDC_EDIT_UP,38,5,176,12,ES_AUTOHSCROLL + EDITTEXT IDC_EDIT_DOWN,38,19,176,12,ES_AUTOHSCROLL + EDITTEXT IDC_EDIT_LEFT,38,33,176,12,ES_AUTOHSCROLL + EDITTEXT IDC_EDIT_RIGHT,38,47,176,12,ES_AUTOHSCROLL + EDITTEXT IDC_EDIT_BUTTON_A,38,61,176,12,ES_AUTOHSCROLL + EDITTEXT IDC_EDIT_BUTTON_B,38,75,176,12,ES_AUTOHSCROLL + EDITTEXT IDC_EDIT_BUTTON_L,38,89,176,12,ES_AUTOHSCROLL + EDITTEXT IDC_EDIT_BUTTON_R,38,103,176,12,ES_AUTOHSCROLL + EDITTEXT IDC_EDIT_BUTTON_SELECT,38,117,176,12,ES_AUTOHSCROLL + EDITTEXT IDC_EDIT_BUTTON_START,38,131,176,12,ES_AUTOHSCROLL + EDITTEXT IDC_EDIT_SPEED,38,145,176,12,ES_AUTOHSCROLL + EDITTEXT IDC_EDIT_CAPTURE,38,159,176,12,ES_AUTOHSCROLL + EDITTEXT IDC_EDIT_BUTTON_GS,38,173,176,12,ES_AUTOHSCROLL + PUSHBUTTON "OK",ID_OK,61,220,40,14 + PUSHBUTTON "Cancel",ID_CANCEL,115,220,40,14 LTEXT "Up:",IDC_STATIC,5,5,35,10 LTEXT "Down:",IDC_STATIC,5,19,35,10 LTEXT "Left:",IDC_STATIC,5,33,35,10 @@ -229,11 +210,11 @@ BEGIN LTEXT "Button R:",IDC_STATIC,5,103,35,10 LTEXT "Select:",IDC_STATIC,5,117,35,10 LTEXT "Start:",IDC_STATIC,5,131,35,10 - LTEXT "Turbo:",IDC_STATIC,5,145,35,10 + LTEXT "Speed:",IDC_STATIC,5,145,35,10 LTEXT "Capture:",IDC_STATIC,5,159,35,10 LTEXT "GS:",IDC_STATIC,5,173,35,10 CONTROL "Assign additional keys to functions",IDC_APPENDMODE, - "Button",BS_AUTOCHECKBOX | WS_TABSTOP,47,193,135,10 + "Button",BS_AUTOCHECKBOX | WS_TABSTOP,47,197,135,10 END IDD_CHEATS DIALOG 0, 0, 276, 253 @@ -331,28 +312,28 @@ BEGIN GROUPBOX "GB Types",IDC_STATIC,3,3,50,69 END -IDD_GBA_ROM_INFO DIALOGEX 0, 0, 208, 126 +IDD_GBA_ROM_INFO DIALOG 0, 0, 220, 142 STYLE DS_SETFONT | DS_MODALFRAME | WS_POPUP | WS_CAPTION | WS_SYSMENU -CAPTION "ROM Header" -FONT 8, "MS Sans Serif", 0, 0, 0x0 +CAPTION "Rom information" +FONT 8, "MS Sans Serif" BEGIN - DEFPUSHBUTTON "OK",ID_OK,6,108,198,12 - RTEXT "Game Title:",IDC_STATIC,6,6,60,12 - RTEXT "Game Code:",IDC_STATIC,6,18,60,12 - RTEXT "Maker Code:",IDC_STATIC,6,30,60,12 - RTEXT "Main Unit Code:",IDC_STATIC,6,54,60,12 - RTEXT "Device Type:",IDC_STATIC,6,66,60,12 - RTEXT "ROM Version:",IDC_STATIC,6,78,60,12 - RTEXT "Complement:",IDC_STATIC,6,90,60,12 - LTEXT "",IDC_ROM_TITLE,78,6,126,12 - LTEXT "",IDC_ROM_GAME_CODE,78,18,126,12 - LTEXT "",IDC_ROM_MAKER_CODE,78,30,126,12 - LTEXT "",IDC_ROM_UNIT_CODE,78,54,126,12 - LTEXT "",IDC_ROM_DEVICE_TYPE,78,66,126,12 - LTEXT "",IDC_ROM_VERSION,78,78,126,12 - LTEXT "",IDC_ROM_CRC,78,90,126,12 - LTEXT "",IDC_ROM_MAKER_NAME,78,42,126,12 - RTEXT "Maker Name:",IDC_STATIC,6,42,60,12 + DEFPUSHBUTTON "OK",ID_OK,84,121,50,14 + LTEXT "Game title:",IDC_STATIC,7,10,60,8 + LTEXT "Game code:",IDC_STATIC,7,24,60,8 + LTEXT "Maker code:",IDC_STATIC,7,38,60,8 + LTEXT "Main unit code:",IDC_STATIC,7,66,60,8 + LTEXT "Device type:",IDC_STATIC,7,80,60,8 + LTEXT "ROM version:",IDC_STATIC,7,94,60,8 + LTEXT "CRC:",IDC_STATIC,7,108,60,8 + LTEXT "Maker name:",IDC_STATIC,7,52,60,8 + LTEXT "",IDC_ROM_TITLE,80,10,133,8,SS_NOPREFIX + LTEXT "",IDC_ROM_GAME_CODE,80,24,133,8,SS_NOPREFIX + LTEXT "",IDC_ROM_MAKER_CODE,80,38,133,8,SS_NOPREFIX + LTEXT "",IDC_ROM_UNIT_CODE,80,66,133,8,SS_NOPREFIX + LTEXT "",IDC_ROM_DEVICE_TYPE,80,80,133,8,SS_NOPREFIX + LTEXT "",IDC_ROM_VERSION,80,94,133,8,SS_NOPREFIX + LTEXT "",IDC_ROM_CRC,80,108,133,8,SS_NOPREFIX + LTEXT "",IDC_ROM_MAKER_NAME,80,52,133,8,SS_NOPREFIX END IDD_GB_ROM_INFO DIALOG 0, 0, 220, 225 @@ -367,25 +348,25 @@ BEGIN LTEXT "Cartridge type:",IDC_STATIC,7,82,60,8 LTEXT "ROM version:",IDC_STATIC,7,152,60,8 LTEXT "CRC:",IDC_STATIC,7,166,60,8 - LTEXT "",IDC_ROM_TITLE,80,10,133,8 - LTEXT "",IDC_ROM_MAKER_CODE,80,38,133,8 - LTEXT "",IDC_ROM_UNIT_CODE,80,68,133,8 - LTEXT "",IDC_ROM_DEVICE_TYPE,80,82,133,8 - LTEXT "",IDC_ROM_VERSION,80,152,133,8 - LTEXT "",IDC_ROM_CRC,80,166,133,8 + LTEXT "",IDC_ROM_TITLE,80,10,133,8,SS_NOPREFIX + LTEXT "",IDC_ROM_MAKER_CODE,80,38,133,8,SS_NOPREFIX + LTEXT "",IDC_ROM_UNIT_CODE,80,68,133,8,SS_NOPREFIX + LTEXT "",IDC_ROM_DEVICE_TYPE,80,82,133,8,SS_NOPREFIX + LTEXT "",IDC_ROM_VERSION,80,152,133,8,SS_NOPREFIX + LTEXT "",IDC_ROM_CRC,80,166,133,8,SS_NOPREFIX LTEXT "Color:",IDC_STATIC,7,24,60,8 - LTEXT "",IDC_ROM_COLOR,80,24,133,8 + LTEXT "",IDC_ROM_COLOR,80,24,133,8,SS_NOPREFIX LTEXT "ROM size:",IDC_STATIC,7,96,60,8 - LTEXT "",IDC_ROM_SIZE,80,96,133,8 + LTEXT "",IDC_ROM_SIZE,80,96,133,8,SS_NOPREFIX LTEXT "RAM size:",IDC_STATIC,7,110,60,8 - LTEXT "",IDC_ROM_RAM_SIZE,80,110,133,8 + LTEXT "",IDC_ROM_RAM_SIZE,80,110,133,8,SS_NOPREFIX LTEXT "Dest. code:",IDC_STATIC,7,124,60,8 - LTEXT "",IDC_ROM_DEST_CODE,80,124,133,8 + LTEXT "",IDC_ROM_DEST_CODE,80,124,133,8,SS_NOPREFIX LTEXT "License code:",IDC_STATIC,7,138,60,8 - LTEXT "",IDC_ROM_LIC_CODE,80,138,133,8 + LTEXT "",IDC_ROM_LIC_CODE,80,138,133,8,SS_NOPREFIX LTEXT "Checksum:",IDC_STATIC,7,180,60,8 - LTEXT "",IDC_ROM_CHECKSUM,80,180,133,8 - LTEXT "",IDC_ROM_MAKER_NAME2,80,52,133,8 + LTEXT "",IDC_ROM_CHECKSUM,80,180,133,8,SS_NOPREFIX + LTEXT "",IDC_ROM_MAKER_NAME2,80,52,133,8,SS_NOPREFIX LTEXT "Maker name:",IDC_STATIC,7,52,60,8 END @@ -435,23 +416,23 @@ BEGIN GROUPBOX "Print Size",IDC_STATIC,7,156,162,25 END -IDD_MOTION_CONFIG DIALOGEX 0, 0, 219, 114 +IDD_MOTION_CONFIG DIALOGEX 0, 0, 234, 107 STYLE DS_SETFONT | DS_MODALFRAME | WS_POPUP | WS_CAPTION | WS_SYSMENU CAPTION "Motion Sensor" FONT 8, "MS Sans Serif", 0, 0, 0x0 BEGIN - EDITTEXT IDC_EDIT_UP,49,2,163,12,ES_AUTOHSCROLL - EDITTEXT IDC_EDIT_DOWN,49,16,163,12,ES_AUTOHSCROLL - EDITTEXT IDC_EDIT_LEFT,49,30,163,12,ES_AUTOHSCROLL - EDITTEXT IDC_EDIT_RIGHT,49,44,163,12,ES_AUTOHSCROLL - PUSHBUTTON "OK",ID_OK,61,92,40,14 - PUSHBUTTON "Cancel",ID_CANCEL,115,92,40,14 + EDITTEXT IDC_EDIT_UP,41,2,186,12,ES_AUTOHSCROLL + EDITTEXT IDC_EDIT_DOWN,41,16,186,12,ES_AUTOHSCROLL + EDITTEXT IDC_EDIT_LEFT,41,30,186,12,ES_AUTOHSCROLL + EDITTEXT IDC_EDIT_RIGHT,41,44,186,12,ES_AUTOHSCROLL + PUSHBUTTON "OK",ID_OK,64,86,40,14 + PUSHBUTTON "Cancel",ID_CANCEL,118,86,40,14 LTEXT "Up:",IDC_STATIC,5,2,35,10 LTEXT "Down:",IDC_STATIC,5,16,35,10 LTEXT "Left:",IDC_STATIC,5,30,35,10 LTEXT "Right:",IDC_STATIC,5,44,35,10 CONTROL "Assign additional keys to functions",IDC_APPENDMODE, - "Button",BS_AUTOCHECKBOX | WS_TABSTOP,47,70,135,10 + "Button",BS_AUTOCHECKBOX | WS_TABSTOP,46,66,135,10 END IDD_LANG_SELECT DIALOG 0, 0, 186, 68 @@ -464,7 +445,7 @@ BEGIN PUSHBUTTON "Cancel",ID_CANCEL,104,49,50,14 LTEXT "Current system language is:",IDC_STATIC,6,9,123,8 LTEXT "Enter language name (3 letter):",IDC_STATIC,6,30,127,8 - LTEXT "",IDC_LANG_NAME,140,9,40,8 + LTEXT "",IDC_LANG_NAME,140,9,40,8,SS_NOPREFIX END IDD_CODE_SELECT DIALOG 0, 0, 316, 235 @@ -496,36 +477,36 @@ BEGIN CONTROL "MapView",IDC_MAP_VIEW,"VbaBitmapControl",WS_GROUP | WS_TABSTOP,187,15,128,128 CONTROL "Zoom",IDC_MAP_VIEW_ZOOM,"VbaZoomControl",WS_GROUP | WS_TABSTOP,7,148,64,64 CONTROL "Color",IDC_COLOR,"VbaColorControl",WS_TABSTOP,187,164,48,47 - LTEXT "",IDC_R,245,173,50,8 - LTEXT "",IDC_G,245,185,50,8 - LTEXT "",IDC_B,245,197,50,8 + LTEXT "",IDC_R,245,173,50,8,SS_NOPREFIX + LTEXT "",IDC_G,245,185,50,8,SS_NOPREFIX + LTEXT "",IDC_B,245,197,50,8,SS_NOPREFIX GROUPBOX "Frame",IDC_STATIC,7,11,63,37 GROUPBOX "Background",IDC_STATIC,7,52,63,67 - LTEXT "",IDC_XY,129,95,53,8 + LTEXT "",IDC_XY,129,95,53,8,SS_NOPREFIX LTEXT "Mode:",IDC_STATIC,80,15,34,8 - LTEXT "",IDC_MODE,130,15,53,8 + LTEXT "",IDC_MODE,130,15,53,8,SS_NOPREFIX LTEXT "Map Base:",IDC_STATIC,80,25,35,8 - LTEXT "",IDC_MAPBASE,130,25,53,8 + LTEXT "",IDC_MAPBASE,130,25,53,8,SS_NOPREFIX LTEXT "Char Base:",IDC_STATIC,80,35,36,8 - LTEXT "",IDC_CHARBASE,130,35,53,8 + LTEXT "",IDC_CHARBASE,130,35,53,8,SS_NOPREFIX LTEXT "Size:",IDC_STATIC,80,45,37,8 - LTEXT "",IDC_DIM,130,45,53,8 + LTEXT "",IDC_DIM,130,45,53,8,SS_NOPREFIX LTEXT "Colors:",IDC_STATIC,80,55,37,8 - LTEXT "",IDC_NUMCOLORS,130,55,53,8 + LTEXT "",IDC_NUMCOLORS,130,55,53,8,SS_NOPREFIX LTEXT "Priority:",IDC_STATIC,80,65,37,8 - LTEXT "",IDC_PRIORITY,130,65,53,8 + LTEXT "",IDC_PRIORITY,130,65,53,8,SS_NOPREFIX LTEXT "Mosaic:",IDC_STATIC,80,75,37,8 - LTEXT "",IDC_MOSAIC,130,75,53,8 + LTEXT "",IDC_MOSAIC,130,75,53,8,SS_NOPREFIX LTEXT "Overflow:",IDC_STATIC,80,85,37,8 - LTEXT "",IDC_OVERFLOW,130,85,53,8 + LTEXT "",IDC_OVERFLOW,130,85,53,8,SS_NOPREFIX LTEXT "Address:",IDC_STATIC,80,105,37,8 - LTEXT "",IDC_ADDRESS,130,105,53,8 + LTEXT "",IDC_ADDRESS,130,105,53,8,SS_NOPREFIX LTEXT "Tile:",IDC_STATIC,80,115,37,8 - LTEXT "",IDC_TILE_NUM,130,115,53,8 + LTEXT "",IDC_TILE_NUM,130,115,53,8,SS_NOPREFIX LTEXT "Flip:",IDC_STATIC,80,125,37,8 - LTEXT "",IDC_FLIP,130,125,53,8 + LTEXT "",IDC_FLIP,130,125,53,8,SS_NOPREFIX LTEXT "Palette:",IDC_STATIC,80,135,37,8 - LTEXT "",IDC_PALETTE_NUM,130,135,53,8 + LTEXT "",IDC_PALETTE_NUM,130,135,53,8,SS_NOPREFIX END IDD_PALETTE_VIEW DIALOG 0, 0, 316, 266 @@ -537,11 +518,11 @@ BEGIN PUSHBUTTON "Save OBJ...",IDC_SAVE_OBJ,98,245,50,14 PUSHBUTTON "&Refresh",IDC_REFRESH2,166,245,50,14 PUSHBUTTON "&Close",IDC_CLOSE,234,245,50,14 - LTEXT "",IDC_ADDRESS,53,168,50,8 - LTEXT "",IDC_R,53,180,50,8 - LTEXT "",IDC_G,53,192,50,8 - LTEXT "",IDC_B,53,204,50,8 - LTEXT "",IDC_VALUE,53,216,50,8 + LTEXT "",IDC_ADDRESS,53,168,50,8,SS_NOPREFIX + LTEXT "",IDC_R,53,180,50,8,SS_NOPREFIX + LTEXT "",IDC_G,53,192,50,8,SS_NOPREFIX + LTEXT "",IDC_B,53,204,50,8,SS_NOPREFIX + LTEXT "",IDC_VALUE,53,216,50,8,SS_NOPREFIX CONTROL "Custom1",IDC_COLOR,"VbaColorControl",WS_TABSTOP,161,168,50,50 CONTROL "PaletteViewBG",IDC_PALETTE_VIEW,"VbaPaletteViewControl",WS_TABSTOP,12,30,128,128 CONTROL "PaletteViewBG",IDC_PALETTE_VIEW_OBJ, @@ -592,18 +573,18 @@ BEGIN CONTROL "MapView",IDC_OAM_VIEW,"VbaBitmapControl",WS_GROUP | WS_TABSTOP,87,7,64,64 CONTROL "Zoom",IDC_OAM_VIEW_ZOOM,"VbaZoomControl",WS_GROUP | WS_TABSTOP,163,7,64,64 CONTROL "Color",IDC_COLOR,"VbaColorControl",WS_TABSTOP,87,79,48,47 - LTEXT "",IDC_POS,31,47,50,8 - LTEXT "",IDC_MODE,31,57,50,8 - LTEXT "",IDC_COLORS,31,67,50,8 - LTEXT "",IDC_PALETTE,31,77,50,8 - LTEXT "",IDC_TILE,31,87,50,8 - LTEXT "",IDC_PRIO,31,97,50,8 - LTEXT "",IDC_SIZE2,31,107,50,8 - LTEXT "",IDC_ROT,31,117,50,8 - LTEXT "",IDC_FLAGS,31,127,50,8 - LTEXT "",IDC_R,145,88,50,8 - LTEXT "",IDC_G,145,100,50,8 - LTEXT "",IDC_B,145,112,50,8 + LTEXT "",IDC_POS,31,47,50,8,SS_NOPREFIX + LTEXT "",IDC_MODE,31,57,50,8,SS_NOPREFIX + LTEXT "",IDC_COLORS,31,67,50,8,SS_NOPREFIX + LTEXT "",IDC_PALETTE,31,77,50,8,SS_NOPREFIX + LTEXT "",IDC_TILE,31,87,50,8,SS_NOPREFIX + LTEXT "",IDC_PRIO,31,97,50,8,SS_NOPREFIX + LTEXT "",IDC_SIZE2,31,107,50,8,SS_NOPREFIX + LTEXT "",IDC_ROT,31,117,50,8,SS_NOPREFIX + LTEXT "",IDC_FLAGS,31,127,50,8,SS_NOPREFIX + LTEXT "",IDC_R,145,88,50,8,SS_NOPREFIX + LTEXT "",IDC_G,145,100,50,8,SS_NOPREFIX + LTEXT "",IDC_B,145,112,50,8,SS_NOPREFIX LTEXT "Pos:",IDC_STATIC,7,47,24,8 LTEXT "Mode:",IDC_STATIC,7,57,24,8 LTEXT "Colors:",IDC_STATIC,7,67,24,8 @@ -659,15 +640,15 @@ BEGIN GROUPBOX "Char Base",IDC_STATIC,7,46,65,64 CONTROL "Zoom",IDC_MAP_VIEW_ZOOM,"VbaZoomControl",WS_GROUP | WS_TABSTOP,7,174,64,64 CONTROL "Color",IDC_COLOR,"VbaColorControl",WS_TABSTOP,98,183,48,47 - LTEXT "",IDC_R,156,192,50,8 - LTEXT "",IDC_G,156,204,50,8 - LTEXT "",IDC_B,156,216,50,8 + LTEXT "",IDC_R,156,192,50,8,SS_NOPREFIX + LTEXT "",IDC_G,156,204,50,8,SS_NOPREFIX + LTEXT "",IDC_B,156,216,50,8,SS_NOPREFIX LTEXT "Palette:",IDC_STATIC,7,113,65,8 CONTROL "Automatic update",IDC_AUTO_UPDATE,"Button",BS_AUTOCHECKBOX | WS_TABSTOP,7,160,71,10 LTEXT "Tile:",IDC_STATIC,79,14,41,8 LTEXT "Address:",IDC_STATIC,79,26,41,8 - LTEXT "",IDC_TILE_NUMBER,135,14,50,8 - LTEXT "",IDC_ADDRESS,135,26,50,8 + LTEXT "",IDC_TILE_NUMBER,135,14,50,8,SS_NOPREFIX + LTEXT "",IDC_ADDRESS,135,26,50,8,SS_NOPREFIX END IDD_GB_COLORS DIALOG 0, 0, 169, 121 @@ -717,22 +698,22 @@ BEGIN LTEXT "R5:",IDC_STATIC,309,47,18,8 LTEXT "R6:",IDC_STATIC,309,55,18,8 LTEXT "R7:",IDC_STATIC,309,63,18,8 - LTEXT "",IDC_R0,344,7,52,8 - LTEXT "",IDC_R1,344,15,52,8 - LTEXT "",IDC_R2,344,23,52,8 - LTEXT "",IDC_R3,344,31,52,8 - LTEXT "",IDC_R4,344,39,52,8 - LTEXT "",IDC_R5,344,47,52,8 - LTEXT "",IDC_R6,344,55,52,8 - LTEXT "",IDC_R7,344,63,52,8 - LTEXT "",IDC_R8,344,71,52,8 - LTEXT "",IDC_R9,344,79,52,8 - LTEXT "",IDC_R10,344,87,52,8 - LTEXT "",IDC_R11,344,95,52,8 - LTEXT "",IDC_R12,344,103,52,8 - LTEXT "",IDC_R13,344,111,52,8 - LTEXT "",IDC_R14,344,119,52,8 - LTEXT "",IDC_R15,344,127,52,8 + LTEXT "",IDC_R0,344,7,52,8,SS_NOPREFIX + LTEXT "",IDC_R1,344,15,52,8,SS_NOPREFIX + LTEXT "",IDC_R2,344,23,52,8,SS_NOPREFIX + LTEXT "",IDC_R3,344,31,52,8,SS_NOPREFIX + LTEXT "",IDC_R4,344,39,52,8,SS_NOPREFIX + LTEXT "",IDC_R5,344,47,52,8,SS_NOPREFIX + LTEXT "",IDC_R6,344,55,52,8,SS_NOPREFIX + LTEXT "",IDC_R7,344,63,52,8,SS_NOPREFIX + LTEXT "",IDC_R8,344,71,52,8,SS_NOPREFIX + LTEXT "",IDC_R9,344,79,52,8,SS_NOPREFIX + LTEXT "",IDC_R10,344,87,52,8,SS_NOPREFIX + LTEXT "",IDC_R11,344,95,52,8,SS_NOPREFIX + LTEXT "",IDC_R12,344,103,52,8,SS_NOPREFIX + LTEXT "",IDC_R13,344,111,52,8,SS_NOPREFIX + LTEXT "",IDC_R14,344,119,52,8,SS_NOPREFIX + LTEXT "",IDC_R15,344,127,52,8,SS_NOPREFIX LTEXT "R8:",IDC_STATIC,309,71,18,8 LTEXT "R9:",IDC_STATIC,309,79,18,8 LTEXT "R10:",IDC_STATIC,309,87,18,8 @@ -741,7 +722,7 @@ BEGIN LTEXT "R13:",IDC_STATIC,309,111,18,8 LTEXT "R14:",IDC_STATIC,309,119,18,8 LTEXT "R15:",IDC_STATIC,309,127,18,8 - LTEXT "",IDC_R16,344,135,52,8 + LTEXT "",IDC_R16,344,135,52,8,SS_NOPREFIX LTEXT "R16:",IDC_STATIC,309,135,20,8 CONTROL "N",IDC_N,"Button",BS_AUTOCHECKBOX | WS_DISABLED | WS_TABSTOP,309,146,21,10 CONTROL "Z",IDC_Z,"Button",BS_AUTOCHECKBOX | WS_DISABLED | WS_TABSTOP,309,156,21,10 @@ -751,7 +732,7 @@ BEGIN CONTROL "I",IDC_I,"Button",BS_AUTOCHECKBOX | WS_DISABLED | WS_TABSTOP,342,146,18,10 CONTROL "T",IDC_T,"Button",BS_AUTOCHECKBOX | WS_DISABLED | WS_TABSTOP,342,166,21,10 LTEXT "Mode:",IDC_STATIC,341,176,21,8 - LTEXT "",IDC_MODE,376,176,20,8 + LTEXT "",IDC_MODE,376,176,20,8,SS_NOPREFIX SCROLLBAR IDC_VSCROLL,283,25,10,161,SBS_VERT PUSHBUTTON "Goto R15",IDC_GOPC,7,204,50,14 END @@ -774,7 +755,7 @@ FONT 8, "MS Sans Serif" BEGIN PUSHBUTTON "Cancel",ID_CANCEL,67,23,50,14 LTEXT "Waiting for connection on port:",IDC_STATIC,7,7,117,8 - LTEXT "",IDC_PORT,143,7,36,8 + LTEXT "",IDC_PORT,143,7,36,8,SS_NOPREFIX END IDD_LOGGING DIALOG 0, 0, 366, 218 @@ -877,11 +858,13 @@ BEGIN LTEXT "DE:",IDC_STATIC,250,45,18,8 LTEXT "HL:",IDC_STATIC,250,55,18,8 LTEXT "IFF:",IDC_STATIC,250,85,18,8 - LTEXT "",IDC_R0,285,25,52,8 - LTEXT "",IDC_R1,285,35,52,8 - LTEXT "",IDC_R2,285,45,52,8 - LTEXT "",IDC_R3,285,55,52,8 - LTEXT "",IDC_R6,285,85,52,8 + LTEXT "LY:",IDC_STATIC,272,95,18,8 + LTEXT "",IDC_R0,285,25,52,8,SS_NOPREFIX + LTEXT "",IDC_R1,285,35,52,8,SS_NOPREFIX + LTEXT "",IDC_R2,285,45,52,8,SS_NOPREFIX + LTEXT "",IDC_R3,285,55,52,8,SS_NOPREFIX + LTEXT "",IDC_R6,285,85,52,8,SS_NOPREFIX + LTEXT "",IDC_LY,285,95,52,8,SS_NOPREFIX CONTROL "N",IDC_N,"Button",BS_AUTOCHECKBOX | WS_DISABLED | WS_TABSTOP,250,109,21,10 CONTROL "Z",IDC_Z,"Button",BS_AUTOCHECKBOX | WS_DISABLED | WS_TABSTOP,250,97,21,10 CONTROL "C",IDC_C,"Button",BS_AUTOCHECKBOX | WS_DISABLED | WS_TABSTOP,250,133,21,10 @@ -889,9 +872,9 @@ BEGIN SCROLLBAR IDC_VSCROLL,229,25,10,161,SBS_VERT PUSHBUTTON "Goto PC",IDC_GOPC,7,204,50,14 LTEXT "SP:",IDC_STATIC,250,65,18,8 - LTEXT "",IDC_R4,285,65,52,8 + LTEXT "",IDC_R4,285,65,52,8,SS_NOPREFIX LTEXT "PC:",IDC_STATIC,250,75,18,8 - LTEXT "",IDC_R5,285,75,52,8 + LTEXT "",IDC_R5,285,75,52,8,SS_NOPREFIX END IDD_GB_OAM_VIEW DIALOG 0, 0, 234, 185 @@ -908,15 +891,15 @@ BEGIN CONTROL "MapView",IDC_OAM_VIEW,"VbaBitmapControl",WS_GROUP | WS_TABSTOP,87,7,64,64 CONTROL "Zoom",IDC_OAM_VIEW_ZOOM,"VbaZoomControl",WS_GROUP | WS_TABSTOP,163,7,64,64 CONTROL "Color",IDC_COLOR,"VbaColorControl",WS_TABSTOP,87,79,48,47 - LTEXT "",IDC_POS,31,47,50,8 - LTEXT "",IDC_PALETTE,31,87,50,8 - LTEXT "",IDC_TILE,31,57,50,8 - LTEXT "",IDC_PRIO,31,67,50,8 - LTEXT "",IDC_OAP,31,77,50,8 - LTEXT "",IDC_FLAGS,31,97,50,8 - LTEXT "",IDC_R,145,88,50,8 - LTEXT "",IDC_G,145,100,50,8 - LTEXT "",IDC_B,145,112,50,8 + LTEXT "",IDC_POS,31,47,50,8,SS_NOPREFIX + LTEXT "",IDC_PALETTE,31,87,50,8,SS_NOPREFIX + LTEXT "",IDC_TILE,31,57,50,8,SS_NOPREFIX + LTEXT "",IDC_PRIO,31,67,50,8,SS_NOPREFIX + LTEXT "",IDC_OAP,31,77,50,8,SS_NOPREFIX + LTEXT "",IDC_FLAGS,31,97,50,8,SS_NOPREFIX + LTEXT "",IDC_R,145,88,50,8,SS_NOPREFIX + LTEXT "",IDC_G,145,100,50,8,SS_NOPREFIX + LTEXT "",IDC_B,145,112,50,8,SS_NOPREFIX LTEXT "Pos:",IDC_STATIC,7,47,24,8 LTEXT "Pal:",IDC_STATIC,7,87,24,8 LTEXT "Tile:",IDC_STATIC,7,57,24,8 @@ -925,7 +908,7 @@ BEGIN LTEXT "Sprite:",IDC_STATIC,7,7,50,8 LTEXT "Flags:",IDC_STATIC,7,97,24,8 CONTROL "Automatic update",IDC_AUTO_UPDATE,"Button",BS_AUTOCHECKBOX | WS_TABSTOP,7,150,71,10 - LTEXT "",IDC_BANK,31,107,50,8 + LTEXT "",IDC_BANK,31,107,50,8,SS_NOPREFIX LTEXT "Bank:",IDC_STATIC,7,107,24,8 END @@ -947,15 +930,15 @@ BEGIN GROUPBOX "Char Base",IDC_STATIC,7,46,65,35 CONTROL "Zoom",IDC_MAP_VIEW_ZOOM,"VbaZoomControl",WS_GROUP | WS_TABSTOP,7,147,64,64 CONTROL "Color",IDC_COLOR,"VbaColorControl",WS_TABSTOP,98,156,48,47 - LTEXT "",IDC_R,156,164,50,8 - LTEXT "",IDC_G,156,176,50,8 - LTEXT "",IDC_B,156,188,50,8 + LTEXT "",IDC_R,156,164,50,8,SS_NOPREFIX + LTEXT "",IDC_G,156,176,50,8,SS_NOPREFIX + LTEXT "",IDC_B,156,188,50,8,SS_NOPREFIX LTEXT "Palette:",IDC_STATIC,7,86,65,8 CONTROL "Automatic update",IDC_AUTO_UPDATE,"Button",BS_AUTOCHECKBOX | WS_TABSTOP,7,133,71,10 LTEXT "Tile:",IDC_STATIC,79,14,41,8 LTEXT "Address:",IDC_STATIC,79,26,41,8 - LTEXT "",IDC_TILE_NUMBER,135,14,50,8 - LTEXT "",IDC_ADDRESS,135,26,50,8 + LTEXT "",IDC_TILE_NUMBER,135,14,50,8,SS_NOPREFIX + LTEXT "",IDC_ADDRESS,135,26,50,8,SS_NOPREFIX CONTROL "Slider1",IDC_PALETTE_SLIDER,"msctls_trackbar32",TBS_AUTOTICKS | WS_GROUP | WS_TABSTOP,1,98,76,22 END @@ -975,23 +958,23 @@ BEGIN CONTROL "MapView",IDC_MAP_VIEW,"VbaBitmapControl",WS_GROUP | WS_TABSTOP,187,15,128,128 CONTROL "Zoom",IDC_MAP_VIEW_ZOOM,"VbaZoomControl",WS_GROUP | WS_TABSTOP,7,148,64,64 CONTROL "Color",IDC_COLOR,"VbaColorControl",WS_TABSTOP,187,164,48,47 - LTEXT "",IDC_R,245,173,50,8 - LTEXT "",IDC_G,245,185,50,8 - LTEXT "",IDC_B,245,197,50,8 + LTEXT "",IDC_R,245,173,50,8,SS_NOPREFIX + LTEXT "",IDC_G,245,185,50,8,SS_NOPREFIX + LTEXT "",IDC_B,245,197,50,8,SS_NOPREFIX GROUPBOX "Char Base",IDC_STATIC,7,11,63,37 GROUPBOX "Map Base",IDC_STATIC,7,52,63,41 CONTROL "Auto update",IDC_AUTO_UPDATE,"Button",BS_AUTOCHECKBOX | WS_TABSTOP,7,134,55,10 - LTEXT "",IDC_XY,129,18,53,8 + LTEXT "",IDC_XY,129,18,53,8,SS_NOPREFIX LTEXT "Priority:",IDC_STATIC,80,68,37,8 - LTEXT "",IDC_PRIORITY,130,68,53,8 + LTEXT "",IDC_PRIORITY,130,68,53,8,SS_NOPREFIX LTEXT "Address:",IDC_STATIC,80,28,37,8 - LTEXT "",IDC_ADDRESS,130,28,53,8 + LTEXT "",IDC_ADDRESS,130,28,53,8,SS_NOPREFIX LTEXT "Tile:",IDC_STATIC,80,38,37,8 - LTEXT "",IDC_TILE_NUM,130,38,53,8 + LTEXT "",IDC_TILE_NUM,130,38,53,8,SS_NOPREFIX LTEXT "Flip:",IDC_STATIC,80,48,37,8 - LTEXT "",IDC_FLIP,130,48,53,8 + LTEXT "",IDC_FLIP,130,48,53,8,SS_NOPREFIX LTEXT "Palette:",IDC_STATIC,80,58,37,8 - LTEXT "",IDC_PALETTE_NUM,130,58,53,8 + LTEXT "",IDC_PALETTE_NUM,130,58,53,8,SS_NOPREFIX END IDD_GB_PALETTE_VIEW DIALOG 0, 0, 196, 234 @@ -1003,11 +986,11 @@ BEGIN PUSHBUTTON "Save OBJ...",IDC_SAVE_OBJ,73,191,50,14 PUSHBUTTON "&Refresh",IDC_REFRESH2,139,191,50,14 PUSHBUTTON "&Close",IDC_CLOSE,73,213,50,14 - LTEXT "",IDC_ADDRESS,53,117,50,8 - LTEXT "",IDC_R,53,129,50,8 - LTEXT "",IDC_G,53,141,50,8 - LTEXT "",IDC_B,53,153,50,8 - LTEXT "",IDC_VALUE,53,165,50,8 + LTEXT "",IDC_ADDRESS,53,117,50,8,SS_NOPREFIX + LTEXT "",IDC_R,53,129,50,8,SS_NOPREFIX + LTEXT "",IDC_G,53,141,50,8,SS_NOPREFIX + LTEXT "",IDC_B,53,153,50,8,SS_NOPREFIX + LTEXT "",IDC_VALUE,53,165,50,8,SS_NOPREFIX CONTROL "Custom1",IDC_COLOR,"VbaColorControl",WS_TABSTOP,119,117,50,50 CONTROL "PaletteViewBG",IDC_PALETTE_VIEW,"VbaPaletteViewControl",WS_TABSTOP,11,30,64,64 CONTROL "PaletteViewBG",IDC_PALETTE_VIEW_OBJ, @@ -1031,7 +1014,7 @@ BEGIN DEFPUSHBUTTON "OK",ID_OK,31,36,50,14 PUSHBUTTON "Cancel",ID_CANCEL,103,36,50,14 CTEXT "Do you want to keep the current mode?",IDC_STATIC,7,7,172,8 - CTEXT "",IDC_TIMER,7,19,172,8 + CTEXT "",IDC_TIMER,7,19,172,8,SS_NOPREFIX,WS_EX_TOOLWINDOW END IDD_REWIND_INTERVAL DIALOG 0, 0, 186, 68 @@ -1099,19 +1082,26 @@ BEGIN DEFPUSHBUTTON "&Copy",IDC_COPY,87,164,50,14 END -IDD_UNIVIDMODE DIALOGEX 0, 0, 268, 114 -STYLE DS_SETFONT | DS_MODALFRAME | DS_3DLOOK | DS_FIXEDSYS | WS_POPUP | WS_CAPTION | WS_SYSMENU -CAPTION "Video Mode" -FONT 8, "MS Shell Dlg", 400, 0, 0x1 +IDD_GAME_OVERRIDES DIALOGEX 0, 0, 268, 132 +STYLE DS_SETFONT | DS_MODALFRAME | WS_POPUP | WS_CAPTION | WS_SYSMENU +CAPTION "Game overrides" +FONT 8, "MS Sans Serif", 0, 0, 0x0 BEGIN - DEFPUSHBUTTON "OK",IDOK,138,96,60,12 - PUSHBUTTON "Cancel",IDCANCEL,204,96,60,12 - CTEXT "API",IDC_APINAME,138,18,126,12,0,WS_EX_CLIENTEDGE - LISTBOX IDC_LISTMODES,6,18,126,90,LBS_NOINTEGRALHEIGHT | WS_VSCROLL | WS_TABSTOP - CTEXT "Select video mode:",IDC_STATIC,6,6,126,8 - PUSHBUTTON "Device Name",IDC_DISPLAYDEVICE,138,36,126,12 - PUSHBUTTON "Max scale...",IDC_BUTTON_MAXSCALE,138,60,60,12 - CONTROL "Stretch to fit",IDC_CHECK_STRETCHTOFIT,"Button",BS_AUTOCHECKBOX | WS_TABSTOP,204,60,60,12 + COMBOBOX IDC_RTC,84,42,180,12,CBS_DROPDOWNLIST | WS_VSCROLL | WS_TABSTOP + COMBOBOX IDC_SAVE_TYPE,84,60,180,12,CBS_DROPDOWNLIST | WS_VSCROLL | WS_TABSTOP + COMBOBOX IDC_FLASH_SIZE,84,78,180,12,CBS_DROPDOWNLIST | WS_VSCROLL | WS_TABSTOP + COMBOBOX IDC_MIRRORING,84,96,180,12,CBS_DROPDOWNLIST | WS_VSCROLL | WS_TABSTOP + DEFPUSHBUTTON "OK",IDOK,6,114,72,12 + PUSHBUTTON "Defaults",IDC_DEFAULTS,108,114,54,12 + PUSHBUTTON "Cancel",IDCANCEL,192,114,72,12 + LTEXT "Game Code",IDC_STATIC,6,6,72,12 + EDITTEXT IDC_NAME,84,6,180,12,ES_AUTOHSCROLL | WS_DISABLED + LTEXT "Real Time Clock:",IDC_STATIC,6,42,72,12 + LTEXT "Save Type:",IDC_STATIC,6,60,72,12 + LTEXT "Flash Size:",IDC_STATIC,6,78,72,12 + LTEXT "Mirroring:",IDC_STATIC,6,96,72,12 + LTEXT "Comment",IDC_STATIC,6,24,72,12 + EDITTEXT IDC_COMMENT,84,24,180,12,ES_AUTOHSCROLL END @@ -1123,12 +1113,6 @@ END #ifdef APSTUDIO_INVOKED GUIDELINES DESIGNINFO BEGIN - IDD_LINKTAB, DIALOG - BEGIN - RIGHTMARGIN, 249 - BOTTOMMARGIN, 195 - END - IDD_OPENDLG, DIALOG BEGIN RIGHTMARGIN, 165 @@ -1137,25 +1121,25 @@ BEGIN IDD_ABOUT, DIALOG BEGIN LEFTMARGIN, 7 - RIGHTMARGIN, 165 + RIGHTMARGIN, 150 TOPMARGIN, 7 - BOTTOMMARGIN, 95 + BOTTOMMARGIN, 71 END IDD_DIRECTORIES, DIALOG BEGIN LEFTMARGIN, 7 - RIGHTMARGIN, 277 + RIGHTMARGIN, 213 TOPMARGIN, 7 - BOTTOMMARGIN, 122 + BOTTOMMARGIN, 294 END IDD_CONFIG, DIALOG BEGIN LEFTMARGIN, 7 - RIGHTMARGIN, 202 + RIGHTMARGIN, 214 TOPMARGIN, 7 - BOTTOMMARGIN, 223 + BOTTOMMARGIN, 234 END IDD_CHEATS, DIALOG @@ -1193,9 +1177,9 @@ BEGIN IDD_GBA_ROM_INFO, DIALOG BEGIN LEFTMARGIN, 7 - RIGHTMARGIN, 201 + RIGHTMARGIN, 213 TOPMARGIN, 7 - BOTTOMMARGIN, 119 + BOTTOMMARGIN, 135 END IDD_GB_ROM_INFO, DIALOG @@ -1233,9 +1217,9 @@ BEGIN IDD_MOTION_CONFIG, DIALOG BEGIN LEFTMARGIN, 7 - RIGHTMARGIN, 212 + RIGHTMARGIN, 227 TOPMARGIN, 7 - BOTTOMMARGIN, 107 + BOTTOMMARGIN, 100 END IDD_LANG_SELECT, DIALOG @@ -1462,12 +1446,12 @@ BEGIN BOTTOMMARGIN, 179 END - IDD_UNIVIDMODE, DIALOG + IDD_GAME_OVERRIDES, DIALOG BEGIN LEFTMARGIN, 7 RIGHTMARGIN, 261 TOPMARGIN, 7 - BOTTOMMARGIN, 107 + BOTTOMMARGIN, 105 END END #endif // APSTUDIO_INVOKED @@ -1482,12 +1466,12 @@ IDR_MENU MENU BEGIN POPUP "&File" BEGIN - MENUITEM "&Open GBA...", ID_FILE_OPEN - MENUITEM "Open GB...", ID_FILE_OPENGAMEBOY + MENUITEM "Open...", ID_FILE_OPEN + MENUITEM "Open Gameboy...", ID_FILE_OPENGAMEBOY MENUITEM SEPARATOR - MENUITEM "&Load...", ID_FILE_LOAD - MENUITEM "&Save...", ID_FILE_SAVE - POPUP "Loa&d Game" + MENUITEM "Load...", ID_FILE_LOAD + MENUITEM "Save...", ID_FILE_SAVE + POPUP "Load Game" BEGIN MENUITEM "Most recent", ID_FILE_LOADGAME_MOSTRECENT MENUITEM "Auto load most recent", ID_FILE_LOADGAME_AUTOLOADMOSTRECENT @@ -1503,7 +1487,7 @@ BEGIN MENUITEM "Slot #9", ID_FILE_LOADGAME_SLOT9 MENUITEM "Slot #10", ID_FILE_LOADGAME_SLOT10 END - POPUP "S&ave Game" + POPUP "Save Game" BEGIN MENUITEM "Oldest slot", ID_FILE_SAVEGAME_OLDESTSLOT MENUITEM SEPARATOR @@ -1519,35 +1503,35 @@ BEGIN MENUITEM "Slot #10", ID_FILE_SAVEGAME_SLOT10 END MENUITEM SEPARATOR - MENUITEM "&Pause", ID_FILE_PAUSE - MENUITEM "&Reset", ID_FILE_RESET + MENUITEM "Pause", ID_FILE_PAUSE + MENUITEM "Reset", ID_FILE_RESET MENUITEM SEPARATOR - POPUP "Re¢" + POPUP "Recent" BEGIN MENUITEM "&Reset", ID_FILE_RECENT_RESET MENUITEM "&Freeze", ID_FILE_RECENT_FREEZE MENUITEM SEPARATOR END MENUITEM SEPARATOR - POPUP "&Import" + POPUP "Import" BEGIN MENUITEM "&Battery file...", ID_FILE_IMPORT_BATTERYFILE MENUITEM "Gameshark &code file...", ID_FILE_IMPORT_GAMESHARKCODEFILE MENUITEM "&Gameshark Snapshot...", ID_FILE_IMPORT_GAMESHARKSNAPSHOT END - POPUP "E&xport" + POPUP "Export" BEGIN MENUITEM "&Battery file...", ID_FILE_EXPORT_BATTERYFILE MENUITEM "&Gameshark Snapshot...", ID_FILE_EXPORT_GAMESHARKSNAPSHOT END MENUITEM SEPARATOR - MENUITEM "S&creen capture...", ID_FILE_SCREENCAPTURE - MENUITEM "RO&M Header...", ID_FILE_ROMINFORMATION - MENUITEM "&Toggle menu", ID_FILE_TOGGLEMENU + MENUITEM "Screen capture...", ID_FILE_SCREENCAPTURE + MENUITEM "Rom information...", ID_FILE_ROMINFORMATION + MENUITEM "Toggle menu", ID_FILE_TOGGLEMENU MENUITEM SEPARATOR - MENUITEM "&Close", ID_FILE_CLOSE + MENUITEM "Close", ID_FILE_CLOSE MENUITEM SEPARATOR - MENUITEM "&Exit", ID_FILE_EXIT + MENUITEM "Exit", ID_FILE_EXIT END POPUP "&Options" BEGIN @@ -1777,10 +1761,8 @@ BEGIN MENUITEM "EEPROM+Sensor", ID_OPTIONS_EMULATOR_SAVETYPE_EEPROMSENSOR MENUITEM "None", ID_OPTIONS_EMULATOR_SAVETYPE_NONE MENUITEM SEPARATOR - MENUITEM "Flash 64K", ID_OPTIONS_EMULATOR_SAVETYPE_FLASH512K - MENUITEM "Flash 128K", ID_OPTIONS_EMULATOR_SAVETYPE_FLASH1M - MENUITEM SEPARATOR - MENUITEM "Enhanced detection", ID_OPTIONS_EMULATOR_SAVETYPE_ENHANCEDDETECTION + MENUITEM "Flash 64 KB", ID_OPTIONS_EMULATOR_SAVETYPE_FLASH512K + MENUITEM "Flash 128 KB", ID_OPTIONS_EMULATOR_SAVETYPE_FLASH1M END MENUITEM SEPARATOR MENUITEM "&Use BIOS file", ID_OPTIONS_EMULATOR_USEBIOSFILE @@ -1792,13 +1774,13 @@ BEGIN END POPUP "&Gameboy" BEGIN - MENUITEM "B&order", ID_OPTIONS_GAMEBOY_BORDER + MENUITEM "&Border", ID_OPTIONS_GAMEBOY_BORDER MENUITEM "&Printer", ID_OPTIONS_GAMEBOY_PRINTER - MENUITEM "Border Au&tomatic", ID_OPTIONS_GAMEBOY_BORDERAUTOMATIC + MENUITEM "Border Automatic", ID_OPTIONS_GAMEBOY_BORDERAUTOMATIC MENUITEM SEPARATOR MENUITEM "&Automatic", ID_OPTIONS_GAMEBOY_AUTOMATIC MENUITEM "&GBA", ID_OPTIONS_GAMEBOY_GBA - MENUITEM "GB&C", ID_OPTIONS_GAMEBOY_CGB + MENUITEM "&CGB/GBC", ID_OPTIONS_GAMEBOY_CGB MENUITEM "&SGB", ID_OPTIONS_GAMEBOY_SGB MENUITEM "SGB&2", ID_OPTIONS_GAMEBOY_SGB2 MENUITEM "G&B", ID_OPTIONS_GAMEBOY_GB @@ -1827,11 +1809,10 @@ BEGIN MENUITEM "&Search for cheats...", ID_CHEATS_SEARCHFORCHEATS MENUITEM "&Cheat list...", ID_CHEATS_CHEATLIST MENUITEM SEPARATOR + MENUITEM "&Automatic save/load cheats", ID_CHEATS_AUTOMATICSAVELOADCHEATS + MENUITEM "Disable cheats", ID_CHEATS_DISABLECHEATS MENUITEM "&Load cheat list...", ID_CHEATS_LOADCHEATLIST MENUITEM "Sa&ve cheat list...", ID_CHEATS_SAVECHEATLIST - MENUITEM SEPARATOR - MENUITEM "Disable cheats", ID_CHEATS_DISABLECHEATS - MENUITEM "&Automatic save/load cheats", ID_CHEATS_AUTOMATICSAVELOADCHEATS END POPUP "&Tools" BEGIN @@ -1843,19 +1824,6 @@ BEGIN MENUITEM "OAM Viewer...", ID_TOOLS_OAMVIEWER MENUITEM "&Palette Viewer...", ID_TOOLS_PALETTEVIEW MENUITEM "Tile Viewer...", ID_TOOLS_TILEVIEWER - POPUP "Show" - BEGIN - MENUITEM "BG 0", ID_OPTIONS_VIDEO_LAYERS_BG0 - MENUITEM "BG 1", ID_OPTIONS_VIDEO_LAYERS_BG1 - MENUITEM "BG 2", ID_OPTIONS_VIDEO_LAYERS_BG2 - MENUITEM "BG 3", ID_OPTIONS_VIDEO_LAYERS_BG3 - MENUITEM "OBJ", ID_OPTIONS_VIDEO_LAYERS_OBJ - MENUITEM "WIN 0", ID_OPTIONS_VIDEO_LAYERS_WIN0 - MENUITEM "WIN 1", ID_OPTIONS_VIDEO_LAYERS_WIN1 - MENUITEM "OBJ WIN", ID_OPTIONS_VIDEO_LAYERS_OBJWIN - MENUITEM SEPARATOR - MENUITEM "SFX", ID_OPTIONS_VIDEO_DISABLESFX - END MENUITEM SEPARATOR MENUITEM "&Next frame", ID_DEBUG_NEXTFRAME POPUP "GDB" @@ -1888,6 +1856,7 @@ BEGIN BEGIN MENUITEM "Bug Report", ID_HELP_BUGREPORT MENUITEM "FAQ (website)", ID_HELP_FAQ + MENUITEM "License...", ID_HELP_GNUPUBLICLICENSE MENUITEM SEPARATOR MENUITEM "&About...", ID_HELP_ABOUT END @@ -1956,47 +1925,6 @@ BEGIN END -///////////////////////////////////////////////////////////////////////////// -// -// Version -// - -VS_VERSION_INFO VERSIONINFO - FILEVERSION 1,7,4,0 - PRODUCTVERSION 1,7,4,0 - FILEFLAGSMASK 0x37L -#ifdef _DEBUG - FILEFLAGS 0x21L -#else - FILEFLAGS 0x20L -#endif - FILEOS 0x4L - FILETYPE 0x1L - FILESUBTYPE 0x0L -BEGIN - BLOCK "StringFileInfo" - BEGIN - BLOCK "040904b0" - BEGIN - VALUE "Comments", "VisualBoyAdvance comes with NO WARRANTY. Use it at your own risk." - VALUE "CompanyName", "Forgotten and the VBA team" - VALUE "FileDescription", "VBA" - VALUE "FileVersion", "1, 7, 4, 0" - VALUE "InternalName", "VisualBoyAdvance" - VALUE "LegalCopyright", "Copyright © 2005 Forgotten and the VBA team" - VALUE "OriginalFilename", "VisualBoyAdvance.exe" - VALUE "ProductName", "VisualBoyAdvance emulator" - VALUE "ProductVersion", "S1.7.4" - VALUE "SpecialBuild", "S - Changes by Spacy" - END - END - BLOCK "VarFileInfo" - BEGIN - VALUE "Translation", 0x409, 1200 - END -END - - ///////////////////////////////////////////////////////////////////////////// // // String Table @@ -2025,7 +1953,7 @@ END STRINGTABLE BEGIN IDS_ERROR_CREATING_FILE "Error creating file %s" - IDS_FAILED_TO_READ_SGM "Failed to read complete save game %s (%d)" + IDS_FAILED_TO_READ_SGM "Battery file's size incompatible with the rom settings %s (%d).\nWarning : save of the battery file is now disabled !" IDS_FAILED_TO_READ_RTC "Failed to read RTC from save game %s (continuing)" IDS_UNSUPPORTED_VB_SGM "Unsupported VisualBoy save game version %d" IDS_CANNOT_LOAD_SGM_FOR "Cannot load save game for %s. Playing %s" @@ -2164,7 +2092,7 @@ BEGIN IDS_FILTER_GBS "Gameboy Snapshot_*.GBS__" IDS_FILTER_GCF "Gameshark Code File_*.GCF__" IDS_SELECT_CODE_FILE "Select code file" - IDS_SAVE_WILL_BE_LOST "Importing a snapshot file will erase any saved games and reset the emulator. Do you want to continue?" + IDS_SAVE_WILL_BE_LOST "Importing a snapshot file will erase any saved games. Do you want to continue?" IDS_CONFIRM_ACTION "Please confirm action" IDS_CODES_WILL_BE_LOST "Importing a code file will erase any entered codes. Do you want to continue?" IDS_FILTER_SPC "Gameshark Code File_*.SPC;*.XPC__" @@ -2192,6 +2120,7 @@ BEGIN IDS_INVALID_THROTTLE_VALUE "Invalid throttle value. Please enter a number between 5 and 1000." IDS_FILTER_INI "Skin INI File_*.INI__" + IDS_SELECT_SKIN_FILE "Select the skin file name" IDS_FILTER_VMV "VisualBoyAdvance Movie_*.VMV__" IDS_SELECT_MOVIE_NAME "Select movie name" IDS_BUG_REPORT "The bug report information is now available on the Windows Clipboard. Please paste it into any bug report made by email or on forums to help solve problems more easily." @@ -2217,8 +2146,7 @@ END // // Generated from the TEXTINCLUDE 3 resource. // - - +#include "vba.rc2" ///////////////////////////////////////////////////////////////////////////// #endif // not APSTUDIO_INVOKED diff --git a/src/win32/VideoMode.cpp b/src/win32/VideoMode.cpp index 3e084ddc..7fec0cdc 100644 --- a/src/win32/VideoMode.cpp +++ b/src/win32/VideoMode.cpp @@ -1,365 +1,395 @@ -// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. -// Copyright (C) 1999-2003 Forgotten -// Copyright (C) 2004 Forgotten and the VBA development team - -// This program is free software; you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation; either version 2, or(at your option) -// any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program; if not, write to the Free Software Foundation, -// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - -// VideoMode.cpp : implementation file -// - -#include "stdafx.h" -#include "VBA.h" - -#define DIRECTDRAW_VERSION 0x0700 -#include - -#include "VideoMode.h" - -#include "../System.h" -#include "..\..\res\resource.h" - - -#ifdef _DEBUG -#define new DEBUG_NEW -#undef THIS_FILE -static char THIS_FILE[] = __FILE__; -#endif - -#define MAX_DRIVERS 32 // 32 drivers maximum - -//----------------------------------------------------------------------------- -// Local structures -//----------------------------------------------------------------------------- -// Keeps data on the available DDraw drivers -struct -{ - char szDescription[128]; - char szName[128]; - GUID *pGUID; - GUID GUIDcopy; - HMONITOR hm; -} Drivers[MAX_DRIVERS]; - -//----------------------------------------------------------------------------- -// Local data -//----------------------------------------------------------------------------- -static int gDriverCnt = 0; // Total number of drivers -static GUID *gpSelectedDriverGUID; - -//----------------------------------------------------------------------------- -// Name: DDEnumCallbackEx() -// Desc: This call back is used to determine the existing available DDraw -// devices, so the user can pick which one to run on. -//----------------------------------------------------------------------------- -BOOL WINAPI -DDEnumCallbackEx(GUID *pGUID, LPSTR pDescription, LPSTR pName, LPVOID pContext, HMONITOR hm) -{ - if (pGUID) - { - Drivers[gDriverCnt].GUIDcopy = *pGUID; - Drivers[gDriverCnt].pGUID = &Drivers[gDriverCnt].GUIDcopy; - } - else - Drivers[gDriverCnt].pGUID = NULL; - Drivers[gDriverCnt].szDescription[127] = '\0'; - Drivers[gDriverCnt].szName[127] = '\0'; - strncpy(Drivers[gDriverCnt].szDescription,pDescription,127); - strncpy(Drivers[gDriverCnt].szName,pName,127); - Drivers[gDriverCnt].hm = hm; - if (gDriverCnt < MAX_DRIVERS) - gDriverCnt++; - else - return DDENUMRET_CANCEL; - return DDENUMRET_OK; -} - - - - -//----------------------------------------------------------------------------- -// Name: DDEnumCallback() -// Desc: This callback is used only with old versions of DDraw. -//----------------------------------------------------------------------------- -BOOL WINAPI -DDEnumCallback(GUID *pGUID, LPSTR pDescription, LPSTR pName, LPVOID context) -{ - return (DDEnumCallbackEx(pGUID, pDescription, pName, context, NULL)); -} - -static HRESULT WINAPI addVideoMode(LPDDSURFACEDESC2 surf, LPVOID lpContext) -{ - HWND h = (HWND)lpContext; - char buffer[50]; - - switch(surf->ddpfPixelFormat.dwRGBBitCount) { - case 16: - case 24: - case 32: - if(surf->dwWidth >= 640 && surf->dwHeight >= 480) { - sprintf(buffer, "%4dx%4dx%2d", surf->dwWidth, surf->dwHeight, - surf->ddpfPixelFormat.dwRGBBitCount); - int pos = ::SendMessage(h, LB_ADDSTRING, 0, (LPARAM)buffer); - ::SendMessage(h, LB_SETITEMDATA, pos, - (surf->ddpfPixelFormat.dwRGBBitCount << 24) | - ((surf->dwWidth & 4095) << 12) | - (surf->dwHeight & 4095)); - } - } - - return DDENUMRET_OK; -} - -int winVideoModeSelect(CWnd *pWnd, GUID **guid) -{ - HINSTANCE h = LoadLibrary("ddraw.dll"); - - // If ddraw.dll doesn't exist in the search path, - // then DirectX probably isn't installed, so fail. - if (!h) - return -1; - - gDriverCnt = 0; - - // Note that you must know which version of the - // function to retrieve (see the following text). - // For this example, we use the ANSI version. - LPDIRECTDRAWENUMERATEEX lpDDEnumEx; - lpDDEnumEx = (LPDIRECTDRAWENUMERATEEX) - GetProcAddress(h,"DirectDrawEnumerateExA"); - - // If the function is there, call it to enumerate all display - // devices attached to the desktop, and any non-display DirectDraw - // devices. - if (lpDDEnumEx) - lpDDEnumEx(DDEnumCallbackEx, NULL, - DDENUM_ATTACHEDSECONDARYDEVICES | - DDENUM_NONDISPLAYDEVICES - ); - else { - /* - * We must be running on an old version of DirectDraw. - * Therefore MultiMon isn't supported. Fall back on - * DirectDrawEnumerate to enumerate standard devices on a - * single-monitor system. - */ - BOOL (WINAPI *lpDDEnum)(LPDDENUMCALLBACK, LPVOID); - - lpDDEnum = (BOOL (WINAPI *)(LPDDENUMCALLBACK, LPVOID)) - GetProcAddress(h, "DirectDrawEnumerateA"); - if(lpDDEnum) - lpDDEnum(DDEnumCallback,NULL); - - /* Note that it could be handy to let the OldCallback function - * be a wrapper for a DDEnumCallbackEx. - * - * Such a function would look like: - * BOOL FAR PASCAL OldCallback(GUID FAR *lpGUID, - * LPSTR pDesc, - * LPSTR pName, - * LPVOID pContext) - * { - * return Callback(lpGUID,pDesc,pName,pContext,NULL); - * } - */ - } - - int selected = 0; - - if(gDriverCnt > 1) { - VideoDriverSelect d(pWnd); - - selected = d.DoModal(); - - if(selected == -1) { - // If the library was loaded by calling LoadLibrary(), - // then you must use FreeLibrary() to let go of it. - FreeLibrary(h); - - return -1; - } - } - - HRESULT (WINAPI *DDrawCreateEx)(GUID *,LPVOID *,REFIID,IUnknown *); - DDrawCreateEx = (HRESULT (WINAPI *)(GUID *,LPVOID *,REFIID,IUnknown *)) - GetProcAddress(h, "DirectDrawCreateEx"); - - LPDIRECTDRAW7 ddraw = NULL; - if(DDrawCreateEx) { - HRESULT hret = DDrawCreateEx(Drivers[selected].pGUID, - (void **)&ddraw, - IID_IDirectDraw7, - NULL); - if(hret != DD_OK) { - systemMessage(0, "Error during DirectDrawCreateEx: %08x", hret); - FreeLibrary(h); - return -1; - } - } else { - // should not happen.... - systemMessage(0, "Error getting DirectDrawCreateEx"); - FreeLibrary(h); - return -1; - } - - VideoMode dlg(ddraw, pWnd); - - int res = dlg.DoModal(); - - if(res != -1) { - *guid = Drivers[selected].pGUID; - } - ddraw->Release(); - ddraw = NULL; - - // If the library was loaded by calling LoadLibrary(), - // then you must use FreeLibrary() to let go of it. - FreeLibrary(h); - - return res; -} - -///////////////////////////////////////////////////////////////////////////// -// VideoMode dialog - - -VideoMode::VideoMode(LPDIRECTDRAW7 pDraw, CWnd* pParent /*=NULL*/) - : CDialog(VideoMode::IDD, pParent) -{ - //{{AFX_DATA_INIT(VideoMode) - // NOTE: the ClassWizard will add member initialization here - //}}AFX_DATA_INIT - pDirectDraw = pDraw; -} - - -void VideoMode::DoDataExchange(CDataExchange* pDX) -{ - CDialog::DoDataExchange(pDX); - //{{AFX_DATA_MAP(VideoMode) - DDX_Control(pDX, IDC_MODES, m_modes); - //}}AFX_DATA_MAP -} - - -BEGIN_MESSAGE_MAP(VideoMode, CDialog) - //{{AFX_MSG_MAP(VideoMode) - ON_LBN_SELCHANGE(IDC_MODES, OnSelchangeModes) - ON_BN_CLICKED(ID_CANCEL, OnCancel) - ON_BN_CLICKED(ID_OK, OnOk) - //}}AFX_MSG_MAP - END_MESSAGE_MAP() - - ///////////////////////////////////////////////////////////////////////////// -// VideoMode message handlers - -void VideoMode::OnSelchangeModes() -{ - int item = m_modes.GetCurSel(); - - GetDlgItem(ID_OK)->EnableWindow(item != -1); -} - -void VideoMode::OnCancel() -{ - EndDialog(-1); -} - -void VideoMode::OnOk() -{ - int cur = m_modes.GetCurSel(); - - if(cur != -1) { - cur = m_modes.GetItemData(cur); - } - EndDialog(cur); -} - -BOOL VideoMode::OnInitDialog() -{ - CDialog::OnInitDialog(); - - // check for available fullscreen modes - pDirectDraw->EnumDisplayModes(DDEDM_STANDARDVGAMODES, NULL, m_modes.m_hWnd, - addVideoMode); - - GetDlgItem(ID_OK)->EnableWindow(FALSE); - CenterWindow(); - - return TRUE; // return TRUE unless you set the focus to a control - // EXCEPTION: OCX Property Pages should return FALSE -} - -///////////////////////////////////////////////////////////////////////////// -// VideoDriverSelect dialog - - -VideoDriverSelect::VideoDriverSelect(CWnd* pParent /*=NULL*/) - : CDialog(VideoDriverSelect::IDD, pParent) -{ - //{{AFX_DATA_INIT(VideoDriverSelect) - // NOTE: the ClassWizard will add member initialization here - //}}AFX_DATA_INIT -} - - -void VideoDriverSelect::DoDataExchange(CDataExchange* pDX) -{ - CDialog::DoDataExchange(pDX); - //{{AFX_DATA_MAP(VideoDriverSelect) - DDX_Control(pDX, IDC_DRIVERS, m_drivers); - //}}AFX_DATA_MAP -} - - -BEGIN_MESSAGE_MAP(VideoDriverSelect, CDialog) - //{{AFX_MSG_MAP(VideoDriverSelect) - ON_BN_CLICKED(ID_OK, OnOk) - ON_BN_CLICKED(ID_CANCEL, OnCancel) - ON_LBN_SELCHANGE(IDC_DRIVERS, OnSelchangeDrivers) - //}}AFX_MSG_MAP - END_MESSAGE_MAP() - - ///////////////////////////////////////////////////////////////////////////// -// VideoDriverSelect message handlers - -void VideoDriverSelect::OnCancel() -{ - EndDialog(-1); -} - -void VideoDriverSelect::OnOk() -{ - EndDialog(m_drivers.GetCurSel()); -} - -BOOL VideoDriverSelect::OnInitDialog() -{ - CDialog::OnInitDialog(); - - for(int i = 0; i < gDriverCnt; i++) { - m_drivers.AddString(Drivers[i].szDescription); - } - - GetDlgItem(ID_OK)->EnableWindow(FALSE); - CenterWindow(); - - return TRUE; // return TRUE unless you set the focus to a control - // EXCEPTION: OCX Property Pages should return FALSE -} - -void VideoDriverSelect::OnSelchangeDrivers() -{ - GetDlgItem(ID_OK)->EnableWindow(m_drivers.GetCurSel() != -1); -} +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +// VideoMode.cpp : implementation file +// + +#include "stdafx.h" +#include "VBA.h" + +#define DIRECTDRAW_VERSION 0x0700 +#include + +#include "VideoMode.h" + +#include "../System.h" +#include "resource.h" + + +#ifdef _DEBUG +#define new DEBUG_NEW +#undef THIS_FILE +static char THIS_FILE[] = __FILE__; +#endif + +#define MAX_DRIVERS 32 // 32 drivers maximum + +//----------------------------------------------------------------------------- +// Local structures +//----------------------------------------------------------------------------- +// Keeps data on the available DDraw drivers +struct +{ + char szDescription[128]; + char szName[128]; + GUID *pGUID; + GUID GUIDcopy; + HMONITOR hm; +} Drivers[MAX_DRIVERS]; + +//----------------------------------------------------------------------------- +// Local data +//----------------------------------------------------------------------------- +static int gDriverCnt = 0; // Total number of drivers +static GUID *gpSelectedDriverGUID; + +//----------------------------------------------------------------------------- +// Name: DDEnumCallbackEx() +// Desc: This call back is used to determine the existing available DDraw +// devices, so the user can pick which one to run on. +//----------------------------------------------------------------------------- +BOOL WINAPI +DDEnumCallbackEx(GUID *pGUID, LPSTR pDescription, LPSTR pName, LPVOID pContext, HMONITOR hm) +{ + if (pGUID) + { + Drivers[gDriverCnt].GUIDcopy = *pGUID; + Drivers[gDriverCnt].pGUID = &Drivers[gDriverCnt].GUIDcopy; + } + else + Drivers[gDriverCnt].pGUID = NULL; + Drivers[gDriverCnt].szDescription[127] = '\0'; + Drivers[gDriverCnt].szName[127] = '\0'; + strncpy(Drivers[gDriverCnt].szDescription,pDescription,127); + strncpy(Drivers[gDriverCnt].szName,pName,127); + Drivers[gDriverCnt].hm = hm; + if (gDriverCnt < MAX_DRIVERS) + gDriverCnt++; + else + return DDENUMRET_CANCEL; + return DDENUMRET_OK; +} + + + + +//----------------------------------------------------------------------------- +// Name: DDEnumCallback() +// Desc: This callback is used only with old versions of DDraw. +//----------------------------------------------------------------------------- +BOOL WINAPI +DDEnumCallback(GUID *pGUID, LPSTR pDescription, LPSTR pName, LPVOID context) +{ + return (DDEnumCallbackEx(pGUID, pDescription, pName, context, NULL)); +} + +static HRESULT WINAPI addVideoMode(LPDDSURFACEDESC2 surf, LPVOID lpContext) +{ + HWND h = (HWND)lpContext; + char buffer[50]; + + switch( surf->ddpfPixelFormat.dwRGBBitCount ) + { + case 16: + case 24: + case 32: + sprintf( + buffer, + _T("%4dx%4dx%2d"), + surf->dwWidth, + surf->dwHeight, + surf->ddpfPixelFormat.dwRGBBitCount + ); + WPARAM pos = ::SendMessage( h, LB_ADDSTRING, 0, (LPARAM)buffer ); + ::SendMessage( + h, + LB_SETITEMDATA, + pos, + (surf->ddpfPixelFormat.dwRGBBitCount << 24) | + ((surf->dwWidth & 4095) << 12) | + (surf->dwHeight & 4095) + ); + break; + } + + return DDENUMRET_OK; +} + +int winVideoModeSelect(CWnd *pWnd, GUID **guid) +{ +#ifdef _AFXDLL + HINSTANCE h = AfxLoadLibrary("ddraw.dll"); +#else + HMODULE h = LoadLibrary( _T("ddraw.dll") ); +#endif + + // If ddraw.dll doesn't exist in the search path, + // then DirectX probably isn't installed, so fail. + if (!h) + return -1; + + gDriverCnt = 0; + + // Note that you must know which version of the + // function to retrieve (see the following text). + // For this example, we use the ANSI version. + LPDIRECTDRAWENUMERATEEX lpDDEnumEx; + lpDDEnumEx = (LPDIRECTDRAWENUMERATEEX) + GetProcAddress(h,"DirectDrawEnumerateExA"); + + // If the function is there, call it to enumerate all display + // devices attached to the desktop, and any non-display DirectDraw + // devices. + if (lpDDEnumEx) + lpDDEnumEx(DDEnumCallbackEx, NULL, + DDENUM_ATTACHEDSECONDARYDEVICES | + DDENUM_NONDISPLAYDEVICES + ); + else { + /* + * We must be running on an old version of DirectDraw. + * Therefore MultiMon isn't supported. Fall back on + * DirectDrawEnumerate to enumerate standard devices on a + * single-monitor system. + */ + BOOL (WINAPI *lpDDEnum)(LPDDENUMCALLBACK, LPVOID); + + lpDDEnum = (BOOL (WINAPI *)(LPDDENUMCALLBACK, LPVOID)) + GetProcAddress(h, "DirectDrawEnumerateA"); + if(lpDDEnum) + lpDDEnum(DDEnumCallback,NULL); + + /* Note that it could be handy to let the OldCallback function + * be a wrapper for a DDEnumCallbackEx. + * + * Such a function would look like: + * BOOL FAR PASCAL OldCallback(GUID FAR *lpGUID, + * LPSTR pDesc, + * LPSTR pName, + * LPVOID pContext) + * { + * return Callback(lpGUID,pDesc,pName,pContext,NULL); + * } + */ + } + + int selected = 0; + + if(gDriverCnt > 1) { + VideoDriverSelect d(pWnd); + + INT_PTR selected = d.DoModal(); + + if(selected == -1) { +#ifdef _AFXDLL + AfxFreeLibrary( h ); +#else + FreeLibrary( h ); +#endif + + return -1; + } + } + + HRESULT (WINAPI *DDrawCreateEx)(GUID *,LPVOID *,REFIID,IUnknown *); + DDrawCreateEx = (HRESULT (WINAPI *)(GUID *,LPVOID *,REFIID,IUnknown *)) + GetProcAddress(h, "DirectDrawCreateEx"); + + LPDIRECTDRAW7 ddraw = NULL; + if(DDrawCreateEx) { + HRESULT hret = DDrawCreateEx(Drivers[selected].pGUID, + (void **)&ddraw, + IID_IDirectDraw7, + NULL); + if(hret != DD_OK) { + systemMessage(0, "Error during DirectDrawCreateEx: %08x", hret); +#ifdef _AFXDLL + AfxFreeLibrary( h ); +#else + FreeLibrary( h ); +#endif + return -1; + } + } else { + // should not happen.... + systemMessage(0, "Error getting DirectDrawCreateEx"); +#ifdef _AFXDLL + AfxFreeLibrary( h ); +#else + FreeLibrary( h ); +#endif + return -1; + } + + VideoMode dlg(ddraw, pWnd); + + INT_PTR res = dlg.DoModal(); + + if(res != -1) { + *guid = Drivers[selected].pGUID; + } + ddraw->Release(); + ddraw = NULL; + + // If the library was loaded by calling LoadLibrary(), + // then you must use FreeLibrary() to let go of it. +#ifdef _AFXDLL + AfxFreeLibrary( h ); +#else + FreeLibrary( h ); +#endif + + return (int)res; +} + +///////////////////////////////////////////////////////////////////////////// +// VideoMode dialog + + +VideoMode::VideoMode(LPDIRECTDRAW7 pDraw, CWnd* pParent /*=NULL*/) + : CDialog(VideoMode::IDD, pParent) +{ + //{{AFX_DATA_INIT(VideoMode) + // NOTE: the ClassWizard will add member initialization here + //}}AFX_DATA_INIT + pDirectDraw = pDraw; +} + + +void VideoMode::DoDataExchange(CDataExchange* pDX) +{ + CDialog::DoDataExchange(pDX); + //{{AFX_DATA_MAP(VideoMode) + DDX_Control(pDX, IDC_MODES, m_modes); + //}}AFX_DATA_MAP +} + + +BEGIN_MESSAGE_MAP(VideoMode, CDialog) + //{{AFX_MSG_MAP(VideoMode) + ON_LBN_SELCHANGE(IDC_MODES, OnSelchangeModes) + ON_BN_CLICKED(ID_CANCEL, OnCancel) + ON_BN_CLICKED(ID_OK, OnOk) + //}}AFX_MSG_MAP + END_MESSAGE_MAP() + + ///////////////////////////////////////////////////////////////////////////// +// VideoMode message handlers + +void VideoMode::OnSelchangeModes() +{ + int item = m_modes.GetCurSel(); + + GetDlgItem(ID_OK)->EnableWindow(item != -1); +} + +void VideoMode::OnCancel() +{ + EndDialog(-1); +} + +void VideoMode::OnOk() +{ + DWORD_PTR cur = m_modes.GetCurSel(); + + if(cur != -1) { + cur = m_modes.GetItemData((int)cur); + } + EndDialog((int)cur); +} + +BOOL VideoMode::OnInitDialog() +{ + CDialog::OnInitDialog(); + + // check for available fullscreen modes + pDirectDraw->EnumDisplayModes( + DDEDM_STANDARDVGAMODES, + NULL, + m_modes.m_hWnd, + addVideoMode); + + GetDlgItem(ID_OK)->EnableWindow(FALSE); + CenterWindow(); + + return TRUE; // return TRUE unless you set the focus to a control + // EXCEPTION: OCX Property Pages should return FALSE +} + +///////////////////////////////////////////////////////////////////////////// +// VideoDriverSelect dialog + + +VideoDriverSelect::VideoDriverSelect(CWnd* pParent /*=NULL*/) + : CDialog(VideoDriverSelect::IDD, pParent) +{ + //{{AFX_DATA_INIT(VideoDriverSelect) + // NOTE: the ClassWizard will add member initialization here + //}}AFX_DATA_INIT +} + + +void VideoDriverSelect::DoDataExchange(CDataExchange* pDX) +{ + CDialog::DoDataExchange(pDX); + //{{AFX_DATA_MAP(VideoDriverSelect) + DDX_Control(pDX, IDC_DRIVERS, m_drivers); + //}}AFX_DATA_MAP +} + + +BEGIN_MESSAGE_MAP(VideoDriverSelect, CDialog) + //{{AFX_MSG_MAP(VideoDriverSelect) + ON_BN_CLICKED(ID_OK, OnOk) + ON_BN_CLICKED(ID_CANCEL, OnCancel) + ON_LBN_SELCHANGE(IDC_DRIVERS, OnSelchangeDrivers) + //}}AFX_MSG_MAP + END_MESSAGE_MAP() + + ///////////////////////////////////////////////////////////////////////////// +// VideoDriverSelect message handlers + +void VideoDriverSelect::OnCancel() +{ + EndDialog(-1); +} + +void VideoDriverSelect::OnOk() +{ + EndDialog(m_drivers.GetCurSel()); +} + +BOOL VideoDriverSelect::OnInitDialog() +{ + CDialog::OnInitDialog(); + + for(int i = 0; i < gDriverCnt; i++) { + m_drivers.AddString(Drivers[i].szDescription); + } + + GetDlgItem(ID_OK)->EnableWindow(FALSE); + CenterWindow(); + + return TRUE; // return TRUE unless you set the focus to a control + // EXCEPTION: OCX Property Pages should return FALSE +} + +void VideoDriverSelect::OnSelchangeDrivers() +{ + GetDlgItem(ID_OK)->EnableWindow(m_drivers.GetCurSel() != -1); +} diff --git a/src/win32/ZoomControl.h b/src/win32/ZoomControl.h index d70b4c98..7a6c7dcd 100644 --- a/src/win32/ZoomControl.h +++ b/src/win32/ZoomControl.h @@ -20,7 +20,7 @@ #if !defined(AFX_ZOOMCONTROL_H__BC193230_D2D6_4240_93AE_28C2EF2C641A__INCLUDED_) #define AFX_ZOOMCONTROL_H__BC193230_D2D6_4240_93AE_28C2EF2C641A__INCLUDED_ -#include "..\System.h" // Added by ClassView +#include "../System.h" // Added by ClassView #if _MSC_VER > 1000 #pragma once #endif // _MSC_VER > 1000 diff --git a/src/win32/dependencies/CVS/Entries b/src/win32/dependencies/CVS/Entries new file mode 100644 index 00000000..b2877be7 --- /dev/null +++ b/src/win32/dependencies/CVS/Entries @@ -0,0 +1,5 @@ +/info.txt/1.1/Fri May 12 21:26:37 2006// +D/cximage//// +D/libpng//// +D/sdl//// +D/zlib//// diff --git a/src/win32/dependencies/CVS/Repository b/src/win32/dependencies/CVS/Repository new file mode 100644 index 00000000..e7e98f4c --- /dev/null +++ b/src/win32/dependencies/CVS/Repository @@ -0,0 +1 @@ +VisualBoyAdvance/win32/dependencies diff --git a/src/win32/dependencies/CVS/Root b/src/win32/dependencies/CVS/Root new file mode 100644 index 00000000..6ceab0dd --- /dev/null +++ b/src/win32/dependencies/CVS/Root @@ -0,0 +1 @@ +:pserver:anonymous@vba.cvs.sourceforge.net:/cvsroot/vba diff --git a/src/win32/dependencies/cximage/CVS/Entries b/src/win32/dependencies/cximage/CVS/Entries new file mode 100644 index 00000000..60478339 --- /dev/null +++ b/src/win32/dependencies/cximage/CVS/Entries @@ -0,0 +1,55 @@ +/cximage.vcproj/1.3/Wed Aug 23 22:16:02 2006// +/license.txt/1.1/Wed Aug 23 22:16:02 2006// +/tif_xfile.cpp/1.3/Wed Aug 23 22:16:02 2006// +/xfile.h/1.3/Wed Aug 23 22:16:02 2006// +/ximabmp.cpp/1.3/Wed Aug 23 22:16:02 2006// +/ximabmp.h/1.3/Wed Aug 23 22:16:02 2006// +/ximacfg.h/1.3/Wed Aug 23 22:16:02 2006// +/ximadef.h/1.3/Wed Aug 23 22:16:02 2006// +/ximadsp.cpp/1.3/Wed Aug 23 22:16:02 2006// +/ximaenc.cpp/1.3/Wed Aug 23 22:16:02 2006// +/ximaexif.cpp/1.3/Wed Aug 23 22:16:02 2006// +/ximage.cpp/1.3/Wed Aug 23 22:16:02 2006// +/ximage.h/1.3/Wed Aug 23 22:16:02 2006// +/ximagif.cpp/1.3/Wed Aug 23 22:16:02 2006// +/ximagif.h/1.3/Wed Aug 23 22:16:02 2006// +/ximahist.cpp/1.3/Wed Aug 23 22:16:02 2006// +/ximaico.cpp/1.3/Wed Aug 23 22:16:02 2006// +/ximaico.h/1.3/Wed Aug 23 22:16:02 2006// +/ximainfo.cpp/1.3/Wed Aug 23 22:16:02 2006// +/ximaint.cpp/1.3/Wed Aug 23 22:16:02 2006// +/ximaiter.h/1.3/Wed Aug 23 22:16:02 2006// +/ximaj2k.cpp/1.3/Wed Aug 23 22:16:02 2006// +/ximaj2k.h/1.3/Wed Aug 23 22:16:02 2006// +/ximajas.cpp/1.3/Wed Aug 23 22:16:02 2006// +/ximajas.h/1.3/Wed Aug 23 22:16:03 2006// +/ximajbg.cpp/1.3/Wed Aug 23 22:16:03 2006// +/ximajbg.h/1.3/Wed Aug 23 22:16:03 2006// +/ximajpg.cpp/1.3/Wed Aug 23 22:16:03 2006// +/ximajpg.h/1.3/Wed Aug 23 22:16:03 2006// +/ximalpha.cpp/1.3/Wed Aug 23 22:16:03 2006// +/ximalyr.cpp/1.3/Wed Aug 23 22:16:03 2006// +/ximamng.cpp/1.3/Wed Aug 23 22:16:03 2006// +/ximamng.h/1.3/Wed Aug 23 22:16:03 2006// +/ximapal.cpp/1.3/Wed Aug 23 22:16:03 2006// +/ximapcx.cpp/1.3/Wed Aug 23 22:16:03 2006// +/ximapcx.h/1.3/Wed Aug 23 22:16:03 2006// +/ximapng.cpp/1.3/Wed Aug 23 22:16:03 2006// +/ximapng.h/1.3/Wed Aug 23 22:16:03 2006// +/ximasel.cpp/1.3/Wed Aug 23 22:16:03 2006// +/ximatga.cpp/1.3/Wed Aug 23 22:16:03 2006// +/ximatga.h/1.3/Wed Aug 23 22:16:03 2006// +/ximath.cpp/1.3/Wed Aug 23 22:16:03 2006// +/ximath.h/1.3/Wed Aug 23 22:16:03 2006// +/ximatif.cpp/1.3/Wed Aug 23 22:16:03 2006// +/ximatif.h/1.3/Wed Aug 23 22:16:03 2006// +/ximatran.cpp/1.3/Wed Aug 23 22:16:03 2006// +/ximawbmp.cpp/1.3/Wed Aug 23 22:16:03 2006// +/ximawbmp.h/1.3/Wed Aug 23 22:16:03 2006// +/ximawmf.cpp/1.3/Wed Aug 23 22:16:03 2006// +/ximawmf.h/1.3/Wed Aug 23 22:16:03 2006// +/ximawnd.cpp/1.3/Wed Aug 23 22:16:03 2006// +/xiofile.h/1.3/Wed Aug 23 22:16:03 2006// +/xmemfile.cpp/1.3/Wed Aug 23 22:16:03 2006// +/xmemfile.h/1.3/Wed Aug 23 22:16:03 2006// +D diff --git a/src/win32/dependencies/cximage/CVS/Repository b/src/win32/dependencies/cximage/CVS/Repository new file mode 100644 index 00000000..aa0db2ac --- /dev/null +++ b/src/win32/dependencies/cximage/CVS/Repository @@ -0,0 +1 @@ +VisualBoyAdvance/win32/dependencies/cximage diff --git a/src/win32/dependencies/cximage/CVS/Root b/src/win32/dependencies/cximage/CVS/Root new file mode 100644 index 00000000..6ceab0dd --- /dev/null +++ b/src/win32/dependencies/cximage/CVS/Root @@ -0,0 +1 @@ +:pserver:anonymous@vba.cvs.sourceforge.net:/cvsroot/vba diff --git a/src/win32/dependencies/cximage/cximage.vcproj b/src/win32/dependencies/cximage/cximage.vcproj new file mode 100644 index 00000000..9799829a --- /dev/null +++ b/src/win32/dependencies/cximage/cximage.vcproj @@ -0,0 +1,367 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/win32/dependencies/cximage/license.txt b/src/win32/dependencies/cximage/license.txt new file mode 100644 index 00000000..755e2c41 --- /dev/null +++ b/src/win32/dependencies/cximage/license.txt @@ -0,0 +1,48 @@ +This copy of the CxImage notices is provided for your convenience. In case of +any discrepancy between this copy and the notices in the file ximage.h that is +included in the CxImage distribution, the latter shall prevail. + +If you modify CxImage you may insert additional notices immediately following +this sentence. + +-------------------------------------------------------------------------------- + +COPYRIGHT NOTICE, DISCLAIMER, and LICENSE: + +CxImage version 5.99c 17/Oct/2004 + +CxImage : Copyright (C) 2001 - 2004, Davide Pizzolato + +Original CImage and CImageIterator implementation are: +Copyright (C) 1995, Alejandro Aguilar Sierra (asierra(at)servidor(dot)unam(dot)mx) + +Covered code is provided under this license on an "as is" basis, without warranty +of any kind, either expressed or implied, including, without limitation, warranties +that the covered code is free of defects, merchantable, fit for a particular purpose +or non-infringing. The entire risk as to the quality and performance of the covered +code is with you. Should any covered code prove defective in any respect, you (not +the initial developer or any other contributor) assume the cost of any necessary +servicing, repair or correction. This disclaimer of warranty constitutes an essential +part of this license. No use of any covered code is authorized hereunder except under +this disclaimer. + +Permission is hereby granted to use, copy, modify, and distribute this +source code, or portions hereof, for any purpose, including commercial applications, +freely and without fee, subject to the following restrictions: + +1. The origin of this software must not be misrepresented; you must not +claim that you wrote the original software. If you use this software +in a product, an acknowledgment in the product documentation would be +appreciated but is not required. + +2. Altered source versions must be plainly marked as such, and must not be +misrepresented as being the original software. + +3. This notice may not be removed or altered from any source distribution. + +-------------------------------------------------------------------------------- + +Other information: about CxImage, and the latest version, can be found at the +CxImage home page: http://www.xdp.it + +-------------------------------------------------------------------------------- diff --git a/src/win32/dependencies/cximage/tif_xfile.cpp b/src/win32/dependencies/cximage/tif_xfile.cpp new file mode 100644 index 00000000..edfc943e --- /dev/null +++ b/src/win32/dependencies/cximage/tif_xfile.cpp @@ -0,0 +1,208 @@ +/* + * TIFF file IO, using CxFile. + */ + +#ifdef WIN32 + #include +#endif +#include + +#include "ximage.h" + +#if CXIMAGE_SUPPORT_TIF + +#include "../tiff/tiffiop.h" + +#include "xfile.h" + +static tsize_t +_tiffReadProcEx(thandle_t fd, tdata_t buf, tsize_t size) +{ + return ((CxFile*)fd)->Read(buf, 1, size); +} + +static tsize_t +_tiffWriteProcEx(thandle_t fd, tdata_t buf, tsize_t size) +{ + return ((CxFile*)fd)->Write(buf, 1, size); +} + +static toff_t +_tiffSeekProcEx(thandle_t fd, toff_t off, int whence) +{ + if ( off == 0xFFFFFFFF ) + return 0xFFFFFFFF; + if (!((CxFile*)fd)->Seek(off, whence)) + return 0xFFFFFFFF; + if (whence == SEEK_SET) + return off; + + return (toff_t)((CxFile*)fd)->Tell(); +} + +// Return nonzero if error +static int +_tiffCloseProcEx(thandle_t fd) +{ +// return !((CxFile*)fd)->Close(); // "//" needed for memory files + return 0; +} + +#include + +static toff_t +_tiffSizeProcEx(thandle_t fd) +{ + return ((CxFile*)fd)->Size(); +} + +static int +_tiffMapProcEx(thandle_t fd, tdata_t* pbase, toff_t* psize) +{ + return (0); +} + +static void +_tiffUnmapProcEx(thandle_t fd, tdata_t base, toff_t size) +{ +} + +// Open a TIFF file descriptor for read/writing. +/* +TIFF* +TIFFOpen(const char* name, const char* mode) +{ + static const char module[] = "TIFFOpen"; + FILE* stream = fopen(name, mode); + if (stream == NULL) + { + TIFFError(module, "%s: Cannot open", name); + return NULL; + } + return (TIFFFdOpen((int)stream, name, mode)); +} +*/ + +TIFF* +_TIFFFdOpen(int fd, const char* name, const char* mode) +{ + TIFF* tif; + + tif = TIFFClientOpen(name, mode, + (thandle_t) fd, + _tiffReadProcEx, _tiffWriteProcEx, _tiffSeekProcEx, _tiffCloseProcEx, + _tiffSizeProcEx, _tiffMapProcEx, _tiffUnmapProcEx); + if (tif) + tif->tif_fd = fd; + return (tif); +} + +extern "C" TIFF* _TIFFOpenEx(CxFile* stream, const char* mode) +{ + return (_TIFFFdOpen((int)stream, "TIFF IMAGE", mode)); +} + +#ifdef __GNUC__ +extern char* malloc(); +extern char* realloc(); +#else +#include +#endif + +tdata_t +_TIFFmalloc(tsize_t s) +{ + return (malloc((size_t) s)); +} + +void +_TIFFfree(tdata_t p) +{ + free(p); +} + +tdata_t +_TIFFrealloc(tdata_t p, tsize_t s) +{ + return (realloc(p, (size_t) s)); +} + +void +_TIFFmemset(tdata_t p, int v, tsize_t c) +{ + memset(p, v, (size_t) c); +} + +void +_TIFFmemcpy(tdata_t d, const tdata_t s, tsize_t c) +{ + memcpy(d, s, (size_t) c); +} + +int +_TIFFmemcmp(const tdata_t p1, const tdata_t p2, tsize_t c) +{ + return (memcmp(p1, p2, (size_t) c)); +} + +static void +Win32WarningHandler(const char* module, const char* fmt, va_list ap) +{ +#ifdef _DEBUG +#if (!defined(_CONSOLE) && defined(WIN32)) + LPTSTR szTitle; + LPTSTR szTmp; + LPCTSTR szTitleText = "%s Warning"; + LPCTSTR szDefaultModule = "TIFFLIB"; + szTmp = (module == NULL) ? (LPTSTR)szDefaultModule : (LPTSTR)module; + if ((szTitle = (LPTSTR)LocalAlloc(LMEM_FIXED, (lstrlen(szTmp) + + lstrlen(szTitleText) + lstrlen(fmt) + 128)*sizeof(TCHAR))) == NULL) + return; + wsprintf(szTitle, szTitleText, szTmp); + szTmp = szTitle + (lstrlen(szTitle)+2)*sizeof(TCHAR); + wvsprintf(szTmp, fmt, ap); + MessageBox(GetFocus(), szTmp, szTitle, MB_OK | MB_ICONINFORMATION); + LocalFree(szTitle); + return; +#else + if (module != NULL) + fprintf(stderr, "%s: ", module); + fprintf(stderr, "Warning, "); + vfprintf(stderr, fmt, ap); + fprintf(stderr, ".\n"); +#endif +#endif +} +TIFFErrorHandler _TIFFwarningHandler = Win32WarningHandler; + +static void +Win32ErrorHandler(const char* module, const char* fmt, va_list ap) +{ +#ifdef _DEBUG +#if (!defined(_CONSOLE) && defined(WIN32)) + LPTSTR szTitle; + LPTSTR szTmp; + LPCTSTR szTitleText = "%s Error"; + LPCTSTR szDefaultModule = "TIFFLIB"; + szTmp = (module == NULL) ? (LPTSTR)szDefaultModule : (LPTSTR)module; + if ((szTitle = (LPTSTR)LocalAlloc(LMEM_FIXED, (lstrlen(szTmp) + + lstrlen(szTitleText) + lstrlen(fmt) + 128)*sizeof(TCHAR))) == NULL) + return; + wsprintf(szTitle, szTitleText, szTmp); + szTmp = szTitle + (lstrlen(szTitle)+2)*sizeof(TCHAR); + wvsprintf(szTmp, fmt, ap); + MessageBox(GetFocus(), szTmp, szTitle, MB_OK | MB_ICONEXCLAMATION); + LocalFree(szTitle); + return; +#else + if (module != NULL) + fprintf(stderr, "%s: ", module); + vfprintf(stderr, fmt, ap); + fprintf(stderr, ".\n"); +#endif +#endif +} +TIFFErrorHandler _TIFFerrorHandler = Win32ErrorHandler; + +#endif + diff --git a/src/win32/dependencies/cximage/xfile.h b/src/win32/dependencies/cximage/xfile.h new file mode 100644 index 00000000..af10b031 --- /dev/null +++ b/src/win32/dependencies/cximage/xfile.h @@ -0,0 +1,76 @@ +/* + * File: xfile.h + * Purpose: General Purpose File Class + */ +/* + -------------------------------------------------------------------------------- + + COPYRIGHT NOTICE, DISCLAIMER, and LICENSE: + + CxFile (c) 11/May/2002 Davide Pizzolato - www.xdp.it + CxFile version 2.00 23/Aug/2002 + + Special thanks to Chris Shearer Cooper for new features, enhancements and bugfixes + + Covered code is provided under this license on an "as is" basis, without warranty + of any kind, either expressed or implied, including, without limitation, warranties + that the covered code is free of defects, merchantable, fit for a particular purpose + or non-infringing. The entire risk as to the quality and performance of the covered + code is with you. Should any covered code prove defective in any respect, you (not + the initial developer or any other contributor) assume the cost of any necessary + servicing, repair or correction. This disclaimer of warranty constitutes an essential + part of this license. No use of any covered code is authorized hereunder except under + this disclaimer. + + Permission is hereby granted to use, copy, modify, and distribute this + source code, or portions hereof, for any purpose, including commercial applications, + freely and without fee, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + + 3. This notice may not be removed or altered from any source distribution. + -------------------------------------------------------------------------------- + */ +#if !defined(__xfile_h) +#define __xfile_h + +#ifdef WIN32 + #include +#endif + +#include +#include + +#include "ximadef.h" + +class DLL_EXP CxFile +{ +public: + CxFile(void) { }; + virtual ~CxFile() { }; + + virtual bool Close() = 0; + virtual size_t Read(void *buffer, size_t size, size_t count) = 0; + virtual size_t Write(const void *buffer, size_t size, size_t count) = 0; + virtual bool Seek(long offset, int origin) = 0; + virtual long Tell() = 0; + virtual long Size() = 0; + virtual bool Flush() = 0; + virtual bool Eof() = 0; + virtual long Error() = 0; + virtual bool PutC(unsigned char c) + { + // Default implementation + size_t nWrote = Write(&c, 1, 1); + return (bool)(nWrote == 1); + } + virtual long GetC() = 0; +}; + +#endif //__xfile_h diff --git a/src/win32/dependencies/cximage/ximabmp.cpp b/src/win32/dependencies/cximage/ximabmp.cpp new file mode 100644 index 00000000..f723ded2 --- /dev/null +++ b/src/win32/dependencies/cximage/ximabmp.cpp @@ -0,0 +1,361 @@ +/* + * File: ximabmp.cpp + * Purpose: Platform Independent BMP Image Class Loader and Writer + * 07/Aug/2001 Davide Pizzolato - www.xdp.it + * CxImage version 5.99c 17/Oct/2004 + */ + +#include "ximabmp.h" + +#if CXIMAGE_SUPPORT_BMP + +#include "ximaiter.h" + +//////////////////////////////////////////////////////////////////////////////// +#if CXIMAGE_SUPPORT_ENCODE +//////////////////////////////////////////////////////////////////////////////// +bool CxImageBMP::Encode(CxFile * hFile) +{ + + if (EncodeSafeCheck(hFile)) return false; + + BITMAPFILEHEADER hdr; + + hdr.bfType = 0x4d42; // 'BM' WINDOWS_BITMAP_SIGNATURE + hdr.bfSize = GetSize() + 14 /*sizeof(BITMAPFILEHEADER)*/; + hdr.bfReserved1 = hdr.bfReserved2 = 0; + hdr.bfOffBits = 14 /*sizeof(BITMAPFILEHEADER)*/ + head.biSize + GetPaletteSize(); + + //copy attributes + memcpy(pDib,&head,sizeof(BITMAPINFOHEADER)); + // Write the file header + hFile->Write(&hdr,min(14,sizeof(BITMAPFILEHEADER)),1); + // Write the DIB header and the pixels + hFile->Write(pDib,GetSize(),1); + return true; +} +//////////////////////////////////////////////////////////////////////////////// +#endif //CXIMAGE_SUPPORT_ENCODE +//////////////////////////////////////////////////////////////////////////////// +#if CXIMAGE_SUPPORT_DECODE +//////////////////////////////////////////////////////////////////////////////// +bool CxImageBMP::Decode(CxFile * hFile) +{ + if (hFile == NULL) return false; + + BITMAPFILEHEADER bf; + DWORD off = hFile->Tell(); // + try { + if (hFile->Read(&bf,min(14,sizeof(bf)),1)==0) throw "Not a BMP"; + if (bf.bfType != BFT_BITMAP) { //do we have a RC HEADER? + bf.bfOffBits = 0L; + hFile->Seek(off,SEEK_SET); + } + + BITMAPINFOHEADER bmpHeader; + if (!DibReadBitmapInfo(hFile,&bmpHeader)) throw "Error reading BMP info"; + DWORD dwCompression=bmpHeader.biCompression; + DWORD dwBitCount=bmpHeader.biBitCount; //preserve for BI_BITFIELDS compression + bool bIsOldBmp = bmpHeader.biSize == sizeof(BITMAPCOREHEADER); + + bool bTopDownDib = bmpHeader.biHeight<0; // check if it's a top-down bitmap + if (bTopDownDib) bmpHeader.biHeight=-bmpHeader.biHeight; + + if (info.nEscape == -1) { + // Return output dimensions only + head.biWidth = bmpHeader.biWidth; + head.biHeight = bmpHeader.biHeight; + throw "output dimensions returned"; + } + + if (!Create(bmpHeader.biWidth,bmpHeader.biHeight,bmpHeader.biBitCount,CXIMAGE_FORMAT_BMP)) + throw "Can't allocate memory"; + + head.biXPelsPerMeter = bmpHeader.biXPelsPerMeter; + head.biYPelsPerMeter = bmpHeader.biYPelsPerMeter; + info.xDPI = (long) floor(bmpHeader.biXPelsPerMeter * 254.0 / 10000.0 + 0.5); + info.yDPI = (long) floor(bmpHeader.biYPelsPerMeter * 254.0 / 10000.0 + 0.5); + + if (info.nEscape) throw "Cancelled"; // - cancel decoding + + RGBQUAD *pRgb = GetPalette(); + if (pRgb){ + if (bIsOldBmp){ + // convert a old color table (3 byte entries) to a new + // color table (4 byte entries) + hFile->Read((void*)pRgb,DibNumColors(&bmpHeader) * sizeof(RGBTRIPLE),1); + for (int i=DibNumColors(&head)-1; i>=0; i--){ + pRgb[i].rgbRed = ((RGBTRIPLE *)pRgb)[i].rgbtRed; + pRgb[i].rgbBlue = ((RGBTRIPLE *)pRgb)[i].rgbtBlue; + pRgb[i].rgbGreen = ((RGBTRIPLE *)pRgb)[i].rgbtGreen; + pRgb[i].rgbReserved = (BYTE)0; + } + } else { + hFile->Read((void*)pRgb,DibNumColors(&bmpHeader) * sizeof(RGBQUAD),1); + //force rgbReserved=0, to avoid problems with some WinXp bitmaps + for (unsigned int i=0; i - cancel decoding + + switch (dwBitCount) { + case 32 : + if (bf.bfOffBits != 0L) hFile->Seek(off + bf.bfOffBits,SEEK_SET); + if (dwCompression == BI_BITFIELDS || dwCompression == BI_RGB){ + long imagesize=4*head.biHeight*head.biWidth; + BYTE* buff32=(BYTE*)malloc(imagesize); + if (buff32){ + hFile->Read(buff32, imagesize,1); // read in the pixels + Bitfield2RGB(buff32,0,0,0,32); + free(buff32); + } else throw "can't allocate memory"; + } else throw "unknown compression"; + break; + case 24 : + if (bf.bfOffBits != 0L) hFile->Seek(off + bf.bfOffBits,SEEK_SET); + if (dwCompression == BI_RGB){ + hFile->Read(info.pImage, head.biSizeImage,1); // read in the pixels + } else throw "unknown compression"; + break; + case 16 : + { + DWORD bfmask[3]; + if (dwCompression == BI_BITFIELDS) + { + hFile->Read(bfmask, 12, 1); + } else { + bfmask[0]=0x7C00; bfmask[1]=0x3E0; bfmask[2]=0x1F; //RGB555 + } + // bf.bfOffBits required after the bitfield mask + if (bf.bfOffBits != 0L) hFile->Seek(off + bf.bfOffBits,SEEK_SET); + // read in the pixels + hFile->Read(info.pImage, head.biHeight*((head.biWidth+1)/2)*4,1); + // transform into RGB + Bitfield2RGB(info.pImage,(WORD)bfmask[0],(WORD)bfmask[1],(WORD)bfmask[2],16); + break; + } + case 8 : + case 4 : + case 1 : + if (bf.bfOffBits != 0L) hFile->Seek(off + bf.bfOffBits,SEEK_SET); + switch (dwCompression) { + case BI_RGB : + hFile->Read(info.pImage, head.biSizeImage,1); // read in the pixels + break; + case BI_RLE4 : + { + BYTE status_byte = 0; + BYTE second_byte = 0; + int scanline = 0; + int bits = 0; + BOOL low_nibble = FALSE; + CImageIterator iter(this); + + for (BOOL bContinue = TRUE; bContinue;) { + hFile->Read(&status_byte, sizeof(BYTE), 1); + switch (status_byte) { + case RLE_COMMAND : + hFile->Read(&status_byte, sizeof(BYTE), 1); + switch (status_byte) { + case RLE_ENDOFLINE : + bits = 0; + scanline++; + low_nibble = FALSE; + break; + case RLE_ENDOFBITMAP : + bContinue=FALSE; + break; + case RLE_DELTA : + { + // read the delta values + BYTE delta_x; + BYTE delta_y; + hFile->Read(&delta_x, sizeof(BYTE), 1); + hFile->Read(&delta_y, sizeof(BYTE), 1); + // apply them + bits += delta_x / 2; + scanline += delta_y; + break; + } + default : + hFile->Read(&second_byte, sizeof(BYTE), 1); + BYTE *sline = iter.GetRow(scanline); + for (int i = 0; i < status_byte; i++) { + if (low_nibble) { + if ((DWORD)(sline+bits) < (DWORD)(info.pImage+head.biSizeImage)){ + *(sline + bits) |= (second_byte & 0x0F); + } + if (i != status_byte - 1) + hFile->Read(&second_byte, sizeof(BYTE), 1); + bits++; + } else { + if ((DWORD)(sline+bits) < (DWORD)(info.pImage+head.biSizeImage)){ + *(sline + bits) = (BYTE)(second_byte & 0xF0); + } + } + low_nibble = !low_nibble; + } + if ((((status_byte+1) >> 1) & 1 )== 1) + hFile->Read(&second_byte, sizeof(BYTE), 1); + break; + }; + break; + default : + { + BYTE *sline = iter.GetRow(scanline); + hFile->Read(&second_byte, sizeof(BYTE), 1); + for (unsigned i = 0; i < status_byte; i++) { + if (low_nibble) { + if ((DWORD)(sline+bits) < (DWORD)(info.pImage+head.biSizeImage)){ + *(sline + bits) |= (second_byte & 0x0F); + } + bits++; + } else { + if ((DWORD)(sline+bits) < (DWORD)(info.pImage+head.biSizeImage)){ + *(sline + bits) = (BYTE)(second_byte & 0xF0); + } + } + low_nibble = !low_nibble; + } + } + break; + }; + } + break; + } + case BI_RLE8 : + { + BYTE status_byte = 0; + BYTE second_byte = 0; + int scanline = 0; + int bits = 0; + CImageIterator iter(this); + + for (BOOL bContinue = TRUE; bContinue; ) { + hFile->Read(&status_byte, sizeof(BYTE), 1); + switch (status_byte) { + case RLE_COMMAND : + hFile->Read(&status_byte, sizeof(BYTE), 1); + switch (status_byte) { + case RLE_ENDOFLINE : + bits = 0; + scanline++; + break; + case RLE_ENDOFBITMAP : + bContinue=FALSE; + break; + case RLE_DELTA : + { + // read the delta values + BYTE delta_x; + BYTE delta_y; + hFile->Read(&delta_x, sizeof(BYTE), 1); + hFile->Read(&delta_y, sizeof(BYTE), 1); + // apply them + bits += delta_x; + scanline += delta_y; + break; + } + default : + hFile->Read((void *)(iter.GetRow(scanline) + bits), sizeof(BYTE) * status_byte, 1); + // align run length to even number of bytes + if ((status_byte & 1) == 1) + hFile->Read(&second_byte, sizeof(BYTE), 1); + bits += status_byte; + break; + }; + break; + default : + BYTE *sline = iter.GetRow(scanline); + hFile->Read(&second_byte, sizeof(BYTE), 1); + for (unsigned i = 0; i < status_byte; i++) { + if ((DWORD)bits + + } catch (char *message) { + strncpy(info.szLastError,message,255); + if (info.nEscape==-1) return true; + return false; + } + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/* ReadDibBitmapInfo() + * + * Will read a file in DIB format and return a global HANDLE to its + * BITMAPINFO. This function will work with both "old" and "new" + * bitmap formats, but will always return a "new" BITMAPINFO. + */ +bool CxImageBMP::DibReadBitmapInfo(CxFile* fh, BITMAPINFOHEADER *pdib) +{ + if ((fh==NULL)||(pdib==NULL)) return false; + + if (fh->Read(pdib,sizeof(BITMAPINFOHEADER),1)==0) return false; + + BITMAPCOREHEADER bc; + + switch (pdib->biSize) // what type of bitmap info is this? + { + case sizeof(BITMAPINFOHEADER): + break; + + case 64: //sizeof(OS2_BMP_HEADER): + fh->Seek((long)(64 - sizeof(BITMAPINFOHEADER)),SEEK_CUR); + break; + + case sizeof(BITMAPCOREHEADER): + bc = *(BITMAPCOREHEADER*)pdib; + pdib->biSize = bc.bcSize; + pdib->biWidth = (DWORD)bc.bcWidth; + pdib->biHeight = (DWORD)bc.bcHeight; + pdib->biPlanes = bc.bcPlanes; + pdib->biBitCount = bc.bcBitCount; + pdib->biCompression = BI_RGB; + pdib->biSizeImage = 0; + pdib->biXPelsPerMeter = 0; + pdib->biYPelsPerMeter = 0; + pdib->biClrUsed = 0; + pdib->biClrImportant = 0; + + fh->Seek((long)(sizeof(BITMAPCOREHEADER)-sizeof(BITMAPINFOHEADER)), SEEK_CUR); + + break; + default: + //give a last chance + if (pdib->biSize>(sizeof(BITMAPINFOHEADER))&& + (pdib->biSizeImage==(unsigned long)(pdib->biHeight*((((pdib->biBitCount*pdib->biWidth)+31)/32)*4)))&& + (pdib->biPlanes==1)&&(pdib->biCompression==BI_RGB)&&(pdib->biClrUsed==0)) + { + fh->Seek((long)(pdib->biSize - sizeof(BITMAPINFOHEADER)),SEEK_CUR); + break; + } + return false; + } + + FixBitmapInfo(pdib); + + return true; +} +//////////////////////////////////////////////////////////////////////////////// +#endif //CXIMAGE_SUPPORT_DECODE +//////////////////////////////////////////////////////////////////////////////// +#endif // CXIMAGE_SUPPORT_BMP +//////////////////////////////////////////////////////////////////////////////// diff --git a/src/win32/dependencies/cximage/ximabmp.h b/src/win32/dependencies/cximage/ximabmp.h new file mode 100644 index 00000000..2bda4feb --- /dev/null +++ b/src/win32/dependencies/cximage/ximabmp.h @@ -0,0 +1,79 @@ +/* + * File: ximabmp.h + * Purpose: BMP Image Class Loader and Writer + */ +/* ========================================================== + * CxImageBMP (c) 07/Aug/2001 Davide Pizzolato - www.xdp.it + * For conditions of distribution and use, see copyright notice in ximage.h + * + * Special thanks to Troels Knakkergaard for new features, enhancements and bugfixes + * + * original CImageBMP and CImageIterator implementation are: + * Copyright: (c) 1995, Alejandro Aguilar Sierra + * + * ========================================================== + */ + +#if !defined(__ximaBMP_h) +#define __ximaBMP_h + +#include "ximage.h" + +const int RLE_COMMAND = 0; +const int RLE_ENDOFLINE = 0; +const int RLE_ENDOFBITMAP = 1; +const int RLE_DELTA = 2; + +#if !defined(BI_RLE8) + #define BI_RLE8 1L +#endif +#if !defined(BI_RLE4) + #define BI_RLE4 2L +#endif + +#if CXIMAGE_SUPPORT_BMP + +class CxImageBMP: public CxImage +{ +public: + CxImageBMP(): CxImage(CXIMAGE_FORMAT_BMP) {}; + + bool Decode(CxFile * hFile); + bool Decode(FILE *hFile) { CxIOFile file(hFile); return Decode(&file); } + +#if CXIMAGE_SUPPORT_ENCODE + bool Encode(CxFile * hFile); + bool Encode(FILE *hFile) { CxIOFile file(hFile); return Encode(&file); } +#endif // CXIMAGE_SUPPORT_ENCODE + +protected: + bool DibReadBitmapInfo(CxFile* fh, BITMAPINFOHEADER *pdib); +}; + +#define BFT_ICON 0x4349 /* 'IC' */ +#define BFT_BITMAP 0x4d42 /* 'BM' */ +#define BFT_CURSOR 0x5450 /* 'PT' */ + +#ifndef WIDTHBYTES +#define WIDTHBYTES(i) ((unsigned)((i+31)&(~31))/8) /* ULONG aligned ! */ +#endif + +#endif + +#define DibWidthBytesN(lpbi, n) (UINT)WIDTHBYTES((UINT)(lpbi)->biWidth * (UINT)(n)) +#define DibWidthBytes(lpbi) DibWidthBytesN(lpbi, (lpbi)->biBitCount) + +#define DibSizeImage(lpbi) ((lpbi)->biSizeImage == 0 \ + ? ((DWORD)(UINT)DibWidthBytes(lpbi) * (DWORD)(UINT)(lpbi)->biHeight) \ + : (lpbi)->biSizeImage) + +#define DibNumColors(lpbi) ((lpbi)->biClrUsed == 0 && (lpbi)->biBitCount <= 8 \ + ? (int)(1 << (int)(lpbi)->biBitCount) \ + : (int)(lpbi)->biClrUsed) + +#define FixBitmapInfo(lpbi) if ((lpbi)->biSizeImage == 0) \ + (lpbi)->biSizeImage = DibSizeImage(lpbi); \ + if ((lpbi)->biClrUsed == 0) \ + (lpbi)->biClrUsed = DibNumColors(lpbi); \ + +#endif diff --git a/src/win32/dependencies/cximage/ximacfg.h b/src/win32/dependencies/cximage/ximacfg.h new file mode 100644 index 00000000..a28b35ed --- /dev/null +++ b/src/win32/dependencies/cximage/ximacfg.h @@ -0,0 +1,52 @@ +#if !defined(__ximaCFG_h) +#define __ximaCFG_h + +///////////////////////////////////////////////////////////////////////////// +// CxImage supported features +#define CXIMAGE_SUPPORT_ALPHA 1 +#define CXIMAGE_SUPPORT_SELECTION 1 +#define CXIMAGE_SUPPORT_TRANSFORMATION 1 +#define CXIMAGE_SUPPORT_DSP 1 +#define CXIMAGE_SUPPORT_LAYERS 1 +#define CXIMAGE_SUPPORT_INTERPOLATION 1 + +#define CXIMAGE_SUPPORT_DECODE 1 +#define CXIMAGE_SUPPORT_ENCODE 1 // +#define CXIMAGE_SUPPORT_WINDOWS 1 +#define CXIMAGE_SUPPORT_WINCE 0 // + +///////////////////////////////////////////////////////////////////////////// +// CxImage supported formats +#define CXIMAGE_SUPPORT_BMP 1 +#define CXIMAGE_SUPPORT_GIF 1 +#define CXIMAGE_SUPPORT_JPG 0 +#define CXIMAGE_SUPPORT_PNG 0 +#define CXIMAGE_SUPPORT_MNG 0 +#define CXIMAGE_SUPPORT_ICO 1 +#define CXIMAGE_SUPPORT_TIF 0 +#define CXIMAGE_SUPPORT_TGA 1 +#define CXIMAGE_SUPPORT_PCX 1 +#define CXIMAGE_SUPPORT_WBMP 1 +#define CXIMAGE_SUPPORT_WMF 1 +#define CXIMAGE_SUPPORT_J2K 0 // Beta, use JP2 +#define CXIMAGE_SUPPORT_JBG 0 // GPL'd see ../jbig/copying.txt & ../jbig/patents.htm + +#define CXIMAGE_SUPPORT_JP2 0 +#define CXIMAGE_SUPPORT_JPC 0 +#define CXIMAGE_SUPPORT_PGX 0 +#define CXIMAGE_SUPPORT_PNM 0 +#define CXIMAGE_SUPPORT_RAS 0 + + +///////////////////////////////////////////////////////////////////////////// +#define CXIMAGE_MAX_MEMORY 256000000 + +#define CXIMAGE_ERR_NOFILE "null file handler" +#define CXIMAGE_ERR_NOIMAGE "null image!!!" + +///////////////////////////////////////////////////////////////////////////// +//color to grey mapping +//#define RGB2GRAY(r,g,b) (((b)*114 + (g)*587 + (r)*299)/1000) +#define RGB2GRAY(r,g,b) (((b)*117 + (g)*601 + (r)*306) >> 10) + +#endif diff --git a/src/win32/dependencies/cximage/ximadef.h b/src/win32/dependencies/cximage/ximadef.h new file mode 100644 index 00000000..6f3d10db --- /dev/null +++ b/src/win32/dependencies/cximage/ximadef.h @@ -0,0 +1,197 @@ +#if !defined(__ximadefs_h) +#define __ximadefs_h + +#include "ximacfg.h" + +#if defined(_AFXDLL)||defined(_USRDLL) + #define DLL_EXP __declspec(dllexport) +#elif defined(_MSC_VER)&&(_MSC_VER<1200) + #define DLL_EXP __declspec(dllimport) +#else + #define DLL_EXP +#endif + +#if CXIMAGE_SUPPORT_JP2 || CXIMAGE_SUPPORT_JPC || CXIMAGE_SUPPORT_PGX || CXIMAGE_SUPPORT_PNM || CXIMAGE_SUPPORT_RAS + #define CXIMAGE_SUPPORT_JASPER 1 +#else + #define CXIMAGE_SUPPORT_JASPER 0 +#endif + +#if CXIMAGE_SUPPORT_DSP +#undef CXIMAGE_SUPPORT_TRANSFORMATION + #define CXIMAGE_SUPPORT_TRANSFORMATION 1 +#endif + +#if CXIMAGE_SUPPORT_TRANSFORMATION || CXIMAGE_SUPPORT_TIF || CXIMAGE_SUPPORT_TGA || CXIMAGE_SUPPORT_BMP || CXIMAGE_SUPPORT_WINDOWS + #define CXIMAGE_SUPPORT_BASICTRANSFORMATIONS 1 +#endif + +#if CXIMAGE_SUPPORT_DSP || CXIMAGE_SUPPORT_TRANSFORMATION +#undef CXIMAGE_SUPPORT_INTERPOLATION + #define CXIMAGE_SUPPORT_INTERPOLATION 1 +#endif + +#if CXIMAGE_SUPPORT_WINCE + #undef CXIMAGE_SUPPORT_WMF + #define CXIMAGE_SUPPORT_WMF 0 + #undef CXIMAGE_SUPPORT_WINDOWS + #define CXIMAGE_SUPPORT_WINDOWS 0 +#endif + +#ifndef WIN32 + #undef CXIMAGE_SUPPORT_WINDOWS + #define CXIMAGE_SUPPORT_WINDOWS 0 +#endif + +#ifndef min +#define min(a,b) (((a)<(b))?(a):(b)) +#endif +#ifndef max +#define max(a,b) (((a)>(b))?(a):(b)) +#endif + +#ifndef PI + #define PI 3.141592653589793f +#endif + + +#ifdef WIN32 +#include +#include +#endif + +#include +#include + + +#ifdef __BORLANDC__ + +#ifndef _COMPLEX_DEFINED + +typedef struct tagcomplex { + double x,y; +} _complex; + +#endif + +#define _cabs(c) sqrt(c.x*c.x+c.y*c.y) + +#endif + + +#ifndef WIN32 + +#include +#include +#include + +typedef unsigned char BYTE; +typedef unsigned short WORD; +typedef unsigned long DWORD; +typedef unsigned int UINT; + +typedef DWORD COLORREF; +typedef unsigned int HANDLE; +typedef void* HRGN; + +#ifndef BOOL +#define BOOL bool +#endif + +#ifndef TRUE +#define TRUE true +#endif + +#ifndef FALSE +#define FALSE false +#endif + +#ifndef TCHAR +#define TCHAR char +#define _T +#endif + +typedef struct tagRECT +{ + long left; + long top; + long right; + long bottom; +} RECT; + +typedef struct tagPOINT +{ + long x; + long y; +} POINT; + +typedef struct tagRGBQUAD { + BYTE rgbBlue; + BYTE rgbGreen; + BYTE rgbRed; + BYTE rgbReserved; +} RGBQUAD; + +#pragma pack(1) + +typedef struct tagBITMAPINFOHEADER{ + DWORD biSize; + long biWidth; + long biHeight; + WORD biPlanes; + WORD biBitCount; + DWORD biCompression; + DWORD biSizeImage; + long biXPelsPerMeter; + long biYPelsPerMeter; + DWORD biClrUsed; + DWORD biClrImportant; +} BITMAPINFOHEADER; + +typedef struct tagBITMAPFILEHEADER { + WORD bfType; + DWORD bfSize; + WORD bfReserved1; + WORD bfReserved2; + DWORD bfOffBits; +} BITMAPFILEHEADER; + +typedef struct tagBITMAPCOREHEADER { + DWORD bcSize; + WORD bcWidth; + WORD bcHeight; + WORD bcPlanes; + WORD bcBitCount; +} BITMAPCOREHEADER; + +typedef struct tagRGBTRIPLE { + BYTE rgbtBlue; + BYTE rgbtGreen; + BYTE rgbtRed; +} RGBTRIPLE; + +#pragma pack() + +#define BI_RGB 0L +#define BI_RLE8 1L +#define BI_RLE4 2L +#define BI_BITFIELDS 3L + +#define GetRValue(rgb) ((BYTE)(rgb)) +#define GetGValue(rgb) ((BYTE)(((WORD)(rgb)) >> 8)) +#define GetBValue(rgb) ((BYTE)((rgb)>>16)) +#define RGB(r,g,b) ((COLORREF)(((BYTE)(r)|((WORD)((BYTE)(g))<<8))|(((DWORD)(BYTE)(b))<<16))) + +#ifndef _COMPLEX_DEFINED + +typedef struct tagcomplex { + double x,y; +} _complex; + +#endif + +#define _cabs(c) sqrt(c.x*c.x+c.y*c.y) + +#endif + +#endif //__ximadefs diff --git a/src/win32/dependencies/cximage/ximadsp.cpp b/src/win32/dependencies/cximage/ximadsp.cpp new file mode 100644 index 00000000..d6a2ef56 --- /dev/null +++ b/src/win32/dependencies/cximage/ximadsp.cpp @@ -0,0 +1,2370 @@ +// xImaDsp.cpp : DSP functions +/* 07/08/2001 v1.00 - Davide Pizzolato - www.xdp.it + * CxImage version 5.99c 17/Oct/2004 + */ + +#include "ximage.h" + +#include "ximaiter.h" + +#if CXIMAGE_SUPPORT_DSP + +//////////////////////////////////////////////////////////////////////////////// +/** + * Converts the image to B&W. + * The Mean() function can be used for calculating the optimal threshold. + * \param level: the lightness threshold. + * \return true if everything is ok + */ +bool CxImage::Threshold(BYTE level) +{ + if (!pDib) return false; + if (head.biBitCount == 1) return true; + + GrayScale(); + + CxImage tmp(head.biWidth,head.biHeight,1); + if (!tmp.IsValid()) return false; + + for (long y=0;ylevel) + tmp.SetPixelIndex(x,y,1); + else + tmp.SetPixelIndex(x,y,0); + } + } + tmp.SetPaletteColor(0,0,0,0); + tmp.SetPaletteColor(1,255,255,255); + Transfer(tmp); + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Extract RGB channels from the image. Each channel is an 8 bit grayscale image. + * \param r,g,b: pointers to CxImage objects, to store the splited channels + * \return true if everything is ok + */ +bool CxImage::SplitRGB(CxImage* r,CxImage* g,CxImage* b) +{ + if (!pDib) return false; + if (r==NULL && g==NULL && b==NULL) return false; + + CxImage tmpr(head.biWidth,head.biHeight,8); + CxImage tmpg(head.biWidth,head.biHeight,8); + CxImage tmpb(head.biWidth,head.biHeight,8); + + RGBQUAD color; + for(long y=0; yTransfer(tmpr); + if (g) g->Transfer(tmpg); + if (b) b->Transfer(tmpb); + + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Extract CMYK channels from the image. Each channel is an 8 bit grayscale image. + * \param c,m,y,k: pointers to CxImage objects, to store the splited channels + * \return true if everything is ok + */ +bool CxImage::SplitCMYK(CxImage* c,CxImage* m,CxImage* y,CxImage* k) +{ + if (!pDib) return false; + if (c==NULL && m==NULL && y==NULL && k==NULL) return false; + + CxImage tmpc(head.biWidth,head.biHeight,8); + CxImage tmpm(head.biWidth,head.biHeight,8); + CxImage tmpy(head.biWidth,head.biHeight,8); + CxImage tmpk(head.biWidth,head.biHeight,8); + + RGBQUAD color; + for(long yy=0; yyTransfer(tmpc); + if (m) m->Transfer(tmpm); + if (y) y->Transfer(tmpy); + if (k) k->Transfer(tmpk); + + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Extract YUV channels from the image. Each channel is an 8 bit grayscale image. + * \param y,u,v: pointers to CxImage objects, to store the splited channels + * \return true if everything is ok + */ +bool CxImage::SplitYUV(CxImage* y,CxImage* u,CxImage* v) +{ + if (!pDib) return false; + if (y==NULL && u==NULL && v==NULL) return false; + + CxImage tmpy(head.biWidth,head.biHeight,8); + CxImage tmpu(head.biWidth,head.biHeight,8); + CxImage tmpv(head.biWidth,head.biHeight,8); + + RGBQUAD color; + for(long yy=0; yyTransfer(tmpy); + if (u) u->Transfer(tmpu); + if (v) v->Transfer(tmpv); + + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Extract YIQ channels from the image. Each channel is an 8 bit grayscale image. + * \param y,i,q: pointers to CxImage objects, to store the splited channels + * \return true if everything is ok + */ +bool CxImage::SplitYIQ(CxImage* y,CxImage* i,CxImage* q) +{ + if (!pDib) return false; + if (y==NULL && i==NULL && q==NULL) return false; + + CxImage tmpy(head.biWidth,head.biHeight,8); + CxImage tmpi(head.biWidth,head.biHeight,8); + CxImage tmpq(head.biWidth,head.biHeight,8); + + RGBQUAD color; + for(long yy=0; yyTransfer(tmpy); + if (i) i->Transfer(tmpi); + if (q) q->Transfer(tmpq); + + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Extract XYZ channels from the image. Each channel is an 8 bit grayscale image. + * \param x,y,z: pointers to CxImage objects, to store the splited channels + * \return true if everything is ok + */ +bool CxImage::SplitXYZ(CxImage* x,CxImage* y,CxImage* z) +{ + if (!pDib) return false; + if (x==NULL && y==NULL && z==NULL) return false; + + CxImage tmpx(head.biWidth,head.biHeight,8); + CxImage tmpy(head.biWidth,head.biHeight,8); + CxImage tmpz(head.biWidth,head.biHeight,8); + + RGBQUAD color; + for(long yy=0; yyTransfer(tmpx); + if (y) y->Transfer(tmpy); + if (z) z->Transfer(tmpz); + + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Extract HSL channels from the image. Each channel is an 8 bit grayscale image. + * \param h,s,l: pointers to CxImage objects, to store the splited channels + * \return true if everything is ok + */ +bool CxImage::SplitHSL(CxImage* h,CxImage* s,CxImage* l) +{ + if (!pDib) return false; + if (h==NULL && s==NULL && l==NULL) return false; + + CxImage tmph(head.biWidth,head.biHeight,8); + CxImage tmps(head.biWidth,head.biHeight,8); + CxImage tmpl(head.biWidth,head.biHeight,8); + + RGBQUAD color; + for(long y=0; yTransfer(tmph); + if (s) s->Transfer(tmps); + if (l) l->Transfer(tmpl); + + return true; +} +//////////////////////////////////////////////////////////////////////////////// +#define HSLMAX 255 /* H,L, and S vary over 0-HSLMAX */ +#define RGBMAX 255 /* R,G, and B vary over 0-RGBMAX */ + /* HSLMAX BEST IF DIVISIBLE BY 6 */ + /* RGBMAX, HSLMAX must each fit in a BYTE. */ +/* Hue is undefined if Saturation is 0 (grey-scale) */ +/* This value determines where the Hue scrollbar is */ +/* initially set for achromatic colors */ +#define HSLUNDEFINED (HSLMAX*2/3) +//////////////////////////////////////////////////////////////////////////////// +RGBQUAD CxImage::RGBtoHSL(RGBQUAD lRGBColor) +{ + BYTE R,G,B; /* input RGB values */ + BYTE H,L,S; /* output HSL values */ + BYTE cMax,cMin; /* max and min RGB values */ + WORD Rdelta,Gdelta,Bdelta; /* intermediate value: % of spread from max*/ + + R = lRGBColor.rgbRed; /* get R, G, and B out of DWORD */ + G = lRGBColor.rgbGreen; + B = lRGBColor.rgbBlue; + + cMax = max( max(R,G), B); /* calculate lightness */ + cMin = min( min(R,G), B); + L = (BYTE)((((cMax+cMin)*HSLMAX)+RGBMAX)/(2*RGBMAX)); + + if (cMax==cMin){ /* r=g=b --> achromatic case */ + S = 0; /* saturation */ + H = HSLUNDEFINED; /* hue */ + } else { /* chromatic case */ + if (L <= (HSLMAX/2)) /* saturation */ + S = (BYTE)((((cMax-cMin)*HSLMAX)+((cMax+cMin)/2))/(cMax+cMin)); + else + S = (BYTE)((((cMax-cMin)*HSLMAX)+((2*RGBMAX-cMax-cMin)/2))/(2*RGBMAX-cMax-cMin)); + /* hue */ + Rdelta = (WORD)((((cMax-R)*(HSLMAX/6)) + ((cMax-cMin)/2) ) / (cMax-cMin)); + Gdelta = (WORD)((((cMax-G)*(HSLMAX/6)) + ((cMax-cMin)/2) ) / (cMax-cMin)); + Bdelta = (WORD)((((cMax-B)*(HSLMAX/6)) + ((cMax-cMin)/2) ) / (cMax-cMin)); + + if (R == cMax) + H = (BYTE)(Bdelta - Gdelta); + else if (G == cMax) + H = (BYTE)((HSLMAX/3) + Rdelta - Bdelta); + else /* B == cMax */ + H = (BYTE)(((2*HSLMAX)/3) + Gdelta - Rdelta); + +// if (H < 0) H += HSLMAX; //always false + if (H > HSLMAX) H -= HSLMAX; + } + RGBQUAD hsl={L,S,H,0}; + return hsl; +} +//////////////////////////////////////////////////////////////////////////////// +float CxImage::HueToRGB(float n1,float n2, float hue) +{ + // fixed implementation for HSL2RGB routine + float rValue; + + if (hue > 360) + hue = hue - 360; + else if (hue < 0) + hue = hue + 360; + + if (hue < 60) + rValue = n1 + (n2-n1)*hue/60.0f; + else if (hue < 180) + rValue = n2; + else if (hue < 240) + rValue = n1+(n2-n1)*(240-hue)/60; + else + rValue = n1; + + return rValue; +} +//////////////////////////////////////////////////////////////////////////////// +RGBQUAD CxImage::HSLtoRGB(COLORREF cHSLColor) +{ + return HSLtoRGB(RGBtoRGBQUAD(cHSLColor)); +} +//////////////////////////////////////////////////////////////////////////////// +RGBQUAD CxImage::HSLtoRGB(RGBQUAD lHSLColor) +{ + // fixed implementation for HSL2RGB routine + float h,s,l; + float m1,m2; + BYTE r,g,b; + + h = (float)lHSLColor.rgbRed * 360.0f/255.0f; + s = (float)lHSLColor.rgbGreen/255.0f; + l = (float)lHSLColor.rgbBlue/255.0f; + + if (l <= 0.5) m2 = l * (1+s); + else m2 = l + s - l*s; + + m1 = 2 * l - m2; + + if (s == 0) { + r=g=b=(BYTE)(l*255.0f); + } else { + r = (BYTE)(HueToRGB(m1,m2,h+120) * 255.0f); + g = (BYTE)(HueToRGB(m1,m2,h) * 255.0f); + b = (BYTE)(HueToRGB(m1,m2,h-120) * 255.0f); + } + + RGBQUAD rgb = {b,g,r,0}; + return rgb; +} +//////////////////////////////////////////////////////////////////////////////// +RGBQUAD CxImage::YUVtoRGB(RGBQUAD lYUVColor) +{ + int U,V,R,G,B; + float Y = lYUVColor.rgbRed; + U = lYUVColor.rgbGreen - 128; + V = lYUVColor.rgbBlue - 128; + +// R = (int)(1.164 * Y + 2.018 * U); +// G = (int)(1.164 * Y - 0.813 * V - 0.391 * U); +// B = (int)(1.164 * Y + 1.596 * V); + R = (int)( Y + 1.403f * V); + G = (int)( Y - 0.344f * U - 0.714f * V); + B = (int)( Y + 1.770f * U); + + R= min(255,max(0,R)); + G= min(255,max(0,G)); + B= min(255,max(0,B)); + RGBQUAD rgb={(BYTE)B,(BYTE)G,(BYTE)R,0}; + return rgb; +} +//////////////////////////////////////////////////////////////////////////////// +RGBQUAD CxImage::RGBtoYUV(RGBQUAD lRGBColor) +{ + int Y,U,V,R,G,B; + R = lRGBColor.rgbRed; + G = lRGBColor.rgbGreen; + B = lRGBColor.rgbBlue; + +// Y = (int)( 0.257 * R + 0.504 * G + 0.098 * B); +// U = (int)( 0.439 * R - 0.368 * G - 0.071 * B + 128); +// V = (int)(-0.148 * R - 0.291 * G + 0.439 * B + 128); + Y = (int)(0.299f * R + 0.587f * G + 0.114f * B); + U = (int)((B-Y) * 0.565f + 128); + V = (int)((R-Y) * 0.713f + 128); + + Y= min(255,max(0,Y)); + U= min(255,max(0,U)); + V= min(255,max(0,V)); + RGBQUAD yuv={(BYTE)V,(BYTE)U,(BYTE)Y,0}; + return yuv; +} +//////////////////////////////////////////////////////////////////////////////// +RGBQUAD CxImage::YIQtoRGB(RGBQUAD lYIQColor) +{ + int I,Q,R,G,B; + float Y = lYIQColor.rgbRed; + I = lYIQColor.rgbGreen - 128; + Q = lYIQColor.rgbBlue - 128; + + R = (int)( Y + 0.956f * I + 0.621f * Q); + G = (int)( Y - 0.273f * I - 0.647f * Q); + B = (int)( Y - 1.104f * I + 1.701f * Q); + + R= min(255,max(0,R)); + G= min(255,max(0,G)); + B= min(255,max(0,B)); + RGBQUAD rgb={(BYTE)B,(BYTE)G,(BYTE)R,0}; + return rgb; +} +//////////////////////////////////////////////////////////////////////////////// +RGBQUAD CxImage::RGBtoYIQ(RGBQUAD lRGBColor) +{ + int Y,I,Q,R,G,B; + R = lRGBColor.rgbRed; + G = lRGBColor.rgbGreen; + B = lRGBColor.rgbBlue; + + Y = (int)( 0.2992f * R + 0.5868f * G + 0.1140f * B); + I = (int)( 0.5960f * R - 0.2742f * G - 0.3219f * B + 128); + Q = (int)( 0.2109f * R - 0.5229f * G + 0.3120f * B + 128); + + Y= min(255,max(0,Y)); + I= min(255,max(0,I)); + Q= min(255,max(0,Q)); + RGBQUAD yiq={(BYTE)Q,(BYTE)I,(BYTE)Y,0}; + return yiq; +} +//////////////////////////////////////////////////////////////////////////////// +RGBQUAD CxImage::XYZtoRGB(RGBQUAD lXYZColor) +{ + int X,Y,Z,R,G,B; + X = lXYZColor.rgbRed; + Y = lXYZColor.rgbGreen; + Z = lXYZColor.rgbBlue; + double k=1.088751; + + R = (int)( 3.240479f * X - 1.537150f * Y - 0.498535f * Z * k); + G = (int)( -0.969256f * X + 1.875992f * Y + 0.041556f * Z * k); + B = (int)( 0.055648f * X - 0.204043f * Y + 1.057311f * Z * k); + + R= min(255,max(0,R)); + G= min(255,max(0,G)); + B= min(255,max(0,B)); + RGBQUAD rgb={(BYTE)B,(BYTE)G,(BYTE)R,0}; + return rgb; +} +//////////////////////////////////////////////////////////////////////////////// +RGBQUAD CxImage::RGBtoXYZ(RGBQUAD lRGBColor) +{ + int X,Y,Z,R,G,B; + R = lRGBColor.rgbRed; + G = lRGBColor.rgbGreen; + B = lRGBColor.rgbBlue; + + X = (int)( 0.412453f * R + 0.357580f * G + 0.180423f * B); + Y = (int)( 0.212671f * R + 0.715160f * G + 0.072169f * B); + Z = (int)((0.019334f * R + 0.119193f * G + 0.950227f * B)*0.918483657f); + + //X= min(255,max(0,X)); + //Y= min(255,max(0,Y)); + //Z= min(255,max(0,Z)); + RGBQUAD xyz={(BYTE)Z,(BYTE)Y,(BYTE)X,0}; + return xyz; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Generates a "rainbow" palette with saturated colors + * \param correction: 1 generates a single hue spectrum. 0.75 is nice for scientific applications. + */ +void CxImage::HuePalette(float correction) +{ + if (head.biClrUsed==0) return; + + for(DWORD j=0; j 1.0f) blend = 1.0f; + int a0 = (int)(256*blend); + int a1 = 256 - a0; + + bool bFullBlend = false; + if (blend > 0.999f) bFullBlend = true; + + RGBQUAD color,hsl; + if (head.biClrUsed==0){ + + long xmin,xmax,ymin,ymax; + if (pSelection){ + xmin = info.rSelectionBox.left; xmax = info.rSelectionBox.right; + ymin = info.rSelectionBox.bottom; ymax = info.rSelectionBox.top; + } else { + xmin = ymin = 0; + xmax = head.biWidth; ymax=head.biHeight; + } + + for(long y=ymin; y>8); + color.rgbBlue = (BYTE)((hsl.rgbBlue * a0 + color.rgbBlue * a1)>>8); + color.rgbGreen = (BYTE)((hsl.rgbGreen * a0 + color.rgbGreen * a1)>>8); + SetPixelColor(x,y,color); + } + } + } + } + } else { + for(DWORD j=0; j + for (int i=0;i<256;i++) { + cTable[i] = (BYTE)max(0,min(255,(int)((i-128)*c + brightness))); + } + + return Lut(cTable); +} +//////////////////////////////////////////////////////////////////////////////// +/** + * \return mean lightness of the image. Useful with Threshold() and Light() + */ +float CxImage::Mean() +{ + if (!pDib) return 0; + + CxImage tmp(*this,true); + if (!tmp.IsValid()) return false; + + tmp.GrayScale(); + float sum=0; + + long xmin,xmax,ymin,ymax; + if (pSelection){ + xmin = info.rSelectionBox.left; xmax = info.rSelectionBox.right; + ymin = info.rSelectionBox.bottom; ymax = info.rSelectionBox.top; + } else { + xmin = ymin = 0; + xmax = head.biWidth; ymax=head.biHeight; + } + if (xmin==xmax || ymin==ymax) return (float)0.0; + + BYTE *iSrc=tmp.info.pImage; + iSrc += tmp.info.dwEffWidth*ymin; // necessary for selections + + for(long y=ymin; y + for(long x=xmin; x 0 && (y+kmax-1) < head.biHeight && x-k2 > 0 && (x+kmax-1) < head.biWidth) + { + b=0; + iCount = 0; + iY2 = ((y-k2)*info.dwEffWidth); + for(long j=-k2;j r) r=c.rgbRed; + if (c.rgbGreen > g) g=c.rgbGreen; + if (c.rgbBlue > b) b=c.rgbBlue; + } + } + c.rgbRed = r; + c.rgbGreen = g; + c.rgbBlue = b; + tmp.SetPixelColor(x,y,c); + } + } + } + Transfer(tmp); + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Enhance the variations between adjacent pixels. + * Similar results can be achieved using Filter(), + * but the algorithms are different both in Edge() and in Contour(). + * \param Ksize: size of the kernel. + * \return true if everything is ok + */ +bool CxImage::Edge(long Ksize) +{ + if (!pDib) return false; + + long k2 = Ksize/2; + long kmax= Ksize-k2; + BYTE r,g,b,rr,gg,bb; + RGBQUAD c; + + CxImage tmp(*this,pSelection!=0,true,true); + if (!tmp.IsValid()) return false; + + long xmin,xmax,ymin,ymax; + if (pSelection){ + xmin = info.rSelectionBox.left; xmax = info.rSelectionBox.right; + ymin = info.rSelectionBox.bottom; ymax = info.rSelectionBox.top; + } else { + xmin = ymin = 0; + xmax = head.biWidth; ymax=head.biHeight; + } + + for(long y=ymin; y r) r=c.rgbRed; + if (c.rgbGreen > g) g=c.rgbGreen; + if (c.rgbBlue > b) b=c.rgbBlue; + + if (c.rgbRed < rr) rr=c.rgbRed; + if (c.rgbGreen < gg) gg=c.rgbGreen; + if (c.rgbBlue < bb) bb=c.rgbBlue; + } + } + c.rgbRed = 255-abs(r-rr); + c.rgbGreen = 255-abs(g-gg); + c.rgbBlue = 255-abs(b-bb); + tmp.SetPixelColor(x,y,c); + } + } + } + Transfer(tmp); + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Blends two images + * \param imgsrc2: image to be mixed with this + * \param op: blending method; see ImageOpType + * \param lXOffset, lYOffset: image displacement + * \param bMixAlpha: if true and imgsrc2 has a valid alpha layer, it will be mixed in the destination image. + * \return true if everything is ok + * + * thanks to Mwolski + */ +// +void CxImage::Mix(CxImage & imgsrc2, ImageOpType op, long lXOffset, long lYOffset, bool bMixAlpha) +{ + long lWide = min(GetWidth(),imgsrc2.GetWidth()-lXOffset); + long lHeight = min(GetHeight(),imgsrc2.GetHeight()-lYOffset); + + bool bEditAlpha = imgsrc2.AlphaIsValid() & bMixAlpha; + + if (bEditAlpha && AlphaIsValid()==false){ + AlphaCreate(); + } + + RGBQUAD rgbBackgrnd = GetTransColor(); + RGBQUAD rgb1, rgb2, rgbDest; + + for(long lY=0;lY>8); + } else { + a=255; + } + + if (a==0){ //transparent + rgbDest = rgb1; + } else if (a==255){ //opaque + rgbDest = rgb2; + } else { //blend + a1 = (BYTE)~a; + rgbDest.rgbBlue = (BYTE)((rgb1.rgbBlue*a1+rgb2.rgbBlue*a)>>8); + rgbDest.rgbGreen = (BYTE)((rgb1.rgbGreen*a1+rgb2.rgbGreen*a)>>8); + rgbDest.rgbRed = (BYTE)((rgb1.rgbRed*a1+rgb2.rgbRed*a)>>8); + } + + if (bEditAlpha) rgbDest.rgbReserved = (BYTE)(((1+rgb1.rgbReserved)*a)>>8); + } + break; + case OpSrcBlend: + if(memcmp(&rgb1,&rgbBackgrnd,sizeof(RGBQUAD))==0) + rgbDest = rgb2; + else + { + long lBDiff = abs(rgb1.rgbBlue - rgbBackgrnd.rgbBlue); + long lGDiff = abs(rgb1.rgbGreen - rgbBackgrnd.rgbGreen); + long lRDiff = abs(rgb1.rgbRed - rgbBackgrnd.rgbRed); + + double lAverage = (lBDiff+lGDiff+lRDiff)/3; + double lThresh = 16; + double dLarge = lAverage/lThresh; + double dSmall = (lThresh-lAverage)/lThresh; + double dSmallAmt = dSmall*((double)rgb2.rgbBlue); + + if( lAverage < lThresh+1){ + rgbDest.rgbBlue = (BYTE)max(0,min(255,(int)(dLarge*((double)rgb1.rgbBlue) + + dSmallAmt))); + rgbDest.rgbGreen = (BYTE)max(0,min(255,(int)(dLarge*((double)rgb1.rgbGreen) + + dSmallAmt))); + rgbDest.rgbRed = (BYTE)max(0,min(255,(int)(dLarge*((double)rgb1.rgbRed) + + dSmallAmt))); + } + else + rgbDest = rgb1; + } + break; + default: + return; + } + SetPixelColor(lX,lY,rgbDest,bEditAlpha); + } + } + } +} +//////////////////////////////////////////////////////////////////////////////// +// thanks to Kenneth Ballard +void CxImage::MixFrom(CxImage & imagesrc2, long lXOffset, long lYOffset) +{ + RGBQUAD rgbBackgrnd = imagesrc2.GetTransColor(); + RGBQUAD rgb1; + + long width = imagesrc2.GetWidth(); + long height = imagesrc2.GetHeight(); + + int x, y; + + for(x = 0; x < width; x++) + { + for(y = 0; y < height; y++) + { + rgb1 = imagesrc2.GetPixelColor(x, y); + if(memcmp(&rgb1, &rgbBackgrnd, sizeof(RGBQUAD)) != 0) + SetPixelColor(x + lXOffset, y + lYOffset, rgb1); + } + } +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Adjusts separately the red, green, and blue values in the image. + * \param r, g, b: can be from -255 to +255. + * \return true if everything is ok + */ +bool CxImage::ShiftRGB(long r, long g, long b) +{ + if (!pDib) return false; + RGBQUAD color; + if (head.biClrUsed==0){ + + long xmin,xmax,ymin,ymax; + if (pSelection){ + xmin = info.rSelectionBox.left; xmax = info.rSelectionBox.right; + ymin = info.rSelectionBox.bottom; ymax = info.rSelectionBox.top; + } else { + xmin = ymin = 0; + xmax = head.biWidth; ymax=head.biHeight; + } + + for(long y=ymin; y + for (int i=0;i<256;i++) { + cTable[i] = (BYTE)max(0,min(255,(int)( pow((double)i, dinvgamma) / dMax))); + } + + return Lut(cTable); +} +//////////////////////////////////////////////////////////////////////////////// +#if CXIMAGE_SUPPORT_WINCE == 0 +/** + * Adjusts the intensity of each pixel to the median intensity of its surrounding pixels. + * \param Ksize: size of the kernel. + * \return true if everything is ok + */ +bool CxImage::Median(long Ksize) +{ + if (!pDib) return false; + + long k2 = Ksize/2; + long kmax= Ksize-k2; + long i,j,k; + + RGBQUAD* kernel = (RGBQUAD*)malloc(Ksize*Ksize*sizeof(RGBQUAD)); + + CxImage tmp(*this,pSelection!=0,true,true); + if (!tmp.IsValid()) return false; + + long xmin,xmax,ymin,ymax; + if (pSelection){ + xmin = info.rSelectionBox.left; xmax = info.rSelectionBox.right; + ymin = info.rSelectionBox.bottom; ymax = info.rSelectionBox.top; + } else { + xmin = ymin = 0; + xmax = head.biWidth; ymax=head.biHeight; + } + + for(long y=ymin; y + for(long x=xmin; xGetWidth(); + h=srcReal->GetHeight(); + } else { + w=srcImag->GetWidth(); + h=srcImag->GetHeight(); + } + + bool bXpow2 = IsPowerof2(w); + bool bYpow2 = IsPowerof2(h); + //if bForceFFT, width AND height must be powers of 2 + if (bForceFFT && !(bXpow2 && bYpow2)) { + long i; + + i=0; + while((1< copy the image + if (srcReal && dstReal) tmpReal->Copy(*srcReal,true,false,false); + if (srcImag && dstImag) tmpImag->Copy(*srcImag,true,false,false); + + // dst&&src are empty -> create new one, else turn to GrayScale + if (srcReal==0 && dstReal==0){ + tmpReal = new CxImage(w,h,8); + tmpReal->Clear(0); + tmpReal->SetGrayPalette(); + } else { + if (!tmpReal->IsGrayScale()) tmpReal->GrayScale(); + } + if (srcImag==0 && dstImag==0){ + tmpImag = new CxImage(w,h,8); + tmpImag->Clear(0); + tmpImag->SetGrayPalette(); + } else { + if (!tmpImag->IsGrayScale()) tmpImag->GrayScale(); + } + + if (!(tmpReal->IsValid() && tmpImag->IsValid())){ + if (srcReal==0 && dstReal==0) delete tmpReal; + if (srcImag==0 && dstImag==0) delete tmpImag; + return false; + } + + //resample for FFT, if necessary + tmpReal->Resample(w,h,0); + tmpImag->Resample(w,h,0); + + //ok, here we have 2 (w x h), grayscale images ready for a FFT + + double* real; + double* imag; + long j,k,m; + + _complex **grid; + //double mean = tmpReal->Mean(); + /* Allocate memory for the grid */ + grid = (_complex **)malloc(w * sizeof(_complex)); + for (k=0;kGetPixelIndex(k,j)-128; + grid[k][j].y = tmpImag->GetPixelIndex(k,j)-128; + } + } + + //DFT buffers + double *real2,*imag2; + real2 = (double*)malloc(max(w,h) * sizeof(double)); + imag2 = (double*)malloc(max(w,h) * sizeof(double)); + + /* Transform the rows */ + real = (double *)malloc(w * sizeof(double)); + imag = (double *)malloc(w * sizeof(double)); + + m=0; + while((1<SetPixelIndex(k,j,(BYTE)max(0,min(255,(nn*(3+log(_cabs(grid[k][j]))))))); + if (grid[k][j].x==0){ + tmpImag->SetPixelIndex(k,j,(BYTE)max(0,min(255,(128+(atan(grid[k][j].y/0.0000000001)*nn))))); + } else { + tmpImag->SetPixelIndex(k,j,(BYTE)max(0,min(255,(128+(atan(grid[k][j].y/grid[k][j].x)*nn))))); + } + } else { + tmpReal->SetPixelIndex(k,j,(BYTE)max(0,min(255,(128 + grid[k][j].x*nn)))); + tmpImag->SetPixelIndex(k,j,(BYTE)max(0,min(255,(128 + grid[k][j].y*nn)))); + } + } + } + + for (k=0;k> 1; + j = 0; + for (i=0;i>= 1; + } + j += k; + } + + /* Compute the FFT */ + c1 = -1.0; + c2 = 0.0; + l2 = 1; + for (l=0;lGetWidth(); + long h = r->GetHeight(); + + Create(w,h,24); + + g->Resample(w,h); + b->Resample(w,h); + + if (a) { + a->Resample(w,h); +#if CXIMAGE_SUPPORT_ALPHA + AlphaCreate(); +#endif //CXIMAGE_SUPPORT_ALPHA + } + + RGBQUAD c; + for (long y=0;y + for (long x=0;xGetPixelIndex(x,y); + c.rgbGreen=g->GetPixelIndex(x,y); + c.rgbBlue=b->GetPixelIndex(x,y); + switch (colorspace){ + case 1: + SetPixelColor(x,y,HSLtoRGB(c)); + break; + case 2: + SetPixelColor(x,y,YUVtoRGB(c)); + break; + case 3: + SetPixelColor(x,y,YIQtoRGB(c)); + break; + case 4: + SetPixelColor(x,y,XYZtoRGB(c)); + break; + default: + SetPixelColor(x,y,c); + } +#if CXIMAGE_SUPPORT_ALPHA + if (a) AlphaSet(x,y,a->GetPixelIndex(x,y)); +#endif //CXIMAGE_SUPPORT_ALPHA + } + } + + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Smart blurring to remove small defects, dithering or artifacts. + * \param radius: normally between 0.01 and 0.5 + * \param niterations: should be trimmed with radius, to avoid blurring should be (radius*niterations)<1 + * \param colorspace: 0 = RGB, 1 = HSL, 2 = YUV, 3 = YIQ, 4 = XYZ + * \return true if everything is ok + */ +bool CxImage::Repair(float radius, long niterations, long colorspace) +{ + if (!IsValid()) return false; + + long w = GetWidth(); + long h = GetHeight(); + + CxImage r,g,b; + + r.Create(w,h,8); + g.Create(w,h,8); + b.Create(w,h,8); + + switch (colorspace){ + case 1: + SplitHSL(&r,&g,&b); + break; + case 2: + SplitYUV(&r,&g,&b); + break; + case 3: + SplitYIQ(&r,&g,&b); + break; + case 4: + SplitXYZ(&r,&g,&b); + break; + default: + SplitRGB(&r,&g,&b); + } + + for (int i=0; iGetWidth()-1; + long h = ch->GetHeight()-1; + + double correction,ix,iy,ixx,ixy,iyy,den,num; + int x,y,xy0,xp1,xm1,yp1,ym1; + for(x=1; xGetPixelIndex(x,y); + xm1 = ch->GetPixelIndex(x-1,y); + xp1 = ch->GetPixelIndex(x+1,y); + ym1 = ch->GetPixelIndex(x,y-1); + yp1 = ch->GetPixelIndex(x,y+1); + + ix= (xp1-xm1)/2.0; + iy= (yp1-ym1)/2.0; + ixx= xp1 - 2.0 * xy0 + xm1; + iyy= yp1 - 2.0 * xy0 + ym1; + ixy=(ch->GetPixelIndex(x+1,y+1)+ch->GetPixelIndex(x-1,y-1)- + ch->GetPixelIndex(x-1,y+1)-ch->GetPixelIndex(x+1,y-1))/4.0; + + num= (1.0+iy*iy)*ixx - ix*iy*ixy + (1.0+ix*ix)*iyy; + den= 1.0+ix*ix+iy*iy; + correction = num/den; + + tmp.SetPixelIndex(x,y,(BYTE)min(255,max(0,(xy0 + radius * correction)))); + } + } + + for (x=0;x<=w;x++){ + tmp.SetPixelIndex(x,0,ch->GetPixelIndex(x,0)); + tmp.SetPixelIndex(x,h,ch->GetPixelIndex(x,h)); + } + for (y=0;y<=h;y++){ + tmp.SetPixelIndex(0,y,ch->GetPixelIndex(0,y)); + tmp.SetPixelIndex(w,y,ch->GetPixelIndex(w,y)); + } + ch->Transfer(tmp); + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Enhance the variations between adjacent pixels. + * Similar results can be achieved using Filter(), + * but the algorithms are different both in Edge() and in Contour(). + * \return true if everything is ok + */ +bool CxImage::Contour() +{ + if (!pDib) return false; + + long Ksize = 3; + long k2 = Ksize/2; + long kmax= Ksize-k2; + long i,j,k; + BYTE maxr,maxg,maxb; + RGBQUAD pix1,pix2; + + CxImage tmp(*this,pSelection!=0,true,true); + if (!tmp.IsValid()) return false; + + long xmin,xmax,ymin,ymax; + if (pSelection){ + xmin = info.rSelectionBox.left; xmax = info.rSelectionBox.right; + ymin = info.rSelectionBox.bottom; ymax = info.rSelectionBox.top; + } else { + xmin = ymin = 0; + xmax = head.biWidth; ymax=head.biHeight; + } + + for(long y=ymin; ymaxb) maxb = pix2.rgbBlue; + if ((pix2.rgbGreen-pix1.rgbGreen)>maxg) maxg = pix2.rgbGreen; + if ((pix2.rgbRed-pix1.rgbRed)>maxr) maxr = pix2.rgbRed; + } + } + pix1.rgbBlue=(BYTE)(255-maxb); + pix1.rgbGreen=(BYTE)(255-maxg); + pix1.rgbRed=(BYTE)(255-maxr); + tmp.SetPixelColor(x,y,pix1); + } + } + } + Transfer(tmp); + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Adds a random offset to each pixel in the image + * \param radius: maximum pixel displacement + * \return true if everything is ok + */ +bool CxImage::Jitter(long radius) +{ + if (!pDib) return false; + + long nx,ny; + + CxImage tmp(*this,pSelection!=0,true,true); + if (!tmp.IsValid()) return false; + + long xmin,xmax,ymin,ymax; + if (pSelection){ + xmin = info.rSelectionBox.left; xmax = info.rSelectionBox.right; + ymin = info.rSelectionBox.bottom; ymax = info.rSelectionBox.top; + } else { + xmin = ymin = 0; + xmax = head.biWidth; ymax=head.biHeight; + } + + for(long y=ymin; y y) + { + for (row = 0; row < y ; row++) + { + scale=0; + /* find the scale factor */ + for (j = 0; j < y ; j++) + { + /* if the index is in bounds, add it to the scale counter */ + if ((j + cmatrix_length/2 - row >= 0) && + (j + cmatrix_length/2 - row < cmatrix_length)) + scale += cmatrix[j + cmatrix_length/2 - row]; + } + for (i = 0; i= row - cmatrix_length/2) && + (j <= row + cmatrix_length/2)) + sum += cur_col[j*bytes + i] * cmatrix[j]; + } + dest_col[row*bytes + i] = (BYTE)(0.5f + sum / scale); + } + } + } + else + { + /* for the edge condition, we only use available info and scale to one */ + for (row = 0; row < cmatrix_middle; row++) + { + /* find scale factor */ + scale=0; + for (j = cmatrix_middle - row; j0; j--) + { + sum += *(ctable_p + *cur_col_p1); + cur_col_p1 += bytes; + ctable_p += 256; + } + cur_col_p++; + *(dest_col_p++) = (BYTE)(0.5f + sum); + } + } + + /* for the edge condition , we only use available info, and scale to one */ + for (; row < y; row++) + { + /* find scale factor */ + scale=0; + for (j = 0; j< y-row + cmatrix_middle; j++) + scale += cmatrix[j]; + for (i = 0; i 255) dest_row[u*3+v] = 255; + else dest_row[u*3+v] = value; + } + } + } + + Transfer(tmp); + + return TRUE; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Apply a look up table to the image. + * \param pLut: BYTE[256] look up table + * \return true if everything is ok + */ +bool CxImage::Lut(BYTE* pLut) +{ + if (!pDib || !pLut) return false; + RGBQUAD color; + + double dbScaler; + if (head.biClrUsed==0){ + + long xmin,xmax,ymin,ymax; + if (pSelection){ + xmin = info.rSelectionBox.left; xmax = info.rSelectionBox.right; + ymin = info.rSelectionBox.bottom; ymax = info.rSelectionBox.top; + } else { + // faster loop for full image + BYTE *iSrc=info.pImage; + for(unsigned long i=0; i < head.biSizeImage ; i++){ + *iSrc++ = pLut[*iSrc]; + } + return true; + } + + dbScaler = 100.0/ymax; + + for(long y=ymin; y + for(long x=xmin; x + if (!pDib) return false; + FloodFill2(x,y,GetPixelColor(x,y),FillColor); + return true; +} +//////////////////////////////////////////////////////////////////////////////// +void CxImage::FloodFill2(int x, int y, RGBQUAD old_color, RGBQUAD new_color) +{ + // Fill in the actual pixels. + // Function steps recursively until it finds borders (color that is not old_color) + if (!IsInside(x,y)) return; + + RGBQUAD r = GetPixelColor(x,y); + COLORREF cr = RGB(r.rgbRed,r.rgbGreen,r.rgbBlue); + + if(cr == RGB(old_color.rgbRed,old_color.rgbGreen,old_color.rgbBlue) + && cr != RGB(new_color.rgbRed,new_color.rgbGreen,new_color.rgbBlue) ) { + + // the above if statement, after && is there to prevent + // stack overflows. The program will continue to find + // colors if you flood-fill an entire region (entire picture) + + SetPixelColor(x,y,new_color); + + FloodFill2((x+1),y,old_color,new_color); + FloodFill2((x-1),y,old_color,new_color); + FloodFill2(x,(y+1),old_color,new_color); + FloodFill2(x,(y-1),old_color,new_color); + } +}*/ +/////////////////////////////////////////////////////////////////////////////// +#endif //CXIMAGE_SUPPORT_DSP diff --git a/src/win32/dependencies/cximage/ximaenc.cpp b/src/win32/dependencies/cximage/ximaenc.cpp new file mode 100644 index 00000000..da95578b --- /dev/null +++ b/src/win32/dependencies/cximage/ximaenc.cpp @@ -0,0 +1,920 @@ +// xImaCodec.cpp : Encode Decode functions +/* 07/08/2001 v1.00 - Davide Pizzolato - www.xdp.it + * CxImage version 5.99c 17/Oct/2004 + */ + +#include "ximage.h" + +#if CXIMAGE_SUPPORT_JPG +#include "ximajpg.h" +#endif + +#if CXIMAGE_SUPPORT_GIF +#include "ximagif.h" +#endif + +#if CXIMAGE_SUPPORT_PNG +#include "ximapng.h" +#endif + +#if CXIMAGE_SUPPORT_MNG +#include "ximamng.h" +#endif + +#if CXIMAGE_SUPPORT_BMP +#include "ximabmp.h" +#endif + +#if CXIMAGE_SUPPORT_ICO +#include "ximaico.h" +#endif + +#if CXIMAGE_SUPPORT_TIF +#include "ximatif.h" +#endif + +#if CXIMAGE_SUPPORT_TGA +#include "ximatga.h" +#endif + +#if CXIMAGE_SUPPORT_PCX +#include "ximapcx.h" +#endif + +#if CXIMAGE_SUPPORT_WBMP +#include "ximawbmp.h" +#endif + +#if CXIMAGE_SUPPORT_WMF +#include "ximawmf.h" // - WMF/EMF support +#endif + +#if CXIMAGE_SUPPORT_J2K +#include "ximaj2k.h" +#endif + +#if CXIMAGE_SUPPORT_JBG +#include "ximajbg.h" +#endif + +#if CXIMAGE_SUPPORT_JASPER +#include "ximajas.h" +#endif + +//////////////////////////////////////////////////////////////////////////////// +#if CXIMAGE_SUPPORT_ENCODE +//////////////////////////////////////////////////////////////////////////////// +bool CxImage::EncodeSafeCheck(CxFile *hFile) +{ + if (hFile==NULL) { + strcpy(info.szLastError,CXIMAGE_ERR_NOFILE); + return true; + } + + if (pDib==NULL){ + strcpy(info.szLastError,CXIMAGE_ERR_NOIMAGE); + return true; + } + return false; +} +//////////////////////////////////////////////////////////////////////////////// +//#ifdef WIN32 +//bool CxImage::Save(LPCWSTR filename, DWORD imagetype) +//{ +// FILE* hFile; //file handle to write the image +// if ((hFile=_wfopen(filename,L"wb"))==NULL) return false; +// bool bOK = Encode(hFile,imagetype); +// fclose(hFile); +// return bOK; +//} +//#endif //WIN32 +//////////////////////////////////////////////////////////////////////////////// +// For UNICODE support: char -> TCHAR +/** + * Saves to disk the image in a specific format. + * \param filename: file name + * \param imagetype: file format, see ENUM_CXIMAGE_FORMATS + * \return true if everything is ok + */ +bool CxImage::Save(const TCHAR * filename, DWORD imagetype) +{ + FILE* hFile; //file handle to write the image + +#ifdef WIN32 + if ((hFile=_tfopen(filename,_T("wb")))==NULL) return false; // For UNICODE support +#else + if ((hFile=fopen(filename,"wb"))==NULL) return false; +#endif + + bool bOK = Encode(hFile,imagetype); + fclose(hFile); + return bOK; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Saves to disk the image in a specific format. + * \param hFile: file handle, open and enabled for writing. + * \param imagetype: file format, see ENUM_CXIMAGE_FORMATS + * \return true if everything is ok + */ +bool CxImage::Encode(FILE *hFile, DWORD imagetype) +{ + CxIOFile file(hFile); + return Encode(&file,imagetype); +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Saves to memory buffer the image in a specific format. + * \param buffer: output memory buffer pointer. Must be NULL, + * the function allocates and fill the memory, + * the application must free the buffer, see also FreeMemory(). + * \param size: output memory buffer size. + * \param imagetype: file format, see ENUM_CXIMAGE_FORMATS + * \return true if everything is ok + */ +bool CxImage::Encode(BYTE * &buffer, long &size, DWORD imagetype) +{ + if (buffer!=NULL){ + strcpy(info.szLastError,"the buffer must be empty"); + return false; + } + CxMemFile file; + file.Open(); + if(Encode(&file,imagetype)){ + buffer=file.GetBuffer(); + size=file.Size(); + return true; + } + return false; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Saves to disk the image in a specific format. + * \param hFile: file handle (implemented using CxMemFile or CxIOFile), + * open and enabled for writing. + * \param imagetype: file format, see ENUM_CXIMAGE_FORMATS + * \return true if everything is ok + * \sa ENUM_CXIMAGE_FORMATS + */ +bool CxImage::Encode(CxFile *hFile, DWORD imagetype) +{ + +#if CXIMAGE_SUPPORT_BMP + + if (imagetype==CXIMAGE_FORMAT_BMP){ + CxImageBMP newima; + newima.Ghost(this); + if (newima.Encode(hFile)){ + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_ICO + if (imagetype==CXIMAGE_FORMAT_ICO){ + CxImageICO newima; + newima.Ghost(this); + if (newima.Encode(hFile)){ + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_TIF + if (imagetype==CXIMAGE_FORMAT_TIF){ + CxImageTIF newima; + newima.Ghost(this); + if (newima.Encode(hFile)){ + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_JPG + if (imagetype==CXIMAGE_FORMAT_JPG){ + CxImageJPG newima; + newima.Ghost(this); + if (newima.Encode(hFile)){ + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_GIF + if (imagetype==CXIMAGE_FORMAT_GIF){ + CxImageGIF newima; + newima.Ghost(this); + if (newima.Encode(hFile)){ + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_PNG + if (imagetype==CXIMAGE_FORMAT_PNG){ + CxImagePNG newima; + newima.Ghost(this); + if (newima.Encode(hFile)){ + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_MNG + if (imagetype==CXIMAGE_FORMAT_MNG){ + CxImageMNG newima; + newima.Ghost(this); + if (newima.Encode(hFile)){ + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_TGA + if (imagetype==CXIMAGE_FORMAT_TGA){ + CxImageTGA newima; + newima.Ghost(this); + if (newima.Encode(hFile)){ + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_PCX + if (imagetype==CXIMAGE_FORMAT_PCX){ + CxImagePCX newima; + newima.Ghost(this); + if (newima.Encode(hFile)){ + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_WBMP + if (imagetype==CXIMAGE_FORMAT_WBMP){ + CxImageWBMP newima; + newima.Ghost(this); + if (newima.Encode(hFile)){ + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_WMF && CXIMAGE_SUPPORT_WINDOWS // - WMF/EMF support + if (imagetype==CXIMAGE_FORMAT_WMF){ + CxImageWMF newima; + newima.Ghost(this); + if (newima.Encode(hFile)){ + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_J2K + if (imagetype==CXIMAGE_FORMAT_J2K){ + CxImageJ2K newima; + newima.Ghost(this); + if (newima.Encode(hFile)){ + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_JBG + if (imagetype==CXIMAGE_FORMAT_JBG){ + CxImageJBG newima; + newima.Ghost(this); + if (newima.Encode(hFile)){ + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_JASPER + if ( + #if CXIMAGE_SUPPORT_JP2 + imagetype==CXIMAGE_FORMAT_JP2 || + #endif + #if CXIMAGE_SUPPORT_JPC + imagetype==CXIMAGE_FORMAT_JPC || + #endif + #if CXIMAGE_SUPPORT_PGX + imagetype==CXIMAGE_FORMAT_PGX || + #endif + #if CXIMAGE_SUPPORT_PNM + imagetype==CXIMAGE_FORMAT_PNM || + #endif + #if CXIMAGE_SUPPORT_RAS + imagetype==CXIMAGE_FORMAT_RAS || + #endif + false ){ + CxImageJAS newima; + newima.Ghost(this); + if (newima.Encode(hFile,imagetype)){ + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif + + strcpy(info.szLastError,"Encode: Unknown format"); + return false; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Saves to disk or memory pagecount images, referenced by an array of CxImage pointers. + * \param hFile: file handle. + * \param pImages: array of CxImage pointers. + * \param pagecount: number of images. + * \param imagetype: can be CXIMAGE_FORMAT_TIF or CXIMAGE_FORMAT_GIF. + * \return true if everything is ok + */ +bool CxImage::Encode(FILE * hFile, CxImage ** pImages, int pagecount, DWORD imagetype) +{ + CxIOFile file(hFile); + return Encode(&file, pImages, pagecount,imagetype); +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Saves to disk or memory pagecount images, referenced by an array of CxImage pointers. + * \param hFile: file handle (implemented using CxMemFile or CxIOFile). + * \param pImages: array of CxImage pointers. + * \param pagecount: number of images. + * \param imagetype: can be CXIMAGE_FORMAT_TIF or CXIMAGE_FORMAT_GIF. + * \return true if everything is ok + */ +bool CxImage::Encode(CxFile * hFile, CxImage ** pImages, int pagecount, DWORD imagetype) +{ +#if CXIMAGE_SUPPORT_TIF + if (imagetype==CXIMAGE_FORMAT_TIF){ + CxImageTIF newima; + newima.Ghost(this); + if (newima.Encode(hFile,pImages,pagecount)){ + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_GIF + if (imagetype==CXIMAGE_FORMAT_GIF){ + CxImageGIF newima; + newima.Ghost(this); + if (newima.Encode(hFile,pImages,pagecount)){ + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif + strcpy(info.szLastError,"Multipage Encode, Unsupported operation for this format"); + return false; +} + +//////////////////////////////////////////////////////////////////////////////// +/** + * exports the image into a RGBA buffer, Useful for OpenGL applications. + * \param buffer: output memory buffer pointer. Must be NULL, + * the function allocates and fill the memory, + * the application must free the buffer, see also FreeMemory(). + * \param size: output memory buffer size. + * \return true if everything is ok + */ +bool CxImage::Encode2RGBA(BYTE * &buffer, long &size) +{ + if (buffer!=NULL){ + strcpy(info.szLastError,"the buffer must be empty"); + return false; + } + CxMemFile file; + file.Open(); + if(Encode2RGBA(&file)){ + buffer=file.GetBuffer(); + size=file.Size(); + return true; + } + return false; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * exports the image into a RGBA buffer, Useful for OpenGL applications. + * \param hFile: file handle (implemented using CxMemFile or CxIOFile). + * \return true if everything is ok + */ +bool CxImage::Encode2RGBA(CxFile *hFile) +{ + if (EncodeSafeCheck(hFile)) return false; + + for (DWORD y = 0; yPutC(color.rgbRed); + hFile->PutC(color.rgbGreen); + hFile->PutC(color.rgbBlue); + hFile->PutC(color.rgbReserved); + } + } + return true; +} + +//////////////////////////////////////////////////////////////////////////////// +#endif //CXIMAGE_SUPPORT_ENCODE +//////////////////////////////////////////////////////////////////////////////// + +//////////////////////////////////////////////////////////////////////////////// +#if CXIMAGE_SUPPORT_DECODE +//////////////////////////////////////////////////////////////////////////////// +// For UNICODE support: char -> TCHAR +/** + * Reads from disk the image in a specific format. + * - If decoding fails using the specified image format, + * the function will try the automatic file format recognition. + * + * \param filename: file name + * \param imagetype: file format, see ENUM_CXIMAGE_FORMATS + * \return true if everything is ok + */ +bool CxImage::Load(const TCHAR * filename, DWORD imagetype) +//bool CxImage::Load(const char * filename, DWORD imagetype) +{ + /*FILE* hFile; //file handle to read the image + if ((hFile=fopen(filename,"rb"))==NULL) return false; + bool bOK = Decode(hFile,imagetype); + fclose(hFile);*/ + + /* automatic file type recognition */ + bool bOK = false; + if ( imagetype > 0 && imagetype < CMAX_IMAGE_FORMATS ){ + FILE* hFile; //file handle to read the image + +#ifdef WIN32 + if ((hFile=_tfopen(filename,_T("rb")))==NULL) return false; // For UNICODE support +#else + if ((hFile=fopen(filename,"rb"))==NULL) return false; +#endif + + bOK = Decode(hFile,imagetype); + fclose(hFile); + if (bOK) return bOK; + } + + char szError[256]; + strcpy(szError,info.szLastError); //save the first error + + // if failed, try automatic recognition of the file... + FILE* hFile; + +#ifdef WIN32 + if ((hFile=_tfopen(filename,_T("rb")))==NULL) return false; // For UNICODE support +#else + if ((hFile=fopen(filename,"rb"))==NULL) return false; +#endif + + bOK = Decode(hFile,CXIMAGE_FORMAT_UNKNOWN); + fclose(hFile); + + if (!bOK && imagetype > 0) strcpy(info.szLastError,szError); //restore the first error + + return bOK; +} +//////////////////////////////////////////////////////////////////////////////// +#ifdef WIN32 +//bool CxImage::Load(LPCWSTR filename, DWORD imagetype) +//{ +// /*FILE* hFile; //file handle to read the image +// if ((hFile=_wfopen(filename, L"rb"))==NULL) return false; +// bool bOK = Decode(hFile,imagetype); +// fclose(hFile);*/ +// +// /* automatic file type recognition */ +// bool bOK = false; +// if ( imagetype > 0 && imagetype < CMAX_IMAGE_FORMATS ){ +// FILE* hFile; //file handle to read the image +// if ((hFile=_wfopen(filename,L"rb"))==NULL) return false; +// bOK = Decode(hFile,imagetype); +// fclose(hFile); +// if (bOK) return bOK; +// } +// +// char szError[256]; +// strcpy(szError,info.szLastError); //save the first error +// +// // if failed, try automatic recognition of the file... +// FILE* hFile; //file handle to read the image +// if ((hFile=_wfopen(filename,L"rb"))==NULL) return false; +// bOK = Decode(hFile,CXIMAGE_FORMAT_UNKNOWN); +// fclose(hFile); +// +// if (!bOK && imagetype > 0) strcpy(info.szLastError,szError); //restore the first error +// +// return bOK; +//} +//////////////////////////////////////////////////////////////////////////////// +/** + * Loads an image from the application resources. + * \param hRes: the resource handle returned by FindResource(). + * \param imagetype: file format, see ENUM_CXIMAGE_FORMATS. + * \param hModule: NULL for internal resource, or external application/DLL hinstance returned by LoadLibray. + * \return true if everything is ok + */ +bool CxImage::LoadResource(HRSRC hRes, DWORD imagetype, HMODULE hModule) +{ + DWORD rsize=SizeofResource(hModule,hRes); + HGLOBAL hMem=::LoadResource(hModule,hRes); + if (hMem){ + char* lpVoid=(char*)LockResource(hMem); + if (lpVoid){ + // FILE* fTmp=tmpfile(); doesn't work with network + /*char tmpPath[MAX_PATH] = {0}; + char tmpFile[MAX_PATH] = {0}; + GetTempPath(MAX_PATH,tmpPath); + GetTempFileName(tmpPath,"IMG",0,tmpFile); + FILE* fTmp=fopen(tmpFile,"w+b"); + if (fTmp){ + fwrite(lpVoid,rsize,1,fTmp); + fseek(fTmp,0,SEEK_SET); + bool bOK = Decode(fTmp,imagetype); + fclose(fTmp); + DeleteFile(tmpFile); + return bOK; + }*/ + + CxMemFile fTmp((BYTE*)lpVoid,rsize); + return Decode(&fTmp,imagetype); + } + } else strcpy(info.szLastError,"Unable to load resource!"); + return false; +} +#endif //WIN32 +//////////////////////////////////////////////////////////////////////////////// +/** + * Constructor from file name, see Load() + * \param filename: file name + * \param imagetype: file format, see ENUM_CXIMAGE_FORMATS + */ +// +// > filename: file name +// > imagetype: specify the image format (CXIMAGE_FORMAT_BMP,...) +// For UNICODE support: char -> TCHAR +CxImage::CxImage(const TCHAR * filename, DWORD imagetype) +//CxImage::CxImage(const char * filename, DWORD imagetype) +{ + Startup(imagetype); + Load(filename,imagetype); +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Constructor from file handle, see Decode() + * \param stream: file handle, with read access. + * \param imagetype: file format, see ENUM_CXIMAGE_FORMATS + */ +CxImage::CxImage(FILE * stream, DWORD imagetype) +{ + Startup(imagetype); + Decode(stream,imagetype); +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Constructor from CxFile object, see Decode() + * \param stream: file handle (implemented using CxMemFile or CxIOFile), with read access. + * \param imagetype: file format, see ENUM_CXIMAGE_FORMATS + */ +CxImage::CxImage(CxFile * stream, DWORD imagetype) +{ + Startup(imagetype); + Decode(stream,imagetype); +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Constructor from memory buffer, see Decode() + * \param buffer: memory buffer + * \param size: size of buffer + * \param imagetype: file format, see ENUM_CXIMAGE_FORMATS + */ +CxImage::CxImage(BYTE * buffer, DWORD size, DWORD imagetype) +{ + Startup(imagetype); + CxMemFile stream(buffer,size); + Decode(&stream,imagetype); +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Loads an image from memory buffer + * \param buffer: memory buffer + * \param size: size of buffer + * \param imagetype: file format, see ENUM_CXIMAGE_FORMATS + * \return true if everything is ok + */ +bool CxImage::Decode(BYTE * buffer, DWORD size, DWORD imagetype) +{ + CxMemFile file(buffer,size); + return Decode(&file,imagetype); +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Loads an image from file handle. + * \param hFile: file handle, with read access. + * \param imagetype: file format, see ENUM_CXIMAGE_FORMATS + * \return true if everything is ok + */ +bool CxImage::Decode(FILE *hFile, DWORD imagetype) +{ + CxIOFile file(hFile); + return Decode(&file,imagetype); +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Loads an image from CxFile object + * \param hFile: file handle (implemented using CxMemFile or CxIOFile), with read access. + * \param imagetype: file format, see ENUM_CXIMAGE_FORMATS + * \return true if everything is ok + * \sa ENUM_CXIMAGE_FORMATS + */ +bool CxImage::Decode(CxFile *hFile, DWORD imagetype) +{ + + if (imagetype==CXIMAGE_FORMAT_UNKNOWN){ + DWORD pos = hFile->Tell(); +#if CXIMAGE_SUPPORT_BMP + { CxImageBMP newima; newima.CopyInfo(*this); if (newima.Decode(hFile)) { Transfer(newima); return true; } else hFile->Seek(pos,SEEK_SET); } +#endif +#if CXIMAGE_SUPPORT_JPG + { CxImageJPG newima; newima.CopyInfo(*this); if (newima.Decode(hFile)) { Transfer(newima); return true; } else hFile->Seek(pos,SEEK_SET); } +#endif +#if CXIMAGE_SUPPORT_ICO + { CxImageICO newima; newima.CopyInfo(*this); if (newima.Decode(hFile)) { Transfer(newima); return true; } else hFile->Seek(pos,SEEK_SET); } +#endif +#if CXIMAGE_SUPPORT_GIF + { CxImageGIF newima; newima.CopyInfo(*this); if (newima.Decode(hFile)) { Transfer(newima); return true; } else hFile->Seek(pos,SEEK_SET); } +#endif +#if CXIMAGE_SUPPORT_PNG + { CxImagePNG newima; newima.CopyInfo(*this); if (newima.Decode(hFile)) { Transfer(newima); return true; } else hFile->Seek(pos,SEEK_SET); } +#endif +#if CXIMAGE_SUPPORT_TIF + { CxImageTIF newima; newima.CopyInfo(*this); if (newima.Decode(hFile)) { Transfer(newima); return true; } else hFile->Seek(pos,SEEK_SET); } +#endif +#if CXIMAGE_SUPPORT_MNG + { CxImageMNG newima; newima.CopyInfo(*this); if (newima.Decode(hFile)) { Transfer(newima); return true; } else hFile->Seek(pos,SEEK_SET); } +#endif +#if CXIMAGE_SUPPORT_TGA + { CxImageTGA newima; newima.CopyInfo(*this); if (newima.Decode(hFile)) { Transfer(newima); return true; } else hFile->Seek(pos,SEEK_SET); } +#endif +#if CXIMAGE_SUPPORT_PCX + { CxImagePCX newima; newima.CopyInfo(*this); if (newima.Decode(hFile)) { Transfer(newima); return true; } else hFile->Seek(pos,SEEK_SET); } +#endif +#if CXIMAGE_SUPPORT_WBMP + { CxImageWBMP newima; newima.CopyInfo(*this); if (newima.Decode(hFile)) { Transfer(newima); return true; } else hFile->Seek(pos,SEEK_SET); } +#endif +#if CXIMAGE_SUPPORT_WMF && CXIMAGE_SUPPORT_WINDOWS + { CxImageWMF newima; newima.CopyInfo(*this); if (newima.Decode(hFile)) { Transfer(newima); return true; } else hFile->Seek(pos,SEEK_SET); } +#endif +#if CXIMAGE_SUPPORT_J2K + { CxImageJ2K newima; newima.CopyInfo(*this); if (newima.Decode(hFile)) { Transfer(newima); return true; } else hFile->Seek(pos,SEEK_SET); } +#endif +#if CXIMAGE_SUPPORT_JBG + { CxImageJBG newima; newima.CopyInfo(*this); if (newima.Decode(hFile)) { Transfer(newima); return true; } else hFile->Seek(pos,SEEK_SET); } +#endif +#if CXIMAGE_SUPPORT_JASPER + { CxImageJAS newima; newima.CopyInfo(*this); if (newima.Decode(hFile)) { Transfer(newima); return true; } else hFile->Seek(pos,SEEK_SET); } +#endif + } + +#if CXIMAGE_SUPPORT_BMP + if (imagetype==CXIMAGE_FORMAT_BMP){ + CxImageBMP newima; + newima.CopyInfo(*this); + if (newima.Decode(hFile)){ + Transfer(newima); + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_JPG + if (imagetype==CXIMAGE_FORMAT_JPG){ + CxImageJPG newima; + newima.CopyInfo(*this); // + if (newima.Decode(hFile)){ + Transfer(newima); + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_ICO + if (imagetype==CXIMAGE_FORMAT_ICO){ + CxImageICO newima; + newima.CopyInfo(*this); + if (newima.Decode(hFile)){ + Transfer(newima); + return true; + } else { + info.nNumFrames = newima.info.nNumFrames; + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_GIF + if (imagetype==CXIMAGE_FORMAT_GIF){ + CxImageGIF newima; + newima.CopyInfo(*this); + if (newima.Decode(hFile)){ + Transfer(newima); + return true; + } else { + info.nNumFrames = newima.info.nNumFrames; + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_PNG + if (imagetype==CXIMAGE_FORMAT_PNG){ + CxImagePNG newima; + newima.CopyInfo(*this); + if (newima.Decode(hFile)){ + Transfer(newima); + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_TIF + if (imagetype==CXIMAGE_FORMAT_TIF){ + CxImageTIF newima; + newima.CopyInfo(*this); + if (newima.Decode(hFile)){ + Transfer(newima); + return true; + } else { + info.nNumFrames = newima.info.nNumFrames; + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_MNG + if (imagetype==CXIMAGE_FORMAT_MNG){ + CxImageMNG newima; + newima.CopyInfo(*this); + if (newima.Decode(hFile)){ + Transfer(newima); + return true; + } else { + info.nNumFrames = newima.info.nNumFrames; + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_TGA + if (imagetype==CXIMAGE_FORMAT_TGA){ + CxImageTGA newima; + newima.CopyInfo(*this); + if (newima.Decode(hFile)){ + Transfer(newima); + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_PCX + if (imagetype==CXIMAGE_FORMAT_PCX){ + CxImagePCX newima; + newima.CopyInfo(*this); + if (newima.Decode(hFile)){ + Transfer(newima); + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_WBMP + if (imagetype==CXIMAGE_FORMAT_WBMP){ + CxImageWBMP newima; + newima.CopyInfo(*this); + if (newima.Decode(hFile)){ + Transfer(newima); + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_WMF && CXIMAGE_SUPPORT_WINDOWS // vho - WMF support + if (imagetype == CXIMAGE_FORMAT_WMF){ + CxImageWMF newima; + newima.CopyInfo(*this); + if (newima.Decode(hFile)){ + Transfer(newima); + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_J2K + if (imagetype==CXIMAGE_FORMAT_J2K){ + CxImageJ2K newima; + newima.CopyInfo(*this); + if (newima.Decode(hFile)){ + Transfer(newima); + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_JBG + if (imagetype==CXIMAGE_FORMAT_JBG){ + CxImageJBG newima; + newima.CopyInfo(*this); + if (newima.Decode(hFile)){ + Transfer(newima); + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif +#if CXIMAGE_SUPPORT_JASPER + if ( + #if CXIMAGE_SUPPORT_JP2 + imagetype==CXIMAGE_FORMAT_JP2 || + #endif + #if CXIMAGE_SUPPORT_JPC + imagetype==CXIMAGE_FORMAT_JPC || + #endif + #if CXIMAGE_SUPPORT_PGX + imagetype==CXIMAGE_FORMAT_PGX || + #endif + #if CXIMAGE_SUPPORT_PNM + imagetype==CXIMAGE_FORMAT_PNM || + #endif + #if CXIMAGE_SUPPORT_RAS + imagetype==CXIMAGE_FORMAT_RAS || + #endif + false ){ + CxImageJAS newima; + newima.CopyInfo(*this); + if (newima.Decode(hFile,imagetype)){ + Transfer(newima); + return true; + } else { + strcpy(info.szLastError,newima.GetLastError()); + return false; + } + } +#endif + + strcpy(info.szLastError,"Decode: Unknown or wrong format"); + return false; +} +//////////////////////////////////////////////////////////////////////////////// +#endif //CXIMAGE_SUPPORT_DECODE +//////////////////////////////////////////////////////////////////////////////// diff --git a/src/win32/dependencies/cximage/ximaexif.cpp b/src/win32/dependencies/cximage/ximaexif.cpp new file mode 100644 index 00000000..17f962a1 --- /dev/null +++ b/src/win32/dependencies/cximage/ximaexif.cpp @@ -0,0 +1,873 @@ +/* + * File: ximaexif.cpp + * Purpose: EXIF reader + * 18/Aug/2002 Davide Pizzolato - www.xdp.it + * CxImage version 5.99c 17/Oct/2004 + * based on jhead-1.8 by Matthias Wandel + */ + +#include "ximajpg.h" + +#if CXIMAGEJPG_SUPPORT_EXIF + +//////////////////////////////////////////////////////////////////////////////// +CxImageJPG::CxExifInfo::CxExifInfo(EXIFINFO* info) +{ + if (info) { + m_exifinfo = info; + freeinfo = false; + } else { + m_exifinfo = new EXIFINFO; + memset(m_exifinfo,0,sizeof(EXIFINFO)); + freeinfo = true; + } + + m_szLastError[0]='\0'; + ExifImageWidth = MotorolaOrder = 0; + SectionsRead=0; + memset(&Sections, 0, MAX_SECTIONS * sizeof(Section_t)); +} +//////////////////////////////////////////////////////////////////////////////// +CxImageJPG::CxExifInfo::~CxExifInfo() +{ + for(int i=0;iGetC(); + + if (a != 0xff || hFile->GetC() != M_SOI){ + return FALSE; + } + + for(;;){ + int itemlen; + int marker = 0; + int ll,lh, got; + BYTE * Data; + + if (SectionsRead >= MAX_SECTIONS){ + strcpy(m_szLastError,"Too many sections in jpg file"); + return false; + } + + for (a=0;a<7;a++){ + marker = hFile->GetC(); + if (marker != 0xff) break; + + if (a >= 6){ + printf("too many padding bytes\n"); + return false; + } + } + + if (marker == 0xff){ + // 0xff is legal padding, but if we get that many, something's wrong. + strcpy(m_szLastError,"too many padding bytes!"); + return false; + } + + Sections[SectionsRead].Type = marker; + + // Read the length of the section. + lh = hFile->GetC(); + ll = hFile->GetC(); + + itemlen = (lh << 8) | ll; + + if (itemlen < 2){ + strcpy(m_szLastError,"invalid marker"); + return false; + } + + Sections[SectionsRead].Size = itemlen; + + Data = (BYTE *)malloc(itemlen); + if (Data == NULL){ + strcpy(m_szLastError,"Could not allocate memory"); + return false; + } + Sections[SectionsRead].Data = Data; + + // Store first two pre-read bytes. + Data[0] = (BYTE)lh; + Data[1] = (BYTE)ll; + + got = hFile->Read(Data+2, 1, itemlen-2); // Read the whole section. + if (got != itemlen-2){ + strcpy(m_szLastError,"Premature end of file?"); + return false; + } + SectionsRead += 1; + + switch(marker){ + + case M_SOS: // stop before hitting compressed data + // If reading entire image is requested, read the rest of the data. + if (nReadMode & EXIF_READ_IMAGE){ + int cp, ep, size; + // Determine how much file is left. + cp = hFile->Tell(); + hFile->Seek(0, SEEK_END); + ep = hFile->Tell(); + hFile->Seek(cp, SEEK_SET); + + size = ep-cp; + Data = (BYTE *)malloc(size); + if (Data == NULL){ + strcpy(m_szLastError,"could not allocate data for entire image"); + return false; + } + + got = hFile->Read(Data, 1, size); + if (got != size){ + strcpy(m_szLastError,"could not read the rest of the image"); + return false; + } + + Sections[SectionsRead].Data = Data; + Sections[SectionsRead].Size = size; + Sections[SectionsRead].Type = PSEUDO_IMAGE_MARKER; + SectionsRead ++; + } + return true; + + case M_EOI: // in case it's a tables-only JPEG stream + printf("No image in jpeg!\n"); + return FALSE; + + case M_COM: // Comment section + if (HaveCom || ((nReadMode & EXIF_READ_EXIF) == 0)){ + // Discard this section. + free(Sections[--SectionsRead].Data); + Sections[SectionsRead].Data=0; + }else{ + process_COM(Data, itemlen); + HaveCom = TRUE; + } + break; + + case M_JFIF: + // Regular jpegs always have this tag, exif images have the exif + // marker instead, althogh ACDsee will write images with both markers. + // this program will re-create this marker on absence of exif marker. + // hence no need to keep the copy from the file. + free(Sections[--SectionsRead].Data); + Sections[SectionsRead].Data=0; + break; + + case M_EXIF: + // Seen files from some 'U-lead' software with Vivitar scanner + // that uses marker 31 for non exif stuff. Thus make sure + // it says 'Exif' in the section before treating it as exif. + if ((nReadMode & EXIF_READ_EXIF) && memcmp(Data+2, "Exif", 4) == 0){ + m_exifinfo->IsExif = process_EXIF((BYTE *)Data+2, itemlen); + }else{ + // Discard this section. + free(Sections[--SectionsRead].Data); + Sections[SectionsRead].Data=0; + } + break; + + case M_SOF0: + case M_SOF1: + case M_SOF2: + case M_SOF3: + case M_SOF5: + case M_SOF6: + case M_SOF7: + case M_SOF9: + case M_SOF10: + case M_SOF11: + case M_SOF13: + case M_SOF14: + case M_SOF15: + process_SOFn(Data, marker); + break; + default: + // Skip any other sections. + //if (ShowTags) printf("Jpeg section marker 0x%02x size %d\n",marker, itemlen); + break; + } + } + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/*-------------------------------------------------------------------------- + Process a EXIF marker + Describes all the drivel that most digital cameras include... +--------------------------------------------------------------------------*/ +bool CxImageJPG::CxExifInfo::process_EXIF(unsigned char * CharBuf, unsigned int length) +{ + m_exifinfo->FlashUsed = 0; + /* If it's from a digicam, and it used flash, it says so. */ + m_exifinfo->Comments[0] = '\0'; /* Initial value - null string */ + + ExifImageWidth = 0; + + { /* Check the EXIF header component */ + static const unsigned char ExifHeader[] = "Exif\0\0"; + if (memcmp(CharBuf+0, ExifHeader,6)){ + strcpy(m_szLastError,"Incorrect Exif header"); + return false; + } + } + + if (memcmp(CharBuf+6,"II",2) == 0){ + MotorolaOrder = 0; + }else{ + if (memcmp(CharBuf+6,"MM",2) == 0){ + MotorolaOrder = 1; + }else{ + strcpy(m_szLastError,"Invalid Exif alignment marker."); + return false; + } + } + + /* Check the next two values for correctness. */ + if (Get16u(CharBuf+8) != 0x2a){ + strcpy(m_szLastError,"Invalid Exif start (1)"); + return false; + } + + int FirstOffset = Get32u(CharBuf+10); + /* + if (FirstOffset < 8 || FirstOffset > 16){ + // I used to ensure this was set to 8 (website I used indicated its 8) + // but PENTAX Optio 230 has it set differently, and uses it as offset. (Sept 11 2002) + strcpy(m_szLastError,"Suspicious offset of first IFD value"); + return false; + }*/ + + unsigned char * LastExifRefd = CharBuf; + + /* First directory starts 16 bytes in. Offsets start at 8 bytes in. */ + if (!ProcessExifDir(CharBuf+14, CharBuf+6, length-6, m_exifinfo, &LastExifRefd)) + return false; + + /* give a chance for a second directory */ + if (FirstOffset > 8) { + if (!ProcessExifDir(CharBuf+14+FirstOffset-8, CharBuf+6, length-6, m_exifinfo, &LastExifRefd)) + return false; + } + + /* This is how far the interesting (non thumbnail) part of the exif went. */ + // int ExifSettingsLength = LastExifRefd - CharBuf; + + /* Compute the CCD width, in milimeters. */ + if (m_exifinfo->FocalplaneXRes != 0){ + m_exifinfo->CCDWidth = (float)(ExifImageWidth * m_exifinfo->FocalplaneUnits / m_exifinfo->FocalplaneXRes); + } + + return true; +} +//-------------------------------------------------------------------------- +// Get 16 bits motorola order (always) for jpeg header stuff. +//-------------------------------------------------------------------------- +int CxImageJPG::CxExifInfo::Get16m(void * Short) +{ + return (((unsigned char *)Short)[0] << 8) | ((unsigned char *)Short)[1]; +} +//////////////////////////////////////////////////////////////////////////////// +/*-------------------------------------------------------------------------- + Convert a 16 bit unsigned value from file's native byte order +--------------------------------------------------------------------------*/ +int CxImageJPG::CxExifInfo::Get16u(void * Short) +{ + if (MotorolaOrder){ + return (((unsigned char *)Short)[0] << 8) | ((unsigned char *)Short)[1]; + }else{ + return (((unsigned char *)Short)[1] << 8) | ((unsigned char *)Short)[0]; + } +} +//////////////////////////////////////////////////////////////////////////////// +/*-------------------------------------------------------------------------- + Convert a 32 bit signed value from file's native byte order +--------------------------------------------------------------------------*/ +long CxImageJPG::CxExifInfo::Get32s(void * Long) +{ + if (MotorolaOrder){ + return ((( char *)Long)[0] << 24) | (((unsigned char *)Long)[1] << 16) + | (((unsigned char *)Long)[2] << 8 ) | (((unsigned char *)Long)[3] << 0 ); + }else{ + return ((( char *)Long)[3] << 24) | (((unsigned char *)Long)[2] << 16) + | (((unsigned char *)Long)[1] << 8 ) | (((unsigned char *)Long)[0] << 0 ); + } +} +//////////////////////////////////////////////////////////////////////////////// +/*-------------------------------------------------------------------------- + Convert a 32 bit unsigned value from file's native byte order +--------------------------------------------------------------------------*/ +unsigned long CxImageJPG::CxExifInfo::Get32u(void * Long) +{ + return (unsigned long)Get32s(Long) & 0xffffffff; +} +//////////////////////////////////////////////////////////////////////////////// + +/* Describes format descriptor */ +static const int BytesPerFormat[] = {0,1,1,2,4,8,1,1,2,4,8,4,8}; +#define NUM_FORMATS 12 + +#define FMT_BYTE 1 +#define FMT_STRING 2 +#define FMT_USHORT 3 +#define FMT_ULONG 4 +#define FMT_URATIONAL 5 +#define FMT_SBYTE 6 +#define FMT_UNDEFINED 7 +#define FMT_SSHORT 8 +#define FMT_SLONG 9 +#define FMT_SRATIONAL 10 +#define FMT_SINGLE 11 +#define FMT_DOUBLE 12 + +/* Describes tag values */ + +#define TAG_EXIF_VERSION 0x9000 +#define TAG_EXIF_OFFSET 0x8769 +#define TAG_INTEROP_OFFSET 0xa005 + +#define TAG_MAKE 0x010F +#define TAG_MODEL 0x0110 + +#define TAG_ORIENTATION 0x0112 +#define TAG_XRESOLUTION 0x011A +#define TAG_YRESOLUTION 0x011B +#define TAG_RESOLUTIONUNIT 0x0128 + +#define TAG_EXPOSURETIME 0x829A +#define TAG_FNUMBER 0x829D + +#define TAG_SHUTTERSPEED 0x9201 +#define TAG_APERTURE 0x9202 +#define TAG_BRIGHTNESS 0x9203 +#define TAG_MAXAPERTURE 0x9205 +#define TAG_FOCALLENGTH 0x920A + +#define TAG_DATETIME_ORIGINAL 0x9003 +#define TAG_USERCOMMENT 0x9286 + +#define TAG_SUBJECT_DISTANCE 0x9206 +#define TAG_FLASH 0x9209 + +#define TAG_FOCALPLANEXRES 0xa20E +#define TAG_FOCALPLANEYRES 0xa20F +#define TAG_FOCALPLANEUNITS 0xa210 +#define TAG_EXIF_IMAGEWIDTH 0xA002 +#define TAG_EXIF_IMAGELENGTH 0xA003 + +/* the following is added 05-jan-2001 vcs */ +#define TAG_EXPOSURE_BIAS 0x9204 +#define TAG_WHITEBALANCE 0x9208 +#define TAG_METERING_MODE 0x9207 +#define TAG_EXPOSURE_PROGRAM 0x8822 +#define TAG_ISO_EQUIVALENT 0x8827 +#define TAG_COMPRESSION_LEVEL 0x9102 + +#define TAG_THUMBNAIL_OFFSET 0x0201 +#define TAG_THUMBNAIL_LENGTH 0x0202 + + +/*-------------------------------------------------------------------------- + Process one of the nested EXIF directories. +--------------------------------------------------------------------------*/ +bool CxImageJPG::CxExifInfo::ProcessExifDir(unsigned char * DirStart, unsigned char * OffsetBase, unsigned ExifLength, + EXIFINFO * const m_exifinfo, unsigned char ** const LastExifRefdP ) +{ + int de; + int a; + int NumDirEntries; + unsigned ThumbnailOffset = 0; + unsigned ThumbnailSize = 0; + + NumDirEntries = Get16u(DirStart); + + if ((DirStart+2+NumDirEntries*12) > (OffsetBase+ExifLength)){ + strcpy(m_szLastError,"Illegally sized directory"); + return false; + } + + for (de=0;de= NUM_FORMATS) { + /* (-1) catches illegal zero case as unsigned underflows to positive large */ + strcpy(m_szLastError,"Illegal format code in EXIF dir"); + return false; + } + + ByteCount = Components * BytesPerFormat[Format]; + + if (ByteCount > 4){ + unsigned OffsetVal; + OffsetVal = Get32u(DirEntry+8); + /* If its bigger than 4 bytes, the dir entry contains an offset.*/ + if (OffsetVal+ByteCount > ExifLength){ + /* Bogus pointer offset and / or bytecount value */ + strcpy(m_szLastError,"Illegal pointer offset value in EXIF."); + return false; + } + ValuePtr = OffsetBase+OffsetVal; + }else{ + /* 4 bytes or less and value is in the dir entry itself */ + ValuePtr = DirEntry+8; + } + + if (*LastExifRefdP < ValuePtr+ByteCount){ + /* Keep track of last byte in the exif header that was + actually referenced. That way, we know where the + discardable thumbnail data begins. + */ + *LastExifRefdP = ValuePtr+ByteCount; + } + + /* Extract useful components of tag */ + switch(Tag){ + + case TAG_MAKE: + strncpy(m_exifinfo->CameraMake, (char*)ValuePtr, 31); + break; + + case TAG_MODEL: + strncpy(m_exifinfo->CameraModel, (char*)ValuePtr, 39); + break; + + case TAG_EXIF_VERSION: + strncpy(m_exifinfo->Version,(char*)ValuePtr, 4); + break; + + case TAG_DATETIME_ORIGINAL: + strncpy(m_exifinfo->DateTime, (char*)ValuePtr, 19); + break; + + case TAG_USERCOMMENT: + // Olympus has this padded with trailing spaces. Remove these first. + for (a=ByteCount;;){ + a--; + if (((char*)ValuePtr)[a] == ' '){ + ((char*)ValuePtr)[a] = '\0'; + }else{ + break; + } + if (a == 0) break; + } + + /* Copy the comment */ + if (memcmp(ValuePtr, "ASCII",5) == 0){ + for (a=5;a<10;a++){ + char c; + c = ((char*)ValuePtr)[a]; + if (c != '\0' && c != ' '){ + strncpy(m_exifinfo->Comments, (char*)ValuePtr+a, 199); + break; + } + } + + }else{ + strncpy(m_exifinfo->Comments, (char*)ValuePtr, 199); + } + break; + + case TAG_FNUMBER: + /* Simplest way of expressing aperture, so I trust it the most. + (overwrite previously computd value if there is one) + */ + m_exifinfo->ApertureFNumber = (float)ConvertAnyFormat(ValuePtr, Format); + break; + + case TAG_APERTURE: + case TAG_MAXAPERTURE: + /* More relevant info always comes earlier, so only + use this field if we don't have appropriate aperture + information yet. + */ + if (m_exifinfo->ApertureFNumber == 0){ + m_exifinfo->ApertureFNumber = (float)exp(ConvertAnyFormat(ValuePtr, Format)*log(2.0f)*0.5); + } + break; + + case TAG_BRIGHTNESS: + m_exifinfo->Brightness = (float)ConvertAnyFormat(ValuePtr, Format); + break; + + case TAG_FOCALLENGTH: + /* Nice digital cameras actually save the focal length + as a function of how farthey are zoomed in. + */ + + m_exifinfo->FocalLength = (float)ConvertAnyFormat(ValuePtr, Format); + break; + + case TAG_SUBJECT_DISTANCE: + /* Inidcates the distacne the autofocus camera is focused to. + Tends to be less accurate as distance increases. + */ + m_exifinfo->Distance = (float)ConvertAnyFormat(ValuePtr, Format); + break; + + case TAG_EXPOSURETIME: + /* Simplest way of expressing exposure time, so I + trust it most. (overwrite previously computd value + if there is one) + */ + m_exifinfo->ExposureTime = + (float)ConvertAnyFormat(ValuePtr, Format); + break; + + case TAG_SHUTTERSPEED: + /* More complicated way of expressing exposure time, + so only use this value if we don't already have it + from somewhere else. + */ + if (m_exifinfo->ExposureTime == 0){ + m_exifinfo->ExposureTime = (float) + (1/exp(ConvertAnyFormat(ValuePtr, Format)*log(2.0f))); + } + break; + + case TAG_FLASH: + if ((int)ConvertAnyFormat(ValuePtr, Format) & 7){ + m_exifinfo->FlashUsed = 1; + }else{ + m_exifinfo->FlashUsed = 0; + } + break; + + case TAG_ORIENTATION: + m_exifinfo->Orientation = (int)ConvertAnyFormat(ValuePtr, Format); + if (m_exifinfo->Orientation < 1 || m_exifinfo->Orientation > 8){ + strcpy(m_szLastError,"Undefined rotation value"); + m_exifinfo->Orientation = 0; + } + break; + + case TAG_EXIF_IMAGELENGTH: + case TAG_EXIF_IMAGEWIDTH: + /* Use largest of height and width to deal with images + that have been rotated to portrait format. + */ + a = (int)ConvertAnyFormat(ValuePtr, Format); + if (ExifImageWidth < a) ExifImageWidth = a; + break; + + case TAG_FOCALPLANEXRES: + m_exifinfo->FocalplaneXRes = (float)ConvertAnyFormat(ValuePtr, Format); + break; + + case TAG_FOCALPLANEYRES: + m_exifinfo->FocalplaneYRes = (float)ConvertAnyFormat(ValuePtr, Format); + break; + + case TAG_RESOLUTIONUNIT: + switch((int)ConvertAnyFormat(ValuePtr, Format)){ + case 1: m_exifinfo->ResolutionUnit = 1.0f; break; /* 1 inch */ + case 2: m_exifinfo->ResolutionUnit = 1.0f; break; + case 3: m_exifinfo->ResolutionUnit = 0.3937007874f; break; /* 1 centimeter*/ + case 4: m_exifinfo->ResolutionUnit = 0.03937007874f; break; /* 1 millimeter*/ + case 5: m_exifinfo->ResolutionUnit = 0.00003937007874f; /* 1 micrometer*/ + } + break; + + case TAG_FOCALPLANEUNITS: + switch((int)ConvertAnyFormat(ValuePtr, Format)){ + case 1: m_exifinfo->FocalplaneUnits = 1.0f; break; /* 1 inch */ + case 2: m_exifinfo->FocalplaneUnits = 1.0f; break; + case 3: m_exifinfo->FocalplaneUnits = 0.3937007874f; break; /* 1 centimeter*/ + case 4: m_exifinfo->FocalplaneUnits = 0.03937007874f; break; /* 1 millimeter*/ + case 5: m_exifinfo->FocalplaneUnits = 0.00003937007874f; /* 1 micrometer*/ + } + break; + + // Remaining cases contributed by: Volker C. Schoech + + case TAG_EXPOSURE_BIAS: + m_exifinfo->ExposureBias = (float) ConvertAnyFormat(ValuePtr, Format); + break; + + case TAG_WHITEBALANCE: + m_exifinfo->Whitebalance = (int)ConvertAnyFormat(ValuePtr, Format); + break; + + case TAG_METERING_MODE: + m_exifinfo->MeteringMode = (int)ConvertAnyFormat(ValuePtr, Format); + break; + + case TAG_EXPOSURE_PROGRAM: + m_exifinfo->ExposureProgram = (int)ConvertAnyFormat(ValuePtr, Format); + break; + + case TAG_ISO_EQUIVALENT: + m_exifinfo->ISOequivalent = (int)ConvertAnyFormat(ValuePtr, Format); + if ( m_exifinfo->ISOequivalent < 50 ) m_exifinfo->ISOequivalent *= 200; + break; + + case TAG_COMPRESSION_LEVEL: + m_exifinfo->CompressionLevel = (int)ConvertAnyFormat(ValuePtr, Format); + break; + + case TAG_XRESOLUTION: + m_exifinfo->Xresolution = (float)ConvertAnyFormat(ValuePtr, Format); + break; + case TAG_YRESOLUTION: + m_exifinfo->Yresolution = (float)ConvertAnyFormat(ValuePtr, Format); + break; + + case TAG_THUMBNAIL_OFFSET: + ThumbnailOffset = (unsigned)ConvertAnyFormat(ValuePtr, Format); + break; + + case TAG_THUMBNAIL_LENGTH: + ThumbnailSize = (unsigned)ConvertAnyFormat(ValuePtr, Format); + break; + + } + + if (Tag == TAG_EXIF_OFFSET || Tag == TAG_INTEROP_OFFSET){ + unsigned char * SubdirStart; + unsigned Offset = Get32u(ValuePtr); + if (Offset>8){ + SubdirStart = OffsetBase + Offset; + if (SubdirStart < OffsetBase || + SubdirStart > OffsetBase+ExifLength){ + strcpy(m_szLastError,"Illegal subdirectory link"); + return false; + } + ProcessExifDir(SubdirStart, OffsetBase, ExifLength, m_exifinfo, LastExifRefdP); + } + continue; + } + } + + + { + /* In addition to linking to subdirectories via exif tags, + there's also a potential link to another directory at the end + of each directory. This has got to be the result of a + committee! + */ + unsigned char * SubdirStart; + unsigned Offset; + Offset = Get16u(DirStart+2+12*NumDirEntries); + if (Offset){ + SubdirStart = OffsetBase + Offset; + if (SubdirStart < OffsetBase + || SubdirStart > OffsetBase+ExifLength){ + strcpy(m_szLastError,"Illegal subdirectory link"); + return false; + } + ProcessExifDir(SubdirStart, OffsetBase, ExifLength, m_exifinfo, LastExifRefdP); + } + } + + + if (ThumbnailSize && ThumbnailOffset){ + if (ThumbnailSize + ThumbnailOffset <= ExifLength){ + /* The thumbnail pointer appears to be valid. Store it. */ + m_exifinfo->ThumbnailPointer = OffsetBase + ThumbnailOffset; + m_exifinfo->ThumbnailSize = ThumbnailSize; + } + } + + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/*-------------------------------------------------------------------------- + Evaluate number, be it int, rational, or float from directory. +--------------------------------------------------------------------------*/ +double CxImageJPG::CxExifInfo::ConvertAnyFormat(void * ValuePtr, int Format) +{ + double Value; + Value = 0; + + switch(Format){ + case FMT_SBYTE: Value = *(signed char *)ValuePtr; break; + case FMT_BYTE: Value = *(unsigned char *)ValuePtr; break; + + case FMT_USHORT: Value = Get16u(ValuePtr); break; + case FMT_ULONG: Value = Get32u(ValuePtr); break; + + case FMT_URATIONAL: + case FMT_SRATIONAL: + { + int Num,Den; + Num = Get32s(ValuePtr); + Den = Get32s(4+(char *)ValuePtr); + if (Den == 0){ + Value = 0; + }else{ + Value = (double)Num/Den; + } + break; + } + + case FMT_SSHORT: Value = (signed short)Get16u(ValuePtr); break; + case FMT_SLONG: Value = Get32s(ValuePtr); break; + + /* Not sure if this is correct (never seen float used in Exif format) + */ + case FMT_SINGLE: Value = (double)*(float *)ValuePtr; break; + case FMT_DOUBLE: Value = *(double *)ValuePtr; break; + } + return Value; +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageJPG::CxExifInfo::process_COM (const BYTE * Data, int length) +{ + int ch; + char Comment[MAX_COMMENT+1]; + int nch; + int a; + + nch = 0; + + if (length > MAX_COMMENT) length = MAX_COMMENT; // Truncate if it won't fit in our structure. + + for (a=2;aComments,Comment); +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageJPG::CxExifInfo::process_SOFn (const BYTE * Data, int marker) +{ + int data_precision, num_components; + + data_precision = Data[2]; + m_exifinfo->Height = Get16m((void*)(Data+3)); + m_exifinfo->Width = Get16m((void*)(Data+5)); + num_components = Data[7]; + + if (num_components == 3){ + m_exifinfo->IsColor = 1; + }else{ + m_exifinfo->IsColor = 0; + } + + m_exifinfo->Process = marker; + + //if (ShowTags) printf("JPEG image is %uw * %uh, %d color components, %d bits per sample\n", + // ImageInfo.Width, ImageInfo.Height, num_components, data_precision); +} +//////////////////////////////////////////////////////////////////////////////// +/** + * this will work only on a CxImageJPG object, if the image originally has valid EXIF data + \verbatim + CxImageJPG jpg; + CxIOFile in,out; + in.Open("D:\\exif_in.jpg","rb"); + out.Open("D:\\exif_out.jpg","w+b"); + jpg.Decode(&in); + if (jpg.IsValid()){ + jpg.RotateLeft(); + jpg.Encode(&out); + } + \endverbatim +*/ +bool CxImageJPG::CxExifInfo::EncodeExif(CxFile * hFile) +{ + int a; + + if (FindSection(M_SOS)==NULL){ + strcpy(m_szLastError,"Can't write exif : didn't read all"); + return false; + } + + // Initial static jpeg marker. + hFile->PutC(0xff); + hFile->PutC(0xd8); + + if (Sections[0].Type != M_EXIF && Sections[0].Type != M_JFIF){ + // The image must start with an exif or jfif marker. If we threw those away, create one. + static BYTE JfifHead[18] = { + 0xff, M_JFIF, + 0x00, 0x10, 'J' , 'F' , 'I' , 'F' , 0x00, 0x01, + 0x01, 0x01, 0x01, 0x2C, 0x01, 0x2C, 0x00, 0x00 + }; + hFile->Write(JfifHead, 18, 1); + } + + // Write all the misc sections + for (a=0;aPutC(0xff); + hFile->PutC(Sections[a].Type); + hFile->Write(Sections[a].Data, Sections[a].Size, 1); + } + + // Write the remaining image data. + hFile->Write(Sections[a].Data, Sections[a].Size, 1); + + return true; +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageJPG::CxExifInfo::DiscardAllButExif() +{ + Section_t ExifKeeper; + Section_t CommentKeeper; + int a; + + memset(&ExifKeeper, 0, sizeof(ExifKeeper)); + memset(&CommentKeeper, 0, sizeof(ExifKeeper)); + + for (a=0;a Use it before Create() + */ +void CxImage::CopyInfo(const CxImage &src) +{ + if (pDib==NULL) memcpy(&info,&src.info,sizeof(CXIMAGEINFO)); +} +//////////////////////////////////////////////////////////////////////////////// +/** + * \sa Copy + */ +CxImage& CxImage::operator = (const CxImage& isrc) +{ + if (this != &isrc) Copy(isrc); + return *this; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Initializes or rebuilds the image. + * \param dwWidth: width + * \param dwHeight: height + * \param wBpp: bit per pixel, can be 1, 4, 8, 24 + * \param imagetype: (optional) set the image format, see ENUM_CXIMAGE_FORMATS + * \return pointer to the internal pDib object; NULL if an error occurs. + */ +void* CxImage::Create(DWORD dwWidth, DWORD dwHeight, DWORD wBpp, DWORD imagetype) +{ + // destroy the existing image (if any) + if (!Destroy()) + return NULL; + + // prevent further actions if width or height are not vaild + if ((dwWidth == 0) || (dwHeight == 0)){ + strcpy(info.szLastError,"CxImage::Create : width and height must be greater than zero"); + return NULL; + } + + // Make sure bits per pixel is valid + if (wBpp <= 1) wBpp = 1; + else if (wBpp <= 4) wBpp = 4; + else if (wBpp <= 8) wBpp = 8; + else wBpp = 24; + + // limit memory requirements (and also a check for bad parameters) + if (((dwWidth*dwHeight*wBpp)>>8) > CXIMAGE_MAX_MEMORY){ + strcpy(info.szLastError,"CXIMAGE_MAX_MEMORY exceeded"); + return NULL; + } + + // set the correct bpp value + switch (wBpp){ + case 1: + head.biClrUsed = 2; break; + case 4: + head.biClrUsed = 16; break; + case 8: + head.biClrUsed = 256; break; + default: + head.biClrUsed = 0; + } + + //set the common image informations + info.dwEffWidth = ((((wBpp * dwWidth) + 31) / 32) * 4); + info.dwType = imagetype; + + // initialize BITMAPINFOHEADER + head.biSize = sizeof(BITMAPINFOHEADER); // + head.biWidth = dwWidth; // fill in width from parameter + head.biHeight = dwHeight; // fill in height from parameter + head.biPlanes = 1; // must be 1 + head.biBitCount = (WORD)wBpp; // from parameter + head.biCompression = BI_RGB; + head.biSizeImage = info.dwEffWidth * dwHeight; +// head.biXPelsPerMeter = 0; See SetXDPI +// head.biYPelsPerMeter = 0; See SetYDPI + head.biClrImportant = 0; + + pDib = malloc(GetSize()); // alloc memory block to store our bitmap + if (!pDib){ + strcpy(info.szLastError,"CxImage::Create can't allocate memory"); + return NULL; + } + + //clear the palette + RGBQUAD* pal=GetPalette(); + if (pal) memset(pal,0,GetPaletteSize()); + //Destroy the existing selection +#if CXIMAGE_SUPPORT_SELECTION + if (pSelection) SelectionDelete(); +#endif //CXIMAGE_SUPPORT_SELECTION + //Destroy the existing alpha channel +#if CXIMAGE_SUPPORT_ALPHA + if (pAlpha) AlphaDelete(); +#endif //CXIMAGE_SUPPORT_ALPHA + + // use our bitmap info structure to fill in first part of + // our DIB with the BITMAPINFOHEADER + BITMAPINFOHEADER* lpbi; + lpbi = (BITMAPINFOHEADER*)(pDib); + *lpbi = head; + + info.pImage=GetBits(); + + return pDib; //return handle to the DIB +} +//////////////////////////////////////////////////////////////////////////////// +/** + * \return pointer to the image pixels. USE CAREFULLY + */ +BYTE* CxImage::GetBits(DWORD row) +{ + if (pDib){ + if (row) { + if (row<(DWORD)head.biHeight){ + return ((BYTE*)pDib + *(DWORD*)pDib + GetPaletteSize() + (info.dwEffWidth * row)); + } else { + return NULL; + } + } else { + return ((BYTE*)pDib + *(DWORD*)pDib + GetPaletteSize()); + } + } + return NULL; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * \return the size in bytes of the internal pDib object + */ +long CxImage::GetSize() +{ + return head.biSize + head.biSizeImage + GetPaletteSize(); +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Checks if the coordinates are inside the image + * \return true if x and y are both inside the image + */ +bool CxImage::IsInside(long x, long y) +{ + return (0<=y && y 0) bval = 255; + } + if (GetBpp() == 4){ + bval = (BYTE)(17*(0x0F & bval)); + } + + memset(info.pImage,bval,head.biSizeImage); +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Transfers the image from an existing source image. The source becomes empty. + * \return true if everything is ok + */ +bool CxImage::Transfer(CxImage &from) +{ + if (!Destroy()) + return false; + + memcpy(&head,&from.head,sizeof(BITMAPINFOHEADER)); + memcpy(&info,&from.info,sizeof(CXIMAGEINFO)); + + pDib = from.pDib; + pSelection = from.pSelection; + pAlpha = from.pAlpha; + pLayers = from.pLayers; + + memset(&from.head,0,sizeof(BITMAPINFOHEADER)); + memset(&from.info,0,sizeof(CXIMAGEINFO)); + from.pDib = from.pSelection = from.pAlpha = NULL; + from.pLayers = NULL; + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * (this) points to the same pDib owned by (*from), the image remains in (*from) + * but (this) has the access to the pixels. Use carefully !!! + */ +void CxImage::Ghost(CxImage *from) +{ + if (from){ + memcpy(&head,&from->head,sizeof(BITMAPINFOHEADER)); + memcpy(&info,&from->info,sizeof(CXIMAGEINFO)); + pDib = from->pDib; + pSelection = from->pSelection; + pAlpha = from->pAlpha; + pLayers = from->pLayers; + info.pGhost=from; + } +} +//////////////////////////////////////////////////////////////////////////////// +/** + * turns a 16 or 32 bit bitfield image into a RGB image + */ +void CxImage::Bitfield2RGB(BYTE *src, WORD redmask, WORD greenmask, WORD bluemask, BYTE bpp) +{ + switch (bpp){ + case 16: + { + DWORD ns[3]={0,0,0}; + // compute the number of shift for each mask + for (int i=0;i<16;i++){ + if ((redmask>>i)&0x01) ns[0]++; + if ((greenmask>>i)&0x01) ns[1]++; + if ((bluemask>>i)&0x01) ns[2]++; + } + ns[1]+=ns[0]; ns[2]+=ns[1]; ns[0]=8-ns[0]; ns[1]-=8; ns[2]-=8; + // dword aligned width for 16 bit image + long effwidth2=(((head.biWidth + 1) / 2) * 4); + WORD w; + long y2,y3,x2,x3; + BYTE *p=info.pImage; + // scan the buffer in reverse direction to avoid reallocations + for (long y=head.biHeight-1; y>=0; y--){ + y2=effwidth2*y; + y3=info.dwEffWidth*y; + for (long x=head.biWidth-1; x>=0; x--){ + x2 = 2*x+y2; + x3 = 3*x+y3; + w = (WORD)(src[x2]+256*src[1+x2]); + p[ x3]=(BYTE)((w & bluemask)<>ns[1]); + p[2+x3]=(BYTE)((w & redmask)>>ns[2]); + } + } + break; + } + case 32: + { + // dword aligned width for 32 bit image + long effwidth4 = head.biWidth * 4; + long y4,y3,x4,x3; + BYTE *p=info.pImage; + // scan the buffer in reverse direction to avoid reallocations + for (long y=head.biHeight-1; y>=0; y--){ + y4=effwidth4*y; + y3=info.dwEffWidth*y; + for (long x=head.biWidth-1; x>=0; x--){ + x4 = 4*x+y4; + x3 = 3*x+y3; + p[ x3]=src[ x4]; + p[1+x3]=src[1+x4]; + p[2+x3]=src[2+x4]; + } + } + } + + } + return; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Creates an image from a generic buffer + * \param pArray: source memory buffer + * \param dwWidth: image width + * \param dwHeight: image height + * \param dwBitsperpixel: can be 1,4,8,24,32 + * \param dwBytesperline: line alignment, in bytes, for a single row stored in pArray + * \param bFlipImage: tune this parameter if the image is upsidedown + * \return true if everything is ok + */ +bool CxImage::CreateFromArray(BYTE* pArray,DWORD dwWidth,DWORD dwHeight,DWORD dwBitsperpixel, DWORD dwBytesperline, bool bFlipImage) +{ + if (pArray==NULL) return false; + if (!((dwBitsperpixel==1)||(dwBitsperpixel==4)||(dwBitsperpixel==8)|| + (dwBitsperpixel==24)||(dwBitsperpixel==32))) return false; + + if (!Create(dwWidth,dwHeight,dwBitsperpixel)) return false; + + if (dwBitsperpixel<24) SetGrayPalette(); + +#if CXIMAGE_SUPPORT_ALPHA + if (dwBitsperpixel==32) AlphaCreate(); +#endif //CXIMAGE_SUPPORT_ALPHA + + BYTE *dst,*src; + + for (DWORD y = 0; yrgbRed,c1->rgbGreen,c1->rgbBlue); + int g2 = (int)RGB2GRAY(c2->rgbRed,c2->rgbGreen,c2->rgbBlue); + + return (g1-g2); +} +//////////////////////////////////////////////////////////////////////////////// +/** + * simply calls "if (memblock) free(memblock);". + * Useful when calling Encode for a memory buffer, + * from a DLL compiled with different memory management options. + * CxImage::FreeMemory will use the same memory environment used by Encode. + */ +void CxImage::FreeMemory(void* memblock) +{ + if (memblock) + free(memblock); +} +//////////////////////////////////////////////////////////////////////////////// +//EOF diff --git a/src/win32/dependencies/cximage/ximage.h b/src/win32/dependencies/cximage/ximage.h new file mode 100644 index 00000000..8d3af351 --- /dev/null +++ b/src/win32/dependencies/cximage/ximage.h @@ -0,0 +1,648 @@ +/* + * File: ximage.h + * Purpose: General Purpose Image Class + */ +/* + -------------------------------------------------------------------------------- + + COPYRIGHT NOTICE, DISCLAIMER, and LICENSE: + + CxImage version 5.99c 17/Oct/2004 + + CxImage : Copyright (C) 2001 - 2004, Davide Pizzolato + + Original CImage and CImageIterator implementation are: + Copyright (C) 1995, Alejandro Aguilar Sierra (asierra(at)servidor(dot)unam(dot)mx) + + Covered code is provided under this license on an "as is" basis, without warranty + of any kind, either expressed or implied, including, without limitation, warranties + that the covered code is free of defects, merchantable, fit for a particular purpose + or non-infringing. The entire risk as to the quality and performance of the covered + code is with you. Should any covered code prove defective in any respect, you (not + the initial developer or any other contributor) assume the cost of any necessary + servicing, repair or correction. This disclaimer of warranty constitutes an essential + part of this license. No use of any covered code is authorized hereunder except under + this disclaimer. + + Permission is hereby granted to use, copy, modify, and distribute this + source code, or portions hereof, for any purpose, including commercial applications, + freely and without fee, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not + claim that you wrote the original software. If you use this software + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original software. + + 3. This notice may not be removed or altered from any source distribution. + + -------------------------------------------------------------------------------- + + Other information: about CxImage, and the latest version, can be found at the + CxImage home page: http://www.xdp.it + + -------------------------------------------------------------------------------- + */ +#if !defined(__CXIMAGE_H) +#define __CXIMAGE_H + +#if _MSC_VER > 1000 +#pragma once +#endif + +///////////////////////////////////////////////////////////////////////////// +#include "xfile.h" +#include "xiofile.h" +#include "xmemfile.h" +#include "ximadef.h" // adjust some #define + +/* see "ximacfg.h" for CxImage configuration options */ + +///////////////////////////////////////////////////////////////////////////// +// CxImage formats enumerator +enum ENUM_CXIMAGE_FORMATS{ +CXIMAGE_FORMAT_UNKNOWN, +#if CXIMAGE_SUPPORT_BMP +CXIMAGE_FORMAT_BMP, +#endif +#if CXIMAGE_SUPPORT_GIF +CXIMAGE_FORMAT_GIF, +#endif +#if CXIMAGE_SUPPORT_JPG +CXIMAGE_FORMAT_JPG, +#endif +#if CXIMAGE_SUPPORT_PNG +CXIMAGE_FORMAT_PNG, +#endif +#if CXIMAGE_SUPPORT_MNG +CXIMAGE_FORMAT_MNG, +#endif +#if CXIMAGE_SUPPORT_ICO +CXIMAGE_FORMAT_ICO, +#endif +#if CXIMAGE_SUPPORT_TIF +CXIMAGE_FORMAT_TIF, +#endif +#if CXIMAGE_SUPPORT_TGA +CXIMAGE_FORMAT_TGA, +#endif +#if CXIMAGE_SUPPORT_PCX +CXIMAGE_FORMAT_PCX, +#endif +#if CXIMAGE_SUPPORT_WBMP +CXIMAGE_FORMAT_WBMP, +#endif +#if CXIMAGE_SUPPORT_WMF +CXIMAGE_FORMAT_WMF, +#endif +#if CXIMAGE_SUPPORT_J2K +CXIMAGE_FORMAT_J2K, +#endif +#if CXIMAGE_SUPPORT_JBG +CXIMAGE_FORMAT_JBG, +#endif +#if CXIMAGE_SUPPORT_JP2 +CXIMAGE_FORMAT_JP2, +#endif +#if CXIMAGE_SUPPORT_JPC +CXIMAGE_FORMAT_JPC, +#endif +#if CXIMAGE_SUPPORT_PGX +CXIMAGE_FORMAT_PGX, +#endif +#if CXIMAGE_SUPPORT_PNM +CXIMAGE_FORMAT_PNM, +#endif +#if CXIMAGE_SUPPORT_RAS +CXIMAGE_FORMAT_RAS, +#endif +CMAX_IMAGE_FORMATS +}; + +///////////////////////////////////////////////////////////////////////////// +// CxImage class +///////////////////////////////////////////////////////////////////////////// +class DLL_EXP CxImage +{ +//extensible information collector +typedef struct tagCxImageInfo { + DWORD dwEffWidth; ///< DWORD aligned scan line width + BYTE* pImage; ///< THE IMAGE BITS + CxImage* pGhost; ///< if this is a ghost, pGhost points to the body + CxImage* pParent; ///< if this is a layer, pParent points to the body + DWORD dwType; ///< original image format + char szLastError[256]; ///< debugging + long nProgress; ///< monitor + long nEscape; ///< escape + long nBkgndIndex; ///< used for GIF, PNG, MNG + RGBQUAD nBkgndColor; ///< used for RGB transparency + BYTE nQuality; ///< used for JPEG + BYTE nJpegScale; ///< used for JPEG [ignacio] + long nFrame; ///< used for TIF, GIF, MNG : actual frame + long nNumFrames; ///< used for TIF, GIF, MNG : total number of frames + DWORD dwFrameDelay; ///< used for GIF, MNG + long xDPI; ///< horizontal resolution + long yDPI; ///< vertical resolution + RECT rSelectionBox; ///< bounding rectangle + BYTE nAlphaMax; ///< max opacity (fade) + bool bAlphaPaletteEnabled; ///< true if alpha values in the palette are enabled. + bool bEnabled; ///< enables the painting functions + long xOffset; + long yOffset; + DWORD dwCodecOpt[CMAX_IMAGE_FORMATS]; ///< for GIF, TIF : 0=def.1=unc,2=fax3,3=fax4,4=pack,5=jpg + RGBQUAD last_c; ///< for GetNearestIndex optimization + BYTE last_c_index; + bool last_c_isvalid; + long nNumLayers; + DWORD dwFlags; ///< 0x??00000 = reserved, 0x00??0000 = blend mode, 0x0000???? = layer id - user flags + +} CXIMAGEINFO; + +public: + //public structures +struct rgb_color { BYTE r,g,b; }; + +#if CXIMAGE_SUPPORT_WINDOWS +// text placement data +// members must be initialized with the InitTextInfo(&this) function. +typedef struct tagCxTextInfo +{ + TCHAR text[4096]; ///< text (char -> TCHAR for UNICODE [Cesar M]) + LOGFONT lfont; ///< font and codepage data + COLORREF fcolor; ///< foreground color + long align; ///< DT_CENTER, DT_RIGHT, DT_LEFT aligment for multiline text + BYTE opaque; ///< text has background or hasn't. Default is true. + ///< data for background (ignored if .opaque==FALSE) + COLORREF bcolor; ///< background color + float b_opacity; ///< opacity value for background between 0.0-1.0 Default is 0. (opaque) + BYTE b_outline; ///< outline width for background (zero: no outline) + BYTE b_round; ///< rounding radius for background rectangle. % of the height, between 0-50. Default is 10. + ///< (backgr. always has a frame: width = 3 pixel + 10% of height by default.) +} CXTEXTINFO; +#endif + +public: +/** \addtogroup Constructors */ //@{ + CxImage(DWORD imagetype = 0); + CxImage(DWORD dwWidth, DWORD dwHeight, DWORD wBpp, DWORD imagetype = 0); + CxImage(const CxImage &src, bool copypixels = true, bool copyselection = true, bool copyalpha = true); + CxImage(const TCHAR * filename, DWORD imagetype); // For UNICODE support: char -> TCHAR + CxImage(FILE * stream, DWORD imagetype); + CxImage(CxFile * stream, DWORD imagetype); + CxImage(BYTE * buffer, DWORD size, DWORD imagetype); + virtual ~CxImage() { Destroy(); }; + CxImage& operator = (const CxImage&); +//@} + +/** \addtogroup Initialization */ //@{ + void* Create(DWORD dwWidth, DWORD dwHeight, DWORD wBpp, DWORD imagetype = 0); + bool Destroy(); + void Clear(BYTE bval=0); + void Copy(const CxImage &src, bool copypixels = true, bool copyselection = true, bool copyalpha = true); + bool Transfer(CxImage &from); + bool CreateFromArray(BYTE* pArray,DWORD dwWidth,DWORD dwHeight,DWORD dwBitsperpixel, DWORD dwBytesperline, bool bFlipImage); + bool CreateFromMatrix(BYTE** ppMatrix,DWORD dwWidth,DWORD dwHeight,DWORD dwBitsperpixel, DWORD dwBytesperline, bool bFlipImage); + void FreeMemory(void* memblock); +//@} + +/** \addtogroup Attributes */ //@{ + long GetSize(); + BYTE* GetBits(DWORD row = 0); + BYTE GetColorType(); + void* GetDIB() const; + DWORD GetHeight() const; + DWORD GetWidth() const; + DWORD GetEffWidth() const; + DWORD GetNumColors() const; + WORD GetBpp() const; + DWORD GetType() const; + const char* GetLastError(); + const TCHAR* GetVersion(); + const float GetVersionNumber(); + + DWORD GetFrameDelay() const; + void SetFrameDelay(DWORD d); + + void GetOffset(long *x,long *y); + void SetOffset(long x,long y); + + BYTE GetJpegQuality() const; + void SetJpegQuality(BYTE q); + + BYTE GetJpegScale() const; + void SetJpegScale(BYTE q); + + long GetXDPI() const; + long GetYDPI() const; + void SetXDPI(long dpi); + void SetYDPI(long dpi); + + DWORD GetClrImportant() const; + void SetClrImportant(DWORD ncolors = 0); + + long GetProgress() const; + long GetEscape() const; + void SetProgress(long p); + void SetEscape(long i); + + long GetTransIndex() const; + RGBQUAD GetTransColor(); + void SetTransIndex(long idx); + void SetTransColor(RGBQUAD rgb); + bool IsTransparent() const; + + DWORD GetCodecOption(DWORD imagetype = 0); + bool SetCodecOption(DWORD opt, DWORD imagetype = 0); + + DWORD GetFlags() const; + void SetFlags(DWORD flags, bool bLockReservedFlags = true); + + //void* GetUserData() const {return info.pUserData;} + //void SetUserData(void* pUserData) {info.pUserData = pUserData;} +//@} + +/** \addtogroup Palette + * These functions have no effects on RGB images and in this case the returned value is always 0. + * @{ */ + bool IsGrayScale(); + bool IsIndexed() const; + bool IsSamePalette(CxImage &img, bool bCheckAlpha = true); + DWORD GetPaletteSize(); + RGBQUAD* GetPalette() const; + RGBQUAD GetPaletteColor(BYTE idx); + bool GetPaletteColor(BYTE i, BYTE* r, BYTE* g, BYTE* b); + BYTE GetNearestIndex(RGBQUAD c); + void BlendPalette(COLORREF cr,long perc); + void SetGrayPalette(); + void SetPalette(DWORD n, BYTE *r, BYTE *g, BYTE *b); + void SetPalette(RGBQUAD* pPal,DWORD nColors=256); + void SetPalette(rgb_color *rgb,DWORD nColors=256); + void SetPaletteColor(BYTE idx, BYTE r, BYTE g, BYTE b, BYTE alpha=0); + void SetPaletteColor(BYTE idx, RGBQUAD c); + void SetPaletteColor(BYTE idx, COLORREF cr); + void SwapIndex(BYTE idx1, BYTE idx2); + void SetStdPalette(); +//@} + +/** \addtogroup Pixel */ //@{ + bool IsInside(long x, long y); + bool IsTransparent(long x,long y); + RGBQUAD GetPixelColor(long x,long y, bool bGetAlpha = true); + BYTE GetPixelIndex(long x,long y); + BYTE GetPixelGray(long x, long y); + void SetPixelColor(long x,long y,RGBQUAD c, bool bSetAlpha = false); + void SetPixelColor(long x,long y,COLORREF cr); + void SetPixelIndex(long x,long y,BYTE i); + void DrawLine(int StartX, int EndX, int StartY, int EndY, RGBQUAD color, bool bSetAlpha=false); + void DrawLine(int StartX, int EndX, int StartY, int EndY, COLORREF cr); + void BlendPixelColor(long x,long y,RGBQUAD c, float blend, bool bSetAlpha = false); +//@} + +protected: +/** \addtogroup Protected */ //@{ + BYTE BlindGetPixelIndex(const long x,const long y); + RGBQUAD BlindGetPixelColor(const long x,const long y); + void *BlindGetPixelPointer(const long x,const long y); +//@} + +public: + +#if CXIMAGE_SUPPORT_INTERPOLATION +/** \addtogroup Interpolation */ //@{ + //overflow methods: + enum OverflowMethod { + OM_COLOR=1, + OM_BACKGROUND=2, + OM_TRANSPARENT=3, + OM_WRAP=4, + OM_REPEAT=5, + OM_MIRROR=6 + }; + void OverflowCoordinates(float &x, float &y, OverflowMethod const ofMethod); + void OverflowCoordinates(long &x, long &y, OverflowMethod const ofMethod); + RGBQUAD GetPixelColorWithOverflow(long x, long y, OverflowMethod const ofMethod=OM_BACKGROUND, RGBQUAD* const rplColor=0); + //interpolation methods: + enum InterpolationMethod { + IM_NEAREST_NEIGHBOUR=1, + IM_BILINEAR =2, + IM_BSPLINE =3, + IM_BICUBIC =4, + IM_BICUBIC2 =5, + IM_LANCZOS =6, + IM_BOX =7, + IM_HERMITE =8, + IM_HAMMING =9, + IM_SINC =10, + IM_BLACKMAN =11, + IM_BESSEL =12, + IM_GAUSSIAN =13, + IM_QUADRATIC =14, + IM_MITCHELL =15, + IM_CATROM =16 + }; + RGBQUAD GetPixelColorInterpolated(float x,float y, InterpolationMethod const inMethod=IM_BILINEAR, OverflowMethod const ofMethod=OM_BACKGROUND, RGBQUAD* const rplColor=0); + RGBQUAD GetAreaColorInterpolated(float const xc, float const yc, float const w, float const h, InterpolationMethod const inMethod, OverflowMethod const ofMethod=OM_BACKGROUND, RGBQUAD* const rplColor=0); +//@} + +protected: +/** \addtogroup Protected */ //@{ + void AddAveragingCont(RGBQUAD const &color, float const surf, float &rr, float &gg, float &bb, float &aa); +//@} + +/** \addtogroup Kernels */ //@{ +public: + static float KernelBSpline(const float x); + static float KernelLinear(const float t); + static float KernelCubic(const float t); + static float KernelGeneralizedCubic(const float t, const float a=-1); + static float KernelLanczosSinc(const float t, const float r = 3); + static float KernelBox(const float x); + static float KernelHermite(const float x); + static float KernelHamming(const float x); + static float KernelSinc(const float x); + static float KernelBlackman(const float x); + static float KernelBessel_J1(const float x); + static float KernelBessel_P1(const float x); + static float KernelBessel_Q1(const float x); + static float KernelBessel_Order1(float x); + static float KernelBessel(const float x); + static float KernelGaussian(const float x); + static float KernelQuadratic(const float x); + static float KernelMitchell(const float x); + static float KernelCatrom(const float x); +//@} +#endif //CXIMAGE_SUPPORT_INTERPOLATION + +/** \addtogroup Painting */ //@{ +#if CXIMAGE_SUPPORT_WINCE + long Blt(HDC pDC, long x=0, long y=0); +#endif +#if CXIMAGE_SUPPORT_WINDOWS + HBITMAP MakeBitmap(HDC hdc = NULL); + HANDLE CopyToHandle(); + bool CreateFromHANDLE(HANDLE hMem); //Windows objects (clipboard) + bool CreateFromHBITMAP(HBITMAP hbmp, HPALETTE hpal=0); //Windows resource + bool CreateFromHICON(HICON hico); + long Draw(HDC hdc, long x=0, long y=0, long cx = -1, long cy = -1, RECT* pClipRect = 0, bool bSmooth = false); + long Draw(HDC hdc, const RECT& rect, RECT* pClipRect=NULL, bool bSmooth = false); + long Stretch(HDC hdc, long xoffset, long yoffset, long xsize, long ysize, DWORD dwRop = SRCCOPY); + long Stretch(HDC hdc, const RECT& rect, DWORD dwRop = SRCCOPY); + long Tile(HDC hdc, RECT *rc); + long Draw2(HDC hdc, long x=0, long y=0, long cx = -1, long cy = -1); + long Draw2(HDC hdc, const RECT& rect); + //long DrawString(HDC hdc, long x, long y, const char* text, RGBQUAD color, const char* font, long lSize=0, long lWeight=400, BYTE bItalic=0, BYTE bUnderline=0, bool bSetAlpha=false); + long DrawString(HDC hdc, long x, long y, const TCHAR* text, RGBQUAD color, const TCHAR* font, long lSize=0, long lWeight=400, BYTE bItalic=0, BYTE bUnderline=0, bool bSetAlpha=false); + // extensions + long DrawStringEx(HDC hdc, long x, long y, CXTEXTINFO *pTextType, bool bSetAlpha=false ); + void InitTextInfo( CXTEXTINFO *txt ); +#endif //CXIMAGE_SUPPORT_WINDOWS +//@} + + // file operations +#if CXIMAGE_SUPPORT_DECODE +/** \addtogroup Decode */ //@{ +#ifdef WIN32 + //bool Load(LPCWSTR filename, DWORD imagetype=0); + bool LoadResource(HRSRC hRes, DWORD imagetype, HMODULE hModule=NULL); +#endif + // For UNICODE support: char -> TCHAR + bool Load(const TCHAR* filename, DWORD imagetype=0); + //bool Load(const char * filename, DWORD imagetype=0); + bool Decode(FILE * hFile, DWORD imagetype); + bool Decode(CxFile * hFile, DWORD imagetype); + bool Decode(BYTE * buffer, DWORD size, DWORD imagetype); +//@} +#endif //CXIMAGE_SUPPORT_DECODE + +#if CXIMAGE_SUPPORT_ENCODE +protected: +/** \addtogroup Protected */ //@{ + bool EncodeSafeCheck(CxFile *hFile); +//@} + +public: +/** \addtogroup Encode */ //@{ +#ifdef WIN32 + //bool Save(LPCWSTR filename, DWORD imagetype=0); +#endif + // For UNICODE support: char -> TCHAR + bool Save(const TCHAR* filename, DWORD imagetype); + //bool Save(const char * filename, DWORD imagetype=0); + bool Encode(FILE * hFile, DWORD imagetype); + bool Encode(CxFile * hFile, DWORD imagetype); + bool Encode(CxFile * hFile, CxImage ** pImages, int pagecount, DWORD imagetype); + bool Encode(FILE *hFile, CxImage ** pImages, int pagecount, DWORD imagetype); + bool Encode(BYTE * &buffer, long &size, DWORD imagetype); + + bool Encode2RGBA(CxFile *hFile); + bool Encode2RGBA(BYTE * &buffer, long &size); +//@} +#endif //CXIMAGE_SUPPORT_ENCODE + +/** \addtogroup Attributes */ //@{ + //misc. + bool IsValid() const; + bool IsEnabled() const; + void Enable(bool enable=true); + + // frame operations + long GetNumFrames() const; + long GetFrame() const; + void SetFrame(long nFrame); +//@} + +#if CXIMAGE_SUPPORT_BASICTRANSFORMATIONS +/** \addtogroup BasicTransformations */ //@{ + bool GrayScale(); + bool Flip(); + bool Mirror(); + bool Negative(); + bool RotateLeft(CxImage* iDst = NULL); + bool RotateRight(CxImage* iDst = NULL); +//@} +#endif //CXIMAGE_SUPPORT_BASICTRANSFORMATIONS + +#if CXIMAGE_SUPPORT_TRANSFORMATION +/** \addtogroup Transformations */ //@{ + // image operations + bool Rotate(float angle, CxImage* iDst = NULL); + bool Rotate2(float angle, CxImage *iDst = NULL, InterpolationMethod inMethod=IM_BILINEAR, + OverflowMethod ofMethod=OM_BACKGROUND, RGBQUAD *replColor=0, + bool const optimizeRightAngles=true, bool const bKeepOriginalSize=false); + bool Rotate180(CxImage* iDst = NULL); + bool Resample(long newx, long newy, int mode = 1, CxImage* iDst = NULL); + bool Resample2(long newx, long newy, InterpolationMethod const inMethod=IM_BICUBIC2, + OverflowMethod const ofMethod=OM_REPEAT, CxImage* const iDst = NULL, + bool const disableAveraging=false); + bool DecreaseBpp(DWORD nbit, bool errordiffusion, RGBQUAD* ppal = 0, DWORD clrimportant = 0); + bool IncreaseBpp(DWORD nbit); + bool Dither(long method = 0); + bool Crop(long left, long top, long right, long bottom, CxImage* iDst = NULL); + bool Crop(const RECT& rect, CxImage* iDst = NULL); + bool CropRotatedRectangle( long topx, long topy, long width, long height, float angle, CxImage* iDst = NULL); + bool Skew(float xgain, float ygain, long xpivot=0, long ypivot=0, bool bEnableInterpolation = false); + bool Expand(long left, long top, long right, long bottom, RGBQUAD canvascolor, CxImage* iDst = 0); + bool Expand(long newx, long newy, RGBQUAD canvascolor, CxImage* iDst = 0); + bool Thumbnail(long newx, long newy, RGBQUAD canvascolor, CxImage* iDst = 0); + bool CircleTransform(int type,long rmax=0,float Koeff=1.0f); + bool RedEyeRemove(); + bool QIShrink(long newx, long newy, CxImage* const iDst = NULL); +//@} +#endif //CXIMAGE_SUPPORT_TRANSFORMATION + +#if CXIMAGE_SUPPORT_DSP +/** \addtogroup DSP */ //@{ + bool Contour(); + bool HistogramStretch(long method = 0); + bool HistogramEqualize(); + bool HistogramNormalize(); + bool HistogramRoot(); + bool HistogramLog(); + long Histogram(long* red, long* green = 0, long* blue = 0, long* gray = 0, long colorspace = 0); + bool Jitter(long radius=2); + bool Repair(float radius = 0.25f, long niterations = 1, long colorspace = 0); + bool Combine(CxImage* r,CxImage* g,CxImage* b,CxImage* a, long colorspace = 0); + bool FFT2(CxImage* srcReal, CxImage* srcImag, CxImage* dstReal, CxImage* dstImag, long direction = 1, bool bForceFFT = true, bool bMagnitude = true); + bool Noise(long level); + bool Median(long Ksize=3); + bool Gamma(float gamma); + bool ShiftRGB(long r, long g, long b); + bool Threshold(BYTE level); + bool Colorize(BYTE hue, BYTE sat, float blend = 1.0f); + bool Light(long brightness, long contrast = 0); + float Mean(); + bool Filter(long* kernel, long Ksize, long Kfactor, long Koffset); + bool Erode(long Ksize=2); + bool Dilate(long Ksize=2); + bool Edge(long Ksize=2); + void HuePalette(float correction=1); + enum ImageOpType { OpAdd, OpAnd, OpXor, OpOr, OpMask, OpSrcCopy, OpDstCopy, OpSub, OpSrcBlend, OpScreen }; + void Mix(CxImage & imgsrc2, ImageOpType op, long lXOffset = 0, long lYOffset = 0, bool bMixAlpha = false); + void MixFrom(CxImage & imagesrc2, long lXOffset, long lYOffset); + bool UnsharpMask(float radius = 5.0, float amount = 0.5, int threshold = 0); + bool Lut(BYTE* pLut); + bool Lut(BYTE* pLutR, BYTE* pLutG, BYTE* pLutB, BYTE* pLutA = 0); +//@} + +protected: +/** \addtogroup Protected */ //@{ + bool IsPowerof2(long x); + bool FFT(int dir,int m,double *x,double *y); + bool DFT(int dir,long m,double *x1,double *y1,double *x2,double *y2); + bool RepairChannel(CxImage *ch, float radius); + // + int gen_convolve_matrix (float radius, float **cmatrix_p); + float* gen_lookup_table (float *cmatrix, int cmatrix_length); + void blur_line (float *ctable, float *cmatrix, int cmatrix_length, BYTE* cur_col, BYTE* dest_col, int y, long bytes); +//@} + +public: +/** \addtogroup ColorSpace */ //@{ + bool SplitRGB(CxImage* r,CxImage* g,CxImage* b); + bool SplitYUV(CxImage* y,CxImage* u,CxImage* v); + bool SplitHSL(CxImage* h,CxImage* s,CxImage* l); + bool SplitYIQ(CxImage* y,CxImage* i,CxImage* q); + bool SplitXYZ(CxImage* x,CxImage* y,CxImage* z); + bool SplitCMYK(CxImage* c,CxImage* m,CxImage* y,CxImage* k); + static RGBQUAD HSLtoRGB(COLORREF cHSLColor); + static RGBQUAD RGBtoHSL(RGBQUAD lRGBColor); + static RGBQUAD HSLtoRGB(RGBQUAD lHSLColor); + static RGBQUAD YUVtoRGB(RGBQUAD lYUVColor); + static RGBQUAD RGBtoYUV(RGBQUAD lRGBColor); + static RGBQUAD YIQtoRGB(RGBQUAD lYIQColor); + static RGBQUAD RGBtoYIQ(RGBQUAD lRGBColor); + static RGBQUAD XYZtoRGB(RGBQUAD lXYZColor); + static RGBQUAD RGBtoXYZ(RGBQUAD lRGBColor); +#endif //CXIMAGE_SUPPORT_DSP + static RGBQUAD RGBtoRGBQUAD(COLORREF cr); + static COLORREF RGBQUADtoRGB (RGBQUAD c); +//@} + +#if CXIMAGE_SUPPORT_SELECTION +/** \addtogroup Selection */ //@{ + bool SelectionClear(); + bool SelectionCreate(); + bool SelectionDelete(); + bool SelectionInvert(); + bool SelectionAddRect(RECT r); + bool SelectionAddEllipse(RECT r); + bool SelectionAddPolygon(POINT *points, long npoints); + bool SelectionAddColor(RGBQUAD c); + bool SelectionAddPixel(int x, int y); + bool SelectionCopy(CxImage &from); + bool SelectionIsInside(long x, long y); + bool SelectionIsValid(); + void SelectionGetBox(RECT& r); + bool SelectionToHRGN(HRGN& region); + bool SelectionSplit(CxImage *dest); +//@} +#endif //CXIMAGE_SUPPORT_SELECTION + +#if CXIMAGE_SUPPORT_ALPHA +/** \addtogroup Alpha */ //@{ + void AlphaClear(); + void AlphaCreate(); + void AlphaDelete(); + void AlphaInvert(); + bool AlphaMirror(); + bool AlphaFlip(); + bool AlphaCopy(CxImage &from); + bool AlphaSplit(CxImage *dest); + void AlphaStrip(); + void AlphaSet(BYTE level); + bool AlphaSet(CxImage &from); + void AlphaSet(const long x,const long y,const BYTE level); + BYTE AlphaGet(const long x,const long y); + BYTE AlphaGetMax() const; + void AlphaSetMax(BYTE nAlphaMax); + bool AlphaIsValid(); + BYTE* AlphaGetPointer(const long x = 0,const long y = 0); + + void AlphaPaletteClear(); + void AlphaPaletteEnable(bool enable=true); + bool AlphaPaletteIsEnabled(); + bool AlphaPaletteIsValid(); + bool AlphaPaletteSplit(CxImage *dest); +//@} + +protected: +/** \addtogroup Protected */ //@{ + BYTE BlindAlphaGet(const long x,const long y); +//@} +#endif //CXIMAGE_SUPPORT_ALPHA + +public: +#if CXIMAGE_SUPPORT_LAYERS +/** \addtogroup Layers */ //@{ + bool LayerCreate(long position = -1); + bool LayerDelete(long position = -1); + void LayerDeleteAll(); + CxImage* GetLayer(long position); + CxImage* GetParent() const; + long GetNumLayers() const; +//@} +#endif //CXIMAGE_SUPPORT_LAYERS + +protected: +/** \addtogroup Protected */ //@{ + void Startup(DWORD imagetype = 0); + void CopyInfo(const CxImage &src); + void Ghost(CxImage *src); + void RGBtoBGR(BYTE *buffer, int length); + static float HueToRGB(float n1,float n2, float hue); + void Bitfield2RGB(BYTE *src, WORD redmask, WORD greenmask, WORD bluemask, BYTE bpp); + static int CompareColors(const void *elem1, const void *elem2); + + void* pDib; //contains the header, the palette, the pixels + BITMAPINFOHEADER head; //standard header + CXIMAGEINFO info; //extended information + BYTE* pSelection; //selected region + BYTE* pAlpha; //alpha channel + CxImage** pLayers; //generic layers +//@} +}; + +//////////////////////////////////////////////////////////////////////////// +#endif // !defined(__CXIMAGE_H) diff --git a/src/win32/dependencies/cximage/ximagif.cpp b/src/win32/dependencies/cximage/ximagif.cpp new file mode 100644 index 00000000..e9aee917 --- /dev/null +++ b/src/win32/dependencies/cximage/ximagif.cpp @@ -0,0 +1,1564 @@ +/* + * File: ximagif.cpp + * Purpose: Platform Independent GIF Image Class Loader and Writer + * 07/Aug/2001 Davide Pizzolato - www.xdp.it + * CxImage version 5.99c 17/Oct/2004 + */ + +#include "ximagif.h" + +#if CXIMAGE_SUPPORT_GIF + +#include "ximaiter.h" + +#if CXIMAGE_SUPPORT_WINCE + #define assert(s) +#else + #include +#endif + +//////////////////////////////////////////////////////////////////////////////// +bool CxImageGIF::Decode(CxFile *fp) +{ + /* AD - for transparency */ + struct_dscgif dscgif; + struct_image image; + struct_TabCol TabCol; + + if (fp == NULL) return false; + + fp->Read(&dscgif,/*sizeof(dscgif)*/13,1); + //if (strncmp(dscgif.header,"GIF8",3)!=0) { + if (strncmp(dscgif.header,"GIF8",4)!=0) return FALSE; + + if (info.nEscape == -1) { + // Return output dimensions only + head.biWidth = dscgif.scrwidth; + head.biHeight = dscgif.scrheight; + return true; + } + + /* AD - for interlace */ + TabCol.sogct = (short)(1 << ((dscgif.pflds & 0x07)+1)); + TabCol.colres = (short)(((dscgif.pflds & 0x70) >> 3) + 1); + + // assume that the image is a truecolor-gif if + // 1) no global color map found + // 2) (image.w, image.h) of the 1st image != (dscgif.scrwidth, dscgif.scrheight) + long bTrueColor=0; + CxImage* imaRGB=NULL; + + // Global colour map? + if (dscgif.pflds & 0x80) + fp->Read(TabCol.paleta,sizeof(struct rgb_color)*TabCol.sogct,1); + else + bTrueColor++; //first chance for a truecolor gif + + long first_transparent_index; + + int iImage = 0; + info.nNumFrames=get_num_frames(fp,&TabCol,&dscgif); + + if ((info.nFrame<0)||(info.nFrame>=info.nNumFrames)) return false; + + //it cannot be a true color GIF with only one frame + if (info.nNumFrames == 1) + bTrueColor=0; + + char ch; + bool bPreviousWasNull = true; + int prevdispmeth = 0; + + for (BOOL bContinue = TRUE; bContinue; ) + { + if (fp->Read(&ch, sizeof(ch), 1) != 1) {break;} + + if (info.nEscape > 0) return false; // - cancel decoding + if (bPreviousWasNull || ch==0) + { + switch (ch) + { + case '!': // extension + { + bContinue = DecodeExtension(fp); + break; + } + case ',': // image + { + assert(sizeof(image) == 9); + fp->Read(&image,sizeof(image),1); + //avoid byte order problems with Solaris + BYTE *byteData = (BYTE *) & image; + image.l = byteData[0]+byteData[1]*256; + image.t = byteData[2]+byteData[3]*256; + image.w = byteData[4]+byteData[5]*256; + image.h = byteData[6]+byteData[7]*256; + + if (((image.l + image.w) > dscgif.scrwidth)||((image.t + image.h) > dscgif.scrheight)) + break; + + // check if it could be a truecolor gif + if ((iImage==0) && (image.w != dscgif.scrwidth) && (image.h != dscgif.scrheight)) + bTrueColor++; + + // Local colour map? + if (image.pf & 0x80) { + TabCol.sogct = (short)(1 << ((image.pf & 0x07) +1)); + assert(3 == sizeof(struct rgb_color)); + fp->Read(TabCol.paleta,sizeof(struct rgb_color)*TabCol.sogct,1); + //log << "Local colour map" << endl; + } + + int bpp; // select the correct bit per pixel value + if (TabCol.sogct <= 2) bpp = 1; + else if (TabCol.sogct <= 16) bpp = 4; + else bpp = 8; + + CxImageGIF backimage; + backimage.CopyInfo(*this); + if (iImage==0){ + //first frame: build image background + backimage.Create(dscgif.scrwidth, dscgif.scrheight, bpp, CXIMAGE_FORMAT_GIF); + first_transparent_index = info.nBkgndIndex; + backimage.Clear((BYTE)gifgce.transpcolindex); + } else { + //generic frame: handle disposal method from previous one + /*Values : 0 - No disposal specified. The decoder is + not required to take any action. + 1 - Do not dispose. The graphic is to be left + in place. + 2 - Restore to background color. The area used by the + graphic must be restored to the background color. + 3 - Restore to previous. The decoder is required to + restore the area overwritten by the graphic with + what was there prior to rendering the graphic. + */ + backimage.Copy(*this); + if (prevdispmeth==2){ + backimage.Clear((BYTE)first_transparent_index); + } + } + + //active frame + Create(image.w, image.h, bpp, CXIMAGE_FORMAT_GIF); + + if ((image.pf & 0x80) || (dscgif.pflds & 0x80)) { + unsigned char r[256], g[256], b[256]; + int i, has_white = 0; + + for (i=0; i < TabCol.sogct; i++) { + r[i] = TabCol.paleta[i].r; + g[i] = TabCol.paleta[i].g; + b[i] = TabCol.paleta[i].b; + + if (RGB(r[i],g[i],b[i]) == 0xFFFFFF) has_white = 1; + } + + // Force transparency colour white... + //if (0) if (info.nBkgndIndex != -1) + // r[info.nBkgndIndex] = g[info.nBkgndIndex] = b[info.nBkgndIndex] = 255; + // Fill in with white // AD + if (info.nBkgndIndex != -1) { + while (i < 256) { + has_white = 1; + r[i] = g[i] = b[i] = 255; + i++; + } + } + + // Force last colour to white... // AD + //if ((info.nBkgndIndex != -1) && !has_white) { + // r[255] = g[255] = b[255] = 255; + //} + + SetPalette((info.nBkgndIndex != -1 ? 256 : TabCol.sogct), r, g, b); + } + + CImageIterator* iter = new CImageIterator(this); + iter->Upset(); + int badcode=0; + ibf = GIFBUFTAM+1; + + interlaced = image.pf & 0x40; + iheight = image.h; + istep = 8; + iypos = 0; + ipass = 0; + + long pos_start = fp->Tell(); + //if (interlaced) log << "Interlaced" << endl; + decoder(fp, iter, image.w, badcode); + delete iter; + + if (info.nEscape) return false; // - cancel decoding + + if (bTrueColor<2 ){ //standard GIF: mix frame with background + backimage.GifMix(*this,image); + backimage.SetTransIndex(first_transparent_index); + backimage.SetPalette(GetPalette()); + Transfer(backimage); + } else { //it's a truecolor gif! + //force full image decoding + info.nFrame=info.nNumFrames-1; + //build the RGB image + if (imaRGB==NULL) imaRGB = new CxImage(dscgif.scrwidth,dscgif.scrheight,24,CXIMAGE_FORMAT_GIF); + //copy the partial image into the full RGB image + for(long y=0;ySetPixelColor(x+image.l,dscgif.scrheight-1-image.t-y,GetPixelColor(x,image.h-y-1)); + } + } + } + + prevdispmeth = gifgce.dispmeth; + + //restore the correct position in the file for the next image + if (badcode){ + seek_next_image(fp,pos_start); + } else { + fp->Seek(-(ibfmax - ibf - 1), SEEK_CUR); + } + + if (info.nFrame==iImage) bContinue=false; else iImage++; + + break; + } + case ';': //terminator + bContinue=false; + break; + default: + bPreviousWasNull = (ch==0); + break; + } + } + } + + if (bTrueColor>=2 && imaRGB){ + if (gifgce.transpcolflag){ + imaRGB->SetTransColor(GetPaletteColor((BYTE)info.nBkgndIndex)); + imaRGB->SetTransIndex(0); + } + Transfer(*imaRGB); + } + delete imaRGB; + + return true; + +} +//////////////////////////////////////////////////////////////////////////////// +bool CxImageGIF::DecodeExtension(CxFile *fp) +{ + bool bContinue; + unsigned char count; + unsigned char fc; + + bContinue = (1 == fp->Read(&fc, sizeof(fc), 1)); + if (bContinue) { + /* AD - for transparency */ + if (fc == 0xF9) { + bContinue = (1 == fp->Read(&count, sizeof(count), 1)); + if (bContinue) { + assert(sizeof(gifgce) == 4); + bContinue = (count == fp->Read(&gifgce, 1, sizeof(gifgce))); + if (bContinue) { + if (gifgce.transpcolflag) info.nBkgndIndex = gifgce.transpcolindex; + info.dwFrameDelay = gifgce.delaytime; + m_dispmeth = gifgce.dispmeth; + } } } + + if (fc == 0xFE) { // Comment block + bContinue = (1 == fp->Read(&count, sizeof(count), 1)); + if (bContinue) { + bContinue = (1 == fp->Read(m_comment, count, 1)); + m_comment[count]='\0'; + } } + + if (fc == 0xFF) { // Application Extension block + bContinue = (1 == fp->Read(&count, sizeof(count), 1)); + if (bContinue) { + bContinue = (count==11); + if (bContinue){ + char AppID[11]; + bContinue = (1 == fp->Read(AppID, count, 1)); + if (bContinue) { + bContinue = (1 == fp->Read(&count, sizeof(count), 1)); + if (bContinue) { + BYTE* dati = (BYTE*)malloc(count); + bContinue = (dati!=NULL); + if (bContinue){ + bContinue = (1 == fp->Read(dati, count, 1)); + if (count>2){ + m_loops = dati[1]+256*dati[2]; + } + } + free(dati); + } } } } } + + while (bContinue && fp->Read(&count, sizeof(count), 1) && count) { + //log << "Skipping " << count << " bytes" << endl; + fp->Seek(count, SEEK_CUR); + } + } + return bContinue; + +} + + +// - This external (machine specific) function is expected to return +// either the next BYTE from the GIF file, or a negative error number. +int CxImageGIF::get_byte(CxFile* file) +{ + if (ibf>=GIFBUFTAM){ + // FW 06/02/98 >>> + ibfmax = file->Read( buf , 1 , GIFBUFTAM) ; + if( ibfmax < GIFBUFTAM ) buf[ ibfmax ] = 255 ; + // FW 06/02/98 <<< + ibf = 0; + } + if (ibf>=ibfmax) return -1; // avoid overflows + return buf[ibf++]; +} +//////////////////////////////////////////////////////////////////////////////// +/* - This function takes a full line of pixels (one BYTE per pixel) and + * displays them (or does whatever your program wants with them...). It + * should return zero, or negative if an error or some other event occurs + * which would require aborting the decode process... Note that the length + * passed will almost always be equal to the line length passed to the + * decoder function, with the sole exception occurring when an ending code + * occurs in an odd place in the GIF file... In any case, linelen will be + * equal to the number of pixels passed... +*/ +int CxImageGIF::out_line(CImageIterator* iter, unsigned char *pixels, int linelen) +{ + // for 1 & 4 bpp images, the pixels are compressed + if (head.biBitCount < 8){ + for(long x=0;x> 3); + if (head.biBitCount==4){ + pos = (BYTE)(4*(1-x%2)); + *iDst &= ~(0x0F<SetY(iheight-iypos-1); + iter->SetRow(pixels, linelen); + + if ((iypos += istep) >= iheight) { + do { + if (ipass++ > 0) istep /= 2; + iypos = istep / 2; + } + while (iypos > iheight); + } + return 0; + } else { + if (iter->ItOK()) { + iter->SetRow(pixels, linelen); + (void)iter->PrevRow(); + return 0; + } else { + // puts("chafeo"); + return -1; + } + } +} +//////////////////////////////////////////////////////////////////////////////// +#if CXIMAGE_SUPPORT_ENCODE +//////////////////////////////////////////////////////////////////////////////// +// SaveFile - writes GIF87a gif file +// Randy Spann 6/15/97 +// R.Spann@ConnRiver.net +bool CxImageGIF::Encode(CxFile * fp) +{ + if (EncodeSafeCheck(fp)) return false; + + if(head.biBitCount > 8) { + //strcpy(info.szLastError,"GIF Images must be 8 bit or less"); + //return FALSE; + return EncodeRGB(fp); + } + + EncodeHeader(fp); + + EncodeExtension(fp); + + EncodeComment(fp); + + EncodeBody(fp); + + fp->PutC(';'); // Write the GIF file terminator + + return true; // done! +} +//////////////////////////////////////////////////////////////////////////////// +bool CxImageGIF::Encode(CxFile * fp, CxImage ** pImages, int pagecount, bool bLocalColorMap) +{ + try{ + if (fp==NULL) throw "invalid file pointer"; + if (pImages==NULL || pagecount==0 || pImages[0]==NULL) throw "multipage GIF, no images!"; + + CxImageGIF ghost; + + //write the first image + ghost.Ghost(pImages[0]); + ghost.EncodeHeader(fp); + + if (m_loops!=1){ + ghost.SetLoops(max(0,m_loops-1)); + ghost.EncodeLoopExtension(fp); + } + + ghost.SetDisposalMethod(GetDisposalMethod()); + ghost.EncodeExtension(fp); + + EncodeComment(fp); + + ghost.EncodeBody(fp); + + for (int i=2; i<=pagecount; i++){ + if (pImages[i-1]==NULL) throw "Bad image pointer"; + ghost.Ghost(pImages[i-1]); + + ghost.SetDisposalMethod(GetDisposalMethod()); + ghost.EncodeExtension(fp); + + ghost.EncodeBody(fp,bLocalColorMap); + } + + fp->PutC(';'); // Write the GIF file terminator + + } catch (char *message) { + strncpy(info.szLastError,message,255); + return false; + } + return true; +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::EncodeHeader(CxFile *fp) +{ + fp->Write("GIF89a",1,6); //GIF Header + + Putword(head.biWidth,fp); //Logical screen descriptor + Putword(head.biHeight,fp); + + BYTE Flags; + if (head.biClrUsed==0){ + Flags=0x11; + } else { + Flags = 0x80; + Flags |=(head.biBitCount - 1) << 5; + Flags |=(head.biBitCount - 1); + } + + fp->PutC(Flags); //GIF "packed fields" + fp->PutC(0); //GIF "BackGround" + fp->PutC(0); //GIF "pixel aspect ratio" + + if (head.biClrUsed!=0){ + RGBQUAD* pPal = GetPalette(); + for(DWORD i=0; iPutC(pPal[i].rgbRed); + fp->PutC(pPal[i].rgbGreen); + fp->PutC(pPal[i].rgbBlue); + } + } +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::EncodeExtension(CxFile *fp) +{ + // TRK BEGIN : transparency + fp->PutC('!'); + fp->PutC(TRANSPARENCY_CODE); + + gifgce.transpcolflag = (info.nBkgndIndex != -1) ? 1 : 0; + gifgce.userinputflag = 0; + gifgce.dispmeth = m_dispmeth; + gifgce.res = 0; + gifgce.delaytime = (WORD)info.dwFrameDelay; + gifgce.transpcolindex = (BYTE)info.nBkgndIndex; + fp->PutC(sizeof(gifgce)); + fp->Write(&gifgce, sizeof(gifgce), 1); + fp->PutC(0); + // TRK END +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::EncodeLoopExtension(CxFile *fp) +{ + fp->PutC('!'); //byte 1 : 33 (hex 0x21) GIF Extension code + fp->PutC(255); //byte 2 : 255 (hex 0xFF) Application Extension Label + fp->PutC(11); //byte 3 : 11 (hex (0x0B) Length of Application Block (eleven bytes of data to follow) + fp->Write("NETSCAPE2.0",11,1); + fp->PutC(3); //byte 15 : 3 (hex 0x03) Length of Data Sub-Block (three bytes of data to follow) + fp->PutC(1); //byte 16 : 1 (hex 0x01) + Putword(m_loops,fp); //bytes 17 to 18 : 0 to 65535, an unsigned integer in lo-hi byte format. + //This indicate the number of iterations the loop should be executed. + fp->PutC(0); //bytes 19 : 0 (hex 0x00) a Data Sub-block Terminator. +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::EncodeBody(CxFile *fp, bool bLocalColorMap) +{ + curx = 0; + cury = head.biHeight - 1; //because we read the image bottom to top + CountDown = (long)head.biWidth * (long)head.biHeight; + + fp->PutC(','); + + Putword(info.xOffset,fp); + Putword(info.yOffset,fp); + Putword(head.biWidth,fp); + Putword(head.biHeight,fp); + + BYTE Flags=0x00; //non-interlaced (0x40 = interlaced) (0x80 = LocalColorMap) + if (bLocalColorMap) { Flags|=0x80; Flags|=head.biBitCount-1; } + fp->PutC(Flags); + + if (bLocalColorMap){ + Flags|=0x87; + RGBQUAD* pPal = GetPalette(); + for(DWORD i=0; iPutC(pPal[i].rgbRed); + fp->PutC(pPal[i].rgbGreen); + fp->PutC(pPal[i].rgbBlue); + } + } + + int InitCodeSize = head.biBitCount <=1 ? 2 : head.biBitCount; + // Write out the initial code size + fp->PutC((BYTE)InitCodeSize); + + // Go and actually compress the data + switch (GetCodecOption(CXIMAGE_FORMAT_GIF)) + { + case 1: //uncompressed + compressNONE(InitCodeSize+1, fp); + break; + case 2: //RLE + compressRLE(InitCodeSize+1, fp); + break; + default: //LZW + compressLZW(InitCodeSize+1, fp); + } + + // Write out a Zero-length packet (to end the series) + fp->PutC(0); +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::EncodeComment(CxFile *fp) +{ + unsigned long n = (unsigned long) strlen(m_comment); + if (n>255) n=255; + if (n) { + fp->PutC('!'); //extension code: + fp->PutC(254); //comment extension + fp->PutC((BYTE)n); //size of comment + fp->Write(m_comment,n,1); + fp->PutC(0); //block terminator + } +} +//////////////////////////////////////////////////////////////////////////////// +bool CxImageGIF::EncodeRGB(CxFile *fp) +{ + EncodeHeader(fp); + +// EncodeLoopExtension(fp); + + EncodeComment(fp); + + unsigned long w,h; + w=h=0; + const long cellw = 17; + const long cellh = 15; + CxImageGIF tmp; + for (long y=0;yPutC(';'); // Write the GIF file terminator + + return true; // done! +} +//////////////////////////////////////////////////////////////////////////////// +#endif // CXIMAGE_SUPPORT_ENCODE +//////////////////////////////////////////////////////////////////////////////// +// Return the next pixel from the image +// fix for 1 & 4 bpp images +int CxImageGIF::GifNextPixel( ) +{ + if( CountDown == 0 ) return EOF; + --CountDown; + int r = GetPixelIndex(curx,cury); + // Bump the current X position + ++curx; + if( curx == head.biWidth ){ + curx = 0; + cury--; //bottom to top + } + return r; +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::Putword(int w, CxFile *fp ) +{ + fp->PutC((BYTE)(w & 0xff)); + fp->PutC((BYTE)((w / 256) & 0xff)); +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::compressNONE( int init_bits, CxFile* outfile) +{ + register long c; + register long ent; + + // g_init_bits - initial number of bits + // g_outfile - pointer to output file + g_init_bits = init_bits; + g_outfile = outfile; + + // Set up the necessary values + cur_accum = cur_bits = clear_flg = 0; + maxcode = (short)MAXCODE(n_bits = g_init_bits); + code_int maxmaxcode = (code_int)1 << MAXBITSCODES; + + ClearCode = (1 << (init_bits - 1)); + EOFCode = ClearCode + 1; + free_ent = (short)(ClearCode + 2); + + a_count=0; + ent = GifNextPixel( ); + + output( (code_int)ClearCode ); + + while ( ent != EOF ) { + c = GifNextPixel(); + + output ( (code_int) ent ); + ent = c; + if ( free_ent < maxmaxcode ) { + free_ent++; + } else { + free_ent=(short)(ClearCode+2); + clear_flg=1; + output((code_int)ClearCode); + } + } + // Put out the final code. + output( (code_int) EOFCode ); +} +//////////////////////////////////////////////////////////////////////////////// + +/*************************************************************************** + * + * GIFCOMPR.C - LZW GIF Image compression routines + * + ***************************************************************************/ + +void CxImageGIF::compressLZW( int init_bits, CxFile* outfile) +{ + register long fcode; + register long c; + register long ent; + register long hshift; + register long disp; + register long i; + + // g_init_bits - initial number of bits + // g_outfile - pointer to output file + g_init_bits = init_bits; + g_outfile = outfile; + + // Set up the necessary values + cur_accum = cur_bits = clear_flg = 0; + maxcode = (short)MAXCODE(n_bits = g_init_bits); + code_int maxmaxcode = (code_int)1 << MAXBITSCODES; + + ClearCode = (1 << (init_bits - 1)); + EOFCode = ClearCode + 1; + free_ent = (short)(ClearCode + 2); + + a_count=0; + ent = GifNextPixel( ); + + hshift = 0; + for ( fcode = (long) HSIZE; fcode < 65536L; fcode *= 2L ) ++hshift; + hshift = 8 - hshift; /* set hash code range bound */ + cl_hash((long)HSIZE); /* clear hash table */ + output( (code_int)ClearCode ); + + while ( (c = GifNextPixel( )) != EOF ) { + + fcode = (long) (((long) c << MAXBITSCODES) + ent); + i = (((code_int)c << hshift) ^ ent); /* xor hashing */ + + if ( HashTabOf (i) == fcode ) { + ent = CodeTabOf (i); + continue; + } else if ( (long)HashTabOf (i) < 0 ) /* empty slot */ + goto nomatch; + disp = HSIZE - i; /* secondary hash (after G. Knott) */ + if ( i == 0 ) disp = 1; +probe: + if ( (i -= disp) < 0 ) i += HSIZE; + if ( HashTabOf (i) == fcode ) { ent = CodeTabOf (i); continue; } + if ( (long)HashTabOf (i) > 0 ) goto probe; +nomatch: + output ( (code_int) ent ); + ent = c; + if ( free_ent < maxmaxcode ) { + CodeTabOf (i) = free_ent++; /* code -> hashtable */ + HashTabOf (i) = fcode; + } else { + cl_hash((long)HSIZE); + free_ent=(short)(ClearCode+2); + clear_flg=1; + output((code_int)ClearCode); + } + } + // Put out the final code. + output( (code_int)ent ); + output( (code_int) EOFCode ); +} +//////////////////////////////////////////////////////////////////////////////// + +static const unsigned long code_mask[] = { 0x0000, 0x0001, 0x0003, 0x0007, 0x000F, + 0x001F, 0x003F, 0x007F, 0x00FF, + 0x01FF, 0x03FF, 0x07FF, 0x0FFF, + 0x1FFF, 0x3FFF, 0x7FFF, 0xFFFF }; + +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::output( code_int code) +{ + cur_accum &= code_mask[ cur_bits ]; + + if( cur_bits > 0 ) + cur_accum |= ((long)code << cur_bits); + else + cur_accum = code; + + cur_bits += n_bits; + + while( cur_bits >= 8 ) { + char_out( (unsigned int)(cur_accum & 0xff) ); + cur_accum >>= 8; + cur_bits -= 8; + } + + /* + * If the next entry is going to be too big for the code size, + * then increase it, if possible. + */ + + if ( free_ent > maxcode || clear_flg ) { + if( clear_flg ) { + maxcode = (short)MAXCODE(n_bits = g_init_bits); + clear_flg = 0; + } else { + ++n_bits; + if ( n_bits == MAXBITSCODES ) + maxcode = (code_int)1 << MAXBITSCODES; /* should NEVER generate this code */ + else + maxcode = (short)MAXCODE(n_bits); + } + } + + if( code == EOFCode ) { + // At EOF, write the rest of the buffer. + while( cur_bits > 0 ) { + char_out( (unsigned int)(cur_accum & 0xff) ); + cur_accum >>= 8; + cur_bits -= 8; + } + + flush_char(); + g_outfile->Flush(); + + if(g_outfile->Error()) strcpy(info.szLastError,"Write Error in GIF file"); + } +} +//////////////////////////////////////////////////////////////////////////////// + +void CxImageGIF::cl_hash(register long hsize) + +{ + register long *htab_p = htab+hsize; + + register long i; + register long m1 = -1L; + + i = hsize - 16; + + do { + *(htab_p-16)=m1; + *(htab_p-15)=m1; + *(htab_p-14)=m1; + *(htab_p-13)=m1; + *(htab_p-12)=m1; + *(htab_p-11)=m1; + *(htab_p-10)=m1; + *(htab_p-9)=m1; + *(htab_p-8)=m1; + *(htab_p-7)=m1; + *(htab_p-6)=m1; + *(htab_p-5)=m1; + *(htab_p-4)=m1; + *(htab_p-3)=m1; + *(htab_p-2)=m1; + *(htab_p-1)=m1; + + htab_p-=16; + } while ((i-=16) >=0); + + for (i+=16;i>0;--i) + *--htab_p=m1; +} + +/******************************************************************************* +* GIF specific +*******************************************************************************/ + +void CxImageGIF::char_out(int c) +{ + accum[a_count++]=(char)c; + if (a_count >=254) + flush_char(); +} + +void CxImageGIF::flush_char() +{ + if (a_count > 0) { + g_outfile->PutC((BYTE)a_count); + g_outfile->Write(accum,1,a_count); + a_count=0; + } +} + +/******************************************************************************* +* GIF decoder +*******************************************************************************/ +/* DECODE.C - An LZW decoder for GIF + * Copyright (C) 1987, by Steven A. Bennett + * Copyright (C) 1994, C++ version by Alejandro Aguilar Sierra +* + * Permission is given by the author to freely redistribute and include + * this code in any program as long as this credit is given where due. + * + * In accordance with the above, I want to credit Steve Wilhite who wrote + * the code which this is heavily inspired by... + * + * GIF and 'Graphics Interchange Format' are trademarks (tm) of + * Compuserve, Incorporated, an H&R Block Company. + * + * Release Notes: This file contains a decoder routine for GIF images + * which is similar, structurally, to the original routine by Steve Wilhite. + * It is, however, somewhat noticably faster in most cases. + * + */ + +//////////////////////////////////////////////////////////////////////////////// + +short CxImageGIF::init_exp(short size) +{ + curr_size = (short)(size + 1); + top_slot = (short)(1 << curr_size); + clear = (short)(1 << size); + ending = (short)(clear + 1); + slot = newcodes = (short)(ending + 1); + navail_bytes = nbits_left = 0; + + memset(stack,0,MAX_CODES + 1); + memset(prefix,0,MAX_CODES + 1); + memset(suffix,0,MAX_CODES + 1); + return(0); +} +//////////////////////////////////////////////////////////////////////////////// + +/* get_next_code() + * - gets the next code from the GIF file. Returns the code, or else + * a negative number in case of file errors... + */ +short CxImageGIF::get_next_code(CxFile* file) +{ + short i, x; + DWORD ret; + + if (nbits_left == 0) { + if (navail_bytes <= 0) { + /* Out of bytes in current block, so read next block */ + pbytes = byte_buff; + if ((navail_bytes = (short)get_byte(file)) < 0) + return(navail_bytes); + else if (navail_bytes) { + for (i = 0; i < navail_bytes; ++i) { + if ((x = (short)get_byte(file)) < 0) return(x); + byte_buff[i] = (BYTE)x; + } + } + } + b1 = *pbytes++; + nbits_left = 8; + --navail_bytes; + } + + if (navail_bytes<0) return ending; // prevent deadlocks (thanks to Mike Melnikov) + + ret = b1 >> (8 - nbits_left); + while (curr_size > nbits_left){ + if (navail_bytes <= 0){ + /* Out of bytes in current block, so read next block*/ + pbytes = byte_buff; + if ((navail_bytes = (short)get_byte(file)) < 0) + return(navail_bytes); + else if (navail_bytes){ + for (i = 0; i < navail_bytes; ++i){ + if ((x = (short)get_byte(file)) < 0) return(x); + byte_buff[i] = (BYTE)x; + } + } + } + b1 = *pbytes++; + ret |= b1 << nbits_left; + nbits_left += 8; + --navail_bytes; + } + nbits_left = (short)(nbits_left-curr_size); + ret &= code_mask[curr_size]; + return((short)(ret)); +} +//////////////////////////////////////////////////////////////////////////////// + +/* short decoder(linewidth) + * short linewidth; * Pixels per line of image * + * + * - This function decodes an LZW image, according to the method used + * in the GIF spec. Every *linewidth* "characters" (ie. pixels) decoded + * will generate a call to out_line(), which is a user specific function + * to display a line of pixels. The function gets it's codes from + * get_next_code() which is responsible for reading blocks of data and + * seperating them into the proper size codes. Finally, get_byte() is + * the global routine to read the next BYTE from the GIF file. + * + * It is generally a good idea to have linewidth correspond to the actual + * width of a line (as specified in the Image header) to make your own + * code a bit simpler, but it isn't absolutely necessary. + * + * Returns: 0 if successful, else negative. (See ERRS.H) + * + */ +/* bad_code_count is incremented each time an out of range code is read. + * When this value is non-zero after a decode, your GIF file is probably + * corrupt in some way... + */ +short CxImageGIF::decoder(CxFile* file, CImageIterator* iter, short linewidth, int &bad_code_count) +{ + register BYTE *sp, *bufptr; + BYTE *buf; + register short code, fc, oc, bufcnt; + short c, size, ret; + + /* Initialize for decoding a new image... */ + bad_code_count = 0; + if ((size = (short)get_byte(file)) < 0) return(size); + if (size < 2 || 9 < size) return(BAD_CODE_SIZE); + // out_line = outline; + init_exp(size); + //printf("L %d %x\n",linewidth,size); + + /* Initialize in case they forgot to put in a clear code. + * (This shouldn't happen, but we'll try and decode it anyway...) + */ + oc = fc = 0; + + /* Allocate space for the decode buffer */ + if ((buf = new BYTE[linewidth + 1]) == NULL) return(OUT_OF_MEMORY); + + /* Set up the stack pointer and decode buffer pointer */ + sp = stack; + bufptr = buf; + bufcnt = linewidth; + + /* This is the main loop. For each code we get we pass through the + * linked list of prefix codes, pushing the corresponding "character" for + * each code onto the stack. When the list reaches a single "character" + * we push that on the stack too, and then start unstacking each + * character for output in the correct order. Special handling is + * included for the clear code, and the whole thing ends when we get + * an ending code. + */ + while ((c = get_next_code(file)) != ending) { + /* If we had a file error, return without completing the decode*/ + if (c < 0){ + delete[] buf; + return(0); + } + /* If the code is a clear code, reinitialize all necessary items.*/ + if (c == clear){ + curr_size = (short)(size + 1); + slot = newcodes; + top_slot = (short)(1 << curr_size); + + /* Continue reading codes until we get a non-clear code + * (Another unlikely, but possible case...) + */ + while ((c = get_next_code(file)) == clear); + + /* If we get an ending code immediately after a clear code + * (Yet another unlikely case), then break out of the loop. + */ + if (c == ending) break; + + /* Finally, if the code is beyond the range of already set codes, + * (This one had better NOT happen... I have no idea what will + * result from this, but I doubt it will look good...) then set it + * to color zero. + */ + if (c >= slot) c = 0; + oc = fc = c; + + /* And let us not forget to put the char into the buffer... And + * if, on the off chance, we were exactly one pixel from the end + * of the line, we have to send the buffer to the out_line() + * routine... + */ + *bufptr++ = (BYTE)c; + if (--bufcnt == 0) { + if ((ret = (short)out_line(iter, buf, linewidth)) < 0) { + delete[] buf; + return(ret); + } + bufptr = buf; + bufcnt = linewidth; + } + } else { + /* In this case, it's not a clear code or an ending code, so + * it must be a code code... So we can now decode the code into + * a stack of character codes. (Clear as mud, right?) + */ + code = c; + + /* Here we go again with one of those off chances... If, on the + * off chance, the code we got is beyond the range of those already + * set up (Another thing which had better NOT happen...) we trick + * the decoder into thinking it actually got the last code read. + * (Hmmn... I'm not sure why this works... But it does...) + */ + if (code >= slot) { + if (code > slot) ++bad_code_count; + code = oc; + *sp++ = (BYTE)fc; + } + + /* Here we scan back along the linked list of prefixes, pushing + * helpless characters (ie. suffixes) onto the stack as we do so. + */ + while (code >= newcodes) { + *sp++ = suffix[code]; + code = prefix[code]; + } + + /* Push the last character on the stack, and set up the new + * prefix and suffix, and if the required slot number is greater + * than that allowed by the current bit size, increase the bit + * size. (NOTE - If we are all full, we *don't* save the new + * suffix and prefix... I'm not certain if this is correct... + * it might be more proper to overwrite the last code... + */ + *sp++ = (BYTE)code; + if (slot < top_slot){ + suffix[slot] = (BYTE)(fc = (BYTE)code); + prefix[slot++] = oc; + oc = c; + } + if (slot >= top_slot){ + if (curr_size < 12) { + top_slot <<= 1; + ++curr_size; + } + } + + /* Now that we've pushed the decoded string (in reverse order) + * onto the stack, lets pop it off and put it into our decode + * buffer... And when the decode buffer is full, write another + * line... + */ + while (sp > stack) { + *bufptr++ = *(--sp); + if (--bufcnt == 0) { + if ((ret = (short)out_line(iter, buf, linewidth)) < 0) { + delete[] buf; + return(ret); + } + bufptr = buf; + bufcnt = linewidth; + } + } + } + } + ret = 0; + if (bufcnt != linewidth) + ret = (short)out_line(iter, buf, (linewidth - bufcnt)); + delete[] buf; + return(ret); +} +//////////////////////////////////////////////////////////////////////////////// +int CxImageGIF::get_num_frames(CxFile *fp,struct_TabCol* TabColSrc,struct_dscgif* dscgif) +{ + struct_image image; + + long pos=fp->Tell(); + int nframes=0; + + struct_TabCol TempTabCol; + memcpy(&TempTabCol,TabColSrc,sizeof(struct_TabCol)); + + char ch; + bool bPreviousWasNull = true; + + for (BOOL bContinue = TRUE; bContinue; ) + { + if (fp->Read(&ch, sizeof(ch), 1) != 1) {break;} + + if (bPreviousWasNull || ch==0) + { + switch (ch) + { + case '!': // extension + { + DecodeExtension(fp); + break; + } + case ',': // image + { + + assert(sizeof(image) == 9); + //log << "Image header" << endl; + fp->Read(&image,sizeof(image),1); + + //avoid byte order problems with Solaris + BYTE *byteData = (BYTE *) & image; + image.l = byteData[0]+byteData[1]*256; + image.t = byteData[2]+byteData[3]*256; + image.w = byteData[4]+byteData[5]*256; + image.h = byteData[6]+byteData[7]*256; + + if (((image.l + image.w) > dscgif->scrwidth)||((image.t + image.h) > dscgif->scrheight)) + break; + + nframes++; + + // Local colour map? + if (image.pf & 0x80) { + TempTabCol.sogct = (short)(1 << ((image.pf & 0x07) +1)); + assert(3 == sizeof(struct rgb_color)); + fp->Read(TempTabCol.paleta,sizeof(struct rgb_color)*TempTabCol.sogct,1); + //log << "Local colour map" << endl; + } + + int bpp; // select the correct bit per pixel value + if (TempTabCol.sogct <= 2) bpp = 1; + else if (TempTabCol.sogct <= 16) bpp = 4; + else bpp = 8; + + Create(image.w, image.h, bpp, CXIMAGE_FORMAT_GIF); + + CImageIterator* iter = new CImageIterator(this); + iter->Upset(); + int badcode=0; + ibf = GIFBUFTAM+1; + + interlaced = image.pf & 0x40; + iheight = image.h; + istep = 8; + iypos = 0; + ipass = 0; + + long pos_start = fp->Tell(); + + //if (interlaced) log << "Interlaced" << endl; + decoder(fp, iter, image.w, badcode); + delete iter; + + if (badcode){ + seek_next_image(fp,pos_start); + } else { + fp->Seek(-(ibfmax - ibf - 1), SEEK_CUR); + } + + break; + } + case ';': //terminator + bContinue=false; + break; + default: + bPreviousWasNull = (ch==0); + break; + } + } + } + + fp->Seek(pos,SEEK_SET); + return nframes; +} +//////////////////////////////////////////////////////////////////////////////// +long CxImageGIF::seek_next_image(CxFile* fp, long position) +{ + fp->Seek(position, SEEK_SET); + char ch1,ch2; + ch1=ch2=0; + while(fp->Read(&ch2,sizeof(char),1)>0){ + if (ch1 == 0 && ch2 == ','){ + fp->Seek(-1,SEEK_CUR); + return fp->Tell(); + } else { + ch1 = ch2; + } + } + return -1; +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::SetDisposalMethod(int dm) +{ m_dispmeth=dm; } +//////////////////////////////////////////////////////////////////////////////// +long CxImageGIF::GetDisposalMethod() +{ return m_dispmeth; } +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::SetLoops(int loops) +{ m_loops=loops; } +//////////////////////////////////////////////////////////////////////////////// +long CxImageGIF::GetLoops() +{ return m_loops; } +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::SetComment(const char* sz_comment_in) +{ if (sz_comment_in) strncpy(m_comment,sz_comment_in,255); } +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::GetComment(char* sz_comment_out) +{ if (sz_comment_out) strncpy(sz_comment_out,m_comment,255); } +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::GifMix(CxImage & imgsrc2, struct_image & imgdesc) +{ + long ymin = max(0,(long)(GetHeight()-imgdesc.t - imgdesc.h)); + long ymax = GetHeight()-imgdesc.t; + long xmin = imgdesc.l; + long xmax = min(GetWidth(), (DWORD)(imgdesc.l + imgdesc.w)); + + long ibg2= imgsrc2.GetTransIndex(); + BYTE i2; + + for(long y = ymin; y < ymax; y++){ + for(long x = xmin; x < xmax; x++){ + i2 = imgsrc2.GetPixelIndex(x-xmin,y-ymin); + if(i2!=ibg2) SetPixelIndex(x,y,i2); + } + } +} +//////////////////////////////////////////////////////////////////////////////// +/*----------------------------------------------------------------------- + * + * miGIF Compression - mouse and ivo's GIF-compatible compression + * + * -run length encoding compression routines- + * + * Copyright (C) 1998 Hutchison Avenue Software Corporation + * http://www.hasc.com + * info@hasc.com + * + * Permission to use, copy, modify, and distribute this software and its + * documentation for any purpose and without fee is hereby granted, provided + * that the above copyright notice appear in all copies and that both that + * copyright notice and this permission notice appear in supporting + * documentation. This software is provided "AS IS." The Hutchison Avenue + * Software Corporation disclaims all warranties, either express or implied, + * including but not limited to implied warranties of merchantability and + * fitness for a particular purpose, with respect to this code and accompanying + * documentation. + * + * The miGIF compression routines do not, strictly speaking, generate files + * conforming to the GIF spec, since the image data is not LZW-compressed + * (this is the point: in order to avoid transgression of the Unisys patent + * on the LZW algorithm.) However, miGIF generates data streams that any + * reasonably sane LZW decompresser will decompress to what we want. + * + * miGIF compression uses run length encoding. It compresses horizontal runs + * of pixels of the same color. This type of compression gives good results + * on images with many runs, for example images with lines, text and solid + * shapes on a solid-colored background. It gives little or no compression + * on images with few runs, for example digital or scanned photos. + * + * der Mouse + * mouse@rodents.montreal.qc.ca + * 7D C8 61 52 5D E7 2D 39 4E F1 31 3E E8 B3 27 4B + * + * ivo@hasc.com + * + * The Graphics Interchange Format(c) is the Copyright property of + * CompuServe Incorporated. GIF(sm) is a Service Mark property of + * CompuServe Incorporated. + * + */ +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::rle_clear(struct_RLE* rle) +{ + rle->out_bits = rle->out_bits_init; + rle->out_bump = rle->out_bump_init; + rle->out_clear = rle->out_clear_init; + rle->out_count = 0; + rle->rl_table_max = 0; + rle->just_cleared = 1; +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::rle_flush(struct_RLE* rle) +{ + if (rle->rl_count == 1){ + rle_output_plain(rle->rl_pixel,rle); + rle->rl_count = 0; + return; + } + if (rle->just_cleared){ + rle_flush_fromclear(rle->rl_count,rle); + } else if ((rle->rl_table_max < 2) || (rle->rl_table_pixel != rle->rl_pixel)) { + rle_flush_clearorrep(rle->rl_count,rle); + } else { + rle_flush_withtable(rle->rl_count,rle); + } + rle->rl_count = 0; +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::rle_output_plain(int c,struct_RLE* rle) +{ + rle->just_cleared = 0; + rle_output(c,rle); + rle->out_count++; + if (rle->out_count >= rle->out_bump){ + rle->out_bits ++; + rle->out_bump += 1 << (rle->out_bits - 1); + } + if (rle->out_count >= rle->out_clear){ + rle_output(rle->code_clear,rle); + rle_clear(rle); + } +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::rle_flush_fromclear(int count,struct_RLE* rle) +{ + int n; + + rle->out_clear = rle->max_ocodes; + rle->rl_table_pixel = rle->rl_pixel; + n = 1; + while (count > 0){ + if (n == 1){ + rle->rl_table_max = 1; + rle_output_plain(rle->rl_pixel,rle); + count --; + } else if (count >= n){ + rle->rl_table_max = n; + rle_output_plain(rle->rl_basecode+n-2,rle); + count -= n; + } else if (count == 1){ + rle->rl_table_max ++; + rle_output_plain(rle->rl_pixel,rle); + count = 0; + } else { + rle->rl_table_max ++; + rle_output_plain(rle->rl_basecode+count-2,rle); + count = 0; + } + if (rle->out_count == 0) n = 1; else n ++; + } + rle_reset_out_clear(rle); +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::rle_reset_out_clear(struct_RLE* rle) +{ + rle->out_clear = rle->out_clear_init; + if (rle->out_count >= rle->out_clear){ + rle_output(rle->code_clear,rle); + rle_clear(rle); + } +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::rle_flush_withtable(int count, struct_RLE* rle) +{ + int repmax; + int repleft; + int leftover; + + repmax = count / rle->rl_table_max; + leftover = count % rle->rl_table_max; + repleft = (leftover ? 1 : 0); + if (rle->out_count+repmax+repleft > rle->max_ocodes){ + repmax = rle->max_ocodes - rle->out_count; + leftover = count - (repmax * rle->rl_table_max); + repleft = 1 + rle_compute_triangle_count(leftover,rle->max_ocodes); + } + if (1+rle_compute_triangle_count(count,rle->max_ocodes) < (unsigned int)(repmax+repleft)){ + rle_output(rle->code_clear,rle); + rle_clear(rle); + rle_flush_fromclear(count,rle); + return; + } + rle->out_clear = rle->max_ocodes; + for (;repmax>0;repmax--) rle_output_plain(rle->rl_basecode+rle->rl_table_max-2,rle); + if (leftover){ + if (rle->just_cleared){ + rle_flush_fromclear(leftover,rle); + } else if (leftover == 1){ + rle_output_plain(rle->rl_pixel,rle); + } else { + rle_output_plain(rle->rl_basecode+leftover-2,rle); + } + } + rle_reset_out_clear(rle); +} +//////////////////////////////////////////////////////////////////////////////// +unsigned int CxImageGIF::rle_compute_triangle_count(unsigned int count, unsigned int nrepcodes) +{ + unsigned int perrep; + unsigned int cost; + + cost = 0; + perrep = (nrepcodes * (nrepcodes+1)) / 2; + while (count >= perrep){ + cost += nrepcodes; + count -= perrep; + } + if (count > 0){ + unsigned int n; + n = rle_isqrt(count); + while ((n*(n+1)) >= 2*count) n --; + while ((n*(n+1)) < 2*count) n ++; + cost += n; + } + return(cost); +} +//////////////////////////////////////////////////////////////////////////////// +unsigned int CxImageGIF::rle_isqrt(unsigned int x) +{ + unsigned int r; + unsigned int v; + + if (x < 2) return(x); + for (v=x,r=1;v;v>>=2,r<<=1) ; + while (1){ + v = ((x / r) + r) / 2; + if ((v == r) || (v == r+1)) return(r); + r = v; + } +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::rle_flush_clearorrep(int count, struct_RLE* rle) +{ + int withclr; + withclr = 1 + rle_compute_triangle_count(count,rle->max_ocodes); + if (withclr < count) { + rle_output(rle->code_clear,rle); + rle_clear(rle); + rle_flush_fromclear(count,rle); + } else { + for (;count>0;count--) rle_output_plain(rle->rl_pixel,rle); + } +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::rle_write_block(struct_RLE* rle) +{ + g_outfile->PutC((BYTE)rle->oblen); + g_outfile->Write(rle->oblock,1,rle->oblen); + rle->oblen = 0; +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::rle_block_out(unsigned char c, struct_RLE* rle) +{ + rle->oblock[rle->oblen++] = c; + if (rle->oblen >= 255) rle_write_block(rle); +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::rle_block_flush(struct_RLE* rle) +{ + if (rle->oblen > 0) rle_write_block(rle); +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::rle_output(int val, struct_RLE* rle) +{ + rle->obuf |= val << rle->obits; + rle->obits += rle->out_bits; + while (rle->obits >= 8){ + rle_block_out(rle->obuf&0xff,rle); + rle->obuf >>= 8; + rle->obits -= 8; + } +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::rle_output_flush(struct_RLE* rle) +{ + if (rle->obits > 0) rle_block_out(rle->obuf,rle); + rle_block_flush(rle); +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageGIF::compressRLE( int init_bits, CxFile* outfile) +{ + g_init_bits = init_bits; + g_outfile = outfile; + + struct_RLE rle; + rle.code_clear = 1 << (init_bits - 1); + rle.code_eof = rle.code_clear + 1; + rle.rl_basecode = rle.code_eof + 1; + rle.out_bump_init = (1 << (init_bits - 1)) - 1; + rle.out_clear_init = (init_bits <= 3) ? 9 : (rle.out_bump_init-1); + rle.out_bits_init = init_bits; + rle.max_ocodes = (1 << MAXBITSCODES) - ((1 << (rle.out_bits_init - 1)) + 3); + rle.rl_count = 0; + rle_clear(&rle); + rle.obuf = 0; + rle.obits = 0; + rle.oblen = 0; + + rle_output(rle.code_clear,&rle); + + int c; + while (1){ + c = GifNextPixel(); + if ((rle.rl_count > 0) && (c != rle.rl_pixel)) rle_flush(&rle); + if (c == EOF) break; + if (rle.rl_pixel == c){ + rle.rl_count++; + } else { + rle.rl_pixel = c; + rle.rl_count = 1; + } + } + rle_output(rle.code_eof,&rle); + rle_output_flush(&rle); +} +//////////////////////////////////////////////////////////////////////////////// +#endif // CXIMAGE_SUPPORT_GIF diff --git a/src/win32/dependencies/cximage/ximagif.h b/src/win32/dependencies/cximage/ximagif.h new file mode 100644 index 00000000..cd4f4516 --- /dev/null +++ b/src/win32/dependencies/cximage/ximagif.h @@ -0,0 +1,248 @@ +/* + * File: ximagif.h + * Purpose: GIF Image Class Loader and Writer + */ +/* ========================================================== + * CxImageGIF (c) 07/Aug/2001 Davide Pizzolato - www.xdp.it + * For conditions of distribution and use, see copyright notice in ximage.h + * + * Special thanks to Troels Knakkergaard for new features, enhancements and bugfixes + * + * original CImageGIF and CImageIterator implementation are: + * Copyright: (c) 1995, Alejandro Aguilar Sierra + * + * 6/15/97 Randy Spann: Added GIF87a writing support + * R.Spann@ConnRiver.net + * + * DECODE.C - An LZW decoder for GIF + * Copyright (C) 1987, by Steven A. Bennett + * Copyright (C) 1994, C++ version by Alejandro Aguilar Sierra + * + * In accordance with the above, I want to credit Steve Wilhite who wrote + * the code which this is heavily inspired by... + * + * GIF and 'Graphics Interchange Format' are trademarks (tm) of + * Compuserve, Incorporated, an H&R Block Company. + * + * Release Notes: This file contains a decoder routine for GIF images + * which is similar, structurally, to the original routine by Steve Wilhite. + * It is, however, somewhat noticably faster in most cases. + * + * ========================================================== + */ + +#if !defined(__ximaGIF_h) +#define __ximaGIF_h + +#include "ximage.h" + +#if CXIMAGE_SUPPORT_GIF + +typedef short int code_int; + +/* Various error codes used by decoder */ +#define OUT_OF_MEMORY -10 +#define BAD_CODE_SIZE -20 +#define READ_ERROR -1 +#define WRITE_ERROR -2 +#define OPEN_ERROR -3 +#define CREATE_ERROR -4 +#define MAX_CODES 4095 +#define GIFBUFTAM 16383 +#define TRANSPARENCY_CODE 0xF9 + +//LZW GIF Image compression +#define MAXBITSCODES 12 +#define HSIZE 5003 /* 80% occupancy */ +#define MAXCODE(n_bits) (((code_int) 1 << (n_bits)) - 1) +#define HashTabOf(i) htab[i] +#define CodeTabOf(i) codetab[i] + + +class CImageIterator; +class DLL_EXP CxImageGIF: public CxImage +{ +#pragma pack(1) + +typedef struct tag_gifgce{ + BYTE transpcolflag:1; + BYTE userinputflag:1; + BYTE dispmeth:3; + BYTE res:3; + WORD delaytime; + BYTE transpcolindex; +} struct_gifgce; + +typedef struct tag_dscgif{ /* Logic Screen Descriptor */ + char header[6]; /* Firma and version */ + WORD scrwidth; + WORD scrheight; + char pflds; + char bcindx; + char pxasrat; +} struct_dscgif; + +typedef struct tag_image{ /* Image Descriptor */ + WORD l; + WORD t; + WORD w; + WORD h; + BYTE pf; +} struct_image; + +typedef struct tag_TabCol{ /* Tabla de colores */ + short colres; /* color resolution */ + short sogct; /* size of global color table */ + rgb_color paleta[256]; /* paleta */ +} struct_TabCol; + +typedef struct tag_RLE{ + int rl_pixel; + int rl_basecode; + int rl_count; + int rl_table_pixel; + int rl_table_max; + int just_cleared; + int out_bits; + int out_bits_init; + int out_count; + int out_bump; + int out_bump_init; + int out_clear; + int out_clear_init; + int max_ocodes; + int code_clear; + int code_eof; + unsigned int obuf; + int obits; + unsigned char oblock[256]; + int oblen; +} struct_RLE; +#pragma pack() + +public: + CxImageGIF(): CxImage(CXIMAGE_FORMAT_GIF) {m_loops=0; m_dispmeth=0; m_comment[0]='\0';} + +// bool Load(const char * imageFileName){ return CxImage::Load(imageFileName,CXIMAGE_FORMAT_GIF);} +// bool Save(const char * imageFileName){ return CxImage::Save(imageFileName,CXIMAGE_FORMAT_GIF);} + + bool Decode(CxFile * fp); + bool Decode(FILE *fp) { CxIOFile file(fp); return Decode(&file); } + +#if CXIMAGE_SUPPORT_ENCODE + bool Encode(CxFile * fp); + bool Encode(CxFile * fp, CxImage ** pImages, int pagecount, bool bLocalColorMap = false); + bool Encode(FILE *fp) { CxIOFile file(fp); return Encode(&file); } + bool Encode(FILE *fp, CxImage ** pImages, int pagecount, bool bLocalColorMap = false) + { CxIOFile file(fp); return Encode(&file, pImages, pagecount, bLocalColorMap); } +#endif // CXIMAGE_SUPPORT_ENCODE + + void SetLoops(int loops); + long GetLoops(); + void SetComment(const char* sz_comment_in); + void GetComment(char* sz_comment_out); + void SetDisposalMethod(int dm); + long GetDisposalMethod(); + +protected: + bool DecodeExtension(CxFile *fp); + void EncodeHeader(CxFile *fp); + void EncodeLoopExtension(CxFile *fp); + void EncodeExtension(CxFile *fp); + void EncodeBody(CxFile *fp, bool bLocalColorMap = false); + void EncodeComment(CxFile *fp); + bool EncodeRGB(CxFile *fp); + void GifMix(CxImage & imgsrc2, struct_image & imgdesc); + + struct_gifgce gifgce; + + int curx, cury; + long CountDown; + unsigned long cur_accum; + int cur_bits; + int interlaced, iypos, istep, iheight, ipass; + int ibf; + int ibfmax; + BYTE buf[GIFBUFTAM + 1]; +// Implementation + int GifNextPixel (); + void Putword (int w, CxFile* fp ); + void compressNONE (int init_bits, CxFile* outfile); + void compressLZW (int init_bits, CxFile* outfile); + void output (code_int code ); + void cl_hash (long hsize); + void char_out (int c); + void flush_char (); + short init_exp(short size); + short get_next_code(CxFile*); + short decoder(CxFile*, CImageIterator* iter, short linewidth, int &bad_code_count); + int get_byte(CxFile*); + int out_line(CImageIterator* iter, unsigned char *pixels, int linelen); + int get_num_frames(CxFile *f,struct_TabCol* TabColSrc,struct_dscgif* dscgif); + long seek_next_image(CxFile* fp, long position); + + short curr_size; /* The current code size */ + short clear; /* Value for a clear code */ + short ending; /* Value for a ending code */ + short newcodes; /* First available code */ + short top_slot; /* Highest code for current size */ + short slot; /* Last read code */ + + /* The following static variables are used + * for seperating out codes */ + short navail_bytes; /* # bytes left in block */ + short nbits_left; /* # bits left in current BYTE */ + BYTE b1; /* Current BYTE */ + BYTE byte_buff[257]; /* Current block */ + BYTE *pbytes; /* Pointer to next BYTE in block */ + /* The reason we have these seperated like this instead of using + * a structure like the original Wilhite code did, is because this + * stuff generally produces significantly faster code when compiled... + * This code is full of similar speedups... (For a good book on writing + * C for speed or for space optomisation, see Efficient C by Tom Plum, + * published by Plum-Hall Associates...) + */ + BYTE stack[MAX_CODES + 1]; /* Stack for storing pixels */ + BYTE suffix[MAX_CODES + 1]; /* Suffix table */ + WORD prefix[MAX_CODES + 1]; /* Prefix linked list */ + +//LZW GIF Image compression routines + long htab [HSIZE]; + unsigned short codetab [HSIZE]; + int n_bits; /* number of bits/code */ + code_int maxcode; /* maximum code, given n_bits */ + code_int free_ent; /* first unused entry */ + int clear_flg; + int g_init_bits; + CxFile* g_outfile; + int ClearCode; + int EOFCode; + + int a_count; + char accum[256]; + + char m_comment[256]; + int m_loops; + int m_dispmeth; + +//RLE compression routines + void compressRLE( int init_bits, CxFile* outfile); + void rle_clear(struct_RLE* rle); + void rle_flush(struct_RLE* rle); + void rle_flush_withtable(int count, struct_RLE* rle); + void rle_flush_clearorrep(int count, struct_RLE* rle); + void rle_flush_fromclear(int count,struct_RLE* rle); + void rle_output_plain(int c,struct_RLE* rle); + void rle_reset_out_clear(struct_RLE* rle); + unsigned int rle_compute_triangle_count(unsigned int count, unsigned int nrepcodes); + unsigned int rle_isqrt(unsigned int x); + void rle_write_block(struct_RLE* rle); + void rle_block_out(unsigned char c, struct_RLE* rle); + void rle_block_flush(struct_RLE* rle); + void rle_output(int val, struct_RLE* rle); + void rle_output_flush(struct_RLE* rle); +}; + +#endif + +#endif diff --git a/src/win32/dependencies/cximage/ximahist.cpp b/src/win32/dependencies/cximage/ximahist.cpp new file mode 100644 index 00000000..8e3542f0 --- /dev/null +++ b/src/win32/dependencies/cximage/ximahist.cpp @@ -0,0 +1,649 @@ +// xImaHist.cpp : histogram functions +/* 28/01/2004 v1.00 - www.xdp.it + * CxImage version 5.99c 17/Oct/2004 + */ + +#include "ximage.h" + +#if CXIMAGE_SUPPORT_DSP + +//////////////////////////////////////////////////////////////////////////////// +long CxImage::Histogram(long* red, long* green, long* blue, long* gray, long colorspace) +{ + if (!pDib) return 0; + RGBQUAD color; + + if (red) memset(red,0,256*sizeof(long)); + if (green) memset(green,0,256*sizeof(long)); + if (blue) memset(blue,0,256*sizeof(long)); + if (gray) memset(gray,0,256*sizeof(long)); + + long xmin,xmax,ymin,ymax; + if (pSelection){ + xmin = info.rSelectionBox.left; xmax = info.rSelectionBox.right; + ymin = info.rSelectionBox.bottom; ymax = info.rSelectionBox.top; + } else { + xmin = ymin = 0; + xmax = head.biWidth; ymax=head.biHeight; + } + + for(long y=ymin; yn) n=red[i]; + if (green && green[i]>n) n=green[i]; + if (blue && blue[i]>n) n=blue[i]; + if (gray && gray[i]>n) n=gray[i]; + } + + return n; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * HistogramStretch + * \param method: 0 = luminance (default), 1 = linked channels , 2 = independent channels. + * \return true if everything is ok + * \author [dave] and [nipper] + */ +bool CxImage::HistogramStretch(long method) +{ + if (!pDib) return false; + + if ((head.biBitCount==8) && IsGrayScale()){ + // get min/max info + BYTE minc = 255, maxc = 0; + BYTE gray; + long y; + + double dbScaler = 50.0/head.biHeight; + + for (y=0; y maxc) maxc = gray; + } + } + + if (minc == 0 && maxc == 255) return true; + + // calculate LUT + BYTE lut[256]; + BYTE range = maxc - minc; + if (range != 0){ + for (long x = minc; x <= maxc; x++){ + lut[x] = (BYTE)(255 * (x - minc) / range); + } + } else lut[minc] = minc; + + for (y=0; y + // get min/max info + BYTE minc = 255, maxc = 0; + RGBQUAD color; + long y; + + for (y=0; y maxc) maxc = color.rgbRed; + if (color.rgbBlue > maxc) maxc = color.rgbBlue; + if (color.rgbGreen > maxc) maxc = color.rgbGreen; + } + } + + if (minc == 0 && maxc == 255) + return true; + + // calculate LUT + BYTE lut[256]; + BYTE range = maxc - minc; + + if (range != 0){ + for (long x = minc; x <= maxc; x++){ + lut[x] = (BYTE)(255 * (x - minc) / range); + } + } else lut[minc] = minc; + + // normalize image + double dbScaler = 100.0/head.biHeight; + + for (y=0; y + // get min/max info + BYTE minR = 255, maxR = 0; + BYTE minG = 255, maxG = 0; + BYTE minB = 255, maxB = 0; + RGBQUAD color; + long y; + + for (y=0; y maxR) maxR = color.rgbRed; + if (color.rgbBlue > maxB) maxB = color.rgbBlue; + if (color.rgbGreen > maxG) maxG = color.rgbGreen; + } + } + + if (minR == 0 && maxR == 255 && minG == 0 && maxG == 255 && minB == 0 && maxB == 255) + return true; + + // calculate LUT + BYTE lutR[256]; + BYTE range = maxR - minR; + if (range != 0) { + for (long x = minR; x <= maxR; x++){ + lutR[x] = (BYTE)(255 * (x - minR) / range); + } + } else lutR[minR] = minR; + + BYTE lutG[256]; + range = maxG - minG; + if (range != 0) { + for (long x = minG; x <= maxG; x++){ + lutG[x] = (BYTE)(255 * (x - minG) / range); + } + } else lutG[minG] = minG; + + BYTE lutB[256]; + range = maxB - minB; + if (range != 0) { + for (long x = minB; x <= maxB; x++){ + lutB[x] = (BYTE)(255 * (x - minB) / range); + } + } else lutB[minB] = minB; + + // normalize image + double dbScaler = 100.0/head.biHeight; + + for (y=0; y + // S = ( R - C ) ( B - A / D - C ) + double alimit = 0.0; + double blimit = 255.0; + double lowerc = 255.0; + double upperd = 0.0; + double tmpGray; + + RGBQUAD color; + RGBQUAD yuvClr; + double stretcheds; + + if ( head.biClrUsed == 0 ){ + long x, y, xmin, xmax, ymin, ymax; + xmin = ymin = 0; + xmax = head.biWidth; + ymax = head.biHeight; + + for( y = ymin; y < ymax; y++ ){ + info.nProgress = (long)(50*y/ymax); + if (info.nEscape) break; + for( x = xmin; x < xmax; x++ ){ + color = GetPixelColor( x, y ); + tmpGray = RGB2GRAY(color.rgbRed, color.rgbGreen, color.rgbBlue); + if ( tmpGray < lowerc ) lowerc = tmpGray; + if ( tmpGray > upperd ) upperd = tmpGray; + } + } + if (upperd==lowerc) return false; + + for( y = ymin; y < ymax; y++ ){ + info.nProgress = (long)(50+50*y/ymax); + if (info.nEscape) break; + for( x = xmin; x < xmax; x++ ){ + + color = GetPixelColor( x, y ); + yuvClr = RGBtoYUV(color); + + // Stretch Luminance + tmpGray = (double)yuvClr.rgbRed; + stretcheds = (double)(tmpGray - lowerc) * ( (blimit - alimit) / (upperd - lowerc) ); // + alimit; + if ( stretcheds < 0.0 ) stretcheds = 0.0; + else if ( stretcheds > 255.0 ) stretcheds = 255.0; + yuvClr.rgbRed = (BYTE)stretcheds; + + color = YUVtoRGB(yuvClr); + SetPixelColor( x, y, color ); + } + } + } else { + DWORD j; + for( j = 0; j < head.biClrUsed; j++ ){ + color = GetPaletteColor( (BYTE)j ); + tmpGray = RGB2GRAY(color.rgbRed, color.rgbGreen, color.rgbBlue); + if ( tmpGray < lowerc ) lowerc = tmpGray; + if ( tmpGray > upperd ) upperd = tmpGray; + } + if (upperd==lowerc) return false; + + for( j = 0; j < head.biClrUsed; j++ ){ + + color = GetPaletteColor( (BYTE)j ); + yuvClr = RGBtoYUV( color ); + + // Stretch Luminance + tmpGray = (double)yuvClr.rgbRed; + stretcheds = (double)(tmpGray - lowerc) * ( (blimit - alimit) / (upperd - lowerc) ); // + alimit; + if ( stretcheds < 0.0 ) stretcheds = 0.0; + else if ( stretcheds > 255.0 ) stretcheds = 255.0; + yuvClr.rgbRed = (BYTE)stretcheds; + + color = YUVtoRGB(yuvClr); + SetPaletteColor( (BYTE)j, color ); + } + } + } + } + } + return true; +} +//////////////////////////////////////////////////////////////////////////////// +// HistogramEqualize function by : dave(at)posortho(dot)com +bool CxImage::HistogramEqualize() +{ + if (!pDib) return false; + + int histogram[256]; + int map[256]; + int equalize_map[256]; + int x, y, i, j; + RGBQUAD color; + RGBQUAD yuvClr; + unsigned int YVal, high, low; + + memset( &histogram, 0, sizeof(int) * 256 ); + memset( &map, 0, sizeof(int) * 256 ); + memset( &equalize_map, 0, sizeof(int) * 256 ); + + // form histogram + for(y=0; y < head.biHeight; y++){ + info.nProgress = (long)(50*y/head.biHeight); + if (info.nEscape) break; + for(x=0; x < head.biWidth; x++){ + color = GetPixelColor( x, y ); + YVal = (unsigned int)RGB2GRAY(color.rgbRed, color.rgbGreen, color.rgbBlue); + histogram[YVal]++; + } + } + + // integrate the histogram to get the equalization map. + j = 0; + for(i=0; i <= 255; i++){ + j += histogram[i]; + map[i] = j; + } + + // equalize + low = map[0]; + high = map[255]; + if (low == high) return false; + for( i = 0; i <= 255; i++ ){ + equalize_map[i] = (unsigned int)((((double)( map[i] - low ) ) * 255) / ( high - low ) ); + } + + // stretch the histogram + if(head.biClrUsed == 0){ // No Palette + for( y = 0; y < head.biHeight; y++ ){ + info.nProgress = (long)(50+50*y/head.biHeight); + if (info.nEscape) break; + for( x = 0; x < head.biWidth; x++ ){ + + color = GetPixelColor( x, y ); + yuvClr = RGBtoYUV(color); + + yuvClr.rgbRed = (BYTE)equalize_map[yuvClr.rgbRed]; + + color = YUVtoRGB(yuvClr); + SetPixelColor( x, y, color ); + } + } + } else { // Palette + for( i = 0; i < (int)head.biClrUsed; i++ ){ + + color = GetPaletteColor((BYTE)i); + yuvClr = RGBtoYUV(color); + + yuvClr.rgbRed = (BYTE)equalize_map[yuvClr.rgbRed]; + + color = YUVtoRGB(yuvClr); + SetPaletteColor( (BYTE)i, color ); + } + } + return true; +} +//////////////////////////////////////////////////////////////////////////////// +// HistogramNormalize function by : dave(at)posortho(dot)com +bool CxImage::HistogramNormalize() +{ + if (!pDib) return false; + + int histogram[256]; + int threshold_intensity, intense; + int x, y, i; + unsigned int normalize_map[256]; + unsigned int high, low, YVal; + + RGBQUAD color; + RGBQUAD yuvClr; + + memset( &histogram, 0, sizeof( int ) * 256 ); + memset( &normalize_map, 0, sizeof( unsigned int ) * 256 ); + + // form histogram + for(y=0; y < head.biHeight; y++){ + info.nProgress = (long)(50*y/head.biHeight); + if (info.nEscape) break; + for(x=0; x < head.biWidth; x++){ + color = GetPixelColor( x, y ); + YVal = (unsigned int)RGB2GRAY(color.rgbRed, color.rgbGreen, color.rgbBlue); + histogram[YVal]++; + } + } + + // find histogram boundaries by locating the 1 percent levels + threshold_intensity = ( head.biWidth * head.biHeight) / 100; + + intense = 0; + for( low = 0; low < 255; low++ ){ + intense += histogram[low]; + if( intense > threshold_intensity ) break; + } + + intense = 0; + for( high = 255; high != 0; high--){ + intense += histogram[ high ]; + if( intense > threshold_intensity ) break; + } + + if ( low == high ){ + // Unreasonable contrast; use zero threshold to determine boundaries. + threshold_intensity = 0; + intense = 0; + for( low = 0; low < 255; low++){ + intense += histogram[low]; + if( intense > threshold_intensity ) break; + } + intense = 0; + for( high = 255; high != 0; high-- ){ + intense += histogram [high ]; + if( intense > threshold_intensity ) break; + } + } + if( low == high ) return false; // zero span bound + + // Stretch the histogram to create the normalized image mapping. + for(i = 0; i <= 255; i++){ + if ( i < (int) low ){ + normalize_map[i] = 0; + } else { + if(i > (int) high) + normalize_map[i] = 255; + else + normalize_map[i] = ( 255 - 1) * ( i - low) / ( high - low ); + } + } + + // Normalize + if( head.biClrUsed == 0 ){ + for( y = 0; y < head.biHeight; y++ ){ + info.nProgress = (long)(50+50*y/head.biHeight); + if (info.nEscape) break; + for( x = 0; x < head.biWidth; x++ ){ + + color = GetPixelColor( x, y ); + yuvClr = RGBtoYUV( color ); + + yuvClr.rgbRed = (BYTE)normalize_map[yuvClr.rgbRed]; + + color = YUVtoRGB( yuvClr ); + SetPixelColor( x, y, color ); + } + } + } else { + for(i = 0; i < (int)head.biClrUsed; i++){ + + color = GetPaletteColor( (BYTE)i ); + yuvClr = RGBtoYUV( color ); + + yuvClr.rgbRed = (BYTE)normalize_map[yuvClr.rgbRed]; + + color = YUVtoRGB( yuvClr ); + SetPaletteColor( (BYTE)i, color ); + } + } + return true; +} +//////////////////////////////////////////////////////////////////////////////// +// HistogramLog function by : dave(at)posortho(dot)com +bool CxImage::HistogramLog() +{ + if (!pDib) return false; + + //q(i,j) = 255/log(1 + |high|) * log(1 + |p(i,j)|); + int x, y, i; + RGBQUAD color; + RGBQUAD yuvClr; + + unsigned int YVal, high = 1; + + // Find Highest Luminance Value in the Image + if( head.biClrUsed == 0 ){ // No Palette + for(y=0; y < head.biHeight; y++){ + info.nProgress = (long)(50*y/head.biHeight); + if (info.nEscape) break; + for(x=0; x < head.biWidth; x++){ + color = GetPixelColor( x, y ); + YVal = (unsigned int)RGB2GRAY(color.rgbRed, color.rgbGreen, color.rgbBlue); + if (YVal > high ) high = YVal; + } + } + } else { // Palette + for(i = 0; i < (int)head.biClrUsed; i++){ + color = GetPaletteColor((BYTE)i); + YVal = (unsigned int)RGB2GRAY(color.rgbRed, color.rgbGreen, color.rgbBlue); + if (YVal > high ) high = YVal; + } + } + + // Logarithm Operator + double k = 255.0 / ::log( 1.0 + (double)high ); + if( head.biClrUsed == 0 ){ + for( y = 0; y < head.biHeight; y++ ){ + info.nProgress = (long)(50+50*y/head.biHeight); + if (info.nEscape) break; + for( x = 0; x < head.biWidth; x++ ){ + + color = GetPixelColor( x, y ); + yuvClr = RGBtoYUV( color ); + + yuvClr.rgbRed = (BYTE)(k * ::log( 1.0 + (double)yuvClr.rgbRed ) ); + + color = YUVtoRGB( yuvClr ); + SetPixelColor( x, y, color ); + } + } + } else { + for(i = 0; i < (int)head.biClrUsed; i++){ + + color = GetPaletteColor( (BYTE)i ); + yuvClr = RGBtoYUV( color ); + + yuvClr.rgbRed = (BYTE)(k * ::log( 1.0 + (double)yuvClr.rgbRed ) ); + + color = YUVtoRGB( yuvClr ); + SetPaletteColor( (BYTE)i, color ); + } + } + + return true; +} + +//////////////////////////////////////////////////////////////////////////////// +// HistogramRoot function by : dave(at)posortho(dot)com +bool CxImage::HistogramRoot() +{ + if (!pDib) return false; + //q(i,j) = sqrt(|p(i,j)|); + + int x, y, i; + RGBQUAD color; + RGBQUAD yuvClr; + double dtmp; + unsigned int YVal, high = 1; + + // Find Highest Luminance Value in the Image + if( head.biClrUsed == 0 ){ // No Palette + for(y=0; y < head.biHeight; y++){ + info.nProgress = (long)(50*y/head.biHeight); + if (info.nEscape) break; + for(x=0; x < head.biWidth; x++){ + color = GetPixelColor( x, y ); + YVal = (unsigned int)RGB2GRAY(color.rgbRed, color.rgbGreen, color.rgbBlue); + if (YVal > high ) high = YVal; + } + } + } else { // Palette + for(i = 0; i < (int)head.biClrUsed; i++){ + color = GetPaletteColor((BYTE)i); + YVal = (unsigned int)RGB2GRAY(color.rgbRed, color.rgbGreen, color.rgbBlue); + if (YVal > high ) high = YVal; + } + } + + // Root Operator + double k = 128.0 / ::log( 1.0 + (double)high ); + if( head.biClrUsed == 0 ){ + for( y = 0; y < head.biHeight; y++ ){ + info.nProgress = (long)(50+50*y/head.biHeight); + if (info.nEscape) break; + for( x = 0; x < head.biWidth; x++ ){ + + color = GetPixelColor( x, y ); + yuvClr = RGBtoYUV( color ); + + dtmp = k * ::sqrt( (double)yuvClr.rgbRed ); + if ( dtmp > 255.0 ) dtmp = 255.0; + if ( dtmp < 0 ) dtmp = 0; + yuvClr.rgbRed = (BYTE)dtmp; + + color = YUVtoRGB( yuvClr ); + SetPixelColor( x, y, color ); + } + } + } else { + for(i = 0; i < (int)head.biClrUsed; i++){ + + color = GetPaletteColor( (BYTE)i ); + yuvClr = RGBtoYUV( color ); + + dtmp = k * ::sqrt( (double)yuvClr.rgbRed ); + if ( dtmp > 255.0 ) dtmp = 255.0; + if ( dtmp < 0 ) dtmp = 0; + yuvClr.rgbRed = (BYTE)dtmp; + + color = YUVtoRGB( yuvClr ); + SetPaletteColor( (BYTE)i, color ); + } + } + + return true; +} +//////////////////////////////////////////////////////////////////////////////// +#endif diff --git a/src/win32/dependencies/cximage/ximaico.cpp b/src/win32/dependencies/cximage/ximaico.cpp new file mode 100644 index 00000000..4523ccf3 --- /dev/null +++ b/src/win32/dependencies/cximage/ximaico.cpp @@ -0,0 +1,299 @@ +/* + * File: ximaico.cpp + * Purpose: Platform Independent ICON Image Class Loader and Writer (MS version) + * 07/Aug/2001 Davide Pizzolato - www.xdp.it + * CxImage version 5.99c 17/Oct/2004 + */ + +#include "ximaico.h" + +#if CXIMAGE_SUPPORT_ICO + +//////////////////////////////////////////////////////////////////////////////// +bool CxImageICO::Decode(CxFile *hFile) +{ + if (hFile==NULL) return false; + + DWORD off = hFile->Tell(); // + int page=info.nFrame; //internal icon structure indexes + + // read the first part of the header + ICONHEADER icon_header; + hFile->Read(&icon_header,sizeof(ICONHEADER),1); + // check if it's an icon or a cursor + if ((icon_header.idReserved == 0) && ((icon_header.idType == 1)||(icon_header.idType == 2))) { + + info.nNumFrames = icon_header.idCount; + + // load the icon descriptions + ICONDIRENTRY *icon_list = (ICONDIRENTRY *)malloc(icon_header.idCount * sizeof(ICONDIRENTRY)); + int c; + for (c = 0; c < icon_header.idCount; c++) + hFile->Read(icon_list + c, sizeof(ICONDIRENTRY), 1); + + if ((page>=0)&&(page + BITMAPINFOHEADER bih; + hFile->Seek(off + icon_list[page].dwImageOffset, SEEK_SET); + hFile->Read(&bih,sizeof(BITMAPINFOHEADER),1); + c = bih.biBitCount; + + // allocate memory for one icon + Create(icon_list[page].bWidth,icon_list[page].bHeight, c, CXIMAGE_FORMAT_ICO); //image creation + + // read the palette + RGBQUAD pal[256]; + hFile->Read(pal,head.biClrUsed*sizeof(RGBQUAD), 1); + SetPalette(pal,head.biClrUsed); //palette assign + + //read the icon + if (c<=24){ + hFile->Read(info.pImage, head.biSizeImage, 1); + } else { // 32 bit icon + BYTE* dst = info.pImage; + BYTE* buf=(BYTE*)malloc(4*head.biHeight*head.biWidth); + BYTE* src = buf; + hFile->Read(buf, 4*head.biHeight*head.biWidth, 1); +#if CXIMAGE_SUPPORT_ALPHA + if (!AlphaIsValid()) AlphaCreate(); +#endif //CXIMAGE_SUPPORT_ALPHA + for (long y = 0; y < head.biHeight; y++) { + for(long x=0;xRead(mask, masksize, 1)){ + + bool bGoodMask=false; + for (int im=0;im>3)]>>(7-x%8))&0x01)){ + AlphaSet(x,y,0); + bNeedAlpha=true; + } + } + } + if (!bNeedAlpha) AlphaDelete(); +#endif //CXIMAGE_SUPPORT_ALPHA + + if (c==24){ //check if there is only one transparent color + RGBQUAD cc,ct; + long* pcc = (long*)&cc; + long* pct = (long*)&ct; + int nTransColors=0; + for (int y = 0; y < head.biHeight; y++){ + for (int x = 0; x < head.biWidth; x++){ + if (((mask[y*maskwdt+(x>>3)] >> (7-x%8)) & 0x01)){ + cc = GetPixelColor(x,y,false); + if (nTransColors==0){ + nTransColors++; + ct = cc; + } else { + if (*pct!=*pcc){ + nTransColors++; + } + } + } + } + } + if (nTransColors==1){ + SetTransColor(ct); + SetTransIndex(0); +#if CXIMAGE_SUPPORT_ALPHA + AlphaDelete(); //because we have a unique transparent color in the image +#endif //CXIMAGE_SUPPORT_ALPHA + } + } + + // - Transparency support w/o Alpha support + if (c <= 8){ // only for icons with less than 256 colors (XP icons need alpha). + + // find a color index, which is not used in the image + // it is almost sure to find one, bcs. nobody uses all possible colors for an icon + + BYTE colorsUsed[256]; + memset(colorsUsed, 0, sizeof(colorsUsed)); + + for (int y = 0; y < head.biHeight; y++){ + for (int x = 0; x < head.biWidth; x++){ + colorsUsed[GetPixelIndex(x,y)] = 1; + } + } + + int iTransIdx = -1; + for (int x = 0; x < (int)head.biClrUsed; x++){ + if (colorsUsed[x] == 0){ + iTransIdx = x; // this one is not in use. we may use it as transparent color + break; + } + } + + // Go thru image and set unused color as transparent index if needed + if (iTransIdx >= 0){ + bool bNeedTrans = false; + for (int y = 0; y < head.biHeight; y++){ + for (int x = 0; x < head.biWidth; x++){ + // AND mask (Each Byte represents 8 Pixels) + if (((mask[y*maskwdt+(x>>3)] >> (7-x%8)) & 0x01)){ + // AND mask is set (!=0). This is a transparent part + SetPixelIndex(x, y, iTransIdx); + bNeedTrans = true; + } + } + } + // set transparent index if needed + if (bNeedTrans) SetTransIndex(iTransIdx); +#if CXIMAGE_SUPPORT_ALPHA + AlphaDelete(); //because we have a transparent color in the palette +#endif //CXIMAGE_SUPPORT_ALPHA + } + } + } else { + SetTransIndex(0); //empty mask, set black as transparent color + Negative(); + } + } + free(mask); + + free(icon_list); + // icon has been loaded successfully! + return true; + } + free(icon_list); + } + return false; +} +//////////////////////////////////////////////////////////////////////////////// +#if CXIMAGE_SUPPORT_ENCODE +//////////////////////////////////////////////////////////////////////////////// +bool CxImageICO::Encode(CxFile * hFile) +{ + if (EncodeSafeCheck(hFile)) return false; + + //check format limits + if ((head.biWidth>255)||(head.biHeight>255)){ + strcpy(info.szLastError,"Can't save this image as icon"); + return false; + } + + //prepare the palette struct + RGBQUAD* pal=GetPalette(); + if (head.biBitCount<=8 && pal==NULL) return false; + + int maskwdt=((head.biWidth+31)/32)*4; //mask line width + int masksize=head.biHeight * maskwdt; //size of mask + int bitcount=head.biBitCount; + int imagesize=head.biSizeImage; +#if CXIMAGE_SUPPORT_ALPHA + if (AlphaIsValid() && head.biClrUsed==0){ + bitcount=32; + imagesize=4*head.biHeight*head.biWidth; + } +#endif + + //fill the icon headers + ICONHEADER icon_header={0,1,1}; + ICONDIRENTRY icon_list={(BYTE)head.biWidth,(BYTE)head.biHeight,(BYTE)head.biClrUsed ,0,0,(WORD)bitcount, + sizeof(BITMAPINFOHEADER)+head.biClrUsed*sizeof(RGBQUAD)+ + imagesize+masksize, + sizeof(ICONHEADER)+sizeof(ICONDIRENTRY)}; + BITMAPINFOHEADER bi={sizeof(BITMAPINFOHEADER),head.biWidth,2*head.biHeight,1,(WORD)bitcount, + 0,imagesize,0,0,0,0}; + + hFile->Write(&icon_header,sizeof(ICONHEADER),1); //write the headers + hFile->Write(&icon_list,sizeof(ICONDIRENTRY),1); + hFile->Write(&bi,sizeof(BITMAPINFOHEADER),1); + if (pal) hFile->Write(pal,head.biClrUsed*sizeof(RGBQUAD),1); //write palette + +#if CXIMAGE_SUPPORT_ALPHA + if (AlphaIsValid() && head.biClrUsed==0){ + BYTE* src = info.pImage; + BYTE* buf=(BYTE*)malloc(imagesize); + BYTE* dst = buf; + for (long y = 0; y < head.biHeight; y++) { + for(long x=0;xWrite(buf,imagesize, 1); + free(buf); + } else { + hFile->Write(info.pImage,imagesize,1); //write image + } +#else + hFile->Write(info.pImage,imagesize,1); //write image +#endif + + //save transparency mask + BYTE* mask=(BYTE*)calloc(masksize,1); //create empty AND/XOR masks + if (!mask) return false; + + //prepare the variables to build the mask + BYTE* iDst; + int pos,i; + RGBQUAD c={0,0,0,0}; + RGBQUAD ct = GetTransColor(); + long* pc = (long*)&c; + long* pct= (long*)&ct; + bool bTransparent = info.nBkgndIndex != -1; +#if CXIMAGE_SUPPORT_ALPHA + bool bAlphaPaletteIsValid = AlphaPaletteIsValid(); + bool bAlphaIsValid = AlphaIsValid(); +#endif + //build the mask + for (int y = 0; y < head.biHeight; y++) { + for (int x = 0; x < head.biWidth; x++) { + i=0; +#if CXIMAGE_SUPPORT_ALPHA + if (bAlphaIsValid && AlphaGet(x,y)==0) i=1; + if (bAlphaPaletteIsValid && GetPixelColor(x,y).rgbReserved==0) i=1; +#endif + c=GetPixelColor(x,y,false); + if (bTransparent && *pc==*pct) i=1; + iDst = mask + y*maskwdt + (x>>3); + pos = 7-x%8; + *iDst &= ~(0x01<Write(mask,masksize,1); + free(mask); + return true; +} +//////////////////////////////////////////////////////////////////////////////// +#endif // CXIMAGE_SUPPORT_ENCODE +//////////////////////////////////////////////////////////////////////////////// +#endif // CXIMAGE_SUPPORT_ICO + diff --git a/src/win32/dependencies/cximage/ximaico.h b/src/win32/dependencies/cximage/ximaico.h new file mode 100644 index 00000000..8f63e060 --- /dev/null +++ b/src/win32/dependencies/cximage/ximaico.h @@ -0,0 +1,52 @@ +/* + * File: ximaico.h + * Purpose: ICON Image Class Loader and Writer + */ +/* ========================================================== + * CxImageICO (c) 07/Aug/2001 Davide Pizzolato - www.xdp.it + * For conditions of distribution and use, see copyright notice in ximage.h + * ========================================================== + */ +#if !defined(__ximaICO_h) +#define __ximaICO_h + +#include "ximage.h" + +#if CXIMAGE_SUPPORT_ICO + +class CxImageICO: public CxImage +{ +typedef struct tagIconDirectoryEntry { + BYTE bWidth; + BYTE bHeight; + BYTE bColorCount; + BYTE bReserved; + WORD wPlanes; + WORD wBitCount; + DWORD dwBytesInRes; + DWORD dwImageOffset; +} ICONDIRENTRY; + +typedef struct tagIconDir { + WORD idReserved; + WORD idType; + WORD idCount; +} ICONHEADER; + +public: + CxImageICO(): CxImage(CXIMAGE_FORMAT_ICO) {} + +// bool Load(const char * imageFileName){ return CxImage::Load(imageFileName,CXIMAGE_FORMAT_ICO);} +// bool Save(const char * imageFileName){ return CxImage::Save(imageFileName,CXIMAGE_FORMAT_ICO);} + bool Decode(CxFile * hFile); + bool Decode(FILE *hFile) { CxIOFile file(hFile); return Decode(&file); } + +#if CXIMAGE_SUPPORT_ENCODE + bool Encode(CxFile * hFile); + bool Encode(FILE *hFile) { CxIOFile file(hFile); return Encode(&file); } +#endif // CXIMAGE_SUPPORT_ENCODE +}; + +#endif + +#endif diff --git a/src/win32/dependencies/cximage/ximainfo.cpp b/src/win32/dependencies/cximage/ximainfo.cpp new file mode 100644 index 00000000..6535ddea --- /dev/null +++ b/src/win32/dependencies/cximage/ximainfo.cpp @@ -0,0 +1,388 @@ +// ximainfo.cpp : main attributes +/* 03/10/2004 v1.00 - Davide Pizzolato - www.xdp.it + * CxImage version 5.99c 17/Oct/2004 + */ + +#include "ximage.h" + +//////////////////////////////////////////////////////////////////////////////// +/** + * \return the color used for transparency, and/or for background color + */ +RGBQUAD CxImage::GetTransColor() +{ + if (head.biBitCount<24 && info.nBkgndIndex != -1) return GetPaletteColor((BYTE)info.nBkgndIndex); + return info.nBkgndColor; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Gets the index used for transparency. Returns -1 for no transparancy. + */ +long CxImage::GetTransIndex() const +{ + return info.nBkgndIndex; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Sets the index used for transparency with 1, 4 and 8 bpp images. Set to -1 to remove the effect. + */ +void CxImage::SetTransIndex(long idx) +{ + info.nBkgndIndex = idx; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Sets the color used for transparency with 24 bpp images. + * You must call SetTransIndex(0) to enable the effect, SetTransIndex(-1) to disable it. + */ +void CxImage::SetTransColor(RGBQUAD rgb) +{ + rgb.rgbReserved=0; + info.nBkgndColor = rgb; +} +//////////////////////////////////////////////////////////////////////////////// +bool CxImage::IsTransparent() const +{ + return info.nBkgndIndex>=0; // +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Returns true if the image has 256 colors or less. + */ +bool CxImage::IsIndexed() const +{ + return head.biClrUsed!=0; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * \return 1 = indexed, 2 = RGB, 4 = RGBA + */ +BYTE CxImage::GetColorType() +{ + BYTE b = (BYTE)((head.biBitCount>8) ? 2 /*COLORTYPE_COLOR*/ : 1 /*COLORTYPE_PALETTE*/); +#if CXIMAGE_SUPPORT_ALPHA + if (AlphaIsValid()) b = 4 /*COLORTYPE_ALPHA*/; +#endif //CXIMAGE_SUPPORT_ALPHA + return b; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * \return Resolution for TIFF, JPEG, PNG and BMP formats. + */ +long CxImage::GetXDPI() const +{ + return info.xDPI; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * \return Resolution for TIFF, JPEG, PNG and BMP formats. + */ +long CxImage::GetYDPI() const +{ + return info.yDPI; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Set resolution for TIFF, JPEG, PNG and BMP formats. + */ +void CxImage::SetXDPI(long dpi) +{ + if (dpi<=0) dpi=96; + info.xDPI = dpi; + head.biXPelsPerMeter = (long) floor(dpi * 10000.0 / 254.0 + 0.5); +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Set resolution for TIFF, JPEG, PNG and BMP formats. + */ +void CxImage::SetYDPI(long dpi) +{ + if (dpi<=0) dpi=96; + info.yDPI = dpi; + head.biYPelsPerMeter = (long) floor(dpi * 10000.0 / 254.0 + 0.5); +} +//////////////////////////////////////////////////////////////////////////////// +/** + * \sa SetFlags + */ +DWORD CxImage::GetFlags() const +{ + return info.dwFlags; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Image flags, for future use + * \param flags + * - 0x??00000 = reserved for 16 bit, CMYK, multilayer + * - 0x00??0000 = blend modes + * - 0x0000???? = layer id or user flags + * + * \param bLockReservedFlags protects the "reserved" and "blend modes" flags + */ +void CxImage::SetFlags(DWORD flags, bool bLockReservedFlags) +{ + if (bLockReservedFlags) info.dwFlags = flags & 0x0000ffff; + else info.dwFlags = flags; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * \sa SetCodecOption + */ +DWORD CxImage::GetCodecOption(DWORD imagetype) +{ + if (imagetype=head.biWidth) x=head.biWidth-(x % head.biWidth + 1); + if (y<0) y=((-y) % head.biHeight); + else if (y>=head.biHeight) y=head.biHeight-(y % head.biHeight + 1); + break; + default: + return; + }//switch +} + +//////////////////////////////////////////////////////////////////////////////// +/** + * See OverflowCoordinates for integer version + * \author ***bd*** 2.2004 + */ +void CxImage::OverflowCoordinates(float &x, float &y, OverflowMethod const ofMethod) +{ + if (x>=0 && x=0 && y=head.biWidth) x=head.biWidth-((float)fmod(x, (float) head.biWidth) + 1); + if (y<0) y=(float)fmod(-y, (float) head.biHeight); + else if (y>=head.biHeight) y=head.biHeight-((float)fmod(y, (float) head.biHeight) + 1); + break; + default: + return; + }//switch +} + +//////////////////////////////////////////////////////////////////////////////// +/** + * Method return pixel color. Different methods are implemented for out of bounds pixels. + * If an image has alpha channel, alpha value is returned in .RGBReserved. + * + * \param x,y : pixel coordinates + * \param ofMethod : out-of-bounds method: + * - OF_WRAP - wrap over to pixels on other side of the image + * - OF_REPEAT - repeat last pixel on the edge + * - OF_COLOR - return input value of color + * - OF_BACKGROUND - return background color (if not set, return input color) + * - OF_TRANSPARENT - return transparent pixel + * + * \param rplColor : input color (returned for out-of-bound coordinates in OF_COLOR mode and if other mode is not applicable) + * + * \return color : color of pixel + * \author ***bd*** 2.2004 + */ +RGBQUAD CxImage::GetPixelColorWithOverflow(long x, long y, OverflowMethod const ofMethod, RGBQUAD* const rplColor) +{ + RGBQUAD color; //color to return + if ((!IsInside(x,y)) || pDib==NULL) { //is pixel within bouns?: + //pixel is out of bounds or no DIB + if (rplColor!=NULL) + color=*rplColor; + else { + color.rgbRed=color.rgbGreen=color.rgbBlue=255; color.rgbReserved=0; //default replacement colour: white transparent + }//if + if (pDib==NULL) return color; + //pixel is out of bounds: + switch (ofMethod) { + case OM_TRANSPARENT: +#if CXIMAGE_SUPPORT_ALPHA + if (AlphaIsValid()) { + //alpha transparency is supported and image has alpha layer + color.rgbReserved=0; + } else { +#endif //CXIMAGE_SUPPORT_ALPHA + //no alpha transparency + if (GetTransIndex()>=0) { + color=GetTransColor(); //single color transparency enabled (return transparent color) + }//if +#if CXIMAGE_SUPPORT_ALPHA + }//if +#endif //CXIMAGE_SUPPORT_ALPHA + return color; + case OM_BACKGROUND: + //return background color (if it exists, otherwise input value) + if (info.nBkgndIndex != -1) { + if (head.biBitCount<24) color = GetPaletteColor((BYTE)info.nBkgndIndex); + else color = info.nBkgndColor; + }//if + return color; + case OM_REPEAT: + case OM_WRAP: + case OM_MIRROR: + OverflowCoordinates(x,y,ofMethod); + break; + default: + //simply return replacement color (OM_COLOR and others) + return color; + }//switch + }//if + //just return specified pixel (it's within bounds) + return BlindGetPixelColor(x,y); +} + +//////////////////////////////////////////////////////////////////////////////// +/** + * This method reconstructs image according to chosen interpolation method and then returns pixel (x,y). + * (x,y) can lie between actual image pixels. If (x,y) lies outside of image, method returns value + * according to overflow method. + * This method is very useful for geometrical image transformations, where destination pixel + * can often assume color value lying between source pixels. + * + * \param (x,y) - coordinates of pixel to return + * GPCI method recreates "analogue" image back from digital data, so x and y + * are float values and color value of point (1.1,1) will generally not be same + * as (1,1). Center of first pixel is at (0,0) and center of pixel right to it is (1,0). + * (0.5,0) is half way between these two pixels. + * \param inMethod - interpolation (reconstruction) method (kernel) to use: + * - IM_NEAREST_NEIGHBOUR - returns colour of nearest lying pixel (causes stairy look of + * processed images) + * - IM_BILINEAR - interpolates colour from four neighbouring pixels (softens image a bit) + * - IM_BICUBIC - interpolates from 16 neighbouring pixels (can produce "halo" artifacts) + * - IM_BICUBIC2 - interpolates from 16 neighbouring pixels (perhaps a bit less halo artifacts + than IM_BICUBIC) + * - IM_BSPLINE - interpolates from 16 neighbouring pixels (softens image, washes colours) + * (As far as I know, image should be prefiltered for this method to give + * good results... some other time :) ) + * This method uses bicubic interpolation kernel from CXImage 5.99a and older + * versions. + * - IM_LANCZOS - interpolates from 12*12 pixels (slow, ringing artifacts) + * + * \param ofMethod - overflow method (see comments at GetPixelColorWithOverflow) + * \param rplColor - pointer to color used for out of borders pixels in OM_COLOR mode + * (and other modes if colour can't calculated in a specified way) + * + * \return interpolated color value (including interpolated alpha value, if image has alpha layer) + * + * \author ***bd*** 2.2004 + */ +RGBQUAD CxImage::GetPixelColorInterpolated( + float x,float y, + InterpolationMethod const inMethod, + OverflowMethod const ofMethod, + RGBQUAD* const rplColor) +{ + //calculate nearest pixel + int xi=(int)(x); if (x<0) xi--; //these replace (incredibly slow) floor (Visual c++ 2003, AMD Athlon) + int yi=(int)(y); if (y<0) yi--; + RGBQUAD color; //calculated colour + + switch (inMethod) { + case IM_NEAREST_NEIGHBOUR: + return GetPixelColorWithOverflow((long)(x+0.5f), (long)(y+0.5f), ofMethod, rplColor); + default: { + //bilinear interpolation + if (xi<-1 || xi>=head.biWidth || yi<-1 || yi>=head.biHeight) { //all 4 points are outside bounds?: + switch (ofMethod) { + case OM_COLOR: case OM_TRANSPARENT: case OM_BACKGROUND: + //we don't need to interpolate anything with all points outside in this case + return GetPixelColorWithOverflow(-999, -999, ofMethod, rplColor); + default: + //recalculate coordinates and use faster method later on + OverflowCoordinates(x,y,ofMethod); + xi=(int)(x); if (x<0) xi--; //x and/or y have changed ... recalculate xi and yi + yi=(int)(y); if (y<0) yi--; + }//switch + }//if + //get four neighbouring pixels + if ((xi+1)=0 && (yi+1)=0 && head.biClrUsed==0) { + //all pixels are inside RGB24 image... optimize reading (and use fixed point arithmetic) + WORD wt1=(WORD)((x-xi)*256.0f), wt2=(WORD)((y-yi)*256.0f); + WORD wd=wt1*wt2>>8; + WORD wb=wt1-wd; + WORD wc=wt2-wd; + WORD wa=256-wt1-wc; + WORD wrr,wgg,wbb; + BYTE *pxptr=(BYTE*)info.pImage+yi*info.dwEffWidth+xi*3; + wbb=wa*(*pxptr++); wgg=wa*(*pxptr++); wrr=wa*(*pxptr++); + wbb+=wb*(*pxptr++); wgg+=wb*(*pxptr++); wrr+=wb*(*pxptr); + pxptr+=(info.dwEffWidth-5); //move to next row + wbb+=wc*(*pxptr++); wgg+=wc*(*pxptr++); wrr+=wc*(*pxptr++); + wbb+=wd*(*pxptr++); wgg+=wd*(*pxptr++); wrr+=wd*(*pxptr); + color.rgbRed=(BYTE) (wrr>>8); color.rgbGreen=(BYTE) (wgg>>8); color.rgbBlue=(BYTE) (wbb>>8); +#if CXIMAGE_SUPPORT_ALPHA + if (pAlpha) { + WORD waa; + //image has alpha layer... we have to do the same for alpha data + pxptr=AlphaGetPointer(xi,yi); //pointer to first byte + waa=wa*(*pxptr++); waa+=wb*(*pxptr); //first two pixels + pxptr+=(head.biWidth-1); //move to next row + waa+=wc*(*pxptr++); waa+=wd*(*pxptr); //and second row pixels + color.rgbReserved=(BYTE) (waa>>8); + } else +#endif + { //Alpha not supported or no alpha at all + color.rgbReserved = 0; + } + return color; + } else { + //default (slower) way to get pixels (not RGB24 or some pixels out of borders) + float t1=x-xi, t2=y-yi; + float d=t1*t2; + float b=t1-d; + float c=t2-d; + float a=1-t1-c; + RGBQUAD rgb11,rgb21,rgb12,rgb22; + rgb11=GetPixelColorWithOverflow(xi, yi, ofMethod, rplColor); + rgb21=GetPixelColorWithOverflow(xi+1, yi, ofMethod, rplColor); + rgb12=GetPixelColorWithOverflow(xi, yi+1, ofMethod, rplColor); + rgb22=GetPixelColorWithOverflow(xi+1, yi+1, ofMethod, rplColor); + //calculate linear interpolation + color.rgbRed=(BYTE) (a*rgb11.rgbRed+b*rgb21.rgbRed+c*rgb12.rgbRed+d*rgb22.rgbRed); + color.rgbGreen=(BYTE) (a*rgb11.rgbGreen+b*rgb21.rgbGreen+c*rgb12.rgbGreen+d*rgb22.rgbGreen); + color.rgbBlue=(BYTE) (a*rgb11.rgbBlue+b*rgb21.rgbBlue+c*rgb12.rgbBlue+d*rgb22.rgbBlue); +#if CXIMAGE_SUPPORT_ALPHA + if (AlphaIsValid()) + color.rgbReserved=(BYTE) (a*rgb11.rgbReserved+b*rgb21.rgbReserved+c*rgb12.rgbReserved+d*rgb22.rgbReserved); + else +#endif + { //Alpha not supported or no alpha at all + color.rgbReserved = 0; + } + return color; + }//if + }//default + case IM_BICUBIC: + case IM_BICUBIC2: + case IM_BSPLINE: + case IM_BOX: + case IM_HERMITE: + case IM_HAMMING: + case IM_SINC: + case IM_BLACKMAN: + case IM_BESSEL: + case IM_GAUSSIAN: + case IM_QUADRATIC: + case IM_MITCHELL: + case IM_CATROM: + //bicubic interpolation(s) + if (((xi+2)<0) || ((xi-1)>=head.biWidth) || ((yi+2)<0) || ((yi-1)>=head.biHeight)) { //all points are outside bounds?: + switch (ofMethod) { + case OM_COLOR: case OM_TRANSPARENT: case OM_BACKGROUND: + //we don't need to interpolate anything with all points outside in this case + return GetPixelColorWithOverflow(-999, -999, ofMethod, rplColor); + break; + default: + //recalculate coordinates and use faster method later on + OverflowCoordinates(x,y,ofMethod); + xi=(int)(x); if (x<0) xi--; //x and/or y have changed ... recalculate xi and yi + yi=(int)(y); if (y<0) yi--; + }//switch + }//if + + //some variables needed from here on + int xii,yii; //x any y integer indexes for loops + float kernel, kernelyc; //kernel cache + float kernelx[12], kernely[4]; //precalculated kernel values + float rr,gg,bb,aa; //accumulated color values + //calculate multiplication factors for all pixels + int i; + switch (inMethod) { + case IM_BICUBIC: + for (i=0; i<4; i++) { + kernelx[i]=KernelCubic((float)(xi+i-1-x)); + kernely[i]=KernelCubic((float)(yi+i-1-y)); + }//for i + break; + case IM_BICUBIC2: + for (i=0; i<4; i++) { + kernelx[i]=KernelGeneralizedCubic((float)(xi+i-1-x), -0.5); + kernely[i]=KernelGeneralizedCubic((float)(yi+i-1-y), -0.5); + }//for i + break; + case IM_BSPLINE: + for (i=0; i<4; i++) { + kernelx[i]=KernelBSpline((float)(xi+i-1-x)); + kernely[i]=KernelBSpline((float)(yi+i-1-y)); + }//for i + break; + case IM_BOX: + for (i=0; i<4; i++) { + kernelx[i]=KernelBox((float)(xi+i-1-x)); + kernely[i]=KernelBox((float)(yi+i-1-y)); + }//for i + break; + case IM_HERMITE: + for (i=0; i<4; i++) { + kernelx[i]=KernelHermite((float)(xi+i-1-x)); + kernely[i]=KernelHermite((float)(yi+i-1-y)); + }//for i + break; + case IM_HAMMING: + for (i=0; i<4; i++) { + kernelx[i]=KernelHamming((float)(xi+i-1-x)); + kernely[i]=KernelHamming((float)(yi+i-1-y)); + }//for i + break; + case IM_SINC: + for (i=0; i<4; i++) { + kernelx[i]=KernelSinc((float)(xi+i-1-x)); + kernely[i]=KernelSinc((float)(yi+i-1-y)); + }//for i + break; + case IM_BLACKMAN: + for (i=0; i<4; i++) { + kernelx[i]=KernelBlackman((float)(xi+i-1-x)); + kernely[i]=KernelBlackman((float)(yi+i-1-y)); + }//for i + break; + case IM_BESSEL: + for (i=0; i<4; i++) { + kernelx[i]=KernelBessel((float)(xi+i-1-x)); + kernely[i]=KernelBessel((float)(yi+i-1-y)); + }//for i + break; + case IM_GAUSSIAN: + for (i=0; i<4; i++) { + kernelx[i]=KernelGaussian((float)(xi+i-1-x)); + kernely[i]=KernelGaussian((float)(yi+i-1-y)); + }//for i + break; + case IM_QUADRATIC: + for (i=0; i<4; i++) { + kernelx[i]=KernelQuadratic((float)(xi+i-1-x)); + kernely[i]=KernelQuadratic((float)(yi+i-1-y)); + }//for i + break; + case IM_MITCHELL: + for (i=0; i<4; i++) { + kernelx[i]=KernelMitchell((float)(xi+i-1-x)); + kernely[i]=KernelMitchell((float)(yi+i-1-y)); + }//for i + break; + case IM_CATROM: + for (i=0; i<4; i++) { + kernelx[i]=KernelCatrom((float)(xi+i-1-x)); + kernely[i]=KernelCatrom((float)(yi+i-1-y)); + }//for i + break; + }//switch + rr=gg=bb=aa=0; + if (((xi+2)=1 && ((yi+2)=1) && !IsIndexed()) { + //optimized interpolation (faster pixel reads) for RGB24 images with all pixels inside bounds + BYTE *pxptr, *pxptra; + for (yii=yi-1; yii255) rr=255; if (rr<0) rr=0; color.rgbRed=(BYTE) rr; + if (gg>255) gg=255; if (gg<0) gg=0; color.rgbGreen=(BYTE) gg; + if (bb>255) bb=255; if (bb<0) bb=0; color.rgbBlue=(BYTE) bb; +#if CXIMAGE_SUPPORT_ALPHA + if (AlphaIsValid()) { + if (aa>255) aa=255; if (aa<0) aa=0; color.rgbReserved=(BYTE) aa; + } else +#endif + { //Alpha not supported or no alpha at all + color.rgbReserved = 0; + } + return color; + case IM_LANCZOS: + //lanczos window (16*16) sinc interpolation + if (((xi+6)<0) || ((xi-5)>=head.biWidth) || ((yi+6)<0) || ((yi-5)>=head.biHeight)) { + //all points are outside bounds + switch (ofMethod) { + case OM_COLOR: case OM_TRANSPARENT: case OM_BACKGROUND: + //we don't need to interpolate anything with all points outside in this case + return GetPixelColorWithOverflow(-999, -999, ofMethod, rplColor); + break; + default: + //recalculate coordinates and use faster method later on + OverflowCoordinates(x,y,ofMethod); + xi=(int)(x); if (x<0) xi--; //x and/or y have changed ... recalculate xi and yi + yi=(int)(y); if (y<0) yi--; + }//switch + }//if + + for (xii=xi-5; xii=0) && ((yi+6)=0) && !IsIndexed()) { + //optimized interpolation (faster pixel reads) for RGB24 images with all pixels inside bounds + BYTE *pxptr, *pxptra; + for (yii=yi-5; yii255) rr=255; if (rr<0) rr=0; color.rgbRed=(BYTE) rr; + if (gg>255) gg=255; if (gg<0) gg=0; color.rgbGreen=(BYTE) gg; + if (bb>255) bb=255; if (bb<0) bb=0; color.rgbBlue=(BYTE) bb; +#if CXIMAGE_SUPPORT_ALPHA + if (AlphaIsValid()) { + if (aa>255) aa=255; if (aa<0) aa=0; color.rgbReserved=(BYTE) aa; + } else +#endif + { //Alpha not supported or no alpha at all + color.rgbReserved = 0; + } + return color; + }//switch +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Helper function for GetAreaColorInterpolated. + * Adds 'surf' portion of image pixel with color 'color' to (rr,gg,bb,aa). + */ +void CxImage::AddAveragingCont(RGBQUAD const &color, float const surf, float &rr, float &gg, float &bb, float &aa) +{ + rr+=color.rgbRed*surf; + gg+=color.rgbGreen*surf; + bb+=color.rgbBlue*surf; +#if CXIMAGE_SUPPORT_ALPHA + aa+=color.rgbReserved*surf; +#endif +} +//////////////////////////////////////////////////////////////////////////////// +/** + * This method is similar to GetPixelColorInterpolated, but this method also properly handles + * subsampling. + * If you need to sample original image with interval of more than 1 pixel (as when shrinking an image), + * you should use this method instead of GetPixelColorInterpolated or aliasing will occur. + * When area width and height are both less than pixel, this method gets pixel color by interpolating + * color of frame center with selected (inMethod) interpolation by calling GetPixelColorInterpolated. + * If width and height are more than 1, method calculates color by averaging color of pixels within area. + * Interpolation method is not used in this case. Pixel color is interpolated by averaging instead. + * If only one of both is more than 1, method uses combination of interpolation and averaging. + * Chosen interpolation method is used, but since it is averaged later on, there is little difference + * between IM_BILINEAR (perhaps best for this case) and better methods. IM_NEAREST_NEIGHBOUR again + * leads to aliasing artifacts. + * This method is a bit slower than GetPixelColorInterpolated and when aliasing is not a problem, you should + * simply use the later. + * + * \param xc, yc - center of (rectangular) area + * \param w, h - width and height of area + * \param inMethod - interpolation method that is used, when interpolation is used (see above) + * \param ofMethod - overflow method used when retrieving individual pixel colors + * \param rplColor - replacement colour to use, in OM_COLOR + * + * \author ***bd*** 2.2004 + */ +RGBQUAD CxImage::GetAreaColorInterpolated( + float const xc, float const yc, float const w, float const h, + InterpolationMethod const inMethod, + OverflowMethod const ofMethod, + RGBQUAD* const rplColor) +{ + RGBQUAD color; //calculated colour + + if (h<=1 && w<=1) { + //both width and height are less than one... we will use interpolation of center point + return GetPixelColorInterpolated(xc, yc, inMethod, ofMethod, rplColor); + } else { + //area is wider and/or taller than one pixel: + CxRect2 area(xc-w/2.0f, yc-h/2.0f, xc+w/2.0f, yc+h/2.0f); //area + int xi1=(int)(area.botLeft.x+0.49999999f); //low x + int yi1=(int)(area.botLeft.y+0.49999999f); //low y + + + int xi2=(int)(area.topRight.x+0.5f); //top x + int yi2=(int)(area.topRight.y+0.5f); //top y (for loops) + + float rr,gg,bb,aa; //red, green, blue and alpha components + rr=gg=bb=aa=0; + int x,y; //loop counters + float s=0; //surface of all pixels + float cps; //surface of current crosssection + if (h>1 && w>1) { + //width and height of area are greater than one pixel, so we can employ "ordinary" averaging + CxRect2 intBL, intTR; //bottom left and top right intersection + intBL=area.CrossSection(CxRect2(((float)xi1)-0.5f, ((float)yi1)-0.5f, ((float)xi1)+0.5f, ((float)yi1)+0.5f)); + intTR=area.CrossSection(CxRect2(((float)xi2)-0.5f, ((float)yi2)-0.5f, ((float)xi2)+0.5f, ((float)yi2)+0.5f)); + float wBL, wTR, hBL, hTR; + wBL=intBL.Width(); //width of bottom left pixel-area intersection + hBL=intBL.Height(); //height of bottom left... + wTR=intTR.Width(); //width of top right... + hTR=intTR.Height(); //height of top right... + + AddAveragingCont(GetPixelColorWithOverflow(xi1,yi1,ofMethod,rplColor), wBL*hBL, rr, gg, bb, aa); //bottom left pixel + AddAveragingCont(GetPixelColorWithOverflow(xi2,yi1,ofMethod,rplColor), wTR*hBL, rr, gg, bb, aa); //bottom right pixel + AddAveragingCont(GetPixelColorWithOverflow(xi1,yi2,ofMethod,rplColor), wBL*hTR, rr, gg, bb, aa); //top left pixel + AddAveragingCont(GetPixelColorWithOverflow(xi2,yi2,ofMethod,rplColor), wTR*hTR, rr, gg, bb, aa); //top right pixel + //bottom and top row + for (x=xi1+1; x255) rr=255; if (rr<0) rr=0; color.rgbRed=(BYTE) rr; + if (gg>255) gg=255; if (gg<0) gg=0; color.rgbGreen=(BYTE) gg; + if (bb>255) bb=255; if (bb<0) bb=0; color.rgbBlue=(BYTE) bb; +#if CXIMAGE_SUPPORT_ALPHA + if (AlphaIsValid()) { + if (aa>255) aa=255; if (aa<0) aa=0; color.rgbReserved=(BYTE) aa; + }//if +#endif + }//if + return color; +} + +//////////////////////////////////////////////////////////////////////////////// +float CxImage::KernelBSpline(const float x) +{ + if (x>2.0f) return 0.0f; + // thanks to Kristian Kratzenstein + float a, b, c, d; + float xm1 = x - 1.0f; // Was calculatet anyway cause the "if((x-1.0f) < 0)" + float xp1 = x + 1.0f; + float xp2 = x + 2.0f; + + if ((xp2) <= 0.0f) a = 0.0f; else a = xp2*xp2*xp2; // Only float, not float -> double -> float + if ((xp1) <= 0.0f) b = 0.0f; else b = xp1*xp1*xp1; + if (x <= 0) c = 0.0f; else c = x*x*x; + if ((xm1) <= 0.0f) d = 0.0f; else d = xm1*xm1*xm1; + + return (0.16666666666666666667f * (a - (4.0f * b) + (6.0f * c) - (4.0f * d))); + + /* equivalent + if (x < -2.0) + return(0.0f); + if (x < -1.0) + return((2.0f+x)*(2.0f+x)*(2.0f+x)*0.16666666666666666667f); + if (x < 0.0) + return((4.0f+x*x*(-6.0f-3.0f*x))*0.16666666666666666667f); + if (x < 1.0) + return((4.0f+x*x*(-6.0f+3.0f*x))*0.16666666666666666667f); + if (x < 2.0) + return((2.0f-x)*(2.0f-x)*(2.0f-x)*0.16666666666666666667f); + return(0.0f); + */ +} + +//////////////////////////////////////////////////////////////////////////////// +/** + * Bilinear interpolation kernel: + \verbatim + / + | 1-t , if 0 <= t <= 1 + h(t) = | t+1 , if -1 <= t < 0 + | 0 , otherwise + \ + \endverbatim + * ***bd*** 2.2004 + */ +float CxImage::KernelLinear(const float t) +{ +// if (0<=t && t<=1) return 1-t; +// if (-1<=t && t<0) return 1+t; +// return 0; + + // + if (t < -1.0f) + return 0.0f; + if (t < 0.0f) + return 1.0f+t; + if (t < 1.0f) + return 1.0f-t; + return 0.0f; +} + +//////////////////////////////////////////////////////////////////////////////// +/** + * Bicubic interpolation kernel (a=-1): + \verbatim + / + | 1-2|t|**2+|t|**3 , if |t| < 1 + h(t) = | 4-8|t|+5|t|**2-|t|**3 , if 1<=|t|<2 + | 0 , otherwise + \ + \endverbatim + * ***bd*** 2.2004 + */ +float CxImage::KernelCubic(const float t) +{ + float abs_t = (float)fabs(t); + float abs_t_sq = abs_t * abs_t; + if (abs_t<1) return 1-2*abs_t_sq+abs_t_sq*abs_t; + if (abs_t<2) return 4 - 8*abs_t +5*abs_t_sq - abs_t_sq*abs_t; + return 0; +} + +//////////////////////////////////////////////////////////////////////////////// +/** + * Bicubic kernel (for a=-1 it is the same as BicubicKernel): + \verbatim + / + | (a+2)|t|**3 - (a+3)|t|**2 + 1 , |t| <= 1 + h(t) = | a|t|**3 - 5a|t|**2 + 8a|t| - 4a , 1 < |t| <= 2 + | 0 , otherwise + \ + \endverbatim + * Often used values for a are -1 and -1/2. + */ +float CxImage::KernelGeneralizedCubic(const float t, const float a) +{ + float abs_t = (float)fabs(t); + float abs_t_sq = abs_t * abs_t; + if (abs_t<1) return (a+2)*abs_t_sq*abs_t - (a+3)*abs_t_sq + 1; + if (abs_t<2) return a*abs_t_sq*abs_t - 5*a*abs_t_sq + 8*a*abs_t - 4*a; + return 0; +} + +//////////////////////////////////////////////////////////////////////////////// +/** + * Lanczos windowed sinc interpolation kernel with radius r. + \verbatim + / + h(t) = | sinc(t)*sinc(t/r) , if |t| r) return 0; + if (t==0) return 1; + float pit=PI*t; + float pitd=pit/r; + return (float)((sin(pit)/pit) * (sin(pitd)/pitd)); +} + +//////////////////////////////////////////////////////////////////////////////// +float CxImage::KernelBox(const float x) +{ + if (x < -0.5f) + return 0.0f; + if (x < 0.5f) + return 1.0f; + return 0.0f; +} +//////////////////////////////////////////////////////////////////////////////// +float CxImage::KernelHermite(const float x) +{ + if (x < -1.0f) + return 0.0f; + if (x < 0.0f) + return (-2.0f*x-3.0f)*x*x+1.0f; + if (x < 1.0f) + return (2.0f*x-3.0f)*x*x+1.0f; + return 0.0f; +// if (fabs(x)>1) return 0.0f; +// return(0.5f+0.5f*(float)cos(PI*x)); +} +//////////////////////////////////////////////////////////////////////////////// +float CxImage::KernelHamming(const float x) +{ + if (x < -1.0f) + return 0.0f; + if (x < 0.0f) + return 0.92f*(-2.0f*x-3.0f)*x*x+1.0f; + if (x < 1.0f) + return 0.92f*(2.0f*x-3.0f)*x*x+1.0f; + return 0.0f; +// if (fabs(x)>1) return 0.0f; +// return(0.54f+0.46f*(float)cos(PI*x)); +} +//////////////////////////////////////////////////////////////////////////////// +float CxImage::KernelSinc(const float x) +{ + if (x == 0.0) + return(1.0); + return((float)sin(PI*x)/(PI*x)); +} +//////////////////////////////////////////////////////////////////////////////// +float CxImage::KernelBlackman(const float x) +{ + //if (fabs(x)>1) return 0.0f; + return (0.42f+0.5f*(float)cos(PI*x)+0.08f*(float)cos(2.0f*PI*x)); +} +//////////////////////////////////////////////////////////////////////////////// +float CxImage::KernelBessel_J1(const float x) +{ + double p, q; + + register long i; + + static const double + Pone[] = + { + 0.581199354001606143928050809e+21, + -0.6672106568924916298020941484e+20, + 0.2316433580634002297931815435e+19, + -0.3588817569910106050743641413e+17, + 0.2908795263834775409737601689e+15, + -0.1322983480332126453125473247e+13, + 0.3413234182301700539091292655e+10, + -0.4695753530642995859767162166e+7, + 0.270112271089232341485679099e+4 + }, + Qone[] = + { + 0.11623987080032122878585294e+22, + 0.1185770712190320999837113348e+20, + 0.6092061398917521746105196863e+17, + 0.2081661221307607351240184229e+15, + 0.5243710262167649715406728642e+12, + 0.1013863514358673989967045588e+10, + 0.1501793594998585505921097578e+7, + 0.1606931573481487801970916749e+4, + 0.1e+1 + }; + + p = Pone[8]; + q = Qone[8]; + for (i=7; i >= 0; i--) + { + p = p*x*x+Pone[i]; + q = q*x*x+Qone[i]; + } + return (float)(p/q); +} +//////////////////////////////////////////////////////////////////////////////// +float CxImage::KernelBessel_P1(const float x) +{ + double p, q; + + register long i; + + static const double + Pone[] = + { + 0.352246649133679798341724373e+5, + 0.62758845247161281269005675e+5, + 0.313539631109159574238669888e+5, + 0.49854832060594338434500455e+4, + 0.2111529182853962382105718e+3, + 0.12571716929145341558495e+1 + }, + Qone[] = + { + 0.352246649133679798068390431e+5, + 0.626943469593560511888833731e+5, + 0.312404063819041039923015703e+5, + 0.4930396490181088979386097e+4, + 0.2030775189134759322293574e+3, + 0.1e+1 + }; + + p = Pone[5]; + q = Qone[5]; + for (i=4; i >= 0; i--) + { + p = p*(8.0/x)*(8.0/x)+Pone[i]; + q = q*(8.0/x)*(8.0/x)+Qone[i]; + } + return (float)(p/q); +} +//////////////////////////////////////////////////////////////////////////////// +float CxImage::KernelBessel_Q1(const float x) +{ + double p, q; + + register long i; + + static const double + Pone[] = + { + 0.3511751914303552822533318e+3, + 0.7210391804904475039280863e+3, + 0.4259873011654442389886993e+3, + 0.831898957673850827325226e+2, + 0.45681716295512267064405e+1, + 0.3532840052740123642735e-1 + }, + Qone[] = + { + 0.74917374171809127714519505e+4, + 0.154141773392650970499848051e+5, + 0.91522317015169922705904727e+4, + 0.18111867005523513506724158e+4, + 0.1038187585462133728776636e+3, + 0.1e+1 + }; + + p = Pone[5]; + q = Qone[5]; + for (i=4; i >= 0; i--) + { + p = p*(8.0/x)*(8.0/x)+Pone[i]; + q = q*(8.0/x)*(8.0/x)+Qone[i]; + } + return (float)(p/q); +} +//////////////////////////////////////////////////////////////////////////////// +float CxImage::KernelBessel_Order1(float x) +{ + float p, q; + + if (x == 0.0) + return (0.0f); + p = x; + if (x < 0.0) + x=(-x); + if (x < 8.0) + return(p*KernelBessel_J1(x)); + q = (float)sqrt(2.0f/(PI*x))*(float)(KernelBessel_P1(x)*(1.0f/sqrt(2.0f)*(sin(x)-cos(x)))-8.0f/x*KernelBessel_Q1(x)* + (-1.0f/sqrt(2.0f)*(sin(x)+cos(x)))); + if (p < 0.0f) + q = (-q); + return (q); +} +//////////////////////////////////////////////////////////////////////////////// +float CxImage::KernelBessel(const float x) +{ + if (x == 0.0f) + return(PI/4.0f); + return(KernelBessel_Order1(PI*x)/(2.0f*x)); +} +//////////////////////////////////////////////////////////////////////////////// +float CxImage::KernelGaussian(const float x) +{ + return (float)(exp(-2.0f*x*x)*0.79788456080287f/*sqrt(2.0f/PI)*/); +} +//////////////////////////////////////////////////////////////////////////////// +float CxImage::KernelQuadratic(const float x) +{ + if (x < -1.5f) + return(0.0f); + if (x < -0.5f) + return(0.5f*(x+1.5f)*(x+1.5f)); + if (x < 0.5f) + return(0.75f-x*x); + if (x < 1.5f) + return(0.5f*(x-1.5f)*(x-1.5f)); + return(0.0f); +} +//////////////////////////////////////////////////////////////////////////////// +float CxImage::KernelMitchell(const float x) +{ +#define KM_B (1.0f/3.0f) +#define KM_C (1.0f/3.0f) +#define KM_P0 (( 6.0f - 2.0f * KM_B ) / 6.0f) +#define KM_P2 ((-18.0f + 12.0f * KM_B + 6.0f * KM_C) / 6.0f) +#define KM_P3 (( 12.0f - 9.0f * KM_B - 6.0f * KM_C) / 6.0f) +#define KM_Q0 (( 8.0f * KM_B + 24.0f * KM_C) / 6.0f) +#define KM_Q1 ((-12.0f * KM_B - 48.0f * KM_C) / 6.0f) +#define KM_Q2 (( 6.0f * KM_B + 30.0f * KM_C) / 6.0f) +#define KM_Q3 (( -1.0f * KM_B - 6.0f * KM_C) / 6.0f) + + if (x < -2.0) + return(0.0f); + if (x < -1.0) + return(KM_Q0-x*(KM_Q1-x*(KM_Q2-x*KM_Q3))); + if (x < 0.0f) + return(KM_P0+x*x*(KM_P2-x*KM_P3)); + if (x < 1.0f) + return(KM_P0+x*x*(KM_P2+x*KM_P3)); + if (x < 2.0f) + return(KM_Q0+x*(KM_Q1+x*(KM_Q2+x*KM_Q3))); + return(0.0f); +} +//////////////////////////////////////////////////////////////////////////////// +float CxImage::KernelCatrom(const float x) +{ + if (x < -2.0) + return(0.0f); + if (x < -1.0) + return(0.5f*(4.0f+x*(8.0f+x*(5.0f+x)))); + if (x < 0.0) + return(0.5f*(2.0f+x*x*(-5.0f-3.0f*x))); + if (x < 1.0) + return(0.5f*(2.0f+x*x*(-5.0f+3.0f*x))); + if (x < 2.0) + return(0.5f*(4.0f+x*(-8.0f+x*(5.0f-x)))); + return(0.0f); +} +//////////////////////////////////////////////////////////////////////////////// + +#endif diff --git a/src/win32/dependencies/cximage/ximaiter.h b/src/win32/dependencies/cximage/ximaiter.h new file mode 100644 index 00000000..39fc5afd --- /dev/null +++ b/src/win32/dependencies/cximage/ximaiter.h @@ -0,0 +1,252 @@ +/* + * File: ImaIter.h + * Purpose: Declaration of the Platform Independent Image Base Class + * Author: Alejandro Aguilar Sierra + * Created: 1995 + * Copyright: (c) 1995, Alejandro Aguilar Sierra + * + * 07/08/2001 Davide Pizzolato - www.xdp.it + * - removed slow loops + * - added safe checks + * + * Permission is given by the author to freely redistribute and include + * this code in any program as long as this credit is given where due. + * + * COVERED CODE IS PROVIDED UNDER THIS LICENSE ON AN "AS IS" BASIS, WITHOUT WARRANTY + * OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, WITHOUT LIMITATION, WARRANTIES + * THAT THE COVERED CODE IS FREE OF DEFECTS, MERCHANTABLE, FIT FOR A PARTICULAR PURPOSE + * OR NON-INFRINGING. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE COVERED + * CODE IS WITH YOU. SHOULD ANY COVERED CODE PROVE DEFECTIVE IN ANY RESPECT, YOU (NOT + * THE INITIAL DEVELOPER OR ANY OTHER CONTRIBUTOR) ASSUME THE COST OF ANY NECESSARY + * SERVICING, REPAIR OR CORRECTION. THIS DISCLAIMER OF WARRANTY CONSTITUTES AN ESSENTIAL + * PART OF THIS LICENSE. NO USE OF ANY COVERED CODE IS AUTHORIZED HEREUNDER EXCEPT UNDER + * THIS DISCLAIMER. + * + * Use at your own risk! + * ========================================================== + */ + +#if !defined(__ImaIter_h) +#define __ImaIter_h + +#include "ximage.h" +#include "ximadef.h" + +class CImageIterator +{ +friend class CxImage; +protected: + int Itx, Ity; // Counters + int Stepx, Stepy; + BYTE* IterImage; // Image pointer + CxImage *ima; +public: + // Constructors + CImageIterator ( void ); + CImageIterator ( CxImage *image ); + operator CxImage* (); + + // Iterators + BOOL ItOK (); + void Reset (); + void Upset (); + void SetRow(BYTE *buf, int n); + void GetRow(BYTE *buf, int n); + BYTE GetByte( ) { return IterImage[Itx]; } + void SetByte(BYTE b) { IterImage[Itx] = b; } + BYTE* GetRow(void); + BYTE* GetRow(int n); + BOOL NextRow(); + BOOL PrevRow(); + BOOL NextByte(); + BOOL PrevByte(); + + void SetSteps(int x, int y=0) { Stepx = x; Stepy = y; } + void GetSteps(int *x, int *y) { *x = Stepx; *y = Stepy; } + BOOL NextStep(); + BOOL PrevStep(); + + void SetY(int y); /* AD - for interlace */ + int GetY() {return Ity;} + BOOL GetCol(BYTE* pCol, DWORD x); + BOOL SetCol(BYTE* pCol, DWORD x); +}; + +///////////////////////////////////////////////////////////////////// +inline +CImageIterator::CImageIterator(void) +{ + ima = 0; + IterImage = 0; + Itx = Ity = 0; + Stepx = Stepy = 0; +} +///////////////////////////////////////////////////////////////////// +inline +CImageIterator::CImageIterator(CxImage *imageImpl): ima(imageImpl) +{ + if (ima) IterImage = ima->GetBits(); + Itx = Ity = 0; + Stepx = Stepy = 0; +} +///////////////////////////////////////////////////////////////////// +inline +CImageIterator::operator CxImage* () +{ + return ima; +} +///////////////////////////////////////////////////////////////////// +inline BOOL CImageIterator::ItOK () +{ + if (ima) return ima->IsInside(Itx, Ity); + else return FALSE; +} +///////////////////////////////////////////////////////////////////// +inline void CImageIterator::Reset() +{ + if (ima) IterImage = ima->GetBits(); + else IterImage=0; + Itx = Ity = 0; +} +///////////////////////////////////////////////////////////////////// +inline void CImageIterator::Upset() +{ + Itx = 0; + Ity = ima->GetHeight()-1; + IterImage = ima->GetBits() + ima->GetEffWidth()*(ima->GetHeight()-1); +} +///////////////////////////////////////////////////////////////////// +inline BOOL CImageIterator::NextRow() +{ + if (++Ity >= (int)ima->GetHeight()) return 0; + IterImage += ima->GetEffWidth(); + return 1; +} +///////////////////////////////////////////////////////////////////// +inline BOOL CImageIterator::PrevRow() +{ + if (--Ity < 0) return 0; + IterImage -= ima->GetEffWidth(); + return 1; +} +/* AD - for interlace */ +inline void CImageIterator::SetY(int y) +{ + if ((y < 0) || (y > (int)ima->GetHeight())) return; + Ity = y; + IterImage = ima->GetBits() + ima->GetEffWidth()*y; +} +///////////////////////////////////////////////////////////////////// +inline void CImageIterator::SetRow(BYTE *buf, int n) +{ + if (n<0) n = (int)ima->GetEffWidth(); + else n = min(n,(int)ima->GetEffWidth()); + + if ((IterImage!=NULL)&&(buf!=NULL)&&(n>0)) memcpy(IterImage,buf,n); +} +///////////////////////////////////////////////////////////////////// +inline void CImageIterator::GetRow(BYTE *buf, int n) +{ + if ((IterImage!=NULL)&&(buf!=NULL)&&(n>0)) memcpy(buf,IterImage,n); +} +///////////////////////////////////////////////////////////////////// +inline BYTE* CImageIterator::GetRow() +{ + return IterImage; +} +///////////////////////////////////////////////////////////////////// +inline BYTE* CImageIterator::GetRow(int n) +{ + SetY(n); + return IterImage; +} +///////////////////////////////////////////////////////////////////// +inline BOOL CImageIterator::NextByte() +{ + if (++Itx < (int)ima->GetEffWidth()) return 1; + else + if (++Ity < (int)ima->GetHeight()){ + IterImage += ima->GetEffWidth(); + Itx = 0; + return 1; + } else + return 0; +} +///////////////////////////////////////////////////////////////////// +inline BOOL CImageIterator::PrevByte() +{ + if (--Itx >= 0) return 1; + else + if (--Ity >= 0){ + IterImage -= ima->GetEffWidth(); + Itx = 0; + return 1; + } else + return 0; +} +///////////////////////////////////////////////////////////////////// +inline BOOL CImageIterator::NextStep() +{ + Itx += Stepx; + if (Itx < (int)ima->GetEffWidth()) return 1; + else { + Ity += Stepy; + if (Ity < (int)ima->GetHeight()){ + IterImage += ima->GetEffWidth(); + Itx = 0; + return 1; + } else + return 0; + } +} +///////////////////////////////////////////////////////////////////// +inline BOOL CImageIterator::PrevStep() +{ + Itx -= Stepx; + if (Itx >= 0) return 1; + else { + Ity -= Stepy; + if (Ity >= 0 && Ity < (int)ima->GetHeight()) { + IterImage -= ima->GetEffWidth(); + Itx = 0; + return 1; + } else + return 0; + } +} +///////////////////////////////////////////////////////////////////// +inline BOOL CImageIterator::GetCol(BYTE* pCol, DWORD x) +{ + if ((pCol==0)||(ima->GetBpp()<8)||(x>=ima->GetWidth())) + return 0; + DWORD h = ima->GetHeight(); + //DWORD line = ima->GetEffWidth(); + BYTE bytes = ima->GetBpp()>>3; + BYTE* pSrc; + for (DWORD y=0;yGetBits(y) + x*bytes; + for (BYTE w=0;wGetBpp()<8)||(x>=ima->GetWidth())) + return 0; + DWORD h = ima->GetHeight(); + //DWORD line = ima->GetEffWidth(); + BYTE bytes = ima->GetBpp()>>3; + BYTE* pSrc; + for (DWORD y=0;yGetBits(y) + x*bytes; + for (BYTE w=0;wSize(); + src=(BYTE*)malloc(len); + hFile->Read(src, len, 1); + + if (!j2k_decode(src, len, &img, &cp)) { + free(src); + throw "failed to decode J2K image!"; + } + + free(src); + + if (img->numcomps==3 && + img->comps[0].dx==img->comps[1].dx && + img->comps[1].dx==img->comps[2].dx && + img->comps[0].dy==img->comps[1].dy && + img->comps[1].dy==img->comps[2].dy && + img->comps[0].prec==img->comps[1].prec && + img->comps[1].prec==img->comps[2].prec) + { + w=CEILDIV(img->x1-img->x0, img->comps[0].dx); + h=CEILDIV(img->y1-img->y0, img->comps[0].dy); + max=(1<comps[0].prec)-1; + + Create(w,h,24,CXIMAGE_FORMAT_J2K); + + RGBQUAD c; + for (i=0,y=0; ycomps[0].data[i]; + c.rgbGreen = img->comps[1].data[i]; + c.rgbBlue = img->comps[2].data[i]; + SetPixelColor(x,h-1-y,c); + } + } + } else { + int compno; + info.nNumFrames = img->numcomps; + if ((info.nFrame<0)||(info.nFrame>=info.nNumFrames)){ + j2k_destroy(&img,&cp); + throw "wrong frame!"; + } + for (compno=0; compno<=info.nFrame; compno++) { + w=CEILDIV(img->x1-img->x0, img->comps[compno].dx); + h=CEILDIV(img->y1-img->y0, img->comps[compno].dy); + max=(1<comps[compno].prec)-1; + Create(w,h,8,CXIMAGE_FORMAT_J2K); + SetGrayPalette(); + for (i=0,y=0; ycomps[compno].data[i]); + } + } + } + } + + j2k_destroy(&img,&cp); + + } catch (char *message) { + strncpy(info.szLastError,message,255); + return FALSE; + } + return true; +} +//////////////////////////////////////////////////////////////////////////////// +bool CxImageJ2K::Encode(CxFile * hFile) +{ + if (EncodeSafeCheck(hFile)) return false; + + if (head.biClrUsed!=0 && !IsGrayScale()){ + strcpy(info.szLastError,"J2K can save only RGB or GrayScale images"); + return false; + } + + int i,x,y; + j2k_image_t *img; + j2k_cp_t *cp; + j2k_tcp_t *tcp; + j2k_tccp_t *tccp; + + img = (j2k_image_t *)calloc(sizeof(j2k_image_t),1); + cp = (j2k_cp_t *)calloc(sizeof(j2k_cp_t),1); + + cp->tx0=0; cp->ty0=0; + cp->tw=1; cp->th=1; + cp->tcps=(j2k_tcp_t*)calloc(sizeof(j2k_tcp_t),1); + tcp=&cp->tcps[0]; + + long w=head.biWidth; + long h=head.biHeight; + + tcp->numlayers=1; + for (i=0;inumlayers;i++) tcp->rates[i]=(w*h*GetJpegQuality())/600; + + + if (IsGrayScale()) { + img->x0=0; + img->y0=0; + img->x1=w; + img->y1=h; + img->numcomps=1; + img->comps=(j2k_comp_t*)calloc(sizeof(j2k_comp_t),1); + img->comps[0].data=(int*)calloc(w*h*sizeof(int),1); + img->comps[0].prec=8; + img->comps[0].sgnd=0; + img->comps[0].dx=1; + img->comps[0].dy=1; + for (i=0,y=0; ycomps[0].data[i]=GetPixelIndex(x,h-1-y); + } + } + } else if (!IsIndexed()) { + img->x0=0; + img->y0=0; + img->x1=w; + img->y1=h; + img->numcomps=3; + img->comps=(j2k_comp_t*)calloc(img->numcomps*sizeof(j2k_comp_t),1); + for (i=0; inumcomps; i++) { + img->comps[i].data=(int*)calloc(w*h*sizeof(int),1); + img->comps[i].prec=8; + img->comps[i].sgnd=0; + img->comps[i].dx=1; + img->comps[i].dy=1; + } + RGBQUAD c; + for (i=0,y=0; ycomps[0].data[i]=c.rgbRed; + img->comps[1].data[i]=c.rgbGreen; + img->comps[2].data[i]=c.rgbBlue; + } + } + } else { + return 0; + } + + cp->tdx=img->x1-img->x0; + cp->tdy=img->y1-img->y0; + + tcp->csty=0; + tcp->prg=0; + tcp->mct=img->numcomps==3?1:0; + tcp->tccps=(j2k_tccp_t*)calloc(img->numcomps*sizeof(j2k_tccp_t),1); + + int ir=0; /* or 1 ???*/ + + for (i=0; inumcomps; i++) { + tccp=&tcp->tccps[i]; + tccp->csty=0; + tccp->numresolutions=6; + tccp->cblkw=6; + tccp->cblkh=6; + tccp->cblksty=0; + tccp->qmfbid=ir?0:1; + tccp->qntsty=ir?J2K_CCP_QNTSTY_SEQNT:J2K_CCP_QNTSTY_NOQNT; + tccp->numgbits=2; + tccp->roishift=0; + j2k_calc_explicit_stepsizes(tccp, img->comps[i].prec); + } + + BYTE* dest=(BYTE*)calloc(tcp->rates[tcp->numlayers-1]+2,1); + long len = j2k_encode(img, cp, dest, tcp->rates[tcp->numlayers-1]+2); + + if (len==0) { + strcpy(info.szLastError,"J2K failed to encode image"); + } else { + hFile->Write(dest, len, 1); + } + + free(dest); + j2k_destroy(&img,&cp); + + return (len!=0); +} +//////////////////////////////////////////////////////////////////////////////// + +static const double dwt_norms_97[4][10]={ + {1.000, 1.965, 4.177, 8.403, 16.90, 33.84, 67.69, 135.3, 270.6, 540.9}, + {2.022, 3.989, 8.355, 17.04, 34.27, 68.63, 137.3, 274.6, 549.0}, + {2.022, 3.989, 8.355, 17.04, 34.27, 68.63, 137.3, 274.6, 549.0}, + {2.080, 3.865, 8.307, 17.18, 34.71, 69.59, 139.3, 278.6, 557.2} +}; + +//////////////////////////////////////////////////////////////////////////////// +void CxImageJ2K::j2k_calc_explicit_stepsizes(j2k_tccp_t *tccp, int prec) { + int numbands, bandno; + numbands=3*tccp->numresolutions-2; + for (bandno=0; bandnonumresolutions-1-resno; + gain=tccp->qmfbid==0?0:(orient==0?0:(orient==1||orient==2?1:2)); + if (tccp->qntsty==J2K_CCP_QNTSTY_NOQNT) { + stepsize=1.0; + } else { + double norm=dwt_norms_97[orient][level]; + stepsize=(1<<(gain+1))/norm; + } + j2k_encode_stepsize((int)floor(stepsize*8192.0), prec+gain, &tccp->stepsizes[bandno].expn, &tccp->stepsizes[bandno].mant); + } +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageJ2K::j2k_encode_stepsize(int stepsize, int numbps, int *expn, int *mant) { + int p, n; + p=j2k_floorlog2(stepsize)-13; + n=11-j2k_floorlog2(stepsize); + *mant=(n<0?stepsize>>-n:stepsize<1; l++) { + a>>=1; + } + return l; +} +//////////////////////////////////////////////////////////////////////////////// +#endif // CXIMAGE_SUPPORT_J2K diff --git a/src/win32/dependencies/cximage/ximaj2k.h b/src/win32/dependencies/cximage/ximaj2k.h new file mode 100644 index 00000000..9f2642eb --- /dev/null +++ b/src/win32/dependencies/cximage/ximaj2k.h @@ -0,0 +1,46 @@ +/* + * File: ximaj2k.h + * Purpose: J2K Image Class Loader and Writer + */ +/* ========================================================== + * CxImageJ2K (c) 04/Aug/2002 Davide Pizzolato - www.xdp.it + * For conditions of distribution and use, see copyright notice in ximage.h + * + * based on LIBJ2K Copyright (c) 2001-2002, David Janssens - All rights reserved. + * ========================================================== + */ +#if !defined(__ximaJ2K_h) +#define __ximaJ2K_h + +#include "ximage.h" + +#if CXIMAGE_SUPPORT_J2K + +#define LIBJ2K_EXPORTS +extern "C" { +#include "../j2k/j2k.h" +}; + +class CxImageJ2K: public CxImage +{ +public: + CxImageJ2K(): CxImage(CXIMAGE_FORMAT_J2K) {} + +// bool Load(const char * imageFileName){ return CxImage::Load(imageFileName,CXIMAGE_FORMAT_J2K);} +// bool Save(const char * imageFileName){ return CxImage::Save(imageFileName,CXIMAGE_FORMAT_J2K);} + bool Decode(CxFile * hFile); + bool Decode(FILE *hFile) { CxIOFile file(hFile); return Decode(&file); } + +#if CXIMAGE_SUPPORT_ENCODE + bool Encode(CxFile * hFile); + bool Encode(FILE *hFile) { CxIOFile file(hFile); return Encode(&file); } +#endif // CXIMAGE_SUPPORT_ENCODE +protected: + void j2k_calc_explicit_stepsizes(j2k_tccp_t *tccp, int prec); + void j2k_encode_stepsize(int stepsize, int numbps, int *expn, int *mant); + int j2k_floorlog2(int a); +}; + +#endif + +#endif diff --git a/src/win32/dependencies/cximage/ximajas.cpp b/src/win32/dependencies/cximage/ximajas.cpp new file mode 100644 index 00000000..fff99463 --- /dev/null +++ b/src/win32/dependencies/cximage/ximajas.cpp @@ -0,0 +1,265 @@ +/* + * File: ximajas.cpp + * Purpose: Platform Independent JasPer Image Class Loader and Writer + * 12/Apr/2003 Davide Pizzolato - www.xdp.it + * CxImage version 5.99c 17/Oct/2004 + */ + +#include "ximajas.h" + +#if CXIMAGE_SUPPORT_JASPER + +//////////////////////////////////////////////////////////////////////////////// +bool CxImageJAS::Decode(CxFile *hFile, DWORD imagetype) +{ + if (hFile == NULL) return false; + + jas_image_t *image=0; + jas_stream_t *in=0; + jas_matrix_t **bufs=0; + long i,error=0; + //jas_setdbglevel(0); + + try + { + if (jas_init()) + throw "cannot initialize jasper"; + + if (!(in = jas_stream_fdopen(0, "rb"))) + throw "error: cannot open standard input"; + + CxFileJas src(hFile,in); + + if (!(image = jas_image_decode(in, -1, 0))) + throw "error: cannot load image data"; + + long x,y,w,h,depth,cmptno; + + w = jas_image_cmptwidth(image,0); + h = jas_image_cmptheight(image,0); + depth = jas_image_cmptprec(image,0); + + if (image->numcmpts_ > 64 || image->numcmpts_ < 0) + throw "error: too much components"; + + if (depth!=1 && depth!=4 && depth!=8){ + jas_image_t *newimage; + jas_cmprof_t *outprof; + //jas_eprintf("forcing conversion to sRGB\n"); + if (!(outprof = jas_cmprof_createfromclrspc(JAS_CLRSPC_SRGB))) { + throw "cannot create sRGB profile"; + } + if (!(newimage = jas_image_chclrspc(image, outprof, JAS_CMXFORM_INTENT_PER))) { + throw "cannot convert to sRGB"; + } + jas_image_destroy(image); + jas_cmprof_destroy(outprof); + image = newimage; + } + + bufs = (jas_matrix_t **)calloc(image->numcmpts_, sizeof(jas_matrix_t**)); + for (i = 0; i < image->numcmpts_; ++i) { + if (!(bufs[i] = jas_matrix_create(1, w))) { + throw "error: cannot allocate memory"; + } + } + + if (image->numcmpts_==3 && + image->cmpts_[0]->width_ == image->cmpts_[1]->width_ && + image->cmpts_[1]->width_ == image->cmpts_[2]->width_ && + image->cmpts_[0]->height_ == image->cmpts_[1]->height_ && + image->cmpts_[1]->height_ == image->cmpts_[2]->height_ && + image->cmpts_[0]->prec_ == image->cmpts_[1]->prec_ && + image->cmpts_[1]->prec_ == image->cmpts_[2]->prec_ ) + { + + if(!Create(w,h,24,imagetype)) + throw "Can't allocate memory"; + + RGBQUAD c; + for (y=0; ynumcmpts_; ++cmptno) { + jas_image_readcmpt(image, cmptno, 0, y, w, 1, bufs[cmptno]); + } + + for (x=0; xnumcmpts_; + if ((info.nFrame<0)||(info.nFrame>=info.nNumFrames)){ + throw "wrong frame!"; + } + for (cmptno=0; cmptno<=info.nFrame; cmptno++) { + w = jas_image_cmptwidth(image,cmptno); + h = jas_image_cmptheight(image,cmptno); + depth = jas_image_cmptprec(image,cmptno); + if (depth>8) depth=8; + if(!Create(w,h,depth,imagetype)) + throw "Can't allocate memory"; + SetGrayPalette(); + for (y=0; ynumcmpts_; ++i){ if (bufs[i]) jas_matrix_destroy(bufs[i]);} + free(bufs); + } + jas_cleanup(); + if (image) jas_image_destroy(image); + if (in) jas_stream_close(in); + return (error==0); +} +//////////////////////////////////////////////////////////////////////////////// +#if CXIMAGE_SUPPORT_ENCODE +//////////////////////////////////////////////////////////////////////////////// +bool CxImageJAS::Encode(CxFile * hFile, DWORD imagetype) +{ + if (EncodeSafeCheck(hFile)) return false; + + if (head.biClrUsed!=0 && !IsGrayScale()){ + strcpy(info.szLastError,"JasPer can save only RGB or GrayScale images"); + return false; + } + + jas_image_t *image=0; + jas_stream_t *out=0; + jas_matrix_t *cmpts[3]; + long x,y,yflip,error=0; + uint_fast16_t cmptno, numcmpts; + jas_image_cmptparm_t cmptparms[3], *cmptparm; + + try + { + + if (jas_init()) + throw "cannot initialize jasper"; + + if (!(out = jas_stream_fdopen(0, "wb"))) + throw "error: cannot open standard output"; + + CxFileJas src(hFile,out); + + numcmpts = head.biClrUsed==0 ? 3 : 1; + + for (cmptno = 0, cmptparm = cmptparms; cmptno < numcmpts; ++cmptno, ++cmptparm) { + cmptparm->tlx = 0; + cmptparm->tly = 0; + cmptparm->hstep = 1; + cmptparm->vstep = 1; + cmptparm->width = head.biWidth; + cmptparm->height = head.biHeight; + cmptparm->prec = 8; + cmptparm->sgnd = false; + } + + /* Create image object. */ + if (!(image = jas_image_create(numcmpts, cmptparms, JAS_CLRSPC_UNKNOWN))) + throw "error : jas_image_create"; + + if (numcmpts == 3) { + jas_image_setclrspc(image, JAS_CLRSPC_SRGB); + jas_image_setcmpttype(image, 0, + JAS_IMAGE_CT_COLOR(JAS_CLRSPC_CHANIND_RGB_R)); + jas_image_setcmpttype(image, 1, + JAS_IMAGE_CT_COLOR(JAS_CLRSPC_CHANIND_RGB_G)); + jas_image_setcmpttype(image, 2, + JAS_IMAGE_CT_COLOR(JAS_CLRSPC_CHANIND_RGB_B)); + } else { + jas_image_setclrspc(image, JAS_CLRSPC_SGRAY); + jas_image_setcmpttype(image, 0, + JAS_IMAGE_CT_COLOR(JAS_CLRSPC_CHANIND_GRAY_Y)); + } + + + for (x = 0; x < numcmpts; ++x) { cmpts[x] = 0; } + /* Create temporary matrices to hold component data. */ + for (x = 0; x < numcmpts; ++x) { + if (!(cmpts[x] = jas_matrix_create(1, head.biWidth))) { + throw "error : can't allocate memory"; + } + } + + RGBQUAD c; + for (y = 0; y < head.biHeight; ++y) { + for (x = 0; x < head.biWidth; ++x) { + if (head.biClrUsed==0){ + c = GetPixelColor(x,y); + jas_matrix_setv(cmpts[0], x, c.rgbRed); + jas_matrix_setv(cmpts[1], x, c.rgbGreen); + jas_matrix_setv(cmpts[2], x, c.rgbBlue); + } else { + jas_matrix_setv(cmpts[0], x, GetPixelIndex(x,y)); + } + } + yflip = head.biHeight - 1 - y; + for (cmptno = 0; cmptno < numcmpts; ++cmptno) { + if (jas_image_writecmpt(image, cmptno, 0, yflip, head.biWidth, 1, cmpts[cmptno])) { + throw "error : jas_image_writecmpt"; + } + } + } + + char szfmt[4]; + *szfmt = '\0'; +#if CXIMAGE_SUPPORT_JP2 + if (imagetype == CXIMAGE_FORMAT_JP2) strcpy(szfmt,"jp2"); +#endif +#if CXIMAGE_SUPPORT_JPC + if (imagetype == CXIMAGE_FORMAT_JPC) strcpy(szfmt,"jpc"); +#endif +#if CXIMAGE_SUPPORT_RAS + if (imagetype == CXIMAGE_FORMAT_RAS) strcpy(szfmt,"ras"); +#endif +#if CXIMAGE_SUPPORT_PNM + if (imagetype == CXIMAGE_FORMAT_PNM) strcpy(szfmt,"pnm"); +#endif +#if CXIMAGE_SUPPORT_PGX + if (imagetype == CXIMAGE_FORMAT_PGX){ + strcpy(szfmt,"pgx"); + if (head.biClrUsed==0) throw "PGX can save only GrayScale images"; + } +#endif + int outfmt = jas_image_strtofmt(szfmt); + + char szoutopts[16]; + sprintf(szoutopts,"rate=%.3f", info.nQuality/100.0f); + + if (jas_image_encode(image, out, outfmt, szoutopts)) { + throw "error: cannot encode image\n"; + } + jas_stream_flush(out); + + } catch (char *message) { + strncpy(info.szLastError,message,255); + error = 1; + } + + for (x = 0; x < numcmpts; ++x) { if (cmpts[x]) { jas_matrix_destroy(cmpts[x]); } } + jas_cleanup(); + if (image) jas_image_destroy(image); + if (out) jas_stream_close(out); + + return (error==0); +} +//////////////////////////////////////////////////////////////////////////////// +#endif // CXIMAGE_SUPPORT_ENCODE +//////////////////////////////////////////////////////////////////////////////// +#endif // CXIMAGE_SUPPORT_JASPER + diff --git a/src/win32/dependencies/cximage/ximajas.h b/src/win32/dependencies/cximage/ximajas.h new file mode 100644 index 00000000..3fd1414a --- /dev/null +++ b/src/win32/dependencies/cximage/ximajas.h @@ -0,0 +1,84 @@ +/* + * File: ximajas.h + * Purpose: Jasper Image Class Loader and Writer + */ +/* ========================================================== + * CxImageJAS (c) 12/Apr/2003 Davide Pizzolato - www.xdp.it + * For conditions of distribution and use, see copyright notice in ximage.h + * + * based on JasPer Copyright (c) 2001-2003 Michael David Adams - All rights reserved. + * ========================================================== + */ +#if !defined(__ximaJAS_h) +#define __ximaJAS_h + +#include "ximage.h" + +#if CXIMAGE_SUPPORT_JASPER + +#include "../jasper/include/jasper/jasper.h" + +class CxImageJAS: public CxImage +{ +public: + CxImageJAS(): CxImage((DWORD)0) {} // cast to DWORD + +// bool Load(const char * imageFileName){ return CxImage::Load(imageFileName,0);} +// bool Save(const char * imageFileName){ return CxImage::Save(imageFileName,0);} + bool Decode(CxFile * hFile, DWORD imagetype = 0); + bool Decode(FILE *hFile, DWORD imagetype = 0) { CxIOFile file(hFile); return Decode(&file,imagetype); } + +#if CXIMAGE_SUPPORT_ENCODE + bool Encode(CxFile * hFile, DWORD imagetype = 0); + bool Encode(FILE *hFile, DWORD imagetype = 0) { CxIOFile file(hFile); return Encode(&file,imagetype); } +#endif // CXIMAGE_SUPPORT_ENCODE +protected: + + class CxFileJas + { + public: + CxFileJas(CxFile* pFile,jas_stream_t *stream) + { + if (stream->obj_) jas_free(stream->obj_); + stream->obj_ = pFile; + + // - cannot set the stream->ops_->functions here, + // because this overwrites a static structure in the Jasper library. + // This structure is used by Jasper for internal operations too, e.g. tempfile. + // However the ops_ pointer in the stream can be overwritten. + + //stream->ops_->close_ = JasClose; + //stream->ops_->read_ = JasRead; + //stream->ops_->seek_ = JasSeek; + //stream->ops_->write_ = JasWrite; + + jas_stream_CxFile.close_ = JasClose; + jas_stream_CxFile.read_ = JasRead; + jas_stream_CxFile.seek_ = JasSeek; + jas_stream_CxFile.write_ = JasWrite; + + stream->ops_ = &jas_stream_CxFile; + + // - end + } + static int JasRead(jas_stream_obj_t *obj, char *buf, int cnt) + { return ((CxFile*)obj)->Read(buf,1,cnt); } + static int JasWrite(jas_stream_obj_t *obj, char *buf, int cnt) + { return ((CxFile*)obj)->Write(buf,1,cnt); } + static long JasSeek(jas_stream_obj_t *obj, long offset, int origin) + { return ((CxFile*)obj)->Seek(offset,origin); } + static int JasClose(jas_stream_obj_t *obj) + { return 1; } + + // + private: + jas_stream_ops_t jas_stream_CxFile; + // - end + + }; + +}; + +#endif + +#endif diff --git a/src/win32/dependencies/cximage/ximajbg.cpp b/src/win32/dependencies/cximage/ximajbg.cpp new file mode 100644 index 00000000..1866b3ed --- /dev/null +++ b/src/win32/dependencies/cximage/ximajbg.cpp @@ -0,0 +1,162 @@ +/* + * File: ximajbg.cpp + * Purpose: Platform Independent JBG Image Class Loader and Writer + * 18/Aug/2002 Davide Pizzolato - www.xdp.it + * CxImage version 5.99c 17/Oct/2004 + */ + +#include "ximajbg.h" + +#if CXIMAGE_SUPPORT_JBG + +#include "ximaiter.h" + +#define JBIG_BUFSIZE 8192 + +//////////////////////////////////////////////////////////////////////////////// +bool CxImageJBG::Decode(CxFile *hFile) +{ + if (hFile == NULL) return false; + + struct jbg_dec_state jbig_state; + unsigned long xmax = 4294967295UL, ymax = 4294967295UL; + unsigned int len, cnt; + BYTE *buffer,*p; + int result; + + try + { + jbg_dec_init(&jbig_state); + jbg_dec_maxsize(&jbig_state, xmax, ymax); + + buffer = (BYTE*)malloc(JBIG_BUFSIZE); + if (!buffer) throw "Sorry, not enough memory available!"; + + result = JBG_EAGAIN; + do { + len = hFile->Read(buffer, 1, JBIG_BUFSIZE); + if (!len) break; + cnt = 0; + p = buffer; + while (len > 0 && (result == JBG_EAGAIN || result == JBG_EOK)) { + result = jbg_dec_in(&jbig_state, p, len, &cnt); + p += cnt; + len -= cnt; + } + } while (result == JBG_EAGAIN || result == JBG_EOK); + + if (hFile->Error()) + throw "Problem while reading input file"; + if (result != JBG_EOK && result != JBG_EOK_INTR) + throw "Problem with input file"; + + int w, h, bpp, planes, ew; + + w = jbg_dec_getwidth(&jbig_state); + h = jbg_dec_getheight(&jbig_state); + planes = jbg_dec_getplanes(&jbig_state); + bpp = (planes+7)>>3; + ew = (w + 7)>>3; + + switch (planes){ + case 1: + { + BYTE* binary_image = jbg_dec_getimage(&jbig_state, 0); + + if (!Create(w,h,1,CXIMAGE_FORMAT_JBG)) + throw "Can't allocate memory"; + + SetPaletteColor(0,255,255,255); + SetPaletteColor(1,0,0,0); + + CImageIterator iter(this); + iter.Upset(); + for (int i=0;i>3; + ew = (w + 7)>>3; + + BYTE mask; + RGBQUAD *rgb = GetPalette(); + if (CompareColors(&rgb[0],&rgb[1])<0) mask=255; else mask=0; + + BYTE *buffer = (BYTE*)malloc(ew*h*2); + if (!buffer) { + strcpy(info.szLastError,"Sorry, not enough memory available!"); + return false; + } + + for (y=0; yError()){ + strcpy(info.szLastError,"Problem while writing JBG file"); + return false; + } + + return true; +} +//////////////////////////////////////////////////////////////////////////////// +#endif // CXIMAGE_SUPPORT_JBG + diff --git a/src/win32/dependencies/cximage/ximajbg.h b/src/win32/dependencies/cximage/ximajbg.h new file mode 100644 index 00000000..bf1fdca7 --- /dev/null +++ b/src/win32/dependencies/cximage/ximajbg.h @@ -0,0 +1,44 @@ +/* + * File: ximajbg.h + * Purpose: JBG Image Class Loader and Writer + */ +/* ========================================================== + * CxImageJBG (c) 18/Aug/2002 Davide Pizzolato - www.xdp.it + * For conditions of distribution and use, see copyright notice in ximage.h + * + * based on LIBJBG Copyright (c) 2002, Markus Kuhn - All rights reserved. + * ========================================================== + */ +#if !defined(__ximaJBG_h) +#define __ximaJBG_h + +#include "ximage.h" + +#if CXIMAGE_SUPPORT_JBG + +extern "C" { +#include "../jbig/jbig.h" +}; + +class CxImageJBG: public CxImage +{ +public: + CxImageJBG(): CxImage(CXIMAGE_FORMAT_JBG) {} + +// bool Load(const char * imageFileName){ return CxImage::Load(imageFileName,CXIMAGE_FORMAT_JBG);} +// bool Save(const char * imageFileName){ return CxImage::Save(imageFileName,CXIMAGE_FORMAT_JBG);} + bool Decode(CxFile * hFile); + bool Decode(FILE *hFile) { CxIOFile file(hFile); return Decode(&file); } + +#if CXIMAGE_SUPPORT_ENCODE + bool Encode(CxFile * hFile); + bool Encode(FILE *hFile) { CxIOFile file(hFile); return Encode(&file); } +#endif // CXIMAGE_SUPPORT_ENCODE +protected: + static void jbig_data_out(BYTE *buffer, unsigned int len, void *file) + {((CxFile*)file)->Write(buffer,len,1);} +}; + +#endif + +#endif diff --git a/src/win32/dependencies/cximage/ximajpg.cpp b/src/win32/dependencies/cximage/ximajpg.cpp new file mode 100644 index 00000000..e1e2f36a --- /dev/null +++ b/src/win32/dependencies/cximage/ximajpg.cpp @@ -0,0 +1,454 @@ +/* + * File: ximajpg.cpp + * Purpose: Platform Independent JPEG Image Class Loader and Writer + * 07/Aug/2001 Davide Pizzolato - www.xdp.it + * CxImage version 5.99c 17/Oct/2004 + */ + +#include "ximajpg.h" + +#if CXIMAGE_SUPPORT_JPG + +#include "../jpeg/jmorecfg.h" + +#include "ximaiter.h" + +#include + +struct jpg_error_mgr { + struct jpeg_error_mgr pub; /* "public" fields */ + jmp_buf setjmp_buffer; /* for return to caller */ + char* buffer; /* error message */ +}; +typedef jpg_error_mgr *jpg_error_ptr; + +//////////////////////////////////////////////////////////////////////////////// +// Here's the routine that will replace the standard error_exit method: +//////////////////////////////////////////////////////////////////////////////// +static void +ima_jpeg_error_exit (j_common_ptr cinfo) +{ + /* cinfo->err really points to a my_error_mgr struct, so coerce pointer */ + jpg_error_ptr myerr = (jpg_error_ptr) cinfo->err; + /* Create the message */ + myerr->pub.format_message (cinfo, myerr->buffer); + /* Send it to stderr, adding a newline */ + /* Return control to the setjmp point */ + longjmp(myerr->setjmp_buffer, 1); +} +//////////////////////////////////////////////////////////////////////////////// +CxImageJPG::CxImageJPG(): CxImage(CXIMAGE_FORMAT_JPG) +{ +#if CXIMAGEJPG_SUPPORT_EXIF + m_exif = NULL; + memset(&m_exifinfo, 0, sizeof(EXIFINFO)); +#endif +} +//////////////////////////////////////////////////////////////////////////////// +CxImageJPG::~CxImageJPG() +{ +#if CXIMAGEJPG_SUPPORT_EXIF + if (m_exif) delete m_exif; +#endif +} +//////////////////////////////////////////////////////////////////////////////// +#if CXIMAGEJPG_SUPPORT_EXIF +bool CxImageJPG::DecodeExif(CxFile * hFile) +{ + m_exif = new CxExifInfo(&m_exifinfo); + if (m_exif){ + long pos=hFile->Tell(); + m_exif->DecodeExif(hFile); + hFile->Seek(pos,SEEK_SET); + return m_exif->m_exifinfo->IsExif; + } else { + return false; + } +} +#endif //CXIMAGEJPG_SUPPORT_EXIF +//////////////////////////////////////////////////////////////////////////////// +bool CxImageJPG::Decode(CxFile * hFile) +{ + + bool is_exif = false; +#if CXIMAGEJPG_SUPPORT_EXIF + is_exif = DecodeExif(hFile); +#endif + + CImageIterator iter(this); + /* This struct contains the JPEG decompression parameters and pointers to + * working space (which is allocated as needed by the JPEG library). + */ + struct jpeg_decompress_struct cinfo; + /* We use our private extension JPEG error handler. */ + struct jpg_error_mgr jerr; + jerr.buffer=info.szLastError; + /* More stuff */ + JSAMPARRAY buffer; /* Output row buffer */ + int row_stride; /* physical row width in output buffer */ + + /* In this example we want to open the input file before doing anything else, + * so that the setjmp() error recovery below can assume the file is open. + * VERY IMPORTANT: use "b" option to fopen() if you are on a machine that + * requires it in order to read binary files. + */ + + /* Step 1: allocate and initialize JPEG decompression object */ + /* We set up the normal JPEG error routines, then override error_exit. */ + cinfo.err = jpeg_std_error(&jerr.pub); + jerr.pub.error_exit = ima_jpeg_error_exit; + + /* Establish the setjmp return context for my_error_exit to use. */ + if (setjmp(jerr.setjmp_buffer)) { + /* If we get here, the JPEG code has signaled an error. + * We need to clean up the JPEG object, close the input file, and return. + */ + jpeg_destroy_decompress(&cinfo); + return 0; + } + /* Now we can initialize the JPEG decompression object. */ + jpeg_create_decompress(&cinfo); + + /* Step 2: specify data source (eg, a file) */ + //jpeg_stdio_src(&cinfo, infile); + CxFileJpg src(hFile); + cinfo.src = &src; + + /* Step 3: read file parameters with jpeg_read_header() */ + (void) jpeg_read_header(&cinfo, TRUE); + + /* Step 4 handle decoder options*/ + if ((GetCodecOption(CXIMAGE_FORMAT_JPG) & DECODE_GRAYSCALE) != 0) + cinfo.out_color_space = JCS_GRAYSCALE; + if ((GetCodecOption(CXIMAGE_FORMAT_JPG) & DECODE_QUANTIZE) != 0) { + cinfo.quantize_colors = TRUE; + cinfo.desired_number_of_colors = info.nQuality; + } + if ((GetCodecOption(CXIMAGE_FORMAT_JPG) & DECODE_DITHER) != 0) + cinfo.dither_mode = m_nDither; + if ((GetCodecOption(CXIMAGE_FORMAT_JPG) & DECODE_ONEPASS) != 0) + cinfo.two_pass_quantize = FALSE; + if ((GetCodecOption(CXIMAGE_FORMAT_JPG) & DECODE_NOSMOOTH) != 0) + cinfo.do_fancy_upsampling = FALSE; + +//: Load true color images as RGB (no quantize) +/* Step 4: set parameters for decompression */ +/* if (cinfo.jpeg_color_space!=JCS_GRAYSCALE) { + * cinfo.quantize_colors = TRUE; + * cinfo.desired_number_of_colors = 128; + *} + */ // + + // Set the scale + cinfo.scale_denom = GetJpegScale(); + + // Borrowed the idea from GIF implementation + if (info.nEscape == -1) { + // Return output dimensions only + jpeg_calc_output_dimensions(&cinfo); + head.biWidth = cinfo.output_width; + head.biHeight = cinfo.output_height; + jpeg_destroy_decompress(&cinfo); + return true; + } + + /* Step 5: Start decompressor */ + jpeg_start_decompress(&cinfo); + + /* We may need to do some setup of our own at this point before reading + * the data. After jpeg_start_decompress() we have the correct scaled + * output image dimensions available, as well as the output colormap + * if we asked for color quantization. + */ + //Create the image using output dimensions + //Create(cinfo.image_width, cinfo.image_height, 8*cinfo.output_components, CXIMAGE_FORMAT_JPG); + Create(cinfo.output_width, cinfo.output_height, 8*cinfo.output_components, CXIMAGE_FORMAT_JPG); + + if (!pDib) longjmp(jerr.setjmp_buffer, 1); // check if the image has been created + + if (is_exif){ +#if CXIMAGEJPG_SUPPORT_EXIF + if ((m_exifinfo.Xresolution != 0.0) && (m_exifinfo.ResolutionUnit != 0)) + SetXDPI((long)(m_exifinfo.Xresolution/m_exifinfo.ResolutionUnit)); + if ((m_exifinfo.Yresolution != 0.0) && (m_exifinfo.ResolutionUnit != 0)) + SetYDPI((long)(m_exifinfo.Yresolution/m_exifinfo.ResolutionUnit)); +#endif + } else { + if (cinfo.density_unit==2){ + SetXDPI((long)floor(cinfo.X_density * 254.0 / 10000.0 + 0.5)); + SetYDPI((long)floor(cinfo.Y_density * 254.0 / 10000.0 + 0.5)); + } else { + SetXDPI(cinfo.X_density); + SetYDPI(cinfo.Y_density); + } + } + + if (cinfo.out_color_space==JCS_GRAYSCALE){ + SetGrayPalette(); + head.biClrUsed =256; + } else { + if (cinfo.quantize_colors==TRUE){ + SetPalette(cinfo.actual_number_of_colors, cinfo.colormap[0], cinfo.colormap[1], cinfo.colormap[2]); + head.biClrUsed=cinfo.actual_number_of_colors; + } else { + head.biClrUsed=0; + } + } + + /* JSAMPLEs per row in output buffer */ + row_stride = cinfo.output_width * cinfo.output_components; + + /* Make a one-row-high sample array that will go away when done with image */ + buffer = (*cinfo.mem->alloc_sarray) + ((j_common_ptr) &cinfo, JPOOL_IMAGE, row_stride, 1); + + /* Step 6: while (scan lines remain to be read) */ + /* jpeg_read_scanlines(...); */ + /* Here we use the library's state variable cinfo.output_scanline as the + * loop counter, so that we don't have to keep track ourselves. + */ + iter.Upset(); + while (cinfo.output_scanline < cinfo.output_height) { + + if (info.nEscape) longjmp(jerr.setjmp_buffer, 1); // - cancel decoding + + (void) jpeg_read_scanlines(&cinfo, buffer, 1); + // info.nProgress = (long)(100*cinfo.output_scanline/cinfo.output_height); + // Step 6a: CMYK->RGB */ + if ((cinfo.num_components==4)&&(cinfo.quantize_colors==FALSE)){ + BYTE k,*dst,*src; + dst=iter.GetRow(); + src=buffer[0]; + for(long x3=0,x4=0; x3<(long)info.dwEffWidth && x4 Step 7A: Swap red and blue components + // not necessary if swapped red and blue definition in jmorecfg.h;ln322 + if ((cinfo.num_components==3)&&(cinfo.quantize_colors==FALSE)){ + BYTE* r0=GetBits(); + for(long y=0;y - cancel decoding + RGBtoBGR(r0,3*head.biWidth); + r0+=info.dwEffWidth; + } + } + + /* Step 8: Release JPEG decompression object */ + /* This is an important step since it will release a good deal of memory. */ + jpeg_destroy_decompress(&cinfo); + + /* At this point you may want to check to see whether any corrupt-data + * warnings occurred (test whether jerr.pub.num_warnings is nonzero). + */ + + /* And we're done! */ + return true; +} +//////////////////////////////////////////////////////////////////////////////// +#if CXIMAGE_SUPPORT_ENCODE +//////////////////////////////////////////////////////////////////////////////// +bool CxImageJPG::Encode(CxFile * hFile) +{ + if (EncodeSafeCheck(hFile)) return false; + + if (head.biClrUsed!=0 && !IsGrayScale()){ + strcpy(info.szLastError,"JPEG can save only RGB or GreyScale images"); + return false; + } + + // necessary for EXIF, and for roll backs + long pos=hFile->Tell(); + + /* This struct contains the JPEG compression parameters and pointers to + * working space (which is allocated as needed by the JPEG library). + * It is possible to have several such structures, representing multiple + * compression/decompression processes, in existence at once. We refer + * to any one struct (and its associated working data) as a "JPEG object". + */ + struct jpeg_compress_struct cinfo; + /* This struct represents a JPEG error handler. It is declared separately + * because applications often want to supply a specialized error handler + * (see the second half of this file for an example). But here we just + * take the easy way out and use the standard error handler, which will + * print a message on stderr and call exit() if compression fails. + * Note that this struct must live as long as the main JPEG parameter + * struct, to avoid dangling-pointer problems. + */ + //struct jpeg_error_mgr jerr; + /* We use our private extension JPEG error handler. */ + struct jpg_error_mgr jerr; + jerr.buffer=info.szLastError; + /* More stuff */ + int row_stride; /* physical row width in image buffer */ + JSAMPARRAY buffer; /* Output row buffer */ + + /* Step 1: allocate and initialize JPEG compression object */ + /* We have to set up the error handler first, in case the initialization + * step fails. (Unlikely, but it could happen if you are out of memory.) + * This routine fills in the contents of struct jerr, and returns jerr's + * address which we place into the link field in cinfo. + */ + //cinfo.err = jpeg_std_error(&jerr); + /* We set up the normal JPEG error routines, then override error_exit. */ + cinfo.err = jpeg_std_error(&jerr.pub); + jerr.pub.error_exit = ima_jpeg_error_exit; + + /* Establish the setjmp return context for my_error_exit to use. */ + if (setjmp(jerr.setjmp_buffer)) { + /* If we get here, the JPEG code has signaled an error. + * We need to clean up the JPEG object, close the input file, and return. + */ + strcpy(info.szLastError, jerr.buffer); // + jpeg_destroy_compress(&cinfo); + return 0; + } + + /* Now we can initialize the JPEG compression object. */ + jpeg_create_compress(&cinfo); + /* Step 2: specify data destination (eg, a file) */ + /* Note: steps 2 and 3 can be done in either order. */ + /* Here we use the library-supplied code to send compressed data to a + * stdio stream. You can also write your own code to do something else. + * VERY IMPORTANT: use "b" option to fopen() if you are on a machine that + * requires it in order to write binary files. + */ + + //jpeg_stdio_dest(&cinfo, outfile); + CxFileJpg dest(hFile); + cinfo.dest = &dest; + + /* Step 3: set parameters for compression */ + /* First we supply a description of the input image. + * Four fields of the cinfo struct must be filled in: + */ + cinfo.image_width = GetWidth(); // image width and height, in pixels + cinfo.image_height = GetHeight(); + + if (IsGrayScale()){ + cinfo.input_components = 1; // # of color components per pixel + cinfo.in_color_space = JCS_GRAYSCALE; /* colorspace of input image */ + } else { + cinfo.input_components = 3; // # of color components per pixel + cinfo.in_color_space = JCS_RGB; /* colorspace of input image */ + } + + /* Now use the library's routine to set default compression parameters. + * (You must set at least cinfo.in_color_space before calling this, + * since the defaults depend on the source color space.) + */ + jpeg_set_defaults(&cinfo); + /* Now you can set any non-default parameters you wish to. + * Here we just illustrate the use of quality (quantization table) scaling: + */ + +//#ifdef C_ARITH_CODING_SUPPORTED + if ((GetCodecOption(CXIMAGE_FORMAT_JPG) & ENCODE_ARITHMETIC) != 0) + cinfo.arith_code = TRUE; +//#endif + +//#ifdef ENTROPY_OPT_SUPPORTED + if ((GetCodecOption(CXIMAGE_FORMAT_JPG) & ENCODE_OPTIMIZE) != 0) + cinfo.optimize_coding = TRUE; +//#endif + + if ((GetCodecOption(CXIMAGE_FORMAT_JPG) & ENCODE_GRAYSCALE) != 0) + jpeg_set_colorspace(&cinfo, JCS_GRAYSCALE); + + if ((GetCodecOption(CXIMAGE_FORMAT_JPG) & ENCODE_SMOOTHING) != 0) + cinfo.smoothing_factor = m_nSmoothing; + + jpeg_set_quality(&cinfo, GetJpegQuality(), (GetCodecOption(CXIMAGE_FORMAT_JPG) & ENCODE_BASELINE) != 0); + +//#ifdef C_PROGRESSIVE_SUPPORTED + if ((GetCodecOption(CXIMAGE_FORMAT_JPG) & ENCODE_PROGRESSIVE) != 0) + jpeg_simple_progression(&cinfo); +//#endif + +#ifdef C_LOSSLESS_SUPPORTED + if ((GetCodecOption(CXIMAGE_FORMAT_JPG) & ENCODE_LOSSLESS) != 0) + jpeg_simple_lossless(&cinfo, m_nPredictor, m_nPointTransform); +#endif + + cinfo.density_unit=1; + cinfo.X_density=(unsigned short)GetXDPI(); + cinfo.Y_density=(unsigned short)GetYDPI(); + + /* Step 4: Start compressor */ + /* TRUE ensures that we will write a complete interchange-JPEG file. + * Pass TRUE unless you are very sure of what you're doing. + */ + jpeg_start_compress(&cinfo, TRUE); + + /* Step 5: while (scan lines remain to be written) */ + /* jpeg_write_scanlines(...); */ + /* Here we use the library's state variable cinfo.next_scanline as the + * loop counter, so that we don't have to keep track ourselves. + * To keep things simple, we pass one scanline per call; you can pass + * more if you wish, though. + */ + row_stride = info.dwEffWidth; /* JSAMPLEs per row in image_buffer */ + + // "8+row_stride" fix heap deallocation problem during debug??? + buffer = (*cinfo.mem->alloc_sarray) + ((j_common_ptr) &cinfo, JPOOL_IMAGE, 8+row_stride, 1); + + CImageIterator iter(this); + + iter.Upset(); + while (cinfo.next_scanline < cinfo.image_height) { + // info.nProgress = (long)(100*cinfo.next_scanline/cinfo.image_height); + iter.GetRow(buffer[0], row_stride); + // not necessary if swapped red and blue definition in jmorecfg.h;ln322 + if (head.biClrUsed==0){ // swap R & B for RGB images + RGBtoBGR(buffer[0], row_stride); // Lance : 1998/09/01 : Bug ID: EXP-2.1.1-9 + } + iter.PrevRow(); + (void) jpeg_write_scanlines(&cinfo, buffer, 1); + } + + /* Step 6: Finish compression */ + jpeg_finish_compress(&cinfo); + + /* Step 7: release JPEG compression object */ + /* This is an important step since it will release a good deal of memory. */ + jpeg_destroy_compress(&cinfo); + + +#if CXIMAGEJPG_SUPPORT_EXIF + if (m_exif && m_exif->m_exifinfo->IsExif){ + // discard useless sections (if any) read from original image + m_exif->DiscardAllButExif(); + // read new created image, to split the sections + hFile->Seek(pos,SEEK_SET); + m_exif->DecodeExif(hFile,EXIF_READ_IMAGE); + // save back the image, adding EXIF section + hFile->Seek(pos,SEEK_SET); + m_exif->EncodeExif(hFile); + } +#endif + + + /* And we're done! */ + return true; +} +//////////////////////////////////////////////////////////////////////////////// +#endif // CXIMAGE_SUPPORT_ENCODE +//////////////////////////////////////////////////////////////////////////////// +#endif // CXIMAGE_SUPPORT_JPG + diff --git a/src/win32/dependencies/cximage/ximajpg.h b/src/win32/dependencies/cximage/ximajpg.h new file mode 100644 index 00000000..8d5f5cec --- /dev/null +++ b/src/win32/dependencies/cximage/ximajpg.h @@ -0,0 +1,315 @@ +/* + * File: ximajpg.h + * Purpose: JPG Image Class Loader and Writer + */ +/* ========================================================== + * CxImageJPG (c) 07/Aug/2001 Davide Pizzolato - www.xdp.it + * For conditions of distribution and use, see copyright notice in ximage.h + * + * Special thanks to Troels Knakkergaard for new features, enhancements and bugfixes + * + * Special thanks to Chris Shearer Cooper for CxFileJpg tips & code + * + * EXIF support based on jhead-1.8 by Matthias Wandel + * + * original CImageJPG and CImageIterator implementation are: + * Copyright: (c) 1995, Alejandro Aguilar Sierra + * + * This software is based in part on the work of the Independent JPEG Group. + * Copyright (C) 1991-1998, Thomas G. Lane. + * ========================================================== + */ +#if !defined(__ximaJPEG_h) +#define __ixmaJPEG_h + +#include "ximage.h" + +#if CXIMAGE_SUPPORT_JPG + +#define CXIMAGEJPG_SUPPORT_EXIF 1 + +extern "C" { + #include "../jpeg/jpeglib.h" + #include "../jpeg/jerror.h" +} + +class DLL_EXP CxImageJPG: public CxImage +{ +public: + CxImageJPG(); + ~CxImageJPG(); + +// bool Load(const char * imageFileName){ return CxImage::Load(imageFileName,CXIMAGE_FORMAT_JPG);} +// bool Save(const char * imageFileName){ return CxImage::Save(imageFileName,CXIMAGE_FORMAT_JPG);} + bool Decode(CxFile * hFile); + bool Decode(FILE *hFile) { CxIOFile file(hFile); return Decode(&file); } + +#if CXIMAGE_SUPPORT_ENCODE + bool Encode(CxFile * hFile); + bool Encode(FILE *hFile) { CxIOFile file(hFile); return Encode(&file); } +#endif // CXIMAGE_SUPPORT_ENCODE + +/* + * EXIF support based on jhead-1.8 by Matthias Wandel + */ + +#if CXIMAGEJPG_SUPPORT_EXIF + +#define MAX_COMMENT 1000 +#define MAX_SECTIONS 20 + +typedef struct tag_ExifInfo { + char Version [5]; + char CameraMake [32]; + char CameraModel [40]; + char DateTime [20]; + int Height, Width; + int Orientation; + int IsColor; + int Process; + int FlashUsed; + float FocalLength; + float ExposureTime; + float ApertureFNumber; + float Distance; + float CCDWidth; + float ExposureBias; + int Whitebalance; + int MeteringMode; + int ExposureProgram; + int ISOequivalent; + int CompressionLevel; + float FocalplaneXRes; + float FocalplaneYRes; + float FocalplaneUnits; + float Xresolution; + float Yresolution; + float ResolutionUnit; + float Brightness; + char Comments[MAX_COMMENT]; + + unsigned char * ThumbnailPointer; /* Pointer at the thumbnail */ + unsigned ThumbnailSize; /* Size of thumbnail. */ + + bool IsExif; +} EXIFINFO; + +//-------------------------------------------------------------------------- +// JPEG markers consist of one or more 0xFF bytes, followed by a marker +// code byte (which is not an FF). Here are the marker codes of interest +// in this program. (See jdmarker.c for a more complete list.) +//-------------------------------------------------------------------------- + +#define M_SOF0 0xC0 // Start Of Frame N +#define M_SOF1 0xC1 // N indicates which compression process +#define M_SOF2 0xC2 // Only SOF0-SOF2 are now in common use +#define M_SOF3 0xC3 +#define M_SOF5 0xC5 // NB: codes C4 and CC are NOT SOF markers +#define M_SOF6 0xC6 +#define M_SOF7 0xC7 +#define M_SOF9 0xC9 +#define M_SOF10 0xCA +#define M_SOF11 0xCB +#define M_SOF13 0xCD +#define M_SOF14 0xCE +#define M_SOF15 0xCF +#define M_SOI 0xD8 // Start Of Image (beginning of datastream) +#define M_EOI 0xD9 // End Of Image (end of datastream) +#define M_SOS 0xDA // Start Of Scan (begins compressed data) +#define M_JFIF 0xE0 // Jfif marker +#define M_EXIF 0xE1 // Exif marker +#define M_COM 0xFE // COMment + +#define PSEUDO_IMAGE_MARKER 0x123; // Extra value. + +#define EXIF_READ_EXIF 0x01 +#define EXIF_READ_IMAGE 0x02 +#define EXIF_READ_ALL 0x03 + +class DLL_EXP CxExifInfo +{ + +typedef struct tag_Section_t{ + BYTE* Data; + int Type; + unsigned Size; +} Section_t; + +public: + EXIFINFO* m_exifinfo; + char m_szLastError[256]; + CxExifInfo(EXIFINFO* info = NULL); + ~CxExifInfo(); + bool DecodeExif(CxFile * hFile, int nReadMode = EXIF_READ_EXIF); + bool EncodeExif(CxFile * hFile); + void DiscardAllButExif(); +protected: + bool process_EXIF(unsigned char * CharBuf, unsigned int length); + void process_COM (const BYTE * Data, int length); + void process_SOFn (const BYTE * Data, int marker); + int Get16u(void * Short); + int Get16m(void * Short); + long Get32s(void * Long); + unsigned long Get32u(void * Long); + double ConvertAnyFormat(void * ValuePtr, int Format); + void* FindSection(int SectionType); + bool ProcessExifDir(unsigned char * DirStart, unsigned char * OffsetBase, unsigned ExifLength, + EXIFINFO * const pInfo, unsigned char ** const LastExifRefdP); + int ExifImageWidth; + int MotorolaOrder; + Section_t Sections[MAX_SECTIONS]; + int SectionsRead; + bool freeinfo; +}; + + CxExifInfo* m_exif; + EXIFINFO m_exifinfo; + bool DecodeExif(CxFile * hFile); + bool DecodeExif(FILE * hFile) { CxIOFile file(hFile); return DecodeExif(&file); } + +#endif //CXIMAGEJPG_SUPPORT_EXIF + +//////////////////////////////////////////////////////////////////////////////////////// +////////////////////// C x F i l e J p g //////////////////////////////// +//////////////////////////////////////////////////////////////////////////////////////// + +// thanks to Chris Shearer Cooper +class CxFileJpg : public jpeg_destination_mgr, public jpeg_source_mgr + { +public: + enum { eBufSize = 4096 }; + + CxFileJpg(CxFile* pFile) + { + m_pFile = pFile; + + init_destination = InitDestination; + empty_output_buffer = EmptyOutputBuffer; + term_destination = TermDestination; + + init_source = InitSource; + fill_input_buffer = FillInputBuffer; + skip_input_data = SkipInputData; + resync_to_restart = jpeg_resync_to_restart; // use default method + term_source = TermSource; + next_input_byte = NULL; //* => next byte to read from buffer + bytes_in_buffer = 0; //* # of bytes remaining in buffer + + m_pBuffer = new unsigned char[eBufSize]; + } + ~CxFileJpg() + { + delete [] m_pBuffer; + } + + static void InitDestination(j_compress_ptr cinfo) + { + CxFileJpg* pDest = (CxFileJpg*)cinfo->dest; + pDest->next_output_byte = pDest->m_pBuffer; + pDest->free_in_buffer = eBufSize; + } + + static boolean EmptyOutputBuffer(j_compress_ptr cinfo) + { + CxFileJpg* pDest = (CxFileJpg*)cinfo->dest; + if (pDest->m_pFile->Write(pDest->m_pBuffer,1,eBufSize)!=(size_t)eBufSize) + ERREXIT(cinfo, JERR_FILE_WRITE); + pDest->next_output_byte = pDest->m_pBuffer; + pDest->free_in_buffer = eBufSize; + return TRUE; + } + + static void TermDestination(j_compress_ptr cinfo) + { + CxFileJpg* pDest = (CxFileJpg*)cinfo->dest; + size_t datacount = eBufSize - pDest->free_in_buffer; + /* Write any data remaining in the buffer */ + if (datacount > 0) { + if (!pDest->m_pFile->Write(pDest->m_pBuffer,1,datacount)) + ERREXIT(cinfo, JERR_FILE_WRITE); + } + pDest->m_pFile->Flush(); + /* Make sure we wrote the output file OK */ + if (pDest->m_pFile->Error()) ERREXIT(cinfo, JERR_FILE_WRITE); + return; + } + + static void InitSource(j_decompress_ptr cinfo) + { + CxFileJpg* pSource = (CxFileJpg*)cinfo->src; + pSource->m_bStartOfFile = TRUE; + } + + static boolean FillInputBuffer(j_decompress_ptr cinfo) + { + size_t nbytes; + CxFileJpg* pSource = (CxFileJpg*)cinfo->src; + nbytes = pSource->m_pFile->Read(pSource->m_pBuffer,1,eBufSize); + if (nbytes <= 0){ + if (pSource->m_bStartOfFile) //* Treat empty input file as fatal error + ERREXIT(cinfo, JERR_INPUT_EMPTY); + WARNMS(cinfo, JWRN_JPEG_EOF); + // Insert a fake EOI marker + pSource->m_pBuffer[0] = (JOCTET) 0xFF; + pSource->m_pBuffer[1] = (JOCTET) JPEG_EOI; + nbytes = 2; + } + pSource->next_input_byte = pSource->m_pBuffer; + pSource->bytes_in_buffer = nbytes; + pSource->m_bStartOfFile = FALSE; + return TRUE; + } + + static void SkipInputData(j_decompress_ptr cinfo, long num_bytes) + { + CxFileJpg* pSource = (CxFileJpg*)cinfo->src; + if (num_bytes > 0){ + while (num_bytes > (long)pSource->bytes_in_buffer){ + num_bytes -= (long)pSource->bytes_in_buffer; + FillInputBuffer(cinfo); + // note we assume that fill_input_buffer will never return FALSE, + // so suspension need not be handled. + } + pSource->next_input_byte += (size_t) num_bytes; + pSource->bytes_in_buffer -= (size_t) num_bytes; + } + } + + static void TermSource(j_decompress_ptr cinfo) + { + return; + } +protected: + CxFile *m_pFile; + unsigned char *m_pBuffer; + bool m_bStartOfFile; +}; + +public: + enum CODEC_OPTION + { + ENCODE_BASELINE = 0x1, + ENCODE_ARITHMETIC = 0x2, + ENCODE_GRAYSCALE = 0x4, + ENCODE_OPTIMIZE = 0x8, + ENCODE_PROGRESSIVE = 0x10, + ENCODE_LOSSLESS = 0x20, + ENCODE_SMOOTHING = 0x40, + DECODE_GRAYSCALE = 0x80, + DECODE_QUANTIZE = 0x100, + DECODE_DITHER = 0x200, + DECODE_ONEPASS = 0x400, + DECODE_NOSMOOTH = 0x800 + }; + + int m_nPredictor; + int m_nPointTransform; + int m_nSmoothing; + int m_nQuantize; + J_DITHER_MODE m_nDither; + +}; + +#endif + +#endif diff --git a/src/win32/dependencies/cximage/ximalpha.cpp b/src/win32/dependencies/cximage/ximalpha.cpp new file mode 100644 index 00000000..b92f33bc --- /dev/null +++ b/src/win32/dependencies/cximage/ximalpha.cpp @@ -0,0 +1,321 @@ +// xImalpha.cpp : Alpha channel functions +/* 07/08/2001 v1.00 - Davide Pizzolato - www.xdp.it + * CxImage version 5.99c 17/Oct/2004 + */ + +#include "ximage.h" + +#if CXIMAGE_SUPPORT_ALPHA + +//////////////////////////////////////////////////////////////////////////////// +/** + * \sa AlphaSetMax + */ +BYTE CxImage::AlphaGetMax() const +{ + return info.nAlphaMax; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Sets global Alpha (opacity) value applied to the whole image, + * valid only for painting functions. + * \param nAlphaMax: can be from 0 to 255 + */ +void CxImage::AlphaSetMax(BYTE nAlphaMax) +{ + info.nAlphaMax=nAlphaMax; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Checks if the image has a valid alpha channel. + */ +bool CxImage::AlphaIsValid() +{ + return pAlpha!=0; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Enables the alpha palette, so the Draw() function changes its behavior. + */ +void CxImage::AlphaPaletteEnable(bool enable) +{ + info.bAlphaPaletteEnabled=enable; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * True if the alpha palette is enabled for painting. + */ +bool CxImage::AlphaPaletteIsEnabled() +{ + return info.bAlphaPaletteEnabled; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Inverts the alpha channel. + */ +void CxImage::AlphaClear() +{ + if (pAlpha) memset(pAlpha,0,head.biWidth * head.biHeight); +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Sets the alpha level for the whole image + */ +void CxImage::AlphaSet(BYTE level) +{ + if (pAlpha) memset(pAlpha,level,head.biWidth * head.biHeight); +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Allocates an empty (opaque) alpha channel. + */ +void CxImage::AlphaCreate() +{ + if (pAlpha==NULL) { + pAlpha = (BYTE*)malloc(head.biWidth * head.biHeight); + if (pAlpha) memset(pAlpha,255,head.biWidth * head.biHeight); + } +} +//////////////////////////////////////////////////////////////////////////////// +void CxImage::AlphaDelete() +{ + if (pAlpha) { free(pAlpha); pAlpha=0; } +} +//////////////////////////////////////////////////////////////////////////////// +void CxImage::AlphaInvert() +{ + if (pAlpha) { + BYTE *iSrc=pAlpha; + long n=head.biHeight*head.biWidth; + for(long i=0; i < n; i++){ + *iSrc=(BYTE)~(*(iSrc)); + iSrc++; + } + } +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Imports an existing alpa channel from another image with the same width and height. + */ +bool CxImage::AlphaCopy(CxImage &from) +{ + if (from.pAlpha == NULL || head.biWidth != from.head.biWidth || head.biHeight != from.head.biHeight) return false; + if (pAlpha==NULL) pAlpha = (BYTE*)malloc(head.biWidth * head.biHeight); + if (pAlpha==NULL) return false; + memcpy(pAlpha,from.pAlpha,head.biWidth * head.biHeight); + info.nAlphaMax=from.info.nAlphaMax; + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Creates the alpha channel from a gray scale image. + */ +bool CxImage::AlphaSet(CxImage &from) +{ + if (!from.IsGrayScale() || head.biWidth != from.head.biWidth || head.biHeight != from.head.biHeight) return false; + if (pAlpha==NULL) pAlpha = (BYTE*)malloc(head.biWidth * head.biHeight); + BYTE* src = from.info.pImage; + BYTE* dst = pAlpha; + if (src==NULL || dst==NULL) return false; + for (long y=0; yTransfer(tmp); + + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Exports the alpha palette channel in a 8bpp grayscale image. + */ +bool CxImage::AlphaPaletteSplit(CxImage *dest) +{ + if (!AlphaPaletteIsValid() || !dest) return false; + + CxImage tmp(head.biWidth,head.biHeight,8); + if (!tmp.IsValid()) return false; + + for(long y=0; yTransfer(tmp); + + return true; +} +//////////////////////////////////////////////////////////////////////////////// +#endif //CXIMAGE_SUPPORT_ALPHA diff --git a/src/win32/dependencies/cximage/ximalyr.cpp b/src/win32/dependencies/cximage/ximalyr.cpp new file mode 100644 index 00000000..04301232 --- /dev/null +++ b/src/win32/dependencies/cximage/ximalyr.cpp @@ -0,0 +1,105 @@ +// xImaLyr.cpp : Layers functions +/* 21/04/2003 v1.00 - Davide Pizzolato - www.xdp.it + * CxImage version 5.99c 17/Oct/2004 + */ + +#include "ximage.h" + +#if CXIMAGE_SUPPORT_LAYERS + +//////////////////////////////////////////////////////////////////////////////// +/** + * If the object is an internal layer, GetParent return its parent in the hierarchy. + */ +CxImage* CxImage::GetParent() const +{ + return info.pParent; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Number of layers allocated directly by the object. + */ +long CxImage::GetNumLayers() const +{ + return info.nNumLayers; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Creates an empty layer. If position is less than 0, the new layer will be placed in the last position + */ +bool CxImage::LayerCreate(long position) +{ + if ( position < 0 || position > info.nNumLayers ) position = info.nNumLayers; + + CxImage** ptmp = (CxImage**)malloc((info.nNumLayers + 1)*sizeof(CxImage**)); + if (ptmp==0) return false; + + int i=0; + for (int n=0; ninfo.pParent = this; + } else { + free(ptmp); + return false; + } + + info.nNumLayers++; + if (pLayers) free(pLayers); + pLayers = ptmp; + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Deletes a layer. If position is less than 0, the last layer will be deleted + */ +bool CxImage::LayerDelete(long position) +{ + if ( position >= info.nNumLayers ) return false; + if ( position < 0) position = info.nNumLayers - 1; + + CxImage** ptmp = (CxImage**)malloc((info.nNumLayers - 1)*sizeof(CxImage**)); + if (ptmp==0) return false; + + int i=0; + for (int n=0; n<(info.nNumLayers - 1); n++){ + if (position == n){ + delete pLayers[n]; + i=1; + } + ptmp[n]=pLayers[n+i]; + } + if (i==0) delete pLayers[info.nNumLayers - 1]; + + info.nNumLayers--; + if (pLayers) free(pLayers); + pLayers = ptmp; + return true; +} +//////////////////////////////////////////////////////////////////////////////// +void CxImage::LayerDeleteAll() +{ + if (pLayers) { + for(long n=0; n= info.nNumLayers ) return 0; + if ( position < 0) position = info.nNumLayers - 1; + return pLayers[position]; +} +//////////////////////////////////////////////////////////////////////////////// +#endif //CXIMAGE_SUPPORT_LAYERS diff --git a/src/win32/dependencies/cximage/ximamng.cpp b/src/win32/dependencies/cximage/ximamng.cpp new file mode 100644 index 00000000..1c43aa9a --- /dev/null +++ b/src/win32/dependencies/cximage/ximamng.cpp @@ -0,0 +1,366 @@ +/* + * File: ximamng.cpp + * Purpose: Platform Independent MNG Image Class Loader and Writer + * Author: 07/Aug/2001 Davide Pizzolato - www.xdp.it + * CxImage version 5.99c 17/Oct/2004 + */ + +#include "ximamng.h" + +#if CXIMAGE_SUPPORT_MNG + +//////////////////////////////////////////////////////////////////////////////// +// callbacks for the mng decoder: +//////////////////////////////////////////////////////////////////////////////// + +//////////////////////////////////////////////////////////////////////////////// +// memory allocation; data must be zeroed +static mng_ptr +mymngalloc( mng_uint32 size ) +{ + return (mng_ptr)calloc(1, size); +} + +//////////////////////////////////////////////////////////////////////////////// +// memory deallocation +static void mymngfree(mng_ptr p, mng_uint32 size) +{ + free(p); +} + +//////////////////////////////////////////////////////////////////////////////// +// Stream open/close: +// since the user is responsible for opening and closing the file, +// we leave the default implementation open +static mng_bool mymngopenstream(mng_handle mng) { return MNG_TRUE; } +static mng_bool mymngopenstreamwrite(mng_handle mng) { return MNG_TRUE; } +static mng_bool mymngclosestream(mng_handle mng) { return MNG_TRUE; } + +//////////////////////////////////////////////////////////////////////////////// +// feed data to the decoder +static mng_bool mymngreadstream(mng_handle mng, mng_ptr buffer, mng_uint32 size, mng_uint32 *bytesread) +{ + mngstuff *mymng = (mngstuff *)mng_get_userdata(mng); + // read the requested amount of data from the file + *bytesread = mymng->file->Read( buffer, sizeof(BYTE), size); + return MNG_TRUE; +} + +//////////////////////////////////////////////////////////////////////////////// +static mng_bool mymngwritestream (mng_handle mng, mng_ptr pBuf, mng_uint32 iSize, mng_uint32 *iWritten) +{ + mngstuff *mymng = (mngstuff *)mng_get_userdata(mng); + // write it + *iWritten = mymng->file->Write (pBuf, 1, iSize); + return MNG_TRUE; +} + +//////////////////////////////////////////////////////////////////////////////// +// the header's been read. set up the display stuff +static mng_bool mymngprocessheader( mng_handle mng, mng_uint32 width, mng_uint32 height ) +{ + // normally the image buffer is allocated here, + // but in this module we don't know nothing about + // the final environment. + + mngstuff *mymng = (mngstuff *)mng_get_userdata(mng); + + mymng->width = width; + mymng->height = height; + mymng->bpp = 24; + mymng->effwdt = ((((width * mymng->bpp) + 31) >> 5) << 2); + + if (mng->bUseBKGD){ + mymng->nBkgndIndex = 0; + mymng->nBkgndColor.rgbRed = mng->iBGred >> 8; + mymng->nBkgndColor.rgbGreen =mng->iBGgreen >> 8; + mymng->nBkgndColor.rgbBlue = mng->iBGblue >> 8; + } + + mymng->image = (BYTE*)malloc(height * mymng->effwdt); + + // tell the mng decoder about our bit-depth choice + mng_set_canvasstyle( mng, MNG_CANVAS_BGR8 ); + return MNG_TRUE; +} + +//////////////////////////////////////////////////////////////////////////////// +// return a row pointer for the decoder to fill +static mng_ptr mymnggetcanvasline( mng_handle mng, mng_uint32 line ) +{ + mngstuff *mymng = (mngstuff *)mng_get_userdata(mng); + return (mng_ptr)(mymng->image + (mymng->effwdt * (mymng->height - 1 - line))); +} + +//////////////////////////////////////////////////////////////////////////////// +// timer +static mng_uint32 mymnggetticks(mng_handle mng) +{ +#ifdef WIN32 + return (mng_uint32)GetTickCount(); +#else + return 0; +#endif +} + +//////////////////////////////////////////////////////////////////////////////// +// Refresh: actual frame need to be updated (Invalidate) +static mng_bool mymngrefresh(mng_handle mng, mng_uint32 x, mng_uint32 y, mng_uint32 w, mng_uint32 h) +{ +// mngstuff *mymng = (mngstuff *)mng_get_userdata(mng); + return MNG_TRUE; +} + +//////////////////////////////////////////////////////////////////////////////// +// interframe delay callback +static mng_bool mymngsettimer(mng_handle mng, mng_uint32 msecs) +{ + mngstuff *mymng = (mngstuff *)mng_get_userdata(mng); + mymng->delay = msecs; // set the timer for when the decoder wants to be woken + return MNG_TRUE; +} + +//////////////////////////////////////////////////////////////////////////////// +static mng_bool mymngerror(mng_handle mng, mng_int32 code, mng_int8 severity, mng_chunkid chunktype, mng_uint32 chunkseq, mng_int32 extra1, mng_int32 extra2, mng_pchar text) +{ + //throw (const char *)text; + return mng_cleanup(&mng); // +} + +//////////////////////////////////////////////////////////////////////////////// +// CxImage members +//////////////////////////////////////////////////////////////////////////////// +CxImageMNG::CxImageMNG(): CxImage(CXIMAGE_FORMAT_MNG) +{ + hmng = NULL; + memset(&mnginfo,0,sizeof(mngstuff)); + mnginfo.nBkgndIndex = -1; + mnginfo.speed = 1.0f; +} +//////////////////////////////////////////////////////////////////////////////// +CxImageMNG::~CxImageMNG() +{ + // cleanup and return + if (mnginfo.thread){ //close the animation thread + mnginfo.animation_enabled=0; + ResumeThread(mnginfo.thread); + WaitForSingleObject(mnginfo.thread,500); + CloseHandle(mnginfo.thread); + } + // free objects + if (mnginfo.image) free(mnginfo.image); + if (hmng) mng_cleanup(&hmng); //be sure it's not needed any more. (active timers ?) +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageMNG::SetCallbacks(mng_handle mng) +{ + // set the callbacks + mng_setcb_errorproc(mng, mymngerror); + mng_setcb_openstream(mng, mymngopenstream); + mng_setcb_closestream(mng, mymngclosestream); + mng_setcb_readdata(mng, mymngreadstream); + mng_setcb_processheader(mng, mymngprocessheader); + mng_setcb_getcanvasline(mng, mymnggetcanvasline); + mng_setcb_refresh(mng, mymngrefresh); + mng_setcb_gettickcount(mng, mymnggetticks); + mng_setcb_settimer(mng, mymngsettimer); + mng_setcb_refresh(mng, mymngrefresh); +} +//////////////////////////////////////////////////////////////////////////////// +// can't use the CxImage implementation because it looses mnginfo +bool CxImageMNG::Load(const char * imageFileName){ + FILE* hFile; //file handle to read the image + if ((hFile=fopen(imageFileName,"rb"))==NULL) return false; + bool bOK = Decode(hFile); + fclose(hFile); + return bOK; +} +//////////////////////////////////////////////////////////////////////////////// +bool CxImageMNG::Decode(CxFile *hFile) +{ + if (hFile == NULL) return false; + + try { + // set up the mng decoder for our stream + hmng = mng_initialize(&mnginfo, mymngalloc, mymngfree, MNG_NULL); + if (hmng == NULL) throw "could not initialize libmng"; + + // set the file we want to play + mnginfo.file = hFile; + + // Set the colorprofile, lcms uses this: + mng_set_srgb(hmng, MNG_TRUE ); + // Set white as background color: + WORD Red,Green,Blue; + Red = Green = Blue = (255 << 8) + 255; + mng_set_bgcolor(hmng, Red, Green, Blue ); + // If PNG Background is available, use it: + mng_set_usebkgd(hmng, MNG_TRUE ); + + // No need to store chunks: + mng_set_storechunks(hmng, MNG_FALSE); + // No need to wait: straight reading + mng_set_suspensionmode(hmng, MNG_FALSE); + + SetCallbacks(hmng); + + mng_datap pData = (mng_datap)hmng; + + // read in the image + info.nNumFrames=0; + mng_readdisplay(hmng); + + // read all + int retval=MNG_NOERROR; + while(pData->bReading){ + retval = mng_display_resume(hmng); + info.nNumFrames++; + } + + // single frame check: + if (retval != MNG_NEEDTIMERWAIT){ + info.nNumFrames--; + } else { + mnginfo.animation=1; + } + + if (info.nNumFrames<=0) info.nNumFrames=1; + + if (mnginfo.animation_enabled==0){ + // select the frame + if (info.nFrame>=0 && info.nFramewidth,OffsetH=mymng->height; + + BYTE *tmpbuffer = new BYTE[ (mymng->effwdt+1) * mymng->height]; + if( tmpbuffer == 0 ) return; + + // Write DEFI chunk. + mng_putchunk_defi( hMNG, 0, 0, 0, MNG_TRUE, OffsetX, OffsetY, MNG_FALSE, 0, 0, 0, 0 ); + + // Write Header: + mng_putchunk_ihdr( + hMNG, + OffsetW, OffsetH, + MNG_BITDEPTH_8, + MNG_COLORTYPE_RGB, + MNG_COMPRESSION_DEFLATE, + MNG_FILTER_ADAPTIVE, + MNG_INTERLACE_NONE + ); + + // transfer data, add Filterbyte: + for( int Row=0; Row No Filter. + tmpbuffer[Row*(mymng->effwdt+1)]=0; + // Copy the scanline: (reverse order) + memcpy(tmpbuffer+Row*(mymng->effwdt+1)+1, + mymng->image+((OffsetH-1-(OffsetY+Row))*(mymng->effwdt))+OffsetX,mymng->effwdt); + // swap red and blue components + RGBtoBGR(tmpbuffer+Row*(mymng->effwdt+1)+1,mymng->effwdt); + } + + // Compress data with ZLib (Deflate): + BYTE *dstbuffer = new BYTE[(mymng->effwdt+1)*OffsetH]; + if( dstbuffer == 0 ) return; + DWORD dstbufferSize=(mymng->effwdt+1)*OffsetH; + + // Compress data: + if(Z_OK != compress2((Bytef *)dstbuffer,(ULONG *)&dstbufferSize,(const Bytef*)tmpbuffer, + (ULONG) (mymng->effwdt+1)*OffsetH,9 )) return; + + // Write Data into MNG File: + mng_putchunk_idat( hMNG, dstbufferSize, (mng_ptr*)dstbuffer); + mng_putchunk_iend(hMNG); + + // Free the stuff: + delete [] tmpbuffer; + delete [] dstbuffer; +} +//////////////////////////////////////////////////////////////////////////////// +long CxImageMNG::Resume() +{ + if (MNG_NEEDTIMERWAIT == mng_display_resume(hmng)){ + if (info.pImage==NULL) Create(mnginfo.width,mnginfo.height,mnginfo.bpp, CXIMAGE_FORMAT_MNG); + if (IsValid()) memcpy(GetBits(), mnginfo.image, mnginfo.effwdt * mnginfo.height); + } else { + mnginfo.animation_enabled = 0; + } + return mnginfo.animation_enabled; +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageMNG::SetSpeed(float speed) +{ + if (speed>10.0) mnginfo.speed = 10.0f; + else if (speed<0.1) mnginfo.speed = 0.1f; + else mnginfo.speed=speed; +} +//////////////////////////////////////////////////////////////////////////////// +#endif // CXIMAGE_SUPPORT_MNG diff --git a/src/win32/dependencies/cximage/ximamng.h b/src/win32/dependencies/cximage/ximamng.h new file mode 100644 index 00000000..1b23d4c0 --- /dev/null +++ b/src/win32/dependencies/cximage/ximamng.h @@ -0,0 +1,86 @@ +/* + * File: ximamng.h + * Purpose: Declaration of the MNG Image Class + * Author: Davide Pizzolato - www.xdp.it + * Created: 2001 + */ +/* ========================================================== + * CxImageMNG (c) 07/Aug/2001 Davide Pizzolato - www.xdp.it + * For conditions of distribution and use, see copyright notice in ximage.h + * + * Special thanks to Frank Haug for suggestions and code. + * + * original mng.cpp code created by Nikolaus Brennig, November 14th, 2000. + * + * LIBMNG Copyright (c) 2000,2001 Gerard Juyn (gerard@libmng.com) + * ========================================================== + */ + +#if !defined(__ximaMNG_h) +#define __ximaMNG_h + +#include "ximage.h" + +#if CXIMAGE_SUPPORT_MNG + +//#define MNG_NO_CMS +#define MNG_SUPPORT_DISPLAY +#define MNG_SUPPORT_READ +#define MNG_SUPPORT_WRITE +#define MNG_ACCESS_CHUNKS +#define MNG_STORE_CHUNKS + +extern "C" { +#include "../mng/libmng.h" +#include "../mng/libmng_data.h" +} + +//unsigned long _stdcall RunMNGThread(void *lpParam); + +typedef struct tagmngstuff +{ + CxFile *file; + BYTE *image; + HANDLE thread; + mng_uint32 delay; + mng_uint32 width; + mng_uint32 height; + mng_uint32 effwdt; + mng_int16 bpp; + mng_bool animation; + mng_bool animation_enabled; + float speed; + long nBkgndIndex; + RGBQUAD nBkgndColor; +} mngstuff; + +class CxImageMNG: public CxImage +{ +public: + CxImageMNG(); + ~CxImageMNG(); + + bool Load(const char * imageFileName); + bool Save(const char * imageFileName){ return CxImage::Save(imageFileName,CXIMAGE_FORMAT_MNG);} + + bool Decode(CxFile * hFile); + bool Decode(FILE *hFile) { CxIOFile file(hFile); return Decode(&file); } + +#if CXIMAGE_SUPPORT_ENCODE + bool Encode(CxFile * hFile); + bool Encode(FILE *hFile) { CxIOFile file(hFile); return Encode(&file); } +#endif // CXIMAGE_SUPPORT_ENCODE + + long Resume(); + void SetSpeed(float speed); + + mng_handle hmng; + mngstuff mnginfo; +protected: + void WritePNG(mng_handle hMNG, int Frame, int FrameCount ); + void SetCallbacks(mng_handle mng); +}; + +#endif + +#endif diff --git a/src/win32/dependencies/cximage/ximapal.cpp b/src/win32/dependencies/cximage/ximapal.cpp new file mode 100644 index 00000000..78cd7053 --- /dev/null +++ b/src/win32/dependencies/cximage/ximapal.cpp @@ -0,0 +1,707 @@ +// xImaPal.cpp : Palette and Pixel functions +/* 07/08/2001 v1.00 - Davide Pizzolato - www.xdp.it + * CxImage version 5.99c 17/Oct/2004 + */ + +#include "ximage.h" + +//////////////////////////////////////////////////////////////////////////////// +/** + * returns the palette dimension in byte + */ +DWORD CxImage::GetPaletteSize() +{ + return (head.biClrUsed * sizeof(RGBQUAD)); +} +//////////////////////////////////////////////////////////////////////////////// +void CxImage::SetPaletteColor(BYTE idx, BYTE r, BYTE g, BYTE b, BYTE alpha) +{ + if ((pDib)&&(head.biClrUsed)){ + BYTE* iDst = (BYTE*)(pDib) + sizeof(BITMAPINFOHEADER); + if (idx=head.biWidth)||(y>=head.biHeight)) { + if (info.nBkgndIndex != -1) return (BYTE)info.nBkgndIndex; + else return *info.pImage; + } + if (head.biBitCount==8){ + return info.pImage[y*info.dwEffWidth + x]; + } else { + BYTE pos; + BYTE iDst= info.pImage[y*info.dwEffWidth + (x*head.biBitCount >> 3)]; + if (head.biBitCount==4){ + pos = (BYTE)(4*(1-x%2)); + iDst &= (0x0F<> pos); + } else if (head.biBitCount==1){ + pos = (BYTE)(7-x%8); + iDst &= (0x01<> pos); + } + } + return 0; +} +//////////////////////////////////////////////////////////////////////////////// +BYTE CxImage::BlindGetPixelIndex(const long x,const long y) +{ +#ifdef _DEBUG + if ((pDib==NULL) || (head.biClrUsed==0) || !IsInside(x,y)) throw 0; +#endif + + if (head.biBitCount==8){ + return info.pImage[y*info.dwEffWidth + x]; + } else { + BYTE pos; + BYTE iDst= info.pImage[y*info.dwEffWidth + (x*head.biBitCount >> 3)]; + if (head.biBitCount==4){ + pos = (BYTE)(4*(1-x%2)); + iDst &= (0x0F<> pos); + } else if (head.biBitCount==1){ + pos = (BYTE)(7-x%8); + iDst &= (0x01<> pos); + } + } + return 0; +} +//////////////////////////////////////////////////////////////////////////////// +RGBQUAD CxImage::GetPixelColor(long x,long y, bool bGetAlpha) +{ +// RGBQUAD rgb={0,0,0,0}; + RGBQUAD rgb=info.nBkgndColor; // + if ((pDib==NULL)||(x<0)||(y<0)|| + (x>=head.biWidth)||(y>=head.biHeight)){ + if (info.nBkgndIndex != -1){ + if (head.biBitCount<24) return GetPaletteColor((BYTE)info.nBkgndIndex); + else return info.nBkgndColor; + } else if (pDib) return GetPixelColor(0,0); + return rgb; + } + + if (head.biClrUsed){ + rgb = GetPaletteColor(GetPixelIndex(x,y)); + } else { + BYTE* iDst = info.pImage + y*info.dwEffWidth + x*3; + rgb.rgbBlue = *iDst++; + rgb.rgbGreen= *iDst++; + rgb.rgbRed = *iDst; + } +#if CXIMAGE_SUPPORT_ALPHA + if (pAlpha && bGetAlpha) rgb.rgbReserved = AlphaGet(x,y); +#else + rgb.rgbReserved = 0; +#endif //CXIMAGE_SUPPORT_ALPHA + return rgb; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * This is (a bit) faster version of GetPixelColor. + * It tests bounds only in debug mode (_DEBUG defined). + * + * It is an error to request out-of-borders pixel with this method. + * In DEBUG mode an exception will be thrown, and data will be violated in non-DEBUG mode. + * \author ***bd*** 2.2004 + */ +RGBQUAD CxImage::BlindGetPixelColor(const long x,const long y) +{ + RGBQUAD rgb; +#ifdef _DEBUG + if ((pDib==NULL) || !IsInside(x,y)) throw 0; +#endif + + if (head.biClrUsed){ + return GetPaletteColor(BlindGetPixelIndex(x,y)); + } + + BYTE* iDst = info.pImage + y*info.dwEffWidth + x*3; + rgb.rgbBlue = *iDst++; + rgb.rgbGreen= *iDst++; + rgb.rgbRed = *iDst; +#if CXIMAGE_SUPPORT_ALPHA + if (pAlpha) rgb.rgbReserved = pAlpha[x+y*head.biWidth]; +#else + rgb.rgbReserved = 0; +#endif //CXIMAGE_SUPPORT_ALPHA + return rgb; +} +//////////////////////////////////////////////////////////////////////////////// +BYTE CxImage::GetPixelGray(long x, long y) +{ + RGBQUAD color = GetPixelColor(x,y); + return (BYTE)RGB2GRAY(color.rgbRed,color.rgbGreen,color.rgbBlue); +} +//////////////////////////////////////////////////////////////////////////////// +void CxImage::SetPixelIndex(long x,long y,BYTE i) +{ + if ((pDib==NULL)||(head.biClrUsed==0)|| + (x<0)||(y<0)||(x>=head.biWidth)||(y>=head.biHeight)) return ; + + if (head.biBitCount==8){ + info.pImage[y*info.dwEffWidth + x]=i; + return; + } else { + BYTE pos; + BYTE* iDst= info.pImage + y*info.dwEffWidth + (x*head.biBitCount >> 3); + if (head.biBitCount==4){ + pos = (BYTE)(4*(1-x%2)); + *iDst &= ~(0x0F<=head.biWidth)||(y>=head.biHeight)) return; + if (head.biClrUsed) + SetPixelIndex(x,y,GetNearestIndex(c)); + else { + BYTE* iDst = info.pImage + y*info.dwEffWidth + x*3; + *iDst++ = c.rgbBlue; + *iDst++ = c.rgbGreen; + *iDst = c.rgbRed; +#if CXIMAGE_SUPPORT_ALPHA + if (bSetAlpha) AlphaSet(x,y,c.rgbReserved); +#endif //CXIMAGE_SUPPORT_ALPHA + } +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Blends the current pixel color with a new color. + * \param x,y = pixel + * \param c = new color + * \param blend = can be from 0 (no effect) to 1 (full effect). + * \param bSetAlpha = if true, blends also the alpha component stored in c.rgbReserved + */ +void CxImage::BlendPixelColor(long x,long y,RGBQUAD c, float blend, bool bSetAlpha) +{ + if ((pDib==NULL)||(x<0)||(y<0)|| + (x>=head.biWidth)||(y>=head.biHeight)) return; + + int a0 = (int)(256*blend); + int a1 = 256 - a0; + + RGBQUAD c0 = BlindGetPixelColor(x,y); + c.rgbRed = (BYTE)((c.rgbRed * a0 + c0.rgbRed * a1)>>8); + c.rgbBlue = (BYTE)((c.rgbBlue * a0 + c0.rgbBlue * a1)>>8); + c.rgbGreen = (BYTE)((c.rgbGreen * a0 + c0.rgbGreen * a1)>>8); + + if (head.biClrUsed) + SetPixelIndex(x,y,GetNearestIndex(c)); + else { + BYTE* iDst = info.pImage + y*info.dwEffWidth + x*3; + *iDst++ = c.rgbBlue; + *iDst++ = c.rgbGreen; + *iDst = c.rgbRed; +#if CXIMAGE_SUPPORT_ALPHA + if (bSetAlpha) AlphaSet(x,y,c.rgbReserved); +#endif //CXIMAGE_SUPPORT_ALPHA + } +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Returns the best palette index that matches a specified color. + */ +BYTE CxImage::GetNearestIndex(RGBQUAD c) +{ + if ((pDib==NULL)||(head.biClrUsed==0)) return 0; + + // check matching with the previous result + if (info.last_c_isvalid && (*(long*)&info.last_c == *(long*)&c)) return info.last_c_index; + info.last_c = c; + info.last_c_isvalid = true; + + BYTE* iDst = (BYTE*)(pDib) + sizeof(BITMAPINFOHEADER); + long distance=200000; + int i,j = 0; + long k,l; + int m = (int)(head.biClrImportant==0 ? head.biClrUsed : head.biClrImportant); + for(i=0,l=0;i100) perc=100; + for(i=0;i=0){ + if (head.biClrUsed){ + if (GetPixelIndex(x,y) == info.nBkgndIndex) return true; + } else { + RGBQUAD ct = info.nBkgndColor; + RGBQUAD c = GetPixelColor(x,y,false); + if (*(long*)&c==*(long*)&ct) return true; + } + } + +#if CXIMAGE_SUPPORT_ALPHA + if (pAlpha) return AlphaGet(x,y)==0; +#endif + + return false; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Checks if image has the same palette, if any. + * \param img = image to compare. + * \param bCheckAlpha = check also the rgbReserved field. + */ +bool CxImage::IsSamePalette(CxImage &img, bool bCheckAlpha) +{ + if (head.biClrUsed != img.head.biClrUsed) + return false; + if (head.biClrUsed == 0) + return false; + + RGBQUAD c1,c2; + for (DWORD n=0; n256) { + head.biClrImportant = 0; + return; + } + + switch(head.biBitCount){ + case 1: + head.biClrImportant = min(ncolors,2); + break; + case 4: + head.biClrImportant = min(ncolors,16); + break; + case 8: + head.biClrImportant = ncolors; + break; + } + return; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Returns pointer to pixel. Currently implemented only for truecolor images. + * + * \param x,y - coordinates + * + * \return pointer to first byte of pixel data + * + * \author ***bd*** 2.2004 + */ +void* CxImage::BlindGetPixelPointer(const long x, const long y) +{ + if (!IsIndexed()) + return info.pImage + y*info.dwEffWidth + x*3; + else + return 0; +} +//////////////////////////////////////////////////////////////////////////////// +void CxImage::DrawLine(int StartX, int EndX, int StartY, int EndY, COLORREF cr) +{ + DrawLine(StartX, EndX, StartY, EndY, RGBtoRGBQUAD(cr)); +} +//////////////////////////////////////////////////////////////////////////////// +void CxImage::DrawLine(int StartX, int EndX, int StartY, int EndY, RGBQUAD color, bool bSetAlpha) +{ + if (!pDib) return; + ////////////////////////////////////////////////////// + // Draws a line using the Bresenham line algorithm + // Thanks to Jordan DeLozier + ////////////////////////////////////////////////////// + int x1 = StartX; + int y1 = StartY; + int x = x1; // Start x off at the first pixel + int y = y1; // Start y off at the first pixel + int x2 = EndX; + int y2 = EndY; + + int xinc1,xinc2,yinc1,yinc2; // Increasing values + int den, num, numadd,numpixels; + int deltax = abs(x2 - x1); // The difference between the x's + int deltay = abs(y2 - y1); // The difference between the y's + + // Get Increasing Values + if (x2 >= x1) { // The x-values are increasing + xinc1 = 1; + xinc2 = 1; + } else { // The x-values are decreasing + xinc1 = -1; + xinc2 = -1; + } + + if (y2 >= y1) { // The y-values are increasing + yinc1 = 1; + yinc2 = 1; + } else { // The y-values are decreasing + yinc1 = -1; + yinc2 = -1; + } + + // Actually draw the line + if (deltax >= deltay) // There is at least one x-value for every y-value + { + xinc1 = 0; // Don't change the x when numerator >= denominator + yinc2 = 0; // Don't change the y for every iteration + den = deltax; + num = deltax / 2; + numadd = deltay; + numpixels = deltax; // There are more x-values than y-values + } + else // There is at least one y-value for every x-value + { + xinc2 = 0; // Don't change the x for every iteration + yinc1 = 0; // Don't change the y when numerator >= denominator + den = deltay; + num = deltay / 2; + numadd = deltax; + numpixels = deltay; // There are more y-values than x-values + } + + for (int curpixel = 0; curpixel <= numpixels; curpixel++) + { + // Draw the current pixel + SetPixelColor(x,y,color,bSetAlpha); + + num += numadd; // Increase the numerator by the top of the fraction + if (num >= den) // Check if numerator >= denominator + { + num -= den; // Calculate the new numerator value + x += xinc1; // Change the x as appropriate + y += yinc1; // Change the y as appropriate + } + x += xinc2; // Change the x as appropriate + y += yinc2; // Change the y as appropriate + } +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Sets a palette with standard colors for 4 and 8 bpp images. + */ +void CxImage::SetStdPalette() +{ + if (!pDib) return; + switch (head.biBitCount){ + case 8: + { + const BYTE pal256[1024] = {0,0,0,0,0,0,128,0,0,128,0,0,0,128,128,0,128,0,0,0,128,0,128,0,128,128,0,0,192,192,192,0, + 192,220,192,0,240,202,166,0,212,240,255,0,177,226,255,0,142,212,255,0,107,198,255,0, + 72,184,255,0,37,170,255,0,0,170,255,0,0,146,220,0,0,122,185,0,0,98,150,0,0,74,115,0,0, + 50,80,0,212,227,255,0,177,199,255,0,142,171,255,0,107,143,255,0,72,115,255,0,37,87,255,0,0, + 85,255,0,0,73,220,0,0,61,185,0,0,49,150,0,0,37,115,0,0,25,80,0,212,212,255,0,177,177,255,0, + 142,142,255,0,107,107,255,0,72,72,255,0,37,37,255,0,0,0,254,0,0,0,220,0,0,0,185,0,0,0,150,0, + 0,0,115,0,0,0,80,0,227,212,255,0,199,177,255,0,171,142,255,0,143,107,255,0,115,72,255,0, + 87,37,255,0,85,0,255,0,73,0,220,0,61,0,185,0,49,0,150,0,37,0,115,0,25,0,80,0,240,212,255,0, + 226,177,255,0,212,142,255,0,198,107,255,0,184,72,255,0,170,37,255,0,170,0,255,0,146,0,220,0, + 122,0,185,0,98,0,150,0,74,0,115,0,50,0,80,0,255,212,255,0,255,177,255,0,255,142,255,0,255,107,255,0, + 255,72,255,0,255,37,255,0,254,0,254,0,220,0,220,0,185,0,185,0,150,0,150,0,115,0,115,0,80,0,80,0, + 255,212,240,0,255,177,226,0,255,142,212,0,255,107,198,0,255,72,184,0,255,37,170,0,255,0,170,0, + 220,0,146,0,185,0,122,0,150,0,98,0,115,0,74,0,80,0,50,0,255,212,227,0,255,177,199,0,255,142,171,0, + 255,107,143,0,255,72,115,0,255,37,87,0,255,0,85,0,220,0,73,0,185,0,61,0,150,0,49,0,115,0,37,0, + 80,0,25,0,255,212,212,0,255,177,177,0,255,142,142,0,255,107,107,0,255,72,72,0,255,37,37,0,254,0, + 0,0,220,0,0,0,185,0,0,0,150,0,0,0,115,0,0,0,80,0,0,0,255,227,212,0,255,199,177,0,255,171,142,0, + 255,143,107,0,255,115,72,0,255,87,37,0,255,85,0,0,220,73,0,0,185,61,0,0,150,49,0,0,115,37,0, + 0,80,25,0,0,255,240,212,0,255,226,177,0,255,212,142,0,255,198,107,0,255,184,72,0,255,170,37,0, + 255,170,0,0,220,146,0,0,185,122,0,0,150,98,0,0,115,74,0,0,80,50,0,0,255,255,212,0,255,255,177,0, + 255,255,142,0,255,255,107,0,255,255,72,0,255,255,37,0,254,254,0,0,220,220,0,0,185,185,0,0,150,150,0, + 0,115,115,0,0,80,80,0,0,240,255,212,0,226,255,177,0,212,255,142,0,198,255,107,0,184,255,72,0, + 170,255,37,0,170,255,0,0,146,220,0,0,122,185,0,0,98,150,0,0,74,115,0,0,50,80,0,0,227,255,212,0, + 199,255,177,0,171,255,142,0,143,255,107,0,115,255,72,0,87,255,37,0,85,255,0,0,73,220,0,0,61,185,0, + 0,49,150,0,0,37,115,0,0,25,80,0,0,212,255,212,0,177,255,177,0,142,255,142,0,107,255,107,0,72,255,72,0, + 37,255,37,0,0,254,0,0,0,220,0,0,0,185,0,0,0,150,0,0,0,115,0,0,0,80,0,0,212,255,227,0,177,255,199,0, + 142,255,171,0,107,255,143,0,72,255,115,0,37,255,87,0,0,255,85,0,0,220,73,0,0,185,61,0,0,150,49,0,0, + 115,37,0,0,80,25,0,212,255,240,0,177,255,226,0,142,255,212,0,107,255,198,0,72,255,184,0,37,255,170,0, + 0,255,170,0,0,220,146,0,0,185,122,0,0,150,98,0,0,115,74,0,0,80,50,0,212,255,255,0,177,255,255,0, + 142,255,255,0,107,255,255,0,72,255,255,0,37,255,255,0,0,254,254,0,0,220,220,0,0,185,185,0,0, + 150,150,0,0,115,115,0,0,80,80,0,242,242,242,0,230,230,230,0,218,218,218,0,206,206,206,0,194,194,194,0, + 182,182,182,0,170,170,170,0,158,158,158,0,146,146,146,0,134,134,134,0,122,122,122,0,110,110,110,0, + 98,98,98,0,86,86,86,0,74,74,74,0,62,62,62,0,50,50,50,0,38,38,38,0,26,26,26,0,14,14,14,0,240,251,255,0, + 164,160,160,0,128,128,128,0,0,0,255,0,0,255,0,0,0,255,255,0,255,0,0,0,255,0,255,0,255,255,0,0,255,255,255,0}; + memcpy(GetPalette(),pal256,1024); + break; + } + case 4: + { + const BYTE pal16[64]={0,0,0,0,0,0,128,0,0,128,0,0,0,128,128,0,128,0,0,0,128,0,128,0,128,128,0,0,192,192,192,0, + 128,128,128,0,0,0,255,0,0,255,0,0,0,255,255,0,255,0,0,0,255,0,255,0,255,255,0,0,255,255,255,0}; + memcpy(GetPalette(),pal16,64); + break; + } + } + return; +} +//////////////////////////////////////////////////////////////////////////////// diff --git a/src/win32/dependencies/cximage/ximapcx.cpp b/src/win32/dependencies/cximage/ximapcx.cpp new file mode 100644 index 00000000..1717ab64 --- /dev/null +++ b/src/win32/dependencies/cximage/ximapcx.cpp @@ -0,0 +1,445 @@ +/* + * File: ximapcx.cpp + * Purpose: Platform Independent PCX Image Class Loader and Writer + * 05/Jan/2002 Davide Pizzolato - www.xdp.it + * CxImage version 5.99c 17/Oct/2004 + * + * based on ppmtopcx.c - convert a portable pixmap to PCX + * Copyright (C) 1994 by Ingo Wilken (Ingo.Wilken@informatik.uni-oldenburg.de) + * based on ppmtopcx.c by Michael Davidson + */ + +#include "ximapcx.h" + +#if CXIMAGE_SUPPORT_PCX + +#include "xmemfile.h" + +#define PCX_MAGIC 0X0A // PCX magic number +#define PCX_256_COLORS 0X0C // magic number for 256 colors +#define PCX_HDR_SIZE 128 // size of PCX header +#define PCX_MAXCOLORS 256 +#define PCX_MAXPLANES 4 +#define PCX_MAXVAL 255 + +//////////////////////////////////////////////////////////////////////////////// +bool CxImagePCX::Decode(CxFile *hFile) +{ + if (hFile == NULL) return false; + + PCXHEADER pcxHeader; + int i, x, y, y2, nbytes, count, Height, Width; + BYTE c, ColorMap[PCX_MAXCOLORS][3]; + BYTE *pcximage = NULL, *lpHead1 = NULL, *lpHead2 = NULL; + BYTE *pcxplanes, *pcxpixels; + + try + { + if (hFile->Read(&pcxHeader,sizeof(PCXHEADER),1)==0) throw "Can't read PCX image"; + + if (pcxHeader.Manufacturer != PCX_MAGIC) throw "Error: Not a PCX file"; + // Check for PCX run length encoding + if (pcxHeader.Encoding != 1) throw "PCX file has unknown encoding scheme"; + + Width = (pcxHeader.Xmax - pcxHeader.Xmin) + 1; + Height = (pcxHeader.Ymax - pcxHeader.Ymin) + 1; + info.xDPI = pcxHeader.Hres; + info.yDPI = pcxHeader.Vres; + + // Check that we can handle this image format + if (pcxHeader.ColorPlanes > 4) + throw "Can't handle image with more than 4 planes"; + + // Create the image + if (pcxHeader.ColorPlanes >= 3 && pcxHeader.BitsPerPixel == 8){ + Create (Width, Height, 24, CXIMAGE_FORMAT_PCX); +#if CXIMAGE_SUPPORT_ALPHA + if (pcxHeader.ColorPlanes==4) AlphaCreate(); +#endif //CXIMAGE_SUPPORT_ALPHA + } else if (pcxHeader.ColorPlanes == 4 && pcxHeader.BitsPerPixel == 1) + Create (Width, Height, 4, CXIMAGE_FORMAT_PCX); + else + Create (Width, Height, pcxHeader.BitsPerPixel, CXIMAGE_FORMAT_PCX); + + if (info.nEscape) throw "Cancelled"; // - cancel decoding + + //Read the image and check if it's ok + nbytes = pcxHeader.BytesPerLine * pcxHeader.ColorPlanes * Height; + lpHead1 = pcximage = (BYTE*)malloc(nbytes); + while (nbytes > 0){ + if (hFile == NULL || hFile->Eof()) throw "corrupted PCX"; + + hFile->Read(&c,1,1); + if ((c & 0XC0) != 0XC0){ // Repeated group + *pcximage++ = c; + --nbytes; + continue; + } + count = c & 0X3F; // extract count + hFile->Read(&c,1,1); + if (count > nbytes) throw "repeat count spans end of image"; + + nbytes -= count; + while (--count >=0) *pcximage++ = c; + } + pcximage = lpHead1; + + //store the palette + for (i = 0; i < 16; i++){ + ColorMap[i][0] = pcxHeader.ColorMap[i][0]; + ColorMap[i][1] = pcxHeader.ColorMap[i][1]; + ColorMap[i][2] = pcxHeader.ColorMap[i][2]; + } + if (pcxHeader.BitsPerPixel == 8 && pcxHeader.ColorPlanes == 1){ + hFile->Read(&c,1,1); + if (c != PCX_256_COLORS) throw "bad color map signature"; + + for (i = 0; i < PCX_MAXCOLORS; i++){ + hFile->Read(&ColorMap[i][0],1,1); + hFile->Read(&ColorMap[i][1],1,1); + hFile->Read(&ColorMap[i][2],1,1); + } + } + if (pcxHeader.BitsPerPixel == 1 && pcxHeader.ColorPlanes == 1){ + ColorMap[0][0] = ColorMap[0][1] = ColorMap[0][2] = 0; + ColorMap[1][0] = ColorMap[1][1] = ColorMap[1][2] = 255; + } + + for (DWORD idx=0; idx - cancel decoding + + y2=Height-1-y; + pcxpixels = lpHead2; + pcxplanes = pcximage + (y * pcxHeader.BytesPerLine * pcxHeader.ColorPlanes); + + if (pcxHeader.ColorPlanes == 3 && pcxHeader.BitsPerPixel == 8){ + // Deal with 24 bit color image + for (x = 0; x < Width; x++){ + SetPixelColor(x,y2,RGB(pcxplanes[x],pcxplanes[pcxHeader.BytesPerLine + x],pcxplanes[2*pcxHeader.BytesPerLine + x])); + } + continue; +#if CXIMAGE_SUPPORT_ALPHA + } else if (pcxHeader.ColorPlanes == 4 && pcxHeader.BitsPerPixel == 8){ + for (x = 0; x < Width; x++){ + SetPixelColor(x,y2,RGB(pcxplanes[x],pcxplanes[pcxHeader.BytesPerLine + x],pcxplanes[2*pcxHeader.BytesPerLine + x])); + AlphaSet(x,y2,pcxplanes[3*pcxHeader.BytesPerLine + x]); + } + continue; +#endif //CXIMAGE_SUPPORT_ALPHA + } else if (pcxHeader.ColorPlanes == 1) { + PCX_UnpackPixels(pcxpixels, pcxplanes, pcxHeader.BytesPerLine, pcxHeader.ColorPlanes, pcxHeader.BitsPerPixel); + } else { + PCX_PlanesToPixels(pcxpixels, pcxplanes, pcxHeader.BytesPerLine, pcxHeader.ColorPlanes, pcxHeader.BitsPerPixel); + } + for (x = 0; x < Width; x++) SetPixelIndex(x,y2,pcxpixels[x]); + } + + } catch (char *message) { + strncpy(info.szLastError,message,255); + if (lpHead1){ free(lpHead1); lpHead1 = NULL; } + if (lpHead2){ free(lpHead2); lpHead2 = NULL; } + return false; + } + if (lpHead1){ free(lpHead1); lpHead1 = NULL; } + if (lpHead2){ free(lpHead2); lpHead2 = NULL; } + return true; +} +//////////////////////////////////////////////////////////////////////////////// +#if CXIMAGE_SUPPORT_ENCODE +//////////////////////////////////////////////////////////////////////////////// +bool CxImagePCX::Encode(CxFile * hFile) +{ + if (EncodeSafeCheck(hFile)) return false; + + try { + PCXHEADER pcxHeader; + memset(&pcxHeader,0,sizeof(pcxHeader)); + pcxHeader.Manufacturer = PCX_MAGIC; + pcxHeader.Version = 5; + pcxHeader.Encoding = 1; + pcxHeader.Xmin = 0; + pcxHeader.Ymin = 0; + pcxHeader.Xmax = (WORD)head.biWidth-1; + pcxHeader.Ymax = (WORD)head.biHeight-1; + pcxHeader.Hres = (WORD)info.xDPI; + pcxHeader.Vres = (WORD)info.yDPI; + pcxHeader.Reserved = 0; + pcxHeader.PaletteType = head.biClrUsed==0; + + switch(head.biBitCount){ + case 24: + case 8: + { + pcxHeader.BitsPerPixel = 8; + pcxHeader.ColorPlanes = head.biClrUsed==0 ? 3 : 1; +#if CXIMAGE_SUPPORT_ALPHA + if (AlphaIsValid() && head.biClrUsed==0) pcxHeader.ColorPlanes =4; +#endif //CXIMAGE_SUPPORT_ALPHA + pcxHeader.BytesPerLine = (WORD)head.biWidth; + break; + } + default: //(4 1) + pcxHeader.BitsPerPixel = 1; + pcxHeader.ColorPlanes = head.biClrUsed==16 ? 4 : 1; + pcxHeader.BytesPerLine = (WORD)((head.biWidth * pcxHeader.BitsPerPixel + 7)>>3); + } + + if (pcxHeader.BitsPerPixel == 1 && pcxHeader.ColorPlanes == 1){ + pcxHeader.ColorMap[0][0] = pcxHeader.ColorMap[0][1] = pcxHeader.ColorMap[0][2] = 0; + pcxHeader.ColorMap[1][0] = pcxHeader.ColorMap[1][1] = pcxHeader.ColorMap[1][2] = 255; + } + if (pcxHeader.BitsPerPixel == 1 && pcxHeader.ColorPlanes == 4){ + RGBQUAD c; + for (int i = 0; i < 16; i++){ + c=GetPaletteColor(i); + pcxHeader.ColorMap[i][0] = c.rgbRed; + pcxHeader.ColorMap[i][1] = c.rgbGreen; + pcxHeader.ColorMap[i][2] = c.rgbBlue; + } + } + + pcxHeader.BytesPerLine = (pcxHeader.BytesPerLine + 1)&(~1); + + if (hFile->Write(&pcxHeader, sizeof(pcxHeader), 1) == 0 ) + throw "cannot write PCX header"; + + CxMemFile buffer; + buffer.Open(); + + BYTE c,n; + long x,y; + if (head.biClrUsed==0){ + for (y = head.biHeight-1; y >=0 ; y--){ + for (int p=0; pWrite(buffer.GetBuffer(false),buffer.Size(),1); + + } else if (head.biBitCount==8) { + + for (y = head.biHeight-1; y >=0 ; y--){ + c=n=0; + for (x = 0; xWrite(buffer.GetBuffer(false),buffer.Size(),1); + + if (head.biBitCount == 8){ + hFile->PutC(0x0C); + BYTE* pal = (BYTE*)malloc(768); + RGBQUAD c; + for (int i=0;i<256;i++){ + c=GetPaletteColor(i); + pal[3*i+0] = c.rgbRed; + pal[3*i+1] = c.rgbGreen; + pal[3*i+2] = c.rgbBlue; + } + hFile->Write(pal,768,1); + free(pal); + } + } else { //(head.biBitCount==4) || (head.biBitCount==1) + + RGBQUAD *rgb = GetPalette(); + bool binvert = false; + if (CompareColors(&rgb[0],&rgb[1])>0) binvert=(head.biBitCount==1); + + BYTE* plane = (BYTE*)malloc(pcxHeader.BytesPerLine); + BYTE* raw = (BYTE*)malloc(head.biWidth); + + for(y = head.biHeight-1; y >=0 ; y--) { + + for( x = 0; x < head.biWidth; x++) raw[x] = (BYTE)GetPixelIndex(x,y); + + if (binvert) for( x = 0; x < head.biWidth; x++) raw[x] = 1-raw[x]; + + for( x = 0; x < pcxHeader.ColorPlanes; x++ ) { + PCX_PixelsToPlanes(raw, head.biWidth, plane, x); + PCX_PackPlanes(plane, pcxHeader.BytesPerLine, buffer); + } + } + + free(plane); + free(raw); + + hFile->Write(buffer.GetBuffer(false),buffer.Size(),1); + + } + + } catch (char *message) { + strncpy(info.szLastError,message,255); + return false; + } + return true; +} +//////////////////////////////////////////////////////////////////////////////// +#endif // CXIMAGE_SUPPORT_ENCODE +//////////////////////////////////////////////////////////////////////////////// +// Convert multi-plane format into 1 pixel per byte +// from unpacked file data bitplanes[] into pixel row pixels[] +// image Height rows, with each row having planes image planes each +// bytesperline bytes +void CxImagePCX::PCX_PlanesToPixels(BYTE * pixels, BYTE * bitplanes, short bytesperline, short planes, short bitsperpixel) +{ + int i, j, npixels; + BYTE * p; + if (planes > 4) throw "Can't handle more than 4 planes"; + if (bitsperpixel != 1) throw "Can't handle more than 1 bit per pixel"; + + // Clear the pixel buffer + npixels = (bytesperline * 8) / bitsperpixel; + p = pixels; + while (--npixels >= 0) *p++ = 0; + + // Do the format conversion + for (i = 0; i < planes; i++){ + int pixbit, bits, mask; + p = pixels; + pixbit = (1 << i); // pixel bit for this plane + for (j = 0; j < bytesperline; j++){ + bits = *bitplanes++; + for (mask = 0X80; mask != 0; mask >>= 1, p++) + if (bits & mask) *p |= pixbit; + } + } +} +//////////////////////////////////////////////////////////////////////////////// +// convert packed pixel format into 1 pixel per byte +// from unpacked file data bitplanes[] into pixel row pixels[] +// image Height rows, with each row having planes image planes each +// bytesperline bytes +void CxImagePCX::PCX_UnpackPixels(BYTE * pixels, BYTE * bitplanes, short bytesperline, short planes, short bitsperpixel) +{ + register int bits; + if (planes != 1) throw "Can't handle packed pixels with more than 1 plane."; + + if (bitsperpixel == 8){ // 8 bits/pixels, no unpacking needed + while (bytesperline-- > 0) *pixels++ = *bitplanes++; + } else if (bitsperpixel == 4){ // 4 bits/pixel, two pixels per byte + while (bytesperline-- > 0){ + bits = *bitplanes++; + *pixels++ = (BYTE)((bits >> 4) & 0X0F); + *pixels++ = (BYTE)((bits) & 0X0F); + } + } else if (bitsperpixel == 2){ // 2 bits/pixel, four pixels per byte + while (bytesperline-- > 0){ + bits = *bitplanes++; + *pixels++ = (BYTE)((bits >> 6) & 0X03); + *pixels++ = (BYTE)((bits >> 4) & 0X03); + *pixels++ = (BYTE)((bits >> 2) & 0X03); + *pixels++ = (BYTE)((bits) & 0X03); + } + } else if (bitsperpixel == 1){ // 1 bits/pixel, 8 pixels per byte + while (bytesperline-- > 0){ + bits = *bitplanes++; + *pixels++ = ((bits & 0X80) != 0); + *pixels++ = ((bits & 0X40) != 0); + *pixels++ = ((bits & 0X20) != 0); + *pixels++ = ((bits & 0X10) != 0); + *pixels++ = ((bits & 0X08) != 0); + *pixels++ = ((bits & 0X04) != 0); + *pixels++ = ((bits & 0X02) != 0); + *pixels++ = ((bits & 0X01) != 0); + } + } +} +//////////////////////////////////////////////////////////////////////////////// +/* PCX_PackPixels(const long p,BYTE &c, BYTE &n, long &l, CxFile &f) + * p = current pixel (-1 ends the line -2 ends odd line) + * c = previous pixel + * n = number of consecutive pixels + */ +void CxImagePCX::PCX_PackPixels(const long p,BYTE &c, BYTE &n, CxFile &f) +{ + if (p!=c && n){ + if (n==1 && c<0xC0){ + f.PutC(c); + } else { + f.PutC(0xC0|n); + f.PutC(c); + } + n=0; + } + if (n==0x3F) { + f.PutC(0xFF); + f.PutC(c); + n=0; + } + if (p==-2) f.PutC(0); + c=(BYTE)p; + n++; +} +//////////////////////////////////////////////////////////////////////////////// +void CxImagePCX::PCX_PackPlanes(BYTE* buff, const long size, CxFile &f) +{ + BYTE *start,*end; + BYTE c, previous, count; + + start = buff; + end = buff + size; + previous = *start++; + count = 1; + + while (start < end) { + c = *start++; + if (c == previous && count < 63) { + ++count; + continue; + } + + if (count > 1 || (previous & 0xc0) == 0xc0) { + f.PutC( count | 0xc0 ); + } + f.PutC(previous); + previous = c; + count = 1; + } + + if (count > 1 || (previous & 0xc0) == 0xc0) { + count |= 0xc0; + f.PutC(count); + } + f.PutC(previous); +} +//////////////////////////////////////////////////////////////////////////////// +void CxImagePCX::PCX_PixelsToPlanes(BYTE* raw, long width, BYTE* buf, long plane) +{ + int cbit, x, mask; + unsigned char *cp = buf-1; + + mask = 1 << plane; + cbit = -1; + for( x = 0; x < width; x++ ) { + if( cbit < 0 ) { + cbit = 7; + *++cp = 0; + } + if( raw[x] & mask ) + *cp |= (1<jmpbuf, 1); +} +//////////////////////////////////////////////////////////////////////////////// +void CxImagePNG::expand2to4bpp(BYTE* prow) +{ + BYTE *psrc,*pdst; + BYTE pos,idx; + for(long x=head.biWidth-1;x>=0;x--){ + psrc = prow + ((2*x)>>3); + pdst = prow + ((4*x)>>3); + pos = (BYTE)(2*(3-x%4)); + idx = (BYTE)((*psrc & (0x03<>pos); + pos = (BYTE)(4*(1-x%2)); + *pdst &= ~(0x0F<jmpbuf)) { + /* Free all of the memory associated with the png_ptr and info_ptr */ + if (row_pointers) delete[] row_pointers; + png_destroy_read_struct(&png_ptr, &info_ptr, (png_infopp)NULL); + throw ""; + } + /* set up the input control */ + //png_init_io(png_ptr, hFile); + + // use custom I/O functions + png_set_read_fn(png_ptr, hFile, (png_rw_ptr)user_read_data); + png_set_error_fn(png_ptr,info.szLastError,(png_error_ptr)user_error_fn,NULL); + + /* read the file information */ + png_read_info(png_ptr, info_ptr); + + /* allocate the memory to hold the image using the fields of png_info. */ + png_color_16 my_background={ 0, 192, 192, 192, 0 }; + png_color_16 *image_background; + + if (info_ptr->pixel_depth != 32){ + // preserve original background info. + if (png_get_bKGD(png_ptr, info_ptr, &image_background)) + png_set_background(png_ptr, image_background,PNG_BACKGROUND_GAMMA_FILE, 1, 1.0); + else + png_set_background(png_ptr, &my_background,PNG_BACKGROUND_GAMMA_SCREEN, 0, 1.0); +// - we call png_set_bgr() below +// // safe check +// if (info_ptr->pixel_depth > 16 ) info_ptr->color_type = COLORTYPE_COLOR; + } + + // hack for images with alpha channel + if (info_ptr->pixel_depth == 32){ +// info.nBkgndIndex = 0; //enable transparency + if (png_get_bKGD(png_ptr, info_ptr, &image_background)){ + info.nBkgndColor.rgbRed = (BYTE)image_background->red; + info.nBkgndColor.rgbGreen = (BYTE)image_background->green; + info.nBkgndColor.rgbBlue = (BYTE)image_background->blue; + info.nBkgndColor.rgbReserved = 0; // + } + } + + /* tell libpng to strip 16 bit depth files down to 8 bits */ + if (info_ptr->bit_depth == 16) png_set_strip_16(png_ptr); + + int pixel_depth=info_ptr->pixel_depth; + if (pixel_depth > 16 ) pixel_depth=24; + if (pixel_depth == 16 ) pixel_depth=8; + + Create(info_ptr->width, info_ptr->height, pixel_depth, CXIMAGE_FORMAT_PNG); + + /* get metrics */ + switch (info_ptr->phys_unit_type) + { + case PNG_RESOLUTION_UNKNOWN: + SetXDPI(info_ptr->x_pixels_per_unit); + SetYDPI(info_ptr->y_pixels_per_unit); + break; + case PNG_RESOLUTION_METER: + SetXDPI((long)floor(info_ptr->x_pixels_per_unit * 254.0 / 10000.0 + 0.5)); + SetYDPI((long)floor(info_ptr->y_pixels_per_unit * 254.0 / 10000.0 + 0.5)); + break; + } + + if (info_ptr->num_palette>0) + SetPalette((rgb_color*)info_ptr->palette,info_ptr->num_palette); + else if (info_ptr->bit_depth ==2) { // needed for 2 bpp grayscale PNGs + SetPaletteColor(0,0,0,0); + SetPaletteColor(1,85,85,85); + SetPaletteColor(2,170,170,170); + SetPaletteColor(3,255,255,255); + } else SetGrayPalette(); // needed for grayscale PNGs + + // simple transparency (the real PGN transparency is more complex) + if (info_ptr->num_trans!=0){ + //palette transparency + RGBQUAD* pal=GetPalette(); + if (pal){ + DWORD ip; + for (ip=0;ipnum_trans);ip++) + pal[ip].rgbReserved=info_ptr->trans[ip]; + if (info_ptr->num_trans==1 && pal[0].rgbReserved==0){ + info.nBkgndIndex = 0; + } else { + info.bAlphaPaletteEnabled=true; + for (;ipcolor_type == PNG_COLOR_TYPE_RGB_ALPHA || //Alpha channel + (info_ptr->color_type == PNG_COLOR_TYPE_GRAY_ALPHA && info_ptr->pixel_depth == 32)){ + if (info_ptr->color_type == PNG_COLOR_TYPE_GRAY_ALPHA){ + png_set_gray_to_rgb(png_ptr); + png_set_expand(png_ptr); + } +#if CXIMAGE_SUPPORT_ALPHA // + png_set_filler(png_ptr, 0xff, PNG_FILLER_AFTER); + AlphaCreate(); +#else + png_set_strip_alpha(png_ptr); +#endif //CXIMAGE_SUPPORT_ALPHA + } + + // - flip the RGB pixels to BGR (or RGBA to BGRA) + if (info_ptr->color_type & PNG_COLOR_MASK_COLOR) png_set_bgr(png_ptr); + + // - handle cancel + if (info.nEscape) longjmp(png_ptr->jmpbuf, 1); + + //allocate the buffer + int row_stride = info_ptr->width * ((info_ptr->pixel_depth+7)>>3); + row_pointers = new BYTE[10+row_stride]; + + // turn on interlace handling + number_passes = png_set_interlace_handling(png_ptr); + + if (number_passes>1){ + SetCodecOption(1); + } else { + SetCodecOption(0); + } + + for (int pass=0; pass< number_passes; pass++) { + iter.Upset(); + int y=0; + do { + + // - handle cancel + if (info.nEscape) longjmp(png_ptr->jmpbuf, 1); + +#if CXIMAGE_SUPPORT_ALPHA // + if (!AlphaIsValid()) +#endif // CXIMAGE_SUPPORT_ALPHA + { + //recover data from previous scan + if (info_ptr->interlace_type && pass>0) + iter.GetRow(row_pointers, info.dwEffWidth); + //read next row + png_read_row(png_ptr, row_pointers, NULL); +// - already done by png_set_bgr() +// //HACK BY OP && ( for interlace, swap only in the last pass) +// if (info_ptr->color_type==COLORTYPE_COLOR && pass==(number_passes-1)) +// RGBtoBGR(row_pointers, info.dwEffWidth); + // expand 2 bpp images only in the last pass + if (info_ptr->bit_depth==2 && pass==(number_passes-1)) + expand2to4bpp(row_pointers); + //copy the pixels + iter.SetRow(row_pointers, info.dwEffWidth); + //go on + iter.PrevRow(); + } +#if CXIMAGE_SUPPORT_ALPHA // + else { //alpha blend + + //compute the correct position of the line + long ax,ay; + ay = head.biHeight-1-y; + BYTE* prow= iter.GetRow(ay); + + //recover data from previous scan + if (info_ptr->interlace_type && pass>0 && pass!=7){ + for(ax=head.biWidth;ax>=0;ax--){ + row_pointers[ax*4]=prow[3*ax]; + row_pointers[ax*4+1]=prow[3*ax+1]; + row_pointers[ax*4+2]=prow[3*ax+2]; + row_pointers[ax*4+3]=AlphaGet(ax,ay); + } + } + + //read next row + png_read_row(png_ptr, row_pointers, NULL); + + //RGBA -> RGB + A + for(ax=0;axjmpbuf)){ + /* If we get here, we had a problem reading the file */ + if (info_ptr->palette) free(info_ptr->palette); + png_destroy_write_struct(&png_ptr, (png_infopp)&info_ptr); + throw "Error saving PNG file"; + } + + int row_stride = info.dwEffWidth; + /* set up the output control */ + //png_init_io(png_ptr, hFile); + + // use custom I/O functions + png_set_write_fn(png_ptr,hFile,(png_rw_ptr)user_write_data,(png_flush_ptr)user_flush_data); + + /* set the file information here */ + info_ptr->width = GetWidth(); + info_ptr->height = GetHeight(); + info_ptr->pixel_depth = (BYTE)GetBpp(); + info_ptr->channels = (GetBpp()>8) ? (BYTE)3: (BYTE)1; + info_ptr->bit_depth = (BYTE)(GetBpp()/info_ptr->channels); + info_ptr->color_type = GetColorType(); + info_ptr->compression_type = info_ptr->filter_type = 0; + info_ptr->valid = 0; + info_ptr->rowbytes = row_stride; + + switch(GetCodecOption(CXIMAGE_FORMAT_PNG)){ + case 1: + info_ptr->interlace_type = PNG_INTERLACE_ADAM7; + break; + default: + info_ptr->interlace_type = PNG_INTERLACE_NONE; + } + + /* set compression level */ + //png_set_compression_level(png_ptr, Z_BEST_COMPRESSION); + + /* set background */ + png_color_16 image_background={ 0, 255, 255, 255, 0 }; + if (info.nBkgndIndex!=-1) { + image_background.blue = info.nBkgndColor.rgbBlue; + image_background.green = info.nBkgndColor.rgbGreen; + image_background.red = info.nBkgndColor.rgbRed; + } + png_set_bKGD(png_ptr, info_ptr, &image_background); + + /* set metrics */ + png_set_pHYs(png_ptr, info_ptr, head.biXPelsPerMeter, head.biYPelsPerMeter, PNG_RESOLUTION_METER); + + /* set the palette if there is one */ + if (GetPalette()){ + png_set_IHDR(png_ptr, info_ptr, info_ptr->width, info_ptr->height, info_ptr->bit_depth, + PNG_COLOR_TYPE_PALETTE, info_ptr->interlace_type, + PNG_COMPRESSION_TYPE_BASE, PNG_FILTER_TYPE_BASE); + info_ptr->valid |= PNG_INFO_PLTE; + + // simple transparency + if (info.nBkgndIndex != -1){ + trans[0]=0; + info_ptr->num_trans = 1; + info_ptr->valid |= PNG_INFO_tRNS; + info_ptr->trans = trans; + // the transparency indexes start from 0 + if (info.nBkgndIndex){ + SwapIndex(0,(BYTE)info.nBkgndIndex); + // the ghost must set the changed attributes in the body + if (info.pGhost) info.pGhost->SetTransIndex(0); + } + } + + int nc = GetNumColors(); + + /* We not need to write unused colors! */ + /* only for small images */ + if ((nc>2)&&((head.biWidth*head.biHeight)<65536)){ + nc = 0; + for (DWORD y=0;ync){ + nc=GetPixelIndex(x,y); + } + } + } + nc++; + } + + if (info.bAlphaPaletteEnabled){ + for(WORD ip=0; ipnum_trans = (WORD)nc; + info_ptr->valid |= PNG_INFO_tRNS; + info_ptr->trans = trans; + } + + // copy the palette colors + info_ptr->palette = new png_color[nc]; + info_ptr->num_palette = (png_uint_16) nc; + for (int i=0; ipalette[i].red, &info_ptr->palette[i].green, &info_ptr->palette[i].blue); + + } + +#if CXIMAGE_SUPPORT_ALPHA // + //Merge the transparent color with the alpha channel + bool bNeedTempAlpha = false; + if (head.biBitCount==24 && info.nBkgndIndex>=0){ + if (!AlphaIsValid()){ + bNeedTempAlpha = true; + AlphaCreate(); + } + RGBQUAD c,ct=GetTransColor(); + for(long y=0; y < head.biHeight; y++){ + for(long x=0; x < head.biWidth ; x++){ + c=GetPixelColor(x,y,false); + if (*(long*)&c==*(long*)&ct) + AlphaSet(x,y,0); + }} + } +#endif // CXIMAGE_SUPPORT_ALPHA // + +#if CXIMAGE_SUPPORT_ALPHA // + if (AlphaIsValid()){ + row_stride = 4 * head.biWidth; + + info_ptr->pixel_depth = 32; + info_ptr->channels = 4; + info_ptr->bit_depth = 8; + info_ptr->color_type = PNG_COLOR_TYPE_RGB_ALPHA; + info_ptr->rowbytes = row_stride; + + /* write the file information */ + png_write_info(png_ptr, info_ptr); + + // "10+row_stride" fix heap deallocation problem during debug??? + BYTE *row_pointers = new BYTE[10+row_stride]; + + //interlace handling + int num_pass = png_set_interlace_handling(png_ptr); + for (int pass = 0; pass < num_pass; pass++){ + + //write image + iter.Upset(); + long ay=head.biHeight-1; + RGBQUAD c; + do { + for (long ax=head.biWidth-1; ax>=0;ax--){ + c=GetPixelColor(ax,ay); + row_pointers[ax*4+3]=(BYTE)((AlphaGet(ax,ay)*info.nAlphaMax)/255); + row_pointers[ax*4+2]=c.rgbBlue; + row_pointers[ax*4+1]=c.rgbGreen; + row_pointers[ax*4]=c.rgbRed; + } + png_write_row(png_ptr, row_pointers); + ay--; + } while(iter.PrevRow()); + } + + delete [] row_pointers; + } + else +#endif //CXIMAGE_SUPPORT_ALPHA // + { + /* write the file information */ + png_write_info(png_ptr, info_ptr); + /* If you are only writing one row at a time, this works */ + BYTE *row_pointers = new BYTE[10+row_stride]; + + //interlace handling + int num_pass = png_set_interlace_handling(png_ptr); + for (int pass = 0; pass < num_pass; pass++){ + + //write image + iter.Upset(); + do { + iter.GetRow(row_pointers, row_stride); + //HACK BY OP + if (info_ptr->color_type == 2 /*COLORTYPE_COLOR*/) + RGBtoBGR(row_pointers, row_stride); + png_write_row(png_ptr, row_pointers); + } while(iter.PrevRow()); + + } + + delete [] row_pointers; + } + +#if CXIMAGE_SUPPORT_ALPHA // + /* remove the temporary alpha channel*/ + if (bNeedTempAlpha) AlphaDelete(); +#endif // CXIMAGE_SUPPORT_ALPHA // + + /* It is REQUIRED to call this to finish writing the rest of the file */ + png_write_end(png_ptr, info_ptr); + + /* if you malloced the palette, free it here */ + if (info_ptr->palette) delete[] (info_ptr->palette); + + /* clean up after the write, and free any memory allocated */ + png_destroy_write_struct(&png_ptr, (png_infopp)&info_ptr); + + } catch (char *message) { + strncpy(info.szLastError,message,255); + return FALSE; + } + /* that's it */ + return TRUE; +} +//////////////////////////////////////////////////////////////////////////////// +#endif // CXIMAGE_SUPPORT_ENCODE +//////////////////////////////////////////////////////////////////////////////// +#endif // CXIMAGE_SUPPORT_PNG diff --git a/src/win32/dependencies/cximage/ximapng.h b/src/win32/dependencies/cximage/ximapng.h new file mode 100644 index 00000000..3c767c1c --- /dev/null +++ b/src/win32/dependencies/cximage/ximapng.h @@ -0,0 +1,73 @@ +/* + * File: ximapng.h + * Purpose: PNG Image Class Loader and Writer + */ +/* ========================================================== + * CxImagePNG (c) 07/Aug/2001 Davide Pizzolato - www.xdp.it + * For conditions of distribution and use, see copyright notice in ximage.h + * + * Special thanks to Troels Knakkergaard for new features, enhancements and bugfixes + * + * original CImagePNG and CImageIterator implementation are: + * Copyright: (c) 1995, Alejandro Aguilar Sierra + * + * libpng Copyright (c) 1998-2003 Glenn Randers-Pehrson + * ========================================================== + */ +#if !defined(__ximaPNG_h) +#define __ximaPNG_h + +#include "ximage.h" + +#if CXIMAGE_SUPPORT_PNG + +extern "C" { +#include "../png/png.h" +} + +class CxImagePNG: public CxImage +{ +public: + CxImagePNG(): CxImage(CXIMAGE_FORMAT_PNG) {} + +// bool Load(const char * imageFileName){ return CxImage::Load(imageFileName,CXIMAGE_FORMAT_PNG);} +// bool Save(const char * imageFileName){ return CxImage::Save(imageFileName,CXIMAGE_FORMAT_PNG);} + bool Decode(CxFile * hFile); + bool Decode(FILE *hFile) { CxIOFile file(hFile); return Decode(&file); } + +#if CXIMAGE_SUPPORT_ENCODE + bool Encode(CxFile * hFile); + bool Encode(FILE *hFile) { CxIOFile file(hFile); return Encode(&file); } +#endif // CXIMAGE_SUPPORT_ENCODE + +protected: + void ima_png_error(png_struct *png_ptr, char *message); + void expand2to4bpp(BYTE* prow); + + static void user_read_data(png_structp png_ptr, png_bytep data, png_size_t length) + { + CxFile* hFile = (CxFile*)png_ptr->io_ptr; + if (hFile->Read(data,1,length) != length) png_error(png_ptr, "Read Error"); + } + + static void user_write_data(png_structp png_ptr, png_bytep data, png_size_t length) + { + CxFile* hFile = (CxFile*)png_ptr->io_ptr; + if (hFile->Write(data,1,length) != length) png_error(png_ptr, "Write Error"); + } + + static void user_flush_data(png_structp png_ptr) + { + CxFile* hFile = (CxFile*)png_ptr->io_ptr; + if (!hFile->Flush()) png_error(png_ptr, "Flush Error"); + } + static void user_error_fn(png_structp png_ptr,png_const_charp error_msg) + { + strncpy((char*)png_ptr->error_ptr,error_msg,255); + longjmp(png_ptr->jmpbuf, 1); + } +}; + +#endif + +#endif diff --git a/src/win32/dependencies/cximage/ximasel.cpp b/src/win32/dependencies/cximage/ximasel.cpp new file mode 100644 index 00000000..c9953cf3 --- /dev/null +++ b/src/win32/dependencies/cximage/ximasel.cpp @@ -0,0 +1,469 @@ +// xImaSel.cpp : Selection functions +/* 07/08/2001 v1.00 - Davide Pizzolato - www.xdp.it + * CxImage version 5.99c 17/Oct/2004 + */ + +#include "ximage.h" + +#if CXIMAGE_SUPPORT_SELECTION + +//////////////////////////////////////////////////////////////////////////////// +/** + * Checks if the image has a valid selection. + */ +bool CxImage::SelectionIsValid() +{ + return pSelection!=0; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Gets the smallest rectangle that contains the selection + */ +void CxImage::SelectionGetBox(RECT& r) +{ + memcpy(&r,&info.rSelectionBox,sizeof(RECT)); +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Empties the selection. + */ +bool CxImage::SelectionClear() +{ + if (pSelection){ + memset(pSelection,0,head.biWidth * head.biHeight); + info.rSelectionBox.left = head.biWidth; + info.rSelectionBox.bottom = head.biHeight; + info.rSelectionBox.right = info.rSelectionBox.top = 0; + return true; + } + return false; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Allocates an empty selection. + */ +bool CxImage::SelectionCreate() +{ + SelectionDelete(); + pSelection = (BYTE*)calloc(head.biWidth * head.biHeight, 1); + return (pSelection!=0); +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Deallocates the selction. + */ +bool CxImage::SelectionDelete() +{ + if (pSelection){ free(pSelection); pSelection=NULL; } + info.rSelectionBox.left = head.biWidth; + info.rSelectionBox.bottom = head.biHeight; + info.rSelectionBox.right = info.rSelectionBox.top = 0; + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Checks if the coordinates are inside the selection. + */ +bool CxImage::SelectionIsInside(long x, long y) +{ + if (IsInside(x,y)){ + if (pSelection==NULL) return true; + return pSelection[x+y*head.biWidth]!=0; + } + return false; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Adds a rectangle to the existing selection. + */ +bool CxImage::SelectionAddRect(RECT r) +{ + if (pSelection==NULL) SelectionCreate(); + if (pSelection==NULL) return false; + + RECT r2; + if (r.left r2.left) info.rSelectionBox.left = max(0L,min(head.biWidth,r2.left)); + if (info.rSelectionBox.right < r2.right) info.rSelectionBox.right = max(0L,min(head.biWidth,r2.right)); + if (info.rSelectionBox.bottom > r2.bottom) info.rSelectionBox.bottom = max(0L,min(head.biHeight,r2.bottom)); + + long ymin = max(0L,min(head.biHeight,r2.bottom)); + long ymax = max(0L,min(head.biHeight,r2.top)); + long xmin = max(0L,min(head.biWidth,r2.left)); + long xmax = max(0L,min(head.biWidth,r2.right)); + + for (long y=ymin; y (xcenter - xradius)) info.rSelectionBox.left = max(0L,min(head.biWidth,(xcenter - xradius))); + if (info.rSelectionBox.right < (xcenter + xradius)) info.rSelectionBox.right = max(0L,min(head.biWidth,(xcenter + xradius))); + if (info.rSelectionBox.bottom > (ycenter - yradius)) info.rSelectionBox.bottom = max(0L,min(head.biHeight,(ycenter - yradius))); + if (info.rSelectionBox.top < (ycenter + yradius)) info.rSelectionBox.top = max(0L,min(head.biHeight,(ycenter + yradius))); + + long xmin = max(0L,min(head.biWidth,xcenter - xradius)); + long xmax = max(0L,min(head.biWidth,xcenter + xradius)); + long ymin = max(0L,min(head.biHeight,ycenter - yradius)); + long ymax = max(0L,min(head.biHeight,ycenter + yradius)); + + long y,yo; + for (y=ymin; yy) pSelection[x + y * head.biWidth] = 255; + } + } + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Inverts the selection. + */ +bool CxImage::SelectionInvert() +{ + if (pSelection) { + BYTE *iSrc=pSelection; + long n=head.biHeight*head.biWidth; + for(long i=0; i < n; i++){ + *iSrc=(BYTE)~(*(iSrc)); + iSrc++; + } + return true; + } + return false; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Imports an existing region from another image with the same width and height. + */ +bool CxImage::SelectionCopy(CxImage &from) +{ + if (from.pSelection == NULL || head.biWidth != from.head.biWidth || head.biHeight != from.head.biHeight) return false; + if (pSelection==NULL) pSelection = (BYTE*)malloc(head.biWidth * head.biHeight); + if (pSelection==NULL) return false; + memcpy(pSelection,from.pSelection,head.biWidth * head.biHeight); + memcpy(&info.rSelectionBox,&from.info.rSelectionBox,sizeof(RECT)); + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Adds a polygonal region to the existing selection. points points to an array of POINT structures. + * Each structure specifies the x-coordinate and y-coordinate of one vertex of the polygon. + * npoints specifies the number of POINT structures in the array pointed to by points. + */ +bool CxImage::SelectionAddPolygon(POINT *points, long npoints) +{ + if (points==NULL || npoints<3) return false; + + if (pSelection==NULL) SelectionCreate(); + if (pSelection==NULL) return false; + + BYTE* plocal = (BYTE*)calloc(head.biWidth*head.biHeight, 1); + RECT localbox = {head.biWidth,0,0,head.biHeight}; + + long x,y,i=0; + POINT *current,*next,*start; + //trace contour + while (i < npoints){ + current = &points[i]; + if (current->x!=-1){ + if (i==0 || (i>0 && points[i-1].x==-1)) start = &points[i]; + + if ((i+1)==npoints || points[i+1].x==-1) + next = start; + else + next = &points[i+1]; + + float beta; + if (current->x != next->x){ + beta = (float)(next->y - current->y)/(float)(next->x - current->x); + if (current->x < next->x){ + for (x=current->x; x<=next->x; x++){ + y = (long)(current->y + (x - current->x) * beta); + if (IsInside(x,y)) plocal[x + y * head.biWidth] = 255; + } + } else { + for (x=current->x; x>=next->x; x--){ + y = (long)(current->y + (x - current->x) * beta); + if (IsInside(x,y)) plocal[x + y * head.biWidth] = 255; + } + } + } + if (current->y != next->y){ + beta = (float)(next->x - current->x)/(float)(next->y - current->y); + if (current->y < next->y){ + for (y=current->y; y<=next->y; y++){ + x = (long)(current->x + (y - current->y) * beta); + if (IsInside(x,y)) plocal[x + y * head.biWidth] = 255; + } + } else { + for (y=current->y; y>=next->y; y--){ + x = (long)(current->x + (y - current->y) * beta); + if (IsInside(x,y)) plocal[x + y * head.biWidth] = 255; + } + } + } + } + + RECT r2; + if (current->x < next->x) {r2.left=current->x; r2.right=next->x; } else {r2.left=next->x ; r2.right=current->x; } + if (current->y < next->y) {r2.bottom=current->y; r2.top=next->y; } else {r2.bottom=next->y ; r2.top=current->y; } + if (localbox.top < r2.top) localbox.top = max(0L,min(head.biHeight-1,r2.top+1)); + if (localbox.left > r2.left) localbox.left = max(0L,min(head.biWidth-1,r2.left-1)); + if (localbox.right < r2.right) localbox.right = max(0L,min(head.biWidth-1,r2.right+1)); + if (localbox.bottom > r2.bottom) localbox.bottom = max(0L,min(head.biHeight-1,r2.bottom-1)); + + i++; + } + + //fill the outer region + long npix=(localbox.right - localbox.left)*(localbox.top - localbox.bottom); + POINT* pix = (POINT*)calloc(npix,sizeof(POINT)); + BYTE back=0, mark=1; + long fx, fy, fxx, fyy, first, last,xmin,xmax,ymin,ymax; + + for (int side=0; side<4; side++){ + switch(side){ + case 0: + xmin=localbox.left; xmax=localbox.right+1; ymin=localbox.bottom; ymax=localbox.bottom+1; + break; + case 1: + xmin=localbox.right; xmax=localbox.right+1; ymin=localbox.bottom; ymax=localbox.top+1; + break; + case 2: + xmin=localbox.left; xmax=localbox.right+1; ymin=localbox.top; ymax=localbox.top+1; + break; + case 3: + xmin=localbox.left; xmax=localbox.left+1; ymin=localbox.bottom; ymax=localbox.top+1; + break; + } + //fill from the border points + for(y=ymin;y=localbox.left && fxx<=localbox.right && fyy>=localbox.bottom && fyy<=localbox.top ) + { + plocal[fxx + fyy*head.biWidth] = mark; + if (fyy > 0 && plocal[fxx + (fyy - 1)*head.biWidth] == back){ + pix[last].x = fx; + pix[last].y = fy - 1; + last++; + if (last == npix) last = 0; + } + if ((fyy + 1)=localbox.left && fxx<=localbox.right && fyy>=localbox.bottom && fyy<=localbox.top ) + { + plocal[fxx + (y + fy)*head.biWidth] = mark; + if (fyy > 0 && plocal[fxx + (fyy - 1)*head.biWidth] == back){ + pix[last].x = fx; + pix[last].y = fy - 1; + last++; + if (last == npix) last = 0; + } + if ((fyy + 1) localbox.left) info.rSelectionBox.left = localbox.left; + if (info.rSelectionBox.right < localbox.right) info.rSelectionBox.right = localbox.right+1; + if (info.rSelectionBox.bottom > localbox.bottom) info.rSelectionBox.bottom = localbox.bottom; + + free(plocal); + free(pix); + + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Adds to the selection all the pixels matching the specified color. + */ +bool CxImage::SelectionAddColor(RGBQUAD c) +{ + if (pSelection==NULL) SelectionCreate(); + if (pSelection==NULL) return false; + + RECT localbox = {head.biWidth,0,0,head.biHeight}; + + for (long y = 0; y < head.biHeight; y++){ + for (long x = 0; x < head.biWidth; x++){ + RGBQUAD color = GetPixelColor(x, y); + if (color.rgbRed == c.rgbRed && + color.rgbGreen == c.rgbGreen && + color.rgbBlue == c.rgbBlue) + { + pSelection[x + y * head.biWidth] = 255; // set the correct mask bit + + if (localbox.top < y) localbox.top = y; + if (localbox.left > x) localbox.left = x; + if (localbox.right < x) localbox.right = x; + if (localbox.bottom > y) localbox.bottom = y; + } + } + } + + if (info.rSelectionBox.top < localbox.top) info.rSelectionBox.top = localbox.top; + if (info.rSelectionBox.left > localbox.left) info.rSelectionBox.left = localbox.left; + if (info.rSelectionBox.right < localbox.right) info.rSelectionBox.right = localbox.right; + if (info.rSelectionBox.bottom > localbox.bottom) info.rSelectionBox.bottom = localbox.bottom; + + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Adds a single pixel to the existing selection. + */ +bool CxImage::SelectionAddPixel(int x, int y) +{ + if (pSelection==NULL) SelectionCreate(); + if (pSelection==NULL) return false; + + if (IsInside(x,y)) { + pSelection[x + y * head.biWidth] = 255; // set the correct mask bit + return true; + } + + return false; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Exports the selection channel in a 8bpp grayscale image. + */ +bool CxImage::SelectionSplit(CxImage *dest) +{ + if (!pSelection || !dest) return false; + + CxImage tmp(head.biWidth,head.biHeight,8); + if (!tmp.IsValid()) return false; + + for(long y=0; yTransfer(tmp); + + return true; +} +//////////////////////////////////////////////////////////////////////////////// +#if CXIMAGE_SUPPORT_WINDOWS +/** + * Converts the selection in a HRGN object. + */ +bool CxImage::SelectionToHRGN(HRGN& region) +{ + if (pSelection && region){ + for(int y = 0; y < head.biHeight; y++){ + HRGN hTemp = NULL; + int iStart = -1; + int x = 0; + for(; x < head.biWidth; x++){ + if (pSelection[x + y * head.biWidth] == 255){ + if (iStart == -1) iStart = x; + continue; + }else{ + if (iStart >= 0){ + hTemp = CreateRectRgn(iStart, y, x, y + 1); + CombineRgn(region, hTemp, region, RGN_OR); + DeleteObject(hTemp); + iStart = -1; + } + } + } + if (iStart >= 0){ + hTemp = CreateRectRgn(iStart, y, x, y + 1); + CombineRgn(region, hTemp, region, RGN_OR); + DeleteObject(hTemp); + iStart = -1; + } + } + return true; + } + return false; +} +#endif //CXIMAGE_SUPPORT_WINDOWS +//////////////////////////////////////////////////////////////////////////////// +#endif //CXIMAGE_SUPPORT_SELECTION diff --git a/src/win32/dependencies/cximage/ximatga.cpp b/src/win32/dependencies/cximage/ximatga.cpp new file mode 100644 index 00000000..a6bdf87e --- /dev/null +++ b/src/win32/dependencies/cximage/ximatga.cpp @@ -0,0 +1,300 @@ +/* + * File: ximatga.cpp + * Purpose: Platform Independent TGA Image Class Loader and Writer + * 05/Jan/2001 Davide Pizzolato - www.xdp.it + * CxImage version 5.99c 17/Oct/2004 + */ + +#include "ximatga.h" + +#if CXIMAGE_SUPPORT_TGA + +#include "ximaiter.h" + +// Definitions for image types. +#define TGA_Null 0 +#define TGA_Map 1 +#define TGA_RGB 2 +#define TGA_Mono 3 +#define TGA_RLEMap 9 +#define TGA_RLERGB 10 +#define TGA_RLEMono 11 +#define TGA_CompMap 32 +#define TGA_CompMap4 33 + +//////////////////////////////////////////////////////////////////////////////// +bool CxImageTGA::Decode(CxFile *hFile) +{ + if (hFile == NULL) return false; + + TGAHEADER tgaHead; + + try + { + if (hFile->Read(&tgaHead,sizeof(tgaHead),1)==0) + throw "Not a TGA"; + + bool bCompressed; + switch (tgaHead.ImageType){ + case TGA_Map: + case TGA_RGB: + case TGA_Mono: + bCompressed = false; + break; + case TGA_RLEMap: + case TGA_RLERGB: + case TGA_RLEMono: + bCompressed = true; + break; + default: + throw "Unknown TGA image type"; + } + + if (tgaHead.ImageWidth==0 || tgaHead.ImageHeight==0 || tgaHead.PixelDepth==0 || tgaHead.CmapLength>256) + throw "bad TGA header"; + + if (tgaHead.PixelDepth!=8 && tgaHead.PixelDepth!=15 && tgaHead.PixelDepth!=16 && tgaHead.PixelDepth!=24 && tgaHead.PixelDepth!=32) + throw "bad TGA header"; + + if (tgaHead.IdLength>0) hFile->Seek(tgaHead.IdLength,SEEK_CUR); //skip descriptor + + Create(tgaHead.ImageWidth, tgaHead.ImageHeight, tgaHead.PixelDepth, CXIMAGE_FORMAT_TGA); +#if CXIMAGE_SUPPORT_ALPHA // + if (tgaHead.PixelDepth==32) AlphaCreate(); // Image has alpha channel +#endif //CXIMAGE_SUPPORT_ALPHA + + if (!IsValid()) throw "TGA Create failed"; + + if (info.nEscape) throw "Cancelled"; // - cancel decoding + + if (tgaHead.CmapType != 0){ // read the palette + rgb_color pal[256]; + hFile->Read(pal,tgaHead.CmapLength*sizeof(rgb_color), 1); + for (int i=0;i - cancel decoding + + if (hFile == NULL || hFile->Eof()) throw "corrupted TGA"; + + if (bYReversed) pDest = iter.GetRow(tgaHead.ImageHeight-y-1); + else pDest = iter.GetRow(y); + + if (bCompressed) rleLeftover = ExpandCompressedLine(pDest,&tgaHead,hFile,tgaHead.ImageWidth,y,rleLeftover); + else ExpandUncompressedLine (pDest,&tgaHead,hFile,tgaHead.ImageWidth,y,0); + } + + if (bXReversed) Mirror(); + +#if CXIMAGE_SUPPORT_ALPHA + if (bYReversed && tgaHead.PixelDepth==32) AlphaFlip(); // +#endif //CXIMAGE_SUPPORT_ALPHA + + } catch (char *message) { + strncpy(info.szLastError,message,255); + return FALSE; + } + return true; +} +//////////////////////////////////////////////////////////////////////////////// +#if CXIMAGE_SUPPORT_ENCODE +//////////////////////////////////////////////////////////////////////////////// +bool CxImageTGA::Encode(CxFile * hFile) +{ + if (EncodeSafeCheck(hFile)) return false; + + if (head.biBitCount<8){ + strcpy(info.szLastError,"Bit depth must be 8 or 24"); + return false; + } + + TGAHEADER tgaHead; + + tgaHead.IdLength = 0; // Image ID Field Length + tgaHead.CmapType = GetPalette()!=0; // Color Map Type + tgaHead.ImageType = (head.biBitCount == 8) ? (BYTE)TGA_Map : (BYTE)TGA_RGB; // Image Type + + tgaHead.CmapIndex=0; // First Entry Index + tgaHead.CmapLength=(head.biBitCount == 8) ? 256 : 0; // Color Map Length + tgaHead.CmapEntrySize=(head.biBitCount == 8) ? (BYTE)24 : (BYTE)0; // Color Map Entry Size + + tgaHead.X_Origin=0; // X-origin of Image + tgaHead.Y_Origin=0; // Y-origin of Image + tgaHead.ImageWidth=(WORD)head.biWidth; // Image Width + tgaHead.ImageHeight=(WORD)head.biHeight; // Image Height + tgaHead.PixelDepth=(BYTE)head.biBitCount; // Pixel Depth + tgaHead.ImagDesc=0; // Image Descriptor + + if (pAlpha && head.biBitCount==24) tgaHead.PixelDepth=32; + + hFile->Write(&tgaHead,sizeof(TGAHEADER),1); + + if (head.biBitCount==8){ + rgb_color pal[256]; + RGBQUAD* ppal = GetPalette(); + for (int i=0;i<256; i++){ + pal[i].r = ppal[i].rgbBlue; + pal[i].g = ppal[i].rgbGreen; + pal[i].b = ppal[i].rgbRed; + } + hFile->Write(&pal,256*sizeof(rgb_color),1); + } + + CImageIterator iter(this); + BYTE* pDest; + if (pAlpha==0 || head.biBitCount==8){ + for (int y=0; y < tgaHead.ImageHeight; y++){ + pDest = iter.GetRow(y); + hFile->Write(pDest,tgaHead.ImageWidth * (head.biBitCount >> 3),1); + } + } else { + pDest = (BYTE*)malloc(4*tgaHead.ImageWidth); + RGBQUAD c; + for (int y=0; y < tgaHead.ImageHeight; y++){ + for(int x=0, x4=0;x + pDest[x4+3]=(BYTE)((AlphaGet(x,y)*info.nAlphaMax)/255); +#else + pDest[x4+3]=0; +#endif //CXIMAGE_SUPPORT_ALPHA + } + hFile->Write(pDest,4*tgaHead.ImageWidth,1); + } + free(pDest); + } + return true; +} +//////////////////////////////////////////////////////////////////////////////// +#endif // CXIMAGE_SUPPORT_ENCODE +//////////////////////////////////////////////////////////////////////////////// +BYTE CxImageTGA::ExpandCompressedLine(BYTE* pDest,TGAHEADER* ptgaHead,CxFile *hFile,int width, int y, BYTE rleLeftover) +{ + try{ + BYTE rle; + long filePos; + for (int x=0; xRead(&rle,1,1); + } + if (rle & 128) { // RLE-Encoded packet + rle -= 127; // Calculate real repeat count. + if ((x+rle)>width){ + rleLeftover = 128 + (rle - (width - x) - 1); + filePos = hFile->Tell(); + rle=width-x; + } + switch (ptgaHead->PixelDepth) + { + case 32: { + RGBQUAD color; + hFile->Read(&color,4,1); + for (int ix = 0; ix < rle; ix++){ + memcpy(&pDest[3*ix],&color,3); +#if CXIMAGE_SUPPORT_ALPHA // + AlphaSet(ix+x,y,color.rgbReserved); +#endif //CXIMAGE_SUPPORT_ALPHA + } + break; + } + case 24: { + rgb_color triple; + hFile->Read(&triple,3,1); + for (int ix = 0; ix < rle; ix++) memcpy(&pDest[3*ix],&triple,3); + break; + } + case 15: + case 16: { + WORD pixel; + hFile->Read(&pixel,2,1); + rgb_color triple; + triple.r = (BYTE)(( pixel & 0x1F ) * 8); // red + triple.g = (BYTE)(( pixel >> 2 ) & 0x0F8); // green + triple.b = (BYTE)(( pixel >> 7 ) & 0x0F8); // blue + for (int ix = 0; ix < rle; ix++){ + memcpy(&pDest[3*ix],&triple,3); + } + break; + } + case 8: { + BYTE pixel; + hFile->Read(&pixel,1,1); + for (int ix = 0; ix < rle; ix++) pDest[ix] = pixel; + } + } + if (rleLeftover!=255) hFile->Seek(filePos, SEEK_SET); + } else { // Raw packet + rle += 1; // Calculate real repeat count. + if ((x+rle)>width){ + rleLeftover = rle - (width - x) - 1; + rle=width-x; + } + ExpandUncompressedLine(pDest,ptgaHead,hFile,rle,y,x); + } + if (head.biBitCount == 24) pDest += rle*3; else pDest += rle; + x += rle; + } + } catch(...){ } + return rleLeftover; +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageTGA::ExpandUncompressedLine(BYTE* pDest,TGAHEADER* ptgaHead,CxFile *hFile,int width, int y, int xoffset) +{ + try{ + switch (ptgaHead->PixelDepth){ + case 8: + hFile->Read(pDest,width,1); + break; + case 15: + case 16:{ + BYTE* dst=pDest; + WORD pixel; + for (int x=0; xRead(&pixel,2,1); + *dst++ = (BYTE)(( pixel & 0x1F ) * 8); // blue + *dst++ = (BYTE)(( pixel >> 2 ) & 0x0F8); // green + *dst++ = (BYTE)(( pixel >> 7 ) & 0x0F8); // red + } + break; + } + case 24: + hFile->Read(pDest,3*width,1); + break; + case 32:{ + BYTE* dst=pDest; + for (int x=0; xRead(&pixel,4,1); + *dst++ = pixel.rgbBlue; + *dst++ = pixel.rgbGreen; + *dst++ = pixel.rgbRed; +#if CXIMAGE_SUPPORT_ALPHA // + AlphaSet(x+xoffset,y,pixel.rgbReserved); //alpha +#endif //CXIMAGE_SUPPORT_ALPHA + } + break; + } + } + } catch(...){ } +} +//////////////////////////////////////////////////////////////////////////////// +#endif // CXIMAGE_SUPPORT_TGA + diff --git a/src/win32/dependencies/cximage/ximatga.h b/src/win32/dependencies/cximage/ximatga.h new file mode 100644 index 00000000..8a6de633 --- /dev/null +++ b/src/win32/dependencies/cximage/ximatga.h @@ -0,0 +1,60 @@ +/* + * File: ximatga.h + * Purpose: TARGA Image Class Loader and Writer + */ +/* ========================================================== + * CxImageTGA (c) 05/Jan/2002 Davide Pizzolato - www.xdp.it + * For conditions of distribution and use, see copyright notice in ximage.h + * + * Parts of the code come from Paintlib : Copyright (c) 1996-1998 Ulrich von Zadow + * ========================================================== + */ +#if !defined(__ximaTGA_h) +#define __ximaTGA_h + +#include "ximage.h" + +#if CXIMAGE_SUPPORT_TGA + +class CxImageTGA: public CxImage +{ +#pragma pack(1) +typedef struct tagTgaHeader +{ + BYTE IdLength; // Image ID Field Length + BYTE CmapType; // Color Map Type + BYTE ImageType; // Image Type + + WORD CmapIndex; // First Entry Index + WORD CmapLength; // Color Map Length + BYTE CmapEntrySize; // Color Map Entry Size + + WORD X_Origin; // X-origin of Image + WORD Y_Origin; // Y-origin of Image + WORD ImageWidth; // Image Width + WORD ImageHeight; // Image Height + BYTE PixelDepth; // Pixel Depth + BYTE ImagDesc; // Image Descriptor +} TGAHEADER; +#pragma pack() + +public: + CxImageTGA(): CxImage(CXIMAGE_FORMAT_TGA) {} + +// bool Load(const char * imageFileName){ return CxImage::Load(imageFileName,CXIMAGE_FORMAT_TGA);} +// bool Save(const char * imageFileName){ return CxImage::Save(imageFileName,CXIMAGE_FORMAT_TGA);} + bool Decode(CxFile * hFile); + bool Decode(FILE *hFile) { CxIOFile file(hFile); return Decode(&file); } + +#if CXIMAGE_SUPPORT_ENCODE + bool Encode(CxFile * hFile); + bool Encode(FILE *hFile) { CxIOFile file(hFile); return Encode(&file); } +#endif // CXIMAGE_SUPPORT_ENCODE +protected: + BYTE ExpandCompressedLine(BYTE* pDest,TGAHEADER* ptgaHead,CxFile *hFile,int width, int y, BYTE rleLeftover); + void ExpandUncompressedLine(BYTE* pDest,TGAHEADER* ptgaHead,CxFile *hFile,int width, int y, int xoffset); +}; + +#endif + +#endif diff --git a/src/win32/dependencies/cximage/ximath.cpp b/src/win32/dependencies/cximage/ximath.cpp new file mode 100644 index 00000000..37533e22 --- /dev/null +++ b/src/win32/dependencies/cximage/ximath.cpp @@ -0,0 +1,97 @@ +#include "ximage.h" +#include "ximath.h" +#include + +//this module should contain some classes for geometrical transformations +//usable with selections, etc... once it's done, that is. :) + +CxPoint2::CxPoint2() +{ + x=y=0.0f; +} + +CxPoint2::CxPoint2(float const x_, float const y_) +{ + x=x_; + y=y_; +} + +CxPoint2::CxPoint2(CxPoint2 const &p) +{ + x=p.x; + y=p.y; +} + +float CxPoint2::Distance(CxPoint2 const p2) +{ + return (float)sqrt((x-p2.x)*(x-p2.x)+(y-p2.y)*(y-p2.y)); +} + +float CxPoint2::Distance(float const x_, float const y_) +{ + return (float)sqrt((x-x_)*(x-x_)+(y-y_)*(y-y_)); +} + +CxRect2::CxRect2() +{ +} + +CxRect2::CxRect2(float const x1_, float const y1_, float const x2_, float const y2_) +{ + botLeft.x=x1_; + botLeft.y=y1_; + topRight.x=x2_; + topRight.y=y2_; +} + +CxRect2::CxRect2(CxRect2 const &p) +{ + botLeft=p.botLeft; + topRight=p.topRight; +} + +float CxRect2::Surface() const +/* + * Returns the surface of rectangle. + */ +{ + return (topRight.x-botLeft.x)*(topRight.y-botLeft.y); +} + +CxRect2 CxRect2::CrossSection(CxRect2 const &r2) const +/* + * Returns crossection with another rectangle. + */ +{ + CxRect2 cs; + cs.botLeft.x=max(botLeft.x, r2.botLeft.x); + cs.botLeft.y=max(botLeft.y, r2.botLeft.y); + cs.topRight.x=min(topRight.x, r2.topRight.x); + cs.topRight.y=min(topRight.y, r2.topRight.y); + if (cs.botLeft.x<=cs.topRight.x && cs.botLeft.y<=cs.topRight.y) { + return cs; + } else { + return CxRect2(0,0,0,0); + }//if +} + +CxPoint2 CxRect2::Center() const +/* + * Returns the center point of rectangle. + */ +{ + return CxPoint2((topRight.x+botLeft.x)/2.0f, (topRight.y+botLeft.y)/2.0f); +} + +float CxRect2::Width() const +//returns rectangle width +{ + return topRight.x-botLeft.x; +} + +float CxRect2::Height() const +//returns rectangle height +{ + return topRight.y-botLeft.y; +} + diff --git a/src/win32/dependencies/cximage/ximath.h b/src/win32/dependencies/cximage/ximath.h new file mode 100644 index 00000000..10b98984 --- /dev/null +++ b/src/win32/dependencies/cximage/ximath.h @@ -0,0 +1,39 @@ +#if !defined(__ximath_h) +#define __ximath_h + +#include "ximadef.h" + +//***bd*** simple floating point point +class DLL_EXP CxPoint2 +{ +public: + CxPoint2(); + CxPoint2(float const x_, float const y_); + CxPoint2(CxPoint2 const &p); + + float Distance(CxPoint2 const p2); + float Distance(float const x_, float const y_); + + float x,y; +}; + +//and simple rectangle +class DLL_EXP CxRect2 +{ +public: + CxRect2(); + CxRect2(float const x1_, float const y1_, float const x2_, float const y2_); + CxRect2(CxPoint2 const &bl, CxPoint2 const &tr); + CxRect2(CxRect2 const &p); + + float Surface() const; + CxRect2 CrossSection(CxRect2 const &r2) const; + CxPoint2 Center() const; + float Width() const; + float Height() const; + + CxPoint2 botLeft; + CxPoint2 topRight; +}; + +#endif diff --git a/src/win32/dependencies/cximage/ximatif.cpp b/src/win32/dependencies/cximage/ximatif.cpp new file mode 100644 index 00000000..4f0bbf89 --- /dev/null +++ b/src/win32/dependencies/cximage/ximatif.cpp @@ -0,0 +1,763 @@ +/* + * File: ximatif.cpp + * Purpose: Platform Independent TIFF Image Class Loader and Writer + * 07/Aug/2001 Davide Pizzolato - www.xdp.it + * CxImage version 5.99c 17/Oct/2004 + */ + +#include "ximatif.h" + +#if CXIMAGE_SUPPORT_TIF + +#include "../tiff/tiffio.h" + +#define CVT(x) (((x) * 255L) / ((1L<<16)-1)) +#define SCALE(x) (((x)*((1L<<16)-1))/255) +#define CalculateLine(width,bitdepth) (((width * bitdepth) + 7) / 8) +#define CalculatePitch(line) (line + 3 & ~3) + +extern "C" TIFF* _TIFFOpenEx(CxFile* stream, const char* mode); + +//////////////////////////////////////////////////////////////////////////////// +CxImageTIF::~CxImageTIF() +{ + if (m_tif2) TIFFClose(m_tif2); +} +//////////////////////////////////////////////////////////////////////////////// +bool CxImageTIF::Decode(CxFile * hFile) +{ + //Comment this line if you need more information on errors + // TIFFSetErrorHandler(NULL); // + + //Open file and fill the TIFF structure + // m_tif = TIFFOpen(imageFileName,"rb"); + TIFF* m_tif = _TIFFOpenEx(hFile, "rb"); + + uint32 height=0; + uint32 width=0; + uint16 bitspersample=1; + uint16 samplesperpixel=1; + uint32 rowsperstrip=(DWORD)-1; + uint16 photometric=0; + uint16 compression=1; + uint16 orientation=ORIENTATION_TOPLEFT; // + uint16 res_unit; // + uint32 x, y; + float resolution, offset; + BOOL isRGB; + BYTE *bits; //pointer to source data + BYTE *bits2; //pointer to destination data + + try{ + //check if it's a tiff file + if (!m_tif) + throw "Error encountered while opening TIFF file"; + + // - 12/2002 : get NumFrames directly, instead of looping + // info.nNumFrames=0; + // while(TIFFSetDirectory(m_tif,(uint16)info.nNumFrames)) info.nNumFrames++; + info.nNumFrames = TIFFNumberOfDirectories(m_tif); + + if (!TIFFSetDirectory(m_tif, (uint16)info.nFrame)) + throw "Error: page not present in TIFF file"; + + //get image info + TIFFGetField(m_tif, TIFFTAG_IMAGEWIDTH, &width); + TIFFGetField(m_tif, TIFFTAG_IMAGELENGTH, &height); + TIFFGetField(m_tif, TIFFTAG_SAMPLESPERPIXEL, &samplesperpixel); + TIFFGetField(m_tif, TIFFTAG_BITSPERSAMPLE, &bitspersample); + TIFFGetField(m_tif, TIFFTAG_ROWSPERSTRIP, &rowsperstrip); + TIFFGetField(m_tif, TIFFTAG_PHOTOMETRIC, &photometric); + TIFFGetField(m_tif, TIFFTAG_ORIENTATION, &orientation); + + if (info.nEscape == -1) { + // Return output dimensions only + head.biWidth = width; + head.biHeight = height; + throw "output dimensions returned"; + } + + TIFFGetFieldDefaulted(m_tif, TIFFTAG_RESOLUTIONUNIT, &res_unit); + if (TIFFGetField(m_tif, TIFFTAG_XRESOLUTION, &resolution)) + { + if (res_unit == RESUNIT_CENTIMETER) resolution = (float)(resolution*2.54f + 0.5f); + SetXDPI((long)resolution); + } + if (TIFFGetField(m_tif, TIFFTAG_YRESOLUTION, &resolution)) + { + if (res_unit == RESUNIT_CENTIMETER) resolution = (float)(resolution*2.54f + 0.5f); + SetYDPI((long)resolution); + } + + if (TIFFGetField(m_tif, TIFFTAG_XPOSITION, &offset)) info.xOffset = (long)offset; + if (TIFFGetField(m_tif, TIFFTAG_YPOSITION, &offset)) info.yOffset = (long)offset; + + head.biClrUsed=0; + info.nBkgndIndex =-1; + + if (rowsperstrip>height){ + rowsperstrip=height; + TIFFSetField(m_tif, TIFFTAG_ROWSPERSTRIP, rowsperstrip); + } + + isRGB = (bitspersample >= 8) && + (photometric == PHOTOMETRIC_RGB) || + (photometric == PHOTOMETRIC_YCBCR) || + (photometric == PHOTOMETRIC_SEPARATED) || + (photometric == PHOTOMETRIC_LOGL) || + (photometric == PHOTOMETRIC_LOGLUV); + + if (isRGB){ + head.biBitCount=24; + }else{ + if ((photometric==PHOTOMETRIC_MINISBLACK)||(photometric==PHOTOMETRIC_MINISWHITE)){ + if (bitspersample == 1){ + head.biBitCount=1; //B&W image + head.biClrUsed =2; + } else if (bitspersample == 4) { + head.biBitCount=4; //16 colors gray scale + head.biClrUsed =16; + } else { + head.biBitCount=8; //gray scale + head.biClrUsed =256; + } + } else if (bitspersample == 4) { + head.biBitCount=4; // 16 colors + head.biClrUsed=16; + } else { + head.biBitCount=8; //256 colors + head.biClrUsed=256; + } + } + + if (info.nEscape) throw "Cancelled"; // - cancel decoding + + Create(width,height,head.biBitCount,CXIMAGE_FORMAT_TIF); //image creation + if (!pDib) throw "CxImageTIF can't create image"; + +#if CXIMAGE_SUPPORT_ALPHA + if (samplesperpixel==4) AlphaCreate(); //add alpha support for 32bpp tiffs + if (samplesperpixel==2 && bitspersample==8) AlphaCreate(); //add alpha support for 8bpp + alpha +#endif //CXIMAGE_SUPPORT_ALPHA + + TIFFGetField(m_tif, TIFFTAG_COMPRESSION, &compression); + SetCodecOption(compression); // save original compression type + + if (isRGB) { + // Read the whole image into one big RGBA buffer using + // the traditional TIFFReadRGBAImage() API that we trust. + uint32* raster; // retrieve RGBA image + uint32 *row; + + raster = (uint32*)_TIFFmalloc(width * height * sizeof (uint32)); + if (raster == NULL) throw "No space for raster buffer"; + + // Read the image in one chunk into an RGBA array + if(!TIFFReadRGBAImage(m_tif, width, height, raster, 1)) { + _TIFFfree(raster); + throw "Corrupted TIFF file!"; + } + + // read the raster lines and save them in the DIB + // with RGB mode, we have to change the order of the 3 samples RGB + row = &raster[0]; + bits2 = info.pImage; + for (y = 0; y < height; y++) { + + if (info.nEscape){ // - cancel decoding + _TIFFfree(raster); + throw "Cancelled"; + } + + bits = bits2; + for (x = 0; x < width; x++) { + *bits++ = (BYTE)TIFFGetB(row[x]); + *bits++ = (BYTE)TIFFGetG(row[x]); + *bits++ = (BYTE)TIFFGetR(row[x]); +#if CXIMAGE_SUPPORT_ALPHA + if (samplesperpixel==4) AlphaSet(x,y,(BYTE)TIFFGetA(row[x])); +#endif //CXIMAGE_SUPPORT_ALPHA + } + row += width; + bits2 += info.dwEffWidth; + } + _TIFFfree(raster); + } else { + RGBQUAD *pal; + pal=(RGBQUAD*)calloc(256,sizeof(RGBQUAD)); + if (pal==NULL) throw "Unable to allocate TIFF palette"; + + // set up the colormap based on photometric + switch(photometric) { + case PHOTOMETRIC_MINISBLACK: // bitmap and greyscale image types + case PHOTOMETRIC_MINISWHITE: + if (bitspersample == 1) { // Monochrome image + if (photometric == PHOTOMETRIC_MINISBLACK) { + pal[1].rgbRed = pal[1].rgbGreen = pal[1].rgbBlue = 255; + } else { + pal[0].rgbRed = pal[0].rgbGreen = pal[0].rgbBlue = 255; + } + } else { // need to build the scale for greyscale images + if (photometric == PHOTOMETRIC_MINISBLACK) { + for (DWORD i=0; i 0) { + if (red[n] >= 256 || green[n] >= 256 || blue[n] >= 256) { + Palette16Bits=TRUE; + break; + } + } + + // load the palette in the DIB + for (int i = (1 << bitspersample) - 1; i >= 0; i--) { + if (Palette16Bits) { + pal[i].rgbRed =(BYTE) CVT(red[i]); + pal[i].rgbGreen = (BYTE) CVT(green[i]); + pal[i].rgbBlue = (BYTE) CVT(blue[i]); + } else { + pal[i].rgbRed = (BYTE) red[i]; + pal[i].rgbGreen = (BYTE) green[i]; + pal[i].rgbBlue = (BYTE) blue[i]; + } + } + break; + } + SetPalette(pal,head.biClrUsed); //palette assign + free(pal); + + // read the tiff lines and save them in the DIB + uint32 nrow; + uint32 ys; + int line = CalculateLine(width, bitspersample * samplesperpixel); + long bitsize= TIFFStripSize(m_tif); + //verify bitsize: could be wrong if StripByteCounts is missing. + if (bitsize>(long)(head.biSizeImage*samplesperpixel)) bitsize=head.biSizeImage*samplesperpixel; + + int tiled_image = TIFFIsTiled(m_tif); + uint32 tw, tl; + BYTE* tilebuf; + if (tiled_image){ + TIFFGetField(m_tif, TIFFTAG_TILEWIDTH, &tw); + TIFFGetField(m_tif, TIFFTAG_TILELENGTH, &tl); + rowsperstrip = tl; + bitsize = TIFFTileSize(m_tif) * (int)(1+width/tw); + tilebuf = (BYTE*)malloc(TIFFTileSize(m_tif)); + } + + bits = (BYTE*)malloc(bitsize); + if (bits==NULL){ + throw "CxImageTIF can't allocate memory"; + } + + for (ys = 0; ys < height; ys += rowsperstrip) { + + if (info.nEscape){ // - cancel decoding + free(bits); + throw "Cancelled"; + } + + nrow = (ys + rowsperstrip > height ? height - ys : rowsperstrip); + + if (tiled_image){ + uint32 imagew = TIFFScanlineSize(m_tif); + uint32 tilew = TIFFTileRowSize(m_tif); + int iskew = imagew - tilew; + uint8* bufp = (uint8*) bits; + + uint32 colb = 0; + for (uint32 col = 0; col < width; col += tw) { + if (TIFFReadTile(m_tif, tilebuf, col, ys, 0, 0) < 0){ + free(tilebuf); + free(bits); + throw "Corrupted tiled TIFF file!"; + } + + if (colb + tw > imagew) { + uint32 owidth = imagew - colb; + uint32 oskew = tilew - owidth; + TileToStrip(bufp + colb, tilebuf, nrow, owidth, oskew + iskew, oskew ); + } else { + TileToStrip(bufp + colb, tilebuf, nrow, tilew, iskew, 0); + } + colb += tilew; + } + + } else { + if (TIFFReadEncodedStrip(m_tif, TIFFComputeStrip(m_tif, ys, 0), bits, nrow * line) == -1) { + free(bits); + throw "Corrupted TIFF file!"; + } + } + + for (y = 0; y < nrow; y++) { + long offset=(nrow-y-1)*line; + if (bitspersample==16) for (DWORD xi=0;xi=(int)width){ + yi--; + xi=0; + } + } + } else { //photometric==PHOTOMETRIC_CIELAB + if (head.biBitCount!=24){ //fix image + Create(width,height,24,CXIMAGE_FORMAT_TIF); +#if CXIMAGE_SUPPORT_ALPHA + if (samplesperpixel==4) AlphaCreate(); +#endif //CXIMAGE_SUPPORT_ALPHA + } + + int xi=0; + int ii=0; + int yi=height-ys-nrow+y; + RGBQUAD c; + int l,a,b,bitsoffset; + double p,cx,cy,cz,cr,cg,cb; + while (ii127) a-=256; + if (b>127) b-=256; + // lab to xyz + p = (l/2.55 + 16) / 116.0; + cx = pow( p + a * 0.002, 3); + cy = pow( p, 3); + cz = pow( p - b * 0.005, 3); + // white point + cx*=0.95047; + //cy*=1.000; + cz*=1.0883; + // xyz to rgb + cr = 3.240479 * cx - 1.537150 * cy - 0.498535 * cz; + cg = -0.969256 * cx + 1.875992 * cy + 0.041556 * cz; + cb = 0.055648 * cx - 0.204043 * cy + 1.057311 * cz; + + if ( cr > 0.00304 ) cr = 1.055 * pow(cr,0.41667) - 0.055; + else cr = 12.92 * cr; + if ( cg > 0.00304 ) cg = 1.055 * pow(cg,0.41667) - 0.055; + else cg = 12.92 * cg; + if ( cb > 0.00304 ) cb = 1.055 * pow(cb,0.41667) - 0.055; + else cb = 12.92 * cb; + + c.rgbRed =(BYTE)max(0,min(255,(int)(cr*255))); + c.rgbGreen=(BYTE)max(0,min(255,(int)(cg*255))); + c.rgbBlue =(BYTE)max(0,min(255,(int)(cb*255))); + + SetPixelColor(xi,yi,c); +#if CXIMAGE_SUPPORT_ALPHA + if (samplesperpixel==4) AlphaSet(xi,yi,bits[bitsoffset+3]); +#endif //CXIMAGE_SUPPORT_ALPHA + ii++; + xi++; + if (xi>=(int)width){ + yi--; + xi=0; + } + } + } + } + } + free(bits); + if (tiled_image) free(tilebuf); + + switch(orientation){ + case ORIENTATION_TOPRIGHT: /* row 0 top, col 0 rhs */ + Mirror(); + break; + case ORIENTATION_BOTRIGHT: /* row 0 bottom, col 0 rhs */ + Flip(); + Mirror(); + break; + case ORIENTATION_BOTLEFT: /* row 0 bottom, col 0 lhs */ + Flip(); + break; + case ORIENTATION_LEFTTOP: /* row 0 lhs, col 0 top */ + RotateRight(); + Mirror(); + break; + case ORIENTATION_RIGHTTOP: /* row 0 rhs, col 0 top */ + RotateLeft(); + break; + case ORIENTATION_RIGHTBOT: /* row 0 rhs, col 0 bottom */ + RotateLeft(); + Mirror(); + break; + case ORIENTATION_LEFTBOT: /* row 0 lhs, col 0 bottom */ + RotateRight(); + break; + } + + } + } catch (char *message) { + strncpy(info.szLastError,message,255); + if (m_tif) TIFFClose(m_tif); + if (info.nEscape==-1) return true; + return false; + } + TIFFClose(m_tif); + return true; +} +//////////////////////////////////////////////////////////////////////////////// +#if CXIMAGE_SUPPORT_ENCODE +//////////////////////////////////////////////////////////////////////////////// +bool CxImageTIF::Encode(CxFile * hFile, bool bAppend) +{ + try{ + if (hFile==NULL) throw CXIMAGE_ERR_NOFILE; + if (pDib==NULL) throw CXIMAGE_ERR_NOIMAGE; + + // replaced "w+b" with "a", to append an image directly on an existing file + if (m_tif2==NULL) m_tif2=_TIFFOpenEx(hFile, "a"); + if (m_tif2==NULL) throw "initialization fail"; + + if (bAppend || m_pages) m_multipage=true; + m_pages++; + + if (!EncodeBody(m_tif2,m_multipage,m_pages,m_pages)) throw "Error saving TIFF file"; + if (bAppend) { + if (!TIFFWriteDirectory(m_tif2)) throw "Error saving TIFF directory"; + } + } catch (char *message) { + strncpy(info.szLastError,message,255); + if (m_tif2){ + TIFFClose(m_tif2); + m_tif2=NULL; + m_multipage=false; + m_pages=0; + } + return false; + } + if (!bAppend){ + TIFFClose(m_tif2); + m_tif2=NULL; + m_multipage=false; + m_pages=0; + } + return true; +} +//////////////////////////////////////////////////////////////////////////////// +// Thanks to Abe +bool CxImageTIF::Encode(CxFile * hFile, CxImage ** pImages, int pagecount) +{ + try{ + if (hFile==NULL) throw "invalid file pointer"; + if (pImages==NULL || pagecount==0) throw "multipage TIFF, no images!"; + + CxImageTIF ghost; + for (int i=1; i<=pagecount; i++){ + if (pImages[i-1]==NULL) throw "Bad image pointer"; + ghost.Ghost(pImages[i-1]); + if (!ghost.Encode(hFile,true)) throw "Error saving TIFF file"; + } + } catch (char *message) { + strncpy(info.szLastError,message,255); + return false; + } + return true; +} +//////////////////////////////////////////////////////////////////////////////// +bool CxImageTIF::EncodeBody(TIFF *m_tif, bool multipage, int page, int pagecount) +{ + uint32 height=head.biHeight; + uint32 width=head.biWidth; + uint16 bitcount=head.biBitCount; + uint16 bitspersample; + uint16 samplesperpixel; + uint16 photometric=0; + uint16 compression; +// uint16 pitch; +// int line; + uint32 x, y; + + samplesperpixel = ((bitcount == 24) || (bitcount == 32)) ? (BYTE)3 : (BYTE)1; +#if CXIMAGE_SUPPORT_ALPHA + if (bitcount==24 && AlphaIsValid()) { bitcount=32; samplesperpixel=4; } +#endif //CXIMAGE_SUPPORT_ALPHA + + bitspersample = bitcount / samplesperpixel; + + //set the PHOTOMETRIC tag + RGBQUAD *rgb = GetPalette(); + switch (bitcount) { + case 1: + if (CompareColors(&rgb[0],&rgb[1])<0) { + /* some viewers do not handle PHOTOMETRIC_MINISBLACK: + * let's transform the image in PHOTOMETRIC_MINISWHITE + */ + //invert the colors + RGBQUAD tempRGB=GetPaletteColor(0); + SetPaletteColor(0,GetPaletteColor(1)); + SetPaletteColor(1,tempRGB); + //invert the pixels + BYTE *iSrc=info.pImage; + for (unsigned long i=0;irgbRed != x)||(rgb->rgbRed != rgb->rgbGreen)||(rgb->rgbRed != rgb->rgbBlue)){ + photometric = PHOTOMETRIC_PALETTE; + break; + } + rgb++; + } + break; + case 24: + case 32: + photometric = PHOTOMETRIC_RGB; + break; + } + +#if CXIMAGE_SUPPORT_ALPHA + if (AlphaIsValid() && bitcount==8) samplesperpixel=2; //8bpp + alpha layer +#endif //CXIMAGE_SUPPORT_ALPHA + +// line = CalculateLine(width, bitspersample * samplesperpixel); +// pitch = (uint16)CalculatePitch(line); + + //prepare the palette struct + RGBQUAD pal[256]; + if (GetPalette()){ + BYTE b; + memcpy(pal,GetPalette(),GetPaletteSize()); + for(WORD a=0;a gives better compression + TIFFSetField(m_tif, TIFFTAG_ROWSPERSTRIP, rowsperstrip); + + // handle metrics + TIFFSetField(m_tif, TIFFTAG_RESOLUTIONUNIT, RESUNIT_INCH); + TIFFSetField(m_tif, TIFFTAG_XRESOLUTION, (float)info.xDPI); + TIFFSetField(m_tif, TIFFTAG_YRESOLUTION, (float)info.yDPI); +// TIFFSetField(m_tif, TIFFTAG_XPOSITION, (float)info.xOffset); +// TIFFSetField(m_tif, TIFFTAG_YPOSITION, (float)info.yOffset); + + // multi-paging - Thanks to Abe + if (multipage) + { + char page_number[20]; + sprintf(page_number, "Page %d", page); + + TIFFSetField(m_tif, TIFFTAG_SUBFILETYPE, FILETYPE_PAGE); + TIFFSetField(m_tif, TIFFTAG_PAGENUMBER, page,pagecount); + TIFFSetField(m_tif, TIFFTAG_PAGENAME, page_number); + } else { + TIFFSetField(m_tif, TIFFTAG_SUBFILETYPE, 0); + } + + // palettes (image colormaps are automatically scaled to 16-bits) + if (photometric == PHOTOMETRIC_PALETTE) { + uint16 *r, *g, *b; + r = (uint16 *) _TIFFmalloc(sizeof(uint16) * 3 * 256); + g = r + 256; + b = g + 256; + + for (int i = 255; i >= 0; i--) { + b[i] = (uint16)SCALE((uint16)pal[i].rgbRed); + g[i] = (uint16)SCALE((uint16)pal[i].rgbGreen); + r[i] = (uint16)SCALE((uint16)pal[i].rgbBlue); + } + + TIFFSetField(m_tif, TIFFTAG_COLORMAP, r, g, b); + _TIFFfree(r); + } + + // compression + if (GetCodecOption(CXIMAGE_FORMAT_TIF)) { + compression = (WORD)GetCodecOption(CXIMAGE_FORMAT_TIF); + } else { + switch (bitcount) { + case 1 : + compression = COMPRESSION_CCITTFAX4; + break; + case 4 : + case 8 : + compression = COMPRESSION_LZW; + break; + case 24 : + case 32 : + compression = COMPRESSION_JPEG; + break; + default : + compression = COMPRESSION_NONE; + break; + } + } + TIFFSetField(m_tif, TIFFTAG_COMPRESSION, compression); + + switch (compression) { + case COMPRESSION_JPEG: + TIFFSetField(m_tif, TIFFTAG_JPEGQUALITY, info.nQuality); + TIFFSetField(m_tif, TIFFTAG_ROWSPERSTRIP, ((7+rowsperstrip)>>3)<<3); + break; + case COMPRESSION_LZW: + if (bitcount>=8) TIFFSetField(m_tif, TIFFTAG_PREDICTOR, 2); + break; + } + + // read the DIB lines from bottom to top and save them in the TIF + + BYTE *bits; + switch(bitcount) { + case 1 : + case 4 : + case 8 : + { + if (samplesperpixel==1){ + for (y = 0; y < height; y++) { + bits= info.pImage + (height - y - 1)*info.dwEffWidth; + if (TIFFWriteScanline(m_tif,bits, y, 0)==-1) return false; + } + } +#if CXIMAGE_SUPPORT_ALPHA + else { //8bpp + alpha layer + bits = (BYTE*)malloc(2*width); + if (!bits) return false; + for (y = 0; y < height; y++) { + for (x=0;x 0) { + uint32 j = cols; + while (j-- > 0) + *out++ = *in++; + out += outskew; + in += inskew; + } +} +//////////////////////////////////////////////////////////////////////////////// +TIFF* CxImageTIF::TIFFOpenEx(CxFile * hFile) +{ + if (hFile) return _TIFFOpenEx(hFile, "rb"); + return NULL; +} +//////////////////////////////////////////////////////////////////////////////// +void CxImageTIF::TIFFCloseEx(TIFF* tif) +{ + if (tif) TIFFClose(tif); +} +//////////////////////////////////////////////////////////////////////////////// +#endif // CXIMAGE_SUPPORT_TIF diff --git a/src/win32/dependencies/cximage/ximatif.h b/src/win32/dependencies/cximage/ximatif.h new file mode 100644 index 00000000..e38182bf --- /dev/null +++ b/src/win32/dependencies/cximage/ximatif.h @@ -0,0 +1,60 @@ +/* + * File: ximatif.h + * Purpose: TIFF Image Class Loader and Writer + */ +/* ========================================================== + * CxImageTIF (c) 07/Aug/2001 Davide Pizzolato - www.xdp.it + * For conditions of distribution and use, see copyright notice in ximage.h + * + * Special thanks to Troels Knakkergaard for new features, enhancements and bugfixes + * + * Special thanks to Abe for MultiPageTIFF code. + * + * LibTIFF is: + * Copyright (c) 1988-1997 Sam Leffler + * Copyright (c) 1991-1997 Silicon Graphics, Inc. + * ========================================================== + */ + +#if !defined(__ximatif_h) +#define __ximatif_h + +#include "ximage.h" + +#if CXIMAGE_SUPPORT_TIF + +#include "../tiff/tiffio.h" + +class DLL_EXP CxImageTIF: public CxImage +{ +public: + CxImageTIF(): CxImage(CXIMAGE_FORMAT_TIF) {m_tif2=NULL; m_multipage=false; m_pages=0;} + ~CxImageTIF(); + + TIFF* TIFFOpenEx(CxFile * hFile); + void TIFFCloseEx(TIFF* tif); + +// bool Load(const char * imageFileName){ return CxImage::Load(imageFileName,CXIMAGE_FORMAT_TIF);} +// bool Save(const char * imageFileName){ return CxImage::Save(imageFileName,CXIMAGE_FORMAT_TIF);} + bool Decode(CxFile * hFile); + bool Decode(FILE *hFile) { CxIOFile file(hFile); return Decode(&file); } + +#if CXIMAGE_SUPPORT_ENCODE + bool Encode(CxFile * hFile, bool bAppend=false); + bool Encode(CxFile * hFile, CxImage ** pImages, int pagecount); + bool Encode(FILE *hFile, bool bAppend=false) { CxIOFile file(hFile); return Encode(&file,bAppend); } + bool Encode(FILE *hFile, CxImage ** pImages, int pagecount) + { CxIOFile file(hFile); return Encode(&file, pImages, pagecount); } +#endif // CXIMAGE_SUPPORT_ENCODE + +protected: + void TileToStrip(uint8* out, uint8* in, uint32 rows, uint32 cols, int outskew, int inskew); + bool EncodeBody(TIFF *m_tif, bool multipage=false, int page=0, int pagecount=0); + TIFF *m_tif2; + bool m_multipage; + int m_pages; +}; + +#endif + +#endif diff --git a/src/win32/dependencies/cximage/ximatran.cpp b/src/win32/dependencies/cximage/ximatran.cpp new file mode 100644 index 00000000..82ea4747 --- /dev/null +++ b/src/win32/dependencies/cximage/ximatran.cpp @@ -0,0 +1,2439 @@ +// xImaTran.cpp : Transformation functions +/* 07/08/2001 v1.00 - Davide Pizzolato - www.xdp.it + * CxImage version 5.99c 17/Oct/2004 + */ + +#include "ximage.h" +#include "ximath.h" + +#if CXIMAGE_SUPPORT_BASICTRANSFORMATIONS +//////////////////////////////////////////////////////////////////////////////// +bool CxImage::GrayScale() +{ + if (!pDib) return false; + if (head.biBitCount<=8){ + RGBQUAD* ppal=GetPalette(); + int gray; + //converts the colors to gray, use the blue channel only + for(DWORD i=0;iIsValid()) return false; + BYTE *iSrc,*iDst; + iSrc=info.pImage + (head.biHeight-1)*info.dwEffWidth; + iDst=imatmp->info.pImage; + for(long y=0; y < head.biHeight; y++){ + memcpy(iDst,iSrc,info.dwEffWidth); + iSrc-=info.dwEffWidth; + iDst+=info.dwEffWidth; + } + +#if CXIMAGE_SUPPORT_ALPHA + imatmp->AlphaFlip(); +#endif //CXIMAGE_SUPPORT_ALPHA + + Transfer(*imatmp); + delete imatmp; + return true; +} +//////////////////////////////////////////////////////////////////////////////// +bool CxImage::Mirror() +{ + if (!pDib) return false; + + CxImage* imatmp = new CxImage(*this,false,false,true); + if (!imatmp) return false; + BYTE *iSrc,*iDst; + long wdt=(head.biWidth-1) * (head.biBitCount==24 ? 3:1); + iSrc=info.pImage + wdt; + iDst=imatmp->info.pImage; + long x,y; + switch (head.biBitCount){ + case 24: + for(y=0; y < head.biHeight; y++){ + for(x=0; x <= wdt; x+=3){ + *(iDst+x)=*(iSrc-x); + *(iDst+x+1)=*(iSrc-x+1); + *(iDst+x+2)=*(iSrc-x+2); + } + iSrc+=info.dwEffWidth; + iDst+=info.dwEffWidth; + } + break; + case 8: + for(y=0; y < head.biHeight; y++){ + for(x=0; x <= wdt; x++) + *(iDst+x)=*(iSrc-x); + iSrc+=info.dwEffWidth; + iDst+=info.dwEffWidth; + } + break; + default: + for(y=0; y < head.biHeight; y++){ + for(x=0; x <= wdt; x++) + imatmp->SetPixelIndex(x,y,GetPixelIndex(wdt-x,y)); + } + } + +#if CXIMAGE_SUPPORT_ALPHA + imatmp->AlphaMirror(); +#endif //CXIMAGE_SUPPORT_ALPHA + + Transfer(*imatmp); + delete imatmp; + return true; +} + +//////////////////////////////////////////////////////////////////////////////// +#define RBLOCK 64 + +//////////////////////////////////////////////////////////////////////////////// +bool CxImage::RotateLeft(CxImage* iDst) +{ + if (!pDib) return false; + + long newWidth = GetHeight(); + long newHeight = GetWidth(); + + CxImage imgDest; + imgDest.CopyInfo(*this); + imgDest.Create(newWidth,newHeight,GetBpp(),GetType()); + imgDest.SetPalette(GetPalette()); + +#if CXIMAGE_SUPPORT_ALPHA + if (AlphaIsValid()) imgDest.AlphaCreate(); +#endif + + long x,x2,y,dlineup; + + // Speedy rotate for BW images + if (head.biBitCount == 1) { + + BYTE *sbits, *dbits, *dbitsmax, bitpos, *nrow,*srcdisp; + div_t div_r; + + BYTE *bsrc = GetBits(), *bdest = imgDest.GetBits(); + dbitsmax = bdest + imgDest.head.biSizeImage - 1; + dlineup = 8 * imgDest.info.dwEffWidth - imgDest.head.biWidth; + + imgDest.Clear(0); + for (y = 0; y < head.biHeight; y++) { + // Figure out the Column we are going to be copying to + div_r = div((int)(y + dlineup), 8); + // set bit pos of src column byte + bitpos = 1 << div_r.rem; + srcdisp = bsrc + y * info.dwEffWidth; + for (x = 0; x < (long)info.dwEffWidth; x++) { + // Get Source Bits + sbits = srcdisp + x; + // Get destination column + nrow = bdest + (x * 8) * imgDest.info.dwEffWidth + imgDest.info.dwEffWidth - 1 - div_r.quot; + for (long z = 0; z < 8; z++) { + // Get Destination Byte + dbits = nrow + z * imgDest.info.dwEffWidth; + if ((dbits < bdest) || (dbits > dbitsmax)) break; + if (*sbits & (128 >> z)) *dbits |= bitpos; + } + } + }//for y + +#if CXIMAGE_SUPPORT_ALPHA + if (AlphaIsValid()) { + for (x = 0; x < newWidth; x++){ + x2=newWidth-x-1; + for (y = 0; y < newHeight; y++){ + imgDest.AlphaSet(x,y,BlindAlphaGet(y, x2)); + }//for y + }//for x + } +#endif //CXIMAGE_SUPPORT_ALPHA + + } else { + //anything other than BW: + //bd, 10. 2004: This optimized version of rotation rotates image by smaller blocks. It is quite + //a bit faster than obvious algorithm, because it produces much less CPU cache misses. + //This optimization can be tuned by changing block size (RBLOCK). 96 is good value for current + //CPUs (tested on Athlon XP and Celeron D). Larger value (if CPU has enough cache) will increase + //speed somehow, but once you drop out of CPU's cache, things will slow down drastically. + //For older CPUs with less cache, lower value would yield better results. + + BYTE *srcPtr, *dstPtr; //source and destionation for 24-bit version + int xs, ys; //x-segment and y-segment + for (xs = 0; xs < newWidth; xs+=RBLOCK) { //for all image blocks of RBLOCK*RBLOCK pixels + for (ys = 0; ys < newHeight; ys+=RBLOCK) { + if (head.biBitCount==24) { + //RGB24 optimized pixel access: + for (x = xs; x < min(newWidth, xs+RBLOCK); x++){ //do rotation + info.nProgress = (long)(100*x/newWidth); + x2=newWidth-x-1; + dstPtr = (BYTE*) imgDest.BlindGetPixelPointer(x,ys); + srcPtr = (BYTE*) BlindGetPixelPointer(ys, x2); + for (y = ys; y < min(newHeight, ys+RBLOCK); y++){ + //imgDest.SetPixelColor(x, y, GetPixelColor(y, x2)); + *(dstPtr) = *(srcPtr); + *(dstPtr+1) = *(srcPtr+1); + *(dstPtr+2) = *(srcPtr+2); + srcPtr += 3; + dstPtr += imgDest.info.dwEffWidth; + }//for y + }//for x + } else { + //anything else than 24bpp (and 1bpp): palette + for (x = xs; x < min(newWidth, xs+RBLOCK); x++){ + info.nProgress = (long)(100*x/newWidth); // + x2=newWidth-x-1; + for (y = ys; y < min(newHeight, ys+RBLOCK); y++){ + imgDest.SetPixelIndex(x, y, BlindGetPixelIndex(y, x2)); + }//for y + }//for x + }//if (version selection) +#if CXIMAGE_SUPPORT_ALPHA + if (pAlpha) { + for (x = xs; x < min(newWidth, xs+RBLOCK); x++){ + x2=newWidth-x-1; + for (y = ys; y < min(newHeight, ys+RBLOCK); y++){ + imgDest.AlphaSet(x,y,BlindAlphaGet(y, x2)); + }//for y + }//for x + }//if (alpha channel) +#endif //CXIMAGE_SUPPORT_ALPHA + }//for ys + }//for xs + }//if + + //select the destination + if (iDst) iDst->Transfer(imgDest); + else Transfer(imgDest); + return true; +} + +//////////////////////////////////////////////////////////////////////////////// +bool CxImage::RotateRight(CxImage* iDst) +{ + if (!pDib) return false; + + long newWidth = GetHeight(); + long newHeight = GetWidth(); + + CxImage imgDest; + imgDest.CopyInfo(*this); + imgDest.Create(newWidth,newHeight,GetBpp(),GetType()); + imgDest.SetPalette(GetPalette()); + +#if CXIMAGE_SUPPORT_ALPHA + if (AlphaIsValid()) imgDest.AlphaCreate(); +#endif + + long x,y,y2; + // Speedy rotate for BW images + if (head.biBitCount == 1) { + + BYTE *sbits, *dbits, *dbitsmax, bitpos, *nrow,*srcdisp; + div_t div_r; + + BYTE *bsrc = GetBits(), *bdest = imgDest.GetBits(); + dbitsmax = bdest + imgDest.head.biSizeImage - 1; + + imgDest.Clear(0); + for (y = 0; y < head.biHeight; y++) { + // Figure out the Column we are going to be copying to + div_r = div((int)y, 8); + // set bit pos of src column byte + bitpos = 128 >> div_r.rem; + srcdisp = bsrc + y * info.dwEffWidth; + for (x = 0; x < (long)info.dwEffWidth; x++) { + // Get Source Bits + sbits = srcdisp + x; + // Get destination column + nrow = bdest + (imgDest.head.biHeight-1-(x*8)) * imgDest.info.dwEffWidth + div_r.quot; + for (long z = 0; z < 8; z++) { + // Get Destination Byte + dbits = nrow - z * imgDest.info.dwEffWidth; + if ((dbits < bdest) || (dbits > dbitsmax)) break; + if (*sbits & (128 >> z)) *dbits |= bitpos; + } + } + } + +#if CXIMAGE_SUPPORT_ALPHA + if (AlphaIsValid()){ + for (y = 0; y < newHeight; y++){ + y2=newHeight-y-1; + for (x = 0; x < newWidth; x++){ + imgDest.AlphaSet(x,y,BlindAlphaGet(y2, x)); + } + } + } +#endif //CXIMAGE_SUPPORT_ALPHA + + } else { + //anything else but BW + BYTE *srcPtr, *dstPtr; //source and destionation for 24-bit version + int xs, ys; //x-segment and y-segment + for (xs = 0; xs < newWidth; xs+=RBLOCK) { + for (ys = 0; ys < newHeight; ys+=RBLOCK) { + if (head.biBitCount==24) { + //RGB24 optimized pixel access: + for (y = ys; y < min(newHeight, ys+RBLOCK); y++){ + info.nProgress = (long)(100*y/newHeight); // + y2=newHeight-y-1; + dstPtr = (BYTE*) imgDest.BlindGetPixelPointer(xs,y); + srcPtr = (BYTE*) BlindGetPixelPointer(y2, xs); + for (x = xs; x < min(newWidth, xs+RBLOCK); x++){ + //imgDest.SetPixelColor(x, y, GetPixelColor(y2, x)); + *(dstPtr) = *(srcPtr); + *(dstPtr+1) = *(srcPtr+1); + *(dstPtr+2) = *(srcPtr+2); + dstPtr += 3; + srcPtr += info.dwEffWidth; + }//for x + }//for y + } else { + //anything else than BW & RGB24: palette + for (y = ys; y < min(newHeight, ys+RBLOCK); y++){ + info.nProgress = (long)(100*y/newHeight); // + y2=newHeight-y-1; + for (x = xs; x < min(newWidth, xs+RBLOCK); x++){ + imgDest.SetPixelIndex(x, y, BlindGetPixelIndex(y2, x)); + }//for x + }//for y + }//if +#if CXIMAGE_SUPPORT_ALPHA + if (pAlpha){ + for (y = ys; y < min(newHeight, ys+RBLOCK); y++){ + y2=newHeight-y-1; + for (x = xs; x < min(newWidth, xs+RBLOCK); x++){ + imgDest.AlphaSet(x,y,BlindAlphaGet(y2, x)); + }//for x + }//for y + }//if (has alpha) +#endif //CXIMAGE_SUPPORT_ALPHA + }//for ys + }//for xs + }//if + + //select the destination + if (iDst) iDst->Transfer(imgDest); + else Transfer(imgDest); + return true; +} + +//////////////////////////////////////////////////////////////////////////////// +bool CxImage::Negative() +{ + if (!pDib) return false; + + if (head.biBitCount<=8){ + if (IsGrayScale()){ //GRAYSCALE, selection + if (pSelection){ + for(long y=info.rSelectionBox.bottom; y invert transparent color too + info.nBkgndColor.rgbBlue = (BYTE)(255-info.nBkgndColor.rgbBlue); + info.nBkgndColor.rgbGreen = (BYTE)(255-info.nBkgndColor.rgbGreen); + info.nBkgndColor.rgbRed = (BYTE)(255-info.nBkgndColor.rgbRed); + } + return true; +} + +//////////////////////////////////////////////////////////////////////////////// +#endif //CXIMAGE_SUPPORT_BASICTRANSFORMATIONS +//////////////////////////////////////////////////////////////////////////////// +#if CXIMAGE_SUPPORT_TRANSFORMATION +//////////////////////////////////////////////////////////////////////////////// + +//////////////////////////////////////////////////////////////////////////////// +bool CxImage::Rotate(float angle, CxImage* iDst) +{ + if (!pDib) return false; + + // Copyright (c) 1996-1998 Ulrich von Zadow + + // Negative the angle, because the y-axis is negative. + double ang = -angle*acos((float)0)/90; + int newWidth, newHeight; + int nWidth = GetWidth(); + int nHeight= GetHeight(); + double cos_angle = cos(ang); + double sin_angle = sin(ang); + + // Calculate the size of the new bitmap + POINT p1={0,0}; + POINT p2={nWidth,0}; + POINT p3={0,nHeight}; + POINT p4={nWidth-1,nHeight}; + POINT newP1,newP2,newP3,newP4, leftTop, rightTop, leftBottom, rightBottom; + + newP1.x = p1.x; + newP1.y = p1.y; + newP2.x = (long)floor(p2.x*cos_angle - p2.y*sin_angle); + newP2.y = (long)floor(p2.x*sin_angle + p2.y*cos_angle); + newP3.x = (long)floor(p3.x*cos_angle - p3.y*sin_angle); + newP3.y = (long)floor(p3.x*sin_angle + p3.y*cos_angle); + newP4.x = (long)floor(p4.x*cos_angle - p4.y*sin_angle); + newP4.y = (long)floor(p4.x*sin_angle + p4.y*cos_angle); + + leftTop.x = min(min(newP1.x,newP2.x),min(newP3.x,newP4.x)); + leftTop.y = min(min(newP1.y,newP2.y),min(newP3.y,newP4.y)); + rightBottom.x = max(max(newP1.x,newP2.x),max(newP3.x,newP4.x)); + rightBottom.y = max(max(newP1.y,newP2.y),max(newP3.y,newP4.y)); + leftBottom.x = leftTop.x; + leftBottom.y = 2+rightBottom.y; + rightTop.x = 2+rightBottom.x; + rightTop.y = leftTop.y; + + newWidth = rightTop.x - leftTop.x; + newHeight= leftBottom.y - leftTop.y; + CxImage imgDest; + imgDest.CopyInfo(*this); + imgDest.Create(newWidth,newHeight,GetBpp(),GetType()); + imgDest.SetPalette(GetPalette()); + +#if CXIMAGE_SUPPORT_ALPHA + if(AlphaIsValid()) //MTA: Fix for rotation problem when the image has an alpha channel + { + imgDest.AlphaCreate(); + imgDest.AlphaClear(); + } +#endif //CXIMAGE_SUPPORT_ALPHA + + int x,y,newX,newY,oldX,oldY; + + if (head.biClrUsed==0){ //RGB + for (y = leftTop.y, newY = 0; y<=leftBottom.y; y++,newY++){ + info.nProgress = (long)(100*newY/newHeight); + if (info.nEscape) break; + for (x = leftTop.x, newX = 0; x<=rightTop.x; x++,newX++){ + oldX = (long)(x*cos_angle + y*sin_angle - 0.5); + oldY = (long)(y*cos_angle - x*sin_angle - 0.5); + imgDest.SetPixelColor(newX,newY,GetPixelColor(oldX,oldY)); +#if CXIMAGE_SUPPORT_ALPHA + imgDest.AlphaSet(newX,newY,AlphaGet(oldX,oldY)); //MTA: copy the alpha value +#endif //CXIMAGE_SUPPORT_ALPHA + } + } + } else { //PALETTE + for (y = leftTop.y, newY = 0; y<=leftBottom.y; y++,newY++){ + info.nProgress = (long)(100*newY/newHeight); + if (info.nEscape) break; + for (x = leftTop.x, newX = 0; x<=rightTop.x; x++,newX++){ + oldX = (long)(x*cos_angle + y*sin_angle - 0.5); + oldY = (long)(y*cos_angle - x*sin_angle - 0.5); + imgDest.SetPixelIndex(newX,newY,GetPixelIndex(oldX,oldY)); +#if CXIMAGE_SUPPORT_ALPHA + imgDest.AlphaSet(newX,newY,AlphaGet(oldX,oldY)); //MTA: copy the alpha value +#endif //CXIMAGE_SUPPORT_ALPHA + } + } + } + //select the destination + if (iDst) iDst->Transfer(imgDest); + else Transfer(imgDest); + + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Rotates image around it's center. + * Method can use interpolation with paletted images, but does not change pallete, so results vary. + * (If you have only four colours in a palette, there's not much room for interpolation.) + * + * \param angle - angle in degrees (positive values rotate clockwise) + * \param *iDst - destination image (if null, this image is changed) + * \param inMethod - interpolation method used + * (IM_NEAREST_NEIGHBOUR produces aliasing (fast), IM_BILINEAR softens picture a bit (slower) + * IM_SHARPBICUBIC is slower and produces some halos...) + * \param ofMethod - overflow method (how to choose colour of pixels that have no source) + * \param replColor - replacement colour to use (OM_COLOR, OM_BACKGROUND with no background colour...) + * \param optimizeRightAngles - call faster methods for 90, 180, and 270 degree rotations. Faster methods + * are called for angles, where error (in location of corner pixels) is less + * than 0.25 pixels. + * \param bKeepOriginalSize - rotates the image without resizing. + * + * \author ***bd*** 2.2004 + */ +bool CxImage::Rotate2(float angle, + CxImage *iDst, + InterpolationMethod inMethod, + OverflowMethod ofMethod, + RGBQUAD *replColor, + bool const optimizeRightAngles, + bool const bKeepOriginalSize) +{ + if (!pDib) return false; //no dib no go + + double ang = -angle*acos(0.0f)/90.0f; //convert angle to radians and invert (positive angle performs clockwise rotation) + float cos_angle = (float) cos(ang); //these two are needed later (to rotate) + float sin_angle = (float) sin(ang); + + //Calculate the size of the new bitmap (rotate corners of image) + CxPoint2 p[4]; //original corners of the image + p[0]=CxPoint2(-0.5f,-0.5f); + p[1]=CxPoint2(GetWidth()-0.5f,-0.5f); + p[2]=CxPoint2(-0.5f,GetHeight()-0.5f); + p[3]=CxPoint2(GetWidth()-0.5f,GetHeight()-0.5f); + CxPoint2 newp[4]; //rotated positions of corners + //(rotate corners) + if (bKeepOriginalSize){ + for (int i=0; i<4; i++) { + newp[i].x = p[i].x; + newp[i].y = p[i].y; + }//for + } else { + for (int i=0; i<4; i++) { + newp[i].x = (p[i].x*cos_angle - p[i].y*sin_angle); + newp[i].y = (p[i].x*sin_angle + p[i].y*cos_angle); + }//for i + + if (optimizeRightAngles) { + //For rotations of 90, -90 or 180 or 0 degrees, call faster routines + if (newp[3].Distance(CxPoint2(GetHeight()-0.5f, 0.5f-GetWidth())) < 0.25) + //rotation right for circa 90 degrees (diagonal pixels less than 0.25 pixel away from 90 degree rotation destination) + return RotateRight(iDst); + if (newp[3].Distance(CxPoint2(0.5f-GetHeight(), -0.5f+GetWidth())) < 0.25) + //rotation left for ~90 degrees + return RotateLeft(iDst); + if (newp[3].Distance(CxPoint2(0.5f-GetWidth(), 0.5f-GetHeight())) < 0.25) + //rotation left for ~180 degrees + return Rotate180(iDst); + if (newp[3].Distance(p[3]) < 0.25) { + //rotation not significant + if (iDst) iDst->Copy(*this); //copy image to iDst, if required + return true; //and we're done + }//if + }//if + }//if + + //(read new dimensions from location of corners) + float minx = (float) min(min(newp[0].x,newp[1].x),min(newp[2].x,newp[3].x)); + float miny = (float) min(min(newp[0].y,newp[1].y),min(newp[2].y,newp[3].y)); + float maxx = (float) max(max(newp[0].x,newp[1].x),max(newp[2].x,newp[3].x)); + float maxy = (float) max(max(newp[0].y,newp[1].y),max(newp[2].y,newp[3].y)); + int newWidth = (int) floor(maxx-minx+0.5f); + int newHeight= (int) floor(maxy-miny+0.5f); + float ssx=((maxx+minx)- ((float) newWidth-1))/2.0f; //start for x + float ssy=((maxy+miny)- ((float) newHeight-1))/2.0f; //start for y + + float newxcenteroffset = 0.5f * newWidth; + float newycenteroffset = 0.5f * newHeight; + if (bKeepOriginalSize){ + ssx -= 0.5f * GetWidth(); + ssy -= 0.5f * GetHeight(); + } + + //create destination image + CxImage imgDest; + imgDest.CopyInfo(*this); + imgDest.Create(newWidth,newHeight,GetBpp(),GetType()); + imgDest.SetPalette(GetPalette()); +#if CXIMAGE_SUPPORT_ALPHA + if(AlphaIsValid()) imgDest.AlphaCreate(); //MTA: Fix for rotation problem when the image has an alpha channel +#endif //CXIMAGE_SUPPORT_ALPHA + + RGBQUAD rgb; //pixel colour + RGBQUAD rc; + if (replColor!=0) + rc=*replColor; + else { + rc.rgbRed=255; rc.rgbGreen=255; rc.rgbBlue=255; rc.rgbReserved=0; + }//if + float x,y; //destination location (float, with proper offset) + float origx, origy; //origin location + int destx, desty; //destination location + + y=ssy; //initialize y + if (!IsIndexed()){ //RGB24 + //optimized RGB24 implementation (direct write to destination): + BYTE *pxptr; +#if CXIMAGE_SUPPORT_ALPHA + BYTE *pxptra=0; +#endif //CXIMAGE_SUPPORT_ALPHA + for (desty=0; destyTransfer(imgDest); + else Transfer(imgDest); + + return true; +} +//////////////////////////////////////////////////////////////////////////////// +bool CxImage::Rotate180(CxImage* iDst) +{ + if (!pDib) return false; + + long wid = GetWidth(); + long ht = GetHeight(); + + CxImage imgDest; + imgDest.CopyInfo(*this); + imgDest.Create(wid,ht,GetBpp(),GetType()); + imgDest.SetPalette(GetPalette()); + +#if CXIMAGE_SUPPORT_ALPHA + if (AlphaIsValid()) imgDest.AlphaCreate(); +#endif //CXIMAGE_SUPPORT_ALPHA + + long x,y,y2; + for (y = 0; y < ht; y++){ + info.nProgress = (long)(100*y/ht); // + y2=ht-y-1; + for (x = 0; x < wid; x++){ + if(head.biClrUsed==0)//RGB + imgDest.SetPixelColor(wid-x-1, y2, GetPixelColor(x, y)); + else //PALETTE + imgDest.SetPixelIndex(wid-x-1, y2, GetPixelIndex(x, y)); + +#if CXIMAGE_SUPPORT_ALPHA + if (AlphaIsValid()) imgDest.AlphaSet(wid-x-1, y2,AlphaGet(x, y)); +#endif //CXIMAGE_SUPPORT_ALPHA + + } + } + + //select the destination + if (iDst) iDst->Transfer(imgDest); + else Transfer(imgDest); + return true; +} + +//////////////////////////////////////////////////////////////////////////////// +/** + * Resizes the image. mode can be 0 for slow (bilinear) method , + * 1 for fast (nearest pixel) method, or 2 for accurate (bicubic spline interpolation) method. + * The function is faster with 24 and 1 bpp images, slow for 4 bpp images and slowest for 8 bpp images. + */ +bool CxImage::Resample(long newx, long newy, int mode, CxImage* iDst) +{ + if (newx==0 || newy==0) return false; + + if (head.biWidth==newx && head.biHeight==newy){ + if (iDst) iDst->Copy(*this); + return true; + } + + float xScale, yScale, fX, fY; + xScale = (float)head.biWidth / (float)newx; + yScale = (float)head.biHeight / (float)newy; + + CxImage newImage; + newImage.CopyInfo(*this); + newImage.Create(newx,newy,head.biBitCount,GetType()); + newImage.SetPalette(GetPalette()); + if (!newImage.IsValid()) return false; + + switch (mode) { + case 1: // nearest pixel + { + for(long y=0; y=head.biHeight) yy = head.biHeight-1; + for(int n=-1; n<3; n++) { + r2 = r1 * KernelBSpline(b - (float)n); + xx = i_x+n; + if (xx<0) xx=0; + if (xx>=head.biWidth) xx=head.biWidth-1; + + if (head.biClrUsed){ + rgb = GetPixelColor(xx,yy); + } else { + iDst = info.pImage + yy*info.dwEffWidth + xx*3; + rgb.rgbBlue = *iDst++; + rgb.rgbGreen= *iDst++; + rgb.rgbRed = *iDst; + } + + rr += rgb.rgbRed * r2; + gg += rgb.rgbGreen * r2; + bb += rgb.rgbBlue * r2; + } + } + + if (head.biClrUsed) + newImage.SetPixelColor(x,y,RGB(rr,gg,bb)); + else { + iDst = newImage.info.pImage + y*newImage.info.dwEffWidth + x*3; + *iDst++ = (BYTE)bb; + *iDst++ = (BYTE)gg; + *iDst = (BYTE)rr; + } + + } + } + break; + } + default: // bilinear interpolation + if (!(head.biWidth>newx && head.biHeight>newy && head.biBitCount==24)) { + //© 1999 Steve McMahon (steve@dogma.demon.co.uk) + long ifX, ifY, ifX1, ifY1, xmax, ymax; + float ir1, ir2, ig1, ig2, ib1, ib2, dx, dy; + BYTE r,g,b; + RGBQUAD rgb1, rgb2, rgb3, rgb4; + xmax = head.biWidth-1; + ymax = head.biHeight-1; + for(long y=0; y + const long ACCURACY = 1000; + long i,j; // index for faValue + long x,y; // coordinates in source image + BYTE* pSource; + BYTE* pDest = newImage.info.pImage; + long* naAccu = new long[3 * newx + 3]; + long* naCarry = new long[3 * newx + 3]; + long* naTemp; + long nWeightX,nWeightY; + float fEndX; + long nScale = (long)(ACCURACY * xScale * yScale); + + memset(naAccu, 0, sizeof(long) * 3 * newx); + memset(naCarry, 0, sizeof(long) * 3 * newx); + + int u, v = 0; // coordinates in dest image + float fEndY = yScale - 1.0f; + for (y = 0; y < head.biHeight; y++){ + info.nProgress = (long)(100*y/head.biHeight); // + if (info.nEscape) break; + pSource = info.pImage + y * info.dwEffWidth; + u = i = 0; + fEndX = xScale - 1.0f; + if ((float)y < fEndY) { // complete source row goes into dest row + for (x = 0; x < head.biWidth; x++){ + if ((float)x < fEndX){ // complete source pixel goes into dest pixel + for (j = 0; j < 3; j++) naAccu[i + j] += (*pSource++) * ACCURACY; + } else { // source pixel is splitted for 2 dest pixels + nWeightX = (long)(((float)x - fEndX) * ACCURACY); + for (j = 0; j < 3; j++){ + naAccu[i] += (ACCURACY - nWeightX) * (*pSource); + naAccu[3 + i++] += nWeightX * (*pSource++); + } + fEndX += xScale; + u++; + } + } + } else { // source row is splitted for 2 dest rows + nWeightY = (long)(((float)y - fEndY) * ACCURACY); + for (x = 0; x < head.biWidth; x++){ + if ((float)x < fEndX){ // complete source pixel goes into 2 pixel + for (j = 0; j < 3; j++){ + naAccu[i + j] += ((ACCURACY - nWeightY) * (*pSource)); + naCarry[i + j] += nWeightY * (*pSource++); + } + } else { // source pixel is splitted for 4 dest pixels + nWeightX = (int)(((float)x - fEndX) * ACCURACY); + for (j = 0; j < 3; j++) { + naAccu[i] += ((ACCURACY - nWeightY) * (ACCURACY - nWeightX)) * (*pSource) / ACCURACY; + *pDest++ = (BYTE)(naAccu[i] / nScale); + naCarry[i] += (nWeightY * (ACCURACY - nWeightX) * (*pSource)) / ACCURACY; + naAccu[i + 3] += ((ACCURACY - nWeightY) * nWeightX * (*pSource)) / ACCURACY; + naCarry[i + 3] = (nWeightY * nWeightX * (*pSource)) / ACCURACY; + i++; + pSource++; + } + fEndX += xScale; + u++; + } + } + if (u < newx){ // possibly not completed due to rounding errors + for (j = 0; j < 3; j++) *pDest++ = (BYTE)(naAccu[i++] / nScale); + } + naTemp = naCarry; + naCarry = naAccu; + naAccu = naTemp; + memset(naCarry, 0, sizeof(int) * 3); // need only to set first pixel zero + pDest = newImage.info.pImage + (++v * newImage.info.dwEffWidth); + fEndY += yScale; + } + } + if (v < newy){ // possibly not completed due to rounding errors + for (i = 0; i < 3 * newx; i++) *pDest++ = (BYTE)(naAccu[i] / nScale); + } + delete [] naAccu; + delete [] naCarry; + } + } + +#if CXIMAGE_SUPPORT_ALPHA + if (AlphaIsValid()){ + newImage.AlphaCreate(); + for(long y=0; yTransfer(newImage); + else Transfer(newImage); + + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * New simpler resample. Adds new interpolation methods and simplifies code (using GetPixelColorInterpolated + * and GetAreaColorInterpolated). It also (unlike old method) interpolates alpha layer. + * + * \param newx, newy - size of resampled image + * \param inMethod - interpolation method to use (see comments at GetPixelColorInterpolated) + * If image size is being reduced, averaging is used instead (or simultaneously with) inMethod. + * \param ofMethod - what to replace outside pixels by (only significant for bordering pixels of enlarged image) + * \param iDst - pointer to destination CxImage or NULL. + * \param disableAveraging - force no averaging when shrinking images (Produces aliasing. + * You probably just want to leave this off...) + * + * \author ***bd*** 2.2004 + */ +bool CxImage::Resample2( + long newx, long newy, + InterpolationMethod const inMethod, + OverflowMethod const ofMethod, + CxImage* const iDst, + bool const disableAveraging) +{ + if (newx<=0 || newy<=0 || !pDib) return false; + + if (head.biWidth==newx && head.biHeight==newy) { + //image already correct size (just copy and return) + if (iDst) iDst->Copy(*this); + return true; + }//if + + //calculate scale of new image (less than 1 for enlarge) + float xScale, yScale; + xScale = (float)head.biWidth / (float)newx; + yScale = (float)head.biHeight / (float)newy; + + //create temporary destination image + CxImage newImage; + newImage.CopyInfo(*this); + newImage.Create(newx,newy,head.biBitCount,GetType()); + newImage.SetPalette(GetPalette()); + if (!newImage.IsValid()) return false; + + //and alpha channel if required +#if CXIMAGE_SUPPORT_ALPHA + if (AlphaIsValid()) newImage.AlphaCreate(); +#endif + + float sX, sY; //source location + long dX,dY; //destination pixel (int value) + if ((xScale<=1 && yScale<=1) || disableAveraging) { + //image is being enlarged (or interpolation on demand) + if (!IsIndexed()) { + //RGB24 image (optimized version with direct writes) + RGBQUAD q; //pixel colour + BYTE *pxptr; //pointer to destination pixel +#if CXIMAGE_SUPPORT_ALPHA + BYTE *pxptra; //and destination alpha data +#endif + for(dY=0; dYTransfer(newImage); + else + Transfer(newImage); + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Reduces the number of bits per pixel to nbit (1, 4 or 8). + * ppal points to a valid palette for the final image; if not supplied the function will use a standard palette. + * ppal is not necessary for reduction to 1 bpp. + */ +bool CxImage::DecreaseBpp(DWORD nbit, bool errordiffusion, RGBQUAD* ppal, DWORD clrimportant) +{ + if (!pDib) return false; + if (head.biBitCount < nbit) return false; + if (head.biBitCount == nbit){ + if (clrimportant==0) return true; + if (head.biClrImportant && (head.biClrImportant4) return false; + + CxImage tmp; + tmp.CopyInfo(*this); + tmp.Create(head.biWidth,head.biHeight,4,info.dwType); + tmp.SetPalette(GetPalette(),GetNumColors()); + if (!tmp.IsValid()) return false; + + +#if CXIMAGE_SUPPORT_SELECTION + tmp.SelectionCopy(*this); +#endif //CXIMAGE_SUPPORT_SELECTION + +#if CXIMAGE_SUPPORT_ALPHA + tmp.AlphaCopy(*this); +#endif //CXIMAGE_SUPPORT_ALPHA + + for (long y=0;y8) return false; + + CxImage tmp; + tmp.CopyInfo(*this); + tmp.Create(head.biWidth,head.biHeight,8,info.dwType); + tmp.SetPalette(GetPalette(),GetNumColors()); + if (!tmp.IsValid()) return false; + +#if CXIMAGE_SUPPORT_SELECTION + tmp.SelectionCopy(*this); +#endif //CXIMAGE_SUPPORT_SELECTION + +#if CXIMAGE_SUPPORT_ALPHA + tmp.AlphaCopy(*this); +#endif //CXIMAGE_SUPPORT_ALPHA + + for (long y=0;y24) return false; + + CxImage tmp; + tmp.CopyInfo(*this); + tmp.Create(head.biWidth,head.biHeight,24,info.dwType); + if (!tmp.IsValid()) return false; + + if (info.nBkgndIndex>=0) //translate transparency + tmp.info.nBkgndColor=GetPaletteColor((BYTE)info.nBkgndIndex); + +#if CXIMAGE_SUPPORT_SELECTION + tmp.SelectionCopy(*this); +#endif //CXIMAGE_SUPPORT_SELECTION + +#if CXIMAGE_SUPPORT_ALPHA + tmp.AlphaCopy(*this); + if (AlphaPaletteIsValid() && !AlphaIsValid()) tmp.AlphaCreate(); +#endif //CXIMAGE_SUPPORT_ALPHA + + for (long y=0;y 128) { + tmp.SetPixelIndex(x, y, 1); + error = level - 255; + } else { + tmp.SetPixelIndex(x, y, 0); + error = level; + } + + nlevel = GetPixelIndex(x + 1, y) + (error * 8) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(x + 1, y, level); + nlevel = GetPixelIndex(x + 2, y) + (error * 4) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(x + 2, y, level); + int i; + for (i = -2; i < 3; i++) { + switch (i) { + case -2: + coeff = 2; + break; + case -1: + coeff = 4; + break; + case 0: + coeff = 8; + break; + case 1: + coeff = 4; + break; + case 2: + coeff = 2; + break; + } + nlevel = GetPixelIndex(x + i, y + 1) + (error * coeff) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(x + i, y + 1, level); + } + } + } + break; + } + case 3: + { + //Stucki error diffusion (Thanks to Franco Gerevini) + int TotalCoeffSum = 42; + long error, nlevel, coeff; + BYTE level; + + for (long y = 0; y < head.biHeight; y++) { + info.nProgress = (long)(100 * y / head.biHeight); + if (info.nEscape) + break; + for (long x = 0; x < head.biWidth; x++) { + level = GetPixelIndex(x, y); + if (level > 128) { + tmp.SetPixelIndex(x, y, 1); + error = level - 255; + } else { + tmp.SetPixelIndex(x, y, 0); + error = level; + } + + nlevel = GetPixelIndex(x + 1, y) + (error * 8) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(x + 1, y, level); + nlevel = GetPixelIndex(x + 2, y) + (error * 4) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(x + 2, y, level); + int i; + for (i = -2; i < 3; i++) { + switch (i) { + case -2: + coeff = 2; + break; + case -1: + coeff = 4; + break; + case 0: + coeff = 8; + break; + case 1: + coeff = 4; + break; + case 2: + coeff = 2; + break; + } + nlevel = GetPixelIndex(x + i, y + 1) + (error * coeff) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(x + i, y + 1, level); + } + for (i = -2; i < 3; i++) { + switch (i) { + case -2: + coeff = 1; + break; + case -1: + coeff = 2; + break; + case 0: + coeff = 4; + break; + case 1: + coeff = 2; + break; + case 2: + coeff = 1; + break; + } + nlevel = GetPixelIndex(x + i, y + 2) + (error * coeff) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(x + i, y + 2, level); + } + } + } + break; + } + case 4: + { + //Jarvis, Judice and Ninke error diffusion (Thanks to Franco Gerevini) + int TotalCoeffSum = 48; + long error, nlevel, coeff; + BYTE level; + + for (long y = 0; y < head.biHeight; y++) { + info.nProgress = (long)(100 * y / head.biHeight); + if (info.nEscape) + break; + for (long x = 0; x < head.biWidth; x++) { + level = GetPixelIndex(x, y); + if (level > 128) { + tmp.SetPixelIndex(x, y, 1); + error = level - 255; + } else { + tmp.SetPixelIndex(x, y, 0); + error = level; + } + + nlevel = GetPixelIndex(x + 1, y) + (error * 7) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(x + 1, y, level); + nlevel = GetPixelIndex(x + 2, y) + (error * 5) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(x + 2, y, level); + int i; + for (i = -2; i < 3; i++) { + switch (i) { + case -2: + coeff = 3; + break; + case -1: + coeff = 5; + break; + case 0: + coeff = 7; + break; + case 1: + coeff = 5; + break; + case 2: + coeff = 3; + break; + } + nlevel = GetPixelIndex(x + i, y + 1) + (error * coeff) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(x + i, y + 1, level); + } + for (i = -2; i < 3; i++) { + switch (i) { + case -2: + coeff = 1; + break; + case -1: + coeff = 3; + break; + case 0: + coeff = 5; + break; + case 1: + coeff = 3; + break; + case 2: + coeff = 1; + break; + } + nlevel = GetPixelIndex(x + i, y + 2) + (error * coeff) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(x + i, y + 2, level); + } + } + } + break; + } + case 5: + { + //Sierra error diffusion (Thanks to Franco Gerevini) + int TotalCoeffSum = 32; + long error, nlevel, coeff; + BYTE level; + + for (long y = 0; y < head.biHeight; y++) { + info.nProgress = (long)(100 * y / head.biHeight); + if (info.nEscape) + break; + for (long x = 0; x < head.biWidth; x++) { + level = GetPixelIndex(x, y); + if (level > 128) { + tmp.SetPixelIndex(x, y, 1); + error = level - 255; + } else { + tmp.SetPixelIndex(x, y, 0); + error = level; + } + + nlevel = GetPixelIndex(x + 1, y) + (error * 5) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(x + 1, y, level); + nlevel = GetPixelIndex(x + 2, y) + (error * 3) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(x + 2, y, level); + int i; + for (i = -2; i < 3; i++) { + switch (i) { + case -2: + coeff = 2; + break; + case -1: + coeff = 4; + break; + case 0: + coeff = 5; + break; + case 1: + coeff = 4; + break; + case 2: + coeff = 2; + break; + } + nlevel = GetPixelIndex(x + i, y + 1) + (error * coeff) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(x + i, y + 1, level); + } + for (i = -1; i < 2; i++) { + switch (i) { + case -1: + coeff = 2; + break; + case 0: + coeff = 3; + break; + case 1: + coeff = 2; + break; + } + nlevel = GetPixelIndex(x + i, y + 2) + (error * coeff) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(x + i, y + 2, level); + } + } + } + break; + } + case 6: + { + //Stevenson and Arce error diffusion (Thanks to Franco Gerevini) + int TotalCoeffSum = 200; + long error, nlevel; + BYTE level; + + for (long y = 0; y < head.biHeight; y++) { + info.nProgress = (long)(100 * y / head.biHeight); + if (info.nEscape) + break; + for (long x = 0; x < head.biWidth; x++) { + level = GetPixelIndex(x, y); + if (level > 128) { + tmp.SetPixelIndex(x, y, 1); + error = level - 255; + } else { + tmp.SetPixelIndex(x, y, 0); + error = level; + } + + int tmp_index_x = x + 2; + int tmp_index_y = y; + int tmp_coeff = 32; + nlevel = GetPixelIndex(tmp_index_x, tmp_index_y) + (error * tmp_coeff) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(tmp_index_x, tmp_index_y, level); + + tmp_index_x = x - 3; + tmp_index_y = y + 1; + tmp_coeff = 12; + nlevel = GetPixelIndex(tmp_index_x, tmp_index_y) + (error * tmp_coeff) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(tmp_index_x, tmp_index_y, level); + + tmp_index_x = x - 1; + tmp_coeff = 26; + nlevel = GetPixelIndex(tmp_index_x, tmp_index_y) + (error * tmp_coeff) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(tmp_index_x, tmp_index_y, level); + + tmp_index_x = x + 1; + tmp_coeff = 30; + nlevel = GetPixelIndex(tmp_index_x, tmp_index_y) + (error * tmp_coeff) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(tmp_index_x, tmp_index_y, level); + + tmp_index_x = x + 3; + tmp_coeff = 16; + nlevel = GetPixelIndex(tmp_index_x, tmp_index_y) + (error * tmp_coeff) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(tmp_index_x, tmp_index_y, level); + + tmp_index_x = x - 2; + tmp_index_y = y + 2; + tmp_coeff = 12; + nlevel = GetPixelIndex(tmp_index_x, tmp_index_y) + (error * tmp_coeff) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(tmp_index_x, tmp_index_y, level); + + tmp_index_x = x; + tmp_coeff = 26; + nlevel = GetPixelIndex(tmp_index_x, tmp_index_y) + (error * tmp_coeff) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(tmp_index_x, tmp_index_y, level); + + tmp_index_x = x + 2; + tmp_coeff = 12; + nlevel = GetPixelIndex(tmp_index_x, tmp_index_y) + (error * tmp_coeff) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(tmp_index_x, tmp_index_y, level); + + tmp_index_x = x - 3; + tmp_index_y = y + 2; + tmp_coeff = 5; + nlevel = GetPixelIndex(tmp_index_x, tmp_index_y) + (error * tmp_coeff) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(tmp_index_x, tmp_index_y, level); + + tmp_index_x = x - 1; + tmp_coeff = 12; + nlevel = GetPixelIndex(tmp_index_x, tmp_index_y) + (error * tmp_coeff) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(tmp_index_x, tmp_index_y, level); + + tmp_index_x = x + 1; + tmp_coeff = 12; + nlevel = GetPixelIndex(tmp_index_x, tmp_index_y) + (error * tmp_coeff) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(tmp_index_x, tmp_index_y, level); + + tmp_index_x = x + 3; + tmp_coeff = 5; + nlevel = GetPixelIndex(tmp_index_x, tmp_index_y) + (error * tmp_coeff) / TotalCoeffSum; + level = (BYTE)min(255, max(0, (int)nlevel)); + SetPixelIndex(tmp_index_x, tmp_index_y, level); + } + } + break; + } + case 7: + { + // Bayer ordered dither + int order = 4; + //create Bayer matrix + if (order>4) order = 4; + int size = (1 << (2*order)); + BYTE* Bmatrix = (BYTE*) malloc(size * sizeof(BYTE)); + for(int i = 0; i < size; i++) { + int n = order; + int x = i / n; + int y = i % n; + int dither = 0; + while (n-- > 0){ + dither = (((dither<<1)|((x&1) ^ (y&1)))<<1) | (y&1); + x >>= 1; + y >>= 1; + } + Bmatrix[i] = dither; + } + + int scale = max(0,(8-2*order)); + int level; + for (long y=0;y> scale; + if(level > Bmatrix[ (x % order) + order * (y % order) ]){ + tmp.SetPixelIndex(x,y,1); + } else { + tmp.SetPixelIndex(x,y,0); + } + } + } + + free(Bmatrix); + + break; + } + default: + { + // Floyd-Steinberg error diffusion (Thanks to Steve McMahon) + long error,nlevel,coeff; + BYTE level; + + for (long y=0;y 128){ + tmp.SetPixelIndex(x,y,1); + error = level-255; + } else { + tmp.SetPixelIndex(x,y,0); + error = level; + } + + nlevel = GetPixelIndex(x+1,y) + (error * 7)/16; + level = (BYTE)min(255,max(0,(int)nlevel)); + SetPixelIndex(x+1,y,level); + for(int i=-1; i<2; i++){ + switch(i){ + case -1: + coeff=3; break; + case 0: + coeff=5; break; + case 1: + coeff=1; break; + } + nlevel = GetPixelIndex(x+i,y+1) + (error * coeff)/16; + level = (BYTE)min(255,max(0,(int)nlevel)); + SetPixelIndex(x+i,y+1,level); + } + } + } + } + } + + tmp.SetPaletteColor(0,0,0,0); + tmp.SetPaletteColor(1,255,255,255); + Transfer(tmp); + + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * CropRotatedRectangle + * \param topx,topy : topmost and leftmost point of the rectangle + (topmost, and if there are 2 topmost points, the left one) + * \param width : size of the right hand side of rect, from (topx,topy) roundwalking clockwise + * \param height : size of the left hand side of rect, from (topx,topy) roundwalking clockwise + * \param angle : angle of the right hand side of rect, from (topx,topy) + * \param iDst : pointer to destination image (if 0, this image is modified) + * \author [VATI] + */ +bool CxImage::CropRotatedRectangle( long topx, long topy, long width, long height, float angle, CxImage* iDst) +{ + if (!pDib) return false; + + + long startx,starty,endx,endy; + double cos_angle = cos(angle/*/57.295779513082320877*/); + double sin_angle = sin(angle/*/57.295779513082320877*/); + + // if there is nothing special, call the original Crop(): + if ( fabs(angle)<0.0002 ) + return Crop( topx, topy, topx+width, topy+height, iDst); + + startx = min(topx, topx - (long)(sin_angle*(double)height)); + endx = topx + (long)(cos_angle*(double)width); + endy = topy + (long)(cos_angle*(double)height + sin_angle*(double)width); + // check: corners of the rectangle must be inside + if ( IsInside( startx, topy )==false || + IsInside( endx, endy ) == false ) + return false; + + // first crop to bounding rectangle + CxImage tmp(*this,false/*pSelection!=0*/,true,true); + tmp.Copy(*this, true, false, true); + if (!tmp.IsValid()) return false; + if ( false == tmp.Crop( startx, topy, endx, endy ) ) + return false; + + // the midpoint of the image now became the same as the midpoint of the rectangle + // rotate new image with minus angle amount + if ( false == tmp.Rotate( (float)(-angle*57.295779513082320877) ) ) // Rotate expects angle in degrees + return false; + + // crop rotated image to the original selection rectangle + endx = (tmp.head.biWidth+width)/2; + startx = (tmp.head.biWidth-width)/2; + starty = (tmp.head.biHeight+height)/2; + endy = (tmp.head.biHeight-height)/2; + if ( false == tmp.Crop( startx, starty, endx, endy ) ) + return false; + + if (iDst) iDst->Transfer(tmp); + else Transfer(tmp); + + return true; +} +//////////////////////////////////////////////////////////////////////////////// +bool CxImage::Crop(const RECT& rect, CxImage* iDst) +{ + return Crop(rect.left, rect.top, rect.right, rect.bottom, iDst); +} +//////////////////////////////////////////////////////////////////////////////// +bool CxImage::Crop(long left, long top, long right, long bottom, CxImage* iDst) +{ + if (!pDib) return false; + + long startx = max(0L,min(left,head.biWidth)); + long endx = max(0L,min(right,head.biWidth)); + long starty = head.biHeight - max(0L,min(top,head.biHeight)); + long endy = head.biHeight - max(0L,min(bottom,head.biHeight)); + + if (startx==endx || starty==endy) return false; + + if (startx>endx) {long tmp=startx; startx=endx; endx=tmp;} + if (starty>endy) {long tmp=starty; starty=endy; endy=tmp;} + + CxImage tmp(endx-startx,endy-starty,head.biBitCount,info.dwType); + if (!tmp.IsValid()) return false; + tmp.SetPalette(GetPalette(),head.biClrUsed); + tmp.info.nBkgndIndex = info.nBkgndIndex; + tmp.info.nBkgndColor = info.nBkgndColor; + + switch (head.biBitCount) { + case 1: + case 4: + { + for(long y=starty, yd=0; y + for(long x=startx, xd=0; x> 3; + BYTE* pDest = tmp.info.pImage; + BYTE* pSrc = info.pImage + starty * info.dwEffWidth + (startx*head.biBitCount >> 3); + for(long y=starty; y + memcpy(pDest,pSrc,linelen); + pDest+=tmp.info.dwEffWidth; + pSrc+=info.dwEffWidth; + } + } + } + +#if CXIMAGE_SUPPORT_ALPHA + if (AlphaIsValid()){ // + tmp.AlphaCreate(); + if (!tmp.AlphaIsValid()) return false; + BYTE* pDest = tmp.pAlpha; + BYTE* pSrc = pAlpha + startx + starty*head.biWidth; + for (long y=starty; yTransfer(tmp); + else Transfer(tmp); + + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * \param xgain, ygain : can be from 0 to 1. + * \param xpivot, ypivot : is the center of the transformation. + * \param bEnableInterpolation : if true, enables bilinear interpolation. + * \return true if everything is ok + */ +bool CxImage::Skew(float xgain, float ygain, long xpivot, long ypivot, bool bEnableInterpolation) +{ + if (!pDib) return false; + float nx,ny; + + CxImage tmp(*this,pSelection!=0,true,true); + if (!tmp.IsValid()) return false; + + long xmin,xmax,ymin,ymax; + if (pSelection){ + xmin = info.rSelectionBox.left; xmax = info.rSelectionBox.right; + ymin = info.rSelectionBox.bottom; ymax = info.rSelectionBox.top; + } else { + xmin = ymin = 0; + xmax = head.biWidth; ymax=head.biHeight; + } + for(long y=ymin; y top) || (x < left) || (x > right)) { + tmp.SetPixelIndex(x,y, pixel); + } else { + tmp.SetPixelIndex(x,y,GetPixelIndex(x-left,y-bottom)); + } + } + } + break; + } + case 8: + case 24: + { + if (head.biBitCount == 8) { + BYTE pixel = tmp.GetNearestIndex( canvascolor); + memset(tmp.info.pImage, pixel, + (tmp.info.dwEffWidth * newHeight)); + } else { + for (long y = 0; y < newHeight; ++y) { + BYTE *pDest = tmp.info.pImage + (y * tmp.info.dwEffWidth); + for (long x = 0; x < newWidth; ++x) { + *pDest++ = canvascolor.rgbBlue; + *pDest++ = canvascolor.rgbGreen; + *pDest++ = canvascolor.rgbRed; + } + } + } + + BYTE* pDest = tmp.info.pImage + (tmp.info.dwEffWidth * bottom) + (left*(head.biBitCount >> 3)); + BYTE* pSrc = info.pImage; + for(long y=bottom; y <= top; y++){ + info.nProgress = (long)(100*y/(1 + top - bottom)); + memcpy(pDest,pSrc,(head.biBitCount >> 3) * (right - left + 1)); + pDest+=tmp.info.dwEffWidth; + pSrc+=info.dwEffWidth; + } + } + } + //select the destination + if (iDst) iDst->Transfer(tmp); + else Transfer(tmp); + + return true; +} +//////////////////////////////////////////////////////////////////////////////// +bool CxImage::Expand(long newx, long newy, RGBQUAD canvascolor, CxImage* iDst) +{ + //thanks to + + if (!pDib) return false; + + if ((newx < head.biWidth) || (newy < head.biHeight)) return false; + + int nAddLeft = (newx - head.biWidth) / 2; + int nAddTop = (newy - head.biHeight) / 2; + + return Expand(nAddLeft, nAddTop, newx - (head.biWidth + nAddLeft), newy - (head.biHeight + nAddTop), canvascolor, iDst); +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Resamples the image with the correct aspect ratio, and fills the borders. + * \param newx, newy = thumbnail size. + * \param canvascolor = border color. + * \param iDst = pointer to destination image (if it's 0, this image is modified). + * \return true if everything is ok. + * \author [Colin Urquhart] + */ +bool CxImage::Thumbnail(long newx, long newy, RGBQUAD canvascolor, CxImage* iDst) +{ + if (!pDib) return false; + + if ((newx <= 0) || (newy <= 0)) return false; + + CxImage tmp(*this); + if (!tmp.IsValid()) return false; + + // determine whether we need to shrink the image + if ((head.biWidth > newx) || (head.biHeight > newy)) { + float fScale; + float fAspect = (float) newx / (float) newy; + if (fAspect * head.biHeight > head.biWidth) { + fScale = (float) newy / head.biHeight; + } else { + fScale = (float) newx / head.biWidth; + } + tmp.Resample((long) (fScale * head.biWidth), (long) (fScale * head.biHeight), 0); + } + + // expand the frame + tmp.Expand(newx, newy, canvascolor, iDst); + + //select the destination + if (iDst) iDst->Transfer(tmp); + else Transfer(tmp); + return true; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Perform circle_based transformations. + * \param type - for different transformations + * - 0 for normal (proturberant) FishEye + * - 1 for reverse (concave) FishEye + * - 2 for Swirle + * - 3 for Cilinder mirror + * - 4 for bathroom + * + * \param rmax - effect radius. If 0, the whole image is processed + * \param Koeff - only for swirle + * \author Arkadiy Olovyannikov ark(at)msun(dot)ru + */ +bool CxImage::CircleTransform(int type,long rmax,float Koeff) +{ + if (!pDib) return false; + + long nx,ny; + double angle,radius,rnew; + + CxImage tmp(*this,pSelection!=0,true,true); + if (!tmp.IsValid()) return false; + + long xmin,xmax,ymin,ymax,xmid,ymid; + if (pSelection){ + xmin = info.rSelectionBox.left; xmax = info.rSelectionBox.right; + ymin = info.rSelectionBox.bottom; ymax = info.rSelectionBox.top; + } else { + xmin = ymin = 0; + xmax = head.biWidth; ymax=head.biHeight; + } + + xmid = (long) (tmp.GetWidth()/2); + ymid = (long) (tmp.GetHeight()/2); + + if (!rmax) rmax=(long)sqrt((float)((xmid-xmin)*(xmid-xmin)+(ymid-ymin)*(ymid-ymin))); + if (Koeff==0.0f) Koeff=1.0f; + + for(long y=ymin; yhead.biWidth || newy>head.biHeight) { + //let me repeat... this method can't enlarge image + strcpy(info.szLastError,"QIShrink can't enlarge image"); + return false; + } + + if (newx==head.biWidth && newy==head.biHeight) { + //image already correct size (just copy and return) + if (iDst) iDst->Copy(*this); + return true; + }//if + + //create temporary destination image + CxImage newImage; + newImage.CopyInfo(*this); + newImage.Create(newx,newy,head.biBitCount,GetType()); + newImage.SetPalette(GetPalette()); + if (!newImage.IsValid()) return false; + + //and alpha channel if required +#if CXIMAGE_SUPPORT_ALPHA + if (AlphaIsValid()) newImage.AlphaCreate(); +#endif + + const int oldx = head.biWidth; + const int oldy = head.biHeight; + + int accuCellSize = 4; +#if CXIMAGE_SUPPORT_ALPHA + BYTE *alphaPtr; + if (AlphaIsValid()) accuCellSize=5; +#endif + + unsigned int *accu = new unsigned int[newx*accuCellSize]; //array for suming pixels... one pixel for every destination column + unsigned int *accuPtr; //pointer for walking through accu + //each cell consists of blue, red, green component and count of pixels summed in this cell + memset(accu, 0, newx * accuCellSize * sizeof(unsigned int)); //clear accu + + if (!IsIndexed()) { + //RGB24 version with pointers + BYTE *destPtr, *srcPtr, *destPtrS, *srcPtrS; //destination and source pixel, and beginnings of current row + srcPtrS=(BYTE*)BlindGetPixelPointer(0,0); + destPtrS=(BYTE*)newImage.BlindGetPixelPointer(0,0); + int ex=0, ey=0; //ex and ey replace division... + int dy=0; + //(we just add pixels, until by adding newx or newy we get a number greater than old size... then + // it's time to move to next pixel) + + for(int y=0; yoldx) { //when we reach oldx, it's time to move to new slot + accuPtr += accuCellSize; + ex -= oldx; //(substract oldx from ex and resume from there on) + }//if (ex overflow) + }//for x + + if (ey>=oldy) { //now when this happens + ey -= oldy; //it's time to move to new destination row + destPtr = destPtrS; //reset pointers to proper initial values + accuPtr = accu; +#if CXIMAGE_SUPPORT_ALPHA + alphaPtr = newImage.AlphaGetPointer(0, dy++); +#endif + for (int k=0; koldx) { //when we reach oldx, it's time to move to new slot + accuPtr += accuCellSize; + ex -= oldx; //(substract oldx from ex and resume from there on) + }//if (ex overflow) + }//for x + + if (ey>=oldy) { //now when this happens + ey -= oldy; //it's time to move to new destination row + accuPtr = accu; + for (int dx=0; dxTransfer(newImage); + else + Transfer(newImage); + return true; + +} + +//////////////////////////////////////////////////////////////////////////////// +#endif //CXIMAGE_SUPPORT_TRANSFORMATION diff --git a/src/win32/dependencies/cximage/ximawbmp.cpp b/src/win32/dependencies/cximage/ximawbmp.cpp new file mode 100644 index 00000000..0cb3f56c --- /dev/null +++ b/src/win32/dependencies/cximage/ximawbmp.cpp @@ -0,0 +1,85 @@ +/* + * File: ximawbmp.cpp + * Purpose: Platform Independent WBMP Image Class Loader and Writer + * 12/Jul/2002 Davide Pizzolato - www.xdp.it + * CxImage version 5.99c 17/Oct/2004 + */ + +#include "ximawbmp.h" + +#if CXIMAGE_SUPPORT_WBMP + +#include "ximaiter.h" + +//////////////////////////////////////////////////////////////////////////////// +bool CxImageWBMP::Decode(CxFile *hFile) +{ + if (hFile == NULL) return false; + + WBMPHEADER wbmpHead; + + try + { + if (hFile->Read(&wbmpHead,sizeof(wbmpHead),1)==0) + throw "Not a WBMP"; + + if (wbmpHead.Type != 0) + throw "Unsupported WBMP type"; + + if (wbmpHead.ImageHeight==0 || wbmpHead.ImageWidth==0) + throw "Corrupted WBMP"; + + Create(wbmpHead.ImageWidth, wbmpHead.ImageHeight, 1, CXIMAGE_FORMAT_WBMP); + if (!IsValid()) throw "WBMP Create failed"; + SetGrayPalette(); + + int linewidth=(wbmpHead.ImageWidth+7)/8; + CImageIterator iter(this); + iter.Upset(); + for (int y=0; y < wbmpHead.ImageHeight; y++){ + hFile->Read(iter.GetRow(),linewidth,1); + iter.PrevRow(); + } + + } catch (char *message) { + strncpy(info.szLastError,message,255); + return FALSE; + } + return true; +} +//////////////////////////////////////////////////////////////////////////////// +#if CXIMAGE_SUPPORT_ENCODE +//////////////////////////////////////////////////////////////////////////////// +bool CxImageWBMP::Encode(CxFile * hFile) +{ + if (EncodeSafeCheck(hFile)) return false; + + //check format limits + if ((head.biWidth>255)||(head.biHeight>255)||(head.biBitCount!=1)){ + strcpy(info.szLastError,"Can't save this image as WBMP"); + return false; + } + + WBMPHEADER wbmpHead; + wbmpHead.Type=0; + wbmpHead.FixHeader=0; + wbmpHead.ImageWidth=(BYTE)head.biWidth; + wbmpHead.ImageHeight=(BYTE)head.biHeight; + + // Write the file header + hFile->Write(&wbmpHead,sizeof(wbmpHead),1); + // Write the pixels + int linewidth=(wbmpHead.ImageWidth+7)/8; + CImageIterator iter(this); + iter.Upset(); + for (int y=0; y < wbmpHead.ImageHeight; y++){ + hFile->Write(iter.GetRow(),linewidth,1); + iter.PrevRow(); + } + return true; +} +//////////////////////////////////////////////////////////////////////////////// +#endif // CXIMAGE_SUPPORT_ENCODE +//////////////////////////////////////////////////////////////////////////////// +#endif // CXIMAGE_SUPPORT_WBMP + diff --git a/src/win32/dependencies/cximage/ximawbmp.h b/src/win32/dependencies/cximage/ximawbmp.h new file mode 100644 index 00000000..5b688c64 --- /dev/null +++ b/src/win32/dependencies/cximage/ximawbmp.h @@ -0,0 +1,44 @@ +/* + * File: ximawbmp.h + * Purpose: WBMP Image Class Loader and Writer + */ +/* ========================================================== + * CxImageWBMP (c) 12/Jul/2002 Davide Pizzolato - www.xdp.it + * For conditions of distribution and use, see copyright notice in ximage.h + * ========================================================== + */ +#if !defined(__ximaWBMP_h) +#define __ximaWBMP_h + +#include "ximage.h" + +#if CXIMAGE_SUPPORT_WBMP + +class CxImageWBMP: public CxImage +{ +#pragma pack(1) +typedef struct tagWbmpHeader +{ + BYTE Type; // 0 + BYTE FixHeader; // 0 + BYTE ImageWidth; // Image Width + BYTE ImageHeight; // Image Height +} WBMPHEADER; +#pragma pack() +public: + CxImageWBMP(): CxImage(CXIMAGE_FORMAT_WBMP) {} + +// bool Load(const char * imageFileName){ return CxImage::Load(imageFileName,CXIMAGE_FORMAT_WBMP);} +// bool Save(const char * imageFileName){ return CxImage::Save(imageFileName,CXIMAGE_FORMAT_WBMP);} + bool Decode(CxFile * hFile); + bool Decode(FILE *hFile) { CxIOFile file(hFile); return Decode(&file); } + +#if CXIMAGE_SUPPORT_ENCODE + bool Encode(CxFile * hFile); + bool Encode(FILE *hFile) { CxIOFile file(hFile); return Encode(&file); } +#endif // CXIMAGE_SUPPORT_ENCODE +}; + +#endif + +#endif diff --git a/src/win32/dependencies/cximage/ximawmf.cpp b/src/win32/dependencies/cximage/ximawmf.cpp new file mode 100644 index 00000000..3c50850f --- /dev/null +++ b/src/win32/dependencies/cximage/ximawmf.cpp @@ -0,0 +1,458 @@ +/* +********************************************************************* + * File: ximawmf.cpp + * Purpose: Windows Metafile Class Loader and Writer + * Author: Volker Horch - vhorch@gmx.de + * created: 13-Jun-2002 + * + * Note: If the code below works, i wrote it. + * If it doesn't work, i don't know who wrote it. +********************************************************************* + */ + +/* +********************************************************************* + Note by Author: +********************************************************************* + + Metafile Formats: + ================= + + There are 2 kinds of Windows Metafiles: + - Standard Windows Metafile + - Placeable Windows Metafile + + A StandardWindows Metafile looks like: + - Metafile Header (MEATAHEADER) + - Metafile Records + + A Placeable Metafile looks like: + - Aldus Header (METAFILEHEADER) + - Metafile Header (METAHEADER) + - Metafile Records + + The "Metafile Header" and the "Metafile Records" are the same + for both formats. However, the Standard Metafile does not contain any + information about the original dimensions or x/y ratio of the Metafile. + + I decided, to allow only placeable Metafiles here. If you also want to + enable Standard Metafiles, you will have to guess the dimensions of + the image. + +********************************************************************* + Limitations: see ximawmf.h + you may configure some stuff there +********************************************************************* +*/ + +#include "ximawmf.h" + +#if CXIMAGE_SUPPORT_WMF && CXIMAGE_SUPPORT_WINDOWS + +///////////////////////////////////////////////////////////////////// +bool CxImageWMF::Decode(CxFile *hFile, long nForceWidth, long nForceHeight) +{ + if (hFile == NULL) return false; + + HENHMETAFILE hMeta; + HDC hDC; + int cx,cy; + + //save the current position of the file + long pos = hFile->Tell(); + + // Read the Metafile and convert to an Enhanced Metafile + METAFILEHEADER mfh; + hMeta = ConvertWmfFiletoEmf(hFile, &mfh); + if (hMeta) { // ok, it's a WMF + +///////////////////////////////////////////////////////////////////// +// We use the original WMF size information, because conversion to +// EMF adjusts the Metafile to Full Screen or does not set rclBounds at all +// ENHMETAHEADER emh; +// UINT uRet; +// uRet = GetEnhMetaFileHeader(hMeta, // handle of enhanced metafile +// sizeof(ENHMETAHEADER), // size of buffer, in bytes +// &emh); // address of buffer to receive data +// if (!uRet){ +// DeleteEnhMetaFile(hMeta); +// return false; +// } +// // calculate size +// cx = emh.rclBounds.right - emh.rclBounds.left; +// cy = emh.rclBounds.bottom - emh.rclBounds.top; +///////////////////////////////////////////////////////////////////// + + // calculate size + // scale the metafile (pixels/inch of metafile => pixels/inch of display) + // mfh.inch already checked to be <> 0 + + hDC = ::GetDC(0); + int cx1 = ::GetDeviceCaps(hDC, LOGPIXELSX); + int cy1 = ::GetDeviceCaps(hDC, LOGPIXELSY); + ::ReleaseDC(0, hDC); + + cx = (mfh.bbox.right - mfh.bbox.left) * cx1 / mfh.inch; + cy = (mfh.bbox.bottom - mfh.bbox.top) * cy1 / mfh.inch; + + } else { // maybe it's an EMF... + + hFile->Seek(pos,SEEK_SET); + + ENHMETAHEADER emh; + hMeta = ConvertEmfFiletoEmf(hFile, &emh); + + if (!hMeta){ + strcpy(info.szLastError,"corrupted WMF"); + return false; // definitively give up + } + // ok, it's an EMF + // calculate size + cx = emh.rclBounds.right - emh.rclBounds.left; + cy = emh.rclBounds.bottom - emh.rclBounds.top; + } + + if (info.nEscape) { // Check if cancelled + DeleteEnhMetaFile(hMeta); + strcpy(info.szLastError,"Cancelled"); + return false; + } + + if (!cx || !cy) { + DeleteEnhMetaFile(hMeta); + strcpy(info.szLastError,"empty WMF"); + return false; + } + + if (nForceWidth) cx=nForceWidth; + if (nForceHeight) cy=nForceHeight; + ShrinkMetafile(cx, cy); // !! Otherwise Bitmap may have bombastic size + + HDC hDC0 = GetDC(0); // DC of screen + HBITMAP hBitmap = CreateCompatibleBitmap(hDC0, cx, cy); // has # colors of display + hDC = CreateCompatibleDC(hDC0); // memory dc compatible with screen + ReleaseDC(0, hDC0); // don't need anymore. get rid of it. + + if (hDC){ + if (hBitmap){ + RECT rc = {0,0,cx,cy}; + int bpp = ::GetDeviceCaps(hDC, BITSPIXEL); + + HBITMAP hBitmapOld = (HBITMAP)SelectObject(hDC, hBitmap); + + // clear out the entire bitmap with windows background + // because the MetaFile may not contain background information + DWORD dwBack = XMF_COLOR_BACK; +#if XMF_SUPPORT_TRANSPARENCY + if (bpp == 24) dwBack = XMF_COLOR_TRANSPARENT; +#endif + DWORD OldColor = SetBkColor(hDC, dwBack); + ExtTextOut(hDC, 0, 0, ETO_OPAQUE, &rc, NULL, 0, NULL); + SetBkColor(hDC, OldColor); + + //retrieves optional palette entries from the specified enhanced metafile + PLOGPALETTE plogPal; + PBYTE pjTmp; + HPALETTE hPal; + int iEntries = GetEnhMetaFilePaletteEntries(hMeta, 0, NULL); + if (iEntries) { + if ((plogPal = (PLOGPALETTE)GlobalAlloc(GMEM_FIXED | GMEM_ZEROINIT, + sizeof(DWORD) + sizeof(PALETTEENTRY)*iEntries )) == NULL) { + DeleteObject(hBitmap); + DeleteDC(hDC); + DeleteEnhMetaFile(hMeta); + strcpy(info.szLastError,"Cancelled"); + return false; + } + + plogPal->palVersion = 0x300; + plogPal->palNumEntries = (WORD) iEntries; + pjTmp = (PBYTE) plogPal; + pjTmp += 4; + + GetEnhMetaFilePaletteEntries(hMeta, iEntries, (PPALETTEENTRY)pjTmp); + hPal = CreatePalette(plogPal); + GlobalFree(plogPal); + + SelectPalette(hDC, hPal, FALSE); + RealizePalette(hDC); + } + + // Play the Metafile into Memory DC + BOOL bRet = PlayEnhMetaFile(hDC, // handle to a device context + hMeta, // handle to an enhanced metafile + &rc); // pointer to bounding rectangle + + SelectObject(hDC, hBitmapOld); + DeleteEnhMetaFile(hMeta); // we are done with this one + + if (info.nEscape) { // Check if cancelled + DeleteObject(hBitmap); + DeleteDC(hDC); + strcpy(info.szLastError,"Cancelled"); + return false; + } + + // the Bitmap now has the image. + // Create our DIB and convert the DDB into DIB + if (!Create(cx, cy, bpp, CXIMAGE_FORMAT_WMF)) { + DeleteObject(hBitmap); + DeleteDC(hDC); + return false; + } + +#if XMF_SUPPORT_TRANSPARENCY + if (bpp == 24) { + RGBQUAD rgbTrans = { XMF_RGBQUAD_TRANSPARENT }; + SetTransColor(rgbTrans); + } +#endif + // We're finally ready to get the DIB. Call the driver and let + // it party on our bitmap. It will fill in the color table, + // and bitmap bits of our global memory block. + bRet = GetDIBits(hDC, hBitmap, 0, + (UINT)cy, GetBits(), (LPBITMAPINFO)pDib, DIB_RGB_COLORS); + + DeleteObject(hBitmap); + DeleteDC(hDC); + + return (bRet!=0); + } else { + DeleteDC(hDC); + } + } else { + if (hBitmap) DeleteObject(hBitmap); + } + + DeleteEnhMetaFile(hMeta); + + return false; +} + +/********************************************************************** + Function: CheckMetafileHeader + Purpose: Check if the Metafileheader of a file is valid +**********************************************************************/ +BOOL CxImageWMF::CheckMetafileHeader(METAFILEHEADER *metafileheader) +{ + WORD *pw; + WORD cs; + int i; + + // check magic # + if (metafileheader->key != 0x9ac6cdd7L) return false; + + // test checksum of header + pw = (WORD *)metafileheader; + cs = *pw; + pw++; + for (i = 0; i < 9; i++) { + cs ^= *pw; + pw++; + } + + if (cs != metafileheader->checksum) return false; + + // check resolution + if ((metafileheader->inch <= 0) || (metafileheader->inch > 2540)) return false; + + return true; +} + +/********************************************************************** + Function: ConvertWmfFiletoEmf + Purpose: Converts a Windows Metafile into an Enhanced Metafile +**********************************************************************/ +HENHMETAFILE CxImageWMF::ConvertWmfFiletoEmf(CxFile *fp, METAFILEHEADER *metafileheader) +{ + HENHMETAFILE hMeta; + long lenFile; + long len; + BYTE *p; + METAHEADER mfHeader; + DWORD seekpos; + + hMeta = 0; + + // get length of the file + lenFile = fp->Size(); + + // a placeable metafile starts with a METAFILEHEADER + // read it and check metafileheader + len = fp->Read(metafileheader, 1, sizeof(METAFILEHEADER)); + if (len < sizeof(METAFILEHEADER)) return (hMeta); + + if (CheckMetafileHeader(metafileheader)) { + // This is a placeable metafile + // Convert the placeable format into something that can + // be used with GDI metafile functions + seekpos = sizeof(METAFILEHEADER); + } else { + // Not a placeable wmf. A windows metafile? + // at least not scaleable. + // we could try to convert, but would loose ratio. don't allow this + return (hMeta); + + //metafileheader->bbox.right = ?; + //metafileheader->bbox.left = ?; + //metafileheader->bbox.bottom = ?; + //metafileheader->bbox.top = ?; + //metafileheader->inch = ?; + // + //seekpos = 0; + // fp->Seek(0, SEEK_SET); // rewind + } + + // At this point we have a metaheader regardless of whether + // the metafile was a windows metafile or a placeable metafile + // so check to see if it is valid. There is really no good + // way to do this so just make sure that the mtType is either + // 1 or 2 (memory or disk file) + // in addition we compare the length of the METAHEADER against + // the length of the file. if filelength < len => no Metafile + + len = fp->Read(&mfHeader, 1, sizeof(METAHEADER)); + if (len < sizeof(METAHEADER)) return (hMeta); + + if ((mfHeader.mtType != 1) && (mfHeader.mtType != 2)) return (hMeta); + + // Length in Bytes from METAHEADER + len = mfHeader.mtSize * 2; + if (len > lenFile) return (hMeta); + + // Allocate memory for the metafile bits + p = (BYTE *)malloc(len); + if (!p) return (hMeta); + + // seek back to METAHEADER and read all the stuff at once + fp->Seek(seekpos, SEEK_SET); + lenFile = fp->Read(p, 1, len); + if (lenFile != len) { + free(p); + return (hMeta); + } + + // the following (commented code) works, but adjusts rclBound of the + // Enhanced Metafile to full screen. + // the METAFILEHEADER from above is needed to scale the image + +// hMeta = SetWinMetaFileBits(len, p, NULL, NULL); + + // scale the metafile (pixels/inch of metafile => pixels/inch of display) + + METAFILEPICT mfp; + int cx1, cy1; + HDC hDC; + + hDC = ::GetDC(0); + cx1 = ::GetDeviceCaps(hDC, LOGPIXELSX); + cy1 = ::GetDeviceCaps(hDC, LOGPIXELSY); + + memset(&mfp, 0, sizeof(mfp)); + + mfp.mm = MM_ANISOTROPIC; + mfp.xExt = (metafileheader->bbox.right - metafileheader->bbox.left) * cx1 / metafileheader->inch; + mfp.yExt = (metafileheader->bbox.bottom - metafileheader->bbox.top) * cy1 / metafileheader->inch; + mfp.hMF = 0; + + // in MM_ANISOTROPIC mode xExt and yExt are in MM_HIENGLISH + // MM_HIENGLISH means: Each logical unit is converted to 0.001 inch + //mfp.xExt *= 1000; + //mfp.yExt *= 1000; + // ???? + //int k = 332800 / ::GetSystemMetrics(SM_CXSCREEN); + //mfp.xExt *= k; mfp.yExt *= k; + + // fix for Win9x + while ((mfp.xExt < 6554) && (mfp.yExt < 6554)) + { + mfp.xExt *= 10; + mfp.yExt *= 10; + } + + hMeta = SetWinMetaFileBits(len, p, hDC, &mfp); + + if (!hMeta){ //try 2nd conversion using a different mapping + mfp.mm = MM_TEXT; + hMeta = SetWinMetaFileBits(len, p, hDC, &mfp); + } + + ::ReleaseDC(0, hDC); + + // Free Memory + free(p); + + return (hMeta); +} +///////////////////////////////////////////////////////////////////// +HENHMETAFILE CxImageWMF::ConvertEmfFiletoEmf(CxFile *pFile, ENHMETAHEADER *pemfh) +{ + HENHMETAFILE hMeta; + long iLen = pFile->Size(); + + BYTE* pBuff = (BYTE *)malloc(iLen); + if (!pBuff) return (FALSE); + + // Read the Enhanced Metafile + long iLenRead = pFile->Read(pBuff, 1, iLen); + if (iLenRead != iLen) { + free(pBuff); + return (FALSE); + } + + // Make it a Memory Metafile + hMeta = SetEnhMetaFileBits(iLen, pBuff); + + free(pBuff); // finished with this one + + if (!hMeta) return (FALSE); // oops. + + // Get the Enhanced Metafile Header + UINT uRet = GetEnhMetaFileHeader(hMeta, // handle of enhanced metafile + sizeof(ENHMETAHEADER), // size of buffer, in bytes + pemfh); // address of buffer to receive data + + if (!uRet) { + DeleteEnhMetaFile(hMeta); + return (FALSE); + } + + return (hMeta); +} + + +///////////////////////////////////////////////////////////////////// +#if CXIMAGE_SUPPORT_ENCODE +///////////////////////////////////////////////////////////////////// +bool CxImageWMF::Encode(CxFile * hFile) +{ + if (hFile == NULL) return false; + strcpy(info.szLastError, "Save WMF not supported"); + return false; +} +#endif // CXIMAGE_SUPPORT_ENCODE +///////////////////////////////////////////////////////////////////// + +/********************************************************************** +Function: ShrinkMetafile +Purpose: Shrink the size of a metafile to be not larger than + the definition +**********************************************************************/ +void CxImageWMF::ShrinkMetafile(int &cx, int &cy) +{ + int xScreen = XMF_MAXSIZE_CX; + int yScreen = XMF_MAXSIZE_CY; + + if (cx > xScreen){ + cy = cy * xScreen / cx; + cx = xScreen; + } + + if (cy > yScreen){ + cx = cx * yScreen / cy; + cy = yScreen; + } +} + +#endif // CIMAGE_SUPPORT_WMF + diff --git a/src/win32/dependencies/cximage/ximawmf.h b/src/win32/dependencies/cximage/ximawmf.h new file mode 100644 index 00000000..75e119a0 --- /dev/null +++ b/src/win32/dependencies/cximage/ximawmf.h @@ -0,0 +1,154 @@ +/* +********************************************************************* + * File: ximawmf.h + * Purpose: Windows Metafile Class Loader and Writer + * Author: Volker Horch - vhorch@gmx.de + * created: 13-Jun-2002 +********************************************************************* + */ + +/* +********************************************************************* + Notes by Author: +********************************************************************* + + Limitations: + ============ + + a) Transparency: + + A Metafile is vector graphics, which has transparency by design. + This class always converts into a Bitmap format. Transparency is + supported, but there is no good way to find out, which parts + of the Metafile are transparent. There are two ways how we can + handle this: + + - Clear the Background of the Bitmap with the background color + you like (i have used COLOR_WINDOW) and don't support transparency. + + below #define XMF_SUPPORT_TRANSPARENCY 0 + #define XMF_COLOR_BACK RGB(Background color you like) + + - Clear the Background of the Bitmap with a very unusual color + (which one ?) and use this color as the transparent color + + below #define XMF_SUPPORT_TRANSPARENCY 1 + #define XMF_COLOR_TRANSPARENT_R ... + #define XMF_COLOR_TRANSPARENT_G ... + #define XMF_COLOR_TRANSPARENT_B ... + + b) Resolution + + Once we have converted the Metafile into a Bitmap and we zoom in + or out, the image may not look very good. If we still had the + original Metafile, zooming would produce good results always. + + c) Size + + Although the filesize of a Metafile may be very small, it might + produce a Bitmap with a bombastic size. Assume you have a Metafile + with an image size of 6000*4000, which contains just one Metafile + record ((e.g. a line from (0,0) to (6000, 4000)). The filesize + of this Metafile would be let's say 100kB. If we convert it to + a 6000*4000 Bitmap with 24 Bits/Pixes, the Bitmap would consume + about 68MB of memory. + + I have choosen, to limit the size of the Bitmap to max. + screensize, to avoid memory problems. + + If you want something else, + modify #define XMF_MAXSIZE_CX / XMF_MAXSIZE_CY below + +********************************************************************* +*/ + +#ifndef _XIMAWMF_H +#define _XIMAWMF_H + +#include "ximage.h" + +#if CXIMAGE_SUPPORT_WMF && CXIMAGE_SUPPORT_WINDOWS + +class CxImageWMF: public CxImage +{ + +#pragma pack(1) + +typedef struct tagRECT16 +{ + short int left; + short int top; + short int right; + short int bottom; +} RECT16; + +// taken from Windos 3.11 SDK Documentation (Programmer's Reference Volume 4: Resources) +typedef struct tagMETAFILEHEADER +{ + DWORD key; // always 0x9ac6cdd7 + WORD reserved1; // reserved = 0 + RECT16 bbox; // bounding rectangle in metafile units as defined in "inch" + WORD inch; // number of metafile units per inch (should be < 1440) + DWORD reserved2; // reserved = 0 + WORD checksum; // sum of the first 10 WORDS (using XOR operator) +} METAFILEHEADER; + +#pragma pack() + +public: + CxImageWMF(): CxImage(CXIMAGE_FORMAT_WMF) { } + + bool Decode(CxFile * hFile, long nForceWidth=0, long nForceHeight=0); + bool Decode(FILE *hFile, long nForceWidth=0, long nForceHeight=0) + { CxIOFile file(hFile); return Decode(&file,nForceWidth,nForceHeight); } + +#if CXIMAGE_SUPPORT_ENCODE + bool Encode(CxFile * hFile); + bool Encode(FILE *hFile) { CxIOFile file(hFile); return Encode(&file); } +#endif // CXIMAGE_SUPPORT_ENCODE + +protected: + void ShrinkMetafile(int &cx, int &cy); + BOOL CheckMetafileHeader(METAFILEHEADER *pmetafileheader); + HENHMETAFILE ConvertWmfFiletoEmf(CxFile *pFile, METAFILEHEADER *pmetafileheader); + HENHMETAFILE ConvertEmfFiletoEmf(CxFile *pFile, ENHMETAHEADER *pemfh); + +}; + +#define METAFILEKEY 0x9ac6cdd7L + +// Background color definition (if no transparency). see Notes above +#define XMF_COLOR_BACK GetSysColor(COLOR_WINDOW) +// alternatives +//#define XMF_COLOR_BACK RGB(192, 192, 192) // lite gray +//#define XMF_COLOR_BACK RGB( 0, 0, 0) // black +//#define XMF_COLOR_BACK RGB(255, 255, 255) // white + + +// transparency support. see Notes above +#define XMF_SUPPORT_TRANSPARENCY 0 +#define XMF_COLOR_TRANSPARENT_R 211 +#define XMF_COLOR_TRANSPARENT_G 121 +#define XMF_COLOR_TRANSPARENT_B 112 +// don't change +#define XMF_COLOR_TRANSPARENT RGB (XMF_COLOR_TRANSPARENT_R, \ + XMF_COLOR_TRANSPARENT_G, \ + XMF_COLOR_TRANSPARENT_B) +// don't change +#define XMF_RGBQUAD_TRANSPARENT XMF_COLOR_TRANSPARENT_B, \ + XMF_COLOR_TRANSPARENT_G, \ + XMF_COLOR_TRANSPARENT_R, \ + 0 +// max. size. see Notes above +// alternatives +//#define XMF_MAXSIZE_CX (GetSystemMetrics(SM_CXSCREEN)-10) +//#define XMF_MAXSIZE_CY (GetSystemMetrics(SM_CYSCREEN)-50) +//#define XMF_MAXSIZE_CX (2*GetSystemMetrics(SM_CXSCREEN)/3) +//#define XMF_MAXSIZE_CY (2*GetSystemMetrics(SM_CYSCREEN)/3) +#define XMF_MAXSIZE_CX 4000 +#define XMF_MAXSIZE_CY 4000 + + +#endif + +#endif diff --git a/src/win32/dependencies/cximage/ximawnd.cpp b/src/win32/dependencies/cximage/ximawnd.cpp new file mode 100644 index 00000000..79d4158f --- /dev/null +++ b/src/win32/dependencies/cximage/ximawnd.cpp @@ -0,0 +1,1110 @@ +// xImaWnd.cpp : Windows functions +/* 07/08/2001 v1.00 - Davide Pizzolato - www.xdp.it + * CxImage version 5.99c 17/Oct/2004 + */ + +#include "ximage.h" + +#include "ximaiter.h" +#include "ximabmp.h" + +//////////////////////////////////////////////////////////////////////////////// +#if CXIMAGE_SUPPORT_WINCE +//////////////////////////////////////////////////////////////////////////////// +long CxImage::Blt(HDC pDC, long x, long y) +{ + if((pDib==0)||(pDC==0)||(!info.bEnabled)) return 0; + + HBRUSH brImage = CreateDIBPatternBrushPt(pDib, DIB_RGB_COLORS); + HBRUSH brOld = (HBRUSH) SelectObject(pDC, brImage); + PatBlt(pDC, 0, 0, head.biWidth, head.biHeight, PATCOPY); + SelectObject(pDC, brOld); + DeleteObject(brImage); + return 1; +} +#endif //CXIMAGE_SUPPORT_WINCE +//////////////////////////////////////////////////////////////////////////////// + +//////////////////////////////////////////////////////////////////////////////// +#if CXIMAGE_SUPPORT_WINDOWS +//////////////////////////////////////////////////////////////////////////////// +/** + * Transfer the image in a global bitmap handle (clipboard copy) + */ +HANDLE CxImage::CopyToHandle() +{ + HANDLE hMem=NULL; + if (pDib){ + hMem= GlobalAlloc(GHND, GetSize()); + if (hMem){ + BYTE* pDst=(BYTE*)GlobalLock(hMem); + if (pDst){ + memcpy(pDst,pDib,GetSize()); + } + GlobalUnlock(hMem); + } + } + return hMem; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Global object (clipboard paste) constructor + * \param hMem: source bitmap object, the clipboard format must be CF_DIB + * \return true if everything is ok + */ +bool CxImage::CreateFromHANDLE(HANDLE hMem) +{ + if (!Destroy()) + return false; + + DWORD dwSize = GlobalSize(hMem); + if (!dwSize) return false; + + BYTE *lpVoid; //pointer to the bitmap + lpVoid = (BYTE *)GlobalLock(hMem); + BITMAPINFOHEADER *pHead; //pointer to the bitmap header + pHead = (BITMAPINFOHEADER *)lpVoid; + if (lpVoid){ + + //CxMemFile hFile(lpVoid,dwSize); + + //copy the bitmap header + memcpy(&head,pHead,sizeof(BITMAPINFOHEADER)); + //create the image + if(!Create(head.biWidth,head.biHeight,head.biBitCount)){ + GlobalUnlock(lpVoid); + return false; + } + //preserve DPI + if (head.biXPelsPerMeter) SetXDPI((long)floor(head.biXPelsPerMeter * 254.0 / 10000.0 + 0.5)); else SetXDPI(96); + if (head.biYPelsPerMeter) SetYDPI((long)floor(head.biYPelsPerMeter * 254.0 / 10000.0 + 0.5)); else SetYDPI(96); + + /*//copy the pixels (old way) + if((pHead->biCompression != BI_RGB) || (pHead->biBitCount == 32)){ // + // BITFIELD case + // set the internal header in the dib + memcpy(pDib,&head,sizeof(head)); + // get the bitfield masks + DWORD bf[3]; + memcpy(bf,lpVoid+pHead->biSize,12); + // transform into RGB + Bitfield2RGB(lpVoid+pHead->biSize+12,(WORD)bf[0],(WORD)bf[1],(WORD)bf[2],(BYTE)pHead->biBitCount); + } else { //normal bitmap + memcpy(pDib,lpVoid,GetSize()); + }*/ + + // + // fill in color map + bool bIsOldBmp = (head.biSize == sizeof(BITMAPCOREHEADER)); + RGBQUAD *pRgb = GetPalette(); + if (pRgb) { + // number of colors to fill in + int nColors = DibNumColors(pHead); + if (bIsOldBmp) { + /* get pointer to BITMAPCOREINFO (old style 1.x) */ + LPBITMAPCOREINFO lpbmc = (LPBITMAPCOREINFO)lpVoid; + for (int i = nColors - 1; i >= 0; i--) { + pRgb[i].rgbRed = lpbmc->bmciColors[i].rgbtRed; + pRgb[i].rgbGreen = lpbmc->bmciColors[i].rgbtGreen; + pRgb[i].rgbBlue = lpbmc->bmciColors[i].rgbtBlue; + pRgb[i].rgbReserved = (BYTE)0; + } + } else { + /* get pointer to BITMAPINFO (new style 3.x) */ + LPBITMAPINFO lpbmi = (LPBITMAPINFO)lpVoid; + for (int i = nColors - 1; i >= 0; i--) { + pRgb[i].rgbRed = lpbmi->bmiColors[i].rgbRed; + pRgb[i].rgbGreen = lpbmi->bmiColors[i].rgbGreen; + pRgb[i].rgbBlue = lpbmi->bmiColors[i].rgbBlue; + pRgb[i].rgbReserved = (BYTE)0; + } + } + } + + // + DWORD dwCompression = pHead->biCompression; + // compressed bitmap ? + if(dwCompression!=BI_RGB || pHead->biBitCount==32) { + // get the bitmap bits + LPSTR lpDIBBits = (LPSTR)((BYTE*)pHead + *(DWORD*)pHead + (WORD)(GetNumColors() * sizeof(RGBQUAD))); + // decode and copy them to our image + switch (pHead->biBitCount) { + case 32 : + { + // BITFIELD case + if (dwCompression == BI_BITFIELDS || dwCompression == BI_RGB) { + // get the bitfield masks + DWORD bf[3]; + memcpy(bf,lpVoid+pHead->biSize,12); + // transform into RGB + Bitfield2RGB(lpVoid+pHead->biSize+12,(WORD)bf[0],(WORD)bf[1],(WORD)bf[2],(BYTE)pHead->biBitCount); + } else + throw "unknown compression"; + } + break; + case 16 : + { + // get the bitfield masks + long offset=0; + DWORD bf[3]; + if (dwCompression == BI_BITFIELDS) { + memcpy(bf,lpVoid+pHead->biSize,12); + offset= 12; + } else { + bf[0] = 0x7C00; + bf[1] = 0x3E0; + bf[2] = 0x1F; // RGB555 + } + // copy the pixels + memcpy(info.pImage, lpDIBBits + offset, head.biHeight*((head.biWidth+1)/2)*4); + // transform into RGB + Bitfield2RGB(info.pImage, (WORD)bf[0], (WORD)bf[1], (WORD)bf[2], 16); + } + break; + case 8 : + case 4 : + case 1 : + { + switch (dwCompression) { + case BI_RLE4: + { + BYTE status_byte = 0; + BYTE second_byte = 0; + int scanline = 0; + int bits = 0; + BOOL low_nibble = FALSE; + CImageIterator iter(this); + + for (BOOL bContinue = TRUE; bContinue; ) { + status_byte = *(lpDIBBits++); + switch (status_byte) { + case RLE_COMMAND : + status_byte = *(lpDIBBits++); + switch (status_byte) { + case RLE_ENDOFLINE : + bits = 0; + scanline++; + low_nibble = FALSE; + break; + case RLE_ENDOFBITMAP : + bContinue = FALSE; + break; + case RLE_DELTA : + { + // read the delta values + BYTE delta_x; + BYTE delta_y; + delta_x = *(lpDIBBits++); + delta_y = *(lpDIBBits++); + // apply them + bits += delta_x / 2; + scanline += delta_y; + break; + } + default : + second_byte = *(lpDIBBits++); + BYTE* sline = iter.GetRow(scanline); + for (int i = 0; i < status_byte; i++) { + if (low_nibble) { + if ((DWORD)(sline + bits) < (DWORD)(info.pImage + head.biSizeImage)) { + *(sline + bits) |= second_byte & 0x0F; + } + if (i != status_byte - 1) + second_byte = *(lpDIBBits++); + bits++; + } else { + if ((DWORD)(sline + bits) <(DWORD)(info.pImage + head.biSizeImage)) { + *(sline + bits) = (BYTE)(second_byte & 0xF0); + } + } + low_nibble = !low_nibble; + } + if ((((status_byte+1) >> 1) & 1 )== 1) + second_byte = *(lpDIBBits++); + break; + }; + break; + default : + { + BYTE* sline = iter.GetRow(scanline); + second_byte = *(lpDIBBits++); + for (unsigned i = 0; i < status_byte; i++) { + if (low_nibble) { + if ((DWORD)(sline + bits) <(DWORD)(info.pImage + head.biSizeImage)) { + *(sline + bits) |= second_byte & 0x0F; + } + bits++; + } else { + if ((DWORD)(sline + bits) <(DWORD)(info.pImage + head.biSizeImage)) { + *(sline + bits) = (BYTE)(second_byte & 0xF0); + } + } + low_nibble = !low_nibble; + } + } + break; + }; + } + } + break; + case BI_RLE8 : + { + BYTE status_byte = 0; + BYTE second_byte = 0; + int scanline = 0; + int bits = 0; + CImageIterator iter(this); + + for (BOOL bContinue = TRUE; bContinue; ) { + status_byte = *(lpDIBBits++); + if (status_byte==RLE_COMMAND) { + status_byte = *(lpDIBBits++); + switch (status_byte) { + case RLE_ENDOFLINE : + bits = 0; + scanline++; + break; + case RLE_ENDOFBITMAP : + bContinue = FALSE; + break; + case RLE_DELTA : + { + // read the delta values + BYTE delta_x; + BYTE delta_y; + delta_x = *(lpDIBBits++); + delta_y = *(lpDIBBits++); + // apply them + bits += delta_x; + scanline += delta_y; + } + break; + default : + int nNumBytes = sizeof(BYTE) * status_byte; + memcpy((void *)(iter.GetRow(scanline) + bits), lpDIBBits, nNumBytes); + lpDIBBits += nNumBytes; + // align run length to even number of bytes + if ((status_byte & 1) == 1) + second_byte = *(lpDIBBits++); + bits += status_byte; + break; + }; + } else { + BYTE *sline = iter.GetRow(scanline); + second_byte = *(lpDIBBits++); + for (unsigned i = 0; i < status_byte; i++) { + if ((DWORD)bits + // // Create a device-independent bitmap + // return CreateBitmap(head.biWidth,head.biHeight, 1, head.biBitCount, GetBits()); + // use instead this code + HDC hMemDC = CreateCompatibleDC(NULL); + LPVOID pBit32; + HBITMAP bmp = CreateDIBSection(hMemDC,(LPBITMAPINFO)pDib,DIB_RGB_COLORS, &pBit32, NULL, 0); + if (pBit32) memcpy(pBit32, GetBits(), head.biSizeImage); + DeleteDC(hMemDC); + return bmp; + } + + // this single line seems to work very well + HBITMAP bmp = CreateDIBitmap(hdc, (LPBITMAPINFOHEADER)pDib, CBM_INIT, + GetBits(), (LPBITMAPINFO)pDib, DIB_RGB_COLORS); + + return bmp; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Bitmap resource constructor + * \param hbmp : bitmap resource handle + * \param hpal : (optional) palette, useful for 8bpp DC + * \return true if everything is ok + */ +bool CxImage::CreateFromHBITMAP(HBITMAP hbmp, HPALETTE hpal) +{ + if (!Destroy()) + return false; + + if (hbmp) { + BITMAP bm; + // get informations about the bitmap + GetObject(hbmp, sizeof(BITMAP), (LPSTR) &bm); + // create the image + if (!Create(bm.bmWidth, bm.bmHeight, bm.bmBitsPixel, 0)) + return false; + // create a device context for the bitmap + HDC dc = ::GetDC(NULL); + if (!dc) + return false; + + if (hpal){ + SelectObject(dc,hpal); //the palette you should get from the user or have a stock one + RealizePalette(dc); + } + + // copy the pixels + if (GetDIBits(dc, hbmp, 0, head.biHeight, info.pImage, + (LPBITMAPINFO)pDib, DIB_RGB_COLORS) == 0){ //replace &head with pDib + strcpy(info.szLastError,"GetDIBits failed"); + ::ReleaseDC(NULL, dc); + return false; + } + ::ReleaseDC(NULL, dc); + return true; + } + return false; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * icon resource constructor + * \param hico : icon resource handle + * \return true if everything is ok + */ +bool CxImage::CreateFromHICON(HICON hico) +{ + if (!Destroy()) + return false; + + if (hico) { + ICONINFO iinfo; + GetIconInfo(hico,&iinfo); + if (!CreateFromHBITMAP(iinfo.hbmColor)) + return false; +#if CXIMAGE_SUPPORT_ALPHA + CxImage mask; + mask.CreateFromHBITMAP(iinfo.hbmMask); + mask.GrayScale(); + mask.Negative(); + AlphaSet(mask); +#endif + DeleteObject(iinfo.hbmColor); // + DeleteObject(iinfo.hbmMask); // + return true; + } + return false; +} +//////////////////////////////////////////////////////////////////////////////// +long CxImage::Draw(HDC hdc, const RECT& rect, RECT* pClipRect, bool bSmooth) +{ + return Draw(hdc, rect.left, rect.top, rect.right - rect.left, rect.bottom - rect.top, pClipRect,bSmooth); +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Draws the image in the specified device context, with support for alpha channel, alpha palette, transparency, opacity. + * \param hdc : destination device context + * \param x,y : (optional) offset + * \param cx,cy : (optional) size. + * - If cx or cy are not specified (or less than 0), the normal width or height will be used + * - If cx or cy are different than width or height, the image will be stretched + * + * \param pClipRect : limit the drawing operations inside a given rectangle in the output device context. + * \param bSmooth : activates a bilinear filter that will enhance the appearence for zommed pictures. + * Quite slow. Needs CXIMAGE_SUPPORT_INTERPOLATION. + * \return true if everything is ok + */ +long CxImage::Draw(HDC hdc, long x, long y, long cx, long cy, RECT* pClipRect, bool bSmooth) +{ + if((pDib==0)||(hdc==0)||(cx==0)||(cy==0)||(!info.bEnabled)) return 0; + + if (cx < 0) cx = head.biWidth; + if (cy < 0) cy = head.biHeight; + bool bTransparent = info.nBkgndIndex != -1; + bool bAlpha = pAlpha != 0; + + RECT mainbox; // (experimental) + if (pClipRect){ + GetClipBox(hdc,&mainbox); + HRGN rgn = CreateRectRgnIndirect(pClipRect); + ExtSelectClipRgn(hdc,rgn,RGN_AND); + DeleteObject(rgn); + } + + //find the smallest area to paint + RECT clipbox,paintbox; + GetClipBox(hdc,&clipbox); + + paintbox.top = min(clipbox.bottom,max(clipbox.top,y)); + paintbox.left = min(clipbox.right,max(clipbox.left,x)); + paintbox.right = max(clipbox.left,min(clipbox.right,x+cx)); + paintbox.bottom = max(clipbox.top,min(clipbox.bottom,y+cy)); + + long destw = paintbox.right - paintbox.left; + long desth = paintbox.bottom - paintbox.top; + + if (!(bTransparent || bAlpha || info.bAlphaPaletteEnabled)){ + if (cx==head.biWidth && cy==head.biHeight){ //NORMAL + SetStretchBltMode(hdc,COLORONCOLOR); + SetDIBitsToDevice(hdc, x, y, cx, cy, 0, 0, 0, cy, + info.pImage,(BITMAPINFO*)pDib,DIB_RGB_COLORS); + } else { //STRETCH + //pixel informations + RGBQUAD c={0,0,0,0}; + //Preparing Bitmap Info + BITMAPINFO bmInfo; + memset(&bmInfo.bmiHeader,0,sizeof(BITMAPINFOHEADER)); + bmInfo.bmiHeader.biSize=sizeof(BITMAPINFOHEADER); + bmInfo.bmiHeader.biWidth=destw; + bmInfo.bmiHeader.biHeight=desth; + bmInfo.bmiHeader.biPlanes=1; + bmInfo.bmiHeader.biBitCount=24; + BYTE *pbase; //points to the final dib + BYTE *pdst; //current pixel from pbase + BYTE *ppix; //current pixel from image + //get the background + HDC TmpDC=CreateCompatibleDC(hdc); + HBITMAP TmpBmp=CreateDIBSection(hdc,&bmInfo,DIB_RGB_COLORS,(void**)&pbase,0,0); + HGDIOBJ TmpObj=SelectObject(TmpDC,TmpBmp); + + if (pbase){ + long xx,yy; + long sx,sy; + float dx,dy; + BYTE *psrc; + + long ew = ((((24 * destw) + 31) / 32) * 4); + long ymax = paintbox.bottom; + long xmin = paintbox.left; + float fx=(float)head.biWidth/(float)cx; + float fy=(float)head.biHeight/(float)cy; + + for(yy=0;yy>8); + + if (head.biClrUsed){ + ci = GetPixelIndex(sx,sy); +#if CXIMAGE_SUPPORT_INTERPOLATION + if (bSmooth){ + c = GetPixelColorInterpolated(dx - 0.5f, dy - 0.5f, CxImage::IM_BILINEAR, CxImage::OM_REPEAT); + } else +#endif //CXIMAGE_SUPPORT_INTERPOLATION + { + c = GetPaletteColor(GetPixelIndex(sx,sy)); + } + if (info.bAlphaPaletteEnabled){ + a = (BYTE)((a*(1+c.rgbReserved))>>8); + } + } else { +#if CXIMAGE_SUPPORT_INTERPOLATION + if (bSmooth){ + c = GetPixelColorInterpolated(dx - 0.5f, dy - 0.5f, CxImage::IM_BILINEAR, CxImage::OM_REPEAT); + } else +#endif //CXIMAGE_SUPPORT_INTERPOLATION + { + ppix = psrc + sx*3; + c.rgbBlue = *ppix++; + c.rgbGreen= *ppix++; + c.rgbRed = *ppix; + } + } + //if (*pc!=*pct || !bTransparent){ + //if ((head.biClrUsed && ci!=cit) || ((!head.biClrUsed||bSmooth) && *pc!=*pct) || !bTransparent){ + if ((head.biClrUsed && ci!=cit) || (!head.biClrUsed && *pc!=*pct) || !bTransparent){ + // DJT, assume many pixels are fully transparent or opaque and thus avoid multiplication + if (a == 0) { // Transparent, retain dest + pdst+=3; + } else if (a == 255) { // opaque, ignore dest + *pdst++= c.rgbBlue; + *pdst++= c.rgbGreen; + *pdst++= c.rgbRed; + } else { // semi transparent + a1=(BYTE)~a; + *pdst++=(BYTE)((*pdst * a1 + a * c.rgbBlue)>>8); + *pdst++=(BYTE)((*pdst * a1 + a * c.rgbGreen)>>8); + *pdst++=(BYTE)((*pdst * a1 + a * c.rgbRed)>>8); + } + } else { + pdst+=3; + } + } + } + } else { + //NORMAL + iy=head.biHeight-ymax+y; + for(yy=0;yy>8); + + if (head.biClrUsed){ + ci = GetPixelIndex(ix,iy); + c = GetPaletteColor((BYTE)ci); + if (info.bAlphaPaletteEnabled){ + a = (BYTE)((a*(1+c.rgbReserved))>>8); + } + } else { + c.rgbBlue = *ppix++; + c.rgbGreen= *ppix++; + c.rgbRed = *ppix++; + } + + //if (*pc!=*pct || !bTransparent){ + if ((head.biClrUsed && ci!=cit) || (!head.biClrUsed && *pc!=*pct) || !bTransparent){ + // DJT, assume many pixels are fully transparent or opaque and thus avoid multiplication + if (a == 0) { // Transparent, retain dest + pdst+=3; + } else if (a == 255) { // opaque, ignore dest + *pdst++= c.rgbBlue; + *pdst++= c.rgbGreen; + *pdst++= c.rgbRed; + } else { // semi transparent + a1=(BYTE)~a; + *pdst++=(BYTE)((*pdst * a1 + a * c.rgbBlue)>>8); + *pdst++=(BYTE)((*pdst * a1 + a * c.rgbGreen)>>8); + *pdst++=(BYTE)((*pdst * a1 + a * c.rgbRed)>>8); + } + } else { + pdst+=3; + } + } + } + } + } + //paint the image & cleanup + SetDIBitsToDevice(hdc,paintbox.left,paintbox.top,destw,desth,0,0,0,desth,pbase,&bmInfo,0); + DeleteObject(SelectObject(TmpDC,TmpObj)); + DeleteDC(TmpDC); + } + + if (pClipRect){ // (experimental) + HRGN rgn = CreateRectRgnIndirect(&mainbox); + ExtSelectClipRgn(hdc,rgn,RGN_OR); + DeleteObject(rgn); + } + + return 1; +} +//////////////////////////////////////////////////////////////////////////////// +long CxImage::Draw2(HDC hdc, const RECT& rect) +{ + return Draw2(hdc, rect.left, rect.top, rect.right - rect.left, rect.bottom - rect.top); +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Draws (stretch) the image with single transparency support + * \param hdc : destination device context + * \param x,y : (optional) offset + * \param cx,cy : (optional) size. + * - If cx or cy are not specified (or less than 0), the normal width or height will be used + * - If cx or cy are different than width or height, the image will be stretched + * + * \return true if everything is ok + */ +long CxImage::Draw2(HDC hdc, long x, long y, long cx, long cy) +{ + if((pDib==0)||(hdc==0)||(cx==0)||(cy==0)||(!info.bEnabled)) return 0; + if (cx < 0) cx = head.biWidth; + if (cy < 0) cy = head.biHeight; + bool bTransparent = (info.nBkgndIndex != -1); + + if (!bTransparent){ + SetStretchBltMode(hdc,COLORONCOLOR); + StretchDIBits(hdc, x, y, cx, cy, 0, 0, head.biWidth, head.biHeight, + info.pImage,(BITMAPINFO*)pDib, DIB_RGB_COLORS,SRCCOPY); + } else { + // draw image with transparent background + const int safe = 0; // or else GDI fails in the following - sometimes + RECT rcDst = {x+safe, y+safe, x+cx, y+cy}; + if (RectVisible(hdc, &rcDst)){ + ///////////////////////////////////////////////////////////////// + // True Mask Method - Thanks to Paul Reynolds and Ron Gery + int nWidth = head.biWidth; + int nHeight = head.biHeight; + // Create two memory dcs for the image and the mask + HDC dcImage=CreateCompatibleDC(hdc); + HDC dcTrans=CreateCompatibleDC(hdc); + // Select the image into the appropriate dc + HBITMAP bm = CreateCompatibleBitmap(hdc, nWidth, nHeight); + HBITMAP pOldBitmapImage = (HBITMAP)SelectObject(dcImage,bm); + SetStretchBltMode(dcImage,COLORONCOLOR); + StretchDIBits(dcImage, 0, 0, nWidth, nHeight, 0, 0, nWidth, nHeight, + info.pImage,(BITMAPINFO*)pDib,DIB_RGB_COLORS,SRCCOPY); + + // Create the mask bitmap + HBITMAP bitmapTrans = CreateBitmap(nWidth, nHeight, 1, 1, NULL); + // Select the mask bitmap into the appropriate dc + HBITMAP pOldBitmapTrans = (HBITMAP)SelectObject(dcTrans, bitmapTrans); + // Build mask based on transparent colour + RGBQUAD rgbBG; + if (head.biBitCount<24) rgbBG = GetPaletteColor((BYTE)info.nBkgndIndex); + else rgbBG = info.nBkgndColor; + COLORREF crColour = RGB(rgbBG.rgbRed, rgbBG.rgbGreen, rgbBG.rgbBlue); + COLORREF crOldBack = SetBkColor(dcImage,crColour); + BitBlt(dcTrans,0, 0, nWidth, nHeight, dcImage, 0, 0, SRCCOPY); + + // Do the work - True Mask method - cool if not actual display + StretchBlt(hdc,x, y,cx,cy, dcImage, 0, 0, nWidth, nHeight, SRCINVERT); + StretchBlt(hdc,x, y,cx,cy, dcTrans, 0, 0, nWidth, nHeight, SRCAND); + StretchBlt(hdc,x, y,cx,cy, dcImage, 0, 0, nWidth, nHeight, SRCINVERT); + + // Restore settings + SelectObject(dcImage,pOldBitmapImage); + SelectObject(dcTrans,pOldBitmapTrans); + SetBkColor(hdc,crOldBack); + DeleteObject( bitmapTrans ); // RG 29/01/2002 + DeleteDC(dcImage); + DeleteDC(dcTrans); + DeleteObject(bm); + } + } + return 1; +} +//////////////////////////////////////////////////////////////////////////////// +long CxImage::Stretch(HDC hdc, const RECT& rect, DWORD dwRop) +{ + return Stretch(hdc, rect.left, rect.top, rect.right - rect.left, rect.bottom - rect.top, dwRop); +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Stretch the image. Obsolete: use Draw() or Draw2() + * \param hdc : destination device context + * \param xoffset,yoffset : (optional) offset + * \param xsize,ysize : size. + * \param dwRop : raster operation code (see BitBlt documentation) + * \return true if everything is ok + */ +long CxImage::Stretch(HDC hdc, long xoffset, long yoffset, long xsize, long ysize, DWORD dwRop) +{ + if((pDib)&&(hdc)) { + //palette must be correctly filled + SetStretchBltMode(hdc,COLORONCOLOR); + StretchDIBits(hdc, xoffset, yoffset, + xsize, ysize, 0, 0, head.biWidth, head.biHeight, + info.pImage,(BITMAPINFO*)pDib,DIB_RGB_COLORS,dwRop); + return 1; + } + return 0; +} +//////////////////////////////////////////////////////////////////////////////// +/** + * Tiles the device context in the specified rectangle with the image. + * \param hdc : destination device context + * \param rc : tiled rectangle in the output device context + * \return true if everything is ok + */ +long CxImage::Tile(HDC hdc, RECT *rc) +{ + if((pDib)&&(hdc)&&(rc)) { + int w = rc->right - rc->left; + int h = rc->bottom - rc->top; + int x,y,z; + int bx=head.biWidth; + int by=head.biHeight; + for (y = 0 ; y < h ; y += by){ + if ((y+by)>h) by=h-y; + z=bx; + for (x = 0 ; x < w ; x += z){ + if ((x+z)>w) z=w-x; + RECT r = {rc->left + x,rc->top + y,rc->left + x + z,rc->top + y + by}; + Draw(hdc,rc->left + x, rc->top + y,-1,-1,&r); + } + } + return 1; + } + return 0; +} +//////////////////////////////////////////////////////////////////////////////// +// For UNICODE support: char -> TCHAR +long CxImage::DrawString(HDC hdc, long x, long y, const TCHAR* text, RGBQUAD color, const TCHAR* font, long lSize, long lWeight, BYTE bItalic, BYTE bUnderline, bool bSetAlpha) +//long CxImage::DrawString(HDC hdc, long x, long y, const char* text, RGBQUAD color, const char* font, long lSize, long lWeight, BYTE bItalic, BYTE bUnderline, bool bSetAlpha) +{ + if (IsValid()){ + //get the background + HDC TmpDC=CreateCompatibleDC(hdc); + //choose the font + HFONT m_Font; + LOGFONT* m_pLF; + m_pLF=(LOGFONT*)calloc(1,sizeof(LOGFONT)); + _tcsncpy(m_pLF->lfFaceName,font,31); // For UNICODE support + //strncpy(m_pLF->lfFaceName,font,31); + m_pLF->lfHeight=lSize; + m_pLF->lfWeight=lWeight; + m_pLF->lfItalic=bItalic; + m_pLF->lfUnderline=bUnderline; + m_Font=CreateFontIndirect(m_pLF); + //select the font in the dc + HFONT pOldFont=NULL; + if (m_Font) + pOldFont = (HFONT)SelectObject(TmpDC,m_Font); + else + pOldFont = (HFONT)SelectObject(TmpDC,GetStockObject(DEFAULT_GUI_FONT)); + + //Set text color + SetTextColor(TmpDC,RGB(255,255,255)); + SetBkColor(TmpDC,RGB(0,0,0)); + //draw the text + SetBkMode(TmpDC,OPAQUE); + //Set text position; + RECT pos = {0,0,0,0}; + //long len = (long)strlen(text); + long len = (long)_tcslen(text); // For UNICODE support + ::DrawText(TmpDC,text,len,&pos,DT_CALCRECT); + pos.right+=pos.bottom; //for italics + + //Preparing Bitmap Info + long width=pos.right; + long height=pos.bottom; + BITMAPINFO bmInfo; + memset(&bmInfo.bmiHeader,0,sizeof(BITMAPINFOHEADER)); + bmInfo.bmiHeader.biSize=sizeof(BITMAPINFOHEADER); + bmInfo.bmiHeader.biWidth=width; + bmInfo.bmiHeader.biHeight=height; + bmInfo.bmiHeader.biPlanes=1; + bmInfo.bmiHeader.biBitCount=24; + BYTE *pbase; //points to the final dib + + HBITMAP TmpBmp=CreateDIBSection(TmpDC,&bmInfo,DIB_RGB_COLORS,(void**)&pbase,0,0); + HGDIOBJ TmpObj=SelectObject(TmpDC,TmpBmp); + memset(pbase,0,height*((((24 * width) + 31) / 32) * 4)); + + ::DrawText(TmpDC,text,len,&pos,0); + + CxImage itext; + itext.CreateFromHBITMAP(TmpBmp); + + y=head.biHeight-y-1; + for (long ix=0;ix +long CxImage::DrawStringEx(HDC hdc, long x, long y, CXTEXTINFO *pTextType, bool bSetAlpha ) +{ + if (!IsValid()) + return -1; + + //get the background + HDC pDC; + if (hdc) pDC=hdc; else pDC = ::GetDC(0); + HDC TmpDC=CreateCompatibleDC(pDC); + + //choose the font + HFONT m_Font; + m_Font=CreateFontIndirect( &pTextType->lfont ); + + // get colors in RGBQUAD + RGBQUAD p_forecolor = RGBtoRGBQUAD(pTextType->fcolor); + RGBQUAD p_backcolor = RGBtoRGBQUAD(pTextType->bcolor); + + // check alignment and re-set default if necessary + if ( pTextType->align != DT_CENTER && + pTextType->align != DT_LEFT && + pTextType->align != DT_RIGHT ) + pTextType->align = DT_CENTER; + + // check rounding radius and re-set default if necessary + if ( pTextType->b_round > 50 || pTextType->b_round < 0 ) + pTextType->b_round = 10; + + // check opacity and re-set default if necessary + if ( pTextType->b_opacity > 1. || pTextType->b_opacity < .0 ) + pTextType->b_opacity = 0.; + + //select the font in the dc + HFONT pOldFont=NULL; + if (m_Font) + pOldFont = (HFONT)SelectObject(TmpDC,m_Font); + else + pOldFont = (HFONT)SelectObject(TmpDC,GetStockObject(DEFAULT_GUI_FONT)); + + //Set text color + SetTextColor(TmpDC,RGB(255,255,255)); + SetBkColor(TmpDC,RGB(0,0,0)); + SetBkMode(TmpDC,OPAQUE); + //Set text position; + RECT pos = {0,0,0,0}; + + // get text length and number of lines + long i=0, numlines=1, len=(long)_tcsclen(pTextType->text); + while (itext[i++]==13 ) + numlines++; + } + + ::DrawText(TmpDC, pTextType->text, len, &pos, /*DT_EDITCONTROL|DT_EXTERNALLEADING|*/DT_NOPREFIX | DT_CALCRECT ); + + // increase only if it's really italics, and only one line height + if ( pTextType->lfont.lfItalic ) + pos.right += pos.bottom/2/numlines; + + // background frame and rounding radius + int frame = 0, roundR = 0; + if ( pTextType->opaque ) + { + roundR= (int)(pos.bottom/numlines * pTextType->b_round / 100 ) ; + frame = (int)(/*3.5 + */0.29289*roundR ) ; + pos.right += pos.bottom/numlines/3 ; // JUST FOR BEAUTY + } + + //Preparing Bitmap Info + long width=pos.right +frame*2; + long height=pos.bottom +frame*2; + BITMAPINFO bmInfo; + memset(&bmInfo.bmiHeader,0,sizeof(BITMAPINFOHEADER)); + bmInfo.bmiHeader.biSize=sizeof(BITMAPINFOHEADER); + bmInfo.bmiHeader.biWidth=width; + bmInfo.bmiHeader.biHeight=height; + bmInfo.bmiHeader.biPlanes=1; + bmInfo.bmiHeader.biBitCount=24; + BYTE *pbase; //points to the final dib + + HBITMAP TmpBmp=CreateDIBSection(TmpDC,&bmInfo,DIB_RGB_COLORS,(void**)&pbase,0,0); + HGDIOBJ TmpObj=SelectObject(TmpDC,TmpBmp); + memset(pbase,0,height*((((24 * width) + 31) / 32) * 4)); + + ::DrawText(TmpDC,pTextType->text,len, &pos, /*DT_EDITCONTROL|DT_EXTERNALLEADING|*/DT_NOPREFIX| pTextType->align ); + + CxImage itext; + itext.CreateFromHBITMAP(TmpBmp); + y=head.biHeight-y-1; + + //move the insertion point according to alignment type + // DT_CENTER: cursor points to the center of text rectangle + // DT_RIGHT: cursor points to right side end of text rectangle + // DT_LEFT: cursor points to left end of text rectangle + if ( pTextType->align == DT_CENTER ) + x -= width/2; + else if ( pTextType->align == DT_RIGHT ) + x -= width; + if (x<0) x=0; + + //draw the background first, if it exists + long ix,iy; + if ( pTextType->opaque ) + { + int ixf=0; + for (ix=0;ix=width-roundR-1 ) + ixf = (int)(.5+roundR-sqrt((float)(roundR*roundR-(width-1-ix-roundR)*(width-1-ix-roundR)))); + else + ixf=0; + + for (iy=0;iy height-ixf-1 || iy < ixf )) || + (ix>=width-roundR-1 && ( iy > height-ixf-1 || iy < ixf )) ) + continue; + else + if ( pTextType->b_opacity > 0.0 && pTextType->b_opacity < 1.0 ) + { + RGBQUAD bcolor, pcolor; + // calculate a transition color from original image to background color: + pcolor = GetPixelColor(x+ix,y+iy); + bcolor.rgbBlue = (unsigned char)(pTextType->b_opacity * pcolor.rgbBlue + (1.0-pTextType->b_opacity) * p_backcolor.rgbBlue ); + bcolor.rgbRed = (unsigned char)(pTextType->b_opacity * pcolor.rgbRed + (1.0-pTextType->b_opacity) * p_backcolor.rgbRed ) ; + bcolor.rgbGreen = (unsigned char)(pTextType->b_opacity * pcolor.rgbGreen + (1.0-pTextType->b_opacity) * p_backcolor.rgbGreen ) ; + bcolor.rgbReserved = 0; + SetPixelColor(x+ix,y+iy,bcolor,bSetAlpha); + } + else + SetPixelColor(x+ix,y+iy,p_backcolor,bSetAlpha); + } + } + } + + // draw the text itself + for (ix=0;ixlfont.lfHeight = -36; + txt->lfont.lfCharSet = EASTEUROPE_CHARSET; // just for Central-European users + txt->lfont.lfWeight = FW_NORMAL; + txt->lfont.lfWidth = 0; + txt->lfont.lfEscapement = 0; + txt->lfont.lfOrientation = 0; + txt->lfont.lfItalic = FALSE; + txt->lfont.lfUnderline = FALSE; + txt->lfont.lfStrikeOut = FALSE; + txt->lfont.lfOutPrecision = OUT_DEFAULT_PRECIS; + txt->lfont.lfClipPrecision = CLIP_DEFAULT_PRECIS; + txt->lfont.lfQuality = PROOF_QUALITY; + txt->lfont.lfPitchAndFamily= DEFAULT_PITCH | FF_DONTCARE ; + _stprintf( txt->lfont.lfFaceName, _T("Arial")); //use TCHAR mappings + + // initial colors + txt->fcolor = RGB( 255,255,160 ); // default foreground: light goldyellow + txt->bcolor = RGB( 32, 96, 0 ); // default background: deep green + + // background + txt->opaque = TRUE; // text has a non-transparent background; + txt->b_opacity = 0.0; // default: opaque background + txt->b_outline = 0; // default: no outline (OUTLINE NOT IMPLEMENTED AT THIS TIME) + txt->b_round = 20; // default: rounding radius is 20% of the rectangle height + // the text + _stprintf( txt->text, _T("Sample Text 01234õû")); // text use TCHAR mappings + txt->align = DT_CENTER; + return; +} +//////////////////////////////////////////////////////////////////////////////// +#endif //CXIMAGE_SUPPORT_WINDOWS +//////////////////////////////////////////////////////////////////////////////// diff --git a/src/win32/dependencies/cximage/xiofile.h b/src/win32/dependencies/cximage/xiofile.h new file mode 100644 index 00000000..46c15f21 --- /dev/null +++ b/src/win32/dependencies/cximage/xiofile.h @@ -0,0 +1,112 @@ +#if !defined(__xiofile_h) +#define __xiofile_h + +#include "xfile.h" + +class DLL_EXP CxIOFile : public CxFile + { +public: + CxIOFile(FILE* fp = NULL) + { + m_fp = fp; + m_bCloseFile = (bool)(fp==0); + } + + ~CxIOFile() + { + Close(); + } +////////////////////////////////////////////////////////// + bool Open(const char *filename, const char *mode) + { + if (m_fp) return false; // Can't re-open without closing first + + m_fp = fopen(filename, mode); + if (!m_fp) return false; + + m_bCloseFile = true; + + return true; + } +////////////////////////////////////////////////////////// + virtual bool Close() + { + int iErr = 0; + if ( (m_fp) && (m_bCloseFile) ){ + iErr = fclose(m_fp); + m_fp = NULL; + } + return (bool)(iErr==0); + } +////////////////////////////////////////////////////////// + virtual size_t Read(void *buffer, size_t size, size_t count) + { + if (!m_fp) return 0; + return fread(buffer, size, count, m_fp); + } +////////////////////////////////////////////////////////// + virtual size_t Write(const void *buffer, size_t size, size_t count) + { + if (!m_fp) return 0; + return fwrite(buffer, size, count, m_fp); + } +////////////////////////////////////////////////////////// + virtual bool Seek(long offset, int origin) + { + if (!m_fp) return false; + return (bool)(fseek(m_fp, offset, origin) == 0); + } +////////////////////////////////////////////////////////// + virtual long Tell() + { + if (!m_fp) return 0; + return ftell(m_fp); + } +////////////////////////////////////////////////////////// + virtual long Size() + { + if (!m_fp) return -1; + long pos,size; + pos = ftell(m_fp); + fseek(m_fp, 0, SEEK_END); + size = ftell(m_fp); + fseek(m_fp, pos,SEEK_SET); + return size; + } +////////////////////////////////////////////////////////// + virtual bool Flush() + { + if (!m_fp) return false; + return (bool)(fflush(m_fp) == 0); + } +////////////////////////////////////////////////////////// + virtual bool Eof() + { + if (!m_fp) return true; + return (bool)(feof(m_fp) != 0); + } +////////////////////////////////////////////////////////// + virtual long Error() + { + if (!m_fp) return -1; + return ferror(m_fp); + } +////////////////////////////////////////////////////////// + virtual bool PutC(unsigned char c) + { + if (!m_fp) return false; + return (bool)(fputc(c, m_fp) == c); + } +////////////////////////////////////////////////////////// + virtual long GetC() + { + if (!m_fp) return EOF; + return getc(m_fp); + } +////////////////////////////////////////////////////////// +protected: + FILE *m_fp; + bool m_bCloseFile; + }; + +#endif diff --git a/src/win32/dependencies/cximage/xmemfile.cpp b/src/win32/dependencies/cximage/xmemfile.cpp new file mode 100644 index 00000000..e253c22e --- /dev/null +++ b/src/win32/dependencies/cximage/xmemfile.cpp @@ -0,0 +1,173 @@ +#include "xmemfile.h" + +////////////////////////////////////////////////////////// +CxMemFile::CxMemFile(BYTE* pBuffer, DWORD size) +{ + m_pBuffer = pBuffer; + m_Position = 0; + m_Size = m_Edge = size; + m_bFreeOnClose = (bool)(pBuffer==0); +} +////////////////////////////////////////////////////////// +CxMemFile::~CxMemFile() +{ + Close(); +} +////////////////////////////////////////////////////////// +bool CxMemFile::Close() +{ + if ( (m_pBuffer) && (m_bFreeOnClose) ){ + free(m_pBuffer); + m_pBuffer = NULL; + m_Size = 0; + } + return true; +} +////////////////////////////////////////////////////////// +bool CxMemFile::Open() +{ + if (m_pBuffer) return false; // Can't re-open without closing first + + m_Position = m_Size = m_Edge = 0; + m_pBuffer=(BYTE*)malloc(1); + m_bFreeOnClose = true; + + return (m_pBuffer!=0); +} +////////////////////////////////////////////////////////// +BYTE* CxMemFile::GetBuffer(bool bDetachBuffer) +{ + m_bFreeOnClose = !bDetachBuffer; + return m_pBuffer; +} +////////////////////////////////////////////////////////// +size_t CxMemFile::Read(void *buffer, size_t size, size_t count) +{ + if (buffer==NULL) return 0; + + if (m_pBuffer==NULL) return 0; + if (m_Position >= (long)m_Size) return 0; + + long nCount = (long)(count*size); + if (nCount == 0) return 0; + + long nRead; + if (m_Position + nCount > (long)m_Size) + nRead = (m_Size - m_Position); + else + nRead = nCount; + + memcpy(buffer, m_pBuffer + m_Position, nRead); + m_Position += nRead; + + return (size_t)(nRead/size); +} +////////////////////////////////////////////////////////// +size_t CxMemFile::Write(const void *buffer, size_t size, size_t count) +{ + if (m_pBuffer==NULL) return 0; + if (buffer==NULL) return 0; + + long nCount = (long)(count*size); + if (nCount == 0) return 0; + + if (m_Position + nCount > m_Edge) Alloc(m_Position + nCount); + + memcpy(m_pBuffer + m_Position, buffer, nCount); + + m_Position += nCount; + + if (m_Position > (long)m_Size) m_Size = m_Position; + + return count; +} +////////////////////////////////////////////////////////// +bool CxMemFile::Seek(long offset, int origin) +{ + if (m_pBuffer==NULL) return false; + long lNewPos = m_Position; + + if (origin == SEEK_SET) lNewPos = offset; + else if (origin == SEEK_CUR) lNewPos += offset; + else if (origin == SEEK_END) lNewPos = m_Size + offset; + else return false; + + if (lNewPos < 0) lNewPos = 0; + + m_Position = lNewPos; + return true; +} +////////////////////////////////////////////////////////// +long CxMemFile::Tell() +{ + if (m_pBuffer==NULL) return -1; + return m_Position; +} +////////////////////////////////////////////////////////// +long CxMemFile::Size() +{ + if (m_pBuffer==NULL) return -1; + return m_Size; +} +////////////////////////////////////////////////////////// +bool CxMemFile::Flush() +{ + if (m_pBuffer==NULL) return false; + return true; +} +////////////////////////////////////////////////////////// +bool CxMemFile::Eof() +{ + if (m_pBuffer==NULL) return true; + return (m_Position >= (long)m_Size); +} +////////////////////////////////////////////////////////// +long CxMemFile::Error() +{ + if (m_pBuffer==NULL) return -1; + return (m_Position > (long)m_Size); +} +////////////////////////////////////////////////////////// +bool CxMemFile::PutC(unsigned char c) +{ + if (m_pBuffer==NULL) return false; + if (m_Position + 1 > m_Edge) Alloc(m_Position + 1); + + memcpy(m_pBuffer + m_Position, &c, 1); + + m_Position += 1; + + if (m_Position > (long)m_Size) m_Size = m_Position; + + return true; +} +////////////////////////////////////////////////////////// +long CxMemFile::GetC() +{ + if (Eof()) return EOF; + return *(BYTE*)((BYTE*)m_pBuffer + m_Position++); +} +////////////////////////////////////////////////////////// +void CxMemFile::Alloc(DWORD dwNewLen) +{ + if (dwNewLen > (DWORD)m_Edge) + { + // find new buffer size + DWORD dwNewBufferSize = (DWORD)(((dwNewLen>>12)+1)<<12); + + // allocate new buffer + if (m_pBuffer == NULL) m_pBuffer = (BYTE*)malloc(dwNewBufferSize); + else m_pBuffer = (BYTE*)realloc(m_pBuffer, dwNewBufferSize); + // I own this buffer now (caller knows nothing about it) + m_bFreeOnClose = true; + + m_Edge = dwNewBufferSize; + } + return; +} +////////////////////////////////////////////////////////// +void CxMemFile::Free() +{ + Close(); +} +////////////////////////////////////////////////////////// diff --git a/src/win32/dependencies/cximage/xmemfile.h b/src/win32/dependencies/cximage/xmemfile.h new file mode 100644 index 00000000..f3077cef --- /dev/null +++ b/src/win32/dependencies/cximage/xmemfile.h @@ -0,0 +1,39 @@ +#if !defined(__xmemfile_h) +#define __xmemfile_h + +#include "xfile.h" + +////////////////////////////////////////////////////////// +class DLL_EXP CxMemFile : public CxFile +{ +public: + CxMemFile(BYTE* pBuffer = NULL, DWORD size = 0); + ~CxMemFile(); + + bool Open(); + BYTE* GetBuffer(bool bDetachBuffer = true); + + virtual bool Close(); + virtual size_t Read(void *buffer, size_t size, size_t count); + virtual size_t Write(const void *buffer, size_t size, size_t count); + virtual bool Seek(long offset, int origin); + virtual long Tell(); + virtual long Size(); + virtual bool Flush(); + virtual bool Eof(); + virtual long Error(); + virtual bool PutC(unsigned char c); + virtual long GetC(); + +protected: + void Alloc(DWORD nBytes); + void Free(); + + BYTE* m_pBuffer; + DWORD m_Size; + bool m_bFreeOnClose; + long m_Position; //current position + long m_Edge; //buffer size +}; + +#endif diff --git a/src/win32/dependencies/info.txt b/src/win32/dependencies/info.txt new file mode 100644 index 00000000..45827b3b --- /dev/null +++ b/src/win32/dependencies/info.txt @@ -0,0 +1 @@ +All those libraries use the Static Multi-Threaded C/C++ [Release / Debug] libraries. \ No newline at end of file diff --git a/src/win32/dependencies/libpng/.cvsignore b/src/win32/dependencies/libpng/.cvsignore new file mode 100644 index 00000000..35bb7040 --- /dev/null +++ b/src/win32/dependencies/libpng/.cvsignore @@ -0,0 +1,3 @@ +release +libpng.vcproj.A2.Spacy.user +debug \ No newline at end of file diff --git a/src/win32/dependencies/libpng/CVS/Entries b/src/win32/dependencies/libpng/CVS/Entries new file mode 100644 index 00000000..fd7f5970 --- /dev/null +++ b/src/win32/dependencies/libpng/CVS/Entries @@ -0,0 +1,20 @@ +/.cvsignore/1.1/Sat May 13 16:41:48 2006// +/libpng.vcproj/1.5/Wed Aug 23 22:13:29 2006// +/png.c/1.1/Fri May 12 21:27:07 2006// +/png.h/1.1/Fri May 12 21:27:07 2006// +/pngconf.h/1.1/Fri May 12 21:27:07 2006// +/pngerror.c/1.1/Fri May 12 21:27:07 2006// +/pngget.c/1.1/Fri May 12 21:27:07 2006// +/pngmem.c/1.1/Fri May 12 21:27:07 2006// +/pngpread.c/1.1/Fri May 12 21:27:07 2006// +/pngread.c/1.1/Fri May 12 21:27:07 2006// +/pngrio.c/1.1/Fri May 12 21:27:07 2006// +/pngrtran.c/1.1/Fri May 12 21:27:07 2006// +/pngrutil.c/1.1/Fri May 12 21:27:07 2006// +/pngset.c/1.1/Fri May 12 21:27:07 2006// +/pngtrans.c/1.1/Fri May 12 21:27:07 2006// +/pngwio.c/1.1/Fri May 12 21:27:07 2006// +/pngwrite.c/1.1/Fri May 12 21:27:07 2006// +/pngwtran.c/1.1/Fri May 12 21:27:07 2006// +/pngwutil.c/1.1/Fri May 12 21:27:07 2006// +D diff --git a/src/win32/dependencies/libpng/CVS/Repository b/src/win32/dependencies/libpng/CVS/Repository new file mode 100644 index 00000000..af98f085 --- /dev/null +++ b/src/win32/dependencies/libpng/CVS/Repository @@ -0,0 +1 @@ +VisualBoyAdvance/win32/dependencies/libpng diff --git a/src/win32/dependencies/libpng/CVS/Root b/src/win32/dependencies/libpng/CVS/Root new file mode 100644 index 00000000..6ceab0dd --- /dev/null +++ b/src/win32/dependencies/libpng/CVS/Root @@ -0,0 +1 @@ +:pserver:anonymous@vba.cvs.sourceforge.net:/cvsroot/vba diff --git a/src/win32/dependencies/libpng/libpng.vcproj b/src/win32/dependencies/libpng/libpng.vcproj new file mode 100644 index 00000000..2045378e --- /dev/null +++ b/src/win32/dependencies/libpng/libpng.vcproj @@ -0,0 +1,238 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/win32/dependencies/libpng/png.c b/src/win32/dependencies/libpng/png.c new file mode 100644 index 00000000..3fe66400 --- /dev/null +++ b/src/win32/dependencies/libpng/png.c @@ -0,0 +1,847 @@ + +/* png.c - location for general purpose libpng functions + * + * Last changed in libpng 1.2.9 April 14, 2006 + * For conditions of distribution and use, see copyright notice in png.h + * Copyright (c) 1998-2006 Glenn Randers-Pehrson + * (Version 0.96 Copyright (c) 1996, 1997 Andreas Dilger) + * (Version 0.88 Copyright (c) 1995, 1996 Guy Eric Schalnat, Group 42, Inc.) + */ + +#define PNG_INTERNAL +#define PNG_NO_EXTERN +#include "png.h" + +/* Generate a compiler error if there is an old png.h in the search path. */ +typedef version_1_2_10 Your_png_h_is_not_version_1_2_10; + +/* Version information for C files. This had better match the version + * string defined in png.h. */ + +#ifdef PNG_USE_GLOBAL_ARRAYS +/* png_libpng_ver was changed to a function in version 1.0.5c */ +const char png_libpng_ver[18] = PNG_LIBPNG_VER_STRING; + +#ifdef PNG_READ_SUPPORTED + +/* png_sig was changed to a function in version 1.0.5c */ +/* Place to hold the signature string for a PNG file. */ +const png_byte FARDATA png_sig[8] = {137, 80, 78, 71, 13, 10, 26, 10}; +#endif /* PNG_READ_SUPPORTED */ + +/* Invoke global declarations for constant strings for known chunk types */ +PNG_IHDR; +PNG_IDAT; +PNG_IEND; +PNG_PLTE; +PNG_bKGD; +PNG_cHRM; +PNG_gAMA; +PNG_hIST; +PNG_iCCP; +PNG_iTXt; +PNG_oFFs; +PNG_pCAL; +PNG_sCAL; +PNG_pHYs; +PNG_sBIT; +PNG_sPLT; +PNG_sRGB; +PNG_tEXt; +PNG_tIME; +PNG_tRNS; +PNG_zTXt; + +#ifdef PNG_READ_SUPPORTED +/* arrays to facilitate easy interlacing - use pass (0 - 6) as index */ + +/* start of interlace block */ +const int FARDATA png_pass_start[] = {0, 4, 0, 2, 0, 1, 0}; + +/* offset to next interlace block */ +const int FARDATA png_pass_inc[] = {8, 8, 4, 4, 2, 2, 1}; + +/* start of interlace block in the y direction */ +const int FARDATA png_pass_ystart[] = {0, 0, 4, 0, 2, 0, 1}; + +/* offset to next interlace block in the y direction */ +const int FARDATA png_pass_yinc[] = {8, 8, 8, 4, 4, 2, 2}; + +/* width of interlace block (used in assembler routines only) */ +#ifdef PNG_HAVE_ASSEMBLER_COMBINE_ROW +const int FARDATA png_pass_width[] = {8, 4, 4, 2, 2, 1, 1}; +#endif + +/* Height of interlace block. This is not currently used - if you need + * it, uncomment it here and in png.h +const int FARDATA png_pass_height[] = {8, 8, 4, 4, 2, 2, 1}; +*/ + +/* Mask to determine which pixels are valid in a pass */ +const int FARDATA png_pass_mask[] = {0x80, 0x08, 0x88, 0x22, 0xaa, 0x55, 0xff}; + +/* Mask to determine which pixels to overwrite while displaying */ +const int FARDATA png_pass_dsp_mask[] + = {0xff, 0x0f, 0xff, 0x33, 0xff, 0x55, 0xff}; + +#endif /* PNG_READ_SUPPORTED */ +#endif /* PNG_USE_GLOBAL_ARRAYS */ + +/* Tells libpng that we have already handled the first "num_bytes" bytes + * of the PNG file signature. If the PNG data is embedded into another + * stream we can set num_bytes = 8 so that libpng will not attempt to read + * or write any of the magic bytes before it starts on the IHDR. + */ + +#ifdef PNG_READ_SUPPORTED +void PNGAPI +png_set_sig_bytes(png_structp png_ptr, int num_bytes) +{ + png_debug(1, "in png_set_sig_bytes\n"); + if (num_bytes > 8) + png_error(png_ptr, "Too many bytes for PNG signature."); + + png_ptr->sig_bytes = (png_byte)(num_bytes < 0 ? 0 : num_bytes); +} + +/* Checks whether the supplied bytes match the PNG signature. We allow + * checking less than the full 8-byte signature so that those apps that + * already read the first few bytes of a file to determine the file type + * can simply check the remaining bytes for extra assurance. Returns + * an integer less than, equal to, or greater than zero if sig is found, + * respectively, to be less than, to match, or be greater than the correct + * PNG signature (this is the same behaviour as strcmp, memcmp, etc). + */ +int PNGAPI +png_sig_cmp(png_bytep sig, png_size_t start, png_size_t num_to_check) +{ + png_byte png_signature[8] = {137, 80, 78, 71, 13, 10, 26, 10}; + if (num_to_check > 8) + num_to_check = 8; + else if (num_to_check < 1) + return (-1); + + if (start > 7) + return (-1); + + if (start + num_to_check > 8) + num_to_check = 8 - start; + + return ((int)(png_memcmp(&sig[start], &png_signature[start], num_to_check))); +} + +#if defined(PNG_1_0_X) || defined(PNG_1_2_X) +/* (Obsolete) function to check signature bytes. It does not allow one + * to check a partial signature. This function might be removed in the + * future - use png_sig_cmp(). Returns true (nonzero) if the file is a PNG. + */ +int PNGAPI +png_check_sig(png_bytep sig, int num) +{ + return ((int)!png_sig_cmp(sig, (png_size_t)0, (png_size_t)num)); +} +#endif +#endif /* PNG_READ_SUPPORTED */ + +#if defined(PNG_READ_SUPPORTED) || defined(PNG_WRITE_SUPPORTED) +/* Function to allocate memory for zlib and clear it to 0. */ +#ifdef PNG_1_0_X +voidpf PNGAPI +#else +voidpf /* private */ +#endif +png_zalloc(voidpf png_ptr, uInt items, uInt size) +{ + png_voidp ptr; + png_structp p=png_ptr; + png_uint_32 save_flags=p->flags; + png_uint_32 num_bytes; + + if (items > PNG_UINT_32_MAX/size) + { + png_warning (png_ptr, "Potential overflow in png_zalloc()"); + return (NULL); + } + num_bytes = (png_uint_32)items * size; + + p->flags|=PNG_FLAG_MALLOC_NULL_MEM_OK; + ptr = (png_voidp)png_malloc((png_structp)png_ptr, num_bytes); + p->flags=save_flags; + +#if defined(PNG_1_0_X) && !defined(PNG_NO_ZALLOC_ZERO) + if (ptr == NULL) + return ((voidpf)ptr); + + if (num_bytes > (png_uint_32)0x8000L) + { + png_memset(ptr, 0, (png_size_t)0x8000L); + png_memset((png_bytep)ptr + (png_size_t)0x8000L, 0, + (png_size_t)(num_bytes - (png_uint_32)0x8000L)); + } + else + { + png_memset(ptr, 0, (png_size_t)num_bytes); + } +#endif + return ((voidpf)ptr); +} + +/* function to free memory for zlib */ +#ifdef PNG_1_0_X +void PNGAPI +#else +void /* private */ +#endif +png_zfree(voidpf png_ptr, voidpf ptr) +{ + png_free((png_structp)png_ptr, (png_voidp)ptr); +} + +/* Reset the CRC variable to 32 bits of 1's. Care must be taken + * in case CRC is > 32 bits to leave the top bits 0. + */ +void /* PRIVATE */ +png_reset_crc(png_structp png_ptr) +{ + png_ptr->crc = crc32(0, Z_NULL, 0); +} + +/* Calculate the CRC over a section of data. We can only pass as + * much data to this routine as the largest single buffer size. We + * also check that this data will actually be used before going to the + * trouble of calculating it. + */ +void /* PRIVATE */ +png_calculate_crc(png_structp png_ptr, png_bytep ptr, png_size_t length) +{ + int need_crc = 1; + + if (png_ptr->chunk_name[0] & 0x20) /* ancillary */ + { + if ((png_ptr->flags & PNG_FLAG_CRC_ANCILLARY_MASK) == + (PNG_FLAG_CRC_ANCILLARY_USE | PNG_FLAG_CRC_ANCILLARY_NOWARN)) + need_crc = 0; + } + else /* critical */ + { + if (png_ptr->flags & PNG_FLAG_CRC_CRITICAL_IGNORE) + need_crc = 0; + } + + if (need_crc) + png_ptr->crc = crc32(png_ptr->crc, ptr, (uInt)length); +} + +/* Allocate the memory for an info_struct for the application. We don't + * really need the png_ptr, but it could potentially be useful in the + * future. This should be used in favour of malloc(png_sizeof(png_info)) + * and png_info_init() so that applications that want to use a shared + * libpng don't have to be recompiled if png_info changes size. + */ +png_infop PNGAPI +png_create_info_struct(png_structp png_ptr) +{ + png_infop info_ptr; + + png_debug(1, "in png_create_info_struct\n"); + if(png_ptr == NULL) return (NULL); +#ifdef PNG_USER_MEM_SUPPORTED + info_ptr = (png_infop)png_create_struct_2(PNG_STRUCT_INFO, + png_ptr->malloc_fn, png_ptr->mem_ptr); +#else + info_ptr = (png_infop)png_create_struct(PNG_STRUCT_INFO); +#endif + if (info_ptr != NULL) + png_info_init_3(&info_ptr, png_sizeof(png_info)); + + return (info_ptr); +} + +/* This function frees the memory associated with a single info struct. + * Normally, one would use either png_destroy_read_struct() or + * png_destroy_write_struct() to free an info struct, but this may be + * useful for some applications. + */ +void PNGAPI +png_destroy_info_struct(png_structp png_ptr, png_infopp info_ptr_ptr) +{ + png_infop info_ptr = NULL; + + png_debug(1, "in png_destroy_info_struct\n"); + if (info_ptr_ptr != NULL) + info_ptr = *info_ptr_ptr; + + if (info_ptr != NULL) + { + png_info_destroy(png_ptr, info_ptr); + +#ifdef PNG_USER_MEM_SUPPORTED + png_destroy_struct_2((png_voidp)info_ptr, png_ptr->free_fn, + png_ptr->mem_ptr); +#else + png_destroy_struct((png_voidp)info_ptr); +#endif + *info_ptr_ptr = NULL; + } +} + +/* Initialize the info structure. This is now an internal function (0.89) + * and applications using it are urged to use png_create_info_struct() + * instead. + */ +#if defined(PNG_1_0_X) || defined(PNG_1_2_X) +#undef png_info_init +void PNGAPI +png_info_init(png_infop info_ptr) +{ + /* We only come here via pre-1.0.12-compiled applications */ + png_info_init_3(&info_ptr, 0); +} +#endif + +void PNGAPI +png_info_init_3(png_infopp ptr_ptr, png_size_t png_info_struct_size) +{ + png_infop info_ptr = *ptr_ptr; + + png_debug(1, "in png_info_init_3\n"); + + if(png_sizeof(png_info) > png_info_struct_size) + { + png_destroy_struct(info_ptr); + info_ptr = (png_infop)png_create_struct(PNG_STRUCT_INFO); + *ptr_ptr = info_ptr; + } + + /* set everything to 0 */ + png_memset(info_ptr, 0, png_sizeof (png_info)); +} + +#ifdef PNG_FREE_ME_SUPPORTED +void PNGAPI +png_data_freer(png_structp png_ptr, png_infop info_ptr, + int freer, png_uint_32 mask) +{ + png_debug(1, "in png_data_freer\n"); + if (png_ptr == NULL || info_ptr == NULL) + return; + if(freer == PNG_DESTROY_WILL_FREE_DATA) + info_ptr->free_me |= mask; + else if(freer == PNG_USER_WILL_FREE_DATA) + info_ptr->free_me &= ~mask; + else + png_warning(png_ptr, + "Unknown freer parameter in png_data_freer."); +} +#endif + +void PNGAPI +png_free_data(png_structp png_ptr, png_infop info_ptr, png_uint_32 mask, + int num) +{ + png_debug(1, "in png_free_data\n"); + if (png_ptr == NULL || info_ptr == NULL) + return; + +#if defined(PNG_TEXT_SUPPORTED) +/* free text item num or (if num == -1) all text items */ +#ifdef PNG_FREE_ME_SUPPORTED +if ((mask & PNG_FREE_TEXT) & info_ptr->free_me) +#else +if (mask & PNG_FREE_TEXT) +#endif +{ + if (num != -1) + { + if (info_ptr->text && info_ptr->text[num].key) + { + png_free(png_ptr, info_ptr->text[num].key); + info_ptr->text[num].key = NULL; + } + } + else + { + int i; + for (i = 0; i < info_ptr->num_text; i++) + png_free_data(png_ptr, info_ptr, PNG_FREE_TEXT, i); + png_free(png_ptr, info_ptr->text); + info_ptr->text = NULL; + info_ptr->num_text=0; + } +} +#endif + +#if defined(PNG_tRNS_SUPPORTED) +/* free any tRNS entry */ +#ifdef PNG_FREE_ME_SUPPORTED +if ((mask & PNG_FREE_TRNS) & info_ptr->free_me) +#else +if ((mask & PNG_FREE_TRNS) && (png_ptr->flags & PNG_FLAG_FREE_TRNS)) +#endif +{ + png_free(png_ptr, info_ptr->trans); + info_ptr->valid &= ~PNG_INFO_tRNS; +#ifndef PNG_FREE_ME_SUPPORTED + png_ptr->flags &= ~PNG_FLAG_FREE_TRNS; +#endif + info_ptr->trans = NULL; +} +#endif + +#if defined(PNG_sCAL_SUPPORTED) +/* free any sCAL entry */ +#ifdef PNG_FREE_ME_SUPPORTED +if ((mask & PNG_FREE_SCAL) & info_ptr->free_me) +#else +if (mask & PNG_FREE_SCAL) +#endif +{ +#if defined(PNG_FIXED_POINT_SUPPORTED) && !defined(PNG_FLOATING_POINT_SUPPORTED) + png_free(png_ptr, info_ptr->scal_s_width); + png_free(png_ptr, info_ptr->scal_s_height); + info_ptr->scal_s_width = NULL; + info_ptr->scal_s_height = NULL; +#endif + info_ptr->valid &= ~PNG_INFO_sCAL; +} +#endif + +#if defined(PNG_pCAL_SUPPORTED) +/* free any pCAL entry */ +#ifdef PNG_FREE_ME_SUPPORTED +if ((mask & PNG_FREE_PCAL) & info_ptr->free_me) +#else +if (mask & PNG_FREE_PCAL) +#endif +{ + png_free(png_ptr, info_ptr->pcal_purpose); + png_free(png_ptr, info_ptr->pcal_units); + info_ptr->pcal_purpose = NULL; + info_ptr->pcal_units = NULL; + if (info_ptr->pcal_params != NULL) + { + int i; + for (i = 0; i < (int)info_ptr->pcal_nparams; i++) + { + png_free(png_ptr, info_ptr->pcal_params[i]); + info_ptr->pcal_params[i]=NULL; + } + png_free(png_ptr, info_ptr->pcal_params); + info_ptr->pcal_params = NULL; + } + info_ptr->valid &= ~PNG_INFO_pCAL; +} +#endif + +#if defined(PNG_iCCP_SUPPORTED) +/* free any iCCP entry */ +#ifdef PNG_FREE_ME_SUPPORTED +if ((mask & PNG_FREE_ICCP) & info_ptr->free_me) +#else +if (mask & PNG_FREE_ICCP) +#endif +{ + png_free(png_ptr, info_ptr->iccp_name); + png_free(png_ptr, info_ptr->iccp_profile); + info_ptr->iccp_name = NULL; + info_ptr->iccp_profile = NULL; + info_ptr->valid &= ~PNG_INFO_iCCP; +} +#endif + +#if defined(PNG_sPLT_SUPPORTED) +/* free a given sPLT entry, or (if num == -1) all sPLT entries */ +#ifdef PNG_FREE_ME_SUPPORTED +if ((mask & PNG_FREE_SPLT) & info_ptr->free_me) +#else +if (mask & PNG_FREE_SPLT) +#endif +{ + if (num != -1) + { + if(info_ptr->splt_palettes) + { + png_free(png_ptr, info_ptr->splt_palettes[num].name); + png_free(png_ptr, info_ptr->splt_palettes[num].entries); + info_ptr->splt_palettes[num].name = NULL; + info_ptr->splt_palettes[num].entries = NULL; + } + } + else + { + if(info_ptr->splt_palettes_num) + { + int i; + for (i = 0; i < (int)info_ptr->splt_palettes_num; i++) + png_free_data(png_ptr, info_ptr, PNG_FREE_SPLT, i); + + png_free(png_ptr, info_ptr->splt_palettes); + info_ptr->splt_palettes = NULL; + info_ptr->splt_palettes_num = 0; + } + info_ptr->valid &= ~PNG_INFO_sPLT; + } +} +#endif + +#if defined(PNG_UNKNOWN_CHUNKS_SUPPORTED) +#ifdef PNG_FREE_ME_SUPPORTED +if ((mask & PNG_FREE_UNKN) & info_ptr->free_me) +#else +if (mask & PNG_FREE_UNKN) +#endif +{ + if (num != -1) + { + if(info_ptr->unknown_chunks) + { + png_free(png_ptr, info_ptr->unknown_chunks[num].data); + info_ptr->unknown_chunks[num].data = NULL; + } + } + else + { + int i; + + if(info_ptr->unknown_chunks_num) + { + for (i = 0; i < (int)info_ptr->unknown_chunks_num; i++) + png_free_data(png_ptr, info_ptr, PNG_FREE_UNKN, i); + + png_free(png_ptr, info_ptr->unknown_chunks); + info_ptr->unknown_chunks = NULL; + info_ptr->unknown_chunks_num = 0; + } + } +} +#endif + +#if defined(PNG_hIST_SUPPORTED) +/* free any hIST entry */ +#ifdef PNG_FREE_ME_SUPPORTED +if ((mask & PNG_FREE_HIST) & info_ptr->free_me) +#else +if ((mask & PNG_FREE_HIST) && (png_ptr->flags & PNG_FLAG_FREE_HIST)) +#endif +{ + png_free(png_ptr, info_ptr->hist); + info_ptr->hist = NULL; + info_ptr->valid &= ~PNG_INFO_hIST; +#ifndef PNG_FREE_ME_SUPPORTED + png_ptr->flags &= ~PNG_FLAG_FREE_HIST; +#endif +} +#endif + +/* free any PLTE entry that was internally allocated */ +#ifdef PNG_FREE_ME_SUPPORTED +if ((mask & PNG_FREE_PLTE) & info_ptr->free_me) +#else +if ((mask & PNG_FREE_PLTE) && (png_ptr->flags & PNG_FLAG_FREE_PLTE)) +#endif +{ + png_zfree(png_ptr, info_ptr->palette); + info_ptr->palette = NULL; + info_ptr->valid &= ~PNG_INFO_PLTE; +#ifndef PNG_FREE_ME_SUPPORTED + png_ptr->flags &= ~PNG_FLAG_FREE_PLTE; +#endif + info_ptr->num_palette = 0; +} + +#if defined(PNG_INFO_IMAGE_SUPPORTED) +/* free any image bits attached to the info structure */ +#ifdef PNG_FREE_ME_SUPPORTED +if ((mask & PNG_FREE_ROWS) & info_ptr->free_me) +#else +if (mask & PNG_FREE_ROWS) +#endif +{ + if(info_ptr->row_pointers) + { + int row; + for (row = 0; row < (int)info_ptr->height; row++) + { + png_free(png_ptr, info_ptr->row_pointers[row]); + info_ptr->row_pointers[row]=NULL; + } + png_free(png_ptr, info_ptr->row_pointers); + info_ptr->row_pointers=NULL; + } + info_ptr->valid &= ~PNG_INFO_IDAT; +} +#endif + +#ifdef PNG_FREE_ME_SUPPORTED + if(num == -1) + info_ptr->free_me &= ~mask; + else + info_ptr->free_me &= ~(mask & ~PNG_FREE_MUL); +#endif +} + +/* This is an internal routine to free any memory that the info struct is + * pointing to before re-using it or freeing the struct itself. Recall + * that png_free() checks for NULL pointers for us. + */ +void /* PRIVATE */ +png_info_destroy(png_structp png_ptr, png_infop info_ptr) +{ + png_debug(1, "in png_info_destroy\n"); + + png_free_data(png_ptr, info_ptr, PNG_FREE_ALL, -1); + +#if defined(PNG_UNKNOWN_CHUNKS_SUPPORTED) + if (png_ptr->num_chunk_list) + { + png_free(png_ptr, png_ptr->chunk_list); + png_ptr->chunk_list=NULL; + png_ptr->num_chunk_list=0; + } +#endif + + png_info_init_3(&info_ptr, png_sizeof(png_info)); +} +#endif /* defined(PNG_READ_SUPPORTED) || defined(PNG_WRITE_SUPPORTED) */ + +/* This function returns a pointer to the io_ptr associated with the user + * functions. The application should free any memory associated with this + * pointer before png_write_destroy() or png_read_destroy() are called. + */ +png_voidp PNGAPI +png_get_io_ptr(png_structp png_ptr) +{ + return (png_ptr->io_ptr); +} + +#if defined(PNG_READ_SUPPORTED) || defined(PNG_WRITE_SUPPORTED) +#if !defined(PNG_NO_STDIO) +/* Initialize the default input/output functions for the PNG file. If you + * use your own read or write routines, you can call either png_set_read_fn() + * or png_set_write_fn() instead of png_init_io(). If you have defined + * PNG_NO_STDIO, you must use a function of your own because "FILE *" isn't + * necessarily available. + */ +void PNGAPI +png_init_io(png_structp png_ptr, png_FILE_p fp) +{ + png_debug(1, "in png_init_io\n"); + png_ptr->io_ptr = (png_voidp)fp; +} +#endif + +#if defined(PNG_TIME_RFC1123_SUPPORTED) +/* Convert the supplied time into an RFC 1123 string suitable for use in + * a "Creation Time" or other text-based time string. + */ +png_charp PNGAPI +png_convert_to_rfc1123(png_structp png_ptr, png_timep ptime) +{ + static PNG_CONST char short_months[12][4] = + {"Jan", "Feb", "Mar", "Apr", "May", "Jun", + "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"}; + + if (png_ptr->time_buffer == NULL) + { + png_ptr->time_buffer = (png_charp)png_malloc(png_ptr, (png_uint_32)(29* + png_sizeof(char))); + } + +#if defined(_WIN32_WCE) + { + wchar_t time_buf[29]; + wsprintf(time_buf, TEXT("%d %S %d %02d:%02d:%02d +0000"), + ptime->day % 32, short_months[(ptime->month - 1) % 12], + ptime->year, ptime->hour % 24, ptime->minute % 60, + ptime->second % 61); + WideCharToMultiByte(CP_ACP, 0, time_buf, -1, png_ptr->time_buffer, 29, + NULL, NULL); + } +#else +#ifdef USE_FAR_KEYWORD + { + char near_time_buf[29]; + sprintf(near_time_buf, "%d %s %d %02d:%02d:%02d +0000", + ptime->day % 32, short_months[(ptime->month - 1) % 12], + ptime->year, ptime->hour % 24, ptime->minute % 60, + ptime->second % 61); + png_memcpy(png_ptr->time_buffer, near_time_buf, + 29*png_sizeof(char)); + } +#else + sprintf(png_ptr->time_buffer, "%d %s %d %02d:%02d:%02d +0000", + ptime->day % 32, short_months[(ptime->month - 1) % 12], + ptime->year, ptime->hour % 24, ptime->minute % 60, + ptime->second % 61); +#endif +#endif /* _WIN32_WCE */ + return ((png_charp)png_ptr->time_buffer); +} +#endif /* PNG_TIME_RFC1123_SUPPORTED */ + +#if 0 +/* Signature string for a PNG file. */ +png_bytep PNGAPI +png_sig_bytes(void) +{ + return ((png_bytep)"\211\120\116\107\015\012\032\012"); +} +#endif +#endif /* defined(PNG_READ_SUPPORTED) || defined(PNG_WRITE_SUPPORTED) */ + +png_charp PNGAPI +png_get_copyright(png_structp png_ptr) +{ + if (&png_ptr != NULL) /* silence compiler warning about unused png_ptr */ + return ((png_charp) "\n libpng version 1.2.10 - April 23, 2006\n\ + Copyright (c) 1998-2006 Glenn Randers-Pehrson\n\ + Copyright (c) 1996-1997 Andreas Dilger\n\ + Copyright (c) 1995-1996 Guy Eric Schalnat, Group 42, Inc.\n"); + return ((png_charp) ""); +} + +/* The following return the library version as a short string in the + * format 1.0.0 through 99.99.99zz. To get the version of *.h files + * used with your application, print out PNG_LIBPNG_VER_STRING, which + * is defined in png.h. + * Note: now there is no difference between png_get_libpng_ver() and + * png_get_header_ver(). Due to the version_nn_nn_nn typedef guard, + * it is guaranteed that png.c uses the correct version of png.h. + */ +png_charp PNGAPI +png_get_libpng_ver(png_structp png_ptr) +{ + /* Version of *.c files used when building libpng */ + if (&png_ptr != NULL) /* silence compiler warning about unused png_ptr */ + return ((png_charp) PNG_LIBPNG_VER_STRING); + return ((png_charp) ""); +} + +png_charp PNGAPI +png_get_header_ver(png_structp png_ptr) +{ + /* Version of *.h files used when building libpng */ + if (&png_ptr != NULL) /* silence compiler warning about unused png_ptr */ + return ((png_charp) PNG_LIBPNG_VER_STRING); + return ((png_charp) ""); +} + +png_charp PNGAPI +png_get_header_version(png_structp png_ptr) +{ + /* Returns longer string containing both version and date */ + if (&png_ptr != NULL) /* silence compiler warning about unused png_ptr */ + return ((png_charp) PNG_HEADER_VERSION_STRING); + return ((png_charp) ""); +} + +#if defined(PNG_READ_SUPPORTED) || defined(PNG_WRITE_SUPPORTED) +#ifdef PNG_HANDLE_AS_UNKNOWN_SUPPORTED +int PNGAPI +png_handle_as_unknown(png_structp png_ptr, png_bytep chunk_name) +{ + /* check chunk_name and return "keep" value if it's on the list, else 0 */ + int i; + png_bytep p; + if((png_ptr == NULL && chunk_name == NULL) || png_ptr->num_chunk_list<=0) + return 0; + p=png_ptr->chunk_list+png_ptr->num_chunk_list*5-5; + for (i = png_ptr->num_chunk_list; i; i--, p-=5) + if (!png_memcmp(chunk_name, p, 4)) + return ((int)*(p+4)); + return 0; +} +#endif + +/* This function, added to libpng-1.0.6g, is untested. */ +int PNGAPI +png_reset_zstream(png_structp png_ptr) +{ + return (inflateReset(&png_ptr->zstream)); +} +#endif /* defined(PNG_READ_SUPPORTED) || defined(PNG_WRITE_SUPPORTED) */ + +/* This function was added to libpng-1.0.7 */ +png_uint_32 PNGAPI +png_access_version_number(void) +{ + /* Version of *.c files used when building libpng */ + return((png_uint_32) PNG_LIBPNG_VER); +} + + +#if defined(PNG_READ_SUPPORTED) +#if !defined(PNG_1_0_X) +#if defined(PNG_ASSEMBLER_CODE_SUPPORTED) + /* GRR: could add this: && defined(PNG_MMX_CODE_SUPPORTED) */ +/* this INTERNAL function was added to libpng 1.2.0 */ +void /* PRIVATE */ +png_init_mmx_flags (png_structp png_ptr) +{ + png_ptr->mmx_rowbytes_threshold = 0; + png_ptr->mmx_bitdepth_threshold = 0; + +# if (defined(PNG_USE_PNGVCRD) || defined(PNG_USE_PNGGCCRD)) + + png_ptr->asm_flags |= PNG_ASM_FLAG_MMX_SUPPORT_COMPILED; + + if (png_mmx_support() > 0) { + png_ptr->asm_flags |= PNG_ASM_FLAG_MMX_SUPPORT_IN_CPU +# ifdef PNG_HAVE_ASSEMBLER_COMBINE_ROW + | PNG_ASM_FLAG_MMX_READ_COMBINE_ROW +# endif +# ifdef PNG_HAVE_ASSEMBLER_READ_INTERLACE + | PNG_ASM_FLAG_MMX_READ_INTERLACE +# endif +# ifndef PNG_HAVE_ASSEMBLER_READ_FILTER_ROW + ; +# else + | PNG_ASM_FLAG_MMX_READ_FILTER_SUB + | PNG_ASM_FLAG_MMX_READ_FILTER_UP + | PNG_ASM_FLAG_MMX_READ_FILTER_AVG + | PNG_ASM_FLAG_MMX_READ_FILTER_PAETH ; + + png_ptr->mmx_rowbytes_threshold = PNG_MMX_ROWBYTES_THRESHOLD_DEFAULT; + png_ptr->mmx_bitdepth_threshold = PNG_MMX_BITDEPTH_THRESHOLD_DEFAULT; +# endif + } else { + png_ptr->asm_flags &= ~( PNG_ASM_FLAG_MMX_SUPPORT_IN_CPU + | PNG_MMX_READ_FLAGS + | PNG_MMX_WRITE_FLAGS ); + } + +# else /* !((PNGVCRD || PNGGCCRD) && PNG_ASSEMBLER_CODE_SUPPORTED)) */ + + /* clear all MMX flags; no support is compiled in */ + png_ptr->asm_flags &= ~( PNG_MMX_FLAGS ); + +# endif /* ?(PNGVCRD || PNGGCCRD) */ +} + +#endif /* !(PNG_ASSEMBLER_CODE_SUPPORTED) */ + +/* this function was added to libpng 1.2.0 */ +#if !defined(PNG_USE_PNGGCCRD) && \ + !(defined(PNG_ASSEMBLER_CODE_SUPPORTED) && defined(PNG_USE_PNGVCRD)) +int PNGAPI +png_mmx_support(void) +{ + return -1; +} +#endif +#endif /* PNG_1_0_X */ +#endif /* PNG_READ_SUPPORTED */ + +#if defined(PNG_READ_SUPPORTED) || defined(PNG_WRITE_SUPPORTED) +#ifdef PNG_SIZE_T +/* Added at libpng version 1.2.6 */ + PNG_EXTERN png_size_t PNGAPI png_convert_size PNGARG((size_t size)); +png_size_t PNGAPI +png_convert_size(size_t size) +{ + if (size > (png_size_t)-1) + PNG_ABORT(); /* We haven't got access to png_ptr, so no png_error() */ + return ((png_size_t)size); +} +#endif /* PNG_SIZE_T */ +#endif /* defined(PNG_READ_SUPPORTED) || defined(PNG_WRITE_SUPPORTED) */ diff --git a/src/win32/dependencies/libpng/png.h b/src/win32/dependencies/libpng/png.h new file mode 100644 index 00000000..5e41c323 --- /dev/null +++ b/src/win32/dependencies/libpng/png.h @@ -0,0 +1,3450 @@ + +/* png.h - header file for PNG reference library + * + * libpng version 1.2.10 - April 23, 2006 + * Copyright (c) 1998-2006 Glenn Randers-Pehrson + * (Version 0.96 Copyright (c) 1996, 1997 Andreas Dilger) + * (Version 0.88 Copyright (c) 1995, 1996 Guy Eric Schalnat, Group 42, Inc.) + * + * Authors and maintainers: + * libpng versions 0.71, May 1995, through 0.88, January 1996: Guy Schalnat + * libpng versions 0.89c, June 1996, through 0.96, May 1997: Andreas Dilger + * libpng versions 0.97, January 1998, through 1.2.10 - April 23, 2006: Glenn + * See also "Contributing Authors", below. + * + * Note about libpng version numbers: + * + * Due to various miscommunications, unforeseen code incompatibilities + * and occasional factors outside the authors' control, version numbering + * on the library has not always been consistent and straightforward. + * The following table summarizes matters since version 0.89c, which was + * the first widely used release: + * + * source png.h png.h shared-lib + * version string int version + * ------- ------ ----- ---------- + * 0.89c "1.0 beta 3" 0.89 89 1.0.89 + * 0.90 "1.0 beta 4" 0.90 90 0.90 [should have been 2.0.90] + * 0.95 "1.0 beta 5" 0.95 95 0.95 [should have been 2.0.95] + * 0.96 "1.0 beta 6" 0.96 96 0.96 [should have been 2.0.96] + * 0.97b "1.00.97 beta 7" 1.00.97 97 1.0.1 [should have been 2.0.97] + * 0.97c 0.97 97 2.0.97 + * 0.98 0.98 98 2.0.98 + * 0.99 0.99 98 2.0.99 + * 0.99a-m 0.99 99 2.0.99 + * 1.00 1.00 100 2.1.0 [100 should be 10000] + * 1.0.0 (from here on, the 100 2.1.0 [100 should be 10000] + * 1.0.1 png.h string is 10001 2.1.0 + * 1.0.1a-e identical to the 10002 from here on, the shared library + * 1.0.2 source version) 10002 is 2.V where V is the source code + * 1.0.2a-b 10003 version, except as noted. + * 1.0.3 10003 + * 1.0.3a-d 10004 + * 1.0.4 10004 + * 1.0.4a-f 10005 + * 1.0.5 (+ 2 patches) 10005 + * 1.0.5a-d 10006 + * 1.0.5e-r 10100 (not source compatible) + * 1.0.5s-v 10006 (not binary compatible) + * 1.0.6 (+ 3 patches) 10006 (still binary incompatible) + * 1.0.6d-f 10007 (still binary incompatible) + * 1.0.6g 10007 + * 1.0.6h 10007 10.6h (testing xy.z so-numbering) + * 1.0.6i 10007 10.6i + * 1.0.6j 10007 2.1.0.6j (incompatible with 1.0.0) + * 1.0.7beta11-14 DLLNUM 10007 2.1.0.7beta11-14 (binary compatible) + * 1.0.7beta15-18 1 10007 2.1.0.7beta15-18 (binary compatible) + * 1.0.7rc1-2 1 10007 2.1.0.7rc1-2 (binary compatible) + * 1.0.7 1 10007 (still compatible) + * 1.0.8beta1-4 1 10008 2.1.0.8beta1-4 + * 1.0.8rc1 1 10008 2.1.0.8rc1 + * 1.0.8 1 10008 2.1.0.8 + * 1.0.9beta1-6 1 10009 2.1.0.9beta1-6 + * 1.0.9rc1 1 10009 2.1.0.9rc1 + * 1.0.9beta7-10 1 10009 2.1.0.9beta7-10 + * 1.0.9rc2 1 10009 2.1.0.9rc2 + * 1.0.9 1 10009 2.1.0.9 + * 1.0.10beta1 1 10010 2.1.0.10beta1 + * 1.0.10rc1 1 10010 2.1.0.10rc1 + * 1.0.10 1 10010 2.1.0.10 + * 1.0.11beta1-3 1 10011 2.1.0.11beta1-3 + * 1.0.11rc1 1 10011 2.1.0.11rc1 + * 1.0.11 1 10011 2.1.0.11 + * 1.0.12beta1-2 2 10012 2.1.0.12beta1-2 + * 1.0.12rc1 2 10012 2.1.0.12rc1 + * 1.0.12 2 10012 2.1.0.12 + * 1.1.0a-f - 10100 2.1.1.0a-f (branch abandoned) + * 1.2.0beta1-2 2 10200 2.1.2.0beta1-2 + * 1.2.0beta3-5 3 10200 3.1.2.0beta3-5 + * 1.2.0rc1 3 10200 3.1.2.0rc1 + * 1.2.0 3 10200 3.1.2.0 + * 1.2.1beta1-4 3 10201 3.1.2.1beta1-4 + * 1.2.1rc1-2 3 10201 3.1.2.1rc1-2 + * 1.2.1 3 10201 3.1.2.1 + * 1.2.2beta1-6 12 10202 12.so.0.1.2.2beta1-6 + * 1.0.13beta1 10 10013 10.so.0.1.0.13beta1 + * 1.0.13rc1 10 10013 10.so.0.1.0.13rc1 + * 1.2.2rc1 12 10202 12.so.0.1.2.2rc1 + * 1.0.13 10 10013 10.so.0.1.0.13 + * 1.2.2 12 10202 12.so.0.1.2.2 + * 1.2.3rc1-6 12 10203 12.so.0.1.2.3rc1-6 + * 1.2.3 12 10203 12.so.0.1.2.3 + * 1.2.4beta1-3 13 10204 12.so.0.1.2.4beta1-3 + * 1.0.14rc1 13 10014 10.so.0.1.0.14rc1 + * 1.2.4rc1 13 10204 12.so.0.1.2.4rc1 + * 1.0.14 10 10014 10.so.0.1.0.14 + * 1.2.4 13 10204 12.so.0.1.2.4 + * 1.2.5beta1-2 13 10205 12.so.0.1.2.5beta1-2 + * 1.0.15rc1-3 10 10015 10.so.0.1.0.15rc1-3 + * 1.2.5rc1-3 13 10205 12.so.0.1.2.5rc1-3 + * 1.0.15 10 10015 10.so.0.1.0.15 + * 1.2.5 13 10205 12.so.0.1.2.5 + * 1.2.6beta1-4 13 10206 12.so.0.1.2.6beta1-4 + * 1.0.16 10 10016 10.so.0.1.0.16 + * 1.2.6 13 10206 12.so.0.1.2.6 + * 1.2.7beta1-2 13 10207 12.so.0.1.2.7beta1-2 + * 1.0.17rc1 10 10017 12.so.0.1.0.17rc1 + * 1.2.7rc1 13 10207 12.so.0.1.2.7rc1 + * 1.0.17 10 10017 12.so.0.1.0.17 + * 1.2.7 13 10207 12.so.0.1.2.7 + * 1.2.8beta1-5 13 10208 12.so.0.1.2.8beta1-5 + * 1.0.18rc1-5 10 10018 12.so.0.1.0.18rc1-5 + * 1.2.8rc1-5 13 10208 12.so.0.1.2.8rc1-5 + * 1.0.18 10 10018 12.so.0.1.0.18 + * 1.2.8 13 10208 12.so.0.1.2.8 + * 1.2.9beta1-3 13 10209 12.so.0.1.2.9beta1-3 + * 1.2.9beta4-11 13 10209 12.so.0.9[.0] + * 1.2.9rc1 13 10209 12.so.0.9[.0] + * 1.2.9 13 10209 12.so.0.9[.0] + * 1.2.10beta1-8 13 10210 12.so.0.10[.0] + * 1.2.10rc1-3 13 10210 12.so.0.10[.0] + * 1.2.10 13 10210 12.so.0.10[.0] + * + * Henceforth the source version will match the shared-library major + * and minor numbers; the shared-library major version number will be + * used for changes in backward compatibility, as it is intended. The + * PNG_LIBPNG_VER macro, which is not used within libpng but is available + * for applications, is an unsigned integer of the form xyyzz corresponding + * to the source version x.y.z (leading zeros in y and z). Beta versions + * were given the previous public release number plus a letter, until + * version 1.0.6j; from then on they were given the upcoming public + * release number plus "betaNN" or "rcN". + * + * Binary incompatibility exists only when applications make direct access + * to the info_ptr or png_ptr members through png.h, and the compiled + * application is loaded with a different version of the library. + * + * DLLNUM will change each time there are forward or backward changes + * in binary compatibility (e.g., when a new feature is added). + * + * See libpng.txt or libpng.3 for more information. The PNG specification + * is available as a W3C Recommendation and as an ISO Specification, + * defines should NOT be changed. + */ +#define PNG_INFO_gAMA 0x0001 +#define PNG_INFO_sBIT 0x0002 +#define PNG_INFO_cHRM 0x0004 +#define PNG_INFO_PLTE 0x0008 +#define PNG_INFO_tRNS 0x0010 +#define PNG_INFO_bKGD 0x0020 +#define PNG_INFO_hIST 0x0040 +#define PNG_INFO_pHYs 0x0080 +#define PNG_INFO_oFFs 0x0100 +#define PNG_INFO_tIME 0x0200 +#define PNG_INFO_pCAL 0x0400 +#define PNG_INFO_sRGB 0x0800 /* GR-P, 0.96a */ +#define PNG_INFO_iCCP 0x1000 /* ESR, 1.0.6 */ +#define PNG_INFO_sPLT 0x2000 /* ESR, 1.0.6 */ +#define PNG_INFO_sCAL 0x4000 /* ESR, 1.0.6 */ +#define PNG_INFO_IDAT 0x8000L /* ESR, 1.0.6 */ + +/* This is used for the transformation routines, as some of them + * change these values for the row. It also should enable using + * the routines for other purposes. + */ +typedef struct png_row_info_struct +{ + png_uint_32 width; /* width of row */ + png_uint_32 rowbytes; /* number of bytes in row */ + png_byte color_type; /* color type of row */ + png_byte bit_depth; /* bit depth of row */ + png_byte channels; /* number of channels (1, 2, 3, or 4) */ + png_byte pixel_depth; /* bits per pixel (depth * channels) */ +} png_row_info; + +typedef png_row_info FAR * png_row_infop; +typedef png_row_info FAR * FAR * png_row_infopp; + +/* These are the function types for the I/O functions and for the functions + * that allow the user to override the default I/O functions with his or her + * own. The png_error_ptr type should match that of user-supplied warning + * and error functions, while the png_rw_ptr type should match that of the + * user read/write data functions. + */ +typedef struct png_struct_def png_struct; +typedef png_struct FAR * png_structp; + +typedef void (PNGAPI *png_error_ptr) PNGARG((png_structp, png_const_charp)); +typedef void (PNGAPI *png_rw_ptr) PNGARG((png_structp, png_bytep, png_size_t)); +typedef void (PNGAPI *png_flush_ptr) PNGARG((png_structp)); +typedef void (PNGAPI *png_read_status_ptr) PNGARG((png_structp, png_uint_32, + int)); +typedef void (PNGAPI *png_write_status_ptr) PNGARG((png_structp, png_uint_32, + int)); + +#ifdef PNG_PROGRESSIVE_READ_SUPPORTED +typedef void (PNGAPI *png_progressive_info_ptr) PNGARG((png_structp, png_infop)); +typedef void (PNGAPI *png_progressive_end_ptr) PNGARG((png_structp, png_infop)); +typedef void (PNGAPI *png_progressive_row_ptr) PNGARG((png_structp, png_bytep, + png_uint_32, int)); +#endif + +#if defined(PNG_READ_USER_TRANSFORM_SUPPORTED) || \ + defined(PNG_WRITE_USER_TRANSFORM_SUPPORTED) || \ + defined(PNG_LEGACY_SUPPORTED) +typedef void (PNGAPI *png_user_transform_ptr) PNGARG((png_structp, + png_row_infop, png_bytep)); +#endif + +#if defined(PNG_USER_CHUNKS_SUPPORTED) +typedef int (PNGAPI *png_user_chunk_ptr) PNGARG((png_structp, png_unknown_chunkp)); +#endif +#if defined(PNG_UNKNOWN_CHUNKS_SUPPORTED) +typedef void (PNGAPI *png_unknown_chunk_ptr) PNGARG((png_structp)); +#endif + +/* Transform masks for the high-level interface */ +#define PNG_TRANSFORM_IDENTITY 0x0000 /* read and write */ +#define PNG_TRANSFORM_STRIP_16 0x0001 /* read only */ +#define PNG_TRANSFORM_STRIP_ALPHA 0x0002 /* read only */ +#define PNG_TRANSFORM_PACKING 0x0004 /* read and write */ +#define PNG_TRANSFORM_PACKSWAP 0x0008 /* read and write */ +#define PNG_TRANSFORM_EXPAND 0x0010 /* read only */ +#define PNG_TRANSFORM_INVERT_MONO 0x0020 /* read and write */ +#define PNG_TRANSFORM_SHIFT 0x0040 /* read and write */ +#define PNG_TRANSFORM_BGR 0x0080 /* read and write */ +#define PNG_TRANSFORM_SWAP_ALPHA 0x0100 /* read and write */ +#define PNG_TRANSFORM_SWAP_ENDIAN 0x0200 /* read and write */ +#define PNG_TRANSFORM_INVERT_ALPHA 0x0400 /* read and write */ +#define PNG_TRANSFORM_STRIP_FILLER 0x0800 /* WRITE only */ + +/* Flags for MNG supported features */ +#define PNG_FLAG_MNG_EMPTY_PLTE 0x01 +#define PNG_FLAG_MNG_FILTER_64 0x04 +#define PNG_ALL_MNG_FEATURES 0x05 + +typedef png_voidp (*png_malloc_ptr) PNGARG((png_structp, png_size_t)); +typedef void (*png_free_ptr) PNGARG((png_structp, png_voidp)); + +/* The structure that holds the information to read and write PNG files. + * The only people who need to care about what is inside of this are the + * people who will be modifying the library for their own special needs. + * It should NOT be accessed directly by an application, except to store + * the jmp_buf. + */ + +struct png_struct_def +{ +#ifdef PNG_SETJMP_SUPPORTED + jmp_buf jmpbuf; /* used in png_error */ +#endif + png_error_ptr error_fn; /* function for printing errors and aborting */ + png_error_ptr warning_fn; /* function for printing warnings */ + png_voidp error_ptr; /* user supplied struct for error functions */ + png_rw_ptr write_data_fn; /* function for writing output data */ + png_rw_ptr read_data_fn; /* function for reading input data */ + png_voidp io_ptr; /* ptr to application struct for I/O functions */ + +#if defined(PNG_READ_USER_TRANSFORM_SUPPORTED) + png_user_transform_ptr read_user_transform_fn; /* user read transform */ +#endif + +#if defined(PNG_WRITE_USER_TRANSFORM_SUPPORTED) + png_user_transform_ptr write_user_transform_fn; /* user write transform */ +#endif + +/* These were added in libpng-1.0.2 */ +#if defined(PNG_USER_TRANSFORM_PTR_SUPPORTED) +#if defined(PNG_READ_USER_TRANSFORM_SUPPORTED) || \ + defined(PNG_WRITE_USER_TRANSFORM_SUPPORTED) + png_voidp user_transform_ptr; /* user supplied struct for user transform */ + png_byte user_transform_depth; /* bit depth of user transformed pixels */ + png_byte user_transform_channels; /* channels in user transformed pixels */ +#endif +#endif + + png_uint_32 mode; /* tells us where we are in the PNG file */ + png_uint_32 flags; /* flags indicating various things to libpng */ + png_uint_32 transformations; /* which transformations to perform */ + + z_stream zstream; /* pointer to decompression structure (below) */ + png_bytep zbuf; /* buffer for zlib */ + png_size_t zbuf_size; /* size of zbuf */ + int zlib_level; /* holds zlib compression level */ + int zlib_method; /* holds zlib compression method */ + int zlib_window_bits; /* holds zlib compression window bits */ + int zlib_mem_level; /* holds zlib compression memory level */ + int zlib_strategy; /* holds zlib compression strategy */ + + png_uint_32 width; /* width of image in pixels */ + png_uint_32 height; /* height of image in pixels */ + png_uint_32 num_rows; /* number of rows in current pass */ + png_uint_32 usr_width; /* width of row at start of write */ + png_uint_32 rowbytes; /* size of row in bytes */ + png_uint_32 irowbytes; /* size of current interlaced row in bytes */ + png_uint_32 iwidth; /* width of current interlaced row in pixels */ + png_uint_32 row_number; /* current row in interlace pass */ + png_bytep prev_row; /* buffer to save previous (unfiltered) row */ + png_bytep row_buf; /* buffer to save current (unfiltered) row */ + png_bytep sub_row; /* buffer to save "sub" row when filtering */ + png_bytep up_row; /* buffer to save "up" row when filtering */ + png_bytep avg_row; /* buffer to save "avg" row when filtering */ + png_bytep paeth_row; /* buffer to save "Paeth" row when filtering */ + png_row_info row_info; /* used for transformation routines */ + + png_uint_32 idat_size; /* current IDAT size for read */ + png_uint_32 crc; /* current chunk CRC value */ + png_colorp palette; /* palette from the input file */ + png_uint_16 num_palette; /* number of color entries in palette */ + png_uint_16 num_trans; /* number of transparency values */ + png_byte chunk_name[5]; /* null-terminated name of current chunk */ + png_byte compression; /* file compression type (always 0) */ + png_byte filter; /* file filter type (always 0) */ + png_byte interlaced; /* PNG_INTERLACE_NONE, PNG_INTERLACE_ADAM7 */ + png_byte pass; /* current interlace pass (0 - 6) */ + png_byte do_filter; /* row filter flags (see PNG_FILTER_ below ) */ + png_byte color_type; /* color type of file */ + png_byte bit_depth; /* bit depth of file */ + png_byte usr_bit_depth; /* bit depth of users row */ + png_byte pixel_depth; /* number of bits per pixel */ + png_byte channels; /* number of channels in file */ + png_byte usr_channels; /* channels at start of write */ + png_byte sig_bytes; /* magic bytes read/written from start of file */ + +#if defined(PNG_READ_FILLER_SUPPORTED) || defined(PNG_WRITE_FILLER_SUPPORTED) +#ifdef PNG_LEGACY_SUPPORTED + png_byte filler; /* filler byte for pixel expansion */ +#else + png_uint_16 filler; /* filler bytes for pixel expansion */ +#endif +#endif + +#if defined(PNG_bKGD_SUPPORTED) + png_byte background_gamma_type; +# ifdef PNG_FLOATING_POINT_SUPPORTED + float background_gamma; +# endif + png_color_16 background; /* background color in screen gamma space */ +#if defined(PNG_READ_GAMMA_SUPPORTED) + png_color_16 background_1; /* background normalized to gamma 1.0 */ +#endif +#endif /* PNG_bKGD_SUPPORTED */ + +#if defined(PNG_WRITE_FLUSH_SUPPORTED) + png_flush_ptr output_flush_fn;/* Function for flushing output */ + png_uint_32 flush_dist; /* how many rows apart to flush, 0 - no flush */ + png_uint_32 flush_rows; /* number of rows written since last flush */ +#endif + +#if defined(PNG_READ_GAMMA_SUPPORTED) || defined(PNG_READ_BACKGROUND_SUPPORTED) + int gamma_shift; /* number of "insignificant" bits 16-bit gamma */ +#ifdef PNG_FLOATING_POINT_SUPPORTED + float gamma; /* file gamma value */ + float screen_gamma; /* screen gamma value (display_exponent) */ +#endif +#endif + +#if defined(PNG_READ_GAMMA_SUPPORTED) || defined(PNG_READ_BACKGROUND_SUPPORTED) + png_bytep gamma_table; /* gamma table for 8-bit depth files */ + png_bytep gamma_from_1; /* converts from 1.0 to screen */ + png_bytep gamma_to_1; /* converts from file to 1.0 */ + png_uint_16pp gamma_16_table; /* gamma table for 16-bit depth files */ + png_uint_16pp gamma_16_from_1; /* converts from 1.0 to screen */ + png_uint_16pp gamma_16_to_1; /* converts from file to 1.0 */ +#endif + +#if defined(PNG_READ_GAMMA_SUPPORTED) || defined(PNG_sBIT_SUPPORTED) + png_color_8 sig_bit; /* significant bits in each available channel */ +#endif + +#if defined(PNG_READ_SHIFT_SUPPORTED) || defined(PNG_WRITE_SHIFT_SUPPORTED) + png_color_8 shift; /* shift for significant bit tranformation */ +#endif + +#if defined(PNG_tRNS_SUPPORTED) || defined(PNG_READ_BACKGROUND_SUPPORTED) \ + || defined(PNG_READ_EXPAND_SUPPORTED) || defined(PNG_READ_BACKGROUND_SUPPORTED) + png_bytep trans; /* transparency values for paletted files */ + png_color_16 trans_values; /* transparency values for non-paletted files */ +#endif + + png_read_status_ptr read_row_fn; /* called after each row is decoded */ + png_write_status_ptr write_row_fn; /* called after each row is encoded */ +#ifdef PNG_PROGRESSIVE_READ_SUPPORTED + png_progressive_info_ptr info_fn; /* called after header data fully read */ + png_progressive_row_ptr row_fn; /* called after each prog. row is decoded */ + png_progressive_end_ptr end_fn; /* called after image is complete */ + png_bytep save_buffer_ptr; /* current location in save_buffer */ + png_bytep save_buffer; /* buffer for previously read data */ + png_bytep current_buffer_ptr; /* current location in current_buffer */ + png_bytep current_buffer; /* buffer for recently used data */ + png_uint_32 push_length; /* size of current input chunk */ + png_uint_32 skip_length; /* bytes to skip in input data */ + png_size_t save_buffer_size; /* amount of data now in save_buffer */ + png_size_t save_buffer_max; /* total size of save_buffer */ + png_size_t buffer_size; /* total amount of available input data */ + png_size_t current_buffer_size; /* amount of data now in current_buffer */ + int process_mode; /* what push library is currently doing */ + int cur_palette; /* current push library palette index */ + +# if defined(PNG_TEXT_SUPPORTED) + png_size_t current_text_size; /* current size of text input data */ + png_size_t current_text_left; /* how much text left to read in input */ + png_charp current_text; /* current text chunk buffer */ + png_charp current_text_ptr; /* current location in current_text */ +# endif /* PNG_PROGRESSIVE_READ_SUPPORTED && PNG_TEXT_SUPPORTED */ + +#endif /* PNG_PROGRESSIVE_READ_SUPPORTED */ + +#if defined(__TURBOC__) && !defined(_Windows) && !defined(__FLAT__) +/* for the Borland special 64K segment handler */ + png_bytepp offset_table_ptr; + png_bytep offset_table; + png_uint_16 offset_table_number; + png_uint_16 offset_table_count; + png_uint_16 offset_table_count_free; +#endif + +#if defined(PNG_READ_DITHER_SUPPORTED) + png_bytep palette_lookup; /* lookup table for dithering */ + png_bytep dither_index; /* index translation for palette files */ +#endif + +#if defined(PNG_READ_DITHER_SUPPORTED) || defined(PNG_hIST_SUPPORTED) + png_uint_16p hist; /* histogram */ +#endif + +#if defined(PNG_WRITE_WEIGHTED_FILTER_SUPPORTED) + png_byte heuristic_method; /* heuristic for row filter selection */ + png_byte num_prev_filters; /* number of weights for previous rows */ + png_bytep prev_filters; /* filter type(s) of previous row(s) */ + png_uint_16p filter_weights; /* weight(s) for previous line(s) */ + png_uint_16p inv_filter_weights; /* 1/weight(s) for previous line(s) */ + png_uint_16p filter_costs; /* relative filter calculation cost */ + png_uint_16p inv_filter_costs; /* 1/relative filter calculation cost */ +#endif + +#if defined(PNG_TIME_RFC1123_SUPPORTED) + png_charp time_buffer; /* String to hold RFC 1123 time text */ +#endif + +/* New members added in libpng-1.0.6 */ + +#ifdef PNG_FREE_ME_SUPPORTED + png_uint_32 free_me; /* flags items libpng is responsible for freeing */ +#endif + +#if defined(PNG_USER_CHUNKS_SUPPORTED) + png_voidp user_chunk_ptr; + png_user_chunk_ptr read_user_chunk_fn; /* user read chunk handler */ +#endif + +#if defined(PNG_UNKNOWN_CHUNKS_SUPPORTED) + int num_chunk_list; + png_bytep chunk_list; +#endif + +/* New members added in libpng-1.0.3 */ +#if defined(PNG_READ_RGB_TO_GRAY_SUPPORTED) + png_byte rgb_to_gray_status; + /* These were changed from png_byte in libpng-1.0.6 */ + png_uint_16 rgb_to_gray_red_coeff; + png_uint_16 rgb_to_gray_green_coeff; + png_uint_16 rgb_to_gray_blue_coeff; +#endif + +/* New member added in libpng-1.0.4 (renamed in 1.0.9) */ +#if defined(PNG_MNG_FEATURES_SUPPORTED) || \ + defined(PNG_READ_EMPTY_PLTE_SUPPORTED) || \ + defined(PNG_WRITE_EMPTY_PLTE_SUPPORTED) +/* changed from png_byte to png_uint_32 at version 1.2.0 */ +#ifdef PNG_1_0_X + png_byte mng_features_permitted; +#else + png_uint_32 mng_features_permitted; +#endif /* PNG_1_0_X */ +#endif + +/* New member added in libpng-1.0.7 */ +#if defined(PNG_READ_GAMMA_SUPPORTED) || defined(PNG_READ_BACKGROUND_SUPPORTED) + png_fixed_point int_gamma; +#endif + +/* New member added in libpng-1.0.9, ifdef'ed out in 1.0.12, enabled in 1.2.0 */ +#if defined(PNG_MNG_FEATURES_SUPPORTED) + png_byte filter_type; +#endif + +#if defined(PNG_1_0_X) || (defined(PNG_DEBUG) && defined(PNG_USE_PNGGCCRD)) +/* New member added in libpng-1.0.10, ifdef'ed out in 1.2.0 */ + png_uint_32 row_buf_size; +#endif + +/* New members added in libpng-1.2.0 */ +#if !defined(PNG_1_0_X) && defined(PNG_ASSEMBLER_CODE_SUPPORTED) + png_byte mmx_bitdepth_threshold; + png_uint_32 mmx_rowbytes_threshold; + png_uint_32 asm_flags; +#endif + +/* New members added in libpng-1.0.2 but first enabled by default in 1.2.0 */ +#ifdef PNG_USER_MEM_SUPPORTED + png_voidp mem_ptr; /* user supplied struct for mem functions */ + png_malloc_ptr malloc_fn; /* function for allocating memory */ + png_free_ptr free_fn; /* function for freeing memory */ +#endif + +/* New member added in libpng-1.0.13 and 1.2.0 */ + png_bytep big_row_buf; /* buffer to save current (unfiltered) row */ + +#if defined(PNG_READ_DITHER_SUPPORTED) +/* The following three members were added at version 1.0.14 and 1.2.4 */ + png_bytep dither_sort; /* working sort array */ + png_bytep index_to_palette; /* where the original index currently is */ + /* in the palette */ + png_bytep palette_to_index; /* which original index points to this */ + /* palette color */ +#endif + +/* New members added in libpng-1.0.16 and 1.2.6 */ + png_byte compression_type; + +#ifdef PNG_SET_USER_LIMITS_SUPPORTED + png_uint_32 user_width_max; + png_uint_32 user_height_max; +#endif + +}; + + +/* This triggers a compiler error in png.c, if png.c and png.h + * do not agree upon the version number. + */ +typedef png_structp version_1_2_10; + +typedef png_struct FAR * FAR * png_structpp; + +/* Here are the function definitions most commonly used. This is not + * the place to find out how to use libpng. See libpng.txt for the + * full explanation, see example.c for the summary. This just provides + * a simple one line description of the use of each function. + */ + +/* Returns the version number of the library */ +extern PNG_EXPORT(png_uint_32,png_access_version_number) PNGARG((void)); + +/* Tell lib we have already handled the first magic bytes. + * Handling more than 8 bytes from the beginning of the file is an error. + */ +extern PNG_EXPORT(void,png_set_sig_bytes) PNGARG((png_structp png_ptr, + int num_bytes)); + +/* Check sig[start] through sig[start + num_to_check - 1] to see if it's a + * PNG file. Returns zero if the supplied bytes match the 8-byte PNG + * signature, and non-zero otherwise. Having num_to_check == 0 or + * start > 7 will always fail (ie return non-zero). + */ +extern PNG_EXPORT(int,png_sig_cmp) PNGARG((png_bytep sig, png_size_t start, + png_size_t num_to_check)); + +/* Simple signature checking function. This is the same as calling + * png_check_sig(sig, n) := !png_sig_cmp(sig, 0, n). + */ +extern PNG_EXPORT(int,png_check_sig) PNGARG((png_bytep sig, int num)); + +/* Allocate and initialize png_ptr struct for reading, and any other memory. */ +extern PNG_EXPORT(png_structp,png_create_read_struct) + PNGARG((png_const_charp user_png_ver, png_voidp error_ptr, + png_error_ptr error_fn, png_error_ptr warn_fn)); + +/* Allocate and initialize png_ptr struct for writing, and any other memory */ +extern PNG_EXPORT(png_structp,png_create_write_struct) + PNGARG((png_const_charp user_png_ver, png_voidp error_ptr, + png_error_ptr error_fn, png_error_ptr warn_fn)); + +#ifdef PNG_WRITE_SUPPORTED +extern PNG_EXPORT(png_uint_32,png_get_compression_buffer_size) + PNGARG((png_structp png_ptr)); +#endif + +#ifdef PNG_WRITE_SUPPORTED +extern PNG_EXPORT(void,png_set_compression_buffer_size) + PNGARG((png_structp png_ptr, png_uint_32 size)); +#endif + +/* Reset the compression stream */ +extern PNG_EXPORT(int,png_reset_zstream) PNGARG((png_structp png_ptr)); + +/* New functions added in libpng-1.0.2 (not enabled by default until 1.2.0) */ +#ifdef PNG_USER_MEM_SUPPORTED +extern PNG_EXPORT(png_structp,png_create_read_struct_2) + PNGARG((png_const_charp user_png_ver, png_voidp error_ptr, + png_error_ptr error_fn, png_error_ptr warn_fn, png_voidp mem_ptr, + png_malloc_ptr malloc_fn, png_free_ptr free_fn)); +extern PNG_EXPORT(png_structp,png_create_write_struct_2) + PNGARG((png_const_charp user_png_ver, png_voidp error_ptr, + png_error_ptr error_fn, png_error_ptr warn_fn, png_voidp mem_ptr, + png_malloc_ptr malloc_fn, png_free_ptr free_fn)); +#endif + +/* Write a PNG chunk - size, type, (optional) data, CRC. */ +extern PNG_EXPORT(void,png_write_chunk) PNGARG((png_structp png_ptr, + png_bytep chunk_name, png_bytep data, png_size_t length)); + +/* Write the start of a PNG chunk - length and chunk name. */ +extern PNG_EXPORT(void,png_write_chunk_start) PNGARG((png_structp png_ptr, + png_bytep chunk_name, png_uint_32 length)); + +/* Write the data of a PNG chunk started with png_write_chunk_start(). */ +extern PNG_EXPORT(void,png_write_chunk_data) PNGARG((png_structp png_ptr, + png_bytep data, png_size_t length)); + +/* Finish a chunk started with png_write_chunk_start() (includes CRC). */ +extern PNG_EXPORT(void,png_write_chunk_end) PNGARG((png_structp png_ptr)); + +/* Allocate and initialize the info structure */ +extern PNG_EXPORT(png_infop,png_create_info_struct) + PNGARG((png_structp png_ptr)); + +#if defined(PNG_1_0_X) || defined (PNG_1_2_X) +/* Initialize the info structure (old interface - DEPRECATED) */ +extern PNG_EXPORT(void,png_info_init) PNGARG((png_infop info_ptr)); +#undef png_info_init +#define png_info_init(info_ptr) png_info_init_3(&info_ptr,\ + png_sizeof(png_info)); +#endif + +extern PNG_EXPORT(void,png_info_init_3) PNGARG((png_infopp info_ptr, + png_size_t png_info_struct_size)); + +/* Writes all the PNG information before the image. */ +extern PNG_EXPORT(void,png_write_info_before_PLTE) PNGARG((png_structp png_ptr, + png_infop info_ptr)); +extern PNG_EXPORT(void,png_write_info) PNGARG((png_structp png_ptr, + png_infop info_ptr)); + +#ifndef PNG_NO_SEQUENTIAL_READ_SUPPORTED +/* read the information before the actual image data. */ +extern PNG_EXPORT(void,png_read_info) PNGARG((png_structp png_ptr, + png_infop info_ptr)); +#endif + +#if defined(PNG_TIME_RFC1123_SUPPORTED) +extern PNG_EXPORT(png_charp,png_convert_to_rfc1123) + PNGARG((png_structp png_ptr, png_timep ptime)); +#endif + +#if !defined(_WIN32_WCE) +/* "time.h" functions are not supported on WindowsCE */ +#if defined(PNG_WRITE_tIME_SUPPORTED) +/* convert from a struct tm to png_time */ +extern PNG_EXPORT(void,png_convert_from_struct_tm) PNGARG((png_timep ptime, + struct tm FAR * ttime)); + +/* convert from time_t to png_time. Uses gmtime() */ +extern PNG_EXPORT(void,png_convert_from_time_t) PNGARG((png_timep ptime, + time_t ttime)); +#endif /* PNG_WRITE_tIME_SUPPORTED */ +#endif /* _WIN32_WCE */ + +#if defined(PNG_READ_EXPAND_SUPPORTED) +/* Expand data to 24-bit RGB, or 8-bit grayscale, with alpha if available. */ +extern PNG_EXPORT(void,png_set_expand) PNGARG((png_structp png_ptr)); +#if !defined(PNG_1_0_X) +extern PNG_EXPORT(void,png_set_expand_gray_1_2_4_to_8) PNGARG((png_structp + png_ptr)); +#endif +extern PNG_EXPORT(void,png_set_palette_to_rgb) PNGARG((png_structp png_ptr)); +extern PNG_EXPORT(void,png_set_tRNS_to_alpha) PNGARG((png_structp png_ptr)); +#if defined(PNG_1_0_X) || defined (PNG_1_2_X) +/* Deprecated */ +extern PNG_EXPORT(void,png_set_gray_1_2_4_to_8) PNGARG((png_structp png_ptr)); +#endif +#endif + +#if defined(PNG_READ_BGR_SUPPORTED) || defined(PNG_WRITE_BGR_SUPPORTED) +/* Use blue, green, red order for pixels. */ +extern PNG_EXPORT(void,png_set_bgr) PNGARG((png_structp png_ptr)); +#endif + +#if defined(PNG_READ_GRAY_TO_RGB_SUPPORTED) +/* Expand the grayscale to 24-bit RGB if necessary. */ +extern PNG_EXPORT(void,png_set_gray_to_rgb) PNGARG((png_structp png_ptr)); +#endif + +#if defined(PNG_READ_RGB_TO_GRAY_SUPPORTED) +/* Reduce RGB to grayscale. */ +#ifdef PNG_FLOATING_POINT_SUPPORTED +extern PNG_EXPORT(void,png_set_rgb_to_gray) PNGARG((png_structp png_ptr, + int error_action, double red, double green )); +#endif +extern PNG_EXPORT(void,png_set_rgb_to_gray_fixed) PNGARG((png_structp png_ptr, + int error_action, png_fixed_point red, png_fixed_point green )); +extern PNG_EXPORT(png_byte,png_get_rgb_to_gray_status) PNGARG((png_structp + png_ptr)); +#endif + +extern PNG_EXPORT(void,png_build_grayscale_palette) PNGARG((int bit_depth, + png_colorp palette)); + +#if defined(PNG_READ_STRIP_ALPHA_SUPPORTED) +extern PNG_EXPORT(void,png_set_strip_alpha) PNGARG((png_structp png_ptr)); +#endif + +#if defined(PNG_READ_SWAP_ALPHA_SUPPORTED) || \ + defined(PNG_WRITE_SWAP_ALPHA_SUPPORTED) +extern PNG_EXPORT(void,png_set_swap_alpha) PNGARG((png_structp png_ptr)); +#endif + +#if defined(PNG_READ_INVERT_ALPHA_SUPPORTED) || \ + defined(PNG_WRITE_INVERT_ALPHA_SUPPORTED) +extern PNG_EXPORT(void,png_set_invert_alpha) PNGARG((png_structp png_ptr)); +#endif + +#if defined(PNG_READ_FILLER_SUPPORTED) || defined(PNG_WRITE_FILLER_SUPPORTED) +/* Add a filler byte to 8-bit Gray or 24-bit RGB images. */ +extern PNG_EXPORT(void,png_set_filler) PNGARG((png_structp png_ptr, + png_uint_32 filler, int flags)); +/* The values of the PNG_FILLER_ defines should NOT be changed */ +#define PNG_FILLER_BEFORE 0 +#define PNG_FILLER_AFTER 1 +/* Add an alpha byte to 8-bit Gray or 24-bit RGB images. */ +#if !defined(PNG_1_0_X) +extern PNG_EXPORT(void,png_set_add_alpha) PNGARG((png_structp png_ptr, + png_uint_32 filler, int flags)); +#endif +#endif /* PNG_READ_FILLER_SUPPORTED || PNG_WRITE_FILLER_SUPPORTED */ + +#if defined(PNG_READ_SWAP_SUPPORTED) || defined(PNG_WRITE_SWAP_SUPPORTED) +/* Swap bytes in 16-bit depth files. */ +extern PNG_EXPORT(void,png_set_swap) PNGARG((png_structp png_ptr)); +#endif + +#if defined(PNG_READ_PACK_SUPPORTED) || defined(PNG_WRITE_PACK_SUPPORTED) +/* Use 1 byte per pixel in 1, 2, or 4-bit depth files. */ +extern PNG_EXPORT(void,png_set_packing) PNGARG((png_structp png_ptr)); +#endif + +#if defined(PNG_READ_PACKSWAP_SUPPORTED) || defined(PNG_WRITE_PACKSWAP_SUPPORTED) +/* Swap packing order of pixels in bytes. */ +extern PNG_EXPORT(void,png_set_packswap) PNGARG((png_structp png_ptr)); +#endif + +#if defined(PNG_READ_SHIFT_SUPPORTED) || defined(PNG_WRITE_SHIFT_SUPPORTED) +/* Converts files to legal bit depths. */ +extern PNG_EXPORT(void,png_set_shift) PNGARG((png_structp png_ptr, + png_color_8p true_bits)); +#endif + +#if defined(PNG_READ_INTERLACING_SUPPORTED) || \ + defined(PNG_WRITE_INTERLACING_SUPPORTED) +/* Have the code handle the interlacing. Returns the number of passes. */ +extern PNG_EXPORT(int,png_set_interlace_handling) PNGARG((png_structp png_ptr)); +#endif + +#if defined(PNG_READ_INVERT_SUPPORTED) || defined(PNG_WRITE_INVERT_SUPPORTED) +/* Invert monochrome files */ +extern PNG_EXPORT(void,png_set_invert_mono) PNGARG((png_structp png_ptr)); +#endif + +#if defined(PNG_READ_BACKGROUND_SUPPORTED) +/* Handle alpha and tRNS by replacing with a background color. */ +#ifdef PNG_FLOATING_POINT_SUPPORTED +extern PNG_EXPORT(void,png_set_background) PNGARG((png_structp png_ptr, + png_color_16p background_color, int background_gamma_code, + int need_expand, double background_gamma)); +#endif +#define PNG_BACKGROUND_GAMMA_UNKNOWN 0 +#define PNG_BACKGROUND_GAMMA_SCREEN 1 +#define PNG_BACKGROUND_GAMMA_FILE 2 +#define PNG_BACKGROUND_GAMMA_UNIQUE 3 +#endif + +#if defined(PNG_READ_16_TO_8_SUPPORTED) +/* strip the second byte of information from a 16-bit depth file. */ +extern PNG_EXPORT(void,png_set_strip_16) PNGARG((png_structp png_ptr)); +#endif + +#if defined(PNG_READ_DITHER_SUPPORTED) +/* Turn on dithering, and reduce the palette to the number of colors available. */ +extern PNG_EXPORT(void,png_set_dither) PNGARG((png_structp png_ptr, + png_colorp palette, int num_palette, int maximum_colors, + png_uint_16p histogram, int full_dither)); +#endif + +#if defined(PNG_READ_GAMMA_SUPPORTED) +/* Handle gamma correction. Screen_gamma=(display_exponent) */ +#ifdef PNG_FLOATING_POINT_SUPPORTED +extern PNG_EXPORT(void,png_set_gamma) PNGARG((png_structp png_ptr, + double screen_gamma, double default_file_gamma)); +#endif +#endif + +#if defined(PNG_1_0_X) || defined (PNG_1_2_X) +#if defined(PNG_READ_EMPTY_PLTE_SUPPORTED) || \ + defined(PNG_WRITE_EMPTY_PLTE_SUPPORTED) +/* Permit or disallow empty PLTE (0: not permitted, 1: permitted) */ +/* Deprecated and will be removed. Use png_permit_mng_features() instead. */ +extern PNG_EXPORT(void,png_permit_empty_plte) PNGARG((png_structp png_ptr, + int empty_plte_permitted)); +#endif +#endif + +#if defined(PNG_WRITE_FLUSH_SUPPORTED) +/* Set how many lines between output flushes - 0 for no flushing */ +extern PNG_EXPORT(void,png_set_flush) PNGARG((png_structp png_ptr, int nrows)); +/* Flush the current PNG output buffer */ +extern PNG_EXPORT(void,png_write_flush) PNGARG((png_structp png_ptr)); +#endif + +/* optional update palette with requested transformations */ +extern PNG_EXPORT(void,png_start_read_image) PNGARG((png_structp png_ptr)); + +/* optional call to update the users info structure */ +extern PNG_EXPORT(void,png_read_update_info) PNGARG((png_structp png_ptr, + png_infop info_ptr)); + +#ifndef PNG_NO_SEQUENTIAL_READ_SUPPORTED +/* read one or more rows of image data. */ +extern PNG_EXPORT(void,png_read_rows) PNGARG((png_structp png_ptr, + png_bytepp row, png_bytepp display_row, png_uint_32 num_rows)); +#endif + +#ifndef PNG_NO_SEQUENTIAL_READ_SUPPORTED +/* read a row of data. */ +extern PNG_EXPORT(void,png_read_row) PNGARG((png_structp png_ptr, + png_bytep row, + png_bytep display_row)); +#endif + +#ifndef PNG_NO_SEQUENTIAL_READ_SUPPORTED +/* read the whole image into memory at once. */ +extern PNG_EXPORT(void,png_read_image) PNGARG((png_structp png_ptr, + png_bytepp image)); +#endif + +/* write a row of image data */ +extern PNG_EXPORT(void,png_write_row) PNGARG((png_structp png_ptr, + png_bytep row)); + +/* write a few rows of image data */ +extern PNG_EXPORT(void,png_write_rows) PNGARG((png_structp png_ptr, + png_bytepp row, png_uint_32 num_rows)); + +/* write the image data */ +extern PNG_EXPORT(void,png_write_image) PNGARG((png_structp png_ptr, + png_bytepp image)); + +/* writes the end of the PNG file. */ +extern PNG_EXPORT(void,png_write_end) PNGARG((png_structp png_ptr, + png_infop info_ptr)); + +#ifndef PNG_NO_SEQUENTIAL_READ_SUPPORTED +/* read the end of the PNG file. */ +extern PNG_EXPORT(void,png_read_end) PNGARG((png_structp png_ptr, + png_infop info_ptr)); +#endif + +/* free any memory associated with the png_info_struct */ +extern PNG_EXPORT(void,png_destroy_info_struct) PNGARG((png_structp png_ptr, + png_infopp info_ptr_ptr)); + +/* free any memory associated with the png_struct and the png_info_structs */ +extern PNG_EXPORT(void,png_destroy_read_struct) PNGARG((png_structpp + png_ptr_ptr, png_infopp info_ptr_ptr, png_infopp end_info_ptr_ptr)); + +/* free all memory used by the read (old method - NOT DLL EXPORTED) */ +extern void png_read_destroy PNGARG((png_structp png_ptr, png_infop info_ptr, + png_infop end_info_ptr)); + +/* free any memory associated with the png_struct and the png_info_structs */ +extern PNG_EXPORT(void,png_destroy_write_struct) + PNGARG((png_structpp png_ptr_ptr, png_infopp info_ptr_ptr)); + +/* free any memory used in png_ptr struct (old method - NOT DLL EXPORTED) */ +extern void png_write_destroy PNGARG((png_structp png_ptr)); + +/* set the libpng method of handling chunk CRC errors */ +extern PNG_EXPORT(void,png_set_crc_action) PNGARG((png_structp png_ptr, + int crit_action, int ancil_action)); + +/* Values for png_set_crc_action() to say how to handle CRC errors in + * ancillary and critical chunks, and whether to use the data contained + * therein. Note that it is impossible to "discard" data in a critical + * chunk. For versions prior to 0.90, the action was always error/quit, + * whereas in version 0.90 and later, the action for CRC errors in ancillary + * chunks is warn/discard. These values should NOT be changed. + * + * value action:critical action:ancillary + */ +#define PNG_CRC_DEFAULT 0 /* error/quit warn/discard data */ +#define PNG_CRC_ERROR_QUIT 1 /* error/quit error/quit */ +#define PNG_CRC_WARN_DISCARD 2 /* (INVALID) warn/discard data */ +#define PNG_CRC_WARN_USE 3 /* warn/use data warn/use data */ +#define PNG_CRC_QUIET_USE 4 /* quiet/use data quiet/use data */ +#define PNG_CRC_NO_CHANGE 5 /* use current value use current value */ + +/* These functions give the user control over the scan-line filtering in + * libpng and the compression methods used by zlib. These functions are + * mainly useful for testing, as the defaults should work with most users. + * Those users who are tight on memory or want faster performance at the + * expense of compression can modify them. See the compression library + * header file (zlib.h) for an explination of the compression functions. + */ + +/* set the filtering method(s) used by libpng. Currently, the only valid + * value for "method" is 0. + */ +extern PNG_EXPORT(void,png_set_filter) PNGARG((png_structp png_ptr, int method, + int filters)); + +/* Flags for png_set_filter() to say which filters to use. The flags + * are chosen so that they don't conflict with real filter types + * below, in case they are supplied instead of the #defined constants. + * These values should NOT be changed. + */ +#define PNG_NO_FILTERS 0x00 +#define PNG_FILTER_NONE 0x08 +#define PNG_FILTER_SUB 0x10 +#define PNG_FILTER_UP 0x20 +#define PNG_FILTER_AVG 0x40 +#define PNG_FILTER_PAETH 0x80 +#define PNG_ALL_FILTERS (PNG_FILTER_NONE | PNG_FILTER_SUB | PNG_FILTER_UP | \ + PNG_FILTER_AVG | PNG_FILTER_PAETH) + +/* Filter values (not flags) - used in pngwrite.c, pngwutil.c for now. + * These defines should NOT be changed. + */ +#define PNG_FILTER_VALUE_NONE 0 +#define PNG_FILTER_VALUE_SUB 1 +#define PNG_FILTER_VALUE_UP 2 +#define PNG_FILTER_VALUE_AVG 3 +#define PNG_FILTER_VALUE_PAETH 4 +#define PNG_FILTER_VALUE_LAST 5 + +#if defined(PNG_WRITE_WEIGHTED_FILTER_SUPPORTED) /* EXPERIMENTAL */ +/* The "heuristic_method" is given by one of the PNG_FILTER_HEURISTIC_ + * defines, either the default (minimum-sum-of-absolute-differences), or + * the experimental method (weighted-minimum-sum-of-absolute-differences). + * + * Weights are factors >= 1.0, indicating how important it is to keep the + * filter type consistent between rows. Larger numbers mean the current + * filter is that many times as likely to be the same as the "num_weights" + * previous filters. This is cumulative for each previous row with a weight. + * There needs to be "num_weights" values in "filter_weights", or it can be + * NULL if the weights aren't being specified. Weights have no influence on + * the selection of the first row filter. Well chosen weights can (in theory) + * improve the compression for a given image. + * + * Costs are factors >= 1.0 indicating the relative decoding costs of a + * filter type. Higher costs indicate more decoding expense, and are + * therefore less likely to be selected over a filter with lower computational + * costs. There needs to be a value in "filter_costs" for each valid filter + * type (given by PNG_FILTER_VALUE_LAST), or it can be NULL if you aren't + * setting the costs. Costs try to improve the speed of decompression without + * unduly increasing the compressed image size. + * + * A negative weight or cost indicates the default value is to be used, and + * values in the range [0.0, 1.0) indicate the value is to remain unchanged. + * The default values for both weights and costs are currently 1.0, but may + * change if good general weighting/cost heuristics can be found. If both + * the weights and costs are set to 1.0, this degenerates the WEIGHTED method + * to the UNWEIGHTED method, but with added encoding time/computation. + */ +#ifdef PNG_FLOATING_POINT_SUPPORTED +extern PNG_EXPORT(void,png_set_filter_heuristics) PNGARG((png_structp png_ptr, + int heuristic_method, int num_weights, png_doublep filter_weights, + png_doublep filter_costs)); +#endif +#endif /* PNG_WRITE_WEIGHTED_FILTER_SUPPORTED */ + +/* Heuristic used for row filter selection. These defines should NOT be + * changed. + */ +#define PNG_FILTER_HEURISTIC_DEFAULT 0 /* Currently "UNWEIGHTED" */ +#define PNG_FILTER_HEURISTIC_UNWEIGHTED 1 /* Used by libpng < 0.95 */ +#define PNG_FILTER_HEURISTIC_WEIGHTED 2 /* Experimental feature */ +#define PNG_FILTER_HEURISTIC_LAST 3 /* Not a valid value */ + +/* Set the library compression level. Currently, valid values range from + * 0 - 9, corresponding directly to the zlib compression levels 0 - 9 + * (0 - no compression, 9 - "maximal" compression). Note that tests have + * shown that zlib compression levels 3-6 usually perform as well as level 9 + * for PNG images, and do considerably fewer caclulations. In the future, + * these values may not correspond directly to the zlib compression levels. + */ +extern PNG_EXPORT(void,png_set_compression_level) PNGARG((png_structp png_ptr, + int level)); + +extern PNG_EXPORT(void,png_set_compression_mem_level) + PNGARG((png_structp png_ptr, int mem_level)); + +extern PNG_EXPORT(void,png_set_compression_strategy) + PNGARG((png_structp png_ptr, int strategy)); + +extern PNG_EXPORT(void,png_set_compression_window_bits) + PNGARG((png_structp png_ptr, int window_bits)); + +extern PNG_EXPORT(void,png_set_compression_method) PNGARG((png_structp png_ptr, + int method)); + +/* These next functions are called for input/output, memory, and error + * handling. They are in the file pngrio.c, pngwio.c, and pngerror.c, + * and call standard C I/O routines such as fread(), fwrite(), and + * fprintf(). These functions can be made to use other I/O routines + * at run time for those applications that need to handle I/O in a + * different manner by calling png_set_???_fn(). See libpng.txt for + * more information. + */ + +#if !defined(PNG_NO_STDIO) +/* Initialize the input/output for the PNG file to the default functions. */ +extern PNG_EXPORT(void,png_init_io) PNGARG((png_structp png_ptr, png_FILE_p fp)); +#endif + +/* Replace the (error and abort), and warning functions with user + * supplied functions. If no messages are to be printed you must still + * write and use replacement functions. The replacement error_fn should + * still do a longjmp to the last setjmp location if you are using this + * method of error handling. If error_fn or warning_fn is NULL, the + * default function will be used. + */ + +extern PNG_EXPORT(void,png_set_error_fn) PNGARG((png_structp png_ptr, + png_voidp error_ptr, png_error_ptr error_fn, png_error_ptr warning_fn)); + +/* Return the user pointer associated with the error functions */ +extern PNG_EXPORT(png_voidp,png_get_error_ptr) PNGARG((png_structp png_ptr)); + +/* Replace the default data output functions with a user supplied one(s). + * If buffered output is not used, then output_flush_fn can be set to NULL. + * If PNG_WRITE_FLUSH_SUPPORTED is not defined at libpng compile time + * output_flush_fn will be ignored (and thus can be NULL). + */ +extern PNG_EXPORT(void,png_set_write_fn) PNGARG((png_structp png_ptr, + png_voidp io_ptr, png_rw_ptr write_data_fn, png_flush_ptr output_flush_fn)); + +/* Replace the default data input function with a user supplied one. */ +extern PNG_EXPORT(void,png_set_read_fn) PNGARG((png_structp png_ptr, + png_voidp io_ptr, png_rw_ptr read_data_fn)); + +/* Return the user pointer associated with the I/O functions */ +extern PNG_EXPORT(png_voidp,png_get_io_ptr) PNGARG((png_structp png_ptr)); + +extern PNG_EXPORT(void,png_set_read_status_fn) PNGARG((png_structp png_ptr, + png_read_status_ptr read_row_fn)); + +extern PNG_EXPORT(void,png_set_write_status_fn) PNGARG((png_structp png_ptr, + png_write_status_ptr write_row_fn)); + +#ifdef PNG_USER_MEM_SUPPORTED +/* Replace the default memory allocation functions with user supplied one(s). */ +extern PNG_EXPORT(void,png_set_mem_fn) PNGARG((png_structp png_ptr, + png_voidp mem_ptr, png_malloc_ptr malloc_fn, png_free_ptr free_fn)); +/* Return the user pointer associated with the memory functions */ +extern PNG_EXPORT(png_voidp,png_get_mem_ptr) PNGARG((png_structp png_ptr)); +#endif + +#if defined(PNG_READ_USER_TRANSFORM_SUPPORTED) || \ + defined(PNG_LEGACY_SUPPORTED) +extern PNG_EXPORT(void,png_set_read_user_transform_fn) PNGARG((png_structp + png_ptr, png_user_transform_ptr read_user_transform_fn)); +#endif + +#if defined(PNG_WRITE_USER_TRANSFORM_SUPPORTED) || \ + defined(PNG_LEGACY_SUPPORTED) +extern PNG_EXPORT(void,png_set_write_user_transform_fn) PNGARG((png_structp + png_ptr, png_user_transform_ptr write_user_transform_fn)); +#endif + +#if defined(PNG_READ_USER_TRANSFORM_SUPPORTED) || \ + defined(PNG_WRITE_USER_TRANSFORM_SUPPORTED) || \ + defined(PNG_LEGACY_SUPPORTED) +extern PNG_EXPORT(void,png_set_user_transform_info) PNGARG((png_structp + png_ptr, png_voidp user_transform_ptr, int user_transform_depth, + int user_transform_channels)); +/* Return the user pointer associated with the user transform functions */ +extern PNG_EXPORT(png_voidp,png_get_user_transform_ptr) + PNGARG((png_structp png_ptr)); +#endif + +#ifdef PNG_USER_CHUNKS_SUPPORTED +extern PNG_EXPORT(void,png_set_read_user_chunk_fn) PNGARG((png_structp png_ptr, + png_voidp user_chunk_ptr, png_user_chunk_ptr read_user_chunk_fn)); +extern PNG_EXPORT(png_voidp,png_get_user_chunk_ptr) PNGARG((png_structp + png_ptr)); +#endif + +#ifdef PNG_PROGRESSIVE_READ_SUPPORTED +/* Sets the function callbacks for the push reader, and a pointer to a + * user-defined structure available to the callback functions. + */ +extern PNG_EXPORT(void,png_set_progressive_read_fn) PNGARG((png_structp png_ptr, + png_voidp progressive_ptr, + png_progressive_info_ptr info_fn, png_progressive_row_ptr row_fn, + png_progressive_end_ptr end_fn)); + +/* returns the user pointer associated with the push read functions */ +extern PNG_EXPORT(png_voidp,png_get_progressive_ptr) + PNGARG((png_structp png_ptr)); + +/* function to be called when data becomes available */ +extern PNG_EXPORT(void,png_process_data) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_bytep buffer, png_size_t buffer_size)); + +/* function that combines rows. Not very much different than the + * png_combine_row() call. Is this even used????? + */ +extern PNG_EXPORT(void,png_progressive_combine_row) PNGARG((png_structp png_ptr, + png_bytep old_row, png_bytep new_row)); +#endif /* PNG_PROGRESSIVE_READ_SUPPORTED */ + +extern PNG_EXPORT(png_voidp,png_malloc) PNGARG((png_structp png_ptr, + png_uint_32 size)); + +#if defined(PNG_1_0_X) +# define png_malloc_warn png_malloc +#else +/* Added at libpng version 1.2.4 */ +extern PNG_EXPORT(png_voidp,png_malloc_warn) PNGARG((png_structp png_ptr, + png_uint_32 size)); +#endif + +/* frees a pointer allocated by png_malloc() */ +extern PNG_EXPORT(void,png_free) PNGARG((png_structp png_ptr, png_voidp ptr)); + +#if defined(PNG_1_0_X) +/* Function to allocate memory for zlib. */ +extern PNG_EXPORT(voidpf,png_zalloc) PNGARG((voidpf png_ptr, uInt items, + uInt size)); + +/* Function to free memory for zlib */ +extern PNG_EXPORT(void,png_zfree) PNGARG((voidpf png_ptr, voidpf ptr)); +#endif + +/* Free data that was allocated internally */ +extern PNG_EXPORT(void,png_free_data) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_uint_32 free_me, int num)); +#ifdef PNG_FREE_ME_SUPPORTED +/* Reassign responsibility for freeing existing data, whether allocated + * by libpng or by the application */ +extern PNG_EXPORT(void,png_data_freer) PNGARG((png_structp png_ptr, + png_infop info_ptr, int freer, png_uint_32 mask)); +#endif +/* assignments for png_data_freer */ +#define PNG_DESTROY_WILL_FREE_DATA 1 +#define PNG_SET_WILL_FREE_DATA 1 +#define PNG_USER_WILL_FREE_DATA 2 +/* Flags for png_ptr->free_me and info_ptr->free_me */ +#define PNG_FREE_HIST 0x0008 +#define PNG_FREE_ICCP 0x0010 +#define PNG_FREE_SPLT 0x0020 +#define PNG_FREE_ROWS 0x0040 +#define PNG_FREE_PCAL 0x0080 +#define PNG_FREE_SCAL 0x0100 +#define PNG_FREE_UNKN 0x0200 +#define PNG_FREE_LIST 0x0400 +#define PNG_FREE_PLTE 0x1000 +#define PNG_FREE_TRNS 0x2000 +#define PNG_FREE_TEXT 0x4000 +#define PNG_FREE_ALL 0x7fff +#define PNG_FREE_MUL 0x4220 /* PNG_FREE_SPLT|PNG_FREE_TEXT|PNG_FREE_UNKN */ + +#ifdef PNG_USER_MEM_SUPPORTED +extern PNG_EXPORT(png_voidp,png_malloc_default) PNGARG((png_structp png_ptr, + png_uint_32 size)); +extern PNG_EXPORT(void,png_free_default) PNGARG((png_structp png_ptr, + png_voidp ptr)); +#endif + +extern PNG_EXPORT(png_voidp,png_memcpy_check) PNGARG((png_structp png_ptr, + png_voidp s1, png_voidp s2, png_uint_32 size)); + +extern PNG_EXPORT(png_voidp,png_memset_check) PNGARG((png_structp png_ptr, + png_voidp s1, int value, png_uint_32 size)); + +#if defined(USE_FAR_KEYWORD) /* memory model conversion function */ +extern void *png_far_to_near PNGARG((png_structp png_ptr,png_voidp ptr, + int check)); +#endif /* USE_FAR_KEYWORD */ + +/* Fatal error in PNG image of libpng - can't continue */ +extern PNG_EXPORT(void,png_error) PNGARG((png_structp png_ptr, + png_const_charp error_message)); + +/* The same, but the chunk name is prepended to the error string. */ +extern PNG_EXPORT(void,png_chunk_error) PNGARG((png_structp png_ptr, + png_const_charp error_message)); + +/* Non-fatal error in libpng. Can continue, but may have a problem. */ +extern PNG_EXPORT(void,png_warning) PNGARG((png_structp png_ptr, + png_const_charp warning_message)); + +/* Non-fatal error in libpng, chunk name is prepended to message. */ +extern PNG_EXPORT(void,png_chunk_warning) PNGARG((png_structp png_ptr, + png_const_charp warning_message)); + +/* The png_set_ functions are for storing values in the png_info_struct. + * Similarly, the png_get_ calls are used to read values from the + * png_info_struct, either storing the parameters in the passed variables, or + * setting pointers into the png_info_struct where the data is stored. The + * png_get_ functions return a non-zero value if the data was available + * in info_ptr, or return zero and do not change any of the parameters if the + * data was not available. + * + * These functions should be used instead of directly accessing png_info + * to avoid problems with future changes in the size and internal layout of + * png_info_struct. + */ +/* Returns "flag" if chunk data is valid in info_ptr. */ +extern PNG_EXPORT(png_uint_32,png_get_valid) PNGARG((png_structp png_ptr, +png_infop info_ptr, png_uint_32 flag)); + +/* Returns number of bytes needed to hold a transformed row. */ +extern PNG_EXPORT(png_uint_32,png_get_rowbytes) PNGARG((png_structp png_ptr, +png_infop info_ptr)); + +#if defined(PNG_INFO_IMAGE_SUPPORTED) +/* Returns row_pointers, which is an array of pointers to scanlines that was +returned from png_read_png(). */ +extern PNG_EXPORT(png_bytepp,png_get_rows) PNGARG((png_structp png_ptr, +png_infop info_ptr)); +/* Set row_pointers, which is an array of pointers to scanlines for use +by png_write_png(). */ +extern PNG_EXPORT(void,png_set_rows) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_bytepp row_pointers)); +#endif + +/* Returns number of color channels in image. */ +extern PNG_EXPORT(png_byte,png_get_channels) PNGARG((png_structp png_ptr, +png_infop info_ptr)); + +#ifdef PNG_EASY_ACCESS_SUPPORTED +/* Returns image width in pixels. */ +extern PNG_EXPORT(png_uint_32, png_get_image_width) PNGARG((png_structp +png_ptr, png_infop info_ptr)); + +/* Returns image height in pixels. */ +extern PNG_EXPORT(png_uint_32, png_get_image_height) PNGARG((png_structp +png_ptr, png_infop info_ptr)); + +/* Returns image bit_depth. */ +extern PNG_EXPORT(png_byte, png_get_bit_depth) PNGARG((png_structp +png_ptr, png_infop info_ptr)); + +/* Returns image color_type. */ +extern PNG_EXPORT(png_byte, png_get_color_type) PNGARG((png_structp +png_ptr, png_infop info_ptr)); + +/* Returns image filter_type. */ +extern PNG_EXPORT(png_byte, png_get_filter_type) PNGARG((png_structp +png_ptr, png_infop info_ptr)); + +/* Returns image interlace_type. */ +extern PNG_EXPORT(png_byte, png_get_interlace_type) PNGARG((png_structp +png_ptr, png_infop info_ptr)); + +/* Returns image compression_type. */ +extern PNG_EXPORT(png_byte, png_get_compression_type) PNGARG((png_structp +png_ptr, png_infop info_ptr)); + +/* Returns image resolution in pixels per meter, from pHYs chunk data. */ +extern PNG_EXPORT(png_uint_32, png_get_pixels_per_meter) PNGARG((png_structp +png_ptr, png_infop info_ptr)); +extern PNG_EXPORT(png_uint_32, png_get_x_pixels_per_meter) PNGARG((png_structp +png_ptr, png_infop info_ptr)); +extern PNG_EXPORT(png_uint_32, png_get_y_pixels_per_meter) PNGARG((png_structp +png_ptr, png_infop info_ptr)); + +/* Returns pixel aspect ratio, computed from pHYs chunk data. */ +#ifdef PNG_FLOATING_POINT_SUPPORTED +extern PNG_EXPORT(float, png_get_pixel_aspect_ratio) PNGARG((png_structp +png_ptr, png_infop info_ptr)); +#endif + +/* Returns image x, y offset in pixels or microns, from oFFs chunk data. */ +extern PNG_EXPORT(png_int_32, png_get_x_offset_pixels) PNGARG((png_structp +png_ptr, png_infop info_ptr)); +extern PNG_EXPORT(png_int_32, png_get_y_offset_pixels) PNGARG((png_structp +png_ptr, png_infop info_ptr)); +extern PNG_EXPORT(png_int_32, png_get_x_offset_microns) PNGARG((png_structp +png_ptr, png_infop info_ptr)); +extern PNG_EXPORT(png_int_32, png_get_y_offset_microns) PNGARG((png_structp +png_ptr, png_infop info_ptr)); + +#endif /* PNG_EASY_ACCESS_SUPPORTED */ + +/* Returns pointer to signature string read from PNG header */ +extern PNG_EXPORT(png_bytep,png_get_signature) PNGARG((png_structp png_ptr, +png_infop info_ptr)); + +#if defined(PNG_bKGD_SUPPORTED) +extern PNG_EXPORT(png_uint_32,png_get_bKGD) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_color_16p *background)); +#endif + +#if defined(PNG_bKGD_SUPPORTED) +extern PNG_EXPORT(void,png_set_bKGD) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_color_16p background)); +#endif + +#if defined(PNG_cHRM_SUPPORTED) +#ifdef PNG_FLOATING_POINT_SUPPORTED +extern PNG_EXPORT(png_uint_32,png_get_cHRM) PNGARG((png_structp png_ptr, + png_infop info_ptr, double *white_x, double *white_y, double *red_x, + double *red_y, double *green_x, double *green_y, double *blue_x, + double *blue_y)); +#endif +#ifdef PNG_FIXED_POINT_SUPPORTED +extern PNG_EXPORT(png_uint_32,png_get_cHRM_fixed) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_fixed_point *int_white_x, png_fixed_point + *int_white_y, png_fixed_point *int_red_x, png_fixed_point *int_red_y, + png_fixed_point *int_green_x, png_fixed_point *int_green_y, png_fixed_point + *int_blue_x, png_fixed_point *int_blue_y)); +#endif +#endif + +#if defined(PNG_cHRM_SUPPORTED) +#ifdef PNG_FLOATING_POINT_SUPPORTED +extern PNG_EXPORT(void,png_set_cHRM) PNGARG((png_structp png_ptr, + png_infop info_ptr, double white_x, double white_y, double red_x, + double red_y, double green_x, double green_y, double blue_x, double blue_y)); +#endif +#ifdef PNG_FIXED_POINT_SUPPORTED +extern PNG_EXPORT(void,png_set_cHRM_fixed) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_fixed_point int_white_x, png_fixed_point int_white_y, + png_fixed_point int_red_x, png_fixed_point int_red_y, png_fixed_point + int_green_x, png_fixed_point int_green_y, png_fixed_point int_blue_x, + png_fixed_point int_blue_y)); +#endif +#endif + +#if defined(PNG_gAMA_SUPPORTED) +#ifdef PNG_FLOATING_POINT_SUPPORTED +extern PNG_EXPORT(png_uint_32,png_get_gAMA) PNGARG((png_structp png_ptr, + png_infop info_ptr, double *file_gamma)); +#endif +extern PNG_EXPORT(png_uint_32,png_get_gAMA_fixed) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_fixed_point *int_file_gamma)); +#endif + +#if defined(PNG_gAMA_SUPPORTED) +#ifdef PNG_FLOATING_POINT_SUPPORTED +extern PNG_EXPORT(void,png_set_gAMA) PNGARG((png_structp png_ptr, + png_infop info_ptr, double file_gamma)); +#endif +extern PNG_EXPORT(void,png_set_gAMA_fixed) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_fixed_point int_file_gamma)); +#endif + +#if defined(PNG_hIST_SUPPORTED) +extern PNG_EXPORT(png_uint_32,png_get_hIST) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_uint_16p *hist)); +#endif + +#if defined(PNG_hIST_SUPPORTED) +extern PNG_EXPORT(void,png_set_hIST) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_uint_16p hist)); +#endif + +extern PNG_EXPORT(png_uint_32,png_get_IHDR) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_uint_32 *width, png_uint_32 *height, + int *bit_depth, int *color_type, int *interlace_method, + int *compression_method, int *filter_method)); + +extern PNG_EXPORT(void,png_set_IHDR) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_uint_32 width, png_uint_32 height, int bit_depth, + int color_type, int interlace_method, int compression_method, + int filter_method)); + +#if defined(PNG_oFFs_SUPPORTED) +extern PNG_EXPORT(png_uint_32,png_get_oFFs) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_int_32 *offset_x, png_int_32 *offset_y, + int *unit_type)); +#endif + +#if defined(PNG_oFFs_SUPPORTED) +extern PNG_EXPORT(void,png_set_oFFs) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_int_32 offset_x, png_int_32 offset_y, + int unit_type)); +#endif + +#if defined(PNG_pCAL_SUPPORTED) +extern PNG_EXPORT(png_uint_32,png_get_pCAL) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_charp *purpose, png_int_32 *X0, png_int_32 *X1, + int *type, int *nparams, png_charp *units, png_charpp *params)); +#endif + +#if defined(PNG_pCAL_SUPPORTED) +extern PNG_EXPORT(void,png_set_pCAL) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_charp purpose, png_int_32 X0, png_int_32 X1, + int type, int nparams, png_charp units, png_charpp params)); +#endif + +#if defined(PNG_pHYs_SUPPORTED) +extern PNG_EXPORT(png_uint_32,png_get_pHYs) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_uint_32 *res_x, png_uint_32 *res_y, int *unit_type)); +#endif + +#if defined(PNG_pHYs_SUPPORTED) +extern PNG_EXPORT(void,png_set_pHYs) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_uint_32 res_x, png_uint_32 res_y, int unit_type)); +#endif + +extern PNG_EXPORT(png_uint_32,png_get_PLTE) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_colorp *palette, int *num_palette)); + +extern PNG_EXPORT(void,png_set_PLTE) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_colorp palette, int num_palette)); + +#if defined(PNG_sBIT_SUPPORTED) +extern PNG_EXPORT(png_uint_32,png_get_sBIT) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_color_8p *sig_bit)); +#endif + +#if defined(PNG_sBIT_SUPPORTED) +extern PNG_EXPORT(void,png_set_sBIT) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_color_8p sig_bit)); +#endif + +#if defined(PNG_sRGB_SUPPORTED) +extern PNG_EXPORT(png_uint_32,png_get_sRGB) PNGARG((png_structp png_ptr, + png_infop info_ptr, int *intent)); +#endif + +#if defined(PNG_sRGB_SUPPORTED) +extern PNG_EXPORT(void,png_set_sRGB) PNGARG((png_structp png_ptr, + png_infop info_ptr, int intent)); +extern PNG_EXPORT(void,png_set_sRGB_gAMA_and_cHRM) PNGARG((png_structp png_ptr, + png_infop info_ptr, int intent)); +#endif + +#if defined(PNG_iCCP_SUPPORTED) +extern PNG_EXPORT(png_uint_32,png_get_iCCP) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_charpp name, int *compression_type, + png_charpp profile, png_uint_32 *proflen)); + /* Note to maintainer: profile should be png_bytepp */ +#endif + +#if defined(PNG_iCCP_SUPPORTED) +extern PNG_EXPORT(void,png_set_iCCP) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_charp name, int compression_type, + png_charp profile, png_uint_32 proflen)); + /* Note to maintainer: profile should be png_bytep */ +#endif + +#if defined(PNG_sPLT_SUPPORTED) +extern PNG_EXPORT(png_uint_32,png_get_sPLT) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_sPLT_tpp entries)); +#endif + +#if defined(PNG_sPLT_SUPPORTED) +extern PNG_EXPORT(void,png_set_sPLT) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_sPLT_tp entries, int nentries)); +#endif + +#if defined(PNG_TEXT_SUPPORTED) +/* png_get_text also returns the number of text chunks in *num_text */ +extern PNG_EXPORT(png_uint_32,png_get_text) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_textp *text_ptr, int *num_text)); +#endif + +/* + * Note while png_set_text() will accept a structure whose text, + * language, and translated keywords are NULL pointers, the structure + * returned by png_get_text will always contain regular + * zero-terminated C strings. They might be empty strings but + * they will never be NULL pointers. + */ + +#if defined(PNG_TEXT_SUPPORTED) +extern PNG_EXPORT(void,png_set_text) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_textp text_ptr, int num_text)); +#endif + +#if defined(PNG_tIME_SUPPORTED) +extern PNG_EXPORT(png_uint_32,png_get_tIME) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_timep *mod_time)); +#endif + +#if defined(PNG_tIME_SUPPORTED) +extern PNG_EXPORT(void,png_set_tIME) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_timep mod_time)); +#endif + +#if defined(PNG_tRNS_SUPPORTED) +extern PNG_EXPORT(png_uint_32,png_get_tRNS) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_bytep *trans, int *num_trans, + png_color_16p *trans_values)); +#endif + +#if defined(PNG_tRNS_SUPPORTED) +extern PNG_EXPORT(void,png_set_tRNS) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_bytep trans, int num_trans, + png_color_16p trans_values)); +#endif + +#if defined(PNG_tRNS_SUPPORTED) +#endif + +#if defined(PNG_sCAL_SUPPORTED) +#ifdef PNG_FLOATING_POINT_SUPPORTED +extern PNG_EXPORT(png_uint_32,png_get_sCAL) PNGARG((png_structp png_ptr, + png_infop info_ptr, int *unit, double *width, double *height)); +#else +#ifdef PNG_FIXED_POINT_SUPPORTED +extern PNG_EXPORT(png_uint_32,png_get_sCAL_s) PNGARG((png_structp png_ptr, + png_infop info_ptr, int *unit, png_charpp swidth, png_charpp sheight)); +#endif +#endif +#endif /* PNG_sCAL_SUPPORTED */ + +#if defined(PNG_sCAL_SUPPORTED) +#ifdef PNG_FLOATING_POINT_SUPPORTED +extern PNG_EXPORT(void,png_set_sCAL) PNGARG((png_structp png_ptr, + png_infop info_ptr, int unit, double width, double height)); +#else +#ifdef PNG_FIXED_POINT_SUPPORTED +extern PNG_EXPORT(void,png_set_sCAL_s) PNGARG((png_structp png_ptr, + png_infop info_ptr, int unit, png_charp swidth, png_charp sheight)); +#endif +#endif +#endif /* PNG_sCAL_SUPPORTED || PNG_WRITE_sCAL_SUPPORTED */ + +#if defined(PNG_UNKNOWN_CHUNKS_SUPPORTED) +/* provide a list of chunks and how they are to be handled, if the built-in + handling or default unknown chunk handling is not desired. Any chunks not + listed will be handled in the default manner. The IHDR and IEND chunks + must not be listed. + keep = 0: follow default behavour + = 1: do not keep + = 2: keep only if safe-to-copy + = 3: keep even if unsafe-to-copy +*/ +extern PNG_EXPORT(void, png_set_keep_unknown_chunks) PNGARG((png_structp + png_ptr, int keep, png_bytep chunk_list, int num_chunks)); +extern PNG_EXPORT(void, png_set_unknown_chunks) PNGARG((png_structp png_ptr, + png_infop info_ptr, png_unknown_chunkp unknowns, int num_unknowns)); +extern PNG_EXPORT(void, png_set_unknown_chunk_location) + PNGARG((png_structp png_ptr, png_infop info_ptr, int chunk, int location)); +extern PNG_EXPORT(png_uint_32,png_get_unknown_chunks) PNGARG((png_structp + png_ptr, png_infop info_ptr, png_unknown_chunkpp entries)); +#endif +#ifdef PNG_HANDLE_AS_UNKNOWN_SUPPORTED +PNG_EXPORT(int,png_handle_as_unknown) PNGARG((png_structp png_ptr, png_bytep + chunk_name)); +#endif + +/* Png_free_data() will turn off the "valid" flag for anything it frees. + If you need to turn it off for a chunk that your application has freed, + you can use png_set_invalid(png_ptr, info_ptr, PNG_INFO_CHNK); */ +extern PNG_EXPORT(void, png_set_invalid) PNGARG((png_structp png_ptr, + png_infop info_ptr, int mask)); + +#if defined(PNG_INFO_IMAGE_SUPPORTED) +/* The "params" pointer is currently not used and is for future expansion. */ +extern PNG_EXPORT(void, png_read_png) PNGARG((png_structp png_ptr, + png_infop info_ptr, + int transforms, + png_voidp params)); +extern PNG_EXPORT(void, png_write_png) PNGARG((png_structp png_ptr, + png_infop info_ptr, + int transforms, + png_voidp params)); +#endif + +/* Define PNG_DEBUG at compile time for debugging information. Higher + * numbers for PNG_DEBUG mean more debugging information. This has + * only been added since version 0.95 so it is not implemented throughout + * libpng yet, but more support will be added as needed. + */ +#ifdef PNG_DEBUG +#if (PNG_DEBUG > 0) +#if !defined(PNG_DEBUG_FILE) && defined(_MSC_VER) +#include +#if (PNG_DEBUG > 1) +#define png_debug(l,m) _RPT0(_CRT_WARN,m) +#define png_debug1(l,m,p1) _RPT1(_CRT_WARN,m,p1) +#define png_debug2(l,m,p1,p2) _RPT2(_CRT_WARN,m,p1,p2) +#endif +#else /* PNG_DEBUG_FILE || !_MSC_VER */ +#ifndef PNG_DEBUG_FILE +#define PNG_DEBUG_FILE stderr +#endif /* PNG_DEBUG_FILE */ +#if (PNG_DEBUG > 1) +#define png_debug(l,m) \ +{ \ + int num_tabs=l; \ + fprintf(PNG_DEBUG_FILE,"%s"m,(num_tabs==1 ? "\t" : \ + (num_tabs==2 ? "\t\t":(num_tabs>2 ? "\t\t\t":"")))); \ +} +#define png_debug1(l,m,p1) \ +{ \ + int num_tabs=l; \ + fprintf(PNG_DEBUG_FILE,"%s"m,(num_tabs==1 ? "\t" : \ + (num_tabs==2 ? "\t\t":(num_tabs>2 ? "\t\t\t":""))),p1); \ +} +#define png_debug2(l,m,p1,p2) \ +{ \ + int num_tabs=l; \ + fprintf(PNG_DEBUG_FILE,"%s"m,(num_tabs==1 ? "\t" : \ + (num_tabs==2 ? "\t\t":(num_tabs>2 ? "\t\t\t":""))),p1,p2); \ +} +#endif /* (PNG_DEBUG > 1) */ +#endif /* _MSC_VER */ +#endif /* (PNG_DEBUG > 0) */ +#endif /* PNG_DEBUG */ +#ifndef png_debug +#define png_debug(l, m) +#endif +#ifndef png_debug1 +#define png_debug1(l, m, p1) +#endif +#ifndef png_debug2 +#define png_debug2(l, m, p1, p2) +#endif + +#if 0 +extern PNG_EXPORT(png_bytep,png_sig_bytes) PNGARG((void)); +#endif + +extern PNG_EXPORT(png_charp,png_get_copyright) PNGARG((png_structp png_ptr)); +extern PNG_EXPORT(png_charp,png_get_header_ver) PNGARG((png_structp png_ptr)); +extern PNG_EXPORT(png_charp,png_get_header_version) PNGARG((png_structp png_ptr)); +extern PNG_EXPORT(png_charp,png_get_libpng_ver) PNGARG((png_structp png_ptr)); + +#ifdef PNG_MNG_FEATURES_SUPPORTED +extern PNG_EXPORT(png_uint_32,png_permit_mng_features) PNGARG((png_structp + png_ptr, png_uint_32 mng_features_permitted)); +#endif + +/* For use in png_set_keep_unknown, added to version 1.2.6 */ +#define PNG_HANDLE_CHUNK_AS_DEFAULT 0 +#define PNG_HANDLE_CHUNK_NEVER 1 +#define PNG_HANDLE_CHUNK_IF_SAFE 2 +#define PNG_HANDLE_CHUNK_ALWAYS 3 + +/* Added to version 1.2.0 */ +#if defined(PNG_ASSEMBLER_CODE_SUPPORTED) +#define PNG_ASM_FLAG_MMX_SUPPORT_COMPILED 0x01 /* not user-settable */ +#define PNG_ASM_FLAG_MMX_SUPPORT_IN_CPU 0x02 /* not user-settable */ +#define PNG_ASM_FLAG_MMX_READ_COMBINE_ROW 0x04 +#define PNG_ASM_FLAG_MMX_READ_INTERLACE 0x08 +#define PNG_ASM_FLAG_MMX_READ_FILTER_SUB 0x10 +#define PNG_ASM_FLAG_MMX_READ_FILTER_UP 0x20 +#define PNG_ASM_FLAG_MMX_READ_FILTER_AVG 0x40 +#define PNG_ASM_FLAG_MMX_READ_FILTER_PAETH 0x80 +#define PNG_ASM_FLAGS_INITIALIZED 0x80000000 /* not user-settable */ + +#define PNG_MMX_READ_FLAGS ( PNG_ASM_FLAG_MMX_READ_COMBINE_ROW \ + | PNG_ASM_FLAG_MMX_READ_INTERLACE \ + | PNG_ASM_FLAG_MMX_READ_FILTER_SUB \ + | PNG_ASM_FLAG_MMX_READ_FILTER_UP \ + | PNG_ASM_FLAG_MMX_READ_FILTER_AVG \ + | PNG_ASM_FLAG_MMX_READ_FILTER_PAETH ) +#define PNG_MMX_WRITE_FLAGS ( 0 ) + +#define PNG_MMX_FLAGS ( PNG_ASM_FLAG_MMX_SUPPORT_COMPILED \ + | PNG_ASM_FLAG_MMX_SUPPORT_IN_CPU \ + | PNG_MMX_READ_FLAGS \ + | PNG_MMX_WRITE_FLAGS ) + +#define PNG_SELECT_READ 1 +#define PNG_SELECT_WRITE 2 + +#if !defined(PNG_1_0_X) +/* pngget.c */ +extern PNG_EXPORT(png_uint_32,png_get_mmx_flagmask) + PNGARG((int flag_select, int *compilerID)); + +/* pngget.c */ +extern PNG_EXPORT(png_uint_32,png_get_asm_flagmask) + PNGARG((int flag_select)); + +/* pngget.c */ +extern PNG_EXPORT(png_uint_32,png_get_asm_flags) + PNGARG((png_structp png_ptr)); + +/* pngget.c */ +extern PNG_EXPORT(png_byte,png_get_mmx_bitdepth_threshold) + PNGARG((png_structp png_ptr)); + +/* pngget.c */ +extern PNG_EXPORT(png_uint_32,png_get_mmx_rowbytes_threshold) + PNGARG((png_structp png_ptr)); + +/* pngset.c */ +extern PNG_EXPORT(void,png_set_asm_flags) + PNGARG((png_structp png_ptr, png_uint_32 asm_flags)); + +/* pngset.c */ +extern PNG_EXPORT(void,png_set_mmx_thresholds) + PNGARG((png_structp png_ptr, png_byte mmx_bitdepth_threshold, + png_uint_32 mmx_rowbytes_threshold)); + +#endif /* PNG_1_0_X */ +#endif /* PNG_ASSEMBLER_CODE_SUPPORTED */ + +#if !defined(PNG_1_0_X) +/* png.c, pnggccrd.c, or pngvcrd.c */ +extern PNG_EXPORT(int,png_mmx_support) PNGARG((void)); + +/* Strip the prepended error numbers ("#nnn ") from error and warning + * messages before passing them to the error or warning handler. */ +#ifdef PNG_ERROR_NUMBERS_SUPPORTED +extern PNG_EXPORT(void,png_set_strip_error_numbers) PNGARG((png_structp + png_ptr, png_uint_32 strip_mode)); +#endif + +#endif /* PNG_1_0_X */ + +/* Added at libpng-1.2.6 */ +#ifdef PNG_SET_USER_LIMITS_SUPPORTED +extern PNG_EXPORT(void,png_set_user_limits) PNGARG((png_structp + png_ptr, png_uint_32 user_width_max, png_uint_32 user_height_max)); +extern PNG_EXPORT(png_uint_32,png_get_user_width_max) PNGARG((png_structp + png_ptr)); +extern PNG_EXPORT(png_uint_32,png_get_user_height_max) PNGARG((png_structp + png_ptr)); +#endif + +/* Maintainer: Put new public prototypes here ^, in libpng.3, and project defs */ + +#ifdef PNG_READ_COMPOSITE_NODIV_SUPPORTED +/* With these routines we avoid an integer divide, which will be slower on + * most machines. However, it does take more operations than the corresponding + * divide method, so it may be slower on a few RISC systems. There are two + * shifts (by 8 or 16 bits) and an addition, versus a single integer divide. + * + * Note that the rounding factors are NOT supposed to be the same! 128 and + * 32768 are correct for the NODIV code; 127 and 32767 are correct for the + * standard method. + * + * [Optimized code by Greg Roelofs and Mark Adler...blame us for bugs. :-) ] + */ + + /* fg and bg should be in `gamma 1.0' space; alpha is the opacity */ + +# define png_composite(composite, fg, alpha, bg) \ + { png_uint_16 temp = (png_uint_16)((png_uint_16)(fg) * (png_uint_16)(alpha) \ + + (png_uint_16)(bg)*(png_uint_16)(255 - \ + (png_uint_16)(alpha)) + (png_uint_16)128); \ + (composite) = (png_byte)((temp + (temp >> 8)) >> 8); } + +# define png_composite_16(composite, fg, alpha, bg) \ + { png_uint_32 temp = (png_uint_32)((png_uint_32)(fg) * (png_uint_32)(alpha) \ + + (png_uint_32)(bg)*(png_uint_32)(65535L - \ + (png_uint_32)(alpha)) + (png_uint_32)32768L); \ + (composite) = (png_uint_16)((temp + (temp >> 16)) >> 16); } + +#else /* standard method using integer division */ + +# define png_composite(composite, fg, alpha, bg) \ + (composite) = (png_byte)(((png_uint_16)(fg) * (png_uint_16)(alpha) + \ + (png_uint_16)(bg) * (png_uint_16)(255 - (png_uint_16)(alpha)) + \ + (png_uint_16)127) / 255) + +# define png_composite_16(composite, fg, alpha, bg) \ + (composite) = (png_uint_16)(((png_uint_32)(fg) * (png_uint_32)(alpha) + \ + (png_uint_32)(bg)*(png_uint_32)(65535L - (png_uint_32)(alpha)) + \ + (png_uint_32)32767) / (png_uint_32)65535L) + +#endif /* PNG_READ_COMPOSITE_NODIV_SUPPORTED */ + +/* Inline macros to do direct reads of bytes from the input buffer. These + * require that you are using an architecture that uses PNG byte ordering + * (MSB first) and supports unaligned data storage. I think that PowerPC + * in big-endian mode and 680x0 are the only ones that will support this. + * The x86 line of processors definitely do not. The png_get_int_32() + * routine also assumes we are using two's complement format for negative + * values, which is almost certainly true. + */ +#if defined(PNG_READ_BIG_ENDIAN_SUPPORTED) +# define png_get_uint_32(buf) ( *((png_uint_32p) (buf))) +# define png_get_uint_16(buf) ( *((png_uint_16p) (buf))) +# define png_get_int_32(buf) ( *((png_int_32p) (buf))) +#else +extern PNG_EXPORT(png_uint_32,png_get_uint_32) PNGARG((png_bytep buf)); +extern PNG_EXPORT(png_uint_16,png_get_uint_16) PNGARG((png_bytep buf)); +extern PNG_EXPORT(png_int_32,png_get_int_32) PNGARG((png_bytep buf)); +#endif /* !PNG_READ_BIG_ENDIAN_SUPPORTED */ +extern PNG_EXPORT(png_uint_32,png_get_uint_31) + PNGARG((png_structp png_ptr, png_bytep buf)); +/* No png_get_int_16 -- may be added if there's a real need for it. */ + +/* Place a 32-bit number into a buffer in PNG byte order (big-endian). + */ +extern PNG_EXPORT(void,png_save_uint_32) + PNGARG((png_bytep buf, png_uint_32 i)); +extern PNG_EXPORT(void,png_save_int_32) + PNGARG((png_bytep buf, png_int_32 i)); + +/* Place a 16-bit number into a buffer in PNG byte order. + * The parameter is declared unsigned int, not png_uint_16, + * just to avoid potential problems on pre-ANSI C compilers. + */ +extern PNG_EXPORT(void,png_save_uint_16) + PNGARG((png_bytep buf, unsigned int i)); +/* No png_save_int_16 -- may be added if there's a real need for it. */ + +/* ************************************************************************* */ + +/* These next functions are used internally in the code. They generally + * shouldn't be used unless you are writing code to add or replace some + * functionality in libpng. More information about most functions can + * be found in the files where the functions are located. + */ + +#if defined(PNG_INTERNAL) + +/* Various modes of operation. Note that after an init, mode is set to + * zero automatically when the structure is created. + */ +#define PNG_HAVE_IHDR 0x01 +#define PNG_HAVE_PLTE 0x02 +#define PNG_HAVE_IDAT 0x04 +#define PNG_AFTER_IDAT 0x08 +#define PNG_HAVE_IEND 0x10 +#define PNG_HAVE_gAMA 0x20 +#define PNG_HAVE_cHRM 0x40 +#define PNG_HAVE_sRGB 0x80 +#define PNG_HAVE_CHUNK_HEADER 0x100 +#define PNG_WROTE_tIME 0x200 +#define PNG_WROTE_INFO_BEFORE_PLTE 0x400 +#define PNG_BACKGROUND_IS_GRAY 0x800 +#define PNG_HAVE_PNG_SIGNATURE 0x1000 + +/* flags for the transformations the PNG library does on the image data */ +#define PNG_BGR 0x0001 +#define PNG_INTERLACE 0x0002 +#define PNG_PACK 0x0004 +#define PNG_SHIFT 0x0008 +#define PNG_SWAP_BYTES 0x0010 +#define PNG_INVERT_MONO 0x0020 +#define PNG_DITHER 0x0040 +#define PNG_BACKGROUND 0x0080 +#define PNG_BACKGROUND_EXPAND 0x0100 + /* 0x0200 unused */ +#define PNG_16_TO_8 0x0400 +#define PNG_RGBA 0x0800 +#define PNG_EXPAND 0x1000 +#define PNG_GAMMA 0x2000 +#define PNG_GRAY_TO_RGB 0x4000 +#define PNG_FILLER 0x8000L +#define PNG_PACKSWAP 0x10000L +#define PNG_SWAP_ALPHA 0x20000L +#define PNG_STRIP_ALPHA 0x40000L +#define PNG_INVERT_ALPHA 0x80000L +#define PNG_USER_TRANSFORM 0x100000L +#define PNG_RGB_TO_GRAY_ERR 0x200000L +#define PNG_RGB_TO_GRAY_WARN 0x400000L +#define PNG_RGB_TO_GRAY 0x600000L /* two bits, RGB_TO_GRAY_ERR|WARN */ + /* 0x800000L Unused */ +#define PNG_ADD_ALPHA 0x1000000L /* Added to libpng-1.2.7 */ +#define PNG_EXPAND_tRNS 0x2000000L /* Added to libpng-1.2.9 */ + /* 0x4000000L unused */ + /* 0x8000000L unused */ + /* 0x10000000L unused */ + /* 0x20000000L unused */ + /* 0x40000000L unused */ + +/* flags for png_create_struct */ +#define PNG_STRUCT_PNG 0x0001 +#define PNG_STRUCT_INFO 0x0002 + +/* Scaling factor for filter heuristic weighting calculations */ +#define PNG_WEIGHT_SHIFT 8 +#define PNG_WEIGHT_FACTOR (1<<(PNG_WEIGHT_SHIFT)) +#define PNG_COST_SHIFT 3 +#define PNG_COST_FACTOR (1<<(PNG_COST_SHIFT)) + +/* flags for the png_ptr->flags rather than declaring a byte for each one */ +#define PNG_FLAG_ZLIB_CUSTOM_STRATEGY 0x0001 +#define PNG_FLAG_ZLIB_CUSTOM_LEVEL 0x0002 +#define PNG_FLAG_ZLIB_CUSTOM_MEM_LEVEL 0x0004 +#define PNG_FLAG_ZLIB_CUSTOM_WINDOW_BITS 0x0008 +#define PNG_FLAG_ZLIB_CUSTOM_METHOD 0x0010 +#define PNG_FLAG_ZLIB_FINISHED 0x0020 +#define PNG_FLAG_ROW_INIT 0x0040 +#define PNG_FLAG_FILLER_AFTER 0x0080 +#define PNG_FLAG_CRC_ANCILLARY_USE 0x0100 +#define PNG_FLAG_CRC_ANCILLARY_NOWARN 0x0200 +#define PNG_FLAG_CRC_CRITICAL_USE 0x0400 +#define PNG_FLAG_CRC_CRITICAL_IGNORE 0x0800 +#define PNG_FLAG_FREE_PLTE 0x1000 +#define PNG_FLAG_FREE_TRNS 0x2000 +#define PNG_FLAG_FREE_HIST 0x4000 +#define PNG_FLAG_KEEP_UNKNOWN_CHUNKS 0x8000L +#define PNG_FLAG_KEEP_UNSAFE_CHUNKS 0x10000L +#define PNG_FLAG_LIBRARY_MISMATCH 0x20000L +#define PNG_FLAG_STRIP_ERROR_NUMBERS 0x40000L +#define PNG_FLAG_STRIP_ERROR_TEXT 0x80000L +#define PNG_FLAG_MALLOC_NULL_MEM_OK 0x100000L +#define PNG_FLAG_ADD_ALPHA 0x200000L /* Added to libpng-1.2.8 */ +#define PNG_FLAG_STRIP_ALPHA 0x400000L /* Added to libpng-1.2.8 */ + /* 0x800000L unused */ + /* 0x1000000L unused */ + /* 0x2000000L unused */ + /* 0x4000000L unused */ + /* 0x8000000L unused */ + /* 0x10000000L unused */ + /* 0x20000000L unused */ + /* 0x40000000L unused */ + +#define PNG_FLAG_CRC_ANCILLARY_MASK (PNG_FLAG_CRC_ANCILLARY_USE | \ + PNG_FLAG_CRC_ANCILLARY_NOWARN) + +#define PNG_FLAG_CRC_CRITICAL_MASK (PNG_FLAG_CRC_CRITICAL_USE | \ + PNG_FLAG_CRC_CRITICAL_IGNORE) + +#define PNG_FLAG_CRC_MASK (PNG_FLAG_CRC_ANCILLARY_MASK | \ + PNG_FLAG_CRC_CRITICAL_MASK) + +/* save typing and make code easier to understand */ + +#define PNG_COLOR_DIST(c1, c2) (abs((int)((c1).red) - (int)((c2).red)) + \ + abs((int)((c1).green) - (int)((c2).green)) + \ + abs((int)((c1).blue) - (int)((c2).blue))) + +/* Added to libpng-1.2.6 JB */ +#define PNG_ROWBYTES(pixel_bits, width) \ + ((pixel_bits) >= 8 ? \ + ((width) * (((png_uint_32)(pixel_bits)) >> 3)) : \ + (( ((width) * ((png_uint_32)(pixel_bits))) + 7) >> 3) ) + +/* PNG_OUT_OF_RANGE returns true if value is outside the range + ideal-delta..ideal+delta. Each argument is evaluated twice. + "ideal" and "delta" should be constants, normally simple + integers, "value" a variable. Added to libpng-1.2.6 JB */ +#define PNG_OUT_OF_RANGE(value, ideal, delta) \ + ( (value) < (ideal)-(delta) || (value) > (ideal)+(delta) ) + +/* variables declared in png.c - only it needs to define PNG_NO_EXTERN */ +#if !defined(PNG_NO_EXTERN) || defined(PNG_ALWAYS_EXTERN) +/* place to hold the signature string for a PNG file. */ +#ifdef PNG_USE_GLOBAL_ARRAYS + PNG_EXPORT_VAR (const png_byte FARDATA) png_sig[8]; +#else +#if 0 +#define png_sig png_sig_bytes(NULL) +#endif +#endif +#endif /* PNG_NO_EXTERN */ + +/* Constant strings for known chunk types. If you need to add a chunk, + * define the name here, and add an invocation of the macro in png.c and + * wherever it's needed. + */ +#define PNG_IHDR const png_byte png_IHDR[5] = { 73, 72, 68, 82, '\0'} +#define PNG_IDAT const png_byte png_IDAT[5] = { 73, 68, 65, 84, '\0'} +#define PNG_IEND const png_byte png_IEND[5] = { 73, 69, 78, 68, '\0'} +#define PNG_PLTE const png_byte png_PLTE[5] = { 80, 76, 84, 69, '\0'} +#define PNG_bKGD const png_byte png_bKGD[5] = { 98, 75, 71, 68, '\0'} +#define PNG_cHRM const png_byte png_cHRM[5] = { 99, 72, 82, 77, '\0'} +#define PNG_gAMA const png_byte png_gAMA[5] = {103, 65, 77, 65, '\0'} +#define PNG_hIST const png_byte png_hIST[5] = {104, 73, 83, 84, '\0'} +#define PNG_iCCP const png_byte png_iCCP[5] = {105, 67, 67, 80, '\0'} +#define PNG_iTXt const png_byte png_iTXt[5] = {105, 84, 88, 116, '\0'} +#define PNG_oFFs const png_byte png_oFFs[5] = {111, 70, 70, 115, '\0'} +#define PNG_pCAL const png_byte png_pCAL[5] = {112, 67, 65, 76, '\0'} +#define PNG_sCAL const png_byte png_sCAL[5] = {115, 67, 65, 76, '\0'} +#define PNG_pHYs const png_byte png_pHYs[5] = {112, 72, 89, 115, '\0'} +#define PNG_sBIT const png_byte png_sBIT[5] = {115, 66, 73, 84, '\0'} +#define PNG_sPLT const png_byte png_sPLT[5] = {115, 80, 76, 84, '\0'} +#define PNG_sRGB const png_byte png_sRGB[5] = {115, 82, 71, 66, '\0'} +#define PNG_tEXt const png_byte png_tEXt[5] = {116, 69, 88, 116, '\0'} +#define PNG_tIME const png_byte png_tIME[5] = {116, 73, 77, 69, '\0'} +#define PNG_tRNS const png_byte png_tRNS[5] = {116, 82, 78, 83, '\0'} +#define PNG_zTXt const png_byte png_zTXt[5] = {122, 84, 88, 116, '\0'} + +#ifdef PNG_USE_GLOBAL_ARRAYS +PNG_EXPORT_VAR (const png_byte FARDATA) png_IHDR[5]; +PNG_EXPORT_VAR (const png_byte FARDATA) png_IDAT[5]; +PNG_EXPORT_VAR (const png_byte FARDATA) png_IEND[5]; +PNG_EXPORT_VAR (const png_byte FARDATA) png_PLTE[5]; +PNG_EXPORT_VAR (const png_byte FARDATA) png_bKGD[5]; +PNG_EXPORT_VAR (const png_byte FARDATA) png_cHRM[5]; +PNG_EXPORT_VAR (const png_byte FARDATA) png_gAMA[5]; +PNG_EXPORT_VAR (const png_byte FARDATA) png_hIST[5]; +PNG_EXPORT_VAR (const png_byte FARDATA) png_iCCP[5]; +PNG_EXPORT_VAR (const png_byte FARDATA) png_iTXt[5]; +PNG_EXPORT_VAR (const png_byte FARDATA) png_oFFs[5]; +PNG_EXPORT_VAR (const png_byte FARDATA) png_pCAL[5]; +PNG_EXPORT_VAR (const png_byte FARDATA) png_sCAL[5]; +PNG_EXPORT_VAR (const png_byte FARDATA) png_pHYs[5]; +PNG_EXPORT_VAR (const png_byte FARDATA) png_sBIT[5]; +PNG_EXPORT_VAR (const png_byte FARDATA) png_sPLT[5]; +PNG_EXPORT_VAR (const png_byte FARDATA) png_sRGB[5]; +PNG_EXPORT_VAR (const png_byte FARDATA) png_tEXt[5]; +PNG_EXPORT_VAR (const png_byte FARDATA) png_tIME[5]; +PNG_EXPORT_VAR (const png_byte FARDATA) png_tRNS[5]; +PNG_EXPORT_VAR (const png_byte FARDATA) png_zTXt[5]; +#endif /* PNG_USE_GLOBAL_ARRAYS */ + +#if defined(PNG_1_0_X) || defined (PNG_1_2_X) +/* Initialize png_ptr struct for reading, and allocate any other memory. + * (old interface - DEPRECATED - use png_create_read_struct instead). + */ +extern PNG_EXPORT(void,png_read_init) PNGARG((png_structp png_ptr)); +#undef png_read_init +#define png_read_init(png_ptr) png_read_init_3(&png_ptr, \ + PNG_LIBPNG_VER_STRING, png_sizeof(png_struct)); +#endif + +extern PNG_EXPORT(void,png_read_init_3) PNGARG((png_structpp ptr_ptr, + png_const_charp user_png_ver, png_size_t png_struct_size)); +#if defined(PNG_1_0_X) || defined (PNG_1_2_X) +extern PNG_EXPORT(void,png_read_init_2) PNGARG((png_structp png_ptr, + png_const_charp user_png_ver, png_size_t png_struct_size, png_size_t + png_info_size)); +#endif + +#if defined(PNG_1_0_X) || defined (PNG_1_2_X) +/* Initialize png_ptr struct for writing, and allocate any other memory. + * (old interface - DEPRECATED - use png_create_write_struct instead). + */ +extern PNG_EXPORT(void,png_write_init) PNGARG((png_structp png_ptr)); +#undef png_write_init +#define png_write_init(png_ptr) png_write_init_3(&png_ptr, \ + PNG_LIBPNG_VER_STRING, png_sizeof(png_struct)); +#endif + +extern PNG_EXPORT(void,png_write_init_3) PNGARG((png_structpp ptr_ptr, + png_const_charp user_png_ver, png_size_t png_struct_size)); +extern PNG_EXPORT(void,png_write_init_2) PNGARG((png_structp png_ptr, + png_const_charp user_png_ver, png_size_t png_struct_size, png_size_t + png_info_size)); + +/* Allocate memory for an internal libpng struct */ +PNG_EXTERN png_voidp png_create_struct PNGARG((int type)); + +/* Free memory from internal libpng struct */ +PNG_EXTERN void png_destroy_struct PNGARG((png_voidp struct_ptr)); + +PNG_EXTERN png_voidp png_create_struct_2 PNGARG((int type, png_malloc_ptr + malloc_fn, png_voidp mem_ptr)); +PNG_EXTERN void png_destroy_struct_2 PNGARG((png_voidp struct_ptr, + png_free_ptr free_fn, png_voidp mem_ptr)); + +/* Free any memory that info_ptr points to and reset struct. */ +PNG_EXTERN void png_info_destroy PNGARG((png_structp png_ptr, + png_infop info_ptr)); + +#ifndef PNG_1_0_X +/* Function to allocate memory for zlib. */ +PNG_EXTERN voidpf png_zalloc PNGARG((voidpf png_ptr, uInt items, uInt size)); + +/* Function to free memory for zlib */ +PNG_EXTERN void png_zfree PNGARG((voidpf png_ptr, voidpf ptr)); + +#ifdef PNG_SIZE_T +/* Function to convert a sizeof an item to png_sizeof item */ + PNG_EXTERN png_size_t PNGAPI png_convert_size PNGARG((size_t size)); +#endif + +/* Next four functions are used internally as callbacks. PNGAPI is required + * but not PNG_EXPORT. PNGAPI added at libpng version 1.2.3. */ + +PNG_EXTERN void PNGAPI png_default_read_data PNGARG((png_structp png_ptr, + png_bytep data, png_size_t length)); + +#ifdef PNG_PROGRESSIVE_READ_SUPPORTED +PNG_EXTERN void PNGAPI png_push_fill_buffer PNGARG((png_structp png_ptr, + png_bytep buffer, png_size_t length)); +#endif + +PNG_EXTERN void PNGAPI png_default_write_data PNGARG((png_structp png_ptr, + png_bytep data, png_size_t length)); + +#if defined(PNG_WRITE_FLUSH_SUPPORTED) +#if !defined(PNG_NO_STDIO) +PNG_EXTERN void PNGAPI png_default_flush PNGARG((png_structp png_ptr)); +#endif +#endif +#else /* PNG_1_0_X */ +#ifdef PNG_PROGRESSIVE_READ_SUPPORTED +PNG_EXTERN void png_push_fill_buffer PNGARG((png_structp png_ptr, + png_bytep buffer, png_size_t length)); +#endif +#endif /* PNG_1_0_X */ + +/* Reset the CRC variable */ +PNG_EXTERN void png_reset_crc PNGARG((png_structp png_ptr)); + +/* Write the "data" buffer to whatever output you are using. */ +PNG_EXTERN void png_write_data PNGARG((png_structp png_ptr, png_bytep data, + png_size_t length)); + +/* Read data from whatever input you are using into the "data" buffer */ +PNG_EXTERN void png_read_data PNGARG((png_structp png_ptr, png_bytep data, + png_size_t length)); + +/* Read bytes into buf, and update png_ptr->crc */ +PNG_EXTERN void png_crc_read PNGARG((png_structp png_ptr, png_bytep buf, + png_size_t length)); + +/* Decompress data in a chunk that uses compression */ +#if defined(PNG_zTXt_SUPPORTED) || defined(PNG_iTXt_SUPPORTED) || \ + defined(PNG_iCCP_SUPPORTED) || defined(PNG_sPLT_SUPPORTED) +PNG_EXTERN png_charp png_decompress_chunk PNGARG((png_structp png_ptr, + int comp_type, png_charp chunkdata, png_size_t chunklength, + png_size_t prefix_length, png_size_t *data_length)); +#endif + +/* Read "skip" bytes, read the file crc, and (optionally) verify png_ptr->crc */ +PNG_EXTERN int png_crc_finish PNGARG((png_structp png_ptr, png_uint_32 skip)); + +/* Read the CRC from the file and compare it to the libpng calculated CRC */ +PNG_EXTERN int png_crc_error PNGARG((png_structp png_ptr)); + +/* Calculate the CRC over a section of data. Note that we are only + * passing a maximum of 64K on systems that have this as a memory limit, + * since this is the maximum buffer size we can specify. + */ +PNG_EXTERN void png_calculate_crc PNGARG((png_structp png_ptr, png_bytep ptr, + png_size_t length)); + +#if defined(PNG_WRITE_FLUSH_SUPPORTED) +PNG_EXTERN void png_flush PNGARG((png_structp png_ptr)); +#endif + +/* simple function to write the signature */ +PNG_EXTERN void png_write_sig PNGARG((png_structp png_ptr)); + +/* write various chunks */ + +/* Write the IHDR chunk, and update the png_struct with the necessary + * information. + */ +PNG_EXTERN void png_write_IHDR PNGARG((png_structp png_ptr, png_uint_32 width, + png_uint_32 height, + int bit_depth, int color_type, int compression_method, int filter_method, + int interlace_method)); + +PNG_EXTERN void png_write_PLTE PNGARG((png_structp png_ptr, png_colorp palette, + png_uint_32 num_pal)); + +PNG_EXTERN void png_write_IDAT PNGARG((png_structp png_ptr, png_bytep data, + png_size_t length)); + +PNG_EXTERN void png_write_IEND PNGARG((png_structp png_ptr)); + +#if defined(PNG_WRITE_gAMA_SUPPORTED) +#ifdef PNG_FLOATING_POINT_SUPPORTED +PNG_EXTERN void png_write_gAMA PNGARG((png_structp png_ptr, double file_gamma)); +#endif +#ifdef PNG_FIXED_POINT_SUPPORTED +PNG_EXTERN void png_write_gAMA_fixed PNGARG((png_structp png_ptr, png_fixed_point + file_gamma)); +#endif +#endif + +#if defined(PNG_WRITE_sBIT_SUPPORTED) +PNG_EXTERN void png_write_sBIT PNGARG((png_structp png_ptr, png_color_8p sbit, + int color_type)); +#endif + +#if defined(PNG_WRITE_cHRM_SUPPORTED) +#ifdef PNG_FLOATING_POINT_SUPPORTED +PNG_EXTERN void png_write_cHRM PNGARG((png_structp png_ptr, + double white_x, double white_y, + double red_x, double red_y, double green_x, double green_y, + double blue_x, double blue_y)); +#endif +#ifdef PNG_FIXED_POINT_SUPPORTED +PNG_EXTERN void png_write_cHRM_fixed PNGARG((png_structp png_ptr, + png_fixed_point int_white_x, png_fixed_point int_white_y, + png_fixed_point int_red_x, png_fixed_point int_red_y, png_fixed_point + int_green_x, png_fixed_point int_green_y, png_fixed_point int_blue_x, + png_fixed_point int_blue_y)); +#endif +#endif + +#if defined(PNG_WRITE_sRGB_SUPPORTED) +PNG_EXTERN void png_write_sRGB PNGARG((png_structp png_ptr, + int intent)); +#endif + +#if defined(PNG_WRITE_iCCP_SUPPORTED) +PNG_EXTERN void png_write_iCCP PNGARG((png_structp png_ptr, + png_charp name, int compression_type, + png_charp profile, int proflen)); + /* Note to maintainer: profile should be png_bytep */ +#endif + +#if defined(PNG_WRITE_sPLT_SUPPORTED) +PNG_EXTERN void png_write_sPLT PNGARG((png_structp png_ptr, + png_sPLT_tp palette)); +#endif + +#if defined(PNG_WRITE_tRNS_SUPPORTED) +PNG_EXTERN void png_write_tRNS PNGARG((png_structp png_ptr, png_bytep trans, + png_color_16p values, int number, int color_type)); +#endif + +#if defined(PNG_WRITE_bKGD_SUPPORTED) +PNG_EXTERN void png_write_bKGD PNGARG((png_structp png_ptr, + png_color_16p values, int color_type)); +#endif + +#if defined(PNG_WRITE_hIST_SUPPORTED) +PNG_EXTERN void png_write_hIST PNGARG((png_structp png_ptr, png_uint_16p hist, + int num_hist)); +#endif + +#if defined(PNG_WRITE_TEXT_SUPPORTED) || defined(PNG_WRITE_pCAL_SUPPORTED) || \ + defined(PNG_WRITE_iCCP_SUPPORTED) || defined(PNG_WRITE_sPLT_SUPPORTED) +PNG_EXTERN png_size_t png_check_keyword PNGARG((png_structp png_ptr, + png_charp key, png_charpp new_key)); +#endif + +#if defined(PNG_WRITE_tEXt_SUPPORTED) +PNG_EXTERN void png_write_tEXt PNGARG((png_structp png_ptr, png_charp key, + png_charp text, png_size_t text_len)); +#endif + +#if defined(PNG_WRITE_zTXt_SUPPORTED) +PNG_EXTERN void png_write_zTXt PNGARG((png_structp png_ptr, png_charp key, + png_charp text, png_size_t text_len, int compression)); +#endif + +#if defined(PNG_WRITE_iTXt_SUPPORTED) +PNG_EXTERN void png_write_iTXt PNGARG((png_structp png_ptr, + int compression, png_charp key, png_charp lang, png_charp lang_key, + png_charp text)); +#endif + +#if defined(PNG_TEXT_SUPPORTED) /* Added at version 1.0.14 and 1.2.4 */ +PNG_EXTERN int png_set_text_2 PNGARG((png_structp png_ptr, + png_infop info_ptr, png_textp text_ptr, int num_text)); +#endif + +#if defined(PNG_WRITE_oFFs_SUPPORTED) +PNG_EXTERN void png_write_oFFs PNGARG((png_structp png_ptr, + png_int_32 x_offset, png_int_32 y_offset, int unit_type)); +#endif + +#if defined(PNG_WRITE_pCAL_SUPPORTED) +PNG_EXTERN void png_write_pCAL PNGARG((png_structp png_ptr, png_charp purpose, + png_int_32 X0, png_int_32 X1, int type, int nparams, + png_charp units, png_charpp params)); +#endif + +#if defined(PNG_WRITE_pHYs_SUPPORTED) +PNG_EXTERN void png_write_pHYs PNGARG((png_structp png_ptr, + png_uint_32 x_pixels_per_unit, png_uint_32 y_pixels_per_unit, + int unit_type)); +#endif + +#if defined(PNG_WRITE_tIME_SUPPORTED) +PNG_EXTERN void png_write_tIME PNGARG((png_structp png_ptr, + png_timep mod_time)); +#endif + +#if defined(PNG_WRITE_sCAL_SUPPORTED) +#if defined(PNG_FLOATING_POINT_SUPPORTED) && !defined(PNG_NO_STDIO) +PNG_EXTERN void png_write_sCAL PNGARG((png_structp png_ptr, + int unit, double width, double height)); +#else +#ifdef PNG_FIXED_POINT_SUPPORTED +PNG_EXTERN void png_write_sCAL_s PNGARG((png_structp png_ptr, + int unit, png_charp width, png_charp height)); +#endif +#endif +#endif + +/* Called when finished processing a row of data */ +PNG_EXTERN void png_write_finish_row PNGARG((png_structp png_ptr)); + +/* Internal use only. Called before first row of data */ +PNG_EXTERN void png_write_start_row PNGARG((png_structp png_ptr)); + +#if defined(PNG_READ_GAMMA_SUPPORTED) +PNG_EXTERN void png_build_gamma_table PNGARG((png_structp png_ptr)); +#endif + +/* combine a row of data, dealing with alpha, etc. if requested */ +PNG_EXTERN void png_combine_row PNGARG((png_structp png_ptr, png_bytep row, + int mask)); + +#if defined(PNG_READ_INTERLACING_SUPPORTED) +/* expand an interlaced row */ +/* OLD pre-1.0.9 interface: +PNG_EXTERN void png_do_read_interlace PNGARG((png_row_infop row_info, + png_bytep row, int pass, png_uint_32 transformations)); + */ +PNG_EXTERN void png_do_read_interlace PNGARG((png_structp png_ptr)); +#endif + +/* GRR TO DO (2.0 or whenever): simplify other internal calling interfaces */ + +#if defined(PNG_WRITE_INTERLACING_SUPPORTED) +/* grab pixels out of a row for an interlaced pass */ +PNG_EXTERN void png_do_write_interlace PNGARG((png_row_infop row_info, + png_bytep row, int pass)); +#endif + +/* unfilter a row */ +PNG_EXTERN void png_read_filter_row PNGARG((png_structp png_ptr, + png_row_infop row_info, png_bytep row, png_bytep prev_row, int filter)); + +/* Choose the best filter to use and filter the row data */ +PNG_EXTERN void png_write_find_filter PNGARG((png_structp png_ptr, + png_row_infop row_info)); + +/* Write out the filtered row. */ +PNG_EXTERN void png_write_filtered_row PNGARG((png_structp png_ptr, + png_bytep filtered_row)); +/* finish a row while reading, dealing with interlacing passes, etc. */ +PNG_EXTERN void png_read_finish_row PNGARG((png_structp png_ptr)); + +/* initialize the row buffers, etc. */ +PNG_EXTERN void png_read_start_row PNGARG((png_structp png_ptr)); +/* optional call to update the users info structure */ +PNG_EXTERN void png_read_transform_info PNGARG((png_structp png_ptr, + png_infop info_ptr)); + +/* these are the functions that do the transformations */ +#if defined(PNG_READ_FILLER_SUPPORTED) +PNG_EXTERN void png_do_read_filler PNGARG((png_row_infop row_info, + png_bytep row, png_uint_32 filler, png_uint_32 flags)); +#endif + +#if defined(PNG_READ_SWAP_ALPHA_SUPPORTED) +PNG_EXTERN void png_do_read_swap_alpha PNGARG((png_row_infop row_info, + png_bytep row)); +#endif + +#if defined(PNG_WRITE_SWAP_ALPHA_SUPPORTED) +PNG_EXTERN void png_do_write_swap_alpha PNGARG((png_row_infop row_info, + png_bytep row)); +#endif + +#if defined(PNG_READ_INVERT_ALPHA_SUPPORTED) +PNG_EXTERN void png_do_read_invert_alpha PNGARG((png_row_infop row_info, + png_bytep row)); +#endif + +#if defined(PNG_WRITE_INVERT_ALPHA_SUPPORTED) +PNG_EXTERN void png_do_write_invert_alpha PNGARG((png_row_infop row_info, + png_bytep row)); +#endif + +#if defined(PNG_WRITE_FILLER_SUPPORTED) || \ + defined(PNG_READ_STRIP_ALPHA_SUPPORTED) +PNG_EXTERN void png_do_strip_filler PNGARG((png_row_infop row_info, + png_bytep row, png_uint_32 flags)); +#endif + +#if defined(PNG_READ_SWAP_SUPPORTED) || defined(PNG_WRITE_SWAP_SUPPORTED) +PNG_EXTERN void png_do_swap PNGARG((png_row_infop row_info, png_bytep row)); +#endif + +#if defined(PNG_READ_PACKSWAP_SUPPORTED) || defined(PNG_WRITE_PACKSWAP_SUPPORTED) +PNG_EXTERN void png_do_packswap PNGARG((png_row_infop row_info, png_bytep row)); +#endif + +#if defined(PNG_READ_RGB_TO_GRAY_SUPPORTED) +PNG_EXTERN int png_do_rgb_to_gray PNGARG((png_structp png_ptr, png_row_infop + row_info, png_bytep row)); +#endif + +#if defined(PNG_READ_GRAY_TO_RGB_SUPPORTED) +PNG_EXTERN void png_do_gray_to_rgb PNGARG((png_row_infop row_info, + png_bytep row)); +#endif + +#if defined(PNG_READ_PACK_SUPPORTED) +PNG_EXTERN void png_do_unpack PNGARG((png_row_infop row_info, png_bytep row)); +#endif + +#if defined(PNG_READ_SHIFT_SUPPORTED) +PNG_EXTERN void png_do_unshift PNGARG((png_row_infop row_info, png_bytep row, + png_color_8p sig_bits)); +#endif + +#if defined(PNG_READ_INVERT_SUPPORTED) || defined(PNG_WRITE_INVERT_SUPPORTED) +PNG_EXTERN void png_do_invert PNGARG((png_row_infop row_info, png_bytep row)); +#endif + +#if defined(PNG_READ_16_TO_8_SUPPORTED) +PNG_EXTERN void png_do_chop PNGARG((png_row_infop row_info, png_bytep row)); +#endif + +#if defined(PNG_READ_DITHER_SUPPORTED) +PNG_EXTERN void png_do_dither PNGARG((png_row_infop row_info, + png_bytep row, png_bytep palette_lookup, png_bytep dither_lookup)); + +# if defined(PNG_CORRECT_PALETTE_SUPPORTED) +PNG_EXTERN void png_correct_palette PNGARG((png_structp png_ptr, + png_colorp palette, int num_palette)); +# endif +#endif + +#if defined(PNG_READ_BGR_SUPPORTED) || defined(PNG_WRITE_BGR_SUPPORTED) +PNG_EXTERN void png_do_bgr PNGARG((png_row_infop row_info, png_bytep row)); +#endif + +#if defined(PNG_WRITE_PACK_SUPPORTED) +PNG_EXTERN void png_do_pack PNGARG((png_row_infop row_info, + png_bytep row, png_uint_32 bit_depth)); +#endif + +#if defined(PNG_WRITE_SHIFT_SUPPORTED) +PNG_EXTERN void png_do_shift PNGARG((png_row_infop row_info, png_bytep row, + png_color_8p bit_depth)); +#endif + +#if defined(PNG_READ_BACKGROUND_SUPPORTED) +#if defined(PNG_READ_GAMMA_SUPPORTED) +PNG_EXTERN void png_do_background PNGARG((png_row_infop row_info, png_bytep row, + png_color_16p trans_values, png_color_16p background, + png_color_16p background_1, + png_bytep gamma_table, png_bytep gamma_from_1, png_bytep gamma_to_1, + png_uint_16pp gamma_16, png_uint_16pp gamma_16_from_1, + png_uint_16pp gamma_16_to_1, int gamma_shift)); +#else +PNG_EXTERN void png_do_background PNGARG((png_row_infop row_info, png_bytep row, + png_color_16p trans_values, png_color_16p background)); +#endif +#endif + +#if defined(PNG_READ_GAMMA_SUPPORTED) +PNG_EXTERN void png_do_gamma PNGARG((png_row_infop row_info, png_bytep row, + png_bytep gamma_table, png_uint_16pp gamma_16_table, + int gamma_shift)); +#endif + +#if defined(PNG_READ_EXPAND_SUPPORTED) +PNG_EXTERN void png_do_expand_palette PNGARG((png_row_infop row_info, + png_bytep row, png_colorp palette, png_bytep trans, int num_trans)); +PNG_EXTERN void png_do_expand PNGARG((png_row_infop row_info, + png_bytep row, png_color_16p trans_value)); +#endif + +/* The following decodes the appropriate chunks, and does error correction, + * then calls the appropriate callback for the chunk if it is valid. + */ + +/* decode the IHDR chunk */ +PNG_EXTERN void png_handle_IHDR PNGARG((png_structp png_ptr, png_infop info_ptr, + png_uint_32 length)); +PNG_EXTERN void png_handle_PLTE PNGARG((png_structp png_ptr, png_infop info_ptr, + png_uint_32 length)); +PNG_EXTERN void png_handle_IEND PNGARG((png_structp png_ptr, png_infop info_ptr, + png_uint_32 length)); + +#if defined(PNG_READ_bKGD_SUPPORTED) +PNG_EXTERN void png_handle_bKGD PNGARG((png_structp png_ptr, png_infop info_ptr, + png_uint_32 length)); +#endif + +#if defined(PNG_READ_cHRM_SUPPORTED) +PNG_EXTERN void png_handle_cHRM PNGARG((png_structp png_ptr, png_infop info_ptr, + png_uint_32 length)); +#endif + +#if defined(PNG_READ_gAMA_SUPPORTED) +PNG_EXTERN void png_handle_gAMA PNGARG((png_structp png_ptr, png_infop info_ptr, + png_uint_32 length)); +#endif + +#if defined(PNG_READ_hIST_SUPPORTED) +PNG_EXTERN void png_handle_hIST PNGARG((png_structp png_ptr, png_infop info_ptr, + png_uint_32 length)); +#endif + +#if defined(PNG_READ_iCCP_SUPPORTED) +extern void png_handle_iCCP PNGARG((png_structp png_ptr, png_infop info_ptr, + png_uint_32 length)); +#endif /* PNG_READ_iCCP_SUPPORTED */ + +#if defined(PNG_READ_iTXt_SUPPORTED) +PNG_EXTERN void png_handle_iTXt PNGARG((png_structp png_ptr, png_infop info_ptr, + png_uint_32 length)); +#endif + +#if defined(PNG_READ_oFFs_SUPPORTED) +PNG_EXTERN void png_handle_oFFs PNGARG((png_structp png_ptr, png_infop info_ptr, + png_uint_32 length)); +#endif + +#if defined(PNG_READ_pCAL_SUPPORTED) +PNG_EXTERN void png_handle_pCAL PNGARG((png_structp png_ptr, png_infop info_ptr, + png_uint_32 length)); +#endif + +#if defined(PNG_READ_pHYs_SUPPORTED) +PNG_EXTERN void png_handle_pHYs PNGARG((png_structp png_ptr, png_infop info_ptr, + png_uint_32 length)); +#endif + +#if defined(PNG_READ_sBIT_SUPPORTED) +PNG_EXTERN void png_handle_sBIT PNGARG((png_structp png_ptr, png_infop info_ptr, + png_uint_32 length)); +#endif + +#if defined(PNG_READ_sCAL_SUPPORTED) +PNG_EXTERN void png_handle_sCAL PNGARG((png_structp png_ptr, png_infop info_ptr, + png_uint_32 length)); +#endif + +#if defined(PNG_READ_sPLT_SUPPORTED) +extern void png_handle_sPLT PNGARG((png_structp png_ptr, png_infop info_ptr, + png_uint_32 length)); +#endif /* PNG_READ_sPLT_SUPPORTED */ + +#if defined(PNG_READ_sRGB_SUPPORTED) +PNG_EXTERN void png_handle_sRGB PNGARG((png_structp png_ptr, png_infop info_ptr, + png_uint_32 length)); +#endif + +#if defined(PNG_READ_tEXt_SUPPORTED) +PNG_EXTERN void png_handle_tEXt PNGARG((png_structp png_ptr, png_infop info_ptr, + png_uint_32 length)); +#endif + +#if defined(PNG_READ_tIME_SUPPORTED) +PNG_EXTERN void png_handle_tIME PNGARG((png_structp png_ptr, png_infop info_ptr, + png_uint_32 length)); +#endif + +#if defined(PNG_READ_tRNS_SUPPORTED) +PNG_EXTERN void png_handle_tRNS PNGARG((png_structp png_ptr, png_infop info_ptr, + png_uint_32 length)); +#endif + +#if defined(PNG_READ_zTXt_SUPPORTED) +PNG_EXTERN void png_handle_zTXt PNGARG((png_structp png_ptr, png_infop info_ptr, + png_uint_32 length)); +#endif + +PNG_EXTERN void png_handle_unknown PNGARG((png_structp png_ptr, + png_infop info_ptr, png_uint_32 length)); + +PNG_EXTERN void png_check_chunk_name PNGARG((png_structp png_ptr, + png_bytep chunk_name)); + +/* handle the transformations for reading and writing */ +PNG_EXTERN void png_do_read_transformations PNGARG((png_structp png_ptr)); +PNG_EXTERN void png_do_write_transformations PNGARG((png_structp png_ptr)); + +PNG_EXTERN void png_init_read_transformations PNGARG((png_structp png_ptr)); + +#ifdef PNG_PROGRESSIVE_READ_SUPPORTED +PNG_EXTERN void png_push_read_chunk PNGARG((png_structp png_ptr, + png_infop info_ptr)); +PNG_EXTERN void png_push_read_sig PNGARG((png_structp png_ptr, + png_infop info_ptr)); +PNG_EXTERN void png_push_check_crc PNGARG((png_structp png_ptr)); +PNG_EXTERN void png_push_crc_skip PNGARG((png_structp png_ptr, + png_uint_32 length)); +PNG_EXTERN void png_push_crc_finish PNGARG((png_structp png_ptr)); +PNG_EXTERN void png_push_save_buffer PNGARG((png_structp png_ptr)); +PNG_EXTERN void png_push_restore_buffer PNGARG((png_structp png_ptr, + png_bytep buffer, png_size_t buffer_length)); +PNG_EXTERN void png_push_read_IDAT PNGARG((png_structp png_ptr)); +PNG_EXTERN void png_process_IDAT_data PNGARG((png_structp png_ptr, + png_bytep buffer, png_size_t buffer_length)); +PNG_EXTERN void png_push_process_row PNGARG((png_structp png_ptr)); +PNG_EXTERN void png_push_handle_unknown PNGARG((png_structp png_ptr, + png_infop info_ptr, png_uint_32 length)); +PNG_EXTERN void png_push_have_info PNGARG((png_structp png_ptr, + png_infop info_ptr)); +PNG_EXTERN void png_push_have_end PNGARG((png_structp png_ptr, + png_infop info_ptr)); +PNG_EXTERN void png_push_have_row PNGARG((png_structp png_ptr, png_bytep row)); +PNG_EXTERN void png_push_read_end PNGARG((png_structp png_ptr, + png_infop info_ptr)); +PNG_EXTERN void png_process_some_data PNGARG((png_structp png_ptr, + png_infop info_ptr)); +PNG_EXTERN void png_read_push_finish_row PNGARG((png_structp png_ptr)); +#if defined(PNG_READ_tEXt_SUPPORTED) +PNG_EXTERN void png_push_handle_tEXt PNGARG((png_structp png_ptr, + png_infop info_ptr, png_uint_32 length)); +PNG_EXTERN void png_push_read_tEXt PNGARG((png_structp png_ptr, + png_infop info_ptr)); +#endif +#if defined(PNG_READ_zTXt_SUPPORTED) +PNG_EXTERN void png_push_handle_zTXt PNGARG((png_structp png_ptr, + png_infop info_ptr, png_uint_32 length)); +PNG_EXTERN void png_push_read_zTXt PNGARG((png_structp png_ptr, + png_infop info_ptr)); +#endif +#if defined(PNG_READ_iTXt_SUPPORTED) +PNG_EXTERN void png_push_handle_iTXt PNGARG((png_structp png_ptr, + png_infop info_ptr, png_uint_32 length)); +PNG_EXTERN void png_push_read_iTXt PNGARG((png_structp png_ptr, + png_infop info_ptr)); +#endif + +#endif /* PNG_PROGRESSIVE_READ_SUPPORTED */ + +#ifdef PNG_MNG_FEATURES_SUPPORTED +PNG_EXTERN void png_do_read_intrapixel PNGARG((png_row_infop row_info, + png_bytep row)); +PNG_EXTERN void png_do_write_intrapixel PNGARG((png_row_infop row_info, + png_bytep row)); +#endif + +#if defined(PNG_ASSEMBLER_CODE_SUPPORTED) +/* png.c */ /* PRIVATE */ +PNG_EXTERN void png_init_mmx_flags PNGARG((png_structp png_ptr)); +#endif +/* Maintainer: Put new private prototypes here ^ and in libpngpf.3 */ + +#endif /* PNG_INTERNAL */ + +#ifdef __cplusplus +} +#endif + +#endif /* PNG_VERSION_INFO_ONLY */ +/* do not put anything past this line */ +#endif /* PNG_H */ diff --git a/src/win32/dependencies/libpng/pngconf.h b/src/win32/dependencies/libpng/pngconf.h new file mode 100644 index 00000000..c12d1825 --- /dev/null +++ b/src/win32/dependencies/libpng/pngconf.h @@ -0,0 +1,1472 @@ + +/* pngconf.h - machine configurable file for libpng + * + * libpng version 1.2.10 - April 23, 2006 + * For conditions of distribution and use, see copyright notice in png.h + * Copyright (c) 1998-2005 Glenn Randers-Pehrson + * (Version 0.96 Copyright (c) 1996, 1997 Andreas Dilger) + * (Version 0.88 Copyright (c) 1995, 1996 Guy Eric Schalnat, Group 42, Inc.) + */ + +/* Any machine specific code is near the front of this file, so if you + * are configuring libpng for a machine, you may want to read the section + * starting here down to where it starts to typedef png_color, png_text, + * and png_info. + */ + +#ifndef PNGCONF_H +#define PNGCONF_H + +#define PNG_1_2_X + +/* + * PNG_USER_CONFIG has to be defined on the compiler command line. This + * includes the resource compiler for Windows DLL configurations. + */ +#ifdef PNG_USER_CONFIG +# ifndef PNG_USER_PRIVATEBUILD +# define PNG_USER_PRIVATEBUILD +# endif +#include "pngusr.h" +#endif + +/* PNG_CONFIGURE_LIBPNG is set by the "configure" script. */ +#ifdef PNG_CONFIGURE_LIBPNG +#ifdef HAVE_CONFIG_H +#include "config.h" +#endif +#endif + +/* + * Added at libpng-1.2.8 + * + * If you create a private DLL you need to define in "pngusr.h" the followings: + * #define PNG_USER_PRIVATEBUILD + * e.g. #define PNG_USER_PRIVATEBUILD "Build by MyCompany for xyz reasons." + * #define PNG_USER_DLLFNAME_POSTFIX + * e.g. // private DLL "libpng13gx.dll" + * #define PNG_USER_DLLFNAME_POSTFIX "gx" + * + * The following macros are also at your disposal if you want to complete the + * DLL VERSIONINFO structure. + * - PNG_USER_VERSIONINFO_COMMENTS + * - PNG_USER_VERSIONINFO_COMPANYNAME + * - PNG_USER_VERSIONINFO_LEGALTRADEMARKS + */ + +#ifdef __STDC__ +#ifdef SPECIALBUILD +# pragma message("PNG_LIBPNG_SPECIALBUILD (and deprecated SPECIALBUILD)\ + are now LIBPNG reserved macros. Use PNG_USER_PRIVATEBUILD instead.") +#endif + +#ifdef PRIVATEBUILD +# pragma message("PRIVATEBUILD is deprecated.\ + Use PNG_USER_PRIVATEBUILD instead.") +# define PNG_USER_PRIVATEBUILD PRIVATEBUILD +#endif +#endif /* __STDC__ */ + +#ifndef PNG_VERSION_INFO_ONLY + +/* End of material added to libpng-1.2.8 */ + +/* This is the size of the compression buffer, and thus the size of + * an IDAT chunk. Make this whatever size you feel is best for your + * machine. One of these will be allocated per png_struct. When this + * is full, it writes the data to the disk, and does some other + * calculations. Making this an extremely small size will slow + * the library down, but you may want to experiment to determine + * where it becomes significant, if you are concerned with memory + * usage. Note that zlib allocates at least 32Kb also. For readers, + * this describes the size of the buffer available to read the data in. + * Unless this gets smaller than the size of a row (compressed), + * it should not make much difference how big this is. + */ + +#ifndef PNG_ZBUF_SIZE +# define PNG_ZBUF_SIZE 8192 +#endif + +/* Enable if you want a write-only libpng */ + +#ifndef PNG_NO_READ_SUPPORTED +# define PNG_READ_SUPPORTED +#endif + +/* Enable if you want a read-only libpng */ + +#ifndef PNG_NO_WRITE_SUPPORTED +# define PNG_WRITE_SUPPORTED +#endif + +/* Enabled by default in 1.2.0. You can disable this if you don't need to + support PNGs that are embedded in MNG datastreams */ +#if !defined(PNG_1_0_X) && !defined(PNG_NO_MNG_FEATURES) +# ifndef PNG_MNG_FEATURES_SUPPORTED +# define PNG_MNG_FEATURES_SUPPORTED +# endif +#endif + +#ifndef PNG_NO_FLOATING_POINT_SUPPORTED +# ifndef PNG_FLOATING_POINT_SUPPORTED +# define PNG_FLOATING_POINT_SUPPORTED +# endif +#endif + +/* If you are running on a machine where you cannot allocate more + * than 64K of memory at once, uncomment this. While libpng will not + * normally need that much memory in a chunk (unless you load up a very + * large file), zlib needs to know how big of a chunk it can use, and + * libpng thus makes sure to check any memory allocation to verify it + * will fit into memory. +#define PNG_MAX_MALLOC_64K + */ +#if defined(MAXSEG_64K) && !defined(PNG_MAX_MALLOC_64K) +# define PNG_MAX_MALLOC_64K +#endif + +/* Special munging to support doing things the 'cygwin' way: + * 'Normal' png-on-win32 defines/defaults: + * PNG_BUILD_DLL -- building dll + * PNG_USE_DLL -- building an application, linking to dll + * (no define) -- building static library, or building an + * application and linking to the static lib + * 'Cygwin' defines/defaults: + * PNG_BUILD_DLL -- (ignored) building the dll + * (no define) -- (ignored) building an application, linking to the dll + * PNG_STATIC -- (ignored) building the static lib, or building an + * application that links to the static lib. + * ALL_STATIC -- (ignored) building various static libs, or building an + * application that links to the static libs. + * Thus, + * a cygwin user should define either PNG_BUILD_DLL or PNG_STATIC, and + * this bit of #ifdefs will define the 'correct' config variables based on + * that. If a cygwin user *wants* to define 'PNG_USE_DLL' that's okay, but + * unnecessary. + * + * Also, the precedence order is: + * ALL_STATIC (since we can't #undef something outside our namespace) + * PNG_BUILD_DLL + * PNG_STATIC + * (nothing) == PNG_USE_DLL + * + * CYGWIN (2002-01-20): The preceding is now obsolete. With the advent + * of auto-import in binutils, we no longer need to worry about + * __declspec(dllexport) / __declspec(dllimport) and friends. Therefore, + * we don't need to worry about PNG_STATIC or ALL_STATIC when it comes + * to __declspec() stuff. However, we DO need to worry about + * PNG_BUILD_DLL and PNG_STATIC because those change some defaults + * such as CONSOLE_IO and whether GLOBAL_ARRAYS are allowed. + */ +#if defined(__CYGWIN__) +# if defined(ALL_STATIC) +# if defined(PNG_BUILD_DLL) +# undef PNG_BUILD_DLL +# endif +# if defined(PNG_USE_DLL) +# undef PNG_USE_DLL +# endif +# if defined(PNG_DLL) +# undef PNG_DLL +# endif +# if !defined(PNG_STATIC) +# define PNG_STATIC +# endif +# else +# if defined (PNG_BUILD_DLL) +# if defined(PNG_STATIC) +# undef PNG_STATIC +# endif +# if defined(PNG_USE_DLL) +# undef PNG_USE_DLL +# endif +# if !defined(PNG_DLL) +# define PNG_DLL +# endif +# else +# if defined(PNG_STATIC) +# if defined(PNG_USE_DLL) +# undef PNG_USE_DLL +# endif +# if defined(PNG_DLL) +# undef PNG_DLL +# endif +# else +# if !defined(PNG_USE_DLL) +# define PNG_USE_DLL +# endif +# if !defined(PNG_DLL) +# define PNG_DLL +# endif +# endif +# endif +# endif +#endif + +/* This protects us against compilers that run on a windowing system + * and thus don't have or would rather us not use the stdio types: + * stdin, stdout, and stderr. The only one currently used is stderr + * in png_error() and png_warning(). #defining PNG_NO_CONSOLE_IO will + * prevent these from being compiled and used. #defining PNG_NO_STDIO + * will also prevent these, plus will prevent the entire set of stdio + * macros and functions (FILE *, printf, etc.) from being compiled and used, + * unless (PNG_DEBUG > 0) has been #defined. + * + * #define PNG_NO_CONSOLE_IO + * #define PNG_NO_STDIO + */ + +#if defined(_WIN32_WCE) +# include + /* Console I/O functions are not supported on WindowsCE */ +# define PNG_NO_CONSOLE_IO +# ifdef PNG_DEBUG +# undef PNG_DEBUG +# endif +#endif + +#ifdef PNG_BUILD_DLL +# ifndef PNG_CONSOLE_IO_SUPPORTED +# ifndef PNG_NO_CONSOLE_IO +# define PNG_NO_CONSOLE_IO +# endif +# endif +#endif + +# ifdef PNG_NO_STDIO +# ifndef PNG_NO_CONSOLE_IO +# define PNG_NO_CONSOLE_IO +# endif +# ifdef PNG_DEBUG +# if (PNG_DEBUG > 0) +# include +# endif +# endif +# else +# if !defined(_WIN32_WCE) +/* "stdio.h" functions are not supported on WindowsCE */ +# include +# endif +# endif + +/* This macro protects us against machines that don't have function + * prototypes (ie K&R style headers). If your compiler does not handle + * function prototypes, define this macro and use the included ansi2knr. + * I've always been able to use _NO_PROTO as the indicator, but you may + * need to drag the empty declaration out in front of here, or change the + * ifdef to suit your own needs. + */ +#ifndef PNGARG + +#ifdef OF /* zlib prototype munger */ +# define PNGARG(arglist) OF(arglist) +#else + +#ifdef _NO_PROTO +# define PNGARG(arglist) () +# ifndef PNG_TYPECAST_NULL +# define PNG_TYPECAST_NULL +# endif +#else +# define PNGARG(arglist) arglist +#endif /* _NO_PROTO */ + +#endif /* OF */ + +#endif /* PNGARG */ + +/* Try to determine if we are compiling on a Mac. Note that testing for + * just __MWERKS__ is not good enough, because the Codewarrior is now used + * on non-Mac platforms. + */ +#ifndef MACOS +# if (defined(__MWERKS__) && defined(macintosh)) || defined(applec) || \ + defined(THINK_C) || defined(__SC__) || defined(TARGET_OS_MAC) +# define MACOS +# endif +#endif + +/* enough people need this for various reasons to include it here */ +#if !defined(MACOS) && !defined(RISCOS) && !defined(_WIN32_WCE) +# include +#endif + +#if !defined(PNG_SETJMP_NOT_SUPPORTED) && !defined(PNG_NO_SETJMP_SUPPORTED) +# define PNG_SETJMP_SUPPORTED +#endif + +#ifdef PNG_SETJMP_SUPPORTED +/* This is an attempt to force a single setjmp behaviour on Linux. If + * the X config stuff didn't define _BSD_SOURCE we wouldn't need this. + */ + +# ifdef __linux__ +# ifdef _BSD_SOURCE +# define PNG_SAVE_BSD_SOURCE +# undef _BSD_SOURCE +# endif +# ifdef _SETJMP_H + /* If you encounter a compiler error here, see the explanation + * near the end of INSTALL. + */ + __png.h__ already includes setjmp.h; + __dont__ include it again.; +# endif +# endif /* __linux__ */ + + /* include setjmp.h for error handling */ +# include + +# ifdef __linux__ +# ifdef PNG_SAVE_BSD_SOURCE +# define _BSD_SOURCE +# undef PNG_SAVE_BSD_SOURCE +# endif +# endif /* __linux__ */ +#endif /* PNG_SETJMP_SUPPORTED */ + +#ifdef BSD +# include +#else +# include +#endif + +/* Other defines for things like memory and the like can go here. */ +#ifdef PNG_INTERNAL + +#include + +/* The functions exported by PNG_EXTERN are PNG_INTERNAL functions, which + * aren't usually used outside the library (as far as I know), so it is + * debatable if they should be exported at all. In the future, when it is + * possible to have run-time registry of chunk-handling functions, some of + * these will be made available again. +#define PNG_EXTERN extern + */ +#define PNG_EXTERN + +/* Other defines specific to compilers can go here. Try to keep + * them inside an appropriate ifdef/endif pair for portability. + */ + +#if defined(PNG_FLOATING_POINT_SUPPORTED) +# if defined(MACOS) + /* We need to check that hasn't already been included earlier + * as it seems it doesn't agree with , yet we should really use + * if possible. + */ +# if !defined(__MATH_H__) && !defined(__MATH_H) && !defined(__cmath__) +# include +# endif +# else +# include +# endif +# if defined(_AMIGA) && defined(__SASC) && defined(_M68881) + /* Amiga SAS/C: We must include builtin FPU functions when compiling using + * MATH=68881 + */ +# include +# endif +#endif + +/* Codewarrior on NT has linking problems without this. */ +#if (defined(__MWERKS__) && defined(WIN32)) || defined(__STDC__) +# define PNG_ALWAYS_EXTERN +#endif + +/* This provides the non-ANSI (far) memory allocation routines. */ +#if defined(__TURBOC__) && defined(__MSDOS__) +# include +# include +#endif + +/* I have no idea why is this necessary... */ +#if defined(_MSC_VER) && (defined(WIN32) || defined(_Windows) || \ + defined(_WINDOWS) || defined(_WIN32) || defined(__WIN32__)) +# include +#endif + +/* This controls how fine the dithering gets. As this allocates + * a largish chunk of memory (32K), those who are not as concerned + * with dithering quality can decrease some or all of these. + */ +#ifndef PNG_DITHER_RED_BITS +# define PNG_DITHER_RED_BITS 5 +#endif +#ifndef PNG_DITHER_GREEN_BITS +# define PNG_DITHER_GREEN_BITS 5 +#endif +#ifndef PNG_DITHER_BLUE_BITS +# define PNG_DITHER_BLUE_BITS 5 +#endif + +/* This controls how fine the gamma correction becomes when you + * are only interested in 8 bits anyway. Increasing this value + * results in more memory being used, and more pow() functions + * being called to fill in the gamma tables. Don't set this value + * less then 8, and even that may not work (I haven't tested it). + */ + +#ifndef PNG_MAX_GAMMA_8 +# define PNG_MAX_GAMMA_8 11 +#endif + +/* This controls how much a difference in gamma we can tolerate before + * we actually start doing gamma conversion. + */ +#ifndef PNG_GAMMA_THRESHOLD +# define PNG_GAMMA_THRESHOLD 0.05 +#endif + +#endif /* PNG_INTERNAL */ + +/* The following uses const char * instead of char * for error + * and warning message functions, so some compilers won't complain. + * If you do not want to use const, define PNG_NO_CONST here. + */ + +#ifndef PNG_NO_CONST +# define PNG_CONST const +#else +# define PNG_CONST +#endif + +/* The following defines give you the ability to remove code from the + * library that you will not be using. I wish I could figure out how to + * automate this, but I can't do that without making it seriously hard + * on the users. So if you are not using an ability, change the #define + * to and #undef, and that part of the library will not be compiled. If + * your linker can't find a function, you may want to make sure the + * ability is defined here. Some of these depend upon some others being + * defined. I haven't figured out all the interactions here, so you may + * have to experiment awhile to get everything to compile. If you are + * creating or using a shared library, you probably shouldn't touch this, + * as it will affect the size of the structures, and this will cause bad + * things to happen if the library and/or application ever change. + */ + +/* Any features you will not be using can be undef'ed here */ + +/* GR-P, 0.96a: Set "*TRANSFORMS_SUPPORTED as default but allow user + * to turn it off with "*TRANSFORMS_NOT_SUPPORTED" or *PNG_NO_*_TRANSFORMS + * on the compile line, then pick and choose which ones to define without + * having to edit this file. It is safe to use the *TRANSFORMS_NOT_SUPPORTED + * if you only want to have a png-compliant reader/writer but don't need + * any of the extra transformations. This saves about 80 kbytes in a + * typical installation of the library. (PNG_NO_* form added in version + * 1.0.1c, for consistency) + */ + +/* The size of the png_text structure changed in libpng-1.0.6 when + * iTXt support was added. iTXt support was turned off by default through + * libpng-1.2.x, to support old apps that malloc the png_text structure + * instead of calling png_set_text() and letting libpng malloc it. It + * was turned on by default in libpng-1.3.0. + */ + +#if defined(PNG_1_0_X) || defined (PNG_1_2_X) +# ifndef PNG_NO_iTXt_SUPPORTED +# define PNG_NO_iTXt_SUPPORTED +# endif +# ifndef PNG_NO_READ_iTXt +# define PNG_NO_READ_iTXt +# endif +# ifndef PNG_NO_WRITE_iTXt +# define PNG_NO_WRITE_iTXt +# endif +#endif + +#if !defined(PNG_NO_iTXt_SUPPORTED) +# if !defined(PNG_READ_iTXt_SUPPORTED) && !defined(PNG_NO_READ_iTXt) +# define PNG_READ_iTXt +# endif +# if !defined(PNG_WRITE_iTXt_SUPPORTED) && !defined(PNG_NO_WRITE_iTXt) +# define PNG_WRITE_iTXt +# endif +#endif + +/* The following support, added after version 1.0.0, can be turned off here en + * masse by defining PNG_LEGACY_SUPPORTED in case you need binary compatibility + * with old applications that require the length of png_struct and png_info + * to remain unchanged. + */ + +#ifdef PNG_LEGACY_SUPPORTED +# define PNG_NO_FREE_ME +# define PNG_NO_READ_UNKNOWN_CHUNKS +# define PNG_NO_WRITE_UNKNOWN_CHUNKS +# define PNG_NO_READ_USER_CHUNKS +# define PNG_NO_READ_iCCP +# define PNG_NO_WRITE_iCCP +# define PNG_NO_READ_iTXt +# define PNG_NO_WRITE_iTXt +# define PNG_NO_READ_sCAL +# define PNG_NO_WRITE_sCAL +# define PNG_NO_READ_sPLT +# define PNG_NO_WRITE_sPLT +# define PNG_NO_INFO_IMAGE +# define PNG_NO_READ_RGB_TO_GRAY +# define PNG_NO_READ_USER_TRANSFORM +# define PNG_NO_WRITE_USER_TRANSFORM +# define PNG_NO_USER_MEM +# define PNG_NO_READ_EMPTY_PLTE +# define PNG_NO_MNG_FEATURES +# define PNG_NO_FIXED_POINT_SUPPORTED +#endif + +/* Ignore attempt to turn off both floating and fixed point support */ +#if !defined(PNG_FLOATING_POINT_SUPPORTED) || \ + !defined(PNG_NO_FIXED_POINT_SUPPORTED) +# define PNG_FIXED_POINT_SUPPORTED +#endif + +#ifndef PNG_NO_FREE_ME +# define PNG_FREE_ME_SUPPORTED +#endif + +#if defined(PNG_READ_SUPPORTED) + +#if !defined(PNG_READ_TRANSFORMS_NOT_SUPPORTED) && \ + !defined(PNG_NO_READ_TRANSFORMS) +# define PNG_READ_TRANSFORMS_SUPPORTED +#endif + +#ifdef PNG_READ_TRANSFORMS_SUPPORTED +# ifndef PNG_NO_READ_EXPAND +# define PNG_READ_EXPAND_SUPPORTED +# endif +# ifndef PNG_NO_READ_SHIFT +# define PNG_READ_SHIFT_SUPPORTED +# endif +# ifndef PNG_NO_READ_PACK +# define PNG_READ_PACK_SUPPORTED +# endif +# ifndef PNG_NO_READ_BGR +# define PNG_READ_BGR_SUPPORTED +# endif +# ifndef PNG_NO_READ_SWAP +# define PNG_READ_SWAP_SUPPORTED +# endif +# ifndef PNG_NO_READ_PACKSWAP +# define PNG_READ_PACKSWAP_SUPPORTED +# endif +# ifndef PNG_NO_READ_INVERT +# define PNG_READ_INVERT_SUPPORTED +# endif +# ifndef PNG_NO_READ_DITHER +# define PNG_READ_DITHER_SUPPORTED +# endif +# ifndef PNG_NO_READ_BACKGROUND +# define PNG_READ_BACKGROUND_SUPPORTED +# endif +# ifndef PNG_NO_READ_16_TO_8 +# define PNG_READ_16_TO_8_SUPPORTED +# endif +# ifndef PNG_NO_READ_FILLER +# define PNG_READ_FILLER_SUPPORTED +# endif +# ifndef PNG_NO_READ_GAMMA +# define PNG_READ_GAMMA_SUPPORTED +# endif +# ifndef PNG_NO_READ_GRAY_TO_RGB +# define PNG_READ_GRAY_TO_RGB_SUPPORTED +# endif +# ifndef PNG_NO_READ_SWAP_ALPHA +# define PNG_READ_SWAP_ALPHA_SUPPORTED +# endif +# ifndef PNG_NO_READ_INVERT_ALPHA +# define PNG_READ_INVERT_ALPHA_SUPPORTED +# endif +# ifndef PNG_NO_READ_STRIP_ALPHA +# define PNG_READ_STRIP_ALPHA_SUPPORTED +# endif +# ifndef PNG_NO_READ_USER_TRANSFORM +# define PNG_READ_USER_TRANSFORM_SUPPORTED +# endif +# ifndef PNG_NO_READ_RGB_TO_GRAY +# define PNG_READ_RGB_TO_GRAY_SUPPORTED +# endif +#endif /* PNG_READ_TRANSFORMS_SUPPORTED */ + +#if !defined(PNG_NO_PROGRESSIVE_READ) && \ + !defined(PNG_PROGRESSIVE_READ_NOT_SUPPORTED) /* if you don't do progressive */ +# define PNG_PROGRESSIVE_READ_SUPPORTED /* reading. This is not talking */ +#endif /* about interlacing capability! You'll */ + /* still have interlacing unless you change the following line: */ + +#define PNG_READ_INTERLACING_SUPPORTED /* required for PNG-compliant decoders */ + +#ifndef PNG_NO_READ_COMPOSITE_NODIV +# ifndef PNG_NO_READ_COMPOSITED_NODIV /* libpng-1.0.x misspelling */ +# define PNG_READ_COMPOSITE_NODIV_SUPPORTED /* well tested on Intel, SGI */ +# endif +#endif + +#if defined(PNG_1_0_X) || defined (PNG_1_2_X) +/* Deprecated, will be removed from version 2.0.0. + Use PNG_MNG_FEATURES_SUPPORTED instead. */ +#ifndef PNG_NO_READ_EMPTY_PLTE +# define PNG_READ_EMPTY_PLTE_SUPPORTED +#endif +#endif + +#endif /* PNG_READ_SUPPORTED */ + +#if defined(PNG_WRITE_SUPPORTED) + +# if !defined(PNG_WRITE_TRANSFORMS_NOT_SUPPORTED) && \ + !defined(PNG_NO_WRITE_TRANSFORMS) +# define PNG_WRITE_TRANSFORMS_SUPPORTED +#endif + +#ifdef PNG_WRITE_TRANSFORMS_SUPPORTED +# ifndef PNG_NO_WRITE_SHIFT +# define PNG_WRITE_SHIFT_SUPPORTED +# endif +# ifndef PNG_NO_WRITE_PACK +# define PNG_WRITE_PACK_SUPPORTED +# endif +# ifndef PNG_NO_WRITE_BGR +# define PNG_WRITE_BGR_SUPPORTED +# endif +# ifndef PNG_NO_WRITE_SWAP +# define PNG_WRITE_SWAP_SUPPORTED +# endif +# ifndef PNG_NO_WRITE_PACKSWAP +# define PNG_WRITE_PACKSWAP_SUPPORTED +# endif +# ifndef PNG_NO_WRITE_INVERT +# define PNG_WRITE_INVERT_SUPPORTED +# endif +# ifndef PNG_NO_WRITE_FILLER +# define PNG_WRITE_FILLER_SUPPORTED /* same as WRITE_STRIP_ALPHA */ +# endif +# ifndef PNG_NO_WRITE_SWAP_ALPHA +# define PNG_WRITE_SWAP_ALPHA_SUPPORTED +# endif +# ifndef PNG_NO_WRITE_INVERT_ALPHA +# define PNG_WRITE_INVERT_ALPHA_SUPPORTED +# endif +# ifndef PNG_NO_WRITE_USER_TRANSFORM +# define PNG_WRITE_USER_TRANSFORM_SUPPORTED +# endif +#endif /* PNG_WRITE_TRANSFORMS_SUPPORTED */ + +#if !defined(PNG_NO_WRITE_INTERLACING_SUPPORTED) && \ + !defined(PNG_WRITE_INTERLACING_SUPPORTED) +#define PNG_WRITE_INTERLACING_SUPPORTED /* not required for PNG-compliant + encoders, but can cause trouble + if left undefined */ +#endif + +#if !defined(PNG_NO_WRITE_WEIGHTED_FILTER) && \ + !defined(PNG_WRITE_WEIGHTED_FILTER) && \ + defined(PNG_FLOATING_POINT_SUPPORTED) +# define PNG_WRITE_WEIGHTED_FILTER_SUPPORTED +#endif + +#ifndef PNG_NO_WRITE_FLUSH +# define PNG_WRITE_FLUSH_SUPPORTED +#endif + +#if defined(PNG_1_0_X) || defined (PNG_1_2_X) +/* Deprecated, see PNG_MNG_FEATURES_SUPPORTED, above */ +#ifndef PNG_NO_WRITE_EMPTY_PLTE +# define PNG_WRITE_EMPTY_PLTE_SUPPORTED +#endif +#endif + +#endif /* PNG_WRITE_SUPPORTED */ + +#ifndef PNG_1_0_X +# ifndef PNG_NO_ERROR_NUMBERS +# define PNG_ERROR_NUMBERS_SUPPORTED +# endif +#endif /* PNG_1_0_X */ + +#if defined(PNG_READ_USER_TRANSFORM_SUPPORTED) || \ + defined(PNG_WRITE_USER_TRANSFORM_SUPPORTED) +# ifndef PNG_NO_USER_TRANSFORM_PTR +# define PNG_USER_TRANSFORM_PTR_SUPPORTED +# endif +#endif + +#ifndef PNG_NO_STDIO +# define PNG_TIME_RFC1123_SUPPORTED +#endif + +/* This adds extra functions in pngget.c for accessing data from the + * info pointer (added in version 0.99) + * png_get_image_width() + * png_get_image_height() + * png_get_bit_depth() + * png_get_color_type() + * png_get_compression_type() + * png_get_filter_type() + * png_get_interlace_type() + * png_get_pixel_aspect_ratio() + * png_get_pixels_per_meter() + * png_get_x_offset_pixels() + * png_get_y_offset_pixels() + * png_get_x_offset_microns() + * png_get_y_offset_microns() + */ +#if !defined(PNG_NO_EASY_ACCESS) && !defined(PNG_EASY_ACCESS_SUPPORTED) +# define PNG_EASY_ACCESS_SUPPORTED +#endif + +/* PNG_ASSEMBLER_CODE was enabled by default in version 1.2.0 + even when PNG_USE_PNGVCRD or PNG_USE_PNGGCCRD is not defined */ +#if defined(PNG_READ_SUPPORTED) && !defined(PNG_NO_ASSEMBLER_CODE) +# ifndef PNG_ASSEMBLER_CODE_SUPPORTED +# define PNG_ASSEMBLER_CODE_SUPPORTED +# endif +# if !defined(PNG_MMX_CODE_SUPPORTED) && !defined(PNG_NO_MMX_CODE) && \ + defined(__MMX__) +# define PNG_MMX_CODE_SUPPORTED +# endif +# if !defined(PNG_USE_PNGGCCRD) && !defined(PNG_NO_MMX_CODE) && \ + !defined(PNG_USE_PNGVCRD) && defined(__MMX__) +# define PNG_USE_PNGGCCRD +# endif +#endif + +/* If you are sure that you don't need thread safety and you are compiling + with PNG_USE_PNGCCRD for an MMX application, you can define this for + faster execution. See pnggccrd.c. +#define PNG_THREAD_UNSAFE_OK +*/ + +#if !defined(PNG_1_0_X) +#if !defined(PNG_NO_USER_MEM) && !defined(PNG_USER_MEM_SUPPORTED) +# define PNG_USER_MEM_SUPPORTED +#endif +#endif /* PNG_1_0_X */ + +/* Added at libpng-1.2.6 */ +#if !defined(PNG_1_0_X) +#ifndef PNG_SET_USER_LIMITS_SUPPORTED +#if !defined(PNG_NO_SET_USER_LIMITS) && !defined(PNG_SET_USER_LIMITS_SUPPORTED) +# define PNG_SET_USER_LIMITS_SUPPORTED +#endif +#endif +#endif /* PNG_1_0_X */ + +/* Added at libpng-1.0.16 and 1.2.6. To accept all valid PNGS no matter + * how large, set these limits to 0x7fffffffL + */ +#ifndef PNG_USER_WIDTH_MAX +# define PNG_USER_WIDTH_MAX 1000000L +#endif +#ifndef PNG_USER_HEIGHT_MAX +# define PNG_USER_HEIGHT_MAX 1000000L +#endif + +/* These are currently experimental features, define them if you want */ + +/* very little testing */ +/* +#ifdef PNG_READ_SUPPORTED +# ifndef PNG_READ_16_TO_8_ACCURATE_SCALE_SUPPORTED +# define PNG_READ_16_TO_8_ACCURATE_SCALE_SUPPORTED +# endif +#endif +*/ + +/* This is only for PowerPC big-endian and 680x0 systems */ +/* some testing */ +/* +#ifndef PNG_READ_BIG_ENDIAN_SUPPORTED +# define PNG_READ_BIG_ENDIAN_SUPPORTED +#endif +*/ + +/* Buggy compilers (e.g., gcc 2.7.2.2) need this */ +/* +#define PNG_NO_POINTER_INDEXING +*/ + +/* These functions are turned off by default, as they will be phased out. */ +/* +#define PNG_USELESS_TESTS_SUPPORTED +#define PNG_CORRECT_PALETTE_SUPPORTED +*/ + +/* Any chunks you are not interested in, you can undef here. The + * ones that allocate memory may be expecially important (hIST, + * tEXt, zTXt, tRNS, pCAL). Others will just save time and make png_info + * a bit smaller. + */ + +#if defined(PNG_READ_SUPPORTED) && \ + !defined(PNG_READ_ANCILLARY_CHUNKS_NOT_SUPPORTED) && \ + !defined(PNG_NO_READ_ANCILLARY_CHUNKS) +# define PNG_READ_ANCILLARY_CHUNKS_SUPPORTED +#endif + +#if defined(PNG_WRITE_SUPPORTED) && \ + !defined(PNG_WRITE_ANCILLARY_CHUNKS_NOT_SUPPORTED) && \ + !defined(PNG_NO_WRITE_ANCILLARY_CHUNKS) +# define PNG_WRITE_ANCILLARY_CHUNKS_SUPPORTED +#endif + +#ifdef PNG_READ_ANCILLARY_CHUNKS_SUPPORTED + +#ifdef PNG_NO_READ_TEXT +# define PNG_NO_READ_iTXt +# define PNG_NO_READ_tEXt +# define PNG_NO_READ_zTXt +#endif +#ifndef PNG_NO_READ_bKGD +# define PNG_READ_bKGD_SUPPORTED +# define PNG_bKGD_SUPPORTED +#endif +#ifndef PNG_NO_READ_cHRM +# define PNG_READ_cHRM_SUPPORTED +# define PNG_cHRM_SUPPORTED +#endif +#ifndef PNG_NO_READ_gAMA +# define PNG_READ_gAMA_SUPPORTED +# define PNG_gAMA_SUPPORTED +#endif +#ifndef PNG_NO_READ_hIST +# define PNG_READ_hIST_SUPPORTED +# define PNG_hIST_SUPPORTED +#endif +#ifndef PNG_NO_READ_iCCP +# define PNG_READ_iCCP_SUPPORTED +# define PNG_iCCP_SUPPORTED +#endif +#ifndef PNG_NO_READ_iTXt +# ifndef PNG_READ_iTXt_SUPPORTED +# define PNG_READ_iTXt_SUPPORTED +# endif +# ifndef PNG_iTXt_SUPPORTED +# define PNG_iTXt_SUPPORTED +# endif +#endif +#ifndef PNG_NO_READ_oFFs +# define PNG_READ_oFFs_SUPPORTED +# define PNG_oFFs_SUPPORTED +#endif +#ifndef PNG_NO_READ_pCAL +# define PNG_READ_pCAL_SUPPORTED +# define PNG_pCAL_SUPPORTED +#endif +#ifndef PNG_NO_READ_sCAL +# define PNG_READ_sCAL_SUPPORTED +# define PNG_sCAL_SUPPORTED +#endif +#ifndef PNG_NO_READ_pHYs +# define PNG_READ_pHYs_SUPPORTED +# define PNG_pHYs_SUPPORTED +#endif +#ifndef PNG_NO_READ_sBIT +# define PNG_READ_sBIT_SUPPORTED +# define PNG_sBIT_SUPPORTED +#endif +#ifndef PNG_NO_READ_sPLT +# define PNG_READ_sPLT_SUPPORTED +# define PNG_sPLT_SUPPORTED +#endif +#ifndef PNG_NO_READ_sRGB +# define PNG_READ_sRGB_SUPPORTED +# define PNG_sRGB_SUPPORTED +#endif +#ifndef PNG_NO_READ_tEXt +# define PNG_READ_tEXt_SUPPORTED +# define PNG_tEXt_SUPPORTED +#endif +#ifndef PNG_NO_READ_tIME +# define PNG_READ_tIME_SUPPORTED +# define PNG_tIME_SUPPORTED +#endif +#ifndef PNG_NO_READ_tRNS +# define PNG_READ_tRNS_SUPPORTED +# define PNG_tRNS_SUPPORTED +#endif +#ifndef PNG_NO_READ_zTXt +# define PNG_READ_zTXt_SUPPORTED +# define PNG_zTXt_SUPPORTED +#endif +#ifndef PNG_NO_READ_UNKNOWN_CHUNKS +# define PNG_READ_UNKNOWN_CHUNKS_SUPPORTED +# ifndef PNG_UNKNOWN_CHUNKS_SUPPORTED +# define PNG_UNKNOWN_CHUNKS_SUPPORTED +# endif +# ifndef PNG_NO_HANDLE_AS_UNKNOWN +# define PNG_HANDLE_AS_UNKNOWN_SUPPORTED +# endif +#endif +#if !defined(PNG_NO_READ_USER_CHUNKS) && \ + defined(PNG_READ_UNKNOWN_CHUNKS_SUPPORTED) +# define PNG_READ_USER_CHUNKS_SUPPORTED +# define PNG_USER_CHUNKS_SUPPORTED +# ifdef PNG_NO_READ_UNKNOWN_CHUNKS +# undef PNG_NO_READ_UNKNOWN_CHUNKS +# endif +# ifdef PNG_NO_HANDLE_AS_UNKNOWN +# undef PNG_NO_HANDLE_AS_UNKNOWN +# endif +#endif +#ifndef PNG_NO_READ_OPT_PLTE +# define PNG_READ_OPT_PLTE_SUPPORTED /* only affects support of the */ +#endif /* optional PLTE chunk in RGB and RGBA images */ +#if defined(PNG_READ_iTXt_SUPPORTED) || defined(PNG_READ_tEXt_SUPPORTED) || \ + defined(PNG_READ_zTXt_SUPPORTED) +# define PNG_READ_TEXT_SUPPORTED +# define PNG_TEXT_SUPPORTED +#endif + +#endif /* PNG_READ_ANCILLARY_CHUNKS_SUPPORTED */ + +#ifdef PNG_WRITE_ANCILLARY_CHUNKS_SUPPORTED + +#ifdef PNG_NO_WRITE_TEXT +# define PNG_NO_WRITE_iTXt +# define PNG_NO_WRITE_tEXt +# define PNG_NO_WRITE_zTXt +#endif +#ifndef PNG_NO_WRITE_bKGD +# define PNG_WRITE_bKGD_SUPPORTED +# ifndef PNG_bKGD_SUPPORTED +# define PNG_bKGD_SUPPORTED +# endif +#endif +#ifndef PNG_NO_WRITE_cHRM +# define PNG_WRITE_cHRM_SUPPORTED +# ifndef PNG_cHRM_SUPPORTED +# define PNG_cHRM_SUPPORTED +# endif +#endif +#ifndef PNG_NO_WRITE_gAMA +# define PNG_WRITE_gAMA_SUPPORTED +# ifndef PNG_gAMA_SUPPORTED +# define PNG_gAMA_SUPPORTED +# endif +#endif +#ifndef PNG_NO_WRITE_hIST +# define PNG_WRITE_hIST_SUPPORTED +# ifndef PNG_hIST_SUPPORTED +# define PNG_hIST_SUPPORTED +# endif +#endif +#ifndef PNG_NO_WRITE_iCCP +# define PNG_WRITE_iCCP_SUPPORTED +# ifndef PNG_iCCP_SUPPORTED +# define PNG_iCCP_SUPPORTED +# endif +#endif +#ifndef PNG_NO_WRITE_iTXt +# ifndef PNG_WRITE_iTXt_SUPPORTED +# define PNG_WRITE_iTXt_SUPPORTED +# endif +# ifndef PNG_iTXt_SUPPORTED +# define PNG_iTXt_SUPPORTED +# endif +#endif +#ifndef PNG_NO_WRITE_oFFs +# define PNG_WRITE_oFFs_SUPPORTED +# ifndef PNG_oFFs_SUPPORTED +# define PNG_oFFs_SUPPORTED +# endif +#endif +#ifndef PNG_NO_WRITE_pCAL +# define PNG_WRITE_pCAL_SUPPORTED +# ifndef PNG_pCAL_SUPPORTED +# define PNG_pCAL_SUPPORTED +# endif +#endif +#ifndef PNG_NO_WRITE_sCAL +# define PNG_WRITE_sCAL_SUPPORTED +# ifndef PNG_sCAL_SUPPORTED +# define PNG_sCAL_SUPPORTED +# endif +#endif +#ifndef PNG_NO_WRITE_pHYs +# define PNG_WRITE_pHYs_SUPPORTED +# ifndef PNG_pHYs_SUPPORTED +# define PNG_pHYs_SUPPORTED +# endif +#endif +#ifndef PNG_NO_WRITE_sBIT +# define PNG_WRITE_sBIT_SUPPORTED +# ifndef PNG_sBIT_SUPPORTED +# define PNG_sBIT_SUPPORTED +# endif +#endif +#ifndef PNG_NO_WRITE_sPLT +# define PNG_WRITE_sPLT_SUPPORTED +# ifndef PNG_sPLT_SUPPORTED +# define PNG_sPLT_SUPPORTED +# endif +#endif +#ifndef PNG_NO_WRITE_sRGB +# define PNG_WRITE_sRGB_SUPPORTED +# ifndef PNG_sRGB_SUPPORTED +# define PNG_sRGB_SUPPORTED +# endif +#endif +#ifndef PNG_NO_WRITE_tEXt +# define PNG_WRITE_tEXt_SUPPORTED +# ifndef PNG_tEXt_SUPPORTED +# define PNG_tEXt_SUPPORTED +# endif +#endif +#ifndef PNG_NO_WRITE_tIME +# define PNG_WRITE_tIME_SUPPORTED +# ifndef PNG_tIME_SUPPORTED +# define PNG_tIME_SUPPORTED +# endif +#endif +#ifndef PNG_NO_WRITE_tRNS +# define PNG_WRITE_tRNS_SUPPORTED +# ifndef PNG_tRNS_SUPPORTED +# define PNG_tRNS_SUPPORTED +# endif +#endif +#ifndef PNG_NO_WRITE_zTXt +# define PNG_WRITE_zTXt_SUPPORTED +# ifndef PNG_zTXt_SUPPORTED +# define PNG_zTXt_SUPPORTED +# endif +#endif +#ifndef PNG_NO_WRITE_UNKNOWN_CHUNKS +# define PNG_WRITE_UNKNOWN_CHUNKS_SUPPORTED +# ifndef PNG_UNKNOWN_CHUNKS_SUPPORTED +# define PNG_UNKNOWN_CHUNKS_SUPPORTED +# endif +# ifndef PNG_NO_HANDLE_AS_UNKNOWN +# ifndef PNG_HANDLE_AS_UNKNOWN_SUPPORTED +# define PNG_HANDLE_AS_UNKNOWN_SUPPORTED +# endif +# endif +#endif +#if defined(PNG_WRITE_iTXt_SUPPORTED) || defined(PNG_WRITE_tEXt_SUPPORTED) || \ + defined(PNG_WRITE_zTXt_SUPPORTED) +# define PNG_WRITE_TEXT_SUPPORTED +# ifndef PNG_TEXT_SUPPORTED +# define PNG_TEXT_SUPPORTED +# endif +#endif + +#endif /* PNG_WRITE_ANCILLARY_CHUNKS_SUPPORTED */ + +/* Turn this off to disable png_read_png() and + * png_write_png() and leave the row_pointers member + * out of the info structure. + */ +#ifndef PNG_NO_INFO_IMAGE +# define PNG_INFO_IMAGE_SUPPORTED +#endif + +/* need the time information for reading tIME chunks */ +#if defined(PNG_tIME_SUPPORTED) +# if !defined(_WIN32_WCE) + /* "time.h" functions are not supported on WindowsCE */ +# include +# endif +#endif + +/* Some typedefs to get us started. These should be safe on most of the + * common platforms. The typedefs should be at least as large as the + * numbers suggest (a png_uint_32 must be at least 32 bits long), but they + * don't have to be exactly that size. Some compilers dislike passing + * unsigned shorts as function parameters, so you may be better off using + * unsigned int for png_uint_16. Likewise, for 64-bit systems, you may + * want to have unsigned int for png_uint_32 instead of unsigned long. + */ + +typedef unsigned long png_uint_32; +typedef long png_int_32; +typedef unsigned short png_uint_16; +typedef short png_int_16; +typedef unsigned char png_byte; + +/* This is usually size_t. It is typedef'ed just in case you need it to + change (I'm not sure if you will or not, so I thought I'd be safe) */ +#ifdef PNG_SIZE_T + typedef PNG_SIZE_T png_size_t; +# define png_sizeof(x) png_convert_size(sizeof (x)) +#else + typedef size_t png_size_t; +# define png_sizeof(x) sizeof (x) +#endif + +/* The following is needed for medium model support. It cannot be in the + * PNG_INTERNAL section. Needs modification for other compilers besides + * MSC. Model independent support declares all arrays and pointers to be + * large using the far keyword. The zlib version used must also support + * model independent data. As of version zlib 1.0.4, the necessary changes + * have been made in zlib. The USE_FAR_KEYWORD define triggers other + * changes that are needed. (Tim Wegner) + */ + +/* Separate compiler dependencies (problem here is that zlib.h always + defines FAR. (SJT) */ +#ifdef __BORLANDC__ +# if defined(__LARGE__) || defined(__HUGE__) || defined(__COMPACT__) +# define LDATA 1 +# else +# define LDATA 0 +# endif + /* GRR: why is Cygwin in here? Cygwin is not Borland C... */ +# if !defined(__WIN32__) && !defined(__FLAT__) && !defined(__CYGWIN__) +# define PNG_MAX_MALLOC_64K +# if (LDATA != 1) +# ifndef FAR +# define FAR __far +# endif +# define USE_FAR_KEYWORD +# endif /* LDATA != 1 */ + /* Possibly useful for moving data out of default segment. + * Uncomment it if you want. Could also define FARDATA as + * const if your compiler supports it. (SJT) +# define FARDATA FAR + */ +# endif /* __WIN32__, __FLAT__, __CYGWIN__ */ +#endif /* __BORLANDC__ */ + + +/* Suggest testing for specific compiler first before testing for + * FAR. The Watcom compiler defines both __MEDIUM__ and M_I86MM, + * making reliance oncertain keywords suspect. (SJT) + */ + +/* MSC Medium model */ +#if defined(FAR) +# if defined(M_I86MM) +# define USE_FAR_KEYWORD +# define FARDATA FAR +# include +# endif +#endif + +/* SJT: default case */ +#ifndef FAR +# define FAR +#endif + +/* At this point FAR is always defined */ +#ifndef FARDATA +# define FARDATA +#endif + +/* Typedef for floating-point numbers that are converted + to fixed-point with a multiple of 100,000, e.g., int_gamma */ +typedef png_int_32 png_fixed_point; + +/* Add typedefs for pointers */ +typedef void FAR * png_voidp; +typedef png_byte FAR * png_bytep; +typedef png_uint_32 FAR * png_uint_32p; +typedef png_int_32 FAR * png_int_32p; +typedef png_uint_16 FAR * png_uint_16p; +typedef png_int_16 FAR * png_int_16p; +typedef PNG_CONST char FAR * png_const_charp; +typedef char FAR * png_charp; +typedef png_fixed_point FAR * png_fixed_point_p; + +#ifndef PNG_NO_STDIO +#if defined(_WIN32_WCE) +typedef HANDLE png_FILE_p; +#else +typedef FILE * png_FILE_p; +#endif +#endif + +#ifdef PNG_FLOATING_POINT_SUPPORTED +typedef double FAR * png_doublep; +#endif + +/* Pointers to pointers; i.e. arrays */ +typedef png_byte FAR * FAR * png_bytepp; +typedef png_uint_32 FAR * FAR * png_uint_32pp; +typedef png_int_32 FAR * FAR * png_int_32pp; +typedef png_uint_16 FAR * FAR * png_uint_16pp; +typedef png_int_16 FAR * FAR * png_int_16pp; +typedef PNG_CONST char FAR * FAR * png_const_charpp; +typedef char FAR * FAR * png_charpp; +typedef png_fixed_point FAR * FAR * png_fixed_point_pp; +#ifdef PNG_FLOATING_POINT_SUPPORTED +typedef double FAR * FAR * png_doublepp; +#endif + +/* Pointers to pointers to pointers; i.e., pointer to array */ +typedef char FAR * FAR * FAR * png_charppp; + +#if defined(PNG_1_0_X) || defined(PNG_1_2_X) +/* SPC - Is this stuff deprecated? */ +/* It'll be removed as of libpng-1.3.0 - GR-P */ +/* libpng typedefs for types in zlib. If zlib changes + * or another compression library is used, then change these. + * Eliminates need to change all the source files. + */ +typedef charf * png_zcharp; +typedef charf * FAR * png_zcharpp; +typedef z_stream FAR * png_zstreamp; +#endif /* (PNG_1_0_X) || defined(PNG_1_2_X) */ + +/* + * Define PNG_BUILD_DLL if the module being built is a Windows + * LIBPNG DLL. + * + * Define PNG_USE_DLL if you want to *link* to the Windows LIBPNG DLL. + * It is equivalent to Microsoft predefined macro _DLL that is + * automatically defined when you compile using the share + * version of the CRT (C Run-Time library) + * + * The cygwin mods make this behavior a little different: + * Define PNG_BUILD_DLL if you are building a dll for use with cygwin + * Define PNG_STATIC if you are building a static library for use with cygwin, + * -or- if you are building an application that you want to link to the + * static library. + * PNG_USE_DLL is defined by default (no user action needed) unless one of + * the other flags is defined. + */ + +#if !defined(PNG_DLL) && (defined(PNG_BUILD_DLL) || defined(PNG_USE_DLL)) +# define PNG_DLL +#endif +/* If CYGWIN, then disallow GLOBAL ARRAYS unless building a static lib. + * When building a static lib, default to no GLOBAL ARRAYS, but allow + * command-line override + */ +#if defined(__CYGWIN__) +# if !defined(PNG_STATIC) +# if defined(PNG_USE_GLOBAL_ARRAYS) +# undef PNG_USE_GLOBAL_ARRAYS +# endif +# if !defined(PNG_USE_LOCAL_ARRAYS) +# define PNG_USE_LOCAL_ARRAYS +# endif +# else +# if defined(PNG_USE_LOCAL_ARRAYS) || defined(PNG_NO_GLOBAL_ARRAYS) +# if defined(PNG_USE_GLOBAL_ARRAYS) +# undef PNG_USE_GLOBAL_ARRAYS +# endif +# endif +# endif +# if !defined(PNG_USE_LOCAL_ARRAYS) && !defined(PNG_USE_GLOBAL_ARRAYS) +# define PNG_USE_LOCAL_ARRAYS +# endif +#endif + +/* Do not use global arrays (helps with building DLL's) + * They are no longer used in libpng itself, since version 1.0.5c, + * but might be required for some pre-1.0.5c applications. + */ +#if !defined(PNG_USE_LOCAL_ARRAYS) && !defined(PNG_USE_GLOBAL_ARRAYS) +# if defined(PNG_NO_GLOBAL_ARRAYS) || (defined(__GNUC__) && defined(PNG_DLL)) +# define PNG_USE_LOCAL_ARRAYS +# else +# define PNG_USE_GLOBAL_ARRAYS +# endif +#endif + +#if defined(__CYGWIN__) +# undef PNGAPI +# define PNGAPI __cdecl +# undef PNG_IMPEXP +# define PNG_IMPEXP +#endif + +/* If you define PNGAPI, e.g., with compiler option "-DPNGAPI=__stdcall", + * you may get warnings regarding the linkage of png_zalloc and png_zfree. + * Don't ignore those warnings; you must also reset the default calling + * convention in your compiler to match your PNGAPI, and you must build + * zlib and your applications the same way you build libpng. + */ + +#if defined(__MINGW32__) && !defined(PNG_MODULEDEF) +# ifndef PNG_NO_MODULEDEF +# define PNG_NO_MODULEDEF +# endif +#endif + +#if !defined(PNG_IMPEXP) && defined(PNG_BUILD_DLL) && !defined(PNG_NO_MODULEDEF) +# define PNG_IMPEXP +#endif + +#if defined(PNG_DLL) || defined(_DLL) || defined(__DLL__ ) || \ + (( defined(_Windows) || defined(_WINDOWS) || \ + defined(WIN32) || defined(_WIN32) || defined(__WIN32__) )) + +# ifndef PNGAPI +# if defined(__GNUC__) || (defined (_MSC_VER) && (_MSC_VER >= 800)) +# define PNGAPI __cdecl +# else +# define PNGAPI _cdecl +# endif +# endif + +# if !defined(PNG_IMPEXP) && (!defined(PNG_DLL) || \ + 0 /* WINCOMPILER_WITH_NO_SUPPORT_FOR_DECLIMPEXP */) +# define PNG_IMPEXP +# endif + +# if !defined(PNG_IMPEXP) + +# define PNG_EXPORT_TYPE1(type,symbol) PNG_IMPEXP type PNGAPI symbol +# define PNG_EXPORT_TYPE2(type,symbol) type PNG_IMPEXP PNGAPI symbol + + /* Borland/Microsoft */ +# if defined(_MSC_VER) || defined(__BORLANDC__) +# if (_MSC_VER >= 800) || (__BORLANDC__ >= 0x500) +# define PNG_EXPORT PNG_EXPORT_TYPE1 +# else +# define PNG_EXPORT PNG_EXPORT_TYPE2 +# if defined(PNG_BUILD_DLL) +# define PNG_IMPEXP __export +# else +# define PNG_IMPEXP /*__import */ /* doesn't exist AFAIK in + VC++ */ +# endif /* Exists in Borland C++ for + C++ classes (== huge) */ +# endif +# endif + +# if !defined(PNG_IMPEXP) +# if defined(PNG_BUILD_DLL) +# define PNG_IMPEXP __declspec(dllexport) +# else +# define PNG_IMPEXP __declspec(dllimport) +# endif +# endif +# endif /* PNG_IMPEXP */ +#else /* !(DLL || non-cygwin WINDOWS) */ +# if (defined(__IBMC__) || defined(__IBMCPP__)) && defined(__OS2__) +# ifndef PNGAPI +# define PNGAPI _System +# endif +# else +# if 0 /* ... other platforms, with other meanings */ +# endif +# endif +#endif + +#ifndef PNGAPI +# define PNGAPI +#endif +#ifndef PNG_IMPEXP +# define PNG_IMPEXP +#endif + +#ifdef PNG_BUILDSYMS +# ifndef PNG_EXPORT +# define PNG_EXPORT(type,symbol) PNG_FUNCTION_EXPORT symbol END +# endif +# ifdef PNG_USE_GLOBAL_ARRAYS +# ifndef PNG_EXPORT_VAR +# define PNG_EXPORT_VAR(type) PNG_DATA_EXPORT +# endif +# endif +#endif + +#ifndef PNG_EXPORT +# define PNG_EXPORT(type,symbol) PNG_IMPEXP type PNGAPI symbol +#endif + +#ifdef PNG_USE_GLOBAL_ARRAYS +# ifndef PNG_EXPORT_VAR +# define PNG_EXPORT_VAR(type) extern PNG_IMPEXP type +# endif +#endif + +/* User may want to use these so they are not in PNG_INTERNAL. Any library + * functions that are passed far data must be model independent. + */ + +#ifndef PNG_ABORT +# define PNG_ABORT() abort() +#endif + +#ifdef PNG_SETJMP_SUPPORTED +# define png_jmpbuf(png_ptr) ((png_ptr)->jmpbuf) +#else +# define png_jmpbuf(png_ptr) \ + (LIBPNG_WAS_COMPILED_WITH__PNG_SETJMP_NOT_SUPPORTED) +#endif + +#if defined(USE_FAR_KEYWORD) /* memory model independent fns */ +/* use this to make far-to-near assignments */ +# define CHECK 1 +# define NOCHECK 0 +# define CVT_PTR(ptr) (png_far_to_near(png_ptr,ptr,CHECK)) +# define CVT_PTR_NOCHECK(ptr) (png_far_to_near(png_ptr,ptr,NOCHECK)) +# define png_strcpy _fstrcpy +# define png_strncpy _fstrncpy /* Added to v 1.2.6 */ +# define png_strlen _fstrlen +# define png_memcmp _fmemcmp /* SJT: added */ +# define png_memcpy _fmemcpy +# define png_memset _fmemset +#else /* use the usual functions */ +# define CVT_PTR(ptr) (ptr) +# define CVT_PTR_NOCHECK(ptr) (ptr) +# define png_strcpy strcpy +# define png_strncpy strncpy /* Added to v 1.2.6 */ +# define png_strlen strlen +# define png_memcmp memcmp /* SJT: added */ +# define png_memcpy memcpy +# define png_memset memset +#endif +/* End of memory model independent support */ + +/* Just a little check that someone hasn't tried to define something + * contradictory. + */ +#if (PNG_ZBUF_SIZE > 65536L) && defined(PNG_MAX_MALLOC_64K) +# undef PNG_ZBUF_SIZE +# define PNG_ZBUF_SIZE 65536L +#endif + +#ifdef PNG_READ_SUPPORTED +/* Prior to libpng-1.0.9, this block was in pngasmrd.h */ +#if defined(PNG_INTERNAL) + +/* These are the default thresholds before the MMX code kicks in; if either + * rowbytes or bitdepth is below the threshold, plain C code is used. These + * can be overridden at runtime via the png_set_mmx_thresholds() call in + * libpng 1.2.0 and later. The values below were chosen by Intel. + */ + +#ifndef PNG_MMX_ROWBYTES_THRESHOLD_DEFAULT +# define PNG_MMX_ROWBYTES_THRESHOLD_DEFAULT 128 /* >= */ +#endif +#ifndef PNG_MMX_BITDEPTH_THRESHOLD_DEFAULT +# define PNG_MMX_BITDEPTH_THRESHOLD_DEFAULT 9 /* >= */ +#endif + +/* Set this in the makefile for VC++ on Pentium, not here. */ +/* Platform must be Pentium. Makefile must assemble and load pngvcrd.c . + * MMX will be detected at run time and used if present. + */ +#ifdef PNG_USE_PNGVCRD +# define PNG_HAVE_ASSEMBLER_COMBINE_ROW +# define PNG_HAVE_ASSEMBLER_READ_INTERLACE +# define PNG_HAVE_ASSEMBLER_READ_FILTER_ROW +#endif + +/* Set this in the makefile for gcc/as on Pentium, not here. */ +/* Platform must be Pentium. Makefile must assemble and load pnggccrd.c . + * MMX will be detected at run time and used if present. + */ +#ifdef PNG_USE_PNGGCCRD +# define PNG_HAVE_ASSEMBLER_COMBINE_ROW +# define PNG_HAVE_ASSEMBLER_READ_INTERLACE +# define PNG_HAVE_ASSEMBLER_READ_FILTER_ROW +#endif +/* - see pnggccrd.c for info about what is currently enabled */ + +#endif /* PNG_INTERNAL */ +#endif /* PNG_READ_SUPPORTED */ + +/* Added at libpng-1.2.8 */ +#endif /* PNG_VERSION_INFO_ONLY */ + +#endif /* PNGCONF_H */ diff --git a/src/win32/dependencies/libpng/pngerror.c b/src/win32/dependencies/libpng/pngerror.c new file mode 100644 index 00000000..ad6ae0e8 --- /dev/null +++ b/src/win32/dependencies/libpng/pngerror.c @@ -0,0 +1,313 @@ + +/* pngerror.c - stub functions for i/o and memory allocation + * + * Last changed in libpng 1.2.9 April 14, 2006 + * For conditions of distribution and use, see copyright notice in png.h + * Copyright (c) 1998-2006 Glenn Randers-Pehrson + * (Version 0.96 Copyright (c) 1996, 1997 Andreas Dilger) + * (Version 0.88 Copyright (c) 1995, 1996 Guy Eric Schalnat, Group 42, Inc.) + * + * This file provides a location for all error handling. Users who + * need special error handling are expected to write replacement functions + * and use png_set_error_fn() to use those functions. See the instructions + * at each function. + */ + +#define PNG_INTERNAL +#include "png.h" + +#if defined(PNG_READ_SUPPORTED) || defined(PNG_WRITE_SUPPORTED) +static void /* PRIVATE */ +png_default_error PNGARG((png_structp png_ptr, + png_const_charp error_message)); +static void /* PRIVATE */ +png_default_warning PNGARG((png_structp png_ptr, + png_const_charp warning_message)); + +/* This function is called whenever there is a fatal error. This function + * should not be changed. If there is a need to handle errors differently, + * you should supply a replacement error function and use png_set_error_fn() + * to replace the error function at run-time. + */ +void PNGAPI +png_error(png_structp png_ptr, png_const_charp error_message) +{ +#ifdef PNG_ERROR_NUMBERS_SUPPORTED + char msg[16]; + if (png_ptr != NULL) + { + if (png_ptr->flags& + (PNG_FLAG_STRIP_ERROR_NUMBERS|PNG_FLAG_STRIP_ERROR_TEXT)) + { + if (*error_message == '#') + { + int offset; + for (offset=1; offset<15; offset++) + if (*(error_message+offset) == ' ') + break; + if (png_ptr->flags&PNG_FLAG_STRIP_ERROR_TEXT) + { + int i; + for (i=0; iflags&PNG_FLAG_STRIP_ERROR_TEXT) + { + msg[0]='0'; + msg[1]='\0'; + error_message=msg; + } + } + } + } +#endif + if (png_ptr != NULL && png_ptr->error_fn != NULL) + (*(png_ptr->error_fn))(png_ptr, error_message); + + /* If the custom handler doesn't exist, or if it returns, + use the default handler, which will not return. */ + png_default_error(png_ptr, error_message); +} + +/* This function is called whenever there is a non-fatal error. This function + * should not be changed. If there is a need to handle warnings differently, + * you should supply a replacement warning function and use + * png_set_error_fn() to replace the warning function at run-time. + */ +void PNGAPI +png_warning(png_structp png_ptr, png_const_charp warning_message) +{ + int offset = 0; + if (png_ptr != NULL) + { +#ifdef PNG_ERROR_NUMBERS_SUPPORTED + if (png_ptr->flags& + (PNG_FLAG_STRIP_ERROR_NUMBERS|PNG_FLAG_STRIP_ERROR_TEXT)) +#endif + { + if (*warning_message == '#') + { + for (offset=1; offset<15; offset++) + if (*(warning_message+offset) == ' ') + break; + } + } + if (png_ptr != NULL && png_ptr->warning_fn != NULL) + (*(png_ptr->warning_fn))(png_ptr, warning_message+offset); + } + else + png_default_warning(png_ptr, warning_message+offset); +} + +/* These utilities are used internally to build an error message that relates + * to the current chunk. The chunk name comes from png_ptr->chunk_name, + * this is used to prefix the message. The message is limited in length + * to 63 bytes, the name characters are output as hex digits wrapped in [] + * if the character is invalid. + */ +#define isnonalpha(c) ((c) < 65 || (c) > 122 || ((c) > 90 && (c) < 97)) +static PNG_CONST char png_digit[16] = { + '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', + 'A', 'B', 'C', 'D', 'E', 'F' +}; + +static void /* PRIVATE */ +png_format_buffer(png_structp png_ptr, png_charp buffer, png_const_charp + error_message) +{ + int iout = 0, iin = 0; + + while (iin < 4) + { + int c = png_ptr->chunk_name[iin++]; + if (isnonalpha(c)) + { + buffer[iout++] = '['; + buffer[iout++] = png_digit[(c & 0xf0) >> 4]; + buffer[iout++] = png_digit[c & 0x0f]; + buffer[iout++] = ']'; + } + else + { + buffer[iout++] = (png_byte)c; + } + } + + if (error_message == NULL) + buffer[iout] = 0; + else + { + buffer[iout++] = ':'; + buffer[iout++] = ' '; + png_strncpy(buffer+iout, error_message, 63); + buffer[iout+63] = 0; + } +} + +void PNGAPI +png_chunk_error(png_structp png_ptr, png_const_charp error_message) +{ + char msg[18+64]; + if (png_ptr == NULL) + png_error(png_ptr, error_message); + png_format_buffer(png_ptr, msg, error_message); + png_error(png_ptr, msg); +} + +void PNGAPI +png_chunk_warning(png_structp png_ptr, png_const_charp warning_message) +{ + char msg[18+64]; + if (png_ptr == NULL) + png_warning(png_ptr, warning_message); + png_format_buffer(png_ptr, msg, warning_message); + png_warning(png_ptr, msg); +} + +/* This is the default error handling function. Note that replacements for + * this function MUST NOT RETURN, or the program will likely crash. This + * function is used by default, or if the program supplies NULL for the + * error function pointer in png_set_error_fn(). + */ +static void /* PRIVATE */ +png_default_error(png_structp png_ptr, png_const_charp error_message) +{ +#ifndef PNG_NO_CONSOLE_IO +#ifdef PNG_ERROR_NUMBERS_SUPPORTED + if (*error_message == '#') + { + int offset; + char error_number[16]; + for (offset=0; offset<15; offset++) + { + error_number[offset] = *(error_message+offset+1); + if (*(error_message+offset) == ' ') + break; + } + if((offset > 1) && (offset < 15)) + { + error_number[offset-1]='\0'; + fprintf(stderr, "libpng error no. %s: %s\n", error_number, + error_message+offset); + } + else + fprintf(stderr, "libpng error: %s, offset=%d\n", error_message,offset); + } + else +#endif + fprintf(stderr, "libpng error: %s\n", error_message); +#endif + +#ifdef PNG_SETJMP_SUPPORTED +# ifdef USE_FAR_KEYWORD + { + jmp_buf jmpbuf; + png_memcpy(jmpbuf,png_ptr->jmpbuf,png_sizeof(jmp_buf)); + longjmp(jmpbuf, 1); + } +# else + longjmp(png_ptr->jmpbuf, 1); +# endif +#else + /* make compiler happy */ ; + if (png_ptr) + PNG_ABORT(); +#endif +#ifdef PNG_NO_CONSOLE_IO + /* make compiler happy */ ; + if (&error_message != NULL) + return; +#endif +} + +/* This function is called when there is a warning, but the library thinks + * it can continue anyway. Replacement functions don't have to do anything + * here if you don't want them to. In the default configuration, png_ptr is + * not used, but it is passed in case it may be useful. + */ +static void /* PRIVATE */ +png_default_warning(png_structp png_ptr, png_const_charp warning_message) +{ +#ifndef PNG_NO_CONSOLE_IO +# ifdef PNG_ERROR_NUMBERS_SUPPORTED + if (*warning_message == '#') + { + int offset; + char warning_number[16]; + for (offset=0; offset<15; offset++) + { + warning_number[offset]=*(warning_message+offset+1); + if (*(warning_message+offset) == ' ') + break; + } + if((offset > 1) && (offset < 15)) + { + warning_number[offset-1]='\0'; + fprintf(stderr, "libpng warning no. %s: %s\n", warning_number, + warning_message+offset); + } + else + fprintf(stderr, "libpng warning: %s\n", warning_message); + } + else +# endif + fprintf(stderr, "libpng warning: %s\n", warning_message); +#else + /* make compiler happy */ ; + if (warning_message) + return; +#endif + /* make compiler happy */ ; + if (png_ptr) + return; +} + +/* This function is called when the application wants to use another method + * of handling errors and warnings. Note that the error function MUST NOT + * return to the calling routine or serious problems will occur. The return + * method used in the default routine calls longjmp(png_ptr->jmpbuf, 1) + */ +void PNGAPI +png_set_error_fn(png_structp png_ptr, png_voidp error_ptr, + png_error_ptr error_fn, png_error_ptr warning_fn) +{ + if (png_ptr == NULL) + return; + png_ptr->error_ptr = error_ptr; + png_ptr->error_fn = error_fn; + png_ptr->warning_fn = warning_fn; +} + + +/* This function returns a pointer to the error_ptr associated with the user + * functions. The application should free any memory associated with this + * pointer before png_write_destroy and png_read_destroy are called. + */ +png_voidp PNGAPI +png_get_error_ptr(png_structp png_ptr) +{ + if (png_ptr == NULL) + return NULL; + return ((png_voidp)png_ptr->error_ptr); +} + + +#ifdef PNG_ERROR_NUMBERS_SUPPORTED +void PNGAPI +png_set_strip_error_numbers(png_structp png_ptr, png_uint_32 strip_mode) +{ + if(png_ptr != NULL) + { + png_ptr->flags &= + ((~(PNG_FLAG_STRIP_ERROR_NUMBERS|PNG_FLAG_STRIP_ERROR_TEXT))&strip_mode); + } +} +#endif +#endif /* PNG_READ_SUPPORTED || PNG_WRITE_SUPPORTED */ diff --git a/src/win32/dependencies/libpng/pngget.c b/src/win32/dependencies/libpng/pngget.c new file mode 100644 index 00000000..df765858 --- /dev/null +++ b/src/win32/dependencies/libpng/pngget.c @@ -0,0 +1,937 @@ + +/* pngget.c - retrieval of values from info struct + * + * Last changed in libpng 1.2.9 April 14, 2006 + * For conditions of distribution and use, see copyright notice in png.h + * Copyright (c) 1998-2006 Glenn Randers-Pehrson + * (Version 0.96 Copyright (c) 1996, 1997 Andreas Dilger) + * (Version 0.88 Copyright (c) 1995, 1996 Guy Eric Schalnat, Group 42, Inc.) + */ + +#define PNG_INTERNAL +#include "png.h" + +#if defined(PNG_READ_SUPPORTED) || defined(PNG_WRITE_SUPPORTED) + +png_uint_32 PNGAPI +png_get_valid(png_structp png_ptr, png_infop info_ptr, png_uint_32 flag) +{ + if (png_ptr != NULL && info_ptr != NULL) + return(info_ptr->valid & flag); + else + return(0); +} + +png_uint_32 PNGAPI +png_get_rowbytes(png_structp png_ptr, png_infop info_ptr) +{ + if (png_ptr != NULL && info_ptr != NULL) + return(info_ptr->rowbytes); + else + return(0); +} + +#if defined(PNG_INFO_IMAGE_SUPPORTED) +png_bytepp PNGAPI +png_get_rows(png_structp png_ptr, png_infop info_ptr) +{ + if (png_ptr != NULL && info_ptr != NULL) + return(info_ptr->row_pointers); + else + return(0); +} +#endif + +#ifdef PNG_EASY_ACCESS_SUPPORTED +/* easy access to info, added in libpng-0.99 */ +png_uint_32 PNGAPI +png_get_image_width(png_structp png_ptr, png_infop info_ptr) +{ + if (png_ptr != NULL && info_ptr != NULL) + { + return info_ptr->width; + } + return (0); +} + +png_uint_32 PNGAPI +png_get_image_height(png_structp png_ptr, png_infop info_ptr) +{ + if (png_ptr != NULL && info_ptr != NULL) + { + return info_ptr->height; + } + return (0); +} + +png_byte PNGAPI +png_get_bit_depth(png_structp png_ptr, png_infop info_ptr) +{ + if (png_ptr != NULL && info_ptr != NULL) + { + return info_ptr->bit_depth; + } + return (0); +} + +png_byte PNGAPI +png_get_color_type(png_structp png_ptr, png_infop info_ptr) +{ + if (png_ptr != NULL && info_ptr != NULL) + { + return info_ptr->color_type; + } + return (0); +} + +png_byte PNGAPI +png_get_filter_type(png_structp png_ptr, png_infop info_ptr) +{ + if (png_ptr != NULL && info_ptr != NULL) + { + return info_ptr->filter_type; + } + return (0); +} + +png_byte PNGAPI +png_get_interlace_type(png_structp png_ptr, png_infop info_ptr) +{ + if (png_ptr != NULL && info_ptr != NULL) + { + return info_ptr->interlace_type; + } + return (0); +} + +png_byte PNGAPI +png_get_compression_type(png_structp png_ptr, png_infop info_ptr) +{ + if (png_ptr != NULL && info_ptr != NULL) + { + return info_ptr->compression_type; + } + return (0); +} + +png_uint_32 PNGAPI +png_get_x_pixels_per_meter(png_structp png_ptr, png_infop info_ptr) +{ + if (png_ptr != NULL && info_ptr != NULL) +#if defined(PNG_pHYs_SUPPORTED) + if (info_ptr->valid & PNG_INFO_pHYs) + { + png_debug1(1, "in %s retrieval function\n", "png_get_x_pixels_per_meter"); + if(info_ptr->phys_unit_type != PNG_RESOLUTION_METER) + return (0); + else return (info_ptr->x_pixels_per_unit); + } +#else + return (0); +#endif + return (0); +} + +png_uint_32 PNGAPI +png_get_y_pixels_per_meter(png_structp png_ptr, png_infop info_ptr) +{ + if (png_ptr != NULL && info_ptr != NULL) +#if defined(PNG_pHYs_SUPPORTED) + if (info_ptr->valid & PNG_INFO_pHYs) + { + png_debug1(1, "in %s retrieval function\n", "png_get_y_pixels_per_meter"); + if(info_ptr->phys_unit_type != PNG_RESOLUTION_METER) + return (0); + else return (info_ptr->y_pixels_per_unit); + } +#else + return (0); +#endif + return (0); +} + +png_uint_32 PNGAPI +png_get_pixels_per_meter(png_structp png_ptr, png_infop info_ptr) +{ + if (png_ptr != NULL && info_ptr != NULL) +#if defined(PNG_pHYs_SUPPORTED) + if (info_ptr->valid & PNG_INFO_pHYs) + { + png_debug1(1, "in %s retrieval function\n", "png_get_pixels_per_meter"); + if(info_ptr->phys_unit_type != PNG_RESOLUTION_METER || + info_ptr->x_pixels_per_unit != info_ptr->y_pixels_per_unit) + return (0); + else return (info_ptr->x_pixels_per_unit); + } +#else + return (0); +#endif + return (0); +} + +#ifdef PNG_FLOATING_POINT_SUPPORTED +float PNGAPI +png_get_pixel_aspect_ratio(png_structp png_ptr, png_infop info_ptr) + { + if (png_ptr != NULL && info_ptr != NULL) +#if defined(PNG_pHYs_SUPPORTED) + if (info_ptr->valid & PNG_INFO_pHYs) + { + png_debug1(1, "in %s retrieval function\n", "png_get_aspect_ratio"); + if (info_ptr->x_pixels_per_unit == 0) + return ((float)0.0); + else + return ((float)((float)info_ptr->y_pixels_per_unit + /(float)info_ptr->x_pixels_per_unit)); + } +#else + return (0.0); +#endif + return ((float)0.0); +} +#endif + +png_int_32 PNGAPI +png_get_x_offset_microns(png_structp png_ptr, png_infop info_ptr) +{ + if (png_ptr != NULL && info_ptr != NULL) +#if defined(PNG_oFFs_SUPPORTED) + if (info_ptr->valid & PNG_INFO_oFFs) + { + png_debug1(1, "in %s retrieval function\n", "png_get_x_offset_microns"); + if(info_ptr->offset_unit_type != PNG_OFFSET_MICROMETER) + return (0); + else return (info_ptr->x_offset); + } +#else + return (0); +#endif + return (0); +} + +png_int_32 PNGAPI +png_get_y_offset_microns(png_structp png_ptr, png_infop info_ptr) +{ + if (png_ptr != NULL && info_ptr != NULL) +#if defined(PNG_oFFs_SUPPORTED) + if (info_ptr->valid & PNG_INFO_oFFs) + { + png_debug1(1, "in %s retrieval function\n", "png_get_y_offset_microns"); + if(info_ptr->offset_unit_type != PNG_OFFSET_MICROMETER) + return (0); + else return (info_ptr->y_offset); + } +#else + return (0); +#endif + return (0); +} + +png_int_32 PNGAPI +png_get_x_offset_pixels(png_structp png_ptr, png_infop info_ptr) +{ + if (png_ptr != NULL && info_ptr != NULL) +#if defined(PNG_oFFs_SUPPORTED) + if (info_ptr->valid & PNG_INFO_oFFs) + { + png_debug1(1, "in %s retrieval function\n", "png_get_x_offset_microns"); + if(info_ptr->offset_unit_type != PNG_OFFSET_PIXEL) + return (0); + else return (info_ptr->x_offset); + } +#else + return (0); +#endif + return (0); +} + +png_int_32 PNGAPI +png_get_y_offset_pixels(png_structp png_ptr, png_infop info_ptr) +{ + if (png_ptr != NULL && info_ptr != NULL) +#if defined(PNG_oFFs_SUPPORTED) + if (info_ptr->valid & PNG_INFO_oFFs) + { + png_debug1(1, "in %s retrieval function\n", "png_get_y_offset_microns"); + if(info_ptr->offset_unit_type != PNG_OFFSET_PIXEL) + return (0); + else return (info_ptr->y_offset); + } +#else + return (0); +#endif + return (0); +} + +#if defined(PNG_INCH_CONVERSIONS) && defined(PNG_FLOATING_POINT_SUPPORTED) +png_uint_32 PNGAPI +png_get_pixels_per_inch(png_structp png_ptr, png_infop info_ptr) +{ + return ((png_uint_32)((float)png_get_pixels_per_meter(png_ptr, info_ptr) + *.0254 +.5)); +} + +png_uint_32 PNGAPI +png_get_x_pixels_per_inch(png_structp png_ptr, png_infop info_ptr) +{ + return ((png_uint_32)((float)png_get_x_pixels_per_meter(png_ptr, info_ptr) + *.0254 +.5)); +} + +png_uint_32 PNGAPI +png_get_y_pixels_per_inch(png_structp png_ptr, png_infop info_ptr) +{ + return ((png_uint_32)((float)png_get_y_pixels_per_meter(png_ptr, info_ptr) + *.0254 +.5)); +} + +float PNGAPI +png_get_x_offset_inches(png_structp png_ptr, png_infop info_ptr) +{ + return ((float)png_get_x_offset_microns(png_ptr, info_ptr) + *.00003937); +} + +float PNGAPI +png_get_y_offset_inches(png_structp png_ptr, png_infop info_ptr) +{ + return ((float)png_get_y_offset_microns(png_ptr, info_ptr) + *.00003937); +} + +#if defined(PNG_pHYs_SUPPORTED) +png_uint_32 PNGAPI +png_get_pHYs_dpi(png_structp png_ptr, png_infop info_ptr, + png_uint_32 *res_x, png_uint_32 *res_y, int *unit_type) +{ + png_uint_32 retval = 0; + + if (png_ptr != NULL && info_ptr != NULL && (info_ptr->valid & PNG_INFO_pHYs)) + { + png_debug1(1, "in %s retrieval function\n", "pHYs"); + if (res_x != NULL) + { + *res_x = info_ptr->x_pixels_per_unit; + retval |= PNG_INFO_pHYs; + } + if (res_y != NULL) + { + *res_y = info_ptr->y_pixels_per_unit; + retval |= PNG_INFO_pHYs; + } + if (unit_type != NULL) + { + *unit_type = (int)info_ptr->phys_unit_type; + retval |= PNG_INFO_pHYs; + if(*unit_type == 1) + { + if (res_x != NULL) *res_x = (png_uint_32)(*res_x * .0254 + .50); + if (res_y != NULL) *res_y = (png_uint_32)(*res_y * .0254 + .50); + } + } + } + return (retval); +} +#endif /* PNG_pHYs_SUPPORTED */ +#endif /* PNG_INCH_CONVERSIONS && PNG_FLOATING_POINT_SUPPORTED */ + +/* png_get_channels really belongs in here, too, but it's been around longer */ + +#endif /* PNG_EASY_ACCESS_SUPPORTED */ + +png_byte PNGAPI +png_get_channels(png_structp png_ptr, png_infop info_ptr) +{ + if (png_ptr != NULL && info_ptr != NULL) + return(info_ptr->channels); + else + return (0); +} + +png_bytep PNGAPI +png_get_signature(png_structp png_ptr, png_infop info_ptr) +{ + if (png_ptr != NULL && info_ptr != NULL) + return(info_ptr->signature); + else + return (NULL); +} + +#if defined(PNG_bKGD_SUPPORTED) +png_uint_32 PNGAPI +png_get_bKGD(png_structp png_ptr, png_infop info_ptr, + png_color_16p *background) +{ + if (png_ptr != NULL && info_ptr != NULL && (info_ptr->valid & PNG_INFO_bKGD) + && background != NULL) + { + png_debug1(1, "in %s retrieval function\n", "bKGD"); + *background = &(info_ptr->background); + return (PNG_INFO_bKGD); + } + return (0); +} +#endif + +#if defined(PNG_cHRM_SUPPORTED) +#ifdef PNG_FLOATING_POINT_SUPPORTED +png_uint_32 PNGAPI +png_get_cHRM(png_structp png_ptr, png_infop info_ptr, + double *white_x, double *white_y, double *red_x, double *red_y, + double *green_x, double *green_y, double *blue_x, double *blue_y) +{ + if (png_ptr != NULL && info_ptr != NULL && (info_ptr->valid & PNG_INFO_cHRM)) + { + png_debug1(1, "in %s retrieval function\n", "cHRM"); + if (white_x != NULL) + *white_x = (double)info_ptr->x_white; + if (white_y != NULL) + *white_y = (double)info_ptr->y_white; + if (red_x != NULL) + *red_x = (double)info_ptr->x_red; + if (red_y != NULL) + *red_y = (double)info_ptr->y_red; + if (green_x != NULL) + *green_x = (double)info_ptr->x_green; + if (green_y != NULL) + *green_y = (double)info_ptr->y_green; + if (blue_x != NULL) + *blue_x = (double)info_ptr->x_blue; + if (blue_y != NULL) + *blue_y = (double)info_ptr->y_blue; + return (PNG_INFO_cHRM); + } + return (0); +} +#endif +#ifdef PNG_FIXED_POINT_SUPPORTED +png_uint_32 PNGAPI +png_get_cHRM_fixed(png_structp png_ptr, png_infop info_ptr, + png_fixed_point *white_x, png_fixed_point *white_y, png_fixed_point *red_x, + png_fixed_point *red_y, png_fixed_point *green_x, png_fixed_point *green_y, + png_fixed_point *blue_x, png_fixed_point *blue_y) +{ + if (png_ptr != NULL && info_ptr != NULL && (info_ptr->valid & PNG_INFO_cHRM)) + { + png_debug1(1, "in %s retrieval function\n", "cHRM"); + if (white_x != NULL) + *white_x = info_ptr->int_x_white; + if (white_y != NULL) + *white_y = info_ptr->int_y_white; + if (red_x != NULL) + *red_x = info_ptr->int_x_red; + if (red_y != NULL) + *red_y = info_ptr->int_y_red; + if (green_x != NULL) + *green_x = info_ptr->int_x_green; + if (green_y != NULL) + *green_y = info_ptr->int_y_green; + if (blue_x != NULL) + *blue_x = info_ptr->int_x_blue; + if (blue_y != NULL) + *blue_y = info_ptr->int_y_blue; + return (PNG_INFO_cHRM); + } + return (0); +} +#endif +#endif + +#if defined(PNG_gAMA_SUPPORTED) +#ifdef PNG_FLOATING_POINT_SUPPORTED +png_uint_32 PNGAPI +png_get_gAMA(png_structp png_ptr, png_infop info_ptr, double *file_gamma) +{ + if (png_ptr != NULL && info_ptr != NULL && (info_ptr->valid & PNG_INFO_gAMA) + && file_gamma != NULL) + { + png_debug1(1, "in %s retrieval function\n", "gAMA"); + *file_gamma = (double)info_ptr->gamma; + return (PNG_INFO_gAMA); + } + return (0); +} +#endif +#ifdef PNG_FIXED_POINT_SUPPORTED +png_uint_32 PNGAPI +png_get_gAMA_fixed(png_structp png_ptr, png_infop info_ptr, + png_fixed_point *int_file_gamma) +{ + if (png_ptr != NULL && info_ptr != NULL && (info_ptr->valid & PNG_INFO_gAMA) + && int_file_gamma != NULL) + { + png_debug1(1, "in %s retrieval function\n", "gAMA"); + *int_file_gamma = info_ptr->int_gamma; + return (PNG_INFO_gAMA); + } + return (0); +} +#endif +#endif + +#if defined(PNG_sRGB_SUPPORTED) +png_uint_32 PNGAPI +png_get_sRGB(png_structp png_ptr, png_infop info_ptr, int *file_srgb_intent) +{ + if (png_ptr != NULL && info_ptr != NULL && (info_ptr->valid & PNG_INFO_sRGB) + && file_srgb_intent != NULL) + { + png_debug1(1, "in %s retrieval function\n", "sRGB"); + *file_srgb_intent = (int)info_ptr->srgb_intent; + return (PNG_INFO_sRGB); + } + return (0); +} +#endif + +#if defined(PNG_iCCP_SUPPORTED) +png_uint_32 PNGAPI +png_get_iCCP(png_structp png_ptr, png_infop info_ptr, + png_charpp name, int *compression_type, + png_charpp profile, png_uint_32 *proflen) +{ + if (png_ptr != NULL && info_ptr != NULL && (info_ptr->valid & PNG_INFO_iCCP) + && name != NULL && profile != NULL && proflen != NULL) + { + png_debug1(1, "in %s retrieval function\n", "iCCP"); + *name = info_ptr->iccp_name; + *profile = info_ptr->iccp_profile; + /* compression_type is a dummy so the API won't have to change + if we introduce multiple compression types later. */ + *proflen = (int)info_ptr->iccp_proflen; + *compression_type = (int)info_ptr->iccp_compression; + return (PNG_INFO_iCCP); + } + return (0); +} +#endif + +#if defined(PNG_sPLT_SUPPORTED) +png_uint_32 PNGAPI +png_get_sPLT(png_structp png_ptr, png_infop info_ptr, + png_sPLT_tpp spalettes) +{ + if (png_ptr != NULL && info_ptr != NULL && spalettes != NULL) + *spalettes = info_ptr->splt_palettes; + return ((png_uint_32)info_ptr->splt_palettes_num); +} +#endif + +#if defined(PNG_hIST_SUPPORTED) +png_uint_32 PNGAPI +png_get_hIST(png_structp png_ptr, png_infop info_ptr, png_uint_16p *hist) +{ + if (png_ptr != NULL && info_ptr != NULL && (info_ptr->valid & PNG_INFO_hIST) + && hist != NULL) + { + png_debug1(1, "in %s retrieval function\n", "hIST"); + *hist = info_ptr->hist; + return (PNG_INFO_hIST); + } + return (0); +} +#endif + +png_uint_32 PNGAPI +png_get_IHDR(png_structp png_ptr, png_infop info_ptr, + png_uint_32 *width, png_uint_32 *height, int *bit_depth, + int *color_type, int *interlace_type, int *compression_type, + int *filter_type) + +{ + if (png_ptr != NULL && info_ptr != NULL && width != NULL && height != NULL && + bit_depth != NULL && color_type != NULL) + { + png_debug1(1, "in %s retrieval function\n", "IHDR"); + *width = info_ptr->width; + *height = info_ptr->height; + *bit_depth = info_ptr->bit_depth; + if (info_ptr->bit_depth < 1 || info_ptr->bit_depth > 16) + png_error(png_ptr, "Invalid bit depth"); + *color_type = info_ptr->color_type; + if (info_ptr->color_type > 6) + png_error(png_ptr, "Invalid color type"); + if (compression_type != NULL) + *compression_type = info_ptr->compression_type; + if (filter_type != NULL) + *filter_type = info_ptr->filter_type; + if (interlace_type != NULL) + *interlace_type = info_ptr->interlace_type; + + /* check for potential overflow of rowbytes */ + if (*width == 0 || *width > PNG_UINT_31_MAX) + png_error(png_ptr, "Invalid image width"); + if (*height == 0 || *height > PNG_UINT_31_MAX) + png_error(png_ptr, "Invalid image height"); + if (info_ptr->width > (PNG_UINT_32_MAX + >> 3) /* 8-byte RGBA pixels */ + - 64 /* bigrowbuf hack */ + - 1 /* filter byte */ + - 7*8 /* rounding of width to multiple of 8 pixels */ + - 8) /* extra max_pixel_depth pad */ + { + png_warning(png_ptr, + "Width too large for libpng to process image data."); + } + return (1); + } + return (0); +} + +#if defined(PNG_oFFs_SUPPORTED) +png_uint_32 PNGAPI +png_get_oFFs(png_structp png_ptr, png_infop info_ptr, + png_int_32 *offset_x, png_int_32 *offset_y, int *unit_type) +{ + if (png_ptr != NULL && info_ptr != NULL && (info_ptr->valid & PNG_INFO_oFFs) + && offset_x != NULL && offset_y != NULL && unit_type != NULL) + { + png_debug1(1, "in %s retrieval function\n", "oFFs"); + *offset_x = info_ptr->x_offset; + *offset_y = info_ptr->y_offset; + *unit_type = (int)info_ptr->offset_unit_type; + return (PNG_INFO_oFFs); + } + return (0); +} +#endif + +#if defined(PNG_pCAL_SUPPORTED) +png_uint_32 PNGAPI +png_get_pCAL(png_structp png_ptr, png_infop info_ptr, + png_charp *purpose, png_int_32 *X0, png_int_32 *X1, int *type, int *nparams, + png_charp *units, png_charpp *params) +{ + if (png_ptr != NULL && info_ptr != NULL && (info_ptr->valid & PNG_INFO_pCAL) + && purpose != NULL && X0 != NULL && X1 != NULL && type != NULL && + nparams != NULL && units != NULL && params != NULL) + { + png_debug1(1, "in %s retrieval function\n", "pCAL"); + *purpose = info_ptr->pcal_purpose; + *X0 = info_ptr->pcal_X0; + *X1 = info_ptr->pcal_X1; + *type = (int)info_ptr->pcal_type; + *nparams = (int)info_ptr->pcal_nparams; + *units = info_ptr->pcal_units; + *params = info_ptr->pcal_params; + return (PNG_INFO_pCAL); + } + return (0); +} +#endif + +#if defined(PNG_sCAL_SUPPORTED) +#ifdef PNG_FLOATING_POINT_SUPPORTED +png_uint_32 PNGAPI +png_get_sCAL(png_structp png_ptr, png_infop info_ptr, + int *unit, double *width, double *height) +{ + if (png_ptr != NULL && info_ptr != NULL && + (info_ptr->valid & PNG_INFO_sCAL)) + { + *unit = info_ptr->scal_unit; + *width = info_ptr->scal_pixel_width; + *height = info_ptr->scal_pixel_height; + return (PNG_INFO_sCAL); + } + return(0); +} +#else +#ifdef PNG_FIXED_POINT_SUPPORTED +png_uint_32 PNGAPI +png_get_sCAL_s(png_structp png_ptr, png_infop info_ptr, + int *unit, png_charpp width, png_charpp height) +{ + if (png_ptr != NULL && info_ptr != NULL && + (info_ptr->valid & PNG_INFO_sCAL)) + { + *unit = info_ptr->scal_unit; + *width = info_ptr->scal_s_width; + *height = info_ptr->scal_s_height; + return (PNG_INFO_sCAL); + } + return(0); +} +#endif +#endif +#endif + +#if defined(PNG_pHYs_SUPPORTED) +png_uint_32 PNGAPI +png_get_pHYs(png_structp png_ptr, png_infop info_ptr, + png_uint_32 *res_x, png_uint_32 *res_y, int *unit_type) +{ + png_uint_32 retval = 0; + + if (png_ptr != NULL && info_ptr != NULL && + (info_ptr->valid & PNG_INFO_pHYs)) + { + png_debug1(1, "in %s retrieval function\n", "pHYs"); + if (res_x != NULL) + { + *res_x = info_ptr->x_pixels_per_unit; + retval |= PNG_INFO_pHYs; + } + if (res_y != NULL) + { + *res_y = info_ptr->y_pixels_per_unit; + retval |= PNG_INFO_pHYs; + } + if (unit_type != NULL) + { + *unit_type = (int)info_ptr->phys_unit_type; + retval |= PNG_INFO_pHYs; + } + } + return (retval); +} +#endif + +png_uint_32 PNGAPI +png_get_PLTE(png_structp png_ptr, png_infop info_ptr, png_colorp *palette, + int *num_palette) +{ + if (png_ptr != NULL && info_ptr != NULL && (info_ptr->valid & PNG_INFO_PLTE) + && palette != NULL) + { + png_debug1(1, "in %s retrieval function\n", "PLTE"); + *palette = info_ptr->palette; + *num_palette = info_ptr->num_palette; + png_debug1(3, "num_palette = %d\n", *num_palette); + return (PNG_INFO_PLTE); + } + return (0); +} + +#if defined(PNG_sBIT_SUPPORTED) +png_uint_32 PNGAPI +png_get_sBIT(png_structp png_ptr, png_infop info_ptr, png_color_8p *sig_bit) +{ + if (png_ptr != NULL && info_ptr != NULL && (info_ptr->valid & PNG_INFO_sBIT) + && sig_bit != NULL) + { + png_debug1(1, "in %s retrieval function\n", "sBIT"); + *sig_bit = &(info_ptr->sig_bit); + return (PNG_INFO_sBIT); + } + return (0); +} +#endif + +#if defined(PNG_TEXT_SUPPORTED) +png_uint_32 PNGAPI +png_get_text(png_structp png_ptr, png_infop info_ptr, png_textp *text_ptr, + int *num_text) +{ + if (png_ptr != NULL && info_ptr != NULL && info_ptr->num_text > 0) + { + png_debug1(1, "in %s retrieval function\n", + (png_ptr->chunk_name[0] == '\0' ? "text" + : (png_const_charp)png_ptr->chunk_name)); + if (text_ptr != NULL) + *text_ptr = info_ptr->text; + if (num_text != NULL) + *num_text = info_ptr->num_text; + return ((png_uint_32)info_ptr->num_text); + } + if (num_text != NULL) + *num_text = 0; + return(0); +} +#endif + +#if defined(PNG_tIME_SUPPORTED) +png_uint_32 PNGAPI +png_get_tIME(png_structp png_ptr, png_infop info_ptr, png_timep *mod_time) +{ + if (png_ptr != NULL && info_ptr != NULL && (info_ptr->valid & PNG_INFO_tIME) + && mod_time != NULL) + { + png_debug1(1, "in %s retrieval function\n", "tIME"); + *mod_time = &(info_ptr->mod_time); + return (PNG_INFO_tIME); + } + return (0); +} +#endif + +#if defined(PNG_tRNS_SUPPORTED) +png_uint_32 PNGAPI +png_get_tRNS(png_structp png_ptr, png_infop info_ptr, + png_bytep *trans, int *num_trans, png_color_16p *trans_values) +{ + png_uint_32 retval = 0; + if (png_ptr != NULL && info_ptr != NULL && (info_ptr->valid & PNG_INFO_tRNS)) + { + png_debug1(1, "in %s retrieval function\n", "tRNS"); + if (info_ptr->color_type == PNG_COLOR_TYPE_PALETTE) + { + if (trans != NULL) + { + *trans = info_ptr->trans; + retval |= PNG_INFO_tRNS; + } + if (trans_values != NULL) + *trans_values = &(info_ptr->trans_values); + } + else /* if (info_ptr->color_type != PNG_COLOR_TYPE_PALETTE) */ + { + if (trans_values != NULL) + { + *trans_values = &(info_ptr->trans_values); + retval |= PNG_INFO_tRNS; + } + if(trans != NULL) + *trans = NULL; + } + if(num_trans != NULL) + { + *num_trans = info_ptr->num_trans; + retval |= PNG_INFO_tRNS; + } + } + return (retval); +} +#endif + +#if defined(PNG_UNKNOWN_CHUNKS_SUPPORTED) +png_uint_32 PNGAPI +png_get_unknown_chunks(png_structp png_ptr, png_infop info_ptr, + png_unknown_chunkpp unknowns) +{ + if (png_ptr != NULL && info_ptr != NULL && unknowns != NULL) + *unknowns = info_ptr->unknown_chunks; + return ((png_uint_32)info_ptr->unknown_chunks_num); +} +#endif + +#if defined(PNG_READ_RGB_TO_GRAY_SUPPORTED) +png_byte PNGAPI +png_get_rgb_to_gray_status (png_structp png_ptr) +{ + return (png_byte)(png_ptr? png_ptr->rgb_to_gray_status : 0); +} +#endif + +#if defined(PNG_USER_CHUNKS_SUPPORTED) +png_voidp PNGAPI +png_get_user_chunk_ptr(png_structp png_ptr) +{ + return (png_ptr? png_ptr->user_chunk_ptr : NULL); +} +#endif + +#ifdef PNG_WRITE_SUPPORTED +png_uint_32 PNGAPI +png_get_compression_buffer_size(png_structp png_ptr) +{ + return (png_uint_32)(png_ptr? png_ptr->zbuf_size : 0L); +} +#endif + +#ifndef PNG_1_0_X +#ifdef PNG_ASSEMBLER_CODE_SUPPORTED +/* this function was added to libpng 1.2.0 and should exist by default */ +png_uint_32 PNGAPI +png_get_asm_flags (png_structp png_ptr) +{ + return (png_uint_32)(png_ptr? png_ptr->asm_flags : 0L); +} + +/* this function was added to libpng 1.2.0 and should exist by default */ +png_uint_32 PNGAPI +png_get_asm_flagmask (int flag_select) +{ + png_uint_32 settable_asm_flags = 0; + + if (flag_select & PNG_SELECT_READ) + settable_asm_flags |= + PNG_ASM_FLAG_MMX_READ_COMBINE_ROW | + PNG_ASM_FLAG_MMX_READ_INTERLACE | + PNG_ASM_FLAG_MMX_READ_FILTER_SUB | + PNG_ASM_FLAG_MMX_READ_FILTER_UP | + PNG_ASM_FLAG_MMX_READ_FILTER_AVG | + PNG_ASM_FLAG_MMX_READ_FILTER_PAETH ; + /* no non-MMX flags yet */ + +#if 0 + /* GRR: no write-flags yet, either, but someday... */ + if (flag_select & PNG_SELECT_WRITE) + settable_asm_flags |= + PNG_ASM_FLAG_MMX_WRITE_ [whatever] ; +#endif /* 0 */ + + return settable_asm_flags; /* _theoretically_ settable capabilities only */ +} +#endif /* PNG_ASSEMBLER_CODE_SUPPORTED */ + + +#if defined(PNG_ASSEMBLER_CODE_SUPPORTED) + /* GRR: could add this: && defined(PNG_MMX_CODE_SUPPORTED) */ +/* this function was added to libpng 1.2.0 */ +png_uint_32 PNGAPI +png_get_mmx_flagmask (int flag_select, int *compilerID) +{ + png_uint_32 settable_mmx_flags = 0; + + if (flag_select & PNG_SELECT_READ) + settable_mmx_flags |= + PNG_ASM_FLAG_MMX_READ_COMBINE_ROW | + PNG_ASM_FLAG_MMX_READ_INTERLACE | + PNG_ASM_FLAG_MMX_READ_FILTER_SUB | + PNG_ASM_FLAG_MMX_READ_FILTER_UP | + PNG_ASM_FLAG_MMX_READ_FILTER_AVG | + PNG_ASM_FLAG_MMX_READ_FILTER_PAETH ; +#if 0 + /* GRR: no MMX write support yet, but someday... */ + if (flag_select & PNG_SELECT_WRITE) + settable_mmx_flags |= + PNG_ASM_FLAG_MMX_WRITE_ [whatever] ; +#endif /* 0 */ + + if (compilerID != NULL) { +#ifdef PNG_USE_PNGVCRD + *compilerID = 1; /* MSVC */ +#else +#ifdef PNG_USE_PNGGCCRD + *compilerID = 2; /* gcc/gas */ +#else + *compilerID = -1; /* unknown (i.e., no asm/MMX code compiled) */ +#endif +#endif + } + + return settable_mmx_flags; /* _theoretically_ settable capabilities only */ +} + +/* this function was added to libpng 1.2.0 */ +png_byte PNGAPI +png_get_mmx_bitdepth_threshold (png_structp png_ptr) +{ + return (png_byte)(png_ptr? png_ptr->mmx_bitdepth_threshold : 0); +} + +/* this function was added to libpng 1.2.0 */ +png_uint_32 PNGAPI +png_get_mmx_rowbytes_threshold (png_structp png_ptr) +{ + return (png_uint_32)(png_ptr? png_ptr->mmx_rowbytes_threshold : 0L); +} +#endif /* ?PNG_ASSEMBLER_CODE_SUPPORTED */ +#endif /* ?PNG_1_0_X */ + +#ifdef PNG_SET_USER_LIMITS_SUPPORTED +/* these functions were added to libpng 1.2.6 */ +png_uint_32 PNGAPI +png_get_user_width_max (png_structp png_ptr) +{ + return (png_ptr? png_ptr->user_width_max : 0); +} +png_uint_32 PNGAPI +png_get_user_height_max (png_structp png_ptr) +{ + return (png_ptr? png_ptr->user_height_max : 0); +} +#endif /* ?PNG_SET_USER_LIMITS_SUPPORTED */ + +#endif /* PNG_READ_SUPPORTED || PNG_WRITE_SUPPORTED */ diff --git a/src/win32/dependencies/libpng/pngmem.c b/src/win32/dependencies/libpng/pngmem.c new file mode 100644 index 00000000..cfe3e0fd --- /dev/null +++ b/src/win32/dependencies/libpng/pngmem.c @@ -0,0 +1,598 @@ + +/* pngmem.c - stub functions for memory allocation + * + * Last changed in libpng 1.2.9 April 14, 2006 + * For conditions of distribution and use, see copyright notice in png.h + * Copyright (c) 1998-2006 Glenn Randers-Pehrson + * (Version 0.96 Copyright (c) 1996, 1997 Andreas Dilger) + * (Version 0.88 Copyright (c) 1995, 1996 Guy Eric Schalnat, Group 42, Inc.) + * + * This file provides a location for all memory allocation. Users who + * need special memory handling are expected to supply replacement + * functions for png_malloc() and png_free(), and to use + * png_create_read_struct_2() and png_create_write_struct_2() to + * identify the replacement functions. + */ + +#define PNG_INTERNAL +#include "png.h" + +#if defined(PNG_READ_SUPPORTED) || defined(PNG_WRITE_SUPPORTED) + +/* Borland DOS special memory handler */ +#if defined(__TURBOC__) && !defined(_Windows) && !defined(__FLAT__) +/* if you change this, be sure to change the one in png.h also */ + +/* Allocate memory for a png_struct. The malloc and memset can be replaced + by a single call to calloc() if this is thought to improve performance. */ +png_voidp /* PRIVATE */ +png_create_struct(int type) +{ +#ifdef PNG_USER_MEM_SUPPORTED + return (png_create_struct_2(type, png_malloc_ptr_NULL, png_voidp_NULL)); +} + +/* Alternate version of png_create_struct, for use with user-defined malloc. */ +png_voidp /* PRIVATE */ +png_create_struct_2(int type, png_malloc_ptr malloc_fn, png_voidp mem_ptr) +{ +#endif /* PNG_USER_MEM_SUPPORTED */ + png_size_t size; + png_voidp struct_ptr; + + if (type == PNG_STRUCT_INFO) + size = png_sizeof(png_info); + else if (type == PNG_STRUCT_PNG) + size = png_sizeof(png_struct); + else + return (png_get_copyright(NULL)); + +#ifdef PNG_USER_MEM_SUPPORTED + if(malloc_fn != NULL) + { + png_struct dummy_struct; + png_structp png_ptr = &dummy_struct; + png_ptr->mem_ptr=mem_ptr; + struct_ptr = (*(malloc_fn))(png_ptr, (png_uint_32)size); + } + else +#endif /* PNG_USER_MEM_SUPPORTED */ + struct_ptr = (png_voidp)farmalloc(size); + if (struct_ptr != NULL) + png_memset(struct_ptr, 0, size); + return (struct_ptr); +} + +/* Free memory allocated by a png_create_struct() call */ +void /* PRIVATE */ +png_destroy_struct(png_voidp struct_ptr) +{ +#ifdef PNG_USER_MEM_SUPPORTED + png_destroy_struct_2(struct_ptr, png_free_ptr_NULL, png_voidp_NULL); +} + +/* Free memory allocated by a png_create_struct() call */ +void /* PRIVATE */ +png_destroy_struct_2(png_voidp struct_ptr, png_free_ptr free_fn, + png_voidp mem_ptr) +{ +#endif + if (struct_ptr != NULL) + { +#ifdef PNG_USER_MEM_SUPPORTED + if(free_fn != NULL) + { + png_struct dummy_struct; + png_structp png_ptr = &dummy_struct; + png_ptr->mem_ptr=mem_ptr; + (*(free_fn))(png_ptr, struct_ptr); + return; + } +#endif /* PNG_USER_MEM_SUPPORTED */ + farfree (struct_ptr); + } +} + +/* Allocate memory. For reasonable files, size should never exceed + * 64K. However, zlib may allocate more then 64K if you don't tell + * it not to. See zconf.h and png.h for more information. zlib does + * need to allocate exactly 64K, so whatever you call here must + * have the ability to do that. + * + * Borland seems to have a problem in DOS mode for exactly 64K. + * It gives you a segment with an offset of 8 (perhaps to store its + * memory stuff). zlib doesn't like this at all, so we have to + * detect and deal with it. This code should not be needed in + * Windows or OS/2 modes, and only in 16 bit mode. This code has + * been updated by Alexander Lehmann for version 0.89 to waste less + * memory. + * + * Note that we can't use png_size_t for the "size" declaration, + * since on some systems a png_size_t is a 16-bit quantity, and as a + * result, we would be truncating potentially larger memory requests + * (which should cause a fatal error) and introducing major problems. + */ + +png_voidp PNGAPI +png_malloc(png_structp png_ptr, png_uint_32 size) +{ + png_voidp ret; + + if (png_ptr == NULL || size == 0) + return (NULL); + +#ifdef PNG_USER_MEM_SUPPORTED + if(png_ptr->malloc_fn != NULL) + ret = ((png_voidp)(*(png_ptr->malloc_fn))(png_ptr, (png_size_t)size)); + else + ret = (png_malloc_default(png_ptr, size)); + if (ret == NULL && (png_ptr->flags&PNG_FLAG_MALLOC_NULL_MEM_OK) == 0) + png_error(png_ptr, "Out of memory!"); + return (ret); +} + +png_voidp PNGAPI +png_malloc_default(png_structp png_ptr, png_uint_32 size) +{ + png_voidp ret; +#endif /* PNG_USER_MEM_SUPPORTED */ + +#ifdef PNG_MAX_MALLOC_64K + if (size > (png_uint_32)65536L) + { + png_warning(png_ptr, "Cannot Allocate > 64K"); + ret = NULL; + } + else +#endif + + if (size != (size_t)size) + ret = NULL; + else if (size == (png_uint_32)65536L) + { + if (png_ptr->offset_table == NULL) + { + /* try to see if we need to do any of this fancy stuff */ + ret = farmalloc(size); + if (ret == NULL || ((png_size_t)ret & 0xffff)) + { + int num_blocks; + png_uint_32 total_size; + png_bytep table; + int i; + png_byte huge * hptr; + + if (ret != NULL) + { + farfree(ret); + ret = NULL; + } + + if(png_ptr->zlib_window_bits > 14) + num_blocks = (int)(1 << (png_ptr->zlib_window_bits - 14)); + else + num_blocks = 1; + if (png_ptr->zlib_mem_level >= 7) + num_blocks += (int)(1 << (png_ptr->zlib_mem_level - 7)); + else + num_blocks++; + + total_size = ((png_uint_32)65536L) * (png_uint_32)num_blocks+16; + + table = farmalloc(total_size); + + if (table == NULL) + { +#ifndef PNG_USER_MEM_SUPPORTED + if ((png_ptr->flags&PNG_FLAG_MALLOC_NULL_MEM_OK) == 0) + png_error(png_ptr, "Out Of Memory."); /* Note "O" and "M" */ + else + png_warning(png_ptr, "Out Of Memory."); +#endif + return (NULL); + } + + if ((png_size_t)table & 0xfff0) + { +#ifndef PNG_USER_MEM_SUPPORTED + if ((png_ptr->flags&PNG_FLAG_MALLOC_NULL_MEM_OK) == 0) + png_error(png_ptr, + "Farmalloc didn't return normalized pointer"); + else + png_warning(png_ptr, + "Farmalloc didn't return normalized pointer"); +#endif + return (NULL); + } + + png_ptr->offset_table = table; + png_ptr->offset_table_ptr = farmalloc(num_blocks * + png_sizeof (png_bytep)); + + if (png_ptr->offset_table_ptr == NULL) + { +#ifndef PNG_USER_MEM_SUPPORTED + if ((png_ptr->flags&PNG_FLAG_MALLOC_NULL_MEM_OK) == 0) + png_error(png_ptr, "Out Of memory."); /* Note "O" and "M" */ + else + png_warning(png_ptr, "Out Of memory."); +#endif + return (NULL); + } + + hptr = (png_byte huge *)table; + if ((png_size_t)hptr & 0xf) + { + hptr = (png_byte huge *)((long)(hptr) & 0xfffffff0L); + hptr = hptr + 16L; /* "hptr += 16L" fails on Turbo C++ 3.0 */ + } + for (i = 0; i < num_blocks; i++) + { + png_ptr->offset_table_ptr[i] = (png_bytep)hptr; + hptr = hptr + (png_uint_32)65536L; /* "+=" fails on TC++3.0 */ + } + + png_ptr->offset_table_number = num_blocks; + png_ptr->offset_table_count = 0; + png_ptr->offset_table_count_free = 0; + } + } + + if (png_ptr->offset_table_count >= png_ptr->offset_table_number) + { +#ifndef PNG_USER_MEM_SUPPORTED + if ((png_ptr->flags&PNG_FLAG_MALLOC_NULL_MEM_OK) == 0) + png_error(png_ptr, "Out of Memory."); /* Note "o" and "M" */ + else + png_warning(png_ptr, "Out of Memory."); +#endif + return (NULL); + } + + ret = png_ptr->offset_table_ptr[png_ptr->offset_table_count++]; + } + else + ret = farmalloc(size); + +#ifndef PNG_USER_MEM_SUPPORTED + if (ret == NULL) + { + if ((png_ptr->flags&PNG_FLAG_MALLOC_NULL_MEM_OK) == 0) + png_error(png_ptr, "Out of memory."); /* Note "o" and "m" */ + else + png_warning(png_ptr, "Out of memory."); /* Note "o" and "m" */ + } +#endif + + return (ret); +} + +/* free a pointer allocated by png_malloc(). In the default + configuration, png_ptr is not used, but is passed in case it + is needed. If ptr is NULL, return without taking any action. */ +void PNGAPI +png_free(png_structp png_ptr, png_voidp ptr) +{ + if (png_ptr == NULL || ptr == NULL) + return; + +#ifdef PNG_USER_MEM_SUPPORTED + if (png_ptr->free_fn != NULL) + { + (*(png_ptr->free_fn))(png_ptr, ptr); + return; + } + else png_free_default(png_ptr, ptr); +} + +void PNGAPI +png_free_default(png_structp png_ptr, png_voidp ptr) +{ +#endif /* PNG_USER_MEM_SUPPORTED */ + + if (png_ptr->offset_table != NULL) + { + int i; + + for (i = 0; i < png_ptr->offset_table_count; i++) + { + if (ptr == png_ptr->offset_table_ptr[i]) + { + ptr = NULL; + png_ptr->offset_table_count_free++; + break; + } + } + if (png_ptr->offset_table_count_free == png_ptr->offset_table_count) + { + farfree(png_ptr->offset_table); + farfree(png_ptr->offset_table_ptr); + png_ptr->offset_table = NULL; + png_ptr->offset_table_ptr = NULL; + } + } + + if (ptr != NULL) + { + farfree(ptr); + } +} + +#else /* Not the Borland DOS special memory handler */ + +/* Allocate memory for a png_struct or a png_info. The malloc and + memset can be replaced by a single call to calloc() if this is thought + to improve performance noticably. */ +png_voidp /* PRIVATE */ +png_create_struct(int type) +{ +#ifdef PNG_USER_MEM_SUPPORTED + return (png_create_struct_2(type, png_malloc_ptr_NULL, png_voidp_NULL)); +} + +/* Allocate memory for a png_struct or a png_info. The malloc and + memset can be replaced by a single call to calloc() if this is thought + to improve performance noticably. */ +png_voidp /* PRIVATE */ +png_create_struct_2(int type, png_malloc_ptr malloc_fn, png_voidp mem_ptr) +{ +#endif /* PNG_USER_MEM_SUPPORTED */ + png_size_t size; + png_voidp struct_ptr; + + if (type == PNG_STRUCT_INFO) + size = png_sizeof(png_info); + else if (type == PNG_STRUCT_PNG) + size = png_sizeof(png_struct); + else + return (NULL); + +#ifdef PNG_USER_MEM_SUPPORTED + if(malloc_fn != NULL) + { + png_struct dummy_struct; + png_structp png_ptr = &dummy_struct; + png_ptr->mem_ptr=mem_ptr; + struct_ptr = (*(malloc_fn))(png_ptr, size); + if (struct_ptr != NULL) + png_memset(struct_ptr, 0, size); + return (struct_ptr); + } +#endif /* PNG_USER_MEM_SUPPORTED */ + +#if defined(__TURBOC__) && !defined(__FLAT__) + struct_ptr = (png_voidp)farmalloc(size); +#else +# if defined(_MSC_VER) && defined(MAXSEG_64K) + struct_ptr = (png_voidp)halloc(size,1); +# else + struct_ptr = (png_voidp)malloc(size); +# endif +#endif + if (struct_ptr != NULL) + png_memset(struct_ptr, 0, size); + + return (struct_ptr); +} + + +/* Free memory allocated by a png_create_struct() call */ +void /* PRIVATE */ +png_destroy_struct(png_voidp struct_ptr) +{ +#ifdef PNG_USER_MEM_SUPPORTED + png_destroy_struct_2(struct_ptr, png_free_ptr_NULL, png_voidp_NULL); +} + +/* Free memory allocated by a png_create_struct() call */ +void /* PRIVATE */ +png_destroy_struct_2(png_voidp struct_ptr, png_free_ptr free_fn, + png_voidp mem_ptr) +{ +#endif /* PNG_USER_MEM_SUPPORTED */ + if (struct_ptr != NULL) + { +#ifdef PNG_USER_MEM_SUPPORTED + if(free_fn != NULL) + { + png_struct dummy_struct; + png_structp png_ptr = &dummy_struct; + png_ptr->mem_ptr=mem_ptr; + (*(free_fn))(png_ptr, struct_ptr); + return; + } +#endif /* PNG_USER_MEM_SUPPORTED */ +#if defined(__TURBOC__) && !defined(__FLAT__) + farfree(struct_ptr); +#else +# if defined(_MSC_VER) && defined(MAXSEG_64K) + hfree(struct_ptr); +# else + free(struct_ptr); +# endif +#endif + } +} + +/* Allocate memory. For reasonable files, size should never exceed + 64K. However, zlib may allocate more then 64K if you don't tell + it not to. See zconf.h and png.h for more information. zlib does + need to allocate exactly 64K, so whatever you call here must + have the ability to do that. */ + +png_voidp PNGAPI +png_malloc(png_structp png_ptr, png_uint_32 size) +{ + png_voidp ret; + +#ifdef PNG_USER_MEM_SUPPORTED + if (png_ptr == NULL || size == 0) + return (NULL); + + if(png_ptr->malloc_fn != NULL) + ret = ((png_voidp)(*(png_ptr->malloc_fn))(png_ptr, (png_size_t)size)); + else + ret = (png_malloc_default(png_ptr, size)); + if (ret == NULL && (png_ptr->flags&PNG_FLAG_MALLOC_NULL_MEM_OK) == 0) + png_error(png_ptr, "Out of Memory!"); + return (ret); +} + +png_voidp PNGAPI +png_malloc_default(png_structp png_ptr, png_uint_32 size) +{ + png_voidp ret; +#endif /* PNG_USER_MEM_SUPPORTED */ + + if (png_ptr == NULL || size == 0) + return (NULL); + +#ifdef PNG_MAX_MALLOC_64K + if (size > (png_uint_32)65536L) + { +#ifndef PNG_USER_MEM_SUPPORTED + if(png_ptr->flags&PNG_FLAG_MALLOC_NULL_MEM_OK) == 0) + png_error(png_ptr, "Cannot Allocate > 64K"); + else +#endif + return NULL; + } +#endif + + /* Check for overflow */ +#if defined(__TURBOC__) && !defined(__FLAT__) + if (size != (unsigned long)size) + ret = NULL; + else + ret = farmalloc(size); +#else +# if defined(_MSC_VER) && defined(MAXSEG_64K) + if (size != (unsigned long)size) + ret = NULL; + else + ret = halloc(size, 1); +# else + if (size != (size_t)size) + ret = NULL; + else + ret = malloc((size_t)size); +# endif +#endif + +#ifndef PNG_USER_MEM_SUPPORTED + if (ret == NULL && (png_ptr->flags&PNG_FLAG_MALLOC_NULL_MEM_OK) == 0) + png_error(png_ptr, "Out of Memory"); +#endif + + return (ret); +} + +/* Free a pointer allocated by png_malloc(). If ptr is NULL, return + without taking any action. */ +void PNGAPI +png_free(png_structp png_ptr, png_voidp ptr) +{ + if (png_ptr == NULL || ptr == NULL) + return; + +#ifdef PNG_USER_MEM_SUPPORTED + if (png_ptr->free_fn != NULL) + { + (*(png_ptr->free_fn))(png_ptr, ptr); + return; + } + else png_free_default(png_ptr, ptr); +} +void PNGAPI +png_free_default(png_structp png_ptr, png_voidp ptr) +{ + if (png_ptr == NULL || ptr == NULL) + return; + +#endif /* PNG_USER_MEM_SUPPORTED */ + +#if defined(__TURBOC__) && !defined(__FLAT__) + farfree(ptr); +#else +# if defined(_MSC_VER) && defined(MAXSEG_64K) + hfree(ptr); +# else + free(ptr); +# endif +#endif +} + +#endif /* Not Borland DOS special memory handler */ + +#if defined(PNG_1_0_X) +# define png_malloc_warn png_malloc +#else +/* This function was added at libpng version 1.2.3. The png_malloc_warn() + * function will set up png_malloc() to issue a png_warning and return NULL + * instead of issuing a png_error, if it fails to allocate the requested + * memory. + */ +png_voidp PNGAPI +png_malloc_warn(png_structp png_ptr, png_uint_32 size) +{ + png_voidp ptr; + png_uint_32 save_flags=png_ptr->flags; + + png_ptr->flags|=PNG_FLAG_MALLOC_NULL_MEM_OK; + ptr = (png_voidp)png_malloc((png_structp)png_ptr, size); + png_ptr->flags=save_flags; + return(ptr); +} +#endif + +png_voidp PNGAPI +png_memcpy_check (png_structp png_ptr, png_voidp s1, png_voidp s2, + png_uint_32 length) +{ + png_size_t size; + + size = (png_size_t)length; + if ((png_uint_32)size != length) + png_error(png_ptr,"Overflow in png_memcpy_check."); + + return(png_memcpy (s1, s2, size)); +} + +png_voidp PNGAPI +png_memset_check (png_structp png_ptr, png_voidp s1, int value, + png_uint_32 length) +{ + png_size_t size; + + size = (png_size_t)length; + if ((png_uint_32)size != length) + png_error(png_ptr,"Overflow in png_memset_check."); + + return (png_memset (s1, value, size)); + +} + +#ifdef PNG_USER_MEM_SUPPORTED +/* This function is called when the application wants to use another method + * of allocating and freeing memory. + */ +void PNGAPI +png_set_mem_fn(png_structp png_ptr, png_voidp mem_ptr, png_malloc_ptr + malloc_fn, png_free_ptr free_fn) +{ + png_ptr->mem_ptr = mem_ptr; + png_ptr->malloc_fn = malloc_fn; + png_ptr->free_fn = free_fn; +} + +/* This function returns a pointer to the mem_ptr associated with the user + * functions. The application should free any memory associated with this + * pointer before png_write_destroy and png_read_destroy are called. + */ +png_voidp PNGAPI +png_get_mem_ptr(png_structp png_ptr) +{ + return ((png_voidp)png_ptr->mem_ptr); +} +#endif /* PNG_USER_MEM_SUPPORTED */ +#endif /* PNG_READ_SUPPORTED || PNG_WRITE_SUPPORTED */ diff --git a/src/win32/dependencies/libpng/pngpread.c b/src/win32/dependencies/libpng/pngpread.c new file mode 100644 index 00000000..4a98a2e2 --- /dev/null +++ b/src/win32/dependencies/libpng/pngpread.c @@ -0,0 +1,1573 @@ + +/* pngpread.c - read a png file in push mode + * + * Last changed in libpng 1.2.8 - December 3, 2004 + * For conditions of distribution and use, see copyright notice in png.h + * Copyright (c) 1998-2004 Glenn Randers-Pehrson + * (Version 0.96 Copyright (c) 1996, 1997 Andreas Dilger) + * (Version 0.88 Copyright (c) 1995, 1996 Guy Eric Schalnat, Group 42, Inc.) + */ + +#define PNG_INTERNAL +#include "png.h" + +#ifdef PNG_PROGRESSIVE_READ_SUPPORTED + +/* push model modes */ +#define PNG_READ_SIG_MODE 0 +#define PNG_READ_CHUNK_MODE 1 +#define PNG_READ_IDAT_MODE 2 +#define PNG_SKIP_MODE 3 +#define PNG_READ_tEXt_MODE 4 +#define PNG_READ_zTXt_MODE 5 +#define PNG_READ_DONE_MODE 6 +#define PNG_READ_iTXt_MODE 7 +#define PNG_ERROR_MODE 8 + +void PNGAPI +png_process_data(png_structp png_ptr, png_infop info_ptr, + png_bytep buffer, png_size_t buffer_size) +{ + png_push_restore_buffer(png_ptr, buffer, buffer_size); + + while (png_ptr->buffer_size) + { + png_process_some_data(png_ptr, info_ptr); + } +} + +/* What we do with the incoming data depends on what we were previously + * doing before we ran out of data... + */ +void /* PRIVATE */ +png_process_some_data(png_structp png_ptr, png_infop info_ptr) +{ + switch (png_ptr->process_mode) + { + case PNG_READ_SIG_MODE: + { + png_push_read_sig(png_ptr, info_ptr); + break; + } + case PNG_READ_CHUNK_MODE: + { + png_push_read_chunk(png_ptr, info_ptr); + break; + } + case PNG_READ_IDAT_MODE: + { + png_push_read_IDAT(png_ptr); + break; + } +#if defined(PNG_READ_tEXt_SUPPORTED) + case PNG_READ_tEXt_MODE: + { + png_push_read_tEXt(png_ptr, info_ptr); + break; + } +#endif +#if defined(PNG_READ_zTXt_SUPPORTED) + case PNG_READ_zTXt_MODE: + { + png_push_read_zTXt(png_ptr, info_ptr); + break; + } +#endif +#if defined(PNG_READ_iTXt_SUPPORTED) + case PNG_READ_iTXt_MODE: + { + png_push_read_iTXt(png_ptr, info_ptr); + break; + } +#endif + case PNG_SKIP_MODE: + { + png_push_crc_finish(png_ptr); + break; + } + default: + { + png_ptr->buffer_size = 0; + break; + } + } +} + +/* Read any remaining signature bytes from the stream and compare them with + * the correct PNG signature. It is possible that this routine is called + * with bytes already read from the signature, either because they have been + * checked by the calling application, or because of multiple calls to this + * routine. + */ +void /* PRIVATE */ +png_push_read_sig(png_structp png_ptr, png_infop info_ptr) +{ + png_size_t num_checked = png_ptr->sig_bytes, + num_to_check = 8 - num_checked; + + if (png_ptr->buffer_size < num_to_check) + { + num_to_check = png_ptr->buffer_size; + } + + png_push_fill_buffer(png_ptr, &(info_ptr->signature[num_checked]), + num_to_check); + png_ptr->sig_bytes = (png_byte)(png_ptr->sig_bytes+num_to_check); + + if (png_sig_cmp(info_ptr->signature, num_checked, num_to_check)) + { + if (num_checked < 4 && + png_sig_cmp(info_ptr->signature, num_checked, num_to_check - 4)) + png_error(png_ptr, "Not a PNG file"); + else + png_error(png_ptr, "PNG file corrupted by ASCII conversion"); + } + else + { + if (png_ptr->sig_bytes >= 8) + { + png_ptr->process_mode = PNG_READ_CHUNK_MODE; + } + } +} + +void /* PRIVATE */ +png_push_read_chunk(png_structp png_ptr, png_infop info_ptr) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_IHDR; + PNG_IDAT; + PNG_IEND; + PNG_PLTE; +#if defined(PNG_READ_bKGD_SUPPORTED) + PNG_bKGD; +#endif +#if defined(PNG_READ_cHRM_SUPPORTED) + PNG_cHRM; +#endif +#if defined(PNG_READ_gAMA_SUPPORTED) + PNG_gAMA; +#endif +#if defined(PNG_READ_hIST_SUPPORTED) + PNG_hIST; +#endif +#if defined(PNG_READ_iCCP_SUPPORTED) + PNG_iCCP; +#endif +#if defined(PNG_READ_iTXt_SUPPORTED) + PNG_iTXt; +#endif +#if defined(PNG_READ_oFFs_SUPPORTED) + PNG_oFFs; +#endif +#if defined(PNG_READ_pCAL_SUPPORTED) + PNG_pCAL; +#endif +#if defined(PNG_READ_pHYs_SUPPORTED) + PNG_pHYs; +#endif +#if defined(PNG_READ_sBIT_SUPPORTED) + PNG_sBIT; +#endif +#if defined(PNG_READ_sCAL_SUPPORTED) + PNG_sCAL; +#endif +#if defined(PNG_READ_sRGB_SUPPORTED) + PNG_sRGB; +#endif +#if defined(PNG_READ_sPLT_SUPPORTED) + PNG_sPLT; +#endif +#if defined(PNG_READ_tEXt_SUPPORTED) + PNG_tEXt; +#endif +#if defined(PNG_READ_tIME_SUPPORTED) + PNG_tIME; +#endif +#if defined(PNG_READ_tRNS_SUPPORTED) + PNG_tRNS; +#endif +#if defined(PNG_READ_zTXt_SUPPORTED) + PNG_zTXt; +#endif +#endif /* PNG_USE_LOCAL_ARRAYS */ + /* First we make sure we have enough data for the 4 byte chunk name + * and the 4 byte chunk length before proceeding with decoding the + * chunk data. To fully decode each of these chunks, we also make + * sure we have enough data in the buffer for the 4 byte CRC at the + * end of every chunk (except IDAT, which is handled separately). + */ + if (!(png_ptr->mode & PNG_HAVE_CHUNK_HEADER)) + { + png_byte chunk_length[4]; + + if (png_ptr->buffer_size < 8) + { + png_push_save_buffer(png_ptr); + return; + } + + png_push_fill_buffer(png_ptr, chunk_length, 4); + png_ptr->push_length = png_get_uint_31(png_ptr,chunk_length); + png_reset_crc(png_ptr); + png_crc_read(png_ptr, png_ptr->chunk_name, 4); + png_ptr->mode |= PNG_HAVE_CHUNK_HEADER; + } + + if (!png_memcmp(png_ptr->chunk_name, png_IHDR, 4)) + { + if (png_ptr->push_length + 4 > png_ptr->buffer_size) + { + png_push_save_buffer(png_ptr); + return; + } + png_handle_IHDR(png_ptr, info_ptr, png_ptr->push_length); + } + else if (!png_memcmp(png_ptr->chunk_name, png_IEND, 4)) + { + if (png_ptr->push_length + 4 > png_ptr->buffer_size) + { + png_push_save_buffer(png_ptr); + return; + } + png_handle_IEND(png_ptr, info_ptr, png_ptr->push_length); + + png_ptr->process_mode = PNG_READ_DONE_MODE; + png_push_have_end(png_ptr, info_ptr); + } +#ifdef PNG_HANDLE_AS_UNKNOWN_SUPPORTED + else if (png_handle_as_unknown(png_ptr, png_ptr->chunk_name)) + { + if (png_ptr->push_length + 4 > png_ptr->buffer_size) + { + png_push_save_buffer(png_ptr); + return; + } + if (!png_memcmp(png_ptr->chunk_name, png_IDAT, 4)) + png_ptr->mode |= PNG_HAVE_IDAT; + png_handle_unknown(png_ptr, info_ptr, png_ptr->push_length); + if (!png_memcmp(png_ptr->chunk_name, png_PLTE, 4)) + png_ptr->mode |= PNG_HAVE_PLTE; + else if (!png_memcmp(png_ptr->chunk_name, png_IDAT, 4)) + { + if (!(png_ptr->mode & PNG_HAVE_IHDR)) + png_error(png_ptr, "Missing IHDR before IDAT"); + else if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE && + !(png_ptr->mode & PNG_HAVE_PLTE)) + png_error(png_ptr, "Missing PLTE before IDAT"); + } + } +#endif + else if (!png_memcmp(png_ptr->chunk_name, png_PLTE, 4)) + { + if (png_ptr->push_length + 4 > png_ptr->buffer_size) + { + png_push_save_buffer(png_ptr); + return; + } + png_handle_PLTE(png_ptr, info_ptr, png_ptr->push_length); + } + else if (!png_memcmp(png_ptr->chunk_name, (png_bytep)png_IDAT, 4)) + { + /* If we reach an IDAT chunk, this means we have read all of the + * header chunks, and we can start reading the image (or if this + * is called after the image has been read - we have an error). + */ + if (!(png_ptr->mode & PNG_HAVE_IHDR)) + png_error(png_ptr, "Missing IHDR before IDAT"); + else if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE && + !(png_ptr->mode & PNG_HAVE_PLTE)) + png_error(png_ptr, "Missing PLTE before IDAT"); + + if (png_ptr->mode & PNG_HAVE_IDAT) + { + if (png_ptr->push_length == 0) + return; + + if (png_ptr->mode & PNG_AFTER_IDAT) + png_error(png_ptr, "Too many IDAT's found"); + } + + png_ptr->idat_size = png_ptr->push_length; + png_ptr->mode |= PNG_HAVE_IDAT; + png_ptr->process_mode = PNG_READ_IDAT_MODE; + png_push_have_info(png_ptr, info_ptr); + png_ptr->zstream.avail_out = (uInt)png_ptr->irowbytes; + png_ptr->zstream.next_out = png_ptr->row_buf; + return; + } +#if defined(PNG_READ_gAMA_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_gAMA, 4)) + { + if (png_ptr->push_length + 4 > png_ptr->buffer_size) + { + png_push_save_buffer(png_ptr); + return; + } + png_handle_gAMA(png_ptr, info_ptr, png_ptr->push_length); + } +#endif +#if defined(PNG_READ_sBIT_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_sBIT, 4)) + { + if (png_ptr->push_length + 4 > png_ptr->buffer_size) + { + png_push_save_buffer(png_ptr); + return; + } + png_handle_sBIT(png_ptr, info_ptr, png_ptr->push_length); + } +#endif +#if defined(PNG_READ_cHRM_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_cHRM, 4)) + { + if (png_ptr->push_length + 4 > png_ptr->buffer_size) + { + png_push_save_buffer(png_ptr); + return; + } + png_handle_cHRM(png_ptr, info_ptr, png_ptr->push_length); + } +#endif +#if defined(PNG_READ_sRGB_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_sRGB, 4)) + { + if (png_ptr->push_length + 4 > png_ptr->buffer_size) + { + png_push_save_buffer(png_ptr); + return; + } + png_handle_sRGB(png_ptr, info_ptr, png_ptr->push_length); + } +#endif +#if defined(PNG_READ_iCCP_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_iCCP, 4)) + { + if (png_ptr->push_length + 4 > png_ptr->buffer_size) + { + png_push_save_buffer(png_ptr); + return; + } + png_handle_iCCP(png_ptr, info_ptr, png_ptr->push_length); + } +#endif +#if defined(PNG_READ_sPLT_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_sPLT, 4)) + { + if (png_ptr->push_length + 4 > png_ptr->buffer_size) + { + png_push_save_buffer(png_ptr); + return; + } + png_handle_sPLT(png_ptr, info_ptr, png_ptr->push_length); + } +#endif +#if defined(PNG_READ_tRNS_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_tRNS, 4)) + { + if (png_ptr->push_length + 4 > png_ptr->buffer_size) + { + png_push_save_buffer(png_ptr); + return; + } + png_handle_tRNS(png_ptr, info_ptr, png_ptr->push_length); + } +#endif +#if defined(PNG_READ_bKGD_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_bKGD, 4)) + { + if (png_ptr->push_length + 4 > png_ptr->buffer_size) + { + png_push_save_buffer(png_ptr); + return; + } + png_handle_bKGD(png_ptr, info_ptr, png_ptr->push_length); + } +#endif +#if defined(PNG_READ_hIST_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_hIST, 4)) + { + if (png_ptr->push_length + 4 > png_ptr->buffer_size) + { + png_push_save_buffer(png_ptr); + return; + } + png_handle_hIST(png_ptr, info_ptr, png_ptr->push_length); + } +#endif +#if defined(PNG_READ_pHYs_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_pHYs, 4)) + { + if (png_ptr->push_length + 4 > png_ptr->buffer_size) + { + png_push_save_buffer(png_ptr); + return; + } + png_handle_pHYs(png_ptr, info_ptr, png_ptr->push_length); + } +#endif +#if defined(PNG_READ_oFFs_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_oFFs, 4)) + { + if (png_ptr->push_length + 4 > png_ptr->buffer_size) + { + png_push_save_buffer(png_ptr); + return; + } + png_handle_oFFs(png_ptr, info_ptr, png_ptr->push_length); + } +#endif +#if defined(PNG_READ_pCAL_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_pCAL, 4)) + { + if (png_ptr->push_length + 4 > png_ptr->buffer_size) + { + png_push_save_buffer(png_ptr); + return; + } + png_handle_pCAL(png_ptr, info_ptr, png_ptr->push_length); + } +#endif +#if defined(PNG_READ_sCAL_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_sCAL, 4)) + { + if (png_ptr->push_length + 4 > png_ptr->buffer_size) + { + png_push_save_buffer(png_ptr); + return; + } + png_handle_sCAL(png_ptr, info_ptr, png_ptr->push_length); + } +#endif +#if defined(PNG_READ_tIME_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_tIME, 4)) + { + if (png_ptr->push_length + 4 > png_ptr->buffer_size) + { + png_push_save_buffer(png_ptr); + return; + } + png_handle_tIME(png_ptr, info_ptr, png_ptr->push_length); + } +#endif +#if defined(PNG_READ_tEXt_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_tEXt, 4)) + { + if (png_ptr->push_length + 4 > png_ptr->buffer_size) + { + png_push_save_buffer(png_ptr); + return; + } + png_push_handle_tEXt(png_ptr, info_ptr, png_ptr->push_length); + } +#endif +#if defined(PNG_READ_zTXt_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_zTXt, 4)) + { + if (png_ptr->push_length + 4 > png_ptr->buffer_size) + { + png_push_save_buffer(png_ptr); + return; + } + png_push_handle_zTXt(png_ptr, info_ptr, png_ptr->push_length); + } +#endif +#if defined(PNG_READ_iTXt_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_iTXt, 4)) + { + if (png_ptr->push_length + 4 > png_ptr->buffer_size) + { + png_push_save_buffer(png_ptr); + return; + } + png_push_handle_iTXt(png_ptr, info_ptr, png_ptr->push_length); + } +#endif + else + { + if (png_ptr->push_length + 4 > png_ptr->buffer_size) + { + png_push_save_buffer(png_ptr); + return; + } + png_push_handle_unknown(png_ptr, info_ptr, png_ptr->push_length); + } + + png_ptr->mode &= ~PNG_HAVE_CHUNK_HEADER; +} + +void /* PRIVATE */ +png_push_crc_skip(png_structp png_ptr, png_uint_32 skip) +{ + png_ptr->process_mode = PNG_SKIP_MODE; + png_ptr->skip_length = skip; +} + +void /* PRIVATE */ +png_push_crc_finish(png_structp png_ptr) +{ + if (png_ptr->skip_length && png_ptr->save_buffer_size) + { + png_size_t save_size; + + if (png_ptr->skip_length < (png_uint_32)png_ptr->save_buffer_size) + save_size = (png_size_t)png_ptr->skip_length; + else + save_size = png_ptr->save_buffer_size; + + png_calculate_crc(png_ptr, png_ptr->save_buffer_ptr, save_size); + + png_ptr->skip_length -= save_size; + png_ptr->buffer_size -= save_size; + png_ptr->save_buffer_size -= save_size; + png_ptr->save_buffer_ptr += save_size; + } + if (png_ptr->skip_length && png_ptr->current_buffer_size) + { + png_size_t save_size; + + if (png_ptr->skip_length < (png_uint_32)png_ptr->current_buffer_size) + save_size = (png_size_t)png_ptr->skip_length; + else + save_size = png_ptr->current_buffer_size; + + png_calculate_crc(png_ptr, png_ptr->current_buffer_ptr, save_size); + + png_ptr->skip_length -= save_size; + png_ptr->buffer_size -= save_size; + png_ptr->current_buffer_size -= save_size; + png_ptr->current_buffer_ptr += save_size; + } + if (!png_ptr->skip_length) + { + if (png_ptr->buffer_size < 4) + { + png_push_save_buffer(png_ptr); + return; + } + + png_crc_finish(png_ptr, 0); + png_ptr->process_mode = PNG_READ_CHUNK_MODE; + } +} + +void PNGAPI +png_push_fill_buffer(png_structp png_ptr, png_bytep buffer, png_size_t length) +{ + png_bytep ptr; + + ptr = buffer; + if (png_ptr->save_buffer_size) + { + png_size_t save_size; + + if (length < png_ptr->save_buffer_size) + save_size = length; + else + save_size = png_ptr->save_buffer_size; + + png_memcpy(ptr, png_ptr->save_buffer_ptr, save_size); + length -= save_size; + ptr += save_size; + png_ptr->buffer_size -= save_size; + png_ptr->save_buffer_size -= save_size; + png_ptr->save_buffer_ptr += save_size; + } + if (length && png_ptr->current_buffer_size) + { + png_size_t save_size; + + if (length < png_ptr->current_buffer_size) + save_size = length; + else + save_size = png_ptr->current_buffer_size; + + png_memcpy(ptr, png_ptr->current_buffer_ptr, save_size); + png_ptr->buffer_size -= save_size; + png_ptr->current_buffer_size -= save_size; + png_ptr->current_buffer_ptr += save_size; + } +} + +void /* PRIVATE */ +png_push_save_buffer(png_structp png_ptr) +{ + if (png_ptr->save_buffer_size) + { + if (png_ptr->save_buffer_ptr != png_ptr->save_buffer) + { + png_size_t i,istop; + png_bytep sp; + png_bytep dp; + + istop = png_ptr->save_buffer_size; + for (i = 0, sp = png_ptr->save_buffer_ptr, dp = png_ptr->save_buffer; + i < istop; i++, sp++, dp++) + { + *dp = *sp; + } + } + } + if (png_ptr->save_buffer_size + png_ptr->current_buffer_size > + png_ptr->save_buffer_max) + { + png_size_t new_max; + png_bytep old_buffer; + + if (png_ptr->save_buffer_size > PNG_SIZE_MAX - + (png_ptr->current_buffer_size + 256)) + { + png_error(png_ptr, "Potential overflow of save_buffer"); + } + new_max = png_ptr->save_buffer_size + png_ptr->current_buffer_size + 256; + old_buffer = png_ptr->save_buffer; + png_ptr->save_buffer = (png_bytep)png_malloc(png_ptr, + (png_uint_32)new_max); + png_memcpy(png_ptr->save_buffer, old_buffer, png_ptr->save_buffer_size); + png_free(png_ptr, old_buffer); + png_ptr->save_buffer_max = new_max; + } + if (png_ptr->current_buffer_size) + { + png_memcpy(png_ptr->save_buffer + png_ptr->save_buffer_size, + png_ptr->current_buffer_ptr, png_ptr->current_buffer_size); + png_ptr->save_buffer_size += png_ptr->current_buffer_size; + png_ptr->current_buffer_size = 0; + } + png_ptr->save_buffer_ptr = png_ptr->save_buffer; + png_ptr->buffer_size = 0; +} + +void /* PRIVATE */ +png_push_restore_buffer(png_structp png_ptr, png_bytep buffer, + png_size_t buffer_length) +{ + png_ptr->current_buffer = buffer; + png_ptr->current_buffer_size = buffer_length; + png_ptr->buffer_size = buffer_length + png_ptr->save_buffer_size; + png_ptr->current_buffer_ptr = png_ptr->current_buffer; +} + +void /* PRIVATE */ +png_push_read_IDAT(png_structp png_ptr) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_IDAT; +#endif + if (!(png_ptr->mode & PNG_HAVE_CHUNK_HEADER)) + { + png_byte chunk_length[4]; + + if (png_ptr->buffer_size < 8) + { + png_push_save_buffer(png_ptr); + return; + } + + png_push_fill_buffer(png_ptr, chunk_length, 4); + png_ptr->push_length = png_get_uint_31(png_ptr,chunk_length); + png_reset_crc(png_ptr); + png_crc_read(png_ptr, png_ptr->chunk_name, 4); + png_ptr->mode |= PNG_HAVE_CHUNK_HEADER; + + if (png_memcmp(png_ptr->chunk_name, (png_bytep)png_IDAT, 4)) + { + png_ptr->process_mode = PNG_READ_CHUNK_MODE; + if (!(png_ptr->flags & PNG_FLAG_ZLIB_FINISHED)) + png_error(png_ptr, "Not enough compressed data"); + return; + } + + png_ptr->idat_size = png_ptr->push_length; + } + if (png_ptr->idat_size && png_ptr->save_buffer_size) + { + png_size_t save_size; + + if (png_ptr->idat_size < (png_uint_32)png_ptr->save_buffer_size) + { + save_size = (png_size_t)png_ptr->idat_size; + /* check for overflow */ + if((png_uint_32)save_size != png_ptr->idat_size) + png_error(png_ptr, "save_size overflowed in pngpread"); + } + else + save_size = png_ptr->save_buffer_size; + + png_calculate_crc(png_ptr, png_ptr->save_buffer_ptr, save_size); + if (!(png_ptr->flags & PNG_FLAG_ZLIB_FINISHED)) + png_process_IDAT_data(png_ptr, png_ptr->save_buffer_ptr, save_size); + png_ptr->idat_size -= save_size; + png_ptr->buffer_size -= save_size; + png_ptr->save_buffer_size -= save_size; + png_ptr->save_buffer_ptr += save_size; + } + if (png_ptr->idat_size && png_ptr->current_buffer_size) + { + png_size_t save_size; + + if (png_ptr->idat_size < (png_uint_32)png_ptr->current_buffer_size) + { + save_size = (png_size_t)png_ptr->idat_size; + /* check for overflow */ + if((png_uint_32)save_size != png_ptr->idat_size) + png_error(png_ptr, "save_size overflowed in pngpread"); + } + else + save_size = png_ptr->current_buffer_size; + + png_calculate_crc(png_ptr, png_ptr->current_buffer_ptr, save_size); + if (!(png_ptr->flags & PNG_FLAG_ZLIB_FINISHED)) + png_process_IDAT_data(png_ptr, png_ptr->current_buffer_ptr, save_size); + + png_ptr->idat_size -= save_size; + png_ptr->buffer_size -= save_size; + png_ptr->current_buffer_size -= save_size; + png_ptr->current_buffer_ptr += save_size; + } + if (!png_ptr->idat_size) + { + if (png_ptr->buffer_size < 4) + { + png_push_save_buffer(png_ptr); + return; + } + + png_crc_finish(png_ptr, 0); + png_ptr->mode &= ~PNG_HAVE_CHUNK_HEADER; + png_ptr->mode |= PNG_AFTER_IDAT; + } +} + +void /* PRIVATE */ +png_process_IDAT_data(png_structp png_ptr, png_bytep buffer, + png_size_t buffer_length) +{ + int ret; + + if ((png_ptr->flags & PNG_FLAG_ZLIB_FINISHED) && buffer_length) + png_error(png_ptr, "Extra compression data"); + + png_ptr->zstream.next_in = buffer; + png_ptr->zstream.avail_in = (uInt)buffer_length; + for(;;) + { + ret = inflate(&png_ptr->zstream, Z_PARTIAL_FLUSH); + if (ret != Z_OK) + { + if (ret == Z_STREAM_END) + { + if (png_ptr->zstream.avail_in) + png_error(png_ptr, "Extra compressed data"); + if (!(png_ptr->zstream.avail_out)) + { + png_push_process_row(png_ptr); + } + + png_ptr->mode |= PNG_AFTER_IDAT; + png_ptr->flags |= PNG_FLAG_ZLIB_FINISHED; + break; + } + else if (ret == Z_BUF_ERROR) + break; + else + png_error(png_ptr, "Decompression Error"); + } + if (!(png_ptr->zstream.avail_out)) + { + if (( +#if defined(PNG_READ_INTERLACING_SUPPORTED) + png_ptr->interlaced && png_ptr->pass > 6) || + (!png_ptr->interlaced && +#endif + png_ptr->row_number == png_ptr->num_rows)) + { + if (png_ptr->zstream.avail_in) + png_warning(png_ptr, "Too much data in IDAT chunks"); + png_ptr->flags |= PNG_FLAG_ZLIB_FINISHED; + break; + } + png_push_process_row(png_ptr); + png_ptr->zstream.avail_out = (uInt)png_ptr->irowbytes; + png_ptr->zstream.next_out = png_ptr->row_buf; + } + else + break; + } +} + +void /* PRIVATE */ +png_push_process_row(png_structp png_ptr) +{ + png_ptr->row_info.color_type = png_ptr->color_type; + png_ptr->row_info.width = png_ptr->iwidth; + png_ptr->row_info.channels = png_ptr->channels; + png_ptr->row_info.bit_depth = png_ptr->bit_depth; + png_ptr->row_info.pixel_depth = png_ptr->pixel_depth; + + png_ptr->row_info.rowbytes = PNG_ROWBYTES(png_ptr->row_info.pixel_depth, + png_ptr->row_info.width); + + png_read_filter_row(png_ptr, &(png_ptr->row_info), + png_ptr->row_buf + 1, png_ptr->prev_row + 1, + (int)(png_ptr->row_buf[0])); + + png_memcpy_check(png_ptr, png_ptr->prev_row, png_ptr->row_buf, + png_ptr->rowbytes + 1); + + if (png_ptr->transformations || (png_ptr->flags&PNG_FLAG_STRIP_ALPHA)) + png_do_read_transformations(png_ptr); + +#if defined(PNG_READ_INTERLACING_SUPPORTED) + /* blow up interlaced rows to full size */ + if (png_ptr->interlaced && (png_ptr->transformations & PNG_INTERLACE)) + { + if (png_ptr->pass < 6) +/* old interface (pre-1.0.9): + png_do_read_interlace(&(png_ptr->row_info), + png_ptr->row_buf + 1, png_ptr->pass, png_ptr->transformations); + */ + png_do_read_interlace(png_ptr); + + switch (png_ptr->pass) + { + case 0: + { + int i; + for (i = 0; i < 8 && png_ptr->pass == 0; i++) + { + png_push_have_row(png_ptr, png_ptr->row_buf + 1); + png_read_push_finish_row(png_ptr); /* updates png_ptr->pass */ + } + if (png_ptr->pass == 2) /* pass 1 might be empty */ + { + for (i = 0; i < 4 && png_ptr->pass == 2; i++) + { + png_push_have_row(png_ptr, png_bytep_NULL); + png_read_push_finish_row(png_ptr); + } + } + if (png_ptr->pass == 4 && png_ptr->height <= 4) + { + for (i = 0; i < 2 && png_ptr->pass == 4; i++) + { + png_push_have_row(png_ptr, png_bytep_NULL); + png_read_push_finish_row(png_ptr); + } + } + if (png_ptr->pass == 6 && png_ptr->height <= 4) + { + png_push_have_row(png_ptr, png_bytep_NULL); + png_read_push_finish_row(png_ptr); + } + break; + } + case 1: + { + int i; + for (i = 0; i < 8 && png_ptr->pass == 1; i++) + { + png_push_have_row(png_ptr, png_ptr->row_buf + 1); + png_read_push_finish_row(png_ptr); + } + if (png_ptr->pass == 2) /* skip top 4 generated rows */ + { + for (i = 0; i < 4 && png_ptr->pass == 2; i++) + { + png_push_have_row(png_ptr, png_bytep_NULL); + png_read_push_finish_row(png_ptr); + } + } + break; + } + case 2: + { + int i; + for (i = 0; i < 4 && png_ptr->pass == 2; i++) + { + png_push_have_row(png_ptr, png_ptr->row_buf + 1); + png_read_push_finish_row(png_ptr); + } + for (i = 0; i < 4 && png_ptr->pass == 2; i++) + { + png_push_have_row(png_ptr, png_bytep_NULL); + png_read_push_finish_row(png_ptr); + } + if (png_ptr->pass == 4) /* pass 3 might be empty */ + { + for (i = 0; i < 2 && png_ptr->pass == 4; i++) + { + png_push_have_row(png_ptr, png_bytep_NULL); + png_read_push_finish_row(png_ptr); + } + } + break; + } + case 3: + { + int i; + for (i = 0; i < 4 && png_ptr->pass == 3; i++) + { + png_push_have_row(png_ptr, png_ptr->row_buf + 1); + png_read_push_finish_row(png_ptr); + } + if (png_ptr->pass == 4) /* skip top two generated rows */ + { + for (i = 0; i < 2 && png_ptr->pass == 4; i++) + { + png_push_have_row(png_ptr, png_bytep_NULL); + png_read_push_finish_row(png_ptr); + } + } + break; + } + case 4: + { + int i; + for (i = 0; i < 2 && png_ptr->pass == 4; i++) + { + png_push_have_row(png_ptr, png_ptr->row_buf + 1); + png_read_push_finish_row(png_ptr); + } + for (i = 0; i < 2 && png_ptr->pass == 4; i++) + { + png_push_have_row(png_ptr, png_bytep_NULL); + png_read_push_finish_row(png_ptr); + } + if (png_ptr->pass == 6) /* pass 5 might be empty */ + { + png_push_have_row(png_ptr, png_bytep_NULL); + png_read_push_finish_row(png_ptr); + } + break; + } + case 5: + { + int i; + for (i = 0; i < 2 && png_ptr->pass == 5; i++) + { + png_push_have_row(png_ptr, png_ptr->row_buf + 1); + png_read_push_finish_row(png_ptr); + } + if (png_ptr->pass == 6) /* skip top generated row */ + { + png_push_have_row(png_ptr, png_bytep_NULL); + png_read_push_finish_row(png_ptr); + } + break; + } + case 6: + { + png_push_have_row(png_ptr, png_ptr->row_buf + 1); + png_read_push_finish_row(png_ptr); + if (png_ptr->pass != 6) + break; + png_push_have_row(png_ptr, png_bytep_NULL); + png_read_push_finish_row(png_ptr); + } + } + } + else +#endif + { + png_push_have_row(png_ptr, png_ptr->row_buf + 1); + png_read_push_finish_row(png_ptr); + } +} + +void /* PRIVATE */ +png_read_push_finish_row(png_structp png_ptr) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + /* arrays to facilitate easy interlacing - use pass (0 - 6) as index */ + + /* start of interlace block */ + const int FARDATA png_pass_start[] = {0, 4, 0, 2, 0, 1, 0}; + + /* offset to next interlace block */ + const int FARDATA png_pass_inc[] = {8, 8, 4, 4, 2, 2, 1}; + + /* start of interlace block in the y direction */ + const int FARDATA png_pass_ystart[] = {0, 0, 4, 0, 2, 0, 1}; + + /* offset to next interlace block in the y direction */ + const int FARDATA png_pass_yinc[] = {8, 8, 8, 4, 4, 2, 2}; + + /* Width of interlace block. This is not currently used - if you need + * it, uncomment it here and in png.h + const int FARDATA png_pass_width[] = {8, 4, 4, 2, 2, 1, 1}; + */ + + /* Height of interlace block. This is not currently used - if you need + * it, uncomment it here and in png.h + const int FARDATA png_pass_height[] = {8, 8, 4, 4, 2, 2, 1}; + */ +#endif + + png_ptr->row_number++; + if (png_ptr->row_number < png_ptr->num_rows) + return; + + if (png_ptr->interlaced) + { + png_ptr->row_number = 0; + png_memset_check(png_ptr, png_ptr->prev_row, 0, + png_ptr->rowbytes + 1); + do + { + png_ptr->pass++; + if ((png_ptr->pass == 1 && png_ptr->width < 5) || + (png_ptr->pass == 3 && png_ptr->width < 3) || + (png_ptr->pass == 5 && png_ptr->width < 2)) + png_ptr->pass++; + + if (png_ptr->pass > 7) + png_ptr->pass--; + if (png_ptr->pass >= 7) + break; + + png_ptr->iwidth = (png_ptr->width + + png_pass_inc[png_ptr->pass] - 1 - + png_pass_start[png_ptr->pass]) / + png_pass_inc[png_ptr->pass]; + + png_ptr->irowbytes = PNG_ROWBYTES(png_ptr->pixel_depth, + png_ptr->iwidth) + 1; + + if (png_ptr->transformations & PNG_INTERLACE) + break; + + png_ptr->num_rows = (png_ptr->height + + png_pass_yinc[png_ptr->pass] - 1 - + png_pass_ystart[png_ptr->pass]) / + png_pass_yinc[png_ptr->pass]; + + } while (png_ptr->iwidth == 0 || png_ptr->num_rows == 0); + } +} + +#if defined(PNG_READ_tEXt_SUPPORTED) +void /* PRIVATE */ +png_push_handle_tEXt(png_structp png_ptr, png_infop info_ptr, png_uint_32 + length) +{ + if (!(png_ptr->mode & PNG_HAVE_IHDR) || (png_ptr->mode & PNG_HAVE_IEND)) + { + png_error(png_ptr, "Out of place tEXt"); + /* to quiet some compiler warnings */ + if(info_ptr == NULL) return; + } + +#ifdef PNG_MAX_MALLOC_64K + png_ptr->skip_length = 0; /* This may not be necessary */ + + if (length > (png_uint_32)65535L) /* Can't hold entire string in memory */ + { + png_warning(png_ptr, "tEXt chunk too large to fit in memory"); + png_ptr->skip_length = length - (png_uint_32)65535L; + length = (png_uint_32)65535L; + } +#endif + + png_ptr->current_text = (png_charp)png_malloc(png_ptr, + (png_uint_32)(length+1)); + png_ptr->current_text[length] = '\0'; + png_ptr->current_text_ptr = png_ptr->current_text; + png_ptr->current_text_size = (png_size_t)length; + png_ptr->current_text_left = (png_size_t)length; + png_ptr->process_mode = PNG_READ_tEXt_MODE; +} + +void /* PRIVATE */ +png_push_read_tEXt(png_structp png_ptr, png_infop info_ptr) +{ + if (png_ptr->buffer_size && png_ptr->current_text_left) + { + png_size_t text_size; + + if (png_ptr->buffer_size < png_ptr->current_text_left) + text_size = png_ptr->buffer_size; + else + text_size = png_ptr->current_text_left; + png_crc_read(png_ptr, (png_bytep)png_ptr->current_text_ptr, text_size); + png_ptr->current_text_left -= text_size; + png_ptr->current_text_ptr += text_size; + } + if (!(png_ptr->current_text_left)) + { + png_textp text_ptr; + png_charp text; + png_charp key; + int ret; + + if (png_ptr->buffer_size < 4) + { + png_push_save_buffer(png_ptr); + return; + } + + png_push_crc_finish(png_ptr); + +#if defined(PNG_MAX_MALLOC_64K) + if (png_ptr->skip_length) + return; +#endif + + key = png_ptr->current_text; + + for (text = key; *text; text++) + /* empty loop */ ; + + if (text != key + png_ptr->current_text_size) + text++; + + text_ptr = (png_textp)png_malloc(png_ptr, + (png_uint_32)png_sizeof(png_text)); + text_ptr->compression = PNG_TEXT_COMPRESSION_NONE; + text_ptr->key = key; +#ifdef PNG_iTXt_SUPPORTED + text_ptr->lang = NULL; + text_ptr->lang_key = NULL; +#endif + text_ptr->text = text; + + ret = png_set_text_2(png_ptr, info_ptr, text_ptr, 1); + + png_free(png_ptr, key); + png_free(png_ptr, text_ptr); + png_ptr->current_text = NULL; + + if (ret) + png_warning(png_ptr, "Insufficient memory to store text chunk."); + } +} +#endif + +#if defined(PNG_READ_zTXt_SUPPORTED) +void /* PRIVATE */ +png_push_handle_zTXt(png_structp png_ptr, png_infop info_ptr, png_uint_32 + length) +{ + if (!(png_ptr->mode & PNG_HAVE_IHDR) || (png_ptr->mode & PNG_HAVE_IEND)) + { + png_error(png_ptr, "Out of place zTXt"); + /* to quiet some compiler warnings */ + if(info_ptr == NULL) return; + } + +#ifdef PNG_MAX_MALLOC_64K + /* We can't handle zTXt chunks > 64K, since we don't have enough space + * to be able to store the uncompressed data. Actually, the threshold + * is probably around 32K, but it isn't as definite as 64K is. + */ + if (length > (png_uint_32)65535L) + { + png_warning(png_ptr, "zTXt chunk too large to fit in memory"); + png_push_crc_skip(png_ptr, length); + return; + } +#endif + + png_ptr->current_text = (png_charp)png_malloc(png_ptr, + (png_uint_32)(length+1)); + png_ptr->current_text[length] = '\0'; + png_ptr->current_text_ptr = png_ptr->current_text; + png_ptr->current_text_size = (png_size_t)length; + png_ptr->current_text_left = (png_size_t)length; + png_ptr->process_mode = PNG_READ_zTXt_MODE; +} + +void /* PRIVATE */ +png_push_read_zTXt(png_structp png_ptr, png_infop info_ptr) +{ + if (png_ptr->buffer_size && png_ptr->current_text_left) + { + png_size_t text_size; + + if (png_ptr->buffer_size < (png_uint_32)png_ptr->current_text_left) + text_size = png_ptr->buffer_size; + else + text_size = png_ptr->current_text_left; + png_crc_read(png_ptr, (png_bytep)png_ptr->current_text_ptr, text_size); + png_ptr->current_text_left -= text_size; + png_ptr->current_text_ptr += text_size; + } + if (!(png_ptr->current_text_left)) + { + png_textp text_ptr; + png_charp text; + png_charp key; + int ret; + png_size_t text_size, key_size; + + if (png_ptr->buffer_size < 4) + { + png_push_save_buffer(png_ptr); + return; + } + + png_push_crc_finish(png_ptr); + + key = png_ptr->current_text; + + for (text = key; *text; text++) + /* empty loop */ ; + + /* zTXt can't have zero text */ + if (text == key + png_ptr->current_text_size) + { + png_ptr->current_text = NULL; + png_free(png_ptr, key); + return; + } + + text++; + + if (*text != PNG_TEXT_COMPRESSION_zTXt) /* check compression byte */ + { + png_ptr->current_text = NULL; + png_free(png_ptr, key); + return; + } + + text++; + + png_ptr->zstream.next_in = (png_bytep )text; + png_ptr->zstream.avail_in = (uInt)(png_ptr->current_text_size - + (text - key)); + png_ptr->zstream.next_out = png_ptr->zbuf; + png_ptr->zstream.avail_out = (uInt)png_ptr->zbuf_size; + + key_size = text - key; + text_size = 0; + text = NULL; + ret = Z_STREAM_END; + + while (png_ptr->zstream.avail_in) + { + ret = inflate(&png_ptr->zstream, Z_PARTIAL_FLUSH); + if (ret != Z_OK && ret != Z_STREAM_END) + { + inflateReset(&png_ptr->zstream); + png_ptr->zstream.avail_in = 0; + png_ptr->current_text = NULL; + png_free(png_ptr, key); + png_free(png_ptr, text); + return; + } + if (!(png_ptr->zstream.avail_out) || ret == Z_STREAM_END) + { + if (text == NULL) + { + text = (png_charp)png_malloc(png_ptr, + (png_uint_32)(png_ptr->zbuf_size - png_ptr->zstream.avail_out + + key_size + 1)); + png_memcpy(text + key_size, png_ptr->zbuf, + png_ptr->zbuf_size - png_ptr->zstream.avail_out); + png_memcpy(text, key, key_size); + text_size = key_size + png_ptr->zbuf_size - + png_ptr->zstream.avail_out; + *(text + text_size) = '\0'; + } + else + { + png_charp tmp; + + tmp = text; + text = (png_charp)png_malloc(png_ptr, text_size + + (png_uint_32)(png_ptr->zbuf_size - png_ptr->zstream.avail_out + + 1)); + png_memcpy(text, tmp, text_size); + png_free(png_ptr, tmp); + png_memcpy(text + text_size, png_ptr->zbuf, + png_ptr->zbuf_size - png_ptr->zstream.avail_out); + text_size += png_ptr->zbuf_size - png_ptr->zstream.avail_out; + *(text + text_size) = '\0'; + } + if (ret != Z_STREAM_END) + { + png_ptr->zstream.next_out = png_ptr->zbuf; + png_ptr->zstream.avail_out = (uInt)png_ptr->zbuf_size; + } + } + else + { + break; + } + + if (ret == Z_STREAM_END) + break; + } + + inflateReset(&png_ptr->zstream); + png_ptr->zstream.avail_in = 0; + + if (ret != Z_STREAM_END) + { + png_ptr->current_text = NULL; + png_free(png_ptr, key); + png_free(png_ptr, text); + return; + } + + png_ptr->current_text = NULL; + png_free(png_ptr, key); + key = text; + text += key_size; + + text_ptr = (png_textp)png_malloc(png_ptr, + (png_uint_32)png_sizeof(png_text)); + text_ptr->compression = PNG_TEXT_COMPRESSION_zTXt; + text_ptr->key = key; +#ifdef PNG_iTXt_SUPPORTED + text_ptr->lang = NULL; + text_ptr->lang_key = NULL; +#endif + text_ptr->text = text; + + ret = png_set_text_2(png_ptr, info_ptr, text_ptr, 1); + + png_free(png_ptr, key); + png_free(png_ptr, text_ptr); + + if (ret) + png_warning(png_ptr, "Insufficient memory to store text chunk."); + } +} +#endif + +#if defined(PNG_READ_iTXt_SUPPORTED) +void /* PRIVATE */ +png_push_handle_iTXt(png_structp png_ptr, png_infop info_ptr, png_uint_32 + length) +{ + if (!(png_ptr->mode & PNG_HAVE_IHDR) || (png_ptr->mode & PNG_HAVE_IEND)) + { + png_error(png_ptr, "Out of place iTXt"); + /* to quiet some compiler warnings */ + if(info_ptr == NULL) return; + } + +#ifdef PNG_MAX_MALLOC_64K + png_ptr->skip_length = 0; /* This may not be necessary */ + + if (length > (png_uint_32)65535L) /* Can't hold entire string in memory */ + { + png_warning(png_ptr, "iTXt chunk too large to fit in memory"); + png_ptr->skip_length = length - (png_uint_32)65535L; + length = (png_uint_32)65535L; + } +#endif + + png_ptr->current_text = (png_charp)png_malloc(png_ptr, + (png_uint_32)(length+1)); + png_ptr->current_text[length] = '\0'; + png_ptr->current_text_ptr = png_ptr->current_text; + png_ptr->current_text_size = (png_size_t)length; + png_ptr->current_text_left = (png_size_t)length; + png_ptr->process_mode = PNG_READ_iTXt_MODE; +} + +void /* PRIVATE */ +png_push_read_iTXt(png_structp png_ptr, png_infop info_ptr) +{ + + if (png_ptr->buffer_size && png_ptr->current_text_left) + { + png_size_t text_size; + + if (png_ptr->buffer_size < png_ptr->current_text_left) + text_size = png_ptr->buffer_size; + else + text_size = png_ptr->current_text_left; + png_crc_read(png_ptr, (png_bytep)png_ptr->current_text_ptr, text_size); + png_ptr->current_text_left -= text_size; + png_ptr->current_text_ptr += text_size; + } + if (!(png_ptr->current_text_left)) + { + png_textp text_ptr; + png_charp key; + int comp_flag; + png_charp lang; + png_charp lang_key; + png_charp text; + int ret; + + if (png_ptr->buffer_size < 4) + { + png_push_save_buffer(png_ptr); + return; + } + + png_push_crc_finish(png_ptr); + +#if defined(PNG_MAX_MALLOC_64K) + if (png_ptr->skip_length) + return; +#endif + + key = png_ptr->current_text; + + for (lang = key; *lang; lang++) + /* empty loop */ ; + + if (lang != key + png_ptr->current_text_size) + lang++; + + comp_flag = *lang++; + lang++; /* skip comp_type, always zero */ + + for (lang_key = lang; *lang_key; lang_key++) + /* empty loop */ ; + lang_key++; /* skip NUL separator */ + + for (text = lang_key; *text; text++) + /* empty loop */ ; + + if (text != key + png_ptr->current_text_size) + text++; + + text_ptr = (png_textp)png_malloc(png_ptr, + (png_uint_32)png_sizeof(png_text)); + text_ptr->compression = comp_flag + 2; + text_ptr->key = key; + text_ptr->lang = lang; + text_ptr->lang_key = lang_key; + text_ptr->text = text; + text_ptr->text_length = 0; + text_ptr->itxt_length = png_strlen(text); + + ret = png_set_text_2(png_ptr, info_ptr, text_ptr, 1); + + png_ptr->current_text = NULL; + + png_free(png_ptr, text_ptr); + if (ret) + png_warning(png_ptr, "Insufficient memory to store iTXt chunk."); + } +} +#endif + +/* This function is called when we haven't found a handler for this + * chunk. If there isn't a problem with the chunk itself (ie a bad chunk + * name or a critical chunk), the chunk is (currently) silently ignored. + */ +void /* PRIVATE */ +png_push_handle_unknown(png_structp png_ptr, png_infop info_ptr, png_uint_32 + length) +{ + png_uint_32 skip=0; + png_check_chunk_name(png_ptr, png_ptr->chunk_name); + + if (!(png_ptr->chunk_name[0] & 0x20)) + { +#if defined(PNG_READ_UNKNOWN_CHUNKS_SUPPORTED) + if(png_handle_as_unknown(png_ptr, png_ptr->chunk_name) != + PNG_HANDLE_CHUNK_ALWAYS +#if defined(PNG_READ_USER_CHUNKS_SUPPORTED) + && png_ptr->read_user_chunk_fn == NULL +#endif + ) +#endif + png_chunk_error(png_ptr, "unknown critical chunk"); + + /* to quiet compiler warnings about unused info_ptr */ + if (info_ptr == NULL) + return; + } + +#if defined(PNG_READ_UNKNOWN_CHUNKS_SUPPORTED) + if (png_ptr->flags & PNG_FLAG_KEEP_UNKNOWN_CHUNKS) + { + png_unknown_chunk chunk; + +#ifdef PNG_MAX_MALLOC_64K + if (length > (png_uint_32)65535L) + { + png_warning(png_ptr, "unknown chunk too large to fit in memory"); + skip = length - (png_uint_32)65535L; + length = (png_uint_32)65535L; + } +#endif + + png_strcpy((png_charp)chunk.name, (png_charp)png_ptr->chunk_name); + chunk.data = (png_bytep)png_malloc(png_ptr, length); + png_crc_read(png_ptr, chunk.data, length); + chunk.size = length; +#if defined(PNG_READ_USER_CHUNKS_SUPPORTED) + if(png_ptr->read_user_chunk_fn != NULL) + { + /* callback to user unknown chunk handler */ + if ((*(png_ptr->read_user_chunk_fn)) (png_ptr, &chunk) <= 0) + { + if (!(png_ptr->chunk_name[0] & 0x20)) + if(png_handle_as_unknown(png_ptr, png_ptr->chunk_name) != + PNG_HANDLE_CHUNK_ALWAYS) + png_chunk_error(png_ptr, "unknown critical chunk"); + } + png_set_unknown_chunks(png_ptr, info_ptr, &chunk, 1); + } + else +#endif + png_set_unknown_chunks(png_ptr, info_ptr, &chunk, 1); + png_free(png_ptr, chunk.data); + } + else +#endif + skip=length; + png_push_crc_skip(png_ptr, skip); +} + +void /* PRIVATE */ +png_push_have_info(png_structp png_ptr, png_infop info_ptr) +{ + if (png_ptr->info_fn != NULL) + (*(png_ptr->info_fn))(png_ptr, info_ptr); +} + +void /* PRIVATE */ +png_push_have_end(png_structp png_ptr, png_infop info_ptr) +{ + if (png_ptr->end_fn != NULL) + (*(png_ptr->end_fn))(png_ptr, info_ptr); +} + +void /* PRIVATE */ +png_push_have_row(png_structp png_ptr, png_bytep row) +{ + if (png_ptr->row_fn != NULL) + (*(png_ptr->row_fn))(png_ptr, row, png_ptr->row_number, + (int)png_ptr->pass); +} + +void PNGAPI +png_progressive_combine_row (png_structp png_ptr, + png_bytep old_row, png_bytep new_row) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + const int FARDATA png_pass_dsp_mask[7] = + {0xff, 0x0f, 0xff, 0x33, 0xff, 0x55, 0xff}; +#endif + if (new_row != NULL) /* new_row must == png_ptr->row_buf here. */ + png_combine_row(png_ptr, old_row, png_pass_dsp_mask[png_ptr->pass]); +} + +void PNGAPI +png_set_progressive_read_fn(png_structp png_ptr, png_voidp progressive_ptr, + png_progressive_info_ptr info_fn, png_progressive_row_ptr row_fn, + png_progressive_end_ptr end_fn) +{ + png_ptr->info_fn = info_fn; + png_ptr->row_fn = row_fn; + png_ptr->end_fn = end_fn; + + png_set_read_fn(png_ptr, progressive_ptr, png_push_fill_buffer); +} + +png_voidp PNGAPI +png_get_progressive_ptr(png_structp png_ptr) +{ + return png_ptr->io_ptr; +} +#endif /* PNG_PROGRESSIVE_READ_SUPPORTED */ diff --git a/src/win32/dependencies/libpng/pngread.c b/src/win32/dependencies/libpng/pngread.c new file mode 100644 index 00000000..e99639f0 --- /dev/null +++ b/src/win32/dependencies/libpng/pngread.c @@ -0,0 +1,1459 @@ + +/* pngread.c - read a PNG file + * + * Last changed in libpng 1.2.9 April 14, 2006 + * For conditions of distribution and use, see copyright notice in png.h + * Copyright (c) 1998-2006 Glenn Randers-Pehrson + * (Version 0.96 Copyright (c) 1996, 1997 Andreas Dilger) + * (Version 0.88 Copyright (c) 1995, 1996 Guy Eric Schalnat, Group 42, Inc.) + * + * This file contains routines that an application calls directly to + * read a PNG file or stream. + */ + +#define PNG_INTERNAL +#include "png.h" + +#if defined(PNG_READ_SUPPORTED) + +/* Create a PNG structure for reading, and allocate any memory needed. */ +png_structp PNGAPI +png_create_read_struct(png_const_charp user_png_ver, png_voidp error_ptr, + png_error_ptr error_fn, png_error_ptr warn_fn) +{ + +#ifdef PNG_USER_MEM_SUPPORTED + return (png_create_read_struct_2(user_png_ver, error_ptr, error_fn, + warn_fn, png_voidp_NULL, png_malloc_ptr_NULL, png_free_ptr_NULL)); +} + +/* Alternate create PNG structure for reading, and allocate any memory needed. */ +png_structp PNGAPI +png_create_read_struct_2(png_const_charp user_png_ver, png_voidp error_ptr, + png_error_ptr error_fn, png_error_ptr warn_fn, png_voidp mem_ptr, + png_malloc_ptr malloc_fn, png_free_ptr free_fn) +{ +#endif /* PNG_USER_MEM_SUPPORTED */ + + png_structp png_ptr; + +#ifdef PNG_SETJMP_SUPPORTED +#ifdef USE_FAR_KEYWORD + jmp_buf jmpbuf; +#endif +#endif + + int i; + + png_debug(1, "in png_create_read_struct\n"); +#ifdef PNG_USER_MEM_SUPPORTED + png_ptr = (png_structp)png_create_struct_2(PNG_STRUCT_PNG, + (png_malloc_ptr)malloc_fn, (png_voidp)mem_ptr); +#else + png_ptr = (png_structp)png_create_struct(PNG_STRUCT_PNG); +#endif + if (png_ptr == NULL) + return (NULL); + +#if !defined(PNG_1_0_X) +#ifdef PNG_ASSEMBLER_CODE_SUPPORTED + png_init_mmx_flags(png_ptr); /* 1.2.0 addition */ +#endif +#endif /* PNG_1_0_X */ + + /* added at libpng-1.2.6 */ +#ifdef PNG_SET_USER_LIMITS_SUPPORTED + png_ptr->user_width_max=PNG_USER_WIDTH_MAX; + png_ptr->user_height_max=PNG_USER_HEIGHT_MAX; +#endif + +#ifdef PNG_SETJMP_SUPPORTED +#ifdef USE_FAR_KEYWORD + if (setjmp(jmpbuf)) +#else + if (setjmp(png_ptr->jmpbuf)) +#endif + { + png_free(png_ptr, png_ptr->zbuf); + png_ptr->zbuf=NULL; +#ifdef PNG_USER_MEM_SUPPORTED + png_destroy_struct_2((png_voidp)png_ptr, + (png_free_ptr)free_fn, (png_voidp)mem_ptr); +#else + png_destroy_struct((png_voidp)png_ptr); +#endif + return (NULL); + } +#ifdef USE_FAR_KEYWORD + png_memcpy(png_ptr->jmpbuf,jmpbuf,png_sizeof(jmp_buf)); +#endif +#endif + +#ifdef PNG_USER_MEM_SUPPORTED + png_set_mem_fn(png_ptr, mem_ptr, malloc_fn, free_fn); +#endif + + png_set_error_fn(png_ptr, error_ptr, error_fn, warn_fn); + + i=0; + do + { + if(user_png_ver[i] != png_libpng_ver[i]) + png_ptr->flags |= PNG_FLAG_LIBRARY_MISMATCH; + } while (png_libpng_ver[i++]); + + if (png_ptr->flags & PNG_FLAG_LIBRARY_MISMATCH) + { + /* Libpng 0.90 and later are binary incompatible with libpng 0.89, so + * we must recompile any applications that use any older library version. + * For versions after libpng 1.0, we will be compatible, so we need + * only check the first digit. + */ + if (user_png_ver == NULL || user_png_ver[0] != png_libpng_ver[0] || + (user_png_ver[0] == '1' && user_png_ver[2] != png_libpng_ver[2]) || + (user_png_ver[0] == '0' && user_png_ver[2] < '9')) + { +#if !defined(PNG_NO_STDIO) && !defined(_WIN32_WCE) + char msg[80]; + if (user_png_ver) + { + sprintf(msg, "Application was compiled with png.h from libpng-%.20s", + user_png_ver); + png_warning(png_ptr, msg); + } + sprintf(msg, "Application is running with png.c from libpng-%.20s", + png_libpng_ver); + png_warning(png_ptr, msg); +#endif +#ifdef PNG_ERROR_NUMBERS_SUPPORTED + png_ptr->flags=0; +#endif + png_error(png_ptr, + "Incompatible libpng version in application and library"); + } + } + + /* initialize zbuf - compression buffer */ + png_ptr->zbuf_size = PNG_ZBUF_SIZE; + png_ptr->zbuf = (png_bytep)png_malloc(png_ptr, + (png_uint_32)png_ptr->zbuf_size); + png_ptr->zstream.zalloc = png_zalloc; + png_ptr->zstream.zfree = png_zfree; + png_ptr->zstream.opaque = (voidpf)png_ptr; + + switch (inflateInit(&png_ptr->zstream)) + { + case Z_OK: /* Do nothing */ break; + case Z_MEM_ERROR: + case Z_STREAM_ERROR: png_error(png_ptr, "zlib memory error"); break; + case Z_VERSION_ERROR: png_error(png_ptr, "zlib version error"); break; + default: png_error(png_ptr, "Unknown zlib error"); + } + + png_ptr->zstream.next_out = png_ptr->zbuf; + png_ptr->zstream.avail_out = (uInt)png_ptr->zbuf_size; + + png_set_read_fn(png_ptr, png_voidp_NULL, png_rw_ptr_NULL); + +#ifdef PNG_SETJMP_SUPPORTED +/* Applications that neglect to set up their own setjmp() and then encounter + a png_error() will longjmp here. Since the jmpbuf is then meaningless we + abort instead of returning. */ +#ifdef USE_FAR_KEYWORD + if (setjmp(jmpbuf)) + PNG_ABORT(); + png_memcpy(png_ptr->jmpbuf,jmpbuf,png_sizeof(jmp_buf)); +#else + if (setjmp(png_ptr->jmpbuf)) + PNG_ABORT(); +#endif +#endif + return (png_ptr); +} + +#if defined(PNG_1_0_X) || defined(PNG_1_2_X) +/* Initialize PNG structure for reading, and allocate any memory needed. + This interface is deprecated in favour of the png_create_read_struct(), + and it will disappear as of libpng-1.3.0. */ +#undef png_read_init +void PNGAPI +png_read_init(png_structp png_ptr) +{ + /* We only come here via pre-1.0.7-compiled applications */ + png_read_init_2(png_ptr, "1.0.6 or earlier", 0, 0); +} + +void PNGAPI +png_read_init_2(png_structp png_ptr, png_const_charp user_png_ver, + png_size_t png_struct_size, png_size_t png_info_size) +{ + /* We only come here via pre-1.0.12-compiled applications */ +#if !defined(PNG_NO_STDIO) && !defined(_WIN32_WCE) + if(png_sizeof(png_struct) > png_struct_size || + png_sizeof(png_info) > png_info_size) + { + char msg[80]; + png_ptr->warning_fn=NULL; + if (user_png_ver) + { + sprintf(msg, "Application was compiled with png.h from libpng-%.20s", + user_png_ver); + png_warning(png_ptr, msg); + } + sprintf(msg, "Application is running with png.c from libpng-%.20s", + png_libpng_ver); + png_warning(png_ptr, msg); + } +#endif + if(png_sizeof(png_struct) > png_struct_size) + { + png_ptr->error_fn=NULL; +#ifdef PNG_ERROR_NUMBERS_SUPPORTED + png_ptr->flags=0; +#endif + png_error(png_ptr, + "The png struct allocated by the application for reading is too small."); + } + if(png_sizeof(png_info) > png_info_size) + { + png_ptr->error_fn=NULL; +#ifdef PNG_ERROR_NUMBERS_SUPPORTED + png_ptr->flags=0; +#endif + png_error(png_ptr, + "The info struct allocated by application for reading is too small."); + } + png_read_init_3(&png_ptr, user_png_ver, png_struct_size); +} +#endif /* PNG_1_0_X || PNG_1_2_X */ + +void PNGAPI +png_read_init_3(png_structpp ptr_ptr, png_const_charp user_png_ver, + png_size_t png_struct_size) +{ +#ifdef PNG_SETJMP_SUPPORTED + jmp_buf tmp_jmp; /* to save current jump buffer */ +#endif + + int i=0; + + png_structp png_ptr=*ptr_ptr; + + do + { + if(user_png_ver[i] != png_libpng_ver[i]) + { +#ifdef PNG_LEGACY_SUPPORTED + png_ptr->flags |= PNG_FLAG_LIBRARY_MISMATCH; +#else + png_ptr->warning_fn=NULL; + png_warning(png_ptr, + "Application uses deprecated png_read_init() and should be recompiled."); + break; +#endif + } + } while (png_libpng_ver[i++]); + + png_debug(1, "in png_read_init_3\n"); + +#ifdef PNG_SETJMP_SUPPORTED + /* save jump buffer and error functions */ + png_memcpy(tmp_jmp, png_ptr->jmpbuf, png_sizeof (jmp_buf)); +#endif + + if(png_sizeof(png_struct) > png_struct_size) + { + png_destroy_struct(png_ptr); + *ptr_ptr = (png_structp)png_create_struct(PNG_STRUCT_PNG); + png_ptr = *ptr_ptr; + } + + /* reset all variables to 0 */ + png_memset(png_ptr, 0, png_sizeof (png_struct)); + +#ifdef PNG_SETJMP_SUPPORTED + /* restore jump buffer */ + png_memcpy(png_ptr->jmpbuf, tmp_jmp, png_sizeof (jmp_buf)); +#endif + + /* added at libpng-1.2.6 */ +#ifdef PNG_SET_USER_LIMITS_SUPPORTED + png_ptr->user_width_max=PNG_USER_WIDTH_MAX; + png_ptr->user_height_max=PNG_USER_HEIGHT_MAX; +#endif + + /* initialize zbuf - compression buffer */ + png_ptr->zbuf_size = PNG_ZBUF_SIZE; + png_ptr->zbuf = (png_bytep)png_malloc(png_ptr, + (png_uint_32)png_ptr->zbuf_size); + png_ptr->zstream.zalloc = png_zalloc; + png_ptr->zstream.zfree = png_zfree; + png_ptr->zstream.opaque = (voidpf)png_ptr; + + switch (inflateInit(&png_ptr->zstream)) + { + case Z_OK: /* Do nothing */ break; + case Z_MEM_ERROR: + case Z_STREAM_ERROR: png_error(png_ptr, "zlib memory"); break; + case Z_VERSION_ERROR: png_error(png_ptr, "zlib version"); break; + default: png_error(png_ptr, "Unknown zlib error"); + } + + png_ptr->zstream.next_out = png_ptr->zbuf; + png_ptr->zstream.avail_out = (uInt)png_ptr->zbuf_size; + + png_set_read_fn(png_ptr, png_voidp_NULL, png_rw_ptr_NULL); +} + +#ifndef PNG_NO_SEQUENTIAL_READ_SUPPORTED +/* Read the information before the actual image data. This has been + * changed in v0.90 to allow reading a file that already has the magic + * bytes read from the stream. You can tell libpng how many bytes have + * been read from the beginning of the stream (up to the maximum of 8) + * via png_set_sig_bytes(), and we will only check the remaining bytes + * here. The application can then have access to the signature bytes we + * read if it is determined that this isn't a valid PNG file. + */ +void PNGAPI +png_read_info(png_structp png_ptr, png_infop info_ptr) +{ + png_debug(1, "in png_read_info\n"); + /* If we haven't checked all of the PNG signature bytes, do so now. */ + if (png_ptr->sig_bytes < 8) + { + png_size_t num_checked = png_ptr->sig_bytes, + num_to_check = 8 - num_checked; + + png_read_data(png_ptr, &(info_ptr->signature[num_checked]), num_to_check); + png_ptr->sig_bytes = 8; + + if (png_sig_cmp(info_ptr->signature, num_checked, num_to_check)) + { + if (num_checked < 4 && + png_sig_cmp(info_ptr->signature, num_checked, num_to_check - 4)) + png_error(png_ptr, "Not a PNG file"); + else + png_error(png_ptr, "PNG file corrupted by ASCII conversion"); + } + if (num_checked < 3) + png_ptr->mode |= PNG_HAVE_PNG_SIGNATURE; + } + + for(;;) + { +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_IHDR; + PNG_IDAT; + PNG_IEND; + PNG_PLTE; +#if defined(PNG_READ_bKGD_SUPPORTED) + PNG_bKGD; +#endif +#if defined(PNG_READ_cHRM_SUPPORTED) + PNG_cHRM; +#endif +#if defined(PNG_READ_gAMA_SUPPORTED) + PNG_gAMA; +#endif +#if defined(PNG_READ_hIST_SUPPORTED) + PNG_hIST; +#endif +#if defined(PNG_READ_iCCP_SUPPORTED) + PNG_iCCP; +#endif +#if defined(PNG_READ_iTXt_SUPPORTED) + PNG_iTXt; +#endif +#if defined(PNG_READ_oFFs_SUPPORTED) + PNG_oFFs; +#endif +#if defined(PNG_READ_pCAL_SUPPORTED) + PNG_pCAL; +#endif +#if defined(PNG_READ_pHYs_SUPPORTED) + PNG_pHYs; +#endif +#if defined(PNG_READ_sBIT_SUPPORTED) + PNG_sBIT; +#endif +#if defined(PNG_READ_sCAL_SUPPORTED) + PNG_sCAL; +#endif +#if defined(PNG_READ_sPLT_SUPPORTED) + PNG_sPLT; +#endif +#if defined(PNG_READ_sRGB_SUPPORTED) + PNG_sRGB; +#endif +#if defined(PNG_READ_tEXt_SUPPORTED) + PNG_tEXt; +#endif +#if defined(PNG_READ_tIME_SUPPORTED) + PNG_tIME; +#endif +#if defined(PNG_READ_tRNS_SUPPORTED) + PNG_tRNS; +#endif +#if defined(PNG_READ_zTXt_SUPPORTED) + PNG_zTXt; +#endif +#endif /* PNG_USE_LOCAL_ARRAYS */ + png_byte chunk_length[4]; + png_uint_32 length; + + png_read_data(png_ptr, chunk_length, 4); + length = png_get_uint_31(png_ptr,chunk_length); + + png_reset_crc(png_ptr); + png_crc_read(png_ptr, png_ptr->chunk_name, 4); + + png_debug2(0, "Reading %s chunk, length=%lu.\n", png_ptr->chunk_name, + length); + + /* This should be a binary subdivision search or a hash for + * matching the chunk name rather than a linear search. + */ + if (!png_memcmp(png_ptr->chunk_name, png_IHDR, 4)) + png_handle_IHDR(png_ptr, info_ptr, length); + else if (!png_memcmp(png_ptr->chunk_name, png_IEND, 4)) + png_handle_IEND(png_ptr, info_ptr, length); +#ifdef PNG_HANDLE_AS_UNKNOWN_SUPPORTED + else if (png_handle_as_unknown(png_ptr, png_ptr->chunk_name)) + { + if (!png_memcmp(png_ptr->chunk_name, png_IDAT, 4)) + png_ptr->mode |= PNG_HAVE_IDAT; + png_handle_unknown(png_ptr, info_ptr, length); + if (!png_memcmp(png_ptr->chunk_name, png_PLTE, 4)) + png_ptr->mode |= PNG_HAVE_PLTE; + else if (!png_memcmp(png_ptr->chunk_name, png_IDAT, 4)) + { + if (!(png_ptr->mode & PNG_HAVE_IHDR)) + png_error(png_ptr, "Missing IHDR before IDAT"); + else if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE && + !(png_ptr->mode & PNG_HAVE_PLTE)) + png_error(png_ptr, "Missing PLTE before IDAT"); + break; + } + } +#endif + else if (!png_memcmp(png_ptr->chunk_name, png_PLTE, 4)) + png_handle_PLTE(png_ptr, info_ptr, length); + else if (!png_memcmp(png_ptr->chunk_name, png_IDAT, 4)) + { + if (!(png_ptr->mode & PNG_HAVE_IHDR)) + png_error(png_ptr, "Missing IHDR before IDAT"); + else if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE && + !(png_ptr->mode & PNG_HAVE_PLTE)) + png_error(png_ptr, "Missing PLTE before IDAT"); + + png_ptr->idat_size = length; + png_ptr->mode |= PNG_HAVE_IDAT; + break; + } +#if defined(PNG_READ_bKGD_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_bKGD, 4)) + png_handle_bKGD(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_cHRM_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_cHRM, 4)) + png_handle_cHRM(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_gAMA_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_gAMA, 4)) + png_handle_gAMA(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_hIST_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_hIST, 4)) + png_handle_hIST(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_oFFs_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_oFFs, 4)) + png_handle_oFFs(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_pCAL_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_pCAL, 4)) + png_handle_pCAL(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_sCAL_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_sCAL, 4)) + png_handle_sCAL(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_pHYs_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_pHYs, 4)) + png_handle_pHYs(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_sBIT_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_sBIT, 4)) + png_handle_sBIT(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_sRGB_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_sRGB, 4)) + png_handle_sRGB(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_iCCP_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_iCCP, 4)) + png_handle_iCCP(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_sPLT_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_sPLT, 4)) + png_handle_sPLT(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_tEXt_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_tEXt, 4)) + png_handle_tEXt(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_tIME_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_tIME, 4)) + png_handle_tIME(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_tRNS_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_tRNS, 4)) + png_handle_tRNS(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_zTXt_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_zTXt, 4)) + png_handle_zTXt(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_iTXt_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_iTXt, 4)) + png_handle_iTXt(png_ptr, info_ptr, length); +#endif + else + png_handle_unknown(png_ptr, info_ptr, length); + } +} +#endif /* PNG_NO_SEQUENTIAL_READ_SUPPORTED */ + +/* optional call to update the users info_ptr structure */ +void PNGAPI +png_read_update_info(png_structp png_ptr, png_infop info_ptr) +{ + png_debug(1, "in png_read_update_info\n"); + if (!(png_ptr->flags & PNG_FLAG_ROW_INIT)) + png_read_start_row(png_ptr); + else + png_warning(png_ptr, + "Ignoring extra png_read_update_info() call; row buffer not reallocated"); + png_read_transform_info(png_ptr, info_ptr); +} + +#ifndef PNG_NO_SEQUENTIAL_READ_SUPPORTED +/* Initialize palette, background, etc, after transformations + * are set, but before any reading takes place. This allows + * the user to obtain a gamma-corrected palette, for example. + * If the user doesn't call this, we will do it ourselves. + */ +void PNGAPI +png_start_read_image(png_structp png_ptr) +{ + png_debug(1, "in png_start_read_image\n"); + if (!(png_ptr->flags & PNG_FLAG_ROW_INIT)) + png_read_start_row(png_ptr); +} +#endif /* PNG_NO_SEQUENTIAL_READ_SUPPORTED */ + +#ifndef PNG_NO_SEQUENTIAL_READ_SUPPORTED +void PNGAPI +png_read_row(png_structp png_ptr, png_bytep row, png_bytep dsp_row) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_IDAT; + const int png_pass_dsp_mask[7] = {0xff, 0x0f, 0xff, 0x33, 0xff, 0x55, 0xff}; + const int png_pass_mask[7] = {0x80, 0x08, 0x88, 0x22, 0xaa, 0x55, 0xff}; +#endif + int ret; + png_debug2(1, "in png_read_row (row %lu, pass %d)\n", + png_ptr->row_number, png_ptr->pass); + if (!(png_ptr->flags & PNG_FLAG_ROW_INIT)) + png_read_start_row(png_ptr); + if (png_ptr->row_number == 0 && png_ptr->pass == 0) + { + /* check for transforms that have been set but were defined out */ +#if defined(PNG_WRITE_INVERT_SUPPORTED) && !defined(PNG_READ_INVERT_SUPPORTED) + if (png_ptr->transformations & PNG_INVERT_MONO) + png_warning(png_ptr, "PNG_READ_INVERT_SUPPORTED is not defined."); +#endif +#if defined(PNG_WRITE_FILLER_SUPPORTED) && !defined(PNG_READ_FILLER_SUPPORTED) + if (png_ptr->transformations & PNG_FILLER) + png_warning(png_ptr, "PNG_READ_FILLER_SUPPORTED is not defined."); +#endif +#if defined(PNG_WRITE_PACKSWAP_SUPPORTED) && !defined(PNG_READ_PACKSWAP_SUPPORTED) + if (png_ptr->transformations & PNG_PACKSWAP) + png_warning(png_ptr, "PNG_READ_PACKSWAP_SUPPORTED is not defined."); +#endif +#if defined(PNG_WRITE_PACK_SUPPORTED) && !defined(PNG_READ_PACK_SUPPORTED) + if (png_ptr->transformations & PNG_PACK) + png_warning(png_ptr, "PNG_READ_PACK_SUPPORTED is not defined."); +#endif +#if defined(PNG_WRITE_SHIFT_SUPPORTED) && !defined(PNG_READ_SHIFT_SUPPORTED) + if (png_ptr->transformations & PNG_SHIFT) + png_warning(png_ptr, "PNG_READ_SHIFT_SUPPORTED is not defined."); +#endif +#if defined(PNG_WRITE_BGR_SUPPORTED) && !defined(PNG_READ_BGR_SUPPORTED) + if (png_ptr->transformations & PNG_BGR) + png_warning(png_ptr, "PNG_READ_BGR_SUPPORTED is not defined."); +#endif +#if defined(PNG_WRITE_SWAP_SUPPORTED) && !defined(PNG_READ_SWAP_SUPPORTED) + if (png_ptr->transformations & PNG_SWAP_BYTES) + png_warning(png_ptr, "PNG_READ_SWAP_SUPPORTED is not defined."); +#endif + } + +#if defined(PNG_READ_INTERLACING_SUPPORTED) + /* if interlaced and we do not need a new row, combine row and return */ + if (png_ptr->interlaced && (png_ptr->transformations & PNG_INTERLACE)) + { + switch (png_ptr->pass) + { + case 0: + if (png_ptr->row_number & 0x07) + { + if (dsp_row != NULL) + png_combine_row(png_ptr, dsp_row, + png_pass_dsp_mask[png_ptr->pass]); + png_read_finish_row(png_ptr); + return; + } + break; + case 1: + if ((png_ptr->row_number & 0x07) || png_ptr->width < 5) + { + if (dsp_row != NULL) + png_combine_row(png_ptr, dsp_row, + png_pass_dsp_mask[png_ptr->pass]); + png_read_finish_row(png_ptr); + return; + } + break; + case 2: + if ((png_ptr->row_number & 0x07) != 4) + { + if (dsp_row != NULL && (png_ptr->row_number & 4)) + png_combine_row(png_ptr, dsp_row, + png_pass_dsp_mask[png_ptr->pass]); + png_read_finish_row(png_ptr); + return; + } + break; + case 3: + if ((png_ptr->row_number & 3) || png_ptr->width < 3) + { + if (dsp_row != NULL) + png_combine_row(png_ptr, dsp_row, + png_pass_dsp_mask[png_ptr->pass]); + png_read_finish_row(png_ptr); + return; + } + break; + case 4: + if ((png_ptr->row_number & 3) != 2) + { + if (dsp_row != NULL && (png_ptr->row_number & 2)) + png_combine_row(png_ptr, dsp_row, + png_pass_dsp_mask[png_ptr->pass]); + png_read_finish_row(png_ptr); + return; + } + break; + case 5: + if ((png_ptr->row_number & 1) || png_ptr->width < 2) + { + if (dsp_row != NULL) + png_combine_row(png_ptr, dsp_row, + png_pass_dsp_mask[png_ptr->pass]); + png_read_finish_row(png_ptr); + return; + } + break; + case 6: + if (!(png_ptr->row_number & 1)) + { + png_read_finish_row(png_ptr); + return; + } + break; + } + } +#endif + + if (!(png_ptr->mode & PNG_HAVE_IDAT)) + png_error(png_ptr, "Invalid attempt to read row data"); + + png_ptr->zstream.next_out = png_ptr->row_buf; + png_ptr->zstream.avail_out = (uInt)png_ptr->irowbytes; + do + { + if (!(png_ptr->zstream.avail_in)) + { + while (!png_ptr->idat_size) + { + png_byte chunk_length[4]; + + png_crc_finish(png_ptr, 0); + + png_read_data(png_ptr, chunk_length, 4); + png_ptr->idat_size = png_get_uint_31(png_ptr,chunk_length); + + png_reset_crc(png_ptr); + png_crc_read(png_ptr, png_ptr->chunk_name, 4); + if (png_memcmp(png_ptr->chunk_name, png_IDAT, 4)) + png_error(png_ptr, "Not enough image data"); + } + png_ptr->zstream.avail_in = (uInt)png_ptr->zbuf_size; + png_ptr->zstream.next_in = png_ptr->zbuf; + if (png_ptr->zbuf_size > png_ptr->idat_size) + png_ptr->zstream.avail_in = (uInt)png_ptr->idat_size; + png_crc_read(png_ptr, png_ptr->zbuf, + (png_size_t)png_ptr->zstream.avail_in); + png_ptr->idat_size -= png_ptr->zstream.avail_in; + } + ret = inflate(&png_ptr->zstream, Z_PARTIAL_FLUSH); + if (ret == Z_STREAM_END) + { + if (png_ptr->zstream.avail_out || png_ptr->zstream.avail_in || + png_ptr->idat_size) + png_error(png_ptr, "Extra compressed data"); + png_ptr->mode |= PNG_AFTER_IDAT; + png_ptr->flags |= PNG_FLAG_ZLIB_FINISHED; + break; + } + if (ret != Z_OK) + png_error(png_ptr, png_ptr->zstream.msg ? png_ptr->zstream.msg : + "Decompression error"); + + } while (png_ptr->zstream.avail_out); + + png_ptr->row_info.color_type = png_ptr->color_type; + png_ptr->row_info.width = png_ptr->iwidth; + png_ptr->row_info.channels = png_ptr->channels; + png_ptr->row_info.bit_depth = png_ptr->bit_depth; + png_ptr->row_info.pixel_depth = png_ptr->pixel_depth; + png_ptr->row_info.rowbytes = PNG_ROWBYTES(png_ptr->row_info.pixel_depth, + png_ptr->row_info.width); + + if(png_ptr->row_buf[0]) + png_read_filter_row(png_ptr, &(png_ptr->row_info), + png_ptr->row_buf + 1, png_ptr->prev_row + 1, + (int)(png_ptr->row_buf[0])); + + png_memcpy_check(png_ptr, png_ptr->prev_row, png_ptr->row_buf, + png_ptr->rowbytes + 1); + +#if defined(PNG_MNG_FEATURES_SUPPORTED) + if((png_ptr->mng_features_permitted & PNG_FLAG_MNG_FILTER_64) && + (png_ptr->filter_type == PNG_INTRAPIXEL_DIFFERENCING)) + { + /* Intrapixel differencing */ + png_do_read_intrapixel(&(png_ptr->row_info), png_ptr->row_buf + 1); + } +#endif + + + if (png_ptr->transformations || (png_ptr->flags&PNG_FLAG_STRIP_ALPHA)) + png_do_read_transformations(png_ptr); + +#if defined(PNG_READ_INTERLACING_SUPPORTED) + /* blow up interlaced rows to full size */ + if (png_ptr->interlaced && + (png_ptr->transformations & PNG_INTERLACE)) + { + if (png_ptr->pass < 6) +/* old interface (pre-1.0.9): + png_do_read_interlace(&(png_ptr->row_info), + png_ptr->row_buf + 1, png_ptr->pass, png_ptr->transformations); + */ + png_do_read_interlace(png_ptr); + + if (dsp_row != NULL) + png_combine_row(png_ptr, dsp_row, + png_pass_dsp_mask[png_ptr->pass]); + if (row != NULL) + png_combine_row(png_ptr, row, + png_pass_mask[png_ptr->pass]); + } + else +#endif + { + if (row != NULL) + png_combine_row(png_ptr, row, 0xff); + if (dsp_row != NULL) + png_combine_row(png_ptr, dsp_row, 0xff); + } + png_read_finish_row(png_ptr); + + if (png_ptr->read_row_fn != NULL) + (*(png_ptr->read_row_fn))(png_ptr, png_ptr->row_number, png_ptr->pass); +} +#endif /* PNG_NO_SEQUENTIAL_READ_SUPPORTED */ + +#ifndef PNG_NO_SEQUENTIAL_READ_SUPPORTED +/* Read one or more rows of image data. If the image is interlaced, + * and png_set_interlace_handling() has been called, the rows need to + * contain the contents of the rows from the previous pass. If the + * image has alpha or transparency, and png_handle_alpha()[*] has been + * called, the rows contents must be initialized to the contents of the + * screen. + * + * "row" holds the actual image, and pixels are placed in it + * as they arrive. If the image is displayed after each pass, it will + * appear to "sparkle" in. "display_row" can be used to display a + * "chunky" progressive image, with finer detail added as it becomes + * available. If you do not want this "chunky" display, you may pass + * NULL for display_row. If you do not want the sparkle display, and + * you have not called png_handle_alpha(), you may pass NULL for rows. + * If you have called png_handle_alpha(), and the image has either an + * alpha channel or a transparency chunk, you must provide a buffer for + * rows. In this case, you do not have to provide a display_row buffer + * also, but you may. If the image is not interlaced, or if you have + * not called png_set_interlace_handling(), the display_row buffer will + * be ignored, so pass NULL to it. + * + * [*] png_handle_alpha() does not exist yet, as of this version of libpng + */ + +void PNGAPI +png_read_rows(png_structp png_ptr, png_bytepp row, + png_bytepp display_row, png_uint_32 num_rows) +{ + png_uint_32 i; + png_bytepp rp; + png_bytepp dp; + + png_debug(1, "in png_read_rows\n"); + rp = row; + dp = display_row; + if (rp != NULL && dp != NULL) + for (i = 0; i < num_rows; i++) + { + png_bytep rptr = *rp++; + png_bytep dptr = *dp++; + + png_read_row(png_ptr, rptr, dptr); + } + else if(rp != NULL) + for (i = 0; i < num_rows; i++) + { + png_bytep rptr = *rp; + png_read_row(png_ptr, rptr, png_bytep_NULL); + rp++; + } + else if(dp != NULL) + for (i = 0; i < num_rows; i++) + { + png_bytep dptr = *dp; + png_read_row(png_ptr, png_bytep_NULL, dptr); + dp++; + } +} +#endif /* PNG_NO_SEQUENTIAL_READ_SUPPORTED */ + +#ifndef PNG_NO_SEQUENTIAL_READ_SUPPORTED +/* Read the entire image. If the image has an alpha channel or a tRNS + * chunk, and you have called png_handle_alpha()[*], you will need to + * initialize the image to the current image that PNG will be overlaying. + * We set the num_rows again here, in case it was incorrectly set in + * png_read_start_row() by a call to png_read_update_info() or + * png_start_read_image() if png_set_interlace_handling() wasn't called + * prior to either of these functions like it should have been. You can + * only call this function once. If you desire to have an image for + * each pass of a interlaced image, use png_read_rows() instead. + * + * [*] png_handle_alpha() does not exist yet, as of this version of libpng + */ +void PNGAPI +png_read_image(png_structp png_ptr, png_bytepp image) +{ + png_uint_32 i,image_height; + int pass, j; + png_bytepp rp; + + png_debug(1, "in png_read_image\n"); + +#ifdef PNG_READ_INTERLACING_SUPPORTED + pass = png_set_interlace_handling(png_ptr); +#else + if (png_ptr->interlaced) + png_error(png_ptr, + "Cannot read interlaced image -- interlace handler disabled."); + pass = 1; +#endif + + + image_height=png_ptr->height; + png_ptr->num_rows = image_height; /* Make sure this is set correctly */ + + for (j = 0; j < pass; j++) + { + rp = image; + for (i = 0; i < image_height; i++) + { + png_read_row(png_ptr, *rp, png_bytep_NULL); + rp++; + } + } +} +#endif /* PNG_NO_SEQUENTIAL_READ_SUPPORTED */ + +#ifndef PNG_NO_SEQUENTIAL_READ_SUPPORTED +/* Read the end of the PNG file. Will not read past the end of the + * file, will verify the end is accurate, and will read any comments + * or time information at the end of the file, if info is not NULL. + */ +void PNGAPI +png_read_end(png_structp png_ptr, png_infop info_ptr) +{ + png_byte chunk_length[4]; + png_uint_32 length; + + png_debug(1, "in png_read_end\n"); + png_crc_finish(png_ptr, 0); /* Finish off CRC from last IDAT chunk */ + + do + { +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_IHDR; + PNG_IDAT; + PNG_IEND; + PNG_PLTE; +#if defined(PNG_READ_bKGD_SUPPORTED) + PNG_bKGD; +#endif +#if defined(PNG_READ_cHRM_SUPPORTED) + PNG_cHRM; +#endif +#if defined(PNG_READ_gAMA_SUPPORTED) + PNG_gAMA; +#endif +#if defined(PNG_READ_hIST_SUPPORTED) + PNG_hIST; +#endif +#if defined(PNG_READ_iCCP_SUPPORTED) + PNG_iCCP; +#endif +#if defined(PNG_READ_iTXt_SUPPORTED) + PNG_iTXt; +#endif +#if defined(PNG_READ_oFFs_SUPPORTED) + PNG_oFFs; +#endif +#if defined(PNG_READ_pCAL_SUPPORTED) + PNG_pCAL; +#endif +#if defined(PNG_READ_pHYs_SUPPORTED) + PNG_pHYs; +#endif +#if defined(PNG_READ_sBIT_SUPPORTED) + PNG_sBIT; +#endif +#if defined(PNG_READ_sCAL_SUPPORTED) + PNG_sCAL; +#endif +#if defined(PNG_READ_sPLT_SUPPORTED) + PNG_sPLT; +#endif +#if defined(PNG_READ_sRGB_SUPPORTED) + PNG_sRGB; +#endif +#if defined(PNG_READ_tEXt_SUPPORTED) + PNG_tEXt; +#endif +#if defined(PNG_READ_tIME_SUPPORTED) + PNG_tIME; +#endif +#if defined(PNG_READ_tRNS_SUPPORTED) + PNG_tRNS; +#endif +#if defined(PNG_READ_zTXt_SUPPORTED) + PNG_zTXt; +#endif +#endif /* PNG_USE_LOCAL_ARRAYS */ + + png_read_data(png_ptr, chunk_length, 4); + length = png_get_uint_31(png_ptr,chunk_length); + + png_reset_crc(png_ptr); + png_crc_read(png_ptr, png_ptr->chunk_name, 4); + + png_debug1(0, "Reading %s chunk.\n", png_ptr->chunk_name); + + if (!png_memcmp(png_ptr->chunk_name, png_IHDR, 4)) + png_handle_IHDR(png_ptr, info_ptr, length); + else if (!png_memcmp(png_ptr->chunk_name, png_IEND, 4)) + png_handle_IEND(png_ptr, info_ptr, length); +#ifdef PNG_HANDLE_AS_UNKNOWN_SUPPORTED + else if (png_handle_as_unknown(png_ptr, png_ptr->chunk_name)) + { + if (!png_memcmp(png_ptr->chunk_name, png_IDAT, 4)) + { + if (length > 0 || png_ptr->mode & PNG_AFTER_IDAT) + png_error(png_ptr, "Too many IDAT's found"); + } + else + png_ptr->mode |= PNG_AFTER_IDAT; + png_handle_unknown(png_ptr, info_ptr, length); + if (!png_memcmp(png_ptr->chunk_name, png_PLTE, 4)) + png_ptr->mode |= PNG_HAVE_PLTE; + } +#endif + else if (!png_memcmp(png_ptr->chunk_name, png_IDAT, 4)) + { + /* Zero length IDATs are legal after the last IDAT has been + * read, but not after other chunks have been read. + */ + if (length > 0 || png_ptr->mode & PNG_AFTER_IDAT) + png_error(png_ptr, "Too many IDAT's found"); + png_crc_finish(png_ptr, length); + } + else if (!png_memcmp(png_ptr->chunk_name, png_PLTE, 4)) + png_handle_PLTE(png_ptr, info_ptr, length); +#if defined(PNG_READ_bKGD_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_bKGD, 4)) + png_handle_bKGD(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_cHRM_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_cHRM, 4)) + png_handle_cHRM(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_gAMA_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_gAMA, 4)) + png_handle_gAMA(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_hIST_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_hIST, 4)) + png_handle_hIST(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_oFFs_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_oFFs, 4)) + png_handle_oFFs(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_pCAL_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_pCAL, 4)) + png_handle_pCAL(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_sCAL_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_sCAL, 4)) + png_handle_sCAL(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_pHYs_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_pHYs, 4)) + png_handle_pHYs(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_sBIT_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_sBIT, 4)) + png_handle_sBIT(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_sRGB_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_sRGB, 4)) + png_handle_sRGB(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_iCCP_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_iCCP, 4)) + png_handle_iCCP(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_sPLT_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_sPLT, 4)) + png_handle_sPLT(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_tEXt_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_tEXt, 4)) + png_handle_tEXt(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_tIME_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_tIME, 4)) + png_handle_tIME(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_tRNS_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_tRNS, 4)) + png_handle_tRNS(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_zTXt_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_zTXt, 4)) + png_handle_zTXt(png_ptr, info_ptr, length); +#endif +#if defined(PNG_READ_iTXt_SUPPORTED) + else if (!png_memcmp(png_ptr->chunk_name, png_iTXt, 4)) + png_handle_iTXt(png_ptr, info_ptr, length); +#endif + else + png_handle_unknown(png_ptr, info_ptr, length); + } while (!(png_ptr->mode & PNG_HAVE_IEND)); +} +#endif /* PNG_NO_SEQUENTIAL_READ_SUPPORTED */ + +/* free all memory used by the read */ +void PNGAPI +png_destroy_read_struct(png_structpp png_ptr_ptr, png_infopp info_ptr_ptr, + png_infopp end_info_ptr_ptr) +{ + png_structp png_ptr = NULL; + png_infop info_ptr = NULL, end_info_ptr = NULL; +#ifdef PNG_USER_MEM_SUPPORTED + png_free_ptr free_fn; + png_voidp mem_ptr; +#endif + + png_debug(1, "in png_destroy_read_struct\n"); + if (png_ptr_ptr != NULL) + png_ptr = *png_ptr_ptr; + + if (info_ptr_ptr != NULL) + info_ptr = *info_ptr_ptr; + + if (end_info_ptr_ptr != NULL) + end_info_ptr = *end_info_ptr_ptr; + +#ifdef PNG_USER_MEM_SUPPORTED + free_fn = png_ptr->free_fn; + mem_ptr = png_ptr->mem_ptr; +#endif + + png_read_destroy(png_ptr, info_ptr, end_info_ptr); + + if (info_ptr != NULL) + { +#if defined(PNG_TEXT_SUPPORTED) + png_free_data(png_ptr, info_ptr, PNG_FREE_TEXT, -1); +#endif + +#ifdef PNG_USER_MEM_SUPPORTED + png_destroy_struct_2((png_voidp)info_ptr, (png_free_ptr)free_fn, + (png_voidp)mem_ptr); +#else + png_destroy_struct((png_voidp)info_ptr); +#endif + *info_ptr_ptr = NULL; + } + + if (end_info_ptr != NULL) + { +#if defined(PNG_READ_TEXT_SUPPORTED) + png_free_data(png_ptr, end_info_ptr, PNG_FREE_TEXT, -1); +#endif +#ifdef PNG_USER_MEM_SUPPORTED + png_destroy_struct_2((png_voidp)end_info_ptr, (png_free_ptr)free_fn, + (png_voidp)mem_ptr); +#else + png_destroy_struct((png_voidp)end_info_ptr); +#endif + *end_info_ptr_ptr = NULL; + } + + if (png_ptr != NULL) + { +#ifdef PNG_USER_MEM_SUPPORTED + png_destroy_struct_2((png_voidp)png_ptr, (png_free_ptr)free_fn, + (png_voidp)mem_ptr); +#else + png_destroy_struct((png_voidp)png_ptr); +#endif + *png_ptr_ptr = NULL; + } +} + +/* free all memory used by the read (old method) */ +void /* PRIVATE */ +png_read_destroy(png_structp png_ptr, png_infop info_ptr, png_infop end_info_ptr) +{ +#ifdef PNG_SETJMP_SUPPORTED + jmp_buf tmp_jmp; +#endif + png_error_ptr error_fn; + png_error_ptr warning_fn; + png_voidp error_ptr; +#ifdef PNG_USER_MEM_SUPPORTED + png_free_ptr free_fn; +#endif + + png_debug(1, "in png_read_destroy\n"); + if (info_ptr != NULL) + png_info_destroy(png_ptr, info_ptr); + + if (end_info_ptr != NULL) + png_info_destroy(png_ptr, end_info_ptr); + + png_free(png_ptr, png_ptr->zbuf); + png_free(png_ptr, png_ptr->big_row_buf); + png_free(png_ptr, png_ptr->prev_row); +#if defined(PNG_READ_DITHER_SUPPORTED) + png_free(png_ptr, png_ptr->palette_lookup); + png_free(png_ptr, png_ptr->dither_index); +#endif +#if defined(PNG_READ_GAMMA_SUPPORTED) + png_free(png_ptr, png_ptr->gamma_table); +#endif +#if defined(PNG_READ_BACKGROUND_SUPPORTED) + png_free(png_ptr, png_ptr->gamma_from_1); + png_free(png_ptr, png_ptr->gamma_to_1); +#endif +#ifdef PNG_FREE_ME_SUPPORTED + if (png_ptr->free_me & PNG_FREE_PLTE) + png_zfree(png_ptr, png_ptr->palette); + png_ptr->free_me &= ~PNG_FREE_PLTE; +#else + if (png_ptr->flags & PNG_FLAG_FREE_PLTE) + png_zfree(png_ptr, png_ptr->palette); + png_ptr->flags &= ~PNG_FLAG_FREE_PLTE; +#endif +#if defined(PNG_tRNS_SUPPORTED) || \ + defined(PNG_READ_EXPAND_SUPPORTED) || defined(PNG_READ_BACKGROUND_SUPPORTED) +#ifdef PNG_FREE_ME_SUPPORTED + if (png_ptr->free_me & PNG_FREE_TRNS) + png_free(png_ptr, png_ptr->trans); + png_ptr->free_me &= ~PNG_FREE_TRNS; +#else + if (png_ptr->flags & PNG_FLAG_FREE_TRNS) + png_free(png_ptr, png_ptr->trans); + png_ptr->flags &= ~PNG_FLAG_FREE_TRNS; +#endif +#endif +#if defined(PNG_READ_hIST_SUPPORTED) +#ifdef PNG_FREE_ME_SUPPORTED + if (png_ptr->free_me & PNG_FREE_HIST) + png_free(png_ptr, png_ptr->hist); + png_ptr->free_me &= ~PNG_FREE_HIST; +#else + if (png_ptr->flags & PNG_FLAG_FREE_HIST) + png_free(png_ptr, png_ptr->hist); + png_ptr->flags &= ~PNG_FLAG_FREE_HIST; +#endif +#endif +#if defined(PNG_READ_GAMMA_SUPPORTED) + if (png_ptr->gamma_16_table != NULL) + { + int i; + int istop = (1 << (8 - png_ptr->gamma_shift)); + for (i = 0; i < istop; i++) + { + png_free(png_ptr, png_ptr->gamma_16_table[i]); + } + png_free(png_ptr, png_ptr->gamma_16_table); + } +#if defined(PNG_READ_BACKGROUND_SUPPORTED) + if (png_ptr->gamma_16_from_1 != NULL) + { + int i; + int istop = (1 << (8 - png_ptr->gamma_shift)); + for (i = 0; i < istop; i++) + { + png_free(png_ptr, png_ptr->gamma_16_from_1[i]); + } + png_free(png_ptr, png_ptr->gamma_16_from_1); + } + if (png_ptr->gamma_16_to_1 != NULL) + { + int i; + int istop = (1 << (8 - png_ptr->gamma_shift)); + for (i = 0; i < istop; i++) + { + png_free(png_ptr, png_ptr->gamma_16_to_1[i]); + } + png_free(png_ptr, png_ptr->gamma_16_to_1); + } +#endif +#endif +#if defined(PNG_TIME_RFC1123_SUPPORTED) + png_free(png_ptr, png_ptr->time_buffer); +#endif + + inflateEnd(&png_ptr->zstream); +#ifdef PNG_PROGRESSIVE_READ_SUPPORTED + png_free(png_ptr, png_ptr->save_buffer); +#endif + +#ifdef PNG_PROGRESSIVE_READ_SUPPORTED +#ifdef PNG_TEXT_SUPPORTED + png_free(png_ptr, png_ptr->current_text); +#endif /* PNG_TEXT_SUPPORTED */ +#endif /* PNG_PROGRESSIVE_READ_SUPPORTED */ + + /* Save the important info out of the png_struct, in case it is + * being used again. + */ +#ifdef PNG_SETJMP_SUPPORTED + png_memcpy(tmp_jmp, png_ptr->jmpbuf, png_sizeof (jmp_buf)); +#endif + + error_fn = png_ptr->error_fn; + warning_fn = png_ptr->warning_fn; + error_ptr = png_ptr->error_ptr; +#ifdef PNG_USER_MEM_SUPPORTED + free_fn = png_ptr->free_fn; +#endif + + png_memset(png_ptr, 0, png_sizeof (png_struct)); + + png_ptr->error_fn = error_fn; + png_ptr->warning_fn = warning_fn; + png_ptr->error_ptr = error_ptr; +#ifdef PNG_USER_MEM_SUPPORTED + png_ptr->free_fn = free_fn; +#endif + +#ifdef PNG_SETJMP_SUPPORTED + png_memcpy(png_ptr->jmpbuf, tmp_jmp, png_sizeof (jmp_buf)); +#endif + +} + +void PNGAPI +png_set_read_status_fn(png_structp png_ptr, png_read_status_ptr read_row_fn) +{ + png_ptr->read_row_fn = read_row_fn; +} + + +#ifndef PNG_NO_SEQUENTIAL_READ_SUPPORTED +#if defined(PNG_INFO_IMAGE_SUPPORTED) +void PNGAPI +png_read_png(png_structp png_ptr, png_infop info_ptr, + int transforms, + voidp params) +{ + int row; + +#if defined(PNG_READ_INVERT_ALPHA_SUPPORTED) + /* invert the alpha channel from opacity to transparency + */ + if (transforms & PNG_TRANSFORM_INVERT_ALPHA) + png_set_invert_alpha(png_ptr); +#endif + + /* png_read_info() gives us all of the information from the + * PNG file before the first IDAT (image data chunk). + */ + png_read_info(png_ptr, info_ptr); + if (info_ptr->height > PNG_UINT_32_MAX/png_sizeof(png_bytep)) + png_error(png_ptr,"Image is too high to process with png_read_png()"); + + /* -------------- image transformations start here ------------------- */ + +#if defined(PNG_READ_16_TO_8_SUPPORTED) + /* tell libpng to strip 16 bit/color files down to 8 bits per color + */ + if (transforms & PNG_TRANSFORM_STRIP_16) + png_set_strip_16(png_ptr); +#endif + +#if defined(PNG_READ_STRIP_ALPHA_SUPPORTED) + /* Strip alpha bytes from the input data without combining with + * the background (not recommended). + */ + if (transforms & PNG_TRANSFORM_STRIP_ALPHA) + png_set_strip_alpha(png_ptr); +#endif + +#if defined(PNG_READ_PACK_SUPPORTED) && !defined(PNG_READ_EXPAND_SUPPORTED) + /* Extract multiple pixels with bit depths of 1, 2, or 4 from a single + * byte into separate bytes (useful for paletted and grayscale images). + */ + if (transforms & PNG_TRANSFORM_PACKING) + png_set_packing(png_ptr); +#endif + +#if defined(PNG_READ_PACKSWAP_SUPPORTED) + /* Change the order of packed pixels to least significant bit first + * (not useful if you are using png_set_packing). + */ + if (transforms & PNG_TRANSFORM_PACKSWAP) + png_set_packswap(png_ptr); +#endif + +#if defined(PNG_READ_EXPAND_SUPPORTED) + /* Expand paletted colors into true RGB triplets + * Expand grayscale images to full 8 bits from 1, 2, or 4 bits/pixel + * Expand paletted or RGB images with transparency to full alpha + * channels so the data will be available as RGBA quartets. + */ + if (transforms & PNG_TRANSFORM_EXPAND) + if ((png_ptr->bit_depth < 8) || + (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE) || + (png_get_valid(png_ptr, info_ptr, PNG_INFO_tRNS))) + png_set_expand(png_ptr); +#endif + + /* We don't handle background color or gamma transformation or dithering. + */ + +#if defined(PNG_READ_INVERT_SUPPORTED) + /* invert monochrome files to have 0 as white and 1 as black + */ + if (transforms & PNG_TRANSFORM_INVERT_MONO) + png_set_invert_mono(png_ptr); +#endif + +#if defined(PNG_READ_SHIFT_SUPPORTED) + /* If you want to shift the pixel values from the range [0,255] or + * [0,65535] to the original [0,7] or [0,31], or whatever range the + * colors were originally in: + */ + if ((transforms & PNG_TRANSFORM_SHIFT) + && png_get_valid(png_ptr, info_ptr, PNG_INFO_sBIT)) + { + png_color_8p sig_bit; + + png_get_sBIT(png_ptr, info_ptr, &sig_bit); + png_set_shift(png_ptr, sig_bit); + } +#endif + +#if defined(PNG_READ_BGR_SUPPORTED) + /* flip the RGB pixels to BGR (or RGBA to BGRA) + */ + if (transforms & PNG_TRANSFORM_BGR) + png_set_bgr(png_ptr); +#endif + +#if defined(PNG_READ_SWAP_ALPHA_SUPPORTED) + /* swap the RGBA or GA data to ARGB or AG (or BGRA to ABGR) + */ + if (transforms & PNG_TRANSFORM_SWAP_ALPHA) + png_set_swap_alpha(png_ptr); +#endif + +#if defined(PNG_READ_SWAP_SUPPORTED) + /* swap bytes of 16 bit files to least significant byte first + */ + if (transforms & PNG_TRANSFORM_SWAP_ENDIAN) + png_set_swap(png_ptr); +#endif + + /* We don't handle adding filler bytes */ + + /* Optional call to gamma correct and add the background to the palette + * and update info structure. REQUIRED if you are expecting libpng to + * update the palette for you (i.e., you selected such a transform above). + */ + png_read_update_info(png_ptr, info_ptr); + + /* -------------- image transformations end here ------------------- */ + +#ifdef PNG_FREE_ME_SUPPORTED + png_free_data(png_ptr, info_ptr, PNG_FREE_ROWS, 0); +#endif + if(info_ptr->row_pointers == NULL) + { + info_ptr->row_pointers = (png_bytepp)png_malloc(png_ptr, + info_ptr->height * png_sizeof(png_bytep)); +#ifdef PNG_FREE_ME_SUPPORTED + info_ptr->free_me |= PNG_FREE_ROWS; +#endif + for (row = 0; row < (int)info_ptr->height; row++) + { + info_ptr->row_pointers[row] = (png_bytep)png_malloc(png_ptr, + png_get_rowbytes(png_ptr, info_ptr)); + } + } + + png_read_image(png_ptr, info_ptr->row_pointers); + info_ptr->valid |= PNG_INFO_IDAT; + + /* read rest of file, and get additional chunks in info_ptr - REQUIRED */ + png_read_end(png_ptr, info_ptr); + + if(transforms == 0 || params == NULL) + /* quiet compiler warnings */ return; + +} +#endif /* PNG_INFO_IMAGE_SUPPORTED */ +#endif /* PNG_NO_SEQUENTIAL_READ_SUPPORTED */ +#endif /* PNG_READ_SUPPORTED */ diff --git a/src/win32/dependencies/libpng/pngrio.c b/src/win32/dependencies/libpng/pngrio.c new file mode 100644 index 00000000..ce05cada --- /dev/null +++ b/src/win32/dependencies/libpng/pngrio.c @@ -0,0 +1,164 @@ + +/* pngrio.c - functions for data input + * + * Last changed in libpng 1.2.9 April 14, 2006 + * For conditions of distribution and use, see copyright notice in png.h + * Copyright (c) 1998-2006 Glenn Randers-Pehrson + * (Version 0.96 Copyright (c) 1996, 1997 Andreas Dilger) + * (Version 0.88 Copyright (c) 1995, 1996 Guy Eric Schalnat, Group 42, Inc.) + * + * This file provides a location for all input. Users who need + * special handling are expected to write a function that has the same + * arguments as this and performs a similar function, but that possibly + * has a different input method. Note that you shouldn't change this + * function, but rather write a replacement function and then make + * libpng use it at run time with png_set_read_fn(...). + */ + +#define PNG_INTERNAL +#include "png.h" + +#if defined(PNG_READ_SUPPORTED) + +/* Read the data from whatever input you are using. The default routine + reads from a file pointer. Note that this routine sometimes gets called + with very small lengths, so you should implement some kind of simple + buffering if you are using unbuffered reads. This should never be asked + to read more then 64K on a 16 bit machine. */ +void /* PRIVATE */ +png_read_data(png_structp png_ptr, png_bytep data, png_size_t length) +{ + png_debug1(4,"reading %d bytes\n", (int)length); + if (png_ptr->read_data_fn != NULL) + (*(png_ptr->read_data_fn))(png_ptr, data, length); + else + png_error(png_ptr, "Call to NULL read function"); +} + +#if !defined(PNG_NO_STDIO) +/* This is the function that does the actual reading of data. If you are + not reading from a standard C stream, you should create a replacement + read_data function and use it at run time with png_set_read_fn(), rather + than changing the library. */ +#ifndef USE_FAR_KEYWORD +void PNGAPI +png_default_read_data(png_structp png_ptr, png_bytep data, png_size_t length) +{ + png_size_t check; + + /* fread() returns 0 on error, so it is OK to store this in a png_size_t + * instead of an int, which is what fread() actually returns. + */ +#if defined(_WIN32_WCE) + if ( !ReadFile((HANDLE)(png_ptr->io_ptr), data, length, &check, NULL) ) + check = 0; +#else + check = (png_size_t)fread(data, (png_size_t)1, length, + (png_FILE_p)png_ptr->io_ptr); +#endif + + if (check != length) + png_error(png_ptr, "Read Error"); +} +#else +/* this is the model-independent version. Since the standard I/O library + can't handle far buffers in the medium and small models, we have to copy + the data. +*/ + +#define NEAR_BUF_SIZE 1024 +#define MIN(a,b) (a <= b ? a : b) + +static void PNGAPI +png_default_read_data(png_structp png_ptr, png_bytep data, png_size_t length) +{ + int check; + png_byte *n_data; + png_FILE_p io_ptr; + + /* Check if data really is near. If so, use usual code. */ + n_data = (png_byte *)CVT_PTR_NOCHECK(data); + io_ptr = (png_FILE_p)CVT_PTR(png_ptr->io_ptr); + if ((png_bytep)n_data == data) + { +#if defined(_WIN32_WCE) + if ( !ReadFile((HANDLE)(png_ptr->io_ptr), data, length, &check, NULL) ) + check = 0; +#else + check = fread(n_data, 1, length, io_ptr); +#endif + } + else + { + png_byte buf[NEAR_BUF_SIZE]; + png_size_t read, remaining, err; + check = 0; + remaining = length; + do + { + read = MIN(NEAR_BUF_SIZE, remaining); +#if defined(_WIN32_WCE) + if ( !ReadFile((HANDLE)(io_ptr), buf, read, &err, NULL) ) + err = 0; +#else + err = fread(buf, (png_size_t)1, read, io_ptr); +#endif + png_memcpy(data, buf, read); /* copy far buffer to near buffer */ + if(err != read) + break; + else + check += err; + data += read; + remaining -= read; + } + while (remaining != 0); + } + if ((png_uint_32)check != (png_uint_32)length) + png_error(png_ptr, "read Error"); +} +#endif +#endif + +/* This function allows the application to supply a new input function + for libpng if standard C streams aren't being used. + + This function takes as its arguments: + png_ptr - pointer to a png input data structure + io_ptr - pointer to user supplied structure containing info about + the input functions. May be NULL. + read_data_fn - pointer to a new input function that takes as its + arguments a pointer to a png_struct, a pointer to + a location where input data can be stored, and a 32-bit + unsigned int that is the number of bytes to be read. + To exit and output any fatal error messages the new write + function should call png_error(png_ptr, "Error msg"). */ +void PNGAPI +png_set_read_fn(png_structp png_ptr, png_voidp io_ptr, + png_rw_ptr read_data_fn) +{ + png_ptr->io_ptr = io_ptr; + +#if !defined(PNG_NO_STDIO) + if (read_data_fn != NULL) + png_ptr->read_data_fn = read_data_fn; + else + png_ptr->read_data_fn = png_default_read_data; +#else + png_ptr->read_data_fn = read_data_fn; +#endif + + /* It is an error to write to a read device */ + if (png_ptr->write_data_fn != NULL) + { + png_ptr->write_data_fn = NULL; + png_warning(png_ptr, + "It's an error to set both read_data_fn and write_data_fn in the "); + png_warning(png_ptr, + "same structure. Resetting write_data_fn to NULL."); + } + +#if defined(PNG_WRITE_FLUSH_SUPPORTED) + png_ptr->output_flush_fn = NULL; +#endif +} +#endif /* PNG_READ_SUPPORTED */ diff --git a/src/win32/dependencies/libpng/pngrtran.c b/src/win32/dependencies/libpng/pngrtran.c new file mode 100644 index 00000000..8b580807 --- /dev/null +++ b/src/win32/dependencies/libpng/pngrtran.c @@ -0,0 +1,4219 @@ + +/* pngrtran.c - transforms the data in a row for PNG readers + * + * Last changed in libpng 1.2.9 April 14, 2006 + * For conditions of distribution and use, see copyright notice in png.h + * Copyright (c) 1998-2006 Glenn Randers-Pehrson + * (Version 0.96 Copyright (c) 1996, 1997 Andreas Dilger) + * (Version 0.88 Copyright (c) 1995, 1996 Guy Eric Schalnat, Group 42, Inc.) + * + * This file contains functions optionally called by an application + * in order to tell libpng how to handle data when reading a PNG. + * Transformations that are used in both reading and writing are + * in pngtrans.c. + */ + +#define PNG_INTERNAL +#include "png.h" + +#if defined(PNG_READ_SUPPORTED) + +/* Set the action on getting a CRC error for an ancillary or critical chunk. */ +void PNGAPI +png_set_crc_action(png_structp png_ptr, int crit_action, int ancil_action) +{ + png_debug(1, "in png_set_crc_action\n"); + /* Tell libpng how we react to CRC errors in critical chunks */ + switch (crit_action) + { + case PNG_CRC_NO_CHANGE: /* leave setting as is */ + break; + case PNG_CRC_WARN_USE: /* warn/use data */ + png_ptr->flags &= ~PNG_FLAG_CRC_CRITICAL_MASK; + png_ptr->flags |= PNG_FLAG_CRC_CRITICAL_USE; + break; + case PNG_CRC_QUIET_USE: /* quiet/use data */ + png_ptr->flags &= ~PNG_FLAG_CRC_CRITICAL_MASK; + png_ptr->flags |= PNG_FLAG_CRC_CRITICAL_USE | + PNG_FLAG_CRC_CRITICAL_IGNORE; + break; + case PNG_CRC_WARN_DISCARD: /* not a valid action for critical data */ + png_warning(png_ptr, "Can't discard critical data on CRC error."); + case PNG_CRC_ERROR_QUIT: /* error/quit */ + case PNG_CRC_DEFAULT: + default: + png_ptr->flags &= ~PNG_FLAG_CRC_CRITICAL_MASK; + break; + } + + switch (ancil_action) + { + case PNG_CRC_NO_CHANGE: /* leave setting as is */ + break; + case PNG_CRC_WARN_USE: /* warn/use data */ + png_ptr->flags &= ~PNG_FLAG_CRC_ANCILLARY_MASK; + png_ptr->flags |= PNG_FLAG_CRC_ANCILLARY_USE; + break; + case PNG_CRC_QUIET_USE: /* quiet/use data */ + png_ptr->flags &= ~PNG_FLAG_CRC_ANCILLARY_MASK; + png_ptr->flags |= PNG_FLAG_CRC_ANCILLARY_USE | + PNG_FLAG_CRC_ANCILLARY_NOWARN; + break; + case PNG_CRC_ERROR_QUIT: /* error/quit */ + png_ptr->flags &= ~PNG_FLAG_CRC_ANCILLARY_MASK; + png_ptr->flags |= PNG_FLAG_CRC_ANCILLARY_NOWARN; + break; + case PNG_CRC_WARN_DISCARD: /* warn/discard data */ + case PNG_CRC_DEFAULT: + default: + png_ptr->flags &= ~PNG_FLAG_CRC_ANCILLARY_MASK; + break; + } +} + +#if defined(PNG_READ_BACKGROUND_SUPPORTED) && \ + defined(PNG_FLOATING_POINT_SUPPORTED) +/* handle alpha and tRNS via a background color */ +void PNGAPI +png_set_background(png_structp png_ptr, + png_color_16p background_color, int background_gamma_code, + int need_expand, double background_gamma) +{ + png_debug(1, "in png_set_background\n"); + if (background_gamma_code == PNG_BACKGROUND_GAMMA_UNKNOWN) + { + png_warning(png_ptr, "Application must supply a known background gamma"); + return; + } + + png_ptr->transformations |= PNG_BACKGROUND; + png_memcpy(&(png_ptr->background), background_color, + png_sizeof(png_color_16)); + png_ptr->background_gamma = (float)background_gamma; + png_ptr->background_gamma_type = (png_byte)(background_gamma_code); + png_ptr->transformations |= (need_expand ? PNG_BACKGROUND_EXPAND : 0); + + /* Note: if need_expand is set and color_type is either RGB or RGB_ALPHA + * (in which case need_expand is superfluous anyway), the background color + * might actually be gray yet not be flagged as such. This is not a problem + * for the current code, which uses PNG_BACKGROUND_IS_GRAY only to + * decide when to do the png_do_gray_to_rgb() transformation. + */ + if ((need_expand && !(png_ptr->color_type & PNG_COLOR_MASK_COLOR)) || + (!need_expand && background_color->red == background_color->green && + background_color->red == background_color->blue)) + png_ptr->mode |= PNG_BACKGROUND_IS_GRAY; +} +#endif + +#if defined(PNG_READ_16_TO_8_SUPPORTED) +/* strip 16 bit depth files to 8 bit depth */ +void PNGAPI +png_set_strip_16(png_structp png_ptr) +{ + png_debug(1, "in png_set_strip_16\n"); + png_ptr->transformations |= PNG_16_TO_8; +} +#endif + +#if defined(PNG_READ_STRIP_ALPHA_SUPPORTED) +void PNGAPI +png_set_strip_alpha(png_structp png_ptr) +{ + png_debug(1, "in png_set_strip_alpha\n"); + png_ptr->flags |= PNG_FLAG_STRIP_ALPHA; +} +#endif + +#if defined(PNG_READ_DITHER_SUPPORTED) +/* Dither file to 8 bit. Supply a palette, the current number + * of elements in the palette, the maximum number of elements + * allowed, and a histogram if possible. If the current number + * of colors is greater then the maximum number, the palette will be + * modified to fit in the maximum number. "full_dither" indicates + * whether we need a dithering cube set up for RGB images, or if we + * simply are reducing the number of colors in a paletted image. + */ + +typedef struct png_dsort_struct +{ + struct png_dsort_struct FAR * next; + png_byte left; + png_byte right; +} png_dsort; +typedef png_dsort FAR * png_dsortp; +typedef png_dsort FAR * FAR * png_dsortpp; + +void PNGAPI +png_set_dither(png_structp png_ptr, png_colorp palette, + int num_palette, int maximum_colors, png_uint_16p histogram, + int full_dither) +{ + png_debug(1, "in png_set_dither\n"); + png_ptr->transformations |= PNG_DITHER; + + if (!full_dither) + { + int i; + + png_ptr->dither_index = (png_bytep)png_malloc(png_ptr, + (png_uint_32)(num_palette * png_sizeof (png_byte))); + for (i = 0; i < num_palette; i++) + png_ptr->dither_index[i] = (png_byte)i; + } + + if (num_palette > maximum_colors) + { + if (histogram != NULL) + { + /* This is easy enough, just throw out the least used colors. + Perhaps not the best solution, but good enough. */ + + int i; + + /* initialize an array to sort colors */ + png_ptr->dither_sort = (png_bytep)png_malloc(png_ptr, + (png_uint_32)(num_palette * png_sizeof (png_byte))); + + /* initialize the dither_sort array */ + for (i = 0; i < num_palette; i++) + png_ptr->dither_sort[i] = (png_byte)i; + + /* Find the least used palette entries by starting a + bubble sort, and running it until we have sorted + out enough colors. Note that we don't care about + sorting all the colors, just finding which are + least used. */ + + for (i = num_palette - 1; i >= maximum_colors; i--) + { + int done; /* to stop early if the list is pre-sorted */ + int j; + + done = 1; + for (j = 0; j < i; j++) + { + if (histogram[png_ptr->dither_sort[j]] + < histogram[png_ptr->dither_sort[j + 1]]) + { + png_byte t; + + t = png_ptr->dither_sort[j]; + png_ptr->dither_sort[j] = png_ptr->dither_sort[j + 1]; + png_ptr->dither_sort[j + 1] = t; + done = 0; + } + } + if (done) + break; + } + + /* swap the palette around, and set up a table, if necessary */ + if (full_dither) + { + int j = num_palette; + + /* put all the useful colors within the max, but don't + move the others */ + for (i = 0; i < maximum_colors; i++) + { + if ((int)png_ptr->dither_sort[i] >= maximum_colors) + { + do + j--; + while ((int)png_ptr->dither_sort[j] >= maximum_colors); + palette[i] = palette[j]; + } + } + } + else + { + int j = num_palette; + + /* move all the used colors inside the max limit, and + develop a translation table */ + for (i = 0; i < maximum_colors; i++) + { + /* only move the colors we need to */ + if ((int)png_ptr->dither_sort[i] >= maximum_colors) + { + png_color tmp_color; + + do + j--; + while ((int)png_ptr->dither_sort[j] >= maximum_colors); + + tmp_color = palette[j]; + palette[j] = palette[i]; + palette[i] = tmp_color; + /* indicate where the color went */ + png_ptr->dither_index[j] = (png_byte)i; + png_ptr->dither_index[i] = (png_byte)j; + } + } + + /* find closest color for those colors we are not using */ + for (i = 0; i < num_palette; i++) + { + if ((int)png_ptr->dither_index[i] >= maximum_colors) + { + int min_d, k, min_k, d_index; + + /* find the closest color to one we threw out */ + d_index = png_ptr->dither_index[i]; + min_d = PNG_COLOR_DIST(palette[d_index], palette[0]); + for (k = 1, min_k = 0; k < maximum_colors; k++) + { + int d; + + d = PNG_COLOR_DIST(palette[d_index], palette[k]); + + if (d < min_d) + { + min_d = d; + min_k = k; + } + } + /* point to closest color */ + png_ptr->dither_index[i] = (png_byte)min_k; + } + } + } + png_free(png_ptr, png_ptr->dither_sort); + png_ptr->dither_sort=NULL; + } + else + { + /* This is much harder to do simply (and quickly). Perhaps + we need to go through a median cut routine, but those + don't always behave themselves with only a few colors + as input. So we will just find the closest two colors, + and throw out one of them (chosen somewhat randomly). + [We don't understand this at all, so if someone wants to + work on improving it, be our guest - AED, GRP] + */ + int i; + int max_d; + int num_new_palette; + png_dsortp t; + png_dsortpp hash; + + t=NULL; + + /* initialize palette index arrays */ + png_ptr->index_to_palette = (png_bytep)png_malloc(png_ptr, + (png_uint_32)(num_palette * png_sizeof (png_byte))); + png_ptr->palette_to_index = (png_bytep)png_malloc(png_ptr, + (png_uint_32)(num_palette * png_sizeof (png_byte))); + + /* initialize the sort array */ + for (i = 0; i < num_palette; i++) + { + png_ptr->index_to_palette[i] = (png_byte)i; + png_ptr->palette_to_index[i] = (png_byte)i; + } + + hash = (png_dsortpp)png_malloc(png_ptr, (png_uint_32)(769 * + png_sizeof (png_dsortp))); + for (i = 0; i < 769; i++) + hash[i] = NULL; +/* png_memset(hash, 0, 769 * png_sizeof (png_dsortp)); */ + + num_new_palette = num_palette; + + /* initial wild guess at how far apart the farthest pixel + pair we will be eliminating will be. Larger + numbers mean more areas will be allocated, Smaller + numbers run the risk of not saving enough data, and + having to do this all over again. + + I have not done extensive checking on this number. + */ + max_d = 96; + + while (num_new_palette > maximum_colors) + { + for (i = 0; i < num_new_palette - 1; i++) + { + int j; + + for (j = i + 1; j < num_new_palette; j++) + { + int d; + + d = PNG_COLOR_DIST(palette[i], palette[j]); + + if (d <= max_d) + { + + t = (png_dsortp)png_malloc_warn(png_ptr, + (png_uint_32)(png_sizeof(png_dsort))); + if (t == NULL) + break; + t->next = hash[d]; + t->left = (png_byte)i; + t->right = (png_byte)j; + hash[d] = t; + } + } + if (t == NULL) + break; + } + + if (t != NULL) + for (i = 0; i <= max_d; i++) + { + if (hash[i] != NULL) + { + png_dsortp p; + + for (p = hash[i]; p; p = p->next) + { + if ((int)png_ptr->index_to_palette[p->left] + < num_new_palette && + (int)png_ptr->index_to_palette[p->right] + < num_new_palette) + { + int j, next_j; + + if (num_new_palette & 0x01) + { + j = p->left; + next_j = p->right; + } + else + { + j = p->right; + next_j = p->left; + } + + num_new_palette--; + palette[png_ptr->index_to_palette[j]] + = palette[num_new_palette]; + if (!full_dither) + { + int k; + + for (k = 0; k < num_palette; k++) + { + if (png_ptr->dither_index[k] == + png_ptr->index_to_palette[j]) + png_ptr->dither_index[k] = + png_ptr->index_to_palette[next_j]; + if ((int)png_ptr->dither_index[k] == + num_new_palette) + png_ptr->dither_index[k] = + png_ptr->index_to_palette[j]; + } + } + + png_ptr->index_to_palette[png_ptr->palette_to_index + [num_new_palette]] = png_ptr->index_to_palette[j]; + png_ptr->palette_to_index[png_ptr->index_to_palette[j]] + = png_ptr->palette_to_index[num_new_palette]; + + png_ptr->index_to_palette[j] = (png_byte)num_new_palette; + png_ptr->palette_to_index[num_new_palette] = (png_byte)j; + } + if (num_new_palette <= maximum_colors) + break; + } + if (num_new_palette <= maximum_colors) + break; + } + } + + for (i = 0; i < 769; i++) + { + if (hash[i] != NULL) + { + png_dsortp p = hash[i]; + while (p) + { + t = p->next; + png_free(png_ptr, p); + p = t; + } + } + hash[i] = 0; + } + max_d += 96; + } + png_free(png_ptr, hash); + png_free(png_ptr, png_ptr->palette_to_index); + png_free(png_ptr, png_ptr->index_to_palette); + png_ptr->palette_to_index=NULL; + png_ptr->index_to_palette=NULL; + } + num_palette = maximum_colors; + } + if (png_ptr->palette == NULL) + { + png_ptr->palette = palette; + } + png_ptr->num_palette = (png_uint_16)num_palette; + + if (full_dither) + { + int i; + png_bytep distance; + int total_bits = PNG_DITHER_RED_BITS + PNG_DITHER_GREEN_BITS + + PNG_DITHER_BLUE_BITS; + int num_red = (1 << PNG_DITHER_RED_BITS); + int num_green = (1 << PNG_DITHER_GREEN_BITS); + int num_blue = (1 << PNG_DITHER_BLUE_BITS); + png_size_t num_entries = ((png_size_t)1 << total_bits); + + png_ptr->palette_lookup = (png_bytep )png_malloc(png_ptr, + (png_uint_32)(num_entries * png_sizeof (png_byte))); + + png_memset(png_ptr->palette_lookup, 0, num_entries * + png_sizeof (png_byte)); + + distance = (png_bytep)png_malloc(png_ptr, (png_uint_32)(num_entries * + png_sizeof(png_byte))); + + png_memset(distance, 0xff, num_entries * png_sizeof(png_byte)); + + for (i = 0; i < num_palette; i++) + { + int ir, ig, ib; + int r = (palette[i].red >> (8 - PNG_DITHER_RED_BITS)); + int g = (palette[i].green >> (8 - PNG_DITHER_GREEN_BITS)); + int b = (palette[i].blue >> (8 - PNG_DITHER_BLUE_BITS)); + + for (ir = 0; ir < num_red; ir++) + { + /* int dr = abs(ir - r); */ + int dr = ((ir > r) ? ir - r : r - ir); + int index_r = (ir << (PNG_DITHER_BLUE_BITS + PNG_DITHER_GREEN_BITS)); + + for (ig = 0; ig < num_green; ig++) + { + /* int dg = abs(ig - g); */ + int dg = ((ig > g) ? ig - g : g - ig); + int dt = dr + dg; + int dm = ((dr > dg) ? dr : dg); + int index_g = index_r | (ig << PNG_DITHER_BLUE_BITS); + + for (ib = 0; ib < num_blue; ib++) + { + int d_index = index_g | ib; + /* int db = abs(ib - b); */ + int db = ((ib > b) ? ib - b : b - ib); + int dmax = ((dm > db) ? dm : db); + int d = dmax + dt + db; + + if (d < (int)distance[d_index]) + { + distance[d_index] = (png_byte)d; + png_ptr->palette_lookup[d_index] = (png_byte)i; + } + } + } + } + } + + png_free(png_ptr, distance); + } +} +#endif + +#if defined(PNG_READ_GAMMA_SUPPORTED) && defined(PNG_FLOATING_POINT_SUPPORTED) +/* Transform the image from the file_gamma to the screen_gamma. We + * only do transformations on images where the file_gamma and screen_gamma + * are not close reciprocals, otherwise it slows things down slightly, and + * also needlessly introduces small errors. + * + * We will turn off gamma transformation later if no semitransparent entries + * are present in the tRNS array for palette images. We can't do it here + * because we don't necessarily have the tRNS chunk yet. + */ +void PNGAPI +png_set_gamma(png_structp png_ptr, double scrn_gamma, double file_gamma) +{ + png_debug(1, "in png_set_gamma\n"); + if ((fabs(scrn_gamma * file_gamma - 1.0) > PNG_GAMMA_THRESHOLD) || + (png_ptr->color_type & PNG_COLOR_MASK_ALPHA) || + (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE)) + png_ptr->transformations |= PNG_GAMMA; + png_ptr->gamma = (float)file_gamma; + png_ptr->screen_gamma = (float)scrn_gamma; +} +#endif + +#if defined(PNG_READ_EXPAND_SUPPORTED) +/* Expand paletted images to RGB, expand grayscale images of + * less than 8-bit depth to 8-bit depth, and expand tRNS chunks + * to alpha channels. + */ +void PNGAPI +png_set_expand(png_structp png_ptr) +{ + png_debug(1, "in png_set_expand\n"); + png_ptr->transformations |= (PNG_EXPAND | PNG_EXPAND_tRNS); +} + +/* GRR 19990627: the following three functions currently are identical + * to png_set_expand(). However, it is entirely reasonable that someone + * might wish to expand an indexed image to RGB but *not* expand a single, + * fully transparent palette entry to a full alpha channel--perhaps instead + * convert tRNS to the grayscale/RGB format (16-bit RGB value), or replace + * the transparent color with a particular RGB value, or drop tRNS entirely. + * IOW, a future version of the library may make the transformations flag + * a bit more fine-grained, with separate bits for each of these three + * functions. + * + * More to the point, these functions make it obvious what libpng will be + * doing, whereas "expand" can (and does) mean any number of things. + * + * GRP 20060307: In libpng-1.4.0, png_set_gray_1_2_4_to_8() was modified + * to expand only the sample depth but not to expand the tRNS to alpha. + */ + +/* Expand paletted images to RGB. */ +void PNGAPI +png_set_palette_to_rgb(png_structp png_ptr) +{ + png_debug(1, "in png_set_palette_to_rgb\n"); + png_ptr->transformations |= (PNG_EXPAND | PNG_EXPAND_tRNS); +} + +#if !defined(PNG_1_0_X) +/* Expand grayscale images of less than 8-bit depth to 8 bits. */ +void PNGAPI +png_set_expand_gray_1_2_4_to_8(png_structp png_ptr) +{ + png_debug(1, "in png_set_expand_gray_1_2_4_to_8\n"); + png_ptr->transformations |= PNG_EXPAND_tRNS; +} +#endif + +#if defined(PNG_1_0_X) || defined(PNG_1_2_X) +/* Expand grayscale images of less than 8-bit depth to 8 bits. */ +/* Deprecated as of libpng-1.2.9 */ +void PNGAPI +png_set_gray_1_2_4_to_8(png_structp png_ptr) +{ + png_debug(1, "in png_set_gray_1_2_4_to_8\n"); + png_ptr->transformations |= (PNG_EXPAND | PNG_EXPAND_tRNS); +} +#endif + + +/* Expand tRNS chunks to alpha channels. */ +void PNGAPI +png_set_tRNS_to_alpha(png_structp png_ptr) +{ + png_debug(1, "in png_set_expand\n"); + png_ptr->transformations |= (PNG_EXPAND | PNG_EXPAND_tRNS); +} +#endif /* defined(PNG_READ_EXPAND_SUPPORTED) */ + +#if defined(PNG_READ_GRAY_TO_RGB_SUPPORTED) +void PNGAPI +png_set_gray_to_rgb(png_structp png_ptr) +{ + png_debug(1, "in png_set_gray_to_rgb\n"); + png_ptr->transformations |= PNG_GRAY_TO_RGB; +} +#endif + +#if defined(PNG_READ_RGB_TO_GRAY_SUPPORTED) +#if defined(PNG_FLOATING_POINT_SUPPORTED) +/* Convert a RGB image to a grayscale of the same width. This allows us, + * for example, to convert a 24 bpp RGB image into an 8 bpp grayscale image. + */ + +void PNGAPI +png_set_rgb_to_gray(png_structp png_ptr, int error_action, double red, + double green) +{ + int red_fixed = (int)((float)red*100000.0 + 0.5); + int green_fixed = (int)((float)green*100000.0 + 0.5); + png_set_rgb_to_gray_fixed(png_ptr, error_action, red_fixed, green_fixed); +} +#endif + +void PNGAPI +png_set_rgb_to_gray_fixed(png_structp png_ptr, int error_action, + png_fixed_point red, png_fixed_point green) +{ + png_debug(1, "in png_set_rgb_to_gray\n"); + switch(error_action) + { + case 1: png_ptr->transformations |= PNG_RGB_TO_GRAY; + break; + case 2: png_ptr->transformations |= PNG_RGB_TO_GRAY_WARN; + break; + case 3: png_ptr->transformations |= PNG_RGB_TO_GRAY_ERR; + } + if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE) +#if defined(PNG_READ_EXPAND_SUPPORTED) + png_ptr->transformations |= PNG_EXPAND; +#else + { + png_warning(png_ptr, "Cannot do RGB_TO_GRAY without EXPAND_SUPPORTED."); + png_ptr->transformations &= ~PNG_RGB_TO_GRAY; + } +#endif + { + png_uint_16 red_int, green_int; + if(red < 0 || green < 0) + { + red_int = 6968; /* .212671 * 32768 + .5 */ + green_int = 23434; /* .715160 * 32768 + .5 */ + } + else if(red + green < 100000L) + { + red_int = (png_uint_16)(((png_uint_32)red*32768L)/100000L); + green_int = (png_uint_16)(((png_uint_32)green*32768L)/100000L); + } + else + { + png_warning(png_ptr, "ignoring out of range rgb_to_gray coefficients"); + red_int = 6968; + green_int = 23434; + } + png_ptr->rgb_to_gray_red_coeff = red_int; + png_ptr->rgb_to_gray_green_coeff = green_int; + png_ptr->rgb_to_gray_blue_coeff = (png_uint_16)(32768-red_int-green_int); + } +} +#endif + +#if defined(PNG_READ_USER_TRANSFORM_SUPPORTED) || \ + defined(PNG_WRITE_USER_TRANSFORM_SUPPORTED) || \ + defined(PNG_LEGACY_SUPPORTED) +void PNGAPI +png_set_read_user_transform_fn(png_structp png_ptr, png_user_transform_ptr + read_user_transform_fn) +{ + png_debug(1, "in png_set_read_user_transform_fn\n"); +#if defined(PNG_READ_USER_TRANSFORM_SUPPORTED) + png_ptr->transformations |= PNG_USER_TRANSFORM; + png_ptr->read_user_transform_fn = read_user_transform_fn; +#endif +#ifdef PNG_LEGACY_SUPPORTED + if(read_user_transform_fn) + png_warning(png_ptr, + "This version of libpng does not support user transforms"); +#endif +} +#endif + +/* Initialize everything needed for the read. This includes modifying + * the palette. + */ +void /* PRIVATE */ +png_init_read_transformations(png_structp png_ptr) +{ + png_debug(1, "in png_init_read_transformations\n"); +#if defined(PNG_USELESS_TESTS_SUPPORTED) + if(png_ptr != NULL) +#endif + { +#if defined(PNG_READ_BACKGROUND_SUPPORTED) || defined(PNG_READ_SHIFT_SUPPORTED) \ + || defined(PNG_READ_GAMMA_SUPPORTED) + int color_type = png_ptr->color_type; +#endif + +#if defined(PNG_READ_EXPAND_SUPPORTED) && defined(PNG_READ_BACKGROUND_SUPPORTED) + if ((png_ptr->transformations & PNG_BACKGROUND_EXPAND) && + (png_ptr->transformations & PNG_EXPAND)) + { + if (!(color_type & PNG_COLOR_MASK_COLOR)) /* i.e., GRAY or GRAY_ALPHA */ + { + /* expand background and tRNS chunks */ + switch (png_ptr->bit_depth) + { + case 1: + png_ptr->background.gray *= (png_uint_16)0xff; + png_ptr->background.red = png_ptr->background.green + = png_ptr->background.blue = png_ptr->background.gray; + if (!(png_ptr->transformations & PNG_EXPAND_tRNS)) + { + png_ptr->trans_values.gray *= (png_uint_16)0xff; + png_ptr->trans_values.red = png_ptr->trans_values.green + = png_ptr->trans_values.blue = png_ptr->trans_values.gray; + } + break; + case 2: + png_ptr->background.gray *= (png_uint_16)0x55; + png_ptr->background.red = png_ptr->background.green + = png_ptr->background.blue = png_ptr->background.gray; + if (!(png_ptr->transformations & PNG_EXPAND_tRNS)) + { + png_ptr->trans_values.gray *= (png_uint_16)0x55; + png_ptr->trans_values.red = png_ptr->trans_values.green + = png_ptr->trans_values.blue = png_ptr->trans_values.gray; + } + break; + case 4: + png_ptr->background.gray *= (png_uint_16)0x11; + png_ptr->background.red = png_ptr->background.green + = png_ptr->background.blue = png_ptr->background.gray; + if (!(png_ptr->transformations & PNG_EXPAND_tRNS)) + { + png_ptr->trans_values.gray *= (png_uint_16)0x11; + png_ptr->trans_values.red = png_ptr->trans_values.green + = png_ptr->trans_values.blue = png_ptr->trans_values.gray; + } + break; + case 8: + case 16: + png_ptr->background.red = png_ptr->background.green + = png_ptr->background.blue = png_ptr->background.gray; + break; + } + } + else if (color_type == PNG_COLOR_TYPE_PALETTE) + { + png_ptr->background.red = + png_ptr->palette[png_ptr->background.index].red; + png_ptr->background.green = + png_ptr->palette[png_ptr->background.index].green; + png_ptr->background.blue = + png_ptr->palette[png_ptr->background.index].blue; + +#if defined(PNG_READ_INVERT_ALPHA_SUPPORTED) + if (png_ptr->transformations & PNG_INVERT_ALPHA) + { +#if defined(PNG_READ_EXPAND_SUPPORTED) + if (!(png_ptr->transformations & PNG_EXPAND_tRNS)) +#endif + { + /* invert the alpha channel (in tRNS) unless the pixels are + going to be expanded, in which case leave it for later */ + int i,istop; + istop=(int)png_ptr->num_trans; + for (i=0; itrans[i] = (png_byte)(255 - png_ptr->trans[i]); + } + } +#endif + + } + } +#endif + +#if defined(PNG_READ_BACKGROUND_SUPPORTED) && defined(PNG_READ_GAMMA_SUPPORTED) + png_ptr->background_1 = png_ptr->background; +#endif +#if defined(PNG_READ_GAMMA_SUPPORTED) && defined(PNG_FLOATING_POINT_SUPPORTED) + + if ((color_type == PNG_COLOR_TYPE_PALETTE && png_ptr->num_trans != 0) + && (fabs(png_ptr->screen_gamma * png_ptr->gamma - 1.0) + < PNG_GAMMA_THRESHOLD)) + { + int i,k; + k=0; + for (i=0; inum_trans; i++) + { + if (png_ptr->trans[i] != 0 && png_ptr->trans[i] != 0xff) + k=1; /* partial transparency is present */ + } + if (k == 0) + png_ptr->transformations &= (~PNG_GAMMA); + } + + if ((png_ptr->transformations & (PNG_GAMMA | PNG_RGB_TO_GRAY)) && + png_ptr->gamma != 0.0) + { + png_build_gamma_table(png_ptr); +#if defined(PNG_READ_BACKGROUND_SUPPORTED) + if (png_ptr->transformations & PNG_BACKGROUND) + { + if (color_type == PNG_COLOR_TYPE_PALETTE) + { + /* could skip if no transparency and + */ + png_color back, back_1; + png_colorp palette = png_ptr->palette; + int num_palette = png_ptr->num_palette; + int i; + if (png_ptr->background_gamma_type == PNG_BACKGROUND_GAMMA_FILE) + { + back.red = png_ptr->gamma_table[png_ptr->background.red]; + back.green = png_ptr->gamma_table[png_ptr->background.green]; + back.blue = png_ptr->gamma_table[png_ptr->background.blue]; + + back_1.red = png_ptr->gamma_to_1[png_ptr->background.red]; + back_1.green = png_ptr->gamma_to_1[png_ptr->background.green]; + back_1.blue = png_ptr->gamma_to_1[png_ptr->background.blue]; + } + else + { + double g, gs; + + switch (png_ptr->background_gamma_type) + { + case PNG_BACKGROUND_GAMMA_SCREEN: + g = (png_ptr->screen_gamma); + gs = 1.0; + break; + case PNG_BACKGROUND_GAMMA_FILE: + g = 1.0 / (png_ptr->gamma); + gs = 1.0 / (png_ptr->gamma * png_ptr->screen_gamma); + break; + case PNG_BACKGROUND_GAMMA_UNIQUE: + g = 1.0 / (png_ptr->background_gamma); + gs = 1.0 / (png_ptr->background_gamma * + png_ptr->screen_gamma); + break; + default: + g = 1.0; /* back_1 */ + gs = 1.0; /* back */ + } + + if ( fabs(gs - 1.0) < PNG_GAMMA_THRESHOLD) + { + back.red = (png_byte)png_ptr->background.red; + back.green = (png_byte)png_ptr->background.green; + back.blue = (png_byte)png_ptr->background.blue; + } + else + { + back.red = (png_byte)(pow( + (double)png_ptr->background.red/255, gs) * 255.0 + .5); + back.green = (png_byte)(pow( + (double)png_ptr->background.green/255, gs) * 255.0 + .5); + back.blue = (png_byte)(pow( + (double)png_ptr->background.blue/255, gs) * 255.0 + .5); + } + + back_1.red = (png_byte)(pow( + (double)png_ptr->background.red/255, g) * 255.0 + .5); + back_1.green = (png_byte)(pow( + (double)png_ptr->background.green/255, g) * 255.0 + .5); + back_1.blue = (png_byte)(pow( + (double)png_ptr->background.blue/255, g) * 255.0 + .5); + } + for (i = 0; i < num_palette; i++) + { + if (i < (int)png_ptr->num_trans && png_ptr->trans[i] != 0xff) + { + if (png_ptr->trans[i] == 0) + { + palette[i] = back; + } + else /* if (png_ptr->trans[i] != 0xff) */ + { + png_byte v, w; + + v = png_ptr->gamma_to_1[palette[i].red]; + png_composite(w, v, png_ptr->trans[i], back_1.red); + palette[i].red = png_ptr->gamma_from_1[w]; + + v = png_ptr->gamma_to_1[palette[i].green]; + png_composite(w, v, png_ptr->trans[i], back_1.green); + palette[i].green = png_ptr->gamma_from_1[w]; + + v = png_ptr->gamma_to_1[palette[i].blue]; + png_composite(w, v, png_ptr->trans[i], back_1.blue); + palette[i].blue = png_ptr->gamma_from_1[w]; + } + } + else + { + palette[i].red = png_ptr->gamma_table[palette[i].red]; + palette[i].green = png_ptr->gamma_table[palette[i].green]; + palette[i].blue = png_ptr->gamma_table[palette[i].blue]; + } + } + } + /* if (png_ptr->background_gamma_type!=PNG_BACKGROUND_GAMMA_UNKNOWN) */ + else + /* color_type != PNG_COLOR_TYPE_PALETTE */ + { + double m = (double)(((png_uint_32)1 << png_ptr->bit_depth) - 1); + double g = 1.0; + double gs = 1.0; + + switch (png_ptr->background_gamma_type) + { + case PNG_BACKGROUND_GAMMA_SCREEN: + g = (png_ptr->screen_gamma); + gs = 1.0; + break; + case PNG_BACKGROUND_GAMMA_FILE: + g = 1.0 / (png_ptr->gamma); + gs = 1.0 / (png_ptr->gamma * png_ptr->screen_gamma); + break; + case PNG_BACKGROUND_GAMMA_UNIQUE: + g = 1.0 / (png_ptr->background_gamma); + gs = 1.0 / (png_ptr->background_gamma * + png_ptr->screen_gamma); + break; + } + + png_ptr->background_1.gray = (png_uint_16)(pow( + (double)png_ptr->background.gray / m, g) * m + .5); + png_ptr->background.gray = (png_uint_16)(pow( + (double)png_ptr->background.gray / m, gs) * m + .5); + + if ((png_ptr->background.red != png_ptr->background.green) || + (png_ptr->background.red != png_ptr->background.blue) || + (png_ptr->background.red != png_ptr->background.gray)) + { + /* RGB or RGBA with color background */ + png_ptr->background_1.red = (png_uint_16)(pow( + (double)png_ptr->background.red / m, g) * m + .5); + png_ptr->background_1.green = (png_uint_16)(pow( + (double)png_ptr->background.green / m, g) * m + .5); + png_ptr->background_1.blue = (png_uint_16)(pow( + (double)png_ptr->background.blue / m, g) * m + .5); + png_ptr->background.red = (png_uint_16)(pow( + (double)png_ptr->background.red / m, gs) * m + .5); + png_ptr->background.green = (png_uint_16)(pow( + (double)png_ptr->background.green / m, gs) * m + .5); + png_ptr->background.blue = (png_uint_16)(pow( + (double)png_ptr->background.blue / m, gs) * m + .5); + } + else + { + /* GRAY, GRAY ALPHA, RGB, or RGBA with gray background */ + png_ptr->background_1.red = png_ptr->background_1.green + = png_ptr->background_1.blue = png_ptr->background_1.gray; + png_ptr->background.red = png_ptr->background.green + = png_ptr->background.blue = png_ptr->background.gray; + } + } + } + else + /* transformation does not include PNG_BACKGROUND */ +#endif /* PNG_READ_BACKGROUND_SUPPORTED */ + if (color_type == PNG_COLOR_TYPE_PALETTE) + { + png_colorp palette = png_ptr->palette; + int num_palette = png_ptr->num_palette; + int i; + + for (i = 0; i < num_palette; i++) + { + palette[i].red = png_ptr->gamma_table[palette[i].red]; + palette[i].green = png_ptr->gamma_table[palette[i].green]; + palette[i].blue = png_ptr->gamma_table[palette[i].blue]; + } + } + } +#if defined(PNG_READ_BACKGROUND_SUPPORTED) + else +#endif +#endif /* PNG_READ_GAMMA_SUPPORTED && PNG_FLOATING_POINT_SUPPORTED */ +#if defined(PNG_READ_BACKGROUND_SUPPORTED) + /* No GAMMA transformation */ + if ((png_ptr->transformations & PNG_BACKGROUND) && + (color_type == PNG_COLOR_TYPE_PALETTE)) + { + int i; + int istop = (int)png_ptr->num_trans; + png_color back; + png_colorp palette = png_ptr->palette; + + back.red = (png_byte)png_ptr->background.red; + back.green = (png_byte)png_ptr->background.green; + back.blue = (png_byte)png_ptr->background.blue; + + for (i = 0; i < istop; i++) + { + if (png_ptr->trans[i] == 0) + { + palette[i] = back; + } + else if (png_ptr->trans[i] != 0xff) + { + /* The png_composite() macro is defined in png.h */ + png_composite(palette[i].red, palette[i].red, + png_ptr->trans[i], back.red); + png_composite(palette[i].green, palette[i].green, + png_ptr->trans[i], back.green); + png_composite(palette[i].blue, palette[i].blue, + png_ptr->trans[i], back.blue); + } + } + } +#endif /* PNG_READ_BACKGROUND_SUPPORTED */ + +#if defined(PNG_READ_SHIFT_SUPPORTED) + if ((png_ptr->transformations & PNG_SHIFT) && + (color_type == PNG_COLOR_TYPE_PALETTE)) + { + png_uint_16 i; + png_uint_16 istop = png_ptr->num_palette; + int sr = 8 - png_ptr->sig_bit.red; + int sg = 8 - png_ptr->sig_bit.green; + int sb = 8 - png_ptr->sig_bit.blue; + + if (sr < 0 || sr > 8) + sr = 0; + if (sg < 0 || sg > 8) + sg = 0; + if (sb < 0 || sb > 8) + sb = 0; + for (i = 0; i < istop; i++) + { + png_ptr->palette[i].red >>= sr; + png_ptr->palette[i].green >>= sg; + png_ptr->palette[i].blue >>= sb; + } + } +#endif /* PNG_READ_SHIFT_SUPPORTED */ + } +#if !defined(PNG_READ_GAMMA_SUPPORTED) && !defined(PNG_READ_SHIFT_SUPPORTED) \ + && !defined(PNG_READ_BACKGROUND_SUPPORTED) + if(png_ptr) + return; +#endif +} + +/* Modify the info structure to reflect the transformations. The + * info should be updated so a PNG file could be written with it, + * assuming the transformations result in valid PNG data. + */ +void /* PRIVATE */ +png_read_transform_info(png_structp png_ptr, png_infop info_ptr) +{ + png_debug(1, "in png_read_transform_info\n"); +#if defined(PNG_READ_EXPAND_SUPPORTED) + if (png_ptr->transformations & PNG_EXPAND) + { + if (info_ptr->color_type == PNG_COLOR_TYPE_PALETTE) + { + if (png_ptr->num_trans && (png_ptr->transformations & PNG_EXPAND_tRNS)) + info_ptr->color_type = PNG_COLOR_TYPE_RGB_ALPHA; + else + info_ptr->color_type = PNG_COLOR_TYPE_RGB; + info_ptr->bit_depth = 8; + info_ptr->num_trans = 0; + } + else + { + if (png_ptr->num_trans) + { + if (png_ptr->transformations & PNG_EXPAND_tRNS) + info_ptr->color_type |= PNG_COLOR_MASK_ALPHA; + else + info_ptr->color_type |= PNG_COLOR_MASK_COLOR; + } + if (info_ptr->bit_depth < 8) + info_ptr->bit_depth = 8; + info_ptr->num_trans = 0; + } + } +#endif + +#if defined(PNG_READ_BACKGROUND_SUPPORTED) + if (png_ptr->transformations & PNG_BACKGROUND) + { + info_ptr->color_type &= ~PNG_COLOR_MASK_ALPHA; + info_ptr->num_trans = 0; + info_ptr->background = png_ptr->background; + } +#endif + +#if defined(PNG_READ_GAMMA_SUPPORTED) + if (png_ptr->transformations & PNG_GAMMA) + { +#ifdef PNG_FLOATING_POINT_SUPPORTED + info_ptr->gamma = png_ptr->gamma; +#endif +#ifdef PNG_FIXED_POINT_SUPPORTED + info_ptr->int_gamma = png_ptr->int_gamma; +#endif + } +#endif + +#if defined(PNG_READ_16_TO_8_SUPPORTED) + if ((png_ptr->transformations & PNG_16_TO_8) && (info_ptr->bit_depth == 16)) + info_ptr->bit_depth = 8; +#endif + +#if defined(PNG_READ_GRAY_TO_RGB_SUPPORTED) + if (png_ptr->transformations & PNG_GRAY_TO_RGB) + info_ptr->color_type |= PNG_COLOR_MASK_COLOR; +#endif + +#if defined(PNG_READ_RGB_TO_GRAY_SUPPORTED) + if (png_ptr->transformations & PNG_RGB_TO_GRAY) + info_ptr->color_type &= ~PNG_COLOR_MASK_COLOR; +#endif + +#if defined(PNG_READ_DITHER_SUPPORTED) + if (png_ptr->transformations & PNG_DITHER) + { + if (((info_ptr->color_type == PNG_COLOR_TYPE_RGB) || + (info_ptr->color_type == PNG_COLOR_TYPE_RGB_ALPHA)) && + png_ptr->palette_lookup && info_ptr->bit_depth == 8) + { + info_ptr->color_type = PNG_COLOR_TYPE_PALETTE; + } + } +#endif + +#if defined(PNG_READ_PACK_SUPPORTED) + if ((png_ptr->transformations & PNG_PACK) && (info_ptr->bit_depth < 8)) + info_ptr->bit_depth = 8; +#endif + + if (info_ptr->color_type == PNG_COLOR_TYPE_PALETTE) + info_ptr->channels = 1; + else if (info_ptr->color_type & PNG_COLOR_MASK_COLOR) + info_ptr->channels = 3; + else + info_ptr->channels = 1; + +#if defined(PNG_READ_STRIP_ALPHA_SUPPORTED) + if (png_ptr->flags & PNG_FLAG_STRIP_ALPHA) + info_ptr->color_type &= ~PNG_COLOR_MASK_ALPHA; +#endif + + if (info_ptr->color_type & PNG_COLOR_MASK_ALPHA) + info_ptr->channels++; + +#if defined(PNG_READ_FILLER_SUPPORTED) + /* STRIP_ALPHA and FILLER allowed: MASK_ALPHA bit stripped above */ + if ((png_ptr->transformations & PNG_FILLER) && + ((info_ptr->color_type == PNG_COLOR_TYPE_RGB) || + (info_ptr->color_type == PNG_COLOR_TYPE_GRAY))) + { + info_ptr->channels++; + /* if adding a true alpha channel not just filler */ +#if !defined(PNG_1_0_X) + if (png_ptr->transformations & PNG_ADD_ALPHA) + info_ptr->color_type |= PNG_COLOR_MASK_ALPHA; +#endif + } +#endif + +#if defined(PNG_USER_TRANSFORM_PTR_SUPPORTED) && \ +defined(PNG_READ_USER_TRANSFORM_SUPPORTED) + if(png_ptr->transformations & PNG_USER_TRANSFORM) + { + if(info_ptr->bit_depth < png_ptr->user_transform_depth) + info_ptr->bit_depth = png_ptr->user_transform_depth; + if(info_ptr->channels < png_ptr->user_transform_channels) + info_ptr->channels = png_ptr->user_transform_channels; + } +#endif + + info_ptr->pixel_depth = (png_byte)(info_ptr->channels * + info_ptr->bit_depth); + + info_ptr->rowbytes = PNG_ROWBYTES(info_ptr->pixel_depth,info_ptr->width); + +#if !defined(PNG_READ_EXPAND_SUPPORTED) + if(png_ptr) + return; +#endif +} + +/* Transform the row. The order of transformations is significant, + * and is very touchy. If you add a transformation, take care to + * decide how it fits in with the other transformations here. + */ +void /* PRIVATE */ +png_do_read_transformations(png_structp png_ptr) +{ + png_debug(1, "in png_do_read_transformations\n"); +#if !defined(PNG_USELESS_TESTS_SUPPORTED) + if (png_ptr->row_buf == NULL) + { +#if !defined(PNG_NO_STDIO) && !defined(_WIN32_WCE) + char msg[50]; + + sprintf(msg, "NULL row buffer for row %ld, pass %d", png_ptr->row_number, + png_ptr->pass); + png_error(png_ptr, msg); +#else + png_error(png_ptr, "NULL row buffer"); +#endif + } +#endif + +#if defined(PNG_READ_EXPAND_SUPPORTED) + if (png_ptr->transformations & PNG_EXPAND) + { + if (png_ptr->row_info.color_type == PNG_COLOR_TYPE_PALETTE) + { + png_do_expand_palette(&(png_ptr->row_info), png_ptr->row_buf + 1, + png_ptr->palette, png_ptr->trans, png_ptr->num_trans); + } + else + { + if (png_ptr->num_trans && (png_ptr->transformations & PNG_EXPAND_tRNS)) + png_do_expand(&(png_ptr->row_info), png_ptr->row_buf + 1, + &(png_ptr->trans_values)); + else + png_do_expand(&(png_ptr->row_info), png_ptr->row_buf + 1, + NULL); + } + } +#endif + +#if defined(PNG_READ_STRIP_ALPHA_SUPPORTED) + if (png_ptr->flags & PNG_FLAG_STRIP_ALPHA) + png_do_strip_filler(&(png_ptr->row_info), png_ptr->row_buf + 1, + PNG_FLAG_FILLER_AFTER | (png_ptr->flags & PNG_FLAG_STRIP_ALPHA)); +#endif + +#if defined(PNG_READ_RGB_TO_GRAY_SUPPORTED) + if (png_ptr->transformations & PNG_RGB_TO_GRAY) + { + int rgb_error = + png_do_rgb_to_gray(png_ptr, &(png_ptr->row_info), png_ptr->row_buf + 1); + if(rgb_error) + { + png_ptr->rgb_to_gray_status=1; + if(png_ptr->transformations == PNG_RGB_TO_GRAY_WARN) + png_warning(png_ptr, "png_do_rgb_to_gray found nongray pixel"); + if(png_ptr->transformations == PNG_RGB_TO_GRAY_ERR) + png_error(png_ptr, "png_do_rgb_to_gray found nongray pixel"); + } + } +#endif + +/* +From Andreas Dilger e-mail to png-implement, 26 March 1998: + + In most cases, the "simple transparency" should be done prior to doing + gray-to-RGB, or you will have to test 3x as many bytes to check if a + pixel is transparent. You would also need to make sure that the + transparency information is upgraded to RGB. + + To summarize, the current flow is: + - Gray + simple transparency -> compare 1 or 2 gray bytes and composite + with background "in place" if transparent, + convert to RGB if necessary + - Gray + alpha -> composite with gray background and remove alpha bytes, + convert to RGB if necessary + + To support RGB backgrounds for gray images we need: + - Gray + simple transparency -> convert to RGB + simple transparency, compare + 3 or 6 bytes and composite with background + "in place" if transparent (3x compare/pixel + compared to doing composite with gray bkgrnd) + - Gray + alpha -> convert to RGB + alpha, composite with background and + remove alpha bytes (3x float operations/pixel + compared with composite on gray background) + + Greg's change will do this. The reason it wasn't done before is for + performance, as this increases the per-pixel operations. If we would check + in advance if the background was gray or RGB, and position the gray-to-RGB + transform appropriately, then it would save a lot of work/time. + */ + +#if defined(PNG_READ_GRAY_TO_RGB_SUPPORTED) + /* if gray -> RGB, do so now only if background is non-gray; else do later + * for performance reasons */ + if ((png_ptr->transformations & PNG_GRAY_TO_RGB) && + !(png_ptr->mode & PNG_BACKGROUND_IS_GRAY)) + png_do_gray_to_rgb(&(png_ptr->row_info), png_ptr->row_buf + 1); +#endif + +#if defined(PNG_READ_BACKGROUND_SUPPORTED) + if ((png_ptr->transformations & PNG_BACKGROUND) && + ((png_ptr->num_trans != 0 ) || + (png_ptr->color_type & PNG_COLOR_MASK_ALPHA))) + png_do_background(&(png_ptr->row_info), png_ptr->row_buf + 1, + &(png_ptr->trans_values), &(png_ptr->background) +#if defined(PNG_READ_GAMMA_SUPPORTED) + , &(png_ptr->background_1), + png_ptr->gamma_table, png_ptr->gamma_from_1, + png_ptr->gamma_to_1, png_ptr->gamma_16_table, + png_ptr->gamma_16_from_1, png_ptr->gamma_16_to_1, + png_ptr->gamma_shift +#endif +); +#endif + +#if defined(PNG_READ_GAMMA_SUPPORTED) + if ((png_ptr->transformations & PNG_GAMMA) && +#if defined(PNG_READ_BACKGROUND_SUPPORTED) + !((png_ptr->transformations & PNG_BACKGROUND) && + ((png_ptr->num_trans != 0) || + (png_ptr->color_type & PNG_COLOR_MASK_ALPHA))) && +#endif + (png_ptr->color_type != PNG_COLOR_TYPE_PALETTE)) + png_do_gamma(&(png_ptr->row_info), png_ptr->row_buf + 1, + png_ptr->gamma_table, png_ptr->gamma_16_table, + png_ptr->gamma_shift); +#endif + +#if defined(PNG_READ_16_TO_8_SUPPORTED) + if (png_ptr->transformations & PNG_16_TO_8) + png_do_chop(&(png_ptr->row_info), png_ptr->row_buf + 1); +#endif + +#if defined(PNG_READ_DITHER_SUPPORTED) + if (png_ptr->transformations & PNG_DITHER) + { + png_do_dither((png_row_infop)&(png_ptr->row_info), png_ptr->row_buf + 1, + png_ptr->palette_lookup, png_ptr->dither_index); + if(png_ptr->row_info.rowbytes == (png_uint_32)0) + png_error(png_ptr, "png_do_dither returned rowbytes=0"); + } +#endif + +#if defined(PNG_READ_INVERT_SUPPORTED) + if (png_ptr->transformations & PNG_INVERT_MONO) + png_do_invert(&(png_ptr->row_info), png_ptr->row_buf + 1); +#endif + +#if defined(PNG_READ_SHIFT_SUPPORTED) + if (png_ptr->transformations & PNG_SHIFT) + png_do_unshift(&(png_ptr->row_info), png_ptr->row_buf + 1, + &(png_ptr->shift)); +#endif + +#if defined(PNG_READ_PACK_SUPPORTED) + if (png_ptr->transformations & PNG_PACK) + png_do_unpack(&(png_ptr->row_info), png_ptr->row_buf + 1); +#endif + +#if defined(PNG_READ_BGR_SUPPORTED) + if (png_ptr->transformations & PNG_BGR) + png_do_bgr(&(png_ptr->row_info), png_ptr->row_buf + 1); +#endif + +#if defined(PNG_READ_PACKSWAP_SUPPORTED) + if (png_ptr->transformations & PNG_PACKSWAP) + png_do_packswap(&(png_ptr->row_info), png_ptr->row_buf + 1); +#endif + +#if defined(PNG_READ_GRAY_TO_RGB_SUPPORTED) + /* if gray -> RGB, do so now only if we did not do so above */ + if ((png_ptr->transformations & PNG_GRAY_TO_RGB) && + (png_ptr->mode & PNG_BACKGROUND_IS_GRAY)) + png_do_gray_to_rgb(&(png_ptr->row_info), png_ptr->row_buf + 1); +#endif + +#if defined(PNG_READ_FILLER_SUPPORTED) + if (png_ptr->transformations & PNG_FILLER) + png_do_read_filler(&(png_ptr->row_info), png_ptr->row_buf + 1, + (png_uint_32)png_ptr->filler, png_ptr->flags); +#endif + +#if defined(PNG_READ_INVERT_ALPHA_SUPPORTED) + if (png_ptr->transformations & PNG_INVERT_ALPHA) + png_do_read_invert_alpha(&(png_ptr->row_info), png_ptr->row_buf + 1); +#endif + +#if defined(PNG_READ_SWAP_ALPHA_SUPPORTED) + if (png_ptr->transformations & PNG_SWAP_ALPHA) + png_do_read_swap_alpha(&(png_ptr->row_info), png_ptr->row_buf + 1); +#endif + +#if defined(PNG_READ_SWAP_SUPPORTED) + if (png_ptr->transformations & PNG_SWAP_BYTES) + png_do_swap(&(png_ptr->row_info), png_ptr->row_buf + 1); +#endif + +#if defined(PNG_READ_USER_TRANSFORM_SUPPORTED) + if (png_ptr->transformations & PNG_USER_TRANSFORM) + { + if(png_ptr->read_user_transform_fn != NULL) + (*(png_ptr->read_user_transform_fn)) /* user read transform function */ + (png_ptr, /* png_ptr */ + &(png_ptr->row_info), /* row_info: */ + /* png_uint_32 width; width of row */ + /* png_uint_32 rowbytes; number of bytes in row */ + /* png_byte color_type; color type of pixels */ + /* png_byte bit_depth; bit depth of samples */ + /* png_byte channels; number of channels (1-4) */ + /* png_byte pixel_depth; bits per pixel (depth*channels) */ + png_ptr->row_buf + 1); /* start of pixel data for row */ +#if defined(PNG_USER_TRANSFORM_PTR_SUPPORTED) + if(png_ptr->user_transform_depth) + png_ptr->row_info.bit_depth = png_ptr->user_transform_depth; + if(png_ptr->user_transform_channels) + png_ptr->row_info.channels = png_ptr->user_transform_channels; +#endif + png_ptr->row_info.pixel_depth = (png_byte)(png_ptr->row_info.bit_depth * + png_ptr->row_info.channels); + png_ptr->row_info.rowbytes = PNG_ROWBYTES(png_ptr->row_info.pixel_depth, + png_ptr->row_info.width); + } +#endif + +} + +#if defined(PNG_READ_PACK_SUPPORTED) +/* Unpack pixels of 1, 2, or 4 bits per pixel into 1 byte per pixel, + * without changing the actual values. Thus, if you had a row with + * a bit depth of 1, you would end up with bytes that only contained + * the numbers 0 or 1. If you would rather they contain 0 and 255, use + * png_do_shift() after this. + */ +void /* PRIVATE */ +png_do_unpack(png_row_infop row_info, png_bytep row) +{ + png_debug(1, "in png_do_unpack\n"); +#if defined(PNG_USELESS_TESTS_SUPPORTED) + if (row != NULL && row_info != NULL && row_info->bit_depth < 8) +#else + if (row_info->bit_depth < 8) +#endif + { + png_uint_32 i; + png_uint_32 row_width=row_info->width; + + switch (row_info->bit_depth) + { + case 1: + { + png_bytep sp = row + (png_size_t)((row_width - 1) >> 3); + png_bytep dp = row + (png_size_t)row_width - 1; + png_uint_32 shift = 7 - (int)((row_width + 7) & 0x07); + for (i = 0; i < row_width; i++) + { + *dp = (png_byte)((*sp >> shift) & 0x01); + if (shift == 7) + { + shift = 0; + sp--; + } + else + shift++; + + dp--; + } + break; + } + case 2: + { + + png_bytep sp = row + (png_size_t)((row_width - 1) >> 2); + png_bytep dp = row + (png_size_t)row_width - 1; + png_uint_32 shift = (int)((3 - ((row_width + 3) & 0x03)) << 1); + for (i = 0; i < row_width; i++) + { + *dp = (png_byte)((*sp >> shift) & 0x03); + if (shift == 6) + { + shift = 0; + sp--; + } + else + shift += 2; + + dp--; + } + break; + } + case 4: + { + png_bytep sp = row + (png_size_t)((row_width - 1) >> 1); + png_bytep dp = row + (png_size_t)row_width - 1; + png_uint_32 shift = (int)((1 - ((row_width + 1) & 0x01)) << 2); + for (i = 0; i < row_width; i++) + { + *dp = (png_byte)((*sp >> shift) & 0x0f); + if (shift == 4) + { + shift = 0; + sp--; + } + else + shift = 4; + + dp--; + } + break; + } + } + row_info->bit_depth = 8; + row_info->pixel_depth = (png_byte)(8 * row_info->channels); + row_info->rowbytes = row_width * row_info->channels; + } +} +#endif + +#if defined(PNG_READ_SHIFT_SUPPORTED) +/* Reverse the effects of png_do_shift. This routine merely shifts the + * pixels back to their significant bits values. Thus, if you have + * a row of bit depth 8, but only 5 are significant, this will shift + * the values back to 0 through 31. + */ +void /* PRIVATE */ +png_do_unshift(png_row_infop row_info, png_bytep row, png_color_8p sig_bits) +{ + png_debug(1, "in png_do_unshift\n"); + if ( +#if defined(PNG_USELESS_TESTS_SUPPORTED) + row != NULL && row_info != NULL && sig_bits != NULL && +#endif + row_info->color_type != PNG_COLOR_TYPE_PALETTE) + { + int shift[4]; + int channels = 0; + int c; + png_uint_16 value = 0; + png_uint_32 row_width = row_info->width; + + if (row_info->color_type & PNG_COLOR_MASK_COLOR) + { + shift[channels++] = row_info->bit_depth - sig_bits->red; + shift[channels++] = row_info->bit_depth - sig_bits->green; + shift[channels++] = row_info->bit_depth - sig_bits->blue; + } + else + { + shift[channels++] = row_info->bit_depth - sig_bits->gray; + } + if (row_info->color_type & PNG_COLOR_MASK_ALPHA) + { + shift[channels++] = row_info->bit_depth - sig_bits->alpha; + } + + for (c = 0; c < channels; c++) + { + if (shift[c] <= 0) + shift[c] = 0; + else + value = 1; + } + + if (!value) + return; + + switch (row_info->bit_depth) + { + case 2: + { + png_bytep bp; + png_uint_32 i; + png_uint_32 istop = row_info->rowbytes; + + for (bp = row, i = 0; i < istop; i++) + { + *bp >>= 1; + *bp++ &= 0x55; + } + break; + } + case 4: + { + png_bytep bp = row; + png_uint_32 i; + png_uint_32 istop = row_info->rowbytes; + png_byte mask = (png_byte)((((int)0xf0 >> shift[0]) & (int)0xf0) | + (png_byte)((int)0xf >> shift[0])); + + for (i = 0; i < istop; i++) + { + *bp >>= shift[0]; + *bp++ &= mask; + } + break; + } + case 8: + { + png_bytep bp = row; + png_uint_32 i; + png_uint_32 istop = row_width * channels; + + for (i = 0; i < istop; i++) + { + *bp++ >>= shift[i%channels]; + } + break; + } + case 16: + { + png_bytep bp = row; + png_uint_32 i; + png_uint_32 istop = channels * row_width; + + for (i = 0; i < istop; i++) + { + value = (png_uint_16)((*bp << 8) + *(bp + 1)); + value >>= shift[i%channels]; + *bp++ = (png_byte)(value >> 8); + *bp++ = (png_byte)(value & 0xff); + } + break; + } + } + } +} +#endif + +#if defined(PNG_READ_16_TO_8_SUPPORTED) +/* chop rows of bit depth 16 down to 8 */ +void /* PRIVATE */ +png_do_chop(png_row_infop row_info, png_bytep row) +{ + png_debug(1, "in png_do_chop\n"); +#if defined(PNG_USELESS_TESTS_SUPPORTED) + if (row != NULL && row_info != NULL && row_info->bit_depth == 16) +#else + if (row_info->bit_depth == 16) +#endif + { + png_bytep sp = row; + png_bytep dp = row; + png_uint_32 i; + png_uint_32 istop = row_info->width * row_info->channels; + + for (i = 0; i> 8)) >> 8; + * + * Approximate calculation with shift/add instead of multiply/divide: + * *dp = ((((png_uint_32)(*sp) << 8) | + * (png_uint_32)((int)(*(sp + 1)) - *sp)) + 128) >> 8; + * + * What we actually do to avoid extra shifting and conversion: + */ + + *dp = *sp + ((((int)(*(sp + 1)) - *sp) > 128) ? 1 : 0); +#else + /* Simply discard the low order byte */ + *dp = *sp; +#endif + } + row_info->bit_depth = 8; + row_info->pixel_depth = (png_byte)(8 * row_info->channels); + row_info->rowbytes = row_info->width * row_info->channels; + } +} +#endif + +#if defined(PNG_READ_SWAP_ALPHA_SUPPORTED) +void /* PRIVATE */ +png_do_read_swap_alpha(png_row_infop row_info, png_bytep row) +{ + png_debug(1, "in png_do_read_swap_alpha\n"); +#if defined(PNG_USELESS_TESTS_SUPPORTED) + if (row != NULL && row_info != NULL) +#endif + { + png_uint_32 row_width = row_info->width; + if (row_info->color_type == PNG_COLOR_TYPE_RGB_ALPHA) + { + /* This converts from RGBA to ARGB */ + if (row_info->bit_depth == 8) + { + png_bytep sp = row + row_info->rowbytes; + png_bytep dp = sp; + png_byte save; + png_uint_32 i; + + for (i = 0; i < row_width; i++) + { + save = *(--sp); + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = save; + } + } + /* This converts from RRGGBBAA to AARRGGBB */ + else + { + png_bytep sp = row + row_info->rowbytes; + png_bytep dp = sp; + png_byte save[2]; + png_uint_32 i; + + for (i = 0; i < row_width; i++) + { + save[0] = *(--sp); + save[1] = *(--sp); + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = save[0]; + *(--dp) = save[1]; + } + } + } + else if (row_info->color_type == PNG_COLOR_TYPE_GRAY_ALPHA) + { + /* This converts from GA to AG */ + if (row_info->bit_depth == 8) + { + png_bytep sp = row + row_info->rowbytes; + png_bytep dp = sp; + png_byte save; + png_uint_32 i; + + for (i = 0; i < row_width; i++) + { + save = *(--sp); + *(--dp) = *(--sp); + *(--dp) = save; + } + } + /* This converts from GGAA to AAGG */ + else + { + png_bytep sp = row + row_info->rowbytes; + png_bytep dp = sp; + png_byte save[2]; + png_uint_32 i; + + for (i = 0; i < row_width; i++) + { + save[0] = *(--sp); + save[1] = *(--sp); + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = save[0]; + *(--dp) = save[1]; + } + } + } + } +} +#endif + +#if defined(PNG_READ_INVERT_ALPHA_SUPPORTED) +void /* PRIVATE */ +png_do_read_invert_alpha(png_row_infop row_info, png_bytep row) +{ + png_debug(1, "in png_do_read_invert_alpha\n"); +#if defined(PNG_USELESS_TESTS_SUPPORTED) + if (row != NULL && row_info != NULL) +#endif + { + png_uint_32 row_width = row_info->width; + if (row_info->color_type == PNG_COLOR_TYPE_RGB_ALPHA) + { + /* This inverts the alpha channel in RGBA */ + if (row_info->bit_depth == 8) + { + png_bytep sp = row + row_info->rowbytes; + png_bytep dp = sp; + png_uint_32 i; + + for (i = 0; i < row_width; i++) + { + *(--dp) = (png_byte)(255 - *(--sp)); + +/* This does nothing: + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = *(--sp); + We can replace it with: +*/ + sp-=3; + dp=sp; + } + } + /* This inverts the alpha channel in RRGGBBAA */ + else + { + png_bytep sp = row + row_info->rowbytes; + png_bytep dp = sp; + png_uint_32 i; + + for (i = 0; i < row_width; i++) + { + *(--dp) = (png_byte)(255 - *(--sp)); + *(--dp) = (png_byte)(255 - *(--sp)); + +/* This does nothing: + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = *(--sp); + We can replace it with: +*/ + sp-=6; + dp=sp; + } + } + } + else if (row_info->color_type == PNG_COLOR_TYPE_GRAY_ALPHA) + { + /* This inverts the alpha channel in GA */ + if (row_info->bit_depth == 8) + { + png_bytep sp = row + row_info->rowbytes; + png_bytep dp = sp; + png_uint_32 i; + + for (i = 0; i < row_width; i++) + { + *(--dp) = (png_byte)(255 - *(--sp)); + *(--dp) = *(--sp); + } + } + /* This inverts the alpha channel in GGAA */ + else + { + png_bytep sp = row + row_info->rowbytes; + png_bytep dp = sp; + png_uint_32 i; + + for (i = 0; i < row_width; i++) + { + *(--dp) = (png_byte)(255 - *(--sp)); + *(--dp) = (png_byte)(255 - *(--sp)); +/* + *(--dp) = *(--sp); + *(--dp) = *(--sp); +*/ + sp-=2; + dp=sp; + } + } + } + } +} +#endif + +#if defined(PNG_READ_FILLER_SUPPORTED) +/* Add filler channel if we have RGB color */ +void /* PRIVATE */ +png_do_read_filler(png_row_infop row_info, png_bytep row, + png_uint_32 filler, png_uint_32 flags) +{ + png_uint_32 i; + png_uint_32 row_width = row_info->width; + + png_byte hi_filler = (png_byte)((filler>>8) & 0xff); + png_byte lo_filler = (png_byte)(filler & 0xff); + + png_debug(1, "in png_do_read_filler\n"); + if ( +#if defined(PNG_USELESS_TESTS_SUPPORTED) + row != NULL && row_info != NULL && +#endif + row_info->color_type == PNG_COLOR_TYPE_GRAY) + { + if(row_info->bit_depth == 8) + { + /* This changes the data from G to GX */ + if (flags & PNG_FLAG_FILLER_AFTER) + { + png_bytep sp = row + (png_size_t)row_width; + png_bytep dp = sp + (png_size_t)row_width; + for (i = 1; i < row_width; i++) + { + *(--dp) = lo_filler; + *(--dp) = *(--sp); + } + *(--dp) = lo_filler; + row_info->channels = 2; + row_info->pixel_depth = 16; + row_info->rowbytes = row_width * 2; + } + /* This changes the data from G to XG */ + else + { + png_bytep sp = row + (png_size_t)row_width; + png_bytep dp = sp + (png_size_t)row_width; + for (i = 0; i < row_width; i++) + { + *(--dp) = *(--sp); + *(--dp) = lo_filler; + } + row_info->channels = 2; + row_info->pixel_depth = 16; + row_info->rowbytes = row_width * 2; + } + } + else if(row_info->bit_depth == 16) + { + /* This changes the data from GG to GGXX */ + if (flags & PNG_FLAG_FILLER_AFTER) + { + png_bytep sp = row + (png_size_t)row_width * 2; + png_bytep dp = sp + (png_size_t)row_width * 2; + for (i = 1; i < row_width; i++) + { + *(--dp) = hi_filler; + *(--dp) = lo_filler; + *(--dp) = *(--sp); + *(--dp) = *(--sp); + } + *(--dp) = hi_filler; + *(--dp) = lo_filler; + row_info->channels = 2; + row_info->pixel_depth = 32; + row_info->rowbytes = row_width * 4; + } + /* This changes the data from GG to XXGG */ + else + { + png_bytep sp = row + (png_size_t)row_width * 2; + png_bytep dp = sp + (png_size_t)row_width * 2; + for (i = 0; i < row_width; i++) + { + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = hi_filler; + *(--dp) = lo_filler; + } + row_info->channels = 2; + row_info->pixel_depth = 32; + row_info->rowbytes = row_width * 4; + } + } + } /* COLOR_TYPE == GRAY */ + else if (row_info->color_type == PNG_COLOR_TYPE_RGB) + { + if(row_info->bit_depth == 8) + { + /* This changes the data from RGB to RGBX */ + if (flags & PNG_FLAG_FILLER_AFTER) + { + png_bytep sp = row + (png_size_t)row_width * 3; + png_bytep dp = sp + (png_size_t)row_width; + for (i = 1; i < row_width; i++) + { + *(--dp) = lo_filler; + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = *(--sp); + } + *(--dp) = lo_filler; + row_info->channels = 4; + row_info->pixel_depth = 32; + row_info->rowbytes = row_width * 4; + } + /* This changes the data from RGB to XRGB */ + else + { + png_bytep sp = row + (png_size_t)row_width * 3; + png_bytep dp = sp + (png_size_t)row_width; + for (i = 0; i < row_width; i++) + { + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = lo_filler; + } + row_info->channels = 4; + row_info->pixel_depth = 32; + row_info->rowbytes = row_width * 4; + } + } + else if(row_info->bit_depth == 16) + { + /* This changes the data from RRGGBB to RRGGBBXX */ + if (flags & PNG_FLAG_FILLER_AFTER) + { + png_bytep sp = row + (png_size_t)row_width * 6; + png_bytep dp = sp + (png_size_t)row_width * 2; + for (i = 1; i < row_width; i++) + { + *(--dp) = hi_filler; + *(--dp) = lo_filler; + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = *(--sp); + } + *(--dp) = hi_filler; + *(--dp) = lo_filler; + row_info->channels = 4; + row_info->pixel_depth = 64; + row_info->rowbytes = row_width * 8; + } + /* This changes the data from RRGGBB to XXRRGGBB */ + else + { + png_bytep sp = row + (png_size_t)row_width * 6; + png_bytep dp = sp + (png_size_t)row_width * 2; + for (i = 0; i < row_width; i++) + { + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = *(--sp); + *(--dp) = hi_filler; + *(--dp) = lo_filler; + } + row_info->channels = 4; + row_info->pixel_depth = 64; + row_info->rowbytes = row_width * 8; + } + } + } /* COLOR_TYPE == RGB */ +} +#endif + +#if defined(PNG_READ_GRAY_TO_RGB_SUPPORTED) +/* expand grayscale files to RGB, with or without alpha */ +void /* PRIVATE */ +png_do_gray_to_rgb(png_row_infop row_info, png_bytep row) +{ + png_uint_32 i; + png_uint_32 row_width = row_info->width; + + png_debug(1, "in png_do_gray_to_rgb\n"); + if (row_info->bit_depth >= 8 && +#if defined(PNG_USELESS_TESTS_SUPPORTED) + row != NULL && row_info != NULL && +#endif + !(row_info->color_type & PNG_COLOR_MASK_COLOR)) + { + if (row_info->color_type == PNG_COLOR_TYPE_GRAY) + { + if (row_info->bit_depth == 8) + { + png_bytep sp = row + (png_size_t)row_width - 1; + png_bytep dp = sp + (png_size_t)row_width * 2; + for (i = 0; i < row_width; i++) + { + *(dp--) = *sp; + *(dp--) = *sp; + *(dp--) = *(sp--); + } + } + else + { + png_bytep sp = row + (png_size_t)row_width * 2 - 1; + png_bytep dp = sp + (png_size_t)row_width * 4; + for (i = 0; i < row_width; i++) + { + *(dp--) = *sp; + *(dp--) = *(sp - 1); + *(dp--) = *sp; + *(dp--) = *(sp - 1); + *(dp--) = *(sp--); + *(dp--) = *(sp--); + } + } + } + else if (row_info->color_type == PNG_COLOR_TYPE_GRAY_ALPHA) + { + if (row_info->bit_depth == 8) + { + png_bytep sp = row + (png_size_t)row_width * 2 - 1; + png_bytep dp = sp + (png_size_t)row_width * 2; + for (i = 0; i < row_width; i++) + { + *(dp--) = *(sp--); + *(dp--) = *sp; + *(dp--) = *sp; + *(dp--) = *(sp--); + } + } + else + { + png_bytep sp = row + (png_size_t)row_width * 4 - 1; + png_bytep dp = sp + (png_size_t)row_width * 4; + for (i = 0; i < row_width; i++) + { + *(dp--) = *(sp--); + *(dp--) = *(sp--); + *(dp--) = *sp; + *(dp--) = *(sp - 1); + *(dp--) = *sp; + *(dp--) = *(sp - 1); + *(dp--) = *(sp--); + *(dp--) = *(sp--); + } + } + } + row_info->channels += (png_byte)2; + row_info->color_type |= PNG_COLOR_MASK_COLOR; + row_info->pixel_depth = (png_byte)(row_info->channels * + row_info->bit_depth); + row_info->rowbytes = PNG_ROWBYTES(row_info->pixel_depth,row_width); + } +} +#endif + +#if defined(PNG_READ_RGB_TO_GRAY_SUPPORTED) +/* reduce RGB files to grayscale, with or without alpha + * using the equation given in Poynton's ColorFAQ at + * + * Copyright (c) 1998-01-04 Charles Poynton poynton at inforamp.net + * + * Y = 0.212671 * R + 0.715160 * G + 0.072169 * B + * + * We approximate this with + * + * Y = 0.21268 * R + 0.7151 * G + 0.07217 * B + * + * which can be expressed with integers as + * + * Y = (6969 * R + 23434 * G + 2365 * B)/32768 + * + * The calculation is to be done in a linear colorspace. + * + * Other integer coefficents can be used via png_set_rgb_to_gray(). + */ +int /* PRIVATE */ +png_do_rgb_to_gray(png_structp png_ptr, png_row_infop row_info, png_bytep row) + +{ + png_uint_32 i; + + png_uint_32 row_width = row_info->width; + int rgb_error = 0; + + png_debug(1, "in png_do_rgb_to_gray\n"); + if ( +#if defined(PNG_USELESS_TESTS_SUPPORTED) + row != NULL && row_info != NULL && +#endif + (row_info->color_type & PNG_COLOR_MASK_COLOR)) + { + png_uint_32 rc = png_ptr->rgb_to_gray_red_coeff; + png_uint_32 gc = png_ptr->rgb_to_gray_green_coeff; + png_uint_32 bc = png_ptr->rgb_to_gray_blue_coeff; + + if (row_info->color_type == PNG_COLOR_TYPE_RGB) + { + if (row_info->bit_depth == 8) + { +#if defined(PNG_READ_GAMMA_SUPPORTED) || defined(PNG_READ_BACKGROUND_SUPPORTED) + if (png_ptr->gamma_from_1 != NULL && png_ptr->gamma_to_1 != NULL) + { + png_bytep sp = row; + png_bytep dp = row; + + for (i = 0; i < row_width; i++) + { + png_byte red = png_ptr->gamma_to_1[*(sp++)]; + png_byte green = png_ptr->gamma_to_1[*(sp++)]; + png_byte blue = png_ptr->gamma_to_1[*(sp++)]; + if(red != green || red != blue) + { + rgb_error |= 1; + *(dp++) = png_ptr->gamma_from_1[ + (rc*red+gc*green+bc*blue)>>15]; + } + else + *(dp++) = *(sp-1); + } + } + else +#endif + { + png_bytep sp = row; + png_bytep dp = row; + for (i = 0; i < row_width; i++) + { + png_byte red = *(sp++); + png_byte green = *(sp++); + png_byte blue = *(sp++); + if(red != green || red != blue) + { + rgb_error |= 1; + *(dp++) = (png_byte)((rc*red+gc*green+bc*blue)>>15); + } + else + *(dp++) = *(sp-1); + } + } + } + + else /* RGB bit_depth == 16 */ + { +#if defined(PNG_READ_GAMMA_SUPPORTED) || defined(PNG_READ_BACKGROUND_SUPPORTED) + if (png_ptr->gamma_16_to_1 != NULL && + png_ptr->gamma_16_from_1 != NULL) + { + png_bytep sp = row; + png_bytep dp = row; + for (i = 0; i < row_width; i++) + { + png_uint_16 red, green, blue, w; + + red = (png_uint_16)(((*(sp))<<8) | *(sp+1)); sp+=2; + green = (png_uint_16)(((*(sp))<<8) | *(sp+1)); sp+=2; + blue = (png_uint_16)(((*(sp))<<8) | *(sp+1)); sp+=2; + + if(red == green && red == blue) + w = red; + else + { + png_uint_16 red_1 = png_ptr->gamma_16_to_1[(red&0xff) >> + png_ptr->gamma_shift][red>>8]; + png_uint_16 green_1 = png_ptr->gamma_16_to_1[(green&0xff) >> + png_ptr->gamma_shift][green>>8]; + png_uint_16 blue_1 = png_ptr->gamma_16_to_1[(blue&0xff) >> + png_ptr->gamma_shift][blue>>8]; + png_uint_16 gray16 = (png_uint_16)((rc*red_1 + gc*green_1 + + bc*blue_1)>>15); + w = png_ptr->gamma_16_from_1[(gray16&0xff) >> + png_ptr->gamma_shift][gray16 >> 8]; + rgb_error |= 1; + } + + *(dp++) = (png_byte)((w>>8) & 0xff); + *(dp++) = (png_byte)(w & 0xff); + } + } + else +#endif + { + png_bytep sp = row; + png_bytep dp = row; + for (i = 0; i < row_width; i++) + { + png_uint_16 red, green, blue, gray16; + + red = (png_uint_16)(((*(sp))<<8) | *(sp+1)); sp+=2; + green = (png_uint_16)(((*(sp))<<8) | *(sp+1)); sp+=2; + blue = (png_uint_16)(((*(sp))<<8) | *(sp+1)); sp+=2; + + if(red != green || red != blue) + rgb_error |= 1; + gray16 = (png_uint_16)((rc*red + gc*green + bc*blue)>>15); + *(dp++) = (png_byte)((gray16>>8) & 0xff); + *(dp++) = (png_byte)(gray16 & 0xff); + } + } + } + } + if (row_info->color_type == PNG_COLOR_TYPE_RGB_ALPHA) + { + if (row_info->bit_depth == 8) + { +#if defined(PNG_READ_GAMMA_SUPPORTED) || defined(PNG_READ_BACKGROUND_SUPPORTED) + if (png_ptr->gamma_from_1 != NULL && png_ptr->gamma_to_1 != NULL) + { + png_bytep sp = row; + png_bytep dp = row; + for (i = 0; i < row_width; i++) + { + png_byte red = png_ptr->gamma_to_1[*(sp++)]; + png_byte green = png_ptr->gamma_to_1[*(sp++)]; + png_byte blue = png_ptr->gamma_to_1[*(sp++)]; + if(red != green || red != blue) + rgb_error |= 1; + *(dp++) = png_ptr->gamma_from_1 + [(rc*red + gc*green + bc*blue)>>15]; + *(dp++) = *(sp++); /* alpha */ + } + } + else +#endif + { + png_bytep sp = row; + png_bytep dp = row; + for (i = 0; i < row_width; i++) + { + png_byte red = *(sp++); + png_byte green = *(sp++); + png_byte blue = *(sp++); + if(red != green || red != blue) + rgb_error |= 1; + *(dp++) = (png_byte)((rc*red + gc*green + bc*blue)>>15); + *(dp++) = *(sp++); /* alpha */ + } + } + } + else /* RGBA bit_depth == 16 */ + { +#if defined(PNG_READ_GAMMA_SUPPORTED) || defined(PNG_READ_BACKGROUND_SUPPORTED) + if (png_ptr->gamma_16_to_1 != NULL && + png_ptr->gamma_16_from_1 != NULL) + { + png_bytep sp = row; + png_bytep dp = row; + for (i = 0; i < row_width; i++) + { + png_uint_16 red, green, blue, w; + + red = (png_uint_16)(((*(sp))<<8) | *(sp+1)); sp+=2; + green = (png_uint_16)(((*(sp))<<8) | *(sp+1)); sp+=2; + blue = (png_uint_16)(((*(sp))<<8) | *(sp+1)); sp+=2; + + if(red == green && red == blue) + w = red; + else + { + png_uint_16 red_1 = png_ptr->gamma_16_to_1[(red&0xff) >> + png_ptr->gamma_shift][red>>8]; + png_uint_16 green_1 = png_ptr->gamma_16_to_1[(green&0xff) >> + png_ptr->gamma_shift][green>>8]; + png_uint_16 blue_1 = png_ptr->gamma_16_to_1[(blue&0xff) >> + png_ptr->gamma_shift][blue>>8]; + png_uint_16 gray16 = (png_uint_16)((rc * red_1 + + gc * green_1 + bc * blue_1)>>15); + w = png_ptr->gamma_16_from_1[(gray16&0xff) >> + png_ptr->gamma_shift][gray16 >> 8]; + rgb_error |= 1; + } + + *(dp++) = (png_byte)((w>>8) & 0xff); + *(dp++) = (png_byte)(w & 0xff); + *(dp++) = *(sp++); /* alpha */ + *(dp++) = *(sp++); + } + } + else +#endif + { + png_bytep sp = row; + png_bytep dp = row; + for (i = 0; i < row_width; i++) + { + png_uint_16 red, green, blue, gray16; + red = (png_uint_16)((*(sp)<<8) | *(sp+1)); sp+=2; + green = (png_uint_16)((*(sp)<<8) | *(sp+1)); sp+=2; + blue = (png_uint_16)((*(sp)<<8) | *(sp+1)); sp+=2; + if(red != green || red != blue) + rgb_error |= 1; + gray16 = (png_uint_16)((rc*red + gc*green + bc*blue)>>15); + *(dp++) = (png_byte)((gray16>>8) & 0xff); + *(dp++) = (png_byte)(gray16 & 0xff); + *(dp++) = *(sp++); /* alpha */ + *(dp++) = *(sp++); + } + } + } + } + row_info->channels -= (png_byte)2; + row_info->color_type &= ~PNG_COLOR_MASK_COLOR; + row_info->pixel_depth = (png_byte)(row_info->channels * + row_info->bit_depth); + row_info->rowbytes = PNG_ROWBYTES(row_info->pixel_depth,row_width); + } + return rgb_error; +} +#endif + +/* Build a grayscale palette. Palette is assumed to be 1 << bit_depth + * large of png_color. This lets grayscale images be treated as + * paletted. Most useful for gamma correction and simplification + * of code. + */ +void PNGAPI +png_build_grayscale_palette(int bit_depth, png_colorp palette) +{ + int num_palette; + int color_inc; + int i; + int v; + + png_debug(1, "in png_do_build_grayscale_palette\n"); + if (palette == NULL) + return; + + switch (bit_depth) + { + case 1: + num_palette = 2; + color_inc = 0xff; + break; + case 2: + num_palette = 4; + color_inc = 0x55; + break; + case 4: + num_palette = 16; + color_inc = 0x11; + break; + case 8: + num_palette = 256; + color_inc = 1; + break; + default: + num_palette = 0; + color_inc = 0; + break; + } + + for (i = 0, v = 0; i < num_palette; i++, v += color_inc) + { + palette[i].red = (png_byte)v; + palette[i].green = (png_byte)v; + palette[i].blue = (png_byte)v; + } +} + +/* This function is currently unused. Do we really need it? */ +#if defined(PNG_READ_DITHER_SUPPORTED) && defined(PNG_CORRECT_PALETTE_SUPPORTED) +void /* PRIVATE */ +png_correct_palette(png_structp png_ptr, png_colorp palette, + int num_palette) +{ + png_debug(1, "in png_correct_palette\n"); +#if defined(PNG_READ_BACKGROUND_SUPPORTED) && \ + defined(PNG_READ_GAMMA_SUPPORTED) && defined(PNG_FLOATING_POINT_SUPPORTED) + if (png_ptr->transformations & (PNG_GAMMA | PNG_BACKGROUND)) + { + png_color back, back_1; + + if (png_ptr->background_gamma_type == PNG_BACKGROUND_GAMMA_FILE) + { + back.red = png_ptr->gamma_table[png_ptr->background.red]; + back.green = png_ptr->gamma_table[png_ptr->background.green]; + back.blue = png_ptr->gamma_table[png_ptr->background.blue]; + + back_1.red = png_ptr->gamma_to_1[png_ptr->background.red]; + back_1.green = png_ptr->gamma_to_1[png_ptr->background.green]; + back_1.blue = png_ptr->gamma_to_1[png_ptr->background.blue]; + } + else + { + double g; + + g = 1.0 / (png_ptr->background_gamma * png_ptr->screen_gamma); + + if (png_ptr->background_gamma_type == PNG_BACKGROUND_GAMMA_SCREEN || + fabs(g - 1.0) < PNG_GAMMA_THRESHOLD) + { + back.red = png_ptr->background.red; + back.green = png_ptr->background.green; + back.blue = png_ptr->background.blue; + } + else + { + back.red = + (png_byte)(pow((double)png_ptr->background.red/255, g) * + 255.0 + 0.5); + back.green = + (png_byte)(pow((double)png_ptr->background.green/255, g) * + 255.0 + 0.5); + back.blue = + (png_byte)(pow((double)png_ptr->background.blue/255, g) * + 255.0 + 0.5); + } + + g = 1.0 / png_ptr->background_gamma; + + back_1.red = + (png_byte)(pow((double)png_ptr->background.red/255, g) * + 255.0 + 0.5); + back_1.green = + (png_byte)(pow((double)png_ptr->background.green/255, g) * + 255.0 + 0.5); + back_1.blue = + (png_byte)(pow((double)png_ptr->background.blue/255, g) * + 255.0 + 0.5); + } + + if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE) + { + png_uint_32 i; + + for (i = 0; i < (png_uint_32)num_palette; i++) + { + if (i < png_ptr->num_trans && png_ptr->trans[i] == 0) + { + palette[i] = back; + } + else if (i < png_ptr->num_trans && png_ptr->trans[i] != 0xff) + { + png_byte v, w; + + v = png_ptr->gamma_to_1[png_ptr->palette[i].red]; + png_composite(w, v, png_ptr->trans[i], back_1.red); + palette[i].red = png_ptr->gamma_from_1[w]; + + v = png_ptr->gamma_to_1[png_ptr->palette[i].green]; + png_composite(w, v, png_ptr->trans[i], back_1.green); + palette[i].green = png_ptr->gamma_from_1[w]; + + v = png_ptr->gamma_to_1[png_ptr->palette[i].blue]; + png_composite(w, v, png_ptr->trans[i], back_1.blue); + palette[i].blue = png_ptr->gamma_from_1[w]; + } + else + { + palette[i].red = png_ptr->gamma_table[palette[i].red]; + palette[i].green = png_ptr->gamma_table[palette[i].green]; + palette[i].blue = png_ptr->gamma_table[palette[i].blue]; + } + } + } + else + { + int i; + + for (i = 0; i < num_palette; i++) + { + if (palette[i].red == (png_byte)png_ptr->trans_values.gray) + { + palette[i] = back; + } + else + { + palette[i].red = png_ptr->gamma_table[palette[i].red]; + palette[i].green = png_ptr->gamma_table[palette[i].green]; + palette[i].blue = png_ptr->gamma_table[palette[i].blue]; + } + } + } + } + else +#endif +#if defined(PNG_READ_GAMMA_SUPPORTED) + if (png_ptr->transformations & PNG_GAMMA) + { + int i; + + for (i = 0; i < num_palette; i++) + { + palette[i].red = png_ptr->gamma_table[palette[i].red]; + palette[i].green = png_ptr->gamma_table[palette[i].green]; + palette[i].blue = png_ptr->gamma_table[palette[i].blue]; + } + } +#if defined(PNG_READ_BACKGROUND_SUPPORTED) + else +#endif +#endif +#if defined(PNG_READ_BACKGROUND_SUPPORTED) + if (png_ptr->transformations & PNG_BACKGROUND) + { + if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE) + { + png_color back; + + back.red = (png_byte)png_ptr->background.red; + back.green = (png_byte)png_ptr->background.green; + back.blue = (png_byte)png_ptr->background.blue; + + for (i = 0; i < (int)png_ptr->num_trans; i++) + { + if (png_ptr->trans[i] == 0) + { + palette[i].red = back.red; + palette[i].green = back.green; + palette[i].blue = back.blue; + } + else if (png_ptr->trans[i] != 0xff) + { + png_composite(palette[i].red, png_ptr->palette[i].red, + png_ptr->trans[i], back.red); + png_composite(palette[i].green, png_ptr->palette[i].green, + png_ptr->trans[i], back.green); + png_composite(palette[i].blue, png_ptr->palette[i].blue, + png_ptr->trans[i], back.blue); + } + } + } + else /* assume grayscale palette (what else could it be?) */ + { + int i; + + for (i = 0; i < num_palette; i++) + { + if (i == (png_byte)png_ptr->trans_values.gray) + { + palette[i].red = (png_byte)png_ptr->background.red; + palette[i].green = (png_byte)png_ptr->background.green; + palette[i].blue = (png_byte)png_ptr->background.blue; + } + } + } + } +#endif +} +#endif + +#if defined(PNG_READ_BACKGROUND_SUPPORTED) +/* Replace any alpha or transparency with the supplied background color. + * "background" is already in the screen gamma, while "background_1" is + * at a gamma of 1.0. Paletted files have already been taken care of. + */ +void /* PRIVATE */ +png_do_background(png_row_infop row_info, png_bytep row, + png_color_16p trans_values, png_color_16p background +#if defined(PNG_READ_GAMMA_SUPPORTED) + , png_color_16p background_1, + png_bytep gamma_table, png_bytep gamma_from_1, png_bytep gamma_to_1, + png_uint_16pp gamma_16, png_uint_16pp gamma_16_from_1, + png_uint_16pp gamma_16_to_1, int gamma_shift +#endif + ) +{ + png_bytep sp, dp; + png_uint_32 i; + png_uint_32 row_width=row_info->width; + int shift; + + png_debug(1, "in png_do_background\n"); + if (background != NULL && +#if defined(PNG_USELESS_TESTS_SUPPORTED) + row != NULL && row_info != NULL && +#endif + (!(row_info->color_type & PNG_COLOR_MASK_ALPHA) || + (row_info->color_type != PNG_COLOR_TYPE_PALETTE && trans_values))) + { + switch (row_info->color_type) + { + case PNG_COLOR_TYPE_GRAY: + { + switch (row_info->bit_depth) + { + case 1: + { + sp = row; + shift = 7; + for (i = 0; i < row_width; i++) + { + if ((png_uint_16)((*sp >> shift) & 0x01) + == trans_values->gray) + { + *sp &= (png_byte)((0x7f7f >> (7 - shift)) & 0xff); + *sp |= (png_byte)(background->gray << shift); + } + if (!shift) + { + shift = 7; + sp++; + } + else + shift--; + } + break; + } + case 2: + { +#if defined(PNG_READ_GAMMA_SUPPORTED) + if (gamma_table != NULL) + { + sp = row; + shift = 6; + for (i = 0; i < row_width; i++) + { + if ((png_uint_16)((*sp >> shift) & 0x03) + == trans_values->gray) + { + *sp &= (png_byte)((0x3f3f >> (6 - shift)) & 0xff); + *sp |= (png_byte)(background->gray << shift); + } + else + { + png_byte p = (png_byte)((*sp >> shift) & 0x03); + png_byte g = (png_byte)((gamma_table [p | (p << 2) | + (p << 4) | (p << 6)] >> 6) & 0x03); + *sp &= (png_byte)((0x3f3f >> (6 - shift)) & 0xff); + *sp |= (png_byte)(g << shift); + } + if (!shift) + { + shift = 6; + sp++; + } + else + shift -= 2; + } + } + else +#endif + { + sp = row; + shift = 6; + for (i = 0; i < row_width; i++) + { + if ((png_uint_16)((*sp >> shift) & 0x03) + == trans_values->gray) + { + *sp &= (png_byte)((0x3f3f >> (6 - shift)) & 0xff); + *sp |= (png_byte)(background->gray << shift); + } + if (!shift) + { + shift = 6; + sp++; + } + else + shift -= 2; + } + } + break; + } + case 4: + { +#if defined(PNG_READ_GAMMA_SUPPORTED) + if (gamma_table != NULL) + { + sp = row; + shift = 4; + for (i = 0; i < row_width; i++) + { + if ((png_uint_16)((*sp >> shift) & 0x0f) + == trans_values->gray) + { + *sp &= (png_byte)((0xf0f >> (4 - shift)) & 0xff); + *sp |= (png_byte)(background->gray << shift); + } + else + { + png_byte p = (png_byte)((*sp >> shift) & 0x0f); + png_byte g = (png_byte)((gamma_table[p | + (p << 4)] >> 4) & 0x0f); + *sp &= (png_byte)((0xf0f >> (4 - shift)) & 0xff); + *sp |= (png_byte)(g << shift); + } + if (!shift) + { + shift = 4; + sp++; + } + else + shift -= 4; + } + } + else +#endif + { + sp = row; + shift = 4; + for (i = 0; i < row_width; i++) + { + if ((png_uint_16)((*sp >> shift) & 0x0f) + == trans_values->gray) + { + *sp &= (png_byte)((0xf0f >> (4 - shift)) & 0xff); + *sp |= (png_byte)(background->gray << shift); + } + if (!shift) + { + shift = 4; + sp++; + } + else + shift -= 4; + } + } + break; + } + case 8: + { +#if defined(PNG_READ_GAMMA_SUPPORTED) + if (gamma_table != NULL) + { + sp = row; + for (i = 0; i < row_width; i++, sp++) + { + if (*sp == trans_values->gray) + { + *sp = (png_byte)background->gray; + } + else + { + *sp = gamma_table[*sp]; + } + } + } + else +#endif + { + sp = row; + for (i = 0; i < row_width; i++, sp++) + { + if (*sp == trans_values->gray) + { + *sp = (png_byte)background->gray; + } + } + } + break; + } + case 16: + { +#if defined(PNG_READ_GAMMA_SUPPORTED) + if (gamma_16 != NULL) + { + sp = row; + for (i = 0; i < row_width; i++, sp += 2) + { + png_uint_16 v; + + v = (png_uint_16)(((*sp) << 8) + *(sp + 1)); + if (v == trans_values->gray) + { + /* background is already in screen gamma */ + *sp = (png_byte)((background->gray >> 8) & 0xff); + *(sp + 1) = (png_byte)(background->gray & 0xff); + } + else + { + v = gamma_16[*(sp + 1) >> gamma_shift][*sp]; + *sp = (png_byte)((v >> 8) & 0xff); + *(sp + 1) = (png_byte)(v & 0xff); + } + } + } + else +#endif + { + sp = row; + for (i = 0; i < row_width; i++, sp += 2) + { + png_uint_16 v; + + v = (png_uint_16)(((*sp) << 8) + *(sp + 1)); + if (v == trans_values->gray) + { + *sp = (png_byte)((background->gray >> 8) & 0xff); + *(sp + 1) = (png_byte)(background->gray & 0xff); + } + } + } + break; + } + } + break; + } + case PNG_COLOR_TYPE_RGB: + { + if (row_info->bit_depth == 8) + { +#if defined(PNG_READ_GAMMA_SUPPORTED) + if (gamma_table != NULL) + { + sp = row; + for (i = 0; i < row_width; i++, sp += 3) + { + if (*sp == trans_values->red && + *(sp + 1) == trans_values->green && + *(sp + 2) == trans_values->blue) + { + *sp = (png_byte)background->red; + *(sp + 1) = (png_byte)background->green; + *(sp + 2) = (png_byte)background->blue; + } + else + { + *sp = gamma_table[*sp]; + *(sp + 1) = gamma_table[*(sp + 1)]; + *(sp + 2) = gamma_table[*(sp + 2)]; + } + } + } + else +#endif + { + sp = row; + for (i = 0; i < row_width; i++, sp += 3) + { + if (*sp == trans_values->red && + *(sp + 1) == trans_values->green && + *(sp + 2) == trans_values->blue) + { + *sp = (png_byte)background->red; + *(sp + 1) = (png_byte)background->green; + *(sp + 2) = (png_byte)background->blue; + } + } + } + } + else /* if (row_info->bit_depth == 16) */ + { +#if defined(PNG_READ_GAMMA_SUPPORTED) + if (gamma_16 != NULL) + { + sp = row; + for (i = 0; i < row_width; i++, sp += 6) + { + png_uint_16 r = (png_uint_16)(((*sp) << 8) + *(sp + 1)); + png_uint_16 g = (png_uint_16)(((*(sp+2)) << 8) + *(sp+3)); + png_uint_16 b = (png_uint_16)(((*(sp+4)) << 8) + *(sp+5)); + if (r == trans_values->red && g == trans_values->green && + b == trans_values->blue) + { + /* background is already in screen gamma */ + *sp = (png_byte)((background->red >> 8) & 0xff); + *(sp + 1) = (png_byte)(background->red & 0xff); + *(sp + 2) = (png_byte)((background->green >> 8) & 0xff); + *(sp + 3) = (png_byte)(background->green & 0xff); + *(sp + 4) = (png_byte)((background->blue >> 8) & 0xff); + *(sp + 5) = (png_byte)(background->blue & 0xff); + } + else + { + png_uint_16 v = gamma_16[*(sp + 1) >> gamma_shift][*sp]; + *sp = (png_byte)((v >> 8) & 0xff); + *(sp + 1) = (png_byte)(v & 0xff); + v = gamma_16[*(sp + 3) >> gamma_shift][*(sp + 2)]; + *(sp + 2) = (png_byte)((v >> 8) & 0xff); + *(sp + 3) = (png_byte)(v & 0xff); + v = gamma_16[*(sp + 5) >> gamma_shift][*(sp + 4)]; + *(sp + 4) = (png_byte)((v >> 8) & 0xff); + *(sp + 5) = (png_byte)(v & 0xff); + } + } + } + else +#endif + { + sp = row; + for (i = 0; i < row_width; i++, sp += 6) + { + png_uint_16 r = (png_uint_16)(((*sp) << 8) + *(sp+1)); + png_uint_16 g = (png_uint_16)(((*(sp+2)) << 8) + *(sp+3)); + png_uint_16 b = (png_uint_16)(((*(sp+4)) << 8) + *(sp+5)); + + if (r == trans_values->red && g == trans_values->green && + b == trans_values->blue) + { + *sp = (png_byte)((background->red >> 8) & 0xff); + *(sp + 1) = (png_byte)(background->red & 0xff); + *(sp + 2) = (png_byte)((background->green >> 8) & 0xff); + *(sp + 3) = (png_byte)(background->green & 0xff); + *(sp + 4) = (png_byte)((background->blue >> 8) & 0xff); + *(sp + 5) = (png_byte)(background->blue & 0xff); + } + } + } + } + break; + } + case PNG_COLOR_TYPE_GRAY_ALPHA: + { + if (row_info->bit_depth == 8) + { +#if defined(PNG_READ_GAMMA_SUPPORTED) + if (gamma_to_1 != NULL && gamma_from_1 != NULL && + gamma_table != NULL) + { + sp = row; + dp = row; + for (i = 0; i < row_width; i++, sp += 2, dp++) + { + png_uint_16 a = *(sp + 1); + + if (a == 0xff) + { + *dp = gamma_table[*sp]; + } + else if (a == 0) + { + /* background is already in screen gamma */ + *dp = (png_byte)background->gray; + } + else + { + png_byte v, w; + + v = gamma_to_1[*sp]; + png_composite(w, v, a, background_1->gray); + *dp = gamma_from_1[w]; + } + } + } + else +#endif + { + sp = row; + dp = row; + for (i = 0; i < row_width; i++, sp += 2, dp++) + { + png_byte a = *(sp + 1); + + if (a == 0xff) + { + *dp = *sp; + } +#if defined(PNG_READ_GAMMA_SUPPORTED) + else if (a == 0) + { + *dp = (png_byte)background->gray; + } + else + { + png_composite(*dp, *sp, a, background_1->gray); + } +#else + *dp = (png_byte)background->gray; +#endif + } + } + } + else /* if (png_ptr->bit_depth == 16) */ + { +#if defined(PNG_READ_GAMMA_SUPPORTED) + if (gamma_16 != NULL && gamma_16_from_1 != NULL && + gamma_16_to_1 != NULL) + { + sp = row; + dp = row; + for (i = 0; i < row_width; i++, sp += 4, dp += 2) + { + png_uint_16 a = (png_uint_16)(((*(sp+2)) << 8) + *(sp+3)); + + if (a == (png_uint_16)0xffff) + { + png_uint_16 v; + + v = gamma_16[*(sp + 1) >> gamma_shift][*sp]; + *dp = (png_byte)((v >> 8) & 0xff); + *(dp + 1) = (png_byte)(v & 0xff); + } +#if defined(PNG_READ_GAMMA_SUPPORTED) + else if (a == 0) +#else + else +#endif + { + /* background is already in screen gamma */ + *dp = (png_byte)((background->gray >> 8) & 0xff); + *(dp + 1) = (png_byte)(background->gray & 0xff); + } +#if defined(PNG_READ_GAMMA_SUPPORTED) + else + { + png_uint_16 g, v, w; + + g = gamma_16_to_1[*(sp + 1) >> gamma_shift][*sp]; + png_composite_16(v, g, a, background_1->gray); + w = gamma_16_from_1[(v&0xff) >> gamma_shift][v >> 8]; + *dp = (png_byte)((w >> 8) & 0xff); + *(dp + 1) = (png_byte)(w & 0xff); + } +#endif + } + } + else +#endif + { + sp = row; + dp = row; + for (i = 0; i < row_width; i++, sp += 4, dp += 2) + { + png_uint_16 a = (png_uint_16)(((*(sp+2)) << 8) + *(sp+3)); + if (a == (png_uint_16)0xffff) + { + png_memcpy(dp, sp, 2); + } +#if defined(PNG_READ_GAMMA_SUPPORTED) + else if (a == 0) +#else + else +#endif + { + *dp = (png_byte)((background->gray >> 8) & 0xff); + *(dp + 1) = (png_byte)(background->gray & 0xff); + } +#if defined(PNG_READ_GAMMA_SUPPORTED) + else + { + png_uint_16 g, v; + + g = (png_uint_16)(((*sp) << 8) + *(sp + 1)); + png_composite_16(v, g, a, background_1->gray); + *dp = (png_byte)((v >> 8) & 0xff); + *(dp + 1) = (png_byte)(v & 0xff); + } +#endif + } + } + } + break; + } + case PNG_COLOR_TYPE_RGB_ALPHA: + { + if (row_info->bit_depth == 8) + { +#if defined(PNG_READ_GAMMA_SUPPORTED) + if (gamma_to_1 != NULL && gamma_from_1 != NULL && + gamma_table != NULL) + { + sp = row; + dp = row; + for (i = 0; i < row_width; i++, sp += 4, dp += 3) + { + png_byte a = *(sp + 3); + + if (a == 0xff) + { + *dp = gamma_table[*sp]; + *(dp + 1) = gamma_table[*(sp + 1)]; + *(dp + 2) = gamma_table[*(sp + 2)]; + } + else if (a == 0) + { + /* background is already in screen gamma */ + *dp = (png_byte)background->red; + *(dp + 1) = (png_byte)background->green; + *(dp + 2) = (png_byte)background->blue; + } + else + { + png_byte v, w; + + v = gamma_to_1[*sp]; + png_composite(w, v, a, background_1->red); + *dp = gamma_from_1[w]; + v = gamma_to_1[*(sp + 1)]; + png_composite(w, v, a, background_1->green); + *(dp + 1) = gamma_from_1[w]; + v = gamma_to_1[*(sp + 2)]; + png_composite(w, v, a, background_1->blue); + *(dp + 2) = gamma_from_1[w]; + } + } + } + else +#endif + { + sp = row; + dp = row; + for (i = 0; i < row_width; i++, sp += 4, dp += 3) + { + png_byte a = *(sp + 3); + + if (a == 0xff) + { + *dp = *sp; + *(dp + 1) = *(sp + 1); + *(dp + 2) = *(sp + 2); + } + else if (a == 0) + { + *dp = (png_byte)background->red; + *(dp + 1) = (png_byte)background->green; + *(dp + 2) = (png_byte)background->blue; + } + else + { + png_composite(*dp, *sp, a, background->red); + png_composite(*(dp + 1), *(sp + 1), a, + background->green); + png_composite(*(dp + 2), *(sp + 2), a, + background->blue); + } + } + } + } + else /* if (row_info->bit_depth == 16) */ + { +#if defined(PNG_READ_GAMMA_SUPPORTED) + if (gamma_16 != NULL && gamma_16_from_1 != NULL && + gamma_16_to_1 != NULL) + { + sp = row; + dp = row; + for (i = 0; i < row_width; i++, sp += 8, dp += 6) + { + png_uint_16 a = (png_uint_16)(((png_uint_16)(*(sp + 6)) + << 8) + (png_uint_16)(*(sp + 7))); + if (a == (png_uint_16)0xffff) + { + png_uint_16 v; + + v = gamma_16[*(sp + 1) >> gamma_shift][*sp]; + *dp = (png_byte)((v >> 8) & 0xff); + *(dp + 1) = (png_byte)(v & 0xff); + v = gamma_16[*(sp + 3) >> gamma_shift][*(sp + 2)]; + *(dp + 2) = (png_byte)((v >> 8) & 0xff); + *(dp + 3) = (png_byte)(v & 0xff); + v = gamma_16[*(sp + 5) >> gamma_shift][*(sp + 4)]; + *(dp + 4) = (png_byte)((v >> 8) & 0xff); + *(dp + 5) = (png_byte)(v & 0xff); + } + else if (a == 0) + { + /* background is already in screen gamma */ + *dp = (png_byte)((background->red >> 8) & 0xff); + *(dp + 1) = (png_byte)(background->red & 0xff); + *(dp + 2) = (png_byte)((background->green >> 8) & 0xff); + *(dp + 3) = (png_byte)(background->green & 0xff); + *(dp + 4) = (png_byte)((background->blue >> 8) & 0xff); + *(dp + 5) = (png_byte)(background->blue & 0xff); + } + else + { + png_uint_16 v, w, x; + + v = gamma_16_to_1[*(sp + 1) >> gamma_shift][*sp]; + png_composite_16(w, v, a, background_1->red); + x = gamma_16_from_1[((w&0xff) >> gamma_shift)][w >> 8]; + *dp = (png_byte)((x >> 8) & 0xff); + *(dp + 1) = (png_byte)(x & 0xff); + v = gamma_16_to_1[*(sp + 3) >> gamma_shift][*(sp + 2)]; + png_composite_16(w, v, a, background_1->green); + x = gamma_16_from_1[((w&0xff) >> gamma_shift)][w >> 8]; + *(dp + 2) = (png_byte)((x >> 8) & 0xff); + *(dp + 3) = (png_byte)(x & 0xff); + v = gamma_16_to_1[*(sp + 5) >> gamma_shift][*(sp + 4)]; + png_composite_16(w, v, a, background_1->blue); + x = gamma_16_from_1[(w & 0xff) >> gamma_shift][w >> 8]; + *(dp + 4) = (png_byte)((x >> 8) & 0xff); + *(dp + 5) = (png_byte)(x & 0xff); + } + } + } + else +#endif + { + sp = row; + dp = row; + for (i = 0; i < row_width; i++, sp += 8, dp += 6) + { + png_uint_16 a = (png_uint_16)(((png_uint_16)(*(sp + 6)) + << 8) + (png_uint_16)(*(sp + 7))); + if (a == (png_uint_16)0xffff) + { + png_memcpy(dp, sp, 6); + } + else if (a == 0) + { + *dp = (png_byte)((background->red >> 8) & 0xff); + *(dp + 1) = (png_byte)(background->red & 0xff); + *(dp + 2) = (png_byte)((background->green >> 8) & 0xff); + *(dp + 3) = (png_byte)(background->green & 0xff); + *(dp + 4) = (png_byte)((background->blue >> 8) & 0xff); + *(dp + 5) = (png_byte)(background->blue & 0xff); + } + else + { + png_uint_16 v; + + png_uint_16 r = (png_uint_16)(((*sp) << 8) + *(sp + 1)); + png_uint_16 g = (png_uint_16)(((*(sp + 2)) << 8) + + *(sp + 3)); + png_uint_16 b = (png_uint_16)(((*(sp + 4)) << 8) + + *(sp + 5)); + + png_composite_16(v, r, a, background->red); + *dp = (png_byte)((v >> 8) & 0xff); + *(dp + 1) = (png_byte)(v & 0xff); + png_composite_16(v, g, a, background->green); + *(dp + 2) = (png_byte)((v >> 8) & 0xff); + *(dp + 3) = (png_byte)(v & 0xff); + png_composite_16(v, b, a, background->blue); + *(dp + 4) = (png_byte)((v >> 8) & 0xff); + *(dp + 5) = (png_byte)(v & 0xff); + } + } + } + } + break; + } + } + + if (row_info->color_type & PNG_COLOR_MASK_ALPHA) + { + row_info->color_type &= ~PNG_COLOR_MASK_ALPHA; + row_info->channels--; + row_info->pixel_depth = (png_byte)(row_info->channels * + row_info->bit_depth); + row_info->rowbytes = PNG_ROWBYTES(row_info->pixel_depth,row_width); + } + } +} +#endif + +#if defined(PNG_READ_GAMMA_SUPPORTED) +/* Gamma correct the image, avoiding the alpha channel. Make sure + * you do this after you deal with the transparency issue on grayscale + * or RGB images. If your bit depth is 8, use gamma_table, if it + * is 16, use gamma_16_table and gamma_shift. Build these with + * build_gamma_table(). + */ +void /* PRIVATE */ +png_do_gamma(png_row_infop row_info, png_bytep row, + png_bytep gamma_table, png_uint_16pp gamma_16_table, + int gamma_shift) +{ + png_bytep sp; + png_uint_32 i; + png_uint_32 row_width=row_info->width; + + png_debug(1, "in png_do_gamma\n"); + if ( +#if defined(PNG_USELESS_TESTS_SUPPORTED) + row != NULL && row_info != NULL && +#endif + ((row_info->bit_depth <= 8 && gamma_table != NULL) || + (row_info->bit_depth == 16 && gamma_16_table != NULL))) + { + switch (row_info->color_type) + { + case PNG_COLOR_TYPE_RGB: + { + if (row_info->bit_depth == 8) + { + sp = row; + for (i = 0; i < row_width; i++) + { + *sp = gamma_table[*sp]; + sp++; + *sp = gamma_table[*sp]; + sp++; + *sp = gamma_table[*sp]; + sp++; + } + } + else /* if (row_info->bit_depth == 16) */ + { + sp = row; + for (i = 0; i < row_width; i++) + { + png_uint_16 v; + + v = gamma_16_table[*(sp + 1) >> gamma_shift][*sp]; + *sp = (png_byte)((v >> 8) & 0xff); + *(sp + 1) = (png_byte)(v & 0xff); + sp += 2; + v = gamma_16_table[*(sp + 1) >> gamma_shift][*sp]; + *sp = (png_byte)((v >> 8) & 0xff); + *(sp + 1) = (png_byte)(v & 0xff); + sp += 2; + v = gamma_16_table[*(sp + 1) >> gamma_shift][*sp]; + *sp = (png_byte)((v >> 8) & 0xff); + *(sp + 1) = (png_byte)(v & 0xff); + sp += 2; + } + } + break; + } + case PNG_COLOR_TYPE_RGB_ALPHA: + { + if (row_info->bit_depth == 8) + { + sp = row; + for (i = 0; i < row_width; i++) + { + *sp = gamma_table[*sp]; + sp++; + *sp = gamma_table[*sp]; + sp++; + *sp = gamma_table[*sp]; + sp++; + sp++; + } + } + else /* if (row_info->bit_depth == 16) */ + { + sp = row; + for (i = 0; i < row_width; i++) + { + png_uint_16 v = gamma_16_table[*(sp + 1) >> gamma_shift][*sp]; + *sp = (png_byte)((v >> 8) & 0xff); + *(sp + 1) = (png_byte)(v & 0xff); + sp += 2; + v = gamma_16_table[*(sp + 1) >> gamma_shift][*sp]; + *sp = (png_byte)((v >> 8) & 0xff); + *(sp + 1) = (png_byte)(v & 0xff); + sp += 2; + v = gamma_16_table[*(sp + 1) >> gamma_shift][*sp]; + *sp = (png_byte)((v >> 8) & 0xff); + *(sp + 1) = (png_byte)(v & 0xff); + sp += 4; + } + } + break; + } + case PNG_COLOR_TYPE_GRAY_ALPHA: + { + if (row_info->bit_depth == 8) + { + sp = row; + for (i = 0; i < row_width; i++) + { + *sp = gamma_table[*sp]; + sp += 2; + } + } + else /* if (row_info->bit_depth == 16) */ + { + sp = row; + for (i = 0; i < row_width; i++) + { + png_uint_16 v = gamma_16_table[*(sp + 1) >> gamma_shift][*sp]; + *sp = (png_byte)((v >> 8) & 0xff); + *(sp + 1) = (png_byte)(v & 0xff); + sp += 4; + } + } + break; + } + case PNG_COLOR_TYPE_GRAY: + { + if (row_info->bit_depth == 2) + { + sp = row; + for (i = 0; i < row_width; i += 4) + { + int a = *sp & 0xc0; + int b = *sp & 0x30; + int c = *sp & 0x0c; + int d = *sp & 0x03; + + *sp = (png_byte)( + ((((int)gamma_table[a|(a>>2)|(a>>4)|(a>>6)]) ) & 0xc0)| + ((((int)gamma_table[(b<<2)|b|(b>>2)|(b>>4)])>>2) & 0x30)| + ((((int)gamma_table[(c<<4)|(c<<2)|c|(c>>2)])>>4) & 0x0c)| + ((((int)gamma_table[(d<<6)|(d<<4)|(d<<2)|d])>>6) )); + sp++; + } + } + if (row_info->bit_depth == 4) + { + sp = row; + for (i = 0; i < row_width; i += 2) + { + int msb = *sp & 0xf0; + int lsb = *sp & 0x0f; + + *sp = (png_byte)((((int)gamma_table[msb | (msb >> 4)]) & 0xf0) + | (((int)gamma_table[(lsb << 4) | lsb]) >> 4)); + sp++; + } + } + else if (row_info->bit_depth == 8) + { + sp = row; + for (i = 0; i < row_width; i++) + { + *sp = gamma_table[*sp]; + sp++; + } + } + else if (row_info->bit_depth == 16) + { + sp = row; + for (i = 0; i < row_width; i++) + { + png_uint_16 v = gamma_16_table[*(sp + 1) >> gamma_shift][*sp]; + *sp = (png_byte)((v >> 8) & 0xff); + *(sp + 1) = (png_byte)(v & 0xff); + sp += 2; + } + } + break; + } + } + } +} +#endif + +#if defined(PNG_READ_EXPAND_SUPPORTED) +/* Expands a palette row to an RGB or RGBA row depending + * upon whether you supply trans and num_trans. + */ +void /* PRIVATE */ +png_do_expand_palette(png_row_infop row_info, png_bytep row, + png_colorp palette, png_bytep trans, int num_trans) +{ + int shift, value; + png_bytep sp, dp; + png_uint_32 i; + png_uint_32 row_width=row_info->width; + + png_debug(1, "in png_do_expand_palette\n"); + if ( +#if defined(PNG_USELESS_TESTS_SUPPORTED) + row != NULL && row_info != NULL && +#endif + row_info->color_type == PNG_COLOR_TYPE_PALETTE) + { + if (row_info->bit_depth < 8) + { + switch (row_info->bit_depth) + { + case 1: + { + sp = row + (png_size_t)((row_width - 1) >> 3); + dp = row + (png_size_t)row_width - 1; + shift = 7 - (int)((row_width + 7) & 0x07); + for (i = 0; i < row_width; i++) + { + if ((*sp >> shift) & 0x01) + *dp = 1; + else + *dp = 0; + if (shift == 7) + { + shift = 0; + sp--; + } + else + shift++; + + dp--; + } + break; + } + case 2: + { + sp = row + (png_size_t)((row_width - 1) >> 2); + dp = row + (png_size_t)row_width - 1; + shift = (int)((3 - ((row_width + 3) & 0x03)) << 1); + for (i = 0; i < row_width; i++) + { + value = (*sp >> shift) & 0x03; + *dp = (png_byte)value; + if (shift == 6) + { + shift = 0; + sp--; + } + else + shift += 2; + + dp--; + } + break; + } + case 4: + { + sp = row + (png_size_t)((row_width - 1) >> 1); + dp = row + (png_size_t)row_width - 1; + shift = (int)((row_width & 0x01) << 2); + for (i = 0; i < row_width; i++) + { + value = (*sp >> shift) & 0x0f; + *dp = (png_byte)value; + if (shift == 4) + { + shift = 0; + sp--; + } + else + shift += 4; + + dp--; + } + break; + } + } + row_info->bit_depth = 8; + row_info->pixel_depth = 8; + row_info->rowbytes = row_width; + } + switch (row_info->bit_depth) + { + case 8: + { + if (trans != NULL) + { + sp = row + (png_size_t)row_width - 1; + dp = row + (png_size_t)(row_width << 2) - 1; + + for (i = 0; i < row_width; i++) + { + if ((int)(*sp) >= num_trans) + *dp-- = 0xff; + else + *dp-- = trans[*sp]; + *dp-- = palette[*sp].blue; + *dp-- = palette[*sp].green; + *dp-- = palette[*sp].red; + sp--; + } + row_info->bit_depth = 8; + row_info->pixel_depth = 32; + row_info->rowbytes = row_width * 4; + row_info->color_type = 6; + row_info->channels = 4; + } + else + { + sp = row + (png_size_t)row_width - 1; + dp = row + (png_size_t)(row_width * 3) - 1; + + for (i = 0; i < row_width; i++) + { + *dp-- = palette[*sp].blue; + *dp-- = palette[*sp].green; + *dp-- = palette[*sp].red; + sp--; + } + row_info->bit_depth = 8; + row_info->pixel_depth = 24; + row_info->rowbytes = row_width * 3; + row_info->color_type = 2; + row_info->channels = 3; + } + break; + } + } + } +} + +/* If the bit depth < 8, it is expanded to 8. Also, if the already + * expanded transparency value is supplied, an alpha channel is built. + */ +void /* PRIVATE */ +png_do_expand(png_row_infop row_info, png_bytep row, + png_color_16p trans_value) +{ + int shift, value; + png_bytep sp, dp; + png_uint_32 i; + png_uint_32 row_width=row_info->width; + + png_debug(1, "in png_do_expand\n"); +#if defined(PNG_USELESS_TESTS_SUPPORTED) + if (row != NULL && row_info != NULL) +#endif + { + if (row_info->color_type == PNG_COLOR_TYPE_GRAY) + { + png_uint_16 gray = (png_uint_16)(trans_value ? trans_value->gray : 0); + + if (row_info->bit_depth < 8) + { + switch (row_info->bit_depth) + { + case 1: + { + gray = (png_uint_16)(gray*0xff); + sp = row + (png_size_t)((row_width - 1) >> 3); + dp = row + (png_size_t)row_width - 1; + shift = 7 - (int)((row_width + 7) & 0x07); + for (i = 0; i < row_width; i++) + { + if ((*sp >> shift) & 0x01) + *dp = 0xff; + else + *dp = 0; + if (shift == 7) + { + shift = 0; + sp--; + } + else + shift++; + + dp--; + } + break; + } + case 2: + { + gray = (png_uint_16)(gray*0x55); + sp = row + (png_size_t)((row_width - 1) >> 2); + dp = row + (png_size_t)row_width - 1; + shift = (int)((3 - ((row_width + 3) & 0x03)) << 1); + for (i = 0; i < row_width; i++) + { + value = (*sp >> shift) & 0x03; + *dp = (png_byte)(value | (value << 2) | (value << 4) | + (value << 6)); + if (shift == 6) + { + shift = 0; + sp--; + } + else + shift += 2; + + dp--; + } + break; + } + case 4: + { + gray = (png_uint_16)(gray*0x11); + sp = row + (png_size_t)((row_width - 1) >> 1); + dp = row + (png_size_t)row_width - 1; + shift = (int)((1 - ((row_width + 1) & 0x01)) << 2); + for (i = 0; i < row_width; i++) + { + value = (*sp >> shift) & 0x0f; + *dp = (png_byte)(value | (value << 4)); + if (shift == 4) + { + shift = 0; + sp--; + } + else + shift = 4; + + dp--; + } + break; + } + } + row_info->bit_depth = 8; + row_info->pixel_depth = 8; + row_info->rowbytes = row_width; + } + + if (trans_value != NULL) + { + if (row_info->bit_depth == 8) + { + sp = row + (png_size_t)row_width - 1; + dp = row + (png_size_t)(row_width << 1) - 1; + for (i = 0; i < row_width; i++) + { + if (*sp == gray) + *dp-- = 0; + else + *dp-- = 0xff; + *dp-- = *sp--; + } + } + else if (row_info->bit_depth == 16) + { + sp = row + row_info->rowbytes - 1; + dp = row + (row_info->rowbytes << 1) - 1; + for (i = 0; i < row_width; i++) + { + if (((png_uint_16)*(sp) | + ((png_uint_16)*(sp - 1) << 8)) == gray) + { + *dp-- = 0; + *dp-- = 0; + } + else + { + *dp-- = 0xff; + *dp-- = 0xff; + } + *dp-- = *sp--; + *dp-- = *sp--; + } + } + row_info->color_type = PNG_COLOR_TYPE_GRAY_ALPHA; + row_info->channels = 2; + row_info->pixel_depth = (png_byte)(row_info->bit_depth << 1); + row_info->rowbytes = PNG_ROWBYTES(row_info->pixel_depth, + row_width); + } + } + else if (row_info->color_type == PNG_COLOR_TYPE_RGB && trans_value) + { + if (row_info->bit_depth == 8) + { + sp = row + (png_size_t)row_info->rowbytes - 1; + dp = row + (png_size_t)(row_width << 2) - 1; + for (i = 0; i < row_width; i++) + { + if (*(sp - 2) == trans_value->red && + *(sp - 1) == trans_value->green && + *(sp - 0) == trans_value->blue) + *dp-- = 0; + else + *dp-- = 0xff; + *dp-- = *sp--; + *dp-- = *sp--; + *dp-- = *sp--; + } + } + else if (row_info->bit_depth == 16) + { + sp = row + row_info->rowbytes - 1; + dp = row + (png_size_t)(row_width << 3) - 1; + for (i = 0; i < row_width; i++) + { + if ((((png_uint_16)*(sp - 4) | + ((png_uint_16)*(sp - 5) << 8)) == trans_value->red) && + (((png_uint_16)*(sp - 2) | + ((png_uint_16)*(sp - 3) << 8)) == trans_value->green) && + (((png_uint_16)*(sp - 0) | + ((png_uint_16)*(sp - 1) << 8)) == trans_value->blue)) + { + *dp-- = 0; + *dp-- = 0; + } + else + { + *dp-- = 0xff; + *dp-- = 0xff; + } + *dp-- = *sp--; + *dp-- = *sp--; + *dp-- = *sp--; + *dp-- = *sp--; + *dp-- = *sp--; + *dp-- = *sp--; + } + } + row_info->color_type = PNG_COLOR_TYPE_RGB_ALPHA; + row_info->channels = 4; + row_info->pixel_depth = (png_byte)(row_info->bit_depth << 2); + row_info->rowbytes = PNG_ROWBYTES(row_info->pixel_depth,row_width); + } + } +} +#endif + +#if defined(PNG_READ_DITHER_SUPPORTED) +void /* PRIVATE */ +png_do_dither(png_row_infop row_info, png_bytep row, + png_bytep palette_lookup, png_bytep dither_lookup) +{ + png_bytep sp, dp; + png_uint_32 i; + png_uint_32 row_width=row_info->width; + + png_debug(1, "in png_do_dither\n"); +#if defined(PNG_USELESS_TESTS_SUPPORTED) + if (row != NULL && row_info != NULL) +#endif + { + if (row_info->color_type == PNG_COLOR_TYPE_RGB && + palette_lookup && row_info->bit_depth == 8) + { + int r, g, b, p; + sp = row; + dp = row; + for (i = 0; i < row_width; i++) + { + r = *sp++; + g = *sp++; + b = *sp++; + + /* this looks real messy, but the compiler will reduce + it down to a reasonable formula. For example, with + 5 bits per color, we get: + p = (((r >> 3) & 0x1f) << 10) | + (((g >> 3) & 0x1f) << 5) | + ((b >> 3) & 0x1f); + */ + p = (((r >> (8 - PNG_DITHER_RED_BITS)) & + ((1 << PNG_DITHER_RED_BITS) - 1)) << + (PNG_DITHER_GREEN_BITS + PNG_DITHER_BLUE_BITS)) | + (((g >> (8 - PNG_DITHER_GREEN_BITS)) & + ((1 << PNG_DITHER_GREEN_BITS) - 1)) << + (PNG_DITHER_BLUE_BITS)) | + ((b >> (8 - PNG_DITHER_BLUE_BITS)) & + ((1 << PNG_DITHER_BLUE_BITS) - 1)); + + *dp++ = palette_lookup[p]; + } + row_info->color_type = PNG_COLOR_TYPE_PALETTE; + row_info->channels = 1; + row_info->pixel_depth = row_info->bit_depth; + row_info->rowbytes = PNG_ROWBYTES(row_info->pixel_depth,row_width); + } + else if (row_info->color_type == PNG_COLOR_TYPE_RGB_ALPHA && + palette_lookup != NULL && row_info->bit_depth == 8) + { + int r, g, b, p; + sp = row; + dp = row; + for (i = 0; i < row_width; i++) + { + r = *sp++; + g = *sp++; + b = *sp++; + sp++; + + p = (((r >> (8 - PNG_DITHER_RED_BITS)) & + ((1 << PNG_DITHER_RED_BITS) - 1)) << + (PNG_DITHER_GREEN_BITS + PNG_DITHER_BLUE_BITS)) | + (((g >> (8 - PNG_DITHER_GREEN_BITS)) & + ((1 << PNG_DITHER_GREEN_BITS) - 1)) << + (PNG_DITHER_BLUE_BITS)) | + ((b >> (8 - PNG_DITHER_BLUE_BITS)) & + ((1 << PNG_DITHER_BLUE_BITS) - 1)); + + *dp++ = palette_lookup[p]; + } + row_info->color_type = PNG_COLOR_TYPE_PALETTE; + row_info->channels = 1; + row_info->pixel_depth = row_info->bit_depth; + row_info->rowbytes = PNG_ROWBYTES(row_info->pixel_depth,row_width); + } + else if (row_info->color_type == PNG_COLOR_TYPE_PALETTE && + dither_lookup && row_info->bit_depth == 8) + { + sp = row; + for (i = 0; i < row_width; i++, sp++) + { + *sp = dither_lookup[*sp]; + } + } + } +} +#endif + +#ifdef PNG_FLOATING_POINT_SUPPORTED +#if defined(PNG_READ_GAMMA_SUPPORTED) +static int png_gamma_shift[] = + {0x10, 0x21, 0x42, 0x84, 0x110, 0x248, 0x550, 0xff0}; + +/* We build the 8- or 16-bit gamma tables here. Note that for 16-bit + * tables, we don't make a full table if we are reducing to 8-bit in + * the future. Note also how the gamma_16 tables are segmented so that + * we don't need to allocate > 64K chunks for a full 16-bit table. + */ +void /* PRIVATE */ +png_build_gamma_table(png_structp png_ptr) +{ + png_debug(1, "in png_build_gamma_table\n"); + + if (png_ptr->bit_depth <= 8) + { + int i; + double g; + + if (png_ptr->screen_gamma > .000001) + g = 1.0 / (png_ptr->gamma * png_ptr->screen_gamma); + else + g = 1.0; + + png_ptr->gamma_table = (png_bytep)png_malloc(png_ptr, + (png_uint_32)256); + + for (i = 0; i < 256; i++) + { + png_ptr->gamma_table[i] = (png_byte)(pow((double)i / 255.0, + g) * 255.0 + .5); + } + +#if defined(PNG_READ_BACKGROUND_SUPPORTED) || \ + defined(PNG_READ_RGB_TO_GRAY_SUPPORTED) + if (png_ptr->transformations & ((PNG_BACKGROUND) | PNG_RGB_TO_GRAY)) + { + + g = 1.0 / (png_ptr->gamma); + + png_ptr->gamma_to_1 = (png_bytep)png_malloc(png_ptr, + (png_uint_32)256); + + for (i = 0; i < 256; i++) + { + png_ptr->gamma_to_1[i] = (png_byte)(pow((double)i / 255.0, + g) * 255.0 + .5); + } + + + png_ptr->gamma_from_1 = (png_bytep)png_malloc(png_ptr, + (png_uint_32)256); + + if(png_ptr->screen_gamma > 0.000001) + g = 1.0 / png_ptr->screen_gamma; + else + g = png_ptr->gamma; /* probably doing rgb_to_gray */ + + for (i = 0; i < 256; i++) + { + png_ptr->gamma_from_1[i] = (png_byte)(pow((double)i / 255.0, + g) * 255.0 + .5); + + } + } +#endif /* PNG_READ_BACKGROUND_SUPPORTED || PNG_RGB_TO_GRAY_SUPPORTED */ + } + else + { + double g; + int i, j, shift, num; + int sig_bit; + png_uint_32 ig; + + if (png_ptr->color_type & PNG_COLOR_MASK_COLOR) + { + sig_bit = (int)png_ptr->sig_bit.red; + if ((int)png_ptr->sig_bit.green > sig_bit) + sig_bit = png_ptr->sig_bit.green; + if ((int)png_ptr->sig_bit.blue > sig_bit) + sig_bit = png_ptr->sig_bit.blue; + } + else + { + sig_bit = (int)png_ptr->sig_bit.gray; + } + + if (sig_bit > 0) + shift = 16 - sig_bit; + else + shift = 0; + + if (png_ptr->transformations & PNG_16_TO_8) + { + if (shift < (16 - PNG_MAX_GAMMA_8)) + shift = (16 - PNG_MAX_GAMMA_8); + } + + if (shift > 8) + shift = 8; + if (shift < 0) + shift = 0; + + png_ptr->gamma_shift = (png_byte)shift; + + num = (1 << (8 - shift)); + + if (png_ptr->screen_gamma > .000001) + g = 1.0 / (png_ptr->gamma * png_ptr->screen_gamma); + else + g = 1.0; + + png_ptr->gamma_16_table = (png_uint_16pp)png_malloc(png_ptr, + (png_uint_32)(num * png_sizeof (png_uint_16p))); + + if (png_ptr->transformations & (PNG_16_TO_8 | PNG_BACKGROUND)) + { + double fin, fout; + png_uint_32 last, max; + + for (i = 0; i < num; i++) + { + png_ptr->gamma_16_table[i] = (png_uint_16p)png_malloc(png_ptr, + (png_uint_32)(256 * png_sizeof (png_uint_16))); + } + + g = 1.0 / g; + last = 0; + for (i = 0; i < 256; i++) + { + fout = ((double)i + 0.5) / 256.0; + fin = pow(fout, g); + max = (png_uint_32)(fin * (double)((png_uint_32)num << 8)); + while (last <= max) + { + png_ptr->gamma_16_table[(int)(last & (0xff >> shift))] + [(int)(last >> (8 - shift))] = (png_uint_16)( + (png_uint_16)i | ((png_uint_16)i << 8)); + last++; + } + } + while (last < ((png_uint_32)num << 8)) + { + png_ptr->gamma_16_table[(int)(last & (0xff >> shift))] + [(int)(last >> (8 - shift))] = (png_uint_16)65535L; + last++; + } + } + else + { + for (i = 0; i < num; i++) + { + png_ptr->gamma_16_table[i] = (png_uint_16p)png_malloc(png_ptr, + (png_uint_32)(256 * png_sizeof (png_uint_16))); + + ig = (((png_uint_32)i * (png_uint_32)png_gamma_shift[shift]) >> 4); + for (j = 0; j < 256; j++) + { + png_ptr->gamma_16_table[i][j] = + (png_uint_16)(pow((double)(ig + ((png_uint_32)j << 8)) / + 65535.0, g) * 65535.0 + .5); + } + } + } + +#if defined(PNG_READ_BACKGROUND_SUPPORTED) || \ + defined(PNG_READ_RGB_TO_GRAY_SUPPORTED) + if (png_ptr->transformations & (PNG_BACKGROUND | PNG_RGB_TO_GRAY)) + { + + g = 1.0 / (png_ptr->gamma); + + png_ptr->gamma_16_to_1 = (png_uint_16pp)png_malloc(png_ptr, + (png_uint_32)(num * png_sizeof (png_uint_16p ))); + + for (i = 0; i < num; i++) + { + png_ptr->gamma_16_to_1[i] = (png_uint_16p)png_malloc(png_ptr, + (png_uint_32)(256 * png_sizeof (png_uint_16))); + + ig = (((png_uint_32)i * + (png_uint_32)png_gamma_shift[shift]) >> 4); + for (j = 0; j < 256; j++) + { + png_ptr->gamma_16_to_1[i][j] = + (png_uint_16)(pow((double)(ig + ((png_uint_32)j << 8)) / + 65535.0, g) * 65535.0 + .5); + } + } + + if(png_ptr->screen_gamma > 0.000001) + g = 1.0 / png_ptr->screen_gamma; + else + g = png_ptr->gamma; /* probably doing rgb_to_gray */ + + png_ptr->gamma_16_from_1 = (png_uint_16pp)png_malloc(png_ptr, + (png_uint_32)(num * png_sizeof (png_uint_16p))); + + for (i = 0; i < num; i++) + { + png_ptr->gamma_16_from_1[i] = (png_uint_16p)png_malloc(png_ptr, + (png_uint_32)(256 * png_sizeof (png_uint_16))); + + ig = (((png_uint_32)i * + (png_uint_32)png_gamma_shift[shift]) >> 4); + for (j = 0; j < 256; j++) + { + png_ptr->gamma_16_from_1[i][j] = + (png_uint_16)(pow((double)(ig + ((png_uint_32)j << 8)) / + 65535.0, g) * 65535.0 + .5); + } + } + } +#endif /* PNG_READ_BACKGROUND_SUPPORTED || PNG_RGB_TO_GRAY_SUPPORTED */ + } +} +#endif +/* To do: install integer version of png_build_gamma_table here */ +#endif + +#if defined(PNG_MNG_FEATURES_SUPPORTED) +/* undoes intrapixel differencing */ +void /* PRIVATE */ +png_do_read_intrapixel(png_row_infop row_info, png_bytep row) +{ + png_debug(1, "in png_do_read_intrapixel\n"); + if ( +#if defined(PNG_USELESS_TESTS_SUPPORTED) + row != NULL && row_info != NULL && +#endif + (row_info->color_type & PNG_COLOR_MASK_COLOR)) + { + int bytes_per_pixel; + png_uint_32 row_width = row_info->width; + if (row_info->bit_depth == 8) + { + png_bytep rp; + png_uint_32 i; + + if (row_info->color_type == PNG_COLOR_TYPE_RGB) + bytes_per_pixel = 3; + else if (row_info->color_type == PNG_COLOR_TYPE_RGB_ALPHA) + bytes_per_pixel = 4; + else + return; + + for (i = 0, rp = row; i < row_width; i++, rp += bytes_per_pixel) + { + *(rp) = (png_byte)((256 + *rp + *(rp+1))&0xff); + *(rp+2) = (png_byte)((256 + *(rp+2) + *(rp+1))&0xff); + } + } + else if (row_info->bit_depth == 16) + { + png_bytep rp; + png_uint_32 i; + + if (row_info->color_type == PNG_COLOR_TYPE_RGB) + bytes_per_pixel = 6; + else if (row_info->color_type == PNG_COLOR_TYPE_RGB_ALPHA) + bytes_per_pixel = 8; + else + return; + + for (i = 0, rp = row; i < row_width; i++, rp += bytes_per_pixel) + { + png_uint_32 s0 = (*(rp ) << 8) | *(rp+1); + png_uint_32 s1 = (*(rp+2) << 8) | *(rp+3); + png_uint_32 s2 = (*(rp+4) << 8) | *(rp+5); + png_uint_32 red = (png_uint_32)((s0+s1+65536L) & 0xffffL); + png_uint_32 blue = (png_uint_32)((s2+s1+65536L) & 0xffffL); + *(rp ) = (png_byte)((red >> 8) & 0xff); + *(rp+1) = (png_byte)(red & 0xff); + *(rp+4) = (png_byte)((blue >> 8) & 0xff); + *(rp+5) = (png_byte)(blue & 0xff); + } + } + } +} +#endif /* PNG_MNG_FEATURES_SUPPORTED */ +#endif /* PNG_READ_SUPPORTED */ diff --git a/src/win32/dependencies/libpng/pngrutil.c b/src/win32/dependencies/libpng/pngrutil.c new file mode 100644 index 00000000..a436d435 --- /dev/null +++ b/src/win32/dependencies/libpng/pngrutil.c @@ -0,0 +1,3124 @@ + +/* pngrutil.c - utilities to read a PNG file + * + * Last changed in libpng 1.2.9 April 14, 2006 + * For conditions of distribution and use, see copyright notice in png.h + * Copyright (c) 1998-2006 Glenn Randers-Pehrson + * (Version 0.96 Copyright (c) 1996, 1997 Andreas Dilger) + * (Version 0.88 Copyright (c) 1995, 1996 Guy Eric Schalnat, Group 42, Inc.) + * + * This file contains routines that are only called from within + * libpng itself during the course of reading an image. + */ + +#define PNG_INTERNAL +#include "png.h" + +#if defined(PNG_READ_SUPPORTED) + +#if defined(_WIN32_WCE) +/* strtod() function is not supported on WindowsCE */ +# ifdef PNG_FLOATING_POINT_SUPPORTED +__inline double strtod(const char *nptr, char **endptr) +{ + double result = 0; + int len; + wchar_t *str, *end; + + len = MultiByteToWideChar(CP_ACP, 0, nptr, -1, NULL, 0); + str = (wchar_t *)malloc(len * sizeof(wchar_t)); + if ( NULL != str ) + { + MultiByteToWideChar(CP_ACP, 0, nptr, -1, str, len); + result = wcstod(str, &end); + len = WideCharToMultiByte(CP_ACP, 0, end, -1, NULL, 0, NULL, NULL); + *endptr = (char *)nptr + (png_strlen(nptr) - len + 1); + free(str); + } + return result; +} +# endif +#endif + +png_uint_32 PNGAPI +png_get_uint_31(png_structp png_ptr, png_bytep buf) +{ + png_uint_32 i = png_get_uint_32(buf); + if (i > PNG_UINT_31_MAX) + png_error(png_ptr, "PNG unsigned integer out of range."); + return (i); +} +#ifndef PNG_READ_BIG_ENDIAN_SUPPORTED +/* Grab an unsigned 32-bit integer from a buffer in big-endian format. */ +png_uint_32 PNGAPI +png_get_uint_32(png_bytep buf) +{ + png_uint_32 i = ((png_uint_32)(*buf) << 24) + + ((png_uint_32)(*(buf + 1)) << 16) + + ((png_uint_32)(*(buf + 2)) << 8) + + (png_uint_32)(*(buf + 3)); + + return (i); +} + +/* Grab a signed 32-bit integer from a buffer in big-endian format. The + * data is stored in the PNG file in two's complement format, and it is + * assumed that the machine format for signed integers is the same. */ +png_int_32 PNGAPI +png_get_int_32(png_bytep buf) +{ + png_int_32 i = ((png_int_32)(*buf) << 24) + + ((png_int_32)(*(buf + 1)) << 16) + + ((png_int_32)(*(buf + 2)) << 8) + + (png_int_32)(*(buf + 3)); + + return (i); +} + +/* Grab an unsigned 16-bit integer from a buffer in big-endian format. */ +png_uint_16 PNGAPI +png_get_uint_16(png_bytep buf) +{ + png_uint_16 i = (png_uint_16)(((png_uint_16)(*buf) << 8) + + (png_uint_16)(*(buf + 1))); + + return (i); +} +#endif /* PNG_READ_BIG_ENDIAN_SUPPORTED */ + +/* Read data, and (optionally) run it through the CRC. */ +void /* PRIVATE */ +png_crc_read(png_structp png_ptr, png_bytep buf, png_size_t length) +{ + png_read_data(png_ptr, buf, length); + png_calculate_crc(png_ptr, buf, length); +} + +/* Optionally skip data and then check the CRC. Depending on whether we + are reading a ancillary or critical chunk, and how the program has set + things up, we may calculate the CRC on the data and print a message. + Returns '1' if there was a CRC error, '0' otherwise. */ +int /* PRIVATE */ +png_crc_finish(png_structp png_ptr, png_uint_32 skip) +{ + png_size_t i; + png_size_t istop = png_ptr->zbuf_size; + + for (i = (png_size_t)skip; i > istop; i -= istop) + { + png_crc_read(png_ptr, png_ptr->zbuf, png_ptr->zbuf_size); + } + if (i) + { + png_crc_read(png_ptr, png_ptr->zbuf, i); + } + + if (png_crc_error(png_ptr)) + { + if (((png_ptr->chunk_name[0] & 0x20) && /* Ancillary */ + !(png_ptr->flags & PNG_FLAG_CRC_ANCILLARY_NOWARN)) || + (!(png_ptr->chunk_name[0] & 0x20) && /* Critical */ + (png_ptr->flags & PNG_FLAG_CRC_CRITICAL_USE))) + { + png_chunk_warning(png_ptr, "CRC error"); + } + else + { + png_chunk_error(png_ptr, "CRC error"); + } + return (1); + } + + return (0); +} + +/* Compare the CRC stored in the PNG file with that calculated by libpng from + the data it has read thus far. */ +int /* PRIVATE */ +png_crc_error(png_structp png_ptr) +{ + png_byte crc_bytes[4]; + png_uint_32 crc; + int need_crc = 1; + + if (png_ptr->chunk_name[0] & 0x20) /* ancillary */ + { + if ((png_ptr->flags & PNG_FLAG_CRC_ANCILLARY_MASK) == + (PNG_FLAG_CRC_ANCILLARY_USE | PNG_FLAG_CRC_ANCILLARY_NOWARN)) + need_crc = 0; + } + else /* critical */ + { + if (png_ptr->flags & PNG_FLAG_CRC_CRITICAL_IGNORE) + need_crc = 0; + } + + png_read_data(png_ptr, crc_bytes, 4); + + if (need_crc) + { + crc = png_get_uint_32(crc_bytes); + return ((int)(crc != png_ptr->crc)); + } + else + return (0); +} + +#if defined(PNG_READ_zTXt_SUPPORTED) || defined(PNG_READ_iTXt_SUPPORTED) || \ + defined(PNG_READ_iCCP_SUPPORTED) +/* + * Decompress trailing data in a chunk. The assumption is that chunkdata + * points at an allocated area holding the contents of a chunk with a + * trailing compressed part. What we get back is an allocated area + * holding the original prefix part and an uncompressed version of the + * trailing part (the malloc area passed in is freed). + */ +png_charp /* PRIVATE */ +png_decompress_chunk(png_structp png_ptr, int comp_type, + png_charp chunkdata, png_size_t chunklength, + png_size_t prefix_size, png_size_t *newlength) +{ + static char msg[] = "Error decoding compressed text"; + png_charp text; + png_size_t text_size; + + if (comp_type == PNG_COMPRESSION_TYPE_BASE) + { + int ret = Z_OK; + png_ptr->zstream.next_in = (png_bytep)(chunkdata + prefix_size); + png_ptr->zstream.avail_in = (uInt)(chunklength - prefix_size); + png_ptr->zstream.next_out = png_ptr->zbuf; + png_ptr->zstream.avail_out = (uInt)png_ptr->zbuf_size; + + text_size = 0; + text = NULL; + + while (png_ptr->zstream.avail_in) + { + ret = inflate(&png_ptr->zstream, Z_PARTIAL_FLUSH); + if (ret != Z_OK && ret != Z_STREAM_END) + { + if (png_ptr->zstream.msg != NULL) + png_warning(png_ptr, png_ptr->zstream.msg); + else + png_warning(png_ptr, msg); + inflateReset(&png_ptr->zstream); + png_ptr->zstream.avail_in = 0; + + if (text == NULL) + { + text_size = prefix_size + png_sizeof(msg) + 1; + text = (png_charp)png_malloc_warn(png_ptr, text_size); + if (text == NULL) + { + png_free(png_ptr,chunkdata); + png_error(png_ptr,"Not enough memory to decompress chunk"); + } + png_memcpy(text, chunkdata, prefix_size); + } + + text[text_size - 1] = 0x00; + + /* Copy what we can of the error message into the text chunk */ + text_size = (png_size_t)(chunklength - (text - chunkdata) - 1); + text_size = png_sizeof(msg) > text_size ? text_size : + png_sizeof(msg); + png_memcpy(text + prefix_size, msg, text_size + 1); + break; + } + if (!png_ptr->zstream.avail_out || ret == Z_STREAM_END) + { + if (text == NULL) + { + text_size = prefix_size + + png_ptr->zbuf_size - png_ptr->zstream.avail_out; + text = (png_charp)png_malloc_warn(png_ptr, text_size + 1); + if (text == NULL) + { + png_free(png_ptr,chunkdata); + png_error(png_ptr,"Not enough memory to decompress chunk."); + } + png_memcpy(text + prefix_size, png_ptr->zbuf, + text_size - prefix_size); + png_memcpy(text, chunkdata, prefix_size); + *(text + text_size) = 0x00; + } + else + { + png_charp tmp; + + tmp = text; + text = (png_charp)png_malloc_warn(png_ptr, + (png_uint_32)(text_size + + png_ptr->zbuf_size - png_ptr->zstream.avail_out + 1)); + if (text == NULL) + { + png_free(png_ptr, tmp); + png_free(png_ptr, chunkdata); + png_error(png_ptr,"Not enough memory to decompress chunk.."); + } + png_memcpy(text, tmp, text_size); + png_free(png_ptr, tmp); + png_memcpy(text + text_size, png_ptr->zbuf, + (png_ptr->zbuf_size - png_ptr->zstream.avail_out)); + text_size += png_ptr->zbuf_size - png_ptr->zstream.avail_out; + *(text + text_size) = 0x00; + } + if (ret == Z_STREAM_END) + break; + else + { + png_ptr->zstream.next_out = png_ptr->zbuf; + png_ptr->zstream.avail_out = (uInt)png_ptr->zbuf_size; + } + } + } + if (ret != Z_STREAM_END) + { +#if !defined(PNG_NO_STDIO) && !defined(_WIN32_WCE) + char umsg[50]; + + if (ret == Z_BUF_ERROR) + sprintf(umsg,"Buffer error in compressed datastream in %s chunk", + png_ptr->chunk_name); + else if (ret == Z_DATA_ERROR) + sprintf(umsg,"Data error in compressed datastream in %s chunk", + png_ptr->chunk_name); + else + sprintf(umsg,"Incomplete compressed datastream in %s chunk", + png_ptr->chunk_name); + png_warning(png_ptr, umsg); +#else + png_warning(png_ptr, + "Incomplete compressed datastream in chunk other than IDAT"); +#endif + text_size=prefix_size; + if (text == NULL) + { + text = (png_charp)png_malloc_warn(png_ptr, text_size+1); + if (text == NULL) + { + png_free(png_ptr, chunkdata); + png_error(png_ptr,"Not enough memory for text."); + } + png_memcpy(text, chunkdata, prefix_size); + } + *(text + text_size) = 0x00; + } + + inflateReset(&png_ptr->zstream); + png_ptr->zstream.avail_in = 0; + + png_free(png_ptr, chunkdata); + chunkdata = text; + *newlength=text_size; + } + else /* if (comp_type != PNG_COMPRESSION_TYPE_BASE) */ + { +#if !defined(PNG_NO_STDIO) && !defined(_WIN32_WCE) + char umsg[50]; + + sprintf(umsg, "Unknown zTXt compression type %d", comp_type); + png_warning(png_ptr, umsg); +#else + png_warning(png_ptr, "Unknown zTXt compression type"); +#endif + + *(chunkdata + prefix_size) = 0x00; + *newlength=prefix_size; + } + + return chunkdata; +} +#endif + +/* read and check the IDHR chunk */ +void /* PRIVATE */ +png_handle_IHDR(png_structp png_ptr, png_infop info_ptr, png_uint_32 length) +{ + png_byte buf[13]; + png_uint_32 width, height; + int bit_depth, color_type, compression_type, filter_type; + int interlace_type; + + png_debug(1, "in png_handle_IHDR\n"); + + if (png_ptr->mode & PNG_HAVE_IHDR) + png_error(png_ptr, "Out of place IHDR"); + + /* check the length */ + if (length != 13) + png_error(png_ptr, "Invalid IHDR chunk"); + + png_ptr->mode |= PNG_HAVE_IHDR; + + png_crc_read(png_ptr, buf, 13); + png_crc_finish(png_ptr, 0); + + width = png_get_uint_31(png_ptr, buf); + height = png_get_uint_31(png_ptr, buf + 4); + bit_depth = buf[8]; + color_type = buf[9]; + compression_type = buf[10]; + filter_type = buf[11]; + interlace_type = buf[12]; + + /* set internal variables */ + png_ptr->width = width; + png_ptr->height = height; + png_ptr->bit_depth = (png_byte)bit_depth; + png_ptr->interlaced = (png_byte)interlace_type; + png_ptr->color_type = (png_byte)color_type; +#if defined(PNG_MNG_FEATURES_SUPPORTED) + png_ptr->filter_type = (png_byte)filter_type; +#endif + png_ptr->compression_type = (png_byte)compression_type; + + /* find number of channels */ + switch (png_ptr->color_type) + { + case PNG_COLOR_TYPE_GRAY: + case PNG_COLOR_TYPE_PALETTE: + png_ptr->channels = 1; + break; + case PNG_COLOR_TYPE_RGB: + png_ptr->channels = 3; + break; + case PNG_COLOR_TYPE_GRAY_ALPHA: + png_ptr->channels = 2; + break; + case PNG_COLOR_TYPE_RGB_ALPHA: + png_ptr->channels = 4; + break; + } + + /* set up other useful info */ + png_ptr->pixel_depth = (png_byte)(png_ptr->bit_depth * + png_ptr->channels); + png_ptr->rowbytes = PNG_ROWBYTES(png_ptr->pixel_depth,png_ptr->width); + png_debug1(3,"bit_depth = %d\n", png_ptr->bit_depth); + png_debug1(3,"channels = %d\n", png_ptr->channels); + png_debug1(3,"rowbytes = %lu\n", png_ptr->rowbytes); + png_set_IHDR(png_ptr, info_ptr, width, height, bit_depth, + color_type, interlace_type, compression_type, filter_type); +} + +/* read and check the palette */ +void /* PRIVATE */ +png_handle_PLTE(png_structp png_ptr, png_infop info_ptr, png_uint_32 length) +{ + png_color palette[PNG_MAX_PALETTE_LENGTH]; + int num, i; +#ifndef PNG_NO_POINTER_INDEXING + png_colorp pal_ptr; +#endif + + png_debug(1, "in png_handle_PLTE\n"); + + if (!(png_ptr->mode & PNG_HAVE_IHDR)) + png_error(png_ptr, "Missing IHDR before PLTE"); + else if (png_ptr->mode & PNG_HAVE_IDAT) + { + png_warning(png_ptr, "Invalid PLTE after IDAT"); + png_crc_finish(png_ptr, length); + return; + } + else if (png_ptr->mode & PNG_HAVE_PLTE) + png_error(png_ptr, "Duplicate PLTE chunk"); + + png_ptr->mode |= PNG_HAVE_PLTE; + + if (!(png_ptr->color_type&PNG_COLOR_MASK_COLOR)) + { + png_warning(png_ptr, + "Ignoring PLTE chunk in grayscale PNG"); + png_crc_finish(png_ptr, length); + return; + } +#if !defined(PNG_READ_OPT_PLTE_SUPPORTED) + if (png_ptr->color_type != PNG_COLOR_TYPE_PALETTE) + { + png_crc_finish(png_ptr, length); + return; + } +#endif + + if (length > 3*PNG_MAX_PALETTE_LENGTH || length % 3) + { + if (png_ptr->color_type != PNG_COLOR_TYPE_PALETTE) + { + png_warning(png_ptr, "Invalid palette chunk"); + png_crc_finish(png_ptr, length); + return; + } + else + { + png_error(png_ptr, "Invalid palette chunk"); + } + } + + num = (int)length / 3; + +#ifndef PNG_NO_POINTER_INDEXING + for (i = 0, pal_ptr = palette; i < num; i++, pal_ptr++) + { + png_byte buf[3]; + + png_crc_read(png_ptr, buf, 3); + pal_ptr->red = buf[0]; + pal_ptr->green = buf[1]; + pal_ptr->blue = buf[2]; + } +#else + for (i = 0; i < num; i++) + { + png_byte buf[3]; + + png_crc_read(png_ptr, buf, 3); + /* don't depend upon png_color being any order */ + palette[i].red = buf[0]; + palette[i].green = buf[1]; + palette[i].blue = buf[2]; + } +#endif + + /* If we actually NEED the PLTE chunk (ie for a paletted image), we do + whatever the normal CRC configuration tells us. However, if we + have an RGB image, the PLTE can be considered ancillary, so + we will act as though it is. */ +#if !defined(PNG_READ_OPT_PLTE_SUPPORTED) + if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE) +#endif + { + png_crc_finish(png_ptr, 0); + } +#if !defined(PNG_READ_OPT_PLTE_SUPPORTED) + else if (png_crc_error(png_ptr)) /* Only if we have a CRC error */ + { + /* If we don't want to use the data from an ancillary chunk, + we have two options: an error abort, or a warning and we + ignore the data in this chunk (which should be OK, since + it's considered ancillary for a RGB or RGBA image). */ + if (!(png_ptr->flags & PNG_FLAG_CRC_ANCILLARY_USE)) + { + if (png_ptr->flags & PNG_FLAG_CRC_ANCILLARY_NOWARN) + { + png_chunk_error(png_ptr, "CRC error"); + } + else + { + png_chunk_warning(png_ptr, "CRC error"); + return; + } + } + /* Otherwise, we (optionally) emit a warning and use the chunk. */ + else if (!(png_ptr->flags & PNG_FLAG_CRC_ANCILLARY_NOWARN)) + { + png_chunk_warning(png_ptr, "CRC error"); + } + } +#endif + + png_set_PLTE(png_ptr, info_ptr, palette, num); + +#if defined(PNG_READ_tRNS_SUPPORTED) + if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE) + { + if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_tRNS)) + { + if (png_ptr->num_trans > (png_uint_16)num) + { + png_warning(png_ptr, "Truncating incorrect tRNS chunk length"); + png_ptr->num_trans = (png_uint_16)num; + } + if (info_ptr->num_trans > (png_uint_16)num) + { + png_warning(png_ptr, "Truncating incorrect info tRNS chunk length"); + info_ptr->num_trans = (png_uint_16)num; + } + } + } +#endif + +} + +void /* PRIVATE */ +png_handle_IEND(png_structp png_ptr, png_infop info_ptr, png_uint_32 length) +{ + png_debug(1, "in png_handle_IEND\n"); + + if (!(png_ptr->mode & PNG_HAVE_IHDR) || !(png_ptr->mode & PNG_HAVE_IDAT)) + { + png_error(png_ptr, "No image in file"); + } + + png_ptr->mode |= (PNG_AFTER_IDAT | PNG_HAVE_IEND); + + if (length != 0) + { + png_warning(png_ptr, "Incorrect IEND chunk length"); + } + png_crc_finish(png_ptr, length); + + if (&info_ptr == NULL) /* quiet compiler warnings about unused info_ptr */ + return; +} + +#if defined(PNG_READ_gAMA_SUPPORTED) +void /* PRIVATE */ +png_handle_gAMA(png_structp png_ptr, png_infop info_ptr, png_uint_32 length) +{ + png_fixed_point igamma; +#ifdef PNG_FLOATING_POINT_SUPPORTED + float file_gamma; +#endif + png_byte buf[4]; + + png_debug(1, "in png_handle_gAMA\n"); + + if (!(png_ptr->mode & PNG_HAVE_IHDR)) + png_error(png_ptr, "Missing IHDR before gAMA"); + else if (png_ptr->mode & PNG_HAVE_IDAT) + { + png_warning(png_ptr, "Invalid gAMA after IDAT"); + png_crc_finish(png_ptr, length); + return; + } + else if (png_ptr->mode & PNG_HAVE_PLTE) + /* Should be an error, but we can cope with it */ + png_warning(png_ptr, "Out of place gAMA chunk"); + + if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_gAMA) +#if defined(PNG_READ_sRGB_SUPPORTED) + && !(info_ptr->valid & PNG_INFO_sRGB) +#endif + ) + { + png_warning(png_ptr, "Duplicate gAMA chunk"); + png_crc_finish(png_ptr, length); + return; + } + + if (length != 4) + { + png_warning(png_ptr, "Incorrect gAMA chunk length"); + png_crc_finish(png_ptr, length); + return; + } + + png_crc_read(png_ptr, buf, 4); + if (png_crc_finish(png_ptr, 0)) + return; + + igamma = (png_fixed_point)png_get_uint_32(buf); + /* check for zero gamma */ + if (igamma == 0) + { + png_warning(png_ptr, + "Ignoring gAMA chunk with gamma=0"); + return; + } + +#if defined(PNG_READ_sRGB_SUPPORTED) + if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_sRGB)) + if (PNG_OUT_OF_RANGE(igamma, 45500L, 500)) + { + png_warning(png_ptr, + "Ignoring incorrect gAMA value when sRGB is also present"); +#ifndef PNG_NO_CONSOLE_IO + fprintf(stderr, "gamma = (%d/100000)\n", (int)igamma); +#endif + return; + } +#endif /* PNG_READ_sRGB_SUPPORTED */ + +#ifdef PNG_FLOATING_POINT_SUPPORTED + file_gamma = (float)igamma / (float)100000.0; +# ifdef PNG_READ_GAMMA_SUPPORTED + png_ptr->gamma = file_gamma; +# endif + png_set_gAMA(png_ptr, info_ptr, file_gamma); +#endif +#ifdef PNG_FIXED_POINT_SUPPORTED + png_set_gAMA_fixed(png_ptr, info_ptr, igamma); +#endif +} +#endif + +#if defined(PNG_READ_sBIT_SUPPORTED) +void /* PRIVATE */ +png_handle_sBIT(png_structp png_ptr, png_infop info_ptr, png_uint_32 length) +{ + png_size_t truelen; + png_byte buf[4]; + + png_debug(1, "in png_handle_sBIT\n"); + + buf[0] = buf[1] = buf[2] = buf[3] = 0; + + if (!(png_ptr->mode & PNG_HAVE_IHDR)) + png_error(png_ptr, "Missing IHDR before sBIT"); + else if (png_ptr->mode & PNG_HAVE_IDAT) + { + png_warning(png_ptr, "Invalid sBIT after IDAT"); + png_crc_finish(png_ptr, length); + return; + } + else if (png_ptr->mode & PNG_HAVE_PLTE) + { + /* Should be an error, but we can cope with it */ + png_warning(png_ptr, "Out of place sBIT chunk"); + } + if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_sBIT)) + { + png_warning(png_ptr, "Duplicate sBIT chunk"); + png_crc_finish(png_ptr, length); + return; + } + + if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE) + truelen = 3; + else + truelen = (png_size_t)png_ptr->channels; + + if (length != truelen || length > 4) + { + png_warning(png_ptr, "Incorrect sBIT chunk length"); + png_crc_finish(png_ptr, length); + return; + } + + png_crc_read(png_ptr, buf, truelen); + if (png_crc_finish(png_ptr, 0)) + return; + + if (png_ptr->color_type & PNG_COLOR_MASK_COLOR) + { + png_ptr->sig_bit.red = buf[0]; + png_ptr->sig_bit.green = buf[1]; + png_ptr->sig_bit.blue = buf[2]; + png_ptr->sig_bit.alpha = buf[3]; + } + else + { + png_ptr->sig_bit.gray = buf[0]; + png_ptr->sig_bit.red = buf[0]; + png_ptr->sig_bit.green = buf[0]; + png_ptr->sig_bit.blue = buf[0]; + png_ptr->sig_bit.alpha = buf[1]; + } + png_set_sBIT(png_ptr, info_ptr, &(png_ptr->sig_bit)); +} +#endif + +#if defined(PNG_READ_cHRM_SUPPORTED) +void /* PRIVATE */ +png_handle_cHRM(png_structp png_ptr, png_infop info_ptr, png_uint_32 length) +{ + png_byte buf[4]; +#ifdef PNG_FLOATING_POINT_SUPPORTED + float white_x, white_y, red_x, red_y, green_x, green_y, blue_x, blue_y; +#endif + png_fixed_point int_x_white, int_y_white, int_x_red, int_y_red, int_x_green, + int_y_green, int_x_blue, int_y_blue; + + png_uint_32 uint_x, uint_y; + + png_debug(1, "in png_handle_cHRM\n"); + + if (!(png_ptr->mode & PNG_HAVE_IHDR)) + png_error(png_ptr, "Missing IHDR before cHRM"); + else if (png_ptr->mode & PNG_HAVE_IDAT) + { + png_warning(png_ptr, "Invalid cHRM after IDAT"); + png_crc_finish(png_ptr, length); + return; + } + else if (png_ptr->mode & PNG_HAVE_PLTE) + /* Should be an error, but we can cope with it */ + png_warning(png_ptr, "Missing PLTE before cHRM"); + + if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_cHRM) +#if defined(PNG_READ_sRGB_SUPPORTED) + && !(info_ptr->valid & PNG_INFO_sRGB) +#endif + ) + { + png_warning(png_ptr, "Duplicate cHRM chunk"); + png_crc_finish(png_ptr, length); + return; + } + + if (length != 32) + { + png_warning(png_ptr, "Incorrect cHRM chunk length"); + png_crc_finish(png_ptr, length); + return; + } + + png_crc_read(png_ptr, buf, 4); + uint_x = png_get_uint_32(buf); + + png_crc_read(png_ptr, buf, 4); + uint_y = png_get_uint_32(buf); + + if (uint_x > 80000L || uint_y > 80000L || + uint_x + uint_y > 100000L) + { + png_warning(png_ptr, "Invalid cHRM white point"); + png_crc_finish(png_ptr, 24); + return; + } + int_x_white = (png_fixed_point)uint_x; + int_y_white = (png_fixed_point)uint_y; + + png_crc_read(png_ptr, buf, 4); + uint_x = png_get_uint_32(buf); + + png_crc_read(png_ptr, buf, 4); + uint_y = png_get_uint_32(buf); + + if (uint_x + uint_y > 100000L) + { + png_warning(png_ptr, "Invalid cHRM red point"); + png_crc_finish(png_ptr, 16); + return; + } + int_x_red = (png_fixed_point)uint_x; + int_y_red = (png_fixed_point)uint_y; + + png_crc_read(png_ptr, buf, 4); + uint_x = png_get_uint_32(buf); + + png_crc_read(png_ptr, buf, 4); + uint_y = png_get_uint_32(buf); + + if (uint_x + uint_y > 100000L) + { + png_warning(png_ptr, "Invalid cHRM green point"); + png_crc_finish(png_ptr, 8); + return; + } + int_x_green = (png_fixed_point)uint_x; + int_y_green = (png_fixed_point)uint_y; + + png_crc_read(png_ptr, buf, 4); + uint_x = png_get_uint_32(buf); + + png_crc_read(png_ptr, buf, 4); + uint_y = png_get_uint_32(buf); + + if (uint_x + uint_y > 100000L) + { + png_warning(png_ptr, "Invalid cHRM blue point"); + png_crc_finish(png_ptr, 0); + return; + } + int_x_blue = (png_fixed_point)uint_x; + int_y_blue = (png_fixed_point)uint_y; + +#ifdef PNG_FLOATING_POINT_SUPPORTED + white_x = (float)int_x_white / (float)100000.0; + white_y = (float)int_y_white / (float)100000.0; + red_x = (float)int_x_red / (float)100000.0; + red_y = (float)int_y_red / (float)100000.0; + green_x = (float)int_x_green / (float)100000.0; + green_y = (float)int_y_green / (float)100000.0; + blue_x = (float)int_x_blue / (float)100000.0; + blue_y = (float)int_y_blue / (float)100000.0; +#endif + +#if defined(PNG_READ_sRGB_SUPPORTED) + if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_sRGB)) + { + if (PNG_OUT_OF_RANGE(int_x_white, 31270, 1000) || + PNG_OUT_OF_RANGE(int_y_white, 32900, 1000) || + PNG_OUT_OF_RANGE(int_x_red, 64000L, 1000) || + PNG_OUT_OF_RANGE(int_y_red, 33000, 1000) || + PNG_OUT_OF_RANGE(int_x_green, 30000, 1000) || + PNG_OUT_OF_RANGE(int_y_green, 60000L, 1000) || + PNG_OUT_OF_RANGE(int_x_blue, 15000, 1000) || + PNG_OUT_OF_RANGE(int_y_blue, 6000, 1000)) + { + + png_warning(png_ptr, + "Ignoring incorrect cHRM value when sRGB is also present"); +#ifndef PNG_NO_CONSOLE_IO +#ifdef PNG_FLOATING_POINT_SUPPORTED + fprintf(stderr,"wx=%f, wy=%f, rx=%f, ry=%f\n", + white_x, white_y, red_x, red_y); + fprintf(stderr,"gx=%f, gy=%f, bx=%f, by=%f\n", + green_x, green_y, blue_x, blue_y); +#else + fprintf(stderr,"wx=%ld, wy=%ld, rx=%ld, ry=%ld\n", + int_x_white, int_y_white, int_x_red, int_y_red); + fprintf(stderr,"gx=%ld, gy=%ld, bx=%ld, by=%ld\n", + int_x_green, int_y_green, int_x_blue, int_y_blue); +#endif +#endif /* PNG_NO_CONSOLE_IO */ + } + png_crc_finish(png_ptr, 0); + return; + } +#endif /* PNG_READ_sRGB_SUPPORTED */ + +#ifdef PNG_FLOATING_POINT_SUPPORTED + png_set_cHRM(png_ptr, info_ptr, + white_x, white_y, red_x, red_y, green_x, green_y, blue_x, blue_y); +#endif +#ifdef PNG_FIXED_POINT_SUPPORTED + png_set_cHRM_fixed(png_ptr, info_ptr, + int_x_white, int_y_white, int_x_red, int_y_red, int_x_green, + int_y_green, int_x_blue, int_y_blue); +#endif + if (png_crc_finish(png_ptr, 0)) + return; +} +#endif + +#if defined(PNG_READ_sRGB_SUPPORTED) +void /* PRIVATE */ +png_handle_sRGB(png_structp png_ptr, png_infop info_ptr, png_uint_32 length) +{ + int intent; + png_byte buf[1]; + + png_debug(1, "in png_handle_sRGB\n"); + + if (!(png_ptr->mode & PNG_HAVE_IHDR)) + png_error(png_ptr, "Missing IHDR before sRGB"); + else if (png_ptr->mode & PNG_HAVE_IDAT) + { + png_warning(png_ptr, "Invalid sRGB after IDAT"); + png_crc_finish(png_ptr, length); + return; + } + else if (png_ptr->mode & PNG_HAVE_PLTE) + /* Should be an error, but we can cope with it */ + png_warning(png_ptr, "Out of place sRGB chunk"); + + if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_sRGB)) + { + png_warning(png_ptr, "Duplicate sRGB chunk"); + png_crc_finish(png_ptr, length); + return; + } + + if (length != 1) + { + png_warning(png_ptr, "Incorrect sRGB chunk length"); + png_crc_finish(png_ptr, length); + return; + } + + png_crc_read(png_ptr, buf, 1); + if (png_crc_finish(png_ptr, 0)) + return; + + intent = buf[0]; + /* check for bad intent */ + if (intent >= PNG_sRGB_INTENT_LAST) + { + png_warning(png_ptr, "Unknown sRGB intent"); + return; + } + +#if defined(PNG_READ_gAMA_SUPPORTED) && defined(PNG_READ_GAMMA_SUPPORTED) + if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_gAMA)) + { + png_fixed_point igamma; +#ifdef PNG_FIXED_POINT_SUPPORTED + igamma=info_ptr->int_gamma; +#else +# ifdef PNG_FLOATING_POINT_SUPPORTED + igamma=(png_fixed_point)(info_ptr->gamma * 100000.); +# endif +#endif + if (PNG_OUT_OF_RANGE(igamma, 45500L, 500)) + { + png_warning(png_ptr, + "Ignoring incorrect gAMA value when sRGB is also present"); +#ifndef PNG_NO_CONSOLE_IO +# ifdef PNG_FIXED_POINT_SUPPORTED + fprintf(stderr,"incorrect gamma=(%d/100000)\n",(int)png_ptr->int_gamma); +# else +# ifdef PNG_FLOATING_POINT_SUPPORTED + fprintf(stderr,"incorrect gamma=%f\n",png_ptr->gamma); +# endif +# endif +#endif + } + } +#endif /* PNG_READ_gAMA_SUPPORTED */ + +#ifdef PNG_READ_cHRM_SUPPORTED +#ifdef PNG_FIXED_POINT_SUPPORTED + if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_cHRM)) + if (PNG_OUT_OF_RANGE(info_ptr->int_x_white, 31270, 1000) || + PNG_OUT_OF_RANGE(info_ptr->int_y_white, 32900, 1000) || + PNG_OUT_OF_RANGE(info_ptr->int_x_red, 64000L, 1000) || + PNG_OUT_OF_RANGE(info_ptr->int_y_red, 33000, 1000) || + PNG_OUT_OF_RANGE(info_ptr->int_x_green, 30000, 1000) || + PNG_OUT_OF_RANGE(info_ptr->int_y_green, 60000L, 1000) || + PNG_OUT_OF_RANGE(info_ptr->int_x_blue, 15000, 1000) || + PNG_OUT_OF_RANGE(info_ptr->int_y_blue, 6000, 1000)) + { + png_warning(png_ptr, + "Ignoring incorrect cHRM value when sRGB is also present"); + } +#endif /* PNG_FIXED_POINT_SUPPORTED */ +#endif /* PNG_READ_cHRM_SUPPORTED */ + + png_set_sRGB_gAMA_and_cHRM(png_ptr, info_ptr, intent); +} +#endif /* PNG_READ_sRGB_SUPPORTED */ + +#if defined(PNG_READ_iCCP_SUPPORTED) +void /* PRIVATE */ +png_handle_iCCP(png_structp png_ptr, png_infop info_ptr, png_uint_32 length) +/* Note: this does not properly handle chunks that are > 64K under DOS */ +{ + png_charp chunkdata; + png_byte compression_type; + png_bytep pC; + png_charp profile; + png_uint_32 skip = 0; + png_uint_32 profile_size, profile_length; + png_size_t slength, prefix_length, data_length; + + png_debug(1, "in png_handle_iCCP\n"); + + if (!(png_ptr->mode & PNG_HAVE_IHDR)) + png_error(png_ptr, "Missing IHDR before iCCP"); + else if (png_ptr->mode & PNG_HAVE_IDAT) + { + png_warning(png_ptr, "Invalid iCCP after IDAT"); + png_crc_finish(png_ptr, length); + return; + } + else if (png_ptr->mode & PNG_HAVE_PLTE) + /* Should be an error, but we can cope with it */ + png_warning(png_ptr, "Out of place iCCP chunk"); + + if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_iCCP)) + { + png_warning(png_ptr, "Duplicate iCCP chunk"); + png_crc_finish(png_ptr, length); + return; + } + +#ifdef PNG_MAX_MALLOC_64K + if (length > (png_uint_32)65535L) + { + png_warning(png_ptr, "iCCP chunk too large to fit in memory"); + skip = length - (png_uint_32)65535L; + length = (png_uint_32)65535L; + } +#endif + + chunkdata = (png_charp)png_malloc(png_ptr, length + 1); + slength = (png_size_t)length; + png_crc_read(png_ptr, (png_bytep)chunkdata, slength); + + if (png_crc_finish(png_ptr, skip)) + { + png_free(png_ptr, chunkdata); + return; + } + + chunkdata[slength] = 0x00; + + for (profile = chunkdata; *profile; profile++) + /* empty loop to find end of name */ ; + + ++profile; + + /* there should be at least one zero (the compression type byte) + following the separator, and we should be on it */ + if ( profile >= chunkdata + slength) + { + png_free(png_ptr, chunkdata); + png_warning(png_ptr, "Malformed iCCP chunk"); + return; + } + + /* compression_type should always be zero */ + compression_type = *profile++; + if (compression_type) + { + png_warning(png_ptr, "Ignoring nonzero compression type in iCCP chunk"); + compression_type=0x00; /* Reset it to zero (libpng-1.0.6 through 1.0.8 + wrote nonzero) */ + } + + prefix_length = profile - chunkdata; + chunkdata = png_decompress_chunk(png_ptr, compression_type, chunkdata, + slength, prefix_length, &data_length); + + profile_length = data_length - prefix_length; + + if ( prefix_length > data_length || profile_length < 4) + { + png_free(png_ptr, chunkdata); + png_warning(png_ptr, "Profile size field missing from iCCP chunk"); + return; + } + + /* Check the profile_size recorded in the first 32 bits of the ICC profile */ + pC = (png_bytep)(chunkdata+prefix_length); + profile_size = ((*(pC ))<<24) | + ((*(pC+1))<<16) | + ((*(pC+2))<< 8) | + ((*(pC+3)) ); + + if(profile_size < profile_length) + profile_length = profile_size; + + if(profile_size > profile_length) + { + png_free(png_ptr, chunkdata); + png_warning(png_ptr, "Ignoring truncated iCCP profile."); + return; + } + + png_set_iCCP(png_ptr, info_ptr, chunkdata, compression_type, + chunkdata + prefix_length, profile_length); + png_free(png_ptr, chunkdata); +} +#endif /* PNG_READ_iCCP_SUPPORTED */ + +#if defined(PNG_READ_sPLT_SUPPORTED) +void /* PRIVATE */ +png_handle_sPLT(png_structp png_ptr, png_infop info_ptr, png_uint_32 length) +/* Note: this does not properly handle chunks that are > 64K under DOS */ +{ + png_bytep chunkdata; + png_bytep entry_start; + png_sPLT_t new_palette; +#ifdef PNG_NO_POINTER_INDEXING + png_sPLT_entryp pp; +#endif + int data_length, entry_size, i; + png_uint_32 skip = 0; + png_size_t slength; + + png_debug(1, "in png_handle_sPLT\n"); + + if (!(png_ptr->mode & PNG_HAVE_IHDR)) + png_error(png_ptr, "Missing IHDR before sPLT"); + else if (png_ptr->mode & PNG_HAVE_IDAT) + { + png_warning(png_ptr, "Invalid sPLT after IDAT"); + png_crc_finish(png_ptr, length); + return; + } + +#ifdef PNG_MAX_MALLOC_64K + if (length > (png_uint_32)65535L) + { + png_warning(png_ptr, "sPLT chunk too large to fit in memory"); + skip = length - (png_uint_32)65535L; + length = (png_uint_32)65535L; + } +#endif + + chunkdata = (png_bytep)png_malloc(png_ptr, length + 1); + slength = (png_size_t)length; + png_crc_read(png_ptr, (png_bytep)chunkdata, slength); + + if (png_crc_finish(png_ptr, skip)) + { + png_free(png_ptr, chunkdata); + return; + } + + chunkdata[slength] = 0x00; + + for (entry_start = chunkdata; *entry_start; entry_start++) + /* empty loop to find end of name */ ; + ++entry_start; + + /* a sample depth should follow the separator, and we should be on it */ + if (entry_start > chunkdata + slength) + { + png_free(png_ptr, chunkdata); + png_warning(png_ptr, "malformed sPLT chunk"); + return; + } + + new_palette.depth = *entry_start++; + entry_size = (new_palette.depth == 8 ? 6 : 10); + data_length = (slength - (entry_start - chunkdata)); + + /* integrity-check the data length */ + if (data_length % entry_size) + { + png_free(png_ptr, chunkdata); + png_warning(png_ptr, "sPLT chunk has bad length"); + return; + } + + new_palette.nentries = (png_int_32) ( data_length / entry_size); + if ((png_uint_32) new_palette.nentries > (png_uint_32) (PNG_SIZE_MAX / + png_sizeof(png_sPLT_entry))) + { + png_warning(png_ptr, "sPLT chunk too long"); + return; + } + new_palette.entries = (png_sPLT_entryp)png_malloc_warn( + png_ptr, new_palette.nentries * png_sizeof(png_sPLT_entry)); + if (new_palette.entries == NULL) + { + png_warning(png_ptr, "sPLT chunk requires too much memory"); + return; + } + +#ifndef PNG_NO_POINTER_INDEXING + for (i = 0; i < new_palette.nentries; i++) + { + png_sPLT_entryp pp = new_palette.entries + i; + + if (new_palette.depth == 8) + { + pp->red = *entry_start++; + pp->green = *entry_start++; + pp->blue = *entry_start++; + pp->alpha = *entry_start++; + } + else + { + pp->red = png_get_uint_16(entry_start); entry_start += 2; + pp->green = png_get_uint_16(entry_start); entry_start += 2; + pp->blue = png_get_uint_16(entry_start); entry_start += 2; + pp->alpha = png_get_uint_16(entry_start); entry_start += 2; + } + pp->frequency = png_get_uint_16(entry_start); entry_start += 2; + } +#else + pp = new_palette.entries; + for (i = 0; i < new_palette.nentries; i++) + { + + if (new_palette.depth == 8) + { + pp[i].red = *entry_start++; + pp[i].green = *entry_start++; + pp[i].blue = *entry_start++; + pp[i].alpha = *entry_start++; + } + else + { + pp[i].red = png_get_uint_16(entry_start); entry_start += 2; + pp[i].green = png_get_uint_16(entry_start); entry_start += 2; + pp[i].blue = png_get_uint_16(entry_start); entry_start += 2; + pp[i].alpha = png_get_uint_16(entry_start); entry_start += 2; + } + pp->frequency = png_get_uint_16(entry_start); entry_start += 2; + } +#endif + + /* discard all chunk data except the name and stash that */ + new_palette.name = (png_charp)chunkdata; + + png_set_sPLT(png_ptr, info_ptr, &new_palette, 1); + + png_free(png_ptr, chunkdata); + png_free(png_ptr, new_palette.entries); +} +#endif /* PNG_READ_sPLT_SUPPORTED */ + +#if defined(PNG_READ_tRNS_SUPPORTED) +void /* PRIVATE */ +png_handle_tRNS(png_structp png_ptr, png_infop info_ptr, png_uint_32 length) +{ + png_byte readbuf[PNG_MAX_PALETTE_LENGTH]; + + png_debug(1, "in png_handle_tRNS\n"); + + if (!(png_ptr->mode & PNG_HAVE_IHDR)) + png_error(png_ptr, "Missing IHDR before tRNS"); + else if (png_ptr->mode & PNG_HAVE_IDAT) + { + png_warning(png_ptr, "Invalid tRNS after IDAT"); + png_crc_finish(png_ptr, length); + return; + } + else if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_tRNS)) + { + png_warning(png_ptr, "Duplicate tRNS chunk"); + png_crc_finish(png_ptr, length); + return; + } + + if (png_ptr->color_type == PNG_COLOR_TYPE_GRAY) + { + png_byte buf[2]; + + if (length != 2) + { + png_warning(png_ptr, "Incorrect tRNS chunk length"); + png_crc_finish(png_ptr, length); + return; + } + + png_crc_read(png_ptr, buf, 2); + png_ptr->num_trans = 1; + png_ptr->trans_values.gray = png_get_uint_16(buf); + } + else if (png_ptr->color_type == PNG_COLOR_TYPE_RGB) + { + png_byte buf[6]; + + if (length != 6) + { + png_warning(png_ptr, "Incorrect tRNS chunk length"); + png_crc_finish(png_ptr, length); + return; + } + png_crc_read(png_ptr, buf, (png_size_t)length); + png_ptr->num_trans = 1; + png_ptr->trans_values.red = png_get_uint_16(buf); + png_ptr->trans_values.green = png_get_uint_16(buf + 2); + png_ptr->trans_values.blue = png_get_uint_16(buf + 4); + } + else if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE) + { + if (!(png_ptr->mode & PNG_HAVE_PLTE)) + { + /* Should be an error, but we can cope with it. */ + png_warning(png_ptr, "Missing PLTE before tRNS"); + } + if (length > (png_uint_32)png_ptr->num_palette || + length > PNG_MAX_PALETTE_LENGTH) + { + png_warning(png_ptr, "Incorrect tRNS chunk length"); + png_crc_finish(png_ptr, length); + return; + } + if (length == 0) + { + png_warning(png_ptr, "Zero length tRNS chunk"); + png_crc_finish(png_ptr, length); + return; + } + png_crc_read(png_ptr, readbuf, (png_size_t)length); + png_ptr->num_trans = (png_uint_16)length; + } + else + { + png_warning(png_ptr, "tRNS chunk not allowed with alpha channel"); + png_crc_finish(png_ptr, length); + return; + } + + if (png_crc_finish(png_ptr, 0)) + return; + + png_set_tRNS(png_ptr, info_ptr, readbuf, png_ptr->num_trans, + &(png_ptr->trans_values)); +} +#endif + +#if defined(PNG_READ_bKGD_SUPPORTED) +void /* PRIVATE */ +png_handle_bKGD(png_structp png_ptr, png_infop info_ptr, png_uint_32 length) +{ + png_size_t truelen; + png_byte buf[6]; + + png_debug(1, "in png_handle_bKGD\n"); + + if (!(png_ptr->mode & PNG_HAVE_IHDR)) + png_error(png_ptr, "Missing IHDR before bKGD"); + else if (png_ptr->mode & PNG_HAVE_IDAT) + { + png_warning(png_ptr, "Invalid bKGD after IDAT"); + png_crc_finish(png_ptr, length); + return; + } + else if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE && + !(png_ptr->mode & PNG_HAVE_PLTE)) + { + png_warning(png_ptr, "Missing PLTE before bKGD"); + png_crc_finish(png_ptr, length); + return; + } + else if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_bKGD)) + { + png_warning(png_ptr, "Duplicate bKGD chunk"); + png_crc_finish(png_ptr, length); + return; + } + + if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE) + truelen = 1; + else if (png_ptr->color_type & PNG_COLOR_MASK_COLOR) + truelen = 6; + else + truelen = 2; + + if (length != truelen) + { + png_warning(png_ptr, "Incorrect bKGD chunk length"); + png_crc_finish(png_ptr, length); + return; + } + + png_crc_read(png_ptr, buf, truelen); + if (png_crc_finish(png_ptr, 0)) + return; + + /* We convert the index value into RGB components so that we can allow + * arbitrary RGB values for background when we have transparency, and + * so it is easy to determine the RGB values of the background color + * from the info_ptr struct. */ + if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE) + { + png_ptr->background.index = buf[0]; + if(info_ptr->num_palette) + { + if(buf[0] > info_ptr->num_palette) + { + png_warning(png_ptr, "Incorrect bKGD chunk index value"); + return; + } + png_ptr->background.red = + (png_uint_16)png_ptr->palette[buf[0]].red; + png_ptr->background.green = + (png_uint_16)png_ptr->palette[buf[0]].green; + png_ptr->background.blue = + (png_uint_16)png_ptr->palette[buf[0]].blue; + } + } + else if (!(png_ptr->color_type & PNG_COLOR_MASK_COLOR)) /* GRAY */ + { + png_ptr->background.red = + png_ptr->background.green = + png_ptr->background.blue = + png_ptr->background.gray = png_get_uint_16(buf); + } + else + { + png_ptr->background.red = png_get_uint_16(buf); + png_ptr->background.green = png_get_uint_16(buf + 2); + png_ptr->background.blue = png_get_uint_16(buf + 4); + } + + png_set_bKGD(png_ptr, info_ptr, &(png_ptr->background)); +} +#endif + +#if defined(PNG_READ_hIST_SUPPORTED) +void /* PRIVATE */ +png_handle_hIST(png_structp png_ptr, png_infop info_ptr, png_uint_32 length) +{ + unsigned int num, i; + png_uint_16 readbuf[PNG_MAX_PALETTE_LENGTH]; + + png_debug(1, "in png_handle_hIST\n"); + + if (!(png_ptr->mode & PNG_HAVE_IHDR)) + png_error(png_ptr, "Missing IHDR before hIST"); + else if (png_ptr->mode & PNG_HAVE_IDAT) + { + png_warning(png_ptr, "Invalid hIST after IDAT"); + png_crc_finish(png_ptr, length); + return; + } + else if (!(png_ptr->mode & PNG_HAVE_PLTE)) + { + png_warning(png_ptr, "Missing PLTE before hIST"); + png_crc_finish(png_ptr, length); + return; + } + else if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_hIST)) + { + png_warning(png_ptr, "Duplicate hIST chunk"); + png_crc_finish(png_ptr, length); + return; + } + + num = length / 2 ; + if (num != (unsigned int) png_ptr->num_palette || num > + (unsigned int) PNG_MAX_PALETTE_LENGTH) + { + png_warning(png_ptr, "Incorrect hIST chunk length"); + png_crc_finish(png_ptr, length); + return; + } + + for (i = 0; i < num; i++) + { + png_byte buf[2]; + + png_crc_read(png_ptr, buf, 2); + readbuf[i] = png_get_uint_16(buf); + } + + if (png_crc_finish(png_ptr, 0)) + return; + + png_set_hIST(png_ptr, info_ptr, readbuf); +} +#endif + +#if defined(PNG_READ_pHYs_SUPPORTED) +void /* PRIVATE */ +png_handle_pHYs(png_structp png_ptr, png_infop info_ptr, png_uint_32 length) +{ + png_byte buf[9]; + png_uint_32 res_x, res_y; + int unit_type; + + png_debug(1, "in png_handle_pHYs\n"); + + if (!(png_ptr->mode & PNG_HAVE_IHDR)) + png_error(png_ptr, "Missing IHDR before pHYs"); + else if (png_ptr->mode & PNG_HAVE_IDAT) + { + png_warning(png_ptr, "Invalid pHYs after IDAT"); + png_crc_finish(png_ptr, length); + return; + } + else if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_pHYs)) + { + png_warning(png_ptr, "Duplicate pHYs chunk"); + png_crc_finish(png_ptr, length); + return; + } + + if (length != 9) + { + png_warning(png_ptr, "Incorrect pHYs chunk length"); + png_crc_finish(png_ptr, length); + return; + } + + png_crc_read(png_ptr, buf, 9); + if (png_crc_finish(png_ptr, 0)) + return; + + res_x = png_get_uint_32(buf); + res_y = png_get_uint_32(buf + 4); + unit_type = buf[8]; + png_set_pHYs(png_ptr, info_ptr, res_x, res_y, unit_type); +} +#endif + +#if defined(PNG_READ_oFFs_SUPPORTED) +void /* PRIVATE */ +png_handle_oFFs(png_structp png_ptr, png_infop info_ptr, png_uint_32 length) +{ + png_byte buf[9]; + png_int_32 offset_x, offset_y; + int unit_type; + + png_debug(1, "in png_handle_oFFs\n"); + + if (!(png_ptr->mode & PNG_HAVE_IHDR)) + png_error(png_ptr, "Missing IHDR before oFFs"); + else if (png_ptr->mode & PNG_HAVE_IDAT) + { + png_warning(png_ptr, "Invalid oFFs after IDAT"); + png_crc_finish(png_ptr, length); + return; + } + else if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_oFFs)) + { + png_warning(png_ptr, "Duplicate oFFs chunk"); + png_crc_finish(png_ptr, length); + return; + } + + if (length != 9) + { + png_warning(png_ptr, "Incorrect oFFs chunk length"); + png_crc_finish(png_ptr, length); + return; + } + + png_crc_read(png_ptr, buf, 9); + if (png_crc_finish(png_ptr, 0)) + return; + + offset_x = png_get_int_32(buf); + offset_y = png_get_int_32(buf + 4); + unit_type = buf[8]; + png_set_oFFs(png_ptr, info_ptr, offset_x, offset_y, unit_type); +} +#endif + +#if defined(PNG_READ_pCAL_SUPPORTED) +/* read the pCAL chunk (described in the PNG Extensions document) */ +void /* PRIVATE */ +png_handle_pCAL(png_structp png_ptr, png_infop info_ptr, png_uint_32 length) +{ + png_charp purpose; + png_int_32 X0, X1; + png_byte type, nparams; + png_charp buf, units, endptr; + png_charpp params; + png_size_t slength; + int i; + + png_debug(1, "in png_handle_pCAL\n"); + + if (!(png_ptr->mode & PNG_HAVE_IHDR)) + png_error(png_ptr, "Missing IHDR before pCAL"); + else if (png_ptr->mode & PNG_HAVE_IDAT) + { + png_warning(png_ptr, "Invalid pCAL after IDAT"); + png_crc_finish(png_ptr, length); + return; + } + else if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_pCAL)) + { + png_warning(png_ptr, "Duplicate pCAL chunk"); + png_crc_finish(png_ptr, length); + return; + } + + png_debug1(2, "Allocating and reading pCAL chunk data (%lu bytes)\n", + length + 1); + purpose = (png_charp)png_malloc_warn(png_ptr, length + 1); + if (purpose == NULL) + { + png_warning(png_ptr, "No memory for pCAL purpose."); + return; + } + slength = (png_size_t)length; + png_crc_read(png_ptr, (png_bytep)purpose, slength); + + if (png_crc_finish(png_ptr, 0)) + { + png_free(png_ptr, purpose); + return; + } + + purpose[slength] = 0x00; /* null terminate the last string */ + + png_debug(3, "Finding end of pCAL purpose string\n"); + for (buf = purpose; *buf; buf++) + /* empty loop */ ; + + endptr = purpose + slength; + + /* We need to have at least 12 bytes after the purpose string + in order to get the parameter information. */ + if (endptr <= buf + 12) + { + png_warning(png_ptr, "Invalid pCAL data"); + png_free(png_ptr, purpose); + return; + } + + png_debug(3, "Reading pCAL X0, X1, type, nparams, and units\n"); + X0 = png_get_int_32((png_bytep)buf+1); + X1 = png_get_int_32((png_bytep)buf+5); + type = buf[9]; + nparams = buf[10]; + units = buf + 11; + + png_debug(3, "Checking pCAL equation type and number of parameters\n"); + /* Check that we have the right number of parameters for known + equation types. */ + if ((type == PNG_EQUATION_LINEAR && nparams != 2) || + (type == PNG_EQUATION_BASE_E && nparams != 3) || + (type == PNG_EQUATION_ARBITRARY && nparams != 3) || + (type == PNG_EQUATION_HYPERBOLIC && nparams != 4)) + { + png_warning(png_ptr, "Invalid pCAL parameters for equation type"); + png_free(png_ptr, purpose); + return; + } + else if (type >= PNG_EQUATION_LAST) + { + png_warning(png_ptr, "Unrecognized equation type for pCAL chunk"); + } + + for (buf = units; *buf; buf++) + /* Empty loop to move past the units string. */ ; + + png_debug(3, "Allocating pCAL parameters array\n"); + params = (png_charpp)png_malloc_warn(png_ptr, (png_uint_32)(nparams + *png_sizeof(png_charp))) ; + if (params == NULL) + { + png_free(png_ptr, purpose); + png_warning(png_ptr, "No memory for pCAL params."); + return; + } + + /* Get pointers to the start of each parameter string. */ + for (i = 0; i < (int)nparams; i++) + { + buf++; /* Skip the null string terminator from previous parameter. */ + + png_debug1(3, "Reading pCAL parameter %d\n", i); + for (params[i] = buf; *buf != 0x00 && buf <= endptr; buf++) + /* Empty loop to move past each parameter string */ ; + + /* Make sure we haven't run out of data yet */ + if (buf > endptr) + { + png_warning(png_ptr, "Invalid pCAL data"); + png_free(png_ptr, purpose); + png_free(png_ptr, params); + return; + } + } + + png_set_pCAL(png_ptr, info_ptr, purpose, X0, X1, type, nparams, + units, params); + + png_free(png_ptr, purpose); + png_free(png_ptr, params); +} +#endif + +#if defined(PNG_READ_sCAL_SUPPORTED) +/* read the sCAL chunk */ +void /* PRIVATE */ +png_handle_sCAL(png_structp png_ptr, png_infop info_ptr, png_uint_32 length) +{ + png_charp buffer, ep; +#ifdef PNG_FLOATING_POINT_SUPPORTED + double width, height; + png_charp vp; +#else +#ifdef PNG_FIXED_POINT_SUPPORTED + png_charp swidth, sheight; +#endif +#endif + png_size_t slength; + + png_debug(1, "in png_handle_sCAL\n"); + + if (!(png_ptr->mode & PNG_HAVE_IHDR)) + png_error(png_ptr, "Missing IHDR before sCAL"); + else if (png_ptr->mode & PNG_HAVE_IDAT) + { + png_warning(png_ptr, "Invalid sCAL after IDAT"); + png_crc_finish(png_ptr, length); + return; + } + else if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_sCAL)) + { + png_warning(png_ptr, "Duplicate sCAL chunk"); + png_crc_finish(png_ptr, length); + return; + } + + png_debug1(2, "Allocating and reading sCAL chunk data (%lu bytes)\n", + length + 1); + buffer = (png_charp)png_malloc_warn(png_ptr, length + 1); + if (buffer == NULL) + { + png_warning(png_ptr, "Out of memory while processing sCAL chunk"); + return; + } + slength = (png_size_t)length; + png_crc_read(png_ptr, (png_bytep)buffer, slength); + + if (png_crc_finish(png_ptr, 0)) + { + png_free(png_ptr, buffer); + return; + } + + buffer[slength] = 0x00; /* null terminate the last string */ + + ep = buffer + 1; /* skip unit byte */ + +#ifdef PNG_FLOATING_POINT_SUPPORTED + width = strtod(ep, &vp); + if (*vp) + { + png_warning(png_ptr, "malformed width string in sCAL chunk"); + return; + } +#else +#ifdef PNG_FIXED_POINT_SUPPORTED + swidth = (png_charp)png_malloc_warn(png_ptr, png_strlen(ep) + 1); + if (swidth == NULL) + { + png_warning(png_ptr, "Out of memory while processing sCAL chunk width"); + return; + } + png_memcpy(swidth, ep, (png_size_t)png_strlen(ep)); +#endif +#endif + + for (ep = buffer; *ep; ep++) + /* empty loop */ ; + ep++; + +#ifdef PNG_FLOATING_POINT_SUPPORTED + height = strtod(ep, &vp); + if (*vp) + { + png_warning(png_ptr, "malformed height string in sCAL chunk"); + return; + } +#else +#ifdef PNG_FIXED_POINT_SUPPORTED + sheight = (png_charp)png_malloc_warn(png_ptr, png_strlen(ep) + 1); + if (swidth == NULL) + { + png_warning(png_ptr, "Out of memory while processing sCAL chunk height"); + return; + } + png_memcpy(sheight, ep, (png_size_t)png_strlen(ep)); +#endif +#endif + + if (buffer + slength < ep +#ifdef PNG_FLOATING_POINT_SUPPORTED + || width <= 0. || height <= 0. +#endif + ) + { + png_warning(png_ptr, "Invalid sCAL data"); + png_free(png_ptr, buffer); +#if defined(PNG_FIXED_POINT_SUPPORTED) && !defined(PNG_FLOATING_POINT_SUPPORTED) + png_free(png_ptr, swidth); + png_free(png_ptr, sheight); +#endif + return; + } + + +#ifdef PNG_FLOATING_POINT_SUPPORTED + png_set_sCAL(png_ptr, info_ptr, buffer[0], width, height); +#else +#ifdef PNG_FIXED_POINT_SUPPORTED + png_set_sCAL_s(png_ptr, info_ptr, buffer[0], swidth, sheight); +#endif +#endif + + png_free(png_ptr, buffer); +#if defined(PNG_FIXED_POINT_SUPPORTED) && !defined(PNG_FLOATING_POINT_SUPPORTED) + png_free(png_ptr, swidth); + png_free(png_ptr, sheight); +#endif +} +#endif + +#if defined(PNG_READ_tIME_SUPPORTED) +void /* PRIVATE */ +png_handle_tIME(png_structp png_ptr, png_infop info_ptr, png_uint_32 length) +{ + png_byte buf[7]; + png_time mod_time; + + png_debug(1, "in png_handle_tIME\n"); + + if (!(png_ptr->mode & PNG_HAVE_IHDR)) + png_error(png_ptr, "Out of place tIME chunk"); + else if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_tIME)) + { + png_warning(png_ptr, "Duplicate tIME chunk"); + png_crc_finish(png_ptr, length); + return; + } + + if (png_ptr->mode & PNG_HAVE_IDAT) + png_ptr->mode |= PNG_AFTER_IDAT; + + if (length != 7) + { + png_warning(png_ptr, "Incorrect tIME chunk length"); + png_crc_finish(png_ptr, length); + return; + } + + png_crc_read(png_ptr, buf, 7); + if (png_crc_finish(png_ptr, 0)) + return; + + mod_time.second = buf[6]; + mod_time.minute = buf[5]; + mod_time.hour = buf[4]; + mod_time.day = buf[3]; + mod_time.month = buf[2]; + mod_time.year = png_get_uint_16(buf); + + png_set_tIME(png_ptr, info_ptr, &mod_time); +} +#endif + +#if defined(PNG_READ_tEXt_SUPPORTED) +/* Note: this does not properly handle chunks that are > 64K under DOS */ +void /* PRIVATE */ +png_handle_tEXt(png_structp png_ptr, png_infop info_ptr, png_uint_32 length) +{ + png_textp text_ptr; + png_charp key; + png_charp text; + png_uint_32 skip = 0; + png_size_t slength; + int ret; + + png_debug(1, "in png_handle_tEXt\n"); + + if (!(png_ptr->mode & PNG_HAVE_IHDR)) + png_error(png_ptr, "Missing IHDR before tEXt"); + + if (png_ptr->mode & PNG_HAVE_IDAT) + png_ptr->mode |= PNG_AFTER_IDAT; + +#ifdef PNG_MAX_MALLOC_64K + if (length > (png_uint_32)65535L) + { + png_warning(png_ptr, "tEXt chunk too large to fit in memory"); + skip = length - (png_uint_32)65535L; + length = (png_uint_32)65535L; + } +#endif + + key = (png_charp)png_malloc_warn(png_ptr, length + 1); + if (key == NULL) + { + png_warning(png_ptr, "No memory to process text chunk."); + return; + } + slength = (png_size_t)length; + png_crc_read(png_ptr, (png_bytep)key, slength); + + if (png_crc_finish(png_ptr, skip)) + { + png_free(png_ptr, key); + return; + } + + key[slength] = 0x00; + + for (text = key; *text; text++) + /* empty loop to find end of key */ ; + + if (text != key + slength) + text++; + + text_ptr = (png_textp)png_malloc_warn(png_ptr, + (png_uint_32)png_sizeof(png_text)); + if (text_ptr == NULL) + { + png_warning(png_ptr, "Not enough memory to process text chunk."); + png_free(png_ptr, key); + return; + } + text_ptr->compression = PNG_TEXT_COMPRESSION_NONE; + text_ptr->key = key; +#ifdef PNG_iTXt_SUPPORTED + text_ptr->lang = NULL; + text_ptr->lang_key = NULL; + text_ptr->itxt_length = 0; +#endif + text_ptr->text = text; + text_ptr->text_length = png_strlen(text); + + ret=png_set_text_2(png_ptr, info_ptr, text_ptr, 1); + + png_free(png_ptr, key); + png_free(png_ptr, text_ptr); + if (ret) + png_warning(png_ptr, "Insufficient memory to process text chunk."); +} +#endif + +#if defined(PNG_READ_zTXt_SUPPORTED) +/* note: this does not correctly handle chunks that are > 64K under DOS */ +void /* PRIVATE */ +png_handle_zTXt(png_structp png_ptr, png_infop info_ptr, png_uint_32 length) +{ + png_textp text_ptr; + png_charp chunkdata; + png_charp text; + int comp_type; + int ret; + png_size_t slength, prefix_len, data_len; + + png_debug(1, "in png_handle_zTXt\n"); + if (!(png_ptr->mode & PNG_HAVE_IHDR)) + png_error(png_ptr, "Missing IHDR before zTXt"); + + if (png_ptr->mode & PNG_HAVE_IDAT) + png_ptr->mode |= PNG_AFTER_IDAT; + +#ifdef PNG_MAX_MALLOC_64K + /* We will no doubt have problems with chunks even half this size, but + there is no hard and fast rule to tell us where to stop. */ + if (length > (png_uint_32)65535L) + { + png_warning(png_ptr,"zTXt chunk too large to fit in memory"); + png_crc_finish(png_ptr, length); + return; + } +#endif + + chunkdata = (png_charp)png_malloc_warn(png_ptr, length + 1); + if (chunkdata == NULL) + { + png_warning(png_ptr,"Out of memory processing zTXt chunk."); + return; + } + slength = (png_size_t)length; + png_crc_read(png_ptr, (png_bytep)chunkdata, slength); + if (png_crc_finish(png_ptr, 0)) + { + png_free(png_ptr, chunkdata); + return; + } + + chunkdata[slength] = 0x00; + + for (text = chunkdata; *text; text++) + /* empty loop */ ; + + /* zTXt must have some text after the chunkdataword */ + if (text == chunkdata + slength) + { + comp_type = PNG_TEXT_COMPRESSION_NONE; + png_warning(png_ptr, "Zero length zTXt chunk"); + } + else + { + comp_type = *(++text); + if (comp_type != PNG_TEXT_COMPRESSION_zTXt) + { + png_warning(png_ptr, "Unknown compression type in zTXt chunk"); + comp_type = PNG_TEXT_COMPRESSION_zTXt; + } + text++; /* skip the compression_method byte */ + } + prefix_len = text - chunkdata; + + chunkdata = (png_charp)png_decompress_chunk(png_ptr, comp_type, chunkdata, + (png_size_t)length, prefix_len, &data_len); + + text_ptr = (png_textp)png_malloc_warn(png_ptr, + (png_uint_32)png_sizeof(png_text)); + if (text_ptr == NULL) + { + png_warning(png_ptr,"Not enough memory to process zTXt chunk."); + png_free(png_ptr, chunkdata); + return; + } + text_ptr->compression = comp_type; + text_ptr->key = chunkdata; +#ifdef PNG_iTXt_SUPPORTED + text_ptr->lang = NULL; + text_ptr->lang_key = NULL; + text_ptr->itxt_length = 0; +#endif + text_ptr->text = chunkdata + prefix_len; + text_ptr->text_length = data_len; + + ret=png_set_text_2(png_ptr, info_ptr, text_ptr, 1); + + png_free(png_ptr, text_ptr); + png_free(png_ptr, chunkdata); + if (ret) + png_error(png_ptr, "Insufficient memory to store zTXt chunk."); +} +#endif + +#if defined(PNG_READ_iTXt_SUPPORTED) +/* note: this does not correctly handle chunks that are > 64K under DOS */ +void /* PRIVATE */ +png_handle_iTXt(png_structp png_ptr, png_infop info_ptr, png_uint_32 length) +{ + png_textp text_ptr; + png_charp chunkdata; + png_charp key, lang, text, lang_key; + int comp_flag; + int comp_type = 0; + int ret; + png_size_t slength, prefix_len, data_len; + + png_debug(1, "in png_handle_iTXt\n"); + + if (!(png_ptr->mode & PNG_HAVE_IHDR)) + png_error(png_ptr, "Missing IHDR before iTXt"); + + if (png_ptr->mode & PNG_HAVE_IDAT) + png_ptr->mode |= PNG_AFTER_IDAT; + +#ifdef PNG_MAX_MALLOC_64K + /* We will no doubt have problems with chunks even half this size, but + there is no hard and fast rule to tell us where to stop. */ + if (length > (png_uint_32)65535L) + { + png_warning(png_ptr,"iTXt chunk too large to fit in memory"); + png_crc_finish(png_ptr, length); + return; + } +#endif + + chunkdata = (png_charp)png_malloc_warn(png_ptr, length + 1); + if (chunkdata == NULL) + { + png_warning(png_ptr, "No memory to process iTXt chunk."); + return; + } + slength = (png_size_t)length; + png_crc_read(png_ptr, (png_bytep)chunkdata, slength); + if (png_crc_finish(png_ptr, 0)) + { + png_free(png_ptr, chunkdata); + return; + } + + chunkdata[slength] = 0x00; + + for (lang = chunkdata; *lang; lang++) + /* empty loop */ ; + lang++; /* skip NUL separator */ + + /* iTXt must have a language tag (possibly empty), two compression bytes, + translated keyword (possibly empty), and possibly some text after the + keyword */ + + if (lang >= chunkdata + slength) + { + comp_flag = PNG_TEXT_COMPRESSION_NONE; + png_warning(png_ptr, "Zero length iTXt chunk"); + } + else + { + comp_flag = *lang++; + comp_type = *lang++; + } + + for (lang_key = lang; *lang_key; lang_key++) + /* empty loop */ ; + lang_key++; /* skip NUL separator */ + + for (text = lang_key; *text; text++) + /* empty loop */ ; + text++; /* skip NUL separator */ + + prefix_len = text - chunkdata; + + key=chunkdata; + if (comp_flag) + chunkdata = png_decompress_chunk(png_ptr, comp_type, chunkdata, + (size_t)length, prefix_len, &data_len); + else + data_len=png_strlen(chunkdata + prefix_len); + text_ptr = (png_textp)png_malloc_warn(png_ptr, + (png_uint_32)png_sizeof(png_text)); + if (text_ptr == NULL) + { + png_warning(png_ptr,"Not enough memory to process iTXt chunk."); + png_free(png_ptr, chunkdata); + return; + } + text_ptr->compression = (int)comp_flag + 1; + text_ptr->lang_key = chunkdata+(lang_key-key); + text_ptr->lang = chunkdata+(lang-key); + text_ptr->itxt_length = data_len; + text_ptr->text_length = 0; + text_ptr->key = chunkdata; + text_ptr->text = chunkdata + prefix_len; + + ret=png_set_text_2(png_ptr, info_ptr, text_ptr, 1); + + png_free(png_ptr, text_ptr); + png_free(png_ptr, chunkdata); + if (ret) + png_error(png_ptr, "Insufficient memory to store iTXt chunk."); +} +#endif + +/* This function is called when we haven't found a handler for a + chunk. If there isn't a problem with the chunk itself (ie bad + chunk name, CRC, or a critical chunk), the chunk is silently ignored + -- unless the PNG_FLAG_UNKNOWN_CHUNKS_SUPPORTED flag is on in which + case it will be saved away to be written out later. */ +void /* PRIVATE */ +png_handle_unknown(png_structp png_ptr, png_infop info_ptr, png_uint_32 length) +{ + png_uint_32 skip = 0; + + png_debug(1, "in png_handle_unknown\n"); + + if (png_ptr->mode & PNG_HAVE_IDAT) + { +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_IDAT; +#endif + if (png_memcmp(png_ptr->chunk_name, png_IDAT, 4)) /* not an IDAT */ + png_ptr->mode |= PNG_AFTER_IDAT; + } + + png_check_chunk_name(png_ptr, png_ptr->chunk_name); + + if (!(png_ptr->chunk_name[0] & 0x20)) + { +#if defined(PNG_READ_UNKNOWN_CHUNKS_SUPPORTED) + if(png_handle_as_unknown(png_ptr, png_ptr->chunk_name) != + PNG_HANDLE_CHUNK_ALWAYS +#if defined(PNG_READ_USER_CHUNKS_SUPPORTED) + && png_ptr->read_user_chunk_fn == NULL +#endif + ) +#endif + png_chunk_error(png_ptr, "unknown critical chunk"); + } + +#if defined(PNG_READ_UNKNOWN_CHUNKS_SUPPORTED) + if ((png_ptr->flags & PNG_FLAG_KEEP_UNKNOWN_CHUNKS) || + (png_ptr->read_user_chunk_fn != NULL)) + { + png_unknown_chunk chunk; + +#ifdef PNG_MAX_MALLOC_64K + if (length > (png_uint_32)65535L) + { + png_warning(png_ptr, "unknown chunk too large to fit in memory"); + skip = length - (png_uint_32)65535L; + length = (png_uint_32)65535L; + } +#endif + png_strcpy((png_charp)chunk.name, (png_charp)png_ptr->chunk_name); + chunk.data = (png_bytep)png_malloc(png_ptr, length); + chunk.size = (png_size_t)length; + png_crc_read(png_ptr, (png_bytep)chunk.data, length); +#if defined(PNG_READ_USER_CHUNKS_SUPPORTED) + if(png_ptr->read_user_chunk_fn != NULL) + { + /* callback to user unknown chunk handler */ + if ((*(png_ptr->read_user_chunk_fn)) (png_ptr, &chunk) <= 0) + { + if (!(png_ptr->chunk_name[0] & 0x20)) + if(png_handle_as_unknown(png_ptr, png_ptr->chunk_name) != + PNG_HANDLE_CHUNK_ALWAYS) + { + png_free(png_ptr, chunk.data); + png_chunk_error(png_ptr, "unknown critical chunk"); + } + png_set_unknown_chunks(png_ptr, info_ptr, &chunk, 1); + } + } + else +#endif + png_set_unknown_chunks(png_ptr, info_ptr, &chunk, 1); + png_free(png_ptr, chunk.data); + } + else +#endif + skip = length; + + png_crc_finish(png_ptr, skip); + +#if !defined(PNG_READ_USER_CHUNKS_SUPPORTED) + if (&info_ptr == NULL) /* quiet compiler warnings about unused info_ptr */ + return; +#endif +} + +/* This function is called to verify that a chunk name is valid. + This function can't have the "critical chunk check" incorporated + into it, since in the future we will need to be able to call user + functions to handle unknown critical chunks after we check that + the chunk name itself is valid. */ + +#define isnonalpha(c) ((c) < 65 || (c) > 122 || ((c) > 90 && (c) < 97)) + +void /* PRIVATE */ +png_check_chunk_name(png_structp png_ptr, png_bytep chunk_name) +{ + png_debug(1, "in png_check_chunk_name\n"); + if (isnonalpha(chunk_name[0]) || isnonalpha(chunk_name[1]) || + isnonalpha(chunk_name[2]) || isnonalpha(chunk_name[3])) + { + png_chunk_error(png_ptr, "invalid chunk type"); + } +} + +/* Combines the row recently read in with the existing pixels in the + row. This routine takes care of alpha and transparency if requested. + This routine also handles the two methods of progressive display + of interlaced images, depending on the mask value. + The mask value describes which pixels are to be combined with + the row. The pattern always repeats every 8 pixels, so just 8 + bits are needed. A one indicates the pixel is to be combined, + a zero indicates the pixel is to be skipped. This is in addition + to any alpha or transparency value associated with the pixel. If + you want all pixels to be combined, pass 0xff (255) in mask. */ +#ifndef PNG_HAVE_ASSEMBLER_COMBINE_ROW +void /* PRIVATE */ +png_combine_row(png_structp png_ptr, png_bytep row, int mask) +{ + png_debug(1,"in png_combine_row\n"); + if (mask == 0xff) + { + png_memcpy(row, png_ptr->row_buf + 1, + PNG_ROWBYTES(png_ptr->row_info.pixel_depth, png_ptr->width)); + } + else + { + switch (png_ptr->row_info.pixel_depth) + { + case 1: + { + png_bytep sp = png_ptr->row_buf + 1; + png_bytep dp = row; + int s_inc, s_start, s_end; + int m = 0x80; + int shift; + png_uint_32 i; + png_uint_32 row_width = png_ptr->width; + +#if defined(PNG_READ_PACKSWAP_SUPPORTED) + if (png_ptr->transformations & PNG_PACKSWAP) + { + s_start = 0; + s_end = 7; + s_inc = 1; + } + else +#endif + { + s_start = 7; + s_end = 0; + s_inc = -1; + } + + shift = s_start; + + for (i = 0; i < row_width; i++) + { + if (m & mask) + { + int value; + + value = (*sp >> shift) & 0x01; + *dp &= (png_byte)((0x7f7f >> (7 - shift)) & 0xff); + *dp |= (png_byte)(value << shift); + } + + if (shift == s_end) + { + shift = s_start; + sp++; + dp++; + } + else + shift += s_inc; + + if (m == 1) + m = 0x80; + else + m >>= 1; + } + break; + } + case 2: + { + png_bytep sp = png_ptr->row_buf + 1; + png_bytep dp = row; + int s_start, s_end, s_inc; + int m = 0x80; + int shift; + png_uint_32 i; + png_uint_32 row_width = png_ptr->width; + int value; + +#if defined(PNG_READ_PACKSWAP_SUPPORTED) + if (png_ptr->transformations & PNG_PACKSWAP) + { + s_start = 0; + s_end = 6; + s_inc = 2; + } + else +#endif + { + s_start = 6; + s_end = 0; + s_inc = -2; + } + + shift = s_start; + + for (i = 0; i < row_width; i++) + { + if (m & mask) + { + value = (*sp >> shift) & 0x03; + *dp &= (png_byte)((0x3f3f >> (6 - shift)) & 0xff); + *dp |= (png_byte)(value << shift); + } + + if (shift == s_end) + { + shift = s_start; + sp++; + dp++; + } + else + shift += s_inc; + if (m == 1) + m = 0x80; + else + m >>= 1; + } + break; + } + case 4: + { + png_bytep sp = png_ptr->row_buf + 1; + png_bytep dp = row; + int s_start, s_end, s_inc; + int m = 0x80; + int shift; + png_uint_32 i; + png_uint_32 row_width = png_ptr->width; + int value; + +#if defined(PNG_READ_PACKSWAP_SUPPORTED) + if (png_ptr->transformations & PNG_PACKSWAP) + { + s_start = 0; + s_end = 4; + s_inc = 4; + } + else +#endif + { + s_start = 4; + s_end = 0; + s_inc = -4; + } + shift = s_start; + + for (i = 0; i < row_width; i++) + { + if (m & mask) + { + value = (*sp >> shift) & 0xf; + *dp &= (png_byte)((0xf0f >> (4 - shift)) & 0xff); + *dp |= (png_byte)(value << shift); + } + + if (shift == s_end) + { + shift = s_start; + sp++; + dp++; + } + else + shift += s_inc; + if (m == 1) + m = 0x80; + else + m >>= 1; + } + break; + } + default: + { + png_bytep sp = png_ptr->row_buf + 1; + png_bytep dp = row; + png_size_t pixel_bytes = (png_ptr->row_info.pixel_depth >> 3); + png_uint_32 i; + png_uint_32 row_width = png_ptr->width; + png_byte m = 0x80; + + + for (i = 0; i < row_width; i++) + { + if (m & mask) + { + png_memcpy(dp, sp, pixel_bytes); + } + + sp += pixel_bytes; + dp += pixel_bytes; + + if (m == 1) + m = 0x80; + else + m >>= 1; + } + break; + } + } + } +} +#endif /* !PNG_HAVE_ASSEMBLER_COMBINE_ROW */ + +#ifdef PNG_READ_INTERLACING_SUPPORTED +#ifndef PNG_HAVE_ASSEMBLER_READ_INTERLACE /* else in pngvcrd.c, pnggccrd.c */ +/* OLD pre-1.0.9 interface: +void png_do_read_interlace(png_row_infop row_info, png_bytep row, int pass, + png_uint_32 transformations) + */ +void /* PRIVATE */ +png_do_read_interlace(png_structp png_ptr) +{ + png_row_infop row_info = &(png_ptr->row_info); + png_bytep row = png_ptr->row_buf + 1; + int pass = png_ptr->pass; + png_uint_32 transformations = png_ptr->transformations; +#ifdef PNG_USE_LOCAL_ARRAYS + /* arrays to facilitate easy interlacing - use pass (0 - 6) as index */ + /* offset to next interlace block */ + const int png_pass_inc[7] = {8, 8, 4, 4, 2, 2, 1}; +#endif + + png_debug(1,"in png_do_read_interlace (stock C version)\n"); + if (row != NULL && row_info != NULL) + { + png_uint_32 final_width; + + final_width = row_info->width * png_pass_inc[pass]; + + switch (row_info->pixel_depth) + { + case 1: + { + png_bytep sp = row + (png_size_t)((row_info->width - 1) >> 3); + png_bytep dp = row + (png_size_t)((final_width - 1) >> 3); + int sshift, dshift; + int s_start, s_end, s_inc; + int jstop = png_pass_inc[pass]; + png_byte v; + png_uint_32 i; + int j; + +#if defined(PNG_READ_PACKSWAP_SUPPORTED) + if (transformations & PNG_PACKSWAP) + { + sshift = (int)((row_info->width + 7) & 0x07); + dshift = (int)((final_width + 7) & 0x07); + s_start = 7; + s_end = 0; + s_inc = -1; + } + else +#endif + { + sshift = 7 - (int)((row_info->width + 7) & 0x07); + dshift = 7 - (int)((final_width + 7) & 0x07); + s_start = 0; + s_end = 7; + s_inc = 1; + } + + for (i = 0; i < row_info->width; i++) + { + v = (png_byte)((*sp >> sshift) & 0x01); + for (j = 0; j < jstop; j++) + { + *dp &= (png_byte)((0x7f7f >> (7 - dshift)) & 0xff); + *dp |= (png_byte)(v << dshift); + if (dshift == s_end) + { + dshift = s_start; + dp--; + } + else + dshift += s_inc; + } + if (sshift == s_end) + { + sshift = s_start; + sp--; + } + else + sshift += s_inc; + } + break; + } + case 2: + { + png_bytep sp = row + (png_uint_32)((row_info->width - 1) >> 2); + png_bytep dp = row + (png_uint_32)((final_width - 1) >> 2); + int sshift, dshift; + int s_start, s_end, s_inc; + int jstop = png_pass_inc[pass]; + png_uint_32 i; + +#if defined(PNG_READ_PACKSWAP_SUPPORTED) + if (transformations & PNG_PACKSWAP) + { + sshift = (int)(((row_info->width + 3) & 0x03) << 1); + dshift = (int)(((final_width + 3) & 0x03) << 1); + s_start = 6; + s_end = 0; + s_inc = -2; + } + else +#endif + { + sshift = (int)((3 - ((row_info->width + 3) & 0x03)) << 1); + dshift = (int)((3 - ((final_width + 3) & 0x03)) << 1); + s_start = 0; + s_end = 6; + s_inc = 2; + } + + for (i = 0; i < row_info->width; i++) + { + png_byte v; + int j; + + v = (png_byte)((*sp >> sshift) & 0x03); + for (j = 0; j < jstop; j++) + { + *dp &= (png_byte)((0x3f3f >> (6 - dshift)) & 0xff); + *dp |= (png_byte)(v << dshift); + if (dshift == s_end) + { + dshift = s_start; + dp--; + } + else + dshift += s_inc; + } + if (sshift == s_end) + { + sshift = s_start; + sp--; + } + else + sshift += s_inc; + } + break; + } + case 4: + { + png_bytep sp = row + (png_size_t)((row_info->width - 1) >> 1); + png_bytep dp = row + (png_size_t)((final_width - 1) >> 1); + int sshift, dshift; + int s_start, s_end, s_inc; + png_uint_32 i; + int jstop = png_pass_inc[pass]; + +#if defined(PNG_READ_PACKSWAP_SUPPORTED) + if (transformations & PNG_PACKSWAP) + { + sshift = (int)(((row_info->width + 1) & 0x01) << 2); + dshift = (int)(((final_width + 1) & 0x01) << 2); + s_start = 4; + s_end = 0; + s_inc = -4; + } + else +#endif + { + sshift = (int)((1 - ((row_info->width + 1) & 0x01)) << 2); + dshift = (int)((1 - ((final_width + 1) & 0x01)) << 2); + s_start = 0; + s_end = 4; + s_inc = 4; + } + + for (i = 0; i < row_info->width; i++) + { + png_byte v = (png_byte)((*sp >> sshift) & 0xf); + int j; + + for (j = 0; j < jstop; j++) + { + *dp &= (png_byte)((0xf0f >> (4 - dshift)) & 0xff); + *dp |= (png_byte)(v << dshift); + if (dshift == s_end) + { + dshift = s_start; + dp--; + } + else + dshift += s_inc; + } + if (sshift == s_end) + { + sshift = s_start; + sp--; + } + else + sshift += s_inc; + } + break; + } + default: + { + png_size_t pixel_bytes = (row_info->pixel_depth >> 3); + png_bytep sp = row + (png_size_t)(row_info->width - 1) * pixel_bytes; + png_bytep dp = row + (png_size_t)(final_width - 1) * pixel_bytes; + + int jstop = png_pass_inc[pass]; + png_uint_32 i; + + for (i = 0; i < row_info->width; i++) + { + png_byte v[8]; + int j; + + png_memcpy(v, sp, pixel_bytes); + for (j = 0; j < jstop; j++) + { + png_memcpy(dp, v, pixel_bytes); + dp -= pixel_bytes; + } + sp -= pixel_bytes; + } + break; + } + } + row_info->width = final_width; + row_info->rowbytes = PNG_ROWBYTES(row_info->pixel_depth,final_width); + } +#if !defined(PNG_READ_PACKSWAP_SUPPORTED) + if (&transformations == NULL) /* silence compiler warning */ + return; +#endif +} +#endif /* !PNG_HAVE_ASSEMBLER_READ_INTERLACE */ +#endif /* PNG_READ_INTERLACING_SUPPORTED */ + +#ifndef PNG_HAVE_ASSEMBLER_READ_FILTER_ROW +void /* PRIVATE */ +png_read_filter_row(png_structp png_ptr, png_row_infop row_info, png_bytep row, + png_bytep prev_row, int filter) +{ + png_debug(1, "in png_read_filter_row\n"); + png_debug2(2,"row = %lu, filter = %d\n", png_ptr->row_number, filter); + switch (filter) + { + case PNG_FILTER_VALUE_NONE: + break; + case PNG_FILTER_VALUE_SUB: + { + png_uint_32 i; + png_uint_32 istop = row_info->rowbytes; + png_uint_32 bpp = (row_info->pixel_depth + 7) >> 3; + png_bytep rp = row + bpp; + png_bytep lp = row; + + for (i = bpp; i < istop; i++) + { + *rp = (png_byte)(((int)(*rp) + (int)(*lp++)) & 0xff); + rp++; + } + break; + } + case PNG_FILTER_VALUE_UP: + { + png_uint_32 i; + png_uint_32 istop = row_info->rowbytes; + png_bytep rp = row; + png_bytep pp = prev_row; + + for (i = 0; i < istop; i++) + { + *rp = (png_byte)(((int)(*rp) + (int)(*pp++)) & 0xff); + rp++; + } + break; + } + case PNG_FILTER_VALUE_AVG: + { + png_uint_32 i; + png_bytep rp = row; + png_bytep pp = prev_row; + png_bytep lp = row; + png_uint_32 bpp = (row_info->pixel_depth + 7) >> 3; + png_uint_32 istop = row_info->rowbytes - bpp; + + for (i = 0; i < bpp; i++) + { + *rp = (png_byte)(((int)(*rp) + + ((int)(*pp++) / 2 )) & 0xff); + rp++; + } + + for (i = 0; i < istop; i++) + { + *rp = (png_byte)(((int)(*rp) + + (int)(*pp++ + *lp++) / 2 ) & 0xff); + rp++; + } + break; + } + case PNG_FILTER_VALUE_PAETH: + { + png_uint_32 i; + png_bytep rp = row; + png_bytep pp = prev_row; + png_bytep lp = row; + png_bytep cp = prev_row; + png_uint_32 bpp = (row_info->pixel_depth + 7) >> 3; + png_uint_32 istop=row_info->rowbytes - bpp; + + for (i = 0; i < bpp; i++) + { + *rp = (png_byte)(((int)(*rp) + (int)(*pp++)) & 0xff); + rp++; + } + + for (i = 0; i < istop; i++) /* use leftover rp,pp */ + { + int a, b, c, pa, pb, pc, p; + + a = *lp++; + b = *pp++; + c = *cp++; + + p = b - c; + pc = a - c; + +#ifdef PNG_USE_ABS + pa = abs(p); + pb = abs(pc); + pc = abs(p + pc); +#else + pa = p < 0 ? -p : p; + pb = pc < 0 ? -pc : pc; + pc = (p + pc) < 0 ? -(p + pc) : p + pc; +#endif + + /* + if (pa <= pb && pa <= pc) + p = a; + else if (pb <= pc) + p = b; + else + p = c; + */ + + p = (pa <= pb && pa <=pc) ? a : (pb <= pc) ? b : c; + + *rp = (png_byte)(((int)(*rp) + p) & 0xff); + rp++; + } + break; + } + default: + png_warning(png_ptr, "Ignoring bad adaptive filter type"); + *row=0; + break; + } +} +#endif /* !PNG_HAVE_ASSEMBLER_READ_FILTER_ROW */ + +void /* PRIVATE */ +png_read_finish_row(png_structp png_ptr) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + /* arrays to facilitate easy interlacing - use pass (0 - 6) as index */ + + /* start of interlace block */ + const int png_pass_start[7] = {0, 4, 0, 2, 0, 1, 0}; + + /* offset to next interlace block */ + const int png_pass_inc[7] = {8, 8, 4, 4, 2, 2, 1}; + + /* start of interlace block in the y direction */ + const int png_pass_ystart[7] = {0, 0, 4, 0, 2, 0, 1}; + + /* offset to next interlace block in the y direction */ + const int png_pass_yinc[7] = {8, 8, 8, 4, 4, 2, 2}; +#endif + + png_debug(1, "in png_read_finish_row\n"); + png_ptr->row_number++; + if (png_ptr->row_number < png_ptr->num_rows) + return; + + if (png_ptr->interlaced) + { + png_ptr->row_number = 0; + png_memset_check(png_ptr, png_ptr->prev_row, 0, png_ptr->rowbytes + 1); + do + { + png_ptr->pass++; + if (png_ptr->pass >= 7) + break; + png_ptr->iwidth = (png_ptr->width + + png_pass_inc[png_ptr->pass] - 1 - + png_pass_start[png_ptr->pass]) / + png_pass_inc[png_ptr->pass]; + + png_ptr->irowbytes = PNG_ROWBYTES(png_ptr->pixel_depth, + png_ptr->iwidth) + 1; + + if (!(png_ptr->transformations & PNG_INTERLACE)) + { + png_ptr->num_rows = (png_ptr->height + + png_pass_yinc[png_ptr->pass] - 1 - + png_pass_ystart[png_ptr->pass]) / + png_pass_yinc[png_ptr->pass]; + if (!(png_ptr->num_rows)) + continue; + } + else /* if (png_ptr->transformations & PNG_INTERLACE) */ + break; + } while (png_ptr->iwidth == 0); + + if (png_ptr->pass < 7) + return; + } + + if (!(png_ptr->flags & PNG_FLAG_ZLIB_FINISHED)) + { +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_IDAT; +#endif + char extra; + int ret; + + png_ptr->zstream.next_out = (Byte *)&extra; + png_ptr->zstream.avail_out = (uInt)1; + for(;;) + { + if (!(png_ptr->zstream.avail_in)) + { + while (!png_ptr->idat_size) + { + png_byte chunk_length[4]; + + png_crc_finish(png_ptr, 0); + + png_read_data(png_ptr, chunk_length, 4); + png_ptr->idat_size = png_get_uint_31(png_ptr, chunk_length); + png_reset_crc(png_ptr); + png_crc_read(png_ptr, png_ptr->chunk_name, 4); + if (png_memcmp(png_ptr->chunk_name, (png_bytep)png_IDAT, 4)) + png_error(png_ptr, "Not enough image data"); + + } + png_ptr->zstream.avail_in = (uInt)png_ptr->zbuf_size; + png_ptr->zstream.next_in = png_ptr->zbuf; + if (png_ptr->zbuf_size > png_ptr->idat_size) + png_ptr->zstream.avail_in = (uInt)png_ptr->idat_size; + png_crc_read(png_ptr, png_ptr->zbuf, png_ptr->zstream.avail_in); + png_ptr->idat_size -= png_ptr->zstream.avail_in; + } + ret = inflate(&png_ptr->zstream, Z_PARTIAL_FLUSH); + if (ret == Z_STREAM_END) + { + if (!(png_ptr->zstream.avail_out) || png_ptr->zstream.avail_in || + png_ptr->idat_size) + png_warning(png_ptr, "Extra compressed data"); + png_ptr->mode |= PNG_AFTER_IDAT; + png_ptr->flags |= PNG_FLAG_ZLIB_FINISHED; + break; + } + if (ret != Z_OK) + png_error(png_ptr, png_ptr->zstream.msg ? png_ptr->zstream.msg : + "Decompression Error"); + + if (!(png_ptr->zstream.avail_out)) + { + png_warning(png_ptr, "Extra compressed data."); + png_ptr->mode |= PNG_AFTER_IDAT; + png_ptr->flags |= PNG_FLAG_ZLIB_FINISHED; + break; + } + + } + png_ptr->zstream.avail_out = 0; + } + + if (png_ptr->idat_size || png_ptr->zstream.avail_in) + png_warning(png_ptr, "Extra compression data"); + + inflateReset(&png_ptr->zstream); + + png_ptr->mode |= PNG_AFTER_IDAT; +} + +void /* PRIVATE */ +png_read_start_row(png_structp png_ptr) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + /* arrays to facilitate easy interlacing - use pass (0 - 6) as index */ + + /* start of interlace block */ + const int png_pass_start[7] = {0, 4, 0, 2, 0, 1, 0}; + + /* offset to next interlace block */ + const int png_pass_inc[7] = {8, 8, 4, 4, 2, 2, 1}; + + /* start of interlace block in the y direction */ + const int png_pass_ystart[7] = {0, 0, 4, 0, 2, 0, 1}; + + /* offset to next interlace block in the y direction */ + const int png_pass_yinc[7] = {8, 8, 8, 4, 4, 2, 2}; +#endif + + int max_pixel_depth; + png_uint_32 row_bytes; + + png_debug(1, "in png_read_start_row\n"); + png_ptr->zstream.avail_in = 0; + png_init_read_transformations(png_ptr); + if (png_ptr->interlaced) + { + if (!(png_ptr->transformations & PNG_INTERLACE)) + png_ptr->num_rows = (png_ptr->height + png_pass_yinc[0] - 1 - + png_pass_ystart[0]) / png_pass_yinc[0]; + else + png_ptr->num_rows = png_ptr->height; + + png_ptr->iwidth = (png_ptr->width + + png_pass_inc[png_ptr->pass] - 1 - + png_pass_start[png_ptr->pass]) / + png_pass_inc[png_ptr->pass]; + + row_bytes = PNG_ROWBYTES(png_ptr->pixel_depth,png_ptr->iwidth) + 1; + + png_ptr->irowbytes = (png_size_t)row_bytes; + if((png_uint_32)png_ptr->irowbytes != row_bytes) + png_error(png_ptr, "Rowbytes overflow in png_read_start_row"); + } + else + { + png_ptr->num_rows = png_ptr->height; + png_ptr->iwidth = png_ptr->width; + png_ptr->irowbytes = png_ptr->rowbytes + 1; + } + max_pixel_depth = png_ptr->pixel_depth; + +#if defined(PNG_READ_PACK_SUPPORTED) + if ((png_ptr->transformations & PNG_PACK) && png_ptr->bit_depth < 8) + max_pixel_depth = 8; +#endif + +#if defined(PNG_READ_EXPAND_SUPPORTED) + if (png_ptr->transformations & PNG_EXPAND) + { + if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE) + { + if (png_ptr->num_trans) + max_pixel_depth = 32; + else + max_pixel_depth = 24; + } + else if (png_ptr->color_type == PNG_COLOR_TYPE_GRAY) + { + if (max_pixel_depth < 8) + max_pixel_depth = 8; + if (png_ptr->num_trans) + max_pixel_depth *= 2; + } + else if (png_ptr->color_type == PNG_COLOR_TYPE_RGB) + { + if (png_ptr->num_trans) + { + max_pixel_depth *= 4; + max_pixel_depth /= 3; + } + } + } +#endif + +#if defined(PNG_READ_FILLER_SUPPORTED) + if (png_ptr->transformations & (PNG_FILLER)) + { + if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE) + max_pixel_depth = 32; + else if (png_ptr->color_type == PNG_COLOR_TYPE_GRAY) + { + if (max_pixel_depth <= 8) + max_pixel_depth = 16; + else + max_pixel_depth = 32; + } + else if (png_ptr->color_type == PNG_COLOR_TYPE_RGB) + { + if (max_pixel_depth <= 32) + max_pixel_depth = 32; + else + max_pixel_depth = 64; + } + } +#endif + +#if defined(PNG_READ_GRAY_TO_RGB_SUPPORTED) + if (png_ptr->transformations & PNG_GRAY_TO_RGB) + { + if ( +#if defined(PNG_READ_EXPAND_SUPPORTED) + (png_ptr->num_trans && (png_ptr->transformations & PNG_EXPAND)) || +#endif +#if defined(PNG_READ_FILLER_SUPPORTED) + (png_ptr->transformations & (PNG_FILLER)) || +#endif + png_ptr->color_type == PNG_COLOR_TYPE_GRAY_ALPHA) + { + if (max_pixel_depth <= 16) + max_pixel_depth = 32; + else + max_pixel_depth = 64; + } + else + { + if (max_pixel_depth <= 8) + { + if (png_ptr->color_type == PNG_COLOR_TYPE_RGB_ALPHA) + max_pixel_depth = 32; + else + max_pixel_depth = 24; + } + else if (png_ptr->color_type == PNG_COLOR_TYPE_RGB_ALPHA) + max_pixel_depth = 64; + else + max_pixel_depth = 48; + } + } +#endif + +#if defined(PNG_READ_USER_TRANSFORM_SUPPORTED) && \ +defined(PNG_USER_TRANSFORM_PTR_SUPPORTED) + if(png_ptr->transformations & PNG_USER_TRANSFORM) + { + int user_pixel_depth=png_ptr->user_transform_depth* + png_ptr->user_transform_channels; + if(user_pixel_depth > max_pixel_depth) + max_pixel_depth=user_pixel_depth; + } +#endif + + /* align the width on the next larger 8 pixels. Mainly used + for interlacing */ + row_bytes = ((png_ptr->width + 7) & ~((png_uint_32)7)); + /* calculate the maximum bytes needed, adding a byte and a pixel + for safety's sake */ + row_bytes = PNG_ROWBYTES(max_pixel_depth,row_bytes) + + 1 + ((max_pixel_depth + 7) >> 3); +#ifdef PNG_MAX_MALLOC_64K + if (row_bytes > (png_uint_32)65536L) + png_error(png_ptr, "This image requires a row greater than 64KB"); +#endif + png_ptr->big_row_buf = (png_bytep)png_malloc(png_ptr, row_bytes+64); + png_ptr->row_buf = png_ptr->big_row_buf+32; +#if defined(PNG_DEBUG) && defined(PNG_USE_PNGGCCRD) + png_ptr->row_buf_size = row_bytes; +#endif + +#ifdef PNG_MAX_MALLOC_64K + if ((png_uint_32)png_ptr->rowbytes + 1 > (png_uint_32)65536L) + png_error(png_ptr, "This image requires a row greater than 64KB"); +#endif + if ((png_uint_32)png_ptr->rowbytes > PNG_SIZE_MAX - 1) + png_error(png_ptr, "Row has too many bytes to allocate in memory."); + png_ptr->prev_row = (png_bytep)png_malloc(png_ptr, (png_uint_32)( + png_ptr->rowbytes + 1)); + + png_memset_check(png_ptr, png_ptr->prev_row, 0, png_ptr->rowbytes + 1); + + png_debug1(3, "width = %lu,\n", png_ptr->width); + png_debug1(3, "height = %lu,\n", png_ptr->height); + png_debug1(3, "iwidth = %lu,\n", png_ptr->iwidth); + png_debug1(3, "num_rows = %lu\n", png_ptr->num_rows); + png_debug1(3, "rowbytes = %lu,\n", png_ptr->rowbytes); + png_debug1(3, "irowbytes = %lu,\n", png_ptr->irowbytes); + + png_ptr->flags |= PNG_FLAG_ROW_INIT; +} +#endif /* PNG_READ_SUPPORTED */ diff --git a/src/win32/dependencies/libpng/pngset.c b/src/win32/dependencies/libpng/pngset.c new file mode 100644 index 00000000..9e499c9b --- /dev/null +++ b/src/win32/dependencies/libpng/pngset.c @@ -0,0 +1,1265 @@ + +/* pngset.c - storage of image information into info struct + * + * Last changed in libpng 1.2.9 April 14, 2006 + * For conditions of distribution and use, see copyright notice in png.h + * Copyright (c) 1998-2006 Glenn Randers-Pehrson + * (Version 0.96 Copyright (c) 1996, 1997 Andreas Dilger) + * (Version 0.88 Copyright (c) 1995, 1996 Guy Eric Schalnat, Group 42, Inc.) + * + * The functions here are used during reads to store data from the file + * into the info struct, and during writes to store application data + * into the info struct for writing into the file. This abstracts the + * info struct and allows us to change the structure in the future. + */ + +#define PNG_INTERNAL +#include "png.h" + +#if defined(PNG_READ_SUPPORTED) || defined(PNG_WRITE_SUPPORTED) + +#if defined(PNG_bKGD_SUPPORTED) +void PNGAPI +png_set_bKGD(png_structp png_ptr, png_infop info_ptr, png_color_16p background) +{ + png_debug1(1, "in %s storage function\n", "bKGD"); + if (png_ptr == NULL || info_ptr == NULL) + return; + + png_memcpy(&(info_ptr->background), background, png_sizeof(png_color_16)); + info_ptr->valid |= PNG_INFO_bKGD; +} +#endif + +#if defined(PNG_cHRM_SUPPORTED) +#ifdef PNG_FLOATING_POINT_SUPPORTED +void PNGAPI +png_set_cHRM(png_structp png_ptr, png_infop info_ptr, + double white_x, double white_y, double red_x, double red_y, + double green_x, double green_y, double blue_x, double blue_y) +{ + png_debug1(1, "in %s storage function\n", "cHRM"); + if (png_ptr == NULL || info_ptr == NULL) + return; + + if (white_x < 0.0 || white_y < 0.0 || + red_x < 0.0 || red_y < 0.0 || + green_x < 0.0 || green_y < 0.0 || + blue_x < 0.0 || blue_y < 0.0) + { + png_warning(png_ptr, + "Ignoring attempt to set negative chromaticity value"); + return; + } + if (white_x > 21474.83 || white_y > 21474.83 || + red_x > 21474.83 || red_y > 21474.83 || + green_x > 21474.83 || green_y > 21474.83 || + blue_x > 21474.83 || blue_y > 21474.83) + { + png_warning(png_ptr, + "Ignoring attempt to set chromaticity value exceeding 21474.83"); + return; + } + + info_ptr->x_white = (float)white_x; + info_ptr->y_white = (float)white_y; + info_ptr->x_red = (float)red_x; + info_ptr->y_red = (float)red_y; + info_ptr->x_green = (float)green_x; + info_ptr->y_green = (float)green_y; + info_ptr->x_blue = (float)blue_x; + info_ptr->y_blue = (float)blue_y; +#ifdef PNG_FIXED_POINT_SUPPORTED + info_ptr->int_x_white = (png_fixed_point)(white_x*100000.+0.5); + info_ptr->int_y_white = (png_fixed_point)(white_y*100000.+0.5); + info_ptr->int_x_red = (png_fixed_point)( red_x*100000.+0.5); + info_ptr->int_y_red = (png_fixed_point)( red_y*100000.+0.5); + info_ptr->int_x_green = (png_fixed_point)(green_x*100000.+0.5); + info_ptr->int_y_green = (png_fixed_point)(green_y*100000.+0.5); + info_ptr->int_x_blue = (png_fixed_point)( blue_x*100000.+0.5); + info_ptr->int_y_blue = (png_fixed_point)( blue_y*100000.+0.5); +#endif + info_ptr->valid |= PNG_INFO_cHRM; +} +#endif +#ifdef PNG_FIXED_POINT_SUPPORTED +void PNGAPI +png_set_cHRM_fixed(png_structp png_ptr, png_infop info_ptr, + png_fixed_point white_x, png_fixed_point white_y, png_fixed_point red_x, + png_fixed_point red_y, png_fixed_point green_x, png_fixed_point green_y, + png_fixed_point blue_x, png_fixed_point blue_y) +{ + png_debug1(1, "in %s storage function\n", "cHRM"); + if (png_ptr == NULL || info_ptr == NULL) + return; + + if (white_x < 0 || white_y < 0 || + red_x < 0 || red_y < 0 || + green_x < 0 || green_y < 0 || + blue_x < 0 || blue_y < 0) + { + png_warning(png_ptr, + "Ignoring attempt to set negative chromaticity value"); + return; + } +#ifdef PNG_FLOATING_POINT_SUPPORTED + if (white_x > (double) PNG_UINT_31_MAX || + white_y > (double) PNG_UINT_31_MAX || + red_x > (double) PNG_UINT_31_MAX || + red_y > (double) PNG_UINT_31_MAX || + green_x > (double) PNG_UINT_31_MAX || + green_y > (double) PNG_UINT_31_MAX || + blue_x > (double) PNG_UINT_31_MAX || + blue_y > (double) PNG_UINT_31_MAX) +#else + if (white_x > (png_fixed_point) PNG_UINT_31_MAX/100000L || + white_y > (png_fixed_point) PNG_UINT_31_MAX/100000L || + red_x > (png_fixed_point) PNG_UINT_31_MAX/100000L || + red_y > (png_fixed_point) PNG_UINT_31_MAX/100000L || + green_x > (png_fixed_point) PNG_UINT_31_MAX/100000L || + green_y > (png_fixed_point) PNG_UINT_31_MAX/100000L || + blue_x > (png_fixed_point) PNG_UINT_31_MAX/100000L || + blue_y > (png_fixed_point) PNG_UINT_31_MAX/100000L) +#endif + { + png_warning(png_ptr, + "Ignoring attempt to set chromaticity value exceeding 21474.83"); + return; + } + info_ptr->int_x_white = white_x; + info_ptr->int_y_white = white_y; + info_ptr->int_x_red = red_x; + info_ptr->int_y_red = red_y; + info_ptr->int_x_green = green_x; + info_ptr->int_y_green = green_y; + info_ptr->int_x_blue = blue_x; + info_ptr->int_y_blue = blue_y; +#ifdef PNG_FLOATING_POINT_SUPPORTED + info_ptr->x_white = (float)(white_x/100000.); + info_ptr->y_white = (float)(white_y/100000.); + info_ptr->x_red = (float)( red_x/100000.); + info_ptr->y_red = (float)( red_y/100000.); + info_ptr->x_green = (float)(green_x/100000.); + info_ptr->y_green = (float)(green_y/100000.); + info_ptr->x_blue = (float)( blue_x/100000.); + info_ptr->y_blue = (float)( blue_y/100000.); +#endif + info_ptr->valid |= PNG_INFO_cHRM; +} +#endif +#endif + +#if defined(PNG_gAMA_SUPPORTED) +#ifdef PNG_FLOATING_POINT_SUPPORTED +void PNGAPI +png_set_gAMA(png_structp png_ptr, png_infop info_ptr, double file_gamma) +{ + double gamma; + png_debug1(1, "in %s storage function\n", "gAMA"); + if (png_ptr == NULL || info_ptr == NULL) + return; + + /* Check for overflow */ + if (file_gamma > 21474.83) + { + png_warning(png_ptr, "Limiting gamma to 21474.83"); + gamma=21474.83; + } + else + gamma=file_gamma; + info_ptr->gamma = (float)gamma; +#ifdef PNG_FIXED_POINT_SUPPORTED + info_ptr->int_gamma = (int)(gamma*100000.+.5); +#endif + info_ptr->valid |= PNG_INFO_gAMA; + if(gamma == 0.0) + png_warning(png_ptr, "Setting gamma=0"); +} +#endif +void PNGAPI +png_set_gAMA_fixed(png_structp png_ptr, png_infop info_ptr, png_fixed_point + int_gamma) +{ + png_fixed_point gamma; + + png_debug1(1, "in %s storage function\n", "gAMA"); + if (png_ptr == NULL || info_ptr == NULL) + return; + + if (int_gamma > (png_fixed_point) PNG_UINT_31_MAX) + { + png_warning(png_ptr, "Limiting gamma to 21474.83"); + gamma=PNG_UINT_31_MAX; + } + else + { + if (int_gamma < 0) + { + png_warning(png_ptr, "Setting negative gamma to zero"); + gamma=0; + } + else + gamma=int_gamma; + } +#ifdef PNG_FLOATING_POINT_SUPPORTED + info_ptr->gamma = (float)(gamma/100000.); +#endif +#ifdef PNG_FIXED_POINT_SUPPORTED + info_ptr->int_gamma = gamma; +#endif + info_ptr->valid |= PNG_INFO_gAMA; + if(gamma == 0) + png_warning(png_ptr, "Setting gamma=0"); +} +#endif + +#if defined(PNG_hIST_SUPPORTED) +void PNGAPI +png_set_hIST(png_structp png_ptr, png_infop info_ptr, png_uint_16p hist) +{ + int i; + + png_debug1(1, "in %s storage function\n", "hIST"); + if (png_ptr == NULL || info_ptr == NULL) + return; + if (info_ptr->num_palette <= 0 || info_ptr->num_palette + > PNG_MAX_PALETTE_LENGTH) + { + png_warning(png_ptr, + "Invalid palette size, hIST allocation skipped."); + return; + } + +#ifdef PNG_FREE_ME_SUPPORTED + png_free_data(png_ptr, info_ptr, PNG_FREE_HIST, 0); +#endif + /* Changed from info->num_palette to PNG_MAX_PALETTE_LENGTH in version + 1.2.1 */ + png_ptr->hist = (png_uint_16p)png_malloc_warn(png_ptr, + (png_uint_32)(PNG_MAX_PALETTE_LENGTH * png_sizeof (png_uint_16))); + if (png_ptr->hist == NULL) + { + png_warning(png_ptr, "Insufficient memory for hIST chunk data."); + return; + } + + for (i = 0; i < info_ptr->num_palette; i++) + png_ptr->hist[i] = hist[i]; + info_ptr->hist = png_ptr->hist; + info_ptr->valid |= PNG_INFO_hIST; + +#ifdef PNG_FREE_ME_SUPPORTED + info_ptr->free_me |= PNG_FREE_HIST; +#else + png_ptr->flags |= PNG_FLAG_FREE_HIST; +#endif +} +#endif + +void PNGAPI +png_set_IHDR(png_structp png_ptr, png_infop info_ptr, + png_uint_32 width, png_uint_32 height, int bit_depth, + int color_type, int interlace_type, int compression_type, + int filter_type) +{ + png_debug1(1, "in %s storage function\n", "IHDR"); + if (png_ptr == NULL || info_ptr == NULL) + return; + + /* check for width and height valid values */ + if (width == 0 || height == 0) + png_error(png_ptr, "Image width or height is zero in IHDR"); +#ifdef PNG_SET_USER_LIMITS_SUPPORTED + if (width > png_ptr->user_width_max || height > png_ptr->user_height_max) + png_error(png_ptr, "image size exceeds user limits in IHDR"); +#else + if (width > PNG_USER_WIDTH_MAX || height > PNG_USER_HEIGHT_MAX) + png_error(png_ptr, "image size exceeds user limits in IHDR"); +#endif + if (width > PNG_UINT_31_MAX || height > PNG_UINT_31_MAX) + png_error(png_ptr, "Invalid image size in IHDR"); + if ( width > (PNG_UINT_32_MAX + >> 3) /* 8-byte RGBA pixels */ + - 64 /* bigrowbuf hack */ + - 1 /* filter byte */ + - 7*8 /* rounding of width to multiple of 8 pixels */ + - 8) /* extra max_pixel_depth pad */ + png_warning(png_ptr, "Width is too large for libpng to process pixels"); + + /* check other values */ + if (bit_depth != 1 && bit_depth != 2 && bit_depth != 4 && + bit_depth != 8 && bit_depth != 16) + png_error(png_ptr, "Invalid bit depth in IHDR"); + + if (color_type < 0 || color_type == 1 || + color_type == 5 || color_type > 6) + png_error(png_ptr, "Invalid color type in IHDR"); + + if (((color_type == PNG_COLOR_TYPE_PALETTE) && bit_depth > 8) || + ((color_type == PNG_COLOR_TYPE_RGB || + color_type == PNG_COLOR_TYPE_GRAY_ALPHA || + color_type == PNG_COLOR_TYPE_RGB_ALPHA) && bit_depth < 8)) + png_error(png_ptr, "Invalid color type/bit depth combination in IHDR"); + + if (interlace_type >= PNG_INTERLACE_LAST) + png_error(png_ptr, "Unknown interlace method in IHDR"); + + if (compression_type != PNG_COMPRESSION_TYPE_BASE) + png_error(png_ptr, "Unknown compression method in IHDR"); + +#if defined(PNG_MNG_FEATURES_SUPPORTED) + /* Accept filter_method 64 (intrapixel differencing) only if + * 1. Libpng was compiled with PNG_MNG_FEATURES_SUPPORTED and + * 2. Libpng did not read a PNG signature (this filter_method is only + * used in PNG datastreams that are embedded in MNG datastreams) and + * 3. The application called png_permit_mng_features with a mask that + * included PNG_FLAG_MNG_FILTER_64 and + * 4. The filter_method is 64 and + * 5. The color_type is RGB or RGBA + */ + if((png_ptr->mode&PNG_HAVE_PNG_SIGNATURE)&&png_ptr->mng_features_permitted) + png_warning(png_ptr,"MNG features are not allowed in a PNG datastream"); + if(filter_type != PNG_FILTER_TYPE_BASE) + { + if(!((png_ptr->mng_features_permitted & PNG_FLAG_MNG_FILTER_64) && + (filter_type == PNG_INTRAPIXEL_DIFFERENCING) && + ((png_ptr->mode&PNG_HAVE_PNG_SIGNATURE) == 0) && + (color_type == PNG_COLOR_TYPE_RGB || + color_type == PNG_COLOR_TYPE_RGB_ALPHA))) + png_error(png_ptr, "Unknown filter method in IHDR"); + if(png_ptr->mode&PNG_HAVE_PNG_SIGNATURE) + png_warning(png_ptr, "Invalid filter method in IHDR"); + } +#else + if(filter_type != PNG_FILTER_TYPE_BASE) + png_error(png_ptr, "Unknown filter method in IHDR"); +#endif + + info_ptr->width = width; + info_ptr->height = height; + info_ptr->bit_depth = (png_byte)bit_depth; + info_ptr->color_type =(png_byte) color_type; + info_ptr->compression_type = (png_byte)compression_type; + info_ptr->filter_type = (png_byte)filter_type; + info_ptr->interlace_type = (png_byte)interlace_type; + if (info_ptr->color_type == PNG_COLOR_TYPE_PALETTE) + info_ptr->channels = 1; + else if (info_ptr->color_type & PNG_COLOR_MASK_COLOR) + info_ptr->channels = 3; + else + info_ptr->channels = 1; + if (info_ptr->color_type & PNG_COLOR_MASK_ALPHA) + info_ptr->channels++; + info_ptr->pixel_depth = (png_byte)(info_ptr->channels * info_ptr->bit_depth); + + /* check for potential overflow */ + if ( width > (PNG_UINT_32_MAX + >> 3) /* 8-byte RGBA pixels */ + - 64 /* bigrowbuf hack */ + - 1 /* filter byte */ + - 7*8 /* rounding of width to multiple of 8 pixels */ + - 8) /* extra max_pixel_depth pad */ + info_ptr->rowbytes = (png_size_t)0; + else + info_ptr->rowbytes = PNG_ROWBYTES(info_ptr->pixel_depth,width); +} + +#if defined(PNG_oFFs_SUPPORTED) +void PNGAPI +png_set_oFFs(png_structp png_ptr, png_infop info_ptr, + png_int_32 offset_x, png_int_32 offset_y, int unit_type) +{ + png_debug1(1, "in %s storage function\n", "oFFs"); + if (png_ptr == NULL || info_ptr == NULL) + return; + + info_ptr->x_offset = offset_x; + info_ptr->y_offset = offset_y; + info_ptr->offset_unit_type = (png_byte)unit_type; + info_ptr->valid |= PNG_INFO_oFFs; +} +#endif + +#if defined(PNG_pCAL_SUPPORTED) +void PNGAPI +png_set_pCAL(png_structp png_ptr, png_infop info_ptr, + png_charp purpose, png_int_32 X0, png_int_32 X1, int type, int nparams, + png_charp units, png_charpp params) +{ + png_uint_32 length; + int i; + + png_debug1(1, "in %s storage function\n", "pCAL"); + if (png_ptr == NULL || info_ptr == NULL) + return; + + length = png_strlen(purpose) + 1; + png_debug1(3, "allocating purpose for info (%lu bytes)\n", length); + info_ptr->pcal_purpose = (png_charp)png_malloc_warn(png_ptr, length); + if (info_ptr->pcal_purpose == NULL) + { + png_warning(png_ptr, "Insufficient memory for pCAL purpose."); + return; + } + png_memcpy(info_ptr->pcal_purpose, purpose, (png_size_t)length); + + png_debug(3, "storing X0, X1, type, and nparams in info\n"); + info_ptr->pcal_X0 = X0; + info_ptr->pcal_X1 = X1; + info_ptr->pcal_type = (png_byte)type; + info_ptr->pcal_nparams = (png_byte)nparams; + + length = png_strlen(units) + 1; + png_debug1(3, "allocating units for info (%lu bytes)\n", length); + info_ptr->pcal_units = (png_charp)png_malloc_warn(png_ptr, length); + if (info_ptr->pcal_units == NULL) + { + png_warning(png_ptr, "Insufficient memory for pCAL units."); + return; + } + png_memcpy(info_ptr->pcal_units, units, (png_size_t)length); + + info_ptr->pcal_params = (png_charpp)png_malloc_warn(png_ptr, + (png_uint_32)((nparams + 1) * png_sizeof(png_charp))); + if (info_ptr->pcal_params == NULL) + { + png_warning(png_ptr, "Insufficient memory for pCAL params."); + return; + } + + info_ptr->pcal_params[nparams] = NULL; + + for (i = 0; i < nparams; i++) + { + length = png_strlen(params[i]) + 1; + png_debug2(3, "allocating parameter %d for info (%lu bytes)\n", i, length); + info_ptr->pcal_params[i] = (png_charp)png_malloc_warn(png_ptr, length); + if (info_ptr->pcal_params[i] == NULL) + { + png_warning(png_ptr, "Insufficient memory for pCAL parameter."); + return; + } + png_memcpy(info_ptr->pcal_params[i], params[i], (png_size_t)length); + } + + info_ptr->valid |= PNG_INFO_pCAL; +#ifdef PNG_FREE_ME_SUPPORTED + info_ptr->free_me |= PNG_FREE_PCAL; +#endif +} +#endif + +#if defined(PNG_READ_sCAL_SUPPORTED) || defined(PNG_WRITE_sCAL_SUPPORTED) +#ifdef PNG_FLOATING_POINT_SUPPORTED +void PNGAPI +png_set_sCAL(png_structp png_ptr, png_infop info_ptr, + int unit, double width, double height) +{ + png_debug1(1, "in %s storage function\n", "sCAL"); + if (png_ptr == NULL || info_ptr == NULL) + return; + + info_ptr->scal_unit = (png_byte)unit; + info_ptr->scal_pixel_width = width; + info_ptr->scal_pixel_height = height; + + info_ptr->valid |= PNG_INFO_sCAL; +} +#else +#ifdef PNG_FIXED_POINT_SUPPORTED +void PNGAPI +png_set_sCAL_s(png_structp png_ptr, png_infop info_ptr, + int unit, png_charp swidth, png_charp sheight) +{ + png_uint_32 length; + + png_debug1(1, "in %s storage function\n", "sCAL"); + if (png_ptr == NULL || info_ptr == NULL) + return; + + info_ptr->scal_unit = (png_byte)unit; + + length = png_strlen(swidth) + 1; + png_debug1(3, "allocating unit for info (%d bytes)\n", length); + info_ptr->scal_s_width = (png_charp)png_malloc_warn(png_ptr, length); + if (info_ptr->scal_s_width == NULL) + { + png_warning(png_ptr, "Memory allocation failed while processing sCAL."); + } + png_memcpy(info_ptr->scal_s_width, swidth, (png_size_t)length); + + length = png_strlen(sheight) + 1; + png_debug1(3, "allocating unit for info (%d bytes)\n", length); + info_ptr->scal_s_height = (png_charp)png_malloc_warn(png_ptr, length); + if (info_ptr->scal_s_height == NULL) + { + png_free (png_ptr, info_ptr->scal_s_width); + png_warning(png_ptr, "Memory allocation failed while processing sCAL."); + } + png_memcpy(info_ptr->scal_s_height, sheight, (png_size_t)length); + + info_ptr->valid |= PNG_INFO_sCAL; +#ifdef PNG_FREE_ME_SUPPORTED + info_ptr->free_me |= PNG_FREE_SCAL; +#endif +} +#endif +#endif +#endif + +#if defined(PNG_pHYs_SUPPORTED) +void PNGAPI +png_set_pHYs(png_structp png_ptr, png_infop info_ptr, + png_uint_32 res_x, png_uint_32 res_y, int unit_type) +{ + png_debug1(1, "in %s storage function\n", "pHYs"); + if (png_ptr == NULL || info_ptr == NULL) + return; + + info_ptr->x_pixels_per_unit = res_x; + info_ptr->y_pixels_per_unit = res_y; + info_ptr->phys_unit_type = (png_byte)unit_type; + info_ptr->valid |= PNG_INFO_pHYs; +} +#endif + +void PNGAPI +png_set_PLTE(png_structp png_ptr, png_infop info_ptr, + png_colorp palette, int num_palette) +{ + + png_debug1(1, "in %s storage function\n", "PLTE"); + if (png_ptr == NULL || info_ptr == NULL) + return; + + if (num_palette < 0 || num_palette > PNG_MAX_PALETTE_LENGTH) + { + if (info_ptr->color_type == PNG_COLOR_TYPE_PALETTE) + png_error(png_ptr, "Invalid palette length"); + else + { + png_warning(png_ptr, "Invalid palette length"); + return; + } + } + + /* + * It may not actually be necessary to set png_ptr->palette here; + * we do it for backward compatibility with the way the png_handle_tRNS + * function used to do the allocation. + */ +#ifdef PNG_FREE_ME_SUPPORTED + png_free_data(png_ptr, info_ptr, PNG_FREE_PLTE, 0); +#endif + + /* Changed in libpng-1.2.1 to allocate PNG_MAX_PALETTE_LENGTH instead + of num_palette entries, + in case of an invalid PNG file that has too-large sample values. */ + png_ptr->palette = (png_colorp)png_malloc(png_ptr, + PNG_MAX_PALETTE_LENGTH * png_sizeof(png_color)); + png_memset(png_ptr->palette, 0, PNG_MAX_PALETTE_LENGTH * + png_sizeof(png_color)); + png_memcpy(png_ptr->palette, palette, num_palette * png_sizeof (png_color)); + info_ptr->palette = png_ptr->palette; + info_ptr->num_palette = png_ptr->num_palette = (png_uint_16)num_palette; + +#ifdef PNG_FREE_ME_SUPPORTED + info_ptr->free_me |= PNG_FREE_PLTE; +#else + png_ptr->flags |= PNG_FLAG_FREE_PLTE; +#endif + + info_ptr->valid |= PNG_INFO_PLTE; +} + +#if defined(PNG_sBIT_SUPPORTED) +void PNGAPI +png_set_sBIT(png_structp png_ptr, png_infop info_ptr, + png_color_8p sig_bit) +{ + png_debug1(1, "in %s storage function\n", "sBIT"); + if (png_ptr == NULL || info_ptr == NULL) + return; + + png_memcpy(&(info_ptr->sig_bit), sig_bit, png_sizeof (png_color_8)); + info_ptr->valid |= PNG_INFO_sBIT; +} +#endif + +#if defined(PNG_sRGB_SUPPORTED) +void PNGAPI +png_set_sRGB(png_structp png_ptr, png_infop info_ptr, int intent) +{ + png_debug1(1, "in %s storage function\n", "sRGB"); + if (png_ptr == NULL || info_ptr == NULL) + return; + + info_ptr->srgb_intent = (png_byte)intent; + info_ptr->valid |= PNG_INFO_sRGB; +} + +void PNGAPI +png_set_sRGB_gAMA_and_cHRM(png_structp png_ptr, png_infop info_ptr, + int intent) +{ +#if defined(PNG_gAMA_SUPPORTED) +#ifdef PNG_FLOATING_POINT_SUPPORTED + float file_gamma; +#endif +#ifdef PNG_FIXED_POINT_SUPPORTED + png_fixed_point int_file_gamma; +#endif +#endif +#if defined(PNG_cHRM_SUPPORTED) +#ifdef PNG_FLOATING_POINT_SUPPORTED + float white_x, white_y, red_x, red_y, green_x, green_y, blue_x, blue_y; +#endif +#ifdef PNG_FIXED_POINT_SUPPORTED + png_fixed_point int_white_x, int_white_y, int_red_x, int_red_y, int_green_x, + int_green_y, int_blue_x, int_blue_y; +#endif +#endif + png_debug1(1, "in %s storage function\n", "sRGB_gAMA_and_cHRM"); + if (png_ptr == NULL || info_ptr == NULL) + return; + + png_set_sRGB(png_ptr, info_ptr, intent); + +#if defined(PNG_gAMA_SUPPORTED) +#ifdef PNG_FLOATING_POINT_SUPPORTED + file_gamma = (float).45455; + png_set_gAMA(png_ptr, info_ptr, file_gamma); +#endif +#ifdef PNG_FIXED_POINT_SUPPORTED + int_file_gamma = 45455L; + png_set_gAMA_fixed(png_ptr, info_ptr, int_file_gamma); +#endif +#endif + +#if defined(PNG_cHRM_SUPPORTED) +#ifdef PNG_FIXED_POINT_SUPPORTED + int_white_x = 31270L; + int_white_y = 32900L; + int_red_x = 64000L; + int_red_y = 33000L; + int_green_x = 30000L; + int_green_y = 60000L; + int_blue_x = 15000L; + int_blue_y = 6000L; + + png_set_cHRM_fixed(png_ptr, info_ptr, + int_white_x, int_white_y, int_red_x, int_red_y, int_green_x, int_green_y, + int_blue_x, int_blue_y); +#endif +#ifdef PNG_FLOATING_POINT_SUPPORTED + white_x = (float).3127; + white_y = (float).3290; + red_x = (float).64; + red_y = (float).33; + green_x = (float).30; + green_y = (float).60; + blue_x = (float).15; + blue_y = (float).06; + + png_set_cHRM(png_ptr, info_ptr, + white_x, white_y, red_x, red_y, green_x, green_y, blue_x, blue_y); +#endif +#endif +} +#endif + + +#if defined(PNG_iCCP_SUPPORTED) +void PNGAPI +png_set_iCCP(png_structp png_ptr, png_infop info_ptr, + png_charp name, int compression_type, + png_charp profile, png_uint_32 proflen) +{ + png_charp new_iccp_name; + png_charp new_iccp_profile; + + png_debug1(1, "in %s storage function\n", "iCCP"); + if (png_ptr == NULL || info_ptr == NULL || name == NULL || profile == NULL) + return; + + new_iccp_name = (png_charp)png_malloc_warn(png_ptr, png_strlen(name)+1); + if (new_iccp_name == NULL) + { + png_warning(png_ptr, "Insufficient memory to process iCCP chunk."); + return; + } + png_strcpy(new_iccp_name, name); + new_iccp_profile = (png_charp)png_malloc_warn(png_ptr, proflen); + if (new_iccp_profile == NULL) + { + png_free (png_ptr, new_iccp_name); + png_warning(png_ptr, "Insufficient memory to process iCCP profile."); + return; + } + png_memcpy(new_iccp_profile, profile, (png_size_t)proflen); + + png_free_data(png_ptr, info_ptr, PNG_FREE_ICCP, 0); + + info_ptr->iccp_proflen = proflen; + info_ptr->iccp_name = new_iccp_name; + info_ptr->iccp_profile = new_iccp_profile; + /* Compression is always zero but is here so the API and info structure + * does not have to change if we introduce multiple compression types */ + info_ptr->iccp_compression = (png_byte)compression_type; +#ifdef PNG_FREE_ME_SUPPORTED + info_ptr->free_me |= PNG_FREE_ICCP; +#endif + info_ptr->valid |= PNG_INFO_iCCP; +} +#endif + +#if defined(PNG_TEXT_SUPPORTED) +void PNGAPI +png_set_text(png_structp png_ptr, png_infop info_ptr, png_textp text_ptr, + int num_text) +{ + int ret; + ret=png_set_text_2(png_ptr, info_ptr, text_ptr, num_text); + if (ret) + png_error(png_ptr, "Insufficient memory to store text"); +} + +int /* PRIVATE */ +png_set_text_2(png_structp png_ptr, png_infop info_ptr, png_textp text_ptr, + int num_text) +{ + int i; + + png_debug1(1, "in %s storage function\n", (png_ptr->chunk_name[0] == '\0' ? + "text" : (png_const_charp)png_ptr->chunk_name)); + + if (png_ptr == NULL || info_ptr == NULL || num_text == 0) + return(0); + + /* Make sure we have enough space in the "text" array in info_struct + * to hold all of the incoming text_ptr objects. + */ + if (info_ptr->num_text + num_text > info_ptr->max_text) + { + if (info_ptr->text != NULL) + { + png_textp old_text; + int old_max; + + old_max = info_ptr->max_text; + info_ptr->max_text = info_ptr->num_text + num_text + 8; + old_text = info_ptr->text; + info_ptr->text = (png_textp)png_malloc_warn(png_ptr, + (png_uint_32)(info_ptr->max_text * png_sizeof (png_text))); + if (info_ptr->text == NULL) + { + png_free(png_ptr, old_text); + return(1); + } + png_memcpy(info_ptr->text, old_text, (png_size_t)(old_max * + png_sizeof(png_text))); + png_free(png_ptr, old_text); + } + else + { + info_ptr->max_text = num_text + 8; + info_ptr->num_text = 0; + info_ptr->text = (png_textp)png_malloc_warn(png_ptr, + (png_uint_32)(info_ptr->max_text * png_sizeof (png_text))); + if (info_ptr->text == NULL) + return(1); +#ifdef PNG_FREE_ME_SUPPORTED + info_ptr->free_me |= PNG_FREE_TEXT; +#endif + } + png_debug1(3, "allocated %d entries for info_ptr->text\n", + info_ptr->max_text); + } + for (i = 0; i < num_text; i++) + { + png_size_t text_length,key_len; + png_size_t lang_len,lang_key_len; + png_textp textp = &(info_ptr->text[info_ptr->num_text]); + + if (text_ptr[i].key == NULL) + continue; + + key_len = png_strlen(text_ptr[i].key); + + if(text_ptr[i].compression <= 0) + { + lang_len = 0; + lang_key_len = 0; + } + else +#ifdef PNG_iTXt_SUPPORTED + { + /* set iTXt data */ + if (text_ptr[i].lang != NULL) + lang_len = png_strlen(text_ptr[i].lang); + else + lang_len = 0; + if (text_ptr[i].lang_key != NULL) + lang_key_len = png_strlen(text_ptr[i].lang_key); + else + lang_key_len = 0; + } +#else + { + png_warning(png_ptr, "iTXt chunk not supported."); + continue; + } +#endif + + if (text_ptr[i].text == NULL || text_ptr[i].text[0] == '\0') + { + text_length = 0; +#ifdef PNG_iTXt_SUPPORTED + if(text_ptr[i].compression > 0) + textp->compression = PNG_ITXT_COMPRESSION_NONE; + else +#endif + textp->compression = PNG_TEXT_COMPRESSION_NONE; + } + else + { + text_length = png_strlen(text_ptr[i].text); + textp->compression = text_ptr[i].compression; + } + + textp->key = (png_charp)png_malloc_warn(png_ptr, + (png_uint_32)(key_len + text_length + lang_len + lang_key_len + 4)); + if (textp->key == NULL) + return(1); + png_debug2(2, "Allocated %lu bytes at %x in png_set_text\n", + (png_uint_32)(key_len + lang_len + lang_key_len + text_length + 4), + (int)textp->key); + + png_memcpy(textp->key, text_ptr[i].key, + (png_size_t)(key_len)); + *(textp->key+key_len) = '\0'; +#ifdef PNG_iTXt_SUPPORTED + if (text_ptr[i].compression > 0) + { + textp->lang=textp->key + key_len + 1; + png_memcpy(textp->lang, text_ptr[i].lang, lang_len); + *(textp->lang+lang_len) = '\0'; + textp->lang_key=textp->lang + lang_len + 1; + png_memcpy(textp->lang_key, text_ptr[i].lang_key, lang_key_len); + *(textp->lang_key+lang_key_len) = '\0'; + textp->text=textp->lang_key + lang_key_len + 1; + } + else +#endif + { +#ifdef PNG_iTXt_SUPPORTED + textp->lang=NULL; + textp->lang_key=NULL; +#endif + textp->text=textp->key + key_len + 1; + } + if(text_length) + png_memcpy(textp->text, text_ptr[i].text, + (png_size_t)(text_length)); + *(textp->text+text_length) = '\0'; + +#ifdef PNG_iTXt_SUPPORTED + if(textp->compression > 0) + { + textp->text_length = 0; + textp->itxt_length = text_length; + } + else +#endif + { + textp->text_length = text_length; +#ifdef PNG_iTXt_SUPPORTED + textp->itxt_length = 0; +#endif + } + info_ptr->text[info_ptr->num_text]= *textp; + info_ptr->num_text++; + png_debug1(3, "transferred text chunk %d\n", info_ptr->num_text); + } + return(0); +} +#endif + +#if defined(PNG_tIME_SUPPORTED) +void PNGAPI +png_set_tIME(png_structp png_ptr, png_infop info_ptr, png_timep mod_time) +{ + png_debug1(1, "in %s storage function\n", "tIME"); + if (png_ptr == NULL || info_ptr == NULL || + (png_ptr->mode & PNG_WROTE_tIME)) + return; + + png_memcpy(&(info_ptr->mod_time), mod_time, png_sizeof (png_time)); + info_ptr->valid |= PNG_INFO_tIME; +} +#endif + +#if defined(PNG_tRNS_SUPPORTED) +void PNGAPI +png_set_tRNS(png_structp png_ptr, png_infop info_ptr, + png_bytep trans, int num_trans, png_color_16p trans_values) +{ + png_debug1(1, "in %s storage function\n", "tRNS"); + if (png_ptr == NULL || info_ptr == NULL) + return; + + if (trans != NULL) + { + /* + * It may not actually be necessary to set png_ptr->trans here; + * we do it for backward compatibility with the way the png_handle_tRNS + * function used to do the allocation. + */ +#ifdef PNG_FREE_ME_SUPPORTED + png_free_data(png_ptr, info_ptr, PNG_FREE_TRNS, 0); +#endif + /* Changed from num_trans to PNG_MAX_PALETTE_LENGTH in version 1.2.1 */ + png_ptr->trans = info_ptr->trans = (png_bytep)png_malloc(png_ptr, + (png_uint_32)PNG_MAX_PALETTE_LENGTH); + if (num_trans <= PNG_MAX_PALETTE_LENGTH) + png_memcpy(info_ptr->trans, trans, (png_size_t)num_trans); +#ifdef PNG_FREE_ME_SUPPORTED + info_ptr->free_me |= PNG_FREE_TRNS; +#else + png_ptr->flags |= PNG_FLAG_FREE_TRNS; +#endif + } + + if (trans_values != NULL) + { + png_memcpy(&(info_ptr->trans_values), trans_values, + png_sizeof(png_color_16)); + if (num_trans == 0) + num_trans = 1; + } + info_ptr->num_trans = (png_uint_16)num_trans; + info_ptr->valid |= PNG_INFO_tRNS; +} +#endif + +#if defined(PNG_sPLT_SUPPORTED) +void PNGAPI +png_set_sPLT(png_structp png_ptr, + png_infop info_ptr, png_sPLT_tp entries, int nentries) +{ + png_sPLT_tp np; + int i; + + if (png_ptr == NULL || info_ptr == NULL) + return; + + np = (png_sPLT_tp)png_malloc_warn(png_ptr, + (info_ptr->splt_palettes_num + nentries) * png_sizeof(png_sPLT_t)); + if (np == NULL) + { + png_warning(png_ptr, "No memory for sPLT palettes."); + return; + } + + png_memcpy(np, info_ptr->splt_palettes, + info_ptr->splt_palettes_num * png_sizeof(png_sPLT_t)); + png_free(png_ptr, info_ptr->splt_palettes); + info_ptr->splt_palettes=NULL; + + for (i = 0; i < nentries; i++) + { + png_sPLT_tp to = np + info_ptr->splt_palettes_num + i; + png_sPLT_tp from = entries + i; + + to->name = (png_charp)png_malloc(png_ptr, + png_strlen(from->name) + 1); + /* TODO: use png_malloc_warn */ + png_strcpy(to->name, from->name); + to->entries = (png_sPLT_entryp)png_malloc(png_ptr, + from->nentries * png_sizeof(png_sPLT_t)); + /* TODO: use png_malloc_warn */ + png_memcpy(to->entries, from->entries, + from->nentries * png_sizeof(png_sPLT_t)); + to->nentries = from->nentries; + to->depth = from->depth; + } + + info_ptr->splt_palettes = np; + info_ptr->splt_palettes_num += nentries; + info_ptr->valid |= PNG_INFO_sPLT; +#ifdef PNG_FREE_ME_SUPPORTED + info_ptr->free_me |= PNG_FREE_SPLT; +#endif +} +#endif /* PNG_sPLT_SUPPORTED */ + +#if defined(PNG_UNKNOWN_CHUNKS_SUPPORTED) +void PNGAPI +png_set_unknown_chunks(png_structp png_ptr, + png_infop info_ptr, png_unknown_chunkp unknowns, int num_unknowns) +{ + png_unknown_chunkp np; + int i; + + if (png_ptr == NULL || info_ptr == NULL || num_unknowns == 0) + return; + + np = (png_unknown_chunkp)png_malloc_warn(png_ptr, + (info_ptr->unknown_chunks_num + num_unknowns) * + png_sizeof(png_unknown_chunk)); + if (np == NULL) + { + png_warning(png_ptr, "Out of memory while processing unknown chunk."); + return; + } + + png_memcpy(np, info_ptr->unknown_chunks, + info_ptr->unknown_chunks_num * png_sizeof(png_unknown_chunk)); + png_free(png_ptr, info_ptr->unknown_chunks); + info_ptr->unknown_chunks=NULL; + + for (i = 0; i < num_unknowns; i++) + { + png_unknown_chunkp to = np + info_ptr->unknown_chunks_num + i; + png_unknown_chunkp from = unknowns + i; + + png_strncpy((png_charp)to->name, (png_charp)from->name, 5); + to->data = (png_bytep)png_malloc_warn(png_ptr, from->size); + if (to->data == NULL) + { + png_warning(png_ptr, "Out of memory processing unknown chunk."); + } + else + { + png_memcpy(to->data, from->data, from->size); + to->size = from->size; + + /* note our location in the read or write sequence */ + to->location = (png_byte)(png_ptr->mode & 0xff); + } + } + + info_ptr->unknown_chunks = np; + info_ptr->unknown_chunks_num += num_unknowns; +#ifdef PNG_FREE_ME_SUPPORTED + info_ptr->free_me |= PNG_FREE_UNKN; +#endif +} +void PNGAPI +png_set_unknown_chunk_location(png_structp png_ptr, png_infop info_ptr, + int chunk, int location) +{ + if(png_ptr != NULL && info_ptr != NULL && chunk >= 0 && chunk < + (int)info_ptr->unknown_chunks_num) + info_ptr->unknown_chunks[chunk].location = (png_byte)location; +} +#endif + +#if defined(PNG_1_0_X) || defined(PNG_1_2_X) +#if defined(PNG_READ_EMPTY_PLTE_SUPPORTED) || \ + defined(PNG_WRITE_EMPTY_PLTE_SUPPORTED) +void PNGAPI +png_permit_empty_plte (png_structp png_ptr, int empty_plte_permitted) +{ + /* This function is deprecated in favor of png_permit_mng_features() + and will be removed from libpng-1.3.0 */ + png_debug(1, "in png_permit_empty_plte, DEPRECATED.\n"); + if (png_ptr == NULL) + return; + png_ptr->mng_features_permitted = (png_byte) + ((png_ptr->mng_features_permitted & (~(PNG_FLAG_MNG_EMPTY_PLTE))) | + ((empty_plte_permitted & PNG_FLAG_MNG_EMPTY_PLTE))); +} +#endif +#endif + +#if defined(PNG_MNG_FEATURES_SUPPORTED) +png_uint_32 PNGAPI +png_permit_mng_features (png_structp png_ptr, png_uint_32 mng_features) +{ + png_debug(1, "in png_permit_mng_features\n"); + if (png_ptr == NULL) + return (png_uint_32)0; + png_ptr->mng_features_permitted = + (png_byte)(mng_features & PNG_ALL_MNG_FEATURES); + return (png_uint_32)png_ptr->mng_features_permitted; +} +#endif + +#if defined(PNG_UNKNOWN_CHUNKS_SUPPORTED) +void PNGAPI +png_set_keep_unknown_chunks(png_structp png_ptr, int keep, png_bytep + chunk_list, int num_chunks) +{ + png_bytep new_list, p; + int i, old_num_chunks; + if (png_ptr == NULL) + return; + if (num_chunks == 0) + { + if(keep == PNG_HANDLE_CHUNK_ALWAYS || keep == PNG_HANDLE_CHUNK_IF_SAFE) + png_ptr->flags |= PNG_FLAG_KEEP_UNKNOWN_CHUNKS; + else + png_ptr->flags &= ~PNG_FLAG_KEEP_UNKNOWN_CHUNKS; + + if(keep == PNG_HANDLE_CHUNK_ALWAYS) + png_ptr->flags |= PNG_FLAG_KEEP_UNSAFE_CHUNKS; + else + png_ptr->flags &= ~PNG_FLAG_KEEP_UNSAFE_CHUNKS; + return; + } + if (chunk_list == NULL) + return; + old_num_chunks=png_ptr->num_chunk_list; + new_list=(png_bytep)png_malloc(png_ptr, + (png_uint_32)(5*(num_chunks+old_num_chunks))); + if(png_ptr->chunk_list != NULL) + { + png_memcpy(new_list, png_ptr->chunk_list, + (png_size_t)(5*old_num_chunks)); + png_free(png_ptr, png_ptr->chunk_list); + png_ptr->chunk_list=NULL; + } + png_memcpy(new_list+5*old_num_chunks, chunk_list, + (png_size_t)(5*num_chunks)); + for (p=new_list+5*old_num_chunks+4, i=0; inum_chunk_list=old_num_chunks+num_chunks; + png_ptr->chunk_list=new_list; +#ifdef PNG_FREE_ME_SUPPORTED + png_ptr->free_me |= PNG_FREE_LIST; +#endif +} +#endif + +#if defined(PNG_READ_USER_CHUNKS_SUPPORTED) +void PNGAPI +png_set_read_user_chunk_fn(png_structp png_ptr, png_voidp user_chunk_ptr, + png_user_chunk_ptr read_user_chunk_fn) +{ + png_debug(1, "in png_set_read_user_chunk_fn\n"); + if (png_ptr == NULL) + return; + png_ptr->read_user_chunk_fn = read_user_chunk_fn; + png_ptr->user_chunk_ptr = user_chunk_ptr; +} +#endif + +#if defined(PNG_INFO_IMAGE_SUPPORTED) +void PNGAPI +png_set_rows(png_structp png_ptr, png_infop info_ptr, png_bytepp row_pointers) +{ + png_debug1(1, "in %s storage function\n", "rows"); + + if (png_ptr == NULL || info_ptr == NULL) + return; + + if(info_ptr->row_pointers && (info_ptr->row_pointers != row_pointers)) + png_free_data(png_ptr, info_ptr, PNG_FREE_ROWS, 0); + info_ptr->row_pointers = row_pointers; + if(row_pointers) + info_ptr->valid |= PNG_INFO_IDAT; +} +#endif + +#ifdef PNG_WRITE_SUPPORTED +void PNGAPI +png_set_compression_buffer_size(png_structp png_ptr, png_uint_32 size) +{ + if (png_ptr == NULL) + return; + if(png_ptr->zbuf) + png_free(png_ptr, png_ptr->zbuf); + png_ptr->zbuf_size = (png_size_t)size; + png_ptr->zbuf = (png_bytep)png_malloc(png_ptr, size); + png_ptr->zstream.next_out = png_ptr->zbuf; + png_ptr->zstream.avail_out = (uInt)png_ptr->zbuf_size; +} +#endif + +void PNGAPI +png_set_invalid(png_structp png_ptr, png_infop info_ptr, int mask) +{ + if (png_ptr && info_ptr) + info_ptr->valid &= ~(mask); +} + + +#ifndef PNG_1_0_X +#ifdef PNG_ASSEMBLER_CODE_SUPPORTED +/* this function was added to libpng 1.2.0 and should always exist by default */ +void PNGAPI +png_set_asm_flags (png_structp png_ptr, png_uint_32 asm_flags) +{ + png_uint_32 settable_asm_flags; + png_uint_32 settable_mmx_flags; + + if (png_ptr == NULL) + return; + + settable_mmx_flags = +#ifdef PNG_HAVE_ASSEMBLER_COMBINE_ROW + PNG_ASM_FLAG_MMX_READ_COMBINE_ROW | +#endif +#ifdef PNG_HAVE_ASSEMBLER_READ_INTERLACE + PNG_ASM_FLAG_MMX_READ_INTERLACE | +#endif +#ifdef PNG_HAVE_ASSEMBLER_READ_FILTER_ROW + PNG_ASM_FLAG_MMX_READ_FILTER_SUB | + PNG_ASM_FLAG_MMX_READ_FILTER_UP | + PNG_ASM_FLAG_MMX_READ_FILTER_AVG | + PNG_ASM_FLAG_MMX_READ_FILTER_PAETH | +#endif + 0; + + /* could be some non-MMX ones in the future, but not currently: */ + settable_asm_flags = settable_mmx_flags; + + if (!(png_ptr->asm_flags & PNG_ASM_FLAG_MMX_SUPPORT_COMPILED) || + !(png_ptr->asm_flags & PNG_ASM_FLAG_MMX_SUPPORT_IN_CPU)) + { + /* clear all MMX flags if MMX isn't supported */ + settable_asm_flags &= ~settable_mmx_flags; + png_ptr->asm_flags &= ~settable_mmx_flags; + } + + /* we're replacing the settable bits with those passed in by the user, + * so first zero them out of the master copy, then logical-OR in the + * allowed subset that was requested */ + + png_ptr->asm_flags &= ~settable_asm_flags; /* zero them */ + png_ptr->asm_flags |= (asm_flags & settable_asm_flags); /* set them */ +} +#endif /* ?PNG_ASSEMBLER_CODE_SUPPORTED */ + +#ifdef PNG_ASSEMBLER_CODE_SUPPORTED +/* this function was added to libpng 1.2.0 */ +void PNGAPI +png_set_mmx_thresholds (png_structp png_ptr, + png_byte mmx_bitdepth_threshold, + png_uint_32 mmx_rowbytes_threshold) +{ + if (png_ptr == NULL) + return; + png_ptr->mmx_bitdepth_threshold = mmx_bitdepth_threshold; + png_ptr->mmx_rowbytes_threshold = mmx_rowbytes_threshold; +} +#endif /* ?PNG_ASSEMBLER_CODE_SUPPORTED */ + +#ifdef PNG_SET_USER_LIMITS_SUPPORTED +/* this function was added to libpng 1.2.6 */ +void PNGAPI +png_set_user_limits (png_structp png_ptr, png_uint_32 user_width_max, + png_uint_32 user_height_max) +{ + /* Images with dimensions larger than these limits will be + * rejected by png_set_IHDR(). To accept any PNG datastream + * regardless of dimensions, set both limits to 0x7ffffffL. + */ + png_ptr->user_width_max = user_width_max; + png_ptr->user_height_max = user_height_max; +} +#endif /* ?PNG_SET_USER_LIMITS_SUPPORTED */ + +#endif /* ?PNG_1_0_X */ +#endif /* PNG_READ_SUPPORTED || PNG_WRITE_SUPPORTED */ diff --git a/src/win32/dependencies/libpng/pngtrans.c b/src/win32/dependencies/libpng/pngtrans.c new file mode 100644 index 00000000..9c4d67cd --- /dev/null +++ b/src/win32/dependencies/libpng/pngtrans.c @@ -0,0 +1,652 @@ + +/* pngtrans.c - transforms the data in a row (used by both readers and writers) + * + * Last changed in libpng 1.2.9 April 14, 2006 + * For conditions of distribution and use, see copyright notice in png.h + * Copyright (c) 1998-2006 Glenn Randers-Pehrson + * (Version 0.96 Copyright (c) 1996, 1997 Andreas Dilger) + * (Version 0.88 Copyright (c) 1995, 1996 Guy Eric Schalnat, Group 42, Inc.) + */ + +#define PNG_INTERNAL +#include "png.h" + +#if defined(PNG_READ_SUPPORTED) || defined(PNG_WRITE_SUPPORTED) +#if defined(PNG_READ_BGR_SUPPORTED) || defined(PNG_WRITE_BGR_SUPPORTED) +/* turn on BGR-to-RGB mapping */ +void PNGAPI +png_set_bgr(png_structp png_ptr) +{ + png_debug(1, "in png_set_bgr\n"); + png_ptr->transformations |= PNG_BGR; +} +#endif + +#if defined(PNG_READ_SWAP_SUPPORTED) || defined(PNG_WRITE_SWAP_SUPPORTED) +/* turn on 16 bit byte swapping */ +void PNGAPI +png_set_swap(png_structp png_ptr) +{ + png_debug(1, "in png_set_swap\n"); + if (png_ptr->bit_depth == 16) + png_ptr->transformations |= PNG_SWAP_BYTES; +} +#endif + +#if defined(PNG_READ_PACK_SUPPORTED) || defined(PNG_WRITE_PACK_SUPPORTED) +/* turn on pixel packing */ +void PNGAPI +png_set_packing(png_structp png_ptr) +{ + png_debug(1, "in png_set_packing\n"); + if (png_ptr->bit_depth < 8) + { + png_ptr->transformations |= PNG_PACK; + png_ptr->usr_bit_depth = 8; + } +} +#endif + +#if defined(PNG_READ_PACKSWAP_SUPPORTED)||defined(PNG_WRITE_PACKSWAP_SUPPORTED) +/* turn on packed pixel swapping */ +void PNGAPI +png_set_packswap(png_structp png_ptr) +{ + png_debug(1, "in png_set_packswap\n"); + if (png_ptr->bit_depth < 8) + png_ptr->transformations |= PNG_PACKSWAP; +} +#endif + +#if defined(PNG_READ_SHIFT_SUPPORTED) || defined(PNG_WRITE_SHIFT_SUPPORTED) +void PNGAPI +png_set_shift(png_structp png_ptr, png_color_8p true_bits) +{ + png_debug(1, "in png_set_shift\n"); + png_ptr->transformations |= PNG_SHIFT; + png_ptr->shift = *true_bits; +} +#endif + +#if defined(PNG_READ_INTERLACING_SUPPORTED) || \ + defined(PNG_WRITE_INTERLACING_SUPPORTED) +int PNGAPI +png_set_interlace_handling(png_structp png_ptr) +{ + png_debug(1, "in png_set_interlace handling\n"); + if (png_ptr->interlaced) + { + png_ptr->transformations |= PNG_INTERLACE; + return (7); + } + + return (1); +} +#endif + +#if defined(PNG_READ_FILLER_SUPPORTED) || defined(PNG_WRITE_FILLER_SUPPORTED) +/* Add a filler byte on read, or remove a filler or alpha byte on write. + * The filler type has changed in v0.95 to allow future 2-byte fillers + * for 48-bit input data, as well as to avoid problems with some compilers + * that don't like bytes as parameters. + */ +void PNGAPI +png_set_filler(png_structp png_ptr, png_uint_32 filler, int filler_loc) +{ + png_debug(1, "in png_set_filler\n"); + png_ptr->transformations |= PNG_FILLER; + png_ptr->filler = (png_byte)filler; + if (filler_loc == PNG_FILLER_AFTER) + png_ptr->flags |= PNG_FLAG_FILLER_AFTER; + else + png_ptr->flags &= ~PNG_FLAG_FILLER_AFTER; + + /* This should probably go in the "do_read_filler" routine. + * I attempted to do that in libpng-1.0.1a but that caused problems + * so I restored it in libpng-1.0.2a + */ + + if (png_ptr->color_type == PNG_COLOR_TYPE_RGB) + { + png_ptr->usr_channels = 4; + } + + /* Also I added this in libpng-1.0.2a (what happens when we expand + * a less-than-8-bit grayscale to GA? */ + + if (png_ptr->color_type == PNG_COLOR_TYPE_GRAY && png_ptr->bit_depth >= 8) + { + png_ptr->usr_channels = 2; + } +} + +#if !defined(PNG_1_0_X) +/* Added to libpng-1.2.7 */ +void PNGAPI +png_set_add_alpha(png_structp png_ptr, png_uint_32 filler, int filler_loc) +{ + png_debug(1, "in png_set_add_alpha\n"); + png_set_filler(png_ptr, filler, filler_loc); + png_ptr->transformations |= PNG_ADD_ALPHA; +} +#endif + +#endif + +#if defined(PNG_READ_SWAP_ALPHA_SUPPORTED) || \ + defined(PNG_WRITE_SWAP_ALPHA_SUPPORTED) +void PNGAPI +png_set_swap_alpha(png_structp png_ptr) +{ + png_debug(1, "in png_set_swap_alpha\n"); + png_ptr->transformations |= PNG_SWAP_ALPHA; +} +#endif + +#if defined(PNG_READ_INVERT_ALPHA_SUPPORTED) || \ + defined(PNG_WRITE_INVERT_ALPHA_SUPPORTED) +void PNGAPI +png_set_invert_alpha(png_structp png_ptr) +{ + png_debug(1, "in png_set_invert_alpha\n"); + png_ptr->transformations |= PNG_INVERT_ALPHA; +} +#endif + +#if defined(PNG_READ_INVERT_SUPPORTED) || defined(PNG_WRITE_INVERT_SUPPORTED) +void PNGAPI +png_set_invert_mono(png_structp png_ptr) +{ + png_debug(1, "in png_set_invert_mono\n"); + png_ptr->transformations |= PNG_INVERT_MONO; +} + +/* invert monochrome grayscale data */ +void /* PRIVATE */ +png_do_invert(png_row_infop row_info, png_bytep row) +{ + png_debug(1, "in png_do_invert\n"); + /* This test removed from libpng version 1.0.13 and 1.2.0: + * if (row_info->bit_depth == 1 && + */ +#if defined(PNG_USELESS_TESTS_SUPPORTED) + if (row == NULL || row_info == NULL) + return; +#endif + if (row_info->color_type == PNG_COLOR_TYPE_GRAY) + { + png_bytep rp = row; + png_uint_32 i; + png_uint_32 istop = row_info->rowbytes; + + for (i = 0; i < istop; i++) + { + *rp = (png_byte)(~(*rp)); + rp++; + } + } + else if (row_info->color_type == PNG_COLOR_TYPE_GRAY_ALPHA && + row_info->bit_depth == 8) + { + png_bytep rp = row; + png_uint_32 i; + png_uint_32 istop = row_info->rowbytes; + + for (i = 0; i < istop; i+=2) + { + *rp = (png_byte)(~(*rp)); + rp+=2; + } + } + else if (row_info->color_type == PNG_COLOR_TYPE_GRAY_ALPHA && + row_info->bit_depth == 16) + { + png_bytep rp = row; + png_uint_32 i; + png_uint_32 istop = row_info->rowbytes; + + for (i = 0; i < istop; i+=4) + { + *rp = (png_byte)(~(*rp)); + *(rp+1) = (png_byte)(~(*(rp+1))); + rp+=4; + } + } +} +#endif + +#if defined(PNG_READ_SWAP_SUPPORTED) || defined(PNG_WRITE_SWAP_SUPPORTED) +/* swaps byte order on 16 bit depth images */ +void /* PRIVATE */ +png_do_swap(png_row_infop row_info, png_bytep row) +{ + png_debug(1, "in png_do_swap\n"); + if ( +#if defined(PNG_USELESS_TESTS_SUPPORTED) + row != NULL && row_info != NULL && +#endif + row_info->bit_depth == 16) + { + png_bytep rp = row; + png_uint_32 i; + png_uint_32 istop= row_info->width * row_info->channels; + + for (i = 0; i < istop; i++, rp += 2) + { + png_byte t = *rp; + *rp = *(rp + 1); + *(rp + 1) = t; + } + } +} +#endif + +#if defined(PNG_READ_PACKSWAP_SUPPORTED)||defined(PNG_WRITE_PACKSWAP_SUPPORTED) +static PNG_CONST png_byte onebppswaptable[256] = { + 0x00, 0x80, 0x40, 0xC0, 0x20, 0xA0, 0x60, 0xE0, + 0x10, 0x90, 0x50, 0xD0, 0x30, 0xB0, 0x70, 0xF0, + 0x08, 0x88, 0x48, 0xC8, 0x28, 0xA8, 0x68, 0xE8, + 0x18, 0x98, 0x58, 0xD8, 0x38, 0xB8, 0x78, 0xF8, + 0x04, 0x84, 0x44, 0xC4, 0x24, 0xA4, 0x64, 0xE4, + 0x14, 0x94, 0x54, 0xD4, 0x34, 0xB4, 0x74, 0xF4, + 0x0C, 0x8C, 0x4C, 0xCC, 0x2C, 0xAC, 0x6C, 0xEC, + 0x1C, 0x9C, 0x5C, 0xDC, 0x3C, 0xBC, 0x7C, 0xFC, + 0x02, 0x82, 0x42, 0xC2, 0x22, 0xA2, 0x62, 0xE2, + 0x12, 0x92, 0x52, 0xD2, 0x32, 0xB2, 0x72, 0xF2, + 0x0A, 0x8A, 0x4A, 0xCA, 0x2A, 0xAA, 0x6A, 0xEA, + 0x1A, 0x9A, 0x5A, 0xDA, 0x3A, 0xBA, 0x7A, 0xFA, + 0x06, 0x86, 0x46, 0xC6, 0x26, 0xA6, 0x66, 0xE6, + 0x16, 0x96, 0x56, 0xD6, 0x36, 0xB6, 0x76, 0xF6, + 0x0E, 0x8E, 0x4E, 0xCE, 0x2E, 0xAE, 0x6E, 0xEE, + 0x1E, 0x9E, 0x5E, 0xDE, 0x3E, 0xBE, 0x7E, 0xFE, + 0x01, 0x81, 0x41, 0xC1, 0x21, 0xA1, 0x61, 0xE1, + 0x11, 0x91, 0x51, 0xD1, 0x31, 0xB1, 0x71, 0xF1, + 0x09, 0x89, 0x49, 0xC9, 0x29, 0xA9, 0x69, 0xE9, + 0x19, 0x99, 0x59, 0xD9, 0x39, 0xB9, 0x79, 0xF9, + 0x05, 0x85, 0x45, 0xC5, 0x25, 0xA5, 0x65, 0xE5, + 0x15, 0x95, 0x55, 0xD5, 0x35, 0xB5, 0x75, 0xF5, + 0x0D, 0x8D, 0x4D, 0xCD, 0x2D, 0xAD, 0x6D, 0xED, + 0x1D, 0x9D, 0x5D, 0xDD, 0x3D, 0xBD, 0x7D, 0xFD, + 0x03, 0x83, 0x43, 0xC3, 0x23, 0xA3, 0x63, 0xE3, + 0x13, 0x93, 0x53, 0xD3, 0x33, 0xB3, 0x73, 0xF3, + 0x0B, 0x8B, 0x4B, 0xCB, 0x2B, 0xAB, 0x6B, 0xEB, + 0x1B, 0x9B, 0x5B, 0xDB, 0x3B, 0xBB, 0x7B, 0xFB, + 0x07, 0x87, 0x47, 0xC7, 0x27, 0xA7, 0x67, 0xE7, + 0x17, 0x97, 0x57, 0xD7, 0x37, 0xB7, 0x77, 0xF7, + 0x0F, 0x8F, 0x4F, 0xCF, 0x2F, 0xAF, 0x6F, 0xEF, + 0x1F, 0x9F, 0x5F, 0xDF, 0x3F, 0xBF, 0x7F, 0xFF +}; + +static PNG_CONST png_byte twobppswaptable[256] = { + 0x00, 0x40, 0x80, 0xC0, 0x10, 0x50, 0x90, 0xD0, + 0x20, 0x60, 0xA0, 0xE0, 0x30, 0x70, 0xB0, 0xF0, + 0x04, 0x44, 0x84, 0xC4, 0x14, 0x54, 0x94, 0xD4, + 0x24, 0x64, 0xA4, 0xE4, 0x34, 0x74, 0xB4, 0xF4, + 0x08, 0x48, 0x88, 0xC8, 0x18, 0x58, 0x98, 0xD8, + 0x28, 0x68, 0xA8, 0xE8, 0x38, 0x78, 0xB8, 0xF8, + 0x0C, 0x4C, 0x8C, 0xCC, 0x1C, 0x5C, 0x9C, 0xDC, + 0x2C, 0x6C, 0xAC, 0xEC, 0x3C, 0x7C, 0xBC, 0xFC, + 0x01, 0x41, 0x81, 0xC1, 0x11, 0x51, 0x91, 0xD1, + 0x21, 0x61, 0xA1, 0xE1, 0x31, 0x71, 0xB1, 0xF1, + 0x05, 0x45, 0x85, 0xC5, 0x15, 0x55, 0x95, 0xD5, + 0x25, 0x65, 0xA5, 0xE5, 0x35, 0x75, 0xB5, 0xF5, + 0x09, 0x49, 0x89, 0xC9, 0x19, 0x59, 0x99, 0xD9, + 0x29, 0x69, 0xA9, 0xE9, 0x39, 0x79, 0xB9, 0xF9, + 0x0D, 0x4D, 0x8D, 0xCD, 0x1D, 0x5D, 0x9D, 0xDD, + 0x2D, 0x6D, 0xAD, 0xED, 0x3D, 0x7D, 0xBD, 0xFD, + 0x02, 0x42, 0x82, 0xC2, 0x12, 0x52, 0x92, 0xD2, + 0x22, 0x62, 0xA2, 0xE2, 0x32, 0x72, 0xB2, 0xF2, + 0x06, 0x46, 0x86, 0xC6, 0x16, 0x56, 0x96, 0xD6, + 0x26, 0x66, 0xA6, 0xE6, 0x36, 0x76, 0xB6, 0xF6, + 0x0A, 0x4A, 0x8A, 0xCA, 0x1A, 0x5A, 0x9A, 0xDA, + 0x2A, 0x6A, 0xAA, 0xEA, 0x3A, 0x7A, 0xBA, 0xFA, + 0x0E, 0x4E, 0x8E, 0xCE, 0x1E, 0x5E, 0x9E, 0xDE, + 0x2E, 0x6E, 0xAE, 0xEE, 0x3E, 0x7E, 0xBE, 0xFE, + 0x03, 0x43, 0x83, 0xC3, 0x13, 0x53, 0x93, 0xD3, + 0x23, 0x63, 0xA3, 0xE3, 0x33, 0x73, 0xB3, 0xF3, + 0x07, 0x47, 0x87, 0xC7, 0x17, 0x57, 0x97, 0xD7, + 0x27, 0x67, 0xA7, 0xE7, 0x37, 0x77, 0xB7, 0xF7, + 0x0B, 0x4B, 0x8B, 0xCB, 0x1B, 0x5B, 0x9B, 0xDB, + 0x2B, 0x6B, 0xAB, 0xEB, 0x3B, 0x7B, 0xBB, 0xFB, + 0x0F, 0x4F, 0x8F, 0xCF, 0x1F, 0x5F, 0x9F, 0xDF, + 0x2F, 0x6F, 0xAF, 0xEF, 0x3F, 0x7F, 0xBF, 0xFF +}; + +static PNG_CONST png_byte fourbppswaptable[256] = { + 0x00, 0x10, 0x20, 0x30, 0x40, 0x50, 0x60, 0x70, + 0x80, 0x90, 0xA0, 0xB0, 0xC0, 0xD0, 0xE0, 0xF0, + 0x01, 0x11, 0x21, 0x31, 0x41, 0x51, 0x61, 0x71, + 0x81, 0x91, 0xA1, 0xB1, 0xC1, 0xD1, 0xE1, 0xF1, + 0x02, 0x12, 0x22, 0x32, 0x42, 0x52, 0x62, 0x72, + 0x82, 0x92, 0xA2, 0xB2, 0xC2, 0xD2, 0xE2, 0xF2, + 0x03, 0x13, 0x23, 0x33, 0x43, 0x53, 0x63, 0x73, + 0x83, 0x93, 0xA3, 0xB3, 0xC3, 0xD3, 0xE3, 0xF3, + 0x04, 0x14, 0x24, 0x34, 0x44, 0x54, 0x64, 0x74, + 0x84, 0x94, 0xA4, 0xB4, 0xC4, 0xD4, 0xE4, 0xF4, + 0x05, 0x15, 0x25, 0x35, 0x45, 0x55, 0x65, 0x75, + 0x85, 0x95, 0xA5, 0xB5, 0xC5, 0xD5, 0xE5, 0xF5, + 0x06, 0x16, 0x26, 0x36, 0x46, 0x56, 0x66, 0x76, + 0x86, 0x96, 0xA6, 0xB6, 0xC6, 0xD6, 0xE6, 0xF6, + 0x07, 0x17, 0x27, 0x37, 0x47, 0x57, 0x67, 0x77, + 0x87, 0x97, 0xA7, 0xB7, 0xC7, 0xD7, 0xE7, 0xF7, + 0x08, 0x18, 0x28, 0x38, 0x48, 0x58, 0x68, 0x78, + 0x88, 0x98, 0xA8, 0xB8, 0xC8, 0xD8, 0xE8, 0xF8, + 0x09, 0x19, 0x29, 0x39, 0x49, 0x59, 0x69, 0x79, + 0x89, 0x99, 0xA9, 0xB9, 0xC9, 0xD9, 0xE9, 0xF9, + 0x0A, 0x1A, 0x2A, 0x3A, 0x4A, 0x5A, 0x6A, 0x7A, + 0x8A, 0x9A, 0xAA, 0xBA, 0xCA, 0xDA, 0xEA, 0xFA, + 0x0B, 0x1B, 0x2B, 0x3B, 0x4B, 0x5B, 0x6B, 0x7B, + 0x8B, 0x9B, 0xAB, 0xBB, 0xCB, 0xDB, 0xEB, 0xFB, + 0x0C, 0x1C, 0x2C, 0x3C, 0x4C, 0x5C, 0x6C, 0x7C, + 0x8C, 0x9C, 0xAC, 0xBC, 0xCC, 0xDC, 0xEC, 0xFC, + 0x0D, 0x1D, 0x2D, 0x3D, 0x4D, 0x5D, 0x6D, 0x7D, + 0x8D, 0x9D, 0xAD, 0xBD, 0xCD, 0xDD, 0xED, 0xFD, + 0x0E, 0x1E, 0x2E, 0x3E, 0x4E, 0x5E, 0x6E, 0x7E, + 0x8E, 0x9E, 0xAE, 0xBE, 0xCE, 0xDE, 0xEE, 0xFE, + 0x0F, 0x1F, 0x2F, 0x3F, 0x4F, 0x5F, 0x6F, 0x7F, + 0x8F, 0x9F, 0xAF, 0xBF, 0xCF, 0xDF, 0xEF, 0xFF +}; + +/* swaps pixel packing order within bytes */ +void /* PRIVATE */ +png_do_packswap(png_row_infop row_info, png_bytep row) +{ + png_debug(1, "in png_do_packswap\n"); + if ( +#if defined(PNG_USELESS_TESTS_SUPPORTED) + row != NULL && row_info != NULL && +#endif + row_info->bit_depth < 8) + { + png_bytep rp, end, table; + + end = row + row_info->rowbytes; + + if (row_info->bit_depth == 1) + table = (png_bytep)onebppswaptable; + else if (row_info->bit_depth == 2) + table = (png_bytep)twobppswaptable; + else if (row_info->bit_depth == 4) + table = (png_bytep)fourbppswaptable; + else + return; + + for (rp = row; rp < end; rp++) + *rp = table[*rp]; + } +} +#endif /* PNG_READ_PACKSWAP_SUPPORTED or PNG_WRITE_PACKSWAP_SUPPORTED */ + +#if defined(PNG_WRITE_FILLER_SUPPORTED) || \ + defined(PNG_READ_STRIP_ALPHA_SUPPORTED) +/* remove filler or alpha byte(s) */ +void /* PRIVATE */ +png_do_strip_filler(png_row_infop row_info, png_bytep row, png_uint_32 flags) +{ + png_debug(1, "in png_do_strip_filler\n"); +#if defined(PNG_USELESS_TESTS_SUPPORTED) + if (row != NULL && row_info != NULL) +#endif + { + png_bytep sp=row; + png_bytep dp=row; + png_uint_32 row_width=row_info->width; + png_uint_32 i; + + if ((row_info->color_type == PNG_COLOR_TYPE_RGB || + (row_info->color_type == PNG_COLOR_TYPE_RGB_ALPHA && + (flags & PNG_FLAG_STRIP_ALPHA))) && + row_info->channels == 4) + { + if (row_info->bit_depth == 8) + { + /* This converts from RGBX or RGBA to RGB */ + if (flags & PNG_FLAG_FILLER_AFTER) + { + dp+=3; sp+=4; + for (i = 1; i < row_width; i++) + { + *dp++ = *sp++; + *dp++ = *sp++; + *dp++ = *sp++; + sp++; + } + } + /* This converts from XRGB or ARGB to RGB */ + else + { + for (i = 0; i < row_width; i++) + { + sp++; + *dp++ = *sp++; + *dp++ = *sp++; + *dp++ = *sp++; + } + } + row_info->pixel_depth = 24; + row_info->rowbytes = row_width * 3; + } + else /* if (row_info->bit_depth == 16) */ + { + if (flags & PNG_FLAG_FILLER_AFTER) + { + /* This converts from RRGGBBXX or RRGGBBAA to RRGGBB */ + sp += 8; dp += 6; + for (i = 1; i < row_width; i++) + { + /* This could be (although png_memcpy is probably slower): + png_memcpy(dp, sp, 6); + sp += 8; + dp += 6; + */ + + *dp++ = *sp++; + *dp++ = *sp++; + *dp++ = *sp++; + *dp++ = *sp++; + *dp++ = *sp++; + *dp++ = *sp++; + sp += 2; + } + } + else + { + /* This converts from XXRRGGBB or AARRGGBB to RRGGBB */ + for (i = 0; i < row_width; i++) + { + /* This could be (although png_memcpy is probably slower): + png_memcpy(dp, sp, 6); + sp += 8; + dp += 6; + */ + + sp+=2; + *dp++ = *sp++; + *dp++ = *sp++; + *dp++ = *sp++; + *dp++ = *sp++; + *dp++ = *sp++; + *dp++ = *sp++; + } + } + row_info->pixel_depth = 48; + row_info->rowbytes = row_width * 6; + } + row_info->channels = 3; + } + else if ((row_info->color_type == PNG_COLOR_TYPE_GRAY || + (row_info->color_type == PNG_COLOR_TYPE_GRAY_ALPHA && + (flags & PNG_FLAG_STRIP_ALPHA))) && + row_info->channels == 2) + { + if (row_info->bit_depth == 8) + { + /* This converts from GX or GA to G */ + if (flags & PNG_FLAG_FILLER_AFTER) + { + for (i = 0; i < row_width; i++) + { + *dp++ = *sp++; + sp++; + } + } + /* This converts from XG or AG to G */ + else + { + for (i = 0; i < row_width; i++) + { + sp++; + *dp++ = *sp++; + } + } + row_info->pixel_depth = 8; + row_info->rowbytes = row_width; + } + else /* if (row_info->bit_depth == 16) */ + { + if (flags & PNG_FLAG_FILLER_AFTER) + { + /* This converts from GGXX or GGAA to GG */ + sp += 4; dp += 2; + for (i = 1; i < row_width; i++) + { + *dp++ = *sp++; + *dp++ = *sp++; + sp += 2; + } + } + else + { + /* This converts from XXGG or AAGG to GG */ + for (i = 0; i < row_width; i++) + { + sp += 2; + *dp++ = *sp++; + *dp++ = *sp++; + } + } + row_info->pixel_depth = 16; + row_info->rowbytes = row_width * 2; + } + row_info->channels = 1; + } + if (flags & PNG_FLAG_STRIP_ALPHA) + row_info->color_type &= ~PNG_COLOR_MASK_ALPHA; + } +} +#endif + +#if defined(PNG_READ_BGR_SUPPORTED) || defined(PNG_WRITE_BGR_SUPPORTED) +/* swaps red and blue bytes within a pixel */ +void /* PRIVATE */ +png_do_bgr(png_row_infop row_info, png_bytep row) +{ + png_debug(1, "in png_do_bgr\n"); + if ( +#if defined(PNG_USELESS_TESTS_SUPPORTED) + row != NULL && row_info != NULL && +#endif + (row_info->color_type & PNG_COLOR_MASK_COLOR)) + { + png_uint_32 row_width = row_info->width; + if (row_info->bit_depth == 8) + { + if (row_info->color_type == PNG_COLOR_TYPE_RGB) + { + png_bytep rp; + png_uint_32 i; + + for (i = 0, rp = row; i < row_width; i++, rp += 3) + { + png_byte save = *rp; + *rp = *(rp + 2); + *(rp + 2) = save; + } + } + else if (row_info->color_type == PNG_COLOR_TYPE_RGB_ALPHA) + { + png_bytep rp; + png_uint_32 i; + + for (i = 0, rp = row; i < row_width; i++, rp += 4) + { + png_byte save = *rp; + *rp = *(rp + 2); + *(rp + 2) = save; + } + } + } + else if (row_info->bit_depth == 16) + { + if (row_info->color_type == PNG_COLOR_TYPE_RGB) + { + png_bytep rp; + png_uint_32 i; + + for (i = 0, rp = row; i < row_width; i++, rp += 6) + { + png_byte save = *rp; + *rp = *(rp + 4); + *(rp + 4) = save; + save = *(rp + 1); + *(rp + 1) = *(rp + 5); + *(rp + 5) = save; + } + } + else if (row_info->color_type == PNG_COLOR_TYPE_RGB_ALPHA) + { + png_bytep rp; + png_uint_32 i; + + for (i = 0, rp = row; i < row_width; i++, rp += 8) + { + png_byte save = *rp; + *rp = *(rp + 4); + *(rp + 4) = save; + save = *(rp + 1); + *(rp + 1) = *(rp + 5); + *(rp + 5) = save; + } + } + } + } +} +#endif /* PNG_READ_BGR_SUPPORTED or PNG_WRITE_BGR_SUPPORTED */ + +#if defined(PNG_READ_USER_TRANSFORM_SUPPORTED) || \ + defined(PNG_WRITE_USER_TRANSFORM_SUPPORTED) || \ + defined(PNG_LEGACY_SUPPORTED) +void PNGAPI +png_set_user_transform_info(png_structp png_ptr, png_voidp + user_transform_ptr, int user_transform_depth, int user_transform_channels) +{ + png_debug(1, "in png_set_user_transform_info\n"); +#if defined(PNG_USER_TRANSFORM_PTR_SUPPORTED) + png_ptr->user_transform_ptr = user_transform_ptr; + png_ptr->user_transform_depth = (png_byte)user_transform_depth; + png_ptr->user_transform_channels = (png_byte)user_transform_channels; +#else + if(user_transform_ptr || user_transform_depth || user_transform_channels) + png_warning(png_ptr, + "This version of libpng does not support user transform info"); +#endif +} +#endif + +/* This function returns a pointer to the user_transform_ptr associated with + * the user transform functions. The application should free any memory + * associated with this pointer before png_write_destroy and png_read_destroy + * are called. + */ +png_voidp PNGAPI +png_get_user_transform_ptr(png_structp png_ptr) +{ +#if defined(PNG_USER_TRANSFORM_PTR_SUPPORTED) + return ((png_voidp)png_ptr->user_transform_ptr); +#else + if(png_ptr) + return (NULL); + return (NULL); +#endif +} +#endif /* PNG_READ_SUPPORTED || PNG_WRITE_SUPPORTED */ diff --git a/src/win32/dependencies/libpng/pngwio.c b/src/win32/dependencies/libpng/pngwio.c new file mode 100644 index 00000000..a9c1dc53 --- /dev/null +++ b/src/win32/dependencies/libpng/pngwio.c @@ -0,0 +1,228 @@ + +/* pngwio.c - functions for data output + * + * Last changed in libpng 1.2.3 - May 21, 2002 + * For conditions of distribution and use, see copyright notice in png.h + * Copyright (c) 1998-2002 Glenn Randers-Pehrson + * (Version 0.96 Copyright (c) 1996, 1997 Andreas Dilger) + * (Version 0.88 Copyright (c) 1995, 1996 Guy Eric Schalnat, Group 42, Inc.) + * + * This file provides a location for all output. Users who need + * special handling are expected to write functions that have the same + * arguments as these and perform similar functions, but that possibly + * use different output methods. Note that you shouldn't change these + * functions, but rather write replacement functions and then change + * them at run time with png_set_write_fn(...). + */ + +#define PNG_INTERNAL +#include "png.h" +#ifdef PNG_WRITE_SUPPORTED + +/* Write the data to whatever output you are using. The default routine + writes to a file pointer. Note that this routine sometimes gets called + with very small lengths, so you should implement some kind of simple + buffering if you are using unbuffered writes. This should never be asked + to write more than 64K on a 16 bit machine. */ + +void /* PRIVATE */ +png_write_data(png_structp png_ptr, png_bytep data, png_size_t length) +{ + if (png_ptr->write_data_fn != NULL ) + (*(png_ptr->write_data_fn))(png_ptr, data, length); + else + png_error(png_ptr, "Call to NULL write function"); +} + +#if !defined(PNG_NO_STDIO) +/* This is the function that does the actual writing of data. If you are + not writing to a standard C stream, you should create a replacement + write_data function and use it at run time with png_set_write_fn(), rather + than changing the library. */ +#ifndef USE_FAR_KEYWORD +void PNGAPI +png_default_write_data(png_structp png_ptr, png_bytep data, png_size_t length) +{ + png_uint_32 check; + +#if defined(_WIN32_WCE) + if ( !WriteFile((HANDLE)(png_ptr->io_ptr), data, length, &check, NULL) ) + check = 0; +#else + check = fwrite(data, 1, length, (png_FILE_p)(png_ptr->io_ptr)); +#endif + if (check != length) + png_error(png_ptr, "Write Error"); +} +#else +/* this is the model-independent version. Since the standard I/O library + can't handle far buffers in the medium and small models, we have to copy + the data. +*/ + +#define NEAR_BUF_SIZE 1024 +#define MIN(a,b) (a <= b ? a : b) + +void PNGAPI +png_default_write_data(png_structp png_ptr, png_bytep data, png_size_t length) +{ + png_uint_32 check; + png_byte *near_data; /* Needs to be "png_byte *" instead of "png_bytep" */ + png_FILE_p io_ptr; + + /* Check if data really is near. If so, use usual code. */ + near_data = (png_byte *)CVT_PTR_NOCHECK(data); + io_ptr = (png_FILE_p)CVT_PTR(png_ptr->io_ptr); + if ((png_bytep)near_data == data) + { +#if defined(_WIN32_WCE) + if ( !WriteFile(io_ptr, near_data, length, &check, NULL) ) + check = 0; +#else + check = fwrite(near_data, 1, length, io_ptr); +#endif + } + else + { + png_byte buf[NEAR_BUF_SIZE]; + png_size_t written, remaining, err; + check = 0; + remaining = length; + do + { + written = MIN(NEAR_BUF_SIZE, remaining); + png_memcpy(buf, data, written); /* copy far buffer to near buffer */ +#if defined(_WIN32_WCE) + if ( !WriteFile(io_ptr, buf, written, &err, NULL) ) + err = 0; +#else + err = fwrite(buf, 1, written, io_ptr); +#endif + if (err != written) + break; + else + check += err; + data += written; + remaining -= written; + } + while (remaining != 0); + } + if (check != length) + png_error(png_ptr, "Write Error"); +} + +#endif +#endif + +/* This function is called to output any data pending writing (normally + to disk). After png_flush is called, there should be no data pending + writing in any buffers. */ +#if defined(PNG_WRITE_FLUSH_SUPPORTED) +void /* PRIVATE */ +png_flush(png_structp png_ptr) +{ + if (png_ptr->output_flush_fn != NULL) + (*(png_ptr->output_flush_fn))(png_ptr); +} + +#if !defined(PNG_NO_STDIO) +void PNGAPI +png_default_flush(png_structp png_ptr) +{ +#if !defined(_WIN32_WCE) + png_FILE_p io_ptr; + io_ptr = (png_FILE_p)CVT_PTR((png_ptr->io_ptr)); + if (io_ptr != NULL) + fflush(io_ptr); +#endif +} +#endif +#endif + +/* This function allows the application to supply new output functions for + libpng if standard C streams aren't being used. + + This function takes as its arguments: + png_ptr - pointer to a png output data structure + io_ptr - pointer to user supplied structure containing info about + the output functions. May be NULL. + write_data_fn - pointer to a new output function that takes as its + arguments a pointer to a png_struct, a pointer to + data to be written, and a 32-bit unsigned int that is + the number of bytes to be written. The new write + function should call png_error(png_ptr, "Error msg") + to exit and output any fatal error messages. + flush_data_fn - pointer to a new flush function that takes as its + arguments a pointer to a png_struct. After a call to + the flush function, there should be no data in any buffers + or pending transmission. If the output method doesn't do + any buffering of ouput, a function prototype must still be + supplied although it doesn't have to do anything. If + PNG_WRITE_FLUSH_SUPPORTED is not defined at libpng compile + time, output_flush_fn will be ignored, although it must be + supplied for compatibility. */ +void PNGAPI +png_set_write_fn(png_structp png_ptr, png_voidp io_ptr, + png_rw_ptr write_data_fn, png_flush_ptr output_flush_fn) +{ + png_ptr->io_ptr = io_ptr; + +#if !defined(PNG_NO_STDIO) + if (write_data_fn != NULL) + png_ptr->write_data_fn = write_data_fn; + else + png_ptr->write_data_fn = png_default_write_data; +#else + png_ptr->write_data_fn = write_data_fn; +#endif + +#if defined(PNG_WRITE_FLUSH_SUPPORTED) +#if !defined(PNG_NO_STDIO) + if (output_flush_fn != NULL) + png_ptr->output_flush_fn = output_flush_fn; + else + png_ptr->output_flush_fn = png_default_flush; +#else + png_ptr->output_flush_fn = output_flush_fn; +#endif +#endif /* PNG_WRITE_FLUSH_SUPPORTED */ + + /* It is an error to read while writing a png file */ + if (png_ptr->read_data_fn != NULL) + { + png_ptr->read_data_fn = NULL; + png_warning(png_ptr, + "Attempted to set both read_data_fn and write_data_fn in"); + png_warning(png_ptr, + "the same structure. Resetting read_data_fn to NULL."); + } +} + +#if defined(USE_FAR_KEYWORD) +#if defined(_MSC_VER) +void *png_far_to_near(png_structp png_ptr,png_voidp ptr, int check) +{ + void *near_ptr; + void FAR *far_ptr; + FP_OFF(near_ptr) = FP_OFF(ptr); + far_ptr = (void FAR *)near_ptr; + if(check != 0) + if(FP_SEG(ptr) != FP_SEG(far_ptr)) + png_error(png_ptr,"segment lost in conversion"); + return(near_ptr); +} +# else +void *png_far_to_near(png_structp png_ptr,png_voidp ptr, int check) +{ + void *near_ptr; + void FAR *far_ptr; + near_ptr = (void FAR *)ptr; + far_ptr = (void FAR *)near_ptr; + if(check != 0) + if(far_ptr != ptr) + png_error(png_ptr,"segment lost in conversion"); + return(near_ptr); +} +# endif +# endif +#endif /* PNG_WRITE_SUPPORTED */ diff --git a/src/win32/dependencies/libpng/pngwrite.c b/src/win32/dependencies/libpng/pngwrite.c new file mode 100644 index 00000000..95984f6e --- /dev/null +++ b/src/win32/dependencies/libpng/pngwrite.c @@ -0,0 +1,1513 @@ + +/* pngwrite.c - general routines to write a PNG file + * + * Last changed in libpng 1.2.9 April 14, 2006 + * For conditions of distribution and use, see copyright notice in png.h + * Copyright (c) 1998-2006 Glenn Randers-Pehrson + * (Version 0.96 Copyright (c) 1996, 1997 Andreas Dilger) + * (Version 0.88 Copyright (c) 1995, 1996 Guy Eric Schalnat, Group 42, Inc.) + */ + +/* get internal access to png.h */ +#define PNG_INTERNAL +#include "png.h" +#ifdef PNG_WRITE_SUPPORTED + +/* Writes all the PNG information. This is the suggested way to use the + * library. If you have a new chunk to add, make a function to write it, + * and put it in the correct location here. If you want the chunk written + * after the image data, put it in png_write_end(). I strongly encourage + * you to supply a PNG_INFO_ flag, and check info_ptr->valid before writing + * the chunk, as that will keep the code from breaking if you want to just + * write a plain PNG file. If you have long comments, I suggest writing + * them in png_write_end(), and compressing them. + */ +void PNGAPI +png_write_info_before_PLTE(png_structp png_ptr, png_infop info_ptr) +{ + png_debug(1, "in png_write_info_before_PLTE\n"); + if (png_ptr == NULL || info_ptr == NULL) + return; + if (!(png_ptr->mode & PNG_WROTE_INFO_BEFORE_PLTE)) + { + png_write_sig(png_ptr); /* write PNG signature */ +#if defined(PNG_MNG_FEATURES_SUPPORTED) + if((png_ptr->mode&PNG_HAVE_PNG_SIGNATURE)&&(png_ptr->mng_features_permitted)) + { + png_warning(png_ptr,"MNG features are not allowed in a PNG datastream"); + png_ptr->mng_features_permitted=0; + } +#endif + /* write IHDR information. */ + png_write_IHDR(png_ptr, info_ptr->width, info_ptr->height, + info_ptr->bit_depth, info_ptr->color_type, info_ptr->compression_type, + info_ptr->filter_type, +#if defined(PNG_WRITE_INTERLACING_SUPPORTED) + info_ptr->interlace_type); +#else + 0); +#endif + /* the rest of these check to see if the valid field has the appropriate + flag set, and if it does, writes the chunk. */ +#if defined(PNG_WRITE_gAMA_SUPPORTED) + if (info_ptr->valid & PNG_INFO_gAMA) + { +# ifdef PNG_FLOATING_POINT_SUPPORTED + png_write_gAMA(png_ptr, info_ptr->gamma); +#else +#ifdef PNG_FIXED_POINT_SUPPORTED + png_write_gAMA_fixed(png_ptr, info_ptr->int_gamma); +# endif +#endif + } +#endif +#if defined(PNG_WRITE_sRGB_SUPPORTED) + if (info_ptr->valid & PNG_INFO_sRGB) + png_write_sRGB(png_ptr, (int)info_ptr->srgb_intent); +#endif +#if defined(PNG_WRITE_iCCP_SUPPORTED) + if (info_ptr->valid & PNG_INFO_iCCP) + png_write_iCCP(png_ptr, info_ptr->iccp_name, PNG_COMPRESSION_TYPE_BASE, + info_ptr->iccp_profile, (int)info_ptr->iccp_proflen); +#endif +#if defined(PNG_WRITE_sBIT_SUPPORTED) + if (info_ptr->valid & PNG_INFO_sBIT) + png_write_sBIT(png_ptr, &(info_ptr->sig_bit), info_ptr->color_type); +#endif +#if defined(PNG_WRITE_cHRM_SUPPORTED) + if (info_ptr->valid & PNG_INFO_cHRM) + { +#ifdef PNG_FLOATING_POINT_SUPPORTED + png_write_cHRM(png_ptr, + info_ptr->x_white, info_ptr->y_white, + info_ptr->x_red, info_ptr->y_red, + info_ptr->x_green, info_ptr->y_green, + info_ptr->x_blue, info_ptr->y_blue); +#else +# ifdef PNG_FIXED_POINT_SUPPORTED + png_write_cHRM_fixed(png_ptr, + info_ptr->int_x_white, info_ptr->int_y_white, + info_ptr->int_x_red, info_ptr->int_y_red, + info_ptr->int_x_green, info_ptr->int_y_green, + info_ptr->int_x_blue, info_ptr->int_y_blue); +# endif +#endif + } +#endif +#if defined(PNG_WRITE_UNKNOWN_CHUNKS_SUPPORTED) + if (info_ptr->unknown_chunks_num) + { + png_unknown_chunk *up; + + png_debug(5, "writing extra chunks\n"); + + for (up = info_ptr->unknown_chunks; + up < info_ptr->unknown_chunks + info_ptr->unknown_chunks_num; + up++) + { + int keep=png_handle_as_unknown(png_ptr, up->name); + if (keep != PNG_HANDLE_CHUNK_NEVER && + up->location && !(up->location & PNG_HAVE_PLTE) && + !(up->location & PNG_HAVE_IDAT) && + ((up->name[3] & 0x20) || keep == PNG_HANDLE_CHUNK_ALWAYS || + (png_ptr->flags & PNG_FLAG_KEEP_UNSAFE_CHUNKS))) + { + png_write_chunk(png_ptr, up->name, up->data, up->size); + } + } + } +#endif + png_ptr->mode |= PNG_WROTE_INFO_BEFORE_PLTE; + } +} + +void PNGAPI +png_write_info(png_structp png_ptr, png_infop info_ptr) +{ +#if defined(PNG_WRITE_TEXT_SUPPORTED) || defined(PNG_WRITE_sPLT_SUPPORTED) + int i; +#endif + + png_debug(1, "in png_write_info\n"); + + if (png_ptr == NULL || info_ptr == NULL) + return; + + png_write_info_before_PLTE(png_ptr, info_ptr); + + if (info_ptr->valid & PNG_INFO_PLTE) + png_write_PLTE(png_ptr, info_ptr->palette, + (png_uint_32)info_ptr->num_palette); + else if (info_ptr->color_type == PNG_COLOR_TYPE_PALETTE) + png_error(png_ptr, "Valid palette required for paletted images"); + +#if defined(PNG_WRITE_tRNS_SUPPORTED) + if (info_ptr->valid & PNG_INFO_tRNS) + { +#if defined(PNG_WRITE_INVERT_ALPHA_SUPPORTED) + /* invert the alpha channel (in tRNS) */ + if ((png_ptr->transformations & PNG_INVERT_ALPHA) && + info_ptr->color_type == PNG_COLOR_TYPE_PALETTE) + { + int j; + for (j=0; j<(int)info_ptr->num_trans; j++) + info_ptr->trans[j] = (png_byte)(255 - info_ptr->trans[j]); + } +#endif + png_write_tRNS(png_ptr, info_ptr->trans, &(info_ptr->trans_values), + info_ptr->num_trans, info_ptr->color_type); + } +#endif +#if defined(PNG_WRITE_bKGD_SUPPORTED) + if (info_ptr->valid & PNG_INFO_bKGD) + png_write_bKGD(png_ptr, &(info_ptr->background), info_ptr->color_type); +#endif +#if defined(PNG_WRITE_hIST_SUPPORTED) + if (info_ptr->valid & PNG_INFO_hIST) + png_write_hIST(png_ptr, info_ptr->hist, info_ptr->num_palette); +#endif +#if defined(PNG_WRITE_oFFs_SUPPORTED) + if (info_ptr->valid & PNG_INFO_oFFs) + png_write_oFFs(png_ptr, info_ptr->x_offset, info_ptr->y_offset, + info_ptr->offset_unit_type); +#endif +#if defined(PNG_WRITE_pCAL_SUPPORTED) + if (info_ptr->valid & PNG_INFO_pCAL) + png_write_pCAL(png_ptr, info_ptr->pcal_purpose, info_ptr->pcal_X0, + info_ptr->pcal_X1, info_ptr->pcal_type, info_ptr->pcal_nparams, + info_ptr->pcal_units, info_ptr->pcal_params); +#endif +#if defined(PNG_WRITE_sCAL_SUPPORTED) + if (info_ptr->valid & PNG_INFO_sCAL) +#if defined(PNG_FLOATING_POINT_SUPPORTED) && !defined(PNG_NO_STDIO) + png_write_sCAL(png_ptr, (int)info_ptr->scal_unit, + info_ptr->scal_pixel_width, info_ptr->scal_pixel_height); +#else +#ifdef PNG_FIXED_POINT_SUPPORTED + png_write_sCAL_s(png_ptr, (int)info_ptr->scal_unit, + info_ptr->scal_s_width, info_ptr->scal_s_height); +#else + png_warning(png_ptr, + "png_write_sCAL not supported; sCAL chunk not written."); +#endif +#endif +#endif +#if defined(PNG_WRITE_pHYs_SUPPORTED) + if (info_ptr->valid & PNG_INFO_pHYs) + png_write_pHYs(png_ptr, info_ptr->x_pixels_per_unit, + info_ptr->y_pixels_per_unit, info_ptr->phys_unit_type); +#endif +#if defined(PNG_WRITE_tIME_SUPPORTED) + if (info_ptr->valid & PNG_INFO_tIME) + { + png_write_tIME(png_ptr, &(info_ptr->mod_time)); + png_ptr->mode |= PNG_WROTE_tIME; + } +#endif +#if defined(PNG_WRITE_sPLT_SUPPORTED) + if (info_ptr->valid & PNG_INFO_sPLT) + for (i = 0; i < (int)info_ptr->splt_palettes_num; i++) + png_write_sPLT(png_ptr, info_ptr->splt_palettes + i); +#endif +#if defined(PNG_WRITE_TEXT_SUPPORTED) + /* Check to see if we need to write text chunks */ + for (i = 0; i < info_ptr->num_text; i++) + { + png_debug2(2, "Writing header text chunk %d, type %d\n", i, + info_ptr->text[i].compression); + /* an internationalized chunk? */ + if (info_ptr->text[i].compression > 0) + { +#if defined(PNG_WRITE_iTXt_SUPPORTED) + /* write international chunk */ + png_write_iTXt(png_ptr, + info_ptr->text[i].compression, + info_ptr->text[i].key, + info_ptr->text[i].lang, + info_ptr->text[i].lang_key, + info_ptr->text[i].text); +#else + png_warning(png_ptr, "Unable to write international text"); +#endif + /* Mark this chunk as written */ + info_ptr->text[i].compression = PNG_TEXT_COMPRESSION_NONE_WR; + } + /* If we want a compressed text chunk */ + else if (info_ptr->text[i].compression == PNG_TEXT_COMPRESSION_zTXt) + { +#if defined(PNG_WRITE_zTXt_SUPPORTED) + /* write compressed chunk */ + png_write_zTXt(png_ptr, info_ptr->text[i].key, + info_ptr->text[i].text, 0, + info_ptr->text[i].compression); +#else + png_warning(png_ptr, "Unable to write compressed text"); +#endif + /* Mark this chunk as written */ + info_ptr->text[i].compression = PNG_TEXT_COMPRESSION_zTXt_WR; + } + else if (info_ptr->text[i].compression == PNG_TEXT_COMPRESSION_NONE) + { +#if defined(PNG_WRITE_tEXt_SUPPORTED) + /* write uncompressed chunk */ + png_write_tEXt(png_ptr, info_ptr->text[i].key, + info_ptr->text[i].text, + 0); +#else + png_warning(png_ptr, "Unable to write uncompressed text"); +#endif + /* Mark this chunk as written */ + info_ptr->text[i].compression = PNG_TEXT_COMPRESSION_NONE_WR; + } + } +#endif +#if defined(PNG_WRITE_UNKNOWN_CHUNKS_SUPPORTED) + if (info_ptr->unknown_chunks_num) + { + png_unknown_chunk *up; + + png_debug(5, "writing extra chunks\n"); + + for (up = info_ptr->unknown_chunks; + up < info_ptr->unknown_chunks + info_ptr->unknown_chunks_num; + up++) + { + int keep=png_handle_as_unknown(png_ptr, up->name); + if (keep != PNG_HANDLE_CHUNK_NEVER && + up->location && (up->location & PNG_HAVE_PLTE) && + !(up->location & PNG_HAVE_IDAT) && + ((up->name[3] & 0x20) || keep == PNG_HANDLE_CHUNK_ALWAYS || + (png_ptr->flags & PNG_FLAG_KEEP_UNSAFE_CHUNKS))) + { + png_write_chunk(png_ptr, up->name, up->data, up->size); + } + } + } +#endif +} + +/* Writes the end of the PNG file. If you don't want to write comments or + * time information, you can pass NULL for info. If you already wrote these + * in png_write_info(), do not write them again here. If you have long + * comments, I suggest writing them here, and compressing them. + */ +void PNGAPI +png_write_end(png_structp png_ptr, png_infop info_ptr) +{ + png_debug(1, "in png_write_end\n"); + if (png_ptr == NULL) + return; + if (!(png_ptr->mode & PNG_HAVE_IDAT)) + png_error(png_ptr, "No IDATs written into file"); + + /* see if user wants us to write information chunks */ + if (info_ptr != NULL) + { +#if defined(PNG_WRITE_TEXT_SUPPORTED) + int i; /* local index variable */ +#endif +#if defined(PNG_WRITE_tIME_SUPPORTED) + /* check to see if user has supplied a time chunk */ + if ((info_ptr->valid & PNG_INFO_tIME) && + !(png_ptr->mode & PNG_WROTE_tIME)) + png_write_tIME(png_ptr, &(info_ptr->mod_time)); +#endif +#if defined(PNG_WRITE_TEXT_SUPPORTED) + /* loop through comment chunks */ + for (i = 0; i < info_ptr->num_text; i++) + { + png_debug2(2, "Writing trailer text chunk %d, type %d\n", i, + info_ptr->text[i].compression); + /* an internationalized chunk? */ + if (info_ptr->text[i].compression > 0) + { +#if defined(PNG_WRITE_iTXt_SUPPORTED) + /* write international chunk */ + png_write_iTXt(png_ptr, + info_ptr->text[i].compression, + info_ptr->text[i].key, + info_ptr->text[i].lang, + info_ptr->text[i].lang_key, + info_ptr->text[i].text); +#else + png_warning(png_ptr, "Unable to write international text"); +#endif + /* Mark this chunk as written */ + info_ptr->text[i].compression = PNG_TEXT_COMPRESSION_NONE_WR; + } + else if (info_ptr->text[i].compression >= PNG_TEXT_COMPRESSION_zTXt) + { +#if defined(PNG_WRITE_zTXt_SUPPORTED) + /* write compressed chunk */ + png_write_zTXt(png_ptr, info_ptr->text[i].key, + info_ptr->text[i].text, 0, + info_ptr->text[i].compression); +#else + png_warning(png_ptr, "Unable to write compressed text"); +#endif + /* Mark this chunk as written */ + info_ptr->text[i].compression = PNG_TEXT_COMPRESSION_zTXt_WR; + } + else if (info_ptr->text[i].compression == PNG_TEXT_COMPRESSION_NONE) + { +#if defined(PNG_WRITE_tEXt_SUPPORTED) + /* write uncompressed chunk */ + png_write_tEXt(png_ptr, info_ptr->text[i].key, + info_ptr->text[i].text, 0); +#else + png_warning(png_ptr, "Unable to write uncompressed text"); +#endif + + /* Mark this chunk as written */ + info_ptr->text[i].compression = PNG_TEXT_COMPRESSION_NONE_WR; + } + } +#endif +#if defined(PNG_WRITE_UNKNOWN_CHUNKS_SUPPORTED) + if (info_ptr->unknown_chunks_num) + { + png_unknown_chunk *up; + + png_debug(5, "writing extra chunks\n"); + + for (up = info_ptr->unknown_chunks; + up < info_ptr->unknown_chunks + info_ptr->unknown_chunks_num; + up++) + { + int keep=png_handle_as_unknown(png_ptr, up->name); + if (keep != PNG_HANDLE_CHUNK_NEVER && + up->location && (up->location & PNG_AFTER_IDAT) && + ((up->name[3] & 0x20) || keep == PNG_HANDLE_CHUNK_ALWAYS || + (png_ptr->flags & PNG_FLAG_KEEP_UNSAFE_CHUNKS))) + { + png_write_chunk(png_ptr, up->name, up->data, up->size); + } + } + } +#endif + } + + png_ptr->mode |= PNG_AFTER_IDAT; + + /* write end of PNG file */ + png_write_IEND(png_ptr); +#if 0 +/* This flush, added in libpng-1.0.8, causes some applications to crash + because they do not set png_ptr->output_flush_fn */ + png_flush(png_ptr); +#endif +} + +#if defined(PNG_WRITE_tIME_SUPPORTED) +#if !defined(_WIN32_WCE) +/* "time.h" functions are not supported on WindowsCE */ +void PNGAPI +png_convert_from_struct_tm(png_timep ptime, struct tm FAR * ttime) +{ + png_debug(1, "in png_convert_from_struct_tm\n"); + ptime->year = (png_uint_16)(1900 + ttime->tm_year); + ptime->month = (png_byte)(ttime->tm_mon + 1); + ptime->day = (png_byte)ttime->tm_mday; + ptime->hour = (png_byte)ttime->tm_hour; + ptime->minute = (png_byte)ttime->tm_min; + ptime->second = (png_byte)ttime->tm_sec; +} + +void PNGAPI +png_convert_from_time_t(png_timep ptime, time_t ttime) +{ + struct tm *tbuf; + + png_debug(1, "in png_convert_from_time_t\n"); + tbuf = gmtime(&ttime); + png_convert_from_struct_tm(ptime, tbuf); +} +#endif +#endif + +/* Initialize png_ptr structure, and allocate any memory needed */ +png_structp PNGAPI +png_create_write_struct(png_const_charp user_png_ver, png_voidp error_ptr, + png_error_ptr error_fn, png_error_ptr warn_fn) +{ +#ifdef PNG_USER_MEM_SUPPORTED + return (png_create_write_struct_2(user_png_ver, error_ptr, error_fn, + warn_fn, png_voidp_NULL, png_malloc_ptr_NULL, png_free_ptr_NULL)); +} + +/* Alternate initialize png_ptr structure, and allocate any memory needed */ +png_structp PNGAPI +png_create_write_struct_2(png_const_charp user_png_ver, png_voidp error_ptr, + png_error_ptr error_fn, png_error_ptr warn_fn, png_voidp mem_ptr, + png_malloc_ptr malloc_fn, png_free_ptr free_fn) +{ +#endif /* PNG_USER_MEM_SUPPORTED */ + png_structp png_ptr; +#ifdef PNG_SETJMP_SUPPORTED +#ifdef USE_FAR_KEYWORD + jmp_buf jmpbuf; +#endif +#endif + int i; + png_debug(1, "in png_create_write_struct\n"); +#ifdef PNG_USER_MEM_SUPPORTED + png_ptr = (png_structp)png_create_struct_2(PNG_STRUCT_PNG, + (png_malloc_ptr)malloc_fn, (png_voidp)mem_ptr); +#else + png_ptr = (png_structp)png_create_struct(PNG_STRUCT_PNG); +#endif /* PNG_USER_MEM_SUPPORTED */ + if (png_ptr == NULL) + return (NULL); + +#if !defined(PNG_1_0_X) +#ifdef PNG_ASSEMBLER_CODE_SUPPORTED + png_init_mmx_flags(png_ptr); /* 1.2.0 addition */ +#endif +#endif /* PNG_1_0_X */ + + /* added at libpng-1.2.6 */ +#ifdef PNG_SET_USER_LIMITS_SUPPORTED + png_ptr->user_width_max=PNG_USER_WIDTH_MAX; + png_ptr->user_height_max=PNG_USER_HEIGHT_MAX; +#endif + +#ifdef PNG_SETJMP_SUPPORTED +#ifdef USE_FAR_KEYWORD + if (setjmp(jmpbuf)) +#else + if (setjmp(png_ptr->jmpbuf)) +#endif + { + png_free(png_ptr, png_ptr->zbuf); + png_ptr->zbuf=NULL; + png_destroy_struct(png_ptr); + return (NULL); + } +#ifdef USE_FAR_KEYWORD + png_memcpy(png_ptr->jmpbuf,jmpbuf,png_sizeof(jmp_buf)); +#endif +#endif + +#ifdef PNG_USER_MEM_SUPPORTED + png_set_mem_fn(png_ptr, mem_ptr, malloc_fn, free_fn); +#endif /* PNG_USER_MEM_SUPPORTED */ + png_set_error_fn(png_ptr, error_ptr, error_fn, warn_fn); + + i=0; + do + { + if(user_png_ver[i] != png_libpng_ver[i]) + png_ptr->flags |= PNG_FLAG_LIBRARY_MISMATCH; + } while (png_libpng_ver[i++]); + + if (png_ptr->flags & PNG_FLAG_LIBRARY_MISMATCH) + { + /* Libpng 0.90 and later are binary incompatible with libpng 0.89, so + * we must recompile any applications that use any older library version. + * For versions after libpng 1.0, we will be compatible, so we need + * only check the first digit. + */ + if (user_png_ver == NULL || user_png_ver[0] != png_libpng_ver[0] || + (user_png_ver[0] == '1' && user_png_ver[2] != png_libpng_ver[2]) || + (user_png_ver[0] == '0' && user_png_ver[2] < '9')) + { +#if !defined(PNG_NO_STDIO) && !defined(_WIN32_WCE) + char msg[80]; + if (user_png_ver) + { + sprintf(msg, "Application was compiled with png.h from libpng-%.20s", + user_png_ver); + png_warning(png_ptr, msg); + } + sprintf(msg, "Application is running with png.c from libpng-%.20s", + png_libpng_ver); + png_warning(png_ptr, msg); +#endif +#ifdef PNG_ERROR_NUMBERS_SUPPORTED + png_ptr->flags=0; +#endif + png_error(png_ptr, + "Incompatible libpng version in application and library"); + } + } + + /* initialize zbuf - compression buffer */ + png_ptr->zbuf_size = PNG_ZBUF_SIZE; + png_ptr->zbuf = (png_bytep)png_malloc(png_ptr, + (png_uint_32)png_ptr->zbuf_size); + + png_set_write_fn(png_ptr, png_voidp_NULL, png_rw_ptr_NULL, + png_flush_ptr_NULL); + +#if defined(PNG_WRITE_WEIGHTED_FILTER_SUPPORTED) + png_set_filter_heuristics(png_ptr, PNG_FILTER_HEURISTIC_DEFAULT, + 1, png_doublep_NULL, png_doublep_NULL); +#endif + +#ifdef PNG_SETJMP_SUPPORTED +/* Applications that neglect to set up their own setjmp() and then encounter + a png_error() will longjmp here. Since the jmpbuf is then meaningless we + abort instead of returning. */ +#ifdef USE_FAR_KEYWORD + if (setjmp(jmpbuf)) + PNG_ABORT(); + png_memcpy(png_ptr->jmpbuf,jmpbuf,png_sizeof(jmp_buf)); +#else + if (setjmp(png_ptr->jmpbuf)) + PNG_ABORT(); +#endif +#endif + return (png_ptr); +} + +/* Initialize png_ptr structure, and allocate any memory needed */ +#if defined(PNG_1_0_X) || defined(PNG_1_2_X) +/* Deprecated. */ +#undef png_write_init +void PNGAPI +png_write_init(png_structp png_ptr) +{ + /* We only come here via pre-1.0.7-compiled applications */ + png_write_init_2(png_ptr, "1.0.6 or earlier", 0, 0); +} + +void PNGAPI +png_write_init_2(png_structp png_ptr, png_const_charp user_png_ver, + png_size_t png_struct_size, png_size_t png_info_size) +{ + /* We only come here via pre-1.0.12-compiled applications */ +#if !defined(PNG_NO_STDIO) && !defined(_WIN32_WCE) + if(png_sizeof(png_struct) > png_struct_size || + png_sizeof(png_info) > png_info_size) + { + char msg[80]; + png_ptr->warning_fn=NULL; + if (user_png_ver) + { + sprintf(msg, "Application was compiled with png.h from libpng-%.20s", + user_png_ver); + png_warning(png_ptr, msg); + } + sprintf(msg, "Application is running with png.c from libpng-%.20s", + png_libpng_ver); + png_warning(png_ptr, msg); + } +#endif + if(png_sizeof(png_struct) > png_struct_size) + { + png_ptr->error_fn=NULL; +#ifdef PNG_ERROR_NUMBERS_SUPPORTED + png_ptr->flags=0; +#endif + png_error(png_ptr, + "The png struct allocated by the application for writing is too small."); + } + if(png_sizeof(png_info) > png_info_size) + { + png_ptr->error_fn=NULL; +#ifdef PNG_ERROR_NUMBERS_SUPPORTED + png_ptr->flags=0; +#endif + png_error(png_ptr, + "The info struct allocated by the application for writing is too small."); + } + png_write_init_3(&png_ptr, user_png_ver, png_struct_size); +} +#endif /* PNG_1_0_X || PNG_1_2_X */ + + +void PNGAPI +png_write_init_3(png_structpp ptr_ptr, png_const_charp user_png_ver, + png_size_t png_struct_size) +{ + png_structp png_ptr=*ptr_ptr; +#ifdef PNG_SETJMP_SUPPORTED + jmp_buf tmp_jmp; /* to save current jump buffer */ +#endif + + int i = 0; + + if (png_ptr == NULL) + return; + + do + { + if (user_png_ver[i] != png_libpng_ver[i]) + { +#ifdef PNG_LEGACY_SUPPORTED + png_ptr->flags |= PNG_FLAG_LIBRARY_MISMATCH; +#else + png_ptr->warning_fn=NULL; + png_warning(png_ptr, + "Application uses deprecated png_write_init() and should be recompiled."); + break; +#endif + } + } while (png_libpng_ver[i++]); + + png_debug(1, "in png_write_init_3\n"); + +#ifdef PNG_SETJMP_SUPPORTED + /* save jump buffer and error functions */ + png_memcpy(tmp_jmp, png_ptr->jmpbuf, png_sizeof (jmp_buf)); +#endif + + if (png_sizeof(png_struct) > png_struct_size) + { + png_destroy_struct(png_ptr); + png_ptr = (png_structp)png_create_struct(PNG_STRUCT_PNG); + *ptr_ptr = png_ptr; + } + + /* reset all variables to 0 */ + png_memset(png_ptr, 0, png_sizeof (png_struct)); + + /* added at libpng-1.2.6 */ +#ifdef PNG_SET_USER_LIMITS_SUPPORTED + png_ptr->user_width_max=PNG_USER_WIDTH_MAX; + png_ptr->user_height_max=PNG_USER_HEIGHT_MAX; +#endif + +#if !defined(PNG_1_0_X) +#ifdef PNG_ASSEMBLER_CODE_SUPPORTED + png_init_mmx_flags(png_ptr); /* 1.2.0 addition */ +#endif +#endif /* PNG_1_0_X */ + +#ifdef PNG_SETJMP_SUPPORTED + /* restore jump buffer */ + png_memcpy(png_ptr->jmpbuf, tmp_jmp, png_sizeof (jmp_buf)); +#endif + + png_set_write_fn(png_ptr, png_voidp_NULL, png_rw_ptr_NULL, + png_flush_ptr_NULL); + + /* initialize zbuf - compression buffer */ + png_ptr->zbuf_size = PNG_ZBUF_SIZE; + png_ptr->zbuf = (png_bytep)png_malloc(png_ptr, + (png_uint_32)png_ptr->zbuf_size); + +#if defined(PNG_WRITE_WEIGHTED_FILTER_SUPPORTED) + png_set_filter_heuristics(png_ptr, PNG_FILTER_HEURISTIC_DEFAULT, + 1, png_doublep_NULL, png_doublep_NULL); +#endif +} + +/* Write a few rows of image data. If the image is interlaced, + * either you will have to write the 7 sub images, or, if you + * have called png_set_interlace_handling(), you will have to + * "write" the image seven times. + */ +void PNGAPI +png_write_rows(png_structp png_ptr, png_bytepp row, + png_uint_32 num_rows) +{ + png_uint_32 i; /* row counter */ + png_bytepp rp; /* row pointer */ + + png_debug(1, "in png_write_rows\n"); + + if (png_ptr == NULL) + return; + + /* loop through the rows */ + for (i = 0, rp = row; i < num_rows; i++, rp++) + { + png_write_row(png_ptr, *rp); + } +} + +/* Write the image. You only need to call this function once, even + * if you are writing an interlaced image. + */ +void PNGAPI +png_write_image(png_structp png_ptr, png_bytepp image) +{ + png_uint_32 i; /* row index */ + int pass, num_pass; /* pass variables */ + png_bytepp rp; /* points to current row */ + + if (png_ptr == NULL) + return; + + png_debug(1, "in png_write_image\n"); +#if defined(PNG_WRITE_INTERLACING_SUPPORTED) + /* intialize interlace handling. If image is not interlaced, + this will set pass to 1 */ + num_pass = png_set_interlace_handling(png_ptr); +#else + num_pass = 1; +#endif + /* loop through passes */ + for (pass = 0; pass < num_pass; pass++) + { + /* loop through image */ + for (i = 0, rp = image; i < png_ptr->height; i++, rp++) + { + png_write_row(png_ptr, *rp); + } + } +} + +/* called by user to write a row of image data */ +void PNGAPI +png_write_row(png_structp png_ptr, png_bytep row) +{ + if (png_ptr == NULL) + return; + png_debug2(1, "in png_write_row (row %ld, pass %d)\n", + png_ptr->row_number, png_ptr->pass); + + /* initialize transformations and other stuff if first time */ + if (png_ptr->row_number == 0 && png_ptr->pass == 0) + { + /* make sure we wrote the header info */ + if (!(png_ptr->mode & PNG_WROTE_INFO_BEFORE_PLTE)) + png_error(png_ptr, + "png_write_info was never called before png_write_row."); + + /* check for transforms that have been set but were defined out */ +#if !defined(PNG_WRITE_INVERT_SUPPORTED) && defined(PNG_READ_INVERT_SUPPORTED) + if (png_ptr->transformations & PNG_INVERT_MONO) + png_warning(png_ptr, "PNG_WRITE_INVERT_SUPPORTED is not defined."); +#endif +#if !defined(PNG_WRITE_FILLER_SUPPORTED) && defined(PNG_READ_FILLER_SUPPORTED) + if (png_ptr->transformations & PNG_FILLER) + png_warning(png_ptr, "PNG_WRITE_FILLER_SUPPORTED is not defined."); +#endif +#if !defined(PNG_WRITE_PACKSWAP_SUPPORTED) && defined(PNG_READ_PACKSWAP_SUPPORTED) + if (png_ptr->transformations & PNG_PACKSWAP) + png_warning(png_ptr, "PNG_WRITE_PACKSWAP_SUPPORTED is not defined."); +#endif +#if !defined(PNG_WRITE_PACK_SUPPORTED) && defined(PNG_READ_PACK_SUPPORTED) + if (png_ptr->transformations & PNG_PACK) + png_warning(png_ptr, "PNG_WRITE_PACK_SUPPORTED is not defined."); +#endif +#if !defined(PNG_WRITE_SHIFT_SUPPORTED) && defined(PNG_READ_SHIFT_SUPPORTED) + if (png_ptr->transformations & PNG_SHIFT) + png_warning(png_ptr, "PNG_WRITE_SHIFT_SUPPORTED is not defined."); +#endif +#if !defined(PNG_WRITE_BGR_SUPPORTED) && defined(PNG_READ_BGR_SUPPORTED) + if (png_ptr->transformations & PNG_BGR) + png_warning(png_ptr, "PNG_WRITE_BGR_SUPPORTED is not defined."); +#endif +#if !defined(PNG_WRITE_SWAP_SUPPORTED) && defined(PNG_READ_SWAP_SUPPORTED) + if (png_ptr->transformations & PNG_SWAP_BYTES) + png_warning(png_ptr, "PNG_WRITE_SWAP_SUPPORTED is not defined."); +#endif + + png_write_start_row(png_ptr); + } + +#if defined(PNG_WRITE_INTERLACING_SUPPORTED) + /* if interlaced and not interested in row, return */ + if (png_ptr->interlaced && (png_ptr->transformations & PNG_INTERLACE)) + { + switch (png_ptr->pass) + { + case 0: + if (png_ptr->row_number & 0x07) + { + png_write_finish_row(png_ptr); + return; + } + break; + case 1: + if ((png_ptr->row_number & 0x07) || png_ptr->width < 5) + { + png_write_finish_row(png_ptr); + return; + } + break; + case 2: + if ((png_ptr->row_number & 0x07) != 4) + { + png_write_finish_row(png_ptr); + return; + } + break; + case 3: + if ((png_ptr->row_number & 0x03) || png_ptr->width < 3) + { + png_write_finish_row(png_ptr); + return; + } + break; + case 4: + if ((png_ptr->row_number & 0x03) != 2) + { + png_write_finish_row(png_ptr); + return; + } + break; + case 5: + if ((png_ptr->row_number & 0x01) || png_ptr->width < 2) + { + png_write_finish_row(png_ptr); + return; + } + break; + case 6: + if (!(png_ptr->row_number & 0x01)) + { + png_write_finish_row(png_ptr); + return; + } + break; + } + } +#endif + + /* set up row info for transformations */ + png_ptr->row_info.color_type = png_ptr->color_type; + png_ptr->row_info.width = png_ptr->usr_width; + png_ptr->row_info.channels = png_ptr->usr_channels; + png_ptr->row_info.bit_depth = png_ptr->usr_bit_depth; + png_ptr->row_info.pixel_depth = (png_byte)(png_ptr->row_info.bit_depth * + png_ptr->row_info.channels); + + png_ptr->row_info.rowbytes = PNG_ROWBYTES(png_ptr->row_info.pixel_depth, + png_ptr->row_info.width); + + png_debug1(3, "row_info->color_type = %d\n", png_ptr->row_info.color_type); + png_debug1(3, "row_info->width = %lu\n", png_ptr->row_info.width); + png_debug1(3, "row_info->channels = %d\n", png_ptr->row_info.channels); + png_debug1(3, "row_info->bit_depth = %d\n", png_ptr->row_info.bit_depth); + png_debug1(3, "row_info->pixel_depth = %d\n", png_ptr->row_info.pixel_depth); + png_debug1(3, "row_info->rowbytes = %lu\n", png_ptr->row_info.rowbytes); + + /* Copy user's row into buffer, leaving room for filter byte. */ + png_memcpy_check(png_ptr, png_ptr->row_buf + 1, row, + png_ptr->row_info.rowbytes); + +#if defined(PNG_WRITE_INTERLACING_SUPPORTED) + /* handle interlacing */ + if (png_ptr->interlaced && png_ptr->pass < 6 && + (png_ptr->transformations & PNG_INTERLACE)) + { + png_do_write_interlace(&(png_ptr->row_info), + png_ptr->row_buf + 1, png_ptr->pass); + /* this should always get caught above, but still ... */ + if (!(png_ptr->row_info.width)) + { + png_write_finish_row(png_ptr); + return; + } + } +#endif + + /* handle other transformations */ + if (png_ptr->transformations) + png_do_write_transformations(png_ptr); + +#if defined(PNG_MNG_FEATURES_SUPPORTED) + /* Write filter_method 64 (intrapixel differencing) only if + * 1. Libpng was compiled with PNG_MNG_FEATURES_SUPPORTED and + * 2. Libpng did not write a PNG signature (this filter_method is only + * used in PNG datastreams that are embedded in MNG datastreams) and + * 3. The application called png_permit_mng_features with a mask that + * included PNG_FLAG_MNG_FILTER_64 and + * 4. The filter_method is 64 and + * 5. The color_type is RGB or RGBA + */ + if((png_ptr->mng_features_permitted & PNG_FLAG_MNG_FILTER_64) && + (png_ptr->filter_type == PNG_INTRAPIXEL_DIFFERENCING)) + { + /* Intrapixel differencing */ + png_do_write_intrapixel(&(png_ptr->row_info), png_ptr->row_buf + 1); + } +#endif + + /* Find a filter if necessary, filter the row and write it out. */ + png_write_find_filter(png_ptr, &(png_ptr->row_info)); + + if (png_ptr->write_row_fn != NULL) + (*(png_ptr->write_row_fn))(png_ptr, png_ptr->row_number, png_ptr->pass); +} + +#if defined(PNG_WRITE_FLUSH_SUPPORTED) +/* Set the automatic flush interval or 0 to turn flushing off */ +void PNGAPI +png_set_flush(png_structp png_ptr, int nrows) +{ + png_debug(1, "in png_set_flush\n"); + if (png_ptr == NULL) + return; + png_ptr->flush_dist = (nrows < 0 ? 0 : nrows); +} + +/* flush the current output buffers now */ +void PNGAPI +png_write_flush(png_structp png_ptr) +{ + int wrote_IDAT; + + png_debug(1, "in png_write_flush\n"); + if (png_ptr == NULL) + return; + /* We have already written out all of the data */ + if (png_ptr->row_number >= png_ptr->num_rows) + return; + + do + { + int ret; + + /* compress the data */ + ret = deflate(&png_ptr->zstream, Z_SYNC_FLUSH); + wrote_IDAT = 0; + + /* check for compression errors */ + if (ret != Z_OK) + { + if (png_ptr->zstream.msg != NULL) + png_error(png_ptr, png_ptr->zstream.msg); + else + png_error(png_ptr, "zlib error"); + } + + if (!(png_ptr->zstream.avail_out)) + { + /* write the IDAT and reset the zlib output buffer */ + png_write_IDAT(png_ptr, png_ptr->zbuf, + png_ptr->zbuf_size); + png_ptr->zstream.next_out = png_ptr->zbuf; + png_ptr->zstream.avail_out = (uInt)png_ptr->zbuf_size; + wrote_IDAT = 1; + } + } while(wrote_IDAT == 1); + + /* If there is any data left to be output, write it into a new IDAT */ + if (png_ptr->zbuf_size != png_ptr->zstream.avail_out) + { + /* write the IDAT and reset the zlib output buffer */ + png_write_IDAT(png_ptr, png_ptr->zbuf, + png_ptr->zbuf_size - png_ptr->zstream.avail_out); + png_ptr->zstream.next_out = png_ptr->zbuf; + png_ptr->zstream.avail_out = (uInt)png_ptr->zbuf_size; + } + png_ptr->flush_rows = 0; + png_flush(png_ptr); +} +#endif /* PNG_WRITE_FLUSH_SUPPORTED */ + +/* free all memory used by the write */ +void PNGAPI +png_destroy_write_struct(png_structpp png_ptr_ptr, png_infopp info_ptr_ptr) +{ + png_structp png_ptr = NULL; + png_infop info_ptr = NULL; +#ifdef PNG_USER_MEM_SUPPORTED + png_free_ptr free_fn = NULL; + png_voidp mem_ptr = NULL; +#endif + + png_debug(1, "in png_destroy_write_struct\n"); + if (png_ptr_ptr != NULL) + { + png_ptr = *png_ptr_ptr; +#ifdef PNG_USER_MEM_SUPPORTED + free_fn = png_ptr->free_fn; + mem_ptr = png_ptr->mem_ptr; +#endif + } + + if (info_ptr_ptr != NULL) + info_ptr = *info_ptr_ptr; + + if (info_ptr != NULL) + { + png_free_data(png_ptr, info_ptr, PNG_FREE_ALL, -1); + +#if defined(PNG_UNKNOWN_CHUNKS_SUPPORTED) + if (png_ptr->num_chunk_list) + { + png_free(png_ptr, png_ptr->chunk_list); + png_ptr->chunk_list=NULL; + png_ptr->num_chunk_list=0; + } +#endif + +#ifdef PNG_USER_MEM_SUPPORTED + png_destroy_struct_2((png_voidp)info_ptr, (png_free_ptr)free_fn, + (png_voidp)mem_ptr); +#else + png_destroy_struct((png_voidp)info_ptr); +#endif + *info_ptr_ptr = NULL; + } + + if (png_ptr != NULL) + { + png_write_destroy(png_ptr); +#ifdef PNG_USER_MEM_SUPPORTED + png_destroy_struct_2((png_voidp)png_ptr, (png_free_ptr)free_fn, + (png_voidp)mem_ptr); +#else + png_destroy_struct((png_voidp)png_ptr); +#endif + *png_ptr_ptr = NULL; + } +} + + +/* Free any memory used in png_ptr struct (old method) */ +void /* PRIVATE */ +png_write_destroy(png_structp png_ptr) +{ +#ifdef PNG_SETJMP_SUPPORTED + jmp_buf tmp_jmp; /* save jump buffer */ +#endif + png_error_ptr error_fn; + png_error_ptr warning_fn; + png_voidp error_ptr; +#ifdef PNG_USER_MEM_SUPPORTED + png_free_ptr free_fn; +#endif + + png_debug(1, "in png_write_destroy\n"); + /* free any memory zlib uses */ + deflateEnd(&png_ptr->zstream); + + /* free our memory. png_free checks NULL for us. */ + png_free(png_ptr, png_ptr->zbuf); + png_free(png_ptr, png_ptr->row_buf); + png_free(png_ptr, png_ptr->prev_row); + png_free(png_ptr, png_ptr->sub_row); + png_free(png_ptr, png_ptr->up_row); + png_free(png_ptr, png_ptr->avg_row); + png_free(png_ptr, png_ptr->paeth_row); + +#if defined(PNG_TIME_RFC1123_SUPPORTED) + png_free(png_ptr, png_ptr->time_buffer); +#endif + +#if defined(PNG_WRITE_WEIGHTED_FILTER_SUPPORTED) + png_free(png_ptr, png_ptr->prev_filters); + png_free(png_ptr, png_ptr->filter_weights); + png_free(png_ptr, png_ptr->inv_filter_weights); + png_free(png_ptr, png_ptr->filter_costs); + png_free(png_ptr, png_ptr->inv_filter_costs); +#endif + +#ifdef PNG_SETJMP_SUPPORTED + /* reset structure */ + png_memcpy(tmp_jmp, png_ptr->jmpbuf, png_sizeof (jmp_buf)); +#endif + + error_fn = png_ptr->error_fn; + warning_fn = png_ptr->warning_fn; + error_ptr = png_ptr->error_ptr; +#ifdef PNG_USER_MEM_SUPPORTED + free_fn = png_ptr->free_fn; +#endif + + png_memset(png_ptr, 0, png_sizeof (png_struct)); + + png_ptr->error_fn = error_fn; + png_ptr->warning_fn = warning_fn; + png_ptr->error_ptr = error_ptr; +#ifdef PNG_USER_MEM_SUPPORTED + png_ptr->free_fn = free_fn; +#endif + +#ifdef PNG_SETJMP_SUPPORTED + png_memcpy(png_ptr->jmpbuf, tmp_jmp, png_sizeof (jmp_buf)); +#endif +} + +/* Allow the application to select one or more row filters to use. */ +void PNGAPI +png_set_filter(png_structp png_ptr, int method, int filters) +{ + png_debug(1, "in png_set_filter\n"); + if (png_ptr == NULL) + return; +#if defined(PNG_MNG_FEATURES_SUPPORTED) + if((png_ptr->mng_features_permitted & PNG_FLAG_MNG_FILTER_64) && + (method == PNG_INTRAPIXEL_DIFFERENCING)) + method = PNG_FILTER_TYPE_BASE; +#endif + if (method == PNG_FILTER_TYPE_BASE) + { + switch (filters & (PNG_ALL_FILTERS | 0x07)) + { + case 5: + case 6: + case 7: png_warning(png_ptr, "Unknown row filter for method 0"); + case PNG_FILTER_VALUE_NONE: png_ptr->do_filter=PNG_FILTER_NONE; break; + case PNG_FILTER_VALUE_SUB: png_ptr->do_filter=PNG_FILTER_SUB; break; + case PNG_FILTER_VALUE_UP: png_ptr->do_filter=PNG_FILTER_UP; break; + case PNG_FILTER_VALUE_AVG: png_ptr->do_filter=PNG_FILTER_AVG; break; + case PNG_FILTER_VALUE_PAETH: png_ptr->do_filter=PNG_FILTER_PAETH;break; + default: png_ptr->do_filter = (png_byte)filters; break; + } + + /* If we have allocated the row_buf, this means we have already started + * with the image and we should have allocated all of the filter buffers + * that have been selected. If prev_row isn't already allocated, then + * it is too late to start using the filters that need it, since we + * will be missing the data in the previous row. If an application + * wants to start and stop using particular filters during compression, + * it should start out with all of the filters, and then add and + * remove them after the start of compression. + */ + if (png_ptr->row_buf != NULL) + { + if ((png_ptr->do_filter & PNG_FILTER_SUB) && png_ptr->sub_row == NULL) + { + png_ptr->sub_row = (png_bytep)png_malloc(png_ptr, + (png_ptr->rowbytes + 1)); + png_ptr->sub_row[0] = PNG_FILTER_VALUE_SUB; + } + + if ((png_ptr->do_filter & PNG_FILTER_UP) && png_ptr->up_row == NULL) + { + if (png_ptr->prev_row == NULL) + { + png_warning(png_ptr, "Can't add Up filter after starting"); + png_ptr->do_filter &= ~PNG_FILTER_UP; + } + else + { + png_ptr->up_row = (png_bytep)png_malloc(png_ptr, + (png_ptr->rowbytes + 1)); + png_ptr->up_row[0] = PNG_FILTER_VALUE_UP; + } + } + + if ((png_ptr->do_filter & PNG_FILTER_AVG) && png_ptr->avg_row == NULL) + { + if (png_ptr->prev_row == NULL) + { + png_warning(png_ptr, "Can't add Average filter after starting"); + png_ptr->do_filter &= ~PNG_FILTER_AVG; + } + else + { + png_ptr->avg_row = (png_bytep)png_malloc(png_ptr, + (png_ptr->rowbytes + 1)); + png_ptr->avg_row[0] = PNG_FILTER_VALUE_AVG; + } + } + + if ((png_ptr->do_filter & PNG_FILTER_PAETH) && + png_ptr->paeth_row == NULL) + { + if (png_ptr->prev_row == NULL) + { + png_warning(png_ptr, "Can't add Paeth filter after starting"); + png_ptr->do_filter &= (png_byte)(~PNG_FILTER_PAETH); + } + else + { + png_ptr->paeth_row = (png_bytep)png_malloc(png_ptr, + (png_ptr->rowbytes + 1)); + png_ptr->paeth_row[0] = PNG_FILTER_VALUE_PAETH; + } + } + + if (png_ptr->do_filter == PNG_NO_FILTERS) + png_ptr->do_filter = PNG_FILTER_NONE; + } + } + else + png_error(png_ptr, "Unknown custom filter method"); +} + +/* This allows us to influence the way in which libpng chooses the "best" + * filter for the current scanline. While the "minimum-sum-of-absolute- + * differences metric is relatively fast and effective, there is some + * question as to whether it can be improved upon by trying to keep the + * filtered data going to zlib more consistent, hopefully resulting in + * better compression. + */ +#if defined(PNG_WRITE_WEIGHTED_FILTER_SUPPORTED) /* GRR 970116 */ +void PNGAPI +png_set_filter_heuristics(png_structp png_ptr, int heuristic_method, + int num_weights, png_doublep filter_weights, + png_doublep filter_costs) +{ + int i; + + png_debug(1, "in png_set_filter_heuristics\n"); + if (png_ptr == NULL) + return; + if (heuristic_method >= PNG_FILTER_HEURISTIC_LAST) + { + png_warning(png_ptr, "Unknown filter heuristic method"); + return; + } + + if (heuristic_method == PNG_FILTER_HEURISTIC_DEFAULT) + { + heuristic_method = PNG_FILTER_HEURISTIC_UNWEIGHTED; + } + + if (num_weights < 0 || filter_weights == NULL || + heuristic_method == PNG_FILTER_HEURISTIC_UNWEIGHTED) + { + num_weights = 0; + } + + png_ptr->num_prev_filters = (png_byte)num_weights; + png_ptr->heuristic_method = (png_byte)heuristic_method; + + if (num_weights > 0) + { + if (png_ptr->prev_filters == NULL) + { + png_ptr->prev_filters = (png_bytep)png_malloc(png_ptr, + (png_uint_32)(png_sizeof(png_byte) * num_weights)); + + /* To make sure that the weighting starts out fairly */ + for (i = 0; i < num_weights; i++) + { + png_ptr->prev_filters[i] = 255; + } + } + + if (png_ptr->filter_weights == NULL) + { + png_ptr->filter_weights = (png_uint_16p)png_malloc(png_ptr, + (png_uint_32)(png_sizeof(png_uint_16) * num_weights)); + + png_ptr->inv_filter_weights = (png_uint_16p)png_malloc(png_ptr, + (png_uint_32)(png_sizeof(png_uint_16) * num_weights)); + for (i = 0; i < num_weights; i++) + { + png_ptr->inv_filter_weights[i] = + png_ptr->filter_weights[i] = PNG_WEIGHT_FACTOR; + } + } + + for (i = 0; i < num_weights; i++) + { + if (filter_weights[i] < 0.0) + { + png_ptr->inv_filter_weights[i] = + png_ptr->filter_weights[i] = PNG_WEIGHT_FACTOR; + } + else + { + png_ptr->inv_filter_weights[i] = + (png_uint_16)((double)PNG_WEIGHT_FACTOR*filter_weights[i]+0.5); + png_ptr->filter_weights[i] = + (png_uint_16)((double)PNG_WEIGHT_FACTOR/filter_weights[i]+0.5); + } + } + } + + /* If, in the future, there are other filter methods, this would + * need to be based on png_ptr->filter. + */ + if (png_ptr->filter_costs == NULL) + { + png_ptr->filter_costs = (png_uint_16p)png_malloc(png_ptr, + (png_uint_32)(png_sizeof(png_uint_16) * PNG_FILTER_VALUE_LAST)); + + png_ptr->inv_filter_costs = (png_uint_16p)png_malloc(png_ptr, + (png_uint_32)(png_sizeof(png_uint_16) * PNG_FILTER_VALUE_LAST)); + + for (i = 0; i < PNG_FILTER_VALUE_LAST; i++) + { + png_ptr->inv_filter_costs[i] = + png_ptr->filter_costs[i] = PNG_COST_FACTOR; + } + } + + /* Here is where we set the relative costs of the different filters. We + * should take the desired compression level into account when setting + * the costs, so that Paeth, for instance, has a high relative cost at low + * compression levels, while it has a lower relative cost at higher + * compression settings. The filter types are in order of increasing + * relative cost, so it would be possible to do this with an algorithm. + */ + for (i = 0; i < PNG_FILTER_VALUE_LAST; i++) + { + if (filter_costs == NULL || filter_costs[i] < 0.0) + { + png_ptr->inv_filter_costs[i] = + png_ptr->filter_costs[i] = PNG_COST_FACTOR; + } + else if (filter_costs[i] >= 1.0) + { + png_ptr->inv_filter_costs[i] = + (png_uint_16)((double)PNG_COST_FACTOR / filter_costs[i] + 0.5); + png_ptr->filter_costs[i] = + (png_uint_16)((double)PNG_COST_FACTOR * filter_costs[i] + 0.5); + } + } +} +#endif /* PNG_WRITE_WEIGHTED_FILTER_SUPPORTED */ + +void PNGAPI +png_set_compression_level(png_structp png_ptr, int level) +{ + png_debug(1, "in png_set_compression_level\n"); + if (png_ptr == NULL) + return; + png_ptr->flags |= PNG_FLAG_ZLIB_CUSTOM_LEVEL; + png_ptr->zlib_level = level; +} + +void PNGAPI +png_set_compression_mem_level(png_structp png_ptr, int mem_level) +{ + png_debug(1, "in png_set_compression_mem_level\n"); + if (png_ptr == NULL) + return; + png_ptr->flags |= PNG_FLAG_ZLIB_CUSTOM_MEM_LEVEL; + png_ptr->zlib_mem_level = mem_level; +} + +void PNGAPI +png_set_compression_strategy(png_structp png_ptr, int strategy) +{ + png_debug(1, "in png_set_compression_strategy\n"); + if (png_ptr == NULL) + return; + png_ptr->flags |= PNG_FLAG_ZLIB_CUSTOM_STRATEGY; + png_ptr->zlib_strategy = strategy; +} + +void PNGAPI +png_set_compression_window_bits(png_structp png_ptr, int window_bits) +{ + if (png_ptr == NULL) + return; + if (window_bits > 15) + png_warning(png_ptr, "Only compression windows <= 32k supported by PNG"); + else if (window_bits < 8) + png_warning(png_ptr, "Only compression windows >= 256 supported by PNG"); +#ifndef WBITS_8_OK + /* avoid libpng bug with 256-byte windows */ + if (window_bits == 8) + { + png_warning(png_ptr, "Compression window is being reset to 512"); + window_bits=9; + } +#endif + png_ptr->flags |= PNG_FLAG_ZLIB_CUSTOM_WINDOW_BITS; + png_ptr->zlib_window_bits = window_bits; +} + +void PNGAPI +png_set_compression_method(png_structp png_ptr, int method) +{ + png_debug(1, "in png_set_compression_method\n"); + if (png_ptr == NULL) + return; + if (method != 8) + png_warning(png_ptr, "Only compression method 8 is supported by PNG"); + png_ptr->flags |= PNG_FLAG_ZLIB_CUSTOM_METHOD; + png_ptr->zlib_method = method; +} + +void PNGAPI +png_set_write_status_fn(png_structp png_ptr, png_write_status_ptr write_row_fn) +{ + if (png_ptr == NULL) + return; + png_ptr->write_row_fn = write_row_fn; +} + +#if defined(PNG_WRITE_USER_TRANSFORM_SUPPORTED) +void PNGAPI +png_set_write_user_transform_fn(png_structp png_ptr, png_user_transform_ptr + write_user_transform_fn) +{ + png_debug(1, "in png_set_write_user_transform_fn\n"); + if (png_ptr == NULL) + return; + png_ptr->transformations |= PNG_USER_TRANSFORM; + png_ptr->write_user_transform_fn = write_user_transform_fn; +} +#endif + + +#if defined(PNG_INFO_IMAGE_SUPPORTED) +void PNGAPI +png_write_png(png_structp png_ptr, png_infop info_ptr, + int transforms, voidp params) +{ + if (png_ptr == NULL || info_ptr == NULL) + return; +#if defined(PNG_WRITE_INVERT_ALPHA_SUPPORTED) + /* invert the alpha channel from opacity to transparency */ + if (transforms & PNG_TRANSFORM_INVERT_ALPHA) + png_set_invert_alpha(png_ptr); +#endif + + /* Write the file header information. */ + png_write_info(png_ptr, info_ptr); + + /* ------ these transformations don't touch the info structure ------- */ + +#if defined(PNG_WRITE_INVERT_SUPPORTED) + /* invert monochrome pixels */ + if (transforms & PNG_TRANSFORM_INVERT_MONO) + png_set_invert_mono(png_ptr); +#endif + +#if defined(PNG_WRITE_SHIFT_SUPPORTED) + /* Shift the pixels up to a legal bit depth and fill in + * as appropriate to correctly scale the image. + */ + if ((transforms & PNG_TRANSFORM_SHIFT) + && (info_ptr->valid & PNG_INFO_sBIT)) + png_set_shift(png_ptr, &info_ptr->sig_bit); +#endif + +#if defined(PNG_WRITE_PACK_SUPPORTED) + /* pack pixels into bytes */ + if (transforms & PNG_TRANSFORM_PACKING) + png_set_packing(png_ptr); +#endif + +#if defined(PNG_WRITE_SWAP_ALPHA_SUPPORTED) + /* swap location of alpha bytes from ARGB to RGBA */ + if (transforms & PNG_TRANSFORM_SWAP_ALPHA) + png_set_swap_alpha(png_ptr); +#endif + +#if defined(PNG_WRITE_FILLER_SUPPORTED) + /* Get rid of filler (OR ALPHA) bytes, pack XRGB/RGBX/ARGB/RGBA into + * RGB (4 channels -> 3 channels). The second parameter is not used. + */ + if (transforms & PNG_TRANSFORM_STRIP_FILLER) + png_set_filler(png_ptr, 0, PNG_FILLER_BEFORE); +#endif + +#if defined(PNG_WRITE_BGR_SUPPORTED) + /* flip BGR pixels to RGB */ + if (transforms & PNG_TRANSFORM_BGR) + png_set_bgr(png_ptr); +#endif + +#if defined(PNG_WRITE_SWAP_SUPPORTED) + /* swap bytes of 16-bit files to most significant byte first */ + if (transforms & PNG_TRANSFORM_SWAP_ENDIAN) + png_set_swap(png_ptr); +#endif + +#if defined(PNG_WRITE_PACKSWAP_SUPPORTED) + /* swap bits of 1, 2, 4 bit packed pixel formats */ + if (transforms & PNG_TRANSFORM_PACKSWAP) + png_set_packswap(png_ptr); +#endif + + /* ----------------------- end of transformations ------------------- */ + + /* write the bits */ + if (info_ptr->valid & PNG_INFO_IDAT) + png_write_image(png_ptr, info_ptr->row_pointers); + + /* It is REQUIRED to call this to finish writing the rest of the file */ + png_write_end(png_ptr, info_ptr); + + if(transforms == 0 || params == NULL) + /* quiet compiler warnings */ return; +} +#endif +#endif /* PNG_WRITE_SUPPORTED */ diff --git a/src/win32/dependencies/libpng/pngwtran.c b/src/win32/dependencies/libpng/pngwtran.c new file mode 100644 index 00000000..0372fe65 --- /dev/null +++ b/src/win32/dependencies/libpng/pngwtran.c @@ -0,0 +1,572 @@ + +/* pngwtran.c - transforms the data in a row for PNG writers + * + * Last changed in libpng 1.2.9 April 14, 2006 + * For conditions of distribution and use, see copyright notice in png.h + * Copyright (c) 1998-2006 Glenn Randers-Pehrson + * (Version 0.96 Copyright (c) 1996, 1997 Andreas Dilger) + * (Version 0.88 Copyright (c) 1995, 1996 Guy Eric Schalnat, Group 42, Inc.) + */ + +#define PNG_INTERNAL +#include "png.h" +#ifdef PNG_WRITE_SUPPORTED + +/* Transform the data according to the user's wishes. The order of + * transformations is significant. + */ +void /* PRIVATE */ +png_do_write_transformations(png_structp png_ptr) +{ + png_debug(1, "in png_do_write_transformations\n"); + + if (png_ptr == NULL) + return; + +#if defined(PNG_WRITE_USER_TRANSFORM_SUPPORTED) + if (png_ptr->transformations & PNG_USER_TRANSFORM) + if(png_ptr->write_user_transform_fn != NULL) + (*(png_ptr->write_user_transform_fn)) /* user write transform function */ + (png_ptr, /* png_ptr */ + &(png_ptr->row_info), /* row_info: */ + /* png_uint_32 width; width of row */ + /* png_uint_32 rowbytes; number of bytes in row */ + /* png_byte color_type; color type of pixels */ + /* png_byte bit_depth; bit depth of samples */ + /* png_byte channels; number of channels (1-4) */ + /* png_byte pixel_depth; bits per pixel (depth*channels) */ + png_ptr->row_buf + 1); /* start of pixel data for row */ +#endif +#if defined(PNG_WRITE_FILLER_SUPPORTED) + if (png_ptr->transformations & PNG_FILLER) + png_do_strip_filler(&(png_ptr->row_info), png_ptr->row_buf + 1, + png_ptr->flags); +#endif +#if defined(PNG_WRITE_PACKSWAP_SUPPORTED) + if (png_ptr->transformations & PNG_PACKSWAP) + png_do_packswap(&(png_ptr->row_info), png_ptr->row_buf + 1); +#endif +#if defined(PNG_WRITE_PACK_SUPPORTED) + if (png_ptr->transformations & PNG_PACK) + png_do_pack(&(png_ptr->row_info), png_ptr->row_buf + 1, + (png_uint_32)png_ptr->bit_depth); +#endif +#if defined(PNG_WRITE_SWAP_SUPPORTED) + if (png_ptr->transformations & PNG_SWAP_BYTES) + png_do_swap(&(png_ptr->row_info), png_ptr->row_buf + 1); +#endif +#if defined(PNG_WRITE_SHIFT_SUPPORTED) + if (png_ptr->transformations & PNG_SHIFT) + png_do_shift(&(png_ptr->row_info), png_ptr->row_buf + 1, + &(png_ptr->shift)); +#endif +#if defined(PNG_WRITE_SWAP_ALPHA_SUPPORTED) + if (png_ptr->transformations & PNG_SWAP_ALPHA) + png_do_write_swap_alpha(&(png_ptr->row_info), png_ptr->row_buf + 1); +#endif +#if defined(PNG_WRITE_INVERT_ALPHA_SUPPORTED) + if (png_ptr->transformations & PNG_INVERT_ALPHA) + png_do_write_invert_alpha(&(png_ptr->row_info), png_ptr->row_buf + 1); +#endif +#if defined(PNG_WRITE_BGR_SUPPORTED) + if (png_ptr->transformations & PNG_BGR) + png_do_bgr(&(png_ptr->row_info), png_ptr->row_buf + 1); +#endif +#if defined(PNG_WRITE_INVERT_SUPPORTED) + if (png_ptr->transformations & PNG_INVERT_MONO) + png_do_invert(&(png_ptr->row_info), png_ptr->row_buf + 1); +#endif +} + +#if defined(PNG_WRITE_PACK_SUPPORTED) +/* Pack pixels into bytes. Pass the true bit depth in bit_depth. The + * row_info bit depth should be 8 (one pixel per byte). The channels + * should be 1 (this only happens on grayscale and paletted images). + */ +void /* PRIVATE */ +png_do_pack(png_row_infop row_info, png_bytep row, png_uint_32 bit_depth) +{ + png_debug(1, "in png_do_pack\n"); + if (row_info->bit_depth == 8 && +#if defined(PNG_USELESS_TESTS_SUPPORTED) + row != NULL && row_info != NULL && +#endif + row_info->channels == 1) + { + switch ((int)bit_depth) + { + case 1: + { + png_bytep sp, dp; + int mask, v; + png_uint_32 i; + png_uint_32 row_width = row_info->width; + + sp = row; + dp = row; + mask = 0x80; + v = 0; + + for (i = 0; i < row_width; i++) + { + if (*sp != 0) + v |= mask; + sp++; + if (mask > 1) + mask >>= 1; + else + { + mask = 0x80; + *dp = (png_byte)v; + dp++; + v = 0; + } + } + if (mask != 0x80) + *dp = (png_byte)v; + break; + } + case 2: + { + png_bytep sp, dp; + int shift, v; + png_uint_32 i; + png_uint_32 row_width = row_info->width; + + sp = row; + dp = row; + shift = 6; + v = 0; + for (i = 0; i < row_width; i++) + { + png_byte value; + + value = (png_byte)(*sp & 0x03); + v |= (value << shift); + if (shift == 0) + { + shift = 6; + *dp = (png_byte)v; + dp++; + v = 0; + } + else + shift -= 2; + sp++; + } + if (shift != 6) + *dp = (png_byte)v; + break; + } + case 4: + { + png_bytep sp, dp; + int shift, v; + png_uint_32 i; + png_uint_32 row_width = row_info->width; + + sp = row; + dp = row; + shift = 4; + v = 0; + for (i = 0; i < row_width; i++) + { + png_byte value; + + value = (png_byte)(*sp & 0x0f); + v |= (value << shift); + + if (shift == 0) + { + shift = 4; + *dp = (png_byte)v; + dp++; + v = 0; + } + else + shift -= 4; + + sp++; + } + if (shift != 4) + *dp = (png_byte)v; + break; + } + } + row_info->bit_depth = (png_byte)bit_depth; + row_info->pixel_depth = (png_byte)(bit_depth * row_info->channels); + row_info->rowbytes = PNG_ROWBYTES(row_info->pixel_depth, + row_info->width); + } +} +#endif + +#if defined(PNG_WRITE_SHIFT_SUPPORTED) +/* Shift pixel values to take advantage of whole range. Pass the + * true number of bits in bit_depth. The row should be packed + * according to row_info->bit_depth. Thus, if you had a row of + * bit depth 4, but the pixels only had values from 0 to 7, you + * would pass 3 as bit_depth, and this routine would translate the + * data to 0 to 15. + */ +void /* PRIVATE */ +png_do_shift(png_row_infop row_info, png_bytep row, png_color_8p bit_depth) +{ + png_debug(1, "in png_do_shift\n"); +#if defined(PNG_USELESS_TESTS_SUPPORTED) + if (row != NULL && row_info != NULL && +#else + if ( +#endif + row_info->color_type != PNG_COLOR_TYPE_PALETTE) + { + int shift_start[4], shift_dec[4]; + int channels = 0; + + if (row_info->color_type & PNG_COLOR_MASK_COLOR) + { + shift_start[channels] = row_info->bit_depth - bit_depth->red; + shift_dec[channels] = bit_depth->red; + channels++; + shift_start[channels] = row_info->bit_depth - bit_depth->green; + shift_dec[channels] = bit_depth->green; + channels++; + shift_start[channels] = row_info->bit_depth - bit_depth->blue; + shift_dec[channels] = bit_depth->blue; + channels++; + } + else + { + shift_start[channels] = row_info->bit_depth - bit_depth->gray; + shift_dec[channels] = bit_depth->gray; + channels++; + } + if (row_info->color_type & PNG_COLOR_MASK_ALPHA) + { + shift_start[channels] = row_info->bit_depth - bit_depth->alpha; + shift_dec[channels] = bit_depth->alpha; + channels++; + } + + /* with low row depths, could only be grayscale, so one channel */ + if (row_info->bit_depth < 8) + { + png_bytep bp = row; + png_uint_32 i; + png_byte mask; + png_uint_32 row_bytes = row_info->rowbytes; + + if (bit_depth->gray == 1 && row_info->bit_depth == 2) + mask = 0x55; + else if (row_info->bit_depth == 4 && bit_depth->gray == 3) + mask = 0x11; + else + mask = 0xff; + + for (i = 0; i < row_bytes; i++, bp++) + { + png_uint_16 v; + int j; + + v = *bp; + *bp = 0; + for (j = shift_start[0]; j > -shift_dec[0]; j -= shift_dec[0]) + { + if (j > 0) + *bp |= (png_byte)((v << j) & 0xff); + else + *bp |= (png_byte)((v >> (-j)) & mask); + } + } + } + else if (row_info->bit_depth == 8) + { + png_bytep bp = row; + png_uint_32 i; + png_uint_32 istop = channels * row_info->width; + + for (i = 0; i < istop; i++, bp++) + { + + png_uint_16 v; + int j; + int c = (int)(i%channels); + + v = *bp; + *bp = 0; + for (j = shift_start[c]; j > -shift_dec[c]; j -= shift_dec[c]) + { + if (j > 0) + *bp |= (png_byte)((v << j) & 0xff); + else + *bp |= (png_byte)((v >> (-j)) & 0xff); + } + } + } + else + { + png_bytep bp; + png_uint_32 i; + png_uint_32 istop = channels * row_info->width; + + for (bp = row, i = 0; i < istop; i++) + { + int c = (int)(i%channels); + png_uint_16 value, v; + int j; + + v = (png_uint_16)(((png_uint_16)(*bp) << 8) + *(bp + 1)); + value = 0; + for (j = shift_start[c]; j > -shift_dec[c]; j -= shift_dec[c]) + { + if (j > 0) + value |= (png_uint_16)((v << j) & (png_uint_16)0xffff); + else + value |= (png_uint_16)((v >> (-j)) & (png_uint_16)0xffff); + } + *bp++ = (png_byte)(value >> 8); + *bp++ = (png_byte)(value & 0xff); + } + } + } +} +#endif + +#if defined(PNG_WRITE_SWAP_ALPHA_SUPPORTED) +void /* PRIVATE */ +png_do_write_swap_alpha(png_row_infop row_info, png_bytep row) +{ + png_debug(1, "in png_do_write_swap_alpha\n"); +#if defined(PNG_USELESS_TESTS_SUPPORTED) + if (row != NULL && row_info != NULL) +#endif + { + if (row_info->color_type == PNG_COLOR_TYPE_RGB_ALPHA) + { + /* This converts from ARGB to RGBA */ + if (row_info->bit_depth == 8) + { + png_bytep sp, dp; + png_uint_32 i; + png_uint_32 row_width = row_info->width; + for (i = 0, sp = dp = row; i < row_width; i++) + { + png_byte save = *(sp++); + *(dp++) = *(sp++); + *(dp++) = *(sp++); + *(dp++) = *(sp++); + *(dp++) = save; + } + } + /* This converts from AARRGGBB to RRGGBBAA */ + else + { + png_bytep sp, dp; + png_uint_32 i; + png_uint_32 row_width = row_info->width; + + for (i = 0, sp = dp = row; i < row_width; i++) + { + png_byte save[2]; + save[0] = *(sp++); + save[1] = *(sp++); + *(dp++) = *(sp++); + *(dp++) = *(sp++); + *(dp++) = *(sp++); + *(dp++) = *(sp++); + *(dp++) = *(sp++); + *(dp++) = *(sp++); + *(dp++) = save[0]; + *(dp++) = save[1]; + } + } + } + else if (row_info->color_type == PNG_COLOR_TYPE_GRAY_ALPHA) + { + /* This converts from AG to GA */ + if (row_info->bit_depth == 8) + { + png_bytep sp, dp; + png_uint_32 i; + png_uint_32 row_width = row_info->width; + + for (i = 0, sp = dp = row; i < row_width; i++) + { + png_byte save = *(sp++); + *(dp++) = *(sp++); + *(dp++) = save; + } + } + /* This converts from AAGG to GGAA */ + else + { + png_bytep sp, dp; + png_uint_32 i; + png_uint_32 row_width = row_info->width; + + for (i = 0, sp = dp = row; i < row_width; i++) + { + png_byte save[2]; + save[0] = *(sp++); + save[1] = *(sp++); + *(dp++) = *(sp++); + *(dp++) = *(sp++); + *(dp++) = save[0]; + *(dp++) = save[1]; + } + } + } + } +} +#endif + +#if defined(PNG_WRITE_INVERT_ALPHA_SUPPORTED) +void /* PRIVATE */ +png_do_write_invert_alpha(png_row_infop row_info, png_bytep row) +{ + png_debug(1, "in png_do_write_invert_alpha\n"); +#if defined(PNG_USELESS_TESTS_SUPPORTED) + if (row != NULL && row_info != NULL) +#endif + { + if (row_info->color_type == PNG_COLOR_TYPE_RGB_ALPHA) + { + /* This inverts the alpha channel in RGBA */ + if (row_info->bit_depth == 8) + { + png_bytep sp, dp; + png_uint_32 i; + png_uint_32 row_width = row_info->width; + for (i = 0, sp = dp = row; i < row_width; i++) + { + /* does nothing + *(dp++) = *(sp++); + *(dp++) = *(sp++); + *(dp++) = *(sp++); + */ + sp+=3; dp = sp; + *(dp++) = (png_byte)(255 - *(sp++)); + } + } + /* This inverts the alpha channel in RRGGBBAA */ + else + { + png_bytep sp, dp; + png_uint_32 i; + png_uint_32 row_width = row_info->width; + + for (i = 0, sp = dp = row; i < row_width; i++) + { + /* does nothing + *(dp++) = *(sp++); + *(dp++) = *(sp++); + *(dp++) = *(sp++); + *(dp++) = *(sp++); + *(dp++) = *(sp++); + *(dp++) = *(sp++); + */ + sp+=6; dp = sp; + *(dp++) = (png_byte)(255 - *(sp++)); + *(dp++) = (png_byte)(255 - *(sp++)); + } + } + } + else if (row_info->color_type == PNG_COLOR_TYPE_GRAY_ALPHA) + { + /* This inverts the alpha channel in GA */ + if (row_info->bit_depth == 8) + { + png_bytep sp, dp; + png_uint_32 i; + png_uint_32 row_width = row_info->width; + + for (i = 0, sp = dp = row; i < row_width; i++) + { + *(dp++) = *(sp++); + *(dp++) = (png_byte)(255 - *(sp++)); + } + } + /* This inverts the alpha channel in GGAA */ + else + { + png_bytep sp, dp; + png_uint_32 i; + png_uint_32 row_width = row_info->width; + + for (i = 0, sp = dp = row; i < row_width; i++) + { + /* does nothing + *(dp++) = *(sp++); + *(dp++) = *(sp++); + */ + sp+=2; dp = sp; + *(dp++) = (png_byte)(255 - *(sp++)); + *(dp++) = (png_byte)(255 - *(sp++)); + } + } + } + } +} +#endif + +#if defined(PNG_MNG_FEATURES_SUPPORTED) +/* undoes intrapixel differencing */ +void /* PRIVATE */ +png_do_write_intrapixel(png_row_infop row_info, png_bytep row) +{ + png_debug(1, "in png_do_write_intrapixel\n"); + if ( +#if defined(PNG_USELESS_TESTS_SUPPORTED) + row != NULL && row_info != NULL && +#endif + (row_info->color_type & PNG_COLOR_MASK_COLOR)) + { + int bytes_per_pixel; + png_uint_32 row_width = row_info->width; + if (row_info->bit_depth == 8) + { + png_bytep rp; + png_uint_32 i; + + if (row_info->color_type == PNG_COLOR_TYPE_RGB) + bytes_per_pixel = 3; + else if (row_info->color_type == PNG_COLOR_TYPE_RGB_ALPHA) + bytes_per_pixel = 4; + else + return; + + for (i = 0, rp = row; i < row_width; i++, rp += bytes_per_pixel) + { + *(rp) = (png_byte)((*rp - *(rp+1))&0xff); + *(rp+2) = (png_byte)((*(rp+2) - *(rp+1))&0xff); + } + } + else if (row_info->bit_depth == 16) + { + png_bytep rp; + png_uint_32 i; + + if (row_info->color_type == PNG_COLOR_TYPE_RGB) + bytes_per_pixel = 6; + else if (row_info->color_type == PNG_COLOR_TYPE_RGB_ALPHA) + bytes_per_pixel = 8; + else + return; + + for (i = 0, rp = row; i < row_width; i++, rp += bytes_per_pixel) + { + png_uint_32 s0 = (*(rp ) << 8) | *(rp+1); + png_uint_32 s1 = (*(rp+2) << 8) | *(rp+3); + png_uint_32 s2 = (*(rp+4) << 8) | *(rp+5); + png_uint_32 red = (png_uint_32)((s0-s1) & 0xffffL); + png_uint_32 blue = (png_uint_32)((s2-s1) & 0xffffL); + *(rp ) = (png_byte)((red >> 8) & 0xff); + *(rp+1) = (png_byte)(red & 0xff); + *(rp+4) = (png_byte)((blue >> 8) & 0xff); + *(rp+5) = (png_byte)(blue & 0xff); + } + } + } +} +#endif /* PNG_MNG_FEATURES_SUPPORTED */ +#endif /* PNG_WRITE_SUPPORTED */ diff --git a/src/win32/dependencies/libpng/pngwutil.c b/src/win32/dependencies/libpng/pngwutil.c new file mode 100644 index 00000000..82bf58a1 --- /dev/null +++ b/src/win32/dependencies/libpng/pngwutil.c @@ -0,0 +1,2745 @@ + +/* pngwutil.c - utilities to write a PNG file + * + * Last changed in libpng 1.2.9 April 14, 2006 + * For conditions of distribution and use, see copyright notice in png.h + * Copyright (c) 1998-2006 Glenn Randers-Pehrson + * (Version 0.96 Copyright (c) 1996, 1997 Andreas Dilger) + * (Version 0.88 Copyright (c) 1995, 1996 Guy Eric Schalnat, Group 42, Inc.) + */ + +#define PNG_INTERNAL +#include "png.h" +#ifdef PNG_WRITE_SUPPORTED + +/* Place a 32-bit number into a buffer in PNG byte order. We work + * with unsigned numbers for convenience, although one supported + * ancillary chunk uses signed (two's complement) numbers. + */ +void PNGAPI +png_save_uint_32(png_bytep buf, png_uint_32 i) +{ + buf[0] = (png_byte)((i >> 24) & 0xff); + buf[1] = (png_byte)((i >> 16) & 0xff); + buf[2] = (png_byte)((i >> 8) & 0xff); + buf[3] = (png_byte)(i & 0xff); +} + +/* The png_save_int_32 function assumes integers are stored in two's + * complement format. If this isn't the case, then this routine needs to + * be modified to write data in two's complement format. + */ +void PNGAPI +png_save_int_32(png_bytep buf, png_int_32 i) +{ + buf[0] = (png_byte)((i >> 24) & 0xff); + buf[1] = (png_byte)((i >> 16) & 0xff); + buf[2] = (png_byte)((i >> 8) & 0xff); + buf[3] = (png_byte)(i & 0xff); +} + +/* Place a 16-bit number into a buffer in PNG byte order. + * The parameter is declared unsigned int, not png_uint_16, + * just to avoid potential problems on pre-ANSI C compilers. + */ +void PNGAPI +png_save_uint_16(png_bytep buf, unsigned int i) +{ + buf[0] = (png_byte)((i >> 8) & 0xff); + buf[1] = (png_byte)(i & 0xff); +} + +/* Write a PNG chunk all at once. The type is an array of ASCII characters + * representing the chunk name. The array must be at least 4 bytes in + * length, and does not need to be null terminated. To be safe, pass the + * pre-defined chunk names here, and if you need a new one, define it + * where the others are defined. The length is the length of the data. + * All the data must be present. If that is not possible, use the + * png_write_chunk_start(), png_write_chunk_data(), and png_write_chunk_end() + * functions instead. + */ +void PNGAPI +png_write_chunk(png_structp png_ptr, png_bytep chunk_name, + png_bytep data, png_size_t length) +{ + png_write_chunk_start(png_ptr, chunk_name, (png_uint_32)length); + png_write_chunk_data(png_ptr, data, length); + png_write_chunk_end(png_ptr); +} + +/* Write the start of a PNG chunk. The type is the chunk type. + * The total_length is the sum of the lengths of all the data you will be + * passing in png_write_chunk_data(). + */ +void PNGAPI +png_write_chunk_start(png_structp png_ptr, png_bytep chunk_name, + png_uint_32 length) +{ + png_byte buf[4]; + png_debug2(0, "Writing %s chunk (%lu bytes)\n", chunk_name, length); + + /* write the length */ + png_save_uint_32(buf, length); + png_write_data(png_ptr, buf, (png_size_t)4); + + /* write the chunk name */ + png_write_data(png_ptr, chunk_name, (png_size_t)4); + /* reset the crc and run it over the chunk name */ + png_reset_crc(png_ptr); + png_calculate_crc(png_ptr, chunk_name, (png_size_t)4); +} + +/* Write the data of a PNG chunk started with png_write_chunk_start(). + * Note that multiple calls to this function are allowed, and that the + * sum of the lengths from these calls *must* add up to the total_length + * given to png_write_chunk_start(). + */ +void PNGAPI +png_write_chunk_data(png_structp png_ptr, png_bytep data, png_size_t length) +{ + /* write the data, and run the CRC over it */ + if (data != NULL && length > 0) + { + png_calculate_crc(png_ptr, data, length); + png_write_data(png_ptr, data, length); + } +} + +/* Finish a chunk started with png_write_chunk_start(). */ +void PNGAPI +png_write_chunk_end(png_structp png_ptr) +{ + png_byte buf[4]; + + /* write the crc */ + png_save_uint_32(buf, png_ptr->crc); + + png_write_data(png_ptr, buf, (png_size_t)4); +} + +/* Simple function to write the signature. If we have already written + * the magic bytes of the signature, or more likely, the PNG stream is + * being embedded into another stream and doesn't need its own signature, + * we should call png_set_sig_bytes() to tell libpng how many of the + * bytes have already been written. + */ +void /* PRIVATE */ +png_write_sig(png_structp png_ptr) +{ + png_byte png_signature[8] = {137, 80, 78, 71, 13, 10, 26, 10}; + /* write the rest of the 8 byte signature */ + png_write_data(png_ptr, &png_signature[png_ptr->sig_bytes], + (png_size_t)8 - png_ptr->sig_bytes); + if(png_ptr->sig_bytes < 3) + png_ptr->mode |= PNG_HAVE_PNG_SIGNATURE; +} + +#if defined(PNG_WRITE_TEXT_SUPPORTED) || defined(PNG_WRITE_iCCP_SUPPORTED) +/* + * This pair of functions encapsulates the operation of (a) compressing a + * text string, and (b) issuing it later as a series of chunk data writes. + * The compression_state structure is shared context for these functions + * set up by the caller in order to make the whole mess thread-safe. + */ + +typedef struct +{ + char *input; /* the uncompressed input data */ + int input_len; /* its length */ + int num_output_ptr; /* number of output pointers used */ + int max_output_ptr; /* size of output_ptr */ + png_charpp output_ptr; /* array of pointers to output */ +} compression_state; + +/* compress given text into storage in the png_ptr structure */ +static int /* PRIVATE */ +png_text_compress(png_structp png_ptr, + png_charp text, png_size_t text_len, int compression, + compression_state *comp) +{ + int ret; + + comp->num_output_ptr = 0; + comp->max_output_ptr = 0; + comp->output_ptr = NULL; + comp->input = NULL; + comp->input_len = 0; + + /* we may just want to pass the text right through */ + if (compression == PNG_TEXT_COMPRESSION_NONE) + { + comp->input = text; + comp->input_len = text_len; + return((int)text_len); + } + + if (compression >= PNG_TEXT_COMPRESSION_LAST) + { +#if !defined(PNG_NO_STDIO) && !defined(_WIN32_WCE) + char msg[50]; + sprintf(msg, "Unknown compression type %d", compression); + png_warning(png_ptr, msg); +#else + png_warning(png_ptr, "Unknown compression type"); +#endif + } + + /* We can't write the chunk until we find out how much data we have, + * which means we need to run the compressor first and save the + * output. This shouldn't be a problem, as the vast majority of + * comments should be reasonable, but we will set up an array of + * malloc'd pointers to be sure. + * + * If we knew the application was well behaved, we could simplify this + * greatly by assuming we can always malloc an output buffer large + * enough to hold the compressed text ((1001 * text_len / 1000) + 12) + * and malloc this directly. The only time this would be a bad idea is + * if we can't malloc more than 64K and we have 64K of random input + * data, or if the input string is incredibly large (although this + * wouldn't cause a failure, just a slowdown due to swapping). + */ + + /* set up the compression buffers */ + png_ptr->zstream.avail_in = (uInt)text_len; + png_ptr->zstream.next_in = (Bytef *)text; + png_ptr->zstream.avail_out = (uInt)png_ptr->zbuf_size; + png_ptr->zstream.next_out = (Bytef *)png_ptr->zbuf; + + /* this is the same compression loop as in png_write_row() */ + do + { + /* compress the data */ + ret = deflate(&png_ptr->zstream, Z_NO_FLUSH); + if (ret != Z_OK) + { + /* error */ + if (png_ptr->zstream.msg != NULL) + png_error(png_ptr, png_ptr->zstream.msg); + else + png_error(png_ptr, "zlib error"); + } + /* check to see if we need more room */ + if (!(png_ptr->zstream.avail_out)) + { + /* make sure the output array has room */ + if (comp->num_output_ptr >= comp->max_output_ptr) + { + int old_max; + + old_max = comp->max_output_ptr; + comp->max_output_ptr = comp->num_output_ptr + 4; + if (comp->output_ptr != NULL) + { + png_charpp old_ptr; + + old_ptr = comp->output_ptr; + comp->output_ptr = (png_charpp)png_malloc(png_ptr, + (png_uint_32)(comp->max_output_ptr * + png_sizeof (png_charpp))); + png_memcpy(comp->output_ptr, old_ptr, old_max + * png_sizeof (png_charp)); + png_free(png_ptr, old_ptr); + } + else + comp->output_ptr = (png_charpp)png_malloc(png_ptr, + (png_uint_32)(comp->max_output_ptr * + png_sizeof (png_charp))); + } + + /* save the data */ + comp->output_ptr[comp->num_output_ptr] = (png_charp)png_malloc(png_ptr, + (png_uint_32)png_ptr->zbuf_size); + png_memcpy(comp->output_ptr[comp->num_output_ptr], png_ptr->zbuf, + png_ptr->zbuf_size); + comp->num_output_ptr++; + + /* and reset the buffer */ + png_ptr->zstream.avail_out = (uInt)png_ptr->zbuf_size; + png_ptr->zstream.next_out = png_ptr->zbuf; + } + /* continue until we don't have any more to compress */ + } while (png_ptr->zstream.avail_in); + + /* finish the compression */ + do + { + /* tell zlib we are finished */ + ret = deflate(&png_ptr->zstream, Z_FINISH); + + if (ret == Z_OK) + { + /* check to see if we need more room */ + if (!(png_ptr->zstream.avail_out)) + { + /* check to make sure our output array has room */ + if (comp->num_output_ptr >= comp->max_output_ptr) + { + int old_max; + + old_max = comp->max_output_ptr; + comp->max_output_ptr = comp->num_output_ptr + 4; + if (comp->output_ptr != NULL) + { + png_charpp old_ptr; + + old_ptr = comp->output_ptr; + /* This could be optimized to realloc() */ + comp->output_ptr = (png_charpp)png_malloc(png_ptr, + (png_uint_32)(comp->max_output_ptr * + png_sizeof (png_charpp))); + png_memcpy(comp->output_ptr, old_ptr, + old_max * png_sizeof (png_charp)); + png_free(png_ptr, old_ptr); + } + else + comp->output_ptr = (png_charpp)png_malloc(png_ptr, + (png_uint_32)(comp->max_output_ptr * + png_sizeof (png_charp))); + } + + /* save off the data */ + comp->output_ptr[comp->num_output_ptr] = + (png_charp)png_malloc(png_ptr, (png_uint_32)png_ptr->zbuf_size); + png_memcpy(comp->output_ptr[comp->num_output_ptr], png_ptr->zbuf, + png_ptr->zbuf_size); + comp->num_output_ptr++; + + /* and reset the buffer pointers */ + png_ptr->zstream.avail_out = (uInt)png_ptr->zbuf_size; + png_ptr->zstream.next_out = png_ptr->zbuf; + } + } + else if (ret != Z_STREAM_END) + { + /* we got an error */ + if (png_ptr->zstream.msg != NULL) + png_error(png_ptr, png_ptr->zstream.msg); + else + png_error(png_ptr, "zlib error"); + } + } while (ret != Z_STREAM_END); + + /* text length is number of buffers plus last buffer */ + text_len = png_ptr->zbuf_size * comp->num_output_ptr; + if (png_ptr->zstream.avail_out < png_ptr->zbuf_size) + text_len += png_ptr->zbuf_size - (png_size_t)png_ptr->zstream.avail_out; + + return((int)text_len); +} + +/* ship the compressed text out via chunk writes */ +static void /* PRIVATE */ +png_write_compressed_data_out(png_structp png_ptr, compression_state *comp) +{ + int i; + + /* handle the no-compression case */ + if (comp->input) + { + png_write_chunk_data(png_ptr, (png_bytep)comp->input, + (png_size_t)comp->input_len); + return; + } + + /* write saved output buffers, if any */ + for (i = 0; i < comp->num_output_ptr; i++) + { + png_write_chunk_data(png_ptr,(png_bytep)comp->output_ptr[i], + png_ptr->zbuf_size); + png_free(png_ptr, comp->output_ptr[i]); + comp->output_ptr[i]=NULL; + } + if (comp->max_output_ptr != 0) + png_free(png_ptr, comp->output_ptr); + comp->output_ptr=NULL; + /* write anything left in zbuf */ + if (png_ptr->zstream.avail_out < (png_uint_32)png_ptr->zbuf_size) + png_write_chunk_data(png_ptr, png_ptr->zbuf, + png_ptr->zbuf_size - png_ptr->zstream.avail_out); + + /* reset zlib for another zTXt/iTXt or image data */ + deflateReset(&png_ptr->zstream); + png_ptr->zstream.data_type = Z_BINARY; +} +#endif + +/* Write the IHDR chunk, and update the png_struct with the necessary + * information. Note that the rest of this code depends upon this + * information being correct. + */ +void /* PRIVATE */ +png_write_IHDR(png_structp png_ptr, png_uint_32 width, png_uint_32 height, + int bit_depth, int color_type, int compression_type, int filter_type, + int interlace_type) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_IHDR; +#endif + png_byte buf[13]; /* buffer to store the IHDR info */ + + png_debug(1, "in png_write_IHDR\n"); + /* Check that we have valid input data from the application info */ + switch (color_type) + { + case PNG_COLOR_TYPE_GRAY: + switch (bit_depth) + { + case 1: + case 2: + case 4: + case 8: + case 16: png_ptr->channels = 1; break; + default: png_error(png_ptr,"Invalid bit depth for grayscale image"); + } + break; + case PNG_COLOR_TYPE_RGB: + if (bit_depth != 8 && bit_depth != 16) + png_error(png_ptr, "Invalid bit depth for RGB image"); + png_ptr->channels = 3; + break; + case PNG_COLOR_TYPE_PALETTE: + switch (bit_depth) + { + case 1: + case 2: + case 4: + case 8: png_ptr->channels = 1; break; + default: png_error(png_ptr, "Invalid bit depth for paletted image"); + } + break; + case PNG_COLOR_TYPE_GRAY_ALPHA: + if (bit_depth != 8 && bit_depth != 16) + png_error(png_ptr, "Invalid bit depth for grayscale+alpha image"); + png_ptr->channels = 2; + break; + case PNG_COLOR_TYPE_RGB_ALPHA: + if (bit_depth != 8 && bit_depth != 16) + png_error(png_ptr, "Invalid bit depth for RGBA image"); + png_ptr->channels = 4; + break; + default: + png_error(png_ptr, "Invalid image color type specified"); + } + + if (compression_type != PNG_COMPRESSION_TYPE_BASE) + { + png_warning(png_ptr, "Invalid compression type specified"); + compression_type = PNG_COMPRESSION_TYPE_BASE; + } + + /* Write filter_method 64 (intrapixel differencing) only if + * 1. Libpng was compiled with PNG_MNG_FEATURES_SUPPORTED and + * 2. Libpng did not write a PNG signature (this filter_method is only + * used in PNG datastreams that are embedded in MNG datastreams) and + * 3. The application called png_permit_mng_features with a mask that + * included PNG_FLAG_MNG_FILTER_64 and + * 4. The filter_method is 64 and + * 5. The color_type is RGB or RGBA + */ + if ( +#if defined(PNG_MNG_FEATURES_SUPPORTED) + !((png_ptr->mng_features_permitted & PNG_FLAG_MNG_FILTER_64) && + ((png_ptr->mode&PNG_HAVE_PNG_SIGNATURE) == 0) && + (color_type == PNG_COLOR_TYPE_RGB || + color_type == PNG_COLOR_TYPE_RGB_ALPHA) && + (filter_type == PNG_INTRAPIXEL_DIFFERENCING)) && +#endif + filter_type != PNG_FILTER_TYPE_BASE) + { + png_warning(png_ptr, "Invalid filter type specified"); + filter_type = PNG_FILTER_TYPE_BASE; + } + +#ifdef PNG_WRITE_INTERLACING_SUPPORTED + if (interlace_type != PNG_INTERLACE_NONE && + interlace_type != PNG_INTERLACE_ADAM7) + { + png_warning(png_ptr, "Invalid interlace type specified"); + interlace_type = PNG_INTERLACE_ADAM7; + } +#else + interlace_type=PNG_INTERLACE_NONE; +#endif + + /* save off the relevent information */ + png_ptr->bit_depth = (png_byte)bit_depth; + png_ptr->color_type = (png_byte)color_type; + png_ptr->interlaced = (png_byte)interlace_type; +#if defined(PNG_MNG_FEATURES_SUPPORTED) + png_ptr->filter_type = (png_byte)filter_type; +#endif + png_ptr->compression_type = (png_byte)compression_type; + png_ptr->width = width; + png_ptr->height = height; + + png_ptr->pixel_depth = (png_byte)(bit_depth * png_ptr->channels); + png_ptr->rowbytes = PNG_ROWBYTES(png_ptr->pixel_depth, width); + /* set the usr info, so any transformations can modify it */ + png_ptr->usr_width = png_ptr->width; + png_ptr->usr_bit_depth = png_ptr->bit_depth; + png_ptr->usr_channels = png_ptr->channels; + + /* pack the header information into the buffer */ + png_save_uint_32(buf, width); + png_save_uint_32(buf + 4, height); + buf[8] = (png_byte)bit_depth; + buf[9] = (png_byte)color_type; + buf[10] = (png_byte)compression_type; + buf[11] = (png_byte)filter_type; + buf[12] = (png_byte)interlace_type; + + /* write the chunk */ + png_write_chunk(png_ptr, (png_bytep)png_IHDR, buf, (png_size_t)13); + + /* initialize zlib with PNG info */ + png_ptr->zstream.zalloc = png_zalloc; + png_ptr->zstream.zfree = png_zfree; + png_ptr->zstream.opaque = (voidpf)png_ptr; + if (!(png_ptr->do_filter)) + { + if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE || + png_ptr->bit_depth < 8) + png_ptr->do_filter = PNG_FILTER_NONE; + else + png_ptr->do_filter = PNG_ALL_FILTERS; + } + if (!(png_ptr->flags & PNG_FLAG_ZLIB_CUSTOM_STRATEGY)) + { + if (png_ptr->do_filter != PNG_FILTER_NONE) + png_ptr->zlib_strategy = Z_FILTERED; + else + png_ptr->zlib_strategy = Z_DEFAULT_STRATEGY; + } + if (!(png_ptr->flags & PNG_FLAG_ZLIB_CUSTOM_LEVEL)) + png_ptr->zlib_level = Z_DEFAULT_COMPRESSION; + if (!(png_ptr->flags & PNG_FLAG_ZLIB_CUSTOM_MEM_LEVEL)) + png_ptr->zlib_mem_level = 8; + if (!(png_ptr->flags & PNG_FLAG_ZLIB_CUSTOM_WINDOW_BITS)) + png_ptr->zlib_window_bits = 15; + if (!(png_ptr->flags & PNG_FLAG_ZLIB_CUSTOM_METHOD)) + png_ptr->zlib_method = 8; + deflateInit2(&png_ptr->zstream, png_ptr->zlib_level, + png_ptr->zlib_method, png_ptr->zlib_window_bits, + png_ptr->zlib_mem_level, png_ptr->zlib_strategy); + png_ptr->zstream.next_out = png_ptr->zbuf; + png_ptr->zstream.avail_out = (uInt)png_ptr->zbuf_size; + /* libpng is not interested in zstream.data_type */ + /* set it to a predefined value, to avoid its evaluation inside zlib */ + png_ptr->zstream.data_type = Z_BINARY; + + png_ptr->mode = PNG_HAVE_IHDR; +} + +/* write the palette. We are careful not to trust png_color to be in the + * correct order for PNG, so people can redefine it to any convenient + * structure. + */ +void /* PRIVATE */ +png_write_PLTE(png_structp png_ptr, png_colorp palette, png_uint_32 num_pal) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_PLTE; +#endif + png_uint_32 i; + png_colorp pal_ptr; + png_byte buf[3]; + + png_debug(1, "in png_write_PLTE\n"); + if (( +#if defined(PNG_MNG_FEATURES_SUPPORTED) + !(png_ptr->mng_features_permitted & PNG_FLAG_MNG_EMPTY_PLTE) && +#endif + num_pal == 0) || num_pal > 256) + { + if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE) + { + png_error(png_ptr, "Invalid number of colors in palette"); + } + else + { + png_warning(png_ptr, "Invalid number of colors in palette"); + return; + } + } + + if (!(png_ptr->color_type&PNG_COLOR_MASK_COLOR)) + { + png_warning(png_ptr, + "Ignoring request to write a PLTE chunk in grayscale PNG"); + return; + } + + png_ptr->num_palette = (png_uint_16)num_pal; + png_debug1(3, "num_palette = %d\n", png_ptr->num_palette); + + png_write_chunk_start(png_ptr, (png_bytep)png_PLTE, num_pal * 3); +#ifndef PNG_NO_POINTER_INDEXING + for (i = 0, pal_ptr = palette; i < num_pal; i++, pal_ptr++) + { + buf[0] = pal_ptr->red; + buf[1] = pal_ptr->green; + buf[2] = pal_ptr->blue; + png_write_chunk_data(png_ptr, buf, (png_size_t)3); + } +#else + /* This is a little slower but some buggy compilers need to do this instead */ + pal_ptr=palette; + for (i = 0; i < num_pal; i++) + { + buf[0] = pal_ptr[i].red; + buf[1] = pal_ptr[i].green; + buf[2] = pal_ptr[i].blue; + png_write_chunk_data(png_ptr, buf, (png_size_t)3); + } +#endif + png_write_chunk_end(png_ptr); + png_ptr->mode |= PNG_HAVE_PLTE; +} + +/* write an IDAT chunk */ +void /* PRIVATE */ +png_write_IDAT(png_structp png_ptr, png_bytep data, png_size_t length) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_IDAT; +#endif + png_debug(1, "in png_write_IDAT\n"); + + /* Optimize the CMF field in the zlib stream. */ + /* This hack of the zlib stream is compliant to the stream specification. */ + if (!(png_ptr->mode & PNG_HAVE_IDAT) && + png_ptr->compression_type == PNG_COMPRESSION_TYPE_BASE) + { + unsigned int z_cmf = data[0]; /* zlib compression method and flags */ + if ((z_cmf & 0x0f) == 8 && (z_cmf & 0xf0) <= 0x70) + { + /* Avoid memory underflows and multiplication overflows. */ + /* The conditions below are practically always satisfied; + however, they still must be checked. */ + if (length >= 2 && + png_ptr->height < 16384 && png_ptr->width < 16384) + { + png_uint_32 uncompressed_idat_size = png_ptr->height * + ((png_ptr->width * + png_ptr->channels * png_ptr->bit_depth + 15) >> 3); + unsigned int z_cinfo = z_cmf >> 4; + unsigned int half_z_window_size = 1 << (z_cinfo + 7); + while (uncompressed_idat_size <= half_z_window_size && + half_z_window_size >= 256) + { + z_cinfo--; + half_z_window_size >>= 1; + } + z_cmf = (z_cmf & 0x0f) | (z_cinfo << 4); + if (data[0] != (png_byte)z_cmf) + { + data[0] = (png_byte)z_cmf; + data[1] &= 0xe0; + data[1] += (png_byte)(0x1f - ((z_cmf << 8) + data[1]) % 0x1f); + } + } + } + else + png_error(png_ptr, + "Invalid zlib compression method or flags in IDAT"); + } + + png_write_chunk(png_ptr, (png_bytep)png_IDAT, data, length); + png_ptr->mode |= PNG_HAVE_IDAT; +} + +/* write an IEND chunk */ +void /* PRIVATE */ +png_write_IEND(png_structp png_ptr) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_IEND; +#endif + png_debug(1, "in png_write_IEND\n"); + png_write_chunk(png_ptr, (png_bytep)png_IEND, png_bytep_NULL, + (png_size_t)0); + png_ptr->mode |= PNG_HAVE_IEND; +} + +#if defined(PNG_WRITE_gAMA_SUPPORTED) +/* write a gAMA chunk */ +#ifdef PNG_FLOATING_POINT_SUPPORTED +void /* PRIVATE */ +png_write_gAMA(png_structp png_ptr, double file_gamma) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_gAMA; +#endif + png_uint_32 igamma; + png_byte buf[4]; + + png_debug(1, "in png_write_gAMA\n"); + /* file_gamma is saved in 1/100,000ths */ + igamma = (png_uint_32)(file_gamma * 100000.0 + 0.5); + png_save_uint_32(buf, igamma); + png_write_chunk(png_ptr, (png_bytep)png_gAMA, buf, (png_size_t)4); +} +#endif +#ifdef PNG_FIXED_POINT_SUPPORTED +void /* PRIVATE */ +png_write_gAMA_fixed(png_structp png_ptr, png_fixed_point file_gamma) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_gAMA; +#endif + png_byte buf[4]; + + png_debug(1, "in png_write_gAMA\n"); + /* file_gamma is saved in 1/100,000ths */ + png_save_uint_32(buf, (png_uint_32)file_gamma); + png_write_chunk(png_ptr, (png_bytep)png_gAMA, buf, (png_size_t)4); +} +#endif +#endif + +#if defined(PNG_WRITE_sRGB_SUPPORTED) +/* write a sRGB chunk */ +void /* PRIVATE */ +png_write_sRGB(png_structp png_ptr, int srgb_intent) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_sRGB; +#endif + png_byte buf[1]; + + png_debug(1, "in png_write_sRGB\n"); + if(srgb_intent >= PNG_sRGB_INTENT_LAST) + png_warning(png_ptr, + "Invalid sRGB rendering intent specified"); + buf[0]=(png_byte)srgb_intent; + png_write_chunk(png_ptr, (png_bytep)png_sRGB, buf, (png_size_t)1); +} +#endif + +#if defined(PNG_WRITE_iCCP_SUPPORTED) +/* write an iCCP chunk */ +void /* PRIVATE */ +png_write_iCCP(png_structp png_ptr, png_charp name, int compression_type, + png_charp profile, int profile_len) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_iCCP; +#endif + png_size_t name_len; + png_charp new_name; + compression_state comp; + + png_debug(1, "in png_write_iCCP\n"); + + comp.num_output_ptr = 0; + comp.max_output_ptr = 0; + comp.output_ptr = NULL; + comp.input = NULL; + comp.input_len = 0; + + if (name == NULL || (name_len = png_check_keyword(png_ptr, name, + &new_name)) == 0) + { + png_warning(png_ptr, "Empty keyword in iCCP chunk"); + return; + } + + if (compression_type != PNG_COMPRESSION_TYPE_BASE) + png_warning(png_ptr, "Unknown compression type in iCCP chunk"); + + if (profile == NULL) + profile_len = 0; + + if (profile_len) + profile_len = png_text_compress(png_ptr, profile, (png_size_t)profile_len, + PNG_COMPRESSION_TYPE_BASE, &comp); + + /* make sure we include the NULL after the name and the compression type */ + png_write_chunk_start(png_ptr, (png_bytep)png_iCCP, + (png_uint_32)name_len+profile_len+2); + new_name[name_len+1]=0x00; + png_write_chunk_data(png_ptr, (png_bytep)new_name, name_len + 2); + + if (profile_len) + png_write_compressed_data_out(png_ptr, &comp); + + png_write_chunk_end(png_ptr); + png_free(png_ptr, new_name); +} +#endif + +#if defined(PNG_WRITE_sPLT_SUPPORTED) +/* write a sPLT chunk */ +void /* PRIVATE */ +png_write_sPLT(png_structp png_ptr, png_sPLT_tp spalette) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_sPLT; +#endif + png_size_t name_len; + png_charp new_name; + png_byte entrybuf[10]; + int entry_size = (spalette->depth == 8 ? 6 : 10); + int palette_size = entry_size * spalette->nentries; + png_sPLT_entryp ep; +#ifdef PNG_NO_POINTER_INDEXING + int i; +#endif + + png_debug(1, "in png_write_sPLT\n"); + if (spalette->name == NULL || (name_len = png_check_keyword(png_ptr, + spalette->name, &new_name))==0) + { + png_warning(png_ptr, "Empty keyword in sPLT chunk"); + return; + } + + /* make sure we include the NULL after the name */ + png_write_chunk_start(png_ptr, (png_bytep)png_sPLT, + (png_uint_32)(name_len + 2 + palette_size)); + png_write_chunk_data(png_ptr, (png_bytep)new_name, name_len + 1); + png_write_chunk_data(png_ptr, (png_bytep)&spalette->depth, 1); + + /* loop through each palette entry, writing appropriately */ +#ifndef PNG_NO_POINTER_INDEXING + for (ep = spalette->entries; epentries+spalette->nentries; ep++) + { + if (spalette->depth == 8) + { + entrybuf[0] = (png_byte)ep->red; + entrybuf[1] = (png_byte)ep->green; + entrybuf[2] = (png_byte)ep->blue; + entrybuf[3] = (png_byte)ep->alpha; + png_save_uint_16(entrybuf + 4, ep->frequency); + } + else + { + png_save_uint_16(entrybuf + 0, ep->red); + png_save_uint_16(entrybuf + 2, ep->green); + png_save_uint_16(entrybuf + 4, ep->blue); + png_save_uint_16(entrybuf + 6, ep->alpha); + png_save_uint_16(entrybuf + 8, ep->frequency); + } + png_write_chunk_data(png_ptr, entrybuf, (png_size_t)entry_size); + } +#else + ep=spalette->entries; + for (i=0; i>spalette->nentries; i++) + { + if (spalette->depth == 8) + { + entrybuf[0] = (png_byte)ep[i].red; + entrybuf[1] = (png_byte)ep[i].green; + entrybuf[2] = (png_byte)ep[i].blue; + entrybuf[3] = (png_byte)ep[i].alpha; + png_save_uint_16(entrybuf + 4, ep[i].frequency); + } + else + { + png_save_uint_16(entrybuf + 0, ep[i].red); + png_save_uint_16(entrybuf + 2, ep[i].green); + png_save_uint_16(entrybuf + 4, ep[i].blue); + png_save_uint_16(entrybuf + 6, ep[i].alpha); + png_save_uint_16(entrybuf + 8, ep[i].frequency); + } + png_write_chunk_data(png_ptr, entrybuf, entry_size); + } +#endif + + png_write_chunk_end(png_ptr); + png_free(png_ptr, new_name); +} +#endif + +#if defined(PNG_WRITE_sBIT_SUPPORTED) +/* write the sBIT chunk */ +void /* PRIVATE */ +png_write_sBIT(png_structp png_ptr, png_color_8p sbit, int color_type) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_sBIT; +#endif + png_byte buf[4]; + png_size_t size; + + png_debug(1, "in png_write_sBIT\n"); + /* make sure we don't depend upon the order of PNG_COLOR_8 */ + if (color_type & PNG_COLOR_MASK_COLOR) + { + png_byte maxbits; + + maxbits = (png_byte)(color_type==PNG_COLOR_TYPE_PALETTE ? 8 : + png_ptr->usr_bit_depth); + if (sbit->red == 0 || sbit->red > maxbits || + sbit->green == 0 || sbit->green > maxbits || + sbit->blue == 0 || sbit->blue > maxbits) + { + png_warning(png_ptr, "Invalid sBIT depth specified"); + return; + } + buf[0] = sbit->red; + buf[1] = sbit->green; + buf[2] = sbit->blue; + size = 3; + } + else + { + if (sbit->gray == 0 || sbit->gray > png_ptr->usr_bit_depth) + { + png_warning(png_ptr, "Invalid sBIT depth specified"); + return; + } + buf[0] = sbit->gray; + size = 1; + } + + if (color_type & PNG_COLOR_MASK_ALPHA) + { + if (sbit->alpha == 0 || sbit->alpha > png_ptr->usr_bit_depth) + { + png_warning(png_ptr, "Invalid sBIT depth specified"); + return; + } + buf[size++] = sbit->alpha; + } + + png_write_chunk(png_ptr, (png_bytep)png_sBIT, buf, size); +} +#endif + +#if defined(PNG_WRITE_cHRM_SUPPORTED) +/* write the cHRM chunk */ +#ifdef PNG_FLOATING_POINT_SUPPORTED +void /* PRIVATE */ +png_write_cHRM(png_structp png_ptr, double white_x, double white_y, + double red_x, double red_y, double green_x, double green_y, + double blue_x, double blue_y) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_cHRM; +#endif + png_byte buf[32]; + png_uint_32 itemp; + + png_debug(1, "in png_write_cHRM\n"); + /* each value is saved in 1/100,000ths */ + if (white_x < 0 || white_x > 0.8 || white_y < 0 || white_y > 0.8 || + white_x + white_y > 1.0) + { + png_warning(png_ptr, "Invalid cHRM white point specified"); +#if !defined(PNG_NO_CONSOLE_IO) + fprintf(stderr,"white_x=%f, white_y=%f\n",white_x, white_y); +#endif + return; + } + itemp = (png_uint_32)(white_x * 100000.0 + 0.5); + png_save_uint_32(buf, itemp); + itemp = (png_uint_32)(white_y * 100000.0 + 0.5); + png_save_uint_32(buf + 4, itemp); + + if (red_x < 0 || red_y < 0 || red_x + red_y > 1.0) + { + png_warning(png_ptr, "Invalid cHRM red point specified"); + return; + } + itemp = (png_uint_32)(red_x * 100000.0 + 0.5); + png_save_uint_32(buf + 8, itemp); + itemp = (png_uint_32)(red_y * 100000.0 + 0.5); + png_save_uint_32(buf + 12, itemp); + + if (green_x < 0 || green_y < 0 || green_x + green_y > 1.0) + { + png_warning(png_ptr, "Invalid cHRM green point specified"); + return; + } + itemp = (png_uint_32)(green_x * 100000.0 + 0.5); + png_save_uint_32(buf + 16, itemp); + itemp = (png_uint_32)(green_y * 100000.0 + 0.5); + png_save_uint_32(buf + 20, itemp); + + if (blue_x < 0 || blue_y < 0 || blue_x + blue_y > 1.0) + { + png_warning(png_ptr, "Invalid cHRM blue point specified"); + return; + } + itemp = (png_uint_32)(blue_x * 100000.0 + 0.5); + png_save_uint_32(buf + 24, itemp); + itemp = (png_uint_32)(blue_y * 100000.0 + 0.5); + png_save_uint_32(buf + 28, itemp); + + png_write_chunk(png_ptr, (png_bytep)png_cHRM, buf, (png_size_t)32); +} +#endif +#ifdef PNG_FIXED_POINT_SUPPORTED +void /* PRIVATE */ +png_write_cHRM_fixed(png_structp png_ptr, png_fixed_point white_x, + png_fixed_point white_y, png_fixed_point red_x, png_fixed_point red_y, + png_fixed_point green_x, png_fixed_point green_y, png_fixed_point blue_x, + png_fixed_point blue_y) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_cHRM; +#endif + png_byte buf[32]; + + png_debug(1, "in png_write_cHRM\n"); + /* each value is saved in 1/100,000ths */ + if (white_x > 80000L || white_y > 80000L || white_x + white_y > 100000L) + { + png_warning(png_ptr, "Invalid fixed cHRM white point specified"); +#if !defined(PNG_NO_CONSOLE_IO) + fprintf(stderr,"white_x=%ld, white_y=%ld\n",white_x, white_y); +#endif + return; + } + png_save_uint_32(buf, (png_uint_32)white_x); + png_save_uint_32(buf + 4, (png_uint_32)white_y); + + if (red_x + red_y > 100000L) + { + png_warning(png_ptr, "Invalid cHRM fixed red point specified"); + return; + } + png_save_uint_32(buf + 8, (png_uint_32)red_x); + png_save_uint_32(buf + 12, (png_uint_32)red_y); + + if (green_x + green_y > 100000L) + { + png_warning(png_ptr, "Invalid fixed cHRM green point specified"); + return; + } + png_save_uint_32(buf + 16, (png_uint_32)green_x); + png_save_uint_32(buf + 20, (png_uint_32)green_y); + + if (blue_x + blue_y > 100000L) + { + png_warning(png_ptr, "Invalid fixed cHRM blue point specified"); + return; + } + png_save_uint_32(buf + 24, (png_uint_32)blue_x); + png_save_uint_32(buf + 28, (png_uint_32)blue_y); + + png_write_chunk(png_ptr, (png_bytep)png_cHRM, buf, (png_size_t)32); +} +#endif +#endif + +#if defined(PNG_WRITE_tRNS_SUPPORTED) +/* write the tRNS chunk */ +void /* PRIVATE */ +png_write_tRNS(png_structp png_ptr, png_bytep trans, png_color_16p tran, + int num_trans, int color_type) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_tRNS; +#endif + png_byte buf[6]; + + png_debug(1, "in png_write_tRNS\n"); + if (color_type == PNG_COLOR_TYPE_PALETTE) + { + if (num_trans <= 0 || num_trans > (int)png_ptr->num_palette) + { + png_warning(png_ptr,"Invalid number of transparent colors specified"); + return; + } + /* write the chunk out as it is */ + png_write_chunk(png_ptr, (png_bytep)png_tRNS, trans, (png_size_t)num_trans); + } + else if (color_type == PNG_COLOR_TYPE_GRAY) + { + /* one 16 bit value */ + if(tran->gray >= (1 << png_ptr->bit_depth)) + { + png_warning(png_ptr, + "Ignoring attempt to write tRNS chunk out-of-range for bit_depth"); + return; + } + png_save_uint_16(buf, tran->gray); + png_write_chunk(png_ptr, (png_bytep)png_tRNS, buf, (png_size_t)2); + } + else if (color_type == PNG_COLOR_TYPE_RGB) + { + /* three 16 bit values */ + png_save_uint_16(buf, tran->red); + png_save_uint_16(buf + 2, tran->green); + png_save_uint_16(buf + 4, tran->blue); + if(png_ptr->bit_depth == 8 && (buf[0] | buf[2] | buf[4])) + { + png_warning(png_ptr, + "Ignoring attempt to write 16-bit tRNS chunk when bit_depth is 8"); + return; + } + png_write_chunk(png_ptr, (png_bytep)png_tRNS, buf, (png_size_t)6); + } + else + { + png_warning(png_ptr, "Can't write tRNS with an alpha channel"); + } +} +#endif + +#if defined(PNG_WRITE_bKGD_SUPPORTED) +/* write the background chunk */ +void /* PRIVATE */ +png_write_bKGD(png_structp png_ptr, png_color_16p back, int color_type) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_bKGD; +#endif + png_byte buf[6]; + + png_debug(1, "in png_write_bKGD\n"); + if (color_type == PNG_COLOR_TYPE_PALETTE) + { + if ( +#if defined(PNG_MNG_FEATURES_SUPPORTED) + (png_ptr->num_palette || + (!(png_ptr->mng_features_permitted & PNG_FLAG_MNG_EMPTY_PLTE))) && +#endif + back->index > png_ptr->num_palette) + { + png_warning(png_ptr, "Invalid background palette index"); + return; + } + buf[0] = back->index; + png_write_chunk(png_ptr, (png_bytep)png_bKGD, buf, (png_size_t)1); + } + else if (color_type & PNG_COLOR_MASK_COLOR) + { + png_save_uint_16(buf, back->red); + png_save_uint_16(buf + 2, back->green); + png_save_uint_16(buf + 4, back->blue); + if(png_ptr->bit_depth == 8 && (buf[0] | buf[2] | buf[4])) + { + png_warning(png_ptr, + "Ignoring attempt to write 16-bit bKGD chunk when bit_depth is 8"); + return; + } + png_write_chunk(png_ptr, (png_bytep)png_bKGD, buf, (png_size_t)6); + } + else + { + if(back->gray >= (1 << png_ptr->bit_depth)) + { + png_warning(png_ptr, + "Ignoring attempt to write bKGD chunk out-of-range for bit_depth"); + return; + } + png_save_uint_16(buf, back->gray); + png_write_chunk(png_ptr, (png_bytep)png_bKGD, buf, (png_size_t)2); + } +} +#endif + +#if defined(PNG_WRITE_hIST_SUPPORTED) +/* write the histogram */ +void /* PRIVATE */ +png_write_hIST(png_structp png_ptr, png_uint_16p hist, int num_hist) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_hIST; +#endif + int i; + png_byte buf[3]; + + png_debug(1, "in png_write_hIST\n"); + if (num_hist > (int)png_ptr->num_palette) + { + png_debug2(3, "num_hist = %d, num_palette = %d\n", num_hist, + png_ptr->num_palette); + png_warning(png_ptr, "Invalid number of histogram entries specified"); + return; + } + + png_write_chunk_start(png_ptr, (png_bytep)png_hIST, (png_uint_32)(num_hist * 2)); + for (i = 0; i < num_hist; i++) + { + png_save_uint_16(buf, hist[i]); + png_write_chunk_data(png_ptr, buf, (png_size_t)2); + } + png_write_chunk_end(png_ptr); +} +#endif + +#if defined(PNG_WRITE_TEXT_SUPPORTED) || defined(PNG_WRITE_pCAL_SUPPORTED) || \ + defined(PNG_WRITE_iCCP_SUPPORTED) || defined(PNG_WRITE_sPLT_SUPPORTED) +/* Check that the tEXt or zTXt keyword is valid per PNG 1.0 specification, + * and if invalid, correct the keyword rather than discarding the entire + * chunk. The PNG 1.0 specification requires keywords 1-79 characters in + * length, forbids leading or trailing whitespace, multiple internal spaces, + * and the non-break space (0x80) from ISO 8859-1. Returns keyword length. + * + * The new_key is allocated to hold the corrected keyword and must be freed + * by the calling routine. This avoids problems with trying to write to + * static keywords without having to have duplicate copies of the strings. + */ +png_size_t /* PRIVATE */ +png_check_keyword(png_structp png_ptr, png_charp key, png_charpp new_key) +{ + png_size_t key_len; + png_charp kp, dp; + int kflag; + int kwarn=0; + + png_debug(1, "in png_check_keyword\n"); + *new_key = NULL; + + if (key == NULL || (key_len = png_strlen(key)) == 0) + { + png_warning(png_ptr, "zero length keyword"); + return ((png_size_t)0); + } + + png_debug1(2, "Keyword to be checked is '%s'\n", key); + + *new_key = (png_charp)png_malloc_warn(png_ptr, (png_uint_32)(key_len + 2)); + if (*new_key == NULL) + { + png_warning(png_ptr, "Out of memory while procesing keyword"); + return ((png_size_t)0); + } + + /* Replace non-printing characters with a blank and print a warning */ + for (kp = key, dp = *new_key; *kp != '\0'; kp++, dp++) + { + if (*kp < 0x20 || (*kp > 0x7E && (png_byte)*kp < 0xA1)) + { +#if !defined(PNG_NO_STDIO) && !defined(_WIN32_WCE) + char msg[40]; + + sprintf(msg, "invalid keyword character 0x%02X", *kp); + png_warning(png_ptr, msg); +#else + png_warning(png_ptr, "invalid character in keyword"); +#endif + *dp = ' '; + } + else + { + *dp = *kp; + } + } + *dp = '\0'; + + /* Remove any trailing white space. */ + kp = *new_key + key_len - 1; + if (*kp == ' ') + { + png_warning(png_ptr, "trailing spaces removed from keyword"); + + while (*kp == ' ') + { + *(kp--) = '\0'; + key_len--; + } + } + + /* Remove any leading white space. */ + kp = *new_key; + if (*kp == ' ') + { + png_warning(png_ptr, "leading spaces removed from keyword"); + + while (*kp == ' ') + { + kp++; + key_len--; + } + } + + png_debug1(2, "Checking for multiple internal spaces in '%s'\n", kp); + + /* Remove multiple internal spaces. */ + for (kflag = 0, dp = *new_key; *kp != '\0'; kp++) + { + if (*kp == ' ' && kflag == 0) + { + *(dp++) = *kp; + kflag = 1; + } + else if (*kp == ' ') + { + key_len--; + kwarn=1; + } + else + { + *(dp++) = *kp; + kflag = 0; + } + } + *dp = '\0'; + if(kwarn) + png_warning(png_ptr, "extra interior spaces removed from keyword"); + + if (key_len == 0) + { + png_free(png_ptr, *new_key); + *new_key=NULL; + png_warning(png_ptr, "Zero length keyword"); + } + + if (key_len > 79) + { + png_warning(png_ptr, "keyword length must be 1 - 79 characters"); + new_key[79] = '\0'; + key_len = 79; + } + + return (key_len); +} +#endif + +#if defined(PNG_WRITE_tEXt_SUPPORTED) +/* write a tEXt chunk */ +void /* PRIVATE */ +png_write_tEXt(png_structp png_ptr, png_charp key, png_charp text, + png_size_t text_len) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_tEXt; +#endif + png_size_t key_len; + png_charp new_key; + + png_debug(1, "in png_write_tEXt\n"); + if (key == NULL || (key_len = png_check_keyword(png_ptr, key, &new_key))==0) + { + png_warning(png_ptr, "Empty keyword in tEXt chunk"); + return; + } + + if (text == NULL || *text == '\0') + text_len = 0; + else + text_len = png_strlen(text); + + /* make sure we include the 0 after the key */ + png_write_chunk_start(png_ptr, (png_bytep)png_tEXt, (png_uint_32)key_len+text_len+1); + /* + * We leave it to the application to meet PNG-1.0 requirements on the + * contents of the text. PNG-1.0 through PNG-1.2 discourage the use of + * any non-Latin-1 characters except for NEWLINE. ISO PNG will forbid them. + * The NUL character is forbidden by PNG-1.0 through PNG-1.2 and ISO PNG. + */ + png_write_chunk_data(png_ptr, (png_bytep)new_key, key_len + 1); + if (text_len) + png_write_chunk_data(png_ptr, (png_bytep)text, text_len); + + png_write_chunk_end(png_ptr); + png_free(png_ptr, new_key); +} +#endif + +#if defined(PNG_WRITE_zTXt_SUPPORTED) +/* write a compressed text chunk */ +void /* PRIVATE */ +png_write_zTXt(png_structp png_ptr, png_charp key, png_charp text, + png_size_t text_len, int compression) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_zTXt; +#endif + png_size_t key_len; + char buf[1]; + png_charp new_key; + compression_state comp; + + png_debug(1, "in png_write_zTXt\n"); + + comp.num_output_ptr = 0; + comp.max_output_ptr = 0; + comp.output_ptr = NULL; + comp.input = NULL; + comp.input_len = 0; + + if (key == NULL || (key_len = png_check_keyword(png_ptr, key, &new_key))==0) + { + png_warning(png_ptr, "Empty keyword in zTXt chunk"); + return; + } + + if (text == NULL || *text == '\0' || compression==PNG_TEXT_COMPRESSION_NONE) + { + png_write_tEXt(png_ptr, new_key, text, (png_size_t)0); + png_free(png_ptr, new_key); + return; + } + + text_len = png_strlen(text); + + png_free(png_ptr, new_key); + + /* compute the compressed data; do it now for the length */ + text_len = png_text_compress(png_ptr, text, text_len, compression, + &comp); + + /* write start of chunk */ + png_write_chunk_start(png_ptr, (png_bytep)png_zTXt, (png_uint_32) + (key_len+text_len+2)); + /* write key */ + png_write_chunk_data(png_ptr, (png_bytep)key, key_len + 1); + buf[0] = (png_byte)compression; + /* write compression */ + png_write_chunk_data(png_ptr, (png_bytep)buf, (png_size_t)1); + /* write the compressed data */ + png_write_compressed_data_out(png_ptr, &comp); + + /* close the chunk */ + png_write_chunk_end(png_ptr); +} +#endif + +#if defined(PNG_WRITE_iTXt_SUPPORTED) +/* write an iTXt chunk */ +void /* PRIVATE */ +png_write_iTXt(png_structp png_ptr, int compression, png_charp key, + png_charp lang, png_charp lang_key, png_charp text) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_iTXt; +#endif + png_size_t lang_len, key_len, lang_key_len, text_len; + png_charp new_lang, new_key; + png_byte cbuf[2]; + compression_state comp; + + png_debug(1, "in png_write_iTXt\n"); + + comp.num_output_ptr = 0; + comp.max_output_ptr = 0; + comp.output_ptr = NULL; + comp.input = NULL; + + if (key == NULL || (key_len = png_check_keyword(png_ptr, key, &new_key))==0) + { + png_warning(png_ptr, "Empty keyword in iTXt chunk"); + return; + } + if (lang == NULL || (lang_len = png_check_keyword(png_ptr, lang, &new_lang))==0) + { + png_warning(png_ptr, "Empty language field in iTXt chunk"); + new_lang = NULL; + lang_len = 0; + } + + if (lang_key == NULL) + lang_key_len = 0; + else + lang_key_len = png_strlen(lang_key); + + if (text == NULL) + text_len = 0; + else + text_len = png_strlen(text); + + /* compute the compressed data; do it now for the length */ + text_len = png_text_compress(png_ptr, text, text_len, compression-2, + &comp); + + + /* make sure we include the compression flag, the compression byte, + * and the NULs after the key, lang, and lang_key parts */ + + png_write_chunk_start(png_ptr, (png_bytep)png_iTXt, + (png_uint_32)( + 5 /* comp byte, comp flag, terminators for key, lang and lang_key */ + + key_len + + lang_len + + lang_key_len + + text_len)); + + /* + * We leave it to the application to meet PNG-1.0 requirements on the + * contents of the text. PNG-1.0 through PNG-1.2 discourage the use of + * any non-Latin-1 characters except for NEWLINE. ISO PNG will forbid them. + * The NUL character is forbidden by PNG-1.0 through PNG-1.2 and ISO PNG. + */ + png_write_chunk_data(png_ptr, (png_bytep)new_key, key_len + 1); + + /* set the compression flag */ + if (compression == PNG_ITXT_COMPRESSION_NONE || \ + compression == PNG_TEXT_COMPRESSION_NONE) + cbuf[0] = 0; + else /* compression == PNG_ITXT_COMPRESSION_zTXt */ + cbuf[0] = 1; + /* set the compression method */ + cbuf[1] = 0; + png_write_chunk_data(png_ptr, cbuf, 2); + + cbuf[0] = 0; + png_write_chunk_data(png_ptr, (new_lang ? (png_bytep)new_lang : cbuf), lang_len + 1); + png_write_chunk_data(png_ptr, (lang_key ? (png_bytep)lang_key : cbuf), lang_key_len + 1); + png_write_compressed_data_out(png_ptr, &comp); + + png_write_chunk_end(png_ptr); + png_free(png_ptr, new_key); + if (new_lang) + png_free(png_ptr, new_lang); +} +#endif + +#if defined(PNG_WRITE_oFFs_SUPPORTED) +/* write the oFFs chunk */ +void /* PRIVATE */ +png_write_oFFs(png_structp png_ptr, png_int_32 x_offset, png_int_32 y_offset, + int unit_type) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_oFFs; +#endif + png_byte buf[9]; + + png_debug(1, "in png_write_oFFs\n"); + if (unit_type >= PNG_OFFSET_LAST) + png_warning(png_ptr, "Unrecognized unit type for oFFs chunk"); + + png_save_int_32(buf, x_offset); + png_save_int_32(buf + 4, y_offset); + buf[8] = (png_byte)unit_type; + + png_write_chunk(png_ptr, (png_bytep)png_oFFs, buf, (png_size_t)9); +} +#endif + +#if defined(PNG_WRITE_pCAL_SUPPORTED) +/* write the pCAL chunk (described in the PNG extensions document) */ +void /* PRIVATE */ +png_write_pCAL(png_structp png_ptr, png_charp purpose, png_int_32 X0, + png_int_32 X1, int type, int nparams, png_charp units, png_charpp params) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_pCAL; +#endif + png_size_t purpose_len, units_len, total_len; + png_uint_32p params_len; + png_byte buf[10]; + png_charp new_purpose; + int i; + + png_debug1(1, "in png_write_pCAL (%d parameters)\n", nparams); + if (type >= PNG_EQUATION_LAST) + png_warning(png_ptr, "Unrecognized equation type for pCAL chunk"); + + purpose_len = png_check_keyword(png_ptr, purpose, &new_purpose) + 1; + png_debug1(3, "pCAL purpose length = %d\n", (int)purpose_len); + units_len = png_strlen(units) + (nparams == 0 ? 0 : 1); + png_debug1(3, "pCAL units length = %d\n", (int)units_len); + total_len = purpose_len + units_len + 10; + + params_len = (png_uint_32p)png_malloc(png_ptr, (png_uint_32)(nparams + *png_sizeof(png_uint_32))); + + /* Find the length of each parameter, making sure we don't count the + null terminator for the last parameter. */ + for (i = 0; i < nparams; i++) + { + params_len[i] = png_strlen(params[i]) + (i == nparams - 1 ? 0 : 1); + png_debug2(3, "pCAL parameter %d length = %lu\n", i, params_len[i]); + total_len += (png_size_t)params_len[i]; + } + + png_debug1(3, "pCAL total length = %d\n", (int)total_len); + png_write_chunk_start(png_ptr, (png_bytep)png_pCAL, (png_uint_32)total_len); + png_write_chunk_data(png_ptr, (png_bytep)new_purpose, purpose_len); + png_save_int_32(buf, X0); + png_save_int_32(buf + 4, X1); + buf[8] = (png_byte)type; + buf[9] = (png_byte)nparams; + png_write_chunk_data(png_ptr, buf, (png_size_t)10); + png_write_chunk_data(png_ptr, (png_bytep)units, (png_size_t)units_len); + + png_free(png_ptr, new_purpose); + + for (i = 0; i < nparams; i++) + { + png_write_chunk_data(png_ptr, (png_bytep)params[i], + (png_size_t)params_len[i]); + } + + png_free(png_ptr, params_len); + png_write_chunk_end(png_ptr); +} +#endif + +#if defined(PNG_WRITE_sCAL_SUPPORTED) +/* write the sCAL chunk */ +#if defined(PNG_FLOATING_POINT_SUPPORTED) && !defined(PNG_NO_STDIO) +void /* PRIVATE */ +png_write_sCAL(png_structp png_ptr, int unit, double width,double height) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_sCAL; +#endif + png_size_t total_len; + char wbuf[32], hbuf[32]; + png_byte bunit = (png_byte)unit; + + png_debug(1, "in png_write_sCAL\n"); + +#if defined(_WIN32_WCE) +/* sprintf() function is not supported on WindowsCE */ + { + wchar_t wc_buf[32]; + swprintf(wc_buf, TEXT("%12.12e"), width); + WideCharToMultiByte(CP_ACP, 0, wc_buf, -1, wbuf, 32, NULL, NULL); + swprintf(wc_buf, TEXT("%12.12e"), height); + WideCharToMultiByte(CP_ACP, 0, wc_buf, -1, hbuf, 32, NULL, NULL); + } +#else + sprintf(wbuf, "%12.12e", width); + sprintf(hbuf, "%12.12e", height); +#endif + total_len = 1 + png_strlen(wbuf)+1 + png_strlen(hbuf); + + png_debug1(3, "sCAL total length = %d\n", (int)total_len); + png_write_chunk_start(png_ptr, (png_bytep)png_sCAL, (png_uint_32)total_len); + png_write_chunk_data(png_ptr, (png_bytep)&bunit, 1); + png_write_chunk_data(png_ptr, (png_bytep)wbuf, png_strlen(wbuf)+1); + png_write_chunk_data(png_ptr, (png_bytep)hbuf, png_strlen(hbuf)); + + png_write_chunk_end(png_ptr); +} +#else +#ifdef PNG_FIXED_POINT_SUPPORTED +void /* PRIVATE */ +png_write_sCAL_s(png_structp png_ptr, int unit, png_charp width, + png_charp height) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_sCAL; +#endif + png_size_t total_len; + char wbuf[32], hbuf[32]; + png_byte bunit = unit; + + png_debug(1, "in png_write_sCAL_s\n"); + + png_strcpy(wbuf,(const char *)width); + png_strcpy(hbuf,(const char *)height); + total_len = 1 + png_strlen(wbuf)+1 + png_strlen(hbuf); + + png_debug1(3, "sCAL total length = %d\n", total_len); + png_write_chunk_start(png_ptr, (png_bytep)png_sCAL, (png_uint_32)total_len); + png_write_chunk_data(png_ptr, (png_bytep)&bunit, 1); + png_write_chunk_data(png_ptr, (png_bytep)wbuf, png_strlen(wbuf)+1); + png_write_chunk_data(png_ptr, (png_bytep)hbuf, png_strlen(hbuf)); + + png_write_chunk_end(png_ptr); +} +#endif +#endif +#endif + +#if defined(PNG_WRITE_pHYs_SUPPORTED) +/* write the pHYs chunk */ +void /* PRIVATE */ +png_write_pHYs(png_structp png_ptr, png_uint_32 x_pixels_per_unit, + png_uint_32 y_pixels_per_unit, + int unit_type) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_pHYs; +#endif + png_byte buf[9]; + + png_debug(1, "in png_write_pHYs\n"); + if (unit_type >= PNG_RESOLUTION_LAST) + png_warning(png_ptr, "Unrecognized unit type for pHYs chunk"); + + png_save_uint_32(buf, x_pixels_per_unit); + png_save_uint_32(buf + 4, y_pixels_per_unit); + buf[8] = (png_byte)unit_type; + + png_write_chunk(png_ptr, (png_bytep)png_pHYs, buf, (png_size_t)9); +} +#endif + +#if defined(PNG_WRITE_tIME_SUPPORTED) +/* Write the tIME chunk. Use either png_convert_from_struct_tm() + * or png_convert_from_time_t(), or fill in the structure yourself. + */ +void /* PRIVATE */ +png_write_tIME(png_structp png_ptr, png_timep mod_time) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + PNG_tIME; +#endif + png_byte buf[7]; + + png_debug(1, "in png_write_tIME\n"); + if (mod_time->month > 12 || mod_time->month < 1 || + mod_time->day > 31 || mod_time->day < 1 || + mod_time->hour > 23 || mod_time->second > 60) + { + png_warning(png_ptr, "Invalid time specified for tIME chunk"); + return; + } + + png_save_uint_16(buf, mod_time->year); + buf[2] = mod_time->month; + buf[3] = mod_time->day; + buf[4] = mod_time->hour; + buf[5] = mod_time->minute; + buf[6] = mod_time->second; + + png_write_chunk(png_ptr, (png_bytep)png_tIME, buf, (png_size_t)7); +} +#endif + +/* initializes the row writing capability of libpng */ +void /* PRIVATE */ +png_write_start_row(png_structp png_ptr) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + /* arrays to facilitate easy interlacing - use pass (0 - 6) as index */ + + /* start of interlace block */ + int png_pass_start[7] = {0, 4, 0, 2, 0, 1, 0}; + + /* offset to next interlace block */ + int png_pass_inc[7] = {8, 8, 4, 4, 2, 2, 1}; + + /* start of interlace block in the y direction */ + int png_pass_ystart[7] = {0, 0, 4, 0, 2, 0, 1}; + + /* offset to next interlace block in the y direction */ + int png_pass_yinc[7] = {8, 8, 8, 4, 4, 2, 2}; +#endif + + png_size_t buf_size; + + png_debug(1, "in png_write_start_row\n"); + buf_size = (png_size_t)(PNG_ROWBYTES( + png_ptr->usr_channels*png_ptr->usr_bit_depth,png_ptr->width)+1); + + /* set up row buffer */ + png_ptr->row_buf = (png_bytep)png_malloc(png_ptr, (png_uint_32)buf_size); + png_ptr->row_buf[0] = PNG_FILTER_VALUE_NONE; + + /* set up filtering buffer, if using this filter */ + if (png_ptr->do_filter & PNG_FILTER_SUB) + { + png_ptr->sub_row = (png_bytep)png_malloc(png_ptr, + (png_ptr->rowbytes + 1)); + png_ptr->sub_row[0] = PNG_FILTER_VALUE_SUB; + } + + /* We only need to keep the previous row if we are using one of these. */ + if (png_ptr->do_filter & (PNG_FILTER_AVG | PNG_FILTER_UP | PNG_FILTER_PAETH)) + { + /* set up previous row buffer */ + png_ptr->prev_row = (png_bytep)png_malloc(png_ptr, (png_uint_32)buf_size); + png_memset(png_ptr->prev_row, 0, buf_size); + + if (png_ptr->do_filter & PNG_FILTER_UP) + { + png_ptr->up_row = (png_bytep )png_malloc(png_ptr, + (png_ptr->rowbytes + 1)); + png_ptr->up_row[0] = PNG_FILTER_VALUE_UP; + } + + if (png_ptr->do_filter & PNG_FILTER_AVG) + { + png_ptr->avg_row = (png_bytep)png_malloc(png_ptr, + (png_ptr->rowbytes + 1)); + png_ptr->avg_row[0] = PNG_FILTER_VALUE_AVG; + } + + if (png_ptr->do_filter & PNG_FILTER_PAETH) + { + png_ptr->paeth_row = (png_bytep )png_malloc(png_ptr, + (png_ptr->rowbytes + 1)); + png_ptr->paeth_row[0] = PNG_FILTER_VALUE_PAETH; + } + } + +#ifdef PNG_WRITE_INTERLACING_SUPPORTED + /* if interlaced, we need to set up width and height of pass */ + if (png_ptr->interlaced) + { + if (!(png_ptr->transformations & PNG_INTERLACE)) + { + png_ptr->num_rows = (png_ptr->height + png_pass_yinc[0] - 1 - + png_pass_ystart[0]) / png_pass_yinc[0]; + png_ptr->usr_width = (png_ptr->width + png_pass_inc[0] - 1 - + png_pass_start[0]) / png_pass_inc[0]; + } + else + { + png_ptr->num_rows = png_ptr->height; + png_ptr->usr_width = png_ptr->width; + } + } + else +#endif + { + png_ptr->num_rows = png_ptr->height; + png_ptr->usr_width = png_ptr->width; + } + png_ptr->zstream.avail_out = (uInt)png_ptr->zbuf_size; + png_ptr->zstream.next_out = png_ptr->zbuf; +} + +/* Internal use only. Called when finished processing a row of data. */ +void /* PRIVATE */ +png_write_finish_row(png_structp png_ptr) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + /* arrays to facilitate easy interlacing - use pass (0 - 6) as index */ + + /* start of interlace block */ + int png_pass_start[7] = {0, 4, 0, 2, 0, 1, 0}; + + /* offset to next interlace block */ + int png_pass_inc[7] = {8, 8, 4, 4, 2, 2, 1}; + + /* start of interlace block in the y direction */ + int png_pass_ystart[7] = {0, 0, 4, 0, 2, 0, 1}; + + /* offset to next interlace block in the y direction */ + int png_pass_yinc[7] = {8, 8, 8, 4, 4, 2, 2}; +#endif + + int ret; + + png_debug(1, "in png_write_finish_row\n"); + /* next row */ + png_ptr->row_number++; + + /* see if we are done */ + if (png_ptr->row_number < png_ptr->num_rows) + return; + +#ifdef PNG_WRITE_INTERLACING_SUPPORTED + /* if interlaced, go to next pass */ + if (png_ptr->interlaced) + { + png_ptr->row_number = 0; + if (png_ptr->transformations & PNG_INTERLACE) + { + png_ptr->pass++; + } + else + { + /* loop until we find a non-zero width or height pass */ + do + { + png_ptr->pass++; + if (png_ptr->pass >= 7) + break; + png_ptr->usr_width = (png_ptr->width + + png_pass_inc[png_ptr->pass] - 1 - + png_pass_start[png_ptr->pass]) / + png_pass_inc[png_ptr->pass]; + png_ptr->num_rows = (png_ptr->height + + png_pass_yinc[png_ptr->pass] - 1 - + png_pass_ystart[png_ptr->pass]) / + png_pass_yinc[png_ptr->pass]; + if (png_ptr->transformations & PNG_INTERLACE) + break; + } while (png_ptr->usr_width == 0 || png_ptr->num_rows == 0); + + } + + /* reset the row above the image for the next pass */ + if (png_ptr->pass < 7) + { + if (png_ptr->prev_row != NULL) + png_memset(png_ptr->prev_row, 0, + (png_size_t)(PNG_ROWBYTES(png_ptr->usr_channels* + png_ptr->usr_bit_depth,png_ptr->width))+1); + return; + } + } +#endif + + /* if we get here, we've just written the last row, so we need + to flush the compressor */ + do + { + /* tell the compressor we are done */ + ret = deflate(&png_ptr->zstream, Z_FINISH); + /* check for an error */ + if (ret == Z_OK) + { + /* check to see if we need more room */ + if (!(png_ptr->zstream.avail_out)) + { + png_write_IDAT(png_ptr, png_ptr->zbuf, png_ptr->zbuf_size); + png_ptr->zstream.next_out = png_ptr->zbuf; + png_ptr->zstream.avail_out = (uInt)png_ptr->zbuf_size; + } + } + else if (ret != Z_STREAM_END) + { + if (png_ptr->zstream.msg != NULL) + png_error(png_ptr, png_ptr->zstream.msg); + else + png_error(png_ptr, "zlib error"); + } + } while (ret != Z_STREAM_END); + + /* write any extra space */ + if (png_ptr->zstream.avail_out < png_ptr->zbuf_size) + { + png_write_IDAT(png_ptr, png_ptr->zbuf, png_ptr->zbuf_size - + png_ptr->zstream.avail_out); + } + + deflateReset(&png_ptr->zstream); + png_ptr->zstream.data_type = Z_BINARY; +} + +#if defined(PNG_WRITE_INTERLACING_SUPPORTED) +/* Pick out the correct pixels for the interlace pass. + * The basic idea here is to go through the row with a source + * pointer and a destination pointer (sp and dp), and copy the + * correct pixels for the pass. As the row gets compacted, + * sp will always be >= dp, so we should never overwrite anything. + * See the default: case for the easiest code to understand. + */ +void /* PRIVATE */ +png_do_write_interlace(png_row_infop row_info, png_bytep row, int pass) +{ +#ifdef PNG_USE_LOCAL_ARRAYS + /* arrays to facilitate easy interlacing - use pass (0 - 6) as index */ + + /* start of interlace block */ + int png_pass_start[7] = {0, 4, 0, 2, 0, 1, 0}; + + /* offset to next interlace block */ + int png_pass_inc[7] = {8, 8, 4, 4, 2, 2, 1}; +#endif + + png_debug(1, "in png_do_write_interlace\n"); + /* we don't have to do anything on the last pass (6) */ +#if defined(PNG_USELESS_TESTS_SUPPORTED) + if (row != NULL && row_info != NULL && pass < 6) +#else + if (pass < 6) +#endif + { + /* each pixel depth is handled separately */ + switch (row_info->pixel_depth) + { + case 1: + { + png_bytep sp; + png_bytep dp; + int shift; + int d; + int value; + png_uint_32 i; + png_uint_32 row_width = row_info->width; + + dp = row; + d = 0; + shift = 7; + for (i = png_pass_start[pass]; i < row_width; + i += png_pass_inc[pass]) + { + sp = row + (png_size_t)(i >> 3); + value = (int)(*sp >> (7 - (int)(i & 0x07))) & 0x01; + d |= (value << shift); + + if (shift == 0) + { + shift = 7; + *dp++ = (png_byte)d; + d = 0; + } + else + shift--; + + } + if (shift != 7) + *dp = (png_byte)d; + break; + } + case 2: + { + png_bytep sp; + png_bytep dp; + int shift; + int d; + int value; + png_uint_32 i; + png_uint_32 row_width = row_info->width; + + dp = row; + shift = 6; + d = 0; + for (i = png_pass_start[pass]; i < row_width; + i += png_pass_inc[pass]) + { + sp = row + (png_size_t)(i >> 2); + value = (*sp >> ((3 - (int)(i & 0x03)) << 1)) & 0x03; + d |= (value << shift); + + if (shift == 0) + { + shift = 6; + *dp++ = (png_byte)d; + d = 0; + } + else + shift -= 2; + } + if (shift != 6) + *dp = (png_byte)d; + break; + } + case 4: + { + png_bytep sp; + png_bytep dp; + int shift; + int d; + int value; + png_uint_32 i; + png_uint_32 row_width = row_info->width; + + dp = row; + shift = 4; + d = 0; + for (i = png_pass_start[pass]; i < row_width; + i += png_pass_inc[pass]) + { + sp = row + (png_size_t)(i >> 1); + value = (*sp >> ((1 - (int)(i & 0x01)) << 2)) & 0x0f; + d |= (value << shift); + + if (shift == 0) + { + shift = 4; + *dp++ = (png_byte)d; + d = 0; + } + else + shift -= 4; + } + if (shift != 4) + *dp = (png_byte)d; + break; + } + default: + { + png_bytep sp; + png_bytep dp; + png_uint_32 i; + png_uint_32 row_width = row_info->width; + png_size_t pixel_bytes; + + /* start at the beginning */ + dp = row; + /* find out how many bytes each pixel takes up */ + pixel_bytes = (row_info->pixel_depth >> 3); + /* loop through the row, only looking at the pixels that + matter */ + for (i = png_pass_start[pass]; i < row_width; + i += png_pass_inc[pass]) + { + /* find out where the original pixel is */ + sp = row + (png_size_t)i * pixel_bytes; + /* move the pixel */ + if (dp != sp) + png_memcpy(dp, sp, pixel_bytes); + /* next pixel */ + dp += pixel_bytes; + } + break; + } + } + /* set new row width */ + row_info->width = (row_info->width + + png_pass_inc[pass] - 1 - + png_pass_start[pass]) / + png_pass_inc[pass]; + row_info->rowbytes = PNG_ROWBYTES(row_info->pixel_depth, + row_info->width); + } +} +#endif + +/* This filters the row, chooses which filter to use, if it has not already + * been specified by the application, and then writes the row out with the + * chosen filter. + */ +#define PNG_MAXSUM (((png_uint_32)(-1)) >> 1) +#define PNG_HISHIFT 10 +#define PNG_LOMASK ((png_uint_32)0xffffL) +#define PNG_HIMASK ((png_uint_32)(~PNG_LOMASK >> PNG_HISHIFT)) +void /* PRIVATE */ +png_write_find_filter(png_structp png_ptr, png_row_infop row_info) +{ + png_bytep prev_row, best_row, row_buf; + png_uint_32 mins, bpp; + png_byte filter_to_do = png_ptr->do_filter; + png_uint_32 row_bytes = row_info->rowbytes; +#if defined(PNG_WRITE_WEIGHTED_FILTER_SUPPORTED) + int num_p_filters = (int)png_ptr->num_prev_filters; +#endif + + png_debug(1, "in png_write_find_filter\n"); + /* find out how many bytes offset each pixel is */ + bpp = (row_info->pixel_depth + 7) >> 3; + + prev_row = png_ptr->prev_row; + best_row = row_buf = png_ptr->row_buf; + mins = PNG_MAXSUM; + + /* The prediction method we use is to find which method provides the + * smallest value when summing the absolute values of the distances + * from zero, using anything >= 128 as negative numbers. This is known + * as the "minimum sum of absolute differences" heuristic. Other + * heuristics are the "weighted minimum sum of absolute differences" + * (experimental and can in theory improve compression), and the "zlib + * predictive" method (not implemented yet), which does test compressions + * of lines using different filter methods, and then chooses the + * (series of) filter(s) that give minimum compressed data size (VERY + * computationally expensive). + * + * GRR 980525: consider also + * (1) minimum sum of absolute differences from running average (i.e., + * keep running sum of non-absolute differences & count of bytes) + * [track dispersion, too? restart average if dispersion too large?] + * (1b) minimum sum of absolute differences from sliding average, probably + * with window size <= deflate window (usually 32K) + * (2) minimum sum of squared differences from zero or running average + * (i.e., ~ root-mean-square approach) + */ + + + /* We don't need to test the 'no filter' case if this is the only filter + * that has been chosen, as it doesn't actually do anything to the data. + */ + if ((filter_to_do & PNG_FILTER_NONE) && + filter_to_do != PNG_FILTER_NONE) + { + png_bytep rp; + png_uint_32 sum = 0; + png_uint_32 i; + int v; + + for (i = 0, rp = row_buf + 1; i < row_bytes; i++, rp++) + { + v = *rp; + sum += (v < 128) ? v : 256 - v; + } + +#if defined(PNG_WRITE_WEIGHTED_FILTER_SUPPORTED) + if (png_ptr->heuristic_method == PNG_FILTER_HEURISTIC_WEIGHTED) + { + png_uint_32 sumhi, sumlo; + int j; + sumlo = sum & PNG_LOMASK; + sumhi = (sum >> PNG_HISHIFT) & PNG_HIMASK; /* Gives us some footroom */ + + /* Reduce the sum if we match any of the previous rows */ + for (j = 0; j < num_p_filters; j++) + { + if (png_ptr->prev_filters[j] == PNG_FILTER_VALUE_NONE) + { + sumlo = (sumlo * png_ptr->filter_weights[j]) >> + PNG_WEIGHT_SHIFT; + sumhi = (sumhi * png_ptr->filter_weights[j]) >> + PNG_WEIGHT_SHIFT; + } + } + + /* Factor in the cost of this filter (this is here for completeness, + * but it makes no sense to have a "cost" for the NONE filter, as + * it has the minimum possible computational cost - none). + */ + sumlo = (sumlo * png_ptr->filter_costs[PNG_FILTER_VALUE_NONE]) >> + PNG_COST_SHIFT; + sumhi = (sumhi * png_ptr->filter_costs[PNG_FILTER_VALUE_NONE]) >> + PNG_COST_SHIFT; + + if (sumhi > PNG_HIMASK) + sum = PNG_MAXSUM; + else + sum = (sumhi << PNG_HISHIFT) + sumlo; + } +#endif + mins = sum; + } + + /* sub filter */ + if (filter_to_do == PNG_FILTER_SUB) + /* it's the only filter so no testing is needed */ + { + png_bytep rp, lp, dp; + png_uint_32 i; + for (i = 0, rp = row_buf + 1, dp = png_ptr->sub_row + 1; i < bpp; + i++, rp++, dp++) + { + *dp = *rp; + } + for (lp = row_buf + 1; i < row_bytes; + i++, rp++, lp++, dp++) + { + *dp = (png_byte)(((int)*rp - (int)*lp) & 0xff); + } + best_row = png_ptr->sub_row; + } + + else if (filter_to_do & PNG_FILTER_SUB) + { + png_bytep rp, dp, lp; + png_uint_32 sum = 0, lmins = mins; + png_uint_32 i; + int v; + +#if defined(PNG_WRITE_WEIGHTED_FILTER_SUPPORTED) + /* We temporarily increase the "minimum sum" by the factor we + * would reduce the sum of this filter, so that we can do the + * early exit comparison without scaling the sum each time. + */ + if (png_ptr->heuristic_method == PNG_FILTER_HEURISTIC_WEIGHTED) + { + int j; + png_uint_32 lmhi, lmlo; + lmlo = lmins & PNG_LOMASK; + lmhi = (lmins >> PNG_HISHIFT) & PNG_HIMASK; + + for (j = 0; j < num_p_filters; j++) + { + if (png_ptr->prev_filters[j] == PNG_FILTER_VALUE_SUB) + { + lmlo = (lmlo * png_ptr->inv_filter_weights[j]) >> + PNG_WEIGHT_SHIFT; + lmhi = (lmhi * png_ptr->inv_filter_weights[j]) >> + PNG_WEIGHT_SHIFT; + } + } + + lmlo = (lmlo * png_ptr->inv_filter_costs[PNG_FILTER_VALUE_SUB]) >> + PNG_COST_SHIFT; + lmhi = (lmhi * png_ptr->inv_filter_costs[PNG_FILTER_VALUE_SUB]) >> + PNG_COST_SHIFT; + + if (lmhi > PNG_HIMASK) + lmins = PNG_MAXSUM; + else + lmins = (lmhi << PNG_HISHIFT) + lmlo; + } +#endif + + for (i = 0, rp = row_buf + 1, dp = png_ptr->sub_row + 1; i < bpp; + i++, rp++, dp++) + { + v = *dp = *rp; + + sum += (v < 128) ? v : 256 - v; + } + for (lp = row_buf + 1; i < row_bytes; + i++, rp++, lp++, dp++) + { + v = *dp = (png_byte)(((int)*rp - (int)*lp) & 0xff); + + sum += (v < 128) ? v : 256 - v; + + if (sum > lmins) /* We are already worse, don't continue. */ + break; + } + +#if defined(PNG_WRITE_WEIGHTED_FILTER_SUPPORTED) + if (png_ptr->heuristic_method == PNG_FILTER_HEURISTIC_WEIGHTED) + { + int j; + png_uint_32 sumhi, sumlo; + sumlo = sum & PNG_LOMASK; + sumhi = (sum >> PNG_HISHIFT) & PNG_HIMASK; + + for (j = 0; j < num_p_filters; j++) + { + if (png_ptr->prev_filters[j] == PNG_FILTER_VALUE_SUB) + { + sumlo = (sumlo * png_ptr->inv_filter_weights[j]) >> + PNG_WEIGHT_SHIFT; + sumhi = (sumhi * png_ptr->inv_filter_weights[j]) >> + PNG_WEIGHT_SHIFT; + } + } + + sumlo = (sumlo * png_ptr->inv_filter_costs[PNG_FILTER_VALUE_SUB]) >> + PNG_COST_SHIFT; + sumhi = (sumhi * png_ptr->inv_filter_costs[PNG_FILTER_VALUE_SUB]) >> + PNG_COST_SHIFT; + + if (sumhi > PNG_HIMASK) + sum = PNG_MAXSUM; + else + sum = (sumhi << PNG_HISHIFT) + sumlo; + } +#endif + + if (sum < mins) + { + mins = sum; + best_row = png_ptr->sub_row; + } + } + + /* up filter */ + if (filter_to_do == PNG_FILTER_UP) + { + png_bytep rp, dp, pp; + png_uint_32 i; + + for (i = 0, rp = row_buf + 1, dp = png_ptr->up_row + 1, + pp = prev_row + 1; i < row_bytes; + i++, rp++, pp++, dp++) + { + *dp = (png_byte)(((int)*rp - (int)*pp) & 0xff); + } + best_row = png_ptr->up_row; + } + + else if (filter_to_do & PNG_FILTER_UP) + { + png_bytep rp, dp, pp; + png_uint_32 sum = 0, lmins = mins; + png_uint_32 i; + int v; + + +#if defined(PNG_WRITE_WEIGHTED_FILTER_SUPPORTED) + if (png_ptr->heuristic_method == PNG_FILTER_HEURISTIC_WEIGHTED) + { + int j; + png_uint_32 lmhi, lmlo; + lmlo = lmins & PNG_LOMASK; + lmhi = (lmins >> PNG_HISHIFT) & PNG_HIMASK; + + for (j = 0; j < num_p_filters; j++) + { + if (png_ptr->prev_filters[j] == PNG_FILTER_VALUE_UP) + { + lmlo = (lmlo * png_ptr->inv_filter_weights[j]) >> + PNG_WEIGHT_SHIFT; + lmhi = (lmhi * png_ptr->inv_filter_weights[j]) >> + PNG_WEIGHT_SHIFT; + } + } + + lmlo = (lmlo * png_ptr->inv_filter_costs[PNG_FILTER_VALUE_UP]) >> + PNG_COST_SHIFT; + lmhi = (lmhi * png_ptr->inv_filter_costs[PNG_FILTER_VALUE_UP]) >> + PNG_COST_SHIFT; + + if (lmhi > PNG_HIMASK) + lmins = PNG_MAXSUM; + else + lmins = (lmhi << PNG_HISHIFT) + lmlo; + } +#endif + + for (i = 0, rp = row_buf + 1, dp = png_ptr->up_row + 1, + pp = prev_row + 1; i < row_bytes; i++) + { + v = *dp++ = (png_byte)(((int)*rp++ - (int)*pp++) & 0xff); + + sum += (v < 128) ? v : 256 - v; + + if (sum > lmins) /* We are already worse, don't continue. */ + break; + } + +#if defined(PNG_WRITE_WEIGHTED_FILTER_SUPPORTED) + if (png_ptr->heuristic_method == PNG_FILTER_HEURISTIC_WEIGHTED) + { + int j; + png_uint_32 sumhi, sumlo; + sumlo = sum & PNG_LOMASK; + sumhi = (sum >> PNG_HISHIFT) & PNG_HIMASK; + + for (j = 0; j < num_p_filters; j++) + { + if (png_ptr->prev_filters[j] == PNG_FILTER_VALUE_UP) + { + sumlo = (sumlo * png_ptr->filter_weights[j]) >> + PNG_WEIGHT_SHIFT; + sumhi = (sumhi * png_ptr->filter_weights[j]) >> + PNG_WEIGHT_SHIFT; + } + } + + sumlo = (sumlo * png_ptr->filter_costs[PNG_FILTER_VALUE_UP]) >> + PNG_COST_SHIFT; + sumhi = (sumhi * png_ptr->filter_costs[PNG_FILTER_VALUE_UP]) >> + PNG_COST_SHIFT; + + if (sumhi > PNG_HIMASK) + sum = PNG_MAXSUM; + else + sum = (sumhi << PNG_HISHIFT) + sumlo; + } +#endif + + if (sum < mins) + { + mins = sum; + best_row = png_ptr->up_row; + } + } + + /* avg filter */ + if (filter_to_do == PNG_FILTER_AVG) + { + png_bytep rp, dp, pp, lp; + png_uint_32 i; + for (i = 0, rp = row_buf + 1, dp = png_ptr->avg_row + 1, + pp = prev_row + 1; i < bpp; i++) + { + *dp++ = (png_byte)(((int)*rp++ - ((int)*pp++ / 2)) & 0xff); + } + for (lp = row_buf + 1; i < row_bytes; i++) + { + *dp++ = (png_byte)(((int)*rp++ - (((int)*pp++ + (int)*lp++) / 2)) + & 0xff); + } + best_row = png_ptr->avg_row; + } + + else if (filter_to_do & PNG_FILTER_AVG) + { + png_bytep rp, dp, pp, lp; + png_uint_32 sum = 0, lmins = mins; + png_uint_32 i; + int v; + +#if defined(PNG_WRITE_WEIGHTED_FILTER_SUPPORTED) + if (png_ptr->heuristic_method == PNG_FILTER_HEURISTIC_WEIGHTED) + { + int j; + png_uint_32 lmhi, lmlo; + lmlo = lmins & PNG_LOMASK; + lmhi = (lmins >> PNG_HISHIFT) & PNG_HIMASK; + + for (j = 0; j < num_p_filters; j++) + { + if (png_ptr->prev_filters[j] == PNG_FILTER_VALUE_AVG) + { + lmlo = (lmlo * png_ptr->inv_filter_weights[j]) >> + PNG_WEIGHT_SHIFT; + lmhi = (lmhi * png_ptr->inv_filter_weights[j]) >> + PNG_WEIGHT_SHIFT; + } + } + + lmlo = (lmlo * png_ptr->inv_filter_costs[PNG_FILTER_VALUE_AVG]) >> + PNG_COST_SHIFT; + lmhi = (lmhi * png_ptr->inv_filter_costs[PNG_FILTER_VALUE_AVG]) >> + PNG_COST_SHIFT; + + if (lmhi > PNG_HIMASK) + lmins = PNG_MAXSUM; + else + lmins = (lmhi << PNG_HISHIFT) + lmlo; + } +#endif + + for (i = 0, rp = row_buf + 1, dp = png_ptr->avg_row + 1, + pp = prev_row + 1; i < bpp; i++) + { + v = *dp++ = (png_byte)(((int)*rp++ - ((int)*pp++ / 2)) & 0xff); + + sum += (v < 128) ? v : 256 - v; + } + for (lp = row_buf + 1; i < row_bytes; i++) + { + v = *dp++ = + (png_byte)(((int)*rp++ - (((int)*pp++ + (int)*lp++) / 2)) & 0xff); + + sum += (v < 128) ? v : 256 - v; + + if (sum > lmins) /* We are already worse, don't continue. */ + break; + } + +#if defined(PNG_WRITE_WEIGHTED_FILTER_SUPPORTED) + if (png_ptr->heuristic_method == PNG_FILTER_HEURISTIC_WEIGHTED) + { + int j; + png_uint_32 sumhi, sumlo; + sumlo = sum & PNG_LOMASK; + sumhi = (sum >> PNG_HISHIFT) & PNG_HIMASK; + + for (j = 0; j < num_p_filters; j++) + { + if (png_ptr->prev_filters[j] == PNG_FILTER_VALUE_NONE) + { + sumlo = (sumlo * png_ptr->filter_weights[j]) >> + PNG_WEIGHT_SHIFT; + sumhi = (sumhi * png_ptr->filter_weights[j]) >> + PNG_WEIGHT_SHIFT; + } + } + + sumlo = (sumlo * png_ptr->filter_costs[PNG_FILTER_VALUE_AVG]) >> + PNG_COST_SHIFT; + sumhi = (sumhi * png_ptr->filter_costs[PNG_FILTER_VALUE_AVG]) >> + PNG_COST_SHIFT; + + if (sumhi > PNG_HIMASK) + sum = PNG_MAXSUM; + else + sum = (sumhi << PNG_HISHIFT) + sumlo; + } +#endif + + if (sum < mins) + { + mins = sum; + best_row = png_ptr->avg_row; + } + } + + /* Paeth filter */ + if (filter_to_do == PNG_FILTER_PAETH) + { + png_bytep rp, dp, pp, cp, lp; + png_uint_32 i; + for (i = 0, rp = row_buf + 1, dp = png_ptr->paeth_row + 1, + pp = prev_row + 1; i < bpp; i++) + { + *dp++ = (png_byte)(((int)*rp++ - (int)*pp++) & 0xff); + } + + for (lp = row_buf + 1, cp = prev_row + 1; i < row_bytes; i++) + { + int a, b, c, pa, pb, pc, p; + + b = *pp++; + c = *cp++; + a = *lp++; + + p = b - c; + pc = a - c; + +#ifdef PNG_USE_ABS + pa = abs(p); + pb = abs(pc); + pc = abs(p + pc); +#else + pa = p < 0 ? -p : p; + pb = pc < 0 ? -pc : pc; + pc = (p + pc) < 0 ? -(p + pc) : p + pc; +#endif + + p = (pa <= pb && pa <=pc) ? a : (pb <= pc) ? b : c; + + *dp++ = (png_byte)(((int)*rp++ - p) & 0xff); + } + best_row = png_ptr->paeth_row; + } + + else if (filter_to_do & PNG_FILTER_PAETH) + { + png_bytep rp, dp, pp, cp, lp; + png_uint_32 sum = 0, lmins = mins; + png_uint_32 i; + int v; + +#if defined(PNG_WRITE_WEIGHTED_FILTER_SUPPORTED) + if (png_ptr->heuristic_method == PNG_FILTER_HEURISTIC_WEIGHTED) + { + int j; + png_uint_32 lmhi, lmlo; + lmlo = lmins & PNG_LOMASK; + lmhi = (lmins >> PNG_HISHIFT) & PNG_HIMASK; + + for (j = 0; j < num_p_filters; j++) + { + if (png_ptr->prev_filters[j] == PNG_FILTER_VALUE_PAETH) + { + lmlo = (lmlo * png_ptr->inv_filter_weights[j]) >> + PNG_WEIGHT_SHIFT; + lmhi = (lmhi * png_ptr->inv_filter_weights[j]) >> + PNG_WEIGHT_SHIFT; + } + } + + lmlo = (lmlo * png_ptr->inv_filter_costs[PNG_FILTER_VALUE_PAETH]) >> + PNG_COST_SHIFT; + lmhi = (lmhi * png_ptr->inv_filter_costs[PNG_FILTER_VALUE_PAETH]) >> + PNG_COST_SHIFT; + + if (lmhi > PNG_HIMASK) + lmins = PNG_MAXSUM; + else + lmins = (lmhi << PNG_HISHIFT) + lmlo; + } +#endif + + for (i = 0, rp = row_buf + 1, dp = png_ptr->paeth_row + 1, + pp = prev_row + 1; i < bpp; i++) + { + v = *dp++ = (png_byte)(((int)*rp++ - (int)*pp++) & 0xff); + + sum += (v < 128) ? v : 256 - v; + } + + for (lp = row_buf + 1, cp = prev_row + 1; i < row_bytes; i++) + { + int a, b, c, pa, pb, pc, p; + + b = *pp++; + c = *cp++; + a = *lp++; + +#ifndef PNG_SLOW_PAETH + p = b - c; + pc = a - c; +#ifdef PNG_USE_ABS + pa = abs(p); + pb = abs(pc); + pc = abs(p + pc); +#else + pa = p < 0 ? -p : p; + pb = pc < 0 ? -pc : pc; + pc = (p + pc) < 0 ? -(p + pc) : p + pc; +#endif + p = (pa <= pb && pa <=pc) ? a : (pb <= pc) ? b : c; +#else /* PNG_SLOW_PAETH */ + p = a + b - c; + pa = abs(p - a); + pb = abs(p - b); + pc = abs(p - c); + if (pa <= pb && pa <= pc) + p = a; + else if (pb <= pc) + p = b; + else + p = c; +#endif /* PNG_SLOW_PAETH */ + + v = *dp++ = (png_byte)(((int)*rp++ - p) & 0xff); + + sum += (v < 128) ? v : 256 - v; + + if (sum > lmins) /* We are already worse, don't continue. */ + break; + } + +#if defined(PNG_WRITE_WEIGHTED_FILTER_SUPPORTED) + if (png_ptr->heuristic_method == PNG_FILTER_HEURISTIC_WEIGHTED) + { + int j; + png_uint_32 sumhi, sumlo; + sumlo = sum & PNG_LOMASK; + sumhi = (sum >> PNG_HISHIFT) & PNG_HIMASK; + + for (j = 0; j < num_p_filters; j++) + { + if (png_ptr->prev_filters[j] == PNG_FILTER_VALUE_PAETH) + { + sumlo = (sumlo * png_ptr->filter_weights[j]) >> + PNG_WEIGHT_SHIFT; + sumhi = (sumhi * png_ptr->filter_weights[j]) >> + PNG_WEIGHT_SHIFT; + } + } + + sumlo = (sumlo * png_ptr->filter_costs[PNG_FILTER_VALUE_PAETH]) >> + PNG_COST_SHIFT; + sumhi = (sumhi * png_ptr->filter_costs[PNG_FILTER_VALUE_PAETH]) >> + PNG_COST_SHIFT; + + if (sumhi > PNG_HIMASK) + sum = PNG_MAXSUM; + else + sum = (sumhi << PNG_HISHIFT) + sumlo; + } +#endif + + if (sum < mins) + { + best_row = png_ptr->paeth_row; + } + } + + /* Do the actual writing of the filtered row data from the chosen filter. */ + + png_write_filtered_row(png_ptr, best_row); + +#if defined(PNG_WRITE_WEIGHTED_FILTER_SUPPORTED) + /* Save the type of filter we picked this time for future calculations */ + if (png_ptr->num_prev_filters > 0) + { + int j; + for (j = 1; j < num_p_filters; j++) + { + png_ptr->prev_filters[j] = png_ptr->prev_filters[j - 1]; + } + png_ptr->prev_filters[j] = best_row[0]; + } +#endif +} + + +/* Do the actual writing of a previously filtered row. */ +void /* PRIVATE */ +png_write_filtered_row(png_structp png_ptr, png_bytep filtered_row) +{ + png_debug(1, "in png_write_filtered_row\n"); + png_debug1(2, "filter = %d\n", filtered_row[0]); + /* set up the zlib input buffer */ + + png_ptr->zstream.next_in = filtered_row; + png_ptr->zstream.avail_in = (uInt)png_ptr->row_info.rowbytes + 1; + /* repeat until we have compressed all the data */ + do + { + int ret; /* return of zlib */ + + /* compress the data */ + ret = deflate(&png_ptr->zstream, Z_NO_FLUSH); + /* check for compression errors */ + if (ret != Z_OK) + { + if (png_ptr->zstream.msg != NULL) + png_error(png_ptr, png_ptr->zstream.msg); + else + png_error(png_ptr, "zlib error"); + } + + /* see if it is time to write another IDAT */ + if (!(png_ptr->zstream.avail_out)) + { + /* write the IDAT and reset the zlib output buffer */ + png_write_IDAT(png_ptr, png_ptr->zbuf, png_ptr->zbuf_size); + png_ptr->zstream.next_out = png_ptr->zbuf; + png_ptr->zstream.avail_out = (uInt)png_ptr->zbuf_size; + } + /* repeat until all data has been compressed */ + } while (png_ptr->zstream.avail_in); + + /* swap the current and previous rows */ + if (png_ptr->prev_row != NULL) + { + png_bytep tptr; + + tptr = png_ptr->prev_row; + png_ptr->prev_row = png_ptr->row_buf; + png_ptr->row_buf = tptr; + } + + /* finish row - updates counters and flushes zlib if last row */ + png_write_finish_row(png_ptr); + +#if defined(PNG_WRITE_FLUSH_SUPPORTED) + png_ptr->flush_rows++; + + if (png_ptr->flush_dist > 0 && + png_ptr->flush_rows >= png_ptr->flush_dist) + { + png_write_flush(png_ptr); + } +#endif +} +#endif /* PNG_WRITE_SUPPORTED */ diff --git a/src/win32/dependencies/sdl/CVS/Entries b/src/win32/dependencies/sdl/CVS/Entries new file mode 100644 index 00000000..096c0f03 --- /dev/null +++ b/src/win32/dependencies/sdl/CVS/Entries @@ -0,0 +1,44 @@ +/SDL.h/1.3/Sat Jun 24 19:59:04 2006// +/SDL_active.h/1.3/Sat Jun 24 19:59:04 2006// +/SDL_audio.h/1.3/Sat Jun 24 19:59:04 2006// +/SDL_byteorder.h/1.3/Sat Jun 24 19:59:04 2006// +/SDL_cdrom.h/1.3/Sat Jun 24 19:59:04 2006// +/SDL_config.h/1.1/Sat Jun 24 19:59:04 2006// +/SDL_config.h.default/1.1/Sat Jun 24 19:59:04 2006// +/SDL_config.h.in/1.1/Sat Jun 24 19:59:04 2006// +/SDL_config_amiga.h/1.1/Sat Jun 24 19:59:04 2006// +/SDL_config_dreamcast.h/1.1/Sat Jun 24 19:59:04 2006// +/SDL_config_macos.h/1.1/Sat Jun 24 19:59:04 2006// +/SDL_config_macosx.h/1.1/Sat Jun 24 19:59:04 2006// +/SDL_config_minimal.h/1.1/Sat Jun 24 19:59:04 2006// +/SDL_config_os2.h/1.1/Sat Jun 24 19:59:04 2006// +/SDL_config_win32.h/1.1/Sat Jun 24 19:59:04 2006// +/SDL_copying.h/1.3/Sat Jun 24 19:59:04 2006// +/SDL_cpuinfo.h/1.3/Sat Jun 24 19:59:04 2006// +/SDL_endian.h/1.3/Sat Jun 24 19:59:04 2006// +/SDL_error.h/1.3/Sat Jun 24 19:59:04 2006// +/SDL_events.h/1.3/Sat Jun 24 19:59:04 2006// +/SDL_getenv.h/1.3/Sat Jun 24 19:59:04 2006// +/SDL_joystick.h/1.3/Sat Jun 24 19:59:04 2006// +/SDL_keyboard.h/1.3/Sat Jun 24 19:59:04 2006// +/SDL_keysym.h/1.3/Sat Jun 24 19:59:04 2006// +/SDL_loadso.h/1.3/Sat Jun 24 19:59:04 2006// +/SDL_main.h/1.3/Sat Jun 24 19:59:04 2006// +/SDL_mouse.h/1.3/Sat Jun 24 19:59:04 2006// +/SDL_mutex.h/1.3/Sat Jun 24 19:59:05 2006// +/SDL_name.h/1.3/Sat Jun 24 19:59:05 2006// +/SDL_opengl.h/1.3/Sat Jun 24 19:59:05 2006// +/SDL_platform.h/1.1/Sat Jun 24 19:59:05 2006// +/SDL_quit.h/1.3/Sat Jun 24 19:59:05 2006// +/SDL_rwops.h/1.3/Sat Jun 24 19:59:05 2006// +/SDL_stdinc.h/1.1/Sat Jun 24 19:59:05 2006// +/SDL_syswm.h/1.3/Sat Jun 24 19:59:05 2006// +/SDL_thread.h/1.3/Sat Jun 24 19:59:05 2006// +/SDL_timer.h/1.3/Sat Jun 24 19:59:05 2006// +/SDL_types.h/1.3/Sat Jun 24 19:59:05 2006// +/SDL_version.h/1.3/Sat Jun 24 19:59:05 2006// +/SDL_video.h/1.3/Sat Jun 24 19:59:05 2006// +/begin_code.h/1.3/Sat Jun 24 19:59:05 2006// +/close_code.h/1.3/Sat Jun 24 19:59:05 2006// +D/SDL_Debug//// +D/SDL_Release//// diff --git a/src/win32/dependencies/sdl/CVS/Repository b/src/win32/dependencies/sdl/CVS/Repository new file mode 100644 index 00000000..5a02aea4 --- /dev/null +++ b/src/win32/dependencies/sdl/CVS/Repository @@ -0,0 +1 @@ +VisualBoyAdvance/win32/dependencies/sdl diff --git a/src/win32/dependencies/sdl/CVS/Root b/src/win32/dependencies/sdl/CVS/Root new file mode 100644 index 00000000..6ceab0dd --- /dev/null +++ b/src/win32/dependencies/sdl/CVS/Root @@ -0,0 +1 @@ +:pserver:anonymous@vba.cvs.sourceforge.net:/cvsroot/vba diff --git a/src/win32/dependencies/sdl/SDL.h b/src/win32/dependencies/sdl/SDL.h new file mode 100644 index 00000000..60ac26ce --- /dev/null +++ b/src/win32/dependencies/sdl/SDL.h @@ -0,0 +1,94 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* Main include header for the SDL library */ + +#ifndef _SDL_H +#define _SDL_H + +#include "SDL_main.h" +#include "SDL_stdinc.h" +#include "SDL_audio.h" +#include "SDL_cdrom.h" +#include "SDL_cpuinfo.h" +#include "SDL_endian.h" +#include "SDL_error.h" +#include "SDL_events.h" +#include "SDL_loadso.h" +#include "SDL_mutex.h" +#include "SDL_rwops.h" +#include "SDL_thread.h" +#include "SDL_timer.h" +#include "SDL_video.h" +#include "SDL_version.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* As of version 0.5, SDL is loaded dynamically into the application */ + +/* These are the flags which may be passed to SDL_Init() -- you should + specify the subsystems which you will be using in your application. +*/ +#define SDL_INIT_TIMER 0x00000001 +#define SDL_INIT_AUDIO 0x00000010 +#define SDL_INIT_VIDEO 0x00000020 +#define SDL_INIT_CDROM 0x00000100 +#define SDL_INIT_JOYSTICK 0x00000200 +#define SDL_INIT_NOPARACHUTE 0x00100000 /* Don't catch fatal signals */ +#define SDL_INIT_EVENTTHREAD 0x01000000 /* Not supported on all OS's */ +#define SDL_INIT_EVERYTHING 0x0000FFFF + +/* This function loads the SDL dynamically linked library and initializes + * the subsystems specified by 'flags' (and those satisfying dependencies) + * Unless the SDL_INIT_NOPARACHUTE flag is set, it will install cleanup + * signal handlers for some commonly ignored fatal signals (like SIGSEGV) + */ +extern DECLSPEC int SDLCALL SDL_Init(Uint32 flags); + +/* This function initializes specific SDL subsystems */ +extern DECLSPEC int SDLCALL SDL_InitSubSystem(Uint32 flags); + +/* This function cleans up specific SDL subsystems */ +extern DECLSPEC void SDLCALL SDL_QuitSubSystem(Uint32 flags); + +/* This function returns mask of the specified subsystems which have + been initialized. + If 'flags' is 0, it returns a mask of all initialized subsystems. +*/ +extern DECLSPEC Uint32 SDLCALL SDL_WasInit(Uint32 flags); + +/* This function cleans up all initialized subsystems and unloads the + * dynamically linked library. You should call it upon all exit conditions. + */ +extern DECLSPEC void SDLCALL SDL_Quit(void); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* _SDL_H */ diff --git a/src/win32/dependencies/sdl/SDL_Debug/CVS/Entries b/src/win32/dependencies/sdl/SDL_Debug/CVS/Entries new file mode 100644 index 00000000..4bf96b0e --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_Debug/CVS/Entries @@ -0,0 +1,3 @@ +/SDL.lib/1.1/Fri Aug 25 13:07:25 2006/-kb/ +/SDLmain.lib/1.1/Fri Aug 25 13:07:25 2006/-kb/ +D diff --git a/src/win32/dependencies/sdl/SDL_Debug/CVS/Repository b/src/win32/dependencies/sdl/SDL_Debug/CVS/Repository new file mode 100644 index 00000000..ff499320 --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_Debug/CVS/Repository @@ -0,0 +1 @@ +VisualBoyAdvance/win32/dependencies/sdl/SDL_Debug diff --git a/src/win32/dependencies/sdl/SDL_Debug/CVS/Root b/src/win32/dependencies/sdl/SDL_Debug/CVS/Root new file mode 100644 index 00000000..6ceab0dd --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_Debug/CVS/Root @@ -0,0 +1 @@ +:pserver:anonymous@vba.cvs.sourceforge.net:/cvsroot/vba diff --git a/src/win32/dependencies/sdl/SDL_Debug/SDL.lib b/src/win32/dependencies/sdl/SDL_Debug/SDL.lib new file mode 100644 index 0000000000000000000000000000000000000000..90060fb0940de0264f59ba6dca6a14af3a50c524 GIT binary patch literal 1787026 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�?viN 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
@y4(hpu;X>ytFmNHj00~<9PB5 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
dN@CHvY#d}+#cT&+~*b1=KNE91b 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�HUclxElQ?~4OzlM=S*DV_oNs7i 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 literal 0 HcmV?d00001 diff --git a/src/win32/dependencies/sdl/SDL_Debug/SDLmain.lib b/src/win32/dependencies/sdl/SDL_Debug/SDLmain.lib new file mode 100644 index 0000000000000000000000000000000000000000..d981135f79399c6bb9483ea125e275800961d04e GIT binary patch literal 19288 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 YycX{gUKj2-BMx!NAa`1~*Wcs+A8o_cZU6uP literal 0 HcmV?d00001 diff --git a/src/win32/dependencies/sdl/SDL_Release/CVS/Entries b/src/win32/dependencies/sdl/SDL_Release/CVS/Entries new file mode 100644 index 00000000..cc87a190 --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_Release/CVS/Entries @@ -0,0 +1,3 @@ +/SDL.lib/1.1/Fri Aug 25 13:07:26 2006/-kb/ +/SDLmain.lib/1.1/Fri Aug 25 13:07:26 2006/-kb/ +D diff --git a/src/win32/dependencies/sdl/SDL_Release/CVS/Repository b/src/win32/dependencies/sdl/SDL_Release/CVS/Repository new file mode 100644 index 00000000..5df783ac --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_Release/CVS/Repository @@ -0,0 +1 @@ +VisualBoyAdvance/win32/dependencies/sdl/SDL_Release diff --git a/src/win32/dependencies/sdl/SDL_Release/CVS/Root b/src/win32/dependencies/sdl/SDL_Release/CVS/Root new file mode 100644 index 00000000..6ceab0dd --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_Release/CVS/Root @@ -0,0 +1 @@ +:pserver:anonymous@vba.cvs.sourceforge.net:/cvsroot/vba diff --git a/src/win32/dependencies/sdl/SDL_Release/SDL.lib b/src/win32/dependencies/sdl/SDL_Release/SDL.lib new file mode 100644 index 0000000000000000000000000000000000000000..613a7a27ec4d214c15fae243f23fef481923ffc8 GIT binary patch literal 664824 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
(XjBIs(GeV3UTJW5YFRH_Ar=xzD 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�m?Cf%vQ|p`T?E{ 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

2tq^ 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�b+$4) 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ċu0W-T70bTn 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 Y8C}*Fs4Y6rgeF5)DQKJH_V-u+0zf2)t^fc4 literal 0 HcmV?d00001 diff --git a/src/win32/dependencies/sdl/SDL_active.h b/src/win32/dependencies/sdl/SDL_active.h new file mode 100644 index 00000000..2cf474c5 --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_active.h @@ -0,0 +1,58 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* Include file for SDL application focus event handling */ + +#ifndef _SDL_active_h +#define _SDL_active_h + +#include "SDL_stdinc.h" +#include "SDL_error.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* The available application states */ +#define SDL_APPMOUSEFOCUS 0x01 /* The app has mouse coverage */ +#define SDL_APPINPUTFOCUS 0x02 /* The app has input focus */ +#define SDL_APPACTIVE 0x04 /* The application is active */ + +/* Function prototypes */ +/* + * This function returns the current state of the application, which is a + * bitwise combination of SDL_APPMOUSEFOCUS, SDL_APPINPUTFOCUS, and + * SDL_APPACTIVE. If SDL_APPACTIVE is set, then the user is able to + * see your application, otherwise it has been iconified or disabled. + */ +extern DECLSPEC Uint8 SDLCALL SDL_GetAppState(void); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* _SDL_active_h */ diff --git a/src/win32/dependencies/sdl/SDL_audio.h b/src/win32/dependencies/sdl/SDL_audio.h new file mode 100644 index 00000000..68ec4759 --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_audio.h @@ -0,0 +1,253 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* Access to the raw audio mixing buffer for the SDL library */ + +#ifndef _SDL_audio_h +#define _SDL_audio_h + +#include "SDL_stdinc.h" +#include "SDL_error.h" +#include "SDL_endian.h" +#include "SDL_mutex.h" +#include "SDL_thread.h" +#include "SDL_rwops.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* The calculated values in this structure are calculated by SDL_OpenAudio() */ +typedef struct SDL_AudioSpec { + int freq; /* DSP frequency -- samples per second */ + Uint16 format; /* Audio data format */ + Uint8 channels; /* Number of channels: 1 mono, 2 stereo */ + Uint8 silence; /* Audio buffer silence value (calculated) */ + Uint16 samples; /* Audio buffer size in samples (power of 2) */ + Uint16 padding; /* Necessary for some compile environments */ + Uint32 size; /* Audio buffer size in bytes (calculated) */ + /* This function is called when the audio device needs more data. + 'stream' is a pointer to the audio data buffer + 'len' is the length of that buffer in bytes. + Once the callback returns, the buffer will no longer be valid. + Stereo samples are stored in a LRLRLR ordering. + */ + void (SDLCALL *callback)(void *userdata, Uint8 *stream, int len); + void *userdata; +} SDL_AudioSpec; + +/* Audio format flags (defaults to LSB byte order) */ +#define AUDIO_U8 0x0008 /* Unsigned 8-bit samples */ +#define AUDIO_S8 0x8008 /* Signed 8-bit samples */ +#define AUDIO_U16LSB 0x0010 /* Unsigned 16-bit samples */ +#define AUDIO_S16LSB 0x8010 /* Signed 16-bit samples */ +#define AUDIO_U16MSB 0x1010 /* As above, but big-endian byte order */ +#define AUDIO_S16MSB 0x9010 /* As above, but big-endian byte order */ +#define AUDIO_U16 AUDIO_U16LSB +#define AUDIO_S16 AUDIO_S16LSB + +/* Native audio byte ordering */ +#if SDL_BYTEORDER == SDL_LIL_ENDIAN +#define AUDIO_U16SYS AUDIO_U16LSB +#define AUDIO_S16SYS AUDIO_S16LSB +#else +#define AUDIO_U16SYS AUDIO_U16MSB +#define AUDIO_S16SYS AUDIO_S16MSB +#endif + + +/* A structure to hold a set of audio conversion filters and buffers */ +typedef struct SDL_AudioCVT { + int needed; /* Set to 1 if conversion possible */ + Uint16 src_format; /* Source audio format */ + Uint16 dst_format; /* Target audio format */ + double rate_incr; /* Rate conversion increment */ + Uint8 *buf; /* Buffer to hold entire audio data */ + int len; /* Length of original audio buffer */ + int len_cvt; /* Length of converted audio buffer */ + int len_mult; /* buffer must be len*len_mult big */ + double len_ratio; /* Given len, final size is len*len_ratio */ + void (SDLCALL *filters[10])(struct SDL_AudioCVT *cvt, Uint16 format); + int filter_index; /* Current audio conversion function */ +} SDL_AudioCVT; + + +/* Function prototypes */ + +/* These functions are used internally, and should not be used unless you + * have a specific need to specify the audio driver you want to use. + * You should normally use SDL_Init() or SDL_InitSubSystem(). + */ +extern DECLSPEC int SDLCALL SDL_AudioInit(const char *driver_name); +extern DECLSPEC void SDLCALL SDL_AudioQuit(void); + +/* This function fills the given character buffer with the name of the + * current audio driver, and returns a pointer to it if the audio driver has + * been initialized. It returns NULL if no driver has been initialized. + */ +extern DECLSPEC char * SDLCALL SDL_AudioDriverName(char *namebuf, int maxlen); + +/* + * This function opens the audio device with the desired parameters, and + * returns 0 if successful, placing the actual hardware parameters in the + * structure pointed to by 'obtained'. If 'obtained' is NULL, the audio + * data passed to the callback function will be guaranteed to be in the + * requested format, and will be automatically converted to the hardware + * audio format if necessary. This function returns -1 if it failed + * to open the audio device, or couldn't set up the audio thread. + * + * When filling in the desired audio spec structure, + * 'desired->freq' should be the desired audio frequency in samples-per-second. + * 'desired->format' should be the desired audio format. + * 'desired->samples' is the desired size of the audio buffer, in samples. + * This number should be a power of two, and may be adjusted by the audio + * driver to a value more suitable for the hardware. Good values seem to + * range between 512 and 8096 inclusive, depending on the application and + * CPU speed. Smaller values yield faster response time, but can lead + * to underflow if the application is doing heavy processing and cannot + * fill the audio buffer in time. A stereo sample consists of both right + * and left channels in LR ordering. + * Note that the number of samples is directly related to time by the + * following formula: ms = (samples*1000)/freq + * 'desired->size' is the size in bytes of the audio buffer, and is + * calculated by SDL_OpenAudio(). + * 'desired->silence' is the value used to set the buffer to silence, + * and is calculated by SDL_OpenAudio(). + * 'desired->callback' should be set to a function that will be called + * when the audio device is ready for more data. It is passed a pointer + * to the audio buffer, and the length in bytes of the audio buffer. + * This function usually runs in a separate thread, and so you should + * protect data structures that it accesses by calling SDL_LockAudio() + * and SDL_UnlockAudio() in your code. + * 'desired->userdata' is passed as the first parameter to your callback + * function. + * + * The audio device starts out playing silence when it's opened, and should + * be enabled for playing by calling SDL_PauseAudio(0) when you are ready + * for your audio callback function to be called. Since the audio driver + * may modify the requested size of the audio buffer, you should allocate + * any local mixing buffers after you open the audio device. + */ +extern DECLSPEC int SDLCALL SDL_OpenAudio(SDL_AudioSpec *desired, SDL_AudioSpec *obtained); + +/* + * Get the current audio state: + */ +typedef enum { + SDL_AUDIO_STOPPED = 0, + SDL_AUDIO_PLAYING, + SDL_AUDIO_PAUSED +} SDL_audiostatus; +extern DECLSPEC SDL_audiostatus SDLCALL SDL_GetAudioStatus(void); + +/* + * This function pauses and unpauses the audio callback processing. + * It should be called with a parameter of 0 after opening the audio + * device to start playing sound. This is so you can safely initialize + * data for your callback function after opening the audio device. + * Silence will be written to the audio device during the pause. + */ +extern DECLSPEC void SDLCALL SDL_PauseAudio(int pause_on); + +/* + * This function loads a WAVE from the data source, automatically freeing + * that source if 'freesrc' is non-zero. For example, to load a WAVE file, + * you could do: + * SDL_LoadWAV_RW(SDL_RWFromFile("sample.wav", "rb"), 1, ...); + * + * If this function succeeds, it returns the given SDL_AudioSpec, + * filled with the audio data format of the wave data, and sets + * 'audio_buf' to a malloc()'d buffer containing the audio data, + * and sets 'audio_len' to the length of that audio buffer, in bytes. + * You need to free the audio buffer with SDL_FreeWAV() when you are + * done with it. + * + * This function returns NULL and sets the SDL error message if the + * wave file cannot be opened, uses an unknown data format, or is + * corrupt. Currently raw and MS-ADPCM WAVE files are supported. + */ +extern DECLSPEC SDL_AudioSpec * SDLCALL SDL_LoadWAV_RW(SDL_RWops *src, int freesrc, SDL_AudioSpec *spec, Uint8 **audio_buf, Uint32 *audio_len); + +/* Compatibility convenience function -- loads a WAV from a file */ +#define SDL_LoadWAV(file, spec, audio_buf, audio_len) \ + SDL_LoadWAV_RW(SDL_RWFromFile(file, "rb"),1, spec,audio_buf,audio_len) + +/* + * This function frees data previously allocated with SDL_LoadWAV_RW() + */ +extern DECLSPEC void SDLCALL SDL_FreeWAV(Uint8 *audio_buf); + +/* + * This function takes a source format and rate and a destination format + * and rate, and initializes the 'cvt' structure with information needed + * by SDL_ConvertAudio() to convert a buffer of audio data from one format + * to the other. + * This function returns 0, or -1 if there was an error. + */ +extern DECLSPEC int SDLCALL SDL_BuildAudioCVT(SDL_AudioCVT *cvt, + Uint16 src_format, Uint8 src_channels, int src_rate, + Uint16 dst_format, Uint8 dst_channels, int dst_rate); + +/* Once you have initialized the 'cvt' structure using SDL_BuildAudioCVT(), + * created an audio buffer cvt->buf, and filled it with cvt->len bytes of + * audio data in the source format, this function will convert it in-place + * to the desired format. + * The data conversion may expand the size of the audio data, so the buffer + * cvt->buf should be allocated after the cvt structure is initialized by + * SDL_BuildAudioCVT(), and should be cvt->len*cvt->len_mult bytes long. + */ +extern DECLSPEC int SDLCALL SDL_ConvertAudio(SDL_AudioCVT *cvt); + +/* + * This takes two audio buffers of the playing audio format and mixes + * them, performing addition, volume adjustment, and overflow clipping. + * The volume ranges from 0 - 128, and should be set to SDL_MIX_MAXVOLUME + * for full audio volume. Note this does not change hardware volume. + * This is provided for convenience -- you can mix your own audio data. + */ +#define SDL_MIX_MAXVOLUME 128 +extern DECLSPEC void SDLCALL SDL_MixAudio(Uint8 *dst, const Uint8 *src, Uint32 len, int volume); + +/* + * The lock manipulated by these functions protects the callback function. + * During a LockAudio/UnlockAudio pair, you can be guaranteed that the + * callback function is not running. Do not call these from the callback + * function or you will cause deadlock. + */ +extern DECLSPEC void SDLCALL SDL_LockAudio(void); +extern DECLSPEC void SDLCALL SDL_UnlockAudio(void); + +/* + * This function shuts down audio processing and closes the audio device. + */ +extern DECLSPEC void SDLCALL SDL_CloseAudio(void); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* _SDL_audio_h */ diff --git a/src/win32/dependencies/sdl/SDL_byteorder.h b/src/win32/dependencies/sdl/SDL_byteorder.h new file mode 100644 index 00000000..3871cfed --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_byteorder.h @@ -0,0 +1,24 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* DEPRECATED */ +#include "SDL_endian.h" diff --git a/src/win32/dependencies/sdl/SDL_cdrom.h b/src/win32/dependencies/sdl/SDL_cdrom.h new file mode 100644 index 00000000..5f8f0c62 --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_cdrom.h @@ -0,0 +1,171 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* This is the CD-audio control API for Simple DirectMedia Layer */ + +#ifndef _SDL_cdrom_h +#define _SDL_cdrom_h + +#include "SDL_stdinc.h" +#include "SDL_error.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* In order to use these functions, SDL_Init() must have been called + with the SDL_INIT_CDROM flag. This causes SDL to scan the system + for CD-ROM drives, and load appropriate drivers. +*/ + +/* The maximum number of CD-ROM tracks on a disk */ +#define SDL_MAX_TRACKS 99 + +/* The types of CD-ROM track possible */ +#define SDL_AUDIO_TRACK 0x00 +#define SDL_DATA_TRACK 0x04 + +/* The possible states which a CD-ROM drive can be in. */ +typedef enum { + CD_TRAYEMPTY, + CD_STOPPED, + CD_PLAYING, + CD_PAUSED, + CD_ERROR = -1 +} CDstatus; + +/* Given a status, returns true if there's a disk in the drive */ +#define CD_INDRIVE(status) ((int)(status) > 0) + +typedef struct SDL_CDtrack { + Uint8 id; /* Track number */ + Uint8 type; /* Data or audio track */ + Uint16 unused; + Uint32 length; /* Length, in frames, of this track */ + Uint32 offset; /* Offset, in frames, from start of disk */ +} SDL_CDtrack; + +/* This structure is only current as of the last call to SDL_CDStatus() */ +typedef struct SDL_CD { + int id; /* Private drive identifier */ + CDstatus status; /* Current drive status */ + + /* The rest of this structure is only valid if there's a CD in drive */ + int numtracks; /* Number of tracks on disk */ + int cur_track; /* Current track position */ + int cur_frame; /* Current frame offset within current track */ + SDL_CDtrack track[SDL_MAX_TRACKS+1]; +} SDL_CD; + +/* Conversion functions from frames to Minute/Second/Frames and vice versa */ +#define CD_FPS 75 +#define FRAMES_TO_MSF(f, M,S,F) { \ + int value = f; \ + *(F) = value%CD_FPS; \ + value /= CD_FPS; \ + *(S) = value%60; \ + value /= 60; \ + *(M) = value; \ +} +#define MSF_TO_FRAMES(M, S, F) ((M)*60*CD_FPS+(S)*CD_FPS+(F)) + +/* CD-audio API functions: */ + +/* Returns the number of CD-ROM drives on the system, or -1 if + SDL_Init() has not been called with the SDL_INIT_CDROM flag. + */ +extern DECLSPEC int SDLCALL SDL_CDNumDrives(void); + +/* Returns a human-readable, system-dependent identifier for the CD-ROM. + Example: + "/dev/cdrom" + "E:" + "/dev/disk/ide/1/master" +*/ +extern DECLSPEC const char * SDLCALL SDL_CDName(int drive); + +/* Opens a CD-ROM drive for access. It returns a drive handle on success, + or NULL if the drive was invalid or busy. This newly opened CD-ROM + becomes the default CD used when other CD functions are passed a NULL + CD-ROM handle. + Drives are numbered starting with 0. Drive 0 is the system default CD-ROM. +*/ +extern DECLSPEC SDL_CD * SDLCALL SDL_CDOpen(int drive); + +/* This function returns the current status of the given drive. + If the drive has a CD in it, the table of contents of the CD and current + play position of the CD will be stored in the SDL_CD structure. +*/ +extern DECLSPEC CDstatus SDLCALL SDL_CDStatus(SDL_CD *cdrom); + +/* Play the given CD starting at 'start_track' and 'start_frame' for 'ntracks' + tracks and 'nframes' frames. If both 'ntrack' and 'nframe' are 0, play + until the end of the CD. This function will skip data tracks. + This function should only be called after calling SDL_CDStatus() to + get track information about the CD. + For example: + // Play entire CD: + if ( CD_INDRIVE(SDL_CDStatus(cdrom)) ) + SDL_CDPlayTracks(cdrom, 0, 0, 0, 0); + // Play last track: + if ( CD_INDRIVE(SDL_CDStatus(cdrom)) ) { + SDL_CDPlayTracks(cdrom, cdrom->numtracks-1, 0, 0, 0); + } + // Play first and second track and 10 seconds of third track: + if ( CD_INDRIVE(SDL_CDStatus(cdrom)) ) + SDL_CDPlayTracks(cdrom, 0, 0, 2, 10); + + This function returns 0, or -1 if there was an error. +*/ +extern DECLSPEC int SDLCALL SDL_CDPlayTracks(SDL_CD *cdrom, + int start_track, int start_frame, int ntracks, int nframes); + +/* Play the given CD starting at 'start' frame for 'length' frames. + It returns 0, or -1 if there was an error. +*/ +extern DECLSPEC int SDLCALL SDL_CDPlay(SDL_CD *cdrom, int start, int length); + +/* Pause play -- returns 0, or -1 on error */ +extern DECLSPEC int SDLCALL SDL_CDPause(SDL_CD *cdrom); + +/* Resume play -- returns 0, or -1 on error */ +extern DECLSPEC int SDLCALL SDL_CDResume(SDL_CD *cdrom); + +/* Stop play -- returns 0, or -1 on error */ +extern DECLSPEC int SDLCALL SDL_CDStop(SDL_CD *cdrom); + +/* Eject CD-ROM -- returns 0, or -1 on error */ +extern DECLSPEC int SDLCALL SDL_CDEject(SDL_CD *cdrom); + +/* Closes the handle for the CD-ROM drive */ +extern DECLSPEC void SDLCALL SDL_CDClose(SDL_CD *cdrom); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* _SDL_video_h */ diff --git a/src/win32/dependencies/sdl/SDL_config.h b/src/win32/dependencies/sdl/SDL_config.h new file mode 100644 index 00000000..8970ec3d --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_config.h @@ -0,0 +1,45 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +#ifndef _SDL_config_h +#define _SDL_config_h + +#include "SDL_platform.h" + +/* Add any platform that doesn't build using the configure system */ +#if defined(__AMIGA__) +#include "SDL_config_amiga.h" +#elif defined(__DREAMCAST__) +#include "SDL_config_dreamcast.h" +#elif defined(__MACOS__) +#include "SDL_config_macos.h" +#elif defined(__MACOSX__) +#include "SDL_config_macosx.h" +#elif defined(__WIN32__) +#include "SDL_config_win32.h" +#elif defined(__OS2__) +#include "SDL_config_os2.h" +#else +#include "SDL_config_minimal.h" +#endif /* platform config */ + +#endif /* _SDL_config_h */ diff --git a/src/win32/dependencies/sdl/SDL_config.h.default b/src/win32/dependencies/sdl/SDL_config.h.default new file mode 100644 index 00000000..8970ec3d --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_config.h.default @@ -0,0 +1,45 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +#ifndef _SDL_config_h +#define _SDL_config_h + +#include "SDL_platform.h" + +/* Add any platform that doesn't build using the configure system */ +#if defined(__AMIGA__) +#include "SDL_config_amiga.h" +#elif defined(__DREAMCAST__) +#include "SDL_config_dreamcast.h" +#elif defined(__MACOS__) +#include "SDL_config_macos.h" +#elif defined(__MACOSX__) +#include "SDL_config_macosx.h" +#elif defined(__WIN32__) +#include "SDL_config_win32.h" +#elif defined(__OS2__) +#include "SDL_config_os2.h" +#else +#include "SDL_config_minimal.h" +#endif /* platform config */ + +#endif /* _SDL_config_h */ diff --git a/src/win32/dependencies/sdl/SDL_config.h.in b/src/win32/dependencies/sdl/SDL_config.h.in new file mode 100644 index 00000000..6f89569c --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_config.h.in @@ -0,0 +1,305 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +#ifndef _SDL_config_h +#define _SDL_config_h + +/* This is a set of defines to configure the SDL features */ + +/* General platform specific identifiers */ +#include "SDL_platform.h" + +/* C language features */ +#undef const +#undef inline +#undef volatile + +/* C datatypes */ +#undef size_t +#undef int8_t +#undef uint8_t +#undef int16_t +#undef uint16_t +#undef int32_t +#undef uint32_t +#undef int64_t +#undef uint64_t +#undef uintptr_t +#undef SDL_HAS_64BIT_TYPE + +/* Endianness */ +#undef SDL_BYTEORDER + +/* Comment this if you want to build without any C library requirements */ +#undef HAVE_LIBC +#if HAVE_LIBC + +/* Useful headers */ +#undef HAVE_ALLOCA_H +#undef HAVE_SYS_TYPES_H +#undef HAVE_STDIO_H +#undef STDC_HEADERS +#undef HAVE_STDLIB_H +#undef HAVE_STDARG_H +#undef HAVE_MALLOC_H +#undef HAVE_MEMORY_H +#undef HAVE_STRING_H +#undef HAVE_STRINGS_H +#undef HAVE_INTTYPES_H +#undef HAVE_STDINT_H +#undef HAVE_CTYPE_H +#undef HAVE_MATH_H +#undef HAVE_ICONV_H +#undef HAVE_SIGNAL_H +#undef HAVE_ALTIVEC_H + +/* C library functions */ +#undef HAVE_MALLOC +#undef HAVE_CALLOC +#undef HAVE_REALLOC +#undef HAVE_FREE +#undef HAVE_ALLOCA +#ifndef _WIN32 /* Don't use C runtime versions of these on Windows */ +#undef HAVE_GETENV +#undef HAVE_PUTENV +#undef HAVE_UNSETENV +#endif +#undef HAVE_QSORT +#undef HAVE_ABS +#undef HAVE_BCOPY +#undef HAVE_MEMSET +#undef HAVE_MEMCPY +#undef HAVE_MEMMOVE +#undef HAVE_MEMCMP +#undef HAVE_STRLEN +#undef HAVE_STRLCPY +#undef HAVE_STRLCAT +#undef HAVE_STRDUP +#undef HAVE__STRREV +#undef HAVE__STRUPR +#undef HAVE__STRLWR +#undef HAVE_INDEX +#undef HAVE_RINDEX +#undef HAVE_STRCHR +#undef HAVE_STRRCHR +#undef HAVE_STRSTR +#undef HAVE_ITOA +#undef HAVE__LTOA +#undef HAVE__UITOA +#undef HAVE__ULTOA +#undef HAVE_STRTOL +#undef HAVE_STRTOUL +#undef HAVE__I64TOA +#undef HAVE__UI64TOA +#undef HAVE_STRTOLL +#undef HAVE_STRTOULL +#undef HAVE_STRTOD +#undef HAVE_ATOI +#undef HAVE_ATOF +#undef HAVE_STRCMP +#undef HAVE_STRNCMP +#undef HAVE__STRICMP +#undef HAVE_STRCASECMP +#undef HAVE__STRNICMP +#undef HAVE_STRNCASECMP +#undef HAVE_SSCANF +#undef HAVE_SNPRINTF +#undef HAVE_VSNPRINTF +#undef HAVE_ICONV +#undef HAVE_SIGACTION +#undef HAVE_SETJMP +#undef HAVE_NANOSLEEP +#undef HAVE_CLOCK_GETTIME +#undef HAVE_DLVSYM + +#else +/* We may need some replacement for stdarg.h here */ +#include +#endif /* HAVE_LIBC */ + +/* Allow disabling of core subsystems */ +#undef SDL_AUDIO_DISABLED +#undef SDL_CDROM_DISABLED +#undef SDL_CPUINFO_DISABLED +#undef SDL_EVENTS_DISABLED +#undef SDL_FILE_DISABLED +#undef SDL_JOYSTICK_DISABLED +#undef SDL_LOADSO_DISABLED +#undef SDL_THREADS_DISABLED +#undef SDL_TIMERS_DISABLED +#undef SDL_VIDEO_DISABLED + +/* Enable various audio drivers */ +#undef SDL_AUDIO_DRIVER_AHI +#undef SDL_AUDIO_DRIVER_ALSA +#undef SDL_AUDIO_DRIVER_ALSA_DYNAMIC +#undef SDL_AUDIO_DRIVER_ARTS +#undef SDL_AUDIO_DRIVER_ARTS_DYNAMIC +#undef SDL_AUDIO_DRIVER_BAUDIO +#undef SDL_AUDIO_DRIVER_BSD +#undef SDL_AUDIO_DRIVER_COREAUDIO +#undef SDL_AUDIO_DRIVER_DART +#undef SDL_AUDIO_DRIVER_DC +#undef SDL_AUDIO_DRIVER_DISK +#undef SDL_AUDIO_DRIVER_DUMMY +#undef SDL_AUDIO_DRIVER_DMEDIA +#undef SDL_AUDIO_DRIVER_DSOUND +#undef SDL_AUDIO_DRIVER_ESD +#undef SDL_AUDIO_DRIVER_ESD_DYNAMIC +#undef SDL_AUDIO_DRIVER_MINT +#undef SDL_AUDIO_DRIVER_MMEAUDIO +#undef SDL_AUDIO_DRIVER_NAS +#undef SDL_AUDIO_DRIVER_OSS +#undef SDL_AUDIO_DRIVER_OSS_SOUNDCARD_H +#undef SDL_AUDIO_DRIVER_PAUD +#undef SDL_AUDIO_DRIVER_QNXNTO +#undef SDL_AUDIO_DRIVER_SNDMGR +#undef SDL_AUDIO_DRIVER_SUNAUDIO +#undef SDL_AUDIO_DRIVER_WAVEOUT + +/* Enable various cdrom drivers */ +#undef SDL_CDROM_AIX +#undef SDL_CDROM_BEOS +#undef SDL_CDROM_BSDI +#undef SDL_CDROM_DC +#undef SDL_CDROM_DUMMY +#undef SDL_CDROM_FREEBSD +#undef SDL_CDROM_LINUX +#undef SDL_CDROM_MACOS +#undef SDL_CDROM_MACOSX +#undef SDL_CDROM_MINT +#undef SDL_CDROM_OPENBSD +#undef SDL_CDROM_OS2 +#undef SDL_CDROM_OSF +#undef SDL_CDROM_QNX +#undef SDL_CDROM_WIN32 + +/* Enable various input drivers */ +#undef SDL_INPUT_TSLIB +#undef SDL_JOYSTICK_AMIGA +#undef SDL_JOYSTICK_BEOS +#undef SDL_JOYSTICK_DC +#undef SDL_JOYSTICK_DUMMY +#undef SDL_JOYSTICK_IOKIT +#undef SDL_JOYSTICK_LINUX +#undef SDL_JOYSTICK_LINUXEV +#undef SDL_JOYSTICK_MACOS +#undef SDL_JOYSTICK_MINT +#undef SDL_JOYSTICK_OS2 +#undef SDL_JOYSTICK_RISCOS +#undef SDL_JOYSTICK_WINMM +#undef SDL_JOYSTICK_USBHID +#undef SDL_JOYSTICK_USBHID_MACHINE_JOYSTICK_H + +/* Enable various shared object loading systems */ +#undef SDL_LOADSO_BEOS +#undef SDL_LOADSO_DLCOMPAT +#undef SDL_LOADSO_DLOPEN +#undef SDL_LOADSO_DUMMY +#undef SDL_LOADSO_LDG +#undef SDL_LOADSO_MACOS +#undef SDL_LOADSO_OS2 +#undef SDL_LOADSO_WIN32 + +/* Enable various threading systems */ +#undef SDL_THREAD_AMIGA +#undef SDL_THREAD_BEOS +#undef SDL_THREAD_DC +#undef SDL_THREAD_EPOC +#undef SDL_THREAD_OS2 +#undef SDL_THREAD_PTH +#undef SDL_THREAD_PTHREAD +#undef SDL_THREAD_PTHREAD_RECURSIVE_MUTEX +#undef SDL_THREAD_PTHREAD_RECURSIVE_MUTEX_NP +#undef SDL_THREAD_SPROC +#undef SDL_THREAD_WIN32 + +/* Enable various timer systems */ +#undef SDL_TIMER_AMIGA +#undef SDL_TIMER_BEOS +#undef SDL_TIMER_DC +#undef SDL_TIMER_DUMMY +#undef SDL_TIMER_EPOC +#undef SDL_TIMER_MACOS +#undef SDL_TIMER_MINT +#undef SDL_TIMER_OS2 +#undef SDL_TIMER_RISCOS +#undef SDL_TIMER_UNIX +#undef SDL_TIMER_WIN32 +#undef SDL_TIMER_WINCE + +/* Enable various video drivers */ +#undef SDL_VIDEO_DRIVER_AALIB +#undef SDL_VIDEO_DRIVER_BWINDOW +#undef SDL_VIDEO_DRIVER_CYBERGRAPHICS +#undef SDL_VIDEO_DRIVER_DC +#undef SDL_VIDEO_DRIVER_DDRAW +#undef SDL_VIDEO_DRIVER_DGA +#undef SDL_VIDEO_DRIVER_DIRECTFB +#undef SDL_VIDEO_DRIVER_DRAWSPROCKET +#undef SDL_VIDEO_DRIVER_DUMMY +#undef SDL_VIDEO_DRIVER_EPOC +#undef SDL_VIDEO_DRIVER_FBCON +#undef SDL_VIDEO_DRIVER_GAPI +#undef SDL_VIDEO_DRIVER_GEM +#undef SDL_VIDEO_DRIVER_GGI +#undef SDL_VIDEO_DRIVER_IPOD +#undef SDL_VIDEO_DRIVER_NANOX +#undef SDL_VIDEO_DRIVER_OS2FS +#undef SDL_VIDEO_DRIVER_PHOTON +#undef SDL_VIDEO_DRIVER_PICOGUI +#undef SDL_VIDEO_DRIVER_PS2GS +#undef SDL_VIDEO_DRIVER_QTOPIA +#undef SDL_VIDEO_DRIVER_QUARTZ +#undef SDL_VIDEO_DRIVER_RISCOS +#undef SDL_VIDEO_DRIVER_SVGALIB +#undef SDL_VIDEO_DRIVER_TOOLBOX +#undef SDL_VIDEO_DRIVER_VGL +#undef SDL_VIDEO_DRIVER_WINDIB +#undef SDL_VIDEO_DRIVER_WSCONS +#undef SDL_VIDEO_DRIVER_X11 +#undef SDL_VIDEO_DRIVER_X11_DGAMOUSE +#undef SDL_VIDEO_DRIVER_X11_DPMS +#undef SDL_VIDEO_DRIVER_X11_DYNAMIC +#undef SDL_VIDEO_DRIVER_X11_DYNAMIC_XEXT +#undef SDL_VIDEO_DRIVER_X11_DYNAMIC_XRANDR +#undef SDL_VIDEO_DRIVER_X11_DYNAMIC_XRENDER +#undef SDL_VIDEO_DRIVER_X11_VIDMODE +#undef SDL_VIDEO_DRIVER_X11_XINERAMA +#undef SDL_VIDEO_DRIVER_X11_XME +#undef SDL_VIDEO_DRIVER_X11_XRANDR +#undef SDL_VIDEO_DRIVER_X11_XV +#undef SDL_VIDEO_DRIVER_XBIOS + +/* Enable OpenGL support */ +#undef SDL_VIDEO_OPENGL +#undef SDL_VIDEO_OPENGL_GLX +#undef SDL_VIDEO_OPENGL_WGL +#undef SDL_VIDEO_OPENGL_OSMESA +#undef SDL_VIDEO_OPENGL_OSMESA_DYNAMIC + +/* Enable assembly routines */ +#undef SDL_ASSEMBLY_ROUTINES +#undef SDL_HERMES_BLITTERS +#undef SDL_ALTIVEC_BLITTERS + +#endif /* _SDL_config_h */ diff --git a/src/win32/dependencies/sdl/SDL_config_amiga.h b/src/win32/dependencies/sdl/SDL_config_amiga.h new file mode 100644 index 00000000..23e08619 --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_config_amiga.h @@ -0,0 +1,80 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +#ifndef _SDL_config_amiga_h +#define _SDL_config_amiga_h + +#include "SDL_platform.h" + +/* This is a set of defines to configure the SDL features */ + +#define SDL_HAS_64BIT_TYPE 1 + +/* Useful headers */ +#define HAVE_SYS_TYPES_H 1 +#define HAVE_STDIO_H 1 +#define STDC_HEADERS 1 +#define HAVE_STRING_H 1 +#define HAVE_INTTYPES_H 1 +#define HAVE_SIGNAL_H 1 + +/* C library functions */ +#define HAVE_MALLOC 1 +#define HAVE_CALLOC 1 +#define HAVE_REALLOC 1 +#define HAVE_FREE 1 +#define HAVE_ALLOCA 1 +#define HAVE_GETENV 1 +#define HAVE_PUTENV 1 +#define HAVE_MEMSET 1 +#define HAVE_MEMCPY 1 +#define HAVE_MEMMOVE 1 +#define HAVE_MEMCMP 1 + +/* Enable various audio drivers */ +#define SDL_AUDIO_DRIVER_AHI 1 +#define SDL_AUDIO_DRIVER_DISK 1 +#define SDL_AUDIO_DRIVER_DUMMY 1 + +/* Enable various cdrom drivers */ +#define SDL_CDROM_DUMMY 1 + +/* Enable various input drivers */ +#define SDL_JOYSTICK_AMIGA 1 + +/* Enable various shared object loading systems */ +#define SDL_LOADSO_DUMMY 1 + +/* Enable various threading systems */ +#define SDL_THREAD_AMIGA 1 + +/* Enable various timer systems */ +#define SDL_TIMER_AMIGA 1 + +/* Enable various video drivers */ +#define SDL_VIDEO_DRIVER_CYBERGRAPHICS 1 +#define SDL_VIDEO_DRIVER_DUMMY 1 + +/* Enable OpenGL support */ +#define SDL_VIDEO_OPENGL 1 + +#endif /* _SDL_config_amiga_h */ diff --git a/src/win32/dependencies/sdl/SDL_config_dreamcast.h b/src/win32/dependencies/sdl/SDL_config_dreamcast.h new file mode 100644 index 00000000..9cbeea31 --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_config_dreamcast.h @@ -0,0 +1,106 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +#ifndef _SDL_config_dreamcast_h +#define _SDL_config_dreamcast_h + +#include "SDL_platform.h" + +/* This is a set of defines to configure the SDL features */ + +typedef signed char int8_t; +typedef unsigned char uint8_t; +typedef signed short int16_t; +typedef unsigned short uint16_t; +typedef signed int int32_t; +typedef unsigned int uint32_t; +typedef signed long long int64_t; +typedef unsigned long long uint64_t; +typedef unsigned long uintptr_t; +#define SDL_HAS_64BIT_TYPE 1 + +/* Useful headers */ +#define HAVE_SYS_TYPES_H 1 +#define HAVE_STDIO_H 1 +#define STDC_HEADERS 1 +#define HAVE_STRING_H 1 +#define HAVE_CTYPE_H 1 + +/* C library functions */ +#define HAVE_MALLOC 1 +#define HAVE_CALLOC 1 +#define HAVE_REALLOC 1 +#define HAVE_FREE 1 +#define HAVE_ALLOCA 1 +#define HAVE_GETENV 1 +#define HAVE_PUTENV 1 +#define HAVE_QSORT 1 +#define HAVE_ABS 1 +#define HAVE_BCOPY 1 +#define HAVE_MEMSET 1 +#define HAVE_MEMCPY 1 +#define HAVE_MEMMOVE 1 +#define HAVE_MEMCMP 1 +#define HAVE_STRLEN 1 +#define HAVE_STRDUP 1 +#define HAVE_INDEX 1 +#define HAVE_RINDEX 1 +#define HAVE_STRCHR 1 +#define HAVE_STRRCHR 1 +#define HAVE_STRSTR 1 +#define HAVE_STRTOL 1 +#define HAVE_STRTOD 1 +#define HAVE_ATOI 1 +#define HAVE_ATOF 1 +#define HAVE_STRCMP 1 +#define HAVE_STRNCMP 1 +#define HAVE_STRICMP 1 +#define HAVE_STRCASECMP 1 +#define HAVE_SSCANF 1 +#define HAVE_SNPRINTF 1 +#define HAVE_VSNPRINTF 1 + +/* Enable various audio drivers */ +#define SDL_AUDIO_DRIVER_DC 1 +#define SDL_AUDIO_DRIVER_DISK 1 +#define SDL_AUDIO_DRIVER_DUMMY 1 + +/* Enable various cdrom drivers */ +#define SDL_CDROM_DC 1 + +/* Enable various input drivers */ +#define SDL_JOYSTICK_DC 1 + +/* Enable various shared object loading systems */ +#define SDL_LOADSO_DUMMY 1 + +/* Enable various threading systems */ +#define SDL_THREAD_DC 1 + +/* Enable various timer systems */ +#define SDL_TIMER_DC 1 + +/* Enable various video drivers */ +#define SDL_VIDEO_DRIVER_DC 1 +#define SDL_VIDEO_DRIVER_DUMMY 1 + +#endif /* _SDL_config_dreamcast_h */ diff --git a/src/win32/dependencies/sdl/SDL_config_macos.h b/src/win32/dependencies/sdl/SDL_config_macos.h new file mode 100644 index 00000000..c4a1c598 --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_config_macos.h @@ -0,0 +1,112 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +#ifndef _SDL_config_macos_h +#define _SDL_config_macos_h + +#include "SDL_platform.h" + +/* This is a set of defines to configure the SDL features */ + +#include + +typedef SInt8 int8_t; +typedef UInt8 uint8_t; +typedef SInt16 int16_t; +typedef UInt16 uint16_t; +typedef SInt32 int32_t; +typedef UInt32 uint32_t; +typedef SInt64 int64_t; +typedef UInt64 uint64_t; +typedef unsigned long uintptr_t; + +#define SDL_HAS_64BIT_TYPE 1 + +/* Useful headers */ +#define HAVE_STDIO_H 1 +#define STDC_HEADERS 1 +#define HAVE_STRING_H 1 +#define HAVE_CTYPE_H 1 +#define HAVE_MATH_H 1 +#define HAVE_SIGNAL_H 1 + +/* C library functions */ +#define HAVE_MALLOC 1 +#define HAVE_CALLOC 1 +#define HAVE_REALLOC 1 +#define HAVE_FREE 1 +#define HAVE_ALLOCA 1 +#define HAVE_ABS 1 +#define HAVE_MEMSET 1 +#define HAVE_MEMCPY 1 +#define HAVE_MEMMOVE 1 +#define HAVE_MEMCMP 1 +#define HAVE_STRLEN 1 +#define HAVE_STRCHR 1 +#define HAVE_STRRCHR 1 +#define HAVE_STRSTR 1 +#define HAVE_ITOA 1 +#define HAVE_STRTOL 1 +#define HAVE_STRTOD 1 +#define HAVE_ATOI 1 +#define HAVE_ATOF 1 +#define HAVE_STRCMP 1 +#define HAVE_STRNCMP 1 +#define HAVE_SSCANF 1 + +/* Enable various audio drivers */ +#define SDL_AUDIO_DRIVER_SNDMGR 1 +#define SDL_AUDIO_DRIVER_DISK 1 +#define SDL_AUDIO_DRIVER_DUMMY 1 + +/* Enable various cdrom drivers */ +#if TARGET_API_MAC_CARBON +#define SDL_CDROM_DUMMY 1 +#else +#define SDL_CDROM_MACOS 1 +#endif + +/* Enable various input drivers */ +#if TARGET_API_MAC_CARBON +#define SDL_JOYSTICK_DUMMY 1 +#else +#define SDL_JOYSTICK_MACOS 1 +#endif + +/* Enable various shared object loading systems */ +#define SDL_LOADSO_MACOS 1 + +/* Enable various threading systems */ +#define SDL_THREADS_DISABLED 1 + +/* Enable various timer systems */ +#define SDL_TIMER_MACOS 1 + +/* Enable various video drivers */ +#define SDL_VIDEO_DRIVER_DUMMY 1 +#define SDL_VIDEO_DRIVER_DRAWSPROCKET 1 +#define SDL_VIDEO_DRIVER_TOOLBOX 1 + +/* Enable OpenGL support */ +#define SDL_VIDEO_OPENGL 1 + +#endif /* _SDL_config_macos_h */ diff --git a/src/win32/dependencies/sdl/SDL_config_macosx.h b/src/win32/dependencies/sdl/SDL_config_macosx.h new file mode 100644 index 00000000..8f04930d --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_config_macosx.h @@ -0,0 +1,132 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +#ifndef _SDL_config_macosx_h +#define _SDL_config_macosx_h + +#include "SDL_platform.h" + +/* This is a set of defines to configure the SDL features */ + +#define SDL_HAS_64BIT_TYPE 1 + +/* Useful headers */ +/* If we specified an SDK or have a post-PowerPC chip, then alloca.h exists. */ +#if ( (MAC_OS_X_VERSION_MIN_REQUIRED >= 1030) || (!defined(__POWERPC__)) ) +#define HAVE_ALLOCA_H 1 +#endif +#define HAVE_SYS_TYPES_H 1 +#define HAVE_STDIO_H 1 +#define STDC_HEADERS 1 +#define HAVE_STRING_H 1 +#define HAVE_INTTYPES_H 1 +#define HAVE_STDINT_H 1 +#define HAVE_CTYPE_H 1 +#define HAVE_MATH_H 1 +#define HAVE_SIGNAL_H 1 + +/* C library functions */ +#define HAVE_MALLOC 1 +#define HAVE_CALLOC 1 +#define HAVE_REALLOC 1 +#define HAVE_FREE 1 +#define HAVE_ALLOCA 1 +#define HAVE_GETENV 1 +#define HAVE_PUTENV 1 +#define HAVE_UNSETENV 1 +#define HAVE_QSORT 1 +#define HAVE_ABS 1 +#define HAVE_BCOPY 1 +#define HAVE_MEMSET 1 +#define HAVE_MEMCPY 1 +#define HAVE_MEMMOVE 1 +#define HAVE_MEMCMP 1 +#define HAVE_STRLEN 1 +#define HAVE_STRLCPY 1 +#define HAVE_STRLCAT 1 +#define HAVE_STRDUP 1 +#define HAVE_STRCHR 1 +#define HAVE_STRRCHR 1 +#define HAVE_STRSTR 1 +#define HAVE_STRTOL 1 +#define HAVE_STRTOUL 1 +#define HAVE_STRTOLL 1 +#define HAVE_STRTOULL 1 +#define HAVE_STRTOD 1 +#define HAVE_ATOI 1 +#define HAVE_ATOF 1 +#define HAVE_STRCMP 1 +#define HAVE_STRNCMP 1 +#define HAVE_STRCASECMP 1 +#define HAVE_STRNCASECMP 1 +#define HAVE_SSCANF 1 +#define HAVE_SNPRINTF 1 +#define HAVE_VSNPRINTF 1 +#define HAVE_SIGACTION 1 +#define HAVE_SETJMP 1 +#define HAVE_NANOSLEEP 1 + +/* Enable various audio drivers */ +#define SDL_AUDIO_DRIVER_COREAUDIO 1 +#define SDL_AUDIO_DRIVER_SNDMGR 1 +#define SDL_AUDIO_DRIVER_DISK 1 +#define SDL_AUDIO_DRIVER_DUMMY 1 + +/* Enable various cdrom drivers */ +#define SDL_CDROM_MACOSX 1 + +/* Enable various input drivers */ +#define SDL_JOYSTICK_IOKIT 1 + +/* Enable various shared object loading systems */ +#ifdef __ppc__ +/* For Mac OS X 10.2 compatibility */ +#define SDL_LOADSO_DLCOMPAT 1 +#else +#define SDL_LOADSO_DLOPEN 1 +#endif + +/* Enable various threading systems */ +#define SDL_THREAD_PTHREAD 1 +#define SDL_THREAD_PTHREAD_RECURSIVE_MUTEX 1 + +/* Enable various timer systems */ +#define SDL_TIMER_UNIX 1 + +/* Enable various video drivers */ +#define SDL_VIDEO_DRIVER_DUMMY 1 +#if TARGET_API_MAC_CARBON +#define SDL_VIDEO_DRIVER_TOOLBOX 1 +#else +#define SDL_VIDEO_DRIVER_QUARTZ 1 +#endif + +/* Enable OpenGL support */ +#define SDL_VIDEO_OPENGL 1 + +/* Enable assembly routines */ +#define SDL_ASSEMBLY_ROUTINES 1 +#ifdef __ppc__ +#define SDL_ALTIVEC_BLITTERS 1 +#endif + +#endif /* _SDL_config_macosx_h */ diff --git a/src/win32/dependencies/sdl/SDL_config_minimal.h b/src/win32/dependencies/sdl/SDL_config_minimal.h new file mode 100644 index 00000000..78b6148c --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_config_minimal.h @@ -0,0 +1,62 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +#ifndef _SDL_config_minimal_h +#define _SDL_config_minimal_h + +#include "SDL_platform.h" + +/* This is the minimal configuration that can be used to build SDL */ + +#include + +typedef signed char int8_t; +typedef unsigned char uint8_t; +typedef signed short int16_t; +typedef unsigned short uint16_t; +typedef signed int int32_t; +typedef unsigned int uint32_t; +typedef unsigned int size_t; +typedef unsigned long uintptr_t; + +/* Enable the dummy audio driver (src/audio/dummy/\*.c) */ +#define SDL_AUDIO_DRIVER_DUMMY 1 + +/* Enable the stub cdrom driver (src/cdrom/dummy/\*.c) */ +#define SDL_CDROM_DISABLED 1 + +/* Enable the stub joystick driver (src/joystick/dummy/\*.c) */ +#define SDL_JOYSTICK_DISABLED 1 + +/* Enable the stub shared object loader (src/loadso/dummy/\*.c) */ +#define SDL_LOADSO_DISABLED 1 + +/* Enable the stub thread support (src/thread/generic/\*.c) */ +#define SDL_THREADS_DISABLED 1 + +/* Enable the stub timer support (src/timer/dummy/\*.c) */ +#define SDL_TIMERS_DISABLED 1 + +/* Enable the dummy video driver (src/video/dummy/\*.c) */ +#define SDL_VIDEO_DRIVER_DUMMY 1 + +#endif /* _SDL_config_minimal_h */ diff --git a/src/win32/dependencies/sdl/SDL_config_os2.h b/src/win32/dependencies/sdl/SDL_config_os2.h new file mode 100644 index 00000000..8cdea9ff --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_config_os2.h @@ -0,0 +1,141 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +#ifndef _SDL_config_os2_h +#define _SDL_config_os2_h + +#include "SDL_platform.h" + +/* This is a set of defines to configure the SDL features */ + +typedef signed char int8_t; +typedef unsigned char uint8_t; +typedef signed short int16_t; +typedef unsigned short uint16_t; +typedef signed int int32_t; +typedef unsigned int uint32_t; +typedef unsigned int size_t; +typedef unsigned long uintptr_t; +typedef signed long long int64_t; +typedef unsigned long long uint64_t; + +#define SDL_HAS_64BIT_TYPE 1 + +/* Use Watcom's LIBC */ +#define HAVE_LIBC 1 + +/* Useful headers */ +#define HAVE_SYS_TYPES_H 1 +#define HAVE_STDIO_H 1 +#define STDC_HEADERS 1 +#define HAVE_STDLIB_H 1 +#define HAVE_STDARG_H 1 +#define HAVE_MALLOC_H 1 +#define HAVE_MEMORY_H 1 +#define HAVE_STRING_H 1 +#define HAVE_STRINGS_H 1 +#define HAVE_INTTYPES_H 1 +#define HAVE_STDINT_H 1 +#define HAVE_CTYPE_H 1 +#define HAVE_MATH_H 1 +#define HAVE_SIGNAL_H 1 + +/* C library functions */ +#define HAVE_MALLOC 1 +#define HAVE_CALLOC 1 +#define HAVE_REALLOC 1 +#define HAVE_FREE 1 +#define HAVE_ALLOCA 1 +#define HAVE_GETENV 1 +#define HAVE_PUTENV 1 +#define HAVE_UNSETENV 1 +#define HAVE_QSORT 1 +#define HAVE_ABS 1 +#define HAVE_BCOPY 1 +#define HAVE_MEMSET 1 +#define HAVE_MEMCPY 1 +#define HAVE_MEMMOVE 1 +#define HAVE_MEMCMP 1 +#define HAVE_STRLEN 1 +#define HAVE_STRLCPY 1 +#define HAVE_STRLCAT 1 +#define HAVE_STRDUP 1 +#define HAVE__STRREV 1 +#define HAVE__STRUPR 1 +#define HAVE__STRLWR 1 +#define HAVE_INDEX 1 +#define HAVE_RINDEX 1 +#define HAVE_STRCHR 1 +#define HAVE_STRRCHR 1 +#define HAVE_STRSTR 1 +#define HAVE_ITOA 1 +#define HAVE__LTOA 1 +#define HAVE__UITOA 1 +#define HAVE__ULTOA 1 +#define HAVE_STRTOL 1 +#define HAVE__I64TOA 1 +#define HAVE__UI64TOA 1 +#define HAVE_STRTOLL 1 +#define HAVE_STRTOD 1 +#define HAVE_ATOI 1 +#define HAVE_ATOF 1 +#define HAVE_STRCMP 1 +#define HAVE_STRNCMP 1 +#define HAVE_STRICMP 1 +#define HAVE_STRCASECMP 1 +#define HAVE_SSCANF 1 +#define HAVE_SNPRINTF 1 +#define HAVE_VSNPRINTF 1 +#define HAVE_SETJMP 1 +#define HAVE_CLOCK_GETTIME 1 + +/* Enable various audio drivers */ +#define SDL_AUDIO_DRIVER_DART 1 +#define SDL_AUDIO_DRIVER_DISK 1 +#define SDL_AUDIO_DRIVER_DUMMY 1 + +/* Enable various cdrom drivers */ +#define SDL_CDROM_OS2 1 + +/* Enable various input drivers */ +#define SDL_JOYSTICK_OS2 1 + +/* Enable various shared object loading systems */ +#define SDL_LOADSO_OS2 1 + +/* Enable various threading systems */ +#define SDL_THREAD_OS2 1 + +/* Enable various timer systems */ +#define SDL_TIMER_OS2 1 + +/* Enable various video drivers */ +#define SDL_VIDEO_DRIVER_DUMMY 1 +#define SDL_VIDEO_DRIVER_OS2FS 1 + +/* Enable OpenGL support */ +/* Nothing here yet for OS/2... :( */ + +/* Enable assembly routines where available */ +#define SDL_ASSEMBLY_ROUTINES 1 + +#endif /* _SDL_config_os2_h */ diff --git a/src/win32/dependencies/sdl/SDL_config_win32.h b/src/win32/dependencies/sdl/SDL_config_win32.h new file mode 100644 index 00000000..3f02fcca --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_config_win32.h @@ -0,0 +1,171 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +#ifndef _SDL_config_win32_h +#define _SDL_config_win32_h + +#include "SDL_platform.h" + +/* This is a set of defines to configure the SDL features */ + +#ifdef _MSC_VER +typedef signed __int8 int8_t; +typedef unsigned __int8 uint8_t; +typedef signed __int16 int16_t; +typedef unsigned __int16 uint16_t; +typedef signed __int32 int32_t; +typedef unsigned __int32 uint32_t; +typedef signed __int64 int64_t; +typedef unsigned __int64 uint64_t; +#ifndef _UINTPTR_T_DEFINED +#ifdef _WIN64 +typedef unsigned __int64 uintptr_t; +#else +typedef unsigned int uintptr_t; +#endif +#define _UINTPTR_T_DEFINED +#endif +#else +typedef signed char int8_t; +typedef unsigned char uint8_t; +typedef signed short int16_t; +typedef unsigned short uint16_t; +typedef signed int int32_t; +typedef unsigned int uint32_t; +typedef signed long long int64_t; +typedef unsigned long long uint64_t; +#ifndef _SIZE_T_DEFINED_ +#define _SIZE_T_DEFINED_ +typedef unsigned int size_t; +#endif +typedef unsigned int uintptr_t; +#endif /* _MSC_VER */ +#define SDL_HAS_64BIT_TYPE 1 + +/* Enabled for SDL 1.2 (binary compatibility) */ +#define HAVE_LIBC 1 +#ifdef HAVE_LIBC +/* Useful headers */ +#define HAVE_STDIO_H 1 +#define STDC_HEADERS 1 +#define HAVE_STRING_H 1 +#define HAVE_CTYPE_H 1 +#define HAVE_MATH_H 1 +#ifndef _WIN32_WCE +#define HAVE_SIGNAL_H 1 +#endif + +/* C library functions */ +#define HAVE_MALLOC 1 +#define HAVE_CALLOC 1 +#define HAVE_REALLOC 1 +#define HAVE_FREE 1 +#define HAVE_ALLOCA 1 +#define HAVE_QSORT 1 +#define HAVE_ABS 1 +#define HAVE_MEMSET 1 +#define HAVE_MEMCPY 1 +#define HAVE_MEMMOVE 1 +#define HAVE_MEMCMP 1 +#define HAVE_STRLEN 1 +#define HAVE__STRREV 1 +#define HAVE__STRUPR 1 +#define HAVE__STRLWR 1 +#define HAVE_STRCHR 1 +#define HAVE_STRRCHR 1 +#define HAVE_STRSTR 1 +#define HAVE_ITOA 1 +#define HAVE__LTOA 1 +#define HAVE__ULTOA 1 +#define HAVE_STRTOL 1 +#define HAVE_STRTOUL 1 +#define HAVE_STRTOLL 1 +#define HAVE_STRTOD 1 +#define HAVE_ATOI 1 +#define HAVE_ATOF 1 +#define HAVE_STRCMP 1 +#define HAVE_STRNCMP 1 +#define HAVE__STRICMP 1 +#define HAVE__STRNICMP 1 +#define HAVE_SSCANF 1 +#else +#define HAVE_STDARG_H 1 +#define HAVE_STDDEF_H 1 +#endif + +/* Enable various audio drivers */ +#ifndef _WIN32_WCE +#define SDL_AUDIO_DRIVER_DSOUND 1 +#endif +#define SDL_AUDIO_DRIVER_WAVEOUT 1 +#define SDL_AUDIO_DRIVER_DISK 1 +#define SDL_AUDIO_DRIVER_DUMMY 1 + +/* Enable various cdrom drivers */ +#ifdef _WIN32_WCE +#define SDL_CDROM_DISABLED 1 +#else +#define SDL_CDROM_WIN32 1 +#endif + +/* Enable various input drivers */ +#ifdef _WIN32_WCE +#define SDL_JOYSTICK_DISABLED 1 +#else +#define SDL_JOYSTICK_WINMM 1 +#endif + +/* Enable various shared object loading systems */ +#define SDL_LOADSO_WIN32 1 + +/* Enable various threading systems */ +#define SDL_THREAD_WIN32 1 + +/* Enable various timer systems */ +#ifdef _WIN32_WCE +#define SDL_TIMER_WINCE 1 +#else +#define SDL_TIMER_WIN32 1 +#endif + +/* Enable various video drivers */ +#ifdef _WIN32_WCE +#define SDL_VIDEO_DRIVER_GAPI 1 +#endif +#ifndef _WIN32_WCE +#define SDL_VIDEO_DRIVER_DDRAW 1 +#endif +#define SDL_VIDEO_DRIVER_DUMMY 1 +#define SDL_VIDEO_DRIVER_WINDIB 1 + +/* Enable OpenGL support */ +#ifndef _WIN32_WCE +#define SDL_VIDEO_OPENGL 1 +#define SDL_VIDEO_OPENGL_WGL 1 +#endif + +/* Enable assembly routines (Win64 doesn't have inline asm) */ +#ifndef _WIN64 +#define SDL_ASSEMBLY_ROUTINES 1 +#endif + +#endif /* _SDL_config_win32_h */ diff --git a/src/win32/dependencies/sdl/SDL_copying.h b/src/win32/dependencies/sdl/SDL_copying.h new file mode 100644 index 00000000..39e122db --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_copying.h @@ -0,0 +1,22 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + diff --git a/src/win32/dependencies/sdl/SDL_cpuinfo.h b/src/win32/dependencies/sdl/SDL_cpuinfo.h new file mode 100644 index 00000000..72acbdd8 --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_cpuinfo.h @@ -0,0 +1,75 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ +/* CPU feature detection for SDL */ + +#ifndef _SDL_cpuinfo_h +#define _SDL_cpuinfo_h + +#include "SDL_stdinc.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* This function returns true if the CPU has the RDTSC instruction + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasRDTSC(void); + +/* This function returns true if the CPU has MMX features + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasMMX(void); + +/* This function returns true if the CPU has MMX Ext. features + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasMMXExt(void); + +/* This function returns true if the CPU has 3DNow features + */ +extern DECLSPEC SDL_bool SDLCALL SDL_Has3DNow(void); + +/* This function returns true if the CPU has 3DNow! Ext. features + */ +extern DECLSPEC SDL_bool SDLCALL SDL_Has3DNowExt(void); + +/* This function returns true if the CPU has SSE features + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasSSE(void); + +/* This function returns true if the CPU has SSE2 features + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasSSE2(void); + +/* This function returns true if the CPU has AltiVec features + */ +extern DECLSPEC SDL_bool SDLCALL SDL_HasAltiVec(void); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* _SDL_cpuinfo_h */ diff --git a/src/win32/dependencies/sdl/SDL_endian.h b/src/win32/dependencies/sdl/SDL_endian.h new file mode 100644 index 00000000..6257a649 --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_endian.h @@ -0,0 +1,192 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* Functions for reading and writing endian-specific values */ + +#ifndef _SDL_endian_h +#define _SDL_endian_h + +#include "SDL_stdinc.h" + +/* The two types of endianness */ +#define SDL_LIL_ENDIAN 1234 +#define SDL_BIG_ENDIAN 4321 + +#ifndef SDL_BYTEORDER /* Not defined in SDL_config.h? */ +#if defined(__hppa__) || \ + defined(__m68k__) || defined(mc68000) || defined(_M_M68K) || \ + (defined(__MIPS__) && defined(__MISPEB__)) || \ + defined(__ppc__) || defined(__POWERPC__) || defined(_M_PPC) || \ + defined(__sparc__) +#define SDL_BYTEORDER SDL_BIG_ENDIAN +#else +#define SDL_BYTEORDER SDL_LIL_ENDIAN +#endif +#endif /* !SDL_BYTEORDER */ + + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* Use inline functions for compilers that support them, and static + functions for those that do not. Because these functions become + static for compilers that do not support inline functions, this + header should only be included in files that actually use them. +*/ +#if defined(__GNUC__) && defined(__i386__) && \ + !(__GNUC__ == 2 && __GNUC_MINOR__ == 95 /* broken gcc version */) +static __inline__ Uint16 SDL_Swap16(Uint16 x) +{ + __asm__("xchgb %b0,%h0" : "=q" (x) : "0" (x)); + return x; +} +#elif defined(__GNUC__) && defined(__x86_64__) +static __inline__ Uint16 SDL_Swap16(Uint16 x) +{ + __asm__("xchgb %b0,%h0" : "=Q" (x) : "0" (x)); + return x; +} +#elif defined(__GNUC__) && (defined(__powerpc__) || defined(__ppc__)) +static __inline__ Uint16 SDL_Swap16(Uint16 x) +{ + Uint16 result; + + __asm__("rlwimi %0,%2,8,16,23" : "=&r" (result) : "0" (x >> 8), "r" (x)); + return result; +} +#elif defined(__GNUC__) && (defined(__M68000__) || defined(__M68020__)) +static __inline__ Uint16 SDL_Swap16(Uint16 x) +{ + __asm__("rorw #8,%0" : "=d" (x) : "0" (x) : "cc"); + return x; +} +#else +static __inline__ Uint16 SDL_Swap16(Uint16 x) { + return((x<<8)|(x>>8)); +} +#endif + +#if defined(__GNUC__) && defined(__i386__) +static __inline__ Uint32 SDL_Swap32(Uint32 x) +{ + __asm__("bswap %0" : "=r" (x) : "0" (x)); + return x; +} +#elif defined(__GNUC__) && defined(__x86_64__) +static __inline__ Uint32 SDL_Swap32(Uint32 x) +{ + __asm__("bswapl %0" : "=r" (x) : "0" (x)); + return x; +} +#elif defined(__GNUC__) && (defined(__powerpc__) || defined(__ppc__)) +static __inline__ Uint32 SDL_Swap32(Uint32 x) +{ + Uint32 result; + + __asm__("rlwimi %0,%2,24,16,23" : "=&r" (result) : "0" (x>>24), "r" (x)); + __asm__("rlwimi %0,%2,8,8,15" : "=&r" (result) : "0" (result), "r" (x)); + __asm__("rlwimi %0,%2,24,0,7" : "=&r" (result) : "0" (result), "r" (x)); + return result; +} +#elif defined(__GNUC__) && (defined(__M68000__) || defined(__M68020__)) +static __inline__ Uint32 SDL_Swap32(Uint32 x) +{ + __asm__("rorw #8,%0\n\tswap %0\n\trorw #8,%0" : "=d" (x) : "0" (x) : "cc"); + return x; +} +#else +static __inline__ Uint32 SDL_Swap32(Uint32 x) { + return((x<<24)|((x<<8)&0x00FF0000)|((x>>8)&0x0000FF00)|(x>>24)); +} +#endif + +#ifdef SDL_HAS_64BIT_TYPE +#if defined(__GNUC__) && defined(__i386__) +static __inline__ Uint64 SDL_Swap64(Uint64 x) +{ + union { + struct { Uint32 a,b; } s; + Uint64 u; + } v; + v.u = x; + __asm__("bswapl %0 ; bswapl %1 ; xchgl %0,%1" + : "=r" (v.s.a), "=r" (v.s.b) + : "0" (v.s.a), "1" (v.s.b)); + return v.u; +} +#elif defined(__GNUC__) && defined(__x86_64__) +static __inline__ Uint64 SDL_Swap64(Uint64 x) +{ + __asm__("bswapq %0" : "=r" (x) : "0" (x)); + return x; +} +#else +static __inline__ Uint64 SDL_Swap64(Uint64 x) +{ + Uint32 hi, lo; + + /* Separate into high and low 32-bit values and swap them */ + lo = (Uint32)(x&0xFFFFFFFF); + x >>= 32; + hi = (Uint32)(x&0xFFFFFFFF); + x = SDL_Swap32(lo); + x <<= 32; + x |= SDL_Swap32(hi); + return(x); +} +#endif +#else +/* This is mainly to keep compilers from complaining in SDL code. + If there is no real 64-bit datatype, then compilers will complain about + the fake 64-bit datatype that SDL provides when it compiles user code. +*/ +#define SDL_Swap64(X) (X) +#endif /* SDL_HAS_64BIT_TYPE */ + + +/* Byteswap item from the specified endianness to the native endianness */ +#if SDL_BYTEORDER == SDL_LIL_ENDIAN +#define SDL_SwapLE16(X) (X) +#define SDL_SwapLE32(X) (X) +#define SDL_SwapLE64(X) (X) +#define SDL_SwapBE16(X) SDL_Swap16(X) +#define SDL_SwapBE32(X) SDL_Swap32(X) +#define SDL_SwapBE64(X) SDL_Swap64(X) +#else +#define SDL_SwapLE16(X) SDL_Swap16(X) +#define SDL_SwapLE32(X) SDL_Swap32(X) +#define SDL_SwapLE64(X) SDL_Swap64(X) +#define SDL_SwapBE16(X) (X) +#define SDL_SwapBE32(X) (X) +#define SDL_SwapBE64(X) (X) +#endif + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* _SDL_endian_h */ diff --git a/src/win32/dependencies/sdl/SDL_error.h b/src/win32/dependencies/sdl/SDL_error.h new file mode 100644 index 00000000..26d6bfae --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_error.h @@ -0,0 +1,61 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* Simple error message routines for SDL */ + +#ifndef _SDL_error_h +#define _SDL_error_h + +#include "SDL_stdinc.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* Public functions */ +extern DECLSPEC void SDLCALL SDL_SetError(const char *fmt, ...); +extern DECLSPEC char * SDLCALL SDL_GetError(void); +extern DECLSPEC void SDLCALL SDL_ClearError(void); + +/* Private error message function - used internally */ +#define SDL_OutOfMemory() SDL_Error(SDL_ENOMEM) +#define SDL_Unsupported() SDL_Error(SDL_UNSUPPORTED) +typedef enum { + SDL_ENOMEM, + SDL_EFREAD, + SDL_EFWRITE, + SDL_EFSEEK, + SDL_UNSUPPORTED, + SDL_LASTERROR +} SDL_errorcode; +extern DECLSPEC void SDLCALL SDL_Error(SDL_errorcode code); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* _SDL_error_h */ diff --git a/src/win32/dependencies/sdl/SDL_events.h b/src/win32/dependencies/sdl/SDL_events.h new file mode 100644 index 00000000..9fe918c7 --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_events.h @@ -0,0 +1,337 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* Include file for SDL event handling */ + +#ifndef _SDL_events_h +#define _SDL_events_h + +#include "SDL_stdinc.h" +#include "SDL_error.h" +#include "SDL_active.h" +#include "SDL_keyboard.h" +#include "SDL_mouse.h" +#include "SDL_joystick.h" +#include "SDL_quit.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* General keyboard/mouse state definitions */ +#define SDL_RELEASED 0 +#define SDL_PRESSED 1 + +/* Event enumerations */ +typedef enum { + SDL_NOEVENT = 0, /* Unused (do not remove) */ + SDL_ACTIVEEVENT, /* Application loses/gains visibility */ + SDL_KEYDOWN, /* Keys pressed */ + SDL_KEYUP, /* Keys released */ + SDL_MOUSEMOTION, /* Mouse moved */ + SDL_MOUSEBUTTONDOWN, /* Mouse button pressed */ + SDL_MOUSEBUTTONUP, /* Mouse button released */ + SDL_JOYAXISMOTION, /* Joystick axis motion */ + SDL_JOYBALLMOTION, /* Joystick trackball motion */ + SDL_JOYHATMOTION, /* Joystick hat position change */ + SDL_JOYBUTTONDOWN, /* Joystick button pressed */ + SDL_JOYBUTTONUP, /* Joystick button released */ + SDL_QUIT, /* User-requested quit */ + SDL_SYSWMEVENT, /* System specific event */ + SDL_EVENT_RESERVEDA, /* Reserved for future use.. */ + SDL_EVENT_RESERVEDB, /* Reserved for future use.. */ + SDL_VIDEORESIZE, /* User resized video mode */ + SDL_VIDEOEXPOSE, /* Screen needs to be redrawn */ + SDL_EVENT_RESERVED2, /* Reserved for future use.. */ + SDL_EVENT_RESERVED3, /* Reserved for future use.. */ + SDL_EVENT_RESERVED4, /* Reserved for future use.. */ + SDL_EVENT_RESERVED5, /* Reserved for future use.. */ + SDL_EVENT_RESERVED6, /* Reserved for future use.. */ + SDL_EVENT_RESERVED7, /* Reserved for future use.. */ + /* Events SDL_USEREVENT through SDL_MAXEVENTS-1 are for your use */ + SDL_USEREVENT = 24, + /* This last event is only for bounding internal arrays + It is the number of bits in the event mask datatype -- Uint32 + */ + SDL_NUMEVENTS = 32 +} SDL_EventType; + +/* Predefined event masks */ +#define SDL_EVENTMASK(X) (1<<(X)) +typedef enum { + SDL_ACTIVEEVENTMASK = SDL_EVENTMASK(SDL_ACTIVEEVENT), + SDL_KEYDOWNMASK = SDL_EVENTMASK(SDL_KEYDOWN), + SDL_KEYUPMASK = SDL_EVENTMASK(SDL_KEYUP), + SDL_KEYEVENTMASK = SDL_EVENTMASK(SDL_KEYDOWN)| + SDL_EVENTMASK(SDL_KEYUP), + SDL_MOUSEMOTIONMASK = SDL_EVENTMASK(SDL_MOUSEMOTION), + SDL_MOUSEBUTTONDOWNMASK = SDL_EVENTMASK(SDL_MOUSEBUTTONDOWN), + SDL_MOUSEBUTTONUPMASK = SDL_EVENTMASK(SDL_MOUSEBUTTONUP), + SDL_MOUSEEVENTMASK = SDL_EVENTMASK(SDL_MOUSEMOTION)| + SDL_EVENTMASK(SDL_MOUSEBUTTONDOWN)| + SDL_EVENTMASK(SDL_MOUSEBUTTONUP), + SDL_JOYAXISMOTIONMASK = SDL_EVENTMASK(SDL_JOYAXISMOTION), + SDL_JOYBALLMOTIONMASK = SDL_EVENTMASK(SDL_JOYBALLMOTION), + SDL_JOYHATMOTIONMASK = SDL_EVENTMASK(SDL_JOYHATMOTION), + SDL_JOYBUTTONDOWNMASK = SDL_EVENTMASK(SDL_JOYBUTTONDOWN), + SDL_JOYBUTTONUPMASK = SDL_EVENTMASK(SDL_JOYBUTTONUP), + SDL_JOYEVENTMASK = SDL_EVENTMASK(SDL_JOYAXISMOTION)| + SDL_EVENTMASK(SDL_JOYBALLMOTION)| + SDL_EVENTMASK(SDL_JOYHATMOTION)| + SDL_EVENTMASK(SDL_JOYBUTTONDOWN)| + SDL_EVENTMASK(SDL_JOYBUTTONUP), + SDL_VIDEORESIZEMASK = SDL_EVENTMASK(SDL_VIDEORESIZE), + SDL_VIDEOEXPOSEMASK = SDL_EVENTMASK(SDL_VIDEOEXPOSE), + SDL_QUITMASK = SDL_EVENTMASK(SDL_QUIT), + SDL_SYSWMEVENTMASK = SDL_EVENTMASK(SDL_SYSWMEVENT) +} SDL_EventMask ; +#define SDL_ALLEVENTS 0xFFFFFFFF + +/* Application visibility event structure */ +typedef struct SDL_ActiveEvent { + Uint8 type; /* SDL_ACTIVEEVENT */ + Uint8 gain; /* Whether given states were gained or lost (1/0) */ + Uint8 state; /* A mask of the focus states */ +} SDL_ActiveEvent; + +/* Keyboard event structure */ +typedef struct SDL_KeyboardEvent { + Uint8 type; /* SDL_KEYDOWN or SDL_KEYUP */ + Uint8 which; /* The keyboard device index */ + Uint8 state; /* SDL_PRESSED or SDL_RELEASED */ + SDL_keysym keysym; +} SDL_KeyboardEvent; + +/* Mouse motion event structure */ +typedef struct SDL_MouseMotionEvent { + Uint8 type; /* SDL_MOUSEMOTION */ + Uint8 which; /* The mouse device index */ + Uint8 state; /* The current button state */ + Uint16 x, y; /* The X/Y coordinates of the mouse */ + Sint16 xrel; /* The relative motion in the X direction */ + Sint16 yrel; /* The relative motion in the Y direction */ +} SDL_MouseMotionEvent; + +/* Mouse button event structure */ +typedef struct SDL_MouseButtonEvent { + Uint8 type; /* SDL_MOUSEBUTTONDOWN or SDL_MOUSEBUTTONUP */ + Uint8 which; /* The mouse device index */ + Uint8 button; /* The mouse button index */ + Uint8 state; /* SDL_PRESSED or SDL_RELEASED */ + Uint16 x, y; /* The X/Y coordinates of the mouse at press time */ +} SDL_MouseButtonEvent; + +/* Joystick axis motion event structure */ +typedef struct SDL_JoyAxisEvent { + Uint8 type; /* SDL_JOYAXISMOTION */ + Uint8 which; /* The joystick device index */ + Uint8 axis; /* The joystick axis index */ + Sint16 value; /* The axis value (range: -32768 to 32767) */ +} SDL_JoyAxisEvent; + +/* Joystick trackball motion event structure */ +typedef struct SDL_JoyBallEvent { + Uint8 type; /* SDL_JOYBALLMOTION */ + Uint8 which; /* The joystick device index */ + Uint8 ball; /* The joystick trackball index */ + Sint16 xrel; /* The relative motion in the X direction */ + Sint16 yrel; /* The relative motion in the Y direction */ +} SDL_JoyBallEvent; + +/* Joystick hat position change event structure */ +typedef struct SDL_JoyHatEvent { + Uint8 type; /* SDL_JOYHATMOTION */ + Uint8 which; /* The joystick device index */ + Uint8 hat; /* The joystick hat index */ + Uint8 value; /* The hat position value: + SDL_HAT_LEFTUP SDL_HAT_UP SDL_HAT_RIGHTUP + SDL_HAT_LEFT SDL_HAT_CENTERED SDL_HAT_RIGHT + SDL_HAT_LEFTDOWN SDL_HAT_DOWN SDL_HAT_RIGHTDOWN + Note that zero means the POV is centered. + */ +} SDL_JoyHatEvent; + +/* Joystick button event structure */ +typedef struct SDL_JoyButtonEvent { + Uint8 type; /* SDL_JOYBUTTONDOWN or SDL_JOYBUTTONUP */ + Uint8 which; /* The joystick device index */ + Uint8 button; /* The joystick button index */ + Uint8 state; /* SDL_PRESSED or SDL_RELEASED */ +} SDL_JoyButtonEvent; + +/* The "window resized" event + When you get this event, you are responsible for setting a new video + mode with the new width and height. + */ +typedef struct SDL_ResizeEvent { + Uint8 type; /* SDL_VIDEORESIZE */ + int w; /* New width */ + int h; /* New height */ +} SDL_ResizeEvent; + +/* The "screen redraw" event */ +typedef struct SDL_ExposeEvent { + Uint8 type; /* SDL_VIDEOEXPOSE */ +} SDL_ExposeEvent; + +/* The "quit requested" event */ +typedef struct SDL_QuitEvent { + Uint8 type; /* SDL_QUIT */ +} SDL_QuitEvent; + +/* A user-defined event type */ +typedef struct SDL_UserEvent { + Uint8 type; /* SDL_USEREVENT through SDL_NUMEVENTS-1 */ + int code; /* User defined event code */ + void *data1; /* User defined data pointer */ + void *data2; /* User defined data pointer */ +} SDL_UserEvent; + +/* If you want to use this event, you should include SDL_syswm.h */ +struct SDL_SysWMmsg; +typedef struct SDL_SysWMmsg SDL_SysWMmsg; +typedef struct SDL_SysWMEvent { + Uint8 type; + SDL_SysWMmsg *msg; +} SDL_SysWMEvent; + +/* General event structure */ +typedef union SDL_Event { + Uint8 type; + SDL_ActiveEvent active; + SDL_KeyboardEvent key; + SDL_MouseMotionEvent motion; + SDL_MouseButtonEvent button; + SDL_JoyAxisEvent jaxis; + SDL_JoyBallEvent jball; + SDL_JoyHatEvent jhat; + SDL_JoyButtonEvent jbutton; + SDL_ResizeEvent resize; + SDL_ExposeEvent expose; + SDL_QuitEvent quit; + SDL_UserEvent user; + SDL_SysWMEvent syswm; +} SDL_Event; + + +/* Function prototypes */ + +/* Pumps the event loop, gathering events from the input devices. + This function updates the event queue and internal input device state. + This should only be run in the thread that sets the video mode. +*/ +extern DECLSPEC void SDLCALL SDL_PumpEvents(void); + +/* Checks the event queue for messages and optionally returns them. + If 'action' is SDL_ADDEVENT, up to 'numevents' events will be added to + the back of the event queue. + If 'action' is SDL_PEEKEVENT, up to 'numevents' events at the front + of the event queue, matching 'mask', will be returned and will not + be removed from the queue. + If 'action' is SDL_GETEVENT, up to 'numevents' events at the front + of the event queue, matching 'mask', will be returned and will be + removed from the queue. + This function returns the number of events actually stored, or -1 + if there was an error. This function is thread-safe. +*/ +typedef enum { + SDL_ADDEVENT, + SDL_PEEKEVENT, + SDL_GETEVENT +} SDL_eventaction; +/* */ +extern DECLSPEC int SDLCALL SDL_PeepEvents(SDL_Event *events, int numevents, + SDL_eventaction action, Uint32 mask); + +/* Polls for currently pending events, and returns 1 if there are any pending + events, or 0 if there are none available. If 'event' is not NULL, the next + event is removed from the queue and stored in that area. + */ +extern DECLSPEC int SDLCALL SDL_PollEvent(SDL_Event *event); + +/* Waits indefinitely for the next available event, returning 1, or 0 if there + was an error while waiting for events. If 'event' is not NULL, the next + event is removed from the queue and stored in that area. + */ +extern DECLSPEC int SDLCALL SDL_WaitEvent(SDL_Event *event); + +/* Add an event to the event queue. + This function returns 0 on success, or -1 if the event queue was full + or there was some other error. + */ +extern DECLSPEC int SDLCALL SDL_PushEvent(SDL_Event *event); + +/* + This function sets up a filter to process all events before they + change internal state and are posted to the internal event queue. + + The filter is protypted as: +*/ +typedef int (SDLCALL *SDL_EventFilter)(const SDL_Event *event); +/* + If the filter returns 1, then the event will be added to the internal queue. + If it returns 0, then the event will be dropped from the queue, but the + internal state will still be updated. This allows selective filtering of + dynamically arriving events. + + WARNING: Be very careful of what you do in the event filter function, as + it may run in a different thread! + + There is one caveat when dealing with the SDL_QUITEVENT event type. The + event filter is only called when the window manager desires to close the + application window. If the event filter returns 1, then the window will + be closed, otherwise the window will remain open if possible. + If the quit event is generated by an interrupt signal, it will bypass the + internal queue and be delivered to the application at the next event poll. +*/ +extern DECLSPEC void SDLCALL SDL_SetEventFilter(SDL_EventFilter filter); + +/* + Return the current event filter - can be used to "chain" filters. + If there is no event filter set, this function returns NULL. +*/ +extern DECLSPEC SDL_EventFilter SDLCALL SDL_GetEventFilter(void); + +/* + This function allows you to set the state of processing certain events. + If 'state' is set to SDL_IGNORE, that event will be automatically dropped + from the event queue and will not event be filtered. + If 'state' is set to SDL_ENABLE, that event will be processed normally. + If 'state' is set to SDL_QUERY, SDL_EventState() will return the + current processing state of the specified event. +*/ +#define SDL_QUERY -1 +#define SDL_IGNORE 0 +#define SDL_DISABLE 0 +#define SDL_ENABLE 1 +extern DECLSPEC Uint8 SDLCALL SDL_EventState(Uint8 type, int state); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* _SDL_events_h */ diff --git a/src/win32/dependencies/sdl/SDL_getenv.h b/src/win32/dependencies/sdl/SDL_getenv.h new file mode 100644 index 00000000..853b9ce4 --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_getenv.h @@ -0,0 +1,24 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* DEPRECATED */ +#include "SDL_stdinc.h" diff --git a/src/win32/dependencies/sdl/SDL_joystick.h b/src/win32/dependencies/sdl/SDL_joystick.h new file mode 100644 index 00000000..e4f72f1a --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_joystick.h @@ -0,0 +1,167 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* Include file for SDL joystick event handling */ + +#ifndef _SDL_joystick_h +#define _SDL_joystick_h + +#include "SDL_stdinc.h" +#include "SDL_error.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* In order to use these functions, SDL_Init() must have been called + with the SDL_INIT_JOYSTICK flag. This causes SDL to scan the system + for joysticks, and load appropriate drivers. +*/ + +/* The joystick structure used to identify an SDL joystick */ +struct _SDL_Joystick; +typedef struct _SDL_Joystick SDL_Joystick; + + +/* Function prototypes */ +/* + * Count the number of joysticks attached to the system + */ +extern DECLSPEC int SDLCALL SDL_NumJoysticks(void); + +/* + * Get the implementation dependent name of a joystick. + * This can be called before any joysticks are opened. + * If no name can be found, this function returns NULL. + */ +extern DECLSPEC const char * SDLCALL SDL_JoystickName(int device_index); + +/* + * Open a joystick for use - the index passed as an argument refers to + * the N'th joystick on the system. This index is the value which will + * identify this joystick in future joystick events. + * + * This function returns a joystick identifier, or NULL if an error occurred. + */ +extern DECLSPEC SDL_Joystick * SDLCALL SDL_JoystickOpen(int device_index); + +/* + * Returns 1 if the joystick has been opened, or 0 if it has not. + */ +extern DECLSPEC int SDLCALL SDL_JoystickOpened(int device_index); + +/* + * Get the device index of an opened joystick. + */ +extern DECLSPEC int SDLCALL SDL_JoystickIndex(SDL_Joystick *joystick); + +/* + * Get the number of general axis controls on a joystick + */ +extern DECLSPEC int SDLCALL SDL_JoystickNumAxes(SDL_Joystick *joystick); + +/* + * Get the number of trackballs on a joystick + * Joystick trackballs have only relative motion events associated + * with them and their state cannot be polled. + */ +extern DECLSPEC int SDLCALL SDL_JoystickNumBalls(SDL_Joystick *joystick); + +/* + * Get the number of POV hats on a joystick + */ +extern DECLSPEC int SDLCALL SDL_JoystickNumHats(SDL_Joystick *joystick); + +/* + * Get the number of buttons on a joystick + */ +extern DECLSPEC int SDLCALL SDL_JoystickNumButtons(SDL_Joystick *joystick); + +/* + * Update the current state of the open joysticks. + * This is called automatically by the event loop if any joystick + * events are enabled. + */ +extern DECLSPEC void SDLCALL SDL_JoystickUpdate(void); + +/* + * Enable/disable joystick event polling. + * If joystick events are disabled, you must call SDL_JoystickUpdate() + * yourself and check the state of the joystick when you want joystick + * information. + * The state can be one of SDL_QUERY, SDL_ENABLE or SDL_IGNORE. + */ +extern DECLSPEC int SDLCALL SDL_JoystickEventState(int state); + +/* + * Get the current state of an axis control on a joystick + * The state is a value ranging from -32768 to 32767. + * The axis indices start at index 0. + */ +extern DECLSPEC Sint16 SDLCALL SDL_JoystickGetAxis(SDL_Joystick *joystick, int axis); + +/* + * Get the current state of a POV hat on a joystick + * The return value is one of the following positions: + */ +#define SDL_HAT_CENTERED 0x00 +#define SDL_HAT_UP 0x01 +#define SDL_HAT_RIGHT 0x02 +#define SDL_HAT_DOWN 0x04 +#define SDL_HAT_LEFT 0x08 +#define SDL_HAT_RIGHTUP (SDL_HAT_RIGHT|SDL_HAT_UP) +#define SDL_HAT_RIGHTDOWN (SDL_HAT_RIGHT|SDL_HAT_DOWN) +#define SDL_HAT_LEFTUP (SDL_HAT_LEFT|SDL_HAT_UP) +#define SDL_HAT_LEFTDOWN (SDL_HAT_LEFT|SDL_HAT_DOWN) +/* + * The hat indices start at index 0. + */ +extern DECLSPEC Uint8 SDLCALL SDL_JoystickGetHat(SDL_Joystick *joystick, int hat); + +/* + * Get the ball axis change since the last poll + * This returns 0, or -1 if you passed it invalid parameters. + * The ball indices start at index 0. + */ +extern DECLSPEC int SDLCALL SDL_JoystickGetBall(SDL_Joystick *joystick, int ball, int *dx, int *dy); + +/* + * Get the current state of a button on a joystick + * The button indices start at index 0. + */ +extern DECLSPEC Uint8 SDLCALL SDL_JoystickGetButton(SDL_Joystick *joystick, int button); + +/* + * Close a joystick previously opened with SDL_JoystickOpen() + */ +extern DECLSPEC void SDLCALL SDL_JoystickClose(SDL_Joystick *joystick); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* _SDL_joystick_h */ diff --git a/src/win32/dependencies/sdl/SDL_keyboard.h b/src/win32/dependencies/sdl/SDL_keyboard.h new file mode 100644 index 00000000..1ad7dcaa --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_keyboard.h @@ -0,0 +1,121 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* Include file for SDL keyboard event handling */ + +#ifndef _SDL_keyboard_h +#define _SDL_keyboard_h + +#include "SDL_stdinc.h" +#include "SDL_error.h" +#include "SDL_keysym.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* Keysym structure + - The scancode is hardware dependent, and should not be used by general + applications. If no hardware scancode is available, it will be 0. + + - The 'unicode' translated character is only available when character + translation is enabled by the SDL_EnableUNICODE() API. If non-zero, + this is a UNICODE character corresponding to the keypress. If the + high 9 bits of the character are 0, then this maps to the equivalent + ASCII character: + char ch; + if ( (keysym.unicode & 0xFF80) == 0 ) { + ch = keysym.unicode & 0x7F; + } else { + An international character.. + } + */ +typedef struct SDL_keysym { + Uint8 scancode; /* hardware specific scancode */ + SDLKey sym; /* SDL virtual keysym */ + SDLMod mod; /* current key modifiers */ + Uint16 unicode; /* translated character */ +} SDL_keysym; + +/* This is the mask which refers to all hotkey bindings */ +#define SDL_ALL_HOTKEYS 0xFFFFFFFF + +/* Function prototypes */ +/* + * Enable/Disable UNICODE translation of keyboard input. + * This translation has some overhead, so translation defaults off. + * If 'enable' is 1, translation is enabled. + * If 'enable' is 0, translation is disabled. + * If 'enable' is -1, the translation state is not changed. + * It returns the previous state of keyboard translation. + */ +extern DECLSPEC int SDLCALL SDL_EnableUNICODE(int enable); + +/* + * Enable/Disable keyboard repeat. Keyboard repeat defaults to off. + * 'delay' is the initial delay in ms between the time when a key is + * pressed, and keyboard repeat begins. + * 'interval' is the time in ms between keyboard repeat events. + */ +#define SDL_DEFAULT_REPEAT_DELAY 500 +#define SDL_DEFAULT_REPEAT_INTERVAL 30 +/* + * If 'delay' is set to 0, keyboard repeat is disabled. + */ +extern DECLSPEC int SDLCALL SDL_EnableKeyRepeat(int delay, int interval); +extern DECLSPEC void SDLCALL SDL_GetKeyRepeat(int *delay, int *interval); + +/* + * Get a snapshot of the current state of the keyboard. + * Returns an array of keystates, indexed by the SDLK_* syms. + * Used: + * Uint8 *keystate = SDL_GetKeyState(NULL); + * if ( keystate[SDLK_RETURN] ) ... is pressed. + */ +extern DECLSPEC Uint8 * SDLCALL SDL_GetKeyState(int *numkeys); + +/* + * Get the current key modifier state + */ +extern DECLSPEC SDLMod SDLCALL SDL_GetModState(void); + +/* + * Set the current key modifier state + * This does not change the keyboard state, only the key modifier flags. + */ +extern DECLSPEC void SDLCALL SDL_SetModState(SDLMod modstate); + +/* + * Get the name of an SDL virtual keysym + */ +extern DECLSPEC char * SDLCALL SDL_GetKeyName(SDLKey key); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* _SDL_keyboard_h */ diff --git a/src/win32/dependencies/sdl/SDL_keysym.h b/src/win32/dependencies/sdl/SDL_keysym.h new file mode 100644 index 00000000..ff44a035 --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_keysym.h @@ -0,0 +1,311 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +#ifndef _SDL_keysym_h +#define _SDL_keysym_h + +/* What we really want is a mapping of every raw key on the keyboard. + To support international keyboards, we use the range 0xA1 - 0xFF + as international virtual keycodes. We'll follow in the footsteps of X11... + The names of the keys + */ + +typedef enum { + /* The keyboard syms have been cleverly chosen to map to ASCII */ + SDLK_UNKNOWN = 0, + SDLK_FIRST = 0, + SDLK_BACKSPACE = 8, + SDLK_TAB = 9, + SDLK_CLEAR = 12, + SDLK_RETURN = 13, + SDLK_PAUSE = 19, + SDLK_ESCAPE = 27, + SDLK_SPACE = 32, + SDLK_EXCLAIM = 33, + SDLK_QUOTEDBL = 34, + SDLK_HASH = 35, + SDLK_DOLLAR = 36, + SDLK_AMPERSAND = 38, + SDLK_QUOTE = 39, + SDLK_LEFTPAREN = 40, + SDLK_RIGHTPAREN = 41, + SDLK_ASTERISK = 42, + SDLK_PLUS = 43, + SDLK_COMMA = 44, + SDLK_MINUS = 45, + SDLK_PERIOD = 46, + SDLK_SLASH = 47, + SDLK_0 = 48, + SDLK_1 = 49, + SDLK_2 = 50, + SDLK_3 = 51, + SDLK_4 = 52, + SDLK_5 = 53, + SDLK_6 = 54, + SDLK_7 = 55, + SDLK_8 = 56, + SDLK_9 = 57, + SDLK_COLON = 58, + SDLK_SEMICOLON = 59, + SDLK_LESS = 60, + SDLK_EQUALS = 61, + SDLK_GREATER = 62, + SDLK_QUESTION = 63, + SDLK_AT = 64, + /* + Skip uppercase letters + */ + SDLK_LEFTBRACKET = 91, + SDLK_BACKSLASH = 92, + SDLK_RIGHTBRACKET = 93, + SDLK_CARET = 94, + SDLK_UNDERSCORE = 95, + SDLK_BACKQUOTE = 96, + SDLK_a = 97, + SDLK_b = 98, + SDLK_c = 99, + SDLK_d = 100, + SDLK_e = 101, + SDLK_f = 102, + SDLK_g = 103, + SDLK_h = 104, + SDLK_i = 105, + SDLK_j = 106, + SDLK_k = 107, + SDLK_l = 108, + SDLK_m = 109, + SDLK_n = 110, + SDLK_o = 111, + SDLK_p = 112, + SDLK_q = 113, + SDLK_r = 114, + SDLK_s = 115, + SDLK_t = 116, + SDLK_u = 117, + SDLK_v = 118, + SDLK_w = 119, + SDLK_x = 120, + SDLK_y = 121, + SDLK_z = 122, + SDLK_DELETE = 127, + /* End of ASCII mapped keysyms */ + + /* International keyboard syms */ + SDLK_WORLD_0 = 160, /* 0xA0 */ + SDLK_WORLD_1 = 161, + SDLK_WORLD_2 = 162, + SDLK_WORLD_3 = 163, + SDLK_WORLD_4 = 164, + SDLK_WORLD_5 = 165, + SDLK_WORLD_6 = 166, + SDLK_WORLD_7 = 167, + SDLK_WORLD_8 = 168, + SDLK_WORLD_9 = 169, + SDLK_WORLD_10 = 170, + SDLK_WORLD_11 = 171, + SDLK_WORLD_12 = 172, + SDLK_WORLD_13 = 173, + SDLK_WORLD_14 = 174, + SDLK_WORLD_15 = 175, + SDLK_WORLD_16 = 176, + SDLK_WORLD_17 = 177, + SDLK_WORLD_18 = 178, + SDLK_WORLD_19 = 179, + SDLK_WORLD_20 = 180, + SDLK_WORLD_21 = 181, + SDLK_WORLD_22 = 182, + SDLK_WORLD_23 = 183, + SDLK_WORLD_24 = 184, + SDLK_WORLD_25 = 185, + SDLK_WORLD_26 = 186, + SDLK_WORLD_27 = 187, + SDLK_WORLD_28 = 188, + SDLK_WORLD_29 = 189, + SDLK_WORLD_30 = 190, + SDLK_WORLD_31 = 191, + SDLK_WORLD_32 = 192, + SDLK_WORLD_33 = 193, + SDLK_WORLD_34 = 194, + SDLK_WORLD_35 = 195, + SDLK_WORLD_36 = 196, + SDLK_WORLD_37 = 197, + SDLK_WORLD_38 = 198, + SDLK_WORLD_39 = 199, + SDLK_WORLD_40 = 200, + SDLK_WORLD_41 = 201, + SDLK_WORLD_42 = 202, + SDLK_WORLD_43 = 203, + SDLK_WORLD_44 = 204, + SDLK_WORLD_45 = 205, + SDLK_WORLD_46 = 206, + SDLK_WORLD_47 = 207, + SDLK_WORLD_48 = 208, + SDLK_WORLD_49 = 209, + SDLK_WORLD_50 = 210, + SDLK_WORLD_51 = 211, + SDLK_WORLD_52 = 212, + SDLK_WORLD_53 = 213, + SDLK_WORLD_54 = 214, + SDLK_WORLD_55 = 215, + SDLK_WORLD_56 = 216, + SDLK_WORLD_57 = 217, + SDLK_WORLD_58 = 218, + SDLK_WORLD_59 = 219, + SDLK_WORLD_60 = 220, + SDLK_WORLD_61 = 221, + SDLK_WORLD_62 = 222, + SDLK_WORLD_63 = 223, + SDLK_WORLD_64 = 224, + SDLK_WORLD_65 = 225, + SDLK_WORLD_66 = 226, + SDLK_WORLD_67 = 227, + SDLK_WORLD_68 = 228, + SDLK_WORLD_69 = 229, + SDLK_WORLD_70 = 230, + SDLK_WORLD_71 = 231, + SDLK_WORLD_72 = 232, + SDLK_WORLD_73 = 233, + SDLK_WORLD_74 = 234, + SDLK_WORLD_75 = 235, + SDLK_WORLD_76 = 236, + SDLK_WORLD_77 = 237, + SDLK_WORLD_78 = 238, + SDLK_WORLD_79 = 239, + SDLK_WORLD_80 = 240, + SDLK_WORLD_81 = 241, + SDLK_WORLD_82 = 242, + SDLK_WORLD_83 = 243, + SDLK_WORLD_84 = 244, + SDLK_WORLD_85 = 245, + SDLK_WORLD_86 = 246, + SDLK_WORLD_87 = 247, + SDLK_WORLD_88 = 248, + SDLK_WORLD_89 = 249, + SDLK_WORLD_90 = 250, + SDLK_WORLD_91 = 251, + SDLK_WORLD_92 = 252, + SDLK_WORLD_93 = 253, + SDLK_WORLD_94 = 254, + SDLK_WORLD_95 = 255, /* 0xFF */ + + /* Numeric keypad */ + SDLK_KP0 = 256, + SDLK_KP1 = 257, + SDLK_KP2 = 258, + SDLK_KP3 = 259, + SDLK_KP4 = 260, + SDLK_KP5 = 261, + SDLK_KP6 = 262, + SDLK_KP7 = 263, + SDLK_KP8 = 264, + SDLK_KP9 = 265, + SDLK_KP_PERIOD = 266, + SDLK_KP_DIVIDE = 267, + SDLK_KP_MULTIPLY = 268, + SDLK_KP_MINUS = 269, + SDLK_KP_PLUS = 270, + SDLK_KP_ENTER = 271, + SDLK_KP_EQUALS = 272, + + /* Arrows + Home/End pad */ + SDLK_UP = 273, + SDLK_DOWN = 274, + SDLK_RIGHT = 275, + SDLK_LEFT = 276, + SDLK_INSERT = 277, + SDLK_HOME = 278, + SDLK_END = 279, + SDLK_PAGEUP = 280, + SDLK_PAGEDOWN = 281, + + /* Function keys */ + SDLK_F1 = 282, + SDLK_F2 = 283, + SDLK_F3 = 284, + SDLK_F4 = 285, + SDLK_F5 = 286, + SDLK_F6 = 287, + SDLK_F7 = 288, + SDLK_F8 = 289, + SDLK_F9 = 290, + SDLK_F10 = 291, + SDLK_F11 = 292, + SDLK_F12 = 293, + SDLK_F13 = 294, + SDLK_F14 = 295, + SDLK_F15 = 296, + + /* Key state modifier keys */ + SDLK_NUMLOCK = 300, + SDLK_CAPSLOCK = 301, + SDLK_SCROLLOCK = 302, + SDLK_RSHIFT = 303, + SDLK_LSHIFT = 304, + SDLK_RCTRL = 305, + SDLK_LCTRL = 306, + SDLK_RALT = 307, + SDLK_LALT = 308, + SDLK_RMETA = 309, + SDLK_LMETA = 310, + SDLK_LSUPER = 311, /* Left "Windows" key */ + SDLK_RSUPER = 312, /* Right "Windows" key */ + SDLK_MODE = 313, /* "Alt Gr" key */ + SDLK_COMPOSE = 314, /* Multi-key compose key */ + + /* Miscellaneous function keys */ + SDLK_HELP = 315, + SDLK_PRINT = 316, + SDLK_SYSREQ = 317, + SDLK_BREAK = 318, + SDLK_MENU = 319, + SDLK_POWER = 320, /* Power Macintosh power key */ + SDLK_EURO = 321, /* Some european keyboards */ + SDLK_UNDO = 322, /* Atari keyboard has Undo */ + + /* Add any other keys here */ + + SDLK_LAST +} SDLKey; + +/* Enumeration of valid key mods (possibly OR'd together) */ +typedef enum { + KMOD_NONE = 0x0000, + KMOD_LSHIFT= 0x0001, + KMOD_RSHIFT= 0x0002, + KMOD_LCTRL = 0x0040, + KMOD_RCTRL = 0x0080, + KMOD_LALT = 0x0100, + KMOD_RALT = 0x0200, + KMOD_LMETA = 0x0400, + KMOD_RMETA = 0x0800, + KMOD_NUM = 0x1000, + KMOD_CAPS = 0x2000, + KMOD_MODE = 0x4000, + KMOD_RESERVED = 0x8000 +} SDLMod; + +#define KMOD_CTRL (KMOD_LCTRL|KMOD_RCTRL) +#define KMOD_SHIFT (KMOD_LSHIFT|KMOD_RSHIFT) +#define KMOD_ALT (KMOD_LALT|KMOD_RALT) +#define KMOD_META (KMOD_LMETA|KMOD_RMETA) + +#endif /* _SDL_keysym_h */ diff --git a/src/win32/dependencies/sdl/SDL_loadso.h b/src/win32/dependencies/sdl/SDL_loadso.h new file mode 100644 index 00000000..ce964494 --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_loadso.h @@ -0,0 +1,74 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ +/* System dependent library loading routines */ + +/* Some things to keep in mind: + - These functions only work on C function names. Other languages may + have name mangling and intrinsic language support that varies from + compiler to compiler. + - Make sure you declare your function pointers with the same calling + convention as the actual library function. Your code will crash + mysteriously if you do not do this. + - Avoid namespace collisions. If you load a symbol from the library, + it is not defined whether or not it goes into the global symbol + namespace for the application. If it does and it conflicts with + symbols in your code or other shared libraries, you will not get + the results you expect. :) +*/ + + +#ifndef _SDL_loadso_h +#define _SDL_loadso_h + +#include "SDL_stdinc.h" +#include "SDL_error.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* This function dynamically loads a shared object and returns a pointer + * to the object handle (or NULL if there was an error). + * The 'sofile' parameter is a system dependent name of the object file. + */ +extern DECLSPEC void * SDLCALL SDL_LoadObject(const char *sofile); + +/* Given an object handle, this function looks up the address of the + * named function in the shared object and returns it. This address + * is no longer valid after calling SDL_UnloadObject(). + */ +extern DECLSPEC void * SDLCALL SDL_LoadFunction(void *handle, const char *name); + +/* Unload a shared object from memory */ +extern DECLSPEC void SDLCALL SDL_UnloadObject(void *handle); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* _SDL_loadso_h */ diff --git a/src/win32/dependencies/sdl/SDL_main.h b/src/win32/dependencies/sdl/SDL_main.h new file mode 100644 index 00000000..cf8b728d --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_main.h @@ -0,0 +1,98 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +#ifndef _SDL_main_h +#define _SDL_main_h + +#include "SDL_stdinc.h" + +/* Redefine main() on Win32 and MacOS so that it is called by winmain.c */ + +#if defined(__WIN32__) || \ + (defined(__MWERKS__) && !defined(__BEOS__)) || \ + defined(__MACOS__) || defined(__MACOSX__) || \ + defined(__SYMBIAN32__) || defined(QWS) + +#ifdef __cplusplus +#define C_LINKAGE "C" +#else +#define C_LINKAGE +#endif /* __cplusplus */ + +/* The application's main() function must be called with C linkage, + and should be declared like this: +#ifdef __cplusplus +extern "C" +#endif + int main(int argc, char *argv[]) + { + } + */ +#define main SDL_main + +/* The prototype for the application's main() function */ +extern C_LINKAGE int SDL_main(int argc, char *argv[]); + + +/* From the SDL library code -- needed for registering the app on Win32 */ +#ifdef __WIN32__ + +#include "begin_code.h" +#ifdef __cplusplus +extern "C" { +#endif + +/* This should be called from your WinMain() function, if any */ +extern DECLSPEC void SDLCALL SDL_SetModuleHandle(void *hInst); +/* This can also be called, but is no longer necessary */ +extern DECLSPEC int SDLCALL SDL_RegisterApp(char *name, Uint32 style, void *hInst); +/* This can also be called, but is no longer necessary (SDL_Quit calls it) */ +extern DECLSPEC void SDLCALL SDL_UnregisterApp(void); +#ifdef __cplusplus +} +#endif +#include "close_code.h" +#endif + +/* From the SDL library code -- needed for registering QuickDraw on MacOS */ +#if defined(__MACOS__) + +#include "begin_code.h" +#ifdef __cplusplus +extern "C" { +#endif + +/* Forward declaration so we don't need to include QuickDraw.h */ +struct QDGlobals; + +/* This should be called from your main() function, if any */ +extern DECLSPEC void SDLCALL SDL_InitQuickDraw(struct QDGlobals *the_qd); + +#ifdef __cplusplus +} +#endif +#include "close_code.h" +#endif + +#endif /* Need to redefine main()? */ + +#endif /* _SDL_main_h */ diff --git a/src/win32/dependencies/sdl/SDL_mouse.h b/src/win32/dependencies/sdl/SDL_mouse.h new file mode 100644 index 00000000..c2364d85 --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_mouse.h @@ -0,0 +1,136 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* Include file for SDL mouse event handling */ + +#ifndef _SDL_mouse_h +#define _SDL_mouse_h + +#include "SDL_stdinc.h" +#include "SDL_error.h" +#include "SDL_video.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +typedef struct WMcursor WMcursor; /* Implementation dependent */ +typedef struct SDL_Cursor { + SDL_Rect area; /* The area of the mouse cursor */ + Sint16 hot_x, hot_y; /* The "tip" of the cursor */ + Uint8 *data; /* B/W cursor data */ + Uint8 *mask; /* B/W cursor mask */ + Uint8 *save[2]; /* Place to save cursor area */ + WMcursor *wm_cursor; /* Window-manager cursor */ +} SDL_Cursor; + +/* Function prototypes */ +/* + * Retrieve the current state of the mouse. + * The current button state is returned as a button bitmask, which can + * be tested using the SDL_BUTTON(X) macros, and x and y are set to the + * current mouse cursor position. You can pass NULL for either x or y. + */ +extern DECLSPEC Uint8 SDLCALL SDL_GetMouseState(int *x, int *y); + +/* + * Retrieve the current state of the mouse. + * The current button state is returned as a button bitmask, which can + * be tested using the SDL_BUTTON(X) macros, and x and y are set to the + * mouse deltas since the last call to SDL_GetRelativeMouseState(). + */ +extern DECLSPEC Uint8 SDLCALL SDL_GetRelativeMouseState(int *x, int *y); + +/* + * Set the position of the mouse cursor (generates a mouse motion event) + */ +extern DECLSPEC void SDLCALL SDL_WarpMouse(Uint16 x, Uint16 y); + +/* + * Create a cursor using the specified data and mask (in MSB format). + * The cursor width must be a multiple of 8 bits. + * + * The cursor is created in black and white according to the following: + * data mask resulting pixel on screen + * 0 1 White + * 1 1 Black + * 0 0 Transparent + * 1 0 Inverted color if possible, black if not. + * + * Cursors created with this function must be freed with SDL_FreeCursor(). + */ +extern DECLSPEC SDL_Cursor * SDLCALL SDL_CreateCursor + (Uint8 *data, Uint8 *mask, int w, int h, int hot_x, int hot_y); + +/* + * Set the currently active cursor to the specified one. + * If the cursor is currently visible, the change will be immediately + * represented on the display. + */ +extern DECLSPEC void SDLCALL SDL_SetCursor(SDL_Cursor *cursor); + +/* + * Returns the currently active cursor. + */ +extern DECLSPEC SDL_Cursor * SDLCALL SDL_GetCursor(void); + +/* + * Deallocates a cursor created with SDL_CreateCursor(). + */ +extern DECLSPEC void SDLCALL SDL_FreeCursor(SDL_Cursor *cursor); + +/* + * Toggle whether or not the cursor is shown on the screen. + * The cursor start off displayed, but can be turned off. + * SDL_ShowCursor() returns 1 if the cursor was being displayed + * before the call, or 0 if it was not. You can query the current + * state by passing a 'toggle' value of -1. + */ +extern DECLSPEC int SDLCALL SDL_ShowCursor(int toggle); + +/* Used as a mask when testing buttons in buttonstate + Button 1: Left mouse button + Button 2: Middle mouse button + Button 3: Right mouse button + Button 4: Mouse wheel up (may also be a real button) + Button 5: Mouse wheel down (may also be a real button) + */ +#define SDL_BUTTON(X) (1 << ((X)-1)) +#define SDL_BUTTON_LEFT 1 +#define SDL_BUTTON_MIDDLE 2 +#define SDL_BUTTON_RIGHT 3 +#define SDL_BUTTON_WHEELUP 4 +#define SDL_BUTTON_WHEELDOWN 5 +#define SDL_BUTTON_LMASK SDL_BUTTON(SDL_BUTTON_LEFT) +#define SDL_BUTTON_MMASK SDL_BUTTON(SDL_BUTTON_MIDDLE) +#define SDL_BUTTON_RMASK SDL_BUTTON(SDL_BUTTON_RIGHT) + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* _SDL_mouse_h */ diff --git a/src/win32/dependencies/sdl/SDL_mutex.h b/src/win32/dependencies/sdl/SDL_mutex.h new file mode 100644 index 00000000..00165281 --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_mutex.h @@ -0,0 +1,162 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +#ifndef _SDL_mutex_h +#define _SDL_mutex_h + +/* Functions to provide thread synchronization primitives + + These are independent of the other SDL routines. +*/ + +#include "SDL_stdinc.h" +#include "SDL_error.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* Synchronization functions which can time out return this value + if they time out. +*/ +#define SDL_MUTEX_TIMEDOUT 1 + +/* This is the timeout value which corresponds to never time out */ +#define SDL_MUTEX_MAXWAIT (~(Uint32)0) + + +/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ +/* Mutex functions */ +/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ + +/* The SDL mutex structure, defined in SDL_mutex.c */ +struct SDL_mutex; +typedef struct SDL_mutex SDL_mutex; + +/* Create a mutex, initialized unlocked */ +extern DECLSPEC SDL_mutex * SDLCALL SDL_CreateMutex(void); + +/* Lock the mutex (Returns 0, or -1 on error) */ +#define SDL_LockMutex(m) SDL_mutexP(m) +extern DECLSPEC int SDLCALL SDL_mutexP(SDL_mutex *mutex); + +/* Unlock the mutex (Returns 0, or -1 on error) + It is an error to unlock a mutex that has not been locked by + the current thread, and doing so results in undefined behavior. + */ +#define SDL_UnlockMutex(m) SDL_mutexV(m) +extern DECLSPEC int SDLCALL SDL_mutexV(SDL_mutex *mutex); + +/* Destroy a mutex */ +extern DECLSPEC void SDLCALL SDL_DestroyMutex(SDL_mutex *mutex); + + +/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ +/* Semaphore functions */ +/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ + +/* The SDL semaphore structure, defined in SDL_sem.c */ +struct SDL_semaphore; +typedef struct SDL_semaphore SDL_sem; + +/* Create a semaphore, initialized with value, returns NULL on failure. */ +extern DECLSPEC SDL_sem * SDLCALL SDL_CreateSemaphore(Uint32 initial_value); + +/* Destroy a semaphore */ +extern DECLSPEC void SDLCALL SDL_DestroySemaphore(SDL_sem *sem); + +/* This function suspends the calling thread until the semaphore pointed + * to by sem has a positive count. It then atomically decreases the semaphore + * count. + */ +extern DECLSPEC int SDLCALL SDL_SemWait(SDL_sem *sem); + +/* Non-blocking variant of SDL_SemWait(), returns 0 if the wait succeeds, + SDL_MUTEX_TIMEDOUT if the wait would block, and -1 on error. +*/ +extern DECLSPEC int SDLCALL SDL_SemTryWait(SDL_sem *sem); + +/* Variant of SDL_SemWait() with a timeout in milliseconds, returns 0 if + the wait succeeds, SDL_MUTEX_TIMEDOUT if the wait does not succeed in + the allotted time, and -1 on error. + On some platforms this function is implemented by looping with a delay + of 1 ms, and so should be avoided if possible. +*/ +extern DECLSPEC int SDLCALL SDL_SemWaitTimeout(SDL_sem *sem, Uint32 ms); + +/* Atomically increases the semaphore's count (not blocking), returns 0, + or -1 on error. + */ +extern DECLSPEC int SDLCALL SDL_SemPost(SDL_sem *sem); + +/* Returns the current count of the semaphore */ +extern DECLSPEC Uint32 SDLCALL SDL_SemValue(SDL_sem *sem); + + +/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ +/* Condition variable functions */ +/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ + +/* The SDL condition variable structure, defined in SDL_cond.c */ +struct SDL_cond; +typedef struct SDL_cond SDL_cond; + +/* Create a condition variable */ +extern DECLSPEC SDL_cond * SDLCALL SDL_CreateCond(void); + +/* Destroy a condition variable */ +extern DECLSPEC void SDLCALL SDL_DestroyCond(SDL_cond *cond); + +/* Restart one of the threads that are waiting on the condition variable, + returns 0 or -1 on error. + */ +extern DECLSPEC int SDLCALL SDL_CondSignal(SDL_cond *cond); + +/* Restart all threads that are waiting on the condition variable, + returns 0 or -1 on error. + */ +extern DECLSPEC int SDLCALL SDL_CondBroadcast(SDL_cond *cond); + +/* Wait on the condition variable, unlocking the provided mutex. + The mutex must be locked before entering this function! + The mutex is re-locked once the condition variable is signaled. + Returns 0 when it is signaled, or -1 on error. + */ +extern DECLSPEC int SDLCALL SDL_CondWait(SDL_cond *cond, SDL_mutex *mut); + +/* Waits for at most 'ms' milliseconds, and returns 0 if the condition + variable is signaled, SDL_MUTEX_TIMEDOUT if the condition is not + signaled in the allotted time, and -1 on error. + On some platforms this function is implemented by looping with a delay + of 1 ms, and so should be avoided if possible. +*/ +extern DECLSPEC int SDLCALL SDL_CondWaitTimeout(SDL_cond *cond, SDL_mutex *mutex, Uint32 ms); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* _SDL_mutex_h */ diff --git a/src/win32/dependencies/sdl/SDL_name.h b/src/win32/dependencies/sdl/SDL_name.h new file mode 100644 index 00000000..511619af --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_name.h @@ -0,0 +1,11 @@ + +#ifndef _SDLname_h_ +#define _SDLname_h_ + +#if defined(__STDC__) || defined(__cplusplus) +#define NeedFunctionPrototypes 1 +#endif + +#define SDL_NAME(X) SDL_##X + +#endif /* _SDLname_h_ */ diff --git a/src/win32/dependencies/sdl/SDL_opengl.h b/src/win32/dependencies/sdl/SDL_opengl.h new file mode 100644 index 00000000..36c0a309 --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_opengl.h @@ -0,0 +1,6551 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* This is a simple file to encapsulate the OpenGL API headers */ + +#include "SDL_config.h" + +#ifdef __WIN32__ +#define WIN32_LEAN_AND_MEAN +#ifndef NOMINMAX +#define NOMINMAX /* Don't defined min() and max() */ +#endif +#include +#endif +#ifndef NO_SDL_GLEXT +#define __glext_h_ /* Don't let gl.h include glext.h */ +#endif +#if defined(__MACOSX__) +#include /* Header File For The OpenGL Library */ +#include /* Header File For The GLU Library */ +#elif defined(__MACOS__) +#include /* Header File For The OpenGL Library */ +#include /* Header File For The GLU Library */ +#else +#include /* Header File For The OpenGL Library */ +#include /* Header File For The GLU Library */ +#endif +#ifndef NO_SDL_GLEXT +#undef __glext_h_ +#endif + +/* This file taken from "GLext.h" from the Jeff Molofee OpenGL tutorials. + It is included here because glext.h is not available on some systems. + If you don't want this version included, simply define "NO_SDL_GLEXT" + */ +#ifndef NO_SDL_GLEXT +#if !defined(__glext_h_) && !defined(GL_GLEXT_LEGACY) +#define __glext_h_ + +#ifdef __cplusplus +extern "C" { +#endif + +/* +** License Applicability. Except to the extent portions of this file are +** made subject to an alternative license as permitted in the SGI Free +** Software License B, Version 1.1 (the "License"), the contents of this +** file are subject only to the provisions of the License. You may not use +** this file except in compliance with the License. You may obtain a copy +** of the License at Silicon Graphics, Inc., attn: Legal Services, 1600 +** Amphitheatre Parkway, Mountain View, CA 94043-1351, or at: +** +** http://oss.sgi.com/projects/FreeB +** +** Note that, as provided in the License, the Software is distributed on an +** "AS IS" basis, with ALL EXPRESS AND IMPLIED WARRANTIES AND CONDITIONS +** DISCLAIMED, INCLUDING, WITHOUT LIMITATION, ANY IMPLIED WARRANTIES AND +** CONDITIONS OF MERCHANTABILITY, SATISFACTORY QUALITY, FITNESS FOR A +** PARTICULAR PURPOSE, AND NON-INFRINGEMENT. +** +** Original Code. The Original Code is: OpenGL Sample Implementation, +** Version 1.2.1, released January 26, 2000, developed by Silicon Graphics, +** Inc. The Original Code is Copyright (c) 1991-2004 Silicon Graphics, Inc. +** Copyright in any portions created by third parties is as indicated +** elsewhere herein. All Rights Reserved. +** +** Additional Notice Provisions: This software was created using the +** OpenGL(R) version 1.2.1 Sample Implementation published by SGI, but has +** not been independently verified as being compliant with the OpenGL(R) +** version 1.2.1 Specification. +*/ + +#if defined(_WIN32) && !defined(APIENTRY) && !defined(__CYGWIN__) && !defined(__SCITECH_SNAP__) +#define WIN32_LEAN_AND_MEAN 1 +#include +#endif + +#ifndef APIENTRY +#define APIENTRY +#endif +#ifndef APIENTRYP +#define APIENTRYP APIENTRY * +#endif +#ifndef GLAPI +#define GLAPI extern +#endif + +/*************************************************************/ + +/* Header file version number, required by OpenGL ABI for Linux */ +/* glext.h last updated 2005/06/20 */ +/* Current version at http://oss.sgi.com/projects/ogl-sample/registry/ */ +#define GL_GLEXT_VERSION 29 + +#ifndef GL_VERSION_1_2 +#define GL_UNSIGNED_BYTE_3_3_2 0x8032 +#define GL_UNSIGNED_SHORT_4_4_4_4 0x8033 +#define GL_UNSIGNED_SHORT_5_5_5_1 0x8034 +#define GL_UNSIGNED_INT_8_8_8_8 0x8035 +#define GL_UNSIGNED_INT_10_10_10_2 0x8036 +#define GL_RESCALE_NORMAL 0x803A +#define GL_TEXTURE_BINDING_3D 0x806A +#define GL_PACK_SKIP_IMAGES 0x806B +#define GL_PACK_IMAGE_HEIGHT 0x806C +#define GL_UNPACK_SKIP_IMAGES 0x806D +#define GL_UNPACK_IMAGE_HEIGHT 0x806E +#define GL_TEXTURE_3D 0x806F +#define GL_PROXY_TEXTURE_3D 0x8070 +#define GL_TEXTURE_DEPTH 0x8071 +#define GL_TEXTURE_WRAP_R 0x8072 +#define GL_MAX_3D_TEXTURE_SIZE 0x8073 +#define GL_UNSIGNED_BYTE_2_3_3_REV 0x8362 +#define GL_UNSIGNED_SHORT_5_6_5 0x8363 +#define GL_UNSIGNED_SHORT_5_6_5_REV 0x8364 +#define GL_UNSIGNED_SHORT_4_4_4_4_REV 0x8365 +#define GL_UNSIGNED_SHORT_1_5_5_5_REV 0x8366 +#define GL_UNSIGNED_INT_8_8_8_8_REV 0x8367 +#define GL_UNSIGNED_INT_2_10_10_10_REV 0x8368 +#define GL_BGR 0x80E0 +#define GL_BGRA 0x80E1 +#define GL_MAX_ELEMENTS_VERTICES 0x80E8 +#define GL_MAX_ELEMENTS_INDICES 0x80E9 +#define GL_CLAMP_TO_EDGE 0x812F +#define GL_TEXTURE_MIN_LOD 0x813A +#define GL_TEXTURE_MAX_LOD 0x813B +#define GL_TEXTURE_BASE_LEVEL 0x813C +#define GL_TEXTURE_MAX_LEVEL 0x813D +#define GL_LIGHT_MODEL_COLOR_CONTROL 0x81F8 +#define GL_SINGLE_COLOR 0x81F9 +#define GL_SEPARATE_SPECULAR_COLOR 0x81FA +#define GL_SMOOTH_POINT_SIZE_RANGE 0x0B12 +#define GL_SMOOTH_POINT_SIZE_GRANULARITY 0x0B13 +#define GL_SMOOTH_LINE_WIDTH_RANGE 0x0B22 +#define GL_SMOOTH_LINE_WIDTH_GRANULARITY 0x0B23 +#define GL_ALIASED_POINT_SIZE_RANGE 0x846D +#define GL_ALIASED_LINE_WIDTH_RANGE 0x846E +#endif + +#ifndef GL_ARB_imaging +#define GL_CONSTANT_COLOR 0x8001 +#define GL_ONE_MINUS_CONSTANT_COLOR 0x8002 +#define GL_CONSTANT_ALPHA 0x8003 +#define GL_ONE_MINUS_CONSTANT_ALPHA 0x8004 +#define GL_BLEND_COLOR 0x8005 +#define GL_FUNC_ADD 0x8006 +#define GL_MIN 0x8007 +#define GL_MAX 0x8008 +#define GL_BLEND_EQUATION 0x8009 +#define GL_FUNC_SUBTRACT 0x800A +#define GL_FUNC_REVERSE_SUBTRACT 0x800B +#define GL_CONVOLUTION_1D 0x8010 +#define GL_CONVOLUTION_2D 0x8011 +#define GL_SEPARABLE_2D 0x8012 +#define GL_CONVOLUTION_BORDER_MODE 0x8013 +#define GL_CONVOLUTION_FILTER_SCALE 0x8014 +#define GL_CONVOLUTION_FILTER_BIAS 0x8015 +#define GL_REDUCE 0x8016 +#define GL_CONVOLUTION_FORMAT 0x8017 +#define GL_CONVOLUTION_WIDTH 0x8018 +#define GL_CONVOLUTION_HEIGHT 0x8019 +#define GL_MAX_CONVOLUTION_WIDTH 0x801A +#define GL_MAX_CONVOLUTION_HEIGHT 0x801B +#define GL_POST_CONVOLUTION_RED_SCALE 0x801C +#define GL_POST_CONVOLUTION_GREEN_SCALE 0x801D +#define GL_POST_CONVOLUTION_BLUE_SCALE 0x801E +#define GL_POST_CONVOLUTION_ALPHA_SCALE 0x801F +#define GL_POST_CONVOLUTION_RED_BIAS 0x8020 +#define GL_POST_CONVOLUTION_GREEN_BIAS 0x8021 +#define GL_POST_CONVOLUTION_BLUE_BIAS 0x8022 +#define GL_POST_CONVOLUTION_ALPHA_BIAS 0x8023 +#define GL_HISTOGRAM 0x8024 +#define GL_PROXY_HISTOGRAM 0x8025 +#define GL_HISTOGRAM_WIDTH 0x8026 +#define GL_HISTOGRAM_FORMAT 0x8027 +#define GL_HISTOGRAM_RED_SIZE 0x8028 +#define GL_HISTOGRAM_GREEN_SIZE 0x8029 +#define GL_HISTOGRAM_BLUE_SIZE 0x802A +#define GL_HISTOGRAM_ALPHA_SIZE 0x802B +#define GL_HISTOGRAM_LUMINANCE_SIZE 0x802C +#define GL_HISTOGRAM_SINK 0x802D +#define GL_MINMAX 0x802E +#define GL_MINMAX_FORMAT 0x802F +#define GL_MINMAX_SINK 0x8030 +#define GL_TABLE_TOO_LARGE 0x8031 +#define GL_COLOR_MATRIX 0x80B1 +#define GL_COLOR_MATRIX_STACK_DEPTH 0x80B2 +#define GL_MAX_COLOR_MATRIX_STACK_DEPTH 0x80B3 +#define GL_POST_COLOR_MATRIX_RED_SCALE 0x80B4 +#define GL_POST_COLOR_MATRIX_GREEN_SCALE 0x80B5 +#define GL_POST_COLOR_MATRIX_BLUE_SCALE 0x80B6 +#define GL_POST_COLOR_MATRIX_ALPHA_SCALE 0x80B7 +#define GL_POST_COLOR_MATRIX_RED_BIAS 0x80B8 +#define GL_POST_COLOR_MATRIX_GREEN_BIAS 0x80B9 +#define GL_POST_COLOR_MATRIX_BLUE_BIAS 0x80BA +#define GL_POST_COLOR_MATRIX_ALPHA_BIAS 0x80BB +#define GL_COLOR_TABLE 0x80D0 +#define GL_POST_CONVOLUTION_COLOR_TABLE 0x80D1 +#define GL_POST_COLOR_MATRIX_COLOR_TABLE 0x80D2 +#define GL_PROXY_COLOR_TABLE 0x80D3 +#define GL_PROXY_POST_CONVOLUTION_COLOR_TABLE 0x80D4 +#define GL_PROXY_POST_COLOR_MATRIX_COLOR_TABLE 0x80D5 +#define GL_COLOR_TABLE_SCALE 0x80D6 +#define GL_COLOR_TABLE_BIAS 0x80D7 +#define GL_COLOR_TABLE_FORMAT 0x80D8 +#define GL_COLOR_TABLE_WIDTH 0x80D9 +#define GL_COLOR_TABLE_RED_SIZE 0x80DA +#define GL_COLOR_TABLE_GREEN_SIZE 0x80DB +#define GL_COLOR_TABLE_BLUE_SIZE 0x80DC +#define GL_COLOR_TABLE_ALPHA_SIZE 0x80DD +#define GL_COLOR_TABLE_LUMINANCE_SIZE 0x80DE +#define GL_COLOR_TABLE_INTENSITY_SIZE 0x80DF +#define GL_CONSTANT_BORDER 0x8151 +#define GL_REPLICATE_BORDER 0x8153 +#define GL_CONVOLUTION_BORDER_COLOR 0x8154 +#endif + +#ifndef GL_VERSION_1_3 +#define GL_TEXTURE0 0x84C0 +#define GL_TEXTURE1 0x84C1 +#define GL_TEXTURE2 0x84C2 +#define GL_TEXTURE3 0x84C3 +#define GL_TEXTURE4 0x84C4 +#define GL_TEXTURE5 0x84C5 +#define GL_TEXTURE6 0x84C6 +#define GL_TEXTURE7 0x84C7 +#define GL_TEXTURE8 0x84C8 +#define GL_TEXTURE9 0x84C9 +#define GL_TEXTURE10 0x84CA +#define GL_TEXTURE11 0x84CB +#define GL_TEXTURE12 0x84CC +#define GL_TEXTURE13 0x84CD +#define GL_TEXTURE14 0x84CE +#define GL_TEXTURE15 0x84CF +#define GL_TEXTURE16 0x84D0 +#define GL_TEXTURE17 0x84D1 +#define GL_TEXTURE18 0x84D2 +#define GL_TEXTURE19 0x84D3 +#define GL_TEXTURE20 0x84D4 +#define GL_TEXTURE21 0x84D5 +#define GL_TEXTURE22 0x84D6 +#define GL_TEXTURE23 0x84D7 +#define GL_TEXTURE24 0x84D8 +#define GL_TEXTURE25 0x84D9 +#define GL_TEXTURE26 0x84DA +#define GL_TEXTURE27 0x84DB +#define GL_TEXTURE28 0x84DC +#define GL_TEXTURE29 0x84DD +#define GL_TEXTURE30 0x84DE +#define GL_TEXTURE31 0x84DF +#define GL_ACTIVE_TEXTURE 0x84E0 +#define GL_CLIENT_ACTIVE_TEXTURE 0x84E1 +#define GL_MAX_TEXTURE_UNITS 0x84E2 +#define GL_TRANSPOSE_MODELVIEW_MATRIX 0x84E3 +#define GL_TRANSPOSE_PROJECTION_MATRIX 0x84E4 +#define GL_TRANSPOSE_TEXTURE_MATRIX 0x84E5 +#define GL_TRANSPOSE_COLOR_MATRIX 0x84E6 +#define GL_MULTISAMPLE 0x809D +#define GL_SAMPLE_ALPHA_TO_COVERAGE 0x809E +#define GL_SAMPLE_ALPHA_TO_ONE 0x809F +#define GL_SAMPLE_COVERAGE 0x80A0 +#define GL_SAMPLE_BUFFERS 0x80A8 +#define GL_SAMPLES 0x80A9 +#define GL_SAMPLE_COVERAGE_VALUE 0x80AA +#define GL_SAMPLE_COVERAGE_INVERT 0x80AB +#define GL_MULTISAMPLE_BIT 0x20000000 +#define GL_NORMAL_MAP 0x8511 +#define GL_REFLECTION_MAP 0x8512 +#define GL_TEXTURE_CUBE_MAP 0x8513 +#define GL_TEXTURE_BINDING_CUBE_MAP 0x8514 +#define GL_TEXTURE_CUBE_MAP_POSITIVE_X 0x8515 +#define GL_TEXTURE_CUBE_MAP_NEGATIVE_X 0x8516 +#define GL_TEXTURE_CUBE_MAP_POSITIVE_Y 0x8517 +#define GL_TEXTURE_CUBE_MAP_NEGATIVE_Y 0x8518 +#define GL_TEXTURE_CUBE_MAP_POSITIVE_Z 0x8519 +#define GL_TEXTURE_CUBE_MAP_NEGATIVE_Z 0x851A +#define GL_PROXY_TEXTURE_CUBE_MAP 0x851B +#define GL_MAX_CUBE_MAP_TEXTURE_SIZE 0x851C +#define GL_COMPRESSED_ALPHA 0x84E9 +#define GL_COMPRESSED_LUMINANCE 0x84EA +#define GL_COMPRESSED_LUMINANCE_ALPHA 0x84EB +#define GL_COMPRESSED_INTENSITY 0x84EC +#define GL_COMPRESSED_RGB 0x84ED +#define GL_COMPRESSED_RGBA 0x84EE +#define GL_TEXTURE_COMPRESSION_HINT 0x84EF +#define GL_TEXTURE_COMPRESSED_IMAGE_SIZE 0x86A0 +#define GL_TEXTURE_COMPRESSED 0x86A1 +#define GL_NUM_COMPRESSED_TEXTURE_FORMATS 0x86A2 +#define GL_COMPRESSED_TEXTURE_FORMATS 0x86A3 +#define GL_CLAMP_TO_BORDER 0x812D +#define GL_COMBINE 0x8570 +#define GL_COMBINE_RGB 0x8571 +#define GL_COMBINE_ALPHA 0x8572 +#define GL_SOURCE0_RGB 0x8580 +#define GL_SOURCE1_RGB 0x8581 +#define GL_SOURCE2_RGB 0x8582 +#define GL_SOURCE0_ALPHA 0x8588 +#define GL_SOURCE1_ALPHA 0x8589 +#define GL_SOURCE2_ALPHA 0x858A +#define GL_OPERAND0_RGB 0x8590 +#define GL_OPERAND1_RGB 0x8591 +#define GL_OPERAND2_RGB 0x8592 +#define GL_OPERAND0_ALPHA 0x8598 +#define GL_OPERAND1_ALPHA 0x8599 +#define GL_OPERAND2_ALPHA 0x859A +#define GL_RGB_SCALE 0x8573 +#define GL_ADD_SIGNED 0x8574 +#define GL_INTERPOLATE 0x8575 +#define GL_SUBTRACT 0x84E7 +#define GL_CONSTANT 0x8576 +#define GL_PRIMARY_COLOR 0x8577 +#define GL_PREVIOUS 0x8578 +#define GL_DOT3_RGB 0x86AE +#define GL_DOT3_RGBA 0x86AF +#endif + +#ifndef GL_VERSION_1_4 +#define GL_BLEND_DST_RGB 0x80C8 +#define GL_BLEND_SRC_RGB 0x80C9 +#define GL_BLEND_DST_ALPHA 0x80CA +#define GL_BLEND_SRC_ALPHA 0x80CB +#define GL_POINT_SIZE_MIN 0x8126 +#define GL_POINT_SIZE_MAX 0x8127 +#define GL_POINT_FADE_THRESHOLD_SIZE 0x8128 +#define GL_POINT_DISTANCE_ATTENUATION 0x8129 +#define GL_GENERATE_MIPMAP 0x8191 +#define GL_GENERATE_MIPMAP_HINT 0x8192 +#define GL_DEPTH_COMPONENT16 0x81A5 +#define GL_DEPTH_COMPONENT24 0x81A6 +#define GL_DEPTH_COMPONENT32 0x81A7 +#define GL_MIRRORED_REPEAT 0x8370 +#define GL_FOG_COORDINATE_SOURCE 0x8450 +#define GL_FOG_COORDINATE 0x8451 +#define GL_FRAGMENT_DEPTH 0x8452 +#define GL_CURRENT_FOG_COORDINATE 0x8453 +#define GL_FOG_COORDINATE_ARRAY_TYPE 0x8454 +#define GL_FOG_COORDINATE_ARRAY_STRIDE 0x8455 +#define GL_FOG_COORDINATE_ARRAY_POINTER 0x8456 +#define GL_FOG_COORDINATE_ARRAY 0x8457 +#define GL_COLOR_SUM 0x8458 +#define GL_CURRENT_SECONDARY_COLOR 0x8459 +#define GL_SECONDARY_COLOR_ARRAY_SIZE 0x845A +#define GL_SECONDARY_COLOR_ARRAY_TYPE 0x845B +#define GL_SECONDARY_COLOR_ARRAY_STRIDE 0x845C +#define GL_SECONDARY_COLOR_ARRAY_POINTER 0x845D +#define GL_SECONDARY_COLOR_ARRAY 0x845E +#define GL_MAX_TEXTURE_LOD_BIAS 0x84FD +#define GL_TEXTURE_FILTER_CONTROL 0x8500 +#define GL_TEXTURE_LOD_BIAS 0x8501 +#define GL_INCR_WRAP 0x8507 +#define GL_DECR_WRAP 0x8508 +#define GL_TEXTURE_DEPTH_SIZE 0x884A +#define GL_DEPTH_TEXTURE_MODE 0x884B +#define GL_TEXTURE_COMPARE_MODE 0x884C +#define GL_TEXTURE_COMPARE_FUNC 0x884D +#define GL_COMPARE_R_TO_TEXTURE 0x884E +#endif + +#ifndef GL_VERSION_1_5 +#define GL_BUFFER_SIZE 0x8764 +#define GL_BUFFER_USAGE 0x8765 +#define GL_QUERY_COUNTER_BITS 0x8864 +#define GL_CURRENT_QUERY 0x8865 +#define GL_QUERY_RESULT 0x8866 +#define GL_QUERY_RESULT_AVAILABLE 0x8867 +#define GL_ARRAY_BUFFER 0x8892 +#define GL_ELEMENT_ARRAY_BUFFER 0x8893 +#define GL_ARRAY_BUFFER_BINDING 0x8894 +#define GL_ELEMENT_ARRAY_BUFFER_BINDING 0x8895 +#define GL_VERTEX_ARRAY_BUFFER_BINDING 0x8896 +#define GL_NORMAL_ARRAY_BUFFER_BINDING 0x8897 +#define GL_COLOR_ARRAY_BUFFER_BINDING 0x8898 +#define GL_INDEX_ARRAY_BUFFER_BINDING 0x8899 +#define GL_TEXTURE_COORD_ARRAY_BUFFER_BINDING 0x889A +#define GL_EDGE_FLAG_ARRAY_BUFFER_BINDING 0x889B +#define GL_SECONDARY_COLOR_ARRAY_BUFFER_BINDING 0x889C +#define GL_FOG_COORDINATE_ARRAY_BUFFER_BINDING 0x889D +#define GL_WEIGHT_ARRAY_BUFFER_BINDING 0x889E +#define GL_VERTEX_ATTRIB_ARRAY_BUFFER_BINDING 0x889F +#define GL_READ_ONLY 0x88B8 +#define GL_WRITE_ONLY 0x88B9 +#define GL_READ_WRITE 0x88BA +#define GL_BUFFER_ACCESS 0x88BB +#define GL_BUFFER_MAPPED 0x88BC +#define GL_BUFFER_MAP_POINTER 0x88BD +#define GL_STREAM_DRAW 0x88E0 +#define GL_STREAM_READ 0x88E1 +#define GL_STREAM_COPY 0x88E2 +#define GL_STATIC_DRAW 0x88E4 +#define GL_STATIC_READ 0x88E5 +#define GL_STATIC_COPY 0x88E6 +#define GL_DYNAMIC_DRAW 0x88E8 +#define GL_DYNAMIC_READ 0x88E9 +#define GL_DYNAMIC_COPY 0x88EA +#define GL_SAMPLES_PASSED 0x8914 +#define GL_FOG_COORD_SRC GL_FOG_COORDINATE_SOURCE +#define GL_FOG_COORD GL_FOG_COORDINATE +#define GL_CURRENT_FOG_COORD GL_CURRENT_FOG_COORDINATE +#define GL_FOG_COORD_ARRAY_TYPE GL_FOG_COORDINATE_ARRAY_TYPE +#define GL_FOG_COORD_ARRAY_STRIDE GL_FOG_COORDINATE_ARRAY_STRIDE +#define GL_FOG_COORD_ARRAY_POINTER GL_FOG_COORDINATE_ARRAY_POINTER +#define GL_FOG_COORD_ARRAY GL_FOG_COORDINATE_ARRAY +#define GL_FOG_COORD_ARRAY_BUFFER_BINDING GL_FOG_COORDINATE_ARRAY_BUFFER_BINDING +#define GL_SRC0_RGB GL_SOURCE0_RGB +#define GL_SRC1_RGB GL_SOURCE1_RGB +#define GL_SRC2_RGB GL_SOURCE2_RGB +#define GL_SRC0_ALPHA GL_SOURCE0_ALPHA +#define GL_SRC1_ALPHA GL_SOURCE1_ALPHA +#define GL_SRC2_ALPHA GL_SOURCE2_ALPHA +#endif + +#ifndef GL_VERSION_2_0 +#define GL_BLEND_EQUATION_RGB GL_BLEND_EQUATION +#define GL_VERTEX_ATTRIB_ARRAY_ENABLED 0x8622 +#define GL_VERTEX_ATTRIB_ARRAY_SIZE 0x8623 +#define GL_VERTEX_ATTRIB_ARRAY_STRIDE 0x8624 +#define GL_VERTEX_ATTRIB_ARRAY_TYPE 0x8625 +#define GL_CURRENT_VERTEX_ATTRIB 0x8626 +#define GL_VERTEX_PROGRAM_POINT_SIZE 0x8642 +#define GL_VERTEX_PROGRAM_TWO_SIDE 0x8643 +#define GL_VERTEX_ATTRIB_ARRAY_POINTER 0x8645 +#define GL_STENCIL_BACK_FUNC 0x8800 +#define GL_STENCIL_BACK_FAIL 0x8801 +#define GL_STENCIL_BACK_PASS_DEPTH_FAIL 0x8802 +#define GL_STENCIL_BACK_PASS_DEPTH_PASS 0x8803 +#define GL_MAX_DRAW_BUFFERS 0x8824 +#define GL_DRAW_BUFFER0 0x8825 +#define GL_DRAW_BUFFER1 0x8826 +#define GL_DRAW_BUFFER2 0x8827 +#define GL_DRAW_BUFFER3 0x8828 +#define GL_DRAW_BUFFER4 0x8829 +#define GL_DRAW_BUFFER5 0x882A +#define GL_DRAW_BUFFER6 0x882B +#define GL_DRAW_BUFFER7 0x882C +#define GL_DRAW_BUFFER8 0x882D +#define GL_DRAW_BUFFER9 0x882E +#define GL_DRAW_BUFFER10 0x882F +#define GL_DRAW_BUFFER11 0x8830 +#define GL_DRAW_BUFFER12 0x8831 +#define GL_DRAW_BUFFER13 0x8832 +#define GL_DRAW_BUFFER14 0x8833 +#define GL_DRAW_BUFFER15 0x8834 +#define GL_BLEND_EQUATION_ALPHA 0x883D +#define GL_POINT_SPRITE 0x8861 +#define GL_COORD_REPLACE 0x8862 +#define GL_MAX_VERTEX_ATTRIBS 0x8869 +#define GL_VERTEX_ATTRIB_ARRAY_NORMALIZED 0x886A +#define GL_MAX_TEXTURE_COORDS 0x8871 +#define GL_MAX_TEXTURE_IMAGE_UNITS 0x8872 +#define GL_FRAGMENT_SHADER 0x8B30 +#define GL_VERTEX_SHADER 0x8B31 +#define GL_MAX_FRAGMENT_UNIFORM_COMPONENTS 0x8B49 +#define GL_MAX_VERTEX_UNIFORM_COMPONENTS 0x8B4A +#define GL_MAX_VARYING_FLOATS 0x8B4B +#define GL_MAX_VERTEX_TEXTURE_IMAGE_UNITS 0x8B4C +#define GL_MAX_COMBINED_TEXTURE_IMAGE_UNITS 0x8B4D +#define GL_SHADER_TYPE 0x8B4F +#define GL_FLOAT_VEC2 0x8B50 +#define GL_FLOAT_VEC3 0x8B51 +#define GL_FLOAT_VEC4 0x8B52 +#define GL_INT_VEC2 0x8B53 +#define GL_INT_VEC3 0x8B54 +#define GL_INT_VEC4 0x8B55 +#define GL_BOOL 0x8B56 +#define GL_BOOL_VEC2 0x8B57 +#define GL_BOOL_VEC3 0x8B58 +#define GL_BOOL_VEC4 0x8B59 +#define GL_FLOAT_MAT2 0x8B5A +#define GL_FLOAT_MAT3 0x8B5B +#define GL_FLOAT_MAT4 0x8B5C +#define GL_SAMPLER_1D 0x8B5D +#define GL_SAMPLER_2D 0x8B5E +#define GL_SAMPLER_3D 0x8B5F +#define GL_SAMPLER_CUBE 0x8B60 +#define GL_SAMPLER_1D_SHADOW 0x8B61 +#define GL_SAMPLER_2D_SHADOW 0x8B62 +#define GL_DELETE_STATUS 0x8B80 +#define GL_COMPILE_STATUS 0x8B81 +#define GL_LINK_STATUS 0x8B82 +#define GL_VALIDATE_STATUS 0x8B83 +#define GL_INFO_LOG_LENGTH 0x8B84 +#define GL_ATTACHED_SHADERS 0x8B85 +#define GL_ACTIVE_UNIFORMS 0x8B86 +#define GL_ACTIVE_UNIFORM_MAX_LENGTH 0x8B87 +#define GL_SHADER_SOURCE_LENGTH 0x8B88 +#define GL_ACTIVE_ATTRIBUTES 0x8B89 +#define GL_ACTIVE_ATTRIBUTE_MAX_LENGTH 0x8B8A +#define GL_FRAGMENT_SHADER_DERIVATIVE_HINT 0x8B8B +#define GL_SHADING_LANGUAGE_VERSION 0x8B8C +#define GL_CURRENT_PROGRAM 0x8B8D +#define GL_POINT_SPRITE_COORD_ORIGIN 0x8CA0 +#define GL_LOWER_LEFT 0x8CA1 +#define GL_UPPER_LEFT 0x8CA2 +#define GL_STENCIL_BACK_REF 0x8CA3 +#define GL_STENCIL_BACK_VALUE_MASK 0x8CA4 +#define GL_STENCIL_BACK_WRITEMASK 0x8CA5 +#endif + +#ifndef GL_ARB_multitexture +#define GL_TEXTURE0_ARB 0x84C0 +#define GL_TEXTURE1_ARB 0x84C1 +#define GL_TEXTURE2_ARB 0x84C2 +#define GL_TEXTURE3_ARB 0x84C3 +#define GL_TEXTURE4_ARB 0x84C4 +#define GL_TEXTURE5_ARB 0x84C5 +#define GL_TEXTURE6_ARB 0x84C6 +#define GL_TEXTURE7_ARB 0x84C7 +#define GL_TEXTURE8_ARB 0x84C8 +#define GL_TEXTURE9_ARB 0x84C9 +#define GL_TEXTURE10_ARB 0x84CA +#define GL_TEXTURE11_ARB 0x84CB +#define GL_TEXTURE12_ARB 0x84CC +#define GL_TEXTURE13_ARB 0x84CD +#define GL_TEXTURE14_ARB 0x84CE +#define GL_TEXTURE15_ARB 0x84CF +#define GL_TEXTURE16_ARB 0x84D0 +#define GL_TEXTURE17_ARB 0x84D1 +#define GL_TEXTURE18_ARB 0x84D2 +#define GL_TEXTURE19_ARB 0x84D3 +#define GL_TEXTURE20_ARB 0x84D4 +#define GL_TEXTURE21_ARB 0x84D5 +#define GL_TEXTURE22_ARB 0x84D6 +#define GL_TEXTURE23_ARB 0x84D7 +#define GL_TEXTURE24_ARB 0x84D8 +#define GL_TEXTURE25_ARB 0x84D9 +#define GL_TEXTURE26_ARB 0x84DA +#define GL_TEXTURE27_ARB 0x84DB +#define GL_TEXTURE28_ARB 0x84DC +#define GL_TEXTURE29_ARB 0x84DD +#define GL_TEXTURE30_ARB 0x84DE +#define GL_TEXTURE31_ARB 0x84DF +#define GL_ACTIVE_TEXTURE_ARB 0x84E0 +#define GL_CLIENT_ACTIVE_TEXTURE_ARB 0x84E1 +#define GL_MAX_TEXTURE_UNITS_ARB 0x84E2 +#endif + +#ifndef GL_ARB_transpose_matrix +#define GL_TRANSPOSE_MODELVIEW_MATRIX_ARB 0x84E3 +#define GL_TRANSPOSE_PROJECTION_MATRIX_ARB 0x84E4 +#define GL_TRANSPOSE_TEXTURE_MATRIX_ARB 0x84E5 +#define GL_TRANSPOSE_COLOR_MATRIX_ARB 0x84E6 +#endif + +#ifndef GL_ARB_multisample +#define GL_MULTISAMPLE_ARB 0x809D +#define GL_SAMPLE_ALPHA_TO_COVERAGE_ARB 0x809E +#define GL_SAMPLE_ALPHA_TO_ONE_ARB 0x809F +#define GL_SAMPLE_COVERAGE_ARB 0x80A0 +#define GL_SAMPLE_BUFFERS_ARB 0x80A8 +#define GL_SAMPLES_ARB 0x80A9 +#define GL_SAMPLE_COVERAGE_VALUE_ARB 0x80AA +#define GL_SAMPLE_COVERAGE_INVERT_ARB 0x80AB +#define GL_MULTISAMPLE_BIT_ARB 0x20000000 +#endif + +#ifndef GL_ARB_texture_env_add +#endif + +#ifndef GL_ARB_texture_cube_map +#define GL_NORMAL_MAP_ARB 0x8511 +#define GL_REFLECTION_MAP_ARB 0x8512 +#define GL_TEXTURE_CUBE_MAP_ARB 0x8513 +#define GL_TEXTURE_BINDING_CUBE_MAP_ARB 0x8514 +#define GL_TEXTURE_CUBE_MAP_POSITIVE_X_ARB 0x8515 +#define GL_TEXTURE_CUBE_MAP_NEGATIVE_X_ARB 0x8516 +#define GL_TEXTURE_CUBE_MAP_POSITIVE_Y_ARB 0x8517 +#define GL_TEXTURE_CUBE_MAP_NEGATIVE_Y_ARB 0x8518 +#define GL_TEXTURE_CUBE_MAP_POSITIVE_Z_ARB 0x8519 +#define GL_TEXTURE_CUBE_MAP_NEGATIVE_Z_ARB 0x851A +#define GL_PROXY_TEXTURE_CUBE_MAP_ARB 0x851B +#define GL_MAX_CUBE_MAP_TEXTURE_SIZE_ARB 0x851C +#endif + +#ifndef GL_ARB_texture_compression +#define GL_COMPRESSED_ALPHA_ARB 0x84E9 +#define GL_COMPRESSED_LUMINANCE_ARB 0x84EA +#define GL_COMPRESSED_LUMINANCE_ALPHA_ARB 0x84EB +#define GL_COMPRESSED_INTENSITY_ARB 0x84EC +#define GL_COMPRESSED_RGB_ARB 0x84ED +#define GL_COMPRESSED_RGBA_ARB 0x84EE +#define GL_TEXTURE_COMPRESSION_HINT_ARB 0x84EF +#define GL_TEXTURE_COMPRESSED_IMAGE_SIZE_ARB 0x86A0 +#define GL_TEXTURE_COMPRESSED_ARB 0x86A1 +#define GL_NUM_COMPRESSED_TEXTURE_FORMATS_ARB 0x86A2 +#define GL_COMPRESSED_TEXTURE_FORMATS_ARB 0x86A3 +#endif + +#ifndef GL_ARB_texture_border_clamp +#define GL_CLAMP_TO_BORDER_ARB 0x812D +#endif + +#ifndef GL_ARB_point_parameters +#define GL_POINT_SIZE_MIN_ARB 0x8126 +#define GL_POINT_SIZE_MAX_ARB 0x8127 +#define GL_POINT_FADE_THRESHOLD_SIZE_ARB 0x8128 +#define GL_POINT_DISTANCE_ATTENUATION_ARB 0x8129 +#endif + +#ifndef GL_ARB_vertex_blend +#define GL_MAX_VERTEX_UNITS_ARB 0x86A4 +#define GL_ACTIVE_VERTEX_UNITS_ARB 0x86A5 +#define GL_WEIGHT_SUM_UNITY_ARB 0x86A6 +#define GL_VERTEX_BLEND_ARB 0x86A7 +#define GL_CURRENT_WEIGHT_ARB 0x86A8 +#define GL_WEIGHT_ARRAY_TYPE_ARB 0x86A9 +#define GL_WEIGHT_ARRAY_STRIDE_ARB 0x86AA +#define GL_WEIGHT_ARRAY_SIZE_ARB 0x86AB +#define GL_WEIGHT_ARRAY_POINTER_ARB 0x86AC +#define GL_WEIGHT_ARRAY_ARB 0x86AD +#define GL_MODELVIEW0_ARB 0x1700 +#define GL_MODELVIEW1_ARB 0x850A +#define GL_MODELVIEW2_ARB 0x8722 +#define GL_MODELVIEW3_ARB 0x8723 +#define GL_MODELVIEW4_ARB 0x8724 +#define GL_MODELVIEW5_ARB 0x8725 +#define GL_MODELVIEW6_ARB 0x8726 +#define GL_MODELVIEW7_ARB 0x8727 +#define GL_MODELVIEW8_ARB 0x8728 +#define GL_MODELVIEW9_ARB 0x8729 +#define GL_MODELVIEW10_ARB 0x872A +#define GL_MODELVIEW11_ARB 0x872B +#define GL_MODELVIEW12_ARB 0x872C +#define GL_MODELVIEW13_ARB 0x872D +#define GL_MODELVIEW14_ARB 0x872E +#define GL_MODELVIEW15_ARB 0x872F +#define GL_MODELVIEW16_ARB 0x8730 +#define GL_MODELVIEW17_ARB 0x8731 +#define GL_MODELVIEW18_ARB 0x8732 +#define GL_MODELVIEW19_ARB 0x8733 +#define GL_MODELVIEW20_ARB 0x8734 +#define GL_MODELVIEW21_ARB 0x8735 +#define GL_MODELVIEW22_ARB 0x8736 +#define GL_MODELVIEW23_ARB 0x8737 +#define GL_MODELVIEW24_ARB 0x8738 +#define GL_MODELVIEW25_ARB 0x8739 +#define GL_MODELVIEW26_ARB 0x873A +#define GL_MODELVIEW27_ARB 0x873B +#define GL_MODELVIEW28_ARB 0x873C +#define GL_MODELVIEW29_ARB 0x873D +#define GL_MODELVIEW30_ARB 0x873E +#define GL_MODELVIEW31_ARB 0x873F +#endif + +#ifndef GL_ARB_matrix_palette +#define GL_MATRIX_PALETTE_ARB 0x8840 +#define GL_MAX_MATRIX_PALETTE_STACK_DEPTH_ARB 0x8841 +#define GL_MAX_PALETTE_MATRICES_ARB 0x8842 +#define GL_CURRENT_PALETTE_MATRIX_ARB 0x8843 +#define GL_MATRIX_INDEX_ARRAY_ARB 0x8844 +#define GL_CURRENT_MATRIX_INDEX_ARB 0x8845 +#define GL_MATRIX_INDEX_ARRAY_SIZE_ARB 0x8846 +#define GL_MATRIX_INDEX_ARRAY_TYPE_ARB 0x8847 +#define GL_MATRIX_INDEX_ARRAY_STRIDE_ARB 0x8848 +#define GL_MATRIX_INDEX_ARRAY_POINTER_ARB 0x8849 +#endif + +#ifndef GL_ARB_texture_env_combine +#define GL_COMBINE_ARB 0x8570 +#define GL_COMBINE_RGB_ARB 0x8571 +#define GL_COMBINE_ALPHA_ARB 0x8572 +#define GL_SOURCE0_RGB_ARB 0x8580 +#define GL_SOURCE1_RGB_ARB 0x8581 +#define GL_SOURCE2_RGB_ARB 0x8582 +#define GL_SOURCE0_ALPHA_ARB 0x8588 +#define GL_SOURCE1_ALPHA_ARB 0x8589 +#define GL_SOURCE2_ALPHA_ARB 0x858A +#define GL_OPERAND0_RGB_ARB 0x8590 +#define GL_OPERAND1_RGB_ARB 0x8591 +#define GL_OPERAND2_RGB_ARB 0x8592 +#define GL_OPERAND0_ALPHA_ARB 0x8598 +#define GL_OPERAND1_ALPHA_ARB 0x8599 +#define GL_OPERAND2_ALPHA_ARB 0x859A +#define GL_RGB_SCALE_ARB 0x8573 +#define GL_ADD_SIGNED_ARB 0x8574 +#define GL_INTERPOLATE_ARB 0x8575 +#define GL_SUBTRACT_ARB 0x84E7 +#define GL_CONSTANT_ARB 0x8576 +#define GL_PRIMARY_COLOR_ARB 0x8577 +#define GL_PREVIOUS_ARB 0x8578 +#endif + +#ifndef GL_ARB_texture_env_crossbar +#endif + +#ifndef GL_ARB_texture_env_dot3 +#define GL_DOT3_RGB_ARB 0x86AE +#define GL_DOT3_RGBA_ARB 0x86AF +#endif + +#ifndef GL_ARB_texture_mirrored_repeat +#define GL_MIRRORED_REPEAT_ARB 0x8370 +#endif + +#ifndef GL_ARB_depth_texture +#define GL_DEPTH_COMPONENT16_ARB 0x81A5 +#define GL_DEPTH_COMPONENT24_ARB 0x81A6 +#define GL_DEPTH_COMPONENT32_ARB 0x81A7 +#define GL_TEXTURE_DEPTH_SIZE_ARB 0x884A +#define GL_DEPTH_TEXTURE_MODE_ARB 0x884B +#endif + +#ifndef GL_ARB_shadow +#define GL_TEXTURE_COMPARE_MODE_ARB 0x884C +#define GL_TEXTURE_COMPARE_FUNC_ARB 0x884D +#define GL_COMPARE_R_TO_TEXTURE_ARB 0x884E +#endif + +#ifndef GL_ARB_shadow_ambient +#define GL_TEXTURE_COMPARE_FAIL_VALUE_ARB 0x80BF +#endif + +#ifndef GL_ARB_window_pos +#endif + +#ifndef GL_ARB_vertex_program +#define GL_COLOR_SUM_ARB 0x8458 +#define GL_VERTEX_PROGRAM_ARB 0x8620 +#define GL_VERTEX_ATTRIB_ARRAY_ENABLED_ARB 0x8622 +#define GL_VERTEX_ATTRIB_ARRAY_SIZE_ARB 0x8623 +#define GL_VERTEX_ATTRIB_ARRAY_STRIDE_ARB 0x8624 +#define GL_VERTEX_ATTRIB_ARRAY_TYPE_ARB 0x8625 +#define GL_CURRENT_VERTEX_ATTRIB_ARB 0x8626 +#define GL_PROGRAM_LENGTH_ARB 0x8627 +#define GL_PROGRAM_STRING_ARB 0x8628 +#define GL_MAX_PROGRAM_MATRIX_STACK_DEPTH_ARB 0x862E +#define GL_MAX_PROGRAM_MATRICES_ARB 0x862F +#define GL_CURRENT_MATRIX_STACK_DEPTH_ARB 0x8640 +#define GL_CURRENT_MATRIX_ARB 0x8641 +#define GL_VERTEX_PROGRAM_POINT_SIZE_ARB 0x8642 +#define GL_VERTEX_PROGRAM_TWO_SIDE_ARB 0x8643 +#define GL_VERTEX_ATTRIB_ARRAY_POINTER_ARB 0x8645 +#define GL_PROGRAM_ERROR_POSITION_ARB 0x864B +#define GL_PROGRAM_BINDING_ARB 0x8677 +#define GL_MAX_VERTEX_ATTRIBS_ARB 0x8869 +#define GL_VERTEX_ATTRIB_ARRAY_NORMALIZED_ARB 0x886A +#define GL_PROGRAM_ERROR_STRING_ARB 0x8874 +#define GL_PROGRAM_FORMAT_ASCII_ARB 0x8875 +#define GL_PROGRAM_FORMAT_ARB 0x8876 +#define GL_PROGRAM_INSTRUCTIONS_ARB 0x88A0 +#define GL_MAX_PROGRAM_INSTRUCTIONS_ARB 0x88A1 +#define GL_PROGRAM_NATIVE_INSTRUCTIONS_ARB 0x88A2 +#define GL_MAX_PROGRAM_NATIVE_INSTRUCTIONS_ARB 0x88A3 +#define GL_PROGRAM_TEMPORARIES_ARB 0x88A4 +#define GL_MAX_PROGRAM_TEMPORARIES_ARB 0x88A5 +#define GL_PROGRAM_NATIVE_TEMPORARIES_ARB 0x88A6 +#define GL_MAX_PROGRAM_NATIVE_TEMPORARIES_ARB 0x88A7 +#define GL_PROGRAM_PARAMETERS_ARB 0x88A8 +#define GL_MAX_PROGRAM_PARAMETERS_ARB 0x88A9 +#define GL_PROGRAM_NATIVE_PARAMETERS_ARB 0x88AA +#define GL_MAX_PROGRAM_NATIVE_PARAMETERS_ARB 0x88AB +#define GL_PROGRAM_ATTRIBS_ARB 0x88AC +#define GL_MAX_PROGRAM_ATTRIBS_ARB 0x88AD +#define GL_PROGRAM_NATIVE_ATTRIBS_ARB 0x88AE +#define GL_MAX_PROGRAM_NATIVE_ATTRIBS_ARB 0x88AF +#define GL_PROGRAM_ADDRESS_REGISTERS_ARB 0x88B0 +#define GL_MAX_PROGRAM_ADDRESS_REGISTERS_ARB 0x88B1 +#define GL_PROGRAM_NATIVE_ADDRESS_REGISTERS_ARB 0x88B2 +#define GL_MAX_PROGRAM_NATIVE_ADDRESS_REGISTERS_ARB 0x88B3 +#define GL_MAX_PROGRAM_LOCAL_PARAMETERS_ARB 0x88B4 +#define GL_MAX_PROGRAM_ENV_PARAMETERS_ARB 0x88B5 +#define GL_PROGRAM_UNDER_NATIVE_LIMITS_ARB 0x88B6 +#define GL_TRANSPOSE_CURRENT_MATRIX_ARB 0x88B7 +#define GL_MATRIX0_ARB 0x88C0 +#define GL_MATRIX1_ARB 0x88C1 +#define GL_MATRIX2_ARB 0x88C2 +#define GL_MATRIX3_ARB 0x88C3 +#define GL_MATRIX4_ARB 0x88C4 +#define GL_MATRIX5_ARB 0x88C5 +#define GL_MATRIX6_ARB 0x88C6 +#define GL_MATRIX7_ARB 0x88C7 +#define GL_MATRIX8_ARB 0x88C8 +#define GL_MATRIX9_ARB 0x88C9 +#define GL_MATRIX10_ARB 0x88CA +#define GL_MATRIX11_ARB 0x88CB +#define GL_MATRIX12_ARB 0x88CC +#define GL_MATRIX13_ARB 0x88CD +#define GL_MATRIX14_ARB 0x88CE +#define GL_MATRIX15_ARB 0x88CF +#define GL_MATRIX16_ARB 0x88D0 +#define GL_MATRIX17_ARB 0x88D1 +#define GL_MATRIX18_ARB 0x88D2 +#define GL_MATRIX19_ARB 0x88D3 +#define GL_MATRIX20_ARB 0x88D4 +#define GL_MATRIX21_ARB 0x88D5 +#define GL_MATRIX22_ARB 0x88D6 +#define GL_MATRIX23_ARB 0x88D7 +#define GL_MATRIX24_ARB 0x88D8 +#define GL_MATRIX25_ARB 0x88D9 +#define GL_MATRIX26_ARB 0x88DA +#define GL_MATRIX27_ARB 0x88DB +#define GL_MATRIX28_ARB 0x88DC +#define GL_MATRIX29_ARB 0x88DD +#define GL_MATRIX30_ARB 0x88DE +#define GL_MATRIX31_ARB 0x88DF +#endif + +#ifndef GL_ARB_fragment_program +#define GL_FRAGMENT_PROGRAM_ARB 0x8804 +#define GL_PROGRAM_ALU_INSTRUCTIONS_ARB 0x8805 +#define GL_PROGRAM_TEX_INSTRUCTIONS_ARB 0x8806 +#define GL_PROGRAM_TEX_INDIRECTIONS_ARB 0x8807 +#define GL_PROGRAM_NATIVE_ALU_INSTRUCTIONS_ARB 0x8808 +#define GL_PROGRAM_NATIVE_TEX_INSTRUCTIONS_ARB 0x8809 +#define GL_PROGRAM_NATIVE_TEX_INDIRECTIONS_ARB 0x880A +#define GL_MAX_PROGRAM_ALU_INSTRUCTIONS_ARB 0x880B +#define GL_MAX_PROGRAM_TEX_INSTRUCTIONS_ARB 0x880C +#define GL_MAX_PROGRAM_TEX_INDIRECTIONS_ARB 0x880D +#define GL_MAX_PROGRAM_NATIVE_ALU_INSTRUCTIONS_ARB 0x880E +#define GL_MAX_PROGRAM_NATIVE_TEX_INSTRUCTIONS_ARB 0x880F +#define GL_MAX_PROGRAM_NATIVE_TEX_INDIRECTIONS_ARB 0x8810 +#define GL_MAX_TEXTURE_COORDS_ARB 0x8871 +#define GL_MAX_TEXTURE_IMAGE_UNITS_ARB 0x8872 +#endif + +#ifndef GL_ARB_vertex_buffer_object +#define GL_BUFFER_SIZE_ARB 0x8764 +#define GL_BUFFER_USAGE_ARB 0x8765 +#define GL_ARRAY_BUFFER_ARB 0x8892 +#define GL_ELEMENT_ARRAY_BUFFER_ARB 0x8893 +#define GL_ARRAY_BUFFER_BINDING_ARB 0x8894 +#define GL_ELEMENT_ARRAY_BUFFER_BINDING_ARB 0x8895 +#define GL_VERTEX_ARRAY_BUFFER_BINDING_ARB 0x8896 +#define GL_NORMAL_ARRAY_BUFFER_BINDING_ARB 0x8897 +#define GL_COLOR_ARRAY_BUFFER_BINDING_ARB 0x8898 +#define GL_INDEX_ARRAY_BUFFER_BINDING_ARB 0x8899 +#define GL_TEXTURE_COORD_ARRAY_BUFFER_BINDING_ARB 0x889A +#define GL_EDGE_FLAG_ARRAY_BUFFER_BINDING_ARB 0x889B +#define GL_SECONDARY_COLOR_ARRAY_BUFFER_BINDING_ARB 0x889C +#define GL_FOG_COORDINATE_ARRAY_BUFFER_BINDING_ARB 0x889D +#define GL_WEIGHT_ARRAY_BUFFER_BINDING_ARB 0x889E +#define GL_VERTEX_ATTRIB_ARRAY_BUFFER_BINDING_ARB 0x889F +#define GL_READ_ONLY_ARB 0x88B8 +#define GL_WRITE_ONLY_ARB 0x88B9 +#define GL_READ_WRITE_ARB 0x88BA +#define GL_BUFFER_ACCESS_ARB 0x88BB +#define GL_BUFFER_MAPPED_ARB 0x88BC +#define GL_BUFFER_MAP_POINTER_ARB 0x88BD +#define GL_STREAM_DRAW_ARB 0x88E0 +#define GL_STREAM_READ_ARB 0x88E1 +#define GL_STREAM_COPY_ARB 0x88E2 +#define GL_STATIC_DRAW_ARB 0x88E4 +#define GL_STATIC_READ_ARB 0x88E5 +#define GL_STATIC_COPY_ARB 0x88E6 +#define GL_DYNAMIC_DRAW_ARB 0x88E8 +#define GL_DYNAMIC_READ_ARB 0x88E9 +#define GL_DYNAMIC_COPY_ARB 0x88EA +#endif + +#ifndef GL_ARB_occlusion_query +#define GL_QUERY_COUNTER_BITS_ARB 0x8864 +#define GL_CURRENT_QUERY_ARB 0x8865 +#define GL_QUERY_RESULT_ARB 0x8866 +#define GL_QUERY_RESULT_AVAILABLE_ARB 0x8867 +#define GL_SAMPLES_PASSED_ARB 0x8914 +#endif + +#ifndef GL_ARB_shader_objects +#define GL_PROGRAM_OBJECT_ARB 0x8B40 +#define GL_SHADER_OBJECT_ARB 0x8B48 +#define GL_OBJECT_TYPE_ARB 0x8B4E +#define GL_OBJECT_SUBTYPE_ARB 0x8B4F +#define GL_FLOAT_VEC2_ARB 0x8B50 +#define GL_FLOAT_VEC3_ARB 0x8B51 +#define GL_FLOAT_VEC4_ARB 0x8B52 +#define GL_INT_VEC2_ARB 0x8B53 +#define GL_INT_VEC3_ARB 0x8B54 +#define GL_INT_VEC4_ARB 0x8B55 +#define GL_BOOL_ARB 0x8B56 +#define GL_BOOL_VEC2_ARB 0x8B57 +#define GL_BOOL_VEC3_ARB 0x8B58 +#define GL_BOOL_VEC4_ARB 0x8B59 +#define GL_FLOAT_MAT2_ARB 0x8B5A +#define GL_FLOAT_MAT3_ARB 0x8B5B +#define GL_FLOAT_MAT4_ARB 0x8B5C +#define GL_SAMPLER_1D_ARB 0x8B5D +#define GL_SAMPLER_2D_ARB 0x8B5E +#define GL_SAMPLER_3D_ARB 0x8B5F +#define GL_SAMPLER_CUBE_ARB 0x8B60 +#define GL_SAMPLER_1D_SHADOW_ARB 0x8B61 +#define GL_SAMPLER_2D_SHADOW_ARB 0x8B62 +#define GL_SAMPLER_2D_RECT_ARB 0x8B63 +#define GL_SAMPLER_2D_RECT_SHADOW_ARB 0x8B64 +#define GL_OBJECT_DELETE_STATUS_ARB 0x8B80 +#define GL_OBJECT_COMPILE_STATUS_ARB 0x8B81 +#define GL_OBJECT_LINK_STATUS_ARB 0x8B82 +#define GL_OBJECT_VALIDATE_STATUS_ARB 0x8B83 +#define GL_OBJECT_INFO_LOG_LENGTH_ARB 0x8B84 +#define GL_OBJECT_ATTACHED_OBJECTS_ARB 0x8B85 +#define GL_OBJECT_ACTIVE_UNIFORMS_ARB 0x8B86 +#define GL_OBJECT_ACTIVE_UNIFORM_MAX_LENGTH_ARB 0x8B87 +#define GL_OBJECT_SHADER_SOURCE_LENGTH_ARB 0x8B88 +#endif + +#ifndef GL_ARB_vertex_shader +#define GL_VERTEX_SHADER_ARB 0x8B31 +#define GL_MAX_VERTEX_UNIFORM_COMPONENTS_ARB 0x8B4A +#define GL_MAX_VARYING_FLOATS_ARB 0x8B4B +#define GL_MAX_VERTEX_TEXTURE_IMAGE_UNITS_ARB 0x8B4C +#define GL_MAX_COMBINED_TEXTURE_IMAGE_UNITS_ARB 0x8B4D +#define GL_OBJECT_ACTIVE_ATTRIBUTES_ARB 0x8B89 +#define GL_OBJECT_ACTIVE_ATTRIBUTE_MAX_LENGTH_ARB 0x8B8A +#endif + +#ifndef GL_ARB_fragment_shader +#define GL_FRAGMENT_SHADER_ARB 0x8B30 +#define GL_MAX_FRAGMENT_UNIFORM_COMPONENTS_ARB 0x8B49 +#define GL_FRAGMENT_SHADER_DERIVATIVE_HINT_ARB 0x8B8B +#endif + +#ifndef GL_ARB_shading_language_100 +#define GL_SHADING_LANGUAGE_VERSION_ARB 0x8B8C +#endif + +#ifndef GL_ARB_texture_non_power_of_two +#endif + +#ifndef GL_ARB_point_sprite +#define GL_POINT_SPRITE_ARB 0x8861 +#define GL_COORD_REPLACE_ARB 0x8862 +#endif + +#ifndef GL_ARB_fragment_program_shadow +#endif + +#ifndef GL_ARB_draw_buffers +#define GL_MAX_DRAW_BUFFERS_ARB 0x8824 +#define GL_DRAW_BUFFER0_ARB 0x8825 +#define GL_DRAW_BUFFER1_ARB 0x8826 +#define GL_DRAW_BUFFER2_ARB 0x8827 +#define GL_DRAW_BUFFER3_ARB 0x8828 +#define GL_DRAW_BUFFER4_ARB 0x8829 +#define GL_DRAW_BUFFER5_ARB 0x882A +#define GL_DRAW_BUFFER6_ARB 0x882B +#define GL_DRAW_BUFFER7_ARB 0x882C +#define GL_DRAW_BUFFER8_ARB 0x882D +#define GL_DRAW_BUFFER9_ARB 0x882E +#define GL_DRAW_BUFFER10_ARB 0x882F +#define GL_DRAW_BUFFER11_ARB 0x8830 +#define GL_DRAW_BUFFER12_ARB 0x8831 +#define GL_DRAW_BUFFER13_ARB 0x8832 +#define GL_DRAW_BUFFER14_ARB 0x8833 +#define GL_DRAW_BUFFER15_ARB 0x8834 +#endif + +#ifndef GL_ARB_texture_rectangle +#define GL_TEXTURE_RECTANGLE_ARB 0x84F5 +#define GL_TEXTURE_BINDING_RECTANGLE_ARB 0x84F6 +#define GL_PROXY_TEXTURE_RECTANGLE_ARB 0x84F7 +#define GL_MAX_RECTANGLE_TEXTURE_SIZE_ARB 0x84F8 +#endif + +#ifndef GL_ARB_color_buffer_float +#define GL_RGBA_FLOAT_MODE_ARB 0x8820 +#define GL_CLAMP_VERTEX_COLOR_ARB 0x891A +#define GL_CLAMP_FRAGMENT_COLOR_ARB 0x891B +#define GL_CLAMP_READ_COLOR_ARB 0x891C +#define GL_FIXED_ONLY_ARB 0x891D +#endif + +#ifndef GL_ARB_half_float_pixel +#define GL_HALF_FLOAT_ARB 0x140B +#endif + +#ifndef GL_ARB_texture_float +#define GL_TEXTURE_RED_TYPE_ARB 0x8C10 +#define GL_TEXTURE_GREEN_TYPE_ARB 0x8C11 +#define GL_TEXTURE_BLUE_TYPE_ARB 0x8C12 +#define GL_TEXTURE_ALPHA_TYPE_ARB 0x8C13 +#define GL_TEXTURE_LUMINANCE_TYPE_ARB 0x8C14 +#define GL_TEXTURE_INTENSITY_TYPE_ARB 0x8C15 +#define GL_TEXTURE_DEPTH_TYPE_ARB 0x8C16 +#define GL_UNSIGNED_NORMALIZED_ARB 0x8C17 +#define GL_RGBA32F_ARB 0x8814 +#define GL_RGB32F_ARB 0x8815 +#define GL_ALPHA32F_ARB 0x8816 +#define GL_INTENSITY32F_ARB 0x8817 +#define GL_LUMINANCE32F_ARB 0x8818 +#define GL_LUMINANCE_ALPHA32F_ARB 0x8819 +#define GL_RGBA16F_ARB 0x881A +#define GL_RGB16F_ARB 0x881B +#define GL_ALPHA16F_ARB 0x881C +#define GL_INTENSITY16F_ARB 0x881D +#define GL_LUMINANCE16F_ARB 0x881E +#define GL_LUMINANCE_ALPHA16F_ARB 0x881F +#endif + +#ifndef GL_ARB_pixel_buffer_object +#define GL_PIXEL_PACK_BUFFER_ARB 0x88EB +#define GL_PIXEL_UNPACK_BUFFER_ARB 0x88EC +#define GL_PIXEL_PACK_BUFFER_BINDING_ARB 0x88ED +#define GL_PIXEL_UNPACK_BUFFER_BINDING_ARB 0x88EF +#endif + +#ifndef GL_EXT_abgr +#define GL_ABGR_EXT 0x8000 +#endif + +#ifndef GL_EXT_blend_color +#define GL_CONSTANT_COLOR_EXT 0x8001 +#define GL_ONE_MINUS_CONSTANT_COLOR_EXT 0x8002 +#define GL_CONSTANT_ALPHA_EXT 0x8003 +#define GL_ONE_MINUS_CONSTANT_ALPHA_EXT 0x8004 +#define GL_BLEND_COLOR_EXT 0x8005 +#endif + +#ifndef GL_EXT_polygon_offset +#define GL_POLYGON_OFFSET_EXT 0x8037 +#define GL_POLYGON_OFFSET_FACTOR_EXT 0x8038 +#define GL_POLYGON_OFFSET_BIAS_EXT 0x8039 +#endif + +#ifndef GL_EXT_texture +#define GL_ALPHA4_EXT 0x803B +#define GL_ALPHA8_EXT 0x803C +#define GL_ALPHA12_EXT 0x803D +#define GL_ALPHA16_EXT 0x803E +#define GL_LUMINANCE4_EXT 0x803F +#define GL_LUMINANCE8_EXT 0x8040 +#define GL_LUMINANCE12_EXT 0x8041 +#define GL_LUMINANCE16_EXT 0x8042 +#define GL_LUMINANCE4_ALPHA4_EXT 0x8043 +#define GL_LUMINANCE6_ALPHA2_EXT 0x8044 +#define GL_LUMINANCE8_ALPHA8_EXT 0x8045 +#define GL_LUMINANCE12_ALPHA4_EXT 0x8046 +#define GL_LUMINANCE12_ALPHA12_EXT 0x8047 +#define GL_LUMINANCE16_ALPHA16_EXT 0x8048 +#define GL_INTENSITY_EXT 0x8049 +#define GL_INTENSITY4_EXT 0x804A +#define GL_INTENSITY8_EXT 0x804B +#define GL_INTENSITY12_EXT 0x804C +#define GL_INTENSITY16_EXT 0x804D +#define GL_RGB2_EXT 0x804E +#define GL_RGB4_EXT 0x804F +#define GL_RGB5_EXT 0x8050 +#define GL_RGB8_EXT 0x8051 +#define GL_RGB10_EXT 0x8052 +#define GL_RGB12_EXT 0x8053 +#define GL_RGB16_EXT 0x8054 +#define GL_RGBA2_EXT 0x8055 +#define GL_RGBA4_EXT 0x8056 +#define GL_RGB5_A1_EXT 0x8057 +#define GL_RGBA8_EXT 0x8058 +#define GL_RGB10_A2_EXT 0x8059 +#define GL_RGBA12_EXT 0x805A +#define GL_RGBA16_EXT 0x805B +#define GL_TEXTURE_RED_SIZE_EXT 0x805C +#define GL_TEXTURE_GREEN_SIZE_EXT 0x805D +#define GL_TEXTURE_BLUE_SIZE_EXT 0x805E +#define GL_TEXTURE_ALPHA_SIZE_EXT 0x805F +#define GL_TEXTURE_LUMINANCE_SIZE_EXT 0x8060 +#define GL_TEXTURE_INTENSITY_SIZE_EXT 0x8061 +#define GL_REPLACE_EXT 0x8062 +#define GL_PROXY_TEXTURE_1D_EXT 0x8063 +#define GL_PROXY_TEXTURE_2D_EXT 0x8064 +#define GL_TEXTURE_TOO_LARGE_EXT 0x8065 +#endif + +#ifndef GL_EXT_texture3D +#define GL_PACK_SKIP_IMAGES_EXT 0x806B +#define GL_PACK_IMAGE_HEIGHT_EXT 0x806C +#define GL_UNPACK_SKIP_IMAGES_EXT 0x806D +#define GL_UNPACK_IMAGE_HEIGHT_EXT 0x806E +#define GL_TEXTURE_3D_EXT 0x806F +#define GL_PROXY_TEXTURE_3D_EXT 0x8070 +#define GL_TEXTURE_DEPTH_EXT 0x8071 +#define GL_TEXTURE_WRAP_R_EXT 0x8072 +#define GL_MAX_3D_TEXTURE_SIZE_EXT 0x8073 +#endif + +#ifndef GL_SGIS_texture_filter4 +#define GL_FILTER4_SGIS 0x8146 +#define GL_TEXTURE_FILTER4_SIZE_SGIS 0x8147 +#endif + +#ifndef GL_EXT_subtexture +#endif + +#ifndef GL_EXT_copy_texture +#endif + +#ifndef GL_EXT_histogram +#define GL_HISTOGRAM_EXT 0x8024 +#define GL_PROXY_HISTOGRAM_EXT 0x8025 +#define GL_HISTOGRAM_WIDTH_EXT 0x8026 +#define GL_HISTOGRAM_FORMAT_EXT 0x8027 +#define GL_HISTOGRAM_RED_SIZE_EXT 0x8028 +#define GL_HISTOGRAM_GREEN_SIZE_EXT 0x8029 +#define GL_HISTOGRAM_BLUE_SIZE_EXT 0x802A +#define GL_HISTOGRAM_ALPHA_SIZE_EXT 0x802B +#define GL_HISTOGRAM_LUMINANCE_SIZE_EXT 0x802C +#define GL_HISTOGRAM_SINK_EXT 0x802D +#define GL_MINMAX_EXT 0x802E +#define GL_MINMAX_FORMAT_EXT 0x802F +#define GL_MINMAX_SINK_EXT 0x8030 +#define GL_TABLE_TOO_LARGE_EXT 0x8031 +#endif + +#ifndef GL_EXT_convolution +#define GL_CONVOLUTION_1D_EXT 0x8010 +#define GL_CONVOLUTION_2D_EXT 0x8011 +#define GL_SEPARABLE_2D_EXT 0x8012 +#define GL_CONVOLUTION_BORDER_MODE_EXT 0x8013 +#define GL_CONVOLUTION_FILTER_SCALE_EXT 0x8014 +#define GL_CONVOLUTION_FILTER_BIAS_EXT 0x8015 +#define GL_REDUCE_EXT 0x8016 +#define GL_CONVOLUTION_FORMAT_EXT 0x8017 +#define GL_CONVOLUTION_WIDTH_EXT 0x8018 +#define GL_CONVOLUTION_HEIGHT_EXT 0x8019 +#define GL_MAX_CONVOLUTION_WIDTH_EXT 0x801A +#define GL_MAX_CONVOLUTION_HEIGHT_EXT 0x801B +#define GL_POST_CONVOLUTION_RED_SCALE_EXT 0x801C +#define GL_POST_CONVOLUTION_GREEN_SCALE_EXT 0x801D +#define GL_POST_CONVOLUTION_BLUE_SCALE_EXT 0x801E +#define GL_POST_CONVOLUTION_ALPHA_SCALE_EXT 0x801F +#define GL_POST_CONVOLUTION_RED_BIAS_EXT 0x8020 +#define GL_POST_CONVOLUTION_GREEN_BIAS_EXT 0x8021 +#define GL_POST_CONVOLUTION_BLUE_BIAS_EXT 0x8022 +#define GL_POST_CONVOLUTION_ALPHA_BIAS_EXT 0x8023 +#endif + +#ifndef GL_SGI_color_matrix +#define GL_COLOR_MATRIX_SGI 0x80B1 +#define GL_COLOR_MATRIX_STACK_DEPTH_SGI 0x80B2 +#define GL_MAX_COLOR_MATRIX_STACK_DEPTH_SGI 0x80B3 +#define GL_POST_COLOR_MATRIX_RED_SCALE_SGI 0x80B4 +#define GL_POST_COLOR_MATRIX_GREEN_SCALE_SGI 0x80B5 +#define GL_POST_COLOR_MATRIX_BLUE_SCALE_SGI 0x80B6 +#define GL_POST_COLOR_MATRIX_ALPHA_SCALE_SGI 0x80B7 +#define GL_POST_COLOR_MATRIX_RED_BIAS_SGI 0x80B8 +#define GL_POST_COLOR_MATRIX_GREEN_BIAS_SGI 0x80B9 +#define GL_POST_COLOR_MATRIX_BLUE_BIAS_SGI 0x80BA +#define GL_POST_COLOR_MATRIX_ALPHA_BIAS_SGI 0x80BB +#endif + +#ifndef GL_SGI_color_table +#define GL_COLOR_TABLE_SGI 0x80D0 +#define GL_POST_CONVOLUTION_COLOR_TABLE_SGI 0x80D1 +#define GL_POST_COLOR_MATRIX_COLOR_TABLE_SGI 0x80D2 +#define GL_PROXY_COLOR_TABLE_SGI 0x80D3 +#define GL_PROXY_POST_CONVOLUTION_COLOR_TABLE_SGI 0x80D4 +#define GL_PROXY_POST_COLOR_MATRIX_COLOR_TABLE_SGI 0x80D5 +#define GL_COLOR_TABLE_SCALE_SGI 0x80D6 +#define GL_COLOR_TABLE_BIAS_SGI 0x80D7 +#define GL_COLOR_TABLE_FORMAT_SGI 0x80D8 +#define GL_COLOR_TABLE_WIDTH_SGI 0x80D9 +#define GL_COLOR_TABLE_RED_SIZE_SGI 0x80DA +#define GL_COLOR_TABLE_GREEN_SIZE_SGI 0x80DB +#define GL_COLOR_TABLE_BLUE_SIZE_SGI 0x80DC +#define GL_COLOR_TABLE_ALPHA_SIZE_SGI 0x80DD +#define GL_COLOR_TABLE_LUMINANCE_SIZE_SGI 0x80DE +#define GL_COLOR_TABLE_INTENSITY_SIZE_SGI 0x80DF +#endif + +#ifndef GL_SGIS_pixel_texture +#define GL_PIXEL_TEXTURE_SGIS 0x8353 +#define GL_PIXEL_FRAGMENT_RGB_SOURCE_SGIS 0x8354 +#define GL_PIXEL_FRAGMENT_ALPHA_SOURCE_SGIS 0x8355 +#define GL_PIXEL_GROUP_COLOR_SGIS 0x8356 +#endif + +#ifndef GL_SGIX_pixel_texture +#define GL_PIXEL_TEX_GEN_SGIX 0x8139 +#define GL_PIXEL_TEX_GEN_MODE_SGIX 0x832B +#endif + +#ifndef GL_SGIS_texture4D +#define GL_PACK_SKIP_VOLUMES_SGIS 0x8130 +#define GL_PACK_IMAGE_DEPTH_SGIS 0x8131 +#define GL_UNPACK_SKIP_VOLUMES_SGIS 0x8132 +#define GL_UNPACK_IMAGE_DEPTH_SGIS 0x8133 +#define GL_TEXTURE_4D_SGIS 0x8134 +#define GL_PROXY_TEXTURE_4D_SGIS 0x8135 +#define GL_TEXTURE_4DSIZE_SGIS 0x8136 +#define GL_TEXTURE_WRAP_Q_SGIS 0x8137 +#define GL_MAX_4D_TEXTURE_SIZE_SGIS 0x8138 +#define GL_TEXTURE_4D_BINDING_SGIS 0x814F +#endif + +#ifndef GL_SGI_texture_color_table +#define GL_TEXTURE_COLOR_TABLE_SGI 0x80BC +#define GL_PROXY_TEXTURE_COLOR_TABLE_SGI 0x80BD +#endif + +#ifndef GL_EXT_cmyka +#define GL_CMYK_EXT 0x800C +#define GL_CMYKA_EXT 0x800D +#define GL_PACK_CMYK_HINT_EXT 0x800E +#define GL_UNPACK_CMYK_HINT_EXT 0x800F +#endif + +#ifndef GL_EXT_texture_object +#define GL_TEXTURE_PRIORITY_EXT 0x8066 +#define GL_TEXTURE_RESIDENT_EXT 0x8067 +#define GL_TEXTURE_1D_BINDING_EXT 0x8068 +#define GL_TEXTURE_2D_BINDING_EXT 0x8069 +#define GL_TEXTURE_3D_BINDING_EXT 0x806A +#endif + +#ifndef GL_SGIS_detail_texture +#define GL_DETAIL_TEXTURE_2D_SGIS 0x8095 +#define GL_DETAIL_TEXTURE_2D_BINDING_SGIS 0x8096 +#define GL_LINEAR_DETAIL_SGIS 0x8097 +#define GL_LINEAR_DETAIL_ALPHA_SGIS 0x8098 +#define GL_LINEAR_DETAIL_COLOR_SGIS 0x8099 +#define GL_DETAIL_TEXTURE_LEVEL_SGIS 0x809A +#define GL_DETAIL_TEXTURE_MODE_SGIS 0x809B +#define GL_DETAIL_TEXTURE_FUNC_POINTS_SGIS 0x809C +#endif + +#ifndef GL_SGIS_sharpen_texture +#define GL_LINEAR_SHARPEN_SGIS 0x80AD +#define GL_LINEAR_SHARPEN_ALPHA_SGIS 0x80AE +#define GL_LINEAR_SHARPEN_COLOR_SGIS 0x80AF +#define GL_SHARPEN_TEXTURE_FUNC_POINTS_SGIS 0x80B0 +#endif + +#ifndef GL_EXT_packed_pixels +#define GL_UNSIGNED_BYTE_3_3_2_EXT 0x8032 +#define GL_UNSIGNED_SHORT_4_4_4_4_EXT 0x8033 +#define GL_UNSIGNED_SHORT_5_5_5_1_EXT 0x8034 +#define GL_UNSIGNED_INT_8_8_8_8_EXT 0x8035 +#define GL_UNSIGNED_INT_10_10_10_2_EXT 0x8036 +#endif + +#ifndef GL_SGIS_texture_lod +#define GL_TEXTURE_MIN_LOD_SGIS 0x813A +#define GL_TEXTURE_MAX_LOD_SGIS 0x813B +#define GL_TEXTURE_BASE_LEVEL_SGIS 0x813C +#define GL_TEXTURE_MAX_LEVEL_SGIS 0x813D +#endif + +#ifndef GL_SGIS_multisample +#define GL_MULTISAMPLE_SGIS 0x809D +#define GL_SAMPLE_ALPHA_TO_MASK_SGIS 0x809E +#define GL_SAMPLE_ALPHA_TO_ONE_SGIS 0x809F +#define GL_SAMPLE_MASK_SGIS 0x80A0 +#define GL_1PASS_SGIS 0x80A1 +#define GL_2PASS_0_SGIS 0x80A2 +#define GL_2PASS_1_SGIS 0x80A3 +#define GL_4PASS_0_SGIS 0x80A4 +#define GL_4PASS_1_SGIS 0x80A5 +#define GL_4PASS_2_SGIS 0x80A6 +#define GL_4PASS_3_SGIS 0x80A7 +#define GL_SAMPLE_BUFFERS_SGIS 0x80A8 +#define GL_SAMPLES_SGIS 0x80A9 +#define GL_SAMPLE_MASK_VALUE_SGIS 0x80AA +#define GL_SAMPLE_MASK_INVERT_SGIS 0x80AB +#define GL_SAMPLE_PATTERN_SGIS 0x80AC +#endif + +#ifndef GL_EXT_rescale_normal +#define GL_RESCALE_NORMAL_EXT 0x803A +#endif + +#ifndef GL_EXT_vertex_array +#define GL_VERTEX_ARRAY_EXT 0x8074 +#define GL_NORMAL_ARRAY_EXT 0x8075 +#define GL_COLOR_ARRAY_EXT 0x8076 +#define GL_INDEX_ARRAY_EXT 0x8077 +#define GL_TEXTURE_COORD_ARRAY_EXT 0x8078 +#define GL_EDGE_FLAG_ARRAY_EXT 0x8079 +#define GL_VERTEX_ARRAY_SIZE_EXT 0x807A +#define GL_VERTEX_ARRAY_TYPE_EXT 0x807B +#define GL_VERTEX_ARRAY_STRIDE_EXT 0x807C +#define GL_VERTEX_ARRAY_COUNT_EXT 0x807D +#define GL_NORMAL_ARRAY_TYPE_EXT 0x807E +#define GL_NORMAL_ARRAY_STRIDE_EXT 0x807F +#define GL_NORMAL_ARRAY_COUNT_EXT 0x8080 +#define GL_COLOR_ARRAY_SIZE_EXT 0x8081 +#define GL_COLOR_ARRAY_TYPE_EXT 0x8082 +#define GL_COLOR_ARRAY_STRIDE_EXT 0x8083 +#define GL_COLOR_ARRAY_COUNT_EXT 0x8084 +#define GL_INDEX_ARRAY_TYPE_EXT 0x8085 +#define GL_INDEX_ARRAY_STRIDE_EXT 0x8086 +#define GL_INDEX_ARRAY_COUNT_EXT 0x8087 +#define GL_TEXTURE_COORD_ARRAY_SIZE_EXT 0x8088 +#define GL_TEXTURE_COORD_ARRAY_TYPE_EXT 0x8089 +#define GL_TEXTURE_COORD_ARRAY_STRIDE_EXT 0x808A +#define GL_TEXTURE_COORD_ARRAY_COUNT_EXT 0x808B +#define GL_EDGE_FLAG_ARRAY_STRIDE_EXT 0x808C +#define GL_EDGE_FLAG_ARRAY_COUNT_EXT 0x808D +#define GL_VERTEX_ARRAY_POINTER_EXT 0x808E +#define GL_NORMAL_ARRAY_POINTER_EXT 0x808F +#define GL_COLOR_ARRAY_POINTER_EXT 0x8090 +#define GL_INDEX_ARRAY_POINTER_EXT 0x8091 +#define GL_TEXTURE_COORD_ARRAY_POINTER_EXT 0x8092 +#define GL_EDGE_FLAG_ARRAY_POINTER_EXT 0x8093 +#endif + +#ifndef GL_EXT_misc_attribute +#endif + +#ifndef GL_SGIS_generate_mipmap +#define GL_GENERATE_MIPMAP_SGIS 0x8191 +#define GL_GENERATE_MIPMAP_HINT_SGIS 0x8192 +#endif + +#ifndef GL_SGIX_clipmap +#define GL_LINEAR_CLIPMAP_LINEAR_SGIX 0x8170 +#define GL_TEXTURE_CLIPMAP_CENTER_SGIX 0x8171 +#define GL_TEXTURE_CLIPMAP_FRAME_SGIX 0x8172 +#define GL_TEXTURE_CLIPMAP_OFFSET_SGIX 0x8173 +#define GL_TEXTURE_CLIPMAP_VIRTUAL_DEPTH_SGIX 0x8174 +#define GL_TEXTURE_CLIPMAP_LOD_OFFSET_SGIX 0x8175 +#define GL_TEXTURE_CLIPMAP_DEPTH_SGIX 0x8176 +#define GL_MAX_CLIPMAP_DEPTH_SGIX 0x8177 +#define GL_MAX_CLIPMAP_VIRTUAL_DEPTH_SGIX 0x8178 +#define GL_NEAREST_CLIPMAP_NEAREST_SGIX 0x844D +#define GL_NEAREST_CLIPMAP_LINEAR_SGIX 0x844E +#define GL_LINEAR_CLIPMAP_NEAREST_SGIX 0x844F +#endif + +#ifndef GL_SGIX_shadow +#define GL_TEXTURE_COMPARE_SGIX 0x819A +#define GL_TEXTURE_COMPARE_OPERATOR_SGIX 0x819B +#define GL_TEXTURE_LEQUAL_R_SGIX 0x819C +#define GL_TEXTURE_GEQUAL_R_SGIX 0x819D +#endif + +#ifndef GL_SGIS_texture_edge_clamp +#define GL_CLAMP_TO_EDGE_SGIS 0x812F +#endif + +#ifndef GL_SGIS_texture_border_clamp +#define GL_CLAMP_TO_BORDER_SGIS 0x812D +#endif + +#ifndef GL_EXT_blend_minmax +#define GL_FUNC_ADD_EXT 0x8006 +#define GL_MIN_EXT 0x8007 +#define GL_MAX_EXT 0x8008 +#define GL_BLEND_EQUATION_EXT 0x8009 +#endif + +#ifndef GL_EXT_blend_subtract +#define GL_FUNC_SUBTRACT_EXT 0x800A +#define GL_FUNC_REVERSE_SUBTRACT_EXT 0x800B +#endif + +#ifndef GL_EXT_blend_logic_op +#endif + +#ifndef GL_SGIX_interlace +#define GL_INTERLACE_SGIX 0x8094 +#endif + +#ifndef GL_SGIX_pixel_tiles +#define GL_PIXEL_TILE_BEST_ALIGNMENT_SGIX 0x813E +#define GL_PIXEL_TILE_CACHE_INCREMENT_SGIX 0x813F +#define GL_PIXEL_TILE_WIDTH_SGIX 0x8140 +#define GL_PIXEL_TILE_HEIGHT_SGIX 0x8141 +#define GL_PIXEL_TILE_GRID_WIDTH_SGIX 0x8142 +#define GL_PIXEL_TILE_GRID_HEIGHT_SGIX 0x8143 +#define GL_PIXEL_TILE_GRID_DEPTH_SGIX 0x8144 +#define GL_PIXEL_TILE_CACHE_SIZE_SGIX 0x8145 +#endif + +#ifndef GL_SGIS_texture_select +#define GL_DUAL_ALPHA4_SGIS 0x8110 +#define GL_DUAL_ALPHA8_SGIS 0x8111 +#define GL_DUAL_ALPHA12_SGIS 0x8112 +#define GL_DUAL_ALPHA16_SGIS 0x8113 +#define GL_DUAL_LUMINANCE4_SGIS 0x8114 +#define GL_DUAL_LUMINANCE8_SGIS 0x8115 +#define GL_DUAL_LUMINANCE12_SGIS 0x8116 +#define GL_DUAL_LUMINANCE16_SGIS 0x8117 +#define GL_DUAL_INTENSITY4_SGIS 0x8118 +#define GL_DUAL_INTENSITY8_SGIS 0x8119 +#define GL_DUAL_INTENSITY12_SGIS 0x811A +#define GL_DUAL_INTENSITY16_SGIS 0x811B +#define GL_DUAL_LUMINANCE_ALPHA4_SGIS 0x811C +#define GL_DUAL_LUMINANCE_ALPHA8_SGIS 0x811D +#define GL_QUAD_ALPHA4_SGIS 0x811E +#define GL_QUAD_ALPHA8_SGIS 0x811F +#define GL_QUAD_LUMINANCE4_SGIS 0x8120 +#define GL_QUAD_LUMINANCE8_SGIS 0x8121 +#define GL_QUAD_INTENSITY4_SGIS 0x8122 +#define GL_QUAD_INTENSITY8_SGIS 0x8123 +#define GL_DUAL_TEXTURE_SELECT_SGIS 0x8124 +#define GL_QUAD_TEXTURE_SELECT_SGIS 0x8125 +#endif + +#ifndef GL_SGIX_sprite +#define GL_SPRITE_SGIX 0x8148 +#define GL_SPRITE_MODE_SGIX 0x8149 +#define GL_SPRITE_AXIS_SGIX 0x814A +#define GL_SPRITE_TRANSLATION_SGIX 0x814B +#define GL_SPRITE_AXIAL_SGIX 0x814C +#define GL_SPRITE_OBJECT_ALIGNED_SGIX 0x814D +#define GL_SPRITE_EYE_ALIGNED_SGIX 0x814E +#endif + +#ifndef GL_SGIX_texture_multi_buffer +#define GL_TEXTURE_MULTI_BUFFER_HINT_SGIX 0x812E +#endif + +#ifndef GL_EXT_point_parameters +#define GL_POINT_SIZE_MIN_EXT 0x8126 +#define GL_POINT_SIZE_MAX_EXT 0x8127 +#define GL_POINT_FADE_THRESHOLD_SIZE_EXT 0x8128 +#define GL_DISTANCE_ATTENUATION_EXT 0x8129 +#endif + +#ifndef GL_SGIS_point_parameters +#define GL_POINT_SIZE_MIN_SGIS 0x8126 +#define GL_POINT_SIZE_MAX_SGIS 0x8127 +#define GL_POINT_FADE_THRESHOLD_SIZE_SGIS 0x8128 +#define GL_DISTANCE_ATTENUATION_SGIS 0x8129 +#endif + +#ifndef GL_SGIX_instruments +#define GL_INSTRUMENT_BUFFER_POINTER_SGIX 0x8180 +#define GL_INSTRUMENT_MEASUREMENTS_SGIX 0x8181 +#endif + +#ifndef GL_SGIX_texture_scale_bias +#define GL_POST_TEXTURE_FILTER_BIAS_SGIX 0x8179 +#define GL_POST_TEXTURE_FILTER_SCALE_SGIX 0x817A +#define GL_POST_TEXTURE_FILTER_BIAS_RANGE_SGIX 0x817B +#define GL_POST_TEXTURE_FILTER_SCALE_RANGE_SGIX 0x817C +#endif + +#ifndef GL_SGIX_framezoom +#define GL_FRAMEZOOM_SGIX 0x818B +#define GL_FRAMEZOOM_FACTOR_SGIX 0x818C +#define GL_MAX_FRAMEZOOM_FACTOR_SGIX 0x818D +#endif + +#ifndef GL_SGIX_tag_sample_buffer +#endif + +#ifndef GL_FfdMaskSGIX +#define GL_TEXTURE_DEFORMATION_BIT_SGIX 0x00000001 +#define GL_GEOMETRY_DEFORMATION_BIT_SGIX 0x00000002 +#endif + +#ifndef GL_SGIX_polynomial_ffd +#define GL_GEOMETRY_DEFORMATION_SGIX 0x8194 +#define GL_TEXTURE_DEFORMATION_SGIX 0x8195 +#define GL_DEFORMATIONS_MASK_SGIX 0x8196 +#define GL_MAX_DEFORMATION_ORDER_SGIX 0x8197 +#endif + +#ifndef GL_SGIX_reference_plane +#define GL_REFERENCE_PLANE_SGIX 0x817D +#define GL_REFERENCE_PLANE_EQUATION_SGIX 0x817E +#endif + +#ifndef GL_SGIX_flush_raster +#endif + +#ifndef GL_SGIX_depth_texture +#define GL_DEPTH_COMPONENT16_SGIX 0x81A5 +#define GL_DEPTH_COMPONENT24_SGIX 0x81A6 +#define GL_DEPTH_COMPONENT32_SGIX 0x81A7 +#endif + +#ifndef GL_SGIS_fog_function +#define GL_FOG_FUNC_SGIS 0x812A +#define GL_FOG_FUNC_POINTS_SGIS 0x812B +#define GL_MAX_FOG_FUNC_POINTS_SGIS 0x812C +#endif + +#ifndef GL_SGIX_fog_offset +#define GL_FOG_OFFSET_SGIX 0x8198 +#define GL_FOG_OFFSET_VALUE_SGIX 0x8199 +#endif + +#ifndef GL_HP_image_transform +#define GL_IMAGE_SCALE_X_HP 0x8155 +#define GL_IMAGE_SCALE_Y_HP 0x8156 +#define GL_IMAGE_TRANSLATE_X_HP 0x8157 +#define GL_IMAGE_TRANSLATE_Y_HP 0x8158 +#define GL_IMAGE_ROTATE_ANGLE_HP 0x8159 +#define GL_IMAGE_ROTATE_ORIGIN_X_HP 0x815A +#define GL_IMAGE_ROTATE_ORIGIN_Y_HP 0x815B +#define GL_IMAGE_MAG_FILTER_HP 0x815C +#define GL_IMAGE_MIN_FILTER_HP 0x815D +#define GL_IMAGE_CUBIC_WEIGHT_HP 0x815E +#define GL_CUBIC_HP 0x815F +#define GL_AVERAGE_HP 0x8160 +#define GL_IMAGE_TRANSFORM_2D_HP 0x8161 +#define GL_POST_IMAGE_TRANSFORM_COLOR_TABLE_HP 0x8162 +#define GL_PROXY_POST_IMAGE_TRANSFORM_COLOR_TABLE_HP 0x8163 +#endif + +#ifndef GL_HP_convolution_border_modes +#define GL_IGNORE_BORDER_HP 0x8150 +#define GL_CONSTANT_BORDER_HP 0x8151 +#define GL_REPLICATE_BORDER_HP 0x8153 +#define GL_CONVOLUTION_BORDER_COLOR_HP 0x8154 +#endif + +#ifndef GL_INGR_palette_buffer +#endif + +#ifndef GL_SGIX_texture_add_env +#define GL_TEXTURE_ENV_BIAS_SGIX 0x80BE +#endif + +#ifndef GL_EXT_color_subtable +#endif + +#ifndef GL_PGI_vertex_hints +#define GL_VERTEX_DATA_HINT_PGI 0x1A22A +#define GL_VERTEX_CONSISTENT_HINT_PGI 0x1A22B +#define GL_MATERIAL_SIDE_HINT_PGI 0x1A22C +#define GL_MAX_VERTEX_HINT_PGI 0x1A22D +#define GL_COLOR3_BIT_PGI 0x00010000 +#define GL_COLOR4_BIT_PGI 0x00020000 +#define GL_EDGEFLAG_BIT_PGI 0x00040000 +#define GL_INDEX_BIT_PGI 0x00080000 +#define GL_MAT_AMBIENT_BIT_PGI 0x00100000 +#define GL_MAT_AMBIENT_AND_DIFFUSE_BIT_PGI 0x00200000 +#define GL_MAT_DIFFUSE_BIT_PGI 0x00400000 +#define GL_MAT_EMISSION_BIT_PGI 0x00800000 +#define GL_MAT_COLOR_INDEXES_BIT_PGI 0x01000000 +#define GL_MAT_SHININESS_BIT_PGI 0x02000000 +#define GL_MAT_SPECULAR_BIT_PGI 0x04000000 +#define GL_NORMAL_BIT_PGI 0x08000000 +#define GL_TEXCOORD1_BIT_PGI 0x10000000 +#define GL_TEXCOORD2_BIT_PGI 0x20000000 +#define GL_TEXCOORD3_BIT_PGI 0x40000000 +#define GL_TEXCOORD4_BIT_PGI 0x80000000 +#define GL_VERTEX23_BIT_PGI 0x00000004 +#define GL_VERTEX4_BIT_PGI 0x00000008 +#endif + +#ifndef GL_PGI_misc_hints +#define GL_PREFER_DOUBLEBUFFER_HINT_PGI 0x1A1F8 +#define GL_CONSERVE_MEMORY_HINT_PGI 0x1A1FD +#define GL_RECLAIM_MEMORY_HINT_PGI 0x1A1FE +#define GL_NATIVE_GRAPHICS_HANDLE_PGI 0x1A202 +#define GL_NATIVE_GRAPHICS_BEGIN_HINT_PGI 0x1A203 +#define GL_NATIVE_GRAPHICS_END_HINT_PGI 0x1A204 +#define GL_ALWAYS_FAST_HINT_PGI 0x1A20C +#define GL_ALWAYS_SOFT_HINT_PGI 0x1A20D +#define GL_ALLOW_DRAW_OBJ_HINT_PGI 0x1A20E +#define GL_ALLOW_DRAW_WIN_HINT_PGI 0x1A20F +#define GL_ALLOW_DRAW_FRG_HINT_PGI 0x1A210 +#define GL_ALLOW_DRAW_MEM_HINT_PGI 0x1A211 +#define GL_STRICT_DEPTHFUNC_HINT_PGI 0x1A216 +#define GL_STRICT_LIGHTING_HINT_PGI 0x1A217 +#define GL_STRICT_SCISSOR_HINT_PGI 0x1A218 +#define GL_FULL_STIPPLE_HINT_PGI 0x1A219 +#define GL_CLIP_NEAR_HINT_PGI 0x1A220 +#define GL_CLIP_FAR_HINT_PGI 0x1A221 +#define GL_WIDE_LINE_HINT_PGI 0x1A222 +#define GL_BACK_NORMALS_HINT_PGI 0x1A223 +#endif + +#ifndef GL_EXT_paletted_texture +#define GL_COLOR_INDEX1_EXT 0x80E2 +#define GL_COLOR_INDEX2_EXT 0x80E3 +#define GL_COLOR_INDEX4_EXT 0x80E4 +#define GL_COLOR_INDEX8_EXT 0x80E5 +#define GL_COLOR_INDEX12_EXT 0x80E6 +#define GL_COLOR_INDEX16_EXT 0x80E7 +#define GL_TEXTURE_INDEX_SIZE_EXT 0x80ED +#endif + +#ifndef GL_EXT_clip_volume_hint +#define GL_CLIP_VOLUME_CLIPPING_HINT_EXT 0x80F0 +#endif + +#ifndef GL_SGIX_list_priority +#define GL_LIST_PRIORITY_SGIX 0x8182 +#endif + +#ifndef GL_SGIX_ir_instrument1 +#define GL_IR_INSTRUMENT1_SGIX 0x817F +#endif + +#ifndef GL_SGIX_calligraphic_fragment +#define GL_CALLIGRAPHIC_FRAGMENT_SGIX 0x8183 +#endif + +#ifndef GL_SGIX_texture_lod_bias +#define GL_TEXTURE_LOD_BIAS_S_SGIX 0x818E +#define GL_TEXTURE_LOD_BIAS_T_SGIX 0x818F +#define GL_TEXTURE_LOD_BIAS_R_SGIX 0x8190 +#endif + +#ifndef GL_SGIX_shadow_ambient +#define GL_SHADOW_AMBIENT_SGIX 0x80BF +#endif + +#ifndef GL_EXT_index_texture +#endif + +#ifndef GL_EXT_index_material +#define GL_INDEX_MATERIAL_EXT 0x81B8 +#define GL_INDEX_MATERIAL_PARAMETER_EXT 0x81B9 +#define GL_INDEX_MATERIAL_FACE_EXT 0x81BA +#endif + +#ifndef GL_EXT_index_func +#define GL_INDEX_TEST_EXT 0x81B5 +#define GL_INDEX_TEST_FUNC_EXT 0x81B6 +#define GL_INDEX_TEST_REF_EXT 0x81B7 +#endif + +#ifndef GL_EXT_index_array_formats +#define GL_IUI_V2F_EXT 0x81AD +#define GL_IUI_V3F_EXT 0x81AE +#define GL_IUI_N3F_V2F_EXT 0x81AF +#define GL_IUI_N3F_V3F_EXT 0x81B0 +#define GL_T2F_IUI_V2F_EXT 0x81B1 +#define GL_T2F_IUI_V3F_EXT 0x81B2 +#define GL_T2F_IUI_N3F_V2F_EXT 0x81B3 +#define GL_T2F_IUI_N3F_V3F_EXT 0x81B4 +#endif + +#ifndef GL_EXT_compiled_vertex_array +#define GL_ARRAY_ELEMENT_LOCK_FIRST_EXT 0x81A8 +#define GL_ARRAY_ELEMENT_LOCK_COUNT_EXT 0x81A9 +#endif + +#ifndef GL_EXT_cull_vertex +#define GL_CULL_VERTEX_EXT 0x81AA +#define GL_CULL_VERTEX_EYE_POSITION_EXT 0x81AB +#define GL_CULL_VERTEX_OBJECT_POSITION_EXT 0x81AC +#endif + +#ifndef GL_SGIX_ycrcb +#define GL_YCRCB_422_SGIX 0x81BB +#define GL_YCRCB_444_SGIX 0x81BC +#endif + +#ifndef GL_SGIX_fragment_lighting +#define GL_FRAGMENT_LIGHTING_SGIX 0x8400 +#define GL_FRAGMENT_COLOR_MATERIAL_SGIX 0x8401 +#define GL_FRAGMENT_COLOR_MATERIAL_FACE_SGIX 0x8402 +#define GL_FRAGMENT_COLOR_MATERIAL_PARAMETER_SGIX 0x8403 +#define GL_MAX_FRAGMENT_LIGHTS_SGIX 0x8404 +#define GL_MAX_ACTIVE_LIGHTS_SGIX 0x8405 +#define GL_CURRENT_RASTER_NORMAL_SGIX 0x8406 +#define GL_LIGHT_ENV_MODE_SGIX 0x8407 +#define GL_FRAGMENT_LIGHT_MODEL_LOCAL_VIEWER_SGIX 0x8408 +#define GL_FRAGMENT_LIGHT_MODEL_TWO_SIDE_SGIX 0x8409 +#define GL_FRAGMENT_LIGHT_MODEL_AMBIENT_SGIX 0x840A +#define GL_FRAGMENT_LIGHT_MODEL_NORMAL_INTERPOLATION_SGIX 0x840B +#define GL_FRAGMENT_LIGHT0_SGIX 0x840C +#define GL_FRAGMENT_LIGHT1_SGIX 0x840D +#define GL_FRAGMENT_LIGHT2_SGIX 0x840E +#define GL_FRAGMENT_LIGHT3_SGIX 0x840F +#define GL_FRAGMENT_LIGHT4_SGIX 0x8410 +#define GL_FRAGMENT_LIGHT5_SGIX 0x8411 +#define GL_FRAGMENT_LIGHT6_SGIX 0x8412 +#define GL_FRAGMENT_LIGHT7_SGIX 0x8413 +#endif + +#ifndef GL_IBM_rasterpos_clip +#define GL_RASTER_POSITION_UNCLIPPED_IBM 0x19262 +#endif + +#ifndef GL_HP_texture_lighting +#define GL_TEXTURE_LIGHTING_MODE_HP 0x8167 +#define GL_TEXTURE_POST_SPECULAR_HP 0x8168 +#define GL_TEXTURE_PRE_SPECULAR_HP 0x8169 +#endif + +#ifndef GL_EXT_draw_range_elements +#define GL_MAX_ELEMENTS_VERTICES_EXT 0x80E8 +#define GL_MAX_ELEMENTS_INDICES_EXT 0x80E9 +#endif + +#ifndef GL_WIN_phong_shading +#define GL_PHONG_WIN 0x80EA +#define GL_PHONG_HINT_WIN 0x80EB +#endif + +#ifndef GL_WIN_specular_fog +#define GL_FOG_SPECULAR_TEXTURE_WIN 0x80EC +#endif + +#ifndef GL_EXT_light_texture +#define GL_FRAGMENT_MATERIAL_EXT 0x8349 +#define GL_FRAGMENT_NORMAL_EXT 0x834A +#define GL_FRAGMENT_COLOR_EXT 0x834C +#define GL_ATTENUATION_EXT 0x834D +#define GL_SHADOW_ATTENUATION_EXT 0x834E +#define GL_TEXTURE_APPLICATION_MODE_EXT 0x834F +#define GL_TEXTURE_LIGHT_EXT 0x8350 +#define GL_TEXTURE_MATERIAL_FACE_EXT 0x8351 +#define GL_TEXTURE_MATERIAL_PARAMETER_EXT 0x8352 +/* reuse GL_FRAGMENT_DEPTH_EXT */ +#endif + +#ifndef GL_SGIX_blend_alpha_minmax +#define GL_ALPHA_MIN_SGIX 0x8320 +#define GL_ALPHA_MAX_SGIX 0x8321 +#endif + +#ifndef GL_SGIX_impact_pixel_texture +#define GL_PIXEL_TEX_GEN_Q_CEILING_SGIX 0x8184 +#define GL_PIXEL_TEX_GEN_Q_ROUND_SGIX 0x8185 +#define GL_PIXEL_TEX_GEN_Q_FLOOR_SGIX 0x8186 +#define GL_PIXEL_TEX_GEN_ALPHA_REPLACE_SGIX 0x8187 +#define GL_PIXEL_TEX_GEN_ALPHA_NO_REPLACE_SGIX 0x8188 +#define GL_PIXEL_TEX_GEN_ALPHA_LS_SGIX 0x8189 +#define GL_PIXEL_TEX_GEN_ALPHA_MS_SGIX 0x818A +#endif + +#ifndef GL_EXT_bgra +#define GL_BGR_EXT 0x80E0 +#define GL_BGRA_EXT 0x80E1 +#endif + +#ifndef GL_SGIX_async +#define GL_ASYNC_MARKER_SGIX 0x8329 +#endif + +#ifndef GL_SGIX_async_pixel +#define GL_ASYNC_TEX_IMAGE_SGIX 0x835C +#define GL_ASYNC_DRAW_PIXELS_SGIX 0x835D +#define GL_ASYNC_READ_PIXELS_SGIX 0x835E +#define GL_MAX_ASYNC_TEX_IMAGE_SGIX 0x835F +#define GL_MAX_ASYNC_DRAW_PIXELS_SGIX 0x8360 +#define GL_MAX_ASYNC_READ_PIXELS_SGIX 0x8361 +#endif + +#ifndef GL_SGIX_async_histogram +#define GL_ASYNC_HISTOGRAM_SGIX 0x832C +#define GL_MAX_ASYNC_HISTOGRAM_SGIX 0x832D +#endif + +#ifndef GL_INTEL_texture_scissor +#endif + +#ifndef GL_INTEL_parallel_arrays +#define GL_PARALLEL_ARRAYS_INTEL 0x83F4 +#define GL_VERTEX_ARRAY_PARALLEL_POINTERS_INTEL 0x83F5 +#define GL_NORMAL_ARRAY_PARALLEL_POINTERS_INTEL 0x83F6 +#define GL_COLOR_ARRAY_PARALLEL_POINTERS_INTEL 0x83F7 +#define GL_TEXTURE_COORD_ARRAY_PARALLEL_POINTERS_INTEL 0x83F8 +#endif + +#ifndef GL_HP_occlusion_test +#define GL_OCCLUSION_TEST_HP 0x8165 +#define GL_OCCLUSION_TEST_RESULT_HP 0x8166 +#endif + +#ifndef GL_EXT_pixel_transform +#define GL_PIXEL_TRANSFORM_2D_EXT 0x8330 +#define GL_PIXEL_MAG_FILTER_EXT 0x8331 +#define GL_PIXEL_MIN_FILTER_EXT 0x8332 +#define GL_PIXEL_CUBIC_WEIGHT_EXT 0x8333 +#define GL_CUBIC_EXT 0x8334 +#define GL_AVERAGE_EXT 0x8335 +#define GL_PIXEL_TRANSFORM_2D_STACK_DEPTH_EXT 0x8336 +#define GL_MAX_PIXEL_TRANSFORM_2D_STACK_DEPTH_EXT 0x8337 +#define GL_PIXEL_TRANSFORM_2D_MATRIX_EXT 0x8338 +#endif + +#ifndef GL_EXT_pixel_transform_color_table +#endif + +#ifndef GL_EXT_shared_texture_palette +#define GL_SHARED_TEXTURE_PALETTE_EXT 0x81FB +#endif + +#ifndef GL_EXT_separate_specular_color +#define GL_LIGHT_MODEL_COLOR_CONTROL_EXT 0x81F8 +#define GL_SINGLE_COLOR_EXT 0x81F9 +#define GL_SEPARATE_SPECULAR_COLOR_EXT 0x81FA +#endif + +#ifndef GL_EXT_secondary_color +#define GL_COLOR_SUM_EXT 0x8458 +#define GL_CURRENT_SECONDARY_COLOR_EXT 0x8459 +#define GL_SECONDARY_COLOR_ARRAY_SIZE_EXT 0x845A +#define GL_SECONDARY_COLOR_ARRAY_TYPE_EXT 0x845B +#define GL_SECONDARY_COLOR_ARRAY_STRIDE_EXT 0x845C +#define GL_SECONDARY_COLOR_ARRAY_POINTER_EXT 0x845D +#define GL_SECONDARY_COLOR_ARRAY_EXT 0x845E +#endif + +#ifndef GL_EXT_texture_perturb_normal +#define GL_PERTURB_EXT 0x85AE +#define GL_TEXTURE_NORMAL_EXT 0x85AF +#endif + +#ifndef GL_EXT_multi_draw_arrays +#endif + +#ifndef GL_EXT_fog_coord +#define GL_FOG_COORDINATE_SOURCE_EXT 0x8450 +#define GL_FOG_COORDINATE_EXT 0x8451 +#define GL_FRAGMENT_DEPTH_EXT 0x8452 +#define GL_CURRENT_FOG_COORDINATE_EXT 0x8453 +#define GL_FOG_COORDINATE_ARRAY_TYPE_EXT 0x8454 +#define GL_FOG_COORDINATE_ARRAY_STRIDE_EXT 0x8455 +#define GL_FOG_COORDINATE_ARRAY_POINTER_EXT 0x8456 +#define GL_FOG_COORDINATE_ARRAY_EXT 0x8457 +#endif + +#ifndef GL_REND_screen_coordinates +#define GL_SCREEN_COORDINATES_REND 0x8490 +#define GL_INVERTED_SCREEN_W_REND 0x8491 +#endif + +#ifndef GL_EXT_coordinate_frame +#define GL_TANGENT_ARRAY_EXT 0x8439 +#define GL_BINORMAL_ARRAY_EXT 0x843A +#define GL_CURRENT_TANGENT_EXT 0x843B +#define GL_CURRENT_BINORMAL_EXT 0x843C +#define GL_TANGENT_ARRAY_TYPE_EXT 0x843E +#define GL_TANGENT_ARRAY_STRIDE_EXT 0x843F +#define GL_BINORMAL_ARRAY_TYPE_EXT 0x8440 +#define GL_BINORMAL_ARRAY_STRIDE_EXT 0x8441 +#define GL_TANGENT_ARRAY_POINTER_EXT 0x8442 +#define GL_BINORMAL_ARRAY_POINTER_EXT 0x8443 +#define GL_MAP1_TANGENT_EXT 0x8444 +#define GL_MAP2_TANGENT_EXT 0x8445 +#define GL_MAP1_BINORMAL_EXT 0x8446 +#define GL_MAP2_BINORMAL_EXT 0x8447 +#endif + +#ifndef GL_EXT_texture_env_combine +#define GL_COMBINE_EXT 0x8570 +#define GL_COMBINE_RGB_EXT 0x8571 +#define GL_COMBINE_ALPHA_EXT 0x8572 +#define GL_RGB_SCALE_EXT 0x8573 +#define GL_ADD_SIGNED_EXT 0x8574 +#define GL_INTERPOLATE_EXT 0x8575 +#define GL_CONSTANT_EXT 0x8576 +#define GL_PRIMARY_COLOR_EXT 0x8577 +#define GL_PREVIOUS_EXT 0x8578 +#define GL_SOURCE0_RGB_EXT 0x8580 +#define GL_SOURCE1_RGB_EXT 0x8581 +#define GL_SOURCE2_RGB_EXT 0x8582 +#define GL_SOURCE0_ALPHA_EXT 0x8588 +#define GL_SOURCE1_ALPHA_EXT 0x8589 +#define GL_SOURCE2_ALPHA_EXT 0x858A +#define GL_OPERAND0_RGB_EXT 0x8590 +#define GL_OPERAND1_RGB_EXT 0x8591 +#define GL_OPERAND2_RGB_EXT 0x8592 +#define GL_OPERAND0_ALPHA_EXT 0x8598 +#define GL_OPERAND1_ALPHA_EXT 0x8599 +#define GL_OPERAND2_ALPHA_EXT 0x859A +#endif + +#ifndef GL_APPLE_specular_vector +#define GL_LIGHT_MODEL_SPECULAR_VECTOR_APPLE 0x85B0 +#endif + +#ifndef GL_APPLE_transform_hint +#define GL_TRANSFORM_HINT_APPLE 0x85B1 +#endif + +#ifndef GL_SGIX_fog_scale +#define GL_FOG_SCALE_SGIX 0x81FC +#define GL_FOG_SCALE_VALUE_SGIX 0x81FD +#endif + +#ifndef GL_SUNX_constant_data +#define GL_UNPACK_CONSTANT_DATA_SUNX 0x81D5 +#define GL_TEXTURE_CONSTANT_DATA_SUNX 0x81D6 +#endif + +#ifndef GL_SUN_global_alpha +#define GL_GLOBAL_ALPHA_SUN 0x81D9 +#define GL_GLOBAL_ALPHA_FACTOR_SUN 0x81DA +#endif + +#ifndef GL_SUN_triangle_list +#define GL_RESTART_SUN 0x0001 +#define GL_REPLACE_MIDDLE_SUN 0x0002 +#define GL_REPLACE_OLDEST_SUN 0x0003 +#define GL_TRIANGLE_LIST_SUN 0x81D7 +#define GL_REPLACEMENT_CODE_SUN 0x81D8 +#define GL_REPLACEMENT_CODE_ARRAY_SUN 0x85C0 +#define GL_REPLACEMENT_CODE_ARRAY_TYPE_SUN 0x85C1 +#define GL_REPLACEMENT_CODE_ARRAY_STRIDE_SUN 0x85C2 +#define GL_REPLACEMENT_CODE_ARRAY_POINTER_SUN 0x85C3 +#define GL_R1UI_V3F_SUN 0x85C4 +#define GL_R1UI_C4UB_V3F_SUN 0x85C5 +#define GL_R1UI_C3F_V3F_SUN 0x85C6 +#define GL_R1UI_N3F_V3F_SUN 0x85C7 +#define GL_R1UI_C4F_N3F_V3F_SUN 0x85C8 +#define GL_R1UI_T2F_V3F_SUN 0x85C9 +#define GL_R1UI_T2F_N3F_V3F_SUN 0x85CA +#define GL_R1UI_T2F_C4F_N3F_V3F_SUN 0x85CB +#endif + +#ifndef GL_SUN_vertex +#endif + +#ifndef GL_EXT_blend_func_separate +#define GL_BLEND_DST_RGB_EXT 0x80C8 +#define GL_BLEND_SRC_RGB_EXT 0x80C9 +#define GL_BLEND_DST_ALPHA_EXT 0x80CA +#define GL_BLEND_SRC_ALPHA_EXT 0x80CB +#endif + +#ifndef GL_INGR_color_clamp +#define GL_RED_MIN_CLAMP_INGR 0x8560 +#define GL_GREEN_MIN_CLAMP_INGR 0x8561 +#define GL_BLUE_MIN_CLAMP_INGR 0x8562 +#define GL_ALPHA_MIN_CLAMP_INGR 0x8563 +#define GL_RED_MAX_CLAMP_INGR 0x8564 +#define GL_GREEN_MAX_CLAMP_INGR 0x8565 +#define GL_BLUE_MAX_CLAMP_INGR 0x8566 +#define GL_ALPHA_MAX_CLAMP_INGR 0x8567 +#endif + +#ifndef GL_INGR_interlace_read +#define GL_INTERLACE_READ_INGR 0x8568 +#endif + +#ifndef GL_EXT_stencil_wrap +#define GL_INCR_WRAP_EXT 0x8507 +#define GL_DECR_WRAP_EXT 0x8508 +#endif + +#ifndef GL_EXT_422_pixels +#define GL_422_EXT 0x80CC +#define GL_422_REV_EXT 0x80CD +#define GL_422_AVERAGE_EXT 0x80CE +#define GL_422_REV_AVERAGE_EXT 0x80CF +#endif + +#ifndef GL_NV_texgen_reflection +#define GL_NORMAL_MAP_NV 0x8511 +#define GL_REFLECTION_MAP_NV 0x8512 +#endif + +#ifndef GL_EXT_texture_cube_map +#define GL_NORMAL_MAP_EXT 0x8511 +#define GL_REFLECTION_MAP_EXT 0x8512 +#define GL_TEXTURE_CUBE_MAP_EXT 0x8513 +#define GL_TEXTURE_BINDING_CUBE_MAP_EXT 0x8514 +#define GL_TEXTURE_CUBE_MAP_POSITIVE_X_EXT 0x8515 +#define GL_TEXTURE_CUBE_MAP_NEGATIVE_X_EXT 0x8516 +#define GL_TEXTURE_CUBE_MAP_POSITIVE_Y_EXT 0x8517 +#define GL_TEXTURE_CUBE_MAP_NEGATIVE_Y_EXT 0x8518 +#define GL_TEXTURE_CUBE_MAP_POSITIVE_Z_EXT 0x8519 +#define GL_TEXTURE_CUBE_MAP_NEGATIVE_Z_EXT 0x851A +#define GL_PROXY_TEXTURE_CUBE_MAP_EXT 0x851B +#define GL_MAX_CUBE_MAP_TEXTURE_SIZE_EXT 0x851C +#endif + +#ifndef GL_SUN_convolution_border_modes +#define GL_WRAP_BORDER_SUN 0x81D4 +#endif + +#ifndef GL_EXT_texture_env_add +#endif + +#ifndef GL_EXT_texture_lod_bias +#define GL_MAX_TEXTURE_LOD_BIAS_EXT 0x84FD +#define GL_TEXTURE_FILTER_CONTROL_EXT 0x8500 +#define GL_TEXTURE_LOD_BIAS_EXT 0x8501 +#endif + +#ifndef GL_EXT_texture_filter_anisotropic +#define GL_TEXTURE_MAX_ANISOTROPY_EXT 0x84FE +#define GL_MAX_TEXTURE_MAX_ANISOTROPY_EXT 0x84FF +#endif + +#ifndef GL_EXT_vertex_weighting +#define GL_MODELVIEW0_STACK_DEPTH_EXT GL_MODELVIEW_STACK_DEPTH +#define GL_MODELVIEW1_STACK_DEPTH_EXT 0x8502 +#define GL_MODELVIEW0_MATRIX_EXT GL_MODELVIEW_MATRIX +#define GL_MODELVIEW1_MATRIX_EXT 0x8506 +#define GL_VERTEX_WEIGHTING_EXT 0x8509 +#define GL_MODELVIEW0_EXT GL_MODELVIEW +#define GL_MODELVIEW1_EXT 0x850A +#define GL_CURRENT_VERTEX_WEIGHT_EXT 0x850B +#define GL_VERTEX_WEIGHT_ARRAY_EXT 0x850C +#define GL_VERTEX_WEIGHT_ARRAY_SIZE_EXT 0x850D +#define GL_VERTEX_WEIGHT_ARRAY_TYPE_EXT 0x850E +#define GL_VERTEX_WEIGHT_ARRAY_STRIDE_EXT 0x850F +#define GL_VERTEX_WEIGHT_ARRAY_POINTER_EXT 0x8510 +#endif + +#ifndef GL_NV_light_max_exponent +#define GL_MAX_SHININESS_NV 0x8504 +#define GL_MAX_SPOT_EXPONENT_NV 0x8505 +#endif + +#ifndef GL_NV_vertex_array_range +#define GL_VERTEX_ARRAY_RANGE_NV 0x851D +#define GL_VERTEX_ARRAY_RANGE_LENGTH_NV 0x851E +#define GL_VERTEX_ARRAY_RANGE_VALID_NV 0x851F +#define GL_MAX_VERTEX_ARRAY_RANGE_ELEMENT_NV 0x8520 +#define GL_VERTEX_ARRAY_RANGE_POINTER_NV 0x8521 +#endif + +#ifndef GL_NV_register_combiners +#define GL_REGISTER_COMBINERS_NV 0x8522 +#define GL_VARIABLE_A_NV 0x8523 +#define GL_VARIABLE_B_NV 0x8524 +#define GL_VARIABLE_C_NV 0x8525 +#define GL_VARIABLE_D_NV 0x8526 +#define GL_VARIABLE_E_NV 0x8527 +#define GL_VARIABLE_F_NV 0x8528 +#define GL_VARIABLE_G_NV 0x8529 +#define GL_CONSTANT_COLOR0_NV 0x852A +#define GL_CONSTANT_COLOR1_NV 0x852B +#define GL_PRIMARY_COLOR_NV 0x852C +#define GL_SECONDARY_COLOR_NV 0x852D +#define GL_SPARE0_NV 0x852E +#define GL_SPARE1_NV 0x852F +#define GL_DISCARD_NV 0x8530 +#define GL_E_TIMES_F_NV 0x8531 +#define GL_SPARE0_PLUS_SECONDARY_COLOR_NV 0x8532 +#define GL_UNSIGNED_IDENTITY_NV 0x8536 +#define GL_UNSIGNED_INVERT_NV 0x8537 +#define GL_EXPAND_NORMAL_NV 0x8538 +#define GL_EXPAND_NEGATE_NV 0x8539 +#define GL_HALF_BIAS_NORMAL_NV 0x853A +#define GL_HALF_BIAS_NEGATE_NV 0x853B +#define GL_SIGNED_IDENTITY_NV 0x853C +#define GL_SIGNED_NEGATE_NV 0x853D +#define GL_SCALE_BY_TWO_NV 0x853E +#define GL_SCALE_BY_FOUR_NV 0x853F +#define GL_SCALE_BY_ONE_HALF_NV 0x8540 +#define GL_BIAS_BY_NEGATIVE_ONE_HALF_NV 0x8541 +#define GL_COMBINER_INPUT_NV 0x8542 +#define GL_COMBINER_MAPPING_NV 0x8543 +#define GL_COMBINER_COMPONENT_USAGE_NV 0x8544 +#define GL_COMBINER_AB_DOT_PRODUCT_NV 0x8545 +#define GL_COMBINER_CD_DOT_PRODUCT_NV 0x8546 +#define GL_COMBINER_MUX_SUM_NV 0x8547 +#define GL_COMBINER_SCALE_NV 0x8548 +#define GL_COMBINER_BIAS_NV 0x8549 +#define GL_COMBINER_AB_OUTPUT_NV 0x854A +#define GL_COMBINER_CD_OUTPUT_NV 0x854B +#define GL_COMBINER_SUM_OUTPUT_NV 0x854C +#define GL_MAX_GENERAL_COMBINERS_NV 0x854D +#define GL_NUM_GENERAL_COMBINERS_NV 0x854E +#define GL_COLOR_SUM_CLAMP_NV 0x854F +#define GL_COMBINER0_NV 0x8550 +#define GL_COMBINER1_NV 0x8551 +#define GL_COMBINER2_NV 0x8552 +#define GL_COMBINER3_NV 0x8553 +#define GL_COMBINER4_NV 0x8554 +#define GL_COMBINER5_NV 0x8555 +#define GL_COMBINER6_NV 0x8556 +#define GL_COMBINER7_NV 0x8557 +/* reuse GL_TEXTURE0_ARB */ +/* reuse GL_TEXTURE1_ARB */ +/* reuse GL_ZERO */ +/* reuse GL_NONE */ +/* reuse GL_FOG */ +#endif + +#ifndef GL_NV_fog_distance +#define GL_FOG_DISTANCE_MODE_NV 0x855A +#define GL_EYE_RADIAL_NV 0x855B +#define GL_EYE_PLANE_ABSOLUTE_NV 0x855C +/* reuse GL_EYE_PLANE */ +#endif + +#ifndef GL_NV_texgen_emboss +#define GL_EMBOSS_LIGHT_NV 0x855D +#define GL_EMBOSS_CONSTANT_NV 0x855E +#define GL_EMBOSS_MAP_NV 0x855F +#endif + +#ifndef GL_NV_blend_square +#endif + +#ifndef GL_NV_texture_env_combine4 +#define GL_COMBINE4_NV 0x8503 +#define GL_SOURCE3_RGB_NV 0x8583 +#define GL_SOURCE3_ALPHA_NV 0x858B +#define GL_OPERAND3_RGB_NV 0x8593 +#define GL_OPERAND3_ALPHA_NV 0x859B +#endif + +#ifndef GL_MESA_resize_buffers +#endif + +#ifndef GL_MESA_window_pos +#endif + +#ifndef GL_EXT_texture_compression_s3tc +#define GL_COMPRESSED_RGB_S3TC_DXT1_EXT 0x83F0 +#define GL_COMPRESSED_RGBA_S3TC_DXT1_EXT 0x83F1 +#define GL_COMPRESSED_RGBA_S3TC_DXT3_EXT 0x83F2 +#define GL_COMPRESSED_RGBA_S3TC_DXT5_EXT 0x83F3 +#endif + +#ifndef GL_IBM_cull_vertex +#define GL_CULL_VERTEX_IBM 103050 +#endif + +#ifndef GL_IBM_multimode_draw_arrays +#endif + +#ifndef GL_IBM_vertex_array_lists +#define GL_VERTEX_ARRAY_LIST_IBM 103070 +#define GL_NORMAL_ARRAY_LIST_IBM 103071 +#define GL_COLOR_ARRAY_LIST_IBM 103072 +#define GL_INDEX_ARRAY_LIST_IBM 103073 +#define GL_TEXTURE_COORD_ARRAY_LIST_IBM 103074 +#define GL_EDGE_FLAG_ARRAY_LIST_IBM 103075 +#define GL_FOG_COORDINATE_ARRAY_LIST_IBM 103076 +#define GL_SECONDARY_COLOR_ARRAY_LIST_IBM 103077 +#define GL_VERTEX_ARRAY_LIST_STRIDE_IBM 103080 +#define GL_NORMAL_ARRAY_LIST_STRIDE_IBM 103081 +#define GL_COLOR_ARRAY_LIST_STRIDE_IBM 103082 +#define GL_INDEX_ARRAY_LIST_STRIDE_IBM 103083 +#define GL_TEXTURE_COORD_ARRAY_LIST_STRIDE_IBM 103084 +#define GL_EDGE_FLAG_ARRAY_LIST_STRIDE_IBM 103085 +#define GL_FOG_COORDINATE_ARRAY_LIST_STRIDE_IBM 103086 +#define GL_SECONDARY_COLOR_ARRAY_LIST_STRIDE_IBM 103087 +#endif + +#ifndef GL_SGIX_subsample +#define GL_PACK_SUBSAMPLE_RATE_SGIX 0x85A0 +#define GL_UNPACK_SUBSAMPLE_RATE_SGIX 0x85A1 +#define GL_PIXEL_SUBSAMPLE_4444_SGIX 0x85A2 +#define GL_PIXEL_SUBSAMPLE_2424_SGIX 0x85A3 +#define GL_PIXEL_SUBSAMPLE_4242_SGIX 0x85A4 +#endif + +#ifndef GL_SGIX_ycrcb_subsample +#endif + +#ifndef GL_SGIX_ycrcba +#define GL_YCRCB_SGIX 0x8318 +#define GL_YCRCBA_SGIX 0x8319 +#endif + +#ifndef GL_SGI_depth_pass_instrument +#define GL_DEPTH_PASS_INSTRUMENT_SGIX 0x8310 +#define GL_DEPTH_PASS_INSTRUMENT_COUNTERS_SGIX 0x8311 +#define GL_DEPTH_PASS_INSTRUMENT_MAX_SGIX 0x8312 +#endif + +#ifndef GL_3DFX_texture_compression_FXT1 +#define GL_COMPRESSED_RGB_FXT1_3DFX 0x86B0 +#define GL_COMPRESSED_RGBA_FXT1_3DFX 0x86B1 +#endif + +#ifndef GL_3DFX_multisample +#define GL_MULTISAMPLE_3DFX 0x86B2 +#define GL_SAMPLE_BUFFERS_3DFX 0x86B3 +#define GL_SAMPLES_3DFX 0x86B4 +#define GL_MULTISAMPLE_BIT_3DFX 0x20000000 +#endif + +#ifndef GL_3DFX_tbuffer +#endif + +#ifndef GL_EXT_multisample +#define GL_MULTISAMPLE_EXT 0x809D +#define GL_SAMPLE_ALPHA_TO_MASK_EXT 0x809E +#define GL_SAMPLE_ALPHA_TO_ONE_EXT 0x809F +#define GL_SAMPLE_MASK_EXT 0x80A0 +#define GL_1PASS_EXT 0x80A1 +#define GL_2PASS_0_EXT 0x80A2 +#define GL_2PASS_1_EXT 0x80A3 +#define GL_4PASS_0_EXT 0x80A4 +#define GL_4PASS_1_EXT 0x80A5 +#define GL_4PASS_2_EXT 0x80A6 +#define GL_4PASS_3_EXT 0x80A7 +#define GL_SAMPLE_BUFFERS_EXT 0x80A8 +#define GL_SAMPLES_EXT 0x80A9 +#define GL_SAMPLE_MASK_VALUE_EXT 0x80AA +#define GL_SAMPLE_MASK_INVERT_EXT 0x80AB +#define GL_SAMPLE_PATTERN_EXT 0x80AC +#define GL_MULTISAMPLE_BIT_EXT 0x20000000 +#endif + +#ifndef GL_SGIX_vertex_preclip +#define GL_VERTEX_PRECLIP_SGIX 0x83EE +#define GL_VERTEX_PRECLIP_HINT_SGIX 0x83EF +#endif + +#ifndef GL_SGIX_convolution_accuracy +#define GL_CONVOLUTION_HINT_SGIX 0x8316 +#endif + +#ifndef GL_SGIX_resample +#define GL_PACK_RESAMPLE_SGIX 0x842C +#define GL_UNPACK_RESAMPLE_SGIX 0x842D +#define GL_RESAMPLE_REPLICATE_SGIX 0x842E +#define GL_RESAMPLE_ZERO_FILL_SGIX 0x842F +#define GL_RESAMPLE_DECIMATE_SGIX 0x8430 +#endif + +#ifndef GL_SGIS_point_line_texgen +#define GL_EYE_DISTANCE_TO_POINT_SGIS 0x81F0 +#define GL_OBJECT_DISTANCE_TO_POINT_SGIS 0x81F1 +#define GL_EYE_DISTANCE_TO_LINE_SGIS 0x81F2 +#define GL_OBJECT_DISTANCE_TO_LINE_SGIS 0x81F3 +#define GL_EYE_POINT_SGIS 0x81F4 +#define GL_OBJECT_POINT_SGIS 0x81F5 +#define GL_EYE_LINE_SGIS 0x81F6 +#define GL_OBJECT_LINE_SGIS 0x81F7 +#endif + +#ifndef GL_SGIS_texture_color_mask +#define GL_TEXTURE_COLOR_WRITEMASK_SGIS 0x81EF +#endif + +#ifndef GL_EXT_texture_env_dot3 +#define GL_DOT3_RGB_EXT 0x8740 +#define GL_DOT3_RGBA_EXT 0x8741 +#endif + +#ifndef GL_ATI_texture_mirror_once +#define GL_MIRROR_CLAMP_ATI 0x8742 +#define GL_MIRROR_CLAMP_TO_EDGE_ATI 0x8743 +#endif + +#ifndef GL_NV_fence +#define GL_ALL_COMPLETED_NV 0x84F2 +#define GL_FENCE_STATUS_NV 0x84F3 +#define GL_FENCE_CONDITION_NV 0x84F4 +#endif + +#ifndef GL_IBM_texture_mirrored_repeat +#define GL_MIRRORED_REPEAT_IBM 0x8370 +#endif + +#ifndef GL_NV_evaluators +#define GL_EVAL_2D_NV 0x86C0 +#define GL_EVAL_TRIANGULAR_2D_NV 0x86C1 +#define GL_MAP_TESSELLATION_NV 0x86C2 +#define GL_MAP_ATTRIB_U_ORDER_NV 0x86C3 +#define GL_MAP_ATTRIB_V_ORDER_NV 0x86C4 +#define GL_EVAL_FRACTIONAL_TESSELLATION_NV 0x86C5 +#define GL_EVAL_VERTEX_ATTRIB0_NV 0x86C6 +#define GL_EVAL_VERTEX_ATTRIB1_NV 0x86C7 +#define GL_EVAL_VERTEX_ATTRIB2_NV 0x86C8 +#define GL_EVAL_VERTEX_ATTRIB3_NV 0x86C9 +#define GL_EVAL_VERTEX_ATTRIB4_NV 0x86CA +#define GL_EVAL_VERTEX_ATTRIB5_NV 0x86CB +#define GL_EVAL_VERTEX_ATTRIB6_NV 0x86CC +#define GL_EVAL_VERTEX_ATTRIB7_NV 0x86CD +#define GL_EVAL_VERTEX_ATTRIB8_NV 0x86CE +#define GL_EVAL_VERTEX_ATTRIB9_NV 0x86CF +#define GL_EVAL_VERTEX_ATTRIB10_NV 0x86D0 +#define GL_EVAL_VERTEX_ATTRIB11_NV 0x86D1 +#define GL_EVAL_VERTEX_ATTRIB12_NV 0x86D2 +#define GL_EVAL_VERTEX_ATTRIB13_NV 0x86D3 +#define GL_EVAL_VERTEX_ATTRIB14_NV 0x86D4 +#define GL_EVAL_VERTEX_ATTRIB15_NV 0x86D5 +#define GL_MAX_MAP_TESSELLATION_NV 0x86D6 +#define GL_MAX_RATIONAL_EVAL_ORDER_NV 0x86D7 +#endif + +#ifndef GL_NV_packed_depth_stencil +#define GL_DEPTH_STENCIL_NV 0x84F9 +#define GL_UNSIGNED_INT_24_8_NV 0x84FA +#endif + +#ifndef GL_NV_register_combiners2 +#define GL_PER_STAGE_CONSTANTS_NV 0x8535 +#endif + +#ifndef GL_NV_texture_compression_vtc +#endif + +#ifndef GL_NV_texture_rectangle +#define GL_TEXTURE_RECTANGLE_NV 0x84F5 +#define GL_TEXTURE_BINDING_RECTANGLE_NV 0x84F6 +#define GL_PROXY_TEXTURE_RECTANGLE_NV 0x84F7 +#define GL_MAX_RECTANGLE_TEXTURE_SIZE_NV 0x84F8 +#endif + +#ifndef GL_NV_texture_shader +#define GL_OFFSET_TEXTURE_RECTANGLE_NV 0x864C +#define GL_OFFSET_TEXTURE_RECTANGLE_SCALE_NV 0x864D +#define GL_DOT_PRODUCT_TEXTURE_RECTANGLE_NV 0x864E +#define GL_RGBA_UNSIGNED_DOT_PRODUCT_MAPPING_NV 0x86D9 +#define GL_UNSIGNED_INT_S8_S8_8_8_NV 0x86DA +#define GL_UNSIGNED_INT_8_8_S8_S8_REV_NV 0x86DB +#define GL_DSDT_MAG_INTENSITY_NV 0x86DC +#define GL_SHADER_CONSISTENT_NV 0x86DD +#define GL_TEXTURE_SHADER_NV 0x86DE +#define GL_SHADER_OPERATION_NV 0x86DF +#define GL_CULL_MODES_NV 0x86E0 +#define GL_OFFSET_TEXTURE_MATRIX_NV 0x86E1 +#define GL_OFFSET_TEXTURE_SCALE_NV 0x86E2 +#define GL_OFFSET_TEXTURE_BIAS_NV 0x86E3 +#define GL_OFFSET_TEXTURE_2D_MATRIX_NV GL_OFFSET_TEXTURE_MATRIX_NV +#define GL_OFFSET_TEXTURE_2D_SCALE_NV GL_OFFSET_TEXTURE_SCALE_NV +#define GL_OFFSET_TEXTURE_2D_BIAS_NV GL_OFFSET_TEXTURE_BIAS_NV +#define GL_PREVIOUS_TEXTURE_INPUT_NV 0x86E4 +#define GL_CONST_EYE_NV 0x86E5 +#define GL_PASS_THROUGH_NV 0x86E6 +#define GL_CULL_FRAGMENT_NV 0x86E7 +#define GL_OFFSET_TEXTURE_2D_NV 0x86E8 +#define GL_DEPENDENT_AR_TEXTURE_2D_NV 0x86E9 +#define GL_DEPENDENT_GB_TEXTURE_2D_NV 0x86EA +#define GL_DOT_PRODUCT_NV 0x86EC +#define GL_DOT_PRODUCT_DEPTH_REPLACE_NV 0x86ED +#define GL_DOT_PRODUCT_TEXTURE_2D_NV 0x86EE +#define GL_DOT_PRODUCT_TEXTURE_CUBE_MAP_NV 0x86F0 +#define GL_DOT_PRODUCT_DIFFUSE_CUBE_MAP_NV 0x86F1 +#define GL_DOT_PRODUCT_REFLECT_CUBE_MAP_NV 0x86F2 +#define GL_DOT_PRODUCT_CONST_EYE_REFLECT_CUBE_MAP_NV 0x86F3 +#define GL_HILO_NV 0x86F4 +#define GL_DSDT_NV 0x86F5 +#define GL_DSDT_MAG_NV 0x86F6 +#define GL_DSDT_MAG_VIB_NV 0x86F7 +#define GL_HILO16_NV 0x86F8 +#define GL_SIGNED_HILO_NV 0x86F9 +#define GL_SIGNED_HILO16_NV 0x86FA +#define GL_SIGNED_RGBA_NV 0x86FB +#define GL_SIGNED_RGBA8_NV 0x86FC +#define GL_SIGNED_RGB_NV 0x86FE +#define GL_SIGNED_RGB8_NV 0x86FF +#define GL_SIGNED_LUMINANCE_NV 0x8701 +#define GL_SIGNED_LUMINANCE8_NV 0x8702 +#define GL_SIGNED_LUMINANCE_ALPHA_NV 0x8703 +#define GL_SIGNED_LUMINANCE8_ALPHA8_NV 0x8704 +#define GL_SIGNED_ALPHA_NV 0x8705 +#define GL_SIGNED_ALPHA8_NV 0x8706 +#define GL_SIGNED_INTENSITY_NV 0x8707 +#define GL_SIGNED_INTENSITY8_NV 0x8708 +#define GL_DSDT8_NV 0x8709 +#define GL_DSDT8_MAG8_NV 0x870A +#define GL_DSDT8_MAG8_INTENSITY8_NV 0x870B +#define GL_SIGNED_RGB_UNSIGNED_ALPHA_NV 0x870C +#define GL_SIGNED_RGB8_UNSIGNED_ALPHA8_NV 0x870D +#define GL_HI_SCALE_NV 0x870E +#define GL_LO_SCALE_NV 0x870F +#define GL_DS_SCALE_NV 0x8710 +#define GL_DT_SCALE_NV 0x8711 +#define GL_MAGNITUDE_SCALE_NV 0x8712 +#define GL_VIBRANCE_SCALE_NV 0x8713 +#define GL_HI_BIAS_NV 0x8714 +#define GL_LO_BIAS_NV 0x8715 +#define GL_DS_BIAS_NV 0x8716 +#define GL_DT_BIAS_NV 0x8717 +#define GL_MAGNITUDE_BIAS_NV 0x8718 +#define GL_VIBRANCE_BIAS_NV 0x8719 +#define GL_TEXTURE_BORDER_VALUES_NV 0x871A +#define GL_TEXTURE_HI_SIZE_NV 0x871B +#define GL_TEXTURE_LO_SIZE_NV 0x871C +#define GL_TEXTURE_DS_SIZE_NV 0x871D +#define GL_TEXTURE_DT_SIZE_NV 0x871E +#define GL_TEXTURE_MAG_SIZE_NV 0x871F +#endif + +#ifndef GL_NV_texture_shader2 +#define GL_DOT_PRODUCT_TEXTURE_3D_NV 0x86EF +#endif + +#ifndef GL_NV_vertex_array_range2 +#define GL_VERTEX_ARRAY_RANGE_WITHOUT_FLUSH_NV 0x8533 +#endif + +#ifndef GL_NV_vertex_program +#define GL_VERTEX_PROGRAM_NV 0x8620 +#define GL_VERTEX_STATE_PROGRAM_NV 0x8621 +#define GL_ATTRIB_ARRAY_SIZE_NV 0x8623 +#define GL_ATTRIB_ARRAY_STRIDE_NV 0x8624 +#define GL_ATTRIB_ARRAY_TYPE_NV 0x8625 +#define GL_CURRENT_ATTRIB_NV 0x8626 +#define GL_PROGRAM_LENGTH_NV 0x8627 +#define GL_PROGRAM_STRING_NV 0x8628 +#define GL_MODELVIEW_PROJECTION_NV 0x8629 +#define GL_IDENTITY_NV 0x862A +#define GL_INVERSE_NV 0x862B +#define GL_TRANSPOSE_NV 0x862C +#define GL_INVERSE_TRANSPOSE_NV 0x862D +#define GL_MAX_TRACK_MATRIX_STACK_DEPTH_NV 0x862E +#define GL_MAX_TRACK_MATRICES_NV 0x862F +#define GL_MATRIX0_NV 0x8630 +#define GL_MATRIX1_NV 0x8631 +#define GL_MATRIX2_NV 0x8632 +#define GL_MATRIX3_NV 0x8633 +#define GL_MATRIX4_NV 0x8634 +#define GL_MATRIX5_NV 0x8635 +#define GL_MATRIX6_NV 0x8636 +#define GL_MATRIX7_NV 0x8637 +#define GL_CURRENT_MATRIX_STACK_DEPTH_NV 0x8640 +#define GL_CURRENT_MATRIX_NV 0x8641 +#define GL_VERTEX_PROGRAM_POINT_SIZE_NV 0x8642 +#define GL_VERTEX_PROGRAM_TWO_SIDE_NV 0x8643 +#define GL_PROGRAM_PARAMETER_NV 0x8644 +#define GL_ATTRIB_ARRAY_POINTER_NV 0x8645 +#define GL_PROGRAM_TARGET_NV 0x8646 +#define GL_PROGRAM_RESIDENT_NV 0x8647 +#define GL_TRACK_MATRIX_NV 0x8648 +#define GL_TRACK_MATRIX_TRANSFORM_NV 0x8649 +#define GL_VERTEX_PROGRAM_BINDING_NV 0x864A +#define GL_PROGRAM_ERROR_POSITION_NV 0x864B +#define GL_VERTEX_ATTRIB_ARRAY0_NV 0x8650 +#define GL_VERTEX_ATTRIB_ARRAY1_NV 0x8651 +#define GL_VERTEX_ATTRIB_ARRAY2_NV 0x8652 +#define GL_VERTEX_ATTRIB_ARRAY3_NV 0x8653 +#define GL_VERTEX_ATTRIB_ARRAY4_NV 0x8654 +#define GL_VERTEX_ATTRIB_ARRAY5_NV 0x8655 +#define GL_VERTEX_ATTRIB_ARRAY6_NV 0x8656 +#define GL_VERTEX_ATTRIB_ARRAY7_NV 0x8657 +#define GL_VERTEX_ATTRIB_ARRAY8_NV 0x8658 +#define GL_VERTEX_ATTRIB_ARRAY9_NV 0x8659 +#define GL_VERTEX_ATTRIB_ARRAY10_NV 0x865A +#define GL_VERTEX_ATTRIB_ARRAY11_NV 0x865B +#define GL_VERTEX_ATTRIB_ARRAY12_NV 0x865C +#define GL_VERTEX_ATTRIB_ARRAY13_NV 0x865D +#define GL_VERTEX_ATTRIB_ARRAY14_NV 0x865E +#define GL_VERTEX_ATTRIB_ARRAY15_NV 0x865F +#define GL_MAP1_VERTEX_ATTRIB0_4_NV 0x8660 +#define GL_MAP1_VERTEX_ATTRIB1_4_NV 0x8661 +#define GL_MAP1_VERTEX_ATTRIB2_4_NV 0x8662 +#define GL_MAP1_VERTEX_ATTRIB3_4_NV 0x8663 +#define GL_MAP1_VERTEX_ATTRIB4_4_NV 0x8664 +#define GL_MAP1_VERTEX_ATTRIB5_4_NV 0x8665 +#define GL_MAP1_VERTEX_ATTRIB6_4_NV 0x8666 +#define GL_MAP1_VERTEX_ATTRIB7_4_NV 0x8667 +#define GL_MAP1_VERTEX_ATTRIB8_4_NV 0x8668 +#define GL_MAP1_VERTEX_ATTRIB9_4_NV 0x8669 +#define GL_MAP1_VERTEX_ATTRIB10_4_NV 0x866A +#define GL_MAP1_VERTEX_ATTRIB11_4_NV 0x866B +#define GL_MAP1_VERTEX_ATTRIB12_4_NV 0x866C +#define GL_MAP1_VERTEX_ATTRIB13_4_NV 0x866D +#define GL_MAP1_VERTEX_ATTRIB14_4_NV 0x866E +#define GL_MAP1_VERTEX_ATTRIB15_4_NV 0x866F +#define GL_MAP2_VERTEX_ATTRIB0_4_NV 0x8670 +#define GL_MAP2_VERTEX_ATTRIB1_4_NV 0x8671 +#define GL_MAP2_VERTEX_ATTRIB2_4_NV 0x8672 +#define GL_MAP2_VERTEX_ATTRIB3_4_NV 0x8673 +#define GL_MAP2_VERTEX_ATTRIB4_4_NV 0x8674 +#define GL_MAP2_VERTEX_ATTRIB5_4_NV 0x8675 +#define GL_MAP2_VERTEX_ATTRIB6_4_NV 0x8676 +#define GL_MAP2_VERTEX_ATTRIB7_4_NV 0x8677 +#define GL_MAP2_VERTEX_ATTRIB8_4_NV 0x8678 +#define GL_MAP2_VERTEX_ATTRIB9_4_NV 0x8679 +#define GL_MAP2_VERTEX_ATTRIB10_4_NV 0x867A +#define GL_MAP2_VERTEX_ATTRIB11_4_NV 0x867B +#define GL_MAP2_VERTEX_ATTRIB12_4_NV 0x867C +#define GL_MAP2_VERTEX_ATTRIB13_4_NV 0x867D +#define GL_MAP2_VERTEX_ATTRIB14_4_NV 0x867E +#define GL_MAP2_VERTEX_ATTRIB15_4_NV 0x867F +#endif + +#ifndef GL_SGIX_texture_coordinate_clamp +#define GL_TEXTURE_MAX_CLAMP_S_SGIX 0x8369 +#define GL_TEXTURE_MAX_CLAMP_T_SGIX 0x836A +#define GL_TEXTURE_MAX_CLAMP_R_SGIX 0x836B +#endif + +#ifndef GL_SGIX_scalebias_hint +#define GL_SCALEBIAS_HINT_SGIX 0x8322 +#endif + +#ifndef GL_OML_interlace +#define GL_INTERLACE_OML 0x8980 +#define GL_INTERLACE_READ_OML 0x8981 +#endif + +#ifndef GL_OML_subsample +#define GL_FORMAT_SUBSAMPLE_24_24_OML 0x8982 +#define GL_FORMAT_SUBSAMPLE_244_244_OML 0x8983 +#endif + +#ifndef GL_OML_resample +#define GL_PACK_RESAMPLE_OML 0x8984 +#define GL_UNPACK_RESAMPLE_OML 0x8985 +#define GL_RESAMPLE_REPLICATE_OML 0x8986 +#define GL_RESAMPLE_ZERO_FILL_OML 0x8987 +#define GL_RESAMPLE_AVERAGE_OML 0x8988 +#define GL_RESAMPLE_DECIMATE_OML 0x8989 +#endif + +#ifndef GL_NV_copy_depth_to_color +#define GL_DEPTH_STENCIL_TO_RGBA_NV 0x886E +#define GL_DEPTH_STENCIL_TO_BGRA_NV 0x886F +#endif + +#ifndef GL_ATI_envmap_bumpmap +#define GL_BUMP_ROT_MATRIX_ATI 0x8775 +#define GL_BUMP_ROT_MATRIX_SIZE_ATI 0x8776 +#define GL_BUMP_NUM_TEX_UNITS_ATI 0x8777 +#define GL_BUMP_TEX_UNITS_ATI 0x8778 +#define GL_DUDV_ATI 0x8779 +#define GL_DU8DV8_ATI 0x877A +#define GL_BUMP_ENVMAP_ATI 0x877B +#define GL_BUMP_TARGET_ATI 0x877C +#endif + +#ifndef GL_ATI_fragment_shader +#define GL_FRAGMENT_SHADER_ATI 0x8920 +#define GL_REG_0_ATI 0x8921 +#define GL_REG_1_ATI 0x8922 +#define GL_REG_2_ATI 0x8923 +#define GL_REG_3_ATI 0x8924 +#define GL_REG_4_ATI 0x8925 +#define GL_REG_5_ATI 0x8926 +#define GL_REG_6_ATI 0x8927 +#define GL_REG_7_ATI 0x8928 +#define GL_REG_8_ATI 0x8929 +#define GL_REG_9_ATI 0x892A +#define GL_REG_10_ATI 0x892B +#define GL_REG_11_ATI 0x892C +#define GL_REG_12_ATI 0x892D +#define GL_REG_13_ATI 0x892E +#define GL_REG_14_ATI 0x892F +#define GL_REG_15_ATI 0x8930 +#define GL_REG_16_ATI 0x8931 +#define GL_REG_17_ATI 0x8932 +#define GL_REG_18_ATI 0x8933 +#define GL_REG_19_ATI 0x8934 +#define GL_REG_20_ATI 0x8935 +#define GL_REG_21_ATI 0x8936 +#define GL_REG_22_ATI 0x8937 +#define GL_REG_23_ATI 0x8938 +#define GL_REG_24_ATI 0x8939 +#define GL_REG_25_ATI 0x893A +#define GL_REG_26_ATI 0x893B +#define GL_REG_27_ATI 0x893C +#define GL_REG_28_ATI 0x893D +#define GL_REG_29_ATI 0x893E +#define GL_REG_30_ATI 0x893F +#define GL_REG_31_ATI 0x8940 +#define GL_CON_0_ATI 0x8941 +#define GL_CON_1_ATI 0x8942 +#define GL_CON_2_ATI 0x8943 +#define GL_CON_3_ATI 0x8944 +#define GL_CON_4_ATI 0x8945 +#define GL_CON_5_ATI 0x8946 +#define GL_CON_6_ATI 0x8947 +#define GL_CON_7_ATI 0x8948 +#define GL_CON_8_ATI 0x8949 +#define GL_CON_9_ATI 0x894A +#define GL_CON_10_ATI 0x894B +#define GL_CON_11_ATI 0x894C +#define GL_CON_12_ATI 0x894D +#define GL_CON_13_ATI 0x894E +#define GL_CON_14_ATI 0x894F +#define GL_CON_15_ATI 0x8950 +#define GL_CON_16_ATI 0x8951 +#define GL_CON_17_ATI 0x8952 +#define GL_CON_18_ATI 0x8953 +#define GL_CON_19_ATI 0x8954 +#define GL_CON_20_ATI 0x8955 +#define GL_CON_21_ATI 0x8956 +#define GL_CON_22_ATI 0x8957 +#define GL_CON_23_ATI 0x8958 +#define GL_CON_24_ATI 0x8959 +#define GL_CON_25_ATI 0x895A +#define GL_CON_26_ATI 0x895B +#define GL_CON_27_ATI 0x895C +#define GL_CON_28_ATI 0x895D +#define GL_CON_29_ATI 0x895E +#define GL_CON_30_ATI 0x895F +#define GL_CON_31_ATI 0x8960 +#define GL_MOV_ATI 0x8961 +#define GL_ADD_ATI 0x8963 +#define GL_MUL_ATI 0x8964 +#define GL_SUB_ATI 0x8965 +#define GL_DOT3_ATI 0x8966 +#define GL_DOT4_ATI 0x8967 +#define GL_MAD_ATI 0x8968 +#define GL_LERP_ATI 0x8969 +#define GL_CND_ATI 0x896A +#define GL_CND0_ATI 0x896B +#define GL_DOT2_ADD_ATI 0x896C +#define GL_SECONDARY_INTERPOLATOR_ATI 0x896D +#define GL_NUM_FRAGMENT_REGISTERS_ATI 0x896E +#define GL_NUM_FRAGMENT_CONSTANTS_ATI 0x896F +#define GL_NUM_PASSES_ATI 0x8970 +#define GL_NUM_INSTRUCTIONS_PER_PASS_ATI 0x8971 +#define GL_NUM_INSTRUCTIONS_TOTAL_ATI 0x8972 +#define GL_NUM_INPUT_INTERPOLATOR_COMPONENTS_ATI 0x8973 +#define GL_NUM_LOOPBACK_COMPONENTS_ATI 0x8974 +#define GL_COLOR_ALPHA_PAIRING_ATI 0x8975 +#define GL_SWIZZLE_STR_ATI 0x8976 +#define GL_SWIZZLE_STQ_ATI 0x8977 +#define GL_SWIZZLE_STR_DR_ATI 0x8978 +#define GL_SWIZZLE_STQ_DQ_ATI 0x8979 +#define GL_SWIZZLE_STRQ_ATI 0x897A +#define GL_SWIZZLE_STRQ_DQ_ATI 0x897B +#define GL_RED_BIT_ATI 0x00000001 +#define GL_GREEN_BIT_ATI 0x00000002 +#define GL_BLUE_BIT_ATI 0x00000004 +#define GL_2X_BIT_ATI 0x00000001 +#define GL_4X_BIT_ATI 0x00000002 +#define GL_8X_BIT_ATI 0x00000004 +#define GL_HALF_BIT_ATI 0x00000008 +#define GL_QUARTER_BIT_ATI 0x00000010 +#define GL_EIGHTH_BIT_ATI 0x00000020 +#define GL_SATURATE_BIT_ATI 0x00000040 +#define GL_COMP_BIT_ATI 0x00000002 +#define GL_NEGATE_BIT_ATI 0x00000004 +#define GL_BIAS_BIT_ATI 0x00000008 +#endif + +#ifndef GL_ATI_pn_triangles +#define GL_PN_TRIANGLES_ATI 0x87F0 +#define GL_MAX_PN_TRIANGLES_TESSELATION_LEVEL_ATI 0x87F1 +#define GL_PN_TRIANGLES_POINT_MODE_ATI 0x87F2 +#define GL_PN_TRIANGLES_NORMAL_MODE_ATI 0x87F3 +#define GL_PN_TRIANGLES_TESSELATION_LEVEL_ATI 0x87F4 +#define GL_PN_TRIANGLES_POINT_MODE_LINEAR_ATI 0x87F5 +#define GL_PN_TRIANGLES_POINT_MODE_CUBIC_ATI 0x87F6 +#define GL_PN_TRIANGLES_NORMAL_MODE_LINEAR_ATI 0x87F7 +#define GL_PN_TRIANGLES_NORMAL_MODE_QUADRATIC_ATI 0x87F8 +#endif + +#ifndef GL_ATI_vertex_array_object +#define GL_STATIC_ATI 0x8760 +#define GL_DYNAMIC_ATI 0x8761 +#define GL_PRESERVE_ATI 0x8762 +#define GL_DISCARD_ATI 0x8763 +#define GL_OBJECT_BUFFER_SIZE_ATI 0x8764 +#define GL_OBJECT_BUFFER_USAGE_ATI 0x8765 +#define GL_ARRAY_OBJECT_BUFFER_ATI 0x8766 +#define GL_ARRAY_OBJECT_OFFSET_ATI 0x8767 +#endif + +#ifndef GL_EXT_vertex_shader +#define GL_VERTEX_SHADER_EXT 0x8780 +#define GL_VERTEX_SHADER_BINDING_EXT 0x8781 +#define GL_OP_INDEX_EXT 0x8782 +#define GL_OP_NEGATE_EXT 0x8783 +#define GL_OP_DOT3_EXT 0x8784 +#define GL_OP_DOT4_EXT 0x8785 +#define GL_OP_MUL_EXT 0x8786 +#define GL_OP_ADD_EXT 0x8787 +#define GL_OP_MADD_EXT 0x8788 +#define GL_OP_FRAC_EXT 0x8789 +#define GL_OP_MAX_EXT 0x878A +#define GL_OP_MIN_EXT 0x878B +#define GL_OP_SET_GE_EXT 0x878C +#define GL_OP_SET_LT_EXT 0x878D +#define GL_OP_CLAMP_EXT 0x878E +#define GL_OP_FLOOR_EXT 0x878F +#define GL_OP_ROUND_EXT 0x8790 +#define GL_OP_EXP_BASE_2_EXT 0x8791 +#define GL_OP_LOG_BASE_2_EXT 0x8792 +#define GL_OP_POWER_EXT 0x8793 +#define GL_OP_RECIP_EXT 0x8794 +#define GL_OP_RECIP_SQRT_EXT 0x8795 +#define GL_OP_SUB_EXT 0x8796 +#define GL_OP_CROSS_PRODUCT_EXT 0x8797 +#define GL_OP_MULTIPLY_MATRIX_EXT 0x8798 +#define GL_OP_MOV_EXT 0x8799 +#define GL_OUTPUT_VERTEX_EXT 0x879A +#define GL_OUTPUT_COLOR0_EXT 0x879B +#define GL_OUTPUT_COLOR1_EXT 0x879C +#define GL_OUTPUT_TEXTURE_COORD0_EXT 0x879D +#define GL_OUTPUT_TEXTURE_COORD1_EXT 0x879E +#define GL_OUTPUT_TEXTURE_COORD2_EXT 0x879F +#define GL_OUTPUT_TEXTURE_COORD3_EXT 0x87A0 +#define GL_OUTPUT_TEXTURE_COORD4_EXT 0x87A1 +#define GL_OUTPUT_TEXTURE_COORD5_EXT 0x87A2 +#define GL_OUTPUT_TEXTURE_COORD6_EXT 0x87A3 +#define GL_OUTPUT_TEXTURE_COORD7_EXT 0x87A4 +#define GL_OUTPUT_TEXTURE_COORD8_EXT 0x87A5 +#define GL_OUTPUT_TEXTURE_COORD9_EXT 0x87A6 +#define GL_OUTPUT_TEXTURE_COORD10_EXT 0x87A7 +#define GL_OUTPUT_TEXTURE_COORD11_EXT 0x87A8 +#define GL_OUTPUT_TEXTURE_COORD12_EXT 0x87A9 +#define GL_OUTPUT_TEXTURE_COORD13_EXT 0x87AA +#define GL_OUTPUT_TEXTURE_COORD14_EXT 0x87AB +#define GL_OUTPUT_TEXTURE_COORD15_EXT 0x87AC +#define GL_OUTPUT_TEXTURE_COORD16_EXT 0x87AD +#define GL_OUTPUT_TEXTURE_COORD17_EXT 0x87AE +#define GL_OUTPUT_TEXTURE_COORD18_EXT 0x87AF +#define GL_OUTPUT_TEXTURE_COORD19_EXT 0x87B0 +#define GL_OUTPUT_TEXTURE_COORD20_EXT 0x87B1 +#define GL_OUTPUT_TEXTURE_COORD21_EXT 0x87B2 +#define GL_OUTPUT_TEXTURE_COORD22_EXT 0x87B3 +#define GL_OUTPUT_TEXTURE_COORD23_EXT 0x87B4 +#define GL_OUTPUT_TEXTURE_COORD24_EXT 0x87B5 +#define GL_OUTPUT_TEXTURE_COORD25_EXT 0x87B6 +#define GL_OUTPUT_TEXTURE_COORD26_EXT 0x87B7 +#define GL_OUTPUT_TEXTURE_COORD27_EXT 0x87B8 +#define GL_OUTPUT_TEXTURE_COORD28_EXT 0x87B9 +#define GL_OUTPUT_TEXTURE_COORD29_EXT 0x87BA +#define GL_OUTPUT_TEXTURE_COORD30_EXT 0x87BB +#define GL_OUTPUT_TEXTURE_COORD31_EXT 0x87BC +#define GL_OUTPUT_FOG_EXT 0x87BD +#define GL_SCALAR_EXT 0x87BE +#define GL_VECTOR_EXT 0x87BF +#define GL_MATRIX_EXT 0x87C0 +#define GL_VARIANT_EXT 0x87C1 +#define GL_INVARIANT_EXT 0x87C2 +#define GL_LOCAL_CONSTANT_EXT 0x87C3 +#define GL_LOCAL_EXT 0x87C4 +#define GL_MAX_VERTEX_SHADER_INSTRUCTIONS_EXT 0x87C5 +#define GL_MAX_VERTEX_SHADER_VARIANTS_EXT 0x87C6 +#define GL_MAX_VERTEX_SHADER_INVARIANTS_EXT 0x87C7 +#define GL_MAX_VERTEX_SHADER_LOCAL_CONSTANTS_EXT 0x87C8 +#define GL_MAX_VERTEX_SHADER_LOCALS_EXT 0x87C9 +#define GL_MAX_OPTIMIZED_VERTEX_SHADER_INSTRUCTIONS_EXT 0x87CA +#define GL_MAX_OPTIMIZED_VERTEX_SHADER_VARIANTS_EXT 0x87CB +#define GL_MAX_OPTIMIZED_VERTEX_SHADER_LOCAL_CONSTANTS_EXT 0x87CC +#define GL_MAX_OPTIMIZED_VERTEX_SHADER_INVARIANTS_EXT 0x87CD +#define GL_MAX_OPTIMIZED_VERTEX_SHADER_LOCALS_EXT 0x87CE +#define GL_VERTEX_SHADER_INSTRUCTIONS_EXT 0x87CF +#define GL_VERTEX_SHADER_VARIANTS_EXT 0x87D0 +#define GL_VERTEX_SHADER_INVARIANTS_EXT 0x87D1 +#define GL_VERTEX_SHADER_LOCAL_CONSTANTS_EXT 0x87D2 +#define GL_VERTEX_SHADER_LOCALS_EXT 0x87D3 +#define GL_VERTEX_SHADER_OPTIMIZED_EXT 0x87D4 +#define GL_X_EXT 0x87D5 +#define GL_Y_EXT 0x87D6 +#define GL_Z_EXT 0x87D7 +#define GL_W_EXT 0x87D8 +#define GL_NEGATIVE_X_EXT 0x87D9 +#define GL_NEGATIVE_Y_EXT 0x87DA +#define GL_NEGATIVE_Z_EXT 0x87DB +#define GL_NEGATIVE_W_EXT 0x87DC +#define GL_ZERO_EXT 0x87DD +#define GL_ONE_EXT 0x87DE +#define GL_NEGATIVE_ONE_EXT 0x87DF +#define GL_NORMALIZED_RANGE_EXT 0x87E0 +#define GL_FULL_RANGE_EXT 0x87E1 +#define GL_CURRENT_VERTEX_EXT 0x87E2 +#define GL_MVP_MATRIX_EXT 0x87E3 +#define GL_VARIANT_VALUE_EXT 0x87E4 +#define GL_VARIANT_DATATYPE_EXT 0x87E5 +#define GL_VARIANT_ARRAY_STRIDE_EXT 0x87E6 +#define GL_VARIANT_ARRAY_TYPE_EXT 0x87E7 +#define GL_VARIANT_ARRAY_EXT 0x87E8 +#define GL_VARIANT_ARRAY_POINTER_EXT 0x87E9 +#define GL_INVARIANT_VALUE_EXT 0x87EA +#define GL_INVARIANT_DATATYPE_EXT 0x87EB +#define GL_LOCAL_CONSTANT_VALUE_EXT 0x87EC +#define GL_LOCAL_CONSTANT_DATATYPE_EXT 0x87ED +#endif + +#ifndef GL_ATI_vertex_streams +#define GL_MAX_VERTEX_STREAMS_ATI 0x876B +#define GL_VERTEX_STREAM0_ATI 0x876C +#define GL_VERTEX_STREAM1_ATI 0x876D +#define GL_VERTEX_STREAM2_ATI 0x876E +#define GL_VERTEX_STREAM3_ATI 0x876F +#define GL_VERTEX_STREAM4_ATI 0x8770 +#define GL_VERTEX_STREAM5_ATI 0x8771 +#define GL_VERTEX_STREAM6_ATI 0x8772 +#define GL_VERTEX_STREAM7_ATI 0x8773 +#define GL_VERTEX_SOURCE_ATI 0x8774 +#endif + +#ifndef GL_ATI_element_array +#define GL_ELEMENT_ARRAY_ATI 0x8768 +#define GL_ELEMENT_ARRAY_TYPE_ATI 0x8769 +#define GL_ELEMENT_ARRAY_POINTER_ATI 0x876A +#endif + +#ifndef GL_SUN_mesh_array +#define GL_QUAD_MESH_SUN 0x8614 +#define GL_TRIANGLE_MESH_SUN 0x8615 +#endif + +#ifndef GL_SUN_slice_accum +#define GL_SLICE_ACCUM_SUN 0x85CC +#endif + +#ifndef GL_NV_multisample_filter_hint +#define GL_MULTISAMPLE_FILTER_HINT_NV 0x8534 +#endif + +#ifndef GL_NV_depth_clamp +#define GL_DEPTH_CLAMP_NV 0x864F +#endif + +#ifndef GL_NV_occlusion_query +#define GL_PIXEL_COUNTER_BITS_NV 0x8864 +#define GL_CURRENT_OCCLUSION_QUERY_ID_NV 0x8865 +#define GL_PIXEL_COUNT_NV 0x8866 +#define GL_PIXEL_COUNT_AVAILABLE_NV 0x8867 +#endif + +#ifndef GL_NV_point_sprite +#define GL_POINT_SPRITE_NV 0x8861 +#define GL_COORD_REPLACE_NV 0x8862 +#define GL_POINT_SPRITE_R_MODE_NV 0x8863 +#endif + +#ifndef GL_NV_texture_shader3 +#define GL_OFFSET_PROJECTIVE_TEXTURE_2D_NV 0x8850 +#define GL_OFFSET_PROJECTIVE_TEXTURE_2D_SCALE_NV 0x8851 +#define GL_OFFSET_PROJECTIVE_TEXTURE_RECTANGLE_NV 0x8852 +#define GL_OFFSET_PROJECTIVE_TEXTURE_RECTANGLE_SCALE_NV 0x8853 +#define GL_OFFSET_HILO_TEXTURE_2D_NV 0x8854 +#define GL_OFFSET_HILO_TEXTURE_RECTANGLE_NV 0x8855 +#define GL_OFFSET_HILO_PROJECTIVE_TEXTURE_2D_NV 0x8856 +#define GL_OFFSET_HILO_PROJECTIVE_TEXTURE_RECTANGLE_NV 0x8857 +#define GL_DEPENDENT_HILO_TEXTURE_2D_NV 0x8858 +#define GL_DEPENDENT_RGB_TEXTURE_3D_NV 0x8859 +#define GL_DEPENDENT_RGB_TEXTURE_CUBE_MAP_NV 0x885A +#define GL_DOT_PRODUCT_PASS_THROUGH_NV 0x885B +#define GL_DOT_PRODUCT_TEXTURE_1D_NV 0x885C +#define GL_DOT_PRODUCT_AFFINE_DEPTH_REPLACE_NV 0x885D +#define GL_HILO8_NV 0x885E +#define GL_SIGNED_HILO8_NV 0x885F +#define GL_FORCE_BLUE_TO_ONE_NV 0x8860 +#endif + +#ifndef GL_NV_vertex_program1_1 +#endif + +#ifndef GL_EXT_shadow_funcs +#endif + +#ifndef GL_EXT_stencil_two_side +#define GL_STENCIL_TEST_TWO_SIDE_EXT 0x8910 +#define GL_ACTIVE_STENCIL_FACE_EXT 0x8911 +#endif + +#ifndef GL_ATI_text_fragment_shader +#define GL_TEXT_FRAGMENT_SHADER_ATI 0x8200 +#endif + +#ifndef GL_APPLE_client_storage +#define GL_UNPACK_CLIENT_STORAGE_APPLE 0x85B2 +#endif + +#ifndef GL_APPLE_element_array +#define GL_ELEMENT_ARRAY_APPLE 0x8768 +#define GL_ELEMENT_ARRAY_TYPE_APPLE 0x8769 +#define GL_ELEMENT_ARRAY_POINTER_APPLE 0x876A +#endif + +#ifndef GL_APPLE_fence +#define GL_DRAW_PIXELS_APPLE 0x8A0A +#define GL_FENCE_APPLE 0x8A0B +#endif + +#ifndef GL_APPLE_vertex_array_object +#define GL_VERTEX_ARRAY_BINDING_APPLE 0x85B5 +#endif + +#ifndef GL_APPLE_vertex_array_range +#define GL_VERTEX_ARRAY_RANGE_APPLE 0x851D +#define GL_VERTEX_ARRAY_RANGE_LENGTH_APPLE 0x851E +#define GL_VERTEX_ARRAY_STORAGE_HINT_APPLE 0x851F +#define GL_VERTEX_ARRAY_RANGE_POINTER_APPLE 0x8521 +#define GL_STORAGE_CACHED_APPLE 0x85BE +#define GL_STORAGE_SHARED_APPLE 0x85BF +#endif + +#ifndef GL_APPLE_ycbcr_422 +#define GL_YCBCR_422_APPLE 0x85B9 +#define GL_UNSIGNED_SHORT_8_8_APPLE 0x85BA +#define GL_UNSIGNED_SHORT_8_8_REV_APPLE 0x85BB +#endif + +#ifndef GL_S3_s3tc +#define GL_RGB_S3TC 0x83A0 +#define GL_RGB4_S3TC 0x83A1 +#define GL_RGBA_S3TC 0x83A2 +#define GL_RGBA4_S3TC 0x83A3 +#endif + +#ifndef GL_ATI_draw_buffers +#define GL_MAX_DRAW_BUFFERS_ATI 0x8824 +#define GL_DRAW_BUFFER0_ATI 0x8825 +#define GL_DRAW_BUFFER1_ATI 0x8826 +#define GL_DRAW_BUFFER2_ATI 0x8827 +#define GL_DRAW_BUFFER3_ATI 0x8828 +#define GL_DRAW_BUFFER4_ATI 0x8829 +#define GL_DRAW_BUFFER5_ATI 0x882A +#define GL_DRAW_BUFFER6_ATI 0x882B +#define GL_DRAW_BUFFER7_ATI 0x882C +#define GL_DRAW_BUFFER8_ATI 0x882D +#define GL_DRAW_BUFFER9_ATI 0x882E +#define GL_DRAW_BUFFER10_ATI 0x882F +#define GL_DRAW_BUFFER11_ATI 0x8830 +#define GL_DRAW_BUFFER12_ATI 0x8831 +#define GL_DRAW_BUFFER13_ATI 0x8832 +#define GL_DRAW_BUFFER14_ATI 0x8833 +#define GL_DRAW_BUFFER15_ATI 0x8834 +#endif + +#ifndef GL_ATI_pixel_format_float +#define GL_TYPE_RGBA_FLOAT_ATI 0x8820 +#define GL_COLOR_CLEAR_UNCLAMPED_VALUE_ATI 0x8835 +#endif + +#ifndef GL_ATI_texture_env_combine3 +#define GL_MODULATE_ADD_ATI 0x8744 +#define GL_MODULATE_SIGNED_ADD_ATI 0x8745 +#define GL_MODULATE_SUBTRACT_ATI 0x8746 +#endif + +#ifndef GL_ATI_texture_float +#define GL_RGBA_FLOAT32_ATI 0x8814 +#define GL_RGB_FLOAT32_ATI 0x8815 +#define GL_ALPHA_FLOAT32_ATI 0x8816 +#define GL_INTENSITY_FLOAT32_ATI 0x8817 +#define GL_LUMINANCE_FLOAT32_ATI 0x8818 +#define GL_LUMINANCE_ALPHA_FLOAT32_ATI 0x8819 +#define GL_RGBA_FLOAT16_ATI 0x881A +#define GL_RGB_FLOAT16_ATI 0x881B +#define GL_ALPHA_FLOAT16_ATI 0x881C +#define GL_INTENSITY_FLOAT16_ATI 0x881D +#define GL_LUMINANCE_FLOAT16_ATI 0x881E +#define GL_LUMINANCE_ALPHA_FLOAT16_ATI 0x881F +#endif + +#ifndef GL_NV_float_buffer +#define GL_FLOAT_R_NV 0x8880 +#define GL_FLOAT_RG_NV 0x8881 +#define GL_FLOAT_RGB_NV 0x8882 +#define GL_FLOAT_RGBA_NV 0x8883 +#define GL_FLOAT_R16_NV 0x8884 +#define GL_FLOAT_R32_NV 0x8885 +#define GL_FLOAT_RG16_NV 0x8886 +#define GL_FLOAT_RG32_NV 0x8887 +#define GL_FLOAT_RGB16_NV 0x8888 +#define GL_FLOAT_RGB32_NV 0x8889 +#define GL_FLOAT_RGBA16_NV 0x888A +#define GL_FLOAT_RGBA32_NV 0x888B +#define GL_TEXTURE_FLOAT_COMPONENTS_NV 0x888C +#define GL_FLOAT_CLEAR_COLOR_VALUE_NV 0x888D +#define GL_FLOAT_RGBA_MODE_NV 0x888E +#endif + +#ifndef GL_NV_fragment_program +#define GL_MAX_FRAGMENT_PROGRAM_LOCAL_PARAMETERS_NV 0x8868 +#define GL_FRAGMENT_PROGRAM_NV 0x8870 +#define GL_MAX_TEXTURE_COORDS_NV 0x8871 +#define GL_MAX_TEXTURE_IMAGE_UNITS_NV 0x8872 +#define GL_FRAGMENT_PROGRAM_BINDING_NV 0x8873 +#define GL_PROGRAM_ERROR_STRING_NV 0x8874 +#endif + +#ifndef GL_NV_half_float +#define GL_HALF_FLOAT_NV 0x140B +#endif + +#ifndef GL_NV_pixel_data_range +#define GL_WRITE_PIXEL_DATA_RANGE_NV 0x8878 +#define GL_READ_PIXEL_DATA_RANGE_NV 0x8879 +#define GL_WRITE_PIXEL_DATA_RANGE_LENGTH_NV 0x887A +#define GL_READ_PIXEL_DATA_RANGE_LENGTH_NV 0x887B +#define GL_WRITE_PIXEL_DATA_RANGE_POINTER_NV 0x887C +#define GL_READ_PIXEL_DATA_RANGE_POINTER_NV 0x887D +#endif + +#ifndef GL_NV_primitive_restart +#define GL_PRIMITIVE_RESTART_NV 0x8558 +#define GL_PRIMITIVE_RESTART_INDEX_NV 0x8559 +#endif + +#ifndef GL_NV_texture_expand_normal +#define GL_TEXTURE_UNSIGNED_REMAP_MODE_NV 0x888F +#endif + +#ifndef GL_NV_vertex_program2 +#endif + +#ifndef GL_ATI_map_object_buffer +#endif + +#ifndef GL_ATI_separate_stencil +#define GL_STENCIL_BACK_FUNC_ATI 0x8800 +#define GL_STENCIL_BACK_FAIL_ATI 0x8801 +#define GL_STENCIL_BACK_PASS_DEPTH_FAIL_ATI 0x8802 +#define GL_STENCIL_BACK_PASS_DEPTH_PASS_ATI 0x8803 +#endif + +#ifndef GL_ATI_vertex_attrib_array_object +#endif + +#ifndef GL_OES_read_format +#define GL_IMPLEMENTATION_COLOR_READ_TYPE_OES 0x8B9A +#define GL_IMPLEMENTATION_COLOR_READ_FORMAT_OES 0x8B9B +#endif + +#ifndef GL_EXT_depth_bounds_test +#define GL_DEPTH_BOUNDS_TEST_EXT 0x8890 +#define GL_DEPTH_BOUNDS_EXT 0x8891 +#endif + +#ifndef GL_EXT_texture_mirror_clamp +#define GL_MIRROR_CLAMP_EXT 0x8742 +#define GL_MIRROR_CLAMP_TO_EDGE_EXT 0x8743 +#define GL_MIRROR_CLAMP_TO_BORDER_EXT 0x8912 +#endif + +#ifndef GL_EXT_blend_equation_separate +#define GL_BLEND_EQUATION_RGB_EXT GL_BLEND_EQUATION +#define GL_BLEND_EQUATION_ALPHA_EXT 0x883D +#endif + +#ifndef GL_MESA_pack_invert +#define GL_PACK_INVERT_MESA 0x8758 +#endif + +#ifndef GL_MESA_ycbcr_texture +#define GL_UNSIGNED_SHORT_8_8_MESA 0x85BA +#define GL_UNSIGNED_SHORT_8_8_REV_MESA 0x85BB +#define GL_YCBCR_MESA 0x8757 +#endif + +#ifndef GL_EXT_pixel_buffer_object +#define GL_PIXEL_PACK_BUFFER_EXT 0x88EB +#define GL_PIXEL_UNPACK_BUFFER_EXT 0x88EC +#define GL_PIXEL_PACK_BUFFER_BINDING_EXT 0x88ED +#define GL_PIXEL_UNPACK_BUFFER_BINDING_EXT 0x88EF +#endif + +#ifndef GL_NV_fragment_program_option +#endif + +#ifndef GL_NV_fragment_program2 +#define GL_MAX_PROGRAM_EXEC_INSTRUCTIONS_NV 0x88F4 +#define GL_MAX_PROGRAM_CALL_DEPTH_NV 0x88F5 +#define GL_MAX_PROGRAM_IF_DEPTH_NV 0x88F6 +#define GL_MAX_PROGRAM_LOOP_DEPTH_NV 0x88F7 +#define GL_MAX_PROGRAM_LOOP_COUNT_NV 0x88F8 +#endif + +#ifndef GL_NV_vertex_program2_option +/* reuse GL_MAX_PROGRAM_EXEC_INSTRUCTIONS_NV */ +/* reuse GL_MAX_PROGRAM_CALL_DEPTH_NV */ +#endif + +#ifndef GL_NV_vertex_program3 +/* reuse GL_MAX_VERTEX_TEXTURE_IMAGE_UNITS_ARB */ +#endif + +#ifndef GL_EXT_framebuffer_object +#define GL_INVALID_FRAMEBUFFER_OPERATION_EXT 0x0506 +#define GL_MAX_RENDERBUFFER_SIZE_EXT 0x84E8 +#define GL_FRAMEBUFFER_BINDING_EXT 0x8CA6 +#define GL_RENDERBUFFER_BINDING_EXT 0x8CA7 +#define GL_FRAMEBUFFER_ATTACHMENT_OBJECT_TYPE_EXT 0x8CD0 +#define GL_FRAMEBUFFER_ATTACHMENT_OBJECT_NAME_EXT 0x8CD1 +#define GL_FRAMEBUFFER_ATTACHMENT_TEXTURE_LEVEL_EXT 0x8CD2 +#define GL_FRAMEBUFFER_ATTACHMENT_TEXTURE_CUBE_MAP_FACE_EXT 0x8CD3 +#define GL_FRAMEBUFFER_ATTACHMENT_TEXTURE_3D_ZOFFSET_EXT 0x8CD4 +#define GL_FRAMEBUFFER_COMPLETE_EXT 0x8CD5 +#define GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT_EXT 0x8CD6 +#define GL_FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT_EXT 0x8CD7 +#define GL_FRAMEBUFFER_INCOMPLETE_DUPLICATE_ATTACHMENT_EXT 0x8CD8 +#define GL_FRAMEBUFFER_INCOMPLETE_DIMENSIONS_EXT 0x8CD9 +#define GL_FRAMEBUFFER_INCOMPLETE_FORMATS_EXT 0x8CDA +#define GL_FRAMEBUFFER_INCOMPLETE_DRAW_BUFFER_EXT 0x8CDB +#define GL_FRAMEBUFFER_INCOMPLETE_READ_BUFFER_EXT 0x8CDC +#define GL_FRAMEBUFFER_UNSUPPORTED_EXT 0x8CDD +#define GL_MAX_COLOR_ATTACHMENTS_EXT 0x8CDF +#define GL_COLOR_ATTACHMENT0_EXT 0x8CE0 +#define GL_COLOR_ATTACHMENT1_EXT 0x8CE1 +#define GL_COLOR_ATTACHMENT2_EXT 0x8CE2 +#define GL_COLOR_ATTACHMENT3_EXT 0x8CE3 +#define GL_COLOR_ATTACHMENT4_EXT 0x8CE4 +#define GL_COLOR_ATTACHMENT5_EXT 0x8CE5 +#define GL_COLOR_ATTACHMENT6_EXT 0x8CE6 +#define GL_COLOR_ATTACHMENT7_EXT 0x8CE7 +#define GL_COLOR_ATTACHMENT8_EXT 0x8CE8 +#define GL_COLOR_ATTACHMENT9_EXT 0x8CE9 +#define GL_COLOR_ATTACHMENT10_EXT 0x8CEA +#define GL_COLOR_ATTACHMENT11_EXT 0x8CEB +#define GL_COLOR_ATTACHMENT12_EXT 0x8CEC +#define GL_COLOR_ATTACHMENT13_EXT 0x8CED +#define GL_COLOR_ATTACHMENT14_EXT 0x8CEE +#define GL_COLOR_ATTACHMENT15_EXT 0x8CEF +#define GL_DEPTH_ATTACHMENT_EXT 0x8D00 +#define GL_STENCIL_ATTACHMENT_EXT 0x8D20 +#define GL_FRAMEBUFFER_EXT 0x8D40 +#define GL_RENDERBUFFER_EXT 0x8D41 +#define GL_RENDERBUFFER_WIDTH_EXT 0x8D42 +#define GL_RENDERBUFFER_HEIGHT_EXT 0x8D43 +#define GL_RENDERBUFFER_INTERNAL_FORMAT_EXT 0x8D44 +#define GL_STENCIL_INDEX1_EXT 0x8D46 +#define GL_STENCIL_INDEX4_EXT 0x8D47 +#define GL_STENCIL_INDEX8_EXT 0x8D48 +#define GL_STENCIL_INDEX16_EXT 0x8D49 +#define GL_RENDERBUFFER_RED_SIZE_EXT 0x8D50 +#define GL_RENDERBUFFER_GREEN_SIZE_EXT 0x8D51 +#define GL_RENDERBUFFER_BLUE_SIZE_EXT 0x8D52 +#define GL_RENDERBUFFER_ALPHA_SIZE_EXT 0x8D53 +#define GL_RENDERBUFFER_DEPTH_SIZE_EXT 0x8D54 +#define GL_RENDERBUFFER_STENCIL_SIZE_EXT 0x8D55 +#endif + +#ifndef GL_GREMEDY_string_marker +#endif + + +/*************************************************************/ + +#include +#ifndef GL_VERSION_2_0 +/* GL type for program/shader text */ +typedef char GLchar; /* native character */ +#endif + +#ifndef GL_VERSION_1_5 +/* GL types for handling large vertex buffer objects */ +typedef ptrdiff_t GLintptr; +typedef ptrdiff_t GLsizeiptr; +#endif + +#ifndef GL_ARB_vertex_buffer_object +/* GL types for handling large vertex buffer objects */ +typedef ptrdiff_t GLintptrARB; +typedef ptrdiff_t GLsizeiptrARB; +#endif + +#ifndef GL_ARB_shader_objects +/* GL types for handling shader object handles and program/shader text */ +typedef char GLcharARB; /* native character */ +typedef unsigned int GLhandleARB; /* shader object handle */ +#endif + +/* GL types for "half" precision (s10e5) float data in host memory */ +#ifndef GL_ARB_half_float_pixel +typedef unsigned short GLhalfARB; +#endif + +#ifndef GL_NV_half_float +typedef unsigned short GLhalfNV; +#endif + +#ifndef GL_VERSION_1_2 +#define GL_VERSION_1_2 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBlendColor (GLclampf, GLclampf, GLclampf, GLclampf); +GLAPI void APIENTRY glBlendEquation (GLenum); +GLAPI void APIENTRY glDrawRangeElements (GLenum, GLuint, GLuint, GLsizei, GLenum, const GLvoid *); +GLAPI void APIENTRY glColorTable (GLenum, GLenum, GLsizei, GLenum, GLenum, const GLvoid *); +GLAPI void APIENTRY glColorTableParameterfv (GLenum, GLenum, const GLfloat *); +GLAPI void APIENTRY glColorTableParameteriv (GLenum, GLenum, const GLint *); +GLAPI void APIENTRY glCopyColorTable (GLenum, GLenum, GLint, GLint, GLsizei); +GLAPI void APIENTRY glGetColorTable (GLenum, GLenum, GLenum, GLvoid *); +GLAPI void APIENTRY glGetColorTableParameterfv (GLenum, GLenum, GLfloat *); +GLAPI void APIENTRY glGetColorTableParameteriv (GLenum, GLenum, GLint *); +GLAPI void APIENTRY glColorSubTable (GLenum, GLsizei, GLsizei, GLenum, GLenum, const GLvoid *); +GLAPI void APIENTRY glCopyColorSubTable (GLenum, GLsizei, GLint, GLint, GLsizei); +GLAPI void APIENTRY glConvolutionFilter1D (GLenum, GLenum, GLsizei, GLenum, GLenum, const GLvoid *); +GLAPI void APIENTRY glConvolutionFilter2D (GLenum, GLenum, GLsizei, GLsizei, GLenum, GLenum, const GLvoid *); +GLAPI void APIENTRY glConvolutionParameterf (GLenum, GLenum, GLfloat); +GLAPI void APIENTRY glConvolutionParameterfv (GLenum, GLenum, const GLfloat *); +GLAPI void APIENTRY glConvolutionParameteri (GLenum, GLenum, GLint); +GLAPI void APIENTRY glConvolutionParameteriv (GLenum, GLenum, const GLint *); +GLAPI void APIENTRY glCopyConvolutionFilter1D (GLenum, GLenum, GLint, GLint, GLsizei); +GLAPI void APIENTRY glCopyConvolutionFilter2D (GLenum, GLenum, GLint, GLint, GLsizei, GLsizei); +GLAPI void APIENTRY glGetConvolutionFilter (GLenum, GLenum, GLenum, GLvoid *); +GLAPI void APIENTRY glGetConvolutionParameterfv (GLenum, GLenum, GLfloat *); +GLAPI void APIENTRY glGetConvolutionParameteriv (GLenum, GLenum, GLint *); +GLAPI void APIENTRY glGetSeparableFilter (GLenum, GLenum, GLenum, GLvoid *, GLvoid *, GLvoid *); +GLAPI void APIENTRY glSeparableFilter2D (GLenum, GLenum, GLsizei, GLsizei, GLenum, GLenum, const GLvoid *, const GLvoid *); +GLAPI void APIENTRY glGetHistogram (GLenum, GLboolean, GLenum, GLenum, GLvoid *); +GLAPI void APIENTRY glGetHistogramParameterfv (GLenum, GLenum, GLfloat *); +GLAPI void APIENTRY glGetHistogramParameteriv (GLenum, GLenum, GLint *); +GLAPI void APIENTRY glGetMinmax (GLenum, GLboolean, GLenum, GLenum, GLvoid *); +GLAPI void APIENTRY glGetMinmaxParameterfv (GLenum, GLenum, GLfloat *); +GLAPI void APIENTRY glGetMinmaxParameteriv (GLenum, GLenum, GLint *); +GLAPI void APIENTRY glHistogram (GLenum, GLsizei, GLenum, GLboolean); +GLAPI void APIENTRY glMinmax (GLenum, GLenum, GLboolean); +GLAPI void APIENTRY glResetHistogram (GLenum); +GLAPI void APIENTRY glResetMinmax (GLenum); +GLAPI void APIENTRY glTexImage3D (GLenum, GLint, GLint, GLsizei, GLsizei, GLsizei, GLint, GLenum, GLenum, const GLvoid *); +GLAPI void APIENTRY glTexSubImage3D (GLenum, GLint, GLint, GLint, GLint, GLsizei, GLsizei, GLsizei, GLenum, GLenum, const GLvoid *); +GLAPI void APIENTRY glCopyTexSubImage3D (GLenum, GLint, GLint, GLint, GLint, GLint, GLint, GLsizei, GLsizei); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLBLENDCOLORPROC) (GLclampf red, GLclampf green, GLclampf blue, GLclampf alpha); +typedef void (APIENTRYP PFNGLBLENDEQUATIONPROC) (GLenum mode); +typedef void (APIENTRYP PFNGLDRAWRANGEELEMENTSPROC) (GLenum mode, GLuint start, GLuint end, GLsizei count, GLenum type, const GLvoid *indices); +typedef void (APIENTRYP PFNGLCOLORTABLEPROC) (GLenum target, GLenum internalformat, GLsizei width, GLenum format, GLenum type, const GLvoid *table); +typedef void (APIENTRYP PFNGLCOLORTABLEPARAMETERFVPROC) (GLenum target, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLCOLORTABLEPARAMETERIVPROC) (GLenum target, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLCOPYCOLORTABLEPROC) (GLenum target, GLenum internalformat, GLint x, GLint y, GLsizei width); +typedef void (APIENTRYP PFNGLGETCOLORTABLEPROC) (GLenum target, GLenum format, GLenum type, GLvoid *table); +typedef void (APIENTRYP PFNGLGETCOLORTABLEPARAMETERFVPROC) (GLenum target, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETCOLORTABLEPARAMETERIVPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLCOLORSUBTABLEPROC) (GLenum target, GLsizei start, GLsizei count, GLenum format, GLenum type, const GLvoid *data); +typedef void (APIENTRYP PFNGLCOPYCOLORSUBTABLEPROC) (GLenum target, GLsizei start, GLint x, GLint y, GLsizei width); +typedef void (APIENTRYP PFNGLCONVOLUTIONFILTER1DPROC) (GLenum target, GLenum internalformat, GLsizei width, GLenum format, GLenum type, const GLvoid *image); +typedef void (APIENTRYP PFNGLCONVOLUTIONFILTER2DPROC) (GLenum target, GLenum internalformat, GLsizei width, GLsizei height, GLenum format, GLenum type, const GLvoid *image); +typedef void (APIENTRYP PFNGLCONVOLUTIONPARAMETERFPROC) (GLenum target, GLenum pname, GLfloat params); +typedef void (APIENTRYP PFNGLCONVOLUTIONPARAMETERFVPROC) (GLenum target, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLCONVOLUTIONPARAMETERIPROC) (GLenum target, GLenum pname, GLint params); +typedef void (APIENTRYP PFNGLCONVOLUTIONPARAMETERIVPROC) (GLenum target, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLCOPYCONVOLUTIONFILTER1DPROC) (GLenum target, GLenum internalformat, GLint x, GLint y, GLsizei width); +typedef void (APIENTRYP PFNGLCOPYCONVOLUTIONFILTER2DPROC) (GLenum target, GLenum internalformat, GLint x, GLint y, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLGETCONVOLUTIONFILTERPROC) (GLenum target, GLenum format, GLenum type, GLvoid *image); +typedef void (APIENTRYP PFNGLGETCONVOLUTIONPARAMETERFVPROC) (GLenum target, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETCONVOLUTIONPARAMETERIVPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETSEPARABLEFILTERPROC) (GLenum target, GLenum format, GLenum type, GLvoid *row, GLvoid *column, GLvoid *span); +typedef void (APIENTRYP PFNGLSEPARABLEFILTER2DPROC) (GLenum target, GLenum internalformat, GLsizei width, GLsizei height, GLenum format, GLenum type, const GLvoid *row, const GLvoid *column); +typedef void (APIENTRYP PFNGLGETHISTOGRAMPROC) (GLenum target, GLboolean reset, GLenum format, GLenum type, GLvoid *values); +typedef void (APIENTRYP PFNGLGETHISTOGRAMPARAMETERFVPROC) (GLenum target, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETHISTOGRAMPARAMETERIVPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETMINMAXPROC) (GLenum target, GLboolean reset, GLenum format, GLenum type, GLvoid *values); +typedef void (APIENTRYP PFNGLGETMINMAXPARAMETERFVPROC) (GLenum target, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETMINMAXPARAMETERIVPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLHISTOGRAMPROC) (GLenum target, GLsizei width, GLenum internalformat, GLboolean sink); +typedef void (APIENTRYP PFNGLMINMAXPROC) (GLenum target, GLenum internalformat, GLboolean sink); +typedef void (APIENTRYP PFNGLRESETHISTOGRAMPROC) (GLenum target); +typedef void (APIENTRYP PFNGLRESETMINMAXPROC) (GLenum target); +typedef void (APIENTRYP PFNGLTEXIMAGE3DPROC) (GLenum target, GLint level, GLint internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLenum format, GLenum type, const GLvoid *pixels); +typedef void (APIENTRYP PFNGLTEXSUBIMAGE3DPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLenum type, const GLvoid *pixels); +typedef void (APIENTRYP PFNGLCOPYTEXSUBIMAGE3DPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLint x, GLint y, GLsizei width, GLsizei height); +#endif + +#ifndef GL_VERSION_1_3 +#define GL_VERSION_1_3 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glActiveTexture (GLenum); +GLAPI void APIENTRY glClientActiveTexture (GLenum); +GLAPI void APIENTRY glMultiTexCoord1d (GLenum, GLdouble); +GLAPI void APIENTRY glMultiTexCoord1dv (GLenum, const GLdouble *); +GLAPI void APIENTRY glMultiTexCoord1f (GLenum, GLfloat); +GLAPI void APIENTRY glMultiTexCoord1fv (GLenum, const GLfloat *); +GLAPI void APIENTRY glMultiTexCoord1i (GLenum, GLint); +GLAPI void APIENTRY glMultiTexCoord1iv (GLenum, const GLint *); +GLAPI void APIENTRY glMultiTexCoord1s (GLenum, GLshort); +GLAPI void APIENTRY glMultiTexCoord1sv (GLenum, const GLshort *); +GLAPI void APIENTRY glMultiTexCoord2d (GLenum, GLdouble, GLdouble); +GLAPI void APIENTRY glMultiTexCoord2dv (GLenum, const GLdouble *); +GLAPI void APIENTRY glMultiTexCoord2f (GLenum, GLfloat, GLfloat); +GLAPI void APIENTRY glMultiTexCoord2fv (GLenum, const GLfloat *); +GLAPI void APIENTRY glMultiTexCoord2i (GLenum, GLint, GLint); +GLAPI void APIENTRY glMultiTexCoord2iv (GLenum, const GLint *); +GLAPI void APIENTRY glMultiTexCoord2s (GLenum, GLshort, GLshort); +GLAPI void APIENTRY glMultiTexCoord2sv (GLenum, const GLshort *); +GLAPI void APIENTRY glMultiTexCoord3d (GLenum, GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glMultiTexCoord3dv (GLenum, const GLdouble *); +GLAPI void APIENTRY glMultiTexCoord3f (GLenum, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glMultiTexCoord3fv (GLenum, const GLfloat *); +GLAPI void APIENTRY glMultiTexCoord3i (GLenum, GLint, GLint, GLint); +GLAPI void APIENTRY glMultiTexCoord3iv (GLenum, const GLint *); +GLAPI void APIENTRY glMultiTexCoord3s (GLenum, GLshort, GLshort, GLshort); +GLAPI void APIENTRY glMultiTexCoord3sv (GLenum, const GLshort *); +GLAPI void APIENTRY glMultiTexCoord4d (GLenum, GLdouble, GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glMultiTexCoord4dv (GLenum, const GLdouble *); +GLAPI void APIENTRY glMultiTexCoord4f (GLenum, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glMultiTexCoord4fv (GLenum, const GLfloat *); +GLAPI void APIENTRY glMultiTexCoord4i (GLenum, GLint, GLint, GLint, GLint); +GLAPI void APIENTRY glMultiTexCoord4iv (GLenum, const GLint *); +GLAPI void APIENTRY glMultiTexCoord4s (GLenum, GLshort, GLshort, GLshort, GLshort); +GLAPI void APIENTRY glMultiTexCoord4sv (GLenum, const GLshort *); +GLAPI void APIENTRY glLoadTransposeMatrixf (const GLfloat *); +GLAPI void APIENTRY glLoadTransposeMatrixd (const GLdouble *); +GLAPI void APIENTRY glMultTransposeMatrixf (const GLfloat *); +GLAPI void APIENTRY glMultTransposeMatrixd (const GLdouble *); +GLAPI void APIENTRY glSampleCoverage (GLclampf, GLboolean); +GLAPI void APIENTRY glCompressedTexImage3D (GLenum, GLint, GLenum, GLsizei, GLsizei, GLsizei, GLint, GLsizei, const GLvoid *); +GLAPI void APIENTRY glCompressedTexImage2D (GLenum, GLint, GLenum, GLsizei, GLsizei, GLint, GLsizei, const GLvoid *); +GLAPI void APIENTRY glCompressedTexImage1D (GLenum, GLint, GLenum, GLsizei, GLint, GLsizei, const GLvoid *); +GLAPI void APIENTRY glCompressedTexSubImage3D (GLenum, GLint, GLint, GLint, GLint, GLsizei, GLsizei, GLsizei, GLenum, GLsizei, const GLvoid *); +GLAPI void APIENTRY glCompressedTexSubImage2D (GLenum, GLint, GLint, GLint, GLsizei, GLsizei, GLenum, GLsizei, const GLvoid *); +GLAPI void APIENTRY glCompressedTexSubImage1D (GLenum, GLint, GLint, GLsizei, GLenum, GLsizei, const GLvoid *); +GLAPI void APIENTRY glGetCompressedTexImage (GLenum, GLint, GLvoid *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLACTIVETEXTUREPROC) (GLenum texture); +typedef void (APIENTRYP PFNGLCLIENTACTIVETEXTUREPROC) (GLenum texture); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1DPROC) (GLenum target, GLdouble s); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1DVPROC) (GLenum target, const GLdouble *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1FPROC) (GLenum target, GLfloat s); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1FVPROC) (GLenum target, const GLfloat *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1IPROC) (GLenum target, GLint s); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1IVPROC) (GLenum target, const GLint *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1SPROC) (GLenum target, GLshort s); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1SVPROC) (GLenum target, const GLshort *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2DPROC) (GLenum target, GLdouble s, GLdouble t); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2DVPROC) (GLenum target, const GLdouble *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2FPROC) (GLenum target, GLfloat s, GLfloat t); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2FVPROC) (GLenum target, const GLfloat *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2IPROC) (GLenum target, GLint s, GLint t); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2IVPROC) (GLenum target, const GLint *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2SPROC) (GLenum target, GLshort s, GLshort t); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2SVPROC) (GLenum target, const GLshort *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3DPROC) (GLenum target, GLdouble s, GLdouble t, GLdouble r); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3DVPROC) (GLenum target, const GLdouble *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3FPROC) (GLenum target, GLfloat s, GLfloat t, GLfloat r); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3FVPROC) (GLenum target, const GLfloat *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3IPROC) (GLenum target, GLint s, GLint t, GLint r); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3IVPROC) (GLenum target, const GLint *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3SPROC) (GLenum target, GLshort s, GLshort t, GLshort r); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3SVPROC) (GLenum target, const GLshort *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4DPROC) (GLenum target, GLdouble s, GLdouble t, GLdouble r, GLdouble q); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4DVPROC) (GLenum target, const GLdouble *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4FPROC) (GLenum target, GLfloat s, GLfloat t, GLfloat r, GLfloat q); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4FVPROC) (GLenum target, const GLfloat *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4IPROC) (GLenum target, GLint s, GLint t, GLint r, GLint q); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4IVPROC) (GLenum target, const GLint *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4SPROC) (GLenum target, GLshort s, GLshort t, GLshort r, GLshort q); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4SVPROC) (GLenum target, const GLshort *v); +typedef void (APIENTRYP PFNGLLOADTRANSPOSEMATRIXFPROC) (const GLfloat *m); +typedef void (APIENTRYP PFNGLLOADTRANSPOSEMATRIXDPROC) (const GLdouble *m); +typedef void (APIENTRYP PFNGLMULTTRANSPOSEMATRIXFPROC) (const GLfloat *m); +typedef void (APIENTRYP PFNGLMULTTRANSPOSEMATRIXDPROC) (const GLdouble *m); +typedef void (APIENTRYP PFNGLSAMPLECOVERAGEPROC) (GLclampf value, GLboolean invert); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXIMAGE3DPROC) (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLsizei imageSize, const GLvoid *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXIMAGE2DPROC) (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLint border, GLsizei imageSize, const GLvoid *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXIMAGE1DPROC) (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLint border, GLsizei imageSize, const GLvoid *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXSUBIMAGE3DPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLsizei imageSize, const GLvoid *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXSUBIMAGE2DPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLsizei imageSize, const GLvoid *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXSUBIMAGE1DPROC) (GLenum target, GLint level, GLint xoffset, GLsizei width, GLenum format, GLsizei imageSize, const GLvoid *data); +typedef void (APIENTRYP PFNGLGETCOMPRESSEDTEXIMAGEPROC) (GLenum target, GLint level, GLvoid *img); +#endif + +#ifndef GL_VERSION_1_4 +#define GL_VERSION_1_4 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBlendFuncSeparate (GLenum, GLenum, GLenum, GLenum); +GLAPI void APIENTRY glFogCoordf (GLfloat); +GLAPI void APIENTRY glFogCoordfv (const GLfloat *); +GLAPI void APIENTRY glFogCoordd (GLdouble); +GLAPI void APIENTRY glFogCoorddv (const GLdouble *); +GLAPI void APIENTRY glFogCoordPointer (GLenum, GLsizei, const GLvoid *); +GLAPI void APIENTRY glMultiDrawArrays (GLenum, GLint *, GLsizei *, GLsizei); +GLAPI void APIENTRY glMultiDrawElements (GLenum, const GLsizei *, GLenum, const GLvoid* *, GLsizei); +GLAPI void APIENTRY glPointParameterf (GLenum, GLfloat); +GLAPI void APIENTRY glPointParameterfv (GLenum, const GLfloat *); +GLAPI void APIENTRY glPointParameteri (GLenum, GLint); +GLAPI void APIENTRY glPointParameteriv (GLenum, const GLint *); +GLAPI void APIENTRY glSecondaryColor3b (GLbyte, GLbyte, GLbyte); +GLAPI void APIENTRY glSecondaryColor3bv (const GLbyte *); +GLAPI void APIENTRY glSecondaryColor3d (GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glSecondaryColor3dv (const GLdouble *); +GLAPI void APIENTRY glSecondaryColor3f (GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glSecondaryColor3fv (const GLfloat *); +GLAPI void APIENTRY glSecondaryColor3i (GLint, GLint, GLint); +GLAPI void APIENTRY glSecondaryColor3iv (const GLint *); +GLAPI void APIENTRY glSecondaryColor3s (GLshort, GLshort, GLshort); +GLAPI void APIENTRY glSecondaryColor3sv (const GLshort *); +GLAPI void APIENTRY glSecondaryColor3ub (GLubyte, GLubyte, GLubyte); +GLAPI void APIENTRY glSecondaryColor3ubv (const GLubyte *); +GLAPI void APIENTRY glSecondaryColor3ui (GLuint, GLuint, GLuint); +GLAPI void APIENTRY glSecondaryColor3uiv (const GLuint *); +GLAPI void APIENTRY glSecondaryColor3us (GLushort, GLushort, GLushort); +GLAPI void APIENTRY glSecondaryColor3usv (const GLushort *); +GLAPI void APIENTRY glSecondaryColorPointer (GLint, GLenum, GLsizei, const GLvoid *); +GLAPI void APIENTRY glWindowPos2d (GLdouble, GLdouble); +GLAPI void APIENTRY glWindowPos2dv (const GLdouble *); +GLAPI void APIENTRY glWindowPos2f (GLfloat, GLfloat); +GLAPI void APIENTRY glWindowPos2fv (const GLfloat *); +GLAPI void APIENTRY glWindowPos2i (GLint, GLint); +GLAPI void APIENTRY glWindowPos2iv (const GLint *); +GLAPI void APIENTRY glWindowPos2s (GLshort, GLshort); +GLAPI void APIENTRY glWindowPos2sv (const GLshort *); +GLAPI void APIENTRY glWindowPos3d (GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glWindowPos3dv (const GLdouble *); +GLAPI void APIENTRY glWindowPos3f (GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glWindowPos3fv (const GLfloat *); +GLAPI void APIENTRY glWindowPos3i (GLint, GLint, GLint); +GLAPI void APIENTRY glWindowPos3iv (const GLint *); +GLAPI void APIENTRY glWindowPos3s (GLshort, GLshort, GLshort); +GLAPI void APIENTRY glWindowPos3sv (const GLshort *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLBLENDFUNCSEPARATEPROC) (GLenum sfactorRGB, GLenum dfactorRGB, GLenum sfactorAlpha, GLenum dfactorAlpha); +typedef void (APIENTRYP PFNGLFOGCOORDFPROC) (GLfloat coord); +typedef void (APIENTRYP PFNGLFOGCOORDFVPROC) (const GLfloat *coord); +typedef void (APIENTRYP PFNGLFOGCOORDDPROC) (GLdouble coord); +typedef void (APIENTRYP PFNGLFOGCOORDDVPROC) (const GLdouble *coord); +typedef void (APIENTRYP PFNGLFOGCOORDPOINTERPROC) (GLenum type, GLsizei stride, const GLvoid *pointer); +typedef void (APIENTRYP PFNGLMULTIDRAWARRAYSPROC) (GLenum mode, GLint *first, GLsizei *count, GLsizei primcount); +typedef void (APIENTRYP PFNGLMULTIDRAWELEMENTSPROC) (GLenum mode, const GLsizei *count, GLenum type, const GLvoid* *indices, GLsizei primcount); +typedef void (APIENTRYP PFNGLPOINTPARAMETERFPROC) (GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLPOINTPARAMETERFVPROC) (GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLPOINTPARAMETERIPROC) (GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLPOINTPARAMETERIVPROC) (GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3BPROC) (GLbyte red, GLbyte green, GLbyte blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3BVPROC) (const GLbyte *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3DPROC) (GLdouble red, GLdouble green, GLdouble blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3DVPROC) (const GLdouble *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3FPROC) (GLfloat red, GLfloat green, GLfloat blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3FVPROC) (const GLfloat *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3IPROC) (GLint red, GLint green, GLint blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3IVPROC) (const GLint *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3SPROC) (GLshort red, GLshort green, GLshort blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3SVPROC) (const GLshort *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3UBPROC) (GLubyte red, GLubyte green, GLubyte blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3UBVPROC) (const GLubyte *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3UIPROC) (GLuint red, GLuint green, GLuint blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3UIVPROC) (const GLuint *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3USPROC) (GLushort red, GLushort green, GLushort blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3USVPROC) (const GLushort *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLORPOINTERPROC) (GLint size, GLenum type, GLsizei stride, const GLvoid *pointer); +typedef void (APIENTRYP PFNGLWINDOWPOS2DPROC) (GLdouble x, GLdouble y); +typedef void (APIENTRYP PFNGLWINDOWPOS2DVPROC) (const GLdouble *v); +typedef void (APIENTRYP PFNGLWINDOWPOS2FPROC) (GLfloat x, GLfloat y); +typedef void (APIENTRYP PFNGLWINDOWPOS2FVPROC) (const GLfloat *v); +typedef void (APIENTRYP PFNGLWINDOWPOS2IPROC) (GLint x, GLint y); +typedef void (APIENTRYP PFNGLWINDOWPOS2IVPROC) (const GLint *v); +typedef void (APIENTRYP PFNGLWINDOWPOS2SPROC) (GLshort x, GLshort y); +typedef void (APIENTRYP PFNGLWINDOWPOS2SVPROC) (const GLshort *v); +typedef void (APIENTRYP PFNGLWINDOWPOS3DPROC) (GLdouble x, GLdouble y, GLdouble z); +typedef void (APIENTRYP PFNGLWINDOWPOS3DVPROC) (const GLdouble *v); +typedef void (APIENTRYP PFNGLWINDOWPOS3FPROC) (GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLWINDOWPOS3FVPROC) (const GLfloat *v); +typedef void (APIENTRYP PFNGLWINDOWPOS3IPROC) (GLint x, GLint y, GLint z); +typedef void (APIENTRYP PFNGLWINDOWPOS3IVPROC) (const GLint *v); +typedef void (APIENTRYP PFNGLWINDOWPOS3SPROC) (GLshort x, GLshort y, GLshort z); +typedef void (APIENTRYP PFNGLWINDOWPOS3SVPROC) (const GLshort *v); +#endif + +#ifndef GL_VERSION_1_5 +#define GL_VERSION_1_5 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGenQueries (GLsizei, GLuint *); +GLAPI void APIENTRY glDeleteQueries (GLsizei, const GLuint *); +GLAPI GLboolean APIENTRY glIsQuery (GLuint); +GLAPI void APIENTRY glBeginQuery (GLenum, GLuint); +GLAPI void APIENTRY glEndQuery (GLenum); +GLAPI void APIENTRY glGetQueryiv (GLenum, GLenum, GLint *); +GLAPI void APIENTRY glGetQueryObjectiv (GLuint, GLenum, GLint *); +GLAPI void APIENTRY glGetQueryObjectuiv (GLuint, GLenum, GLuint *); +GLAPI void APIENTRY glBindBuffer (GLenum, GLuint); +GLAPI void APIENTRY glDeleteBuffers (GLsizei, const GLuint *); +GLAPI void APIENTRY glGenBuffers (GLsizei, GLuint *); +GLAPI GLboolean APIENTRY glIsBuffer (GLuint); +GLAPI void APIENTRY glBufferData (GLenum, GLsizeiptr, const GLvoid *, GLenum); +GLAPI void APIENTRY glBufferSubData (GLenum, GLintptr, GLsizeiptr, const GLvoid *); +GLAPI void APIENTRY glGetBufferSubData (GLenum, GLintptr, GLsizeiptr, GLvoid *); +GLAPI GLvoid* APIENTRY glMapBuffer (GLenum, GLenum); +GLAPI GLboolean APIENTRY glUnmapBuffer (GLenum); +GLAPI void APIENTRY glGetBufferParameteriv (GLenum, GLenum, GLint *); +GLAPI void APIENTRY glGetBufferPointerv (GLenum, GLenum, GLvoid* *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLGENQUERIESPROC) (GLsizei n, GLuint *ids); +typedef void (APIENTRYP PFNGLDELETEQUERIESPROC) (GLsizei n, const GLuint *ids); +typedef GLboolean (APIENTRYP PFNGLISQUERYPROC) (GLuint id); +typedef void (APIENTRYP PFNGLBEGINQUERYPROC) (GLenum target, GLuint id); +typedef void (APIENTRYP PFNGLENDQUERYPROC) (GLenum target); +typedef void (APIENTRYP PFNGLGETQUERYIVPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETQUERYOBJECTIVPROC) (GLuint id, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETQUERYOBJECTUIVPROC) (GLuint id, GLenum pname, GLuint *params); +typedef void (APIENTRYP PFNGLBINDBUFFERPROC) (GLenum target, GLuint buffer); +typedef void (APIENTRYP PFNGLDELETEBUFFERSPROC) (GLsizei n, const GLuint *buffers); +typedef void (APIENTRYP PFNGLGENBUFFERSPROC) (GLsizei n, GLuint *buffers); +typedef GLboolean (APIENTRYP PFNGLISBUFFERPROC) (GLuint buffer); +typedef void (APIENTRYP PFNGLBUFFERDATAPROC) (GLenum target, GLsizeiptr size, const GLvoid *data, GLenum usage); +typedef void (APIENTRYP PFNGLBUFFERSUBDATAPROC) (GLenum target, GLintptr offset, GLsizeiptr size, const GLvoid *data); +typedef void (APIENTRYP PFNGLGETBUFFERSUBDATAPROC) (GLenum target, GLintptr offset, GLsizeiptr size, GLvoid *data); +typedef GLvoid* (APIENTRYP PFNGLMAPBUFFERPROC) (GLenum target, GLenum access); +typedef GLboolean (APIENTRYP PFNGLUNMAPBUFFERPROC) (GLenum target); +typedef void (APIENTRYP PFNGLGETBUFFERPARAMETERIVPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETBUFFERPOINTERVPROC) (GLenum target, GLenum pname, GLvoid* *params); +#endif + +#ifndef GL_VERSION_2_0 +#define GL_VERSION_2_0 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBlendEquationSeparate (GLenum, GLenum); +GLAPI void APIENTRY glDrawBuffers (GLsizei, const GLenum *); +GLAPI void APIENTRY glStencilOpSeparate (GLenum, GLenum, GLenum, GLenum); +GLAPI void APIENTRY glStencilFuncSeparate (GLenum, GLenum, GLint, GLuint); +GLAPI void APIENTRY glStencilMaskSeparate (GLenum, GLuint); +GLAPI void APIENTRY glAttachShader (GLuint, GLuint); +GLAPI void APIENTRY glBindAttribLocation (GLuint, GLuint, const GLchar *); +GLAPI void APIENTRY glCompileShader (GLuint); +GLAPI GLuint APIENTRY glCreateProgram (void); +GLAPI GLuint APIENTRY glCreateShader (GLenum); +GLAPI void APIENTRY glDeleteProgram (GLuint); +GLAPI void APIENTRY glDeleteShader (GLuint); +GLAPI void APIENTRY glDetachShader (GLuint, GLuint); +GLAPI void APIENTRY glDisableVertexAttribArray (GLuint); +GLAPI void APIENTRY glEnableVertexAttribArray (GLuint); +GLAPI void APIENTRY glGetActiveAttrib (GLuint, GLuint, GLsizei, GLsizei *, GLint *, GLenum *, GLchar *); +GLAPI void APIENTRY glGetActiveUniform (GLuint, GLuint, GLsizei, GLsizei *, GLint *, GLenum *, GLchar *); +GLAPI void APIENTRY glGetAttachedShaders (GLuint, GLsizei, GLsizei *, GLuint *); +GLAPI GLint APIENTRY glGetAttribLocation (GLuint, const GLchar *); +GLAPI void APIENTRY glGetProgramiv (GLuint, GLenum, GLint *); +GLAPI void APIENTRY glGetProgramInfoLog (GLuint, GLsizei, GLsizei *, GLchar *); +GLAPI void APIENTRY glGetShaderiv (GLuint, GLenum, GLint *); +GLAPI void APIENTRY glGetShaderInfoLog (GLuint, GLsizei, GLsizei *, GLchar *); +GLAPI void APIENTRY glGetShaderSource (GLuint, GLsizei, GLsizei *, GLchar *); +GLAPI GLint APIENTRY glGetUniformLocation (GLuint, const GLchar *); +GLAPI void APIENTRY glGetUniformfv (GLuint, GLint, GLfloat *); +GLAPI void APIENTRY glGetUniformiv (GLuint, GLint, GLint *); +GLAPI void APIENTRY glGetVertexAttribdv (GLuint, GLenum, GLdouble *); +GLAPI void APIENTRY glGetVertexAttribfv (GLuint, GLenum, GLfloat *); +GLAPI void APIENTRY glGetVertexAttribiv (GLuint, GLenum, GLint *); +GLAPI void APIENTRY glGetVertexAttribPointerv (GLuint, GLenum, GLvoid* *); +GLAPI GLboolean APIENTRY glIsProgram (GLuint); +GLAPI GLboolean APIENTRY glIsShader (GLuint); +GLAPI void APIENTRY glLinkProgram (GLuint); +GLAPI void APIENTRY glShaderSource (GLuint, GLsizei, const GLchar* *, const GLint *); +GLAPI void APIENTRY glUseProgram (GLuint); +GLAPI void APIENTRY glUniform1f (GLint, GLfloat); +GLAPI void APIENTRY glUniform2f (GLint, GLfloat, GLfloat); +GLAPI void APIENTRY glUniform3f (GLint, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glUniform4f (GLint, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glUniform1i (GLint, GLint); +GLAPI void APIENTRY glUniform2i (GLint, GLint, GLint); +GLAPI void APIENTRY glUniform3i (GLint, GLint, GLint, GLint); +GLAPI void APIENTRY glUniform4i (GLint, GLint, GLint, GLint, GLint); +GLAPI void APIENTRY glUniform1fv (GLint, GLsizei, const GLfloat *); +GLAPI void APIENTRY glUniform2fv (GLint, GLsizei, const GLfloat *); +GLAPI void APIENTRY glUniform3fv (GLint, GLsizei, const GLfloat *); +GLAPI void APIENTRY glUniform4fv (GLint, GLsizei, const GLfloat *); +GLAPI void APIENTRY glUniform1iv (GLint, GLsizei, const GLint *); +GLAPI void APIENTRY glUniform2iv (GLint, GLsizei, const GLint *); +GLAPI void APIENTRY glUniform3iv (GLint, GLsizei, const GLint *); +GLAPI void APIENTRY glUniform4iv (GLint, GLsizei, const GLint *); +GLAPI void APIENTRY glUniformMatrix2fv (GLint, GLsizei, GLboolean, const GLfloat *); +GLAPI void APIENTRY glUniformMatrix3fv (GLint, GLsizei, GLboolean, const GLfloat *); +GLAPI void APIENTRY glUniformMatrix4fv (GLint, GLsizei, GLboolean, const GLfloat *); +GLAPI void APIENTRY glValidateProgram (GLuint); +GLAPI void APIENTRY glVertexAttrib1d (GLuint, GLdouble); +GLAPI void APIENTRY glVertexAttrib1dv (GLuint, const GLdouble *); +GLAPI void APIENTRY glVertexAttrib1f (GLuint, GLfloat); +GLAPI void APIENTRY glVertexAttrib1fv (GLuint, const GLfloat *); +GLAPI void APIENTRY glVertexAttrib1s (GLuint, GLshort); +GLAPI void APIENTRY glVertexAttrib1sv (GLuint, const GLshort *); +GLAPI void APIENTRY glVertexAttrib2d (GLuint, GLdouble, GLdouble); +GLAPI void APIENTRY glVertexAttrib2dv (GLuint, const GLdouble *); +GLAPI void APIENTRY glVertexAttrib2f (GLuint, GLfloat, GLfloat); +GLAPI void APIENTRY glVertexAttrib2fv (GLuint, const GLfloat *); +GLAPI void APIENTRY glVertexAttrib2s (GLuint, GLshort, GLshort); +GLAPI void APIENTRY glVertexAttrib2sv (GLuint, const GLshort *); +GLAPI void APIENTRY glVertexAttrib3d (GLuint, GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glVertexAttrib3dv (GLuint, const GLdouble *); +GLAPI void APIENTRY glVertexAttrib3f (GLuint, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glVertexAttrib3fv (GLuint, const GLfloat *); +GLAPI void APIENTRY glVertexAttrib3s (GLuint, GLshort, GLshort, GLshort); +GLAPI void APIENTRY glVertexAttrib3sv (GLuint, const GLshort *); +GLAPI void APIENTRY glVertexAttrib4Nbv (GLuint, const GLbyte *); +GLAPI void APIENTRY glVertexAttrib4Niv (GLuint, const GLint *); +GLAPI void APIENTRY glVertexAttrib4Nsv (GLuint, const GLshort *); +GLAPI void APIENTRY glVertexAttrib4Nub (GLuint, GLubyte, GLubyte, GLubyte, GLubyte); +GLAPI void APIENTRY glVertexAttrib4Nubv (GLuint, const GLubyte *); +GLAPI void APIENTRY glVertexAttrib4Nuiv (GLuint, const GLuint *); +GLAPI void APIENTRY glVertexAttrib4Nusv (GLuint, const GLushort *); +GLAPI void APIENTRY glVertexAttrib4bv (GLuint, const GLbyte *); +GLAPI void APIENTRY glVertexAttrib4d (GLuint, GLdouble, GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glVertexAttrib4dv (GLuint, const GLdouble *); +GLAPI void APIENTRY glVertexAttrib4f (GLuint, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glVertexAttrib4fv (GLuint, const GLfloat *); +GLAPI void APIENTRY glVertexAttrib4iv (GLuint, const GLint *); +GLAPI void APIENTRY glVertexAttrib4s (GLuint, GLshort, GLshort, GLshort, GLshort); +GLAPI void APIENTRY glVertexAttrib4sv (GLuint, const GLshort *); +GLAPI void APIENTRY glVertexAttrib4ubv (GLuint, const GLubyte *); +GLAPI void APIENTRY glVertexAttrib4uiv (GLuint, const GLuint *); +GLAPI void APIENTRY glVertexAttrib4usv (GLuint, const GLushort *); +GLAPI void APIENTRY glVertexAttribPointer (GLuint, GLint, GLenum, GLboolean, GLsizei, const GLvoid *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLBLENDEQUATIONSEPARATEPROC) (GLenum modeRGB, GLenum modeAlpha); +typedef void (APIENTRYP PFNGLDRAWBUFFERSPROC) (GLsizei n, const GLenum *bufs); +typedef void (APIENTRYP PFNGLSTENCILOPSEPARATEPROC) (GLenum face, GLenum sfail, GLenum dpfail, GLenum dppass); +typedef void (APIENTRYP PFNGLSTENCILFUNCSEPARATEPROC) (GLenum frontfunc, GLenum backfunc, GLint ref, GLuint mask); +typedef void (APIENTRYP PFNGLSTENCILMASKSEPARATEPROC) (GLenum face, GLuint mask); +typedef void (APIENTRYP PFNGLATTACHSHADERPROC) (GLuint program, GLuint shader); +typedef void (APIENTRYP PFNGLBINDATTRIBLOCATIONPROC) (GLuint program, GLuint index, const GLchar *name); +typedef void (APIENTRYP PFNGLCOMPILESHADERPROC) (GLuint shader); +typedef GLuint (APIENTRYP PFNGLCREATEPROGRAMPROC) (void); +typedef GLuint (APIENTRYP PFNGLCREATESHADERPROC) (GLenum type); +typedef void (APIENTRYP PFNGLDELETEPROGRAMPROC) (GLuint program); +typedef void (APIENTRYP PFNGLDELETESHADERPROC) (GLuint shader); +typedef void (APIENTRYP PFNGLDETACHSHADERPROC) (GLuint program, GLuint shader); +typedef void (APIENTRYP PFNGLDISABLEVERTEXATTRIBARRAYPROC) (GLuint index); +typedef void (APIENTRYP PFNGLENABLEVERTEXATTRIBARRAYPROC) (GLuint index); +typedef void (APIENTRYP PFNGLGETACTIVEATTRIBPROC) (GLuint program, GLuint index, GLsizei bufSize, GLsizei *length, GLint *size, GLenum *type, GLchar *name); +typedef void (APIENTRYP PFNGLGETACTIVEUNIFORMPROC) (GLuint program, GLuint index, GLsizei bufSize, GLsizei *length, GLint *size, GLenum *type, GLchar *name); +typedef void (APIENTRYP PFNGLGETATTACHEDSHADERSPROC) (GLuint program, GLsizei maxCount, GLsizei *count, GLuint *obj); +typedef GLint (APIENTRYP PFNGLGETATTRIBLOCATIONPROC) (GLuint program, const GLchar *name); +typedef void (APIENTRYP PFNGLGETPROGRAMIVPROC) (GLuint program, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETPROGRAMINFOLOGPROC) (GLuint program, GLsizei bufSize, GLsizei *length, GLchar *infoLog); +typedef void (APIENTRYP PFNGLGETSHADERIVPROC) (GLuint shader, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETSHADERINFOLOGPROC) (GLuint shader, GLsizei bufSize, GLsizei *length, GLchar *infoLog); +typedef void (APIENTRYP PFNGLGETSHADERSOURCEPROC) (GLuint shader, GLsizei bufSize, GLsizei *length, GLchar *source); +typedef GLint (APIENTRYP PFNGLGETUNIFORMLOCATIONPROC) (GLuint program, const GLchar *name); +typedef void (APIENTRYP PFNGLGETUNIFORMFVPROC) (GLuint program, GLint location, GLfloat *params); +typedef void (APIENTRYP PFNGLGETUNIFORMIVPROC) (GLuint program, GLint location, GLint *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBDVPROC) (GLuint index, GLenum pname, GLdouble *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBFVPROC) (GLuint index, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBIVPROC) (GLuint index, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBPOINTERVPROC) (GLuint index, GLenum pname, GLvoid* *pointer); +typedef GLboolean (APIENTRYP PFNGLISPROGRAMPROC) (GLuint program); +typedef GLboolean (APIENTRYP PFNGLISSHADERPROC) (GLuint shader); +typedef void (APIENTRYP PFNGLLINKPROGRAMPROC) (GLuint program); +typedef void (APIENTRYP PFNGLSHADERSOURCEPROC) (GLuint shader, GLsizei count, const GLchar* *string, const GLint *length); +typedef void (APIENTRYP PFNGLUSEPROGRAMPROC) (GLuint program); +typedef void (APIENTRYP PFNGLUNIFORM1FPROC) (GLint location, GLfloat v0); +typedef void (APIENTRYP PFNGLUNIFORM2FPROC) (GLint location, GLfloat v0, GLfloat v1); +typedef void (APIENTRYP PFNGLUNIFORM3FPROC) (GLint location, GLfloat v0, GLfloat v1, GLfloat v2); +typedef void (APIENTRYP PFNGLUNIFORM4FPROC) (GLint location, GLfloat v0, GLfloat v1, GLfloat v2, GLfloat v3); +typedef void (APIENTRYP PFNGLUNIFORM1IPROC) (GLint location, GLint v0); +typedef void (APIENTRYP PFNGLUNIFORM2IPROC) (GLint location, GLint v0, GLint v1); +typedef void (APIENTRYP PFNGLUNIFORM3IPROC) (GLint location, GLint v0, GLint v1, GLint v2); +typedef void (APIENTRYP PFNGLUNIFORM4IPROC) (GLint location, GLint v0, GLint v1, GLint v2, GLint v3); +typedef void (APIENTRYP PFNGLUNIFORM1FVPROC) (GLint location, GLsizei count, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORM2FVPROC) (GLint location, GLsizei count, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORM3FVPROC) (GLint location, GLsizei count, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORM4FVPROC) (GLint location, GLsizei count, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORM1IVPROC) (GLint location, GLsizei count, const GLint *value); +typedef void (APIENTRYP PFNGLUNIFORM2IVPROC) (GLint location, GLsizei count, const GLint *value); +typedef void (APIENTRYP PFNGLUNIFORM3IVPROC) (GLint location, GLsizei count, const GLint *value); +typedef void (APIENTRYP PFNGLUNIFORM4IVPROC) (GLint location, GLsizei count, const GLint *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX2FVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX3FVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX4FVPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLVALIDATEPROGRAMPROC) (GLuint program); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1DPROC) (GLuint index, GLdouble x); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1DVPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1FPROC) (GLuint index, GLfloat x); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1FVPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1SPROC) (GLuint index, GLshort x); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1SVPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2DPROC) (GLuint index, GLdouble x, GLdouble y); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2DVPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2FPROC) (GLuint index, GLfloat x, GLfloat y); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2FVPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2SPROC) (GLuint index, GLshort x, GLshort y); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2SVPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3DPROC) (GLuint index, GLdouble x, GLdouble y, GLdouble z); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3DVPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3FPROC) (GLuint index, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3FVPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3SPROC) (GLuint index, GLshort x, GLshort y, GLshort z); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3SVPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NBVPROC) (GLuint index, const GLbyte *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NIVPROC) (GLuint index, const GLint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NSVPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NUBPROC) (GLuint index, GLubyte x, GLubyte y, GLubyte z, GLubyte w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NUBVPROC) (GLuint index, const GLubyte *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NUIVPROC) (GLuint index, const GLuint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NUSVPROC) (GLuint index, const GLushort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4BVPROC) (GLuint index, const GLbyte *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4DPROC) (GLuint index, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4DVPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4FPROC) (GLuint index, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4FVPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4IVPROC) (GLuint index, const GLint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4SPROC) (GLuint index, GLshort x, GLshort y, GLshort z, GLshort w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4SVPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4UBVPROC) (GLuint index, const GLubyte *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4UIVPROC) (GLuint index, const GLuint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4USVPROC) (GLuint index, const GLushort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBPOINTERPROC) (GLuint index, GLint size, GLenum type, GLboolean normalized, GLsizei stride, const GLvoid *pointer); +#endif + +#ifndef GL_ARB_multitexture +#define GL_ARB_multitexture 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glActiveTextureARB (GLenum); +GLAPI void APIENTRY glClientActiveTextureARB (GLenum); +GLAPI void APIENTRY glMultiTexCoord1dARB (GLenum, GLdouble); +GLAPI void APIENTRY glMultiTexCoord1dvARB (GLenum, const GLdouble *); +GLAPI void APIENTRY glMultiTexCoord1fARB (GLenum, GLfloat); +GLAPI void APIENTRY glMultiTexCoord1fvARB (GLenum, const GLfloat *); +GLAPI void APIENTRY glMultiTexCoord1iARB (GLenum, GLint); +GLAPI void APIENTRY glMultiTexCoord1ivARB (GLenum, const GLint *); +GLAPI void APIENTRY glMultiTexCoord1sARB (GLenum, GLshort); +GLAPI void APIENTRY glMultiTexCoord1svARB (GLenum, const GLshort *); +GLAPI void APIENTRY glMultiTexCoord2dARB (GLenum, GLdouble, GLdouble); +GLAPI void APIENTRY glMultiTexCoord2dvARB (GLenum, const GLdouble *); +GLAPI void APIENTRY glMultiTexCoord2fARB (GLenum, GLfloat, GLfloat); +GLAPI void APIENTRY glMultiTexCoord2fvARB (GLenum, const GLfloat *); +GLAPI void APIENTRY glMultiTexCoord2iARB (GLenum, GLint, GLint); +GLAPI void APIENTRY glMultiTexCoord2ivARB (GLenum, const GLint *); +GLAPI void APIENTRY glMultiTexCoord2sARB (GLenum, GLshort, GLshort); +GLAPI void APIENTRY glMultiTexCoord2svARB (GLenum, const GLshort *); +GLAPI void APIENTRY glMultiTexCoord3dARB (GLenum, GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glMultiTexCoord3dvARB (GLenum, const GLdouble *); +GLAPI void APIENTRY glMultiTexCoord3fARB (GLenum, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glMultiTexCoord3fvARB (GLenum, const GLfloat *); +GLAPI void APIENTRY glMultiTexCoord3iARB (GLenum, GLint, GLint, GLint); +GLAPI void APIENTRY glMultiTexCoord3ivARB (GLenum, const GLint *); +GLAPI void APIENTRY glMultiTexCoord3sARB (GLenum, GLshort, GLshort, GLshort); +GLAPI void APIENTRY glMultiTexCoord3svARB (GLenum, const GLshort *); +GLAPI void APIENTRY glMultiTexCoord4dARB (GLenum, GLdouble, GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glMultiTexCoord4dvARB (GLenum, const GLdouble *); +GLAPI void APIENTRY glMultiTexCoord4fARB (GLenum, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glMultiTexCoord4fvARB (GLenum, const GLfloat *); +GLAPI void APIENTRY glMultiTexCoord4iARB (GLenum, GLint, GLint, GLint, GLint); +GLAPI void APIENTRY glMultiTexCoord4ivARB (GLenum, const GLint *); +GLAPI void APIENTRY glMultiTexCoord4sARB (GLenum, GLshort, GLshort, GLshort, GLshort); +GLAPI void APIENTRY glMultiTexCoord4svARB (GLenum, const GLshort *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLACTIVETEXTUREARBPROC) (GLenum texture); +typedef void (APIENTRYP PFNGLCLIENTACTIVETEXTUREARBPROC) (GLenum texture); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1DARBPROC) (GLenum target, GLdouble s); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1DVARBPROC) (GLenum target, const GLdouble *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1FARBPROC) (GLenum target, GLfloat s); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1FVARBPROC) (GLenum target, const GLfloat *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1IARBPROC) (GLenum target, GLint s); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1IVARBPROC) (GLenum target, const GLint *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1SARBPROC) (GLenum target, GLshort s); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1SVARBPROC) (GLenum target, const GLshort *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2DARBPROC) (GLenum target, GLdouble s, GLdouble t); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2DVARBPROC) (GLenum target, const GLdouble *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2FARBPROC) (GLenum target, GLfloat s, GLfloat t); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2FVARBPROC) (GLenum target, const GLfloat *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2IARBPROC) (GLenum target, GLint s, GLint t); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2IVARBPROC) (GLenum target, const GLint *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2SARBPROC) (GLenum target, GLshort s, GLshort t); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2SVARBPROC) (GLenum target, const GLshort *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3DARBPROC) (GLenum target, GLdouble s, GLdouble t, GLdouble r); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3DVARBPROC) (GLenum target, const GLdouble *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3FARBPROC) (GLenum target, GLfloat s, GLfloat t, GLfloat r); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3FVARBPROC) (GLenum target, const GLfloat *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3IARBPROC) (GLenum target, GLint s, GLint t, GLint r); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3IVARBPROC) (GLenum target, const GLint *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3SARBPROC) (GLenum target, GLshort s, GLshort t, GLshort r); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3SVARBPROC) (GLenum target, const GLshort *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4DARBPROC) (GLenum target, GLdouble s, GLdouble t, GLdouble r, GLdouble q); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4DVARBPROC) (GLenum target, const GLdouble *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4FARBPROC) (GLenum target, GLfloat s, GLfloat t, GLfloat r, GLfloat q); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4FVARBPROC) (GLenum target, const GLfloat *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4IARBPROC) (GLenum target, GLint s, GLint t, GLint r, GLint q); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4IVARBPROC) (GLenum target, const GLint *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4SARBPROC) (GLenum target, GLshort s, GLshort t, GLshort r, GLshort q); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4SVARBPROC) (GLenum target, const GLshort *v); +#endif + +#ifndef GL_ARB_transpose_matrix +#define GL_ARB_transpose_matrix 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glLoadTransposeMatrixfARB (const GLfloat *); +GLAPI void APIENTRY glLoadTransposeMatrixdARB (const GLdouble *); +GLAPI void APIENTRY glMultTransposeMatrixfARB (const GLfloat *); +GLAPI void APIENTRY glMultTransposeMatrixdARB (const GLdouble *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLLOADTRANSPOSEMATRIXFARBPROC) (const GLfloat *m); +typedef void (APIENTRYP PFNGLLOADTRANSPOSEMATRIXDARBPROC) (const GLdouble *m); +typedef void (APIENTRYP PFNGLMULTTRANSPOSEMATRIXFARBPROC) (const GLfloat *m); +typedef void (APIENTRYP PFNGLMULTTRANSPOSEMATRIXDARBPROC) (const GLdouble *m); +#endif + +#ifndef GL_ARB_multisample +#define GL_ARB_multisample 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glSampleCoverageARB (GLclampf, GLboolean); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLSAMPLECOVERAGEARBPROC) (GLclampf value, GLboolean invert); +#endif + +#ifndef GL_ARB_texture_env_add +#define GL_ARB_texture_env_add 1 +#endif + +#ifndef GL_ARB_texture_cube_map +#define GL_ARB_texture_cube_map 1 +#endif + +#ifndef GL_ARB_texture_compression +#define GL_ARB_texture_compression 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glCompressedTexImage3DARB (GLenum, GLint, GLenum, GLsizei, GLsizei, GLsizei, GLint, GLsizei, const GLvoid *); +GLAPI void APIENTRY glCompressedTexImage2DARB (GLenum, GLint, GLenum, GLsizei, GLsizei, GLint, GLsizei, const GLvoid *); +GLAPI void APIENTRY glCompressedTexImage1DARB (GLenum, GLint, GLenum, GLsizei, GLint, GLsizei, const GLvoid *); +GLAPI void APIENTRY glCompressedTexSubImage3DARB (GLenum, GLint, GLint, GLint, GLint, GLsizei, GLsizei, GLsizei, GLenum, GLsizei, const GLvoid *); +GLAPI void APIENTRY glCompressedTexSubImage2DARB (GLenum, GLint, GLint, GLint, GLsizei, GLsizei, GLenum, GLsizei, const GLvoid *); +GLAPI void APIENTRY glCompressedTexSubImage1DARB (GLenum, GLint, GLint, GLsizei, GLenum, GLsizei, const GLvoid *); +GLAPI void APIENTRY glGetCompressedTexImageARB (GLenum, GLint, GLvoid *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXIMAGE3DARBPROC) (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLsizei imageSize, const GLvoid *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXIMAGE2DARBPROC) (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLint border, GLsizei imageSize, const GLvoid *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXIMAGE1DARBPROC) (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLint border, GLsizei imageSize, const GLvoid *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXSUBIMAGE3DARBPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLsizei imageSize, const GLvoid *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXSUBIMAGE2DARBPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLsizei imageSize, const GLvoid *data); +typedef void (APIENTRYP PFNGLCOMPRESSEDTEXSUBIMAGE1DARBPROC) (GLenum target, GLint level, GLint xoffset, GLsizei width, GLenum format, GLsizei imageSize, const GLvoid *data); +typedef void (APIENTRYP PFNGLGETCOMPRESSEDTEXIMAGEARBPROC) (GLenum target, GLint level, GLvoid *img); +#endif + +#ifndef GL_ARB_texture_border_clamp +#define GL_ARB_texture_border_clamp 1 +#endif + +#ifndef GL_ARB_point_parameters +#define GL_ARB_point_parameters 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPointParameterfARB (GLenum, GLfloat); +GLAPI void APIENTRY glPointParameterfvARB (GLenum, const GLfloat *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLPOINTPARAMETERFARBPROC) (GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLPOINTPARAMETERFVARBPROC) (GLenum pname, const GLfloat *params); +#endif + +#ifndef GL_ARB_vertex_blend +#define GL_ARB_vertex_blend 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glWeightbvARB (GLint, const GLbyte *); +GLAPI void APIENTRY glWeightsvARB (GLint, const GLshort *); +GLAPI void APIENTRY glWeightivARB (GLint, const GLint *); +GLAPI void APIENTRY glWeightfvARB (GLint, const GLfloat *); +GLAPI void APIENTRY glWeightdvARB (GLint, const GLdouble *); +GLAPI void APIENTRY glWeightubvARB (GLint, const GLubyte *); +GLAPI void APIENTRY glWeightusvARB (GLint, const GLushort *); +GLAPI void APIENTRY glWeightuivARB (GLint, const GLuint *); +GLAPI void APIENTRY glWeightPointerARB (GLint, GLenum, GLsizei, const GLvoid *); +GLAPI void APIENTRY glVertexBlendARB (GLint); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLWEIGHTBVARBPROC) (GLint size, const GLbyte *weights); +typedef void (APIENTRYP PFNGLWEIGHTSVARBPROC) (GLint size, const GLshort *weights); +typedef void (APIENTRYP PFNGLWEIGHTIVARBPROC) (GLint size, const GLint *weights); +typedef void (APIENTRYP PFNGLWEIGHTFVARBPROC) (GLint size, const GLfloat *weights); +typedef void (APIENTRYP PFNGLWEIGHTDVARBPROC) (GLint size, const GLdouble *weights); +typedef void (APIENTRYP PFNGLWEIGHTUBVARBPROC) (GLint size, const GLubyte *weights); +typedef void (APIENTRYP PFNGLWEIGHTUSVARBPROC) (GLint size, const GLushort *weights); +typedef void (APIENTRYP PFNGLWEIGHTUIVARBPROC) (GLint size, const GLuint *weights); +typedef void (APIENTRYP PFNGLWEIGHTPOINTERARBPROC) (GLint size, GLenum type, GLsizei stride, const GLvoid *pointer); +typedef void (APIENTRYP PFNGLVERTEXBLENDARBPROC) (GLint count); +#endif + +#ifndef GL_ARB_matrix_palette +#define GL_ARB_matrix_palette 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glCurrentPaletteMatrixARB (GLint); +GLAPI void APIENTRY glMatrixIndexubvARB (GLint, const GLubyte *); +GLAPI void APIENTRY glMatrixIndexusvARB (GLint, const GLushort *); +GLAPI void APIENTRY glMatrixIndexuivARB (GLint, const GLuint *); +GLAPI void APIENTRY glMatrixIndexPointerARB (GLint, GLenum, GLsizei, const GLvoid *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLCURRENTPALETTEMATRIXARBPROC) (GLint index); +typedef void (APIENTRYP PFNGLMATRIXINDEXUBVARBPROC) (GLint size, const GLubyte *indices); +typedef void (APIENTRYP PFNGLMATRIXINDEXUSVARBPROC) (GLint size, const GLushort *indices); +typedef void (APIENTRYP PFNGLMATRIXINDEXUIVARBPROC) (GLint size, const GLuint *indices); +typedef void (APIENTRYP PFNGLMATRIXINDEXPOINTERARBPROC) (GLint size, GLenum type, GLsizei stride, const GLvoid *pointer); +#endif + +#ifndef GL_ARB_texture_env_combine +#define GL_ARB_texture_env_combine 1 +#endif + +#ifndef GL_ARB_texture_env_crossbar +#define GL_ARB_texture_env_crossbar 1 +#endif + +#ifndef GL_ARB_texture_env_dot3 +#define GL_ARB_texture_env_dot3 1 +#endif + +#ifndef GL_ARB_texture_mirrored_repeat +#define GL_ARB_texture_mirrored_repeat 1 +#endif + +#ifndef GL_ARB_depth_texture +#define GL_ARB_depth_texture 1 +#endif + +#ifndef GL_ARB_shadow +#define GL_ARB_shadow 1 +#endif + +#ifndef GL_ARB_shadow_ambient +#define GL_ARB_shadow_ambient 1 +#endif + +#ifndef GL_ARB_window_pos +#define GL_ARB_window_pos 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glWindowPos2dARB (GLdouble, GLdouble); +GLAPI void APIENTRY glWindowPos2dvARB (const GLdouble *); +GLAPI void APIENTRY glWindowPos2fARB (GLfloat, GLfloat); +GLAPI void APIENTRY glWindowPos2fvARB (const GLfloat *); +GLAPI void APIENTRY glWindowPos2iARB (GLint, GLint); +GLAPI void APIENTRY glWindowPos2ivARB (const GLint *); +GLAPI void APIENTRY glWindowPos2sARB (GLshort, GLshort); +GLAPI void APIENTRY glWindowPos2svARB (const GLshort *); +GLAPI void APIENTRY glWindowPos3dARB (GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glWindowPos3dvARB (const GLdouble *); +GLAPI void APIENTRY glWindowPos3fARB (GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glWindowPos3fvARB (const GLfloat *); +GLAPI void APIENTRY glWindowPos3iARB (GLint, GLint, GLint); +GLAPI void APIENTRY glWindowPos3ivARB (const GLint *); +GLAPI void APIENTRY glWindowPos3sARB (GLshort, GLshort, GLshort); +GLAPI void APIENTRY glWindowPos3svARB (const GLshort *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLWINDOWPOS2DARBPROC) (GLdouble x, GLdouble y); +typedef void (APIENTRYP PFNGLWINDOWPOS2DVARBPROC) (const GLdouble *v); +typedef void (APIENTRYP PFNGLWINDOWPOS2FARBPROC) (GLfloat x, GLfloat y); +typedef void (APIENTRYP PFNGLWINDOWPOS2FVARBPROC) (const GLfloat *v); +typedef void (APIENTRYP PFNGLWINDOWPOS2IARBPROC) (GLint x, GLint y); +typedef void (APIENTRYP PFNGLWINDOWPOS2IVARBPROC) (const GLint *v); +typedef void (APIENTRYP PFNGLWINDOWPOS2SARBPROC) (GLshort x, GLshort y); +typedef void (APIENTRYP PFNGLWINDOWPOS2SVARBPROC) (const GLshort *v); +typedef void (APIENTRYP PFNGLWINDOWPOS3DARBPROC) (GLdouble x, GLdouble y, GLdouble z); +typedef void (APIENTRYP PFNGLWINDOWPOS3DVARBPROC) (const GLdouble *v); +typedef void (APIENTRYP PFNGLWINDOWPOS3FARBPROC) (GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLWINDOWPOS3FVARBPROC) (const GLfloat *v); +typedef void (APIENTRYP PFNGLWINDOWPOS3IARBPROC) (GLint x, GLint y, GLint z); +typedef void (APIENTRYP PFNGLWINDOWPOS3IVARBPROC) (const GLint *v); +typedef void (APIENTRYP PFNGLWINDOWPOS3SARBPROC) (GLshort x, GLshort y, GLshort z); +typedef void (APIENTRYP PFNGLWINDOWPOS3SVARBPROC) (const GLshort *v); +#endif + +#ifndef GL_ARB_vertex_program +#define GL_ARB_vertex_program 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glVertexAttrib1dARB (GLuint, GLdouble); +GLAPI void APIENTRY glVertexAttrib1dvARB (GLuint, const GLdouble *); +GLAPI void APIENTRY glVertexAttrib1fARB (GLuint, GLfloat); +GLAPI void APIENTRY glVertexAttrib1fvARB (GLuint, const GLfloat *); +GLAPI void APIENTRY glVertexAttrib1sARB (GLuint, GLshort); +GLAPI void APIENTRY glVertexAttrib1svARB (GLuint, const GLshort *); +GLAPI void APIENTRY glVertexAttrib2dARB (GLuint, GLdouble, GLdouble); +GLAPI void APIENTRY glVertexAttrib2dvARB (GLuint, const GLdouble *); +GLAPI void APIENTRY glVertexAttrib2fARB (GLuint, GLfloat, GLfloat); +GLAPI void APIENTRY glVertexAttrib2fvARB (GLuint, const GLfloat *); +GLAPI void APIENTRY glVertexAttrib2sARB (GLuint, GLshort, GLshort); +GLAPI void APIENTRY glVertexAttrib2svARB (GLuint, const GLshort *); +GLAPI void APIENTRY glVertexAttrib3dARB (GLuint, GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glVertexAttrib3dvARB (GLuint, const GLdouble *); +GLAPI void APIENTRY glVertexAttrib3fARB (GLuint, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glVertexAttrib3fvARB (GLuint, const GLfloat *); +GLAPI void APIENTRY glVertexAttrib3sARB (GLuint, GLshort, GLshort, GLshort); +GLAPI void APIENTRY glVertexAttrib3svARB (GLuint, const GLshort *); +GLAPI void APIENTRY glVertexAttrib4NbvARB (GLuint, const GLbyte *); +GLAPI void APIENTRY glVertexAttrib4NivARB (GLuint, const GLint *); +GLAPI void APIENTRY glVertexAttrib4NsvARB (GLuint, const GLshort *); +GLAPI void APIENTRY glVertexAttrib4NubARB (GLuint, GLubyte, GLubyte, GLubyte, GLubyte); +GLAPI void APIENTRY glVertexAttrib4NubvARB (GLuint, const GLubyte *); +GLAPI void APIENTRY glVertexAttrib4NuivARB (GLuint, const GLuint *); +GLAPI void APIENTRY glVertexAttrib4NusvARB (GLuint, const GLushort *); +GLAPI void APIENTRY glVertexAttrib4bvARB (GLuint, const GLbyte *); +GLAPI void APIENTRY glVertexAttrib4dARB (GLuint, GLdouble, GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glVertexAttrib4dvARB (GLuint, const GLdouble *); +GLAPI void APIENTRY glVertexAttrib4fARB (GLuint, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glVertexAttrib4fvARB (GLuint, const GLfloat *); +GLAPI void APIENTRY glVertexAttrib4ivARB (GLuint, const GLint *); +GLAPI void APIENTRY glVertexAttrib4sARB (GLuint, GLshort, GLshort, GLshort, GLshort); +GLAPI void APIENTRY glVertexAttrib4svARB (GLuint, const GLshort *); +GLAPI void APIENTRY glVertexAttrib4ubvARB (GLuint, const GLubyte *); +GLAPI void APIENTRY glVertexAttrib4uivARB (GLuint, const GLuint *); +GLAPI void APIENTRY glVertexAttrib4usvARB (GLuint, const GLushort *); +GLAPI void APIENTRY glVertexAttribPointerARB (GLuint, GLint, GLenum, GLboolean, GLsizei, const GLvoid *); +GLAPI void APIENTRY glEnableVertexAttribArrayARB (GLuint); +GLAPI void APIENTRY glDisableVertexAttribArrayARB (GLuint); +GLAPI void APIENTRY glProgramStringARB (GLenum, GLenum, GLsizei, const GLvoid *); +GLAPI void APIENTRY glBindProgramARB (GLenum, GLuint); +GLAPI void APIENTRY glDeleteProgramsARB (GLsizei, const GLuint *); +GLAPI void APIENTRY glGenProgramsARB (GLsizei, GLuint *); +GLAPI void APIENTRY glProgramEnvParameter4dARB (GLenum, GLuint, GLdouble, GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glProgramEnvParameter4dvARB (GLenum, GLuint, const GLdouble *); +GLAPI void APIENTRY glProgramEnvParameter4fARB (GLenum, GLuint, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glProgramEnvParameter4fvARB (GLenum, GLuint, const GLfloat *); +GLAPI void APIENTRY glProgramLocalParameter4dARB (GLenum, GLuint, GLdouble, GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glProgramLocalParameter4dvARB (GLenum, GLuint, const GLdouble *); +GLAPI void APIENTRY glProgramLocalParameter4fARB (GLenum, GLuint, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glProgramLocalParameter4fvARB (GLenum, GLuint, const GLfloat *); +GLAPI void APIENTRY glGetProgramEnvParameterdvARB (GLenum, GLuint, GLdouble *); +GLAPI void APIENTRY glGetProgramEnvParameterfvARB (GLenum, GLuint, GLfloat *); +GLAPI void APIENTRY glGetProgramLocalParameterdvARB (GLenum, GLuint, GLdouble *); +GLAPI void APIENTRY glGetProgramLocalParameterfvARB (GLenum, GLuint, GLfloat *); +GLAPI void APIENTRY glGetProgramivARB (GLenum, GLenum, GLint *); +GLAPI void APIENTRY glGetProgramStringARB (GLenum, GLenum, GLvoid *); +GLAPI void APIENTRY glGetVertexAttribdvARB (GLuint, GLenum, GLdouble *); +GLAPI void APIENTRY glGetVertexAttribfvARB (GLuint, GLenum, GLfloat *); +GLAPI void APIENTRY glGetVertexAttribivARB (GLuint, GLenum, GLint *); +GLAPI void APIENTRY glGetVertexAttribPointervARB (GLuint, GLenum, GLvoid* *); +GLAPI GLboolean APIENTRY glIsProgramARB (GLuint); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLVERTEXATTRIB1DARBPROC) (GLuint index, GLdouble x); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1DVARBPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1FARBPROC) (GLuint index, GLfloat x); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1FVARBPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1SARBPROC) (GLuint index, GLshort x); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1SVARBPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2DARBPROC) (GLuint index, GLdouble x, GLdouble y); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2DVARBPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2FARBPROC) (GLuint index, GLfloat x, GLfloat y); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2FVARBPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2SARBPROC) (GLuint index, GLshort x, GLshort y); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2SVARBPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3DARBPROC) (GLuint index, GLdouble x, GLdouble y, GLdouble z); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3DVARBPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3FARBPROC) (GLuint index, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3FVARBPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3SARBPROC) (GLuint index, GLshort x, GLshort y, GLshort z); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3SVARBPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NBVARBPROC) (GLuint index, const GLbyte *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NIVARBPROC) (GLuint index, const GLint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NSVARBPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NUBARBPROC) (GLuint index, GLubyte x, GLubyte y, GLubyte z, GLubyte w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NUBVARBPROC) (GLuint index, const GLubyte *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NUIVARBPROC) (GLuint index, const GLuint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4NUSVARBPROC) (GLuint index, const GLushort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4BVARBPROC) (GLuint index, const GLbyte *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4DARBPROC) (GLuint index, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4DVARBPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4FARBPROC) (GLuint index, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4FVARBPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4IVARBPROC) (GLuint index, const GLint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4SARBPROC) (GLuint index, GLshort x, GLshort y, GLshort z, GLshort w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4SVARBPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4UBVARBPROC) (GLuint index, const GLubyte *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4UIVARBPROC) (GLuint index, const GLuint *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4USVARBPROC) (GLuint index, const GLushort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBPOINTERARBPROC) (GLuint index, GLint size, GLenum type, GLboolean normalized, GLsizei stride, const GLvoid *pointer); +typedef void (APIENTRYP PFNGLENABLEVERTEXATTRIBARRAYARBPROC) (GLuint index); +typedef void (APIENTRYP PFNGLDISABLEVERTEXATTRIBARRAYARBPROC) (GLuint index); +typedef void (APIENTRYP PFNGLPROGRAMSTRINGARBPROC) (GLenum target, GLenum format, GLsizei len, const GLvoid *string); +typedef void (APIENTRYP PFNGLBINDPROGRAMARBPROC) (GLenum target, GLuint program); +typedef void (APIENTRYP PFNGLDELETEPROGRAMSARBPROC) (GLsizei n, const GLuint *programs); +typedef void (APIENTRYP PFNGLGENPROGRAMSARBPROC) (GLsizei n, GLuint *programs); +typedef void (APIENTRYP PFNGLPROGRAMENVPARAMETER4DARBPROC) (GLenum target, GLuint index, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +typedef void (APIENTRYP PFNGLPROGRAMENVPARAMETER4DVARBPROC) (GLenum target, GLuint index, const GLdouble *params); +typedef void (APIENTRYP PFNGLPROGRAMENVPARAMETER4FARBPROC) (GLenum target, GLuint index, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +typedef void (APIENTRYP PFNGLPROGRAMENVPARAMETER4FVARBPROC) (GLenum target, GLuint index, const GLfloat *params); +typedef void (APIENTRYP PFNGLPROGRAMLOCALPARAMETER4DARBPROC) (GLenum target, GLuint index, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +typedef void (APIENTRYP PFNGLPROGRAMLOCALPARAMETER4DVARBPROC) (GLenum target, GLuint index, const GLdouble *params); +typedef void (APIENTRYP PFNGLPROGRAMLOCALPARAMETER4FARBPROC) (GLenum target, GLuint index, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +typedef void (APIENTRYP PFNGLPROGRAMLOCALPARAMETER4FVARBPROC) (GLenum target, GLuint index, const GLfloat *params); +typedef void (APIENTRYP PFNGLGETPROGRAMENVPARAMETERDVARBPROC) (GLenum target, GLuint index, GLdouble *params); +typedef void (APIENTRYP PFNGLGETPROGRAMENVPARAMETERFVARBPROC) (GLenum target, GLuint index, GLfloat *params); +typedef void (APIENTRYP PFNGLGETPROGRAMLOCALPARAMETERDVARBPROC) (GLenum target, GLuint index, GLdouble *params); +typedef void (APIENTRYP PFNGLGETPROGRAMLOCALPARAMETERFVARBPROC) (GLenum target, GLuint index, GLfloat *params); +typedef void (APIENTRYP PFNGLGETPROGRAMIVARBPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETPROGRAMSTRINGARBPROC) (GLenum target, GLenum pname, GLvoid *string); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBDVARBPROC) (GLuint index, GLenum pname, GLdouble *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBFVARBPROC) (GLuint index, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBIVARBPROC) (GLuint index, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBPOINTERVARBPROC) (GLuint index, GLenum pname, GLvoid* *pointer); +typedef GLboolean (APIENTRYP PFNGLISPROGRAMARBPROC) (GLuint program); +#endif + +#ifndef GL_ARB_fragment_program +#define GL_ARB_fragment_program 1 +/* All ARB_fragment_program entry points are shared with ARB_vertex_program. */ +#endif + +#ifndef GL_ARB_vertex_buffer_object +#define GL_ARB_vertex_buffer_object 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBindBufferARB (GLenum, GLuint); +GLAPI void APIENTRY glDeleteBuffersARB (GLsizei, const GLuint *); +GLAPI void APIENTRY glGenBuffersARB (GLsizei, GLuint *); +GLAPI GLboolean APIENTRY glIsBufferARB (GLuint); +GLAPI void APIENTRY glBufferDataARB (GLenum, GLsizeiptrARB, const GLvoid *, GLenum); +GLAPI void APIENTRY glBufferSubDataARB (GLenum, GLintptrARB, GLsizeiptrARB, const GLvoid *); +GLAPI void APIENTRY glGetBufferSubDataARB (GLenum, GLintptrARB, GLsizeiptrARB, GLvoid *); +GLAPI GLvoid* APIENTRY glMapBufferARB (GLenum, GLenum); +GLAPI GLboolean APIENTRY glUnmapBufferARB (GLenum); +GLAPI void APIENTRY glGetBufferParameterivARB (GLenum, GLenum, GLint *); +GLAPI void APIENTRY glGetBufferPointervARB (GLenum, GLenum, GLvoid* *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLBINDBUFFERARBPROC) (GLenum target, GLuint buffer); +typedef void (APIENTRYP PFNGLDELETEBUFFERSARBPROC) (GLsizei n, const GLuint *buffers); +typedef void (APIENTRYP PFNGLGENBUFFERSARBPROC) (GLsizei n, GLuint *buffers); +typedef GLboolean (APIENTRYP PFNGLISBUFFERARBPROC) (GLuint buffer); +typedef void (APIENTRYP PFNGLBUFFERDATAARBPROC) (GLenum target, GLsizeiptrARB size, const GLvoid *data, GLenum usage); +typedef void (APIENTRYP PFNGLBUFFERSUBDATAARBPROC) (GLenum target, GLintptrARB offset, GLsizeiptrARB size, const GLvoid *data); +typedef void (APIENTRYP PFNGLGETBUFFERSUBDATAARBPROC) (GLenum target, GLintptrARB offset, GLsizeiptrARB size, GLvoid *data); +typedef GLvoid* (APIENTRYP PFNGLMAPBUFFERARBPROC) (GLenum target, GLenum access); +typedef GLboolean (APIENTRYP PFNGLUNMAPBUFFERARBPROC) (GLenum target); +typedef void (APIENTRYP PFNGLGETBUFFERPARAMETERIVARBPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETBUFFERPOINTERVARBPROC) (GLenum target, GLenum pname, GLvoid* *params); +#endif + +#ifndef GL_ARB_occlusion_query +#define GL_ARB_occlusion_query 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGenQueriesARB (GLsizei, GLuint *); +GLAPI void APIENTRY glDeleteQueriesARB (GLsizei, const GLuint *); +GLAPI GLboolean APIENTRY glIsQueryARB (GLuint); +GLAPI void APIENTRY glBeginQueryARB (GLenum, GLuint); +GLAPI void APIENTRY glEndQueryARB (GLenum); +GLAPI void APIENTRY glGetQueryivARB (GLenum, GLenum, GLint *); +GLAPI void APIENTRY glGetQueryObjectivARB (GLuint, GLenum, GLint *); +GLAPI void APIENTRY glGetQueryObjectuivARB (GLuint, GLenum, GLuint *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLGENQUERIESARBPROC) (GLsizei n, GLuint *ids); +typedef void (APIENTRYP PFNGLDELETEQUERIESARBPROC) (GLsizei n, const GLuint *ids); +typedef GLboolean (APIENTRYP PFNGLISQUERYARBPROC) (GLuint id); +typedef void (APIENTRYP PFNGLBEGINQUERYARBPROC) (GLenum target, GLuint id); +typedef void (APIENTRYP PFNGLENDQUERYARBPROC) (GLenum target); +typedef void (APIENTRYP PFNGLGETQUERYIVARBPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETQUERYOBJECTIVARBPROC) (GLuint id, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETQUERYOBJECTUIVARBPROC) (GLuint id, GLenum pname, GLuint *params); +#endif + +#ifndef GL_ARB_shader_objects +#define GL_ARB_shader_objects 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDeleteObjectARB (GLhandleARB); +GLAPI GLhandleARB APIENTRY glGetHandleARB (GLenum); +GLAPI void APIENTRY glDetachObjectARB (GLhandleARB, GLhandleARB); +GLAPI GLhandleARB APIENTRY glCreateShaderObjectARB (GLenum); +GLAPI void APIENTRY glShaderSourceARB (GLhandleARB, GLsizei, const GLcharARB* *, const GLint *); +GLAPI void APIENTRY glCompileShaderARB (GLhandleARB); +GLAPI GLhandleARB APIENTRY glCreateProgramObjectARB (void); +GLAPI void APIENTRY glAttachObjectARB (GLhandleARB, GLhandleARB); +GLAPI void APIENTRY glLinkProgramARB (GLhandleARB); +GLAPI void APIENTRY glUseProgramObjectARB (GLhandleARB); +GLAPI void APIENTRY glValidateProgramARB (GLhandleARB); +GLAPI void APIENTRY glUniform1fARB (GLint, GLfloat); +GLAPI void APIENTRY glUniform2fARB (GLint, GLfloat, GLfloat); +GLAPI void APIENTRY glUniform3fARB (GLint, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glUniform4fARB (GLint, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glUniform1iARB (GLint, GLint); +GLAPI void APIENTRY glUniform2iARB (GLint, GLint, GLint); +GLAPI void APIENTRY glUniform3iARB (GLint, GLint, GLint, GLint); +GLAPI void APIENTRY glUniform4iARB (GLint, GLint, GLint, GLint, GLint); +GLAPI void APIENTRY glUniform1fvARB (GLint, GLsizei, const GLfloat *); +GLAPI void APIENTRY glUniform2fvARB (GLint, GLsizei, const GLfloat *); +GLAPI void APIENTRY glUniform3fvARB (GLint, GLsizei, const GLfloat *); +GLAPI void APIENTRY glUniform4fvARB (GLint, GLsizei, const GLfloat *); +GLAPI void APIENTRY glUniform1ivARB (GLint, GLsizei, const GLint *); +GLAPI void APIENTRY glUniform2ivARB (GLint, GLsizei, const GLint *); +GLAPI void APIENTRY glUniform3ivARB (GLint, GLsizei, const GLint *); +GLAPI void APIENTRY glUniform4ivARB (GLint, GLsizei, const GLint *); +GLAPI void APIENTRY glUniformMatrix2fvARB (GLint, GLsizei, GLboolean, const GLfloat *); +GLAPI void APIENTRY glUniformMatrix3fvARB (GLint, GLsizei, GLboolean, const GLfloat *); +GLAPI void APIENTRY glUniformMatrix4fvARB (GLint, GLsizei, GLboolean, const GLfloat *); +GLAPI void APIENTRY glGetObjectParameterfvARB (GLhandleARB, GLenum, GLfloat *); +GLAPI void APIENTRY glGetObjectParameterivARB (GLhandleARB, GLenum, GLint *); +GLAPI void APIENTRY glGetInfoLogARB (GLhandleARB, GLsizei, GLsizei *, GLcharARB *); +GLAPI void APIENTRY glGetAttachedObjectsARB (GLhandleARB, GLsizei, GLsizei *, GLhandleARB *); +GLAPI GLint APIENTRY glGetUniformLocationARB (GLhandleARB, const GLcharARB *); +GLAPI void APIENTRY glGetActiveUniformARB (GLhandleARB, GLuint, GLsizei, GLsizei *, GLint *, GLenum *, GLcharARB *); +GLAPI void APIENTRY glGetUniformfvARB (GLhandleARB, GLint, GLfloat *); +GLAPI void APIENTRY glGetUniformivARB (GLhandleARB, GLint, GLint *); +GLAPI void APIENTRY glGetShaderSourceARB (GLhandleARB, GLsizei, GLsizei *, GLcharARB *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLDELETEOBJECTARBPROC) (GLhandleARB obj); +typedef GLhandleARB (APIENTRYP PFNGLGETHANDLEARBPROC) (GLenum pname); +typedef void (APIENTRYP PFNGLDETACHOBJECTARBPROC) (GLhandleARB containerObj, GLhandleARB attachedObj); +typedef GLhandleARB (APIENTRYP PFNGLCREATESHADEROBJECTARBPROC) (GLenum shaderType); +typedef void (APIENTRYP PFNGLSHADERSOURCEARBPROC) (GLhandleARB shaderObj, GLsizei count, const GLcharARB* *string, const GLint *length); +typedef void (APIENTRYP PFNGLCOMPILESHADERARBPROC) (GLhandleARB shaderObj); +typedef GLhandleARB (APIENTRYP PFNGLCREATEPROGRAMOBJECTARBPROC) (void); +typedef void (APIENTRYP PFNGLATTACHOBJECTARBPROC) (GLhandleARB containerObj, GLhandleARB obj); +typedef void (APIENTRYP PFNGLLINKPROGRAMARBPROC) (GLhandleARB programObj); +typedef void (APIENTRYP PFNGLUSEPROGRAMOBJECTARBPROC) (GLhandleARB programObj); +typedef void (APIENTRYP PFNGLVALIDATEPROGRAMARBPROC) (GLhandleARB programObj); +typedef void (APIENTRYP PFNGLUNIFORM1FARBPROC) (GLint location, GLfloat v0); +typedef void (APIENTRYP PFNGLUNIFORM2FARBPROC) (GLint location, GLfloat v0, GLfloat v1); +typedef void (APIENTRYP PFNGLUNIFORM3FARBPROC) (GLint location, GLfloat v0, GLfloat v1, GLfloat v2); +typedef void (APIENTRYP PFNGLUNIFORM4FARBPROC) (GLint location, GLfloat v0, GLfloat v1, GLfloat v2, GLfloat v3); +typedef void (APIENTRYP PFNGLUNIFORM1IARBPROC) (GLint location, GLint v0); +typedef void (APIENTRYP PFNGLUNIFORM2IARBPROC) (GLint location, GLint v0, GLint v1); +typedef void (APIENTRYP PFNGLUNIFORM3IARBPROC) (GLint location, GLint v0, GLint v1, GLint v2); +typedef void (APIENTRYP PFNGLUNIFORM4IARBPROC) (GLint location, GLint v0, GLint v1, GLint v2, GLint v3); +typedef void (APIENTRYP PFNGLUNIFORM1FVARBPROC) (GLint location, GLsizei count, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORM2FVARBPROC) (GLint location, GLsizei count, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORM3FVARBPROC) (GLint location, GLsizei count, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORM4FVARBPROC) (GLint location, GLsizei count, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORM1IVARBPROC) (GLint location, GLsizei count, const GLint *value); +typedef void (APIENTRYP PFNGLUNIFORM2IVARBPROC) (GLint location, GLsizei count, const GLint *value); +typedef void (APIENTRYP PFNGLUNIFORM3IVARBPROC) (GLint location, GLsizei count, const GLint *value); +typedef void (APIENTRYP PFNGLUNIFORM4IVARBPROC) (GLint location, GLsizei count, const GLint *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX2FVARBPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX3FVARBPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLUNIFORMMATRIX4FVARBPROC) (GLint location, GLsizei count, GLboolean transpose, const GLfloat *value); +typedef void (APIENTRYP PFNGLGETOBJECTPARAMETERFVARBPROC) (GLhandleARB obj, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETOBJECTPARAMETERIVARBPROC) (GLhandleARB obj, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETINFOLOGARBPROC) (GLhandleARB obj, GLsizei maxLength, GLsizei *length, GLcharARB *infoLog); +typedef void (APIENTRYP PFNGLGETATTACHEDOBJECTSARBPROC) (GLhandleARB containerObj, GLsizei maxCount, GLsizei *count, GLhandleARB *obj); +typedef GLint (APIENTRYP PFNGLGETUNIFORMLOCATIONARBPROC) (GLhandleARB programObj, const GLcharARB *name); +typedef void (APIENTRYP PFNGLGETACTIVEUNIFORMARBPROC) (GLhandleARB programObj, GLuint index, GLsizei maxLength, GLsizei *length, GLint *size, GLenum *type, GLcharARB *name); +typedef void (APIENTRYP PFNGLGETUNIFORMFVARBPROC) (GLhandleARB programObj, GLint location, GLfloat *params); +typedef void (APIENTRYP PFNGLGETUNIFORMIVARBPROC) (GLhandleARB programObj, GLint location, GLint *params); +typedef void (APIENTRYP PFNGLGETSHADERSOURCEARBPROC) (GLhandleARB obj, GLsizei maxLength, GLsizei *length, GLcharARB *source); +#endif + +#ifndef GL_ARB_vertex_shader +#define GL_ARB_vertex_shader 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBindAttribLocationARB (GLhandleARB, GLuint, const GLcharARB *); +GLAPI void APIENTRY glGetActiveAttribARB (GLhandleARB, GLuint, GLsizei, GLsizei *, GLint *, GLenum *, GLcharARB *); +GLAPI GLint APIENTRY glGetAttribLocationARB (GLhandleARB, const GLcharARB *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLBINDATTRIBLOCATIONARBPROC) (GLhandleARB programObj, GLuint index, const GLcharARB *name); +typedef void (APIENTRYP PFNGLGETACTIVEATTRIBARBPROC) (GLhandleARB programObj, GLuint index, GLsizei maxLength, GLsizei *length, GLint *size, GLenum *type, GLcharARB *name); +typedef GLint (APIENTRYP PFNGLGETATTRIBLOCATIONARBPROC) (GLhandleARB programObj, const GLcharARB *name); +#endif + +#ifndef GL_ARB_fragment_shader +#define GL_ARB_fragment_shader 1 +#endif + +#ifndef GL_ARB_shading_language_100 +#define GL_ARB_shading_language_100 1 +#endif + +#ifndef GL_ARB_texture_non_power_of_two +#define GL_ARB_texture_non_power_of_two 1 +#endif + +#ifndef GL_ARB_point_sprite +#define GL_ARB_point_sprite 1 +#endif + +#ifndef GL_ARB_fragment_program_shadow +#define GL_ARB_fragment_program_shadow 1 +#endif + +#ifndef GL_ARB_draw_buffers +#define GL_ARB_draw_buffers 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDrawBuffersARB (GLsizei, const GLenum *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLDRAWBUFFERSARBPROC) (GLsizei n, const GLenum *bufs); +#endif + +#ifndef GL_ARB_texture_rectangle +#define GL_ARB_texture_rectangle 1 +#endif + +#ifndef GL_ARB_color_buffer_float +#define GL_ARB_color_buffer_float 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glClampColorARB (GLenum, GLenum); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLCLAMPCOLORARBPROC) (GLenum target, GLenum clamp); +#endif + +#ifndef GL_ARB_half_float_pixel +#define GL_ARB_half_float_pixel 1 +#endif + +#ifndef GL_ARB_texture_float +#define GL_ARB_texture_float 1 +#endif + +#ifndef GL_ARB_pixel_buffer_object +#define GL_ARB_pixel_buffer_object 1 +#endif + +#ifndef GL_EXT_abgr +#define GL_EXT_abgr 1 +#endif + +#ifndef GL_EXT_blend_color +#define GL_EXT_blend_color 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBlendColorEXT (GLclampf, GLclampf, GLclampf, GLclampf); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLBLENDCOLOREXTPROC) (GLclampf red, GLclampf green, GLclampf blue, GLclampf alpha); +#endif + +#ifndef GL_EXT_polygon_offset +#define GL_EXT_polygon_offset 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPolygonOffsetEXT (GLfloat, GLfloat); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLPOLYGONOFFSETEXTPROC) (GLfloat factor, GLfloat bias); +#endif + +#ifndef GL_EXT_texture +#define GL_EXT_texture 1 +#endif + +#ifndef GL_EXT_texture3D +#define GL_EXT_texture3D 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTexImage3DEXT (GLenum, GLint, GLenum, GLsizei, GLsizei, GLsizei, GLint, GLenum, GLenum, const GLvoid *); +GLAPI void APIENTRY glTexSubImage3DEXT (GLenum, GLint, GLint, GLint, GLint, GLsizei, GLsizei, GLsizei, GLenum, GLenum, const GLvoid *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLTEXIMAGE3DEXTPROC) (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLint border, GLenum format, GLenum type, const GLvoid *pixels); +typedef void (APIENTRYP PFNGLTEXSUBIMAGE3DEXTPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLsizei width, GLsizei height, GLsizei depth, GLenum format, GLenum type, const GLvoid *pixels); +#endif + +#ifndef GL_SGIS_texture_filter4 +#define GL_SGIS_texture_filter4 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGetTexFilterFuncSGIS (GLenum, GLenum, GLfloat *); +GLAPI void APIENTRY glTexFilterFuncSGIS (GLenum, GLenum, GLsizei, const GLfloat *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLGETTEXFILTERFUNCSGISPROC) (GLenum target, GLenum filter, GLfloat *weights); +typedef void (APIENTRYP PFNGLTEXFILTERFUNCSGISPROC) (GLenum target, GLenum filter, GLsizei n, const GLfloat *weights); +#endif + +#ifndef GL_EXT_subtexture +#define GL_EXT_subtexture 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTexSubImage1DEXT (GLenum, GLint, GLint, GLsizei, GLenum, GLenum, const GLvoid *); +GLAPI void APIENTRY glTexSubImage2DEXT (GLenum, GLint, GLint, GLint, GLsizei, GLsizei, GLenum, GLenum, const GLvoid *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLTEXSUBIMAGE1DEXTPROC) (GLenum target, GLint level, GLint xoffset, GLsizei width, GLenum format, GLenum type, const GLvoid *pixels); +typedef void (APIENTRYP PFNGLTEXSUBIMAGE2DEXTPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLenum type, const GLvoid *pixels); +#endif + +#ifndef GL_EXT_copy_texture +#define GL_EXT_copy_texture 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glCopyTexImage1DEXT (GLenum, GLint, GLenum, GLint, GLint, GLsizei, GLint); +GLAPI void APIENTRY glCopyTexImage2DEXT (GLenum, GLint, GLenum, GLint, GLint, GLsizei, GLsizei, GLint); +GLAPI void APIENTRY glCopyTexSubImage1DEXT (GLenum, GLint, GLint, GLint, GLint, GLsizei); +GLAPI void APIENTRY glCopyTexSubImage2DEXT (GLenum, GLint, GLint, GLint, GLint, GLint, GLsizei, GLsizei); +GLAPI void APIENTRY glCopyTexSubImage3DEXT (GLenum, GLint, GLint, GLint, GLint, GLint, GLint, GLsizei, GLsizei); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLCOPYTEXIMAGE1DEXTPROC) (GLenum target, GLint level, GLenum internalformat, GLint x, GLint y, GLsizei width, GLint border); +typedef void (APIENTRYP PFNGLCOPYTEXIMAGE2DEXTPROC) (GLenum target, GLint level, GLenum internalformat, GLint x, GLint y, GLsizei width, GLsizei height, GLint border); +typedef void (APIENTRYP PFNGLCOPYTEXSUBIMAGE1DEXTPROC) (GLenum target, GLint level, GLint xoffset, GLint x, GLint y, GLsizei width); +typedef void (APIENTRYP PFNGLCOPYTEXSUBIMAGE2DEXTPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint x, GLint y, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLCOPYTEXSUBIMAGE3DEXTPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLint x, GLint y, GLsizei width, GLsizei height); +#endif + +#ifndef GL_EXT_histogram +#define GL_EXT_histogram 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGetHistogramEXT (GLenum, GLboolean, GLenum, GLenum, GLvoid *); +GLAPI void APIENTRY glGetHistogramParameterfvEXT (GLenum, GLenum, GLfloat *); +GLAPI void APIENTRY glGetHistogramParameterivEXT (GLenum, GLenum, GLint *); +GLAPI void APIENTRY glGetMinmaxEXT (GLenum, GLboolean, GLenum, GLenum, GLvoid *); +GLAPI void APIENTRY glGetMinmaxParameterfvEXT (GLenum, GLenum, GLfloat *); +GLAPI void APIENTRY glGetMinmaxParameterivEXT (GLenum, GLenum, GLint *); +GLAPI void APIENTRY glHistogramEXT (GLenum, GLsizei, GLenum, GLboolean); +GLAPI void APIENTRY glMinmaxEXT (GLenum, GLenum, GLboolean); +GLAPI void APIENTRY glResetHistogramEXT (GLenum); +GLAPI void APIENTRY glResetMinmaxEXT (GLenum); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLGETHISTOGRAMEXTPROC) (GLenum target, GLboolean reset, GLenum format, GLenum type, GLvoid *values); +typedef void (APIENTRYP PFNGLGETHISTOGRAMPARAMETERFVEXTPROC) (GLenum target, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETHISTOGRAMPARAMETERIVEXTPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETMINMAXEXTPROC) (GLenum target, GLboolean reset, GLenum format, GLenum type, GLvoid *values); +typedef void (APIENTRYP PFNGLGETMINMAXPARAMETERFVEXTPROC) (GLenum target, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETMINMAXPARAMETERIVEXTPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLHISTOGRAMEXTPROC) (GLenum target, GLsizei width, GLenum internalformat, GLboolean sink); +typedef void (APIENTRYP PFNGLMINMAXEXTPROC) (GLenum target, GLenum internalformat, GLboolean sink); +typedef void (APIENTRYP PFNGLRESETHISTOGRAMEXTPROC) (GLenum target); +typedef void (APIENTRYP PFNGLRESETMINMAXEXTPROC) (GLenum target); +#endif + +#ifndef GL_EXT_convolution +#define GL_EXT_convolution 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glConvolutionFilter1DEXT (GLenum, GLenum, GLsizei, GLenum, GLenum, const GLvoid *); +GLAPI void APIENTRY glConvolutionFilter2DEXT (GLenum, GLenum, GLsizei, GLsizei, GLenum, GLenum, const GLvoid *); +GLAPI void APIENTRY glConvolutionParameterfEXT (GLenum, GLenum, GLfloat); +GLAPI void APIENTRY glConvolutionParameterfvEXT (GLenum, GLenum, const GLfloat *); +GLAPI void APIENTRY glConvolutionParameteriEXT (GLenum, GLenum, GLint); +GLAPI void APIENTRY glConvolutionParameterivEXT (GLenum, GLenum, const GLint *); +GLAPI void APIENTRY glCopyConvolutionFilter1DEXT (GLenum, GLenum, GLint, GLint, GLsizei); +GLAPI void APIENTRY glCopyConvolutionFilter2DEXT (GLenum, GLenum, GLint, GLint, GLsizei, GLsizei); +GLAPI void APIENTRY glGetConvolutionFilterEXT (GLenum, GLenum, GLenum, GLvoid *); +GLAPI void APIENTRY glGetConvolutionParameterfvEXT (GLenum, GLenum, GLfloat *); +GLAPI void APIENTRY glGetConvolutionParameterivEXT (GLenum, GLenum, GLint *); +GLAPI void APIENTRY glGetSeparableFilterEXT (GLenum, GLenum, GLenum, GLvoid *, GLvoid *, GLvoid *); +GLAPI void APIENTRY glSeparableFilter2DEXT (GLenum, GLenum, GLsizei, GLsizei, GLenum, GLenum, const GLvoid *, const GLvoid *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLCONVOLUTIONFILTER1DEXTPROC) (GLenum target, GLenum internalformat, GLsizei width, GLenum format, GLenum type, const GLvoid *image); +typedef void (APIENTRYP PFNGLCONVOLUTIONFILTER2DEXTPROC) (GLenum target, GLenum internalformat, GLsizei width, GLsizei height, GLenum format, GLenum type, const GLvoid *image); +typedef void (APIENTRYP PFNGLCONVOLUTIONPARAMETERFEXTPROC) (GLenum target, GLenum pname, GLfloat params); +typedef void (APIENTRYP PFNGLCONVOLUTIONPARAMETERFVEXTPROC) (GLenum target, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLCONVOLUTIONPARAMETERIEXTPROC) (GLenum target, GLenum pname, GLint params); +typedef void (APIENTRYP PFNGLCONVOLUTIONPARAMETERIVEXTPROC) (GLenum target, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLCOPYCONVOLUTIONFILTER1DEXTPROC) (GLenum target, GLenum internalformat, GLint x, GLint y, GLsizei width); +typedef void (APIENTRYP PFNGLCOPYCONVOLUTIONFILTER2DEXTPROC) (GLenum target, GLenum internalformat, GLint x, GLint y, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLGETCONVOLUTIONFILTEREXTPROC) (GLenum target, GLenum format, GLenum type, GLvoid *image); +typedef void (APIENTRYP PFNGLGETCONVOLUTIONPARAMETERFVEXTPROC) (GLenum target, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETCONVOLUTIONPARAMETERIVEXTPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETSEPARABLEFILTEREXTPROC) (GLenum target, GLenum format, GLenum type, GLvoid *row, GLvoid *column, GLvoid *span); +typedef void (APIENTRYP PFNGLSEPARABLEFILTER2DEXTPROC) (GLenum target, GLenum internalformat, GLsizei width, GLsizei height, GLenum format, GLenum type, const GLvoid *row, const GLvoid *column); +#endif + +#ifndef GL_EXT_color_matrix +#define GL_EXT_color_matrix 1 +#endif + +#ifndef GL_SGI_color_table +#define GL_SGI_color_table 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glColorTableSGI (GLenum, GLenum, GLsizei, GLenum, GLenum, const GLvoid *); +GLAPI void APIENTRY glColorTableParameterfvSGI (GLenum, GLenum, const GLfloat *); +GLAPI void APIENTRY glColorTableParameterivSGI (GLenum, GLenum, const GLint *); +GLAPI void APIENTRY glCopyColorTableSGI (GLenum, GLenum, GLint, GLint, GLsizei); +GLAPI void APIENTRY glGetColorTableSGI (GLenum, GLenum, GLenum, GLvoid *); +GLAPI void APIENTRY glGetColorTableParameterfvSGI (GLenum, GLenum, GLfloat *); +GLAPI void APIENTRY glGetColorTableParameterivSGI (GLenum, GLenum, GLint *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLCOLORTABLESGIPROC) (GLenum target, GLenum internalformat, GLsizei width, GLenum format, GLenum type, const GLvoid *table); +typedef void (APIENTRYP PFNGLCOLORTABLEPARAMETERFVSGIPROC) (GLenum target, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLCOLORTABLEPARAMETERIVSGIPROC) (GLenum target, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLCOPYCOLORTABLESGIPROC) (GLenum target, GLenum internalformat, GLint x, GLint y, GLsizei width); +typedef void (APIENTRYP PFNGLGETCOLORTABLESGIPROC) (GLenum target, GLenum format, GLenum type, GLvoid *table); +typedef void (APIENTRYP PFNGLGETCOLORTABLEPARAMETERFVSGIPROC) (GLenum target, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETCOLORTABLEPARAMETERIVSGIPROC) (GLenum target, GLenum pname, GLint *params); +#endif + +#ifndef GL_SGIX_pixel_texture +#define GL_SGIX_pixel_texture 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPixelTexGenSGIX (GLenum); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLPIXELTEXGENSGIXPROC) (GLenum mode); +#endif + +#ifndef GL_SGIS_pixel_texture +#define GL_SGIS_pixel_texture 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPixelTexGenParameteriSGIS (GLenum, GLint); +GLAPI void APIENTRY glPixelTexGenParameterivSGIS (GLenum, const GLint *); +GLAPI void APIENTRY glPixelTexGenParameterfSGIS (GLenum, GLfloat); +GLAPI void APIENTRY glPixelTexGenParameterfvSGIS (GLenum, const GLfloat *); +GLAPI void APIENTRY glGetPixelTexGenParameterivSGIS (GLenum, GLint *); +GLAPI void APIENTRY glGetPixelTexGenParameterfvSGIS (GLenum, GLfloat *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLPIXELTEXGENPARAMETERISGISPROC) (GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLPIXELTEXGENPARAMETERIVSGISPROC) (GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLPIXELTEXGENPARAMETERFSGISPROC) (GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLPIXELTEXGENPARAMETERFVSGISPROC) (GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLGETPIXELTEXGENPARAMETERIVSGISPROC) (GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETPIXELTEXGENPARAMETERFVSGISPROC) (GLenum pname, GLfloat *params); +#endif + +#ifndef GL_SGIS_texture4D +#define GL_SGIS_texture4D 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTexImage4DSGIS (GLenum, GLint, GLenum, GLsizei, GLsizei, GLsizei, GLsizei, GLint, GLenum, GLenum, const GLvoid *); +GLAPI void APIENTRY glTexSubImage4DSGIS (GLenum, GLint, GLint, GLint, GLint, GLint, GLsizei, GLsizei, GLsizei, GLsizei, GLenum, GLenum, const GLvoid *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLTEXIMAGE4DSGISPROC) (GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLsizei depth, GLsizei size4d, GLint border, GLenum format, GLenum type, const GLvoid *pixels); +typedef void (APIENTRYP PFNGLTEXSUBIMAGE4DSGISPROC) (GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint zoffset, GLint woffset, GLsizei width, GLsizei height, GLsizei depth, GLsizei size4d, GLenum format, GLenum type, const GLvoid *pixels); +#endif + +#ifndef GL_SGI_texture_color_table +#define GL_SGI_texture_color_table 1 +#endif + +#ifndef GL_EXT_cmyka +#define GL_EXT_cmyka 1 +#endif + +#ifndef GL_EXT_texture_object +#define GL_EXT_texture_object 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLboolean APIENTRY glAreTexturesResidentEXT (GLsizei, const GLuint *, GLboolean *); +GLAPI void APIENTRY glBindTextureEXT (GLenum, GLuint); +GLAPI void APIENTRY glDeleteTexturesEXT (GLsizei, const GLuint *); +GLAPI void APIENTRY glGenTexturesEXT (GLsizei, GLuint *); +GLAPI GLboolean APIENTRY glIsTextureEXT (GLuint); +GLAPI void APIENTRY glPrioritizeTexturesEXT (GLsizei, const GLuint *, const GLclampf *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef GLboolean (APIENTRYP PFNGLARETEXTURESRESIDENTEXTPROC) (GLsizei n, const GLuint *textures, GLboolean *residences); +typedef void (APIENTRYP PFNGLBINDTEXTUREEXTPROC) (GLenum target, GLuint texture); +typedef void (APIENTRYP PFNGLDELETETEXTURESEXTPROC) (GLsizei n, const GLuint *textures); +typedef void (APIENTRYP PFNGLGENTEXTURESEXTPROC) (GLsizei n, GLuint *textures); +typedef GLboolean (APIENTRYP PFNGLISTEXTUREEXTPROC) (GLuint texture); +typedef void (APIENTRYP PFNGLPRIORITIZETEXTURESEXTPROC) (GLsizei n, const GLuint *textures, const GLclampf *priorities); +#endif + +#ifndef GL_SGIS_detail_texture +#define GL_SGIS_detail_texture 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDetailTexFuncSGIS (GLenum, GLsizei, const GLfloat *); +GLAPI void APIENTRY glGetDetailTexFuncSGIS (GLenum, GLfloat *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLDETAILTEXFUNCSGISPROC) (GLenum target, GLsizei n, const GLfloat *points); +typedef void (APIENTRYP PFNGLGETDETAILTEXFUNCSGISPROC) (GLenum target, GLfloat *points); +#endif + +#ifndef GL_SGIS_sharpen_texture +#define GL_SGIS_sharpen_texture 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glSharpenTexFuncSGIS (GLenum, GLsizei, const GLfloat *); +GLAPI void APIENTRY glGetSharpenTexFuncSGIS (GLenum, GLfloat *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLSHARPENTEXFUNCSGISPROC) (GLenum target, GLsizei n, const GLfloat *points); +typedef void (APIENTRYP PFNGLGETSHARPENTEXFUNCSGISPROC) (GLenum target, GLfloat *points); +#endif + +#ifndef GL_EXT_packed_pixels +#define GL_EXT_packed_pixels 1 +#endif + +#ifndef GL_SGIS_texture_lod +#define GL_SGIS_texture_lod 1 +#endif + +#ifndef GL_SGIS_multisample +#define GL_SGIS_multisample 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glSampleMaskSGIS (GLclampf, GLboolean); +GLAPI void APIENTRY glSamplePatternSGIS (GLenum); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLSAMPLEMASKSGISPROC) (GLclampf value, GLboolean invert); +typedef void (APIENTRYP PFNGLSAMPLEPATTERNSGISPROC) (GLenum pattern); +#endif + +#ifndef GL_EXT_rescale_normal +#define GL_EXT_rescale_normal 1 +#endif + +#ifndef GL_EXT_vertex_array +#define GL_EXT_vertex_array 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glArrayElementEXT (GLint); +GLAPI void APIENTRY glColorPointerEXT (GLint, GLenum, GLsizei, GLsizei, const GLvoid *); +GLAPI void APIENTRY glDrawArraysEXT (GLenum, GLint, GLsizei); +GLAPI void APIENTRY glEdgeFlagPointerEXT (GLsizei, GLsizei, const GLboolean *); +GLAPI void APIENTRY glGetPointervEXT (GLenum, GLvoid* *); +GLAPI void APIENTRY glIndexPointerEXT (GLenum, GLsizei, GLsizei, const GLvoid *); +GLAPI void APIENTRY glNormalPointerEXT (GLenum, GLsizei, GLsizei, const GLvoid *); +GLAPI void APIENTRY glTexCoordPointerEXT (GLint, GLenum, GLsizei, GLsizei, const GLvoid *); +GLAPI void APIENTRY glVertexPointerEXT (GLint, GLenum, GLsizei, GLsizei, const GLvoid *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLARRAYELEMENTEXTPROC) (GLint i); +typedef void (APIENTRYP PFNGLCOLORPOINTEREXTPROC) (GLint size, GLenum type, GLsizei stride, GLsizei count, const GLvoid *pointer); +typedef void (APIENTRYP PFNGLDRAWARRAYSEXTPROC) (GLenum mode, GLint first, GLsizei count); +typedef void (APIENTRYP PFNGLEDGEFLAGPOINTEREXTPROC) (GLsizei stride, GLsizei count, const GLboolean *pointer); +typedef void (APIENTRYP PFNGLGETPOINTERVEXTPROC) (GLenum pname, GLvoid* *params); +typedef void (APIENTRYP PFNGLINDEXPOINTEREXTPROC) (GLenum type, GLsizei stride, GLsizei count, const GLvoid *pointer); +typedef void (APIENTRYP PFNGLNORMALPOINTEREXTPROC) (GLenum type, GLsizei stride, GLsizei count, const GLvoid *pointer); +typedef void (APIENTRYP PFNGLTEXCOORDPOINTEREXTPROC) (GLint size, GLenum type, GLsizei stride, GLsizei count, const GLvoid *pointer); +typedef void (APIENTRYP PFNGLVERTEXPOINTEREXTPROC) (GLint size, GLenum type, GLsizei stride, GLsizei count, const GLvoid *pointer); +#endif + +#ifndef GL_EXT_misc_attribute +#define GL_EXT_misc_attribute 1 +#endif + +#ifndef GL_SGIS_generate_mipmap +#define GL_SGIS_generate_mipmap 1 +#endif + +#ifndef GL_SGIX_clipmap +#define GL_SGIX_clipmap 1 +#endif + +#ifndef GL_SGIX_shadow +#define GL_SGIX_shadow 1 +#endif + +#ifndef GL_SGIS_texture_edge_clamp +#define GL_SGIS_texture_edge_clamp 1 +#endif + +#ifndef GL_SGIS_texture_border_clamp +#define GL_SGIS_texture_border_clamp 1 +#endif + +#ifndef GL_EXT_blend_minmax +#define GL_EXT_blend_minmax 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBlendEquationEXT (GLenum); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLBLENDEQUATIONEXTPROC) (GLenum mode); +#endif + +#ifndef GL_EXT_blend_subtract +#define GL_EXT_blend_subtract 1 +#endif + +#ifndef GL_EXT_blend_logic_op +#define GL_EXT_blend_logic_op 1 +#endif + +#ifndef GL_SGIX_interlace +#define GL_SGIX_interlace 1 +#endif + +#ifndef GL_SGIX_pixel_tiles +#define GL_SGIX_pixel_tiles 1 +#endif + +#ifndef GL_SGIX_texture_select +#define GL_SGIX_texture_select 1 +#endif + +#ifndef GL_SGIX_sprite +#define GL_SGIX_sprite 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glSpriteParameterfSGIX (GLenum, GLfloat); +GLAPI void APIENTRY glSpriteParameterfvSGIX (GLenum, const GLfloat *); +GLAPI void APIENTRY glSpriteParameteriSGIX (GLenum, GLint); +GLAPI void APIENTRY glSpriteParameterivSGIX (GLenum, const GLint *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLSPRITEPARAMETERFSGIXPROC) (GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLSPRITEPARAMETERFVSGIXPROC) (GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLSPRITEPARAMETERISGIXPROC) (GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLSPRITEPARAMETERIVSGIXPROC) (GLenum pname, const GLint *params); +#endif + +#ifndef GL_SGIX_texture_multi_buffer +#define GL_SGIX_texture_multi_buffer 1 +#endif + +#ifndef GL_EXT_point_parameters +#define GL_EXT_point_parameters 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPointParameterfEXT (GLenum, GLfloat); +GLAPI void APIENTRY glPointParameterfvEXT (GLenum, const GLfloat *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLPOINTPARAMETERFEXTPROC) (GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLPOINTPARAMETERFVEXTPROC) (GLenum pname, const GLfloat *params); +#endif + +#ifndef GL_SGIS_point_parameters +#define GL_SGIS_point_parameters 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPointParameterfSGIS (GLenum, GLfloat); +GLAPI void APIENTRY glPointParameterfvSGIS (GLenum, const GLfloat *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLPOINTPARAMETERFSGISPROC) (GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLPOINTPARAMETERFVSGISPROC) (GLenum pname, const GLfloat *params); +#endif + +#ifndef GL_SGIX_instruments +#define GL_SGIX_instruments 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLint APIENTRY glGetInstrumentsSGIX (void); +GLAPI void APIENTRY glInstrumentsBufferSGIX (GLsizei, GLint *); +GLAPI GLint APIENTRY glPollInstrumentsSGIX (GLint *); +GLAPI void APIENTRY glReadInstrumentsSGIX (GLint); +GLAPI void APIENTRY glStartInstrumentsSGIX (void); +GLAPI void APIENTRY glStopInstrumentsSGIX (GLint); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef GLint (APIENTRYP PFNGLGETINSTRUMENTSSGIXPROC) (void); +typedef void (APIENTRYP PFNGLINSTRUMENTSBUFFERSGIXPROC) (GLsizei size, GLint *buffer); +typedef GLint (APIENTRYP PFNGLPOLLINSTRUMENTSSGIXPROC) (GLint *marker_p); +typedef void (APIENTRYP PFNGLREADINSTRUMENTSSGIXPROC) (GLint marker); +typedef void (APIENTRYP PFNGLSTARTINSTRUMENTSSGIXPROC) (void); +typedef void (APIENTRYP PFNGLSTOPINSTRUMENTSSGIXPROC) (GLint marker); +#endif + +#ifndef GL_SGIX_texture_scale_bias +#define GL_SGIX_texture_scale_bias 1 +#endif + +#ifndef GL_SGIX_framezoom +#define GL_SGIX_framezoom 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glFrameZoomSGIX (GLint); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLFRAMEZOOMSGIXPROC) (GLint factor); +#endif + +#ifndef GL_SGIX_tag_sample_buffer +#define GL_SGIX_tag_sample_buffer 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTagSampleBufferSGIX (void); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLTAGSAMPLEBUFFERSGIXPROC) (void); +#endif + +#ifndef GL_SGIX_polynomial_ffd +#define GL_SGIX_polynomial_ffd 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDeformationMap3dSGIX (GLenum, GLdouble, GLdouble, GLint, GLint, GLdouble, GLdouble, GLint, GLint, GLdouble, GLdouble, GLint, GLint, const GLdouble *); +GLAPI void APIENTRY glDeformationMap3fSGIX (GLenum, GLfloat, GLfloat, GLint, GLint, GLfloat, GLfloat, GLint, GLint, GLfloat, GLfloat, GLint, GLint, const GLfloat *); +GLAPI void APIENTRY glDeformSGIX (GLbitfield); +GLAPI void APIENTRY glLoadIdentityDeformationMapSGIX (GLbitfield); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLDEFORMATIONMAP3DSGIXPROC) (GLenum target, GLdouble u1, GLdouble u2, GLint ustride, GLint uorder, GLdouble v1, GLdouble v2, GLint vstride, GLint vorder, GLdouble w1, GLdouble w2, GLint wstride, GLint worder, const GLdouble *points); +typedef void (APIENTRYP PFNGLDEFORMATIONMAP3FSGIXPROC) (GLenum target, GLfloat u1, GLfloat u2, GLint ustride, GLint uorder, GLfloat v1, GLfloat v2, GLint vstride, GLint vorder, GLfloat w1, GLfloat w2, GLint wstride, GLint worder, const GLfloat *points); +typedef void (APIENTRYP PFNGLDEFORMSGIXPROC) (GLbitfield mask); +typedef void (APIENTRYP PFNGLLOADIDENTITYDEFORMATIONMAPSGIXPROC) (GLbitfield mask); +#endif + +#ifndef GL_SGIX_reference_plane +#define GL_SGIX_reference_plane 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glReferencePlaneSGIX (const GLdouble *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLREFERENCEPLANESGIXPROC) (const GLdouble *equation); +#endif + +#ifndef GL_SGIX_flush_raster +#define GL_SGIX_flush_raster 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glFlushRasterSGIX (void); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLFLUSHRASTERSGIXPROC) (void); +#endif + +#ifndef GL_SGIX_depth_texture +#define GL_SGIX_depth_texture 1 +#endif + +#ifndef GL_SGIS_fog_function +#define GL_SGIS_fog_function 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glFogFuncSGIS (GLsizei, const GLfloat *); +GLAPI void APIENTRY glGetFogFuncSGIS (GLfloat *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLFOGFUNCSGISPROC) (GLsizei n, const GLfloat *points); +typedef void (APIENTRYP PFNGLGETFOGFUNCSGISPROC) (GLfloat *points); +#endif + +#ifndef GL_SGIX_fog_offset +#define GL_SGIX_fog_offset 1 +#endif + +#ifndef GL_HP_image_transform +#define GL_HP_image_transform 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glImageTransformParameteriHP (GLenum, GLenum, GLint); +GLAPI void APIENTRY glImageTransformParameterfHP (GLenum, GLenum, GLfloat); +GLAPI void APIENTRY glImageTransformParameterivHP (GLenum, GLenum, const GLint *); +GLAPI void APIENTRY glImageTransformParameterfvHP (GLenum, GLenum, const GLfloat *); +GLAPI void APIENTRY glGetImageTransformParameterivHP (GLenum, GLenum, GLint *); +GLAPI void APIENTRY glGetImageTransformParameterfvHP (GLenum, GLenum, GLfloat *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLIMAGETRANSFORMPARAMETERIHPPROC) (GLenum target, GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLIMAGETRANSFORMPARAMETERFHPPROC) (GLenum target, GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLIMAGETRANSFORMPARAMETERIVHPPROC) (GLenum target, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLIMAGETRANSFORMPARAMETERFVHPPROC) (GLenum target, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLGETIMAGETRANSFORMPARAMETERIVHPPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETIMAGETRANSFORMPARAMETERFVHPPROC) (GLenum target, GLenum pname, GLfloat *params); +#endif + +#ifndef GL_HP_convolution_border_modes +#define GL_HP_convolution_border_modes 1 +#endif + +#ifndef GL_SGIX_texture_add_env +#define GL_SGIX_texture_add_env 1 +#endif + +#ifndef GL_EXT_color_subtable +#define GL_EXT_color_subtable 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glColorSubTableEXT (GLenum, GLsizei, GLsizei, GLenum, GLenum, const GLvoid *); +GLAPI void APIENTRY glCopyColorSubTableEXT (GLenum, GLsizei, GLint, GLint, GLsizei); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLCOLORSUBTABLEEXTPROC) (GLenum target, GLsizei start, GLsizei count, GLenum format, GLenum type, const GLvoid *data); +typedef void (APIENTRYP PFNGLCOPYCOLORSUBTABLEEXTPROC) (GLenum target, GLsizei start, GLint x, GLint y, GLsizei width); +#endif + +#ifndef GL_PGI_vertex_hints +#define GL_PGI_vertex_hints 1 +#endif + +#ifndef GL_PGI_misc_hints +#define GL_PGI_misc_hints 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glHintPGI (GLenum, GLint); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLHINTPGIPROC) (GLenum target, GLint mode); +#endif + +#ifndef GL_EXT_paletted_texture +#define GL_EXT_paletted_texture 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glColorTableEXT (GLenum, GLenum, GLsizei, GLenum, GLenum, const GLvoid *); +GLAPI void APIENTRY glGetColorTableEXT (GLenum, GLenum, GLenum, GLvoid *); +GLAPI void APIENTRY glGetColorTableParameterivEXT (GLenum, GLenum, GLint *); +GLAPI void APIENTRY glGetColorTableParameterfvEXT (GLenum, GLenum, GLfloat *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLCOLORTABLEEXTPROC) (GLenum target, GLenum internalFormat, GLsizei width, GLenum format, GLenum type, const GLvoid *table); +typedef void (APIENTRYP PFNGLGETCOLORTABLEEXTPROC) (GLenum target, GLenum format, GLenum type, GLvoid *data); +typedef void (APIENTRYP PFNGLGETCOLORTABLEPARAMETERIVEXTPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETCOLORTABLEPARAMETERFVEXTPROC) (GLenum target, GLenum pname, GLfloat *params); +#endif + +#ifndef GL_EXT_clip_volume_hint +#define GL_EXT_clip_volume_hint 1 +#endif + +#ifndef GL_SGIX_list_priority +#define GL_SGIX_list_priority 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGetListParameterfvSGIX (GLuint, GLenum, GLfloat *); +GLAPI void APIENTRY glGetListParameterivSGIX (GLuint, GLenum, GLint *); +GLAPI void APIENTRY glListParameterfSGIX (GLuint, GLenum, GLfloat); +GLAPI void APIENTRY glListParameterfvSGIX (GLuint, GLenum, const GLfloat *); +GLAPI void APIENTRY glListParameteriSGIX (GLuint, GLenum, GLint); +GLAPI void APIENTRY glListParameterivSGIX (GLuint, GLenum, const GLint *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLGETLISTPARAMETERFVSGIXPROC) (GLuint list, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETLISTPARAMETERIVSGIXPROC) (GLuint list, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLLISTPARAMETERFSGIXPROC) (GLuint list, GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLLISTPARAMETERFVSGIXPROC) (GLuint list, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLLISTPARAMETERISGIXPROC) (GLuint list, GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLLISTPARAMETERIVSGIXPROC) (GLuint list, GLenum pname, const GLint *params); +#endif + +#ifndef GL_SGIX_ir_instrument1 +#define GL_SGIX_ir_instrument1 1 +#endif + +#ifndef GL_SGIX_calligraphic_fragment +#define GL_SGIX_calligraphic_fragment 1 +#endif + +#ifndef GL_SGIX_texture_lod_bias +#define GL_SGIX_texture_lod_bias 1 +#endif + +#ifndef GL_SGIX_shadow_ambient +#define GL_SGIX_shadow_ambient 1 +#endif + +#ifndef GL_EXT_index_texture +#define GL_EXT_index_texture 1 +#endif + +#ifndef GL_EXT_index_material +#define GL_EXT_index_material 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glIndexMaterialEXT (GLenum, GLenum); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLINDEXMATERIALEXTPROC) (GLenum face, GLenum mode); +#endif + +#ifndef GL_EXT_index_func +#define GL_EXT_index_func 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glIndexFuncEXT (GLenum, GLclampf); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLINDEXFUNCEXTPROC) (GLenum func, GLclampf ref); +#endif + +#ifndef GL_EXT_index_array_formats +#define GL_EXT_index_array_formats 1 +#endif + +#ifndef GL_EXT_compiled_vertex_array +#define GL_EXT_compiled_vertex_array 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glLockArraysEXT (GLint, GLsizei); +GLAPI void APIENTRY glUnlockArraysEXT (void); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLLOCKARRAYSEXTPROC) (GLint first, GLsizei count); +typedef void (APIENTRYP PFNGLUNLOCKARRAYSEXTPROC) (void); +#endif + +#ifndef GL_EXT_cull_vertex +#define GL_EXT_cull_vertex 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glCullParameterdvEXT (GLenum, GLdouble *); +GLAPI void APIENTRY glCullParameterfvEXT (GLenum, GLfloat *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLCULLPARAMETERDVEXTPROC) (GLenum pname, GLdouble *params); +typedef void (APIENTRYP PFNGLCULLPARAMETERFVEXTPROC) (GLenum pname, GLfloat *params); +#endif + +#ifndef GL_SGIX_ycrcb +#define GL_SGIX_ycrcb 1 +#endif + +#ifndef GL_SGIX_fragment_lighting +#define GL_SGIX_fragment_lighting 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glFragmentColorMaterialSGIX (GLenum, GLenum); +GLAPI void APIENTRY glFragmentLightfSGIX (GLenum, GLenum, GLfloat); +GLAPI void APIENTRY glFragmentLightfvSGIX (GLenum, GLenum, const GLfloat *); +GLAPI void APIENTRY glFragmentLightiSGIX (GLenum, GLenum, GLint); +GLAPI void APIENTRY glFragmentLightivSGIX (GLenum, GLenum, const GLint *); +GLAPI void APIENTRY glFragmentLightModelfSGIX (GLenum, GLfloat); +GLAPI void APIENTRY glFragmentLightModelfvSGIX (GLenum, const GLfloat *); +GLAPI void APIENTRY glFragmentLightModeliSGIX (GLenum, GLint); +GLAPI void APIENTRY glFragmentLightModelivSGIX (GLenum, const GLint *); +GLAPI void APIENTRY glFragmentMaterialfSGIX (GLenum, GLenum, GLfloat); +GLAPI void APIENTRY glFragmentMaterialfvSGIX (GLenum, GLenum, const GLfloat *); +GLAPI void APIENTRY glFragmentMaterialiSGIX (GLenum, GLenum, GLint); +GLAPI void APIENTRY glFragmentMaterialivSGIX (GLenum, GLenum, const GLint *); +GLAPI void APIENTRY glGetFragmentLightfvSGIX (GLenum, GLenum, GLfloat *); +GLAPI void APIENTRY glGetFragmentLightivSGIX (GLenum, GLenum, GLint *); +GLAPI void APIENTRY glGetFragmentMaterialfvSGIX (GLenum, GLenum, GLfloat *); +GLAPI void APIENTRY glGetFragmentMaterialivSGIX (GLenum, GLenum, GLint *); +GLAPI void APIENTRY glLightEnviSGIX (GLenum, GLint); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLFRAGMENTCOLORMATERIALSGIXPROC) (GLenum face, GLenum mode); +typedef void (APIENTRYP PFNGLFRAGMENTLIGHTFSGIXPROC) (GLenum light, GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLFRAGMENTLIGHTFVSGIXPROC) (GLenum light, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLFRAGMENTLIGHTISGIXPROC) (GLenum light, GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLFRAGMENTLIGHTIVSGIXPROC) (GLenum light, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLFRAGMENTLIGHTMODELFSGIXPROC) (GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLFRAGMENTLIGHTMODELFVSGIXPROC) (GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLFRAGMENTLIGHTMODELISGIXPROC) (GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLFRAGMENTLIGHTMODELIVSGIXPROC) (GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLFRAGMENTMATERIALFSGIXPROC) (GLenum face, GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLFRAGMENTMATERIALFVSGIXPROC) (GLenum face, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLFRAGMENTMATERIALISGIXPROC) (GLenum face, GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLFRAGMENTMATERIALIVSGIXPROC) (GLenum face, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLGETFRAGMENTLIGHTFVSGIXPROC) (GLenum light, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETFRAGMENTLIGHTIVSGIXPROC) (GLenum light, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETFRAGMENTMATERIALFVSGIXPROC) (GLenum face, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETFRAGMENTMATERIALIVSGIXPROC) (GLenum face, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLLIGHTENVISGIXPROC) (GLenum pname, GLint param); +#endif + +#ifndef GL_IBM_rasterpos_clip +#define GL_IBM_rasterpos_clip 1 +#endif + +#ifndef GL_HP_texture_lighting +#define GL_HP_texture_lighting 1 +#endif + +#ifndef GL_EXT_draw_range_elements +#define GL_EXT_draw_range_elements 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDrawRangeElementsEXT (GLenum, GLuint, GLuint, GLsizei, GLenum, const GLvoid *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLDRAWRANGEELEMENTSEXTPROC) (GLenum mode, GLuint start, GLuint end, GLsizei count, GLenum type, const GLvoid *indices); +#endif + +#ifndef GL_WIN_phong_shading +#define GL_WIN_phong_shading 1 +#endif + +#ifndef GL_WIN_specular_fog +#define GL_WIN_specular_fog 1 +#endif + +#ifndef GL_EXT_light_texture +#define GL_EXT_light_texture 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glApplyTextureEXT (GLenum); +GLAPI void APIENTRY glTextureLightEXT (GLenum); +GLAPI void APIENTRY glTextureMaterialEXT (GLenum, GLenum); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLAPPLYTEXTUREEXTPROC) (GLenum mode); +typedef void (APIENTRYP PFNGLTEXTURELIGHTEXTPROC) (GLenum pname); +typedef void (APIENTRYP PFNGLTEXTUREMATERIALEXTPROC) (GLenum face, GLenum mode); +#endif + +#ifndef GL_SGIX_blend_alpha_minmax +#define GL_SGIX_blend_alpha_minmax 1 +#endif + +#ifndef GL_EXT_bgra +#define GL_EXT_bgra 1 +#endif + +#ifndef GL_SGIX_async +#define GL_SGIX_async 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glAsyncMarkerSGIX (GLuint); +GLAPI GLint APIENTRY glFinishAsyncSGIX (GLuint *); +GLAPI GLint APIENTRY glPollAsyncSGIX (GLuint *); +GLAPI GLuint APIENTRY glGenAsyncMarkersSGIX (GLsizei); +GLAPI void APIENTRY glDeleteAsyncMarkersSGIX (GLuint, GLsizei); +GLAPI GLboolean APIENTRY glIsAsyncMarkerSGIX (GLuint); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLASYNCMARKERSGIXPROC) (GLuint marker); +typedef GLint (APIENTRYP PFNGLFINISHASYNCSGIXPROC) (GLuint *markerp); +typedef GLint (APIENTRYP PFNGLPOLLASYNCSGIXPROC) (GLuint *markerp); +typedef GLuint (APIENTRYP PFNGLGENASYNCMARKERSSGIXPROC) (GLsizei range); +typedef void (APIENTRYP PFNGLDELETEASYNCMARKERSSGIXPROC) (GLuint marker, GLsizei range); +typedef GLboolean (APIENTRYP PFNGLISASYNCMARKERSGIXPROC) (GLuint marker); +#endif + +#ifndef GL_SGIX_async_pixel +#define GL_SGIX_async_pixel 1 +#endif + +#ifndef GL_SGIX_async_histogram +#define GL_SGIX_async_histogram 1 +#endif + +#ifndef GL_INTEL_parallel_arrays +#define GL_INTEL_parallel_arrays 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glVertexPointervINTEL (GLint, GLenum, const GLvoid* *); +GLAPI void APIENTRY glNormalPointervINTEL (GLenum, const GLvoid* *); +GLAPI void APIENTRY glColorPointervINTEL (GLint, GLenum, const GLvoid* *); +GLAPI void APIENTRY glTexCoordPointervINTEL (GLint, GLenum, const GLvoid* *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLVERTEXPOINTERVINTELPROC) (GLint size, GLenum type, const GLvoid* *pointer); +typedef void (APIENTRYP PFNGLNORMALPOINTERVINTELPROC) (GLenum type, const GLvoid* *pointer); +typedef void (APIENTRYP PFNGLCOLORPOINTERVINTELPROC) (GLint size, GLenum type, const GLvoid* *pointer); +typedef void (APIENTRYP PFNGLTEXCOORDPOINTERVINTELPROC) (GLint size, GLenum type, const GLvoid* *pointer); +#endif + +#ifndef GL_HP_occlusion_test +#define GL_HP_occlusion_test 1 +#endif + +#ifndef GL_EXT_pixel_transform +#define GL_EXT_pixel_transform 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPixelTransformParameteriEXT (GLenum, GLenum, GLint); +GLAPI void APIENTRY glPixelTransformParameterfEXT (GLenum, GLenum, GLfloat); +GLAPI void APIENTRY glPixelTransformParameterivEXT (GLenum, GLenum, const GLint *); +GLAPI void APIENTRY glPixelTransformParameterfvEXT (GLenum, GLenum, const GLfloat *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLPIXELTRANSFORMPARAMETERIEXTPROC) (GLenum target, GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLPIXELTRANSFORMPARAMETERFEXTPROC) (GLenum target, GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLPIXELTRANSFORMPARAMETERIVEXTPROC) (GLenum target, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLPIXELTRANSFORMPARAMETERFVEXTPROC) (GLenum target, GLenum pname, const GLfloat *params); +#endif + +#ifndef GL_EXT_pixel_transform_color_table +#define GL_EXT_pixel_transform_color_table 1 +#endif + +#ifndef GL_EXT_shared_texture_palette +#define GL_EXT_shared_texture_palette 1 +#endif + +#ifndef GL_EXT_separate_specular_color +#define GL_EXT_separate_specular_color 1 +#endif + +#ifndef GL_EXT_secondary_color +#define GL_EXT_secondary_color 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glSecondaryColor3bEXT (GLbyte, GLbyte, GLbyte); +GLAPI void APIENTRY glSecondaryColor3bvEXT (const GLbyte *); +GLAPI void APIENTRY glSecondaryColor3dEXT (GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glSecondaryColor3dvEXT (const GLdouble *); +GLAPI void APIENTRY glSecondaryColor3fEXT (GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glSecondaryColor3fvEXT (const GLfloat *); +GLAPI void APIENTRY glSecondaryColor3iEXT (GLint, GLint, GLint); +GLAPI void APIENTRY glSecondaryColor3ivEXT (const GLint *); +GLAPI void APIENTRY glSecondaryColor3sEXT (GLshort, GLshort, GLshort); +GLAPI void APIENTRY glSecondaryColor3svEXT (const GLshort *); +GLAPI void APIENTRY glSecondaryColor3ubEXT (GLubyte, GLubyte, GLubyte); +GLAPI void APIENTRY glSecondaryColor3ubvEXT (const GLubyte *); +GLAPI void APIENTRY glSecondaryColor3uiEXT (GLuint, GLuint, GLuint); +GLAPI void APIENTRY glSecondaryColor3uivEXT (const GLuint *); +GLAPI void APIENTRY glSecondaryColor3usEXT (GLushort, GLushort, GLushort); +GLAPI void APIENTRY glSecondaryColor3usvEXT (const GLushort *); +GLAPI void APIENTRY glSecondaryColorPointerEXT (GLint, GLenum, GLsizei, const GLvoid *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3BEXTPROC) (GLbyte red, GLbyte green, GLbyte blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3BVEXTPROC) (const GLbyte *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3DEXTPROC) (GLdouble red, GLdouble green, GLdouble blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3DVEXTPROC) (const GLdouble *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3FEXTPROC) (GLfloat red, GLfloat green, GLfloat blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3FVEXTPROC) (const GLfloat *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3IEXTPROC) (GLint red, GLint green, GLint blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3IVEXTPROC) (const GLint *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3SEXTPROC) (GLshort red, GLshort green, GLshort blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3SVEXTPROC) (const GLshort *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3UBEXTPROC) (GLubyte red, GLubyte green, GLubyte blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3UBVEXTPROC) (const GLubyte *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3UIEXTPROC) (GLuint red, GLuint green, GLuint blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3UIVEXTPROC) (const GLuint *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3USEXTPROC) (GLushort red, GLushort green, GLushort blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3USVEXTPROC) (const GLushort *v); +typedef void (APIENTRYP PFNGLSECONDARYCOLORPOINTEREXTPROC) (GLint size, GLenum type, GLsizei stride, const GLvoid *pointer); +#endif + +#ifndef GL_EXT_texture_perturb_normal +#define GL_EXT_texture_perturb_normal 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTextureNormalEXT (GLenum); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLTEXTURENORMALEXTPROC) (GLenum mode); +#endif + +#ifndef GL_EXT_multi_draw_arrays +#define GL_EXT_multi_draw_arrays 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glMultiDrawArraysEXT (GLenum, GLint *, GLsizei *, GLsizei); +GLAPI void APIENTRY glMultiDrawElementsEXT (GLenum, const GLsizei *, GLenum, const GLvoid* *, GLsizei); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLMULTIDRAWARRAYSEXTPROC) (GLenum mode, GLint *first, GLsizei *count, GLsizei primcount); +typedef void (APIENTRYP PFNGLMULTIDRAWELEMENTSEXTPROC) (GLenum mode, const GLsizei *count, GLenum type, const GLvoid* *indices, GLsizei primcount); +#endif + +#ifndef GL_EXT_fog_coord +#define GL_EXT_fog_coord 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glFogCoordfEXT (GLfloat); +GLAPI void APIENTRY glFogCoordfvEXT (const GLfloat *); +GLAPI void APIENTRY glFogCoorddEXT (GLdouble); +GLAPI void APIENTRY glFogCoorddvEXT (const GLdouble *); +GLAPI void APIENTRY glFogCoordPointerEXT (GLenum, GLsizei, const GLvoid *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLFOGCOORDFEXTPROC) (GLfloat coord); +typedef void (APIENTRYP PFNGLFOGCOORDFVEXTPROC) (const GLfloat *coord); +typedef void (APIENTRYP PFNGLFOGCOORDDEXTPROC) (GLdouble coord); +typedef void (APIENTRYP PFNGLFOGCOORDDVEXTPROC) (const GLdouble *coord); +typedef void (APIENTRYP PFNGLFOGCOORDPOINTEREXTPROC) (GLenum type, GLsizei stride, const GLvoid *pointer); +#endif + +#ifndef GL_REND_screen_coordinates +#define GL_REND_screen_coordinates 1 +#endif + +#ifndef GL_EXT_coordinate_frame +#define GL_EXT_coordinate_frame 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTangent3bEXT (GLbyte, GLbyte, GLbyte); +GLAPI void APIENTRY glTangent3bvEXT (const GLbyte *); +GLAPI void APIENTRY glTangent3dEXT (GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glTangent3dvEXT (const GLdouble *); +GLAPI void APIENTRY glTangent3fEXT (GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glTangent3fvEXT (const GLfloat *); +GLAPI void APIENTRY glTangent3iEXT (GLint, GLint, GLint); +GLAPI void APIENTRY glTangent3ivEXT (const GLint *); +GLAPI void APIENTRY glTangent3sEXT (GLshort, GLshort, GLshort); +GLAPI void APIENTRY glTangent3svEXT (const GLshort *); +GLAPI void APIENTRY glBinormal3bEXT (GLbyte, GLbyte, GLbyte); +GLAPI void APIENTRY glBinormal3bvEXT (const GLbyte *); +GLAPI void APIENTRY glBinormal3dEXT (GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glBinormal3dvEXT (const GLdouble *); +GLAPI void APIENTRY glBinormal3fEXT (GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glBinormal3fvEXT (const GLfloat *); +GLAPI void APIENTRY glBinormal3iEXT (GLint, GLint, GLint); +GLAPI void APIENTRY glBinormal3ivEXT (const GLint *); +GLAPI void APIENTRY glBinormal3sEXT (GLshort, GLshort, GLshort); +GLAPI void APIENTRY glBinormal3svEXT (const GLshort *); +GLAPI void APIENTRY glTangentPointerEXT (GLenum, GLsizei, const GLvoid *); +GLAPI void APIENTRY glBinormalPointerEXT (GLenum, GLsizei, const GLvoid *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLTANGENT3BEXTPROC) (GLbyte tx, GLbyte ty, GLbyte tz); +typedef void (APIENTRYP PFNGLTANGENT3BVEXTPROC) (const GLbyte *v); +typedef void (APIENTRYP PFNGLTANGENT3DEXTPROC) (GLdouble tx, GLdouble ty, GLdouble tz); +typedef void (APIENTRYP PFNGLTANGENT3DVEXTPROC) (const GLdouble *v); +typedef void (APIENTRYP PFNGLTANGENT3FEXTPROC) (GLfloat tx, GLfloat ty, GLfloat tz); +typedef void (APIENTRYP PFNGLTANGENT3FVEXTPROC) (const GLfloat *v); +typedef void (APIENTRYP PFNGLTANGENT3IEXTPROC) (GLint tx, GLint ty, GLint tz); +typedef void (APIENTRYP PFNGLTANGENT3IVEXTPROC) (const GLint *v); +typedef void (APIENTRYP PFNGLTANGENT3SEXTPROC) (GLshort tx, GLshort ty, GLshort tz); +typedef void (APIENTRYP PFNGLTANGENT3SVEXTPROC) (const GLshort *v); +typedef void (APIENTRYP PFNGLBINORMAL3BEXTPROC) (GLbyte bx, GLbyte by, GLbyte bz); +typedef void (APIENTRYP PFNGLBINORMAL3BVEXTPROC) (const GLbyte *v); +typedef void (APIENTRYP PFNGLBINORMAL3DEXTPROC) (GLdouble bx, GLdouble by, GLdouble bz); +typedef void (APIENTRYP PFNGLBINORMAL3DVEXTPROC) (const GLdouble *v); +typedef void (APIENTRYP PFNGLBINORMAL3FEXTPROC) (GLfloat bx, GLfloat by, GLfloat bz); +typedef void (APIENTRYP PFNGLBINORMAL3FVEXTPROC) (const GLfloat *v); +typedef void (APIENTRYP PFNGLBINORMAL3IEXTPROC) (GLint bx, GLint by, GLint bz); +typedef void (APIENTRYP PFNGLBINORMAL3IVEXTPROC) (const GLint *v); +typedef void (APIENTRYP PFNGLBINORMAL3SEXTPROC) (GLshort bx, GLshort by, GLshort bz); +typedef void (APIENTRYP PFNGLBINORMAL3SVEXTPROC) (const GLshort *v); +typedef void (APIENTRYP PFNGLTANGENTPOINTEREXTPROC) (GLenum type, GLsizei stride, const GLvoid *pointer); +typedef void (APIENTRYP PFNGLBINORMALPOINTEREXTPROC) (GLenum type, GLsizei stride, const GLvoid *pointer); +#endif + +#ifndef GL_EXT_texture_env_combine +#define GL_EXT_texture_env_combine 1 +#endif + +#ifndef GL_APPLE_specular_vector +#define GL_APPLE_specular_vector 1 +#endif + +#ifndef GL_APPLE_transform_hint +#define GL_APPLE_transform_hint 1 +#endif + +#ifndef GL_SGIX_fog_scale +#define GL_SGIX_fog_scale 1 +#endif + +#ifndef GL_SUNX_constant_data +#define GL_SUNX_constant_data 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glFinishTextureSUNX (void); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLFINISHTEXTURESUNXPROC) (void); +#endif + +#ifndef GL_SUN_global_alpha +#define GL_SUN_global_alpha 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGlobalAlphaFactorbSUN (GLbyte); +GLAPI void APIENTRY glGlobalAlphaFactorsSUN (GLshort); +GLAPI void APIENTRY glGlobalAlphaFactoriSUN (GLint); +GLAPI void APIENTRY glGlobalAlphaFactorfSUN (GLfloat); +GLAPI void APIENTRY glGlobalAlphaFactordSUN (GLdouble); +GLAPI void APIENTRY glGlobalAlphaFactorubSUN (GLubyte); +GLAPI void APIENTRY glGlobalAlphaFactorusSUN (GLushort); +GLAPI void APIENTRY glGlobalAlphaFactoruiSUN (GLuint); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLGLOBALALPHAFACTORBSUNPROC) (GLbyte factor); +typedef void (APIENTRYP PFNGLGLOBALALPHAFACTORSSUNPROC) (GLshort factor); +typedef void (APIENTRYP PFNGLGLOBALALPHAFACTORISUNPROC) (GLint factor); +typedef void (APIENTRYP PFNGLGLOBALALPHAFACTORFSUNPROC) (GLfloat factor); +typedef void (APIENTRYP PFNGLGLOBALALPHAFACTORDSUNPROC) (GLdouble factor); +typedef void (APIENTRYP PFNGLGLOBALALPHAFACTORUBSUNPROC) (GLubyte factor); +typedef void (APIENTRYP PFNGLGLOBALALPHAFACTORUSSUNPROC) (GLushort factor); +typedef void (APIENTRYP PFNGLGLOBALALPHAFACTORUISUNPROC) (GLuint factor); +#endif + +#ifndef GL_SUN_triangle_list +#define GL_SUN_triangle_list 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glReplacementCodeuiSUN (GLuint); +GLAPI void APIENTRY glReplacementCodeusSUN (GLushort); +GLAPI void APIENTRY glReplacementCodeubSUN (GLubyte); +GLAPI void APIENTRY glReplacementCodeuivSUN (const GLuint *); +GLAPI void APIENTRY glReplacementCodeusvSUN (const GLushort *); +GLAPI void APIENTRY glReplacementCodeubvSUN (const GLubyte *); +GLAPI void APIENTRY glReplacementCodePointerSUN (GLenum, GLsizei, const GLvoid* *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUISUNPROC) (GLuint code); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUSSUNPROC) (GLushort code); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUBSUNPROC) (GLubyte code); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUIVSUNPROC) (const GLuint *code); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUSVSUNPROC) (const GLushort *code); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUBVSUNPROC) (const GLubyte *code); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEPOINTERSUNPROC) (GLenum type, GLsizei stride, const GLvoid* *pointer); +#endif + +#ifndef GL_SUN_vertex +#define GL_SUN_vertex 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glColor4ubVertex2fSUN (GLubyte, GLubyte, GLubyte, GLubyte, GLfloat, GLfloat); +GLAPI void APIENTRY glColor4ubVertex2fvSUN (const GLubyte *, const GLfloat *); +GLAPI void APIENTRY glColor4ubVertex3fSUN (GLubyte, GLubyte, GLubyte, GLubyte, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glColor4ubVertex3fvSUN (const GLubyte *, const GLfloat *); +GLAPI void APIENTRY glColor3fVertex3fSUN (GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glColor3fVertex3fvSUN (const GLfloat *, const GLfloat *); +GLAPI void APIENTRY glNormal3fVertex3fSUN (GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glNormal3fVertex3fvSUN (const GLfloat *, const GLfloat *); +GLAPI void APIENTRY glColor4fNormal3fVertex3fSUN (GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glColor4fNormal3fVertex3fvSUN (const GLfloat *, const GLfloat *, const GLfloat *); +GLAPI void APIENTRY glTexCoord2fVertex3fSUN (GLfloat, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glTexCoord2fVertex3fvSUN (const GLfloat *, const GLfloat *); +GLAPI void APIENTRY glTexCoord4fVertex4fSUN (GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glTexCoord4fVertex4fvSUN (const GLfloat *, const GLfloat *); +GLAPI void APIENTRY glTexCoord2fColor4ubVertex3fSUN (GLfloat, GLfloat, GLubyte, GLubyte, GLubyte, GLubyte, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glTexCoord2fColor4ubVertex3fvSUN (const GLfloat *, const GLubyte *, const GLfloat *); +GLAPI void APIENTRY glTexCoord2fColor3fVertex3fSUN (GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glTexCoord2fColor3fVertex3fvSUN (const GLfloat *, const GLfloat *, const GLfloat *); +GLAPI void APIENTRY glTexCoord2fNormal3fVertex3fSUN (GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glTexCoord2fNormal3fVertex3fvSUN (const GLfloat *, const GLfloat *, const GLfloat *); +GLAPI void APIENTRY glTexCoord2fColor4fNormal3fVertex3fSUN (GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glTexCoord2fColor4fNormal3fVertex3fvSUN (const GLfloat *, const GLfloat *, const GLfloat *, const GLfloat *); +GLAPI void APIENTRY glTexCoord4fColor4fNormal3fVertex4fSUN (GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glTexCoord4fColor4fNormal3fVertex4fvSUN (const GLfloat *, const GLfloat *, const GLfloat *, const GLfloat *); +GLAPI void APIENTRY glReplacementCodeuiVertex3fSUN (GLuint, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glReplacementCodeuiVertex3fvSUN (const GLuint *, const GLfloat *); +GLAPI void APIENTRY glReplacementCodeuiColor4ubVertex3fSUN (GLuint, GLubyte, GLubyte, GLubyte, GLubyte, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glReplacementCodeuiColor4ubVertex3fvSUN (const GLuint *, const GLubyte *, const GLfloat *); +GLAPI void APIENTRY glReplacementCodeuiColor3fVertex3fSUN (GLuint, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glReplacementCodeuiColor3fVertex3fvSUN (const GLuint *, const GLfloat *, const GLfloat *); +GLAPI void APIENTRY glReplacementCodeuiNormal3fVertex3fSUN (GLuint, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glReplacementCodeuiNormal3fVertex3fvSUN (const GLuint *, const GLfloat *, const GLfloat *); +GLAPI void APIENTRY glReplacementCodeuiColor4fNormal3fVertex3fSUN (GLuint, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glReplacementCodeuiColor4fNormal3fVertex3fvSUN (const GLuint *, const GLfloat *, const GLfloat *, const GLfloat *); +GLAPI void APIENTRY glReplacementCodeuiTexCoord2fVertex3fSUN (GLuint, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glReplacementCodeuiTexCoord2fVertex3fvSUN (const GLuint *, const GLfloat *, const GLfloat *); +GLAPI void APIENTRY glReplacementCodeuiTexCoord2fNormal3fVertex3fSUN (GLuint, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glReplacementCodeuiTexCoord2fNormal3fVertex3fvSUN (const GLuint *, const GLfloat *, const GLfloat *, const GLfloat *); +GLAPI void APIENTRY glReplacementCodeuiTexCoord2fColor4fNormal3fVertex3fSUN (GLuint, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glReplacementCodeuiTexCoord2fColor4fNormal3fVertex3fvSUN (const GLuint *, const GLfloat *, const GLfloat *, const GLfloat *, const GLfloat *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLCOLOR4UBVERTEX2FSUNPROC) (GLubyte r, GLubyte g, GLubyte b, GLubyte a, GLfloat x, GLfloat y); +typedef void (APIENTRYP PFNGLCOLOR4UBVERTEX2FVSUNPROC) (const GLubyte *c, const GLfloat *v); +typedef void (APIENTRYP PFNGLCOLOR4UBVERTEX3FSUNPROC) (GLubyte r, GLubyte g, GLubyte b, GLubyte a, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLCOLOR4UBVERTEX3FVSUNPROC) (const GLubyte *c, const GLfloat *v); +typedef void (APIENTRYP PFNGLCOLOR3FVERTEX3FSUNPROC) (GLfloat r, GLfloat g, GLfloat b, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLCOLOR3FVERTEX3FVSUNPROC) (const GLfloat *c, const GLfloat *v); +typedef void (APIENTRYP PFNGLNORMAL3FVERTEX3FSUNPROC) (GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLNORMAL3FVERTEX3FVSUNPROC) (const GLfloat *n, const GLfloat *v); +typedef void (APIENTRYP PFNGLCOLOR4FNORMAL3FVERTEX3FSUNPROC) (GLfloat r, GLfloat g, GLfloat b, GLfloat a, GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLCOLOR4FNORMAL3FVERTEX3FVSUNPROC) (const GLfloat *c, const GLfloat *n, const GLfloat *v); +typedef void (APIENTRYP PFNGLTEXCOORD2FVERTEX3FSUNPROC) (GLfloat s, GLfloat t, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLTEXCOORD2FVERTEX3FVSUNPROC) (const GLfloat *tc, const GLfloat *v); +typedef void (APIENTRYP PFNGLTEXCOORD4FVERTEX4FSUNPROC) (GLfloat s, GLfloat t, GLfloat p, GLfloat q, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +typedef void (APIENTRYP PFNGLTEXCOORD4FVERTEX4FVSUNPROC) (const GLfloat *tc, const GLfloat *v); +typedef void (APIENTRYP PFNGLTEXCOORD2FCOLOR4UBVERTEX3FSUNPROC) (GLfloat s, GLfloat t, GLubyte r, GLubyte g, GLubyte b, GLubyte a, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLTEXCOORD2FCOLOR4UBVERTEX3FVSUNPROC) (const GLfloat *tc, const GLubyte *c, const GLfloat *v); +typedef void (APIENTRYP PFNGLTEXCOORD2FCOLOR3FVERTEX3FSUNPROC) (GLfloat s, GLfloat t, GLfloat r, GLfloat g, GLfloat b, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLTEXCOORD2FCOLOR3FVERTEX3FVSUNPROC) (const GLfloat *tc, const GLfloat *c, const GLfloat *v); +typedef void (APIENTRYP PFNGLTEXCOORD2FNORMAL3FVERTEX3FSUNPROC) (GLfloat s, GLfloat t, GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLTEXCOORD2FNORMAL3FVERTEX3FVSUNPROC) (const GLfloat *tc, const GLfloat *n, const GLfloat *v); +typedef void (APIENTRYP PFNGLTEXCOORD2FCOLOR4FNORMAL3FVERTEX3FSUNPROC) (GLfloat s, GLfloat t, GLfloat r, GLfloat g, GLfloat b, GLfloat a, GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLTEXCOORD2FCOLOR4FNORMAL3FVERTEX3FVSUNPROC) (const GLfloat *tc, const GLfloat *c, const GLfloat *n, const GLfloat *v); +typedef void (APIENTRYP PFNGLTEXCOORD4FCOLOR4FNORMAL3FVERTEX4FSUNPROC) (GLfloat s, GLfloat t, GLfloat p, GLfloat q, GLfloat r, GLfloat g, GLfloat b, GLfloat a, GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +typedef void (APIENTRYP PFNGLTEXCOORD4FCOLOR4FNORMAL3FVERTEX4FVSUNPROC) (const GLfloat *tc, const GLfloat *c, const GLfloat *n, const GLfloat *v); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUIVERTEX3FSUNPROC) (GLuint rc, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUIVERTEX3FVSUNPROC) (const GLuint *rc, const GLfloat *v); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUICOLOR4UBVERTEX3FSUNPROC) (GLuint rc, GLubyte r, GLubyte g, GLubyte b, GLubyte a, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUICOLOR4UBVERTEX3FVSUNPROC) (const GLuint *rc, const GLubyte *c, const GLfloat *v); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUICOLOR3FVERTEX3FSUNPROC) (GLuint rc, GLfloat r, GLfloat g, GLfloat b, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUICOLOR3FVERTEX3FVSUNPROC) (const GLuint *rc, const GLfloat *c, const GLfloat *v); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUINORMAL3FVERTEX3FSUNPROC) (GLuint rc, GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUINORMAL3FVERTEX3FVSUNPROC) (const GLuint *rc, const GLfloat *n, const GLfloat *v); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUICOLOR4FNORMAL3FVERTEX3FSUNPROC) (GLuint rc, GLfloat r, GLfloat g, GLfloat b, GLfloat a, GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUICOLOR4FNORMAL3FVERTEX3FVSUNPROC) (const GLuint *rc, const GLfloat *c, const GLfloat *n, const GLfloat *v); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUITEXCOORD2FVERTEX3FSUNPROC) (GLuint rc, GLfloat s, GLfloat t, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUITEXCOORD2FVERTEX3FVSUNPROC) (const GLuint *rc, const GLfloat *tc, const GLfloat *v); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUITEXCOORD2FNORMAL3FVERTEX3FSUNPROC) (GLuint rc, GLfloat s, GLfloat t, GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUITEXCOORD2FNORMAL3FVERTEX3FVSUNPROC) (const GLuint *rc, const GLfloat *tc, const GLfloat *n, const GLfloat *v); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUITEXCOORD2FCOLOR4FNORMAL3FVERTEX3FSUNPROC) (GLuint rc, GLfloat s, GLfloat t, GLfloat r, GLfloat g, GLfloat b, GLfloat a, GLfloat nx, GLfloat ny, GLfloat nz, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLREPLACEMENTCODEUITEXCOORD2FCOLOR4FNORMAL3FVERTEX3FVSUNPROC) (const GLuint *rc, const GLfloat *tc, const GLfloat *c, const GLfloat *n, const GLfloat *v); +#endif + +#ifndef GL_EXT_blend_func_separate +#define GL_EXT_blend_func_separate 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBlendFuncSeparateEXT (GLenum, GLenum, GLenum, GLenum); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLBLENDFUNCSEPARATEEXTPROC) (GLenum sfactorRGB, GLenum dfactorRGB, GLenum sfactorAlpha, GLenum dfactorAlpha); +#endif + +#ifndef GL_INGR_blend_func_separate +#define GL_INGR_blend_func_separate 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBlendFuncSeparateINGR (GLenum, GLenum, GLenum, GLenum); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLBLENDFUNCSEPARATEINGRPROC) (GLenum sfactorRGB, GLenum dfactorRGB, GLenum sfactorAlpha, GLenum dfactorAlpha); +#endif + +#ifndef GL_INGR_color_clamp +#define GL_INGR_color_clamp 1 +#endif + +#ifndef GL_INGR_interlace_read +#define GL_INGR_interlace_read 1 +#endif + +#ifndef GL_EXT_stencil_wrap +#define GL_EXT_stencil_wrap 1 +#endif + +#ifndef GL_EXT_422_pixels +#define GL_EXT_422_pixels 1 +#endif + +#ifndef GL_NV_texgen_reflection +#define GL_NV_texgen_reflection 1 +#endif + +#ifndef GL_SUN_convolution_border_modes +#define GL_SUN_convolution_border_modes 1 +#endif + +#ifndef GL_EXT_texture_env_add +#define GL_EXT_texture_env_add 1 +#endif + +#ifndef GL_EXT_texture_lod_bias +#define GL_EXT_texture_lod_bias 1 +#endif + +#ifndef GL_EXT_texture_filter_anisotropic +#define GL_EXT_texture_filter_anisotropic 1 +#endif + +#ifndef GL_EXT_vertex_weighting +#define GL_EXT_vertex_weighting 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glVertexWeightfEXT (GLfloat); +GLAPI void APIENTRY glVertexWeightfvEXT (const GLfloat *); +GLAPI void APIENTRY glVertexWeightPointerEXT (GLsizei, GLenum, GLsizei, const GLvoid *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLVERTEXWEIGHTFEXTPROC) (GLfloat weight); +typedef void (APIENTRYP PFNGLVERTEXWEIGHTFVEXTPROC) (const GLfloat *weight); +typedef void (APIENTRYP PFNGLVERTEXWEIGHTPOINTEREXTPROC) (GLsizei size, GLenum type, GLsizei stride, const GLvoid *pointer); +#endif + +#ifndef GL_NV_light_max_exponent +#define GL_NV_light_max_exponent 1 +#endif + +#ifndef GL_NV_vertex_array_range +#define GL_NV_vertex_array_range 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glFlushVertexArrayRangeNV (void); +GLAPI void APIENTRY glVertexArrayRangeNV (GLsizei, const GLvoid *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLFLUSHVERTEXARRAYRANGENVPROC) (void); +typedef void (APIENTRYP PFNGLVERTEXARRAYRANGENVPROC) (GLsizei length, const GLvoid *pointer); +#endif + +#ifndef GL_NV_register_combiners +#define GL_NV_register_combiners 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glCombinerParameterfvNV (GLenum, const GLfloat *); +GLAPI void APIENTRY glCombinerParameterfNV (GLenum, GLfloat); +GLAPI void APIENTRY glCombinerParameterivNV (GLenum, const GLint *); +GLAPI void APIENTRY glCombinerParameteriNV (GLenum, GLint); +GLAPI void APIENTRY glCombinerInputNV (GLenum, GLenum, GLenum, GLenum, GLenum, GLenum); +GLAPI void APIENTRY glCombinerOutputNV (GLenum, GLenum, GLenum, GLenum, GLenum, GLenum, GLenum, GLboolean, GLboolean, GLboolean); +GLAPI void APIENTRY glFinalCombinerInputNV (GLenum, GLenum, GLenum, GLenum); +GLAPI void APIENTRY glGetCombinerInputParameterfvNV (GLenum, GLenum, GLenum, GLenum, GLfloat *); +GLAPI void APIENTRY glGetCombinerInputParameterivNV (GLenum, GLenum, GLenum, GLenum, GLint *); +GLAPI void APIENTRY glGetCombinerOutputParameterfvNV (GLenum, GLenum, GLenum, GLfloat *); +GLAPI void APIENTRY glGetCombinerOutputParameterivNV (GLenum, GLenum, GLenum, GLint *); +GLAPI void APIENTRY glGetFinalCombinerInputParameterfvNV (GLenum, GLenum, GLfloat *); +GLAPI void APIENTRY glGetFinalCombinerInputParameterivNV (GLenum, GLenum, GLint *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLCOMBINERPARAMETERFVNVPROC) (GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLCOMBINERPARAMETERFNVPROC) (GLenum pname, GLfloat param); +typedef void (APIENTRYP PFNGLCOMBINERPARAMETERIVNVPROC) (GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLCOMBINERPARAMETERINVPROC) (GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLCOMBINERINPUTNVPROC) (GLenum stage, GLenum portion, GLenum variable, GLenum input, GLenum mapping, GLenum componentUsage); +typedef void (APIENTRYP PFNGLCOMBINEROUTPUTNVPROC) (GLenum stage, GLenum portion, GLenum abOutput, GLenum cdOutput, GLenum sumOutput, GLenum scale, GLenum bias, GLboolean abDotProduct, GLboolean cdDotProduct, GLboolean muxSum); +typedef void (APIENTRYP PFNGLFINALCOMBINERINPUTNVPROC) (GLenum variable, GLenum input, GLenum mapping, GLenum componentUsage); +typedef void (APIENTRYP PFNGLGETCOMBINERINPUTPARAMETERFVNVPROC) (GLenum stage, GLenum portion, GLenum variable, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETCOMBINERINPUTPARAMETERIVNVPROC) (GLenum stage, GLenum portion, GLenum variable, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETCOMBINEROUTPUTPARAMETERFVNVPROC) (GLenum stage, GLenum portion, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETCOMBINEROUTPUTPARAMETERIVNVPROC) (GLenum stage, GLenum portion, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETFINALCOMBINERINPUTPARAMETERFVNVPROC) (GLenum variable, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETFINALCOMBINERINPUTPARAMETERIVNVPROC) (GLenum variable, GLenum pname, GLint *params); +#endif + +#ifndef GL_NV_fog_distance +#define GL_NV_fog_distance 1 +#endif + +#ifndef GL_NV_texgen_emboss +#define GL_NV_texgen_emboss 1 +#endif + +#ifndef GL_NV_blend_square +#define GL_NV_blend_square 1 +#endif + +#ifndef GL_NV_texture_env_combine4 +#define GL_NV_texture_env_combine4 1 +#endif + +#ifndef GL_MESA_resize_buffers +#define GL_MESA_resize_buffers 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glResizeBuffersMESA (void); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLRESIZEBUFFERSMESAPROC) (void); +#endif + +#ifndef GL_MESA_window_pos +#define GL_MESA_window_pos 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glWindowPos2dMESA (GLdouble, GLdouble); +GLAPI void APIENTRY glWindowPos2dvMESA (const GLdouble *); +GLAPI void APIENTRY glWindowPos2fMESA (GLfloat, GLfloat); +GLAPI void APIENTRY glWindowPos2fvMESA (const GLfloat *); +GLAPI void APIENTRY glWindowPos2iMESA (GLint, GLint); +GLAPI void APIENTRY glWindowPos2ivMESA (const GLint *); +GLAPI void APIENTRY glWindowPos2sMESA (GLshort, GLshort); +GLAPI void APIENTRY glWindowPos2svMESA (const GLshort *); +GLAPI void APIENTRY glWindowPos3dMESA (GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glWindowPos3dvMESA (const GLdouble *); +GLAPI void APIENTRY glWindowPos3fMESA (GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glWindowPos3fvMESA (const GLfloat *); +GLAPI void APIENTRY glWindowPos3iMESA (GLint, GLint, GLint); +GLAPI void APIENTRY glWindowPos3ivMESA (const GLint *); +GLAPI void APIENTRY glWindowPos3sMESA (GLshort, GLshort, GLshort); +GLAPI void APIENTRY glWindowPos3svMESA (const GLshort *); +GLAPI void APIENTRY glWindowPos4dMESA (GLdouble, GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glWindowPos4dvMESA (const GLdouble *); +GLAPI void APIENTRY glWindowPos4fMESA (GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glWindowPos4fvMESA (const GLfloat *); +GLAPI void APIENTRY glWindowPos4iMESA (GLint, GLint, GLint, GLint); +GLAPI void APIENTRY glWindowPos4ivMESA (const GLint *); +GLAPI void APIENTRY glWindowPos4sMESA (GLshort, GLshort, GLshort, GLshort); +GLAPI void APIENTRY glWindowPos4svMESA (const GLshort *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLWINDOWPOS2DMESAPROC) (GLdouble x, GLdouble y); +typedef void (APIENTRYP PFNGLWINDOWPOS2DVMESAPROC) (const GLdouble *v); +typedef void (APIENTRYP PFNGLWINDOWPOS2FMESAPROC) (GLfloat x, GLfloat y); +typedef void (APIENTRYP PFNGLWINDOWPOS2FVMESAPROC) (const GLfloat *v); +typedef void (APIENTRYP PFNGLWINDOWPOS2IMESAPROC) (GLint x, GLint y); +typedef void (APIENTRYP PFNGLWINDOWPOS2IVMESAPROC) (const GLint *v); +typedef void (APIENTRYP PFNGLWINDOWPOS2SMESAPROC) (GLshort x, GLshort y); +typedef void (APIENTRYP PFNGLWINDOWPOS2SVMESAPROC) (const GLshort *v); +typedef void (APIENTRYP PFNGLWINDOWPOS3DMESAPROC) (GLdouble x, GLdouble y, GLdouble z); +typedef void (APIENTRYP PFNGLWINDOWPOS3DVMESAPROC) (const GLdouble *v); +typedef void (APIENTRYP PFNGLWINDOWPOS3FMESAPROC) (GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLWINDOWPOS3FVMESAPROC) (const GLfloat *v); +typedef void (APIENTRYP PFNGLWINDOWPOS3IMESAPROC) (GLint x, GLint y, GLint z); +typedef void (APIENTRYP PFNGLWINDOWPOS3IVMESAPROC) (const GLint *v); +typedef void (APIENTRYP PFNGLWINDOWPOS3SMESAPROC) (GLshort x, GLshort y, GLshort z); +typedef void (APIENTRYP PFNGLWINDOWPOS3SVMESAPROC) (const GLshort *v); +typedef void (APIENTRYP PFNGLWINDOWPOS4DMESAPROC) (GLdouble x, GLdouble y, GLdouble z, GLdouble w); +typedef void (APIENTRYP PFNGLWINDOWPOS4DVMESAPROC) (const GLdouble *v); +typedef void (APIENTRYP PFNGLWINDOWPOS4FMESAPROC) (GLfloat x, GLfloat y, GLfloat z, GLfloat w); +typedef void (APIENTRYP PFNGLWINDOWPOS4FVMESAPROC) (const GLfloat *v); +typedef void (APIENTRYP PFNGLWINDOWPOS4IMESAPROC) (GLint x, GLint y, GLint z, GLint w); +typedef void (APIENTRYP PFNGLWINDOWPOS4IVMESAPROC) (const GLint *v); +typedef void (APIENTRYP PFNGLWINDOWPOS4SMESAPROC) (GLshort x, GLshort y, GLshort z, GLshort w); +typedef void (APIENTRYP PFNGLWINDOWPOS4SVMESAPROC) (const GLshort *v); +#endif + +#ifndef GL_IBM_cull_vertex +#define GL_IBM_cull_vertex 1 +#endif + +#ifndef GL_IBM_multimode_draw_arrays +#define GL_IBM_multimode_draw_arrays 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glMultiModeDrawArraysIBM (const GLenum *, const GLint *, const GLsizei *, GLsizei, GLint); +GLAPI void APIENTRY glMultiModeDrawElementsIBM (const GLenum *, const GLsizei *, GLenum, const GLvoid* const *, GLsizei, GLint); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLMULTIMODEDRAWARRAYSIBMPROC) (const GLenum *mode, const GLint *first, const GLsizei *count, GLsizei primcount, GLint modestride); +typedef void (APIENTRYP PFNGLMULTIMODEDRAWELEMENTSIBMPROC) (const GLenum *mode, const GLsizei *count, GLenum type, const GLvoid* const *indices, GLsizei primcount, GLint modestride); +#endif + +#ifndef GL_IBM_vertex_array_lists +#define GL_IBM_vertex_array_lists 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glColorPointerListIBM (GLint, GLenum, GLint, const GLvoid* *, GLint); +GLAPI void APIENTRY glSecondaryColorPointerListIBM (GLint, GLenum, GLint, const GLvoid* *, GLint); +GLAPI void APIENTRY glEdgeFlagPointerListIBM (GLint, const GLboolean* *, GLint); +GLAPI void APIENTRY glFogCoordPointerListIBM (GLenum, GLint, const GLvoid* *, GLint); +GLAPI void APIENTRY glIndexPointerListIBM (GLenum, GLint, const GLvoid* *, GLint); +GLAPI void APIENTRY glNormalPointerListIBM (GLenum, GLint, const GLvoid* *, GLint); +GLAPI void APIENTRY glTexCoordPointerListIBM (GLint, GLenum, GLint, const GLvoid* *, GLint); +GLAPI void APIENTRY glVertexPointerListIBM (GLint, GLenum, GLint, const GLvoid* *, GLint); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLCOLORPOINTERLISTIBMPROC) (GLint size, GLenum type, GLint stride, const GLvoid* *pointer, GLint ptrstride); +typedef void (APIENTRYP PFNGLSECONDARYCOLORPOINTERLISTIBMPROC) (GLint size, GLenum type, GLint stride, const GLvoid* *pointer, GLint ptrstride); +typedef void (APIENTRYP PFNGLEDGEFLAGPOINTERLISTIBMPROC) (GLint stride, const GLboolean* *pointer, GLint ptrstride); +typedef void (APIENTRYP PFNGLFOGCOORDPOINTERLISTIBMPROC) (GLenum type, GLint stride, const GLvoid* *pointer, GLint ptrstride); +typedef void (APIENTRYP PFNGLINDEXPOINTERLISTIBMPROC) (GLenum type, GLint stride, const GLvoid* *pointer, GLint ptrstride); +typedef void (APIENTRYP PFNGLNORMALPOINTERLISTIBMPROC) (GLenum type, GLint stride, const GLvoid* *pointer, GLint ptrstride); +typedef void (APIENTRYP PFNGLTEXCOORDPOINTERLISTIBMPROC) (GLint size, GLenum type, GLint stride, const GLvoid* *pointer, GLint ptrstride); +typedef void (APIENTRYP PFNGLVERTEXPOINTERLISTIBMPROC) (GLint size, GLenum type, GLint stride, const GLvoid* *pointer, GLint ptrstride); +#endif + +#ifndef GL_SGIX_subsample +#define GL_SGIX_subsample 1 +#endif + +#ifndef GL_SGIX_ycrcba +#define GL_SGIX_ycrcba 1 +#endif + +#ifndef GL_SGIX_ycrcb_subsample +#define GL_SGIX_ycrcb_subsample 1 +#endif + +#ifndef GL_SGIX_depth_pass_instrument +#define GL_SGIX_depth_pass_instrument 1 +#endif + +#ifndef GL_3DFX_texture_compression_FXT1 +#define GL_3DFX_texture_compression_FXT1 1 +#endif + +#ifndef GL_3DFX_multisample +#define GL_3DFX_multisample 1 +#endif + +#ifndef GL_3DFX_tbuffer +#define GL_3DFX_tbuffer 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTbufferMask3DFX (GLuint); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLTBUFFERMASK3DFXPROC) (GLuint mask); +#endif + +#ifndef GL_EXT_multisample +#define GL_EXT_multisample 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glSampleMaskEXT (GLclampf, GLboolean); +GLAPI void APIENTRY glSamplePatternEXT (GLenum); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLSAMPLEMASKEXTPROC) (GLclampf value, GLboolean invert); +typedef void (APIENTRYP PFNGLSAMPLEPATTERNEXTPROC) (GLenum pattern); +#endif + +#ifndef GL_SGIX_vertex_preclip +#define GL_SGIX_vertex_preclip 1 +#endif + +#ifndef GL_SGIX_convolution_accuracy +#define GL_SGIX_convolution_accuracy 1 +#endif + +#ifndef GL_SGIX_resample +#define GL_SGIX_resample 1 +#endif + +#ifndef GL_SGIS_point_line_texgen +#define GL_SGIS_point_line_texgen 1 +#endif + +#ifndef GL_SGIS_texture_color_mask +#define GL_SGIS_texture_color_mask 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTextureColorMaskSGIS (GLboolean, GLboolean, GLboolean, GLboolean); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLTEXTURECOLORMASKSGISPROC) (GLboolean red, GLboolean green, GLboolean blue, GLboolean alpha); +#endif + +#ifndef GL_SGIX_igloo_interface +#define GL_SGIX_igloo_interface 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glIglooInterfaceSGIX (GLenum, const GLvoid *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLIGLOOINTERFACESGIXPROC) (GLenum pname, const GLvoid *params); +#endif + +#ifndef GL_EXT_texture_env_dot3 +#define GL_EXT_texture_env_dot3 1 +#endif + +#ifndef GL_ATI_texture_mirror_once +#define GL_ATI_texture_mirror_once 1 +#endif + +#ifndef GL_NV_fence +#define GL_NV_fence 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDeleteFencesNV (GLsizei, const GLuint *); +GLAPI void APIENTRY glGenFencesNV (GLsizei, GLuint *); +GLAPI GLboolean APIENTRY glIsFenceNV (GLuint); +GLAPI GLboolean APIENTRY glTestFenceNV (GLuint); +GLAPI void APIENTRY glGetFenceivNV (GLuint, GLenum, GLint *); +GLAPI void APIENTRY glFinishFenceNV (GLuint); +GLAPI void APIENTRY glSetFenceNV (GLuint, GLenum); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLDELETEFENCESNVPROC) (GLsizei n, const GLuint *fences); +typedef void (APIENTRYP PFNGLGENFENCESNVPROC) (GLsizei n, GLuint *fences); +typedef GLboolean (APIENTRYP PFNGLISFENCENVPROC) (GLuint fence); +typedef GLboolean (APIENTRYP PFNGLTESTFENCENVPROC) (GLuint fence); +typedef void (APIENTRYP PFNGLGETFENCEIVNVPROC) (GLuint fence, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLFINISHFENCENVPROC) (GLuint fence); +typedef void (APIENTRYP PFNGLSETFENCENVPROC) (GLuint fence, GLenum condition); +#endif + +#ifndef GL_NV_evaluators +#define GL_NV_evaluators 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glMapControlPointsNV (GLenum, GLuint, GLenum, GLsizei, GLsizei, GLint, GLint, GLboolean, const GLvoid *); +GLAPI void APIENTRY glMapParameterivNV (GLenum, GLenum, const GLint *); +GLAPI void APIENTRY glMapParameterfvNV (GLenum, GLenum, const GLfloat *); +GLAPI void APIENTRY glGetMapControlPointsNV (GLenum, GLuint, GLenum, GLsizei, GLsizei, GLboolean, GLvoid *); +GLAPI void APIENTRY glGetMapParameterivNV (GLenum, GLenum, GLint *); +GLAPI void APIENTRY glGetMapParameterfvNV (GLenum, GLenum, GLfloat *); +GLAPI void APIENTRY glGetMapAttribParameterivNV (GLenum, GLuint, GLenum, GLint *); +GLAPI void APIENTRY glGetMapAttribParameterfvNV (GLenum, GLuint, GLenum, GLfloat *); +GLAPI void APIENTRY glEvalMapsNV (GLenum, GLenum); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLMAPCONTROLPOINTSNVPROC) (GLenum target, GLuint index, GLenum type, GLsizei ustride, GLsizei vstride, GLint uorder, GLint vorder, GLboolean packed, const GLvoid *points); +typedef void (APIENTRYP PFNGLMAPPARAMETERIVNVPROC) (GLenum target, GLenum pname, const GLint *params); +typedef void (APIENTRYP PFNGLMAPPARAMETERFVNVPROC) (GLenum target, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLGETMAPCONTROLPOINTSNVPROC) (GLenum target, GLuint index, GLenum type, GLsizei ustride, GLsizei vstride, GLboolean packed, GLvoid *points); +typedef void (APIENTRYP PFNGLGETMAPPARAMETERIVNVPROC) (GLenum target, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETMAPPARAMETERFVNVPROC) (GLenum target, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETMAPATTRIBPARAMETERIVNVPROC) (GLenum target, GLuint index, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETMAPATTRIBPARAMETERFVNVPROC) (GLenum target, GLuint index, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLEVALMAPSNVPROC) (GLenum target, GLenum mode); +#endif + +#ifndef GL_NV_packed_depth_stencil +#define GL_NV_packed_depth_stencil 1 +#endif + +#ifndef GL_NV_register_combiners2 +#define GL_NV_register_combiners2 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glCombinerStageParameterfvNV (GLenum, GLenum, const GLfloat *); +GLAPI void APIENTRY glGetCombinerStageParameterfvNV (GLenum, GLenum, GLfloat *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLCOMBINERSTAGEPARAMETERFVNVPROC) (GLenum stage, GLenum pname, const GLfloat *params); +typedef void (APIENTRYP PFNGLGETCOMBINERSTAGEPARAMETERFVNVPROC) (GLenum stage, GLenum pname, GLfloat *params); +#endif + +#ifndef GL_NV_texture_compression_vtc +#define GL_NV_texture_compression_vtc 1 +#endif + +#ifndef GL_NV_texture_rectangle +#define GL_NV_texture_rectangle 1 +#endif + +#ifndef GL_NV_texture_shader +#define GL_NV_texture_shader 1 +#endif + +#ifndef GL_NV_texture_shader2 +#define GL_NV_texture_shader2 1 +#endif + +#ifndef GL_NV_vertex_array_range2 +#define GL_NV_vertex_array_range2 1 +#endif + +#ifndef GL_NV_vertex_program +#define GL_NV_vertex_program 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLboolean APIENTRY glAreProgramsResidentNV (GLsizei, const GLuint *, GLboolean *); +GLAPI void APIENTRY glBindProgramNV (GLenum, GLuint); +GLAPI void APIENTRY glDeleteProgramsNV (GLsizei, const GLuint *); +GLAPI void APIENTRY glExecuteProgramNV (GLenum, GLuint, const GLfloat *); +GLAPI void APIENTRY glGenProgramsNV (GLsizei, GLuint *); +GLAPI void APIENTRY glGetProgramParameterdvNV (GLenum, GLuint, GLenum, GLdouble *); +GLAPI void APIENTRY glGetProgramParameterfvNV (GLenum, GLuint, GLenum, GLfloat *); +GLAPI void APIENTRY glGetProgramivNV (GLuint, GLenum, GLint *); +GLAPI void APIENTRY glGetProgramStringNV (GLuint, GLenum, GLubyte *); +GLAPI void APIENTRY glGetTrackMatrixivNV (GLenum, GLuint, GLenum, GLint *); +GLAPI void APIENTRY glGetVertexAttribdvNV (GLuint, GLenum, GLdouble *); +GLAPI void APIENTRY glGetVertexAttribfvNV (GLuint, GLenum, GLfloat *); +GLAPI void APIENTRY glGetVertexAttribivNV (GLuint, GLenum, GLint *); +GLAPI void APIENTRY glGetVertexAttribPointervNV (GLuint, GLenum, GLvoid* *); +GLAPI GLboolean APIENTRY glIsProgramNV (GLuint); +GLAPI void APIENTRY glLoadProgramNV (GLenum, GLuint, GLsizei, const GLubyte *); +GLAPI void APIENTRY glProgramParameter4dNV (GLenum, GLuint, GLdouble, GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glProgramParameter4dvNV (GLenum, GLuint, const GLdouble *); +GLAPI void APIENTRY glProgramParameter4fNV (GLenum, GLuint, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glProgramParameter4fvNV (GLenum, GLuint, const GLfloat *); +GLAPI void APIENTRY glProgramParameters4dvNV (GLenum, GLuint, GLuint, const GLdouble *); +GLAPI void APIENTRY glProgramParameters4fvNV (GLenum, GLuint, GLuint, const GLfloat *); +GLAPI void APIENTRY glRequestResidentProgramsNV (GLsizei, const GLuint *); +GLAPI void APIENTRY glTrackMatrixNV (GLenum, GLuint, GLenum, GLenum); +GLAPI void APIENTRY glVertexAttribPointerNV (GLuint, GLint, GLenum, GLsizei, const GLvoid *); +GLAPI void APIENTRY glVertexAttrib1dNV (GLuint, GLdouble); +GLAPI void APIENTRY glVertexAttrib1dvNV (GLuint, const GLdouble *); +GLAPI void APIENTRY glVertexAttrib1fNV (GLuint, GLfloat); +GLAPI void APIENTRY glVertexAttrib1fvNV (GLuint, const GLfloat *); +GLAPI void APIENTRY glVertexAttrib1sNV (GLuint, GLshort); +GLAPI void APIENTRY glVertexAttrib1svNV (GLuint, const GLshort *); +GLAPI void APIENTRY glVertexAttrib2dNV (GLuint, GLdouble, GLdouble); +GLAPI void APIENTRY glVertexAttrib2dvNV (GLuint, const GLdouble *); +GLAPI void APIENTRY glVertexAttrib2fNV (GLuint, GLfloat, GLfloat); +GLAPI void APIENTRY glVertexAttrib2fvNV (GLuint, const GLfloat *); +GLAPI void APIENTRY glVertexAttrib2sNV (GLuint, GLshort, GLshort); +GLAPI void APIENTRY glVertexAttrib2svNV (GLuint, const GLshort *); +GLAPI void APIENTRY glVertexAttrib3dNV (GLuint, GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glVertexAttrib3dvNV (GLuint, const GLdouble *); +GLAPI void APIENTRY glVertexAttrib3fNV (GLuint, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glVertexAttrib3fvNV (GLuint, const GLfloat *); +GLAPI void APIENTRY glVertexAttrib3sNV (GLuint, GLshort, GLshort, GLshort); +GLAPI void APIENTRY glVertexAttrib3svNV (GLuint, const GLshort *); +GLAPI void APIENTRY glVertexAttrib4dNV (GLuint, GLdouble, GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glVertexAttrib4dvNV (GLuint, const GLdouble *); +GLAPI void APIENTRY glVertexAttrib4fNV (GLuint, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glVertexAttrib4fvNV (GLuint, const GLfloat *); +GLAPI void APIENTRY glVertexAttrib4sNV (GLuint, GLshort, GLshort, GLshort, GLshort); +GLAPI void APIENTRY glVertexAttrib4svNV (GLuint, const GLshort *); +GLAPI void APIENTRY glVertexAttrib4ubNV (GLuint, GLubyte, GLubyte, GLubyte, GLubyte); +GLAPI void APIENTRY glVertexAttrib4ubvNV (GLuint, const GLubyte *); +GLAPI void APIENTRY glVertexAttribs1dvNV (GLuint, GLsizei, const GLdouble *); +GLAPI void APIENTRY glVertexAttribs1fvNV (GLuint, GLsizei, const GLfloat *); +GLAPI void APIENTRY glVertexAttribs1svNV (GLuint, GLsizei, const GLshort *); +GLAPI void APIENTRY glVertexAttribs2dvNV (GLuint, GLsizei, const GLdouble *); +GLAPI void APIENTRY glVertexAttribs2fvNV (GLuint, GLsizei, const GLfloat *); +GLAPI void APIENTRY glVertexAttribs2svNV (GLuint, GLsizei, const GLshort *); +GLAPI void APIENTRY glVertexAttribs3dvNV (GLuint, GLsizei, const GLdouble *); +GLAPI void APIENTRY glVertexAttribs3fvNV (GLuint, GLsizei, const GLfloat *); +GLAPI void APIENTRY glVertexAttribs3svNV (GLuint, GLsizei, const GLshort *); +GLAPI void APIENTRY glVertexAttribs4dvNV (GLuint, GLsizei, const GLdouble *); +GLAPI void APIENTRY glVertexAttribs4fvNV (GLuint, GLsizei, const GLfloat *); +GLAPI void APIENTRY glVertexAttribs4svNV (GLuint, GLsizei, const GLshort *); +GLAPI void APIENTRY glVertexAttribs4ubvNV (GLuint, GLsizei, const GLubyte *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef GLboolean (APIENTRYP PFNGLAREPROGRAMSRESIDENTNVPROC) (GLsizei n, const GLuint *programs, GLboolean *residences); +typedef void (APIENTRYP PFNGLBINDPROGRAMNVPROC) (GLenum target, GLuint id); +typedef void (APIENTRYP PFNGLDELETEPROGRAMSNVPROC) (GLsizei n, const GLuint *programs); +typedef void (APIENTRYP PFNGLEXECUTEPROGRAMNVPROC) (GLenum target, GLuint id, const GLfloat *params); +typedef void (APIENTRYP PFNGLGENPROGRAMSNVPROC) (GLsizei n, GLuint *programs); +typedef void (APIENTRYP PFNGLGETPROGRAMPARAMETERDVNVPROC) (GLenum target, GLuint index, GLenum pname, GLdouble *params); +typedef void (APIENTRYP PFNGLGETPROGRAMPARAMETERFVNVPROC) (GLenum target, GLuint index, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETPROGRAMIVNVPROC) (GLuint id, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETPROGRAMSTRINGNVPROC) (GLuint id, GLenum pname, GLubyte *program); +typedef void (APIENTRYP PFNGLGETTRACKMATRIXIVNVPROC) (GLenum target, GLuint address, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBDVNVPROC) (GLuint index, GLenum pname, GLdouble *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBFVNVPROC) (GLuint index, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBIVNVPROC) (GLuint index, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBPOINTERVNVPROC) (GLuint index, GLenum pname, GLvoid* *pointer); +typedef GLboolean (APIENTRYP PFNGLISPROGRAMNVPROC) (GLuint id); +typedef void (APIENTRYP PFNGLLOADPROGRAMNVPROC) (GLenum target, GLuint id, GLsizei len, const GLubyte *program); +typedef void (APIENTRYP PFNGLPROGRAMPARAMETER4DNVPROC) (GLenum target, GLuint index, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +typedef void (APIENTRYP PFNGLPROGRAMPARAMETER4DVNVPROC) (GLenum target, GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLPROGRAMPARAMETER4FNVPROC) (GLenum target, GLuint index, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +typedef void (APIENTRYP PFNGLPROGRAMPARAMETER4FVNVPROC) (GLenum target, GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLPROGRAMPARAMETERS4DVNVPROC) (GLenum target, GLuint index, GLuint count, const GLdouble *v); +typedef void (APIENTRYP PFNGLPROGRAMPARAMETERS4FVNVPROC) (GLenum target, GLuint index, GLuint count, const GLfloat *v); +typedef void (APIENTRYP PFNGLREQUESTRESIDENTPROGRAMSNVPROC) (GLsizei n, const GLuint *programs); +typedef void (APIENTRYP PFNGLTRACKMATRIXNVPROC) (GLenum target, GLuint address, GLenum matrix, GLenum transform); +typedef void (APIENTRYP PFNGLVERTEXATTRIBPOINTERNVPROC) (GLuint index, GLint fsize, GLenum type, GLsizei stride, const GLvoid *pointer); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1DNVPROC) (GLuint index, GLdouble x); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1DVNVPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1FNVPROC) (GLuint index, GLfloat x); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1FVNVPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1SNVPROC) (GLuint index, GLshort x); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1SVNVPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2DNVPROC) (GLuint index, GLdouble x, GLdouble y); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2DVNVPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2FNVPROC) (GLuint index, GLfloat x, GLfloat y); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2FVNVPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2SNVPROC) (GLuint index, GLshort x, GLshort y); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2SVNVPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3DNVPROC) (GLuint index, GLdouble x, GLdouble y, GLdouble z); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3DVNVPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3FNVPROC) (GLuint index, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3FVNVPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3SNVPROC) (GLuint index, GLshort x, GLshort y, GLshort z); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3SVNVPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4DNVPROC) (GLuint index, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4DVNVPROC) (GLuint index, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4FNVPROC) (GLuint index, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4FVNVPROC) (GLuint index, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4SNVPROC) (GLuint index, GLshort x, GLshort y, GLshort z, GLshort w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4SVNVPROC) (GLuint index, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4UBNVPROC) (GLuint index, GLubyte x, GLubyte y, GLubyte z, GLubyte w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4UBVNVPROC) (GLuint index, const GLubyte *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS1DVNVPROC) (GLuint index, GLsizei count, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS1FVNVPROC) (GLuint index, GLsizei count, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS1SVNVPROC) (GLuint index, GLsizei count, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS2DVNVPROC) (GLuint index, GLsizei count, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS2FVNVPROC) (GLuint index, GLsizei count, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS2SVNVPROC) (GLuint index, GLsizei count, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS3DVNVPROC) (GLuint index, GLsizei count, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS3FVNVPROC) (GLuint index, GLsizei count, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS3SVNVPROC) (GLuint index, GLsizei count, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS4DVNVPROC) (GLuint index, GLsizei count, const GLdouble *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS4FVNVPROC) (GLuint index, GLsizei count, const GLfloat *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS4SVNVPROC) (GLuint index, GLsizei count, const GLshort *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS4UBVNVPROC) (GLuint index, GLsizei count, const GLubyte *v); +#endif + +#ifndef GL_SGIX_texture_coordinate_clamp +#define GL_SGIX_texture_coordinate_clamp 1 +#endif + +#ifndef GL_SGIX_scalebias_hint +#define GL_SGIX_scalebias_hint 1 +#endif + +#ifndef GL_OML_interlace +#define GL_OML_interlace 1 +#endif + +#ifndef GL_OML_subsample +#define GL_OML_subsample 1 +#endif + +#ifndef GL_OML_resample +#define GL_OML_resample 1 +#endif + +#ifndef GL_NV_copy_depth_to_color +#define GL_NV_copy_depth_to_color 1 +#endif + +#ifndef GL_ATI_envmap_bumpmap +#define GL_ATI_envmap_bumpmap 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glTexBumpParameterivATI (GLenum, const GLint *); +GLAPI void APIENTRY glTexBumpParameterfvATI (GLenum, const GLfloat *); +GLAPI void APIENTRY glGetTexBumpParameterivATI (GLenum, GLint *); +GLAPI void APIENTRY glGetTexBumpParameterfvATI (GLenum, GLfloat *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLTEXBUMPPARAMETERIVATIPROC) (GLenum pname, const GLint *param); +typedef void (APIENTRYP PFNGLTEXBUMPPARAMETERFVATIPROC) (GLenum pname, const GLfloat *param); +typedef void (APIENTRYP PFNGLGETTEXBUMPPARAMETERIVATIPROC) (GLenum pname, GLint *param); +typedef void (APIENTRYP PFNGLGETTEXBUMPPARAMETERFVATIPROC) (GLenum pname, GLfloat *param); +#endif + +#ifndef GL_ATI_fragment_shader +#define GL_ATI_fragment_shader 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLuint APIENTRY glGenFragmentShadersATI (GLuint); +GLAPI void APIENTRY glBindFragmentShaderATI (GLuint); +GLAPI void APIENTRY glDeleteFragmentShaderATI (GLuint); +GLAPI void APIENTRY glBeginFragmentShaderATI (void); +GLAPI void APIENTRY glEndFragmentShaderATI (void); +GLAPI void APIENTRY glPassTexCoordATI (GLuint, GLuint, GLenum); +GLAPI void APIENTRY glSampleMapATI (GLuint, GLuint, GLenum); +GLAPI void APIENTRY glColorFragmentOp1ATI (GLenum, GLuint, GLuint, GLuint, GLuint, GLuint, GLuint); +GLAPI void APIENTRY glColorFragmentOp2ATI (GLenum, GLuint, GLuint, GLuint, GLuint, GLuint, GLuint, GLuint, GLuint, GLuint); +GLAPI void APIENTRY glColorFragmentOp3ATI (GLenum, GLuint, GLuint, GLuint, GLuint, GLuint, GLuint, GLuint, GLuint, GLuint, GLuint, GLuint, GLuint); +GLAPI void APIENTRY glAlphaFragmentOp1ATI (GLenum, GLuint, GLuint, GLuint, GLuint, GLuint); +GLAPI void APIENTRY glAlphaFragmentOp2ATI (GLenum, GLuint, GLuint, GLuint, GLuint, GLuint, GLuint, GLuint, GLuint); +GLAPI void APIENTRY glAlphaFragmentOp3ATI (GLenum, GLuint, GLuint, GLuint, GLuint, GLuint, GLuint, GLuint, GLuint, GLuint, GLuint, GLuint); +GLAPI void APIENTRY glSetFragmentShaderConstantATI (GLuint, const GLfloat *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef GLuint (APIENTRYP PFNGLGENFRAGMENTSHADERSATIPROC) (GLuint range); +typedef void (APIENTRYP PFNGLBINDFRAGMENTSHADERATIPROC) (GLuint id); +typedef void (APIENTRYP PFNGLDELETEFRAGMENTSHADERATIPROC) (GLuint id); +typedef void (APIENTRYP PFNGLBEGINFRAGMENTSHADERATIPROC) (void); +typedef void (APIENTRYP PFNGLENDFRAGMENTSHADERATIPROC) (void); +typedef void (APIENTRYP PFNGLPASSTEXCOORDATIPROC) (GLuint dst, GLuint coord, GLenum swizzle); +typedef void (APIENTRYP PFNGLSAMPLEMAPATIPROC) (GLuint dst, GLuint interp, GLenum swizzle); +typedef void (APIENTRYP PFNGLCOLORFRAGMENTOP1ATIPROC) (GLenum op, GLuint dst, GLuint dstMask, GLuint dstMod, GLuint arg1, GLuint arg1Rep, GLuint arg1Mod); +typedef void (APIENTRYP PFNGLCOLORFRAGMENTOP2ATIPROC) (GLenum op, GLuint dst, GLuint dstMask, GLuint dstMod, GLuint arg1, GLuint arg1Rep, GLuint arg1Mod, GLuint arg2, GLuint arg2Rep, GLuint arg2Mod); +typedef void (APIENTRYP PFNGLCOLORFRAGMENTOP3ATIPROC) (GLenum op, GLuint dst, GLuint dstMask, GLuint dstMod, GLuint arg1, GLuint arg1Rep, GLuint arg1Mod, GLuint arg2, GLuint arg2Rep, GLuint arg2Mod, GLuint arg3, GLuint arg3Rep, GLuint arg3Mod); +typedef void (APIENTRYP PFNGLALPHAFRAGMENTOP1ATIPROC) (GLenum op, GLuint dst, GLuint dstMod, GLuint arg1, GLuint arg1Rep, GLuint arg1Mod); +typedef void (APIENTRYP PFNGLALPHAFRAGMENTOP2ATIPROC) (GLenum op, GLuint dst, GLuint dstMod, GLuint arg1, GLuint arg1Rep, GLuint arg1Mod, GLuint arg2, GLuint arg2Rep, GLuint arg2Mod); +typedef void (APIENTRYP PFNGLALPHAFRAGMENTOP3ATIPROC) (GLenum op, GLuint dst, GLuint dstMod, GLuint arg1, GLuint arg1Rep, GLuint arg1Mod, GLuint arg2, GLuint arg2Rep, GLuint arg2Mod, GLuint arg3, GLuint arg3Rep, GLuint arg3Mod); +typedef void (APIENTRYP PFNGLSETFRAGMENTSHADERCONSTANTATIPROC) (GLuint dst, const GLfloat *value); +#endif + +#ifndef GL_ATI_pn_triangles +#define GL_ATI_pn_triangles 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPNTrianglesiATI (GLenum, GLint); +GLAPI void APIENTRY glPNTrianglesfATI (GLenum, GLfloat); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLPNTRIANGLESIATIPROC) (GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLPNTRIANGLESFATIPROC) (GLenum pname, GLfloat param); +#endif + +#ifndef GL_ATI_vertex_array_object +#define GL_ATI_vertex_array_object 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLuint APIENTRY glNewObjectBufferATI (GLsizei, const GLvoid *, GLenum); +GLAPI GLboolean APIENTRY glIsObjectBufferATI (GLuint); +GLAPI void APIENTRY glUpdateObjectBufferATI (GLuint, GLuint, GLsizei, const GLvoid *, GLenum); +GLAPI void APIENTRY glGetObjectBufferfvATI (GLuint, GLenum, GLfloat *); +GLAPI void APIENTRY glGetObjectBufferivATI (GLuint, GLenum, GLint *); +GLAPI void APIENTRY glFreeObjectBufferATI (GLuint); +GLAPI void APIENTRY glArrayObjectATI (GLenum, GLint, GLenum, GLsizei, GLuint, GLuint); +GLAPI void APIENTRY glGetArrayObjectfvATI (GLenum, GLenum, GLfloat *); +GLAPI void APIENTRY glGetArrayObjectivATI (GLenum, GLenum, GLint *); +GLAPI void APIENTRY glVariantArrayObjectATI (GLuint, GLenum, GLsizei, GLuint, GLuint); +GLAPI void APIENTRY glGetVariantArrayObjectfvATI (GLuint, GLenum, GLfloat *); +GLAPI void APIENTRY glGetVariantArrayObjectivATI (GLuint, GLenum, GLint *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef GLuint (APIENTRYP PFNGLNEWOBJECTBUFFERATIPROC) (GLsizei size, const GLvoid *pointer, GLenum usage); +typedef GLboolean (APIENTRYP PFNGLISOBJECTBUFFERATIPROC) (GLuint buffer); +typedef void (APIENTRYP PFNGLUPDATEOBJECTBUFFERATIPROC) (GLuint buffer, GLuint offset, GLsizei size, const GLvoid *pointer, GLenum preserve); +typedef void (APIENTRYP PFNGLGETOBJECTBUFFERFVATIPROC) (GLuint buffer, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETOBJECTBUFFERIVATIPROC) (GLuint buffer, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLFREEOBJECTBUFFERATIPROC) (GLuint buffer); +typedef void (APIENTRYP PFNGLARRAYOBJECTATIPROC) (GLenum array, GLint size, GLenum type, GLsizei stride, GLuint buffer, GLuint offset); +typedef void (APIENTRYP PFNGLGETARRAYOBJECTFVATIPROC) (GLenum array, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETARRAYOBJECTIVATIPROC) (GLenum array, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLVARIANTARRAYOBJECTATIPROC) (GLuint id, GLenum type, GLsizei stride, GLuint buffer, GLuint offset); +typedef void (APIENTRYP PFNGLGETVARIANTARRAYOBJECTFVATIPROC) (GLuint id, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETVARIANTARRAYOBJECTIVATIPROC) (GLuint id, GLenum pname, GLint *params); +#endif + +#ifndef GL_EXT_vertex_shader +#define GL_EXT_vertex_shader 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBeginVertexShaderEXT (void); +GLAPI void APIENTRY glEndVertexShaderEXT (void); +GLAPI void APIENTRY glBindVertexShaderEXT (GLuint); +GLAPI GLuint APIENTRY glGenVertexShadersEXT (GLuint); +GLAPI void APIENTRY glDeleteVertexShaderEXT (GLuint); +GLAPI void APIENTRY glShaderOp1EXT (GLenum, GLuint, GLuint); +GLAPI void APIENTRY glShaderOp2EXT (GLenum, GLuint, GLuint, GLuint); +GLAPI void APIENTRY glShaderOp3EXT (GLenum, GLuint, GLuint, GLuint, GLuint); +GLAPI void APIENTRY glSwizzleEXT (GLuint, GLuint, GLenum, GLenum, GLenum, GLenum); +GLAPI void APIENTRY glWriteMaskEXT (GLuint, GLuint, GLenum, GLenum, GLenum, GLenum); +GLAPI void APIENTRY glInsertComponentEXT (GLuint, GLuint, GLuint); +GLAPI void APIENTRY glExtractComponentEXT (GLuint, GLuint, GLuint); +GLAPI GLuint APIENTRY glGenSymbolsEXT (GLenum, GLenum, GLenum, GLuint); +GLAPI void APIENTRY glSetInvariantEXT (GLuint, GLenum, const GLvoid *); +GLAPI void APIENTRY glSetLocalConstantEXT (GLuint, GLenum, const GLvoid *); +GLAPI void APIENTRY glVariantbvEXT (GLuint, const GLbyte *); +GLAPI void APIENTRY glVariantsvEXT (GLuint, const GLshort *); +GLAPI void APIENTRY glVariantivEXT (GLuint, const GLint *); +GLAPI void APIENTRY glVariantfvEXT (GLuint, const GLfloat *); +GLAPI void APIENTRY glVariantdvEXT (GLuint, const GLdouble *); +GLAPI void APIENTRY glVariantubvEXT (GLuint, const GLubyte *); +GLAPI void APIENTRY glVariantusvEXT (GLuint, const GLushort *); +GLAPI void APIENTRY glVariantuivEXT (GLuint, const GLuint *); +GLAPI void APIENTRY glVariantPointerEXT (GLuint, GLenum, GLuint, const GLvoid *); +GLAPI void APIENTRY glEnableVariantClientStateEXT (GLuint); +GLAPI void APIENTRY glDisableVariantClientStateEXT (GLuint); +GLAPI GLuint APIENTRY glBindLightParameterEXT (GLenum, GLenum); +GLAPI GLuint APIENTRY glBindMaterialParameterEXT (GLenum, GLenum); +GLAPI GLuint APIENTRY glBindTexGenParameterEXT (GLenum, GLenum, GLenum); +GLAPI GLuint APIENTRY glBindTextureUnitParameterEXT (GLenum, GLenum); +GLAPI GLuint APIENTRY glBindParameterEXT (GLenum); +GLAPI GLboolean APIENTRY glIsVariantEnabledEXT (GLuint, GLenum); +GLAPI void APIENTRY glGetVariantBooleanvEXT (GLuint, GLenum, GLboolean *); +GLAPI void APIENTRY glGetVariantIntegervEXT (GLuint, GLenum, GLint *); +GLAPI void APIENTRY glGetVariantFloatvEXT (GLuint, GLenum, GLfloat *); +GLAPI void APIENTRY glGetVariantPointervEXT (GLuint, GLenum, GLvoid* *); +GLAPI void APIENTRY glGetInvariantBooleanvEXT (GLuint, GLenum, GLboolean *); +GLAPI void APIENTRY glGetInvariantIntegervEXT (GLuint, GLenum, GLint *); +GLAPI void APIENTRY glGetInvariantFloatvEXT (GLuint, GLenum, GLfloat *); +GLAPI void APIENTRY glGetLocalConstantBooleanvEXT (GLuint, GLenum, GLboolean *); +GLAPI void APIENTRY glGetLocalConstantIntegervEXT (GLuint, GLenum, GLint *); +GLAPI void APIENTRY glGetLocalConstantFloatvEXT (GLuint, GLenum, GLfloat *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLBEGINVERTEXSHADEREXTPROC) (void); +typedef void (APIENTRYP PFNGLENDVERTEXSHADEREXTPROC) (void); +typedef void (APIENTRYP PFNGLBINDVERTEXSHADEREXTPROC) (GLuint id); +typedef GLuint (APIENTRYP PFNGLGENVERTEXSHADERSEXTPROC) (GLuint range); +typedef void (APIENTRYP PFNGLDELETEVERTEXSHADEREXTPROC) (GLuint id); +typedef void (APIENTRYP PFNGLSHADEROP1EXTPROC) (GLenum op, GLuint res, GLuint arg1); +typedef void (APIENTRYP PFNGLSHADEROP2EXTPROC) (GLenum op, GLuint res, GLuint arg1, GLuint arg2); +typedef void (APIENTRYP PFNGLSHADEROP3EXTPROC) (GLenum op, GLuint res, GLuint arg1, GLuint arg2, GLuint arg3); +typedef void (APIENTRYP PFNGLSWIZZLEEXTPROC) (GLuint res, GLuint in, GLenum outX, GLenum outY, GLenum outZ, GLenum outW); +typedef void (APIENTRYP PFNGLWRITEMASKEXTPROC) (GLuint res, GLuint in, GLenum outX, GLenum outY, GLenum outZ, GLenum outW); +typedef void (APIENTRYP PFNGLINSERTCOMPONENTEXTPROC) (GLuint res, GLuint src, GLuint num); +typedef void (APIENTRYP PFNGLEXTRACTCOMPONENTEXTPROC) (GLuint res, GLuint src, GLuint num); +typedef GLuint (APIENTRYP PFNGLGENSYMBOLSEXTPROC) (GLenum datatype, GLenum storagetype, GLenum range, GLuint components); +typedef void (APIENTRYP PFNGLSETINVARIANTEXTPROC) (GLuint id, GLenum type, const GLvoid *addr); +typedef void (APIENTRYP PFNGLSETLOCALCONSTANTEXTPROC) (GLuint id, GLenum type, const GLvoid *addr); +typedef void (APIENTRYP PFNGLVARIANTBVEXTPROC) (GLuint id, const GLbyte *addr); +typedef void (APIENTRYP PFNGLVARIANTSVEXTPROC) (GLuint id, const GLshort *addr); +typedef void (APIENTRYP PFNGLVARIANTIVEXTPROC) (GLuint id, const GLint *addr); +typedef void (APIENTRYP PFNGLVARIANTFVEXTPROC) (GLuint id, const GLfloat *addr); +typedef void (APIENTRYP PFNGLVARIANTDVEXTPROC) (GLuint id, const GLdouble *addr); +typedef void (APIENTRYP PFNGLVARIANTUBVEXTPROC) (GLuint id, const GLubyte *addr); +typedef void (APIENTRYP PFNGLVARIANTUSVEXTPROC) (GLuint id, const GLushort *addr); +typedef void (APIENTRYP PFNGLVARIANTUIVEXTPROC) (GLuint id, const GLuint *addr); +typedef void (APIENTRYP PFNGLVARIANTPOINTEREXTPROC) (GLuint id, GLenum type, GLuint stride, const GLvoid *addr); +typedef void (APIENTRYP PFNGLENABLEVARIANTCLIENTSTATEEXTPROC) (GLuint id); +typedef void (APIENTRYP PFNGLDISABLEVARIANTCLIENTSTATEEXTPROC) (GLuint id); +typedef GLuint (APIENTRYP PFNGLBINDLIGHTPARAMETEREXTPROC) (GLenum light, GLenum value); +typedef GLuint (APIENTRYP PFNGLBINDMATERIALPARAMETEREXTPROC) (GLenum face, GLenum value); +typedef GLuint (APIENTRYP PFNGLBINDTEXGENPARAMETEREXTPROC) (GLenum unit, GLenum coord, GLenum value); +typedef GLuint (APIENTRYP PFNGLBINDTEXTUREUNITPARAMETEREXTPROC) (GLenum unit, GLenum value); +typedef GLuint (APIENTRYP PFNGLBINDPARAMETEREXTPROC) (GLenum value); +typedef GLboolean (APIENTRYP PFNGLISVARIANTENABLEDEXTPROC) (GLuint id, GLenum cap); +typedef void (APIENTRYP PFNGLGETVARIANTBOOLEANVEXTPROC) (GLuint id, GLenum value, GLboolean *data); +typedef void (APIENTRYP PFNGLGETVARIANTINTEGERVEXTPROC) (GLuint id, GLenum value, GLint *data); +typedef void (APIENTRYP PFNGLGETVARIANTFLOATVEXTPROC) (GLuint id, GLenum value, GLfloat *data); +typedef void (APIENTRYP PFNGLGETVARIANTPOINTERVEXTPROC) (GLuint id, GLenum value, GLvoid* *data); +typedef void (APIENTRYP PFNGLGETINVARIANTBOOLEANVEXTPROC) (GLuint id, GLenum value, GLboolean *data); +typedef void (APIENTRYP PFNGLGETINVARIANTINTEGERVEXTPROC) (GLuint id, GLenum value, GLint *data); +typedef void (APIENTRYP PFNGLGETINVARIANTFLOATVEXTPROC) (GLuint id, GLenum value, GLfloat *data); +typedef void (APIENTRYP PFNGLGETLOCALCONSTANTBOOLEANVEXTPROC) (GLuint id, GLenum value, GLboolean *data); +typedef void (APIENTRYP PFNGLGETLOCALCONSTANTINTEGERVEXTPROC) (GLuint id, GLenum value, GLint *data); +typedef void (APIENTRYP PFNGLGETLOCALCONSTANTFLOATVEXTPROC) (GLuint id, GLenum value, GLfloat *data); +#endif + +#ifndef GL_ATI_vertex_streams +#define GL_ATI_vertex_streams 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glVertexStream1sATI (GLenum, GLshort); +GLAPI void APIENTRY glVertexStream1svATI (GLenum, const GLshort *); +GLAPI void APIENTRY glVertexStream1iATI (GLenum, GLint); +GLAPI void APIENTRY glVertexStream1ivATI (GLenum, const GLint *); +GLAPI void APIENTRY glVertexStream1fATI (GLenum, GLfloat); +GLAPI void APIENTRY glVertexStream1fvATI (GLenum, const GLfloat *); +GLAPI void APIENTRY glVertexStream1dATI (GLenum, GLdouble); +GLAPI void APIENTRY glVertexStream1dvATI (GLenum, const GLdouble *); +GLAPI void APIENTRY glVertexStream2sATI (GLenum, GLshort, GLshort); +GLAPI void APIENTRY glVertexStream2svATI (GLenum, const GLshort *); +GLAPI void APIENTRY glVertexStream2iATI (GLenum, GLint, GLint); +GLAPI void APIENTRY glVertexStream2ivATI (GLenum, const GLint *); +GLAPI void APIENTRY glVertexStream2fATI (GLenum, GLfloat, GLfloat); +GLAPI void APIENTRY glVertexStream2fvATI (GLenum, const GLfloat *); +GLAPI void APIENTRY glVertexStream2dATI (GLenum, GLdouble, GLdouble); +GLAPI void APIENTRY glVertexStream2dvATI (GLenum, const GLdouble *); +GLAPI void APIENTRY glVertexStream3sATI (GLenum, GLshort, GLshort, GLshort); +GLAPI void APIENTRY glVertexStream3svATI (GLenum, const GLshort *); +GLAPI void APIENTRY glVertexStream3iATI (GLenum, GLint, GLint, GLint); +GLAPI void APIENTRY glVertexStream3ivATI (GLenum, const GLint *); +GLAPI void APIENTRY glVertexStream3fATI (GLenum, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glVertexStream3fvATI (GLenum, const GLfloat *); +GLAPI void APIENTRY glVertexStream3dATI (GLenum, GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glVertexStream3dvATI (GLenum, const GLdouble *); +GLAPI void APIENTRY glVertexStream4sATI (GLenum, GLshort, GLshort, GLshort, GLshort); +GLAPI void APIENTRY glVertexStream4svATI (GLenum, const GLshort *); +GLAPI void APIENTRY glVertexStream4iATI (GLenum, GLint, GLint, GLint, GLint); +GLAPI void APIENTRY glVertexStream4ivATI (GLenum, const GLint *); +GLAPI void APIENTRY glVertexStream4fATI (GLenum, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glVertexStream4fvATI (GLenum, const GLfloat *); +GLAPI void APIENTRY glVertexStream4dATI (GLenum, GLdouble, GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glVertexStream4dvATI (GLenum, const GLdouble *); +GLAPI void APIENTRY glNormalStream3bATI (GLenum, GLbyte, GLbyte, GLbyte); +GLAPI void APIENTRY glNormalStream3bvATI (GLenum, const GLbyte *); +GLAPI void APIENTRY glNormalStream3sATI (GLenum, GLshort, GLshort, GLshort); +GLAPI void APIENTRY glNormalStream3svATI (GLenum, const GLshort *); +GLAPI void APIENTRY glNormalStream3iATI (GLenum, GLint, GLint, GLint); +GLAPI void APIENTRY glNormalStream3ivATI (GLenum, const GLint *); +GLAPI void APIENTRY glNormalStream3fATI (GLenum, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glNormalStream3fvATI (GLenum, const GLfloat *); +GLAPI void APIENTRY glNormalStream3dATI (GLenum, GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glNormalStream3dvATI (GLenum, const GLdouble *); +GLAPI void APIENTRY glClientActiveVertexStreamATI (GLenum); +GLAPI void APIENTRY glVertexBlendEnviATI (GLenum, GLint); +GLAPI void APIENTRY glVertexBlendEnvfATI (GLenum, GLfloat); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLVERTEXSTREAM1SATIPROC) (GLenum stream, GLshort x); +typedef void (APIENTRYP PFNGLVERTEXSTREAM1SVATIPROC) (GLenum stream, const GLshort *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM1IATIPROC) (GLenum stream, GLint x); +typedef void (APIENTRYP PFNGLVERTEXSTREAM1IVATIPROC) (GLenum stream, const GLint *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM1FATIPROC) (GLenum stream, GLfloat x); +typedef void (APIENTRYP PFNGLVERTEXSTREAM1FVATIPROC) (GLenum stream, const GLfloat *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM1DATIPROC) (GLenum stream, GLdouble x); +typedef void (APIENTRYP PFNGLVERTEXSTREAM1DVATIPROC) (GLenum stream, const GLdouble *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM2SATIPROC) (GLenum stream, GLshort x, GLshort y); +typedef void (APIENTRYP PFNGLVERTEXSTREAM2SVATIPROC) (GLenum stream, const GLshort *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM2IATIPROC) (GLenum stream, GLint x, GLint y); +typedef void (APIENTRYP PFNGLVERTEXSTREAM2IVATIPROC) (GLenum stream, const GLint *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM2FATIPROC) (GLenum stream, GLfloat x, GLfloat y); +typedef void (APIENTRYP PFNGLVERTEXSTREAM2FVATIPROC) (GLenum stream, const GLfloat *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM2DATIPROC) (GLenum stream, GLdouble x, GLdouble y); +typedef void (APIENTRYP PFNGLVERTEXSTREAM2DVATIPROC) (GLenum stream, const GLdouble *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM3SATIPROC) (GLenum stream, GLshort x, GLshort y, GLshort z); +typedef void (APIENTRYP PFNGLVERTEXSTREAM3SVATIPROC) (GLenum stream, const GLshort *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM3IATIPROC) (GLenum stream, GLint x, GLint y, GLint z); +typedef void (APIENTRYP PFNGLVERTEXSTREAM3IVATIPROC) (GLenum stream, const GLint *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM3FATIPROC) (GLenum stream, GLfloat x, GLfloat y, GLfloat z); +typedef void (APIENTRYP PFNGLVERTEXSTREAM3FVATIPROC) (GLenum stream, const GLfloat *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM3DATIPROC) (GLenum stream, GLdouble x, GLdouble y, GLdouble z); +typedef void (APIENTRYP PFNGLVERTEXSTREAM3DVATIPROC) (GLenum stream, const GLdouble *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM4SATIPROC) (GLenum stream, GLshort x, GLshort y, GLshort z, GLshort w); +typedef void (APIENTRYP PFNGLVERTEXSTREAM4SVATIPROC) (GLenum stream, const GLshort *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM4IATIPROC) (GLenum stream, GLint x, GLint y, GLint z, GLint w); +typedef void (APIENTRYP PFNGLVERTEXSTREAM4IVATIPROC) (GLenum stream, const GLint *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM4FATIPROC) (GLenum stream, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +typedef void (APIENTRYP PFNGLVERTEXSTREAM4FVATIPROC) (GLenum stream, const GLfloat *coords); +typedef void (APIENTRYP PFNGLVERTEXSTREAM4DATIPROC) (GLenum stream, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +typedef void (APIENTRYP PFNGLVERTEXSTREAM4DVATIPROC) (GLenum stream, const GLdouble *coords); +typedef void (APIENTRYP PFNGLNORMALSTREAM3BATIPROC) (GLenum stream, GLbyte nx, GLbyte ny, GLbyte nz); +typedef void (APIENTRYP PFNGLNORMALSTREAM3BVATIPROC) (GLenum stream, const GLbyte *coords); +typedef void (APIENTRYP PFNGLNORMALSTREAM3SATIPROC) (GLenum stream, GLshort nx, GLshort ny, GLshort nz); +typedef void (APIENTRYP PFNGLNORMALSTREAM3SVATIPROC) (GLenum stream, const GLshort *coords); +typedef void (APIENTRYP PFNGLNORMALSTREAM3IATIPROC) (GLenum stream, GLint nx, GLint ny, GLint nz); +typedef void (APIENTRYP PFNGLNORMALSTREAM3IVATIPROC) (GLenum stream, const GLint *coords); +typedef void (APIENTRYP PFNGLNORMALSTREAM3FATIPROC) (GLenum stream, GLfloat nx, GLfloat ny, GLfloat nz); +typedef void (APIENTRYP PFNGLNORMALSTREAM3FVATIPROC) (GLenum stream, const GLfloat *coords); +typedef void (APIENTRYP PFNGLNORMALSTREAM3DATIPROC) (GLenum stream, GLdouble nx, GLdouble ny, GLdouble nz); +typedef void (APIENTRYP PFNGLNORMALSTREAM3DVATIPROC) (GLenum stream, const GLdouble *coords); +typedef void (APIENTRYP PFNGLCLIENTACTIVEVERTEXSTREAMATIPROC) (GLenum stream); +typedef void (APIENTRYP PFNGLVERTEXBLENDENVIATIPROC) (GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLVERTEXBLENDENVFATIPROC) (GLenum pname, GLfloat param); +#endif + +#ifndef GL_ATI_element_array +#define GL_ATI_element_array 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glElementPointerATI (GLenum, const GLvoid *); +GLAPI void APIENTRY glDrawElementArrayATI (GLenum, GLsizei); +GLAPI void APIENTRY glDrawRangeElementArrayATI (GLenum, GLuint, GLuint, GLsizei); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLELEMENTPOINTERATIPROC) (GLenum type, const GLvoid *pointer); +typedef void (APIENTRYP PFNGLDRAWELEMENTARRAYATIPROC) (GLenum mode, GLsizei count); +typedef void (APIENTRYP PFNGLDRAWRANGEELEMENTARRAYATIPROC) (GLenum mode, GLuint start, GLuint end, GLsizei count); +#endif + +#ifndef GL_SUN_mesh_array +#define GL_SUN_mesh_array 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDrawMeshArraysSUN (GLenum, GLint, GLsizei, GLsizei); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLDRAWMESHARRAYSSUNPROC) (GLenum mode, GLint first, GLsizei count, GLsizei width); +#endif + +#ifndef GL_SUN_slice_accum +#define GL_SUN_slice_accum 1 +#endif + +#ifndef GL_NV_multisample_filter_hint +#define GL_NV_multisample_filter_hint 1 +#endif + +#ifndef GL_NV_depth_clamp +#define GL_NV_depth_clamp 1 +#endif + +#ifndef GL_NV_occlusion_query +#define GL_NV_occlusion_query 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGenOcclusionQueriesNV (GLsizei, GLuint *); +GLAPI void APIENTRY glDeleteOcclusionQueriesNV (GLsizei, const GLuint *); +GLAPI GLboolean APIENTRY glIsOcclusionQueryNV (GLuint); +GLAPI void APIENTRY glBeginOcclusionQueryNV (GLuint); +GLAPI void APIENTRY glEndOcclusionQueryNV (void); +GLAPI void APIENTRY glGetOcclusionQueryivNV (GLuint, GLenum, GLint *); +GLAPI void APIENTRY glGetOcclusionQueryuivNV (GLuint, GLenum, GLuint *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLGENOCCLUSIONQUERIESNVPROC) (GLsizei n, GLuint *ids); +typedef void (APIENTRYP PFNGLDELETEOCCLUSIONQUERIESNVPROC) (GLsizei n, const GLuint *ids); +typedef GLboolean (APIENTRYP PFNGLISOCCLUSIONQUERYNVPROC) (GLuint id); +typedef void (APIENTRYP PFNGLBEGINOCCLUSIONQUERYNVPROC) (GLuint id); +typedef void (APIENTRYP PFNGLENDOCCLUSIONQUERYNVPROC) (void); +typedef void (APIENTRYP PFNGLGETOCCLUSIONQUERYIVNVPROC) (GLuint id, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGETOCCLUSIONQUERYUIVNVPROC) (GLuint id, GLenum pname, GLuint *params); +#endif + +#ifndef GL_NV_point_sprite +#define GL_NV_point_sprite 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPointParameteriNV (GLenum, GLint); +GLAPI void APIENTRY glPointParameterivNV (GLenum, const GLint *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLPOINTPARAMETERINVPROC) (GLenum pname, GLint param); +typedef void (APIENTRYP PFNGLPOINTPARAMETERIVNVPROC) (GLenum pname, const GLint *params); +#endif + +#ifndef GL_NV_texture_shader3 +#define GL_NV_texture_shader3 1 +#endif + +#ifndef GL_NV_vertex_program1_1 +#define GL_NV_vertex_program1_1 1 +#endif + +#ifndef GL_EXT_shadow_funcs +#define GL_EXT_shadow_funcs 1 +#endif + +#ifndef GL_EXT_stencil_two_side +#define GL_EXT_stencil_two_side 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glActiveStencilFaceEXT (GLenum); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLACTIVESTENCILFACEEXTPROC) (GLenum face); +#endif + +#ifndef GL_ATI_text_fragment_shader +#define GL_ATI_text_fragment_shader 1 +#endif + +#ifndef GL_APPLE_client_storage +#define GL_APPLE_client_storage 1 +#endif + +#ifndef GL_APPLE_element_array +#define GL_APPLE_element_array 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glElementPointerAPPLE (GLenum, const GLvoid *); +GLAPI void APIENTRY glDrawElementArrayAPPLE (GLenum, GLint, GLsizei); +GLAPI void APIENTRY glDrawRangeElementArrayAPPLE (GLenum, GLuint, GLuint, GLint, GLsizei); +GLAPI void APIENTRY glMultiDrawElementArrayAPPLE (GLenum, const GLint *, const GLsizei *, GLsizei); +GLAPI void APIENTRY glMultiDrawRangeElementArrayAPPLE (GLenum, GLuint, GLuint, const GLint *, const GLsizei *, GLsizei); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLELEMENTPOINTERAPPLEPROC) (GLenum type, const GLvoid *pointer); +typedef void (APIENTRYP PFNGLDRAWELEMENTARRAYAPPLEPROC) (GLenum mode, GLint first, GLsizei count); +typedef void (APIENTRYP PFNGLDRAWRANGEELEMENTARRAYAPPLEPROC) (GLenum mode, GLuint start, GLuint end, GLint first, GLsizei count); +typedef void (APIENTRYP PFNGLMULTIDRAWELEMENTARRAYAPPLEPROC) (GLenum mode, const GLint *first, const GLsizei *count, GLsizei primcount); +typedef void (APIENTRYP PFNGLMULTIDRAWRANGEELEMENTARRAYAPPLEPROC) (GLenum mode, GLuint start, GLuint end, const GLint *first, const GLsizei *count, GLsizei primcount); +#endif + +#ifndef GL_APPLE_fence +#define GL_APPLE_fence 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glGenFencesAPPLE (GLsizei, GLuint *); +GLAPI void APIENTRY glDeleteFencesAPPLE (GLsizei, const GLuint *); +GLAPI void APIENTRY glSetFenceAPPLE (GLuint); +GLAPI GLboolean APIENTRY glIsFenceAPPLE (GLuint); +GLAPI GLboolean APIENTRY glTestFenceAPPLE (GLuint); +GLAPI void APIENTRY glFinishFenceAPPLE (GLuint); +GLAPI GLboolean APIENTRY glTestObjectAPPLE (GLenum, GLuint); +GLAPI void APIENTRY glFinishObjectAPPLE (GLenum, GLint); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLGENFENCESAPPLEPROC) (GLsizei n, GLuint *fences); +typedef void (APIENTRYP PFNGLDELETEFENCESAPPLEPROC) (GLsizei n, const GLuint *fences); +typedef void (APIENTRYP PFNGLSETFENCEAPPLEPROC) (GLuint fence); +typedef GLboolean (APIENTRYP PFNGLISFENCEAPPLEPROC) (GLuint fence); +typedef GLboolean (APIENTRYP PFNGLTESTFENCEAPPLEPROC) (GLuint fence); +typedef void (APIENTRYP PFNGLFINISHFENCEAPPLEPROC) (GLuint fence); +typedef GLboolean (APIENTRYP PFNGLTESTOBJECTAPPLEPROC) (GLenum object, GLuint name); +typedef void (APIENTRYP PFNGLFINISHOBJECTAPPLEPROC) (GLenum object, GLint name); +#endif + +#ifndef GL_APPLE_vertex_array_object +#define GL_APPLE_vertex_array_object 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBindVertexArrayAPPLE (GLuint); +GLAPI void APIENTRY glDeleteVertexArraysAPPLE (GLsizei, const GLuint *); +GLAPI void APIENTRY glGenVertexArraysAPPLE (GLsizei, const GLuint *); +GLAPI GLboolean APIENTRY glIsVertexArrayAPPLE (GLuint); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLBINDVERTEXARRAYAPPLEPROC) (GLuint array); +typedef void (APIENTRYP PFNGLDELETEVERTEXARRAYSAPPLEPROC) (GLsizei n, const GLuint *arrays); +typedef void (APIENTRYP PFNGLGENVERTEXARRAYSAPPLEPROC) (GLsizei n, const GLuint *arrays); +typedef GLboolean (APIENTRYP PFNGLISVERTEXARRAYAPPLEPROC) (GLuint array); +#endif + +#ifndef GL_APPLE_vertex_array_range +#define GL_APPLE_vertex_array_range 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glVertexArrayRangeAPPLE (GLsizei, GLvoid *); +GLAPI void APIENTRY glFlushVertexArrayRangeAPPLE (GLsizei, GLvoid *); +GLAPI void APIENTRY glVertexArrayParameteriAPPLE (GLenum, GLint); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLVERTEXARRAYRANGEAPPLEPROC) (GLsizei length, GLvoid *pointer); +typedef void (APIENTRYP PFNGLFLUSHVERTEXARRAYRANGEAPPLEPROC) (GLsizei length, GLvoid *pointer); +typedef void (APIENTRYP PFNGLVERTEXARRAYPARAMETERIAPPLEPROC) (GLenum pname, GLint param); +#endif + +#ifndef GL_APPLE_ycbcr_422 +#define GL_APPLE_ycbcr_422 1 +#endif + +#ifndef GL_S3_s3tc +#define GL_S3_s3tc 1 +#endif + +#ifndef GL_ATI_draw_buffers +#define GL_ATI_draw_buffers 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDrawBuffersATI (GLsizei, const GLenum *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLDRAWBUFFERSATIPROC) (GLsizei n, const GLenum *bufs); +#endif + +#ifndef GL_ATI_pixel_format_float +#define GL_ATI_pixel_format_float 1 +/* This is really a WGL extension, but defines some associated GL enums. + * ATI does not export "GL_ATI_pixel_format_float" in the GL_EXTENSIONS string. + */ +#endif + +#ifndef GL_ATI_texture_env_combine3 +#define GL_ATI_texture_env_combine3 1 +#endif + +#ifndef GL_ATI_texture_float +#define GL_ATI_texture_float 1 +#endif + +#ifndef GL_NV_float_buffer +#define GL_NV_float_buffer 1 +#endif + +#ifndef GL_NV_fragment_program +#define GL_NV_fragment_program 1 +/* Some NV_fragment_program entry points are shared with ARB_vertex_program. */ +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glProgramNamedParameter4fNV (GLuint, GLsizei, const GLubyte *, GLfloat, GLfloat, GLfloat, GLfloat); +GLAPI void APIENTRY glProgramNamedParameter4dNV (GLuint, GLsizei, const GLubyte *, GLdouble, GLdouble, GLdouble, GLdouble); +GLAPI void APIENTRY glProgramNamedParameter4fvNV (GLuint, GLsizei, const GLubyte *, const GLfloat *); +GLAPI void APIENTRY glProgramNamedParameter4dvNV (GLuint, GLsizei, const GLubyte *, const GLdouble *); +GLAPI void APIENTRY glGetProgramNamedParameterfvNV (GLuint, GLsizei, const GLubyte *, GLfloat *); +GLAPI void APIENTRY glGetProgramNamedParameterdvNV (GLuint, GLsizei, const GLubyte *, GLdouble *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLPROGRAMNAMEDPARAMETER4FNVPROC) (GLuint id, GLsizei len, const GLubyte *name, GLfloat x, GLfloat y, GLfloat z, GLfloat w); +typedef void (APIENTRYP PFNGLPROGRAMNAMEDPARAMETER4DNVPROC) (GLuint id, GLsizei len, const GLubyte *name, GLdouble x, GLdouble y, GLdouble z, GLdouble w); +typedef void (APIENTRYP PFNGLPROGRAMNAMEDPARAMETER4FVNVPROC) (GLuint id, GLsizei len, const GLubyte *name, const GLfloat *v); +typedef void (APIENTRYP PFNGLPROGRAMNAMEDPARAMETER4DVNVPROC) (GLuint id, GLsizei len, const GLubyte *name, const GLdouble *v); +typedef void (APIENTRYP PFNGLGETPROGRAMNAMEDPARAMETERFVNVPROC) (GLuint id, GLsizei len, const GLubyte *name, GLfloat *params); +typedef void (APIENTRYP PFNGLGETPROGRAMNAMEDPARAMETERDVNVPROC) (GLuint id, GLsizei len, const GLubyte *name, GLdouble *params); +#endif + +#ifndef GL_NV_half_float +#define GL_NV_half_float 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glVertex2hNV (GLhalfNV, GLhalfNV); +GLAPI void APIENTRY glVertex2hvNV (const GLhalfNV *); +GLAPI void APIENTRY glVertex3hNV (GLhalfNV, GLhalfNV, GLhalfNV); +GLAPI void APIENTRY glVertex3hvNV (const GLhalfNV *); +GLAPI void APIENTRY glVertex4hNV (GLhalfNV, GLhalfNV, GLhalfNV, GLhalfNV); +GLAPI void APIENTRY glVertex4hvNV (const GLhalfNV *); +GLAPI void APIENTRY glNormal3hNV (GLhalfNV, GLhalfNV, GLhalfNV); +GLAPI void APIENTRY glNormal3hvNV (const GLhalfNV *); +GLAPI void APIENTRY glColor3hNV (GLhalfNV, GLhalfNV, GLhalfNV); +GLAPI void APIENTRY glColor3hvNV (const GLhalfNV *); +GLAPI void APIENTRY glColor4hNV (GLhalfNV, GLhalfNV, GLhalfNV, GLhalfNV); +GLAPI void APIENTRY glColor4hvNV (const GLhalfNV *); +GLAPI void APIENTRY glTexCoord1hNV (GLhalfNV); +GLAPI void APIENTRY glTexCoord1hvNV (const GLhalfNV *); +GLAPI void APIENTRY glTexCoord2hNV (GLhalfNV, GLhalfNV); +GLAPI void APIENTRY glTexCoord2hvNV (const GLhalfNV *); +GLAPI void APIENTRY glTexCoord3hNV (GLhalfNV, GLhalfNV, GLhalfNV); +GLAPI void APIENTRY glTexCoord3hvNV (const GLhalfNV *); +GLAPI void APIENTRY glTexCoord4hNV (GLhalfNV, GLhalfNV, GLhalfNV, GLhalfNV); +GLAPI void APIENTRY glTexCoord4hvNV (const GLhalfNV *); +GLAPI void APIENTRY glMultiTexCoord1hNV (GLenum, GLhalfNV); +GLAPI void APIENTRY glMultiTexCoord1hvNV (GLenum, const GLhalfNV *); +GLAPI void APIENTRY glMultiTexCoord2hNV (GLenum, GLhalfNV, GLhalfNV); +GLAPI void APIENTRY glMultiTexCoord2hvNV (GLenum, const GLhalfNV *); +GLAPI void APIENTRY glMultiTexCoord3hNV (GLenum, GLhalfNV, GLhalfNV, GLhalfNV); +GLAPI void APIENTRY glMultiTexCoord3hvNV (GLenum, const GLhalfNV *); +GLAPI void APIENTRY glMultiTexCoord4hNV (GLenum, GLhalfNV, GLhalfNV, GLhalfNV, GLhalfNV); +GLAPI void APIENTRY glMultiTexCoord4hvNV (GLenum, const GLhalfNV *); +GLAPI void APIENTRY glFogCoordhNV (GLhalfNV); +GLAPI void APIENTRY glFogCoordhvNV (const GLhalfNV *); +GLAPI void APIENTRY glSecondaryColor3hNV (GLhalfNV, GLhalfNV, GLhalfNV); +GLAPI void APIENTRY glSecondaryColor3hvNV (const GLhalfNV *); +GLAPI void APIENTRY glVertexWeighthNV (GLhalfNV); +GLAPI void APIENTRY glVertexWeighthvNV (const GLhalfNV *); +GLAPI void APIENTRY glVertexAttrib1hNV (GLuint, GLhalfNV); +GLAPI void APIENTRY glVertexAttrib1hvNV (GLuint, const GLhalfNV *); +GLAPI void APIENTRY glVertexAttrib2hNV (GLuint, GLhalfNV, GLhalfNV); +GLAPI void APIENTRY glVertexAttrib2hvNV (GLuint, const GLhalfNV *); +GLAPI void APIENTRY glVertexAttrib3hNV (GLuint, GLhalfNV, GLhalfNV, GLhalfNV); +GLAPI void APIENTRY glVertexAttrib3hvNV (GLuint, const GLhalfNV *); +GLAPI void APIENTRY glVertexAttrib4hNV (GLuint, GLhalfNV, GLhalfNV, GLhalfNV, GLhalfNV); +GLAPI void APIENTRY glVertexAttrib4hvNV (GLuint, const GLhalfNV *); +GLAPI void APIENTRY glVertexAttribs1hvNV (GLuint, GLsizei, const GLhalfNV *); +GLAPI void APIENTRY glVertexAttribs2hvNV (GLuint, GLsizei, const GLhalfNV *); +GLAPI void APIENTRY glVertexAttribs3hvNV (GLuint, GLsizei, const GLhalfNV *); +GLAPI void APIENTRY glVertexAttribs4hvNV (GLuint, GLsizei, const GLhalfNV *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLVERTEX2HNVPROC) (GLhalfNV x, GLhalfNV y); +typedef void (APIENTRYP PFNGLVERTEX2HVNVPROC) (const GLhalfNV *v); +typedef void (APIENTRYP PFNGLVERTEX3HNVPROC) (GLhalfNV x, GLhalfNV y, GLhalfNV z); +typedef void (APIENTRYP PFNGLVERTEX3HVNVPROC) (const GLhalfNV *v); +typedef void (APIENTRYP PFNGLVERTEX4HNVPROC) (GLhalfNV x, GLhalfNV y, GLhalfNV z, GLhalfNV w); +typedef void (APIENTRYP PFNGLVERTEX4HVNVPROC) (const GLhalfNV *v); +typedef void (APIENTRYP PFNGLNORMAL3HNVPROC) (GLhalfNV nx, GLhalfNV ny, GLhalfNV nz); +typedef void (APIENTRYP PFNGLNORMAL3HVNVPROC) (const GLhalfNV *v); +typedef void (APIENTRYP PFNGLCOLOR3HNVPROC) (GLhalfNV red, GLhalfNV green, GLhalfNV blue); +typedef void (APIENTRYP PFNGLCOLOR3HVNVPROC) (const GLhalfNV *v); +typedef void (APIENTRYP PFNGLCOLOR4HNVPROC) (GLhalfNV red, GLhalfNV green, GLhalfNV blue, GLhalfNV alpha); +typedef void (APIENTRYP PFNGLCOLOR4HVNVPROC) (const GLhalfNV *v); +typedef void (APIENTRYP PFNGLTEXCOORD1HNVPROC) (GLhalfNV s); +typedef void (APIENTRYP PFNGLTEXCOORD1HVNVPROC) (const GLhalfNV *v); +typedef void (APIENTRYP PFNGLTEXCOORD2HNVPROC) (GLhalfNV s, GLhalfNV t); +typedef void (APIENTRYP PFNGLTEXCOORD2HVNVPROC) (const GLhalfNV *v); +typedef void (APIENTRYP PFNGLTEXCOORD3HNVPROC) (GLhalfNV s, GLhalfNV t, GLhalfNV r); +typedef void (APIENTRYP PFNGLTEXCOORD3HVNVPROC) (const GLhalfNV *v); +typedef void (APIENTRYP PFNGLTEXCOORD4HNVPROC) (GLhalfNV s, GLhalfNV t, GLhalfNV r, GLhalfNV q); +typedef void (APIENTRYP PFNGLTEXCOORD4HVNVPROC) (const GLhalfNV *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1HNVPROC) (GLenum target, GLhalfNV s); +typedef void (APIENTRYP PFNGLMULTITEXCOORD1HVNVPROC) (GLenum target, const GLhalfNV *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2HNVPROC) (GLenum target, GLhalfNV s, GLhalfNV t); +typedef void (APIENTRYP PFNGLMULTITEXCOORD2HVNVPROC) (GLenum target, const GLhalfNV *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3HNVPROC) (GLenum target, GLhalfNV s, GLhalfNV t, GLhalfNV r); +typedef void (APIENTRYP PFNGLMULTITEXCOORD3HVNVPROC) (GLenum target, const GLhalfNV *v); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4HNVPROC) (GLenum target, GLhalfNV s, GLhalfNV t, GLhalfNV r, GLhalfNV q); +typedef void (APIENTRYP PFNGLMULTITEXCOORD4HVNVPROC) (GLenum target, const GLhalfNV *v); +typedef void (APIENTRYP PFNGLFOGCOORDHNVPROC) (GLhalfNV fog); +typedef void (APIENTRYP PFNGLFOGCOORDHVNVPROC) (const GLhalfNV *fog); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3HNVPROC) (GLhalfNV red, GLhalfNV green, GLhalfNV blue); +typedef void (APIENTRYP PFNGLSECONDARYCOLOR3HVNVPROC) (const GLhalfNV *v); +typedef void (APIENTRYP PFNGLVERTEXWEIGHTHNVPROC) (GLhalfNV weight); +typedef void (APIENTRYP PFNGLVERTEXWEIGHTHVNVPROC) (const GLhalfNV *weight); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1HNVPROC) (GLuint index, GLhalfNV x); +typedef void (APIENTRYP PFNGLVERTEXATTRIB1HVNVPROC) (GLuint index, const GLhalfNV *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2HNVPROC) (GLuint index, GLhalfNV x, GLhalfNV y); +typedef void (APIENTRYP PFNGLVERTEXATTRIB2HVNVPROC) (GLuint index, const GLhalfNV *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3HNVPROC) (GLuint index, GLhalfNV x, GLhalfNV y, GLhalfNV z); +typedef void (APIENTRYP PFNGLVERTEXATTRIB3HVNVPROC) (GLuint index, const GLhalfNV *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4HNVPROC) (GLuint index, GLhalfNV x, GLhalfNV y, GLhalfNV z, GLhalfNV w); +typedef void (APIENTRYP PFNGLVERTEXATTRIB4HVNVPROC) (GLuint index, const GLhalfNV *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS1HVNVPROC) (GLuint index, GLsizei n, const GLhalfNV *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS2HVNVPROC) (GLuint index, GLsizei n, const GLhalfNV *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS3HVNVPROC) (GLuint index, GLsizei n, const GLhalfNV *v); +typedef void (APIENTRYP PFNGLVERTEXATTRIBS4HVNVPROC) (GLuint index, GLsizei n, const GLhalfNV *v); +#endif + +#ifndef GL_NV_pixel_data_range +#define GL_NV_pixel_data_range 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPixelDataRangeNV (GLenum, GLsizei, GLvoid *); +GLAPI void APIENTRY glFlushPixelDataRangeNV (GLenum); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLPIXELDATARANGENVPROC) (GLenum target, GLsizei length, GLvoid *pointer); +typedef void (APIENTRYP PFNGLFLUSHPIXELDATARANGENVPROC) (GLenum target); +#endif + +#ifndef GL_NV_primitive_restart +#define GL_NV_primitive_restart 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glPrimitiveRestartNV (void); +GLAPI void APIENTRY glPrimitiveRestartIndexNV (GLuint); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLPRIMITIVERESTARTNVPROC) (void); +typedef void (APIENTRYP PFNGLPRIMITIVERESTARTINDEXNVPROC) (GLuint index); +#endif + +#ifndef GL_NV_texture_expand_normal +#define GL_NV_texture_expand_normal 1 +#endif + +#ifndef GL_NV_vertex_program2 +#define GL_NV_vertex_program2 1 +#endif + +#ifndef GL_ATI_map_object_buffer +#define GL_ATI_map_object_buffer 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLvoid* APIENTRY glMapObjectBufferATI (GLuint); +GLAPI void APIENTRY glUnmapObjectBufferATI (GLuint); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef GLvoid* (APIENTRYP PFNGLMAPOBJECTBUFFERATIPROC) (GLuint buffer); +typedef void (APIENTRYP PFNGLUNMAPOBJECTBUFFERATIPROC) (GLuint buffer); +#endif + +#ifndef GL_ATI_separate_stencil +#define GL_ATI_separate_stencil 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glStencilOpSeparateATI (GLenum, GLenum, GLenum, GLenum); +GLAPI void APIENTRY glStencilFuncSeparateATI (GLenum, GLenum, GLint, GLuint); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLSTENCILOPSEPARATEATIPROC) (GLenum face, GLenum sfail, GLenum dpfail, GLenum dppass); +typedef void (APIENTRYP PFNGLSTENCILFUNCSEPARATEATIPROC) (GLenum frontfunc, GLenum backfunc, GLint ref, GLuint mask); +#endif + +#ifndef GL_ATI_vertex_attrib_array_object +#define GL_ATI_vertex_attrib_array_object 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glVertexAttribArrayObjectATI (GLuint, GLint, GLenum, GLboolean, GLsizei, GLuint, GLuint); +GLAPI void APIENTRY glGetVertexAttribArrayObjectfvATI (GLuint, GLenum, GLfloat *); +GLAPI void APIENTRY glGetVertexAttribArrayObjectivATI (GLuint, GLenum, GLint *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLVERTEXATTRIBARRAYOBJECTATIPROC) (GLuint index, GLint size, GLenum type, GLboolean normalized, GLsizei stride, GLuint buffer, GLuint offset); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBARRAYOBJECTFVATIPROC) (GLuint index, GLenum pname, GLfloat *params); +typedef void (APIENTRYP PFNGLGETVERTEXATTRIBARRAYOBJECTIVATIPROC) (GLuint index, GLenum pname, GLint *params); +#endif + +#ifndef GL_OES_read_format +#define GL_OES_read_format 1 +#endif + +#ifndef GL_EXT_depth_bounds_test +#define GL_EXT_depth_bounds_test 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glDepthBoundsEXT (GLclampd, GLclampd); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLDEPTHBOUNDSEXTPROC) (GLclampd zmin, GLclampd zmax); +#endif + +#ifndef GL_EXT_texture_mirror_clamp +#define GL_EXT_texture_mirror_clamp 1 +#endif + +#ifndef GL_EXT_blend_equation_separate +#define GL_EXT_blend_equation_separate 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glBlendEquationSeparateEXT (GLenum, GLenum); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLBLENDEQUATIONSEPARATEEXTPROC) (GLenum modeRGB, GLenum modeAlpha); +#endif + +#ifndef GL_MESA_pack_invert +#define GL_MESA_pack_invert 1 +#endif + +#ifndef GL_MESA_ycbcr_texture +#define GL_MESA_ycbcr_texture 1 +#endif + +#ifndef GL_EXT_pixel_buffer_object +#define GL_EXT_pixel_buffer_object 1 +#endif + +#ifndef GL_NV_fragment_program_option +#define GL_NV_fragment_program_option 1 +#endif + +#ifndef GL_NV_fragment_program2 +#define GL_NV_fragment_program2 1 +#endif + +#ifndef GL_NV_vertex_program2_option +#define GL_NV_vertex_program2_option 1 +#endif + +#ifndef GL_NV_vertex_program3 +#define GL_NV_vertex_program3 1 +#endif + +#ifndef GL_EXT_framebuffer_object +#define GL_EXT_framebuffer_object 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI GLboolean APIENTRY glIsRenderbufferEXT (GLuint); +GLAPI void APIENTRY glBindRenderbufferEXT (GLenum, GLuint); +GLAPI void APIENTRY glDeleteRenderbuffersEXT (GLsizei, const GLuint *); +GLAPI void APIENTRY glGenRenderbuffersEXT (GLsizei, GLuint *); +GLAPI void APIENTRY glRenderbufferStorageEXT (GLenum, GLenum, GLsizei, GLsizei); +GLAPI void APIENTRY glGetRenderbufferParameterivEXT (GLenum, GLenum, GLint *); +GLAPI GLboolean APIENTRY glIsFramebufferEXT (GLuint); +GLAPI void APIENTRY glBindFramebufferEXT (GLenum, GLuint); +GLAPI void APIENTRY glDeleteFramebuffersEXT (GLsizei, const GLuint *); +GLAPI void APIENTRY glGenFramebuffersEXT (GLsizei, GLuint *); +GLAPI GLenum APIENTRY glCheckFramebufferStatusEXT (GLenum); +GLAPI void APIENTRY glFramebufferTexture1DEXT (GLenum, GLenum, GLenum, GLuint, GLint); +GLAPI void APIENTRY glFramebufferTexture2DEXT (GLenum, GLenum, GLenum, GLuint, GLint); +GLAPI void APIENTRY glFramebufferTexture3DEXT (GLenum, GLenum, GLenum, GLuint, GLint, GLint); +GLAPI void APIENTRY glFramebufferRenderbufferEXT (GLenum, GLenum, GLenum, GLuint); +GLAPI void APIENTRY glGetFramebufferAttachmentParameterivEXT (GLenum, GLenum, GLenum, GLint *); +GLAPI void APIENTRY glGenerateMipmapEXT (GLenum); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef GLboolean (APIENTRYP PFNGLISRENDERBUFFEREXTPROC) (GLuint renderbuffer); +typedef void (APIENTRYP PFNGLBINDRENDERBUFFEREXTPROC) (GLenum target, GLuint renderbuffer); +typedef void (APIENTRYP PFNGLDELETERENDERBUFFERSEXTPROC) (GLsizei n, const GLuint *renderbuffers); +typedef void (APIENTRYP PFNGLGENRENDERBUFFERSEXTPROC) (GLsizei n, GLuint *renderbuffers); +typedef void (APIENTRYP PFNGLRENDERBUFFERSTORAGEEXTPROC) (GLenum target, GLenum internalformat, GLsizei width, GLsizei height); +typedef void (APIENTRYP PFNGLGETRENDERBUFFERPARAMETERIVEXTPROC) (GLenum target, GLenum pname, GLint *params); +typedef GLboolean (APIENTRYP PFNGLISFRAMEBUFFEREXTPROC) (GLuint framebuffer); +typedef void (APIENTRYP PFNGLBINDFRAMEBUFFEREXTPROC) (GLenum target, GLuint framebuffer); +typedef void (APIENTRYP PFNGLDELETEFRAMEBUFFERSEXTPROC) (GLsizei n, const GLuint *framebuffers); +typedef void (APIENTRYP PFNGLGENFRAMEBUFFERSEXTPROC) (GLsizei n, GLuint *framebuffers); +typedef GLenum (APIENTRYP PFNGLCHECKFRAMEBUFFERSTATUSEXTPROC) (GLenum target); +typedef void (APIENTRYP PFNGLFRAMEBUFFERTEXTURE1DEXTPROC) (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level); +typedef void (APIENTRYP PFNGLFRAMEBUFFERTEXTURE2DEXTPROC) (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level); +typedef void (APIENTRYP PFNGLFRAMEBUFFERTEXTURE3DEXTPROC) (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, GLint level, GLint zoffset); +typedef void (APIENTRYP PFNGLFRAMEBUFFERRENDERBUFFEREXTPROC) (GLenum target, GLenum attachment, GLenum renderbuffertarget, GLuint renderbuffer); +typedef void (APIENTRYP PFNGLGETFRAMEBUFFERATTACHMENTPARAMETERIVEXTPROC) (GLenum target, GLenum attachment, GLenum pname, GLint *params); +typedef void (APIENTRYP PFNGLGENERATEMIPMAPEXTPROC) (GLenum target); +#endif + +#ifndef GL_GREMEDY_string_marker +#define GL_GREMEDY_string_marker 1 +#ifdef GL_GLEXT_PROTOTYPES +GLAPI void APIENTRY glStringMarkerGREMEDY (GLsizei, const GLvoid *); +#endif /* GL_GLEXT_PROTOTYPES */ +typedef void (APIENTRYP PFNGLSTRINGMARKERGREMEDYPROC) (GLsizei len, const GLvoid *string); +#endif + + +#ifdef __cplusplus +} +#endif + +#endif +#endif /* NO_SDL_GLEXT */ diff --git a/src/win32/dependencies/sdl/SDL_platform.h b/src/win32/dependencies/sdl/SDL_platform.h new file mode 100644 index 00000000..80274bc4 --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_platform.h @@ -0,0 +1,104 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* Try to get a standard set of platform defines */ + +#ifndef _SDL_platform_h +#define _SDL_platform_h + +#if defined(_AIX) +#undef __AIX__ +#define __AIX__ 1 +#endif +#if defined(AMIGA) || defined(__AMIGA) || defined(__amigados__) +#undef __AMIGA__ +#define __AMIGA__ 1 +#endif +#if defined(__BEOS__) +#undef __BEOS__ +#define __BEOS__ 1 +#endif +#if defined(bsdi) || defined(__bsdi) || defined(__bsdi__) +#undef __BSDI__ +#define __BSDI__ 1 +#endif +#if defined(_arch_dreamcast) +#undef __DREAMCAST__ +#define __DREAMCAST__ 1 +#endif +#if defined(__FreeBSD__) || defined(__DragonFly__) +#undef __FREEBSD__ +#define __FREEBSD__ 1 +#endif +#if defined(hpux) || defined(__hpux) || defined(__hpux__) +#undef __HPUX__ +#define __HPUX__ 1 +#endif +#if defined(sgi) || defined(__sgi) || defined(__sgi__) || defined(_SGI_SOURCE) +#undef __IRIX__ +#define __IRIX__ 1 +#endif +#if defined(linux) || defined(__linux) || defined(__linux__) +#undef __LINUX__ +#define __LINUX__ 1 +#endif +#if defined(__APPLE__) +#undef __MACOSX__ +#define __MACOSX__ 1 +#elif defined(macintosh) +#undef __MACOS__ +#define __MACOS__ 1 +#endif +#if defined(__NetBSD__) +#undef __NETBSD__ +#define __NETBSD__ 1 +#endif +#if defined(__OpenBSD__) +#undef __OPENBSD__ +#define __OPENBSD__ 1 +#endif +#if defined(__OS2__) +#undef __OS2__ +#define __OS2__ 1 +#endif +#if defined(osf) || defined(__osf) || defined(__osf__) || defined(_OSF_SOURCE) +#undef __OSF__ +#define __OSF__ 1 +#endif +#if defined(__QNXNTO__) +#undef __QNXNTO__ +#define __QNXNTO__ 1 +#endif +#if defined(riscos) || defined(__riscos) || defined(__riscos__) +#undef __RISCOS__ +#define __RISCOS__ 1 +#endif +#if defined(__SVR4) +#undef __SOLARIS__ +#define __SOLARIS__ 1 +#endif +#if defined(WIN32) || defined(_WIN32) +#undef __WIN32__ +#define __WIN32__ 1 +#endif + +#endif /* _SDL_platform_h */ diff --git a/src/win32/dependencies/sdl/SDL_quit.h b/src/win32/dependencies/sdl/SDL_quit.h new file mode 100644 index 00000000..fcf40fbd --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_quit.h @@ -0,0 +1,50 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* Include file for SDL quit event handling */ + +#ifndef _SDL_quit_h +#define _SDL_quit_h + +#include "SDL_stdinc.h" +#include "SDL_error.h" + +/* + An SDL_QUITEVENT is generated when the user tries to close the application + window. If it is ignored or filtered out, the window will remain open. + If it is not ignored or filtered, it is queued normally and the window + is allowed to close. When the window is closed, screen updates will + complete, but have no effect. + + SDL_Init() installs signal handlers for SIGINT (keyboard interrupt) + and SIGTERM (system termination request), if handlers do not already + exist, that generate SDL_QUITEVENT events as well. There is no way + to determine the cause of an SDL_QUITEVENT, but setting a signal + handler in your application will override the default generation of + quit events for that signal. +*/ + +/* There are no functions directly affecting the quit event */ +#define SDL_QuitRequested() \ + (SDL_PumpEvents(), SDL_PeepEvents(NULL,0,SDL_PEEKEVENT,SDL_QUITMASK)) + +#endif /* _SDL_quit_h */ diff --git a/src/win32/dependencies/sdl/SDL_rwops.h b/src/win32/dependencies/sdl/SDL_rwops.h new file mode 100644 index 00000000..d7e01d8f --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_rwops.h @@ -0,0 +1,139 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* This file provides a general interface for SDL to read and write + data sources. It can easily be extended to files, memory, etc. +*/ + +#ifndef _SDL_rwops_h +#define _SDL_rwops_h + +#include "SDL_stdinc.h" +#include "SDL_error.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* This is the read/write operation structure -- very basic */ + +typedef struct SDL_RWops { + /* Seek to 'offset' relative to whence, one of stdio's whence values: + SEEK_SET, SEEK_CUR, SEEK_END + Returns the final offset in the data source. + */ + int (SDLCALL *seek)(struct SDL_RWops *context, int offset, int whence); + + /* Read up to 'num' objects each of size 'objsize' from the data + source to the area pointed at by 'ptr'. + Returns the number of objects read, or -1 if the read failed. + */ + int (SDLCALL *read)(struct SDL_RWops *context, void *ptr, int size, int maxnum); + + /* Write exactly 'num' objects each of size 'objsize' from the area + pointed at by 'ptr' to data source. + Returns 'num', or -1 if the write failed. + */ + int (SDLCALL *write)(struct SDL_RWops *context, const void *ptr, int size, int num); + + /* Close and free an allocated SDL_FSops structure */ + int (SDLCALL *close)(struct SDL_RWops *context); + + Uint32 type; + union { +#ifdef __WIN32__ + struct { + int append; + void* h; + } win32io; +#endif +#ifdef HAVE_STDIO_H + struct { + int autoclose; + FILE *fp; + } stdio; +#endif + struct { + Uint8 *base; + Uint8 *here; + Uint8 *stop; + } mem; + struct { + void *data1; + } unknown; + } hidden; + +} SDL_RWops; + + +/* Functions to create SDL_RWops structures from various data sources */ + +extern DECLSPEC SDL_RWops * SDLCALL SDL_RWFromFile(const char *file, const char *mode); + +#ifdef HAVE_STDIO_H +extern DECLSPEC SDL_RWops * SDLCALL SDL_RWFromFP(FILE *fp, int autoclose); +#endif + +extern DECLSPEC SDL_RWops * SDLCALL SDL_RWFromMem(void *mem, int size); +extern DECLSPEC SDL_RWops * SDLCALL SDL_RWFromConstMem(const void *mem, int size); + +extern DECLSPEC SDL_RWops * SDLCALL SDL_AllocRW(void); +extern DECLSPEC void SDLCALL SDL_FreeRW(SDL_RWops *area); + +#define RW_SEEK_SET 0 /* Seek from the beginning of data */ +#define RW_SEEK_CUR 1 /* Seek relative to current read point */ +#define RW_SEEK_END 2 /* Seek relative to the end of data */ + +/* Macros to easily read and write from an SDL_RWops structure */ +#define SDL_RWseek(ctx, offset, whence) (ctx)->seek(ctx, offset, whence) +#define SDL_RWtell(ctx) (ctx)->seek(ctx, 0, RW_SEEK_CUR) +#define SDL_RWread(ctx, ptr, size, n) (ctx)->read(ctx, ptr, size, n) +#define SDL_RWwrite(ctx, ptr, size, n) (ctx)->write(ctx, ptr, size, n) +#define SDL_RWclose(ctx) (ctx)->close(ctx) + + +/* Read an item of the specified endianness and return in native format */ +extern DECLSPEC Uint16 SDLCALL SDL_ReadLE16(SDL_RWops *src); +extern DECLSPEC Uint16 SDLCALL SDL_ReadBE16(SDL_RWops *src); +extern DECLSPEC Uint32 SDLCALL SDL_ReadLE32(SDL_RWops *src); +extern DECLSPEC Uint32 SDLCALL SDL_ReadBE32(SDL_RWops *src); +extern DECLSPEC Uint64 SDLCALL SDL_ReadLE64(SDL_RWops *src); +extern DECLSPEC Uint64 SDLCALL SDL_ReadBE64(SDL_RWops *src); + +/* Write an item of native format to the specified endianness */ +extern DECLSPEC int SDLCALL SDL_WriteLE16(SDL_RWops *dst, Uint16 value); +extern DECLSPEC int SDLCALL SDL_WriteBE16(SDL_RWops *dst, Uint16 value); +extern DECLSPEC int SDLCALL SDL_WriteLE32(SDL_RWops *dst, Uint32 value); +extern DECLSPEC int SDLCALL SDL_WriteBE32(SDL_RWops *dst, Uint32 value); +extern DECLSPEC int SDLCALL SDL_WriteLE64(SDL_RWops *dst, Uint64 value); +extern DECLSPEC int SDLCALL SDL_WriteBE64(SDL_RWops *dst, Uint64 value); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* _SDL_rwops_h */ diff --git a/src/win32/dependencies/sdl/SDL_stdinc.h b/src/win32/dependencies/sdl/SDL_stdinc.h new file mode 100644 index 00000000..f790cd56 --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_stdinc.h @@ -0,0 +1,584 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* This is a general header that includes C language support */ + +#ifndef _SDL_stdinc_h +#define _SDL_stdinc_h + +#include "SDL_config.h" + + +#ifdef HAVE_SYS_TYPES_H +#include +#endif +#ifdef HAVE_STDIO_H +#include +#endif +#if defined(STDC_HEADERS) +# include +# include +# include +#else +# if defined(HAVE_STDLIB_H) +# include +# elif defined(HAVE_MALLOC_H) +# include +# endif +# if defined(HAVE_STDDEF_H) +# include +# endif +# if defined(HAVE_STDARG_H) +# include +# endif +#endif +#ifdef HAVE_STRING_H +# if !defined(STDC_HEADERS) && defined(HAVE_MEMORY_H) +# include +# endif +# include +#endif +#ifdef HAVE_STRINGS_H +# include +#endif +#if defined(HAVE_INTTYPES_H) +# include +#elif defined(HAVE_STDINT_H) +# include +#endif +#ifdef HAVE_CTYPE_H +# include +#endif +#ifdef HAVE_ICONV_H +# include +#endif + +/* The number of elements in an array */ +#define SDL_arraysize(array) (sizeof(array)/sizeof(array[0])) +#define SDL_TABLESIZE(table) SDL_arraysize(table) + +/* Basic data types */ +typedef enum SDL_bool { + SDL_FALSE = 0, + SDL_TRUE = 1 +} SDL_bool; + +typedef int8_t Sint8; +typedef uint8_t Uint8; +typedef int16_t Sint16; +typedef uint16_t Uint16; +typedef int32_t Sint32; +typedef uint32_t Uint32; + +#ifdef SDL_HAS_64BIT_TYPE +typedef int64_t Sint64; +typedef uint64_t Uint64; +#else +/* This is really just a hack to prevent the compiler from complaining */ +typedef struct { + Uint32 hi; + Uint32 lo; +} Uint64, Sint64; +#endif + +/* Make sure the types really have the right sizes */ +#define SDL_COMPILE_TIME_ASSERT(name, x) \ + typedef int SDL_dummy_ ## name[(x) * 2 - 1] + +SDL_COMPILE_TIME_ASSERT(uint8, sizeof(Uint8) == 1); +SDL_COMPILE_TIME_ASSERT(sint8, sizeof(Sint8) == 1); +SDL_COMPILE_TIME_ASSERT(uint16, sizeof(Uint16) == 2); +SDL_COMPILE_TIME_ASSERT(sint16, sizeof(Sint16) == 2); +SDL_COMPILE_TIME_ASSERT(uint32, sizeof(Uint32) == 4); +SDL_COMPILE_TIME_ASSERT(sint32, sizeof(Sint32) == 4); +SDL_COMPILE_TIME_ASSERT(uint64, sizeof(Uint64) == 8); +SDL_COMPILE_TIME_ASSERT(sint64, sizeof(Sint64) == 8); + +/* Check to make sure enums are the size of ints, for structure packing. + For both Watcom C/C++ and Borland C/C++ the compiler option that makes + enums having the size of an int must be enabled. + This is "-b" for Borland C/C++ and "-ei" for Watcom C/C++ (v11). +*/ +/* Enable enums always int in CodeWarrior (for MPW use "-enum int") */ +#ifdef __MWERKS__ +#pragma enumsalwaysint on +#endif + +typedef enum { + DUMMY_ENUM_VALUE +} SDL_DUMMY_ENUM; + +SDL_COMPILE_TIME_ASSERT(enum, sizeof(SDL_DUMMY_ENUM) == sizeof(int)); + + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +#ifdef HAVE_MALLOC +#define SDL_malloc malloc +#else +extern DECLSPEC void * SDLCALL SDL_malloc(size_t size); +#endif + +#ifdef HAVE_CALLOC +#define SDL_calloc calloc +#else +extern DECLSPEC void * SDLCALL SDL_calloc(size_t nmemb, size_t size); +#endif + +#ifdef HAVE_REALLOC +#define SDL_realloc realloc +#else +extern DECLSPEC void * SDLCALL SDL_realloc(void *mem, size_t size); +#endif + +#ifdef HAVE_FREE +#define SDL_free free +#else +extern DECLSPEC void SDLCALL SDL_free(void *mem); +#endif + +#if defined(HAVE_ALLOCA) && !defined(alloca) +# if defined(HAVE_ALLOCA_H) +# include +# elif defined(__GNUC__) +# define alloca __builtin_alloca +# elif defined(_MSC_VER) +# include +# define alloca _alloca +# elif defined(__WATCOMC__) +# include +# elif defined(__AIX__) + #pragma alloca +# elif defined(__MRC__) + void *alloca (unsigned); +# else + char *alloca (); +# endif +#endif +#ifdef HAVE_ALLOCA +#define SDL_stack_alloc(type, count) (type*)alloca(sizeof(type)*count) +#define SDL_stack_free(data) +#else +#define SDL_stack_alloc(type, count) (type*)SDL_malloc(sizeof(type)*count) +#define SDL_stack_free(data) SDL_free(data) +#endif + +#ifdef HAVE_GETENV +#define SDL_getenv getenv +#else +extern DECLSPEC char * SDLCALL SDL_getenv(const char *name); +#endif + +#ifdef HAVE_PUTENV +#define SDL_putenv putenv +#else +extern DECLSPEC int SDLCALL SDL_putenv(const char *variable); +#endif + +#ifdef HAVE_QSORT +#define SDL_qsort qsort +#else +extern DECLSPEC void SDLCALL SDL_qsort(void *base, size_t nmemb, size_t size, + int (*compare)(const void *, const void *)); +#endif + +#ifdef HAVE_ABS +#define SDL_abs abs +#else +#define SDL_abs(X) ((X) < 0 ? -(X) : (X)) +#endif + +#define SDL_min(x, y) (((x) < (y)) ? (x) : (y)) +#define SDL_max(x, y) (((x) > (y)) ? (x) : (y)) + +#ifdef HAVE_CTYPE_H +#define SDL_isdigit(X) isdigit(X) +#define SDL_isspace(X) isspace(X) +#define SDL_toupper(X) toupper(X) +#define SDL_tolower(X) tolower(X) +#else +#define SDL_isdigit(X) (((X) >= '0') && ((X) <= '9')) +#define SDL_isspace(X) (((X) == ' ') || ((X) == '\t') || ((X) == '\r') || ((X) == '\n')) +#define SDL_toupper(X) (((X) >= 'a') && ((X) <= 'z') ? ('A'+((X)-'a')) : (X)) +#define SDL_tolower(X) (((X) >= 'A') && ((X) <= 'Z') ? ('a'+((X)-'A')) : (X)) +#endif + +#ifdef HAVE_MEMSET +#define SDL_memset memset +#else +extern DECLSPEC void * SDLCALL SDL_memset(void *dst, int c, size_t len); +#endif + +#if defined(__GNUC__) && defined(i386) +#define SDL_memset4(dst, val, len) \ +do { \ + int u0, u1, u2; \ + __asm__ __volatile__ ( \ + "cld\n\t" \ + "rep ; stosl\n\t" \ + : "=&D" (u0), "=&a" (u1), "=&c" (u2) \ + : "0" (dst), "1" (val), "2" ((Uint32)(len)) \ + : "memory" ); \ +} while(0) +#endif +#ifndef SDL_memset4 +#define SDL_memset4(dst, val, len) \ +do { \ + unsigned _count = (len); \ + unsigned _n = (_count + 3) / 4; \ + Uint32 *_p = (Uint32 *)(dst); \ + Uint32 _val = (val); \ + switch (_count % 4) { \ + case 0: do { *_p++ = _val; \ + case 3: *_p++ = _val; \ + case 2: *_p++ = _val; \ + case 1: *_p++ = _val; \ + } while ( --_n ); \ + } \ +} while(0) +#endif + +#if defined(__GNUC__) && defined(i386) +#define SDL_memcpy(dst, src, len) \ +do { \ + int u0, u1, u2; \ + __asm__ __volatile__ ( \ + "cld\n\t" \ + "rep ; movsl\n\t" \ + "testb $2,%b4\n\t" \ + "je 1f\n\t" \ + "movsw\n" \ + "1:\ttestb $1,%b4\n\t" \ + "je 2f\n\t" \ + "movsb\n" \ + "2:" \ + : "=&c" (u0), "=&D" (u1), "=&S" (u2) \ + : "0" ((unsigned)(len)/4), "q" (len), "1" (dst),"2" (src) \ + : "memory" ); \ +} while(0) +#endif +#ifndef SDL_memcpy +#ifdef HAVE_MEMCPY +#define SDL_memcpy memcpy +#elif defined(HAVE_BCOPY) +#define SDL_memcpy(d, s, n) bcopy((s), (d), (n)) +#else +extern DECLSPEC void * SDLCALL SDL_memcpy(void *dst, const void *src, size_t len); +#endif +#endif + +#if defined(__GNUC__) && defined(i386) +#define SDL_memcpy4(dst, src, len) \ +do { \ + int ecx, edi, esi; \ + __asm__ __volatile__ ( \ + "cld\n\t" \ + "rep ; movsl" \ + : "=&c" (ecx), "=&D" (edi), "=&S" (esi) \ + : "0" ((unsigned)(len)), "1" (dst), "2" (src) \ + : "memory" ); \ +} while(0) +#endif +#ifndef SDL_memcpy4 +#define SDL_memcpy4(dst, src, len) SDL_memcpy(dst, src, (len) << 2) +#endif + +#if defined(__GNUC__) && defined(i386) +#define SDL_revcpy(dst, src, len) \ +do { \ + int u0, u1, u2; \ + char *dstp = (char *)(dst); \ + char *srcp = (char *)(src); \ + int n = (len); \ + if ( n >= 4 ) { \ + __asm__ __volatile__ ( \ + "std\n\t" \ + "rep ; movsl\n\t" \ + : "=&c" (u0), "=&D" (u1), "=&S" (u2) \ + : "0" (n >> 2), \ + "1" (dstp+(n-4)), "2" (srcp+(n-4)) \ + : "memory" ); \ + } \ + switch (n & 3) { \ + case 3: dstp[2] = srcp[2]; \ + case 2: dstp[1] = srcp[1]; \ + case 1: dstp[0] = srcp[0]; \ + break; \ + default: \ + break; \ + } \ +} while(0) +#endif +#ifndef SDL_revcpy +extern DECLSPEC void * SDLCALL SDL_revcpy(void *dst, const void *src, size_t len); +#endif + +#ifdef HAVE_MEMMOVE +#define SDL_memmove memmove +#elif defined(HAVE_BCOPY) +#define SDL_memmove(d, s, n) bcopy((s), (d), (n)) +#else +#define SDL_memmove(dst, src, len) \ +do { \ + if ( dst < src ) { \ + SDL_memcpy(dst, src, len); \ + } else { \ + SDL_revcpy(dst, src, len); \ + } \ +} while(0) +#endif + +#ifdef HAVE_MEMCMP +#define SDL_memcmp memcmp +#else +extern DECLSPEC int SDLCALL SDL_memcmp(const void *s1, const void *s2, size_t len); +#endif + +#ifdef HAVE_STRLEN +#define SDL_strlen strlen +#else +extern DECLSPEC size_t SDLCALL SDL_strlen(const char *string); +#endif + +#ifdef HAVE_STRLCPY +#define SDL_strlcpy strlcpy +#else +extern DECLSPEC size_t SDLCALL SDL_strlcpy(char *dst, const char *src, size_t maxlen); +#endif + +#ifdef HAVE_STRLCAT +#define SDL_strlcat strlcat +#else +extern DECLSPEC size_t SDLCALL SDL_strlcat(char *dst, const char *src, size_t maxlen); +#endif + +#ifdef HAVE_STRDUP +#define SDL_strdup strdup +#else +extern DECLSPEC char * SDLCALL SDL_strdup(const char *string); +#endif + +#ifdef HAVE__STRREV +#define SDL_strrev _strrev +#else +extern DECLSPEC char * SDLCALL SDL_strrev(char *string); +#endif + +#ifdef HAVE__STRUPR +#define SDL_strupr _strupr +#else +extern DECLSPEC char * SDLCALL SDL_strupr(char *string); +#endif + +#ifdef HAVE__STRLWR +#define SDL_strlwr _strlwr +#else +extern DECLSPEC char * SDLCALL SDL_strlwr(char *string); +#endif + +#ifdef HAVE_STRCHR +#define SDL_strchr strchr +#elif defined(HAVE_INDEX) +#define SDL_strchr index +#else +extern DECLSPEC char * SDLCALL SDL_strchr(const char *string, int c); +#endif + +#ifdef HAVE_STRRCHR +#define SDL_strrchr strrchr +#elif defined(HAVE_RINDEX) +#define SDL_strrchr rindex +#else +extern DECLSPEC char * SDLCALL SDL_strrchr(const char *string, int c); +#endif + +#ifdef HAVE_STRSTR +#define SDL_strstr strstr +#else +extern DECLSPEC char * SDLCALL SDL_strstr(const char *haystack, const char *needle); +#endif + +#ifdef HAVE_ITOA +#define SDL_itoa itoa +#else +#define SDL_itoa(value, string, radix) SDL_ltoa((long)value, string, radix) +#endif + +#ifdef HAVE__LTOA +#define SDL_ltoa _ltoa +#else +extern DECLSPEC char * SDLCALL SDL_ltoa(long value, char *string, int radix); +#endif + +#ifdef HAVE__UITOA +#define SDL_uitoa _uitoa +#else +#define SDL_uitoa(value, string, radix) SDL_ultoa((long)value, string, radix) +#endif + +#ifdef HAVE__ULTOA +#define SDL_ultoa _ultoa +#else +extern DECLSPEC char * SDLCALL SDL_ultoa(unsigned long value, char *string, int radix); +#endif + +#ifdef HAVE_STRTOL +#define SDL_strtol strtol +#else +extern DECLSPEC long SDLCALL SDL_strtol(const char *string, char **endp, int base); +#endif + +#ifdef HAVE_STRTOUL +#define SDL_strtoul strtoul +#else +extern DECLSPEC unsigned long SDLCALL SDL_strtoul(const char *string, char **endp, int base); +#endif + +#ifdef SDL_HAS_64BIT_TYPE + +#ifdef HAVE__I64TOA +#define SDL_lltoa _i64toa +#else +extern DECLSPEC char* SDLCALL SDL_lltoa(Sint64 value, char *string, int radix); +#endif + +#ifdef HAVE__UI64TOA +#define SDL_ulltoa _ui64toa +#else +extern DECLSPEC char* SDLCALL SDL_ulltoa(Uint64 value, char *string, int radix); +#endif + +#ifdef HAVE_STRTOLL +#define SDL_strtoll strtoll +#else +extern DECLSPEC Sint64 SDLCALL SDL_strtoll(const char *string, char **endp, int base); +#endif + +#ifdef HAVE_STRTOULL +#define SDL_strtoull strtoull +#else +extern DECLSPEC Uint64 SDLCALL SDL_strtoull(const char *string, char **endp, int base); +#endif + +#endif /* SDL_HAS_64BIT_TYPE */ + +#ifdef HAVE_STRTOD +#define SDL_strtod strtod +#else +extern DECLSPEC double SDLCALL SDL_strtod(const char *string, char **endp); +#endif + +#ifdef HAVE_ATOI +#define SDL_atoi atoi +#else +#define SDL_atoi(X) SDL_strtol(X, NULL, 0) +#endif + +#ifdef HAVE_ATOF +#define SDL_atof atof +#else +#define SDL_atof(X) SDL_strtod(X, NULL) +#endif + +#ifdef HAVE_STRCMP +#define SDL_strcmp strcmp +#else +extern DECLSPEC int SDLCALL SDL_strcmp(const char *str1, const char *str2); +#endif + +#ifdef HAVE_STRNCMP +#define SDL_strncmp strncmp +#else +extern DECLSPEC int SDLCALL SDL_strncmp(const char *str1, const char *str2, size_t maxlen); +#endif + +#ifdef HAVE_STRCASECMP +#define SDL_strcasecmp strcasecmp +#elif defined(HAVE__STRICMP) +#define SDL_strcasecmp _stricmp +#else +extern DECLSPEC int SDLCALL SDL_strcasecmp(const char *str1, const char *str2); +#endif + +#ifdef HAVE_STRNCASECMP +#define SDL_strncasecmp strncasecmp +#elif defined(HAVE__STRNICMP) +#define SDL_strncasecmp _strnicmp +#else +extern DECLSPEC int SDLCALL SDL_strncasecmp(const char *str1, const char *str2, size_t maxlen); +#endif + +#ifdef HAVE_SSCANF +#define SDL_sscanf sscanf +#else +extern DECLSPEC int SDLCALL SDL_sscanf(const char *text, const char *fmt, ...); +#endif + +#ifdef HAVE_SNPRINTF +#define SDL_snprintf snprintf +#else +extern DECLSPEC int SDLCALL SDL_snprintf(char *text, size_t maxlen, const char *fmt, ...); +#endif + +#ifdef HAVE_VSNPRINTF +#define SDL_vsnprintf vsnprintf +#else +extern DECLSPEC int SDLCALL SDL_vsnprintf(char *text, size_t maxlen, const char *fmt, va_list ap); +#endif + +/* The SDL implementation of iconv() returns these error codes */ +#define SDL_ICONV_ERROR (size_t)-1 +#define SDL_ICONV_E2BIG (size_t)-2 +#define SDL_ICONV_EILSEQ (size_t)-3 +#define SDL_ICONV_EINVAL (size_t)-4 + +#ifdef HAVE_ICONV +#define SDL_iconv_t iconv_t +#define SDL_iconv_open iconv_open +#define SDL_iconv_close iconv_close +extern DECLSPEC size_t SDLCALL SDL_iconv(SDL_iconv_t cd, char **inbuf, size_t *inbytesleft, char **outbuf, size_t *outbytesleft); +#else +typedef struct _SDL_iconv_t *SDL_iconv_t; +extern DECLSPEC SDL_iconv_t SDLCALL SDL_iconv_open(const char *tocode, const char *fromcode); +extern DECLSPEC int SDLCALL SDL_iconv_close(SDL_iconv_t cd); +extern DECLSPEC size_t SDLCALL SDL_iconv(SDL_iconv_t cd, char **inbuf, size_t *inbytesleft, char **outbuf, size_t *outbytesleft); +#endif +/* This function converts a string between encodings in one pass, returning a + string that must be freed with SDL_free() or NULL on error. +*/ +extern DECLSPEC char * SDLCALL SDL_iconv_string(const char *tocode, const char *fromcode, char *inbuf, size_t inbytesleft); +#define SDL_iconv_utf8_ascii(S) SDL_iconv_string("ASCII", "UTF-8", S, SDL_strlen(S)+1) +#define SDL_iconv_utf8_latin1(S) SDL_iconv_string("LATIN1", "UTF-8", S, SDL_strlen(S)+1) +#define SDL_iconv_utf8_ucs2(S) (Uint16 *)SDL_iconv_string("UCS-2", "UTF-8", S, SDL_strlen(S)+1) +#define SDL_iconv_utf8_ucs4(S) (Uint32 *)SDL_iconv_string("UCS-4", "UTF-8", S, SDL_strlen(S)+1) + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* _SDL_stdinc_h */ diff --git a/src/win32/dependencies/sdl/SDL_syswm.h b/src/win32/dependencies/sdl/SDL_syswm.h new file mode 100644 index 00000000..e7f35139 --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_syswm.h @@ -0,0 +1,210 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* Include file for SDL custom system window manager hooks */ + +#ifndef _SDL_syswm_h +#define _SDL_syswm_h + +#include "SDL_stdinc.h" +#include "SDL_error.h" +#include "SDL_version.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* Your application has access to a special type of event 'SDL_SYSWMEVENT', + which contains window-manager specific information and arrives whenever + an unhandled window event occurs. This event is ignored by default, but + you can enable it with SDL_EventState() +*/ +#ifdef SDL_PROTOTYPES_ONLY +struct SDL_SysWMinfo; +typedef struct SDL_SysWMinfo SDL_SysWMinfo; +#else + +/* This is the structure for custom window manager events */ +#if defined(SDL_VIDEO_DRIVER_X11) +#if defined(__APPLE__) && defined(__MACH__) +/* conflicts with Quickdraw.h */ +#define Cursor X11Cursor +#endif + +#include +#include + +#if defined(__APPLE__) && defined(__MACH__) +/* matches the re-define above */ +#undef Cursor +#endif + +/* These are the various supported subsystems under UNIX */ +typedef enum { + SDL_SYSWM_X11 +} SDL_SYSWM_TYPE; + +/* The UNIX custom event structure */ +struct SDL_SysWMmsg { + SDL_version version; + SDL_SYSWM_TYPE subsystem; + union { + XEvent xevent; + } event; +}; + +/* The UNIX custom window manager information structure. + When this structure is returned, it holds information about which + low level system it is using, and will be one of SDL_SYSWM_TYPE. + */ +typedef struct SDL_SysWMinfo { + SDL_version version; + SDL_SYSWM_TYPE subsystem; + union { + struct { + Display *display; /* The X11 display */ + Window window; /* The X11 display window */ + /* These locking functions should be called around + any X11 functions using the display variable. + They lock the event thread, so should not be + called around event functions or from event filters. + */ + void (*lock_func)(void); + void (*unlock_func)(void); + + /* Introduced in SDL 1.0.2 */ + Window fswindow; /* The X11 fullscreen window */ + Window wmwindow; /* The X11 managed input window */ + } x11; + } info; +} SDL_SysWMinfo; + +#elif defined(SDL_VIDEO_DRIVER_NANOX) +#include + +/* The generic custom event structure */ +struct SDL_SysWMmsg { + SDL_version version; + int data; +}; + +/* The windows custom window manager information structure */ +typedef struct SDL_SysWMinfo { + SDL_version version ; + GR_WINDOW_ID window ; /* The display window */ +} SDL_SysWMinfo; + +#elif defined(SDL_VIDEO_DRIVER_WINDIB) || defined(SDL_VIDEO_DRIVER_DDRAW) || defined(SDL_VIDEO_DRIVER_GAPI) +#define WIN32_LEAN_AND_MEAN +#include + +/* The windows custom event structure */ +struct SDL_SysWMmsg { + SDL_version version; + HWND hwnd; /* The window for the message */ + UINT msg; /* The type of message */ + WPARAM wParam; /* WORD message parameter */ + LPARAM lParam; /* LONG message parameter */ +}; + +/* The windows custom window manager information structure */ +typedef struct SDL_SysWMinfo { + SDL_version version; + HWND window; /* The Win32 display window */ + HGLRC hglrc; /* The OpenGL context, if any */ +} SDL_SysWMinfo; + +#elif defined(SDL_VIDEO_DRIVER_RISCOS) + +/* RISC OS custom event structure */ +struct SDL_SysWMmsg { + SDL_version version; + int eventCode; /* The window for the message */ + int pollBlock[64]; +}; + +/* The RISC OS custom window manager information structure */ +typedef struct SDL_SysWMinfo { + SDL_version version; + int wimpVersion; /* Wimp version running under */ + int taskHandle; /* The RISC OS task handle */ + int window; /* The RISC OS display window */ +} SDL_SysWMinfo; + +#elif defined(SDL_VIDEO_DRIVER_PHOTON) +#include +#include + +/* The QNX custom event structure */ +struct SDL_SysWMmsg { + SDL_version version; + int data; +}; + +/* The QNX custom window manager information structure */ +typedef struct SDL_SysWMinfo { + SDL_version version; + int data; +} SDL_SysWMinfo; + +#else + +/* The generic custom event structure */ +struct SDL_SysWMmsg { + SDL_version version; + int data; +}; + +/* The generic custom window manager information structure */ +typedef struct SDL_SysWMinfo { + SDL_version version; + int data; +} SDL_SysWMinfo; + +#endif /* video driver type */ + +#endif /* SDL_PROTOTYPES_ONLY */ + +/* Function prototypes */ +/* + * This function gives you custom hooks into the window manager information. + * It fills the structure pointed to by 'info' with custom information and + * returns 1 if the function is implemented. If it's not implemented, or + * the version member of the 'info' structure is invalid, it returns 0. + * + * You typically use this function like this: + * SDL_SysWMInfo info; + * SDL_VERSION(&info.version); + * if ( SDL_GetWMInfo(&info) ) { ... } + */ +extern DECLSPEC int SDLCALL SDL_GetWMInfo(SDL_SysWMinfo *info); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* _SDL_syswm_h */ diff --git a/src/win32/dependencies/sdl/SDL_thread.h b/src/win32/dependencies/sdl/SDL_thread.h new file mode 100644 index 00000000..80746fe3 --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_thread.h @@ -0,0 +1,119 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +#ifndef _SDL_thread_h +#define _SDL_thread_h + +/* Header for the SDL thread management routines + + These are independent of the other SDL routines. +*/ + +#include "SDL_stdinc.h" +#include "SDL_error.h" + +/* Thread synchronization primitives */ +#include "SDL_mutex.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* The SDL thread structure, defined in SDL_thread.c */ +struct SDL_Thread; +typedef struct SDL_Thread SDL_Thread; + +/* Create a thread */ +#if (defined(__WIN32__) && !defined(HAVE_LIBC)) || defined(__OS2__) +/* + We compile SDL into a DLL on OS/2. This means, that it's the DLL which + creates a new thread for the calling process with the SDL_CreateThread() + API. There is a problem with this, that only the RTL of the SDL.DLL will + be initialized for those threads, and not the RTL of the calling application! + To solve this, we make a little hack here. + We'll always use the caller's _beginthread() and _endthread() APIs to + start a new thread. This way, if it's the SDL.DLL which uses this API, + then the RTL of SDL.DLL will be used to create the new thread, and if it's + the application, then the RTL of the application will be used. + So, in short: + Always use the _beginthread() and _endthread() of the calling runtime library! +*/ +#define SDL_PASSED_BEGINTHREAD_ENDTHREAD +#ifndef _WIN32_WCE +#include /* This has _beginthread() and _endthread() defined! */ +#endif + +#ifdef __OS2__ +typedef int (*pfnSDL_CurrentBeginThread)(void (*func)(void *), void *, unsigned, void *arg); +typedef void (*pfnSDL_CurrentEndThread)(void); +#elif __GNUC__ +typedef unsigned long (__cdecl *pfnSDL_CurrentBeginThread) (void *, unsigned, + unsigned (__stdcall *func)(void *), void *arg, + unsigned, unsigned *threadID); +typedef void (__cdecl *pfnSDL_CurrentEndThread)(unsigned code); +#else +typedef uintptr_t (__cdecl *pfnSDL_CurrentBeginThread) (void *, unsigned, + unsigned (__stdcall *func)(void *), void *arg, + unsigned, unsigned *threadID); +typedef void (__cdecl *pfnSDL_CurrentEndThread)(unsigned code); +#endif + +extern DECLSPEC SDL_Thread * SDLCALL SDL_CreateThread(int (*fn)(void *), void *data, pfnSDL_CurrentBeginThread pfnBeginThread, pfnSDL_CurrentEndThread pfnEndThread); + +#ifdef __OS2__ +#define SDL_CreateThread(fn, data) SDL_CreateThread(fn, data, _beginthread, _endthread) +#elif defined(_WIN32_WCE) +#define SDL_CreateThread(fn, data) SDL_CreateThread(fn, data, NULL, NULL) +#else +#define SDL_CreateThread(fn, data) SDL_CreateThread(fn, data, _beginthreadex, _endthreadex) +#endif +#else +extern DECLSPEC SDL_Thread * SDLCALL SDL_CreateThread(int (SDLCALL *fn)(void *), void *data); +#endif + +/* Get the 32-bit thread identifier for the current thread */ +extern DECLSPEC Uint32 SDLCALL SDL_ThreadID(void); + +/* Get the 32-bit thread identifier for the specified thread, + equivalent to SDL_ThreadID() if the specified thread is NULL. + */ +extern DECLSPEC Uint32 SDLCALL SDL_GetThreadID(SDL_Thread *thread); + +/* Wait for a thread to finish. + The return code for the thread function is placed in the area + pointed to by 'status', if 'status' is not NULL. + */ +extern DECLSPEC void SDLCALL SDL_WaitThread(SDL_Thread *thread, int *status); + +/* Forcefully kill a thread without worrying about its state */ +extern DECLSPEC void SDLCALL SDL_KillThread(SDL_Thread *thread); + + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* _SDL_thread_h */ diff --git a/src/win32/dependencies/sdl/SDL_timer.h b/src/win32/dependencies/sdl/SDL_timer.h new file mode 100644 index 00000000..d21159fe --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_timer.h @@ -0,0 +1,115 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +#ifndef _SDL_timer_h +#define _SDL_timer_h + +/* Header for the SDL time management routines */ + +#include "SDL_stdinc.h" +#include "SDL_error.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* This is the OS scheduler timeslice, in milliseconds */ +#define SDL_TIMESLICE 10 + +/* This is the maximum resolution of the SDL timer on all platforms */ +#define TIMER_RESOLUTION 10 /* Experimentally determined */ + +/* Get the number of milliseconds since the SDL library initialization. + * Note that this value wraps if the program runs for more than ~49 days. + */ +extern DECLSPEC Uint32 SDLCALL SDL_GetTicks(void); + +/* Wait a specified number of milliseconds before returning */ +extern DECLSPEC void SDLCALL SDL_Delay(Uint32 ms); + +/* Function prototype for the timer callback function */ +typedef Uint32 (SDLCALL *SDL_TimerCallback)(Uint32 interval); + +/* Set a callback to run after the specified number of milliseconds has + * elapsed. The callback function is passed the current timer interval + * and returns the next timer interval. If the returned value is the + * same as the one passed in, the periodic alarm continues, otherwise a + * new alarm is scheduled. If the callback returns 0, the periodic alarm + * is cancelled. + * + * To cancel a currently running timer, call SDL_SetTimer(0, NULL); + * + * The timer callback function may run in a different thread than your + * main code, and so shouldn't call any functions from within itself. + * + * The maximum resolution of this timer is 10 ms, which means that if + * you request a 16 ms timer, your callback will run approximately 20 ms + * later on an unloaded system. If you wanted to set a flag signaling + * a frame update at 30 frames per second (every 33 ms), you might set a + * timer for 30 ms: + * SDL_SetTimer((33/10)*10, flag_update); + * + * If you use this function, you need to pass SDL_INIT_TIMER to SDL_Init(). + * + * Under UNIX, you should not use raise or use SIGALRM and this function + * in the same program, as it is implemented using setitimer(). You also + * should not use this function in multi-threaded applications as signals + * to multi-threaded apps have undefined behavior in some implementations. + * + * This function returns 0 if successful, or -1 if there was an error. + */ +extern DECLSPEC int SDLCALL SDL_SetTimer(Uint32 interval, SDL_TimerCallback callback); + +/* New timer API, supports multiple timers + * Written by Stephane Peter + */ + +/* Function prototype for the new timer callback function. + * The callback function is passed the current timer interval and returns + * the next timer interval. If the returned value is the same as the one + * passed in, the periodic alarm continues, otherwise a new alarm is + * scheduled. If the callback returns 0, the periodic alarm is cancelled. + */ +typedef Uint32 (SDLCALL *SDL_NewTimerCallback)(Uint32 interval, void *param); + +/* Definition of the timer ID type */ +typedef struct _SDL_TimerID *SDL_TimerID; + +/* Add a new timer to the pool of timers already running. + Returns a timer ID, or NULL when an error occurs. + */ +extern DECLSPEC SDL_TimerID SDLCALL SDL_AddTimer(Uint32 interval, SDL_NewTimerCallback callback, void *param); + +/* Remove one of the multiple timers knowing its ID. + * Returns a boolean value indicating success. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_RemoveTimer(SDL_TimerID t); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* _SDL_timer_h */ diff --git a/src/win32/dependencies/sdl/SDL_types.h b/src/win32/dependencies/sdl/SDL_types.h new file mode 100644 index 00000000..853b9ce4 --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_types.h @@ -0,0 +1,24 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* DEPRECATED */ +#include "SDL_stdinc.h" diff --git a/src/win32/dependencies/sdl/SDL_version.h b/src/win32/dependencies/sdl/SDL_version.h new file mode 100644 index 00000000..52dd52f4 --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_version.h @@ -0,0 +1,85 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* This header defines the current SDL version */ + +#ifndef _SDL_version_h +#define _SDL_version_h + +#include "SDL_stdinc.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* Printable format: "%d.%d.%d", MAJOR, MINOR, PATCHLEVEL +*/ +#define SDL_MAJOR_VERSION 1 +#define SDL_MINOR_VERSION 2 +#define SDL_PATCHLEVEL 10 + +typedef struct SDL_version { + Uint8 major; + Uint8 minor; + Uint8 patch; +} SDL_version; + +/* This macro can be used to fill a version structure with the compile-time + * version of the SDL library. + */ +#define SDL_VERSION(X) \ +{ \ + (X)->major = SDL_MAJOR_VERSION; \ + (X)->minor = SDL_MINOR_VERSION; \ + (X)->patch = SDL_PATCHLEVEL; \ +} + +/* This macro turns the version numbers into a numeric value: + (1,2,3) -> (1203) + This assumes that there will never be more than 100 patchlevels +*/ +#define SDL_VERSIONNUM(X, Y, Z) \ + ((X)*1000 + (Y)*100 + (Z)) + +/* This is the version number macro for the current SDL version */ +#define SDL_COMPILEDVERSION \ + SDL_VERSIONNUM(SDL_MAJOR_VERSION, SDL_MINOR_VERSION, SDL_PATCHLEVEL) + +/* This macro will evaluate to true if compiled with SDL at least X.Y.Z */ +#define SDL_VERSION_ATLEAST(X, Y, Z) \ + (SDL_COMPILEDVERSION >= SDL_VERSIONNUM(X, Y, Z)) + +/* This function gets the version of the dynamically linked SDL library. + it should NOT be used to fill a version structure, instead you should + use the SDL_Version() macro. + */ +extern DECLSPEC const SDL_version * SDLCALL SDL_Linked_Version(void); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* _SDL_version_h */ diff --git a/src/win32/dependencies/sdl/SDL_video.h b/src/win32/dependencies/sdl/SDL_video.h new file mode 100644 index 00000000..720022e3 --- /dev/null +++ b/src/win32/dependencies/sdl/SDL_video.h @@ -0,0 +1,889 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2006 Sam Lantinga + + This library 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 library 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 library; if not, write to the Free Software + Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* Header file for access to the SDL raw framebuffer window */ + +#ifndef _SDL_video_h +#define _SDL_video_h + +#include "SDL_stdinc.h" +#include "SDL_error.h" +#include "SDL_rwops.h" + +#include "begin_code.h" +/* Set up for C function definitions, even when using C++ */ +#ifdef __cplusplus +extern "C" { +#endif + +/* Transparency definitions: These define alpha as the opacity of a surface */ +#define SDL_ALPHA_OPAQUE 255 +#define SDL_ALPHA_TRANSPARENT 0 + +/* Useful data types */ +typedef struct SDL_Rect { + Sint16 x, y; + Uint16 w, h; +} SDL_Rect; + +typedef struct SDL_Color { + Uint8 r; + Uint8 g; + Uint8 b; + Uint8 unused; +} SDL_Color; +#define SDL_Colour SDL_Color + +typedef struct SDL_Palette { + int ncolors; + SDL_Color *colors; +} SDL_Palette; + +/* Everything in the pixel format structure is read-only */ +typedef struct SDL_PixelFormat { + SDL_Palette *palette; + Uint8 BitsPerPixel; + Uint8 BytesPerPixel; + Uint8 Rloss; + Uint8 Gloss; + Uint8 Bloss; + Uint8 Aloss; + Uint8 Rshift; + Uint8 Gshift; + Uint8 Bshift; + Uint8 Ashift; + Uint32 Rmask; + Uint32 Gmask; + Uint32 Bmask; + Uint32 Amask; + + /* RGB color key information */ + Uint32 colorkey; + /* Alpha value information (per-surface alpha) */ + Uint8 alpha; +} SDL_PixelFormat; + +/* This structure should be treated as read-only, except for 'pixels', + which, if not NULL, contains the raw pixel data for the surface. +*/ +typedef struct SDL_Surface { + Uint32 flags; /* Read-only */ + SDL_PixelFormat *format; /* Read-only */ + int w, h; /* Read-only */ + Uint16 pitch; /* Read-only */ + void *pixels; /* Read-write */ + int offset; /* Private */ + + /* Hardware-specific surface info */ + struct private_hwdata *hwdata; + + /* clipping information */ + SDL_Rect clip_rect; /* Read-only */ + Uint32 unused1; /* for binary compatibility */ + + /* Allow recursive locks */ + Uint32 locked; /* Private */ + + /* info for fast blit mapping to other surfaces */ + struct SDL_BlitMap *map; /* Private */ + + /* format version, bumped at every change to invalidate blit maps */ + unsigned int format_version; /* Private */ + + /* Reference count -- used when freeing surface */ + int refcount; /* Read-mostly */ +} SDL_Surface; + +/* These are the currently supported flags for the SDL_surface */ +/* Available for SDL_CreateRGBSurface() or SDL_SetVideoMode() */ +#define SDL_SWSURFACE 0x00000000 /* Surface is in system memory */ +#define SDL_HWSURFACE 0x00000001 /* Surface is in video memory */ +#define SDL_ASYNCBLIT 0x00000004 /* Use asynchronous blits if possible */ +/* Available for SDL_SetVideoMode() */ +#define SDL_ANYFORMAT 0x10000000 /* Allow any video depth/pixel-format */ +#define SDL_HWPALETTE 0x20000000 /* Surface has exclusive palette */ +#define SDL_DOUBLEBUF 0x40000000 /* Set up double-buffered video mode */ +#define SDL_FULLSCREEN 0x80000000 /* Surface is a full screen display */ +#define SDL_OPENGL 0x00000002 /* Create an OpenGL rendering context */ +#define SDL_OPENGLBLIT 0x0000000A /* Create an OpenGL rendering context and use it for blitting */ +#define SDL_RESIZABLE 0x00000010 /* This video mode may be resized */ +#define SDL_NOFRAME 0x00000020 /* No window caption or edge frame */ +/* Used internally (read-only) */ +#define SDL_HWACCEL 0x00000100 /* Blit uses hardware acceleration */ +#define SDL_SRCCOLORKEY 0x00001000 /* Blit uses a source color key */ +#define SDL_RLEACCELOK 0x00002000 /* Private flag */ +#define SDL_RLEACCEL 0x00004000 /* Surface is RLE encoded */ +#define SDL_SRCALPHA 0x00010000 /* Blit uses source alpha blending */ +#define SDL_PREALLOC 0x01000000 /* Surface uses preallocated memory */ + +/* Evaluates to true if the surface needs to be locked before access */ +#define SDL_MUSTLOCK(surface) \ + (surface->offset || \ + ((surface->flags & (SDL_HWSURFACE|SDL_ASYNCBLIT|SDL_RLEACCEL)) != 0)) + +/* typedef for private surface blitting functions */ +typedef int (*SDL_blit)(struct SDL_Surface *src, SDL_Rect *srcrect, + struct SDL_Surface *dst, SDL_Rect *dstrect); + + +/* Useful for determining the video hardware capabilities */ +typedef struct SDL_VideoInfo { + Uint32 hw_available :1; /* Flag: Can you create hardware surfaces? */ + Uint32 wm_available :1; /* Flag: Can you talk to a window manager? */ + Uint32 UnusedBits1 :6; + Uint32 UnusedBits2 :1; + Uint32 blit_hw :1; /* Flag: Accelerated blits HW --> HW */ + Uint32 blit_hw_CC :1; /* Flag: Accelerated blits with Colorkey */ + Uint32 blit_hw_A :1; /* Flag: Accelerated blits with Alpha */ + Uint32 blit_sw :1; /* Flag: Accelerated blits SW --> HW */ + Uint32 blit_sw_CC :1; /* Flag: Accelerated blits with Colorkey */ + Uint32 blit_sw_A :1; /* Flag: Accelerated blits with Alpha */ + Uint32 blit_fill :1; /* Flag: Accelerated color fill */ + Uint32 UnusedBits3 :16; + Uint32 video_mem; /* The total amount of video memory (in K) */ + SDL_PixelFormat *vfmt; /* Value: The format of the video surface */ + int current_w; /* Value: The current video mode width */ + int current_h; /* Value: The current video mode height */ +} SDL_VideoInfo; + + +/* The most common video overlay formats. + For an explanation of these pixel formats, see: + http://www.webartz.com/fourcc/indexyuv.htm + + For information on the relationship between color spaces, see: + http://www.neuro.sfc.keio.ac.jp/~aly/polygon/info/color-space-faq.html + */ +#define SDL_YV12_OVERLAY 0x32315659 /* Planar mode: Y + V + U (3 planes) */ +#define SDL_IYUV_OVERLAY 0x56555949 /* Planar mode: Y + U + V (3 planes) */ +#define SDL_YUY2_OVERLAY 0x32595559 /* Packed mode: Y0+U0+Y1+V0 (1 plane) */ +#define SDL_UYVY_OVERLAY 0x59565955 /* Packed mode: U0+Y0+V0+Y1 (1 plane) */ +#define SDL_YVYU_OVERLAY 0x55595659 /* Packed mode: Y0+V0+Y1+U0 (1 plane) */ + +/* The YUV hardware video overlay */ +typedef struct SDL_Overlay { + Uint32 format; /* Read-only */ + int w, h; /* Read-only */ + int planes; /* Read-only */ + Uint16 *pitches; /* Read-only */ + Uint8 **pixels; /* Read-write */ + + /* Hardware-specific surface info */ + struct private_yuvhwfuncs *hwfuncs; + struct private_yuvhwdata *hwdata; + + /* Special flags */ + Uint32 hw_overlay :1; /* Flag: This overlay hardware accelerated? */ + Uint32 UnusedBits :31; +} SDL_Overlay; + + +/* Public enumeration for setting the OpenGL window attributes. */ +typedef enum { + SDL_GL_RED_SIZE, + SDL_GL_GREEN_SIZE, + SDL_GL_BLUE_SIZE, + SDL_GL_ALPHA_SIZE, + SDL_GL_BUFFER_SIZE, + SDL_GL_DOUBLEBUFFER, + SDL_GL_DEPTH_SIZE, + SDL_GL_STENCIL_SIZE, + SDL_GL_ACCUM_RED_SIZE, + SDL_GL_ACCUM_GREEN_SIZE, + SDL_GL_ACCUM_BLUE_SIZE, + SDL_GL_ACCUM_ALPHA_SIZE, + SDL_GL_STEREO, + SDL_GL_MULTISAMPLEBUFFERS, + SDL_GL_MULTISAMPLESAMPLES, + SDL_GL_ACCELERATED_VISUAL, + SDL_GL_SWAP_CONTROL +} SDL_GLattr; + +/* flags for SDL_SetPalette() */ +#define SDL_LOGPAL 0x01 +#define SDL_PHYSPAL 0x02 + +/* Function prototypes */ + +/* These functions are used internally, and should not be used unless you + * have a specific need to specify the video driver you want to use. + * You should normally use SDL_Init() or SDL_InitSubSystem(). + * + * SDL_VideoInit() initializes the video subsystem -- sets up a connection + * to the window manager, etc, and determines the current video mode and + * pixel format, but does not initialize a window or graphics mode. + * Note that event handling is activated by this routine. + * + * If you use both sound and video in your application, you need to call + * SDL_Init() before opening the sound device, otherwise under Win32 DirectX, + * you won't be able to set full-screen display modes. + */ +extern DECLSPEC int SDLCALL SDL_VideoInit(const char *driver_name, Uint32 flags); +extern DECLSPEC void SDLCALL SDL_VideoQuit(void); + +/* This function fills the given character buffer with the name of the + * video driver, and returns a pointer to it if the video driver has + * been initialized. It returns NULL if no driver has been initialized. + */ +extern DECLSPEC char * SDLCALL SDL_VideoDriverName(char *namebuf, int maxlen); + +/* + * This function returns a pointer to the current display surface. + * If SDL is doing format conversion on the display surface, this + * function returns the publicly visible surface, not the real video + * surface. + */ +extern DECLSPEC SDL_Surface * SDLCALL SDL_GetVideoSurface(void); + +/* + * This function returns a read-only pointer to information about the + * video hardware. If this is called before SDL_SetVideoMode(), the 'vfmt' + * member of the returned structure will contain the pixel format of the + * "best" video mode. + */ +extern DECLSPEC const SDL_VideoInfo * SDLCALL SDL_GetVideoInfo(void); + +/* + * Check to see if a particular video mode is supported. + * It returns 0 if the requested mode is not supported under any bit depth, + * or returns the bits-per-pixel of the closest available mode with the + * given width and height. If this bits-per-pixel is different from the + * one used when setting the video mode, SDL_SetVideoMode() will succeed, + * but will emulate the requested bits-per-pixel with a shadow surface. + * + * The arguments to SDL_VideoModeOK() are the same ones you would pass to + * SDL_SetVideoMode() + */ +extern DECLSPEC int SDLCALL SDL_VideoModeOK(int width, int height, int bpp, Uint32 flags); + +/* + * Return a pointer to an array of available screen dimensions for the + * given format and video flags, sorted largest to smallest. Returns + * NULL if there are no dimensions available for a particular format, + * or (SDL_Rect **)-1 if any dimension is okay for the given format. + * + * If 'format' is NULL, the mode list will be for the format given + * by SDL_GetVideoInfo()->vfmt + */ +extern DECLSPEC SDL_Rect ** SDLCALL SDL_ListModes(SDL_PixelFormat *format, Uint32 flags); + +/* + * Set up a video mode with the specified width, height and bits-per-pixel. + * + * If 'bpp' is 0, it is treated as the current display bits per pixel. + * + * If SDL_ANYFORMAT is set in 'flags', the SDL library will try to set the + * requested bits-per-pixel, but will return whatever video pixel format is + * available. The default is to emulate the requested pixel format if it + * is not natively available. + * + * If SDL_HWSURFACE is set in 'flags', the video surface will be placed in + * video memory, if possible, and you may have to call SDL_LockSurface() + * in order to access the raw framebuffer. Otherwise, the video surface + * will be created in system memory. + * + * If SDL_ASYNCBLIT is set in 'flags', SDL will try to perform rectangle + * updates asynchronously, but you must always lock before accessing pixels. + * SDL will wait for updates to complete before returning from the lock. + * + * If SDL_HWPALETTE is set in 'flags', the SDL library will guarantee + * that the colors set by SDL_SetColors() will be the colors you get. + * Otherwise, in 8-bit mode, SDL_SetColors() may not be able to set all + * of the colors exactly the way they are requested, and you should look + * at the video surface structure to determine the actual palette. + * If SDL cannot guarantee that the colors you request can be set, + * i.e. if the colormap is shared, then the video surface may be created + * under emulation in system memory, overriding the SDL_HWSURFACE flag. + * + * If SDL_FULLSCREEN is set in 'flags', the SDL library will try to set + * a fullscreen video mode. The default is to create a windowed mode + * if the current graphics system has a window manager. + * If the SDL library is able to set a fullscreen video mode, this flag + * will be set in the surface that is returned. + * + * If SDL_DOUBLEBUF is set in 'flags', the SDL library will try to set up + * two surfaces in video memory and swap between them when you call + * SDL_Flip(). This is usually slower than the normal single-buffering + * scheme, but prevents "tearing" artifacts caused by modifying video + * memory while the monitor is refreshing. It should only be used by + * applications that redraw the entire screen on every update. + * + * If SDL_RESIZABLE is set in 'flags', the SDL library will allow the + * window manager, if any, to resize the window at runtime. When this + * occurs, SDL will send a SDL_VIDEORESIZE event to you application, + * and you must respond to the event by re-calling SDL_SetVideoMode() + * with the requested size (or another size that suits the application). + * + * If SDL_NOFRAME is set in 'flags', the SDL library will create a window + * without any title bar or frame decoration. Fullscreen video modes have + * this flag set automatically. + * + * This function returns the video framebuffer surface, or NULL if it fails. + * + * If you rely on functionality provided by certain video flags, check the + * flags of the returned surface to make sure that functionality is available. + * SDL will fall back to reduced functionality if the exact flags you wanted + * are not available. + */ +extern DECLSPEC SDL_Surface * SDLCALL SDL_SetVideoMode + (int width, int height, int bpp, Uint32 flags); + +/* + * Makes sure the given list of rectangles is updated on the given screen. + * If 'x', 'y', 'w' and 'h' are all 0, SDL_UpdateRect will update the entire + * screen. + * These functions should not be called while 'screen' is locked. + */ +extern DECLSPEC void SDLCALL SDL_UpdateRects + (SDL_Surface *screen, int numrects, SDL_Rect *rects); +extern DECLSPEC void SDLCALL SDL_UpdateRect + (SDL_Surface *screen, Sint32 x, Sint32 y, Uint32 w, Uint32 h); + +/* + * On hardware that supports double-buffering, this function sets up a flip + * and returns. The hardware will wait for vertical retrace, and then swap + * video buffers before the next video surface blit or lock will return. + * On hardware that doesn not support double-buffering, this is equivalent + * to calling SDL_UpdateRect(screen, 0, 0, 0, 0); + * The SDL_DOUBLEBUF flag must have been passed to SDL_SetVideoMode() when + * setting the video mode for this function to perform hardware flipping. + * This function returns 0 if successful, or -1 if there was an error. + */ +extern DECLSPEC int SDLCALL SDL_Flip(SDL_Surface *screen); + +/* + * Set the gamma correction for each of the color channels. + * The gamma values range (approximately) between 0.1 and 10.0 + * + * If this function isn't supported directly by the hardware, it will + * be emulated using gamma ramps, if available. If successful, this + * function returns 0, otherwise it returns -1. + */ +extern DECLSPEC int SDLCALL SDL_SetGamma(float red, float green, float blue); + +/* + * Set the gamma translation table for the red, green, and blue channels + * of the video hardware. Each table is an array of 256 16-bit quantities, + * representing a mapping between the input and output for that channel. + * The input is the index into the array, and the output is the 16-bit + * gamma value at that index, scaled to the output color precision. + * + * You may pass NULL for any of the channels to leave it unchanged. + * If the call succeeds, it will return 0. If the display driver or + * hardware does not support gamma translation, or otherwise fails, + * this function will return -1. + */ +extern DECLSPEC int SDLCALL SDL_SetGammaRamp(const Uint16 *red, const Uint16 *green, const Uint16 *blue); + +/* + * Retrieve the current values of the gamma translation tables. + * + * You must pass in valid pointers to arrays of 256 16-bit quantities. + * Any of the pointers may be NULL to ignore that channel. + * If the call succeeds, it will return 0. If the display driver or + * hardware does not support gamma translation, or otherwise fails, + * this function will return -1. + */ +extern DECLSPEC int SDLCALL SDL_GetGammaRamp(Uint16 *red, Uint16 *green, Uint16 *blue); + +/* + * Sets a portion of the colormap for the given 8-bit surface. If 'surface' + * is not a palettized surface, this function does nothing, returning 0. + * If all of the colors were set as passed to SDL_SetColors(), it will + * return 1. If not all the color entries were set exactly as given, + * it will return 0, and you should look at the surface palette to + * determine the actual color palette. + * + * When 'surface' is the surface associated with the current display, the + * display colormap will be updated with the requested colors. If + * SDL_HWPALETTE was set in SDL_SetVideoMode() flags, SDL_SetColors() + * will always return 1, and the palette is guaranteed to be set the way + * you desire, even if the window colormap has to be warped or run under + * emulation. + */ +extern DECLSPEC int SDLCALL SDL_SetColors(SDL_Surface *surface, + SDL_Color *colors, int firstcolor, int ncolors); + +/* + * Sets a portion of the colormap for a given 8-bit surface. + * 'flags' is one or both of: + * SDL_LOGPAL -- set logical palette, which controls how blits are mapped + * to/from the surface, + * SDL_PHYSPAL -- set physical palette, which controls how pixels look on + * the screen + * Only screens have physical palettes. Separate change of physical/logical + * palettes is only possible if the screen has SDL_HWPALETTE set. + * + * The return value is 1 if all colours could be set as requested, and 0 + * otherwise. + * + * SDL_SetColors() is equivalent to calling this function with + * flags = (SDL_LOGPAL|SDL_PHYSPAL). + */ +extern DECLSPEC int SDLCALL SDL_SetPalette(SDL_Surface *surface, int flags, + SDL_Color *colors, int firstcolor, + int ncolors); + +/* + * Maps an RGB triple to an opaque pixel value for a given pixel format + */ +extern DECLSPEC Uint32 SDLCALL SDL_MapRGB + (SDL_PixelFormat *format, Uint8 r, Uint8 g, Uint8 b); + +/* + * Maps an RGBA quadruple to a pixel value for a given pixel format + */ +extern DECLSPEC Uint32 SDLCALL SDL_MapRGBA(SDL_PixelFormat *format, + Uint8 r, Uint8 g, Uint8 b, Uint8 a); + +/* + * Maps a pixel value into the RGB components for a given pixel format + */ +extern DECLSPEC void SDLCALL SDL_GetRGB(Uint32 pixel, SDL_PixelFormat *fmt, + Uint8 *r, Uint8 *g, Uint8 *b); + +/* + * Maps a pixel value into the RGBA components for a given pixel format + */ +extern DECLSPEC void SDLCALL SDL_GetRGBA(Uint32 pixel, SDL_PixelFormat *fmt, + Uint8 *r, Uint8 *g, Uint8 *b, Uint8 *a); + +/* + * Allocate and free an RGB surface (must be called after SDL_SetVideoMode) + * If the depth is 4 or 8 bits, an empty palette is allocated for the surface. + * If the depth is greater than 8 bits, the pixel format is set using the + * flags '[RGB]mask'. + * If the function runs out of memory, it will return NULL. + * + * The 'flags' tell what kind of surface to create. + * SDL_SWSURFACE means that the surface should be created in system memory. + * SDL_HWSURFACE means that the surface should be created in video memory, + * with the same format as the display surface. This is useful for surfaces + * that will not change much, to take advantage of hardware acceleration + * when being blitted to the display surface. + * SDL_ASYNCBLIT means that SDL will try to perform asynchronous blits with + * this surface, but you must always lock it before accessing the pixels. + * SDL will wait for current blits to finish before returning from the lock. + * SDL_SRCCOLORKEY indicates that the surface will be used for colorkey blits. + * If the hardware supports acceleration of colorkey blits between + * two surfaces in video memory, SDL will try to place the surface in + * video memory. If this isn't possible or if there is no hardware + * acceleration available, the surface will be placed in system memory. + * SDL_SRCALPHA means that the surface will be used for alpha blits and + * if the hardware supports hardware acceleration of alpha blits between + * two surfaces in video memory, to place the surface in video memory + * if possible, otherwise it will be placed in system memory. + * If the surface is created in video memory, blits will be _much_ faster, + * but the surface format must be identical to the video surface format, + * and the only way to access the pixels member of the surface is to use + * the SDL_LockSurface() and SDL_UnlockSurface() calls. + * If the requested surface actually resides in video memory, SDL_HWSURFACE + * will be set in the flags member of the returned surface. If for some + * reason the surface could not be placed in video memory, it will not have + * the SDL_HWSURFACE flag set, and will be created in system memory instead. + */ +#define SDL_AllocSurface SDL_CreateRGBSurface +extern DECLSPEC SDL_Surface * SDLCALL SDL_CreateRGBSurface + (Uint32 flags, int width, int height, int depth, + Uint32 Rmask, Uint32 Gmask, Uint32 Bmask, Uint32 Amask); +extern DECLSPEC SDL_Surface * SDLCALL SDL_CreateRGBSurfaceFrom(void *pixels, + int width, int height, int depth, int pitch, + Uint32 Rmask, Uint32 Gmask, Uint32 Bmask, Uint32 Amask); +extern DECLSPEC void SDLCALL SDL_FreeSurface(SDL_Surface *surface); + +/* + * SDL_LockSurface() sets up a surface for directly accessing the pixels. + * Between calls to SDL_LockSurface()/SDL_UnlockSurface(), you can write + * to and read from 'surface->pixels', using the pixel format stored in + * 'surface->format'. Once you are done accessing the surface, you should + * use SDL_UnlockSurface() to release it. + * + * Not all surfaces require locking. If SDL_MUSTLOCK(surface) evaluates + * to 0, then you can read and write to the surface at any time, and the + * pixel format of the surface will not change. In particular, if the + * SDL_HWSURFACE flag is not given when calling SDL_SetVideoMode(), you + * will not need to lock the display surface before accessing it. + * + * No operating system or library calls should be made between lock/unlock + * pairs, as critical system locks may be held during this time. + * + * SDL_LockSurface() returns 0, or -1 if the surface couldn't be locked. + */ +extern DECLSPEC int SDLCALL SDL_LockSurface(SDL_Surface *surface); +extern DECLSPEC void SDLCALL SDL_UnlockSurface(SDL_Surface *surface); + +/* + * Load a surface from a seekable SDL data source (memory or file.) + * If 'freesrc' is non-zero, the source will be closed after being read. + * Returns the new surface, or NULL if there was an error. + * The new surface should be freed with SDL_FreeSurface(). + */ +extern DECLSPEC SDL_Surface * SDLCALL SDL_LoadBMP_RW(SDL_RWops *src, int freesrc); + +/* Convenience macro -- load a surface from a file */ +#define SDL_LoadBMP(file) SDL_LoadBMP_RW(SDL_RWFromFile(file, "rb"), 1) + +/* + * Save a surface to a seekable SDL data source (memory or file.) + * If 'freedst' is non-zero, the source will be closed after being written. + * Returns 0 if successful or -1 if there was an error. + */ +extern DECLSPEC int SDLCALL SDL_SaveBMP_RW + (SDL_Surface *surface, SDL_RWops *dst, int freedst); + +/* Convenience macro -- save a surface to a file */ +#define SDL_SaveBMP(surface, file) \ + SDL_SaveBMP_RW(surface, SDL_RWFromFile(file, "wb"), 1) + +/* + * Sets the color key (transparent pixel) in a blittable surface. + * If 'flag' is SDL_SRCCOLORKEY (optionally OR'd with SDL_RLEACCEL), + * 'key' will be the transparent pixel in the source image of a blit. + * SDL_RLEACCEL requests RLE acceleration for the surface if present, + * and removes RLE acceleration if absent. + * If 'flag' is 0, this function clears any current color key. + * This function returns 0, or -1 if there was an error. + */ +extern DECLSPEC int SDLCALL SDL_SetColorKey + (SDL_Surface *surface, Uint32 flag, Uint32 key); + +/* + * This function sets the alpha value for the entire surface, as opposed to + * using the alpha component of each pixel. This value measures the range + * of transparency of the surface, 0 being completely transparent to 255 + * being completely opaque. An 'alpha' value of 255 causes blits to be + * opaque, the source pixels copied to the destination (the default). Note + * that per-surface alpha can be combined with colorkey transparency. + * + * If 'flag' is 0, alpha blending is disabled for the surface. + * If 'flag' is SDL_SRCALPHA, alpha blending is enabled for the surface. + * OR:ing the flag with SDL_RLEACCEL requests RLE acceleration for the + * surface; if SDL_RLEACCEL is not specified, the RLE accel will be removed. + * + * The 'alpha' parameter is ignored for surfaces that have an alpha channel. + */ +extern DECLSPEC int SDLCALL SDL_SetAlpha(SDL_Surface *surface, Uint32 flag, Uint8 alpha); + +/* + * Sets the clipping rectangle for the destination surface in a blit. + * + * If the clip rectangle is NULL, clipping will be disabled. + * If the clip rectangle doesn't intersect the surface, the function will + * return SDL_FALSE and blits will be completely clipped. Otherwise the + * function returns SDL_TRUE and blits to the surface will be clipped to + * the intersection of the surface area and the clipping rectangle. + * + * Note that blits are automatically clipped to the edges of the source + * and destination surfaces. + */ +extern DECLSPEC SDL_bool SDLCALL SDL_SetClipRect(SDL_Surface *surface, const SDL_Rect *rect); + +/* + * Gets the clipping rectangle for the destination surface in a blit. + * 'rect' must be a pointer to a valid rectangle which will be filled + * with the correct values. + */ +extern DECLSPEC void SDLCALL SDL_GetClipRect(SDL_Surface *surface, SDL_Rect *rect); + +/* + * Creates a new surface of the specified format, and then copies and maps + * the given surface to it so the blit of the converted surface will be as + * fast as possible. If this function fails, it returns NULL. + * + * The 'flags' parameter is passed to SDL_CreateRGBSurface() and has those + * semantics. You can also pass SDL_RLEACCEL in the flags parameter and + * SDL will try to RLE accelerate colorkey and alpha blits in the resulting + * surface. + * + * This function is used internally by SDL_DisplayFormat(). + */ +extern DECLSPEC SDL_Surface * SDLCALL SDL_ConvertSurface + (SDL_Surface *src, SDL_PixelFormat *fmt, Uint32 flags); + +/* + * This performs a fast blit from the source surface to the destination + * surface. It assumes that the source and destination rectangles are + * the same size. If either 'srcrect' or 'dstrect' are NULL, the entire + * surface (src or dst) is copied. The final blit rectangles are saved + * in 'srcrect' and 'dstrect' after all clipping is performed. + * If the blit is successful, it returns 0, otherwise it returns -1. + * + * The blit function should not be called on a locked surface. + * + * The blit semantics for surfaces with and without alpha and colorkey + * are defined as follows: + * + * RGBA->RGB: + * SDL_SRCALPHA set: + * alpha-blend (using alpha-channel). + * SDL_SRCCOLORKEY ignored. + * SDL_SRCALPHA not set: + * copy RGB. + * if SDL_SRCCOLORKEY set, only copy the pixels matching the + * RGB values of the source colour key, ignoring alpha in the + * comparison. + * + * RGB->RGBA: + * SDL_SRCALPHA set: + * alpha-blend (using the source per-surface alpha value); + * set destination alpha to opaque. + * SDL_SRCALPHA not set: + * copy RGB, set destination alpha to source per-surface alpha value. + * both: + * if SDL_SRCCOLORKEY set, only copy the pixels matching the + * source colour key. + * + * RGBA->RGBA: + * SDL_SRCALPHA set: + * alpha-blend (using the source alpha channel) the RGB values; + * leave destination alpha untouched. [Note: is this correct?] + * SDL_SRCCOLORKEY ignored. + * SDL_SRCALPHA not set: + * copy all of RGBA to the destination. + * if SDL_SRCCOLORKEY set, only copy the pixels matching the + * RGB values of the source colour key, ignoring alpha in the + * comparison. + * + * RGB->RGB: + * SDL_SRCALPHA set: + * alpha-blend (using the source per-surface alpha value). + * SDL_SRCALPHA not set: + * copy RGB. + * both: + * if SDL_SRCCOLORKEY set, only copy the pixels matching the + * source colour key. + * + * If either of the surfaces were in video memory, and the blit returns -2, + * the video memory was lost, so it should be reloaded with artwork and + * re-blitted: + while ( SDL_BlitSurface(image, imgrect, screen, dstrect) == -2 ) { + while ( SDL_LockSurface(image) < 0 ) + Sleep(10); + -- Write image pixels to image->pixels -- + SDL_UnlockSurface(image); + } + * This happens under DirectX 5.0 when the system switches away from your + * fullscreen application. The lock will also fail until you have access + * to the video memory again. + */ +/* You should call SDL_BlitSurface() unless you know exactly how SDL + blitting works internally and how to use the other blit functions. +*/ +#define SDL_BlitSurface SDL_UpperBlit + +/* This is the public blit function, SDL_BlitSurface(), and it performs + rectangle validation and clipping before passing it to SDL_LowerBlit() +*/ +extern DECLSPEC int SDLCALL SDL_UpperBlit + (SDL_Surface *src, SDL_Rect *srcrect, + SDL_Surface *dst, SDL_Rect *dstrect); +/* This is a semi-private blit function and it performs low-level surface + blitting only. +*/ +extern DECLSPEC int SDLCALL SDL_LowerBlit + (SDL_Surface *src, SDL_Rect *srcrect, + SDL_Surface *dst, SDL_Rect *dstrect); + +/* + * This function performs a fast fill of the given rectangle with 'color' + * The given rectangle is clipped to the destination surface clip area + * and the final fill rectangle is saved in the passed in pointer. + * If 'dstrect' is NULL, the whole surface will be filled with 'color' + * The color should be a pixel of the format used by the surface, and + * can be generated by the SDL_MapRGB() function. + * This function returns 0 on success, or -1 on error. + */ +extern DECLSPEC int SDLCALL SDL_FillRect + (SDL_Surface *dst, SDL_Rect *dstrect, Uint32 color); + +/* + * This function takes a surface and copies it to a new surface of the + * pixel format and colors of the video framebuffer, suitable for fast + * blitting onto the display surface. It calls SDL_ConvertSurface() + * + * If you want to take advantage of hardware colorkey or alpha blit + * acceleration, you should set the colorkey and alpha value before + * calling this function. + * + * If the conversion fails or runs out of memory, it returns NULL + */ +extern DECLSPEC SDL_Surface * SDLCALL SDL_DisplayFormat(SDL_Surface *surface); + +/* + * This function takes a surface and copies it to a new surface of the + * pixel format and colors of the video framebuffer (if possible), + * suitable for fast alpha blitting onto the display surface. + * The new surface will always have an alpha channel. + * + * If you want to take advantage of hardware colorkey or alpha blit + * acceleration, you should set the colorkey and alpha value before + * calling this function. + * + * If the conversion fails or runs out of memory, it returns NULL + */ +extern DECLSPEC SDL_Surface * SDLCALL SDL_DisplayFormatAlpha(SDL_Surface *surface); + + +/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ +/* YUV video surface overlay functions */ +/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ + +/* This function creates a video output overlay + Calling the returned surface an overlay is something of a misnomer because + the contents of the display surface underneath the area where the overlay + is shown is undefined - it may be overwritten with the converted YUV data. +*/ +extern DECLSPEC SDL_Overlay * SDLCALL SDL_CreateYUVOverlay(int width, int height, + Uint32 format, SDL_Surface *display); + +/* Lock an overlay for direct access, and unlock it when you are done */ +extern DECLSPEC int SDLCALL SDL_LockYUVOverlay(SDL_Overlay *overlay); +extern DECLSPEC void SDLCALL SDL_UnlockYUVOverlay(SDL_Overlay *overlay); + +/* Blit a video overlay to the display surface. + The contents of the video surface underneath the blit destination are + not defined. + The width and height of the destination rectangle may be different from + that of the overlay, but currently only 2x scaling is supported. +*/ +extern DECLSPEC int SDLCALL SDL_DisplayYUVOverlay(SDL_Overlay *overlay, SDL_Rect *dstrect); + +/* Free a video overlay */ +extern DECLSPEC void SDLCALL SDL_FreeYUVOverlay(SDL_Overlay *overlay); + + +/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ +/* OpenGL support functions. */ +/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ + +/* + * Dynamically load an OpenGL library, or the default one if path is NULL + * + * If you do this, you need to retrieve all of the GL functions used in + * your program from the dynamic library using SDL_GL_GetProcAddress(). + */ +extern DECLSPEC int SDLCALL SDL_GL_LoadLibrary(const char *path); + +/* + * Get the address of a GL function + */ +extern DECLSPEC void * SDLCALL SDL_GL_GetProcAddress(const char* proc); + +/* + * Set an attribute of the OpenGL subsystem before intialization. + */ +extern DECLSPEC int SDLCALL SDL_GL_SetAttribute(SDL_GLattr attr, int value); + +/* + * Get an attribute of the OpenGL subsystem from the windowing + * interface, such as glX. This is of course different from getting + * the values from SDL's internal OpenGL subsystem, which only + * stores the values you request before initialization. + * + * Developers should track the values they pass into SDL_GL_SetAttribute + * themselves if they want to retrieve these values. + */ +extern DECLSPEC int SDLCALL SDL_GL_GetAttribute(SDL_GLattr attr, int* value); + +/* + * Swap the OpenGL buffers, if double-buffering is supported. + */ +extern DECLSPEC void SDLCALL SDL_GL_SwapBuffers(void); + +/* + * Internal functions that should not be called unless you have read + * and understood the source code for these functions. + */ +extern DECLSPEC void SDLCALL SDL_GL_UpdateRects(int numrects, SDL_Rect* rects); +extern DECLSPEC void SDLCALL SDL_GL_Lock(void); +extern DECLSPEC void SDLCALL SDL_GL_Unlock(void); + +/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ +/* These functions allow interaction with the window manager, if any. */ +/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ + +/* + * Sets/Gets the title and icon text of the display window (UTF-8 encoded) + */ +extern DECLSPEC void SDLCALL SDL_WM_SetCaption(const char *title, const char *icon); +extern DECLSPEC void SDLCALL SDL_WM_GetCaption(char **title, char **icon); + +/* + * Sets the icon for the display window. + * This function must be called before the first call to SDL_SetVideoMode(). + * It takes an icon surface, and a mask in MSB format. + * If 'mask' is NULL, the entire icon surface will be used as the icon. + */ +extern DECLSPEC void SDLCALL SDL_WM_SetIcon(SDL_Surface *icon, Uint8 *mask); + +/* + * This function iconifies the window, and returns 1 if it succeeded. + * If the function succeeds, it generates an SDL_APPACTIVE loss event. + * This function is a noop and returns 0 in non-windowed environments. + */ +extern DECLSPEC int SDLCALL SDL_WM_IconifyWindow(void); + +/* + * Toggle fullscreen mode without changing the contents of the screen. + * If the display surface does not require locking before accessing + * the pixel information, then the memory pointers will not change. + * + * If this function was able to toggle fullscreen mode (change from + * running in a window to fullscreen, or vice-versa), it will return 1. + * If it is not implemented, or fails, it returns 0. + * + * The next call to SDL_SetVideoMode() will set the mode fullscreen + * attribute based on the flags parameter - if SDL_FULLSCREEN is not + * set, then the display will be windowed by default where supported. + * + * This is currently only implemented in the X11 video driver. + */ +extern DECLSPEC int SDLCALL SDL_WM_ToggleFullScreen(SDL_Surface *surface); + +/* + * This function allows you to set and query the input grab state of + * the application. It returns the new input grab state. + */ +typedef enum { + SDL_GRAB_QUERY = -1, + SDL_GRAB_OFF = 0, + SDL_GRAB_ON = 1, + SDL_GRAB_FULLSCREEN /* Used internally */ +} SDL_GrabMode; +/* + * Grabbing means that the mouse is confined to the application window, + * and nearly all keyboard input is passed directly to the application, + * and not interpreted by a window manager, if any. + */ +extern DECLSPEC SDL_GrabMode SDLCALL SDL_WM_GrabInput(SDL_GrabMode mode); + +/* Not in public API at the moment - do not use! */ +extern DECLSPEC int SDLCALL SDL_SoftStretch(SDL_Surface *src, SDL_Rect *srcrect, + SDL_Surface *dst, SDL_Rect *dstrect); + +/* Ends C function definitions when using C++ */ +#ifdef __cplusplus +} +#endif +#include "close_code.h" + +#endif /* _SDL_video_h */ diff --git a/src/win32/dependencies/sdl/begin_code.h b/src/win32/dependencies/sdl/begin_code.h new file mode 100644 index 00000000..40279337 --- /dev/null +++ b/src/win32/dependencies/sdl/begin_code.h @@ -0,0 +1,150 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2004 Sam Lantinga + + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Library General Public + License as published by the Free Software Foundation; either + version 2 of the License, or (at your option) any later version. + + This library 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 + Library General Public License for more details. + + You should have received a copy of the GNU Library General Public + License along with this library; if not, write to the Free + Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* This file sets things up for C dynamic library function definitions, + static inlined functions, and structures aligned at 4-byte alignment. + If you don't like ugly C preprocessor code, don't look at this file. :) +*/ + +/* This shouldn't be nested -- included it around code only. */ +#ifdef _begin_code_h +#error Nested inclusion of begin_code.h +#endif +#define _begin_code_h + +/* Some compilers use a special export keyword */ +#ifndef DECLSPEC +# if defined(__BEOS__) +# if defined(__GNUC__) +# define DECLSPEC __declspec(dllexport) +# else +# define DECLSPEC __declspec(export) +# endif +# elif defined(__WIN32__) +# ifdef __BORLANDC__ +# ifdef BUILD_SDL +# define DECLSPEC +# else +# define DECLSPEC __declspec(dllimport) +# endif +# else +# define DECLSPEC __declspec(dllexport) +# endif +# elif defined(__OS2__) +# ifdef __WATCOMC__ +# ifdef BUILD_SDL +# define DECLSPEC __declspec(dllexport) +# else +# define DECLSPEC +# endif +# else +# define DECLSPEC +# endif +# else +# if defined(__GNUC__) && __GNUC__ >= 4 +# define DECLSPEC __attribute__ ((visibility("default"))) +# else +# define DECLSPEC +# endif +# endif +#endif + +/* By default SDL uses the C calling convention */ +#ifndef SDLCALL +#if defined(__WIN32__) && !defined(__GNUC__) +#define SDLCALL __cdecl +#else +#ifdef __OS2__ +/* But on OS/2, we use the _System calling convention */ +/* to be compatible with every compiler */ +#define SDLCALL _System +#else +#define SDLCALL +#endif +#endif +#endif /* SDLCALL */ + +/* Removed DECLSPEC on Symbian OS because SDL cannot be a DLL in EPOC */ +#ifdef __SYMBIAN32__ +#undef DECLSPEC +#define DECLSPEC +#endif /* __SYMBIAN32__ */ + +/* Force structure packing at 4 byte alignment. + This is necessary if the header is included in code which has structure + packing set to an alternate value, say for loading structures from disk. + The packing is reset to the previous value in close_code.h + */ +#if defined(_MSC_VER) || defined(__MWERKS__) || defined(__BORLANDC__) +#ifdef _MSC_VER +#pragma warning(disable: 4103) +#endif +#ifdef __BORLANDC__ +#pragma nopackwarning +#endif +#pragma pack(push,4) +#elif (defined(__MWERKS__) && defined(__MACOS__)) +#pragma options align=mac68k4byte +#pragma enumsalwaysint on +#endif /* Compiler needs structure packing set */ + +/* Set up compiler-specific options for inlining functions */ +#ifndef SDL_INLINE_OKAY +#ifdef __GNUC__ +#define SDL_INLINE_OKAY +#else +/* Add any special compiler-specific cases here */ +#if defined(_MSC_VER) || defined(__BORLANDC__) || \ + defined(__DMC__) || defined(__SC__) || \ + defined(__WATCOMC__) || defined(__LCC__) || \ + defined(__DECC) +#ifndef __inline__ +#define __inline__ __inline +#endif +#define SDL_INLINE_OKAY +#else +#if !defined(__MRC__) && !defined(_SGI_SOURCE) +#define __inline__ inline +#define SDL_INLINE_OKAY +#endif /* Not a funky compiler */ +#endif /* Visual C++ */ +#endif /* GNU C */ +#endif /* SDL_INLINE_OKAY */ + +/* If inlining isn't supported, remove "__inline__", turning static + inlined functions into static functions (resulting in code bloat + in all files which include the offending header files) +*/ +#ifndef SDL_INLINE_OKAY +#define __inline__ +#endif + +/* Apparently this is needed by several Windows compilers */ +#if !defined(__MACH__) +#ifndef NULL +#ifdef __cplusplus +#define NULL 0 +#else +#define NULL ((void *)0) +#endif +#endif /* NULL */ +#endif /* ! Mac OS X - breaks precompiled headers */ diff --git a/src/win32/dependencies/sdl/close_code.h b/src/win32/dependencies/sdl/close_code.h new file mode 100644 index 00000000..afbb6504 --- /dev/null +++ b/src/win32/dependencies/sdl/close_code.h @@ -0,0 +1,41 @@ +/* + SDL - Simple DirectMedia Layer + Copyright (C) 1997-2004 Sam Lantinga + + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Library General Public + License as published by the Free Software Foundation; either + version 2 of the License, or (at your option) any later version. + + This library 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 + Library General Public License for more details. + + You should have received a copy of the GNU Library General Public + License along with this library; if not, write to the Free + Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + + Sam Lantinga + slouken@libsdl.org +*/ + +/* This file reverses the effects of begin_code.h and should be included + after you finish any function and structure declarations in your headers +*/ + +#undef _begin_code_h + +/* Reset structure packing at previous byte alignment */ +#if defined(_MSC_VER) || defined(__MWERKS__) || defined(__WATCOMC__) || defined(__BORLANDC__) +#ifdef __BORLANDC__ +#pragma nopackwarning +#endif +#if (defined(__MWERKS__) && defined(__MACOS__)) +#pragma options align=reset +#pragma enumsalwaysint reset +#else +#pragma pack(pop) +#endif +#endif /* Compiler needs structure packing set */ + diff --git a/src/win32/dependencies/zlib/.cvsignore b/src/win32/dependencies/zlib/.cvsignore new file mode 100644 index 00000000..c4fc3bb4 --- /dev/null +++ b/src/win32/dependencies/zlib/.cvsignore @@ -0,0 +1,3 @@ +zlib.vcproj.A2.Spacy.user +debug +release \ No newline at end of file diff --git a/src/win32/dependencies/zlib/CVS/Entries b/src/win32/dependencies/zlib/CVS/Entries new file mode 100644 index 00000000..e17dbf2b --- /dev/null +++ b/src/win32/dependencies/zlib/CVS/Entries @@ -0,0 +1,25 @@ +/.cvsignore/1.1/Sat May 13 16:42:49 2006// +/adler32.c/1.1/Fri May 12 21:27:36 2006// +/compress.c/1.1/Fri May 12 21:27:36 2006// +/crc32.c/1.1/Fri May 12 21:27:36 2006// +/crc32.h/1.1/Fri May 12 21:27:36 2006// +/deflate.c/1.1/Fri May 12 21:27:36 2006// +/deflate.h/1.1/Fri May 12 21:27:36 2006// +/gzio.c/1.1/Fri May 12 21:27:36 2006// +/infback.c/1.1/Fri May 12 21:27:36 2006// +/inffast.c/1.1/Fri May 12 21:27:37 2006// +/inffast.h/1.1/Fri May 12 21:27:37 2006// +/inffixed.h/1.1/Fri May 12 21:27:37 2006// +/inflate.c/1.1/Fri May 12 21:27:37 2006// +/inflate.h/1.1/Fri May 12 21:27:37 2006// +/inftrees.c/1.1/Fri May 12 21:27:37 2006// +/inftrees.h/1.1/Fri May 12 21:27:37 2006// +/trees.c/1.1/Fri May 12 21:27:37 2006// +/trees.h/1.1/Fri May 12 21:27:37 2006// +/uncompr.c/1.1/Fri May 12 21:27:37 2006// +/zconf.h/1.1/Fri May 12 21:27:37 2006// +/zlib.h/1.1/Fri May 12 21:27:37 2006// +/zlib.vcproj/1.5/Wed Aug 23 22:13:30 2006// +/zutil.c/1.1/Fri May 12 21:27:37 2006// +/zutil.h/1.1/Fri May 12 21:27:37 2006// +D diff --git a/src/win32/dependencies/zlib/CVS/Repository b/src/win32/dependencies/zlib/CVS/Repository new file mode 100644 index 00000000..1764d38b --- /dev/null +++ b/src/win32/dependencies/zlib/CVS/Repository @@ -0,0 +1 @@ +VisualBoyAdvance/win32/dependencies/zlib diff --git a/src/win32/dependencies/zlib/CVS/Root b/src/win32/dependencies/zlib/CVS/Root new file mode 100644 index 00000000..6ceab0dd --- /dev/null +++ b/src/win32/dependencies/zlib/CVS/Root @@ -0,0 +1 @@ +:pserver:anonymous@vba.cvs.sourceforge.net:/cvsroot/vba diff --git a/zlib/adler32.c b/src/win32/dependencies/zlib/adler32.c similarity index 98% rename from zlib/adler32.c rename to src/win32/dependencies/zlib/adler32.c index 007ba262..43594716 100644 --- a/zlib/adler32.c +++ b/src/win32/dependencies/zlib/adler32.c @@ -3,7 +3,7 @@ * For conditions of distribution and use, see copyright notice in zlib.h */ -/* @(#) $Id$ */ +/* @(#) $Id: adler32.c,v 1.1 2006/05/12 21:27:36 spacy51 Exp $ */ #define ZLIB_INTERNAL #include "zlib.h" diff --git a/zlib/compress.c b/src/win32/dependencies/zlib/compress.c similarity index 97% rename from zlib/compress.c rename to src/win32/dependencies/zlib/compress.c index df04f014..65517162 100644 --- a/zlib/compress.c +++ b/src/win32/dependencies/zlib/compress.c @@ -3,7 +3,7 @@ * For conditions of distribution and use, see copyright notice in zlib.h */ -/* @(#) $Id$ */ +/* @(#) $Id: compress.c,v 1.1 2006/05/12 21:27:36 spacy51 Exp $ */ #define ZLIB_INTERNAL #include "zlib.h" diff --git a/zlib/crc32.c b/src/win32/dependencies/zlib/crc32.c similarity index 99% rename from zlib/crc32.c rename to src/win32/dependencies/zlib/crc32.c index f658a9ef..39219115 100644 --- a/zlib/crc32.c +++ b/src/win32/dependencies/zlib/crc32.c @@ -9,7 +9,7 @@ * factor of two increase in speed on a Power PC G4 (PPC7455) using gcc -O3. */ -/* @(#) $Id$ */ +/* @(#) $Id: crc32.c,v 1.1 2006/05/12 21:27:36 spacy51 Exp $ */ /* Note on the use of DYNAMIC_CRC_TABLE: there is no mutex or semaphore diff --git a/zlib/crc32.h b/src/win32/dependencies/zlib/crc32.h similarity index 100% rename from zlib/crc32.h rename to src/win32/dependencies/zlib/crc32.h diff --git a/zlib/deflate.c b/src/win32/dependencies/zlib/deflate.c similarity index 99% rename from zlib/deflate.c rename to src/win32/dependencies/zlib/deflate.c index 29ce1f64..cd0fe2eb 100644 --- a/zlib/deflate.c +++ b/src/win32/dependencies/zlib/deflate.c @@ -47,7 +47,7 @@ * */ -/* @(#) $Id$ */ +/* @(#) $Id: deflate.c,v 1.1 2006/05/12 21:27:36 spacy51 Exp $ */ #include "deflate.h" diff --git a/zlib/deflate.h b/src/win32/dependencies/zlib/deflate.h similarity index 99% rename from zlib/deflate.h rename to src/win32/dependencies/zlib/deflate.h index 05a5ab3a..cd7355d1 100644 --- a/zlib/deflate.h +++ b/src/win32/dependencies/zlib/deflate.h @@ -8,7 +8,7 @@ subject to change. Applications should only use zlib.h. */ -/* @(#) $Id$ */ +/* @(#) $Id: deflate.h,v 1.1 2006/05/12 21:27:36 spacy51 Exp $ */ #ifndef DEFLATE_H #define DEFLATE_H diff --git a/zlib/gzio.c b/src/win32/dependencies/zlib/gzio.c similarity index 99% rename from zlib/gzio.c rename to src/win32/dependencies/zlib/gzio.c index 7e90f492..40cd6af6 100644 --- a/zlib/gzio.c +++ b/src/win32/dependencies/zlib/gzio.c @@ -5,7 +5,7 @@ * Compile this file with -DNO_GZCOMPRESS to avoid the compression code. */ -/* @(#) $Id$ */ +/* @(#) $Id: gzio.c,v 1.1 2006/05/12 21:27:36 spacy51 Exp $ */ #include diff --git a/zlib/infback.c b/src/win32/dependencies/zlib/infback.c similarity index 100% rename from zlib/infback.c rename to src/win32/dependencies/zlib/infback.c diff --git a/zlib/inffast.c b/src/win32/dependencies/zlib/inffast.c similarity index 100% rename from zlib/inffast.c rename to src/win32/dependencies/zlib/inffast.c diff --git a/zlib/inffast.h b/src/win32/dependencies/zlib/inffast.h similarity index 100% rename from zlib/inffast.h rename to src/win32/dependencies/zlib/inffast.h diff --git a/zlib/inffixed.h b/src/win32/dependencies/zlib/inffixed.h similarity index 100% rename from zlib/inffixed.h rename to src/win32/dependencies/zlib/inffixed.h diff --git a/zlib/inflate.c b/src/win32/dependencies/zlib/inflate.c similarity index 100% rename from zlib/inflate.c rename to src/win32/dependencies/zlib/inflate.c diff --git a/zlib/inflate.h b/src/win32/dependencies/zlib/inflate.h similarity index 100% rename from zlib/inflate.h rename to src/win32/dependencies/zlib/inflate.h diff --git a/zlib/inftrees.c b/src/win32/dependencies/zlib/inftrees.c similarity index 100% rename from zlib/inftrees.c rename to src/win32/dependencies/zlib/inftrees.c diff --git a/zlib/inftrees.h b/src/win32/dependencies/zlib/inftrees.h similarity index 100% rename from zlib/inftrees.h rename to src/win32/dependencies/zlib/inftrees.h diff --git a/zlib/trees.c b/src/win32/dependencies/zlib/trees.c similarity index 99% rename from zlib/trees.c rename to src/win32/dependencies/zlib/trees.c index 395e4e16..d5dd0020 100644 --- a/zlib/trees.c +++ b/src/win32/dependencies/zlib/trees.c @@ -29,7 +29,7 @@ * Addison-Wesley, 1983. ISBN 0-201-06672-6. */ -/* @(#) $Id$ */ +/* @(#) $Id: trees.c,v 1.1 2006/05/12 21:27:37 spacy51 Exp $ */ /* #define GEN_TREES_H */ diff --git a/zlib/trees.h b/src/win32/dependencies/zlib/trees.h similarity index 100% rename from zlib/trees.h rename to src/win32/dependencies/zlib/trees.h diff --git a/zlib/uncompr.c b/src/win32/dependencies/zlib/uncompr.c similarity index 96% rename from zlib/uncompr.c rename to src/win32/dependencies/zlib/uncompr.c index b59e3d0d..8f760b34 100644 --- a/zlib/uncompr.c +++ b/src/win32/dependencies/zlib/uncompr.c @@ -3,7 +3,7 @@ * For conditions of distribution and use, see copyright notice in zlib.h */ -/* @(#) $Id$ */ +/* @(#) $Id: uncompr.c,v 1.1 2006/05/12 21:27:37 spacy51 Exp $ */ #define ZLIB_INTERNAL #include "zlib.h" diff --git a/zlib/zconf.h b/src/win32/dependencies/zlib/zconf.h similarity index 99% rename from zlib/zconf.h rename to src/win32/dependencies/zlib/zconf.h index 03a9431c..73697bdd 100644 --- a/zlib/zconf.h +++ b/src/win32/dependencies/zlib/zconf.h @@ -3,7 +3,7 @@ * For conditions of distribution and use, see copyright notice in zlib.h */ -/* @(#) $Id$ */ +/* @(#) $Id: zconf.h,v 1.1 2006/05/12 21:27:37 spacy51 Exp $ */ #ifndef ZCONF_H #define ZCONF_H diff --git a/zlib/zlib.h b/src/win32/dependencies/zlib/zlib.h similarity index 100% rename from zlib/zlib.h rename to src/win32/dependencies/zlib/zlib.h diff --git a/src/win32/dependencies/zlib/zlib.vcproj b/src/win32/dependencies/zlib/zlib.vcproj new file mode 100644 index 00000000..65375f87 --- /dev/null +++ b/src/win32/dependencies/zlib/zlib.vcproj @@ -0,0 +1,255 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/zlib/zutil.c b/src/win32/dependencies/zlib/zutil.c similarity index 99% rename from zlib/zutil.c rename to src/win32/dependencies/zlib/zutil.c index d55f5948..4b64e6f8 100644 --- a/zlib/zutil.c +++ b/src/win32/dependencies/zlib/zutil.c @@ -3,7 +3,7 @@ * For conditions of distribution and use, see copyright notice in zlib.h */ -/* @(#) $Id$ */ +/* @(#) $Id: zutil.c,v 1.1 2006/05/12 21:27:37 spacy51 Exp $ */ #include "zutil.h" diff --git a/zlib/zutil.h b/src/win32/dependencies/zlib/zutil.h similarity index 99% rename from zlib/zutil.h rename to src/win32/dependencies/zlib/zutil.h index b7d5eff8..07168ba1 100644 --- a/zlib/zutil.h +++ b/src/win32/dependencies/zlib/zutil.h @@ -8,7 +8,7 @@ subject to change. Applications should only use zlib.h. */ -/* @(#) $Id$ */ +/* @(#) $Id: zutil.h,v 1.1 2006/05/12 21:27:37 spacy51 Exp $ */ #ifndef ZUTIL_H #define ZUTIL_H diff --git a/src/win32/display.cpp b/src/win32/display.cpp new file mode 100644 index 00000000..bd6ffa3f --- /dev/null +++ b/src/win32/display.cpp @@ -0,0 +1,123 @@ +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#include "stdafx.h" +#include "Display.h" + +void copyImage( void *source, void *destination, unsigned int width, unsigned int height, unsigned int destinationPitch, unsigned int colorDepth ) +{ +#ifdef ASM + + // Copy the image at [source] to the locked Direct3D texture + __asm + { + mov eax, width ; Initialize + mov ebx, height ; + mov edi, destination ; + mov edx, destinationPitch ; + + cmp colorDepth, 16 ; Check colorDepth==16bit + jnz gbaOtherColor ; + sub edx, eax ; + sub edx, eax ; + mov esi, source ; + lea esi,[esi+2*eax+4] ; + shr eax, 1 ; +gbaLoop16bit: + mov ecx, eax ; + rep movsd ; + add esi, 4 ; + add edi, edx ; + dec ebx ; + jnz gbaLoop16bit ; + jmp gbaLoopEnd ; +gbaOtherColor: + cmp colorDepth, 32 ; Check colorDepth==32bit + jnz gbaOtherColor2 ; + + lea esi, [eax*4] ; + sub edx, esi ; + mov esi, source ; + lea esi, [esi+4*eax+4] ; +gbaLoop32bit: + mov ecx, eax ; + rep movsd ; ECX times: Move DWORD at [ESI] to [EDI] | ESI++ EDI++ + add esi, 4 ; + add edi, edx ; + dec ebx ; + jnz gbaLoop32bit ; + jmp gbaLoopEnd ; +gbaOtherColor2: + lea eax, [eax+2*eax] ; Works like colorDepth==24bit + sub edx, eax ; +gbaLoop24bit: + mov ecx, eax ; + shr ecx, 2 ; + rep movsd ; + add edi, edx ; + dec ebx ; + jnz gbaLoop24bit ; +gbaLoopEnd: + } + +#else // #ifdef ASM + + // optimized C version + register unsigned int lineSize; + register unsigned char *src, *dst; + switch(colorDepth) + { + case 16: + lineSize = width<<1; + src = ((unsigned char*)source) + lineSize + 4; + dst = (unsigned char*)destination; + do { + MoveMemory( dst, src, lineSize ); + src+=lineSize; + dst+=lineSize; + src += 2; + dst += (destinationPitch - lineSize); + } while ( --height); + break; + case 32: + lineSize = width<<2; + src = ((unsigned char*)source) + lineSize + 4; + dst = (unsigned char*)destination; + do { + MoveMemory( dst, src, lineSize ); + src+=lineSize; + dst+=lineSize; + src += 4; + dst += (destinationPitch - lineSize); + } while ( --height); + break; + } + + // very compatible but slow C version + //unsigned int nBytesPerPixel = colorDepth>>3; + //unsigned int i, x, y, srcPitch = (width+1) * nBytesPerPixel; + //unsigned char * src = ((unsigned char*)source)+srcPitch; + //unsigned char * dst = (unsigned char*)destination; + //for (y=0;yX= zoq6+q^LF%hrF|j|(kMrc91+#~4I&@WS4=q*5E*5zqr>+I$7kCNF z27C^F8GZpi4nGL*!1v3Vbi;ST2jSb{i}364dH7ZM415|s4ZjFK4L=M&0N({~;0?Ti zH}D4D!0WNbah>B5N1P*U`6B!}d>(!kJ_E1O)6k31)3Wrz8R!A%F5+*%=iry&7vSUY zgYXV~KYTaQJK=-y?eN51S%II2AAt|S2jK<&+3b+D4fq`VGW-I39DWeqf$xX!CYzmP zGe|buMSlMNs+e{7Jp3wr20jg+hF^rAh98C>fbY_0h4=>Ez#Di2Z{XR`8XLOK?YhM6 zigUZdVUZ&I<A9fbX10ihvnqSlQJ+cASX_o zkgl#SIez@O96fqeT3cIrBL0S2tSVs3t{by*cUweuPcuGkr33Z3@^oi+cl)W}a^2WN zS7xk4eeT0DcIlnemowr%jJe84(h7e{L%Z_1mAI*xyPFz`xp6BuHZt*K>-W-xy*j6f z((l`{z4a(s(W5Okg#7Kc_(o;d2X)nd%$;NZ{C~EC;61eJ`zXP?tEzwReN?@|OZ2?T zc$rlLG^(ZQ(HX2$`id$PZtILX6FM)_JUH(31e_s$ty5!TlOlnS$HM`~!S>bkX&*NJ z->iK9(D=R`!9R5|tPik0!vW6;2Rtt1}C+<|^zRJG(obL4$1y4oJ-xFy{=%e=%()IUJ(vdZL5YLu(Mf zo=7xeR8_Ns!+PSdo;a*0_+h-D;5w`)s(%>&bziAsw3u+8+e?j8fjY$ZQCQVzpQ)_e zsSmjmi$?t@O8yzM-qKd-F-OT7F;@u@8%h07YJ?e$QcG3*cD-LzRZ1(??XBM`Dpx7G z_WfFm>H1>1TvUH@X8*v9mCMv`+HXB7+x5v0-)&2_eVfV5B-Q^%IaXGGjO8y{AA`WJ zU(@&dRh?hj={i*FzfdEMuM2Da%+)H^3ZVYI3)FgT_-GV=DwXn6?beFe(I_sF&AGV> zp&Qzhdgtcm@B=8Z=dpuNtao!$uh|5$U9CB*{r23Z@2%|W8DP6^v;SkGzG}dF#dXSG DsvOIK literal 0 HcmV?d00001 diff --git a/src/win32/resource.h b/src/win32/resource.h new file mode 100644 index 00000000..681be17a --- /dev/null +++ b/src/win32/resource.h @@ -0,0 +1,807 @@ +//{{NO_DEPENDENCIES}} +// Microsoft Visual C++ generated include file. +// Used by VBA.rc +// +#define IDS_UNSUPPORTED_VBA_SGM 1 +#define IDS_CANNOT_LOAD_SGM 2 +#define IDS_SAVE_GAME_NOT_USING_BIOS 3 +#define IDC_DEFAULTS 3 +#define IDS_SAVE_GAME_USING_BIOS 4 +#define IDS_UNSUPPORTED_SAVE_TYPE 5 +#define IDS_CANNOT_OPEN_FILE 6 +#define IDS_BAD_ZIP_FILE 7 +#define IDS_NO_IMAGE_ON_ZIP 8 +#define IDS_ERROR_OPENING_IMAGE 9 +#define IDS_ERROR_READING_IMAGE 10 +#define IDS_UNSUPPORTED_BIOS_FUNCTION 11 +#define IDS_INVALID_BIOS_FILE_SIZE 12 +#define IDS_INVALID_CHEAT_CODE 13 +#define IDS_UNKNOWN_ARM_OPCDOE 14 +#define IDS_UNKNOWN_THUMB_OPCODE 15 +#define IDS_ERROR_CREATING_FILE 16 +#define IDS_FAILED_TO_READ_SGM 17 +#define IDS_FAILED_TO_READ_RTC 18 +#define IDS_UNSUPPORTED_VB_SGM 19 +#define IDS_CANNOT_LOAD_SGM_FOR 20 +#define IDS_ERROR_OPENING_IMAGE_FROM 21 +#define IDS_ERROR_READING_IMAGE_FROM 22 +#define IDS_UNSUPPORTED_ROM_SIZE 23 +#define IDS_UNSUPPORTED_RAM_SIZE 24 +#define IDS_UNKNOWN_CARTRIDGE_TYPE 25 +#define IDS_MAXIMUM_NUMBER_OF_CHEATS 26 +#define IDS_INVALID_GAMESHARK_CODE 27 +#define IDS_INVALID_GAMEGENIE_CODE 28 +#define IDS_INVALID_CHEAT_TO_REMOVE 29 +#define IDS_INVALID_CHEAT_CODE_ADDRESS 30 +#define IDS_UNSUPPORTED_CHEAT_LIST_VERSION 31 +#define IDS_UNSUPPORTED_CHEAT_LIST_TYPE 32 +#define IDS_INVALID_GSA_CODE 33 +#define IDS_CANNOT_IMPORT_SNAPSHOT_FOR 34 +#define IDS_UNSUPPORTED_SNAPSHOT_FILE 35 +#define IDS_UNSUPPORTED_ARM_MODE 36 +#define IDS_UNSUPPORTED_CODE_FILE 37 +#define IDS_GSA_CODE_WARNING 38 +#define IDS_INVALID_CBA_CODE 39 +#define IDS_CBA_CODE_WARNING 40 +#define IDS_OUT_OF_MEMORY 41 +#define IDS_WRONG_GAMESHARK_CODE 42 +#define IDS_UNSUPPORTED_GAMESHARK_CODE 43 +#define IDD_REGISTERS 102 +#define IDD_DEBUG 103 +#define IDR_MENU 104 +#define IDD_ABOUT 105 +#define IDR_ACCELERATOR 106 +#define IDD_CHEATS 107 +#define IDD_ADD_CHEAT 108 +#define IDD_DIRECTORIES 109 +#define IDD_CONFIG 110 +#define IDD_GS 111 +#define IDD_GG 112 +#define IDD_CHEAT_LIST 113 +#define IDD_ASSOCIATIONS 114 +#define IDR_GB_PRINTER 115 +#define IDD_GBA_ROM_INFO 116 +#define IDD_GB_ROM_INFO 117 +#define IDD_GB_CHEAT_LIST 118 +#define IDD_ADD_CHEAT_DLG 119 +#define IDD_GB_PRINTER 120 +#define IDD_MOTION_CONFIG 121 +#define IDD_LANG_SELECT 122 +#define IDD_CODE_SELECT 123 +#define IDD_OPENDLG 124 +#define IDD_MAP_VIEW 126 +#define IDD_PALETTE_VIEW 127 +#define IDD_MEM_VIEWER 128 +#define IDD_OAM_VIEW 130 +#define IDD_ACCEL_EDITOR 131 +#define IDD_TILE_VIEWER 132 +#define IDD_GB_COLORS 133 +#define IDD_DISASSEMBLE 134 +#define IDD_GDB_PORT 135 +#define IDD_GDB_WAITING 136 +#define IDD_LOGGING 137 +#define IDD_EXPORT_SPS 138 +#define IDD_ADDR_SIZE 139 +#define IDD_MODES 140 +#define IDD_DRIVERS 142 +#define IDD_THROTTLE 143 +#define IDD_GB_DISASSEMBLE 144 +#define IDD_GB_OAM_VIEW 145 +#define IDD_GB_TILE_VIEWER 146 +#define IDD_GB_MAP_VIEW 147 +#define IDD_GB_PALETTE_VIEW 148 +#define IDD_MODE_CONFIRM 149 +#define IDD_REWIND_INTERVAL 150 +#define IDD_IO_VIEWER 151 +#define IDD_MAX_SCALE 154 +#define IDD_BUG_REPORT 155 +#define IDD_GAME_OVERRIDES 156 +#define IDC_R0 1000 +#define IDC_EDIT_UP 1000 +#define IDC_R1 1001 +#define IDC_EDIT_DOWN 1001 +#define IDC_R2 1002 +#define IDC_EDIT_LEFT 1002 +#define IDC_R3 1003 +#define IDC_EDIT_RIGHT 1003 +#define IDC_R4 1004 +#define IDC_EDIT_BUTTON_A 1004 +#define IDC_R5 1005 +#define IDC_EDIT_BUTTON_B 1005 +#define IDC_R6 1006 +#define IDC_EDIT_BUTTON_SELECT 1006 +#define IDC_R7 1007 +#define IDC_EDIT_BUTTON_START 1007 +#define IDC_R8 1008 +#define ID_OK 1008 +#define IDC_R9 1009 +#define ID_CANCEL 1009 +#define ID_SAVE 1009 +#define IDC_R10 1010 +#define IDC_EDIT_SPEED 1010 +#define IDC_R11 1011 +#define IDC_EDIT_CAPTURE 1011 +#define IDC_R12 1012 +#define IDC_EDIT_BUTTON_L 1012 +#define IDC_R13 1013 +#define IDC_EDIT_BUTTON_GS 1013 +#define IDC_R14 1014 +#define IDC_EDIT_BUTTON_R 1014 +#define IDC_R15 1015 +#define IDC_R16 1016 +#define IDC_R17 1017 +#define IDC_N_FLAG 1018 +#define IDC_ROM_DIR 1018 +#define IDC_Z_FLAG 1019 +#define IDC_NEXT 1019 +#define IDC_BATTERY_DIR 1019 +#define IDC_C_FLAG 1020 +#define IDC_CONTINUE 1020 +#define IDC_SAVE_DIR 1020 +#define IDC_V_FLAG 1021 +#define IDC_CAPTURE_DIR 1021 +#define IDC_CHEAT_LIST 1021 +#define IDC_IRQ 1022 +#define IDC_ROM_PATH 1022 +#define IDC_START 1022 +#define IDC_T_FLAG 1023 +#define IDC_BATTERY_PATH 1023 +#define IDC_SEARCH 1023 +#define IDS_DIRECTX_7_REQUIRED 1024 +#define IDC_SAVE_PATH 1024 +#define IDC_ADD_CHEAT 1024 +#define IDC_M4 1025 +#define IDC_CAPTURE_PATH 1025 +#define IDC_OLD_VALUE 1025 +#define IDC_ADD_GS_CHEAT 1025 +#define IDS_DISABLING_VIDEO_MEMORY 1025 +#define IDC_ADD_GAMESHARK 1025 +#define IDC_M3 1026 +#define IDC_SPECIFIC_VALUE 1026 +#define IDS_SETTING_WILL_BE_EFFECTIVE 1026 +#define IDC_GBROM_DIR 1026 +#define IDC_M2 1027 +#define IDS_DISABLING_EMULATION_ONLY 1027 +#define IDC_GBROM_PATH 1027 +#define IDC_M1 1028 +#define IDC_SIZE_8 1028 +#define IDS_FAILED_TO_OPEN_FILE 1028 +#define IDC_ROM_DIR_RESET 1028 +#define IDC_M0 1029 +#define IDC_SIZE_16 1029 +#define IDS_FAILED_TO_READ_ZIP_DIR 1029 +#define IDC_GBROM_DIR_RESET 1029 +#define IDC_SIZE_32 1030 +#define IDS_UNSUPPORTED_FILE_TYPE 1030 +#define IDC_BATTERY_DIR_RESET 1030 +#define IDC_EQ 1031 +#define IDS_CANNOT_CREATE_DIRECTSOUND 1031 +#define IDC_SAVE_DIR_RESET 1031 +#define IDC_NE 1032 +#define IDS_CANNOT_SETCOOPERATIVELEVEL 1032 +#define IDC_CAPTURE_DIR_RESET 1032 +#define IDC_LT 1033 +#define IDS_CANNOT_CREATESOUNDBUFFER 1033 +#define IDC_LE 1034 +#define IDS_CANNOT_SETFORMAT_PRIMARY 1034 +#define IDC_GT 1035 +#define IDS_CANNOT_CREATESOUNDBUFFER_SEC 1035 +#define IDC_GE 1036 +#define IDS_CANNOT_PLAY_PRIMARY 1036 +#define IDC_SIGNED 1037 +#define IDS_SEARCH_PRODUCED_TOO_MANY 1037 +#define IDC_UNSIGNED 1038 +#define IDS_NUMBER_CANNOT_BE_EMPTY 1038 +#define IDS_INVALID_ADDRESS 1039 +#define IDC_HEXADECIMAL 1040 +#define IDS_MISALIGNED_HALFWORD 1040 +#define IDC_VALUE 1041 +#define IDS_MISALIGNED_WORD 1041 +#define IDC_ADDRESS 1042 +#define IDS_VALUE_CANNOT_BE_EMPTY 1042 +#define IDS_ERROR_ON_STARTDOC 1043 +#define IDC_R 1043 +#define IDS_ERROR_ON_STARTPAGE 1044 +#define IDC_G 1044 +#define IDS_ERROR_PRINTING_ON_STRETCH 1045 +#define IDC_B 1045 +#define IDC_UPDATE 1046 +#define IDS_ERROR_ON_ENDPAGE 1046 +#define IDC_TILE_NUM 1046 +#define IDC_GGDESC 1047 +#define IDS_ERROR_ON_ENDDOC 1047 +#define IDC_FLIP 1047 +#define IDC_GGCODE 1048 +#define IDS_ERROR 1048 +#define IDC_PALETTE_NUM 1048 +#define IDC_GGADD 1049 +#define IDS_JOY_LEFT 1049 +#define IDC_GGDEL 1050 +#define IDS_JOY_RIGHT 1050 +#define IDC_GGLIST 1051 +#define IDS_JOY_UP 1051 +#define IDC_GGRES 1052 +#define IDS_JOY_DOWN 1052 +#define IDC_GGQUIT 1053 +#define IDS_JOY_BUTTON 1053 +#define IDC_GSDESC 1054 +#define IDS_SELECT_ROM_DIR 1054 +#define IDC_GSCODE 1055 +#define IDS_SELECT_BATTERY_DIR 1055 +#define IDC_GSADD 1056 +#define IDS_SELECT_SAVE_DIR 1056 +#define IDC_GSDEL 1057 +#define IDS_SELECT_CAPTURE_DIR 1057 +#define IDC_GSLIST 1058 +#define IDS_SELECT_BIOS_FILE 1058 +#define IDC_GSRES 1059 +#define IDS_RESET 1059 +#define IDC_GSQUIT 1060 +#define IDS_AUTOFIRE_A_DISABLED 1060 +#define IDC_FREEZE 1061 +#define IDS_AUTOFIRE_A 1061 +#define IDS_AUTOFIRE_B_DISABLED 1062 +#define IDS_AUTOFIRE_B 1063 +#define IDS_AUTOFIRE_L_DISABLED 1064 +#define IDS_AUTOFIRE_L 1065 +#define IDS_AUTOFIRE_R_DISABLED 1066 +#define IDC_REMOVE 1067 +#define IDS_AUTOFIRE_R 1067 +#define IDC_REMOVE_ALL 1068 +#define IDS_SELECT_ROM 1068 +#define IDS_SELECT_SAVE_GAME_NAME 1069 +#define IDC_ENABLE 1070 +#define IDS_LOADED_STATE 1070 +#define IDS_LOADED_STATE_N 1071 +#define IDS_WROTE_STATE 1072 +#define IDS_WROTE_STATE_N 1073 +#define IDC_RESTORE 1074 +#define IDS_LOADED_BATTERY 1074 +#define IDC_GBA 1075 +#define IDS_SELECT_CAPTURE_NAME 1075 +#define IDC_AGB 1076 +#define IDS_SCREEN_CAPTURE 1076 +#define IDC_BIN 1077 +#define IDS_ADDRESS 1077 +#define IDC_GB 1078 +#define IDS_OLD_VALUE 1078 +#define IDC_SGB 1079 +#define IDC_ROM_TITLE 1079 +#define IDS_NEW_VALUE 1079 +#define IDC_CGB 1080 +#define IDC_ROM_GAME_CODE 1080 +#define IDS_ADD_CHEAT_CODE 1080 +#define IDC_GBC 1081 +#define IDC_ROM_MAKER_CODE 1081 +#define IDS_CODE 1081 +#define IDC_ROM_UNIT_CODE 1082 +#define IDS_DESCRIPTION 1082 +#define IDC_ROM_DEVICE_TYPE 1083 +#define IDS_STATUS 1083 +#define IDC_ROM_VERSION 1084 +#define IDS_ADD_GG_CODE 1084 +#define IDC_ROM_CRC 1085 +#define IDS_ADD_GS_CODE 1085 +#define IDC_ROM_COLOR 1086 +#define IDC_CODE 1086 +#define IDS_POCKET_PRINTER 1086 +#define IDC_ROM_MAKER_NAME 1086 +#define IDC_ROM_SIZE 1087 +#define IDC_DESC 1087 +#define IDS_UNKNOWN 1087 +#define IDC_ROM_RAM_SIZE 1088 +#define IDC_ADD_GG_CHEAT 1088 +#define IDS_NONE 1088 +#define IDC_ROM_DEST_CODE 1089 +#define IDC_GB_PRINTER 1089 +#define IDS_FAILED_TO_LOAD_LIBRARY 1089 +#define IDC_ROM_LIC_CODE 1090 +#define IDC_1X 1090 +#define IDS_FAILED_TO_GET_LOCINFO 1090 +#define IDC_ROM_CHECKSUM 1091 +#define IDC_2X 1091 +#define IDS_SELECT_CHEAT_LIST_NAME 1091 +#define IDC_3X 1092 +#define IDS_FILTER_BIOS 1092 +#define IDC_4X 1093 +#define IDS_FILTER_ROM 1093 +#define IDC_ROM_MAKER_NAME2 1093 +#define ID_PRINT 1094 +#define IDS_FILTER_SGM 1094 +#define IDC_ADD_GSA 1095 +#define IDC_ADD_CODE 1095 +#define IDS_FILTER_CHEAT_LIST 1095 +#define IDC_TRANSLATION_BY 1096 +#define IDS_FILTER_PNG 1096 +#define IDC_LANG_STRING 1097 +#define IDS_LOADED_CHEATS 1097 +#define IDC_LANG_NAME 1098 +#define IDS_ERROR_DISP_COLOR 1098 +#define IDS_ADD_GSA_CODE 1099 +#define IDC_GAME_LIST 1099 +#define IDS_FILTER_SPS 1100 +#define IDS_SELECT_SNAPSHOT_FILE 1101 +#define IDC_ADD_CODEBREAKER 1101 +#define IDS_FILTER_SAV 1102 +#define IDS_SELECT_BATTERY_FILE 1103 +#define IDS_FILTER_GBS 1104 +#define IDS_FILTER_GCF 1105 +#define IDS_SELECT_CODE_FILE 1106 +#define IDS_SAVE_WILL_BE_LOST 1107 +#define IDS_CONFIRM_ACTION 1108 +#define IDS_CODES_WILL_BE_LOST 1109 +#define IDS_FILTER_SPC 1110 +#define IDS_ADD_CBA_CODE 1111 +#define IDS_FILTER_WAV 1112 +#define IDS_SELECT_WAV_NAME 1113 +#define IDC_FRAME_0 1113 +#define IDS_FILTER_GBROM 1114 +#define IDC_FRAME_1 1114 +#define IDC_BG0 1115 +#define IDS_FILTER_PAL 1115 +#define IDC_BG1 1116 +#define IDS_SELECT_PALETTE_NAME 1116 +#define IDC_BG2 1117 +#define IDS_SEARCH_PRODUCED_NO_RESULTS 1117 +#define IDC_BG3 1118 +#define IDS_ERROR_BINDING 1118 +#define IDS_ERROR_LISTENING 1119 +#define IDS_ERROR_CREATING_SOCKET 1120 +#define IDS_ACK_NOT_RECEIVED 1121 +#define IDS_ERROR_NOT_GBA_IMAGE 1122 +#define IDS_EEPROM_NOT_SUPPORTED 1123 +#define IDC_MAP_VIEW 1124 +#define IDS_FILTER_DUMP 1124 +#define IDC_PALETTE_VIEW 1125 +#define IDS_SELECT_DUMP_FILE 1125 +#define IDC_PALETTE_VIEW_OBJ 1126 +#define IDC_REFRESH 1126 +#define IDS_FILTER_AVI 1126 +#define IDC_SAVE 1127 +#define IDC_GOPC 1127 +#define IDS_SELECT_AVI_NAME 1127 +#define IDC_APPLY 1127 +#define IDS_INVALID_THROTTLE_VALUE 1128 +#define IDC_REFRESH2 1129 +#define IDS_FILTER_INI 1129 +#define IDS_SELECT_SKIN_FILE 1130 +#define IDC_CLOSE 1131 +#define IDS_FILTER_VMV 1131 +#define IDS_SELECT_MOVIE_NAME 1132 +#define IDS_BUG_REPORT 1133 +#define IDS_UNSUPPORTED_MOVIE_VERSION 1134 +#define IDS_END_OF_MOVIE 1135 +#define IDC_COLOR 1136 +#define IDS_INVALID_INTERVAL_VALUE 1136 +#define IDC_SAVE_BG 1137 +#define IDS_REGISTRY 1137 +#define IDC_SAVE_OBJ 1138 +#define IDC_MAP_VIEW_ZOOM 1138 +#define IDS_MOVIE_PLAY 1138 +#define IDC_VIEWER 1140 +#define IDC_ADDRESSES 1141 +#define IDC_GO 1143 +#define IDC_8_BIT 1144 +#define IDC_16_BIT 1145 +#define IDC_32_BIT 1146 +#define IDC_OAM_VIEW 1147 +#define IDC_OAM_VIEW_ZOOM 1148 +#define IDC_SPRITE 1150 +#define IDC_POS 1151 +#define IDC_MODE 1152 +#define IDC_COLORS 1153 +#define IDC_MAPBASE 1153 +#define IDC_PALETTE 1154 +#define IDC_CHARBASE 1154 +#define IDC_TILE 1155 +#define IDC_DIM 1155 +#define IDC_PRIO 1156 +#define IDC_NUMCOLORS 1156 +#define IDC_SCROLLBAR 1157 +#define IDC_PRIORITY 1157 +#define IDC_MOSAIC 1158 +#define IDC_SIZE2 1159 +#define IDC_OVERFLOW 1159 +#define IDC_ROT 1160 +#define IDC_FLAGS 1161 +#define IDC_COMMANDS 1162 +#define IDC_BANK 1162 +#define IDC_CURRENTS 1163 +#define IDC_ASSIGN 1164 +#define IDC_RESET 1165 +#define IDC_EDIT_KEY 1166 +#define IDC_ALREADY_AFFECTED 1167 +#define IDC_TILE_VIEW 1168 +#define IDC_16_COLORS 1169 +#define IDC_256_COLORS 1170 +#define IDC_CHARBASE_0 1173 +#define IDC_CHARBASE_1 1174 +#define IDC_CHARBASE_2 1175 +#define IDC_CHARBASE_3 1176 +#define IDC_PALETTE_SLIDER 1177 +#define IDC_CHARBASE_4 1178 +#define IDC_COLOR_BG0 1178 +#define IDC_COLOR_BG1 1179 +#define IDC_URL 1179 +#define IDC_COLOR_BG2 1180 +#define IDC_STRETCH 1180 +#define IDC_COLOR_BG3 1181 +#define IDC_COLOR_OB0 1182 +#define IDC_COLOR_OB1 1183 +#define IDC_COLOR_OB2 1184 +#define IDC_COLOR_OB3 1185 +#define IDC_TRANSLATOR_URL 1186 +#define IDC_STATIC1 1187 +#define IDC_STATIC2 1188 +#define IDC_STATIC3 1189 +#define IDC_STATIC4 1190 +#define IDC_DEFAULT 1191 +#define IDC_USER1 1192 +#define IDC_USER2 1193 +#define IDC_DISASSEMBLE 1196 +#define IDC_AUTOMATIC 1199 +#define IDC_ARM 1200 +#define IDC_THUMB 1201 +#define IDC_AUTO_UPDATE 1204 +#define IDC_N 1210 +#define IDC_Z 1211 +#define IDC_C 1212 +#define IDC_V 1213 +#define IDC_F 1214 +#define IDC_I 1215 +#define IDC_T 1216 +#define IDC_PORT 1217 +#define IDC_VSCROLL 1218 +#define IDC_VERSION 1219 +#define IDC_VERBOSE_SWI 1223 +#define IDC_VERBOSE_UNALIGNED_ACCESS 1224 +#define IDC_VERBOSE_ILLEGAL_WRITE 1225 +#define IDC_VERBOSE_ILLEGAL_READ 1226 +#define IDC_LOG 1227 +#define IDC_CLEAR 1228 +#define IDC_VERBOSE_DMA0 1229 +#define IDC_VERBOSE_DMA1 1230 +#define IDC_TILE_NUMBER 1230 +#define IDC_VERBOSE_DMA2 1231 +#define IDC_XY 1231 +#define IDC_VERBOSE_DMA3 1232 +#define IDC_VERBOSE_UNDEFINED 1233 +#define IDC_TITLE 1234 +#define IDC_VERBOSE_AGBPRINT 1234 +#define IDC_CURRENT_ADDRESS 1235 +#define IDC_NOTES 1236 +#define IDC_CURRENT_ADDRESS_LABEL 1236 +#define IDC_LOAD 1238 +#define IDC_SIZE_CONTROL 1240 +#define IDC_MODES 1240 +#define IDC_DRIVERS 1241 +#define IDC_THROTTLE 1242 +#define IDC_H 1243 +#define IDC_OAP 1244 +#define IDC_BANK_0 1245 +#define IDC_BANK_1 1246 +#define IDC_TIMER 1247 +#define IDC_INTERVAL 1248 +#define IDC_BIT_0 1250 +#define IDC_BIT_1 1251 +#define IDC_PREDEFINED 1251 +#define IDC_BIT_2 1252 +#define IDC_BUG_REPORT 1252 +#define IDC_BIT_3 1253 +#define IDC_COPY 1253 +#define IDC_BIT_4 1254 +#define IDC_NAME 1254 +#define IDC_BIT_5 1255 +#define IDC_RTC 1255 +#define IDC_BIT_6 1256 +#define IDC_SAVE_TYPE 1256 +#define IDC_BIT_7 1257 +#define IDC_FLASH_SIZE 1257 +#define IDC_BIT_8 1258 +#define IDC_COMMENT 1258 +#define IDC_BIT_9 1259 +#define IDC_BIT_10 1260 +#define IDC_BIT_11 1261 +#define IDC_BIT_12 1262 +#define IDC_BIT_13 1263 +#define IDC_BIT_14 1264 +#define IDC_BIT_15 1265 +#define IDC_MIRRORING 1266 +#define IDC_LY 1267 +#define IDC_APPENDMODE 1268 +#define ID_HELP_ABOUT 40001 +#define ID_FILE_EXIT 40002 +#define ID_OPTIONS_VIDEO_FRAMESKIP_0 40003 +#define ID_OPTIONS_VIDEO_FRAMESKIP_1 40004 +#define ID_OPTIONS_VIDEO_FRAMESKIP_2 40005 +#define ID_OPTIONS_VIDEO_FRAMESKIP_3 40006 +#define ID_OPTIONS_VIDEO_FRAMESKIP_4 40007 +#define ID_OPTIONS_VIDEO_FRAMESKIP_5 40008 +#define ID_OPTIONS_VIDEO_VSYNC 40009 +#define ID_OPTIONS_VIDEO_X1 40010 +#define ID_OPTIONS_VIDEO_X2 40011 +#define ID_OPTIONS_VIDEO_X3 40012 +#define ID_OPTIONS_VIDEO_X4 40013 +#define ID_FILE_PAUSE 40014 +#define ID_OPTIONS_EMULATOR_DIRECTORIES 40015 +#define ID_OPTIONS_JOYPAD 40016 +#define ID_OPTIONS_EMULATOR_SYNCHRONIZE 40017 +#define ID_FILE_RESET 40018 +#define ID_FILE_LOAD 40019 +#define ID_OPTIONS_SOUND_DIRECTSOUNDA 40020 +#define ID_OPTIONS_SOUND_DIRECTSOUNDB 40021 +#define ID_OPTIONS_SOUND_OFF 40022 +#define ID_OPTIONS_SOUND_MUTE 40023 +#define ID_OPTIONS_SOUND_ON 40024 +#define ID_OPTIONS_SOUND_CHANNEL1 40025 +#define ID_OPTIONS_SOUND_CHANNEL2 40026 +#define ID_OPTIONS_SOUND_CHANNEL3 40027 +#define ID_OPTIONS_SOUND_CHANNEL4 40028 +#define ID_OPTIONS_EMULATOR_USEBIOSFILE 40029 +#define ID_OPTIONS_EMULATOR_SELECTBIOSFILE 40030 +#define ID_CHEATS_SEARCHFORCHEATS 40031 +#define ID_CHEATS_ADDCHEAT 40032 +#define ID_OPTIONS_VIDEO_DISABLESFX 40033 +#define ID_OPTIONS_GAMEBOY_BORDER 40034 +#define ID_FILE_SAVEGAME_SLOT1 40035 +#define ID_FILE_SAVEGAME_SLOT2 40036 +#define ID_FILE_SAVEGAME_SLOT3 40037 +#define ID_FILE_SAVEGAME_SLOT4 40038 +#define ID_FILE_SAVEGAME_SLOT5 40039 +#define ID_FILE_SAVEGAME_SLOT6 40040 +#define ID_FILE_SAVEGAME_SLOT7 40041 +#define ID_FILE_SAVEGAME_SLOT8 40042 +#define ID_FILE_SAVEGAME_SLOT9 40043 +#define ID_FILE_SAVEGAME_SLOT10 40044 +#define ID_FILE_LOADGAME_SLOT1 40045 +#define ID_FILE_LOADGAME_SLOT2 40046 +#define ID_FILE_LOADGAME_SLOT3 40047 +#define ID_FILE_LOADGAME_SLOT4 40048 +#define ID_FILE_LOADGAME_SLOT5 40049 +#define ID_FILE_LOADGAME_SLOT6 40050 +#define ID_FILE_LOADGAME_SLOT7 40051 +#define ID_FILE_LOADGAME_SLOT8 40052 +#define ID_FILE_LOADGAME_SLOT9 40053 +#define ID_FILE_LOADGAME_SLOT10 40054 +#define ID_OPTIONS_GAMEBOY_AUTOMATIC 40057 +#define ID_OPTIONS_GAMEBOY_CGB 40058 +#define ID_OPTIONS_GAMEBOY_GBA 40059 +#define ID_OPTIONS_GAMEBOY_SGB 40060 +#define ID_OPTIONS_GAMEBOY_GB 40062 +#define ID_OPTIONS_GAMEBOY_REALCOLORS 40063 +#define ID_OPTIONS_GAMEBOY_GAMEBOYCOLORS 40064 +#define ID_CHEATS_GAMEBOY 40065 +#define ID_OPTIONS_SOUND_11KHZ 40067 +#define ID_OPTIONS_SOUND_22KHZ 40068 +#define ID_OPTIONS_SOUND_44KHZ 40069 +#define ID_OPTIONS_VIDEO_DDRAWEMULATIONONLY 40070 +#define ID_OPTIONS_VIDEO_DDRAWUSEVIDEOMEMORY 40071 +#define ID_OPTIONS_PRIORITY_HIGHEST 40072 +#define ID_OPTIONS_PRIORITY_ABOVENORMAL 40073 +#define ID_OPTIONS_PRIORITY_NORMAL 40074 +#define ID_OPTIONS_PRIORITY_BELOWNORMAL 40075 +#define ID_OPTIONS_VIDEO_FULLSCREEN320X240 40076 +#define ID_OPTIONS_VIDEO_FULLSCREEN640X480 40077 +#define ID_OPTIONS_FILTER_NORMAL 40078 +#define ID_OPTIONS_FILTER_2XSAI 40079 +#define ID_OPTIONS_FILTER_SUPER2XSAI 40081 +#define ID_OPTIONS_FILTER_SUPEREAGLE 40082 +#define ID_OPTIONS_FILTER_TVMODE 40083 +#define ID_CHEATS_CHEATLIST 40084 +#define ID_OPTIONS_JOYPAD_AUTOFIRE_A 40085 +#define ID_OPTIONS_JOYPAD_AUTOFIRE_B 40086 +#define ID_OPTIONS_JOYPAD_AUTOFIRE_L 40087 +#define ID_OPTIONS_JOYPAD_AUTOFIRE_R 40088 +#define ID_OPTIONS_VIDEO_FULLSCREENSTRETCHTOFIT 40089 +#define ID_OPTIONS_EMULATOR_ASSOCIATE 40091 +#define ID_OPTIONS_FILTER_DISABLEMMX 40093 +#define ID_OPTIONS_1X 40096 +#define ID_OPTIONS_2X 40097 +#define ID_OPTIONS_3X 40098 +#define ID_OPTIONS_4X 40099 +#define ID_FILE_ROMINFORMATION 40100 +#define ID_CHEATS_ADDCHEATCODE 40101 +#define ID_OPTIONS_EMULATOR_DISABLESTATUSMESSAGES 40102 +#define ID_OPTIONS_JOYPAD_MOTIONCONFIGURE 40103 +#define ID_FILE_SCREENCAPTURE 40104 +#define ID_OPTIONS_LANGUAGE_SYSTEM 40105 +#define ID_OPTIONS_LANGUAGE_ENGLISH 40106 +#define ID_OPTIONS_LANGUAGE_OTHER 40107 +#define ID_OPTIONS_GAMEBOY_PRINTER 40108 +#define ID_FILE_RECENT_RESET 40109 +#define ID_CHEATS_SAVECHEATLIST 40110 +#define ID_CHEATS_LOADCHEATLIST 40111 +#define ID_CHEATS_AUTOMATICSAVELOADCHEATS 40112 +#define ID_FILE_IMPORT_GAMESHARKSNAPSHOT 40115 +#define ID_FILE_IMPORT_BATTERYFILE 40116 +#define ID_FILE_IMPORT_GAMESHARKCODEFILE 40117 +#define ID_FILE_EXPORT_BATTERYFILE 40118 +#define ID_OPTIONS_FILTER16BIT_PIXELATEEXPERIMENTAL 40121 +#define ID_OPTIONS_FILTER16BIT_MOTIONBLUREXPERIMENTAL 40122 +#define ID_OPTIONS_EMULATOR_PAUSEWHENINACTIVE 40124 +#define ID_OPTIONS_SOUND_STARTRECORDING 40125 +#define ID_OPTIONS_SOUND_STOPRECORDING 40126 +#define ID_OPTIONS_VIDEO_LAYERS_BG0 40127 +#define ID_OPTIONS_VIDEO_LAYERS_BG1 40128 +#define ID_OPTIONS_VIDEO_LAYERS_BG2 40129 +#define ID_OPTIONS_VIDEO_LAYERS_BG3 40130 +#define ID_OPTIONS_VIDEO_LAYERS_OBJ 40131 +#define ID_OPTIONS_VIDEO_LAYERS_WIN0 40132 +#define ID_OPTIONS_VIDEO_LAYERS_WIN1 40133 +#define ID_OPTIONS_VIDEO_LAYERS_OBJWIN 40134 +#define ID_FILE_OPENGAMEBOY 40135 +#define ID_OPTIONS_SOUND_USEOLDSYNCHRONIZATION 40136 +#define ID_DEBUG_NEXTFRAME 40137 +#define ID_TOOLS_MAPVIEW 40138 +#define ID_TOOLS_PALETTEVIEW 40139 +#define ID_OPTIONS_EMULATOR_PNGFORMAT 40140 +#define ID_OPTIONS_EMULATOR_BMPFORMAT 40141 +#define ID_TOOLS_MEMORYVIEWER 40143 +#define ID_TOOLS_OAMVIEWER 40144 +#define ID_TOOLS_CUSTOMIZE 40145 +#define ID_TOOLS_TILEVIEWER 40146 +#define ID_OPTIONS_GAMEBOY_COLORS 40147 +#define ID_OPTIONS_SOUND_ECHO 40148 +#define ID_OPTIONS_SOUND_LOWPASSFILTER 40149 +#define ID_OPTIONS_SOUND_REVERSESTEREO 40150 +#define ID_TOOLS_DISASSEMBLE 40151 +#define ID_TOOLS_DEBUG_GDB 40152 +#define ID_TOOLS_DEBUG_LOADANDWAIT 40153 +#define ID_TOOLS_DEBUG_DISCONNECT 40154 +#define ID_TOOLS_DEBUG_BREAK 40155 +#define ID_TOOLS_LOGGING 40156 +#define ID_OPTIONS_EMULATOR_SPEEDHACK 40157 +#define ID_OPTIONS_EMULATOR_SPEEDUPTOGGLE 40158 +#define ID_OPTIONS_FILTER16BIT_ADVANCEMAMESCALE2X 40160 +#define ID_OPTIONS_FILTER16BIT_SIMPLE2X 40161 +#define ID_FILE_RECENT_FREEZE 40162 +#define ID_FILE_EXPORT_GAMESHARKSNAPSHOT 40163 +#define ID_OPTIONS_VIDEO_FULLSCREEN800X600 40164 +#define ID_OPTIONS_VIDEO_FRAMESKIP_6 40165 +#define ID_OPTIONS_VIDEO_FRAMESKIP_7 40166 +#define ID_OPTIONS_VIDEO_FRAMESKIP_8 40167 +#define ID_OPTIONS_VIDEO_FRAMESKIP_9 40168 +#define ID_OPTIONS_EMULATOR_SAVETYPE_AUTOMATIC 40169 +#define ID_OPTIONS_EMULATOR_SAVETYPE_EEPROM 40170 +#define ID_OPTIONS_EMULATOR_SAVETYPE_SRAM 40171 +#define ID_OPTIONS_EMULATOR_SAVETYPE_FLASH 40172 +#define ID_OPTIONS_EMULATOR_SAVETYPE_EEPROMSENSOR 40173 +#define ID_OPTIONS_EMULATOR_SAVETYPE_FLASH512K 40174 +#define ID_OPTIONS_EMULATOR_SAVETYPE_FLASH1M 40175 +#define ID_OPTIONS_EMULATOR_AUTOMATICALLYIPSPATCH 40176 +#define ID_TOOLS_RECORD_STARTAVIRECORDING 40178 +#define ID_TOOLS_RECORD_STOPAVIRECORDING 40179 +#define ID_OPTIONS_SOUND_VOLUME_1X 40182 +#define ID_OPTIONS_SOUND_VOLUME_2X 40183 +#define ID_OPTIONS_SOUND_VOLUME_3X 40184 +#define ID_OPTIONS_SOUND_VOLUME_4X 40185 +#define ID_OPTIONS_FILTER_BILINEAR 40186 +#define ID_OPTIONS_FILTER_BILINEARPLUS 40187 +#define ID_OPTIONS_FILTER_INTERFRAMEBLENDING_NONE 40188 +#define ID_OPTIONS_FILTER_INTERFRAMEBLENDING_MOTIONBLUR 40189 +#define ID_OPTIONS_FILTER_INTERFRAMEBLENDING_SMART 40190 +#define ID_OPTIONS_VIDEO_FULLSCREEN 40191 +#define ID_OPTIONS_VIDEO_TRIPLEBUFFERING 40192 +#define ID_OPTIONS_FRAMESKIP_AUTOMATIC 40194 +#define ID_OPTIONS_EMULATOR_SHOWSPEED_NONE 40195 +#define ID_OPTIONS_EMULATOR_SHOWSPEED_PERCENTAGE 40196 +#define ID_OPTIONS_EMULATOR_SHOWSPEED_DETAILED 40197 +#define ID_OPTIONS_EMULATOR_SHOWSPEED_TRANSPARENT 40198 +#define ID_OPTIONS_JOYPAD_CONFIGURE_1 40199 +#define ID_OPTIONS_JOYPAD_CONFIGURE_2 40200 +#define ID_OPTIONS_JOYPAD_CONFIGURE_3 40201 +#define ID_OPTIONS_JOYPAD_CONFIGURE_4 40202 +#define ID_OPTIONS_JOYPAD_DEFAULTJOYPAD_1 40208 +#define ID_OPTIONS_JOYPAD_DEFAULTJOYPAD_2 40209 +#define ID_OPTIONS_JOYPAD_DEFAULTJOYPAD_3 40210 +#define ID_OPTIONS_JOYPAD_DEFAULTJOYPAD_4 40211 +#define ID_OPTIONS_EMULATOR_STORESETTINGSINREGISTRY 40214 +#define ID_FILE_EXPORT_SETTINGSTOINI 40215 +#define ID_OPTIONS_FRAMESKIP_THROTTLE_NOTHROTTLE 40216 +#define ID_OPTIONS_FRAMESKIP_THROTTLE_50 40217 +#define ID_OPTIONS_FRAMESKIP_THROTTLE_150 40218 +#define ID_OPTIONS_FRAMESKIP_THROTTLE_200 40219 +#define ID_OPTIONS_FRAMESKIP_THROTTLE_25 40220 +#define ID_OPTIONS_FRAMESKIP_THROTTLE_OTHER 40221 +#define ID_OPTIONS_FRAMESKIP_THROTTLE_100 40222 +#define ID_OPTIONS_FILTER_SCANLINES 40223 +#define ID_OPTIONS_VIDEO_RENDERMETHOD_GDI 40228 +#define ID_OPTIONS_VIDEO_RENDERMETHOD_DIRECTDRAW 40229 +#define ID_OPTIONS_VIDEO_RENDERMETHOD_DIRECT3D 40230 +#define ID_OPTIONS_VIDEO_RENDERMETHOD_OPENGL 40231 +#define ID_OPTIONS_VIDEO_RENDEROPTIONS_D3DNOFILTER 40233 +#define ID_OPTIONS_VIDEO_RENDEROPTIONS_D3DBILINEAR 40234 +#define ID_OPTIONS_VIDEO_RENDEROPTIONS_D3DTRILINEAR 40235 +#define ID_OPTIONS_VIDEO_RENDEROPTIONS_D3DANISOTROPIC 40236 +#define ID_OPTIONS_VIDEO_RENDEROPTIONS_GLNEAREST 40237 +#define ID_OPTIONS_VIDEO_RENDEROPTIONS_GLBILINEAR 40238 +#define ID_OPTIONS_VIDEO_RENDEROPTIONS_GLTRIANGLE 40239 +#define ID_OPTIONS_VIDEO_RENDEROPTIONS_GLQUADS 40240 +#define ID_OPTIONS_VIDEO_RENDEROPTIONS_SELECTSKIN 40245 +#define ID_OPTIONS_VIDEO_RENDEROPTIONS_SKIN 40246 +#define ID_OPTIONS_EMULATOR_AGBPRINT 40247 +#define ID_OPTIONS_EMULATOR_REALTIMECLOCK 40248 +#define ID_OPTIONS_GAMEBOY_SGB2 40249 +#define ID_SYSTEM_MINIMIZE 40250 +#define ID_TOOLS_RECORD_STARTMOVIERECORDING 40251 +#define ID_TOOLS_RECORD_STOPMOVIERECORDING 40252 +#define ID_TOOLS_PLAY_STARTMOVIEPLAYING 40253 +#define ID_TOOLS_PLAY_STOPMOVIEPLAYING 40254 +#define ID_OPTIONS_EMULATOR_AUTOHIDEMENU 40255 +#define ID_OPTIONS_GAMEBOY_BORDERAUTOMATIC 40256 +#define ID_OPTIONS_EMULATOR_REWIND 40257 +#define ID_TOOLS_REWIND 40258 +#define ID_OPTIONS_EMULATOR_SKIPBIOS 40259 +#define ID_HELP_BUGREPORT 40260 +#define ID_HELP_FAQ 40261 +#define ID_OPTIONS_EMULATOR_REWINDINTERVAL 40262 +#define ID_FILE_TOGGLEMENU 40263 +#define ID_OPTIONS_EMULATOR_SAVETYPE_NONE 40264 +#define ID_TOOLS_IOVIEWER 40266 +#define ID_FILE_LOADGAME_MOSTRECENT 40267 +#define ID_FILE_SAVEGAME_OLDESTSLOT 40268 +#define ID_FILE_LOADGAME_AUTOLOADMOSTRECENT 40269 +#define ID_OPTIONS_SOUND_VOLUME_5X 40270 +#define ID_OPTIONS_SOUND_VOLUME_25X 40271 +#define ID_CHEATS_DISABLECHEATS 40272 +#define ID_OPTIONS_VIDEO_FULLSCREENMAXSCALE 40273 +#define ID_OPTIONS_FILTER_HQ2X 40274 +#define ID_OPTIONS_FILTER_LQ2X 40275 +#define ID_OPTIONS_EMULATOR_GAMEOVERRIDES 40276 +#define ID_HELP_GNUPUBLICLICENSE 40277 +#define ID_OPTIONS_EMULATOR_GENERICFLASHCARD 40279 +#define ID_OPTIONS_SOUND_HARDWAREACCELERATION 40281 +#define ID_OPTIONS_VIDEO_FULLSCREEN1024X768 40282 +#define ID_OPTIONS_VIDEO_FULLSCREEN1280X1024 40283 +#define ID_OPTIONS_FILTER_SIMPLE3X 40287 +#define ID_OPTIONS_FILTER_SIMPLE4X 40288 +#define ID_OPTIONS_FILTER_HQ3X 40290 +#define ID_OPTIONS_FILTER_HQ4X 40291 +#define ID_VIDEO_WINDOWED 40292 +#define ID_VIDEO_FULL 40293 +#define ID_OPTIONS_SOUND_PCMINTERPOLATION_NONE 40294 +#define ID_OPTIONS_SOUND_PCMINTERPOLATION_LINEAR 40295 +#define ID_OPTIONS_SOUND_PCMINTERPOLATION_CUBIC 40296 +#define ID_OPTIONS_SOUND_PCMINTERPOLATION_FIR 40297 +#define ID_OPTIONS_SOUND_PCMINTERPOLATION_LIBRESAMPLE 40298 +#define ID_OPTIONS_FILTER_LCDCOLORS 40299 +#define ID_OPTIONS_EMULATOR_REMOVEINTROSGBA 40331 +#define IDD_LINKTAB1 40300 +#define IDD_LINKTAB 40301 +#define IDD_LINKTAB2 40302 +#define IDD_LINKTAB3 40303 +#define IDD_SERVERWAIT 40304 +#define IDC_TAB1 40305 +#define IDC_LINK_SINGLE 40306 +#define IDC_LINK_TIMEOUT 40307 +#define IDC_LINK_LAN 40308 +#define IDC_LINK2P 40309 +#define IDC_LINKTCP 40310 +#define IDC_SSPEED 40311 +#define IDC_SERVERSTART 40312 +#define IDC_SERVERIP 40313 +#define IDC_CLINKIP 40314 +#define IDC_SPEEDOFF 40315 +#define IDC_LINKCONNECT 40316 +#define ID_OPTIONS_LINK_OPTIONS 40318 +#define ID_OPTIONS_LINK_LOG 40319 +#define ID_OPTIONS_LINK_WIRELESSADAPTER 40320 +#define IDC_LINKTIMEOUT 40321 +#define IDC_CLINKTCP 40322 +#define IDC_SERVERWAIT 40323 +#define IDC_LINKUDP 40324 +#define IDC_LINK3P 40325 +#define IDC_LINK4P 40326 +#define IDC_CLINKUDP 40327 +#define IDC_SPEEDON 40328 + +// Next default values for new objects +// +#ifdef APSTUDIO_INVOKED +#ifndef APSTUDIO_READONLY_SYMBOLS +#define _APS_NEXT_RESOURCE_VALUE 159 +#define _APS_NEXT_COMMAND_VALUE 40332 +#define _APS_NEXT_CONTROL_VALUE 1269 +#define _APS_NEXT_SYMED_VALUE 103 +#endif +#endif diff --git a/src/win32/resource2.h b/src/win32/resource2.h new file mode 100644 index 00000000..3192ab96 --- /dev/null +++ b/src/win32/resource2.h @@ -0,0 +1 @@ +#define IDI_ICON 101 diff --git a/src/win32/skin.cpp b/src/win32/skin.cpp new file mode 100644 index 00000000..b123c1d3 --- /dev/null +++ b/src/win32/skin.cpp @@ -0,0 +1,588 @@ +// =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- +// +// WINDOWS SKINNING TUTORIAL - by Vander Nunes - virtware.net +// This is the source-code that shows what is discussed in the tutorial. +// The code is simplified for the sake of clarity, but all the needed +// features for handling skinned windows is present. Please read +// the article for more information. +// +// skin.cpp : CSkin class implementation +// 28/02/2002 : initial release. +// +// =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- + +#include "stdafx.h" +#include "skin.h" +#include +#include "xImage.h" + +#include "../System.h" + +#ifdef _DEBUG +#define new DEBUG_NEW +#undef THIS_FILE +static char THIS_FILE[] = __FILE__; +#endif + +// ---------------------------------------------------------------------------- +// constructor 1 - use it when you have not already created the app window. +// this one will not subclass automatically, you must call Hook() and Enable() +// to subclass the app window and enable the skin respectively. +// will throw an exception if unable to initialize skin from resource. +// ---------------------------------------------------------------------------- + +CSkin::CSkin() +{ + // default starting values + m_bHooked = false; + m_OldWndProc = NULL; + + m_rect.top = 0; + m_rect.bottom = 0; + m_rect.right = 0; + m_rect.left = 0; + + m_dOldStyle = 0; + + m_oldRect = m_rect; + m_nButtons = 0; + m_buttons = NULL; +} + +// ---------------------------------------------------------------------------- +// destructor - just call the destroyer +// ---------------------------------------------------------------------------- +CSkin::~CSkin() +{ + Destroy(); +} + +HBITMAP CSkin::LoadImage(const char *filename) +{ + CxImage image; + image.Load(filename); + if(!image.IsValid()) { + return NULL; + } + + return image.MakeBitmap(NULL); +} + +// ---------------------------------------------------------------------------- +// Initialize the skin +// ---------------------------------------------------------------------------- +bool CSkin::Initialize(const char *skinFile) +{ + // try to retrieve the skin data from resource. + bool res = GetSkinData(skinFile); + if(!res) + systemMessage(0, m_error); + return res; +} + +// ---------------------------------------------------------------------------- +// destroy skin resources and free allocated resources +// ---------------------------------------------------------------------------- +void CSkin::Destroy() +{ + if (m_buttons) { + delete[] m_buttons; + m_buttons = NULL; + } + + // unhook the window + UnHook(); + + // free bitmaps and device context + if (m_dcSkin) { SelectObject(m_dcSkin, m_hOldBmp); DeleteDC(m_dcSkin); m_dcSkin = NULL; } + if (m_hBmp) { DeleteObject(m_hBmp); m_hBmp = NULL; } + + // free skin region + if (m_rgnSkin) { DeleteObject(m_rgnSkin); m_rgnSkin = NULL; } + +} + + + +// ---------------------------------------------------------------------------- +// toggle skin on/off - must be Hooked() before attempting to enable skin. +// ---------------------------------------------------------------------------- +bool CSkin::Enable(bool bEnable) +{ + // refuse to enable if there is no window subclassed yet. + if (!Hooked()) return false; + + // toggle + m_bEnabled = bEnable; + + // force window repainting + InvalidateRect(m_hWnd, NULL, TRUE); + + return true; +} + + + +// ---------------------------------------------------------------------------- +// tell if the skinning is enabled +// ---------------------------------------------------------------------------- +bool CSkin::Enabled() +{ + return m_bEnabled; +} + + + +// ---------------------------------------------------------------------------- +// hook a window +// ---------------------------------------------------------------------------- +bool CSkin::Hook(CWnd *pWnd) +{ + // unsubclass any other window + if (Hooked()) UnHook(); + + // this will be our new subclassed window + m_hWnd = (HWND)*pWnd; + + // -------------------------------------------------- + // change window style (get rid of the caption bar) + // -------------------------------------------------- + LONG_PTR dwStyle = GetWindowLongPtr(m_hWnd, GWL_STYLE); + m_dOldStyle = dwStyle; + dwStyle &= ~(WS_CAPTION|WS_SIZEBOX); + SetWindowLongPtr(m_hWnd, GWL_STYLE, dwStyle); + + RECT r; + pWnd->GetWindowRect(&r); + m_oldRect = r; + pWnd->MoveWindow(r.left, + r.top, + m_iWidth, + m_iHeight, + FALSE); + + pWnd->SetMenu(NULL); + + if(m_rgnSkin != NULL) + // set the skin region to the window + pWnd->SetWindowRgn(m_rgnSkin, true); + + // subclass the window procedure + m_OldWndProc = (WNDPROC)SetWindowLongPtr( m_hWnd, GWLP_WNDPROC, (LONG_PTR)SkinWndProc ); + + // store a pointer to our class instance inside the window procedure. + if (!SetProp(m_hWnd, "skin", (void*)this)) + { + // if we fail to do so, we just can't activate the skin. + UnHook(); + return false; + } + + + // update flag + m_bHooked = ( m_OldWndProc ? true : false ); + + for(int i = 0; i < m_nButtons; i++) { + RECT r; + m_buttons[i].GetRect(r); + m_buttons[i].CreateButton("", WS_VISIBLE, r, pWnd, 0); + } + + // force window repainting + RedrawWindow(NULL,NULL,NULL,RDW_INVALIDATE|RDW_ERASE|RDW_ALLCHILDREN); + + // successful return if we're hooked. + return m_bHooked; +} + + + +// ---------------------------------------------------------------------------- +// unhook the window +// ---------------------------------------------------------------------------- +bool CSkin::UnHook() +{ + // just to be safe we'll check this + WNDPROC OurWnd; + + // cannot unsubclass if there is no window subclassed + // returns true anyways. + if (!Hooked()) return true; + + if(m_rgnSkin != NULL) + // remove the skin region from the window + SetWindowRgn(m_hWnd, NULL, true); + + // unsubclass the window procedure + OurWnd = (WNDPROC)SetWindowLongPtr( m_hWnd, GWLP_WNDPROC, (LONG_PTR)m_OldWndProc ); + + // remove the pointer to our class instance, but if we fail we don't care. + RemoveProp(m_hWnd, "skin"); + + // update flag - if we can't get our window procedure address again, + // we failed to unhook the window. + m_bHooked = ( OurWnd ? false : true ); + + SetWindowLongPtr(m_hWnd, GWL_STYLE, m_dOldStyle); + + RECT r; + + GetWindowRect(m_hWnd, &r); + + MoveWindow(m_hWnd, + r.left, + r.top, + m_oldRect.right - m_oldRect.left, + m_oldRect.bottom - m_oldRect.top, + FALSE); + + // force window repainting + RedrawWindow(NULL,NULL,NULL,RDW_INVALIDATE|RDW_ERASE|RDW_ALLCHILDREN); + + // successful return if we're unhooked. + return !m_bHooked; +} + + + +// ---------------------------------------------------------------------------- +// tell us if there is a window subclassed +// ---------------------------------------------------------------------------- +bool CSkin::Hooked() +{ + return m_bHooked; +} + + + +// ---------------------------------------------------------------------------- +// return the skin bitmap width +// ---------------------------------------------------------------------------- +int CSkin::Width() +{ + return m_iWidth; +} + + + +// ---------------------------------------------------------------------------- +// return the skin bitmap height +// ---------------------------------------------------------------------------- +int CSkin::Height() +{ + return m_iHeight; +} + + + +// ---------------------------------------------------------------------------- +// return the skin device context +// ---------------------------------------------------------------------------- +HDC CSkin::HDC() +{ + return m_dcSkin; +} + +bool CSkin::ParseRect(char *buffer, RECT& rect) +{ + char *token = strtok(buffer, ","); + + if(token == NULL) + return false; + rect.left = atoi(token); + + token = strtok(NULL, ","); + if(token == NULL) + return false; + rect.top = atoi(token); + + token = strtok(NULL, ","); + if(token == NULL) + return false; + rect.right = rect.left + atoi(token); + + token = strtok(NULL, ","); + if(token == NULL) + return false; + rect.bottom = rect.top + atoi(token); + + token = strtok(NULL, ","); + if(token != NULL) + return false; + + return true; +} + +HRGN CSkin::LoadRegion(const char *rgn) +{ + // ------------------------------------------------- + // then, we retrieve the skin region from resource. + // ------------------------------------------------- + FILE *f = fopen(rgn, "rb"); + if(!f) return NULL; + + fseek(f, 0, SEEK_END); + int size = ftell(f); + LPRGNDATA pSkinData = (LPRGNDATA)malloc(size); + if(!pSkinData) { + fclose(f); + return NULL; + } + + fseek(f, 0, SEEK_SET); + + fread(pSkinData, 1, size, f); + + fclose(f); + + // create the region using the binary data. + HRGN r = ExtCreateRegion(NULL, size, pSkinData); + + // free the allocated resource + free(pSkinData); + + return r; +} + +// ---------------------------------------------------------------------------- +// skin retrieval helper +// ---------------------------------------------------------------------------- +bool CSkin::GetSkinData(const char *skinFile) +{ + // ------------------------------------------------- + // retrieve the skin bitmap from resource. + // ------------------------------------------------- + + char buffer[2048]; + + if(!GetPrivateProfileString("skin", "image", "", buffer, 2048, skinFile)) { + m_error = "Missing skin bitmap"; + return false; + } + CString bmpName = buffer; + CString rgn = ""; + if(GetPrivateProfileString("skin", "region", "", buffer, 2048, skinFile)) { + rgn = buffer; + } + + if(!GetPrivateProfileString("skin", "draw", "", buffer, 2048, skinFile)) { + m_error = "Missing draw rectangle"; + return false; + } + + if(!ParseRect(buffer, m_rect)) { + m_error = "Invalid draw rectangle"; + return false; + } + + m_nButtons = GetPrivateProfileInt("skin", "buttons", 0, skinFile); + + if(m_nButtons) { + m_buttons = new SkinButton[m_nButtons]; + for(int i = 0; i < m_nButtons; i++) { + if(!ReadButton(skinFile, i)) + return false; + } + } + + CString path = skinFile; + int index = path.ReverseFind('\\'); + if(index != -1) { + path = path.Left(index+1); + } + + bmpName = path + bmpName; + if(strcmp(rgn, "")) + rgn = path + rgn; + + m_hBmp = LoadImage(bmpName); + + if (!m_hBmp) { + m_error = "Error loading skin bitmap " + bmpName; + return false; + } + + // get skin info + BITMAP bmp; + GetObject(m_hBmp, sizeof(bmp), &bmp); + + // get skin dimensions + m_iWidth = bmp.bmWidth; + m_iHeight = bmp.bmHeight; + + if(strcmp(rgn, "")) { + m_rgnSkin = LoadRegion(rgn); + if(m_rgnSkin == NULL) { + m_error = "Error loading skin region " + rgn; + return false; + } + } + + // ------------------------------------------------- + // well, things are looking good... + // as a quick providence, just create and keep + // a device context for our later blittings. + // ------------------------------------------------- + + // create a context compatible with the user desktop + m_dcSkin = CreateCompatibleDC(0); + if (!m_dcSkin) return false; + + // select our bitmap + m_hOldBmp = (HBITMAP)SelectObject(m_dcSkin, m_hBmp); + + + // ------------------------------------------------- + // done + // ------------------------------------------------- + return true; +} + +bool CSkin::ReadButton(const char *skinFile, int num) +{ + char buffer[2048]; + + CString path = skinFile; + int index = path.ReverseFind('\\'); + if(index != -1) { + path = path.Left(index+1); + } + sprintf(buffer, "button-%d", num); + CString name = buffer; + + if(!GetPrivateProfileString(name, "normal", "", buffer, 2048, skinFile)) { + m_error = "Missing button bitmap for " + name; + return false; + } + + CString normalBmp = path + buffer; + + HBITMAP bmp = LoadImage(normalBmp); + if(!bmp) { + m_error = "Error loading button bitmap " + normalBmp; + return false; + } + m_buttons[num].SetNormalBitmap(bmp); + + if(!GetPrivateProfileString(name, "down", "", buffer, 2048, skinFile)) { + m_error = "Missing button down bitmap " + name; + return false; + } + + CString downBmp = path + buffer; + + bmp = LoadImage(downBmp); + + if (!bmp) { + m_error = "Error loading button down bitmap " + downBmp; + return false; + } + m_buttons[num].SetDownBitmap(bmp); + + if(GetPrivateProfileString(name, "over", "", buffer, 2048, skinFile)) { + CString overBmp = path + buffer; + + bmp = LoadImage(overBmp); + + if (!bmp) { + m_error = "Error loading button over bitmap " + overBmp; + return false; + } + m_buttons[num].SetOverBitmap(bmp); + } + + if(GetPrivateProfileString(name, "region", "", buffer, 2048, skinFile)) { + CString region = path + buffer; + + HRGN rgn = LoadRegion(region); + if(!rgn) { + m_error = "Error loading button region " + region; + return false; + } + m_buttons[num].SetRegion(rgn); + } + + if(!GetPrivateProfileString(name, "id", "", buffer, 2048, skinFile)) { + "Missing button ID for " + name; + return false; + } + m_buttons[num].SetId(buffer); + + if(!GetPrivateProfileString(name, "rect", "", buffer, 2048, skinFile)) { + m_error = "Missing button rectangle for " + name; + return false; + } + + RECT r; + if(!ParseRect(buffer, r)) { + m_error = "Invalid button rectangle for " + name; + return false; + } + m_buttons[num].SetRect(r); + + return true; +} + +// ------------------------------------------------------------------------ +// Default skin window procedure. +// Here the class will handle WM_PAINT and WM_LBUTTONDOWN, originally sent +// to the application window, but now subclassed. Any other messages will +// just pass through the procedure and reach the original app procedure. +// ------------------------------------------------------------------------ +LRESULT CALLBACK SkinWndProc(HWND hWnd, UINT uMessage, WPARAM wParam, LPARAM lParam) +{ + // we will need a pointer to the associated class instance + // (it was stored in the window before, remember?) + CSkin *pSkin = (CSkin*)GetProp(hWnd, _T("skin")); + + // to handle WM_PAINT + PAINTSTRUCT ps; + + // if we fail to get our class instance, we can't handle anything. + if (!pSkin) return DefWindowProc(hWnd,uMessage,wParam,lParam); + + switch(uMessage) + { + case WM_WINDOWPOSCHANGING: + { + LPWINDOWPOS pos = (LPWINDOWPOS)lParam; + pos->cx = pSkin->Width(); + pos->cy = pSkin->Height(); + return 0L; + } + break; + + case WM_PAINT: + { + // --------------------------------------------------------- + // here we just need to blit our skin + // directly to the device context + // passed by the painting message. + // --------------------------------------------------------- + BeginPaint(hWnd,&ps); + + // blit the skin + BitBlt(ps.hdc,0,0,pSkin->Width(),pSkin->Height(),pSkin->HDC(),0,0,SRCCOPY); + + EndPaint(hWnd,&ps); + break; + } + + case WM_LBUTTONDOWN: + { + // --------------------------------------------------------- + // this is a common trick for easy dragging of the window. + // this message fools windows telling that the user is + // actually dragging the application caption bar. + // --------------------------------------------------------- + SendMessage(hWnd, WM_NCLBUTTONDOWN, HTCAPTION,NULL); + break; + } + + } + + // --------------------------------------------------------- + // call the default window procedure to keep things going. + // --------------------------------------------------------- + return CallWindowProc(pSkin->m_OldWndProc, hWnd, uMessage, wParam, lParam); +} diff --git a/src/win32/skin.h b/src/win32/skin.h new file mode 100644 index 00000000..f8301e5a --- /dev/null +++ b/src/win32/skin.h @@ -0,0 +1,162 @@ +// =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- +// +// WINDOWS SKINNING TUTORIAL - by Vander Nunes - virtware.net +// This is the source-code that shows what is discussed in the tutorial. +// The code is simplified for the sake of clarity, but all the needed +// features for handling skinned windows is present. Please read +// the article for more information. +// +// skin.h : CSkin class declaration +// 28/02/2002 : initial release. +// +// =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- + + +#ifndef _SKIN_H_ + +#define _SKIN_H_ + +#include "skinButton.h" + + + +// -------------------------------------------------------------------------- +// The CSkin class will load the skin from a resource +// and subclass the associated window, so that the +// WM_PAINT message will be redirected to the provided +// window procedure. All the skin handling will be automatized. +// -------------------------------------------------------------------------- + +class CSkin +{ + + // -------------------------------------------------------------------------- + // the skin window procedure, where the class + // will handle WM_PAINT and WM_LBUTTONDOWN automatically. + // -------------------------------------------------------------------------- + friend LRESULT CALLBACK SkinWndProc(HWND hWnd, UINT uMessage, WPARAM wParam, LPARAM lParam); + + private: + + // the associated window handle + HWND m_hWnd; + + // the old window procedure + WNDPROC m_OldWndProc; + + // skin region + HRGN m_rgnSkin; + + // the internal skin device context handle + HDC m_dcSkin; + + // bitmap and old bitmap from the device context + HBITMAP m_hBmp, m_hOldBmp; + + // skin dimensions + int m_iWidth, m_iHeight; + + // on|off toggle + bool m_bEnabled; + + // tell the class if it has a window subclassed. + bool m_bHooked; + + // skin retrieval helper + bool GetSkinData(const char *skin); + + RECT m_rect; + + LONG_PTR m_dOldStyle; + RECT m_oldRect; + int m_nButtons; + SkinButton *m_buttons; + CString m_error; + + public: + + // ---------------------------------------------------------------------------- + // constructor 1 - use it when you have not already created the app window. + // this one will not subclass automatically, you must call Hook() to subclass. + // will throw an exception if unable to initialize skin from resource. + // ---------------------------------------------------------------------------- + + CSkin(); + + // ---------------------------------------------------------------------------- + // destructor - just call the destroyer + // ---------------------------------------------------------------------------- + + virtual ~CSkin(); + + // ---------------------------------------------------------------------------- + // Initialize the skin + // ---------------------------------------------------------------------------- + bool Initialize(const char *); + + + // ---------------------------------------------------------------------------- + // destroy skin resources and free allocated resources + // ---------------------------------------------------------------------------- + + void Destroy(); + + // ---------------------------------------------------------------------------- + // subclass a window. + // ---------------------------------------------------------------------------- + + bool Hook(CWnd *pWnd); + + // ---------------------------------------------------------------------------- + // unsubclass the subclassed window. + // ---------------------------------------------------------------------------- + + bool UnHook(); + + // ---------------------------------------------------------------------------- + // tell us if we have a window subclassed. + // ---------------------------------------------------------------------------- + + bool Hooked(); + + // ---------------------------------------------------------------------------- + // toggle skin on/off. + // ---------------------------------------------------------------------------- + + bool Enable(bool bEnable); + + // ---------------------------------------------------------------------------- + // tell if the skinning is enabled + // ---------------------------------------------------------------------------- + + bool Enabled(); + + // ---------------------------------------------------------------------------- + // return the skin bitmap width. + // ---------------------------------------------------------------------------- + + int Width(); + + // ---------------------------------------------------------------------------- + // return the skin bitmap height. + // ---------------------------------------------------------------------------- + + int Height(); + + // Return blit rect + RECT &GetBlitRect() { return m_rect; } + + // ---------------------------------------------------------------------------- + // return the skin device context. + // ---------------------------------------------------------------------------- + + HDC HDC(); + + private: + HBITMAP LoadImage(const char *); + bool ReadButton(const char *, int); + static bool ParseRect(char *, RECT &); + static HRGN LoadRegion(const char *); +}; + +#endif diff --git a/src/win32/skinButton.cpp b/src/win32/skinButton.cpp new file mode 100644 index 00000000..f2525f5b --- /dev/null +++ b/src/win32/skinButton.cpp @@ -0,0 +1,330 @@ +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +// skinButton.cpp : implementation file +// + +#include "stdafx.h" +#include "vba.h" +#include "skinButton.h" + +#ifdef _DEBUG +#define new DEBUG_NEW +#undef THIS_FILE +static char THIS_FILE[] = __FILE__; +#endif + +extern bool winAccelGetID(const char *command, WORD& id); + +///////////////////////////////////////////////////////////////////////////// +// SkinButton + +SkinButton::SkinButton() +{ + normalBmp = NULL; + downBmp = NULL; + overBmp = NULL; + mouseOver = false; + id = ""; + idCommand = 0; + region = NULL; + buttonMask = 0; + menu = -1; +} + +SkinButton::~SkinButton() +{ + DestroyWindow(); + if(normalBmp) { + DeleteObject(normalBmp); + normalBmp = NULL; + } + if(downBmp) { + DeleteObject(downBmp); + downBmp = NULL; + } + if(overBmp) { + DeleteObject(overBmp); + overBmp = NULL; + } + if(region) { + DeleteObject(region); + region = NULL; + } +} + + +BEGIN_MESSAGE_MAP(SkinButton, CWnd) + //{{AFX_MSG_MAP(SkinButton) + ON_WM_ERASEBKGND() + ON_WM_PAINT() + ON_WM_KILLFOCUS() + ON_WM_CAPTURECHANGED() + ON_WM_CONTEXTMENU() + //}}AFX_MSG_MAP + ON_MESSAGE(WM_LBUTTONUP, OnLButtonUpMsg) + ON_MESSAGE(WM_LBUTTONDOWN, OnLButtonDownMsg) + ON_MESSAGE(WM_MOUSEMOVE, OnMouseMoveMsg) + ON_MESSAGE(WM_MOUSELEAVE, OnMouseLeaveMsg) + END_MESSAGE_MAP() + + + ///////////////////////////////////////////////////////////////////////////// +// SkinButton message handlers + +BOOL SkinButton::OnEraseBkgnd(CDC* pDC) +{ + return TRUE; +} + +void SkinButton::OnPaint() +{ + PAINTSTRUCT ps; + HDC hDC = ::BeginPaint(m_hWnd, &ps); + HDC memDC = ::CreateCompatibleDC(hDC); + LRESULT state = ::SendMessage(m_hWnd, BM_GETSTATE, 0, 0); + HBITMAP oldBitmap; + if(state & BST_PUSHED) + oldBitmap = (HBITMAP)SelectObject(memDC, downBmp); + else if(mouseOver && overBmp != NULL) + oldBitmap = (HBITMAP)SelectObject(memDC, overBmp); + else + oldBitmap = (HBITMAP)SelectObject(memDC, normalBmp); + SelectClipRgn(hDC, region); + BitBlt(hDC, 0, 0, theApp.rect.right - theApp.rect.left, + theApp.rect.bottom - theApp.rect.top, memDC, 0, 0, SRCCOPY); + SelectClipRgn(hDC, NULL); + SelectObject(memDC, oldBitmap); + DeleteDC(memDC); + + ::EndPaint(m_hWnd, &ps); +} + +LRESULT SkinButton::OnLButtonUpMsg(WPARAM wParam, LPARAM lParam) +{ + POINT pt; + pt.x = LOWORD(lParam); + pt.y = HIWORD(lParam); + RECT r; + GetClientRect(&r); + BOOL inside = PtInRect(&r, pt); + if(region != NULL) + inside &= PtInRegion(region, pt.x, pt.y); + if(inside) { + ReleaseCapture(); + Invalidate(); + HWND hWnd = m_hWnd; + if(idCommand != 0) + GetParent()->SendMessage(WM_COMMAND, idCommand, 0); + else if(buttonMask) + theApp.skinButtons = 0; + else if(menu != -1) { + HMENU m = GetSubMenu(theApp.menu, menu); + pt.x = r.left; + pt.y = r.bottom; + ClientToScreen(&pt); + theApp.m_pMainWnd->SendMessage(WM_INITMENUPOPUP, (WPARAM)m, menu); + TrackPopupMenu(m, 0, pt.x, pt.y, 0, *theApp.m_pMainWnd, NULL); + } + + return ::DefWindowProc(hWnd, WM_LBUTTONUP, wParam, lParam); + } + return GetParent()->SendMessage(WM_LBUTTONUP, wParam, lParam); +} + +LRESULT SkinButton::OnLButtonDownMsg(WPARAM wParam, LPARAM lParam) +{ + if(idCommand != 0) + return Default(); + + POINT pt; + pt.x = LOWORD(lParam); + pt.y = HIWORD(lParam); + RECT r; + GetClientRect(&r); + BOOL inside = PtInRect(&r, pt); + if(region != NULL) + inside &= PtInRegion(region, pt.x, pt.y); + if(inside) { + if(buttonMask) + theApp.skinButtons = buttonMask; + return Default(); + } + return GetParent()->SendMessage(WM_LBUTTONDOWN, wParam, lParam); +} + +LRESULT SkinButton::OnMouseMoveMsg(WPARAM wParam, LPARAM lParam) +{ + if(wParam & MK_LBUTTON && !mouseOver) + return Default(); + + if(GetCapture() != this) { + SetCapture(); + } + POINT pt; + pt.x = LOWORD(lParam); + pt.y = HIWORD(lParam); + // ClientToScreen(getHandle(), &p); + RECT r; + GetClientRect(&r); + BOOL inside = PtInRect(&r, pt); + if(region != NULL) + inside &= PtInRegion(region, pt.x, pt.y); + + if(!inside) { + // HWND h = WindowFromPoint(p); + // if(h != getHandle()) { + if(mouseOver) { + mouseOver = false; + Invalidate(); + } + if(!(wParam & MK_LBUTTON)) + ReleaseCapture(); + } else { + if(!mouseOver) { + mouseOver = true; + Invalidate(); + } + } + return Default(); +} + +void SkinButton::OnKillFocus(CWnd* pNewWnd) +{ + mouseOver = false; + Invalidate(); + + CWnd::OnKillFocus(pNewWnd); +} + +void SkinButton::OnCaptureChanged(CWnd *pWnd) +{ + if(mouseOver) { + ReleaseCapture(); + Invalidate(); + } + + CWnd::OnCaptureChanged(pWnd); +} + +LRESULT SkinButton::OnMouseLeaveMsg(WPARAM wParam, LPARAM lParam) +{ + if(mouseOver) { + ReleaseCapture(); + mouseOver = false; + Invalidate(); + } + + return Default(); +} + +void SkinButton::OnContextMenu(CWnd* pWnd, CPoint point) +{ +} + +void SkinButton::SetNormalBitmap(HBITMAP bmp) +{ + normalBmp = bmp; +} + +void SkinButton::SetDownBitmap(HBITMAP bmp) +{ + downBmp = bmp; +} + +void SkinButton::SetOverBitmap(HBITMAP bmp) +{ + overBmp = bmp; +} + +void SkinButton::SetRect(const RECT& r) +{ + rect = r; +} + +void SkinButton::SetId(const char *id) +{ + this->id = id; + if(!winAccelGetID(id, idCommand)) { + if(!strcmp(id, "A")) + buttonMask = 1; + else if(!strcmp("B", id)) + buttonMask = 2; + else if(!strcmp("SEL", id)) + buttonMask = 4; + else if(!strcmp("START", id)) + buttonMask = 8; + else if(!strcmp("R", id)) + buttonMask = 16; + else if(!strcmp("L", id)) + buttonMask = 32; + else if(!strcmp("U", id)) + buttonMask = 64; + else if(!strcmp("D", id)) + buttonMask = 128; + else if(!strcmp("BR", id)) + buttonMask = 256; + else if(!strcmp("BL", id)) + buttonMask = 512; + else if(!strcmp("SPEED", id)) + buttonMask = 1024; + else if(!strcmp("CAPTURE", id)) + buttonMask = 2048; + else if(!strcmp("GS", id)) + buttonMask = 4096; + else if(!strcmp("UR", id)) + buttonMask = 64+16; + else if(!strcmp("UL", id)) + buttonMask = 64+32; + else if(!strcmp("DR", id)) + buttonMask = 128+16; + else if(!strcmp("DL", id)) + buttonMask = 128+32; + else if(!strcmp("MENUFILE", id)) + menu = 0; + else if(!strcmp("MENUOPTIONS", id)) + menu = 1; + else if(!strcmp("MENUCHEATS", id)) + menu = 2; + else if(!strcmp("MENUTOOLS", id)) + menu = 3; + else if(!strcmp("MENUHELP", id)) + menu = 4; + } +} + +void SkinButton::SetRegion(HRGN rgn) +{ + region = rgn; +} + +void SkinButton::GetRect(RECT& r) +{ + r = rect; +} + +BOOL SkinButton::CreateButton(const char *name, DWORD style, const RECT& r, + CWnd *parent, UINT id) +{ + return CWnd::Create("BUTTON", + name, + style|WS_CHILDWINDOW, + r, + parent, + id); +} diff --git a/src/win32/skinButton.h b/src/win32/skinButton.h new file mode 100644 index 00000000..8fa7312b --- /dev/null +++ b/src/win32/skinButton.h @@ -0,0 +1,93 @@ +// -*- C++ -*- +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +#if !defined(AFX_SKINBUTTON_H__E51B4507_EAD7_43EE_9F54_204BC485D59C__INCLUDED_) +#define AFX_SKINBUTTON_H__E51B4507_EAD7_43EE_9F54_204BC485D59C__INCLUDED_ + +#if _MSC_VER > 1000 +#pragma once +#endif // _MSC_VER > 1000 +// skinButton.h : header file +// + +///////////////////////////////////////////////////////////////////////////// +// SkinButton window + +class SkinButton : public CWnd +{ + // Construction + public: + SkinButton(); + + // Attributes + private: + HBITMAP normalBmp; + HBITMAP downBmp; + HBITMAP overBmp; + RECT rect; + bool mouseOver; + CString id; + HRGN region; + WORD idCommand; + int buttonMask; + int menu; + + // Operations + public: + BOOL CreateButton(const char *, DWORD, const RECT&, CWnd *, UINT); + + void SetNormalBitmap(HBITMAP); + void SetDownBitmap(HBITMAP); + void SetOverBitmap(HBITMAP); + void SetRect(const RECT &); + void GetRect(RECT& r); + void SetId(const char *); + void SetRegion(HRGN); + + // Overrides + // ClassWizard generated virtual function overrides + //{{AFX_VIRTUAL(SkinButton) + //}}AFX_VIRTUAL + + // Implementation + public: + afx_msg LRESULT OnMouseLeaveMsg(WPARAM wParam, LPARAM lParam); + afx_msg LRESULT OnMouseMoveMsg(WPARAM wParam, LPARAM lParam); + afx_msg LRESULT OnLButtonDownMsg(WPARAM wParam, LPARAM lParam); + afx_msg LRESULT OnLButtonUpMsg(WPARAM wParam, LPARAM lParam); + virtual ~SkinButton(); + + // Generated message map functions + protected: + //{{AFX_MSG(SkinButton) + afx_msg BOOL OnEraseBkgnd(CDC* pDC); + afx_msg void OnPaint(); + afx_msg void OnKillFocus(CWnd* pNewWnd); + afx_msg void OnCaptureChanged(CWnd *pWnd); + afx_msg void OnContextMenu(CWnd* pWnd, CPoint point); + //}}AFX_MSG + DECLARE_MESSAGE_MAP() + }; + + ///////////////////////////////////////////////////////////////////////////// + +//{{AFX_INSERT_LOCATION}} +// Microsoft Visual C++ will insert additional declarations immediately before the previous line. + +#endif // !defined(AFX_SKINBUTTON_H__E51B4507_EAD7_43EE_9F54_204BC485D59C__INCLUDED_) diff --git a/src/win32/stdafx.cpp b/src/win32/stdafx.cpp index 1577c4e3..b141a727 100644 --- a/src/win32/stdafx.cpp +++ b/src/win32/stdafx.cpp @@ -1 +1,19 @@ +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + #include "stdafx.h" \ No newline at end of file diff --git a/src/win32/stdafx.h b/src/win32/stdafx.h index 78b29ad5..01f6d5b2 100644 --- a/src/win32/stdafx.h +++ b/src/win32/stdafx.h @@ -1,37 +1,53 @@ -#pragma once - -#define VERSION "S1.7.6" - -#ifndef DIRECT3D_VERSION -#define DIRECT3D_VERSION 0x0900 -#endif - -#ifndef WINVER -#define WINVER 0x0500 -#endif - -#ifndef _WIN32_WINNT -#define _WIN32_WINNT 0x0500 -#endif - -#ifndef _WIN32_WINDOWS -#define _WIN32_WINDOWS 0x0500 -#endif - -#ifndef VC_EXTRALEAN -#define VC_EXTRALEAN -#endif - -#define _ATL_CSTRING_EXPLICIT_CONSTRUCTORS - -#define _AFX_ALL_WARNINGS - -#include -#include - -#include -#ifndef _AFX_NO_AFXCMN_SUPPORT -#include -#endif - -#include "VBA.h" \ No newline at end of file +// -*- C++ -*- +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. +// Copyright (C) 1999-2003 Forgotten +// Copyright (C) 2004 Forgotten and the VBA development team + +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2, or(at your option) +// any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software Foundation, +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +// stdafx.h : include file for standard system include files, +// or project specific include files that are used frequently, but +// are changed infrequently +// + +#if !defined(AFX_STDAFX_H__A7126ECB_A234_4116_A7D0_BE50547E87F8__INCLUDED_) +#define AFX_STDAFX_H__A7126ECB_A234_4116_A7D0_BE50547E87F8__INCLUDED_ + +#if _MSC_VER > 1000 +#pragma once +#endif // _MSC_VER > 1000 + + +// Insert your headers here +//#define WIN32_LEAN_AND_MEAN // Exclude rarely-used stuff from Windows headers + +// Target for Windows 2000 +#define WINVER 0x0500 +#define _WIN32_WINNT 0x0500 + +#define STRICT + + +#include +#include +#include +#include "VBA.h" + +// TODO: reference additional headers your program requires here + +//{{AFX_INSERT_LOCATION}} +// Microsoft Visual C++ will insert additional declarations immediately before the previous line. + +#endif // !defined(AFX_STDAFX_H__A7126ECB_A234_4116_A7D0_BE50547E87F8__INCLUDED_) diff --git a/src/win32/vba.rc2 b/src/win32/vba.rc2 new file mode 100644 index 00000000..02f7ae02 --- /dev/null +++ b/src/win32/vba.rc2 @@ -0,0 +1,56 @@ +#ifndef _MAC +#include "../AutoBuild.h" + +#if _MSC_VER >= 1400 +// use modern icon +IDI_ICON ICON DISCARDABLE "vbavista.ico" +#else +// use traditional icon +IDI_ICON ICON DISCARDABLE "gbadvance.ico" +#endif + + + +///////////////////////////////////////////////////////////////////////////// +// +// Version +// + +VS_VERSION_INFO VERSIONINFO + FILEVERSION FILEVER + PRODUCTVERSION PRODUCTVER + FILEFLAGSMASK 0x3fL +#ifdef _DEBUG + FILEFLAGS 0x29L +#else + FILEFLAGS 0x28L +#endif + FILEOS 0x40004L + FILETYPE 0x1L + FILESUBTYPE 0x0L +BEGIN + BLOCK "StringFileInfo" + BEGIN + BLOCK "040904b0" + BEGIN + VALUE "Comments", "VisualBoyAdvance comes with NO WARRANTY. Use it at your own risk.\0" + VALUE "CompanyName", "http://vba.ngemu.com/\0" + VALUE "FileDescription", "VisualBoyAdvance emulator\0" + VALUE "FileVersion", STRFILEVER + VALUE "InternalName", "VisualBoyAdvance\0" + VALUE "LegalCopyright", "Copyright © 2006 VBA development team\0" + VALUE "LegalTrademarks", "\0" + VALUE "OriginalFilename", "VisualBoyAdvance.exe\0" + VALUE "PrivateBuild", "0\0" + VALUE "ProductName", "VisualBoyAdvance emulator\0" + VALUE "ProductVersion", STRPRODUCTVER + VALUE "SpecialBuild", "0\0" + END + END + BLOCK "VarFileInfo" + BEGIN + VALUE "Translation", 0x409, 1200 + END +END + +#endif // !_MAC diff --git a/src/win32/vbavista.ico b/src/win32/vbavista.ico new file mode 100644 index 0000000000000000000000000000000000000000..b9227f4ea25ed990b81db192468ae1f0c56b0696 GIT binary patch literal 501830 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%0DU+9|HA=lwIR zZ`8x%K8Kv{PPMkBNZwo|!<|QMj^gaRR~~$To>6krEEKgO zRRBf4z<=-V{#i8o!}g92yR+64=F`?TWZ0UFrn0Q;29>eX z$&P#YI)}5x7Z`Q8`xvgv>S$4^YbDa^yn^cd{JKJElSEc>KI_#7AAo$Gd1&3|V^{pe zPh4}DrNFoy#e{328+E)rAE@A(d6o6{3C8}t;5yK2Zn9VDC=*Lt84NK zDzbA+&Si$ro-fYGs<@CBOh5nHS06$S=eY;gecrO-H^(pQS1kCK&-<2t6k0y*T|Ub# zU-U1Z=4kUb4~wrKeQ7=J=it*%A-Be#N$1ki1Lx02vUAGw@@opE^*UpxCotq-s_l+O z-alZmcQIU@OjVYdjr6STbei+{$w1m!|CzJgne=ae?O}-eg?p)V_M77q>y-z-afSZf z<;AC0O5Ru=Jh8%mY=!L2<)vvW7fe)G`^|CU*$ud#L(e>IJ9f-|_^9ph5$5=bAaaHm za*&p3!~MYQTb-3QdxOs(HCmcYw(<+Pp{(3$i>0T&Mt|-+b2=RfKOH&^|Za#LxdHCo%k3Nd$_3|C-KQFlEFCM4AzH$HdttVc3 z$$sigm7ha8(XUavj-PTLJ83%jedf&BYOS$Nqc3&P{dA~wHTjZyr=u<$XeZZl61~#s;apll6`@HFs20l=EzFb>(<#Yd})uPdj(|rI(O1eD%f+ z|L~o47?S%lBG13(`u2fMc|UkxZNIH&WZ%b=&kR%+`m-9OkH0Z!dU{6MpSeGuyZ2CV z@87+Bef+F5cWp+lGXFc@jbvqyg^DN2D(a|$dZC(OYC`^|vf}bWNrOxozn>0+8Tm}4 zMh97cGw&8QH4Rum^vsQokek}GTwH~R|R*j*X=ZETQ%T=0Ym99KHFL*9P^7i+T zihMbB<0toS`isZO8xoKURloPXKO?gQI*vqrk-U!XN{Q-5r`CXn)oe6heqJu7!(^*3 zEULppl*n3@nkJdDDqjNG2GY)H-v98nT{~XhvT1Hl!e2a&rzYOMXQ$!)4?QPOc~6}V zXI(%BxE%HB0x1%t4RTeR!Q3J!7jCJ;%XipZl?BqWoV=RCBBXGurSi&rX-Q^I_-uyb z_$mCf=QnQZT%Yj8*!;yxd=EB%@4%;*_kWm}c*ovd?jy&8>F2;Ys&fm9vvbOF^Q-eE zNI|0@j~i%oUCtYi)-^^M4)+kuyb#IBtIRK~mdYR>WRl@8U?IP9!=C+#i4UybbpQH| z_tSFI0~`LYaX)Qt#`3=3H}2iA>Gmy258iTXjMQhX{n~+ z*EfTeOTEz_^bh{1r+=xoVYa+F>S9{8dgQU&l-fp#3`Hm|^Md;5@q!Nzo_g!;cV2wu zji;Xd)?-gR_w`4f`s%}vKltFI_um&E-?;m($M3uMrEfg)=95o;_^of9`1YF@4!oy4 za?Eos1K?~dEb4@7Fj`u)h8~A&DqJ!hDfzUe{j=`grOvM3yq~3!;jxbP;pWDf?#@q# zhJPNL{z+HgCpC@ZUVqeNjiUHybM)C==#^-;FcqdysUe6_q7Ov0e2I>)(z}}V<~F;l zLaypC+q!Meev5t3#SD44A%?|rz~$*TT6>I^ac^L;wCc0^mY?+WeG=`P>FOM8tQ%}< zn2z@T=KaLi|NG+NXVX&`$3{OIAN%p_?5~zS`TOYbrBptjOOeXqV0Ag$ zRP1S_!8De5n##S+jcmoBho35|o~)?(q`rANQa0-g&ITiwB4r=)7DUyV%tO|2&KSqQCHqzqA?eL;UmpX3^p3UljiS#`XTx?G~I(H~8Cs|CaV& z9l!tI(*}tT{(1gUol>ZP{(t{er(A}fXz`yY0Z{^?1Vjml5)dUINXN?^jpjXS&(jBmp2n{*3H7dPqh zO*&ZYk2%>1y9Z;^>YA{+#;xu#vm=HTvwh6sjG3L|ShqMwEbcLrW7OmvGCBv0jzPVB z&|n|X+lCDG5xsfbj3XU5498DcoMRSu%;Jcd?6}~B%{^iB2uo(d!Q#`Hor&4p`1T1m zkMA9KFcWsynAJIMXU6TGe?`HFU-chb0#TcL*ybE|vSV&O=J8LlfoYF#(&L+D1G9{O z*5#$8o1eqd!OS?AS*I6c&dJU=JhOJsoE>}E87%G0w8MjyX`36%DVuxBM!21%B{N}h zOj?{+nY2JDoX~<{gLT;G95L8OjrK9Kd)(}pFxw}s&Pj`7iZ0@sw#An&eCdqcJ>$T) zc;@j%HuoI9h`!4$th4y)SsY+vuz!{YGwt-^(`hF=>*i*h?39C@a(bp*-YFM1;lz@U zi8}Hh8@7M;-S=4R;2!-pX4vT+cXLw=H^cBVZttAOKkM|&xx90XZ`Q?O9hjd3!d>2Z z7q{SK7Z~57$G_@Ju?{ahrS0#tfNVgGT$H$u(ql4;$^H7SEW)6$AL4-U-0f?w%wK0`hSkQnWd! zfRhw$-i^Du{MB?zxIH(w$oiKUZjtff#<0XqF1o!-xJfs+gq<$$5;oo30xfxbgcV$e zeubNxck}omQ!d;#KjG%aT-={6S@DE_ml6Oy05_NhOL2In>@GkWwh^c%LoJ|d+Tw)S z1OOIX9Iyn~ECJqtkcVGl{qxwwk~Vz+7_?yC>tEnP3m$&n7n$WlvtB=zGn{|M>!0C5 z(_8>!k`IitzDX`H<@HZuaG?oraMJ6;W{mYuuz?9rV8rDew73VXOrNjp)9!v~<)GO) zWMjtcu2Hv_PG|jCPIv>8EI);%H!$t>V`Y{LPP4vg4?oKXXSl$O*Ei?k=lIYZ7o78O z3tV8H3(d2>MQ>n{^E|D4A+=N2{M-zC0(-%SY?*bDLiBMF?(*xb`r=R8;nkPq4b!eGbJ zGw)y)oE{iY03K)oP6+&bWRVLk`itlJFb1Fwc+dI6(_C=Y7n%(ePx(UA0G=;0<&RAJ z!jpV()?Yluhp;}w2d90JNj@~i1!7z<<_pL8-~^AA&^RBaJ)9rQF*Y#f@ekWQ{Z{u7 z7w+vCxioRHKUfh(L}2y6_KvyO3D!5x1)wz(bc%n%7n<;gC%GVfsqkG>_&R@h#uvnL zHdulioA!sN0^w+tv<4)kWm?U@04VZ5^*?<^O9Ii2$t$H%NM zLY5qe@)4+j`I%tZWT1E^SUiQD{z!}uP52@)e`Ly6JP}_9&iUZDH!$Y)jbRC#djq3h zKgO_!8};xb9^a_Phvf*vjk@_k2RmS6qTa|*UC(&yXqPA4YqAg7Ji`ul*v*ZAgy1Y_ z0fvVk^?1jpH!$w?(>LL%MsCXe%Iv0p6 zh091Ig5miUh)N@4j30X|3zNV0zUK!A0*Y_f}i;!P>0X>;HOX! zcnyaC$iqW5KH>tq6Bq@8)yAv9)8SA%Mqa0&5khqD8mmi99D+h>@fB-{D6ZQa1hQ1 zt?sDP*Y7WnvXLGeGXOe-rOh?y^o}yVVVvvn4ZFP10=gPq(~B$P>td`AYBfpP!B6l3 z_y}kO{0@8#JP{NDn{?;F(#b$%CK!PV%mgBHf+Ea^%jX28m=BlEMJnJd7DL5L@DYKM z#X#{QAG!p+-~)?L1U~plp!j2d@yCJUPXpmk{ozmf;AeaY<8S!Tr(Qoa;vz99IEVlg zVcsJu!oQM23UMn^F&=Om;xi5wuopr?pq*O)Uqt}x^bp`V|E!0bW&w3%b$qj4-z*=Q z@c1TZXJC@yz^euJ5{dbUpG*XbV_bOL7a8Yay@R7%aFq9t`@$n!a0HuzNgn|qS)Ra* zB}14TbbA5UA;vrGW`~^2AQ0|k23+i*iyd%!`W;MsL>=yayE}?ubA#~r**sCI1-N@G zu5Oc~&+I~VKnTz2umQVs(BX!r3_3jnE)T9V=wyZ+%&5yV;$+}N#z?J5FKET>rF(OG zVqQPo1axp5D&Px4Pr<&(9r%Kz&TtK(lKD{SEYtz4i5x-6d>}Fp@IwXo&;lP=@CO!s z!6iO;$xp6fDG2V-aBp zBL)lqB8o8X_Rb(bgN#i)^+$ZhgF(p|5O^3a^5!7zfq6>$@iX4wEFYc$LjW`9!;>EG zl-oPy@go}(bFt$t&p6|adH9&u2YwEk0d_G$1V+HQ$2-jW2i-8j0@OXeAsSexM`4UQT8(1yG-_Oqpibe?KC@j zEbcA~biv(icJ`W`?Ivrd$%>VziB9h}JD?6dR#%_Z(Pwo!4jyzJXYX~oh(xB)6f@> zZ<;~H$~ysgyHTZL#$8Oz;fWz9>GX^v6$z&8bi;;^GG5qKP>w-22h%;^;`#~L-T|0& zx0lQ>%sckG***shxO2eH^x51Y?9M*Bd&tQ`0ZA8Jp15yPx|fU`h)3bA&~!%Z^C&@$65&kQ^x$&H)wu!xD~{gFA&zZeKD_(QZ5z`qm> zUkVjp^oN&tKceZSVDTj`aLMaO5KVN5;g^s}LxvV9U%PwU;`lF#*?)!~J;K3E6_a-k zmcg+S;E|f6EKPXH7n%14khP{Hq%VY+Zh@lLFl7Q^!36wMRg~dl1bCE0oudw9-QD9h z*O<*UYIUICH3}l(Bszht%do>W47=*|^x27@gQfJ^Tx7s)t|*ocrpNArDTXD7S;puv zSpdjRldT7q++YFD39DLTi_X-nF|=rmEgD0s#@M7bG^=&ZDt*)Cp=(rX8&$d%wV__A zZBpy%RoVuHrdFw`SLy2H>L!)83Hw#LW`(9frNO0|RCKi_wZ2JXXjB_o)%q5l5em|# zGk56CU3v>tquXHVCjUZiqQ?rq;~-B259RXqgG9ODAfVA6DDsGlfnI<_(QtdlVG|gB z-0gu87_~V@?aoQNYtrSJaC;}+o+&qjT zO30I_FGeaTq2UWtWfnXK+#ZQ>J~U6M_~4w^kDAo98=VO(5gzCQo~F|sb9zQ?F2H^S zrCy7Dz+xY=J775xkPh03q7%okxd9$P2s{KPolLID+HNEZ3^U#WL#;ElXiWr2jlM;# zZxFEBBv-a5warRhlS0!dSJlfk^+2y&Sx4*YI)$cQMmuZd%6g@yR<5d%scPh^I_#0k zs$?o0P$yGV%ak=mikc#2t&C2@p7`mwLakg0t*KY28*}!JQm~R(LMgpxreD?qsuLIpT1Qpy*(8 z44G{s7TcH=9Tbk39pYkQjAshf2gW3;;&Rztkgk+>NvEpK>0E+4) z3_jBGfs$!o02yzXdK6eE91QX~)Ukl1CF@7t60zH&m$E&S(!}vl6FxxXmXe&IVXJ+> z>Kd{*2F$j8i!*ArfxYxton6EzoE>IIyPhaVi{9J{oNG282GaIWK7nEca3pS);}`Hayxc5{fJ;cM zp$-nejEtO>5^;-Eo~G*aMJi8IE)K!kJj)~V5cBx{D`EWi{8q%9Bdi}WCV2RS%L|-S z72d_pdiWW}KZE>?iyd>g#+`7`9x8?+(FYsCqNp0AdJt&54bO;)xxw%`fFJpAVnld4 zNNgfxfKs4V#Ya$tof7J>M3qgpUSPmr>N1!*K^}~@c7v@|Zzd|PF~Y`zel!r42-{@6 z<(dYCwnnC|gMF646srO0ZmX@^=IFJ%VOo3a2xA?c7F(;)T(310OXX~Kp5b)5^6;^O z_dm?~-n;2DOORf9pG$-+M=P=!oX{3G2BmHn*@C>ZZ|mKwn&J6nP}j7z0!RjR?n+EkiWjiFPAp$Bv8G?;r0 zmR^IgS7+$go1!}7fWb6kwhh@`aFpX9he#{AKoMQzj0gCi@OUO!4qAW_gGPARX?O>( z7X)b<{sL7MJSnSl#^PK+ZQbTVClV?zPyyI~gvMyeK#aVA3N`BK-oPB5i|{nZxX|yx z$A89Mqo^PGBja8^26&^;52AvwH12Q#W}9oA#O}fVn9$7_)fzgL>Q0q5YA}!3-Jt8^ zi0075g%imLvTTclsmb=D z;mu&Y6v4!!vD z>yJME+yf7O?T*{;yXnSTckkG@EoI-<$mLOam&usTX*faX~(vkccx<8 zgt22AcHXjc`>ne$cHpe|xP8yA+jsB0Y3KHvcWm3ged~>>TW{Ewa${-=uCqUN>%LU1 zZ^a5ex+!(*&D&|^D|`3cchml_-*MZs_uv2KV~-zw<<;Cn-#4Dm3P_7;^u{(Mi#%K( z>+5weZ5lWZWxb4i7Jg<6f)OQY1gV4C(5}+8!%Zl)9V%^?M%Sy=_vuaj26MmJGHA17 zj5yuHcKfKqHR^PZyWHb$1|9(#G3s(oxCNR72Z0qA5)0^Awa3i1Nwamv2A!kdCAA$d5z#j`sm%xby}nwqkf9 zEfk!3(V{bMLH8HAU9iwG2MmrEt(b#w$_8^E7>Gnxmz!Uom)|8<4(iP?P2=bjqt;>v zdVnz?VPtYfk)p9Vhp+^>C5nsuIpH>HaiX-^g-}svYNN2!D3AmdY-1zEnBdjgIw2+n z^XWm%YjbtkoONm)pIab3bokiIufFogHy^v}uDfsCw=Z?uo|G-Swj}ON-m+&)((WyZ zyOWc4ByGmnnY4NLmV{kN2|KqW>`hMEo1A#V)?^ynlJ{;+-jk99jfn3d1ra_1ux})U z1M1su+z!ld1ITgwKIqQYYZgzIDFCMd=B zZCDDLlDt1PW&e)UJ8vNEc4QLF*dNam|(K>EI!>}FQk_gsD z0eECQtdtLgkLWd;yN#AEy$N$ky1{!jWaNSQMmYh#5y2yfO%WniO*hD}#SRWoAyZn? z(lg$E`>jVGd+hGJ?z~~|P1{p}F#u`bw&Xp@TYzm?b%HlRJuyCZB*r&4Z%^EWkK$T@ z1F*6^QP{a90cx@%DFGOUUhLYEfc;p91`(>06Za%1VSOiX4uzs+5)Q!WySF68ziS75 z*JhlE({UXfyek<>hZDD8xicA?Nze`qD8>G`LeY{`Ptb^MxPLg2{ep_!wrl5|`}W*> z)4p%s^_91tek$X@ySme7n2hWysjNkzsx2t0&dY0%%A2K%rXod4k)lKL=xCXiM_ zm@bWfmGjUTYRHf662fL1Kr#y=tW!$ z;3=T0`?jXg3Ms-CXfb}#o-e${zSLwes5qMfCA;Uwy${{8{~KSv?fn;DkbdxiB|%?a#fo`)vnTXsMK8=O|MSh2iFWOa3GK59Jb+0#>)QzYdtv9369J+9v9K#Fj{ZbCRMVe_#Q1E>aQwMc=`OVu?L7YInutX&C<-L3Ezj^;_-~4*sfdihi=jw~(?Q&&hc6M!E zL1U4u4Y9vmiCC;tt?kk4dyGbSgaL#JW;2lAZ!q=ijU!s)xYjhTGYuJxqbBPRJOUmu zn61q*;c`zfEG1MuEXwhyNMf=N64a>0gFaEGH|6@6S!!INjznkySQ84G3urp^f;CZ! zo5BRNsqi6mqzwF6ieKs-2M=_4 zhwaV*B%)EHF`B!y2)H!e8r^`-)Q4(Ok)kfIpuRv-n_pO)lUJWt*ezE9{CzO~I#a*a zFsL^nB7mtMLh9aX8z4IB=r>w=^`v105t7`tXBWuQ%U^$3cIZ&GRMuy+_Zm#iG8x=!8?->7>QZZZ;0yG| zKE0t&XN+oeeH#6UQZuG84#8gOO~WP&oC3-U;7emx`#6T3$kK$vP03XoCc)V-pkgf& zS=3|&S}^4YJs8uRe+or64lBO6YXA#EO9mD9d6cAsSVx$ElsM`{<6ICKuzzW%@b33M z{O+4?<)1j&f(Fad>e=Fo`QnO2LVhuNa%ceg$X?RijR5RCx&;t{qQZj4ts&5Rr)St~ zjUwx)*0n3u?Fv;(k-Uw#iZUt~d06;P6hI54Z4z0xTn*SmDaZw=^+*9hyhzaXA}L|C z^qJu`>`2{rnB29#%`?i7S zfZLEgM05$0C=hQl_<%YYd;uXq5%vpV>=ueciKvqq2jqZN*kNIl*2(e{v7x0vO9 zIo3h`0YYGJd*a48GI2aXT{t*?DV&9ag`Knn{>hIBzJp*-`a&SbVNi$o=~&XY3#t?+ zW>8NYL-&dA0y`3-W_*Ohum`~c8Grl)K@sq);UJ&__@(&0f`fns*n;4i0tbOIVF`7B zcc7S>l2IvN-m~k`yY9?>_uWWpv)eFnpj$qX$RFquY7)?u@46qpAGHd<*IEO_pX1=5^Et7Kkjd zypYrO&-#MUh^(!0>AerOky!^-}qV&pwO12W%J64l<7zFrG_<*~b&7 zK~JCXK!!bqAR)kjo<3MKKmxWeK!v;jFo0|fc?#iN!Z4sHM6Ut@p$G&yKoqEp<3Apy z!d#Oc3CB~Wh>$6a%h4+gass{@S_2sGLbOXdLS~#S|7F!f2uz0&&IP9+;F%Z{z6iG= z+$S+2+9dC=33r11;5_(hgvKEJlrJKvQ%*_v)uhF^f67n^fX51aoWS~^s-z2~JJ1Uu z2Zitf14aXKfKz}LKrimvxA*L~-|Vv5;kcV+idLx{j=KZVflSdMS4HKj0W6i8UKlQ9 zkM*!!7&5l03V?JS3zzfQKQyY1C*zG90;9^e9du zSVk;N4SSgQ#?PYYCTIkd0*wJ^w?oYcljG3!ib+OncJ}+Pz4qa|@BXu>Kz>GMK(CL5 z%jQC5G{XWBI^q;B5>}y)ODaFEtY0dtLI)b$R?N-y+nF|`)TDC28+-->`Lw87H*ByV zGlrxkRG|b=)vK>wwF>*Of{$o(^(ri}XT#dHsR^6cuf2BlN}RG1S6hAMl{j(rl`FSw zTrb#mO8XGJi3PI4NRl}{3`L%*GrBAMJlT^ryf+Qx8C2|=QCI~sPr|!KwoL|^u zG&K}RV0l_Fq_R$_qPIxVCs%bV)Ln8_uS(m8gaaZ4l?DJn5(8{@P^pb7wD9j8a36xN zZ+az)jfX*TXE;69e4`>ZTI^(F>7DLg2<~aB!;S12tjSNyCgs~K< zxEMy$BVJH|Da>9l5LBY4_yFqZ(xb=DeE&$snY6!H3o_F(RC&3zDEUPyW(nazv>ad- z2}sTyIt|bnUQ%_jv}ytN9}_D){+OEwMS(OL^Cc|>MX&>Srw5Y2^a2FT>=u?|XjxS2 z0DI&I#tfD*qzDX1npp;+`)1o98WNDeq|PogEIg`{omwN+@zLml9HvZ#bVQR(UN4m) zJ5iOJR|P$iDL#1a*_(Hy3O1iImatcNWWYe2^a_s^PdU!*i3SURp$ARo@$x2_O#yPS zu}Pb+U$-MANtvBl$~X-YiN~mW?fK`fS+)AETW;dbdd_UvaNW8WpZMm9gCDM3zCxLk zWmU+6c5BFK4>PXoue&aFOH#4h9<*D07UO$wz5&=j{et zE3tp|ij_EP)yfsPEcQd864qZ&2d}~@D+L`~g%ek;TnU}M?rPHCbyr^vja$EVE$J8W zIwEFRhrSDu0#uBWuFwK}N;?J02jRr62P6hX07WDfAXh@30%5W%J-t(-t%D!WFKjN5 zc1UEs5_wdn>X0eARoY&-Y%Jjml$sv7iUukRg|gT*bL)E^T4Ft zO^U$sm@DVb&doh@DktsiU#0?@g1lO%YbpfGUy9gv3XS(jPl0g>Sy}8v2(VC6iRFBx z3~vj;8*+wiE+{O}j)BS)y#U1SLLV%8YthUwf%2{y+|PpS{iqo=FXsrBX2+n>3RB+; zvyb+8452Ln&5dZnK@Vi3LRTkM)Mn?`T%fE@RaQ=QPG04O+_H?!x&q1j&pwUVR>$Z-X3a`sGF)&zgAt(Lh>4Z(2 zk~XX#@98eMkbUUDfq~YR_rCMiKugo9!$(gaIXcwd_VxSj4?69q4j*2*d^tub%l6jQ zg&5D3D^{4~(rA5MUqfAQLtRHzrBkEYyL0=7b?X|!fnx{WsSk&GYpXpbGhojebltU8 z=T97mF6`dA)n~Khr>E5h{C$lL(Z+=gJeODlb zHvutj%r8Vl&{Rl~U<>pmUxFGCE(MYRf`mj0ru>Y0ILucPW-&x&1DKIN%)ulYH1;D9t@}vM@9A&s71vR8nHG%!Mmv_%L1?1GSpLyb?s*XtSZIc0tUL z#pF|(IapkYSw$3s2;Dn`c51N4D`?OV!cM!<%)r8bO?v1h8|XmBrO=8wN_}W}0f=<~ z-8ecUB!;Y7hro8IKS8B$mMZHDq_z3d>YRd#tn7+&S!HK4FiO+UmtuJ?v-Ess{@ZUO zSpmXBHa{NH3eOJU7nm)j!SO)J5eR8~IDH%jI|dlU-Mzr5DNiUocZzmXHf{LYJ$J>T zy{TKbF*;2w8nx*Sz|vHI|3qIj!ucFpwN;~SttyWV3}mLIk92ijNI$nUIfUzxH`K4b5=Q1!GQ@Ux+)16AH$kltKsuAQ?d&1g1o}GDP$Vd+FfB8@HxBdgmRL z`I73a?An}s;2Gu`beCw1TnP%)C6#wdWX-vSKz>7ZUITIz7jkIiAXArvEFVrmWgoC7 z!hjX zg1{tRO#m_mM}Wzsc;SHXMjZqQ06f}dQ4mL77i~H5PFlPJ1oN%XJ%YSF@IQpKKe}`E zM&xih(Y~utHY?Pva!oUAMZTmqE4MP^LODQwHnSw{eB|`mNZPs3>GbgF^y0Mh;k1lM z`uV`w%scnp5KmAD_@}InzO7e97pM3y9}}lD2K$ zeESW1#(TQH_P~Q%leX;Lg+Ml`03a+aZLO-5psl`t4WW zm>V7KZR@HoDZAs=+pk`|+NIMpS5|U%$52Q6se=ax+gg$~ZCbPP%G|Wmyxm^Td&jyv zL4j5;U$J8O3az9t|6F=az<=n#duP6X$ZNG+3DsD!LY|e`U0;iH@4fRb=!H&F=r$O# zPM&BkD|zeHSF=u^gkE&l*B$)cJB1k;E}eF)tMm4o_OD&N>fw9ue(A|4MmjoqyF-?n zmztcsYWec(uenBmy%6$1GK6`G0^&Xqasak>2nrHU83|A)YJ`tKelB}e90mmu3UJU1 z&`#tWnhPaWnb}CeA?O2E!C{~UT{0#5+|ZqfNSUY((tUWcg_3#{XtML_a`PJU3P7A% zC9)<-QH!Jq*`(GYd8A!Jv5@KpOgLkTS*VXc)W`&qa*IL+=f+XJ!^46F#kG$Oy(%cSTtvfANBes{6V}JVK!V!FBZW2Ksf)j7wtV9toGws@fd&A&izA=;~Y^af;y5dO3S`5^X^qUL-l%taF{MovL>RxXTu_{=%~>9ZIC zj8kd;lV^O`Jae|}eAair`6$tTD(FxiJD$~yn=il{g8&~-^?acP^c;mOhmi23%#85d z$;3ey;4%;<;OSqpdKG6jcGlFq_0o$s?%aOIjW^u9dv~n2XFM7`@Xi64)0v^6*|CZJ zdvBN?9$K23e&(rXTR>M#<`uGp58)Z18J=J#K-X3dEY55kyN`MJiD;;y>7!H#wS z7$NiS?b~}B>JPl}dUt(2XS1(awJPJ}$@-GwyS{Q4=upm?(>jR+A9Yk$v{jZ3v^LB0 z@&?=6qm2z6)s=0PrKb-cy8fE0VcrE}4$KQHWDUsWVH3+6kt+}wBaXp9+zsJCMDhKw zEs%_Gyl^fVjRY8z`);~{O+R0mnO%`_p(ZN_aBVFBftPj2l>Remf%6yIHTqH3I~T#* zV}j$Xw@X2xF^UC;!)*jX!zXcl@;P#2Nm?8SZ47?Z~mO?A^V8dnyb*C9VnBLZVv$JZ!$u zR6{%Q5XrwM(>fljpclgPg%cMNAOs0Yl2h>)*TECtws}*+`t=}g|ZvFb})}^LwP2IY6Tk_VcRW*!xTa#1p(e+nf4a7nY!jjUwf>0 z52!4V78J>l-vd*sx{zI+or_LL@*24X&>caE^t~{ldIQR-PytMfrnYSddJI7R=`CSs zu7LHPa(OTl96gP)^XKsp|KBRWm<(aNZ6;DNA1p_G0aOb0Xbg1kplm_ms?mxjG|cNn zAsZAZ#`)+yIe>i-Z^((@6-;<3(~On{5EaLAxDu zi>UpB_mA4$V0Vaj>az3wxmm?-TczJy9pEZBk4>e}egCNE_(}Ho3I6yg@6qF)6Q|yP z>dD)8?*Q|MUxwYL0%Csw)XfV+LL3a=x zU=vhfJ&H*Y`@ zdF$qlXs!^J*u=^Pp}r=hJrmGzvbNDK<1xv{-_-!P^D}~0ySD7n~HS^ z0laMwbX;hPKz%`gbP|pyzzg*+Qv4KxJ+eCSemp^F^z%tdg6RO};|>C+!#Kci;!`vf zQA$=QC7{Uo)tmR1K3sGD0*S5OFM+wXApF-)_`SR#{hxp}_4{6KDg zql;~Fdpa0)$QQU&S@Use*_;Qt5xfl;MPxndXdURafNM~yTIFB`R4uB^$tlmuuFT3t z0tDeU_E3>jiL9xv3%c-ZZA@%dNaDqkcvb@1$hA{gGrsBF_-}2&dJsm{J&g) z+ic}+)+pW;hHR%d2;+}z52j*af))DhhmELESVpX_VY5T1SXsdH=`A#5ujXJU=zUPWF&6K%@eI4%BbXwANu%NLXO%Udl$1^L{z03qL!)U{D4UQ& z&Mj=rDF8EU3iAymp@!mMeI!svBdE^K_MAKe74RH6=01FsJ#q4}JMTdImXMJk<1g54 zJWU}wiseWA|{M4Dhpg{sRP4njfs zpL$2*`G4ROCKvg9)V8QDK$SHb0;&_XQOS(T&}bj};!~knNT|UK?mw3{!?HDyKM9h-~!+y)z?2 zo0eW?b5^_9c61P-H_RXSxT5moVw6_>6PTt4r=U=xtlX|pHcBWpR0Hs5WtX1IsK_Lu z1^BJgLLsccpv?(w%*e;G9Ns4|=Z~S3>!==(P=*xiJ(Pc1p>QU8FqpPY}u0XP@ zEtjeWEoymxQ+;b?X;WFGu_V%17D0lMb2tqL53{GzJV#Hs4~F2APTygkom#I$#mc&_yR$6 zxDJF53Y7Kw>!Gx>;k2`e0!lNokgu!E$sPCkW=l((39Rht2j&kh*-M0-}y$7P|Px@PaW7rTF#G4N`KPSe5t&00*Y_NtV1J4hfdd_H?(UpK~+sTPeem%eFsv< zh4L;mpDT+l4n?C)_3c&VZPgWBb(OKM_6WS(SOiVomlbe-Mkw2b!~3)N?`x&mCmQmC;=7qxqY$p-l7C;WYfZ`Cs1s#d^*9nbDP_8)V5lB1mjAQUa2-pHA z6eL4;7#9V82+RU*8`8iJgwOzKs0KC#3r}XBeme<-Z2k>f1vL>0=8~aUJL##Zvr5jL zD>LFKaTNTP4ouN&s>cJd6I5@2q zVa|}+0OX@MhzT!E(xOiWI(|4eGt(EH>h1byeE5^;Nvlj1$++M=n5cI(-Xu_R0QxTJ#n&fLd=UtkAndW!nqJ5 zU{Czf;~0VW#vwsFVgHeF1?dI&(H=%kq?eh`mc2lfz&Fr)gK-(LNz}s}H|&?0jL@+I zn-ACn^XPvDZc&ebBH+8wN`y_24S@y4k0I}XP3Z5=q)k}jFzR6tn*ZrKRMn;H2sDNe z`)gf>9H0dNIqrvk2oV@c4wDNI01v&6|709VX#_Iwum&V6{8;D+_LC_f1*apa|A9UN zaT2s6B?;Fg~Virm~#cDCa@nrDoxTIV$w8(40(t@E?~ zp-($Hr^?Iwylfw)(w9|!TwV8RQ_Fl+&Bex+PdmE*rn6_Rx~|J?u23qR=)KVy0y1!xF? zQFTIf3~&Fl=Hz+ImfyvHDeI{VqFxY*>kH;=;^knNQ4Zse*Q1T&t;HrASQ(mgaB;MD zplKn-Q1*3%f&GV$;Jy0DMA@Py6B3&UY=@mraGO@Ss#S*67V=uk9*wpW1#!KhN2P5; zl>`9}8bRd>v_gT)brew4(~CDJn2kT48UNJ}Kl|m6KKt>~LQNpVX5^G+=kjOLJx7nb zjvUW^?G5zsqrDB!1<)leAo@yqX}}v#7N$w?@`QXccl0UPJC(x(MZgL*#t3jh|#InkRh zpo?`1ErE6+Y$2i|u&m2BFF67U?i^F+Q5& z`c>L?i5!KGUbzO0t`!8y@2~TCB3gZ^M1d-B_*@2eDvdpQ(thZ0`in2#j``yt-U8po zvx`3tvQ$D22Y`=z0Rj22_Lv4mZO?ILOOFp((>VK$+c7*-f&3HCr~PF11uII(!5(`0 z0ue$(H3$#B7t{-;4FTjN2HdXkKwz=Da;BtYG+fdaEUt92!?g_;`v!kD zKK`?T;h)B)e?A`j>CngzqWwz^O-t2vpENXm($qQt+RQM64D)~3d+#tQ%k%Bu1w@1x z6;Pu|tO0D;OQObV~&T}sn4AEbzkMY&TFre1Y^@gZLPPUuv54|8Et9l9T{1K z+h7Ld(=A98FjfWcq81hzx|Bj6m2P_GCJ3@9{=5m@6`Hd;$}9D?wGN~H%S)iyZ0T~^ zhbUk%E&+-iwPU;{_6w@LR>yXmxJ9^L2(F?TNaah^-OT77vjim71@ji8X+}$9i$ZXf ztLu>02$Yoi;Aa+_?@8`$sA&Mk_1*rDbFChW^URzEa3eWy3{fLq%|+ zDu>IcfXL9|n5b=_E{vLoC%4dQ-N@>d`V{BYeb7r*JH0t3=7F1GtigGNm1nY~nkx`brO2aujCEhzEg5~zYmkibuz zD-a@<2CppM@kiV*fGVK{>`5O?gu}tL!^46Gnfrq{$>2 z;Fs_$AvyTsykHQD0_S`-;kStQ0 z5w}nW#YPZ(Bi@BiCB0qnui{~>8ooOpl@Ga!^yy_6-+cMXXMcCUGcDa58*5KWY)VaS z%g*r@7Vdzk+~M|Bl>23}fx6lqq<}iRUh(hTbwKN*PjQ@6 z5qfuagzQjpVaW*aH5Fzmw6Ne)sFY(Wfw3IPo11qCof3B$rr# zO879+x?@G)Y-25XDNxM^0NA`HC*TL%247$L3McHX0PR0v$+jZBMG$@xye0JV0O=cv zWhEJ5;h6($-jp{IA5Q{5&jp{-7ZbY4qnQ(c-jV|D7%L3qVMFk#c~=F<9H6tEy%;gf+ETs z6jSgdThcQjJUEh4&9RBte_d2u`0svq}23@%Zyk{CBxH5s#5hmM`O6x@Vd6MDZv7 z`-A1nsD=HI^X1Dp7q258DPB+<=Pc*FxL33_{MX~WdNzT;d_UxRV z;u1%0uD!I(Ay?{9t8=(Ecl%!3`TUXn2R?jm-}_rPzcbMPPJiH&iOG*gM?adJ`ebta z)9I;WJ9d1+E-*Fos<&sSt8LWP*4NZxtygrJ%>7W2t#-<4APTcD(oMKP>6s)#VI0wY zLppH`A|=UEdSeCt5z@5p6k^vc`|VDeF7baT2cQpWkjdP(ue3~dH#K{m&Ep7+qo_>Q zlwjG~MDS)JmrG$H+cLJ}F|)7U6tURHr=ixD|)!=yqnR9yqq1%-*SU2z$h z_hbY398tW>$>U@5;k^(A%PPhzs}Q!r)rTTW@&_-DSiV2M1d1%k@5w1}XXbXKXS-6; zVgA?>pgrLHCpJbU=p&-Ses}+kq~C=~iSX+A6*wdtC@c~OX~wh9JoBytj=PhT~yFz$AECJyoiH$~D4#NMrV%5|4{KlS=Vv0|&B_=kdrL|>e_Y@aT zQ{^```sy2+%E}u{EBYL*yZZ)qySv`ny5sGwTi@Hc^TX$!`}p~NM_+vL*zUa_j8D89 z2)q~Y9~m6@aCGd*;NS-nldlhsY<9JFH7K;%*^bikHdHAj`wx)X(lT39NrYyR1p;h> zHALaKu5KKW7PV#u=Uj4Kc8DQ9t)zhV=uIzK1T_v>SAAt=r~Rw=ce%W-rU`QsWile) z=XFLX69l|SA`7{DwB^lqUUQ3~QOMnMi~2pOPO%G!TN3^_|L~E9t7`~&LB0>?r?`ZG z*H={3LnBxrn6DhAORkbAppxLon_ozZ7*e0;dJu{w6%@7m@`@mkbmtbfXXUkJ%Z#T=j17@uT}jx$8YY9gaU?|TUu#hUx zD9J#LdaKR;oZIt!kMHH)zQ6bM9`JVmy+3ehYU=%+yFQVUKOgTu@X7ucKis+F{Vki{ z>GvNF1dfc1y*)Veg4?~f-My!^?fJI$-8MU}MGtN>E<{#NYib6R;MO$C!MT1yU`eC} zbw}!|Q)(?}++wzpw?Q~)kEr1`VgKPLWB(_WI%Acr)A1Gj_cXhDnkZ~J5jBOj3ULmJ zHi()SdBLm`+Tg~mLUqJ|ofkuV^ythpngn@mPLS6&&1kgr=aSO&mz4F1wnV~S2(Bd{ zA7=(;Ec{rAOth|W>*ZVI=29Z(k^QCRfS)KPkXJNB2wYm(TU6GOThN@AVM$1`CZ!o- z6Adx(x~OPvc%&{WRud7u@!orHK`n3v`A-Z{a8njO&y^PocZd_Rg5W*|Ytw4Ip{lCWZvC?V-_zXWb-5;yabSK21b!$J zrH#O7o&os=)3lyRB^us(prQ)r2QXvaIY2v~wp$|a zb>(8%^TB*qZXSCUawXn^A~?_RpP_Rx*V+>Jo+-f3YH{A0Za?U~KzZ8*)Bz(~A51J{`=HNd%Z1LPj^G+V$@(LXL0Q@hP z;^GtbKX{R)s9EUCf&VA&`X6gjk})RM93Kzu8ULrVun0=n3xS~*`uux5p3SZ8TU;$9 ze+Yix9UOXZp#SjD;Kzi(Tefg}WOU^1k>OYS{4e|a-kh3xb9(dZfxyAhiB|@P|2{DC zg1d98$v7^Tk5<T@qSPk|Ds;UNSY6t;9<1}6h4m?xapePf` zYW2eY3+^%amwnCu@isSmn_N@E{xf69Du$KgK`{VDNDJzi%O=3jAt9d~VOnS4^(kRb zA%47|ras>(%QwGy!vu!KLmk7zbvA{+ zF`BAmp0C<}Z%cD;vuj4+-$s+KXy9c$iV)SE&bQL(D9hze`9={ zCNk0x6KzgTbY^C@=jHX+*7+Oc&7~!Z^bA8;RiCqYdzbgQo}Po_Q-?Nhd6zuTu3blV z?)Y$O>VuK7_XY;v>hqBlIy^jdcz6`r*xQ4{FLZf_%{F^^rJu| zCa1cRQd(i0<`wxX5e8BAm6iMHAb@NG@01Y0Xkls_c^mY`ne>E|ioieFOP$FqtM0Ua zW%_`(#pP|r{-ePN^QRHbMzr5|=*`0aftZaW#QZ^n1%YLp|7qb4Aq;L{rr{K9AIe^E z+Z2l7+FJG?Ur8C6e_~z)sc`Sn0T9}cdgErf0uPOhP)jB$&5V}JjP|T-SRy=K4IIKlqka@FNe{^mFO z*jNHIQ*5l65I8-pEjO=^WMD(XIH97$v9qOZy18v@Ysb!Z&tA9tz`(#u0oXocZ;Xu} z-nr+)efy8>dG6gUGj9$A{^9L@%kO{7A9$^&XHUC(TT}D)rluVff3-SqX_+wo*|}7H z1>Gn)1tL&aeo23gYM@%~6MTcpex!1s7HbX6N7!O@ZDrUQAuw}kp@16Adl;IaYiz6r z|6hrJy)7;H|5MV~AnZTHW`-mpDvtB>zjVb7BtNaVK+}Lx`3(J5)0~MKjjG zbgF9BAog5n7#0<8l`D3tHFPoH=aUcy?JXG@M74b8j_jWA=<__>MJS>6qL|3 zMDV+(&;8u=)P%2h%-b{G+r49SaL?pKabk)!5&nM?#V=JvWGMJwwL;=vO4TK)nj;le zMVXQVzA^YuiCR2&Xx_==TV8?B56!=Nh4BA_p%&IgN(7058v8Hy|K=-KK5_S5x|kSE zc!WMC#uOK4O-^pf&h}PTj;pmjl@(r@EFf=~frsGfKH&AgJvMP@Xylc?zQ6nW4)}bp z_ye!{`(GOze0gyA_3^2#l0n^xIomx zNCeH8bsxR<;u|mSdwJLPgU|0d{L0^F`Ue}*vm6XSNlG!s#F76Az3*N)Cs05|X*F2Z z7kjaw!|~H_d+85F{wFPJ@$75!Pafa$3VigdsN06dw=%o6<9w*}bo)cUN2cgt^gMQ{AsrZ*OuD%pU0Rz3S^b*gx>j z=-A=O>4Sa!ulM?1?C##x-tj_L&+|Rr7yNrY zGFMl3JHKxKdtJ@b%uHnbC9_x18bVr}+Lo3g7fp+dm$h%fn`(4yG1;*EOclZTXDT9N zfW}qoaV7ITfvRUXVw(xJ3PkPGU_%I&73nG~pieTl!U z41Y-+fS=}f61i2?JKLQfzkcB8J8ypS=09H8yxCq^)0tmvOUp1PBpRYc{xkgk`)^t$ zrmF;#pTuJLJd*z}c4P_v!vD#LUOe~r1t*Vhc?FKYyx^KuO9k>JN(K~YmO#+dU~Pi_ z|1-b+tu8tiYN9qO+7KJtn2hm8O7=#P-GAz)po~@M!)*8@ThxFDllVi%-Ji|a|L1S<*^b-mkW41PHjQBrY zU(Nsbwzm3OTBeM426o{8(Cq@@A9{lrWsVob7+mIbW9Kz<)GCXCQ3F7*@?xC`>h19U)gD=2Kw$$|DpS+I-HxVQ+;Cz}OT^6uhd zxFS80-1cTV79b8J-eEAbS!OmUwMsY}5n4BmYDDA^s<3AN&J; zj1n6GHVRG}n*7-R9PIzTQ%@e>@(RrEJN1SumvNni!!KHXLwbH$ z4*W;Out6JR;|#b!;2(2WR_?2>4=7dMx;jsF4f@Q6>>PVZSyN4Ivr_9dnkU=b2S&zU z9UgscbbLpVvFexuv9x=?O5GSOp|}SPt&ITu(thEfSsr@qlh=yuzqV(hY_P zS5XaFf}%E|1OYNWo?FyAGM!1Ojj{2%h^T}={^5p|qWzY=n7I`=Kl~8v$f6xsO4HF` z3;y%)|Mx6Bd3@U|uy^52S1skj5&{e3FS-3J0lJ)O`U zv%i0EPnVZ`5!Q8}uFegGxV+p~TQ{Q7_LWt%X5~1NQ=Q4FZJ9Y9u)&}oS^a=q->bIy zH1>eHv0rT*)HV*AY!jBIDey0`H+Nbb+r+3uM#VKU=-FIb+uigP`|oXS>1%Bp*IQ7g zqVpFi9v(@}3vr=oy?NSTnKW9!|D@5%dK}j^j%xHH@UfMep$5f(OeQS93^#_!n-$)& zQeRnFZ&`U4R6Q_XR0PgR^#J>h+}!ruTyF7*I`i`c`GE$RswxDWMdDPb+EP0!MfMQl z1QCH36)Z+t5Hb_*3nE;KS;BwvLl1GyDFI7HN!TR{Vc=hc10>}xxhQD=3&{WMIQ8W5 zEw8}r&Qou`TKGTEwlVvX@fVg~B!7e&fUy5h-F-KnKU;t)E}jG8iY+{7HZm9n)tcU#n$`-LzOd9-T49&fnk!}POv17^ZEJP! z?dY5|nR~0NTj?UHuJxB!wq#^El4!9>cW37@vn)_96O}K8p;zPV)!1?V`?cmlodx4R zVRcR6|JXzbOe4%z2F8JZjJZ;4uB+?%YW&;N(&BGz8`GJ$;U+N_oUu(X9LRPWEUej? zMjP3rDPFVDK4CCVXiXz3JRdRKnW#Jf=Le1lRJZanod4d^Qg4ayZ8#wcU<&~M!sZLi z=ZmRQOiw`Fs~p|_V*b=wP{1wAgqc&`qEhyP9x_O*dQzi+ALB1nFUWUgIN-v?C9%n- zJ^YuO86HXtyvX^8bc9s=g^?)M0t?~7^UuTn7lHpR3r`;3@(RpuU3l}=XJb`@4nH1d zkaj3eXoX%${^#jOi5E9>R4Rnl+|>rG~;N;Q9jVo*VzL)>yW`VW^S6aHci``XKclT#`H))$(xt;tEHs8=VW429M9jv1%T(U;n7~`Jn6vI{Y zt;^8u9Fr>p!gQB`d2a~-#^x6ZFqf1toIg?`TtVzKfUne@hz{w?DWE0SD%f+C))t#MGL&JGg1mv5`U2V|NGR@ z(Hi&3zGT$!1zIXP`PV)t(+B|;2XQrd&CPfhJWy*yA`H>6bI zCUw`zyKCybbq)P0ZEsC&ds#(eenD+&Mr}s6I=@hzk=+EHCa-{8I*F7%IQCU_er6fh ztNP?xpF-zTHTt#ofZj3G*gR%&P1>5LZLZB0CvhwO{|xge#Ry~bgi>dzZ|HJ3zRZ8~ zwzc`&+eWmFgbBDw_&@mn+l(fW?Xfs!Y|S(FmPuRFWTSmV%;c>gQ5j(Ofw7NBb%jiT zpL|YH34v`Vv^R(ilz=%1YzvB!E(g^f+SWw#3;F-t4orGZHW4vdEEd8)58A=<33Yje zuDn7le^+i{7fT?&xFv(yS(Ym`6a5EmR802Y9u@gdL4gwnU+7IqtAOrW(ZhxOA0o_W zo>`9nKfdtf@olfb#KPOI!T&!2ekuPWgfFl*kUwd}W+=>t`+p1gCHo&wr4kApo3&NA zK-p&cf)N6x;-8a4QLHtag^&Y5lzd-*{z9+?Oq zI;&*Oxp~Uixccb04qB*k^ZkfLL#U{%?yIiz)hUQ9dT4v$3CpOULL>&?ERGI%>;_E0g?V+(O$geN08JS2JWe7a$4;KO9)QLqr0&l1Q7 z$!(e0v`DvRWs%60R)JU)kOsNs+fm?fi^u09a@&@kC6-K9elsPoq_m*_pZUlmTz2pu z%>M{mL=&Hsl#t96_WuXqzbb0U+{mJn$G5x!v!jb{yY_4>Kl?1^1)qmo@Q?nC!2jhd zm_$$iyAJ$EM@h;5B)HHHNehKlfc>|oq&6ldS(1~TX_-ygc}>~5&DpGltmdq&R+?M0 zvzxQC+j51E)Rv#uQB>GbRN|&>L{@_eP)BK5OA$7(Fj!6kPQK!DKLduK*)Zvze7mg9 zSKH9npy*R*0tCMX$Dq+UY;GF2x3UGySZLi6Gr1)En-~~6(x9>_mLkzO>ajZFf!CTgax4)GLK5duiqKj+O%*-F{g^Zw{oL;@+ZCdK)E= z1=ocwpumm6&o9L03-qF4o|YwbGG<}Po4E*S=`^nip%{pKD7rw6$jofX&LO&|qTPnZ zU2gtZt-QCmvXv)N=_65)mSIauCI7F9h{}EJafx=gDj5F?E|wsr3xXnMirb+5uLl1E zi%uTj_6iK1{_|_k0`nw4Sq1-eenc*a6c9oLw_bh4hTq+0jPz5$7B+v+EPD(y9I%!4*a8%d&8k7O>uhI|b%)U=U4n6_z&PepfZAgrnE4IOi=wzmMQRWaZcG> zgc4JB*N9wAa)x6-t{c|by@eHB+4+Hd;pBi=$;rWo5q=H|0Ni5FMV(qG_Gh4v(UhFt zl$v2rN_D2BNk?iEEbjD7mWC@mqX}K3%q%+ET*+w^e3?7YU#)bdWfJVSXJm5$QGqr@ z#eo03$DhFeVP37oc@UVF7J@)N3xVUJ^M(Ik1OEFKeYb~y(HuwcU4rLd+;Zpt{;#ur zi;nj#db?xMN8O7!7q9Tega3IV9^)5B@K5}gV$FRnW) zXk@Mum=Bf!#K>T_T?#SqA3}fsdsR%VDkjUF!)J0LLygbO-Mwt*p`xJ zNl7!r#p`3^^l@=8B`6c{FmqC}IXS5@CB>eS${t}!N+NyAb6Gy<9<)Otz$MNr0Aq}M zV%*`AOb)bPhFAZEywu_vcArKCHE zl2X%Pj5OmKrDruIr!}Q!Hs_Z%WfuuOFzb@W`U4)}ViN(-kYFus66J zi?A1qbd{t=c(D}!)(@d!gfAYimYwlZkt{Oc*iyK z_v*jHulD7OHtSvV>+8?P?IE{-!Dk_eLI9Z`;s0E6{mM&kz4~(GxK-qTD1pVs>Ehyy z31SH}CM03}saiS`#m{X`NNP-AT_iRpCi5`j1x-}6Dmsc?1MZwQCdLpGYlsu{AqzTM zDXGnwS;8N{CC)3v!4TjtDeo>R?Jg-JuP&9A@;#iCmiJUt_0`IK4JyAXr=u?xfhuz-`{%F}*v?fm)kLJl`C@)j=So}|An%z5%uPIQS2Z&!X4Vv7tA~5XIF}sihWo--IjjZ=?gMl0_lrw1pudAz4o?c+;_n zxBAs}3ueE|Z};WvcGnGO1^xe^|0B!}$xmVd?EmE}=zj@+@b~J-NK!yr@E;qii;u^) z8$f$}0)V$BCG+dE1ONaPOmGgH9sAEJ5GVXMKEa4~N zKL?^55Fvv8k0Shga~(^*yW?2<;U80`0l>x;=6A= zOE4dT{qIr=6w4FL|4@az<;s;YfBLgR7<2F+tBQ?Q$Hi)30maAB6vILQ-omgaP!~wz zM`Vd zepW$o8NLv!gYk&IsygyT0l9iWV;C{v|5N{UO<0_hkk*V=Xgo{`p#C+Z)Dc!U>2&|X z|8E}fxV=iZuDtMoUir^)>TPTM+X2 zU8k-1(Sjc?pSQbyamL@oV>D5xHK#MHbiO)j(LFbw`oJwfddcahhMd22{*!ktTzba5 zJ6A9G(6J=(q0`P7Lxvi`Fhmeq_vMQ;7dig<(2;+`|tmqrrcSDmF$L8>5bm(ZenL#R&@pI|MCgppFs?V#fIn`sjhpSGI6uUsN-eQQPgRYlvU1GmEt0 zi@v+B$t&M?<5?8hssEDvA5oEsg?Zvc9CW;A*t@p%9}!~!P!j9RVTw$t7->SI`BW%XdlB}VHYr2W}tCu^oZg@ zewdVNJmBAG{96BGZwvi~?k*MhS8P{lg&?g$4fze8|ER%&F5IZmGHhrZ)|vecN^f<2 zcZH0yRIn2PO23}Xg~x5&7{762!ln%|&#sSI_iX&8 z4GC-4=B-^{vnk9XhS^af&vK?^Ik51!{iF-n5fJ=gi%d*vBr_l-4J=fCQ_{FfFl-fJ zk+mB({q*vcH(z-L7ActX#1LI67%3tnhsb9T|GKd6&)LrS?vA6@GtNJ2{`XIxH6d9 zA2ojOE046FA>Xt(>(SF={)F>r?q$@1(KOz6D zkn;amtV(|55qVUkKt4+#K3)|U%Pydb0r^onNYN3|%FqZU_z#Py3k|Cc4X@i2S|1iJ z508|GN3bZQKscJjk%7c1$seSo)6&U!*fKI`C3a1%Dr{&%a5V+floHROL7%R2?VS!*28n?{Z1VWV|GV+<(NUbxd0Rh>m;ol<=YC#_v5 zycGyXh~CV?QvTasSYod%3beziYi4@Upu2Unr_cf$ACb9L=dsFF)_)M zltc%CElp$v9T}O<3|2vAQ)afP6l4l21^hQCG0vAacMG>CTUdTr4Y`!MiaH94+Y3tC^NU)ua$ECDI#o4W2Lls6 z@1(zXs;_sV$FpUyf3nxhJWv9SyrksFhaWP2=+5_oZz1%XGV^2ct{5&B$FmG z5G*Gq8%cX6CK+jdjE>b|F+#%|!Xs|GVij%I%zu=SA1ulUqZAATg7)9E@r*gc$>ZB! zf$u$V)0tf7AianC0#wavH zRW+1!$LbX=l`^-*>iMev15W8+m&c4>kN?6I_`^+Sg85*t$A7c5NRX1pfiICn-GQZ@>B_>s=lhg{?3qryEhvCfZHQ zw9x0CoK7f2Km^-O&z=~^5fA=ZFW|p{oir@`m)Bf<%T-r^e+)i9ghT-nCx~)`{~w=s zF7dBz!}n)ZCy)Qa6?pKbGbO`vA!eTpo0N9Am=%Ep4dQPUB6HV({|I?hlp;Dt8K0<% zOHjwesv}5ugexM2Q&b-lU$k*k^1Ai0>(<6RyDo0s(=ltFOkML-`T7kt;nDSBk!%nR z8$0sG)msWIU3NjmJzb&VrT`Yk@Y(6b%wv1d$TDCbgt1%HUXStQKZG!__ zhX(-vSa0{Jw`#iyI)i$&w*RkOP4rK}y7&oN7x-v!=kWCN9Ws%_*SQ(o!r?I+ulTZHj%%`Irj%C~3aHl}78QqsY&HZDmU6{k*2b1N#r|8Q65NVi8i z++#hR za-Qt71X(PBe-f@Py$Jgc{>#>{uaAkt^HECpSH>l=5h%i=8bTu)qGHR^Qj86C{T*%F zhWodV4q_7BHgj8}!DZB%RSjj?>1ELZ0}Y!*n9Q$CAEiz>y-9kQ5{Q0C#!wI^zy>|w zBu3)FMuH^?dGJ(gT9z|2w<)i%jn)#V`1Q(OxjLZM_Y0@pi1#w0R1ekEk5|{@(hpVF zSu3i#98Qn@EBt54{&#we79Ke%g#Z&gMWGfg_9`95zqel5%^ae1A-&Q@h9@V#1tM!s zfjun;??oGzsALI8#?*y_&xrcy_zrm`0q=mPv$w@1#3fprkr)lMw@vkVyPI3q{_W8} z-g0Bvnl&X)udN^(urW-wDYP~$OddtjEKZHoTS6jN0JkStJ0p#T?76@{$)63OWX~SB z)TwF<^QywbV8QXa}prOrj~da-h1_T3HGC+4I%@7YYAvPooX| zPe6WXP?1m?ua}QNC{my)r0K10=*%tb5T3tO-jy^coas5n_+*r6<>4{48^dL!w$^T_ zTDP%sQ$$-`*|wpHvt=ib|H2jcn}q+XF1rYyUkbPc^DP(--+blew_SCGj0{gqG@g$F z{Ks+|uZ)aVk@hEbsnd2kEZt4EPP@6?Y(!?$rPuVey12(}H@mH-c8d{@s6HoCxiL&l zz9^CuP`o}q8Sq=tPCQldG9wH%>K~ zt)xNSSFh};sG-g)cs^MLGu;}V%Lrb0wb5<8m zi_E8QI^dfL^v(49SOOD$z2n|)t5*N#`|i8z>Z|e|{afjpHIUih^H9~@fZv9c&U(kiej8$_izJce=@Pt+$Q zn-irpU_vtB7oM;feU34V(PAD+69I|vC{Uq`h=F5(7v#vwYa$U^E^|Z9p(9L*-PZQY z)kF1){_-ke&+Fs?`s-z~o~ExmKP&{|-%guzpr&D@rjGu{k*b=}x`shGrF9AlzV3pO zPK-a1Z%!VqH!V4Z4aq6_F>z@dLsOpJn6iFD;@b5I&pZ>iW=-sqYa$+hQdL?o($z8H z>zM)n1OAzQ|8$^_a=<{RyRM?@kw5+U{$Kto^pQuRAAc<7i6>&8d@Am#HSteAmGty8 zY0o^9|MW9e&u$bHDZ(F8pwTROu>`DAhR7&Yj7WQ`;^Nf_i5g5| zbX-Gpe4wMFzoXsP;_|gNu?kQf>}zr1ehj$VnYY;EwD&k|+{s$-*eu5EEX0kqQL$Qh zaS6$Ufu;n~fP}z!LG*pcX;J?Zn2%P$qY$^TittE9XgD5-f!;v8Ac_KT((r=V0_s(w zDF!)Enzl;I$7daAT@HM$JCCWFSQ(=_R|Ek=V)t+6N-jq>^qs|f+;$v*GiiP3|5cO88G z`RArKPxbjJezjRF`nq~!eXT{MaOgBPjoPYITh&UNT4hzMH8pivkukaJpCt;Q3ed>R zUxFQWfyfvt=~_~zVd0d{pT7HU{GaPq{6zSoQWU@{U@h=75uF9`-wpnAP9Fb-EAZsa zXJYoHw2yE&xxz3XNIoFJbld7He|gPnL4J#k*CZrssGU)9OHMPRj*?f{Qd;hYqEjRH*2=No1J!ll zAFqbwr?;}&Qdh^Qi!XaV!E^HezUG!*83RmfcPh0bmDN0cphoVK)piwEbY>TL$b9DH z!+G{r*1F2_iQ@a(TKE?JR+q@_^md}#ALwY6NDfZ!AM9-3H#PR&f&HHxeD&ztZ+!B` zq5WI84|VtO7`PC=mL?JcquuUlzjw^r&DReI!SnXMw)Te7vJDSEL|zmkoHaQEiXe$l z_C8jD9^Wz?FDrb*Z}0o*RhL6P#0wS3m)sta{}&0m6ZrQ&arSJ+(*MUJbLnjG_~bA?6xD>Xu5dGJ4PyXNY~Dh`q-DZ02sb!04|fqYYV!PC#gJ@r)Vv+Gl$Bhq7|lEXt2o?V~2Ze0UQ zfVdzf!%AAFxU9XhwyRdrQz!4Mt?#EOP*O2b*ML$Al5tkK{NMUNgI(@{c6V20Es}#w zcpojV8m_ME=PAg~7M68E@1%QOI6vh*l{F6Bo1RXdGTGOIOAnZPn(R1ofi|(z@n5D_ zO!~U_PmjN}dFquNTi)Ea=dJzwUfsQ8Di9d(bPRxYG25b*oruS?3Mdv+CV+#{PZe5L z^~{5!pUzI#40KR3y(EE|0!D%~E`j(+84(%v;2(c_)fGYlPDEtFDJ4P2rY!9LWfwEd z6#w7%_}O!*OaIg3rI;Hk{+jvZ1I~vY^n^XCvxaSWA@$aTu!e-_hWg~xuf1AR>H5Ta zd7?_5sFNo)Hta3FJ@|mx)TPCqqQpSMKZ>sk-g7*4S$2PA!ic=2H^0PJ#7W}li6d!L z&XGA(do1moFP<*%OwJuF&Ku0nAIL5k%qSd8DISPxw?Ftv<`2I789q;2-etNYa_p%! zlTU??JYuol{$bX6Up!!6#U(Y4Yin&+O;umaN7`9&;Z)@hUoX7iiwAtelV@%^3#-C) zh6M-sPh4|}2=TLPuDa)@8(=$-ufPb4^ffYNeQ*54B8fx-euI`Y*ZKJ<-|nRFefWI&YYHI&o6ANsP@QeDTnn{)qr8x$mH_}vE=2I zPPMwH=_};76ZXH;J=odVSyeYsS+i52VGi|BbuIA%Wi_0io@~+HNaTwK9qv$5Wicf! zL78r5aJGnU*u@C*@p<~hmvRB`k?u|w{%~j8c#mhR-%Igp+W=93kC7H!jX`XJ8*A?h zj(X{DZ=UdWjr(9nc+{0uPe1s8DkfgEMlu4DVn+hv{~{cQ90GNg+(OdB4-JK+`PHj`E#?U&dn{Hn_E0P`+DIuU%dFE zd)k)Gy|sMq)$`_FIDc;E`ExTrnj8J`od4%G(_(|{q&zD_x>%<+mkKR&jzv6h(vH|%jongg`<(GZ&F5mEE{Y__) zUJmAbg15rvuotYj6#PH%vs=m6k+i2=WuQ<+!!JD@DQQ$lhdVl6-?MAS$S`Q>cG(?T zty8O`_9TXzbhPw)*zsEd|JK2wUBiKflvE5IoOw%Psu``zPqLtHXRp8C*;TyoQhn!ySg`f|ukAivz5#Q@)YSk~ygbywE3$Vx}O-M!7N zF1;Rij&WTqgdT@=(9`A@hMiPD34a%h)TvdubgFifuG`+|Z*h!xI|Uu0o7+w{f)m%F zqoudm>9JXRp9>**b^u1V^G0a0$h`S;-6X=_)q)mBe!08IpVAQ1Q>jN4$1SE zh!8&k|I5LD|08G5#xDJa$EUGBxbKF+Q_i0axo9qA)$Ex!ez^ROcTWA~=ioo&k8_u; zISBqw37rc~G|k4Hc+F4a&s(NB4E{r!PvCz=;N_!n=kSgLH4laynhiNTch2GW4yRqh zJx5Y6JoWQq!2Z@(x5W%8{|)}<=6tG$zxcFyeI-Hs|M=MRmmb}E>CuS`KOequ&VSKd z*YzJ*Ue3S#c-+#s@yJ#0RbT#g@eQwM{(0M`TlPnR|LgZXbNh}*cNbsA13pf<;I@f7 zZy&j3W5?>_u}fdb|ItXrC5JN3;rAWv>fjg>hntiODB3nO z{OayqF1f5Fw}6gUHhqh9K!Y){5yXfDNKR;&v7w>OY$UF9=+q?DcZ>}9TbwP82COc3 zl2fv2l&Zo4f<$X_x|3#^a#?p(T`zrM6*VaHOg1P26|#2JRE(ycrmrVISq>O;7OA_Y z0X;XgmnXnKI#*?tNYVNVOM8*xWL8&6*>LEe0dGtZt{k zzcjt8xv#aQ%juL?lvCrDm#}JoeOwQ5PJITCQ9Z7joZR$e-r!T|0Q` z`qx83=e`&B@jv2L2VZkv@~!hUv*7=H$J;l#c4PlT#%3oA@8z#0_JUIm&4K@rPiJde z)4Au3)K&9mj|2N(?;q9LqJ#K?<#plK;Z<7aM`taB+l)f3Lmzkzf5nOlOOw zffpGEdY94>FmVahnp|9k-2;P1-+S+)ci!GNvw5b!ALx$zdd7Xd6FwjReSUiL`>($G z()5V4w73hE<5U)oFnZ+GMbii6t(b(uHEYcZWrxK=5v|){Yc}bq9Xm8CstN686B|L7 zoj;jb1$M2{n3iUaOLEeFNPkvUO%EDb71b!SO*Sa|%Bxz-%G=E5uZiCV_kwg)=1xIK<0 zx78+m9@ek7dCDPwa<=DMR{%N`u)0sp5Q zo;&5Uxs`9d{Bh)YFJ)dIGCK?Gf4{S5r1JhC{vX)gcPwn_vG9K$U;4B=i+;Xz?rr>^ zCw7>`1EfDqRQ~+O68<0BUs2SvcEu6J%J-_&yoVr^H%Z?zj)YxdJ=lW*_U7R5eOYdf|d|M;WNKl|*xgNI(;wdcjH+xBnS_R6lk@4oul(Zlb&`P`09 zStT=bJ(*1P&1y=|rXSEL(ST_Z){k_)@zV*`I-+b|wAPGzeT)A;0 zEs!@}u?qb2TY!9EPar^;#yvs&kNkP*(T&T#$??&K^Ugf$L-2peWv_j|Lm9W*z!w!eU&V{7So{~QoQZ{!={oE-M{-b4z=aOzc)qnhy(YX~PFMkwz z-i3P(O8B1z|M>Y%?fPwbKL_(1kIb|lN?5hv_;Kz(|HxZ1Te8e9j%s`IKN41b@oC?W z{P6`x4}$+|4^L@r8@2Xl1MXYRX?(S$fZ zLhy&p8vHY&OQOf4ismn`q801)ZQGB&_x9}3&t^YA_W8#j9XtH~@sB?|`pz2%rbhfq z)nG-HHwRKZ8i091AuS_Ya(#sWh%qLKw|4u`J zYfw=#7(tAb5Lm)LGmKClqB;itjrqAp-}w9S504x_^8WGn-#`Ard*bBC;p0c%n|=TN zWADEE;Vu2zqre3)Q;P%OYtW2r)cd+YgK#}B_d`^ks1pM5-g?6bMgk8(|Bj~$!+ z{PWq*j=uleKa6$q^?&{&#BYW@p#Xyph9fD}Oe&cEFA6K7Z!`w3Aoz!Tf-lN%feR$O zAXb4?3pjtdh<``^xb%~C%f88RWbOG&zW;H^vboil@BMV0c-GtNF1hs10~g)(%0<6> z<)VjPx#-cC_iVT|DZDO-|1}1`SQ`yDMT$zG!L=_a`M?h zn&9@QWGKUOTW z!eTO(lv10E34>gA`Sa*;$K#Qz`wyVzyykV1u0M@BZiqNR#k)a5cVHs{yUyh2U`dFw)uK?5Bm0x4Zb`*_UhKD*LQ3_v}@a2dw0G4-0tVcMvD^SBOZOEIU7=U zvDgA6|A&@&i4B!RM*!XQaI0<+E%2fRM%n^`QGnP8SO`J;Kl+DdpR76On;aiMea@;2 z{to^Z|LD`#pSt42Q=VS_$(N5c=e)H3y2Y`dV*jtsd4VOfs`|hw+PQg_xslj=c}j^a z5Bq=0*16S_`$G=Qg8%g9%3toC0QU1{XY<=igZPi|sd>A5b`5+D|BL75N}KY&__QBC zz5I#IO+ozsU~X41{}aUjU-y)~zy5-#k%Sc=2>)lO_-9`{i<3`z@vV=7|2sE7@bNR} zZR+?L_80i}ca?zI``OVLxuU{(SUt#}AxC!S+n!hF_SgC)-{Q1vtR4lYCC8t?Y zHO$O%1*I${xn|%urqsMKIQIU;mREZFcbM%HDyF4tr^qpqoGht8NEtaPWHZru6D_oa zuGx8P0f?csr)I&RvnkbE2M2~aJt9@uhERVifv_kU+b!^jf)xZ&hiY%3);8j9aVDW3 zA`x|R^BKa?5N!b#&b8@=L3d5>*Ac%KThKevU%iiW7NWPJMQ!rXX{yyAz6$+iE7|i{%IZcxY@@ zeF)99w4G${}kM6IkVc(&-5)~8| z5j{#l$w+{v!H5vLN%4uaK;C-wmHa6DBCG-wL0JfRL8ODum-0Vj4=n%aiF3Z;vFxzIa|A3gcMebzHk4)7n6KPOL0IkoO{@PE4bvx94Y!n^E^y!F)4&%uAl-nozi zbKu`DTkEQRI%H0`{pWpjIEeq0jw)V1%Bu?Ef9;S}rHNK+qBWZ6!Mxvn`P0(o4gSxa zdm*!vSt0j#mOt?4z4D;{Q|nlxH9oxJ110fqvUly5pLOJ!3vS;j?EeGfzuTO;@~4wO zzis5!ApY;`y5yzs^S}JyZ+ISiorr(gTZ18h*mGCn|6GCuCH1e=zdcIwlYVV7)ebU0 zsJZ4Apk9LU7lRH<%7z+L2m6NJ9-i3O+%aBLKU^UjF0aDu51?35D)bB?6?7qHg2E?) z$rwHHbR;zMpq%0VYb&aspPGO?fJBw3lXkSWH5$zdIT|LxOuIBH9RZWhtwV!-HiHnM zqBB>#b$MTz7|4au2DiOsRW`^E%}xJG{tx1Rw5O-b;y^?KvK%8I8Ot&vQHI3p6qN=f z=~4Q~ChL{k3}$4dci3BYHnr^#NBa(!dwa8IM@#2sm%BVMalG3@mWdk2pu3GiFkGO~ z9``^;yXc5-c4kF}7l&@7FgK#sO)y-^>_n%RX~j%yLa-Xql__l_%doeslB>~*DM-nr z*)v!UfC?pq^%4^AxcZ8p3JL_^$L$d%fM5tr|JV7;MgDX2x91%B+qvKHn0WB+h0BkD z|9KbArrcHg-lISKw@;jT=2xeteUAUXy!?pp|Fv_gG%tPdH}OUvJbLbJ?yXR4VDu z70!Q7ix_%{l)W&&gpGmns*wgYQ{=|#4x@u;(dvXR?vL0WJ%~lv>gsx%U0)Ob4}zatYS3PX%U@nST2;MAqaUlP z#r{K|z;W^-`&?YkQevtF-DC{K+pIP0Fj{t+ox7~AUG~;pj+Whaj#kdSYArp#ZmYwl zRk2mIkf)NE-b4XRne1`cGs8DoGBU^Is%ed3i=lD5$+n}>vBNBMsu?sqr8P~c^n>+k z+L^&W;zB_=EOfE(Faf_Aa$r)*&#$@qXREJdN5(Gk)11KnWx^dgf&cr?IsBLZ;7Gr% zD&zu@|378b-1R@(DgS-UbAP$+oxh%c_>uG8c;v#7$A9zwU(V(DU7}yazvZ(*{Qs$< z^L^=ExGk&5I%Uh;DU$s^@4!3n{OtlhK$Sl2%lNO-By(q9<^%Jj`qy2%N17_vx;U!V zag6_e?7atYUFEs=d(JoCm(1MEy@7;a8v_XeW7DyL1QI%g5<(9UYUm`8gko^FEL*az z-h1y`(yrRtUD7VXGI#DwX5Lxz?zPuid+oLU&+|V0 z*?Rx-JvVQ>^@sm4f&c6N0eybhZ&eXbzEKnC#sAH-DTjWyB0TST@Q?rNXQSIyVYiRH z_7nH^Yk$@Bg4h0&6@AOi4flfo=d3@Ue}2^~+O^>SF3SaldsZ+1#5bILblnfI|L}wn z`$0ZDMk|idE>OpbG)pvOh@hFD3LTY#S&AuW0hJXD`xA^2W^#_LAeXbK~6|4P|B3Hv9-? z#fu8;G1{Ba4qD19ruzDa>MN-tZXs4}FCADyO2Z@3FA&IGtYMy+@Ovsi?pPq zLqV>(RazBD!w%XISEpr{$EVmsqHsM^N0ek|#R^0~>=rRHOIXOJ2XDFo;nTY)^;ok7b#(fRt%{!jC1FaGbUtn%(Z_~K>f9Xl)h zKnRI3@3MdF<#qpOY3cL+U-aMaVI6I8EnjWs*^4haIi&c1k^h_U|9a(r?EgN-zi`<6 zGZ(z&`ZNA7{J&RCY`aeUzwdbM|JzmfY$>=C{NG%ATUNl89NhfQkMFh%{vR@&-}l<8 zh@Od3FEl8W=!J6p zSs`I9FyzP>V)zSFk`X_?0Q=>SP!nA+Fz>_U!GB#;EcmC|Cjp1xV^Ck3=1MZ8(U`Oe&Qy6aGd#sbCRM&BBfT&^#h8;ps76tC2F?q#6@mb_zWHjt z?=I*FRj45~voT8r2_XD5*56RUYTXoCM7BBw)J@r3w{`}hVPb@yV@|TuIFaF=piCf8a$}iym$A7iFR<(Qb=^uNKnSFBg{_LmLPxui3 zaP64HKWylGZCB!VkHpTBz-Dme{h zXmd#`ZMv-1W#!dUvxVXnB(Fuq(cqZ6j>SIM#P-1d@dYc+@ofk{;2$mc*}L!l#kE(- z8(DLycWD+W2*1o_m#qAu$p5u>pFjVbmH(}=%JIN&-n}>Oj$c)+f3S7K<1KgnuHu8| zVmcq+pj>T5>5ISF`CiD*4?}it3F&?6+Ry#O3iAg~?TiW8nHb{v^8?HKdtbVKXI|*e zybzzfkj6jU>b*!({5nrUdZme5dWN@na3{jp1AOvm72XTMy7linfiWk$~)01 zZ^tFQ?*8!h<;#9{*ouMYFU{S5M_By*slm5(ymHyXLo3SPza}Q~=ds~8Xnn3;K7Qu0 z3sU@kob0zcdH1T`7gsOu|4%1duNM9f{^jvv|5q8Pk$y&Ng@?7 z{`=6wVmJ7w9uD|7!_o+l@T$J0Nhn7oCE%S zfmnn}{OPC_>kHx*%pav2{DV^PPs9RkkK7T_#rS`-vcJ;#dlvsDbCuOQj?TcOIbOp3}xihW#mB)4asSq^t1}Am6~!kwZ@Q} zm!FbiQmGsoOjUFlGg5tPq-691R1(U)DaLgD{)cbq@_z>04( z<{mhI?tvBF?YFtk|KHEucI6M(A^#|>MmjHhRe-gZ-g5Q2UtW6^8Cz1l8UC*r|LNhP zh6E6SgaiU10Ynjs_S=I)>~K*?AT$C|+#M2Lwl4$@KW;7S%2Ac>3l( zz&eUt1aCa(a7~tYI`h=Vz>uPtm@2iZ1ADvER$HX4(G=Df>aeUfr=;UV2LIOmLAIcf zvdB0nqCzS^fd34Z8v~WE@^JoswecSk@R&_iHmfBr#T6FboRCBpu*yjA9|s^E5s`L9 z3xNM~W8zZ?4JTQ+Dn3c5fW;y;lR|TKsabW2sWsWzlRe#|ojn6B?R`xxh-dvxErTuX zgRSi&?VS#-;e(f6DA?swNlHR|68yg=IklF?WYk)TSJKTwB2arRyZuQwlVR;pAKY^`R zoOjtptG)Oa{TCC^<(Hxhko<-GgYh%#y$|v>Z5IAd!tX9$hK644etQj6Yz_!24ho@5 zg*7i!G2wx-17FLIRkH!Xm4Z(y2y^ zL#iq)vKq5tY=S2y0ruXUQ#fd{;aNCZ)7(w=77LAne|WVsoSy7)Kony>U9FvAk^1l` zDnEP;|EC0Qjl*tAOb6R_vGMJRX_b+&eNm9atvvNlDcU(isj z3TIhLMh(VHI%p)PI%4BaFCG+U_sJ77PAJ=xd^~w-@ldV93I5f-KB(kX1ezwLRwYTH z6;FJMi_Fd7aFlnm*nWM*4!nCW&J-iF{_B6SV(P9{{~F)^32a|?-eniA#`5V^ewpC^x?$}ajGwn&dlePi2+_>l zv^j6vHWlSMeS8aq1a_-;iz2M2s~t51gF-C9p;V5wg+)+8(;i08Xo)LRNC3_s%p8jF z3P=tFBy2?TEn5sn;h6C~p^+5~(z3#%DdxlQrl>b(0ppKH z1Q?~BLvmVaShO}W#;#F+q4N2%^2>zS`_bgT~IUyw3i!Ac>PI*onylr=})O z&Kx{EI$V^Mz3G*gbpC!-NKkPJG|?`PPxM5Sn;e6d4fw6Q$u*<#6D79%iuUX!k_xMW zTAE_)kU$8G|CEK<_|eW6Ucml?3dT!eMaC)5MEvrX{q#o)|F8e~%7eGB`q%jOPhiJd z@Q?imv(IIKAITs87u#Q6d*xjlt|L2JdO8sKOo_Igl(O{6M+;!=LKoi6LZDZ40b@X5 zaZr#@K`;{F>b<`7t;~>HnCc^`7 zkC6EMaxDDektD6V!y=t{(`3mA@vb9f5m6N^6A@7qb8DpSOK5~9CKlr-VacDz{1@iL zrkbhsWYZANO4AMSU*fl)wvVh?kO0JjkDxp{zKS>v<+pSZ zaX}m3jrZA{?6)Ir->!_nJz2s2*};3Wg8WkV`R;%3FFW6Qqck)e`G>^?Jp%C4zrz|3 zObJJ&Un*sF{RRKj*Fy54Gaw;w+jh_y1aF`s|7^+pVAKBBUMK!%10m3?#lru^JEM@m zrC$6u+_duGFIN3)eETQhv*x_ZFB17*mRb@1#Snz^7yax0n{J>K1NayFCxvQ#d{s1X z^x0Xsle*12sNP?=TXX@v|6aqs0J>Kcp%e!MS|y80NT4hXTTqz97>iOIj>pqZJM~bR zD`uiFQIqi`oJG5&v{^VgT~Z@1;COG6^uq2WDViHRuOya)N5)PyiI8XUFLfcp|Mn!Z0=he!;&i3L+^1by*Ie6kxCH7kXjs2nrROJOc>|V-aGHVqYo^4Rc4w zRL3Q=Mo8K7m^f5#O;Qr!2VbNALJRQ!QjOjo98OnvOrUkq@wkAgpn?kEj7cmB3P(@3 zBl)ldKxi<3%s~Y2i8>I1s_tgdiHf0#06yaS#MGvwRQjkkre)Gfm>y)US@}$w(sOYi zpg4O-E{KRO35#SAA;rwGd#`o(KFV1e{r5>_Q3-zDtwEC9;RF6@2&maaA~42Jr8NMF z!0&zVbEWw#wrDS3!tx`N%Y^@f|EB9!joxsPuQDxd+T*q zKXv@f!hF_=IFj*w80%%T9=0XAL(AqRyPn6pO~g&^^r2_Xn|h-U%yp)?{AyCq_| zJ0z?lEfY-IkbhFsh?n~U!~eoZlF(g6#n#Yh7LNAhbP~ZRnO>ccT%C|4S^$C>OAO6V ziCNkogdil|J^%eI2be}#3s{V}4Q#FaBJ8~;KCvb)p)M&^Mj|7nHX(_!isW4Kmf+_dK57!dw(ZbR4npz&@~A2QnEv4=D(_JR z8gqnLfg~tPjGvf*&f@=?RioEk@UQXhpTO=*ezfMI3&itvx$u9l^9#$vZR@Xl{;s=H z-~KZN*`#F&-L7fff|$CEem%GYm?+I{*|}va{ahFj0?pqI(kdl@9(Kin!K?))Y(%uQ zV~=!xXDb6LQ0%3lwvbTq8ekg50#7!g!uc^1iTH#D5gt_*7F8A=jZj3aGU1x8=mh*q z)o}@kE<`f)Wve_mf3g2$WDqj+1;BsdW3`R6M6)F#7PX=!DYZ2zwO)*qSv4tH6^Usc zq##8e<7SA6@W!l4mYRA3je3ylh)S{dzvs4tO`VQZ&DKCBt=^Aj~4*@CoW(N z5lSMkgk~|Yg8t~-bA|t3xpMgGRsR~_{t4{)@sBQDy~?}J!fJTntOW}GuYLJfze-W? zPnjk<^(d>e^sW+f(3Y*X-f$oVnfri(36Tp9PID!*VZFtjUgvU4|qp^8b#ECTqrY#2csF*%g zAy!HY$?v&&*nhqN_%D2vXd;`<92JjBLLum`w9LB1 zRsj5P2BN1hc&ZtaNw__K{ z{L`>@^}xCd|24k-6WDXnic42t0FzNTzX(6*0_eZDuD|xJ2Y;RX&R?j`hYFz7r`?#f z@k6?uqM-3H`@@gKUi~4d@iae`!ADjHayk@2Xs;LZ#gwiE|LDl{w!>wh&;n}#O&&#G z@ZX0^3t=0ip?*q?D8X_74?-X^lOWPy45#5B789Z~Hr|azF(I*%Mvw&9#K!kz=C~rF zOwn-||G%>G!{|b-yzs9$J0{$1S_~`|!2({wz(s|jt1SrXmSO4XaN5H>Sqop{f6l;~1j_L1G z#_fGNT=SYBha6g*KnqX=?XINJ`F3gJwR=wy>M-mM!22WaO30jk&mIY!kZ!;Gg(&>! zuH_H@2@NG80%6-LDhT71P_$4IasiTbJ_VMFr;PUQRBZMnrBo!PvkEk(q_?MKKpI6c z@pg^&D=S}p0snO+CFZ0ILQ*N2GOE^BC#O}W&;=mflaTC;NdUThMK}k-7!&a|2yKef z0bd^Sj00VO;0>ZrapwieP!i|mK2c-{c2#cJi~kTBY9L3M140mfENu6O@ZSahG{r;g zN9JdX+iSaQ)f7afCNSlI2(_h;{MPMH-F3(9;D4R){|)kXBq(eBn#<1O|0k>aF1heu zzHk`;jt zCbV>fSxI~M%>>Aa@FNM5yKw7BnA{$J6g>z5{L@j0tylh0T3zq=iq=*|LI(L*T@;xi z5EhD&QwDGfiL?h3oD&8z(LD(kbz)kf>^ZniO6|zWZ%Rr5c)GawQk~BFRh4f)kAFg- zDcVdU7!Z#0t2ZYP{8uEW(*Mq#lv)-YkBf~id*THP3KfNAZvY-E$dkt0L~e^9v}eCC ze-;S*+QDMKCFD=^WdLaOiYN4ARpY~*a1+2{Ps(Ixm|wJN*K@Y4u;a*hKkcW zeDyng^xJ&2o44ct%B4FV#WAP|v}OBC4?c(q0xei95?;GB2D-}y|K}n9T)4XXqJNG5 zfhXX<;(YkOu>EywNP)j`-Q_%U`mce1-)EntXlEAjpC5e0>a5-*L$hVOZtG6n)*afd z+qJUYL349zo9MUgG;G_cv`zFUL7p@jFpnUFI$#j~?=SHZ=mq}!VE*8WU>-`B!1;yw z`v=G%p!%~IX@nw}Cn%ybg*7Og;v|p&hI)~IVAn~h<%vn1dHD?qN#I|bnCLX<|54`4 z=SiU6SxO@S*gJu9y{bY2dn$;@1pmotB?NCscwPXAHItm^zYmGs=r0M77a;-KdC=gJ$WOQg z`9waxt5D)Xmu(oX{_@_k7pPu~fZP#DTOYvIdMEO;C)OqI!{_8JT-Fd<4j=vx5 z`SQ-sAM=mC{qp|hJ)fU2|3{|Pn8~)=EBW1a2K>| zw{FvI-=!mgecKMb^r!RDZ{1$Bb*FyI4)z!*P68DWChbcz-#w^%5CT--b(4XAu%rn2 zf;0){l%VJ!MZw(f-4FH{roa#?JXnGwtdx2S4o4M5>L6B)b}gQ?Oj`F=B&GD`7gk5d z;#$=vCzBvd>FLiy{VV5ZSd~H}#Izd}ssw?f%LH=IkbwC9VEgP34r3KS_@{|*@t*yq zkx91jSpDvOMRb|*73Z%s`$iq0qAB?=JT&4TNV3M5wrwZEKV`l)JtrX$BL85NBQ?)! zm%jY6AD(m0kG}u?>hmvbUU88!yc4$GvAlD6k8*$<|M%m3?~#9h{K^ZP8Ouk0e!jfZ zJCUzA_j3m`_a47|*5@b7^X0?7aIW{`=MVeJ17zoCzw15tAD!~${a@a*;zIxPR{Z3G zl?eQ>8itquV|ilgBui^&dskPAthD1$vEKQlAaboET4%!+WP{veiG)VlT}Qpz(md(h9CvxM0z0J6qCyX z!b$23U>j(HkQIT)91_L>{52@rL=$1Z0B2OPBPLzFJAj0DmIM61=)!OY+Db}1;r5-n zP22LNA~c=BcTnI(wPm~C8*eHhKx@4biHZm$CeR-!_%Azeb@?|N72o(Uj`R+${Q0w% z=gZ%I!=L`A9_GEPzq_{|e#J9cKHwkS%16_7AQ*q>5AC;g!$tj7W)s1 zFp>Z|uM~nn3y_RRB%Hty3$(Du-z@H9d<22G41kjXXz>+j3w;zb5s>sdF`r#_5xj1MlL;tJ&9?E zUMxe!q-BMK7l(uq*QDLMUv3+_rIPY&CiRjf0EA1hU!*B0%oq@?$IQDAmli@@0CrAD zz|6NGA1NqQ0!qO@d5%TK3rghzI!MpquIJsgXv6)Y-|&Y z5Jf?^?F|0&yJ%r>A?y-f5lH;MYcBo4Ip2r>7k+=G?)$5hArq~9TpqHYJ>H$#&*t7e zU)bu-@t(Cxf6hw%Iji*FUnxW8a#{J-yPvJ{bAS8lZ$G#EyK?Z^TUwrYFYwF! zhUT-cn0)0pd*V6D71(+QXLCDs{>shgoPP=Ozt}v_dVX022!5ai z8x1Kk-`$w=!AJ0aZwi>K0vM~I0>G~f;s0BA6m9ouP0AS86;*~t=(qY9cK8}7lO_1y z%~qr!;s0U?BI1{~FA{~4K>(`?*6uQ`f8Yt$FGmgQ=3fcB^yFwitvB!*)MG+sWMG}F>39S_Rg~6w^y^rA4dQdA&Z zIR2x}Kf7|RG(r^-NW5TB!TQVB!2gl|Bfs~fsQ>G{D7G@rQ|5eBChV85{(jEK@0}O< zr6GqYBU-r>+vUmME`Ih`yuWq!7r6nh_LbNC{I5iO@4PRJ@b8`XD_4k}ndhvCKvbT*YyEXp zK2N7U)w>%hIVg5%lrphSd?o_nzQKG)XLjE`bk0IApq>xbjH2zkPy~3(I-gxSilsmT zXnDK+2opjI0#4%kS3-sid;COlK-mN{Xq}Q2Eck~;fCq6Dd@Gis{orsvrX&ait+PBe z9f_qTTSak_9-XlwJ5G@ z?eCoXM`bKe`1ptKexvc$chB{XH@|!C+3}ro-(>LdJLkUf-OtA7=C8B!JLkOi-E%)T z_;}81?EKC-uPR&be0k^DkDqFL#lMG+PsD64=JfaR+_eu&)2KL7(KDfv)1hKwQM zrqFN>;6!Xk#ruLVMUwdrk409m%F3(HE$A<{_L*#rDzz~+&0TDwSmH9szh*MfP(MG= z*XFJma=4oEHRINj>a0ALfU@XVvY4^+W9?M5@*rd$?JnO+wLudW?TyU&%*N}Z+391z zr7K?DePYytDE5hXCZ3CB?I4)6pdd5kz#hE7=sUi9rSJu`%-dEhr znqYdd3;7{rvMA*5-2KbzuO$l@*u$NK|6>9Y{GUVq3*(&6jpdJ@pDZ8do%~a`pT8OJ z{Ga-B|FakW@v0TDm$S<66_-pd-2|TvfleJZ{y~|kG3M9>IMJ1NZBs-Whp;v zyW?dc8!(^<$>!eXgCR&V`*4Bn^r5&V?@>~mQ1KL9Ktts#)_vdI)DePw6puiFgn?06 zck>op3F1@UwAoBFW=aN^6}GQLM4*{6usss1P)#a&zGlQ^8+JHbHF}T1*i~CgdgIr? ze)+4;>gr}{JDY7IWu8`z;h@7=ms@}|#j+ypz_91z{8DVNMZ5h9eSC`Psk(cwa%0@v zCUJIrv>ETMGc(VKVkLMJ}Qam~jUj!${+;HMX^zI3$= z4p7DgB!=J{x@5JS!nbTM9=Aw%@>@v0& zUdWMrbt&UQImWx?g5LdWl&{uYd?BZB3|d>%%K@u7-aA)LkrTP#8m@Nsce&>aIKXRO!^!BCf@r>Z*G=pH@X$lJeG5L^q6Yui zIenz+AmF$7q6T8YmNz9YOOU*qZ{cIE(8$*0wDyeb(!fxb3GAhusFpBMA5vym*Ah0A4Dzp^RF~1S$u?5zY1f;_ZDp+mS_CSV0P%t)q__zL zrJ5o><$VFfClH(}fm-j94MLw=nDZn{EFgX_=wT%Cc#+Ul02)d^>1&br#!3EG0eq;=0TnWGWGa z(z9`dxe}7e2rdo`lC(tZz0##(I|fONGm;0Vn1V@s)A;zH5sPB3ARk|tEs%)5U_>CX zd;YU*EcqZ_FT*Etjb=!AB0x$=b1~HCdQ`#T& z%mWYIx#8MduDzP~A`n%e#T;k7;6dK3V1}=LCO8!2@a-}QFi~(NNn{+!oDg451Tf%} zJO)erI6)^LyOdu{U$WHIK@+*K%ZsYQsu2j&jn98}-&#t_JR2>D1awi-ksJ>zi z$!mNoq#@h2oW242l7GQ(3065$!5){DgOxpU1&#;99Pj}*ha*CWNzs?`O@^Z?a+;j6yP+8nNe}XL+ z;P&0TjbIf99Ny$h*+W_DVN-B^G+gvVbVC+*d|r3gzjyg3e_nI+ox9rKD-Dkz(c4U+ zW54|#rQD5e?|y3UNVbd8X{9A)^FcQtTo`z-?|NEh{EfP zit!{QdI(fV%L1Z^D-Ehb>^8Hm3h>{cGkgKq-|%Cvt+e0n>^E85w|tT&PsxVSSGiK1wb(cZ%4pGo;Q>1AZjm8VM*-~Uxws9gb?@* z_%ZMjFCl?L1a&A{0GuEBM{%zCvxeeRfOUxeiwjJQ34Yj7=|_xb!MHDI(<`sj{O(sj zze#c-&w|#Aa0R!*>V+{1-~J4)3b2KTd)Xl0GFM0>H;%?7X4^IBe3BtFM%kimO0q=;#v6>Aq_7%@KIK-b?ac!UNo{tAqPRD097FY zMY`i(2-OH7u#hR>RIVnsrw|m|zsqQRPu^9p!(?_Kq8_yYq(Y@4W6B7JB;fs6X6-X^xl&tP@D8CepXD!-%aT5YCT! z3-d4b^`m}%+LkRh4{UsRDQwNh@BX;uSyeC&Q-4gF97!%PvHDEJTS}WP4C7KYE70KM zWA*b_Z-@Esf)LCEi29;g`*2NYib&~_$Y@u5A~HX)tjW%=%PVM9XHUq8;)-u0SyY z`?HwBB8sHa0kCqU{5nIq4Q|hShss4@(emqFoO?m&y*N9;u-up|MrZF({)j@t z{0e_i9s-o>B}}=#_dZ!c~Y zfM@Gv;0iJFl`<5%P@a-<^RkY3??;}7ccnNhXvvYd8t=Yw{XI9Wrv%WGx89Psm9<_X za&;eV)oyZ#XS(Hk)zES$KOlt@c)uv+#9l#oIg7hjc@ zC2T)8zdo;!I?UZ#!$FPEM9LkXr&sw z5rY^Xu_kdaZs>hMcnT1ca(#UN@Yv%KyLUfu_q~@%ruL=OC1x$aG>Vz-uA4Td9E3p;-;>Md^dgAe?@4GMdz4yg*uM}k^lC5a#_TsHO zP1|;wP}UUT2T1_woh(1lj*%Ss2V*Ah5pITFAcfrmLZs%iQV1GW3G$%=IHUk0hI)fB z&bYYJn7Fc-n6j8y98b>ZI3OVchGIxzEpSE0R;Q#lVCN~+b!ZGd`l69y>#U>nkh^@* zQ#DiOnJDutRM)o`o63vnwQBwXy#LdWUDZ{10ou!)3pI_$MkXkT52y`UMh+1=(-yrm& z5Q`UJ0(Ifo5W;7$rzGqZLRNxXcs2YVQx7=gth2lw%JR~ha-z^3prxHLR|cpcnRD6c zr59oT%JQTj`7BjJ1+o^LB^l9cyrilqG72Gk>4HmLwT}1@h#eEzFR#1i*6Xh(ZjHqe zeh$cQyY3n+`J7JQoV7o`=#o_{aAaJy=CXV5xc#?}KKk5~Pdxj?6Hh<<$nPG1?9G>6 zc;lI8Ha`1&`lc;7V)S^ga6xY0K_r8DyN_kZZmTeVsd+BZEMoB^V=JJF=843oDdLa> za4L*Tq+GJdr373A;W}_~M|cF_$GhZ?Pw*rrO6p2_W<_ddWf~)+ig{8}S(wxaA|@Gt zH6$kWOygF^xW#_ZRyJAUoU}8yG7(I=Dv!2w%(%;Ii;Bo+A*Ag;1^gEd=&1Dc z*44MTN(Y^u#rl?$2c{0UbSUke*fiH|NPR+@4fQMUtW3T!?)hr_{X>Z^7xE?K2L#pJd=0&6pcuhO_4BLO$r>jm zA}^(ZJ1gTQ7p%N!g#@!*w{Gp7x7_@j`|o@7*S~)Ffd?OX@PS7jeDLYV9)0ndr(byb zse5m`^=H?wzvJdx9=z+W-~ReHuRrnIJ#W38v2{CMEzA*;tf8c7n0D?mZS%3w=hxSt z!Z;MCLQtnB1AL74=nf9A43k2IQtFnZ4_V(8U5^3|fH^#Zv|OsRm14k&j&;SxRU{@? zr4qWCRg;ASD-Y3vjuLJ8nwGqR=FF_-wDiu*oQWdSoV{$?>KH33?kS}4NJHzC4>tL|`Nu6UzxL59uWfqmjrU)EW#?c1virji{Wfp*`S8One|h)4 zx88d5#g~5n%-aqeOn z+90^)qt~y5cqYIWYPymb%UiDUDz>Z|>rm(EcYw+Fil1Dx=3+e2t1e!7{zd2i=;HIw zzj)=!H5Xre)n%7nvu^DzH{N*fFK>PD&O07?-~q-%_x+0h^8C|JJ^$?QUU}~M*Iszx zna3Wz|BgF;b=&O^-E+_55B&PY$DjPuOE3HU>Cb`heHiu8=9H~F3;p&Rc#^>p?(k?& zaJW=jlgA#)TaXF{JaGR2C(m5UpzU|kr6MQ{sM9&IB_+KjHN7@E-ox9BPpV7FtVv3( ziHoNcAwE-K`;5qF1OOKdkhdQj=Z;T=D*_Az25(0RVBe=L8Z%nP48`MG{jko^t19Tq zRgDx|#w`vIHD+-P7MTW2wt->`LjHjg7t;Q0ee1FA!IQm%pR{(4R#rEc*jn7K?wTq( zt^cQh|MCI-^)-XdO&yhxfV0a}I#yG?(9!+r=){SM=@Sz($0uh_%*>yhSvWp9dwgu_ z~pMutaYi^G$s)Lo(Sa5OdasO3V%-Lkk8Q$$>je{k%U zT_GEH?0s+Rwm-c0{wr_);klRJeERv!dd^c_O z-MsmOx8Hg14}W~~_rHJb`R9KB^fNC%{q)Pv{*LkLbI-l_^wZBj_0)4uJn`g14?TSU z{SV%A_pk5!)r0rm`@r3IKXA|8_uqBby?5Sz-<@~zk@%qJ4e&*3fo_*}m=N^0F z)n}i3=k?dOy#L-uZ@=^Y>unE(+~dow-5d1#V4M4>xGxLzWt}&?`@3Ou`7K0 z&e)y4DZYL={sG#ca0}7$2`QBC=E+yb((yTlKA~*s@=W$C)xUvsImKq_SjU#yvs63| zi=?QURO1W?f&aMK28Qr1fVa2S4V|{?6O!7J)991fTA*uHGYrk?6b4VGmN-UoCrx$Z zsM3&Bt_-rYVEZ^&8*}mrw(r*&51MR?jZaou|F<#Z%o{-kjQQO!?)}W+w8aR10TP?Z1Z{hBmWP!?|y&V z_IEaJ`Qv-X=K0@0_1tT}d*Rh*o`2)n7a4Cp|I%B}|NgC) zUVi@ZC!cxv(Wig&@KX;1;K!bO^zq+4^5_!}KJ+9zAAR)2r=EH0;YXi*=;5axdF*$O zJpRbN_dW5zuU~rd$ya{+)bo!#{MIwi{OP&pHop4$wmLi^Sc7vgFtn8F_p}#DPOGB;U}AD@B1O#Drqm@S(NU$I{S+ljrZibfeUj8A zt4m1Y7Gb!$esX8!vUGDD=moptNy;t04C9I(RQq{Vz_I z03v+a>4&8$YGobQ89NJgEom9lk|LeFIwg%Lg^Coqx@6U6=GJFsx8~&!=uJ}=$Kf*f z5tnDtSw2%zHdW%Ba+Fc|f5PFMc2&++H7quF92*!pJ~4A<{_xVm(Z9|gTAH3eJvQ;l z;PBy|zPa{}@rL>ykE^ZJ(Ol+ea+VKD6aAyd zI|q(6bsVZ~JW|{Aab5HA#EO@pa1^G<8@Rnx|Q&P$TQ?OSI#SjIshl$}6q;e6x38d&;GD|RT zzdA0lIy#=d8?;5E=&sa|kEO0%N@+x#Jv`1Fm8=g>)rO|1Ly`+alJi573xbpLgOdtE zl9=ZQB{9znOk|rMl&E4b4@%4nOi%?SKqJaTzEy=J6@(@;G@+@Qu++k^RBd>=Iy_Ao z%u{vYDMjJ&#?WX>RDwM^(Giv4h>EpF$6Ki|j1Mt7-Vz>#UxV}qa(}WmZ%LDiI-Aly z_%QPWdcR675jw3CGbVOLticG|n92zo;J@=^ViTn3y`n&&LxEL9N-$X!pIDttHM6vO zV4sp+3+(BV88C`-NVFOdz6o%J;>?2c%OWv6ijZ-QVr_NU0P$O!PKEI?lM~Yv_?uXz{i?0avIYz8nP5QCtuyGH%?j{ z3uVqDu8IX`1$G;P7Ur0jS0AcsI9%6!w5|8}(DHH|h|cV0mU2@5%Sy?F%#g&M55Q_iY` zWt9gV?qQ>&HK(8|E}kWTQN{b^wZ_DHsi2Z>#Vp5On zO-dT=_K>EiQ(X}!MI9>>QYvY4oRr~?VHHR&i_5gfXWQddwm6j~HqR25XJrznGRNgv z;@RdhvBt9}kGX8)RmHKnrZ{E9%D1MtJlPYQ%klQ4LPv_)o~(8x6_zGz>`4Xek)6qf zWyu9)@#(HOA_-Hf5;Lk2QmNEkk(i1+hBQ)|v2q!lAApv4zq&HvNNyA(&o{`;>p4pz2 zJ8rZrmU~t>?(y71Nc@jUt(_%%-iZ53@hMob_4#dp{DvKSIIvD_BY&*7BJo0b>!ed zdbqSzc$(d17=@ahrHy3{%u0==b_juaW0}3hS)#~D&aQHn24`=j2U?hDs9R|3Kp#KV zJA8^iHGlZb;?Yy%Gsn9IKW^^)7%FINKi$&xX?^Q_iF;UE)LWqLFVqfd^dm)<(P9Uq z-(YFUDv-xcbRR2nLNd=Ep2llM&#I4(uZxIoPtTHdnO6`aoS$l?G?bCXOlcXlz@FiK zi2#7p0b3j-kXTMh#w4n{q*o?lo6T~jwVOOiE?WUW0}YfIADlXaFvttClgP1Z26 zCh4q6?AKY6n9Ck>B3q3dkf^aGX)TF`*2IE>kd%UuG;L&dVOXX*EL|O%p$^N?glDKD z(lp`enuv5=M4B!%&KMeH2#PR=L{Y(@I4Il{98nY!p$iT#3X3X^j5Wt3SQAp5saZv# zkrD6wIq`#yl$On<11!;oKD(OJGpqUU*(zz8mY&6&HVV}uy@({!kd#7aiUe|xs7TUB__Xg%VpTy#}Tm%69Q%8_j_)E{eVKiNBcW?=M8|H$c{fzy4%hhg#! z^#hgeZg*+7h~L8GTb&MJ@&F#3GwhZ+ho#nGtpofD{_7EbJmnMJz2hCNZI$kC2=cE# ztgo(Sw!iDx^z>+FcV~5NtH+IIfqbRF-T~u}XBLi~92)<$z2}pLwoe<{PO=;{cO9#0z??gzD<0J72DQck zgSlT<+?HKHyI~4S*D^3BB&IYbrwyvqooQK(F$s;)iJeJl19@uRKJ2b8IR(r&X5`2s zn~~iBbCYdmeQHKiavDWVsmfapu~2|LF{Ls&qk=+tR5#5~m1bxiX?k0V(VARjPAW1d z8jBN(iW7{c1SUqK%w-fM7?>O44Tc1LQM^IVBwqF~Hzym@0#YDXqIuk4Qhr z>nqguVWLzO_7`Z7@IS7o{kXDW&Qm?(s>Ez~q@nFZx1b%&pX%s7(cZbx+&orO+3P9k zE(bhi9RNuYHV|x@N`=GM+O2hVORe2fW207~rN(ZqwVS=*r#o;*Rn2t&;B-%CS5^7f zqx!$}?Lb}4OmElm*{Q{e$${opa%5UPvKF8eD1=Z#)d*P*%1T%fm^YQm5JGUuQq|!q z>tsFI-WC%OlZw{#tE>NwTjbGp6fczxTHrG!;r zK&|W3=wTgGdhB4v)-;x5T5nU-E~_got0gX_DK@bwF1aZtp_7*m&XZDDuj|t@W$n(& zX~|YW0!OsCS(Z~SR+IWK| zPN$96>*5W%1c5q38>eSKTlp$h$5uwHPM=^Xh$>J;7VxnkxWeE?>L!i8DqrJBN^L3Bwrh)~OI(A7;sJ70b;j<(BCuVXk%=V%eYQR&jn%L>Gp_^W zqga9b>|6{B?Ya3Ks)7zxVMneCoD28`_PIzqZ3_5XvvXRQFo-39E^_lBlQy8P40)L8 znMkL7S-CB-iLjlfn8cR2B#Pt@stS4R)b~cMZsVspJ|nwFrRgiw;|A)>*9;p>^UjLn z^{uS(ixqfIYv-%#KW^(f-7|Q)tM6n-?}_#uC_;#$45yoyB1Bn0G=dOV5jxPCF(Q?_ddrqol# z&Ws*hnwUYmIWskXYH<8S*TAW^o>QHDXFB_NVl&A9L^P=iXHE7)=F-u8T~D@3YzwKG ztqG~EiK%@mZF^2Z6Z3>r+FW8}khMNr)uvMOe%n~tvvT2n=+|DV=}gUL9ibAn=m=Og z5>v|&xl=Nn>3JpTI%~4glvoV-^$A7#M5Ew5!Jv-Q7slz>GB5Pb6RTCn z>-efDxi~X42Qtyb>oP(z^CDHcI9+m38gvx5H_4c6ObJM_r<>CP(yeLw{E&Eae3~^P z))kLzmQ@@afl0(h1D)(VM`CJ+-bC?@MoeKw^MJ|jM*L9gI*ZKKxWpc#xwqK5=&Bks zSq6(OeFhUjFCF=sW)W2At&rK2meG}&Gmw+tjWCmu)sdNlRiP8)=c^U`=eOtPwew_h za*^#@GP7|l$pVp))smIn#vWFIY)An7^Yn7t^B|V|b|DZ|8<*uGNokF-NzKXW9ceku z2`Q~{DZLpv9T~YTsTqCQ`6DXL7;!lKx(Yc~H&$dm?5;l5*#2olE3*A!W&P2H)=xWn z&ma%O<(k_T8-aZ_Mn$+iut%f^Cn89Ag$jpN_`F@Vuzg_f9dP?vhbXzVC60zNS8I9I zAeS0FaBO0Hvb&?F?wnCR^vZtLuCXzr_T z?5?fps%BKRRaCT`|$E^3w}ZHv993toJjV9CLh#?g@{yC@MM<2*RBZbrTI0CEG>Csc zAqC;HD^R>hI+WvgagwcAey!KRM$EWpeN2(hRRX2!Y+unP+wd+Jv+tG%m zsoI);5eYCPmSI0Zsl=}0<(UW)!tL#d^mgW;9g0A}CX%7i>1=YkP;c8Rs=I3Hhg#aF z2ZoPMP92|_S{Uv>(BAwFIr+CgY;k1Z5aQf)slSd98KXmAmg@wb@ zvvXq;ll?C2x zH+*Jjd}(^&%v_WLSzbt(Hu?J*UK#Yj&pPIB6FM z&1UDLt<)014$se3cW8>b4VGSO*{H)gSynND56D%CE>NAFTb--wH`_*SrAW~e_R<5^ zk`YTOB+#!nbrAHPuR**4z5RIwqd1cC3c7PuJt{FN<3d7=>@5WUh3tWo%oJOvjXW9i zq|P*{FIKjegF4ee?C=K5RFQSM*fCpdKUiczfWi0L&2{p%J=myIkUX<{3JjAb$56hy zGbvSGL{9!dZsB-=VYE;;ZnS<Kb@I9 zIX8Q1ZswDziNm8qQ#~Eu9@78qD;zyAeDuJ`vGLJQCdQ6WPMnyYJT*Ui`tTw|apvfe zQ%8?}x_Ic=+`{3hnYjZK5bQ*E-&kAcP;=WLeSsUBhZ-A(8|sJaYlrJ=X4=|M42+!_ zJg_u2_1E#)Gs6>S`VXA$9y;CJb*8c7Ol#j!cg=*sJd|75n*lS=8^|f($@gWddNOi{ z3bccTx&;+((rBMF+NRCUX;bNxv2?ho zq+V?;FL0OSyDTcVN#!x-x{bLWeXd)V?b2nt_1UhXT(^ObIWBfGigH~BMh<)2I`+%A zE_1$Hl~HQQaavSPtIBODC^zMKN(##z1?9E^k2T+I%lDKNxZQ>2 zZgY;B9fL-54+eI$=KMk`z77>=CJGJ1xDgA~0~ph^MFVJ3TK%9}2U|d^0^SIdi>8u8 zrjiAdW5HZ9&(>5jZ!TG|x)%7#QhLZz#^9?Fjebb2#VCyei~Q4*nb)0>+nZC+lcnlQ z&*@Ihl((QN94*u@7THg@s!!H5oowtlTHkW4vHfJ{z+d}EPj~cv+Ssu7ry@R%7KUK|@&a6ThApPvI!p#8+m)TdJu zA0I^289O>Qd}w4~y07cMjoW|TiHC;z=LdTkM`?jd=hNwfr)DO(|I-U|XAUny6labd zJ$>}($-_rJT|9hje&NXU?Bc}1`OyP&gF`bty@!WJPYoZ$6t*-t{?{QU2QUqtX>4C= z>s=Z;xHLTZac%RI-aM332>u6i3gJH^1-em?uQd*73?q8e0fTv-_sgrax#mke3nk_A zCFOIa6>|>HY)Sdd*=_l>xoo<~K2q##Eh?+hmOIo{w!&(gy2eykQ(RbQET}aW))f`h znF?x*-ceX<%&#pfs3}suG8R-96*8|jvcIs#Qdncv)EU(EMcQVQp{-cgYSOi^E!MRZ zX=Gy3HCgn{c3pjy*6mEqDNRVNPt9pg%dSb!A&H41tBCy#h5DufT{}Kgees~#-fyzk z=4;xDOe3YPAxGJey-dW7FB(-}r|#!Ub^6LEjpY;9V|NBhiuLx zma@a`*BvXbIoZ;E zs&inee{`vT>{QRt$9=<7IA>d14z#r%Xm1~D?>yMmJA)f*WbE+x!DCa?pU%#nn4dqn zFn?-c9)z5ppF24_4HpIKp8)moQQ@u!Mh=e*{qqp}PrY($MMb^4Y_O$yw7qSrr)zPj zk3aS4zsYg(Ihir%reFFLe)| zY41DJ+_BW!wbV7R)Hn9m;Ymp2OxF;$(<7eh#nOrytj+r3F`bcdz+jp{Nwkz6v$~Gh zU584`4>_wA$|~o}Di=yC7s{%ji22fr`O?aHVRvN*%&tz8r&eF>)Ydq3b#_gSRb6Y5 zQD@TBGq-BwqwTXfd(7%uXvC_nvlP}A7cj526aaqq*I6_*P>@|)XVccnR#R)!a&C=7 zTj$W$mS}4$^)>Z6Prb(4T411Y^N7ZPk+55BK$Gp&7Yp;ZIVW9JbCtD6>Y9+8C)}08 zgj4~3N9nM=WTM1*vY~a+T{TteAv|Cb{1;nc??^cC`5`UH2mc178gM(pRuAT7J7zE) zFj~eJrLqPpQsZSzud=U+Q| zm%93vI(wHo`<6QTm%0X*h7T@{&o50bE-CWT>He|Py(4FOM4gx4WcU3)&)mHMHNu}yRQw!%JDF4F%$f6?_kG?nb1vSG?B2t6 z&tV7tuS;QP2XE^9q^0YV*3F-GY>WII?c9E}YsWE6*|%=(4tI42 z+m9k6hbYV&`ZUmdG~9Nqr3=ghVn60-`nbXIab3fw4fbxUtJ~)4Zgg~;Y{#%#-tPG{ z(EcfMdVl`7`^e#A`}g35ezJ2rUha|Y6m>+72vj!uB137FUla*aP_w zw!jwjgxb4fm3?=I(8f`~y9^4tDQ4*t6?U&&~tgyASs4 z{t(;Uox6IP;9_m>-n_fJYgbR#E^chz)7`Zjr@FW9foAbw*Pb38+}pinU-#y{Y!GR) z8N1lt-Mwvp_ZFO04SJjH>)yJ*d+Pz7+R96gq4Bl@-CK7b>)hFmzpkC#ZQH4*{?=p7 zUAR{dMs@H=&!LZd4jt(^c$j~G{X#4X3}`?(25&ugBp79UZecyRFV1NWRtC z-RL}8-*^z-W@Y_hWU%+`>;7;*Ec+)rw!+ZE+~b|@YH8Z)_q90f0f>5o*=;g8O=e4N z{l8c6zTtP@+un3!Ygd=ky}in`wWNA?McsjF@b;ijr?-b^J>3@fF|+MsgK1wG zJ+ZAD%XgcspKPb=ez2nz0U&0hw)j0Q4x78dY^ya|YK`A;dHW_0?|?n9ySCwjiu%2V zhW$0>gVpB4wT*u^J3h5}jzJe70FVa=Jmo*!?E-h<&)*aAl$5=ooWNe-y+ht0hrYIM zH}nO<)7k?8hU7!wsR9sk>_#OS(jMA!3?kjMo&Q4Y!`qGp&;U*XPIXbv!`pgz97ncw zhuFTipGbQMf1m?>LIsZbTH(3Df56{SjNImiP<4AkbfhsP>T>6w?cNXJPJ4nM`J2O) z27hsxzqlf>sT3Pqs_QU0c%Y{CV4dlh%XiEZIOYxF-)WEIM4c2?AFFRt2!ILl;ft1x_%Md<&)*sy41e}n5#qwAo_e#q=RY_uP& zvmG=v9;&q*;=c{}_W)i^m1$pj-TumYOhFzp+CQwf9mKyT$04JQyGF-{xUi;SPk9~x z4Ay-{*KVVGx7oYb2e&7_sf$YxM8pk-%>Jnf!Zt{{nkVKJKC=8{^Nn%ji>W zd}zOy%>t1N(aCNzj(GQ(eS3_qJx0%-dgq>6%dQ&p&N9P}O%=Nlk5?J@)mlC@V{v%k zfZ5SpTIX6<8ZI?dq-UisTamhA)!M|=nA!7IOr4%MdoDircH@U`vI}-q)g3Y04r5-m zw(($%34aGF>kyeD>vgcqaImuWV2$~Z!Te#3=|e-qVMD`*1~dPyHXg>G!F0sfcm!fx zZ#`nNeq^*Dp8d#VJJR4dVzM8pZ9toks!c}>jfZO+KH^m-TzCkyV`w;7+i=imJ7Bc! zsWI=bv+Orok*96i@RdKa{XZa$|HenyWApB_`3~7Uhw*QN^Fy0bO#alE_tu#3?>_!D96%ehYj3S>XSMl*TH6Qpt{q0twmR?jdf(Q1?>1v# z3;s0)w$}T%*5Qb6OPz09y?=A9cXOQ=yZGBy=jR5F*G9JeTk1Sp&=Sq+{i+B5PIKr( zd+UBn(;iEBcVlS3ExfNWxWnk$XTtQ5V|R^tYjNcV6?MA|mi_hiJ%$EM=WWHRNQ+~a zRsA_VyIh`KUVkTk(8J=`Zg*|9ICj`vdz|h)F7FPrb!T}EmJlDPHbLe0RMzb-heDO( z6RMq+$n@e{661a-MQtPg-4CN#+i<|pxVPH0ugZwL*&(A9$M+e`IDW9sasZO7{yl(} z)oj1F+Poh?)i&Y?@Z4Wh4}r(u0f21r9JILim~EZq2K*ApzflbIt?{KyS7jYm3+$+E z*k!ctHQD!@9ee8@Z*B0Dk5AU{z_So>(w96jaWpCmU2mjq`Z{BTh+U>x<;a#>cPVKUV@VDC<-h*b? z;K491L^B-0E{{9H`|M47?V-KSW^Oy1_S$eq&mOaVzsbJOY~N#c>}z!GvwOE!8#~vR zV(oYP`m(m7O&x1DwXZMfSYOsrRNS$S$8n@{?WWD^%lL0mF*ChIn|5rhIB0PlvU(3S zxDJB{O!mVj8wsHS9~?Q6_W8)_K5TJ(gp80486ocxyB~`a_EsCW6<6%5s;8(|)$y;o z{yRsiAU8M$$>Vu8+WOkI*bd^?p69%crpb^jrXaqC@8Uc-fMnEH= z5zq)|1T+E~0gZr0KqH_L&He>7(D}F5{{@PFd-^Y);J}H_e-45v zCp!Nb1b>BHwDIu{PJH?Cm+*HkKmLe@Xor7YM=(4AHqnN+5yZ~u-rv^5=Cle*rpdbK94k@BfAH0H7~~ z=cJR)`UlUS+{sb?{r^tx{QKv4=HyQN^9BEX+wcV^aY~=p_WNW&YPc`TUpp zfB*FJU*f;s_Qn43m-)~Cg7aTUFJJ1vf0=*(<>#k-;Q+oQz5qUkFS;J_OXojw9s$-D z5-j4UFH}&(UtcI7w;_Q0qVeJM7d?OqAiv-SkqGwhjxZWx8~)A^cz6EyckXaJK9&UY1B;SW40 zBOZK6-hS#S9D7|#=FXEqeA2dHQrfC;X6{4FCcb-yGc9X_FC|^9D_2X}yYI=+OZv%G z0|&@MzxlP?@w1!d=ht5=x88WYTyg2ea>XV6rEuDGac5+SY3T~FBqWzO(lR^iSFDnX z*>h#(=(lD3t7prH&EGj;nLheQk~n^XG;J!DvV}{eF(qS*IXSIq6RypjI!%_0ctxf> z^<$C3(_}}&sp4I9iue|vD*Mw;RmbK%f37Tgb+mNb$;5lDBxVlq_BD zsf&#-sa>&Js#eEJ`nU-)aabP-{%o+=Zof`U58ou_$9^J~hX+gZ9T&=stNWqNcqy4P zPfQ6(QoS;!+?l^-aq-**z;?c@ojgr`J7}Pc=+jqT>C;bM>Dy06uwa|@qk{*_nn{yo z&AeGsjP}-?{G}Z>+b=pT)=+1ivAe0DNKC8aB!B)&SvGa4EPZdDBu$wnxwB?V$>L>V z#<@^VfwWdu_p};~?M)`r?NM0>+wG?v_5^M?;tY)X(B1sb0e9Ow2VJdWcQrc3?Qd|5 zJ7V{aJ7RT@Iq34-*5+~jD0Z(&zY|0)mb%PBm=pOSfBC@m|YIW7B8OGd7Q(zC^xkRpxo z$zqI&m*V-0C2#sH$%I|VLdnN=@%%+%ib)WAVybvh!fBb(nw2AM=~*AOCTFCyX5>B; z$|^V^1Hsl`(~`S(qBl9C-W{KO)V4BK8kejP`>Hr8m^@V`JoBvl?v7u{FRs2y2KBv2 zF1_%4xvb9xC>P4$etpz2*0k|IziOB~dfTls_Q@w@-He&yfDQM=CWtRSNrK5~N1L+p zOreaNN$?SFATH|?60h;3WPTjZ%99|*97s(U+sYUzUa&~w{xnLizU)#N(!Z}Nzr5+E zGU}BPGI{)XdFk0_H%Mwp=n%Lvw#k_0<+AI=dYMP9B^idglaX+~n z_m7q#*k>8k``pAMT8P2r{aGUIuK*r1pLNy*Y#Q6()UWnx&mOltA2yvgZO7oSvM z%gm{FW80pbDmA!o1^f>K?EboC%jC@mA5{0R-0(fw(faMLRGh}|$*@cN%amuImHMO< z*==r+T}Go+dFQdmaR2_|gU==v-#=ce7ky94=6_Gt&iJ0Jn;9woB#iNM zZFc0HB3BN;{hxV8GRBURO!!xYlc&h0nX{#A(Nbwl&ym{r#Cl_L>Y94U<_73c5yrVT zDNzzfkHIzLWi0Mb-!wSuOp&BBj+c&kr^~@rXUP5)XGnMA8FFCd8M1fzXXn{==h8DH z_y7H~68qL@SuuLFB%|NlN$*MV?0I5@j#kCSm(?eyrdBRkB5M~dlERqPVl3Gt2{UI% z;ydGHB7AK=e|XdnMW+2wIwoE%Tc%tg9rLc11M@Jhn4ik_C6~kZzESoqxJY&_yioSe zgP)7GyC?imt{QNOO#kBxvH{OjR*);{(1pY?<0J|AmO{VFW8w;GQ__|ipcBU2JgMKX zL5g!S#a3P>+0$mojOU(L{(0c3AIT@f&ylxpUm-7yNSFIpmdU;53b`XxCmWVMDX*KC z$Oy|jlCkP~Nn3WQYP;jPA1U2k^6Kki1rNE3*1?aRDjQOgF_t{p1Yfr_HZHC@E#tkq zCCkMG&avhfO4ZsmisMS514~|c6*7FW*nWAg?7sIrDf;EZvH?0#_2k{M{*8wu>%C`X z?fA!~@Vy76V$@BNKmJ;&dASeTo-b{8pC{Kq|5v{`N*D)J;Tg*j!`QQP#kXOj)IbJG z5)!A?lyzna(->?hi+Ci6r z!`_ta2@@fwv+&GwRo|7+eGH}n$H_3H_2dZbeX3zjtUpx4+$T8H{@|>!< z{g>sTU;JFwzB^eqqTj^$nc$a&(6gCRgP1V5zIc1h`gLbnDyzQLZm|Tr8X9GDol!Pd z)rt#tzzt3|AeJk`yU4=WW5-XF=_AL;gqPot(a#T;KRx?`yz%rON7Vp^(ZnvRy6qid!WtDW))k{--eM@Cg(f`l}eb5x@+v)SJ zYIWKzoi68}_cyo5A$LHwtSc7UrPll+31t^bXQfum3aE)P1+v z=@@)?%jVx4ZgkH3*cAwTeKDK%G9kIAJAFOMPKe)N`kDWfxFT4#6Xa4p5^u9(wBk=D-;CRB| z1RIzHoQ#RWv!*bG@W1$v*YWxjbbb5~^Ay*hJdbibPcaZX`ag|;MnEH=5zq)|1pea@ zuq33M5lqV-(3G0}lW=m@uS1FHPXyvqUhu}HjC92$jdHC{_>(^_`ITUN%5$NF)W<@J zX?KT{(r*kUXI$b>%sA~|2>|l0=e8tg{-Gr)Jr=oFds9lrq1KE%Y01o&7Q|!7AqwUh z1@n{QNAAd#nj>x$=J7%})`W8{S^3hMoi80Io!IWo%-z?OnP)>zHW}q55b$x4k&e1m9vl-(-ZdiQC?|D+Q5Qk=nEh$Yb$m2F3S8GImwQfa>7?z_f zTdlTvtP$s|XyfE{=p%qQGK73xb4Iqb=j2IePQG+y<#o5GWK^`KX8taio_EGqeU`?o z!v6{66y6ld%zqF0=4xM3+Q*)_Byr&W0gTa$;zqH?CP@wEWHXSDTQKqs8H-#n=YOBP z^G&WJCcN7zp`5~>*)t3NVlF!RY;TVm_dn_L7row;vnH!0 zr*M}mE?F=)EoO|*x@@I1EM6|w<*USkd`{u}Q)CqKxA$YNnRzgk$3=d2z(p6xB^R8h z3iH_9zx1N>kLPeBx!eoYzREja+*e)8T;kn7xj{xDC%6gorj1LNixc;9#l(qcb-dV- z&+#L#+zi=k&d%Q(NXyP$HgVG1lgCdu{wy0};}=5*B@CHt$t@IrLNeqdUEIizH)8zN z$O~j6lx$%@M<8+?#)rU*VlZa?+8!^iVv{&xQ?=^?0B4%yT2hVnDvffcH~_ zyi+~K;6`pJfH{W%v!c^j)mz;9FHzTLV51H^Jq)HHZv_K}>WM*M*B2gNU!>i8D zkUXpf7=F(^a)rv1W9+zJBxl}Nh7Gt>euDXds|O8~o36cDuEjGB#k(1H*#Note7utj z&r@>?>bZ~a~Cv2kGyy;KA#KnXTm!)ASYFWyf>fQin;f-(`L$3kl|?Fe{e7M z%c_~jyE;Qui zrFt=L8~E6a`2xd|rN~XrlM>`?OQ4VC$f;H%S6PH-e;V?`d4q?3iShfs_qCQ$&-_@; zE4==|Z&hA-6Xq<+u$FCOa?I4Sld;KGCpn zDdvD?%SPl#i?D8_5d6FrGPiN+Ov#`4o;>rbU#h(OBe!Av2F$zsRrxdKS=w_?mbRRe zWlO=yu(v0}-kvNSIM#%k$Nu`7=>A(zoU6t^^1%nxyi7Lox>?BA=Dqixtic+f4ak$1 z0?P{63R8NXxF9nniAmX2*}01ycs?`mFUNez`pMJO+++gQ)+E3Ej^s^vS8~U{D_Q98 z4|m?7-tl9%MArOp-OuOCw)j(J#IL_6D~F#fYsZ};bANxbvdLrUUc>nloS<)rr{4Y_{KIVHdKw*2w#yVTs-6E~eJ5{J2` zud%#yUu67mV(v6<)M#1#&RAJKVZ5vy`6u8wM!kz%$iRk~n1=*+Hek#f5|dU`<`=xv zfM+Vj-18d9PAuk<@>i}9M@_ZNpD;m|zWuhuVty+bc)Se$=e*66H)8y=fAF=Iad(}M zIhjjj%o9&YBJe8C%~Eq)xyzPH3b-$I^g9xRXGw!Blq`o&1-rT~DQQ-9QPBt^=DbS5 zGZo26QeK!RRYiqTvu=&lZCWq28`nuZ<_9w{U-BpTF7z3my7pY@8FRKOju&s2Hy?P(gdP^ACQ0V|Q(V%)e$-Z z2hUTb>;3`q&aDe&$s=#eY*;DR;`E z)%VDvWjD))F&B$z#Ap4Cz3w74hqZ9{2w98yurl~OoR6ykr%*PWd4&>gY>-f8C45%+ z2Fq7UX@1_lz<;n2{8PPjrJ5_OUA9UZvJrb=u5<&|7SuqtHo$ktpY*=WeeosXT-lQY z&X;YsT=2D)cdocd&Fd}2+-UCmlck7!2_J@YYvt-W7OV9RCite5of@n^EX~io#;|e2 z=_c3+BlxER&sENDB%*_{0nX2YTH*h^7=3zPKR|$Wn4)S71Op=nkyz}UT8Z)!^8xvA6pF2<1 z!^V~3IXIVDg7>s();uYiJRP>|eMtfjaPITb^Dhw7xfgw|T===|=y4wVqQZ8coqxSg zAGvYpU`ZYIHf-f2rzLrjc*a{^Rbz$y;&}6+ zD{Jr!o3P%a1im-%=lm=E2d=sKDb}%Eg?rt0?get|x#y#>zCdm}?{CYk>ez+C`WIhN zZo#=5`t_CD;d|$dognLALyJ(dm&}y4cn2H#9I#*c;2^F=wIh}>t|{^pd;S+Fulhqr zNd?w&WMj|Fx&57Cq``54rU@hdd6!%#VTRJyus*?W-jHLwrB-hMhU5j*lYyL<1KNA9{?9{ANAa^LN@%RTV7qvh9#H}Ct!E%L;#@0RC&`&)S( zYdFUI;W?R#HOPx$Z_~jK)V(Umu8BSZ?5hE~+6Z4cTv#McYl>vMpKTdfcTx*bH|~l{7b)WIN`S4+WZJpV=ncN~^?=cV@*JF)P<_A)c`zCgu9x znz%IC7*inYR^_Wwv@%cDEYFsr71>y~oPs&|Sj18(vJrY`L_FF6U(pV}a{?O|uymzm zsM?W&wMvMs8qudeGfz4&N4}%NfH`j5-)fU)vq_v)<>IWalqRdiV5uxSIXZsUt&YZT zAKbe6;XNH4i*_`H%es8t{agG#+1t`0pLA}MkF0LlzNtbwFz4QZx$~{-N~CkmCUNJN z$cBt+S({!JDQQ(wlwK_xGHawNW1~3Xr-#>_1k;*w9Ce)-fHkWT}xa@5--pW40h2{1ZT+wjqc!Eo;W z&h~km{l4K%cH7OC+PYIuG{&!bthv1QJG;$}pYAj?JoQ0E-E+Gt>z~_EZ+~t}z5Dqs z^`1X&W~p<(u({s-;trGd#T`b+OFIk=FSo5P9p1K~Ys%H8zBeXyoFgV_ zoHHhIgll!&v)1_J$2{?=zw#zz{?Zkfe61xe@yfu;`14%}sb|)&iu>;0eF%L*Bk*62 zfH5WIJB>-HXS-5!hq+U;Z}BIl-tSAwdOnzvH5O&2KPhurFfC_IFg2$tl$v8f9SEgl z?Fyyjd>l&3IEwS#f#eJcCZ?+ricgUsia$P8e6dLq;C5`H_~Mc!5SxtS*vGaf4(DT% z#2b?=uGPunSe+!!m5Jh9l_-vt3F1QSSs9CKd0mpahSvvitq<1*;?Q?YqPkZwAytA2 zxWB5?B$$Np#3lCx64Qo$+aefp`&EASJ`u$01wZX)cVqbi6jz2MNfj=>QN+2cs zEq_AV??cI1cX^UhZ}cZ-TmtSm?Y~?!^yB{rBak+C-VfI=U3pdI>iAzbCZ@e?O-P;O zPDoztPR=OsrDoRz(sRAxjJ&-~={cW-(sPcrWaLRpTCTLDWJc=LY-#0wQkpcQv?Sv= zN-K1|IVoLQpzp1TJdSoDqY0{8%RNY&6y=xqo(985?Tfbg9qHJ z;>8Uf@IddqtHBqssW_J;L2ySXJ{fJ3B&_;QLI1o5>l8J%=0w~NI^T>jwV_Q5?$^q( zsPUxs?q{emwy-U(=lK>@&=;PIeKg@&T2iy5IXR;z3~mc2r+*wy$=ni7$!ZKHXKo6o zWXA>))29WJGX4}uNWb3|mvVz6G3`g2X3YQAf3T?Nq5p>>P#%~3Epu{ue_LY8ZyVzh zC%~Uhu_vTi+{u~y{Ha;T!fBA@v}|ckh2Ew^my@Bh(C^l?9O!wL+NZv^Qm;|B;aEF# zpdIHs(7pr5+EF^vb5z}qzPR589cg8s*l*)LYSkZ(D?Uht9^qb1EI1zqe}qypB!qo# z2cdT%+{X_+^+6Y1u_@w&&O2jLu$>AX00+dS!_KF`e}L{|d_lAiLq|i3SE%>wKf*<= z3@9>adSs?ioT zx7okK0q5Glk-(xmoRaZTb82=dn3`4UOUYUqNJxV}k@4$bTHdABRf*sJ4;3#x>~9zX z@K^hI6SE$IPdf|#Yi@8TDA%;$A&zf~}>^qpY?=zo`$D`A+II zbs#-kI-wVBJjPlbgHB+ZJkXYjI;%k1(XK6LjkIJJNDGd!ZqCY;W^6a(d{b7QDxvH= z@n_^n2>W523$fsQFf$+9IqFydeFx!h1k>}xgYh`w&pYDN#Fn#Owl~-XJSI-~1uk%d z56|X>&Z$0e9qV9bo`f>8(HClzCftMDEm`@}g!4@pYfE-M#+8Th<$(L(%V4~%INzEM zoyPGtjHwN6J8)eG>Mk7P^LJsKU1;Bh!>RZ5d7#ATj_rG z8tH)icjT^>w(LSxsN>Y>*4#qqalW+VU_WoIgi$w9m!SLM+%?jaQz#+qx8T|^+H=1d z`%O3>!hRTf6u|WX91q}l5G8;<{J4)g4$ffhgZ}&AW4JL^J8}iCtU|FEJ+e92CHDM{ zVr5<+9zFm#%Lm^f2)*;8Z}Lb`^#Kk+zd?*Qg#OgnddCyym~gx~2jfQFl3%2rr3u^N zJUm}6o-==~wBq^NaE{O1f@?ZZci}M}J)ye<$v<1!XI?$pKq2j%~nm8}7Re z+gp3Lw{sucTYxutVk@}=+g+$5$ACYcdvkW-r(K!34q(0%@AlEA)T|$U!^BSy=ih(; zV#)w6pH;Nx}?}M%Nv8H~b_QEc+xG^R#Y^n#k;6-8Wfz9;7CvhTfcfihDGuMc#yixqM zUNNVG3y|A$A*biYc{iTJgW_X92ToWcorUWoypRuW1SX0jPi?%xH!NAUpexea(Ij=;Tl05fvJ4jkJ~uE5y0;5}`@_{kAla18v=-IbMZhp#XV zG1X5HGw6K6zd`^09k+C+WcT+Y*WQ$#*Vvqr+0z1@ppLiEruXVO?K*R9ttL^;FFGAFB$h^{X9BV=e zLe5!y)NRN&^*)F`e2{S-V@(}YIu1SdI&GXoH zz&~-|`KaGc*newsj@WZIh&5x4G=dKtj0spH$LB=+??o))W`E!gH=dO|pzfpYm#OZ{ zF$GY_IU&@3#0Wu*mHq;Gf&K~o5&A9UG#*pWoxcuzw+4K-MsYwhwp)9-pbgy6+RF`X zh2V!Z8h0iH zcKY+me}_KMwnuVU)N{nEkoi{Fdgj!Wu0x(%pkqxL)N#bVkee3R-WK@vL9`93bBK+h zcZ`oCemZP7Y*v7}0C^0+-}XbV{g72JY^WEu%A1Y&8hY-*wui@hZMX;hd!S27cUi-y zpq@Jsms2NQ*mpy>UC6;Xsn@6-@JXB~4vg83W6@$wN)!yfa}Q{PW0zCknevIYKaGh{`@-PjJn4-P}V!-%JuXAhwGp$mTKWdM2_ z05>rHrq4`2*@NPX>Nl=+L*|_5n?AbIad3!B74CBzx(auGCqu83#ESVk3-S}u8uJs_Q9k zRC}-9cfcQQhn*n@v~e4<)Q0n%LvMoYB8MzZkhdTUWFv?^v91L=pC$qDNdViNYxCn8 z%ma!q8S`vt=S6$kNBYoC9H)(TAirkEIGiXB+Hmx*Y%1DPcX-SOy|yM`jSlq8Mq3QO z+(w_B`ba%TX@p*yq1Q^MV-lnudR<5T#=L&rsyOV&NIll`7_nWuVzt!cxM5|C7+1te z-O4zrMqRUfwbZPL6$5JH3S74=Myi&sK>OIpH7n2_yn!`+Qj2Tq(1#KCFynsY5i@xN zbB7JF2^do%cp_2k*tRi-K(WDpuz?#~z=d3;_y9Z^^=G{B4d~}6|AD>%xFM3)03Sdn z$q)1$IH$pUWB^!#ixnbsc%|5#R%;x>xr*!2_ES zM|6Q#I^laU&)5mSU^Dn&7twXd2z~la*yK+5;BC4#fzjk49Kv6fb>;EzVwbJ|$iZQ5KGV@#aVH52r$k#-w;Nc~p&2%R-R zKWmn)62mfVqpn3=y>yk-K!jOXZ?vQnq** zwpU2`!lk$d=Xf6bH8@wfWSP20^^(Yas+Xc)+^Y)x)MDEJZXv(afisNY8#8!<9MOQW zHYooi;xjY?BMXXyehYZc1>Ld(N5&M2A6Vl(d4Mzdfw6@u;0QnY7Vp51ZE~v@zK0j@ zA%wm{XcxqFA)E`Rf*0`aS{SEg(N92bh9w(wudw;5M$FL8JYx>#Xt3Ry4PA!}bbt@k z+yi)9=-VG9mptj;y?& zSW~m#g%4ei{L|3@YzTF-2{~M)x3KZe+=hKoYt>LU)7Hc8(l?y_y zn7_xmT%5z)9Jqk5Y(MGqxFj8)3`qVHfE`8(<@Asoy9yOR3|qrO*}H;xgzJ?J;$?5<04M96CiErHwAd z`BLa+F?6&9x>|<10=id)Yia-Msbk<1BlJqe28<6d1~YjFq^lDYGxf@2|9sBz;NI3|<| za0C5?N{qXjg}jC5Frd_tBfxQW;5)_}^l8kDDNyi@lJbKr@PF(Wr-S%n>~``j_MLb} z7B@J=gJYi9IPgLuVlu2>#Cr`u?}P9og5ZN-D%LQj@jYYxBK-oDFJQa?{|_9{PMOHW zHh80%z5@IQuHodmWHm2@xQlZ^9pL}S+Qdxhg8p+ZNEON#u0`IrGmxH>>`u@9y(uZZ zZ^h~~J*W7UwfwUS8Z)xKZO_WP(wUk68uDiao{Zex%$G5~4nps#-&hB#biD<~k=I9w z*lQo;zz;d_QOEIqI1k17sW9}JYgg3z705H=du2DkAu5Ltz2^M2iVG+woXcYlf!u%+ zhHt^Obzu~)i%~HyWYt1>T@@o{6cc@4l=@W((D&F#j14)by`)cFsq`VD&&p0i&nuxP zrO>5v+F$5*)#9Z}ha2M27wnw{_Kh~O5w+4^=xy!lSgC*>uAR3~GN#RxRpTbg;x|Xj z>{nlxDK8C|iO;_vW1fCmMnCnmy!pE)<&7tvl-C}86y-5_iMTmX{xTNM6OU z*B(Q8{Bari_!FxA=qI0&G0!|J|-dwYz;PnsebGv>%zauV#j zk^F&oVTHf!z&mgPYvwRH4{5=;Y5VDal+RtHxUC92P)R(%YvelvaAB#%b2DzKL(EYN zJ~ORgS*hY46Z{4XIKr}OwODZ6jy|o3I~>5=1CDcnqv(tI8GFIzGhXwhKn}nG)c+v# zKLq_J53mHt3&KW46gei2hjdAN60nI z%on!fdDZ$&$XYAqBO|v5GS=+NDu{KbSodke2bvHy`9Pa?%%SCjSx{Kmh?;P(I=L3)dd}k(B=!FTgWP**cLT)^` zhB3JxeK9V_x^d_}-aqULzEwjW4>^3CQ}HG2f)D*xxEJcHus;mjJkD*lxf=B~sJZ>qp+hjftK_a5ua}2! zy;YvO_daeOKxWJD0Y=R)vLJIOV z$PdsjATM}R@J$rk2@=BlXo2q*hD`B0iWcZT=N-7FxeawE zY(#_uB5S!j$pw%juIcWC&EZ<^HsqBsZ*er3k!$g1A-*6G0)o4 zi+pQK|L7~+m=P%0uwAVs1Ba@74lq={Icoa(Sc8vmQu&=q#OGri=)*hl;ho?cJFIDk zZwMK%(XXDnK+0##l8TwL#Eh8L2|G+1;pg`)c%K3Ix)y zF>&^c*)s2qH)ZsbPs>wx|3>b;`DXdqmBXN0m&y?6^Wc7cWN80>GNfN$xg6V<_wOqM z`(CKZAk>%lyGRE2y+{V39*T2A`}Y9{^pBKF`^!*l54pIn;)UpO#TC76sLNL>y{3*| zDpy`ceZN%c_7%8(82JQkFGs)Z|8n%jaSTNr=@<9HIESJgxrcqR&mrue*JJx?as&Eg z9pR1ssKFrv2gprVM0n!fpWP&XeDDF8^zv|7KJi`2m^Dj^79sxtjxgc<*23PCE6e6D zRh&@GxB@;>HF6i#;ENi>9`)F-2li(8RgHN6R>*-BTORXrUc{>=p!I! zkd%a6LZo=1>t4)L29&=5-3Mo=cmntJBMxas9MueY2|`{1$W;c=mOcX4g0#YyXyqCN z7UUW`m^Xm#tM3tDCpfo+bw!aJG-uAesHfB?~~u9@SAtq zZx49dP5p#E*<+ydhy@#92O6PoCd8n%bLLCU%(-I0wgY~o8#>G4!8`Rshkb}41L&8z zBQJFwwIe<M25SI8gp z^cx|caT6xW`!BvI&);*e-21beA@@U}4+E6GP#1{Pv(ODy6{dth||d%Uhq8U?YM@G`wr&I zmM#|)uBnHOF+g`IPn4}1)RipoDVXP~m=9YIUa6;y;X2mqA%{!f8Z9FcUqAlKU&_y~ zyfWfL!#7s85jE}U6_-SO=!m}_(S;$@8>OQcDE%G;-HPZl^bop#*+u80Tp)v>`vbU5 zUGF`1Y4jZQl{!0w`i%NAltFBVYt(+PP7lOB+YMId$ScrgY!AYH)bZovh}=7REW#lX z?um>q($8h&0TkYs_hNe%^4XwXeu(k_{RVQ}Fkqmzq5Fy-p#Q{W$mjfsVFUWBxFQ;7 zFoya0l|$v1*Ip&R{>k<7>f?{eyw_fnq%mWVGg*lDh5X~}xl%EAo>Vh$ihTp#V;#7n z94Xoc;#+0P_j_UX$xY`CWJ`VoZJ;LcgF*<(J_L zAa6)6U{Pzv;TvG>)2H~Jqt=~OFvgx)cx`P;_J7p(%rz+m-}bL5zAjX>X>zb&y)9g@ z?ikm;t97x^W5(%W6wa-&X8xY*XM&9Dp^J>geX#FbGv-DvnrmR}@B{2|37BW+@75r{ zjrYlSTL+s<8&C}ysD=%;!MC=-M%&;A*rChReFt+yz3-6udN(-5g?sr>bN-IGWix#C znptzDde%Itg5OX+eWsLSzjWFRDV{n*DsbG0{uZUU8fFQ1umdoTz+x?hzOOUA z0O|o3U8HL2Hg%c$uWGiX&ZDsHJrp=F?jbKQ zrl8LdjUyud0dow?{B<{FVlbq&kGPM44^Yy1BG?rZI1Mugofmd2kTR^PyeR)26x~8yKmod*S&)u zyzjSnC(oF5vw!V|TUyqYEo~_(X>TtoR^N5-`*qIEk^iXg%D;z?Pg{@g=zG`D5jT8S z#`p?xz8gN41OApBg?_ksIdeVmt&!_-;CNn{i^T|E zzXG;oHFDk~kt4t7rkhlrj5#zVi_~$*B5gc%ow+h4yO3eZ^p&*vs3Wm7ePZnQ`r14e z^`E2qr|dcO@3M<%lOuW`(f_EveL$b{mF`jBmHwk0 z^_l#jdgmLMN828qNb^bN=bloGn>z9UY=?R2GQ%!(V!R(hJW$-#Kf{J7Rz>ZCzI)Eo(PQ8%jsP zI>bWw_6>PERjBviKYkk*0vFJ}bDb;mYFyXBZ%JH?>A?x^*m&gYF^>j2UJuT$$9pn> z&+C*wzF4fiww-p}1f8v(Jx}U@aRaug^W+BQ17MqT0DPyME3!cUsp~f6ayW;~Ir;{? zQ|1hqN3KM^pmh2yDMemR={$6Q`U@}1W4GNVHx9jA`SL1H1|5&c-6bl=HVn4?N)+bU zD5sQP+OSA09kKP)ab@d!<(j@e&qs20ka27??xu}Z{yB7rdN&w)6FJtahtT_|ePvev^R!eO-`WSMLv$qU0f_b0Uz9R^Gz~s_;BPB8T%)~Pn;j&0q{Cy%naK?E@*&X zV8*^3_QQo3!il*Da)A>vM4!O6Y85yDaU|r%5ra8L=(QView9@~VF(4~!oD?yts;erNOS@e{-VEL+!X zkam8{jWzSM^{x2EtpjUiw&K`ka6vnGpc!#Kf7=4w1)n|}>s_&y1i3ic^g6`sY7Ph3 zQD2#NA>W#y=QVQ{NF{u5+Vx5lGh%Mqau@76f4k0uJe~{jney*}H*LKt=-bYmJ9OIw zK5qcWH-O(8makCvZDjv=pGLfI=8_oKm(7`n`(YgmVu{x=C(isd?Fr-0VM?Yxn-8O3 zPq|b@$!SD>l`Y49)TT$}lyNZiopC%(HrU@ z^(dc8{t1%Q5>w{16UYOLXND67p$xyL#)XMP8P}{WX}nE!+7&}%n3V0UNK@W zL;0PMYgca$n)2&{40>UYICstYNPaICg8nx{PpB*EcT})8sfBBkdU=7|z_qOSCJAu^ z&Ox8}9VNe;%0R61>NC&C>%)hCJnpSABgc;(_qDD)dTI3dZ;g9<{M)nMpL{eASOoA* zWoyBD{LTY@*MWIW<~K0E!QZ&qTC_>FZmN*Z^~J*PYUuL>-~-qZ$JZe*R}IXm)AZk2 zIDcaW&TjCg2ZeKU#I?*q{IEYZ1MtjqkN3l$*#LS42jYZHG)= zu4I+AsIT(zsXw&gD(3}#QubZR@Mkiu+VqR$04~1Zd^z`rKa?Mxd8V9m)|sk4`;0T> z;`7gwvra!v&OZGN`QhoO%bBO0DrcN}8tTu=4{-b}?DIUg*+w11b*y>(ytB`SUw@&T zd-f0I^i$Z!De69_pL(k5>vZh*JOA8>E`wX5I#1muCq%d>!VAZr4Lm?S=ef)KNAy|Q zdbB4G(MKS+5d-BbpdQpO($_#<1I);Ys)l|t_rb!LiP-TRuT>m?eey*VPsI&5MxP+c zO}v-#7oxnw`;iOOeb65`fqjvuRn7oBKyG;8mY+!>V#Y?~04rzDRlIKi_Ztu^n2_(a zKpt(-{|3ZXLfhW}Z zQoJLsJFSB6z?^V3-b)4EGxNm7GiIWIUvT`jM;?Zcj~G(f`-q&711Lk3SIQcF(umAa z4mmd-u|ev%l6BZVC4Uh)rQT3hY4ajFM0?p+F8uL1G7x<&o;F$5r6tR%xpO6b^$L0B z(MRM5r=2c0UwgG=$F7vD*i~}gkIs?D@BfXA8Tq=La`N|O@$@MYzi5HvC&o)ba-tNb zrO1ze^dlJn&RLt9h~sgR6SrDkflo-CzwgdFaE}xzN>7zXF$a0Z_rEVG%NNT%zxtJY z@4MfX)3AT~Dc@Ioa{4K!$f@5ySx(2Xv%nK)fmc|c$!%~;pL5RvuaHkpkyFs_4AeY- z`u9&!{hWq=cu$@ur(JaJk0aNjpEGeh!dIuMzD`3M^4*Wnj&095>nz1f=YqEs{}5YM z`lOBbC7kg z|F!6cK8Er&=FZ35G<{7J)qbJsld&t~3?JeI=Dz*N3kKoulMmGIz2#yak-87N+kwKh z=*^f*;5zpZ;t9nC-~s*~Qz~-SGscgV=@Z^9n>2RZKm42OvnJ1cYuW6XJ!P2l2*c*{ zcUqXw3nPE0zN2Sc0^fxB3!Y;RpdCH{>$WxPVgIq72l2fdI!>;mp0jY>4Ra#+ZFu!N zv;3Zu-_i$hjK5z`|A_PSUK9`3-gsCees>~PppCcC<|77ZVCh|R%Wu!Pww~X_RV~80 zdGLAJG{p5&BYFH{CBE8K+r^~m9uga2QB_(8QME|R~G%t6x z{;OJJgX6UIv`%9i1M^R>zx0y0>T1POT_c9# z61fci!T6D{h@-Af>;{88^Y|08YW{3-noKhAl1s%{x=}K*b}i^|%IvArW#T*IO#tK@?>#=iL`_~c|& z)@5XfrMz4Uu&(7)=zh$?d19?77h7eySWt|`nKiMqz9Q zabrwQJio2FTBeK}t9&BbeXci%&WjGk`~Y!ejwOO8eUOO$t2hMbcwG4n;M^$Y(fC1e zfO?nc1G>+5#qx`5u9RfNvGn`Pkn^oV?xz;>JSOPB5prXOu5(_9Yf2ia_pl=t*m)ED z*J`w_fDBb3XH*Rts=_)mUSCBnL7|PSfM2QRl%VJIGZ@QOfhU+3sD}R27ch`NaGdK^ zcn|uvoTp|koWG&s1rPA|Ve)noEH~@U0@)gi)as_`+x{2fD94W>f#(YT=)?S5B)Bg$KdAX*a-_`OS{Kn23 z58jBwx(wu79LTjafXXuj(N|`Cz-zd^hVppv1?S0~KfhT5R*MXPKBui%hHa}PAU{EU2;1!vv{_{h z^1?~Wmr6YBQ*H4E2{;@w8Ec6=W|K^vI7vD~A$4xf)G5;M!VBexu%&gIHo=crC37dC zozW=qxW1vXLQY5gz!)GSX0@zD4t(*9>5`0`$Z7B=Snj>!4yi9KmHrp?k%#WSOKgT( zSvhZ>E0ecIyaFEEDE)E&XW>7Yzy+%o z&X-y5y(f3ydW)DbZt8mq=5KrrX1VR>KbO9kzkKoce~?kHj}*VfB6FuqmOnoIB>bU^ z;TN1C=ltLY;6Uad$e-i^=1S=|MRJG4Q~6XlN1j#j0qThT@AbdP&1e_l1o#5vXLXEu zQM?m%Z6CyK7h^u>M%e`YD}kJpL6@n&bb&xGc^&*+>U~x3u_#AGxqx|W9;-)88u10-^THS4x>j-k ze=~}|tH$4WreC0bgBknH-~oQmq&R@~6oucb@HZv1z(2DfGYe+U$elBF+E@NIaq--R zePb6d+LH^J3M0?Q-=F1g$Oq`B<39NRSVYEw{DWGb1MUf6d?DOB$k>Ix0AdC8Td&|U z^-UFW1=MwZlgYJ~oFCSf&l^d=Y z0$XjCJF%YslD=p=c!>1t(^qCwx7MwLzHgGbh)rI6=4lDoY!Y_4B@wZzyU`*oUa$P& z>EBB$YD=A2N{ZIYjn~}(-8o%Sz$G={fGm82>@wEN*pVY)TMg0|{=pg0{}~g;NjBEG zY{1|31p{xpQO|c%u|{`VHvy_&>cNnelOQDmel^!9wV1Lq(bD@0Ocx zR{XF6*Q6pRKINS;QnS8Dp2xRFQ^vic_@M#*!^oFjmO1agk8wf&!CN<6eHHQ=r^y|N zzn}Q+{op7Q#*-uq@T>#+_Xii9EI)?+M|3`dGr5)X57B%CabzLBJU6hHE247`SeY0Ix9SW4^a$&Ri*;JWb63F*ikhq`yucFeCTG?|b;281sh8 z$LE|BZRIlY;X0lSa* zC5Sat}XoP*_JOOhm_0)S{N_=Zz=gG;{C>5~xCEx(s_QLU#Bp15R zlE*T3g5W) z*+iS5{eIu6E)lN%fB)zA*5hd{&01$wO?6duclG)9{&qZP+PSBnZdIH*wZ_9+BLuV9kv3T9NvBH)mPb1$&;1eZ{h5j7A-EcKmF!v zxOR=roi)p9ONuRF#iMrlk1w^lk`gOIrw_KZ+B)=mU3s}3gYV#=kt6NoZ-3jmxTd_I zzz%~;%tSv+cUKhTT3uPGwPPRjBU6M``4CV4?su^bYHa+CHy{^kEH^pDzJ^b70qADCO8D4Yq>Z-Tsyq-lLY|5snA#p1UW5|F1ip?&)q} zJ;v>M{!Xs4)iKH4vyin;uJU|!zJ|}a^n3ZQd$GlsT=wsWOc585?EdjNXIeJ;AOilw z=#R>UiybDL(AiB(AH@%_iaB$i1?DfphrEG4sO0vR(91QSKFI916FZQPP`Ly0>$P#d z?Ehx@0N7W)LDNd?|5Y4^OX&YbWP$vR(*Kn_(+cqj$q)3HbeZ#cV-NK5eyVHg1_NE_ zfL`UQ;xAKQ#y+s2{5Y3q1SgOW&h@C`xb(ARg8Ig-nL}OZLl6IB&7&*M|NkoklGm-j zB@=tD5nMhA_8*7$3-ddWL-I9jOI5xOx%l|>x9~qb_FP{q*DD(QZikns&#UYJ^?6n= zN5@s*20QYU{{YsJTie*mcez!#XJ1<#S07pRx9nqHDOc0Q@OT#T-zz6TG1Wmh!B*^o zUVMF$naRwx)pKXtql^{kp_J(}Ef3u1VI$|yV9dPN@))_Z?z7~n)2UTE9o>J5J~?1}9=>=!@( zr6sIcZ96w_wVj(bTV;9J`Oa>@f7sPf502}tC@a&>7(-p(*=JiG_mCa1o_l9)jI%Y& zyQA^5g#!Uwc;9{K|548F|2Ev@X#9<`|PL2JV0Ii6_~ShaK+z|N5ba+6lxB z4*S|y?VH~?&iMmQ#BMl}Scd$DM{(aH(c#ClJ%V$OKH^BuIfnZkWv6hTQ}79#e8Nd~ zC|v(UxX-C4pJHPeC!KJjJLhQpfX>&)bMv;IKgIW>I_yi{^4yhMrRS>OK);LA`|RsA zAKNCc-J0VOyuaj$CtEhaf6;*q<-|z_sOO@qd8U4KbGmGQ@DCqQ z?y>ar4nCjyR;u3YQEW2iR_FSaEB~i3dYQX@7LOA9%aYsM_gCVww)#%#-(Hp8-J&bGULb2YWDW1QVDeklFya`j?c;ncG&y?Bz- zGaU2r?l3Kzk|CWhz3$|Mn`1tQ@N8CH>&p|_p_+%{xW@=5#~y1Jo_n5~|KjjRGXI4u z#rwX6jQ9rrwr?K$4fNvS!~%{mVN^EqaU4IM^S;43!q{P)FAi{o{OD}un?L@@!|gb( zk^f#gT|8d**S)>iB5_2Hj~+FW`yWca@v$zq@Fa4Hm3!=MbxixmdU?luZej7e*zw9k z)OpHF9z(2Ad4wnPzvBMV!^#iTam511aE|hd6lc``F3t!3&XBxhD}RY{7d;H~>^awA zo#&-zuA!V^{YLWdyBuPktMIR9uV?Ch2mTkv-EhB)WI>FB>0Dk@t$~YSa$NXVt@AI4 z2b6)q!0dZ1IA?*g1!};bM)kwwl`7^>o?j1r1RC(8ls)>0<)W|i;qU?QT8D0|hmW^r z($_4X9)giddoGCHBrkXyd8W!0=th4jzo-?=E1#`p6?JOJgL-TQ>HWy!#TLR35n(9j zTs}!1ce!oY1ImeO7au_Hu@1AlUl;%Pfj!m9D8{B-W5)x~)9Q^T&aPe+8R(964?irQ z(GCAI4v>+M^k7~x`8Z(pNyYhy>r3~mf0BCas8^2a=Lf<40I|5CBse90c`q*~(b@N| z54_Xu@?CO$KyeKA(f?a>z<3V5UwKyB;0kUH3$VQjTVRl!>LIo})SIoi*k0s0K97yN z6U_G`3tFk+Xu$TbUy5x7=dXkF%kHW#&a>LGBC9Pew3?EDRl8AWYsibpg7-@XWTEpj z@P%j3oMpel@1vR$=jT@pnz`ZWXU#KD_c;GP`|>qwKFKyXmAJR?{OP$Kwly!sAv8xd z$7C1iyi=K*ZjN$H$DJ(Tm^g!Sflp&!IPvDU=BH$ea#+YG=JTC1Mz#Co8oPbfA=Rei+ci)-s;MD_oc-OPi zGm(!@GE@6H*PZX4-KTyQ@0gx-%>T!|^jvkFa5JXX$O3)?zkhVmCy3j2)Uyag7% z-?K%;Ka`i?>}xoH^6Zs6tiDdG7cKz* ztGB1DUHvcXi8ii_KO7&?-^!6&JHfvCuFI#W9;=UmL9JsUpM!c|DqcB&KYk}`M{Ugy zB8TwL6MGX6aPdp@x%#*6Kpt-8drQa5uHTFdReu)k4`txjKn^?>U~L!V<*PMy_DW5y zJ&r{9R~c*-E6YV@ z$4$E1vatiQrp*NZQ!Nu&Anva^T;WIcXs!;GeHR~Mu4sPv^F%t{$F8r3G>26;E1BT& ze{X&{T&-X$7sU}gAAy$}>~i6luiARKE8LU$sCb0r)48Y8&*QW)ZeR8J zs-0IZi_Xz|OHOF$d-$Kz>1S~1^f8!p`c*Q9_=H|lUXXBq2AC9%mCGx>ER4EZ1FrF9 zfnO`28Uy9`xVpmFb-I`Iw!2PSCs$(k(mma4-f?{f$sXx+UFXYB>1OYlf_L^M3;eB~ zrT^UZUG+Tmn|QqmV!x69Tk*QiQ9hBLy`HgqA70mWdT%GAbq??2cv$Rr(QhvsVBrln zSkZlR@ej{+et_VDg)V*+S-gZb8pzkjH`}mcg;m4lo5+D103({qTgZtSWIT=y@MN^o z28)WUnRy%HykdM*P3XTc`nVCU+a*1Q?%PdY@@F~z4A{_Gl0zG@@5xh_tx$u^DF^?- zdGnkskQ`DSf%0Ev3nfT!@$hF$g7^oBgR9;ZeXM?0ipQz0K$sN=aDIC1fGoI5 zB0h-B`1ou76$i*lN}XAZkEo-Vbr*=|t5%6MnO$y=_SO4Jem=z}}JQHr63}CB$&kvxjbhmW5r#oCt zPi+1P73emqWfK~(U@3&4)nZm zuIo=%-5mSsBjA4r&2=A>;^8irU$_;1kpaT0aO(7N47;wzm}Ao0ejU>3ye9oE+~~M> zjjq%1_`2J7_mwR1aP7U1#|I=coJ^4H5w;6q?pX$xU=zi?miFD_8gwyxu>3Nv#!0C zxRCHqoCrTaDK>K%I$eJHDs)aKSnVjrj>T{Lc(lT{z)yFTf@AVFhM2ppswZS#Zz&`X z4mq$pQeiJQG}|s-Z_mrK5U&@?X2ow9!GGNhPCCJ98$PZl3t5{e7_t|^^8mI&`2K}f zHgBGl&FB3WE_U$+={(`S8SdMD0nd_l7Ot>;UX@t(Ty>A(vwV z7t#8Xt;mZF*fLoO3DJLs|McYaVS%)aZ6UtPR$~3CN8XNa&*d9S4sZ{x*`PRt>Y3D2 zN47^d>qmFTvsMJ(trr_*GyF#}y)Eeft;ogA$VusWar=JRP>I+L*cY!m~Sf^Y5KwIq3aN^~W1G&h^1M>$_geUvtKrJ02ewPBkxtcOCa- zg0QMQR)=F=*L7YkyUz3GrMSEN{JKta)Xg)_)0}nh!@iC?%(K<`o_+f1oYZU5|Egh; zZK}Rqs%dexg%0apt^8Qwg6o|O(R(mFe0unn{+B+NF84S&`d#~AT^N+^7v9xtMEG>L z2CMo!sxi|%0oRT-CNk! z=XcM7t)E}%H~^nP&rb5ue}=lw`E2?AvJpHP>^*163-|oFrzb1D>x6SBYmg=0*83e) ze{nwgVA5}Y>-2vS_%CF>6wjUKWI%Yy5-Y-vOeJ3=z+B8`j_07GYMI}UHMiM|y#w|F zvCJKzkoDxSb_2(EmBaU=)%HStqdiekY0uO(a!!Z6)YggYsHQh-f`wPD#Ae{0%<};G zY|$j@V$hF+35moa7CHQv!UGhSkUu~^?iTT0@UOYAUO3NEch?2yZe^ZJ_YUC)P~MQ% zi`7~+@&PK3MtMS7rz!~_M^-}ON5ppjTR+0&_@uLf*;!VNEZRXWk1s#e?@Vj2IQ*mg z)e~E(VX>osE2zuhcyowW0`jzt|be!&sO)*Tt(;j{MG5|81I*^a{Pj=)A!up&FR?I z;~Ad*)8`Whke#CbIC>AA>*XGLazvlo{Ei0`Q9tlvbR!963%OOMO%eU`CvblzB=yOSgMX>^T#TVDnX^LkysxxTM^ z4OyXHBkG4Z{#U=U0NBr;J)5}kJyrx4CZZ8mz*;y8{UEwH$-1tgrW8en6E353O zs%q;--&Ugg^5jp%2T;N`yhyP~IDkF_a!7GW5RpOS;?yY=7xMza zr2DtvuN&fCTJKSEU^DNjoLuGTsyBz?5VGxuupwNo0d2lg*scM6I`MxdK0LrE4A_quzQIl*1;@8!1 zjuJRU5RRlgz+z-*F7|)+G-87E+?xZ>yO7`eeA&|AM?<{OpA*8X!-Jc1!kIT;q-R|W zpRM$>=AYB|ZmYS4PKw<}`%cGezHvWc)Y4OG5@*fvuWtwx~D#)>@mmxW3TDnr=R#uS0kvtr*~fUD=WnI z&z(7oc>lc)|0VGM=z~jGyCKPXn8Tge^?e!m^oX54Mc%@$vNGFLTxQM0xFW>b2ay5K z^z_@`wrsJ#^$psaTX)*qTejMp14H(5cdtENTaRr}ZO^lPzP`!&Wb@)f=x1z4U+pR^ zq;8cy@W`BqvjMO{z<)V%NV#pA@4~(MT*(j6PFz4bQgtv|AI`1u4F6Xhfc$P+gIY0r z^(WtfTqp-yY23dcBQ5Yn{1<2Egen60)~s9|eE!d1rzls~@c{7$WWpw7!Vs@(tF`3S zk9#w*fX(CrdHG4`eq@ehfYy$dkG~)M_wwA7zu@X|;ZN%6T)*rQt3j422VXT9vH_Zx zgL~wH@+S0pD>9%B{MW(p+sQ+0V852!73EX5Fq+{2E#SO`o?OlF2=#;Rgoh|UP(Ad+ z#0CT8%PQ`liSEysI?HDN?)P>9u|Hq`YaY0H!W@;}_U&?~x0yR0&b_%PoC#B!bEj(_ zolL(@&9m4Xbe!~LU$;m%YktP=!M^6F$5&(bndkG;QXv%THl&wjVP)Y57% z)YjO`O-=UJrcL%HulvxTt?+C(_aOF%@`ZPjxA!>s-;Pb7^>5T` zuL|2bh0kA-m9e`xBja;5!oj?P$E)yNv~r$mcC{9g`k+gOIP8ndvp+-(z^z@&IfGzb zYc{#}Wq%0Yfbw<*g?~OvzwnRU-;+o$O)#&%UFyNq!nO)ug!=bHk#&lFC+E^LrixPbcjR1g;|heMTMM+M0L z%>n-zQ)gN__W#u1{09E7elea-6_58Z@6An5XK23r81i(GKOeoh}VlsUOBpdt}%|q`lf_LM?Z^w^5fS&IsF7;$h zt?dZ1emPwJshWEGTUVdG-U8>RhUwjo4*Lff|8St+KI-na52)vOom}td8yjple(o20 zdhGR0TkJ8o|C0?Z))T3+Es=;l#s8agwI(Dy8SYx6{NTk8S_r)_Ut|d*0{)ec(2U+! zZGvjrwN{GOg4DW6J>YH#oI7qV{U5-FhXbhA<_WN>-h@r)#?)0StSmR@*1L&&cEN#EQ>|QT`>}Jq<43yxz2isjy^qe(v2Pr8q{|2Trv7JN|GOL@ zZDaS<|Jv8*(d*u4`1EysHt!toHSZqYy|n+&#<52o;l0<)Tcx- zv358>`$l4iiNxL={;3%Rmz%NIx2ql({Hr&a_`h-jv>t3a`aeWGqAVx(NZ$r1&JPTa z78X;dS!^Ad=zn4dPowMQJ5Zky<>@H5cZlsE$KAk2&>Gcy*RUndp!Mj5dDSZnAOrg0 z0sX1;zQ=#kOphgD+|@wf$578L;XTZE3?XM-?i0Azch+~WLdPqIx&tn+wLE0ws|HiP zf7wCuhc+ys#%eL~8E{{@2zj~S0iF|n8Faq#{`1KF$(t$Lf3{`aJzGSLl9p#x~>KvQz7|#Szxf1 zdbq8%jkYyZVXqEsveyO%?9I)akp(;K{hhn)!zZ4wkG5{NcRD(Wao5|M^>y|(b=mK> zwX+{QK?nLZL}2LtAO0>GIXJ@|AmFJxeKfm8@5coNbs**09Qu471LWPJr?-K{C&8`i0kr018n#9RKVL8>`&1wQMR|osMaf&J zD#HJe4etm4^84*Xb|{BfzCNuv@7AP8jtn6S^nU&690T`miya`Y@9_X{t60Bmf7LIv zfPeY=g!>wCf8?rseKq7IR-(IQ^Eht9G37?-HThnu;0TJrsy52?QGoxb=1;kiweWo9 zYDcm2gZJ~SsL>Gi^X|FNa>(<~o_P=XzW2cW@%>MkZpl-o+JvjFCf|d+nlbp*V|u~O z3Ggo4STQx}e&v#=USIg1aOc6c=ab?4$3FB7{_*z_#3Ya1YdiSd-;V+Kd_XbQi>b+U z_{ZkY!2Zvfg%6N@*^p7}f90XHDRxENdLWOS+EAE0fEs(Iy4Ieqtj3QYwWrbhFTwf$ zwt0(v5cB=N4>$PW@h9xv&716R{qTc^I(rLQ@HR1m*P5E`u~5ji2Sc`vJc6C%!ffT< z;>eY`d?gnHFMDOaFJ_tIxoX9x0Ii&ph47e_27vv6aO|R!QuyD~q@R=c?v$8?k(?!J~DUrQ2QI zk^Bv5)E;nOw?+>7UF+TUaZLKY4;i4*k9_EZ1N7qSSN}c5@anLoYUT3-yRPO@TW~M# zE}LFGI8_Uw9v#|N={YczgQ@EpkdMN?tD6S<&GG@l2ddeY!vTu1lX9r{$wl|4!}-(6 z`Aa3|FJ;mcw&0&!pgXU)!Y!o>>tDTkA3JF zeE1&tKjCPP<4evX(}j6)0m*lt`^yh3{r}g?F0(x8|7pViOz?l71;KyCgT&p4`3(@W zuE8H&FP+Og-yDqCtMqJmrLo06=pV56`g-k6bp0#M&GsU?{>7SVdkK5swcb8^V`!7T zH8^B%U<16feY?H4>oNOa=i~M)@}na&$6B!8+mSC?r>IT-KX|e5ul^OK^X6L-@oUwr zD;J<%{%-llsV8c}pQoB2^|I{(k9}Zu2)-`-Z$ zt;Mhdne!C!j-7m;p0)8-0C!4y=mAS67chS5gSGu=G^ z==~0Sfc^b$`!==KuTeL>o9mvbtF>3)2e0(>5&NsR7u!1R*`^kIy0H}=5W$a@K_4q} z-QmfO^D^&+clrK`!N2@~PXD`D7d4%`9lgI5ABB4Is7JZe z|I)|cHeLE3{vXND|Dlinl0flkRmIr<=>Kk~|M3C98MfhDRKJc*V688S*rx{hh7S9} zzc>N77cb}mlX`#UK=y!r<@$9Y*E-<-vQrxPywd;T{ld3&z5IN_xa^z=GC=!^duXdV zs5;qRT)TyPJ%;?<&QKp9rzax2@~tv9g4F*l#*SNc+jwey!M=C^ z_)nNL+3vXfa##DOS{mi+$RDpcAuizg0X+UM-=O;S3IEe4ga3EH|NgPhJcEzk8g8ea zcvQ^a@5KbXIEd%p@#78WpGyCQGfuU?{_--*0RJiUCd-^L-N}L?;%`A>Pigo9im{v1 z;Go5cDb`*dv7v@mdv)_xdu!Wvdt+e0UaqgTSF5YA_3P}NmKJ-T@g6?FcaQ-e^mN&K z@c%dH(e^TO;;9hzE#wYt#fGRtPglUTtH4b&{Cz8Ull?E<7+OFM5V$WQe_b{}m^oj! zf*j)2^tZz|rg*sU-$ra*eT}=pU0*iqE(`bQe%0D;LH}z_GOelCPd-m3_z$E1!T*na z{1@jJomNG!h-z$rUWb3`2-B(0L%+&@FvLBy)^#8MD|crgEywxy)W2VA>GUEC+&~ue zaK3USy0J$($eWPwuL%q*e{V1T(dRzSrSp~VD1G0AeCVgvN^3Okf_L;1H&m|##R{Ut zK9nn$FC13M+5!OZ|cST#Qzm* zR4mlX-%>nLxz!h)d74T8XUv=dN1TG)IMuT0p&20mC%AYqYq+nsI_&J?N0!+}_;~vA zRaT3ye5j(zc2E!SVh1sQdNRHt|30}tZ)5Mji>`kMJO54ksl7@rfMNr0wKOx@T+PdC z=>EsJz6HGIUBa_2SwPRe`>Dl34(mGgSE^f# z&j6pFdNgdU4_jMxlwNeyXN5|wf!>-e(U7&*)Y!cfr{K?-X=zhc`^#_3|927RRs7GZ zrDZO}V$Uwu1rDINy<~v4s*RKWzh?^i|J4!u$3FB7K7M7moq5V<@b7#8ey)Q2yxzX_ z|3zn>Zh!sH|FkrGfASS&vdxwca2_!L^inOHyb+9b$j3#F>E@C$>meUtptKx6K-jk9 z$9oE2z?1dL^KZA8+dJ$n`Zm40>v4N~8@WFHz4!_0?FIaVuhx+-R9A1$Avd<+`x_+g zHdIJX9YgtrRm2&c?uY*>@2`lQgd%Ja)dExy?`l|!PFxdDew*@V@r8q*7GzT!I=UOo z_EV!bK#bn`0Kvc3%~em3ZM@b;JU@#%TJit#+`OxN{1@d1z<-Ie|GUZO+m7y+{#PyU zR<>FjTipM1>tm-Y_J>T7zDF)d-}gxWga0nHC(`cdi`Uz5*YttI8wdZF z{m)QMyYPSC-QfQv^#A^`&pd;VUm9WG8{^skzK=)0v1~kT_1la8N&kO;?5TF+Wxs?c z;v>RFOk#+q>&Zp!CAR(;IsH#n)!A$9o%T|5t36vw zJb?V5r_epmg7MvT4fY(dzGu+&&%+U38`^Ad3=U#1P@l%JKKu%G#P1s5$BpF2X{|El z!7DGO94wU1qn91AKaFxYa@Eq+oJ;QfI@U!e@2zGnJ#VSGX(Io=4L!`-!PW=&7ygG_ z44{}EnrW_ohw$Hz|1JytUr~@x{4Z-S{>uWzW2$5L7l%-PkH^?*{q3!6-CEmx{{eVF zA9`PYKIQ!lkSowDIRNgv#RG(WWMn7upp*04$xD!JqS(KB?z_B3VtHFqQ;1*SheXeZ zkgtmU3;U7*$_0>LAgJe#?^S+*Fu0Fm1Bg3Rb066N>aD6+KyP-cz4zP`_VN?E?3pb? z_TtWM_SUmcTW>>?#qspuKC=zskG6Rx}h`~Or|1L$Ml<(#7PRcGU3!^{zF zPo~d}@IM#)?|E*-{;>}|gOB;FXO212{f=i-<0E-4J3zm=mn-PmeToAq|7RS2p)_p8 z{V67>o(qZxl;2N0fI29b2jnnMo)30`><2f5ePn>v>{2e_=3v0yfBq@^@by>h zgI8X#$LN=xPkvxz$x_RMOJvU?7KjZX{7<^-D(ZiT|DUSff3cVlv48P>&;Iv#p@)Bu z|IeR_{(t&_{bL_`2A@20fSrBX5gzWb@q~TV0Bh^ZcyR&W{=bF#zf|UkYQ$2A8K+|d zWMKp3lLwFp|IQ>AC6{?znUQ4;x#Yru=Vw|w;QcMGwr>-$vt8r`JXsO8=W40%p)TMh z{QfUt|EnJ08F0UoYX(Y6ts46{f-kNVe5w97gsyDhzWsUh$l~78i{*|FBLmQXQDSD* z=(<|gv}oRtO7DP7tJ#=tHPqhMrQ}+DTCO!S+A|BNCoHyJa`O}eP#w@Fc)MzVwxPHC zcu(>F3gZ9a{DQss$Nzs!b!jQRvr4QxgMNJIaq)lk`B1IDa(er?hx7GgOAM&~mpY?f zwmtY2g!?W=H*!Ec$zfmIA8ymE*dP8A*(y$VvyG7RBfVd)?+Mq>eW}<_woOoH)f2p^Y`+8=7r}2@bcBAC;a@s`@#R7Cl1&@_MvC+$&&}zIj4L0 zR}Gf4^}xS;z1lwuP9XoEY{84M|8GMtq|hHGO)+Eo#HA4fkUt=YoPg5Bi>>gX2Q6Xl z9M;&HW*OMB`PAM;@rO4{ua?sT3%tM3-DfX!cG@#=&nKz#dmO#L3(l|uJAXT}VH@_s zc7|}?n_FPD_{MUX<8ko&bmn^=`K=}R#Izn|*&_OpGAfwg&G7q1a(8Q1Q8%|f3GR_; z)rsUICg)gf3N^c_dDfgspD*>{Az!Z#y+4#6TXR$RPp2OTI9BW~2Y+FdI=^uKr}lqA zR>4u#r6u%kr^XpRzXhMZu2Fy8E%1OL@qVuF<+yZxA3DFEm_;vnLOtxu?;zeU8PI{f z(Z=U(k0&NUU$j>2nkIOS`gYZ#zZ<}6l;@~@UNg!LfYUo}6-l4kbic2&^|A9m z&H44~p-y!Dm6ZQ?+pn%b|ATw5FRqw@4UjGyK=_wV0`oQW{i-LPU4^|KW=?DU-E{PH zKJ&T|Top6#tI&^KW%vW(2D|a~KiAe_o4|idW|lQcPok6ih|zZuBWncz9qHhbyuXf& z46DSqP>2pJAtqJ`|1X+DKVtM_74y3a?ANZQ7tRJazwn=w3;w}=3UveY0B#1i9n#16 z|AhZR^#5k^c(r!UHZZGox3%_e7N1l0f3%=*ul^6@6@2wm{ZHMl@Xye?gz70Y0C$j& zUp9WX_K^wF``!Akj85?1!FipG4t#$t^ZSyZCWqxkWa zTM>rmOD@zo{~#Dg4#?LpnGmAxCyb0x4~8)EK<7r;>fAOk?$&2jEfCK~xx>nhl#fBa zLfK29`;|YjfLZ|he-H;qh1<t z-!^jp*q5Hc_K~c$a)kRWmHQ{XFHYy{fAvvR4w(G^@&}E({6F#k5i17!>J_6rAo>5Y zs0Azs|24$ftHsr+0j_|9*5Y^TM*nqZ((4mGA0@_Bh^?9qZUe}LFgg8| zutB*0cND zxHtZ%`kyNFe{CtB3qOL^-B8~@_1M>S%JC za(~Ia)EWv}tJlR66(^NH04||5zf=#PxM|g5FSj6q+ziTp>S}?g1&HDQ-ar1){ocv} zl%oc{~wh9e-rwD|JY}q!Jf?r+V{Wb@qg9)yR}`B>3+P|`TpU5 z=YoIf|C=tm%-Q|o0GTspx}jY1JZ%4>`NW{5tC+iW_yC&Vwhj2TTJQn3qVMXlb(_F$ zD>1MRYIvK7YgHpJO3Bkt!%vlg?Vi7Son<2zg5Oy{0W0RL62dc^E zt;7Fc0%t9_M?Fa9;TuQ)KTOOEf4|H7!S1h44lwea?cad@Z(=Hbcz-!1+R{^bMQ zgpED`C)q0eOLw!Mjm)XQ{};^5yWEfe1+sI8S7Q6uf&X^6#Gq^j&L0BH>M7a}&ilc= z0>xt{tDPNlJQ}Za87iWjdSFv@zK=$Pa z5C;(coxXQ?U*ci~$^&rgq#zrs;Qcx$g3X|`S@0d!>y!R}SorTBxqs|K z&tT8MNbrBSljFXvFYNm@Uv3}$uil>*(GU7o;=-<0gtbw{{c~r|c7Edm{Qr6M1q-k( zMkkfQO*OV3ui6$jJ&ZiL@9s^$m3R^7%lXZKgb{p(WU0IB%? z((|mHy1gzi-$UH5H!naQaKP2<_aSe#fl;le*8%=>uwi8Xm*?jH#K(VOcK-0HlCpPe z$!l)MkJt}qkj*fNEj@%BP+u|S_V#9j`&{f7 zdgc4T_l5T=_`YIy72wukm;W1)g|4R;yj``t^5d!IPwOF6A`?PR|9c#u`~l*Eixn$m zeFSX!h++erBYEKck4(THsAnY^5QYPkBR9+Ddofh;sGJ%1x*WiH|Me%=@8x`CfYblz zc`r8P#f6;y$2TN-aMJN!{QnW*zX$y9AN$NR*u!si@mb*C$#zfgyB?rUzI(s5WPtGR z+5cBs61l(0%KwEc=79fv@<+r0ln0zo{+(n&5x6bIPgRDGT0Aw#yeyZ$ig{j%T~LR8 z-G~gRhtD@74`ee$snd@VtFL0e0eiffbAs^vEave#aK4UtnuLv!#I_L3EAFLuxP0NY z@Z82#*=*wOU$o^m|qS4zq%CgpH^VaS;hFisog>M_rUx6z`p!*vj2PO5wazf zetdBEPU`=1@W)q>!xSkf*qi@f7AP53RaW*!U9b!eL#-d5Q8t74zwH0P3}Soo^J7o+ z@*chUCA=5%2L0bk|IvnJ>cvyQaswEaymHu={~p~eUtWaoqd0yPSx~7VUmm#C z8r{Ni7`^S{0pjlH?{evI`hUnTs5pqr6@VjnTlEA&4m)|k^Iyz8xW0mWRY(R9`zzsj zic1yES>XErrl9{9{^dF^{s;d*`@5d5ck5@w|DF!)|NUd% zc?O*$?UJ))|9^^o=L3NAd$u0&!0(>ydO`m1dt+@p{r`mh6sQ03`;j-M8X$4UTxSE2 zN230K?28MOFejB8REqzvjD6J#Dt=XlOi)}aBEHHTkBS2$3+nMtJAO~EkD7J(^4Nya z{gMw6I7BhmCcry3P_I&s|1Zp_14j+Z@ih<^s70^Vpl9WGsG;vyZDIz#zg%lb4_H%X zk+tO%Tc_|32k6BP7?kY~|Iqq@is#9v*Uoi$ zd>@{mzG3R|?Q;L|?JLf&+yd$TUO0x{r%ha6aY^j%ZsbfKGNuL@p!|HvgBphXdE)Ah zx5v1?eEZ`4QLYnaBh>CI*TLZ(ex$8%TSh*BdN7uG~S=RdpO{5_g9BPupm_NA`2uNiWz0hPtRsx zjz5f@3l>7~;7YKsbpXoY$R+$AMmPF7F~UU6Qu6!3mdlyvoJz&d@pad%i$ew=2Z(jm z%lDT~U$2Y+bp(aj5T%X-^rH6%!RHXUJRbhhOUQ-r;mESZ8>ByU6r2VnKmmDEalebXc}0_6YX;s00sKNsr_WggZ z9)vmK_x_O1Pz`{u|JMlrjo^R(*ms^m(?P85dziz!a(sNhpC8+CaRB(gdcj`&y)$h5 zudhP?Ps9H|)#U+Zko%)vK-ts;YW)c3LxL0Jh!Ze36&J{yHPht?WpON<+<!k6h`-zL1Zh8#&O5&X@09qmea0 zTJT3yJ+y*)-X*Nb?QwhI(&P5x>mI*XUsYd^E7w1GAHEevA^fV4wYv+kiSog0fd9S0 znhcz$Ufz-sMabd8dGth~MyO2x)8|jQe%s+64QmqcpDb{`(;#w0zm?VjPKWyo|BG+9 z(eeND-TI%N|Icy6nBULU{|Wz$6OQxj|Fyz@E%^U3M;-64;eQ7I|Icpqn*aa*$9a2r z?Gwgp<)iJr=xEdb?!KSB_A~qLeE$E$KgL$aG~TZqZEu9xucJoNzt8^zTYa9No_iQF z-i!O63D&>obiWth^<;wbf2jW<|98S~e#08S)QD31mp+qv(V46lMIW$C>3V$eempLh zTvDweo5$SDVWbl)NQ1YhQ7@ppVdV_ynryDgW=-lWI7B|2L3|+xS)!Pt?kAa|{4re_Q^A3jVLcR*)S~k8Z9fp4SwM{kIAK@&SU+Ui|+3 z_~!={r^mif4X}KAsxio?rdIeb$<6&7{;B`%X{0W>o*Y5>47UjXyw8BP_yQI4>qe$@ z!~doGyO9BHyyo@d()xw?`oN&DFa59lyb#!zes{bc%m?A<8sg|$<4`>f6rU>rzr~CK z#cVf){w{8mfuL2`q()ft? zz4WXR_R9;%L97Oc{D0%+2V&a}x1xok?7`bc@;!#*>ltO^t|agElo9qe*lt}r${k;M z*GP+->YZP@cogy7(ZoMS*tBa$+Kz9)bM*@}AKbf15o#qhFkNxa0g< z<3sU&{njoQ06uW8D*5>1v#8Ao1z%q<@l>?SbSIF$`1SJ z{3uwD65CdM+~HsG2JPbuti-Qg3(u{A%hx4$A3=$mloGHgv+fC{P!qkz%^QH!}a(A|1Q@De}7ku14zzv z@E$GrB@JK-2= z5cv#gvqstQ;lqsI&r+t3WbLX0+T6==M`* zkHr2tzz!I}Ii;h~DWh!Ke~n~K^@Hqy;a{;Ij`iMS`nBFY^X84RgGLT>vStrl;mi|< zvzGWUxb8?BHf)&Pf71c>O2KFwKI|*@b7aAuFX6jU`e0uNXjz@K5ewl4$f5%z&>J>wd5caCMUc6zp~u?qdtrO%Hoo$ zrf`_JLW=X{X&t{F-eVw%eqXZxmG`4~KQcyfM!i=D_ix3vXvR)ZZocyPWV?jn{^jrh z#ov5=FZ|0#ry8F!hHxvMuO8ibv+=2bcdgB<=jrnG*%xm0jD>e`fI`Uthi~L??3ijJ z)$2QO?|g8tzQ0<#jrw8UE5Lio7E>%IUHyM1PPP@d-R}H<7h?Z=I^i=pVc=f#;wC3wUd=t&;s4xI4zN!+=i~B&?GuK%@nt=? zgYB{l4s-QAUd`|4;{b~D%KxW4u#3i?YSVuI2ixd$xA2c0Fny{eQ3H^yIsohe^#+kF z5T>0j2h-W&1)8I5vuD5qls814xoS<*l|!hx%y4-`oFiLSb5(vR#izrI$ZZAx@`*+8 zjVX>;&NhtCyquhukgzW~0H$OQNKRDaYpWq1sQ5uW_IG1)y0ub=+lp@PBnH@te(r+* z_oDliv)@kKK>0tta0U4s3+b)n`hOJ!zWQ1G!~csL!y&8UyskLb5(fW$@c(|~K@Zn= zDSn7d>Ezy$0qtyCi3v0zALNg$S&Lus(KW~^_2ML78vKR8UpYEonD@RjF%sm0_BE8_ zpdQ`w&1r28<>#vYO1fIlFdG>lo~1af^S{auBmDEf_<)WnKTCdt5_G+M{iW~=x9$fq zg*;?K0q>u~Z=>Gd>i@Ok_VI5057m6jAMC~cz5Gx=2gvb6@PA^={!g6vHQQ5i$d@)g zEIz~z1Ea$KAAfS7&HeLe{Fok|Q>Kiz$>9IM1BO}KvV-gw@bFRW-|)*(tiO8zvSPSx z-EgqY|BH8i^4+6&P6r`RMp)9_qwVqw4z#N;5{3`4gAN>K#~m?(URWdTh_4N|pI{rj zTX2ZY{^Ka#f4I&4%P4nT`u~;egY6gR9AH;mcmR5Hv>o=<;TD{K5PsN$?BQES+r50o zXHtlpeu>ZRa^W9&E}YBW_x5G)$;MMo;ECXWj9r8-m~rj3mVoY0g#RZ_M^`YC7%A}f zWJU_OPv(Ew2#Lf9)h8_ZGx}aR1Ag9slL;}srS&JYo~Y)!^pcBfG55>R{i=6&zAzWl z_WWZ(WI+&|yEp;9KNmxUPpeK=Ijvrd8y_KZpdK5b8T_{>_aD8lq56Vuc)9v-_QTwicNRF{+)rR z%}lM$$ND@elXJ2ei#a z9tiunoSVy5ayc!g|5x9Bhs*zy&qw)x;`)AkNHsrB2E>ki=Nn%9FB$w>A$sgf8Xtne z!ww#XeujsiaDcs+cZe!S-g>A=a??AltMWea-jU!_alQ{?+tDY%9k<40zvJukB7c*mfr$ zV&|g&4<0ql-sAos^LpQ-2V1S)JJY*I#;h+A0%!Qf%b**ijGTD-_*OQ(82mk2)B=&Voe!-1X?shhU^oVT9 zBx(f{I2J#7vL)&@`IwOpsv{8Z7mt;%Q2i%zHJF$A48;gsUJ4wz$i)jiou#;;`VEzV zbHxmU%5^2arr3B0ebx=uw+D({51q*3hsd!d*9gC!@ZSzMZx<&Q4*;jVU{n1*wKk}7 zf4kz7S#uctXJml?{C9&{pRWJo;XfGP5DHmEqVT`bhS2+NtxaM9J={n2{cYF)S|6wl z-rtf=%>lm0I%Gr@d5KZ-eUx+K_bpHje+aDm`8**$ZwOvcy=t7rUziXEKuDmASY!0t$tKUI&og0xW>&D+@KcL>@`^0>u7v%4k43I7u%N%hzV(e>e zKOg_86Av}ZLk4_FQ)ButN_TZq19qWd7Ieq3cH5A)|&{*5re1(iJ1!e#e8n z_AfqT-y?@u+N6VQ*V@=Q+(Y+q<6m6!#e41H{2$^wI0|m?UKTdd7q8>>_`43a`>#95 zru}ZT)hzhepXa|_`^9_x@BYtc`PGHsU;Z8ap6~#-b-%UZf!_M+7b^aT{=b*Jkodc& zINk4L09-#2{EG)9P806&>8UoD8uEk*ld&bi|HR3*e!@gsH(`>kziWbR;8;99=6K|T zWP|4mB%bK}2-vgo{o@132mkU*=gmi-5fd!J7g)%)7@k~0Ot2U^P>Ku)VhgzVIGk4f zeqL*Au~!?K-MZS+sTJ@9>0kB!Xv5zx8PEy-B?Ee7vm+a{-ky4ZNv8+FXh}w_SgV3=c@+^NP{k-YXSxdcfbxEN$(YLRyGD7{a zdJt0EKreF1sabbUwD~uVbNE#ZPqF+WI7K1yP<}oAuQ*Bo{VzKq8;+0({xc*4G_VcS z(^WQ^Y>Rk)rww-s|El>u!{vZD?1OvR{f-NuAN*s!{hvMo{AYueFLC@m^HBQ-!~YB) zWqr+b>|?ec#`Zt?zwr6~k##8Vi#_x&-s|rfU$e?N2iwpx=l|C6tG@5~|K#&? zejVh1ek*PY@?y{s`}hV1FYzDgMq0mT>1J zTX)BuwhB(UlCf$$`;7IB#Ji9WoGUIXUw~o(@+W4XqcWM3>MM{(zJTIb1>_e87|Ja! zLmCO3p*btRqt{!{68QrYPw|7uE0?r)BUt%8_hr{n!eHpsS{SU@K+!w%k~ zRqulgXhNRUBOj{Qk<(1hb4310>3d;cTk?9e9*61{%duU857OI*{Zd{lU4m~jajgwB z)zh=D%DU=U1HHD|IxEB0-_&4T^ib}ptD&D~$QFa)DcAlHzaa5D`2x`Uh2TFR{KK8{ z*>_wYZlJiKVg~7)mx&z6KpyB>XTT3KsOe0VTn7Jf$mUCs_v-n0_IFtW$l+f4Uc<}( zbM_%}z}NqocY^)&{bS#H24A(SFZ{auef^p){nqEg`FuX096Dfd5^58!k`|hgOVCenZ6xRp%oA zxke4^-PWW39saQaWRtgp{dUFc#o5u_&DiD2;cLfVSVGNBC?^;FpTG07*w=ndY5CM9 zdKZP!>urkr;aluS|0@>Qsq3Zxg@0=Onz(nhct7@p;uO;RVK6DKug|Nvf6(FnVJpLi z5#~$a0n*+1?mKox^@Va-fCxZA119aI72bch}d@%Qs^4r_ZvRF2Bt44XQ2^ zT<6NpN6rZQ1#Be~wBC>Sfved9|1SO_KP@9wasc_8Ed2BPZJ0F4estzo*Yia^K*jr2 zALiu(cs0P11F8w|@t<=S*iYF%_O)m5TktQu`?)^uw?@vp{9k@|`G1xFqxj#Vzutr$ z5B~@68}aWs|DR-ldVu`WPth#DYlTNxhdU5eo&6=05G4&*2^7)53sKq1IdFD@c?wa z@?kZ^1%&+w`m`1vpgh1vaM_H#@9gq4>iqb>E7{ZI>W?>=1b-)Ipb%Xh%FC7h@A>ce zFTJx__y^xDoZAcTW&d|Fy739L!3*STY=T!bfd3le{>n*qwKiZ;e8lBCE2kH(QmTP0 zD8f!DMDOPdv&8vx@Bs?{LyZmA%UTJYwN=)N?r*E27P`8^hFY2&{_8_wy9ZtUm!JR4 z$i+oc)dOC6;K~JiV4Uy|zQO)RbWH+tDUrF9ggi)s7bG%oHe%DuHrRkY`7n0my|>+N z({H;KztwFv!#hC5+1XNQX z8IUVkfIgF5;A8^aKz^X|*CQ9|@ehP->g{*hpMG+w)7$ItuO#r?6UCv>_36X_GTk7zU)Yyj z1=d~u0Ou)>D}~=szF5`$%XZr!{KMV;%gXZ0I9!NGx%%<`)$Hl=( z2>$$T`2dg~9qZ#*Hx&GX`DQpk1MgMOel_=tJb=F+y)MpQ{s3Gb-CxQmxqk`z{sGJ4 zGpD2Th1V2p_!QRQNSZy@LK{~){I^s@tQDE#4c6?y2iSoO=!YBB&^K}7&9~YgFS*$H zSXRTM)=8$yM#C;hLN=%J?9yc4bN>u%gLK6bHMHI@BZ*jye6X^;wDynizu~UC?8*x- zB%k(7w+68A@97BX2QL>$azMk^|3$Zh{k8kYKJ^UNeBJ&C{=Img@7syRdib67n@c80 z{|o=iZwLSIM)CfH82%HOv*H39;qGzh?>Knm^0~~lIM%Z(FSEgx1{-Q?ayl%OmyK^N z-HI~kv6C2Q%NNeG<@gaaU*qobe1@8b>Mxcm{*SLJjX9e}9+7MT=`NQ`toh84t((dG zR!&?Fd2Om;%biR73;tY)AE1offRJkJ@#|H?0qVhJgR{-a$3suI(wnFmo!$2{U6BrBK~X3$}eq+Ml6i~K-YDFefa=7br1aft_LZ&m+r6TJ)`iC@H}jO zaTEFVkpU%a3z1Lxa~CQJm)Li@xvhFaM9n3z-+v5%7LLHmv#atz$j@Uvk@FW~=s( zeeD_i>HA;b`+Lgoubh96rBHxG{}vQ%_;2_RhR@TwS>Y^t34{4gc>e$# zU$KS``kAzH|3>bqoa5>>^bmmGmpi*ZmeU9RW$y=&2^r+;Yu(NGNz-f{F_P7ak=#0- z^*HaemB@gy^~-Irxe5PYjq?F^;s5U@4xnvkT^06!wS@zPcKz@FXum%HT+hF9!wt3s zKg&Z-7Tj#h@v*EX&ay$=ir*#~IiP&3bY!;TEsFOmW|PeG)0$3+!asJHIKXdzaIsy2 z{+Ipl?=W2J9zporV`B&r6<%|_w;}0X1M>u z=>Pp=-+2a053?JiiwfKt(^bV_D3A*W$hNJW?N*kZPCvaYix$YYp6g^oI3@>j6XLwQL3E*V zf^@Gmu%C?`>|$2z3;WJ)W$tF6H7FI{S1v{z zGG6&w*Z=Hi@PE($=Xifi@4Fn)m<^y@QJ?=uZUXyD_K$t-8Qci|^_%+s9L0IX_rwKM z-{tVnJh+tpA93iBMDU-4{!e5iVNWJYhk?g9Y}Wena%}z}YvdG>r&j3XfY#Bf<$rPN z#!%3zz*Z&ul^oY=k_TaIg^Y#wyR~Oin2#yoKLgCmXO&6bs_X#qt9Uu-sOwRQN0jo{u={o8M+re=gy1WJ) zXVaIrswf~Iz=SX2zabbJA^X3Az8>|+6UmJs?EV&NhID-+GNcCHU&(tZuTS__Tv9bj z>Q9zBZ%&N&WLZ&qsuiXu!NU>O_WEr}c9QpxyQlPnj%Ms899_e*4(WXXKL`;n0k zzzO6BkW9^DsCLl#0eDS$n<+d~#r~9|dGlqLIsSj{7xpJcF^0wzaYJD>nN zpah(U(92P9TeXh5VDxk&yj}Pgk5}xk3XJBgUyqC`g1Z#n_(lBJ7MBjk|Nl-6ez$6R zjkdxUHp3@cxlY$Na_<_BRV*duw~*XE#r=>!s`ss0xz37~FL8d*+OiV&9O|%#oIk%h zg4`@cewOk)l-nC{uQx@4wyT@<(Vu$UK6r7refZkT_QA_9+2aF4)*P*L=PMt$rlipM zTHB%_Y=$aggi-fQ>)l%4L8~QB5ckM(8&95~;)ALM%B2=qHCT!Ticd)nq_MvdAEac! z_+MRVKOp~4wO_IY#Rb(DLY&{r5%K)QKK`q2JlyuoKjQz~Sirsf@qhBZ|4-ga&+dQm z`RUpGANVfg$p7{0eD%9J{~y1%_9gda|6M{Z$S==3m$?26r~i}D`zh#;G<@>uU{rQ` z9CK-FdprEJ-kKxg0cFhN0%z}U>uQ5DvldrLvDG0Ls!N#T3~_`yu+SQ?^)be_s@U>q{8Wwu#@E@hzH*f!e?iYS2X>O+wH!olWpGgyY1dd6Kod$KXBV^ z*uBUA`3JChbLDdfgF!G@0SBl=PuH$vU0^tT4ShK};qK}it~G>m;=d{s z9@bP5dbKe~4G{Oz8d{sce;YPL3$mm^_t2V}57C!oAvJyE^eEr22D`tK;|1dM#P7HV zdYyH+!KtuY$yy__h3d;Z-lSu7!D8#GiQ3~m9rk2jw>`V5-(K0Z&0c+cm+juV3IAU` z9H1OdRfZg3U4kI81Ky%%DmkFsLFEi6U%4S1w&c~T>=ybrD0eIWUgZOlTdlY$zgsqb zROJMxGUR_rAb0ETYkmj*&v5czI>B)Oc%ouO;)YKDGe^F8jA#GX+;F(roFo3X$CJ~} zw-tX{V1Kk=t!He%RTGj8sNe zoNZrv>E3hq@cEX8AGX=`arVN}Gxr|K*TU&6Bt;;*py4Rd&dcVypzGK}h zziH1ubo}Se_3!<^vb}}+AL)C?|EU8~ynpZS4gU38|M2WH?eYuHcX__@A4|8om@;$7 z`P;?Q$-~*)(gIiQLjP1+Gjm+JERvtY91mDQN)j=wB5R#A;sL3`KUhu%`|I%&&LFq~XS^&OW2|tMj3NQU4{`G%z zB($|D5}_Y4Yj%O@&G`D;@GmsuLzMqOIY;9DmB<{wcZt?3l1)&!a;eMVaW%xq#wu_x z{v#eFpQ-qMt$Q}u7ht*-TSKz6CBm9R#7+j#?aKWb1l#fnDi)#V+$4F6ZRM`j^F?+v zFgnQ>>ZqyVwXh|T*KqSSSA+kVUXL*Rk9o?SK)$N>Pd-8QbWx3_{JHmBd#&^NXboWL z0ImBi-S6Z;OgCu%_@j?@`G57-f&bay|KA+%&H9#2{*P1}b>fHimDBdvSI#vX_5-#* zWqZXQJLcLq?7-_jv0*oxjhyfi^5CM+z1N=Ehucr$yX`C4W?w1dnh2wgQ*z?>p7-X0 zlkK1v|8B$H+GBI-*4m!fXZ^>VZ`dIp{oTG|W@o*$)2`Un;P!{{KOH@cZXb3a`y&|# z@7V+H&fa@Zec$W0mD!Q@svTv!?O1!vjhbb}49`0n0wcP=OX*7+m5wCy5Wntz6Ua?g?U zkvz>()5q8okDjvkynpw(-wys2)AjxR{?`AUWd{iVSO4&07e7*LUO74G%Ee*+f0qFh>@cRc8&&Eero=crQn9w}eTD>j!!}P!OsPYT4Hmw?{U&2L@&FEGc;nJkbD(DVKb{*@0T{LlTZtX|S0onYT z6Q14g+5hqh9Dnpt4*v~*hQCgSll~jy@6*3wSDxQwUpa1%4Lbqcg8#2y@U9*8qu1@g zpMC7`KkRpAr;K~i;r}ZW%qGmvv^_IEeZRlYJl2j(e#_y1Sn;R$A5;H~eK_N2_qiI^ zUSq>v;yvCnJNm6R?X9^dx#Mro`Hmg<@kb8x7w;alY0U}$8vn?En#5~A_dOTaF58R$ zZ-2DYPX2I!(P`iLxW&E${!g$9JMWzwduGY$_6eVPM#mI8?aeiII`}{T`33fe!Mp6D z$M1IdKX2D9cIl?;Z0pif+_~@Hcf4KF`4hXe^&-2o{4D!~=lj@_Z&@q6;HCM;y8r*3 zXYOBf#|4M6o~t+=a$Y=--&8tZzqw?A*8aHkobTE1esl?bdF6`}cL)FK=chb8tsSpk zU1j*WD_EoF?cLAXJ1@RqL&$&*c!1_#Q$@KO&1|Kw7;tH|X z#Z7Z!Lpkf2;sV$N@>MUwFP#w=M=T-3`Ns;-{i==I)Y=Fp0#-t9k@Tp1XNouEB_~)S zYZA-1mCL;X#NdKpG>Y!7R(~KctiJnQ;J+UH2C|YB~~K*%MT#^U0Yg$4(x7=ddkCpb!d z5L^ITK=s0`TZ{}NMnTR+E3&F7Twxnmtgvyvy&4@q%k>peU9jsBz%zHff6)K3|C6x~ zlHg4DUw?zE`BL7O=7O&egn3^c$S2_I|E513VV~T69f7Vtevci0+ACIa z{Z01n-N)O9cOPX5<7U{fE5Y;c&91n$$xgogc^d})zdGZ6dtnB6h`r~odoQ-($=n0| ze_X+9c45(0r~ikw@3F3ju5jlq%YV?m@)Gv~|HJ;i$JUk3bH`tvbDE9V^NGX!FCK5R zl)R~X@qg_g{a)8x{kien{IPrASMOhtb^BiYFRoo;AIv!3jhE(~YE!!wApRV3Bl}4#|0{|IU>q4LJs7oBv=Nu0@?T!0{G&CkI;Vy?q3NeYx%rdcc+Uy z-dc2dDQj+)WoP5F$lnt!EFAgI@ZS(B4|HHxhlukxav!b#(~T@>fj`IxD49@;yKWoy-5XwoVO>jT}?`!5QcP0sgO< z|GvGQeU}{q{tt!!r$n9it?%{eKh#Y$F~i%^pQt%t=;XJYoFiy zTj%_&Z{FkiA3XLCcG&*=cFX_L_hk2X-Vfc^`f2@-vO6xkfVkYJf|xvbzt1UW{KOI4 zo*-OBVqA)%Cfc&r7TfXJr}otsU)aW#t88(5r+wViZ1WqNZAtrV+pu(*ZQr!s)^xR6 z#_$iU0IZ}Fv#WV~am-ay{;F7Li{`8#x{284(!qKTyuSdfsvfrZe*wHZ7ObYtl3y}* z1GPWCmHR8C1;y#<6V3Do)c!8~+|FNx_JKt8!nQJTQSIwP=Of=hS3X7eGry?WFii)!F zBZedUP!ny&WXns4hIZf^sHx=qxwe|?tep2TZ9eM@ZKbX`e}S#~c%k2CX-60G|A%(> z)mQmEAo(N`p||n;yBPF0&;1DvkgZj|RoMVPAl6egK#wE#Tlm*_5bguJaNr}-_7MN& z19!9S_wW9b+o$*Sw%_l%)$xDeAD7t|_xJK^)ZKAG5B)#j|NrCW?Y88hqpa6Mn{1aS z+4d5c9ZLT$cpgQI;Q#UAW3B7Clk9f|@bn6^L&_K1rw{hF{W@1h@XvYRU;2IQgnO)c z$Qh1z;s1vEBwO`NKl|+tYz?gU*}T?<&wAd5M+a>L{R;N?eAiDN-PihVU+MVozo{1a z;eMMCeXos)yU)&DG3^`vU;KpK)%;2~{+A+m^t{fOe821w$ASNwP56Hz<6PfAdSd@> z{8tU#|6AwuJpbn`KSKUr=l@_`vi>oLsNVMWPi`~ z4?dM&B98g@%EO~Er8+hRU`si6mN&N`2duNNw|`+@qg#CiJ@|6dC(wnhwtf9-TUc98 zZf<<=U@(Vsg@47!CyJiX0v(k({>Z0Oo_N|mz`h`PR`zYTnaQ^RzT>AbhaWq+HVs$hO9^d=7chGi@5QOZl-9k;~JN)n%v4e>cd-EBjwAHo91v>=aqCG4n&b@7$CB z|1IG^BWjA<|LQach*$!}5R`Kr#SrF1@SjT_AJr*PoRIK8W#}8Wef1*Sv30ZU*vz;1 zgtld4u)KK_{YLwmZC`HKU>`TP;v-FhrYVMcDs76#4oJTdjnf>UwR75S$EFST^|r0{ z^%tMn*IzQIzuNBK`|B^ir2mTT+wGHOD{Klh>YjmD`&{ZwY@y1luKdc%y*M2iOW$}R zx`FaIC>~hzv4nrceoG(Jc#sa@dVuHvO*Vl(y#o7R`PIAGw!3%#$!+spee6H}^E2CJ zcWiHm&9=|)4z9EA?gQq*Z;-&Vlp9k{(sS6ve9Q{DpSG%)f zmY=)trqz!Bo$Y^A@UWDTxA2_be%BLMHTJOowOe0nfbUm;{Q}4TN$V%uqGt}pR`_N& z{+B;}@IQR7O^@trCwAQ8_`kH~a{Ge!h#&qZ@PC4xQF63B5!27M-LuC(ocD9Sz0P@b zfcXD@(01wmft^SB-UFNOH}e0(S6=0{VU_3K^Yk$nH)b&SXTDuLV<{O7dz>5mu6q0= z>q^VE`9)>6tiH}xwYS*%kLTE`*=;tT`K-0@an;!=g;wN|4=)G*yz10>?kmv%t|hr5 z4t}kgHy`2epGvGwCAcfVzAYLc`=|G1dyopB$o7p0~?=lGvIgIePweGZU(fXX3U zh8|D=pH+fYrn%ANPURabzoO)Sr$`i)pPhRZT_}lOy&i{YY1Z}CAZ@Vmn?}PuB z6F#uBXIDGk{{^{!X!1M0Ec`!`H{SG27qu6EC;rLUZSTYl`Q{mUp3iv3TN~oK@xPa~ zciaC%{9oVsu6=S(Uwf?l89QW4Ape*C^^|X2N8j#q-MfkUtMLCb8!orc2KBZFQcrRI zFZ`cb&<{KI{{L{^PyO!U@sAJKEwKMN|Cg^Pko}M6-w!zSU{ei9#s54paG=MF&p;NR z365r>Qw#s`uSIIiApqbVy{rO^s2;9r z;i@KvYTl{7kLSRXjSdB&$$_c-*FlbL$tl^;oNQ!)Qgn0iXYuJ-_{d~)r|xe`i1(fQ{N$vE zr2l8~PSXD?k^2<`R6#6&<`Nfl^M821VxGmLteL0uQ6x(&B0cQEr2-)@cilSLlxf_{-xJQ#>q#nDaPkN_U$+9{pYCJ^~inp z)?@eEJ5N1oA3XP@4SW768}`D}HsXb6+5eQi!5pi%m>XQgGfO^E?lI*7kC$B(9UzW) zAjJbHF4*fcU?)%xu#<@Wc3ZIczvKYMglK{E0O$V^`+w!-z3kIl_xM@u?qg;;{(plz z`@z$v*r&Js_;m(77UTHe1^IsrH0*aHzjFNl&*@*-%DWEs=a~H{IzR>6!T&C^X$#@; z;6Hl!T^2v$PVgP*{JlTLZvd{|8SkngZkQOZ2|wk zyy`M*c=1p>x#kQzwfrRK|HA(znFrc}hx`1)xj*OoKY2`G|C2ly`u<$rLkDnOfO&t4 z|9SDon|xlb@<+;V93#6wvcK&1%F`|0FZ|~)-b;~(SLT=6y0U7Un}D4PjFe$tFN5EU z=NG?2UT*kpn6Ez*exzEL8OX~S;!S={^7}#uG>1m|{{(Dxb84$S_Ej}F6f>xJxn%hM z^s&sd$L6iLf*=+K{~7z_r*C2oUH&ZQ9zajh(0N5;a%7)_FFQ8j&BA9cy1=XvU;~)} z9bo*j0+scK1WEzhFh}i!Mr3dhy$}d!d{L(ymy4M_T zjZU<-xMXYfmSU~(Y1WdEZmo$K*1+<6)KICyrXfFs;w4IWSH%KqK0rlwniUgc?Ri0w zfnI&|Vf*WY_nTsw%EwQ%YUGkSa-6HaX(M$G8)gQzj+L9dmiz~zS-HHszH^djEO`@@ zJ1q`-tNekQW8t-&o_W?zC+7R)qmFPr&~<;u{WrE@<^Sr_EAanS{CN-CeB+)!t5sih zf&E)A{BPj@U-tXjUO4~nw(+LDfACuMgU+$vy#%(wf3Np82l_w!zhBe>+j3*@EL(2c z(@srm`WF8G1^-|BV<+0CTl?B!tB@HYxDWdw_z3hKqoeRUuVrJkEBpPYp0j0b zZ8opI-siuRD)t4rUNJ9e@P6gUmH)d08$k)aaM{u8i%M)kbE9QqcTdOvulcxHU^5R| zQcRq_{P)t|700jK!Lvp}2jKtA|9d;c`_6rSO4>P!SImh)zMz&z1=#M4nq-y3u>ZsV z6;mu-A_rPlKrDc0OffkHm1n4)oI?#0CR;1{1KPl7JNRsmPO|oxMC*VibZD8DGoF0K znd$KU7@LLfU;fTS`8Uu5G;dmSC-V{$i5ZBt!4E%RGhP{NRpc+IL&j*FJk46B&azf= z9k){JQ1y))p*ywYJ*XZ>4wO;MgU0qL{HyjrSQA9`L8R9S`_r)lDp%MU(EU@8{S*T( z{-7}u#Q(|;5a`VVYzoVSkEo26@VRw70df}pPy>H`Fr{e=$=J7ul z%}XMW%r!P^_&~d->TlinKY2}*ZR9G_)=9SR=00|3LO;j0l?Ui!vu$hE+S1N8;@pXEg%8VT z7R12`^J#%k96Lc%X&Lin=Gw>D8C3^66Z@g+2RZKtv$8)(H*Y|%knKJXyM6Q+<_DoS z;CFvyXWWPDc?qdUX%0|+BJs_fBmJQR{7a_^<^WOW2fCsfWd+JT%)6`Zm*xjLKFL4U zjD56~+JG(KzAYxnTA~xIiPk(b!P;n*)274oQ}IQBf9CKA^GWXCoZ|C^($NJJTbYT! zl~_X}l1#s2s>aatf6aDeUAYj=|@WLGQxhlKRg}$|E53fl&|f##Q*fVbc5}E-5UGPo44A(4f@g+U4NM0CwtI9+hr&; zqX+(9dF7pNJwyB5C))2Cku4(lKc~E+d;juV54K)gzLEVU1AH{{G2g%Vj>GJC_yE4k z{zt}syP~P6`#O5>)i>^E`+o6pB>qqOe}l*Wh5Enb|G}w``Z?>a?`!uJJ>u~{ivK%; z_`m)h|MyV0|No^~=h^cyXFL8+hW}6f>!1AG#)tN|b8`+b;s4U)J#7xQkN?VdfA;9z z?a=-9ah@-JFWkEi5SlOFzvupC9?0LArz#t9P@{u<^4R@i!Bql$KLwdO9c;;_ruw`& zG{wNE?!Elrl7XdP=L`GrS;g826XL&$nUT*xaWazsvqS?pNBOZ;ze8~}x>gY~dK9w8 zI(%e1wr;VtOBPs1RW&@m*nQyggKO@Aa6hNM(KanzW}hsYZ~4rn%jNuRXp`3u z!}rApiaewosfpMFWouPkP_H%Q_}9G4=j`%RPYL*czrZFG@PFln`Nki(*X|MgpTDQA zzjCktV(V=g7Y?+Ok88Al=?_l_|GzmKz5QYu{{LNWLLR@h z!2hQ|a@V(>VFS-}T4sGW{-2mW`dj;-P8#(s8DR3e_xk<~SN67jHm?3&{P$b8lH+>! zxK?jFeR27B;(!0IGwrU5*KNgh`*$C|X5ilT_Oxqlz_P&}|9kY}ApZB*Id|EsQ_i#2 z2M@E8n=W(wpIv#TExl>K?sG@KbD-n@ytF;+@u)p)-PJ$;yFYi-?smw&d;2_J$$7^h z|4R-K_O-6OpC_XSe)zz{o+EMw^Ve0^+im~gF$vi}g*;oS)ZSI?ZDCjQT{Y)TxEFty zte*$w6=PEbHcD9*?=2=4$9W_&g4Si168^o3SK}MV2fLCTJij*lD;3+~ro{_w$LHJZ z%Wd226XFZx_g+35|H3?M!7EnbXWVRCmM)y0|7XoYf zqJylLuwAy%YGmXFn=RSi<=kbrw(v;ouHABpWsCI7=9OKUk1fA&WWfJ5&tGz${&W9B4#wYmkmvnA@rZst4|LSSl~Um{#JRYtbjlpev==01EN->pnT6d%R1b@Xx!+E>gyE70`excyTqf zry8u*z?*BRO(2?3hYhb}5;aM*6>VgG;Q z9{+#7=8J#Y-42udAEEKb9(qXdzo7#Iko!(K@-W805uTUF>-38MM|}UWui>K?4Ui7t zGywZO7|VvwX3^)o3mp(26b%3m(y@!dK*3ws#o+5pvkI(ZI{86}$rDZsbu99Nu$n8` z8SKkXDE&bEY$;3)-rT?Mg#ZStQNW?MCJ^Hs#X*TyEg-?0&1sMtc` zKZ`yaJQiU06J00+r?Shd?w4eeWN3_HaEbjP_J8KL>i+Ib14=T|ODgayW{!^H++o%@ zb{u(t$t8>oPzqfU)^djv^UHI~4j}%ocky@w^d997tbm3&9gyq*4N@(l8n9Y}FR=<- zubACwi-?t+*VJlr8k$_EnM*&P_4&jGEo^N^21q8?&n@IR8jS4``#;wd|F1%)spguM zT)T{Io|6c@OE$mE*ZEiAU-25MH6;8i4p@Fb%>|NwXF9#|FFttX75?v)10diFlKZ6# z(8D7n2kg1qZm$2=oYC7>UC`&(_6z$4FC5;>4o3bD@$dFObO5c(9-tiXlK(aDZz^pj zvFOhKUm->p|DRg?8}Jcq0Leq~-80_Pz*z<@3x1RJK5{*_@m$7zE_f?~4#+O9d3Vwe zBnRZd>($FXAQ?Ry+97*@j&m9yp8*)HfREKNz7=az!FnZpyd1lF330dL>5bUb=ci|T z+`;CmM(adoD1=T({ulnmj|;KWl}PpngYw5i2+<5#7=(kIWc^+&m%oC)nC)LZ`f$nxrL7ZIAR85!>k~-qLRF( z704sXMP7ofuLylVi|;4fX9igJT;$0Any;zYjfB_W{rCao|5J>F>afcPApZZvz-!R| z4|f}o>x9BSd?Az*r4Q`6dw~DiQ+wOWbNc++eqrC>rNhAgK6|_0FSPy12dMa;<7o2z zoeuw3T^`NBn*si3zR294mzh6?{V2w10ORNNz!sP^6n#MTuE1I*{9JV}v*E3(ftN!s zyhs)fXn^=Wjdjt5P*=#6uE4p%uImcO0J0UzXJ3L2QKkll%AqUD-LDqr;A}+SU7k~9 z^WvE|k31gC#etp(|D2x>4JvV2J(9l{+~uRkCqPqj;_>}Q&uR^E{(U|3>EEfME?*`x ziJnRELmlv1#rc(if6oJmJfN80Z1}EpZpGZ?fIImzb42_24$2`~Cb_|70mVl}c|GE| zcZXX|!eraBXr66d{IP9Yw$MIbzSurrvBb8oBu0Mqa@(|M0rP>U*|T@vW{E?GV0%EX zU>-v`Hh@xc8J9qV3el}(hZOd+z`N>?DF?Y~C#Uc%nztc4Zrsbza`?Z;0)zb-$N-bj z|6jS~7PtS)2BfhenP0J?t_QgMe~|4F;s3QK?Pbf)?DK2;g?)pU`w9QD|I6-=-cM7# zFUh8B(4R&56vET@(E04WS+xHciPpq5+~EiWeva zs}<0J%EZ`f4whfRC;O`H185S#GUYxefz)Cj-t z9-g0g3&}Ll9@!U!{XFJ3XK`QEK$E>M16ki|4hsLQyWGz<+5R;TI2N0(;;aJsAO9co zrQW{lZjb*z7W}(B==?v36A5U*AG`5icLMl775x9&eo^0G@FBhI(0v2_-(@`Ly4!oe z|FQV~W&gS8gyTHNuJG^vzn7W!3*VSESonX1SRiD9S6*fw0CWHvAbUU(x^ z2B8I6U?rph1z<+__W82n_l#l1rRC7)K_|2x_zXBkbO7uZ3C|oOTH*dfuwN*hLRjY* zk15u1qAQI30_+NkF%}Jx-0e1acy77mVlXe?pYs08HW3H!4DcU4HN^S%_2knSDE^;~ z9U>oERLt|tmJBcck6obz`cg!^fBpz;ZqmE4b;y??-Y@(YVAqsSQ!+`OzYo_Sj!^PS z75J4MU>Y`(v3%c=)H@rAtTkEcfQm-(&w7kE+Kl86d=e+{-UQ1BfS;J-}@O%rSu$$Ul$<4M_hRvH*I4>|!}!Ts}9q z3xc0)Vp0MR1xe)$BT1&R|?9HeZV#ga{+0hRD_#S7MA8)!hE zZTF0iR1MCV;6GY& z0OKcG@c@zmL=$ML!55EC5GUGzOdwjI7(vAeWWt9t-$5Q0-Xkr8bv13zfJUfkS#%)x zJ>rJItz-k?Tv%2dfzt%7dy{Pddx3Z$=w19o`9qM@56!Njj%ReWbyO{MhFeSWY zgWvcG*a4V7loc2MP>A#I>kCrT_E&DrqLegf*56&0Xa)bu1*96{mFVD=_ypVzz`gS% zgU|xs#yhSHc$^e8K{A1SgpvmoGpYDU`6KmQ;^|e$CsoKMHP8WX&;->QtmgWvIVSt3 zbO^ntXrRwILeI#d)}ZE9DYiOQ_=g@SPAiF)01c4NA47|K8JP|FKVCV3`L~My(?1se zCq4bF*L_kQV8w;F9{`*Gv5b>|1|012zxaQ{QQ*Hn`2V&2qQ1fF2kt@qPhbDP{rRur z|6%NpV*kz<(BE!8=Ny|r?mgAoQ5_z|hR=X!2=~$00%uT@Q?_R`P?(q$Kb4oDtIm%o7ZbZy5@?s9@;1&=3?%})M9^ah&85ojG5LUOam z6!^O61emAEzF&a+TYzm)x`F27YEE7e`ncj_D#4Oubj9t?p^jGx*ikN^q{)+!|B3yH zi@E;$a35|b_V0I<*;(6F3n-JkUa~nffzM`I?f3~E-%}Mujvjb(iSz(4EqWw;${!fc zcl4NPu$eRDua+yF0NFeryykOE0mr&+1HP`@q5(ZXzMytVIrOU(8-whJZU@nOBSXlp zkc~_sTSF$_Qt_0E%S_dGkMRGb*Rl8G2Z(_-JMD&E$7o(O`1ks0U|w`U_*Wc&e7~9l zAX~6xLfMHVAF9a@xW^y<;PQXNpZ2l^NA~%({ldP%8~elm_wLgz--*WSUrQbc@PD-3 zalr)^6~Vu1Mod@kPnQ!VD~ey>69Dfso>Q;6BLNQy;z{rqvM-L9fdt|OQm_l8M&bp9 zbuBA)AZ)4971*YIq6>;C%#u71HpaEv)1V*O>{CpSe1@98CtX7Gcob*k4I5Y?*pa`l zjCnq?vx_&ouN?k5XI8BH;WD8QsWWB<_W#(JGeVqyUr+q+F67=@uK5QUA0p?YgQym8 z2l7PCN5tSEo2$OK>VX!a&%2%io@IxS{O|HP?~x5Yl_M|{8juA(v%!B3*bL|ZK2et& z0=b}w^A#&C8%PoFs(D3{DZ=;$#W?5C^i70)pF_d5g?-Uo&C`&tFHQ}+f!a*|$xPY) z)WrXh|Hc0`zdOwFc)dI+>@RzO?0=^ZINI*J5T_tw`IZ(cFJbKlK^t zM5Jk(?zZ+mwjmI&;cut02SC3%Ern6J}$t&@QVMz*;FEQFjF*>@1gH1xnDYe5aWlR z4_m+MWc-I%t7 zPV@Ld)|1{~E+KRwRby8&fM^5TWw)06syTv+3zqI2%DB!;W$yzA`O*u#$xk4g06GD( zeE~AG>hhKl7gIKFlFI@0V6=52wLqZ}De&l=_ynWw-`C8!`H!AB|Gs7LpIk$J&5UvA z4e(s$5MLCRM9dv=Kv5H|6nj7kG@%eVT^Q9diiwu);&URvuby4;&F%*fo<##>51<#V z5Y5m#sRg-2gZGmyL~|#DcMb86J%w%3FETh!vVW%fx8L*cnIJzuwwie9{NUbu#r(X) zI(?jKEfV{$`JKw|pqxO;@p$byXW5y;KfJ)@ejg{n+>imp0Cwa5fW2+bf&2X0eqrBW zXdm$J{6F$fq3@Faecms2pHurEVNYLuwM}~PF`GsW4#faT22d_QwTa+g^8#Ehc=BnN z0bEBA9f0p^u8_U4vp#L?6PVm^IV zTDF+jq#Vt=}(GuMXemtg}aA+ETX`xM{@Pz+KwIQKXp zp2=+;yo2I$MAOoFXU%(dT>!hC^M1||O_OXPzlY{fx&4s$)%*(2F~B~>7b@O5n`QY{ z6n8BfYdSVSuRnzEPxJhh|0MNgHS92o17bae^(5#)(u??P@c$|PX9g`AdZ0W{Gmr~z zBk!YZz^ec5c46fH&=)9rApG}&|JUuaw{;Rn`fK}TeS@LB_jLa6JfDB89H7UO^Hcx( zB85*f`~jIUzK>j@S)g!<>(gQcis;jkc~@o3|)U_pXMA&HxTa= z4zjQTWZ`#H?rqTk`Qlyv7f%M0rQ{uLpUK=iZ0N#&0y(w|latT^V!A?n|5(o)z-MX; z^DP}4qIkOI5_Qdr^;rHgXhNy*4=u=3>@&}(xzWO92C|CYPkMPO?<=_`Y*}D7OR_)5 zXr8p5O}@6rFI<77eCgmKMf5RKD*`A+34QJk;rCGz*Bf%`P<=oU)y(Y zYp1n+N7MH2+qd)aKX%;r?eDpL=Vh(`SnKHVj6IJHpD)~|_3v8__l4(#>*4V~)Q9Kv zJf`RQ;q$2vm&0SfZ=3dq`+8oh=Xr1LDgKZBPca=d{Zr9*;a_q8vi}aflfbbCL_5p5u?jD#B#sga!v_W_RSHhE8Iy_N%eKXa*#ATWG!Hskat!uN_mOg5Ft=Lv zkl>v~13aEeJYV~v6_Nw=O;W*l8gfMbh;f!ZjJhMl0Hnb8Q;7vq?4RO)MF->uh(i{L z*S}HxFPK+umuUWz>MyBg%Ck2GF~BbW_fw1r_J4Z!55xa=-*s2p1O8vR=RVfhd*4V4 zm(_=@=e}@T*t5^i57&G2dxqzQeUE*6v8>j8kMCITxi4Jyb9B9(&eeK&y|C}OEnM!o zFWjd7huY5P{Lrz!ugCL+=Z5|Fo%?|K<0Q3^4qz zdu$4Ixu=o;L-rx*@-ehn{Cn;@R{U6`Nv~AR>qO*&1jdi}Z!Ph5l1GLAWNZN9JIVM1 z16crC01h;FNV>oK2k6BkB`3Qc%a{)M9b;WKbNHC%vMEkLHbTcfm`MQ>Y2uNJxqEj6 z@&P(F+Y1yEi@sk1UoNLrg3WmeDeebMhi@iOi%a!0Gh<_>h4}ulo;g5g){(QfI5nMm zY{V1c`)`A$G|q^&3he)-(3v9Sja>0-0n%ZzwkrynT(u7TsFD>ko6@K zDE?3OK+)YeWHO(>#lK6${vRzL0RK#WT;)oi#XlT!&%Kd30R6yxfcrrH4}5^TMetwH zYi}$48LjMRo}=@^ru*)Eou2DGkNY{#_}_U>_#R;&zH8a;-~YXP9{Xd@{`zjcZT~)d z`CQ*1<|okqPk;`bPVC=F%>P$C?WySgGb8wS84=k~nD^LGFz@xGBE9m;2>WTTyn#(T z)wb6*S{)cnmOK!V0hE(V{wU}D@Iu)FMFWz+MH>1*8q3PnDLa68p7@*QvSiAx1zs|s z5yFI;VhDtZWMM?d2xr*7+%Lek0{rA9@Zxgj!&iX)I%M{)ge35f-Jf|$$y3Q4PCj93 ze!Tl*xDWRe1AIswbxSL) zz^dj)B=BDT&b*`ev2>#h;`Lkx5UwT9xA2TTr<3G*#FS=!+mj_|J<&7+JV^r1Akvc zuItAfAn>o)-_wpd!lE7~_80t5NB0;14|4#y900En_7zJSHs}A0pA_WEX6)lj6Uki* z-%BFbY%+W(nb^}L$3HRvz3Tza{{vru=A~+!iYLk!n5r=it~JixPR2M^u3_m3zCUtI z3V28sujN?B2FudBRqMBenEEpLFla*#4X8XD6pyIWfAW(3&FT9^GFy%yclEA-V+qQ^2ld9pPVb z_|nDY<4_KC&xI^G9vMdSs8d-^2K&iiK8fpy|A)3feUEJ7Bs1UR+dv1DXTj$iaBL>> zzjFLb7m)2m`93`7C-=}??Rf5|SOED2+h`}8|FMUC`0u^!L-pQR_H}KGENea77A}YDI)*mldzxP(a?IZ& z_fX$+`CIqyeva^Y!d~~-!|(G$_uKh7hI8-lYQxz6eeHwDKK0+#-e-H*-s`cv^Rc@A z&fByep8H+BuBZ0yf9!7WcYxX|4w_=j$9||5W#T68IPY zR}C%q56d=$eCT5&)B%v|6;rBuAj$`xjD1=50p;(PojC;?fouU9hl%(B5}^Su12Fcb z2WZZK^Z;SPV^k#*(2{8`_reo3r&xBsO!!*{V_5hPeTRxK)^YAPfL@3W$oDOIT6VP3 z3Czy}|5fmQ<=&p1$oybrfplztdC&sY3&@O(zwXCyAMP(mPyLs=qWm>owY650l4-Lu z^QyUd$)=>Pv+bO}L;t+DN1uFj0>GlxE zs_C3?`9YS0_6K@3t@rG|Z#i<_zp~guNFT2E+}FLW$1ys75a-`X-`%v{qus&s9qbR!Uu*bWXLxkIucYgM8{lD{>{$JgD ze_{ZX>#N)L2k%$ir?ZdiZ?~R*9(jMr`}-92bMgC0S4?vJzv?+8)37P0g1gY4q_LxA zHPHpnGYRhH3-G*Cj4@$fwdXXxJO>R}2y@ZIbH423T8|T5fu4xRr-Eglv%*-NMf|a1 z3_RYD*y9ZNY##c(?&&4 zjGNTEd#+FXrTTt82QY$v+5e;;sJ==n{~}S1e-qH#Krit8AbitvPdvfSI_?&k!S z*Yjg_zhGVc!MtD3b_mBDvhTj0GwWdW!1_t;^d z>g?$_@s}d(Sk2=nS}T~y$44ibLHIsO3q%)r<7aP0y zpTzr-CG@XUUqv!ps&q5Hz38Xp zF&(e`fc`(}FFfruJD+&~!oSZ0VvNW?9Qp*6C*~Mp0Ym>lU?Wr>nIMkD{RQ}yJ+C~+ zDt;i8AqF5*9K*H&$ecPaY;I@JbtBtCyMykl`{~-+CVBOQi0>*q#;?KkJZ}-l%Wl!l z+c-AVakXE5uM^bMbiR&Lb3eH5&9feRSU*2U_wo1E{dG*_{<59vJ^XvHtX^@_>h&x- z)}M`IbX{FT-%>nx0N3b-sr_{}7jj{6+p|0&1|k^vI2Jt)Rl^{Yi2B(u7l4IfnQ0MGMJ++W2+c>lDSRyCcv zIPodA0<12`E4Jp8G;(W?w}h!vzs`LC&$HP{X;w33 zy6YRR3t$USEKt!{YVYtKY4|NY=ay_IJhSFZ%GV~_yYL_SSd~jD9yvjFLC@g@{uKkH zc%W>qAwQwy9QXVEjko|c#s5QJG@m$|m?_0T$fqj5e+u+KvcKnO()Sks=RYVeFp=jF z4G`^B%(eSopdqrUUVrhWcJ{GH+bKsM**%8BvE}24vE_6@SYb?QpT^mVjxUX=U|hM} z5E+MB4)77=7SeeEZj^Ug7-YM)(L_J|K7Q?BTYz017vNm9NB7n7IyTf{od&S(bmC7D zocr?ycslVE;F(SUx4P!BqCXs~`v_mbyM%ZWrh{{X z?ZI^e{0jR49r5Rlyk|&%y06K#MVExn;C{h()O~~V^o}h1yK=qAHQh$bxqh7pp8eZt zo4$kIm1TVoy<>PSe@=Z%zL9c%pLg;JPKQQ6^sw{X7{&ad!z*@NO?*8a{;%3R;xpnI zk{4yylYL9}EXkgV_c^!raXa+WN2q)Jke$=|IQDq_^YESw&R4uk39_+{%|>2TK7G&g z!MIjTT_yA*;U#>DvIkIeGy!=b8QEC#i<20$nvX2(M-#Uq-X&XKA$+!kxjrRePO*5x zpk(qo;`3ymUqfAD;lD01dG(ahW1gHmX529|#!uPhyRiMwY|BZ_{GVOf`CrXNu9%ye z=`^4gdQ(FzKsEQRnodlBWC5O~2stK`ca%LN6)cMWDK1*_zWd#v0kW$GK33kB?d}`I z2jFpz@OkMIUS9w@AR3?;0nMcr_T?*8{IJg@fL3U(f#`wwzU2QP_K5rPj}&hp8twK! zXu4tnz2+kSGY#6IxPw%#_3WK@+fmg1x!|M|@B<#sxCyZt;yQv)VOuz4>^pz%fyn^N z!MF=~mW~U?VK63jj33ANbb8|BRs22JuQ3|nTkGHOeaE|x|A3bdh+yPIVO3+DF)w_C zwBj2)!)rMv#BYE-j?=Y-6MAiPx#uwapWHLVnYO#!K!2QLD!`?lCD4=f9RfPyw18&{ zu@~GYc&C8Y1lSf9!}S2G!FLYmXYd~3xsn6)9sz9#aIELic_F5|ug7=R_TbwE-`Jmx z?Yf_?AAFynKMTG8Z1lae`yb`H=nUEYkpC0m|BCmOomhHfI=ZXmY@d_IxD(%(O;5Z( z4SeUqKPECC_~dy{+2uPvutT;!X8SkYNS$rPx}iJ3XC&JyAE4&rYuV%67_XX_Bl$g# zF`Wz#%wqh;f&UcpiHH_tXs#f5mz_-a6TQ*A;XHKeB64O(-xubpsI^ryGX`5HZIJt5^ojgH}ES&2!c~$3X6n z*!^@K-ztly`~l+oS`Oz%q)JEPIzG3OZ!g*-*-bP!4f#I>S%toW-S=4c9Z z;>d#z!WaA}JL~uX@QA})*4G#c$E%Mu@GtzjEh(~3+a1%4F=5)rb@#Xp_@wjDkdJa+ zFvbJyh2!0?&$UD^g6n`kEr;9awH#pgFhA}D{Fc6r?ZQSt3v`Zf!ak=9$l-o&a6cU{ z9;;*h+QO96fdFd(UIL5-*N!|7$A|b1u@&B1?;yPTyYPNGE_^S&Pe=px9Qsx+%W%(t z|8sr6fB1f!8=+0z&lfpQ&*t>Z?-yWM*9+*A&I{f__YUtLF00pb$>)F3$;aED4%pwG zyyiNu389!@<%de7$^MtdSjhz6l1nv5M{{t5U-u=0L-7{jU-Q?+lh2uaw_X0}n|8p$ z+wJ73H~m|1%WNswUkIu=l8)HLkqSYdMRAPUqn1Wp}ig-ip0gBlX zz0*7ZeG8|3@b?1I0ltggE!5`~>*zE?_$Ovbv_pAOJU5Eu0Biutksw__{th*LAN86) zD80esCAgR9fX5`i!T(c3{!gR1y_S6*Z^iK$@c%UQFvTA|bi*z1{v(M6Ji;zue4N(* zNaXu~zxvpTj4A1v8kepU(}#F*{v+Hu&NWWI!JE?yjcNAjc;{=v1@yu(F1n#>vMHDgnuo6LnE9%Kr@1E?Bf|^2mc0t!eU4_ zg6qgP>dzR#q|OcQ9h|TC3@{kbiQwA7JBD;e?{UK6!F_^n7u+X!PQ4H7ypNv8WhSl} z(k>knvv7yc8G z|C643$x^Z9r^EkUchpz`|ME2}_pRtb5*Sr{zi^rfb}O*`O{G5F{iB|-dqzHK(ZkT| z(dXTNjohzX|J9!33L8GyuOc>Id90NSPqaZ;Pz-$$G(vPjdA>Zy9{1C|)8YA&d%d>6 zN96Xv=3XUR-*oC~&5E%mFyEG#W?jh{wmiSsJ}IvP`vutl3#=$McGf?G?;koQYSftD z<|M>lRh^mf#kz_bTUt_OOLGhD^ZI66Usz&G(lVd{DeiZW{vjJg75a$F0nmUf@UHoP z;`wRl1R1=K`|zP}l4WxA{@^{2@1a~G%3-Eu$qKn2a14E3BtKF%aX^|2JB7Cv#JokM)c=>v}Pd19xKEBd6Pf|zk(PyE-%F8FmG=W=;4mNlM*b;q^Ftd==0 zkac@(_c02_>2;3dz{h_-k4yLlBO2EMc70q6=Mk(6b9$B#CjmYV^LuMsU;_w_7gqG_ z0ewM^=*CyD-S>$$`7r_ZoUaO-q5=L+0Ugu1{@jjj_OUJ+r{h9-z~7&9j@Pqu4bey4 zBjn}ZTtm+(I>9~tUctTf9NpV>UF-$Arf{w8zUW4B&-zKz1>FI&9ZsgV1H{S?LYLK8B4ZW;C?#*yO9mAj@CUM4={b9=ykn(Tsa zV1q}$7j5wT{N&xJA`V9RF65I_yqd#S)8x4>@o3-OU1|p^X7tB zxlC-`dFazc%;Bv-zjmIF-dzjdSAE^igk)Qczi(q{rELR~s|w3(d2xk(oRw<@3Gpp; zg~k8WeBbY*0ptL?slB4oYIAdKJ~oA|o%3utG+=FUxoxRwuq~CfwuyVMMmLz7nq?jM z93+P{pc7P0oE*?MFe_PIGJx!P1tW-8#C{=rp;|HX6pKHKV(y~m8L%DtSU5^`#+2Jk zInI>tNZ(F5(U1@H9l^ZY29XJ515oZE`4JTZC_kiZgNlvvcuHh7)mcwPc9T9QI^pN? zkCbO6MY;pueFpY{`>(p%hLhv@;&ac(m(xu@c$#9h zWvB7nZNzPyiA;aSfMYz5i)wjjUZ2i8{J{P3lkVmI*i(-`!EXD@)pp&57lQpK0v|m# ze%a_XZzlztR2s5>I?L(cJCyx3W~HYq50!8(-?rp``R+8n#Ajq5@LE2^u05lguL72U@~4M05PUdOtOPHtxZ< zQ~b4Pr+hG{@oi3rZ)<*zWH0e=G*~Lk_m{&pOkt zx$;W8Y2aV%#=rc5u#UfW8!Q+GRqun7M;;?(6a=OCs3?n&GZ=bnrP-PPNR`uQ1SrZs~9?2jO|1D zP7C!tp{t&Al<(v@fx&+1@DJU-BHc-OMOFK_nAig4CQzQSJY+T1T6cQQzbv3B@0ff$ zqT}hv2$|S>-nH)>$?H-l|xd{r6IQP+vQYv3gX5RtPV`yKGTsXnaF6&PF#C ztq|Q%Jkq%*9PittvueNc)CvR2S9cn{$1Ku|hMWp!L`P0O>Tt)=NfCN-B3MywW8qo% z6p#O=Mb^Qd%LnW`P7TaUE)cy4w+H7)RtSz2jneW-qFLNS?G%=UVaeQjrqjWy=$xKa ze)cnXZt-Zni)3=?Qm6Mnf^(rwJezD2iqkm)o-MiSK<4c3+dG)M(;wO3>{Cv*%PzXm zuKe?#?edE+u`4dQn0AR>ch!}4>-E>!E!SOZ=R==QJZ6BMb@C~82|0bQ`SWFV_g}8D zH|~4TCd1zq$0M7t$Bbj&SI&rZ=s-F;e>VSJwIr0wUbQ!*i!0x*ur1zF{4O&HuN%lyT*sbijPZ0@x6(1+FKHP8k0gwks28Y2@x0~QcF+=bjSmpR2t zm`AV@`Cxs1k*&$evn9O8oTPNuY3k5zYM}*H$O9GFJIas)f|_3=`JR7w+llNY$_cJs zb;;}TIW%*88}d<0Op=u|H?e|$C7DhCs|eXlHB*X^)e5Q4T!hYF#CkEk|8siPQz>{8 zSx)jC%btT!`#Daz$_kmYRLC5MOs*FT-I#z)aX4|~L+*XRp1{)2V6L;Kc zuRQvwz5evmHiTTUgXy1t^f7xJdT}o_;`Zxru-h23H(zs&{pGUDoL(G*?Xlm12f)ij zGl(yw9RvmrCe}}U?(hTlvm*}LA9+K$2?KsAnIo)U>HHJiNhUrk!Y>28cYy1@;-kXg z*~g*3NRNT8h}N8mJR-T|9LX&7!l$-{@iEeA9Lr8;(5H`rM{{478(d#FEXb`(&P4f8 z4?lo-Jmezf+}&@lJ#F7T_ptr;+>_l~OnybqJLUM}?flbDw@c1G%dWT>EYmK% z@B;rHH-qJyZ@j_oyy<4U>y}&WdN6(wH1&cr&a_LTQONJH(D-EjUmkI##r&%haO-t*{GapR`H!AgR{t8CPoe%deF3s!A-bUQ&kEx; z)E!cj8;-%8(3cs-BXNF^DO2aPDM9B zK9c@ZhHao!GL(2cdV%LT#eUF?FQgT}Ll^X;GbO`1_@-?Msa8WQLN$4dMF-?>kxZrd z3#ad3STz!qYglrEH~zinAbyt~IWNpjrdmq!*Q(bwiWpb&2NiQ&UniGIk?JJqKE!IP zE@=U^lhdhlHht*3HtP9T?Y*a7wl^Pt!Ct)oaeLy<2kriw?y!5WV|-tGqdjoT?e;iy zd>ruFiRF0xC`JJ@)4cv(NTz zTEFm|vwi!;>MGv|W^cOcFE;4Lo9v;x@3!Zkc-)@hJ)XGxUc2v>oBjE(Bk%4tmtJPK^33<$ za+|&I;KTMhcDdo`V8f8jM=`%93b|}L_)H+*Ri@^-kEOQ!creL-uO!E`YU6mFJ!paGf3bT0x5k4o9p=y=B7M9zp(khp+H#D`|Cvz6qXG>Pv=d0FJA9suG*s$4l ztlel|EnH&jRYSXknBV+dQi zH!QPHTIbs4`c}6mtgooE^~f(9Dr&uLq~AF5(d8q-NXf^jz!2N7;ctq=ndsCg`qeh=^8{GA=~NHBP*yBL7~CI>DInV>n0S!JEby|J!R3sz;|&eT7oV7|?_wmc2IPNc=ma#(7@0ybq{T_yIYJ z-WzSP?~Jioe;s4f-x_5z-i9`?JmarpZRR^4*^Ix9waILo@a9PBpS@=zUVPhzJ^!YC z@Z1o4>&d}33!#l6P{7Sp-vdisSbmSYay2@_CzH!49 zSK41Lxzq+~AA0ny*Iv*5E1@SGcg0ogzuK-i|3do<{P(tj1MSXh;L}%JZug;EKMW0e zYS7)#w+HMsa_Rq-T$#i0t&If7W6-U~gXJkh-n3|Bsd)4;<+92LUuED}wV`XE?^WQb znx-7~)!2h;_{Y`4p7z6YYxwUqV5*j;`Y< j|?=5hpxqd!_r^urXmrR*f0a3DuaX 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 - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/zlib/zlib.vcproj.7.10.old b/zlib/zlib.vcproj.7.10.old deleted file mode 100644 index 9ae9e0c9..00000000 --- a/zlib/zlib.vcproj.7.10.old +++ /dev/null @@ -1,167 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -