From fbbb115b9805f5068a149066cc2844d6753c31d5 Mon Sep 17 00:00:00 2001 From: peterdell Date: Sun, 30 Dec 2018 16:42:36 +0100 Subject: [PATCH] Initial upload --- com.wudsn.ide.asm.compilers.test/.classpath | 7 + com.wudsn.ide.asm.compilers.test/.project | 28 + .../.settings/org.eclipse.jdt.core.prefs | 7 + .../META-INF/MANIFEST.MF | 10 + .../bin/.gitignore | 1 + .../build.properties | 16 + .../icons/editor-test-16x16.gif | Bin 0 -> 160 bytes .../icons/os/k_acorn.gif | Bin 0 -> 214 bytes .../icons/os/k_alambik.gif | Bin 0 -> 632 bytes .../icons/os/k_amiga.gif | Bin 0 -> 634 bytes .../icons/os/k_amiga_aga.gif | Bin 0 -> 631 bytes .../icons/os/k_amiga_ppc.gif | Bin 0 -> 640 bytes .../icons/os/k_amplus.gif | Bin 0 -> 375 bytes .../icons/os/k_apple2.gif | Bin 0 -> 619 bytes .../icons/os/k_atari_falcon.gif | Bin 0 -> 211 bytes .../icons/os/k_atari_st.gif | Bin 0 -> 214 bytes .../icons/os/k_atari_ste.gif | Bin 0 -> 207 bytes .../icons/os/k_atari_vcs.gif | Bin 0 -> 293 bytes .../icons/os/k_atari_xl_xe.gif | Bin 0 -> 211 bytes .../icons/os/k_bbcmicro.gif | Bin 0 -> 640 bytes .../icons/os/k_beos.gif | Bin 0 -> 274 bytes .../icons/os/k_bk.gif | Bin 0 -> 962 bytes .../icons/os/k_c16.gif | Bin 0 -> 216 bytes .../icons/os/k_coco.gif | Bin 0 -> 626 bytes .../icons/os/k_commodore.gif | Bin 0 -> 614 bytes .../icons/os/k_cpc.gif | Bin 0 -> 265 bytes .../icons/os/k_dreamcast.gif | Bin 0 -> 288 bytes .../icons/os/k_ds.gif | Bin 0 -> 208 bytes .../icons/os/k_dtv.gif | Bin 0 -> 621 bytes .../icons/os/k_ent.gif | Bin 0 -> 625 bytes .../icons/os/k_flash.gif | Bin 0 -> 217 bytes .../icons/os/k_freebsd.gif | Bin 0 -> 628 bytes .../icons/os/k_gameboy.gif | Bin 0 -> 408 bytes .../icons/os/k_gamepark32.gif | Bin 0 -> 215 bytes .../icons/os/k_gba.gif | Bin 0 -> 624 bytes .../icons/os/k_gbc.gif | Bin 0 -> 279 bytes .../icons/os/k_gg.gif | Bin 0 -> 378 bytes .../icons/os/k_gp2x.gif | Bin 0 -> 979 bytes .../icons/os/k_gus.gif | Bin 0 -> 309 bytes .../icons/os/k_iigs.gif | Bin 0 -> 407 bytes .../icons/os/k_intellivision.gif | Bin 0 -> 417 bytes .../icons/os/k_ipod.gif | Bin 0 -> 212 bytes .../icons/os/k_jag.gif | Bin 0 -> 382 bytes .../icons/os/k_java.gif | Bin 0 -> 630 bytes .../icons/os/k_js.gif | Bin 0 -> 201 bytes .../icons/os/k_linux.gif | Bin 0 -> 634 bytes .../icons/os/k_lynx.gif | Bin 0 -> 639 bytes .../icons/os/k_mac.gif | Bin 0 -> 282 bytes .../icons/os/k_macb.gif | Bin 0 -> 439 bytes .../icons/os/k_macosx.gif | Bin 0 -> 292 bytes .../icons/os/k_mastersystem.gif | Bin 0 -> 273 bytes .../icons/os/k_megadrive.gif | Bin 0 -> 273 bytes .../icons/os/k_mobiles.gif | Bin 0 -> 285 bytes .../icons/os/k_msdos.gif | Bin 0 -> 276 bytes .../icons/os/k_msx.gif | Bin 0 -> 169 bytes .../icons/os/k_msx2.gif | Bin 0 -> 171 bytes .../icons/os/k_msx2p.gif | Bin 0 -> 171 bytes .../icons/os/k_msxt.gif | Bin 0 -> 171 bytes .../icons/os/k_n64.gif | Bin 0 -> 640 bytes .../icons/os/k_neogeopocket.gif | Bin 0 -> 215 bytes .../icons/os/k_nes.gif | Bin 0 -> 263 bytes .../icons/os/k_ngc.gif | Bin 0 -> 635 bytes .../icons/os/k_oric.gif | Bin 0 -> 280 bytes .../icons/os/k_palmos.gif | Bin 0 -> 631 bytes .../icons/os/k_pce.gif | Bin 0 -> 268 bytes .../icons/os/k_php.gif | Bin 0 -> 204 bytes .../icons/os/k_pocketpc.gif | Bin 0 -> 301 bytes .../icons/os/k_pokemon.gif | Bin 0 -> 394 bytes .../icons/os/k_processing.gif | Bin 0 -> 194 bytes .../icons/os/k_ps1.gif | Bin 0 -> 634 bytes .../icons/os/k_ps2.gif | Bin 0 -> 285 bytes .../icons/os/k_ps3.gif | Bin 0 -> 289 bytes .../icons/os/k_psp.gif | Bin 0 -> 202 bytes .../icons/os/k_samcoupe.gif | Bin 0 -> 616 bytes .../icons/os/k_sgi.gif | Bin 0 -> 228 bytes .../icons/os/k_snes.gif | Bin 0 -> 286 bytes .../icons/os/k_solaris.gif | Bin 0 -> 649 bytes .../icons/os/k_sv.gif | Bin 0 -> 278 bytes .../icons/os/k_thomson.gif | Bin 0 -> 222 bytes .../icons/os/k_ti8x.gif | Bin 0 -> 277 bytes .../icons/os/k_tt.gif | Bin 0 -> 397 bytes .../icons/os/k_vb.gif | Bin 0 -> 221 bytes .../icons/os/k_vectrex.gif | Bin 0 -> 209 bytes .../icons/os/k_vic20.gif | Bin 0 -> 287 bytes .../icons/os/k_wii.gif | Bin 0 -> 212 bytes .../icons/os/k_wild.gif | Bin 0 -> 275 bytes .../icons/os/k_win.gif | Bin 0 -> 286 bytes .../icons/os/k_wonderswan.gif | Bin 0 -> 214 bytes .../icons/os/k_xbox.gif | Bin 0 -> 405 bytes .../icons/os/k_xbox360.gif | Bin 0 -> 1041 bytes .../icons/os/k_zx.gif | Bin 0 -> 394 bytes .../icons/os/k_zx81.gif | Bin 0 -> 396 bytes .../plugin.properties | 10 + com.wudsn.ide.asm.compilers.test/plugin.xml | 64 + .../plugin_de_DE.properties | 10 + .../ide/asm/compiler/test/TestCompiler.java | 50 + .../ide/asm/compiler/test/TestCompiler.xml | 362 +++ .../test/TestCompilerProcessLogParser.java | 174 ++ .../test/TestCompilerSourceParser.java | 52 + .../wudsn/ide/asm/editor/test/TestEditor.java | 18 + ...AssemblerPreferencesTestCompilersPage.java | 42 + com.wudsn.ide.asm.compilers/.classpath | 7 + com.wudsn.ide.asm.compilers/.project | 40 + .../.settings/org.eclipse.jdt.core.prefs | 7 + .../META-INF/MANIFEST.MF | 12 + com.wudsn.ide.asm.compilers/bin/.gitignore | 1 + .../binaries/.gitignore | 1 + com.wudsn.ide.asm.compilers/build.properties | 15 + .../icons/editor-acme-16x16.gif | Bin 0 -> 1763 bytes .../icons/editor-asm6-16x16.gif | Bin 0 -> 263 bytes .../icons/editor-atasm-16x16.gif | Bin 0 -> 909 bytes .../icons/editor-dasm-16x16.gif | Bin 0 -> 195 bytes .../icons/editor-kickass-16x16.gif | Bin 0 -> 1639 bytes .../icons/editor-mads-16x16.gif | Bin 0 -> 918 bytes .../icons/editor-merlin32-16x16.gif | Bin 0 -> 619 bytes .../icons/editor-tass-16x16.gif | Bin 0 -> 1639 bytes .../icons/editor-xasm-16x16.gif | Bin 0 -> 915 bytes com.wudsn.ide.asm.compilers/plugin.properties | 75 + com.wudsn.ide.asm.compilers/plugin.xml | 529 ++++ .../plugin_de_DE.properties | 75 + .../wudsn/ide/asm/compiler/CompilerId.java | 38 + .../ide/asm/compiler/acme/AcmeCompiler.java | 49 + .../ide/asm/compiler/acme/AcmeCompiler.xml | 95 + .../acme/AcmeCompilerProcessLogParser.java | 123 + .../acme/AcmeCompilerSourceParser.java | 59 + .../ide/asm/compiler/asm6/Asm6Compiler.java | 50 + .../ide/asm/compiler/asm6/Asm6Compiler.xml | 74 + .../asm6/Asm6CompilerProcessLogParser.java | 111 + .../asm6/Asm6CompilerSourceParser.java | 52 + .../ide/asm/compiler/atasm/AtasmCompiler.java | 50 + .../ide/asm/compiler/atasm/AtasmCompiler.xml | 252 ++ .../atasm/AtasmCompilerProcessLogParser.java | 154 ++ .../atasm/AtasmCompilerSourceParser.java | 51 + .../com/wudsn/ide/asm/compiler/dasm/DASM.txt | 750 ++++++ .../ide/asm/compiler/dasm/DasmCompiler.java | 50 + .../ide/asm/compiler/dasm/DasmCompiler.xml | 417 +++ .../dasm/DasmCompilerProcessLogParser.java | 270 ++ .../dasm/DasmCompilerSourceParser.java | 52 + .../compiler/kickass/KickAss-Libraries.txt | 424 +++ .../asm/compiler/kickass/KickAssCompiler.java | 50 + .../asm/compiler/kickass/KickAssCompiler.txt | 52 + .../asm/compiler/kickass/KickAssCompiler.xml | 400 +++ .../KickAssCompilerProcessLogParser.java | 118 + .../kickass/KickAssCompilerSourceParser.java | 74 + .../ide/asm/compiler/mads/MadsCompiler.java | 50 + .../ide/asm/compiler/mads/MadsCompiler.xml | 909 +++++++ .../mads/MadsCompilerProcessLogParser.java | 228 ++ .../mads/MadsCompilerSourceParser.java | 70 + .../compiler/merlin32/Merlin32Compiler.java | 50 + .../compiler/merlin32/Merlin32Compiler.xml | 80 + .../Merlin32CompilerProcessLogParser.java | 180 ++ .../Merlin32CompilerSourceParser.java | 55 + .../ide/asm/compiler/tass/TassCompiler.java | 50 + .../ide/asm/compiler/tass/TassCompiler.xml | 20 + .../tass/TassCompilerProcessLogParser.java | 171 ++ .../tass/TassCompilerSourceParser.java | 52 + .../ide/asm/compiler/xasm/XasmCompiler.java | 58 + .../ide/asm/compiler/xasm/XasmCompiler.xml | 268 ++ .../xasm/XasmCompilerProcessLogParser.java | 104 + .../xasm/XasmCompilerSourceParser.java | 53 + .../wudsn/ide/asm/editor/acme/AcmeEditor.java | 37 + .../wudsn/ide/asm/editor/asm6/Asm6Editor.java | 37 + .../ide/asm/editor/atasm/AtasmEditor.java | 38 + .../wudsn/ide/asm/editor/dasm/DasmEditor.java | 37 + .../ide/asm/editor/kickass/KickAssEditor.java | 38 + .../wudsn/ide/asm/editor/mads/MadsEditor.java | 38 + .../asm/editor/merlin32/Merlin32Editor.java | 38 + .../wudsn/ide/asm/editor/tass/TassEditor.java | 38 + .../wudsn/ide/asm/editor/xasm/XasmEditor.java | 38 + ...semblerPreferencesApple2CompilersPage.java | 42 + ...blerPreferencesAtari2600CompilersPage.java | 42 + ...blerPreferencesAtari7800CompilersPage.java | 42 + ...semblerPreferencesAtari8CompilersPage.java | 43 + .../AssemblerPreferencesC64CompilersPage.java | 42 + .../AssemblerPreferencesNESCompilersPage.java | 42 + .../wudsn/ide/asm/runner/atari8/Altirra.java | 78 + .../ide/asm/runner/atari8/Atari800Win.java | 32 + com.wudsn.ide.asm/.classpath | 7 + com.wudsn.ide.asm/.project | 28 + .../org.eclipse.core.resources.prefs | 4 + .../.settings/org.eclipse.jdt.core.prefs | 11 + .../.settings/org.eclipse.jdt.ui.prefs | 4 + com.wudsn.ide.asm/META-INF/MANIFEST.MF | 81 + com.wudsn.ide.asm/WUDSN-IDE (de).launch | 61 + com.wudsn.ide.asm/WUDSN-IDE (pl).launch | 61 + com.wudsn.ide.asm/WUDSN-IDE.launch | 56 + com.wudsn.ide.asm/bin/.gitignore | 1 + com.wudsn.ide.asm/build.properties | 16 + com.wudsn.ide.asm/help/create-links.bat | 21 + .../help/ide-credits.section.html | 204 ++ com.wudsn.ide.asm/help/ide-faq.section.html | 500 ++++ .../help/ide-features.section.html | 1014 +++++++ .../help/ide-installation.section.html | 548 ++++ .../help/ide-releases.section.html | 201 ++ .../help/ide-tutorials-videos.html | 42 + .../help/ide-tutorials.section.html | 85 + .../help/productions/java/.gitignore | 1 + .../help/productions/java/ide.txt | 1 + .../help/www.oxyron.de/html/opcodes.html | 24 + .../help/www.oxyron.de/html/opcodes02.html | 1124 ++++++++ .../help/www.oxyron.de/html/opcodes816.html | 562 ++++ .../help/www.oxyron.de/html/opcodesc02.html | 492 ++++ .../help/www.oxyron.de/html/opcodesce02.html | 560 ++++ .../www.oxyron.de/html/registers_antic.html | 298 +++ .../www.oxyron.de/html/registers_gtia.html | 314 +++ .../www.oxyron.de/html/registers_pokey.html | 196 ++ .../www.oxyron.de/html/registers_rec.html | 113 + .../www.oxyron.de/html/registers_sid.html | 254 ++ .../www.oxyron.de/html/registers_ted.html | 254 ++ .../www.oxyron.de/html/registers_vic1.html | 107 + .../www.oxyron.de/html/registers_vic2.html | 385 +++ .../help/www.oxyron.de/pics/noise.gif | Bin 0 -> 2858 bytes .../help/www.oxyron.de/pics/pulse.gif | Bin 0 -> 2112 bytes .../help/www.oxyron.de/pics/pulse1.gif | Bin 0 -> 1505 bytes .../help/www.oxyron.de/pics/pulse2.gif | Bin 0 -> 1510 bytes .../help/www.oxyron.de/pics/pulse3.gif | Bin 0 -> 1513 bytes .../help/www.oxyron.de/pics/sawtooth.gif | Bin 0 -> 2162 bytes .../help/www.oxyron.de/pics/triangle.gif | Bin 0 -> 2156 bytes .../help/www.oxyron.de/pics/vic2palette.png | Bin 0 -> 1047 bytes .../minidig/docs/2600_advanced_prog_guide.txt | 712 +++++ .../minidig/docs/2600_mem_map.txt | 632 +++++ .../minidig/docs/2600_riot_map.txt | 140 + .../minidig/docs/2600_tia_map.txt | 81 + ...mer's Guide (Unofficial HTML version).html | 2375 +++++++++++++++++ .../www.qotile.net/minidig/docs/sizes.txt | 1966 ++++++++++++++ .../www.qotile.net/minidig/docs/stella.pdf | Bin 0 -> 772501 bytes .../www.qotile.net/minidig/docs/stella.txt | 1938 ++++++++++++++ .../minidig/docs/tia_color.html | 898 +++++++ com.wudsn.ide.asm/icons/brkp_obj.gif | Bin 0 -> 197 bytes com.wudsn.ide.asm/icons/brkpd_obj.gif | Bin 0 -> 139 bytes .../icons/hardware-apple2-16x16.gif | Bin 0 -> 619 bytes .../icons/hardware-atari2600-16x16.gif | Bin 0 -> 293 bytes .../icons/hardware-atari7800-16x16.gif | Bin 0 -> 395 bytes .../icons/hardware-atari8bit-16x16.gif | Bin 0 -> 211 bytes .../icons/hardware-atari8bit-32x32.gif | Bin 0 -> 351 bytes .../icons/hardware-c64-16x16.gif | Bin 0 -> 614 bytes .../icons/hardware-generic-16x16.gif | Bin 0 -> 947 bytes .../icons/hardware-nes-16x16.gif | Bin 0 -> 263 bytes .../icons/hardware-test-16x16.gif | Bin 0 -> 961 bytes com.wudsn.ide.asm/icons/help-16x16.gif | Bin 0 -> 618 bytes .../icons/help-topic-pdf-small.gif | Bin 0 -> 924 bytes com.wudsn.ide.asm/icons/help-topic-small.gif | Bin 0 -> 910 bytes .../instruction-type-directive-16x16.gif | Bin 0 -> 595 bytes .../instruction-type-illegal-opcode-16x16.gif | Bin 0 -> 928 bytes .../instruction-type-legal-opcode-16x16.gif | Bin 0 -> 208 bytes .../instruction-type-pseudo-opcode-16x16.gif | Bin 0 -> 928 bytes .../icons/outline-binary-include-16x16.gif | Bin 0 -> 974 bytes .../icons/outline-binary-output-16x16.gif | Bin 0 -> 968 bytes .../outline-constant-definition-16x16.gif | Bin 0 -> 889 bytes .../icons/outline-default-16x16.gif | Bin 0 -> 193 bytes .../outline-definition-section-16x16.gif | Bin 0 -> 193 bytes com.wudsn.ide.asm/icons/outline-diamond.gif | Bin 0 -> 879 bytes .../outline-enum-definition-section-16x16.gif | Bin 0 -> 1657 bytes .../icons/outline-equate-definition-16x16.gif | Bin 0 -> 899 bytes .../outline-implementation-section-16x16.gif | Bin 0 -> 1660 bytes .../icons/outline-label-definition-16x16.gif | Bin 0 -> 152 bytes .../icons/outline-local-section-16x16.gif | Bin 0 -> 888 bytes ...outline-macro-definition-section-16x16.gif | Bin 0 -> 887 bytes .../icons/outline-pages-section-16x16.gif | Bin 0 -> 887 bytes ...ine-procedure-definition-section-16x16.gif | Bin 0 -> 887 bytes .../icons/outline-repeat-section-16x16.gif | Bin 0 -> 883 bytes com.wudsn.ide.asm/icons/outline-sort.gif | Bin 0 -> 153 bytes .../icons/outline-source-include-16x16.gif | Bin 0 -> 185 bytes ...ine-structure-definition-section-16x16.gif | Bin 0 -> 1657 bytes .../icons/outline-variable-16x16 - Copy.gif | Bin 0 -> 888 bytes com.wudsn.ide.asm/icons/sample.gif | Bin 0 -> 983 bytes com.wudsn.ide.asm/icons/topic.gif | Bin 0 -> 315 bytes com.wudsn.ide.asm/plugin.properties | 52 + com.wudsn.ide.asm/plugin.xml | 425 +++ com.wudsn.ide.asm/plugin_de_DE.properties | 52 + com.wudsn.ide.asm/schema/compilers.exsd | 197 ++ com.wudsn.ide.asm/schema/runners.exsd | 144 + .../src/com/wudsn/ide/asm/Actions.properties | 3 + .../com/wudsn/ide/asm/AssemblerPlugin.java | 305 +++ .../wudsn/ide/asm/AssemblerProperties.java | 144 + .../src/com/wudsn/ide/asm/CPU.java | 31 + .../src/com/wudsn/ide/asm/Hardware.java | 31 + .../com/wudsn/ide/asm/HardwareUtility.java | 170 ++ .../src/com/wudsn/ide/asm/Texts.java | 288 ++ .../src/com/wudsn/ide/asm/Texts.properties | 206 ++ .../com/wudsn/ide/asm/Texts_de_DE.properties | 204 ++ .../com/wudsn/ide/asm/compiler/Compiler.java | 98 + .../com/wudsn/ide/asm/compiler/Compiler.xml | 113 + .../ide/asm/compiler/CompilerConsole.java | 102 + .../ide/asm/compiler/CompilerDefinition.java | 436 +++ .../ide/asm/compiler/CompilerFileWriter.java | 39 + .../wudsn/ide/asm/compiler/CompilerFiles.java | 256 ++ .../compiler/CompilerOutputFolderMode.java | 73 + .../compiler/CompilerProcessLogParser.java | 286 ++ .../ide/asm/compiler/CompilerRegistry.java | 244 ++ .../ide/asm/compiler/CompilerSymbol.java | 156 ++ .../ide/asm/compiler/CompilerSymbolType.java | 77 + .../ide/asm/compiler/CompilerVariables.java | 75 + .../compiler/parser/CompilerSourceFile.java | 460 ++++ .../compiler/parser/CompilerSourceParser.java | 1214 +++++++++ .../CompilerSourceParserFileReference.java | 72 + ...CompilerSourceParserFileReferenceType.java | 38 + .../CompilerSourceParserLineCallback.java | 85 + .../CompilerSourceParserTreeObject.java | 411 +++ .../CompilerSourceParserTreeObjectType.java | 143 + .../CompilerSourcePartitionScanner.java | 141 + .../asm/compiler/syntax/CompilerSyntax.java | 936 +++++++ .../syntax/CompilerSyntaxUtility.java | 110 + .../ide/asm/compiler/syntax/Directive.java | 57 + .../ide/asm/compiler/syntax/Instruction.java | 205 ++ .../asm/compiler/syntax/InstructionSet.java | 273 ++ .../asm/compiler/syntax/InstructionType.java | 63 + .../wudsn/ide/asm/compiler/syntax/Opcode.java | 199 ++ .../asm/compiler/writer/AppleFileWriter.java | 343 +++ ...emblerBreakpoinDebugModelPresentation.java | 114 + .../ide/asm/editor/AssemblerBreakpoint.java | 172 ++ .../AssemblerBreakpointAdapterFactory.java | 88 + .../editor/AssemblerBreakpointsTarget.java | 168 ++ .../AssemblerContentAssistProcessor.java | 573 ++++ ...blerContentAssistProcessorDefaultCase.java | 39 + .../editor/AssemblerContentOutlinePage.java | 590 ++++ ...blerContentOutlineTreeContentProvider.java | 182 ++ .../wudsn/ide/asm/editor/AssemblerEditor.java | 558 ++++ ...semblerEditorCompileAndRunCommandMenu.java | 64 + .../editor/AssemblerEditorCompileCommand.java | 802 ++++++ ...AssemblerEditorCompileCommandDelegate.java | 243 ++ .../AssemblerEditorCompileCommandHandler.java | 63 + ...erEditorCompileCommandPositioningMode.java | 39 + ...mblerEditorCompilerHelpCommandHandler.java | 74 + .../AssemblerEditorFilesCommandHandler.java | 98 + .../asm/editor/AssemblerEditorFilesLogic.java | 372 +++ ...erEditorOpenDeclarationCommandHandler.java | 66 + ...semblerEditorOpenFolderCommandHandler.java | 70 + .../AssemblerEditorToggleCommentAction.java | 168 ++ .../ide/asm/editor/AssemblerHyperlink.java | 236 ++ .../editor/AssemblerHyperlinkDetector.java | 332 +++ ...ssemblerInstructionCompletionProposal.java | 167 ++ .../editor/AssemblerReconcilingStategy.java | 109 + .../asm/editor/AssemblerRuleBasedScanner.java | 102 + .../asm/editor/AssemblerSourceScanner.java | 516 ++++ .../AssemblerSourceViewerConfiguration.java | 225 ++ ...erSourceParserTreeObjectLabelProvider.java | 201 ++ .../editor/CompilerSymbolLabelProvider.java | 111 + .../ide/asm/editor/CompilerSymbolsView.java | 296 ++ .../help/AssemblerHelpContentProducer.java | 684 +++++ .../ide/asm/help/AssemblerTocProvider.java | 414 +++ .../asm/help/AssemblerTocProvider.properties | 23 + .../AssemblerTocProvider_de_DE.properties | 23 + .../com/wudsn/ide/asm/help/HTMLConstants.java | 45 + .../ide/asm/help/HTMLWrapperInputStream.java | 104 + .../com/wudsn/ide/asm/help/HTMLWriter.java | 211 ++ .../asm/preferences/AssemblerPreferences.java | 192 ++ .../AssemblerPreferencesChangeListener.java | 43 + .../AssemblerPreferencesCompilersPage.java | 653 +++++ .../AssemblerPreferencesConstants.java | 419 +++ .../AssemblerPreferencesInitializer.java | 171 ++ .../preferences/AssemblerPreferencesPage.java | 597 +++++ .../asm/preferences/CompilerPreferences.java | 262 ++ .../preferences/CompilerRunPreferences.java | 154 ++ .../DirectoryFieldDownloadEditor.java | 137 + .../preferences/FileFieldDownloadEditor.java | 137 + .../preferences/TextAttributeConverter.java | 162 ++ .../TextAttributeListContentProvider.java | 61 + .../preferences/TextAttributeListItem.java | 98 + .../TextAttributeListItemProvider.java | 70 + .../src/com/wudsn/ide/asm/runner/Runner.java | 111 + .../ide/asm/runner/RunnerDefinition.java | 245 ++ .../com/wudsn/ide/asm/runner/RunnerId.java | 40 + .../wudsn/ide/asm/runner/RunnerRegistry.java | 233 ++ com.wudsn.ide.base.feature/.project | 17 + com.wudsn.ide.base.feature/build.properties | 3 + com.wudsn.ide.base.feature/feature.xml | 321 +++ com.wudsn.ide.base/.classpath | 7 + com.wudsn.ide.base/.project | 28 + .../.settings/org.eclipse.jdt.core.prefs | 11 + com.wudsn.ide.base/META-INF/MANIFEST.MF | 26 + com.wudsn.ide.base/bin/.gitignore | 1 + com.wudsn.ide.base/build.properties | 14 + .../fonts/atari8/AtariClassic-Regular.ttf | Bin 0 -> 29096 bytes .../fonts/atari8/AtariClassicInt-Regular.ttf | Bin 0 -> 30944 bytes com.wudsn.ide.base/fonts/atari8/ReadMe.rtf | 317 +++ .../fonts/c64/C64Classic-Regular.ttf | Bin 0 -> 35036 bytes com.wudsn.ide.base/icons/copy-disabled.gif | Bin 0 -> 366 bytes com.wudsn.ide.base/icons/copy-enabled.gif | Bin 0 -> 594 bytes com.wudsn.ide.base/icons/cut-disabled.gif | Bin 0 -> 138 bytes com.wudsn.ide.base/icons/cut-enabled.gif | Bin 0 -> 212 bytes com.wudsn.ide.base/icons/delete-disabled.gif | Bin 0 -> 221 bytes com.wudsn.ide.base/icons/hex-editor-16x16.gif | Bin 0 -> 365 bytes .../icons/hex-editor-segment-16x16.gif | Bin 0 -> 183 bytes com.wudsn.ide.base/icons/import-disabled.gif | Bin 0 -> 143 bytes com.wudsn.ide.base/icons/import-enabled.gif | Bin 0 -> 327 bytes com.wudsn.ide.base/icons/paste-disabled.gif | Bin 0 -> 361 bytes com.wudsn.ide.base/icons/paste-enabled.gif | Bin 0 -> 605 bytes com.wudsn.ide.base/icons/save-as-disabled.gif | Bin 0 -> 232 bytes com.wudsn.ide.base/icons/save-as-enabled.gif | Bin 0 -> 583 bytes com.wudsn.ide.base/plugin.properties | 58 + com.wudsn.ide.base/plugin.xml | 632 +++++ com.wudsn.ide.base/plugin_de_DE.properties | 58 + .../src/com/wudsn/ide/base/BasePlugin.java | 87 + .../src/com/wudsn/ide/base/Texts.java | 116 + .../src/com/wudsn/ide/base/Texts.properties | 63 + .../com/wudsn/ide/base/Texts_de_DE.properties | 63 + .../ide/base/common/AbstractIDEPlugin.java | 293 ++ .../ide/base/common/ByteArrayUtility.java | 31 + .../ide/base/common/ClassPathUtility.java | 97 + .../wudsn/ide/base/common/EnumUtility.java | 69 + .../wudsn/ide/base/common/FileUtility.java | 544 ++++ .../com/wudsn/ide/base/common/HexUtility.java | 159 ++ .../wudsn/ide/base/common/IPathUtility.java | 77 + .../ide/base/common/InputStreamMonitor.java | 176 ++ .../wudsn/ide/base/common/MarkerUtility.java | 125 + .../wudsn/ide/base/common/NumberFactory.java | 57 + .../wudsn/ide/base/common/NumberUtility.java | 89 + .../ide/base/common/OutputStreamMonitor.java | 206 ++ .../ide/base/common/ProcessWithLogs.java | 227 ++ .../com/wudsn/ide/base/common/Profiler.java | 88 + .../ide/base/common/PropertiesSerializer.java | 233 ++ .../ide/base/common/ResourceUtility.java | 317 +++ .../ide/base/common/RunnableWithLogging.java | 45 + .../ide/base/common/SequencedProperties.java | 80 + .../wudsn/ide/base/common/StreamsProxy.java | 100 + .../wudsn/ide/base/common/StringUtility.java | 70 + .../wudsn/ide/base/common/TextUtility.java | 71 + .../CommonOpenEditorCommandHandler.java | 151 ++ .../CommonOpenFolderCommandHandler.java | 154 ++ .../wudsn/ide/base/editor/hex/Hardware.java | 31 + .../wudsn/ide/base/editor/hex/HexEditor.java | 648 +++++ .../editor/hex/HexEditorCharacterSet.java | 270 ++ .../hex/HexEditorClipboardCommandHandler.java | 184 ++ .../HexEditorContentOutlineLabelProvider.java | 86 + .../hex/HexEditorContentOutlinePage.java | 88 + ...itorContentOutlineTreeContentProvider.java | 94 + .../HexEditorContentOutlineTreeObject.java | 148 + .../editor/hex/HexEditorFileContentMode.java | 40 + .../hex/HexEditorOpenCommandHandler.java | 39 + .../ide/base/editor/hex/HexEditorParser.java | 226 ++ .../editor/hex/HexEditorParserComponent.java | 727 +++++ ...exEditorSaveSelectionAsCommandHandler.java | 79 + .../base/editor/hex/HexEditorSelection.java | 106 + .../hex/HexEditorSelectionCommandHandler.java | 77 + .../hex/HexEditorSelectionTransfer.java | 96 + .../editor/hex/parser/AtariCOMParser.java | 36 + .../hex/parser/AtariDiskImageKFileParser.java | 47 + .../hex/parser/AtariDiskImageParser.java | 89 + .../editor/hex/parser/AtariMADSParser.java | 221 ++ .../base/editor/hex/parser/AtariParser.java | 127 + .../editor/hex/parser/AtariSAPParser.java | 51 + .../editor/hex/parser/AtariSDXParser.java | 195 ++ .../base/editor/hex/parser/BinaryParser.java | 39 + .../base/editor/hex/parser/C64PRGParser.java | 63 + .../text/TextEditorConvertNumbersCommand.java | 296 ++ .../text/TextEditorLinesCommandHandler.java | 139 + .../TextEditorReverseLinesCommandHandler.java | 51 + ...ortLinesCaseInsensitiveCommandHandler.java | 37 + ...rSortLinesCaseSensitiveCommandHandler.java | 50 + .../TextEditorSortLinesCommandHandler.java | 71 + ...tEditorSortLinesNumericCommandHandler.java | 81 + .../src/com/wudsn/ide/base/gui/Action.java | 109 + .../wudsn/ide/base/gui/ActionListener.java | 35 + .../com/wudsn/ide/base/gui/Application.java | 33 + .../com/wudsn/ide/base/gui/CheckBoxField.java | 133 + .../src/com/wudsn/ide/base/gui/EnumField.java | 226 ++ .../src/com/wudsn/ide/base/gui/Field.java | 100 + .../com/wudsn/ide/base/gui/FilePathField.java | 277 ++ .../com/wudsn/ide/base/gui/IntegerField.java | 291 ++ .../wudsn/ide/base/gui/MessageManager.java | 370 +++ .../ide/base/gui/MultiLineTextField.java | 210 ++ .../com/wudsn/ide/base/gui/SWTFactory.java | 133 + .../src/com/wudsn/ide/base/gui/TextField.java | 133 + com.wudsn.ide.feature/.project | 17 + com.wudsn.ide.feature/build.properties | 4 + com.wudsn.ide.feature/feature.xml | 378 +++ com.wudsn.ide.feature/gpl-2_0.txt | 280 ++ 468 files changed, 58526 insertions(+) create mode 100644 com.wudsn.ide.asm.compilers.test/.classpath create mode 100644 com.wudsn.ide.asm.compilers.test/.project create mode 100644 com.wudsn.ide.asm.compilers.test/.settings/org.eclipse.jdt.core.prefs create mode 100644 com.wudsn.ide.asm.compilers.test/META-INF/MANIFEST.MF create mode 100644 com.wudsn.ide.asm.compilers.test/bin/.gitignore create mode 100644 com.wudsn.ide.asm.compilers.test/build.properties create mode 100644 com.wudsn.ide.asm.compilers.test/icons/editor-test-16x16.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_acorn.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_alambik.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_amiga.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_amiga_aga.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_amiga_ppc.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_amplus.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_apple2.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_atari_falcon.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_atari_st.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_atari_ste.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_atari_vcs.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_atari_xl_xe.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_bbcmicro.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_beos.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_bk.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_c16.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_coco.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_commodore.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_cpc.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_dreamcast.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_ds.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_dtv.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_ent.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_flash.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_freebsd.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_gameboy.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_gamepark32.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_gba.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_gbc.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_gg.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_gp2x.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_gus.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_iigs.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_intellivision.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_ipod.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_jag.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_java.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_js.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_linux.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_lynx.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_mac.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_macb.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_macosx.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_mastersystem.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_megadrive.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_mobiles.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_msdos.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_msx.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_msx2.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_msx2p.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_msxt.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_n64.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_neogeopocket.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_nes.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_ngc.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_oric.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_palmos.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_pce.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_php.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_pocketpc.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_pokemon.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_processing.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_ps1.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_ps2.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_ps3.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_psp.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_samcoupe.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_sgi.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_snes.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_solaris.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_sv.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_thomson.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_ti8x.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_tt.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_vb.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_vectrex.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_vic20.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_wii.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_wild.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_win.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_wonderswan.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_xbox.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_xbox360.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_zx.gif create mode 100644 com.wudsn.ide.asm.compilers.test/icons/os/k_zx81.gif create mode 100644 com.wudsn.ide.asm.compilers.test/plugin.properties create mode 100644 com.wudsn.ide.asm.compilers.test/plugin.xml create mode 100644 com.wudsn.ide.asm.compilers.test/plugin_de_DE.properties create mode 100644 com.wudsn.ide.asm.compilers.test/src/com/wudsn/ide/asm/compiler/test/TestCompiler.java create mode 100644 com.wudsn.ide.asm.compilers.test/src/com/wudsn/ide/asm/compiler/test/TestCompiler.xml create mode 100644 com.wudsn.ide.asm.compilers.test/src/com/wudsn/ide/asm/compiler/test/TestCompilerProcessLogParser.java create mode 100644 com.wudsn.ide.asm.compilers.test/src/com/wudsn/ide/asm/compiler/test/TestCompilerSourceParser.java create mode 100644 com.wudsn.ide.asm.compilers.test/src/com/wudsn/ide/asm/editor/test/TestEditor.java create mode 100644 com.wudsn.ide.asm.compilers.test/src/com/wudsn/ide/asm/preferences/test/AssemblerPreferencesTestCompilersPage.java create mode 100644 com.wudsn.ide.asm.compilers/.classpath create mode 100644 com.wudsn.ide.asm.compilers/.project create mode 100644 com.wudsn.ide.asm.compilers/.settings/org.eclipse.jdt.core.prefs create mode 100644 com.wudsn.ide.asm.compilers/META-INF/MANIFEST.MF create mode 100644 com.wudsn.ide.asm.compilers/bin/.gitignore create mode 100644 com.wudsn.ide.asm.compilers/binaries/.gitignore create mode 100644 com.wudsn.ide.asm.compilers/build.properties create mode 100644 com.wudsn.ide.asm.compilers/icons/editor-acme-16x16.gif create mode 100644 com.wudsn.ide.asm.compilers/icons/editor-asm6-16x16.gif create mode 100644 com.wudsn.ide.asm.compilers/icons/editor-atasm-16x16.gif create mode 100644 com.wudsn.ide.asm.compilers/icons/editor-dasm-16x16.gif create mode 100644 com.wudsn.ide.asm.compilers/icons/editor-kickass-16x16.gif create mode 100644 com.wudsn.ide.asm.compilers/icons/editor-mads-16x16.gif create mode 100644 com.wudsn.ide.asm.compilers/icons/editor-merlin32-16x16.gif create mode 100644 com.wudsn.ide.asm.compilers/icons/editor-tass-16x16.gif create mode 100644 com.wudsn.ide.asm.compilers/icons/editor-xasm-16x16.gif create mode 100644 com.wudsn.ide.asm.compilers/plugin.properties create mode 100644 com.wudsn.ide.asm.compilers/plugin.xml create mode 100644 com.wudsn.ide.asm.compilers/plugin_de_DE.properties create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/CompilerId.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/acme/AcmeCompiler.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/acme/AcmeCompiler.xml create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/acme/AcmeCompilerProcessLogParser.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/acme/AcmeCompilerSourceParser.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/asm6/Asm6Compiler.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/asm6/Asm6Compiler.xml create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/asm6/Asm6CompilerProcessLogParser.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/asm6/Asm6CompilerSourceParser.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/atasm/AtasmCompiler.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/atasm/AtasmCompiler.xml create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/atasm/AtasmCompilerProcessLogParser.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/atasm/AtasmCompilerSourceParser.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/dasm/DASM.txt create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/dasm/DasmCompiler.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/dasm/DasmCompiler.xml create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/dasm/DasmCompilerProcessLogParser.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/dasm/DasmCompilerSourceParser.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/kickass/KickAss-Libraries.txt create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/kickass/KickAssCompiler.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/kickass/KickAssCompiler.txt create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/kickass/KickAssCompiler.xml create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/kickass/KickAssCompilerProcessLogParser.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/kickass/KickAssCompilerSourceParser.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/mads/MadsCompiler.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/mads/MadsCompiler.xml create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/mads/MadsCompilerProcessLogParser.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/mads/MadsCompilerSourceParser.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/merlin32/Merlin32Compiler.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/merlin32/Merlin32Compiler.xml create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/merlin32/Merlin32CompilerProcessLogParser.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/merlin32/Merlin32CompilerSourceParser.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/tass/TassCompiler.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/tass/TassCompiler.xml create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/tass/TassCompilerProcessLogParser.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/tass/TassCompilerSourceParser.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/xasm/XasmCompiler.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/xasm/XasmCompiler.xml create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/xasm/XasmCompilerProcessLogParser.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/xasm/XasmCompilerSourceParser.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/acme/AcmeEditor.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/asm6/Asm6Editor.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/atasm/AtasmEditor.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/dasm/DasmEditor.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/kickass/KickAssEditor.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/mads/MadsEditor.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/merlin32/Merlin32Editor.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/tass/TassEditor.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/xasm/XasmEditor.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesApple2CompilersPage.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesAtari2600CompilersPage.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesAtari7800CompilersPage.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesAtari8CompilersPage.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesC64CompilersPage.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesNESCompilersPage.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/runner/atari8/Altirra.java create mode 100644 com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/runner/atari8/Atari800Win.java create mode 100644 com.wudsn.ide.asm/.classpath create mode 100644 com.wudsn.ide.asm/.project create mode 100644 com.wudsn.ide.asm/.settings/org.eclipse.core.resources.prefs create mode 100644 com.wudsn.ide.asm/.settings/org.eclipse.jdt.core.prefs create mode 100644 com.wudsn.ide.asm/.settings/org.eclipse.jdt.ui.prefs create mode 100644 com.wudsn.ide.asm/META-INF/MANIFEST.MF create mode 100644 com.wudsn.ide.asm/WUDSN-IDE (de).launch create mode 100644 com.wudsn.ide.asm/WUDSN-IDE (pl).launch create mode 100644 com.wudsn.ide.asm/WUDSN-IDE.launch create mode 100644 com.wudsn.ide.asm/bin/.gitignore create mode 100644 com.wudsn.ide.asm/build.properties create mode 100644 com.wudsn.ide.asm/help/create-links.bat create mode 100644 com.wudsn.ide.asm/help/ide-credits.section.html create mode 100644 com.wudsn.ide.asm/help/ide-faq.section.html create mode 100644 com.wudsn.ide.asm/help/ide-features.section.html create mode 100644 com.wudsn.ide.asm/help/ide-installation.section.html create mode 100644 com.wudsn.ide.asm/help/ide-releases.section.html create mode 100644 com.wudsn.ide.asm/help/ide-tutorials-videos.html create mode 100644 com.wudsn.ide.asm/help/ide-tutorials.section.html create mode 100644 com.wudsn.ide.asm/help/productions/java/.gitignore create mode 100644 com.wudsn.ide.asm/help/productions/java/ide.txt create mode 100644 com.wudsn.ide.asm/help/www.oxyron.de/html/opcodes.html create mode 100644 com.wudsn.ide.asm/help/www.oxyron.de/html/opcodes02.html create mode 100644 com.wudsn.ide.asm/help/www.oxyron.de/html/opcodes816.html create mode 100644 com.wudsn.ide.asm/help/www.oxyron.de/html/opcodesc02.html create mode 100644 com.wudsn.ide.asm/help/www.oxyron.de/html/opcodesce02.html create mode 100644 com.wudsn.ide.asm/help/www.oxyron.de/html/registers_antic.html create mode 100644 com.wudsn.ide.asm/help/www.oxyron.de/html/registers_gtia.html create mode 100644 com.wudsn.ide.asm/help/www.oxyron.de/html/registers_pokey.html create mode 100644 com.wudsn.ide.asm/help/www.oxyron.de/html/registers_rec.html create mode 100644 com.wudsn.ide.asm/help/www.oxyron.de/html/registers_sid.html create mode 100644 com.wudsn.ide.asm/help/www.oxyron.de/html/registers_ted.html create mode 100644 com.wudsn.ide.asm/help/www.oxyron.de/html/registers_vic1.html create mode 100644 com.wudsn.ide.asm/help/www.oxyron.de/html/registers_vic2.html create mode 100644 com.wudsn.ide.asm/help/www.oxyron.de/pics/noise.gif create mode 100644 com.wudsn.ide.asm/help/www.oxyron.de/pics/pulse.gif create mode 100644 com.wudsn.ide.asm/help/www.oxyron.de/pics/pulse1.gif create mode 100644 com.wudsn.ide.asm/help/www.oxyron.de/pics/pulse2.gif create mode 100644 com.wudsn.ide.asm/help/www.oxyron.de/pics/pulse3.gif create mode 100644 com.wudsn.ide.asm/help/www.oxyron.de/pics/sawtooth.gif create mode 100644 com.wudsn.ide.asm/help/www.oxyron.de/pics/triangle.gif create mode 100644 com.wudsn.ide.asm/help/www.oxyron.de/pics/vic2palette.png create mode 100644 com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/2600_advanced_prog_guide.txt create mode 100644 com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/2600_mem_map.txt create mode 100644 com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/2600_riot_map.txt create mode 100644 com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/2600_tia_map.txt create mode 100644 com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/Stella Programmer's Guide (Unofficial HTML version).html create mode 100644 com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/sizes.txt create mode 100644 com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/stella.pdf create mode 100644 com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/stella.txt create mode 100644 com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/tia_color.html create mode 100644 com.wudsn.ide.asm/icons/brkp_obj.gif create mode 100644 com.wudsn.ide.asm/icons/brkpd_obj.gif create mode 100644 com.wudsn.ide.asm/icons/hardware-apple2-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/hardware-atari2600-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/hardware-atari7800-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/hardware-atari8bit-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/hardware-atari8bit-32x32.gif create mode 100644 com.wudsn.ide.asm/icons/hardware-c64-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/hardware-generic-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/hardware-nes-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/hardware-test-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/help-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/help-topic-pdf-small.gif create mode 100644 com.wudsn.ide.asm/icons/help-topic-small.gif create mode 100644 com.wudsn.ide.asm/icons/instruction-type-directive-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/instruction-type-illegal-opcode-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/instruction-type-legal-opcode-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/instruction-type-pseudo-opcode-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/outline-binary-include-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/outline-binary-output-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/outline-constant-definition-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/outline-default-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/outline-definition-section-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/outline-diamond.gif create mode 100644 com.wudsn.ide.asm/icons/outline-enum-definition-section-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/outline-equate-definition-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/outline-implementation-section-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/outline-label-definition-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/outline-local-section-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/outline-macro-definition-section-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/outline-pages-section-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/outline-procedure-definition-section-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/outline-repeat-section-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/outline-sort.gif create mode 100644 com.wudsn.ide.asm/icons/outline-source-include-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/outline-structure-definition-section-16x16.gif create mode 100644 com.wudsn.ide.asm/icons/outline-variable-16x16 - Copy.gif create mode 100644 com.wudsn.ide.asm/icons/sample.gif create mode 100644 com.wudsn.ide.asm/icons/topic.gif create mode 100644 com.wudsn.ide.asm/plugin.properties create mode 100644 com.wudsn.ide.asm/plugin.xml create mode 100644 com.wudsn.ide.asm/plugin_de_DE.properties create mode 100644 com.wudsn.ide.asm/schema/compilers.exsd create mode 100644 com.wudsn.ide.asm/schema/runners.exsd create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/Actions.properties create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/AssemblerPlugin.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/AssemblerProperties.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/CPU.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/Hardware.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/HardwareUtility.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/Texts.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/Texts.properties create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/Texts_de_DE.properties create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/Compiler.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/Compiler.xml create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerConsole.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerDefinition.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerFileWriter.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerFiles.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerOutputFolderMode.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerProcessLogParser.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerRegistry.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerSymbol.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerSymbolType.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerVariables.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceFile.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceParser.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceParserFileReference.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceParserFileReferenceType.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceParserLineCallback.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceParserTreeObject.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceParserTreeObjectType.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourcePartitionScanner.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/CompilerSyntax.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/CompilerSyntaxUtility.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/Directive.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/Instruction.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/InstructionSet.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/InstructionType.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/Opcode.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/writer/AppleFileWriter.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerBreakpoinDebugModelPresentation.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerBreakpoint.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerBreakpointAdapterFactory.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerBreakpointsTarget.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerContentAssistProcessor.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerContentAssistProcessorDefaultCase.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerContentOutlinePage.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerContentOutlineTreeContentProvider.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditor.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorCompileAndRunCommandMenu.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorCompileCommand.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorCompileCommandDelegate.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorCompileCommandHandler.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorCompileCommandPositioningMode.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorCompilerHelpCommandHandler.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorFilesCommandHandler.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorFilesLogic.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorOpenDeclarationCommandHandler.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorOpenFolderCommandHandler.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorToggleCommentAction.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerHyperlink.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerHyperlinkDetector.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerInstructionCompletionProposal.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerReconcilingStategy.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerRuleBasedScanner.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerSourceScanner.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerSourceViewerConfiguration.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/CompilerSourceParserTreeObjectLabelProvider.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/CompilerSymbolLabelProvider.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/CompilerSymbolsView.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/AssemblerHelpContentProducer.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/AssemblerTocProvider.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/AssemblerTocProvider.properties create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/AssemblerTocProvider_de_DE.properties create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/HTMLConstants.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/HTMLWrapperInputStream.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/HTMLWriter.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/AssemblerPreferences.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesChangeListener.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesCompilersPage.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesConstants.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesInitializer.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesPage.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/CompilerPreferences.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/CompilerRunPreferences.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/DirectoryFieldDownloadEditor.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/FileFieldDownloadEditor.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/TextAttributeConverter.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/TextAttributeListContentProvider.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/TextAttributeListItem.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/TextAttributeListItemProvider.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/runner/Runner.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/runner/RunnerDefinition.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/runner/RunnerId.java create mode 100644 com.wudsn.ide.asm/src/com/wudsn/ide/asm/runner/RunnerRegistry.java create mode 100644 com.wudsn.ide.base.feature/.project create mode 100644 com.wudsn.ide.base.feature/build.properties create mode 100644 com.wudsn.ide.base.feature/feature.xml create mode 100644 com.wudsn.ide.base/.classpath create mode 100644 com.wudsn.ide.base/.project create mode 100644 com.wudsn.ide.base/.settings/org.eclipse.jdt.core.prefs create mode 100644 com.wudsn.ide.base/META-INF/MANIFEST.MF create mode 100644 com.wudsn.ide.base/bin/.gitignore create mode 100644 com.wudsn.ide.base/build.properties create mode 100644 com.wudsn.ide.base/fonts/atari8/AtariClassic-Regular.ttf create mode 100644 com.wudsn.ide.base/fonts/atari8/AtariClassicInt-Regular.ttf create mode 100644 com.wudsn.ide.base/fonts/atari8/ReadMe.rtf create mode 100644 com.wudsn.ide.base/fonts/c64/C64Classic-Regular.ttf create mode 100644 com.wudsn.ide.base/icons/copy-disabled.gif create mode 100644 com.wudsn.ide.base/icons/copy-enabled.gif create mode 100644 com.wudsn.ide.base/icons/cut-disabled.gif create mode 100644 com.wudsn.ide.base/icons/cut-enabled.gif create mode 100644 com.wudsn.ide.base/icons/delete-disabled.gif create mode 100644 com.wudsn.ide.base/icons/hex-editor-16x16.gif create mode 100644 com.wudsn.ide.base/icons/hex-editor-segment-16x16.gif create mode 100644 com.wudsn.ide.base/icons/import-disabled.gif create mode 100644 com.wudsn.ide.base/icons/import-enabled.gif create mode 100644 com.wudsn.ide.base/icons/paste-disabled.gif create mode 100644 com.wudsn.ide.base/icons/paste-enabled.gif create mode 100644 com.wudsn.ide.base/icons/save-as-disabled.gif create mode 100644 com.wudsn.ide.base/icons/save-as-enabled.gif create mode 100644 com.wudsn.ide.base/plugin.properties create mode 100644 com.wudsn.ide.base/plugin.xml create mode 100644 com.wudsn.ide.base/plugin_de_DE.properties create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/BasePlugin.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/Texts.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/Texts.properties create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/Texts_de_DE.properties create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/common/AbstractIDEPlugin.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/common/ByteArrayUtility.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/common/ClassPathUtility.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/common/EnumUtility.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/common/FileUtility.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/common/HexUtility.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/common/IPathUtility.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/common/InputStreamMonitor.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/common/MarkerUtility.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/common/NumberFactory.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/common/NumberUtility.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/common/OutputStreamMonitor.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/common/ProcessWithLogs.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/common/Profiler.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/common/PropertiesSerializer.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/common/ResourceUtility.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/common/RunnableWithLogging.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/common/SequencedProperties.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/common/StreamsProxy.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/common/StringUtility.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/common/TextUtility.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/CommonOpenEditorCommandHandler.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/CommonOpenFolderCommandHandler.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/Hardware.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditor.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorCharacterSet.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorClipboardCommandHandler.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorContentOutlineLabelProvider.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorContentOutlinePage.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorContentOutlineTreeContentProvider.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorContentOutlineTreeObject.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorFileContentMode.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorOpenCommandHandler.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorParser.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorParserComponent.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorSaveSelectionAsCommandHandler.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorSelection.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorSelectionCommandHandler.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorSelectionTransfer.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariCOMParser.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariDiskImageKFileParser.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariDiskImageParser.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariMADSParser.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariParser.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariSAPParser.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariSDXParser.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/BinaryParser.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/C64PRGParser.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorConvertNumbersCommand.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorLinesCommandHandler.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorReverseLinesCommandHandler.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorSortLinesCaseInsensitiveCommandHandler.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorSortLinesCaseSensitiveCommandHandler.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorSortLinesCommandHandler.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorSortLinesNumericCommandHandler.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/gui/Action.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/gui/ActionListener.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/gui/Application.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/gui/CheckBoxField.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/gui/EnumField.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/gui/Field.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/gui/FilePathField.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/gui/IntegerField.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/gui/MessageManager.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/gui/MultiLineTextField.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/gui/SWTFactory.java create mode 100644 com.wudsn.ide.base/src/com/wudsn/ide/base/gui/TextField.java create mode 100644 com.wudsn.ide.feature/.project create mode 100644 com.wudsn.ide.feature/build.properties create mode 100644 com.wudsn.ide.feature/feature.xml create mode 100644 com.wudsn.ide.feature/gpl-2_0.txt diff --git a/com.wudsn.ide.asm.compilers.test/.classpath b/com.wudsn.ide.asm.compilers.test/.classpath new file mode 100644 index 00000000..8a8f1668 --- /dev/null +++ b/com.wudsn.ide.asm.compilers.test/.classpath @@ -0,0 +1,7 @@ + + + + + + + diff --git a/com.wudsn.ide.asm.compilers.test/.project b/com.wudsn.ide.asm.compilers.test/.project new file mode 100644 index 00000000..0ef5742c --- /dev/null +++ b/com.wudsn.ide.asm.compilers.test/.project @@ -0,0 +1,28 @@ + + + com.wudsn.ide.asm.compilers.test + + + + + + org.eclipse.jdt.core.javabuilder + + + + + org.eclipse.pde.ManifestBuilder + + + + + org.eclipse.pde.SchemaBuilder + + + + + + org.eclipse.pde.PluginNature + org.eclipse.jdt.core.javanature + + diff --git a/com.wudsn.ide.asm.compilers.test/.settings/org.eclipse.jdt.core.prefs b/com.wudsn.ide.asm.compilers.test/.settings/org.eclipse.jdt.core.prefs new file mode 100644 index 00000000..f287d53c --- /dev/null +++ b/com.wudsn.ide.asm.compilers.test/.settings/org.eclipse.jdt.core.prefs @@ -0,0 +1,7 @@ +eclipse.preferences.version=1 +org.eclipse.jdt.core.compiler.codegen.inlineJsrBytecode=enabled +org.eclipse.jdt.core.compiler.codegen.targetPlatform=1.6 +org.eclipse.jdt.core.compiler.compliance=1.6 +org.eclipse.jdt.core.compiler.problem.assertIdentifier=error +org.eclipse.jdt.core.compiler.problem.enumIdentifier=error +org.eclipse.jdt.core.compiler.source=1.6 diff --git a/com.wudsn.ide.asm.compilers.test/META-INF/MANIFEST.MF b/com.wudsn.ide.asm.compilers.test/META-INF/MANIFEST.MF new file mode 100644 index 00000000..680be15e --- /dev/null +++ b/com.wudsn.ide.asm.compilers.test/META-INF/MANIFEST.MF @@ -0,0 +1,10 @@ +Manifest-Version: 1.0 +Bundle-ManifestVersion: 2 +Bundle-Name: WUDSN IDE Assembler Test Compilers Plug-in +Bundle-SymbolicName: com.wudsn.ide.asm.compilers.test;singleton:=true +Bundle-Version: 1.7.0.qualifier +Bundle-Vendor: Peter Dell +Bundle-RequiredExecutionEnvironment: JavaSE-1.6 +Bundle-Localization: plugin +Require-Bundle: com.wudsn.ide.asm +Bundle-ActivationPolicy: lazy diff --git a/com.wudsn.ide.asm.compilers.test/bin/.gitignore b/com.wudsn.ide.asm.compilers.test/bin/.gitignore new file mode 100644 index 00000000..43e58b99 --- /dev/null +++ b/com.wudsn.ide.asm.compilers.test/bin/.gitignore @@ -0,0 +1 @@ +/com diff --git a/com.wudsn.ide.asm.compilers.test/build.properties b/com.wudsn.ide.asm.compilers.test/build.properties new file mode 100644 index 00000000..74239926 --- /dev/null +++ b/com.wudsn.ide.asm.compilers.test/build.properties @@ -0,0 +1,16 @@ +source.. = src/ +output.. = bin/ +bin.includes = .,\ + plugin.xml,\ + icons/,\ + META-INF/,\ + plugin.properties,\ + src/,\ + build.properties,\ + plugin_de_DE.properties,\ + .classpath,\ + .project +jars.compile.order = . +src.includes = bin/,\ + icons/ + diff --git a/com.wudsn.ide.asm.compilers.test/icons/editor-test-16x16.gif b/com.wudsn.ide.asm.compilers.test/icons/editor-test-16x16.gif new file mode 100644 index 0000000000000000000000000000000000000000..751df8b8e3874d6986ee86aa93f10006100d8e4c GIT binary patch literal 160 zcmZ?wbhEHb6krfwc+ABB1U6xNo#M{9r`-0+eCCt$HlXNNP|5#G*Z=?e_3J+dQ2fcl z$i=|VpaaqgGJ}D|*MP&2O>wD2@x1!r($w1*YaJJ@%lTldsl9uJ&HjyRFU76yuiq|tK?Z98aS=xv literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_acorn.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_acorn.gif new file mode 100644 index 0000000000000000000000000000000000000000..2eec3b57dcaf2337ac4f2f9b432a9616054e780b GIT binary patch literal 214 zcmV;{04e`RNk%w1VGsZi0J9GO3;Q&vSjzMT zuUIL9dl=+~MQMVYaOVsJ1Np#J_`rDt0eT2tJ_86kQgDTHhXpSb3J3`Y3<)X&4$B<3&fF>iJ{{%V~%yXjaTS)UobT&!#K$HyzdvC@>Am zO6Xi&G-GG!?A;B^k5|mtWFAvdG;L$!ietSi_iKig7$r3YH!t(dZn6nYE1k8qX5pSE zuRllx6r1Hw7E?F!Z(b76Fh9I*dU)-W`sD}OmuypstyPb!)6Z!4>0GTJn3_3t{mmz@ zZasPR^1~Owpc3JbQfZe+8OKnq#M*#{8PV-?i>EHX{`8Hgg{PKJj9=UQwB7}?w;h_d z{qVB=r}v$|zIN9ES(gyo@(E5A{dv<@6wX?`@%V+AyN+Fb`1Jd)-+%u8oilIFlxb61 z+FI7ETl4?_e+C8y2nl3R9Vq@}VPs%PV$cDZ42ly5_Nxs^Nz5%wE$!`1&786EF|lzR zob0SjOij!^F`R6CR^BqxSee_HV`JHwxzybQy%?Ao7Be!(b8a-S@wc}W+`!Dz7Q@EO zYZ4k2?W4`Tzb%#}QOVCWAtAy|i-F}Ja~!j%Sy=eB$Y3e4BW>HQ16(}pY;{9?m09m{ zG8wBnm`X}0G;QE0Y+zW+%;YoSLqT#=i@2fQ5|0aK9T?c810GEHc+i!di&bEPl)wT8 y9S)~S9+ICFQXHKbIeL^*7940~;oy@n&{!d2p1{%|6y$Ph(#6HokFYQ?SOWk!!O?;M literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_amiga.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_amiga.gif new file mode 100644 index 0000000000000000000000000000000000000000..36791b8561ab07088ca3959905c63a480d084b89 GIT binary patch literal 634 zcmZ?wbhEHb6krfwc*ejmi;MSuRqgx9GiN8K%}mP-b#Z+;Z)t5(>QxQfB}rL3DjM|E z)Zd&q`|RrV-LvM;V~|>AAG9hw^;YZTU|+u=AK$I<1@D({OY{l2l3aT;tNB3tl+~UI z+Y-tiR8HQMQ7z0ba6P_ZTX^B^-0qXmKnqOG~*#ipZAHypJ#w=^?0&CbZ;;o^~zl2K7oF*P#X9(bLzz_x1Cuubi?^!KVLcX`RcjP4{m?nz5eswji0Za{V=C@VQKEI<#XSyUie}A`k#%(L00A;Hm!bh za8H1xxx0<^^Tw)4SsBL{Em${ac96Bz)SBwawKes=J})Ltn4A#r@9g~l|9=JwfZ|UU zMh1p#1|5(Ypg3V*zuu6YZP8-Z(%#RXF{1z7% z8w*2L9!Wk83zJ2r7VdTy@<|af1`5hPEtYN0))ryj!M<*?s>$4~7A}_d;UR8;p&2@& z?3M>CTrKnx0^Izf7###T54UXf*lu}bwlarOh#Ad+ZkBc7$g)}r=Mq-c9NlCHG`5i!-jnfQ|2>d9S1|I49rZ7yCN7WtQb_d z8IJ2StWjssI0TexWEvPIEm+9P=OLIRnb_3AF;|Sk=;?vr^KACg R8d|9l1_0r{h%5jA literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_amiga_ppc.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_amiga_ppc.gif new file mode 100644 index 0000000000000000000000000000000000000000..9af7ceea13b8ac51e167010e6a28ae3d1794ec7d GIT binary patch literal 640 zcmV-`0)PESNk%w1VGsZi0OkMy-4GG*7Z_j?6ZkeaeJLt|G&O%eK(=CI@wBypIXiJU zI#W73f?Zx$KR{haNP1ynM=&xpCn$t;c4b{&WLjK*dwm=l93&znG%_?iIy_WTR7^@t zWMX7nSzCO1d~9lLH#Rp%Mn^(HLWzipf`Nhn0RaIA2mJs4{r~^||Nr~^{Cj02nY5K4fqfb{23Vk8X58>B>O2S_bVy$Dk}FnIrl$4`A0|KK|%CTPWV_> z@Ksa(Tw4BOVEScb{B3LaZEXK^bN6y^_j7Ufb8`K5cK?5V_JM)%e}9fmP5h0Fpj%s6 zGc)y&kM@<5{+XHfmzMjUp8BAl_oAWsq@(qtqxPny>!hUdtE>01u<)?2^tZP6y1Mkc zyYs-m^u@&Y%*_AQ)cD=qBOxL3=H~zZ{{R30|Ns900000000000A^8LW004XdEC2ui z01yBW000NwfP8!ygcpQ|hk}D2EiWuBE*>8o8W$IW7#}MzAbSxD3JDAl6c~mWEiocM z1Z-qWIVe~K8W5AiI70g#8r7$`PKjOb8dp&<(yY7yF?fx`z6 a95yf%bShl23`YqmGG(c$(@T5;0RTI{0~Y51 literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_amplus.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_amplus.gif new file mode 100644 index 0000000000000000000000000000000000000000..754e255f989c6dbb69a60ac704ab8f4afdd349e5 GIT binary patch literal 375 zcmZ?wbhEHb6krfwxXJ(m|NsA2QBpB6GO@9;2?+}E@$yMYN~x);$;ryuuzrKKR#`;& zqLk!y_V!mjJdOtj?(gY&R#i3Q=FRVO=FOQhZAwd9%bIm-rc7;Evo`hrf1a=36kfen zy>nOV{CSJ(H|!1^@K~@QyRWZaT1pCN5!ga%5{f@r7#SE;7<52}g8an5HpM|G(?Lh7 z^HAcVqJ<$k9Y+=_tkBY)($Z+KB8)pWY3ZSY8Rws|R+#Kiky*sfp;VZ1hPf%h!O+4= zT1rMzUP_VArCO9FqnD4Ridl@gUP?l)mo1TVN?$tbL{6r8e1d%JEII6p=Lz$1PGnQ= v6BK3?=1Np%Los)|_T3IF3WkO}lMP|>~xEpa!jttfSC+uj& literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_apple2.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_apple2.gif new file mode 100644 index 0000000000000000000000000000000000000000..9755078c9b8b424081817e7764103dd6ce1e5ce9 GIT binary patch literal 619 zcmZ?wbhEHb6krfwc*elM!^I;bC8MIGVq#=sV`UQ(6yoFMlaiEDQ&p3bm7{(^P4%dX z;yMKuQ5IGyRvvvG9z7l&B_1AOo;ma8Oqn*NrLASnx;6ZPeEQ#X^gd|ozSh!trKNRS zUG=cC(q;vP6><`d;s*cq^nd8;f78|bqNDd&NAHu4-g|A`ciOsdv~^x+={(ocd8VcP zSX29frq*2zt=k$J7u3{FsL0Nek?NF`Xb_jE6BjEM70DD92;%2;;*psoEm12jk|xZ< z!^6P9@c;jRAUSk_;!hSv1_lQP9gr75al*iUqQSv|xrM2vy}hZKQ$$QyM3jS*ot25H ziMdCZlZ}OW+BDYrEg~Z9%=Rm-xOtfw7BMo5ajsiwFKcSg(8AIt%(hP4%+gH3TyXbd z5f*z3RV8VC6-65cmc7iP%ywMr+FH^YMnVh++P3iNNol`Pli$U9gOkb5Q1gwplng%u zJIh9P7KR&*jQ6{ExOr3#q$!BCh-n7QxM9%b#>ldmN#w@W=GJye1BsX~=^6}-fh{~D f784i_Hu^ELNJ%bDpd}g N4jC&fqAU^-06UUnLQ4Pu literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_atari_st.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_atari_st.gif new file mode 100644 index 0000000000000000000000000000000000000000..2ed2102e9d99082f6e57cb82099b0a45c7962ffa GIT binary patch literal 214 zcmV;{04e`RNk%w1VGsZi0J8u9t*)&sD=d_hlz)DI8yXr+N=#K#RhOBUX=iCXIy~d! z<0K*^0RRF2|Nk^HGynhqA^8LW000jFEC2ui01yBW000Do@I6PXrE;PJ&KU!O7)z;n zNRc>YX|NPZF*NNkFW~^l-7vo&gOXS{4uzY7v0U0LtPf<#po*^J3X=iDFet&W9 z_5c6>aj*ZLasPCE{{R30A^8LW000jFEC2ui01yBW000Dh@V!8*rE(&`K}rZx9HhpH z6iF7rL7mhf%fZ0908%dl&kq$GYak#RX{`XDW-Jzwu0-w77*rbv*)5BVW`N2A~N*$jG2Im3`_; z#>vY;1}gq!0aEe|IzTqiMgax})g(vCqq=of|B`8S&mZrvN)3&5w%ogUnhFzBW}vH$oMd@Y z@S6Fp%V#xqSEd!k`1Mw&M>(4L8fxnC@hB@Pn;4m>D5=<3*)T9LOqn)CMoK0pE9XDM z|B#>%A1|Mjq?Fpsh^G(t&Y3r-B+h^Twi)Xdb*x#pW^P}3OIu4pl+VZaCnq=MK6$Vw z$XqYM-R8*dx%Y2vHx?F{-CbIr9rfh??$d{t)Kt}IsB1W>${*N1)56?hT3ex-s@kEQ zv%h{mb9mRBWBcY$sL$TCto!|&W3Qea+_`@8?W>zlA6h(rLgm?`%ig>^vUx>sO?vpt zrw8=)^j|$cw01!o(5Zt5DE?$&WMGhC&;dCK6ekSqcN=6RL|epK+S?_XM5QIwlq5CG zq~&GB#3Y(~y?OZ++)a)7nVCe}L?tCX1bH|ewHZ_e)n!GcMAfAEge-lyxVY_w1VqK# zl;mt(g*nujISiecM3$`(XSd^FV^tPp*5o%oCaNLopvB_rugS>Bp>U;5TGU!ghsBGH z(bD14B+$PDoHlN>YlCm(P@GQ)HxM zYN~45SlQ;xo5QK$+0xd+=GwMq-I`^pYXAQOl@kDpKUo+V7(^I!K zPI{D~F=??wPos>}TCKi`0uHhAwv(J16puJ{wb7e5nTqi!TRLP%vW_t1w$Wr*CK| sJC7qbd)Q3Q*SeQe?0O0Ep>DQUCw| literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_bk.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_bk.gif new file mode 100644 index 0000000000000000000000000000000000000000..7876d9d521e7db47197301e5c212e94a30a60e50 GIT binary patch literal 962 zcmYLIT}TvB6#iy+aT_(t4QCax+;A+>j8qAE~GwdI&~Ka{VEO zk`gn>ii!w&Xo+B}*Txj~+fM7s`q-Q2qc#;WxEk6q8Vc zJ3)mT5e35NOyi-SqjvVB;m~5jJ7$0x1pH2B=b<7L=x*U{K+2wtkl!ZyjTELLcCATF z1BTwjYQan}1B}T%3!@kMFx2>H+L@(9 z;K%0xTa0J{%`%<F^5yK}K97ctl?8r3! z=U_)1t_#9bRIMtVyF=-A4zoWWi&NKi32$4@PJ&rrNsB>h>vnk57w%VHqEu8H#{FCK zGtXU;6a#ma=I1^zlH|SmyQZh+DqZ3T4yl9|adQ6mqaj}dYQp9BF7$kqW9>rEZ;(TY0D@&8MJat1I3@)n(B_?v4 zq0}#r>Y3MC`|aq`)q5Jp*Stc*#?~6O<8~@rI#xG*raqIpRx-A0Wa8`Ovx}?Wwzjoe f()PiC980J>@90Ctqeb7f@WA8qDjf=i%dqh;xeO5| literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_c16.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_c16.gif new file mode 100644 index 0000000000000000000000000000000000000000..81d60af1385cd206b4b12b21fb81a6d84b9f7fa2 GIT binary patch literal 216 zcmV;}04M)PNk%w1VGsZi0J9GOX=iEt{QNXBGywnsJUToiA|zE)RZL1u?(XiFnU@V`DhdxJ S6kA0rpaK~zEvGFK5db^aDMY*g literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_coco.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_coco.gif new file mode 100644 index 0000000000000000000000000000000000000000..25fce6264718e2886265e6b3b819cfd6c921b6ee GIT binary patch literal 626 zcmZ?wbhEHb6krfwc*f0OVq~JCq{73+V`F8*z`zg^6e1%fGiBP8oUELbq!b@7pZ^U1 zdH8wsbbRN`n^RL&bA*YRndt}*mmu?DW^PU`<|E8&)~)%Vug|5y#qfZkE;gRik&}5p zvw?xZcX4rx5Sth!B__s44Dt*D0vGvobou!%$TKilTRNCZON1Dk^0{*AXdB*?kqz+l zW@KaJ;?oom_GM&zXltv$!0;6af3mO$aPbImam&j;(BkBll8Q)ANczUe$i?B*($=D` z#;M_`%*?$191pMl2Ym~33wt>Q3qxZ!9!|0AV&0wsT%4!59J#nPxH(<r<%qiMkZrJ3y!}Wrp6(PiVjCPIf0=;2cY_wsRYF*8h`&bpw5ot?9V ziDeZN1JhzQ1~CpdOEohd9|;~IUS9P!KHX4p9Sv_`8F^d50NyrsO>r9mBhe^P6;)%e z=(dnRxd?YHNl9M=MMJB|HYq1l0ecS<=Wszkix}p2vHl7U%Y1HpP&&b(sm*reR^X9S zyy6lr9~L<%da~Csh#V0*Fj4e4lZ%I>#)VbN?yRhGDhDzW4rn-O2rC4b7#RNl{|_XGC{X;#!pOkj$e;sqKPXNZ*bg^2I&!vfw6wQ3H4B=Wo0ysj2=eoC za5QoDmhLDh>X0ho(>DsI!>-O6Jc3xWd#jQ zdlnYfR!&n%RTb&e%!c{~qFe_!&6IQ%_c%Bh8EY}|Ef&20a4&<6h3$qrf>M{RurOO& zTG{b)ZIY8wNaJbcOl1*sdGPE&gKmtT&4(t|#3_wzOcF5xsfrEFYy$rzjTShFEbQl3 cYI&35FkxZ*yln?nmb?&NF?#_ED-(k?00o0|D*ylh literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_cpc.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_cpc.gif new file mode 100644 index 0000000000000000000000000000000000000000..470d5f41b5579e53e3ecb2614165fe3474895ec1 GIT binary patch literal 265 zcmV+k0rvh!Nk%w1VGsZi0K^{vGc(CFGBhM2Bs@AiY-(&&QdAon8kCllWMX7YN=$ru ze4L(~{{H}khJ>uHtREg~(9s^Cp;<*mvx0(i!^14Jv^>kqGL)2GZf=ZBOP1;C6yDz% zJUqKtSE7iCbpQYVA^8LW0018VEC2ui01yBW000Gs;P(Y&X_Ded9L{ny3V{?ttsV=J z3_)ENJ<{Qj0T_?Q2hb335C9WSMS=lDjDdk5ut)$P8pbAKVO#*D?O}jy0|weaBk*N- zDD6a$KzAPtaRdT>M+gH<2!n(G1aS!livS8i1_cj<9fn&513)AR1qcKR2s;KJH6(CZ PJgWgIJwCENA|U`f2=Q5& literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_dreamcast.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_dreamcast.gif new file mode 100644 index 0000000000000000000000000000000000000000..b4ca761a29becad0c9f948a1ad4567157c7317b5 GIT binary patch literal 288 zcmZ?wbhEHb6krfwIKsdnBPFAvq+()ZVq;|!5)|U&<&%<>Qd3ovla(`P)|{@6uCHId zPM||h_yI>h-io?9+7D~aYzE>J0Tm(9pv=xsk+%P-TZ!%NEX!;VP zWI-ktQJ1uj4JJ|oCzO;WW~j7wFBYtvaeU&W1fv}p$8W4kejpmaB5q{)=z05AVXNk%w1VGsZi0J8u93xIy`J@Y*bQItFWw0N=#&8 zWR#YabbEGxet(>voB#j-A^8LW000jFEC2ui01yBW000Di@V!8*rE;2u+-i-X973rr zj*%2WSspF5EJ^^}_ze#MVidVxp&$r25QHQFs8Awe90H(_6eMxPKm&QB5}E``fXEse%C9#$im0|$NpfC^ay1uzr|2MTWrD+3QD6hi@+m;o6r KE}t$E5db@}FF-5+ literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_dtv.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_dtv.gif new file mode 100644 index 0000000000000000000000000000000000000000..337e4f7896cb0d03d34377259a4738fa078c9ce0 GIT binary patch literal 621 zcmZ?wbhEHb6krfwc;?PvVq~JCq+(-b^Z)<vbqEh)_>Wnp1%aqr%}Lx&DMdi3b; z-@lxkoE)OY+??F3tgN?h-<~~twzZ}8)2B~QoHzk=+fV_DKUo+V7~C0jKn@1Q2?P77 z26uOc7RHwL_NHc5E=fTy1s+yT7DmP)Fx;tzf-_e+fI~Cx|M;;LQcbojZM~uS6zqc0E2=Uv%nrxz^nCUt9Ffs_~1m*<*sYW`wbrLN)F$iQF? E0LI9X;Q#;t literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_ent.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_ent.gif new file mode 100644 index 0000000000000000000000000000000000000000..bdecb692e5f54845957424b93fd742f25ac4d107 GIT binary patch literal 625 zcmZ?wbhEHb6krfwc*ezGVr0U@#iOF6Vq;~)z`*eT|9=@NnVhVgkf4w$)294q`0wN8 zlaiG3>(`Iq;2>UZUPngxj zeb1g~e9+gA;Bjr_dai4!NBJ$u^1 z+#)|O&(zrT_wQdB>FF*`&Tn47GB7Zhw_x7w+c#rlVt)VrDJ3Dv$HNyF8ygr9;Aro# zX5E_IyLWy4`bAGqS64^Z#K@SRm;c|tf1<*oX{o7ds%ohzDZoIWCs6#!!pOkD#h?SS z6ci^6?B^P|SQuKETH4!Lni$wQ-DGs-6z!ck(Sq(TC+88)Fc^pJp z1-ztLZ5WrbG5E7HFb1)zF@{Izv@o{`bMrHbD4R2$J*&BF1@l=yM-Nr;b5=(f%yuyf zvieB47@sw{xby5oUm+=@vype$na)~xDxEtU%)E`~E2Dx*fxrXDsjA#v1c^u*aTP@9xgaAK}v{gPQby2*^P`GItwZ$E<8L@MqWba#q^g~8Wl+BOxO`Iy--UfB*mgA^8LW000jFEC2ui01yBW000Dr@V!N=rE-G7hjRcBFr>zb z%|eoaN}W_uBnm+wEwc>3A{Pk4K|n|~bc5DUkvI?(2nw?COWbUTLgTn__+$zLkRT}J z7QJQARt|6h3}D!LNXP(h2GJ0-%K~^hBTiNdKW1J!3;}BgS!MwsTnr5@6ajz$AZ`o~ TBotc&1eArBXiijs)*#$9}$f;khN- z{Y2|o!SuBp>Ab@-Oh1nAKeS->dufJ8Vhq1H*Dg#ud!Wb;3d`)m7S+6jz#?Zg;S&*3+#H@Yj%%3JD01 zl9W_dQu_b@zk`E=sHmu*pdbSS1CR?OsRa~&vM@3*q%r7#3v^3V1*0z?W zW-du_2}y2lE>3oqsa=v1TpVnyQ>U`eZIP7pU^U|7S4>i1^;i(kD$d2`f+N<;R9acv)YU6)!+uFN9ywV<2d}8`7~wtpxmoQsw48kXgTmZo&a!f`+8LPp z_=N-n>q*^y#UkQjVHL^{XzLR3f>oQ5k;A*~$%2iKUAcug(*8PqXzgjyb+6c9P`Kz| zW8VIKGCz#idKy$T?G%2@I1tVlw}4}<&w>QUCg1ooAxc+PJDi`#DpdN(<;vs*4h+@+ DFZ#hE literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_gameboy.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_gameboy.gif new file mode 100644 index 0000000000000000000000000000000000000000..2bb2310501ae3460673ae14d739cdee654cee59e GIT binary patch literal 408 zcmV;J0cZY4Nk%w1VGsZi0M$PLe0qF)e>H`PI-H)I*VfsDhJdWDtkKZb!okD*`}?-F z!|dtw`t$VD(ADzp^8Nbzn3~z~SB9 z@9pj2-Q?fh-tg$_-`w8y^7KMRO~k>%)X~<5h-k90ywJ|iv9PqhyT$kS`Qg^wBO)WO zuDR{%_L!N3($Cl7-QVcv>%K;AeSLo%8XT0CluSxYWMX7&YHU=BPg;V@@23~U8J4yWt&gg)jVciZ`_ zR^69qv*COc6714A`FP@*&yahkFr`u<#Ut*F12;T83IPHF0VabiV>Ua38it51gAso| z1#dhV5HA)J2ZJ|bb%PEI6)q19gF2TvJ%b1$1Pn70gE%{KII%ngC?Ov!4m>wGmp5`e z2`U{j92;-|V>LQ`9w0FwBQ!byx<)lSVnjDOI62^5H9|%{J6`edUOq}+VEJG|K>$1V C4#B_x literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_gamepark32.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_gamepark32.gif new file mode 100644 index 0000000000000000000000000000000000000000..31a52215f18a7f1506276c2be1daae4196fd823f GIT binary patch literal 215 zcmZ?wbhEHb6krfw*v!wcV&#gnXV0c2rTBXLh6II7o;vxhj0z7IPhozcjg^gzl#C$1 zpo)^p|Ns9@j7)%};!hSv1_pix9R?r(nZdx)^1;@9_0pn6OkM6B4hBoU@pd{-YVoRM zD%Ck%XyLxvf}u&ji_IZ$v5G|Bp#zLL5>mX6&W3uJ%x_aSEc7v&``|$FMwee=i8;xR zFO($0SClDwykY5yG>}$R-oq@c*d#UmvlB_k!HqNE}uDCF+qZenC&V`UQ(6yoFMlaiED zQ&p3bm9u)@4m)edbxRN1**L4K7XV7#MK!mK$L zVq$9+FF9di>SJpcws6U9Z=b~U?722}(ZQhwWffDh3YMf~&g|^nn3T0THD_U3&g`0| z4T&jL*?G;*eq9Z{*Xz5^l{fA!YuK4lxY)%r!zZY?ciQQ?wypXuYt>CFT>>Y@XUxbf zUN~XSo#y`I>gJWT{r6J}mPqRr$s3lWm96WZd%t!1)!N<@UP<%04cioq6QuM)nM^0E z+Q!S9N0|9$b9>FQ%U`K(7tI>AO)!3!e%oW7h)Ld)k8)%#6P@*jappV5nQs}F{kNZU zj&aT3IrHXBnKq@Rt!2%+H4F?4|Ns97lGFl4W=t0KK^+ZuL` zwnPSgaZLd}NjWhQL8iq?!n#H_)~+s&2A-BuyV;Xf6-~UYyn@60-0bxh^UBJY28Bn) zM~B)uNZjBRb<&DRh>4Aha(4GoXxQp-q({PMp36d}rJO=C`WrMg6b`!e3#x~0a42MM zX$)i*+VHC&ySa&lM=U|Q!9ir2KUo+V z7^E0a(zWaHTR9*2Xh& zGKq8Yib}E-y2Z0Mh;!u$O*E^H_w-^cPBRIb&$Dp0x0s2F5Ho`?KZA{glzo(Kf#j~O OO7{B?>{n7`um%9hxmGy< literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_gg.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_gg.gif new file mode 100644 index 0000000000000000000000000000000000000000..ecfc955241f004ece9f910755ab38892ad2dada4 GIT binary patch literal 378 zcmZ?wbhEHb6krfwxXQrrpFw{WgM1u=AP*N02Rnz1l#H~bw2G37iIItom5sHfwTiMz zNKlB6mrqJkN=;QwPF7BKX10a7g|w7BL(+c+myHbDCo)teGT0i-nKx(3v?(oZEo;`T z;bdlh@!~~DNC*SaivRzCB#nULPZmZ721N!Pkoh1#F|bWP5WwMRBhc^uP+-x+6qDKP zM-E!Zv~Z-cUGh*^6LxI+ii;Lv3$Z~Bu*U?@N31gwMiYieB1y>HK zw(g!b&N>ch1$jBS-ZtK30flL8Q|2o0Oqe}wX^$))XCWWkvSln>0uy*y=g-;B%)_6| q%EQb(Zz}Jum2Rw@NiK)^1o-)4S)JT!rw* literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_gp2x.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_gp2x.gif new file mode 100644 index 0000000000000000000000000000000000000000..1594ebfe368e7e3ef1d5f012a0a0372cfce51a8f GIT binary patch literal 979 zcmZ?wbhEHb6krfwXlG#H0D|KG0vrMo5)z7viiU=UmWl$FmX=;#UO_=Y$;rvth62Ue z#Vsu@Q>IK=8ziuH?b>U}9QU$0{#P^n9|fZ!FvLSZ@h1y20|O(24g(N?@&p6N1_ovp z4TlX29GjTfjS3hJHafHMh^Qzy1U__TVdn~1GQ&aHv7J|=gGFJI6El-6yGX-<2BwDT z2A)D11qsWTS@hhe*?3-f);K}U^N>oygT}K=ObJp33qqP2XYz1OF)=u($*_t$ODw^G zL5+2lpSn&!fx}WJCI{)5T7iQa%xgK=cC_qFV7kF7$F9Qg>*~9!tD@LtUU+h($ literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_gus.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_gus.gif new file mode 100644 index 0000000000000000000000000000000000000000..33fa35aa54091afc2c4495288fef0a3930c48ccb GIT binary patch literal 309 zcmV-50m}YINk%w1VGsZi0K^{vp^3^)FpT~7|2ZXLY-((il$L5ssBK_`N;qp%PFTdR zA^8LW0018VEC2ui01yBW000HD;P**nX_Ded6qQ#r4M8PStri}U zU~`g93CV<1C{Qp7Po;5KARIx5WJzg61cHddGQdbMLyzU4)o2VB#bb~WN(zR6K-0l0 zG=TsDoM{*|5f%a+8fgXtK@%AVXLnc{000^T3w#H84-9|;3=9t%2N+Tm0|6JG0cj5q z5TQc}8wddqnPwLd1`Tdd90&s%533go4M#%`NEsMN3ls;%I}RT;B+Fl3))FZ~HQnA5 HA|U`feDh>P literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_iigs.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_iigs.gif new file mode 100644 index 0000000000000000000000000000000000000000..ec06b373dadbde9c76e6f07b7e28627245a1f18d GIT binary patch literal 407 zcmZ?wbhEHb6krfwSjqqbJX}0Lz#_^bCMG5$B_k&%r=p~ytgNi3r)OwrXkuhyV`by$ z=;-6+6A%y(5)=|19-fkvl9QE_ot<4%Ra0GET~kxj)YR0{*3#G4Hw6f$O_?)q&iwiF zmoHzwX5E_2n>TOWy7lnk!^e&tJAeNCrAwD?-n@DH_U(K3?mc|?@cHxSuU@@+^XAR_ z_wPS`{P_9v=PzHreEat8$B!R>|NaF#5FerVlZBCs!Ja_}qzL3E2DU8+?&mw&2=u!@ zEI7niYBBSa3zMZx3wK)jqXeBbVJD`ac<3S4P?BsC!E&UPHRF*VvzvrN!wG{jnLK`R zK`{w2j_PC?9tIITV;x;X6EQ}fT5d^hCN*81`TEK%j16ujY?|{I$X5x>5N2mmoi|@k zor_V3r%;HUQQBygwt)(t@C?2}PAwfhJsl${0nubufoc(LB{>bn%e&lId6HcC1cZb| Zg<@Hq+_w>~N>Yl9l+2WAQ);Se z=FFSZ($=zO-J1XZ|119I_Hzvhc6JPKHPSO+W@KPsQ2fcl%Fn>hpaWzB4HaNuV5$DF z*J1V2BC)44vb-B5oK|Z36bd*-O6e}u;9zAq%)r7hhoz6{ecR=?Oe~HKg%U#ZMV2-d zoPIj5C1ru(mB?P-r5`F)rM9p#-mGvIsN*Ot<+Bjk(k_2DBjShb(}P&+cZM$J(~U%;WHkrTn+&x@(@s zs$2%9{5PANM(5oJr3+3n9(KrkR()=Z2qS*;-#3foDSs$cGsx5WF(X~apMd#t_ZWN< V)T%;O&h1~-E6=dFB-fR}8USjHgWmuE literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_ipod.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_ipod.gif new file mode 100644 index 0000000000000000000000000000000000000000..3a1bfbcd64affce79f8b9cb55972aa7dd17701e3 GIT binary patch literal 212 zcmZ?wbhEHb6krfw*v!um;1`gYo;iKybQ2>J9xk30D^{o|sl0#x-p0yi>(;H(Qd0l_ z|G#kILVaC*(QP2}zv538Mg|6c1|0?<0GYwS()eMoYyA=_$;oDN zoL6Ga=18-K8#~WuYBcn*yy0L?G_#EJ;9@Y3bYWmnjN#(Y6X{V>WGLlV6^VA!n!_xq Mt+QB1NRYuA0FJ>(=>Px# literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_jag.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_jag.gif new file mode 100644 index 0000000000000000000000000000000000000000..46d72705ecad10d9177e7c9719d4cabbd2f5500e GIT binary patch literal 382 zcmZ?wbhEHb6krfwxXJ(m|NsC0!oYZ+k?9U2(=jIIb<8YNSy(5tu(q(WRkN~}u(9W` zv1hWer?asqvU9|2h)j*oZ@6R=^}R)mRs$E$RVEm8Kys6DuNbsCY(2-;Z{3WUvMR-56a? literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_java.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_java.gif new file mode 100644 index 0000000000000000000000000000000000000000..61be6df91ea474d9b4b17d67f798bdc4aa6131fa GIT binary patch literal 630 zcmZ?wbhEHb6krfwc*ekRwY>4h_S1jA{(gGpsf?72ijszk@}PVL_}Ys!J?6L)V|b?w~A*I&Q<&P|(Nn6)@Nb#`g~in9C_RVAw@b{%SO+B|pq z@tIQ&typ~K{Fx_>)%$uoH?LT7cF)cmFJFAFuim+K)s zhd+-T|2SvfoGH_$w6wLXS-0l@lZXG`zx~!;_wUlVe>blFe)8yrpX~EobD8KyS~0;26tRHaqsP9e(aXZac!+53<>_T~cy&_VACEW8z%Qd(Y6s zhMAGU+FXj`HZNgdwCt$V}^RJ9jE3mmQ@-Wj*Scq F)&Qi{3{U_7 literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_js.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_js.gif new file mode 100644 index 0000000000000000000000000000000000000000..9c7ed9d1705fb8aac4527b97712328d95fd32820 GIT binary patch literal 201 zcmV;)05<+00IC1{{R30A^8LW000gEEC2ui01yBW000Db&^bn{rE(@Cj2T0c6icaj z3864$S+Eq!AWhvM^NR}!V;{B#MX_KQ2(+LLM>HyezyPqJXbyb8#sN@M6?--SP+TiC z!BOBLa7!p~;Mz7wa2LJG8lo_0?pHww1blWg3NI9cA|WaW4kbe?Rg)_jEG?KV5)lA9 DV{<>H literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_linux.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_linux.gif new file mode 100644 index 0000000000000000000000000000000000000000..563d90514367f823757a242b74a497c77e9acd97 GIT binary patch literal 634 zcmZ?wbhEHb6krfwc*ekx5EswG#UsEcAS56pBPAm*E3cxYVq#=sV`UQ(6cXqk=;P(1 zp{}8-qMDMFlAM^Fla=G<>J}FpC%`W#B`NLZ>Mkd%*xNISot>MLliSwT#mg%|Lqj_x zBr-le;rH)9w{PDG4U4vO@U*mc_VNk3eec<;w_m^i{QvyLYei)Z6LZ@;ckdrMe3X}u zKRq*j&b&EOrcG&SYgw~yjhvk9zCG*yUtsxvN$}ZGgHQVfcJ(lP*~7Sgf{wql%&`^j z3n#?<-^XxgA=8xwg1488o?mAAWPilE=|OdQPCc~&dOB)HCX4sxO6;Aay>m{``sqN@`nm*avNE{$zu^vdBEg)TAY<( eQx4l9QEFUsrEpWNKn;Dk3Js#>uCks=Ru`nmP04Oqn*t z!BJ-BH1FJ0+1Xv%s>)){_L{$M*?zxl`TvH^mrKsA6}HoxEo$;CgMADw%=Aw!H+#6- zzCOoj)BO0zKr=rN)0}XL^DC|AG+W!-7?&sM&u((mR+nu~Q*O#K4)Au!iuaV3lBi42 z>dZDv3N-$|(f0plr^z|SOLGmnAvaRm&p-J_cwmw^q2~ zt-ab-xy@Mip{v?bTQw_b5jA1{#&Ew0C3#ydlr@FgB#7GxQ4~t?RF%dpVO;_KPsQ5evuA;f3!hBx#Dyn+UZaxWxMf^PM z%mrCF2QFv`x}4|{Rw@gasJwt%T-$$<2nS;mOQ)c2+y{dT$4-hGWOKwsG%P&KE@@Z7 zkomxYkHvCDfNO(;)6bxHT8vpSO$d(fpD2#IwVFA5bzwn-~%KHEC3jpiF%D$EEw7XrQoRB z9Uk{D2vtP>fC?N-aRUhr4~dEo4GRMdO@w+MKP0O%F#(VtHCZ}v1-u0*KsCa{2_hi?I|zbPb^rhX literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_macb.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_macb.gif new file mode 100644 index 0000000000000000000000000000000000000000..6a03865734e81603ad96b5fb56823d3748e1bb4f GIT binary patch literal 439 zcmZ?wbhEHb6krfwxXQrLbM0RB(R;;5@2y?CHF@j3T}82)&Cj(XD4SzB^N7Z?^R6R8Wa?y7`0R}XpWbcm!j7c zE2khurzwg~J&IO9)M902WoYGOXlSUZX{e~ED5FN zAjF^pG6v))297HXsvZ)K8x}Y=F|qp@7#wI|WaU-!*%7d4kyE>{NsEMHlhaXdWiNpv z9s#ZF9MT~@0gRuHbTA6LEpfQG@GvvCT95#b;-v;QAsGn~i-bf+7C|lNJ_CV6PD~5| ztTGM^j;Y3ubpZwn4R8qn00kWq0T4}&0+S7fmKT{q7fl?94;2Lu qOBrVZ7D<1z7Zx#f0uVqXwK6d}0v|P5J9Grb1Svr^%*_TOApko^;#qzG literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_mastersystem.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_mastersystem.gif new file mode 100644 index 0000000000000000000000000000000000000000..37af7cf0214f55814972fbaf10a8d7697bb9726f GIT binary patch literal 273 zcmZ?wbhEHb6krfwI3mnoVr26F|9=%Fl}VE(@o@3jSlQ%c<%9%<_;~rGB&AH5Hbq8C zrnRNDrmAMnyg5&wK3%hJO-ozL%o#I*x)_K6ia%Kx85o2abU$T}QY)okh@hqIIzMRb&wLBbbIUK&NW!}7w-0Up* oEb}VZc{`%GSshe3*g4r2S#t}UM%ZLCY+=x}wcBNDn(sj8VXZ%#{F%Yn=P*Q{GJbH+@dE(Rii;!hSv1_ogU9gtd(oeZpf4@?+SGABLC z(3rT`rgx%@(^{>*MiGZtdD}@&6B#_1PN*>MU~u7Ka7%Dl%I3k!c%YHNXR_C%ut0^H z~@9PGA?N@~nSd+qiyYdM-7 LJ!USZ$Y2cs(pXNo literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_msdos.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_msdos.gif new file mode 100644 index 0000000000000000000000000000000000000000..879bc89028cac80f52450cf396632affd2c40446 GIT binary patch literal 276 zcmZ?wbhEHb6krfwIKsfd!^I;bC8MIGVq#=sV`ZbPq#P0y;^XC`p{}8(s+N+JqN}4@ zQ&p3bm1ALUF=yVKDbuF3w6&~Rx90HS!v_u=*sx*4lqpjh8X6oN91IK$l$4Yh7#RNl z{|_V~K=CIFBLjmBgARxbvXg;zqQY_BB@XkJTbNDw;vjfgqfMcM`L_wX;SICH{Va_d z9u2-i1`Hg`%$|!^vP7&^X?j_BAV6cr>myMTP8YHV3ZJT?sX=@c+Rrx zVR@WIyi9SK;S!?UECF0h<=)}!l11zR;gkIMoEh1CW-_p|@o*I}vN)+M=Hgvy%c!Kr U%wo57iygC;y~ADyB}E2n09!s%Bme*a literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_msx.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_msx.gif new file mode 100644 index 0000000000000000000000000000000000000000..01ca621b25f6f5c205c7a9e9e837d944b3d60414 GIT binary patch literal 169 zcmZ?wbhEHb6krfw*v!UYY;4THz>t=f_W%F?nKNf{amoH?_CMz{a2h5(k;Vz!G+W#bNbQM<&NA@dXXOGKz7tzp6UkR(i`hBk%310|GCOJa}(0 z-%s3vgYRJMg3Fg8T@I&x)M$QJ!}4?i*M^%ZQ<}DiGH^4o7Ov&-3w2uH#qE0D=fyN1 H76xkoa#l6I literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_msx2.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_msx2.gif new file mode 100644 index 0000000000000000000000000000000000000000..38a9d0d8715ce2a18c8dbe017e191e6065e19f21 GIT binary patch literal 171 zcmZ?wbhEHb6krfw*v!sgY;4THz>t=f_W%EXW8*V3XU;S>p822Q|CuvqfZ~`y@h1x- z0|PsQ4oC`Q1_MjP2Ns9bOC6a4tHc*H_{J#4&HgIsa$Cx+fGI8E{thl4#(16s^Y~6Y zW))yMvO)aCjS8{AkCi^>me?jVTJB&fNVwVhR7*N-Hp3bpi^aiH1y&?E-!vB9tt!mO GU=08%MKv)1 literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_msx2p.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_msx2p.gif new file mode 100644 index 0000000000000000000000000000000000000000..eb5e009941520289c1634ab97c31f083d5dad6c0 GIT binary patch literal 171 zcmZ?wbhEHb6krfw*v!sgY;4THz>t=f_W%F?($ZHmXU?psIQF06|CuvqfZ~`y@h1x- z0|PsQ4oC`Q1_MjP2Ns9bOC6a4tHc*H_{J#4&HgIsa$Cx+fGI8E{thl4#(16s^Y~6Y zW))yMvO)aCjS8{AkCi^>me?jVTJB&fNVwVhR7*N-Hp3bpi^aiH1y&?E-!vB9tt!mO GU=0B1pf-vC literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_msxt.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_msxt.gif new file mode 100644 index 0000000000000000000000000000000000000000..420e76f4a28bc2f4e824ff9aa51dc2147d696e49 GIT binary patch literal 171 zcmZ?wbhEHb6krfw*v!FTY;4THz>t=f_W%F?GyTRhfuO+n%$YO)8U6#sF@fSw7DfgJ z4h9{N6vzw)mIwh>ht*3RnF6cC7c}_BD8|kHD(P}t%B_GYE#dwSE*{2so&)pvPCRDi zYdX9^{Kbt5vA~a&KIfL$CNx^^U@Azs+4@vVI&C(?8Xt?r!BYiRBst$S7T&EY%*bF3 E08BkKY5)KL literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_n64.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_n64.gif new file mode 100644 index 0000000000000000000000000000000000000000..761f3c1757444de87e49812b08a13c897b16854f GIT binary patch literal 640 zcmZ?wbhEHb6krfwc*ekRx~6RhyG=V=ga)eyCp#xMCpQll4;w2RI~zMU7q^U*jGT;| zl7fu~3@j2%k}nyoj;JgAXOLdapmK;I`V(8u zXSUi!*026Ctl!VDd>xB}6Z78>d^g)3DUcumI%f7svp)H=_Og+P9Cx)HndH;AB zG#MHH|NqZG98mnp!pOi-#GnIG4vG^7_PY&5Mf@$iE$!`1%@QGD!6BjI5~4!9yiNQ) z!4e`tsaY00yh8kK{2?J?0)f5>3`Q*6?2`Ok{9zKwnXysfhO&_<=^27;!6Gr)1-VIS zi9XTNGE0{WdU)sM#OJHq*lOq<&=J66c1iTMu!y~f(OG|+5Ag73Y=Kn`Yf2^W(c@DHa4@c2{3H@xX_`IQ+$QygoFo; xjT1O{0wxqJY_eb#^bnjC$>{9M#mhRgOW+}6;9^&K4XxCbD=&K&F)=b&0|4tQv2y?b literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_neogeopocket.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_neogeopocket.gif new file mode 100644 index 0000000000000000000000000000000000000000..49425f5c32f6ebc459657c3c9a2ae60815e5f230 GIT binary patch literal 215 zcmV;|04V=QNk%w1VGsZi0J9GOlL;Fr3J=%`BSH!iR|*t=et(pflx%8jOiD}|85uk} zJnso53yrqD<>o<#K+4gB`HN(N@~iqDK%9! zbLP!iZLH42#naN(vS!_yHOA`y|NjT-A_ge_WMO1r5Ms~)DF)fez}m7vg)t>_(xV6s zfyFl7jS|jlwR{Q%9Ac$yCpk4Lu=cq(FwbCc;9_Vjxy3PogPGy6z}%pw4J8KrZZ$SYYw<9MTUv2?=e?ufEi-XE^ b4pwdsYfd54aGOkqRScT8b{p)J6d9}mj-)>F literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_ngc.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_ngc.gif new file mode 100644 index 0000000000000000000000000000000000000000..1449e44296cd7b529e25becb028bf11abd24c1cc GIT binary patch literal 635 zcmZ?wbhEHb6krfwc*elM!^I;bC8MIGqM@c?Y-ns`U}R=uW@2PwV`XD)X&n+25*!#D z6cFU=?d#*^laiF8qped@Rg;sIla-NWWodQw^3}ZDJP{#L8!KCDOB**w&xhBa>uBoT zz500B!c`eDMe54h3bM*-%96LSLos`NV%_nI{+3U%9 zBsfe-cbl8w+MDA)uQY6ht4WffnoDiQLJz%a6UDGt*Df2qFhg}`NfD)~9qVRISmSF} zXQLBe7QQ9Yxh*xQ+d$jO-zU`9sv_K}%~;*9AaX^BbB&&=ld7DljJQTq;i_=IYy(~M zmV%v{%9dfCb6yo`QF?U5-RS9WZ?)7O4z&b&EOrcG&S zYgw~y&Hw-Z8HfOiKUo+V7>XHmKx#p8!oYr`p}3g6g{h^zy{TEKpeVnfP*8}Ui;1a; zy(eFYk5gZ78WR_L8+$1Y|g@LS8VsE^*T2Rn-t+um%7D3c+*$ literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_oric.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_oric.gif new file mode 100644 index 0000000000000000000000000000000000000000..14e7c040e5883ee48d12eec9b11c0f0c7df436ef GIT binary patch literal 280 zcmZ?wbhEHb6krfwI3mhmVr0U@#iOF6Vq<0VpMl~3|NpJ5Y&lsuAweOB4juCG@<~Zb znKEsPw3O8UhK8D|nmP04w6wLXS-0jvV&Y6mDWG}=yg>0M3nK%AD1#12706Bo)~N}h zj47Fu9#t3!9dS9oQNd}g(S?OP4zbF%Ld_c$So_=^7;Z3l@SRFtG{JzygPD5QhAnyFdv-A!LzBMWBUTL8s=Et5GgaC+Z+Y#Ozv5H@vQFL%)Z<$rNVr|Y;0WY z1zfx$EOX}0;ox`V#Mlcv6U|Es8^;_Ug$bLP%J zbol7peIJTSD;rndsh@LT_KrK=VaW_q7OdPt-nCmRX6*Hg$@a8yUD=B{Qdub4Nc9t3l{wO|KHuiqo=ob>yB-5 z@^W+L&6zT7N=sYInssX?PoCJ;*5d2yQ&m;j)L19L&zqT%mY177ZR+HuOBMwM2DrI6 zySunJ+S^A(MA+KcSXo;B|NsBSjT@&-rm&AY42!f@4&&y z&dSKx#N1=Y$;QGwZ5r$R7JGa4#02kFppk`xuY^hAgn)tnojReQFPu{X1Dco^8LR=ismN6T literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_pce.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_pce.gif new file mode 100644 index 0000000000000000000000000000000000000000..c8b61d2753f6d4ea3928e36c439ecd3968c033fa GIT binary patch literal 268 zcmZ?wbhEHb6krfwXklb9F*5o8|G$co%A`q?c({0MtZZ_!azcVae7t;8l2WEjn<676 z)7sKnQ&lr(-khgTpRQTArlqX~sD((N_>+Z^fkB=@hXDvcb~3PbKKRazw@W4r z!dhv$YaQUkL7UkD4kQeWc#YvuI0p$}OezTne+DlU1%(G8DgzHCLn{WAmH`vpeem literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_pocketpc.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_pocketpc.gif new file mode 100644 index 0000000000000000000000000000000000000000..8983388ab1d90381eaad811188210aec553b84a7 GIT binary patch literal 301 zcmV+|0n+|QNk%w1VGsZi0K^{vG%_?KA|#ufn+sb1JUTp-mXvI2Z2$lNprD{sQdES6 zgiK0Ia&mHpuKt38f{u=k5MKNNR{j|n7-V8(d3kxv%*+v6`+RzQtgftxh=^TWT|a&N zN`?IdSpNuC|NsC0A^8LW0018VEC2ui01yBW000H5;P(M!X_De-6qrybmoX(ltrmi@ z5I|igJyN0IKpA{__ ze*gpw3^QlWbaZgsv}w~!1_nn4#?H>pNGa*gj?U-LpPMl-zGPtV_4eiG=YRkHy|bjW ztE_x;Q?sIiVt-$MY;3Fx1H+j!XYB3le={(INJ$$qFr)!R7#Oq}81(e?7$^XWKUo+V z7<7OT2ta;fU|a4W#_gaZ)pM+Gmn7<^Rm6=JfR#rw^nvaROjDahSmqUa{K$4q>sj@tWX{wrl zfavV>i7brm5=ts^+}wRE%$cj0#I+Q~HY)S8Zz*TzTrID_!^79aHo2UWtt2Ix{melI fPUbk5D}e!#oH`+;Q4<)RGI&PEypC~lWUvMR>UUlb literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_processing.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_processing.gif new file mode 100644 index 0000000000000000000000000000000000000000..1e09d3038f85124ea0b1306ad6b5be1f4f998c1a GIT binary patch literal 194 zcmZ?wbhEHb6krfwc+A0|q^M+KWMX4w6X+k1nVwl!UpIU1?EnA&t0-||MT$RJ7#SEi z7<3qb0AvOOOTNHDht*4qgbw>^vKTTfIo9dqFsa$A^mro6M@JT20f`ow%Tt(L7Z~%T zuq_nu(BztAv@F1ZM?^1Z&g*NNFD7kwoXW7_>kN@YJ7j{QuIvi@zU@eCX@l4`zHrVj b7R-tYNemn+&YB^N-Hei2+7q>f1R1OWa?&~a literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_ps1.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_ps1.gif new file mode 100644 index 0000000000000000000000000000000000000000..ff6d5a12cb8a627313e6b33b99fe1ba196c31210 GIT binary patch literal 634 zcmZ?wbhEHb6krfwc*ejmNkgkeO+8OV^{TD2i=wKPqN=UA#Z+_uKvg3iE*^O)c^N4g z6(toDBNH1dn~|sA`KT>S`ERYG*Fj%UP$Ot|us~A*-TgXzipEU&ALUFD$DhBCoFX-ZfQs@_Z>LH^bx%mCAaff>coj**Wv(Oqn*NrLASnx;6C^(>C|HeOT$Ts?vOJ zfmwZ=;k>1y}rX#fBJpMhMU_>+Z^fgy)M2V?*! zP8is)H{|5-w(zvHw>LEldwP3#dI<>&^7HUC@%DHK3-Iwyo5nxC#nV%;MI|{So7GlL zmY18?TbNhRP$D5VE1JbdjIYf@Kv%;;GCiItios4~u_vF7yf{}}CSzo(HQzyAFJ66f zdn@J`4t7x^fyG+`eViPU(oD3~mH6)o^Z5G(1Vx+>&=gwY#=^cy$)F*y_0$xD@L3X) zMJY|@9NZy2GZK~C8MS2$GA0;g9c*UdHEZ}VVUbG*yAYpL)B^_*MGk%?pN^ZEjS9}Z VyI8-fY-wENzhF|6kO&8ZH2`~9r%?a^ literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_ps2.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_ps2.gif new file mode 100644 index 0000000000000000000000000000000000000000..d7d017d8a6c7d9e431438d5240d8a0511fa11b11 GIT binary patch literal 285 zcmZ?wbhEHb6krfwIKsfd!^I;bC8MIGVq#=sV`UQ(6yoFMlaiEDQ&p3bm1E&iT{!Vu zRL}dMrW>yLn-Ug$R&$Ouj;v1G{4MdoHx-}Q*t6fFu6@_duAVb*&Xj3WTH0FHtXspt z!0`Y7e;^42ia%Kx85m?4bU+-CoeZqA60Y+uahSK$!Qdxkz`()W?7L_sOT=20CeGx93dT71!_iVH95UyaHw36$WqIphJmChT5XZed z2|x94VL=9AFV-SXCQ$|vF)?w5J|T9N0CuJ@uh1|y&LXyeaPKfy4rfMIpG6F8OQ)zY gvN)-%W#{Bzw`EjPV=mlhx1U+d-r<;ok|KjO0H%#s^Z)<= literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_ps3.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_ps3.gif new file mode 100644 index 0000000000000000000000000000000000000000..234803b7dd2390001af11aae3009b260472d08fc GIT binary patch literal 289 zcmZ?wbhEHb6krfwIKlt||NsBz;o^~zl2K7oF)=c+v9bvX3i0*!_3`paNlK}ys>#X9 zNlZvwzi$1Ud2^;ro6^$OvS!_y*RNk+x^!vJo;}m1O>1s$wzs#pu&@vl6JudvVP+Z^fkB=@2V@h-P6pO_559}4IL=#cVKm{3!^J}y?2a9*zYRJAZ)`i< z&(f&j!Q>DWAi%-S%DH%@NW@x|=9d#URG8m8cJz{lr~(_q@q;TpcbHYn-rxIy`J@Y*bQItFWw0N=#&8 zWR#YabbEGxet(>voB#j-A^8LW000jFEC2ui01yBW000Dc@V!8*rE;2u+-i-X973rr zj*%2WSspF5ED8nOcL2acfEa~=a5yX!1OZ24KxjIhJ>()tBoYS4KqFAKMgs=!bBskA z1kA3`*i)ddI9vjVT&Cl4C4uU3Rze%6oCo`4GJp*462d^!fk9t;u2hNf%`6{;$3`o5I z94v(5PZmZ727U$|kb^*R!oYsCfuEJ3rM0bvwTXe5Tb-L*RDhY8gK=7O4>vO(>)g4F z91M&MrnY(}f`URa_Rd19T%CMh=*v$~2dxJ?*-s7&Bgm^b2tJ4{!yX*=YJ}g{sS>3&OHT{jPN~9bT zrZBV44T;FOaKNdNlS83{X#vB*mUfYOff@o6A84?$I58GHTFBYQI7#M#c0#8BBa>>J T0K*5NiAvrU#c@>v91PX~NPlz= literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_sgi.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_sgi.gif new file mode 100644 index 0000000000000000000000000000000000000000..bcae67229687828e350740cbdb1f524bbb07c060 GIT binary patch literal 228 zcmZ?wbhEHb6krfwc+ANF1Uy_kN{UJ*MkY2^Hi7;DndzBz^>wr7&i;=LDE^&0(%@_JoGH>F{b5>g=D{oh zhPvlDN(l=c(X@iKxOUA^8LW0018VEC2ui01yBW000G>;P(M!X_Df|K$!(;6!R=btwI*# zBtTs+JrV@+AQg^BKmj-aWEjfGg&;tZ5(a?9p%f6l2x}0MI3lLhMhBC?0yN42BJynn z?%XU?QmIIe7NJ|ZCiJ79)V>Hq)$ literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_solaris.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_solaris.gif new file mode 100644 index 0000000000000000000000000000000000000000..b0cc1c6ac636afe0f008933d96375d6c413845e6 GIT binary patch literal 649 zcmZ?wbhEHb6krfwc*el+9|RbH@IM2H17rayAXZUQ5fK(KF*1>nma(z2k(ZNKQC3lq zR|p9T@$vHU^6=77*HBkeS5s9}RZ&ezN{Nq+H#Rah&^OT4(ap)qv9q=F_w)Dl@pW-_ zsjsUqEh&|dkX*fC&7662rc9gC+}bQDAt5Cxsivx`t)=zl%a@NIKfZkV^6As3ckkZ4 zaN)w1EnAi^Up{Z%yy?@YPnCo{tD~c%t*x!Gv9Y$cwx*`0s;a8Aw6wUm zI3pt?EiElMIXN*gF(M)&ARxfs-`~&A&)3)2$H&Lh)6>n(&Dq)6$;rvV!NK0%-qzOE z+S=O6%F5Ex(%jtK%*@Qt&`?iLPghr0M@L6pU0qdGRZ&qPS zE|!*(N=Zt&a3CNjD<>o<+| zz#zq-0}=$;$-p{!!LrDd%t?=auy6><9KP%jx|Zz-b3jeX2veD>a#uKv8N9M7|W(Y7l@(6J8EnLJmyDN@I#6g8YfR~?{*P2JlR7J#QyVy=K NOtCtq(EfGjMFiFB?Wo35(OKRqZ zDboX7W7&`1W;rFq!E-cWK>~*cyAy+=1fQs-x@)I}#oPv_2YMUy>C(w(B72h_-KQ-02@zAxHESg6R%&RQ-lF88!Kn3zdS>=7caZM2ooD8_f#!j VepL_cSjOdyk~+F;b%g{OtO1GKLC^pI literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_ti8x.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_ti8x.gif new file mode 100644 index 0000000000000000000000000000000000000000..119d40c125f75cc7deb52368248763980c1702f9 GIT binary patch literal 277 zcmZ?wbhEHb6krfwI3mtqVr26F|9>7X9u*}O8!MZftelXb5Fam}l%$j?)26hww#i7z z)Kt~XnK!4krFG4^HHQuznlgDxOIyp5B}*d0BWBHpa~2(fZ|UUMg|6P1|5(* z$W8{l+Xn8WtTL zo0OE0nwpWDTTxrf!OO$I!mOGTs8<^0(30uiR~$0CA##38(#pQXSlfMR+uX+|HvPPpxS~60Wl!Vr)g9+HO}w#t?&IUH{(og)VPfUyU|?Zf zzHY{*O)EN@o7dpWBCi>bVvjuwN0jHw`JG9L?rnu?A#gX~fUu8Dli z4ANpcI)W;j7`pf(_9T~UynOt; XoV%Rd%Hk$Gckzl(c$eVh$Y2cs=J9*s literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_vb.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_vb.gif new file mode 100644 index 0000000000000000000000000000000000000000..80a7b256ae54d4cf738a2cffa16ed777a9de6c1a GIT binary patch literal 221 zcmV<303!cKNk%w1VGsZi0J9GOg$5~1N=zIY90>>sWMX7gQdA5I3~Xv_mzkGs41t^k zB0M@gBqAi}04y{zH2?qqA^8LW000jFEC2ui01yBW000Dv@I6PXrE(s``8h= zxd4E-kUcOGY5+IvB`k0R#-KScG}1f=SpfnA3_X58S7k*41$_pQk_I4I0t+k@0R;pf X1SbLyBNSK;rKJuTDJrZg5)lA9i6KLM literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_vectrex.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_vectrex.gif new file mode 100644 index 0000000000000000000000000000000000000000..9d75b2160f6dae099f068314ee56579cebec0301 GIT binary patch literal 209 zcmV;?051PWNk%w1VGsZi0J9GO3xIy`J@Y*bQItFWw0N=#&8 zWR#YabbEGxet(>voB#j-A^8LW000jFEC2ui01yBW000Dj@V!8*rE;2u+-i-X973rr zj*%2WSspF5ED8nOcL2acfEYTQkWvr?9EqS1M=>r641!eJlNO4BhAv7CNC8PB@D@dQ z3AY2*Ai#0aH$|bRXuIKH12)lH;tpCy0|x*~Hwl1pf(0-XhzbS`3M&H-B}6P&0h$3B LEiRxg5)lA9_Sis! literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_vic20.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_vic20.gif new file mode 100644 index 0000000000000000000000000000000000000000..4b6899289ce99d33bd17561242aab73891938e0c GIT binary patch literal 287 zcmV+)0pR{eNk%w1VGsZi0K^{vmzkF|GBf}H05vxVBqAg{Iy_WTRBUQ&OiE1t{s3fR zWE&b9{r~^~kn?7Yt-1$N z1a)ojB$t9DC>8^f-y%>s1O$=EBQZDz3WNXv5r}Z6R3j-1CrE;bt&KX0J7)z;n zNue}lX|NR9AWY35^NR~fV>1DQaHUu@4g~>2DIhqUMB_o>08|~ptfJT$Gz3(I(!n6B zjJ)Hf1+Mq&7xG4DeKiOM16+kGKL`befHMj&6pbPwDhLlH6k82j Oo+}wFEu$?G5db^He?n0J literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_wild.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_wild.gif new file mode 100644 index 0000000000000000000000000000000000000000..9bbbcd6083250c4e7a3e83adb783b9cbf79e7d9b GIT binary patch literal 275 zcmZ?wbhEHb6krfwIKsfd!^I;bC8MIGVq#=sV`UQ(6yoFMlaiEDQ&p3bl{07FoGH_$ zw6wLXS-0lyy}PepzfMRKXKxKjj%4F3X7guA3K#I@ raAssppYFxR%F(69$l|23n4NQ}Eu)eebI}&Nt;|~X4tpGw6d9}mfelK$ literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_win.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_win.gif new file mode 100644 index 0000000000000000000000000000000000000000..bf216199ca80c8dad339d936647c0668c6377196 GIT binary patch literal 286 zcmV+(0pb2fNk%w1VGsZi0K^{vo1L2s3Jexq+x4a@QBP2ypG!)p@MQK6X=iC7AtL_M zH0-Zu=T0azGBp3xR6)8JPiBfdIz0a!1g3#oet&+8`4^hh|8!tGUxrIyNJ3y>cfr{6 zvC5C3-c-lDjsO4uA^8LW0018VEC2ui01yBW000G>;P*9TX_DfI04zXJ2tp*Z%F0ng z(}YgmjfrEC2?QE}UZMd&=mri31EZrfB1=FoGf+q-k3)q+%0>c{#X?dDS11u|!x}Jb z3>wU%(upV=P6-AU0vZDa1s+Lq2ni4t1_BdZ3W^AfjuaXK16>XT1eZ%QNJbZ39ti|i kOR7gku%Iw<34SG@F)##JA2cL!S2@59DLOmHJ0c+fJGg^mLjV8( literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_wonderswan.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_wonderswan.gif new file mode 100644 index 0000000000000000000000000000000000000000..d71dad8f6a58b0123c91da685c87766d686e965d GIT binary patch literal 214 zcmZ?wbhEHb6krfw*v!w6#UMFr?rbIohCu(ovy29r>6uFz)r}ZycQKIIcvR% zP@AK0ehas_k%LJu;|WeTO#`4}kCu1tl140SPZ=**2#E0{9-0^-Bi_{fRbhjuP?IZj zfx{P`?#6F>6nZ#MWGD75b1UayW2#nFwrprjh|za7>1$EpQ5IolR#YhB;@1(0bkv$R OOGszQ(j`KI4AuZ`^*vnx literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_xbox.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_xbox.gif new file mode 100644 index 0000000000000000000000000000000000000000..99cf3abd37578462f58bbf95249f2b6f79e100ff GIT binary patch literal 405 zcmZ?wbhEHb6krfwxXQr5!^I;bC8MIGVq#=sV`UQ(6yoFMlaiEDQ&p3bmBYZn#2~=T za!!+ZjyzMc2*Wv5MoVr+O%8@LvJ5AM8C01WwlgqHW?;x+U`S?Q2w`CGXJ9a6VBln6 zU}s=pWSBE=&Xj3WTH0FHtXspt!0`Y7e;`R+p!k!8k%2*lK?h_k$WIJxn;e8P9Yh5B z-5(k-3e`F&x;zRH;Amlfrk<$4v*y~-=^RYC3UaMU8_P1dnh!WkkucShOC$lnlH(w1;4(nV_ zCI(&}zJ)csT`XNRz~B(zk>s4q#<+Lkv4vcG9LbEV3=1wYu(5LNcVlEJaJk3M Y`5>0j$*m!7&U+WH_=Il>PL2%L09JlYUH||9 literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_xbox360.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_xbox360.gif new file mode 100644 index 0000000000000000000000000000000000000000..9360895d04cb2dbf436ea925d79f05c776923218 GIT binary patch literal 1041 zcmZ?wbhEHb6krfw_|5(*^hQ8k`0aaLY_nWJ++U`QOV zpln)J85f^~h@^6TQ%6H%v!{Qgy|agzm0eCzg|~ljaCmeQxSHvnMU(V5-%i;7 zB6I!uTmv)Z1?!q8&8aJ!Y_|DE%)Te-?cH_zUb!9p5U}oW$MjzfL@THU01ZJHP%< z{q=v|_g`IKzOMTJ|MCC-AOHXVym(RV&mWJsY+AZ<#ZoJC6C(pX87WD9U0^sHE?K;| zudfdn1>mR{CFmUjia%Kx85o)vbU+q^@&p6NI|f!xD~AmX2bBB;;gJJ4Auah1{=x% literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_zx.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_zx.gif new file mode 100644 index 0000000000000000000000000000000000000000..9d54103af75e10f4ad9454eaa0511f8b61010cc4 GIT binary patch literal 394 zcmZ?wbhEHb6krfwxXQp#z`~lMroqF-BO@iFqNHMCWMX4w6A~2Sf*lu}bwlarOh z(D0ujVLx-yAy%I$8ZMy>4f`1y>KPgsn3I@Qyj14Qn=@tFl$N%ZHS5+mOk!wQ!eHyo z{{I9+e*wdx8O;BuG5k+qIONMb#fJ4i1H*m>hI$4N`2QcsqXJO;$->CMpuwO6G7ID< z2Dar6BC8x!1p3_{78IV0(33c-RHz`(!n!QA(LiL)wZqdZcB*tNn{?vvN{%DVPHBMx zbG%dn+?^Xmc-+J|O4U>pWu=9M_+44W+0>aTs-%Pk`Pdp*C0g1#x_Ee--Ac+UtEchg z&*EljZSNFX$i&H3$jQR6e5&9o4(?eT%#6zN8#l9aB{Q=#l`>sm;yAh0jhQXUXJoE#af0gF6i2mk;8 literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/icons/os/k_zx81.gif b/com.wudsn.ide.asm.compilers.test/icons/os/k_zx81.gif new file mode 100644 index 0000000000000000000000000000000000000000..b6a56118dbb14992ef160d8d3b0e4d8f9573c29b GIT binary patch literal 396 zcmZ?wbhEHbT6XOh|?jVSKPj~rEYIq)$z zw?0l_ImA4fO^884U}@;{gBBWGDuV2c%pCIS8ln=sF04vi42 z(n3wFQtV|FRnw&N>!n!dYHM~aWEAIO7GGbXHC1x8pj5peGrOR&;%1?IIag+3Moz{{ lmjs37wyoz%WaqvrE_FSI$;oAAUFYJva& literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers.test/plugin.properties b/com.wudsn.ide.asm.compilers.test/plugin.properties new file mode 100644 index 00000000..51dfa568 --- /dev/null +++ b/com.wudsn.ide.asm.compilers.test/plugin.properties @@ -0,0 +1,10 @@ +com.wudsn.ide.asm.preferences.test.AssemblerPreferencesTestCompilersPage.name=Test Compilers + +com.wudsn.ide.asm.compiler.test.TestCompiler.name=TEST + +com.wudsn.ide.asm.editor.test.TestEditor.name=Test Editor + +com.wudsn.ide.asm.editor.test.TestSourceFile.name=Test Source File + +com.wudsn.ide.asm.runner.test.runner1.name=First Application +com.wudsn.ide.asm.runner.test.runner2.name=Second Application diff --git a/com.wudsn.ide.asm.compilers.test/plugin.xml b/com.wudsn.ide.asm.compilers.test/plugin.xml new file mode 100644 index 00000000..d8947fd6 --- /dev/null +++ b/com.wudsn.ide.asm.compilers.test/plugin.xml @@ -0,0 +1,64 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/com.wudsn.ide.asm.compilers.test/plugin_de_DE.properties b/com.wudsn.ide.asm.compilers.test/plugin_de_DE.properties new file mode 100644 index 00000000..eaf6a521 --- /dev/null +++ b/com.wudsn.ide.asm.compilers.test/plugin_de_DE.properties @@ -0,0 +1,10 @@ +com.wudsn.ide.asm.preferences.test.AssemblerPreferencesTestCompilersPage.name=Test Kompiler + +com.wudsn.ide.asm.compiler.test.TestCompiler.name=TEST + +com.wudsn.ide.asm.editor.test.TestEditor.name=Test Editor + +com.wudsn.ide.asm.editor.test.TestSourceFile.name=Test Quell-Datei + +com.wudsn.ide.asm.runner.test.runner1.name=Erste Anwendung +com.wudsn.ide.asm.runner.test.runner2.name=Zweite Anwendung diff --git a/com.wudsn.ide.asm.compilers.test/src/com/wudsn/ide/asm/compiler/test/TestCompiler.java b/com.wudsn.ide.asm.compilers.test/src/com/wudsn/ide/asm/compiler/test/TestCompiler.java new file mode 100644 index 00000000..dc70e2f3 --- /dev/null +++ b/com.wudsn.ide.asm.compilers.test/src/com/wudsn/ide/asm/compiler/test/TestCompiler.java @@ -0,0 +1,50 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.test; + +import com.wudsn.ide.asm.compiler.Compiler; +import com.wudsn.ide.asm.compiler.CompilerProcessLogParser; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; + +/** + * Compiler class for TEST. + * + * @author Peter Dell + */ +public final class TestCompiler extends Compiler { + + /** + * Creates a new instance. + */ + public TestCompiler() { + } + + @Override + public CompilerSourceParser createSourceParser() { + return new TestCompilerSourceParser(); + } + + @Override + public CompilerProcessLogParser createLogParser() { + + return new TestCompilerProcessLogParser(); + } + +} diff --git a/com.wudsn.ide.asm.compilers.test/src/com/wudsn/ide/asm/compiler/test/TestCompiler.xml b/com.wudsn.ide.asm.compilers.test/src/com/wudsn/ide/asm/compiler/test/TestCompiler.xml new file mode 100644 index 00000000..a86fd45e --- /dev/null +++ b/com.wudsn.ide.asm.compilers.test/src/com/wudsn/ide/asm/compiler/test/TestCompiler.xml @@ -0,0 +1,362 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + /> + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers.test/src/com/wudsn/ide/asm/compiler/test/TestCompilerProcessLogParser.java b/com.wudsn.ide.asm.compilers.test/src/com/wudsn/ide/asm/compiler/test/TestCompilerProcessLogParser.java new file mode 100644 index 00000000..3dc665f1 --- /dev/null +++ b/com.wudsn.ide.asm.compilers.test/src/com/wudsn/ide/asm/compiler/test/TestCompilerProcessLogParser.java @@ -0,0 +1,174 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.test; + +import java.util.List; +import java.util.StringTokenizer; +import java.util.regex.Matcher; +import java.util.regex.Pattern; + +import org.eclipse.core.resources.IMarker; + +import com.wudsn.ide.asm.compiler.CompilerProcessLogParser; +import com.wudsn.ide.asm.compiler.CompilerSymbol; + +/** + * Process log parser for {@link TestCompiler}. + * + * + * Sample error message: + * + * "Pass 1: In Abbuc99-NetBalls-Init.asm, line 24-- Warning: Resizing + * 'BALLSCOLORTAB1', forcing another pass" + * + * "Pass 2: In Abbuc99-NetBalls-Init.asm, line 25-- Warning: Resizing + * 'BALLSCOLORTAB2', forcing another pass" + * + * @author Peter Dell + */ +final class TestCompilerProcessLogParser extends CompilerProcessLogParser { + + private Pattern pattern; + private String sourceFilePattern; + + @Override + protected void initialize() { + pattern = Pattern.compile("In .* line "); + sourceFilePattern = "In " + mainSourceFilePath + ", line "; + } + + @Override + protected void findNextMarker() { + + int index; + String line; + boolean include; + String includeFile; + Matcher matcher = pattern.matcher(errorLog); + + if (matcher.find()) { + index = matcher.start(); + line = errorLog.substring(index); + if (line.startsWith(sourceFilePattern)) { + include = false; + includeFile = ""; + } else { + include = true; + includeFile = line.substring(3, matcher.end() - matcher.start() - 7); + } + index = matcher.end(); + String lineNumberLine = errorLog.substring(index); + errorLog = lineNumberLine; + int numberEndIndex = lineNumberLine.indexOf("--"); + if (numberEndIndex > 0) { + String lineNumberString; + lineNumberString = lineNumberLine.substring(0, numberEndIndex); + + try { + lineNumber = Integer.parseInt(lineNumberString); + int nextIndex = lineNumberLine.indexOf('\n'); + if (index > 0) { + message = lineNumberLine.substring(nextIndex + 1); + int nextIndex2 = message.indexOf('\n'); + if (nextIndex > 0) { + message = message.substring(0, nextIndex2 - 1); + } + message = message.trim(); + } + } catch (NumberFormatException ex) { + lineNumber = -1; + severity = IMarker.SEVERITY_ERROR; + message = ex.getMessage(); + } + } + + if (message.startsWith("Error:")) { + severity = IMarker.SEVERITY_ERROR; + message = message.substring(6); + } else if (message.startsWith("Warning:")) { + severity = IMarker.SEVERITY_WARNING; + message = message.substring(8); + } + + if (include) { + if (lineNumber >= 0) { + message = includeFile + " line " + lineNumber + ": " + message; + } else { + message = includeFile + ": " + message; + } + lineNumber = -1; + } + message = message.trim(); + + // Message mapping. + if (severity == IMarker.SEVERITY_WARNING && message.startsWith("Using bank")) { + severity = IMarker.SEVERITY_INFO; + } + markerAvailable = true; + } + } + + @Override + public void addCompilerSymbols(List compilerSymbols) { + final String EQUATES = "Equates:"; + final String SYMBOL = "Symbol"; + final String TABLE = "table:"; + + String log; + int index; + + log = outputLog; + index = log.indexOf(EQUATES); + if (index >= 0) { + log = log.substring(index + EQUATES.length()); + + StringTokenizer st = new StringTokenizer(log); + String token; + String name; + String hexValue; + while (st.hasMoreTokens()) { + token = st.nextToken(); + if (token.equals(SYMBOL)) { + break; + } + name = token.substring(0, token.length() - 1); + hexValue = st.nextToken(); // Must be there + compilerSymbols.add(CompilerSymbol.createNumberHexSymbol(name, hexValue)); + } + + if (st.hasMoreTokens()) { + token = st.nextToken(); + if (token.equals(TABLE)) { + while (st.hasMoreTokens()) { + token = st.nextToken(); + if (!token.endsWith(":")) { + break; + } + name = token.substring(0, token.length() - 1); + hexValue = st.nextToken(); // Must be there + compilerSymbols.add(CompilerSymbol.createNumberHexSymbol(name, hexValue)); + } + } + } + } + + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers.test/src/com/wudsn/ide/asm/compiler/test/TestCompilerSourceParser.java b/com.wudsn.ide.asm.compilers.test/src/com/wudsn/ide/asm/compiler/test/TestCompilerSourceParser.java new file mode 100644 index 00000000..ee96654a --- /dev/null +++ b/com.wudsn.ide.asm.compilers.test/src/com/wudsn/ide/asm/compiler/test/TestCompilerSourceParser.java @@ -0,0 +1,52 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.test; + +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; + +/** + * Source parser for {@link TestCompiler}. + * + * @author Peter Dell + */ +final class TestCompilerSourceParser extends CompilerSourceParser { + + @Override + protected void parseLine(int startOffset, String symbol, int symbolOffset, String instruction, + int instructionOffset, String operand, String comment) { + + if (symbol.length() > 0) { + + // Check for origin statement + if (symbol.equals("*")) { //$NON-NLS-1$ + beginImplementationSection(startOffset, startOffset + symbolOffset, operand, comment); + + } else { + if (instruction.equals("=")) { //$NON-NLS-1$ + createEquateDefinitionChild(startOffset, startOffset + symbolOffset, symbol, operand, comment); + } else { + createLabelDefinitionChild(startOffset, startOffset + symbolOffset, symbol, comment); + + } + } + + } // Symbol not empty + } +} diff --git a/com.wudsn.ide.asm.compilers.test/src/com/wudsn/ide/asm/editor/test/TestEditor.java b/com.wudsn.ide.asm.compilers.test/src/com/wudsn/ide/asm/editor/test/TestEditor.java new file mode 100644 index 00000000..2c497f73 --- /dev/null +++ b/com.wudsn.ide.asm.compilers.test/src/com/wudsn/ide/asm/editor/test/TestEditor.java @@ -0,0 +1,18 @@ +package com.wudsn.ide.asm.editor.test; + +import com.wudsn.ide.asm.editor.AssemblerEditor; + +public final class TestEditor extends AssemblerEditor { + + /** + * Creation is public. Called by the extension "org.eclipse.ui.editors". + */ + public TestEditor() { + + } + + @Override + public String getCompilerId() { + return "test"; + } +} diff --git a/com.wudsn.ide.asm.compilers.test/src/com/wudsn/ide/asm/preferences/test/AssemblerPreferencesTestCompilersPage.java b/com.wudsn.ide.asm.compilers.test/src/com/wudsn/ide/asm/preferences/test/AssemblerPreferencesTestCompilersPage.java new file mode 100644 index 00000000..57d83797 --- /dev/null +++ b/com.wudsn.ide.asm.compilers.test/src/com/wudsn/ide/asm/preferences/test/AssemblerPreferencesTestCompilersPage.java @@ -0,0 +1,42 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ +package com.wudsn.ide.asm.preferences.test; + +import com.wudsn.ide.asm.Hardware; +import com.wudsn.ide.asm.preferences.AssemblerPreferencesCompilersPage; + +/** + * Visual editor page for the assembler preferences regarding arbitrary compilers. + * + * @author Peter Dell + * + */ +public final class AssemblerPreferencesTestCompilersPage extends + AssemblerPreferencesCompilersPage { + + /** + * Create is public. Used by extension point + * "org.eclipse.ui.preferencePages". + */ + public AssemblerPreferencesTestCompilersPage() { + super(Hardware.TEST); + + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers/.classpath b/com.wudsn.ide.asm.compilers/.classpath new file mode 100644 index 00000000..8a8f1668 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/.classpath @@ -0,0 +1,7 @@ + + + + + + + diff --git a/com.wudsn.ide.asm.compilers/.project b/com.wudsn.ide.asm.compilers/.project new file mode 100644 index 00000000..1c2697d4 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/.project @@ -0,0 +1,40 @@ + + + com.wudsn.ide.asm.compilers + + + + + + org.eclipse.jdt.core.javabuilder + + + + + org.eclipse.pde.ManifestBuilder + + + + + org.eclipse.pde.SchemaBuilder + + + + + + org.eclipse.pde.PluginNature + org.eclipse.jdt.core.javanature + + + + binaries/build-compilers-new.bat + 1 + C:/jac/wudsn/Tools/WWW/build-compilers-new.bat + + + binaries/build.bat + 1 + C:/jac/wudsn/Tools/WWW/build.bat + + + diff --git a/com.wudsn.ide.asm.compilers/.settings/org.eclipse.jdt.core.prefs b/com.wudsn.ide.asm.compilers/.settings/org.eclipse.jdt.core.prefs new file mode 100644 index 00000000..f287d53c --- /dev/null +++ b/com.wudsn.ide.asm.compilers/.settings/org.eclipse.jdt.core.prefs @@ -0,0 +1,7 @@ +eclipse.preferences.version=1 +org.eclipse.jdt.core.compiler.codegen.inlineJsrBytecode=enabled +org.eclipse.jdt.core.compiler.codegen.targetPlatform=1.6 +org.eclipse.jdt.core.compiler.compliance=1.6 +org.eclipse.jdt.core.compiler.problem.assertIdentifier=error +org.eclipse.jdt.core.compiler.problem.enumIdentifier=error +org.eclipse.jdt.core.compiler.source=1.6 diff --git a/com.wudsn.ide.asm.compilers/META-INF/MANIFEST.MF b/com.wudsn.ide.asm.compilers/META-INF/MANIFEST.MF new file mode 100644 index 00000000..696a81fa --- /dev/null +++ b/com.wudsn.ide.asm.compilers/META-INF/MANIFEST.MF @@ -0,0 +1,12 @@ +Manifest-Version: 1.0 +Bundle-ManifestVersion: 2 +Bundle-Name: WUDSN IDE Assembler Compilers Plug-in +Bundle-SymbolicName: com.wudsn.ide.asm.compilers;singleton:=true +Bundle-Version: 1.7.0.qualifier +Bundle-Vendor: Peter Dell +Bundle-RequiredExecutionEnvironment: JavaSE-1.6 +Bundle-Localization: plugin +Require-Bundle: com.wudsn.ide.asm, + org.eclipse.ui.ide +Bundle-ActivationPolicy: lazy +Bundle-ClassPath: . diff --git a/com.wudsn.ide.asm.compilers/bin/.gitignore b/com.wudsn.ide.asm.compilers/bin/.gitignore new file mode 100644 index 00000000..43e58b99 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/bin/.gitignore @@ -0,0 +1 @@ +/com diff --git a/com.wudsn.ide.asm.compilers/binaries/.gitignore b/com.wudsn.ide.asm.compilers/binaries/.gitignore new file mode 100644 index 00000000..d1d483cb --- /dev/null +++ b/com.wudsn.ide.asm.compilers/binaries/.gitignore @@ -0,0 +1 @@ +/compilers diff --git a/com.wudsn.ide.asm.compilers/build.properties b/com.wudsn.ide.asm.compilers/build.properties new file mode 100644 index 00000000..236bd606 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/build.properties @@ -0,0 +1,15 @@ +source.. = src/ +output.. = bin/ +bin.includes = .,\ + plugin.xml,\ + icons/,\ + META-INF/,\ + plugin.properties,\ + src/,\ + build.properties,\ + .project,\ + .classpath,\ + plugin_de_DE.properties +jars.compile.order = . + + diff --git a/com.wudsn.ide.asm.compilers/icons/editor-acme-16x16.gif b/com.wudsn.ide.asm.compilers/icons/editor-acme-16x16.gif new file mode 100644 index 0000000000000000000000000000000000000000..d6d12fd1f7411c32898ed145609c16cb6a74fc65 GIT binary patch literal 1763 zcmZ?wbhEHb6krfw_|5+{b|BpV#-lj`7!0Fe z_=N!e9Kmsbp`XKsX@NompEQq{NQS^676$pY7cT^r54B1$^euSsu+fRRL#yS%gv3Ob zjvj3b3C6%jjVuhtA$wLlXlQg0cc~HKbWCPs5SFwHQJDCVgIj{Ff5wTz2M2h0mAdpo zPHkD)E5dHXBeLU5L$9>CQOe$lQ?)OPBnEHsNL>7~Pujj&CZYP#YtHTFL328OE4#PJ zsC{^4n#jb^CT&*s=D@GAf<$4S>1>%bO&|Q5qJ3EkYj(Z}V=&io{yrqD<>o<#K+4gB`HN(N@~iqDK%9! zbLP!iZLH42#naN(vS!_yHOA`y|NjT-A_ge_WMO1r5Ms~)DF)fez}m7vg)t>_(xV6s zfyFl7jS|jlwR{Q%9Ac$yCpk4Lu=cq(FwbCc;9_Vjxy3PogPGy6z}%pw4J8KrZZ$SYYw<9MTUv2?=e?ufEi-XE^ b4pwdsYfd54aGOkqRScT8b{p)J6d9}mj-)>F literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers/icons/editor-atasm-16x16.gif b/com.wudsn.ide.asm.compilers/icons/editor-atasm-16x16.gif new file mode 100644 index 0000000000000000000000000000000000000000..e4633f21502d6140df67b76c5b2517ee94e0749b GIT binary patch literal 909 zcmZ?wbhEHb6krfw_|Cxa9|##34jee}|NsAnh6W&e6pV(z$O!?(pDc_Z-|Bz}P@Z7t z$YNmRkkK$`U|?bq=HoF~py<@fEzKuTAfUw1#>W{@<1s<8kx`yM?+wQ$71xPOOd>jx zlad&i3__DkUVc(woGBE&Y|Bca28MRw>|-)Ngr9jfJ9CO%?f9a0WU0a2sFquo7x_;J NT^o01mVp9;H2~E%Fk%1z literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers/icons/editor-dasm-16x16.gif b/com.wudsn.ide.asm.compilers/icons/editor-dasm-16x16.gif new file mode 100644 index 0000000000000000000000000000000000000000..933dc281d22a2fa71f76050a631336193aef1615 GIT binary patch literal 195 zcmZ?wbhEHb6krfw*vtS1GmRPk|4&Q%f5!L>hyrqP1I3>#j0_A+3_2hsATydl3{p&B zaR@l!xq5Fz!&yf*7LB^Zmx&G&*Uf)5?^+teXCD6|-`k=BEYb`3JGhHl?E2CUd|_bo WQ!C-@wVHpq&t>Jqw0#da7_0$F2sEVt literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm.compilers/icons/editor-kickass-16x16.gif b/com.wudsn.ide.asm.compilers/icons/editor-kickass-16x16.gif new file mode 100644 index 0000000000000000000000000000000000000000..21a1fbd34df3eb6aab913397c066b7ab8abc96ba GIT binary patch literal 1639 zcmZ?wbhEHb6krfw_|5`=Z zK(SFU8UlkV1pfd2uwcOwTq#!ZCkrFUxjG;MlqcGObORWV<_KUgjDq180{C+ThYJHE zhm6OD1&-Vtf>J6o3?4c)^2+MfkkK$`U|?bq=HoF~py<@fEzKuTAdu9^B%~iy<`d;s*cq^nd8;f78|bqNDd&NAHu4-g|A`ciOsdv~^x+={(ocd8VcP zSX29frq*2zt=k$J7u3{FsL0Nek?NF`Xb_jE6BjEM70DD92;%2;;*psoEm12jk|xZ< z!^6P9@c;jRAUSk_;!hSv1_lQP9gr75al*iUqQSv|xrM2vy}hZKQ$$QyM3jS*ot25H ziMdCZlZ}OW+BDYrEg~Z9%=Rm-xOtfw7BMo5ajsiwFKcSg(8AIt%(hP4%+gH3TyXbd z5f*z3RV8VC6-65cmc7iP%ywMr+FH^YMnVh++P3iNNol`Pli$U9gOkb5Q1gwplng%u zJIh9P7KR&*jQ6{ExOr3#q$!BCh-n7QxM9%b#>ldmN#w@W=GJye1BsX~=^6}-fh{~D f784i_Hu^ELNJ`=Z zK(SFU8UlkV1pfd2uwcOwTq#!ZCkrFUxjG;MlqcGObORWV<_KUgjDq180{C+ThYJHE zhm6OD1&-Vtf>J6o3?4c)^2+M;Cjxvex z2k``KNb#Ja&3Pi_ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/com.wudsn.ide.asm.compilers/plugin_de_DE.properties b/com.wudsn.ide.asm.compilers/plugin_de_DE.properties new file mode 100644 index 00000000..4ba0fae0 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/plugin_de_DE.properties @@ -0,0 +1,75 @@ +com.wudsn.ide.asm.compiler.acme.AcmeCompiler.name=ACME +com.wudsn.ide.asm.compiler.asm6.Asm6Compiler.name=ASM6 +com.wudsn.ide.asm.compiler.atasm.AtasmCompiler.name=ATASM +com.wudsn.ide.asm.compiler.dasm.DasmCompiler.name=DASM +com.wudsn.ide.asm.compiler.kickass.KickAssCompiler.name=KICKASS +com.wudsn.ide.asm.compiler.mads.MadsCompiler.name=MADS +com.wudsn.ide.asm.compiler.merlin32.Merlin32Compiler.name=MERLIN32 +com.wudsn.ide.asm.compiler.tass.TassCompiler.name=TASS +com.wudsn.ide.asm.compiler.xasm.XasmCompiler.name=XASM + +com.wudsn.ide.asm.compiler.acme.AcmeSourceFile.name=ACME Quelll-Datei +com.wudsn.ide.asm.compiler.asm6.Asm6SourceFile.name=ASM6 Quelll-Datei +com.wudsn.ide.asm.compiler.atasm.AtasmSourceFile.name=ATASM Quelll-Datei +com.wudsn.ide.asm.compiler.dasm.DasmSourceFile.name=DASM Quelll-Datei +com.wudsn.ide.asm.compiler.kickass.KickAssSourceFile.name=KICKASS Quelll-Datei +com.wudsn.ide.asm.compiler.mads.MadsSourceFile.name=MADS Quelll-Datei +com.wudsn.ide.asm.compiler.merlin32.Merlin32SourceFile.name=MERLIN32 Quelll-Datei +com.wudsn.ide.asm.compiler.tass.TassSourceFile.name=TASS Quelll-Datei +com.wudsn.ide.asm.compiler.xasm.XasmSourceFile.name=XASM Quelll-Datei + +com.wudsn.ide.asm.editor.acme.AcmeEditor.name=ACME Editor +com.wudsn.ide.asm.editor.asm6.Asm6Editor.name=ASM6 Editor +com.wudsn.ide.asm.editor.atasm.AtasmEditor.name=ATASM Editor +com.wudsn.ide.asm.editor.dasm.DasmEditor.name=DASM Editor +com.wudsn.ide.asm.editor.kickass.KickAssEditor.name=KICKASS Editor +com.wudsn.ide.asm.editor.mads.MadsEditor.name=MADS Editor +com.wudsn.ide.asm.editor.merlin32.Merlin32Editor.name=MERLIN32 Editor +com.wudsn.ide.asm.editor.tass.TassEditor.name=TASS Editor +com.wudsn.ide.asm.editor.xasm.XasmEditor.name=XASM Editor + +# +# Hardware: APPLE2 +# +com.wudsn.ide.asm.preferences.AssemblerPreferencesApple2CompilersPage.name=Apple II Kompiler +com.wudsn.ide.asm.runner.apple2.applewin.name=AppleWin +com.wudsn.ide.asm.runner.apple2.jace.name=JACE +com.wudsn.ide.asm.runner.apple2.virtu.name=Virtu +com.wudsn.ide.asm.runner.apple2.virtualii.name=Virtual ][ + +# +# Hardware: ATARI2600 +# +com.wudsn.ide.asm.preferences.AssemblerPreferencesAtari2600CompilersPage.name=Atari 2600 Kompiler +com.wudsn.ide.asm.runner.atari2600.stella.name=Stella + +# +# Hardware: ATARI7800 +# +com.wudsn.ide.asm.preferences.AssemblerPreferencesAtari7800CompilersPage.name=Atari 7800 Kompiler +com.wudsn.ide.asm.runner.atari7800.emu7800.name=EMU7800 + +# +# Hardware: ATARI8 +# +com.wudsn.ide.asm.preferences.AssemblerPreferencesAtari8CompilersPage.name=Atari 8-bit Kompiler +com.wudsn.ide.asm.runner.atari8.atari800.name=Atari800 +com.wudsn.ide.asm.runner.atari8.atari800_win.name=Atari800Win +com.wudsn.ide.asm.runner.atari8.atari800_macx.name=Atari800MacX +com.wudsn.ide.asm.runner.atari8.altirra.name=Altirra +com.wudsn.ide.asm.runner.atari8.atari_plus_plus.name=Atari++ + +# +# Hardware: C64 +# +com.wudsn.ide.asm.preferences.AssemblerPreferencesC64CompilersPage.name=C64 Kompiler +com.wudsn.ide.asm.runner.c64.ccs64.name=CCS64 +com.wudsn.ide.asm.runner.c64.vice.name=VICE + +# +# Hardware: NES +# +com.wudsn.ide.asm.preferences.AssemblerPreferencesNESCompilersPage.name=NES Kompiler +com.wudsn.ide.asm.runner.nes.bsnes.name=BSNES +com.wudsn.ide.asm.runner.nes.fceux.name=FCEUX +com.wudsn.ide.asm.runner.nes.nintendulator.name=Nintendulator diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/CompilerId.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/CompilerId.java new file mode 100644 index 00000000..cbc5569b --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/CompilerId.java @@ -0,0 +1,38 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler; + +/** + * Compiler IDs for static use. + * + * @author Peter Dell + * @since 1.6.1 + */ +public class CompilerId { + public static final String ACME = "acme"; + public static final String ASM6 = "asm6"; + public static final String ATASM = "atasm"; + public static final String DASM = "dasm"; + public static final String KICKASS = "kickass"; + public static final String MADS = "mads"; + public static final String MERLIN32 = "merlin32"; + public static final String TASS = "tass"; + public static final String XASM = "xasm"; +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/acme/AcmeCompiler.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/acme/AcmeCompiler.java new file mode 100644 index 00000000..c3fa7461 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/acme/AcmeCompiler.java @@ -0,0 +1,49 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ +package com.wudsn.ide.asm.compiler.acme; + +import com.wudsn.ide.asm.compiler.Compiler; +import com.wudsn.ide.asm.compiler.CompilerProcessLogParser; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; + +/** + * Compiler class for ACME. + * + * @author Peter Dell + */ +public final class AcmeCompiler extends Compiler { + + /** + * Creates a new instance. + */ + public AcmeCompiler() { + } + + @Override + public CompilerSourceParser createSourceParser() { + return new AcmeCompilerSourceParser(); + } + + @Override + public CompilerProcessLogParser createLogParser() { + + return new AcmeCompilerProcessLogParser(); + } + +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/acme/AcmeCompiler.xml b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/acme/AcmeCompiler.xml new file mode 100644 index 00000000..2b62c867 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/acme/AcmeCompiler.xml @@ -0,0 +1,95 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/acme/AcmeCompilerProcessLogParser.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/acme/AcmeCompilerProcessLogParser.java new file mode 100644 index 00000000..c79f5762 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/acme/AcmeCompilerProcessLogParser.java @@ -0,0 +1,123 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.acme; + +import java.io.BufferedReader; +import java.io.IOException; +import java.io.StringReader; +import java.util.List; + +import org.eclipse.core.resources.IMarker; + +import com.wudsn.ide.asm.compiler.CompilerProcessLogParser; +import com.wudsn.ide.asm.compiler.CompilerSymbol; + +/** + * Process log parser for {@link AcmeCompiler}. + * + * Sample error message: + * + *
+ * Error - File include/ACME-Reference-Source-Include.a, line 3 (Zone ): Value not yet defined.
+ * Error - File C:\Users\D025328\Documents\Eclipse\workspace.jac\com.wudsn.ide.ref\ASM\C64\ACME\ACME-Error-Reference.a, line 9 (Zone ): Value not yet defined.
+ * 
+ * + * @author Peter Dell + */ +final class AcmeCompilerProcessLogParser extends CompilerProcessLogParser { + private BufferedReader bufferedReader; + + @Override + protected void initialize() { + bufferedReader = new BufferedReader(new StringReader(errorLog)); + } + + @Override + protected void findNextMarker() { + + String line; + line = ""; + + while (line != null && !markerAvailable) { + try { + line = bufferedReader.readLine(); + } catch (IOException ex) { + throw new RuntimeException("Cannot read line", ex); + } + if (line != null) { + String pattern; + int index; + pattern = "Error - File "; + severity = IMarker.SEVERITY_ERROR; + index = line.indexOf(pattern); + if (index < 0) { + pattern = "Warning - File "; + severity = IMarker.SEVERITY_WARNING; + index = line.indexOf(pattern); + } + if (index >= 0) { + + index = index + pattern.length(); + pattern = ", line "; + int i = line.indexOf(pattern); + if (i == -1) { + continue; + } + filePath = line.substring(index, i); + + i = i + pattern.length(); + int j = i; + while (line.charAt(j) != ' ' && j < line.length()) { + j++; + } + + String lineNumberString = line.substring(i, j); + + try { + lineNumber = Integer.parseInt(lineNumberString); + } catch (NumberFormatException ex) { + lineNumber = -1; + severity = IMarker.SEVERITY_ERROR; + message = ex.getMessage(); + } + + pattern = "): "; + line = line.substring(j); + j = line.indexOf(pattern); + if (j > -1) { + j = j + pattern.length(); + message = line.substring(j); + } else { + message = line; + } + + markerAvailable = true; + } + + } + } + } + + @Override + public void addCompilerSymbols(List compilerSymbols) { + + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/acme/AcmeCompilerSourceParser.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/acme/AcmeCompilerSourceParser.java new file mode 100644 index 00000000..2f265526 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/acme/AcmeCompilerSourceParser.java @@ -0,0 +1,59 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.acme; + +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; + +/** + * Source parser for {@link AcmeCompiler}. + * + * @author Peter Dell + */ +final class AcmeCompilerSourceParser extends CompilerSourceParser { + + /** + * Creation is package local. + */ + AcmeCompilerSourceParser() { + + } + + @Override + protected void parseLine(int startOffset, String symbol, int symbolOffset, String instruction, + int instructionOffset, String operand, String comment) { + + if (symbol.length() > 0) { + + // Check for origin statement + if (symbol.equals("*")) { + beginImplementationSection(startOffset, startOffset + symbolOffset, operand, comment); + + } else { + if (instruction.equals("=")) { + createEquateDefinitionChild(startOffset, startOffset + symbolOffset, symbol, operand, comment); + } else { + createLabelDefinitionChild(startOffset, startOffset + symbolOffset, symbol, comment); + + } + } + + } // Symbol not empty + } +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/asm6/Asm6Compiler.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/asm6/Asm6Compiler.java new file mode 100644 index 00000000..e29d08d1 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/asm6/Asm6Compiler.java @@ -0,0 +1,50 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.asm6; + +import com.wudsn.ide.asm.compiler.Compiler; +import com.wudsn.ide.asm.compiler.CompilerProcessLogParser; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; + +/** + * Compiler class for ASM6. + * + * @author Peter Dell + */ +public final class Asm6Compiler extends Compiler { + + /** + * Creates a new instance. + */ + public Asm6Compiler() { + } + + @Override + public CompilerSourceParser createSourceParser() { + return new Asm6CompilerSourceParser(); + } + + @Override + public CompilerProcessLogParser createLogParser() { + + return new Asm6CompilerProcessLogParser(); + } + +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/asm6/Asm6Compiler.xml b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/asm6/Asm6Compiler.xml new file mode 100644 index 00000000..a03129c1 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/asm6/Asm6Compiler.xml @@ -0,0 +1,74 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/asm6/Asm6CompilerProcessLogParser.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/asm6/Asm6CompilerProcessLogParser.java new file mode 100644 index 00000000..f08713da --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/asm6/Asm6CompilerProcessLogParser.java @@ -0,0 +1,111 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.asm6; + +import java.io.BufferedReader; +import java.io.IOException; +import java.io.StringReader; +import java.util.List; + +import org.eclipse.core.resources.IMarker; + +import com.wudsn.ide.asm.compiler.CompilerProcessLogParser; +import com.wudsn.ide.asm.compiler.CompilerSymbol; + +/** + * Process log parser for {@link Asm6Compiler}. + * + * Sample error message: + * + *
+ * include/ASM6-Reference-Source-Include.asm(2): Illegal instruction.
+ * include/ASM6-Reference-Source-Include.asm(4): Illegal instruction.
+ * 
+ * + * @author Peter Dell + */ +final class Asm6CompilerProcessLogParser extends CompilerProcessLogParser { + + private BufferedReader bufferedReader; + + @Override + protected void initialize() { + bufferedReader = new BufferedReader(new StringReader(errorLog)); + } + + @Override + protected void findNextMarker() { + + String line; + line = ""; + + while (line != null && !markerAvailable) { + try { + line = bufferedReader.readLine(); + } catch (IOException ex) { + throw new RuntimeException("Cannot read line", ex); + } + if (line != null) { + String pattern; + int index; + pattern = "): "; + severity = IMarker.SEVERITY_ERROR; + index = line.indexOf(pattern); + if (index < 2) { + pattern = "): "; + severity = IMarker.SEVERITY_WARNING; + index = line.indexOf(pattern); + } + if (index > 2) { + + int i = index - 2; + while (line.charAt(i) != '(' && i >= 0) { + i--; + } + + if (line.charAt(i) == '(') { + String lineNumberString = line.substring(i + 1, index); + + try { + lineNumber = Integer.parseInt(lineNumberString); + } catch (NumberFormatException ex) { + lineNumber = -1; + severity = IMarker.SEVERITY_ERROR; + message = ex.getMessage(); + } + } else { + lineNumber = -1; + } + message = line.substring(index + pattern.length()).trim(); + + filePath = line.substring(0, i); + markerAvailable = true; + } + + } + } + } + + @Override + public void addCompilerSymbols(List compilerSymbols) { + + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/asm6/Asm6CompilerSourceParser.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/asm6/Asm6CompilerSourceParser.java new file mode 100644 index 00000000..ee30c3a3 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/asm6/Asm6CompilerSourceParser.java @@ -0,0 +1,52 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.asm6; + +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; + +/** + * Source parser for {@link Asm6Compiler}. + * + * @author Peter Dell + */ +final class Asm6CompilerSourceParser extends CompilerSourceParser { + + @Override + protected void parseLine(int startOffset, String symbol, int symbolOffset, String instruction, + int instructionOffset, String operand, String comment) { + + if (symbol.length() > 0) { + + // Check for origin statement + if (symbol.equals("*")) { + beginImplementationSection(startOffset, startOffset + symbolOffset, operand, comment); + + } else { + if (instruction.equals("=")) { + createEquateDefinitionChild(startOffset, startOffset + symbolOffset, symbol, operand, comment); + } else { + createLabelDefinitionChild(startOffset, startOffset + symbolOffset, symbol, comment); + + } + } + + } // Symbol not empty + } +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/atasm/AtasmCompiler.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/atasm/AtasmCompiler.java new file mode 100644 index 00000000..cb361dec --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/atasm/AtasmCompiler.java @@ -0,0 +1,50 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.atasm; + +import com.wudsn.ide.asm.compiler.Compiler; +import com.wudsn.ide.asm.compiler.CompilerProcessLogParser; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; + +/** + * Compiler class for ATASM. + * + * @author Peter Dell + */ +public final class AtasmCompiler extends Compiler { + + /** + * Creates a new instance. + */ + public AtasmCompiler() { + } + + @Override + public CompilerSourceParser createSourceParser() { + return new AtasmCompilerSourceParser(); + } + + @Override + public CompilerProcessLogParser createLogParser() { + + return new AtasmCompilerProcessLogParser(); + } + +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/atasm/AtasmCompiler.xml b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/atasm/AtasmCompiler.xml new file mode 100644 index 00000000..160487f6 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/atasm/AtasmCompiler.xml @@ -0,0 +1,252 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + /> + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/atasm/AtasmCompilerProcessLogParser.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/atasm/AtasmCompilerProcessLogParser.java new file mode 100644 index 00000000..271eac64 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/atasm/AtasmCompilerProcessLogParser.java @@ -0,0 +1,154 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.atasm; + +import java.util.List; +import java.util.StringTokenizer; +import java.util.regex.Matcher; +import java.util.regex.Pattern; + +import org.eclipse.core.resources.IMarker; + +import com.wudsn.ide.asm.compiler.CompilerProcessLogParser; +import com.wudsn.ide.asm.compiler.CompilerSymbol; + +/** + * Process log parser for {@link AtasmCompiler}. + * + * Sample error message: + * + *
+ * "Pass 1: In Abbuc99-NetBalls-Init.asm, line 24-- Warning: Resizing 'BALLSCOLORTAB1', forcing another pass"
+ * "Pass 2: In Abbuc99-NetBalls-Init.asm, line 25-- Warning: Resizing 'BALLSCOLORTAB2', forcing another pass"
+ * 
+ * + * @author Peter Dell + */ +final class AtasmCompilerProcessLogParser extends CompilerProcessLogParser { + + private Pattern pattern; + + @Override + protected void initialize() { + pattern = Pattern.compile("In .* line "); + } + + @Override + protected void findNextMarker() { + + int index; + String line; + Matcher matcher = pattern.matcher(errorLog); + + if (matcher.find()) { + index = matcher.start(); + line = errorLog.substring(index); + filePath = line.substring(3, matcher.end() - matcher.start() - 7); + index = matcher.end(); + String lineNumberLine = errorLog.substring(index); + errorLog = lineNumberLine; + int numberEndIndex = lineNumberLine.indexOf("--"); + if (numberEndIndex > 0) { + String lineNumberString; + lineNumberString = lineNumberLine.substring(0, numberEndIndex); + + try { + lineNumber = Integer.parseInt(lineNumberString); + int nextIndex = lineNumberLine.indexOf("\n"); + if (nextIndex > 0) { + message = lineNumberLine.substring(nextIndex + 1); + int nextIndex2 = message.indexOf("\n"); + if (nextIndex2 > 0) { + message = message.substring(0, nextIndex2 - 1); + } + message = message.trim(); + } + } catch (NumberFormatException ex) { + lineNumber = -1; + severity = IMarker.SEVERITY_ERROR; + message = ex.getMessage(); + } + } + + if (message.startsWith("Error:")) { + severity = IMarker.SEVERITY_ERROR; + message = message.substring(6); + } else if (message.startsWith("Warning:")) { + severity = IMarker.SEVERITY_WARNING; + message = message.substring(8); + } + + message = message.trim(); + + // Message mapping. + if (severity == IMarker.SEVERITY_WARNING && message.startsWith("Using bank")) { + severity = IMarker.SEVERITY_INFO; + } + markerAvailable = true; + } + } + + @Override + public void addCompilerSymbols(List compilerSymbols) { + final String EQUATES = "Equates:"; + final String SYMBOL = "Symbol"; + final String TABLE = "table:"; + + String log; + int index; + + log = outputLog; + index = log.indexOf(EQUATES); + if (index >= 0) { + log = log.substring(index + EQUATES.length()); + + StringTokenizer st = new StringTokenizer(log); + String token; + String name; + String hexValue; + while (st.hasMoreTokens()) { + token = st.nextToken(); + if (token.equals(SYMBOL)) { + break; + } + name = token.substring(0, token.length() - 1); + hexValue = st.nextToken(); // Must be there + compilerSymbols.add(CompilerSymbol.createNumberHexSymbol(name, hexValue)); + } + + if (st.hasMoreTokens()) { + token = st.nextToken(); + if (token.equals(TABLE)) { + while (st.hasMoreTokens()) { + token = st.nextToken(); + if (!token.endsWith(":")) { + break; + } + name = token.substring(0, token.length() - 1); + hexValue = st.nextToken(); // Must be there + compilerSymbols.add(CompilerSymbol.createNumberHexSymbol(name, hexValue)); + } + } + } + } + + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/atasm/AtasmCompilerSourceParser.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/atasm/AtasmCompilerSourceParser.java new file mode 100644 index 00000000..ad3f0f5d --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/atasm/AtasmCompilerSourceParser.java @@ -0,0 +1,51 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.atasm; + +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; + +/** + * Source parser for {@link AtasmCompiler}. + * + * @author Peter Dell + */ +final class AtasmCompilerSourceParser extends CompilerSourceParser { + + @Override + protected void parseLine(int startOffset, String symbol, int symbolOffset, String instruction, + int instructionOffset, String operand, String comment) { + + if (symbol.length() > 0) { + + // Check for origin statement + if (symbol.equals("*")) { + beginImplementationSection(startOffset, startOffset, operand, comment); + + } else { + if (instruction.equals("=")) { + createEquateDefinitionChild(startOffset, startOffset + symbolOffset, symbol, operand, comment); + } else { + createLabelDefinitionChild(startOffset, startOffset + symbolOffset, symbol, comment); + } + } + + } // Symbol not empty + } +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/dasm/DASM.txt b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/dasm/DASM.txt new file mode 100644 index 00000000..f524c3f6 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/dasm/DASM.txt @@ -0,0 +1,750 @@ + + +DOCUMENTATION FOR DASM, a high level macro cross assembler for: + + -6502 (and 6507) + -68705 + -6803 + -HD6303 (extension of 6803) + -68HC11 + +DASM was created in 1988 by Matthew Dillon (dillon@apollo.backplane.com) +and modified in 1995 by Olaf 'Rhialto' Seibert (rhialto@polderland.nl), and +is now updated and maintained by Andrew Davie (atari2600@taswegian.com) + +DASM's homepage is http://www.atari2600.org/dasm + +DASM is freely re-distributable, and provided with full source-code to +allow recompilation for other target-platforms. + + +PREFACE FROM MATT: + + Over the last year my work has included writing software to drive small + single-chip microcomputers for various things (remote telemetry units, + for instance). I have had need to program quite a few different + processors over that time. + + At the beginning, I used an awful macro assembler running on an IBM-PC. + I *really* wanted to do it on my Amiga. Thus the writing of this + program. + + Feel free to suggest other similar processors for me to add to the list! + The processor type is specified with a pseudo-op (see below). This + assembler produces only binary output in one of three formats described + below. In general, one has a master assembly file which INCLUDEs all + the modules. + + Also provided is FTOHEX which converts an output file in one of the + three formats to an intel-hex format suitable for many intelligent + prom programmers (I have a GTEK). + + YES it's packed with features! + +FEATURES: + + -fast assembly + -supports several common 8 bit processor models (NOT 8086, thank god!) + -takes as many passes as needed + -automatic checksum generation, special symbol '...' + -several binary output formats available. Format 2 allows reverse + indexed origins. + -multiple segments, BSS segments (no generation), relocatable origin. + -expressions, as in C but [] is used instead of () for parenthesis. + (all expressions are computed with 32 bit integers) + -no real limitation on label size, label values are 32 bits. + -complex pseudo ops, repeat loops, macros, etc.... + + +PREFACE FROM ANDREW (APRIL/2003) + +The documentation is lagging a bit behind the modifications. Essentially the information contained herein is correct, but there have been minor changes to the formatting of output data. + +Please see the accompanying ‘README.TXT’ in the distributable root directory for a list of recent changes to the package. + + +COMMAND LINE: + + dasm srcfile [options] + + options: -f# select output format 1-3 (default 1, see below) + -oname select output file name (else a.out) + -lname select list file name (else none generated) + -Lname list file, containing all passes + -sname select symbol dump file (else none generated) + -v# select verboseness 0-4 (default 0, see below) + -d debug mode + -DSYMBOL predefine a symbol, set to 0 + -DSYMBOL=EXPRESSION predefine a symbol, set to exp + -MSYMBOL=EXPRESSION define label as in EQM (same as –D) + + -Idir search directory for include and incbin + -p# max number of passes + -P# max number of passes, with less checks + + -T# sort symbol table by (0 = alphabetic (default) + or (non-zero= address/value) + + Example: dasm master.asm -f2 -oout -llist -v3 -DVER=4 + +Note: A slash (/) or dash (-) must prefix options. + + +Return Value: + + The assembler will return 0 on successful compilation, 1 otherwise. + + +FORMAT OPTIONS: + + 1 (DEFAULT) + + The output file contains a two byte origin in LSB,MSB order, then + data until the end of the file. + + Restrictions: Any instructions which generate output (within an + initialized segment) must do so with an ascending PC. Initialized + segments must occur in ascending order. + + 2 RAS (Random Access Segment) + + The output file contains one or more hunks. Each hunk consists + of a 2 byte origin (LSB,MSB), 2 byte length (LSB,MSB), and that + number of data bytes. The hunks occur in the same order as + initialized segments in the assembly. There are no restrictions + to segment ordering (i.e. reverse indexed ORG statements are + allowed). The next hunk begins after the previous hunk's data, + until the end of the file. + + 3 RAW (Raw) + + The output file contains data only (format #1 without the 2 byte + header). Restrictions are the same as for format #1. + + Format 3 RAW (Raw format) + Same as format 1, but NO header origin is generated. You get + nothing but data. + +VERBOSE OPTIONS: + + 0 (default) + + Only warnings and errors are generated + + 1 + -Segment list information generated after each pass + -Include file names are displayed + -statistics on why the assembler is going to make another pass + R1,R2 reason code: R3 + where R1 is the number of times the assembler encountered + something requiring another pass to resolve. R2 is the + number of references to unknown symbols which occured in the + pass (but only R1 determines the need for another pass). R3 + is a BITMASK of the reasons why another pass is required. + See the end of this document for bit designations. + + 2 + mismatches between program labels and equates are displayed + on every pass (usually none occur in the first pass unless you + have re-declared a symbol name). + + displayed information for symbols: + ???? = unknown value + str = symbol is a string + eqm = symbol is an eqm macro + (r) = symbol has been referenced + (s) = symbol created with SET or EQM pseudo-op + + 3 + Unresolved and unreferenced symbols are displayed every pass + (unsorted, sorry) + + 4 + An entire symbol list is displayed every pass to STDOUT. + (unsorted, sorry) + + + +PROCESSOR MODEL: + + The processor model is chosen with the PROCESSOR pseudo-op and should + be the first thing you do in your assembly file. Different processor + models use different integer formats (see below). The word order does + not effect the headers in the output files (-f1 and -f2), which are + always LSB,MSB. The word ordering effects all address, word, and + long generation. + + Only one PROCESSOR pseudo-op may be declared in the entire assembly, + and should be the first thing encountered. + + -6502 LSB,MSB + -68HC11 MSB,LSB + -68705 MSB,LSB + -6803 MSB,LSB + -HD6303 MSB,LSB + -F8 ? + +SEGMENTS: + The SEG pseudo-op creates/sets the current segment. Each segment has + it's own origin and is optionally an 'uninitialized' segment. + Unitialized segments produce no output and have no restrictions. This + is useful for determining the size of a certain assembly sequence + without generating code, and for assigning RAM to labels. + + 'Initialized' segments produce output. The following should be + considered when generating roms: + + (1) The default fill character when using ORG (and format 1 or 3) to + skip forward is 00. This is a GLOBAL default and effects all + segments. See ORG. + + (2) The default fill character for DS is 00 and is independant of + the default fill character for ORG (see DS). + +GENERAL: + Most everything is recursive. You cannot have a macro DEFINITION + within a macro definition, but can nest macro calls, repeat loops, + and include files. + + The other major feature in this assembler is the SUBROUTINE pseudo-op, + which logically separates local labels (starting with a dot). This + allows you to reuse label names (for example, .1 .fail) rather than + think up crazy combinations of the current subroutine to keep it all + unique. + + Almost nothing need be resolved in pass 1. The assembler will make + multiple passes in an attempt to resolve the assembly (including just + one pass if everything is resolved immediately). + + +PSEUDOPS: + + INCLUDE "name" + + Include another assembly file. + +#if OlafIncbin + +[label] INCBIN "name" + + Include another file literally in the output. +#endif +#if OlafIncdir + + INCDIR "directory name" + + Add the given directory name to the list of places where + INCLUDE and INCBIN search their files. First, the names are + tried relative to the current directory, if that fails and + the name is not an absolute pathname, the list is tried. + You can optionally end the name with a /. AmigaDOS filename + conventions imply that two slashes at the + end of an INCDIR (dir//) indicates the parent directory, and + so does an INCLUDE /filename. + + The command-line option -Idir is equivalent to an INCDIR + directive placed before the source file. + + Currently the list is not cleared between passes, but each + exact directory name is added to the list only once. + This may change in subsequent releases. +#endif +[label] SEG[.U] name + + This sets the current segment, creating it if neccessary. If + a .U extension is specified on segment creation, the segment + is an UNINITIALIZED segment. The .U is not needed when going + back to an already created uninitialized segment, though it + makes the code more readable. + +[label] DC[.BWL] exp,exp,exp ... + + Declare data in the current segment. No output is generated if + within a .U segment. Note that the byte ordering for the + selected processor is used for each entry. + + The default size extension is a byte. +#if OlafByte + BYTE, WORD and LONG are synonyms for DC.B, DC.W and DC.L. +#endif + +[label] DS[.BWL] exp[,filler] + + declare space (default filler is 0). Data is not generated if + within an uninitialized segment. Note that the number of bytes + generated is exp * entrysize (1,2, or 4) + + The default size extension is a byte. + + Note that the default filler is always 0 (has nothing to do + with the ORG default filler). + +[label] DV[.BWL] eqmlabel exp,exp,exp.... + + This is equivalent to DC, but each exp in the list is passed + through the symbolic expression specified by the EQM label. + The expression is held in a special symbol dotdot '..' on each + call to the EQM label. + + See EQM below + +[label] HEX hh hh hh.. + + This sets down raw HEX data. Spaces are optional between bytes. + NO EXPRESSIONS are allowed. Note that you do NOT place a $ + in front of the digits. This is a short form for creating + tables compactly. Data is always layed down on a byte-by-byte + basis. + + Example: HEX 1A45 45 13254F 3E12 + + ERR + + Abort assembly. + +[label] ORG exp[,DefaultFillVal] + + This pseudop sets the current origin. You can also set the + global default fill character (a byte value) with this + pseudoop. NOTE that no filler is generated until the first + data-generating opcode/psueoop is encountered after this one. + Sequences like: + + org 0,255 + org 100,0 + org 200 + dc 23 + + will result in 200 zero's and a 23. This allows you to specify + some ORG, then change your mind and specify some other (lower + address) ORG without causing an error (assuming nothing is + generated inbetween). + + Normally, DS and ALIGN are used to generate specific filler + values. + +[label] RORG exp + + This activates the relocatable origin. All generated + addresses, including '.', although physically placed at the + true origin, will use values from the relocatable origin. + While in effect both the physical origin and relocatable origin + are updated. + + The relocatable origin can skip around (no limitations). The + relocatable origin is a function of the segment. That is, you + can still SEG to another segment that does not have a + relocatable origin activated, do other (independant) stuff + there, and then switch back to the current segment and continue + where you left off. + + PROCESSOR model + + do not quote. model is one of: 6502,6803,HD6303,68705,68HC11,F8 + Can only be executed once, and should be the first thing + encountered by the assembler. + + ECHO exp,exp,exp + + The expressions (which may also be strings), are echoed on the + screen and into the list file. + +[label] REND + + Deactivate the relocatable origin for the current segment. + Generation uses the real origin for reference. + +[label] ALIGN N[,fill] + + Align the current PC to an N byte boundry. The default + fill character is always 0, and has nothing to do with + the default fill character specifiable in an ORG. + +[label] SUBROUTINE name + + This isn't really a subroutine, but a boundry between sets of + temporary labels (which begin with a dot). Temporary label + names are unique within segments of code bounded by SUBROUTINE: + + CHARLIE subroutine + ldx #10 + .1 dex + bne .1 + BEN subroutine + ldx #20 + .qq dex + bne .qq + + Automatic temporary label boundries occur for each macro level. + Usually temporary labels are used in macros and within actual + subroutines (so you don't have to think up a thousand different + names) + + +symbol EQU exp +#if OlafAsgn +symbol = exp +#endif + + The expression is evaluated and the result assigned to the + symbol. + +#if OlafDotAssign + If this option is enabled, you can use the common idiom of + + . EQU . + 3 (or . = .+3) + + in other words, you can assign to "." (or "*" if OlafStar + is also enabled) instead of an ORG or RORG directive. + More formally, a directive of the form ". EQU expr" is + interpreted as if it were written " (R)ORG expr". + The RORG is used if a relocatable origin is already in effect, + otherwise ORG is used. Note that the first example is NOT + equivalent with "DS.B 3" when the rorg is in effect. +#endif + +symbol EQM exp + + The STRING representing the expression is assigned to the + symbol. Occurances of the label in later expressions causes + the string to be evaluated for each occurance. Also used in + conjuction with the DV psuedo-op. + +symbol SET exp + + Same as EQU, but the symbol may be reassigned later. + + MAC name + + Declare a macro. lines between MAC and ENDM are the macro. + You cannot recursively declare a macro. You CAN recursively + use a macro (reference a macro in a macro). No label is + allowed to the left of MAC or ENDM. + + Arguments passed to macros are referenced with: {#}. The first + argument passed to a macro would thus be {1}. You should + always use LOCAL labels (.name) inside macros which you use + more than once. {0} represents an EXACT substitution of the + ENTIRE argument line. + + ENDM + + end of macro def. NO LABEL ALLOWED ON THE LEFT! + + MEXIT + + Used in conjuction with conditionals. Exits the current macro + level. + +[label] IFCONST exp + + Is TRUE if the expression result is defined, FALSE otherwise + and NO error is generated if the expression is undefined. + +[label] IFNCONST exp + + Is TRUE if the expression result is undefined, FALSE otherwise + and NO error is generated if the expression is undefined. + +[label] IF exp + + Is TRUE if the expression result is defined AND non-zero. + Is FALSE if the expression result is defined AND zero. + Neither IF or ELSE will be executed if the expression result + is undefined. If the expression is undefined, another assembly + pass is automatically taken. +#if OlafPhase + If this happens, phase errors in the next pass only will not + be reported unless the verboseness is 1 or more. +#endif + +[label] ELSE + + ELSE the current IF. + +[label] ENDIF +[label] EIF + + Terminate an IF. ENDIF and EIF are equivalent. + +[label] REPEAT exp +[label] REPEND + + Repeat code between REPEAT/REPEND 'exp' times. + +#if DAD + if exp == 0, + the code repeats forever. exp is evaluated once. + + If exp == 0, the repeat loop is ignored. + If exp < 0, a warning “REPEAT parameter < 0 (ignored)” is generated and the repeat loop is ignored. +#endif + + Y SET 0 + REPEAT 10 + X SET 0 + REPEAT 10 + DC X,Y + X SET X + 1 + REPEND + Y SET Y + 1 + REPEND + + generates an output table: 0,0 1,0 2,0 ... 9,0 0,1 1,1 2,1 + ... 9,1, etc... + + Labels within a REPEAT/REPEND should be temporary labels with a + SUBROUTINE pseudo-op to keep them unique. + + The Label to the left of REPEND is assigned AFTER the loop + FINISHES. + + +[label] XXX[.force] operand + + XXX is some mnemonic, not necessarily three characters long. + The .FORCE optional extension is used to force specific + addressing modes (see below). + +[label] LIST ON or OFF + + Globally turns listing on or off, starting with the current + line. +#if OlafList + When you give LOCALON or LOCALOFF the effect is local to the + current macro or included file. For a line to be listed both + the global and local list switches must be on. +#endif + +#if OlafDotop + All pseudo-ops (and incidentally also the mnemonics) can be + prefixed with a . for compatibility with other assemblers. + So .IF is the same as IF. This works only because lone .FORCE + extensions are meaningless. +#endif + +#if OlafFreeFormat + The format of each input line is free: first all leading + spaces are discarded, and the first word is examined. If it + does not look like a directive or opcode (as known at that point), + it is taken as a label. This is sort-of nasty if you like labels + with names like END. + The two xxxFormat options are mutually exclusive. +#endif + +#if OlafHashFormat + With this option an initial # (after optional initial spaces) + turns the next word into a directive/opcode. + A ^ skips more spaces and makes the next word a label. +#endif + +GENERAL: + + The label will be set to the current ORG/RORG either before or after + a pseudo-op is executed. Most of the time, the label to the left of a + pseudo-op is the current ORG/RORG. The following pseudo-op's labels are + created AFTER execution of the pseudo-op: + + SEG, ORG, RORG, REND, ALIGN + +EXTENSIONS: + + FORCE extensions are used to force an addressing mode. In some cases, + you can optimize the assembly to take fewer passes by telling it the + addressing mode. Force extensions are also used with DS,DC, and DV + to determine the element size. NOT ALL EXTENSIONS APPLY TO ALL + PROCESSORS! + + example: lda.z charlie + + i -implied + ind -indirect word + 0 -implied + 0x -implied indexing (0,x) + 0y -implied indexing (0,y) + b -byte address + bx -byte address indexed x + by -byte address indexed y + w -word address + wx -word address indexed x + wy -word address indexed y + l -longword (4 bytes) (DS/DC/DV) + r -relative + u -uninitialized (SEG) + + First character equivalent substitutions: + + b z d (byte, zeropage, direct) + w e a (word, extended, absolute) + + +ASSEMBLER PASSES: + The assembler may have to make several passes through the source + code to resolve all generation. The number of passes is not + limited to two. Since this may result in an unexpected, verbose + option 2, 3, and 4 have been provided to allow determination of the + cause. The assembler will give up if it thinks it can't do the + assembly in *any* number of passes. + + Error reporting could be better.... + +#if OlafPasses + + The check if another pass might resolve the source is pretty good, but + not perfect. You can specify the maximum number of passes to do + (default -p10), and with the -P option you can override the normal check. + This allows the following contrived example to resolve in 12 passes: + + org 1 + repeat [[x < 11] ? [x-11]] + 11 + dc.b x + repend + x: + +#endif + +EXPRESSIONS: + [] may be used to group expressions. The precedense of operators + is the same as for the C language in almost all respects. Use + brackets [] when you are unsure. The reason () cannot be used to + group expressions is due to a conflict with the 6502 and other + assembly languages. +#if OlafBraKet + It is possible to use () instead of [] in expressions following + pseudo-ops, but not following mnemonics. So this works: + if target & (pet3001 | pet4001), but this doesn't: + lda #target & (pet3001 | pet4001). +#endif + + Some operators, such as ||, can return a resolved value even if + one of the expressions is not resolved. Operators are as follows: + + NOTE WELL! Some operations will result in non-byte values when a + byte value was wanted. For example: ~1 is NOT $FF, but + $FFFFFFFF. Preceding it with a < (take LSB of) will solve the + problem. ALL ARITHMETIC IS CARRIED OUT IN 32 BITS. The final + Result will be automatically truncated to the maximum handleable + by the particular machine language (usually a word) when applied + to standard mnemonics. + + prec UNARY + + 20 ~exp one's complement. + 20 -exp negation + 20 !exp not expression (returns 0 if exp non-zero, 1 if exp zero) + 20 exp take MSB byte of an expression + + BINARY + + 19 * multiplication + 19 / division + 19 % mod + 18 + addition + 18 - subtraction + 17 >>,<< shift right, shift left + 16 >,>= greater, greater equal + 16 <,<= smaller, smaller equal + 15 == equal to. Try to use this instead of = + 15 = exactly the same as == (exists compatibility) + 15 != not equal to + 14 & logical and + 13 ^ logical xor + 12 | logical or + 11 && left expression is true AND right expression is true + 10 || left expression is true OR right expression is true + 9 ? if left expression is true, result is right expression, + else result is 0. [10 ? 20] returns 20 + 8 [] group expressions + 7 , separate expressions in list (also used in + addressing mode resolution, BE CAREFUL! + + Note: The effect of the C conditional operator a ? b : c can be + had with [a ? b - c] + c. + + Constants: + + nnn decimal + 0nnn octal + %nnn binary + $nnn hex + 'c character + "cc.." string (NOT zero terminated if in DC/DS/DV) + [exp]d the constant expressions is evaluated and it's decimal + result turned into an ascii string. + + Symbols: + + ... -holds CHECKSUM so far (of actual-generated stuff) + .. -holds evaluated value in DV pseudo op + .name -represents a temporary symbol name. Temporary symbols + may be reused inside MACROS and between SUBROUTINES, but + may not be referenced across macros or across SUBROUTINEs. + . -current program counter (as of the beginning of the + instruction). + name -beginning with an alpha character and containing letters, + numbers, or '_'. Represents some global symbol name. +#if OlafStar + * -synonym for ., when not confused as an operator. +#endif +#if OlafDol + nnn$ -temporary label, much like .name, except that defining + a non-temporary label has the effect that SUBROUTINE + has on .name. They are unique within macros, like + .name. Note that 0$ and 00$ are distinct, as are 8$ + and 010$. + (mainly for compatibility with other assemblers.) +#endif + + +WHY codes: + Each bit in the WHY word (verbose option 1) is a reason (why + the assembler needs to do another pass), as follows: + + bit 0 expression in mnemonic not resolved + 1 - + 2 expression in a DC not resolved + 3 expression in a DV not resolved (probably in DV's EQM symbol) + 4 expression in a DV not resolved (could be in DV's EQM symbol) + 5 expression in a DS not resolved + 6 expression in an ALIGN not resolved + 7 ALIGN: Relocatable origin not known (if in RORG at the time) + 8 ALIGN: Normal origin not known (if in ORG at the time) + 9 EQU: expression not resolved + 10 EQU: value mismatch from previous pass (phase error) + 11 IF: expression not resolved + 12 REPEAT: expression not resolved + + 13 a program label has been defined after it has been + referenced (forward reference) and thus we need another + pass + 14 a program label's value is different from that of the + previous pass (phase error) + + Certain errors will cause the assembly to abort immediately, others + will wait until the current pass is other. The remaining allow another + pass to occur in the hopes the error will fix itself. + + +VERSIONS: + + V2.12 + -Fixed macro naming bug (macros wouldn't work if the name after + the 'mac' was in upper case). + + V2.11 + -Fixed exp bug, exp MSB (it used to + be reversed). + -Fixed many bugs in macros and other things + -Added automatic checksumming ... no more doing checksums manually! + -Added several new processors, including 6502. + -Source is now 16/32 bit int compatible, and will compile on an + IBM-PC (the ultimate portability test) + + V2.01 + + -can now have REPEAT/REPEND's within macros + -fill field for DS.W is a word (used to be a byte fill) + -fill field for DS.L is a long (used to be a byte fill) + + diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/dasm/DasmCompiler.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/dasm/DasmCompiler.java new file mode 100644 index 00000000..33bbfed2 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/dasm/DasmCompiler.java @@ -0,0 +1,50 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.dasm; + +import com.wudsn.ide.asm.compiler.Compiler; +import com.wudsn.ide.asm.compiler.CompilerProcessLogParser; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; + +/** + * Compiler class for DASM. + * + * @author Peter Dell + */ +public final class DasmCompiler extends Compiler { + + /** + * Creates a new instance. + */ + public DasmCompiler() { + } + + @Override + public CompilerSourceParser createSourceParser() { + return new DasmCompilerSourceParser(); + } + + @Override + public CompilerProcessLogParser createLogParser() { + + return new DasmCompilerProcessLogParser(); + } + +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/dasm/DasmCompiler.xml b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/dasm/DasmCompiler.xml new file mode 100644 index 00000000..dccb2126 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/dasm/DasmCompiler.xml @@ -0,0 +1,417 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/dasm/DasmCompilerProcessLogParser.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/dasm/DasmCompilerProcessLogParser.java new file mode 100644 index 00000000..cb7c1700 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/dasm/DasmCompilerProcessLogParser.java @@ -0,0 +1,270 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.dasm; + +import java.io.File; +import java.io.FileInputStream; +import java.io.FileNotFoundException; +import java.util.List; +import java.util.StringTokenizer; +import java.util.regex.Matcher; +import java.util.regex.Pattern; + +import org.eclipse.core.resources.IMarker; +import org.eclipse.core.runtime.CoreException; + +import com.wudsn.ide.asm.compiler.CompilerProcessLogParser; +import com.wudsn.ide.asm.compiler.CompilerSymbol; +import com.wudsn.ide.base.common.FileUtility; +import com.wudsn.ide.base.common.StringUtility; + +/** + * Process log parser for {@link DasmCompiler}. + * + * Sample error message: + * + *
+ * ------- FILE C:\Users\D025328\Documents\Eclipse\workspace.jac\com.wudsn.ide.ref\ASM\Atari2600\DASM\DASM-Error-Reference.asm LEVEL 1 PASS 1
+ *       1  0000 ????
+ * ------- FILE include/DASM-Reference-Source-Include.asm LEVEL 2 PASS 1
+ *       0  0000 ????				      include	"include/DASM-Reference-Source-Include.asm"
+ *       1  0000 ????						;	@com.wudsn.ide.asm.mainsourcefile=../DASM-Error-Reference.asm
+ *       2  0000 ????
+ *       3  0000 ????						;	Reference source include file for DASM
+ *       4  0000 ????
+ * include/DASM-Reference-Source-Include.asm (5): error: Unknown Mnemonic 'jmp'.
+ *       5  0000 ????				      jmp	unknownTest
+ * ------- FILE C:\Users\D025328\Documents\Eclipse\workspace.jac\com.wudsn.ide.ref\ASM\Atari2600\DASM\DASM-Error-Reference.asm
+ *       3  0000 ????
+ * C:\Users\D025328\Documents\Eclipse\workspace.jac\com.wudsn.ide.ref\ASM\Atari2600\DASM\DASM-Error-Reference.asm (4): error: Unknown Mnemonic 'jmp'.
+ *       4  0000 ????				      jmp	nowhere
+ *       5  0000 ????
+ * 
+ * + * @author Peter Dell + */ +final class DasmCompilerProcessLogParser extends CompilerProcessLogParser { + + private Pattern pattern; + + private String listLog; + private String listLogErrorMessage; + private boolean fatalErrorFound; + private boolean unresolvedSymbolsFound; + + @Override + protected void initialize() { + pattern = Pattern.compile(".* (.*): error:"); + + File listFile = new File(files.outputFolder, files.mainSourceFile.fileNameWithoutExtension + ".lst"); + if (listFile.exists()) { + try { + listLog = FileUtility.readString(listFile.getPath(), new FileInputStream(listFile), + FileUtility.MAX_SIZE_UNLIMITED); + listLogErrorMessage = null; + } catch (FileNotFoundException ex) { + listLog = ""; + listLogErrorMessage = ex.getMessage(); + } catch (CoreException ex) { + listLog = ""; + listLogErrorMessage = ex.getStatus().getMessage(); + } + } else { + listLog = ""; + listLogErrorMessage = "Expected list file '" + + listFile.getPath() + + "' does not exist. Check the compiler preferences and make sure you have set the option '-l${outputFilePathWithoutExtension}.lst'."; + } + fatalErrorFound = false; + unresolvedSymbolsFound = false; + } + + @Override + protected void findNextMarker() { + + if (listLogErrorMessage != null) { + filePath = mainSourceFilePath; + lineNumber = 0; + severity = IMarker.SEVERITY_ERROR; + message = listLogErrorMessage; + markerAvailable = true; + listLogErrorMessage = null; + return; + } + + int index; + String line; + Matcher matcher = pattern.matcher(listLog); + + if (matcher.find()) { + index = matcher.start(); + line = listLog.substring(index); + listLog = listLog.substring(matcher.end()); + int numberIndex = line.indexOf(" ("); + + if (numberIndex > 0) { + + filePath = line.substring(0, numberIndex); + + String lineNumberString; + int numberEndIndex = line.indexOf(')'); + if (numberEndIndex > 0) { + lineNumberString = line.substring(numberIndex + 2, numberEndIndex); + } else { + lineNumberString = "-1"; + } + + try { + lineNumber = Integer.parseInt(lineNumberString); + + int nextIndex = line.indexOf("\n"); + if (nextIndex > 0) { + message = line.substring(numberEndIndex + 10, nextIndex - 1); + } + } catch (NumberFormatException ex) { + lineNumber = -1; + message = ex.getMessage(); + } + + severity = IMarker.SEVERITY_ERROR; + message = message.trim(); + markerAvailable = true; + return; + } + } else { + index = outputLog.indexOf("Fatal assembly error: "); + if (index > 0 && !fatalErrorFound) { + int nextIndex = outputLog.indexOf("\n", index); + if (nextIndex > 0) { + message = outputLog.substring(index, nextIndex - 1); + } else { + message = outputLog.substring(index); + } + fatalErrorFound = true; + + severity = IMarker.SEVERITY_ERROR; + message = message.trim(); + markerAvailable = true; + return; + } + + if (fatalErrorFound) { + final String UNRESOLVED = "--- Unresolved Symbol List"; + index = outputLog.lastIndexOf(UNRESOLVED); + if (index > 0) { + unresolvedSymbolsFound = true; + outputLog = outputLog.substring(index + UNRESOLVED.length()).trim(); + } + } + if (unresolvedSymbolsFound) { + index = outputLog.indexOf('\n'); + if (index > 0) { + line = outputLog.substring(0, index - 1); + if (!line.startsWith("--")) { + outputLog = outputLog.substring(index + 1); + severity = IMarker.SEVERITY_ERROR; + index = line.indexOf(" "); + if (index > 0) { + line = line.substring(0, index); + } + message = "Unresolved symbol " + line + "."; + markerAvailable = true; + return; + } + } + } + } + + return; + } + + /** + * Type tokens in labels file:
+ * + * ???? = unknown value
+ * str = symbol is a string
+ * eqm = symbol is an eqm macro
+ * (r) = symbol has been referenced
+ * (s) = symbol created with SET or EQM pseudo-op
+ */ + @Override + public void addCompilerSymbols(List compilerSymbols) { + final String SYMBOLS = "--- Symbol List (sorted by symbol)"; + + String log; + int index; + + log = outputLog; + index = log.indexOf(SYMBOLS); + if (index >= 0) { + log = log.substring(index + SYMBOLS.length()).trim(); + + index = log.indexOf('\n'); + while (index > 0) { + String line = log.substring(0, index - 1); + if (line.startsWith("--- End of Symbol List.")) { + break; + } + StringTokenizer st = new StringTokenizer(line); + String name = st.nextToken(); + String hexValue = ""; + if (st.hasMoreTokens()) { + hexValue = st.nextToken(); + } + + String token = ""; + int valueType = CompilerSymbol.NUMBER; + String stringValue = ""; + while (st.hasMoreTokens()) { + token = st.nextToken(); + // "str" indicates that the symbol is a string value + if (token.equals("str")) { + valueType = CompilerSymbol.STRING; + } + + // String values are enclosed in double quotes. + if (valueType == CompilerSymbol.STRING && token.startsWith("\"") && token.length() >= 2 + && token.endsWith("\"")) { + stringValue = token.substring(1, token.length() - 1); + } + } + + switch (valueType) { + case CompilerSymbol.NUMBER: + // Ignore unnamed symbol with value "0000" + if (StringUtility.isSpecified(hexValue)) { + compilerSymbols.add(CompilerSymbol.createNumberHexSymbol(name, hexValue)); + } + break; + case CompilerSymbol.STRING: + compilerSymbols.add(CompilerSymbol.createStringSymbol(name, stringValue)); + break; + default: + throw new IllegalStateException("Unsupported value type '" + valueType + "'."); + } + + log = log.substring(index).trim(); + index = log.indexOf('\n'); + + } + } + + } +} \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/dasm/DasmCompilerSourceParser.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/dasm/DasmCompilerSourceParser.java new file mode 100644 index 00000000..74098666 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/dasm/DasmCompilerSourceParser.java @@ -0,0 +1,52 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.dasm; + +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; + +/** + * Source parser for {@link DasmCompiler}. + * + * @author Peter Dell + */ +final class DasmCompilerSourceParser extends CompilerSourceParser { + + @Override + protected void parseLine(int startOffset, String symbol, int symbolOffset, String instruction, + int instructionOffset, String operand, String comment) { + + if (symbol.length() > 0) { + + // Check for origin statement + if (symbol.equals("*")) { + beginImplementationSection(startOffset, startOffset + symbolOffset, operand, comment); + + } else { + if (instruction.equals("=")) { + createEquateDefinitionChild(startOffset, startOffset + symbolOffset, symbol, operand, comment); + } else { + createLabelDefinitionChild(startOffset, startOffset + symbolOffset, symbol, comment); + + } + } + + } // Symbol not empty + } +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/kickass/KickAss-Libraries.txt b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/kickass/KickAss-Libraries.txt new file mode 100644 index 00000000..4e3345ae --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/kickass/KickAss-Libraries.txt @@ -0,0 +1,424 @@ +Questions based on your list: +>- There is "ANC_IMM" for "anc" but not "anc2" for "ANC2_IMM"? +Yes thats right, but don't they do the same? + +>- BRA_REL is missing? +From the constants? No, its after RLA_ABSY + + +>- What are the "AT_.." constants? +These are argument type constants. Pseudocommands are a kind of advanced macros that takes instruction arguments. An argument consist of a type (immediate, absolute, etc) and a value and you can manipulate these as you want. Let's say you want to define you own 16bit amgia/pc style move command that can be used like this: + +:mov16 #$1234 ; target // Move immediate to absolute +:mov16 $1000 ; $2000,x // Move from absolute to absolute,x +:mov16 $10,x ; $2000,y // zeropage,x to absolute + +Then you can define it like this: + +.pseudocommand mov16 src;tar { + lda src + sta tar + lda _16bit_nextArgument(src) + sta _16bit_nextArgument(tar) +} + +.function _16bit_nextArgument(arg) { + .if (arg.getType()==AT_IMMEDIATE) .return CmdArgument(arg.getType(),>arg.getValue()) + .return CmdArgument(arg.getType(),arg.getValue()+1) +} + +Btw. Don't worry if you missed a detail in the above. The point is that you use AT_IMMEDIATE constant to detect if the argument is immediate, since you have to treat these differently. + + +>- Is there a list of descriptions & values for the constants? In the content assist, I'd like to display that information. +They are described in the manual. Most of them are seen in tables (left column=constant, right one - description). + + +Hi Peter + +> Why "else" and not ".else"? +Because else is considered part of the .if directive. (In the parse tree the if directive is a node with a true-condition branch and an optional false-condition branch (else)) + +>Is everything case-senstive? +Thats correct + + +Eg: +- All directives start with '.'. For Example: '.byte' '.word' '.import' '.fill' '.for' +- All labels end with ':'. For example: 'label1:' +- All macro / pseudocommand executions start with ':'. For Example: ':MyMacro(27, "Hello")' or ':mov #5 ; $d020' +- Comments are like java / C++ (/* Block comment */, // Line comment) + +The Directives: + +".align" +".assert" +".asserterror" +".by" +".byte" +".const" +".define" +".dw" +".dword" +".easteregg" +".enum" +".error" +".eval" +".filenamespace" +".fill" +".for" +".function" +".if" +".import" +".importonce" +".label" +".macro" +".namespace" +".pc" +".print" +".printnow" +".pseudocommand" +".pseudopc" +".return" +".struct" +".te" +".text" +".var" +".wo" +".word" +"else" + + + +Asm Commands (Including the illegal ones); +"tas"|"php"|"sax"|"rti"|"sta"|"tsx"|"cpy"|"cpx"|"pha"|"bvs"|"bpl"|"dcp"|"adc"| +"tya"|"eor"|"iny"|"inx"|"lax"|"bcs"|"bvc"|"las"|"bit"|"slo"|"txs"|"inc"|"bcc"| +"lsr"|"sre"|"plp"|"txa"|"rla"|"jmp"|"jsr"|"rra"|"pla"|"sei"|"ahx"|"sed"|"sec"| +"ora"|"isc"|"bne"|"and"|"anc"|"cmp"|"bmi"|"asl"|"ldy"|"clv"|"ldx"|"ror"|"cli"| +"alr"|"dey"|"rol"|"cld"|"dex"|"clc"|"arr"|"brk"|"lda"|"shy"|"axs"|"shx"|"sty"| +"stx"|"beq"|"nop"|"dec"|"sbc2"|"sbc"|"tay"|"xaa"|"tax"|"rts" + + +Illegal Ones: +"tas"|"sax"|"dcp"|"lax"|"las"|"slo"|"sre"|"rla"|""rra"|"ahx"|"isc"|"anc" +"alr"|"arr"|"shy"|"axs"|"shx"|"sbc2"|"xaa"| + + +DTV Commands: +"sac"|"sir"|"bra" + +Default macros: +:BasicUpstart(address) +:BasicUpstart2(address) + + +Default Constants: +PI +E +AT_ABSOLUTE +AT_ZEROPAGE +AT_ABSOLUTEX +AT_ABSOLUTEY +AT_IMMEDIATE +AT_INDIRECT +AT_IZEROPAGEX +AT_IZEROPAGEY +AT_NONE +BLACK +WHITE +RED +CYAN +PURPLE +GREEN +BLUE +YELLOW +ORANGE +BROWN +LIGHT_RED +DARK_GRAY +DARK_GREY +GRAY +GREY +LIGHT_GREEN +LIGHT_BLUE +LIGHT_GRAY +LIGHT_GREY +BF_C64FILE +BF_KOALA +BF_FLI +BF_BITMAP_SINGLECOLOR +PLA +SAC_IMM +SLO_ZP +SLO_ZPX +SLO_IZPX +SLO_IZPY +SLO_ABS +SLO_ABSX +SLO_ABSY +PHA +BVC_REL +BRK +CLC +CLD +PLP +EOR_IMM +EOR_ZP +EOR_ZPX +EOR_IZPX +EOR_IZPY +EOR_ABS +EOR_ABSX +EOR_ABSY +INC_ZP +INC_ZPX +INC_ABS +INC_ABSX +PHP +RLA_ZP +RLA_ZPX +RLA_IZPX +RLA_IZPY +RLA_ABS +RLA_ABSX +RLA_ABSY +BRA_REL +JMP_ABS +JMP_IND +BVS_REL +ROR +ROR_ZP +ROR_ZPX +ROR_ABS +ROR_ABSX +CLV +STA_ZP +STA_ZPX +STA_IZPX +STA_IZPY +STA_ABS +STA_ABSX +STA_ABSY +ARR_IMM +SBC_IMM +SBC_ZP +SBC_ZPX +SBC_IZPX +SBC_IZPY +SBC_ABS +SBC_ABSX +SBC_ABSY +CLI +BEQ_REL +LSR +LSR_ZP +LSR_ZPX +LSR_ABS +LSR_ABSX +CPX_IMM +CPX_ZP +CPX_ABS +CPY_IMM +CPY_ZP +CPY_ABS +NOP +BNE_REL +JSR_ABS +ANC_IMM +LDA_IMM +LDA_ZP +LDA_ZPX +LDA_IZPX +LDA_IZPY +LDA_ABS +LDA_ABSX +LDA_ABSY +AND_IMM +AND_ZP +AND_ZPX +AND_IZPX +AND_IZPY +AND_ABS +AND_ABSX +AND_ABSY +RTI +BIT_ZP +BIT_ZPX +BIT_ABS +BIT_ABSX +STX_ZP +STX_ZPY +STX_ABS +TAY +TAX +RTS +TAS_ABSY +SAX_ZP +SAX_ZPY +SAX_IZPX +SAX_ABS +ROL +ROL_ZP +ROL_ZPX +ROL_ABS +ROL_ABSX +INX +INY +STY_ZP +STY_ZPX +STY_ABS +ASL +ASL_ZP +ASL_ZPX +ASL_ABS +ASL_ABSX +TYA +BMI_REL +AHX_IZPY +AHX_ABSY +DEX +DEY +CMP_IMM +CMP_ZP +CMP_ZPX +CMP_IZPX +CMP_IZPY +CMP_ABS +CMP_ABSX +CMP_ABSY +AXS_IMM +ADC_IMM +ADC_ZP +ADC_ZPX +ADC_IZPX +ADC_IZPY +ADC_ABS +ADC_ABSX +ADC_ABSY +TXS +DEC_ZP +DEC_ZPX +DEC_ABS +DEC_ABSX +ISC_ZP +ISC_ZPX +ISC_IZPX +ISC_IZPY +ISC_ABS +ISC_ABSX +ISC_ABSY +ALR_IMM +SRE_ZP +SRE_ZPX +SRE_IZPX +SRE_IZPY +SRE_ABS +SRE_ABSX +SRE_ABSY +TSX +LAS_ABSY +LAX_IMM +LAX_ZP +LAX_ZPY +LAX_IZPX +LAX_IZPY +LAX_ABS +LAX_ABSY +TXA +BCC_REL +SHY_ABSX +SHX_ABSY +RRA_ZP +RRA_ZPX +RRA_IZPX +RRA_IZPY +RRA_ABS +RRA_ABSX +RRA_ABSY +BPL_REL +SEI +XAA_IMM +SIR_IMM +DCP_ZP +DCP_ZPX +DCP_IZPX +DCP_IZPY +DCP_ABS +DCP_ABSX +DCP_ABSY +SED +ORA_IMM +ORA_ZP +ORA_ZPX +ORA_IZPX +ORA_IZPY +ORA_ABS +ORA_ABSX +ORA_ABSY +SEC +ANC2_IMM +BCS_REL +SBC2_IMM +LDY_IMM +LDY_ZP +LDY_ZPX +LDY_ABS +LDY_ABSX +LDX_IMM +LDX_ZP +LDX_ZPY +LDX_ABS +LDX_ABSY + +Default functions: +abs +acos +asin +atan +atan2 +cbrt +ceil +cos +cosh +exp +expm1 +floor +hypot +IEEEremainder +log +log10 +log1p +max +min +pow +random +round +signum +sin +sinh +sqrt +tan +tanh +toDegrees +toRadians +mod +toIntString +toHexString +toOctalString +toBinaryString +Vector +Vector +Matrix +RotationMatrix +ScaleMatrix +MoveMatrix +PerspectiveMatrix +createFile +List +List +Hashtable +CmdArgument +LoadSid +LoadPicture +LoadBinary +asmCommandSize \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/kickass/KickAssCompiler.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/kickass/KickAssCompiler.java new file mode 100644 index 00000000..712a040a --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/kickass/KickAssCompiler.java @@ -0,0 +1,50 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.kickass; + +import com.wudsn.ide.asm.compiler.Compiler; +import com.wudsn.ide.asm.compiler.CompilerProcessLogParser; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; + +/** + * Compiler class for KICKASS. + * + * @author Peter Dell + */ +public final class KickAssCompiler extends Compiler { + + /** + * Creates a new instance. + */ + public KickAssCompiler() { + } + + @Override + public CompilerSourceParser createSourceParser() { + return new KickAssCompilerSourceParser(); + } + + @Override + public CompilerProcessLogParser createLogParser() { + + return new KickAssCompilerProcessLogParser(); + } + +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/kickass/KickAssCompiler.txt b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/kickass/KickAssCompiler.txt new file mode 100644 index 00000000..803a8b0c --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/kickass/KickAssCompiler.txt @@ -0,0 +1,52 @@ + if (title.equals(name)) { + int index = name.indexOf("_"); + String opcode; + String addressing; + if (index >= 0) { + opcode = name.substring(0, index); + + addressing = name.substring(index + 1); + } else { + index=name.length(); + opcode = name; + addressing = ""; + } + + StringBuilder result = new StringBuilder("Opcode "); + for (int i = 0; i < index; i++) { + result.append("_"); + result.append(opcode.charAt(i)); + } + if (addressing.equals("ABS")) { + result.append(" _a_b_solute"); + } else if (addressing.equals("ABSX")) { + result.append(" _a_b_solute,_x"); + } else if (addressing.equals("ABSY")) { + result.append(" _a_b_solute,_y"); + } else if (addressing.equals("IMM")) { + result.append(" #_i_m_mediate"); + } else if (addressing.equals("IZPX")) { + result.append(" (_indirect _zero_page,_x)"); + } else if (addressing.equals("IZPY")) { + result.append(" (_indirect _zero_page),_y"); + } else if (addressing.equals("ZP")) { + result.append(" _zero_page"); + } else if (addressing.equals("ZPX")) { + result.append(" _zero_page,_x"); + } else if (addressing.equals("ZPY")) { + result.append(" _zero_page,_y"); + }else if (addressing.equals("REL")) { + result.append(" _r_e_lative"); + } else if (addressing.equals("IND")) { + result.append(" (_i_n_direct)"); + } else if (addressing.equals("")) { + } else { + throw new RuntimeException( + "Unknown addressing mode " + addressing); + } + title = result.toString(); + } + System.out.println(" "); + } \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/kickass/KickAssCompiler.xml b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/kickass/KickAssCompiler.xml new file mode 100644 index 00000000..8e82a2fc --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/kickass/KickAssCompiler.xml @@ -0,0 +1,400 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/kickass/KickAssCompilerProcessLogParser.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/kickass/KickAssCompilerProcessLogParser.java new file mode 100644 index 00000000..4a2e8f6a --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/kickass/KickAssCompilerProcessLogParser.java @@ -0,0 +1,118 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.kickass; + +import java.io.BufferedReader; +import java.io.IOException; +import java.io.StringReader; +import java.util.List; + +import org.eclipse.core.resources.IMarker; + +import com.wudsn.ide.asm.compiler.CompilerProcessLogParser; +import com.wudsn.ide.asm.compiler.CompilerSymbol; + +/** + * Process log parser for {@link KickAssCompiler}. + * + * Sample error message: + * + *
+ * Error: Unknown symbol 'unknownLabel'
+ * at line 9, column 6 in C:\Users\D025328\Documents\Eclipse\workspace.jac\com.wudsn.ide.ref\ASM\C64\KICKASS\KICKASS-Error-Reference.asm
+ * 
+ * + * @author Peter Dell + */ +final class KickAssCompilerProcessLogParser extends CompilerProcessLogParser { + + private BufferedReader bufferedReader; + + @Override + protected void initialize() { + bufferedReader = new BufferedReader(new StringReader(outputLog)); + } + + @Override + protected void findNextMarker() { + + String line; + line = ""; + + while (line != null && !markerAvailable) { + try { + line = bufferedReader.readLine(); + } catch (IOException ex) { + throw new RuntimeException("Cannot read line", ex); + } + if (line != null) { + String pattern; + int index; + pattern = "Error: "; + severity = IMarker.SEVERITY_ERROR; + index = line.indexOf(pattern); + if (index < 0) { + pattern = "Warning: "; + severity = IMarker.SEVERITY_WARNING; + index = line.indexOf(pattern); + } + if (index == 0) { + + message = line.substring(index + pattern.length()).trim(); + try { + line = bufferedReader.readLine(); + } catch (IOException ex) { + throw new RuntimeException("Cannot read line", ex); + } + if (line != null) { + index = line.indexOf(','); + if (index > 8) { + + String lineNumberString = line.substring(8, + index); + + try { + lineNumber = Integer.parseInt(lineNumberString); + } catch (NumberFormatException ex) { + lineNumber = -1; + severity = IMarker.SEVERITY_ERROR; + message = ex.getMessage(); + } + } else { + lineNumber = -1; + } + index = line.indexOf(" in "); + if (index > 0) { + filePath = line.substring(index + 4); + } + } + markerAvailable = true; + } + + } + } + } + + @Override + public void addCompilerSymbols(List compilerSymbols) { + + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/kickass/KickAssCompilerSourceParser.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/kickass/KickAssCompilerSourceParser.java new file mode 100644 index 00000000..b4e8b83f --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/kickass/KickAssCompilerSourceParser.java @@ -0,0 +1,74 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.kickass; + +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; + +/** + * Source parser for {@link KickAssCompiler}. + * + * @author Peter Dell + */ +final class KickAssCompilerSourceParser extends CompilerSourceParser { + + @Override + protected void parseLine(int startOffset, String symbol, int symbolOffset, String instruction, + int instructionOffset, String operand, String comment) { + + int symbolLength = symbol.length(); + if (symbolLength > 0) { + + if (symbol.charAt(symbolLength - 1) == ':') { + symbol = symbol.substring(0, symbolLength - 1); + } + if (instruction.equals("=")) { + createEquateDefinitionChild(startOffset, startOffset + symbolOffset, symbol, operand, comment); + } else { + createLabelDefinitionChild(startOffset, startOffset + symbolOffset, symbol, comment); + + } + + } // Symbol not empty + + // TODO Make .VAR an own type of instruction. Same code as in + // MadsCompilerSourceParser! + if (instruction.equals(".var")) { + operand = operand.trim(); + int index = operand.indexOf('='); + String variable; + String value; + if (index < 0) { + variable = operand; + value = ""; + } else { + variable = operand.substring(0, index).trim(); + value = operand.substring(index).trim(); + if (value.startsWith("=")) { + value = value.substring(1).trim(); + } + } + if (value.length() > 0) { + createEquateDefinitionChild(startOffset, startOffset + instructionOffset, variable, value, comment); + } else { + createLabelDefinitionChild(startOffset, startOffset + instructionOffset, variable, comment); + } + } + } +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/mads/MadsCompiler.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/mads/MadsCompiler.java new file mode 100644 index 00000000..e13a55f7 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/mads/MadsCompiler.java @@ -0,0 +1,50 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.mads; + +import com.wudsn.ide.asm.compiler.Compiler; +import com.wudsn.ide.asm.compiler.CompilerProcessLogParser; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; + +/** + * Compiler class for MADS. + * + * @author Peter Dell + */ +public final class MadsCompiler extends Compiler { + + /** + * Creates a new instance. + */ + public MadsCompiler() { + } + + @Override + public CompilerSourceParser createSourceParser() { + return new MadsCompilerSourceParser(); + } + + @Override + public final CompilerProcessLogParser createLogParser() { + + return new MadsCompilerProcessLogParser(); + } + +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/mads/MadsCompiler.xml b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/mads/MadsCompiler.xml new file mode 100644 index 00000000..ef47be91 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/mads/MadsCompiler.xml @@ -0,0 +1,909 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/mads/MadsCompilerProcessLogParser.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/mads/MadsCompilerProcessLogParser.java new file mode 100644 index 00000000..7f8ed397 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/mads/MadsCompilerProcessLogParser.java @@ -0,0 +1,228 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.mads; + +import java.io.BufferedReader; +import java.io.File; +import java.io.IOException; +import java.io.StringReader; +import java.util.List; + +import org.eclipse.core.resources.IMarker; +import org.eclipse.core.runtime.CoreException; + +import com.wudsn.ide.asm.AssemblerPlugin; +import com.wudsn.ide.asm.compiler.CompilerProcessLogParser; +import com.wudsn.ide.asm.compiler.CompilerSymbol; +import com.wudsn.ide.asm.compiler.CompilerSymbolType; +import com.wudsn.ide.base.common.FileUtility; + +/** + * Process log parser for {@link MadsCompiler}. + * + * @author Peter Dell + */ +final class MadsCompilerProcessLogParser extends CompilerProcessLogParser { + + private BufferedReader bufferedReader; + + @Override + protected void initialize() { + bufferedReader = new BufferedReader(new StringReader(outputLog)); + } + + @Override + protected void findNextMarker() { + + String line; + line = ""; + + while (line != null && !markerAvailable) { + try { + line = bufferedReader.readLine(); + } catch (IOException ex) { + throw new RuntimeException("Cannot read line", ex); + } + if (line != null) { + String pattern; + int index; + pattern = ") ERROR: "; + severity = IMarker.SEVERITY_ERROR; + index = line.indexOf(pattern); + if (index < 2) { + pattern = ") WARNING: "; + severity = IMarker.SEVERITY_WARNING; + index = line.indexOf(pattern); + } + if (index > 2) { + + int i = index - 2; + while (line.charAt(i) != '(' && i >= 0) { + i--; + } + + if (line.charAt(i) == '(') { + String lineNumberString = line.substring(i + 1, index); + + try { + lineNumber = Integer.parseInt(lineNumberString); + } catch (NumberFormatException ex) { + lineNumber = -1; + severity = IMarker.SEVERITY_ERROR; + message = ex.getMessage(); + } + } else { + lineNumber = -1; + } + message = line.substring(index + pattern.length()).trim(); + + filePath = line.substring(0, i - 1); + markerAvailable = true; + } + + } + } + } + + /** + *
+     * From the MADS documentation
+     * 
+     * .LAB file format
+     * 
+     * As with XASM, the *.LAB file stores information about labels in the program. There are three columns:
+     * • The first column is the virtual bank number assigned to the label (if <>0).
+     * • The second column is the label value.
+     * • The third column is the label name.
+     * 
+     * Virtual bank numbers with values >= $FFF9 have special meanings:
+     * • $FFF9   label for parameter in procedure defined by .PROC
+     * • $FFFA   label for array defined by .ARRAY
+     * • $FFFB   label for structured data defined by the pseudo-command DTA STRUCT_LABEL
+     * • $FFFC   label for SpartaDOS X symbol defined by SMB
+     * • $FFFD   label for macro defined by .MACRO directive
+     * • $FFFE   label for structure defined by .STRUCT directive
+     * • $FFFF   label for procedure defined by .PROC directive
+     * 
+     * 
+     * Characters with special meanings in label names: 
+     * • label with two colons :: is defined in a macro
+     * • a dot ('.') separates the name of a scope (.MACRO, .PROC, .LOCAL, .STRUCT) from the field name in the scope
+     * 
+     * The numeric value after :: is the number of the macro call. This is an example file:
+     * 
+     * Mad-Assembler v1.4.2beta by TeBe/Madteam
+     * Label table:
+     * 00	0400	@STACK_ADDRESS
+     * 00	00FF	@STACK_POINTER
+     * 00	2000	MAIN
+     * 00	2019	LOOP
+     * 00	201C	LOOP::1
+     * 00	201C	LHEX
+     * 00	0080	LHEX.HLP
+     * 00	204C	LHEX.THEX
+     * 00	205C	HEX
+     * 00	205C	HEX.@GETPAR0.LOOP
+     * 00	2079	HEX.@GETPAR1.LOOP
+     * 
+ */ + @Override + // TODO Parser symbol type and add source tree object type for .DEFs + public void addCompilerSymbols(List compilerSymbols) throws CoreException { + if (compilerSymbols == null) { + throw new IllegalArgumentException("Parameter 'compilerSymbols' must not be null."); + } + String labelsFilePath = files.outputFilePathWithoutExtension + ".lab"; + File labelsFile = new File(labelsFilePath); + if (labelsFile.exists()) { + + String labelsFileContent = FileUtility.readString(labelsFile, FileUtility.MAX_SIZE_UNLIMITED); + String[] lines = labelsFileContent.split("[\\r\\n]+"); + if (lines.length > 2 || lines[0].toLowerCase().startsWith("mads") + || lines[1].toLowerCase().startsWith("label table:")) { + for (int i = 2; i < lines.length; i++) { + String[] parts = lines[i].split("\\t"); + if (parts.length == 3) { + int type = CompilerSymbolType.LABEL_DEFINITION; + String bankString = parts[0]; + + String name = parts[2]; + String valueString = parts[1]; + try { + long bank = Long.parseLong(bankString, 16); + int symbolBank; + if (bank >= 0 && bank < 0xfff9) { + symbolBank = (int) bank; + } else { + symbolBank = CompilerSymbol.UNDEFINED_BANK; + if (bank == 0xfff9) { + // Label for parameter in procedure defined + // by .PROC + // TODO: This would actually be a separate + // type + type = CompilerSymbolType.PROCEDURE_DEFINITION_SECTION; + } else if (bank == 0xfffa) { + // Label for array defined by .ARRAY + // TODO: This would actually be a separate + // type + type = CompilerSymbolType.LABEL_DEFINITION; + } else if (bank == 0xfffb) { + // Label for structured data defined by the + // pseudo-command DTA STRUCT_LABEL + type = CompilerSymbolType.STRUCTURE_DEFINITION_SECTION; + } else if (bank == 0xfffc) { + // Label for SpartaDOS X symbol defined by + // SMB + // TODO: This would actually be a separate + // type + type = CompilerSymbolType.LABEL_DEFINITION; + } else if (bank == 0xfffd) { + // Label for macro defined by .MACRO + // directive + // TODO: This would actually be a separate + // type + type = CompilerSymbolType.MACRO_DEFINITION_SECTION; + } else if (bank == 0xfffe) { + // Label for structure defined by .STRUCT + // directive + type = CompilerSymbolType.STRUCTURE_DEFINITION_SECTION; + } else if (bank == 0xffff) { + // Label for procedure defined by .PROC + // directive + type = CompilerSymbolType.PROCEDURE_DEFINITION_SECTION; + } + } + long value = Long.parseLong(valueString, 16); + CompilerSymbol compilerSymbol = CompilerSymbol.createNumberSymbol(type, name, symbolBank, + value); + compilerSymbols.add(compilerSymbol); + } catch (NumberFormatException ex) { + AssemblerPlugin.getInstance().logError("Cannot parse value {1} of symbol {0}.", + new Object[] { name, valueString }, ex); + } + + } + } + } + + } + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/mads/MadsCompilerSourceParser.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/mads/MadsCompilerSourceParser.java new file mode 100644 index 00000000..21132606 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/mads/MadsCompilerSourceParser.java @@ -0,0 +1,70 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.mads; + +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; + +/** + * Source parser for {@link MadsCompiler}. + * + * @author Peter Dell + */ +final class MadsCompilerSourceParser extends CompilerSourceParser { + + @Override + protected void parseLine(int startOffset, String symbol, int symbolOffset, String instruction, + int instructionOffset, String operand, String comment) { + + if (symbol.length() > 0) { + + if (instruction.equals("=") || instruction.equals("EQU")) { + createEquateDefinitionChild(startOffset, startOffset + symbolOffset, symbol, operand, comment); + + } else { + createLabelDefinitionChild(startOffset, startOffset + symbolOffset, symbol, comment); + } + + } // Symbol not empty + + // TODO Make .VAR an own type of instruction + if (instruction.equals(".VAR") || instruction.equals(".ZPVAR")) { + operand = operand.trim(); + int index = operand.indexOf('='); + String variable; + String value; + if (index < 0) { + variable = operand; + value = ""; + } else { + variable = operand.substring(0, index).trim(); + value = operand.substring(index).trim(); + if (value.startsWith("=")) { + value = value.substring(1).trim(); + } + } + if (value.length() > 0) { + createEquateDefinitionChild(startOffset, startOffset + instructionOffset, variable, value, comment); + } else { + createLabelDefinitionChild(startOffset, startOffset + instructionOffset, variable, comment); + } + + } + } +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/merlin32/Merlin32Compiler.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/merlin32/Merlin32Compiler.java new file mode 100644 index 00000000..2084fc90 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/merlin32/Merlin32Compiler.java @@ -0,0 +1,50 @@ +/** +* Copyright (C) 2009 - 2016 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.merlin32; + +import com.wudsn.ide.asm.compiler.Compiler; +import com.wudsn.ide.asm.compiler.CompilerProcessLogParser; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; + +/** + * Compiler class for MERLIN32. + * + * @author Peter Dell + */ +public final class Merlin32Compiler extends Compiler { + + /** + * Creates a new instance. + */ + public Merlin32Compiler() { + } + + @Override + public CompilerSourceParser createSourceParser() { + return new Merlin32CompilerSourceParser(); + } + + @Override + public final CompilerProcessLogParser createLogParser() { + + return new Merlin32CompilerProcessLogParser(); + } + +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/merlin32/Merlin32Compiler.xml b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/merlin32/Merlin32Compiler.xml new file mode 100644 index 00000000..6bb205de --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/merlin32/Merlin32Compiler.xml @@ -0,0 +1,80 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/merlin32/Merlin32CompilerProcessLogParser.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/merlin32/Merlin32CompilerProcessLogParser.java new file mode 100644 index 00000000..5baabfe4 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/merlin32/Merlin32CompilerProcessLogParser.java @@ -0,0 +1,180 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.merlin32; + +import java.io.BufferedReader; +import java.io.IOException; +import java.io.StringReader; +import java.util.List; + +import org.eclipse.core.resources.IMarker; +import org.eclipse.core.runtime.CoreException; + +import com.wudsn.ide.asm.compiler.CompilerProcessLogParser; +import com.wudsn.ide.asm.compiler.CompilerSymbol; + +/** + * Process log parser for {@link Merlin32Compiler}. + * + * @author Peter Dell + */ +final class Merlin32CompilerProcessLogParser extends CompilerProcessLogParser { + + private BufferedReader bufferedReader; + + @Override + protected void initialize() { + bufferedReader = new BufferedReader(new StringReader(outputLog)); + } + + @Override + protected void findNextMarker() { + + String line; + line = ""; + + while (line != null && !markerAvailable) { + try { + line = bufferedReader.readLine(); + } catch (IOException ex) { + throw new RuntimeException("Cannot read line", ex); + } + if (line != null) { + String pattern; + int index; + pattern = ") ERROR: "; + severity = IMarker.SEVERITY_ERROR; + index = line.indexOf(pattern); + if (index < 2) { + pattern = ") WARNING: "; + severity = IMarker.SEVERITY_WARNING; + index = line.indexOf(pattern); + } + if (index > 2) { + + int i = index - 2; + while (line.charAt(i) != '(' && i >= 0) { + i--; + } + + if (line.charAt(i) == '(') { + String lineNumberString = line.substring(i + 1, index); + + try { + lineNumber = Integer.parseInt(lineNumberString); + } catch (NumberFormatException ex) { + lineNumber = -1; + severity = IMarker.SEVERITY_ERROR; + message = ex.getMessage(); + } + } else { + lineNumber = -1; + } + message = line.substring(index + pattern.length()).trim(); + + filePath = line.substring(0, i - 1); + markerAvailable = true; + } + + } + } + } + @Override + public void addCompilerSymbols(List compilerSymbols) throws CoreException { + if (compilerSymbols == null) { + throw new IllegalArgumentException("Parameter 'compilerSymbols' must not be null."); + } +// String labelsFilePath = files.outputFilePathWithoutExtension + ".lab"; +// File labelsFile = new File(labelsFilePath); +// if (labelsFile.exists()) { +// +// String labelsFileContent = FileUtility.readString(labelsFile, FileUtility.MAX_SIZE_UNLIMITED); +// String[] lines = labelsFileContent.split("[\\r\\n]+"); +// if (lines.length > 2 || lines[0].toLowerCase().startsWith("mads") +// || lines[1].toLowerCase().startsWith("label table:")) { +// for (int i = 2; i < lines.length; i++) { +// String[] parts = lines[i].split("\\t"); +// if (parts.length == 3) { +// int type = CompilerSymbolType.LABEL_DEFINITION; +// String bankString = parts[0]; +// +// String name = parts[2]; +// String valueString = parts[1]; +// try { +// long bank = Long.parseLong(bankString, 16); +// int symbolBank; +// if (bank >= 0 && bank < 0xfff9) { +// symbolBank = (int) bank; +// } else { +// symbolBank = CompilerSymbol.UNDEFINED_BANK; +// if (bank == 0xfff9) { +// // Label for parameter in procedure defined +// // by .PROC +// // TODO: This would actually be a separate +// // type +// type = CompilerSymbolType.PROCEDURE_DEFINITION_SECTION; +// } else if (bank == 0xfffa) { +// // Label for array defined by .ARRAY +// // TODO: This would actually be a separate +// // type +// type = CompilerSymbolType.LABEL_DEFINITION; +// } else if (bank == 0xfffb) { +// // Label for structured data defined by the +// // pseudo-command DTA STRUCT_LABEL +// type = CompilerSymbolType.STRUCTURE_DEFINITION_SECTION; +// } else if (bank == 0xfffc) { +// // Label for SpartaDOS X symbol defined by +// // SMB +// // TODO: This would actually be a separate +// // type +// type = CompilerSymbolType.LABEL_DEFINITION; +// } else if (bank == 0xfffd) { +// // Label for macro defined by .MACRO +// // directive +// // TODO: This would actually be a separate +// // type +// type = CompilerSymbolType.MACRO_DEFINITION_SECTION; +// } else if (bank == 0xfffe) { +// // Label for structure defined by .STRUCT +// // directive +// type = CompilerSymbolType.STRUCTURE_DEFINITION_SECTION; +// } else if (bank == 0xffff) { +// // Label for procedure defined by .PROC +// // directive +// type = CompilerSymbolType.PROCEDURE_DEFINITION_SECTION; +// } +// } +// long value = Long.parseLong(valueString, 16); +// CompilerSymbol compilerSymbol = CompilerSymbol.createNumberSymbol(type, name, symbolBank, +// value); +// compilerSymbols.add(compilerSymbol); +// } catch (NumberFormatException ex) { +// AssemblerPlugin.getInstance().logError("Cannot parse value {1} of symbol {0}.", +// new Object[] { name, valueString }, ex); +// } +// +// } +// } +// } +// +// } + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/merlin32/Merlin32CompilerSourceParser.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/merlin32/Merlin32CompilerSourceParser.java new file mode 100644 index 00000000..7e00fd21 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/merlin32/Merlin32CompilerSourceParser.java @@ -0,0 +1,55 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.merlin32; + +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; + +/** + * Source parser for {@link Merlin32Compiler}. + * + * @author Peter Dell + */ +final class Merlin32CompilerSourceParser extends CompilerSourceParser { + + @Override + protected void parseLine(int startOffset, String symbol, int symbolOffset, String instruction, + int instructionOffset, String operand, String comment) { + + if (symbol.length() > 0) { + + if (instruction.equals("=") || instruction.equals("EQU")) { + createEquateDefinitionChild(startOffset, startOffset + symbolOffset, symbol, operand, comment); + + } else { + createLabelDefinitionChild(startOffset, startOffset + symbolOffset, symbol, comment); + } + + } // Symbol not empty + + // TODO Implement macro and Lup handling +// if (instruction.equals("<<<<")) { +// endMacroDefinition(); +// } else if (instruction.equals(">>>>")) { +// // Same as "PMC" +// } else if (instruction.equals("--^")) { +// endRepeatSection(); +// } + } +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/tass/TassCompiler.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/tass/TassCompiler.java new file mode 100644 index 00000000..97f95609 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/tass/TassCompiler.java @@ -0,0 +1,50 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.tass; + +import com.wudsn.ide.asm.compiler.Compiler; +import com.wudsn.ide.asm.compiler.CompilerProcessLogParser; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; + +/** + * Compiler class for ATASM. + * + * @author Peter Dell + */ +public final class TassCompiler extends Compiler { + + /** + * Creates a new instance. + */ + public TassCompiler() { + } + + @Override + public CompilerSourceParser createSourceParser() { + return new TassCompilerSourceParser(); + } + + @Override + public CompilerProcessLogParser createLogParser() { + + return new TassCompilerProcessLogParser(); + } + +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/tass/TassCompiler.xml b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/tass/TassCompiler.xml new file mode 100644 index 00000000..6d4188ad --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/tass/TassCompiler.xml @@ -0,0 +1,20 @@ + + + + + + + \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/tass/TassCompilerProcessLogParser.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/tass/TassCompilerProcessLogParser.java new file mode 100644 index 00000000..cac2205c --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/tass/TassCompilerProcessLogParser.java @@ -0,0 +1,171 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.tass; + +import java.util.List; +import java.util.StringTokenizer; +import java.util.regex.Matcher; +import java.util.regex.Pattern; + +import org.eclipse.core.resources.IMarker; + +import com.wudsn.ide.asm.compiler.CompilerProcessLogParser; +import com.wudsn.ide.asm.compiler.CompilerSymbol; + +/** + * Process log parser for {@link TassCompiler}. + * + * + * TODO Implement TASS Support + * + * @author Peter Dell + */ +final class TassCompilerProcessLogParser extends CompilerProcessLogParser { + + private Pattern pattern; + private String sourceFilePattern; + + @Override + protected void initialize() { + pattern = Pattern.compile("In .* line "); + sourceFilePattern = "In " + mainSourceFilePath + ", line "; + } + + @Override + protected void findNextMarker() { + + int index; + String line; + boolean include; + String includeFile; + Matcher matcher = pattern.matcher(errorLog); + + if (matcher.find()) { + index = matcher.start(); + line = errorLog.substring(index); + if (line.startsWith(sourceFilePattern)) { + include = false; + includeFile = ""; + } else { + include = true; + includeFile = line.substring(3, matcher.end() - matcher.start() + - 7); + } + index = matcher.end(); + String lineNumberLine = errorLog.substring(index); + errorLog = lineNumberLine; + int numberEndIndex = lineNumberLine.indexOf("--"); + if (numberEndIndex > 0) { + String lineNumberString; + lineNumberString = lineNumberLine.substring(0, numberEndIndex); + + try { + lineNumber = Integer.parseInt(lineNumberString); + int nextIndex = lineNumberLine.indexOf("\n"); + if (index > 0) { + message = lineNumberLine.substring(nextIndex + 1); + int nextIndex2 = message.indexOf("\n"); + if (nextIndex > 0) { + message = message.substring(0, nextIndex2 - 1); + } + message = message.trim(); + } + } catch (NumberFormatException ex) { + lineNumber = -1; + severity = IMarker.SEVERITY_ERROR; + message = ex.getMessage(); + } + } + + if (message.startsWith("Error:")) { + severity = IMarker.SEVERITY_ERROR; + message = message.substring(6); + } else if (message.startsWith("Warning:")) { + severity = IMarker.SEVERITY_WARNING; + message = message.substring(8); + } + + if (include) { + if (lineNumber >= 0) { + message = includeFile + " line " + lineNumber + ": " + + message; + } else { + message = includeFile + ": " + message; + } + lineNumber = -1; + } + message = message.trim(); + + // Message mapping. + if (severity == IMarker.SEVERITY_WARNING + && message.startsWith("Using bank")) { + severity = IMarker.SEVERITY_INFO; + } + markerAvailable = true; + } + } + + @Override + public void addCompilerSymbols(List compilerSymbols) { + final String EQUATES = "Equates:"; + final String SYMBOL = "Symbol"; + final String TABLE = "table:"; + + String log; + int index; + + log = outputLog; + index = log.indexOf(EQUATES); + if (index >= 0) { + log = log.substring(index + EQUATES.length()); + + StringTokenizer st = new StringTokenizer(log); + String token; + String name; + String hexValue; + while (st.hasMoreTokens()) { + token = st.nextToken(); + if (token.equals(SYMBOL)) { + break; + } + name = token.substring(0, token.length() - 1); + hexValue = st.nextToken(); // Must be there + compilerSymbols.add(CompilerSymbol.createNumberHexSymbol(name, hexValue)); + } + + if (st.hasMoreTokens()) { + token = st.nextToken(); + if (token.equals(TABLE)) { + while (st.hasMoreTokens()) { + token = st.nextToken(); + if (!token.endsWith(":")) { + break; + } + name = token.substring(0, token.length() - 1); + hexValue = st.nextToken(); // Must be there + compilerSymbols.add(CompilerSymbol.createNumberHexSymbol(name, hexValue)); + } + } + } + } + + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/tass/TassCompilerSourceParser.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/tass/TassCompilerSourceParser.java new file mode 100644 index 00000000..94188246 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/tass/TassCompilerSourceParser.java @@ -0,0 +1,52 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.tass; + +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; + +/** + * Source parser for {@link TassCompiler}. + * + * @author Peter Dell + */ +final class TassCompilerSourceParser extends CompilerSourceParser { + + @Override + protected void parseLine(int startOffset, String symbol, int symbolOffset, String instruction, + int instructionOffset, String operand, String comment) { + + if (symbol.length() > 0) { + + // Check for origin statement + if (symbol.equals("*")) { + beginImplementationSection(startOffset, startOffset + symbolOffset, operand, comment); + + } else { + if (instruction.equals("=")) { + createEquateDefinitionChild(startOffset, startOffset + symbolOffset, symbol, operand, comment); + } else { + createLabelDefinitionChild(startOffset, startOffset + symbolOffset, symbol, comment); + + } + } + + } // Symbol not empty + } +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/xasm/XasmCompiler.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/xasm/XasmCompiler.java new file mode 100644 index 00000000..b6eac70e --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/xasm/XasmCompiler.java @@ -0,0 +1,58 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.xasm; + +import com.wudsn.ide.asm.compiler.Compiler; +import com.wudsn.ide.asm.compiler.CompilerProcessLogParser; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; + +/** + * Compiler class for XASM. + * + * @author Peter Dell + */ +public final class XasmCompiler extends Compiler { + + /** + * Creates a new instance. + */ + public XasmCompiler() { + + } + + @Override + public CompilerSourceParser createSourceParser() { + return new XasmCompilerSourceParser(); + } + + @Override + public boolean isSuccessExitValue(int exitValue) { + // XASM returns 0 if there is no warning, 1 if there are only warnings, + // 2 if there is at least one error. + return exitValue == 0 || exitValue == 1; + } + + @Override + public CompilerProcessLogParser createLogParser() { + + return new XasmCompilerProcessLogParser(); + } + +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/xasm/XasmCompiler.xml b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/xasm/XasmCompiler.xml new file mode 100644 index 00000000..14cb8415 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/xasm/XasmCompiler.xml @@ -0,0 +1,268 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/xasm/XasmCompilerProcessLogParser.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/xasm/XasmCompilerProcessLogParser.java new file mode 100644 index 00000000..fec3ba1b --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/xasm/XasmCompilerProcessLogParser.java @@ -0,0 +1,104 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.xasm; + +import java.io.BufferedReader; +import java.io.IOException; +import java.io.StringReader; +import java.util.List; + +import org.eclipse.core.resources.IMarker; + +import com.wudsn.ide.asm.compiler.CompilerProcessLogParser; +import com.wudsn.ide.asm.compiler.CompilerSymbol; + +/** + * Process log parser for {@link XasmCompiler}. Identical to + * MadsCompilerProcessLogParser, except that it is based on the errorLog. + * + * @author Peter Dell + */ +final class XasmCompilerProcessLogParser extends CompilerProcessLogParser { + + private BufferedReader bufferedReader; + + @Override + protected void initialize() { + bufferedReader = new BufferedReader(new StringReader(errorLog)); + } + + @Override + protected void findNextMarker() { + + String line; + line = ""; + + while (line != null && !markerAvailable) { + try { + line = bufferedReader.readLine(); + } catch (IOException ex) { + throw new RuntimeException("Cannot read line", ex); + } + if (line != null) { + String pattern; + int index; + pattern = ") ERROR: "; + severity = IMarker.SEVERITY_ERROR; + index = line.indexOf(pattern); + if (index < 2) { + pattern = ") WARNING: "; + severity = IMarker.SEVERITY_WARNING; + index = line.indexOf(pattern); + } + if (index > 2) { + + int i = index - 2; + while (line.charAt(i) != '(' && i >= 0) { + i--; + } + + if (line.charAt(i) == '(') { + String lineNumberString = line.substring(i + 1, index); + + try { + lineNumber = Integer.parseInt(lineNumberString); + } catch (NumberFormatException ex) { + lineNumber = -1; + severity = IMarker.SEVERITY_ERROR; + message = ex.getMessage(); + } + } else { + lineNumber = -1; + } + message = line.substring(index + pattern.length()).trim(); + filePath = line.substring(0, i - 1); + markerAvailable = true; + } + + } + } + } + + @Override + public void addCompilerSymbols(List compilerSymbols) { + + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/xasm/XasmCompilerSourceParser.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/xasm/XasmCompilerSourceParser.java new file mode 100644 index 00000000..1aacc209 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/compiler/xasm/XasmCompilerSourceParser.java @@ -0,0 +1,53 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.xasm; + +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; + +/** + * Source parser for {@link XasmCompiler}. + * + * @author Peter Dell + */ +final class XasmCompilerSourceParser extends CompilerSourceParser { + + /** + * Creation is package local. + */ + XasmCompilerSourceParser() { + + } + + @Override + protected void parseLine(int startOffset, String symbol, int symbolOffset, String instruction, + int instructionOffset, String operand, String comment) { + if (symbol.length() > 0) { + + if (instruction.equals("EQU")) { + createEquateDefinitionChild(startOffset, startOffset + symbolOffset, symbol, operand, comment); + + } else { + createLabelDefinitionChild(startOffset, startOffset + symbolOffset, symbol, comment); + } + + } // Symbol not empty + } + +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/acme/AcmeEditor.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/acme/AcmeEditor.java new file mode 100644 index 00000000..7e03eab7 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/acme/AcmeEditor.java @@ -0,0 +1,37 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.editor.acme; + +import com.wudsn.ide.asm.compiler.CompilerId; +import com.wudsn.ide.asm.editor.AssemblerEditor; + +public final class AcmeEditor extends AssemblerEditor { + + /** + * Creation is public. Called by the extension "org.eclipse.ui.editors". + */ + public AcmeEditor() { + } + + @Override + public String getCompilerId() { + return CompilerId.ACME; + } +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/asm6/Asm6Editor.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/asm6/Asm6Editor.java new file mode 100644 index 00000000..525506ea --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/asm6/Asm6Editor.java @@ -0,0 +1,37 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.editor.asm6; + +import com.wudsn.ide.asm.compiler.CompilerId; +import com.wudsn.ide.asm.editor.AssemblerEditor; + +public final class Asm6Editor extends AssemblerEditor { + + /** + * Creation is public. Called by the extension "org.eclipse.ui.editors". + */ + public Asm6Editor() { + } + + @Override + public String getCompilerId() { + return CompilerId.ASM6; + } +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/atasm/AtasmEditor.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/atasm/AtasmEditor.java new file mode 100644 index 00000000..d89fb2b5 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/atasm/AtasmEditor.java @@ -0,0 +1,38 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.editor.atasm; + +import com.wudsn.ide.asm.compiler.CompilerId; +import com.wudsn.ide.asm.editor.AssemblerEditor; + +public final class AtasmEditor extends AssemblerEditor { + + /** + * Creation is public. Called by the extension "org.eclipse.ui.editors". + */ + public AtasmEditor() { + } + + + @Override + public String getCompilerId() { + return CompilerId.ATASM; + } +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/dasm/DasmEditor.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/dasm/DasmEditor.java new file mode 100644 index 00000000..de597fc6 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/dasm/DasmEditor.java @@ -0,0 +1,37 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.editor.dasm; + +import com.wudsn.ide.asm.compiler.CompilerId; +import com.wudsn.ide.asm.editor.AssemblerEditor; + +public final class DasmEditor extends AssemblerEditor { + + /** + * Creation is public. Called by the extension "org.eclipse.ui.editors". + */ + public DasmEditor() { + } + + @Override + public String getCompilerId() { + return CompilerId.DASM; + } +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/kickass/KickAssEditor.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/kickass/KickAssEditor.java new file mode 100644 index 00000000..f5edf71c --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/kickass/KickAssEditor.java @@ -0,0 +1,38 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.editor.kickass; + +import com.wudsn.ide.asm.compiler.CompilerId; +import com.wudsn.ide.asm.editor.AssemblerEditor; + +public final class KickAssEditor extends AssemblerEditor { + + /** + * Creation is public. Called by the extension "org.eclipse.ui.editors". + */ + public KickAssEditor() { + + } + + @Override + public String getCompilerId() { + return CompilerId.KICKASS; + } +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/mads/MadsEditor.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/mads/MadsEditor.java new file mode 100644 index 00000000..006cfa3a --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/mads/MadsEditor.java @@ -0,0 +1,38 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.editor.mads; + +import com.wudsn.ide.asm.compiler.CompilerId; +import com.wudsn.ide.asm.editor.AssemblerEditor; + +public final class MadsEditor extends AssemblerEditor { + + /** + * Creation is public. Called by the extension "org.eclipse.ui.editors". + */ + public MadsEditor() { + + } + + @Override + public String getCompilerId() { + return CompilerId.MADS; + } +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/merlin32/Merlin32Editor.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/merlin32/Merlin32Editor.java new file mode 100644 index 00000000..5b693050 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/merlin32/Merlin32Editor.java @@ -0,0 +1,38 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.editor.merlin32; + +import com.wudsn.ide.asm.compiler.CompilerId; +import com.wudsn.ide.asm.editor.AssemblerEditor; + +public final class Merlin32Editor extends AssemblerEditor { + + /** + * Creation is public. Called by the extension "org.eclipse.ui.editors". + */ + public Merlin32Editor() { + + } + + @Override + public String getCompilerId() { + return CompilerId.MERLIN32; + } +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/tass/TassEditor.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/tass/TassEditor.java new file mode 100644 index 00000000..9a12756d --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/tass/TassEditor.java @@ -0,0 +1,38 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.editor.tass; + +import com.wudsn.ide.asm.compiler.CompilerId; +import com.wudsn.ide.asm.editor.AssemblerEditor; + +public final class TassEditor extends AssemblerEditor { + + /** + * Creation is public. Called by the extension "org.eclipse.ui.editors". + */ + public TassEditor() { + + } + + @Override + public String getCompilerId() { + return CompilerId.TASS; + } +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/xasm/XasmEditor.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/xasm/XasmEditor.java new file mode 100644 index 00000000..1e713dfe --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/editor/xasm/XasmEditor.java @@ -0,0 +1,38 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.editor.xasm; + +import com.wudsn.ide.asm.compiler.CompilerId; +import com.wudsn.ide.asm.editor.AssemblerEditor; + +public final class XasmEditor extends AssemblerEditor { + + /** + * Creation is public. Called by the extension "org.eclipse.ui.editors". + */ + public XasmEditor() { + + } + + @Override + public String getCompilerId() { + return CompilerId.XASM; + } +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesApple2CompilersPage.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesApple2CompilersPage.java new file mode 100644 index 00000000..b57f5e55 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesApple2CompilersPage.java @@ -0,0 +1,42 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ +package com.wudsn.ide.asm.preferences; + +import com.wudsn.ide.asm.Hardware; +import com.wudsn.ide.asm.preferences.AssemblerPreferencesCompilersPage; + +/** + * Visual editor page for the assembler preferences regarding Apple 2 compilers. + * + * @author Peter Dell + * + */ +public final class AssemblerPreferencesApple2CompilersPage extends + AssemblerPreferencesCompilersPage { + + /** + * Create is public. Used by extension point + * "org.eclipse.ui.preferencePages". + */ + public AssemblerPreferencesApple2CompilersPage() { + super(Hardware.APPLE2); + + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesAtari2600CompilersPage.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesAtari2600CompilersPage.java new file mode 100644 index 00000000..df0a6dcb --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesAtari2600CompilersPage.java @@ -0,0 +1,42 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ +package com.wudsn.ide.asm.preferences; + +import com.wudsn.ide.asm.Hardware; +import com.wudsn.ide.asm.preferences.AssemblerPreferencesCompilersPage; + +/** + * Visual editor page for the assembler preferences regarding Apple 2 compilers. + * + * @author Peter Dell + * + */ +public final class AssemblerPreferencesAtari2600CompilersPage extends + AssemblerPreferencesCompilersPage { + + /** + * Create is public. Used by extension point + * "org.eclipse.ui.preferencePages". + */ + public AssemblerPreferencesAtari2600CompilersPage() { + super(Hardware.ATARI2600); + + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesAtari7800CompilersPage.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesAtari7800CompilersPage.java new file mode 100644 index 00000000..190d8f7e --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesAtari7800CompilersPage.java @@ -0,0 +1,42 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ +package com.wudsn.ide.asm.preferences; + +import com.wudsn.ide.asm.Hardware; +import com.wudsn.ide.asm.preferences.AssemblerPreferencesCompilersPage; + +/** + * Visual editor page for the assembler preferences regarding Apple 2 compilers. + * + * @author Peter Dell + * + */ +public final class AssemblerPreferencesAtari7800CompilersPage extends + AssemblerPreferencesCompilersPage { + + /** + * Create is public. Used by extension point + * "org.eclipse.ui.preferencePages". + */ + public AssemblerPreferencesAtari7800CompilersPage() { + super(Hardware.ATARI7800); + + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesAtari8CompilersPage.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesAtari8CompilersPage.java new file mode 100644 index 00000000..9d207e4b --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesAtari8CompilersPage.java @@ -0,0 +1,43 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ +package com.wudsn.ide.asm.preferences; + +import com.wudsn.ide.asm.Hardware; +import com.wudsn.ide.asm.preferences.AssemblerPreferencesCompilersPage; + +/** + * Visual editor page for the assembler preferences regarding Atari 8-bit + * compilers. + * + * @author Peter Dell + * + */ +public final class AssemblerPreferencesAtari8CompilersPage extends + AssemblerPreferencesCompilersPage { + + /** + * Create is public. Used by extension point + * "org.eclipse.ui.preferencePages". + */ + public AssemblerPreferencesAtari8CompilersPage() { + super(Hardware.ATARI8BIT); + + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesC64CompilersPage.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesC64CompilersPage.java new file mode 100644 index 00000000..71396673 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesC64CompilersPage.java @@ -0,0 +1,42 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ +package com.wudsn.ide.asm.preferences; + +import com.wudsn.ide.asm.Hardware; +import com.wudsn.ide.asm.preferences.AssemblerPreferencesCompilersPage; + +/** + * Visual editor page for the assembler preferences regarding C64 compilers. + * + * @author Peter Dell + * + */ +public final class AssemblerPreferencesC64CompilersPage extends + AssemblerPreferencesCompilersPage { + + /** + * Create is public. Used by extension point + * "org.eclipse.ui.preferencePages". + */ + public AssemblerPreferencesC64CompilersPage() { + super(Hardware.C64); + + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesNESCompilersPage.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesNESCompilersPage.java new file mode 100644 index 00000000..7cf380d8 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesNESCompilersPage.java @@ -0,0 +1,42 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ +package com.wudsn.ide.asm.preferences; + +import com.wudsn.ide.asm.Hardware; +import com.wudsn.ide.asm.preferences.AssemblerPreferencesCompilersPage; + +/** + * Visual editor page for the assembler preferences regarding NES compilers. + * + * @author Peter Dell + * + */ +public final class AssemblerPreferencesNESCompilersPage extends + AssemblerPreferencesCompilersPage { + + /** + * Create is public. Used by extension point + * "org.eclipse.ui.preferencePages". + */ + public AssemblerPreferencesNESCompilersPage() { + super(Hardware.NES); + + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/runner/atari8/Altirra.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/runner/atari8/Altirra.java new file mode 100644 index 00000000..fbcf910b --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/runner/atari8/Altirra.java @@ -0,0 +1,78 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.runner.atari8; + +import java.io.File; + +import org.eclipse.core.resources.IMarker; +import org.eclipse.core.runtime.CoreException; +import org.eclipse.debug.core.model.IBreakpoint; + +import com.wudsn.ide.asm.compiler.CompilerFiles; +import com.wudsn.ide.asm.editor.AssemblerBreakpoint; +import com.wudsn.ide.asm.runner.Runner; + +/** + * Runner for Altirra 2.0 and above which support source line breakpoints. + * + * @author Peter Dell + * @since 1.6.1 + */ +public final class Altirra extends Runner { + + @Override + public File createBreakpointsFile(CompilerFiles files) { + if (files == null) { + throw new IllegalArgumentException("Parameter 'files' must not be null."); + } + return new File(files.outputFilePathWithoutExtension + ".atdbg"); + } + + @Override + public int createBreakpointsFileContent(AssemblerBreakpoint[] breakpoints, StringBuilder breakpointBuilder) { + if (breakpoints == null) { + throw new IllegalArgumentException("Parameter 'breakpoints' must not be null."); + } + int activeBreakpoints = 0; + breakpointBuilder.append(".sourcemode on\n"); + breakpointBuilder.append(".echo\n"); + breakpointBuilder.append(".echo \"Loading executable...\"\n"); + breakpointBuilder.append(".echo\n"); + breakpointBuilder.append("bc *\n"); + breakpointBuilder.append(".onexerun .echo \"Launching executable...\"\n"); + for (IBreakpoint breakpoint : breakpoints) { + try { + if (breakpoint.isEnabled()) { + AssemblerBreakpoint assemberBreakpoint = (AssemblerBreakpoint) breakpoint; + IMarker marker = breakpoint.getMarker(); + String sourceFilePath = marker.getResource().getLocation().toOSString(); + breakpointBuilder + .append("bp \"`" + sourceFilePath + ":" + assemberBreakpoint.getLineNumber() + "`\"\n"); + activeBreakpoints++; + } + } catch (CoreException ex) { + throw new RuntimeException(ex); + } + + } + return activeBreakpoints; + } + +} diff --git a/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/runner/atari8/Atari800Win.java b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/runner/atari8/Atari800Win.java new file mode 100644 index 00000000..690b0c71 --- /dev/null +++ b/com.wudsn.ide.asm.compilers/src/com/wudsn/ide/asm/runner/atari8/Atari800Win.java @@ -0,0 +1,32 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.runner.atari8; + +import com.wudsn.ide.asm.runner.Runner; + +/** + * Runner for Atar800Win. + * + * @author Peter Dell + * @since 1.7.0. + */ +public final class Atari800Win extends Runner { + +} diff --git a/com.wudsn.ide.asm/.classpath b/com.wudsn.ide.asm/.classpath new file mode 100644 index 00000000..8a8f1668 --- /dev/null +++ b/com.wudsn.ide.asm/.classpath @@ -0,0 +1,7 @@ + + + + + + + diff --git a/com.wudsn.ide.asm/.project b/com.wudsn.ide.asm/.project new file mode 100644 index 00000000..b53cba16 --- /dev/null +++ b/com.wudsn.ide.asm/.project @@ -0,0 +1,28 @@ + + + com.wudsn.ide.asm + + + + + + org.eclipse.jdt.core.javabuilder + + + + + org.eclipse.pde.ManifestBuilder + + + + + org.eclipse.pde.SchemaBuilder + + + + + + org.eclipse.pde.PluginNature + org.eclipse.jdt.core.javanature + + diff --git a/com.wudsn.ide.asm/.settings/org.eclipse.core.resources.prefs b/com.wudsn.ide.asm/.settings/org.eclipse.core.resources.prefs new file mode 100644 index 00000000..e0a59bd1 --- /dev/null +++ b/com.wudsn.ide.asm/.settings/org.eclipse.core.resources.prefs @@ -0,0 +1,4 @@ +#Thu Jun 28 19:08:33 CEST 2012 +eclipse.preferences.version=1 +encoding//help/create-links.bat=ISO-8859-1 +encoding/help=UTF-8 diff --git a/com.wudsn.ide.asm/.settings/org.eclipse.jdt.core.prefs b/com.wudsn.ide.asm/.settings/org.eclipse.jdt.core.prefs new file mode 100644 index 00000000..54e493c0 --- /dev/null +++ b/com.wudsn.ide.asm/.settings/org.eclipse.jdt.core.prefs @@ -0,0 +1,11 @@ +eclipse.preferences.version=1 +org.eclipse.jdt.core.compiler.codegen.inlineJsrBytecode=enabled +org.eclipse.jdt.core.compiler.codegen.targetPlatform=1.6 +org.eclipse.jdt.core.compiler.codegen.unusedLocal=preserve +org.eclipse.jdt.core.compiler.compliance=1.6 +org.eclipse.jdt.core.compiler.debug.lineNumber=generate +org.eclipse.jdt.core.compiler.debug.localVariable=generate +org.eclipse.jdt.core.compiler.debug.sourceFile=generate +org.eclipse.jdt.core.compiler.problem.assertIdentifier=error +org.eclipse.jdt.core.compiler.problem.enumIdentifier=error +org.eclipse.jdt.core.compiler.source=1.6 diff --git a/com.wudsn.ide.asm/.settings/org.eclipse.jdt.ui.prefs b/com.wudsn.ide.asm/.settings/org.eclipse.jdt.ui.prefs new file mode 100644 index 00000000..9985b1d5 --- /dev/null +++ b/com.wudsn.ide.asm/.settings/org.eclipse.jdt.ui.prefs @@ -0,0 +1,4 @@ +#Wed Feb 08 19:13:41 CET 2006 +eclipse.preferences.version=1 +internal.default.compliance=default +org.eclipse.jdt.ui.text.custom_code_templates= diff --git a/com.wudsn.ide.asm/META-INF/MANIFEST.MF b/com.wudsn.ide.asm/META-INF/MANIFEST.MF new file mode 100644 index 00000000..0ce7ce4d --- /dev/null +++ b/com.wudsn.ide.asm/META-INF/MANIFEST.MF @@ -0,0 +1,81 @@ +Manifest-Version: 1.0 +Bundle-ManifestVersion: 2 +Bundle-Name: WUDSN IDE Assembler Plug-in +Bundle-SymbolicName: com.wudsn.ide.asm;singleton:=true +Bundle-Version: 1.7.0.qualifier +Bundle-Activator: com.wudsn.ide.asm.AssemblerPlugin +Bundle-Localization: plugin +Require-Bundle: org.eclipse.core.runtime;visibility:=reexport, + org.eclipse.core.resources;visibility:=reexport, + org.eclipse.core.filesystem, + org.eclipse.jface.text;visibility:=reexport, + org.eclipse.ui;visibility:=reexport, + org.eclipse.ui.console, + org.eclipse.ui.editors;visibility:=reexport, + org.eclipse.ui.ide, + org.eclipse.ui.workbench.texteditor, + org.eclipse.ui.views, + org.eclipse.e4.ui.di;bundle-version="1.0.0", + org.eclipse.e4.ui.model.workbench;bundle-version="1.0.1", + org.eclipse.help, + com.wudsn.ide.base;visibility:=reexport, + com.wudsn.ide.dsk, + com.wudsn.ide.gfx, + org.eclipse.debug.core;visibility:=reexport, + org.eclipse.debug.ui;visibility:=reexport, + org.eclipse.ui.workbench +Bundle-ActivationPolicy: lazy +Bundle-Vendor: Peter Dell +Bundle-RequiredExecutionEnvironment: JavaSE-1.6 +Export-Package: com.wudsn.ide.asm; + uses:="org.eclipse.osgi.util, + org.eclipse.core.runtime, + com.wudsn.ide.base.common, + org.osgi.framework, + com.wudsn.ide.asm.runner, + com.wudsn.ide.asm.preferences, + com.wudsn.ide.asm.compiler", + com.wudsn.ide.asm.compiler; + uses:="org.eclipse.jface.text, + org.eclipse.ui.console, + org.eclipse.jface.viewers, + com.wudsn.ide.asm, + org.eclipse.jface.text.rules", + com.wudsn.ide.asm.compiler.parser, + com.wudsn.ide.asm.editor; + uses:="org.eclipse.jface.text, + org.eclipse.core.runtime, + org.eclipse.core.commands, + org.eclipse.jface.text.rules, + org.eclipse.debug.core.model, + org.eclipse.jface.action, + org.eclipse.ui.views.contentoutline, + org.eclipse.core.resources, + org.eclipse.jface.viewers, + com.wudsn.ide.asm, + com.wudsn.ide.asm.compiler, + com.wudsn.ide.asm.preferences, + org.eclipse.jface.text.hyperlink, + org.eclipse.ui.texteditor, + org.eclipse.jface.text.reconciler, + org.eclipse.jface.text.source, + org.eclipse.jface.resource, + org.eclipse.ui, + org.eclipse.jface.text.contentassist, + org.eclipse.debug.ui.actions, + org.eclipse.jface.preference, + org.eclipse.jface.text.presentation, + org.eclipse.swt.graphics, + org.eclipse.jface.util, + org.eclipse.swt.widgets, + org.eclipse.ui.editors.text", + com.wudsn.ide.asm.preferences; + uses:="org.eclipse.jface.text, + org.eclipse.jface.preference, + org.eclipse.ui, + org.eclipse.swt.graphics, + org.eclipse.jface.viewers, + com.wudsn.ide.asm, + org.eclipse.core.runtime.preferences, + org.eclipse.swt.widgets", + com.wudsn.ide.asm.runner;uses:="com.wudsn.ide.asm" diff --git a/com.wudsn.ide.asm/WUDSN-IDE (de).launch b/com.wudsn.ide.asm/WUDSN-IDE (de).launch new file mode 100644 index 00000000..481beba3 --- /dev/null +++ b/com.wudsn.ide.asm/WUDSN-IDE (de).launch @@ -0,0 +1,61 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/com.wudsn.ide.asm/WUDSN-IDE (pl).launch b/com.wudsn.ide.asm/WUDSN-IDE (pl).launch new file mode 100644 index 00000000..a6360938 --- /dev/null +++ b/com.wudsn.ide.asm/WUDSN-IDE (pl).launch @@ -0,0 +1,61 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/com.wudsn.ide.asm/WUDSN-IDE.launch b/com.wudsn.ide.asm/WUDSN-IDE.launch new file mode 100644 index 00000000..66779f82 --- /dev/null +++ b/com.wudsn.ide.asm/WUDSN-IDE.launch @@ -0,0 +1,56 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/com.wudsn.ide.asm/bin/.gitignore b/com.wudsn.ide.asm/bin/.gitignore new file mode 100644 index 00000000..43e58b99 --- /dev/null +++ b/com.wudsn.ide.asm/bin/.gitignore @@ -0,0 +1 @@ +/com diff --git a/com.wudsn.ide.asm/build.properties b/com.wudsn.ide.asm/build.properties new file mode 100644 index 00000000..21624a63 --- /dev/null +++ b/com.wudsn.ide.asm/build.properties @@ -0,0 +1,16 @@ +source.. = src/ +output.. = bin/ +bin.includes = META-INF/,\ + .,\ + plugin.xml,\ + plugin.properties,\ + icons/,\ + help/,\ + src/,\ + plugin_de_DE.properties,\ + build.properties,\ + .classpath,\ + .project,\ + schema/ +bin.excludes = help/productions/java/ide/downloads/ + diff --git a/com.wudsn.ide.asm/help/create-links.bat b/com.wudsn.ide.asm/help/create-links.bat new file mode 100644 index 00000000..2711a1d8 --- /dev/null +++ b/com.wudsn.ide.asm/help/create-links.bat @@ -0,0 +1,21 @@ +echo off +rem +rem The image and file resource files for the IDE help are original in the "site\productions\java\ide" folder. +rem The are linked into the "com.wudsn.ide.asm/help/productions/java/ide project" folder via a symbolic link. +rem This way, the Eclipse build uses the latest versions automatically. +rem The HTML files for the online help are original in the com.wudsn.ide.asm/help" folder. +rem They have to be copied into the Joomla using the "export.bat" script when something is changed. +rem +rem Important: Run this script in an Administrator shell. +rem + +echo on +setlocal +set WORKSPACE=C:\jac\system\Java\Programming\Workspaces\WUDSN-IDE +set SITE=C:\jac\system\WWW\Sites\www.wudsn.com +set SYMBOLIC_LINK=%WORKSPACE%\com.wudsn.ide.asm\help\productions\java\ide +set REAL_FOLDER=%SITE%\productions\java\ide +echo on +rmdir %SYMBOLIC_LINK% +mklink /D %SYMBOLIC_LINK% %REAL_FOLDER% +pause diff --git a/com.wudsn.ide.asm/help/ide-credits.section.html b/com.wudsn.ide.asm/help/ide-credits.section.html new file mode 100644 index 00000000..c7aa0923 --- /dev/null +++ b/com.wudsn.ide.asm/help/ide-credits.section.html @@ -0,0 +1,204 @@ + +A project like WUDSN IDE is not possible and not worth anything without the contributions provided by others. So here's +the list of credits of all involved people and related projects. +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
CompilerContributor
+ ACME + Krzysztof Dabrowski (BruSH/ElysiuM)
+ ATASM + M. Schmelzenbach (schmelze)
+ DASM + Peter H. Froehlich (pfh)
+ KickAss + Mads Nielsen (Slammer)
+ MADS + Tomasz Biela (tebe)
+ XASM + Piotr Fusik (fox)
Linux compilingCarsten Strohmann (cas)
Mac OS X compilingSpookt
Graphic library or toolContributor
+ FAIL + Piotr Fusik (fox)
+ Grafx2 + Adrien Destugues (PulkoMandy), Yves Rizoud (yrizoud)
Sound libraryContributor
+ ASAP + Piotr Fusik (fox)
+ JSIDPLAY2 + Ken + Händel + (kenchis) +
File system libraryContributor
+ AppleCommander + Robert Greene (robgreene)
EmulatorContributor
+ AppleWin (Apple II) + Nick Westgate (sicklittlemonkey)
+ Altirra (Atari 8-bit) + Avery Lee (phaeron)
+ Atari800Win (Atari 8-bit) + Marcin Lewandowski (jaskier)
+ CCS64 (C64) + Håkan Sundell (phs)
+ JACE (Apple II) + Brendan Robert (BLuRry)
+ Stella (Atari VCS) + Stephen Anthony, Bradford Mott, Eckhard Stolberg, Brian Watson
+ Virtu (Apple II) + Sean Fausett (fool)
Eclipse plugin or libraryContributor
+ Eclipse + Eclipse Foundation
+ x86 ASM Plugin + Andy Reek, Daniel Mitte
+ Rhino JavaScript engine + Mozilla Foundation
+ Eclipse Hex Editor Plugin (used as inspiration) + Marcel Palko (randallco)
+ Java Hex Editor & Plugin + Pordi Estaqual (pestatije)
+ Java Expression Language Parser + Aaron Gadberry (aaron)
diff --git a/com.wudsn.ide.asm/help/ide-faq.section.html b/com.wudsn.ide.asm/help/ide-faq.section.html new file mode 100644 index 00000000..f48f1653 --- /dev/null +++ b/com.wudsn.ide.asm/help/ide-faq.section.html @@ -0,0 +1,500 @@ + +Here you can find the answers to some frequently asked questions. If your question is not answered here, please have a look at +the video tutorials or contact me. + +

Installation

+
+

How to I install Eclipse at all?

+

+ If you are not familiar with Eclipse at all, make sure that you have installed the + Eclipse platform distribution + only - without Java or J2EE tools. This distribution is much smaller (normally around 50-70 MB instead of 170 MD) and will not + confuse you with tons of features and buttons you do not need at all. See the section "Installing Eclipse" on the tab + "Installation" for the required steps. If you use Windows, you can use one of the zero installation distributions of WUDSN IDE + which are linked on the tab "Installation". They are simply ".zip" archives that already contain Eclipse and everything else." +

+
+
+

When I try to start Eclipse is get an error like "Failed to load the JNI shared library 'C:\Program Files + (x86)\Java\jre6\bin\client\jvm.dll'". What is wrong? +

+

The Eclipse version and the Java version on your system are not compatible. The Eclipse is not pure Java, but uses platform + specific native libraries to run and debug Java efficiently. For example you must install the 64-bit version Java (JRE or JDK) + if you want to use the 64-bit version of the Eclipse. This is a frequent issue under Windows 7, because be default there is + only the 32-bit version of Java installed. See the section "Installing Eclipse" on the tab "Installation" for the required + steps. +

+
+
+

How to I use Eclipse at all?

+

Start the build-in help of Eclipse via the menu "Help/Help Content" and read the section "Workbench User Guide". +

+ Workbench user guide +
+
+

Why is WUDSN IDE not available via the update site?

+

Most likely you typed in the wrong update site URL, for example using "wusdn" instead of "wudsn". The correct URL is + "http://www.wudsn.com/update". In addition you should disable the checkbox "Hide items that are already installed", so you see + what really is there. See the section "Installing WUDSN IDE" the tab "Installation" for the required steps. +

+ Installation dialog with update site +
+
+

Why do I get "Unable to read repository at ... Read timed out" when accessing the update site?

+

This error message indicates the the Eclipse program is somehow blocked from accessing the site. If you are behind a proxy + server, check the general proxy server setttings in the Eclipse preferences. If you use a firewall or internet security tool, + make sure "Eclipse.exe", "java.exe", "javaw.exe" or the corresponding program on your host platform are allowed to cannot to + the internet. Maybe you have to change the settings, so you are prompted to allow access interactively. +

+
+
+

Why is the "Assembler" section not visible in the preferences?

+

+ If the WUDSN IDE specific sections and features are not visible after a successful installation, you are probably using an + outdated Java version. For example Java 1.6 is + out of maintenance since 2013/02 + . This seems to be a common problem on Mac OS X even in Mavericks (10.9), which still uses Java 1.6 by default. Make sure you + have at least the Java version that is mentioned in the installation section for the IDE installed and that Eclipse is + actually started using that version. +

+
+

Configuration

+
+

Why do I see wrong messages in the "Problems" view?

+

The default configuration of the "Problems" view show all error from all files in the current project. While this is a good + default for Java programming, it does not fit at all for compiling single independent assembler files. Therefore you have to + configure the "Problems" view accordingly. See the section "Installing Eclipse" on the tab "Installation" for the required + steps. +

+
+
+

How to I associate my source file extensions with the correct editor?

+

+ The IDE support many different compilers and provides a specialized editor for each of them. Typically you have some preferred + source file extension (".asm" or ".a") and a preferred compiler. The procedure to associate the file extension with the editor + via the preferences is described in this video tutorial + WUDSN IDE Tutorial 3: Setting up Editors and File Extensions correctly + . +

+
+
+

Why do I have to put ;@com.wudsn.ide.asm.hardware=...in the source file?

+

The association with the file extension with the editor for your compiler (that is done in the preferences, see before) + does not determine for which platform you want to create output. Therefore this additional annotation in the main source file + is required used to tell the IDE which is the target platform. It is used to fine the correct compiler and emulator setting, + which can be different per platform. Every compiler has a default platform (see the online help in the IDE) but can also be + used to every other platform. Therefore you have to specify the target platform in the main source file, if you use a + non-default platform. +

+
+

Editing

+
+

Why is editing sometimes slow or even everything is blocked showing the wait cursor?

+

The core of WUDSN IDE uses the Eclipse Platform Runtime only and does not require any additional plugins. It starts and runs + very fast with that configuration and I personally used it one daily basis. So if you experience performance problems, try to + download and run the zero installation distribution of WUDSN IDE. Performance problem they are very likely caused by + additional plugins or themes installed. Often these plugins are not only slow, but broken. Check the ".metadata/.log" file in + the workspace folder. In some Eclipse versions, this is also available via "Window/Show View.../Error Log". +

+
+
+

Is there support for source version control?

+

Yes, there are several plugins available to connect Eclipse to CVS or subversion. Also the "Local History" feature is + installed by default. You can configure it in the preferences. It automatically records all changes to the source file and + let's you compare versions in-place. +

+ Source version control +
+
+

Is there support for block selection mode?

+

Yes, there is a toolbar button and the shortcut "ALT+SHIFT+A" to toggle block selection mode in all text editors. This can + be very useful for adding and removing common prefixes such as line numbers. +

+ Toggle block selection mode +

In case the toolbar button is not visible, you have to set it to visible via the menu entry "Customize Perspective" in the + context menu of the main toolbar. In the customizing dialog you have to activate the commend group "Editor Presentation" and + then the toolbar entries you want to see. +

+ Toggle block selection mode +
+
+

Why does CTRL-Space not open content assist?

+

There is a known key conflict when using Messenger Plus Live! v4.85.0.386 with Microsoft Messenger 2009 on Windows 7 + Ultimate. This may also occur in other version of course. Justin Payne has provided the following description of the solution. +

+
    +
  1. Start up and log into MS Messenger.
  2. +
  3. From the main window, hit the ALT key to bring up the main menu and select "Plus! | Preferences & Options". +
  4. +
  5. From the Preferences Windows, Select the Messenger tab and uncheck "Activate Messenger Lock with a system-wide shortcut" + OR change the value in it's text box to something other than "CTRL + Space" +
  6. +
  7. Select OK button.
  8. +
+
+

Why do CTRL-SHIFT-0/9 and other key combinations not work?

+

You probably have another program outside of Eclipse that has already captured these keys or key combinations. A frequent + problem is the Windows Input Methods Editor (IME) which is used to switch keyboard layouts. For example if you are using + multiple keyboard layouts, the CTRL-Space is mapped to allow you to cycle between the different keyboard regional layouts. + You should probably be aware of how to turn the feature off since you're probably using this features, but if you don't... +

+

Windows 7

+
    +
  1. Within "Windows Control Panel", open "Region and Settings".
  2. +
  3. Select the "Keyboard and Languages" tab, select "Change Keyboards...".
  4. +
  5. In the "Text Services and Input Languages" windows, select "Advanced Key Settings".
  6. +
  7. In the "Hot Keys for input languages" list box, select "Between input languages" and then select "Change Key Sequence..." +
  8. +
  9. In the "Change Key Sequence" window, choose another radio button other than the one next to CTRL+Shift. At best you + choose "(None)". +
  10. +
  11. Click OK until you closed all popup windows.
  12. +
+

Windows 10

+
    +
  1. Open "Control Panel\Clock, Language, and Region\Language\Advanced settings.
  2. +
  3. Click "Change Language bar hot keys".
  4. +
  5. Set all key sequences to "(None)".
  6. +
  7. Click OK until you closed all popup windows.
  8. +
+

+ Of course, this is Windows 7/10 and we know how Microsoft loves to change their layouts and names, but for fact this option + is available back to Windows XP. +
+ Configure Windows IME hot keys +

+
+

Compiling

+
+

Why is MADS the primary compiler?

+

When I started with WUDSN IDE, ATASM was the first supported compiler. The reason was simple: 90% of my sources are in + ATASM format. ATASM is very comprehensive and fast. Its capabilities to define constants an byte sequence is very complete + (".BYTE", ".WORD", ".DBYTE", ".FLOAT", ."SBYTE" for ATASCII, ".CBYTE" for terminated strings, separate offset for all + constants). Over time additional platform compilers haven been added and support for them will be completed step by step. + When the support for MADS was rather complete, I found that it is the most powerful compiler I have ever seen and used. The + support for ".PROC/.ENDPROC" has revolutionized the way I write assembler code now. It allows logical structuring and + visibility control without any runtime overhead. At the same time MADS is compatible to MAC/65 and XASM and even ATASM + sources can be adapted to MADS with a few minor changes described below. Therefore MADS is the primary compiler since WUDSN + IDE version 1.6.0. +

+
+
+

Why do I get the error "No ORG defined" when compiling the example from the tutorial?

+

Since WUDSN IDE version 1.6.0 MADS is the primary compiler which is registered for the file extensions ".asm" upon + installation. You are trying to run the code example for version 1.5.0 or before, which is in ATASM format. Therefore you can + either +

+ +
+
+

Why are the errors and warnings from an include file assigned to the main source file in the problems view?

+

You use a case-insensitive file system and have used different upper or lower case writing in the source include statement + than in the actual file system. For example you have written "ICL 'example.asm'" for a file named "Example.asm" on the file + system. In Eclipse the file names of all resources are treated as case-sensitive, even if the underlying file system is + case-insensitive. Therefore the file name issued by the compiler will no match with the file name of the source include. As a + fall-back, the IDE assigns the error message to the main source file. To fix this, you have to adapt the spelling of the file + name in the source include statement. +

+
+
+

How to I convert an ATASM source for to MADS format?

+

Because both ATASM and MADS syntax are based on the MAC/65 syntax, there are not really many differences. Therefore manual + conversion using "Find/Replace (CTRL-F)" is very easy. +

+
    +
  • Replace the origin definition "* = address" by "ORG address".
  • +
  • Replace ".INCLUDE" by "ICL" for source includes.
  • +
  • Replace ".INCBIN" by "INS" for binary includes.
  • +
  • Replace "* = $2E0;.WORD address" by "RUN address" to specify the run address.
  • +
  • Replace "* = $2E2;.WORD address" by "INI address" to specify the init address.
  • +
  • Check the quotes of ".BYTE" and "DTA" statements. In MADS, single quotes result in ASCII codes, double quotes in ATARI + screen codes. +
  • +
  • Remove all ".BANK" statements. While ATASM sorts and merges all segments within one bank statement, MADS simply uses the + order of segments as defined in the source file. Every "ORG" statement automatically generates new file segment. +
  • +
+
+
+

How does ATASM generate segments in executable files?

+

A helpful feature for small projects is that by default ATASM sorts the segments by address and warns if the same address + is overwritten by code or data. Since version 1.05 the ".BANK" directive is available, which allows you to create COM files + with "INITAD" segments and arbitrary segment counts. If you don't use the ".BANK" directive, ATASM will sort the segments by + their address and will put consecutive blocks into a single segment by default. While this saves some bytes, it might be + confusing if you are used to other assemblers. Note that you have to use the ".SET 6" directive to set the assembler origin + offset in every bank if you use it in one of the banks. +

+
+ ; Bank 0 +
+ .bank +
+ .set 6,0 +
+ * = $8000 +
+ start lda #0 +
+ jmp * +
+
+ ; Bank 1 +
+ .bank .set 6,0 +
+ * = $2e0 +
+ .word start; +
+
+ ; Bank 2 +
+ .bank +
+ .set 6,$4300-$C000 +
+ * = $C000 +
+ lda #1 +
+ sta label+1 +
+ label lda #2 +
+ jmp * +
+
+
+
+

How do I compile into ROM images?

+

Plain ROM Images do not have header bytes by default, or at least they do not have the same header bytes as executable + files. In order to create raw object files without headers, compiler specific options have to be used. Some cases are listed + below. See the manual of the specific compiler for more details. +

+
    +
  • + ACME: Use the compiler parameter " + -f plain + " instead of "-f cbm" (default) to switch to "plain" mode without header +
  • +
  • + ATASM: Extend the compiler parameter "-o${outputFilePath}" to " + -r + -o${outputFilePath}" to switch to "raw" mode without header +
  • +
  • + MADS: Use " + OPT h-f+ + " at the very beginning of the source file disables header and enabled "fill" mode, i.e. no segments are created if there + are gaps in the object code +
  • +
+
+
+

How do I compile into disk images?

+
Atari 8-bit
+

+ For Atari 8-bit, the ATASM compiler has dedicated parameter to write the executable file directly into ".ATR" or ".XFD" disk + images. The disk image must be formatted with Atari DOS 2.0S, Atari DOS DOS 2.5 or a compatible DOS. All Atari 8-bit disk + formats can be created using the "dir2atr.exe" tool of the + AtariSIO tools + by Matthias Reichl (hias). The tool can create a complete disk image with arbitrary DOS (Atari DOS 2.5, MyDos, SpartaDOS) and + size based on a folder which contains "DOS.SYS", "DUP.SYS" (or the equivalent files of the respective DOS) and all other + files requires. I have packaged an example including the "dir2atr" tool, a batch script to call the tool and the emulator and + the "files" folder in this + archive + . Unpack the archive to your output folder. Double-click "makefile.bat" to see how the disk image is created and started. + Read section + How to run a makefile script instead of an emulator? + for the details how to configure the call to "makefile.bat". For productive usage you should of course put the "hias" folder + into a central location and use the most recent version from hias' site. For MacOS X users, the download also contains a + "makefile.sh" script and MacOS X binaries of Matthias Reichl's tools. The binaries have been provided by Fredrick Holst + (freetz) and you can find the latest versions on his + web site + . +

+
Apple II
+

+ For Apple II, WUDSN IDE automatically generates a bootable AppleDos 3.3 disk image with extension ".dsk" if one of the + predefined emulators is used for execution. If you want to use another DOS or disk size or if you want to but more files into + the disk after compilation, you can use the command line version of + AppleCommander + to achieve this. Create a makefile script and configured it as described in + How to run a makefile script instead of an emulator? + . In case of Apple Disk images always remember using the correct the file content/load/run address. The IDE needs to know the + load address of an executable file in order to store this information in the directory entry. To detect the load address from + the executable file, the IDE evaluate the file extension. Supported extensions are ".b", ".prg" and ".xex". Here's the logic + begin the built in disk image creation: +

+
    +
  • + File extension ".b" +
    + // AppleDos 3.3 binary file: start-lo,start-hi,length-lo,length-hi,data +
    + address = getWord(outputFileContent, 0);length = length - 4;content = getData(outputFileContent, 4); +
    +
  • +
  • + File extension ".prg" and length > 2 +
    + // C64 program file: start-lo,start-hi,data +
    + address = getWord(outputFileContent, 0);length = length - 2;content = getData(outputFileContent, 2); +
    +
  • +
  • + File extension ".xex" and length > 6 and (getWord(outputFileContent, 0) & 0xffff) == 0xffff) +
    + // AtariDOS 2.5 binary file:$ff,$ff,start-lo,start-hi,end-lo,end-hi,data +
    + address = getWord(outputFileContent, 2);length = length - 6;content = getData(outputFileContent, 6); +
    +
  • +
+
Other hardwares
+

+ If you find a tool similar to "dir2atr.exe" for the Atari 8-bit or "AppleCommander" for the Apple II, you can use create your + own script an run it as described in + How to run a makefile script instead of an emulator? + . +

+
+
+

How can I run a makefile or script instead of an emulator?

+

+ Sometimes it is useful to run a makefile script instead of the emulator, for example if the output file shall be combined + with other files into a single ATR file. To execute a such a script select "User Defined Application" as the "Default + Application to open Output File". Specify the path to the shell as "Path to Application". In the command line you can then + use the standard variables to start the shell, pass the name of the script and pass the file path of the compiled output + file. Since the working directory at the time of execution is the output folder of the compiler you must place the script + file there or you must specify the script file with its absolute path. If you are using Windows and "cmd.exe" as shell, you + must specify "/c" before the name of the script to prevent "cmd.exe" from remaining as process after the script has finished. + The resulting command line is "${runnerExecutablePath} /c makefile.bat ${outputFilePath}", assuming "makefile.bat" is located + in the output folder". See + How do I compile into disk images? + for the example how to use this for compiling complete disk images. +

+ Configure makefile script +
+

Emulation

+
+

How can I use other emulators?

+

You can "re-use" the existing tabs and simply specify another emulators' executable. If you use "User Defined + Application", you can specify whatever you want. When using "User Defined Application", no disk image is created or updated. + You can use this setting to have you own script which put the executable file onto a disk image of your choice using + additional tools like "dir2atr.exe" or "AppleCommander" for example. +

+
\ No newline at end of file diff --git a/com.wudsn.ide.asm/help/ide-features.section.html b/com.wudsn.ide.asm/help/ide-features.section.html new file mode 100644 index 00000000..734aa16e --- /dev/null +++ b/com.wudsn.ide.asm/help/ide-features.section.html @@ -0,0 +1,1014 @@ + +The latest version contains the following features: +
+ +
General + IDE enhancements   + » top +
+
    +
  • "Open Folder" context menu available for all folders and files
  • +
  • + "Sort" context menu with sub-menu available for all text files including +
    +
      +
    • Case sensitive, case insensitive and numeric sorting
    • +
    • Sorting with and without removal of duplicates
    • +
    • Reverse ordering
    • +
    + Open folder action in context menu + Sort action in text editor context menu +
  • +
  • + The online help contains the entry "WUDSN IDE Guide" which contains multiple sections. +
      +
    • + WUDSN IDE - mainly the documentation from the web site +
        +
      • Video links to the tutorial and release news
      • +
      • Features
      • +
      • Installation
      • +
      • FQA
      • +
      • Link
      • +
      +
    • +
    • + Assemblers - the information on the supported assemblers and their properties +
        +
      • General - links, syntax and support features
      • +
      • Instructions - all supported instruction including their descriptions grouped by type
      • +
      • Manual - direct access to the PDF, HTML or text manual file or files which are part of the compiler installation
      • +
      +
    • +
    • + Hardware - the information on the supported hardwares and the corresponding emulators and links +
        +
      • Reference documentation - Hardware specific PDF, HTML or text files with CPU and customer chips reference sheets and programming guides
      • +
      +
    • +
    +
    + Integrated Help +
  • +
+
Fully + integrated assembler editor   + » top +
+
    +
  • + The Eclipse platform contains editors like for example the generic text editor. Editors can support one or more + content types identified by file extensions. One editor can be the default for a specific file extension. WUDSN IDE + provides an assembler + editor and a corresponding content type for every compiler. Every assembler + editor has a default compiler associated which in turn defines via the preferences which application is used to run the + output file. +
    + IDE editors, content types and file associations +
  • +
  • + New content types and file associations for Atari 2600 compilers +
      +
    • DASM Source File (*.asm)
    • +
    +
  • +
  • + New content types and file associations for Atari 8-bit compilers +
      +
    • ATASM Source File (*.asm)
    • +
    • MADS Source File (*.asm, default)
    • +
    • XASM Source File (*.asx, default)
    • +
    +
  • +
  • + New content types and file associations for C64 compilers +
      +
    • ACME Source File (*.a, default)
    • +
    +
  • +
  • + New content types and file associations for NES compilers +
      +
    • ASM6 Source File (*.asm)
    • +
    +
  • +
  • + The default editor for an extension can be configured in section "File Associations" of the preferences with the + button "Default" +
    + IDE file associations +
  • +
  • Syntax highlighting colors and styles for comments, directives, all types of identifiers, legal opcodes, illegal + opcodes, pseudo opcodes and strings +
  • +
  • Single line comments for the current line selection can be toggled using "CTRL-7" or the editor context menu entry + "Toggle Comment" +
  • +
  • Configurable support for 16-bit opcodes of 65816
  • +
  • + Built-in support for + ATASM compiler + , + MADS compiler + , + XASM compiler + , + ACME compiler +
  • +
  • Build in support for running compiler output files with the operating system default application without + configuration effort +
  • +
  • + Build in support for + Altirra emulator + , + Atari800Win emulator + , + Atari800MacX emulator + , + Atari++ emulator +
  • +
  • + Adding support for new compilers and application is possible via Eclipse extension points +
    + IDE compiler extensions +
  • +
  • Compile or compile and run with a single key stroke or via menu
  • +
  • Toolbar button for "Compile and Run" which adapts to the hardware of the currently opened editor, for example it + will show a C64 icon when using the "ACME (C64)" editor +
  • +
  • + The toolbar button for "Compile and Run" offers a pulldown menu which allows to run the output file with any of the + applications configured in the preferences. This way you can run the output file easily with different emulators + without changing the + preferences. That can be very helpful if one emulator show a different behaviour than another or + in case you have specified a build script as user defined application in the preferences +
    + Run with... for Atari 2600 + Run with... for Atari 8-bit + Run with... for C64 +
  • +
  • Open source and output folder directly from menu
  • +
  • + Open compiler help directly from menu if the documentation is available in the default folder structure of the + compiler executable +
    + IDE assembler menu +
  • +
  • Complete list of compiler error and warning messages in the problems view
  • +
  • Direct navigation to the source location of the via double click on the problem message
  • +
  • + Problem markers in the scroll area including tooltip with problem message +
    + IDE example source +
  • +
  • + German localization for all Eclipse plugin texts. The original compiler messages are not translated but some are + mapped automatically +
    +
      +
    • ATASM ".bank" warnings are mapped to info messages
    • +
    +
  • +
  • + Console view with original compiler output opens automatically in the background after compiling which useful in + cases where the plugin is not yet complete. The "Compiler Console" is automatically brought to front when the + compiler starts. This prevents + the compiler output from being hidden behind other consoles in the console view +
    + IDE compiler console view +
  • +
+
Content + outline and source folding   + » top +
+
    +
  • Activated via the standard menu "Window/Show View/Outline"
  • +
  • Automated asynchronous parsing while typing
  • +
  • Automated recursive parsing of source includes and merge with current outline
  • +
  • Positioning in the content outline remains stable while typing unless structure changes appear
  • +
  • Folding is active automatically when outline is visible
  • +
  • Folding for if/else/endif blocks
  • +
  • Outline and folding for definition section and implementation sections
  • +
  • Outline for equate definitions (including defining expression), label definitions and variable definitions
  • +
  • Outline and folding for enum and struct definitions
  • +
  • Outline and folding for macros definitions
  • +
  • Outline and folding for repeat sections
  • +
  • Outline and folding for procedure definitions
  • +
  • Outline and folding for local sections
  • +
  • Outline for source includes and binary includes
  • +
  • Type specific outline icons
  • +
  • Nested folding with tooltip for folded sections
  • +
  • Line end comment is used as short description
  • +
  • + Toolbar with button to toggle the sorting order of the sections and labels. The state of the button is persisted + automatically along with the respective source file. For files which do not have a setting yet, the setting from the + currently opened file + will be used +
    + IDE content outline +
    + IDE content folding +
  • +
+
Content + assist and code completion   + » top +
+
    +
  • Content assist by pressing CTRL-Space for directives, legal opcode, illegal opcodes and pseudo opcodes
  • +
  • The content assist also recognizes if there is already and instruction in the current line and suggests the + available identifiers instead if this is the case. This is the first version and it supports global identifiers in + the current source file and + source file included from there. Scoped identifiers of the form "a.b" are not yet + supported +
  • +
  • Completion proposal auto activation without pressing CTRL-Space after typing compiler dependent characters, for + example "." in ATASM, "." or "#" in MADS and "!" in ACME +
  • +
  • Type specific icon and mnemonic highlighting
  • +
  • Progressive filtering as you type
  • +
  • Automatic detection of lower case / upper case based on current input
  • +
  • Default case configurable in preferences
  • +
  • Illegal opcodes can be hidden via preferences
  • +
  • + 16-bit opcodes of 65816 can hidden via preferences +
    + IDE content assist +
  • +
  • + Multi-line content completion and explicit cursor positioning, for example ".MA" becomes ".MACRO <cursor is + here> <newline> .ENDM" +
    + Muli-line content completion and cursor positioning +
  • +
+ +
    +
  • Hyperlink navigation via CTRL-click to source includes and binary includes
  • +
  • Support for relative and absolute file paths
  • +
  • Source file are always opened with the same assembler editor, irrespective of the extension
  • +
  • The ".asm" extension for "ICL" source includes is appended automatically in MADS, if it is missing
  • +
  • The ".asx" extension for "ICL" source includes is appended automatically in XASM, if it is missing
  • +
  • + Choice for binary includes to open the file with +
      +
    • the build in hex editor
    • +
    • the build in graphics editor
    • +
    • the default Eclipse editor (e.g. a text editor)
    • +
    • system editor (e.g. an emulator or a paint program)
    • +
    +
  • +
  • + Hyperlink navigation via CTRL-click to labels, equates, local definitions, macro definitions and procedure + definitions. In case there is only one target, direct navigation takes place. In case there is more than one possible + target, the type and line + number of the target are displayed in a hyperlink popup. All included source files are also + taken into account like in the content outline. In case there are targets from different files, the file name is also + displayed as differentiator in the hyperlink + popup. +
    +
    + IDE navigation to source file +
    + IDE navigation to binary file +
    + IDE navigation to labels, equates, local definitions, macro definitions and procedure definitions +
  • +
+
Hex + Editor   + » top +
+
    +
  • Read-only hex editor to inspect arbitrary files
  • +
  • + Available via context menu including multi-file-selection to open several files at once +
    + Open With Hex Editor +
  • +
  • The "Open With Hex Editor" menu entry closes an existing editor in case the file is already opened and thereby + forces the file to be opened with the hex editor. In addition the hex editor is set as the default editor this one + file. As a consequence + double clicking the file in the package explorer will open it automatically with the hex + editor, no matter which type of file is actually is. This saves you from always using the context menu, just because + the file has no known file type. You can change + the default editor again by just using the "Open With" context menu + entry +
  • +
  • Available via hyperlink navigation for binary includes
  • +
  • + Context menu to copy parts of the file into the clipboard in different formats: +
      +
    • hex values (".byte $01,$02,...")
    • +
    • decimal values (".byte 1,2,...")
    • +
    • ASCII string
    • +
    + This is very helpful to turn parts of a file into source code. The possibility to copy/paste into the binary file + itself in order to modify is prepared but not working yet and will be completed later. +
    + Hex Editor context menu +
  • +
  • + Support for binary files, Atari COM files and Atari Disk Images +
    + Hex Editor file mode selection +
  • +
  • The possible file modes for a binary file are computed automatically and used as default when opening the file +
  • +
  • Unsupported file modes are detected and cannot be used
  • +
  • Files with a block structure get an outline in the outline view
  • +
  • + Files with a corrupted block structure are detected and displayed as good as possible +
    + Hex Editor outline view +
    + Hex Editor with corrupted Atari COM file +
  • +
+
Graphics + Editor   + » +
+
    +
  • + Viewer for binary files and 8-bit image files. +
    + The most suitable supported converter and the corresponding default parameters are computed based on the file + content, size and extension. +
    + The converters for the Atari standard image formats are based on + FAIL + , the excellent First Atari Image Library created by Piotr Fusik and Adrian Matoga. +
    + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    PlatformFile ExtensionFile Format
    Atari 8-bitAP380x192, 256 colors, interlaced
    Atari 8-bitAPCAny Point, Any Color, 80x96, 256 colors, interlaced
    Atari 8-bitCHR8x8 charset, mono or multicolor
    Atari 8-bitCCIChampions' Interlace, 160x192, compressed
    Atari 8-bitCINChampions' Interlace, 160x192
    Atari 8-bitCHGGephard Hires Graphics, up to 320x200, mono
    Atari 8-bitCPRTrzmiel, 320x192, mono, compressed
    Atari 8-bitFNTStandard 8x8 font, mono
    Atari 8-bitGR8 Standard 320x192, mono
    Atari 8-bitGR9Standard 80x192, grayscale
    Atari 8-bitHIP Hard Interlace Picture, 160x200, grayscale
    Atari 8-bitHRHires 256x239, 3 colors, interlaced
    Atari 8-bitHR2Hires 320x200, 5 colors, interlaced
    Atari 8-bitILCAPAC 80x192, 256 colors interlaced
    Atari 8-bitINPInterlace Picture 160x200, 7 colors, interlaced
    Atari 8-bitINTINT95a, up to 160x239, 16 colors, interlaced
    Atari 8-bitMCPMcPainter, 160x200, 16 colors, interlaced
    Atari 8-bitMICMicropainter 160x192, 4 colors
    Atari 8-bitPICKoala MicroIllustrator, 160x192, 4 colors, compressed
    Atari 8-bitPLMPlama 256, 80x96, 256 colors
    Atari 8-bitRIPRocky Interlace Picture, up to 160x239
    Atari 8-bitSXS16x16 font, mono
    Atari 8-bitTIPTaquart Interlace Picture, up to 160x119
    C6464CCharset, mono or multi color
    C64SPRSprite, mono or multi color
    +
  • +
  • Viewer for GIF/JPG/BMP/PNG images
  • +
  • + Available via context menu including multi-file-selection to open several files at once +
    + Open With Graphics Editor +
  • +
  • The "Open With Graphics Editor" menu entry closes an existing editor in case the file is already opened and + thereby forces the file to be opened with the graphics editor. In addition the graphics editor is set as the default + editor this one file. As a + consequence double clicking the file in the package explorer will open it automatically + with the graphics editor, no matter which type of file is actually is. This saves you from always using the context + menu, just because the file has no known file + type. You can change the default editor again by just using the "Open + With" context menu entry +
  • +
  • Available via hyperlink navigation for binary includes
  • +
  • + Conversion from binary files or 8-bit images files to GIF/JPG/BMP/PNG images +
      +
    • Up to different 3 source files depending on the converter
    • +
    • Configurable start offset for every source file, useful for extracting character sets
    • +
    • Configurable palette via the Image Palette view
    • +
    • Configurable number of rows and columns
    • +
    • Configurable spacing width for and spacing color to separate tiles and unused areas
    • +
    • Separate aspect ratio for display and saving the image file
    • +
    + Conversion from binary files or 8-bit images files to GIF/JPG/BMP/PNG images +
  • +
  • + Conversion from GIF/JPG/BMP/PNG images to binary files or 8-bit images using JavaScript. +
      +
    • Separate aspect ratio for loading and displaying the image file
    • +
    • Default scripts are included. They can adjusted and saved along with the other parameters
    • +
    +
    + Conversion from GIF/JPG/BMP/PNG images to binary files or 8-bit images files +
  • +
  • + The image palette view associated with the graphics editor displays the palette entries and the color histogram. It + supports filtering of unused colors, sorting by index and color frequencies, pre-settings and editing the palette + when displaying 8-bit + images +
    + Image palette view +
  • +
  • Conversion files with extension ".cnv" are used to save the conversion direction and all conversion parameters +
  • +
  • Support for relative file paths. If the file path is in the same folder as the ".cnv" file, it it automatically + converted to a relative file path when the file name is defaulted or picked from the file browser dialog. This allows + to move the ".cnv" + file and the referenced source files around together without breaking the saved paths +
  • +
+
Preferences + for editing   + » top +
+
    +
  • Available via entry "Preferences" from the menu "Window" (in Windows and Linux) or the menu "Eclipse" (in Mac OS + X) +
  • +
  • Configuration of syntax highlighting colors and styles for comments, directives, all types of identifiers, legal + opcodes, illegal opcodes, numbers, pseudo opcodes and strings +
  • +
  • For non unique identifiers, the syntax highlighting will try to be a best guess
  • +
  • + Configuration of default case in content assist +
    + Preferences for compilers +
  • +
+
Preferences + for compiling   + » top +
+
    +
  • Available via entry "Preferences" from the menu "Window" (in Windows and Linux) or the menu "Eclipse" (in Mac OS + X) +
  • +
  • Separate compiler preferences page for every type of hardware, i.e. Atari 8-bit and C64; Apple 2 support is in + development by Nick Westgate +
  • +
  • Preferences can be maintained for all compilers in parallel
  • +
  • Upon opening the compiler preferences, the tab for the compiler of the active editor is activated automatically +
  • +
  • + Configuration of illegal opcodes and 65816 opcodes support in syntax highlighting and content assist +
    + The configuration is offered only if the compiler supports the respective feature +
  • +
  • Download links, configurable paths and default parameters for all compilers
  • +
  • The default file extension filter is set to "*.exe" on Windows and "*.*" on all other operating systems
  • +
  • The button "Apply Defaults" does explicitly not reset the paths to the executables
  • +
  • + Configuration of parameters per compiler including predefined defaults to run out-of-the-box +
    + Possible variables are: +
      +
    • {$sourceFolderPath} The absolute path to the source folder
    • +
    • {$sourceFilePath} The absolute path to the source file
    • +
    • {$outputFolderPath} The absolute path to the output folder
    • +
    • {$outputFilePath} The absolute path to the output file
    • +
    • {$outputFileName} The name of the output file including its extension, for example "TestFile123.asm"
    • +
    • {$outputFileNameWithoutExtension} The name of the output file without extension, for example "TestFile123"
    • +
    • {$outputFileNameShortWithoutExtension} The name of the output file without extension shortened to 8 alphanumeric + characters, for example "TESTFILE" +
    • +
    + Make sure that you don't remove parameters when you specify your own values +
    + If not, either creating the label definition file or parsing the compiler log may fail +
      +
    • ATASM requires the "-s" parameter to be present"
    • +
    • MADS requires the "-p" parameter to be present
    • +
    • ACME requires the "!to" directive in the source to be commented out to ensure the output file name from the + command line is used +
    • +
    +
  • +
  • + Preferences can be maintained for all possible applications of all compilers in parallel +
      +
    • + Possibility to use the "Operating Sytem Default Application" of the platform to open the output file +
      + For Windows and Mac OS X choosing the "Operating Sytem Default Application" means that for ".xex" file the emulator will be started + automatically without any further configuration +
    • +
    • Possibility to use one of the pre-defined applications and optionally change the command line
    • +
    • Possibility to use a user defined application to open the output file
    • +
    • + Possible variables are: +
        +
      • {$runnerExecutablePath} The absolute path to the executable of the application
      • +
      • {$sourceFolderPath} The absolute path to the source folder
      • +
      • {$sourceFilePath} The absolute path to the source file
      • +
      • {$outputFolderPath} The absolute path to the output folder
      • +
      • {$outputFilePath} The absolute path to the output file
      • +
      • {$outputFileName} The name of the output file including its extension, for example "TestFile123.asm"
      • +
      • {$outputFileNameWithoutExtension} The name of the output file without extension, for example "TestFile123" +
      • +
      • {$outputFileNameShortWithoutExtension} The name of the output file without extension shortened to 8 + alphanumeric characters, for example "TESTFILE" +
      • +
      +
    • +
    +
  • +
  • Download links, configurable paths and default command lines for Atari++, Atari800Win, Atari800MacX emulators per + compiler +
  • +
  • + The default file extension filter is set to "*.exe" on Windows and "*.*" on all other operating systems +
    + Preferences for compilers +
  • +
+
Annotations + for compiling   + » top +
+
    +
  • For most use-cases, the defaults provided for the compilers and in the preferences are sufficient. But if you want have develop in parallel in multiple projects, for different platforms (Apple II and Atari 8-bit) or in different output formats + (".XEX" and ".BIN") with the same compiler it may become cumbersome to change the preferences every time. Therefore WUDSN IDE offers annotations which you can put into the source code files. These annotations override the defaults and the preferences. +
  • +
  • All annotations start with the prefix "@com.wudsn.ide.asm." followed by the lower case name of the annotation, an equals sign and the unquoted value. Example: "@com.wudsn.ide.asm.hardware=ATARI8BIT"
  • +
  • All annotations can be placed in comment lines at the begin of a source file. Some of the annotations are only relevant for the main source file, some are only relevant in include source files, some are + relevant for all source files. +
  • +
  • + @com.wudsn.ide.asm.hardware +
      +
    • Defines the target hardware for which the preferences shall be evaluated, in particular which emulator is used to run the output file.
    • +
    • Allowed values are "APPLE2", "ATARI2600", "ATARI7800", "ATARI8BIT", "C64", "NES".
    • +
    • This annotation is relevant for all source files.
    • +
    • This annotation is only evaluated when a file is opened. So if you add this annotation or change its value, you have to close and re-open the file once.
    • +
    • Example: @com.wudsn.ide.asm.hardware=ATARI8BIT
    • +
    +
  • +
  • + @com.wudsn.ide.asm.mainsourcefile +
      +
    • Defines the main source file to which the current include source file belongs. When the "Compile" action is executed, the main source file is compiler instead of the current file.
    • +
    • Allowed values are file paths relative to the folder of the current include source file and absolute file paths.
    • +
    • This annotation is only relevant for include source files.
    • +
    • Example: @com.wudsn.ide.asm.mainsourcefile=ExampleMain.asm
    • +
    +
  • +
  • + @com.wudsn.ide.asm.outputfoldermode (planned for 1.7.0) +
      +
    • Overrides the "Output Folder Mode" from the preferences.
    • +
    • Allowed values are "SOURCE_FOLDER", "TEMP_FOLDER", "FIXED_FOLDER".
    • +
    • This annotation is only relevant for the main source file.
    • +
    • Example: @com.wudsn.ide.asm.outputfoldermode=SOURCE_FOLDER
    • +
    +
  • +
  • + @com.wudsn.ide.asm.outputfolder (planned for 1.7.0) +
      +
    • Overrides the "Output Folder" from the preferences and the "@com.wudsn.ide.asm.outputfoldermode" annotation.
    • +
    • Allowed values are file paths relative to the folder of the main source file and absolute file paths.
    • +
    • This annotation is only relevant for the main source file.
    • +
    • Example: @com.wudsn.ide.asm.outputfolder=..\out
    • +
    +
  • +
  • + @com.wudsn.ide.asm.outputfileextension (planned for 1.7.0) +
      +
    • Overrides the "Output File Extension" from the preferences.
    • +
    • Allowed values have to start with a period.
    • +
    • This annotation is only relevant for the main source file.
    • +
    • Example: @com.wudsn.ide.asm.outputfileextension=.bin
    • +
    +
  • +
  • + @com.wudsn.ide.asm.outputfile (planned for 1.7.0) +
      +
    • Overrides the "Output Folder", "Output File Extension" from the preferences and the automatic computation of the out file name based on the main source file name.
    • +
    • Allowed values are file paths relative to the folder of the main source file and absolute file paths.
    • +
    • This annotation is only relevant for the main source file.
    • +
    • Example: @com.wudsn.ide.asm.outputfile=..\out\output.bin
    • +
    +
  • +
+
Known + bugs   + » top +
+Open bugs: +
    +
  • If you find any, please contact me
  • +
+Fixed bugs: +
    +
  • + 1.6.5 +
    +
      +
    • Automatic creation of ".DSK" disk images for Apple II now works correctly +
    • +
    • Dirty indicator in Graphics Editor is now updated correctly +
    • +
    +
  • +
  • + 1.6.4 +
    +
      +
    • The toolbar icons now work correctly with newer Eclipse versions
    • +
    • Atari 8-bit Graphics 12 Converter works again
    • +
    • Graphics editor now properly closes input stream for image files
    • +
    • Empty selection and too large numbers no longer lead to exceptions when opening the context menu for "Convert..."
    • +
    • Pressing refresh in the graphics converter now always updates the image pane correctly and not only if the size has changed
    • +
    • Inline repeats like ":64" in MADS are no longer detected as labels
    • +
    • Hex Editor now correctly detects erroneous COM files in case the segment length exceeds the file length (by one)
    • +
    • Disk images (for Apple II) are now updated also if only "Compile" instead of "Compile and Run" is used
    • +
    • Apple Commander integration is now part of the installation, as it should have been with 1.6.3
    • +
    • + The HELLO program generated for the auto-start disk images of Apple II now displays a title and uses "BLOAD/CALL" instead of "BRUN" because of this + bug in Apple DOS +
      10 PRINT "Loading <title>" : PRINT CHR$(4);"BRUN WORLD" : CALL <address>
      +
    • +
    +
  • +
  • + 1.6.3 +
    +
      +
    • First character of numbers is now correctly highlighted in #123
    • +
    • Source file includes via "INCSRC" are now correctly detected for ASM6
    • +
    +
  • +
  • + 1.6.2 +
    +
      +
    • Typing a "." to trigger the automatic content assist no longer locks-up
    • +
    • Default color for illegal opcodes changed to red as it was intended
    • +
    • German localization for the graphics editor is finally completed
    • +
    • Sorting of application in the preferences now is "Default, A...Z, User Defined" also in non English localizations
    • +
    +
  • +
  • + 1.6.1 +
    +
      +
    • Clicking in the outline always positions the cursor correctly in the source, not only the first time
    • +
    • All names of content types are now translated correctly in the preferences
    • +
    • "Open Folder" command now also works for objects which are no resource or file themselves but can be adapted to one of these types. For example project explorer entries for Java classes
    • +
    +
  • +
  • + 1.6.0 +
    +
      +
    • Hyperlink navigation now also works for labels which contain an underscore
    • +
    • The key binding for the "Compile" menu is now "Shift-Ctrl-9" because it turned out that "Ctrl-0" is not available in all cases
    • +
    • The HexEditor is now also detecting the situation that the first block of a COM file is incorrect and display this correctly
    • +
    • Content assist has now correct new lines when inserting #IF, #WHILE and .TEST
    • +
    • Fonts and colors resources are disposed correctly now
    • +
    • Resetting to default syntax colors in the preferences works now
    • +
    • Elements of .ENUM are recognized as equates now also if they do not start of position 0
    • +
    • Elements of .STRUCT are recognized as labels now also if they do not start of position 0
    • +
    • Cursor is now positioned to the first character of an equate or label also if it is not defined starting at position 0
    • +
    • Cursor is now positioned to the first character of an equate or label also if it is not defined starting at position + 0 +
    • +
    +
  • +
  • + 1.5.0 +
    + +
      +
    • Syntax highlighting remains active now also after "Save as..."
    • +
    • Cursor is now placed correctly by content assist
    • +
    • The key binding for the "Compile" menu is now "Ctrl-0" instead of "Ctrl-Alt-0", so entering "}" is now possible again
    • +
    • XASM editor now correctly detects the "ORG" directive
    • +
    • The assembler editor toolbar contribution now also displays the label "Assembler" in the customizing dialog for the perspective
    • +
    +
  • +
  • + 1.4.4 +
    +
      +
    • The directive ".LOCAL" in ATASM is a normal directive now and does not start a folding section while in MADS it is + really is a folding section from ".LOCAL" to ".ENDL" +
    • +
    • "SIN()" and "RND()" in MADS are now recognized correctly even if there are no spaces before or after the directive +
    • +
    • ".EN" and ."END" in MADS are now recognized correctly
    • +
    • Labels in "ORG" lines are now recognized correctly and rendered as separate tree entries
    • +
    • Preferences for lower/upper case instructions in content assist are now also evaluated if the for directives which + do not start with letters like ".end" +
    • +
    • When opening a file which is located outside of the workspace, the action to open folders and to compile the file + are now not doing anything and will not cause exceptions. They will also be disabled once the new Eclipse version is + mandatory +
    • +
    • The 16 bytes per row in the hex editor are now separated by a space
    • +
    +
  • +
  • + 1.4.3 +
    +
      +
    • The output file is not opened anymore in case it has been there before but was not updated by the compiler due to + compiler errors +
    • +
    • The scroll bar and cursor in the editor are now stable also in the cases when the outline is changed by the latest + user input +
    • +
    • The hex editor now gets the focus correctly also when clicking on already opened files
    • +
    +
  • +
  • + 1.4.2 +
    +
      +
    • The folder which contains the output file and the symbols file is refreshed automatically after compiling to + ensure that the Eclipse resource cache is in sync with the file system +
    • +
    • Line end comments are now used as description for source and binary includes in the content outline
    • +
    • The "Open Folder" context menu entry works again, bug was introduced in version 1.4.0
    • +
    +
  • +
  • + 1.4.0 +
    +
      +
    • The "Assembler" menu is visible now only if an Assembler editor is active. The contained entries and their short + cuts are now disabled if no Assembler editor is active +
    • +
    • The output file is not deleted anymore before the compiler is started, instead it is only checked for being + writeable. This allows for direct compilation into an existing ATR image +
    • +
    • The name of the application used to open the output file is now included in the info message
    • +
    +
  • +
  • + 1.3.2 +
    +
      +
    • The content outline sometimes only found the first macro definition
    • +
    • Folding did not show up in all situations
    • +
    • + Under Mac OS X, the file select dialog can now choose an "*.app" application like "Atari800MacX.app" since this is + folder. The solution is a workaround to the a general Eclipse problem under MacOS X, see + BUG 82155 +
    • +
    +
  • +
  • + Before 1.3.2 +
    +
      +
    • The "OK" and "Apply" buttons in the preferences are now always enabled, irrespective of the specified compiler and + emulator paths +
    • +
    • The focus is not back in editor window after compiling
    • +
    • The syntax highlighting of illegal opcodes fixed, now also uses the preferences setting
    • +
    • On German operating systems, the properties for the correct locale are now found
    • +
    +
  • +
+
Planned + features   + » top +
+
    +
  • + See the table below for the current feature set for the features which depend on the compiler. +
    + Features which are not supported by the compiler itself and hence cannot be supported by the IDE are marked as "n/a". +
    + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    CompilerDefault HardwareAuto Completion ActivationSingle-Line CommentsBlock CommentsStringsOpcodesDirectivesCompile Log ParsingInclude Log ParsingContent OutlineLabel Definition File Format
    ACMEC64Yes: !Yes: ;|n/aYes: "Yes + Partly + YesTBD + Partly + ACME
    ATASMAtari 8-bitYes: .Yes: ;n/aYes: "YesYesYesTBDYesXASM 3.0.1
    MADSAtari 8-bitYes: . #Yes: ; * //YesYes: ' "YesYesYesTBDYesMADS
    XASMAtari 8-bitn/aYes: ; * |n/aYes: ' "YesYes + Planned + TBDYesXASM 3.0.1
    +
  • +
diff --git a/com.wudsn.ide.asm/help/ide-installation.section.html b/com.wudsn.ide.asm/help/ide-installation.section.html new file mode 100644 index 00000000..1e557552 --- /dev/null +++ b/com.wudsn.ide.asm/help/ide-installation.section.html @@ -0,0 +1,548 @@ +

+There are two ways of installing WUDSN IDE. If you are using a Windows operating system, I recommend downloading the ready to run zero installation distribution: +
+

+ + Click to download the Windows 64-bit version +
+It contains the Eclipse Platform, the Java Runtime Environment, the latest stable version of the WUDSN IDE plugin, all supported compilers and an emulator for +each supported platform. All paths to folders, compilers and emulators are pre-configured. Unpack the content of this archive to +the directory "C:\jac\wudsn" and click the "WUDSN IDE" link. Eclipse will open with the predefined workspace that contains +"Hello World" examples for different platforms. +
+
+If you use another operating system, or want to use the latest version of WUDSN or want to install compilers and emulators more +selectively, read the descriptions of the installation steps below. In addition, the following previous versions of the zero installation distribution are available + +In case something is not correct or not working, please contact me. +

+ +
+ Installing Eclipse   + » top +
+
    +
  • + Download Eclipse from + http://www.eclipse.org/downloads + . +
  • +
  • + If you don't know which version to take, use + Eclipse 4.3.2 + Platform Runtime Binary (61 MB) + . +
    + This is a minimum size installation which does not include the Java Development Toolkit (JDT). WUDSN IDE has no dependency to + the JDT but of course to the Java Runtime Environment (JRE). Make sure you also have the 64-bit version of the JRE installed + if you want to use the 64-bit version of Eclipse. +
  • +
  • Unzip the downloaded archive file and store the contained folder "eclipse" where you want Eclipse to be located on you + local hard drive.
  • +
  • Start the Eclipse executable from that folder.
  • +
  • Upon the first start you are prompted to specify the folder where the workspace shall be located.
  • +
  • Normally a start link a created to this end. You can use the parameter "-nl" to specify the locale if you want. Use "en_US" + for English or "de_DE" for German. Example: "C:\Program Files\Eclipse\4.3.1\eclipse\eclipse.exe" -Xmx512M -nl en_US"
  • +
  • It is recommended to create the workspace folder in your home directory.
  • +
  • After you have read the welcome page and got familiar with the Eclipse UI, just switch to the Resource perspective.
  • +
  • + Open the view "Problems" via the menu "Window/Show View/Problems" and then click the entry "Configure Contents..." from its + view menu. +
    + Open configuration of the problems view +
  • +
  • + Select the configuration "All Errors", the radio button "On selected element and its children" and the check boxes ""Error", + "Warning" and "Info". +
    + Configuration of the problems view +
  • +
  • + Now Eclipse itself is ready and you can proceed with step + Installing WUDSN IDE + . +
  • +
+
+ Installing WUDSN IDE   + » top +
+
    +
  • Start Eclipse
  • +
  • Select the entry "Install New Software..." from the menu "Help".
  • +
  • Enter "http://www.wudsn.com/update" in the "Work with" field and press ENTER.
  • +
  • + Select the latest version of the feature "WUDSN IDE" for in and press the button "Next". +
    + You don't need to install the "General Eclipse Enhancements" feature as it is already included in the "WUDSN IDE" feature". +
  • +
  • Review the installation details and press the button "Next".
  • +
  • Read the license agreement, choose the option "I accept..." and press the button "Finish".
  • +
  • In case you get a security warning that the content is unsigned, confirm the warning by pressing the button "OK".
  • +
  • When you are prompted to restart Eclipse now, press the button "Yes".
  • +
  • + As always with updates, it may happen that the update fails for whatever reason or the installed version turns out to have + severe issues. For example the required Java Version might not be available on your machine. In this case you can uninstall it + via the link "What is already installed" in the "Install New Software..." dialog and restart the IDE. Then you can reinstall + the latest version from "http://www.wudsn.com/update" or previously released versions from the locations listed in the + "Releases" chapter. +
    + IDE installation steps +
  • +
+
+ Installing compilers like ATASM, MADS, XASM...   + » top +
+
    +
  • Start Eclipse
  • +
  • Select the entry "Preferences" from the menu "Window" (in Windows and Linux) or the menu "Eclipse" (in Mac OS X)
  • +
  • Open the preferences page "Assembler/Atari 8-bit Compilers" or the respective page for your platform.
  • +
  • + Select the tab for the compiler of your choice. +
    + The following compilers are or will be supported: +
    + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    CompilerDefault Target Platform
    ACMEC64
    ASM6NES
    ATASMAtari 8-bit
    DASMAtari 2600
    KickAssC64
    MADSAtari 8-bit
    XASMAtari 8-bit
    TASSC64 (in preparation)
    +
  • +
  • Click the "Download"" link to open the home page of the compiler.
  • +
  • Follow the instruction on the download site for install the compiler to the folder of your choice.
  • +
  • + Alternatively, you can download the single + archive containing all supported 6502 + compilers + including the Linux and Mac OS X versions where available. The archive contains a readme file with the date of the last update + and the included compiler versions.  +
    + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    CompilerOperating SystemArchitectureCompiler VersionExecutable PathCompile DateCompiled By
    ACMEWindowsIntel - 32 bit0.90ACME / acme.exe2006-03-19Krzysztof Dabrowski
    ASM6WindowsIntel - 32 bit1.6ASM6 / asm6.exe2011-03-11loopy
    ATASMWindowsIntel - 32 bit1.07 (non-final)ATASM / atasm.exe2010-05-10M. Schmelzenbach
    ATASMMac OS XIntel1.07 (non-final)ATASM / atasm.macosx-i3862015-05-01JAC!
    ATASMMac OS XPowerPC1.07 (non-final)ATASM / atasm.macosx-powerpc2015-05-01JAC!
    ATASMLinuxIntel - 32 bit1.07 (non-final)ATASM / atasm.linux-i3862015-05-01JAC!
    ATASMLinuxIntel - 64 bit1.07 (non-final)ATASM / atasm.linux-x86-642015-05-01JAC!
    DASMWindowsIntel - 32 bit2.20.11DASM / bin / dasm.exe2015-05-08JAC!
    DASMMac OS XIntel2.20.11DASM / bin / dasm.macosx-i3862015-05-01JAC!
    DASMMac OS XPowerPC2.20.11DASM / bin / dasm.macosx-powerpc2015-05-01JAC!
    DASMLinuxIntel 32-bit2.20.11-20140304DASM / bin / dasm.linux-i3862015-05-01JAC!
    DASMLinuxIntel 64-bit2.20.11-20140304DASM / bin / dasm.linux-x86-642015-05-01JAC!
    KickAssAllJavaV3.39KICKASS / KickAss.jar2015-03-26Mads Nielsen
    MADSWindowsIntel - 32 bit2.0.7 (new features since 1.9.0 are not fully supported in WUDSN IDE yet)MADS / mads.exe2014-05-20JAC!
    MADSMac OS XIntel2.0.7 (new features since 1.9.0 are not fully supported in WUDSN IDE yet)MADS / mads.macosx-i3862017-11-05JAC!
    MADSMac OS XPowerPC2.0.7 (new features since 1.9.0 are not fully supported in WUDSN IDE yet)MADS / mads.macosx-powerpc2017-11-05JAC!
    MADSLinuxIntel - 32 bit2.0.7 (new features since 1.9.0 are not fully supported in WUDSN IDE yet)MADS / mads.linux-i3862017-11-05JAC!
    TASSWindowsIntel - 32 bit1.46 r38TASS / 64tass.exe2011-07-03Soci
    XASMWindowsIntel - 32 bit3.1.0XASM / xasm.exe2014-07-200xF
    XASMLinuxIntel - 32 bit3.1.0XASM / xasm.linux-i3862014-07-200xF
    XASMMac OS XIntel - 32 bit3.1.0XASM / xasm.macosx-i3862014-07-200xF
    +
  • +
  • In the section "Browse..." button for the field "Path to Compiler" to locate the executable. Note that under Mac OS-X, you + must specify the path to the actual executable inside the ".app" folder. +
  • +
  • If not explicit compiler parameters are specified, the default parameters are used.
  • +
  • If explicit compiler parameters are specified, the default parameters are ignored.
  • +
  • + The variable "${sourceFilePath}" is replaced by the absolute path to the source file. +
    + The variable "${outputFilePath}" is replaced by the absolute path to the output file. +
    + For more variables see section + preferences for compiling. +
  • +
  • Choose if you want to use the source folder or the temporary folder as output folder.
  • +
  • Choose the file extension for the output file, for example ".xex" or ".bin"
  • +
  • Press the button "OK".
  • +
  • + Using the button "Restore Defaults" all values are reset, except for the paths to the compilers. +
    +
    + Configuration of compiler executable path +
  • +
  • + Using the button "Default" in the "File Associations" preferences you can set the default editor for a file extension, for + example "MADS" for "*.asm". +
    + IDE file associations +
  • +
+
+ Installing Atari800Win and other emulators   + » top +
+
    +
  • Start Eclipse.
  • +
  • Select the entry "Preferences" from the menu "Window" (for Windows and Linux) or "Eclipse" (for Mac OS X).
  • +
  • Open the preferences page "Assembler/Atari 8-bit Compilers" and select to tab for your compiler.
  • +
  • + The field "Default Application to open Output File" is defaulted to "Operating Sytstem Default Application". +
      +
    • If you have the emulator registered as default application for the extension of the output file, for example + "Atari800Win" or "Atari800MacX" for ".xex", you don't need to configure anything in addition and can skip the steps below. +
    • +
    • If your operating system, like for example Linux, does not support default applications or if you need a special output + file extension and special parameters to open the output file in the emulator you can choose the emulator in the field + "Application to open Output File" and configure the command line on the corresponding tab. +
    • +
    • You can also use the "User Defined Application" for open the output file with and arbitrary application like a script, + linker or whatever.
    • +
    +
  • +
  • If you have not yet downloaded the corresponding emulator, you can use the download link on the tab of the emulator and + follow the instruction on the download site for install the emulator
  • +
  • Use the "Browse..." button for the field "Path to Application" to locate the executable. If required you can specify your + own command line based on the default command line displayed. Note that under Mac OS-X, you must specify the path to the + actual executable inside the ".app" folder. +
  • +
  • If not explicit command line is specified, the default command line is used.
  • +
  • If an explicit command line is specified, the default command line is ignored.
  • +
  • + The variable "${runnerExecutablePath}" is replaced by the path to the application executable. +
    + The variable "${outputFilePath}" is replaced by the absolute path to the output file. +
    + For more variables see section + preferences for compiling. +
  • +
  • Press the button "OK"
  • +
  • + In case you also need the Atari ROM files, you can find them in the file + PCXF380.ZIP + which is available at + http://www.emulators.com +
  • +
  • + The patch to convert the English version into a German version including a German keyboard layout is available from + ABBUC + . It also contains the ROM files +
    + Configuration of emulator executable path +
  • +
+
+ Creating and compiling an example project   + » top +
+
    +
  • Start Eclipse.
  • +
  • Select the entry "New/Project" from the menu "File".
  • +
  • Select the wizard "General/Project" and press the button "Next".
  • +
  • Enter the project name "Atari800" and press the button "Finish".
  • +
  • The new project will appear in the "Project Explorer".
  • +
  • Select the newly created project and open its context menu.
  • +
  • Select the entry "New/File", enter the file name "Example.asm" and press the button "Finish".
  • +
  • The new empty file will now be opened in the MADS editor.
  • +
  • + Copy the following source text and paste it into the file: +
    +
    +
    + ; WUDSN IDE Atari Rainbow Example - MADS syntax +
    +
    +       org $4000 ;Start of code +
    +
    + start lda #0 ;Disable screen DMA +
    +       sta 559 +
    + loop  lda $d40b ;Load VCOUNT +
    +       clc +
    +       adc 20 ;Add counter +
    +       sta $d40a +
    +       sta $d01a ;Change background color +
    +       jmp loop +
    +
    +       run start ;Define run address +
    +
  • +
  • Select the entry "Compile and run" from the menu "Assembler" or press "SHIFT-CTRL-0".
  • +
  • Enjoy your famous first rainbow effect.
  • +
  • + In case of problems, open the view "Problems" and the output of the view "Console" for details and post a message with the + screenshots in the + English AtariAge forum + if you get stuck: +
    +
    + Congratulations, when you did everything correctly - this is your first rainbow effect +
  • +
+
+ Further information on assembler programming   + » top +
+ \ No newline at end of file diff --git a/com.wudsn.ide.asm/help/ide-releases.section.html b/com.wudsn.ide.asm/help/ide-releases.section.html new file mode 100644 index 00000000..87bbd698 --- /dev/null +++ b/com.wudsn.ide.asm/help/ide-releases.section.html @@ -0,0 +1,201 @@ + +The following table lists all releases of WUDSN IDE, the required minimum Java and Eclipse version the link to the release news +article and release new video. For releases before 1.6.0 no downloads and no videos are offered. The latest version is always +available via the update site "http://www.wudsn.com/update". Older versions are available via version specific update sites +listed below. +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
WUDSN IDE VersionRelease DateMinimum Java VersionMinimum Eclipse VersionUpdate Site URLRelease News ArticleRelease News Video
1.6.62014-06-11Java 1.64.3.1http://www.wudsn.com/update/1.6.6 + read + planned
1.6.52014-01-02Java 1.74.3.1http://www.wudsn.com/update/1.6.5 + read + planned
1.6.42013-09-13Java 1.63.6.0http://www.wudsn.com/update/1.6.4 + read + planned
1.6.32012-06-09Java 1.63.6.0http://www.wudsn.com/update/1.6.3 + read + planned
1.6.22012-05-16Java 1.63.6.0http://www.wudsn.com/update/1.6.2 + read + + watch +
1.6.12012-05-12Java 1.63.6.0not available + read + + watch +
1.6.02011-04-02Java 1.53.6.0http://www.wudsn.com/update/1.6.0 + read + + watch +
1.5.02010-10-27Java 1.53.6.0not available + read + not available
1.4.42010-09-01Java 1.53.3.0not available + read + not available
1.4.32010-04-28Java 1.53.3.0not available + read + not available
1.4.22009-10-11Java 1.53.3.0not available + read + not available
1.4.02009-09-13Java 1.53.3.0not available + read + not available
1.3.22009-07-26Java 1.53.3.0not available + read + not available
1.2.02009-07-12Java 1.53.3.0not available + read + not available
1.1.02009-06-28Java 1.53.3.0not available + read + not available
1.0.02009-06-08Java 1.53.3.0not available + read + not available
\ No newline at end of file diff --git a/com.wudsn.ide.asm/help/ide-tutorials-videos.html b/com.wudsn.ide.asm/help/ide-tutorials-videos.html new file mode 100644 index 00000000..aef0fe15 --- /dev/null +++ b/com.wudsn.ide.asm/help/ide-tutorials-videos.html @@ -0,0 +1,42 @@ + +Cheat sheet for creating video screen shots - Do not format the source! + +
+
+ + + + + \ No newline at end of file diff --git a/com.wudsn.ide.asm/help/ide-tutorials.section.html b/com.wudsn.ide.asm/help/ide-tutorials.section.html new file mode 100644 index 00000000..5ea869e5 --- /dev/null +++ b/com.wudsn.ide.asm/help/ide-tutorials.section.html @@ -0,0 +1,85 @@ + +When I was thinking about creating tutorials, I decided not to simply write text but to create short videos instead. +They are best viewed in full screen mode and in HD video resolution. +I think this is the best way to show how things are intended to be +used. On the other hand, these tutorials cannot +tackle every detail of a supported feature. So please also check the features section. +
+
+ + + + + + + + + + + + + + + + + + + + + +
+ Part 1: Introduction, Installation and Use +
+ + Tutorial part 1 + +
+ Part 2: Setting up Perspective, Views and Editors +
+ + Tutorial part 2 + +
+ Part 3: Setting up Editors and File Extensions correctly +
+ + Tutorial part 3 + +
+ Part 4: Syntax highlighting and Content Assist +
+ + Tutorial part 4 + +
+ Part 5: Working with Projects, Folders and Files +
+ + Tutorial part 5 + +
+ Part 6: Content Outline and Navigation - the Heart of the IDE +
+ + Tutorial part 6 + +
+ Part 7: New Features in version 1.6.0 +
+ + Tutorial part 7 + +
+ Part 8: New Features in version 1.6.2 +
+ + Tutorial part 8 + +
+ Part 9: Source Level Debugging +
+ + Tutorial part 9 + +
+
diff --git a/com.wudsn.ide.asm/help/productions/java/.gitignore b/com.wudsn.ide.asm/help/productions/java/.gitignore new file mode 100644 index 00000000..92ae9f65 --- /dev/null +++ b/com.wudsn.ide.asm/help/productions/java/.gitignore @@ -0,0 +1 @@ +/ide diff --git a/com.wudsn.ide.asm/help/productions/java/ide.txt b/com.wudsn.ide.asm/help/productions/java/ide.txt new file mode 100644 index 00000000..daec6b32 --- /dev/null +++ b/com.wudsn.ide.asm/help/productions/java/ide.txt @@ -0,0 +1 @@ +This "ide" folder is only symbolic link. \ No newline at end of file diff --git a/com.wudsn.ide.asm/help/www.oxyron.de/html/opcodes.html b/com.wudsn.ide.asm/help/www.oxyron.de/html/opcodes.html new file mode 100644 index 00000000..3017746b --- /dev/null +++ b/com.wudsn.ide.asm/help/www.oxyron.de/html/opcodes.html @@ -0,0 +1,24 @@ + + + + + + + +65xx processor series opcodes + + + + +
Welcome to the 65xx resources.

+
Here you will find detailed information about some processors of the 65xx series. Especially the 6502 +document provides a deep look into the internal behaviour of the processor. Also you will find an illegal opcode list +which has been validated for being 100% several times now. The other documents only extend the 6502 document with +additional information. They do not completely describe all opcodes.

+6502/6510/8500/8502 Opcodes

+65C02/65SC02 Opcodes

+65CE02 Opcodes

+65816 Opcodes

+
Main page

+Last change on 30/1/2009. For comments contact Graham
+ diff --git a/com.wudsn.ide.asm/help/www.oxyron.de/html/opcodes02.html b/com.wudsn.ide.asm/help/www.oxyron.de/html/opcodes02.html new file mode 100644 index 00000000..3d074aa6 --- /dev/null +++ b/com.wudsn.ide.asm/help/www.oxyron.de/html/opcodes02.html @@ -0,0 +1,1124 @@ + + + + + + +6502/6510/8500/8502 Opcodes + + + + +

6502/6510/8500/8502 Opcode matrix:

+ +imm = #$00
+zp = $00
+zpx = $00,X
+zpy = $00,Y
+izx = ($00,X)
+izy = ($00),Y
+abs = $0000
+abx = $0000,X
+aby = $0000,Y
+ind = ($0000)
+rel = $0000 (PC-relative) +

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
 x0x1x2x3x4x5x6x7x8x9xAxBxCxDxExF
0xBRK
7
ORA
izx 6
KILSLO
izx 8
NOP
zp 3
ORA
zp 3
ASL
zp 5
SLO
zp 5
PHP
3
ORA
imm 2
ASL
2
ANC
imm 2
NOP
abs 4
ORA
abs 4
ASL
abs 6
SLO
abs 6
1xBPL
rel 2*
ORA
izy 5*
KILSLO
izy 8
NOP
zpx 4
ORA
zpx 4
ASL
zpx 6
SLO
zpx 6
CLC
2
ORA
aby 4*
NOP
2
SLO
aby 7
NOP
abx 4*
ORA
abx 4*
ASL
abx 7
SLO
abx 7
2xJSR
abs 6
AND
izx 6
KILRLA
izx 8
BIT
zp 3
AND
zp 3
ROL
zp 5
RLA
zp 5
PLP
4
AND
imm 2
ROL
2
ANC
imm 2
BIT
abs 4
AND
abs 4
ROL
abs 6
RLA
abs 6
3xBMI
rel 2*
AND
izy 5*
KILRLA
izy 8
NOP
zpx 4
AND
zpx 4
ROL
zpx 6
RLA
zpx 6
SEC
2
AND
aby 4*
NOP
2
RLA
aby 7
NOP
abx 4*
AND
abx 4*
ROL
abx 7
RLA
abx 7
4xRTI
6
EOR
izx 6
KILSRE
izx 8
NOP
zp 3
EOR
zp 3
LSR
zp 5
SRE
zp 5
PHA
3
EOR
imm 2
LSR
2
ALR
imm 2
JMP
abs 3
EOR
abs 4
LSR
abs 6
SRE
abs 6
5xBVC
rel 2*
EOR
izy 5*
KILSRE
izy 8
NOP
zpx 4
EOR
zpx 4
LSR
zpx 6
SRE
zpx 6
CLI
2
EOR
aby 4*
NOP
2
SRE
aby 7
NOP
abx 4*
EOR
abx 4*
LSR
abx 7
SRE
abx 7
6xRTS
6
ADC
izx 6
KILRRA
izx 8
NOP
zp 3
ADC
zp 3
ROR
zp 5
RRA
zp 5
PLA
4
ADC
imm 2
ROR
2
ARR
imm 2
JMP
ind 5
ADC
abs 4
ROR
abs 6
RRA
abs 6
7xBVS
rel 2*
ADC
izy 5*
KILRRA
izy 8
NOP
zpx 4
ADC
zpx 4
ROR
zpx 6
RRA
zpx 6
SEI
2
ADC
aby 4*
NOP
2
RRA
aby 7
NOP
abx 4*
ADC
abx 4*
ROR
abx 7
RRA
abx 7
8xNOP
imm 2
STA
izx 6
NOP
imm 2
SAX
izx 6
STY
zp 3
STA
zp 3
STX
zp 3
SAX
zp 3
DEY
2
NOP
imm 2
TXA
2
XAA
imm 2
STY
abs 4
STA
abs 4
STX
abs 4
SAX
abs 4
9xBCC
rel 2*
STA
izy 6
KILAHX
izy 6
STY
zpx 4
STA
zpx 4
STX
zpy 4
SAX
zpy 4
TYA
2
STA
aby 5
TXS
2
TAS
aby 5
SHY
abx 5
STA
abx 5
SHX
aby 5
AHX
aby 5
AxLDY
imm 2
LDA
izx 6
LDX
imm 2
LAX
izx 6
LDY
zp 3
LDA
zp 3
LDX
zp 3
LAX
zp 3
TAY
2
LDA
imm 2
TAX
2
LAX
imm 2
LDY
abs 4
LDA
abs 4
LDX
abs 4
LAX
abs 4
BxBCS
rel 2*
LDA
izy 5*
KILLAX
izy 5*
LDY
zpx 4
LDA
zpx 4
LDX
zpy 4
LAX
zpy 4
CLV
2
LDA
aby 4*
TSX
2
LAS
aby 4*
LDY
abx 4*
LDA
abx 4*
LDX
aby 4*
LAX
aby 4*
CxCPY
imm 2
CMP
izx 6
NOP
imm 2
DCP
izx 8
CPY
zp 3
CMP
zp 3
DEC
zp 5
DCP
zp 5
INY
2
CMP
imm 2
DEX
2
AXS
imm 2
CPY
abs 4
CMP
abs 4
DEC
abs 6
DCP
abs 6
DxBNE
rel 2*
CMP
izy 5*
KILDCP
izy 8
NOP
zpx 4
CMP
zpx 4
DEC
zpx 6
DCP
zpx 6
CLD
2
CMP
aby 4*
NOP
2
DCP
aby 7
NOP
abx 4*
CMP
abx 4*
DEC
abx 7
DCP
abx 7
ExCPX
imm 2
SBC
izx 6
NOP
imm 2
ISC
izx 8
CPX
zp 3
SBC
zp 3
INC
zp 5
ISC
zp 5
INX
2
SBC
imm 2
NOP
2
SBC
imm 2
CPX
abs 4
SBC
abs 4
INC
abs 6
ISC
abs 6
FxBEQ
rel 2*
SBC
izy 5*
KILISC
izy 8
NOP
zpx 4
SBC
zpx 4
INC
zpx 6
ISC
zpx 6
SED
2
SBC
aby 4*
NOP
2
ISC
aby 7
NOP
abx 4*
SBC
abx 4*
INC
abx 7
ISC
abx 7
+ +
+"*" : add 1 cycle if page boundary is crossed.
+add 1 cycle on branches if taken.
+
+
+Logical and arithmetic commands: + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
OpcodeimpimmzpzpxzpyizxizyabsabxabyindrelFunctionNVBDIZC
ORA $09$05$15 $01$11$0D$1D$19  A:=A or {adr}*    * 
AND $29$25$35 $21$31$2D$3D$39  A:=A&{adr}*    * 
EOR $49$45$55 $41$51$4D$5D$59  A:=A exor {adr}*    * 
ADC $69$65$75 $61$71$6D$7D$79  A:=A+{adr}**   **
SBC $E9$E5$F5 $E1$F1$ED$FD$F9  A:=A-{adr}**   **
CMP $C9$C5$D5 $C1$D1$CD$DD$D9  A-{adr}*    **
CPX $E0$E4    $EC    X-{adr}*    **
CPY $C0$C4    $CC    Y-{adr}*    **
DEC  $C6$D6   $CE$DE   {adr}:={adr}-1*    * 
DEX$CA           X:=X-1*    * 
DEY$88           Y:=Y-1*    * 
INC  $E6$F6   $EE$FE   {adr}:={adr}+1*    * 
INX$E8           X:=X+1*    * 
INY$C8           Y:=Y+1*    * 
ASL$0A $06$16   $0E$1E   {adr}:={adr}*2*    **
ROL$2A $26$36   $2E$3E   {adr}:={adr}*2+C*    **
LSR$4A $46$56   $4E$5E   {adr}:={adr}/2*    **
ROR$6A $66$76   $6E$7E   {adr}:={adr}/2+C*128*    **
+ +

+Move commands: + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
OpcodeimpimmzpzpxzpyizxizyabsabxabyindrelFunctionNVBDIZC
LDA $A9$A5$B5 $A1$B1$AD$BD$B9  A:={adr}*    * 
STA  $85$95 $81$91$8D$9D$99  {adr}:=A       
LDX $A2$A6 $B6  $AE $BE  X:={adr}*    * 
STX  $86 $96  $8E    {adr}:=X       
LDY $A0$A4$B4   $AC$BC   Y:={adr}*    * 
STY  $84$94   $8C    {adr}:=Y       
TAX$AA           X:=A*    * 
TXA$8A           A:=X*    * 
TAY$A8           Y:=A*    * 
TYA$98           A:=Y*    * 
TSX$BA           X:=S*    * 
TXS$9A           S:=X       
PLA$68           A:=+(S)*    * 
PHA$48           (S)-:=A       
PLP$28           P:=+(S)** ****
PHP$08           (S)-:=P       
+ +

+Jump/Flag commands: + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
OpcodeimpimmzpzpxzpyizxizyabsabxabyindrelFunctionNVBDIZC
BPL           $10branch on N=0       
BMI           $30branch on N=1       
BVC           $50branch on V=0       
BVS           $70branch on V=1       
BCC           $90branch on C=0       
BCS           $B0branch on C=1       
BNE           $D0branch on Z=0       
BEQ           $F0branch on Z=1       
BRK$00           (S)-=:PC,P PC:=($FFFE)  1 1  
RTI$40           P,PC:=+(S)** ****
JSR       $20    (S)-:=PC PC:={adr}       
RTS$60           PC:=+(S)       
JMP       $4C  $6C PC:={adr}       
BIT  $24    $2C    N:=b7 V:=b6 Z:=A&{adr}**   * 
CLC$18           C:=0      0
SEC$38           C:=1      1
CLD$D8           D:=0   0   
SED$F8           D:=1   1   
CLI$58           I:=0    0  
SEI$78           I:=1    1  
CLV$B8           V:=0 0     
NOP$EA                   
+ +

+Flags of the status register:
+
+The processor status register has 8 bits, where 7 are used as flags:
+
+N = negative flag (1 when result is negative)
+V = overflow flag (1 on signed overflow)
+# = unused (always 1)
+B = break flag (1 when interupt was caused by a BRK)
+D = decimal flag (1 when CPU in BCD mode)
+I = IRQ flag (when 1, no interupts will occur (exceptions are IRQs forced by BRK and NMIs))
+Z = zero flag (1 when all bits of a result are 0)
+C = carry flag (1 on unsigned overflow)
+

+Hardware vectors:
+
+$FFFA = NMI vector (NMI=not maskable interupts)
+$FFFC = Reset vector
+$FFFE = IRQ vector
+

+Illegal opcodes: + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
OpcodeimpimmzpzpxzpyizxizyabsabxabyindrelFunctionNVBDIZC
SLO  $07$17 $03$13$0F$1F$1B  {adr}:={adr}*2 A:=A or {adr}*    **
RLA  $27$37 $23$33$2F$3F$3B  {adr}:={adr}rol A:=A and {adr}*    **
SRE  $47$57 $43$53$4F$5F$5B  {adr}:={adr}/2 A:=A exor {adr}*    **
RRA  $67$77 $63$73$6F$7F$7B  {adr}:={adr}ror A:=A adc {adr}**   **
SAX  $87 $97$83 $8F    {adr}:=A&X       
LAX  $A7 $B7$A3$B3$AF $BF  A,X:={adr}*    * 
DCP  $C7$D7 $C3$D3$CF$DF$DB  {adr}:={adr}-1 A-{adr}*    **
ISC  $E7$F7 $E3$F3$EF$FF$FB  {adr}:={adr}+1 A:=A-{adr}**   **
ANC $0B          A:=A&#{imm}*    **
ANC $2B          A:=A&#{imm}*    **
ALR $4B          A:=(A&#{imm})/2*    **
ARR $6B          A:=(A&#{imm})/2**   **
XAA² $8B          A:=X&#{imm}*    * 
LAX² $AB          A,X:=#{imm}*    * 
AXS $CB          X:=A&X-#{imm}*    **
SBC $EB          A:=A-#{imm}**   **
AHX¹      $93  $9F  {adr}:=A&X&H       
SHY¹        $9C   {adr}:=Y&H       
SHX¹         $9E  {adr}:=X&H       
TAS¹         $9B  S:=A&X {adr}:=S&H       
LAS         $BB  A,X,S:={adr}&S*    * 
+ +

+¹ = unstable in certain matters
+² = highly unstable (results are not predictable on some machines)
+A = Akkumulator
+X = X-Register
+Y = Y-Register
+S = Stack-Pointer
+P = Status-Register
++(S) = Stack-Pointer relative with pre-increment
+(S)- = Stack-Pointer relative with post-decrement

+

+Combinations of two operations with the same addressing mode:
+
+SLO {adr} = ASL {adr} + ORA {adr}
+RLA {adr} = ROL {adr} + AND {adr}
+SRE {adr} = LSR {adr} + EOR {adr}
+RRA {adr} = ROR {adr} + ADC {adr}
+SAX {adr} = store A&X into {adr}
+LAX {adr} = LDA {adr} + LDX {adr}
+DCP {adr} = DEC {adr} + CMP {adr}
+ISC {adr} = INC {adr} + SBC {adr}

+
+note to SAX: the A&X operation is a result of A and X put onto the bus at the same time.
+

+Combinations of an immediate and an implied command:
+
+ANC #{imm} = AND #{imm} + (ASL)
+ANC #{imm} = AND #{imm} + (ROL)
+ALR #{imm} = AND #{imm} + LSR
+ARR #{imm} = AND #{imm} + ROR
+XAA #{imm} = TXA + AND #{imm}
+LAX #{imm} = LDA #{imm} + TAX
+AXS #{imm} = A&X minus #{imm} into X
+SBC #{imm} = SBC #{imm} + NOP

+
+note to ANC: this command performs an AND operation only, but bit 7 is put into the carry, as if the ASL/ROL would have been executed.
+note to ARR: part of this command are some ADC mechanisms. following effects appear after AND but before ROR: +the V-Flag is set according to (A and #{imm})+#{imm}, bit 0 does NOT go into carry, but bit 7 is exchanged with the carry.
+note to XAA: DO NOT USE!!! Highly unstable!!!
+note to LAX: DO NOT USE!!! On my C128, this opcode is stable, but on my C64-II it loses bits so that the operation +looks like this: ORA #? AND #{imm} TAX.
+note to AXS: performs CMP and DEX at the same time, so that the MINUS sets the flag like CMP, not SBC.
+

+Combinations of STA/STX/STY:
+
+AHX {adr} = stores A&X&H into {adr}
+SHX {adr} = stores X&H into {adr}
+SHY {adr} = stores Y&H into {adr}

+
+note: sometimes the &H drops off. Also page boundary crossing will not work as expected (the bank where the value is stored may be equal to the value stored).
+

+Combinations of STA/TXS and LDA/TSX:
+
+TAS {adr} = stores A&X into S and A&X&H into {adr}
+LAS {adr} = stores {adr}&S into A, X and S

+
+note to LAS: is called as "propably unreliable" in one source.
+

+Bit configuration does not allow any operation on these ones:
+
+NOP = has no effects
+NOP #{imm} = fetches #{imm} but has no effects
+NOP {adr} = fetches {adr} but has no effects
+

+KIL = halts the CPU. the data bus will be set to #$FF

+
+
+Aliases used in other illegal opcode sources:
+
+SLO = ASO
+SRE = LSE
+ISC = ISB
+ALR = ASR
+SHX = A11 (A11 was a result of only having tested this one on adress $1000)
+SHY = A11
+LAS = LAR
+KIL = JAM, HLT
+
+
+The 6502 bugs:
+
+Zeropage index will not leave zeropage when page boundary is crossed:
+
+LDX #$01
+LDA $FF,X

+
+...will fetch from adress $0000 and not $0100 as indexed.
+

+Indirect adressing modes are not able to fetch an adress which crosses the page boundary:
+
+Four examples to illustrate this:
+
+LDA ($FF),Y
+
+LDX #$00
+LDA ($FF,X)
+
+LDX #$FF
+LDA ($00,X)

+
+... will all fetch the low-byte from $00FF and the high-byte from $0000
+
+JMP ($12FF)
+
+... will fetch the low-byte from $12FF and the high-byte from $1200
+

+The N, V and Z flags do not work correctly in BCD mode:
+
+N will always carry bit 7.
+V will always be ((U eor N) nand (U eor V)) (while U is bit 7 of operand 1, V is bit 7 of operand 2 and N is the N flag after the ADC is performed).
+please note that SBC is truly ADC with an inverted operand!
+Z will be 0 when the non-BCD operation WOULD have resulted in $00, no matter what value the result of the BCD operation is.
+
+example to Z:
+
+SED
+CLC
+LDA #$80
+ADC #$80

+
+... results in A=$60, but the Z flag is 1.
+

+BCD and non BCD values:
+
+Since only nibble values from 0 to 9 are valid in BCD, it's interesting to see what happens when using A to F:
+
+$00+$0F=$15 (an easy way to convert a hex-digit into BCD...)
+$00+$1F=$25 (can be claimed as being "ok" since 10+$0F=25)
+$10+$1F=$35 ("ok")
+$05+$1F=$2A (a non-BCD result, still somewhat "ok" since 5+10+$0F=20+$0A)
+$0F+$0A=$1F ("ok", since $0F+$0A=$0F+10)
+$0F+$0B=$10 (now, this is plain bullshit!)

+

+ +Different versions of the 6502:
+
+In the C64/C128 series of computers, slightly modified versions of the 6502 +were used. The modifications did not affect the functional part of the processor +itself. Only a so-called processor port was added. +This port, in combination with an external PLA, was used to map ROM and I/O areas into the 64KB RAM of the C64. +Also, some bits of the port were used for the legendary Datasette.
+
+The port can be accessed through memory adresses $0000 and $0001, while $0001 +is the port itself, and $0000 is the data direction register for it.
+
+Explanation for the bits of $0001:
+
+7 - unused (Flash 8: 0=8MHz/1=1MHz)
+6 - unused (C128: ASCII/DIN sense/switch (1=ASCII/0=DIN))
+5 - Cassette motor control (0 = motor on)
+4 - Cassette switch sense (0 = PLAY pressed)
+3 - Cassette write line
+2 - CHAREN (0=Character ROM instead of I/O area)
+1 - HIRAM ($E000-$FFFF)
+0 - LORAM ($A000-$BFFF)
+
+If HIRAM or LORAM is set, the I/O area is mapped to $D000-$DFFF.
+
+$0000 should always be set to $2F (%00101111)
+
+Note to bit 6: This bit is used to select either the ASCII or the DIN +character ROM of a C128. When data direction is set to INPUT, the charset +is selected externally with the ASCII/DIN key.
+
+CPU versions:
+
+6502: NMOS, used in Commodore disk drives, PET, various other 8 bit computers
+6502C: 6502 with additional HALT pin, used in Atari 8 bit computer range
+6510: 6502 with additional processor port, used in C64
+8500: CMOS version of the 6510, used in C64C and C64G
+8502: 2 MHz version of the 8500, used in C128
+7501: HMOS-1 version of the 6502, used in C16/C116/Plus4
+8501: HMOS-2 version of the 6502, used in C16/C116/Plus4
+
+All of these processors are the same concerning the software-side.
+
+Some processors of the family which are not 100% compatible:
+
+65C02: Extension of the 6502.
+65SC02: Small version of the 65C02 which lost a few opcodes again.
+65CE02: Extension of the 65C02, used in the C65.
+65816: Extended 6502 with new opcodes and 16 bit operation modes.
+

+Zeropage/Stack:
+
+The first 256 bytes of adressable memory are called Zeropage. The 6502 +processor family offers a wide selection of adressing modes to work with +this part of the memory, which generally results in shorter and (even more +important) faster code.
+
+Following the Zeropage, the next 256 bytes (located at $0100-$01FF) are +used as processor stack. The stack function of this part is defined as +it is in most other CPU's: Writing to stack will automatically decrement +the stack pointer, while reading from it will increment it.
+

© 2002-2010 Graham. Last change on 15.12.2010. + + diff --git a/com.wudsn.ide.asm/help/www.oxyron.de/html/opcodes816.html b/com.wudsn.ide.asm/help/www.oxyron.de/html/opcodes816.html new file mode 100644 index 00000000..51d3a8a5 --- /dev/null +++ b/com.wudsn.ide.asm/help/www.oxyron.de/html/opcodes816.html @@ -0,0 +1,562 @@ + +65816 Opcodes + + +65816 Opcode matrix: + + +

+imm = #$00
+sr = $00,S
+dp = $00
+dpx = $00,X
+dpy = $00,Y
+idp = ($00)
+idx = ($00,X)
+idy = ($00),Y
+idl = [$00]
+idly = [$00],Y
+isy = ($00,S),Y
+abs = $0000
+abx = $0000,X
+aby = $0000,Y
+abl = $000000
+alx = $000000,X
+ind = ($0000)
+iax = ($0000,X)
+ial = [$000000]
+rel = $0000 (8 bits PC-relative)
+rell = $0000 (16 bits PC-relative)
+bm = $00,$00
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
x0x1x2x3x4x5x6x7x8x9xAxBxCxDxExF
0xBRK
7
ORA
idx 6
COP
imm 7
ORA
sr 4
TSB
dp 5
ORA
dp 3
ASL
dp 5
ORA
idl 6
PHP
3
ORA
imm 2
ASL
2
PHD
4
TSB
abs 6
ORA
abs 4
ASL
abs 6
ORA
abl 5
1xBPL
rel 2*
ORA
idy 5*
ORA
idp 5
ORA
isy 7
TRB
dp 5
ORA
dpx 4
ASL
dpx 6
ORA
idly 6
CLC
2
ORA
aby 4*
INC
2
TCS
2
TRB
abs 6
ORA
abx 4*
ASL
abx 7
ORA
alx 4
2xJSR
abs 6
AND
idx 6
JSR
abl 8
AND
sr 4
BIT
dp 3
AND
dp 3
ROL
dp 5
AND
idl 6
PLP
4
AND
imm 2
ROL
2
PLD
5
BIT
abs 4
AND
abs 4
ROL
abs 6
AND
abl 5
3xBMI
rel 2*
AND
idy 5*
AND
idp 5
AND
isy 7
BIT
dpx 4
AND
dpx 4
ROL
dpx 6
AND
idly 6
SEC
2
AND
aby 4*
DEC
2
TSC
2
BIT
abx 4*
AND
abx 4*
ROL
abx 7
AND
alx 5
4xRTI
6
EOR
idx 6
WDM
n/a
EOR
sr 4
MVP
bm 1#
EOR
dp 3
LSR
dp 5
EOR
idl 6
PHA
3
EOR
imm 2
LSR
2
PHK
3
JMP
abs 3
EOR
abs 4
LSR
abs 6
EOR
abl 5
5xBVC
rel 2*
EOR
idy 5*
EOR
idp 5
EOR
isy 6
MVN
bm 1#
EOR
dpx 4
LSR
dpx 6
EOR
idly 6
CLI
2
EOR
aby 4*
PHY
3
TCD
2
JMP
abl 4
EOR
abx 4*
LSR
abx 7
EOR
alx 5
6xRTS
6
ADC
idx 6
PER
rell 6
ADC
sr 4
STZ
dp 3
ADC
dp 3
ROR
zp 5
ADC
idl 6
PLA
4
ADC
imm 2
ROR
2
RTL
6
JMP
ind 5
ADC
abs 4
ROR
abs 6
ADC
abl 5
7xBVS
rel 2*
ADC
idy 5*
ADC
idp 5
ADC
isy 7
STZ
dpx 4
ADC
dpx 4
ROR
zpx 6
ADC
idly 6
SEI
2
ADC
aby 4*
PLY
4
TDC
2
JMP
ial 6
ADC
abx 4*
ROR
abx 7
ADC
alx 5
8xBRA
rel 3*
STA
idx 6
BRL
rell 4
STA
sr 4
STY
dp 3
STA
dp 3
STX
dp 3
STA
idl 6
DEY
2
BIT
imm 2
TXA
2
PHB
3
STY
abs 4
STA
abs 4
STX
abs 4
STA
abl 5
9xBCC
rel 2*
STA
idy 6
STA
idp 5
STA
isy 7
STY
dpx 4
STA
dpx 4
STX
dpy 4
STA
idly 6
TYA
2
STA
aby 5
TXS
2
TXY
2
STZ
abs 4
STA
abx 5
STZ
abx 5
STA
alx 5
AxLDY
imm 2
LDA
idx 6
LDX
imm 2
LDA
sr 4
LDY
dp 3
LDA
dp 3
LDX
dp 3
LDA
idl 6
TAY
2
LDA
imm 2
TAX
2
PLB
4
LDY
abs 4
LDA
abs 4
LDX
abs 4
LDA
abl 5
BxBCS
rel 2*
LDA
idy 5*
LDA
idp 5
LDA
isy 7
LDY
dpx 4
LDA
dpx 4
LDX
dpy 4
LDA
idly 6
CLV
2
LDA
aby 4*
TSX
2
TYX
2
LDY
abx 4*
LDA
abx 4*
LDX
aby 4*
LDA
alx 5
CxCPY
imm 2
CMP
idx 6
REP
imm 3
CMP
sr 4
CPY
dp 3
CMP
dp 3
DEC
dp 5
CMP
idl 6
INY
2
CMP
imm 2
DEX
2
WAI
3
CPY
abs 4
CMP
abs 4
DEC
abs 6
CMP
abl 5
DxBNE
rel 2*
CMP
idy 5*
CMP
idp 5
CMP
isy 7
PEI
idp 6
CMP
dpx 4
DEC
dpx 6
CMP
idly 6
CLD
2
CMP
aby 4*
PHX
3
STP
3
JMP
iax 6
CMP
abx 4*
DEC
abx 7
CMP
alx 5
ExCPX
imm 2
SBC
idx 6
SEP
imm 3
SBC
sr 4
CPX
dp 3
SBC
dp 3
INC
dp 5
SBC
idl 6
INX
2
SBC
imm 2
NOP
2
XBA
3
CPX
abs 4
SBC
abs 4
INC
abs 6
SBC
abl 5
FxBEQ
rel 2*
SBC
idy 5*
SBC
idp 5
SBC
isy 7
PEA
abs 5
SBC
dpx 4
INC
dpx 6
SBC
idly 6
SED
2
SBC
aby 4*
PLX
4
XCE
2
JSR
iax 8
SBC
abx 4*
INC
abx 7
SBC
alx 5
+
+"*" : add 1 cycle if page boundary is crossed.
+"#" : add 7 cycles for every byte moved and 1 cycle if page boundary is crossed.
+add 1 cycle if m=0: ADC, AND, BIT, CMP, EOR, LDA, ORA, PHA, PLA, SBC, STA, STZ
+add 2 cycles if m=0 (NOT the implied ones): ASL, DEC, INC, LSR, ROL, ROR, TRB, TSB
+add 1 cycle if x=0: CPX, CPY, LDX, LDY, STX, STY, PLX, PLY, PHX, PHY
+add 1 cycle if e=0: BRK, RTI
+add 1 cycle if direct page register is non zero on direct page adressing modes.
+add 1 cycle on conditional branches if taken.
+native 65816 only: branches do not take one additional cycle when page boundary is crossed.
+
+
+A = Akkumulator
+B = Akkumulator, upper 8 bits
+C = Akkumulator (but always 16 bit not depending on M)
+X = X-Register
+Y = Y-Register
+S = Stack-Pointer
+P = Status-Register
++(S) = Stack-Pointer relative with pre-increment
+(S)- = Stack-Pointer relative with post-decrement
+DBR = Data Bank Register (all data movements with 16 bit adresses will refer to this bank)
+DPR = Direct Page Register (all direct page accesses will use this as adress base)
+PBR = Program Bank Register (the bank the actual code is executed)

+
+Please note that 3 new flags have been added to the P: E, M and X.
+
+While E=1 the 65816 is in 6502 emulation mode and will act like a 6502 in all legal matters.
+DBR, DPR and PBR are still active! Take care: irq's will force PBR=0 without saving the PBR!
+While E=0 the 65816 is in native mode.
+While M=1 the Akku is 8 bits wide.
+While M=0 the Akku is 16 bits wide.
+While X=1 the X and Y registers are 8 bits wide.
+While X=0 the X and Y registers are 16 bits wide.
+
+Also note that the P has virtually lost the B flag in some matters.
+
+
+65816 opcodes: + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
OpcodeFunctionENVMXDIZC
BRAbranch always         
BRLbranch always (long)         
COPcoprocessor enable     **  
MVNmove block backward         
MVPmove block forward         
PEA(S)-:=adr         
PEI(S)-:=adr         
PER(S)-:=adr         
PHB(S)-:=DBR         
PHD(S)-:=DPR         
PHK(S)-:=PBR         
PHX(S)-:=X         
PHY(S)-:=Y         
PLBDBR:=+(S) *     * 
PLDDPR:=+(S) *     * 
PLXX:=+(S) *     * 
PLYY:=+(S) *     * 
REPP:=P nand #{imm} ????????
SEPP:=P or #{imm} ????????
RTLPC:=+(S) (long)         
STPstop CPU         
WAIwait for IRQ         
STZ{adr}:=0         
TCDDPR:=C         
TDCC:=DPR *     * 
TCSS:=C         
TSCC:=S *     * 
TXYY:=X *     * 
TYXX:=Y *     * 
XBAexchange B with A *     * 
XCEexchange C with E*       *
TRB{adr}:={adr} nand A       * 
TSB{adr}:={adr} or A       * 
WDMn/a         
+

+note to STP, WAI: these opcodes need 3 cycles to shut down the CPU.
+
+ + + diff --git a/com.wudsn.ide.asm/help/www.oxyron.de/html/opcodesc02.html b/com.wudsn.ide.asm/help/www.oxyron.de/html/opcodesc02.html new file mode 100644 index 00000000..c3a09d70 --- /dev/null +++ b/com.wudsn.ide.asm/help/www.oxyron.de/html/opcodesc02.html @@ -0,0 +1,492 @@ + + + + + + +65C02 Opcodes + + + + +

65C02 Opcode matrix:

+ +imm = #$00
+zp = $00
+zpx = $00,X
+zpy = $00,Y
+izp = ($00)
+izx = ($00,X)
+izy = ($00),Y
+abs = $0000
+abx = $0000,X
+aby = $0000,Y
+ind = ($0000)
+iax = ($0000,X)
+rel = $0000 (PC-relative) +

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
 x0x1x2x3x4x5x6x7x8x9xAxBxCxDxExF
0xBRK
7
ORA
izx 6
NOP
imm 2
NOPTSB
zp 5
ORA
zp 3
ASL
zp 5
RMB0
zp 5
PHP
3
ORA
imm 2
ASL
2
NOPTSB
abs 6
ORA
abs 4
ASL
abs 6
BBR0
rel 2*
1xBPL
rel 2*
ORA
izy 5*
ORA
izp 5
NOPTRB
zp 5
ORA
zpx 4
ASL
zpx 6
RMB1
zp 5
CLC
2
ORA
aby 4*
INC
2
NOPTRB
abs 6
ORA
abx 4*
ASL
abx 6*
BBR1
rel 2*
2xJSR
abs 6
AND
izx 6
NOP
imm 2
NOPBIT
zp 3
AND
zp 3
ROL
zp 5
RMB2
zp 5
PLP
4
AND
imm 2
ROL
2
NOPBIT
abs 4
AND
abs 4
ROL
abs 6
BBR2
rel 2*
3xBMI
rel 2*
AND
izy 5*
AND
izp 5
NOPBIT
zpx 4
AND
zpx 4
ROL
zpx 6
RMB3
zp 5
SEC
2
AND
aby 4*
DEC
2
NOPBIT
abx 4*
AND
abx 4*
ROL
abx 6*
BBR3
rel 2*
4xRTI
6
EOR
izx 6
NOP
imm 2
NOPNOP
zp 3
EOR
zp 3
LSR
zp 5
RMB4
zp 5
PHA
3
EOR
imm 2
LSR
2
NOPJMP
abs 3
EOR
abs 4
LSR
abs 6
BBR4
rel 2*
5xBVC
rel 2*
EOR
izy 5*
EOR
izp 5
NOPNOP
zpx 4
EOR
zpx 4
LSR
zpx 6
RMB5
zp 5
CLI
2
EOR
aby 4*
PHY
3
NOPNOPEOR
abx 4*
LSR
abx 6*
BBR5
rel 2*
6xRTS
6
ADC
izx 6
NOP
imm 2
NOPSTZ
zp 3
ADC
zp 3
ROR
zp 5
RMB6
zp 5
PLA
4
ADC
imm 2
ROR
2
NOPJMP
ind 6
ADC
abs 4
ROR
abs 6
BBR6
rel 2*
7xBVS
rel 2*
ADC
izy 5*
ADC
izp 5
NOPSTZ
zpx 4
ADC
zpx 4
ROR
zpx 6
RMB7
zp 5
SEI
2
ADC
aby 4*
PLY
4
NOPJMP
iax 6
ADC
abx 4*
ROR
abx 6*
BBR7
rel 2*
8xBRA
rel 3*
STA
izx 6
NOP
imm 2
NOPSTY
zp 3
STA
zp 3
STX
zp 3
SMB0
zp 5
DEY
2
BIT
imm 2
TXA
2
NOPSTY
abs 4
STA
abs 4
STX
abs 4
BBS0
rel 2*
9xBCC
rel 2*
STA
izy 6
STA
izp 5
NOPSTY
zpx 4
STA
zpx 4
STX
zpy 4
SMB1
zp 5
TYA
2
STA
aby 5
TXS
2
NOPSTZ
abs 4
STA
abx 5
STZ
abx 5
BBS1
rel 2*
AxLDY
imm 2
LDA
izx 6
LDX
imm 2
NOPLDY
zp 3
LDA
zp 3
LDX
zp 3
SMB2
zp 5
TAY
2
LDA
imm 2
TAX
2
NOPLDY
abs 4
LDA
abs 4
LDX
abs 4
BBS2
rel 2*
BxBCS
rel 2*
LDA
izy 5*
LDA
izp 5
NOPLDY
zpx 4
LDA
zpx 4
LDX
zpy 4
SMB3
zp 5
CLV
2
LDA
aby 4*
TSX
2
NOPLDY
abx 4*
LDA
abx 4*
LDX
aby 4*
BBS3
rel 2*
CxCPY
imm 2
CMP
izx 6
NOP
imm 2
NOPCPY
zp 3
CMP
zp 3
DEC
zp 5
SMB4
zp 5
INY
2
CMP
imm 2
DEX
2
STP ¹
3
CPY
abs 4
CMP
abs 4
DEC
abs 6
BBS4
rel 2*
DxBNE
rel 2*
CMP
izy 5*
CMP
izp 5
NOPNOP
zpx 4
CMP
zpx 4
DEC
zpx 6
SMB5
zp 5
CLD
2
CMP
aby 4*
PHX
3
WAI ¹
3
NOPCMP
abx 4*
DEC
abx 6*
BBS5
rel 2*
ExCPX
imm 2
SBC
izx 6
NOP
imm 2
NOPCPX
zp 3
SBC
zp 3
INC
zp 5
SMB6
zp 5
INX
2
SBC
imm 2
NOP
2
NOPCPX
abs 4
SBC
abs 4
INC
abs 6
BBS6
rel 2*
FxBEQ
rel 2*
SBC
izy 5*
SBC
izp 5
NOPNOP
zpx 4
SBC
zpx 4
INC
zpx 6
SMB7
zp 5
SED
2
SBC
aby 4*
PLX
4
NOPNOPSBC
abx 4*
INC
abx 6*
BBS7
rel 2*
+ +
+¹ - Only available on WDC 65C02
+
+"*" : add 1 cycle if page boundary is crossed.
+add 1 cycle if direct page register is non zero on direct page adressing modes.
+add 1 cycle on conditional branches if taken.
+add 1 cycle on these commands if D=1: ADC, SBC
+

+A = Akkumulator
+X = X-Register
+Y = Y-Register
+S = Stack-Pointer
+P = Status-Register
++(S) = Stack-Pointer relative with pre-increment
+(S)- = Stack-Pointer relative with post-decrement

+

+These things have changed from 6502 to 65C02:
+
+- new instructions.
+- new adressing modes for a few instrucions.
+- one adressing mode for fetch-modify-write instructions has been optimized, so it takes 1 less cycle in some cases.
+- flags now work correctly in BCD mode (takes 1 additional cycle).
+- absolute-indirect adressing mode now also works on a page boundary (also 1 additional cycle).
+- BRK also affects the decimal flag now.
+- illegal opcodes perform a NOP.
+
+ +65C02 opcodes: + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
OpcodeimpimmzpzpxzpyizxizyabsabxabyindrelFunctionNVBDIZC
BRA           $80branch always       
PHX$DA           (S)-:=X       
PHY$5A           (S)-:=Y       
PLX$FA           X:=+(S)*    * 
PLY$7A           Y:=+(S)*    * 
STZ  $64$74   $9C$9E   {adr}:=0       
TRB  $14    $1C    {adr}:={adr} nand A     * 
TSB  $04    $0C    {adr}:={adr} or A     * 
BBRn           $xFbranch on bit n reset       
BBSn           $xFbranch on bit n set       
RMBn  $x7         {adr}:={adr} nand 2^n     * 
SMBn  $x7         {adr}:={adr} or 2^n     * 
+ +
+The WDC 65C02 also includes the STP and WAI instructions known from the 65816.
+Another version of this processor, called 65SC02, doesn't have the single-bit instructions BBR/BBS/RMB/SMB.
+

+Changes on instructions of the 6502 instruction set:
+
+INC and DEC now have implied adressing modes.
+All major instructions (ORA, ADC, STA...) now also feature the indirect-zeropage adressing mode without index.
+JMP features a new absolute-indirect-indexed adressing mode.
+BIT now works with a few more adressing modes.
+

© 2009-2010 Graham. Last change on 17.12.2010. + + diff --git a/com.wudsn.ide.asm/help/www.oxyron.de/html/opcodesce02.html b/com.wudsn.ide.asm/help/www.oxyron.de/html/opcodesce02.html new file mode 100644 index 00000000..3f3c485a --- /dev/null +++ b/com.wudsn.ide.asm/help/www.oxyron.de/html/opcodesce02.html @@ -0,0 +1,560 @@ + + + + + + +65CE02 Opcodes + + + + +

65CE02 Opcode matrix:

+ + +imm = #$00
+imw = #$0000
+zp = $00
+zpx = $00,X
+zpy = $00,Y
+izx = ($00,X)
+izy = ($00),Y
+izz = ($00),Z
+isy = ($00,S),Y
+abs = $0000
+abx = $0000,X
+aby = $0000,Y
+ind = ($0000)
+iax = ($0000,X)
+rel = $0000 (8 bits PC-relative)
+rell = $0000 (16 bits PC-relative)
+

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
 x0x1x2x3x4x5x6x7x8x9xAxBxCxDxExF
0xBRK
7
ORA
izx 6
CLE
2
SEE
2
TSB
zp 5
ORA
zp 3
ASL
zp 5
RMB0
zp 5
PHP
3
ORA
imm 2
ASL
2
TSY
2
TSB
abs 6
ORA
abs 4
ASL
abs 6
BBR0
rel 2*
1xBPL
rel 2*
ORA
izy 5*
ORA
izz 5
BPL
rell 3*
TRB
zp 5
ORA
zpx 4
ASL
zpx 6
RMB1
zp 5
CLC
2
ORA
aby 4*
INC
2
INZ
2
TRB
abs 6
ORA
abx 4*
ASL
abx 6*
BBR1
rel 2*
2xJSR
abs 6
AND
izx 6
JSR
iab 6
JSR
iax 6
BIT
zp 3
AND
zp 3
ROL
zp 5
RMB2
zp 5
PLP
4
AND
imm 2
ROL
2
TYS
2
BIT
abs 4
AND
abs 4
ROL
abs 6
BBR2
rel 2*
3xBMI
rel 2*
AND
izy 5*
AND
izz 5
BMI
rell 3*
BIT
zpx 4
AND
zpx 4
ROL
zpx 6
RMB3
zp 5
SEC
2
AND
aby 4*
DEC
2
DEZ
2
BIT
abx 4*
AND
abx 4*
ROL
abx 6*
BBR3
rel 2*
4xRTI
6
EOR
izx 6
NEG
2
ASR
2
ASR
zp 5
EOR
zp 3
LSR
zp 5
RMB4
zp 5
PHA
3
EOR
imm 2
LSR
2
TAZ
2
JMP
abs 3
EOR
abs 4
LSR
abs 6
BBR4
rel 2*
5xBVC
rel 2*
EOR
izy 5*
EOR
izz 5
BVC
rell 3*
ASR
zpx 6
EOR
zpx 4
LSR
zpx 6
RMB5
zp 5
CLI
2
EOR
aby 4*
PHY
3
TAB
2
MAP
2
EOR
abx 4*
LSR
abx 6*
BBR5
rel 2*
6xRTS
6
ADC
izx 6
RTS
imm 6
BSR
rell 3*
STZ
zp 3
ADC
zp 3
ROR
zp 5
RMB6
zp 5
PLA
4
ADC
imm 2
ROR
2
TZA
2
JMP
ind 6
ADC
abs 4
ROR
abs 6
BBR6
rel 2*
7xBVS
rel 2*
ADC
izy 5*
ADC
izz 5
BVS
rell 3*
STZ
zpx 4
ADC
zpx 4
ROR
zpx 6
RMB7
zp 5
SEI
2
ADC
aby 4*
PLY
4
TBA
2
JMP
iax 6
ADC
abx 4*
ROR
abx 6*
BBR7
rel 2*
8xBRA
rel 3*
STA
izx 6
STA
isy 7
BRA
rell 4
STY
zp 3
STA
zp 3
STX
zp 3
SMB0
zp 5
DEY
2
BIT
imm 2
TXA
2
STY
abx 5
STY
abs 4
STA
abs 4
STX
abs 4
BBS0
rel 2*
9xBCC
rel 2*
STA
izy 6
STA
izz 5
BCC
rell 3*
STY
zpx 4
STA
zpx 4
STX
zpy 4
SMB1
zp 5
TYA
2
STA
aby 5
TXS
2
STX
aby 5
STZ
abs 4
STA
abx 5
STZ
abx 5
BBS1
rel 2*
AxLDY
imm 2
LDA
izx 6
LDX
imm 2
LDZ
imm 2
LDY
zp 3
LDA
zp 3
LDX
zp 3
SMB2
zp 5
TAY
2
LDA
imm 2
TAX
2
LDZ
abs 4
LDY
abs 4
LDA
abs 4
LDX
abs 4
BBS2
rel 2*
BxBCS
rel 2*
LDA
izy 5*
LDA
izz 5
BCS
rell 3*
LDY
zpx 4
LDA
zpx 4
LDX
zpy 4
SMB3
zp 5
CLV
2
LDA
aby 4*
TSX
2
LDZ
abx 4*
LDY
abx 4*
LDA
abx 4*
LDX
aby 4*
BBS3
rel 2*
CxCPY
imm 2
CMP
izx 6
CPZ
imm 2
DEW
zp 6
CPY
zp 3
CMP
zp 3
DEC
zp 5
SMB4
zp 5
INY
2
CMP
imm 2
DEX
2
ASW
abs 7
CPY
abs 4
CMP
abs 4
DEC
abs 6
BBS4
rel 2*
DxBNE
rel 2*
CMP
izy 5*
CMP
izz 5
BNE
rell 3*
CPZ
zp 3
CMP
zpx 4
DEC
zpx 6
SMB5
zp 5
CLD
2
CMP
aby 4*
PHX
3
PHZ
3
CPZ
abs 4
CMP
abx 4*
DEC
abx 6*
BBS5
rel 2*
ExCPX
imm 2
SBC
izx 6
LDA
isy 7
INW
zp 6
CPX
zp 3
SBC
zp 3
INC
zp 5
SMB6
zp 5
INX
2
SBC
imm 2
NOP
2
ROW
abs 7
CPX
abs 4
SBC
abs 4
INC
abs 6
BBS6
rel 2*
FxBEQ
rel 2*
SBC
izy 5*
SBC
izz 5
BEQ
rell 3*
PHW
imw 4
SBC
zpx 4
INC
zpx 6
SMB7
zp 5
SED
2
SBC
aby 4*
PLX
4
PLZ
4
PHW
abs 4
SBC
abx 4*
INC
abx 6*
BBS7
rel 2*
+ +
+"*" : add 1 cycle if page boundary is crossed.
+add 1 cycle if direct page register is non zero on direct page adressing modes.
+add 1 cycle on conditional branches if taken.
+add 1 cycle on these commands if D=1: ADC, SBC
+

+A = Akkumulator
+B = Bank-Register
+X = X-Register
+Y = Y-Register
+Z = Z-Register
+S = Stack-Pointer
+P = Status-Register
++(S) = Stack-Pointer relative with pre-increment
+(S)- = Stack-Pointer relative with post-decrement

+

+These things have changed from 65C02 to 65CE02:
+
+- new instructions.
+- new adressing modes for a few instrucions.
+- Z-Register is introduced.
+- with the introduction of the Z-Register, also the IZP adressing mode has been changed to IZZ.
+
+ +65CE02 opcodes: + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
OpcodeFunctionNVBDIZC
CPZZ-{adr}*    **
DECA:=A-1*    * 
DEZZ:=Z-1*    * 
DEW{adr}:={adr}-1 (16 bit)*    * 
INCA:=A+1*    * 
INZZ:=Z+1*    * 
INW{adr}:={adr}+1 (16 bit)*    * 
NEGA:=0-A*    * 
ASR{adr}:={adr}/2*    **
ASW{adr}:={adr}*2 (16 bit)*    **
ROW{adr}:={adr}*2+C (16 bit)*    **
LDZZ:={adr}*    * 
STZ{adr}:=Z       
TABB:=A*    * 
TBAA:=B*    * 
TAZZ:=A*    * 
TZAA:=Z*    * 
TSYY:=S*    * 
TYSS:=Y       
PHW(S)-:={adr} (16 bit)       
PHX(S)-:=X       
PHY(S)-:=Y       
PHZ(S)-:=Z       
PLXX:=+(S)*    * 
PLYY:=+(S)*    * 
PLZZ:=+(S)*    * 
TRB{adr}:={adr} nand A     * 
TSB{adr}:={adr} or A     * 
CLEE:=0 ?       
SEEE:=1 ?       
BRAbranch always       
BSR(S)-:=PC / branch always       
BBRnbranch on bit n reset       
BBSnbranch on bit n set       
RMBn{adr}:={adr} nand 2^n     * 
SMBn{adr}:={adr} or 2^n     * 
MAP?       
+ +
+Warning! Some information is based on 65C02 documents, others is just +guessed. There might be some mistakes in here.
+

© 2009 Graham. Last change on 30.1.2009. + + diff --git a/com.wudsn.ide.asm/help/www.oxyron.de/html/registers_antic.html b/com.wudsn.ide.asm/help/www.oxyron.de/html/registers_antic.html new file mode 100644 index 00000000..86bf0ed9 --- /dev/null +++ b/com.wudsn.ide.asm/help/www.oxyron.de/html/registers_antic.html @@ -0,0 +1,298 @@ + + + + + + +ANTIC reference + + + +

ANTIC (AlphaNumeric Television Interface Controller) reference

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
ANTIC register set:
  76543210
$D400 (W)DMACTLunusedDL DMAPM Res.P DMAM DMAPlayfield Width
$D401 (W)CHACTLunusedTurnedInverseOpaque
$D402 (W)DLISTLDisplay List Low (A7-A0)
$D403 (W)DLISTHDisplay List High (A15-A8)
$D404 (W)HSCROLunusedHorizontal Scrolling
$D405 (W)VSCROLunusedVertical Scrolling
$D406 (W) unused
$D407 (W)PMBASEPlayer/Missile Base High (A15-A10)unused
$D408 (W) unused
$D409 (W)CHBASECharacter Set Base High (A15-A9)unused
$D40A (W)WSYNCunused
$D40B (R)VCOUNTVertical Line Counter / 2
$D40C (R)PENHHorizontal Light Pen Trigger Position
$D40D (R)PENVVertical Light Pen Trigger Position
$D40E (W)NMIENDLIVBIResetunused
$D40F (R)NMISTDLIVBIResetunused ¹
$D40F (W)NMIRESunused
+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
ANTIC instruction set:
 76543210
Blank LineDLINumber of scan lines to skip - 10
Mode LineDLILMSVScrollHScrollANTIC Mode (2-15)
JumpDLI0unused$1
Jump/Wait for VBLDLI1unused$1
+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
ANTIC display modes:
ModeColorsResolutionMode HeightPixel Height
22 (Monochrome)32081
32 (Monochrome)320101
45 (4 per char)16081
55 (4 per char)160162
65 (2 per char)16081
75 (2 per char)160162
844088
92 (Monochrome)8044
1048044
112 (Monochrome)16022
122 (Monochrome)16011
13416022
14416011
152 (Monochrome)32011
GTIA 1 (15)16 Lumas/1 Chroma8011
GTIA 2 (15)98011
GTIA 3 (15)1 Luma/16 Chromas8011
+
+¹ - Unused bits read back 1
+

+© 2009-2010 Graham + diff --git a/com.wudsn.ide.asm/help/www.oxyron.de/html/registers_gtia.html b/com.wudsn.ide.asm/help/www.oxyron.de/html/registers_gtia.html new file mode 100644 index 00000000..051f4f9a --- /dev/null +++ b/com.wudsn.ide.asm/help/www.oxyron.de/html/registers_gtia.html @@ -0,0 +1,314 @@ + + + + + + +CTIA/GTIA reference + + + +

CTIA/GTIA (Graphic Television Interface Adapter) reference

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
CTIA/GTIA register set:
  76543210
$D000 (R)M0PFunused ¹Missile 0 to playfield collision
$D000 (W)HPOSP0Horizontal position of player 0
$D001 (R)M1PFunused ¹Missile 1 to playfield collision
$D001 (W)HPOSP1Horizontal position of player 1
$D002 (R)M2PFunused ¹Missile 2 to playfield collision
$D002 (W)HPOSP2Horizontal position of player 2
$D003 (R)M3PFunused ¹Missile 3 to playfield collision
$D003 (W)HPOSP3Horizontal position of player 3
$D004 (R)P0PFunused ¹Player 0 to playfield collision
$D004 (W)HPOSM0Horizontal position of missile 0
$D005 (R)P1PFunused ¹Player 1 to playfield collision
$D005 (W)HPOSM1Horizontal position of missile 1
$D006 (R)P2PFunused ¹Player 2 to playfield collision
$D006 (W)HPOSM2Horizontal position of missile 2
$D007 (R)P3PFunused ¹Player 3 to playfield collision
$D007 (W)HPOSM3Horizontal position of missile 3
$D008 (R)M0PLunused ¹Missile 0 to player collision
$D008 (W)SIZEP0unusedPlayer 0 size
$D009 (R)M1PLunused ¹Missile 1 to player collision
$D009 (W)SIZEP1unusedPlayer 1 size
$D00A (R)M2PLunused ¹Missile 2 to player collision
$D00A (W)SIZEP2unusedPlayer 2 size
$D00B (R)M3PLunused ¹Missile 3 to player collision
$D00B (W)SIZEP3unusedPlayer 3 size
$D00C (R)P0PLunused ¹Player 0 to player collision
$D00C (W)SIZEMMissile 3 sizeMissile 2 sizeMissile 1 sizeMissile 0 size
$D00D (R)P1PLunused ¹Player 1 to player collision
$D00D (W)GRAFP0Player 0 graphic data
$D00E (R)P2PLunused ¹Player 2 to player collision
$D00E (W)GRAFP1Player 1 graphic data
$D00F (R)P3PLunused ¹Player 3 to player collision
$D00F (W)GRAFP2Player 2 graphic data
$D010 (R)TRIG0unused ¹/Trigger
$D010 (W)GRAFP3Player 3 graphic data
$D011 (R)TRIG1unused ¹/Trigger
$D011 (W)GRAFMMissiles graphic data
$D012 (R)TRIG2unused ¹/Trigger
$D012 (W)COLPM0ChromaLumaunused
$D013 (R)TRIG3unused ¹/Trigger
$D013 (W)COLPM1ChromaLumaunused
$D014 (R)PALunused ¹$F (NTSC) / $1 (PAL)
$D014 (W)COLPM2ChromaLumaunused
$D015 (W)COLPM3ChromaLumaunused
$D016 (W)COLPF0ChromaLumaunused
$D017 (W)COLPF1ChromaLumaunused
$D018 (W)COLPF2ChromaLumaunused
$D019 (W)COLPF3ChromaLumaunused
$D01A (W)COLBKChromaLumaunused
$D01B (W)PRIORGTIA mode ²Overlap5th Pl.Priority
$D01C (W)VDELAYP3P2P1P0M3M2M1M0
$D01D (W)GRACTLunusedLatch Tr.enable Penable M
$D01E (W)HITCLRunused
$D01F (R)CONSOLunused ¹/Option/Select/Start
$D01F (W)CONSOLunusedReset?unknown
+
+¹ - Unused bits read back 0
+² - Only available on GTIA
+

+© 2009-2010 Graham + diff --git a/com.wudsn.ide.asm/help/www.oxyron.de/html/registers_pokey.html b/com.wudsn.ide.asm/help/www.oxyron.de/html/registers_pokey.html new file mode 100644 index 00000000..e4d129a6 --- /dev/null +++ b/com.wudsn.ide.asm/help/www.oxyron.de/html/registers_pokey.html @@ -0,0 +1,196 @@ + + + + + + +POKEY reference + + + +

POKEY (POtentiometer and KEYboard Integrated Circuit) reference

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
POKEY register set:
  76543210
$D200 (R)POT0Potentiometer 0 input
$D200 (W)AUDF1Frequency
$D201 (R)POT1Potentiometer 1 input
$D201 (W)AUDC1Sound DistortionOn/OffVolume Control
$D202 (R)POT2Potentiometer 2 input
$D202 (W)AUDF2Frequency
$D203 (R)POT3Potentiometer 3 input
$D203 (W)AUDC2Sound DistortionOn/OffVolume Control
$D204 (R)POT4Potentiometer 4 input
$D204 (W)AUDF3Frequency
$D205 (R)POT5Potentiometer 5 input
$D205 (W)AUDC3Sound DistortionOn/OffVolume Control
$D206 (R)POT6Potentiometer 6 input
$D206 (W)AUDF4Frequency
$D207 (R)POT7Potentiometer 7 input
$D207 (W)AUDC4Sound DistortionOn/OffVolume Control
$D208 (R)POTSTPOT7POT6POT5POT4POT3POT2POT1POT0
$D208 (W)AUDCTLDistortionC1 clockC3 clockC1+C2C3+C4C1 filterC2 filterbase clock
$D209 (R)KBCODECTRLSHIFTKey code
$D209 (W)STIMERunused
$D20A (R)RANDOMRandom generator value
$D20A (W)SKSTRESunused
$D20B (W)POTGOunused
$D20D (R)SERINData
$D20D (W)SEROUTData
$D20E (R)IRQSTBREAKKBCODESERINSEROUTEOTTimer 4Timer 2Timer 1
$D20E (W)IRQENBREAKKBCODESERINSEROUTEOTTimer 4Timer 2Timer 1
$D20F (R)SKCTLSPACESerial speedtwo-toneADC spdPOKEY ctrl
$D20F (W)SKSTATFrameESERIN EKB ESER ESHIFTKEYTransferunused
+
+© 2009-2010 Graham + diff --git a/com.wudsn.ide.asm/help/www.oxyron.de/html/registers_rec.html b/com.wudsn.ide.asm/help/www.oxyron.de/html/registers_rec.html new file mode 100644 index 00000000..0d5b8763 --- /dev/null +++ b/com.wudsn.ide.asm/help/www.oxyron.de/html/registers_rec.html @@ -0,0 +1,113 @@ + + + + + + +REC reference + + + +

REC 8726 (RAM Expansion Controller) reference

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
REC register set:
  76543210
$DF00 (R)SRIRQEOBVerifySizeVersion
$DF01 (R/W)CRExecutereserved ²LoadFF00reserved ²Transfer Type
$DF02 (R/W) C64/C128 start address (A7-A0)
$DF03 (R/W) C64/C128 start address (A15-A8)
$DF04 (R/W) REU start address (A7-A0)
$DF05 (R/W) REU start address (A15-A8)
$DF06 (R/W) unused ¹REU start address (A18-A16)
$DF07 (R/W) Transfer length (D7-D0)
$DF08 (R/W) Transfer length (D15-D8)
$DF09 (R/W) IRQEEOBEVerifyEunused ¹
$DF0A (R/W)ACRC64 fixedREU fixedunused ¹
+
+¹ - Unused bits read back 1
+² - Reserved bits read back 0
+

+ + + + + + + + + + + + + + + + + + +
Transfer Types:
0C64 to REU
1REU to C64
2Swap
3Verify
+

+© 2010 Graham + diff --git a/com.wudsn.ide.asm/help/www.oxyron.de/html/registers_sid.html b/com.wudsn.ide.asm/help/www.oxyron.de/html/registers_sid.html new file mode 100644 index 00000000..678789cc --- /dev/null +++ b/com.wudsn.ide.asm/help/www.oxyron.de/html/registers_sid.html @@ -0,0 +1,254 @@ + + + + + + +SID reference + + + +

SID 6581/8580 (Sound Interface Device) reference

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
SID register set:
  76543210
$D400 (W)FREQLO1Channel 1 Frequency Low-Byte
$D401 (W)FREQHI1Channel 1 Frequency High-Byte
$D402 (W)PWLO1Channel 1 Pulse Width (PW7-0)
$D403 (W)PWHI1unusedChannel 1 Pulse Width (PW11-8)
$D404 (W)CR1NOISEPULSESAWTRITESTRINGSYNCGATE
$D405 (W)AD1Channel 1 AttackChannel 1 Decay
$D406 (W)SR1Channel 1 SustainChannel 1 Release
$D407 (W)FREQLO2Channel 2 Frequency Low-Byte
$D408 (W)FREQHI2Channel 2 Frequency High-Byte
$D409 (W)PWLO2Channel 2 Pulse Width (PW7-0)
$D40A (W)PWHI2unusedChannel 2 Pulse Width (PW11-8)
$D40B (W)CR2NOISEPULSESAWTRITESTRINGSYNCGATE
$D40C (W)AD2Channel 2 AttackChannel 2 Decay
$D40D (W)SR2Channel 2 SustainChannel 2 Release
$D40E (W)FREQLO3Channel 3 Frequency Low-Byte
$D40F (W)FREQHI3Channel 3 Frequency High-Byte
$D410 (W)PWLO3Channel 3 Pulse Width (PW7-0)
$D411 (W)PWHI3unusedChannel 3 Pulse Width (PW11-8)
$D412 (W)CR3NOISEPULSESAWTRITESTRINGSYNCGATE
$D413 (W)AD3Channel 3 AttackChannel 3 Decay
$D414 (W)SR3Channel 3 SustainChannel 3 Release
$D415 (W)FCLOunusedFilter Cutoff Low (FC2-FC0)
$D416 (W)FCHIFilter Cutoff High (FC10-FC3)
$D417 (W)Res/FiltFilter ResonanceFilt ExFilt 3Filt 2Filt 1
$D418 (W)Mode/VolChan 3 OffHigh PassBand PassLow PassVolume
$D419 (R)POTXPotentiometer X
$D41A (R)POTYPotentiometer Y
$D41B (R)OSC3Channel 3 Oscillator
$D41C (R)ENV3Channel 3 Envelope
+
+
+Frequency:
+
+The frequency registers of the SID are written with a 16 bit value which is added to the oscillator every +clock cycle. Calculating a frequency register value for a given frequency can be done using these formulas:
+
+PAL:  x = f * (18*2^24)/17734475 (0 - 3848 Hz)
+NTSC: x = f * (14*2^24)/14318318 (0 - 3995 Hz)
+
+
+Waveforms:
+
+The SID offers four basic waveforms which can be selected by the control register of each channel.
+
+ + + + + + + + + + + + + + + + +
Triangle:Triangle
 
Sawtooth:Sawtooth
 
Pulse:Pulse
 
Noise:Noise
+
+
+Pulse width:
+
+The pulse waveform offers additional control via the 12 bit pulse width registers which are available for +every channel. A value of $800 will result in a square wave.
+
+ + + + + + + + + + + + +
$400Pulse width 25%
 
$800Pulse width 50%
 
$C00Pulse width 75%
+
+
+Noise waveform LFSR:
+
+The noise waveform of the SID is generated by a simple 23 bit LFSR. On shifting, bit 0 is filled with bit 22 EXOR bit 17.
+Unlike the other waveforms, the noise waveform does not use the topmost 8 bits for output. The 8 bits of the noise waveform +are assembled from bits 20, 18, 14, 11, 9, 5, 2 and 0 of the oscillator.
+
+
+Playing a sound:
+
+1. Set frequency and ADSR, and in case you use the pulse wave also set the pulse width.
+2. Set the waveform and the GATE bit in the control register. Setting the gate bit will start the ADSR envelope generator.
+3. Wait for as long as you wish the note to be played.
+4. Clear gate bit. This will start the release phase of the ADSR.
+
+
+© 2009-2012 Graham + diff --git a/com.wudsn.ide.asm/help/www.oxyron.de/html/registers_ted.html b/com.wudsn.ide.asm/help/www.oxyron.de/html/registers_ted.html new file mode 100644 index 00000000..8f1fa916 --- /dev/null +++ b/com.wudsn.ide.asm/help/www.oxyron.de/html/registers_ted.html @@ -0,0 +1,254 @@ + + + + + + +TED reference + + + +

TED 7360 (Text Editing Device) reference

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
TED register set:
  76543210
$FF00 (R/W) Timer 1 High-Byte
$FF01 (R/W) Timer 1 Low-Byte
$FF02 (R/W) Timer 2 High-Byte
$FF03 (R/W) Timer 2 Low-Byte
$FF04 (R/W) Timer 3 High-Byte
$FF05 (R/W) Timer 3 Low-Byte
$FF06 (R/W)CR1TESTECMBMMDENRSELYSCROLL
$FF07 (R/W)CR2REVERSEPAL/NTSCSTOPMCMCSELXSCROLL
$FF08 (R/W) Keyboard Input Latch
$FF09 (R/W)IRQSTIRQICNT3unused ¹ICNT2ICNT1ILPIRSTunused ¹
$FF0A (R/W)IRQENunused ¹ECNT3unused ¹ECNT2ECNT1ELPERSTRST8
$FF0B (R)RASTERRaster Counter bits 7-0
$FF0B (W)RSTCMPRaster Comparator bits 7-0
$FF0C (R/W) unused ¹Cursor Position bits 9-8
$FF0D (R/W) Cursor Position Low-Byte
$FF0E (R/W) Channel 1 Frequency Low-Byte
$FF0F (R/W) Channel 2 Frequency Low-Byte
$FF10 (R/W) unused ¹Chan. 1 Freq. bits 9-8
$FF11 (R/W) D/A modeC2 noiseC2 squareC1 enableVolume (0-8)
$FF12 (R/W) unused ¹Bitmap Address (A15-A13)CHARROMChan. 2 Freq. bits 9-8
$FF13 (R/W) Charset Address (A15-A10)1 MHzROM
$FF14 (R/W) Video RAM Address (A15-A11)unused ¹
$FF15 (R/W)B0Cunused ¹LumaChroma
$FF16 (R/W)B1Cunused ¹LumaChroma
$FF17 (R/W)B2Cunused ¹LumaChroma
$FF18 (R/W)B3Cunused ¹LumaChroma
$FF19 (R/W)ECunused ¹LumaChroma
$FF1A (R/W) unused ¹V. RAM Pos (A9-A8)
$FF1B (R/W) Video RAM Position (A7-A0)
$FF1C (R/W) unused ¹VSCAN8
$FF1D (R/W) Vertical Scan Position Low-Byte
$FF1E (R/W) Horizontal Scan Position
$FF1F (R/W) unused ¹Flash CounterVertical Character Scan Position
$FF3E (W)ROMENunused
$FF3F (W)RAMENunused
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
TED video modes:
ECMBMMMCMMode
000Hires character mode (40x25)
001Multicolor character mode (40x25)
010Hires bitmap mode (320x200)
011Multicolor bitmap mode (160x200)
100Hires character mode with extended background colors (40x25)
+
+¹ - Unused bits read back 1
+

+© 2009-2011 Graham + diff --git a/com.wudsn.ide.asm/help/www.oxyron.de/html/registers_vic1.html b/com.wudsn.ide.asm/help/www.oxyron.de/html/registers_vic1.html new file mode 100644 index 00000000..efefcd6c --- /dev/null +++ b/com.wudsn.ide.asm/help/www.oxyron.de/html/registers_vic1.html @@ -0,0 +1,107 @@ + + + + + + +VIC reference + + + +

VIC 6560/6561 (Video Interface Chip) reference

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
VIC register set:
  76543210
$9000 (R/W) InterlaceScreen Origin X
$9001 (R/W) Screen Origin Y
$9002 (R/W) Screen A9Number of Video Columns
$9003 (R/W) Raster b0Number of Video RowsChar Size
$9004 (R/W) Raster Value bits 8-1
$9005 (R/W) Screen Memory Location (A13-A10)Char Memory Location (A13-A10)
$9006 (R/W) Lightpen X
$9007 (R/W) Lightpen Y
$9008 (R/W) Paddle X
$9009 (R/W) Paddle Y
$900A (R/W) SwitchBass Frequency
$900B (R/W) SwitchAlto Frequency
$900C (R/W) SwitchSoprano Frequency
$900D (R/W) SwitchNoise Frequency
$900E (R/W) Auxiliary ColorVolume Control
$900F (R/W) Screen ColorReverseBorder Color
+

+© 2011 Graham + diff --git a/com.wudsn.ide.asm/help/www.oxyron.de/html/registers_vic2.html b/com.wudsn.ide.asm/help/www.oxyron.de/html/registers_vic2.html new file mode 100644 index 00000000..4064c07f --- /dev/null +++ b/com.wudsn.ide.asm/help/www.oxyron.de/html/registers_vic2.html @@ -0,0 +1,385 @@ + + + + + + +VIC-II reference + + + +

VIC-II 6567/6569/856x (Video Interface Chip 2) reference

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
VIC-II register set:
  76543210
$D000 (R/W)M0XSprite 0 X-position
$D001 (R/W)M0YSprite 0 Y-position
$D002 (R/W)M1XSprite 1 X-position
$D003 (R/W)M1YSprite 1 Y-position
$D004 (R/W)M2XSprite 2 X-position
$D005 (R/W)M2YSprite 2 Y-position
$D006 (R/W)M3XSprite 3 X-position
$D007 (R/W)M3YSprite 3 Y-position
$D008 (R/W)M4XSprite 4 X-position
$D009 (R/W)M4YSprite 4 Y-position
$D00A (R/W)M5XSprite 5 X-position
$D00B (R/W)M5YSprite 5 Y-position
$D00C (R/W)M6XSprite 6 X-position
$D00D (R/W)M6YSprite 6 Y-position
$D00E (R/W)M7XSprite 7 X-position
$D00F (R/W)M7YSprite 7 Y-position
$D010 (R/W)M?X8M7X8M6X8M5X8M4X8M3X8M2X8M1X8M0X8
$D011 (R/W)CR1RST8ECMBMMDENRSELYSCROLL
$D012 (R)RASTERRaster Counter bits 7-0
$D012 (W)RSTCMPRaster Comparator bits 7-0
$D013 (R)LPXLight Pen X-position
$D014 (R)LPYLight Pen Y-position
$D015 (R/W)M?EM7EM6EM5EM4EM3EM2EM1EM0E
$D016 (R/W)CR2unused ¹RESMCMCSELXSCROLL
$D017 (R/W)M?YEM7YEM6YEM5YEM4YEM3YEM2YEM1YEM0YE
$D018 (R/W)VM/CBScreen Pointer (A13-A10)Bitmap/Charset Pointer (A13-A11)unused ¹
$D019 (R/W)IRQSTIRQunused ¹ILPIMMCIMBCIRST
$D01A (R/W)IRQENunused ¹ELPEMMCEMBCERST
$D01B (R/W)M?DPM7DPM6DPM5DPM4DPM3DPM2DPM1DPM0DP
$D01C (R/W)M?MCM7MCM6MCM5MCM4MCM3MCM2MCM1MCM0MC
$D01D (R/W)M?XEM7XEM6XEM5XEM4XEM3XEM2XEM1XEM0XE
$D01E (R/W)M?MM7MM6MM5MM4MM3MM2MM1MM0M
$D01F (R/W)M?DM7DM6DM5DM4DM3DM2DM1DM0D
$D020 (R/W)ECunused ¹Border Color
$D021 (R/W)B0Cunused ¹Background Color 0
$D022 (R/W)B1Cunused ¹Background Color 1
$D023 (R/W)B2Cunused ¹Background Color 2
$D024 (R/W)B3Cunused ¹Background Color 3
$D025 (R/W)MM0unused ¹Sprite Multicolor 0
$D026 (R/W)MM1unused ¹Sprite Multicolor 1
$D027 (R/W)M0Cunused ¹Sprite 0 Color
$D028 (R/W)M1Cunused ¹Sprite 1 Color
$D029 (R/W)M2Cunused ¹Sprite 2 Color
$D02A (R/W)M3Cunused ¹Sprite 3 Color
$D02B (R/W)M4Cunused ¹Sprite 4 Color
$D02C (R/W)M5Cunused ¹Sprite 5 Color
$D02D (R/W)M6Cunused ¹Sprite 6 Color
$D02E (R/W)M7Cunused ¹Sprite 7 Color
$D02F (R/W)KCR ²unused ¹Keyboard Interface ²
$D030 (R/W)FAST ²unused ¹TEST ²2 MHz ²
+
+¹ - Unused bits read back 1
+² - Only available on C128
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + +
VIC-II video modes:
ECMBMMMCMMode
000Hires character mode (40x25)
001Multicolor character mode (40x25)
010Hires bitmap mode (320x200)
011Multicolor bitmap mode (160x200)
100Hires character mode with extended background colors (40x25)
+

+ + + + + + + + + + + + + + + + + + + + + + + + + +
VIC-II color palette:
01234567
VIC-II color palette
89ABCDEF
+

+© 2009-2011 Graham + diff --git a/com.wudsn.ide.asm/help/www.oxyron.de/pics/noise.gif b/com.wudsn.ide.asm/help/www.oxyron.de/pics/noise.gif new file mode 100644 index 0000000000000000000000000000000000000000..f29b6a90127986309158a74b5f632e69fc94df17 GIT binary patch literal 2858 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 INI(DpJ0~Nt0RR91 literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/help/www.oxyron.de/pics/pulse.gif b/com.wudsn.ide.asm/help/www.oxyron.de/pics/pulse.gif new file mode 100644 index 0000000000000000000000000000000000000000..d80230ddccdde06ba996d67714df39851def195f GIT binary patch literal 2112 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 literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/help/www.oxyron.de/pics/pulse1.gif b/com.wudsn.ide.asm/help/www.oxyron.de/pics/pulse1.gif new file mode 100644 index 0000000000000000000000000000000000000000..8058ebaa2cc9aad68bd5bbf9f65c7da4660c5677 GIT binary patch literal 1505 zcmV<71s?iGNk%v~VE_R_0OuY6000010RaL60s{jB1Ox;H1qB8M1_uWR2nYxX2?+`c z3JVJh3=9kn4Gj(s4i66x5D*X%5fKs+5)%^>6ciK{6%`g178e&67#J8C85tTH8XFrM z92^`S9UUGX9v>ecARr(iAt53nA|oRsBqSsyB_$>%CMPE+C@3f?DJd!{Dl021EG#T7 zEiEoCE-x=HFfcGNF)=bSGBYzXG&D3dH8nOiHa9mnI5;>tIXOByIy*Z%JUl!-Jv}}? zK0iM{KtMo2K|w-7LPJACL_|bIMMXwNMn^|SNJvOYNl8jdN=r*iOiWBoO-)WtPESuy zP*6}&QBhJ-Qd3h?R8&+|RaI72R##V7SXfwDSy@_IT3cINTwGjTU0q&YUSD5dU|?Wj zVPRroVq;@tWMpJzWo2e&W@l$-XlQ6@X=!R|YHMq2Y;0_8ZEbFDZf|dIaBy&OadC2T za&vQYbaZreb#-=jc6WDoczAeud3kzzdV70&e0+R;eSLm@et&;|fPjF3fq{a8f`fyD zgoK2Jg@uNOhKGlTh=_=ZiHVAeii?YjjEszpjg5|uj*pLzkdTm(k&%*;l9Q8@l$4Z} zm6ev3mY0{8n3$NEnVFiJnwy)OoSdAUot>VZo}ZteprD|kp`oIpqNAguq@<*!rKP5( zrl+T;sHmu^si~@}s;jH3tgNi9t*x%EuCK4Ju&}VPv9YqUva_?Zw6wIfwY9dkwzs#p zxVX5vxw*Q!y1To(yu7@dCU$jHda z$;ryf%FD~k%*@Qq&CSlv&d<-!(9qD)(b3Y<($mw^)YR0~)z#M4*4Nk9*x1lt)=I7_<=;-L_>FMg~>g((4 z?Ck9A?d|UF?(gsK@bK{Q@$vHV^7Hfa^z`)g_4W4l_V@Sq`1ttw`T6?#`uqF){QUg= z{r&#_{{R2~EC2ui0003(000R70RIUbNU)&6g9sBUT*$DY!-o(fN}NcsqD6@SGHTq& zv7^V2AU8e?NwTC#jU!L0T#3>l%a$7V8?wS>y?}sFeTGwo0LuLIsZ)Dj^X~(ub88=ehvwaT*o?Ez2;=hgWL_S>kP3Fg))0BP@ z`f}<#c~{n+nR{pNp~3!_7N44YYxA}LHpX7j`g8Ua-A_lqTm57BKk;MKPn17X|E{eS z9DM`YcN>AlA()(k%`q6AgV#a$o`mxm*c*k_0f?S}30k7$c2D9@t}tNnW@lk4=6UWr$Ob zSY?S7J2v>Fqi6blEqqjteVRX z8m*$!vRSR0*TR{toZHgL>Y}MO>T9!_K1(UNu$qf0y5p+Lsk@%S3#z=L(o3qnrs9jL zzN+f^ZMT5_JFKz;Gs~~6{?b~ov$;?7_w+jO@h$ zXAH8lB121Zp#pnMu*(LgeD2BR%DgVk>Dv6R&hhd*ug{;Y6sXWf61|wwBiE_)(o8pf HP(T1X=tA{3 literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/help/www.oxyron.de/pics/pulse2.gif b/com.wudsn.ide.asm/help/www.oxyron.de/pics/pulse2.gif new file mode 100644 index 0000000000000000000000000000000000000000..be454a953ab430413ec2248e0e673772340541e7 GIT binary patch literal 1510 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 P(!FIyE%nqH1q1*)fxG%e literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/help/www.oxyron.de/pics/sawtooth.gif b/com.wudsn.ide.asm/help/www.oxyron.de/pics/sawtooth.gif new file mode 100644 index 0000000000000000000000000000000000000000..8530ad3c48d9cbe8435dcd82d500c8832846d13d GIT binary patch literal 2162 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 literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/help/www.oxyron.de/pics/triangle.gif b/com.wudsn.ide.asm/help/www.oxyron.de/pics/triangle.gif new file mode 100644 index 0000000000000000000000000000000000000000..14c3b8bd837ab798bb5dc72a8ec9e1e2c71356fd GIT binary patch literal 2156 zcmV-y2$T0mNk%v~VE_U`0OuY6000010RaL60s{jB1Ox;H1qB8M1_uWR2nYxX2?+`c z3JVJh3=9kn4Gj(s4i66x5D*X%5fKs+5)%^>6ciK{6%`g178e&67#J8C85tTH8XFrM z92^`S9UUGX9v>ecARr(iAt53nA|oRsBqSsyB_$>%CMPE+C@3f?DJd!{Dl021EG#T7 zEiEoCE-x=HFfcGNF)=bSGBYzXG&D3dH8nOiHa9mnI5;>tIXOByIy*Z%JUl!-Jv}}? zK0iM{KtMo2K|w-7LPJACL_|bIMMXwNMn^|SNJvOYNl8jdN=r*iOiWBoO-)WtPESuy zP*6}&QBhJ-Qd3h?R8&+|RaI72R##V7SXfwDSy@_IT3cINTwGjTU0q&YUSD5dU|?Wj zVPRroVq;@tWMpJzWo2e&W@l$-XlQ6@X=!R|YHMq2Y;0_8ZEbFDZf|dIaBy&OadC2T za&vQYbaZreb#-=jc6WDoczAeud3kzzdV70&e0+R;eSLm@et&;|fPjF3fq{a8f`fyD zgoK2Jg@uNOhKGlTh=_=ZiHVAeii?YjjEszpjg5|uj*pLzkdTm(k&%*;l9Q8@l$4Z} zm6ev3mY0{8n3$NEnVFiJnwy)OoSdAUot>VZo}ZteprD|kp`oIpqNAguq@<*!rKP5( zrl+T;sHmu^si~@}s;jH3tgNi9t*x%EuCK4Ju&}VPv9YqUva_?Zw6wIfwY9dkwzs#p zxVX5vxw*Q!y1To(yu7@dCU$jHda z$;ryf%FD~k%*@Qq&CSlv&d<-!(9qD)(b3Y<($mw^)YR0~)z#M4*4Nk9*x1lt)=I7_<=;-L_>FMg~>g((4 z?Ck9A?d|UF?(gsK@bK{Q@$vHV^7Hfa^z`)g_4W4l_V@Sq`1ttw`T6?#`uqF){QUg= z{r&#_{{R2~EC2ui0006)000R70RIUbNU)&6g98B|T*$DY!-onVN}NcMV8n_TGakIC zv13M#9z#k5DYE23k|tAHRLRmH%9bWy!W1cUrpKB#HR8;PaVJlTK7WoBT2f(9q9^|y zMY>Tb(~d@;>U1i#C)KJzuVNj#bZgYD6TgCmsC6tuv1ZSfP2039Rv@Tzl3I7({D>$&e0TEeu~${UUHW+N@9|f6zaRd0|M>yf zAAjwsb)SIAHH2PDI4u~Hg8%7PAc1c|c%Xz88W&dY?OCXud0J&Co`@Rx=$nx`%I2XwC0gg8mqMDUq@7awsikguYSU?o z2CAw`sD^5)dHztjD(XzFmWnB@vD*4-dbu_Vs-IKoIxDQf`bui0y&_wzvcHa+th38P z+pM4~0{bkr*Pe=^vD0c>EtlUidn~rmlG`n@+lqTGOADsT9;3L@1a7+0mWyt<-qLHX zy}Zh+?zs16+poU);u|l)0UOLNgY^o`@Pqq03~^Bn^ZPKx5nDX5zXc<_@Wu&CjPb|^ zb6m2=7lX{|#~~}MGRh>!yz7St+(A+@BJmyX$uY`;EUF*G}wkG-t^yfLw>i~hf5wfdQ-?J@%zXV322PKE6@=gR zMFVwBH9g_&xlE(0{}G*9k>9!0@%Ogzux^3jO!O7>*Ex1gzK$5m&<;`VfQu zJ5*l}|7SxB!q9|7RN)Victay5QHM(eViaXKMeq$VidTH%1BF#ZCr%KGT9jfKsW?U} zVrhe7+@h?$sKh2BFN;5f;}hlhMLH(Xj%s8h81u-jBSJ8UsKa9l-RQ+ZS__bb6C@uM zIY>q}C_-Hnqa!cKMML(nj3%5U8Z!w=OiHqodQ>7K)i}!Gm6AT7w4|_3iAdwI(vPc* z4<9ErOIf|Lh<^-aE(wXjQa;d^iUFo}Iw?$C9+H<4D`q6srA$vUb2z#@W-TYx%xLBk zn9RXuS+rT3<8afNfBdF3k10-MN~@gH+*CRZg-%tf)17zHW@6-d6?#%;JN8yIC$jRn z&Q0+Xp8iD4JOx@$g5L8x|Kuk<{jyM}G!!nGc_=_DB+-fzl%W@OXhtC#8jD_(qnnXv iM*FEH*BEr969v?ARP#}kHk70)l_fM!I#K`x1OPi`TQc$h literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/help/www.oxyron.de/pics/vic2palette.png b/com.wudsn.ide.asm/help/www.oxyron.de/pics/vic2palette.png new file mode 100644 index 0000000000000000000000000000000000000000..c2b683d8c85f64b11d2557c7f044700e18831355 GIT binary patch literal 1047 zcmeAS@N?(olHy`uVBq!ia0y~yU<5K595|SP2>rL)|=mOLs z0n*`|pPQSSSHcjSUs{x$YNZgCnVhO%q-T^YT?NW$HJ&bxAr-gYUSMQwa$sQCC@v_# z`g^ NZC (Ind,X) 2/8 + +Double NOP: + No operation command that takes 3 cycles and uses two bytes. + DOP Opcode:$04 [no operation] (Z-Page) 2/3 + +Load A and X: + Loads both A and X with the memory. + LAX Opcode:$AF A <- M, X <- M (Absolute) 3/4 + + + +============================================================== +============================================================== +Insurance Against Too Many VBLANK/Overscan Cycles +-------------------------------------------------------------- +Paul Slocum +============================================================== +============================================================== + +It's difficult to make sure that there is no unusual case where your game logic +in VBlank and Overscan will use too many cycles and cause the number of +scanlines in a frame to fluctuate. To insure against this problem, pad your +VBlank and Overscan with a few STA WSYNCs to add extra cycles during development +. Optimize your game so that it runs fine with these in. Then when the game is +finished and ready for release, comment out those extra WSYNCs. This is also an +easy way to estimate how much time you have left in VBlank and Overscan: keep +adding WSYNCs (each line is 76 cycles) until the screen jumps. + + + +============================================================== +============================================================== +The Multi-Sprite Trick +-------------------------------------------------------------- +Christopher Tumber +Todo: Add ASCII Images +============================================================== +============================================================== + +The multi-sprite trick is a technique which allows a programmer to put more sprites on a scanline than would normally be +allowable. Using this trick, up to 18 [?] sprites may be displayed on a scanline. + +This trick may be used with Player0, Player1, Missile0 and/or Missile1 in any combination. The most common use is with +P0 and P1 to create a row of sprites. + +There are important limitations with this trick. This trick essentially allows you to create more than the normal limit +of 3 copies of a sprite. However, in doing so you probably will not have time to change the bitmap, colour or size of the +sprite (as applicable). So the kinds of displays created with this technique will usually be repetitive patterns. + +The trick is accomplished by first setting NUSIZ0 and/or NUSIZ1 to display 2 or more copies of the sprites you want to +display [Is the exact setting significant?]. Your program then must strobe RESP0, RESP1, RESM0 and/or RESM1 repeatedly. +If this is timed correctly so that it occurs after the first copy is drawn but before the second then the TIA is tricked +into thinking it's just started drawing sprites and contiues drawing the sprite as if it's the first copy. You just +continue this for every copy you need. + +The tightest formation of sprites possible with this trick is: + +[image] + +which is done by the following code: + + +sta RESP0 +sta RESP1 +sta RESP0 +sta RESP1 +sta RESP0 +sta RESP1 +sta RESP0 +sta RESP1 +sta RESP0 +sta RESP1 + +However, this formation has one problem in that the last sprite on a row is shifted one pixel to the left. There is no +known soloution to this problem. If you can work this "glitch" into your game (for example as part of an asynchronous +display) then you can use this layout. If not, the closest "stable" formation is: + +[image] + +which is done by the following code: + +sta RESP0,x +sta RESP1,x +sta RESP0,x +sta RESP1,x +sta RESP0,x +sta RESP1,x +sta RESP0,x +sta RESP1,x +sta RESP0,x +sta RESP1,x + +or + +sta.w RESP0 +sta.w RESP1 +sta.w RESP0 +sta.w RESP1 +sta.w RESP0 +sta.w RESP1 +sta.w RESP0 +sta.w RESP1 +sta.w RESP0 +sta.w RESP1 + +The ,x and .w in this example are "dummies". They're only used to increased the length of time taken by each instruction +by 1 cycle. The tradeoff being that the former requires the X register be set to zero and the latter results in slightly +larger code. + +If you need to turn off some of the sprites, you can do this by simply skipping their spot by inserting a dummy command. +For example: + +[image] + +sta RESP0 +sta RESP1 +sta Dummy +sta RESP1 +sta RESP0 +sta Dummy +sta RESP0 +sta RESP1 +sta RESP0 +sta RESP1 + +Where Dummy is an unsued (or scratch or temporary) RAM location. + +This trick is quite easy to use to generate static displays. However, if you want a fully dynamic display things get +considerably more complicated. + +Since there is no time available while drawing the sprites to do any calculations, if you want a variable number of +sprites on and off you must predetermine which sprites to display. A simple way to do this is to create a subroutine +for every possible combination of on/off sprites. Then your program just needs to call the appropriate subroutine. +The problem with this approach is that if you have a lot of sprites, the number of subroutines becomes very large. +An alternative method is to place you drawing routine in RAM, adjusted for the current display - In the above example, +copy all the STA RESP0 and STA RESP1 commands into RAM and then where sprites don't appear, substitue in a dummy RAM +location for the relevant RESP0 or RESP1. + +One thing you must be aware of if you are allowing individual sprites to be switched on and off. There are a number +of cominations which you must treat as an exception. They cannot be displayed using the general display as above. +For example, if either RESP0 or RESP1 needs to display only 1 copy of a sprite (because all other copies are off) +then you must reset NUSIZ0/NUSIZ1. This would also be the case when two copies of a sprite are set too far apart +for the second STA RESPn to occur before a "normal" copy is drawn. + +In addition, these sprites are not positioned vertically like normal sprites. Rather their position is determined by which +display cycle the first STA RESPn or RESPMn occurs. So if you want to be able to reposition your sprites verically, +you will either need to add more subroutines (as above) or adjust your RAM routine further. Or some combination of the +two. Space Instigators uses a different subroutine for each possible vertical position, copies that routine into RAM +and then modifies the STA RESPn commands to turn off dead Instigators. The new, single scanline repositioning routine may +be of help here, however the multi-sprite trick tends to use up so much of your scanline time that if you're +trying to do other things (ie: display other sprites) on that scanline you may not have the luxury of enough +cycles for general purpose positioning code. + +The multi-sprite trick has a side effect in that the formation of sprites is shifted [?exact number?] pixels right +as compared to where a normal sprite would appear with an STA RESPn at that cycle. This results in a left margin +that's not at the left edge of the screen. "Illegal" HMOVE/HMMn combination tricks may be used to fix this but with a +corresponding increase in complexity. + +[Some more example code here] + +References: The multi-sprite trick was originally used in Galaxian and was pioneered by Eckhard Stolberg, John +Saeger, Erik Mooney and Thomas Jentzsch. Search Stella List under "Grid demo","trick18","trick12" and "inv3". + + +============================================================== +============================================================== +Paddles +-------------------------------------------------------------- +Thomas Jentzch + Todo: Explain and include how to discharge cap +============================================================== +============================================================== + +Assumes Y is your kernal line counter. + + lda INPT0 ;3 + bmi paddles1 ;2 or 3 + .byte $2c ;1 bit abs opcode + paddles1: + sty padVal1 ;3 + + + +============================================================== +============================================================== +Showing Missiles/Ball using PHP +============================================================== +============================================================== + +This trick is originally from Combat and is probably the most efficient way to +display the missiles and/or ball. This trick just requires that you don't use +the stack during your kernal. Recall that: + + ENABL = $1F + ENAM1 = $1E + ENAM0 = $1D + +In this example I'll show how to use the trick for both missiles. You can +easily adapt it for the ball too. To set the trick up, before your kernal save +the stack pointer and set the top of the stack to ENAM1+1. + + tsx ; Transfer stack pointer to X + stx SavedStackPointer ; Store it in RAM + ldx #ENAM1+1 + txs ; Set the top of the stack to ENAM1+1 + +Now during the kernal you can compare your scanline counter to your missile +position register and this will set the zero flag in the processor. Then to +enable/disable the missile for that scanline, just push the processor flags onto +the stack. The ENAxx registers use bit 1 to enable/disable which corresponds +with the zero flag in the processor, so the enable/disable will be automatic. +It takes few cycles and doesn't vary the number of cycles depending on the +result like branching usually does. + + ; On each line of your the kernal... + cpy MissilePos1 ; Assumes Y is your kernal line counter + php + cpy MissilePos0 + php + +Then before you do it again, somewhere on each scanline you need to pull off the +stack again using two PLA's or PLP's, or you can manually reset the stack +pointer with ldx #ENAM1+1, txs. + +After your kernal, restore the stack pointer: + + ldx SavedStackPointer + txs + + +============================================================== +============================================================== +Skipdraw +-------------------------------------------------------------- +Thomas Jentzch + Todo: Explain and clean up +============================================================== +============================================================== + +The best way, i knew until now, was (if y contains linecounter): + + tya ; 2 +; sec ; 2) <- this can sometimes be avoided + sbc SpriteEnd ; 3 + adc #SPRITEHEIGHT ; 2 + bcx .skipDraw ; 2 = 9-11 cycles + ... + +---------- or --------- + +If you like using illegal opcodes, you can use dcp (dec,cmp) here: + + lda #SPRITEHEIGHT ; 2 + dcp SpriteEnd ; 5 initial value has to be adjusted + bcx .skipDraw ; 2 = 9 + ... + +Advantages: +- state of carry flag doesn't matter anymore (may save 2 cycles) +- a remains constant, could be useful for a 2nd sprite +- you could use the content of SpriteEnd instead of y for accesing sprite data + +;================================== +;An Example: +; + ; skipDraw routine for right player + TXA ; 2 A-> Current scannline + SEC ; 2 Set Carry + SBC slowP1YCoordFromBottom+1 ; 3 + ADC #SPRITEHEIGHT+1 ; 2 calc if sprite is drawn + BCC skipDrawRight ; 2/3 To skip or not to skip? + TAY ; 2 + lda P1Graphic,y ; 4 +continueRight: + STA GRP0 + +;----- this part outside of kernel + +skipDrawRight ; 3 from BCC + LDA #0 ; 2 + BEQ continueRight ; 3 Return... + + + +============================================================== +============================================================== +Sound and Music +-------------------------------------------------------------- +============================================================== +============================================================== + +Atari 2600 Music Programming Guide and music driver code: + +http://qotile.net/sequencer.html + +Eckhard Strolberg's Frequency and Waveform Guide: + +http://buerger.metropolis.de/estolberg + + + +============================================================== +============================================================== +Using BRK with RESXX +-------------------------------------------------------------- +Eckhard Strolberg +============================================================== +============================================================== + +(I'm not sure what you'd do with this trick but it's pretty interesting. Maybe +somebody will figure out how to use it.) + +Pole Position puts the stack pointer over the RESxx registers and then does a +BRK. There are three write cycles in a BRK instruction, so the three position +registers for the objects that make up the road in PP, get accessed in three +consecutive cycles. This is how PP managed to get the road to meet so closely in +the horizon. + + + +============================================================== +============================================================== +Wasting Cycles +-------------------------------------------------------------- +Christopher Tumbler, Chris Wilkson, Andrew Davie +============================================================== +============================================================== + +These are the most efficient ways to waste processor cycles. + +Note that locations $2D-$3F do nothin and aren't decoded, and so they are used +often here. In some bankswitching schemes this could cause problems though. + +----------------- +1 Cycle (0 or 1 byte) + .w (Change a zero page instruction to absolute, adds 1 byte of code) + ,x (Change a zero page or absolute instruction to an indexed instruction. + Make sure x=0. Can also use Y) +----------------- +2 Cycles (1 byte) + nop +----------------- +3 Cycles (2 bytes) + sta $2D + - or - + lda $2D + - or - + dop (Double NOP illegal opcode) +----------------- +4 Cycles (2 bytes) + nop + nop +----------------- +5 Cycles (2 bytes) + dec $2D + - or - + sta $1800,X ; asssumes you can write to ROM without problems +----------------- +6 Cycles (2 bytes) + lda ($80,X) ; assumes possible reads from 0-$7f have no effect + +6 Cycles (3 bytes) + nop + nop + nop +----------------- +7 Cycles (2 bytes, need 1 byte free on stack) + pha + pla +----------------- +8 Cycles (3 bytes) + lda ($80,X) ; assumes possible reads from 0-$7f have no effect + nop +----------------- +9 Cycles (3 bytes, need 1 byte free on stack) + pha + pla + nop + +9 Cycles (4 bytes) + dec $2D + nop + nop +----------------- +10 Cycles (4 bytes) + dec $2D + dec $2D + - or - + rol $80 + rol $80 ; leaves $80 unchanged +----------------- +11 Cycles (4 bytes) + ASSUMING we can safely write to ROM and have nothing disasterous... + STA $8000,X + LDA ($80,X) ; assumes possible reads from 0-$7f have no effect +----------------- +12 Cycles (3 bytes, need 2 bytes free on stack) + jsr return + ; somewhere else + return: + rts +----------------- +12 Cycles (4 bytes) + LDA ($80,X) ; assumes possible reads from 0-$7f have no effect + LDA ($80,X) ; assumes possible reads from 0-$7f have no effect + +Also: +You can use PHA/PHP (1 byte 3 cycles) or PLA/PLP (1 byte 4 cycles) alone but +you have to be carefull not to mess up your stack (PLP/PHA would be usefull if +you have no stack! + + + +============================================================== +============================================================== +HMOVE Timing Chart +-------------------------------------------------------------- +Brad Mott +============================================================== +============================================================== + +Typically HMOVE is executed right after WSYNC, but hitting HMOVE at other times +during the line has effects that can sometimes be useful. It's possible to move +objects without the black HMOVE bars and/or move objects farther than would +normally be possible with HMPx registers. This test was performed using player +graphics, but will probably work with the ball and missiles as well. + + + HMPx values + 0 1 2 3 4 5 6 7 8 9 a b c d e f +Cyc +10 0 -1 -2 -2 -2 -2 -2 -2 8 7 6 5 4 3 2 1 ** HBLANK +11 0 -1 -1 -1 -1 -1 -1 -1 8 7 6 5 4 3 2 1 HBLANK +12 0 0 0 0 0 0 0 0 8 7 6 5 4 3 2 1 HBLANK +13 1 1 1 1 1 1 1 1 8 7 6 5 4 3 2 1 HBLANK +14 1 1 1 1 1 1 1 1 8 7 6 5 4 3 2 1 ** HBLANK +15 2 2 2 2 2 2 2 2 8 7 6 5 4 3 2 2 HBLANK +16 3 3 3 3 3 3 3 3 8 7 6 5 4 3 3 3 HBLANK +17 4 4 4 4 4 4 4 4 8 7 6 5 4 4 4 4 HBLANK +18 4 4 4 4 4 4 4 4 8 7 6 5 4 4 4 4 ** HBLANK +19 5 5 5 5 5 5 5 5 8 7 6 5 5 5 5 5 HBLANK +20 6 6 6 6 6 6 6 6 8 7 6 6 6 6 6 6 HBLANK +21 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +22 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 + . + . +53 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +54 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +55 0 0 0 0 0 0 0 -1 0 0 0 0 0 0 0 0 +56 0 0 0 0 0 0 -1 -2 0 0 0 0 0 0 0 0 +57 0 0 0 0 0 -1 -2 -3 0 0 0 0 0 0 0 0 +58 0 0 0 0 0 -1 -2 -3 0 0 0 0 0 0 0 0 ** +59 0 0 0 0 -1 -2 -3 -4 0 0 0 0 0 0 0 0 +60 0 0 0 -1 -2 -3 -4 -5 0 0 0 0 0 0 0 0 +61 0 0 -1 -2 -3 -4 -5 -6 0 0 0 0 0 0 0 0 +62 0 0 -1 -2 -3 -4 -5 -6 0 0 0 0 0 0 0 0 ** +63 0 -1 -2 -3 -4 -5 -6 -7 0 0 0 0 0 0 0 0 +64 -1 -2 -3 -4 -5 -6 -7 -8 0 0 0 0 0 0 0 0 +65 -2 -3 -4 -5 -6 -7 -8 -9 0 0 0 0 0 0 0 -1 +66 -2 -3 -4 -5 -6 -7 -8 -9 0 0 0 0 0 0 0 -1 ** +67 -3 -4 -5 -6 -7 -8 -9 -10 0 0 0 0 0 0 -1 -2 +68 -4 -5 -6 -7 -8 -9 -10 -11 0 0 0 0 0 -1 -2 -3 +69 -5 -6 -7 -8 -9 -10 -11 -12 0 0 0 0 -1 -2 -3 -4 +70 -5 -6 -7 -8 -9 -10 -11 -12 0 0 0 0 -1 -2 -3 -4 ** +71 -6 -7 -8 -9 -10 -11 -12 -13 0 0 0 -1 -2 -3 -4 -5 +72 -7 -8 -9 -10 -11 -12 -13 -14 0 0 -1 -2 -3 -4 -5 -6 +73 -8 -9 -10 -11 -12 -13 -14 -15 0 -1 -2 -3 -4 -5 -6 -7 +74 -8 -9 -10 -11 -12 -13 -14 -15 0 -1 -2 -3 -4 -5 -6 -7 ** +75 0 -1 -2 -3 -4 -5 -6 -7 8 7 6 5 4 3 2 1 HBLANK +76 0 -1 -2 -3 -4 -5 -6 -7 8 7 6 5 4 3 2 1 HBLANK +77 0 -1 -2 -3 -4 -5 -6 -7 8 7 6 5 4 3 2 1 HBLANK +78 0 -1 -2 -3 -4 -5 -6 -7 8 7 6 5 4 3 2 1 HBLANK +79 0 -1 -2 -3 -4 -5 -6 -7 8 7 6 5 4 3 2 1 HBLANK +80 0 -1 -2 -3 -4 -5 -6 -6 8 7 6 5 4 3 2 1 HBLANK +81 0 -1 -2 -3 -4 -5 -5 -5 8 7 6 5 4 3 2 1 HBLANK +82 0 -1 -2 -3 -4 -5 -5 -5 8 7 6 5 4 3 2 1 ** HBLANK +83 0 -1 -2 -3 -4 -4 -4 -4 8 7 6 5 4 3 2 1 HBLANK +84 0 -1 -2 -3 -3 -3 -3 -3 8 7 6 5 4 3 2 1 HBLANK +85 0 -1 -2 -2 -2 -2 -2 -2 8 7 6 5 4 3 2 1 HBLANK +( table repeats at this point) \ No newline at end of file diff --git a/com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/2600_mem_map.txt b/com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/2600_mem_map.txt new file mode 100644 index 00000000..9a7635f5 --- /dev/null +++ b/com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/2600_mem_map.txt @@ -0,0 +1,632 @@ +*************************************************** +* $0000-$003F = TIA Addresses $00-$3F (zero page) * +* ----------------------------------------------- * +* * +* mirror: $xyz0 * +* * +* x = {even} * +* y = {anything} * +* z = {0, 4} * +* * +*************************************************** + +************************************** +* $0080-$00FF = RIOT RAM (zero page) * +* ---------------------------------- * +* * +* mirror: $xy80 * +* * +* x = {even} * +* y = {0,1,4,5,8,9,$C,$D} * +* * +************************************** + +**************************************** +* $0280-$029F = RIOT Addresses $00-$1F * +* ------------------------------------ * +* * +* mirror: $xyz0 * +* * +* x = {even} * +* y = {2,3,6,7,$A,$B,$E,$F} * +* z = {8,$A,$C,$E} * +* * +**************************************** + +***************************************** +* $1000-$1FFF = ROM Addresses $000-$FFF * +* ------------------------------------- * +* * +* mirror: $x000 * +* * +* x = {odd} * +* * +***************************************** + + +Mirrors: + +$0000-$003F = TIA Addresses $00-$3F (zero page) +$0040-$007F = TIA Addresses $00-$3F (mirror) +$0080-$00FF = RIOT RAM (zero page) +$0100-$013F = TIA Addresses $00-$3F (mirror) +$0140-$017F = TIA Addresses $00-$3F (mirror) +$0180-$01FF = RIOT RAM (mirror) +$0200-$023F = TIA Addresses $00-$3F (mirror) +$0240-$027F = TIA Addresses $00-$3F (mirror) +$0280-$029F = RIOT Addresses $00-$1F +$02A0-$02BF = RIOT Addresses $00-$1F (mirror) +$02C0-$02DF = RIOT Addresses $00-$1F (mirror) +$02E0-$02FF = RIOT Addresses $00-$1F (mirror) +$0300-$033F = TIA Addresses $00-$3F (mirror) +$0340-$037F = TIA Addresses $00-$3F (mirror) +$0380-$039F = RIOT Addresses $00-$1F (mirror) +$03A0-$03BF = RIOT Addresses $00-$1F (mirror) +$03C0-$03DF = RIOT Addresses $00-$1F (mirror) +$03E0-$03FF = RIOT Addresses $00-$1F (mirror) +$0400-$043F = TIA Addresses $00-$3F (mirror) +$0440-$047F = TIA Addresses $00-$3F (mirror) +$0480-$04FF = RIOT RAM (mirror) +$0500-$053F = TIA Addresses $00-$3F (mirror) +$0540-$057F = TIA Addresses $00-$3F (mirror) +$0580-$05FF = RIOT RAM (mirror) +$0600-$063F = TIA Addresses $00-$3F (mirror) +$0640-$067F = TIA Addresses $00-$3F (mirror) +$0680-$069F = RIOT Addresses $00-$1F (mirror) +$06A0-$06BF = RIOT Addresses $00-$1F (mirror) +$06C0-$06DF = RIOT Addresses $00-$1F (mirror) +$06E0-$06FF = RIOT Addresses $00-$1F (mirror) +$0700-$073F = TIA Addresses $00-$3F (mirror) +$0740-$077F = TIA Addresses $00-$3F (mirror) +$0780-$079F = RIOT Addresses $00-$1F (mirror) +$07A0-$07BF = RIOT Addresses $00-$1F (mirror) +$07C0-$07DF = RIOT Addresses $00-$1F (mirror) +$07E0-$07FF = RIOT Addresses $00-$1F (mirror) +$0800-$083F = TIA Addresses $00-$3F (mirror) +$0840-$087F = TIA Addresses $00-$3F (mirror) +$0880-$08FF = RIOT RAM (mirror) +$0900-$093F = TIA Addresses $00-$3F (mirror) +$0940-$097F = TIA Addresses $00-$3F (mirror) +$0980-$09FF = RIOT RAM (mirror) +$0A00-$0A3F = TIA Addresses $00-$3F (mirror) +$0A40-$0A7F = TIA Addresses $00-$3F (mirror) +$0A80-$0A9F = RIOT Addresses $00-$1F (mirror) +$0AA0-$0ABF = RIOT Addresses $00-$1F (mirror) +$0AC0-$0ADF = RIOT Addresses $00-$1F (mirror) +$0AE0-$0AFF = RIOT Addresses $00-$1F (mirror) +$0B00-$0B3F = TIA Addresses $00-$3F (mirror) +$0B40-$0B7F = TIA Addresses $00-$3F (mirror) +$0B80-$0B9F = RIOT Addresses $00-$1F (mirror) +$0BA0-$0BBF = RIOT Addresses $00-$1F (mirror) +$0BC0-$0BDF = RIOT Addresses $00-$1F (mirror) +$0BE0-$0BFF = RIOT Addresses $00-$1F (mirror) +$0C00-$0C3F = TIA Addresses $00-$3F (mirror) +$0C40-$0C7F = TIA Addresses $00-$3F (mirror) +$0C80-$0CFF = RIOT RAM (mirror) +$0D00-$0D3F = TIA Addresses $00-$3F (mirror) +$0D40-$0D7F = TIA Addresses $00-$3F (mirror) +$0D80-$0DFF = RIOT RAM (mirror) +$0E00-$0E3F = TIA Addresses $00-$3F (mirror) +$0E40-$0E7F = TIA Addresses $00-$3F (mirror) +$0E80-$0E9F = RIOT Addresses $00-$1F (mirror) +$0EA0-$0EBF = RIOT Addresses $00-$1F (mirror) +$0EC0-$0EDF = RIOT Addresses $00-$1F (mirror) +$0EE0-$0EFF = RIOT Addresses $00-$1F (mirror) +$0F00-$0F3F = TIA Addresses $00-$3F (mirror) +$0F40-$0F7F = TIA Addresses $00-$3F (mirror) +$0F80-$0F9F = RIOT Addresses $00-$1F (mirror) +$0FA0-$0FBF = RIOT Addresses $00-$1F (mirror) +$0FC0-$0FDF = RIOT Addresses $00-$1F (mirror) +$0FE0-$0FFF = RIOT Addresses $00-$1F (mirror) +$1000-$1FFF = ROM Addresses $000-$FFF +$2000-$203F = TIA Addresses $00-$3F (mirror) +$2040-$207F = TIA Addresses $00-$3F (mirror) +$2080-$20FF = RIOT RAM (mirror) +$2100-$213F = TIA Addresses $00-$3F (mirror) +$2140-$217F = TIA Addresses $00-$3F (mirror) +$2180-$21FF = RIOT RAM (mirror) +$2200-$223F = TIA Addresses $00-$3F (mirror) +$2240-$227F = TIA Addresses $00-$3F (mirror) +$2280-$229F = RIOT Addresses $00-$1F (mirror) +$22A0-$22BF = RIOT Addresses $00-$1F (mirror) +$22C0-$22DF = RIOT Addresses $00-$1F (mirror) +$22E0-$22FF = RIOT Addresses $00-$1F (mirror) +$2300-$233F = TIA Addresses $00-$3F (mirror) +$2340-$237F = TIA Addresses $00-$3F (mirror) +$2380-$239F = RIOT Addresses $00-$1F (mirror) +$23A0-$23BF = RIOT Addresses $00-$1F (mirror) +$23C0-$23DF = RIOT Addresses $00-$1F (mirror) +$23E0-$23FF = RIOT Addresses $00-$1F (mirror) +$2400-$243F = TIA Addresses $00-$3F (mirror) +$2440-$247F = TIA Addresses $00-$3F (mirror) +$2480-$24FF = RIOT RAM (mirror) +$2500-$253F = TIA Addresses $00-$3F (mirror) +$2540-$257F = TIA Addresses $00-$3F (mirror) +$2580-$25FF = RIOT RAM (mirror) +$2600-$263F = TIA Addresses $00-$3F (mirror) +$2640-$267F = TIA Addresses $00-$3F (mirror) +$2680-$269F = RIOT Addresses $00-$1F (mirror) +$26A0-$26BF = RIOT Addresses $00-$1F (mirror) +$26C0-$26DF = RIOT Addresses $00-$1F (mirror) +$26E0-$26FF = RIOT Addresses $00-$1F (mirror) +$2700-$273F = TIA Addresses $00-$3F (mirror) +$2740-$277F = TIA Addresses $00-$3F (mirror) +$2780-$279F = RIOT Addresses $00-$1F (mirror) +$27A0-$27BF = RIOT Addresses $00-$1F (mirror) +$27C0-$27DF = RIOT Addresses $00-$1F (mirror) +$27E0-$27FF = RIOT Addresses $00-$1F (mirror) +$2800-$283F = TIA Addresses $00-$3F (mirror) +$2840-$287F = TIA Addresses $00-$3F (mirror) +$2880-$28FF = RIOT RAM (mirror) +$2900-$293F = TIA Addresses $00-$3F (mirror) +$2940-$297F = TIA Addresses $00-$3F (mirror) +$2980-$29FF = RIOT RAM (mirror) +$2A00-$2A3F = TIA Addresses $00-$3F (mirror) +$2A40-$2A7F = TIA Addresses $00-$3F (mirror) +$2A80-$2A9F = RIOT Addresses $00-$1F (mirror) +$2AA0-$2ABF = RIOT Addresses $00-$1F (mirror) +$2AC0-$2ADF = RIOT Addresses $00-$1F (mirror) +$2AE0-$2AFF = RIOT Addresses $00-$1F (mirror) +$2B00-$2B3F = TIA Addresses $00-$3F (mirror) +$2B40-$2B7F = TIA Addresses $00-$3F (mirror) +$2B80-$2B9F = RIOT Addresses $00-$1F (mirror) +$2BA0-$2BBF = RIOT Addresses $00-$1F (mirror) +$2BC0-$2BDF = RIOT Addresses $00-$1F (mirror) +$2BE0-$2BFF = RIOT Addresses $00-$1F (mirror) +$2C00-$2C3F = TIA Addresses $00-$3F (mirror) +$2C40-$2C7F = TIA Addresses $00-$3F (mirror) +$2C80-$2CFF = RIOT RAM (mirror) +$2D00-$2D3F = TIA Addresses $00-$3F (mirror) +$2D40-$2D7F = TIA Addresses $00-$3F (mirror) +$2D80-$2DFF = RIOT RAM (mirror) +$2E00-$2E3F = TIA Addresses $00-$3F (mirror) +$2E40-$2E7F = TIA Addresses $00-$3F (mirror) +$2E80-$2E9F = RIOT Addresses $00-$1F (mirror) +$2EA0-$2EBF = RIOT Addresses $00-$1F (mirror) +$2EC0-$2EDF = RIOT Addresses $00-$1F (mirror) +$2EE0-$2EFF = RIOT Addresses $00-$1F (mirror) +$2F00-$2F3F = TIA Addresses $00-$3F (mirror) +$2F40-$2F7F = TIA Addresses $00-$3F (mirror) +$2F80-$2F9F = RIOT Addresses $00-$1F (mirror) +$2FA0-$2FBF = RIOT Addresses $00-$1F (mirror) +$2FC0-$2FDF = RIOT Addresses $00-$1F (mirror) +$2FE0-$2FFF = RIOT Addresses $00-$1F (mirror) +$3000-$3FFF = ROM Addresses $000-$FFF (mirror) +$4000-$403F = TIA Addresses $00-$3F (mirror) +$4040-$407F = TIA Addresses $00-$3F (mirror) +$4080-$40FF = RIOT RAM (mirror) +$4100-$413F = TIA Addresses $00-$3F (mirror) +$4140-$417F = TIA Addresses $00-$3F (mirror) +$4180-$41FF = RIOT RAM (mirror) +$4200-$423F = TIA Addresses $00-$3F (mirror) +$4240-$427F = TIA Addresses $00-$3F (mirror) +$4280-$429F = RIOT Addresses $00-$1F (mirror) +$42A0-$42BF = RIOT Addresses $00-$1F (mirror) +$42C0-$42DF = RIOT Addresses $00-$1F (mirror) +$42E0-$42FF = RIOT Addresses $00-$1F (mirror) +$4300-$433F = TIA Addresses $00-$3F (mirror) +$4340-$437F = TIA Addresses $00-$3F (mirror) +$4380-$439F = RIOT Addresses $00-$1F (mirror) +$43A0-$43BF = RIOT Addresses $00-$1F (mirror) +$43C0-$43DF = RIOT Addresses $00-$1F (mirror) +$43E0-$43FF = RIOT Addresses $00-$1F (mirror) +$4400-$443F = TIA Addresses $00-$3F (mirror) +$4440-$447F = TIA Addresses $00-$3F (mirror) +$4480-$44FF = RIOT RAM (mirror) +$4500-$453F = TIA Addresses $00-$3F (mirror) +$4540-$457F = TIA Addresses $00-$3F (mirror) +$4580-$45FF = RIOT RAM (mirror) +$4600-$463F = TIA Addresses $00-$3F (mirror) +$4640-$467F = TIA Addresses $00-$3F (mirror) +$4680-$469F = RIOT Addresses $00-$1F (mirror) +$46A0-$46BF = RIOT Addresses $00-$1F (mirror) +$46C0-$46DF = RIOT Addresses $00-$1F (mirror) +$46E0-$46FF = RIOT Addresses $00-$1F (mirror) +$4700-$473F = TIA Addresses $00-$3F (mirror) +$4740-$477F = TIA Addresses $00-$3F (mirror) +$4780-$479F = RIOT Addresses $00-$1F (mirror) +$47A0-$47BF = RIOT Addresses $00-$1F (mirror) +$47C0-$47DF = RIOT Addresses $00-$1F (mirror) +$47E0-$47FF = RIOT Addresses $00-$1F (mirror) +$4800-$483F = TIA Addresses $00-$3F (mirror) +$4840-$487F = TIA Addresses $00-$3F (mirror) +$4880-$48FF = RIOT RAM (mirror) +$4900-$493F = TIA Addresses $00-$3F (mirror) +$4940-$497F = TIA Addresses $00-$3F (mirror) +$4980-$49FF = RIOT RAM (mirror) +$4A00-$4A3F = TIA Addresses $00-$3F (mirror) +$4A40-$4A7F = TIA Addresses $00-$3F (mirror) +$4A80-$4A9F = RIOT Addresses $00-$1F (mirror) +$4AA0-$4ABF = RIOT Addresses $00-$1F (mirror) +$4AC0-$4ADF = RIOT Addresses $00-$1F (mirror) +$4AE0-$4AFF = RIOT Addresses $00-$1F (mirror) +$4B00-$4B3F = TIA Addresses $00-$3F (mirror) +$4B40-$4B7F = TIA Addresses $00-$3F (mirror) +$4B80-$4B9F = RIOT Addresses $00-$1F (mirror) +$4BA0-$4BBF = RIOT Addresses $00-$1F (mirror) +$4BC0-$4BDF = RIOT Addresses $00-$1F (mirror) +$4BE0-$4BFF = RIOT Addresses $00-$1F (mirror) +$4C00-$4C3F = TIA Addresses $00-$3F (mirror) +$4C40-$4C7F = TIA Addresses $00-$3F (mirror) +$4C80-$4CFF = RIOT RAM (mirror) +$4D00-$4D3F = TIA Addresses $00-$3F (mirror) +$4D40-$4D7F = TIA Addresses $00-$3F (mirror) +$4D80-$4DFF = RIOT RAM (mirror) +$4E00-$4E3F = TIA Addresses $00-$3F (mirror) +$4E40-$4E7F = TIA Addresses $00-$3F (mirror) +$4E80-$4E9F = RIOT Addresses $00-$1F (mirror) +$4EA0-$4EBF = RIOT Addresses $00-$1F (mirror) +$4EC0-$4EDF = RIOT Addresses $00-$1F (mirror) +$4EE0-$4EFF = RIOT Addresses $00-$1F (mirror) +$4F00-$4F3F = TIA Addresses $00-$3F (mirror) +$4F40-$4F7F = TIA Addresses $00-$3F (mirror) +$4F80-$4F9F = RIOT Addresses $00-$1F (mirror) +$4FA0-$4FBF = RIOT Addresses $00-$1F (mirror) +$4FC0-$4FDF = RIOT Addresses $00-$1F (mirror) +$4FE0-$4FFF = RIOT Addresses $00-$1F (mirror) +$5000-$5FFF = ROM Addresses $000-$FFF (mirror) +$6000-$603F = TIA Addresses $00-$3F (mirror) +$6040-$607F = TIA Addresses $00-$3F (mirror) +$6080-$60FF = RIOT RAM (mirror) +$6100-$613F = TIA Addresses $00-$3F (mirror) +$6140-$617F = TIA Addresses $00-$3F (mirror) +$6180-$61FF = RIOT RAM (mirror) +$6200-$623F = TIA Addresses $00-$3F (mirror) +$6240-$627F = TIA Addresses $00-$3F (mirror) +$6280-$629F = RIOT Addresses $00-$1F (mirror) +$62A0-$62BF = RIOT Addresses $00-$1F (mirror) +$62C0-$62DF = RIOT Addresses $00-$1F (mirror) +$62E0-$62FF = RIOT Addresses $00-$1F (mirror) +$6300-$633F = TIA Addresses $00-$3F (mirror) +$6340-$637F = TIA Addresses $00-$3F (mirror) +$6380-$639F = RIOT Addresses $00-$1F (mirror) +$63A0-$63BF = RIOT Addresses $00-$1F (mirror) +$63C0-$63DF = RIOT Addresses $00-$1F (mirror) +$63E0-$63FF = RIOT Addresses $00-$1F (mirror) +$6400-$643F = TIA Addresses $00-$3F (mirror) +$6440-$647F = TIA Addresses $00-$3F (mirror) +$6480-$64FF = RIOT RAM (mirror) +$6500-$653F = TIA Addresses $00-$3F (mirror) +$6540-$657F = TIA Addresses $00-$3F (mirror) +$6580-$65FF = RIOT RAM (mirror) +$6600-$663F = TIA Addresses $00-$3F (mirror) +$6640-$667F = TIA Addresses $00-$3F (mirror) +$6680-$669F = RIOT Addresses $00-$1F (mirror) +$66A0-$66BF = RIOT Addresses $00-$1F (mirror) +$66C0-$66DF = RIOT Addresses $00-$1F (mirror) +$66E0-$66FF = RIOT Addresses $00-$1F (mirror) +$6700-$673F = TIA Addresses $00-$3F (mirror) +$6740-$677F = TIA Addresses $00-$3F (mirror) +$6780-$679F = RIOT Addresses $00-$1F (mirror) +$67A0-$67BF = RIOT Addresses $00-$1F (mirror) +$67C0-$67DF = RIOT Addresses $00-$1F (mirror) +$67E0-$67FF = RIOT Addresses $00-$1F (mirror) +$6800-$683F = TIA Addresses $00-$3F (mirror) +$6840-$687F = TIA Addresses $00-$3F (mirror) +$6880-$68FF = RIOT RAM (mirror) +$6900-$693F = TIA Addresses $00-$3F (mirror) +$6940-$697F = TIA Addresses $00-$3F (mirror) +$6980-$69FF = RIOT RAM (mirror) +$6A00-$6A3F = TIA Addresses $00-$3F (mirror) +$6A40-$6A7F = TIA Addresses $00-$3F (mirror) +$6A80-$6A9F = RIOT Addresses $00-$1F (mirror) +$6AA0-$6ABF = RIOT Addresses $00-$1F (mirror) +$6AC0-$6ADF = RIOT Addresses $00-$1F (mirror) +$6AE0-$6AFF = RIOT Addresses $00-$1F (mirror) +$6B00-$6B3F = TIA Addresses $00-$3F (mirror) +$6B40-$6B7F = TIA Addresses $00-$3F (mirror) +$6B80-$6B9F = RIOT Addresses $00-$1F (mirror) +$6BA0-$6BBF = RIOT Addresses $00-$1F (mirror) +$6BC0-$6BDF = RIOT Addresses $00-$1F (mirror) +$6BE0-$6BFF = RIOT Addresses $00-$1F (mirror) +$6C00-$6C3F = TIA Addresses $00-$3F (mirror) +$6C40-$6C7F = TIA Addresses $00-$3F (mirror) +$6C80-$6CFF = RIOT RAM (mirror) +$6D00-$6D3F = TIA Addresses $00-$3F (mirror) +$6D40-$6D7F = TIA Addresses $00-$3F (mirror) +$6D80-$6DFF = RIOT RAM (mirror) +$6E00-$6E3F = TIA Addresses $00-$3F (mirror) +$6E40-$6E7F = TIA Addresses $00-$3F (mirror) +$6E80-$6E9F = RIOT Addresses $00-$1F (mirror) +$6EA0-$6EBF = RIOT Addresses $00-$1F (mirror) +$6EC0-$6EDF = RIOT Addresses $00-$1F (mirror) +$6EE0-$6EFF = RIOT Addresses $00-$1F (mirror) +$6F00-$6F3F = TIA Addresses $00-$3F (mirror) +$6F40-$6F7F = TIA Addresses $00-$3F (mirror) +$6F80-$6F9F = RIOT Addresses $00-$1F (mirror) +$6FA0-$6FBF = RIOT Addresses $00-$1F (mirror) +$6FC0-$6FDF = RIOT Addresses $00-$1F (mirror) +$6FE0-$6FFF = RIOT Addresses $00-$1F (mirror) +$7000-$7FFF = ROM Addresses $000-$FFF (mirror) +$8000-$803F = TIA Addresses $00-$3F (mirror) +$8040-$807F = TIA Addresses $00-$3F (mirror) +$8080-$80FF = RIOT RAM (mirror) +$8100-$813F = TIA Addresses $00-$3F (mirror) +$8140-$817F = TIA Addresses $00-$3F (mirror) +$8180-$81FF = RIOT RAM (mirror) +$8200-$823F = TIA Addresses $00-$3F (mirror) +$8240-$827F = TIA Addresses $00-$3F (mirror) +$8280-$829F = RIOT Addresses $00-$1F (mirror) +$82A0-$82BF = RIOT Addresses $00-$1F (mirror) +$82C0-$82DF = RIOT Addresses $00-$1F (mirror) +$82E0-$82FF = RIOT Addresses $00-$1F (mirror) +$8300-$833F = TIA Addresses $00-$3F (mirror) +$8340-$837F = TIA Addresses $00-$3F (mirror) +$8380-$839F = RIOT Addresses $00-$1F (mirror) +$83A0-$83BF = RIOT Addresses $00-$1F (mirror) +$83C0-$83DF = RIOT Addresses $00-$1F (mirror) +$83E0-$83FF = RIOT Addresses $00-$1F (mirror) +$8400-$843F = TIA Addresses $00-$3F (mirror) +$8440-$847F = TIA Addresses $00-$3F (mirror) +$8480-$84FF = RIOT RAM (mirror) +$8500-$853F = TIA Addresses $00-$3F (mirror) +$8540-$857F = TIA Addresses $00-$3F (mirror) +$8580-$85FF = RIOT RAM (mirror) +$8600-$863F = TIA Addresses $00-$3F (mirror) +$8640-$867F = TIA Addresses $00-$3F (mirror) +$8680-$869F = RIOT Addresses $00-$1F (mirror) +$86A0-$86BF = RIOT Addresses $00-$1F (mirror) +$86C0-$86DF = RIOT Addresses $00-$1F (mirror) +$86E0-$86FF = RIOT Addresses $00-$1F (mirror) +$8700-$873F = TIA Addresses $00-$3F (mirror) +$8740-$877F = TIA Addresses $00-$3F (mirror) +$8780-$879F = RIOT Addresses $00-$1F (mirror) +$87A0-$87BF = RIOT Addresses $00-$1F (mirror) +$87C0-$87DF = RIOT Addresses $00-$1F (mirror) +$87E0-$87FF = RIOT Addresses $00-$1F (mirror) +$8800-$883F = TIA Addresses $00-$3F (mirror) +$8840-$887F = TIA Addresses $00-$3F (mirror) +$8880-$88FF = RIOT RAM (mirror) +$8900-$893F = TIA Addresses $00-$3F (mirror) +$8940-$897F = TIA Addresses $00-$3F (mirror) +$8980-$89FF = RIOT RAM (mirror) +$8A00-$8A3F = TIA Addresses $00-$3F (mirror) +$8A40-$8A7F = TIA Addresses $00-$3F (mirror) +$8A80-$8A9F = RIOT Addresses $00-$1F (mirror) +$8AA0-$8ABF = RIOT Addresses $00-$1F (mirror) +$8AC0-$8ADF = RIOT Addresses $00-$1F (mirror) +$8AE0-$8AFF = RIOT Addresses $00-$1F (mirror) +$8B00-$8B3F = TIA Addresses $00-$3F (mirror) +$8B40-$8B7F = TIA Addresses $00-$3F (mirror) +$8B80-$8B9F = RIOT Addresses $00-$1F (mirror) +$8BA0-$8BBF = RIOT Addresses $00-$1F (mirror) +$8BC0-$8BDF = RIOT Addresses $00-$1F (mirror) +$8BE0-$8BFF = RIOT Addresses $00-$1F (mirror) +$8C00-$8C3F = TIA Addresses $00-$3F (mirror) +$8C40-$8C7F = TIA Addresses $00-$3F (mirror) +$8C80-$8CFF = RIOT RAM (mirror) +$8D00-$8D3F = TIA Addresses $00-$3F (mirror) +$8D40-$8D7F = TIA Addresses $00-$3F (mirror) +$8D80-$8DFF = RIOT RAM (mirror) +$8E00-$8E3F = TIA Addresses $00-$3F (mirror) +$8E40-$8E7F = TIA Addresses $00-$3F (mirror) +$8E80-$8E9F = RIOT Addresses $00-$1F (mirror) +$8EA0-$8EBF = RIOT Addresses $00-$1F (mirror) +$8EC0-$8EDF = RIOT Addresses $00-$1F (mirror) +$8EE0-$8EFF = RIOT Addresses $00-$1F (mirror) +$8F00-$8F3F = TIA Addresses $00-$3F (mirror) +$8F40-$8F7F = TIA Addresses $00-$3F (mirror) +$8F80-$8F9F = RIOT Addresses $00-$1F (mirror) +$8FA0-$8FBF = RIOT Addresses $00-$1F (mirror) +$8FC0-$8FDF = RIOT Addresses $00-$1F (mirror) +$8FE0-$8FFF = RIOT Addresses $00-$1F (mirror) +$9000-$9FFF = ROM Addresses $000-$FFF (mirror) +$A000-$A03F = TIA Addresses $00-$3F (mirror) +$A040-$A07F = TIA Addresses $00-$3F (mirror) +$A080-$A0FF = RIOT RAM (mirror) +$A100-$A13F = TIA Addresses $00-$3F (mirror) +$A140-$A17F = TIA Addresses $00-$3F (mirror) +$A180-$A1FF = RIOT RAM (mirror) +$A200-$A23F = TIA Addresses $00-$3F (mirror) +$A240-$A27F = TIA Addresses $00-$3F (mirror) +$A280-$A29F = RIOT Addresses $00-$1F (mirror) +$A2A0-$A2BF = RIOT Addresses $00-$1F (mirror) +$A2C0-$A2DF = RIOT Addresses $00-$1F (mirror) +$A2E0-$A2FF = RIOT Addresses $00-$1F (mirror) +$A300-$A33F = TIA Addresses $00-$3F (mirror) +$A340-$A37F = TIA Addresses $00-$3F (mirror) +$A380-$A39F = RIOT Addresses $00-$1F (mirror) +$A3A0-$A3BF = RIOT Addresses $00-$1F (mirror) +$A3C0-$A3DF = RIOT Addresses $00-$1F (mirror) +$A3E0-$A3FF = RIOT Addresses $00-$1F (mirror) +$A400-$A43F = TIA Addresses $00-$3F (mirror) +$A440-$A47F = TIA Addresses $00-$3F (mirror) +$A480-$A4FF = RIOT RAM (mirror) +$A500-$A53F = TIA Addresses $00-$3F (mirror) +$A540-$A57F = TIA Addresses $00-$3F (mirror) +$A580-$A5FF = RIOT RAM (mirror) +$A600-$A63F = TIA Addresses $00-$3F (mirror) +$A640-$A67F = TIA Addresses $00-$3F (mirror) +$A680-$A69F = RIOT Addresses $00-$1F (mirror) +$A6A0-$A6BF = RIOT Addresses $00-$1F (mirror) +$A6C0-$A6DF = RIOT Addresses $00-$1F (mirror) +$A6E0-$A6FF = RIOT Addresses $00-$1F (mirror) +$A700-$A73F = TIA Addresses $00-$3F (mirror) +$A740-$A77F = TIA Addresses $00-$3F (mirror) +$A780-$A79F = RIOT Addresses $00-$1F (mirror) +$A7A0-$A7BF = RIOT Addresses $00-$1F (mirror) +$A7C0-$A7DF = RIOT Addresses $00-$1F (mirror) +$A7E0-$A7FF = RIOT Addresses $00-$1F (mirror) +$A800-$A83F = TIA Addresses $00-$3F (mirror) +$A840-$A87F = TIA Addresses $00-$3F (mirror) +$A880-$A8FF = RIOT RAM (mirror) +$A900-$A93F = TIA Addresses $00-$3F (mirror) +$A940-$A97F = TIA Addresses $00-$3F (mirror) +$A980-$A9FF = RIOT RAM (mirror) +$AA00-$AA3F = TIA Addresses $00-$3F (mirror) +$AA40-$AA7F = TIA Addresses $00-$3F (mirror) +$AA80-$AA9F = RIOT Addresses $00-$1F (mirror) +$AAA0-$AABF = RIOT Addresses $00-$1F (mirror) +$AAC0-$AADF = RIOT Addresses $00-$1F (mirror) +$AAE0-$AAFF = RIOT Addresses $00-$1F (mirror) +$AB00-$AB3F = TIA Addresses $00-$3F (mirror) +$AB40-$AB7F = TIA Addresses $00-$3F (mirror) +$AB80-$AB9F = RIOT Addresses $00-$1F (mirror) +$ABA0-$ABBF = RIOT Addresses $00-$1F (mirror) +$ABC0-$ABDF = RIOT Addresses $00-$1F (mirror) +$ABE0-$ABFF = RIOT Addresses $00-$1F (mirror) +$AC00-$AC3F = TIA Addresses $00-$3F (mirror) +$AC40-$AC7F = TIA Addresses $00-$3F (mirror) +$AC80-$ACFF = RIOT RAM (mirror) +$AD00-$AD3F = TIA Addresses $00-$3F (mirror) +$AD40-$AD7F = TIA Addresses $00-$3F (mirror) +$AD80-$ADFF = RIOT RAM (mirror) +$AE00-$AE3F = TIA Addresses $00-$3F (mirror) +$AE40-$AE7F = TIA Addresses $00-$3F (mirror) +$AE80-$AE9F = RIOT Addresses $00-$1F (mirror) +$AEA0-$AEBF = RIOT Addresses $00-$1F (mirror) +$AEC0-$AEDF = RIOT Addresses $00-$1F (mirror) +$AEE0-$AEFF = RIOT Addresses $00-$1F (mirror) +$AF00-$AF3F = TIA Addresses $00-$3F (mirror) +$AF40-$AF7F = TIA Addresses $00-$3F (mirror) +$AF80-$AF9F = RIOT Addresses $00-$1F (mirror) +$AFA0-$AFBF = RIOT Addresses $00-$1F (mirror) +$AFC0-$AFDF = RIOT Addresses $00-$1F (mirror) +$AFE0-$AFFF = RIOT Addresses $00-$1F (mirror) +$B000-$BFFF = ROM Addresses $000-$FFF (mirror) +$C000-$C03F = TIA Addresses $00-$3F (mirror) +$C040-$C07F = TIA Addresses $00-$3F (mirror) +$C080-$C0FF = RIOT RAM (mirror) +$C100-$C13F = TIA Addresses $00-$3F (mirror) +$C140-$C17F = TIA Addresses $00-$3F (mirror) +$C180-$C1FF = RIOT RAM (mirror) +$C200-$C23F = TIA Addresses $00-$3F (mirror) +$C240-$C27F = TIA Addresses $00-$3F (mirror) +$C280-$C29F = RIOT Addresses $00-$1F (mirror) +$C2A0-$C2BF = RIOT Addresses $00-$1F (mirror) +$C2C0-$C2DF = RIOT Addresses $00-$1F (mirror) +$C2E0-$C2FF = RIOT Addresses $00-$1F (mirror) +$C300-$C33F = TIA Addresses $00-$3F (mirror) +$C340-$C37F = TIA Addresses $00-$3F (mirror) +$C380-$C39F = RIOT Addresses $00-$1F (mirror) +$C3A0-$C3BF = RIOT Addresses $00-$1F (mirror) +$C3C0-$C3DF = RIOT Addresses $00-$1F (mirror) +$C3E0-$C3FF = RIOT Addresses $00-$1F (mirror) +$C400-$C43F = TIA Addresses $00-$3F (mirror) +$C440-$C47F = TIA Addresses $00-$3F (mirror) +$C480-$C4FF = RIOT RAM (mirror) +$C500-$C53F = TIA Addresses $00-$3F (mirror) +$C540-$C57F = TIA Addresses $00-$3F (mirror) +$C580-$C5FF = RIOT RAM (mirror) +$C600-$C63F = TIA Addresses $00-$3F (mirror) +$C640-$C67F = TIA Addresses $00-$3F (mirror) +$C680-$C69F = RIOT Addresses $00-$1F (mirror) +$C6A0-$C6BF = RIOT Addresses $00-$1F (mirror) +$C6C0-$C6DF = RIOT Addresses $00-$1F (mirror) +$C6E0-$C6FF = RIOT Addresses $00-$1F (mirror) +$C700-$C73F = TIA Addresses $00-$3F (mirror) +$C740-$C77F = TIA Addresses $00-$3F (mirror) +$C780-$C79F = RIOT Addresses $00-$1F (mirror) +$C7A0-$C7BF = RIOT Addresses $00-$1F (mirror) +$C7C0-$C7DF = RIOT Addresses $00-$1F (mirror) +$C7E0-$C7FF = RIOT Addresses $00-$1F (mirror) +$C800-$C83F = TIA Addresses $00-$3F (mirror) +$C840-$C87F = TIA Addresses $00-$3F (mirror) +$C880-$C8FF = RIOT RAM (mirror) +$C900-$C93F = TIA Addresses $00-$3F (mirror) +$C940-$C97F = TIA Addresses $00-$3F (mirror) +$C980-$C9FF = RIOT RAM (mirror) +$CA00-$CA3F = TIA Addresses $00-$3F (mirror) +$CA40-$CA7F = TIA Addresses $00-$3F (mirror) +$CA80-$CA9F = RIOT Addresses $00-$1F (mirror) +$CAA0-$CABF = RIOT Addresses $00-$1F (mirror) +$CAC0-$CADF = RIOT Addresses $00-$1F (mirror) +$CAE0-$CAFF = RIOT Addresses $00-$1F (mirror) +$CB00-$CB3F = TIA Addresses $00-$3F (mirror) +$CB40-$CB7F = TIA Addresses $00-$3F (mirror) +$CB80-$CB9F = RIOT Addresses $00-$1F (mirror) +$CBA0-$CBBF = RIOT Addresses $00-$1F (mirror) +$CBC0-$CBDF = RIOT Addresses $00-$1F (mirror) +$CBE0-$CBFF = RIOT Addresses $00-$1F (mirror) +$CC00-$CC3F = TIA Addresses $00-$3F (mirror) +$CC40-$CC7F = TIA Addresses $00-$3F (mirror) +$CC80-$CCFF = RIOT RAM (mirror) +$CD00-$CD3F = TIA Addresses $00-$3F (mirror) +$CD40-$CD7F = TIA Addresses $00-$3F (mirror) +$CD80-$CDFF = RIOT RAM (mirror) +$CE00-$CE3F = TIA Addresses $00-$3F (mirror) +$CE40-$CE7F = TIA Addresses $00-$3F (mirror) +$CE80-$CE9F = RIOT Addresses $00-$1F (mirror) +$CEA0-$CEBF = RIOT Addresses $00-$1F (mirror) +$CEC0-$CEDF = RIOT Addresses $00-$1F (mirror) +$CEE0-$CEFF = RIOT Addresses $00-$1F (mirror) +$CF00-$CF3F = TIA Addresses $00-$3F (mirror) +$CF40-$CF7F = TIA Addresses $00-$3F (mirror) +$CF80-$CF9F = RIOT Addresses $00-$1F (mirror) +$CFA0-$CFBF = RIOT Addresses $00-$1F (mirror) +$CFC0-$CFDF = RIOT Addresses $00-$1F (mirror) +$CFE0-$CFFF = RIOT Addresses $00-$1F (mirror) +$D000-$DFFF = ROM Addresses $000-$FFF (mirror) +$E000-$E03F = TIA Addresses $00-$3F (mirror) +$E040-$E07F = TIA Addresses $00-$3F (mirror) +$E080-$E0FF = RIOT RAM (mirror) +$E100-$E13F = TIA Addresses $00-$3F (mirror) +$E140-$E17F = TIA Addresses $00-$3F (mirror) +$E180-$E1FF = RIOT RAM (mirror) +$E200-$E23F = TIA Addresses $00-$3F (mirror) +$E240-$E27F = TIA Addresses $00-$3F (mirror) +$E280-$E29F = RIOT Addresses $00-$1F (mirror) +$E2A0-$E2BF = RIOT Addresses $00-$1F (mirror) +$E2C0-$E2DF = RIOT Addresses $00-$1F (mirror) +$E2E0-$E2FF = RIOT Addresses $00-$1F (mirror) +$E300-$E33F = TIA Addresses $00-$3F (mirror) +$E340-$E37F = TIA Addresses $00-$3F (mirror) +$E380-$E39F = RIOT Addresses $00-$1F (mirror) +$E3A0-$E3BF = RIOT Addresses $00-$1F (mirror) +$E3C0-$E3DF = RIOT Addresses $00-$1F (mirror) +$E3E0-$E3FF = RIOT Addresses $00-$1F (mirror) +$E400-$E43F = TIA Addresses $00-$3F (mirror) +$E440-$E47F = TIA Addresses $00-$3F (mirror) +$E480-$E4FF = RIOT RAM (mirror) +$E500-$E53F = TIA Addresses $00-$3F (mirror) +$E540-$E57F = TIA Addresses $00-$3F (mirror) +$E580-$E5FF = RIOT RAM (mirror) +$E600-$E63F = TIA Addresses $00-$3F (mirror) +$E640-$E67F = TIA Addresses $00-$3F (mirror) +$E680-$E69F = RIOT Addresses $00-$1F (mirror) +$E6A0-$E6BF = RIOT Addresses $00-$1F (mirror) +$E6C0-$E6DF = RIOT Addresses $00-$1F (mirror) +$E6E0-$E6FF = RIOT Addresses $00-$1F (mirror) +$E700-$E73F = TIA Addresses $00-$3F (mirror) +$E740-$E77F = TIA Addresses $00-$3F (mirror) +$E780-$E79F = RIOT Addresses $00-$1F (mirror) +$E7A0-$E7BF = RIOT Addresses $00-$1F (mirror) +$E7C0-$E7DF = RIOT Addresses $00-$1F (mirror) +$E7E0-$E7FF = RIOT Addresses $00-$1F (mirror) +$E800-$E83F = TIA Addresses $00-$3F (mirror) +$E840-$E87F = TIA Addresses $00-$3F (mirror) +$E880-$E8FF = RIOT RAM (mirror) +$E900-$E93F = TIA Addresses $00-$3F (mirror) +$E940-$E97F = TIA Addresses $00-$3F (mirror) +$E980-$E9FF = RIOT RAM (mirror) +$EA00-$EA3F = TIA Addresses $00-$3F (mirror) +$EA40-$EA7F = TIA Addresses $00-$3F (mirror) +$EA80-$EA9F = RIOT Addresses $00-$1F (mirror) +$EAA0-$EABF = RIOT Addresses $00-$1F (mirror) +$EAC0-$EADF = RIOT Addresses $00-$1F (mirror) +$EAE0-$EAFF = RIOT Addresses $00-$1F (mirror) +$EB00-$EB3F = TIA Addresses $00-$3F (mirror) +$EB40-$EB7F = TIA Addresses $00-$3F (mirror) +$EB80-$EB9F = RIOT Addresses $00-$1F (mirror) +$EBA0-$EBBF = RIOT Addresses $00-$1F (mirror) +$EBC0-$EBDF = RIOT Addresses $00-$1F (mirror) +$EBE0-$EBFF = RIOT Addresses $00-$1F (mirror) +$EC00-$EC3F = TIA Addresses $00-$3F (mirror) +$EC40-$EC7F = TIA Addresses $00-$3F (mirror) +$EC80-$ECFF = RIOT RAM (mirror) +$ED00-$ED3F = TIA Addresses $00-$3F (mirror) +$ED40-$ED7F = TIA Addresses $00-$3F (mirror) +$ED80-$EDFF = RIOT RAM (mirror) +$EE00-$EE3F = TIA Addresses $00-$3F (mirror) +$EE40-$EE7F = TIA Addresses $00-$3F (mirror) +$EE80-$EE9F = RIOT Addresses $00-$1F (mirror) +$EEA0-$EEBF = RIOT Addresses $00-$1F (mirror) +$EEC0-$EEDF = RIOT Addresses $00-$1F (mirror) +$EEE0-$EEFF = RIOT Addresses $00-$1F (mirror) +$EF00-$EF3F = TIA Addresses $00-$3F (mirror) +$EF40-$EF7F = TIA Addresses $00-$3F (mirror) +$EF80-$EF9F = RIOT Addresses $00-$1F (mirror) +$EFA0-$EFBF = RIOT Addresses $00-$1F (mirror) +$EFC0-$EFDF = RIOT Addresses $00-$1F (mirror) +$EFE0-$EFFF = RIOT Addresses $00-$1F (mirror) +$F000-$FFFF = ROM Addresses $000-$FFF (mirror) diff --git a/com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/2600_riot_map.txt b/com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/2600_riot_map.txt new file mode 100644 index 00000000..6abfc0aa --- /dev/null +++ b/com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/2600_riot_map.txt @@ -0,0 +1,140 @@ +-------------------------------- +RIOT Addresses: names taken from +Atari VCS Stella Documentation +-------------------------------- +$0280 = (RIOT $00) - SWCHA (read/write) +$0281 = (RIOT $01) - SWACNT (read/write) +$0282 = (RIOT $02) - SWCHB (read/write) (*) +$0283 = (RIOT $03) - SWBCNT (read/write) (*) +$0284 = (RIOT $04) - INTIM (read), edge detect control (write) +$0285 = (RIOT $05) - read interrupt flag (read), edge detect control (write) +$0286 = (RIOT $06) - INTIM (read), edge detect control (write) +$0287 = (RIOT $07) - read interrupt flag (read), edge detect control (write) +$0288 = (RIOT $08) - SWCHA (read/write) +$0289 = (RIOT $09) - SWACNT (read/write) +$028A = (RIOT $0A) - SWCHB (read/write) (*) +$028B = (RIOT $0B) - SWBCNT (read/write) (*) +$028C = (RIOT $0C) - INTIM (read) , edge detect control (write) +$028D = (RIOT $0D) - read interrupt flag (read), edge detect control (write) +$028E = (RIOT $0E) - INTIM (read) , edge detect control (write) +$028F = (RIOT $0F) - read interrupt flag (read), edge detect control (write) +$0290 = (RIOT $10) - SWCHA (read/write) +$0291 = (RIOT $11) - SWACNT (read/write) +$0292 = (RIOT $12) - SWCHB (read/write) (*) +$0293 = (RIOT $13) - SWBCNT (read/write) (*) +$0294 = (RIOT $14) - INTIM (read), TIM1T (write) +$0295 = (RIOT $15) - read interrupt flag (read), TIM8T (write) +$0296 = (RIOT $16) - INTIM (read), TIM64T (write) +$0297 = (RIOT $17) - read interrupt flag (read), TIM1024T (write) +$0298 = (RIOT $18) - SWCHA (read/write) +$0299 = (RIOT $19) - SWACNT (read/write) +$029A = (RIOT $1A) - SWCHB (read/write) (*) +$029B = (RIOT $1B) - SWBCNT (read/write) (*) +$029C = (RIOT $1C) - INTIM (read), TIM1T (write) +$029D = (RIOT $1D) - read interrupt flag (read), TIM8T (write) +$029E = (RIOT $1E) - INTIM (read), TIM64T (write) +$029F = (RIOT $1F) - read interrupt flag (read), TIM1024T (write) + +(*) The Stella documentation states that Port B is hardwired for input + only. In reality, Port B is fully configurable. Configuring Port B + as an output, however, will cause direct hardware conflicts with the + console switches. DO NOT ATTEMPT THIS. + +------------------------------------------------------------------------------ + +********************** +* RIOT Documentation * +********************** + +--------------------------------------- +RIOT Addressing Notes for the Atari VCS +--------------------------------------- +A12 is connected to /CS2 (Chip Select 2 - active low) +A7 is connected to CS1 (Chip Select 1 - active high) +A9 is connected to /RS (RAM Select - active low) +A11, A10, and A8 are not connected to the RIOT + +------------------------------------- +$0080 - $00FF = RIOT RAM (read/write) +------------------------------------- +$0080 - $00FF = RIOT RAM, Addresses $00-$7F + +--------------------------------------------- +$0280 - $029F = RIOT Read Addresses (not RAM) +--------------------------------------------- +$0280 = (RIOT $00) - Read DRA +$0281 = (RIOT $01) - Read DDRA +$0282 = (RIOT $02) - Read DRB +$0283 = (RIOT $03) - Read DDRB +$0284 = (RIOT $04) - Read timer, disable interrupt (2) +$0285 = (RIOT $05) - Read interrupt flag +$0286 = (RIOT $06) - Read timer, disable interrupt (2) +$0287 = (RIOT $07) - Read interrupt flag +$0288 = (RIOT $08) - Read DRA +$0289 = (RIOT $09) - Read DDRA +$028A = (RIOT $0A) - Read DRB +$028B = (RIOT $0B) - Read DDRB +$028C = (RIOT $0C) - Read timer, enable interrupt (2) +$028D = (RIOT $0D) - Read interrupt flag +$028E = (RIOT $0E) - Read timer, enable interrupt (2) +$028F = (RIOT $0F) - Read interrupt flag +$0290 = (RIOT $10) - Read DRA +$0291 = (RIOT $11) - Read DDRA +$0292 = (RIOT $12) - Read DRB +$0293 = (RIOT $13) - Read DDRB +$0294 = (RIOT $14) - Read timer, disable interrupt (2) +$0295 = (RIOT $15) - Read interrupt flag +$0296 = (RIOT $16) - Read timer, disable interrupt (2) +$0297 = (RIOT $17) - Read interrupt flag +$0298 = (RIOT $18) - Read DRA +$0299 = (RIOT $19) - Read DDRA +$029A = (RIOT $1A) - Read DRB +$029B = (RIOT $1B) - Read DDRB +$029C = (RIOT $1C) - Read timer, enable interrupt (2) +$029D = (RIOT $1D) - Read interrupt flag +$029E = (RIOT $1E) - Read timer, enable interrupt (2) +$029F = (RIOT $1F) - Read interrupt flag + +---------------------------------------------- +$0280 - $029F = RIOT Write Addresses (not RAM) +---------------------------------------------- +$0280 = (RIOT $00) - Write DRA +$0281 = (RIOT $01) - Write DDRA +$0282 = (RIOT $02) - Write DRB +$0283 = (RIOT $03) - Write DDRB +$0284 = (RIOT $04) - Write edge detect control - negative edge, disable int (1) +$0285 = (RIOT $05) - Write edge detect control - positive edge, disable int (1) +$0286 = (RIOT $06) - Write edge detect control - negative edge, enable int (1) +$0287 = (RIOT $07) - Write edge detect control - positive edge, enable int (1) +$0288 = (RIOT $08) - Write DRA +$0289 = (RIOT $09) - Write DDRA +$028A = (RIOT $0A) - Write DRB +$028B = (RIOT $0B) - Write DDRB +$028C = (RIOT $0C) - Write edge detect control - negative edge, disable int (1) +$028D = (RIOT $0D) - Write edge detect control - positive edge, disable int (1) +$028E = (RIOT $0E) - Write edge detect control - negative edge, enable int (1) +$028F = (RIOT $0F) - Write edge detect control - positive edge, enable int (1) +$0290 = (RIOT $10) - Write DRA +$0291 = (RIOT $11) - Write DDRA +$0292 = (RIOT $12) - Write DRB +$0293 = (RIOT $13) - Write DDRB +$0294 = (RIOT $14) - Write timer (div by 1) - disable int (2) +$0295 = (RIOT $15) - Write timer (div by 8) - disable int (2) +$0296 = (RIOT $16) - Write timer (div by 64) - disable int (2) +$0297 = (RIOT $17) - Write timer (div by 1024) - disable int (2) +$0298 = (RIOT $18) - Write DRA +$0299 = (RIOT $19) - Write DDRA +$029A = (RIOT $1A) - Write DRB +$029B = (RIOT $1B) - Write DDRB +$029C = (RIOT $1C) - Write timer (div by 1) - enable int (2) +$029D = (RIOT $1D) - Write timer (div by 8) - enable int (2) +$029E = (RIOT $1E) - Write timer (div by 64) - enable int (2) +$029F = (RIOT $1F) - Write timer (div by 1024) - enable int (2) + +(1) A0 = 0 for negative edge detect + A0 = 1 for positive edge detect + A1 = 0 to disable interrupt from PA7 to /IRQ + A1 = 1 to enable interrupt from PA7 to /IRQ + +(2) A3 = 0 to disable interrupt from timer to /IRQ + A3 = 1 to enable interrupt from timer to /IRQ diff --git a/com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/2600_tia_map.txt b/com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/2600_tia_map.txt new file mode 100644 index 00000000..6305383d --- /dev/null +++ b/com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/2600_tia_map.txt @@ -0,0 +1,81 @@ +********************* +* TIA Documentation * +********************* + +-------------------------------------- +TIA Addressing Notes for the Atari VCS +-------------------------------------- +A12 is connected to /CS0 (Chip Select 0 - active low) +A7 is connected to /CS3 (Chip Select 3 - active low) +A[11:8] and A6 are not connected to the TIA + +VCC is connected to CS1 (Chip Select 1 - active high) +GND is connected to /CS2 (Chip Select 2 - active low) + +-------------------------------------------- +$0000 - $003F = TIA.......(write)......(read) +-------------------------------------------- +$0000 = TIA Address $00 - (VSYNC)......(CXM0P) +$0001 = TIA Address $01 - (VBLANK).....(CXM1P) +$0002 = TIA Address $02 - (WSYNC)......(CXP0FB) +$0003 = TIA Address $03 - (RSYNC)......(CXP1FB) +$0004 = TIA Address $04 - (NUSIZ0).....(CXM0FB) +$0005 = TIA Address $05 - (NUSIZ1).....(CXM1FB) +$0006 = TIA Address $06 - (COLUP0).....(CXBLPF) +$0007 = TIA Address $07 - (COLUP1).....(CXPPMM) +$0008 = TIA Address $08 - (COLUPF).....(INPT0) +$0009 = TIA Address $09 - (COLUBK).....(INPT1) +$000A = TIA Address $0A - (CTRLPF).....(INPT2) +$000B = TIA Address $0B - (REFP0)......(INPT3) +$000C = TIA Address $0C - (REFP1)......(INPT4) +$000D = TIA Address $0D - (PF0)........(INPT5) +$000E = TIA Address $0E - (PF1)........(UNDEFINED) +$000F = TIA Address $0F - (PF2)........(UNDEFINED) +$0010 = TIA Address $10 - (RESP0)......(CXM0P) +$0011 = TIA Address $11 - (RESP1)......(CXM1P) +$0012 = TIA Address $12 - (RESM0)......(CXP0FB) +$0013 = TIA Address $13 - (RESM1)......(CXP1FB) +$0014 = TIA Address $14 - (RESBL)......(CXM0FB) +$0015 = TIA Address $15 - (AUDC0)......(CXM1FB) +$0016 = TIA Address $16 - (AUDC1)......(CXBLPF) +$0017 = TIA Address $17 - (AUDF0)......(CXPPMM) +$0018 = TIA Address $18 - (AUDF1)......(INPT0) +$0019 = TIA Address $19 - (AUDV0)......(INPT1) +$001A = TIA Address $1A - (AUDV1)......(INPT2) +$001B = TIA Address $1B - (GRP0).......(INPT3) +$001C = TIA Address $1C - (GRP1).......(INPT4) +$001D = TIA Address $1D - (ENAM0)......(INPT5) +$001E = TIA Address $1E - (ENAM1)......(UNDEFINED) +$001F = TIA Address $1F - (ENABL)......(UNDEFINED) +$0020 = TIA Address $20 - (HMP0).......(CXM0P) +$0021 = TIA Address $21 - (HMP1).......(CXM1P) +$0022 = TIA Address $22 - (HMM0).......(CXP0FB) +$0023 = TIA Address $23 - (HMM1).......(CXP1FB) +$0024 = TIA Address $24 - (HMBL).......(CXM0FB) +$0025 = TIA Address $25 - (VDELP0).....(CXM1FB) +$0026 = TIA Address $26 - (VDELP1).....(CXBLPF) +$0027 = TIA Address $27 - (VDELBL).....(CXPPMM) +$0028 = TIA Address $28 - (RESMP0).....(INPT0) +$0029 = TIA Address $29 - (RESMP1).....(INPT1) +$002A = TIA Address $2A - (HMOVE)......(INPT2) +$002B = TIA Address $2B - (HMCLR)......(INPT3) +$002C = TIA Address $2C - (CXCLR)......(INPT4) +$002D = TIA Address $2D - (UNDEFINED)..(INPT5) +$002E = TIA Address $2E - (UNDEFINED)..(UNDEFINED) +$002F = TIA Address $2F - (UNDEFINED)..(UNDEFINED) +$0030 = TIA Address $30 - (UNDEFINED)..(CXM0P) +$0031 = TIA Address $31 - (UNDEFINED)..(CXM1P) +$0032 = TIA Address $32 - (UNDEFINED)..(CXP0FB) +$0033 = TIA Address $33 - (UNDEFINED)..(CXP1FB) +$0034 = TIA Address $34 - (UNDEFINED)..(CXM0FB) +$0035 = TIA Address $35 - (UNDEFINED)..(CXM1FB) +$0036 = TIA Address $36 - (UNDEFINED)..(CXBLPF) +$0037 = TIA Address $37 - (UNDEFINED)..(CXPPMM) +$0038 = TIA Address $38 - (UNDEFINED)..(INPT0) +$0039 = TIA Address $39 - (UNDEFINED)..(INPT1) +$003A = TIA Address $3A - (UNDEFINED)..(INPT2) +$003B = TIA Address $3B - (UNDEFINED)..(INPT3) +$003C = TIA Address $3C - (UNDEFINED)..(INPT4) +$003D = TIA Address $3D - (UNDEFINED)..(INPT5) +$003E = TIA Address $3E - (UNDEFINED)..(UNDEFINED) +$003F = TIA Address $3F - (UNDEFINED)..(UNDEFINED) diff --git a/com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/Stella Programmer's Guide (Unofficial HTML version).html b/com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/Stella Programmer's Guide (Unofficial HTML version).html new file mode 100644 index 00000000..d07bfafd --- /dev/null +++ b/com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/Stella Programmer's Guide (Unofficial HTML version).html @@ -0,0 +1,2375 @@ + + + + + Stella Programmer's Guide (Unofficial HTML version) + + + +
+

+STELLA
+PROGRAMMER'S
+GUIDE
+

+by Steve Wright 12/03/79

+Reconstructed by Charles Sinnett 6/11/93
+ Email: cas@mentor.cc.purdue.edu

+HTMLified by B. Watson 9/14/2001
+Email: atari@hardcoders.org
+(Editor's notes --BW) +

+ +

+ +

TABLE OF CONTENTS

+ + +
+ + + +

TELEVISION PROTOCOL

+ (The TV picture according to Atari)
+

+ +For the purposes of Stella programming, a single television "frame" consists +of 262 horizontal lines, and each line is divided by 228 clock counts +(3.58MHz). The actual TV picture is drawn line by line from the top down +60 times a second, and actaully consists of only a portion of the entire +"frame" (see diag. #1). A typical frame will consists of 3 vertical sync +(VSYNC) lines*, 37 vertical blank (VBLANK) lines, 192 TV picture lines, +and 30 overscan lines. Atari's research has shown that this pattern will +work on all types of TV sets. Each scan lines starts with 68 clock counts of +horizontal blank (not seen on the TV screen) followed by 160 clock counts to +fully scan one line of TV picture. When the electron beam reaches the end +of a scan line, it returns to the left side of the screen, waits for the 68 +horizontal blank clock counts, and proceeds to draw the next line below. +

+All horizontal timing is taken care of by hardware, but the microprocessor +must "manually" control vertical timing to signal the start of the next +frame. When the last line of the previous frame is detected, the +microprocessor must generate 3 lines of VSYNC, 37 lines of VBLANK, 192 +lines of actual TV picture, and 30 lines of overscan. Fortunately, both +VSYNC and VBLANK can simply be turned on and off at the appropriate +times, freeing the microprocessor for other activities during their execution. + +

+* (to signal the TV set to start a new frame) + +

+The actual TV picture is drawn one line at a time by having the +microprocessor enter the data for that line into the Television Interface +Adaptor (TIA) chip, which then converts the data into video signals. The +TIA can only have data in it that pertains to the line being currently drawn, +so the microprocessor must be "one step ahead" of the electron beam on each +line. Since one microprocessor machine cycle occurs every 3 clock counts, +the programmer has only 76 machine cycles per line (228/3 = 76) to +construct the actual picture (actually less because the microprocessor must +be ahead of the raster). To allow more time for the software, it is customary +(but not required) to update the TIA every two scan lines. The portion of +the program that constructs this TV picture is referred to as the "Kernel", as +it is the essence or kernel of the game. + +

+In general, the remaining 70 scan lines (3 for VSYNC, 37 for VBLANK, and +30 for overscan) will provides 5,320 machine cycles (70 lines x 76 machine +cycles) for housekeeping and game logic. Such activities as calculating the +new position of a player, updating the score, and checking for new inputs +are typically done during this time. + +

+ +Diagram 1 - Atari TV Frame +

+
+ + +

The TIA (as seen by the programmer)

+ + +1.0 General Description

+

+The TIA is a custom IC designed to create the TV picture and sound from +the instructions sent to it by the microprocessor. It converts the 8 bit +parallel data from the microprocessor into signals that are sent to video +modulation circuits which combine and shape those signals to be compatible +with ordinary TV reception. A "playfield" and 5 moveable objects can be +created and manipulated by software. +

+A playfield consisting of walls, clouds, barriers, and other seldom moved +objects can be created over a colored background. The 5 moveable objects +can be positioned anywhere, and consists of 2 players, 2 missiles, and a ball. +The playfield, players, missiles, and ball are created and manipulated by a +series of registers in the TIA that the microprocessor can address and write +into. Each type of object has certain defined capabilities. For example, a +player can be moved with one instruction, but the playfield must be +completely re-drawn in order to make it "move". +

+Color and luminosity (brightness) can be assigned to the background, +playfield, and 5 moveable objects. Sound can also be generated and +controlled for volume, pitch, and type of sound. Collisions between the +various objects on the TV screen are detected by the TIA and can be read by +the microprocessor . Input ports which can be read by the microprocessor +give the status of some of the various hand held controllers. +

+

+ +2.0 The Registers +

+

+All instructions to the TIA are achieved by addressing and writing to +various registers in the chip. A key point to remember is data written in a +register is latched an retained until altered by another write operation into +that register. For example, if the color register for a player is set for red, +that player will be red every time it is drawn until that color register is +changed. All of the registers are addressed by the microprocessor as part of +the overall RAM/ROM memory space. +

+All registers have fixed address locations and pre-assigned address names for handy +reference. Many registers do not use all 8 data bits, and some registers are used to +"strobe" or trigger events. A "strobe" register executes its function the instant it is +written to (the data written is ignored). The only registers the microprocessor can read +are the collision registers and input port registers. These registers are conveniently +arranged so that the data bits of interest always appear as data bits 6 or 7 for easy access. +

+ +

+ + +3.0 Synchronization

+

+

+ +3.1 Horizontal Timing +
+ When the electron beam scans across the TV screen and reaches the right + edge, it must be turned off and moved back to the left edge of the screen to + begin the next scan line. The TIA takes care of this automatically, + independent of the microprocessor. A 3.58 MHz oscillator generates clock + pulses called "color clocks" which go into a pulse counter in the TIA. This + counter allows 160 color clocks for the beam to reach the right edge, then + generates a horizontal sync signal (HSYNC) to return the beam to the left + edge. It also generates the signal to turn the beam off (horizontal blanking) + during its return time of 68 color clocks. Total round trip for the electron + beam is 160 + 68 = 228 color clocks. Again, all the horizontal timing is + taken care of by the TIA without assistance from the microprocessor. +

+

+ +3.2 Microprocessor Synchronization +
+ The microprocessor's clock is the 3.58 MHz oscillator divided by 3, so one + machine cycle is 3 color clocks. Therefore, one complete scan line of 228 + color clocks allows only 76 machine cycles (228/3 = 76) per scan line. The + microprocessor must be synchronized with the TIA on a line-by-line basis, + but program loops and branches take unpredictable lengths of time. To + solve this software sync. problem, the programmer can use the WSYNC + (Wait for SYNC) strobe register. Simply writing to the WSYNC causes the + microprocessor to halt until the electron beam reaches the right edge of the + screen, then the microprocessor resumes operation at the beginning of the + 68 color clocks for horizontal blanking. Since the TIA latches all + instructions until altered by another write operation, it could be updated + every 2 or 3 lines. The advantage is the programmer gains more time to + execute software, but at a price paid with lower vertical resolution in the + graphics. +

+ NOTE: WSYNC and all the following addresses' bit structures are itemized + in the TIA hardware manual. The purpose of this document is to make + them understandable. +

+

+ +3.3 Vertical timing +
+ When the electron beam has scanned 262 lines, the TV set must be signaled + to blank the beam and position it at the top of the screen to start a new + frame. This signal is called vertical sync, and the TIA must transmit this + signal for at least 3 scan lines. This is accomplished by writing a "1" in D1 + of VSYNC to turn it on, count at least 2 scan lines, then write a "0" to D1 of + VSYNC to turn it off. +

+ To physically turn the beam off during its repositioning time, the TV set + needs 37 scan lines of vertical blanks signal from the TIA. This is + accomplished by writing a "1" in D1 of VBLANK to turn it on, count 37 + lines, then write a "0" to D1 of VBLANK to turn it off. The microprocessor + is of course free to execute other software during the vertical timing + commands, VSYNC and VBLANK. +

+

+
+ +4.0 Color and Luminosity +

+

+ Color and luminosity can be assigned to the background (BK), playfield + (PF), ball (BL), player 0 (P0), player 1(P1), missile 0 (M0), and missile 1 + (M1). There are only four color-lum registers for these 7 objects, so the + objects are paired to share the same register according to the following list: +

+

+ + + + + + + +
color-lum register Objects colored
COLUMP0 P0, M0 (player 0, missile 0)
COLUMP1 P1, M1 (player 1, missile 1)
COLUMPF PF, BL (playfield, ball)
COLUMBK BK (background)
+
+

+ For example, if the COLUMP0 register is set for light red, both P0 and M0 + will be light red when drawn. +

+ A color-lum register is set for both color and luminosity by writing a single 7 + bit instruction to that register. Four of the bits select one of the 16 + available colors, and the other 3 bits select one of 8 levels of luminosity + (brightness). The specific codes required to create specific color and lum are + listed in the Detailed Address List of the TIA hardware manual. As with all + registers (except the "strobe" registers), the data written to them is latched + until altered by another write operation. +

+(These registers are referred to here as COLUMxx, but everywhere else +they are COLUxx, including later in this document. --BW) +

+ +5.0 Playfield +

+

+ The PF register is used to create a playfield of walls, clouds, barriers, etc., that are + seldom moved. This low resolution register is written into to draw the left half of the + TV screen only. The right half of the screen is drawn by software selection of whether a + duplication or a reflection of the right half. +

+ The PF register is 20 bits wide, so the 20 bits are written into 3 addresses: + PF0, PF1, and PF2. PF0 is only 4 bits wide and constructs the first 4 "bits" + of the playfield, starting at the left edge of the TV screen. PF1 constructs + the next 8 "bits", and PF2 the last 8 bits" which end at the center of the + screen. The PF register is scanned from left to right and where a "1" is + found the PF color is drawn, and where a "0" is found, the BK color is + drawn. To clear the playfield, obviously zeros must be written into PF0, + PF1, and PF2. +

+ To make the right half of the playfield into a duplication or copy of the left + half, a "0" is written to D0 of the CTLPF (control playfield) register. + Writing a "1" will cause the reflection to be displayed. +

+

+ +6.0 The Moveable Objects Graphics +

+

+ All 5 moveable objects (P0, M0, P1, M1, BL) can be assigned a horizontal + location on the screen and moved left or right relative to their location. + Vertical positions, however, are treated in an entirely different manner. In + principle, these objects appear at whatever scan lines their graphics + registers are enabled. For example, let us assume the ball is to be + positioned vertically in the center of the screen. The screen has 192 scan + lines and we want the ball to be 2 scan lines "thick". The ball graphics + would be disabled until scan line 96, enabled for 2 scan lines, then disabled + for the rest of the frame. Each type of object (players, missiles, and ball) + has its own characteristics and limitations. +

+

+ +6.1 Missile Graphics (M0, M1) +
+ The two missile graphics registers will draw a missile on any scan line by + writing a "1" to the one bit enable missile registers (ENAM0, ENAM1). + Writing a "0" to these registers will disable the graphics. The missiles' left + edge is positioned by a horizontal position register, but the right edge is a + function of how wide the missile is make. Width of a missile is controlled by + writing into bits D4 and D5 of the number-size registers (NUSIZ0, + NUSIZ1). This has the effect of "stretching" the missile out over 1,2,4, or 8 + color clock counts (a full scan line is 160 color clocks). +

+

+ +6.2 Ball Graphics (BL) +
+ The ball graphics register works just like the missile registers. Writing a + "1" to the enable ball register (ENABL) enables the ball graphics until the + register is disabled. The ball can also be "stretched" to widths of 1, 2, 4, or 8 + color clock counts by writing to bits D4 and D5 of the CTRLPF register. +

+ The ball can also be vertically delayed one can line. For example, if the ball + graphics were enabled on scan line 95, it could be delayed to not display on + the screen until scan line 96 by writing a "1" to D0 of the vertical delay + (VDELBL) register. The reason for having a vertical delay capability is + because most programs will update the TIA every 2 lines. This confines all + vertical movements of objects to 2 scan line "jumps". The use of vertical + delay allows the objects to move one scan line at a time. +

+

+ +6.3 Player Graphics (P0, P1) +
+ The player graphics are the most sophisticated of all the moveable objects. + They have all the capabilities of the missile and ball graphics, plus three + move capabilities. Players can take on a "shape" such as a man or an + airplane, and the player can easily be flipped over horizontally to display + the mirror image (reflection) instead of the original image, plus multiple + copies of the players can be created. +

+ The player graphics are drawn line-by-line like all other graphics. The + difference here is each scan line of the player is 8 "bits" wide, whereas the + missiles and ball are one "bit" wide. Therefore, a player can be though of as + being drawn of graph paper 8 squares wide and as tall as desired. To "color + in the squares" of this imaginary graph paper, 8 data bits are written into + the players graphics registers (GP0, GP1). This 8 bit register is scanned + from D7 to D0, and wherever a "1" is found that "square" gets the players' + color (from the color-lum register) and where a "0" is found that "square" + gets the background color. To position a player vertically, simply leave all + "0's" in the graphics registers (GP0, GP1) until the electron beam is on the + scan line desired, write to the graphics register line-by-line describing the + player, then write all "0's" to turn off the players' graphics until the end of + that frame. +

+ To display a mirror image (reflection) instead of the original figure, write a + "1" to D3 of the one bit reflection register (REFP0, REFP1). A "0" written to + these registers restores the original figure. +

+ Multiple copies of players as well as their size are controlled by writing 3 + bits (D0, D1, D2) into the number-size registers (NUSIZ0, NUSIZ1). These + three bits select from 1 to 3 copies of the player, spacing of those copies, as + well as the size of the player (each "square" of the player can be 1, 2, or 4 + clocks wide). Whenever multiple copies are selected, the TIA automatically + creates the same number of copies of the missile for that player. Again, the + specifics of all this are laid out in the TIA hardware manual. +

+ Vertical delay for the players works exactly like the ball by writing a "1" to + D0 in the players' vertical delay registers (VDELP0, VDELP1). Writing a + "0" to these locations disables the vertical delay. +

+

+ +7.0 Horizontal Positioning +
+

+ The horizontal position of each object is set by writing to its associated reset + register (RESP0, RESP1, RESM0, RESM1, RESBL) which are all "strobe" + registers (they trigger their function as soon as they are addressed). That + causes the object to be positioned wherever the electron bean was in its + sweep across the screen when the register was reset. for example, if the + electron beam was 60 color clocks into a scan line when RESP0 was written + to, player 0 would be positioned 60 color clocks "in" on the next scan line. + Whether or not P0 is actually drawn on the screen is a function of the data + in the GP0 register, but if it were drawn, it would show up at 60. Resets to + these registers anywhere during horizontal blanking will position objects at + the left edge of the screen (color clock 0). Since there are 3 color clocks per + machine cycle, and it can take up to 5 machine cycles to write the register, + the programmer is confined to positioning the objects at 15 color clock + intervals across the screen. This "course" positioning is "fine tuned" by the + Horizontal Motion, explained in section 8.0. +

+ Missiles have an additional positioning command. Writing a "1" to D1 of + the reset missile-to-player register (RESMP0, RESMP1) disables that + missiles' graphics (turns it off) and repositions it horizontally to the center + of its associated player. Until a "0" is written to the register, the missile's + horizontal position is locked to the center of its player in preparation to be + fired again. +

+

+ +8.0 Horizontal Motion +
+

+ Horizontal motion allows the programmer to move any of the 5 graphics + objects relative to their current horizontal position. Each object has a 4 bit + horizontal motion register (HMP0, HMP1, HMM0, HMM1, HMBL) that can + be loaded with a value in the range of +7 to -8 (negative values are + expressed in two's complement from). This motion is not executed until the + HMOVE register is written to, at which time all motion registers move their + respective objects. Objects can be moved repeatedly by simply executing + HMOVE. Any object that is not to move must have a 0 in its motion + register. With the horizontal positioning command confined to positioning + objects at 15 color clock intervals, the motion registers fills in the gaps by + moving objects +7 to -8 color clocks. Objects can not be placed at any color + clock position across the screen. All 5 motion registers can be set to zero + simultaneously by writing to the horizontal motion clear register (HMCLR). +

+ There are timing constraints for the HMOVE command. The HMOVE + command must immediately follow a WSYNC (Wait for SYNC) to insure the + HMOVE operation occurs during horizontal blanking. This is to allow + sufficient time for the motion registers to do their thing before the electron + beam starts drawing the next scan line. Also, for mysterious internal + hardware considerations, the motion registers should not be modified for at + least 24 machine cycles after an HMOVE command. +

+

+ +9.0 Object Priorities +
+

+ Each object is assigned a priority so when any two objects overlap the one + with the highest priority will appear to move in front of the other. To + simplify hardware logic, the missiles have the same priority as their + associated player, and the ball has the same priority as the playfield. The + background, of course, has the lowest priority. The following table + illustrates the normal (default) priority assignments. +

+

+ + + + + + +
Priority Objects
1 P0, M0
2 P1, M1
3 BL, PF
4 BK
+
+

+ This priority assignment means that players and missiles will move in front + of the playfield. To make the players and missiles move behind the + playfield, a "1" must be written to D2 of the CTRLPF register. The + following table illustrates how the priorities are affected: +

+

+ + + + + + +
Priority Objects
1 BL, PF
2 P0, M0
3 P1, M1
4 BK
+
+

+ One more priority control is available to be used for displaying the score. + When a "1" is written to D1 of the CTRLPF register, the left half of the + playfield takes on the color of player 0, and the right half the color of player + 1. The game score can now be displayed using the PF graphics register, and + the score will be in the same color as its associated player. +

+

+ +10.0 Collisions +
+

+ The TIA detects collisions between any of the 6 objects it generates (the + playfield and 5 moveable objects). There are 15 possible two-object + collisions which are stored in 15 one bit latches. Each collision register + contains two of these latches which are read by the microprocessor on D6 + and D7 of the data bus for easy access. A "1" on the data line indicates the + collision it records has occurred. The collision registers could be read at any + time but is usually done during vertical blank after all possible collisions + have occurred. The collision registers are all reset simultaneously by + writing to the collision reset register (CXCLR). +

+

+ +11.0 Sound +
+

+ There are two audio circuits for generating sound. They are identical but + completely independent and can be operated simultaneously to produce + sound effects through the TV speaker. Each audio circuit has three + registers that control a noise-tone generator (what kind of sound), a + frequency selection (high or low pitch of the sound), and a volume control. +

+ +11.1 Tone +

+ The noise-tone generator is controlled by writing to the 4 bit audio control + registers (AUDC0, AUDC1). The values written cause different kinds of + sounds to be generated. Some are pure tones like a flute, others have + various "noise" content like a rocket motor or explosion. Even though the + TIA hardware manual lists the sounds created by each value, some + experimentation will be necessary to find "your sound". +

+

+ +11.2 Frequency +
+ Frequency selection is controlled by writing to a 5 bit audio frequency + register (AUDF0, AUDF1). The value written is used to divide a 30KHz + reference frequency creating higher or lower pitch of whatever type of sound + is created by the noise-tone generator. By combining the pure tones + available from the noise-tone generator with frequency selection a wide + range of tones can be generated. +

+

+ +11.3 Volume +
+ Volume is controlled by writing to a 4 bit audio volume register (AUDV0, + AUDV1). Writing 0 to these registers turns sound off completely, and + writing any value up to 15 increases the volume accordingly. +

+

+
+ +12.0 Input Ports +
+

+ There are six input ports whose logic states can be read on D7 by reading + the input port addresses (INPT0, thru INPT5). These six ports are divided + into two types, "dumped" and "latched". +

+ +12.1 Dumped Input Ports (INPT0 thru INPT3) +

+ These four ports are used to read up to four paddle controllers. Each paddle controller + contains an adjustable pot controlled by the knob on the controller. The output of the + pot is used to charge a capacitor in the console, and when the capacitor is charged the + input port goes HI. The microprocessor discharges this capacitor by writing a "1" to D7 + of VBLANK then measures the time it takes to detect a logic one at that port. This + information can be used to position objects on the screen based on the position of the + knob on the paddle controller. +

+

+ +12.2 Latched Input Ports (INPT4, INPT5) +
+ These two ports have latches that are both enabled by writing a "1" or + disabled by writing a "0" to D6 of VBLANK. When disabled the + microprocessor reads the logic level of the port directly. When enabled, the + latch is set for logic one and remains that way until its port goes LO. When + the port goes LO the latch goes LO and remains that way regardless of what + the port does. The trigger buttons of the joystick controllers connect to + these ports. +

+

+
+
+ +

THE PIA (6532)

+

+ +1.0 General +

+

+ The PIA chip is an off-the-shelf 6532 Peripheral Interface Adaptor which + has three functions: a programmable timer, 128 bytes of RAM, and two 8 + bit parallel I/O ports. +

+

+ +2.0 Interval timer +
+

+ The PIA uses the same clock as the microprocessor so that one PIA cycle + occurs for each machine cycle. The PIA can be set for one of four different + "intervals", where each interval is some multiple of the clock (and therefore + machine cycles). A value from 1 to 255 is loaded into the PIA which will be + decremented by one at each interval. The timer can now be read by the + microprocessor to determine elapsed time for timing various software + operations and keep them synchronized with the hardware (TIA chip). +

+ +2.1 Setting the timer +

+ The timer is set by writing a value or count (from 1 to 255) to the address of + the desired interval setting according to the following table : +

+

+ + + + + + +
Hex Address Interval Mnemonic
294 1 clock TIM1T
295 8 clocks TIM8T
296 64 clocks TIM64T
297 1024 clocks T1024T
+
+

+ For example, if the value of 100 were written to TIM64T (HEX address 296) + the timer would decrement to 0 in 6400 clocks (64 clocks per interval x 100 + intervals) which would also be 6400 microprocessor machine cycles. +

+

+ +2.2 Reading the timer +
+ The timer may be read any number of times after it is loaded of course, but + the programmer is usually interested in whether or not the timer has + reached 0. The timer is read by reading INTIM at hex address 284. +

+

+ +2.3 When the timer reaches zero +
+ The PIA decrements the value or count loaded into it once each interval + until it reaches 0. It holds that 0 counts for one interval, then the counter + flips to FF(HEX) and decrements once each clock cycle, rather than once per + interval. The purpose of this feature is to allow the programmer to + determine how long ago the timer zeroed out in the event the timer was + read after it passed zero. +
+
+

+ +3.0 RAM +

+

+ The PIA has 128 bytes of RAM located in the Stella memory map from HEX + address 80 to FF. The microprocessor stack is normally located from FF on + down, and variables are normally located from 80 on up (hoping the two + never meet). +

+

+ +4.0 The I/O ports +
+

+ The two ports (Port A and Port B) are 8 bits wide and can be set for either + input or output. Port A is used to interface to carious hand-held controllers + but Port B is dedicated to reading the status of the Stella console switches. +

+ +4.1 Port B - Console Switches (read only) +

+ Port B is hardwired to be an input port only that is read by addressing + SWCHB (HEX 282) to determine the status of all the console switches + according to the following table: +

+

+ + + + + + + + + +
Data Bit Switch Bit Meaning
D7 P1 difficulty 0 = amateur (B), 1 = pro (A)
D6 P0 difficulty 0 = amateur (B), 1 = pro (A)
D5/D4 (not used)
D3 color - B/W 0 = B/W, 1 = color
D2 (not used)
D1 game select 0 = switch pressed
D0 game reset 0 = switch pressed
+
+
+

+

+ +5.0 Port A - Hand Controllers +
+

+ Port A is under full software control to be configured as an input or an + output port. It can then be used to read or control various hand-head + controllers with the data bits defined differently depending on the type of + controller used. +

+ +5.1 Setting for input or output +

+ Port A has an 8 bit wide Data Direction Register (DDR) that is written to at + SWACNT (HEX 281) to set each individual pin of Port A to either input or + output. The Port A pins are labeled PA0 thru PA7, and writing a "0" to a + pins' DDR bit sets that pin as input, and a "1" sets it as an output. For + example, writing all 0's to SWACNT (the DDR for Port A) sets PA0 thru + PA7 (all 8 pins of Port A) as inputs. If F0 (11110000) were written to + SWACNT then PA7, PA6, PA5 & PA4 would be outputs, and PA3, PA2, PA1 + & PA0 would be inputs. +
+

+ +5.2 Inputting and Outputting +

+ Once the DDR has set the pins of Port A for input or output they may be + read or written to by addressing SWCHA (HEX 280). +
+

+ +5.3 Joystick Controllers +

+ Two joysticks can be read by configuring the entire port as input and + reading the data at SWCHA according to the following table: +

+

+ + + + + + + + + + +
Data BitDirectionPlayer
D7rightP0
D6leftP0
D5downP0
D4upP0
D3rightP1
D2leftP1
D1downP1
D0upP1
+ (P0 = left player, P1 = right player) +

+

+ A "0" in a data bit indicates the joystick has been moved to close that + switch. All "1's" in a player's nibble indicates that joystick is not moving. +

+

+ + 5.4 Paddle (pot) controllers +
+ Only the paddle triggers are read from the PIA. The paddles themselves + are read at INP0 thru INPT3 of the TIA. The paddle triggers can be read at + SWCHA according to the following table : +

+

+ + + + + + + + +
Data Bit Paddle #
D7 P0
D6 P1
D5/D4 (not used)
D3 P2
D2 P3
D1/D0 (not used)
+
+

+

+ +5.5 Keyboard controllers +
+ The keyboard controller has 12 buttons arranged into 4 rows and 3 columns. + A signal is sent to a row, then the columns are checked to see if a button is + pushed, then the next row is signaled and all columns sensed, etc. until the + entire keyboard is scanned and sensed. The PIA sends the signals to the + rows, and the columns are sensed by reading INPT0, INPT1, and INPT4 of + the TIA. With Port A configured as an output port, the data bits will send a + signal to the keyboard controller rows according to the following table : +

+

+ + + + + + + + + + +
Data BitKeyboard RowPlayer
D7bottomP0
D6thirdP0
D5secondP0
D4topP0
D3bottomP1
D2thirdP1
D1secondP1
D0topP1
+(P0 = left player, P1 = right player) +

+ NOTE : a delay of 400 microseconds is necessary between writing to this + port and reading the TIA input ports. +

+
+
+

+ +6.0 Address summary table +

+

+ + + + + + + + + + + +
Hex Address Mnemonic Purpose
280 SWCHA Port A; input or output (read or write)
281 SWACNT Port A DDR, 0= input, 1=output
282 SWCHB Port B; console switches (read only)
283 SWBCNT Port B DDR (hardwired as input)
284 INTIM Timer output (read only)
294 TIM1T set 1 clock interval (838 nsec/interval)
295 TIM8T set 8 clock interval (6.7 usec/interval)
296 TIM64T set 64 clock interval (53.6 usec/interval)
297 T1024T set 1024 clock interval (858.2 usec/interval)
+

+ NOTE: one clock is also one microprocessor machine cycle +

+
+

+ +

PAL/SECAM CONVERSIONS

+

+PAL +

+1. The number of scan lines, and therefore the frame time increases from + NTSC to PAL according to the following table: +

+

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
NTSCPAL
scanlinesmicrosecondsscanlinesmicroseconds
VBLANK402548483085
KERNEL1921222822814656
OVERSCAN301910362314
FRAME2621668631220055
+
+ +

+ 2. Sounds will drop a little in pitch (frequency) because of a slower crystal + clock. Some sounds may need the AUDF0/AUDF1 touched up. +

+ 3. PAL operates at 50 Hz compared to NTSC 60Hz, a 17% reduction. If + game play speed is based on frames per second, it will slow down by 17%. + This can be disastrous for most skill/action carts. If the NTSC version is + designed with 2 byte fractional addition techniques (or anything not based + on frames per second) to move objects, then PAL conversion can be as + simple as changing the fraction tables, avoiding major surgery on the + program. +

+

+SECAM +
+1. SECAM is a little weird. It takes the PAL software, but the console + color/black & white switch is hardwired as black & white. Therefore, it + reads the PAL black & white tables in software and assigns a fixed color to + each lum of black & white according to the following table: +

+

+ + + + + + + + + + +
Lum Color
0 Black
2 Blue
4 Red
6 Magenta
8 green
A cyan
C yellow
E white
+
+

+ There is a trap here: the manual is the same for NTSC, PAL, & SECAM. + This means that the descriptions for black & white must jive between NTSC + & PAL. If you make major changes to PAL black & white to achieve good + SECAM color, NTSC black & white must be made similar. +

+ 2. PAL sounds work fine on SECAM with one exception: when a sound is to + be turned off, it must be done by setting AUDV0/AUDV1 to 0, not by setting + AUDC0/AUDC1 to 0. Otherwise, you get an obnoxious background sound. +

+

+
+ +

TIA 1A - TELEVISION INTERFACE ADAPTOR (MODEL 1A)

+

+ +GENERAL DESCRIPTION +

+

+ The TIA is an MOS integrated circuit designed to interface between an + eight (8) bit microprocessor and a television video modulator and to convert + eight (8) bit parallel data into serial outputs for the color, luminosity, and + composite sync required by a video modulator. +

+ This circuit operates on a line by line basis, always outputting the same + information every television line unless new data is written into it by the + microprocessor. +

+ A hardware sync counter produces horizontal sync timing independent of + the microprocessor. Vertical sync timing is supplied to this circuit by the + microprocessor and combined into composite sync. +

+ Horizontal position counters are used to trigger the serial output of five (5) + horizontally movable objects; two players, two missiles and a ball. The + microprocessor can add or subtract from these position counters to move + these objects right or left. +

+ The microprocessor determines all vertical position and motion by writing + zeros or ones into object registers before each appropriate horizontal line. +

+ Walls, clouds and other seldom moved objects are produced by a low + resolution data register called the playfield register. +

+ A fifteen (15) bit collision register detects all fifteen possible two object + collisions between these six (6) objects (five moveable and one playfield). + This collision register can be read and reset by the microprocessor. Six input + ports are also provided on this chip that can be read by the microprocessor. + These input ports and the collision register are the only chip addresses that + can be read by the microprocessor. All other addresses are write only. +

+ Color luminosity registers are included that can be programmed by the + microprocessor with eight (8) luminosity and fifteen (15) color values. A + digital phase shifter is included on this chip to provide a single color output + with fifteen (15) phase angles. +

+ Two (2) independent audio generating circuits are included, each with + programmable frequency, noise content, and volume control registers. +

+

+ +DETAILED DESCRIPTION +
+

+ +1. Data and addressing +

+

+ Registers on this chip are addressed by the microprocessor as part of its + overall RAM-ROM memory space. The attached table of read-write + addresses summarizes the addressable functions. There are no registers + that are both read and write. Some addresses however are both read and + write, with write data going into one register and read data returning from + a different register. +

+ If the read-write line is low, the data bits indicated in this table will be + written into the addressed write location when the 02 clock goes from high + to low. Some registers are eight bits wide, some only one bit, and some + (strobes) have no bits, performing only control functions (such as resets) + when their address is written. +

+ If the read-write line is high, the addressed location can be read by the + microprocessor on data lines 6 and 7 while the 02 clock is high. +

+ The addresses given in the table refer only to the six (6) real address lines. + If any of the four (4) chip select lines are used for addressing, the addresses + must be modified accordingly. +

+

+ +2. Synchronization +
+

+ A. Horizontal Timing +

+ A hardware counter on this chip produces all horizontal timing (such as + sync, blank, burst) independent of the microprocessor, This counter is + driven from an external 3.58 Mhz oscillator and has a total count of 228. + Blank is decoded as 68 counts and sync and color burst as 16 counts. +
+

+ B. Vertical Timing +

+ There are one bit, addressable registers on this chip for vertical sync and + vertical blank. The timing for these functions is established by the + microprocessor by writing zero or one into these bits. (VSYNC, VBLANK ) +
+

+ C. Composite Sync +

+ Horizontal sync and the output of the vertical sync bit are combined + together to produce composite sync. This composite sync signal drives a + chip output pad to an external composite video resistor network. +
+

+ D. Microprocessor Synchronization +

+ The 3.58 MHz oscillator also clocks a divide by three counter on this chip + whose output (1.19 Mhz) is buffered to drive an output pad called 00. This + pad provides the input phase zero clock to the microprocessor which then + produces the system 02 clock (1.19 Mhz). + Software program loops require different lengths of time to run depending + on branch decisions made within the program. Additional synchronization + between the software and hardware. This is done with a one bit latch called + WSYNC (wait for sync). When the microprocessor finishes a routine such + as loading registers for a horizontal line, or computing new vertical + locations during vertical blank, it can address WSYNC, setting this latch + high. When this latch is high, it drives an output pad to zero connected to + the microprocessor ready line (RDY). A zero on this line causes the + microprocessor to halt and wait. As shown in figure 2, WSYNC latch is + automatically reset to zero by the leading edge of the next horizontal blank + timing signal, releasing the RDY line, allowing the microprocessor to begin + its computation and register writing for this horizontal television line or + line pair. +

+

+
+ +3. Playfield graphics Register +

+

+ A. Description +
+ Objects such as walls, clouds, and score) which are not required to move, are + written into a 20 bit register called the playfield register. This register + (Figure 5) is loaded from the data bus by three separate write addresses + (PF0, PFl, PF2). Playfield may be loaded at any time. To clear the playfield, + zeros must be written into all three addresses. +

+

+ B. Normal Serial Output +
+ The playfield register is automatically scanned (and converted to serial + output) by a bi-directional shift register clocked at a rate which spreads the + twenty (20) bits out over the left half of a horizontal line. This scanning is + initiated by the end of horizontal blank (left edge of television screen). + Normally the same scan is then repeated, duplicating the same twenty (20) + bit sequence over the right half of the horizontal line. +

+

+ C. Reflected Serial Output +
+ A reflected playfield may be requested by writing a one into bit zero of the + playfield control register (CTRLPF). When this bit is true the scanning + shift register will scan the opposite direction during the right half of the + horizontal line, reversing the twenty (20) bit sequence. +

+

+ D. Timing Constraints +
+ Even though the playfield bytes (PF0, PFl, PF2) may be written to any time, + if one of them is changed while being serially scanned, part of the new value + may both show up on the television horizontal line. +

+

+
+ +4. Horizontal Position Counters +
+

+ A. Description +

+ The playfield is a fixed graphics register, always starting its serial output + when triggered by the beginning of each television line. This chip also + includes five "moveable" graphics registers, whose serial outputs are + triggered by five separate horizontal position counters every time these + counters pass through zero count. These position counters are clocked + continuously during the unblanked portion of every horizontal line and + their count length is exactly equal to the normal number of clocks supplied + to them during this time. They will therefore pass through zero at the same + time during each horizontal television line and the triggered outputs will + have no horizontal motion. A typical horizontal counter is shown in figure 4. +

+ If extra clocks are supplied to these counters (or normal clocks suppressed) + the zero crossing time will shift and the object will have moved left (extra + clocks) or right (fewer clocks). Some position counters have extra decoders + (in addition to a zero decode) to trigger multiple copies of the same object + across a horizontal line. +

+ All position counters can be reset to zero count by the microprocessor at any + time, by a write instruction to the reset addresses (RESBL, RESM0, RESMl, + RESP0, RESPl). If reset occurs during horizontal blank, the object will + appear at the left side of the television screen. Properly timed resets may + position an object at any horizontal location consistent with the + microprocessor cycle time. +

+

+ B. Ball position Counter +
+ The ball position counter has only the zero crossing decode and therefore + cannot trigger multiple copies of the ball graphics. +

+

+ C. Player Position Counters +
+ Each player position counter has three decodes in addition to the zero + crossing decode. These decodes are controlled by bits 0,1,2 of the number + size control registers (NUSIZ0, NUSIZ1), and trigger 1,2, or 3 copies of the + player (at various spacings) across a horizontal line as shown on page ___. + These same control bits are used for the decodes on the missile position + counter, insuring an equal number of players and missiles. +
+ D. Missile Position Counters +
+ Missile position counters are identical to player position counters except + that they have another type of reset in addition to the previously discussed + horizontal position reset. These extra reset addresses (RESMP0, RESMP1) + write data bit 1 into a one bit register whose output is used to position the + missile (horizontally) directly on top of its corresponding player, and to + disable the missile serial output. +

+

+
+ +5. Horizontal Motion Registers +

+

+ A. General Description +
+ There are five write only registers on this chip that contain the horizontal + motion values for each of the five moving objects. A typical horizontal + motion register is shown in figure 4 . The data bus (bits 4 through 7) is + written into these addresses (HMP0, HMPl, HMM0, HMMl, HMBL) to load + these registers with motion values. These registers supply extra (or fewer) + clocks to the horizontal position counters only when commanded to do so by + an HMOVE address from the microprocessor. These registers may all be + cleared to zero (no motion) simultaneously by an HMCLR command from + the microprocessor, or individually by loading zeros into each register. + These registers are each four bits in length and may be loaded with positive + (left motion), negative (right motion) or zero (no motion) values. Negative + values are represented in twos complement format. +

+

+ B. Timing constraints +
+ These registers may be loaded or cleared at almost any time. The motion + values they contain will be used only when an HMOVE command is + addressed, and then all five motion values will be used simultaneously into + all five horizontal position counters once. The only timing constraint on this + operation involves the HMOVE command. The HMOVE command must be + located in the microprocessor program immediately after a wait for sync + (WSYNC) command. This assures that the HMOVE operation begins at the + leading edge of horizontal blank, and has the full blank time to supply extra + or fewer clocks to the horizontal position counters. These registers should + not be modified for at least 24 Computer cycles after the HMOVE command. +

+

+
+ +6. Moving Object Graphics Registers +

+

+ A. General Description +
+ There are five graphics registers for moving objects on this chip. These + graphics registers are loaded (written) in parallel by the microprocessor and + like the playfield register are scanned and converted to serial output. + Unlike the playfield register, which is always scanned beginning at the left + side of each horizontal line, moving object graphics registers are scanned + only when triggered by a start decode from their horizontal position + counter. A typical graphics register is shown in figure 4 . +

+

+ B. Missile Graphics +
+ The graphics registers for both missiles are identical and very simple. They + each consist of a one bit register called missile enable (ENAM0, ENAM1). + This graphics bit is scanned (outputted) only when triggered by its + corresponding position counter. There are control bits (bits 4, 5, of NUSIZ0, + NUSIZ1) that can stretch this single graphics bit out over widths of 1, 2, 4, + or 8 clocks of horizontal line time. (A full line is 160 clocks). +

+

+ C. Player Graphics +
+ The graphics registers for both players are identical and are rather complex. + They each consist of eight bit parallel registers (GRP0, GRP1) and a bi- + directional parallel to serial scan counter that converts the parallel data + into serial output. A one bit control register (REFP0, REFP1) determines + the direction (reflection) of the parallel to serial scan, outputing either D7 + through D0, or D0 though D7. This allows reflection (horizontal flipping) of + player serial graphics data without having to flip the microprocessor data. +

+ The clock into the scan counter can be controlled (three bits of NUSIZ0 and + NUSIZ1) to slow the scan rate and stretch the eight bits of serial graphics + out over widths of 8, 16, or 32 clocks of horizontal line time. These same + control bits are used in the player-missile motion counters to control + multiple copies, so only three player widths ( scan + rates) are available. +

+

+ D. Vertical Delay +
+ Each of the player graphics registers actually consists of two 8 bit parallel + registers. The first (GRP0, GRP1) is loaded (written) from the + microprocessor 8 bit data bus. The second is automatically loaded from the + output of the first. The reason for this is a complex subject called vertical + delay. A large amount of microprocessor time is required to generate + player, missile and playfield graphics (table look up, masking, comparisons, + etc.) and load these into this chip's registers. For most game programs this + time is just too large to fit into one horizontal line time. In fact for most + games it will barely fit into two line times (127 microseconds). Therefore, + individual graphics registers are loaded (written) every two lines, and used + twice for serial output between loads. This type of programing will + obviously limit the vertical height resolution of objects to multiples of two + lines. It also will limit the resolution of vertical motion to two lines jumps. + Nothing can be done about the vertical height resolution; however, vertical + motion can be resolved to a single line by addition of a second graphics + register that is automatically parallel loaded from the output of the first, + one line time after the first was loaded from the data bus. This second + graphics register output is therefore always delayed vertically by one line. A + control bit called vertical delay (VDEL0, VDEL1) selects which of these two + registers is to be used for serial output. If this control bit is set by the + microprocessor between picture frames, the object will be moved down + (delayed) by one line during the next frame. In most programming + applications player 0 graphics and player 1 graphics are loaded (written) + alternately, during the blank time just prior to each line as shown in (figure + 1). Since GRP0 and GRP1 addresses from the microprocessor alternate, + they are delayed by one line from each other. The GRP0 address decode can + therefore be used to load the delayed graphics register for player 1, and + GRP1 likewise to load the delayed graphics register for player 0. The two + vertical delay bits (VDEL0, VDELl) then select delayed or undelayed + registers for player 0 and player 1 as serial outputs. +

+

+ E. Ball Graphics +
+ The ball graphics register is almost identical to the missile graphics + register. It also consists of a single enable bit (ENABL) whose output is + triggered by the ball position counter. It also has two control bits (bits 4, 5 + of CTRLPF) that can stretch this single graphics bit out over widths of 1, 2, + 4, or 8 clocks of horizontal line time. Unlike the missile graphics; however, + the ball graphics register has capability for vertical delay similar to the + player graphics. A second graphics (enable) bit is alternately loaded from + the output of the first, one line after the first was loaded from the data bus. + A ball vertical delay bit (VDELBL) selects which of these two graphics bits + is used for the ball serial output. The first graphics bit (ENABL) should be + loaded during the same horizontal blank time as player 0 (GRP0), because + GRP1 is used to load the second enable bit from the output of the first on + alternate lines. +

+

+
+ +7. Collision Detection Latches +

+

+ A. Definitions +
+ The serial outputs from all the graphics registers represent real time + horizontal location of objects on the television screen. If any of these outputs + occur at the same time, they will overlap (collide) on the screen. There are + six objects generated on this chip (five moving and playfield) allowing + fifteen possible two object collisions. These overlaps (collisions) are detected + by fifteen "and" gates whenever they occur, and are stored in fifteen + individual latch register bits, as shown in figure 6. +

+

+ B. Reading Collision +
+ The microprocessor can read these fifteen collision bits on data lines 6 and 7 + by addressing them two at a time. This could be done at any time but is + usually done between frames (during vertical blank) after all possible + collisions have serially occurred. +

+

+ C. Reset +
+ All collision bits are reset simultaneously by the microprocessor using the + reset address CXCLR. This is usually done near the end of vertical blank, + after collisions have been tested. +

+

+
+

+ +8. Input ports +

+

+ A. General Description +
+ There are 6 input ports on this chip whose logic state may be read on data + line 7 with read addresses INPT0 through INPT5. These 6 ports are + divided into two types, "dumped" and "latched". See Figure 8. +

+

+ B. Dumped Input Ports (I0 through I3) +
+ These 4 input ports are normally used to read paddle position from an + external potentiometer-capacitor circuit. In order to discharge these + capacitors each of these input ports has a large transistor, which may be + turned on (grounding the input ports) by writing into bit 7 of the register + VBLANK. When this control bit is cleared the potentiometers begin to + recharge the capacitors and the microprocessor measures the time required + to detect a logic 1 at each input port. +

+ As long as bit 7 of register VBLANK is zero, these four ports are general + purpose high impedance input ports. When this bit is a 1 these ports are + grounded. +

+

+ C. Latched Input ports (I4, I5) +
+ These two input ports have latches which can be enabled or disabled by + writing into bit 6 of register VBLANK. +

+ When disabled, these latches are removed from the circuit completely and + these ports become two general purpose input ports, whose present logic + state can be read directly by the microprocessor. +

+ When enabled, these latches will store negative (zero logic level) signals + appearing on these two input ports, and the input port addresses will read + the latches instead of the input ports. +

+ When first enabled these latches will remain positive as long as the input + ports remain positive (logic one). A zero input port signal will clear a latch + value to zero, where it will remain (even after the port returns positive) + until disabled. Both latches may be simultaneously disabled by writing a + zero into bit 6 of register VBLANK. +

+

+
+ +8.5 Priority Encoder +

+

+ A. Purpose +
+ As discussed in the section on collisions, simultaneous serial outputs from + the graphics registers represent overlap on the television screen. In order to + have color-luminosity values assigned to individual objects it is necessary to + establish priorities between objects when overlapped. The priority encoder + is shown in figure 3. +

+

+ B. Priority Assignment +
+ The lack of any objects results in a color-lum value called the background. + The background (BK) has lowest priority and only appears when no objects + are outputing. In order to simplify the logic, each missile is given the same + color-lum value and priority as its corresponding player (P0, M0) and the + ball is given the same color-lum value and priority as the playfield (PF, BL). +

+ The following table illustrates the normal priority assignment: +

+

+ + + + + +
Highest Priority P0, M0
Second Highest P1, M1
Third Highest PF, BL
Lowest Priority BK
+
+

+ Objects with higher priority will appear to move in front of objects with + lower priority. Players will therefore move in front of playfield (clouds, + walls, etc.). +

+

+ C. Priority Control +
+ There are two playfield control bits that affect priority, one called playfield + priority (PFP) (bit 2 of CTRLPF) and one called score (bit 1 of CTRLPF). + When a one is written into the PFP bit the priority assignment is modified + as shown below. +

+

+ + + + + +
Highest Priority PF, BL
Second Highest P0, M0
Third Highest P1, M1
Lowest Priority BK
+
+

+ Players will then move behind playfield (clouds, wall, etc.). When a one is + written into the score control bit, the playfield is forced to take the color-lum + of player 0 in the left half of the screen and player 1 in the right half of the + screen. This is used when displaying score and identifies the score with the + correct player. The priority encoder produces 4 register select lines shown + in figure 3) that are mutually exclusive. These 4 lines select either + background, player 0, player 1 or playfield, and only one of them can be + true at a time. +

+

+
+ +9. Color Luminance Registers +

+

+ A. Description +
+ There are four registers (shown in figure 3) that contain color-lum codes. + Four bits of color code and three its of luminance code may be written into + each of these registers (COLUP0, COLUP1, COLUPF, COLUBK) by the + microprocessor at any time. These codes (representing 16 color values and 8 + luminance values) are given in the Detailed Address List. +

+

+ B. Multiplexing +
+ The serial graphics output from all six objects is examined by the priority + encoder which activates one of the four select lines into a 4 x 7 multiplexer. + This multiplexer (shown in figure 3) then selects one of the four color-lum + registers as a 7 line output. Three of these lines are binary coded + luminosity and go directly to chip output pads. The other four lines go to the + color phase shifter. +

+

+
+ +10. Color Phase Shifter +

+

+ This portion of the chip (shown in figure 3) produces a reference color + output (color burst) during horizontal blank and then during the unblanked + portion of the line it produces a color output shifted in phase with respect to + the color burst. The amount of phase shift determines the color and is + selected by the four color code lines from the Color-lum multiplexer. Binary + code 0 selects no color. Code 1 selects gold (same phase as color burst). + Codes 2 (0010) through 15 (1111) shift the phase from zero through almost + 360 degrees allowing selection of 15 total colors around the television color + wheel. +

+

+ +11. Audio Circuits +

+

+ Two audio circuits are incorporated on this chip. They are identical and + completely independent, although their outputs could be combined + externally into one speaker. Each audio circuit consists of parts described + below, and in figure 7. +

+ A. Frequency Select +

+ Clock pulses (at approximately 30 KHz) from the horizontal sync counter + pass through a divide by N circuit which is controlled by the output code + from a five bit frequency register (AUDF). This register can be loaded + (written) by the microprocessor at any time, and causes the 30 KHz clocks + to be divided by 1 (code 00000) through 32 (code 11111). This produces + pulses that are digitally adjustable from approximately 30 KHz to 1 KHz + and are used to clock the noise-tone generator. +
+ B. Noise-Tone Generator +
+ This circuit contains a nine bit shift counter which may be controlled by the + output code from a four bit audio control register(AUDC), and is clocked by + the frequency select circuit. The control register can be loaded by the + microprocessor at any time, and selects different shift counter feedback taps + and count lengths to produce a variety of noise and tone qualities. +

+

+ C. Volume Select +
+ The shift counter output is used to drive the audio output pad through four + driver transistors that are graduated in size. Each transistor is twice as + large as the previous one and is enable by one bit from the audio volume + register (AUDV). This audio volume register may be loaded by the + microprocessor at any time. As binary codes 0 through 15 are loaded, the + pad drive transistors are enabled in a binary sequence. The shift counter + output therefore can pull down on the audio output pad with 16 selectable + impedance levels. +
+
+

+


+

+Figure 1. Vertical Delay

+Figure 2. Synchronization
Figure 3. Color-Luminance

+Figure 4. Typical Horizontal Motion Circuit

+Figure 5. Playfield Graphics

+Figure 6. Collision Detection

+Figure 7. Audio Circuit

+Figure 8. Input Ports

+Figure 9. Game System

+

+


+

+ +

Write Address Detailed Functions

+

+ +WSYNC (wait for sync) +

+ This address halts microprocessor by clearing RDY latch to zero. RDY is set true again + by the leading edge of horizontal blank. +

+ Data bits not used +

+ +RSYNC (reset sync) +
+ This address resets the horizontal sync counter to define the beginning of horizontal + blank time, and is used in chip testing. +

+ Data bits not used. +

+ +VSYNC +
+This address controls vertical sync time by writing D1 into the VSYNC latch +
+ + + + +
D1
0Stop vertical sync
1Start vertical sync
+
+
+ +VBLANK +
+ This address controls vertical blank and the latches and dumping transistors on the input + ports by writing into bits D7, D6 and D1 of the VBLANK register. +
+ + + + + + + + +
Data BitValueEffect
D10Stop vert. blank
1Start vert. blank
D60Disable INPT4, INPT5 latches
1Enable INPT4, INPT5 latches
D70Remove INPT1,2,3,6 dump path to ground
1Dump INPT1,2,3,6 to ground
+ Note : Disable latches (D6 = 0) also resets latches to logic true +
+
+ +PF0 (PF1, PF2) +
+ These addresses are used to write into playfield registers +
+ + + + +
PF0D7D6D5D4(not used)
PF1D7D6D5D4D3D2D1D0
PF2D7D6D5D4D3D2D1D0
+
+
+ + PLAYFIELD REGISTERS SERIAL OUTPUT +
+ + + + + +
1 horizontal line ( 160 clocks)Playfield Reflect Control
4..7 7......0 0......7 4..7 7......0 0......7 REF = 0
PF0PF1PF2PF0PF1PF2
center
+ + + +
4..7 7......0 0......7 7......0 0......7 7..4 REF = 1
PF0 PF1 PF2 PF2 PF1 PF0
+

+ each bit = 4 clocks +

+ +CTRLPF +
+ This address is used to write into the playfield control register +(a logic 1 causes action as described below) +
+ + + + + + +
D0REF (reflect playfield)
D1SCORE (left half of playfield gets color of player 0, right half gets color of player
D2PFP (playfield gets priority over players so they can move behind the playfield)
D4Ball Size (see next table)
D5
+

+ + D4 & D5 = BALL SIZE + + + + + + +
D5D4Width
001 clock
012 clocks
104 clocks
118 clocks
+

+
+ +NUSIZ0 (NUSIZ1) +
+ These addresses control the number and size of players and missiles. +
+ Missile Size + + + + + + +
D5D4Width
001 clock
012 clocks
104 clocks
118 clocks
+

+Player-Missile number & player size + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
D2D1D01/2 television line (80 clocks)
8 clocks per square
Description
000X        one copy
001X X      two copies - close
010X   X    two copies - med
011X X X    three copies - close
100X       Xtwo copies - wide
101XX       double size player
110X   X   X3 copies medium
111XXXX     quad sized player
+

+
+ + + +RESP0 (RESP1, RESM0, RESM1, RESBL) +
+ These addresses are used to reset players, missiles and the ball. The object will begin its + serial graphics at the time of a horizontal line at which the reset address occurs. +

+ No data bits are used +

+ +RESMP0 (RESMP1) +
+ These addresses are used to reset the horiz. location of a missile to the center of its + corresponding player. As long as this control bit is true (1) the missile will remain + locked to the center of its player and the missile graphics will be disabled. When a + zero is written into this location, the missile is enabled, and can be moved independently + from the player. +

+

+ D1 = RESMP (missile-player reset) +
+
+ +HMOVE +
+ This address causes the horizontal motion register values to be acted upon during the + horizontal blank time in which it occurs. It must occur at the beginning of horiz. blank in + order to allow time for generation of extra clock pulses into the horizontal position + counters if motion is desired this command must immediately follow a WSYNC + command in the program. +

+ No data bits are used. +

+ +HMCLR +
+This address clears all horizontal motion registers to zero (no motion). + No data bits are used. +
+ + + +HMP0 (HMP1, HMM0, HMM1, HMBL) +
+ These addresses write data (horizontal motion values) into the horizontal motion + registers. These registers will cause horizontal motion only when commanded to do so + by the horiz. move command HMOVE. + The motion values are coded as shown below : +
+ + + + + + + + + + + + + + + + + + +
D7D6D5D4ClocksEffect
0111+7Move left indicated number of clocks
0110+6
0101+5
0100+4
0011+3
0010+2
0001+1
00000No Motion
1111-1Move right indicated number of clocks
1110-2
1101-3
1100-4
1011-5
1010-6
1001-7
1000-8
+
+ WARNING : These motion registers should not be modified during the 24 computer + cycles immediately following an HMOVE command. Unpredictable motion values may + result. +
+ + +ENAM0 (ENAM1, ENABL) +
+ These addresses write D1 into the 1 bit missile or ball graphics registers. +
+ + + + +
D1
0Disables object
1Enables object
+
+
+ +GRP0 (GRP1) +
+ These addresses write data into the player graphics registers. +
+ + + +
D7D6D5D4D3D2D1D0
Graphics Data
+
+ Note: serial output begins with D7, unless REFP0 (REFP1) = 1 +
+ +REFP0 (REFP1) +
+ These addesses write D3 into the 1 bit player reflect registers +
+ + + + +
D3
0no reflect, D7 of GRPx on left
1reflect, D0 of GRPx on left
+
+
+ + +VDELP0 (VDELP1, VDELBL) +
+ These addresses write D0 into the 1 bit vertical delay registers, to delay players or ball by + one vertical line.

+

+ + + + +
D0
0no delay
1delay
+
+
+ +CXCLR +
+This adderess clears all collision latches to zero (no collision). + No data bits are used. +
+ +COLUP0 (COLUP1, COLUPF, COLUBK) +
+ These addresses write data into the player, playfield, and background color-luminance + registers +

+

+ + + + + + + + + + + + + + + + + + +
COLORD7D6D5D4D3D2D1LUM
grey - gold0000000black
 0001001dark grey
orange, brt-org0010010
 0011011grey
pink - purple0100100
 0101101
purp-blue, blue0110110light grey
 0111111white
blue - lt. blue1000
 1001
torq. - grn. blue1010
 1011
grn. - yel. grn.1100
 1101
org. grn - lt org.1110
 1111
+
+
+ +AUDF0 (AUDF1) +
+ These addresses write data into the audio frequency divider registers. +

+

+ + + + + + + + +
D4 D3 D2 D1 D0 30KHz divided by
00000no division
00001divide by 2
00010divide by 3
..................
11110divide by 31
11111divide by 32
+
+
+ +AUDC0 (AUDC1) +
+ These addresses write data into the audio control registers which control the noise + content and additional division of the audio output. +

+

+ + + + + + + + + + + + + + + + + + +
D3 D2 D1 D0 Type of noise or division
0000set to 1
00014 bit poly
0010div 15 -> 4 bit poly
00115 bit poly -> 4 bit poly
0100div 2 : pure tone
0101div 2 : pure tone
0110div 31 : pure tone
01115 bit poly -> div 2
10009 bit poly (white noise)
10015 bit poly
1010div 31 : pure tone
1011set last 4 bits to 1
1100div 6 : pure tone
1101div 6 : pure tone
1110div 93 : pure tone
11115 bit poly div 6
+
+
+ +AUDV0 (AUDV1) +
+ These addresses write data into the audio volume registers which set the pull down + impedance driving the audio output pads. +

+

+ + + + + + + + +
D3 D2 D1 D0 Audio Output Pull down current
0 0 0 0 No output current
0 0 0 1 lowest
0 0 1 0
... ... ... ...
1 1 1 0
1 1 1 1 highest
+
+
+

+


+

+ +

TIA WRITE ADDRESS SUMMARY

+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
6-bit AddressAddress Name76543210Function
00VSYNC ......1. vertical sync set-clear
01VBLANK 11....1. vertical blank set-clear
02WSYNC strobe wait for leading edge of horizontal blank
03RSYNC strobe reset horizontal sync counter
04NUSIZ0 ..111111 number-size player-missile 0
05NUSIZ1 ..111111 number-size player-missile 1
06COLUP0 1111111. color-lum player 0
07COLUP1 1111111. color-lum player 1
08COLUPF 1111111. color-lum playfield
09COLUBK 1111111. color-lum background
0ACTRLPF ..11.111 control playfield ball size & collisions
0BREFP0 ....1... reflect player 0
0CREFP1 ....1... reflect player 1
0DPF0 1111.... playfield register byte 0
0EPF1 11111111 playfield register byte 1
0FPF2 11111111 playfield register byte 2
10RESP0 strobe reset player 0
11RESP1 strobe reset player 1
12RESM0 strobe reset missile 0
13RESM1 strobe reset missile 1
14RESBL strobe reset ball
15AUDC0 ....1111 audio control 0
16AUDC1 ...11111 audio control 1
17AUDF0 ...11111 audio frequency 0
18AUDF1 ....1111 audio frequency 1
19AUDV0 ....1111 audio volume 0
1AAUDV1 ....1111 audio volume 1
1BGRP0 11111111 graphics player 0
1CGRP1 11111111 graphics player 1
1DENAM0 ......1. graphics (enable) missile 0
1EENAM1 ......1. graphics (enable) missile 1
1FENABL ......1. graphics (enable) ball
20HMP0 1111.... horizontal motion player 0
21HMP1 1111.... horizontal motion player 1
22HMM0 1111.... horizontal motion missile 0
23HMM1 1111.... horizontal motion missile 1
24HMBL 1111.... horizontal motion ball
25VDELP0 .......1 vertical delay player 0
26VDELP1 .......1 vertical delay player 1
27VDELBL .......1 vertical delay ball
28RESMP0 ......1. reset missile 0 to player 0
29RESMP1 ......1. reset missile 1 to player 1
2AHMOVE strobe apply horizontal motion
2BHMCLR strobe clear horizontal motion registers
2CCXCLR strobe clear collision latches
+
+

+


+

+ +

TIA READ ADDRESS SUMMARY

+ +
+ + + + + + + + + + + + + + + + +
6-bit AddressAddress Name 76543210 FunctionD7D6
0CXM0P 11...... read collision M0 P1 M0 P0
1CXM1P 11...... read collision M1 P0 M1 P1
2CXP0FB 11...... read collision P0 PF P0 BL
3CXP1FB 11...... read collision P1 PF P1 BL
4CXM0FB 11...... read collision M0 PF M0 BL
5CXM1FB 11...... read collision M1 PF M1 BL
6CXBLPF 1....... read collision BL PF unused
7CXPPMM 11...... read collision P0 P1 M0 M1
8INPT0 1....... read pot port
9INPT1 1....... read pot port
AINPT2 1....... read pot port
BINPT3 1....... read pot port
CINPT4 1....... read input
DINPT5 1....... read input
+ Note : I0, I2, I2, I3 can be grounded + under software control. + I4, I5 can be converted to latched + inputs under software control +
+

+(For PIA I/O addresses, see section 6,0 --BW) +

+


+

+ +Editor's Notes

+(This section is *not* part of the manual, so you can ignore it if +you want! --BW) +

+I did this HTML version of the Stella Programming Guide for a couple of +reasons: +
    +
  • +The existing PDF version is great, but I really hate having to use a PDF +reader to read it. HTML is (supposedly) cross-platform, and since I already +keep 10-15 browser windows open at any given time, I'd rather use the same +GUI for reading the Stella docs as I do for everything else that's graphical. +
  • +The existing plain-text version I found is ok, but it's missing all the +diagrams (of course), and some of the text formatting is a bit strange. +
  • +I tried numerous programs that claim to convert PDF (or PostScript) to HTML, +but they all produced ugly, unusable results. +
  • +I wanted internal links in the document, such as the table of contents. The +existing PDF version doesn't seem to have them. The next version of this +document will probably have links to external `how-to' docs, with code +snippets and such. +
  • +The Stella/Atari development community are some of the nicest people I've +ever met, and this is my attempt to give back to the community in some small +way. +
  • +Finally, I needed an excuse to procrastinate... my poor 2600 game is in a +nasty state of disrepair, and I'm at the point where I'd be better starting +from scratch. Taking a break from 6502 asm and writing in a `language' like +HTML gives me a much-needed breather. +
+

+The original PDF version has a few typos. I don't know if those are in the +original paper version, or mistakes in the transcription, but I fixed as many +of them as I could find (most of them were silly, like it's instead of +its, or sidable for disable). Also, I didn't try to +format the tables exactly as they appear in the PDF version, since HTML isn't +much good for that sort of thing anyway. I did my best to preserve the content, +but took a few liberties with the presentation.

+ +Although I have tried to avoid it, I'm sure I introduced a few errors and +typos of my own. If anybody spots one, let me know, and I'll fix it ASAP.

+ +As far as copyright goes, who knows what the official, legal status of the +Stella Programmer's Guide is? I sure don't. But this HTML version is free for +anyone to use and/or redistribute, as much as the original version is. If +you modify this version, I'd really appreciate it if you'd clearly mark your +new version as modified. Better yet, if you fix some of my mistakes, show me +what I did wrong so I can learn from them!

+ +I looked at this in Netscape 4.7, Opera 5.0b6, and Konqueror 2.1 on Linux, and IE 5.0 on Win98; it looks OK on all of them. +If you have any problems with the HTML formatting, please let me know. I tried to +use `bog standard' HTML as much as possible, so it should work OK on any +browser, on any OS, but if it doesn't, I'll fix it the best I am able.

+ +B. Watson
+atari@hardcoders.org
+09/15/2001
+ +

\ No newline at end of file diff --git a/com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/sizes.txt b/com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/sizes.txt new file mode 100644 index 00000000..2aabc284 --- /dev/null +++ b/com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/sizes.txt @@ -0,0 +1,1966 @@ + + Cart Information + ---- ----------- + + + + Info about cart sizes and bankswitching methods + + + + By Kevin Horton + khorton@iquest.net + + +V6.00 +Last modified on 4/18/97 +(614 images) + +Copyright 1997 K Horton, all rights reserved! You may copy and distribute +this file as long as it remains intact and un-modified. + + +This text has been modified to be machine-readable by Bankzilla's database +generator. This includes the '####Data Start####' and '####Data End####' +flags, as well as including an 'overflow' category and including controller +type. + + + How Bankswitching Works + --- ------------- ----- + +Bankswitching allows game programmers to include more data into a cartridge, +therefore making (hopefully) a better game with more graphics/ levels. + +Bankswitching works under similar principals in all cases. Basically, by +reading a certain location in ROM switches banks. + + +(this is the F8-style of bankswitching) + +Bank #1 Bank #2 +-------------------------------------------------------------- +1000 JSR $1800 (do subroutine) . +1003 (program continues) 1200 _subroutine goes here_ +. 1209 RTS +. . +1800 LDA $1FF9 (switch to bank 2) 1802 (rest of program) +1803 NOP 1803 JSR $1200 +1804 NOP . +1805 NOP . +1806 NOP 1806 LDA $1FF8 (Switch back to bank 1) +1807 NOP . +1808 NOP . +1809 RTS (We're done w/ routine) 1809 (rest of program) + +OK, we start out in bank #1 and we want to run a subroutine in bank #2. + +What happens is this- the processor starts at 1000h in bank #1. We call +our subroutine from here. 1800h: We do a read to change us to bank #2. +Remember that when we change banks, we are basically doing a ROM swap. +(You can think of bankswitching as 'hot-swapping' ROMs) Now that we're +in bank #2, the processor sees that JSR to $1200, which is the subroutine +that we wanted to execute. We execute the subroutine and exit it with an +RTS. This brings us back to 1806h. We then do another read to select bank +#1. After this instruction finishes, the processor is now in bank #1, with +the program counter pointing to 1809, which is an RTS which will take us +back to 1003 and let us continue on with our program. + + + Extra RAM in Carts + ----- --- -- ----- + +Some carts have extra RAM; There are three known formats for this: + +Atari's 'Super Chip' is nothing more than a 128-byte RAM chip that maps +itsself in the first 256 bytes of cart memory. (1000-10FFh) +The first 128 bytes is the write port, while the second 128 bytes is the +read port. This is needed, because there is no R/W line to the cart. + +CBS RAM Plus (RAM+) This maps in 256 bytes of RAM in the first 512 bytes +of the cart; 1000-11FF. The lower 256 addresses are the write port, while +the upper 256 addresses are the read port. To store a byte and retrieve it: + +LDA #$69 ; byte to store +STA $1000 ; store it +. +. ; rest of program goes here +. +LDA $1100 ; read it back +. ; acc=$69, which is what we stored here earlier. + + +M-network (AFAIK it has no name) + +OK, the RAM setup in these carts is very complex. There is a total of 2K +of RAM broken up into 2 1K pieces. One 1K piece goes into 1000-17FF +if the bankswitch is set to $1FE7. The other is broken up into 4 256-byte +parts. + +You select which part to use by issuing a fake read to 1FE8-1FEB. The +RAM is then available for use by all banks at 1800-19FF. + +Similar to other schemes, 1800-18FF is write while 1900-19FF is read. +Low RAM uses 1000-13FF for write and 1400-17FF for read. + + +Note that the 256-byte banks and the large 1K bank are seperate entities. +The M-Network carts are about as complex as it gets. + + + Descriptions of the Various Bankswitch Modes + -------------------------------------------- + +2K: + + -These carts are not bankswitched, however the data repeats twice in the + 4K address space. You'll need to manually double-up these images to 4K + if you want to put these in say, a 4K cart. + +4K: + + -These images are not bankswitched. + +6K: + + -AR: The Arcadia (aka Starpath) Supercharger uses 6K of RAM to store the + games loaded from tape. + +8K: + + -F8: This is the 'standard' method to implement 8K carts. There are two + addresses which select between two unique 4K sections. They are 1FF8 + and 1FF9. Any access to either one of these locations switches banks. + Accessing 1FF8 switches in the first 4K, and accessing 1FF9 switches in + the last 4K. Note that you can only access one 4K at a time! + + -FE: Used only on two carts (Robot Tank and Decathlon). You select banks + via accesses to the stack. You set the stack pointer to FF, and then a + JSR switches banks one way, while RTS switches you back to the original + bank (both banks are 4K). This allows the programmers to perform + 'automatic' bankswitching. All the subroutines are in one bank, while + all the game code is in another. When you perform a JSR; you switch banks + to the bank containg the subroutines. Upon encoutering an RTS, the bank + is switched back to the original calling bank. Pretty spiffy! + + -E0: Parker Brothers was the main user of this method. This cart is + segmented into 4 1K segments. Each segment can point to one 1K slice of + the ROM image. You select the desired 1K slice by accessing 1FE0 to 1FE7 + for the first 1K (1FE0 selects slice 0, 1FE1 selects slice 1, etc). + 1FE8 to 1FEF selects the slice for the second 1K, and 1FF0 to 1FF8 selects + the slice for the third 1K. The last 1K always points to the last 1K + of the ROM image so that the cart always starts up in the exact same place. + + -3F: Tigervision was the only user of this intresting method. This works + in a similar fashion to the above method; however, there are only 4 2K + segments instead of 4 1K ones, and the ROM image is broken up into 4 2K + slices. As before, the last 2K always points to the last 2K of the image. + You select the desired bank by performing an STA $3F instruction. The + accumulator holds the desired bank number (0-3; only the lower two bits + are used). Any STA in the $00-$3F range will change banks. This appears to + interfere with the TIA addresses, which it does; however you just use + $40 to $7F instead! :-) $3F does not have a corresponding TIA register, so + writing here has no effect other than switching banks. Very clever; + especially since you can implement this with only one chip! (a 74LS173) + +12K: + + -FA: Used only by CBS. Similar to F8, except you have three 4K banks + instead of two. You select the desired bank via 1FF8, 1FF9, and 1FFA. + These carts also have 256 bytes of RAM mapped in at 1000-11FF. 1000-10FF + is the write port while 1100-11FF is the read port. + +16K: + + -F6: The 'standard' method for implementing 16K of data. It is identical + to the F8 method above, except there are 4 4K banks. You select which + 4K bank by accessing 1FF6, 1FF7, 1FF8, and 1FF9. + + -E7: Only M-Network used this scheme. This has to be the most complex + method used in any cart! :-) It allows for the capability of 2K of RAM; + although it doesn't have to be used (in fact, only one cart used it- + Burgertime). This is similar to the 3F type with a few changes. There are + now 8 2K banks, instead of 4. The last 2K in the cart always points to + the last 2K of the ROM image, while the first 2K is selectable. You + access 1FE0 to 1FE6 to select which 2K bank. Note that you cannot select + the last 2K of the ROM image into the lower 2K of the cart! Accessing + 1FE7 selects 1K of RAM at 1000-17FF instead of ROM! The 2K of RAM is + broken up into two 1K sections. One 1K section is mapped in at 1000-17FF + if 1FE7 has been accessed. 1000-13FF is the write port, while 1400-17FF + is the read port. The second 1K of RAM appears at 1800-19FF. 1800-18FF + is the write port while 1900-19FF is the read port. You select which + 256 byte block appears here by accessing 1FF8 to 1FFB. + +32K + + -F4: The 'standard' method for implementing 32K. Only one cart is known + to use it- Fatal Run. Like the F6 method, however there are 8 4K + banks instead of 4. You use 1FF4 to 1FFB to select the desired bank. + +64K + + -F0: Only used one cart, AFAIK. (the 'Megaboy' cart from Dynacom) It + has 16 4K banks. Accessing 1FF0 will increment the current bank. The + program uses location 1FEC to tell it which bank it's in. There's a + little loop at 1FE0 that checks this location against the accumulator, + and if they're equal it does an RTS. Otherwise it does an STA 1FF0 + and repeats the loop. + + + /-----------------------------------------------------------------------\ + | KEY | + | --- | + | Name - Game Name | + | Part # - Part Number of the actual cart | + | RA - Rarity, according to VGR's guide and my observations | + | SZ - Size of the ROM image in K | + | SC - If the cart has a Special Chip | + | BS - Bankswitch method used (see below) | + | IM - 'X'ed if I have the image | + | SP - Special Attribute (See the end of a section for details) | + | CT - Controller type (See below) | + | Filename - The filename of the ROM image | + | | + | Bankswitch Types: | + | ----------------- | + | (See above for full descriptions) | + | | + | - (nothing); Not bankswitched (2K and 4K only) | + | F8 - 'Standard' 8K; uses 1FF8 and 1FF9 | + | F6 - 'Standard' 16K; uses 1FF6 to 1FF9 | + | F4 - 'Standard' 32K; uses 1FF4 to 1FFB | + | F0 - Megaboy 64K; uses 1FF0 to increment bank # | + | SC - Superchip; 128 bytes of RAM @ 1000-10FF (i.e. F8+SC, F4+SC) | + | FA - 'RAM+' 12K; uses 1FF8 to 1FFA; 256 bytes of RAM @ 1000 to 11FF | + | FE - 'Activision' 8K; uses 01FE and 01FF to determine bank | + | E0 - 'Parker Brothers' 8K; uses 1FE0 to 1FF7 | + | E7 - 'M-Network' 16K; Uses 1FE0 to 1FE7 and 2K of RAM at 1800-19FF | + | 3F - 'Tigervision' 8K; Uses STA $3F to determine bank # | + | AR - 'Arcadia' 6K; Used on the Supercharger | + | ?? - Unknown at this time | + | | + | Controller Types: | + | ----------------- | + | | + | - (nothing); Unknown at this time | + | J - Joystick | + | P - Paddles | + | K - Keypad | + | JK - Joystick and keypad (Star Raiders) | + | D - Driving Controllers | + | B - Joystick plus Booster Grip (Omega Race) | + | T - Track & Field controller | + | O - Other | + | L - Light Gun | + | | + | | + | I have added two new categories to the rarity rating: | + | | + | PR - This was only available as a prototype | + | DM - This image is only a demo, and not really a game | + | | + \-----------------------------------------------------------------------/ + + + + + Hot Wants (that I know I probably won't get :-) + ----------------------------------------------- + + +Tempest (Atari) +Good Luck, Charlie Brown (Atari) +Miss Piggy's Wedding (exists?) (Atari) +Wizard (Atari) +BMX Airmaster (atari ver) (Atari) +White Water Madness (exists?) (Atari) +Rodeo (Atari) +Rabbit Transit (Atari) +Nightmare Manor (Atari) +Pink Panther (Probe 2000) +The Impossible Game (Telesys) +Ewoks Adventure (Parker Bros) +Thwoker (Activision) +Out of Control (Avalon Hill) +Berenstein Bears (Coleco) +Video Life (Commavid) +Aerial Ace (exist?) (Imagic) +Lady in Wading (Playaround) +Snowplow (Sunrise) +Noah and the Ark (Sunrise) +Meltdown (exist?) (20th cent.) +Tomarc the Barabrrian (exist?) (Xonox) +Motocross Racer (exist?) (Xonox) + + + +Anything by Action Hi-tech + + + + + +####Data Start#### + + +*********************************** +* Atari * +*********************************** + +[If SC is marked, cart uses a 'Super Chip'; aka CO20231] + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Combat CX2601 C 2 X J COMBAT +Air-Sea Battle CX2602 U 2 X J AIR_SEA +Star Ship CX2603 R 2 X STARSHIP +Space War CX2604 U 2 X SPACEWAR +Outlaw CX2605 U 2 X J OUTLAW +Slot Racers CX2606 U 2 X SLOTRACE +Canyon Bomber CX2607 U 2 X CANYONB +Super Breakout CX2608 U 4 X P SUPERB +Defender CX2609 C 4 X J DEFENDER +Warlords CX2610 U 4 X P WARLORDS +Indy 500 CX2611 U 2 X D INDY_500 +Street Racer CX2612 U 2 X STRTRACE +Adventure CX2613 C 4 X J ADVNTURE +Steeple Chase CX2614 NR 2 X P STEPLCHS +Demons to Diamonds CX2615 U 4 X DEMONDIM +Hot Rox CX2615 NR 4 X DEMONDIM +Pele's Soccer CX2616 C 4 X PELE +Backgammon CX2617 U 4 X BACKGAM +3D- Tic-Tac-Toe CX2618 U 2 X J 3D_TIC +Stellar Track CX2619 NR 4 X STELRTRK +BASIC Programming CX2620 R 4 X K BASIC +Video Olympics CX2621 C 2 X P VID_OLYM +Breakout CX2622 C 2 X P BREAKOUT +Homerun CX2623 C 2 X HOMERUN +Basketball CX2624 C 2 X BASKETBL +Football CX2625 C 2 X FOOTBALL +Minature Golf CX2626 U 2 X MIN_GOLF +Human Cannonball CX2627 U 2 X HUMAN_CB +Bowling CX2628 C 2 X J BOWLING +Sky Diver CX2629 U 2 X SKYDIVER +Circus Atari CX2630 C 4 X P CIRCATRI +Superman CX2631 C 4 X J SUPRMAN1 +Space Invaders CX2632 C 4 X J SPCINVAD +Night Driver CX2633 C 2 X NIGHTDRV +Golf CX2634 C 2 X GOLF +Maze Craze CX2635 U 4 X J MAZECRZ +Video Checkers CX2636 R 4 X J CHECKERS +Dodge 'Em CX2637 U 4 X DODGE_EM +Missile Command CX2638 C 4 X J MISSCOMM +Othello CX2639 R 2 X OTHELLO +Realsports Baseball CX2640 U 8 F8 X RS_BASEB +Surround CX2641 C 2 X J SURROUND +A Game of Concentration CX2642 C 2 X CONCENTR +Code Breaker CX2643 U 2 X CODEBRK +Flag Capture CX2644 U 2 X FLAGCAP +Video Chess CX2645 U 4 X VIDCHESS +Pac-Man CX2646 C 4 X J PACMAN +Submarine Commander CX2647 PR 4 X J SUBCOMDR +Video Pinball CX2648 U 4 X J VIDPIN +Asteroids CX2649 C 8 F8 X J ASTEROID +Berzerk CX2650 C 4 X J BERZERK +Blackjack CX2651 R 2 X BLACK_J +Casino CX2652 U 4 X CASINO +Slot Machine CX2653 R 2 X SLOTMACH +Haunted House CX2654 C 4 X J HAUNTHSE +Yar's Revenge CX2655 C 4 X J YAR_REV +Swordquest Earthworld CX2656 C 8 F8 X J SQ_EARTH +Swordquest Fireworld CX2657 C 8 F8 X J SQ_FIRE +Math Gran Prix CX2658 C 4 X J MATH_GPX +Raiders of the Lost Ark CX2659 C 8 F8 X J RAIDERS +Star Raiders CX2660 U 8 F8 X JK STARRAID +Basic Math CX2661 C 2 X J BASMATH +Hangman CX2662 U 4 X J HANGMAN +Road Runner CX2663 ER 16 F6 X J ROADRUNR +Brain Games CX2664 U 2 X BRAINGMS +Frog Pond CX2665 PR 8 F8 X J FROGPOND +Realsports Volleyball CX2666 U 4 X RS_VOLLY +Realsports Soccer CX2667 U 8 F8 X RSSOCCER +Realsports Football CX2668 C 8 F8 X RS_FOOTB +Vanguard CX2669 C 8 F8 X VANGUARD +Atari Video Cube CX2670 R 4 X J VIDCUBE +Swordquest Waterworld CX2671 UR 8 F8 X J SQ_WATER +Swordquest Airworld CX2672 NR ?? ---No Known Copies Exist--- +Phoenix CX2673 C 8 F8 X J PHOENIX +E.T. The Extra-Terrestrial CX2674 C 8 F8 X J E_T +Ms. Pac-Man CX2675 C 8 F8 X J MSPACMAN +Centipede CX2676 C 8 F8 X J CENTIPED +Dig Dug CX2677 U 16 X F6 X J DIGDUG +Dukes of Hazzard CX2678 PR 16 F6 X J DUKES +Realsports Basketball CX2679 NR ?? ---No Known Copies Exist--- +Realsports Tennis CX2680 U 8 F8 X J RSTENNIS +Battlezone CX2681 U 8 F8 X J BATLZONE +Krull CX2682 R 8 F8 X J KRULL +Crazy Climber CX2683 ER 8 F8 X J CRAZCLMB +Galaxian CX2684 U 8 F8 X J GALAXIAN +Gravitar CX2685 U 8 F8 X J GRAVITAR +Quadrun CX2686 ER 8 F8 X J QUADRUN +Tempest CX2687 PR ?? ---A Few Proto's Exist--- +Junglehunt CX2688 U 8 F8 X J JNGLHUNT +Kangaroo CX2689 U 8 F8 X J KANGAROO +Pengo CX2690 ER 8 F8 X J PENGO +Joust CX2691 C 8 F8 X J JOUST +Moon Patrol CX2692 U 8 F8 X J MOONPTRL +Food Fight CX2693 NR ?? ---No Known Copies Exist--- +Pole Position CX2694 C 8 F8 X J POLEPSN +Xevious CX2695 PR 8 F8 X J XEVIOUS +Asterix CX2696 ER 8 F8 X J ASTERPAL +Mario Bros. CX2697 U 8 F8 X J MARIOBRO +Rubik's Cube CX2698 ER 4 X J RUBIKS +Taz CX2699 R 8 F8 X J TAZ +Oscar's Trash Race CX26101 R 8 F8 X K OSCAR +Cookie Monster Crunch CX26102 R 8 F8 X K COOKMONS +Alpha-Beam with Ernie CX26103 R 8 F8 X K ALPHBEAM +Big Bird's Egg Catch CX26104 R 8 F8 X K EGGCATCH +3-D Asteroids CX26105 NR ?? +Grover's Music Maker CX26106 PR 8 F8 X GROVER +Snow White CX26107 NR ?? +Donald Duck's Speedboat CX26108 PR 8 F8 X J DDUCKSBT +Sourcerer's Apprentice CX26109 R 8 F8 X J SORCAPRN +Crystal Castles CX26110 U 16 X F6 X J XTALCAST +Snoopy and the Red Barron CX26111 R 8 F8 X J SNOOPY +Good Luck; Charlie Brown CX26112 PR ?? ---One Proto Exists--- +Miss Piggie's Wedding CX26113 UR ?? +Pigs in Space CX26114 ER 8 F8 X J PIGSPACE +Dumbo's Flying Circus CX26115 PR 8 F8 X J DUMBO_N +Galaga CX26116 NR ?? +Obelix CX26117 ER 8 F8 X J OBELIX +Millipede CX26118 R 16 X F6 X J MILLIPED +Saboteur CX26119 PR 8 F8 X J SABOTEUR +Star Gate CX26120 U 8 X F8 X J STARGATE +Defender ][ CX26120 R 8 X F8 X J DEFENDR2 +Zookeeper CX26121 NR ?? +Sinistar CX26122 PR 8 F8 X J SINISTAR +Jr. Pac-Man CX26123 U 16 X F6 X J JRPACMAN +Choplifter CX26124 NR ?? +Track & Field CX26125 R 16 F6 X T TRACK +Elevator Action CX26126 NR ?? +Gremlins CX26127 ER 8 F8 X J GREMLINS +Boing CX26128 NR ?? +Midnight Magic CX26129 R 16 F6 X J MIDNIGHT +Honker Bonker CX26130 NR ?? +Monstercise CX26131 PR 8 F8 X MONS +Garfield CX26132 NR ?? +The A-Team CX26133 PR 8 F8 X J ATEAM +The Last Starfighter CX26134 NR ?? +Star Raiders ][ CX26134 NR ?? +Realsports Boxing CX26135 U 16 F6 X J RSBOXING +Solaris CX26136 U 16 F6 X J SOLARIS +Peek-A-Boo CX26137 PR 4 X PEEKABOO +Super Soccer CX26138 NR ?? +Crossbow CX26139 U 16 F6 X J CROSSBOW +Desert Falcon CX26140 C 16 X F6 X J DSRTFALC +Motor Psycho CX26141 NR ?? +Crack'ed CX26142 ?? +Donkey Kong CX26143 U 4 X J DK +Donkey Kong Jr. CX26144 R 8 F8 X J DKJR +Venture CX26145 R 4 X J VENTURE +Mousetrap CX26146 R 4 X J MOUSETRP +Frogger CX26147 NR 4 X J FROGGER +Turbo CX26148 NR ?? +Zaxxon CX26149 NR ?? +Q*Bert CX26150 R 4 X J QBERT_PB +Dark Chambers CX26151 U 16 X F6 X X3 J DARKC +Super Baseball CX26152 U 16 F6 X J SUPBBALL +Super Football CX26154 U 16 X F6 X J SPRFOOTB +Sprintmaster CX26155 R 16 X F6 X J SPRNMAST +Combat II (Wizard?) CX26156 NR ?? + CX26157 ?? +Surround II CX26158 NR ?? +Double Dunk CX26159 R 16 F6 X J DOUBDUNK + CX26160 ?? + CX26161 ?? +Fatal Run CX26162 UR 32 X F4 X J FATALRUN +32-in-1 CX26163 ER 64 32 banks of 2K -- 32IN1 + CX26164 ?? +Jinks CX26165 NR ?? + CX26166 ?? +Street Fight CX26167 ?? +Off the Wall CX26168 ER 16 X F6 X J OFTHWALL +Shooting Arcade CX26169 PR 16 X F6 X L SHOOTING +Secret Quest CX26170 R 16 X F6 X J SECRETQ +Motorodeo CX26171 UR 16 F6 X J MOTOR +Xenophobe CX26172 ER 16 F6 X J XENOPHOB + CX26173 ?? + CX26174 ?? + CX26175 ?? +Radar Lock CX26176 R 16 X F6 X J RADARLOK +Ikari Warriors CX26177 R 16 F6 X J IKARIWAR +Save Mary! CX26178 PR 16 X F6 X J SAVEMARY + CX26179 ?? + CX26180 ?? + CX26181 ?? + CX26182 ?? +Sentinel CX26183 R 16 F6 X L SENTINEL +White Water Madness CX26184 UR ?? + CX26185 ?? + CX26186 ?? + CX26187 ?? + CX26188 ?? + CX26189 ?? +BMX Airmaster CX26190 ER ?? + CX26191 ?? +KLAX CX26192 ER 16 X F6 X X2 J KLAXNTSC + + +Aquaventure CX26??? PR 8 F8 X J AQUAVENT +Bionic Breakthrough CX26??? PR 8 F8 X O MINDLINK +Bugs Bunny CX26??? PR 8 F8 X J BUGSBUN +Coke Wins! CX26??? UR 4 X J COKEWINS +Holy Moley CX26??? PR 8 F8 X HOLEMOLE +Polo CX26??? PR 2 X J POLO +Rodeo CX26??? PR ?? +Rabbit Transit CX26??? PR ?? +Standalone Test Tape MAO17600 DM 2 X -- MAO17600 +Nightmare Manor CX26??? PR ?? +Super Stunt Cycle PR 2 X J STUNT-1 +Dukes of Hazzard (not CX2678) PR 2 X J STUNT-2 + +X2: Special Best Prototype NTSC version. + +X3: Doesn't like my test cart; have to disable SC for it to start. I + can then re-enable the SC and it'll work. Also, the cart itsself + doesn't work on my 7800 (or my test cart for that matter). + + +*********************************** +* Action Hi-Tech * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Crab Control 605077 UR ?? +F-18 vs. Aliens ???? UR ?? +Galaxy Invader ???? UR ?? +Space Grid ???? UR ?? +Tank City ???? UR ?? +War Zone ???? UR ?? + + +*********************************** +* Activision * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Dragster AG-001 U 2 X J DRAGSTER +Boxing AG-002 U 2 X J BOXING +Checkers AG-003 ER 2 X J CHECKERA +Fishing Derby AG-004 U 2 X J FISHDRBY +Skiing AG-005 U 2 X J SKIING +Bridge AX-006 U 4 X J BRIDGE +Tennis AG-007 C 2 X J TENNIS +Laser Blast AG-008 C 2 X J LASRBLST +Freeway AG-009 U 2 X J FREEWAY +Kaboom AG-010 U 2 X P KABOOM +Stampede AG-011 U 2 X J STAMPEDE +Ice hockey AX-012 U 4 X J ICEHOCKY +Barnstroming AX-013 U 4 X J BARNSTRM +Gran Prix AX-014 U 4 X J GRANDPRX +Chopper Command AX-015 U 4 X J CHOPRCMD +Starmaster AX-016 U 4 X J STARMAST +Megamania AX-017 U 4 X J MEGAMAN +Pitfall AX-018 C 4 X J PITFALL +Sky Jinks AG-019 R 2 X J SKYJINKS +River Raid AX-020 U 4 X J RIVERAID +Spider Fighter AX-021 U 4 X J SPIDRFTR +Seaquest AX-022 R 4 X J SEAQUEST +Oink! AX-023 R 4 X J OINK +Dolphin AX-024 R 4 X J DOLPHIN +Keystone Kapers AX-025 U 4 X J KEYSTONE +Enduro AX-026 U 4 X J ENDURO_A +Plaque Attack AX-027 R 4 X J PLAQATTK +Robot Tank AZ-028 R 8 F8 X X2 J ROBO_FIX +Crackpots AX-029 R 4 X J CRACKPOT +Decathlon AZ-030 R 8 FE X J DECATHLN +Frostbite AX-031 R 4 X J FROSTBIT +Pressure Cooker AZ-032 R 8 F8 X J PRESSURE +Space Shuttle AZ-033 R 8 F8 X J SPCSHUTL +Private Eye AG-034-04 R 8 F8 X J PRIVEYE +Pitfall ][: Lost Caverns AB-035-04 R 8 F8 X X1 J PITFALL2 +H.E.R.O. AZ-036-04 R 8 F8 X J HERO +Beamrider AZ-037-04 R 8 F8 X J BEAMRIDE +Cosmic Commuter AG-038 R 4 X J CSMCOMTR +Kung-Fu Master AX-039 R 8 F8 X J KUNG_FU +Contenders AK-041 ---Never Released by Activision--- +Commando AK-043 R 16 F6 X J COMMANDO +Fighter Pilot AK-046 UR 16 F6 X J FIGHTERP +River Raid ][ AK-048-04 ER 16 F6 X J RIVRAID2 +Rampage AK-049 R 16 F6 X J RAMPAGE +Double Dragon AK-050-04 R 16 F6 X J DBLDRAGN +Ghostbusters AZ-108-04 R 8 F8 X J GHOSTBST +Ghostbusters ][ A?-??? NR ?? ---See Salu--- +Dreadnaught Factor A?-??? NR ?? +Wing War A?-??? NR ?? ---See Imagic--- +Thwocker A?-??? PR ?? ---Proto Exists--- +Zenji A?-??? NR ?? + + +X1: Uses Activision's version of the 'Super Chip' No extra RAM this time, + however it does have extra ROM! Does three-channel sound, and even + includes several random # generators. + + +X2: This used to be an FE cart, but has been fixed to run as an F8. + +*********************************** +* Absolute Entertainment * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Title Match Pro Wresting AG-041 R 8 F8 X J PROWREST +Skateboardin' AG-042 R 8 F8 X J SKATEBRD +Pete Rose Baseball AK-045 R 16 F6 X J PETEROSE +Tomcat F-14 Simulator AK-046 ER 16 F6 X X1 J FIGHTERP +My Golf A?-??? ER 8 F8 X J MYGOLF + +X1: This is identical to the Activision one. I read both in and compared; +they are byte-for-byte identical. + +*********************************** +* American Videogame * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Tax Avoiders ???? R 8 F8 X J TAXAVOID + + +*********************************** +* Amiga * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Mogul Maniac 3120 ER 4 X J MOGULMAN +Surf's Up 3125 PR 8 F8 X J SURF_FIX +Off your Rocker 3130 PR 4 X J OFFROCKR + + +*********************************** +* Answer Software * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Malagai ASC1001 UR 4 X J MALAGAI +Gauntlet ASC1002 UR 4 X J GAUNTLET +Confrontation ASC2001 UR ?? +Personal Game Programmer PGP-1 UR XX ---Hardware--- + + +*********************************** +* Artic * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Space Robot SM8001 UR 4 X J SPCROBOT +Astrowar SM8002 UR 4 X J ASTROWAR + + +*********************************** +* Apollo * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Skeet Shoot AP 1001 R 2 X SKEETSHT +Spacechase AP 2001 U 4 X SPACHASE +Space Cavern AP 2002 U 4 X SPACECAV +Racquetball AP 2003 U 4 X RACQUETB +Lost Luggage AP 2004 U 4 X J LOSTLUGG +Lochjaw AP 2005 ER 4 X J LOCHJAW +Shark Attack AP 2005 U 4 X J SHARKATK +Infiltrate AP 2006 U 4 X J INFILTRT +Kyphus AP 2007 R ?? +Guardian AP 2008 R 4 X GUARDIAN +Final Approach AP 2009 R 4 X J FINLAPCH +Wabbit AP 2010 R 4 X J WABBIT +Pompeii AP 2011 R ?? +Squoosh AP 2012 PR ?? + + +*********************************** +* Avalon Hill * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Death Trap 50010 UR 4 X J DETHTRAP +London Blitz 50020 ER 4 X J LONDBLTZ +Wall Ball 50030 ER 4 X J WALLBALL +Shuttle Orbiter 50040 UR 4 X J SHTLORBT +Out of Control 50050 UR ?? Need! + + +*********************************** +* Bit Corp. * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Sea Monster PG201 R 4 X SEAMNSTR +Space Tunnel PG202 R 4 X SPACT_TW +Phantom Tank PG203 R 4 X PHANTOMT +Open Sesame PG204 R 4 X OPENSESM +Dancing Plate PG205 R 4 X DANCPLAT +Bobby is Going Home PG206 R 4 X J BOBBY +Mission 3000 AD PG207 R 4 X MISN3000 +Snail Against Squirrel PG208 R 4 X SNALSQRL +Mr. Postman PG209 R 4 X MRPOSTMN +Superman (CCE) ???? ?? 4 X SUPERCCE + +*********************************** +* Bob Colbert * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Okie Dokie (Lights Out) ???? -- 2 X J OKIEDOKE + + +*********************************** +* Bomb * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Assault CA281 ER 4 X J ASSAULT +Great Escape CA282 ER 4 X J GRESCAPE +Z-Tack CA283 ER 4 X J Z_TACK +Wall Defender CA285 ER 4 X J WALLDFND + + +*********************************** +* CBS Electronics * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Wizard of Wor M8774 R 4 X J WIZRDWOR +Gorf M8793 U 4 X J GORF +Blueprint 4L-2486 U 8 F8 X J BLUEPRNT +Solar Fox 4L-2487 U 8 F8 X J SOLARFOX +Tunnel Runner 4L-2520 R 12 FA X J TUNLRUNR +Omega Race 4L-2737 U 12 FA X B OMEGARAC +Mountain King 4L-2738 R 12 FA X J MTNKING +Wings ???? NR ?? ?? + +'SC' in this case refers to RAM+ + +*********************************** +* Coleco * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Donkey Kong 2451 C 4 X J DK +Zaxxon 2454 U 8 F8 X J ZAXXON +Venture 2457 C 4 X J VENTURE +Mouse Trap 2459 U 4 X J MOUSETRP +Lady Bug 2463 UR ?? +Cosmic Avenger 2464 UR ?? +Smurf: RIGC 2465 U 8 F8 X J SMURFRES +Carnival 2468 U 4 X J CARNIVAL +Smurfs Save the Day 2511 UR 8 F8 X J SMRFSAVE +Donkey Kong Jr. 2653 R 8 F8 X J DKJR +Mr. Do! 2656 R 8 F8 X J MRDO +Berinstein Bears 2658 UR ?? +Time Pilot 2663 R 8 F8 X J TIMEPLT +Front Line 2665 R 8 F8 X J FRNTLINE +Roc 'N Rope 2667 R 8 F8 X J ROCNROPE + + + +*addendum* +Got Mr. Do! to read out, and Time Pilot seems to have read out correctly, +yet it doesn't play on the emu. I'm going to pull the EPROM off the board +and read it directly! +*end* + +*addendum2* +Pulled the EPROM and copied it. Just as I suspected, it did indeed read out +correctly. Guess the emu isn't as good as the Real Thing. :-) I'll try it +out on the real hardware RSN. +*end* + +*addendum3* +The saga continues. I tried the ROM image of Time Pilot out on the Real +Thing, and it behaved the exact same way that it did on the emu. It appears +that the RC delay in the cart is required so it doesn't switch banks +immediately. I hope I can fix it so it can work as a normal F8 cart. +*end* + +*addendum4* +Yes! I got Time Pilot to work! This is intresting. The bank is *only* +flipped in two parts of the cart- once at the beginning of bank0 and once +at the beginning of bank1. The tip-off was bank #0 had a BIT $1FF8 +instruction and bank #1 had a BIT $1FF9 instruction! This of course will +*not* flip banks!!! I changed the 4K blocks around, ran it on the emu, and +it worked perfectly! +*end* + +*addendum5* +Finally figured out what was wrong with Smurfs Save the Day. The places +in the code that switch banks was right on top of each other. (the +STA $1FFx instructions were at the same addresses in diffrent banks) +As a result, there were only STA $1FF8 instructions rather than both +STA $1FF8 and STA $1FF9 instructions. Fixing these resulted in a working +ROM image!!!!!! Now all I need is Berinstein Bears to round out the +Coleco Collection. +*end* + +*********************************** +* CommaVid * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +MagiCard CM-001 UR ?? ----Hardware---- +Video Life CM-002 UR ?? +Cosmic Swarm CM-003 R 2 X J COSMSWRM +Room of Doom CM-004 ER 4 X J ROOMDOOM +Mines of Minos CM-005 ER 4 X MINEMNOS +Cakewalk CM-008 ER 4 X J CAKEWALK +Stronghold CM-009 ER 4 X STRNGHLD + + +*********************************** +* Data Age * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Encounter at L-5 DA 1001 U 4 X ENCONTL5 +Warplock DA 1002 C 4 X WARPLOCK +SSSnake DA 1003 C 4 X SSSNAKE +Airlock DA 1004 C 4 X AIRLOCK +Bugs DA 1005 U 4 X P BUGS +Journey Escape 112-006 C 4 X J JRNYESCP +Rock 'n Roll Escape 112-006 R 4 X X1 J JRNYESCP +Bermuda Triangle 112-007 R 4 X J BERMDTRI +Frankenstein's Monster 112-008 R 4 X J FRANKMON +Secret Agent ???? UR ?? + + +X1: This is the same as Journey Escape. + + +*********************************** +* DSD/Camelot * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Tooth Protectors ???? UR 8 E0 X X1 J TOOTHPRO + + +X1: Intresting!!! This cart uses the Parker Bros. 8K bankswitch!!! It's +the only non-PB cart to use this format. + + + +*********************************** +* Dynacom * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Mega Boy ???? ?? 64 F0 X J MEGABOY + + +This is a very intresting cart. It's designed as an educational product! +It was only test-marketed in Brazil, and most of it is in Portugese. It +contains several different learning tools- Math, English, Games, and Music! +This cart goes with a hand-held 2600 called the 'Mega Boy'... it's similar +to a TV Boy except it can accept regular 2600 carts! It runs on batteries +and transmits through the air to a TV in a similar fashion to a TV Boy. All +in all a very cool device. + + + +*********************************** +* Ed Fendermyer * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- +SoundX ???? ER 4 X J SOUNDX +EdTris ???? ER 4 X J EDTRIS + + +*********************************** +* Emag * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +I Want My Mommy GN-010 ER 4 X IWANTMOM +Dishaster GN-020 ER 4 X J DISHASTR +Tanks But No Tanks GN-030 ER 4 X TANKSBUT +Cosmic Corridor GN-040 ER 4 X COSMCORR +Pizza Chef GN-050 ER 4 X J PIZZA +Immies & Aggies GN-060 ER 4 X IMMIES +A Mysterious Thief GN-070 ER ?? +Fire Spinner GN-080 ER 4 X FIRESPIN + + +*********************************** +* Epyx * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Summer Games 8056100250 R 16 F6 X J SUMMERGA +Winter Games 8056100251 R 16 F6 X J WINTERGA +California Games 8056100286 R 16 F6 X J CALIFGMS +Super Cycle ???? ?? + + +*********************************** +* Exus * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Video Jogger ???? ?? 4 X O VIDJOGGR +Video Reflex ???? ?? 4 X O VIDREFLX + +*********************************** +* First Star Software * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Boing! ???? ER 4 X J BOING + +*********************************** +* Froggo * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Karate FG 1001 R 4 X J KARATE +Spiderdroid FG 1002 R 4 X J SPIDROID +Task Force FG 1003 R 4 X J TASKFORC +Cruise Missile FG 1007 R 4 X J CRUSMISL +Sea Hawk FG 1008 R 4 X J SEAHWK_F +Sea Hunt FG 1009 R 4 X J SEA_HUNT + + +*********************************** +* Funvision * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Ocean City Defender ???? ?? 4 X J OCEANCTY +Spider Maze ???? ?? 4 X J SPDRMAZE + + +*********************************** +* HES * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Challenge ???? ?? 8 F8 X J CHALLANG +My Golf 535 ?? 8 F8 X J MYGOLF +Pigs 'n Wolf ???? ?? ?? +Star Warrior ???? ?? ?? + +Go for the Gold Pak (really two carts) + -Winter Games ?? 16 F6 X J WINTERG2 + -Summer Games ?? 16 F6 X J SUMMERG2 + +**** +Special menued multicarts: +**** + +Super Action Pak 223 ?? 16 F6 X J SUPERACT + -Pitfall + -Grand Prix + -Laser Blast + -Barnstorming +Smash Hit Pak 498 ?? 16 F6 X J SMASHHIT + -Frogger + -Stampede + -Seaquest + -Boxing + -Skiing +Hot Action Pak 542 ?? 16 F6 X J HOTPAK + -Ghostbusters + -Plaque Attack + -Tennis +Rad Action Pak 559 ?? 16 F6 X J RADACT + -Kung-Fu Master + -Frostbite + -Freeway +Mega Fun Pak ???? ?? ?? F6 X J MEGAPAK + -Gorf + -Planet Patrol + -Pac-Man + -Skeet Shoot +Sports Action Pak ???? ?? 16 F6 X J SPORTACT + -Enduro + -Ice hockey + -Fishing Derby + -Dragster +Super Hit Pak ???? ?? 16 F6 X J SUPERHIT + -River Raid + -Grand Prix + -Fishing Derby + -Jink + -Checkers +2 Pak Special #1 ???? ?? 16 F6 X J P0 + -Dungeon Master - Venture + -Creature Strike - Demon Attack +2 Pak Special #2 ???? ?? 16 F6 X J P1 + -Star Warrior - Starwars: Empire Strikes Back + -Frogger +2 Pak Special #3 ???? ?? 16 F6 X J P2 + -Wall Defender + -Planet Patrol +2 Pak Special #4 ???? ?? 16 F6 X J P3 + -Space Voyage - Starmaster + -Fire Alert - Fire Fighter +2 Pak Special #5 ???? ?? ?? + -Alien Force + -Hoppy +2 Pak Special #6 ???? ?? ?? + -Cavern Blaster + -City War +2 Pak Special #7 ???? ?? ?? + -Challenge + -Surfing +2 Pak Special #8 ???? ?? ?? + -Dolphin + -Pigs 'n Wolf +2 Pak Special #9 ???? ?? ?? + -Motocross + -Boom Bang + + +Notes about menued carts: These are very intresting! They consist of +several games in seperate banks of a 16K F6 bankswitched ROM. There's a +very slick looking menu that comes up displaying the co's logo (HES), and +to press the fire button. After doing so, the user is given a choice of +what game to play. The choices are actually written out onto the screen in +hi-res text! You highlight the desired game and hit the button. The tech +behind it is pretty simple, yet clever. On startup, the bank is pointed to +the menu system's bank, and then is run just like any other F6 cart. The +games are stored in seperate banks, or the upper 2K of a 4K block with the +lower 2K being the menuing program. When the user selects a game, a small +'stub' of code is written to RAM then executed; this stub is usually +something like this: + +0080: LDA $1FF8 ;change banks +0083: JMP $1000 ;run game + +So that when 0080 is called, the cart is switched to bank #2, and then the +game in said bank will be run. Pretty nifty! Note that since this is +already a bankswitched game, 8K bankswitched games can be run in it, as well +as non-bankswitched games. Check out the game linup on the 'Hot Action Pak'. +It is: + +Ghost Busters 8K +Plaque Attack 4K +Tennis 2K + +Total: 14K + +That gives us the extra 2K for the menuing system. + +On those '2 Pak Special' carts, they are still 16K, but almost half of this +goes to waste; they could've made '3 Pak Specials' to use up most of the +space no problem. I still don't know why they didn't do this. Examining the +ROM shows almost half of it is 'FF'. What a waste! :-) + + +*********************************** +* Homevision * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + + +Robot Fight 1 ER 4 X J ROBOFGHT +War 2000 2 ?? +Gogo? Home Monster 3 ?? +World? Trap? 9 ?? +Asteroid Fire 11 ?? +Sky Alien 12 ?? +Base Attack 13 4 X BASEATTK +Wall Break 14 ?? +Lilly Adventure 17 4 X J LILLY +Col 'N' ???? 4 X J COLN +Cosmic War ???? ?? +Frisco ???? ?? +IQ 180 ???? 4 X J IQ180 +Magic Carpet ???? ?? +Panda Chase ???? 4 X J PANDCHSE +Parachute ???? 4 X PARCHUTE +Plate Mania ???? ?? +Racing Car ???? ?? +Repro Cart 83014 ?? +Tanks War ???? ?? +Teddy Apple ???? ?? +Tennis Topsy ???? ?? +Zoo Fun ???? 4 X J ZOOFUN + + +*********************************** +* Imagic * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Trick Shot IA3000 R 4 X J TRICKSHT +Demon Attack IA3200 C 4 X J DEMONATK +Star Voyager IA3201 C 4 X STARVYGR +Atlantis IA3203 U 4 X J ATLANTIS +Cosmic Ark IA3204 U 4 X COSMCARK +No Escape! IA3312 R 4 X NOESCAPE +Fire Fighter IA3400 R 4 X J FIREFITE +Aerial Ace IA3409 ER ?? +Shootin' Gallery IA3410 ER 4 X J SHOOTIN +Riddle of the Sphinx IA3600 U 4 X J RIDDLE +Dragon Fire IA3611 C 4 X J DRGNFIRE +Fathom O3205 R 8 F8 X J FATHOM +Solar Storm O3206 R 4 X SOLRSTRM +Moonsweeper O3207 ER 8 F8 X J MOONSWEP +Laser Gates O3208 U 4 X J LASRGATE +Quick Step O3211 R 4 X J QUICKSTP +Subterra O3213 R 8 F8 X J SUBTERRA +Wing War EIZ-002-04 ER 8 F8 X J WINGWAR +Cubicolour ???? UR 4 X J CUBICOL +Imagic Selector ???? DM 4 X -- IMAGSLCT + + +*********************************** +* ITT * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Aliens Return ???? ?? 4 X J ALIENRET +Fire Birds ???? ?? 4 X J FIREBIRD +Meteor Defence ???? ?? 4 X J METDEF + +*********************************** +* Konami * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Pooyan 001 ER 4 X POOYAN +Strategy X 010 ER 4 X STRATGYX +Marine Wars 011 ER 4 X MARINWAR + +Note: The part numbers are labelled in binary notation. :-) + +*********************************** +* M-Network * +*********************************** + +[If SC is marked, chip uses an extra RAM, #TMM2009P-25. It's +really a 6116 in disguise] + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Star Strike MT4313 R 4 X J STARSTRK +Adventures of TRON MT4317 U 4 X J ADVNTRON +MOTU: Power of He-Man MT4319 R 16 E7 X J HE_MAN +Burgertime MT4518 R 12 X E7 X J BURGTIME +Kool-Aid man MT4648 U 4 X J KOOLAIDE +SC Football MT5658 C 4 X J SUPRFOOT +Space Attack MT5659 C 4 X J SPACATTK +Armour Ambush MT5661 C 4 X J ARMAMBSH +TRON Deadly Discs MT5662 U 4 X J TRONDEAD +Lock 'n Chase MT5663 C 4 X J LOCKCHSE +Frogs and Flies MT5664 U 4 X J FROGFLYS +SC Baseball MT5665 C 4 X J SUPRBASE +Astroblast MT5666 U 4 X J ASTRBLST +Dark Cavern MT5667 U 4 X J DARKCVRN +International Soccer MT5687 U 4 X J INTRSCCR +Air Raiders MT5861 R 4 X J AIRAIDRS +Bump 'n Jump MT7045 R 8 E7 X J BNJ + + +'SC' in this case refers to extra RAM in the cart. + +Notes: + +All three E7 carts have been read in as 16K. This makes it much easier +to write emulators and build hardware, as there's just one standard +size. The RAM can still be included in any cart; however it has no +effect in Bump 'n Jump or He-Man. Fortunately, it doesn't hinder operation +either, so I chose to just include the extra RAM under the E7 label. + + + +*********************************** +* Milton Bradley * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Survival Run 4362 R 4 X J SURVLRUN +Spitfire Attack 4363 U 4 X J SPITFIRE + + +*********************************** +* Mystique (hehe!) * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Custer's Revenge 1001 ER 4 X J CUSTEREV +Bachelor Party 1002 ER 4 X P BACHELOR +Beat 'em and Eat 'em 1003 ER 4 X P BEATEM +Bachelorette Party 1004 ER 4 X P BACHLRTT +Gigolo 1009 ER 4 X J GIGOLO +Jungle Fever 1011 ER 4 X J JNGLFEVR +Burning Desire ???? ER 4 X J BURNDESR +Cathouse Blues ???? ER 4 X J CATHOUSE +Knight on the Town ???? ER 4 X J KNIGHTWN +Lady in Wading ???? ER ?? +Philly Flasher ???? ER 4 X P PHILLY + + +*********************************** +* Mythicon * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Sourcerer MA-1001 U 4 X J SORCERER +Fire Fly MA-1002 U 4 X J FIREFLY +Star Fox MA-1003 U 4 X J STARFOX + +*********************************** +* Panda Inc. * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Space Canyon 100 R 4 X J SPACANYN +Tank Brigade 101 R 4 X TANKBRIG +Scuba Diver 104 R 4 X J SCUDIV_P +Stuntman 105 R 4 X J STNTMAN +Dice Puzzle 106 R 4 X J DICEPUZL +Sea Hawk 108 R 4 X SEAHWK_P +Exocet 109 R 4 X EXOCET +Harbour Escape 110 R 4 X J HARBRESC + + +*********************************** +* Parker Bros. * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +SW: Jedi Arena PB5000 R 4 X P JEDIAREN +SW: Empire Strikes Back PB5050 C 4 X J STAREMPR +SW: Death Star Battle PB5060 R 8 E0 X J DETHSTAR +SW: Ewok Adventure PB5065 PR ?? ---Proto Exists--- +Gyruss PB5080 R 8 E0 X J GYRUSS +James Bond 007 PB5110 R 8 E0 X J JAMEBOND +Frogger PB5300 C 4 X J FROGGER +Amidar PB5310 U 4 X J AMIDAR +Super Cobra PB5320 U 8 E0 X J SPRCOBRA +Reactor PB5330 U 4 X J REACTOR +Tutankham PB5340 U 8 E0 X J TUTANK +Sky Skipper PB5350 R 4 X SKYSKIPR +Q*Bert PB5360 C 4 X J QBERT_PB +Popeye PB5370 C 8 E0 X J POPEYE +SW: Arcade Game PB5540 ER 8 E0 X J SWARCADE +Q*Bert's Qubes PB5550 ER 8 E0 X J QBRTQUBE +Frogger ][: Threeedeep PB5590 ER 8 E0 X J FROGGER2 +Circus Charlie PB5750 ?? ---No Known Copies Exist--- +Montezuma's Revenge PB5760 ER 8 E0 X J MONTZREV +Mr. Do's Castle PB5820 ER 8 E0 X J DOCASTLE +Spider-Man PB5900 R 4 X J SPIDRMAN +Strawberry Shortcake PB5910 U 4 X J STRWBERY +GI Joe: Cobra Strike PB5920 U 4 X J GIJOE +Action Force PB5920 ER 4 X J ACTIONMN +Lord of the Rings PB???? UR ?? ---No Known Copies Exist--- +McDonald's PB???? UR ?? ---No Known Copies Exist--- + + +*********************************** +* Playaround * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +See Mystique for #201-205 + +/General retreat 206 ER ?? +\Westward Ho 206 ER ?? + + +*********************************** +* Puzzy * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Seesaw ???? ER 4 X J SEESW_TW +Football ???? ER 4 X FB_PIR8 +Earth Attack ???? ER ?? +Puzzled World ???? ER 4 X PUZZL_TW +Chess ???? ER ?? +Boom Bang ???? ER 4 X J BOOMBANG +Pitfall ???? ER ?? +Spider ???? ER ?? +Tennis ???? ER ?? +Pyramid War ???? ER 4 X PYRMDWAR +Bobby is Going Home ???? ER 4 X J BOBBY +Mr. Postman ???? ER 4 X MRPOSTMN +Space Tunnel ???? ER 4 X SPACTUNL +Fancy Car ???? ER ?? +My Way ???? ER ?? +S.O.S. ???? ER ?? +Frogger ???? ER ?? +Fishing ???? ER ?? +Cross Force ???? ER 4 X CROSFRCE +Farmer Dan ???? ER ?? +Dancing Plate ???? ER 4 X DANCPLAT +Volley Ball ???? ER ?? +Little Bear ???? ER ?? + + +*********************************** +* Rainbow Vision * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Pac-Kong 55-003 R 4 X PACKONG +Netmaker 55-006 R ?? +Mafia 55-010 R ?? +Hey! Stop! 55-012 R 4 X J HEY_STOP +Bi! Bi! 55-013 R 4 X BIBI +Catch Time 55-015 R ?? +Boom Bang 55-016 R 4 X BOOMBANG +Mariana 55-017 R ?? +Curtiss 55-019 R ?? +Tuby Bird 55-020 R 4 X TUBYBIRD +Tomboy 55-??? R 4 X J TOMBOY + + +*********************************** +* Salu * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Acid Drop ???? ER 16 F6 X J ACIDDROP +Ghostbusters ][ ???? ER 16 F6 X J GHOSTBS2 +Pick 'N' Pile ???? ER 16 F6 X J PICKPILE + + +*********************************** +* Sancho * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Exocet TEC001 ER ?? +Sea Hawk TEC002 ER ?? +Skin Diver TEC003 ER 4 X SKINDIVR +Nightmare TEC004 ER 4 X NGHTMARE +Dice Puzzle TEC005 ER 4 X DICEPUZL +Forest TEC006 ER 4 X FOREST + + +*********************************** +* Sega * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Tac-Scan 001-01 U 4 X TACSCAN +Sub Scan 002-01 U 4 X SUBSCAN +Thunderground 003-01 R 4 X THUNDGRD +Star Trek: SOS 004-01 R 8 F8 X J STARTREK +Buck Rodgers 005-01 U 8 F8 X J BUCKROG +Congo Bongo 006-01 R 8 F8 X J CONGBONG +Up 'n Down 009-01 ER 8 F8 X J UPNDOWN +Tapper 010-01 ER 8 F8 X J TAPPER +Spy Hunter 011-02 ER 8 F8 X J SPYHUNTR + + +*********************************** +* Selchow & Righter * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Glib ???? UR 4 X GLIB + + +*********************************** +* Simage * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Eli's Ladder ???? UR 4 X ELILADDR + + +*********************************** +* Sparrow * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Music Machine GCG 1001T UR 4 X MUSCMACH + + +*********************************** +* Spectravision * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Gangster Alley SA-201 R 4 X J GANGALLY +Planet Patrol SA-202 R 4 X J PLANTPAT +Cross Force SA-203 R 4 X CROSFRCE +Tape Worm SA-204 R 4 X J TAPEWORM +China Syndrome SA-205 R 4 X CHINASYN +The Challange of... Nexar SA-206 R 4 X NEXAR +Master Builder SA-210 R 4 X J MASTBULD +Galactic Tactic SA-211 R ?? +Mangia SA-212 R 4 X J MANGIA +Gas Hog SA-217 ER 4 X J GASHOG +Bumper Bash SA-218 ER 4 X J BUMPER +Save the Whales SA-??? UR ?? +Cave-In SA-??? UR ?? +Chase the Chuckwagon SA-??? UR 4 X J CHUCKWGN + + +*********************************** +* StarPath * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Supercharger BIOS ROM 330030 -- 2 X STARPATH +Phasor Patrol RA-4000 R 6 AR X J PHASOR +Communist Mutants RA-4001 R 6 AR X COMMIE +Suicide Mission RA-4002 R 6 AR X J SUICIDE +Killer Satellites RA-4103 R 6 AR X J KILLRSAT +Rabbit Transit RA-4104 ER 6 AR X J RABBIT +Frogger RA-4105 ER 6 AR X J FROGGER +Escape from the Mindmaster RA-4200 ER 6*4 AR XXXX J MINDMAS1-4 +Sword of Saros RA-4201 ER 6 AR X SWOSAROS +Excalibur RA-4201 PR ?? +Fireball RA-4300 R 6 AR X FIREBALL +Party Mix RA-4302 ER 6*3 AR XXX PRTYMIX1-3 +Dragonstomper RA-4400 ER 6*3 AR XXX J DRAGON1-3 +Survival Island RA-4401 ER 6*3 AR XXX SURVIVAL1-3 +Sweat! RA-4??? PR 6*2 AR XX SWEAT1,2 +Comm. Mutants Demo ???? DM 6 AR X COMMDEMO +Dragonstomper Demo ???? DM 6 AR X DRAGDEMO +Fireball Demo ???? DM 6 AR X FIREDEMO +Frogger Demo ???? DM 6 AR X FROGDEMO +Killer Satellites Demo ???? DM 6 AR X KILLDEMO +Esc. from Mindmaster Demo ???? DM 6 AR X MINDDEMO +Party Mix Demo ???? DM 6 AR X PRTYDEMO +Rabbit Transit Demo ???? DM 6 AR X RABTDEMO +Suicide Mission Demo ???? DM 6 AR X SUICDEMO + + +*********************************** +* Sunrise * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Quest for Quinta Roo 1603 ER 8 F8 X J QUINTROO +Snowplow ???? ER ?? +Glacier Patrol ???? ER 4 X J GLACIER +Noah and the Ark ???? ?? + + +*********************************** +* Suntek * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Time Race 1 ER ?? +Galactic 2 ER ?? +Pac-Kong 3 ER 4 X PACKONG +Pyramid War 4 ER 4 X PYRMDWAR +Netmaker 6 ER ?? +Bermuda 9 ER 4 X BERMUDA + + +*********************************** +* Technovision * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Adventures of GX-12 ???? ?? +Flipper ???? ?? +Formula I ???? ?? +Jungle Jim ???? ?? +Laser Raid ???? ?? +Moonbase ???? ?? +Motor Mouth ???? ?? +Mouse Highway ???? 4 X J CATMOUSE +Nuts ???? 4 X NUTS +Pharoah's Curse ???? 4 X PHARHCRS +Save Our Ship ???? 4 X J SAVESHIP +Silly Safari ???? ?? +Shoot-Out ???? ?? +Stone Age ???? ?? +Tachion Beam ???? ?? + + +*********************************** +* Tele-Games * +*********************************** + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + + +Bogey Blaster 5861 A030 R 4 X J BOGYBLST +Night Stalker ???? R 4 X J NIGHTSTK +Universal Chaos ???? 4 X J UNIVCHOS +Bump 'n Jump ???? 8 F8 X X1 J BUMPHUMP + + +X1: Intresting... uses F8 instead of E7 bankswitching! + + +*********************************** +* Telesys * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Coco Nuts 1001 R 4 X J COCONUTS +Cosmic Creeps 1002 R 4 X J COSMCREP +Fast Food 1003 R 4 X J FASTFOOD +Ram- It 1004 ER 4 X J RAMIT +Star Gunner 1005 ER 4 X STARGN +Demolition Herby 1006 ER 4 X J DEMOHRBY +Bouncing Baby Monkeys ???? UR ?? +The Impossible Game ???? UR ?? + + +*********************************** +* Tigervision * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +King Kong 7-001 R 4 X J KINGKONG +Jawbreaker 7-002 R 4 X J JAWBREAK +Threshold 7-003 R 4 X J THRSHOLD +River Patrol 7-004 UR 8 3F X J RIVERP +Marauder 7-005 ER 4 X J MARAUDER +Springer 7-006 UR 8 3F X J SPRINGER +Polaris 7-007 ER 8 3F X J POLARIS +Miner 2049'er 7-008 ER 8 3F X J MNR2049R +Intuition 7-009 NR ?? +Scraper Caper 7-010 NR ?? +Miner 2049'er Volume ][ 7-011 ER 8 3F X J MINRVOL2 +Espial 7-012 ER 8 3F X J ESPIAL + +*********************************** +* TNT Games * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +BMX Airmaster 26192 ER 16 F6 X J BMX_TNT + +*********************************** +* 20th Century Fox * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Worm War I 11001 U 4 X J WORMWAR1 +Beany Bopper 11002 R 4 X J BEANYBOP +Fast Eddie 11003 U 4 X FASTEDIE +Deadly Duck 11004 R 4 X J DEADDUCK +Mega Force 11005 R 4 X MEGAFRCE +Alien 11006 R 4 X J ALIEN +Turmoil 11007 U 4 X J TURMOIL +Fantastic Voyage 11008 R 4 X FANTCVOY +Crypts of Chaos 11009 R 4 X CRPTCHOS +M*A*S*H 11011 U 4 X J MASH +Bank Heist 11012 R 4 X BANKHEST +Porky's 11013 R 8 F8 X J PORKYS +Flash Gordon 11015 R 4 X FLASHGRD +Revenge of the BS Tomatoes 11016 R 4 X J REVNGTOM +The Earth Dies Screaming 11020 R 4 X J EARTHDIE +Spacemaster X-7 11022 ER 4 X SPACMAST +Meltdown 11029 ER ?? Need! +Crash Dive 11031 ER 4 X CRSHDIVE +Alligator People ????? PR 4 X J ALIGPEPL + +*********************************** +* U.S. Games * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Space Jockey VC 1001 C 2 X J SPACJOCK +Sneak 'n Peek VC 1002 R 4 X SNEKPEEK +Word Zapper VC 1003 U 4 X J WORDZAPR +Commando Raid VC 1004 R 4 X COMANDRD +"Name This Game" VC 1007 R 4 X X1 J NAMEGAME +Octopus VC 1007 ER 4 X X1 J NAMEGAME +Towering Inferno VC 1009 U 4 X J TOWERINF +M.A.D VC 1012 R 4 X J M_A_D +Gopher VC 2001 R 4 X J GOPHER +Squeeze Box VC 2002 ER 4 X J SQUEEZBX +Eggomania VC 2003 R 4 X P EGGOMANA +Picnic VC 2004 ER 4 X P PICNIC +Piece 'o Cake VC 2005 ER 4 X P PIECECKE +Raft Rider VC 2006 ER 4 X J RAFTRIDR +Entombed VC 2007 ER 4 X J ENTOMBED + + +X1: These two carts are the same except the name. + + +*********************************** +* Universal Gamex * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +X-Man GX-001 UR 4 X J XMAN + + +*********************************** +* Venturevision * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Rescue Terra I VV2001 ER 4 X RESCTER1 +Inner Space ???? + + +*********************************** +* Video Gems * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Missile Control ???? ER 4 X MISLCONT +Mission Survive ???? ER +Steeple Chase ???? ER +Surfer's Paradise ???? ER 4 X SURFPRDS +Treasure Below ???? ER + + +*********************************** +* Wizard Video * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Texas Chainsaw Massacre 008 ER 4 X J TXSCHAIN +Halloween 007 ER 4 X J HALOWEEN + + +*********************************** +* Xonox * +*********************************** + +Note: No repeats are listed (i.e. when the same game is on two double-enders) + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Spike's Peak 99001 ER 8 F8 X J SPIKE_PK +Ghost Manor 99002 ER 8 F8 X J GHOSTMAN +Chuck Norris Super Kicks 99003 ER 8 F8 X J CHUCKICK +Artillery Duel 99004 ER 8 F8 X J ART_DUEL +Robin Hood 99005 ER 8 F8 X J ROBH_P +Sir Lancelot 99006 ER 8 F8 X J SIRL_N +Tomarc the Barbarrian 99007 R ?? +Motocross Racer 99008 R ?? + + +*********************************** +* Zellers * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Time Warp ???? ER 4 X TIMEWARP + +*********************************** +* Zimag * +*********************************** + + +See Emag + + +*********************************** +* Un-marked / Other * +*********************************** + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Dragon Defender TP-605 ER 4 X DRGNDFND +Hole Hunter TP-606 ER 4 X HOLEHUNT +Farmyard Fun TP-617 ER 4 X FARMYARD +16 Games in 1 ???? ER ?? +Brick Kick ???? ER 4 X J BRICKICK +Challange ???? ER 4 X CHALENGE +Clown Down Town ???? ER 4 X CLWNDOWN +Criminal Pursuit ???? ER 4 X CRIMLPUR +Dragon Treasure ???? ER ?? +Frontline ???? ER ?? +Inca Gold ???? ER ?? +laser Volley ???? ER 4 X LASRVOLY +Ski Run ???? ER 4 X SKI_RUN +Lie Low ???? ER ?? +Lost and Found ???? ER ?? +Missile Attack ???? ER ?? +Oops ???? ER ?? +Planet Protector ???? ER ?? +Ski Hunt ???? ER 4 X SKIHUNT +Super-Ferrari ???? ER 4 X SUPFERRI +Tom Boy ???? ER 4 X J TOMBOY +UFO Patrol ???? ER 4 X UFOPATRL +Wolf Fighting ???? ER ?? +Pink Panther / Probe 2000 ???? PR ?? ---One Proto Exists--- + + +********************************************** +* Odd proto's and other intresting things * +********************************************** + + Name / Description SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Air Raid (Not Air Raiders! By Men-A-Vision) 4 X AIR_RAID +Bira Bira (modified Burning Desire) 4 X J BIRABIRA +Challange of Nexar (Changed GFX) 4 X NEXAR-CH +Circus Atari (Uses joysticks) 4 X J CIRCUS-J +Colourbar Generator 4 X -- COLORBAR +Condor (Similar to Condor Attack) 4 X CONDOR +Donkey Kong (Changed GFX) 4 X J DK-CH +Duck Shot (Positively Weird!) 4 X DUCKSHOT +Elk Attack (Mark R. Hahn; 1987) 8 F8 X ELK +Fishing Derby (changed GFX; from 32-in-1) 2 X J FISHN-CH +Freeway (Changed GFX; from the 32-in-1) 2 X J FREWY-CH +Galaga (River Raid ripoff) 4 X J GALAGA +Joust Hack (#1) 8 F8 X J J_HCK1 +Joust Hack (#2) 8 F8 X J J_HCK2 +Joust; Super (modified) 8 F8 X J SJOUST +Marflegr (PAL version of Sea Hawk) 4 X MARFLEGR +MASH w/ subs (Modified GFX) 4 X J MASH_SUB +Missile Command (Changed GFX) 4 X J MC-CH +Pac-Kong (Changed GFX) 4 X J PK-CH +Snail Against Squirrel (Changed GFX) 4 X J SVS-CH +Space Raid (Vaguely Threshold-like) 4 X SPACRAID +Test (Neat little TIA test) 4 X -- TEST +Traffic (Dunno; intresting) 4 X J TRAFFIC +World End (I Believe this is World? Trap?) 4 X J WORLDEND +Galactic (Starsoft title; not a pirate) 4 X J GALACTIC +Air-sea Battle (changed GFX) 2 X J AIRSEA2 +Space Invaders (changed GFX) 4 X J SPACE2 +Magazine Demo 4 X -- MAGDEMO +4-game ROM... uses F6 bankswitch 16 F6 X J GORF_RIP + +*********************************** +* Overflow * +*********************************** + +Used to keep Bankzilla's database program happy (mainly PAL versions and +game revisions. Prevents cluttering up the main listing) + + + Name Part # RA SZ SC BS IM SP CT Filename +--------------------------------------------------------------------------- + +Pal version of Pele's Soc. CX2616 ?? 4 X J CX2616PL +Taiwanese Enduro ???? ?? 4 X J ENDRO_TW +Taiwanese Pele's ???? ?? 4 X J PELE_TW +CCE version of Pitfall ???? ?? 4 X J PITF_CCE +Taiwanese River Raid ???? ?? 4 X J RIVER_TW +Diagnostic Program ???? ?? 4 X -- SALTDIAG +Superman; revision 2 ???? ?? 4 X J SUPRMAN2 +Dunno ???? ?? 2 X -- XX +Pal version of KLAX ???? ?? 16 X F6 X J KLAX +Okie Dokie; Limited ???? ?? 4 X J OKIEDLIM +Robot Tank; F8-fixed ---- -- 8 F8 X J ROBO_FIX +Surf's Up; original ---- -- 8 F8 X J SURFSUP +Dumbo's F. Circ.; PAL ???? PR 8 F8 X J DUMBO_P +Sir Lancelot; NTSC ???? ?? 8 F8 X J SIRL_P +Smurf Rescue; PAL ???? ?? 8 F8 X J SMURF_PL +'Chess'; Taiwanese ???? ?? 2 X J CHESS_TW +Outerspace; Sears ???? ?? 2 X J OUTERSPC +Okie Dokie Proto ---- -- 2 X J LT +Taiwanese Bowling ???? ?? 2 X J BOWLG_TW + +*Not* a ROM image! **** ** ** ** ** X -- EMPTY + +####Data End#### + +*********************************** +* World of Dead ROM Images * +*********************************** + +Yeppers, some ROM images are better off dead. Either they're exact copies, +damaged, or just plain DOA. + +XTACK Same as Z-Tack by Bomb. +POLEPOS Pole Position... bad read. 5 bytes bad. +ASTROSMS Same as M-net's Astroblast + +*********************************** +* Proto's Read * +*********************************** + + +Thanks to many people, I have been able to read some prototype carts in! +However, the data contained on the carts is identical to the production +ones. + +Bermuda Triangle, #371 +Pac-Man, in an Imagic case!! The board has been wave-soldered; said board + is an actual Imagic board, and the chip is a ROM. diff --git a/com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/stella.pdf b/com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/stella.pdf new file mode 100644 index 0000000000000000000000000000000000000000..5a76bc1b4a6aa17ff16a9314a1bb32c727820040 GIT binary patch literal 772501 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 z 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�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 vJTfX#C9`n1%Af}{)s*;+#Vsw_$ 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¬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
9GX$P%=}{_Ynd|5CCHOPO%3-K;_X&Ofe4dkUlab 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

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
PAL TIA colors (104 unique colors)
HUE
L
U
M
0123456789ABCDEF
0-1                
2-3                
4-5                
6-7                
8-9                
A-B                
C-D                
E-F                
+

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
SECAM TIA colors (8 unique colors)
HUE
L
U
M
0123456789ABCDEF
0-1                
2-3                
4-5                
6-7                
8-9                
A-B                
C-D                
E-F                

diff --git a/com.wudsn.ide.asm/icons/brkp_obj.gif b/com.wudsn.ide.asm/icons/brkp_obj.gif new file mode 100644 index 0000000000000000000000000000000000000000..a831fe7278277363fd535031d3957d687752dea9 GIT binary patch literal 197 zcmZ?wbhEHb6krfwIKsg2|IdHD(0Vn$BD19KoTaC;=O1off3|7enWi--X6(GM^3=mE z7oP@C-4`|MQ1Qw$6ZYJgxaazegSR7R9r*wMKLZg!@h1x-1A{Pw4oEG?P6k$|2P%Cj zne#GcG3l&kI;FtF)xhLGLC5pM4YdOdf|^Qf2ZS6PG7MB$94D#hSgd1x!YOXxz_(A+ T)NJ+MsHCY?XY(dIF<1it*a<>% literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/icons/brkpd_obj.gif b/com.wudsn.ide.asm/icons/brkpd_obj.gif new file mode 100644 index 0000000000000000000000000000000000000000..74b26c31a4d412ed57701fe474957bc7e6462036 GIT binary patch literal 139 zcmZ?wbhEHb6krfw*v!E2`1KF}0RR90|DQa0a@)3T?d|PVRaKdpnd#~2Nl8hep`k!2 z1|*>PlZBCiftx`GBnC2rfyJWWr043r7U>6fia8gGG&eMIC@^?ba(13#?9AkJJpMHQ f&c$tA4G95^JS__*Ea1?XBtJLJYi*RVAcHjk@C7j? literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/icons/hardware-apple2-16x16.gif b/com.wudsn.ide.asm/icons/hardware-apple2-16x16.gif new file mode 100644 index 0000000000000000000000000000000000000000..9755078c9b8b424081817e7764103dd6ce1e5ce9 GIT binary patch literal 619 zcmZ?wbhEHb6krfwc*elM!^I;bC8MIGVq#=sV`UQ(6yoFMlaiEDQ&p3bm7{(^P4%dX z;yMKuQ5IGyRvvvG9z7l&B_1AOo;ma8Oqn*NrLASnx;6ZPeEQ#X^gd|ozSh!trKNRS zUG=cC(q;vP6><`d;s*cq^nd8;f78|bqNDd&NAHu4-g|A`ciOsdv~^x+={(ocd8VcP zSX29frq*2zt=k$J7u3{FsL0Nek?NF`Xb_jE6BjEM70DD92;%2;;*psoEm12jk|xZ< z!^6P9@c;jRAUSk_;!hSv1_lQP9gr75al*iUqQSv|xrM2vy}hZKQ$$QyM3jS*ot25H ziMdCZlZ}OW+BDYrEg~Z9%=Rm-xOtfw7BMo5ajsiwFKcSg(8AIt%(hP4%+gH3TyXbd z5f*z3RV8VC6-65cmc7iP%ywMr+FH^YMnVh++P3iNNol`Pli$U9gOkb5Q1gwplng%u zJIh9P7KR&*jQ6{ExOr3#q$!BCh-n7QxM9%b#>ldmN#w@W=GJye1BsX~=^6}-fh{~D f784i_Hu^ELNJBVW`N2A~N*$jG2Im3`_; z#>vY;1}gq!0aEe|IzTqiMgax});Rut~RQAcs8P~2``-2A}~93_iz;O&HQ8!?rUV zS5RSK;8Xd-AmAim{&6MK|NsAihSLHl{$v5#ti+%LG92V52DTLr6FD4oq&g2JK6J2@ z5$SR|D6m3nF6$A7fVEOPTyz2*)RPd*IDd)l{|a!gp4HQVczvhvf+a0#XNTPkmF zyqnCj;;)(GJiSz#u9m}x8;^aFoou|aBytN=sne~8fph$J?|)!nm6zN8Vse#^TJY)b zYV10xx6k_o7sScel+__}fYOPbT8*l6 zmo4a8HCtk7;`&J&mu!j5UAQA?bKm?u8y4?xX>D6{Wc2|LW&Wr|E03b7#RNl{|_XGC{X;#!pOkj$e;sqKPXNZ*bg^2I&!vfw6wQ3H4B=Wo0ysj2=eoC za5QoDmhLDh>X0ho(>DsI!>-O6Jc3xWd#jQ zdlnYfR!&n%RTb&e%!c{~qFe_!&6IQ%_c%Bh8EY}|Ef&20a4&<6h3$qrf>M{RurOO& zTG{b)ZIY8wNaJbcOl1*sdGPE&gKmtT&4(t|#3_wzOcF5xsfrEFYy$rzjTShFEbQl3 cYI&35FkxZ*yln?nmb?&NF?#_ED-(k?00o0|D*ylh literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/icons/hardware-generic-16x16.gif b/com.wudsn.ide.asm/icons/hardware-generic-16x16.gif new file mode 100644 index 0000000000000000000000000000000000000000..b58a71ded14df9c783b309b82edb224c4a12ad89 GIT binary patch literal 947 zcmW-f!HX9}9L1k0rXF$;6G{+_m5`BeeG4XaXfG?jE z=nxV_bIC)3-aJ?!eGih1;6Jb+)TKk>={dV|7-r@R^SJH@1Og82}?MA5XWEc z*##WLDa^I~_RaPI1Ox<-fKh-#6rlu4q8JsZL=|eFCaTeZMl_)XTA~>p=tLKKpeMR9 zfI$pl1V&MDyd>ss8Us`p_-~zgBsPO z7HX+xb*NKa>Y<+M)_?{zq!AjaVHq-I$swma5lrF?C=h|1M4})HqbN#5$taGBsEn$p z5jCSa8lo|pqD8cf=IDsd=!zcEGrD6S24g5j#K;&9Bh0YE5zg?WKh0)BiAdx$BMY)H zi?T$P%;K!b%B;#7Su?A%Ase$PTV%^@&W`NNuI!OLvpWZJFo$wPj?CdS(o8EI=}db& z2-A&;WcD?SGYSPRbdgKAq>EkQN>{mtYr5JEZgi7dxTTxj;ZAqChkLr)10M8{M|h;i zw%82XbbBJ7bMJ`nq<&}H_dQd+PitSl_WJEd4^KZ?Uj6dnw<}B6{`hER^UUJZ zy>~WttsL8bX3tx@+hX&?;?KvI|NHr`qkFIXaB2PGvq!hi>JS&Xb>P-Aq2 zH2LebZTxe2W#d7^2d@D9)wnyrqD<>o<#K+4gB`HN(N@~iqDK%9! zbLP!iZLH42#naN(vS!_yHOA`y|NjT-A_ge_WMO1r5Ms~)DF)fez}m7vg)t>_(xV6s zfyFl7jS|jlwR{Q%9Ac$yCpk4Lu=cq(FwbCc;9_Vjxy3PogPGy6z}%pw4J8KrZZ$SYYw<9MTUv2?=e?ufEi-XE^ b4pwdsYfd54aGOkqRScT8b{p)J6d9}mj-)>F literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/icons/hardware-test-16x16.gif b/com.wudsn.ide.asm/icons/hardware-test-16x16.gif new file mode 100644 index 0000000000000000000000000000000000000000..a5bb5a8c71418a048dd3520175538d8efa04110b GIT binary patch literal 961 zcmW-fJ%}Gf6vQVnTw*xQrStHwQS#5gg)34%47X2Q513*~c84fe0v&v#ltI!YC0%Q8J37A}XUsR7K6Gj)rKA z7SR+fqd7XFGkQc<^o;Hph`|^ULoqUj!w54RVTCh15lrF~LXeP?hzeAw5-L(j6{|v( zs-Y^?RJ9t^s1|BcOEs%Qo$8@3^;EY8G^i08(nt-#lIHfD=#%9h!j9od;ZvMYOLcMjxWj>w@LnZs$MnU1v5nf7iF zrW+H<>}wWh6bfAE5-xH{7rVlhuHh=zbhR7Y=oW5rOEp`vGw4Ew|{#5+s)%I{rJJr ztruU|{`1E7n~OWItzKWf7b|ZaJF|Rx>D`@I-udd-b12vT{%m#W{PND)r$1a@T^!$7 q$1~sDvwiuEb02R0_58DZ{L;toE!{qR_QbWVTb~?!>hSU*JpDg~V7NK} literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/icons/help-16x16.gif b/com.wudsn.ide.asm/icons/help-16x16.gif new file mode 100644 index 0000000000000000000000000000000000000000..86550fe9f4505c88287a70add9bb89bd24d0e8b2 GIT binary patch literal 618 zcmZ?wbhEHb6krfwc*el+@8AEApT0hL{PNG=|DQg8&8?ZTbTI7l9pM;Aadrs z)q57K*|TWvo`q|6FJ8NQ^PUs8?mzqR`TO%%?+Tk|_Rd*9f6bnGt9LJ4yJyG2Gh6na zIC$#nnJf3^uGuqp_3o3G@14JP@4~gar?1?7^77r^fB#QixqIW`%j@@_J$Ui%-RJLL zfB*mW_y3CzU+=wm_wv*CFTekP|MUOtr?21s{D1T5>(9Uc|NQ&^_0RtwfB%2~`~Ua9 z|KI-p|Mlt$qs9JUr40 zer^8R{6g$3jO;?JY^{odOr}OcLMjYQg2K&y{Gz4;9DIBnTBf4>{r>WDZb}@IZp?gK zZf=_LO;Uz7UM40!%DjADULsOWJT9)THZ~SqTspU%xSKd#Jf2(Kvv7Ux^4MQO*U{0_ zm_ydn^NX%TlZN`|rY9DO$tNZ$I{GaTc=F^>ud=(9!h?d86BDJy@@zAc9v$nG=cswZ nP;}U3vXigQj)sei*?Gjp{G^ILEplmMQH{%JIT9S8z+epk+|z~IlI!vF*zv34Nc0LBeqssYGrXgB}_cMdE&^&ypk zA%lS-i-94ZfuWFrp}NGszRa(+!l$*;r;UN3n}MOH#Fg~zW@356A=CS@a5OXFTX#2 z{rz+G>Yr=Y{M@(i*Z%##J3D{Rp8fmTvp-o`f7Y-6bNTXLTid@rK7XU4{`U0zUBCYC zh7JE}YW~fe_y7NYpkGJ9hzf!KzdtNku%w}(0TlQD8Nf*ZNPt*Cz){ICpF_rDLxQ6K zhp?2$3a7R%K_#aG1IDC>?P9tiQzRS{TRHjJm@Nu6Bs4Oy8rzgb>~L)Fle3L^bz_48 zGoOr0-<=~96BXN}ohSWq__X+VYpZC6iQ%Q@&It;lVs0FVTG~Qa3yRqZI6ZFVUav2g RQR}cx=VogLCmRccH2~uS#sL5T literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/icons/help-topic-small.gif b/com.wudsn.ide.asm/icons/help-topic-small.gif new file mode 100644 index 0000000000000000000000000000000000000000..1cfe6f3a9c2b945144d8968edc67e99c5cb000d4 GIT binary patch literal 910 zcmZ?wbhEHb6ky-l|Gy!F4g(N?#M-}n`}6V3?^o}C{`~c)ZPvb?`G@Ym z`0?n~kEd^bzI^{{^0H&I)|_6n`TYGC-=Dqx@$&u8x1WC9fBECl>mMJ#{{Hag*Y}@) z?!Wy0_S4TV-+zDj`uqFO-`{@zt}gMfFY{}y^y#Vb?5Xi;t?-%C;xeVdeR6}xs&0qH z9nN!FTpyh1-!j2=L!aG-KKs?(4p%m1T-lU)Yj4ffO_~4y{|7m56pV%dEkhtYscOrX zEgabl$s95s8x#(Ta0pA;SS)zRAg(FK6%&}~%*v~uuwp_4Q?omRM%0t*uliUlR4-H2S@pCkBl`GWDcS`bX zRg{^br?JjZ^EfZtK~cuDa?F=R7#|8S95d!UtSECvMgEI5+kXp&|27OS?1h#&*zC)* zz8x;{-<|ti0MGwG&i{ej|3f(5`-uGx7W^MB^gogRf0FROG>PM$dZ$B_4@Bu-39x@2 zXZR>l|9Of2ybzbwPEKpx-FJF8?ez3q6&kubETAbPY)W#}mkmBr?z|OjXD%*xEk z-DS}DAjM0dNuxqWz#=h0NI`2J&lDvUZgEqN4Ld9!KW*-8_RZPCA;@^vu}MFvry^zo O2fOkVV__Z+25SJoAm3*I literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/icons/instruction-type-illegal-opcode-16x16.gif b/com.wudsn.ide.asm/icons/instruction-type-illegal-opcode-16x16.gif new file mode 100644 index 0000000000000000000000000000000000000000..da0e61cab602ce3e270982cb1c3496e6fec7355c GIT binary patch literal 928 zcmZ?wbhEHb6krfw_|CxapW*-S_V&v~MeUxRkE^QQ)YsoBFaO%oGBYUX|NQy?Cr|!A zefs~tzSFt6|GT$`kAywG50LG9DWi9Bk&`wyT&> zuz-PyQBEQw;lTq&Mn)MC1%tu{$4(v@n;SP67@3(^q%8s-JaA}a;$`EB2oPjoWa1Vv z==h05AVXNk%w1VGsZi0J8u9Dk>^AHa1^hUuS1$aBy&UcX$5)|A2sihlhuYi;I$y zl9!j4prD|vtgN-QwaUuMA^8Le000I6EC2ui01yBW000Pm01XHnNU)&6g9r;EK#*|Y zfC35-5D-A%V1fY(791!r5r9F73>F~RC?EiX0t5vtXs{r`fdc{n0B}&zK!E}d8Zd~6 zAm)PuH4PkiptEI9nF~-3Oz^+H$+MR literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/icons/instruction-type-pseudo-opcode-16x16.gif b/com.wudsn.ide.asm/icons/instruction-type-pseudo-opcode-16x16.gif new file mode 100644 index 0000000000000000000000000000000000000000..92d2f6f7f31055cdc49c3775456d678e6cfd3296 GIT binary patch literal 928 zcmZ?wbhEHb6krfw_|CxapW%P|z4oHrMV=L&RVS*P0kc!Dt8!qYzO1$-)Tojt+BcqInfM-*wfPz92l$t>OMwK literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/icons/outline-binary-include-16x16.gif b/com.wudsn.ide.asm/icons/outline-binary-include-16x16.gif new file mode 100644 index 0000000000000000000000000000000000000000..b7db916868b1db24ba69a4aff410afa6cf07b54b GIT binary patch literal 974 zcmW-fF=*ID7{#9@l_r8_2$Z5CRSC-36#X13Qb7#WKm`$xba04Z-XIE6T`U$ZM4H7- zNcc8~4st?45EMi)XbZMa=yE8cqfi_oIJAhv^Pk*s+}#KFzVG+miz{bNJ@dvq=J5(} z{NI_ofD?EE)7pRA+CLxw0tpB(3Qz)tD2XBzqXIQhiJGWFH5$+YjcAD`G@}DO(21Vt zLN^950)rTdAq-D{!vPacSnzb2v?IVk1~k>8AWB4Gl#HS%j*6%el~FUQqB5=>-56Ok;)5?Pof zvnY$RB5P!2*37D`&W3D}joC7rvN=1lM|NhO98FI*! zQBp9G=@eb@TZ{ z8@mthy|DQ9FW+67ePHj#*Jszh-8y_<|LEMw-=5o9czkK$?v=HlE-sutvwUso)u%7~ zw|#Zt*7@T;(v*XKU{_}A5!AKQ8G z+@-aLo}9UUd-t7h{yw{M?E3ujjr)(D{Nu`7J6Dg)tzLQR*r6Ym=eB1z7S}(#bMfcP LU!4E5;o1KI)*r}n literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/icons/outline-binary-output-16x16.gif b/com.wudsn.ide.asm/icons/outline-binary-output-16x16.gif new file mode 100644 index 0000000000000000000000000000000000000000..a61ee2d33460bbd054451384b1344bd76558d600 GIT binary patch literal 968 zcmW-fJ%|`Y6oe;t2^s~BsA!>((Frq6*hv1f}KBk zSmlU?i=rTk7D|K|$uU7ewvYr2XknqzB5;VsxOZDDyZd10y>I4$2UqUC=lNaOg~x!U zf46S~_TqNTwfXkO<^cf^NI-y5fD$M~NfePV__< zx-oze7{o{nVR-%x2TV9&!Sgb;BfvlgG;2{1C897&Mo|<;MbwDOs2NpJ9SzYU8lz=2 zMRRmSkLZk^(G}e>5F=tRM#fMKhY^l2!x>h1_A$o}f+<=h{yu*4-No0JuXy^=-yb@*aPG{u-E`jAPsfUi$Ci1?=gGU;v{qh#?GP1QVErNlalHGg!bXEMf`ESlNFg zg%A;9WH%eT3M{DLJ*$RlRgKkD&DByJ)vG$ItGcVF25M9d)=&-CNKMqNnyjgsu9;e> zRkc`4wOlI|RI0*?Dz0oEd+j1fi14Px2DV}&o7l`2cCZ&a*~M=5aDbyY$RQ4MgcF>_ zNltN^GhE;*E^>*>Tw%ZzLq?35Ex}g21PT(o6;Xp)sZmX8R*O2+OP%Ucw|X?7Q5w{c zhBcxI&C;Z%G_4seXq6VVq-CurP)ea9#mZKHn_UJC8Qx~N!L8irCO5mq9q#2$ce&d= z9`Gm+ddR~b@q}l2(o>%Hj2FDhi(c}wR~$Iy(2-*&hahY>wj`U^9_}bKq*WTzl;*Ug zBfZj@u5_m-0~wXU3}rYYnaHe6W-8N}$wF3TF-uv_I)Lq8F+=9et!D(E6*R{#{Y*PqTv7l literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/icons/outline-default-16x16.gif b/com.wudsn.ide.asm/icons/outline-default-16x16.gif new file mode 100644 index 0000000000000000000000000000000000000000..7d24707ee82f54aa9fb10d1d9050013cbf161a7a GIT binary patch literal 193 zcmV;y06zamNk%w1VGsZi0K@VRxXubL!4|)qjO}gg>klxZ?TGXw~#-V zU_Y2&N}FX?r*L1YbYiM-aj|xBv2}#Mgo3?-guaA=wSS1Yfrz+)iMWB7#*ml2h^x<; ztIwFU(w+bR{{R30A^8LW0015UEC2ui01yBW000F(peK%GX`X1Rt}L1aL$Vf5mpMgx vG+WO#2NYmJDM}^)l;8n@L?90V%CN9pFcyU&MPO(u48jTlL$uClRtNw)MiWcq literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/icons/outline-definition-section-16x16.gif b/com.wudsn.ide.asm/icons/outline-definition-section-16x16.gif new file mode 100644 index 0000000000000000000000000000000000000000..7d24707ee82f54aa9fb10d1d9050013cbf161a7a GIT binary patch literal 193 zcmV;y06zamNk%w1VGsZi0K@VRxXubL!4|)qjO}gg>klxZ?TGXw~#-V zU_Y2&N}FX?r*L1YbYiM-aj|xBv2}#Mgo3?-guaA=wSS1Yfrz+)iMWB7#*ml2h^x<; ztIwFU(w+bR{{R30A^8LW0015UEC2ui01yBW000F(peK%GX`X1Rt}L1aL$Vf5mpMgx vG+WO#2NYmJDM}^)l;8n@L?90V%CN9pFcyU&MPO(u48jTlL$uClRtNw)MiWcq literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/icons/outline-diamond.gif b/com.wudsn.ide.asm/icons/outline-diamond.gif new file mode 100644 index 0000000000000000000000000000000000000000..36b7561ba617cb71498227f61204fba3b4470375 GIT binary patch literal 879 zcmZ?wbhEHb6krfw_|Cxa|I>#5FDL$gHF5i7)qQg`cF$1XJyZSvvl;8!vO0cpRG!z_UWK(dIlSr6wps`<3a!!v&fMW}% lL86;SgFsWuY;!@4l7`QY&Ro1ICOi}9X2buY*KRBq2l_# zcJ^`^yH&CdtK{t0DLDS?S@Cb$hJQIz{{R2aFbYOPU?_)x;!hSv1_luZ9grtLd7^zZ zM-1io9CbUTA;1yFpvfWQv0=f%W)5L3l?4+N7#Vn^?N}HB6B!wqnI#-{JYZnv;&GCb zNSJVdv7gCRuEQh1v4vAa?L^N621OQ5W>tlN7Y)P^q%YmN< zChVSa;{OSTQ7{?;!!ZODf3h%w{G|gTKzV|lBY}aDL&jsnf`iQ*!dfvN1_v5h*_kv# z3LG36y0`?`A`T=dG_wlvav4krZ1iMi;!`r1;E>S7!X_Xi67Ybbf!Tt4iHSr)Lt`Tw dTZ@TDf9J5m4u literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/icons/outline-implementation-section-16x16.gif b/com.wudsn.ide.asm/icons/outline-implementation-section-16x16.gif new file mode 100644 index 0000000000000000000000000000000000000000..49ccefda86facb6803cf705238899e17744770a8 GIT binary patch literal 1660 zcmZ?wbhEHb6krfw_|CxaJR$LfrojmVlM|YTCuDU#`Nn+ojrm&Ea6(e+`Q+}C&Ne6A z?M~WReVf$xJR#w0P0icp=KufyGmL`K5EvFAp!k!8k%2*wK?md;P@ZTX%@M;QVn;ne zW(aUZGAMJ%cx+g3u$eg1lb);hTDFLZjLs&~-@&N7aU#yw3L zj6O*!7R)g$izT`gnhO|N#M)YoHf)w*<7u8^I3bd&xuGoJf(!$5!&GOzd?f~J0FXsH AVE_OC literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/icons/outline-local-section-16x16.gif b/com.wudsn.ide.asm/icons/outline-local-section-16x16.gif new file mode 100644 index 0000000000000000000000000000000000000000..86df7aae8bef0a125ebebc09da075baa0b1d8ab3 GIT binary patch literal 888 zcmZ?wbhEHb6krfw_|CwP@Vs5~grC8QC?F|&!q)dwj_=3pvakJ;CrrDaZ?!pDYST7`x2*}!yKBBKYJR)q|Ns9CqhK@yhD8YIfP4YU3k)2w49Xlb9vc=MY~~Qwijhcc zJlxLDua)7jF^QR#T~fwCV4(veGrL4e%L4_bhRFghQYjZ6IJU4GNUzY*NJu{3&(0#l i<4~l)z`@8TySOWmgiahxM literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/icons/outline-macro-definition-section-16x16.gif b/com.wudsn.ide.asm/icons/outline-macro-definition-section-16x16.gif new file mode 100644 index 0000000000000000000000000000000000000000..05f3a129776e56c038312fd2ab0b8c982d8426b6 GIT binary patch literal 887 zcmZ?wbhEHb6krfw_|CwPkoa8F@PvW!2_Pw}bHX>~lW+9Lvih%*S|=uVKX1KD* z*6QTMzHbQ$&uePFHaEZh|NlS3C>RZaVG#m4AYXv;0s}`3gEEJV$A$$5n>mEFVk8n9 z54ZF4Yh^fWOk!qbmy|IOSm?mQ%r24A@*qKhfkTPcDC0xIgJw2OZmt_QK0an-WM`hx h!_c7K!obd{BH?Ai*x1Z5Nh|c!hLx9>F)}h(0|2gJJaPa4 literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/icons/outline-pages-section-16x16.gif b/com.wudsn.ide.asm/icons/outline-pages-section-16x16.gif new file mode 100644 index 0000000000000000000000000000000000000000..53ec991991c9f43a0767642f8fc3805b4a593c65 GIT binary patch literal 887 zcmZ?wbhEHb6krfw_|Cxa{93|^MVcqp7=Xxb*-zVjKW_8=dbaFDtK{Y9-MdfPY(Hst z@}$+LZ+*{iC47BZ^Y&5m|Ns9PM!{$Z42uxZ0r>)y7Z^BV7?e3=JT@#i*vuiU6(f<@ zc(|RPUn|35V-hnfyQGYP!G!~iY?6u=9SjXBEECuyO)3mNI5JORWzFdcSkU0k$j;oM h!pXqU!obd{BH?Ai*x1Z5Nh|c!hLx9>F)}h(0|0MHK^Fi3 literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/icons/outline-procedure-definition-section-16x16.gif b/com.wudsn.ide.asm/icons/outline-procedure-definition-section-16x16.gif new file mode 100644 index 0000000000000000000000000000000000000000..cbb53ccb37ca8d68e239a50cea01addcf2334355 GIT binary patch literal 887 zcmZ?wbhEHb6krfw_|Cv!fA)jG-cw@hPXS4eQ=4@DKLw(&|LXTE6RyLem-9 zX=e)0|F=K#A?C*a)JtFf|NqZ03PwXUHj z!|nY1S{V)-lbBiAC1o@WHXLANlT@_mU}#WbnZPD#Qep7Hk$DO$YfewVf@h43?93f1 goD2*t4D6gL5?&^Zjm;dBv_em9Sb2FFBO`+~015^_Z~y=R literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/icons/outline-repeat-section-16x16.gif b/com.wudsn.ide.asm/icons/outline-repeat-section-16x16.gif new file mode 100644 index 0000000000000000000000000000000000000000..572f08fc35883a9f33a370440ace221f93feacc0 GIT binary patch literal 883 zcmZ?wbhEHb6krfw_|CwvE8%&o=7~Oo6F@Rw_QV|DPqTeLZZG?qC3)g;_w&s*C%4<3 zTyAypLEpE%3D3{hd_B|r_W%F?45MH)1cpTj=zx3y$_oq}5e&*4G9DWi9Bk$g){2ox zY&_h~&##r?urZ05m0eQCz~I6GMm9-Biw=ec6_yEXk|q@f9~_ydu(IZeL>yS$(9GyE d$wcyjLL-YqvzkJIK;y;5?vh%ut4tIatO41bKy&~A literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/icons/outline-sort.gif b/com.wudsn.ide.asm/icons/outline-sort.gif new file mode 100644 index 0000000000000000000000000000000000000000..6311cc00f80dadd6f5082e4381dc2bdaa28e6b66 GIT binary patch literal 153 zcmZ?wbhEHb6krfw*v!Dt=Qd?o<*NL$mekrQF|AAe`uB!(Z}yFfm2Rtco!n8ea?$_) z{~3^g;!hSv1_o{h9grBv3N_ARJ`qXtn-g`OJ{8w(; v#98&Xg*(Ax3QtReYl4Pg43pQg1got&3nw){NJ!P%?BU7qTUxJOkii-NY)mvJ literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/icons/outline-source-include-16x16.gif b/com.wudsn.ide.asm/icons/outline-source-include-16x16.gif new file mode 100644 index 0000000000000000000000000000000000000000..ea94702caff9d46b70fc7a6b341b6ada7bf0d612 GIT binary patch literal 185 zcmZ?wbhEHb6krfwIKsfNdh?kryDu(Ub7IN5lWVq~UAOJ*)P)CTEvbMZ)V_pB& z_GvqN=j@%k@@RhZ+RC2IO_R6hHm_lHtI|p7&hB6O|Nnmmyg>0M3nK%AD1#12706Bo zR4q+{o1rrn)8F-}aSQr8m85x}>of%xx-P3dKSQs22%nvk)YWMFz35v|NZ_mBx=^Y!c8E zZk7BA!9t?b$^Hk{mWn8h`@1b3`}V=kyqS6b#(QtR^~ott@j3AFKS$4CpR0#*c>C&b zK>-8=7zi5B3XNz&Gg{DrUg$&@y3vCHjKUy>FpLpQU=}7Zg=x%S0jscxB`jlQ|BVzv zM2L~yZ0stqpn~_T8md(_R#P=sOLbJQ>a4EnuAUmGQ8idYHC!V#QL}2YrfRxoYN1xu zVlCBjtyEB{3M;C(vU%*aiy$Gwn-&|`ij8byGh5ifUhHHSyV=75j^ZGPILr}Fa26*y z#c9rPfvdR4B`$M?0aFYaF=n;|Tk#SoNbpuf4Qi!EHK|!G>QFCrs!QGK(SSy2P(vEl zh$b{klbX`BX0)JHTGW!3wW2^Ng^Cm_Tm5Zz88l>go8bnxa-*Bv>=t*pmpk3%ZufY= zqde##4|~KDp5;kTdD=5x@G38Q$;)1G;FLp0j-4EWu-({_Y+ie~qtK96X-rd^(~^$# zN@u##ot_M2R0cDY;f!P=voe{fOlKwwS(U{sWjX5rwu83a-jeUVFTgKSe^HJ*{yur> z{LzCacaJZh-9A74_@|o>A6)tJuaocJz4OBJ*MEEM@vFbwy?^%SH@D9IcyjZVckcc7 w?a7ZfKKl3iymox)-W%WD|N7z6^WQK1`r)Tve0KEQ2e1G9!}*KP9^vi(0lNjC?*IS* literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/icons/sample.gif b/com.wudsn.ide.asm/icons/sample.gif new file mode 100644 index 0000000000000000000000000000000000000000..34fb3c9d8cb7d489681b7f7aee4bdcd7eaf53610 GIT binary patch literal 983 zcmZ?wbhEHb6krfw_|CxKYUg-n!?izO{@9*?jxd%4aX0yzy`dymabz zw#(eg=y~&N&n)dZv2xzduG}5lraiApo3(c4*{Ylg5#|$JO_EEZ<^|a2`Z*=9ns7DV zy=TR&gYw*7f%auV?ip3tvjRPmcdoho{K?x$_vR?C#t5&<;~V}S*>OMCr>h}%%bLZ9 zmo3`hYEwTICo-TTCZwgTsC&VjZRgJ1eE#fBa^%9R zmmfWS@;bnyJ27HWY}kxYzv(Hl>yu;FCPlAEh+34Muq-8Rb6C)<8qA3{r2e5 z`$vyngh#H=FWlqqvnapfc5%(!sQ4v?r7J61-&eJNEN^;KTK}T7{#i-gJh%G*9vcYdwv_*~xdw!Gz4Va?T!sXyyF@8?w<>X`X=#j%uHV4GRvj@+tE@ zQ%F!a)GKcn^~8abN>4la1UNXVL;{ZWi)lEwyeatDu%Lr6;aASiLrXXW zQm#;zxe*_?T?r5f4=?n>;B6hk6!=y`1SXPFTcM3{B!^1_qU&Ze)<0U!`I*6 zfBydV^LKTLe|?!>Yo$+5jb~4dS8IjOoEDcU4epa0JXUo(Ebefg)8hKzME{lvwj28F zHuTxA?smAcDdWnf%v*bFu5QZw|NlP&(LnJh3nK%AA%hM`2gpwhY!L?{bX;|~`j1~p zuq;?Q;dp3j@Wnj|S{Z9vA~hIRgm<%=vZ_ttX%#YXbQN#g?$9u2!Q%D@2mNh40^~W0 jxJvjG0u0%X~ErwEF%$p*;Go-@bYk--`OMEH@B literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/plugin.properties b/com.wudsn.ide.asm/plugin.properties new file mode 100644 index 00000000..959f4286 --- /dev/null +++ b/com.wudsn.ide.asm/plugin.properties @@ -0,0 +1,52 @@ +com.wudsn.ide.asm.Hardware.GENERIC=Generic +com.wudsn.ide.asm.Hardware.APPLE2=Apple 2 +com.wudsn.ide.asm.Hardware.ATARI2600=Atari 2600 +com.wudsn.ide.asm.Hardware.ATARI7800=Atari 7800 +com.wudsn.ide.asm.Hardware.ATARI8BIT=Atari 8-bit +com.wudsn.ide.asm.Hardware.C64=C64 +com.wudsn.ide.asm.Hardware.NES=NES +com.wudsn.ide.asm.Hardware.TEST=Test + +com.wudsn.ide.asm.CPU.MOS6502=6502 +com.wudsn.ide.asm.CPU.MOS6502_ILLEGAL=6502 with illegal opcodes +com.wudsn.ide.asm.CPU.MOS65C02=65C02 +com.wudsn.ide.asm.CPU.MOS6502_DTV=6502 DTV +com.wudsn.ide.asm.CPU.MOS65816=65816 + +com.wudsn.ide.asm.compiler.AssemblerSourceFile.name=Assembler Source File + +com.wudsn.ide.asm.editor.AssemblerEditorAssemblerMenu.label=Assembler +com.wudsn.ide.asm.editor.AssemblerEditorAssemblerMenu.mnemonic=A + +com.wudsn.ide.asm.editor.AssemblerEditorAssemblerToolbar.label=Assembler + +com.wudsn.ide.asm.editor.AssemblerEditorCommands.name=Assembler Editor Commands +com.wudsn.ide.asm.editor.AssemblerEditorOpenSourceFolderCommand.name=Open Source Folder +com.wudsn.ide.asm.editor.AssemblerEditorOpenSourceFolderCommand.mnemonic=S +com.wudsn.ide.asm.editor.AssemblerEditorOpenOutputFolderCommand.name=Open Output Folder +com.wudsn.ide.asm.editor.AssemblerEditorOpenOutputFolderCommand.mnemonic=O +com.wudsn.ide.asm.editor.AssemblerEditorCompileCommand.name=Compile +com.wudsn.ide.asm.editor.AssemblerEditorCompileCommand.mnemonic=C +com.wudsn.ide.asm.editor.AssemblerEditorCompileAndRunCommand.name=Compile and Run +com.wudsn.ide.asm.editor.AssemblerEditorCompileAndRunWithCommand.name=Compile and Run with... +com.wudsn.ide.asm.editor.AssemblerEditorCompileAndRunCommand.mnemonic=R +com.wudsn.ide.asm.editor.AssemblerEditorCompilerHelpCommand.name=Compiler Help +com.wudsn.ide.asm.editor.AssemblerEditorCompilerHelpCommand.mnemonic=H +com.wudsn.ide.asm.editor.AssemblerEditorToggleCommentCommand.name=Toggle Comment +com.wudsn.ide.asm.editor.AssemblerEditorToggleBreakpointCommand.name=Toggle Breakpoint +com.wudsn.ide.asm.editor.AssemblerEditorOpenDeclarationCommand.name=Open Declaration +com.wudsn.ide.asm.editor.AssemblerEditorEnableDisableBreakpointCommand.name=Enable/Disable Breakpoint + +com.wudsn.ide.asm.editor.AssemblerHyperlinkDetector.name=Hyperlink Detector +com.wudsn.ide.asm.editor.AssemblerHyperlinkDetectorEditorTarget.name=Assembler Editor + +com.wudsn.ide.asm.editor.AssemblerBreakpoint.name=Breakpoint + +com.wudsn.ide.asm.editor.CompilerSymbolsView.name=Symbols + +# Preferences +com.wudsn.ide.asm.preferences.AssemblerPreferences.name=Assembler Preferences +com.wudsn.ide.asm.preferences.AssemblerPreferencesPage.name=Assembler + +com.wudsn.ide.asm.runner.DefaultApplication.name=Operating System Default Application +com.wudsn.ide.asm.runner.UserDefinedApplication.name=User Defined Application diff --git a/com.wudsn.ide.asm/plugin.xml b/com.wudsn.ide.asm/plugin.xml new file mode 100644 index 00000000..46fb9ff9 --- /dev/null +++ b/com.wudsn.ide.asm/plugin.xml @@ -0,0 +1,425 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + label="%com.wudsn.ide.asm.editor.AssemblerEditorAssemblerMenu.label" + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/com.wudsn.ide.asm/plugin_de_DE.properties b/com.wudsn.ide.asm/plugin_de_DE.properties new file mode 100644 index 00000000..6c148013 --- /dev/null +++ b/com.wudsn.ide.asm/plugin_de_DE.properties @@ -0,0 +1,52 @@ +com.wudsn.ide.asm.Hardware.GENERIC=Generisch +com.wudsn.ide.asm.Hardware.APPLE2=Apple 2 +com.wudsn.ide.asm.Hardware.ATARI2600=Atari 2600 +com.wudsn.ide.asm.Hardware.ATARI7800=Atari 7800 +com.wudsn.ide.asm.Hardware.ATARI8BIT=Atari 8-bit +com.wudsn.ide.asm.Hardware.C64=C64 +com.wudsn.ide.asm.Hardware.NES=NES +com.wudsn.ide.asm.Hardware.TEST=Test + +com.wudsn.ide.asm.CPU.MOS6502=6502 +com.wudsn.ide.asm.CPU.MOS6502_ILLEGAL=6502 mit illegalen Opcodes +com.wudsn.ide.asm.CPU.MOS65C02=65C02 +com.wudsn.ide.asm.CPU.MOS6502_DTV=6502 DTV +com.wudsn.ide.asm.CPU.MOS65816=65816 + +com.wudsn.ide.asm.compiler.AssemblerSourceFile.name=Assembler Quell-Datei + +com.wudsn.ide.asm.editor.AssemblerEditorAssemblerMenu.label=Assembler +com.wudsn.ide.asm.editor.AssemblerEditorAssemblerMenu.mnemonic=A + +com.wudsn.ide.asm.editor.AssemblerEditorAssemblerToolbar.label=Assembler + +com.wudsn.ide.asm.editor.AssemblerEditorCommands.name=Assembler Editor Befehle +com.wudsn.ide.asm.editor.AssemblerEditorOpenSourceFolderCommand.name=Quell-Ordner öffnen +com.wudsn.ide.asm.editor.AssemblerEditorOpenSourceFolderCommand.mnemonic=Q +com.wudsn.ide.asm.editor.AssemblerEditorOpenOutputFolderCommand.name=Ausgabe-Ordner öffnen +com.wudsn.ide.asm.editor.AssemblerEditorOpenOutputFolderCommand.mnemonic=A +com.wudsn.ide.asm.editor.AssemblerEditorCompileCommand.name=Kompilieren +com.wudsn.ide.asm.editor.AssemblerEditorCompileCommand.mnemonic=K +com.wudsn.ide.asm.editor.AssemblerEditorCompileAndRunCommand.name=Kompilieren und Ausführen +com.wudsn.ide.asm.editor.AssemblerEditorCompileAndRunWithCommand.name=Kompilieren und Ausführen mit ... +com.wudsn.ide.asm.editor.AssemblerEditorCompileAndRunCommand.mnemonic=A +com.wudsn.ide.asm.editor.AssemblerEditorCompilerHelpCommand.name=Kompiler Hilfe +com.wudsn.ide.asm.editor.AssemblerEditorCompilerHelpCommand.mnemonic=H +com.wudsn.ide.asm.editor.AssemblerEditorToggleCommentCommand.name=Kommentar umschalten +com.wudsn.ide.asm.editor.AssemblerEditorToggleBreakpointCommand.name=Breakpoint umschalten +com.wudsn.ide.asm.editor.AssemblerEditorOpenDeclarationCommand.name=Deklaration öffnen +com.wudsn.ide.asm.editor.AssemblerEditorEnableDisableBreakpointCommand.name=Breakpoint ein/ausschalten + +com.wudsn.ide.asm.editor.AssemblerHyperlinkDetector.name=Hyperlink Detector +com.wudsn.ide.asm.editor.AssemblerHyperlinkDetectorEditorTarget.name=Assembler Editor + +com.wudsn.ide.asm.editor.AssemblerBreakpoint.name=Unterbrechungspunkt + +com.wudsn.ide.asm.editor.CompilerSymbolsView.name=Symbole + +# Preferences +com.wudsn.ide.asm.preferences.AssemblerPreferences.name=Assembler Einstellungen +com.wudsn.ide.asm.preferences.AssemblerPreferencesPage.name=Assembler + +com.wudsn.ide.asm.runner.DefaultApplication.name=Standardanwendung des Betriebssystems +com.wudsn.ide.asm.runner.UserDefinedApplication.name=Benutzerdefinierte Anwendung \ No newline at end of file diff --git a/com.wudsn.ide.asm/schema/compilers.exsd b/com.wudsn.ide.asm/schema/compilers.exsd new file mode 100644 index 00000000..fb88ff2b --- /dev/null +++ b/com.wudsn.ide.asm/schema/compilers.exsd @@ -0,0 +1,197 @@ + + + + + + + + + This extension point allows for adding new 6502 assembler compilers to the WUDSN IDE. + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + The technical id of the compiler. Must consist of lower case letters or digits. + + + + + + + Translatable name of the compiler. Typically contains only upper case letters. + + + + + + + + + + The name of the compiler class. + + + + + + + + + + Supported (minimum) version of the compiler. All compiler directives and major compiler features up to this version have been incorporated into this version of WUDSN IDE. + + + + + + + The absolute URL pointing to the homepage of the compiler where the latest version can be downloaded. + + + + + + + The relative paths to the help documents. Multiple paths are separated by comma. Each path is evluated based on the absolute path of the compiler executable in the preferences. + + + + + + + The default compiler parameters which are use if not other parameters are specified in the preferences. + + + + + + + The default hardware which is assumed, if no source annotation is found. + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + WUDSN IDE 1.1.0 + + + + + + + + + + To implement this extension point, you have to create a sub-class of com.wudsn.ide.asm.compiler.Compiler, a sub-class of com.wudsn.ide.asm.compiler.parser.CompilerSourceParser and a sub-class of com.wudsn.ide.asm.compiler.CompilerLogParser. In the constructor of the compiler you have to set the source parser instance. In the "createLogParser" method of the compiler you have to create a new instance of the corresponding log parser. + + + + + + + + + + (c) 2009 Peter Dell + + + + diff --git a/com.wudsn.ide.asm/schema/runners.exsd b/com.wudsn.ide.asm/schema/runners.exsd new file mode 100644 index 00000000..e7703290 --- /dev/null +++ b/com.wudsn.ide.asm/schema/runners.exsd @@ -0,0 +1,144 @@ + + + + + + + + + This extension point allows for adding new runner, for example runners, to the WUDSN IDE. + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Optional implementation class to provide runner specific breakouts, for example for creating breakpoint files. + + + + + + + + + + + + + + + + + + + + + + + + + + + + + WUDSN IDE 1.3.0 + + + + + + + + + + + + (c) 2009 Peter Dell + + + + diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/Actions.properties b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/Actions.properties new file mode 100644 index 00000000..b9deb5e4 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/Actions.properties @@ -0,0 +1,3 @@ +# Used by AssemblerEditor +com.wudsn.ide.asm.editor.ContentAssistProposal_label=Content Assist +com.wudsn.ide.asm.editor.AssemblerEditorToggleCommentCommand_label=Toggle Comment2 \ No newline at end of file diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/AssemblerPlugin.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/AssemblerPlugin.java new file mode 100644 index 00000000..735bfd0d --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/AssemblerPlugin.java @@ -0,0 +1,305 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm; + +import java.util.HashMap; +import java.util.Map; +import java.util.Set; + +import org.eclipse.core.runtime.ListenerList; +import org.eclipse.core.runtime.QualifiedName; +import org.eclipse.jface.preference.JFacePreferences; +import org.eclipse.jface.resource.JFaceResources; +import org.eclipse.jface.util.IPropertyChangeListener; +import org.eclipse.jface.util.PropertyChangeEvent; +import org.osgi.framework.BundleContext; + +import com.wudsn.ide.asm.compiler.CompilerConsole; +import com.wudsn.ide.asm.compiler.CompilerRegistry; +import com.wudsn.ide.asm.preferences.AssemblerPreferences; +import com.wudsn.ide.asm.preferences.AssemblerPreferencesChangeListener; +import com.wudsn.ide.asm.preferences.AssemblerPreferencesConstants; +import com.wudsn.ide.asm.runner.RunnerRegistry; +import com.wudsn.ide.base.common.AbstractIDEPlugin; + +/** + * The main plugin class to be used in the desktop. + * + * @author Peter Dell + */ +public final class AssemblerPlugin extends AbstractIDEPlugin { + + /** + * The plugin id. + */ + public static final String ID = "com.wudsn.ide.asm"; + + /** + * The shared instance. + */ + private static AssemblerPlugin plugin; + + /** + * The preferences. + */ + private AssemblerPreferences preferences; + private ListenerList preferencesChangeListeners; + + /** + * The compiler registry. + */ + private CompilerRegistry compilerRegistry; + + /** + * The compiler console. + */ + private CompilerConsole compilerConsole; + + /** + * The runner registry. + */ + private RunnerRegistry runnerRegistry; + + /** + * The UI properties. + */ + private Map properties; + + /** + * Creates a new instance. Must be public for dynamic instantiation. + */ + public AssemblerPlugin() { + preferences = null; + preferencesChangeListeners = new ListenerList(ListenerList.IDENTITY); + compilerRegistry = new CompilerRegistry(); + compilerConsole = null; + runnerRegistry = new RunnerRegistry(); + properties = new HashMap(10); + } + + /** + * {@inheritDoc} + */ + @Override + protected String getPluginId() { + return ID; + } + + /** + * {@inheritDoc} + */ + @Override + public void start(BundleContext context) throws Exception { + super.start(context); + preferences = new AssemblerPreferences(getPreferenceStore()); + plugin = this; + try { + compilerRegistry.init(); + } catch (Exception ex) { + logError("Cannot initialize compiler registry", null, ex); + throw ex; + } + compilerConsole = new CompilerConsole(); + try { + runnerRegistry.init(); + } catch (Exception ex) { + logError("Cannot initialize runner registry", null, ex); + throw ex; + } + + // Register for global JFace preferences that also affect the editors. + JFacePreferences.getPreferenceStore().addPropertyChangeListener(new IPropertyChangeListener() { + final static String BLOCK_SELECTION_MODE_FONT = "org.eclipse.ui.workbench.texteditor.blockSelectionModeFont"; + + @Override + public void propertyChange(PropertyChangeEvent event) { + if (event.getProperty().equals(JFaceResources.TEXT_FONT) + || event.getProperty().equals(BLOCK_SELECTION_MODE_FONT)) { + firePreferencesChangeEvent(AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTES); + } + + } + }); + + } + + /** + * {@inheritDoc} + */ + @Override + public void stop(BundleContext context) throws Exception { + super.stop(context); + plugin = null; + } + + /** + * Gets the shared plugin instance. + * + * @return The plug-in, not null. + */ + public static AssemblerPlugin getInstance() { + if (plugin == null) { + throw new IllegalStateException("Plugin not initialized or already stopped"); + } + return plugin; + } + + /** + * Gets the compiler registry for this plugin. + * + * @return The compiler registry, not null. + */ + public CompilerRegistry getCompilerRegistry() { + if (compilerRegistry == null) { + throw new IllegalStateException("Field 'compilerRegistry' must not be null."); + } + return compilerRegistry; + } + + /** + * Gets the compiler console for this plugin. + * + * @return The compiler console, not null. + */ + public final CompilerConsole getCompilerConsole() { + if (compilerConsole == null) { + throw new IllegalStateException("Field 'compilerConsole' must not be null."); + } + return compilerConsole; + } + + /** + * Gets the runner registry for this plugin. + * + * @return The runner registry, not null. + */ + public RunnerRegistry getRunnerRegistry() { + if (runnerRegistry == null) { + throw new IllegalStateException("Field 'runnerRegistry' must not be null."); + } + return runnerRegistry; + } + + /** + * Gets the preferences for this plugin. + * + * @return The preferences, not null. + */ + public AssemblerPreferences getPreferences() { + if (preferences == null) { + throw new IllegalStateException("Field 'preferences' must not be null."); + } + return preferences; + } + + /** + * Adds a listener for immediate preferences changes. + * + * @param listener + * The listener, not null. + * @since 1.6.3 + */ + public void addPreferencesChangeListener(AssemblerPreferencesChangeListener listener) { + if (listener == null) { + throw new IllegalArgumentException("Parameter 'listener' must not be null."); + } + preferencesChangeListeners.add(listener); + } + + /** + * Removes a listener for immediate preferences changes. + * + * @param listener + * The listener, not null. + * @since 1.6.3 + */ + public void removePreferencesChangeListener(AssemblerPreferencesChangeListener listener) { + if (listener == null) { + throw new IllegalArgumentException("Parameter 'listener' must not be null."); + } + preferencesChangeListeners.remove(listener); + } + + /** + * Fire the change events for all registered listeners. + * + * @param changedPropertyNames + * The set of property changed property names, not + * null. + * + * @since 1.6.3 + */ + public void firePreferencesChangeEvent(Set changedPropertyNames) { + if (changedPropertyNames == null) { + throw new IllegalArgumentException("Parameter 'changedPropertyNames' must not be null."); + } + if (!changedPropertyNames.isEmpty()) { + + for (Object listener : preferencesChangeListeners.getListeners()) { + ((AssemblerPreferencesChangeListener) listener).preferencesChanged(preferences, changedPropertyNames); + } + } + } + + /** + * Gets a UI property. + * + * @param key + * The property key, not null. + * + * @return The UI property, may be empty, not null. + */ + public String getProperty(QualifiedName key) { + if (key == null) { + throw new IllegalArgumentException("Parameter 'key' must not be null."); + } + + String result; + synchronized (properties) { + result = properties.get(key); + } + if (result == null) { + result = ""; + } + return result; + } + + /** + * Set a UI property. + * + * @param key + * The property key, not null. + * + * @param value + * The UI property, may be empty, not null. + */ + public void setProperty(QualifiedName key, String value) { + if (key == null) { + throw new IllegalArgumentException("Parameter 'key' must not be null."); + } + if (value == null) { + throw new IllegalArgumentException("Parameter 'value' must not be null."); + } + synchronized (properties) { + properties.put(key, value); + } + + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/AssemblerProperties.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/AssemblerProperties.java new file mode 100644 index 00000000..ff051fc2 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/AssemblerProperties.java @@ -0,0 +1,144 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm; + +import java.util.Map; +import java.util.TreeMap; + +import org.eclipse.core.resources.IMarker; + +import com.wudsn.ide.base.common.StringUtility; + +public final class AssemblerProperties { + + /** + * A single key value pair. + * + * @since 1.6.1 + */ + public static final class AssemblerProperty { + public final String key; + public final String value; + public final int lineNumber; + + AssemblerProperty(String key, String value, int lineNumber) { + this.key = key; + this.value = value; + this.lineNumber = lineNumber; + } + + @Override + public String toString() { + return key + "=" + value + " in line " + lineNumber; + } + } + + @SuppressWarnings("serial") + public final static class InvalidAssemblerPropertyException extends Exception { + public final AssemblerProperty property; + public final IMarker marker; + + public InvalidAssemblerPropertyException(AssemblerProperty property, IMarker marker) { + if (property == null) { + throw new IllegalArgumentException("Parameter 'property' must not be null."); + } + this.property = property; + this.marker = marker; + } + } + + /** + * Source code constants. + */ + public final static String PREFIX = "@com.wudsn.ide.asm."; + public final static String HARDWARE = "@com.wudsn.ide.asm.hardware"; + public final static String MAIN_SOURCE_FILE = "@com.wudsn.ide.asm.mainsourcefile"; + public final static String OUTPUT_FOLDER_MODE = "@com.wudsn.ide.asm.outputfoldermode"; + public final static String OUTPUT_FOLDER = "@com.wudsn.ide.asm.outputfolder"; + public final static String OUTPUT_FILE_EXTENSION = "@com.wudsn.ide.asm.outputfileextension"; + public final static String OUTPUT_FILE = "@com.wudsn.ide.asm.outputfile"; + + private Map properties; + + /** + * Creation is public. + */ + public AssemblerProperties() { + properties = new TreeMap(); + } + + /** + * Puts a new value into the properties provided not other value is already + * there. + * + * @param key + * The property key, not empty and not null. + * @param value + * The property value, may be empty, not null. + * @param lineNumber + * The line number, a positive integer or 0 if the line number is + * undefined. + * @since 1.6.1 + */ + public void put(String key, String value, int lineNumber) { + if (key == null) { + throw new IllegalArgumentException("Parameter 'key' must not be null."); + } + if (StringUtility.isEmpty(key)) { + throw new IllegalArgumentException("Parameter 'key' must not be empty."); + } + if (value == null) { + throw new IllegalArgumentException("Parameter 'value' must not be null."); + } + if (lineNumber < 0l) { + throw new IllegalArgumentException("Parameter 'lineNumber' must not be negative. Specified value is " + + lineNumber + "."); + } + if (!properties.containsKey(key)) { + AssemblerProperty property = new AssemblerProperty(key, value, lineNumber); + properties.put(key, property); + } + } + + /** + * Gets a property from the properties map. + * + * @param key + * The property key, not empty and not null. + * @return The property or null if the property is not defined. + * + * @since 1.6.1 + */ + public AssemblerProperty get(String key) { + if (key == null) { + throw new IllegalArgumentException("Parameter 'key' must not be null."); + } + if (StringUtility.isEmpty(key)) { + throw new IllegalArgumentException("Parameter 'key' must not be empty."); + } + return properties.get(key); + } + + @Override + public String toString() { + return properties.toString(); + } + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/CPU.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/CPU.java new file mode 100644 index 00000000..2c0b38d5 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/CPU.java @@ -0,0 +1,31 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ +package com.wudsn.ide.asm; + +/** + * Enum for the supported CPUs. Used for restricting the visible instructions. + * + * @author Peter Dell + * + * @sinbce 1.6.1 + */ +public enum CPU { + + MOS6502, MOS6502_ILLEGAL, MOS65C02, MOS6502_DTV, MOS65816 +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/Hardware.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/Hardware.java new file mode 100644 index 00000000..7478f01a --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/Hardware.java @@ -0,0 +1,31 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ +package com.wudsn.ide.asm; + +/** + * Enum for the supported hardware platforms. Used for classifying compilers, + * emulators and converters. + * + * @author Peter Dell + * + */ +public enum Hardware { + + GENERIC, APPLE2, ATARI2600, ATARI7800, ATARI8BIT, C64, NES, TEST +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/HardwareUtility.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/HardwareUtility.java new file mode 100644 index 00000000..73b76cff --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/HardwareUtility.java @@ -0,0 +1,170 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ +package com.wudsn.ide.asm; + +import org.eclipse.jface.resource.ImageDescriptor; + +import com.wudsn.ide.asm.compiler.CompilerFileWriter; +import com.wudsn.ide.asm.compiler.writer.AppleFileWriter; + +/** + * Map value of {@link Hardware} to icon paths and descriptors. + * + * @author Peter Dell + * + */ +public final class HardwareUtility { + + /** + * Creation is private. + */ + private HardwareUtility() { + } + + /** + * Gets the default file extension for executable files on a hardware. + * + * @param hardware + * The hardware, not null. + * @return The default file extension, may be empty, not null. + * + * @since 1.6.1 + */ + public static String getDefaultFileExtension(Hardware hardware) { + if (hardware == null) { + throw new IllegalArgumentException( + "Parameter 'hardware' must not be null."); + } + switch (hardware) { + case APPLE2: + // AppleDos 3.3 binary file + // start-lo,start-hi,length-lo,length-hi,data + return ".b"; + case ATARI2600: + // Atari VCS ROM cartridge + return ".bin"; + case ATARI7800: + // Atari 7800 ROM cartridge + return ".bin"; + case ATARI8BIT: + // AtariDOS 2.5 compound file, + // $ff,$ff,start-lo,start-hi,end-lo,end-hi,data + return ".xex"; + case C64: + // C64 program file + // start-lo,start-hi,data + return ".prg"; + case NES: + // NES ROM file + return ".nes"; + case TEST: + return ".tst"; + default: + return ""; + } + } + + /** + * Gets the image path for a hardware image. + * + * @param hardware + * The hardware, not null. + * @return The image path for the hardware image, not empty and not + * null. + */ + public static String getImagePath(Hardware hardware) { + if (hardware == null) { + throw new IllegalArgumentException( + "Parameter 'hardware' must not be null."); + } + String path; + switch (hardware) { + case GENERIC: + path = "hardware-generic-16x16.gif"; + break; + case APPLE2: + path = "hardware-apple2-16x16.gif"; + break; + case ATARI2600: + path = "hardware-atari2600-16x16.gif"; + break; + case ATARI7800: + path = "hardware-atari7800-16x16.gif"; + break; + case ATARI8BIT: + path = "hardware-atari8bit-16x16.gif"; + break; + case C64: + path = "hardware-c64-16x16.gif"; + break; + case NES: + path = "hardware-nes-16x16.gif"; + break; + case TEST: + path = "hardware-test-16x16.gif"; + break; + default: + throw new IllegalArgumentException("Unknown hardware " + hardware + + "."); + } + return path; + } + + /** + * Gets the image descriptor for a hardware image. + * + * @param hardware + * The hardware, not null. + * @return The image descriptor for the hardware image, not + * null. + */ + public static ImageDescriptor getImageDescriptor(Hardware hardware) { + if (hardware == null) { + throw new IllegalArgumentException( + "Parameter 'hardware' must not be null."); + } + ImageDescriptor result; + result = AssemblerPlugin.getInstance().getImageDescriptor( + getImagePath(hardware)); + return result; + } + + /** + * Gets the compiler file writer a hardware. + * + * @param hardware + * The hardware, not null. + * @return The image descriptor for the hardware image, not + * null. + * @since 1.6.4 + */ + public static CompilerFileWriter getCompilerFileWriter(Hardware hardware) { + if (hardware == null) { + throw new IllegalArgumentException( + "Parameter 'hardware' must not be null."); + } + CompilerFileWriter compilerFileWriter; + if (hardware.equals(Hardware.APPLE2)) { + compilerFileWriter = new AppleFileWriter(); + } else { + compilerFileWriter = new CompilerFileWriter(); + } + return compilerFileWriter; + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/Texts.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/Texts.java new file mode 100644 index 00000000..931ac98b --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/Texts.java @@ -0,0 +1,288 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm; + +import org.eclipse.osgi.util.NLS; + +/** + * Class which holds the localized text constants. + * + * @author Peter Dell + */ +public final class Texts extends NLS { + + public static String TOC_ASSEMBLER_SYNTAX_YES; + public static String TOC_ASSEMBLER_SYNTAX_NO; + public static String TOC_ASSEMBLER_SYNTAX_BLOCK_DEFINITION_CHARACTERS; + public static String TOC_ASSEMBLER_SYNTAX_COMPLETION_PROPOSAL_AUTO_ACTIVATION_CHARACTERS; + public static String TOC_ASSEMBLER_SYNTAX_IDENTIFIER_PART_CHARACTERS; + public static String TOC_ASSEMBLER_SYNTAX_IDENTIFIER_SEPARATOR_CHARACTER; + public static String TOC_ASSEMBLER_SYNTAX_IDENTIFIER_START_CHARACTERS; + public static String TOC_ASSEMBLER_SYNTAX_IDENTIFIERS_CASE_SENSITIVE; + public static String TOC_ASSEMBLER_SYNTAX_INSTRUCTIONS_CASE_SENSITIVE; + public static String TOC_ASSEMBLER_SYNTAX_LABEL_DEFINITION_SUFFIX_CHARACTER; + public static String TOC_ASSEMBLER_SYNTAX_MACRO_USAGE_PREFIX_CHARACTER; + public static String TOC_ASSEMBLER_SYNTAX_MULTIPLE_LINES_COMMENT_DELIMITERS; + public static String TOC_ASSEMBLER_SYNTAX_SINGLE_LINE_COMMENT_DELIMITERS; + public static String TOC_ASSEMBLER_SYNTAX_SOURCE_INCLUDE_DEFAULT_EXTENSION; + public static String TOC_ASSEMBLER_SYNTAX_STRING_DELIMITERS; + + /** + * Compiler console. + */ + public static String COMPILER_CONSOLE_TITLE; + + /** + * Compiler source parser tree. + */ + public static String COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_DEFAULT; + public static String COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_DEFINITION_SECTION; + public static String COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_IMPLEMENTATION_SECTION; + public static String COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_EQUATE_DEFINITION; + public static String COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_LABEL_DEFINITION; + public static String COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_ENUM_DEFINITION_SECTION; + public static String COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_STRUCTURE_DEFINITION_SECTION; + public static String COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_LOCAL_SECTION; + public static String COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_MACRO_DEFINITION_SECTION; + public static String COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_PAGES_SECTION; + public static String COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_PROCEDURE_DEFINITION_SECTION; + public static String COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_REPEAT_SECTION; + public static String COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_SOURCE_INCLUDE; + public static String COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_BINARY_INCLUDE; + public static String COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_BINARY_OUTPUT; + + /** + * Compiler syntax. + */ + public static String COMPILER_SYNTAX_INSTRUCTION_DIRECTIVE; + public static String COMPILER_SYNTAX_LEGAL_OPCODE; + public static String COMPILER_SYNTAX_ILLEGAL_OPCODE; + public static String COMPILER_SYNTAX_PSEUDO_OPCODE; + public static String COMPILER_SYNTAX_W65816_ONLY; + + /** + * Assembler toolbar and menu. + * + */ + public static String ASSEMBLER_TOOLBAR_RUN_WITH_DEFAULT_LABEL; + + /** + * Assembler content outline. + */ + public static String ASSEMBLER_CONTENT_OUTLINE_SORT_BUTTON_TOOL_TIP; + public static String ASSEMBLER_CONTENT_OUTLINE_TREE_TYPE_DEFAULT; + public static String ASSEMBLER_CONTENT_OUTLINE_TREE_TYPE_DEFINITION_SECTION; + public static String ASSEMBLER_CONTENT_OUTLINE_TREE_TYPE_IMPLEMENTATION_SECTION; + + /** + * Assembler hyperlink detector. + */ + public static String ASSEMBLER_HYPERLINK_DETECTOR_OPEN_SOURCE_WITH_ASSEMBLER_EDITOR; + public static String ASSEMBLER_HYPERLINK_DETECTOR_OPEN_BINARY_WITH_HEX_EDITOR; + public static String ASSEMBLER_HYPERLINK_DETECTOR_OPEN_BINARY_WITH_GRAPHICS_EDITOR; + public static String ASSEMBLER_HYPERLINK_DETECTOR_OPEN_BINARY_WITH_DEFAULT_EDITOR; + public static String ASSEMBLER_HYPERLINK_DETECTOR_OPEN_BINARY_WITH_SYSTEM_EDITOR; + public static String ASSEMBLER_HYPERLINK_DETECTOR_OPEN_IDENTIFIER; + public static String ASSEMBLER_HYPERLINK_DETECTOR_OPEN_IDENTIFIER_IN_INCLUDE; + public static String ASSEMBLER_HYPERLINK_FILE_NOT_EXISTS; + + /** + * Assembler breakpoints. + */ + public static String ASSEMBLER_BREAKPOINT_MARKER_MESSAGE; + public static String ASSEMBLER_BREAKPOINT_TOGGLE_TYPE_MENU_TEXT; + + /** + * Compiler symbols + */ + public static String COMPILER_SYMBOLS_VIEW_FILTER_TOOLTIP; + public static String COMPILER_SYMBOLS_VIEW_SOURCE_LABEL; + public static String COMPILER_SYMBOLS_VIEW_SOURCE_NONE; + public static String COMPILER_SYMBOLS_VIEW_SOURCE_TOTAL_COUNT; + public static String COMPILER_SYMBOLS_VIEW_SOURCE_FILTERED_COUNT; + public static String COMPILER_SYMBOLS_VIEW_TYPE_COLUMN_LABEL; + public static String COMPILER_SYMBOLS_VIEW_BANK_COLUMN_LABEL; + public static String COMPILER_SYMBOLS_VIEW_NAME_COLUMN_LABEL; + public static String COMPILER_SYMBOLS_VIEW_HEX_VALUE_COLUMN_LABEL; + public static String COMPILER_SYMBOLS_VIEW_DECIMAL_VALUE_COLUMN_LABEL; + public static String COMPILER_SYMBOLS_VIEW_STRING_VALUE_COLUMN_LABEL; + + /** + * Preferences: syntax highlighting. + */ + public static String PREFERENCES_SYNTAX_HIGHLIGHTING_GROUP_TITLE; + + public static String PREFERENCES_TEXT_ATTRIBUTE_COMMENT_NAME; + public static String PREFERENCES_TEXT_ATTRIBUTE_STRING_NAME; + public static String PREFERENCES_TEXT_ATTRIBUTE_NUMBER_NAME; + public static String PREFERENCES_TEXT_ATTRIBUTE_DIRECTIVE; + public static String PREFERENCES_TEXT_ATTRIBUTE_OPCODE_LEGAL_NAME; + public static String PREFERENCES_TEXT_ATTRIBUTE_OPCODE_ILLEGAL_NAME; + public static String PREFERENCES_TEXT_ATTRIBUTE_OPCODE_PSEUDO_NAME; + public static String PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_EQUATE; + public static String PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_LABEL; + public static String PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_ENUM_DEFINITION_SECTION; + public static String PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_STRUCTURE_DEFINITION_SECTION; + public static String PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_LOCAL_SECTION; + public static String PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_MACRO_DEFINITION_SECTION; + public static String PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_PROCEDURE_DEFINITION_SECTION; + + public static String PREFERENCES_FOREGROUND_COLOR_LABEL; + public static String PREFERENCES_BOLD_LABEL; + public static String PREFERENCES_ITALIC_LABEL; + + /** + * Preferences: editor content assist and parsing. + */ + public static String PREFERENCES_EDITOR_GROUP_TITLE; + + public static String PREFERENCES_CONTENT_ASSIST_PROCESSOR_DEFAULT_CASE_LABEL; + public static String PREFERENCES_CONTENT_ASSIST_PROCESSOR_DEFAULT_CASE_UPPER_CASE_TEXT; + public static String PREFERENCES_CONTENT_ASSIST_PROCESSOR_DEFAULT_CASE_LOWER_CASE_TEXT; + + public static String PREFERENCES_COMPILE_COMMAND_POSITIONING_MODE_LABEL; + public static String PREFERENCES_COMPILE_COMMAND_POSITIONING_MODE_FIRST_ERROR_OR_WARNING_TEXT; + public static String PREFERENCES_COMPILE_COMMAND_POSITIONING_MODE_FIRST_ERROR_TEXT; + + /** + * Preferences: compiler and runner + */ + public static String PREFERENCES_DOWNLOAD_LINK; + public static String PREFERENCES_DOWNLOAD_LINK_TOOL_TIP; + + /** + * Preferences: compiler. + */ + public static String PREFERENCES_COMPILER_CPU_LABEL; + + public static String PREFERENCES_COMPILER_EXECUTABLE_PATH_LABEL; + public static String PREFERENCES_COMPILER_HARDWARE_ACTIVE_LABEL; + public static String PREFERENCES_COMPILER_DEFAULT_PARAMETERS_LABEL; + public static String PREFERENCES_COMPILER_PARAMETERS_LABEL; + public static String PREFERENCES_COMPILER_PARAMETERS_HELP; + public static String PREFERENCES_COMPILER_PARAMETERS_VARIABLES; + + public static String PREFERENCES_COMPILER_OUTPUT_FOLDER_MODE_LABEL; + public static String PREFERENCES_COMPILER_OUTPUT_FOLDER_MODE_SOURCE_FOLDER_TEXT; + public static String PREFERENCES_COMPILER_OUTPUT_FOLDER_MODE_TEMP_FOLDER_TEXT; + public static String PREFERENCES_COMPILER_OUTPUT_FOLDER_MODE_FIXED_FOLDER_TEXT; + public static String PREFERENCES_COMPILER_OUTPUT_FOLDER_PATH_LABEL; + public static String PREFERENCES_COMPILER_OUTPUT_FILE_EXTENSION_LABEL; + + public static String PREFERENCES_COMPILER_RUNNER_ID_LABEL; + public static String PREFERENCES_COMPILER_RUNNER_EXECUTABLE_PATH_LABEL; + public static String PREFERENCES_COMPILER_RUNNER_DEFAULT_COMMAND_LINE_LABEL; + public static String PREFERENCES_COMPILER_RUNNER_COMMAND_LINE_LABEL; + public static String PREFERENCES_COMPILER_RUNNER_COMMAND_LINE_HELP; + public static String PREFERENCES_COMPILER_RUNNER_COMMAND_LINE_VARIABLES; + public static String PREFERENCES_COMPILER_RUNNER_WAIT_FOR_COMPLETION_LABEL; + + /** + * Help table of contents + */ + public static String TOC_WUDSN_IDE_LABEL; + + public static String TOC_IDE_TOPIC_LABEL; + + public static String TOC_ASSEMBLERS_TOPIC_LABEL; + public static String TOC_ASSEMBLER_GENERAL_TOPIC_LABEL; + public static String TOC_ASSEMBLER_NAME_LABEL; + public static String TOC_ASSEMBLER_HOME_PAGE_LABEL; + public static String TOC_ASSEMBLER_DEFAULT_HARDWARE_LABEL; + public static String TOC_ASSEMBLER_SUPPORTED_CPUS_LABEL; + public static String TOC_ASSEMBLER_DEFAULT_PARAMETERS_LABEL; + public static String TOC_ASSEMBLER_INSTRUCTIONS_TOPIC_LABEL; + public static String TOC_ASSEMBLER_INSTRUCTION_TYPE_DIRECTIVES_LABEL; + public static String TOC_ASSEMBLER_INSTRUCTION_TYPE_LEGAL_OPCODES_LABEL; + public static String TOC_ASSEMBLER_INSTRUCTION_TYPE_PSEUDO_OPCODES_LABEL; + public static String TOC_ASSEMBLER_INSTRUCTION_TYPE_ILLEGAL_OPCODES_LABEL; + public static String TOC_ASSEMBLER_INSTRUCTION_TYPE_W65816_OPCODES_LABEL; + public static String TOC_ASSEMBLER_INSTRUCTION_TYPE_LABEL; + public static String TOC_ASSEMBLER_INSTRUCTION_NAME_LABEL; + public static String TOC_ASSEMBLER_INSTRUCTION_DESCRIPTION_LABEL; + public static String TOC_ASSEMBLER_MANUAL_TOPIC_LABEL; + + public static String TOC_HARDWARES_TOPIC_LABEL; + public static String TOC_HARDWARE_NAME_LABEL; + public static String TOC_HARDWARE_ID_LABEL; + public static String TOC_HARDWARE_ICON_LABEL; + public static String TOC_HARDWARE_DEFAULT_FILE_EXTENSION_LABEL; + public static String TOC_HARDWARE_EMULATOR_LABEL; + public static String TOC_HARDWARE_HOME_PAGE_LABEL; + public static String TOC_HARDWARE_DEFAULT_PARAMETERS_LABEL; + + public static String TOC_CPUS_TOPIC_LABEL; + public static String TOC_CPU_NAME_LABEL; + public static String TOC_CPU_OPCODE_LABEL; + + /** + * Messages for the Assembler plugin. + */ + public static String MESSAGE_E100; + public static String MESSAGE_E101; + public static String MESSAGE_E102; + public static String MESSAGE_E103; + public static String MESSAGE_E104; + public static String MESSAGE_E105; + public static String MESSAGE_E106; + public static String MESSAGE_E107; + public static String MESSAGE_E108; + public static String MESSAGE_I109; + public static String MESSAGE_I110; + public static String MESSAGE_E111; + public static String MESSAGE_E112; + public static String MESSAGE_E113; + public static String MESSAGE_E114; + public static String MESSAGE_E115; + public static String MESSAGE_E116; + public static String MESSAGE_E117; + public static String MESSAGE_I118; + public static String MESSAGE_E119; + public static String MESSAGE_W120; + public static String MESSAGE_I121; + public static String MESSAGE_E122; + public static String MESSAGE_E123; + public static String MESSAGE_E124; + public static String MESSAGE_E125; + public static String MESSAGE_E126; + public static String MESSAGE_E127; + public static String MESSAGE_E128; + public static String MESSAGE_E129; + public static String MESSAGE_E130; + public static String MESSAGE_E131; + public static String MESSAGE_E132; + public static String MESSAGE_E133; + public static String MESSAGE_E134; + public static String MESSAGE_E135; + public static String MESSAGE_E136; + public static String MESSAGE_E137; + public static String MESSAGE_E138; + public static String MESSAGE_E139; + public static String MESSAGE_E140; + public static String MESSAGE_E141; + public static String MESSAGE_E142; + public static String MESSAGE_S143; + /** + * Initializes the constants. + */ + static { + NLS.initializeMessages(Texts.class.getName(), Texts.class); + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/Texts.properties b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/Texts.properties new file mode 100644 index 00000000..6dd13a77 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/Texts.properties @@ -0,0 +1,206 @@ +COMPILER_CONSOLE_TITLE=Compiler Console + +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_DEFAULT=Default +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_DEFINITION_SECTION=Definition Section +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_IMPLEMENTATION_SECTION=Implementation Section +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_EQUATE_DEFINITION=Equate Definition +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_LABEL_DEFINITION=Label Definition +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_ENUM_DEFINITION_SECTION=Enum Definition Section +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_STRUCTURE_DEFINITION_SECTION=Structure Definition Section +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_LOCAL_SECTION=Local Section +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_MACRO_DEFINITION_SECTION=Macro Definition Section +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_PAGES_SECTION=Pages Section +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_PROCEDURE_DEFINITION_SECTION=Procedure Definition Section +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_REPEAT_SECTION=Repeat Section +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_SOURCE_INCLUDE=Source Include +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_BINARY_INCLUDE=Binary Include +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_BINARY_OUTPUT=Binary Output + +COMPILER_SYNTAX_INSTRUCTION_DIRECTIVE=Directive +COMPILER_SYNTAX_LEGAL_OPCODE=Legal Opcode +COMPILER_SYNTAX_ILLEGAL_OPCODE=Illegal Opcode +COMPILER_SYNTAX_PSEUDO_OPCODE=Pseudo Opcode +COMPILER_SYNTAX_W65816_ONLY=W65816 only + +ASSEMBLER_TOOLBAR_RUN_WITH_DEFAULT_LABEL=(default) + +ASSEMBLER_CONTENT_OUTLINE_SORT_BUTTON_TOOL_TIP=Sort +ASSEMBLER_CONTENT_OUTLINE_TREE_TYPE_DEFAULT=Default +ASSEMBLER_CONTENT_OUTLINE_TREE_TYPE_DEFINITION_SECTION=Definition Section +ASSEMBLER_CONTENT_OUTLINE_TREE_TYPE_IMPLEMENTATION_SECTION=Implementation Section + +ASSEMBLER_HYPERLINK_DETECTOR_OPEN_SOURCE_WITH_ASSEMBLER_EDITOR=Open with Assembler Editor +ASSEMBLER_HYPERLINK_DETECTOR_OPEN_BINARY_WITH_HEX_EDITOR=Open with Hex Editor +ASSEMBLER_HYPERLINK_DETECTOR_OPEN_BINARY_WITH_GRAPHICS_EDITOR=Open with Graphics Editor +ASSEMBLER_HYPERLINK_DETECTOR_OPEN_BINARY_WITH_DEFAULT_EDITOR=Open with Default Editor +ASSEMBLER_HYPERLINK_DETECTOR_OPEN_BINARY_WITH_SYSTEM_EDITOR=Open with System Editor +ASSEMBLER_HYPERLINK_DETECTOR_OPEN_IDENTIFIER={0} {1} in line {2} +ASSEMBLER_HYPERLINK_DETECTOR_OPEN_IDENTIFIER_IN_INCLUDE={0} {1} in line {2} of {3} +ASSEMBLER_HYPERLINK_FILE_NOT_EXISTS=Target file '{0}' not exists. Do you want to create the file now? + +ASSEMBLER_BREAKPOINT_MARKER_MESSAGE={0} [line: {1}] - {2} +ASSEMBLER_BREAKPOINT_TOGGLE_TYPE_MENU_TEXT=Assembler Breakpoints + +COMPILER_SYMBOLS_VIEW_FILTER_TOOLTIP=Type one or more substrings separated by spaces to filter the result list +COMPILER_SYMBOLS_VIEW_SOURCE_LABEL=Source: +COMPILER_SYMBOLS_VIEW_SOURCE_NONE=None +COMPILER_SYMBOLS_VIEW_SOURCE_TOTAL_COUNT={0} symbols +COMPILER_SYMBOLS_VIEW_SOURCE_FILTERED_COUNT={0} of {1} symbols +COMPILER_SYMBOLS_VIEW_SOURCE_NONE=Symbols +COMPILER_SYMBOLS_VIEW_TYPE_COLUMN_LABEL=Type +COMPILER_SYMBOLS_VIEW_BANK_COLUMN_LABEL=Bank +COMPILER_SYMBOLS_VIEW_NAME_COLUMN_LABEL=Name +COMPILER_SYMBOLS_VIEW_HEX_VALUE_COLUMN_LABEL=Hex Value +COMPILER_SYMBOLS_VIEW_DECIMAL_VALUE_COLUMN_LABEL=Decimal Value +COMPILER_SYMBOLS_VIEW_STRING_VALUE_COLUMN_LABEL=String Value + +PREFERENCES_SYNTAX_HIGHLIGHTING_GROUP_TITLE=Syntax Highlighting + +PREFERENCES_TEXT_ATTRIBUTE_COMMENT_NAME=Comments +PREFERENCES_TEXT_ATTRIBUTE_STRING_NAME=Strings +PREFERENCES_TEXT_ATTRIBUTE_NUMBER_NAME=Numbers +PREFERENCES_TEXT_ATTRIBUTE_DIRECTIVE=Directives +PREFERENCES_TEXT_ATTRIBUTE_OPCODE_LEGAL_NAME=Opcodes +PREFERENCES_TEXT_ATTRIBUTE_OPCODE_ILLEGAL_NAME=Illegal Opcodes +PREFERENCES_TEXT_ATTRIBUTE_OPCODE_PSEUDO_NAME=Pseudo Opcodes +PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_EQUATE=Identifiers of Equates +PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_LABEL=Identifiers of Labels +PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_ENUM_DEFINITION_SECTION=Identifiers of Enums +PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_STRUCTURE_DEFINITION_SECTION=Identifiers of Structures +PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_LOCAL_SECTION=Identifiers of Local Sections +PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_MACRO_DEFINITION_SECTION=Identifiers of Macros +PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_PROCEDURE_DEFINITION_SECTION=Identifiers of Procedures + +PREFERENCES_EDITOR_GROUP_TITLE=Editor + +PREFERENCES_CONTENT_ASSIST_PROCESSOR_DEFAULT_CASE_LABEL=Default Case for Content Assist +PREFERENCES_CONTENT_ASSIST_PROCESSOR_DEFAULT_CASE_LOWER_CASE_TEXT=Lower Case Instructions +PREFERENCES_CONTENT_ASSIST_PROCESSOR_DEFAULT_CASE_UPPER_CASE_TEXT=Upper Case Instructions + +PREFERENCES_COMPILE_COMMAND_POSITIONING_MODE_LABEL=Position after compiling +PREFERENCES_COMPILE_COMMAND_POSITIONING_MODE_FIRST_ERROR_OR_WARNING_TEXT=To first error or warning +PREFERENCES_COMPILE_COMMAND_POSITIONING_MODE_FIRST_ERROR_TEXT=To first error + + +PREFERENCES_FOREGROUND_COLOR_LABEL=&Color: +PREFERENCES_BOLD_LABEL=&Bold +PREFERENCES_ITALIC_LABEL=&Italic + +PREFERENCES_DOWNLOAD_LINK=Download +PREFERENCES_DOWNLOAD_LINK_TOOL_TIP=Open download page {0} + +PREFERENCES_COMPILER_CPU_LABEL=CP&U +PREFERENCES_COMPILER_EXECUTABLE_PATH_LABEL=Path to &Compiler +PREFERENCES_COMPILER_HARDWARE_ACTIVE_LABEL=Use for {0} +PREFERENCES_COMPILER_DEFAULT_PARAMETERS_LABEL=&Default Parameters +PREFERENCES_COMPILER_PARAMETERS_LABEL=&Parameters +PREFERENCES_COMPILER_PARAMETERS_HELP=Available variables are: +PREFERENCES_COMPILER_PARAMETERS_VARIABLES=${sourceFolderPath} The absolute path to the source folder\n${sourceFilePath} The absolute path to the source file\n${outputFolderPath} The absolute path to the output folder\n${outputFilePath} The absolute path to the output file\n${outputFilePathWithoutExtension} The absolute path to the output file without extension\n${outputFileName} The name of the output file including its extension\n${outputFileNameWithoutExtension} The name of the output file without extension\n${outputFileNameShortWithoutExtension} The name of the output file without extension shortened to 8 alphanumeric characters +PREFERENCES_COMPILER_OUTPUT_FOLDER_MODE_LABEL=&Output Folder +PREFERENCES_COMPILER_OUTPUT_FOLDER_MODE_SOURCE_FOLDER_TEXT=&Source Folder +PREFERENCES_COMPILER_OUTPUT_FOLDER_MODE_TEMP_FOLDER_TEXT=&Temporary Folder +PREFERENCES_COMPILER_OUTPUT_FOLDER_MODE_FIXED_FOLDER_TEXT=&Fixed Folder +PREFERENCES_COMPILER_OUTPUT_FOLDER_PATH_LABEL=&Output Folder +PREFERENCES_COMPILER_OUTPUT_FILE_EXTENSION_LABEL=Output File E&xtension +PREFERENCES_COMPILER_RUNNER_ID_LABEL=Default &Application to open Output File +PREFERENCES_COMPILER_RUNNER_EXECUTABLE_PATH_LABEL=Path to A&pplication +PREFERENCES_COMPILER_RUNNER_DEFAULT_COMMAND_LINE_LABEL=&Default Command Line +PREFERENCES_COMPILER_RUNNER_COMMAND_LINE_LABEL=Command &Line +PREFERENCES_COMPILER_RUNNER_COMMAND_LINE_HELP=Available variables are: +PREFERENCES_COMPILER_RUNNER_COMMAND_LINE_VARIABLES=${runnerExecutablePath} The absolute path to the application\n${sourceFolderPath} The absolute path to the source folder\n${sourceFilePath} The absolute path to the source file\n${outputFolderPath} The absolute path to the output folder\n${outputFilePath} The absolute path to the output file\n${outputFilePathWithoutExtension} The absolute path to the output file without extension\n${outputFileName} The name of the output file including its extension\n${outputFileNameWithoutExtension} The name of the output file without extension\n${outputFileNameShortWithoutExtension} The name of the output file without extension shortened to 8 alphanumeric characters +PREFERENCES_COMPILER_RUNNER_WAIT_FOR_COMPLETION_LABEL=Wait for end of application + +TOC_WUDSN_IDE_LABEL=WUDSN IDE Guide + +TOC_IDE_TOPIC_LABEL=WUDSN IDE + +TOC_ASSEMBLERS_TOPIC_LABEL=Assemblers +TOC_ASSEMBLER_GENERAL_TOPIC_LABEL=General +TOC_ASSEMBLER_NAME_LABEL=Name +TOC_ASSEMBLER_HOME_PAGE_LABEL=Home Page +TOC_ASSEMBLER_DEFAULT_HARDWARE_LABEL=Default Hardware +TOC_ASSEMBLER_SUPPORTED_CPUS_LABEL=Supported CPUs +TOC_ASSEMBLER_DEFAULT_PARAMETERS_LABEL=Default Parameters +TOC_ASSEMBLER_INSTRUCTIONS_TOPIC_LABEL=Instructions +TOC_ASSEMBLER_INSTRUCTION_TYPE_DIRECTIVES_LABEL=Directives +TOC_ASSEMBLER_INSTRUCTION_TYPE_LEGAL_OPCODES_LABEL=Legal Opcodes +TOC_ASSEMBLER_INSTRUCTION_TYPE_PSEUDO_OPCODES_LABEL=Pseudo Opcodes +TOC_ASSEMBLER_INSTRUCTION_TYPE_ILLEGAL_OPCODES_LABEL=Illegal Opcodes +TOC_ASSEMBLER_INSTRUCTION_TYPE_W65816_OPCODES_LABEL=W65816 Opcodes +TOC_ASSEMBLER_INSTRUCTION_TYPE_LABEL=Type +TOC_ASSEMBLER_INSTRUCTION_NAME_LABEL=Name +TOC_ASSEMBLER_INSTRUCTION_DESCRIPTION_LABEL=Description +TOC_ASSEMBLER_SYNTAX_YES=Yes +TOC_ASSEMBLER_SYNTAX_NO=No +TOC_ASSEMBLER_SYNTAX_BLOCK_DEFINITION_CHARACTERS=Block Definition Characters +TOC_ASSEMBLER_SYNTAX_COMPLETION_PROPOSAL_AUTO_ACTIVATION_CHARACTERS=Completion Proposal Characters +TOC_ASSEMBLER_SYNTAX_IDENTIFIER_PART_CHARACTERS=Identifier Part Characters +TOC_ASSEMBLER_SYNTAX_IDENTIFIER_SEPARATOR_CHARACTER=Identifier Separator Character +TOC_ASSEMBLER_SYNTAX_IDENTIFIER_START_CHARACTERS=Identifier Start Characters +TOC_ASSEMBLER_SYNTAX_IDENTIFIERS_CASE_SENSITIVE=Identifiers Case Sensitive +TOC_ASSEMBLER_SYNTAX_INSTRUCTIONS_CASE_SENSITIVE=Instructions Case Sensitive +TOC_ASSEMBLER_SYNTAX_LABEL_DEFINITION_SUFFIX_CHARACTER=Label Definition Suffix Character +TOC_ASSEMBLER_SYNTAX_MACRO_USAGE_PREFIX_CHARACTER=Macro Usage Prefix Character +TOC_ASSEMBLER_SYNTAX_MULTIPLE_LINES_COMMENT_DELIMITERS=Multiple Lines Comment Delimiters +TOC_ASSEMBLER_SYNTAX_SINGLE_LINE_COMMENT_DELIMITERS=Single Line Comment Delimiters +TOC_ASSEMBLER_SYNTAX_SOURCE_INCLUDE_DEFAULT_EXTENSION=Source Include Default Extension +TOC_ASSEMBLER_SYNTAX_STRING_DELIMITERS=String Delimiters +TOC_ASSEMBLER_MANUAL_TOPIC_LABEL=Manual + +TOC_HARDWARES_TOPIC_LABEL=Hardwares +TOC_HARDWARE_NAME_LABEL=Hardware +TOC_HARDWARE_ID_LABEL=Identifier +TOC_HARDWARE_ICON_LABEL=Icon +TOC_HARDWARE_DEFAULT_FILE_EXTENSION_LABEL=Default File Extension +TOC_HARDWARE_EMULATOR_LABEL=Emulator +TOC_HARDWARE_HOME_PAGE_LABEL=Home Page +TOC_HARDWARE_DEFAULT_PARAMETERS_LABEL=Default Parameters + +TOC_CPUS_TOPIC_LABEL=CPUs +TOC_CPU_NAME_LABEL=CPU +TOC_CPU_OPCODE_LABEL=Opcode + +MESSAGE_E100=Path to '{0}' compiler executable is not set in the 'Assembler' preferences. +MESSAGE_E101=The compiler '{0}' does not specify default parameters. +MESSAGE_E102=The compiler '{0}' does not specify a help file path. +MESSAGE_E103=Path to '{0}' compiler executable in the preferences points to non-existing file '{1}'. +MESSAGE_E104=Output file extension must be set in the preferences of compiler '{0}' or via the annotation '{1}'. +MESSAGE_E105=Cannot execute compiler process '{0}' in working directory '{1}'. System error: {2} +MESSAGE_E106=Output file '{0}' cannot be opened for writing. End all applications which may keep the file open. +MESSAGE_E107=Output file not created. Check the error messages and the console log. +MESSAGE_E108=Output file not updated. Check the error messages and the console log. +MESSAGE_I109=Output file '{0}' created or updated with {1} (${2}) bytes. +MESSAGE_I110=Symbols file '{0}' created with {1} symbols. +MESSAGE_E111=Cannot open symbols file '{0}' for output. System error: {1} +MESSAGE_E112=Path to application executable is not set in the preferences of application '{0}'. +MESSAGE_E113=Cannot execute application '{0}' process '{1}' in working directory '{2}'. System error: {3} +MESSAGE_E114=Path to '{0}' application executable in the preferences points to non-existing file '{1}'. +MESSAGE_E115=Cannot open output file '{0}' with the standard application since no application is registered for the file extension '{1}'. +MESSAGE_E116=Definition for application '{0}' from the preferences of hardware '{1}' is not registered. +MESSAGE_E117=Cannot delete empty symbols file '{0}'. +MESSAGE_I118=Opening output file '{0}' with application '{1}'. +MESSAGE_E119=Cannot open output file '{0}' with application '{1}'. +MESSAGE_W120=Breakpoints will be ignored because the application '{0}' does not support passing source level breakpoints. +MESSAGE_I121=Breakpoints file '{0}' created with {1} active breakpoints. +MESSAGE_E122=Cannot open breakpoints file '{0}' for output. System error: {1} +MESSAGE_E123=Cannot delete empty breakpoints file '{0}'. +MESSAGE_E124=Unknown hardware '{0}'. Specify one of the following valid values '{1}'. +MESSAGE_E125=Main source file '{0}' does not exist. +MESSAGE_E126=Output file created but empty. Check the error messages and the console log. +MESSAGE_E127=Compiler process ended with return code {0}. Check the error messages and the console log. +MESSAGE_E128=Hardware not specified. Specify one of the following valid values '{0}'. +MESSAGE_E129=Main source file specifies or defaults to hardware '{0}' while include file specifies or defaults to hardware '{1}'. +MESSAGE_E130=Help for the '{0}' compiler cannot be displayed because the path to the compiler executable is not set in the preferences. +MESSAGE_E131=Help for the '{0}' compiler cannot be displayed because no help file was found in the paths '{1}' relative to the compiler executable path '{0}'. +MESSAGE_E132=Disk image file '{0}' does not exist. Create a bootable disk image where the output file '{1}' can be stored. +MESSAGE_E133=Disk image file '{0}' is not writeable. Make the disk image file is writeable, so the output file '{1}' can be stored. +MESSAGE_E134=Disk image file '{0}' cannot be opened for reading. System error: {1} +MESSAGE_E135=Disk image file '{0}' does not contain a valid file system. Make sure the disk image is properly formatted, so the output file '{1}' can be stored. +MESSAGE_E136=Disk image file '{0}' is full. System error: {1} +MESSAGE_E137=Disk image file '{0}' cannot be opened for writing. System error: {1} +MESSAGE_E138=Output file {0} has an unsupported file extension or invalid content. Supported file extensions are ".b" (Apple II binary file with format [start, length, code]), ".prg" (C64 program file with format [start, code]) and ".xex" (Atari compound file with format [$ffff, start,end,code]). +MESSAGE_E139=Output file extension '{0}' must start with ".". +MESSAGE_E140=Output folder mode be set in the preferences of compiler '{0}' or via the annotation '{1}'. +MESSAGE_E141=Unknown output folder mode '{0}'. Specify one of the following valid values '{1}'. +MESSAGE_E142=Include statement for file '{0}' uses a file name that has a different case different from real file system name '{1}'. Correct the file name in the include statement. +MESSAGE_S143=In include file '{0}', line {1}. \ No newline at end of file diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/Texts_de_DE.properties b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/Texts_de_DE.properties new file mode 100644 index 00000000..ad857909 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/Texts_de_DE.properties @@ -0,0 +1,204 @@ +COMPILER_CONSOLE_TITLE=Kompiler Konsole + +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_DEFAULT=Default +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_DEFINITION_SECTION=Definition Section +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_IMPLEMENTATION_SECTION=Implementation Section +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_EQUATE_DEFINITION=Equate Definition +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_LABEL_DEFINITION=Label Definition +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_ENUM_DEFINITION_SECTION=Enum Definition Section +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_STRUCTURE_DEFINITION_SECTION=Structure Definition Section +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_LOCAL_SECTION=Local Section +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_MACRO_DEFINITION_SECTION=Macro Definition Section +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_PAGES_SECTION=Pages Section +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_PROCEDURE_DEFINITION_SECTION=Procedure Definition Section +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_REPEAT_SECTION=Repeat Section +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_SOURCE_INCLUDE=Source Include +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_BINARY_INCLUDE=Binary Include +COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_BINARY_OUTPUT=Binary Output + +COMPILER_SYNTAX_INSTRUCTION_DIRECTIVE=Direktive +COMPILER_SYNTAX_LEGAL_OPCODE=Legaler Opcode +COMPILER_SYNTAX_ILLEGAL_OPCODE=Illegaler Opcode +COMPILER_SYNTAX_PSEUDO_OPCODE=Pseudo Opcode +COMPILER_SYNTAX_W65816_ONLY=nur W65816 + +ASSEMBLER_TOOLBAR_RUN_WITH_DEFAULT_LABEL=(Standard) + +ASSEMBLER_CONTENT_OUTLINE_SORT_BUTTON_TOOL_TIP=Sortieren +ASSEMBLER_CONTENT_OUTLINE_TREE_TYPE_DEFAULT=Standard +ASSEMBLER_CONTENT_OUTLINE_TREE_TYPE_DEFINITION_SECTION=Definitionsabschnitt +ASSEMBLER_CONTENT_OUTLINE_TREE_TYPE_IMPLEMENTATION_SECTION=Implementierungsabschnitt + +ASSEMBLER_HYPERLINK_DETECTOR_OPEN_SOURCE_WITH_ASSEMBLER_EDITOR=Öffnen mit Assembler Editor +ASSEMBLER_HYPERLINK_DETECTOR_OPEN_BINARY_WITH_HEX_EDITOR=Öffnen mit Hex Editor +ASSEMBLER_HYPERLINK_DETECTOR_OPEN_BINARY_WITH_GRAPHICS_EDITOR=Öffnen mit Grafik Editor +ASSEMBLER_HYPERLINK_DETECTOR_OPEN_BINARY_WITH_DEFAULT_EDITOR=Öffnen mit Standardeditor +ASSEMBLER_HYPERLINK_DETECTOR_OPEN_BINARY_WITH_SYSTEM_EDITOR=Öffnen mit Systemeditor +ASSEMBLER_HYPERLINK_DETECTOR_OPEN_IDENTIFIER={0} {1} in Zeile {2} +ASSEMBLER_HYPERLINK_DETECTOR_OPEN_IDENTIFIER_IN_INCLUDE={0} {1} in Zeile {2} von {3} +ASSEMBLER_HYPERLINK_FILE_NOT_EXISTS=Ziel-Datei '{0}' existiert nicht. Möchten Sie die Ziel-Datei nun anlegen? + +ASSEMBLER_BREAKPOINT_MARKER_MESSAGE={0} [Zeile: {1}] - {2} +ASSEMBLER_BREAKPOINT_TOGGLE_TYPE_MENU_TEXT=Assembler Breakpoints + +COMPILER_SYMBOLS_VIEW_FILTER_TOOLTIP=Geben sie eine oder mehrere Teilzeichenfolgen durch Leerzeichen getrennt an +COMPILER_SYMBOLS_VIEW_SOURCE_LABEL=Quelle: +COMPILER_SYMBOLS_VIEW_SOURCE_NONE=Keine +COMPILER_SYMBOLS_VIEW_SOURCE_TOTAL_COUNT={0} Symbole +COMPILER_SYMBOLS_VIEW_SOURCE_FILTERED_COUNT={0} von {1} Symbolen +COMPILER_SYMBOLS_VIEW_TYPE_COLUMN_LABEL=Typ +COMPILER_SYMBOLS_VIEW_BANK_COLUMN_LABEL=Bank +COMPILER_SYMBOLS_VIEW_NAME_COLUMN_LABEL=Name +COMPILER_SYMBOLS_VIEW_HEX_VALUE_COLUMN_LABEL=Hexadezimalwert +COMPILER_SYMBOLS_VIEW_DECIMAL_VALUE_COLUMN_LABEL=Dezimalwert +COMPILER_SYMBOLS_VIEW_STRING_VALUE_COLUMN_LABEL=Textwert + +PREFERENCES_SYNTAX_HIGHLIGHTING_GROUP_TITLE=Syntax Hervorhebung + +PREFERENCES_TEXT_ATTRIBUTE_COMMENT_NAME=Kommentare +PREFERENCES_TEXT_ATTRIBUTE_STRING_NAME=Textkonstanten +PREFERENCES_TEXT_ATTRIBUTE_NUMBER_NAME=Zahlen +PREFERENCES_TEXT_ATTRIBUTE_DIRECTIVE=Direktive +PREFERENCES_TEXT_ATTRIBUTE_OPCODE_LEGAL_NAME=Opcodes +PREFERENCES_TEXT_ATTRIBUTE_OPCODE_ILLEGAL_NAME=Illegale Opcodes +PREFERENCES_TEXT_ATTRIBUTE_OPCODE_PSEUDO_NAME=Pseudo Opcodes +PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_EQUATE=Namen von Equates +PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_LABEL=Namen von Labels +PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_ENUM_DEFINITION_SECTION=Namen von Enums +PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_STRUCTURE_DEFINITION_SECTION=Namen von von Strukturen +PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_LOCAL_SECTION=Namen von Lokalen Bereichen +PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_MACRO_DEFINITION_SECTION=Namen von Macros +PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_PROCEDURE_DEFINITION_SECTION=Namen von Prozeduren + +PREFERENCES_EDITOR_GROUP_TITLE=Editor + +PREFERENCES_CONTENT_ASSIST_PROCESSOR_DEFAULT_CASE_LABEL=Standard Schreibweise für Befehlsvorschläge +PREFERENCES_CONTENT_ASSIST_PROCESSOR_DEFAULT_CASE_LOWER_CASE_TEXT=Kleingeschriebene Befehle +PREFERENCES_CONTENT_ASSIST_PROCESSOR_DEFAULT_CASE_UPPER_CASE_TEXT=Großgeschriebene Befehle + +PREFERENCES_COMPILE_COMMAND_POSITIONING_MODE_LABEL=Positionieren nach Kompilieren +PREFERENCES_COMPILE_COMMAND_POSITIONING_MODE_FIRST_ERROR_OR_WARNING_TEXT=Auf ersten Fehler oder erste Warnung +PREFERENCES_COMPILE_COMMAND_POSITIONING_MODE_FIRST_ERROR_TEXT=Auf ersten Fehler + +PREFERENCES_FOREGROUND_COLOR_LABEL=&Farbe: +PREFERENCES_BOLD_LABEL=F&ett +PREFERENCES_ITALIC_LABEL=&Kursiv + +PREFERENCES_DOWNLOAD_LINK=Download +PREFERENCES_DOWNLOAD_LINK_TOOL_TIP=Webseite {0} für Download öffnen + +PREFERENCES_COMPILER_CPU_LABEL=CP&U +PREFERENCES_COMPILER_EXECUTABLE_PATH_LABEL=Pfad zum &Kompiler +PREFERENCES_COMPILER_HARDWARE_ACTIVE_LABEL=Verwenden für {0} +PREFERENCES_COMPILER_DEFAULT_PARAMETERS_LABEL=&Standardparameter +PREFERENCES_COMPILER_PARAMETERS_LABEL=&Parameter +PREFERENCES_COMPILER_PARAMETERS_HELP=Verfügbare Variablen sind: +PREFERENCES_COMPILER_PARAMETERS_VARIABLES=${sourceFolderPath} Absoluter Pfad zum Quell-Ordner\n${sourceFilePath} Absoluter Pfad zur Quell-Datei\n${outputFolderPath} Absoluter Pfad zum Ausgabe-Ordner\n${outputFilePath} Absoluter Pfad der Ausgabe-Datei\n${outputFilePathWithoutExtension} Absoluter Pfad der Ausgabe-Datei ohne ihre Dateierweiterung\n${outputFileName} Name der Ausgabe-Datei inklusive ihrer Dateierweiterung\n${outputFileNameWithoutExtension} Name der Ausgabe-Datei ohne ihre Dateierweiterung\n${outputFileNameShortWithoutExtension} Name der Ausgabe-Datei ohne ihre Dateierweiterung, gekürzt auf 8 alpha-numerische Zeichen +PREFERENCES_COMPILER_OUTPUT_FOLDER_MODE_LABEL=&Ausgabe-Ordner +PREFERENCES_COMPILER_OUTPUT_FOLDER_MODE_SOURCE_FOLDER_TEXT=&Quell-Ordner +PREFERENCES_COMPILER_OUTPUT_FOLDER_MODE_TEMP_FOLDER_TEXT=&Temporärer Ordner +PREFERENCES_COMPILER_OUTPUT_FOLDER_MODE_FIXED_FOLDER_TEXT=&Fester Ordner +PREFERENCES_COMPILER_OUTPUT_FOLDER_PATH_LABEL=Ausgabe-O&rdner +PREFERENCES_COMPILER_OUTPUT_FILE_EXTENSION_LABEL=Ausgabe-&Dateierweiterung +PREFERENCES_COMPILER_RUNNER_ID_LABEL=Standardan&wendung zum Öffnen der Ausgabe-Datei +PREFERENCES_COMPILER_RUNNER_EXECUTABLE_PATH_LABEL=P&fad zur Anwendung +PREFERENCES_COMPILER_RUNNER_DEFAULT_COMMAND_LINE_LABEL=S&tandard-Kommandozeile +PREFERENCES_COMPILER_RUNNER_COMMAND_LINE_LABEL=Kommando&zeile +PREFERENCES_COMPILER_RUNNER_COMMAND_LINE_HELP=Verfügbare Variablen sind: +PREFERENCES_COMPILER_RUNNER_COMMAND_LINE_VARIABLES=${runnerExecutablePath} Absoluter Pfad zur Anwendung\n${sourceFolderPath} Absoluter Pfad zum Quell-Ordner\n${sourceFilePath} Absoluter Pfad zur Quell-Datei\n${outputFolderPath} Absoluter Pfad zum Ausgabe-Ordner\n${outputFilePath} Absoluter Pfad der Ausgabe-Datei\n${outputFilePathWithoutExtension} Absoluter Pfad der Ausgabe-Datei ohne ihre Dateierweiterung\n${outputFileName} Name der Ausgabe-Datei inklusive ihrer Dateierweiterung\n${outputFileNameWithoutExtension} Name der Ausgabe-Datei ohne ihre Dateierweiterung\n${outputFileNameShortWithoutExtension} Name der Ausgabe-Datei ohne ihre Dateierweiterung, gekürzt auf 8 alpha-numerische Zeichen +PREFERENCES_COMPILER_RUNNER_WAIT_FOR_COMPLETION_LABEL=Auf Ende der Anwendung warten + +TOC_WUDSN_IDE_LABEL=WUDSN IDE Handbuch + +TOC_IDE_TOPIC_LABEL=WUDSN IDE + +TOC_ASSEMBLERS_TOPIC_LABEL=Assembler +TOC_ASSEMBLER_GENERAL_TOPIC_LABEL=Allgemein +TOC_ASSEMBLER_NAME_LABEL=Name +TOC_ASSEMBLER_HOME_PAGE_LABEL=Home Page +TOC_ASSEMBLER_DEFAULT_HARDWARE_LABEL=Standardhardware +TOC_ASSEMBLER_SUPPORTED_CPUS_LABEL=Unterstütze CPUs +TOC_ASSEMBLER_DEFAULT_PARAMETERS_LABEL=Standardparameter +TOC_ASSEMBLER_INSTRUCTIONS_TOPIC_LABEL=Befehle +TOC_ASSEMBLER_INSTRUCTION_TYPE_DIRECTIVES_LABEL=Direktiven +TOC_ASSEMBLER_INSTRUCTION_TYPE_LEGAL_OPCODES_LABEL=Legale Opcodes +TOC_ASSEMBLER_INSTRUCTION_TYPE_PSEUDO_OPCODES_LABEL=Pseudo Opcodes +TOC_ASSEMBLER_INSTRUCTION_TYPE_ILLEGAL_OPCODES_LABEL=Illegale Opcodes +TOC_ASSEMBLER_INSTRUCTION_TYPE_W65816_OPCODES_LABEL=W65816 Opcodes +TOC_ASSEMBLER_INSTRUCTION_TYPE_LABEL=Typ +TOC_ASSEMBLER_INSTRUCTION_NAME_LABEL=Name +TOC_ASSEMBLER_INSTRUCTION_DESCRIPTION_LABEL=Beschreibung +TOC_ASSEMBLER_SYNTAX_YES=Ja +TOC_ASSEMBLER_SYNTAX_NO=Nein +TOC_ASSEMBLER_SYNTAX_BLOCK_DEFINITION_CHARACTERS=Zeichen für Block Definitionen +TOC_ASSEMBLER_SYNTAX_COMPLETION_PROPOSAL_AUTO_ACTIVATION_CHARACTERS=Zeichen für Vervollständigungsvorschlag +TOC_ASSEMBLER_SYNTAX_IDENTIFIER_PART_CHARACTERS=Zeichen für Namen +TOC_ASSEMBLER_SYNTAX_IDENTIFIER_SEPARATOR_CHARACTER=Trennzeichen für Namen +TOC_ASSEMBLER_SYNTAX_IDENTIFIER_START_CHARACTERS=Anfangszeichen für Namen +TOC_ASSEMBLER_SYNTAX_IDENTIFIERS_CASE_SENSITIVE=Namen unterscheiden Groß-/Kleinschreibung +TOC_ASSEMBLER_SYNTAX_INSTRUCTIONS_CASE_SENSITIVE=Instruktionen unterscheiden Groß-/Kleinschreibung +TOC_ASSEMBLER_SYNTAX_LABEL_DEFINITION_SUFFIX_CHARACTER=Suffixzeichen für Label Definitionen +TOC_ASSEMBLER_SYNTAX_MACRO_USAGE_PREFIX_CHARACTER=Zeichen für Verwendung von Makros +TOC_ASSEMBLER_SYNTAX_MULTIPLE_LINES_COMMENT_DELIMITERS=Trennzeichen für mehrzeilige Kommentare +TOC_ASSEMBLER_SYNTAX_SINGLE_LINE_COMMENT_DELIMITERS=Trennzeichen für einzeilige Kommentare +TOC_ASSEMBLER_SYNTAX_SOURCE_INCLUDE_DEFAULT_EXTENSION=Standard Datei-Erweiterung für Quell-Dateien +TOC_ASSEMBLER_SYNTAX_STRING_DELIMITERS=Trennzeichen für Textkonstanten +TOC_ASSEMBLER_MANUAL_TOPIC_LABEL=Handbuch + +TOC_HARDWARES_TOPIC_LABEL=Hardware Platformen +TOC_HARDWARE_NAME_LABEL=Hardware +TOC_HARDWARE_ID_LABEL=Identifikation +TOC_HARDWARE_ICON_LABEL=Ikone +TOC_HARDWARE_DEFAULT_FILE_EXTENSION_LABEL=Standard-Dateierweiterung +TOC_HARDWARE_EMULATOR_LABEL=Emulator +TOC_HARDWARE_HOME_PAGE_LABEL=Home Page +TOC_HARDWARE_DEFAULT_PARAMETERS_LABEL=Standardparameter + +TOC_CPUS_TOPIC_LABEL=CPUs +TOC_CPU_NAME_LABEL=CPU +TOC_CPU_OPCODE_LABEL=Opcode + +MESSAGE_E100=Pfad zur ausführbaren Datei des Kompilers '{0}' ist in den 'Assembler' Voreinstellungen nicht angegeben. +MESSAGE_E101=Der Kompiler '{0}' definiert keine Standardparameter. +MESSAGE_E102=Der Kompiler '{0}' getHelpFile keine Pfade zu Hilfe-Dateien. +MESSAGE_E103=Pfad zur ausführbaren Datei des Kompilers '{0}' in den Voreinstellungen verweist auf die nicht existierende Datei '{1}'. +MESSAGE_E104=Ausgabe-Dateierweiterung muss in den Voreinstellungen des Kompilers '{0}' oder mit der Annotation '{1}' angegeben werden. +MESSAGE_E105=Kompilerprozess '{0}' kann im Arbeitsverzeichnis '{1}' nicht ausgeführt werden. Systemfehler: {2} +MESSAGE_E106=Ausgabe-Datei '{0}' kann nicht zum Schreiben geöffnet werden. Beenden Sie alle Anwendungen welche die Datei geöffnet haben könnten. +MESSAGE_E107=Ausgabe-Datei wurde nicht erzeugt. Prüfen Sie die Fehlermeldungen und die Konsolenausgabe. +MESSAGE_E108=Ausgabe-Datei wurde nicht aktualisiert. Prüfen Sie die Fehlermeldungen und die Konsolenausgabe. +MESSAGE_I109=Ausgabe-Datei '{0}' mit einer Größe von {1} (${2}) Bytes erzeugt. +MESSAGE_I110=Symbol-Datei '{0}' mit {1} Symbolen erzeugt. +MESSAGE_E111=Symbol-Datei '{0}' kann nicht zum Schreiben geöffnet werden. Systemfehler: {1} +MESSAGE_E112=Pfad zur ausführbaren Datei des Emulators ist in den Voreinstellungen des Emulators '{0}' nicht angegeben. +MESSAGE_E113=Prozess '{1}' for den Emulator '{0}' kann im Arbeitsverzeichnis '{1}' nicht ausgeführt werden. Systemfehler: {2} +MESSAGE_E114=Pfad zur ausführbaren Datei des Anwendung '{0}' in der Voreinstellungen verweist auf die nicht existierende Datei '{1}'. +MESSAGE_E115=Ausgabe-Datei '{0}' kann nicht geöffnet werden, da keine Standardanwendung für die Dateierweiterung '{1}' registiert ist. +MESSAGE_E116=Definition für die Anwendung '{0}' aus den Voreinstellungen für die Hardware '{1}' ist nicht registriert. +MESSAGE_E117=Symbol-Datei '{0}' ist leer, aber kann nicht gelöscht werden. +MESSAGE_I118=Ausgabe-Datei '{0}' wird mit Anwendung '{1}' geöffnet. +MESSAGE_E119=Ausgabe-Datei '{0}' kann nicht mit Anwendung '{1}' geöffnet werden. +MESSAGE_W120=Unterbrechungspunkte werden ignoriert, da die Anwendung '{0}' die Übergabe von Quelltext-Unterbrechungspunkte nicht unterstützt. +MESSAGE_I121=Unterbrechungspunkte-Datei '{0}' mit {1} aktiven Unterbrechungspunkten erzeugt. +MESSAGE_E122=Unterbrechungspunkte-Datei '{0}' kann nicht zum Schreiben geöffnet werden. Systemfehler: {1} +MESSAGE_E123=Unterbrechungspunkte-Datei '{0}' ist leer, aber kann nicht gelöscht werden. +MESSAGE_E124=Unbekannte Hardware '{0}'. Geben Sie einen der folgenden gültigen Werte an '{1}'. +MESSAGE_E125=Haupt-Quell-Datei '{0}' existiert nicht. +MESSAGE_E126=Ausgabe-Datei erzeugt is aber leer. Prüfen Sie die Fehlermeldungen und die Konsolenausgabe. +MESSAGE_E127=Kompilerprozess endete mit dem Rückgabewert {0}. Prüfen Sie die Fehlermeldungen und die Konsolenausgabe. +MESSAGE_E128=Hardware nicht angegeben. Gültige Werte sind '{0}'. +MESSAGE_E129=Haupt-Quell-Datei spezifiziert oder verwendet die Hardware '{0}' während die Include-Quell-Datei die Hardware '{1}' spezifiziert oder verwendet. +MESSAGE_E130=Handbuch zum Kompiler '{0}' kann nicht angezeigt werden, da der Pfad zur ausführbaren Datei des Kompilers in den Voreinstellungen nicht angegeben ist. +MESSAGE_E131=Handbuch zum Kompiler '{0}' kann nicht angezeigt werden, da keine Hilfe-Datei in den Pfaden '{1}' relativ zur ausführbaren Datei '{2}' des Kompilers gefunden wurde. +MESSAGE_E132=Disketten-Image-Datei '{0}' existiert nicht. Lege ein bootfähiges Disketten-Image an, auf dem die Ausgabe-Datei '{1}' gespeichert werden kann. +MESSAGE_E133=Disketten-Image-Datei '{0}' ist nicht schreibbar. Mache das Diskettenimage schreibbbar, damit die Ausgabe-Datei '{1}' gespeichert werden kann. +MESSAGE_E134=Disketten-Image-Datei '{0}' kann not nicht zum Lesen geöffnet werden. Systemfehler: {1} +MESSAGE_E135=Disketten-Image-Datei '{0}' enthält kein gültiges Dateisystem. Stelle sicher, dass das Diskettenimage korrekt formatiert ist, damit die Ausgabe-Datei '{1}' gespeichert werden kann. +MESSAGE_E136=Disketten-Image-Datei '{0}' ist voll. Systemfehler: {1} +MESSAGE_E137=Disketten-Image-Datei '{0}' kann nicht zum Schreiben geöffnet werden. Systemfehler: {1} +MESSAGE_E138=Ausgabe-Datei '{0}' has eine nicht unterstütze Dateierweiterunge oder ungültigen Inhalt. Unterstützte Dateierweiterungen sind ".b" (Apple II Binär-Datei im Format [start, length, code]), ".prg" (C64 Programm-Datei im Format [start, code]) und ".xex" (Atari Compound-Datei im Format [$ffff, start,end,code]). +MESSAGE_E139=Ausgabe-Dateierweiterung '{0}' muss mit "." beginnen. +MESSAGE_E140=Ausgabe-Ordnermodus muss in den Voreinstellungen des Kompilers '{0}' oder mit der Annotation '{1}' angegeben werden. +MESSAGE_E141=Unbekannter Ausgabe-Ordnermodus '{0}'. Geben Sie einen der folgenden gültigen Werte an '{1}'. +MESSAGE_E142=Include Anweisung für die Datei '{0}' verwendet im Dateinamen eine andere Groß-Kleinschreibung als der echte Dateiname {1} auf dem Dateisystem. Korrigieren Sie den Dateinamen in der Include Anweisung. +MESSAGE_S143=In Include-Datei '{0}', Zeile {1}. \ No newline at end of file diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/Compiler.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/Compiler.java new file mode 100644 index 00000000..671ef52f --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/Compiler.java @@ -0,0 +1,98 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler; + +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; + +/** + * Base class for compiler implementations. Sub classes have to be stateless. + * + * @author Peter Dell + */ +public abstract class Compiler { + + // See {@link CompilerId} for predefined ids. + private CompilerDefinition definition; + + /** + * Creation is protected. + */ + protected Compiler() { + + } + + /** + * Sets the definition of the compiler. Called by {@link CompilerRegistry} + * only. + * + * @param definition + * The definition if the compiler, not null. + */ + final void setDefinition(CompilerDefinition definition) { + if (definition == null) { + throw new IllegalArgumentException("Parameter 'type' must not be null."); + } + this.definition = definition; + } + + /** + * Gets the definition of the compiler. + * + * @return The definition of the compiler, not null. + */ + public final CompilerDefinition getDefinition() { + if (definition == null) { + throw new IllegalStateException("Field 'definition' must not be null."); + } + return definition; + } + + /** + * Creates a compiler source parser. + * + * @return The compiler source parser, not null. + */ + public abstract CompilerSourceParser createSourceParser(); + + /** + * Checks if the exit code of the compiler process represents success. By + * default 0 is interpreted as success, but a compiler may + * override this. + * + * @param exitValue + * The exit code of the compiler process. + * @return true if the exit code represents success (only + * information and warning messages) or a failure (at least one error + * message). + * + * @since 1.7.0 + */ + public boolean isSuccessExitValue(int exitValue) { + return exitValue == 0; + } + + /** + * Creates the parser to for the compiler output. + * + * @return The parser to for the compiler output, not null. + */ + public abstract CompilerProcessLogParser createLogParser(); + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/Compiler.xml b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/Compiler.xml new file mode 100644 index 00000000..9ca3c1ab --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/Compiler.xml @@ -0,0 +1,113 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerConsole.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerConsole.java new file mode 100644 index 00000000..adcfa0da --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerConsole.java @@ -0,0 +1,102 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler; + +import java.io.PrintStream; + +import org.eclipse.ui.console.ConsolePlugin; +import org.eclipse.ui.console.IConsole; +import org.eclipse.ui.console.IConsoleManager; +import org.eclipse.ui.console.IConsoleView; +import org.eclipse.ui.console.MessageConsole; +import org.eclipse.ui.console.MessageConsoleStream; + +import com.wudsn.ide.asm.Texts; + +/** + * The console to show the user the output from the compiler. + * + * @author Peter Dell + * @author Daniel Mitte + */ +public final class CompilerConsole { + + private IConsoleManager consoleManager; + public MessageConsole console; + + private MessageConsoleStream messageStream; + private PrintStream printStream; + + /** + * Create a new console-window. + * + */ + public CompilerConsole() { + consoleManager = ConsolePlugin.getDefault().getConsoleManager(); + console = new MessageConsole(Texts.COMPILER_CONSOLE_TITLE, null); + consoleManager.addConsoles(new IConsole[] { console }); + + messageStream = console.newMessageStream(); + messageStream.setActivateOnWrite(false); + messageStream.print(""); + printStream = new PrintStream(messageStream); + } + + /** + * Brings this console view instance to front in the console view editor + * part. + * + * @param consoleView + * The console view editor part, not null. + */ + + public void display(IConsoleView consoleView) { + if (consoleView == null) { + throw new IllegalArgumentException( + "Parameter 'consoleView' must not be null."); + } + consoleView.display(console); + + } + + /** + * Add a line to console. + * + * @param message + * The message to print, not null. + */ + public void println(String message) { + if (message == null) { + throw new IllegalArgumentException( + "Parameter 'message' must not be null."); + } + messageStream.println(message); + } + + /** + * Gets a print messageStream to write to this console. + * + * @return The print messageStream, not null. + */ + public PrintStream getPrintStream() { + return printStream; + } + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerDefinition.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerDefinition.java new file mode 100644 index 00000000..ec6cb7e3 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerDefinition.java @@ -0,0 +1,436 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler; + +import java.io.File; +import java.util.List; +import java.util.Locale; +import java.util.StringTokenizer; + +import org.eclipse.core.runtime.CoreException; +import org.eclipse.core.runtime.IStatus; +import org.eclipse.core.runtime.Status; + +import com.wudsn.ide.asm.AssemblerPlugin; +import com.wudsn.ide.asm.CPU; +import com.wudsn.ide.asm.Hardware; +import com.wudsn.ide.asm.Texts; +import com.wudsn.ide.asm.compiler.syntax.CompilerSyntax; +import com.wudsn.ide.base.common.FileUtility; +import com.wudsn.ide.base.common.StringUtility; +import com.wudsn.ide.base.common.TextUtility; + +/** + * Definition of a compiler. The definition contains all static meta information + * about the compiler. It is normally defined via an extension. The id of a + * compiler must be unique across all compilers. + * + * @author Peter Dell + */ +public final class CompilerDefinition implements Comparable { + + // Id + private String id; + private String name; + private String className; + + // Installation and use. + private String helpFilePaths; + private String homePageURL; + + // Editing and source parsing. + private List supportedCPUs; + private CompilerSyntax syntax; + + // Compiling. + private String defaultParameters; + + // Assignment to default hardware type. + private Hardware defaultHardware; + + /** + * Creation is package local. Called by {@link CompilerRegistry} only. + */ + CompilerDefinition() { + + } + + /** + * Sets the id of the compiler. Called by {@link CompilerRegistry} only. + * + * @param id + * The id of the compiler, not empty and not null. + */ + final void setId(String id) { + if (id == null) { + throw new IllegalArgumentException("Parameter 'id' must not be null."); + } + if (StringUtility.isEmpty(id)) { + throw new IllegalArgumentException("Parameter 'id' must not be empty."); + } + this.id = id; + } + + /** + * Gets the id of the compiler. + * + * @return The id of the compiler, not empty and not null. + */ + public final String getId() { + if (id == null) { + throw new IllegalStateException("Field 'id' must not be null."); + } + return id; + } + + /** + * Sets the localized name of the compiler. Called by + * {@link CompilerRegistry} only. + * + * @param name + * The localized name of the compiler, not empty and not + * null. + */ + final void setName(String name) { + if (name == null) { + throw new IllegalArgumentException("Parameter 'name' must not be null."); + } + if (StringUtility.isEmpty(id)) { + throw new IllegalArgumentException("Parameter 'name' must not be empty."); + } + this.name = name; + } + + /** + * Gets the localized name of the compiler. + * + * @return The localized name of the compiler, not empty and not + * null. + */ + public final String getName() { + if (name == null) { + throw new IllegalStateException("Field 'name' must not be null."); + } + return name; + } + + /** + * Sets the class name of the compiler. Called by {@link CompilerRegistry} + * only. + * + * @param className + * The class name of the compiler, not empty and not + * null. + */ + final void setClassName(String className) { + if (className == null) { + throw new IllegalArgumentException("Parameter 'className' must not be null."); + } + this.className = className; + } + + /** + * Gets the class name of the compiler. + * + * @return The class name of the compiler, not empty and not + * null. + */ + public final String getClassName() { + if (className == null) { + throw new IllegalStateException("Field 'className' must not be null."); + } + return className; + } + + /** + * Sets the absolute URL of the home page where the compiler can be + * downloaded. Called by {@link CompilerRegistry} only. + * + * @param homePageURL + * The absolute URL of the home page where the compiler can be + * downloaded. May be empty or null. + */ + final void setHomePageURL(String homePageURL) { + if (homePageURL == null) { + homePageURL = ""; + } + this.homePageURL = homePageURL; + } + + /** + * Gets the absolute URL of the home page where the compiler can be + * downloaded. + * + * @return The absolute URL of the home page where the compiler can be + * downloaded. The result may be empty or null. + */ + public final String getHomePageURL() { + if (homePageURL == null) { + homePageURL = ""; + } + return homePageURL; + } + + /** + * Sets the help file paths to locate the help file for the compiler. Called + * by {@link CompilerRegistry} only. + * + * @param helpFilePaths + * The relative file path to locate the help file for the + * compiler based on the folder of the executable. ".", ".." and + * "/" may be used to specify the path. A path may end with a + * language in the form "(en)". Multiple paths are separated by + * ",". May be empty or null. + */ + final void setHelpFilePaths(String helpFilePaths) { + if (helpFilePaths == null) { + helpFilePaths = ""; + } + this.helpFilePaths = helpFilePaths; + } + + /** + * Gets the help file paths to locate the help file for the compiler. + * + * @return The relative file path to locate the help file for the compiler + * based on the folder of the executable. ".", ".." and "/" may be + * used to specify the path. A path may end with a language in the + * form "(en)".Multiple paths are separated by ",". The result may + * be empty, not null. + */ + public final String getHelpFilePaths() { + if (helpFilePaths == null) { + throw new IllegalStateException("Field 'helpFilePaths' must not be null."); + } + return helpFilePaths; + } + + /** + * Determines if this compiler offers a help file at all. + * + * @return true if this compiler offers a help file, + * false otherwise. + */ + public final boolean hasHelpFile() { + return StringUtility.isSpecified(helpFilePaths); + } + + /** + * Gets the help file for the compiler. This includes locating the most + * appropriate file in the file system based on the current locale. + * + * @param compilerExecutablePath + * the compiler executable path, may be empty, not + * null. + * @return The help file, or null not help file could be found. + * + * @throws CoreException + * if the compilerExecutablePath is empty, the compiler does not + * specify a help path or no help file can be found. + */ + public final File getHelpFile(String compilerExecutablePath) throws CoreException { + if (compilerExecutablePath == null) { + throw new IllegalArgumentException("Parameter 'compilerExecutablePath' must not be null."); + } + if (StringUtility.isEmpty(compilerExecutablePath)) { + // ERROR: Help for the '{0}' compiler cannot be + // displayed because the path to the compiler executable + // is not set in the preferences. + throw new CoreException(new Status(IStatus.ERROR, AssemblerPlugin.ID, TextUtility.format( + Texts.MESSAGE_E130, name))); + } + if (!hasHelpFile()) { + // ERROR: The compiler '{0}' does not specify a help file path. + throw new CoreException(new Status(IStatus.ERROR, AssemblerPlugin.ID, TextUtility.format( + Texts.MESSAGE_E102, name))); + } + + String localeLanguage = Locale.getDefault().getLanguage(); + File firstFile = null; + File firstLanguageFile = null; + StringTokenizer tokenizer = new StringTokenizer(helpFilePaths, ","); + while (tokenizer.hasMoreTokens()) { + String helpFilePath = tokenizer.nextToken().trim(); + String helpFileLanguage = ""; + int index = helpFilePath.lastIndexOf("("); + if (index > 0) { + helpFileLanguage = helpFilePath.substring(index + 1, index + 3); + helpFilePath = helpFilePath.substring(0, index - 1).trim(); + } + File file = FileUtility.getCanonicalFile(new File(new File(compilerExecutablePath).getParent(), + helpFilePath)); + if (file.exists()) { + if (firstFile == null) { + firstFile = file; + } + if (firstLanguageFile == null && helpFileLanguage.equals(localeLanguage)) { + firstLanguageFile = file; + } + } + } + // Use language specific file if present, use first file otherwise. + File result = firstLanguageFile; + if (result == null) { + result = firstFile; + } + if (result == null) { + // ERROR: Help for the '{0}' compiler cannot be displayed because no + // help file was found in the paths '{1}' for the compiler + // executable path '{0}'. + throw new CoreException(new Status(IStatus.ERROR, AssemblerPlugin.ID, TextUtility.format( + Texts.MESSAGE_E131, name, helpFilePaths, compilerExecutablePath))); + } + return result; + + } + + /** + * Sets the list of supported CPUs Called by {@link CompilerRegistry} only. + * + * @param supportedCPUs + * The unmodifiable list of supported CPUs, not empty and not + * null. + * @since 1.6.1 + */ + final void setSupportedCPUs(List supportedCPUs) { + if (supportedCPUs == null) { + throw new IllegalArgumentException("Parameter 'supportedCPUs' must not be null."); + } + if (supportedCPUs.isEmpty()) { + throw new IllegalArgumentException("Parameter 'supportedCPUs' must not be empty."); + } + this.supportedCPUs = supportedCPUs; + } + + /** + * Gets the unmodifiable list of CPUs supported by this compiler. The first + * entry defines the default CPU. + * + * @return The unmodifiable list of CPUs supported by this compiler, not + * empty and, not null. + * + * @since 1.6.1 + */ + public final List getSupportedCPUs() { + if (supportedCPUs == null) { + throw new IllegalStateException("Field 'supportedCPUs' must not be null."); + } + return supportedCPUs; + } + + /** + * Sets the compiler syntax. Called by {@link CompilerRegistry} only. + * + * @param syntax + * The compiler syntax, not null. + */ + final void setSyntax(CompilerSyntax syntax) { + if (syntax == null) { + throw new IllegalArgumentException("Parameter 'syntax' must not be null."); + } + this.syntax = syntax; + } + + /** + * Gets the compiler syntax. + * + * @return The compiler syntax, not null. + */ + public final CompilerSyntax getSyntax() { + if (syntax == null) { + throw new IllegalStateException("Field 'syntax' must not be null."); + } + return syntax; + } + + /** + * Sets the compiler default parameters. Called by {@link CompilerRegistry} + * only. + * + * @param defaultParameters + * The compiler default parameters, not null. + */ + final void setDefaultParameters(String defaultParameters) { + if (defaultParameters == null) { + throw new IllegalArgumentException("Parameter 'defaultParameters' must not be null."); + } + this.defaultParameters = defaultParameters; + } + + /** + * Gets the compiler default parameters. + * + * @return The compiler default parameters, not null. + */ + public final String getDefaultParameters() { + if (defaultParameters == null) { + throw new IllegalStateException("Field 'defaultParameters' must not be null."); + } + return defaultParameters; + } + + /** + * Sets the default hardware to be assumed for this compiler. Called by + * {@link CompilerRegistry} only. + * + * @param hardware + * The default hardware, not null. + */ + final void setDefaultHardware(Hardware hardware) { + if (hardware == null) { + throw new IllegalArgumentException("Parameter 'hardware' must not be null."); + } + this.defaultHardware = hardware; + } + + /** + * Gets the default hardware for this compiler. + * + * @return The default hardwares, not null. + */ + public final Hardware getDefaultHardware() { + if (defaultHardware == null) { + throw new IllegalStateException("Field 'defaultHardware' must not be null."); + } + return defaultHardware; + } + + /** + * See {@link Comparable}. + */ + @Override + public int compareTo(CompilerDefinition o) { + if (o == null) { + throw new IllegalArgumentException("Parameter 'o' must not be null."); + } + if (id == null || o.id == null) { + if (id == null) { + throw new IllegalStateException("Field 'id' must not be null for this or for argument."); + } + } + return id.compareTo(o.id); + } + + @Override + public String toString() { + return id; + } + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerFileWriter.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerFileWriter.java new file mode 100644 index 00000000..2ed2727a --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerFileWriter.java @@ -0,0 +1,39 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler; + +public class CompilerFileWriter { + /** + * Creates or updates a disk image file with the output file. + * + * @param files The compiler files, not null. + * + * @return true if no disk image is required or it is required and has been updated. + * + * @since 1.6.3 + */ + public boolean createOrUpdateDiskImage(CompilerFiles files) { + if (files == null) { + throw new IllegalArgumentException( + "Parameter 'outputFile' must not be null."); + } + return true; + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerFiles.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerFiles.java new file mode 100644 index 00000000..cde61fe2 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerFiles.java @@ -0,0 +1,256 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler; + +import java.io.File; + +import org.eclipse.core.resources.IFile; + +import com.wudsn.ide.asm.AssemblerProperties; +import com.wudsn.ide.asm.AssemblerProperties.AssemblerProperty; +import com.wudsn.ide.asm.preferences.CompilerPreferences; +import com.wudsn.ide.base.common.FileUtility; +import com.wudsn.ide.base.common.StringUtility; + +/** + * Container class for the folder, file names and paths of the source file, the + * output file and the symbols file. + * + * @author Peter Dell + * + */ +public final class CompilerFiles { + + public final class SourceFile { + + public final IFile iFile; + + public final File folder; + public final String folderPath; + + public final File file; + public final String filePath; + public final String fileName; + public final String fileNameWithoutExtension; + + public final AssemblerProperties assemblerProperties; + + SourceFile(IFile iFile, AssemblerProperties assemblerProperties) { + + if (iFile == null) { + throw new IllegalArgumentException("Parameter 'iFile' must not be null."); + } + if (assemblerProperties == null) { + throw new IllegalArgumentException("Parameter 'assemblerProperties' must not be null."); + } + + this.iFile = iFile; + this.assemblerProperties = assemblerProperties; + + // Source file. + filePath = iFile.getLocation().toOSString(); + file = FileUtility.getCanonicalFile(new File(filePath)); + fileName = file.getName(); + + String extension = iFile.getFileExtension(); + if (extension == null) { + extension = ""; + } + fileNameWithoutExtension = fileName.substring(0, fileName.length() - extension.length() - 1); + + // Source folder. + folder = file.getParentFile(); + folderPath = folder.getPath(); + } + } + + /** + * The actual source file which is currently open. + */ + public final SourceFile sourceFile; + + /** + * The main source file which is either the current file or the file + * indicated by the property "@com.wudsn.ide.asm.editor.MainSourceFile". + */ + public final SourceFile mainSourceFile; + + public final AssemblerProperty outputFolderModeProperty; + public final String outputFolderMode; + + public final AssemblerProperty outputFolderProperty; + public final File outputFolder; + public final String outputFolderPath; + + public final File outputFile; + public final String outputFilePath; + public final String outputFilePathWithoutExtension; + + public final AssemblerProperty outputFileProperty; + public final String outputFileName; + public final String outputFileNameWithoutExtension; + public final AssemblerProperty outputFileExtensionProperty; + public final String outputFileExtension; + public final String outputFileNameShortWithoutExtension; + + public final File symbolsFile; + public final String symbolsFilePath; + public final String symbolsFileName; + + public CompilerFiles(IFile mainSourceIFile, AssemblerProperties mainSourceFileAssemblerProperties, + IFile sourceIFile, AssemblerProperties sourceFileAssemblerProperties, + CompilerPreferences compilerPreferences) { + + if (mainSourceIFile == null) { + throw new IllegalArgumentException("Parameter 'mainSourceIFile' must not be null."); + } + if (sourceIFile == null) { + throw new IllegalArgumentException("Parameter 'sourceIFile' must not be null."); + } + if (compilerPreferences == null) { + throw new IllegalArgumentException("Parameter 'compilerPreferences' must not be null."); + } + this.mainSourceFile = new SourceFile(mainSourceIFile, mainSourceFileAssemblerProperties); + this.sourceFile = new SourceFile(sourceIFile, sourceFileAssemblerProperties); + + // Output folder mode + // Can be overridden via annotation property in main source file + String localOutputFolderPath = compilerPreferences.getOutputFolderPath(); + String localOutputFolderMode = compilerPreferences.getOutputFolderMode(); + String localOutputFileExtension = compilerPreferences.getOutputFileExtension(); + + // Properties which override the preferences + outputFolderModeProperty = mainSourceFileAssemblerProperties.get(AssemblerProperties.OUTPUT_FOLDER_MODE); + outputFolderProperty = mainSourceFileAssemblerProperties.get(AssemblerProperties.OUTPUT_FOLDER); + outputFileExtensionProperty = mainSourceFileAssemblerProperties.get(AssemblerProperties.OUTPUT_FILE_EXTENSION); + outputFileProperty = mainSourceFileAssemblerProperties.get(AssemblerProperties.OUTPUT_FILE); + + // The following sequence sets the instance fields "outputFolder" and + // "outputFileNameWithoutExtension" as well as the + // "outputFileNameWithoutExtension". + // If the output file is specified explicitly, it overrides all output + // properties. + if (outputFileProperty != null) { + + // Make the file an absolute file. + File file = new File(outputFileProperty.value); + if (!file.isAbsolute()) { + file = new File(mainSourceFile.file.getParentFile(), file.getPath()); + } + file = FileUtility.getCanonicalFile(file); + + outputFolderMode = CompilerOutputFolderMode.FIXED_FOLDER; + outputFolder = file.getParentFile(); + + // Split the file name and file extension. + String fileName = file.getName(); + int index = fileName.lastIndexOf('.'); + if (index > 0) { + outputFileNameWithoutExtension = fileName.substring(0, index); + localOutputFileExtension = fileName.substring(index); + } else { + outputFileNameWithoutExtension = fileName; + localOutputFileExtension = ""; + } + } else { + // The output file extension is independent of the rest. + if (outputFileExtensionProperty != null) { + localOutputFileExtension = outputFileExtensionProperty.value; + } + // If the output folder mode is specified explicitly, it overrides + // the output + // folder mode preferences. + if (outputFolderModeProperty != null) { + localOutputFolderMode = outputFolderModeProperty.value; + } + + // If the output folder is specified explicitly, it overrides the + // output folder mode and folder preferences. + if (outputFolderProperty != null) { + localOutputFolderMode = CompilerOutputFolderMode.FIXED_FOLDER; + localOutputFolderPath = outputFolderProperty.value; + } + + if (localOutputFolderMode.equals(CompilerOutputFolderMode.SOURCE_FOLDER)) { + localOutputFolderPath = mainSourceFile.folderPath; + } else if (localOutputFolderMode.equals(CompilerOutputFolderMode.FIXED_FOLDER)) { + // Fallback + if (StringUtility.isEmpty(localOutputFolderPath)) { + localOutputFolderPath = System.getProperty("java.io.tmpdir"); + } + } else { + localOutputFolderPath = System.getProperty("java.io.tmpdir"); + } + + File file = new File(localOutputFolderPath); + if (!file.isAbsolute()) { + file = new File(sourceFile.file, file.getPath()); + } + file = FileUtility.getCanonicalFile(file); + + outputFolderMode = localOutputFolderMode; + outputFolder = file; + + // Output file. + outputFileNameWithoutExtension = mainSourceFile.fileNameWithoutExtension; + } + + // Common output parts. + outputFolderPath = outputFolder.getPath(); + outputFileNameShortWithoutExtension = getFileNameShort(outputFileNameWithoutExtension); + outputFileName = outputFileNameWithoutExtension + localOutputFileExtension; + outputFileExtension = localOutputFileExtension; + outputFile = new File(outputFolder, outputFileName); + outputFilePath = outputFile.getPath(); + outputFilePathWithoutExtension = new File(outputFolder, outputFileNameWithoutExtension).getPath(); + + // Symbols file. + symbolsFileName = mainSourceFile.fileNameWithoutExtension + ".lbl"; + symbolsFile = new File(outputFolder, symbolsFileName); + symbolsFilePath = symbolsFile.getPath(); + + } + + /** + * Computes a short file consisting of at most 8 ASCII letters and digits + * which starts with a letter. + * + * @param fileName + * The file name, may be empty, not null. + * @return The short file, name, not empty and not null. + */ + private String getFileNameShort(String fileName) { + if (fileName == null) { + throw new IllegalArgumentException("Parameter 'fileName' must not be null."); + } + StringBuilder builder = new StringBuilder(fileName); + for (int i = 0; i < fileName.length() && builder.length() < 8; i++) { + char c = fileName.charAt(i); + c = Character.toUpperCase(c); + if (c >= 'A' && c <= 'Z' && (builder.length() > 0) || c >= '0' && c <= '9') { + builder.append(c); + } + } + if (builder.length() == 0) { + return "UNKNOWN"; + } + return builder.toString(); + } +} \ No newline at end of file diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerOutputFolderMode.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerOutputFolderMode.java new file mode 100644 index 00000000..7991b23e --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerOutputFolderMode.java @@ -0,0 +1,73 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler; + +import com.wudsn.ide.base.common.StringUtility; + +/** + * The target directory mode controls which folder is use to create the binary + * file in. + * + * @author Peter Dell + */ +public final class CompilerOutputFolderMode { + + /** + * Creation is private. + */ + private CompilerOutputFolderMode() { + } + + public final static String SOURCE_FOLDER = "SOURCE_FOLDER"; + public final static String TEMP_FOLDER = "TEMP_FOLDER"; + public final static String FIXED_FOLDER = "FIXED_FOLDER"; + + /** + * Determines if the output folder mode is defined. + * + * @param outputFolderMode + * The output folder mode, not null. + * @return true if the output folder mode is not specified or + * defined. + */ + public static boolean isDefined(String outputFolderMode) { + if (outputFolderMode == null) { + throw new IllegalArgumentException("Parameter 'outputFolderMode' must not be null."); + } + if (StringUtility.isEmpty(outputFolderMode)) { + return true; + } + if (outputFolderMode.equals(SOURCE_FOLDER) || outputFolderMode.equals(TEMP_FOLDER) + || outputFolderMode.equals(FIXED_FOLDER)) { + return true; + } + return false; + } + + /** + * Gets the comma separated list of allowed values. + * + * @return The comma separated list of allowed values, not null + * . + */ + public static String getAllowedValues() { + return SOURCE_FOLDER + ", " + TEMP_FOLDER + ", " + FIXED_FOLDER; + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerProcessLogParser.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerProcessLogParser.java new file mode 100644 index 00000000..69dbc158 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerProcessLogParser.java @@ -0,0 +1,286 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler; + +import java.io.File; +import java.util.List; + +import org.eclipse.core.resources.IFile; +import org.eclipse.core.resources.IMarker; +import org.eclipse.core.resources.IResource; +import org.eclipse.core.runtime.CoreException; +import org.eclipse.core.runtime.Path; + +import com.wudsn.ide.asm.AssemblerPlugin; +import com.wudsn.ide.asm.Texts; +import com.wudsn.ide.base.common.NumberUtility; +import com.wudsn.ide.base.common.TextUtility; + +/** + * Base class for compiler process log parsing. + * + * @author Peter Dell + */ +public abstract class CompilerProcessLogParser { + + /** + * Proxy class for creating instances of {@link IMarker}. + * + * @since 1.6.1 + */ + public static final class Marker { + private IFile iFile; + private int lineNumber; + private int severity; + private String message; + private Marker detailMarker; + + Marker(IFile iFile, int lineNumber, int severity, String message, Marker detailMarker) { + if (iFile == null) { + throw new IllegalArgumentException("Parameter 'iFile' must not be null."); + } + if (message == null) { + throw new IllegalArgumentException("Parameter 'message' must not be null."); + } + this.iFile = iFile; + this.lineNumber = lineNumber; + this.severity = severity; + this.message = message; + this.detailMarker = detailMarker; + } + + /** + * Gets the iFile the marker refers to. + * + * @return The iFile, not null. + */ + public IFile getIFile() { + return iFile; + } + + /** + * Gets the line number. + * + * @return The liner number or 0, if there is no known line number. + */ + public int getLineNumber() { + return lineNumber; + } + + /** + * Gets the severity. + * + * @return The severity, see {@link IMarker#SEVERITY}. + */ + public int getSeverity() { + return severity; + } + + /** + * Gets the message. + * + * @return The message, may be empty, not null. + */ + public String getMessage() { + return message; + } + + /** + * Gets the detail marker, describing this marker in more detail. + * + * @return The detail marker or null. + */ + public Marker getDetailMarker() { + return detailMarker; + } + + @Override + public String toString() { + return iFile.getFullPath() + ":" + lineNumber + ":" + severity + ":" + message; + } + + @Override + public boolean equals(Object o) { + if (o == null || !(o instanceof Marker)) { + return false; + } + Marker other = (Marker) o; + return iFile.getFullPath().equals(other.iFile.getFullPath()) + && lineNumber == other.lineNumber + && severity == other.severity + && message.equals(other.message) + && ((detailMarker == null && other.detailMarker == null) || detailMarker.equals(other.detailMarker)); + + } + + @Override + public int hashCode() { + return lineNumber; + + } + } + + private boolean initialized; + protected CompilerFiles files; + protected String mainSourceFilePath; + protected String outputLog; + protected String errorLog; + protected boolean markerAvailable; + protected String filePath; + protected int lineNumber; + protected int severity; + protected String message; + + protected CompilerProcessLogParser() { + } + + public final void setLogs(CompilerFiles files, String outputLog, String errorLog) { + if (files == null) { + throw new IllegalArgumentException("Parameter 'files' must not be null."); + } + if (outputLog == null) { + throw new IllegalArgumentException("Parameter 'outputLog' must not be null."); + } + if (errorLog == null) { + throw new IllegalArgumentException("Parameter 'errorLog' must not be null."); + } + this.files = files; + this.mainSourceFilePath = files.mainSourceFile.filePath; + this.outputLog = outputLog; + this.errorLog = errorLog; + initialize(); + initialized = true; + markerAvailable = false; + return; + } + + protected void initialize() { + } + + public final boolean nextMarker() { + filePath = ""; + lineNumber = 0; + severity = 0; + message = ""; + markerAvailable = false; + findNextMarker(); + return markerAvailable; + } + + protected abstract void findNextMarker(); + + /** + * Adds the compiler symbols from the process output to the specified list. + * + * @param list + * The modifiable list to which the compiler symbols shall be + * added, not null. + * + * @throws CoreException + * if the symbols information is present, but cannot be read or + * parsed. + */ + public void addCompilerSymbols(List list) throws CoreException { + } + + /** + * Creates a new marker proxy for a file. + * + * @return The marker proxy, not null. + * + * @since 1.6.1 + */ + public final Marker getMarker() { + + if (!initialized) { + throw new IllegalStateException("No log set."); + } + if (!markerAvailable) { + throw new IllegalStateException("No marker available."); + } + IFile mainSourceIFile = files.mainSourceFile.iFile; + IFile iFile = mainSourceIFile; + String normalizedFilePath = filePath.replace(File.separatorChar, '/'); + String normalizedMainSourceFilePath = mainSourceFilePath.replace(File.separatorChar, '/'); + Marker detailMarker = null; + if (normalizedFilePath.length() > 0 && !normalizedFilePath.equals(normalizedMainSourceFilePath)) { + String folderPath = mainSourceIFile.getParent().getLocation().toOSString(); + String normalizedFolderPath = folderPath.replace(File.separatorChar, '/'); + + String relativePath; + + // Absolute include path, may even be a brother or parent path? + if (normalizedFilePath.startsWith(normalizedFolderPath)) { + relativePath = normalizedFilePath.substring(folderPath.length() + 1); + } else { + // Simple relative path. + relativePath = normalizedFilePath; + } + + // Create absolute iFile. + iFile = mainSourceIFile.getParent().getFile(new Path(relativePath)); + + // Check if it exists. This requires the file path to be in exactly + // the right case, even if the file system is case insensitive. + if (iFile == null || !iFile.exists()) { + + // If the file exists, but the include was specified with a + // different case, an additional detail error message is issued. + IFile caseInsenstiveIFile = null; + if (iFile != null) { + try { + IResource[] members = iFile.getParent().members(); + for (int i = 0; i < members.length && caseInsenstiveIFile == null; i++) { + if (members[i] instanceof IFile && members[i].getName().equalsIgnoreCase(iFile.getName())) { + caseInsenstiveIFile = (IFile) members[i]; + } + } + } catch (CoreException ex) { + AssemblerPlugin.getInstance().logError("Could not retrieve members of {0}", + new Object[] { iFile }, ex); + } + } + + // Use the case insensitive file if found and the main file + // otherwise. + if (iFile != null && caseInsenstiveIFile != null) { + // ERROR: Include statement for file '{0}' uses a file name + // that has a different case different from file system name + // {1}. Correct the file name in the include statement. + String caseMessage = TextUtility.format(Texts.MESSAGE_E142, iFile.getName(), + caseInsenstiveIFile.getName()); + detailMarker = new Marker(mainSourceIFile, 0, IMarker.SEVERITY_ERROR, caseMessage, null); + + iFile = caseInsenstiveIFile; + } else { + iFile = mainSourceIFile; + // INFO: In include file '{0}', line {1}. + message += " " + + TextUtility.format(Texts.MESSAGE_S143, filePath, + NumberUtility.getLongValueDecimalString(lineNumber)); + lineNumber = 0; + } + } + + } + + return new Marker(iFile, lineNumber, severity, message.trim(), detailMarker); + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerRegistry.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerRegistry.java new file mode 100644 index 00000000..6ff56bcc --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerRegistry.java @@ -0,0 +1,244 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler; + +import java.util.ArrayList; +import java.util.Collections; +import java.util.List; +import java.util.Map; +import java.util.TreeMap; + +import org.eclipse.core.runtime.CoreException; +import org.eclipse.core.runtime.IConfigurationElement; +import org.eclipse.core.runtime.IExtension; +import org.eclipse.core.runtime.IExtensionPoint; +import org.eclipse.core.runtime.IExtensionRegistry; +import org.eclipse.core.runtime.Platform; + +import com.wudsn.ide.asm.CPU; +import com.wudsn.ide.asm.Hardware; +import com.wudsn.ide.asm.compiler.syntax.CompilerSyntax; + +/** + * Registry for compilers, based on the extension points + * {@value CompilerRegistry#COMPILERS}. + * + * @author Peter Dell + * + */ +public final class CompilerRegistry { + + /** + * The id of the extension point which provides the compilers. + */ + private static final String COMPILERS = "com.wudsn.ide.asm.compilers"; + + /** + * The registered compiler definition. + */ + private List compilerDefinitionList; + + /** + * The cached map of compiler instances. + */ + private Map compilerMap; + + /** + * Creation is public. + */ + public CompilerRegistry() { + compilerDefinitionList = Collections.emptyList(); + compilerMap = Collections.emptyMap(); + + } + + /** + * Initializes the list of available compilers. + */ + public void init() { + + compilerDefinitionList = new ArrayList(); + compilerMap = new TreeMap(); + + IExtensionRegistry extensionRegistry = Platform.getExtensionRegistry(); + IExtensionPoint extensionPoint = extensionRegistry + .getExtensionPoint(COMPILERS); + if (extensionPoint == null) { + throw new IllegalStateException("Extension point '" + COMPILERS + + "' is not defined."); + } + + IExtension[] extensions = extensionPoint.getExtensions(); + + for (IExtension extension : extensions) { + IConfigurationElement[] configurationElements = extension + .getConfigurationElements(); + for (IConfigurationElement configurationElement : configurationElements) { + + try { + CompilerDefinition compilerDefinition; + compilerDefinition = new CompilerDefinition(); + compilerDefinition.setId(configurationElement + .getAttribute("id")); + compilerDefinition.setName(configurationElement + .getAttribute("name")); + compilerDefinition.setClassName(configurationElement + .getAttribute("class")); + compilerDefinition.setHelpFilePaths(configurationElement + .getAttribute("helpFilePaths")); + compilerDefinition.setHomePageURL(configurationElement + .getAttribute("homePageURL")); + compilerDefinition + .setDefaultParameters(configurationElement + .getAttribute("defaultParameters")); + + configurationElement.getChildren("supportedCPU"); + IConfigurationElement[] supportedCPUArray; + supportedCPUArray = configurationElement + .getChildren("supportedCPU"); + List supportedCPUs = new ArrayList( + supportedCPUArray.length); + for (IConfigurationElement supportedCPU : supportedCPUArray) { + supportedCPUs.add(CPU.valueOf(supportedCPU + .getAttribute("cpu"))); + } + supportedCPUs = Collections.unmodifiableList(supportedCPUs); + compilerDefinition.setSupportedCPUs(supportedCPUs); + compilerDefinition.setDefaultHardware(Hardware + .valueOf(configurationElement + .getAttribute("defaultHardware"))); + + compilerDefinitionList.add(compilerDefinition); + + addCompiler(configurationElement, compilerDefinition); + } catch (RuntimeException ex) { + throw new RuntimeException( + "Error during registration of compiler '" + + configurationElement.getAttribute("id") + + "'.", ex); + } + } + } + + compilerDefinitionList = new ArrayList( + compilerDefinitionList); + Collections.sort(compilerDefinitionList); + compilerDefinitionList = Collections + .unmodifiableList(compilerDefinitionList); + compilerMap = Collections.unmodifiableMap(compilerMap); + } + + /** + * Adds a new compiler. + * + * @param configurationElement + * The configuration element used as class instance factory, not + * null. + * + * @param compilerDefinition + * The compiler definition, not null. + */ + private void addCompiler(IConfigurationElement configurationElement, + CompilerDefinition compilerDefinition) { + if (configurationElement == null) { + throw new IllegalArgumentException( + "Parameter 'configurationElement' must not be null."); + } + if (compilerDefinition == null) { + throw new IllegalArgumentException( + "Parameter 'compilerDefinition' must not be null."); + } + + String id = compilerDefinition.getId(); + Compiler compiler; + try { + // The class loading must be delegated to the framework. + compiler = (Compiler) configurationElement + .createExecutableExtension("class"); + } catch (CoreException ex) { + throw new RuntimeException( + "Cannot create compiler instance for id '" + id + "'.", ex); + } + + // Build the list of common and specific syntax definition files. + List> compilerClasses = new ArrayList>(2); + compilerClasses.add(compiler.getClass()); + compilerClasses.add(Compiler.class); + + CompilerSyntax syntax; + syntax = new CompilerSyntax(id); + + syntax.loadXMLData(compilerClasses); + + compilerDefinition.setSyntax(syntax); + + compiler.setDefinition(compilerDefinition); + + compiler = compilerMap.put(id, compiler); + if (compiler != null) { + throw new RuntimeException("Compiler id '" + id + + "' is already registered to class '" + + compiler.getClass().getName() + "'."); + } + + } + + /** + * Gets the unmodifiable list of compiler definitions, sorted by their id. + * + * + * @return The unmodifiable list of compiler definitions, sorted by their + * id, may be empty, not null + * + * @since 1.6.1 + */ + public List getCompilerDefinitions() { + return compilerDefinitionList; + } + + /** + * Gets the compiler for a given id. Instances of compiler are stateless + * singletons within the plugin. + * + * @param compilerId + * The compiler id, not null. + * + * @return The compiler, not null. + */ + public Compiler getCompiler(String compilerId) { + if (compilerId == null) { + throw new IllegalArgumentException( + "Parameter 'compilerId' must not be null."); + } + Compiler result; + synchronized (compilerMap) { + + result = compilerMap.get(compilerId); + } + if (result == null) { + + throw new IllegalArgumentException("Unknown compiler id '" + + compilerId + "'."); + } + + return result; + } + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerSymbol.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerSymbol.java new file mode 100644 index 00000000..63771105 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerSymbol.java @@ -0,0 +1,156 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler; + +import com.wudsn.ide.base.common.HexUtility; +import com.wudsn.ide.base.common.NumberUtility; +import com.wudsn.ide.base.common.StringUtility; + +/** + * A symbols in the symbol table. A symbol may be an equate or a label and it + * may represent a numeric numericValue or an address. + * + * @author Peter Dell + */ +public final class CompilerSymbol { + + public static final int NUMBER = 1; + public static final int STRING = 2; + + private final int type; + private final String name; + private final String nameUpperCase; + private final String bankString; + private final int valueType; + private final String numberValueHexString; + private final String numberValueHexStringUpperCase; + private final String numberValueDecimalString; + private final String stringValue; + private final String stringValueUpperCase; + + public static final int UNDEFINED_BANK = -1; + + private CompilerSymbol(int type, String name, int bank, int valueType, long numberValue, String stringValue) { + if (name == null) + throw new IllegalArgumentException("Parameter 'name' must not be null."); + if (StringUtility.isEmpty(name)) { + throw new IllegalArgumentException("Parameter 'name' must not be empty."); + } + if (stringValue == null) { + throw new IllegalArgumentException("Parameter 'stringValue' must not be null."); + } + this.type = type; + this.name = name.trim(); + this.nameUpperCase = name.toUpperCase(); + this.bankString = bank != UNDEFINED_BANK ? NumberUtility.getLongValueDecimalString(bank) : ""; + this.valueType = valueType; + switch (valueType) { + case NUMBER: + numberValue = numberValue & 0xfffffff; + int length = (HexUtility.getLongValueHexLength(numberValue) + 1) & 0xffffffe; + this.numberValueHexString = HexUtility.getLongValueHexString(numberValue, length); + this.numberValueHexStringUpperCase = numberValueHexString.toUpperCase(); + this.numberValueDecimalString = NumberUtility.getLongValueDecimalString(numberValue); + this.stringValue = ""; + this.stringValueUpperCase = stringValue; + break; + case STRING: + this.numberValueHexString = ""; + this.numberValueHexStringUpperCase = ""; + this.numberValueDecimalString = ""; + this.stringValue = stringValue; + this.stringValueUpperCase = stringValue.toUpperCase(); + break; + default: + throw new IllegalArgumentException("Value type '" + valueType + "' is not supported."); + } + } + + public static CompilerSymbol createNumberSymbol(int type, String name, int bank, long numberValue) { + return new CompilerSymbol(type, name, bank, NUMBER, numberValue, ""); + } + + public static CompilerSymbol createNumberHexSymbol(String name, String hexValue) throws NumberFormatException { + return new CompilerSymbol(CompilerSymbolType.DEFAULT, name, UNDEFINED_BANK, NUMBER, + Long.parseLong(hexValue, 16), ""); + } + + public static CompilerSymbol createStringSymbol(String name, String stringValue) { + return new CompilerSymbol(CompilerSymbolType.DEFAULT, name, UNDEFINED_BANK, STRING, 0, stringValue); + } + + public int getType() { + return type; + } + + public String getName() { + return name; + } + + public String getNameUpperCase() { + return nameUpperCase; + } + + public String getBankString() { + return bankString; + } + + public String getValueAsHexString() { + return numberValueHexString; + } + + /** + * Upper case version of the hex string for fuzzy search. + * + * @return The upper case version of the hex string. + */ + public String getValueAsHexStringUpperCase() { + return numberValueHexStringUpperCase; + } + + public String getValueAsDecimalString() { + return numberValueDecimalString; + } + + public String getValueAsString() { + return stringValue; + } + + /** + * Upper case version of the hex string for fuzzy search. + * + * @return The upper case version of the hex string. + */ + public String getValueAsStringUpperCase() { + return stringValueUpperCase; + } + + @Override + public String toString() { + switch (valueType) { + case NUMBER: + return name + "=" + getValueAsHexString(); + case STRING: + return name + "=" + getValueAsString(); + } + throw new IllegalStateException(); + + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerSymbolType.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerSymbolType.java new file mode 100644 index 00000000..5c4c0493 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerSymbolType.java @@ -0,0 +1,77 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ +package com.wudsn.ide.asm.compiler; + +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParserTreeObjectType; + +/** + * Constant values for symbol types. Symbol types are a subset of the source + * tree object types. + * + * @author Peter Dell + */ +public final class CompilerSymbolType { + + /** + * Creation is private. + */ + private CompilerSymbolType() { + } + + /** + * Default-type + */ + public static final int DEFAULT = CompilerSourceParserTreeObjectType.DEFAULT; + + public static final int EQUATE_DEFINITION = CompilerSourceParserTreeObjectType.EQUATE_DEFINITION; + public static final int LABEL_DEFINITION = CompilerSourceParserTreeObjectType.LABEL_DEFINITION; + + public static final int ENUM_DEFINITION_SECTION = CompilerSourceParserTreeObjectType.ENUM_DEFINITION_SECTION; + public static final int STRUCTURE_DEFINITION_SECTION = CompilerSourceParserTreeObjectType.STRUCTURE_DEFINITION_SECTION; + public static final int MACRO_DEFINITION_SECTION = CompilerSourceParserTreeObjectType.MACRO_DEFINITION_SECTION; + public static final int LOCAL_SECTION = CompilerSourceParserTreeObjectType.LOCAL_SECTION; + public static final int PROCEDURE_DEFINITION_SECTION = CompilerSourceParserTreeObjectType.PROCEDURE_DEFINITION_SECTION; + + /** + * Gets the localized text for a compiler symbol type. + * + * @param type + * The type, see constants of this class. + * @return The localized text, may be empty but not null. + */ + public static String getText(int type) { + String result; + switch (type) { + case DEFAULT: + case EQUATE_DEFINITION: + case LABEL_DEFINITION: + case ENUM_DEFINITION_SECTION: + case STRUCTURE_DEFINITION_SECTION: + case MACRO_DEFINITION_SECTION: + case LOCAL_SECTION: + case PROCEDURE_DEFINITION_SECTION: + result = CompilerSourceParserTreeObjectType.getText(type); + break; + default: + throw new IllegalArgumentException("Unknown type " + type + "."); + } + return result; + + } +} \ No newline at end of file diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerVariables.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerVariables.java new file mode 100644 index 00000000..bc0ac64e --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/CompilerVariables.java @@ -0,0 +1,75 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler; + +/** + * Utility class to handle dynamic variables. + * + * @author Peter Dell + */ +public final class CompilerVariables { + + /** + * Creation is private. + */ + private CompilerVariables() { + } + + public static final String SOURCE_FOLDER_PATH = "${sourceFolderPath}"; + public static final String SOURCE_FILE_PATH = "${sourceFilePath}"; + public static final String OUTPUT_FOLDER_PATH = "${outputFolderPath}"; + public static final String OUTPUT_FILE_PATH = "${outputFilePath}"; + public static final String OUTPUT_FILE_PATH_WITHOUT_EXTENSION = "${outputFilePathWithoutExtension}"; + public static final String OUTPUT_FILE_NAME = "${outputFileName}"; + public static final String OUTPUT_FILE_NAME_WITHOUT_EXTENSION = "${outputFileNameWithoutExtension}"; + public static final String OUTPUT_FILE_NAME_SHORT_WITHOUT_EXTENSION = "${outputFileNameShortWithoutExtension}"; + + public static String replaceVariables(String parameter, + CompilerFiles files) { + if (parameter == null) { + throw new IllegalArgumentException( + "Parameter 'parameter' must not be null."); + } + if (files == null) { + throw new IllegalArgumentException( + "Parameter 'files' must not be null."); + } + + // When referring to the source file and folder at compiler time, + // this always means the main source file. + parameter = parameter.replace(SOURCE_FOLDER_PATH, + files.mainSourceFile.folderPath); + parameter = parameter.replace(SOURCE_FILE_PATH, + files.mainSourceFile.filePath); + parameter = parameter.replace(OUTPUT_FOLDER_PATH, + files.outputFolderPath); + parameter = parameter.replace(OUTPUT_FILE_PATH, files.outputFilePath); + parameter = parameter.replace(OUTPUT_FILE_PATH_WITHOUT_EXTENSION, + files.outputFilePathWithoutExtension); + parameter = parameter.replace(OUTPUT_FILE_NAME, files.outputFileName); + parameter = parameter.replace(OUTPUT_FILE_NAME_WITHOUT_EXTENSION, + files.outputFileNameWithoutExtension); + parameter = parameter.replace(OUTPUT_FILE_NAME_SHORT_WITHOUT_EXTENSION, + files.outputFileNameShortWithoutExtension); + + return parameter; + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceFile.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceFile.java new file mode 100644 index 00000000..89b28816 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceFile.java @@ -0,0 +1,460 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.parser; + +import java.io.File; +import java.util.ArrayList; +import java.util.Collections; +import java.util.List; + +import org.eclipse.jface.text.IDocument; +import org.eclipse.jface.text.Position; + +import com.wudsn.ide.asm.Texts; +import com.wudsn.ide.asm.compiler.syntax.CompilerSyntax; +import com.wudsn.ide.base.common.FileUtility; +import com.wudsn.ide.base.common.StringUtility; + +/** + * Container for recursive parsing of source files using + * {@link CompilerSourceParser#parse(CompilerSourceFile, CompilerSourceParserLineCallback)} + * . + * + * @author Peter Dell + */ +public final class CompilerSourceFile { + + private final static class FoldingStackEntry { + public final int startOffset; + public final boolean forSection; + + public FoldingStackEntry(int startOffset, boolean forSection) { + this.startOffset = startOffset; + this.forSection = forSection; + } + + @Override + public String toString() { + return "startOffset=" + startOffset + ", forSection=" + forSection; + } + } + + private CompilerSyntax compilerSyntax; + private File documentFile; + private File documentDirectory; + private IDocument document; + + private List foldingPositions; + + /** + * Temporary data during the parse process. + */ + private List foldingStack; + + /** + * The result of the last parse process. + */ + private CompilerSourceParserTreeObject definitionSection; + private List implementationSections; + private List sectionStack; + + /** + * Creates a new compiler source file. Instances are only created by + * {@link CompilerSourceParser#createCompilerSourceFile(File, IDocument)}. + * + * @param compilerSyntax + * The compiler syntax used to parse this file, not + * null. + * @param documentFile + * The file in the file system. Its directory is used to resolve + * relative file paths, or null if the document is + * not yet persistent. + * @param document + * The document, not null. + */ + CompilerSourceFile(CompilerSyntax compilerSyntax, File documentFile, IDocument document) { + if (compilerSyntax == null) { + throw new IllegalArgumentException("Parameter 'compilerSyntax' must not be null."); + } + if (document == null) { + throw new IllegalArgumentException("Parameter 'document' must not be null."); + } + this.compilerSyntax = compilerSyntax; + if (documentFile != null) { + documentFile = FileUtility.getCanonicalFile(documentFile); + this.documentFile = documentFile; + this.documentDirectory = documentFile.getParentFile(); + } + + this.document = document; + + // Folding. + foldingPositions = new ArrayList(); + foldingStack = new ArrayList(); + + // Sections. + definitionSection = new CompilerSourceParserTreeObject(this, 0, + CompilerSourceParserTreeObjectType.DEFINITION_SECTION, "DefinitionSection", + Texts.ASSEMBLER_CONTENT_OUTLINE_TREE_TYPE_DEFINITION_SECTION, ""); + implementationSections = new ArrayList(); + sectionStack = new ArrayList(); + } + + /** + * Gets the compiler syntax user to parse this file. + * + * @return The compiler syntax, not null. + * + * @since 1.6.1 + */ + public CompilerSyntax getCompilerSyntax() { + return compilerSyntax; + } + + /** + * Gets the canonical document file. + * + * @return The file where the document is located, or null if + * the document is not yet persistent. + */ + public File getDocumentFile() { + return documentFile; + } + + /** + * Gets the document directory. + * + * @return The directory which is used to resolved relative file paths, or + * null if the document is not yet persistent. + */ + public File getDocumentDirectory() { + return documentDirectory; + } + + /** + * Gets the document. + * + * @return The document, not null. + */ + public IDocument getDocument() { + return document; + } + + /** + * Gets the sections of this file. + * + * @return The unmodifiable list of section, may be empty, not + * null. + */ + public List getSections() { + List result; + result = new ArrayList(); + if (definitionSection != null && definitionSection.hasChildren()) { + result.add(definitionSection); + } + + result.addAll(implementationSections); + result = Collections.unmodifiableList(result); + return result; + } + + final CompilerSourceParserTreeObject getDefinitionSection() { + return definitionSection; + } + + final List getImplementationSections() { + return implementationSections; + } + + /** + * Gets the list of foldingPositions for folding after parsed has completed. + * + * @return The non-modifiable sorted list of foldingPositions, may be empty, + * not null. + */ + public List getFoldingPositions() { + return Collections.unmodifiableList(foldingPositions); + } + + /** + * Determines if the currently active folding was started for a section. + * + * @return true if the folding was started for a section, + * false if not. + */ + final boolean isFoldingForSection() { + if (foldingStack.isEmpty()) { + return false; // Ignore wrong nesting. + } + FoldingStackEntry entry = foldingStack.get(foldingStack.size() - 1); + return entry.forSection; + } + + /** + * Starts a new active folding level. + * + * @param startOffset + * The start offset, a non-negative integer. + * @param forSection + * true if the folding is started for a section, + * false if not. + */ + final void beginFolding(int startOffset, boolean forSection) { + if (startOffset < 0) { + throw new IllegalArgumentException("Parameter 'startOffset' must not be negative. Specified value is " + + startOffset + "."); + } + foldingStack.add(new FoldingStackEntry(startOffset, forSection)); + } + + /** + * Ends the currently active folding level. + * + * @param endOffset + * The end offset, a non-negative integer. This must be the + * actual offset of the last character in the line, including the + * line end delimiter, if present. + */ + final void endFolding(int endOffset) { + if (endOffset < 0) { + throw new IllegalArgumentException("Parameter 'endOffset' must not be negative. Specified value is " + + endOffset + "."); + } + + if (foldingStack.isEmpty()) { + return; // Ignore wrong nesting. + } + FoldingStackEntry entry = foldingStack.remove(foldingStack.size() - 1); + int length = endOffset - entry.startOffset; + if (length < 0) { + throw new IllegalArgumentException("End offset " + endOffset + " is less than the start offset " + + entry.startOffset + "."); + } + // Add only non-empty positions. + if (length > 0) { + foldingPositions.add(new Position(entry.startOffset, length)); + } + } + + /** + * Ends all currently active folding levels. + * + */ + final void endAllFoldings() { + + while (!foldingStack.isEmpty()) { + endFolding(document.getLength() > 0 ? document.getLength() - 1 : 0); + + } + } + + /** + * Ends all currently active sections. + * + */ + public void endAllSections() { + while (!sectionStack.isEmpty()) { + endSection(document.getLength() > 0 ? document.getLength() - 1 : 0); + } + } + + /** + * Starts a new active section. + * + * @param startOffset + * The start offset, a non-negative integer. + * @param section + * The new section, not null. + * @param withFolding + * true if the section is also a folding section, + * false otherwise. + */ + final void beginSection(int startOffset, CompilerSourceParserTreeObject section, boolean withFolding) { + if (startOffset < 0) { + throw new IllegalArgumentException("Parameter 'startOffset' must not be negative. Specified value is " + + startOffset + "."); + } + if (section == null) { + throw new IllegalArgumentException("Parameter 'section' must not be null."); + } + + if (!sectionStack.isEmpty()) { + CompilerSourceParserTreeObject parent = sectionStack.get(sectionStack.size() - 1); + parent.addChild(section); + } + sectionStack.add(section); + if (withFolding) { + beginFolding(startOffset, true); + } + + } + + /** + * Ends the currently active section. + * + * @param endOffset + * The end offset, a non-negative integer. This must be the + * actual offset of the last character in the line, including the + * line end delimiter, if present. + * + * @return The new top of the section stack, may be null. + */ + final CompilerSourceParserTreeObject endSection(int endOffset) { + if (endOffset < 0) { + throw new IllegalArgumentException("Parameter 'endOffset' must not be negative. Specified value is " + + endOffset + "."); + } + + CompilerSourceParserTreeObject result; + result = null; + + // Remove top of stack. + if (!sectionStack.isEmpty()) { + sectionStack.remove(sectionStack.size() - 1); + + // Make top of stack the new active section, accepting illegal pops. + if (!sectionStack.isEmpty()) { + result = sectionStack.get(sectionStack.size() - 1); + } + endFolding(endOffset); + } + return result; + } + + public List getIdentifiers() { + List foundElements; + foundElements = new ArrayList(); + getIdentifiers(getSections(), foundElements); + foundElements = Collections.unmodifiableList(foundElements); + return foundElements; + } + + private void getIdentifiers(List allElements, + List foundElements) { + if (allElements == null) { + throw new IllegalArgumentException("Parameter 'allElements' must not be null."); + } + if (foundElements == null) { + throw new IllegalArgumentException("Parameter 'foundElements' must not be null."); + } + CompilerSourceParserTreeObject element = null; + for (int i = 0; i < allElements.size(); i++) { + element = allElements.get(i); + switch (element.getType()) { + case CompilerSourceParserTreeObjectType.EQUATE_DEFINITION: + case CompilerSourceParserTreeObjectType.LABEL_DEFINITION: + case CompilerSourceParserTreeObjectType.LOCAL_SECTION: + case CompilerSourceParserTreeObjectType.MACRO_DEFINITION_SECTION: + case CompilerSourceParserTreeObjectType.PROCEDURE_DEFINITION_SECTION: + + foundElements.add(element); + break; + } + if (element.hasChildren()) { + getIdentifiers(element.getChildren(), foundElements); + } + + } + } + + /** + * Find the definition elements for a given identifier. + * + * @param identifier + * The identifier to search for, not empty and not + * null. + * @return The unmodifiable list of compiler source tree object with the + * label, may be empty, not null. The result may + * contain entries from source include files. + */ + public List getIdentifierDefinitionElements(String identifier) { + + if (identifier == null) { + throw new IllegalArgumentException("Parameter 'identifier' must not be null."); + } + if (StringUtility.isEmpty(identifier)) { + throw new IllegalArgumentException("Parameter 'identifier' must not be empty."); + } + List foundElements; + foundElements = new ArrayList(); + getIdentifierDefinitionElements(getSections(), identifier, foundElements); + foundElements = Collections.unmodifiableList(foundElements); + return foundElements; + + } + + private void getIdentifierDefinitionElements(List allElements, String identifier, + List foundElements) { + if (allElements == null) { + throw new IllegalArgumentException("Parameter 'allElements' must not be null."); + } + if (identifier == null) { + throw new IllegalArgumentException("Parameter 'identifier' must not be null."); + } + if (foundElements == null) { + throw new IllegalArgumentException("Parameter 'foundElements' must not be null."); + } + String compoundIdentifierSuffix = null; + char c = compilerSyntax.getIdentifierSeparatorCharacter(); + if (c != CompilerSyntax.NO_CHARACTER) { + compoundIdentifierSuffix = c + identifier; + } + + boolean identifiersCaseSensitive = compilerSyntax.areIdentifiersCaseSensitive(); + CompilerSourceParserTreeObject element = null; + for (int i = 0; i < allElements.size(); i++) { + element = allElements.get(i); + switch (element.getType()) { + case CompilerSourceParserTreeObjectType.EQUATE_DEFINITION: + case CompilerSourceParserTreeObjectType.LABEL_DEFINITION: + case CompilerSourceParserTreeObjectType.ENUM_DEFINITION_SECTION: + case CompilerSourceParserTreeObjectType.STRUCTURE_DEFINITION_SECTION: + case CompilerSourceParserTreeObjectType.LOCAL_SECTION: + case CompilerSourceParserTreeObjectType.MACRO_DEFINITION_SECTION: + case CompilerSourceParserTreeObjectType.PROCEDURE_DEFINITION_SECTION: + // Check if name equals the identifier, compound name equals the + // identifier or compound name ends with + // compoundIdentifierSuffix. + if (identifiersCaseSensitive) { + if (element.getName().equals(identifier) + || element.getCompoundName().equals(identifier) + || (compoundIdentifierSuffix != null && element.getCompoundName().endsWith( + compoundIdentifierSuffix))) { + foundElements.add(element); + } + } else { + if (element.getName().equalsIgnoreCase(identifier) + || element.getCompoundName().equals(identifier) + || (compoundIdentifierSuffix != null && element.getCompoundName().regionMatches(true, + element.getCompoundName().length() - compoundIdentifierSuffix.length(), + compoundIdentifierSuffix, 0, compoundIdentifierSuffix.length()))) { + foundElements.add(element); + } + } + break; + } + + if (element.hasChildren()) { + getIdentifierDefinitionElements(element.getChildren(), identifier, foundElements); + } + + } + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceParser.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceParser.java new file mode 100644 index 00000000..fb06d661 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceParser.java @@ -0,0 +1,1214 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.parser; + +import java.io.File; +import java.util.HashMap; +import java.util.Iterator; +import java.util.List; +import java.util.Map; + +import org.eclipse.core.runtime.CoreException; +import org.eclipse.jface.text.BadLocationException; +import org.eclipse.jface.text.Document; +import org.eclipse.jface.text.IDocument; +import org.eclipse.jface.text.IRegion; + +import com.wudsn.ide.asm.AssemblerPlugin; +import com.wudsn.ide.asm.AssemblerProperties; +import com.wudsn.ide.asm.compiler.Compiler; +import com.wudsn.ide.asm.compiler.syntax.CompilerSyntax; +import com.wudsn.ide.asm.compiler.syntax.Instruction; +import com.wudsn.ide.asm.compiler.syntax.InstructionSet; +import com.wudsn.ide.asm.compiler.syntax.InstructionType; +import com.wudsn.ide.base.BasePlugin; +import com.wudsn.ide.base.common.FileUtility; +import com.wudsn.ide.base.common.StringUtility; + +/** + * Source parser for creating {@link CompilerSourceParserTreeObject} instances. + * The performance is so good, that no caching for the results of source include + * parsing is required. + * + * @author Peter Dell + * + */ +public abstract class CompilerSourceParser { + + // The sections of a single line + private static final class LineSection { + public static final int NONE = 0; + public static final int SYMBOL = 1; + public static final int INSTRUCTION = 2; + public static final int OPERAND = 3; + } + + // The compiler syntax and instruction set. + private CompilerSyntax compilerSyntax; + private InstructionSet instructionSet; + + // Fields set once for parsing a file. + private CompilerSourceFile compilerSourceFile; + + // Fields modified during parsing. + private CompilerSourceParserTreeObject section; + + // Fields modified per line. Preserved to and restored from local variables + // during recursive parsing of SOURCE_INCLUDE_DIRECTIVE instructions. + private CompilerSourceParserTreeObject child; + private CompilerSourceParserTreeObject labelChild; + private boolean blockStarting; + private boolean blockEnding; + + // For debugging. + private boolean logEnabled = false; + + /** + * Extract all {@link AssemblerProperties} properties from a document. + * + * @param document + * The document, not null. + * @return The properties, may be empty, not null. + */ + public static AssemblerProperties getDocumentProperties(IDocument document) { + if (document == null) { + throw new IllegalArgumentException( + "Parameter 'document' must not be null."); + } + String content = document.get(); + AssemblerProperties properties = new AssemblerProperties(); + + int index1 = content.indexOf(AssemblerProperties.PREFIX); + while (index1 >= 0) { + + int indexEqualSign = content.indexOf('=', index1); + int indexNewLine = content.indexOf('\n', index1); + if (indexNewLine < 0) { + indexNewLine = content.indexOf('\r', index1); + } + if (indexNewLine < 0) { + indexNewLine = content.length(); + } + + if (indexEqualSign >= 0 && indexEqualSign < indexNewLine) { + String key = content.substring(index1, indexEqualSign).trim(); + String value = content.substring(indexEqualSign + 1, + indexNewLine).trim(); + int lineNumber; + try { + lineNumber = document.getLineOfOffset(index1) + 1; + } catch (BadLocationException ex) { + lineNumber = 0; + } + properties.put(key, value, lineNumber); + } + index1 = content.indexOf(AssemblerProperties.PREFIX, indexNewLine); + } + return properties; + } + + /** + * Creation is protected. + */ + protected CompilerSourceParser() { + + } + + /** + * Gets the compiler syntax for this parser. + * + * @return The compiler syntax, not null. + */ + public final CompilerSyntax getCompilerSyntax() { + if (compilerSyntax == null) { + throw new IllegalStateException( + "Field 'compilerSyntax' must not be null."); + } + return compilerSyntax; + } + + /** + * Gets the instruction for the currently active CPU. + * + * @return The instruction set, not null. + * + * @since 1.6.1 + */ + public final InstructionSet getInstructionSet() { + if (instructionSet == null) { + throw new IllegalStateException( + "Field 'instructionSet' must not be null."); + } + return instructionSet; + } + + /** + * Logs a message to the error log in case logging is enabled. + * + * @param message + * The message with place holders, may be empty, not + * null. + * @param parameters + * The parameters for the place holders, may be empty or + * null. + */ + private void log(String message, Object... parameters) { + if (logEnabled) { + BasePlugin.getInstance().log(message, parameters); + } + + } + + /** + * Called by {@link Compiler} to link the parser to the compile syntax. + * + * @param instructionSet + * The instruction set, not null. + */ + public final void init(InstructionSet instructionSet) { + if (instructionSet == null) { + throw new IllegalArgumentException( + "Parameter 'instructionSet' must not be null."); + } + this.instructionSet = instructionSet; + this.compilerSyntax = instructionSet.getCompilerSyntax(); + + } + + /** + * Detects a file references in the given source line. This method is + * stateless. + * + * @param line + * The source line, may be empty, not null. + * @param include + * The modifiable include statement description, not + * null. + */ + public final void detectFileReference(String line, + CompilerSourceParserFileReference include) { + if (line == null) { + throw new IllegalArgumentException( + "Parameter 'line' must not be null."); + } + if (include == null) { + throw new IllegalArgumentException( + "Parameter 'include' must not be null."); + } + + boolean caseSenstive = instructionSet.areInstructionsCaseSensitive(); + if (!caseSenstive) { + line = line.toUpperCase(); + } + + // Find next (possible) quote. + int quoteOffset = -1; + Iterator i = compilerSyntax.getStringDelimiters().iterator(); + while (i.hasNext() && quoteOffset == -1) { + String quote = i.next(); + quoteOffset = line.indexOf(quote); + } + + for (Instruction instruction : instructionSet + .getFileReferenceInstructions()) { + int instructionOffset; + String instructionName; + + if (caseSenstive) { + instructionName = instruction.getName(); + } else { + instructionName = instruction.getUpperCaseName(); + } + instructionOffset = line.indexOf(instructionName); + + // Key word found before the quote? + if (quoteOffset > -1 && instructionOffset > -1 + && instructionOffset < quoteOffset) { + if (instruction.getType() == InstructionType.SOURCE_INCLUDE_DIRECTIVE) { + include.setType(CompilerSourceParserFileReferenceType.SOURCE); + } else if (instruction.getType() == InstructionType.BINARY_INCLUDE_DIRECTIVE + || instruction.getType() == InstructionType.BINARY_OUTPUT_DIRECTIVE) { + include.setType(CompilerSourceParserFileReferenceType.BINARY); + } else { + throw new IllegalStateException("Include instruction '" + + instructionName + "' has the unsupported type '" + + instruction.getType() + "'"); + } + + include.setDirectiveEndOffset(instructionOffset + + instructionName.length()); + return; + } + + } + } + + /** + * Enhances the file path of an include, for example add a default extension + * for source includes. + * + * @param type + * The type of include, see + * {@link CompilerSourceParserFileReferenceType}. + * + * @param documentDirectory + * The current directory which act as the basis for relative + * paths or not null if it is not known. + * + * @param filePath + * The possibly relative file path of the include file in OS + * specific notation, not empty and not null. + * @return The enhanced absolute file path of the include file in OS + * specific notation, not empty and not null. + */ + public final String getIncludeAbsoluteFilePath(int type, + File documentDirectory, String filePath) { + + if (filePath == null) { + throw new IllegalArgumentException( + "Parameter 'filePath' must not be null."); + } + + String relativeCurrentPrefix; + String relativeParentPrefix; + File absoluteFile; + relativeCurrentPrefix = "."; + relativeParentPrefix = ".."; + + // Ensure documentDirectory is a canonical file. + if (documentDirectory != null) { + documentDirectory = FileUtility.getCanonicalFile(documentDirectory); + } + + // Check double or single dots followed by path delimiter first. + int relativeParentPrefixLength = relativeParentPrefix.length() + 1; + int relativeCurrentPrefixLength = relativeCurrentPrefix.length() + 1; + if (filePath.startsWith(relativeParentPrefix) + && filePath.length() >= relativeParentPrefixLength) { + if (documentDirectory == null) { + return null; + } + absoluteFile = new File(documentDirectory.getParentFile(), + filePath.substring(relativeParentPrefixLength)); + + } else if (filePath.startsWith(relativeCurrentPrefix) + && filePath.length() >= relativeCurrentPrefixLength) { + if (documentDirectory == null) { + return null; + } + absoluteFile = new File(documentDirectory, + filePath.substring(relativeCurrentPrefixLength)); + } else { + + // If there is no file separator in the file name, we can assume a + // relative path based on the current directory. + File file = new File(filePath); + if (file.exists()) { + absoluteFile = file; + } else { + absoluteFile = new File(documentDirectory, filePath); + } + } + + // Ensure the complete file path is in OS notation. + absoluteFile = FileUtility.getCanonicalFile(absoluteFile); + String absoluteFilePath = absoluteFile.getPath(); + if (type == CompilerSourceParserFileReferenceType.SOURCE) { + int index = absoluteFilePath.lastIndexOf(File.separator); + if (index < 1) { + index = 0; + } + String fileName = absoluteFilePath.substring(index); + index = fileName.lastIndexOf('.'); + if (index == -1) { + String extension = compilerSyntax + .getSourceIncludeDefaultExtension(); + if (extension.length() > 0) { + absoluteFilePath = absoluteFilePath + "." + extension; + } + } + } + return absoluteFilePath; + } + + /** + * Creates a new, yet empty compiler source file. + * + * @param file + * The file, not null. + * @param document + * The document, not null. + * @return The compiler source file for use in + * {@link #parse(CompilerSourceFile, CompilerSourceParserLineCallback)} + * , not null. + * @since 1.6.1 + */ + public final CompilerSourceFile createCompilerSourceFile(File file, + IDocument document) { + if (compilerSyntax == null) { + throw new IllegalArgumentException( + "Parameter 'compilerSyntax' must not be null."); + } + if (document == null) { + throw new IllegalArgumentException( + "Parameter 'document' must not be null."); + } + return new CompilerSourceFile(compilerSyntax, file, document); + } + + /** + * Creates a new compiler source file for persistent file. + * + * @param filePath + * The absolute file path, not empty and not null. + * @return The compiler source file, not null. + * + * @since 1.6.3 + */ + private CompilerSourceFile createCompilerSourceFile(String filePath) { + if (filePath == null) { + throw new IllegalArgumentException( + "Parameter 'filePath' must not be null."); + } + if (StringUtility.isEmpty(filePath)) { + throw new IllegalArgumentException( + "Parameter 'filePath' must not be empty."); + } + File newDocumentFile = new File(filePath); + String newDocumentContent; + try { + newDocumentContent = FileUtility.readString(newDocumentFile, + FileUtility.MAX_SIZE_UNLIMITED); + } catch (CoreException ex) { + newDocumentContent = compilerSyntax + .getSingleLineCommentDelimiters().get(0) + + " " + + ex.getMessage(); + } + IDocument newDocument = new Document(newDocumentContent); + CompilerSourceFile newSourceFile = createCompilerSourceFile( + newDocumentFile, newDocument); + return newSourceFile; + } + + /** + * Parse the new input and builds up the parse tree. + * + * @param compilerSourceFile + * The file to be parsed, not null. + * @param compilerSourceParserLineCallback + * The callback to be notified when a certain line is encountered + * or null. + */ + public final void parse(CompilerSourceFile compilerSourceFile, + CompilerSourceParserLineCallback compilerSourceParserLineCallback) { + if (compilerSourceFile == null) { + throw new IllegalArgumentException( + "Parameter 'compilerSourceFile' must not be null."); + } + Map parsedFiles; + parsedFiles = new HashMap(); + parseInternal(compilerSourceFile, parsedFiles, + compilerSourceParserLineCallback); + return; + } + + /** + * Parse the new input and builds up the parse tree recursively with + * collecting already parsed includes. + * + * @param compilerSourceFile + * The file to be parsed, not null. + * @param parsedFiles + * The list of already parsed file names to prevent recursion, + * not null. + * @param compilerSourceParserLineCallback + * The callback to be notified when a certain line is encountered + * or null. + * @return true if the file was parsed now, false + * if the file is already in the list of parsed files. + */ + private boolean parseInternal(CompilerSourceFile compilerSourceFile, + Map parsedFiles, + CompilerSourceParserLineCallback compilerSourceParserLineCallback) { + if (compilerSourceFile == null) { + throw new IllegalArgumentException( + "Parameter 'compilerSourceFile' must not be null."); + } + if (parsedFiles == null) { + throw new IllegalArgumentException( + "Parameter 'parsedFiles' must not be null."); + } + + // Ensure every persistent file is parsed only once to prevent infinite + // recursions cause by circular includes. Non-persistent files cannot + // cause recursions since they cannot be included yet. + if (compilerSourceFile.getDocumentFile() != null) { + String key = compilerSourceFile.getDocumentFile().getPath(); + if (parsedFiles.containsKey(key)) { + return false; + } + parsedFiles.put(key, compilerSourceFile); + } + + this.compilerSourceFile = compilerSourceFile; + + // To allow folding for introduction comment at the begin of the source, + // the definition section is always open already, even if it does not + // contain any definitions. + child = compilerSourceFile.getDefinitionSection(); + beginSection(0, true); + + IDocument document = compilerSourceFile.getDocument(); + int lines = document.getNumberOfLines(); + int lineOffset, lineLength, startOffset, endOffset; + String stringLine = ""; + + // Prepare line and document offsets. + lineOffset = 0; + lineLength = 0; + startOffset = 0; + endOffset = 0; + + // Prepare line section buffers. + StringBuilder symbolBuffer; + StringBuilder instructionBuffer; + StringBuilder operandBuffer; + StringBuilder commentBuffer; + char blockDefinitonStartCharacter; + char blockDefinitonEndCharacter; + + symbolBuffer = new StringBuilder(100); + instructionBuffer = new StringBuilder(100); + operandBuffer = new StringBuilder(100); + commentBuffer = new StringBuilder(100); + blockDefinitonStartCharacter = compilerSyntax + .getBlockDefinitionStartCharacter(); + blockDefinitonEndCharacter = compilerSyntax + .getBlockDefinitionEndCharacter(); + + for (int lineNumber = 0; lineNumber < lines; lineNumber++) { + + /** + * Part 1: Parse line segments from line strin. + */ + + int symbolOffset = 0; + boolean symbolOffsetFound = false; + symbolBuffer.setLength(0); + int instructionOffset = 0; + boolean instructionOffsetFound = false; + instructionBuffer.setLength(0); + int operandOffset = 0; + boolean operandOffsetFound = false; + operandBuffer.setLength(0); + int commentOffset = 0; + boolean commentOffsetFound = false; + commentBuffer.setLength(0); + + try { + IRegion region = document.getLineInformation(lineNumber); + lineOffset = region.getOffset(); + lineLength = region.getLength(); + stringLine = document.get(lineOffset, lineLength); + + int pos = 0; + char lastChar = 0; + int lineSection = LineSection.NONE; + while (pos < lineLength) { + char ch = stringLine.charAt(pos); + boolean whiteSpace = Character.isWhitespace(ch); + // Find the next word. + if (pos == 0 + || (!whiteSpace && Character.isWhitespace(lastChar))) { + + // Does the current section allow instructions? + if (CompilerSourceParserTreeObjectType + .areInstructionsAllowed(section.getType())) { + if (lineSection == LineSection.NONE) { + lineSection = LineSection.SYMBOL; + } else if (lineSection == LineSection.SYMBOL) { + lineSection = LineSection.INSTRUCTION; + if (symbolBuffer.length() > 0) { + String possibleInstruction = symbolBuffer + .toString().toUpperCase(); + if (isInstruction(possibleInstruction)) { + + instructionOffset = symbolOffset; + instructionOffsetFound = true; + instructionBuffer.append(symbolBuffer); + symbolOffset = 0; + symbolBuffer.setLength(0); + lineSection = LineSection.OPERAND; + } + } + } else if (lineSection == LineSection.INSTRUCTION) { + lineSection = LineSection.OPERAND; + } + } else { + // No instructions allowed. + if (!symbolOffsetFound) { + if (!whiteSpace + && lineSection == LineSection.NONE) { + lineSection = LineSection.SYMBOL; + } + } else { + lineSection = LineSection.OPERAND; + } + } + + } + String type = document.getPartition(lineOffset + pos) + .getType(); + if (type.equals(IDocument.DEFAULT_CONTENT_TYPE)) { + if (lineSection == LineSection.SYMBOL) { + + // TODO: Does not work with kernel equates + // if (symbolBuffer.length() == 0 && + // compilerSyntax.isIdentifierStartCharacter(ch) + // || symbolBuffer.length() > 0 && + // compilerSyntax.isIdentifierPartCharacter(ch)) + if (compilerSyntax.isIdentifierCharacter(ch)) { + if (!symbolOffsetFound) { + symbolOffsetFound = true; + symbolOffset = pos; + } + symbolBuffer.append(ch); + + } + } else if (lineSection == LineSection.INSTRUCTION) { + if (!whiteSpace) { + if (!instructionOffsetFound) { + instructionOffsetFound = true; + instructionOffset = pos; + } + instructionBuffer.append(ch); + } + } else { + if (!operandOffsetFound) { + operandOffsetFound = true; + operandOffset = pos; + } + operandBuffer.append(ch); + } + } else if (type + .equals(CompilerSourcePartitionScanner.PARTITION_COMMENT_SINGLE)) { + if (!commentOffsetFound) { + commentOffsetFound = true; + commentOffset = pos; + } + // Keep spaces within comments and convert tabs to + // spaces. + if (ch == 0x9) { + ch = ' '; + } + if (ch != 0xa && ch != 0xd) { + commentBuffer.append(ch); + } + } else if (type + .equals(CompilerSourcePartitionScanner.PARTITION_STRING)) { + operandBuffer.append(ch); + } + + lastChar = ch; + pos++; + } + + } catch (BadLocationException ex) { + throw new RuntimeException(ex); + } + + /** + * Part 2: Post processing of line segments + */ + startOffset = lineOffset; + try { + endOffset = startOffset + document.getLineLength(lineNumber); + } catch (BadLocationException ex) { + throw new RuntimeException(ex); + } + + // Check if the single symbol in the line is actually an + // instruction. + String possibleInstruction = symbolBuffer.toString().toUpperCase(); + if (isInstruction(possibleInstruction)) { + + instructionOffset = symbolOffset; + instructionBuffer.append(symbolBuffer); + symbolOffset = 0; + symbolBuffer.setLength(0); + } + + String symbol = symbolBuffer.toString(); + String instruction = instructionBuffer.toString(); + if (!instructionSet.areInstructionsCaseSensitive()) { + instruction = instruction.toUpperCase(); + } + + // Refine operand and detect block start and end. + String operand = operandBuffer.toString().trim(); + blockStarting = false; + if (operandOffsetFound) { + int blockStartOffset = operand + .indexOf(blockDefinitonStartCharacter); + if (blockStartOffset > -1) { + operand = operand.substring(0, blockStartOffset); + blockStarting = true; + } + } + blockEnding = false; + int blockEndOffset = stringLine.indexOf(blockDefinitonEndCharacter); + if (blockEndOffset > -1) { + if (!commentOffsetFound || blockEndOffset < commentOffset) { + blockEnding = true; + } + } + + // Refine comment. Strip leading single comment sign. + String comment = commentBuffer.toString(); + if (comment.length() > 0) { + for (String singleLineCommentDelimiter : compilerSyntax + .getSingleLineCommentDelimiters()) { + if (comment.startsWith(singleLineCommentDelimiter)) { + comment = comment.substring(singleLineCommentDelimiter + .length()); + } + } + comment = comment.trim(); + } + + /** + * Part 3: Parse labels or equates, either directly or via + * delegation. + */ + // This is an instance variable modified per line. + labelChild = null; + + // Depending on the current context parsing is delegated or not. + switch (section.getType()) { + case CompilerSourceParserTreeObjectType.ENUM_DEFINITION_SECTION: + if (symbol.length() > 0) { + String value = operand.trim(); + + if (value.startsWith("=")) { + value = value.substring(1).trim(); + } + if (value.length() == 0) { + value = "(auto)"; + } + createEquateDefinitionChild(startOffset, startOffset + + symbolOffset, symbol, value, comment); + } + break; + case CompilerSourceParserTreeObjectType.STRUCTURE_DEFINITION_SECTION: + if (symbol.length() > 0) { + createLabelDefinitionChild(startOffset, startOffset + + symbolOffset, symbol, comment); + } + break; + default: + parseLine(startOffset, symbol, symbolOffset, instruction, + instructionOffset, operand, comment); + break; + } + + /** + * Part 4: Instruction based parsing + */ + int positionStartOffset = startOffset + instructionOffset; + parseInstruction(startOffset, endOffset, positionStartOffset, + symbol, instruction, operand, comment, parsedFiles, + compilerSourceParserLineCallback); + // Label not yet consumed by instruction, so it is a single label. + if (labelChild != null) { + section.addChild(labelChild); + } + + /** + * Part 6: Handle generic block starts and ends. + */ + if (blockStarting && child == null) { + // Block started but not yet converted into a child like a macro + // or procedure. + beginFolding(startOffset); + } + if (blockEnding) { + // Arbitrary block end, can be section or folding. + if (compilerSourceFile.isFoldingForSection()) { + endSection(startOffset + blockEndOffset); + } else { + endFolding(startOffset + blockEndOffset); + } + } + + /** + * Part 7: Callback for dedicated source file lines. + */ + // Was the current file and line relevant for the callback? + if (compilerSourceParserLineCallback != null + && compilerSourceParserLineCallback.getSourceFilePath() + .equals(compilerSourceFile.getDocumentFile() + .getPath()) + && lineNumber == compilerSourceParserLineCallback + .getLineNumber()) { + compilerSourceParserLineCallback.processLine(this, + compilerSourceFile, lineNumber, startOffset, + symbolOffset, isInstruction(instruction), + instructionOffset, instruction, operandOffset, section); + } + } + + // End last section. + endSection(document.getLength() > 0 ? document.getLength() - 1 : 0); + + // End incomplete sections. + compilerSourceFile.endAllSections(); + + // End incomplete sections. + compilerSourceFile.endAllFoldings(); + + return true; + } + + private boolean isInstruction(String instructionName) { + if (instructionName == null) { + throw new IllegalArgumentException( + "Parameter 'instructionName' must not be null."); + } + boolean result; + result = instructionSet.getInstruction(instructionName) != null; + return result; + } + + protected void parseLine(int startOffset, String symbol, int symbolOffset, + String instruction, int instructionOffset, String operand, + String comment) { + return; + } + + protected final void createEquateDefinitionChild(int startOffset, + int positionStartOffset, String symbol, String operand, + String comment) { + ensureDefinitionSection(startOffset, positionStartOffset); + createChild(positionStartOffset, + CompilerSourceParserTreeObjectType.EQUATE_DEFINITION, symbol, + symbol + " = " + operand, comment); + section.addChild(child); + + } + + protected final void createLabelDefinitionChild(int startOffset, + int positionStartOffset, String symbol, String comment) { + createChild(positionStartOffset, + CompilerSourceParserTreeObjectType.LABEL_DEFINITION, symbol, + symbol, comment); + // Remember the label child. It will be added only if the label is not + // consumed by some other instruction. + labelChild = child; + } + + protected final void beginImplementationSection(int startOffset, + int positionStartOffset, String operand, String comment) { + + if (section.getParent() != null) { + return; + } + int endOffset = startOffset - 1; + if (endOffset < 0) { + endOffset = 0; + } + endSection(endOffset); + + // Folding for an implementation section should only bebe active if + // there is´a name section in the code. + boolean withFolding = (StringUtility.isSpecified(operand)); + if (StringUtility.isEmpty(operand)) { + operand = "Implementation Section"; + } + + createChild(positionStartOffset, + CompilerSourceParserTreeObjectType.IMPLEMENTATION_SECTION, + operand, operand, comment); + + // This will always be a top level section. + beginSection(startOffset, withFolding); + compilerSourceFile.getImplementationSections().add(section); + labelChild = null; + } + + /** + * Parse the instruction in a single line. + * + * @param startOffset + * The start offset of the line. + * @param endOffset + * The end offset of the line, including its last character and + * the line delimiter is available. + * @param positionStartOffset + * The start offset for positioning the cursor. + * @param symbol + * The symbol name, may be empty, not null. + * @param instructionName + * The instruction name, may be empty, not null. + * @param operand + * The operand, may be empty, not null. + * @param comment + * The comment, may be empty, not null. + * @param parsedFiles + * The parsed files, not null. + * @param compilerSourceParserLineCallback + * The callback to be notified when a certain line is encountered + * or null. + */ + private void parseInstruction(int startOffset, int endOffset, + int positionStartOffset, String symbol, String instructionName, + String operand, String comment, + Map parsedFiles, + CompilerSourceParserLineCallback compilerSourceParserLineCallback) { + + if (StringUtility.isEmpty(instructionName)) { + return; + } + Instruction instruction = instructionSet + .getInstruction(instructionName); + + if (instruction == null) { + return; + } + + if (logEnabled) { + log("parseInstruction: startOffset={0} endOffset={1} positionStartOffset={2}, symbol={3} instructionName={4} symbol={5} instructionType={6} labelChild={7}", + Integer.toString(startOffset), Integer.toString(endOffset), + Integer.toString(positionStartOffset), symbol, + instructionName, operand, + Integer.toString(instruction.getType()), labelChild); + } + + String symbolOrOperand; + String symbolOrOperandFirstWord; + symbolOrOperand = symbol; + if (StringUtility.isEmpty(symbolOrOperand)) { + symbolOrOperand = operand; + } + symbolOrOperandFirstWord = symbolOrOperand; + int i = 0; + for (; i < symbolOrOperandFirstWord.length(); i++) { + if (Character.isWhitespace(symbolOrOperandFirstWord.charAt(i))) { + break; + } + } + symbolOrOperandFirstWord = symbolOrOperandFirstWord.substring(0, i); + + switch (instruction.getType()) { + case InstructionType.BEGIN_IMPLEMENTATION_SECTION_DIRECTIVE: + beginImplementationSection(startOffset, positionStartOffset, + operand, comment); + break; + case InstructionType.DIRECTIVE: + if (blockStarting) { + beginFolding(startOffset); + } + break; + case InstructionType.BEGIN_FOLDING_BLOCK_DIRECTIVE: + beginFolding(startOffset); + break; + case InstructionType.END_FOLDING_BLOCK_DIRECTIVE: + endFolding(endOffset); + break; + case InstructionType.END_SECTION_DIRECTIVE: + endSection(endOffset); + break; + + case InstructionType.BEGIN_ENUM_DEFINITION_SECTION_DIRECTIVE: + ensureDefinitionSection(startOffset, positionStartOffset); + createChild(positionStartOffset, + CompilerSourceParserTreeObjectType.ENUM_DEFINITION_SECTION, + symbolOrOperandFirstWord, symbolOrOperand, comment); + beginSection(startOffset, true); + break; + + case InstructionType.BEGIN_STRUCTURE_DEFINITION_SECTION_DIRECTIVE: + ensureDefinitionSection(startOffset, positionStartOffset); + createChild( + positionStartOffset, + CompilerSourceParserTreeObjectType.STRUCTURE_DEFINITION_SECTION, + symbolOrOperandFirstWord, symbolOrOperand, comment); + beginSection(startOffset, true); + break; + + case InstructionType.BEGIN_LOCAL_SECTION_DIRECTIVE: + ensureImplementationSection(startOffset, positionStartOffset); + createChild(positionStartOffset, + CompilerSourceParserTreeObjectType.LOCAL_SECTION, + symbolOrOperandFirstWord, symbolOrOperand, comment); + beginSection(startOffset, true); + + break; + case InstructionType.BEGIN_MACRO_DEFINITION_SECTION_DIRECTIVE: + ensureDefinitionSection(startOffset, positionStartOffset); + createChild( + positionStartOffset, + CompilerSourceParserTreeObjectType.MACRO_DEFINITION_SECTION, + symbolOrOperandFirstWord, symbolOrOperand, comment); + beginSection(startOffset, true); + break; + case InstructionType.BEGIN_PROCEDURE_DEFINITION_SECTION_DIRECTIVE: + ensureImplementationSection(startOffset, positionStartOffset); + createChild( + positionStartOffset, + CompilerSourceParserTreeObjectType.PROCEDURE_DEFINITION_SECTION, + symbolOrOperandFirstWord, symbolOrOperand, comment); + beginSection(startOffset, true); + break; + case InstructionType.BEGIN_PAGES_SECTION_DIRECTIVE: + ensureImplementationSection(startOffset, positionStartOffset); + createChild(positionStartOffset, + CompilerSourceParserTreeObjectType.PAGES_SECTION, "", + symbolOrOperand, comment); + beginSection(startOffset, true); + break; + case InstructionType.BEGIN_REPEAT_SECTION_DIRECTIVE: + createChild(positionStartOffset, + CompilerSourceParserTreeObjectType.REPEAT_SECTION, "", + symbolOrOperand, comment); + beginSection(startOffset, true); + break; + + case InstructionType.SOURCE_INCLUDE_DIRECTIVE: + // Remove leading and trailing string delimiters. + String filePath = operand; + for (String stringDelimiter : compilerSyntax.getStringDelimiters()) { + if (filePath.startsWith(stringDelimiter)) { + filePath = filePath.substring(stringDelimiter.length()); + break; + } + } + for (String stringDelimiter : compilerSyntax.getStringDelimiters()) { + if (filePath.endsWith(stringDelimiter)) { + filePath = filePath.substring(0, filePath.length() + - stringDelimiter.length()); + break; + } + } + filePath = getIncludeAbsoluteFilePath( + CompilerSourceParserFileReferenceType.SOURCE, + compilerSourceFile.getDocumentDirectory(), filePath); + + createChild(positionStartOffset, + CompilerSourceParserTreeObjectType.SOURCE_INCLUDE, "", + operand, comment); + + // If there is no file, the include is a normal child. + if (filePath == null) { + if (labelChild == null) { + section.addChild(child); + } else { + labelChild.addChild(child); + } + } else { + // If there is a file, the include is a section. + beginSection(startOffset, true); + + // Preserve current line specific state into local variables. + CompilerSourceFile oldSourceFile = compilerSourceFile; + CompilerSourceParserTreeObject oldSection = section; + CompilerSourceParserTreeObject oldChild = child; + CompilerSourceParserTreeObject oldLabelChild = labelChild; + boolean oldBlockStarting = blockStarting; + boolean oldBlockEnding = blockEnding; + + CompilerSourceFile newSourceFile = createCompilerSourceFile(filePath); + + // This might be moved to the createCompilerSourceFile() method. + CompilerSourcePartitionScanner partitionScanner = new CompilerSourcePartitionScanner( + compilerSyntax); + partitionScanner.createDocumentPartitioner(newSourceFile + .getDocument()); + boolean parsed = parseInternal(newSourceFile, parsedFiles, + compilerSourceParserLineCallback); + + if (parsed) { + // Restore old line specific state from local variables. + section = oldSection; + compilerSourceFile = oldSourceFile; + child = oldChild; + labelChild = oldLabelChild; + blockStarting = oldBlockStarting; + blockEnding = oldBlockEnding; + + section.setIncludedCompilerSourceFile(newSourceFile); + List newSourceFileSections = newSourceFile + .getSections(); + if (newSourceFileSections.size() == 1 + && newSourceFileSections.get(0).getType() == CompilerSourceParserTreeObjectType.SOURCE_INCLUDE) { + newSourceFileSections = newSourceFileSections.get(0) + .getChildren(); + } + for (CompilerSourceParserTreeObject newChild : newSourceFileSections) { + section.addChild(newChild); + } + } else { + AssemblerPlugin + .getInstance() + .log("Include file '{0}' was already parsed. Stopping recursion.", + new Object[] { newSourceFile + .getDocumentFile().getPath() }); + } + endSection(endOffset); + } + break; + case InstructionType.BINARY_INCLUDE_DIRECTIVE: + ensureImplementationSection(startOffset, positionStartOffset); + createChild(positionStartOffset, + CompilerSourceParserTreeObjectType.BINARY_INCLUDE, "", + operand, comment); + if (labelChild == null) { + section.addChild(child); + } else { + labelChild.addChild(child); + } + + break; + case InstructionType.BINARY_OUTPUT_DIRECTIVE: + + createChild(positionStartOffset, + CompilerSourceParserTreeObjectType.BINARY_OUTPUT, "", + operand, comment); + if (labelChild == null) { + section.addChild(child); + } else { + labelChild.addChild(child); + } + + break; + } + + } + + private void ensureDefinitionSection(int startOffset, + int positionStartOffset) { + // To allow folding for introduction comment at the begin of the source, + // the definition section is always open already. + } + + private void ensureImplementationSection(int startOffset, + int positionStartOffset) { + if (section.getType() == CompilerSourceParserTreeObjectType.DEFINITION_SECTION) { + beginImplementationSection(startOffset, positionStartOffset, "", ""); + } + } + + private void createChild(int startOffset, int type, String name, + String displayName, String comment) { + if (startOffset < 0) { + throw new IllegalArgumentException( + "Parameter 'startOffset' must not be negative. Specified value is " + + startOffset + "."); + } + if (name == null) { + throw new IllegalArgumentException( + "Parameter 'name' must not be null."); + } + if (displayName == null) { + throw new IllegalArgumentException( + "Parameter 'displayName' must not be null."); + } + if (comment == null) { + throw new IllegalArgumentException( + "Parameter 'comment' must not be null."); + } + + child = new CompilerSourceParserTreeObject(compilerSourceFile, + startOffset, type, name, displayName, comment); + } + + /** + * Starts a new active section. + * + * @param startOffset + * The start offset, a non-negative integer. + * @param withFolding + * true if the section is also a folding section, + * false otherwise. + */ + private void beginSection(int startOffset, boolean withFolding) { + if (startOffset < 0) { + throw new IllegalArgumentException( + "Parameter 'startOffset' must not be negative. Specified value is " + + startOffset + "."); + } + if (child == null) { + throw new IllegalStateException("Field 'child' must not be null."); + } + section = child; + compilerSourceFile.beginSection(startOffset, section, withFolding); + labelChild = null; + blockStarting = false; + } + + /** + * Ends the currently active section. + * + * @param endOffset + * The end offset, a non-negative integer. + */ + private void endSection(int endOffset) { + if (section == null) { + throw new IllegalStateException( + "Variable 'section' must not be null."); + } + if (endOffset < 0) { + throw new IllegalArgumentException( + "Parameter 'endOffset' must not be negative. Specified value is " + + endOffset + "."); + } + CompilerSourceParserTreeObject section; + section = compilerSourceFile.endSection(endOffset); + if (section != null) { + this.section = section; + } + labelChild = null; + blockEnding = false; + } + + /** + * Starts a new active folding which does not belong to a section. + * + * @param startOffset + * The start offset, a non-negative integer. + */ + private void beginFolding(int startOffset) { + compilerSourceFile.beginFolding(startOffset, false); + blockStarting = false; + } + + /** + * Ends the currently active folding which does not belong to a section. + * + * @param endOffset + * The end offset, a non-negative integer. This must be the + * actual offset of the last character in the line, including the + * line end delimiter, if present. + */ + private void endFolding(int endOffset) { + compilerSourceFile.endFolding(endOffset); + blockEnding = false; + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceParserFileReference.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceParserFileReference.java new file mode 100644 index 00000000..a3c191c6 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceParserFileReference.java @@ -0,0 +1,72 @@ +package com.wudsn.ide.asm.compiler.parser; + +public final class CompilerSourceParserFileReference { + + private int type; + private int directiveEndOffset; + + /** + * Creation is public. + */ + public CompilerSourceParserFileReference() { + type = CompilerSourceParserFileReferenceType.NONE; + directiveEndOffset = 0; + + } + + /** + * Sets the type of the include. + * + * @param type + * The type of the include, see + * {@link CompilerSourceParserFileReferenceType}. + */ + public void setType(int type) { + switch (type) { + case CompilerSourceParserFileReferenceType.SOURCE: + case CompilerSourceParserFileReferenceType.BINARY: + break; + + default: + throw new IllegalArgumentException("Illegal include type " + type + + "."); + } + this.type = type; + } + + /** + * Get the type of the include. + * + * @return The type of the include, see + * {@link CompilerSourceParserFileReferenceType}. + */ + public int getType() { + return type; + } + + /** + * Sets the include directive end offset. The value must be the offset of + * first character after the directive. + * + * @param directiveEndOffset + * The include directive end offset, a non-negative integer. + */ + public void setDirectiveEndOffset(int directiveEndOffset) { + if (directiveEndOffset < 0) { + throw new IllegalArgumentException( + "Parameter 'directiveEndOffset' must not be be negative. Specified value was " + + directiveEndOffset + "."); + } + this.directiveEndOffset = directiveEndOffset; + } + + /** + * Gets the include directive end offset. + * + * @return The include directive end offset, a non-negative integer. + */ + public int getDirectiveEndOffset() { + return directiveEndOffset; + } + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceParserFileReferenceType.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceParserFileReferenceType.java new file mode 100644 index 00000000..de62c379 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceParserFileReferenceType.java @@ -0,0 +1,38 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.parser; + +/** + * The possible types of referenced files. + * + * @author Peter Dell + */ +public final class CompilerSourceParserFileReferenceType { + + /** + * Creation is private. + */ + private CompilerSourceParserFileReferenceType() { + } + + public final static int NONE=1; + public final static int SOURCE = 2; + public final static int BINARY = 3; +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceParserLineCallback.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceParserLineCallback.java new file mode 100644 index 00000000..630a287b --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceParserLineCallback.java @@ -0,0 +1,85 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.parser; + +/** + * Callback interface to react on the parsing of certain source file lines. + * + * @author Peter Dell + * + * @since 1.6.0 + */ +public abstract class CompilerSourceParserLineCallback { + private String filePath; + private int lineNumber; + + /** + * Creation is protected. + * + * @param filePath + * The absolute path of the source file, not empty and not + * null. + * @param lineNumber + * The line number, a non-negative integer or -1 to + * indicate that no line number is relevant. + */ + protected CompilerSourceParserLineCallback(String filePath, int lineNumber) { + if (filePath == null) { + throw new IllegalArgumentException( + "Parameter 'filePath' must not be null."); + } + if (lineNumber < 0) { + throw new IllegalArgumentException( + "Parameter 'lineNumber' must not be negative. Specified value is " + + lineNumber + "."); + } + this.filePath = filePath; + this.lineNumber = lineNumber; + } + + /** + * Gets the path of the source file for which for which the callback shall + * be triggered. + * + * @return The absolute path of the source file, not empty and not + * null. + */ + public final String getSourceFilePath() { + return filePath; + } + + /** + * Gets the line number for which the callback shall be triggered. + * + * @return The line number, a non-negative integer or -1 to + * indicate that no line number is relevant. + */ + public final int getLineNumber() { + return lineNumber; + } + + // Most of the parameters are currently not used by the consumer. + public abstract void processLine(CompilerSourceParser compilerSourceParser, + CompilerSourceFile compilerSourceFile, int lineNumber, + int startOffset, int symbolOffset, boolean instructionFound, + int instructionOffset, String instruction, int operandOffset, + CompilerSourceParserTreeObject section); + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceParserTreeObject.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceParserTreeObject.java new file mode 100644 index 00000000..b2faaa1f --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceParserTreeObject.java @@ -0,0 +1,411 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.parser; + +import java.util.ArrayList; +import java.util.List; + +import org.eclipse.jface.viewers.StyledString; + +import com.wudsn.ide.asm.compiler.syntax.CompilerSyntax; +import com.wudsn.ide.base.common.StringUtility; + +/** + * The object representing a node in the content outline tree. + * + * @author Peter Dell + * @author Andy Reek + */ +public final class CompilerSourceParserTreeObject { + + private final CompilerSourceFile compilerSourceFile; + + private final int startOffset; + + private final int type; + + private final String name; + + private final String displayName; + + private final String description; + + private final StyledString styledString; + + private final List children; + + private CompilerSourceParserTreeObject parent; + + private String treePath; + + private String compoundName; + + private CompilerSourceFile includedCompilerSourceFile; + + /** + * Create a new instance. + * + * @param compilerSourceFile + * The source file to which this parser tree object belongs. + * + * @param startOffset + * The start offset of the instance, a non-negative integer. + * @param type + * The type of a instance, see + * {@link CompilerSourceParserTreeObjectType}. + * @param name + * The name of the tree object, not null. It is used + * to build the full tree path, + * @param displayName + * The display name of the tree object, not null. It + * is used in the tree view. + * @param description + * The description of the tree object, not null. + */ + CompilerSourceParserTreeObject(CompilerSourceFile compilerSourceFile, + int startOffset, int type, String name, String displayName, + String description) { + if (compilerSourceFile == null) { + throw new IllegalArgumentException( + "Parameter 'compilerSourceFile' must not be null."); + } + if (startOffset < 0) { + throw new IllegalArgumentException( + "Parameter 'startOffset' must not be negative. Specified value is " + + startOffset + "."); + } + this.compilerSourceFile = compilerSourceFile; + this.startOffset = startOffset; + + switch (type) { + case CompilerSourceParserTreeObjectType.DEFAULT: + + case CompilerSourceParserTreeObjectType.DEFINITION_SECTION: + case CompilerSourceParserTreeObjectType.IMPLEMENTATION_SECTION: + + case CompilerSourceParserTreeObjectType.EQUATE_DEFINITION: + case CompilerSourceParserTreeObjectType.LABEL_DEFINITION: + + case CompilerSourceParserTreeObjectType.ENUM_DEFINITION_SECTION: + case CompilerSourceParserTreeObjectType.STRUCTURE_DEFINITION_SECTION: + case CompilerSourceParserTreeObjectType.LOCAL_SECTION: + case CompilerSourceParserTreeObjectType.MACRO_DEFINITION_SECTION: + case CompilerSourceParserTreeObjectType.PAGES_SECTION: + case CompilerSourceParserTreeObjectType.PROCEDURE_DEFINITION_SECTION: + case CompilerSourceParserTreeObjectType.REPEAT_SECTION: + + case CompilerSourceParserTreeObjectType.SOURCE_INCLUDE: + case CompilerSourceParserTreeObjectType.BINARY_INCLUDE: + case CompilerSourceParserTreeObjectType.BINARY_OUTPUT: + + break; + + default: + throw new IllegalArgumentException("Unknown type " + type + "."); + } + if (name == null) { + throw new IllegalArgumentException( + "Parameter 'name' must not be null."); + } + if (displayName == null) { + throw new IllegalArgumentException( + "Parameter 'displayName' must not be null."); + } + if (description == null) { + throw new IllegalArgumentException( + "Parameter 'description' must not be null."); + } + this.type = type; + this.name = name; + this.displayName = displayName; + this.description = description; + + styledString = new StyledString(displayName); + if (description.length() > 0) { + styledString.append(" "); + styledString.append(description, StyledString.QUALIFIER_STYLER); + } + this.children = new ArrayList(); + } + + /** + * Gets the compiler source file, this parser tree object belongs to. + * + * @return The compiler source file, this parser tree object belongs to, not + * null. + */ + public CompilerSourceFile getCompilerSourceFile() { + return compilerSourceFile; + } + + /** + * Gets the start offset of this parser tree object in the compiler source + * file. + * + * @return The start offset of this parser tree object in the compiler + * source file, a non-negative integer. + */ + public int getStartOffset() { + return startOffset; + } + + /** + * Gets the type of the object. + * + * @return The type, see {@link CompilerSourceParserTreeObjectType}. + */ + public int getType() { + return type; + } + + /** + * Gets the name of the object. + * + * @return The name, not null. + */ + public String getName() { + return name; + } + + /** + * Gets the compound name of the object. + * + * @return The compound name, not null. + */ + public String getCompoundName() { + if (compoundName == null) { + char identifierSeparatorCharacter = compilerSourceFile + .getCompilerSyntax().getIdentifierSeparatorCharacter(); + switch (type) { + case CompilerSourceParserTreeObjectType.DEFINITION_SECTION: + case CompilerSourceParserTreeObjectType.IMPLEMENTATION_SECTION: + case CompilerSourceParserTreeObjectType.REPEAT_SECTION: + case CompilerSourceParserTreeObjectType.SOURCE_INCLUDE: + case CompilerSourceParserTreeObjectType.BINARY_INCLUDE: + case CompilerSourceParserTreeObjectType.BINARY_OUTPUT: + if (parent != null) { + compoundName = parent.getCompoundName(); + } else { + compoundName = ""; + } + return compoundName; + } + + compoundName = name; + + // Compound identifiers supported? + if (identifierSeparatorCharacter != CompilerSyntax.NO_CHARACTER) { + + // Find next named parent. + CompilerSourceParserTreeObject namedParent = parent; + while (namedParent != null + // && namedParent.getType() != + // CompilerSourceParserTreeObjectType.DEFINITION_SECTION + // && namedParent.getType() != + // CompilerSourceParserTreeObjectType.IMPLEMENTATION_SECTION + // && namedParent.getType() != + // CompilerSourceParserTreeObjectType.REPEAT_SECTION + // && namedParent.getType() != + // CompilerSourceParserTreeObjectType.SOURCE_INCLUDE + // && namedParent.getType() != + // CompilerSourceParserTreeObjectType.BINARY_INCLUDE + // && namedParent.getType() != + // CompilerSourceParserTreeObjectType.BINARY_OUTPUT + + && namedParent.getType() != CompilerSourceParserTreeObjectType.ENUM_DEFINITION_SECTION + && namedParent.getType() != CompilerSourceParserTreeObjectType.STRUCTURE_DEFINITION_SECTION + && namedParent.getType() != CompilerSourceParserTreeObjectType.LOCAL_SECTION + && namedParent.getType() != CompilerSourceParserTreeObjectType.MACRO_DEFINITION_SECTION + && namedParent.getType() != CompilerSourceParserTreeObjectType.PROCEDURE_DEFINITION_SECTION) { + namedParent = namedParent.getParent(); + } + if (namedParent != null + && StringUtility.isSpecified(compoundName)) { + String parentCompoundName; + parentCompoundName = parent.getCompoundName(); + if (StringUtility.isSpecified(parentCompoundName)) { + compoundName = parentCompoundName + + identifierSeparatorCharacter + compoundName; + } + } + } + } + + return compoundName; + + } + + /** + * Gets the display name of the object. + * + * @return The display name, not null. + */ + public String getDisplayName() { + return displayName; + } + + /** + * Gets the unique tree path of this tree object within the tree. + * + * @return The unique tree path, not empty and not null. + */ + public String getTreePath() { + if (treePath == null) { + treePath = "\"" + type + "/" + name + "\""; + if (parent != null) { + treePath = parent.getTreePath() + "/" + treePath; + } + } + return treePath; + } + + /** + * Gets the description of the object. + * + * @return The description, not null. + */ + public String getDescription() { + return description; + } + + /** + * Gets the styled string of the object. + * + * @return The styled string, not null. + */ + public StyledString getStyledString() { + return styledString; + } + + /** + * Gets the parent object of the tree object. + * + * @return The parent, may be null. + */ + public CompilerSourceParserTreeObject getParent() { + return parent; + } + + /** + * Sets a new parent. + * + * @param parent + * The new parent, may be null. + */ + final void setParent(CompilerSourceParserTreeObject parent) { + this.parent = parent; + this.treePath = null; + } + + /** + * Determines if this tree object has children. + * + * @return true if this tree object has children, + * false otherwise. + */ + public boolean hasChildren() { + return !children.isEmpty(); + } + + /** + * Gets all children of the tree object. + * + * @return The array of children, not null.. + */ + public Object[] getChildrenAsArray() { + return children.toArray(new Object[0]); + } + + /** + * Gets all children of the tree object. + * + * @return The array of children, not null. Do not modify. + */ + public List getChildren() { + return children; + } + + /** + * Adds a new Child. + * + * @param child + * The new child, not null. + */ + final void addChild(CompilerSourceParserTreeObject child) { + if (child == null) { + throw new IllegalArgumentException( + "Parameter 'child' must not be null."); + } + if (child == this) { + throw new IllegalArgumentException( + "Parameter 'child' must not be this: " + toString()); + } + children.add(child); + child.setParent(this); + + } + + /** + * Gets the included compiler source file for a SOURCE_INCLUDE tree object. + * + * @return The included compiler source file or null. + */ + final CompilerSourceFile setIncludedCompilerSourceFile() { + return includedCompilerSourceFile; + } + + /** + * Sets the included compiler source file for a SOURCE_INCLUDE tree object. + * + * @param includedCompilerSourceFile + * The included compiler source file, may be null. + */ + final void setIncludedCompilerSourceFile( + CompilerSourceFile includedCompilerSourceFile) { + if (type != CompilerSourceParserTreeObjectType.SOURCE_INCLUDE) { + throw new IllegalStateException("The type of this tree object is " + + type + " and not SOURCE_INCLUDE"); + } + this.includedCompilerSourceFile = includedCompilerSourceFile; + + } + + + @Override + public boolean equals(Object object) { + if (!(object instanceof CompilerSourceParserTreeObject)) { + return false; + } + CompilerSourceParserTreeObject other; + other = (CompilerSourceParserTreeObject) object; + return getTreePath().equals(other.getTreePath()); + } + + @Override + public int hashCode() { + return getTreePath().hashCode(); + + } + + @Override + public String toString() { + return getTreePath(); + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceParserTreeObjectType.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceParserTreeObjectType.java new file mode 100644 index 00000000..6f74f2af --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourceParserTreeObjectType.java @@ -0,0 +1,143 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ +package com.wudsn.ide.asm.compiler.parser; + +import com.wudsn.ide.asm.Texts; + +/** + * Constant values for tree object types. + * + * @author Peter Dell + */ +public final class CompilerSourceParserTreeObjectType { + + /** + * Creation is private. + */ + private CompilerSourceParserTreeObjectType() { + } + + /** + * Default-type + */ + public static final int DEFAULT = 0; + + public static final int DEFINITION_SECTION = 1; + public static final int IMPLEMENTATION_SECTION = 2; + + public static final int EQUATE_DEFINITION = 3; + public static final int LABEL_DEFINITION = 4; + + public static final int ENUM_DEFINITION_SECTION = 5; + public static final int STRUCTURE_DEFINITION_SECTION = 6; + public static final int LOCAL_SECTION = 7; + public static final int MACRO_DEFINITION_SECTION = 8; + public static final int PAGES_SECTION = 9; + public static final int PROCEDURE_DEFINITION_SECTION = 10; + public static final int REPEAT_SECTION = 11; + + public static final int SOURCE_INCLUDE = 12; + public static final int BINARY_INCLUDE = 13; + public static final int BINARY_OUTPUT = 14; + + /** + * Gets the localized text for a compiler source parser tree object type. + * + * @param type + * The type, see constants of this class. + * @return The localized text, may be empty but not null. + */ + public static String getText(int type) { + String result; + switch (type) { + case DEFAULT: + result = Texts.COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_DEFAULT; + break; + case DEFINITION_SECTION: + result = Texts.COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_DEFINITION_SECTION; + break; + case IMPLEMENTATION_SECTION: + result = Texts.COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_IMPLEMENTATION_SECTION; + break; + case EQUATE_DEFINITION: + result = Texts.COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_EQUATE_DEFINITION; + break; + case LABEL_DEFINITION: + result = Texts.COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_LABEL_DEFINITION; + break; + case ENUM_DEFINITION_SECTION: + result = Texts.COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_ENUM_DEFINITION_SECTION; + break; + case STRUCTURE_DEFINITION_SECTION: + result = Texts.COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_STRUCTURE_DEFINITION_SECTION; + break; + case LOCAL_SECTION: + result = Texts.COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_LOCAL_SECTION; + break; + case MACRO_DEFINITION_SECTION: + result = Texts.COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_MACRO_DEFINITION_SECTION; + break; + case PAGES_SECTION: + result = Texts.COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_PAGES_SECTION; + break; + case PROCEDURE_DEFINITION_SECTION: + result = Texts.COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_PROCEDURE_DEFINITION_SECTION; + break; + case REPEAT_SECTION: + result = Texts.COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_REPEAT_SECTION; + break; + case SOURCE_INCLUDE: + result = Texts.COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_SOURCE_INCLUDE; + break; + case BINARY_INCLUDE: + result = Texts.COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_BINARY_INCLUDE; + break; + case BINARY_OUTPUT: + result = Texts.COMPILER_SOURCE_PARSER_TREE_OBJECT_TYPE_BINARY_OUTPUT; + break; + default: + throw new IllegalArgumentException("Unknown type " + type + "."); + } + return result; + + } + + /** + * Determines if instructions are allowed in the given type of section. + * + * @param type + * The type of the tree object, {link + * {@link CompilerSourceParserTreeObjectType} + * @return true if instructions are allowed, false + * otherwise. + * + * @since 1.6.0 + */ + public static boolean areInstructionsAllowed(int type) { + // Within ENUM_DEFINITION_SECTION everything is an equate. + // Within STRUCTURE_DEFINITION_SECTION everything is a label. + switch (type) { + case CompilerSourceParserTreeObjectType.ENUM_DEFINITION_SECTION: + case CompilerSourceParserTreeObjectType.STRUCTURE_DEFINITION_SECTION: + return false; + default: + return true; + } + } +} \ No newline at end of file diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourcePartitionScanner.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourcePartitionScanner.java new file mode 100644 index 00000000..8e3e08c0 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/parser/CompilerSourcePartitionScanner.java @@ -0,0 +1,141 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.parser; + +import java.util.ArrayList; +import java.util.List; + +import org.eclipse.jface.text.IDocument; +import org.eclipse.jface.text.rules.EndOfLineRule; +import org.eclipse.jface.text.rules.FastPartitioner; +import org.eclipse.jface.text.rules.IPredicateRule; +import org.eclipse.jface.text.rules.IRule; +import org.eclipse.jface.text.rules.IToken; +import org.eclipse.jface.text.rules.MultiLineRule; +import org.eclipse.jface.text.rules.RuleBasedPartitionScanner; +import org.eclipse.jface.text.rules.SingleLineRule; +import org.eclipse.jface.text.rules.Token; + +import com.wudsn.ide.asm.compiler.syntax.CompilerSyntax; +import com.wudsn.ide.asm.editor.AssemblerEditor; + +/** + * A partition scanner for the comments and strings. + * + * @author Peter Dell + * @author Andy Reek + */ +public final class CompilerSourcePartitionScanner extends + RuleBasedPartitionScanner { + + /** + * Name for the single line comment partition. + */ + public static final String PARTITION_COMMENT_SINGLE = "partition.comment.single"; //$NON-NLS-1$ + + /** + * Name for the multiple lines comment partition. + */ + public static final String PARTITION_COMMENT_MULTIPLE = "partition.comment.multiple"; //$NON-NLS-1$ + + /** + * Name for the string partition. + */ + public static final String PARTITION_STRING = "partition.string"; //$NON-NLS-1$ + + /** + * Creates a new instance. + * + * Called by + * {@link AssemblerEditor#init(org.eclipse.ui.IEditorSite, org.eclipse.ui.IEditorInput)} + * . + * + * @param compilerSyntax + * The compiler syntax, not null. + */ + public CompilerSourcePartitionScanner(CompilerSyntax compilerSyntax) { + if (compilerSyntax == null) { + throw new IllegalArgumentException( + "Parameter 'compilerSyntax' must not be null."); + } + IToken commentSingleToken = new Token(PARTITION_COMMENT_SINGLE); + IToken commentMultipleToken = new Token(PARTITION_COMMENT_MULTIPLE); + IToken stringToken = new Token(PARTITION_STRING); + + List rules = new ArrayList(); + for (String singleLineCommentDelimiter : compilerSyntax + .getSingleLineCommentDelimiters()) { + + // A "*" is only a comment start token if it is followed by a space + // or a tab. + // It is allowed as part of an expression to refer to the program + // counter. + if (singleLineCommentDelimiter.equals("*")) { + rules.add(new EndOfLineRule(singleLineCommentDelimiter + " ", + commentSingleToken)); + rules.add(new EndOfLineRule(singleLineCommentDelimiter + "\t", + commentSingleToken)); + } else { + rules.add(new EndOfLineRule(singleLineCommentDelimiter, + commentSingleToken)); + } + } + List multipleLinesCommentDelimiters = compilerSyntax + .getMultipleLinesCommentDelimiters(); + for (int i = 0; i < multipleLinesCommentDelimiters.size();) { + String startSequence = multipleLinesCommentDelimiters.get(i++); + String endSequence = multipleLinesCommentDelimiters.get(i++); + rules.add(new MultiLineRule(startSequence, endSequence, + commentMultipleToken)); + } + + for (String stringDelimiter : compilerSyntax.getStringDelimiters()) { + rules.add(new SingleLineRule(stringDelimiter, stringDelimiter, + stringToken)); + } + + + IPredicateRule[] rulesArray = new IPredicateRule[rules.size()]; + rules.toArray(rulesArray); + setPredicateRules(rulesArray); + } + + /** + * Creates a new FastPartitioner based on this partition scanner and + * connects it to the document. + * + * @param document + * The document, not null. + */ + public void createDocumentPartitioner(IDocument document) { + + if (document == null) { + throw new IllegalArgumentException( + "Parameter 'document' must not be null."); + } + FastPartitioner partitioner = new FastPartitioner(this, new String[] { + CompilerSourcePartitionScanner.PARTITION_COMMENT_SINGLE, + CompilerSourcePartitionScanner.PARTITION_COMMENT_MULTIPLE, + CompilerSourcePartitionScanner.PARTITION_STRING }); + partitioner.connect(document); + document.setDocumentPartitioner(partitioner); + + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/CompilerSyntax.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/CompilerSyntax.java new file mode 100644 index 00000000..c01c76f5 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/CompilerSyntax.java @@ -0,0 +1,936 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.syntax; + +import java.io.IOException; +import java.io.InputStream; +import java.util.ArrayList; +import java.util.Collections; +import java.util.List; +import java.util.Map; +import java.util.Set; +import java.util.StringTokenizer; +import java.util.TreeMap; +import java.util.TreeSet; + +import javax.xml.parsers.ParserConfigurationException; +import javax.xml.parsers.SAXParser; +import javax.xml.parsers.SAXParserFactory; + +import org.xml.sax.Attributes; +import org.xml.sax.SAXException; +import org.xml.sax.SAXParseException; +import org.xml.sax.helpers.DefaultHandler; + +import com.wudsn.ide.asm.CPU; +import com.wudsn.ide.asm.compiler.CompilerRegistry; +import com.wudsn.ide.base.common.StringUtility; + +/** + * Container for a set of directives, legal, illegal and pseudo opcodes and + * their properties. + * + * @author Peter Dell + * @author Daniel Mitte + * + */ +public final class CompilerSyntax { + + public static final char NO_CHARACTER = (char) 0; + + private static final class XMLHandler extends DefaultHandler { + + // Completion proposal auto activation characters, possibly empty array + // of characters. + String completionProposalAutoActivationCharactersText; + + // Single line delimiters, list of strings separated by spaces. + String singleLineCommentDelimitersText; + + // Single line delimiters, list of strings separated by spaces. + String multipleLinesCommentDelimitersText; + + // Single line delimiters, non empty array of characters. + String stringDelimiterCharactersText; + + // The (possibly empty) string of character pairs which define the begin + // and end of a named or unnamed (folding) block. + String blockDefinitionCharactersText; + + // The boolean value indicating if identifiers are case sensitive or + // not. + boolean identifiersCaseSensitive; + + // The string containing all allowed identifier start characters. + String identifierStartCharactersText; + + // The string containing all allowed identifier start characters. + String identifierPartCharactersText; + + // The string containing the character which separates two parts of a + // compound identifier or + // the empty string if compound identifiers are not supported. + String identifierSeparatorCharacterText; + + // The string containing the character which end a label definition or + // the empty string if label definitions end at the first white space. + String labelDefinitionSuffixCharacterText; + + // The string containing the character which start a macro usage or + // the empty string if macro usages are not prefixed with a character. + String macroUsagePrefixCharacterText; + + // The boolean value indicating if instructions are case sensitive or + // not. + boolean instructionsCaseSensitive; + + // Default file extension to be added if it is missing for source + // include directives. + String sourceIncludeDefaultExtension; + + List instructionsList; + + public XMLHandler() { + instructionsList = new ArrayList(); + } + + @Override + public void startElement(String uri, String localName, String qName, Attributes attributes) throws SAXException { + if (qName == null) { + throw new IllegalArgumentException("Parameter 'qName' must not be null."); + } + if (attributes == null) { + throw new IllegalArgumentException("Parameter 'attributes' must not be null."); + } + + if (qName.equals("instructionset")) { + + // Completion proposal auto activation characters + completionProposalAutoActivationCharactersText = attributes + .getValue("completionProposalAutoActivationCharacters"); + if (completionProposalAutoActivationCharactersText == null) { + throw new SAXException("No completionProposalAutoActivationCharacters specified."); + } + + // Single lines comments. + singleLineCommentDelimitersText = attributes.getValue("singleLineCommentDelimiters"); + if (singleLineCommentDelimitersText == null) { + throw new SAXException("No singleLineCommentDelimiters specified."); + } + if (StringUtility.isEmpty(singleLineCommentDelimitersText)) { + throw new SAXException("Attribute singleLineCommentDelimiterst must not be empty."); + } + + // Multiple lines comments. + multipleLinesCommentDelimitersText = attributes.getValue("multipleLinesCommentDelimiters"); + if (multipleLinesCommentDelimitersText == null) { + throw new SAXException("No multipleLinesCommentDelimiters specified."); + } + + // Strings. + stringDelimiterCharactersText = attributes.getValue("stringDelimiterCharacters"); + if (stringDelimiterCharactersText == null) { + throw new SAXException("No stringDelimiterCharacters specified."); + } + if (StringUtility.isEmpty(stringDelimiterCharactersText)) { + throw new SAXException("Attribute stringDelimiterCharacters must not be empty."); + } + + // Block definition characters + blockDefinitionCharactersText = attributes.getValue("blockDefinitionCharacters"); + if (blockDefinitionCharactersText == null) { + throw new SAXException("No blockDefinitionCharacters specified."); + } + if (blockDefinitionCharactersText.length() % 2 != 0) { + throw new SAXException("Attribute blockDefinitionCharacters must and even number of characters."); + } + + // Identifiers: Case sensitive. + String value = attributes.getValue("identifiersCaseSensitive"); + if (value == null) { + throw new SAXException("No identifiersCaseSensitive specified."); + } + if (!value.equals("true") && !value.equals("false")) { + throw new SAXException("Attribute identifiersCaseSensitive must be \"true\" or \"false\"."); + } + identifiersCaseSensitive = Boolean.parseBoolean(value); + + // Identifiers: Start characters. + identifierStartCharactersText = attributes.getValue("identifierStartCharacters"); + if (identifierStartCharactersText == null) { + throw new SAXException("No identifierStartCharacters specified."); + } + if (StringUtility.isEmpty(identifierStartCharactersText)) { + throw new SAXException("Attribute identifierStartCharacters must not be empty."); + } + + // Identifiers: Start characters. + identifierPartCharactersText = attributes.getValue("identifierPartCharacters"); + if (identifierPartCharactersText == null) { + throw new SAXException("No identifierPartCharacters specified."); + } + if (StringUtility.isEmpty(identifierPartCharactersText)) { + throw new SAXException("Attribute identifierPartCharacters must not be empty."); + } + + // Identifiers: Separator character. + identifierSeparatorCharacterText = attributes.getValue("identifierSeparatorCharacter"); + if (identifierSeparatorCharacterText == null) { + throw new SAXException("No identifierSeparatorCharacter specified."); + } + if (identifierSeparatorCharacterText.length() > 1) { + throw new SAXException( + "Attribute identifierSeparatorCharacter must not contain more than 1 characters."); + } + + // Identifiers: Label definition suffix character. + labelDefinitionSuffixCharacterText = attributes.getValue("labelDefinitionSuffixCharacter"); + if (labelDefinitionSuffixCharacterText == null) { + throw new SAXException("No labelDefinitionSuffixCharacter specified."); + } + if (labelDefinitionSuffixCharacterText.length() > 1) { + throw new SAXException( + "Attribute labelDefinitionSuffixCharacter must not contain more than 1 characters."); + } + + // Identifiers: Macro usage prefix character. + macroUsagePrefixCharacterText = attributes.getValue("macroUsagePrefixCharacter"); + if (macroUsagePrefixCharacterText == null) { + throw new SAXException("No macroUsagePrefixCharacter specified."); + } + if (macroUsagePrefixCharacterText.length() > 1) { + throw new SAXException( + "Attribute macroUsagePrefixCharacter must not contain more than 1 characters."); + } + + // Instructions case sensitive. + value = attributes.getValue("instructionsCaseSensitive"); + if (value == null) { + throw new SAXException("No instructionsCaseSensitive specified."); + } + if (!value.equals("true") && !value.equals("false")) { + throw new SAXException("Attribute instructionsCaseSensitive must be \"true\" or \"false\"."); + } + instructionsCaseSensitive = Boolean.parseBoolean(value); + + // Source include default extension. + sourceIncludeDefaultExtension = attributes.getValue("sourceIncludeDefaultExtension"); + if (sourceIncludeDefaultExtension == null) { + throw new SAXException("No sourceIncludeDefaultExtension specified."); + } + + } else if (qName.equals("opcodes")) { + // Nothing to do. + } else { + + // Begin parsing of instructions. + String cpusString; + Set cpus; + String name; + String title; + String proposal; + String typeString; + int type; + + name = attributes.getValue("name"); + if (name == null) { + throw new SAXException("No name specified for '" + qName + "'."); + } + + cpusString = attributes.getValue("cpus"); + if (cpusString == null) { + throw new SAXException("No CPUs specified for '" + name + "'."); + } + cpus = new TreeSet(); + StringTokenizer tokenizer = new StringTokenizer(cpusString, ","); + while (tokenizer.hasMoreTokens()) { + String token = tokenizer.nextToken(); + boolean found = false; + + // Wild card? + if (token.endsWith("*")) { + token = token.substring(0, token.length() - 1); + for (CPU cpu : CPU.values()) { + if (cpu.name().startsWith(token) || cpu == CPU.MOS65C02 && token.equals("MOS6502")) { + cpus.add(cpu); + found = true; + } + } + } else { + // Exact match + for (CPU cpu : CPU.values()) { + if (cpu.name().equals(token)) { + cpus.add(cpu); + found = true; + } + } + } + if (!found) { + throw new SAXException("No cpu matches the cpus '" + cpusString + "' for '" + name + "'."); + } + } + + title = attributes.getValue("title"); + if (title == null) { + throw new SAXException("No title specified for '" + name + "'."); + } + + typeString = attributes.getValue("type"); + if (typeString == null) { + typeString = ""; + } + + proposal = attributes.getValue("proposal"); + if (qName.equals("constant")) { + // Constants always have a simple proposal + proposal = name + "_"; + } + if (proposal == null) { + throw new SAXException("No proposal specified for '" + name + "'."); + } + + // TODO Have constant as own instruction class + if (qName.equals("constant")) { + qName = "directive"; + typeString = "DIRECTIVE"; + } + + if (qName.equals("directive")) { + + // Default and default nesting for folding blocks and + // sections. + if (typeString.equals("DIRECTIVE")) { + type = InstructionType.DIRECTIVE; + } else if (typeString.equals("BEGIN_IMPLEMENTATION_SECTION_DIRECTIVE")) { + type = InstructionType.BEGIN_IMPLEMENTATION_SECTION_DIRECTIVE; + } else if (typeString.equals("BEGIN_FOLDING_BLOCK_DIRECTIVE")) { + type = InstructionType.BEGIN_FOLDING_BLOCK_DIRECTIVE; + } else if (typeString.equals("END_FOLDING_BLOCK_DIRECTIVE")) { + type = InstructionType.END_FOLDING_BLOCK_DIRECTIVE; + } else if (typeString.equals("END_SECTION_DIRECTIVE")) { + type = InstructionType.END_SECTION_DIRECTIVE; + } + + // Begin of sections. + else if (typeString.equals("BEGIN_ENUM_DEFINITION_SECTION_DIRECTIVE")) { + type = InstructionType.BEGIN_ENUM_DEFINITION_SECTION_DIRECTIVE; + } else if (typeString.equals("BEGIN_STRUCTURE_DEFINITION_SECTION_DIRECTIVE")) { + type = InstructionType.BEGIN_STRUCTURE_DEFINITION_SECTION_DIRECTIVE; + } else if (typeString.equals("BEGIN_LOCAL_SECTION_DIRECTIVE")) { + type = InstructionType.BEGIN_LOCAL_SECTION_DIRECTIVE; + } else if (typeString.equals("BEGIN_MACRO_DEFINITION_SECTION_DIRECTIVE")) { + type = InstructionType.BEGIN_MACRO_DEFINITION_SECTION_DIRECTIVE; + } else if (typeString.equals("BEGIN_PAGES_SECTION_DIRECTIVE")) { + type = InstructionType.BEGIN_PAGES_SECTION_DIRECTIVE; + } else if (typeString.equals("BEGIN_PROCEDURE_DEFINITION_SECTION_DIRECTIVE")) { + type = InstructionType.BEGIN_PROCEDURE_DEFINITION_SECTION_DIRECTIVE; + } else if (typeString.equals("BEGIN_REPEAT_SECTION_DIRECTIVE")) { + type = InstructionType.BEGIN_REPEAT_SECTION_DIRECTIVE; + } + + // File references. + else if (typeString.equals("SOURCE_INCLUDE_DIRECTIVE")) { + type = InstructionType.SOURCE_INCLUDE_DIRECTIVE; + } else if (typeString.equals("BINARY_INCLUDE_DIRECTIVE")) { + type = InstructionType.BINARY_INCLUDE_DIRECTIVE; + } else if (typeString.equals("BINARY_OUTPUT_DIRECTIVE")) { + type = InstructionType.BINARY_OUTPUT_DIRECTIVE; + } else if (typeString.equals("")) { + throw new SAXException("No directive type specified for directive '" + name + "'."); + } else { + throw new SAXException("Unknown directive type '" + typeString + "' for directive '" + name + + "'."); + } + instructionsList.add(new Directive(cpus, type, instructionsCaseSensitive, name, title, proposal)); + } else if (qName.equals("opcode") || qName.equals("illegalopcode") || qName.equals("pseudoopcode")) { + + if (qName.equals("opcode")) { + type = InstructionType.LEGAL_OPCODE; + } else if (qName.equals("illegalopcode")) { + type = InstructionType.ILLEGAL_OPCODE; + } else if (qName.equals("pseudoopcode")) { + type = InstructionType.PSEUDO_OPCODE; + } else { + throw new SAXException("Unknown qName '" + qName + "'."); + } + + String flags = attributes.getValue("flags"); + if (flags == null) { + flags = "@todo flags"; + } + String modes = attributes.getValue("modes"); + if (modes == null) { + modes = "todo=$00"; + } + + // Currently only KickAss uses case sensitive instructions + // and they are lower case. There this logic is applied here + // instead of having an additional XML attribute to control + // the lower/upper case setting for opcodes. + if (instructionsCaseSensitive) { + name = name.toLowerCase(); + proposal = proposal.toLowerCase(); + } + instructionsList.add(new Opcode(cpus, type, instructionsCaseSensitive, name, title, proposal, cpus + .contains(CPU.MOS65816), flags, modes)); + } + } + } + } + + private String compilerId; + + private char[] completionProposalAutoActivationCharacters; + + private List singleLineCommentDelimiters; + + private List multipleLinesCommentDelimiters; + + private List stringDelimiters; + + private String blockDefinitionCharacters; + private char blockDefinitionStartCharacter; + private char blockDefinitionEndCharacter; + + private boolean identifiersCaseSensitive; + + private String identifierStartCharacters; + private boolean identifierStartCharactersArray[]; + + private String identifierPartCharacters; + private boolean identifierPartCharactersArray[]; + + private char identifierSeparatorCharacter; + + private char labelDefinitionSuffixCharacter; + + private char macroUsagePrefixCharacter; + + private boolean instructionsCaseSensitive; + + private String sourceIncludeDefaultExtension; + + private List instructionList; + private Map instructionSets; + + /** + * Creates new instance. Called by {@link CompilerRegistry}. + * + * @param compilerId + * The compiler id, not empty and null. + */ + public CompilerSyntax(String compilerId) { + if (compilerId == null) { + throw new IllegalArgumentException("Parameter 'compilerId' must not be null."); + } + if (StringUtility.isEmpty(compilerId)) { + throw new IllegalArgumentException("Parameter 'compilerId' must not be empty."); + } + this.compilerId = compilerId; + } + + /** + * Load XML data. + * + * @param compilerClasses + * The list of compiler classes for which the XML files shall be + * loaded, not empty and not null. The content of + * subsequent files will extend the content of previous files. + */ + public void loadXMLData(List> compilerClasses) { + + if (compilerClasses == null) { + throw new IllegalArgumentException("Parameter 'compilerClasses' must not be null."); + } + if (compilerClasses.isEmpty()) { + throw new IllegalArgumentException("Parameter 'compilerClasses' must not be empty."); + } + XMLHandler xmlHandler; + xmlHandler = new XMLHandler(); + for (int i = 0; i < compilerClasses.size(); i++) { + Class compilerClass = compilerClasses.get(i); + if (compilerClass == null) { + throw new IllegalArgumentException("Parameter 'compilerClasses[" + i + "]' must not be null."); + } + + SAXParser parser; + try { + parser = SAXParserFactory.newInstance().newSAXParser(); + } catch (ParserConfigurationException ex) { + throw new RuntimeException( + "Cannot create parser for compiler class '" + compilerClass.getName() + "'.", ex); + } catch (SAXException ex) { + throw new RuntimeException( + "Cannot create parser for compiler class '" + compilerClass.getName() + "'.", ex); + } + + String syntaxFileName = "/" + compilerClass.getName().replace('.', '/') + ".xml"; + try { + + InputStream inputStream = compilerClass.getResourceAsStream(syntaxFileName); + parser.parse(inputStream, xmlHandler); + } catch (SAXParseException ex) { + throw new RuntimeException("Cannot create parser for file '" + syntaxFileName + "'. Error in line " + + ex.getLineNumber() + ", column " + ex.getColumnNumber() + ".", ex); + } catch (SAXException ex) { + throw new RuntimeException("Cannot create parser for file '" + syntaxFileName + "'.", ex); + } catch (IOException ex) { + throw new RuntimeException("Cannot create parser for file '" + syntaxFileName + "'.", ex); + } + } + + // Completion proposal auto activation characters + completionProposalAutoActivationCharacters = xmlHandler.completionProposalAutoActivationCharactersText + .toCharArray(); + + // Single line comments. + String delimiterText = xmlHandler.singleLineCommentDelimitersText; + if (delimiterText == null) { + throw new IllegalStateException("Attribute 'singleLineCommentDelimiters' is no set."); + } + StringTokenizer tokenizer = new StringTokenizer(delimiterText, " "); + singleLineCommentDelimiters = new ArrayList(tokenizer.countTokens()); + while (tokenizer.hasMoreTokens()) { + String token = tokenizer.nextToken(); + singleLineCommentDelimiters.add(token); + } + singleLineCommentDelimiters = Collections.unmodifiableList(singleLineCommentDelimiters); + + // Multiple lines comments. + delimiterText = xmlHandler.multipleLinesCommentDelimitersText; + if (delimiterText == null) { + throw new IllegalStateException("Attribute 'singleLineCommentDelimiters' is no set."); + } + tokenizer = new StringTokenizer(delimiterText, " "); + int tokenCount = tokenizer.countTokens(); + if ((tokenCount & 1) == 1) { + throw new IllegalStateException("Attribute 'multipleLinesCommentDelimiters' has an odd number of tokens: '" + + delimiterText + "'."); + } + multipleLinesCommentDelimiters = new ArrayList(tokenCount); + while (tokenizer.hasMoreTokens()) { + multipleLinesCommentDelimiters.add(tokenizer.nextToken()); + } + multipleLinesCommentDelimiters = Collections.unmodifiableList(multipleLinesCommentDelimiters); + + // Strings. + delimiterText = xmlHandler.stringDelimiterCharactersText; + if (delimiterText == null) { + throw new IllegalStateException("Attribute 'stringDelimiters' is not set."); + } + stringDelimiters = new ArrayList(delimiterText.length()); + for (int i = 0; i < delimiterText.length(); i++) { + stringDelimiters.add(delimiterText.substring(i, i + 1)); + } + stringDelimiters = Collections.unmodifiableList(stringDelimiters); + + // Block definitions. + if (xmlHandler.blockDefinitionCharactersText == null) { + throw new IllegalArgumentException("Attribute 'blockDefinitionCharacters' is not set."); + } + blockDefinitionCharacters = xmlHandler.blockDefinitionCharactersText; + if (blockDefinitionCharacters.length() > 0) { + if (blockDefinitionCharacters.length() == 2) { + blockDefinitionStartCharacter = blockDefinitionCharacters.charAt(0); + blockDefinitionEndCharacter = blockDefinitionCharacters.charAt(1); + } else { + throw new IllegalArgumentException("Attribute 'blockDefinitionCharacters' has the value '" + + blockDefinitionCharacters + "' and does not have 2 characters."); + } + } else { + blockDefinitionStartCharacter = NO_CHARACTER; + blockDefinitionEndCharacter = NO_CHARACTER; + } + + // Identifiers: Case sensitive. + identifiersCaseSensitive = xmlHandler.identifiersCaseSensitive; + + // Identifiers: Start characters. + if (xmlHandler.identifierStartCharactersText == null) { + throw new IllegalArgumentException("Attribute 'identifierStartCharacters' is not set."); + } + identifierStartCharacters = xmlHandler.identifierStartCharactersText; + identifierStartCharactersArray = createBooleanArray(identifierStartCharacters); + + // Identifiers: Start characters. + if (xmlHandler.identifierPartCharactersText == null) { + throw new IllegalArgumentException("Attribute 'identifierPartCharacters' is not set."); + } + identifierPartCharacters = xmlHandler.identifierPartCharactersText; + identifierPartCharactersArray = createBooleanArray(identifierPartCharacters); + + // Identifiers: Separator characters. + if (xmlHandler.identifierSeparatorCharacterText == null) { + throw new IllegalArgumentException("Attribute 'identifierSeparatorCharacter' is not set."); + } + if (xmlHandler.identifierSeparatorCharacterText.length() == 0) { + identifierSeparatorCharacter = NO_CHARACTER; + } else { + identifierSeparatorCharacter = xmlHandler.identifierSeparatorCharacterText.charAt(0); + } + + // Identifiers: Label definition suffix character. + if (xmlHandler.labelDefinitionSuffixCharacterText == null) { + throw new IllegalArgumentException("Attribute 'labelDefinitionSuffixCharacterText' is not set."); + } + if (xmlHandler.labelDefinitionSuffixCharacterText.length() == 0) { + labelDefinitionSuffixCharacter = NO_CHARACTER; + } else { + labelDefinitionSuffixCharacter = xmlHandler.labelDefinitionSuffixCharacterText.charAt(0); + } + + // Identifiers: Macro usage prefix character + if (xmlHandler.macroUsagePrefixCharacterText == null) { + throw new IllegalArgumentException("Attribute 'macroUsagePrefixCharacterText' is not set."); + } + if (xmlHandler.macroUsagePrefixCharacterText.length() == 0) { + macroUsagePrefixCharacter = NO_CHARACTER; + } else { + macroUsagePrefixCharacter = xmlHandler.macroUsagePrefixCharacterText.charAt(0); + } + + // Instructions case sensitive. + instructionsCaseSensitive = xmlHandler.instructionsCaseSensitive; + + // Source include default extension. + sourceIncludeDefaultExtension = xmlHandler.sourceIncludeDefaultExtension; + + instructionList = Collections.unmodifiableList(xmlHandler.instructionsList); + + // Create instruction set map. + instructionSets = new TreeMap(); + } + + static boolean[] createBooleanArray(String string) { + int length = string.length(); + int max = 0; + for (int i = 0; i < length; i++) { + char c = string.charAt(i); + if (c > max) { + max = c; + } + } + boolean[] result = new boolean[max + 1]; + + for (int i = 0; i < length; i++) { + char c = string.charAt(i); + result[c] = true; + } + return result; + } + + /** + * Gets the completion proposal auto activation characters. + * + * @return The array of completion proposal auto activation characters, may + * be empty, not null. + */ + public char[] getCompletionProposalAutoActivationCharacters() { + if (completionProposalAutoActivationCharacters == null) { + throw new IllegalStateException("Attribute 'completionProposalAutoActivationCharacters' is not set."); + } + return completionProposalAutoActivationCharacters; + } + + /** + * Gets the delimiter prefixes for single line comments. + * + * @return The unmodifiable list of delimiter prefixes for single line + * comments, not empty and not null. + */ + public List getSingleLineCommentDelimiters() { + if (singleLineCommentDelimiters == null) { + throw new IllegalStateException("Attribute 'singleLineCommentDelimiters' is not set."); + } + return singleLineCommentDelimiters; + } + + /** + * Gets the delimiter prefixes for multiple lines comments. + * + * @return The unmodifiable list of delimiter prefixes for single line + * comments, not empty and not null. The list contains + * an even number of entries where two entries constitute the start + * sequence and the end sequence of the multiple lines rules. + */ + public List getMultipleLinesCommentDelimiters() { + if (multipleLinesCommentDelimiters == null) { + throw new IllegalStateException("Attribute 'multipleLinesCommentDelimiters' is not set."); + } + return multipleLinesCommentDelimiters; + } + + /** + * Gets the delimiter characters for strings. + * + * @return The unmodifiable list of delimiter characters for strings, not + * empty and not null. + */ + public List getStringDelimiters() { + if (stringDelimiters == null) { + throw new IllegalStateException("Attribute 'stringDelimiters' is not set."); + } + return stringDelimiters; + } + + /** + * Gets the (possibly empty) string of character pairs which define the + * begin and end of a named or unnamed (folding) block. + * + * @return The string of character pairs which define the begin and end of a + * named or unnamed (folding) block, may be empty, not + * null. + * @since 1.6.1 + */ + public String getBlockDefinitionCharacters() { + return blockDefinitionCharacters; + } + + /** + * Gets the block definition start character if defined. + * + * @return The block definition start character or {@link #NO_CHARACTER}. + */ + public char getBlockDefinitionStartCharacter() { + return blockDefinitionStartCharacter; + } + + /** + * Gets the block definition start character if defined. + * + * @return The block definition start character or {@link #NO_CHARACTER}. + */ + public char getBlockDefinitionEndCharacter() { + return blockDefinitionEndCharacter; + } + + /** + * Determines if identifiers are case sensitive. + * + * @return true if identifiers are case sensitive, + * false otherwise. + * + * @since 1.6.1 + */ + public boolean areIdentifiersCaseSensitive() { + return identifiersCaseSensitive; + } + + /** + * Determines if a character can be the start or part of an identifier. + * + * @param c + * The character to be checked. + * @return true if the character can be the start of an + * identifier, false otherwise. + */ + public boolean isIdentifierCharacter(char c) { + return isIdentifierStartCharacter(c) || isIdentifierPartCharacter(c) + || (c == identifierSeparatorCharacter && c != NO_CHARACTER); + } + + /** + * Gets the valid identifier start characters. + * + * @return The non empty string of identifier start characters, not + * null. + * + * @since 1.6.3 + */ + public String getIdentifierStartCharacters() { + return identifierStartCharacters; + } + + /** + * Determines if a character can be the start of an identifier. + * + * @param c + * The character to be checked. + * @return true if the character can be the start of an + * identifier, false otherwise. + * + * @since 1.6.1 + */ + public boolean isIdentifierStartCharacter(char c) { + return c < identifierStartCharactersArray.length && identifierStartCharactersArray[c]; + } + + /** + * Gets the valid identifier part characters. + * + * @return The non empty string of identifier part characters, not + * null. + * + * @since 1.6.3 + */ + public String getIdentifierPartCharacters() { + return identifierPartCharacters; + } + + /** + * Determines if a character can be the part of an identifier. + * + * @param c + * The character to be checked. + * @return true if the character can be part of an identifier, + * false otherwise. + * @since 1.6.1 + */ + public boolean isIdentifierPartCharacter(char c) { + return c < identifierPartCharactersArray.length && identifierPartCharactersArray[c]; + } + + /** + * Gets the character which separates two parts of a compound identifier. + * + * @return The character which separates two parts of a compound identifier + * or {@link #NO_CHARACTER} if compound identifiers are not + * supported. + */ + public char getIdentifierSeparatorCharacter() { + return identifierSeparatorCharacter; + } + + /** + * Determines if a character can be part of an identifier and separates two + * parts of a compound identifier characters. + * + * @param c + * The character to be checked. + * @return true if the character can be part of an identifier + * and separates two parts of a compound identifier, + * false otherwise. + * @since 1.6.1 + */ + public boolean isIdentifierSeparatorCharacter(char c) { + return (c != NO_CHARACTER) && (c == identifierSeparatorCharacter); + } + + /** + * Gets the character which end a label definition, e.g. ':'. + * + * @return The character which ends a label definition or + * {@link #NO_CHARACTER} if label definitions end at the first white + * space. + * + * @since 1.6.1 + */ + public char getLabelDefinitionSuffixCharacter() { + return labelDefinitionSuffixCharacter; + } + + /** + * Gets the character which separates two parts of a compound identifier. + * + * @return The character which start a macro usage or {@link #NO_CHARACTER} + * if macro usages are not prefixed with a character. + * + * @since 1.6.1 + */ + public char getMacroUsagePrefixCharacter() { + return macroUsagePrefixCharacter; + } + + /** + * Determines if instructions are case sensitive. + * + * @return true if instructions are case sensitive, + * false otherwise. + * + * @since 1.6.1 + */ + public boolean areInstructionsCaseSensitive() { + return instructionsCaseSensitive; + } + + /** + * Determines if a character can be the start of a number. + * + * @param c + * The character to be checked. + * @return true if the character can be the start of a number, + * false otherwise. + * + * @since 1.6.1 + */ + public boolean isNumberStartCharacter(char c) { + return c == '$' || c == '%' || c == '~' || (c >= '0' && c <= '9'); + } + + /** + * Determines if a character can be the part of a number. + * + * @param c + * The character to be checked. + * @return true if the character can be part of a number, + * false otherwise. + * @since 1.6.1 + */ + public boolean isNumberPartCharacter(char c) { + return (c >= '0' && c <= '9') || (c >= 'A' && c <= 'F') || (c >= 'a' && c <= 'f'); + } + + /** + * Gets the default extension (excluding the period) to use in case source + * include file name does not end with an extension. + * + * @return Gets the default extension (excluding the period) to use in case + * source include file name does not end with an extension, may be + * empty, not null. + */ + public String getSourceIncludeDefaultExtension() { + if (sourceIncludeDefaultExtension == null) { + throw new IllegalStateException("Variable 'sourceIncludeDefaultExtension' not yet initialized."); + } + return sourceIncludeDefaultExtension; + } + + /** + * Gets the instruction set for a CPU. + * + * @param cpu + * The CPU this which the allowed list of instructions shall be + * returned, not null. + * @return The set of instructions supported by the compiler for the + * specified CPU, not null. + * + * @since 1.6.1 + */ + public InstructionSet getInstructionSet(CPU cpu) { + if (cpu == null) { + throw new IllegalArgumentException("Parameter 'cpu' must not be null."); + } + InstructionSet result; + synchronized (instructionSets) { + result = instructionSets.get(cpu); + if (result == null) { + result = new InstructionSet(this, instructionList, cpu); + instructionSets.put(cpu, result); + } + } + + return result; + } + + @Override + public String toString() { + return compilerId; + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/CompilerSyntaxUtility.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/CompilerSyntaxUtility.java new file mode 100644 index 00000000..3da119b0 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/CompilerSyntaxUtility.java @@ -0,0 +1,110 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.syntax; + +import com.wudsn.ide.asm.Texts; + +/** + * Utility class for compiler syntax and instructions. + * + * @author Peter Dell + * @since 1.6.1 + */ +public final class CompilerSyntaxUtility { + + /** + * Creation private. + */ + private CompilerSyntaxUtility() { + + } + + /** + * Gets the image path for an instruction type image. + * + * @param instruction + * The instruction, not null. + * @return The image path for the instruction type image, not empty and not + * null. + */ + public static String getTypeImagePath(Instruction instruction) { + if (instruction == null) { + throw new IllegalArgumentException( + "Parameter 'instruction' must not be null."); + } + String path; + + if (instruction instanceof Directive) { + path = "instruction-type-directive-16x16.gif"; + } else { + Opcode opcode = (Opcode) instruction; + switch (opcode.getType()) { + case InstructionType.LEGAL_OPCODE: + path = "instruction-type-legal-opcode-16x16.gif"; + break; + case InstructionType.ILLEGAL_OPCODE: + path = "instruction-type-illegal-opcode-16x16.gif"; + break; + case InstructionType.PSEUDO_OPCODE: + path = "instruction-type-pseudo-opcode-16x16.gif"; + break; + default: + throw new IllegalStateException("Unknown opcode type " + + opcode.getType() + "."); + } + + } + return path; + } + + /** + * Gets the text for an instruction type. + * + * @param instruction + * The instruction, not null. + * @return The text for the instruction type image, may be empty, not + * null. + */ + public static String getTypeText(Instruction instruction) { + String text; + + if (instruction instanceof Directive) { + text = Texts.COMPILER_SYNTAX_INSTRUCTION_DIRECTIVE; + } else { + Opcode opcode = (Opcode) instruction; + switch (opcode.getType()) { + case InstructionType.LEGAL_OPCODE: + text = Texts.COMPILER_SYNTAX_LEGAL_OPCODE; + break; + case InstructionType.ILLEGAL_OPCODE: + text = Texts.COMPILER_SYNTAX_ILLEGAL_OPCODE; + break; + case InstructionType.PSEUDO_OPCODE: + text = Texts.COMPILER_SYNTAX_PSEUDO_OPCODE; + break; + default: + throw new IllegalStateException("Unknown opcode type " + + opcode.getType() + "."); + } + + } + return text; + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/Directive.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/Directive.java new file mode 100644 index 00000000..eb982852 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/Directive.java @@ -0,0 +1,57 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.syntax; + +import java.util.Set; + +import com.wudsn.ide.asm.CPU; + +public final class Directive extends Instruction { + + Directive(Set cpus, int type, boolean caseSensitive, + String name, String title, String proposal) { + super(cpus, type, caseSensitive, name, title, proposal); + + switch (type) { + case InstructionType.DIRECTIVE: + case InstructionType.BEGIN_IMPLEMENTATION_SECTION_DIRECTIVE: + case InstructionType.BEGIN_FOLDING_BLOCK_DIRECTIVE: + case InstructionType.END_FOLDING_BLOCK_DIRECTIVE: + case InstructionType.END_SECTION_DIRECTIVE: + + case InstructionType.BEGIN_ENUM_DEFINITION_SECTION_DIRECTIVE: + case InstructionType.BEGIN_STRUCTURE_DEFINITION_SECTION_DIRECTIVE: + case InstructionType.BEGIN_LOCAL_SECTION_DIRECTIVE: + case InstructionType.BEGIN_MACRO_DEFINITION_SECTION_DIRECTIVE: + case InstructionType.BEGIN_PROCEDURE_DEFINITION_SECTION_DIRECTIVE: + case InstructionType.BEGIN_PAGES_SECTION_DIRECTIVE: + case InstructionType.BEGIN_REPEAT_SECTION_DIRECTIVE: + + case InstructionType.SOURCE_INCLUDE_DIRECTIVE: + case InstructionType.BINARY_INCLUDE_DIRECTIVE: + case InstructionType.BINARY_OUTPUT_DIRECTIVE: + break; + + default: + throw new IllegalArgumentException("Unknown type " + type + + " for directive '" + name + "'."); + } + } +} \ No newline at end of file diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/Instruction.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/Instruction.java new file mode 100644 index 00000000..456755dd --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/Instruction.java @@ -0,0 +1,205 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.syntax; + +import java.util.ArrayList; +import java.util.List; +import java.util.Set; + +import com.wudsn.ide.asm.CPU; +import com.wudsn.ide.base.common.NumberFactory; + +public abstract class Instruction implements Comparable { + + public static final char CURSOR = '_'; + public static final char NEWLINE = '\n'; + + private Set cpus; + private int type; + private String name; + private String upperCaseName; + private String lowerCaseName; + private String title; + private String formattedTitle; + private String styledTitle; + private int[] styledTitleOffsets; + private String proposal; + + protected Instruction(Set cpus, int type, boolean caseSensitive, + String name, String title, String proposal) { + if (cpus == null) { + throw new IllegalArgumentException( + "Parameter 'cpus' must not be null."); + } + if (name == null) { + throw new IllegalArgumentException( + "Parameter 'name' must not be null."); + } + if (title == null) { + throw new IllegalArgumentException( + "Parameter 'title' must not be null for instruction '" + + name + "'."); + } + if (proposal == null) { + throw new IllegalArgumentException( + "Parameter 'proposal' must not be null for instruction '" + + name + "'."); + } + this.cpus = cpus; + this.type = type; + this.name = name; + if (caseSensitive) { + this.upperCaseName = name; + this.lowerCaseName = name; + } else { + this.upperCaseName = name.toUpperCase(); + this.lowerCaseName = name.toLowerCase(); + } + this.title = title; + + int length = title.length(); + StringBuilder mnemonicBuilder = new StringBuilder(3); + StringBuilder formattedTitleBuilder = new StringBuilder(length); + StringBuilder styledTitleBuilder = new StringBuilder(length); + List styledTitleOffsetsList = new ArrayList(10); + for (int i = 0; i < length; i++) { + char c = title.charAt(i); + char fc; + if (c == '_') { + i++; + if (i >= title.length()) { + throw new RuntimeException("Instruction '" + name + + "' has invalid title '" + title + "'."); + } + c = title.charAt(i); + fc = Character.toUpperCase(c); + mnemonicBuilder.append(fc); + styledTitleOffsetsList.add(NumberFactory + .getInteger(styledTitleBuilder.length())); + } else { + fc = c; + } + formattedTitleBuilder.append(fc); + styledTitleBuilder.append(c); + } + this.formattedTitle = formattedTitleBuilder.toString(); + this.styledTitle = styledTitleBuilder.toString(); + int size = styledTitleOffsetsList.size(); + this.styledTitleOffsets = new int[size]; + for (int i = 0; i < size; i++) { + this.styledTitleOffsets[i] = styledTitleOffsetsList.get(i) + .intValue(); + } + + proposal = proposal.replace("\\n", "" + NEWLINE); + if (!proposal.startsWith(name)) { + throw new RuntimeException("Proposal '" + proposal + + "' of instruction '" + name + "' does not start with '" + + name + "'."); + } + if (proposal.indexOf(CURSOR) == -1) { + + throw new RuntimeException("Proposal '" + proposal + + "' of instruction '" + name + + "' does not contain cursor positioning via '_'."); + } + + this.proposal = proposal; + + // Remove all special characters like + StringBuilder mnemonicNameBuilder = new StringBuilder(upperCaseName); + for (int i = 0; i < mnemonicNameBuilder.length(); i++) { + if (!Character.isLetter(mnemonicNameBuilder.charAt(i)) + && !Character.isDigit(mnemonicNameBuilder.charAt(i))) { + mnemonicNameBuilder.deleteCharAt(i); + } + } + + String mnemonic = mnemonicBuilder.toString(); + String mnemonicName = mnemonicNameBuilder.toString(); + if (!mnemonicName.equalsIgnoreCase(mnemonic)) { + throw new RuntimeException("Menmonic '" + mnemonic + + "' derived from title '" + title + + "' with of instruction '" + name + + "' does match mnemonic '" + mnemonicName + + " derived from the name'."); + + } + } + + public Set getCPUs() { + return cpus; + } + + public int getType() { + return type; + } + + public final String getName() { + return name; + } + + public final String getUpperCaseName() { + return upperCaseName; + } + + public final String getLowerCaseName() { + return lowerCaseName; + } + + public final String getTitle() { + return title; + } + + public final String getFormattedTitle() { + return formattedTitle; + } + + public final String getStyledTitle() { + return styledTitle; + } + + public final int[] getStyledTitleOffsets() { + return styledTitleOffsets; + } + + /** + * Gets the upper case proposal text for the instruction. By default the + * proposal is the same as the {@link #getName()}. In case the instruction + * has a mandatory parameter, a space is added. In case the proposal span + * several line, they are separated via {@link #NEWLINE}. The position of + * the cursor is defined by the character {@link #CURSOR}. + * + * @return The proposal, not empty, not null. + */ + public final String getProposal() { + return proposal; + } + + @Override + public final int compareTo(Instruction o) { + return name.compareToIgnoreCase(o.name); + } + + @Override + public final String toString() { + return "type=" + type + ", name=" + name; + } +} \ No newline at end of file diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/InstructionSet.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/InstructionSet.java new file mode 100644 index 00000000..4c1eec09 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/InstructionSet.java @@ -0,0 +1,273 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.syntax; + +import java.util.ArrayList; +import java.util.Collections; +import java.util.List; +import java.util.Map; +import java.util.TreeMap; + +import com.wudsn.ide.asm.CPU; +import com.wudsn.ide.asm.compiler.CompilerRegistry; +import com.wudsn.ide.asm.compiler.syntax.Opcode.OpcodeAddressingMode; + +/** + * Container for a set of directives, legal, illegal and pseudo opcodes and + * their properties. + * + * TODO Maintain address modes for all illegal opcodes for online help + * + * @author Peter Dell + * @since 1.6.1 + */ +public final class InstructionSet { + + private CompilerSyntax compilerSyntax; + + private boolean instructionStartCharactersArray[]; + private boolean instructionPartCharactersArray[]; + + private List instructionsList; + + private Map instructionsMap; + + /** + * Array list of opcode addressing modes. + * + * @since 1.7.0 + */ + private List> opcodeAddessingModesList; + + private List fileReferenceInstructionsList; + + /** + * Creates new instance. Called by {@link CompilerRegistry}. + * + * @param compilerSyntax + * The compiler syntax, not null. + * + * @param instructionsList + * The non-filtered list of compiler instructions. + * @param cpu + * The cpu to filter by, not null. + */ + InstructionSet(CompilerSyntax compilerSyntax, List instructionsList, CPU cpu) { + + if (compilerSyntax == null) { + throw new IllegalArgumentException("Parameter 'compilerSyntax' must not be null."); + } + if (instructionsList == null) { + throw new IllegalArgumentException("Parameter 'instructionsList' must not be null."); + } + if (cpu == null) { + throw new IllegalArgumentException("Parameter 'cpu' must not be null."); + } + // Compute the list of all include instructions. + this.compilerSyntax = compilerSyntax; + this.instructionsList = new ArrayList(instructionsList.size()); + instructionsMap = new TreeMap(); + + opcodeAddessingModesList = new ArrayList>(Opcode.MAX_OPCODES); + for (int i = 0; i < Opcode.MAX_OPCODES; i++) { + opcodeAddessingModesList.add(new ArrayList()); + } + fileReferenceInstructionsList = new ArrayList(10); + + // Collect all start and part characters. + boolean caseSenstive = compilerSyntax.areIdentifiersCaseSensitive(); + StringBuilder instructionStartCharacters = new StringBuilder(512); + StringBuilder instructionPartCharacters = new StringBuilder(2048); + + for (Instruction instruction : instructionsList) { + if (!instruction.getCPUs().contains(cpu)) { + continue; + } + this.instructionsList.add(instruction); + + // If not case sensitive, the upper case and lower case + // representation is allowed + if (caseSenstive) { + instructionStartCharacters.append(instruction.getName().substring(0, 1)); + instructionPartCharacters.append(instruction.getName().substring(1)); + instructionsMap.put(instruction.getName(), instruction); + + } else { + instructionStartCharacters.append(instruction.getUpperCaseName().substring(0, 1)); + instructionPartCharacters.append(instruction.getUpperCaseName().substring(1)); + instructionStartCharacters.append(instruction.getLowerCaseName().substring(0, 1)); + instructionPartCharacters.append(instruction.getLowerCaseName().substring(1)); + instructionsMap.put(instruction.getUpperCaseName(), instruction); + } + + switch (instruction.getType()) { + case InstructionType.LEGAL_OPCODE: + case InstructionType.ILLEGAL_OPCODE: + case InstructionType.PSEUDO_OPCODE: + Opcode opcode = (Opcode) instruction; + for (OpcodeAddressingMode opcodeAddressingMode : opcode.getAddressingModes()) { + // Even if an instruction is supported by all CPUs, not all + // addressing modes may be supported by the CPU, + if (opcodeAddressingMode.getCPUs().contains(cpu)) { + List list = opcodeAddessingModesList.get(opcodeAddressingMode + .getOpcodeValue()); + if (list == null) { + list = new ArrayList(); + } + list.add(opcodeAddressingMode); + } + } + break; + + case InstructionType.SOURCE_INCLUDE_DIRECTIVE: + case InstructionType.BINARY_INCLUDE_DIRECTIVE: + case InstructionType.BINARY_OUTPUT_DIRECTIVE: + fileReferenceInstructionsList.add(instruction); + break; + + } + + } + + instructionStartCharactersArray = CompilerSyntax.createBooleanArray(instructionStartCharacters.toString()); + instructionPartCharactersArray = CompilerSyntax.createBooleanArray(instructionPartCharacters.toString()); + + this.instructionsList = Collections.unmodifiableList(this.instructionsList); + instructionsMap = Collections.unmodifiableMap(instructionsMap); + fileReferenceInstructionsList = Collections.unmodifiableList(fileReferenceInstructionsList); + } + + /** + * Gets the compiler syntax. + * + * @return The compiler syntax, not null. + */ + public CompilerSyntax getCompilerSyntax() { + return compilerSyntax; + } + + /** + * Determines if instructions are case sensitive. + * + * @return true if instructions are case sensitive, + * false otherwise. + * + * @since 1.6.1 + */ + public boolean areInstructionsCaseSensitive() { + return compilerSyntax.areInstructionsCaseSensitive(); + } + + /** + * Determines if a character can be the start of an instruction. + * + * @param c + * The character to be checked. + * @return true if the character can be the start of an + * instruction, false otherwise. + * + * @since 1.6.1 + */ + public boolean isInstructionStartCharacter(char c) { + return c < instructionStartCharactersArray.length && instructionStartCharactersArray[c]; + } + + /** + * Determines if a character can be the part of an instruction. + * + * @param c + * The character to be checked. + * @return true if the character can be part of an instruction, + * false otherwise. + * @since 1.6.1 + */ + public boolean isInstructionPartCharacter(char c) { + return c < instructionPartCharactersArray.length && instructionPartCharactersArray[c]; + } + + /** + * Gets the list of all instructions. + * + * @return The unmodifiable list of instructions, not null. + */ + public List getInstructions() { + + if (instructionsList == null) { + throw new IllegalStateException("Variable 'instructionsList' not yet initialized."); + } + return instructionsList; + } + + /** + * Gets list of opcode address modes for the given opcode value. Only + * instances that are support by the CPU of the instruction set are + * returned. + * + * @param opcodeValue + * The opcode value. + * @return The list of opcode address modes, may be empty, not + * null. + * + * @since 1.7.0 + */ + public List getOpcodeAddressingModes(int opcodeValue) { + if (opcodeAddessingModesList == null) { + throw new IllegalStateException("Variable 'opcodeAddessingModesList' not yet initialized."); + } + List result = null; + if (opcodeValue > 0 && opcodeValue < opcodeAddessingModesList.size()) { + result = opcodeAddessingModesList.get(opcodeValue); + } + if (result == null) { + result = Collections.emptyList(); + } + return result; + } + + /** + * Gets an instruction by its upper case name. + * + * @param instructionName + * The upper case name + * + * @return The instruction or null. + */ + public Instruction getInstruction(String instructionName) { + return instructionsMap.get(instructionName); + } + + /** + * Gets the list of all include instructions. + * + * @return The unmodifiable list of include instructions, not + * null. + */ + public List getFileReferenceInstructions() { + if (fileReferenceInstructionsList == null) { + throw new IllegalStateException("Variable 'fileReferenceInstructionsList' not yet initialized."); + } + return fileReferenceInstructionsList; + } + + @Override + public String toString() { + return compilerSyntax.toString() + ": " + instructionsMap.keySet().toString(); + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/InstructionType.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/InstructionType.java new file mode 100644 index 00000000..b11b608d --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/InstructionType.java @@ -0,0 +1,63 @@ +/** +* Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.syntax; + +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParserTreeObjectType; + +/** + * Instruction types as defined in the XML syntax definition. See also + * {@link CompilerSourceParserTreeObjectType}. + * + * @author Peter Dell + * + */ +public final class InstructionType { + /** + * Creation is private. + */ + private InstructionType() { + + } + + // Types of directives. + public static final int DIRECTIVE = 100; + public static final int BEGIN_IMPLEMENTATION_SECTION_DIRECTIVE = 101; + public static final int BEGIN_FOLDING_BLOCK_DIRECTIVE = 102; + public static final int END_FOLDING_BLOCK_DIRECTIVE = 103; + public static final int END_SECTION_DIRECTIVE = 104; + + public static final int BEGIN_ENUM_DEFINITION_SECTION_DIRECTIVE = 105; + public static final int BEGIN_STRUCTURE_DEFINITION_SECTION_DIRECTIVE = 106; + public static final int BEGIN_LOCAL_SECTION_DIRECTIVE = 107; + public static final int BEGIN_MACRO_DEFINITION_SECTION_DIRECTIVE = 108; + public static final int BEGIN_PROCEDURE_DEFINITION_SECTION_DIRECTIVE = 109; + public static final int BEGIN_PAGES_SECTION_DIRECTIVE = 110; + public static final int BEGIN_REPEAT_SECTION_DIRECTIVE = 111; + + public static final int SOURCE_INCLUDE_DIRECTIVE = 120; + public static final int BINARY_INCLUDE_DIRECTIVE = 121; + public static final int BINARY_OUTPUT_DIRECTIVE = 122; + + + // Types of opcodes. + public static final int LEGAL_OPCODE = 200; + public static final int ILLEGAL_OPCODE = 201; + public static final int PSEUDO_OPCODE = 202; +} \ No newline at end of file diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/Opcode.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/Opcode.java new file mode 100644 index 00000000..55fcd537 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/syntax/Opcode.java @@ -0,0 +1,199 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.syntax; + +import java.util.ArrayList; +import java.util.List; +import java.util.Set; +import java.util.TreeSet; + +import com.wudsn.ide.asm.CPU; + +public final class Opcode extends Instruction { + + public static final int MAX_OPCODES = 256; + + public final static class OpcodeAddressingMode { + private Opcode opcode; + private Set cpus; + private String addressingMode; + private int opcodeValue; + + OpcodeAddressingMode(Opcode opcode, Set cpus, String addressingMode, int opcodeValue) { + if (opcode == null) { + throw new IllegalArgumentException("Parameter 'opcode' must not be null."); + } + if (cpus == null) { + throw new IllegalArgumentException("Parameter 'cpus' must not be null for opcode '" + opcode.getName() + + "'."); + } + if (addressingMode == null) { + throw new IllegalArgumentException("Parameter 'addressingMode' must not be nullfor opcode '" + + opcode.getName() + "'."); + } + if (opcodeValue < 0 || opcodeValue > 255) { + throw new IllegalArgumentException("Parameter 'opcodeValue' has value " + opcodeValue + " for opcode '" + + opcode.getName() + "' but must be between $00 and $ff."); + + } + this.opcode = opcode; + this.cpus = cpus; + this.addressingMode = addressingMode; + this.opcodeValue = opcodeValue; + } + + public Opcode getOpcode() { + return opcode; + } + + public Set getCPUs() { + return cpus; + } + + public String getAddressingMode() { + return addressingMode; + } + + public String getFormattedText() { + + StringBuffer result = new StringBuffer(opcode.getName()); + if (addressingMode.equals("imp")) { + + } else if (addressingMode.equals("imm")) { + result.append(" #$nn"); + } else if (addressingMode.equals("zp")) { + result.append(" zp"); + } else if (addressingMode.equals("zpx")) { + result.append(" zp,x"); + } else if (addressingMode.equals("zpy")) { + result.append(" zp,y"); + } else if (addressingMode.equals("izx")) { + result.append(" (zp,x)"); + } else if (addressingMode.equals("izy")) { + result.append(" (zp),y"); + } else if (addressingMode.equals("abs")) { + result.append(" abs"); + } else if (addressingMode.equals("abx")) { + result.append(" abs,x"); + } else if (addressingMode.equals("aby")) { + result.append(" abs,y"); + } else if (addressingMode.equals("ind")) { + result.append(" (abs)"); + } else if (addressingMode.equals("rel")) { + result.append(" rel"); + } else + + // 65816 modes + if (addressingMode.equals("abl")) { + result.append(" adr (long)"); + } else if (addressingMode.equals("bm")) { + result.append(" $nn,$nn"); + } else if (addressingMode.equals("dp")) { + result.append(" (zp)"); + } else if (addressingMode.equals("ial")) { + result.append(" abs (long)"); + } else if (addressingMode.equals("iax")) { + result.append(" (abs,x)"); + } else if (addressingMode.equals("idp")) { + result.append(" (zp)"); + } else if (addressingMode.equals("rell")) { + result.append(" rel (long)"); + } else { + throw new RuntimeException("Unmapped addressing mode " + addressingMode + " for opcode " + + opcode.getName()); + } + + return result.toString(); + } + + public int getOpcodeValue() { + return opcodeValue; + } + } + + private boolean w65816; + private String flags; + private String modes; + private List addressingModes; + + Opcode(Set cpus, int type, boolean instructionsCaseSensitive, String name, String title, String proposal, + boolean w65816, String flags, String modes) { + + super(cpus, type, instructionsCaseSensitive, name, title, proposal); + switch (type) { + case InstructionType.LEGAL_OPCODE: + case InstructionType.ILLEGAL_OPCODE: + case InstructionType.PSEUDO_OPCODE: + break; + + default: + throw new IllegalArgumentException("Unknown type " + type + " for opcode '" + name + "'."); + } + if (flags == null) { + throw new IllegalArgumentException("Parameter 'flags' must not be null for opcode '" + name + "'."); + } + if (modes == null) { + throw new IllegalArgumentException("Parameter 'modes' must not be null for opcode '" + name + "'."); + + } + this.w65816 = w65816; + this.flags = flags; + this.modes = modes; + addressingModes = new ArrayList(); + Set addressingModeCPUs = cpus; + for (String mode : modes.split(",")) { + mode = mode.trim(); + String values[] = mode.split("="); + String addressingMode = values[0]; + String value = values[1].substring(1); + int index = value.indexOf("["); + if (index >= 0) { + String[] cpuNameList = value.substring(index + 1, value.length() - 1).split(";"); + value = value.substring(0, index); + addressingModeCPUs = new TreeSet(); + for (String cpuName : cpuNameList) { + addressingModeCPUs.add(CPU.valueOf(cpuName)); + } + } + int opcode = Integer.parseInt(value, 16); + OpcodeAddressingMode addressingModeInstance = new OpcodeAddressingMode(this, addressingModeCPUs, + addressingMode, opcode); + addressingModes.add(addressingModeInstance); + } + + } + + public boolean isW65816() { + return w65816; + } + + public String getFlags() { + return flags; + } + + public String getModes() { + return modes; + } + + public List getAddressingModes() { + return addressingModes; + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/writer/AppleFileWriter.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/writer/AppleFileWriter.java new file mode 100644 index 00000000..8db967c1 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/compiler/writer/AppleFileWriter.java @@ -0,0 +1,343 @@ +/** + * Copyright (C) 2009 - 2014 Peter Dell + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see . + */ + +package com.wudsn.ide.asm.compiler.writer; + +import java.io.ByteArrayOutputStream; +import java.io.File; +import java.io.IOException; +import java.io.InputStream; +import java.io.UnsupportedEncodingException; + +import org.eclipse.core.resources.IMarker; +import org.eclipse.core.runtime.CoreException; + +import com.webcodepro.applecommander.storage.Disk; +import com.webcodepro.applecommander.storage.DiskFullException; +import com.webcodepro.applecommander.storage.FileEntry; +import com.webcodepro.applecommander.storage.FormattedDisk; +import com.wudsn.ide.asm.Texts; +import com.wudsn.ide.asm.compiler.CompilerFileWriter; +import com.wudsn.ide.asm.compiler.CompilerFiles; +import com.wudsn.ide.base.common.FileUtility; +import com.wudsn.ide.base.common.MarkerUtility; +import com.wudsn.ide.base.common.StringUtility; + +/** + * Compiler file writer for Apple II + * + * @author Peter Dell + * @since 1.6.3 + */ +public final class AppleFileWriter extends CompilerFileWriter { + + private final static String HELLO_ENTRY_NAME = "HELLO"; + private final static String OUTPUT_IMAGE_ENTRY_NAME = "WORLD"; + + private final class AppleSoftTokens { + public final static byte PRINT = (byte) 0xba; + public final static byte CHRS = (byte) 0xe7; + public final static byte CALL = (byte) 0x8c; + + // public String[] tokenNames = { "END", "FOR ", "NEXT ", "DATA ", + // "INPUT ", "DEL", "DIM ", "READ ", "GR", "TEXT", "PR#", "IN#", + // "CALL ", "PLOT", "HLIN ", "VLIN ", "HGR2", "HGR", "HCOLOR=", + // "HPLOT ", "DRAW ", "XDRAW ", "HTAB ", "HOME", "ROT=", "SCALE=", + // "SHLOAD", "TRACE", "NOTRACE", "NORMAL", "INVERSE", "FLASH", + // "COLOR=", "POP", "VTAB ", "HIMEM:", "LOMEM:", "ONERR ", + // "RESUME", "RECALL", "STORE", "SPEED=", "LET ", "GOTO ", "RUN", + // "IF ", "RESTORE", "& ", "GOSUB ", "RETURN", "REM ", "STOP", + // "ON ", "WAIT", "LOAD", "SAVE", "DEF", "POKE ", "PRINT ", + // "CONT", "LIST", "CLEAR", "GET ", "NEW", "TAB(", "TO ", "FN", + // "SPC(", "THEN ", "AT ", "NOT ", "STEP ", "+ ", "- ", "* ", + // "/ ", "^ ", "AND ", "OR ", "> ", "= ", "< ", "SGN", "INT", + // "ABS", "USR", "FRE", "SCRN(", "PDL", "POS ", "SQR", "RND", + // "LOG", "EXP", "COS", "SIN", "TAN", "ATN", "PEEK", "LEN", + // "STR$", "VAL", "ASC", "CHR$", "LEFT$", "RIGHT$", "MID$" }; + + // public int[] tokenAddresses = { 0xD870, 0xD766, 0xDCF9, 0xD995, + // 0xDBB2, + // 0xF331, 0xDFD9, 0xDBE2, 0xF390, 0xF399, 0xF1E5, 0xF1DE, 0xF1D5, + // 0xF225, 0xF232, 0xF241, 0xF3D8, 0xF3E2, 0xF6E9, 0xF6FE, 0xF769, + // 0xF76F, 0xF7E7, 0xFC58, 0xF721, 0xF727, 0xF775, 0xF26D, 0xF26F, + // 0xF273, 0xF277, 0xF280, 0xF24F, 0xD96B, 0xF256, 0xF286, 0xF2A6, + // 0xF2CB, 0xF318, 0xF3BC, 0xF39F, 0xF262, 0xDA46, 0xD93E, 0xD912, + // 0xD9C9, 0xD849, 0x03F5, 0xD921, 0xD96B, 0xD9DC, 0xD86E, 0xD9EC, + // 0xE784, 0xD8C9, 0xD8B0, 0xE313, 0xE77B, 0xFDAD5, 0xD896, + // 0xD6A5, 0xD66A, 0xDBA0, 0xD649, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + // 0, 0, 0, 0, 0, 0, 0, 0, 0xEB90, 0xEC23, 0xEBAF, 0x000A, 0xE2DE, + // 0xD412, 0xDFCD, 0xE2FF, 0xEE8D, 0xEFAE, 0xE941, 0xEF09, 0xEFEA, + // 0xEFF1, 0xF03A, 0xF09E, 0xE764, 0xE6D6, 0xE3C5, 0xE707, 0xE6E5, + // 0xE646, 0xE65A, 0xE686, 0xE691 }; + } + + @Override + public boolean createOrUpdateDiskImage(CompilerFiles files) { + String imageFilePath = files.outputFilePathWithoutExtension + ".dsk"; + String outputImageEntryName = OUTPUT_IMAGE_ENTRY_NAME; + String outputImageEntryTitle = "Loading " + files.outputFileNameWithoutExtension; + boolean autoCreate = true; + + if (StringUtility.isSpecified(imageFilePath)) { + File imageFile = new File(imageFilePath); + if (!imageFile.exists()) { + + // When auto creation is active, we copy a template file. + // This is the only way to get a disk image which is not only + // formatted but also has a boot loader which runs "HELLO". + if (autoCreate) { + String resource = "/lib/AppleDos.dsk"; + InputStream inputStream = Disk.class.getClassLoader().getResourceAsStream(resource); + if (inputStream == null) { + throw new RuntimeException("Cannot get input stream for '" + resource + "'"); + } + ByteArrayOutputStream bos = new ByteArrayOutputStream(); + byte[] buffer = new byte[8192]; + int count; + do { + try { + count = inputStream.read(buffer); + } catch (IOException ex) { + throw new RuntimeException("Cannot read input stream for '" + resource + "'", ex); + } + + if (count > 0) { + bos.write(buffer, 0, count); + + } + } while (count > -1); + + try { + bos.close(); + } catch (IOException ex) { + throw new RuntimeException(ex); + } + + try { + FileUtility.writeBytes(imageFile, bos.toByteArray()); + } catch (CoreException ex) { + MarkerUtility.createMarker(files.mainSourceFile.iFile, 0, ex.getStatus().getSeverity(), "{0}", + ex.getStatus().getMessage()); + return false; + } + + } else { // no auto creation + + // ERROR: Disk image file '{0}' does not exist. Create a + // bootable disk image where the output file '{1}' can be + // stored. + MarkerUtility.createMarker(files.mainSourceFile.iFile, 0, IMarker.SEVERITY_ERROR, + Texts.MESSAGE_E132, imageFilePath, outputImageEntryName); + return false; + } + } + if (!imageFile.canWrite()) { + // ERROR: Disk image file '{0}' is not writeable. Make the disk + // image file writeable, so the output file '{1}' can be stored. + MarkerUtility.createMarker(files.mainSourceFile.iFile, 0, IMarker.SEVERITY_ERROR, Texts.MESSAGE_E133, + imageFilePath, outputImageEntryName); + return false; + } + } + Disk disk; + try { + disk = new Disk(imageFilePath); + } catch (IOException ex) { + // ERROR: Disk image file '{0}' cannot be opened for reading. System + // error: {1} + MarkerUtility.createMarker(files.mainSourceFile.iFile, 0, IMarker.SEVERITY_ERROR, Texts.MESSAGE_E134, + imageFilePath, ex.getMessage()); + return false; + } + + FormattedDisk[] formattedDisks = disk.getFormattedDisks(); + if (formattedDisks == null || formattedDisks.length == 0) { + // ERROR: Disk image file '{0}' does not contain a valid file + // system. Make sure the disk image is properly formatted, so + // the output file '{1}' can be stored. + MarkerUtility.createMarker(files.mainSourceFile.iFile, 0, IMarker.SEVERITY_ERROR, Texts.MESSAGE_E135, + imageFilePath, outputImageEntryName); + return false; + } + + byte[] outputFileContent; + try { + outputFileContent = FileUtility.readBytes(files.outputFile, 65536, true); + } catch (CoreException ex) { + // ERROR: Cannot read output file. + MarkerUtility.createMarker(files.mainSourceFile.iFile, 0, ex.getStatus().getSeverity(), "{0}", ex + .getStatus().getMessage()); + return false; + } + + FormattedDisk formattedDisk; + try { + formattedDisk = formattedDisks[0]; + // Create HELLO with dummy address. + createHello(formattedDisk, outputImageEntryName, outputImageEntryTitle, 1234); + + FileEntry entry = formattedDisk.getFile(outputImageEntryName); + if (entry == null) { + entry = formattedDisk.createFile(); + // TODO This is required due to a BUG in AppleCommander + entry.setFilename(outputImageEntryName); + } + entry.setFiletype("B"); + + String fileName = files.outputFileName.toLowerCase(); + int length = outputFileContent.length; + byte[] content = outputFileContent; + if (entry.needsAddress()) { + int address; + if (fileName.endsWith(".b") && length > 4) { + // // AppleDos 3.3 binary file: + // start-lo,start-hi,length-lo,length-hi,data + address = getWord(outputFileContent, 0); + length = length - 4; + content = getData(outputFileContent, 4); + } else if (fileName.endsWith(".prg") && length > 2) { + // C64 program file + // start-lo,start-hi,data + address = getWord(outputFileContent, 0); + length = length - 2; + content = getData(outputFileContent, 2); + } else if (fileName.endsWith(".xex") && length > 6 + && ((getWord(outputFileContent, 0) & 0xffff) == 0xffff)) { + // AtariDOS 2.5 binary file: + // $ff,$ff,start-lo,start-hi,end-lo,end-hi,data + address = getWord(outputFileContent, 2); + length = length - 6; + content = getData(outputFileContent, 6); + } else { + // ERROR: Output file {0} has unknown executable file + // extension or content. File extensions ".b", ".prg" and + // ".xex" are allowed. + MarkerUtility.createMarker(files.mainSourceFile.iFile, 0, IMarker.SEVERITY_ERROR, + Texts.MESSAGE_E138, files.outputFilePath); + return false; + } + // Method setAddress must be called after method setFileData! + entry.setFileData(content); + entry.setAddress(address); + // Update HELLO with acual start address. + createHello(formattedDisk, outputImageEntryName, outputImageEntryTitle, address); + } else { + entry.setFileData(outputFileContent); + + } + + } catch (DiskFullException ex) { + // ERROR: Disk image file '{0}' is full. System + // error: {1} + MarkerUtility.createMarker(files.mainSourceFile.iFile, 0, IMarker.SEVERITY_ERROR, Texts.MESSAGE_E136, + imageFilePath, ex.getMessage()); + return false; + + } + + try { + formattedDisk.save(); + + } catch (IOException ex) { + + // ERROR: Disk image file '{0}' cannot be opened for writing. System + // error: {1} + MarkerUtility.createMarker(files.mainSourceFile.iFile, 0, IMarker.SEVERITY_ERROR, Texts.MESSAGE_E137, + imageFilePath, ex.getMessage()); + return false; + } + + return true; + + } + + private static FileEntry createHello(FormattedDisk formattedDisk, String outputImageEntryName, + String outputImageEntryTitle, int runAddress) throws DiskFullException { + if (formattedDisk == null) { + throw new IllegalArgumentException("Parameter 'formattedDisk' must not be null."); + } + if (outputImageEntryName == null) { + throw new IllegalArgumentException("Parameter 'OUTPUT_IMAGE_ENTRY_NAME' must not be null."); + } + if (StringUtility.isEmpty(outputImageEntryName)) { + throw new IllegalArgumentException("Parameter 'OUTPUT_IMAGE_ENTRY_NAME' must not be empty."); + } + if (outputImageEntryTitle == null) { + throw new IllegalArgumentException("Parameter 'outputImageEntryTitle' must not be null."); + } + FileEntry entry; + + entry = formattedDisk.getFile(HELLO_ENTRY_NAME); + if (entry == null) { + entry = formattedDisk.createFile(); + entry.setFilename(HELLO_ENTRY_NAME); + } + entry.setFiletype("A"); + byte[] program; + try { + // See "Beneath Apple DOS.pdf", 4-11/12 for the binary and AppleSoft + // file format. + // See http://www.textfiles.com/apple/ANATOMY/cmd.brun.bload.txt for + // the bug in the BRUN routine. Instead of BRUN now BLOAD is sent to + // DOS via PRINT CHR$(4) and the program is started via CALL. + // 10 PRINT "LOADING " : PRINT CHR$(4);"BRUN WORLD" : CALL + // <address> + + ByteArrayOutputStream bos = new ByteArrayOutputStream(); + bos.write(new byte[] { 0x00, 0x08, 0x0A, 0x00 }); + bos.write(new byte[] { AppleSoftTokens.PRINT, '"' }); + bos.write(outputImageEntryTitle.getBytes("US-ASCII")); + bos.write(new byte[] { '"', ':' }); + bos.write(new byte[] { AppleSoftTokens.PRINT, AppleSoftTokens.CHRS, '(', '4', ')', ';', '"', 'B', 'L', 'O', + 'A', 'D', ' ' }); + bos.write(outputImageEntryName.getBytes("US-ASCII")); + bos.write(new byte[] { '"', ':', AppleSoftTokens.CALL }); + bos.write(Integer.toString(runAddress).getBytes("US-ASCII")); + bos.write(new byte[] { 0, 0, 0 }); + program = bos.toByteArray(); + program[0] = (byte) (program.length); + } catch (UnsupportedEncodingException ex) { + throw new RuntimeException(ex); + } catch (IOException ex) { + throw new RuntimeException(ex); + } + + // Byte is the line length in bytes + entry.setFileData(program); + return entry; + } + + private static int getWord(byte[] bytes, int index) { + if (bytes == null) { + throw new IllegalArgumentException("Parameter 'bytes' must not be null."); + } + + return (0xff & bytes[index]) + 256 * (0xff & bytes[index + 1]); + } + + private static byte[] getData(byte[] bytes, int index) { + if (bytes == null) { + throw new IllegalArgumentException("Parameter 'bytes' must not be null."); + } + int length = bytes.length - index; + byte[] result = new byte[length]; + System.arraycopy(bytes, index, result, 0, length); + return result; + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerBreakpoinDebugModelPresentation.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerBreakpoinDebugModelPresentation.java new file mode 100644 index 00000000..70a473d3 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerBreakpoinDebugModelPresentation.java @@ -0,0 +1,114 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import org.eclipse.debug.core.model.IValue; +import org.eclipse.debug.ui.IDebugModelPresentation; +import org.eclipse.debug.ui.IValueDetailListener; +import org.eclipse.jface.viewers.ILabelProviderListener; +import org.eclipse.swt.graphics.Image; +import org.eclipse.ui.IEditorInput; +import org.eclipse.ui.IEditorPart; +import org.eclipse.ui.IWorkbenchPage; +import org.eclipse.ui.IWorkbenchWindow; +import org.eclipse.ui.PartInitException; +import org.eclipse.ui.ide.IDE; + +import com.wudsn.ide.asm.AssemblerPlugin; + +/** + * Implementation class for extension + * "com.wudsn.ide.asm.editor.AssemblerBreakpoinDebugModelPresentation". This is + * the binding logic which enables navigation from transient and persistent + * break point markers to the corresponding editor. + * + * @author Peter Dell + */ +public class AssemblerBreakpoinDebugModelPresentation implements IDebugModelPresentation { + + @Override + public void dispose() { + } + + @Override + public boolean isLabelProperty(Object element, String property) { + return false; + } + + @Override + public void addListener(ILabelProviderListener listener) { + } + + @Override + public void removeListener(ILabelProviderListener listener) { + + } + + @Override + public IEditorInput getEditorInput(Object element) { + AssemblerBreakpoint assemblerBreakpoint = (AssemblerBreakpoint) element; + IEditorInput result = assemblerBreakpoint.getEditorInput(); + if (result == null) { + IWorkbenchWindow activeWindow = AssemblerPlugin.getInstance().getWorkbench().getActiveWorkbenchWindow(); + if (activeWindow == null) { + return null; + } + IWorkbenchPage activePage = activeWindow.getActivePage(); + if (activePage == null) { + return null; + } + IEditorPart part; + try { + part = IDE.openEditor(activePage, assemblerBreakpoint.getMarker(), false); + } catch (PartInitException ex) { + return null; + } + return part.getEditorInput(); + } + return result; + } + + @Override + public String getEditorId(IEditorInput input, Object element) { + AssemblerBreakpoint assemblerBreakpoint = (AssemblerBreakpoint) element; + return assemblerBreakpoint.getEditorId(); + } + + @Override + public void setAttribute(String attribute, Object value) { + + } + + @Override + public Image getImage(Object element) { + return null; + } + + @Override + public String getText(Object element) { + return null; + } + + @Override + public void computeDetail(IValue value, IValueDetailListener listener) { + + } + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerBreakpoint.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerBreakpoint.java new file mode 100644 index 00000000..72e00810 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerBreakpoint.java @@ -0,0 +1,172 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import org.eclipse.core.resources.IMarker; +import org.eclipse.core.resources.IResource; +import org.eclipse.core.resources.IWorkspaceRunnable; +import org.eclipse.core.runtime.CoreException; +import org.eclipse.core.runtime.IProgressMonitor; +import org.eclipse.debug.core.model.IBreakpoint; +import org.eclipse.debug.core.model.LineBreakpoint; +import org.eclipse.ui.IEditorInput; + +import com.wudsn.ide.asm.AssemblerPlugin; +import com.wudsn.ide.asm.Texts; +import com.wudsn.ide.base.common.NumberUtility; +import com.wudsn.ide.base.common.TextUtility; + +/** + * Implementation class for assembler breakpoints. See + * http://eclipse.org/articles/Article-Debugger/how-to.html for details. + * + * @author Peter Dell + * @since 1.6.1 + * + */ + +public final class AssemblerBreakpoint extends LineBreakpoint { + + /** + * Attributes stored with the marker. + */ + public static final String EDITOR_ID = "editorId"; + + /** + * Grouping ID for all breakpoint of this type. + */ + public static final String DEBUG_MODEL_ID = AssemblerPlugin.ID; + + /** + * Marker type as defined by the extension + * "org.eclipse.core.resources.markers" + */ + public static final String MARKER_TYPE = "org.eclipse.debug.core.lineBreakpointMarker"; + + private IEditorInput editorInput; + + /** + * Default constructor is required for the breakpoint manager to re-create + * persisted breakpoints. After instantiating a breakpoint, the + * <code>setMarker(...)</code> method is called to restore this breakpoint's + * attributes. + */ + public AssemblerBreakpoint() { + } + + @Override + @SuppressWarnings({ "rawtypes", "unchecked" }) + public Object getAdapter(Class adapter) { + if (adapter.isAssignableFrom(IMarker.class)) { + return getMarker(); + } + if (adapter.isAssignableFrom(IResource.class)) { + // IResource result = getMarker().getResource(); + // System.out.println(adapter.getName() + ":" + result); + // return result; + } + return super.getAdapter(adapter); + } + + /** + * Constructs a line breakpoint on the given resource at the given line + * number. The line number is 1-based (i.e. the first line of a file is line + * number 1). + * + * @param editorId + * The editor id, not <code>null</code>. + * + * @param editorInput + * The editor input, not <code>null</code>. + * @param resource + * The file on which to set the breakpoint, not <code>null</code> + * . + * @param lineNumber + * The line number of the breakpoint, a positive integer. + * @param description + * The description of the break point, may be empty not + * <code>null</code>. + * @throws CoreException + * if unable to create the breakpoint + */ + public AssemblerBreakpoint(final String editorId, IEditorInput editorInput, final IResource resource, + final int lineNumber, final String description) throws CoreException { + if (editorId == null) { + throw new IllegalArgumentException("Parameter 'editorId' must not be null."); + } + if (editorInput == null) { + throw new IllegalArgumentException("Parameter 'editorInput' must not be null."); + } + if (resource == null) { + throw new IllegalArgumentException("Parameter 'resource' must not be null."); + } + if (lineNumber < 1) { + throw new IllegalArgumentException("Parameter 'lineNumber' must be positive. Specified value is " + + lineNumber + "."); + } + if (description == null) { + throw new IllegalArgumentException("Parameter 'description' must not be null."); + } + this.editorInput = editorInput; + IWorkspaceRunnable runnable = new IWorkspaceRunnable() { + @Override + public void run(IProgressMonitor monitor) throws CoreException { + IMarker marker = resource.createMarker(MARKER_TYPE); + // This must be the first operation before setting marker + // attributes. + setMarker(marker); + + marker.setAttribute(EDITOR_ID, editorId); + marker.setAttribute(IBreakpoint.ENABLED, Boolean.TRUE); + marker.setAttribute(IMarker.LINE_NUMBER, lineNumber); + marker.setAttribute(IBreakpoint.ID, getModelIdentifier()); + marker.setAttribute( + IMarker.MESSAGE, + TextUtility.format(Texts.ASSEMBLER_BREAKPOINT_MARKER_MESSAGE, resource.getName(), + NumberUtility.getLongValueDecimalString(lineNumber), description)); + + } + }; + run(getMarkerRule(resource), runnable); + } + + /* + * (non-Javadoc) + * + * @see org.eclipse.debug.core.model.IBreakpoint#getModelIdentifier() + */ + @Override + public String getModelIdentifier() { + return DEBUG_MODEL_ID; + } + + final String getEditorId() { + return getMarker().getAttribute(EDITOR_ID, null); + } + + final void setEditorInput(IEditorInput editorInput) { + this.editorInput = editorInput; + } + + final IEditorInput getEditorInput() { + return editorInput; + } + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerBreakpointAdapterFactory.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerBreakpointAdapterFactory.java new file mode 100644 index 00000000..8a600f65 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerBreakpointAdapterFactory.java @@ -0,0 +1,88 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import java.util.Collections; +import java.util.HashSet; +import java.util.Set; + +import org.eclipse.debug.ui.actions.IToggleBreakpointsTarget; +import org.eclipse.debug.ui.actions.IToggleBreakpointsTargetFactory; +import org.eclipse.jface.viewers.ISelection; +import org.eclipse.ui.IWorkbenchPart; + +import com.wudsn.ide.asm.Texts; + +/** + * Factory for {@AssemblerBreakpointsTarget} + * instances. Used by extension + * "org.eclipse.debug.ui.toggleBreakpointsTargetFactories" + * + * @author Peter Dell + * @since 1.6.1 + */ +public final class AssemblerBreakpointAdapterFactory implements + IToggleBreakpointsTargetFactory { + + private String TARGET_ID = AssemblerBreakpointsTarget.class.getName(); + private Set<String> defaultSet; + + public AssemblerBreakpointAdapterFactory() { + defaultSet = new HashSet<String>(); + defaultSet.add(TARGET_ID); + } + + @Override + public Set<String> getToggleTargets(IWorkbenchPart part, + ISelection selection) { + if (part instanceof AssemblerEditor) { + return defaultSet; + } + return Collections.emptySet(); + } + + @Override + public String getDefaultToggleTarget(IWorkbenchPart part, + ISelection selection) { + if (part instanceof AssemblerEditor) { + return TARGET_ID; + } + return null; + } + + @Override + public IToggleBreakpointsTarget createToggleTarget(String targetID) { + if (TARGET_ID.equals(targetID)) { + return new AssemblerBreakpointsTarget(); + } + return null; + } + + @Override + public String getToggleTargetName(String targetID) { + return Texts.ASSEMBLER_BREAKPOINT_TOGGLE_TYPE_MENU_TEXT; + } + + @Override + public String getToggleTargetDescription(String targetID) { + return TARGET_ID; + } + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerBreakpointsTarget.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerBreakpointsTarget.java new file mode 100644 index 00000000..c27045ef --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerBreakpointsTarget.java @@ -0,0 +1,168 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ +package com.wudsn.ide.asm.editor; + +import org.eclipse.core.resources.IResource; +import org.eclipse.core.runtime.CoreException; +import org.eclipse.debug.core.DebugPlugin; +import org.eclipse.debug.core.IBreakpointManager; +import org.eclipse.debug.core.model.IBreakpoint; +import org.eclipse.debug.core.model.ILineBreakpoint; +import org.eclipse.debug.ui.actions.IToggleBreakpointsTarget; +import org.eclipse.jface.text.BadLocationException; +import org.eclipse.jface.text.IDocument; +import org.eclipse.jface.text.ITextSelection; +import org.eclipse.jface.viewers.ISelection; +import org.eclipse.ui.IEditorInput; +import org.eclipse.ui.IWorkbenchPart; +import org.eclipse.ui.texteditor.IDocumentProvider; + +import com.wudsn.ide.base.common.StringUtility; + +/** + * Target which creates {@link AssemblerBreakpoint} instances. Used by + * {@link AssemblerBreakpointAdapterFactory}. + */ +public final class AssemblerBreakpointsTarget implements IToggleBreakpointsTarget { + /* + * (non-Javadoc) + * + * @see + * org.eclipse.debug.ui.actions.IToggleBreakpointsTarget#toggleLineBreakpoints + * (org.eclipse.ui.IWorkbenchPart, org.eclipse.jface.viewers.ISelection) + */ + @Override + public void toggleLineBreakpoints(IWorkbenchPart part, ISelection selection) throws CoreException { + AssemblerEditor assemblerEditor = getEditor(part); + if (assemblerEditor != null) { + IBreakpointManager breakPointManager = DebugPlugin.getDefault().getBreakpointManager(); + String editorId=assemblerEditor.getClass().getName(); + IEditorInput editorInput = assemblerEditor.getEditorInput(); + IResource resource = (IResource) editorInput.getAdapter(IResource.class); + ITextSelection textSelection = (ITextSelection) selection; + int lineNumber = textSelection.getStartLine(); + IBreakpoint[] breakpoints = breakPointManager.getBreakpoints(AssemblerBreakpoint.DEBUG_MODEL_ID); + for (int i = 0; i < breakpoints.length; i++) { + IBreakpoint breakpoint = breakpoints[i]; + if (resource.equals(breakpoint.getMarker().getResource())) { + if (((ILineBreakpoint) breakpoint).getLineNumber() == (lineNumber + 1)) { + // Remove existing breakpoint + breakpoint.delete(); + return; + } + } + } + // Create line breakpoint (doc line numbers start at 0) + String description; + IDocumentProvider provider = assemblerEditor.getDocumentProvider(); + IDocument document = provider.getDocument(assemblerEditor.getEditorInput()); + try { + int startOffset = document.getLineOffset(lineNumber); + int lineLength = document.getLineLength(lineNumber); + description = document.get(startOffset, lineLength).trim(); + description = description.replace('\t', ' '); + } catch (BadLocationException ex) { + throw new RuntimeException(ex); + } + + // No break points on empty lines + if (StringUtility.isEmpty(description)) { + return; + } + AssemblerBreakpoint breakpoint = new AssemblerBreakpoint(editorId, editorInput, resource, lineNumber + 1, description); + breakPointManager.addBreakpoint(breakpoint); + } + } + + /* + * (non-Javadoc) + * + * @see org.eclipse.debug.ui.actions.IToggleBreakpointsTarget# + * canToggleLineBreakpoints(org.eclipse.ui.IWorkbenchPart, + * org.eclipse.jface.viewers.ISelection) + */ + @Override + public boolean canToggleLineBreakpoints(IWorkbenchPart part, ISelection selection) { + return getEditor(part) != null; + } + + /* + * (non-Javadoc) + * + * @see + * org.eclipse.debug.ui.actions.IToggleBreakpointsTarget#toggleMethodBreakpoints + * (org.eclipse.ui.IWorkbenchPart, org.eclipse.jface.viewers.ISelection) + */ + @Override + public void toggleMethodBreakpoints(IWorkbenchPart part, ISelection selection) throws CoreException { + } + + /* + * (non-Javadoc) + * + * @see org.eclipse.debug.ui.actions.IToggleBreakpointsTarget# + * canToggleMethodBreakpoints(org.eclipse.ui.IWorkbenchPart, + * org.eclipse.jface.viewers.ISelection) + */ + @Override + public boolean canToggleMethodBreakpoints(IWorkbenchPart part, ISelection selection) { + return false; + } + + /* + * (non-Javadoc) + * + * @see + * org.eclipse.debug.ui.actions.IToggleBreakpointsTarget#toggleWatchpoints + * (org.eclipse.ui.IWorkbenchPart, org.eclipse.jface.viewers.ISelection) + */ + @Override + public void toggleWatchpoints(IWorkbenchPart part, ISelection selection) throws CoreException { + } + + /* + * (non-Javadoc) + * + * @see + * org.eclipse.debug.ui.actions.IToggleBreakpointsTarget#canToggleWatchpoints + * (org.eclipse.ui.IWorkbenchPart, org.eclipse.jface.viewers.ISelection) + */ + @Override + public boolean canToggleWatchpoints(IWorkbenchPart part, ISelection selection) { + return false; + } + + /** + * Determines of the specified workbench part is an assembler editor with a + * valid resource. + * + * @param part + * The editor part or <code>null</code>. + * @return The assembler editor or <code>null</code>. + */ + private AssemblerEditor getEditor(IWorkbenchPart part) { + if (part instanceof AssemblerEditor) { + AssemblerEditor assemblerEditor = (AssemblerEditor) part; + if (assemblerEditor.getCurrentFile() != null) { + return assemblerEditor; + } + } + return null; + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerContentAssistProcessor.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerContentAssistProcessor.java new file mode 100644 index 00000000..3e76dbad --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerContentAssistProcessor.java @@ -0,0 +1,573 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import java.util.ArrayList; +import java.util.List; + +import org.eclipse.jface.text.BadLocationException; +import org.eclipse.jface.text.IDocument; +import org.eclipse.jface.text.ITextSelection; +import org.eclipse.jface.text.ITextViewer; +import org.eclipse.jface.text.Region; +import org.eclipse.jface.text.contentassist.ICompletionProposal; +import org.eclipse.jface.text.contentassist.IContentAssistProcessor; +import org.eclipse.jface.text.contentassist.IContextInformation; +import org.eclipse.jface.text.contentassist.IContextInformationValidator; +import org.eclipse.jface.viewers.DelegatingStyledCellLabelProvider.IStyledLabelProvider; +import org.eclipse.jface.viewers.StyledString; +import org.eclipse.jface.viewers.StyledString.Styler; +import org.eclipse.swt.graphics.Image; +import org.eclipse.swt.graphics.TextStyle; + +import com.wudsn.ide.asm.AssemblerPlugin; +import com.wudsn.ide.asm.compiler.CompilerFiles; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceFile; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParserLineCallback; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParserTreeObject; +import com.wudsn.ide.asm.compiler.syntax.CompilerSyntax; +import com.wudsn.ide.asm.compiler.syntax.Directive; +import com.wudsn.ide.asm.compiler.syntax.Instruction; +import com.wudsn.ide.asm.compiler.syntax.InstructionSet; +import com.wudsn.ide.asm.compiler.syntax.InstructionType; +import com.wudsn.ide.asm.compiler.syntax.Opcode; +import com.wudsn.ide.asm.preferences.AssemblerPreferences; +import com.wudsn.ide.base.common.StringUtility; + +/** + * Class for content assist. Creates the content assist list. + * + * @author Peter Dell + * @author Daniel Mitte + */ +final class AssemblerContentAssistProcessor implements IContentAssistProcessor { + + /** + * Empty styler + */ + private final static class InstructionStyler extends Styler { + + InstructionStyler() { + } + + @Override + public void applyStyles(TextStyle textStyle) { + + } + } + + /** + * Underline styler + */ + private final static class HighlightStyler extends Styler { + + HighlightStyler() { + } + + @Override + public void applyStyles(TextStyle textStyle) { + textStyle.underline = true; + + } + } + + /** + * Callback to find out if a given line already contains an instruction. + * + * @since 1.6.0 + */ + private static final class SourceParserCallback extends + CompilerSourceParserLineCallback { + private boolean instructionFound; + private int instructionEndOffset; + + /** + * Create a new callback. + * + * @param filePath + * The absolute path of the source file, not empty and not + * <code>null</code>. + * @param lineNumber + * The line number, a non-negative integer or <code>-1</code> + * to indicate that no line number is relevant. + */ + public SourceParserCallback(String filePath, int lineNumber) { + super(filePath, lineNumber); + } + + @Override + public void processLine(CompilerSourceParser compilerSourceParser, + CompilerSourceFile compilerSourceFile, int lineNumber, + int startOffset, int symbolOffset, boolean instructionFound, + int instructionOffset, String instruction, int operandOffset, + CompilerSourceParserTreeObject section) { + + this.instructionFound = instructionFound; + if (instructionFound) { + instructionEndOffset = instructionOffset + instruction.length(); + } else { + instructionEndOffset = -1; + } + } + + /** + * Determines if the specified line in the source file already contains + * an instruction. + * + * @return <code>true</code> if the specified line in the source file + * already contains an instruction, <code>false</code> + * otherwise. + */ + public boolean wasInstructionFound() { + return instructionFound; + } + + /** + * Gets the offset of the last character of the instruction if an + * instruction was found. + * + * @return The offset or -1 if no instruction was found. + */ + public int getInstructionEndOffset() { + return instructionEndOffset; + } + } + + private AssemblerEditor editor; + + private Image directiveImage; + private Image legalOpcodeImage; + private Image illegalOpcodeImage; + private Image pseudoOpcodeImage; + private Styler instructionStyler; + private Styler highlightStyler; + + /** + * Creates a new instance. + * + * Called by + * {@link AssemblerSourceViewerConfiguration#getContentAssistant(org.eclipse.jface.text.source.ISourceViewer)} + * . + * + * @param editor + * The assembler editor for which this instance is created, not + * <code>null</code>. + */ + AssemblerContentAssistProcessor(AssemblerEditor editor) { + if (editor == null) { + throw new IllegalArgumentException( + "Parameter 'editor' must not be null."); + } + + this.editor = editor; + + AssemblerPlugin plugin = editor.getPlugin(); + directiveImage = plugin + .getImage("instruction-type-directive-16x16.gif"); + legalOpcodeImage = plugin + .getImage("instruction-type-legal-opcode-16x16.gif"); + illegalOpcodeImage = plugin + .getImage("instruction-type-illegal-opcode-16x16.gif"); + pseudoOpcodeImage = plugin + .getImage("instruction-type-pseudo-opcode-16x16.gif"); + instructionStyler = new InstructionStyler(); + highlightStyler = new HighlightStyler(); + } + + /** + * {@inheritDoc} + */ + @Override + public ICompletionProposal[] computeCompletionProposals(ITextViewer viewer, + int offset) { + if (viewer == null) { + throw new IllegalArgumentException( + "Parameter 'viewer' must not be null."); + } + ITextSelection selection = (ITextSelection) viewer + .getSelectionProvider().getSelection(); + + int selectionOffset = offset; + + if (selection.getOffset() != offset) { + selectionOffset = selection.getOffset(); + } + + List<ICompletionProposal> proposalList = new ArrayList<ICompletionProposal>(); + + // Convert offset into line number. + int lineNumber; + int lineOffset; + try { + lineNumber = viewer.getDocument().getLineOfOffset(offset); + lineOffset = viewer.getDocument().getLineOffset(lineNumber); + } catch (BadLocationException ex) { + lineNumber = -1; + lineOffset = -1; + } + + // Parse the current assembler file and try to find the line in the + // correct source file. + CompilerFiles files = AssemblerEditorFilesLogic.createInstance(editor).createCompilerFiles(); + if (files==null){ + return null; + } + SourceParserCallback compilerSourceCallback = new SourceParserCallback( + files.sourceFile.filePath, lineNumber); + CompilerSourceParser compilerSourceParser = editor + .createCompilerSourceParser(); + CompilerSourceFile compilerSourceFile = compilerSourceParser + .createCompilerSourceFile(files.sourceFile.file, + viewer.getDocument()); + compilerSourceParser.parse(compilerSourceFile, compilerSourceCallback); + + // If there is no instruction in the line yet or the cursor is exactly + // at the last character of that instruction, propose one. + if (!compilerSourceCallback.wasInstructionFound() + || selectionOffset == lineOffset + + compilerSourceCallback.getInstructionEndOffset()) { + String prefix = getPrefix(viewer, + compilerSourceParser.getCompilerSyntax(), selectionOffset, + false); + Region region = new Region(selectionOffset - prefix.length(), + prefix.length() + selection.getLength()); + addInstructionProposals(region, prefix, proposalList); + } else { + // Otherwise propose to use an identifier as operand. + String prefix = getPrefix(viewer, + compilerSourceParser.getCompilerSyntax(), selectionOffset, + true); + Region region = new Region(selectionOffset - prefix.length(), + prefix.length() + selection.getLength()); + addIdentifierProposals(region, prefix, compilerSourceFile, + proposalList); + } + + // If there is no proposal entry, return null instead of an empty array. + int size = proposalList.size(); + if (proposalList.size() == 0) { + return null; + } + + return proposalList.toArray(new ICompletionProposal[size]); + } + + /** + * Gets the prefix of the document starting at a given offset to the start + * of the document until a space or control character is found. + * + * @param viewer + * The viewer, not <code>null</code>. + * @param compilerSyntax + * The compiler syntax, not <code>null</code>. + * @param offset + * The offset, a non-negative integer. + * @param onlyIdentifiers + * <code>true</code> if only identifier characters shall be + * considered as part of the prefix. + * + * @return The prefix, may be empty, not <code>null</code>. + */ + private String getPrefix(ITextViewer viewer, CompilerSyntax compilerSyntax, + int offset, boolean onlyIdentifiers) { + if (viewer == null) { + throw new IllegalArgumentException( + "Parameter 'viewer' must not be null."); + } + if (compilerSyntax == null) { + throw new IllegalArgumentException( + "Parameter 'compilerSyntax' must not be null."); + } + int i = offset; + IDocument document = viewer.getDocument(); + + int l = document.getLength(); + if (i > l) { + return ""; + } + + try { + while (i > 0) { + char ch = document.getChar(i - 1); + + if (onlyIdentifiers) { + if (!compilerSyntax.isIdentifierCharacter(ch)) { + break; + } + } else { + if (Character.isWhitespace(ch)) { + break; + } + } + + i--; + } + + return document.get(i, offset - i); + } catch (BadLocationException ex) { + throw new RuntimeException(ex); + } + } + + private void addInstructionProposals(Region region, String prefix, + List<ICompletionProposal> proposalList) { + if (region == null) { + throw new IllegalArgumentException( + "Parameter 'region' must not be null."); + } + if (prefix == null) { + throw new IllegalArgumentException( + "Parameter 'prefix' must not be null."); + } + if (proposalList == null) { + throw new IllegalArgumentException( + "Parameter 'proposalList' must not be null."); + } + AssemblerPreferences assemblerPreferences = editor.getPlugin() + .getPreferences(); + + int offset = region.getOffset(); + boolean lowerCase; + + // Prefix is empty or prefix does not end with a letter but for + // example "." + if (StringUtility.isEmpty(prefix) + || !Character.isLetter(prefix.charAt(prefix.length() - 1))) { + String defaultCase; + defaultCase = assemblerPreferences + .getEditorContentAssistProcessorDefaultCase(); + lowerCase = AssemblerContentAssistProcessorDefaultCase.LOWER_CASE + .equals(defaultCase); + } else { + char lastchar = prefix.charAt(prefix.length() - 1); + lowerCase = ((lastchar < 'a') || (lastchar > 'z')) ? false : true; + } + + CompilerSourceParser compilerSourceParser = editor + .createCompilerSourceParser(); + InstructionSet instructionSet = compilerSourceParser + .getInstructionSet(); + + boolean caseSenstive = instructionSet.areInstructionsCaseSensitive(); + if (!caseSenstive) { + prefix = prefix.toUpperCase(); + } + + List<Instruction> instructions = instructionSet.getInstructions(); + for (int i = 0; i < instructions.size(); i++) { + Instruction instruction = instructions.get(i); + + String name = null; + if (caseSenstive) { + if (instruction.getName().indexOf(prefix) == 0) { + name = instruction.getName(); + } + } else { + if (instruction.getUpperCaseName().indexOf(prefix) == 0) { + + name = lowerCase ? instruction.getLowerCaseName() + : instruction.getUpperCaseName(); + } + } + + if (name != null) { + Image image; + + if (instruction instanceof Directive) { + image = directiveImage; + } else { + Opcode opcode = (Opcode) instruction; + switch (opcode.getType()) { + case InstructionType.LEGAL_OPCODE: + image = legalOpcodeImage; + break; + case InstructionType.ILLEGAL_OPCODE: + image = illegalOpcodeImage; + + break; + case InstructionType.PSEUDO_OPCODE: + image = pseudoOpcodeImage; + break; + default: + throw new IllegalStateException("Unknown opcode type " + + opcode.getType() + "."); + } + } + + String separator = " - "; + String displayString = name + separator + + instruction.getTitle(); + StyledString styledDisplayString = new StyledString(); + styledDisplayString.append(name); + styledDisplayString.append(separator); + int start = styledDisplayString.length(); + styledDisplayString.append(instruction.getStyledTitle()); + styledDisplayString.setStyle(0, name.length(), + instructionStyler); + int[] offsets = instruction.getStyledTitleOffsets(); + + for (int j = 0; j < offsets.length; j++) { + styledDisplayString.setStyle(start + offsets[j], 1, + highlightStyler); + } + + // Adapt proposal. + String proposal = instruction.getProposal(); + proposal = lowerCase ? proposal.toLowerCase() : proposal; + int proposalIndex; + int newCursorOffset; + // Must be positive. + proposalIndex = proposal.indexOf('_'); + // Remove cursor positioning. + proposal = proposal.replace("_", ""); + // Apply leading tabulator. + proposal = proposal.replace("\n", "\n\t"); + newCursorOffset = offset + proposalIndex; + + proposalList.add(new AssemblerInstructionCompletionProposal( + proposal, offset, region.getLength(), newCursorOffset, + image, displayString, styledDisplayString, null)); + } + } + } + + // TODO Handle prefixes which contain "." or end with it. + // TODO Handle identifier case sensitivity correctly + private void addIdentifierProposals(Region region, String prefix, + CompilerSourceFile compilerSourceFile, + List<ICompletionProposal> proposalList) { + if (region == null) { + throw new IllegalArgumentException( + "Parameter 'region' must not be null."); + } + if (prefix == null) { + throw new IllegalArgumentException( + "Parameter 'prefix' must not be null."); + } + if (compilerSourceFile == null) { + throw new IllegalArgumentException( + "Parameter 'compilerSourceFile' must not be null."); + } + if (proposalList == null) { + throw new IllegalArgumentException( + "Parameter 'proposalList' must not be null."); + } + + CompilerSourceParserTreeObjectLabelProvider imageProvider = new CompilerSourceParserTreeObjectLabelProvider(); + IStyledLabelProvider styledStringProvider = imageProvider + .getStyledStringProvider(); + int regionOffset = region.getOffset(); + int regionLength = region.getLength(); + String lowerCasePrefix = prefix.toLowerCase(); + + // Find last separator as basis for the prefix. + char identifierSeparatorCharacter = editor.getCompilerDefinition() + .getSyntax().getIdentifierSeparatorCharacter(); + if (identifierSeparatorCharacter != CompilerSyntax.NO_CHARACTER) { + int index = lowerCasePrefix + .lastIndexOf(identifierSeparatorCharacter); + if (index >= 0) { + regionOffset += index + 1; + regionLength -= index + 1; + lowerCasePrefix = lowerCasePrefix.substring(index + 1); + } + } + List<CompilerSourceParserTreeObject> identifiers = compilerSourceFile + .getIdentifiers(); + String separator = " - "; + for (int i = 0; i < identifiers.size(); i++) { + CompilerSourceParserTreeObject element = identifiers.get(i); + String lowerCaseName = element.getName().toLowerCase(); + if (lowerCaseName.indexOf(lowerCasePrefix) == 0) { + String proposal = element.getName(); + Image image = imageProvider.getImage(element); + String displayName; + String description; + String displayString; + displayName = element.getDisplayName(); + description = element.getDescription(); + if (StringUtility.isSpecified(description)) { + displayString = displayName + separator + description; + } else { + displayString = displayName; + + } + StyledString styledDisplayString = styledStringProvider + .getStyledText(element); + + int newCursorOffset = regionOffset + proposal.length(); + + proposalList.add(new AssemblerInstructionCompletionProposal( + proposal, regionOffset, regionLength, newCursorOffset, + image, displayString, styledDisplayString, null)); + } + } + + } + + /** + * {@inheritDoc} + */ + @Override + public IContextInformation[] computeContextInformation(ITextViewer viewer, + int offset) { + return null; + } + + /** + * {@inheritDoc} + */ + @Override + public char[] getCompletionProposalAutoActivationCharacters() { + CompilerSyntax compilerSyntax = editor.getCompilerDefinition() + .getSyntax(); + char[] result = compilerSyntax + .getCompletionProposalAutoActivationCharacters(); + if (result.length == 0) { + result = null; + } + return result; + } + + /** + * {@inheritDoc} + */ + @Override + public char[] getContextInformationAutoActivationCharacters() { + return null; + } + + /** + * {@inheritDoc} + */ + @Override + public String getErrorMessage() { + return null; + } + + /** + * {@inheritDoc} + */ + @Override + public IContextInformationValidator getContextInformationValidator() { + return null; + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerContentAssistProcessorDefaultCase.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerContentAssistProcessorDefaultCase.java new file mode 100644 index 00000000..a8e3c450 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerContentAssistProcessorDefaultCase.java @@ -0,0 +1,39 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +/** + * Constant definition for completion proposals. + * + * @author Peter Dell + * + */ +public final class AssemblerContentAssistProcessorDefaultCase { + + /** + * Creation is private. + */ + private AssemblerContentAssistProcessorDefaultCase() { + + } + + public static final String UPPER_CASE = "UPPER_CASE"; + public static final String LOWER_CASE = "LOWER_CASE"; +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerContentOutlinePage.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerContentOutlinePage.java new file mode 100644 index 00000000..a8d9d14b --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerContentOutlinePage.java @@ -0,0 +1,590 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import java.util.ArrayList; +import java.util.Collections; +import java.util.List; + +import org.eclipse.core.resources.IFile; +import org.eclipse.core.runtime.CoreException; +import org.eclipse.core.runtime.QualifiedName; +import org.eclipse.jface.action.Action; +import org.eclipse.jface.action.IToolBarManager; +import org.eclipse.jface.resource.ImageDescriptor; +import org.eclipse.jface.text.Position; +import org.eclipse.jface.text.TextSelection; +import org.eclipse.jface.viewers.ContentViewer; +import org.eclipse.jface.viewers.IBaseLabelProvider; +import org.eclipse.jface.viewers.ILabelProvider; +import org.eclipse.jface.viewers.ISelection; +import org.eclipse.jface.viewers.IStructuredSelection; +import org.eclipse.jface.viewers.SelectionChangedEvent; +import org.eclipse.jface.viewers.TreePath; +import org.eclipse.jface.viewers.TreeSelection; +import org.eclipse.jface.viewers.TreeViewer; +import org.eclipse.jface.viewers.Viewer; +import org.eclipse.jface.viewers.ViewerComparator; +import org.eclipse.swt.custom.BusyIndicator; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Control; +import org.eclipse.ui.IEditorInput; +import org.eclipse.ui.plugin.AbstractUIPlugin; +import org.eclipse.ui.views.contentoutline.ContentOutlinePage; + +import com.wudsn.ide.asm.AssemblerPlugin; +import com.wudsn.ide.asm.Texts; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceFile; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParserTreeObject; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParserTreeObjectType; +import com.wudsn.ide.base.common.Profiler; +import com.wudsn.ide.base.common.RunnableWithLogging; + +/** + * Outline page for the assembler editor. + * + * @author Peter Dell. + * @author Andy Reek + */ +final class AssemblerContentOutlinePage extends ContentOutlinePage { + + /* + * Toggle action to toggle the sorting in the outline tree. The state of the + * action will be persisted along with the file in the editor. If there is + * no state store yet along with in the file, the default is taken from the + * last file which was opened. + */ + private static final class OutlineViewerSortAction extends Action { + private static final QualifiedName CHECKED = new QualifiedName( + "OutlineViewerSortAction", "Checked"); + + final AssemblerEditor editor; + final TreeViewer treeViewer; + + /** + * Creates a new sort action. + * + * @param editor + * The editor which holds the respective file, not + * <code>null</code>. + * @param treeViewer + * The tree viewer which displays the outline. + */ + public OutlineViewerSortAction(AssemblerEditor editor, + TreeViewer treeViewer) { + super("", AS_CHECK_BOX); + if (editor == null) { + throw new IllegalArgumentException( + "Parameter 'editor' must not be null."); + } + if (treeViewer == null) { + throw new IllegalArgumentException( + "Parameter 'treeViewer' must not be null."); + } + setToolTipText(Texts.ASSEMBLER_CONTENT_OUTLINE_SORT_BUTTON_TOOL_TIP); + ImageDescriptor imageDescriptor = AbstractUIPlugin + .imageDescriptorFromPlugin(AssemblerPlugin.ID, + "icons/outline-sort.gif"); + setImageDescriptor(imageDescriptor); + this.editor = editor; + this.treeViewer = treeViewer; + + String checkedProperty; + try { + IFile iFile = editor.getCurrentIFile(); + if (iFile != null) { + checkedProperty = iFile.getPersistentProperty(CHECKED); + } else { + checkedProperty = ""; + } + } catch (CoreException ignore) { + checkedProperty = null; + } + if (checkedProperty == null) { + checkedProperty = editor.getPlugin().getProperty(CHECKED); + } + + boolean checked = Boolean.parseBoolean(checkedProperty); + setChecked(checked); + + } + + @Override + public void run() { + + // Get current state and update the UI. + boolean checked = isChecked(); + setChecked(checked); + + // Store the property. + String checkedProperty = Boolean.toString(checked); + try { + IFile iFile = editor.getCurrentIFile(); + if (iFile != null) { + iFile.setPersistentProperty(CHECKED, checkedProperty); + } + } catch (CoreException ex) { + editor.getPlugin().logError("Cannot set property {0}", + new Object[] { CHECKED }, ex); + } + editor.getPlugin().setProperty(CHECKED, checkedProperty); + + // Refresh the tree viewer. + BusyIndicator.showWhile(treeViewer.getControl().getDisplay(), + new RunnableWithLogging() { + @Override + protected void runWithLogging() { + treeViewer.refresh(false); + } + }); + } + } + + private static final class OutlineViewerComparator extends ViewerComparator { + private final OutlineViewerSortAction sortAction; + + OutlineViewerComparator(OutlineViewerSortAction sortAction) { + if (sortAction == null) { + throw new IllegalArgumentException( + "Parameter 'sortAction' must not be null."); + } + this.sortAction = sortAction; + } + + @Override + public int category(Object element) { + int result; + CompilerSourceParserTreeObject object = (CompilerSourceParserTreeObject) element; + + // Treat equate definition and label definition as equal. + result = object.getType(); + + switch (object.getType()) { + case CompilerSourceParserTreeObjectType.DEFAULT: + case CompilerSourceParserTreeObjectType.DEFINITION_SECTION: + case CompilerSourceParserTreeObjectType.IMPLEMENTATION_SECTION: + break; + + case CompilerSourceParserTreeObjectType.EQUATE_DEFINITION: + result = CompilerSourceParserTreeObjectType.EQUATE_DEFINITION; + break; + case CompilerSourceParserTreeObjectType.LABEL_DEFINITION: + result = CompilerSourceParserTreeObjectType.EQUATE_DEFINITION; + break; + + case CompilerSourceParserTreeObjectType.ENUM_DEFINITION_SECTION: + case CompilerSourceParserTreeObjectType.STRUCTURE_DEFINITION_SECTION: + case CompilerSourceParserTreeObjectType.LOCAL_SECTION: + case CompilerSourceParserTreeObjectType.MACRO_DEFINITION_SECTION: + case CompilerSourceParserTreeObjectType.PAGES_SECTION: + case CompilerSourceParserTreeObjectType.PROCEDURE_DEFINITION_SECTION: + case CompilerSourceParserTreeObjectType.REPEAT_SECTION: + break; + + case CompilerSourceParserTreeObjectType.SOURCE_INCLUDE: + case CompilerSourceParserTreeObjectType.BINARY_INCLUDE: + break; + + default: + throw new RuntimeException("Element '" + object.getName() + + "' has unknown type " + object.getType() + "."); + + } + return result; + } + + @Override + public int compare(Viewer viewer, Object e1, Object e2) { + + if (!sortAction.isChecked()) { + return 0; + } + + int cat1 = category(e1); + int cat2 = category(e2); + + // Never sort definition or implementation sections. + if (cat1 == CompilerSourceParserTreeObjectType.DEFINITION_SECTION + && cat2 == CompilerSourceParserTreeObjectType.DEFINITION_SECTION) { + return 0; + } + + if (cat1 != cat2) { + return cat1 - cat2; + } + + String name1; + String name2; + + if (viewer == null || !(viewer instanceof ContentViewer)) { + name1 = e1.toString(); + name2 = e2.toString(); + } else { + IBaseLabelProvider prov = ((ContentViewer) viewer) + .getLabelProvider(); + if (prov instanceof ILabelProvider) { + ILabelProvider lprov = (ILabelProvider) prov; + name1 = lprov.getText(e1); + name2 = lprov.getText(e2); + } else { + name1 = e1.toString(); + name2 = e2.toString(); + } + } + if (name1 == null) { + name1 = "";//$NON-NLS-1$ + } + if (name2 == null) { + name2 = "";//$NON-NLS-1$ + } + + // Use direct comparison as identified are ASCII only. + return name1.compareTo(name2); + + } + } + + /** + * Editor updater for selection changes in the content outline page. + */ + private final static class EditorUpdater extends RunnableWithLogging { + private final Profiler profiler; + + private final AssemblerEditor editor; + private final AssemblerContentOutlinePage outlinePage; + private final TreeViewer viewer; + private final AssemblerContentOutlineTreeContentProvider contentProvider; + + EditorUpdater(AssemblerEditor editor, + AssemblerContentOutlinePage outlinePage, TreeViewer viewer) { + if (editor == null) { + throw new IllegalArgumentException( + "Parameter 'editor' must not be null."); + } + if (outlinePage == null) { + throw new IllegalArgumentException( + "Parameter 'outlinePage' must not be null."); + } + if (viewer == null) { + throw new IllegalArgumentException( + "Parameter 'viewer' must not be null."); + } + this.editor = editor; + this.outlinePage = outlinePage; + this.viewer = viewer; + this.contentProvider = (AssemblerContentOutlineTreeContentProvider) viewer + .getContentProvider(); + profiler = new Profiler(this); + + } + + /** + * Triggers a new + * {@link AssemblerContentOutlineTreeContentProvider#parse} run and + * updates the display. + */ + @Override + protected void runWithLogging() { + synchronized (outlinePage) { + try { + outlinePage.inputUpdateCounter++; + runSynchronized(); + } finally { + outlinePage.inputUpdateCounter--; + } + } + } + + private void runSynchronized() { + // Stop drawing the control. + Control control = viewer.getControl(); + + // Check if this call is caused by closing the editor. + if (control.isDisposed()) { + return; + } + + profiler.begin("runSynchronized"); + + profiler.begin("updateOutline"); + control.setRedraw(false); + + // Remember the currently selected tree object in the content + // outline tree viewer. + ISelection selection = viewer.getSelection(); + Object[] expandedElements = viewer.getExpandedElements(); + + // Trigger the the new parse run. + viewer.setInput(outlinePage.input); + // viewer.refresh(); Not required? + + profiler.begin("expandElements"); + if (expandedElements.length > 0) { + viewer.setExpandedElements(expandedElements); + } else { + viewer.expandToLevel(2); + } + profiler.end("expandElements"); + + restoreSelection(selection); + // Now that all changes are done, draw the control again. + control.setRedraw(true); + profiler.end("updateOutline"); + + // // Reselect the previous text selection in the editor. + // editor.getSelectionProvider().setSelection(textSelection); + + CompilerSourceFile compilerSourceFile; + compilerSourceFile = contentProvider.getCompilerSourceFile(); + + // Update the identifiers to be highlighted + profiler.begin("updateIdentifiers"); + editor.updateIdentifiers(compilerSourceFile); + profiler.end("updateIdentifiers"); + + // Update the folding structure. + profiler.begin("updateFoldingStructure"); + List<Position> foldingPositions; + if (compilerSourceFile != null) { + foldingPositions = compilerSourceFile.getFoldingPositions(); + } else { + foldingPositions = Collections.emptyList(); + } + editor.updateFoldingStructure(foldingPositions); + + profiler.end("updateFoldingStructure"); + + profiler.end("runSynchronized"); + + } + + private void restoreSelection(ISelection selection) { + if (selection instanceof TreeSelection) { + TreeSelection treeSelection = (TreeSelection) selection; + TreePath[] selectedTreePaths = treeSelection.getPaths(); + List<TreePath> reselectedTreePaths = new ArrayList<TreePath>( + selectedTreePaths.length); + for (int i = 0; i < selectedTreePaths.length; i++) { + TreePath treePath = selectedTreePaths[i]; + List<CompilerSourceParserTreeObject> treeObjects = contentProvider + .getCompilerSourceFile().getSections(); + + List<Object> segments = new ArrayList<Object>( + treePath.getSegmentCount()); + + for (int j = 0; j < treePath.getSegmentCount(); j++) { + CompilerSourceParserTreeObject oldTreeObject; + CompilerSourceParserTreeObject newTreeObject; + oldTreeObject = (CompilerSourceParserTreeObject) treePath + .getSegment(j); + newTreeObject = null; + + for (int k = 0; newTreeObject == null + && k < treeObjects.size(); k++) { + if (treeObjects.get(k).getTreePath() + .equals(oldTreeObject.getTreePath())) { + newTreeObject = treeObjects.get(k); + segments.add(newTreeObject); + treeObjects = newTreeObject.getChildren(); + } + } + + } + if (!segments.isEmpty()) { + reselectedTreePaths + .add(new TreePath(segments.toArray())); + } + + } + + TreePath[] reselectedTreePathsArray = new TreePath[reselectedTreePaths + .size()]; + reselectedTreePaths.toArray(reselectedTreePathsArray); + selection = new TreeSelection(reselectedTreePathsArray); + + // Reselect the previously selected tree object in the + // content outline tree viewer. + viewer.setSelection(selection); + } + } + } + + /** + * The owning editor. + */ + final AssemblerEditor editor; + + /** + * The visual components. + */ + private OutlineViewerSortAction treeViewerSortAction; + private OutlineViewerComparator treeViewerComparator; + + /** + * The current input. + */ + IEditorInput input; + int inputUpdateCounter; + + /** + * Creates a new instance. + * + * @param editor + * The assembler editor, not <code>null</code>. + */ + AssemblerContentOutlinePage(AssemblerEditor editor) { + if (editor == null) { + throw new IllegalArgumentException( + "Parameter 'editor' must not be null."); + } + this.editor = editor; + } + + /** + * Sets the input for the outline page. + * + * @param input + * The new input, not <code>null</code>. + */ + final void setInput(IEditorInput input) { + if (input == null) { + throw new IllegalArgumentException( + "Parameter 'input' must not be null."); + } + this.input = input; + + runEditorUpdater(); + } + + private void runEditorUpdater() { + final TreeViewer viewer = getTreeViewer(); + + if ((viewer != null) && (viewer.getContentProvider() != null)) { + editor.getSite().getShell().getDisplay() + .asyncExec(new EditorUpdater(editor, this, viewer)); + } + } + + /** + * Create the control and configures it. See code of + * org.eclipse.jdt.internal.ui.text.JavaOutlineInformationControl for + * similar use case. + * + * @param parent + * ´The parent, not <code>null</code>. + */ + @Override + public void createControl(Composite parent) { + super.createControl(parent); + + TreeViewer treeViewer = getTreeViewer(); + + // Configure the toolbar. + treeViewerSortAction = new OutlineViewerSortAction(editor, treeViewer); + treeViewerComparator = new OutlineViewerComparator(treeViewerSortAction); + IToolBarManager toolBarManager = getSite().getActionBars() + .getToolBarManager(); + + // Configure the content. + treeViewer + .setContentProvider(new AssemblerContentOutlineTreeContentProvider( + this)); + treeViewer + .setLabelProvider(new CompilerSourceParserTreeObjectLabelProvider()); + treeViewer.setComparator(treeViewerComparator); + treeViewer.addSelectionChangedListener(this); + + toolBarManager.add(treeViewerSortAction); + toolBarManager.update(true); + + if (input != null) { + runEditorUpdater(); + } + + } + + @Override + public void selectionChanged(SelectionChangedEvent event) { + super.selectionChanged(event); + + synchronized (this) { + if (inputUpdateCounter > 0) { + return; + } + } + + ISelection selection = event.getSelection(); + + if (selection.isEmpty()) { + editor.resetHighlightRange(); + } else { + if (selection instanceof IStructuredSelection) { + Object object = ((IStructuredSelection) selection) + .getFirstElement(); + + if (object instanceof CompilerSourceParserTreeObject) { + + CompilerSourceParserTreeObject treeObject; + AssemblerContentOutlineTreeContentProvider contentProvider; + contentProvider = (AssemblerContentOutlineTreeContentProvider) getTreeViewer() + .getContentProvider(); + treeObject = (CompilerSourceParserTreeObject) object; + + // If this is the tree object from another (source + // include) file, step off the tree to find the source + // include statement. + while (treeObject != null + && treeObject.getCompilerSourceFile() != contentProvider + .getCompilerSourceFile()) { + treeObject = treeObject.getParent(); + } + if (treeObject != null) { + try { + editor.setHighlightRange( + treeObject.getStartOffset(), 0, true); + editor.getSelectionProvider().setSelection( + new TextSelection(treeObject + .getStartOffset(), 1)); + } catch (IllegalArgumentException x) { + editor.resetHighlightRange(); + } + } + + } + } + } + } + + /** + * Gets the compiler source file of the last parse process. + * + * @return The compiler source file of the last parse process or + * <code>null</code>. + */ + final CompilerSourceFile getCompilerSourceFile() { + AssemblerContentOutlineTreeContentProvider contentProvider; + contentProvider = (AssemblerContentOutlineTreeContentProvider) getTreeViewer() + .getContentProvider(); + CompilerSourceFile compilerSourceFile = contentProvider + .getCompilerSourceFile(); + return compilerSourceFile; + } + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerContentOutlineTreeContentProvider.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerContentOutlineTreeContentProvider.java new file mode 100644 index 00000000..a1cdb2d5 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerContentOutlineTreeContentProvider.java @@ -0,0 +1,182 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import java.util.List; + +import org.eclipse.jface.text.IDocument; +import org.eclipse.jface.viewers.ITreeContentProvider; +import org.eclipse.jface.viewers.Viewer; +import org.eclipse.ui.IEditorInput; + +import com.wudsn.ide.asm.compiler.parser.CompilerSourceFile; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParserTreeObject; +import com.wudsn.ide.base.common.Profiler; + +/** + * Tree content provider to {@link AssemblerContentOutlinePage}. + * + * @author Peter Dell + * @author Andy Reek + */ +final class AssemblerContentOutlineTreeContentProvider implements + ITreeContentProvider { + + /** + * The surrounding content outline page. + */ + private final AssemblerContentOutlinePage assemblerContentOutlinePage; + + /** + * The last editor input which was parsed. + */ + private IEditorInput input; + + /** + * The result of the last parse process. + */ + private CompilerSourceFile compilerSourceFile; + + /** + * Called by + * {@link AssemblerContentOutlinePage#createControl(org.eclipse.swt.widgets.Composite)} + * . + * + * @param assemblerContentOutlinePage + * The outline page, not <code>null</code>. + */ + AssemblerContentOutlineTreeContentProvider( + AssemblerContentOutlinePage assemblerContentOutlinePage) { + if (assemblerContentOutlinePage == null) { + throw new IllegalArgumentException( + "Parameter 'assemblerContentOutlinePage' must not be null."); + } + this.assemblerContentOutlinePage = assemblerContentOutlinePage; + } + + /** + * {@inheritDoc} + */ + @Override + public Object getParent(Object element) { + if (element instanceof CompilerSourceParserTreeObject) { + return ((CompilerSourceParserTreeObject) element).getParent(); + } + + return null; + } + + /** + * {@inheritDoc} + */ + @Override + public boolean hasChildren(Object element) { + if (element instanceof CompilerSourceParserTreeObject) { + return (((CompilerSourceParserTreeObject) element).hasChildren()); + } + + return false; + } + + /** + * {@inheritDoc} + */ + @Override + public Object[] getChildren(Object parentElement) { + if (parentElement instanceof CompilerSourceParserTreeObject) { + return ((CompilerSourceParserTreeObject) parentElement) + .getChildrenAsArray(); + } + return null; + } + + /** + * {@inheritDoc} + */ + @Override + public Object[] getElements(Object inputElement) { + Object[] result; + if (inputElement == input && compilerSourceFile != null) { + List<CompilerSourceParserTreeObject> sections; + sections = compilerSourceFile.getSections(); + result = sections.toArray(new Object[sections.size()]); + } else { + result = new Object[0]; + } + + return result; + } + + /** + * {@inheritDoc} + */ + @Override + public void dispose() { + input = null; + } + + /** + * Gets the compiler source file of the last parse process. + * + * @return The compiler source file of the last parse process or + * <code>null</code>. + */ + CompilerSourceFile getCompilerSourceFile() { + return compilerSourceFile; + } + + /** + * {@inheritDoc} + */ + @Override + public void inputChanged(Viewer viewer, Object oldInput, Object newInput) { + if (oldInput instanceof IEditorInput) { + input = null; + } + + if (newInput instanceof IEditorInput) { + input = (IEditorInput) newInput; + parse(); + } + } + + /** + * Parses the new input and builds up the parse tree. + */ + private void parse() { + + AssemblerEditor editor = this.assemblerContentOutlinePage.editor; + IDocument document = editor.getDocumentProvider().getDocument( + editor.getEditorInput()); + + if (document != null) { + CompilerSourceParser parser = editor.createCompilerSourceParser(); + compilerSourceFile = parser.createCompilerSourceFile( + editor.getCurrentFile(), document); + Profiler profiler = new Profiler(parser); + profiler.begin("parse", editor.getTitle()); + parser.parse(compilerSourceFile, null); + profiler.end("parse"); + } + + } + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditor.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditor.java new file mode 100644 index 00000000..1ce697b7 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditor.java @@ -0,0 +1,558 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import java.io.File; +import java.util.ArrayList; +import java.util.Collections; +import java.util.HashMap; +import java.util.Iterator; +import java.util.List; +import java.util.Locale; +import java.util.ResourceBundle; + +import org.eclipse.core.resources.IFile; +import org.eclipse.core.runtime.CoreException; +import org.eclipse.debug.ui.actions.ToggleBreakpointAction; +import org.eclipse.jface.action.IAction; +import org.eclipse.jface.text.BadLocationException; +import org.eclipse.jface.text.IDocument; +import org.eclipse.jface.text.Position; +import org.eclipse.jface.text.source.Annotation; +import org.eclipse.jface.text.source.ISourceViewer; +import org.eclipse.jface.text.source.IVerticalRuler; +import org.eclipse.jface.text.source.SourceViewer; +import org.eclipse.jface.text.source.projection.ProjectionAnnotation; +import org.eclipse.jface.text.source.projection.ProjectionAnnotationModel; +import org.eclipse.jface.text.source.projection.ProjectionSupport; +import org.eclipse.jface.text.source.projection.ProjectionViewer; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.ui.IEditorInput; +import org.eclipse.ui.editors.text.TextEditor; +import org.eclipse.ui.ide.IDE; +import org.eclipse.ui.part.FileEditorInput; +import org.eclipse.ui.texteditor.IDocumentProvider; +import org.eclipse.ui.texteditor.ITextEditorActionDefinitionIds; +import org.eclipse.ui.texteditor.TextOperationAction; +import org.eclipse.ui.views.contentoutline.IContentOutlinePage; + +import com.wudsn.ide.asm.AssemblerPlugin; +import com.wudsn.ide.asm.AssemblerProperties; +import com.wudsn.ide.asm.AssemblerProperties.InvalidAssemblerPropertyException; +import com.wudsn.ide.asm.CPU; +import com.wudsn.ide.asm.Hardware; +import com.wudsn.ide.asm.compiler.Compiler; +import com.wudsn.ide.asm.compiler.CompilerDefinition; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceFile; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParserTreeObject; +import com.wudsn.ide.asm.compiler.parser.CompilerSourcePartitionScanner; +import com.wudsn.ide.asm.preferences.CompilerPreferences; +import com.wudsn.ide.base.common.Profiler; + +/** + * The assembler editor. + * + * @author Peter Dell + * @author Andy Reek + */ +public abstract class AssemblerEditor extends TextEditor { + + private AssemblerPlugin plugin; + private AssemblerEditorFilesLogic filesLogic; + + private Compiler compiler; + + private AssemblerContentOutlinePage contentOutlinePage; + private ProjectionAnnotationModel annotationModel; + + private Hardware hardware; + + /** + * Creates a new instance. Constructor parameters are not useful, because + * the super constructor inverts the flow of control, so + * {@link #initializeEditor} is called before the code in this constructor + * is executed. + */ + protected AssemblerEditor() { + filesLogic = AssemblerEditorFilesLogic.createInstance(this); + } + + /** + * Gets the files logic associated with this editor. + * + * @return The files logic, not <code>null</code>. + */ + public AssemblerEditorFilesLogic getFilesLogic() { + return filesLogic; + } + + /** + * Gets the hardware for this editor. + * + * @return The hardware for this editor, not <code>null</code>. + * + * @since 1.6.1 + */ + protected final Hardware getHardware() { + if (hardware != null) { + return hardware; + } + return compiler.getDefinition().getDefaultHardware(); + } + + /** + * Gets the compiler id for this editor. + * + * @return The compiler id for this editor, not empty and not + * <code>null</code>. + */ + protected abstract String getCompilerId(); + + @Override + protected final void initializeEditor() { + super.initializeEditor(); + + plugin = AssemblerPlugin.getInstance(); + compiler = plugin.getCompilerRegistry().getCompiler(getCompilerId()); + + setSourceViewerConfiguration(new AssemblerSourceViewerConfiguration(this, getPreferenceStore())); + + } + + /** + * Gets the plugin this compiler instance belongs to. + * + * @return The plugin this compiler instance belongs to, not + * <code>null</code>. + */ + public final AssemblerPlugin getPlugin() { + if (plugin == null) { + throw new IllegalStateException("Field 'plugin' must not be null."); + } + return plugin; + } + + /** + * Gets the compiler preferences. + * + * @return The compiler preferences, not <code>null</code>. + */ + public final CompilerPreferences getCompilerPreferences() { + return plugin.getPreferences().getCompilerPreferences(getCompilerId(), getHardware()); + } + + /** + * Gets the compiler for this editor. + * + * @return The compiler for this editor, not <code>null</code>. + */ + public final Compiler getCompiler() { + if (compiler == null) { + throw new IllegalStateException("Field 'compiler' must not be null."); + } + return compiler; + } + + /** + * Gets the compiler definition for this editor. + * + * @return The compiler definition for this editor, not <code>null</code>. + * + * @sine 1.6.1 + */ + public final CompilerDefinition getCompilerDefinition() { + if (compiler == null) { + throw new IllegalStateException("Field 'compiler' must not be null."); + } + return compiler.getDefinition(); + } + + /** + * Gets the compiler source parser for this editor and the currently + * selected instruction set. + * + * @return The compiler source parser for this editor, not <code>null</code> + * . + */ + public final CompilerSourceParser createCompilerSourceParser() { + CPU cpu; + CompilerSourceParser result; + if (compiler == null) { + throw new IllegalStateException("Field 'compiler' must not be null."); + } + cpu = getCompilerPreferences().getCPU(); + result = compiler.createSourceParser(); + result.init(compiler.getDefinition().getSyntax().getInstructionSet(cpu)); + return result; + } + + /** + * This method is called whenever the input changes, i.e. after loading and + * after saving as new file. + * + * @param input + * The new input, may be <code>null</code> + */ + @Override + protected final void doSetInput(IEditorInput input) throws CoreException { + super.doSetInput(input); + + hardware = null; + if (input != null) { + IDocument document = getDocumentProvider().getDocument(getEditorInput()); + + CompilerSourcePartitionScanner partitionScanner = new CompilerSourcePartitionScanner(compiler + .getDefinition().getSyntax()); + partitionScanner.createDocumentPartitioner(document); + + AssemblerProperties properties = CompilerSourceParser.getDocumentProperties(document); + + IFile iFile = getCurrentIFile(); + if (iFile != null) { + try { + hardware = filesLogic.getHardware(iFile, properties); + } catch (InvalidAssemblerPropertyException ex) { + // Do not use MarkerUtility.gotoMarker to make sure this + // editor instance is used. + IDE.gotoMarker(this, ex.marker); + hardware = null; + } + } + } + + } + + @Override + protected final void createActions() { + super.createActions(); + + ResourceBundle bundle = ResourceBundle.getBundle("com.wudsn.ide.asm.Actions", Locale.getDefault(), + AssemblerEditor.class.getClassLoader()); + + String actionDefintionId; + String actionId; + actionDefintionId = ITextEditorActionDefinitionIds.CONTENT_ASSIST_PROPOSALS; + actionId = "com.wudsn.ide.asm.editor.ContentAssistProposal"; + IAction action = new TextOperationAction(bundle, actionId + ".", this, ISourceViewer.CONTENTASSIST_PROPOSALS); + action.setActionDefinitionId(actionDefintionId); + setAction(actionId, action); + markAsStateDependentAction(actionId, true); + + SourceViewer sourceViewer = (SourceViewer) getSourceViewer(); + actionDefintionId = "com.wudsn.ide.asm.editor.AssemblerEditorToggleCommentCommand"; + actionId = actionDefintionId; + action = new AssemblerEditorToggleCommentAction(bundle, actionId + ".", this, sourceViewer); + action.setActionDefinitionId(actionId); + setAction(actionId, action); + markAsStateDependentAction(actionId, true); + + // Register rule double click. + ToggleBreakpointAction toggleBreakpointAction; + actionDefintionId = "org.eclipse.debug.ui.commands.ToggleBreakpoint"; + actionId = "RulerDoubleClick"; + action.setActionDefinitionId(actionId); + toggleBreakpointAction = new ToggleBreakpointAction(this, getDocumentProvider().getDocument(getEditorInput()), + getVerticalRuler()); + toggleBreakpointAction.setId(actionId); + setAction(actionId, toggleBreakpointAction); + markAsStateDependentAction(actionId, true); + toggleBreakpointAction.update(); + } + + final ISourceViewer getSourceViewerInternal() { + return getSourceViewer(); + } + + /** + * Refreshes the editor after changes to the text attributes or the text + * content. + * + * Called by {@link #updateIdentifiers(CompilerSourceFile)} and + * {@link AssemblerSourceViewerConfiguration#preferencesChanged(com.wudsn.ide.asm.preferences.AssemblerPreferences, java.util.Set)} + * . + */ + final void refreshSourceViewer() { + ISourceViewer isv = getSourceViewer(); + if (isv instanceof SourceViewer) { + ((SourceViewer) getSourceViewer()).invalidateTextPresentation(); + } + } + + @Override + public final void dispose() { + AssemblerSourceViewerConfiguration asvc; + asvc = (AssemblerSourceViewerConfiguration) getSourceViewerConfiguration(); + asvc.dispose(); + super.dispose(); + } + + @Override + public final Object getAdapter(@SuppressWarnings("rawtypes") Class adapter) { + if (IContentOutlinePage.class.equals(adapter)) { + if (contentOutlinePage == null) { + contentOutlinePage = new AssemblerContentOutlinePage(this); + // This causes double parsing upon starting with a new file + // currently. + updateContentOutlinePage(); + } + return contentOutlinePage; + } + return super.getAdapter(adapter); + } + + /** + * Updates the content in view of the outline page. Called by + * {@link AssemblerReconcilingStategy#parse}. + */ + final void updateContentOutlinePage() { + if (contentOutlinePage != null) { + IEditorInput input = getEditorInput(); + + if (input != null) { + contentOutlinePage.setInput(input); + } + } + } + + /** + * Gets the compiler source file of the last parse process. + * + * @return The compiler source file of the last parse process or + * <code>null</code>. + */ + final CompilerSourceFile getCompilerSourceFile() { + CompilerSourceFile result; + if (contentOutlinePage != null) { + result = contentOutlinePage.getCompilerSourceFile(); + } else { + result = null; + } + return result; + } + + @Override + public final void createPartControl(Composite parent) { + if (parent == null) { + throw new IllegalArgumentException("Parameter 'parent' must not be null."); + } + super.createPartControl(parent); + ProjectionViewer viewer = (ProjectionViewer) getSourceViewer(); + + ProjectionSupport projectionSupport = new ProjectionSupport(viewer, getAnnotationAccess(), getSharedColors()); + projectionSupport.install(); + + // turn projection mode on + viewer.doOperation(ProjectionViewer.TOGGLE); + + annotationModel = viewer.getProjectionAnnotationModel(); + + } + + @Override + protected final ISourceViewer createSourceViewer(Composite parent, IVerticalRuler ruler, int styles) { + if (parent == null) { + throw new IllegalArgumentException("Parameter 'parent' must not be null."); + } + if (ruler == null) { + throw new IllegalArgumentException("Parameter 'ruler' must not be null."); + } + ISourceViewer viewer = new ProjectionViewer(parent, ruler, getOverviewRuler(), isOverviewRulerVisible(), styles); + + // Ensure decoration support has been created and configured. + getSourceViewerDecorationSupport(viewer); + + // The first single line comment delimiter is used as the default. + List<String> singleLineCommentDelimiters = compiler.getDefinition().getSyntax() + .getSingleLineCommentDelimiters(); + String[] array = singleLineCommentDelimiters.toArray(new String[singleLineCommentDelimiters.size()]); + viewer.setDefaultPrefixes(array, IDocument.DEFAULT_CONTENT_TYPE); + viewer.setDefaultPrefixes(array, CompilerSourcePartitionScanner.PARTITION_COMMENT_SINGLE); + + return viewer; + } + + /** + * Update the identifiers to be highlighted + * + * @param compilerSourceFile + * The compiler source file or <code>null</code>. + */ + final void updateIdentifiers(CompilerSourceFile compilerSourceFile) { + Profiler profiler = new Profiler(this.getClass()); + + AssemblerSourceViewerConfiguration asvc; + AssemblerSourceScanner ais; + asvc = (AssemblerSourceViewerConfiguration) getSourceViewerConfiguration(); + ais = asvc.getAssemblerInstructionScanner(); + + List<CompilerSourceParserTreeObject> newIdentifiers; + if (compilerSourceFile == null) { + newIdentifiers = Collections.emptyList(); + } else { + newIdentifiers = compilerSourceFile.getIdentifiers(); + } + + ais.setIdentifiers(newIdentifiers); + profiler.begin("refreshSourceViewer"); + // refreshSourceViewer(); + profiler.end("refreshSourceViewer"); + } + + /** + * Update the folding structure with a given list of foldingPositions. Used + * by the editor updater of {@link AssemblerReconcilingStategy}. + * + * @param foldingPositions + * The list of foldingPositions, may be empty, not + * <code>null</code>. + */ + final void updateFoldingStructure(List<Position> foldingPositions) { + if (foldingPositions == null) { + throw new IllegalArgumentException("Parameter 'foldingPositions' must not be null."); + } + + // Create a working copy. + foldingPositions = new ArrayList<Position>(foldingPositions); + List<ProjectionAnnotation> deletions = new ArrayList<ProjectionAnnotation>(); + Object annotationObject = null; + ProjectionAnnotation annotation = null; + Position position = null; + + // Access to the annotationModel is intentionally not synchronized, as + // otherwise deadlock would be the result. + for (@SuppressWarnings("rawtypes") + Iterator iter = annotationModel.getAnnotationIterator(); iter.hasNext();) { + annotationObject = iter.next(); + + if (annotationObject instanceof ProjectionAnnotation) { + annotation = (ProjectionAnnotation) annotationObject; + + position = annotationModel.getPosition(annotation); + + if (foldingPositions.contains(position)) { + foldingPositions.remove(position); + } else { + deletions.add(annotation); + } + } + + } + + Annotation[] removeAnnotations = deletions.toArray(new Annotation[deletions.size()]); + + // This will hold the new annotations along + // with their corresponding folding positions. + HashMap<ProjectionAnnotation, Position> newAnnotations = new HashMap<ProjectionAnnotation, Position>(); + + for (int i = 0; i < foldingPositions.size(); i++) { + annotation = new ProjectionAnnotation(); + newAnnotations.put(annotation, foldingPositions.get(i)); + } + + // Do not update anything if there is actual change to preserve the + // current cursor positioning. + if (removeAnnotations.length == 0 && newAnnotations.isEmpty()) { + return; + } + + annotationModel.modifyAnnotations(removeAnnotations, newAnnotations, new Annotation[] {}); + } + + /** + * Gets the directory of the current file. + * + * @return The directory of the current file or <code>null</code>. + */ + public final File getCurrentDirectory() { + File result; + result = getCurrentFile(); + if (result != null) { + result = result.getParentFile(); + } + return result; + } + + /** + * Gets the the current file. + * + * @return The current file or <code>null</code>. + */ + public final File getCurrentFile() { + File result; + IEditorInput editorInput = getEditorInput(); + if (editorInput instanceof FileEditorInput) { + FileEditorInput fileEditorInput = (FileEditorInput) editorInput; + result = new File(fileEditorInput.getPath().toOSString()); + } else { + result = null; + } + return result; + } + + /** + * Gets the the current file. + * + * @return The current file or <code>null</code>. + */ + public final IFile getCurrentIFile() { + IFile result; + IEditorInput editorInput = getEditorInput(); + if (editorInput instanceof FileEditorInput) { + FileEditorInput fileEditorInput = (FileEditorInput) editorInput; + result = fileEditorInput.getFile(); + + } else { + result = null; + } + return result; + } + + /** + * Position the cursor to the specified line in the document. + * + * @param line + * The line number, a positive integer. + * + * @return <code>true</code> if the positioning was successful. + */ + public final boolean gotoLine(int line) { + if (line < 1) { + throw new IllegalArgumentException("Parameter 'line' must be positive. Specified value is " + line + "."); + } + IDocumentProvider provider = getDocumentProvider(); + IDocument document = provider.getDocument(getEditorInput()); + boolean result = false; + try { + int startOffset = document.getLineOffset(line - 1); + int lineLength = document.getLineLength(line - 1); + if (lineLength > 0) { + lineLength = lineLength - 1; + } + selectAndReveal(startOffset, lineLength); + result = true; + } catch (BadLocationException ex) { + plugin.logError("Cannot position to line {0}.", new Object[] { String.valueOf(line) }, ex); + result = false; + } + return result; + } + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorCompileAndRunCommandMenu.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorCompileAndRunCommandMenu.java new file mode 100644 index 00000000..4d5fc0d0 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorCompileAndRunCommandMenu.java @@ -0,0 +1,64 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import java.util.Date; +import java.util.List; + +import org.eclipse.e4.ui.di.AboutToShow; +import org.eclipse.e4.ui.model.application.ui.menu.MDirectMenuItem; +import org.eclipse.e4.ui.model.application.ui.menu.MMenuElement; +import org.eclipse.e4.ui.model.application.ui.menu.MMenuFactory; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Control; +import org.eclipse.ui.menus.WorkbenchWindowControlContribution; + +// TODO: Dynamic menu contribution is not working! +public final class AssemblerEditorCompileAndRunCommandMenu extends + WorkbenchWindowControlContribution { + + public AssemblerEditorCompileAndRunCommandMenu() { + new Exception().printStackTrace(); + } + + public AssemblerEditorCompileAndRunCommandMenu(String id) { + super(id); + } + + @AboutToShow + public void aboutToShow(List<MMenuElement> items) { + MDirectMenuItem dynamicItem = MMenuFactory.INSTANCE + .createDirectMenuItem(); + dynamicItem.setLabel("Dynamic Menu Item (" + new Date() + ")"); + dynamicItem + .setContributorURI("platform:/plugin/at.descher.eclipse.bug389063"); + dynamicItem + .setContributionURI("bundleclass://at.descher.eclipse.bug389063/at.descher.eclipse.bug389063.dynamic.DirectMenuItemAHandler"); + items.add(dynamicItem); + + } + + @Override + protected Control createControl(Composite parent) { + // TODO Auto-generated method stub + return null; + } + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorCompileCommand.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorCompileCommand.java new file mode 100644 index 00000000..7766909b --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorCompileCommand.java @@ -0,0 +1,802 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import java.io.File; +import java.io.FileNotFoundException; +import java.io.FileOutputStream; +import java.io.IOException; +import java.text.SimpleDateFormat; +import java.util.ArrayList; +import java.util.Collections; +import java.util.Comparator; +import java.util.Date; +import java.util.HashSet; +import java.util.List; +import java.util.Set; + +import org.eclipse.core.resources.IFile; +import org.eclipse.core.resources.IMarker; +import org.eclipse.core.resources.IResource; +import org.eclipse.core.resources.ResourcesPlugin; +import org.eclipse.core.runtime.CoreException; +import org.eclipse.debug.core.DebugPlugin; +import org.eclipse.debug.core.IBreakpointManager; +import org.eclipse.debug.core.model.IBreakpoint; +import org.eclipse.swt.program.Program; +import org.eclipse.ui.IPageLayout; +import org.eclipse.ui.IViewReference; +import org.eclipse.ui.IWorkbenchPage; +import org.eclipse.ui.IWorkbenchPart; +import org.eclipse.ui.PartInitException; +import org.eclipse.ui.console.IConsoleConstants; +import org.eclipse.ui.console.IConsoleView; +import org.eclipse.ui.ide.IDE; + +import com.wudsn.ide.asm.AssemblerPlugin; +import com.wudsn.ide.asm.Hardware; +import com.wudsn.ide.asm.HardwareUtility; +import com.wudsn.ide.asm.Texts; +import com.wudsn.ide.asm.compiler.Compiler; +import com.wudsn.ide.asm.compiler.CompilerConsole; +import com.wudsn.ide.asm.compiler.CompilerDefinition; +import com.wudsn.ide.asm.compiler.CompilerFileWriter; +import com.wudsn.ide.asm.compiler.CompilerFiles; +import com.wudsn.ide.asm.compiler.CompilerProcessLogParser; +import com.wudsn.ide.asm.compiler.CompilerProcessLogParser.Marker; +import com.wudsn.ide.asm.compiler.CompilerSymbol; +import com.wudsn.ide.asm.compiler.CompilerVariables; +import com.wudsn.ide.asm.preferences.CompilerRunPreferences; +import com.wudsn.ide.asm.runner.Runner; +import com.wudsn.ide.asm.runner.RunnerDefinition; +import com.wudsn.ide.asm.runner.RunnerId; +import com.wudsn.ide.base.common.FileUtility; +import com.wudsn.ide.base.common.HexUtility; +import com.wudsn.ide.base.common.MarkerUtility; +import com.wudsn.ide.base.common.NumberUtility; +import com.wudsn.ide.base.common.ProcessWithLogs; +import com.wudsn.ide.base.common.StringUtility; + +/** + * Implementation of the "Compile" command. + * + * @author Peter Dell + */ +final class AssemblerEditorCompileCommand { + + /** + * Commands. + */ + public static final String COMPILE = "com.wudsn.ide.asm.editor.AssemblerEditorCompileCommand"; + public static final String COMPILE_AND_RUN = "com.wudsn.ide.asm.editor.AssemblerEditorCompileAndRunCommand"; + public static final String COMPILE_AND_RUN_WITH = "com.wudsn.ide.asm.editor.AssemblerEditorCompileAndRunWithCommand"; + + /** + * The owning plugin. + */ + private AssemblerPlugin plugin; + + /** + * Creation is private. + */ + private AssemblerEditorCompileCommand() { + plugin = AssemblerPlugin.getInstance(); + } + + /** + * Creates a message associated with the main source file of an + * {@link CompilerFiles} instance. The message is not bound to a line + * number. + * + * @param files + * The {@link CompilerFiles} not <code>null</code>. + * @param severity + * The message severity, see {@link IMarker#SEVERITY} + * @param message + * The message, may contain parameter "{0}" to "{9}". May be + * empty, not <code>null</code>. + * @param parameters + * The format parameters for the message, may be empty, not + * <code>null</code>. + */ + private void createMainSourceFileMessage(CompilerFiles files, int severity, String message, String... parameters) { + if (files == null) { + throw new IllegalArgumentException("Parameter 'files' must not be null."); + } + MarkerUtility.createMarker(files.mainSourceFile.iFile, 0, severity, message, parameters); + } + + /** + * Executes a compile command. + * + * @param assemblerEditor + * The assembler editor, not <code>null</code>. + * @param files + * The compiler files, not <code>null</code>. + * @param commandId + * The command id, see {@link #COMPILE}, {@link #COMPILE_AND_RUN} + * , {@link #COMPILE_AND_RUN_WITH}. + * @param runnerId + * The runner id, may be empty or <code>null</code>. + * + * @throws RuntimeException + */ + public static void execute(AssemblerEditor assemblerEditor, CompilerFiles files, String commandId, String runnerId) + throws RuntimeException { + + if (assemblerEditor == null) { + throw new IllegalArgumentException("Parameter 'assemblerEditor' must not be null."); + } + if (files == null) { + throw new IllegalArgumentException("Parameter 'files' must not be null."); + } + if (commandId == null) { + throw new IllegalArgumentException("Parameter 'commandId' must not be null."); + } + + // Ensure all current changes in all open editors are saved. + if (!IDE.saveAllEditors(new IResource[] { ResourcesPlugin.getWorkspace().getRoot() }, false)) { + return; + } + + assemblerEditor.doSave(null); + + IWorkbenchPage page = assemblerEditor.getSite().getPage(); + + AssemblerEditorCompileCommand instance; + instance = new AssemblerEditorCompileCommand(); + + try { + instance.executeInternal(assemblerEditor, files, commandId, runnerId); + } catch (RuntimeException ex) { + throw ex; + } + + try { + // Remember active editor. + IWorkbenchPart activePart = page.getActivePart(); + + // Show console. + CompilerConsole compilerConsole = instance.plugin.getCompilerConsole(); + IConsoleView consoleView = (IConsoleView) page.showView(IConsoleConstants.ID_CONSOLE_VIEW); + compilerConsole.display(consoleView); + + // Show problems view. + page.showView(IPageLayout.ID_PROBLEM_VIEW); + + // Reactivate previously active editor. + page.activate(activePart); + + } catch (PartInitException ex) { + throw new RuntimeException("Cannot show view.", ex); + } + } + + private boolean executeInternal(AssemblerEditor assemblerEditor, CompilerFiles files, String commandId, + String runnerId) { + + if (assemblerEditor == null) { + throw new IllegalArgumentException("Parameter 'assemblerEditor' must not be null."); + } + if (files == null) { + throw new IllegalArgumentException("Parameter 'files' must not be null."); + } + if (commandId == null) { + throw new IllegalArgumentException("Parameter 'commandId' must not be null."); + } + + AssemblerEditorFilesLogic assemblerEditorFilesLogic = AssemblerEditorFilesLogic.createInstance(assemblerEditor); + + // Remove existing problem markers from all files. + if (!assemblerEditorFilesLogic.removeMarkers(files)) { + return false; + } + + // Determine and check hardware. + Hardware hardware = assemblerEditorFilesLogic.getHardware(files); + if (hardware == null) { + return false; + } + + // Check files based on the compiler definition. + if (!assemblerEditorFilesLogic.validateOutputFile(files)) { + return false; + } + + // Create wrapper for run properties. + CompilerDefinition compilerDefinition = assemblerEditor.getCompilerDefinition(); + CompilerRunPreferences compilerRunPreferences = new CompilerRunPreferences( + assemblerEditor.getCompilerPreferences(), files.mainSourceFile.assemblerProperties); + + // Check if output file is modifiable in case it already exists. + long outputFileLastModified = -1; + if (files.outputFile.exists()) { + boolean canWrite = files.outputFile.canWrite(); + if (canWrite) { + try { + FileOutputStream fos = new FileOutputStream(files.outputFile, true); + try { + fos.close(); + } catch (IOException ex) { + plugin.logError("Cannot close file output stream", null, ex); + } + } catch (FileNotFoundException ex) { + canWrite = false; + } + + } + if (!canWrite) { + // ERROR: Output file '{0}' cannot be opened for writing. End + // all applications which may keep the file open. + createMainSourceFileMessage(files, IMarker.SEVERITY_ERROR, Texts.MESSAGE_E106, files.outputFileName); + return false; + } + outputFileLastModified = files.outputFile.lastModified(); + } + + // Get and check path to compiler executable. + String compilerExecutablePath = plugin.getPreferences().getCompilerExecutablePath(compilerDefinition.getId()); + if (StringUtility.isEmpty(compilerExecutablePath)) { + // ERROR: Path to '{0}' compiler executable is not set in the + // preferences. + createMainSourceFileMessage(files, IMarker.SEVERITY_ERROR, Texts.MESSAGE_E100, compilerDefinition.getName()); + return false; + } + File compilerExecutableFile = new File(compilerExecutablePath); + if (!compilerExecutableFile.exists()) { + // Path to '{0}' compiler executable in the preferences points + // to non-existing file '{1}'. + createMainSourceFileMessage(files, IMarker.SEVERITY_ERROR, Texts.MESSAGE_E103, + compilerDefinition.getName(), compilerExecutablePath); + return false; + } + + // Get and check compiler executable parameters. + String compilerParameters = compilerRunPreferences.getParameters(); + if (StringUtility.isEmpty(compilerParameters)) { + compilerParameters = compilerDefinition.getDefaultParameters(); + } + + // The parameters are first split and then substituted. + // This allows for parameters and file paths inner spaces to be used. + // In some case addition quotes must be places around parameters, for + // example for the "${sourceFilePath}". This can be used to avoid + // problems with absolute file path under Unix starting with "/" or path + // containing white spaces. + compilerParameters = compilerParameters.trim(); + String compilerParameterArray[] = compilerParameters.split(" "); + if (compilerParameterArray.length == 0) { + // ERROR: The compiler '{0}' does not specify default + // parameters. + createMainSourceFileMessage(files, IMarker.SEVERITY_ERROR, Texts.MESSAGE_E101); + return false; + } + + // From here on, the method is linear, i.e. there is no "return" until + // the end. + + // Special handling for direct execution of ".jar" files. + String[] fullCommandLineArray; + int offset; + if (compilerExecutablePath.toLowerCase().endsWith(".jar")) { + offset = 3; + fullCommandLineArray = new String[offset + compilerParameterArray.length]; + fullCommandLineArray[0] = "java"; + fullCommandLineArray[1] = "-jar"; + fullCommandLineArray[2] = compilerExecutablePath; + } else { + offset = 1; + fullCommandLineArray = new String[offset + compilerParameterArray.length]; + fullCommandLineArray[0] = compilerExecutablePath; + } + + // Map parameter with variables replacement. + for (int i = 0; i < compilerParameterArray.length; i++) { + String parameter = compilerParameterArray[i]; + parameter = CompilerVariables.replaceVariables(parameter, files); + fullCommandLineArray[i + offset] = parameter; + } + + ProcessWithLogs compilerProcess = new ProcessWithLogs(fullCommandLineArray, files.mainSourceFile.folder); + CompilerConsole compilerConsole = plugin.getCompilerConsole(); + compilerConsole.println(""); + compilerConsole.println("Compiling for hardware " + hardware.name() + " on " + + new SimpleDateFormat().format(new Date()) + ": " + compilerProcess.getCommandArrayString()); + + try { + compilerProcess.exec(System.out, System.err, true); + } catch (IOException ex) { + // ERROR: Cannot execute compiler process '{0}' in working directory + // '{1}'. System error: {2} + createMainSourceFileMessage(files, IMarker.SEVERITY_ERROR, Texts.MESSAGE_E105, + compilerProcess.getCommandArrayString(), compilerProcess.getWorkingDirectory().getPath(), + ex.getMessage()); + } + + // Refresh the output and the symbols file resource. + if (files.outputFolderPath.equals(files.mainSourceFile.folderPath) + || files.outputFolderPath.equals(files.sourceFile.folderPath)) { + try { + IResource outputResource = files.mainSourceFile.iFile.getParent(); + if (outputResource != null) { + outputResource.refreshLocal(IResource.DEPTH_ONE, null); + } + outputResource = files.sourceFile.iFile.getParent(); + if (outputResource != null) { + outputResource.refreshLocal(IResource.DEPTH_ONE, null); + } + + } catch (CoreException ex) { + createMainSourceFileMessage(files, IMarker.SEVERITY_ERROR, ex.getMessage()); + return false; + } + } + + compilerConsole.println(""); + compilerConsole.println("Compiler '" + compilerDefinition.getName() + "' output:"); + compilerConsole.println(compilerProcess.getErrorLog()); + compilerConsole.println(compilerProcess.getOutputLog()); + + // Compiling is over, check the result. + Compiler compiler = assemblerEditor.getCompiler(); + boolean compilerSuccess = compiler.isSuccessExitValue(compilerProcess.getExitValue()); + if (compilerSuccess) { + if (files.outputFile.exists()) { + if (files.outputFile.length() > 0) { + if (files.outputFile.lastModified() != outputFileLastModified) { + // INFO: Output file '{0}' created or updated with {1} + // (${2}) bytes. + long fileLength = files.outputFile.length(); + createMainSourceFileMessage(files, IMarker.SEVERITY_INFO, Texts.MESSAGE_I109, + files.outputFilePath, Long.toString(fileLength), + HexUtility.getLongValueHexString(fileLength)); + + // Handle disk images + CompilerFileWriter compilerFileWriter = HardwareUtility + .getCompilerFileWriter(compilerRunPreferences.getHardware()); + if (!compilerFileWriter.createOrUpdateDiskImage(files)) { + return false; + } + + if (commandId.equals(COMPILE_AND_RUN) || commandId.equals(COMPILE_AND_RUN_WITH)) { + + openOutputFile(assemblerEditor, files, compilerRunPreferences, compilerConsole, runnerId); + + } + } else { + // ERROR: Output file not updated. Check the error + // messages and the console log. + createMainSourceFileMessage(files, IMarker.SEVERITY_ERROR, Texts.MESSAGE_E108); + + } + } else { + // ERROR: Output file created but empty. Check the error + // messages and the console log. + createMainSourceFileMessage(files, IMarker.SEVERITY_ERROR, Texts.MESSAGE_E126); + } + } else { + // ERROR: No output file created. Check the error messages and + // the + // console log. + createMainSourceFileMessage(files, IMarker.SEVERITY_ERROR, Texts.MESSAGE_E107); + } + } + + // Output an additional message if the reason for the compiler's exit + // value is not already contained in the error messages. + boolean errorFound = parseLogs(assemblerEditor, files, compilerProcess); + if (!compilerSuccess && !errorFound) { + // ERROR: Compiler process ended with error code {0}. Check the + // error messages and the console log. + createMainSourceFileMessage(files, IMarker.SEVERITY_ERROR, Texts.MESSAGE_E127, + NumberUtility.getLongValueDecimalString(compilerProcess.getExitValue())); + } + + return true; + } + + /** + * Creates or deletes the breakpoints file based on the runner. + * + * @param assemblerEditor + * The assembler editor, not <code>null</code>. + * @param files + * The assembler files, not <code>null</code>. + * @param runner + * The runner, not <code>null</code>. + * + * @since 1.6.1 + */ + private void createBreakpointsFile(AssemblerEditor assemblerEditor, CompilerFiles files, Runner runner) { + if (assemblerEditor == null) { + throw new IllegalArgumentException("Parameter 'assemblerEditor' must not be null."); + } + if (files == null) { + throw new IllegalArgumentException("Parameter 'files' must not be null."); + } + if (runner == null) { + throw new IllegalArgumentException("Parameter 'runner' must not be null."); + } + File breakpointsFile = runner.createBreakpointsFile(files); + IBreakpointManager breakpointManager = DebugPlugin.getDefault().getBreakpointManager(); + IBreakpoint breakpoints[] = breakpointManager.getBreakpoints(AssemblerBreakpoint.DEBUG_MODEL_ID); + if (breakpointsFile == null) { + if (breakpoints.length > 0) { + // WARNING: Breakpoints will be ignored because the application + // '{0}' does not support passing source level breakpoints. + createMainSourceFileMessage(files, IMarker.SEVERITY_WARNING, Texts.MESSAGE_W120, new String[] { runner + .getDefinition().getName() }); + } + return; + } + + // If breakpoints are present, a breakpoints file is generated. + if (breakpoints.length >= 0) { + AssemblerBreakpoint[] assemblerBreakpoints = new AssemblerBreakpoint[breakpoints.length]; + System.arraycopy(breakpoints, 0, assemblerBreakpoints, 0, breakpoints.length); + StringBuilder breakpointBuilder = new StringBuilder(); + int activeBreakpointCount = runner.createBreakpointsFileContent(assemblerBreakpoints, breakpointBuilder); + try { + FileUtility.writeString(breakpointsFile, breakpointBuilder.toString()); + } catch (CoreException ex) { + // ERROR: Cannot open breakpoints file '{0}' for output. System + // error: {1} + createMainSourceFileMessage(files, IMarker.SEVERITY_ERROR, Texts.MESSAGE_E122, new String[] { + breakpointsFile.getPath(), ex.getMessage() }); + return; + } + // INFO: Breakpoints file '{0}' created with {1} active breakpoints. + createMainSourceFileMessage(files, IMarker.SEVERITY_INFO, Texts.MESSAGE_I121, new String[] { + breakpointsFile.getPath(), NumberUtility.getLongValueDecimalString(activeBreakpointCount) }); + + } else { + // If no breakpoints are present, the breakpoints file is deleted. + if (breakpointsFile.exists()) { + if (!breakpointsFile.delete()) { + // ERROR: Cannot delete empty breakpoints file '{0}'. + createMainSourceFileMessage(files, IMarker.SEVERITY_ERROR, Texts.MESSAGE_E123, + new String[] { breakpointsFile.getPath() }); + } + } + } + } + + private void openOutputFile(AssemblerEditor assemblerEditor, CompilerFiles files, + CompilerRunPreferences compilerRunPreferences, CompilerConsole compilerConsole, String runnerId) { + + if (assemblerEditor == null) { + throw new IllegalArgumentException("Parameter 'assemblerEditor' must not be null."); + } + if (files == null) { + throw new IllegalArgumentException("Parameter 'files' must not be null."); + } + if (compilerRunPreferences == null) { + throw new IllegalArgumentException("Parameter 'compilerRunPreferences' must not be null."); + } + if (compilerConsole == null) { + throw new IllegalArgumentException("Parameter 'compilerConsole' must not be null."); + } + String[] fullCommandLineArray; + ProcessWithLogs runnerProcess; + + fullCommandLineArray = null; + + // If the runner id was no specified explicitly, the default from the + // preferences is used. + if (runnerId == null || StringUtility.isEmpty(runnerId)) { + runnerId = compilerRunPreferences.getRunnerId(); + } + Hardware hardware = compilerRunPreferences.getHardware(); + Runner runner; + runner = plugin.getRunnerRegistry().getRunner(hardware, runnerId); + if (runner == null) { + // ERROR: Definition for application '{0}' from the preferences of + // hardware '{1}' is not registered. + createMainSourceFileMessage(files, IMarker.SEVERITY_ERROR, Texts.MESSAGE_E116, runnerId, + hardware.toString()); + return; + } + + createBreakpointsFile(assemblerEditor, files, runner); + + String runnerCommandLine; + runnerCommandLine = compilerRunPreferences.getRunnerCommandLine(runnerId); + if (StringUtility.isEmpty(runnerCommandLine)) { + runnerCommandLine = runner.getDefinition().getDefaultCommandLine(); + } + runnerCommandLine = runnerCommandLine.trim(); + + // The parameters are first split and then substituted. + // This allows for parameters and file paths inner spaces to + // be used. In some case addition quotes must be places around + // parameters, for example for the "${outputFilePath}" for + // MADS. Otherwise using absolute file path under Unix starting + // "/" may cause conflicts. + String[] commandLineArray; + commandLineArray = runnerCommandLine.split(" "); + + // Execution type: DEFAULT_APPLICATION + if (runnerId.equals(RunnerId.DEFAULT_APPLICATION)) { + String extension = files.outputFileName; + int index = extension.lastIndexOf('.'); + if (index > 0) { + extension = extension.substring(index); + } + Program program = Program.findProgram(extension); + if (program == null) { + // ERROR: Cannot open output file '{0}' with the + // standard application since no application is + // registered for the file extension '{1}'. + createMainSourceFileMessage(files, IMarker.SEVERITY_ERROR, Texts.MESSAGE_E115, files.outputFilePath, + extension); + } else { + + if (Program.launch(files.outputFilePath)) { + // INFO: Opening output file '{0}' with + // application + // '{1}'. + createMainSourceFileMessage(files, IMarker.SEVERITY_INFO, Texts.MESSAGE_I118, files.outputFilePath, + program.getName()); + + compilerConsole.println("Running '" + runner.getDefinition().getName() + "': " + program.getName() + + " " + files.outputFilePath); + } else { + // ERROR: Cannot open output file '{0}' with + // application '{1}'. + createMainSourceFileMessage(files, IMarker.SEVERITY_ERROR, Texts.MESSAGE_E119, + files.outputFilePath, program.getName()); + } + } + } + // Execution type: pre-defined or USER_DEFINED_APPLICATION + else { + boolean error = false; + + String runnerExecutablePath = compilerRunPreferences.getRunnerExecutablePath(runnerId); + if (runnerCommandLine.contains(RunnerDefinition.RUNNER_EXECUTABLE_PATH)) { + if (StringUtility.isEmpty(runnerExecutablePath)) { + // ERROR: Path to application executable is not + // set in the preferences of application '{0}'. + createMainSourceFileMessage(files, IMarker.SEVERITY_ERROR, Texts.MESSAGE_E112, runner + .getDefinition().getName()); + error = true; + } else { + File runnerExecutableFile = new File(runnerExecutablePath); + if (!runnerExecutableFile.exists()) { + // ERROR: Path to '{0}' application + // executable in the preferences points to + // non-existing file '{1}'. + createMainSourceFileMessage(files, IMarker.SEVERITY_ERROR, Texts.MESSAGE_E114, runner + .getDefinition().getName(), runnerExecutablePath); + error = true; + } + } + } + if (!error) { + + fullCommandLineArray = new String[commandLineArray.length]; + for (int i = 0; i < commandLineArray.length; i++) { + String parameter = commandLineArray[i]; + parameter = CompilerVariables.replaceVariables(parameter, files); + parameter = replaceRunnerParameters(parameter, runnerExecutablePath); + parameter = parameter.replace(RunnerDefinition.OUTPUT_FILE_PATH, files.outputFilePath); + fullCommandLineArray[i] = parameter; + } + + // INFO: Opening output file '{0}' with application '{1}'. + createMainSourceFileMessage(files, IMarker.SEVERITY_INFO, Texts.MESSAGE_I118, files.outputFilePath, + runner.getDefinition().getName()); + + runnerProcess = new ProcessWithLogs(fullCommandLineArray, files.outputFolder); + + compilerConsole.println("Running '" + runner.getDefinition().getName() + "': " + + runnerProcess.getCommandArrayString()); + + try { + boolean wait = compilerRunPreferences.isRunnerWaitForCompletion(runnerId); + runnerProcess.exec(compilerConsole.getPrintStream(), compilerConsole.getPrintStream(), wait); + compilerConsole + .println("Application returned with exit code " + runnerProcess.getExitValue() + "."); + } catch (IOException ex) { + // ERROR: Cannot execute application '{0}' process '{1}' in + // working directory '{2}'. System error: {3} + createMainSourceFileMessage(files, IMarker.SEVERITY_ERROR, Texts.MESSAGE_E113, runner + .getDefinition().getName(), runnerProcess.getCommandArrayString(), runnerProcess + .getWorkingDirectory().getPath(), ex.getMessage()); + } + } + } + } + + private String replaceRunnerParameters(String parameter, String runnerExecutablePath) { + if (parameter == null) { + throw new IllegalArgumentException("Parameter 'parameter' must not be null."); + } + if (runnerExecutablePath == null) { + throw new IllegalArgumentException("Parameter 'runnerExecutablePath' must not be null."); + } + parameter = parameter.replace(RunnerDefinition.RUNNER_EXECUTABLE_PATH, runnerExecutablePath); + return parameter; + } + + private boolean parseLogs(AssemblerEditor assemblerEditor, CompilerFiles files, ProcessWithLogs compileProcess) { + + if (assemblerEditor == null) { + throw new IllegalArgumentException("Parameter 'assemblerEditor' must not be null."); + } + if (files == null) { + throw new IllegalArgumentException("Parameter 'files' must not be null."); + } + if (compileProcess == null) { + throw new IllegalArgumentException("Parameter 'compileProcess' must not be null."); + } + Compiler compiler = assemblerEditor.getCompiler(); + CompilerProcessLogParser logParser = compiler.createLogParser(); + + // Line parser with main source file and logs. + logParser.setLogs(files, compileProcess.getOutputLog(), compileProcess.getErrorLog()); + + Set<Marker> set = new HashSet<Marker>(); + List<IMarker> markers = new ArrayList<IMarker>(); + while (logParser.nextMarker()) { + Marker markerProxy = logParser.getMarker(); + while (markerProxy != null) { // Loop to add main marker and its detail markers + if (!set.contains(markerProxy)) { + set.add(markerProxy); + try { + IFile iFile = markerProxy.getIFile(); + IMarker marker = iFile.createMarker(IMarker.PROBLEM); + marker.setAttribute(IMarker.SEVERITY, markerProxy.getSeverity()); + marker.setAttribute(IMarker.MESSAGE, markerProxy.getMessage()); + int lineNumber = markerProxy.getLineNumber(); + if (lineNumber > 0) { + marker.setAttribute(IMarker.LINE_NUMBER, lineNumber); + } + marker.setAttribute(IMarker.TRANSIENT, true); + markers.add(marker); + } catch (CoreException ex) { + throw new RuntimeException(ex); + } + } + markerProxy = markerProxy.getDetailMarker(); + } + } + boolean errorOccurred = positionToFirstErrorOrWarning(assemblerEditor, markers); + + parseCompilerSymbols(assemblerEditor, files, logParser); + + return errorOccurred; + } + + /** + * Positions to the first error or warning in any file for which markers + * have been created. + * + * @param assemblerEditor + * The assembler editor, not <code>null</code>. Used as basis for + * opening another editor when required. + * @param markers + * The modifiable list of marker, may be empty, not + * <code>null</code>. + * @return <code>true</code> if an error was found. + */ + private boolean positionToFirstErrorOrWarning(AssemblerEditor assemblerEditor, List<IMarker> markers) { + if (assemblerEditor == null) { + throw new IllegalArgumentException("Parameter 'assemblerEditor' must not be null."); + } + if (markers == null) { + throw new IllegalArgumentException("Parameter 'markers' must not be null."); + } + + Collections.sort(markers, new Comparator<IMarker>() { + + @Override + public int compare(IMarker o1, IMarker o2) { + int result = o1.getResource().getFullPath().toString() + .compareTo(o2.getResource().getFullPath().toString()); + if (result == 0) { + result = o1.getAttribute(IMarker.LINE_NUMBER, 0) - o2.getAttribute(IMarker.LINE_NUMBER, 0); + } + return result; + } + + }); + + String positioningMode = plugin.getPreferences().getEditorCompileCommandPositioningMode(); + boolean ignoreWarnings = positioningMode.equals(AssemblerEditorCompileCommandPositioningMode.FIRST_ERROR); + IMarker firstWarningMarker = null; + IMarker firstErrorMarker = null; + for (IMarker marker : markers) { + + switch (marker.getAttribute(IMarker.SEVERITY, 0)) { + case IMarker.SEVERITY_WARNING: + if (!ignoreWarnings) { + + if (firstWarningMarker == null) { + firstWarningMarker = marker; + } + + } + break; + case IMarker.SEVERITY_ERROR: + if (firstErrorMarker == null) { + firstErrorMarker = marker; + } + break; + } + + } + IMarker firstMarker = firstErrorMarker; + if (firstMarker == null) { + firstMarker = firstWarningMarker; + } + + if (firstMarker != null) { + MarkerUtility.gotoMarker(assemblerEditor, firstMarker); + } + return firstErrorMarker != null; + } + + private void parseCompilerSymbols(AssemblerEditor assemblerEditor, CompilerFiles files, + CompilerProcessLogParser logParser) { + + if (assemblerEditor == null) + throw new IllegalArgumentException("Parameter 'assemblerEditor' must not be null."); + if (files == null) + throw new IllegalArgumentException("Parameter 'files' must not be null."); + if (logParser == null) + throw new IllegalArgumentException("Parameter 'logParser' must not be null."); + + List<CompilerSymbol> compilerSymbols; + + compilerSymbols = new ArrayList<CompilerSymbol>(); + try { + logParser.addCompilerSymbols(compilerSymbols); + } catch (RuntimeException ex) { + String message = ex.getMessage(); + if (message == null) { + message = ex.getClass().getName(); + } + createMainSourceFileMessage(files, IMarker.SEVERITY_ERROR, message); + AssemblerPlugin.getInstance().logError("Error in addCompilerSymbols()", null, ex); + + } catch (CoreException ex) { + String message = ex.getMessage(); + if (message == null) { + message = ex.getClass().getName(); + } + createMainSourceFileMessage(files, IMarker.SEVERITY_ERROR, message); + + } + + // Display symbols. + IViewReference[] references = assemblerEditor.getSite().getPage().getViewReferences(); + for (IViewReference reference : references) { + if (reference.getId().equals(CompilerSymbolsView.ID)) { + CompilerSymbolsView compilerSymbolsView = (CompilerSymbolsView) reference.getView(true); + if (compilerSymbolsView != null) { + compilerSymbolsView.setSymbols(files, compilerSymbols); + } + } + + } + + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorCompileCommandDelegate.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorCompileCommandDelegate.java new file mode 100644 index 00000000..f9bd3968 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorCompileCommandDelegate.java @@ -0,0 +1,243 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import java.util.List; + +import org.eclipse.jface.action.Action; +import org.eclipse.jface.action.ActionContributionItem; +import org.eclipse.jface.action.IAction; +import org.eclipse.jface.resource.ImageDescriptor; +import org.eclipse.jface.viewers.ISelection; +import org.eclipse.swt.widgets.Control; +import org.eclipse.swt.widgets.Event; +import org.eclipse.swt.widgets.Menu; +import org.eclipse.ui.IActionDelegate2; +import org.eclipse.ui.IEditorPart; +import org.eclipse.ui.IWorkbenchPage; +import org.eclipse.ui.IWorkbenchWindow; +import org.eclipse.ui.IWorkbenchWindowPulldownDelegate2; + +import com.wudsn.ide.asm.AssemblerPlugin; +import com.wudsn.ide.asm.Hardware; +import com.wudsn.ide.asm.HardwareUtility; +import com.wudsn.ide.asm.Texts; +import com.wudsn.ide.asm.preferences.CompilerPreferences; +import com.wudsn.ide.asm.runner.RunnerDefinition; +import com.wudsn.ide.asm.runner.RunnerId; +import com.wudsn.ide.asm.runner.RunnerRegistry; +import com.wudsn.ide.base.common.StringUtility; + +/** + * Delegate class to provide a dynamic drop-down menu for the toolbar based on + * the configured runners for the currently active compiler's hardware. + * + * @author Peter Dell + * + */ +public final class AssemblerEditorCompileCommandDelegate implements IActionDelegate2, IWorkbenchWindowPulldownDelegate2 { + + private IWorkbenchWindow window; + private Menu menu; + + /** + * Private action class. + * + * @author Peter Dell + */ + private final class CompileAndRunAction extends Action { + private String runnerId; + + public CompileAndRunAction(String runnerId) { + if (runnerId == null) { + throw new IllegalArgumentException("Parameter 'runnerId' must not be null."); + } + this.runnerId = runnerId; + } + + /** + * @see org.eclipse.jface.action.IAction#run() + */ + @Override + public void run() { + + } + + @Override + public void runWithEvent(Event event) { + compileAndRun(runnerId); + } + } + + /** + * Creation is public. + */ + public AssemblerEditorCompileCommandDelegate() { + } + + @Override + public void init(IWorkbenchWindow window) { + this.window = window; + } + + @Override + public Menu getMenu(Control parent) { + + AssemblerEditor assemblerEditor = getAssemblerEditor(); + if (assemblerEditor == null) { + return null; + } + + AssemblerPlugin assemblerPlugin = assemblerEditor.getPlugin(); + RunnerRegistry runnerRegistry = assemblerPlugin.getRunnerRegistry(); + Hardware hardware = assemblerEditor.getHardware(); + List<RunnerDefinition> runnerDefinitions = runnerRegistry.getDefinitions(hardware); + CompilerPreferences compilerPreferences = assemblerEditor.getCompilerPreferences(); + + Menu menu = new Menu(parent); + setMenu(menu); + + // Collect all runner definition for which the executable path is + // maintained correctly. + ImageDescriptor imageDescriptor = HardwareUtility.getImageDescriptor(hardware); + for (RunnerDefinition runnerDefinition : runnerDefinitions) { + String runnerId = runnerDefinition.getId(); + String runnerName = runnerDefinition.getName(); + // The system default application does not need an executable path. + if (!runnerId.equals(RunnerId.DEFAULT_APPLICATION)) { + if (StringUtility.isEmpty(compilerPreferences.getRunnerExecutablePath(runnerId))) { + continue; + } + } + + Action action = new CompileAndRunAction(runnerId); + action.setActionDefinitionId(AssemblerEditorCompileCommand.COMPILE_AND_RUN_WITH); + action.setImageDescriptor(imageDescriptor); + if (runnerId.equals(compilerPreferences.getRunnerId())) { + runnerName = runnerName + " " + Texts.ASSEMBLER_TOOLBAR_RUN_WITH_DEFAULT_LABEL; + } + action.setText(runnerName); + ActionContributionItem item = new ActionContributionItem(action); + item.fill(menu, -1); + } + + return menu; + } + + @Override + public Menu getMenu(Menu parent) { + return null; + } + + @Override + public void dispose() { + setMenu(null); + } + + @Override + public void init(IAction action) { + AssemblerEditor assemblerEditor = getAssemblerEditor(); + boolean enabled = assemblerEditor != null && assemblerEditor.getCurrentIFile() != null; + action.setEnabled(enabled); + + Hardware hardware; + if (assemblerEditor != null) { + hardware = assemblerEditor.getHardware(); + + } else { + hardware = Hardware.GENERIC; + } + ImageDescriptor imageDescriptor = HardwareUtility.getImageDescriptor(hardware); + action.setImageDescriptor(imageDescriptor); + } + + @Override + public void run(IAction action) { + // Not invoked + } + + @Override + public void runWithEvent(IAction action, Event event) { + try { + compileAndRun(null); + } catch (RuntimeException ex) { + AssemblerPlugin.getInstance().showError(window.getShell(), "Error in compileAndRun()", ex); + } + } + + @Override + public void selectionChanged(IAction action, ISelection selection) { + init(action); + // TODO: Check disabling of action based on current selection. + // System.out.println(action.getActionDefinitionId()); + // System.out.println(selection); + } + + /** + * Helper method to correctly retain and dispose menu instances. + * + * @param menu + * The menu to be retained or <code>null</code>. + */ + private void setMenu(Menu menu) { + if (this.menu != null) { + this.menu.dispose(); + } + this.menu = menu; + } + + /** + * Gets the currently active assembler editor. + * + * @return The currently active assembler editor or <code>null</code>. + */ + private AssemblerEditor getAssemblerEditor() { + if (window == null) { + return null; + } + + IWorkbenchPage workbenchPage = window.getActivePage(); + if (workbenchPage == null) { + return null; + } + IEditorPart editorPart = workbenchPage.getActiveEditor(); + if (!(editorPart instanceof AssemblerEditor)) { + return null; + } + return (AssemblerEditor) editorPart; + } + + /** + * Call the compiler and run command with the specified runner id. + * + * @param runnerId + * The runner id or <code>null</code> to use the default. + */ + final void compileAndRun(String runnerId) { + AssemblerEditor assemblerEditor = getAssemblerEditor(); + if (assemblerEditor == null) { + throw new IllegalStateException("Action is active but no assembler editor is active."); + } + AssemblerEditorCompileCommand.execute(assemblerEditor, AssemblerEditorFilesLogic + .createInstance(assemblerEditor).createCompilerFiles(), AssemblerEditorCompileCommand.COMPILE_AND_RUN, + runnerId); + } + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorCompileCommandHandler.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorCompileCommandHandler.java new file mode 100644 index 00000000..23eb8058 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorCompileCommandHandler.java @@ -0,0 +1,63 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import org.eclipse.core.commands.ExecutionEvent; +import org.eclipse.core.commands.ExecutionException; + +import com.wudsn.ide.asm.compiler.CompilerFiles; + +/** + * Event handler for the "Compile" command. + * + * @author Peter Dell + */ +public final class AssemblerEditorCompileCommandHandler extends + AssemblerEditorFilesCommandHandler { + + /** + * Creates a new instance. Called by the extension point + * "org.eclipse.ui.handlers". + */ + public AssemblerEditorCompileCommandHandler() { + + } + + @Override + protected void execute(ExecutionEvent event, + AssemblerEditor assemblerEditor, CompilerFiles files) + throws ExecutionException { + if (event == null) { + throw new IllegalArgumentException("Parameter 'event' must not be null."); + } + if (assemblerEditor == null) { + throw new IllegalArgumentException("Parameter 'assemblerEditor' must not be null."); + } + if (files == null) { + throw new IllegalArgumentException("Parameter 'files' must not be null."); + } + try { + AssemblerEditorCompileCommand.execute(assemblerEditor, files, event + .getCommand().getId(), null); + } catch (RuntimeException ex) { + throw new ExecutionException("Cannot execute event " + event, ex); + } + } +} \ No newline at end of file diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorCompileCommandPositioningMode.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorCompileCommandPositioningMode.java new file mode 100644 index 00000000..2c1ee5fc --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorCompileCommandPositioningMode.java @@ -0,0 +1,39 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +/** + * Constant definition for positioning after compiling. + * + * @author Peter Dell + * + */ +public final class AssemblerEditorCompileCommandPositioningMode { + + /** + * Creation is private. + */ + private AssemblerEditorCompileCommandPositioningMode() { + + } + + public static final String FIRST_ERROR_OR_WARNING = "FIRST_ERROR_OR_WARNING"; + public static final String FIRST_ERROR = "FIRST_ERROR"; +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorCompilerHelpCommandHandler.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorCompilerHelpCommandHandler.java new file mode 100644 index 00000000..b0f122c7 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorCompilerHelpCommandHandler.java @@ -0,0 +1,74 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import java.io.File; + +import org.eclipse.core.commands.AbstractHandler; +import org.eclipse.core.commands.ExecutionEvent; +import org.eclipse.core.commands.ExecutionException; +import org.eclipse.core.runtime.CoreException; +import org.eclipse.jface.dialogs.MessageDialog; +import org.eclipse.swt.program.Program; +import org.eclipse.swt.widgets.Shell; +import org.eclipse.ui.IEditorPart; +import org.eclipse.ui.handlers.HandlerUtil; + +import com.wudsn.ide.asm.AssemblerPlugin; +import com.wudsn.ide.asm.compiler.CompilerDefinition; +import com.wudsn.ide.asm.preferences.AssemblerPreferences; + +/** + * Event handler for the "Compiler Help" command. + * + * @author Peter Dell + */ +public final class AssemblerEditorCompilerHelpCommandHandler extends AbstractHandler { + + @Override + public Object execute(ExecutionEvent event) throws ExecutionException { + Shell shell; + shell = HandlerUtil.getActiveShell(event); + + IEditorPart editor; + editor = HandlerUtil.getActiveEditorChecked(event); + if (!(editor instanceof AssemblerEditor)) { + return null; + } + + AssemblerEditor assemblerEditor; + assemblerEditor = (AssemblerEditor) editor; + + CompilerDefinition compilerDefinition = assemblerEditor.getCompilerDefinition(); + AssemblerPreferences assemblerPreferences = AssemblerPlugin.getInstance().getPreferences(); + String compilerExecutablePath = assemblerPreferences.getCompilerExecutablePath(compilerDefinition.getId()); + + try { + File file = compilerDefinition.getHelpFile(compilerExecutablePath); + Program.launch(file.getPath()); + + } catch (CoreException ex) { + // ERROR: Display text from core exception. + MessageDialog.openInformation(shell, com.wudsn.ide.base.Texts.DIALOG_TITLE, ex.getMessage()); + } + + return null; + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorFilesCommandHandler.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorFilesCommandHandler.java new file mode 100644 index 00000000..9b84c63a --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorFilesCommandHandler.java @@ -0,0 +1,98 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import org.eclipse.core.commands.AbstractHandler; +import org.eclipse.core.commands.ExecutionEvent; +import org.eclipse.core.commands.ExecutionException; +import org.eclipse.core.commands.common.NotDefinedException; +import org.eclipse.ui.IEditorPart; +import org.eclipse.ui.handlers.HandlerUtil; + +import com.wudsn.ide.asm.AssemblerPlugin; +import com.wudsn.ide.asm.compiler.CompilerFiles; + +/** + * Base class for commands which operate on the current file of an assembler + * editor, in case the file is within the work space. The base class ensures + * that the corresponding command is disabled, if there is no active assembler + * editor or the editor contains a file from outside of the work space. + * + * @author Peter Dell + */ +public abstract class AssemblerEditorFilesCommandHandler extends + AbstractHandler { + + public AssemblerEditorFilesCommandHandler() { + super(); + } + + @Override + public Object execute(ExecutionEvent event) throws ExecutionException { + IEditorPart editor; + editor = HandlerUtil.getActiveEditorChecked(event); + if (!(editor instanceof AssemblerEditor)) { + return null; + } + + AssemblerEditor assemblerEditor; + assemblerEditor = (AssemblerEditor) editor; + + CompilerFiles files; + files = AssemblerEditorFilesLogic.createInstance(assemblerEditor).createCompilerFiles(); + + if (files != null) { + execute(event, assemblerEditor, files); + } else { + try { + AssemblerPlugin + .getInstance() + .showError( + assemblerEditor.getSite().getShell(), + "Operation '" + + event.getCommand().getName() + + "' is not possible because the file in the editor is not located in the worksapce.", + new Exception()); + } catch (NotDefinedException ignore) { + // Ignore + } + } + return null; + } + + /** + * Perform the action on the current editor and file. + * + * @param event + * The event, not <code>null</code>. + * @param assemblerEditor + * The assembler editor, not <code>null</code> and with current + * files which are not <code>null</code>. + * @param files + * The current compiler files of the editor, not <code>null</code> + * . + * @throws ExecutionException + * if an exception occurred during execution. + */ + protected abstract void execute(ExecutionEvent event, + AssemblerEditor assemblerEditor, CompilerFiles files) + throws ExecutionException; + +} \ No newline at end of file diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorFilesLogic.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorFilesLogic.java new file mode 100644 index 00000000..818a8291 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorFilesLogic.java @@ -0,0 +1,372 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import java.io.File; +import java.util.Map; +import java.util.TreeMap; + +import org.eclipse.core.resources.IFile; +import org.eclipse.core.resources.IMarker; +import org.eclipse.core.resources.IResource; +import org.eclipse.core.resources.ResourcesPlugin; +import org.eclipse.core.runtime.CoreException; +import org.eclipse.core.runtime.IPath; +import org.eclipse.jface.text.Document; +import org.eclipse.jface.text.IDocument; + +import com.wudsn.ide.asm.AssemblerPlugin; +import com.wudsn.ide.asm.AssemblerProperties; +import com.wudsn.ide.asm.AssemblerProperties.AssemblerProperty; +import com.wudsn.ide.asm.AssemblerProperties.InvalidAssemblerPropertyException; +import com.wudsn.ide.asm.Hardware; +import com.wudsn.ide.asm.Texts; +import com.wudsn.ide.asm.compiler.CompilerDefinition; +import com.wudsn.ide.asm.compiler.CompilerFiles; +import com.wudsn.ide.asm.compiler.CompilerOutputFolderMode; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; +import com.wudsn.ide.base.common.FileUtility; +import com.wudsn.ide.base.common.MarkerUtility; +import com.wudsn.ide.base.common.StringUtility; + +/** + * Logic to handle the {@link CompilerFiles} of an {@link AssemblerEditor} + * + * @author Peter Dell + * + * @since 1.7.0 + */ +public final class AssemblerEditorFilesLogic { + + private AssemblerEditor assemblerEditor; + + /** + * Create a new instance of the logic. + * + * @param assemblerEditor + * The assembler editor, not <code>null</code>. + * + * @return The new instance, not <code>null</code>. + */ + static AssemblerEditorFilesLogic createInstance(AssemblerEditor assemblerEditor) { + if (assemblerEditor == null) { + throw new IllegalArgumentException("Parameter 'assemblerEditor' must not be null."); + } + + AssemblerEditorFilesLogic result = new AssemblerEditorFilesLogic(); + result.assemblerEditor = assemblerEditor; + return result; + } + + /** + * Gets the container with all files related to the current files. + * + * @return The container with all files related to the current files or + * <code>null</code>. + */ + CompilerFiles createCompilerFiles() { + IFile sourceIFile; + CompilerFiles result; + sourceIFile = assemblerEditor.getCurrentIFile(); + if (sourceIFile != null) { + + IDocument document = assemblerEditor.getDocumentProvider().getDocument(assemblerEditor.getEditorInput()); + AssemblerProperties sourceFileProperties = CompilerSourceParser.getDocumentProperties(document); + + IFile mainSourceIFile; + AssemblerProperties mainSourceFileProperties; + + mainSourceIFile = sourceIFile; + mainSourceFileProperties = sourceFileProperties; + + AssemblerProperty property = sourceFileProperties.get(AssemblerProperties.MAIN_SOURCE_FILE); + if (property != null) { + if (StringUtility.isSpecified(property.value)) { + IPath mainSourceFileIPath; + mainSourceFileIPath = sourceIFile.getFullPath().removeLastSegments(1).append(property.value); + mainSourceIFile = ResourcesPlugin.getWorkspace().getRoot().getFile(mainSourceFileIPath); + File mainSourceFile = new File(mainSourceIFile.getLocation().toOSString()); + + try { + String mainSource; + + mainSource = FileUtility.readString(mainSourceFile, FileUtility.MAX_SIZE_UNLIMITED); + document = new Document(mainSource); + mainSourceFileProperties = CompilerSourceParser.getDocumentProperties(document); + } catch (CoreException ex) { + AssemblerPlugin plugin = AssemblerPlugin.getInstance(); + + plugin.logError("Cannot read main source file '{0'}", + new Object[] { mainSourceFile.getPath() }, ex); + mainSourceFileProperties = new AssemblerProperties(); + } + + } + } + result = new CompilerFiles(mainSourceIFile, mainSourceFileProperties, sourceIFile, sourceFileProperties, + assemblerEditor.getCompilerPreferences()); + } else { + result = null; + } + return result; + } + + public boolean removeMarkers(CompilerFiles files) { + if (files == null) { + throw new IllegalArgumentException("Parameter 'files' must not be null."); + } + + // Remove markers from the current file. + try { + files.sourceFile.iFile.deleteMarkers(IMarker.PROBLEM, true, IResource.DEPTH_INFINITE); + } catch (CoreException ex) { + assemblerEditor.getPlugin().logError("Cannot remove markers", null, ex); + } + + // If the main source file is not there, the error shall be related to + // the include source file. In all other cases, the main source file + // exists and is the main message target. + if (!files.mainSourceFile.iFile.exists()) { + int lineNumber = files.sourceFile.assemblerProperties.get(AssemblerProperties.MAIN_SOURCE_FILE).lineNumber; + + // ERROR: Main source file '{0}' does not exist. + IMarker marker = MarkerUtility.createMarker(files.sourceFile.iFile, lineNumber, IMarker.SEVERITY_ERROR, + Texts.MESSAGE_E125, files.mainSourceFile.filePath); + MarkerUtility.gotoMarker(assemblerEditor, marker); + return false; + } + + // Remove markers from the other relevant files. + // TODO Use (only) include files parsed from main source file + try { + files.mainSourceFile.iFile.getParent().deleteMarkers(IMarker.PROBLEM, true, IResource.DEPTH_INFINITE); + } catch (CoreException ex) { + assemblerEditor.getPlugin().logError("Cannot remove markers", null, ex); + } + return true; + } + + /** + * Determine the hardware defined by the property + * {@link AssemblerProperties#HARDWARE}. + * + * @param iFile + * The iFIle to which error message will be associated, not + * <code>null</code>. + * @param properties + * The assembler properties, not <code>null</code>. + * + * @return The hardware or <code>null</code> if is the not defined in the + * properties. + * @throws InvalidAssemblerPropertyException + * If the hardware is specified but invalid. Error message will + * be assigned to the iFile in this case. + * + * @since 1.6.1 + */ + Hardware getHardware(IFile iFile, AssemblerProperties properties) throws InvalidAssemblerPropertyException { + if (iFile == null) { + throw new IllegalArgumentException("Parameter 'iFile' must not be null."); + } + Hardware hardware = null; + AssemblerProperty hardwareProperty = properties.get(AssemblerProperties.HARDWARE); + if (hardwareProperty != null) { + Map<String, Hardware> allowedValues = new TreeMap<String, Hardware>(); + StringBuilder allowedValuesBuilder = new StringBuilder(); + for (Hardware value : Hardware.values()) { + + if (value != Hardware.GENERIC) { + if (allowedValuesBuilder.length() > 0) { + allowedValuesBuilder.append(","); + } + allowedValues.put(value.name(), value); + allowedValuesBuilder.append(value.name()); + } + } + + String hardwarePropertyValue = hardwareProperty.value.toUpperCase(); + if (StringUtility.isEmpty(hardwarePropertyValue)) { + try { + iFile.deleteMarkers(IMarker.PROBLEM, true, IResource.DEPTH_ZERO); + } catch (CoreException ex) { + AssemblerPlugin.getInstance().logError("Cannot remove markers", null, ex); + } + // ERROR: Hardware not specified. Specify one of the + // following valid values '{0}'. + IMarker marker = MarkerUtility.createMarker(iFile, hardwareProperty.lineNumber, IMarker.SEVERITY_ERROR, + Texts.MESSAGE_E128, new String[] { allowedValuesBuilder.toString() }); + throw new InvalidAssemblerPropertyException(hardwareProperty, marker); + } + hardware = allowedValues.get(hardwarePropertyValue); + + if (hardware == null) { + try { + iFile.deleteMarkers(IMarker.PROBLEM, true, IResource.DEPTH_ZERO); + } catch (CoreException ex) { + assemblerEditor.getPlugin().logError("Cannot remove markers", null, ex); + } + // ERROR: Unknown hardware {0}. Specify one of the + // following valid values '{1}'. + IMarker marker = MarkerUtility.createMarker(iFile, hardwareProperty.lineNumber, IMarker.SEVERITY_ERROR, + Texts.MESSAGE_E124, new String[] { hardwarePropertyValue, allowedValuesBuilder.toString() }); + throw new InvalidAssemblerPropertyException(hardwareProperty, marker); + } + + } + return hardware; + } + + /** + * Determine the hardware of the main source file and makes sure that the + * include file uses the same hardware. + * + * @param files + * The compiler files, not <code>null</code>. + * @return The hardware to be used, or <code>null</code> if errors have + * occurred. + * + * @since 1.6.1 + */ + public Hardware getHardware(CompilerFiles files) { + if (files == null) { + throw new IllegalArgumentException("Parameter 'files' must not be null."); + } + Hardware mainSourceFileHardware; + Hardware sourceFileHardware; + try { + sourceFileHardware = getHardware(files.sourceFile.iFile, files.sourceFile.assemblerProperties); + } catch (InvalidAssemblerPropertyException ex) { + MarkerUtility.gotoMarker(assemblerEditor, ex.marker); + return null; + } + try { + mainSourceFileHardware = getHardware(files.mainSourceFile.iFile, files.mainSourceFile.assemblerProperties); + } catch (InvalidAssemblerPropertyException ex) { + MarkerUtility.gotoMarker(assemblerEditor, ex.marker); + return null; + } + Hardware defaultHardware = assemblerEditor.getCompilerDefinition().getDefaultHardware(); + int sourceFileLineNumber; + int mainSourceFileLineNumber; + if (sourceFileHardware == null) { + sourceFileHardware = defaultHardware; + sourceFileLineNumber = 0; + } else { + sourceFileLineNumber = files.sourceFile.assemblerProperties.get(AssemblerProperties.HARDWARE).lineNumber; + } + if (mainSourceFileHardware == null) { + mainSourceFileHardware = defaultHardware; + mainSourceFileLineNumber = 0; + } else { + mainSourceFileLineNumber = files.mainSourceFile.assemblerProperties.get(AssemblerProperties.HARDWARE).lineNumber; + } + + if (!sourceFileHardware.equals(mainSourceFileHardware)) { + // ERROR: Main source file specifies or defaults to hardware {0} + // while include file specifies or defaults to hardware '{1}'. + MarkerUtility.createMarker(files.mainSourceFile.iFile, mainSourceFileLineNumber, IMarker.SEVERITY_ERROR, + Texts.MESSAGE_E129, new String[] { mainSourceFileHardware.name(), sourceFileHardware.name() }); + MarkerUtility.createMarker(files.sourceFile.iFile, sourceFileLineNumber, IMarker.SEVERITY_ERROR, + Texts.MESSAGE_E129, new String[] { mainSourceFileHardware.name(), sourceFileHardware.name() }); + return null; + } + return mainSourceFileHardware; + } + + /** + * Validates the output file related settings of the compiler files. + * + * @param files + * The compiler files, not <code>null</code>. + * @return <code>true</code> if all settings are correct, <code>false</code> + * otherwise. + */ + public boolean validateOutputFile(CompilerFiles files) { + if (files == null) { + throw new IllegalArgumentException("Parameter 'files' must not be null."); + } + CompilerDefinition compilerDefinition = assemblerEditor.getCompilerDefinition(); + if (StringUtility.isEmpty(files.outputFileExtension)) { + // ERROR: Output file extension must be set in the preferences of + // compiler '{0}' or via the annotation '{1}'. + createMainSourceFileMessage(files, files.outputFileExtensionProperty, IMarker.SEVERITY_ERROR, + Texts.MESSAGE_E104, compilerDefinition.getName(), AssemblerProperties.OUTPUT_FILE_EXTENSION); + + return false; + } + if (!files.outputFileExtension.startsWith(".")) { + // ERROR: Output file extension {0} must start with ".". + createMainSourceFileMessage(files, files.outputFileExtensionProperty, IMarker.SEVERITY_ERROR, + Texts.MESSAGE_E139, files.outputFileExtension); + return false; + } + + if (StringUtility.isEmpty(files.outputFolderMode)) { + // ERROR: Output folder mode be set in the preferences of + // compiler '{0}' or via the annotation '{1}'. + createMainSourceFileMessage(files, files.outputFolderModeProperty, IMarker.SEVERITY_ERROR, + Texts.MESSAGE_E140, compilerDefinition.getName(), AssemblerProperties.OUTPUT_FOLDER_MODE); + + return false; + } + if (!CompilerOutputFolderMode.isDefined(files.outputFolderMode)) { + // ERROR: Unknown output folder mode {0}. Specify one of the + // following valid values '{1}'. + createMainSourceFileMessage(files, files.outputFolderModeProperty, IMarker.SEVERITY_ERROR, + Texts.MESSAGE_E141, files.outputFolderMode, CompilerOutputFolderMode.getAllowedValues()); + return false; + + } + + return true; + } + + /** + * Creates a message associated with the main source file of an + * {@link AssemblerEditorFilesLogic} instance. The message is is bound to + * the line number number of the property (if available). Also the editor is + * position to the marker. + * + * @param files + * The {@link AssemblerEditorFilesLogic} not <code>null</code>. + * @param property + * The assembler editor property to which the message belongs or + * <code>null</code>. + * @param severity + * The message severity, see {@link IMarker#SEVERITY} + * @param message + * The message, may contain parameter "{0}" to "{9}". May be + * empty, not <code>null</code>. + * @param parameters + * The format parameters for the message, may be empty, not + * <code>null</code>. + */ + private void createMainSourceFileMessage(CompilerFiles files, AssemblerProperty property, int severity, + String message, String... parameters) { + if (files == null) { + throw new IllegalArgumentException("Parameter 'files' must not be null."); + } + IMarker marker = MarkerUtility.createMarker(files.mainSourceFile.iFile, (property == null ? 0 + : property.lineNumber), severity, message, parameters); + MarkerUtility.gotoMarker(assemblerEditor, marker); + + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorOpenDeclarationCommandHandler.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorOpenDeclarationCommandHandler.java new file mode 100644 index 00000000..eaaf863d --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorOpenDeclarationCommandHandler.java @@ -0,0 +1,66 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import java.util.ArrayList; +import java.util.List; + +import org.eclipse.core.commands.AbstractHandler; +import org.eclipse.core.commands.ExecutionEvent; +import org.eclipse.core.commands.ExecutionException; +import org.eclipse.jface.text.IDocument; +import org.eclipse.jface.text.ITextSelection; +import org.eclipse.ui.IEditorPart; +import org.eclipse.ui.handlers.HandlerUtil; + +/** + * Handler to open the definition of an identifier or include file via "F3". + * + * @author Peter Dell + * + * @since 1.7.0 + */ +public final class AssemblerEditorOpenDeclarationCommandHandler extends AbstractHandler { + + @Override + public Object execute(ExecutionEvent event) throws ExecutionException { + IEditorPart editor; + editor = HandlerUtil.getActiveEditorChecked(event); + if (!(editor instanceof AssemblerEditor)) { + return null; + } + + AssemblerEditor assemblerEditor; + assemblerEditor = (AssemblerEditor) editor; + ITextSelection textSelection = (ITextSelection) assemblerEditor.getSite().getSelectionProvider().getSelection(); + if (textSelection != null) { + IDocument document = assemblerEditor.getDocumentProvider().getDocument(assemblerEditor.getEditorInput()); + int offset = textSelection.getOffset(); + List<AssemblerHyperlink> hyperlinks = new ArrayList<AssemblerHyperlink>(); + AssemblerHyperlinkDetector.detectHyperlinks(assemblerEditor, document, offset, false, hyperlinks); + if (!hyperlinks.isEmpty()){ + AssemblerHyperlink hyperlink=hyperlinks.get(0); + hyperlink.open(); + } + } + return null; + } + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorOpenFolderCommandHandler.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorOpenFolderCommandHandler.java new file mode 100644 index 00000000..3d874197 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorOpenFolderCommandHandler.java @@ -0,0 +1,70 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import org.eclipse.core.commands.ExecutionEvent; +import org.eclipse.core.commands.ExecutionException; +import org.eclipse.swt.program.Program; + +import com.wudsn.ide.asm.compiler.CompilerFiles; + +/** + * Event handler for the "Open Source Folder" and "Open Output Folder" command. + * + * @author Peter Dell + */ +public final class AssemblerEditorOpenFolderCommandHandler extends + AssemblerEditorFilesCommandHandler { + + public static final String OPEN_SOURCE_FOLDER = "com.wudsn.ide.asm.editor.AssemblerEditorOpenSourceFolderCommand"; + public static final String OPEN_OUTPUT_FOLDER = "com.wudsn.ide.asm.editor.AssemblerEditorOpenOutputFolderCommand"; + + @Override + protected void execute(ExecutionEvent event, + AssemblerEditor assemblerEditor, CompilerFiles files) + throws ExecutionException { + + if (event == null) { + throw new IllegalArgumentException( + "Parameter 'event' must not be null."); + } + if (assemblerEditor == null) { + throw new IllegalArgumentException( + "Parameter 'assemblerEditor' must not be null."); + } + if (files == null) { + throw new IllegalArgumentException( + "Parameter 'files' must not be null."); + } + + String folderPath; + if (event.getCommand().getId().equals(OPEN_SOURCE_FOLDER)) { + folderPath = files.sourceFile.folderPath; + } else if (event.getCommand().getId().equals(OPEN_OUTPUT_FOLDER)) { + folderPath = files.outputFolderPath; + } else { + folderPath = null; + } + if (folderPath != null) { + Program.launch(folderPath); + } + + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorToggleCommentAction.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorToggleCommentAction.java new file mode 100644 index 00000000..9f79d1f5 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerEditorToggleCommentAction.java @@ -0,0 +1,168 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import java.util.List; +import java.util.ResourceBundle; + +import org.eclipse.jface.text.BadLocationException; +import org.eclipse.jface.text.IDocument; +import org.eclipse.jface.text.ITextOperationTarget; +import org.eclipse.jface.text.TextSelection; +import org.eclipse.jface.text.source.SourceViewer; +import org.eclipse.jface.viewers.ISelection; +import org.eclipse.ui.texteditor.TextEditorAction; + +import com.wudsn.ide.asm.AssemblerPlugin; + +/** + * Action to toggle a comment. + * + * @author Peter Dell + * @author Andy Reek + */ +final class AssemblerEditorToggleCommentAction extends TextEditorAction { + + /** + * The owning editor. + */ + private final AssemblerEditor editor; + + /** + * The editor's source viewer. + */ + private final SourceViewer sourceViewer; + + /** + * Creates a new instance. + * + * @param bundle + * The resource bundle, not <code>null</code>. + * @param prefix + * The resource bundle key prefix, not <code>null</code>. + * + * @param editor + * The assembler editor, not <code>null</code>. + * @param sourceViewer + * The assembler editor's source viewer. + * + */ + AssemblerEditorToggleCommentAction(ResourceBundle bundle, String prefix, + AssemblerEditor editor, SourceViewer sourceViewer) { + super(bundle, prefix, editor); + if (editor == null) { + throw new IllegalArgumentException( + "Parameter 'editor' must not be null."); + } + if (sourceViewer == null) { + throw new IllegalArgumentException( + "Parameter 'sourceViewer' must not be null."); + } + this.editor = editor; + this.sourceViewer = sourceViewer; + } + + @Override + public void update() { + setEnabled(canModifyEditor()); + } + + @Override + public void run() { + IDocument document = sourceViewer.getDocument(); + ISelection selection = sourceViewer.getSelection(); + TextSelection textSelection; + if (selection instanceof TextSelection) { + textSelection = (TextSelection) selection; + boolean isCommented = isCommented(document, textSelection); + if (isCommented) { + sourceViewer.doOperation(ITextOperationTarget.STRIP_PREFIX); + } else { + sourceViewer.doOperation(ITextOperationTarget.PREFIX); + } + } + } + + /** + * Checks, if the selection in the given document is commented fully using + * one of the single line comment delimiters. + * + * @param document + * The document, not <code>null</code>. + * @param selection + * The selection, not <code>null</code>. + * + * @return <code>true</code>, if commented. + */ + private boolean isCommented(IDocument document, TextSelection selection) { + if (document == null) { + throw new IllegalArgumentException( + "Parameter 'document' must not be null."); + } + if (selection == null) { + throw new IllegalArgumentException( + "Parameter 'selection' must not be null."); + } + try { + int startLine = selection.getStartLine(); + int endLine = selection.getEndLine(); + for (int line = startLine; line <= endLine; line++) { + int lineOffset = document.getLineOffset(line); + int lineLength = document.getLineLength(line); + String lineText = document.get(lineOffset, lineLength); + + if (!isCommented(lineText)) { + return false; + } + } + } catch (BadLocationException ex) { + AssemblerPlugin.getInstance().logError("Cannot find location", null, ex); + } + + return true; + } + + /** + * Checks, if the line is commented using one of the single line comment + * delimiters. + * + * @param lineText + * The line text, may be empty, not <code>null</code>. + * + * @return <code>true</code>, if commented. + */ + private boolean isCommented(String lineText) { + + if (lineText == null) { + throw new IllegalArgumentException( + "Parameter 'lineText' must not be null."); + } + + List<String> singleLineCommentDeliminters; + singleLineCommentDeliminters = editor.getCompilerDefinition() + .getSyntax().getSingleLineCommentDelimiters(); + for (String delimiter : singleLineCommentDeliminters) { + if (lineText.startsWith(delimiter)) { + return true; + } + } + return false; + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerHyperlink.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerHyperlink.java new file mode 100644 index 00000000..f03bcb54 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerHyperlink.java @@ -0,0 +1,236 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import java.io.File; +import java.net.URI; + +import org.eclipse.core.filesystem.EFS; +import org.eclipse.core.filesystem.IFileStore; +import org.eclipse.core.runtime.CoreException; +import org.eclipse.jface.dialogs.ErrorDialog; +import org.eclipse.jface.dialogs.MessageDialog; +import org.eclipse.jface.text.IRegion; +import org.eclipse.jface.text.hyperlink.IHyperlink; +import org.eclipse.swt.program.Program; +import org.eclipse.ui.IEditorPart; +import org.eclipse.ui.IWorkbenchPage; +import org.eclipse.ui.PartInitException; +import org.eclipse.ui.ide.IDE; + +import com.wudsn.ide.asm.AssemblerPlugin; +import com.wudsn.ide.asm.Texts; +import com.wudsn.ide.base.common.FileUtility; +import com.wudsn.ide.base.common.StringUtility; +import com.wudsn.ide.base.common.TextUtility; + +/** + * Hyperlink implementation for opening source or binary include files. + * + * @author Peter Dell + */ +final class AssemblerHyperlink implements IHyperlink { + + public static final String DEFAULT_EDITOR = "DEFAULT_EDITOR"; + public static final String SYSTEM_EDITOR = "SYSTEM_EDITOR"; + + private IRegion region; + private IWorkbenchPage workbenchPage; + private String absoluteFilePath; + private URI uri; + private String editorId; + private int lineNumber; + private String hyperlinkText; + + /** + * Creates a new hyper link. + * + * @param region + * The region of the text, not <code>null</code>. + * + * @param workbenchPage + * The active workbench page used for the new editor instance, + * not <code>null</code>. + * + * @param absoluteFilePath + * The absolute file path, not <code>null</code>. + * @param uri + * The uri to be opened, not <code>null</code>. + * + * @param editorId + * The id of the editor to be opened, not empty and not + * <code>null</code>. + * + * @param lineNumber + * The liner number to position to, in case the editor is an + * {@link AssemblerEditor}. The line numbers are starting at 1. + * The value 0 indicates that no positioning shall take place. + * + * @param hyperlinkText + * The localized text to display in case there is more than one + * hyperlink for the same location, may be empty, not + * <code>null</code>. + */ + AssemblerHyperlink(IRegion region, IWorkbenchPage workbenchPage, + String absoluteFilePath, URI uri, String editorId, int lineNumber, + String hyperlinkText) { + if (region == null) { + throw new IllegalArgumentException( + "Parameter 'region' must not be null."); + } + if (workbenchPage == null) { + throw new IllegalArgumentException( + "Parameter 'workbenchPage' must not be null."); + } + if (absoluteFilePath == null) { + throw new IllegalArgumentException( + "Parameter 'absoluteFilePath' must not be null."); + } + if (uri == null) { + throw new IllegalArgumentException( + "Parameter 'uri' must not be null."); + } + if (editorId == null) { + throw new IllegalArgumentException( + "Parameter 'editorId' must not be null."); + } + if (StringUtility.isEmpty(editorId)) { + throw new IllegalArgumentException( + "Parameter 'editorId' must not be empty."); + } + if (lineNumber < 0) { + throw new IllegalArgumentException( + "Parameter 'lineNumber' must not be negative. Specified value is " + + lineNumber + "."); + } + + if (hyperlinkText == null) { + throw new IllegalArgumentException( + "Parameter 'hyperlinkText' must not be null."); + } + this.region = region; + this.workbenchPage = workbenchPage; + this.absoluteFilePath = absoluteFilePath; + this.uri = uri; + this.editorId = editorId; + this.lineNumber = lineNumber; + this.hyperlinkText = hyperlinkText; + } + + /** + * {@inheritDoc} + */ + @Override + public IRegion getHyperlinkRegion() { + return region; + } + + /** + * {@inheritDoc} + */ + @Override + public String getTypeLabel() { + return null; + } + + /** + * {@inheritDoc} + */ + @Override + public String getHyperlinkText() { + return hyperlinkText; + } + + /** + * {@inheritDoc} + */ + @Override + public void open() { + if (uri != null) { + File fileToOpen = new File(absoluteFilePath); + if (!fileToOpen.exists()) { + String message = TextUtility.format( + // ERROR: Target file '{0}' not exists. Do you + // want to create the file now? + Texts.ASSEMBLER_HYPERLINK_FILE_NOT_EXISTS, + absoluteFilePath); + boolean result = MessageDialog.openQuestion(workbenchPage + .getWorkbenchWindow().getShell(), + com.wudsn.ide.base.Texts.DIALOG_TITLE, message); + // Try to create the file, if OK was pressed. + if (result) { + try { + FileUtility.writeString(fileToOpen, ""); + } catch (CoreException ex) { + ErrorDialog.openError(workbenchPage + .getWorkbenchWindow().getShell(), + com.wudsn.ide.base.Texts.DIALOG_TITLE, null, ex + .getStatus()); + } + + } else { + return; + } + } + IEditorPart editorPart; + editorPart = null; + if (editorId.equals(DEFAULT_EDITOR)) { + + if (fileToOpen.exists() && fileToOpen.isFile()) { + IFileStore fileStore = EFS.getLocalFileSystem().getStore( + fileToOpen.toURI()); + + try { + editorPart = IDE.openEditorOnFileStore(workbenchPage, + fileStore); + } catch (PartInitException ex) { + + AssemblerPlugin.getInstance().logError( + "Cannot default editor editor for '{0}'.", + new Object[] { uri }, ex); + + } + } else { + // Do something if the file does not exist + } + + } else if (editorId.equals(SYSTEM_EDITOR)) { + Program.launch(absoluteFilePath); + } else { + + try { + + editorPart = IDE.openEditor(workbenchPage, uri, editorId, + true); + } catch (PartInitException ex) { + AssemblerPlugin.getInstance().logError( + "Cannot system editor editor for '{0}'.", + new Object[] { uri }, ex); + } + } + if (editorPart instanceof AssemblerEditor && lineNumber > 0) { + AssemblerEditor assemblerEditor = (AssemblerEditor) editorPart; + assemblerEditor.gotoLine(lineNumber); + } + + uri = null; + } + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerHyperlinkDetector.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerHyperlinkDetector.java new file mode 100644 index 00000000..bcacdcb3 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerHyperlinkDetector.java @@ -0,0 +1,332 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import java.io.File; +import java.net.URI; +import java.util.ArrayList; +import java.util.Iterator; +import java.util.List; + +import org.eclipse.jface.text.BadLocationException; +import org.eclipse.jface.text.IDocument; +import org.eclipse.jface.text.IRegion; +import org.eclipse.jface.text.ITextViewer; +import org.eclipse.jface.text.Region; +import org.eclipse.jface.text.hyperlink.AbstractHyperlinkDetector; +import org.eclipse.jface.text.hyperlink.IHyperlink; +import org.eclipse.ui.IEditorSite; +import org.eclipse.ui.IWorkbenchPage; + +import com.wudsn.ide.asm.Texts; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceFile; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParserFileReference; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParserFileReferenceType; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParserTreeObject; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParserTreeObjectType; +import com.wudsn.ide.asm.compiler.syntax.CompilerSyntax; +import com.wudsn.ide.base.common.NumberUtility; +import com.wudsn.ide.base.common.TextUtility; +import com.wudsn.ide.base.editor.hex.HexEditor; +import com.wudsn.ide.gfx.editor.GraphicsEditor; + +/** + * Hyperlink detector implementation for opening source or binary include files. + * + * @author Peter Dell + */ +public final class AssemblerHyperlinkDetector extends AbstractHyperlinkDetector { + + /** + * Hyperlink detector target as defined in the extension + * "org.eclipse.ui.workbench.texteditor.hyperlinkDetectorTargets". + */ + public static final String TARGET = "com.wudsn.ide.asm.editor.AssemblerHyperlinkDetectorEditorTarget"; + + /** + * Creates a new instance. Called by extension point + * "org.eclipse.ui.workbench.texteditor.hyperlinkDetectors". + * + */ + public AssemblerHyperlinkDetector() { + } + + /** + * Tries to detect hyperlinks for the given region in the given text viewer + * and returns them. + * <p> + * In most of the cases only one hyperlink should be returned. + * </p> + * + * @param textViewer + * the text viewer on which the hover popup should be shown + * @param region + * the text range in the text viewer which is used to detect the + * hyperlinks + * @param canShowMultipleHyperlinks + * tells whether the caller is able to show multiple links to the + * user. If <code>true</code> {@link IHyperlink#open()} should + * directly open the link and not show any additional UI to + * select from a list. If <code>false</code> this method should + * only return one hyperlink which upon {@link IHyperlink#open()} + * may allow to select from a list. + * @return The hyperlinks or <code>null</code> if no hyperlink was detected + */ + @Override + public IHyperlink[] detectHyperlinks(ITextViewer textViewer, IRegion region, boolean canShowMultipleHyperlinks) { + + List<AssemblerHyperlink> hyperlinks; + hyperlinks = new ArrayList<AssemblerHyperlink>(2); + + if (region == null || textViewer == null) + return null; + + AssemblerEditor assemblerEditor; + + assemblerEditor = (AssemblerEditor) getAdapter(AssemblerEditor.class); + if (assemblerEditor == null) { + return null; + } + + IDocument document = textViewer.getDocument(); + + if (document == null) + return null; + + int offset = region.getOffset(); + + detectHyperlinks(assemblerEditor, document, offset, canShowMultipleHyperlinks, hyperlinks); + + if (!hyperlinks.isEmpty()) { + return hyperlinks.toArray(new IHyperlink[hyperlinks.size()]); + } + return null; + + } + + final static void detectHyperlinks(AssemblerEditor assemblerEditor, IDocument document, int offset, + boolean canShowMultipleHyperlinks, List<AssemblerHyperlink> hyperlinks) { + IRegion lineInfo; + int lineNumber; + String line; + try { + lineInfo = document.getLineInformationOfOffset(offset); + lineNumber = document.getLineOfOffset(offset) + 1; + line = document.get(lineInfo.getOffset(), lineInfo.getLength()); + } catch (BadLocationException ex) { + throw new RuntimeException("Region with offset " + offset + " no valid", ex); + } + + int offsetInLine = offset - lineInfo.getOffset(); + + if (offsetInLine >= line.length()) { + return; + } + detectInclude(assemblerEditor, lineInfo, lineNumber, line, offsetInLine, canShowMultipleHyperlinks, hyperlinks); + if (hyperlinks.isEmpty()) { + detectIdentifier(assemblerEditor, lineInfo, lineNumber, line, offsetInLine, canShowMultipleHyperlinks, + hyperlinks); + } + } + + private static void detectInclude(AssemblerEditor assemblerEditor, IRegion lineInfo, int lineNumber, String line, + int offsetInLine, boolean canShowMultipleHyperlinks, List<AssemblerHyperlink> hyperlinks) { + // Try to detect binary or source includes + CompilerSourceParser compilerSourceParser = assemblerEditor.createCompilerSourceParser(); + CompilerSourceParserFileReference fileReference; + fileReference = new CompilerSourceParserFileReference(); + compilerSourceParser.detectFileReference(line, fileReference); + if (fileReference.getType() != CompilerSourceParserFileReferenceType.NONE) { + + CompilerSyntax syntax = compilerSourceParser.getCompilerSyntax(); + int startQuoteOffset = 0; + String filePath = null; + Iterator<String> i = syntax.getStringDelimiters().iterator(); + while (i.hasNext() && filePath == null) { + String quote = i.next(); + startQuoteOffset = line.indexOf(quote, fileReference.getDirectiveEndOffset()); + if (startQuoteOffset == -1) { + continue; + } + int endQuoteOffset = line.indexOf(quote, startQuoteOffset + 1); + if (endQuoteOffset == -1) { + continue; + } + + if (startQuoteOffset < offsetInLine && offsetInLine < endQuoteOffset) { + filePath = line.substring(startQuoteOffset + 1, endQuoteOffset); + } else { + continue; + } + } + if (filePath == null) { + return; + } + + // Perform resolution of relative paths and compiler specific + // default extension. + File currentDirectory = assemblerEditor.getCurrentDirectory(); + String absoluteFilePath = compilerSourceParser.getIncludeAbsoluteFilePath(fileReference.getType(), + currentDirectory, filePath); + if (absoluteFilePath == null) { + return; + } + + URI uri; + uri = new File(absoluteFilePath).toURI(); + + IRegion linkRegion = new Region(lineInfo.getOffset() + startQuoteOffset + 1, filePath.length()); + + IEditorSite site = assemblerEditor.getEditorSite(); + if (site == null) { + return; + } + IWorkbenchPage workbenchPage = site.getWorkbenchWindow().getActivePage(); + + switch (fileReference.getType()) { + case CompilerSourceParserFileReferenceType.SOURCE: + hyperlinks + .add(new AssemblerHyperlink(linkRegion, workbenchPage, absoluteFilePath, uri, assemblerEditor + .getClass().getName(), 0, + Texts.ASSEMBLER_HYPERLINK_DETECTOR_OPEN_SOURCE_WITH_ASSEMBLER_EDITOR)); + break; + case CompilerSourceParserFileReferenceType.BINARY: + hyperlinks.add(new AssemblerHyperlink(linkRegion, workbenchPage, absoluteFilePath, uri, HexEditor.ID, + 0, Texts.ASSEMBLER_HYPERLINK_DETECTOR_OPEN_BINARY_WITH_HEX_EDITOR)); + if (canShowMultipleHyperlinks) { + hyperlinks.add(new AssemblerHyperlink(linkRegion, workbenchPage, absoluteFilePath, uri, + GraphicsEditor.ID, 0, Texts.ASSEMBLER_HYPERLINK_DETECTOR_OPEN_BINARY_WITH_GRAPHICS_EDITOR)); + hyperlinks.add(new AssemblerHyperlink(linkRegion, workbenchPage, absoluteFilePath, uri, + AssemblerHyperlink.DEFAULT_EDITOR, 0, + Texts.ASSEMBLER_HYPERLINK_DETECTOR_OPEN_BINARY_WITH_DEFAULT_EDITOR)); + hyperlinks.add(new AssemblerHyperlink(linkRegion, workbenchPage, absoluteFilePath, uri, + AssemblerHyperlink.SYSTEM_EDITOR, 0, + Texts.ASSEMBLER_HYPERLINK_DETECTOR_OPEN_BINARY_WITH_SYSTEM_EDITOR)); + } + break; + default: + throw new IllegalStateException("Unknown include type " + fileReference.getType()); + + } + } + } + + private static void detectIdentifier(AssemblerEditor assemblerEditor, IRegion lineInfo, int lineNumber, + String line, int offsetInLine, boolean canShowMultipleHyperlinks, List<AssemblerHyperlink> hyperlinks) { + + CompilerSyntax compilerSyntax = assemblerEditor.createCompilerSourceParser().getCompilerSyntax(); + + int startIdentifierOffset = offsetInLine; + int endIdentifierOffset = offsetInLine; + + if (!compilerSyntax.isIdentifierCharacter(line.charAt(startIdentifierOffset))) { + return; + } + + while (startIdentifierOffset > 0 + && compilerSyntax.isIdentifierCharacter(line.charAt(startIdentifierOffset - 1))) { + startIdentifierOffset--; + } + while (endIdentifierOffset < line.length() + && compilerSyntax.isIdentifierCharacter(line.charAt(endIdentifierOffset))) { + // If we find an identifier separator character when moving right, + // we stop if we pass the cursor position. + if (endIdentifierOffset > offsetInLine + && compilerSyntax.isIdentifierSeparatorCharacter(line.charAt(endIdentifierOffset))) { + break; + } + endIdentifierOffset++; + } + String identifier = line.substring(startIdentifierOffset, endIdentifierOffset); + CompilerSourceFile compilerSourceFile = assemblerEditor.getCompilerSourceFile(); + if (compilerSourceFile == null) { + return; + } + + List<CompilerSourceParserTreeObject> foundElements; + foundElements = compilerSourceFile.getIdentifierDefinitionElements(identifier); + if (foundElements.isEmpty()) { + return; + } + + IEditorSite site = assemblerEditor.getEditorSite(); + if (site == null) { + return; + } + IWorkbenchPage workbenchPage = site.getWorkbenchWindow().getActivePage(); + + int size; + if (canShowMultipleHyperlinks) { + size = foundElements.size(); + } else { + size = 1; + } + for (int i = 0; i < size; i++) { + CompilerSourceParserTreeObject element = foundElements.get(i); + + String absoluteFilePath; + String fileName; + File documentFile = element.getCompilerSourceFile().getDocumentFile(); + if (documentFile != null) { + absoluteFilePath = documentFile.getPath(); + fileName = documentFile.getName(); + } else { + absoluteFilePath = ""; + fileName = ""; + } + int elementLineNumber; + try { + elementLineNumber = element.getCompilerSourceFile().getDocument() + .getLineOfOffset(element.getStartOffset()) + 1; + } catch (BadLocationException ex) { + continue; + } + + // Ignore if the found element is the start element only. + File currentFile = assemblerEditor.getCurrentFile(); + boolean inSameFile = foundElements.size() == 1 && currentFile != null + && currentFile.getPath().equals(absoluteFilePath); + + if (inSameFile && lineNumber == elementLineNumber) { + continue; + } + + URI uri; + uri = new File(absoluteFilePath).toURI(); + + IRegion linkRegion = new Region(lineInfo.getOffset() + startIdentifierOffset, identifier.length()); + + String hyperlinkText; + if (inSameFile) { + hyperlinkText = TextUtility.format(Texts.ASSEMBLER_HYPERLINK_DETECTOR_OPEN_IDENTIFIER, + CompilerSourceParserTreeObjectType.getText(element.getType()), element.getCompoundName(), + NumberUtility.getLongValueDecimalString(elementLineNumber)); + } else { + hyperlinkText = TextUtility.format(Texts.ASSEMBLER_HYPERLINK_DETECTOR_OPEN_IDENTIFIER_IN_INCLUDE, + CompilerSourceParserTreeObjectType.getText(element.getType()), element.getCompoundName(), + NumberUtility.getLongValueDecimalString(elementLineNumber), fileName); + } + hyperlinks.add(new AssemblerHyperlink(linkRegion, workbenchPage, absoluteFilePath, uri, assemblerEditor + .getClass().getName(), elementLineNumber, hyperlinkText)); + } + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerInstructionCompletionProposal.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerInstructionCompletionProposal.java new file mode 100644 index 00000000..5e5b08a9 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerInstructionCompletionProposal.java @@ -0,0 +1,167 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import org.eclipse.core.runtime.Assert; +import org.eclipse.jface.text.BadLocationException; +import org.eclipse.jface.text.IDocument; +import org.eclipse.jface.text.contentassist.ICompletionProposal; +import org.eclipse.jface.text.contentassist.ICompletionProposalExtension6; +import org.eclipse.jface.text.contentassist.IContextInformation; +import org.eclipse.jface.viewers.StyledString; +import org.eclipse.swt.graphics.Image; +import org.eclipse.swt.graphics.Point; + +/** + * The standard implementation of the <code>ICompletionProposal</code> + * interface. + * + * @author Peter Dell + */ +final class AssemblerInstructionCompletionProposal implements + ICompletionProposal, ICompletionProposalExtension6 { + + /** The string to be displayed in the completion proposal popup. */ + private String displayString; + private StyledString styledDisplayString; + + /** The replacement string. */ + private String replacementString; + /** The replacement offset. */ + private int replacementOffset; + /** The replacement length. */ + private int replacementLength; + /** The offset of the cursor after applying the replacement. */ + private int cursorOffset; + + /** The image to be displayed in the completion proposal popup. */ + private Image image; + /** The context information of this proposal. */ + private IContextInformation contextInformation; + + + /** + * Creates a new completion proposal. All fields are initialized based on + * the provided information. + * + * @param replacementString + * The actual string to be inserted into the document. + * @param replacementOffset + * The offset of the text to be replaced. + * @param replacementLength + * The length of the text to be replaced. + * @param cursorOffset + * The offset of the cursor after applying the replacement. + * @param image + * The image to display for this proposal. + * @param displayString + * The string to be displayed for the proposal. + * @param styledDisplayString + * The styles display string for the proposal. + * @param contextInformation + * The context information associated with this proposal. + */ + AssemblerInstructionCompletionProposal(String replacementString, + int replacementOffset, int replacementLength, int cursorOffset, Image image, + String displayString, StyledString styledDisplayString, + IContextInformation contextInformation) { + Assert.isNotNull(replacementString); + Assert.isNotNull(displayString); + Assert.isTrue(replacementOffset >= 0); + Assert.isTrue(replacementLength >= 0); + Assert.isTrue(cursorOffset >= 0); + + this.replacementString = replacementString; + this.replacementOffset = replacementOffset; + this.replacementLength = replacementLength; + this.cursorOffset=cursorOffset; + this.image = image; + this.displayString = displayString; + this.styledDisplayString = styledDisplayString; + this.contextInformation = contextInformation; + } + + /* + * @see ICompletionProposal#apply(IDocument) + */ + @Override + public void apply(IDocument document) { + try { + document.replace(replacementOffset, replacementLength, + replacementString); + } catch (BadLocationException ex) { + throw new RuntimeException("Replacement offset " + + replacementOffset + " no valid", ex); + + } + } + + /* + * @see ICompletionProposal#getSelection(IDocument) + */ + @Override + public Point getSelection(IDocument document) { + return new Point(cursorOffset, 0); + } + + /* + * @see ICompletionProposal#getContextInformation() + */ + @Override + public IContextInformation getContextInformation() { + return contextInformation; + } + + /* + * @see ICompletionProposal#getImage() + */ + @Override + public Image getImage() { + return image; + } + + /* + * @see ICompletionProposal#getDisplayString() + */ + @Override + public String getDisplayString() { + if (displayString != null) + return displayString; + return replacementString; + } + + /* + * @see ICompletionProposalExtension6#getStyledDisplayString() + */ + @Override + public StyledString getStyledDisplayString() { + + return styledDisplayString; + } + + /* + * @see ICompletionProposal#getAdditionalProposalInfo() + */ + @Override + public String getAdditionalProposalInfo() { + return ""; + } + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerReconcilingStategy.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerReconcilingStategy.java new file mode 100644 index 00000000..0ddbbd5b --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerReconcilingStategy.java @@ -0,0 +1,109 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import org.eclipse.core.runtime.IProgressMonitor; +import org.eclipse.jface.text.IDocument; +import org.eclipse.jface.text.IRegion; +import org.eclipse.jface.text.reconciler.DirtyRegion; +import org.eclipse.jface.text.reconciler.IReconcilingStrategy; +import org.eclipse.jface.text.reconciler.IReconcilingStrategyExtension; + +/** + * The reconciling strategy for the AssemblerEditor. Builds the folding + * structure for folding and notifies the editor. + * + * @author Peter Dell + * @author Andy Reek + */ +final class AssemblerReconcilingStategy implements IReconcilingStrategy, + IReconcilingStrategyExtension { + + private final AssemblerEditor editor; + private IDocument document; + + /** + * Creates a new instance. Called by + * {@link AssemblerSourceViewerConfiguration#getReconciler(org.eclipse.jface.text.source.ISourceViewer)} + * . + * + * * @param editor The underlying assembler editor, not <code>null</code>. + */ + AssemblerReconcilingStategy(AssemblerEditor editor) { + if (editor == null) { + throw new IllegalArgumentException( + "Parameter 'editor' must not be null."); + } + this.editor = editor; + } + + /** + * {@inheritDoc} + */ + @Override + public void setDocument(IDocument document) { + this.document = document; + } + + /** + * {@inheritDoc} + */ + @Override + public final void setProgressMonitor(final IProgressMonitor monitor) { + // Not needed + } + + /** + * {@inheritDoc} + */ + @Override + public void initialReconcile() { + parse(); + } + + /** + * {@inheritDoc} + */ + @Override + public void reconcile(DirtyRegion dirtyRegion, IRegion subRegion) { + parse(); + } + + /** + * {@inheritDoc} + */ + @Override + public void reconcile(IRegion partition) { + parse(); + } + + /** + * Parses the current document for the content outline and the folding + * structure. + */ + private void parse() { + if (document == null) { + return; + } + + editor.updateContentOutlinePage(); + + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerRuleBasedScanner.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerRuleBasedScanner.java new file mode 100644 index 00000000..4f0a986f --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerRuleBasedScanner.java @@ -0,0 +1,102 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import java.util.Set; + +import org.eclipse.jface.text.TextAttribute; +import org.eclipse.jface.text.rules.RuleBasedScanner; +import org.eclipse.jface.text.rules.Token; + +import com.wudsn.ide.asm.AssemblerPlugin; +import com.wudsn.ide.asm.preferences.AssemblerPreferences; +import com.wudsn.ide.asm.preferences.TextAttributeConverter; + +/** + * Rule based scanner for comments and strings. + * + * @author Peter Dell + * @author Daniel Mitte + */ +final class AssemblerRuleBasedScanner extends RuleBasedScanner { + + /** Default Token for the text attributes * */ + private Token defaultToken; + + /** Key for preference store * */ + private String preferencesKey; + + /** + * Creates a new instance. Called by + * {@link AssemblerSourceViewerConfiguration}. + * + * @param preferencesKey + * The preference key to listen to for text attribute changes, + * not <code>null</code>.. + */ + AssemblerRuleBasedScanner(String preferencesKey) { + + if (preferencesKey == null) { + throw new IllegalArgumentException("Parameter 'preferencesKey' must not be null."); + } + + this.preferencesKey = preferencesKey; + + AssemblerPreferences preferences = AssemblerPlugin.getInstance().getPreferences(); + defaultToken = new Token(preferences.getEditorTextAttribute(preferencesKey)); + + super.setDefaultReturnToken(defaultToken); + } + + /** + * Dispose UI resources. + */ + final void dispose() { + TextAttributeConverter.dispose((TextAttribute) defaultToken.getData()); + } + + /** + * Update the token based on the preferences. Called by + * {@link AssemblerSourceViewerConfiguration}. + * + * @param preferences + * The preferences, not <code>null</code>. + * @param changedPropertyNames + * The set of changed property names, not <code>null</code>. + * + * @return <code>true</code> If the editor has to be refreshed. + */ + final boolean preferencesChanged(AssemblerPreferences preferences, Set<String> changedPropertyNames) { + if (preferences == null) { + throw new IllegalArgumentException("Parameter 'preferences' must not be null."); + } + if (changedPropertyNames == null) { + throw new IllegalArgumentException("Parameter 'changedPropertyNames' must not be null."); + } + boolean refresh = false; + if (changedPropertyNames.contains(preferencesKey)) { + TextAttributeConverter.dispose((TextAttribute) defaultToken.getData()); + defaultToken.setData(preferences.getEditorTextAttribute(preferencesKey)); + refresh = true; + } + return refresh; + + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerSourceScanner.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerSourceScanner.java new file mode 100644 index 00000000..49dbe314 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerSourceScanner.java @@ -0,0 +1,516 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import java.util.ArrayList; +import java.util.List; +import java.util.Map; +import java.util.Set; +import java.util.TreeMap; + +import org.eclipse.jface.text.TextAttribute; +import org.eclipse.jface.text.rules.ICharacterScanner; +import org.eclipse.jface.text.rules.IRule; +import org.eclipse.jface.text.rules.IToken; +import org.eclipse.jface.text.rules.RuleBasedScanner; +import org.eclipse.jface.text.rules.Token; + +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParser; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParserTreeObject; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParserTreeObjectType; +import com.wudsn.ide.asm.compiler.syntax.CompilerSyntax; +import com.wudsn.ide.asm.compiler.syntax.Directive; +import com.wudsn.ide.asm.compiler.syntax.Instruction; +import com.wudsn.ide.asm.compiler.syntax.InstructionSet; +import com.wudsn.ide.asm.compiler.syntax.InstructionType; +import com.wudsn.ide.asm.compiler.syntax.Opcode; +import com.wudsn.ide.asm.preferences.AssemblerPreferences; +import com.wudsn.ide.asm.preferences.AssemblerPreferencesConstants; +import com.wudsn.ide.asm.preferences.TextAttributeConverter; + +/** + * A rule based scanner for instructions. + * + * @author Peter Dell + * @author Andy Reek + */ +final class AssemblerSourceScanner extends RuleBasedScanner { + + private final class AssemblerWordRule implements IRule { + public final class State { + public CompilerSourceParser compilerSourceParser; + + public CompilerSyntax compilerSyntax; + public InstructionSet instructionSet; + + public boolean instructionsCaseSensitive; + public Map<String, IToken> instructionWordTokens; + + public boolean identifiersCaseSensitive; + public Map<String, IToken> identifierWordTokens; + + public State() { + instructionWordTokens = new TreeMap<String, IToken>(); + identifierWordTokens = new TreeMap<String, IToken>(); + } + + public State createDeepCopy() { + State result = new State(); + result.compilerSourceParser = compilerSourceParser; + result.compilerSyntax = compilerSyntax; + result.instructionSet = instructionSet; + result.instructionsCaseSensitive = instructionsCaseSensitive; + result.instructionWordTokens = instructionWordTokens; + result.identifiersCaseSensitive = identifiersCaseSensitive; + result.identifierWordTokens = identifierWordTokens; + return result; + } + } + + // State of the AssemblerWordRule instance. + State state; + + /** Buffer used for pattern detection in evaluate() only. */ + private StringBuilder instructionBuffer = new StringBuilder(); + private StringBuilder identifierBuffer = new StringBuilder(); + private StringBuilder numberBuffer = new StringBuilder(); + + public AssemblerWordRule() { + + } + + public void setCompilerSourceParser(CompilerSourceParser compilerSourceParser) { + if (compilerSourceParser == null) { + throw new IllegalArgumentException("Parameter 'compilerSourceParser' must not be null."); + } + + State state = new State(); + state.compilerSourceParser = compilerSourceParser; + state.instructionSet = compilerSourceParser.getInstructionSet(); + state.compilerSyntax = compilerSourceParser.getCompilerSyntax(); + + state.instructionsCaseSensitive = state.compilerSyntax.areInstructionsCaseSensitive(); + state.identifiersCaseSensitive = state.compilerSyntax.areIdentifiersCaseSensitive(); + + synchronized (this) { + this.state = state; + } + } + + public void setInstructions() { + synchronized (this) { + state.instructionWordTokens.clear(); + + state.instructionsCaseSensitive = state.compilerSyntax.areInstructionsCaseSensitive(); + state.instructionSet = state.compilerSourceParser.getInstructionSet(); + + List<Instruction> instructions = state.instructionSet.getInstructions(); + + // Map with lower case name and corresponding token. + for (Instruction instruction : instructions) { + IToken token; + if (instruction instanceof Directive) { + token = directiveToken; + } else if (instruction instanceof Opcode) { + + Opcode opcode = (Opcode) instruction; + + switch (opcode.getType()) { + + case InstructionType.LEGAL_OPCODE: + token = legalOpcodeToken; + break; + case InstructionType.ILLEGAL_OPCODE: + + token = illegalOpcodeToken; + break; + case InstructionType.PSEUDO_OPCODE: + token = pseudoOpcodeToken; + break; + default: + throw new IllegalStateException("Unknown opcode type " + opcode.getType() + "."); + + } + } else { + throw new IllegalStateException("Unknown instruction type " + instruction.toString() + "."); + + } + // Case insensitive word rules expect upper case words. + if (state.instructionsCaseSensitive) { + state.instructionWordTokens.put(instruction.getName(), token); + } else { + state.instructionWordTokens.put(instruction.getUpperCaseName(), token); + } + } + } + + } + + /** + * Update the list of identifiers to be highlighted + * + * @param identifiers + * The list of identifiers, not <code>null</code>. + */ + public void setIdentifiers(List<CompilerSourceParserTreeObject> identifiers) { + if (identifiers == null) { + throw new IllegalArgumentException("Parameter 'identifiers' must not be null."); + } + synchronized (this) { + + for (CompilerSourceParserTreeObject element : identifiers) { + IToken token; + switch (element.getType()) { + case CompilerSourceParserTreeObjectType.EQUATE_DEFINITION: + token = equateIdentifierToken; + break; + case CompilerSourceParserTreeObjectType.LABEL_DEFINITION: + token = labelIdentifierToken; + break; + case CompilerSourceParserTreeObjectType.ENUM_DEFINITION_SECTION: + token = enumDefinitionSectionIdentifierToken; + break; + case CompilerSourceParserTreeObjectType.STRUCTURE_DEFINITION_SECTION: + token = structureDefinitionSectionIdentifierToken; + break; + case CompilerSourceParserTreeObjectType.LOCAL_SECTION: + token = localSectionIdentifierToken; + break; + case CompilerSourceParserTreeObjectType.MACRO_DEFINITION_SECTION: + token = macroDefinitionSectionIdentifierToken; + break; + case CompilerSourceParserTreeObjectType.PROCEDURE_DEFINITION_SECTION: + token = procedureDefinitionSectionIdentifierToken; + break; + + default: + throw new RuntimeException("Unexpected identifier element type " + element.getType() + " - " + + element.getTreePath() + "."); + } + if (element.getDescription().startsWith("@style=(")) { + String value = element.getDescription().substring(8); + int index = value.indexOf(")"); + if (index > 0) { + value = value.substring(0, index); + TextAttribute textAttribute = TextAttributeConverter.fromString(value); + token = new Token(textAttribute); + } + } + if (state.identifiersCaseSensitive) { + state.identifierWordTokens.put(element.getName(), token); + } else { + state.identifierWordTokens.put(element.getName().toUpperCase(), token); + + } + } + } + System.out.println(""+this+":"+state.identifierWordTokens.size()); + } + + /* + * @see IRule#evaluate(ICharacterScanner) + */ + @Override + public IToken evaluate(ICharacterScanner scanner) { + + // Create a local copy to prevent synchronization issues. + State localState; + synchronized (this) { + localState = state.createDeepCopy(); + } + + int c = scanner.read(); + boolean instructionStartCharacter = localState.instructionSet.isInstructionStartCharacter((char) c); + boolean identifierStartCharacter = localState.compilerSyntax.isIdentifierStartCharacter((char) c); + boolean numberStartCharacter = localState.compilerSyntax.isNumberStartCharacter((char) c); + if (c != ICharacterScanner.EOF + && (instructionStartCharacter || identifierStartCharacter || numberStartCharacter)) { + + instructionBuffer.setLength(0); + identifierBuffer.setLength(0); + numberBuffer.setLength(0); + int charactersRead = 0; + boolean instructionPartCharacter = instructionStartCharacter; + boolean identifierPartCharacter = identifierStartCharacter; + boolean numberPartCharacter = numberStartCharacter; + do { + charactersRead++; + if (instructionPartCharacter) { + instructionBuffer.append((char) c); + } + if (identifierPartCharacter) { + identifierBuffer.append((char) c); + } + if (numberPartCharacter) { + numberBuffer.append((char) c); + } + c = scanner.read(); + instructionPartCharacter = instructionPartCharacter + && localState.instructionSet.isInstructionPartCharacter((char) c); + identifierPartCharacter = identifierPartCharacter + && (localState.compilerSyntax.isIdentifierPartCharacter((char) c)); + numberPartCharacter = numberPartCharacter + && localState.compilerSyntax.isNumberPartCharacter((char) c); + + } while (c != ICharacterScanner.EOF + && (instructionPartCharacter || identifierPartCharacter || numberPartCharacter)); + scanner.unread(); + + String instructionString = instructionBuffer.toString(); + String identifierString = identifierBuffer.toString(); + String numberString = numberBuffer.toString(); + // System.out.println(instructionString + "/" + identifierString + // + "/" + numberString); + + // If case-insensitive, convert to upper case before + // accessing the map + if (!localState.instructionsCaseSensitive) { + instructionString = instructionString.toUpperCase(); + } + + IToken instructionToken = localState.instructionWordTokens.get(instructionString); + + // Anything found at all? + if (instructionToken == null && identifierString.length() == 0 && numberString.length() == 0) { + unreadBuffer(scanner, charactersRead); + return Token.UNDEFINED; + } + + // If the identifier string is longer, use it. + IToken token; + if (instructionToken == null || identifierString.length() > instructionString.length()) { + if (identifierString.length() >= numberString.length()) { + if (identifierString.length() == 0) { + return Token.UNDEFINED; + } + if (!localState.identifiersCaseSensitive) { + identifierString = identifierString.toUpperCase(); + } + token = localState.identifierWordTokens.get(identifierString); + + // Consume the next separator if there is one. + if (localState.compilerSyntax.isIdentifierSeparatorCharacter((char) c)) { + charactersRead--; + } + unreadBuffer(scanner, charactersRead - identifierString.length()); + if (token == null) { + token = Token.UNDEFINED; + } + return token; + } + unreadBuffer(scanner, charactersRead - numberString.length()); + return numberToken; + + } + if (instructionString.length() >= numberString.length()) { + unreadBuffer(scanner, charactersRead - instructionString.length()); + return instructionToken; + } else if (numberString.length() > 0) { + return numberToken; + } + + return Token.UNDEFINED; + } + + scanner.unread(); + return Token.UNDEFINED; + + } + + /** + * Returns the specified number of characters to the scanner. + * + * @param scanner + * The scanner to be used, not <code>null</code>. + * @param count + * The count. If the count is 0 or negative, no characters + * will be returned. + */ + private void unreadBuffer(ICharacterScanner scanner, int count) { + for (int i = 0; i < count; i++) { + scanner.unread(); + } + } + } + + private AssemblerEditor editor; + private Map<String, Token> tokens; + + // Numbers + IToken numberToken; + + // Instructions. + IToken directiveToken; + IToken legalOpcodeToken; + IToken illegalOpcodeToken; + IToken pseudoOpcodeToken; + + // Identifiers. + IToken equateIdentifierToken; + IToken labelIdentifierToken; + IToken enumDefinitionSectionIdentifierToken; + IToken structureDefinitionSectionIdentifierToken; + IToken localSectionIdentifierToken; + IToken macroDefinitionSectionIdentifierToken; + IToken procedureDefinitionSectionIdentifierToken; + + // Word rule + private AssemblerWordRule wordRule; + + /** + * Creates a new instance. Called by the + * {@link AssemblerSourceViewerConfiguration}. + * + * @param editor + * The underlying AssemblerEditor for the code scanner, not + * <code>null</code>. + */ + AssemblerSourceScanner(AssemblerEditor editor) { + if (editor == null) { + throw new IllegalArgumentException("Parameter 'editor' must not be null."); + } + this.editor = editor; + this.tokens = new TreeMap<String, Token>(); + + createTokens(); + createRules(); + } + + /** + * The token are stable over the life time of the editor, whereas the rules + * may change if the preferences are changed. + */ + private void createTokens() { + + // Numbers + numberToken = createToken(AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_NUMBER); + + // Instructions + directiveToken = createToken(AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_DIRECTVE); + legalOpcodeToken = createToken(AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_OPCODE_LEGAL); + illegalOpcodeToken = createToken(AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_OPCODE_ILLEGAL); + pseudoOpcodeToken = createToken(AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_OPCODE_PSEUDO); + + // Identifiers + equateIdentifierToken = createToken(AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_EQUATE); + labelIdentifierToken = createToken(AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_LABEL); + enumDefinitionSectionIdentifierToken = createToken(AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_ENUM_DEFINITION_SECTION); + structureDefinitionSectionIdentifierToken = createToken(AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_STRUCTURE_DEFINITION_SECTION); + localSectionIdentifierToken = createToken(AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_LOCAL_SECTION); + macroDefinitionSectionIdentifierToken = createToken(AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_MACRO_DEFINITION_SECTION); + procedureDefinitionSectionIdentifierToken = createToken(AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_PROCEDURE_DEFINITION_SECTION); + + } + + /** + * Creates a token bound to a text attribute name. + * + * @param textAttributeName + * The text attribute name, not empty and not <code>null</code>. + * @return The new token, not <code>null</code>. + */ + private IToken createToken(String textAttributeName) { + if (textAttributeName == null) { + throw new IllegalArgumentException("Parameter 'textAttributeName' must not be null."); + } + AssemblerPreferences preferences; + Token token; + preferences = editor.getPlugin().getPreferences(); + token = new Token(preferences.getEditorTextAttribute(textAttributeName)); + tokens.put(textAttributeName, token); + return token; + } + + /** + * Creates and sets the rules based on the the compiler syntax. + */ + private void createRules() { + + // Instructions, identifiers and numbers. + wordRule = new AssemblerWordRule(); + List<IRule> rules = new ArrayList<IRule>(4); + rules.add(wordRule); + setRules(rules.toArray(new IRule[rules.size()])); + + CompilerSourceParser compilerSourceParser = editor.createCompilerSourceParser(); + wordRule.setCompilerSourceParser(compilerSourceParser); + wordRule.setInstructions(); + } + + /** + * Update the list of identifiers to be highlighted + * + * @param identifiers + * The list of identifiers, not <code>null</code>. + */ + final void setIdentifiers(List<CompilerSourceParserTreeObject> identifiers) { + if (identifiers == null) { + throw new IllegalArgumentException("Parameter 'identifiers' must not be null."); + } + wordRule.setIdentifiers(identifiers); + } + + /** + * Dispose UI resources. + */ + final void dispose() { + for (Token token : tokens.values()) { + TextAttributeConverter.dispose((TextAttribute) token.getData()); + } + } + + /** + * Update the token based on the preferences. Called by + * {@link AssemblerSourceViewerConfiguration}. + * + * @param preferences + * The preferences, not <code>null</code>. + * @param changedPropertyNames + * The set of changed property names, not <code>null</code>. + * + * @return <code>true</code> If the editor has to be refreshed. + */ + final boolean preferencesChanged(AssemblerPreferences preferences, Set<String> changedPropertyNames) { + if (preferences == null) { + throw new IllegalArgumentException("Parameter 'preferences' must not be null."); + } + if (changedPropertyNames == null) { + throw new IllegalArgumentException("Parameter 'changedPropertyNames' must not be null."); + } + boolean refresh = false; + for (String propertyName : changedPropertyNames) { + Token token = tokens.get(propertyName); + if (token != null) { + TextAttributeConverter.dispose((TextAttribute) token.getData()); + token.setData(preferences.getEditorTextAttribute(propertyName)); + refresh = true; + + } else if (AssemblerPreferencesConstants.isCompilerCPUName(propertyName)) { + CompilerSourceParser compilerSourceParser = editor.createCompilerSourceParser(); + wordRule.setCompilerSourceParser(compilerSourceParser); + wordRule.setInstructions(); + refresh = true; + } + } + return refresh; + + } + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerSourceViewerConfiguration.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerSourceViewerConfiguration.java new file mode 100644 index 00000000..261c3bb4 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/AssemblerSourceViewerConfiguration.java @@ -0,0 +1,225 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import java.util.Map; +import java.util.Set; + +import org.eclipse.core.runtime.IAdaptable; +import org.eclipse.core.runtime.NullProgressMonitor; +import org.eclipse.jface.preference.IPreferenceStore; +import org.eclipse.jface.text.IDocument; +import org.eclipse.jface.text.contentassist.ContentAssistant; +import org.eclipse.jface.text.contentassist.IContentAssistant; +import org.eclipse.jface.text.presentation.IPresentationReconciler; +import org.eclipse.jface.text.presentation.PresentationReconciler; +import org.eclipse.jface.text.reconciler.IReconciler; +import org.eclipse.jface.text.reconciler.IReconcilingStrategy; +import org.eclipse.jface.text.reconciler.MonoReconciler; +import org.eclipse.jface.text.rules.DefaultDamagerRepairer; +import org.eclipse.jface.text.source.ISourceViewer; +import org.eclipse.ui.editors.text.TextSourceViewerConfiguration; + +import com.wudsn.ide.asm.compiler.parser.CompilerSourcePartitionScanner; +import com.wudsn.ide.asm.preferences.AssemblerPreferences; +import com.wudsn.ide.asm.preferences.AssemblerPreferencesChangeListener; +import com.wudsn.ide.asm.preferences.AssemblerPreferencesConstants; + +/** + * Source configuration for the assembler editor. Provides syntax highlighting. + * + * Created and disposed by {@link AssemblerEditor}. + * + * @author Peter Dell + * @author Andy Reek + */ +final class AssemblerSourceViewerConfiguration extends TextSourceViewerConfiguration implements + AssemblerPreferencesChangeListener { + + /** + * The underlying assembler editor. + */ + private AssemblerEditor editor; + + /** + * Rule scanner for single line comments. + */ + private AssemblerRuleBasedScanner commentSingleScanner; + + /** + * Rule scanner for multiple lines comments. + */ + private AssemblerRuleBasedScanner commentMultipleScanner; + + /** + * Rule scanner for strings. + */ + private AssemblerRuleBasedScanner stringScanner; + + /** + * Rule scanner for assembler instructions. + */ + private AssemblerSourceScanner instructionScanner; + + /** + * Creates a new instance. Called by + * {@link AssemblerEditor#initializeEditor()}. + * + * @param editor + * The underlying assembler editor, not <code>null</code>. + * + * @param preferenceStore + * The preferences store, not <code>null</code>. + */ + AssemblerSourceViewerConfiguration(AssemblerEditor editor, IPreferenceStore preferenceStore) { + super(preferenceStore); + if (editor == null) { + throw new IllegalArgumentException("Parameter 'editor' must not be null."); + } + this.editor = editor; + editor.getPlugin().addPreferencesChangeListener(this); + + } + + /** + * Called by + * {@link AssemblerEditor#updateIdentifiers(com.wudsn.ide.asm.compiler.parser.CompilerSourceFile)} + * + * @return The instruction scanner, not <code>null</code>. + */ + final AssemblerSourceScanner getAssemblerInstructionScanner() { + if (instructionScanner == null) { + throw new IllegalStateException("Instruction scanner not yet created"); + } + return instructionScanner; + } + + /** + * Remove all rule scanners from property change listener. Used by + * {@link AssemblerEditor#dispose()}. + */ + final void dispose() { + if (commentSingleScanner != null) { + commentSingleScanner.dispose(); + commentSingleScanner = null; + } + if (commentMultipleScanner != null) { + commentMultipleScanner.dispose(); + commentMultipleScanner = null; + } + if (stringScanner != null) { + stringScanner.dispose(); + stringScanner = null; + } + if (instructionScanner != null) { + instructionScanner.dispose(); + instructionScanner = null; + } + editor.getPlugin().removePreferencesChangeListener(this); + } + + @Override + public void preferencesChanged(AssemblerPreferences preferences, Set<String> changedPropertyNames) { + if (preferences == null) { + throw new IllegalArgumentException("Parameter 'preferences' must not be null."); + } + if (changedPropertyNames == null) { + throw new IllegalArgumentException("Parameter 'changedPropertyNames' must not be null."); + } + boolean refresh = false; + refresh |= commentSingleScanner.preferencesChanged(preferences, changedPropertyNames); + refresh |= commentMultipleScanner.preferencesChanged(preferences, changedPropertyNames); + refresh |= stringScanner.preferencesChanged(preferences, changedPropertyNames); + refresh |= instructionScanner.preferencesChanged(preferences, changedPropertyNames); + if (refresh) { + editor.refreshSourceViewer(); + } + } + + @Override + public IContentAssistant getContentAssistant(ISourceViewer sourceViewer) { + ContentAssistant assistant = new ContentAssistant(); + assistant + .setContentAssistProcessor(new AssemblerContentAssistProcessor(editor), IDocument.DEFAULT_CONTENT_TYPE); + assistant.setContextInformationPopupOrientation(IContentAssistant.CONTEXT_INFO_ABOVE); + assistant.enableAutoActivation(true); + assistant.enableAutoInsert(true); + assistant.setAutoActivationDelay(500); + assistant.enableColoredLabels(true); + + return assistant; + } + + @Override + public IPresentationReconciler getPresentationReconciler(ISourceViewer sourceViewer) { + if (sourceViewer == null) { + throw new IllegalArgumentException("Parameter 'sourceViewer' must not be null."); + } + PresentationReconciler reconciler = new PresentationReconciler(); + DefaultDamagerRepairer dr; + + commentSingleScanner = new AssemblerRuleBasedScanner( + AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_COMMENT); + dr = new DefaultDamagerRepairer(commentSingleScanner); + reconciler.setDamager(dr, CompilerSourcePartitionScanner.PARTITION_COMMENT_SINGLE); + reconciler.setRepairer(dr, CompilerSourcePartitionScanner.PARTITION_COMMENT_SINGLE); + + commentMultipleScanner = new AssemblerRuleBasedScanner( + AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_COMMENT); + dr = new DefaultDamagerRepairer(commentMultipleScanner); + reconciler.setDamager(dr, CompilerSourcePartitionScanner.PARTITION_COMMENT_MULTIPLE); + reconciler.setRepairer(dr, CompilerSourcePartitionScanner.PARTITION_COMMENT_MULTIPLE); + + stringScanner = new AssemblerRuleBasedScanner(AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_STRING); + dr = new DefaultDamagerRepairer(stringScanner); + reconciler.setDamager(dr, CompilerSourcePartitionScanner.PARTITION_STRING); + reconciler.setRepairer(dr, CompilerSourcePartitionScanner.PARTITION_STRING); + + instructionScanner = new AssemblerSourceScanner(editor); + dr = new DefaultDamagerRepairer(instructionScanner); + reconciler.setDamager(dr, IDocument.DEFAULT_CONTENT_TYPE); + reconciler.setRepairer(dr, IDocument.DEFAULT_CONTENT_TYPE); + + return reconciler; + } + + @Override + public IReconciler getReconciler(ISourceViewer sourceViewer) { + if (sourceViewer == null) { + throw new IllegalArgumentException("Parameter 'sourceViewer' must not be null."); + } + IReconcilingStrategy reconcilingStrategy = new AssemblerReconcilingStategy(editor); + + MonoReconciler reconciler = new MonoReconciler(reconcilingStrategy, false); + reconciler.setProgressMonitor(new NullProgressMonitor()); + reconciler.setDelay(500); + + return reconciler; + } + + @SuppressWarnings("unchecked") + @Override + protected Map<String, IAdaptable> getHyperlinkDetectorTargets(ISourceViewer sourceViewer) { + Map<String, IAdaptable> targets = super.getHyperlinkDetectorTargets(sourceViewer); + targets.put(AssemblerHyperlinkDetector.TARGET, editor); + return targets; + } + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/CompilerSourceParserTreeObjectLabelProvider.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/CompilerSourceParserTreeObjectLabelProvider.java new file mode 100644 index 00000000..71ebeac2 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/CompilerSourceParserTreeObjectLabelProvider.java @@ -0,0 +1,201 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import org.eclipse.jface.viewers.DelegatingStyledCellLabelProvider; +import org.eclipse.jface.viewers.LabelProvider; +import org.eclipse.jface.viewers.StyledString; +import org.eclipse.swt.graphics.Image; + +import com.wudsn.ide.asm.AssemblerPlugin; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParserTreeObject; +import com.wudsn.ide.asm.compiler.parser.CompilerSourceParserTreeObjectType; + +/** + * LabelProvider for the {@link CompilerSourceParserTreeObject} instances in the + * outline page and the content assist popup. + * + * @author Peter Dell + * @author Daniel Mitte + */ +final class CompilerSourceParserTreeObjectLabelProvider extends DelegatingStyledCellLabelProvider { + + /** Default tree image */ + private final Image defaultImage; + + /** Outline definition section image */ + private final Image definitionSectionImage; + + /** Outline implementation section image */ + private final Image implementationSectionImage; + + /** Outline equate definition image */ + private final Image equateDefintionImage; + + /** Outline label definition image */ + private final Image labelDefinitionImage; + + /** Outline enum definition section image */ + private final Image enumDefinitionSectionImage; + + /** Outline structure definition section image */ + private final Image structureDefinitionSectionImage; + + /** Outline local section image */ + private final Image localSectionImage; + + /** Outline macro definition section image */ + private final Image macroDefinitionSectionImage; + + /** Outline pages section image */ + private final Image pagesSectionImage; + + /** Outline procedure definition section image */ + private final Image procedureDefinitionSectionImage; + + /** Outline repeat section image */ + private final Image repeatSectionImage; + + /** Outline source include image */ + private final Image sourceIncludeImage; + + /** Outline binary include image */ + private final Image binaryIncludeImage; + + /** Outline binary include image */ + private final Image binaryOutputImage; + + private static class StyledLabelProvider extends LabelProvider implements IStyledLabelProvider { + + /** + * Creation is local. + */ + StyledLabelProvider() { + + } + + @Override + public StyledString getStyledText(Object element) { + if (element == null) { + throw new IllegalArgumentException("Parameter 'element' must not be null."); + } + if (element instanceof CompilerSourceParserTreeObject) { + CompilerSourceParserTreeObject elem = (CompilerSourceParserTreeObject) element; + return elem.getStyledString(); + } + + return new StyledString(getText(element)); + } + } + + /** + * Creates a new instance. + * + * Called by + * {@link AssemblerContentOutlinePage#createControl(org.eclipse.swt.widgets.Composite)} + * . + */ + CompilerSourceParserTreeObjectLabelProvider() { + super(new StyledLabelProvider()); + AssemblerPlugin plugin; + plugin = AssemblerPlugin.getInstance(); + defaultImage = plugin.getImage("outline-default-16x16.gif"); + definitionSectionImage = plugin.getImage("outline-definition-section-16x16.gif"); + implementationSectionImage = plugin.getImage("outline-implementation-section-16x16.gif"); + + equateDefintionImage = plugin.getImage("outline-equate-definition-16x16.gif"); + labelDefinitionImage = plugin.getImage("outline-label-definition-16x16.gif"); + + enumDefinitionSectionImage = plugin.getImage("outline-enum-definition-section-16x16.gif"); + structureDefinitionSectionImage = plugin.getImage("outline-structure-definition-section-16x16.gif"); + localSectionImage = plugin.getImage("outline-local-section-16x16.gif"); + macroDefinitionSectionImage = plugin.getImage("outline-macro-definition-section-16x16.gif"); + pagesSectionImage = plugin.getImage("outline-pages-section-16x16.gif"); + procedureDefinitionSectionImage = plugin.getImage("outline-procedure-definition-section-16x16.gif"); + repeatSectionImage = plugin.getImage("outline-repeat-section-16x16.gif"); + + sourceIncludeImage = plugin.getImage("outline-source-include-16x16.gif"); + binaryIncludeImage = plugin.getImage("outline-binary-include-16x16.gif"); + binaryOutputImage = plugin.getImage("outline-binary-output-16x16.gif"); + } + + @Override + public Image getImage(Object element) { + Image result; + if (element instanceof CompilerSourceParserTreeObject) { + CompilerSourceParserTreeObject elem = (CompilerSourceParserTreeObject) element; + int type = elem.getType(); + + switch (type) { + case CompilerSourceParserTreeObjectType.DEFINITION_SECTION: + result = definitionSectionImage; + break; + case CompilerSourceParserTreeObjectType.IMPLEMENTATION_SECTION: + result = implementationSectionImage; + break; + + case CompilerSourceParserTreeObjectType.EQUATE_DEFINITION: + result = equateDefintionImage; + break; + case CompilerSourceParserTreeObjectType.LABEL_DEFINITION: + result = labelDefinitionImage; + break; + + case CompilerSourceParserTreeObjectType.ENUM_DEFINITION_SECTION: + result = enumDefinitionSectionImage; + break; + case CompilerSourceParserTreeObjectType.STRUCTURE_DEFINITION_SECTION: + result = structureDefinitionSectionImage; + break; + case CompilerSourceParserTreeObjectType.LOCAL_SECTION: + result = localSectionImage; + break; + case CompilerSourceParserTreeObjectType.MACRO_DEFINITION_SECTION: + result = macroDefinitionSectionImage; + break; + case CompilerSourceParserTreeObjectType.PAGES_SECTION: + result = pagesSectionImage; + break; + case CompilerSourceParserTreeObjectType.PROCEDURE_DEFINITION_SECTION: + result = procedureDefinitionSectionImage; + break; + case CompilerSourceParserTreeObjectType.REPEAT_SECTION: + result = repeatSectionImage; + break; + + case CompilerSourceParserTreeObjectType.SOURCE_INCLUDE: + result = sourceIncludeImage; + break; + case CompilerSourceParserTreeObjectType.BINARY_INCLUDE: + result = binaryIncludeImage; + break; + case CompilerSourceParserTreeObjectType.BINARY_OUTPUT: + result = binaryOutputImage; + break; + default: + result = defaultImage; + } + } else { + result = null; + } + + return result; + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/CompilerSymbolLabelProvider.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/CompilerSymbolLabelProvider.java new file mode 100644 index 00000000..86652278 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/CompilerSymbolLabelProvider.java @@ -0,0 +1,111 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import org.eclipse.swt.graphics.Image; + +import com.wudsn.ide.asm.AssemblerPlugin; +import com.wudsn.ide.asm.compiler.CompilerSymbol; +import com.wudsn.ide.asm.compiler.CompilerSymbolType; + + +/** + * LabelProvider for the {@link CompilerSymbol} instances in the compiler + * symbols view. + * + * @author Peter Dell + */ +final class CompilerSymbolLabelProvider { + + /** Default image */ + private final Image defaultImage; + + /** Outline equate definition image */ + private final Image equateDefintionImage; + + /** Outline label definition image */ + private final Image labelDefinitionImage; + + /** Outline enum definition section image */ + private final Image enumDefinitionSectionImage; + + /** Outline structure definition section image */ + private final Image structureDefinitionSectionImage; + + /** Outline local section image */ + private final Image localSectionImage; + + /** Outline procedure definition section image */ + private final Image procedureDefinitionSectionImage; + + + CompilerSymbolLabelProvider() { + AssemblerPlugin plugin; + plugin = AssemblerPlugin.getInstance(); + defaultImage = plugin.getImage("outline-default-16x16.gif"); + + equateDefintionImage = plugin.getImage("outline-equate-definition-16x16.gif"); + labelDefinitionImage = plugin.getImage("outline-label-definition-16x16.gif"); + + enumDefinitionSectionImage = plugin.getImage("outline-enum-definition-section-16x16.gif"); + structureDefinitionSectionImage = plugin.getImage("outline-structure-definition-section-16x16.gif"); + localSectionImage = plugin.getImage("outline-local-section-16x16.gif"); + procedureDefinitionSectionImage = plugin.getImage("outline-procedure-definition-section-16x16.gif"); + + } + + public Image getImage(Object element) { + Image result; + if (element instanceof CompilerSymbol) { + CompilerSymbol elem = (CompilerSymbol) element; + int type = elem.getType(); + + switch (type) { + case CompilerSymbolType.EQUATE_DEFINITION: + result = equateDefintionImage; + break; + case com.wudsn.ide.asm.compiler.CompilerSymbolType.LABEL_DEFINITION: + result = labelDefinitionImage; + break; + + case CompilerSymbolType.ENUM_DEFINITION_SECTION: + result = enumDefinitionSectionImage; + break; + case CompilerSymbolType.STRUCTURE_DEFINITION_SECTION: + result = structureDefinitionSectionImage; + break; + case CompilerSymbolType.LOCAL_SECTION: + result = localSectionImage; + break; + + case CompilerSymbolType.PROCEDURE_DEFINITION_SECTION: + result = procedureDefinitionSectionImage; + break; + + default: + result = defaultImage; + } + } else { + result = null; + } + + return result; + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/CompilerSymbolsView.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/CompilerSymbolsView.java new file mode 100644 index 00000000..95b4b27b --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/editor/CompilerSymbolsView.java @@ -0,0 +1,296 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.editor; + +import java.text.DateFormat; +import java.util.ArrayList; +import java.util.Collections; +import java.util.Date; +import java.util.List; + +import org.eclipse.jface.viewers.ArrayContentProvider; +import org.eclipse.jface.viewers.ColumnLabelProvider; +import org.eclipse.jface.viewers.ColumnViewerToolTipSupport; +import org.eclipse.jface.viewers.TableViewer; +import org.eclipse.jface.viewers.TableViewerColumn; +import org.eclipse.jface.window.ToolTip; +import org.eclipse.swt.SWT; +import org.eclipse.swt.events.KeyAdapter; +import org.eclipse.swt.events.KeyEvent; +import org.eclipse.swt.graphics.Image; +import org.eclipse.swt.layout.GridData; +import org.eclipse.swt.layout.GridLayout; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Label; +import org.eclipse.swt.widgets.Table; +import org.eclipse.swt.widgets.TableColumn; +import org.eclipse.swt.widgets.Text; +import org.eclipse.ui.part.ViewPart; + +import com.wudsn.ide.asm.Texts; +import com.wudsn.ide.asm.compiler.CompilerFiles; +import com.wudsn.ide.asm.compiler.CompilerSymbol; +import com.wudsn.ide.asm.compiler.CompilerSymbolType; +import com.wudsn.ide.base.common.NumberUtility; +import com.wudsn.ide.base.common.TextUtility; + +// TODO Add column sorting +public final class CompilerSymbolsView extends ViewPart { + + public static final String ID = "com.wudsn.ide.asm.editor.CompilerSymbolsView"; + + // Model + private CompilerFiles compilerFiles; + private List<CompilerSymbol> compilerSymbols; + private Date updateTimestamp; + + // View + private Text filterTextField; + private Label sourceFileNameText; + private Label symbolsCountText; + private TableViewer viewer; + + public CompilerSymbolsView() { + compilerFiles = null; + compilerSymbols = Collections.emptyList(); + updateTimestamp = null; + } + + @Override + public void createPartControl(Composite parent) { + GridLayout layout = new GridLayout(4, false); + parent.setLayout(layout); + filterTextField = new Text(parent, SWT.BORDER | SWT.SEARCH); + filterTextField.setToolTipText(Texts.COMPILER_SYMBOLS_VIEW_FILTER_TOOLTIP); + filterTextField.setMessage(filterTextField.getToolTipText()); + filterTextField.setLayoutData(new GridData(GridData.GRAB_HORIZONTAL | GridData.HORIZONTAL_ALIGN_FILL)); + + // Filter as you type... + filterTextField.addKeyListener(new KeyAdapter() { + @Override + public void keyReleased(KeyEvent e) { + dataToUI(); + } + }); + Label sourceFileNameLabel = new Label(parent, SWT.NONE); + sourceFileNameLabel.setText(Texts.COMPILER_SYMBOLS_VIEW_SOURCE_LABEL); + sourceFileNameText = new Label(parent, SWT.NONE); + symbolsCountText = new Label(parent, SWT.NONE); + createViewer(parent); + } + + private void createViewer(Composite parent) { + if (parent == null) { + throw new IllegalArgumentException("Parameter 'parent' must not be null."); + } + viewer = new TableViewer(parent, SWT.MULTI | SWT.H_SCROLL | SWT.V_SCROLL | SWT.FULL_SELECTION | SWT.BORDER); + createColumns(parent, viewer); + final Table table = viewer.getTable(); + table.setHeaderVisible(true); + table.setLinesVisible(true); + + viewer.setContentProvider(new ArrayContentProvider()); + dataToUI(); + + // Make the selection available to other views + getSite().setSelectionProvider(viewer); + + // Set the sorter for the table + + // Define layout for the viewer + GridData gridData = new GridData(); + gridData.verticalAlignment = GridData.FILL; + gridData.horizontalSpan = 4; + gridData.grabExcessHorizontalSpace = true; + gridData.grabExcessVerticalSpace = true; + gridData.horizontalAlignment = GridData.FILL; + viewer.getControl().setLayoutData(gridData); + + // Activate the tooltip support for the viewer + ColumnViewerToolTipSupport.enableFor(viewer, ToolTip.NO_RECREATE); + } + + // create the columns for the table + private void createColumns(final Composite parent, final TableViewer viewer) { + if (parent == null) { + throw new IllegalArgumentException("Parameter 'parent' must not be null."); + } + if (viewer == null) { + throw new IllegalArgumentException("Parameter 'viewer' must not be null."); + } + int index = 0; + + // Column: Type + final CompilerSymbolLabelProvider labelProvider = new CompilerSymbolLabelProvider(); + TableViewerColumn col = createTableViewerColumn(Texts.COMPILER_SYMBOLS_VIEW_TYPE_COLUMN_LABEL, 40, index++); + col.setLabelProvider(new ColumnLabelProvider() { + @Override + public String getText(Object element) { + return null; + } + + @Override + public String getToolTipText(Object element) { + CompilerSymbol compilerSymbol = (CompilerSymbol) element; + return CompilerSymbolType.getText(compilerSymbol.getType()); + } + + @Override + public Image getImage(Object element) { + return labelProvider.getImage(element); + } + }); + + // Column: Bank + col = createTableViewerColumn(Texts.COMPILER_SYMBOLS_VIEW_BANK_COLUMN_LABEL, 40, index++); + col.setLabelProvider(new ColumnLabelProvider() { + @Override + public String getText(Object element) { + CompilerSymbol compilerSymbol = (CompilerSymbol) element; + return compilerSymbol.getBankString(); + } + }); + + // Column: Name + col = createTableViewerColumn(Texts.COMPILER_SYMBOLS_VIEW_NAME_COLUMN_LABEL, 200, index++); + col.setLabelProvider(new ColumnLabelProvider() { + @Override + public String getText(Object element) { + CompilerSymbol compilerSymbol = (CompilerSymbol) element; + return compilerSymbol.getName(); + } + }); + + // Column: Hex Value + col = createTableViewerColumn(Texts.COMPILER_SYMBOLS_VIEW_HEX_VALUE_COLUMN_LABEL, 100, index++); + col.setLabelProvider(new ColumnLabelProvider() { + @Override + public String getText(Object element) { + CompilerSymbol compilerSymbol = (CompilerSymbol) element; + return compilerSymbol.getValueAsHexString(); + } + }); + + // Column: Decimal Value + col = createTableViewerColumn(Texts.COMPILER_SYMBOLS_VIEW_DECIMAL_VALUE_COLUMN_LABEL, 100, index++); + col.setLabelProvider(new ColumnLabelProvider() { + @Override + public String getText(Object element) { + CompilerSymbol compilerSymbol = (CompilerSymbol) element; + return compilerSymbol.getValueAsDecimalString(); + } + }); + + // Column: String Value + col = createTableViewerColumn(Texts.COMPILER_SYMBOLS_VIEW_STRING_VALUE_COLUMN_LABEL, 100, index++); + col.setLabelProvider(new ColumnLabelProvider() { + @Override + public String getText(Object element) { + CompilerSymbol compilerSymbol = (CompilerSymbol) element; + return compilerSymbol.getValueAsString(); + } + }); + } + + private TableViewerColumn createTableViewerColumn(String title, int bound, int colNumber) { + if (title == null) { + throw new IllegalArgumentException("Parameter 'title' must not be null."); + } + final TableViewerColumn viewerColumn = new TableViewerColumn(viewer, SWT.NONE); + final TableColumn column = viewerColumn.getColumn(); + column.setText(title); + column.setWidth(bound); + column.setResizable(true); + column.setMoveable(true); + return viewerColumn; + } + + @Override + public void setFocus() { + filterTextField.setFocus(); + } + + public void setSymbols(CompilerFiles compilerFiles, List<CompilerSymbol> compilerSymbols) { + if (compilerFiles == null) { + throw new IllegalArgumentException("Parameter 'compilerFiles' must not be null."); + } + if (compilerSymbols == null) { + throw new IllegalArgumentException("Parameter 'compilerSymbols' must not be null."); + } + this.compilerFiles = compilerFiles; + this.compilerSymbols = new ArrayList<CompilerSymbol>(compilerSymbols); + this.updateTimestamp = new Date(); + if (viewer != null) { + dataToUI(); + } + } + + void dataToUI() { + String text = Texts.COMPILER_SYMBOLS_VIEW_SOURCE_NONE; + if (compilerFiles != null) { + text = compilerFiles.mainSourceFile.fileName; + text += " " + DateFormat.getTimeInstance().format(updateTimestamp); + } + sourceFileNameText.setText(text); + String filterTextSequence = filterTextField.getText().toUpperCase(); + + // A leading ! reverses the filter + boolean matchTarget= true; + if (filterTextSequence.startsWith("!")){ + filterTextSequence=filterTextSequence.substring(1); + matchTarget=false; + } + + String[] filterTexts = filterTextSequence.split("[ ]+"); + List<CompilerSymbol> filteredCompilerSymbols = compilerSymbols; + if (filterTexts.length > 0) { + filteredCompilerSymbols = new ArrayList<CompilerSymbol>(); + for (CompilerSymbol compilerSymbol : compilerSymbols) { + boolean match = true; + for (int i = 0; i < filterTexts.length && match; i++) { + String filterText = filterTexts[i]; + boolean symbolMatch = (compilerSymbol.getNameUpperCase().contains(filterText) + || compilerSymbol.getValueAsHexStringUpperCase().contains(filterText) || compilerSymbol + .getValueAsDecimalString().contains(filterText)) + || compilerSymbol.getValueAsStringUpperCase().contains(filterText); + match &= symbolMatch; + } + + if (match == matchTarget) { + filteredCompilerSymbols.add(compilerSymbol); + } + } + } + int totalCount = compilerSymbols.size(); + String totalCountText = NumberUtility.getLongValueDecimalString(totalCount); + + int filteredCount = filteredCompilerSymbols.size(); + if (totalCount > 0 && filteredCount < totalCount) { + String filteredCountText = NumberUtility.getLongValueDecimalString(filteredCount); + text = TextUtility.format(Texts.COMPILER_SYMBOLS_VIEW_SOURCE_FILTERED_COUNT, filteredCountText, + totalCountText); + } else { + text = TextUtility.format(Texts.COMPILER_SYMBOLS_VIEW_SOURCE_TOTAL_COUNT, totalCountText); + } + symbolsCountText.setText(text); + // Calling setInput will call getElements in the contentProvider + viewer.setInput(filteredCompilerSymbols); + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/AssemblerHelpContentProducer.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/AssemblerHelpContentProducer.java new file mode 100644 index 00000000..aeed488f --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/AssemblerHelpContentProducer.java @@ -0,0 +1,684 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.help; + +import java.io.ByteArrayInputStream; +import java.io.File; +import java.io.FileInputStream; +import java.io.FileNotFoundException; +import java.io.IOException; +import java.io.InputStream; +import java.io.UnsupportedEncodingException; +import java.net.URI; +import java.net.URISyntaxException; +import java.net.URLDecoder; +import java.util.ArrayList; +import java.util.Collections; +import java.util.HashMap; +import java.util.List; +import java.util.Locale; +import java.util.Map; + +import org.eclipse.core.runtime.CoreException; +import org.eclipse.core.runtime.FileLocator; +import org.eclipse.core.runtime.IPath; +import org.eclipse.core.runtime.Path; +import org.eclipse.help.IHelpContentProducer; + +import com.wudsn.ide.asm.AssemblerPlugin; +import com.wudsn.ide.asm.CPU; +import com.wudsn.ide.asm.Hardware; +import com.wudsn.ide.asm.HardwareUtility; +import com.wudsn.ide.asm.Texts; +import com.wudsn.ide.asm.compiler.Compiler; +import com.wudsn.ide.asm.compiler.CompilerDefinition; +import com.wudsn.ide.asm.compiler.CompilerRegistry; +import com.wudsn.ide.asm.compiler.syntax.CompilerSyntax; +import com.wudsn.ide.asm.compiler.syntax.CompilerSyntaxUtility; +import com.wudsn.ide.asm.compiler.syntax.Directive; +import com.wudsn.ide.asm.compiler.syntax.Instruction; +import com.wudsn.ide.asm.compiler.syntax.InstructionSet; +import com.wudsn.ide.asm.compiler.syntax.InstructionType; +import com.wudsn.ide.asm.compiler.syntax.Opcode; +import com.wudsn.ide.asm.compiler.syntax.Opcode.OpcodeAddressingMode; +import com.wudsn.ide.asm.runner.RunnerDefinition; +import com.wudsn.ide.asm.runner.RunnerId; +import com.wudsn.ide.asm.runner.RunnerRegistry; +import com.wudsn.ide.base.common.EnumUtility; +import com.wudsn.ide.base.common.HexUtility; +import com.wudsn.ide.base.common.StringUtility; +import com.wudsn.ide.base.common.TextUtility; + +/** + * Dynamic help content provider. Uses static pages and the meta data from the + * compiler definitions to build a comprehensive help. + * + * @author Peter Dell + * + * @since 1.6.1 + * + * TODO Complete opcode entries in Compiler.xml, also for extended and + * illegal opcodes + */ +public final class AssemblerHelpContentProducer implements IHelpContentProducer { + + // In order to get the navigation breadcrumbs automatically, + // the files have to have this suffix (see BreadcrumbsFilter). + public static final String EXTENSION = ".html"; + public static final String SECTION_EXTENSION = ".section.html"; + private static final String SECTION_PREFIX = "<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.01 Transitional//EN\">" + + "<html><head>" + + "<meta http-equiv=\"content-type\" content=\"text/html;charset=utf-8\">" + + "<link rel=\"stylesheet\" href=\"../../../content/org.eclipse.platform/book.css\" type=\"text/css\"></link>" + + "<style type=\"text/css\">table,tbody,td,th {border-style:solid;border-width:1px;border-collapse:collapse; " + + "transition-property: background-color;transition-duration: 0.25s;transition-timing-function: linear;transition-delay: 0ms;} " + + "th {background-color:#0074cc;color:#fff } </style>" + "</head><body>"; + private static final String SECTION_SUFFIX = "</body></html>"; + + public static final String SCHEMA_COMPILER = "compiler/"; + public static final String SCHEMA_HARDWARE = "hardware/"; + + public static final String SECTION_GENERAL = "general"; + public static final String SECTION_INSTRUCTIONS = "instructions"; + public static final String SECTION_MANUAL = "manual"; + public static final String SECTION_MANUAL_FILE = "file"; + + public static final String SCHEMA_CPU = "cpu/"; + + private static final String ICONS_PATH = "/help/topic/com.wudsn.ide.asm/icons/"; + + @Override + public InputStream getInputStream(String pluginID, String href, Locale locale) { + if (AssemblerPlugin.ID.equals(pluginID)) { + int index = href.indexOf("?"); + if (index >= 0) { + href = href.substring(0, index); + } + if (href.startsWith(SCHEMA_COMPILER)) { + return getCompilerInputStream(href); + } else if (href.startsWith(SCHEMA_HARDWARE)) { + return getHardwareInputStream(href); + } else if (href.startsWith(SCHEMA_CPU)) { + return getCPUInputStream(href); + } else if (href.endsWith(".html")) { // Web site documents + return getHTMLInputStream(href); + } + } + + return null; + } + + /** + * Gets the HTML input stream for HTML files. + * + * @param href + * The href ending with ".html" + * @return The input stream, not <code>null</code>. + */ + private InputStream getHTMLInputStream(String href) { + if (href == null) { + throw new IllegalArgumentException("Parameter 'href' must not be null."); + } + AssemblerPlugin plugin = AssemblerPlugin.getInstance(); + IPath path = new Path(href); + InputStream result; + try { + result = FileLocator.openStream(plugin.getBundle(), path, false); + // HTML files that do not have an own <html> and <body> tag must be + // wrapped. + if (href.endsWith(SECTION_EXTENSION)) { + result = new HTMLWrapperInputStream(SECTION_PREFIX, SECTION_SUFFIX, result); + } + } catch (IOException ex) { + plugin.logError("Cannot open stream for {0}", new Object[] { href }, ex); + result = null; + } + + return result; + } + + /** + * Create a string build for formatted help content. + * + * @return The HTML write, not <code>null</code>. + */ + private HTMLWriter createHeader() { + HTMLWriter writer = new HTMLWriter(); + + writer.writeText(HTMLConstants.PREFIX); + return writer; + } + + /** + * Convert a string builder to an input stream. + * + * @param writer + * The HTML writer, not <code>null</code>. + * @return The input stream, not <code>null</code>. + */ + private InputStream getInputStream(HTMLWriter writer) { + if (writer == null) { + throw new IllegalArgumentException("Parameter 'writer' must not be null."); + } + writer.writeText(HTMLConstants.SUFFIX); + + try { + return new ByteArrayInputStream(writer.toHTML().getBytes(HTMLConstants.UTF8)); + } catch (UnsupportedEncodingException ex) { + throw new RuntimeException(ex); + } + } + + private String getPath(String prefix, String href) { + if (prefix == null) { + throw new IllegalArgumentException("Parameter 'prefix' must not be null."); + } + if (href == null) { + throw new IllegalArgumentException("Parameter 'href' must not be null."); + } + String path = href.substring(prefix.length()); + int index = path.indexOf("?"); + if (index >= 0) { + path = path.substring(0, index); + } + + index = path.lastIndexOf('.'); + if (index > -1) { + path = path.substring(0, index); + return path; + } + return null; + } + + private Map<String, List<String>> getQueryParameters(String href) { + if (href == null) { + throw new IllegalArgumentException("Parameter 'href' must not be null."); + } + Map<String, List<String>> params = new HashMap<String, List<String>>(); + + try { + URI uri; + try { + uri = new URI(href); + } catch (URISyntaxException ex1) { + throw new RuntimeException("Cannot parse '" + href + "'", ex1); + } + String query = uri.getQuery(); + if (query != null) { + for (String param : query.split("&")) { + String pair[] = param.split("="); + String key; + + key = URLDecoder.decode(pair[0], "UTF-8"); + + String value = ""; + if (pair.length > 1) { + value = URLDecoder.decode(pair[1], "UTF-8"); + } + List<String> values = params.get(key); + if (values == null) { + values = new ArrayList<String>(); + params.put(key, values); + } + values.add(value); + } + } + } catch (UnsupportedEncodingException ex) { + throw new RuntimeException(ex); + } + return params; + } + + private InputStream getCompilerInputStream(String href) { + if (href == null) { + throw new IllegalArgumentException("Parameter 'href' must not be null."); + } + + String path = getPath(SCHEMA_COMPILER, href); + if (path == null) { + return null; + } + int index = path.indexOf("/"); + if (index <= 0) { + return null; + } + String compilerId = path.substring(0, index); + String section = path.substring(index + 1); + AssemblerPlugin assemblerPlugin = AssemblerPlugin.getInstance(); + CompilerRegistry compilerRegistry = assemblerPlugin.getCompilerRegistry(); + + // Find non-empty compiler executable path. + Compiler compiler = compilerRegistry.getCompiler(compilerId); + + if (section.startsWith(SECTION_GENERAL)) { + return getInputStream(getCompilerGeneralSection(compiler)); + } else if (section.startsWith(SECTION_MANUAL)) { + + String compilerExecutablePath = assemblerPlugin.getPreferences().getCompilerExecutablePath(compilerId); + if (StringUtility.isEmpty(compilerExecutablePath)) { + // ERROR: Help for the '{0}' compiler cannot be + // displayed because the path to the compiler executable + // is not set in the preferences. + String message = TextUtility.format(Texts.MESSAGE_E130, new String[] { compiler.getDefinition() + .getName() }); + HTMLWriter writer = createHeader(); + writer.writeText(message); + return getInputStream(writer); + } + + try { + File file = compiler.getDefinition().getHelpFile(compilerExecutablePath); + + // Handle file requests within the help directory. + List<String> fileNames = getQueryParameters(href).get(SECTION_MANUAL_FILE); + String fileName; + if (fileNames != null && fileNames.size() == 1) { + fileName = fileNames.get(0); + } else { + fileName = ""; + } + + if (StringUtility.isSpecified(fileName)) { + file = new File(file, fileName); + } + InputStream inputStream = new FileInputStream(file); + + index = file.getName().indexOf("."); + if (index == -1 || file.getName().substring(index).equalsIgnoreCase(".txt")) { + inputStream = new HTMLWrapperInputStream("<html><body><pre>", "</pre></body></html>", inputStream); + } + return inputStream; + + } catch (CoreException ex) { + HTMLWriter writer = createHeader(); + writer.writeText(ex.getMessage()); + return getInputStream(writer); + + } catch (FileNotFoundException ex) { + HTMLWriter writer = createHeader(); + writer.writeText(ex.getMessage()); + return getInputStream(writer); + + } + + } else if (section.equals(SECTION_INSTRUCTIONS)) { + return getInputStream(getCompilerInstructionsSection(compiler.getDefinition())); + + } + return null; + } + + private HTMLWriter getCompilerGeneralSection(Compiler compiler) { + CompilerDefinition compilerDefinition = compiler.getDefinition(); + + HTMLWriter writer = createHeader(); + + writer.beginTable(true); + + writer.writeTableRow(Texts.TOC_ASSEMBLER_NAME_LABEL, compilerDefinition.getName()); + + writer.writeTableRow(Texts.TOC_ASSEMBLER_HOME_PAGE_LABEL, + HTMLWriter.getLink(compilerDefinition.getHomePageURL(), compilerDefinition.getHomePageURL())); + + Hardware hardware = compilerDefinition.getDefaultHardware(); + writer.writeTableRow( + Texts.TOC_ASSEMBLER_DEFAULT_HARDWARE_LABEL, + HTMLWriter.getImage(ICONS_PATH + HardwareUtility.getImagePath(hardware), hardware.name(), + hardware.name())); + + List<CPU> cpus = compilerDefinition.getSupportedCPUs(); + writer.beginTableRow(); + writer.writeTableHeader(Texts.TOC_ASSEMBLER_SUPPORTED_CPUS_LABEL); + StringBuilder builder = new StringBuilder(); + for (int i = 0; i < cpus.size(); i++) { + builder.append(EnumUtility.getText(cpus.get(i))); + if (i < cpus.size() - 1) { + builder.append(", "); + } + } + writer.writeTableCell(builder.toString()); + writer.end(); + + writer.beginTableRow(); + writer.writeTableHeader(Texts.TOC_ASSEMBLER_DEFAULT_PARAMETERS_LABEL); + + writer.writeTableCell(compilerDefinition.getDefaultParameters()); + writer.end(); + + CompilerSyntax compilerSyntax = compilerDefinition.getSyntax(); + + writer.writeTableRow(Texts.TOC_ASSEMBLER_SYNTAX_IDENTIFIERS_CASE_SENSITIVE, compilerSyntax + .areIdentifiersCaseSensitive() ? Texts.TOC_ASSEMBLER_SYNTAX_YES : Texts.TOC_ASSEMBLER_SYNTAX_NO); + + writer.writeTableRowCode(Texts.TOC_ASSEMBLER_SYNTAX_IDENTIFIER_START_CHARACTERS, + getCompilerGeneralCharactersWrapped(compilerSyntax.getIdentifierStartCharacters())); + + writer.writeTableRowCode(Texts.TOC_ASSEMBLER_SYNTAX_IDENTIFIER_PART_CHARACTERS, + getCompilerGeneralCharactersWrapped(compilerSyntax.getIdentifierPartCharacters())); + + writer.writeTableRowCode(Texts.TOC_ASSEMBLER_SYNTAX_IDENTIFIER_SEPARATOR_CHARACTER, + compilerSyntax.getIdentifierSeparatorCharacter()); + + writer.writeTableRow(Texts.TOC_ASSEMBLER_SYNTAX_INSTRUCTIONS_CASE_SENSITIVE, compilerSyntax + .areInstructionsCaseSensitive() ? Texts.TOC_ASSEMBLER_SYNTAX_YES : Texts.TOC_ASSEMBLER_SYNTAX_NO); + + writer.writeTableRowCode(Texts.TOC_ASSEMBLER_SYNTAX_BLOCK_DEFINITION_CHARACTERS, + compilerSyntax.getBlockDefinitionCharacters()); + + writer.writeTableRowCode(Texts.TOC_ASSEMBLER_SYNTAX_COMPLETION_PROPOSAL_AUTO_ACTIVATION_CHARACTERS, new String( + compilerSyntax.getCompletionProposalAutoActivationCharacters())); + + writer.writeTableRowCode(Texts.TOC_ASSEMBLER_SYNTAX_LABEL_DEFINITION_SUFFIX_CHARACTER, + compilerSyntax.getLabelDefinitionSuffixCharacter()); + + writer.writeTableRowCode(Texts.TOC_ASSEMBLER_SYNTAX_MACRO_USAGE_PREFIX_CHARACTER, + compilerSyntax.getMacroUsagePrefixCharacter()); + + writer.writeTableRowCode(Texts.TOC_ASSEMBLER_SYNTAX_SOURCE_INCLUDE_DEFAULT_EXTENSION, + compilerSyntax.getSourceIncludeDefaultExtension()); + + writer.writeTableRowCode(Texts.TOC_ASSEMBLER_SYNTAX_MULTIPLE_LINES_COMMENT_DELIMITERS, + HTMLWriter.getString(compilerSyntax.getMultipleLinesCommentDelimiters())); + + writer.writeTableRowCode(Texts.TOC_ASSEMBLER_SYNTAX_SINGLE_LINE_COMMENT_DELIMITERS, + HTMLWriter.getString(compilerSyntax.getSingleLineCommentDelimiters())); + + writer.writeTableRowCode(Texts.TOC_ASSEMBLER_SYNTAX_STRING_DELIMITERS, + HTMLWriter.getString(compilerSyntax.getStringDelimiters())); + + writer.end(); + + return writer; + + } + + private String getCompilerGeneralCharactersWrapped(String text) { + if (text == null) { + throw new IllegalArgumentException("Parameter 'text' must not be null."); + } + int index = text.indexOf('Z'); + if (index > -1 && index < text.length() - 1) { + text = text.substring(0, index + 1) + " " + text.substring(index + 1); + } + return text; + } + + /** + * Creates an HTML string describing all instructions of the compiler. + * + * @param compilerDefinition + * The compiler definition, not <code>null</code>. + * @return The HTML string builder describing all instructions of the + * compiler, may be empty, not <code>null</code>. + */ + private HTMLWriter getCompilerInstructionsSection(CompilerDefinition compilerDefinition) { + if (compilerDefinition == null) { + throw new IllegalArgumentException("Parameter 'compilerDefinition' must not be null."); + } + HTMLWriter result = createHeader(); + + CompilerSyntax syntax = compilerDefinition.getSyntax(); + List<Instruction> directives = new ArrayList<Instruction>(); + List<Instruction> legalOpcodes = new ArrayList<Instruction>(); + List<Instruction> illegalOpcodes = new ArrayList<Instruction>(); + List<Instruction> pseudoOpcodes = new ArrayList<Instruction>(); + List<Instruction> w65816Opcodes = new ArrayList<Instruction>(); + + List<CPU> cpus = compilerDefinition.getSupportedCPUs(); + for (CPU cpu : cpus) { + for (Instruction instruction : syntax.getInstructionSet(cpu).getInstructions()) { + + if (instruction instanceof Directive) { + if (!directives.contains(instruction)) { + directives.add(instruction); + } + } + if (instruction instanceof Opcode) { + Opcode opcode = (Opcode) instruction; + switch (opcode.getType()) { + case InstructionType.LEGAL_OPCODE: + if (!opcode.isW65816()) { + if (!legalOpcodes.contains(instruction)) { + + legalOpcodes.add(opcode); + } + + } else { + if (!w65816Opcodes.contains(instruction)) { + w65816Opcodes.add(opcode); + } + } + break; + case InstructionType.ILLEGAL_OPCODE: + if (!illegalOpcodes.contains(instruction)) { + + illegalOpcodes.add(opcode); + } + break; + case InstructionType.PSEUDO_OPCODE: + if (!pseudoOpcodes.contains(instruction)) { + pseudoOpcodes.add(opcode); + } + break; + default: + + } + } + } + } + + getCompilerInstructions(result, Texts.TOC_ASSEMBLER_INSTRUCTION_TYPE_DIRECTIVES_LABEL, directives); + getCompilerInstructions(result, Texts.TOC_ASSEMBLER_INSTRUCTION_TYPE_LEGAL_OPCODES_LABEL, legalOpcodes); + getCompilerInstructions(result, Texts.TOC_ASSEMBLER_INSTRUCTION_TYPE_PSEUDO_OPCODES_LABEL, pseudoOpcodes); + getCompilerInstructions(result, Texts.TOC_ASSEMBLER_INSTRUCTION_TYPE_ILLEGAL_OPCODES_LABEL, illegalOpcodes); + getCompilerInstructions(result, Texts.TOC_ASSEMBLER_INSTRUCTION_TYPE_W65816_OPCODES_LABEL, w65816Opcodes); + + return result; + } + + private void getCompilerInstructions(HTMLWriter writer, String title, List<Instruction> instructions) { + if (writer == null) { + throw new IllegalArgumentException("Parameter 'writer' must not be null."); + } + if (title == null) { + throw new IllegalArgumentException("Parameter 'title' must not be null."); + } + if (instructions == null) { + throw new IllegalArgumentException("Parameter 'instructions' must not be null."); + } + if (instructions.isEmpty()) { + return; + } + + Collections.sort(instructions); + + writer.begin("h3", null); + writer.writeText(title); + writer.end(); + + writer.beginTable(); + writer.beginTableRow(); + writer.writeTableHeader(Texts.TOC_ASSEMBLER_INSTRUCTION_TYPE_LABEL); + writer.writeTableHeader(Texts.TOC_ASSEMBLER_INSTRUCTION_NAME_LABEL); + writer.writeTableHeader(Texts.TOC_ASSEMBLER_INSTRUCTION_DESCRIPTION_LABEL); + + writer.end(); + + for (Instruction instruction : instructions) { + String typeImagePath = ICONS_PATH + CompilerSyntaxUtility.getTypeImagePath(instruction); + String typeText = CompilerSyntaxUtility.getTypeText(instruction); + StringBuilder styledTitle = new StringBuilder(instruction.getStyledTitle()); + int[] offsets = instruction.getStyledTitleOffsets(); + + for (int j = offsets.length - 1; j >= 0; j--) { + styledTitle.insert(offsets[j] + 1, "</u>"); + styledTitle.insert(offsets[j], "<u>"); + } + + writer.beginTableRow(); + writer.writeTableCell(HTMLWriter.getImage(typeImagePath, typeText, "")); + writer.writeTableCell(instruction.getName()); + writer.writeTableCell(styledTitle.toString()); + writer.end(); + + } + writer.end(); + } + + private InputStream getHardwareInputStream(String href) { + if (href == null) { + throw new IllegalArgumentException("Parameter 'href' must not be null."); + } + String path = getPath(SCHEMA_HARDWARE, href); + if (path == null) { + return null; + } + Hardware hardware = Hardware.valueOf(path.toUpperCase()); + AssemblerPlugin assemblerPlugin = AssemblerPlugin.getInstance(); + RunnerRegistry runnerRegistry = assemblerPlugin.getRunnerRegistry(); + + HTMLWriter writer = createHeader(); + + writer.beginTable(); + writer.writeTableRow(Texts.TOC_HARDWARE_NAME_LABEL, EnumUtility.getText(hardware)); + writer.writeTableRow(Texts.TOC_HARDWARE_ID_LABEL, hardware.name()); + writer.writeTableRow(Texts.TOC_HARDWARE_ICON_LABEL, + HTMLWriter.getImage(ICONS_PATH + HardwareUtility.getImagePath(hardware), hardware.name(), "")); + + writer.writeTableRow(Texts.TOC_HARDWARE_DEFAULT_FILE_EXTENSION_LABEL, + HardwareUtility.getDefaultFileExtension(hardware)); + writer.end(); + + writer.begin("br", null); + writer.end(); + + writer.beginTable(); + writer.beginTableRow(); + writer.writeTableHeader(Texts.TOC_HARDWARE_EMULATOR_LABEL); + writer.writeTableHeader(Texts.TOC_HARDWARE_HOME_PAGE_LABEL); + writer.writeTableHeader(Texts.TOC_HARDWARE_DEFAULT_PARAMETERS_LABEL); + writer.end(); + + List<RunnerDefinition> runnerDefinitions = runnerRegistry.getDefinitions(hardware); + for (RunnerDefinition runnerDefinition : runnerDefinitions) { + + String runnerId = runnerDefinition.getId(); + if (runnerId.equals(RunnerId.DEFAULT_APPLICATION) || runnerId.equals(RunnerId.USER_DEFINED_APPLICATION)) { + continue; + } + writer.beginTableRow(); + writer.writeTableCell(runnerDefinition.getName()); + writer.writeTableCell(HTMLWriter.getLink(runnerDefinition.getHomePageURL(), + runnerDefinition.getHomePageURL())); + writer.writeTableCell(runnerDefinition.getDefaultCommandLine()); + + writer.end(); + + } + writer.end(); + + return getInputStream(writer); + } + + private InputStream getCPUInputStream(String href) { + + if (href == null) { + throw new IllegalArgumentException("Parameter 'href' must not be null."); + } + String path = getPath(SCHEMA_CPU, href); + if (path == null) { + return null; + } + + CPU cpu = CPU.valueOf(path.toUpperCase()); + AssemblerPlugin assemblerPlugin = AssemblerPlugin.getInstance(); + CompilerRegistry compilerRegistry = assemblerPlugin.getCompilerRegistry(); + + HTMLWriter writer = createHeader(); + + writer.beginTable(); + writer.writeTableRow(Texts.TOC_CPU_NAME_LABEL, EnumUtility.getText(cpu)); + writer.end(); + + writer.begin("br", null); + writer.end(); + + writer.beginTable(); + writer.beginTableRow(); + + writer.writeTableHeader(Texts.TOC_CPU_OPCODE_LABEL); + writer.writeTableHeader(Texts.TOC_ASSEMBLER_INSTRUCTION_DESCRIPTION_LABEL); + + List<CompilerDefinition> compilerDefinitions = compilerRegistry.getCompilerDefinitions(); + int compilerDefinitionCount = compilerDefinitions.size(); + InstructionSet[] instructionSets = new InstructionSet[compilerDefinitions.size()]; + for (int c = 0; c < compilerDefinitionCount; c++) { + CompilerDefinition compilerDefinition = compilerDefinitions.get(c); + instructionSets[c] = compilerDefinition.getSyntax().getInstructionSet(cpu); + writer.writeTableHeader(compilerDefinition.getName()); + + } + writer.end(); + + List<String> cellContents = new ArrayList<String>(compilerDefinitionCount); + + for (int opcode = 0; opcode < Opcode.MAX_OPCODES; opcode++) { + String opcodeText = null; + cellContents.clear(); + + for (int c = 0; c < compilerDefinitionCount; c++) { + InstructionSet instructionSet = instructionSets[c]; + List<OpcodeAddressingMode> opcodeAddressingModes = instructionSet.getOpcodeAddressingModes(opcode); + StringBuffer cellBuffer = new StringBuffer(); + for (int m = 0; m < opcodeAddressingModes.size(); m++) { + // There should only be one, but be robust here. + if (m > 0 && m < opcodeAddressingModes.size()) { + cellBuffer.append("<br/>"); + } + OpcodeAddressingMode opcodeAddressingMode = opcodeAddressingModes.get(m); + if (opcodeText == null) { + opcodeText = opcodeAddressingMode.getOpcode().getStyledTitle(); + } + cellBuffer.append(opcodeAddressingMode.getFormattedText()); + } + cellContents.add(cellBuffer.toString()); + + } + + if (opcodeText != null) { + writer.beginTableRow(); + writer.writeTableCell(HexUtility.getByteValueHexString(opcode)); + writer.writeTableCell(opcodeText); + + for (int c = 0; c < compilerDefinitionCount; c++) { + + writer.writeTableCell(cellContents.get(c)); + + } + writer.end(); + } + } + + writer.end(); + + return getInputStream(writer); + } + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/AssemblerTocProvider.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/AssemblerTocProvider.java new file mode 100644 index 00000000..ad2f0aee --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/AssemblerTocProvider.java @@ -0,0 +1,414 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.help; + +import java.io.File; +import java.io.UnsupportedEncodingException; +import java.net.URLEncoder; +import java.util.ArrayList; +import java.util.List; +import java.util.Locale; +import java.util.MissingResourceException; +import java.util.ResourceBundle; + +import org.eclipse.core.expressions.IEvaluationContext; +import org.eclipse.core.runtime.CoreException; +import org.eclipse.help.AbstractTocProvider; +import org.eclipse.help.IToc; +import org.eclipse.help.ITocContribution; +import org.eclipse.help.ITopic; +import org.eclipse.help.ITopic2; +import org.eclipse.help.IUAElement; + +import com.wudsn.ide.asm.AssemblerPlugin; +import com.wudsn.ide.asm.CPU; +import com.wudsn.ide.asm.Hardware; +import com.wudsn.ide.asm.Texts; +import com.wudsn.ide.asm.compiler.CompilerDefinition; +import com.wudsn.ide.asm.compiler.CompilerRegistry; +import com.wudsn.ide.asm.preferences.AssemblerPreferences; +import com.wudsn.ide.base.common.EnumUtility; + +/** + * Dynamic help content provider. Uses static pages and the meta data from the + * compiler definitions to build a comprehensive help. + * + * @author Peter Dell + * + * @since 1.6.1 + */ +public final class AssemblerTocProvider extends AbstractTocProvider { + + private static final class Toc implements IToc { + private ITopic[] topics; + + public Toc() { + + topics = createTopics(); + } + + @Override + public boolean isEnabled(IEvaluationContext context) { + return true; + } + + @Override + public IUAElement[] getChildren() { + return topics; + } + + @Override + public String getHref() { + return ""; + } + + @Override + public String getLabel() { + return Texts.TOC_WUDSN_IDE_LABEL; + } + + @Override + public ITopic[] getTopics() { + return topics; + } + + @Override + public ITopic getTopic(String href) { + return topics[0]; + } + } + + private static final class TocContribution implements ITocContribution { + + public TocContribution() { + } + + @Override + public String getCategoryId() { + return "CategoryID"; + } + + @Override + public String getContributorId() { + return AssemblerPlugin.ID; + } + + @Override + public String[] getExtraDocuments() { + return new String[0]; + } + + @Override + public String getId() { + return "ID"; + } + + @Override + public String getLocale() { + return ""; + } + + @Override + public String getLinkTo() { + return ""; + } + + @Override + public IToc getToc() { + return new Toc(); + } + + @Override + public boolean isPrimary() { + return true; + } + + } + + public AssemblerTocProvider() { + } + + @Override + public ITocContribution[] getTocContributions(String locale) { + return new ITocContribution[] { new TocContribution() }; + } + + private static ITopic createTopic(String href) { + if (href == null) { + throw new IllegalArgumentException( + "Parameter 'href' must not be null."); + } + + String label; + String key = href; + try { + ResourceBundle resourceBundle; + resourceBundle = ResourceBundle.getBundle( + "com/wudsn/ide/asm/help/AssemblerTocProvider", + Locale.getDefault(), + AssemblerTocProvider.class.getClassLoader()); + label = resourceBundle.getString(key); + } catch (MissingResourceException ex) { + label = href + " - Text missing"; + AssemblerPlugin.getInstance().logError( + "Resource for enum value {0} is missing.", + new Object[] { key }, ex); + } + return createTopic("", label, href, null); + } + + // See[Bug 382599] Help: Icons not taken from IToc2/ITopic2 Implementations + // https://bugs.eclipse.org/bugs/show_bug.cgi?id=382599 + @SuppressWarnings("restriction") + private static ITopic createTopic(String icon, String label, String href, + ITopic[] subtopics) { + org.eclipse.help.internal.Topic result = new org.eclipse.help.internal.Topic(); + result.setAttribute(org.eclipse.help.internal.Topic.ATTRIBUTE_ICON, + icon); + result.setLabel(label); + result.setHref(href); + if (subtopics == null) { + subtopics = new ITopic2[0]; + } + result.appendChildren(subtopics); + return result; + } + + private static ITopic[] createTopicsArray(List<ITopic> topics) { + if (topics == null) { + throw new IllegalArgumentException( + "Parameter 'topics' must not be null."); + } + ITopic[] topicsArray; + topicsArray = new ITopic[topics.size()]; + topics.toArray(topicsArray); + return topicsArray; + } + + static ITopic[] createTopics() { + AssemblerPlugin assemblerPlugin = AssemblerPlugin.getInstance(); + CompilerRegistry compilerRegistry = assemblerPlugin + .getCompilerRegistry(); + List<CompilerDefinition> compilerDefinitions = compilerRegistry + .getCompilerDefinitions(); + + List<ITopic> ideTopics = createIDETopics(); + List<ITopic> assemblerTopics = createAssemblerTopics(compilerDefinitions); + List<ITopic> hardwareTopics = createHardwareTopics(); + List<ITopic> cpuTopics = createCPUTopics(); + + List<ITopic> topics = new ArrayList<ITopic>(); + + topics.add(createTopic("", Texts.TOC_IDE_TOPIC_LABEL, "", + createTopicsArray(ideTopics))); + topics.add(createTopic("", Texts.TOC_ASSEMBLERS_TOPIC_LABEL, "", + createTopicsArray(assemblerTopics))); + topics.add(createTopic("", Texts.TOC_HARDWARES_TOPIC_LABEL, "", + createTopicsArray(hardwareTopics))); + topics.add(createTopic("", Texts.TOC_CPUS_TOPIC_LABEL, "", + createTopicsArray(cpuTopics))); + + return createTopicsArray(topics); + } + + private static List<ITopic> createIDETopics() { + List<ITopic> topics = new ArrayList<ITopic>(); + + topics.add(createTopic("help/ide-tutorials.section.html")); + topics.add(createTopic("help/ide-features.section.html")); + topics.add(createTopic("help/ide-installation.section.html")); + topics.add(createTopic("help/ide-releases.section.html")); + topics.add(createTopic("help/ide-faq.section.html")); + topics.add(createTopic("help/ide-credits.section.html")); + return topics; + } + + private static List<ITopic> createAssemblerTopics( + List<CompilerDefinition> compilerDefinitions) { + if (compilerDefinitions == null) { + throw new IllegalArgumentException( + "Parameter 'compilerDefinitions' must not be null."); + } + int size = compilerDefinitions.size(); + List<ITopic> assemblerTopics = new ArrayList<ITopic>(size); + + for (int i = 0; i < size; i++) { + CompilerDefinition compilerDefinition = compilerDefinitions.get(i); + + String href = AssemblerHelpContentProducer.SCHEMA_COMPILER + + compilerDefinition.getId() + "/" + + AssemblerHelpContentProducer.SECTION_GENERAL + + AssemblerHelpContentProducer.EXTENSION; + + ITopic generalTopic = createTopic("", + Texts.TOC_ASSEMBLER_GENERAL_TOPIC_LABEL, href, null); + + href = AssemblerHelpContentProducer.SCHEMA_COMPILER + + compilerDefinition.getId() + "/" + + AssemblerHelpContentProducer.SECTION_INSTRUCTIONS + + AssemblerHelpContentProducer.EXTENSION; + ITopic opcodesTopic = createTopic("", + Texts.TOC_ASSEMBLER_INSTRUCTIONS_TOPIC_LABEL, href, null); + + AssemblerPreferences assemblerPreferences = AssemblerPlugin + .getInstance().getPreferences(); + String compilerExecutablePath = assemblerPreferences + .getCompilerExecutablePath(compilerDefinition.getId()); + + String icon = ""; + List<ITopic> manualTopics = new ArrayList<ITopic>(); + try { + File file = compilerDefinition + .getHelpFile(compilerExecutablePath); + + // Appending the help file path with the correct file + // extension allows in-place display for example for ".html" + // files. + String extension = file.getPath(); + int index = extension.lastIndexOf('.'); + if (index > 0) { + extension = extension.substring(index); + if (extension.equalsIgnoreCase(".pdf")) { + icon = "pdf"; + } + } else { + extension = ".html"; + } + + href = AssemblerHelpContentProducer.SCHEMA_COMPILER + + compilerDefinition.getId() + "/" + + AssemblerHelpContentProducer.SECTION_MANUAL + + extension; + + if (file.isDirectory()) { + File[] files = file.listFiles(); + if (files != null) { + for (File file2 : files) { + String encodedPath; + try { + encodedPath = URLEncoder.encode( + file2.getName(), "UTF-8"); + } catch (UnsupportedEncodingException ex) { + throw new IllegalArgumentException( + "Cannot encode file name '" + + file2.getName() + "'"); + } + + manualTopics + .add(createTopic( + "", + file2.getName(), + href + + "?" + + AssemblerHelpContentProducer.SECTION_MANUAL_FILE + + "=" + encodedPath, null)); + } + } + // if the file is folder, the manual does not have own + // content but only sub-topics + href = ""; + } + } catch (CoreException ex) { + href = AssemblerHelpContentProducer.SCHEMA_COMPILER + + compilerDefinition.getId() + "/" + + AssemblerHelpContentProducer.SECTION_MANUAL + ".html"; + } + + ITopic manualTopic = createTopic(icon, + Texts.TOC_ASSEMBLER_MANUAL_TOPIC_LABEL, href, + createTopicsArray(manualTopics)); + + assemblerTopics.add(createTopic("", compilerDefinition.getName(), + "", + new ITopic[] { generalTopic, opcodesTopic, manualTopic })); + } + return assemblerTopics; + } + + private static List<ITopic> createHardwareTopics() { + // Build hardware topics + List<ITopic> hardwareTopics = new ArrayList<ITopic>( + Hardware.values().length - 1); + for (Hardware hardware : Hardware.values()) { + if (hardware.equals(Hardware.GENERIC)) { + continue; + } + List<ITopic> chipTopics = new ArrayList<ITopic>(); + switch (hardware) { + + case ATARI2600: + chipTopics + .add(createTopic("help/www.qotile.net/minidig/docs/sizes.txt")); + chipTopics + .add(createTopic("help/www.qotile.net/minidig/docs/2600_mem_map.txt")); + chipTopics + .add(createTopic("help/www.qotile.net/minidig/docs/stella.pdf")); + chipTopics + .add(createTopic("help/www.qotile.net/minidig/docs/2600_advanced_prog_guide.txt")); + chipTopics + .add(createTopic("help/www.qotile.net/minidig/docs/tia_color.html")); + chipTopics + .add(createTopic("help/www.qotile.net/minidig/docs/2600_tia_map.txt")); + chipTopics + .add(createTopic("help/www.qotile.net/minidig/docs/2600_riot_map.txt")); + break; + + case ATARI8BIT: + chipTopics + .add(createTopic("help/www.oxyron.de/html/registers_antic.html")); + chipTopics + .add(createTopic("help/www.oxyron.de/html/registers_gtia.html")); + chipTopics + .add(createTopic("help/www.oxyron.de/html/registers_pokey.html")); + break; + case C64: + chipTopics + .add(createTopic("help/www.oxyron.de/html/registers_rec.html")); + chipTopics + .add(createTopic("help/www.oxyron.de/html/registers_sid.html")); + chipTopics + .add(createTopic("help/www.oxyron.de/html/registers_vic2.html")); + break; + default: + // Nothing available. + + } + String href = AssemblerHelpContentProducer.SCHEMA_HARDWARE + + hardware.name().toLowerCase() + + AssemblerHelpContentProducer.EXTENSION; + hardwareTopics.add(createTopic("", EnumUtility.getText(hardware), + href, createTopicsArray(chipTopics))); + } + return hardwareTopics; + } + + private static List<ITopic> createCPUTopics() { + List<ITopic> cpuTopics = new ArrayList<ITopic>(CPU.values().length - 1); + for (CPU cpu : CPU.values()) { + String href = AssemblerHelpContentProducer.SCHEMA_CPU + + cpu.name().toLowerCase() + + AssemblerHelpContentProducer.EXTENSION; + cpuTopics + .add(createTopic("", EnumUtility.getText(cpu), href, null)); + } + return cpuTopics; + } + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/AssemblerTocProvider.properties b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/AssemblerTocProvider.properties new file mode 100644 index 00000000..9045fd2a --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/AssemblerTocProvider.properties @@ -0,0 +1,23 @@ +help/ide-credits.section.html=Credits +help/ide-faq.section.html=FAQ +help/ide-features.section.html=Features +help/ide-installation.section.html=Installation +help/ide-releases.section.html=Releases +help/ide-tutorials.section.html=Video Tutorials + +help/www.oxyron.de/html/registers_antic.html=ANTIC Reference +help/www.oxyron.de/html/registers_gtia.html=GTIA Reference +help/www.oxyron.de/html/registers_pokey.html=POKEY Reference +help/www.oxyron.de/html/registers_rec.html=REC Reference +help/www.oxyron.de/html/registers_sid.html=SID Reference +help/www.oxyron.de/html/registers_ted.html=TED Reference +help/www.oxyron.de/html/registers_vic1.html=VIC I Reference +help/www.oxyron.de/html/registers_vic2.html=VIC II Reference + +help/www.qotile.net/minidig/docs/2600_advanced_prog_guide.txt=STELLA Programming Guide (Advanced) +help/www.qotile.net/minidig/docs/2600_mem_map.txt=Memory Map +help/www.qotile.net/minidig/docs/2600_riot_map.txt=RIOT Map +help/www.qotile.net/minidig/docs/2600_tia_map.txt=TIA Map +help/www.qotile.net/minidig/docs/sizes.txt=Cartridges +help/www.qotile.net/minidig/docs/stella.pdf=STELLA Programming Guide +help/www.qotile.net/minidig/docs/tia_color.html=TIA Colors \ No newline at end of file diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/AssemblerTocProvider_de_DE.properties b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/AssemblerTocProvider_de_DE.properties new file mode 100644 index 00000000..e8b8d714 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/AssemblerTocProvider_de_DE.properties @@ -0,0 +1,23 @@ +help/ide-credits.section.html=Danke an... +help/ide-faq.section.html=Häufig gestellte Fragen +help/ide-features.section.html=Funktionen +help/ide-installation.section.html=Installation +help/ide-releases.section.html=Versionen +help/ide-tutorials.section.html=Video Lehrgänge + +help/www.oxyron.de/html/registers_antic.html=ANTIC Referenz +help/www.oxyron.de/html/registers_gtia.html=GTIA Referenz +help/www.oxyron.de/html/registers_pokey.html=POKEY Referenz +help/www.oxyron.de/html/registers_rec.html=REC Referenz +help/www.oxyron.de/html/registers_sid.html=SID Referenz +help/www.oxyron.de/html/registers_ted.html=TED Referenz +help/www.oxyron.de/html/registers_vic1.html=VIC I Referenz +help/www.oxyron.de/html/registers_vic2.html=VIC II Referenz + +help/www.qotile.net/minidig/docs/2600_advanced_prog_guide.txt=STELLA Programmierleitfaden (erweitert) +help/www.qotile.net/minidig/docs/2600_mem_map.txt=Speicher Addressübersicht +help/www.qotile.net/minidig/docs/2600_riot_map.txt=RIOT Addressübersicht +help/www.qotile.net/minidig/docs/2600_tia_map.txt=TIA Addressübersicht +help/www.qotile.net/minidig/docs/sizes.txt=Steckmodule +help/www.qotile.net/minidig/docs/stella.pdf=STELLA Programmierleitfaden +help/www.qotile.net/minidig/docs/tia_color.html=TIA Farben \ No newline at end of file diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/HTMLConstants.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/HTMLConstants.java new file mode 100644 index 00000000..6f607208 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/HTMLConstants.java @@ -0,0 +1,45 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.help; + +/** + * HTML Constants. + * + * @author Peter Dell + * @since 1.6.3 + */ +final class HTMLConstants { + public static final String UTF8 = "UTF-8"; + + private static final String NL = "\n"; + private static final String DOC_TYPE = "<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Strict//EN\" \"http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd\">"; + private static final String CONTENT_TYPE = "<meta http-equiv=\"Content-Type\" content=\"text/html; charset=" + + UTF8 + "\" />"; + private static final String STYLE_SHEETS = "<link rel=\"STYLESHEET\" href=\"/help/content/org.eclipse.platform/book.css\" type=\"text/css\" />" + + NL + + "<link rel=\"stylesheet\" type=\"text/css\" media=\"all\" href=\"stylesheets/ide.css\"/>"; + + public static final String PREFIX = DOC_TYPE + NL + + "<html xmlns=\"http://www.w3.org/1999/xhtml\">" + NL + "<head>" + + NL + HTMLConstants.CONTENT_TYPE + NL + HTMLConstants.STYLE_SHEETS + + NL + "</head><body>"; + public static final String SUFFIX = "</body>" + NL + "</html>"; + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/HTMLWrapperInputStream.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/HTMLWrapperInputStream.java new file mode 100644 index 00000000..0c5d142b --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/HTMLWrapperInputStream.java @@ -0,0 +1,104 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.help; + +import java.io.ByteArrayInputStream; +import java.io.IOException; +import java.io.InputStream; +import java.io.UnsupportedEncodingException; + +/** + * Wraps plain text files into a HTML document envelope. + * + * @author Peter Dell + * @since 1.6.3 + * + */ +final class HTMLWrapperInputStream extends InputStream { + + private InputStream prefixInputStream; + private InputStream innerInputStream; + private InputStream suffixInputStream; + + public HTMLWrapperInputStream(String prefix, String suffix, + InputStream inputStream) { + if (prefix == null) { + throw new IllegalArgumentException( + "Parameter 'prefix' must not be null."); + } + if (suffix == null) { + throw new IllegalArgumentException( + "Parameter 'suffix' must not be null."); + } + if (inputStream == null) { + throw new IllegalArgumentException( + "Parameter 'inputStream' must not be null."); + } + + try { + prefixInputStream = new ByteArrayInputStream( + prefix.getBytes("UTF-8")); + + innerInputStream = inputStream; + suffixInputStream = new ByteArrayInputStream( + suffix.getBytes("UTF-8")); + } catch (UnsupportedEncodingException ex) { + throw new RuntimeException(ex); + } + + } + + @Override + public int read() throws IOException { + int result; + + if (prefixInputStream != null) { + result = prefixInputStream.read(); + if (result != -1) { +// System.out.print((char)result); + + return result; + } + prefixInputStream = null; + } + + if (innerInputStream != null) { + result = innerInputStream.read(); + if (result != -1) { +// System.out.print((char)result); + return result; + } + innerInputStream = null; + } + + if (suffixInputStream != null) { + result = suffixInputStream.read(); + if (result != -1) { +// System.out.print((char)result); + + return result; + } + suffixInputStream = null; + } + return -1; + + } + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/HTMLWriter.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/HTMLWriter.java new file mode 100644 index 00000000..8b38b805 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/help/HTMLWriter.java @@ -0,0 +1,211 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.help; + +import java.util.ArrayList; +import java.util.List; + +/** + * Convenience HTML writer. + * + * @author Peter Dell + * @since 1.6.3 + */ +final class HTMLWriter { + private static final String BORDER_STYLE = "border-style:solid;border-width:1px;border-collapse:collapse;"; + + private StringBuilder builder; + private List<String> stack; + private boolean tableBorder; + + public HTMLWriter() { + builder = new StringBuilder(); + stack = new ArrayList<String>(); + } + + public void begin(String tag, String attributes) { + if (tag == null) { + throw new IllegalArgumentException("Parameter 'tag' must not be null."); + } + builder.append("<"); + builder.append(tag); + + if (attributes != null) { + builder.append(" "); + builder.append(attributes); + + } else { + if ((tag.equals("th") || tag.equals("td")) && tableBorder) { + builder.append(" style=\"border:1px solid\""); + } + } + builder.append(">"); + stack.add(tag); + } + + public void end() { + if (stack.isEmpty()) { + throw new RuntimeException("No open tag: " + builder); + } + String tag = stack.remove(stack.size() - 1); + builder.append("</"); + builder.append(tag); + builder.append(">\n"); + + if (tag.equals("table")) { + tableBorder = false; + } + + } + + public void writeText(String text) { + if (text == null) { + throw new IllegalArgumentException("Parameter 'text' must not be null."); + } + builder.append(text); + } + + public void beginTable() { + beginTable(true); + } + + public void beginTable(boolean border) { + tableBorder = border; + + begin("table", "style=\"text-align:left;" + (tableBorder ? BORDER_STYLE : "") + "\""); + } + + public void beginTableRow() { + begin("tr", null); + } + + public void writeTableRow(String header, String text) { + if (header == null) { + throw new IllegalArgumentException("Parameter 'header' must not be null."); + } + if (text == null) { + throw new IllegalArgumentException("Parameter 'text' must not be null."); + } + beginTableRow(); + writeTableHeader(header); + writeTableCell(text); + end(); + } + + public void writeTableRowCode(String header, String text) { + beginTableRow(); + writeTableHeader(header); + begin("td", "style=\"vertical-align:text-top;font-family:Courier New, Courier, monospace;border:1px solid;\""); + + // Cell content must not be empty, otherwise the border is not + // displayed. + if (text.trim().length() == 0) { + text = " "; + } + builder.append(text); + end(); + end(); + } + + public void writeTableRowCode(String header, char character) { + String text = ""; + if (character >= 32) { + text = Character.toString(character); + } + writeTableRowCode(header, text); + } + + public void writeTableHeader(String text) { + if (text == null) { + throw new IllegalArgumentException("Parameter 'text' must not be null."); + } + begin("th", "style=\"" + (tableBorder ? BORDER_STYLE : "") + ";vertical-align:text-top;white-space:nowrap;\""); + builder.append(text); + end(); + } + + public void writeTableCell(String text) { + if (text == null) { + throw new IllegalArgumentException("Parameter 'text' must not be null."); + } + begin("td", "style=\"" + (tableBorder ? BORDER_STYLE : "") + ";vertical-align:text-top;white-space:nowrap;\""); + builder.append(text); + end(); + } + + public void beginList() { + begin("ul", null); + } + + public void writeListItem(String text) { + if (text == null) { + throw new IllegalArgumentException("Parameter 'text' must not be null."); + } + begin("li", null); + builder.append(text); + end(); + } + + public String toHTML() { + if (!stack.isEmpty()) { + throw new IllegalStateException("There are still open tags: " + stack + "\n" + builder.toString()); + } + return builder.toString(); + } + + public static String getImage(String src, String alt, String text) { + if (src == null) { + throw new IllegalArgumentException("Parameter 'src' must not be null."); + } + if (alt == null) { + throw new IllegalArgumentException("Parameter 'alt' must not be null."); + } + if (text == null) { + throw new IllegalArgumentException("Parameter 'text' must not be null."); + } + return "<img style=\"vertical-align:middle\" src=\"" + src + "\" alt=\"" + alt + "\"/> " + text; + } + + public static String getLink(String href, String text) { + if (href == null) { + throw new IllegalArgumentException("Parameter 'href' must not be null."); + } + + if (text == null) { + throw new IllegalArgumentException("Parameter 'text' must not be null."); + } + return "<a href=\"" + href + "\" >" + text + "</a>"; + } + + public static String getString(List<String> list) { + if (list == null) { + throw new IllegalArgumentException("Parameter 'list' must not be null."); + } + StringBuilder builder = new StringBuilder(); + int size = list.size(); + for (int i = 0; i < size; i++) { + builder.append(list.get(i)); + if (i < size - 1) { + builder.append(" "); + } + } + return builder.toString(); + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/AssemblerPreferences.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/AssemblerPreferences.java new file mode 100644 index 00000000..10796a07 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/AssemblerPreferences.java @@ -0,0 +1,192 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.preferences; + +import org.eclipse.jface.preference.IPreferenceStore; +import org.eclipse.jface.text.TextAttribute; + +import com.wudsn.ide.asm.Hardware; +import com.wudsn.ide.asm.editor.AssemblerContentAssistProcessorDefaultCase; +import com.wudsn.ide.asm.editor.AssemblerEditorCompileCommandPositioningMode; +import com.wudsn.ide.base.common.AbstractIDEPlugin; +import com.wudsn.ide.base.common.StringUtility; + +/** + * Facade class for typed access to the plugin preferences. + * + * @author Peter Dell + */ +public final class AssemblerPreferences { + + /** + * The preference store to which all calls are delegated. + */ + private IPreferenceStore preferenceStore; + + /** + * Created by {@link AbstractIDEPlugin} only. + * + * @param preferenceStore + * The preference store, not <code>null</code>. + */ + public AssemblerPreferences(IPreferenceStore preferenceStore) { + if (preferenceStore == null) { + throw new IllegalArgumentException( + "Parameter 'preferenceStore' must not be null."); + } + this.preferenceStore = preferenceStore; + } + + /** + * Gets the text attribute for a token type. + * + * @param name + * The name of the preferences for the token type, see + * {@link AssemblerPreferencesConstants}. + * + * @return The text attribute, not <code>null</code>. + */ + public TextAttribute getEditorTextAttribute(String name) { + if (name == null) { + throw new IllegalArgumentException( + "Parameter 'name' must not be null."); + } + return TextAttributeConverter.fromString(preferenceStore + .getString(name)); + } + + /** + * Gets the default case content assist. + * + * @return The default case content assist, may be empty, not + * <code>null</code>. See + * {@link AssemblerContentAssistProcessorDefaultCase}. + */ + public String getEditorContentAssistProcessorDefaultCase() { + return getString(AssemblerPreferencesConstants.EDITOR_CONTENT_ASSIST_PROCESSOR_DEFAULT_CASE); + } + + /** + * Gets the compile command positioning mode. + * + * @return The positioning mode, may be empty, not <code>null</code>. See + * {@link AssemblerEditorCompileCommandPositioningMode}. + * @since 1.6.1 + */ + public String getEditorCompileCommandPositioningMode() { + return getString(AssemblerPreferencesConstants.EDITOR_COMPILE_COMMAND_POSITIONING_MODE); + } + + /** + * Gets the executable path for the compiler. + * + * @param compilerId + * The compiler id, not empty and not <code>null</code>. + * + * @return The executable path for the compiler, may be empty, not + * <code>null</code>. + */ + public String getCompilerExecutablePath(String compilerId) { + if (compilerId == null) { + throw new IllegalArgumentException( + "Parameter 'compilerId' must not be null."); + } + if (StringUtility.isEmpty(compilerId)) { + throw new IllegalArgumentException( + "Parameter 'compilerId' must not be empty."); + } + return getString(AssemblerPreferencesConstants + .getCompilerExecutablePathName(compilerId)); + } + + /** + * Gets the preferences for a compiler. + * + * @param compilerId + * The compiler id, not empty and not <code>null</code>. + * @param hardware + * The preferences or <code>null</code> if the compiler is not + * active for that hardware. + * + * @return The compiler preferences, not <code>null</code>. + */ + public CompilerPreferences getCompilerPreferences(String compilerId, + Hardware hardware) { + if (compilerId == null) { + throw new IllegalArgumentException( + "Parameter 'compilerId' must not be null."); + } + if (StringUtility.isEmpty(compilerId)) { + throw new IllegalArgumentException( + "Parameter 'compilerId' must not be empty."); + } + if (hardware == null) { + throw new IllegalArgumentException( + "Parameter 'hardware' must not be null."); + } + return new CompilerPreferences(this, compilerId, hardware); + + } + + /** + * Gets the current value of the string-valued preference with the given + * name. Returns the default-default value (the empty string <code>""</code> + * ) if there is no preference with the given name, or if the current value + * cannot be treated as a string. + * + * @param name + * The name of the preference, not <code>null</code>. + * @return The preference value, may be empty, not <code>null</code>. + */ + String getString(String name) { + if (name == null) { + throw new IllegalArgumentException( + "Parameter 'key' must not be null."); + } + String result; + result = preferenceStore.getString(name); + if (result == null) { + result = ""; + } else { + result = result.trim(); + } + + return result; + } + + /** + * Gets the current value of the boolean preference with the given name. + * Returns the default-default value <code>false</code> if there is no + * preference with the given name, or if the current value cannot be treated + * as a boolean. + * + * @param name + * The name of the preference, not <code>null</code>. + * @return The preference value. + */ + boolean getBoolean(String name) { + if (name == null) { + throw new IllegalArgumentException( + "Parameter 'key' must not be null."); + } + return preferenceStore.getBoolean(name); + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesChangeListener.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesChangeListener.java new file mode 100644 index 00000000..8a3f4f18 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesChangeListener.java @@ -0,0 +1,43 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.preferences; + +import java.util.Set; + +/** + * Listener interface for global preferences changes. + * + * @author Peter Dell + * + * @sine 1.7.0 + */ +public interface AssemblerPreferencesChangeListener { + + /** + * Notify of changed properties. + * @param preferences + * The assembler preferences containing the new values, not + * <code>null</code>. + * @param changedPropertyNames + * The set of names of properties that have been changed, may be + * empty, not <code>null</code>. + */ + public void preferencesChanged(AssemblerPreferences preferences, Set<String> changedPropertyNames); +} \ No newline at end of file diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesCompilersPage.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesCompilersPage.java new file mode 100644 index 00000000..12997cd4 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesCompilersPage.java @@ -0,0 +1,653 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.preferences; + +import java.util.ArrayList; +import java.util.List; +import java.util.Map; +import java.util.TreeMap; + +import org.eclipse.jface.fieldassist.ControlDecoration; +import org.eclipse.jface.preference.BooleanFieldEditor; +import org.eclipse.jface.preference.ComboFieldEditor; +import org.eclipse.jface.preference.FieldEditor; +import org.eclipse.jface.preference.FieldEditorPreferencePage; +import org.eclipse.jface.preference.IPreferenceStore; +import org.eclipse.jface.preference.RadioGroupFieldEditor; +import org.eclipse.jface.preference.StringFieldEditor; +import org.eclipse.jface.util.IPropertyChangeListener; +import org.eclipse.jface.util.PropertyChangeEvent; +import org.eclipse.swt.SWT; +import org.eclipse.swt.events.SelectionEvent; +import org.eclipse.swt.events.SelectionListener; +import org.eclipse.swt.layout.GridData; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Control; +import org.eclipse.swt.widgets.Label; +import org.eclipse.swt.widgets.TabFolder; +import org.eclipse.swt.widgets.TabItem; +import org.eclipse.swt.widgets.Text; +import org.eclipse.ui.IEditorPart; +import org.eclipse.ui.IWorkbench; +import org.eclipse.ui.IWorkbenchPreferencePage; + +import com.wudsn.ide.asm.AssemblerPlugin; +import com.wudsn.ide.asm.CPU; +import com.wudsn.ide.asm.Hardware; +import com.wudsn.ide.asm.Texts; +import com.wudsn.ide.asm.compiler.CompilerDefinition; +import com.wudsn.ide.asm.compiler.CompilerOutputFolderMode; +import com.wudsn.ide.asm.compiler.CompilerRegistry; +import com.wudsn.ide.asm.editor.AssemblerEditor; +import com.wudsn.ide.asm.runner.RunnerDefinition; +import com.wudsn.ide.asm.runner.RunnerId; +import com.wudsn.ide.asm.runner.RunnerRegistry; +import com.wudsn.ide.base.common.EnumUtility; +import com.wudsn.ide.base.common.ProcessWithLogs; +import com.wudsn.ide.base.common.StringUtility; +import com.wudsn.ide.base.common.TextUtility; +import com.wudsn.ide.base.gui.SWTFactory; + +/** + * Visual editor page for the assembler preferences regarding compilers. There + * is a separate page per {@link Hardware}. Subclasses only implement the + * constructor. + * + * @author Peter Dell + */ +public abstract class AssemblerPreferencesCompilersPage extends FieldEditorPreferencePage implements + IWorkbenchPreferencePage { + + private static final class Tab { + + public final String compilerId; + public final int tabIndex; + public final TabItem tabItem; + public final Control enabledControl; + public final Control disabledControl; + public final List<ControlDecoration> controlDecorations; + public boolean initialized; + public boolean enabled; + + public Tab(String compilerId, int tabIndex, TabItem tabItem, Control enabledControl, Control disabledControl, + List<ControlDecoration> controlDecorations) { + this.compilerId = compilerId; + this.tabIndex = tabIndex; + this.tabItem = tabItem; + this.enabledControl = enabledControl; + this.disabledControl = disabledControl; + this.controlDecorations = controlDecorations; + initialized = false; + enabled = false; + } + } + + /** + * Local workaround class to react on changes of a radio group field editor. + * By default the selection listener is set to the containing page, so we + * need a second listener mechanism. + */ + private final class RadioGroupFieldEditorWithAction extends RadioGroupFieldEditor { + + private IPropertyChangeListener propertyChangeListener; + + public RadioGroupFieldEditorWithAction(String name, String labelText, int numColumns, + String[][] labelAndValues, Composite parent) { + super(name, labelText, numColumns, labelAndValues, parent); + } + + @Override + protected void fireValueChanged(String property, Object oldValue, Object newValue) { + super.fireValueChanged(property, oldValue, newValue); + if (propertyChangeListener != null) { + propertyChangeListener.propertyChange(new PropertyChangeEvent(this, property, oldValue, newValue)); + } + + } + + public void setAdditionalPropertyChangeListener(IPropertyChangeListener propertyChangeListener) { + this.propertyChangeListener = propertyChangeListener; + + } + } + + /** + * Property change listener to set the enabled state of the output folder + * path field based on the output folder mode field. + */ + private final class OutputFolderModeChangeListener implements IPropertyChangeListener { + + private Composite outputFolderPathFieldEditorParent; + private StringFieldEditor outputFolderPathFieldEditor; + + public OutputFolderModeChangeListener(Composite outputFolderPathFieldEditorParent, + StringFieldEditor outputFolderPathFieldEditor) { + if (outputFolderPathFieldEditorParent == null) { + throw new IllegalArgumentException("Parameter 'outputFolderPathFieldEditorParent' must not be null."); + } + if (outputFolderPathFieldEditor == null) { + throw new IllegalArgumentException("Parameter 'outputFolderPathFieldEditor' must not be null."); + } + this.outputFolderPathFieldEditorParent = outputFolderPathFieldEditorParent; + this.outputFolderPathFieldEditor = outputFolderPathFieldEditor; + } + + @Override + public void propertyChange(PropertyChangeEvent event) { + if (event == null) { + throw new IllegalArgumentException("Parameter 'event' must not be null."); + } + setOutputFolderMode((String) event.getNewValue()); + } + + public void setOutputFolderMode(String newValue) { + boolean enabled; + enabled = CompilerOutputFolderMode.FIXED_FOLDER.equals(newValue); + outputFolderPathFieldEditor.setEnabled(enabled, outputFolderPathFieldEditorParent); + } + + } + + /** + * The type of hardware used to filter the compilers and emulators. + */ + final Hardware hardware; + + /** + * The owning plugin. + */ + private final AssemblerPlugin plugin; + + /** + * The tab folder and all visible tab items. + */ + private TabFolder tabFolder; + private final Map<String, Tab> tabs; + + /** + * The id of the compiler and runner to be used as default. + */ + private String activeCompilerId; + private String activeRunnerId; + + /** + * Creation is protected for sub-classes. + * + * @param hardware + * The type of hardware used to filter the compilers and + * emulators, not <code>null</code>. + */ + protected AssemblerPreferencesCompilersPage(Hardware hardware) { + super(GRID); + if (hardware == null) { + throw new IllegalArgumentException("Parameter 'hardware' must not be null."); + } + this.hardware = hardware; + plugin = AssemblerPlugin.getInstance(); + IPreferenceStore preferencesStore = plugin.getPreferenceStore(); + setPreferenceStore(preferencesStore); + + tabs = new TreeMap<String, Tab>(); + } + + @Override + public final void init(IWorkbench workbench) { + if (workbench == null) { + throw new IllegalArgumentException("Parameter 'workbench' must not be null."); + } + IEditorPart editor = workbench.getActiveWorkbenchWindow().getActivePage().getActiveEditor(); + if (editor instanceof AssemblerEditor) { + AssemblerEditor assemblerEditor; + assemblerEditor = (AssemblerEditor) editor; + activeCompilerId = assemblerEditor.getCompilerDefinition().getId(); + activeRunnerId = assemblerEditor.getCompilerPreferences().getRunnerId(); + } else { + activeCompilerId = ""; + activeRunnerId = ""; + } + } + + @Override + public final void createFieldEditors() { + + Composite parent = getFieldEditorParent(); + GridData gridData = new GridData(); + gridData.verticalIndent = 0; + gridData.horizontalIndent = 0; + parent.setLayoutData(gridData); + + createCompilerFieldEditors(parent); + setTabsStatus(); + } + + private void createCompilerFieldEditors(Composite parent) { + if (parent == null) { + throw new IllegalArgumentException("Parameter 'parent' must not be null."); + } + + // Create the editors for all compilers of the hardware. + CompilerRegistry compilerRegistry = plugin.getCompilerRegistry(); + List<CompilerDefinition> compilerDefinitions = compilerRegistry.getCompilerDefinitions(); + + tabFolder = new TabFolder(parent, SWT.FLAT); + for (CompilerDefinition compilerDefinition : compilerDefinitions) { + + createTabItem(tabFolder, compilerDefinition); + } + + // Default to tab item for active compiler or to first. + TabItem selectedTabItem = null; + if (activeCompilerId != null) { + Tab selectedTab = tabs.get(activeCompilerId); + if (selectedTab != null) { + selectedTabItem = selectedTab.tabItem; + } + } + if (selectedTabItem == null && tabFolder.getItemCount() > 0) { + selectedTabItem = tabFolder.getItem(0); + } + if (selectedTabItem != null) { + tabFolder.setSelection(selectedTabItem); + } + + // Make sure the control decorations are updated as required + tabFolder.addSelectionListener(new SelectionListener() { + + @Override + public void widgetSelected(SelectionEvent e) { + setTabsStatus(); + } + + @Override + public void widgetDefaultSelected(SelectionEvent e) { + } + }); + } + + private void createTabItem(TabFolder tabFolder, CompilerDefinition compilerDefinition) { + if (tabFolder == null) { + throw new IllegalArgumentException("Parameter 'tabFolder' must not be null."); + } + if (compilerDefinition == null) { + throw new IllegalArgumentException("Parameter 'compilerDefinition' must not be null."); + } + + String[][] labelsAndValues; + labelsAndValues = new String[][] { + { Texts.PREFERENCES_COMPILER_OUTPUT_FOLDER_MODE_SOURCE_FOLDER_TEXT, + CompilerOutputFolderMode.SOURCE_FOLDER }, + { Texts.PREFERENCES_COMPILER_OUTPUT_FOLDER_MODE_TEMP_FOLDER_TEXT, CompilerOutputFolderMode.TEMP_FOLDER }, + { Texts.PREFERENCES_COMPILER_OUTPUT_FOLDER_MODE_FIXED_FOLDER_TEXT, + CompilerOutputFolderMode.FIXED_FOLDER } + + }; + + String compilerId = compilerDefinition.getId(); + TabItem tabItem = new TabItem(tabFolder, SWT.NONE); + tabItem.setText(compilerDefinition.getName()); + + Composite tabContent; + tabContent = SWTFactory.createComposite(tabFolder, 1, 1, GridData.FILL_BOTH); + + List<ControlDecoration> controlDecorations; + controlDecorations = new ArrayList<ControlDecoration>(); + + Composite composite; + + // Field: cpu + composite = SWTFactory.createComposite(tabContent, 2, 3, GridData.FILL_HORIZONTAL); + + // Filtering of CPU based on hardware is currently not implemented + // because expansion boards like a W65816 board might be there/added + // for a hardware. + List<CPU> cpus = compilerDefinition.getSupportedCPUs(); + String[][] entryNamesAndValues = new String[cpus.size()][]; + int i = 0; + for (CPU cpu : cpus) { + entryNamesAndValues[i] = new String[2]; + entryNamesAndValues[i][1] = cpu.name(); + entryNamesAndValues[i][0] = EnumUtility.getText(cpu); + i++; + } + + FieldEditor comboFieldEditor = new ComboFieldEditor(AssemblerPreferencesConstants.getCompilerCPUName( + compilerId, hardware), Texts.PREFERENCES_COMPILER_CPU_LABEL, entryNamesAndValues, composite); + comboFieldEditor.setEnabled(entryNamesAndValues.length > 1, composite); + addField(comboFieldEditor); + + String name; + + // Field: defaultParameters + composite = SWTFactory.createComposite(tabContent, 2, 2, GridData.FILL_HORIZONTAL); + + Label label = new Label(composite, SWT.LEFT); + label.setText(Texts.PREFERENCES_COMPILER_DEFAULT_PARAMETERS_LABEL); + Text textField = new Text(composite, SWT.SINGLE | SWT.BORDER); + textField.setEditable(false); + textField.setText(compilerDefinition.getDefaultParameters()); + GridData gd = new GridData(); + gd.horizontalSpan = 2; + gd.horizontalAlignment = GridData.FILL; + gd.grabExcessHorizontalSpace = true; + textField.setLayoutData(gd); + + // Field: parameters + composite = SWTFactory.createComposite(tabContent, 2, 2, GridData.FILL_HORIZONTAL); + StringFieldEditor parametersFieldEditor; + name = AssemblerPreferencesConstants.getCompilerParametersName(compilerId, hardware); + parametersFieldEditor = new StringFieldEditor(name, Texts.PREFERENCES_COMPILER_PARAMETERS_LABEL, tabContent); + + String compilerParametersHelp = Texts.PREFERENCES_COMPILER_PARAMETERS_HELP + "\n" + + Texts.PREFERENCES_COMPILER_PARAMETERS_VARIABLES; + controlDecorations.add(createHelpDecoration(parametersFieldEditor, tabContent, compilerParametersHelp)); + + gd = new GridData(); + gd.horizontalSpan = 1; + gd.horizontalAlignment = GridData.FILL; + gd.grabExcessHorizontalSpace = true; + gd.horizontalIndent = 14; + parametersFieldEditor.getTextControl(tabContent).setLayoutData(gd); + + addField(parametersFieldEditor); + + // Field: outputFolderMode + composite = SWTFactory.createComposite(tabContent, 2, 2, GridData.FILL_HORIZONTAL); + RadioGroupFieldEditorWithAction outputFolderModeChoiceEditor = new RadioGroupFieldEditorWithAction( + AssemblerPreferencesConstants.getCompilerOutputFolderModeName(compilerId, hardware), + Texts.PREFERENCES_COMPILER_OUTPUT_FOLDER_MODE_LABEL, 3, labelsAndValues, composite); + addField(outputFolderModeChoiceEditor); + + // Field: outputFolderPath + composite = SWTFactory.createComposite(tabContent, 2, 2, GridData.FILL_HORIZONTAL); + StringFieldEditor outputFolderPathFieldEditor; + outputFolderPathFieldEditor = new DirectoryFieldDownloadEditor( + AssemblerPreferencesConstants.getCompilerOutputFolderPathName(compilerId, hardware), + Texts.PREFERENCES_COMPILER_OUTPUT_FOLDER_PATH_LABEL, composite); + addField(outputFolderPathFieldEditor); + + // Create a connection between the output mode field and the output + // path field. + OutputFolderModeChangeListener outputFolderModeChangeListener; + outputFolderModeChangeListener = new OutputFolderModeChangeListener(composite, outputFolderPathFieldEditor); + // Set initial status based on current output folder mode. + outputFolderModeChangeListener.setOutputFolderMode(getPreferenceStore().getString( + outputFolderModeChoiceEditor.getPreferenceName())); + // Register for changes. + outputFolderModeChoiceEditor.setAdditionalPropertyChangeListener(outputFolderModeChangeListener); + + // Field: outputFileExtension + composite = SWTFactory.createComposite(tabContent, 2, 2, GridData.FILL_HORIZONTAL); + StringFieldEditor outputFileExtensionFieldEditor; + outputFileExtensionFieldEditor = new StringFieldEditor( + AssemblerPreferencesConstants.getCompilerOutputFileExtensionName(compilerId, hardware), + Texts.PREFERENCES_COMPILER_OUTPUT_FILE_EXTENSION_LABEL, composite); + + gd = new GridData(SWT.BEGINNING, SWT.FILL, true, false); + gd.widthHint = convertWidthInCharsToPixels(5); + outputFileExtensionFieldEditor.getTextControl(composite).setLayoutData(gd); + outputFileExtensionFieldEditor.getTextControl(composite); + addField(outputFileExtensionFieldEditor); + + composite = SWTFactory.createComposite(tabContent, 1, 2, GridData.FILL_HORIZONTAL); + + RunnerRegistry runnerRegistry = plugin.getRunnerRegistry(); + List<RunnerDefinition> runnerDefinitions; + runnerDefinitions = runnerRegistry.getDefinitions(hardware); + entryNamesAndValues = new String[runnerDefinitions.size()][]; + i = 0; + for (RunnerDefinition runnerDefinition : runnerDefinitions) { + entryNamesAndValues[i] = new String[2]; + entryNamesAndValues[i][1] = runnerDefinition.getId(); + entryNamesAndValues[i][0] = runnerDefinition.getName(); + i++; + } + comboFieldEditor = new ComboFieldEditor(AssemblerPreferencesConstants.getCompilerRunnerIdName(compilerId, + hardware), Texts.PREFERENCES_COMPILER_RUNNER_ID_LABEL, entryNamesAndValues, composite); + addField(comboFieldEditor); + createRunnerFieldEdiors(compilerId, composite, controlDecorations); + + Composite disabledControl = SWTFactory.createComposite(tabFolder, 1, 1, GridData.FILL_BOTH); + label = new Label(disabledControl, SWT.NONE); + label.setText(TextUtility.format(Texts.MESSAGE_E100, compilerDefinition.getName())); + Tab tab = new Tab(compilerId, tabs.size(), tabItem, tabContent, disabledControl, controlDecorations); + tabs.put(compilerId, tab); + + } + + void setTabsStatus() { + for (Tab tab : tabs.values()) { + setTabStatus(tab); + + } + // tabFolder.layout(); + // tabFolder.getParent().getParent().redraw(); + } + + private void setTabStatus(Tab tab) { + if (tab == null) { + throw new IllegalArgumentException("Parameter 'tab' must not be null."); + } + + AssemblerPreferences assemblerPreferences = plugin.getPreferences(); + + boolean enabled = StringUtility.isSpecified(assemblerPreferences.getCompilerExecutablePath(tab.compilerId)); + + if (!tab.initialized || enabled != tab.enabled) { + tab.initialized = true; + tab.enabled = enabled; + if (enabled) { + tab.tabItem.setControl(tab.enabledControl); + } else { + tab.tabItem.setControl(tab.disabledControl); + } + tab.disabledControl.setVisible(!enabled); + tab.enabledControl.setVisible(enabled); + } + boolean tabActive = tab.tabIndex == tabFolder.getSelectionIndex(); + for (ControlDecoration controlDecoration : tab.controlDecorations) { + if (enabled && tabActive) { + controlDecoration.show(); + } else { + controlDecoration.hide(); + } + } + + } + + private void createRunnerFieldEdiors(String compilerId, Composite parent, List<ControlDecoration> controlDecorations) { + if (compilerId == null) { + throw new IllegalArgumentException("Parameter 'compilerId' must not be null."); + } + if (parent == null) { + throw new IllegalArgumentException("Parameter 'parent' must not be null."); + } + if (controlDecorations == null) { + throw new IllegalArgumentException("Parameter 'controlDecorations' must not be null."); + } + + TabFolder tabFolder = new TabFolder(parent, SWT.NONE); + TabItem selectedTabItem = null; + GridData gd; + + gd = new GridData(); + gd.horizontalSpan = 1; + gd.horizontalAlignment = GridData.FILL; + gd.grabExcessHorizontalSpace = true; + tabFolder.setLayoutData(gd); + + RunnerRegistry runnerRegistry = plugin.getRunnerRegistry(); + List<RunnerDefinition> runnerDefinitions; + runnerDefinitions = runnerRegistry.getDefinitions(hardware); + + String runnerCommandLineHelp = Texts.PREFERENCES_COMPILER_RUNNER_COMMAND_LINE_HELP + "\n" + + Texts.PREFERENCES_COMPILER_RUNNER_COMMAND_LINE_VARIABLES; + + for (RunnerDefinition runnerDefinition : runnerDefinitions) { + + String runnerId = runnerDefinition.getId(); + + if (runnerId.equals(RunnerId.DEFAULT_APPLICATION)) { + continue; + } + + TabItem tabItem = new TabItem(tabFolder, SWT.NONE); + tabItem.setText(runnerDefinition.getName()); + + Composite tabContent; + tabContent = SWTFactory.createComposite(tabFolder, 2, 1, GridData.FILL_BOTH); + + String name = AssemblerPreferencesConstants.getCompilerRunnerExecutablePathName(compilerId, hardware, + runnerId); + + Composite composite; + composite = SWTFactory.createComposite(tabContent, 4, 2, GridData.FILL_HORIZONTAL); + FileFieldDownloadEditor fileFieldEditor = new FileFieldDownloadEditor(name, + Texts.PREFERENCES_COMPILER_RUNNER_EXECUTABLE_PATH_LABEL, composite); + fileFieldEditor.setFileExtensions(ProcessWithLogs.getExecutableExtensions()); + fileFieldEditor.setEnabled(runnerDefinition.isRunnerExecutablePathPossible(), composite); + addField(fileFieldEditor); + + // Field: defaultEmulatorParameters + composite = SWTFactory.createComposite(tabContent, 2, 2, GridData.FILL_HORIZONTAL); + + Label label = new Label(composite, SWT.LEFT); + label.setText(Texts.PREFERENCES_COMPILER_RUNNER_DEFAULT_COMMAND_LINE_LABEL); + Text textField = new Text(composite, SWT.SINGLE | SWT.BORDER); + textField.setEditable(false); + textField.setText(runnerDefinition.getDefaultCommandLine()); + gd = new GridData(); + gd.horizontalSpan = 2; + gd.horizontalAlignment = GridData.FILL; + gd.grabExcessHorizontalSpace = true; + textField.setLayoutData(gd); + + // Field: parameters + composite = SWTFactory.createComposite(tabContent, 2, 2, GridData.FILL_HORIZONTAL); + StringFieldEditor commandLineFieldEditor; + name = AssemblerPreferencesConstants.getCompilerRunnerCommandLineName(compilerId, hardware, runnerId); + commandLineFieldEditor = new StringFieldEditor(name, Texts.PREFERENCES_COMPILER_RUNNER_COMMAND_LINE_LABEL, + tabContent); + + gd = new GridData(); + gd.horizontalSpan = 1; + gd.horizontalAlignment = GridData.FILL; + gd.grabExcessHorizontalSpace = true; + gd.horizontalIndent = 14; + commandLineFieldEditor.getTextControl(tabContent).setLayoutData(gd); + addField(commandLineFieldEditor); + + controlDecorations.add(createHelpDecoration(commandLineFieldEditor, tabContent, runnerCommandLineHelp)); + String url = runnerDefinition.getHomePageURL(); + fileFieldEditor.setLinkURL(url); + + // Field: illegalOpcodesVisible + composite = SWTFactory.createComposite(tabContent, 2, 3, GridData.FILL_HORIZONTAL); + FieldEditor booleanFieldEditor = new BooleanFieldEditor( + AssemblerPreferencesConstants + .getCompilerRunnerWaitForCompletionName(compilerId, hardware, runnerId), + Texts.PREFERENCES_COMPILER_RUNNER_WAIT_FOR_COMPLETION_LABEL, composite); + + addField(booleanFieldEditor); + + tabItem.setControl(tabContent); + + if (runnerId.equals(activeRunnerId)) { + selectedTabItem = tabItem; + } + + } + + // Default to selected tab item. + if (selectedTabItem != null) { + tabFolder.setSelection(selectedTabItem); + } + } + + private ControlDecoration createHelpDecoration(StringFieldEditor parametersFieldEditor, Composite tabContent, + String text) { + if (parametersFieldEditor == null) { + throw new IllegalArgumentException("Parameter 'parametersFieldEditor' must not be null."); + } + if (tabContent == null) { + throw new IllegalArgumentException("Parameter 'tabContent' must not be null."); + } + if (text == null) { + throw new IllegalArgumentException("Parameter 'text' must not be null."); + } + Text textControl = parametersFieldEditor.getTextControl(tabContent); + ControlDecoration controlDecoration = new ControlDecoration(textControl, SWT.LEFT | SWT.CENTER); + controlDecoration.hide(); + controlDecoration.setShowHover(true); + controlDecoration.setDescriptionText(text); + + controlDecoration.setImage(AssemblerPlugin.getInstance().getImage("help-16x16.gif")); + + return controlDecoration; + } + + /** + * {@inheritDoc} + */ + @Override + public final boolean performOk() { + if (super.performOk()) { + saveChanges(); + return true; + } + return false; + } + + /** + * The field editor preference page implementation of a + * <code>PreferencePage</code> method loads all the field editors with their + * default values except for the executable paths. + */ + @Override + protected final void performDefaults() { + + super.performDefaults(); + + } + + @Override + public final void dispose() { + super.dispose(); + + } + + @Override + public void setVisible(boolean visible) { + super.setVisible(visible); + + if (visible) { + setTabsStatus(); + } + + } + + /** + * Saves all changes to the {@link IPreferenceStore}. + */ + private void saveChanges() { + + plugin.savePreferences(); + + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesConstants.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesConstants.java new file mode 100644 index 00000000..054d5f6a --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesConstants.java @@ -0,0 +1,419 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.preferences; + +import java.util.HashSet; +import java.util.Set; + +import com.wudsn.ide.asm.Hardware; + +/** + * Constants for preferences. + * + * @author Peter Dell + */ +public final class AssemblerPreferencesConstants { + + /** + * Creation is private. + */ + private AssemblerPreferencesConstants() { + } + + /** + * Preference key for comment text style. + */ + public static final String EDITOR_TEXT_ATTRIBUTE_COMMENT = "editor.text.attribute.comment"; //$NON-NLS-1$ + + /** + * Preferences key for string text style. + */ + public static final String EDITOR_TEXT_ATTRIBUTE_STRING = "editor.text.attribute.string"; //$NON-NLS-1$ + + /** + * Preferences key for number text style. + */ + public static final String EDITOR_TEXT_ATTRIBUTE_NUMBER = "editor.text.attribute.number"; //$NON-NLS-1$ + + /** + * Preference key for directive text style. + */ + public static final String EDITOR_TEXT_ATTRIBUTE_DIRECTVE = "editor.text.attribute.directive"; //$NON-NLS-1$ + + /** + * Preference key for legal opcode text style. + */ + public static final String EDITOR_TEXT_ATTRIBUTE_OPCODE_LEGAL = "editor.text.attribute.opcode.legal"; //$NON-NLS-1$ + + /** + * Preference key for illegal opcode text style. + */ + public static final String EDITOR_TEXT_ATTRIBUTE_OPCODE_ILLEGAL = "editor.text.attribute.opcode.illegal"; //$NON-NLS-1$ + + /** + * Preference key for pseudo opcode text style. + */ + public static final String EDITOR_TEXT_ATTRIBUTE_OPCODE_PSEUDO = "editor.text.attribute.opcode.pseudo"; //$NON-NLS-1$ + + /** + * Preference key for equate identifier text style. + */ + public static final String EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_EQUATE = "editor.text.attribute.identifier.equate"; //$NON-NLS-1$ + + /** + * Preference key for label identifier text style. + */ + public static final String EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_LABEL = "editor.text.attribute.identifier.label"; //$NON-NLS-1$ + + /** + * Preference key for enum identifier text style. + */ + public static final String EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_ENUM_DEFINITION_SECTION = "editor.text.attribute.identifier.enum"; //$NON-NLS-1$ + + /** + * Preference key for structure identifier text style. + */ + public static final String EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_STRUCTURE_DEFINITION_SECTION = "editor.text.attribute.identifier.structure"; //$NON-NLS-1$ + + /** + * Preference key for local identifier text style. + */ + public static final String EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_LOCAL_SECTION = "editor.text.attribute.identifier.local"; //$NON-NLS-1$ + + /** + * Preference key for macro identifier text style. + */ + public static final String EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_MACRO_DEFINITION_SECTION = "editor.text.attribute.identifier.macro"; //$NON-NLS-1$ + + /** + * Preference key for procedure identifier text style. + */ + public static final String EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_PROCEDURE_DEFINITION_SECTION = "editor.text.attribute.identifier.procedure"; //$NON-NLS-1$ + + /** + * Set of all preferences keys that depend on the global JFact text font + * setting. + */ + public static final Set<String> EDITOR_TEXT_ATTRIBUTES; + + /** + * Preference key for default case for content assist. + */ + static final String EDITOR_CONTENT_ASSIST_PROCESSOR_DEFAULT_CASE = "editor.content.assist.processor.default.case"; //$NON-NLS-1$ + + /** + * Preference key for positioning for for compiling. + * + * @since 1.6.1 + */ + static final String EDITOR_COMPILE_COMMAND_POSITIONING_MODE = "editor.compile.command.positioning.mode"; //$NON-NLS-1$ + + /** + * Static initialization. + */ + static { + EDITOR_TEXT_ATTRIBUTES = new HashSet<String>(); + EDITOR_TEXT_ATTRIBUTES.add(EDITOR_TEXT_ATTRIBUTE_DIRECTVE); + EDITOR_TEXT_ATTRIBUTES.add(EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_ENUM_DEFINITION_SECTION); + EDITOR_TEXT_ATTRIBUTES.add(EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_EQUATE); + EDITOR_TEXT_ATTRIBUTES.add(EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_LABEL); + EDITOR_TEXT_ATTRIBUTES.add(EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_LOCAL_SECTION); + EDITOR_TEXT_ATTRIBUTES.add(EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_MACRO_DEFINITION_SECTION); + EDITOR_TEXT_ATTRIBUTES.add(EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_PROCEDURE_DEFINITION_SECTION); + EDITOR_TEXT_ATTRIBUTES.add(EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_STRUCTURE_DEFINITION_SECTION); + EDITOR_TEXT_ATTRIBUTES.add(EDITOR_TEXT_ATTRIBUTE_NUMBER); + EDITOR_TEXT_ATTRIBUTES.add(EDITOR_TEXT_ATTRIBUTE_OPCODE_ILLEGAL); + EDITOR_TEXT_ATTRIBUTES.add(EDITOR_TEXT_ATTRIBUTE_OPCODE_LEGAL); + EDITOR_TEXT_ATTRIBUTES.add(EDITOR_TEXT_ATTRIBUTE_OPCODE_PSEUDO); + EDITOR_TEXT_ATTRIBUTES.add(EDITOR_TEXT_ATTRIBUTE_STRING); + EDITOR_TEXT_ATTRIBUTES.add(EDITOR_TEXT_ATTRIBUTE_COMMENT); + } + + /** + * Determines if preference key name represents a setting for compiler + * opcodes visibility. + * + * @param name + * The name of the preferences key, not <code>null</code>. + * @return <code>true</code> if preference key name represents a setting for + * compiler opcodes visibility, <code>false</code> otherwise. + */ + public static boolean isCompilerCPUName(String name) { + if (name == null) { + throw new IllegalArgumentException("Parameter 'name' must not be null."); + } + boolean result = name.startsWith("compiler.") && name.endsWith(".cpu"); + return result; + } + + /** + * Gets preference key name for the compiler executable path. This is the + * only hardware independent compiler setting. + * + * @param compilerId + * The compiler id, not <code>null</code>. + * + * @return The preference key name for the compiler executable path, not + * empty and not <code>null</code>. + */ + static String getCompilerExecutablePathName(String compilerId) { + + if (compilerId == null) { + throw new IllegalArgumentException("Parameter 'compilerId' must not be null."); + } + return "compiler." + compilerId + ".executable.path"; //$NON-NLS-1$ + } + + /** + * Gets preference key name for the compiler CPU visibility. + * + * @param compilerId + * The compiler id, not <code>null</code>. + * @param hardware + * The hardware, not <code>null</code>. + * + * @return The preference key name for the compiler CPU, not empty and not + * <code>null</code>. + */ + static String getCompilerCPUName(String compilerId, Hardware hardware) { + if (compilerId == null) { + throw new IllegalArgumentException("Parameter 'compilerId' must not be null."); + } + if (hardware == null) { + throw new IllegalArgumentException("Parameter 'hardware' must not be null."); + } + return getCompilerHardwarePrefix(compilerId, hardware) + ".cpu"; //$NON-NLS-1$ + } + + /** + * Gets preference key name for the compiler parameters. + * + * @param compilerId + * The compiler id, not <code>null</code>. + * @param hardware + * The hardware, not <code>null</code>. + * + * @return The preference key name for the compiler parameters, not empty + * and not <code>null</code>. + */ + static String getCompilerParametersName(String compilerId, Hardware hardware) { + if (compilerId == null) { + throw new IllegalArgumentException("Parameter 'compilerId' must not be null."); + } + if (hardware == null) { + throw new IllegalArgumentException("Parameter 'hardware' must not be null."); + } + return getCompilerHardwarePrefix(compilerId, hardware) + ".default.parameters"; //$NON-NLS-1$ + } + + /** + * Gets preference key name for the compiler output folder mode. + * + * @param compilerId + * The compiler id, not <code>null</code>. + * @param hardware + * The hardware, not <code>null</code>. + * + * @return The preference key name for the compiler output folder mode, not + * empty and not <code>null</code>. + */ + static String getCompilerOutputFolderModeName(String compilerId, Hardware hardware) { + if (compilerId == null) { + throw new IllegalArgumentException("Parameter 'compilerId' must not be null."); + } + if (hardware == null) { + throw new IllegalArgumentException("Parameter 'hardware' must not be null."); + } + return getCompilerHardwarePrefix(compilerId, hardware) + ".output.folder.mode"; //$NON-NLS-1$ + + } + + /** + * Gets preference key name for the compiler output folder path. + * + * @param compilerId + * The compiler id, not <code>null</code>. + * @param hardware + * The hardware, not <code>null</code>. + * + * @return The preference key name for the compiler output folder path, not + * empty and not <code>null</code>. + */ + static String getCompilerOutputFolderPathName(String compilerId, Hardware hardware) { + if (compilerId == null) { + throw new IllegalArgumentException("Parameter 'compilerId' must not be null."); + } + if (hardware == null) { + throw new IllegalArgumentException("Parameter 'hardware' must not be null."); + } + return getCompilerHardwarePrefix(compilerId, hardware) + ".output.folder.path"; //$NON-NLS-1$ + + } + + /** + * Gets preference key name for the compiler output file extension. + * + * @param compilerId + * The compiler id, not <code>null</code>. + * @param hardware + * The hardware, not <code>null</code>. + * + * @return The preference key name for the compiler output file extension, + * not empty and not <code>null</code>. + */ + static String getCompilerOutputFileExtensionName(String compilerId, Hardware hardware) { + if (compilerId == null) { + throw new IllegalArgumentException("Parameter 'compilerId' must not be null."); + } + if (hardware == null) { + throw new IllegalArgumentException("Parameter 'hardware' must not be null."); + } + return getCompilerHardwarePrefix(compilerId, hardware) + ".output.file.extension"; //$NON-NLS-1$ + + } + + /** + * Gets preference key name for the runner to run the output file. + * + * @param compilerId + * The compiler id, not <code>null</code>. + * @param hardware + * The hardware, not <code>null</code>. + * + * @return The preference key name for the for the runner to run the output + * file, not empty and not <code>null</code>. + */ + static String getCompilerRunnerIdName(String compilerId, Hardware hardware) { + if (compilerId == null) { + throw new IllegalArgumentException("Parameter 'compilerId' must not be null."); + } + if (hardware == null) { + throw new IllegalArgumentException("Parameter 'hardware' must not be null."); + } + return getCompilerHardwarePrefix(compilerId, hardware) + ".runner.id"; //$NON-NLS-1$ + + } + + /** + * Gets preference key name for the runner executable path. + * + * @param compilerId + * The compiler id, not <code>null</code>. + * @param hardware + * The hardware, not <code>null</code>. + * @param runnerId + * The runner id, not <code>null</code>. + * + * @return The preference key name for the runner executable path, not empty + * and not <code>null</code>. + */ + static String getCompilerRunnerExecutablePathName(String compilerId, Hardware hardware, String runnerId) { + if (compilerId == null) { + throw new IllegalArgumentException("Parameter 'compilerId' must not be null."); + } + if (hardware == null) { + throw new IllegalArgumentException("Parameter 'hardware' must not be null."); + } + if (runnerId == null) { + throw new IllegalArgumentException("Parameter 'runnerId' must not be null."); + } + return getCompilerHardwarePrefix(compilerId, hardware) + ".runner." + runnerId + ".executable.path"; //$NON-NLS-1$ + + } + + /** + * Gets preference key name for the runner command line. + * + * @param compilerId + * The compiler id, not <code>null</code>. + * @param hardware + * The hardware, not <code>null</code>. + * @param runnerId + * The runner id, not <code>null</code>. + * + * @return The preference key name for the runner command line, not empty + * and not <code>null</code>. + */ + static String getCompilerRunnerCommandLineName(String compilerId, Hardware hardware, String runnerId) { + if (compilerId == null) { + throw new IllegalArgumentException("Parameter 'compilerId' must not be null."); + } + if (hardware == null) { + throw new IllegalArgumentException("Parameter 'hardware' must not be null."); + } + if (runnerId == null) { + throw new IllegalArgumentException("Parameter 'runnerId' must not be null."); + } + return getCompilerHardwarePrefix(compilerId, hardware) + ".runner." + runnerId + ".parameters"; //$NON-NLS-1$ + + } + + /** + * Gets preference key name for the runner wait for completion flag. + * + * @param compilerId + * The compiler id, not <code>null</code>. + * @param hardware + * The hardware, not <code>null</code>. + * @param runnerId + * The runner id, not <code>null</code>. + * + * @return The preference key name for the runner command line, not empty + * and not <code>null</code>. + * @since 1.6.1 + */ + static String getCompilerRunnerWaitForCompletionName(String compilerId, Hardware hardware, String runnerId) { + + if (compilerId == null) { + throw new IllegalArgumentException("Parameter 'compilerId' must not be null."); + } + if (hardware == null) { + throw new IllegalArgumentException("Parameter 'hardware' must not be null."); + } + if (runnerId == null) { + throw new IllegalArgumentException("Parameter 'runnerId' must not be null."); + } + return getCompilerHardwarePrefix(compilerId, hardware) + ".runner." + runnerId + ".waitForCompletion"; //$NON-NLS-1$ + + } + + /** + * Gets preference key name prefix for a given hardware and compiler. + * + * @param hardware + * The hardware, not <code>null</code>. + * @param compilerId + * The compiler id, not <code>null</code>. + * + * + * @return The preference key name prefix without trailing dot, not empty + * and not <code>null</code>. + */ + private static String getCompilerHardwarePrefix(String compilerId, Hardware hardware) { + + if (compilerId == null) { + throw new IllegalArgumentException("Parameter 'compilerId' must not be null."); + } + if (hardware == null) { + throw new IllegalArgumentException("Parameter 'hardware' must not be null."); + } + return "compiler." + compilerId + "." + hardware.name().toLowerCase(); + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesInitializer.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesInitializer.java new file mode 100644 index 00000000..01ac5c54 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesInitializer.java @@ -0,0 +1,171 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.preferences; + +import java.util.List; + +import org.eclipse.core.runtime.preferences.AbstractPreferenceInitializer; +import org.eclipse.jface.preference.IPreferenceStore; +import org.eclipse.jface.text.TextAttribute; +import org.eclipse.swt.SWT; +import org.eclipse.swt.graphics.Color; +import org.eclipse.swt.widgets.Display; + +import com.wudsn.ide.asm.AssemblerPlugin; +import com.wudsn.ide.asm.Hardware; +import com.wudsn.ide.asm.HardwareUtility; +import com.wudsn.ide.asm.compiler.CompilerDefinition; +import com.wudsn.ide.asm.compiler.CompilerOutputFolderMode; +import com.wudsn.ide.asm.compiler.CompilerRegistry; +import com.wudsn.ide.asm.editor.AssemblerContentAssistProcessorDefaultCase; +import com.wudsn.ide.asm.editor.AssemblerEditorCompileCommandPositioningMode; +import com.wudsn.ide.asm.runner.RunnerId; + +/** + * Initializer for setting defaults values in the preferences. + * + * @author Peter Dell + */ +public final class AssemblerPreferencesInitializer extends + AbstractPreferenceInitializer { + + /** + * Creation must be public default. + */ + public AssemblerPreferencesInitializer() { + } + + @Override + public void initializeDefaultPreferences() { + IPreferenceStore store = AssemblerPlugin.getInstance() + .getPreferenceStore(); + + initializeEditorPreferences(store); + + initializeCompilerPreferences(store); + + AssemblerPlugin.getInstance().savePreferences(); + } + + private void initializeEditorPreferences(IPreferenceStore store) { + if (store == null) { + throw new IllegalArgumentException( + "Parameter 'store' must not be null."); + } + // Editor. + Display display = Display.getCurrent(); + + TextAttribute textAttribute = new TextAttribute(new Color(display, 0, + 128, 0), null, SWT.ITALIC); + store.setDefault( + AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_COMMENT, + TextAttributeConverter.toString(textAttribute)); + + textAttribute = new TextAttribute(new Color(display, 0, 0, 255), null, + SWT.NORMAL); + store.setDefault( + AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_STRING, + TextAttributeConverter.toString(textAttribute)); + + textAttribute = new TextAttribute(new Color(display, 0, 0, 255), null, + SWT.BOLD); + store.setDefault( + AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_NUMBER, + TextAttributeConverter.toString(textAttribute)); + + textAttribute = new TextAttribute(new Color(display, 128, 64, 0), null, + SWT.BOLD); + store.setDefault( + AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_DIRECTVE, + TextAttributeConverter.toString(textAttribute)); + + textAttribute = new TextAttribute(new Color(display, 0, 0, 128), null, + SWT.BOLD); + store.setDefault( + AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_OPCODE_LEGAL, + TextAttributeConverter.toString(textAttribute)); + + textAttribute = new TextAttribute(new Color(display, 255, 32, 32), + null, SWT.BOLD); + store.setDefault( + AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_OPCODE_ILLEGAL, + TextAttributeConverter.toString(textAttribute)); + + textAttribute = new TextAttribute(new Color(display, 32, 128, 32), + null, SWT.BOLD); + store.setDefault( + AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_OPCODE_PSEUDO, + TextAttributeConverter.toString(textAttribute)); + + // Content assist. + store.setDefault( + AssemblerPreferencesConstants.EDITOR_CONTENT_ASSIST_PROCESSOR_DEFAULT_CASE, + AssemblerContentAssistProcessorDefaultCase.LOWER_CASE); + + // Compiling. + store.setDefault( + AssemblerPreferencesConstants.EDITOR_COMPILE_COMMAND_POSITIONING_MODE, + AssemblerEditorCompileCommandPositioningMode.FIRST_ERROR_OR_WARNING); + } + + private void initializeCompilerPreferences(IPreferenceStore store) { + if (store == null) { + throw new IllegalArgumentException( + "Parameter 'store' must not be null."); + } + + CompilerRegistry compilerRegistry = AssemblerPlugin.getInstance() + .getCompilerRegistry(); + + List<CompilerDefinition> compilerDefinitions = compilerRegistry + .getCompilerDefinitions(); + for (CompilerDefinition compilerDefinition : compilerDefinitions) { + String compilerId; + String name; + compilerId = compilerDefinition.getId(); + + for (Hardware hardware : Hardware.values()) { + if (hardware.equals(Hardware.GENERIC)) { + continue; + } + store.setDefault(AssemblerPreferencesConstants + .getCompilerCPUName(compilerId, hardware), + compilerDefinition.getSupportedCPUs().get(0).toString()); + + name = AssemblerPreferencesConstants.getCompilerParametersName( + compilerId, hardware); + store.setDefault(name, + compilerDefinition.getDefaultParameters()); + name = AssemblerPreferencesConstants + .getCompilerOutputFolderModeName(compilerId, hardware); + store.setDefault(name, CompilerOutputFolderMode.TEMP_FOLDER); + name = AssemblerPreferencesConstants + .getCompilerOutputFileExtensionName(compilerId, + hardware); + store.setDefault(name, + HardwareUtility.getDefaultFileExtension(hardware)); + name = AssemblerPreferencesConstants.getCompilerRunnerIdName( + compilerId, hardware); + store.setDefault(name, RunnerId.DEFAULT_APPLICATION); + } + + } + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesPage.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesPage.java new file mode 100644 index 00000000..d333da6c --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/AssemblerPreferencesPage.java @@ -0,0 +1,597 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.preferences; + +import java.util.ArrayList; +import java.util.List; +import java.util.Set; +import java.util.TreeSet; + +import org.eclipse.jface.preference.ColorSelector; +import org.eclipse.jface.preference.FieldEditor; +import org.eclipse.jface.preference.FieldEditorPreferencePage; +import org.eclipse.jface.preference.IPreferenceStore; +import org.eclipse.jface.preference.RadioGroupFieldEditor; +import org.eclipse.jface.text.TextAttribute; +import org.eclipse.jface.util.PropertyChangeEvent; +import org.eclipse.jface.viewers.ISelectionChangedListener; +import org.eclipse.jface.viewers.IStructuredSelection; +import org.eclipse.jface.viewers.SelectionChangedEvent; +import org.eclipse.jface.viewers.StructuredSelection; +import org.eclipse.jface.viewers.TableViewer; +import org.eclipse.swt.SWT; +import org.eclipse.swt.events.SelectionEvent; +import org.eclipse.swt.events.SelectionListener; +import org.eclipse.swt.graphics.Color; +import org.eclipse.swt.graphics.Font; +import org.eclipse.swt.graphics.FontData; +import org.eclipse.swt.layout.GridData; +import org.eclipse.swt.layout.GridLayout; +import org.eclipse.swt.widgets.Button; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Display; +import org.eclipse.swt.widgets.Group; +import org.eclipse.swt.widgets.Label; +import org.eclipse.swt.widgets.TabFolder; +import org.eclipse.swt.widgets.TabItem; +import org.eclipse.ui.IEditorPart; +import org.eclipse.ui.IWorkbench; +import org.eclipse.ui.IWorkbenchPreferencePage; +import org.eclipse.ui.model.WorkbenchViewerComparator; + +import com.wudsn.ide.asm.AssemblerPlugin; +import com.wudsn.ide.asm.Texts; +import com.wudsn.ide.asm.compiler.CompilerDefinition; +import com.wudsn.ide.asm.compiler.CompilerRegistry; +import com.wudsn.ide.asm.editor.AssemblerContentAssistProcessorDefaultCase; +import com.wudsn.ide.asm.editor.AssemblerEditor; +import com.wudsn.ide.asm.editor.AssemblerEditorCompileCommandPositioningMode; +import com.wudsn.ide.base.common.ProcessWithLogs; +import com.wudsn.ide.base.gui.SWTFactory; + +/** + * Visual editor page for the assembler preferences. + * + * @author Peter Dell + */ +public final class AssemblerPreferencesPage extends FieldEditorPreferencePage implements IWorkbenchPreferencePage { + + private abstract class TextAttributeSelectionListener implements SelectionListener { + + /** + * Creation is public. + */ + public TextAttributeSelectionListener() { + } + + @Override + public void widgetDefaultSelected(SelectionEvent e) { + } + + @Override + public void widgetSelected(SelectionEvent e) { + TextAttributeListItem item = getTextAttributeListItem(); + if (item == null) { + throw new IllegalStateException("No item selected."); + } + updateItem(item); + textAttributeListItemsViewer.refresh(); + addChangedProperty(item.getPreferencesKey()); + } + + abstract protected void updateItem(TextAttributeListItem item); + } + + /** + * The owning plugin. + */ + private AssemblerPlugin plugin; + + /** + * The set of property names for which the value was changed since the page + * was opened. + */ + private Set<String> changedPropertyNames; + + /** + * The id of the compiler to be used as default. + */ + private String activeCompilerId; + + /** + * The list of all text attributes and the corresponding preferences keys. + */ + private String[][] textAttributeListItemKeys; + + /** + * List for text attribute items. + */ + List<TextAttributeListItem> textAttributeListItems; + + /** + * Highlighting color list viewer + */ + TableViewer textAttributeListItemsViewer; + + /** + * Color selector for foreground color. + */ + ColorSelector textAttributeForegroundColorSelector; + + /** + * Check box for bold setting. + */ + Button textAttributeBoldCheckBox; + + /** + * Check box for italic setting. + */ + Button textAttributeItalicCheckBox; + + /** + * Creation must be public default. + */ + public AssemblerPreferencesPage() { + super(GRID); + plugin = AssemblerPlugin.getInstance(); + setPreferenceStore(plugin.getPreferenceStore()); + changedPropertyNames = new TreeSet<String>(); + textAttributeListItemKeys = new String[][] { + { Texts.PREFERENCES_TEXT_ATTRIBUTE_COMMENT_NAME, + AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_COMMENT }, + { Texts.PREFERENCES_TEXT_ATTRIBUTE_NUMBER_NAME, + AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_NUMBER }, + { Texts.PREFERENCES_TEXT_ATTRIBUTE_STRING_NAME, + AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_STRING }, + { Texts.PREFERENCES_TEXT_ATTRIBUTE_DIRECTIVE, + AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_DIRECTVE }, + { Texts.PREFERENCES_TEXT_ATTRIBUTE_OPCODE_LEGAL_NAME, + AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_OPCODE_LEGAL }, + { Texts.PREFERENCES_TEXT_ATTRIBUTE_OPCODE_ILLEGAL_NAME, + AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_OPCODE_ILLEGAL }, + { Texts.PREFERENCES_TEXT_ATTRIBUTE_OPCODE_PSEUDO_NAME, + AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_OPCODE_PSEUDO }, + + { Texts.PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_EQUATE, + AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_EQUATE }, + { Texts.PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_LABEL, + AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_LABEL }, + { Texts.PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_ENUM_DEFINITION_SECTION, + AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_ENUM_DEFINITION_SECTION }, + { Texts.PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_STRUCTURE_DEFINITION_SECTION, + AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_STRUCTURE_DEFINITION_SECTION }, + { Texts.PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_LOCAL_SECTION, + AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_LOCAL_SECTION }, + { Texts.PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_MACRO_DEFINITION_SECTION, + AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_MACRO_DEFINITION_SECTION }, + { Texts.PREFERENCES_TEXT_ATTRIBUTE_IDENTIFIER_PROCEDURE_DEFINITION_SECTION, + AssemblerPreferencesConstants.EDITOR_TEXT_ATTRIBUTE_IDENTIFIER_PROCEDURE_DEFINITION_SECTION } }; + } + + @Override + public void init(IWorkbench workbench) { + IEditorPart editor = workbench.getActiveWorkbenchWindow().getActivePage().getActiveEditor(); + if (editor instanceof AssemblerEditor) { + AssemblerEditor assemblerEditor; + assemblerEditor = (AssemblerEditor) editor; + activeCompilerId = assemblerEditor.getCompilerDefinition().getId(); + + } + changedPropertyNames.clear(); + } + + @Override + public void createFieldEditors() { + + Composite parent = getFieldEditorParent(); + + parent = SWTFactory.createComposite(parent, 1, 1, GridData.FILL_BOTH); + initializeTextAttributesList(); + createSyntaxHighlightingGroup(parent); + createEditorGroup(parent); + createCompilersGroup(parent); + } + + void addChangedProperty(String key) { + if (key == null) { + throw new IllegalArgumentException("Parameter 'key' must not be null."); + } + changedPropertyNames.add(key); + } + + @Override + public void dispose() { + + disposeTextAttributesList(); + super.dispose(); + } + + /** + * Creates all visual controls. + * + * @param parent + * The parent object, not <code>null</code>. + */ + private void createSyntaxHighlightingGroup(Composite parent) { + if (parent == null) { + throw new IllegalArgumentException("Parameter 'parent' must not be null."); + } + Group group = SWTFactory.createGroup(parent, Texts.PREFERENCES_SYNTAX_HIGHLIGHTING_GROUP_TITLE, 2, 1, + GridData.FILL_HORIZONTAL); + Label label; + GridLayout layout; + GridData gd; + + textAttributeListItemsViewer = new TableViewer(group, SWT.SINGLE | SWT.V_SCROLL | SWT.BORDER + | SWT.FULL_SELECTION); + textAttributeListItemsViewer.setLabelProvider(new TextAttributeListItemProvider()); + textAttributeListItemsViewer.setContentProvider(new TextAttributeListContentProvider()); + textAttributeListItemsViewer.setComparator(new WorkbenchViewerComparator()); + gd = new GridData(SWT.BEGINNING, SWT.FILL, false, true); + gd.heightHint = convertHeightInCharsToPixels(textAttributeListItems.size()); + textAttributeListItemsViewer.getControl().setLayoutData(gd); + + Composite stylesComposite = new Composite(group, SWT.NONE); + layout = new GridLayout(); + layout.marginHeight = 0; + layout.marginWidth = 0; + layout.numColumns = 2; + stylesComposite.setLayout(layout); + stylesComposite.setLayoutData(new GridData(SWT.BEGINNING, SWT.BEGINNING, false, false)); + + gd = new GridData(GridData.FILL_HORIZONTAL); + gd.horizontalAlignment = GridData.BEGINNING; + gd.horizontalSpan = 2; + + label = new Label(stylesComposite, SWT.LEFT); + label.setText(Texts.PREFERENCES_FOREGROUND_COLOR_LABEL); + gd = new GridData(GridData.HORIZONTAL_ALIGN_BEGINNING); + gd.horizontalIndent = 20; + label.setLayoutData(gd); + + textAttributeForegroundColorSelector = new ColorSelector(stylesComposite); + Button foregroundColorButton = textAttributeForegroundColorSelector.getButton(); + gd = new GridData(GridData.HORIZONTAL_ALIGN_BEGINNING); + foregroundColorButton.setLayoutData(gd); + + textAttributeBoldCheckBox = new Button(stylesComposite, SWT.CHECK); + textAttributeBoldCheckBox.setText(Texts.PREFERENCES_BOLD_LABEL); + gd = new GridData(GridData.HORIZONTAL_ALIGN_BEGINNING); + gd.horizontalIndent = 20; + gd.horizontalSpan = 2; + textAttributeBoldCheckBox.setLayoutData(gd); + + textAttributeItalicCheckBox = new Button(stylesComposite, SWT.CHECK); + textAttributeItalicCheckBox.setText(Texts.PREFERENCES_ITALIC_LABEL); + gd = new GridData(GridData.HORIZONTAL_ALIGN_BEGINNING); + gd.horizontalIndent = 20; + gd.horizontalSpan = 2; + textAttributeItalicCheckBox.setLayoutData(gd); + + textAttributeListItemsViewer.addSelectionChangedListener(new ISelectionChangedListener() { + @Override + public void selectionChanged(SelectionChangedEvent event) { + handleSyntaxColorListSelection(); + } + }); + + foregroundColorButton.addSelectionListener(new TextAttributeSelectionListener() { + @Override + protected void updateItem(TextAttributeListItem item) { + if (item == null) { + throw new IllegalArgumentException("Parameter 'item' must not be null."); + } + TextAttribute textAttribute = item.getTextAttribute(); + Color foreground = textAttribute.getForeground(); + foreground.dispose(); + foreground = new Color(Display.getCurrent(), textAttributeForegroundColorSelector.getColorValue()); + item.setTextAttribute(new TextAttribute(foreground, textAttribute.getBackground(), textAttribute + .getStyle(), textAttribute.getFont())); + } + + }); + + textAttributeBoldCheckBox.addSelectionListener(new TextAttributeSelectionListener() { + @Override + protected void updateItem(TextAttributeListItem item) { + if (item == null) { + throw new IllegalArgumentException("Parameter 'item' must not be null."); + } + TextAttribute textAttribute = item.getTextAttribute(); + int style = (textAttribute.getStyle() & ~SWT.BOLD) + | (textAttributeBoldCheckBox.getSelection() ? SWT.BOLD : SWT.NONE); + Font font = textAttribute.getFont(); + FontData fontData = font.getFontData()[0]; + fontData = new FontData(fontData.getName(), fontData.getHeight(), style); + font.dispose(); + font = new Font(Display.getCurrent(), fontData); + item.setTextAttribute(new TextAttribute(textAttribute.getForeground(), textAttribute.getBackground(), + style, font)); + } + }); + + textAttributeItalicCheckBox.addSelectionListener(new TextAttributeSelectionListener() { + @Override + protected void updateItem(TextAttributeListItem item) { + if (item == null) { + throw new IllegalArgumentException("Parameter 'item' must not be null."); + } + TextAttribute textAttribute = item.getTextAttribute(); + int style = (textAttribute.getStyle() & ~SWT.ITALIC) + | (textAttributeItalicCheckBox.getSelection() ? SWT.ITALIC : SWT.NONE); + Font font = textAttribute.getFont(); + FontData fontData = font.getFontData()[0]; + fontData = new FontData(fontData.getName(), fontData.getHeight(), style); + font.dispose(); + font = new Font(Display.getCurrent(), fontData); + item.setTextAttribute(new TextAttribute(textAttribute.getForeground(), textAttribute.getBackground(), + style, font)); + } + }); + + inittialzeTextAttributesListViewer(); + + parent.layout(); + + } + + /** + * FIll the list items view an set the selection to the first item. + */ + private void inittialzeTextAttributesListViewer() { + textAttributeListItemsViewer.setInput(textAttributeListItems); + textAttributeListItemsViewer + .setSelection(new StructuredSelection(textAttributeListItemsViewer.getElementAt(0))); + } + + /** + * Dispose the text attribute list. + */ + private void disposeTextAttributesList() { + if (textAttributeListItems == null) { + throw new IllegalStateException("Attribute 'textAttributeListItems' must not be null."); + } + for (TextAttributeListItem item : textAttributeListItems) { + TextAttributeConverter.dispose(item.getTextAttribute()); + } + textAttributeListItems = null; + } + + /** + * Setup the text attribute list. + */ + private void initializeTextAttributesList() { + if (textAttributeListItems != null) { + throw new IllegalStateException("Attribute 'textAttributeListItems' must be null."); + } + textAttributeListItems = new ArrayList<TextAttributeListItem>(textAttributeListItemKeys.length); + + for (int i = 0, n = textAttributeListItemKeys.length; i < n; i++) { + String data; + TextAttribute textAttribute; + TextAttributeListItem item; + + data = getPreferenceStore().getString(textAttributeListItemKeys[i][1]); + textAttribute = TextAttributeConverter.fromString(data); + + item = new TextAttributeListItem(textAttributeListItemKeys[i][0], textAttributeListItemKeys[i][1]); + item.setTextAttribute(textAttribute); + textAttributeListItems.add(item); + } + } + + private void createEditorGroup(Composite parent) { + if (parent == null) { + throw new IllegalArgumentException("Parameter 'parent' must not be null."); + } + Group group = SWTFactory.createGroup(parent, Texts.PREFERENCES_EDITOR_GROUP_TITLE, 1, 1, + GridData.FILL_HORIZONTAL); + + Composite space = SWTFactory.createComposite(group, 2, 1, GridData.FILL_HORIZONTAL); + + String[][] labelsAndValues; + labelsAndValues = new String[][] { + { + + Texts.PREFERENCES_CONTENT_ASSIST_PROCESSOR_DEFAULT_CASE_LOWER_CASE_TEXT, + AssemblerContentAssistProcessorDefaultCase.LOWER_CASE }, + { + + Texts.PREFERENCES_CONTENT_ASSIST_PROCESSOR_DEFAULT_CASE_UPPER_CASE_TEXT, + AssemblerContentAssistProcessorDefaultCase.UPPER_CASE } + + }; + + FieldEditor choiceFieldEditor = new RadioGroupFieldEditor( + AssemblerPreferencesConstants.EDITOR_CONTENT_ASSIST_PROCESSOR_DEFAULT_CASE, + Texts.PREFERENCES_CONTENT_ASSIST_PROCESSOR_DEFAULT_CASE_LABEL, 2, labelsAndValues, space); + addField(choiceFieldEditor); + + labelsAndValues = new String[][] { + { + + Texts.PREFERENCES_COMPILE_COMMAND_POSITIONING_MODE_FIRST_ERROR_OR_WARNING_TEXT, + AssemblerEditorCompileCommandPositioningMode.FIRST_ERROR_OR_WARNING }, + { + + Texts.PREFERENCES_COMPILE_COMMAND_POSITIONING_MODE_FIRST_ERROR_TEXT, + AssemblerEditorCompileCommandPositioningMode.FIRST_ERROR } + + }; + + choiceFieldEditor = new RadioGroupFieldEditor( + AssemblerPreferencesConstants.EDITOR_COMPILE_COMMAND_POSITIONING_MODE, + Texts.PREFERENCES_COMPILE_COMMAND_POSITIONING_MODE_LABEL, 2, labelsAndValues, space); + addField(choiceFieldEditor); + + } + + private void createCompilersGroup(Composite parent) { + if (parent == null) { + throw new IllegalArgumentException("Parameter 'parent' must not be null."); + } + TabFolder tabFolder = new TabFolder(parent, SWT.NONE); + TabItem selectedTabItem = null; + + // Create the editors for all compilers of the hardware. + CompilerRegistry compilerRegistry = plugin.getCompilerRegistry(); + List<CompilerDefinition> compilerDefinitions = compilerRegistry.getCompilerDefinitions(); + + for (CompilerDefinition compilerDefinition : compilerDefinitions) { + String compilerId = compilerDefinition.getId(); + TabItem tabItem = new TabItem(tabFolder, SWT.NONE); + tabItem.setText(compilerDefinition.getName()); + + Composite tabContent; + tabContent = SWTFactory.createComposite(tabFolder, 1, 1, GridData.FILL_HORIZONTAL); + tabItem.setControl(tabContent); + + Composite composite; + + String name; + name = AssemblerPreferencesConstants.getCompilerExecutablePathName(compilerId); + + // Field: executablePath + composite = SWTFactory.createComposite(tabContent, 4, 2, GridData.FILL_HORIZONTAL); + FileFieldDownloadEditor fileFieldEditor = new FileFieldDownloadEditor(name, + Texts.PREFERENCES_COMPILER_EXECUTABLE_PATH_LABEL, composite); + fileFieldEditor.setFileExtensions(ProcessWithLogs.getExecutableExtensions()); + + addField(fileFieldEditor); + + // Set URL only after editor was added. + String url = compilerDefinition.getHomePageURL(); + fileFieldEditor.setLinkURL(url); + + if (compilerId.equals(activeCompilerId)) { + selectedTabItem = tabItem; + } + } + + // Default to selected tab item. + if (selectedTabItem != null) { + tabFolder.setSelection(selectedTabItem); + } + } + + /** + * {@inheritDoc} + * + * This method is called when "Apply" or "OK" is pressed. + */ + @Override + public boolean performOk() { + if (super.performOk()) { + saveChanges(); + plugin.firePreferencesChangeEvent(changedPropertyNames); + return true; + } + return false; + } + + /** + * The field editor preference page implementation of a + * <code>PreferencePage</code> method loads all the field editors with their + * default values. + */ + @Override + protected void performDefaults() { + + super.performDefaults(); + + IPreferenceStore preferencesStore = getPreferenceStore(); + for (int i = 0, n = textAttributeListItemKeys.length; i < n; i++) { + String key = textAttributeListItemKeys[i][1]; + preferencesStore.setValue(key, preferencesStore.getDefaultString(key)); + addChangedProperty(key); + } + + disposeTextAttributesList(); + initializeTextAttributesList(); + inittialzeTextAttributesListViewer(); + } + + /** + * Saves all changes to the {@link IPreferenceStore}. + */ + private void saveChanges() { + String data; + IPreferenceStore store = getPreferenceStore(); + + for (TextAttributeListItem listItem : textAttributeListItems) { + data = TextAttributeConverter.toString(listItem.getTextAttribute()); + store.setValue(listItem.getPreferencesKey(), data); + + } + + plugin.savePreferences(); + + } + + @Override + public void propertyChange(PropertyChangeEvent event) { + super.propertyChange(event); + if (event.getSource() instanceof FieldEditor) { + FieldEditor fieldEditor = (FieldEditor) event.getSource(); + addChangedProperty(fieldEditor.getPreferenceName()); + } + } + + /** + * Update controls after item select. + */ + void handleSyntaxColorListSelection() { + TextAttributeListItem item = getTextAttributeListItem(); + + if (item == null) { + return; + } + + Color color; + boolean bold; + boolean italic; + + TextAttribute textAttribute = item.getTextAttribute(); + color = textAttribute.getForeground(); + bold = (textAttribute.getStyle() & SWT.BOLD) == SWT.BOLD; + italic = (textAttribute.getStyle() & SWT.ITALIC) == SWT.ITALIC; + + textAttributeForegroundColorSelector.setColorValue(color.getRGB()); + textAttributeBoldCheckBox.setSelection(bold); + textAttributeItalicCheckBox.setSelection(italic); + + textAttributeForegroundColorSelector.getButton().setEnabled(true); + textAttributeBoldCheckBox.setEnabled(true); + textAttributeItalicCheckBox.setEnabled(true); + } + + /** + * Returns the current highlighting color list item. + * + * @return The current highlighting color list item or <code>null</code>. + */ + TextAttributeListItem getTextAttributeListItem() { + TextAttributeListItem listItem; + IStructuredSelection selection = (IStructuredSelection) textAttributeListItemsViewer.getSelection(); + listItem = (TextAttributeListItem) selection.getFirstElement(); + return listItem; + } + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/CompilerPreferences.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/CompilerPreferences.java new file mode 100644 index 00000000..6dc68904 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/CompilerPreferences.java @@ -0,0 +1,262 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.preferences; + +import com.wudsn.ide.asm.CPU; +import com.wudsn.ide.asm.Hardware; +import com.wudsn.ide.asm.compiler.CompilerOutputFolderMode; +import com.wudsn.ide.asm.runner.RunnerId; +import com.wudsn.ide.base.common.StringUtility; + +/** + * Facade class for typed access to the global compiler preferences for a given + * hardware. + * + * @author Peter Dell + */ +public final class CompilerPreferences { + + private AssemblerPreferences assemblerPreferences; + private Hardware hardware; + private String compilerId; + + CompilerPreferences(AssemblerPreferences assemblerPreferences, + String compilerId, Hardware hardware) { + if (assemblerPreferences == null) { + throw new IllegalArgumentException( + "Parameter 'assemblerPreferences' must not be null."); + } + + if (compilerId == null) { + throw new IllegalArgumentException( + "Parameter 'compilerId' must not be null."); + } + if (StringUtility.isEmpty(compilerId)) { + throw new IllegalArgumentException( + "Parameter 'compilerId' must not be empty."); + } + if (hardware == null) { + throw new IllegalArgumentException( + "Parameter 'hardware' must not be null."); + } + this.assemblerPreferences = assemblerPreferences; + this.compilerId = compilerId; + this.hardware = hardware; + } + + /** + * Gets the compiler id of the compiler. + * + * @return The compiler id of the compiler, not empty and not + * <code>null</code>. + */ + public String getCompilerId() { + return compilerId; + } + + /** + * Gets the hardware for which the compiler is invoked. + * + * @return The hardware, not <code>null</code>. + * + * @since 1.6.1 + */ + public Hardware getHardware() { + return hardware; + } + + /** + * Gets the CPU for which the instructions shall be active. + * + * @return The CPU, not <code>null</code>. + * + * @since 1.6.1 + */ + public CPU getCPU() { + CPU result; + String cpuString = assemblerPreferences + .getString(AssemblerPreferencesConstants.getCompilerCPUName( + compilerId, hardware)); + + if (StringUtility.isEmpty(cpuString)) { + result = CPU.MOS6502; + } else { + result = CPU.valueOf(cpuString); + } + return result; + } + + /** + * Determines if illegal opcodes shall be highlighted and proposed. + * + * @return <code>true</code> if yet, <code>false</code> otherwise. + */ + @Deprecated + public boolean isIllegalOpcodesVisible() { + return getCPU() == CPU.MOS6502_ILLEGAL; + } + + /** + * Determines if W65816 opcodes shall be highlighted and proposed. + * + * @return <code>true</code> if yet, <code>false</code> otherwise. + */ + @Deprecated + public boolean isW65816OpcodesVisible() { + return getCPU() == CPU.MOS65816; + } + + /** + * Gets the parameters for the compiler. + * + * @return The parameters path for the compiler, may be empty, not + * <code>null</code>. + */ + public String getParameters() { + return assemblerPreferences.getString(AssemblerPreferencesConstants + .getCompilerParametersName(compilerId, hardware)); + } + + /** + * Gets the output folder mode for the compiler. + * + * @return The output folder mode for the compiler, see + * {@link CompilerOutputFolderMode}, may be empty, not + * <code>null</code>. + */ + public String getOutputFolderMode() { + + return assemblerPreferences.getString(AssemblerPreferencesConstants + .getCompilerOutputFolderModeName(compilerId, hardware)); + } + + /** + * Gets the output folder for the compiler in case the output folder mode is + * {@link CompilerOutputFolderMode#FIXED_FOLDER}. + * + * @return The output folder mode for the compiler, see + * {@link CompilerOutputFolderMode#FIXED_FOLDER}, may be empty, not + * <code>null</code>. + */ + public String getOutputFolderPath() { + + return assemblerPreferences.getString(AssemblerPreferencesConstants + .getCompilerOutputFolderPathName(compilerId, hardware)); + } + + /** + * Gets the output file extension for the compiler. + * + * @return The output file extension may be empty, not <code>null</code>. + */ + public String getOutputFileExtension() { + + return assemblerPreferences.getString(AssemblerPreferencesConstants + .getCompilerOutputFileExtensionName(compilerId, hardware)); + } + + /** + * Gets the id of the default runner to run the output file. + * + * @return The id of the runner to run the output file, not empty and not + * <code>null</code>. + */ + public String getRunnerId() { + String result = assemblerPreferences + .getString(AssemblerPreferencesConstants + .getCompilerRunnerIdName(compilerId, hardware)); + if (StringUtility.isEmpty(result)) { + result = RunnerId.DEFAULT_APPLICATION; + } + return result; + } + + /** + * Gets the executable path for the runner. + * + * @param runnerId + * The runner id, not empty and not <code>null</code>. + * + * @return The executable path for the runner, may be empty, not + * <code>null</code>. + */ + public String getRunnerExecutablePath(String runnerId) { + if (runnerId == null) { + throw new IllegalArgumentException( + "Parameter 'runnerId' must not be null."); + } + if (StringUtility.isEmpty(runnerId)) { + throw new IllegalArgumentException( + "Parameter 'runnerId' must not be empty."); + } + return assemblerPreferences.getString(AssemblerPreferencesConstants + .getCompilerRunnerExecutablePathName(compilerId, hardware, + runnerId)); + } + + /** + * Gets the parameters for the runner. + * + * @param runnerId + * The runner id, not empty and not <code>null</code>. + * + * @return The parameters for the runner, may be empty, not + * <code>null</code>. + */ + public String getRunnerCommandLine(String runnerId) { + if (runnerId == null) { + throw new IllegalArgumentException( + "Parameter 'runnerId' must not be null."); + } + if (StringUtility.isEmpty(runnerId)) { + throw new IllegalArgumentException( + "Parameter 'runnerId' must not be empty."); + } + return assemblerPreferences.getString(AssemblerPreferencesConstants + .getCompilerRunnerCommandLineName(compilerId, hardware, + runnerId)); + } + + /** + * Gets the wait for completion indicator for the runner. + * + * @param runnerId + * The runner id, not empty and not <code>null</code>. + * + * @return <code>true</code>if waiting for completion is requested, + * <code>false</code> otherwise. + * + * @since 1.6.1 + */ + public boolean isRunnerWaitForCompletion(String runnerId) { + if (runnerId == null) { + throw new IllegalArgumentException( + "Parameter 'runnerId' must not be null."); + } + if (StringUtility.isEmpty(runnerId)) { + throw new IllegalArgumentException( + "Parameter 'runnerId' must not be empty."); + } + return assemblerPreferences.getBoolean(AssemblerPreferencesConstants + .getCompilerRunnerWaitForCompletionName(compilerId, hardware, + runnerId)); + } + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/CompilerRunPreferences.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/CompilerRunPreferences.java new file mode 100644 index 00000000..2e8fbc29 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/CompilerRunPreferences.java @@ -0,0 +1,154 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.preferences; + +import com.wudsn.ide.asm.AssemblerProperties; +import com.wudsn.ide.asm.Hardware; +import com.wudsn.ide.base.common.StringUtility; + +/** + * Facade class to mix compiler run specific preferences into the global + * preferences. + * + * @author Peter Dell + * + */ +public final class CompilerRunPreferences { + + private CompilerPreferences compilerPreferences; + private AssemblerProperties mainSourceFileProperties; + + public CompilerRunPreferences(CompilerPreferences compilerPreferences, AssemblerProperties mainSourceFileProperties) { + if (compilerPreferences == null) { + throw new IllegalArgumentException("Parameter 'compilerPreferences' must not be null."); + } + if (mainSourceFileProperties == null) { + throw new IllegalArgumentException("Parameter 'properties' must not be null."); + } + this.compilerPreferences = compilerPreferences; + this.mainSourceFileProperties=mainSourceFileProperties; + } + + /** + * Gets the hardware for which the compiler is invoked. + * + * @return The hardware, not <code>null</code>. + * + * @since 1.6.1 + */ + public Hardware getHardware() { + return compilerPreferences.getHardware(); + } + + /** + * Gets the parameters for the compiler. + * + * @return The parameters path for the compiler, may be empty, not + * <code>null</code>. + */ + public String getParameters() { + + String result; + + result = compilerPreferences.getParameters(); + return result; + } + + /** + * Gets the id of the runner to run the output file. + * + * @return The id of the runner to run the output file, not empty and not + * <code>null</code>. + */ + public String getRunnerId() { + String result; + + result = compilerPreferences.getRunnerId(); + return result; + } + + /** + * Gets the executable path for the runner. + * + * @param runnerId + * The runner id, not empty and not <code>null</code>. + * + * @return The executable path for the runner, may be empty, not + * <code>null</code>. + */ + public String getRunnerExecutablePath(String runnerId) { + if (runnerId == null) { + throw new IllegalArgumentException("Parameter 'runnerId' must not be null."); + } + if (StringUtility.isEmpty(runnerId)) { + throw new IllegalArgumentException("Parameter 'runnerId' must not be empty."); + } + + String result; + + result = compilerPreferences.getRunnerExecutablePath(runnerId); + return result; + } + + /** + * Gets the parameters for the runner. + * + * @param runnerId + * The runner id, not empty and not <code>null</code>. + * + * @return The parameters for the runner, may be empty, not + * <code>null</code>. + */ + public String getRunnerCommandLine(String runnerId) { + if (runnerId == null) { + throw new IllegalArgumentException("Parameter 'runnerId' must not be null."); + } + if (StringUtility.isEmpty(runnerId)) { + throw new IllegalArgumentException("Parameter 'runnerId' must not be empty."); + } + String result; + result = compilerPreferences.getRunnerCommandLine(runnerId); + return result; + } + + /** + * Gets the wait for completion indicator for the runner. + * + * @param runnerId + * The runner id, not empty and not <code>null</code>. + * + * @return <code>true</code>if waiting for completion is requested, + * <code>false</code> otherwise. + * + * @since 1.6.1 + */ + public boolean isRunnerWaitForCompletion(String runnerId) { + if (runnerId == null) { + throw new IllegalArgumentException("Parameter 'runnerId' must not be null."); + } + if (StringUtility.isEmpty(runnerId)) { + throw new IllegalArgumentException("Parameter 'runnerId' must not be empty."); + } + boolean result; + result = compilerPreferences.isRunnerWaitForCompletion(runnerId); + return result; + } + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/DirectoryFieldDownloadEditor.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/DirectoryFieldDownloadEditor.java new file mode 100644 index 00000000..05f1e9d1 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/DirectoryFieldDownloadEditor.java @@ -0,0 +1,137 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.preferences; + +import org.eclipse.jface.preference.DirectoryFieldEditor; +import org.eclipse.jface.preference.FieldEditor; +import org.eclipse.swt.SWT; +import org.eclipse.swt.events.DisposeEvent; +import org.eclipse.swt.events.DisposeListener; +import org.eclipse.swt.layout.GridData; +import org.eclipse.swt.program.Program; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Event; +import org.eclipse.swt.widgets.Link; +import org.eclipse.swt.widgets.Listener; + +import com.wudsn.ide.asm.Texts; +import com.wudsn.ide.base.common.TextUtility; + +/** + * Extended directory field editor with build-in download link. + * + * @author Peter Dell + */ +final class DirectoryFieldDownloadEditor extends DirectoryFieldEditor { + + Link link; + + public DirectoryFieldDownloadEditor(String name, String labelText, + Composite parent) { + super(name, labelText, parent); + } + + /** + * Override the method declared in {@link FieldEditor}. + */ + @Override + public int getNumberOfControls() { + return 4; + } + + @Override + protected void doFillIntoGrid(Composite parent, int numColumns) { + super.doFillIntoGrid(parent, numColumns - 1); + if (link == null) { + link = new Link(parent, SWT.NONE); + link.addDisposeListener(new DisposeListener() { + @Override + public void widgetDisposed(DisposeEvent event) { + link = null; + } + }); + + link.addListener(SWT.Selection, new Listener() { + @Override + public void handleEvent(Event event) { + String url = event.text; + if (url != null && url.length() > 0) { + Program.launch(event.text); + } + } + }); + + } + GridData gd = new GridData(); + gd.horizontalAlignment = GridData.FILL; + link.setLayoutData(gd); + } + + @Override + protected void adjustForNumColumns(int numColumns) { + ((GridData) getTextControl().getLayoutData()).horizontalSpan = numColumns - 3; + } + + /** + * Do not reset path to default. + */ + @Override + public void loadDefault() { + } + + /** + * Do not check input as file to allow selecting ".app" directories on MacOS X. + * + * @return <code>true</code> in all cases. + */ + @Override + protected boolean checkState() { + return true; + + } + + /** + * Sets the URL for the link label. + * + * @param url + * The URL, may be empty, not <code>null</code>. + */ + public void setLinkURL(String url) { + if (link == null) { + throw new IllegalArgumentException( + "Parameter 'link' must not be null."); + } + if (url == null) { + throw new IllegalArgumentException( + "Parameter 'url' must not be null."); + } + + if (url.length() > 0) { + link.setText("<a href=\"" + url + "\">" + + Texts.PREFERENCES_DOWNLOAD_LINK + "</a>"); + link.setToolTipText(TextUtility.format( + Texts.PREFERENCES_DOWNLOAD_LINK_TOOL_TIP, url)); + + } else { + link.setText(""); + link.setToolTipText(""); + } + } +} \ No newline at end of file diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/FileFieldDownloadEditor.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/FileFieldDownloadEditor.java new file mode 100644 index 00000000..bfc8cc7a --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/FileFieldDownloadEditor.java @@ -0,0 +1,137 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.preferences; + +import org.eclipse.jface.preference.FieldEditor; +import org.eclipse.jface.preference.FileFieldEditor; +import org.eclipse.swt.SWT; +import org.eclipse.swt.events.DisposeEvent; +import org.eclipse.swt.events.DisposeListener; +import org.eclipse.swt.layout.GridData; +import org.eclipse.swt.program.Program; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Event; +import org.eclipse.swt.widgets.Link; +import org.eclipse.swt.widgets.Listener; + +import com.wudsn.ide.asm.Texts; +import com.wudsn.ide.base.common.TextUtility; + +/** + * Extended file field editor with build-in download link. + * + * @author Peter Dell + */ +final class FileFieldDownloadEditor extends FileFieldEditor { + + Link link; + + public FileFieldDownloadEditor(String name, String labelText, + Composite parent) { + super(name, labelText, parent); + } + + /** + * Override the method declared in {@link FieldEditor}. + */ + @Override + public int getNumberOfControls() { + return 4; + } + + @Override + protected void doFillIntoGrid(Composite parent, int numColumns) { + super.doFillIntoGrid(parent, numColumns - 1); + if (link == null) { + link = new Link(parent, SWT.NONE); + link.addDisposeListener(new DisposeListener() { + @Override + public void widgetDisposed(DisposeEvent event) { + link = null; + } + }); + + link.addListener(SWT.Selection, new Listener() { + @Override + public void handleEvent(Event event) { + String url = event.text; + if (url != null && url.length() > 0) { + Program.launch(event.text); + } + } + }); + + } + GridData gd = new GridData(); + gd.horizontalAlignment = GridData.FILL; + link.setLayoutData(gd); + } + + @Override + protected void adjustForNumColumns(int numColumns) { + ((GridData) getTextControl().getLayoutData()).horizontalSpan = numColumns - 3; + } + + /** + * Do not reset path to default. + */ + @Override + public void loadDefault() { + } + + /** + * Do not check input as file to allow selecting ".app" directories on MacOS X. + * + * @return <code>true</code> in all cases. + */ + @Override + protected boolean checkState() { + return true; + + } + + /** + * Sets the URL for the link label. + * + * @param url + * The URL, may be empty, not <code>null</code>. + */ + public void setLinkURL(String url) { + if (link == null) { + throw new IllegalArgumentException( + "Parameter 'link' must not be null."); + } + if (url == null) { + throw new IllegalArgumentException( + "Parameter 'url' must not be null."); + } + + if (url.length() > 0) { + link.setText("<a href=\"" + url + "\">" + + Texts.PREFERENCES_DOWNLOAD_LINK + "</a>"); + link.setToolTipText(TextUtility.format( + Texts.PREFERENCES_DOWNLOAD_LINK_TOOL_TIP, url)); + + } else { + link.setText(""); + link.setToolTipText(""); + } + } +} \ No newline at end of file diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/TextAttributeConverter.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/TextAttributeConverter.java new file mode 100644 index 00000000..187a9432 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/TextAttributeConverter.java @@ -0,0 +1,162 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.preferences; + +import org.eclipse.jface.resource.JFaceResources; +import org.eclipse.jface.text.TextAttribute; +import org.eclipse.swt.SWT; +import org.eclipse.swt.graphics.Color; +import org.eclipse.swt.graphics.Font; +import org.eclipse.swt.graphics.FontData; +import org.eclipse.swt.widgets.Display; + +import com.wudsn.ide.base.common.StringUtility; + +/** + * Convertdf go connvert the class TextAttributes into a PreferenceStore + * compatible form. + * + * @author Peter Dell + * @author Daniel Mitte + */ +public final class TextAttributeConverter { + + /** + * Converts preferences string to a color value. + * + * @param value + * The color as an RGB string of the for "r, g, b", may be empty + * or <code>null</code>. + * + * @return The color, not <code>null</code>. + */ + public static TextAttribute fromString(String value) { + + TextAttribute result; + Display display = Display.getCurrent(); + Color foregroundColor; + Color backgroundColor; + int style; + if (value != null) { + + String[] data = value.split(","); + + try { + int r, g, b; + + if (StringUtility.isEmpty(data[0] + data[1] + data[2])) { + foregroundColor = null; + } else { + r = new Integer(data[0]).intValue(); + g = new Integer(data[1]).intValue(); + b = new Integer(data[2]).intValue(); + foregroundColor = new Color(display, r, g, b); + } + if (StringUtility.isEmpty(data[3] + data[4] + data[5])) { + backgroundColor = null; + } else { + r = new Integer(data[3]).intValue(); + g = new Integer(data[4]).intValue(); + b = new Integer(data[5]).intValue(); + backgroundColor = new Color(display, r, g, b); + } + style = b = new Integer(data[6]).intValue(); + } catch (Exception ex) { + foregroundColor = new Color(display, 0, 0, 0); + backgroundColor = null; + style = SWT.NORMAL; + } + } else { + foregroundColor = new Color(display, 0, 0, 0); + backgroundColor = null; + style = SWT.NORMAL; + } + Font font = JFaceResources.getTextFont(); + FontData fontData = font.getFontData()[0]; + fontData = new FontData(fontData.getName(), fontData.getHeight(), style); + font = new Font(display, fontData); + result = new TextAttribute(foregroundColor, backgroundColor, style, + font); + return result; + } + + /** + * Converts a text attribute to a string, except for the font. + * + * @param textAttribute + * The text attribute, not <code>null</code>. + * @return The string, not <code>null</code>. + */ + public static String toString(TextAttribute textAttribute) { + if (textAttribute == null) { + throw new IllegalArgumentException( + "Parameter 'textAttribute' must not be null."); + } + + String result; + result = toString(textAttribute.getForeground()) + "," + + toString(textAttribute.getBackground()) + "," + + Integer.toString(textAttribute.getStyle()); + return result; + } + + /** + * Converts a color to a comma separated RGB string. + * + * @param color + * The color, may be <code>null</code>. + * @return The comma separated RGB string, not <code>null</code>. + */ + private static String toString(Color color) { + String result; + if (color == null) { + result = ",,"; + } else { + String red = Integer.toString(color.getRed()); + String green = Integer.toString(color.getGreen()); + String blue = Integer.toString(color.getBlue()); + result = red + "," + green + "," + blue; + } + return result; + } + + /** + * Dispose the colors and the font of the text attribute created by this class. + * + * @param textAttribute + * The text attribute or <code>null</code>. + * + * @since 1.6.0 + */ + public static void dispose(TextAttribute textAttribute) { + if (textAttribute != null) { + if (textAttribute.getForeground() != null) { + textAttribute.getForeground().dispose(); + } + if (textAttribute.getBackground() != null) { + textAttribute.getBackground().dispose(); + } + if (textAttribute.getFont() != null) { + textAttribute.getFont().dispose(); + } + } + + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/TextAttributeListContentProvider.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/TextAttributeListContentProvider.java new file mode 100644 index 00000000..556b28ce --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/TextAttributeListContentProvider.java @@ -0,0 +1,61 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.preferences; + +import org.eclipse.jface.viewers.IStructuredContentProvider; +import org.eclipse.jface.viewers.Viewer; +import java.util.List; + +/** + * List content provider. + * + * @author Peter Dell + */ +final class TextAttributeListContentProvider implements + IStructuredContentProvider { + + TextAttributeListContentProvider() { + } + + /** + * {@inheritDoc} + */ + @Override + @SuppressWarnings("unchecked") + public Object[] getElements(Object inputElement) { + return ((List<TextAttributeListItem>) inputElement).toArray(); + } + + /** + * {@inheritDoc} + */ + @Override + public void dispose() { + // Not used. + } + + /** + * {@inheritDoc} + */ + @Override + public void inputChanged(Viewer viewer, Object oldInput, Object newInput) { + // Not used. + } +} \ No newline at end of file diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/TextAttributeListItem.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/TextAttributeListItem.java new file mode 100644 index 00000000..89773053 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/TextAttributeListItem.java @@ -0,0 +1,98 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.preferences; + +import org.eclipse.jface.text.TextAttribute; + +/** + * Item in the highlighting color list. + * + * @author Peter Dell + */ +final class TextAttributeListItem { + + /** Display name */ + private String displayName; + + /** Color preference key */ + private String preferencesKey; + + /** Text attribute */ + private TextAttribute textAttribute; + + TextAttributeListItem(String displayName, String preferencesKey) { + if (displayName == null) { + throw new IllegalArgumentException( + "Parameter 'displayName' must not be null."); + } + if (preferencesKey == null) { + throw new IllegalArgumentException( + "Parameter 'preferencesKey' must not be null."); + } + this.displayName = displayName; + this.preferencesKey = preferencesKey; + } + + /** + * Gets the preferences key. + * + * @return The preferences key, not empty and not <code>null</code>. + */ + public String getPreferencesKey() { + return preferencesKey; + } + + /** + * Gets the display name. + * + * @return The display name, not empty and not <code>null</code>. + */ + public String getDisplayName() { + return displayName; + } + + /** + * Sets the text attribute. + * + * @param textAttribute + * The text attribute, not <code>null</code>. + * + */ + public void setTextAttribute(TextAttribute textAttribute) { + if (textAttribute == null) { + throw new IllegalArgumentException( + "Parameter 'textAttribute' must not be null."); + } + this.textAttribute = textAttribute; + } + + /** + * Gets the text attribute. + * + * @return The text attribute, not <code>null</code>. + */ + public TextAttribute getTextAttribute() { + if (textAttribute == null) { + throw new IllegalStateException( + "Field 'textAttribute' must not be null."); + } + return textAttribute; + } +} \ No newline at end of file diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/TextAttributeListItemProvider.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/TextAttributeListItemProvider.java new file mode 100644 index 00000000..6999038b --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/preferences/TextAttributeListItemProvider.java @@ -0,0 +1,70 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.preferences; + +import org.eclipse.jface.viewers.IColorProvider; +import org.eclipse.jface.viewers.IFontProvider; +import org.eclipse.jface.viewers.LabelProvider; +import org.eclipse.swt.graphics.Color; +import org.eclipse.swt.graphics.Font; + +/** + * Color list label provider. + * + * @author Peter Dell + */ +final class TextAttributeListItemProvider extends LabelProvider implements + IColorProvider, IFontProvider { + + TextAttributeListItemProvider() { + + } + + @Override + public String getText(Object element) { + return ((TextAttributeListItem) element).getDisplayName(); + } + + /** + * {@inheritDoc} + */ + @Override + public Color getForeground(Object element) { + return ((TextAttributeListItem) element).getTextAttribute() + .getForeground(); + } + + /** + * {@inheritDoc} + */ + @Override + public Color getBackground(Object element) { + return ((TextAttributeListItem) element).getTextAttribute() + .getBackground(); + } + + /** + * {@inheritDoc} + */ + @Override + public Font getFont(Object element) { + return ((TextAttributeListItem) element).getTextAttribute().getFont(); + } +} \ No newline at end of file diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/runner/Runner.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/runner/Runner.java new file mode 100644 index 00000000..d3f47280 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/runner/Runner.java @@ -0,0 +1,111 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.runner; + +import java.io.File; + +import com.wudsn.ide.asm.compiler.CompilerFiles; +import com.wudsn.ide.asm.editor.AssemblerBreakpoint; + +/** + * Base class for runner implementations. + * + * @author Peter Dell + */ +public class Runner { + + private RunnerDefinition definition; + + /** + * Creation is protected. + */ + protected Runner() { + + } + + /** + * Sets the definition of the Runner. Called by {@link RunnerRegistry} only. + * + * @param definition + * The definition if the Runner, not <code>null</code>. + */ + final void setDefinition(RunnerDefinition definition) { + if (definition == null) { + throw new IllegalArgumentException("Parameter 'type' must not be null."); + } + this.definition = definition; + } + + /** + * Gets the definition of the Runner. + * + * @return The definition of the Runner, not <code>null</code>. + */ + public final RunnerDefinition getDefinition() { + if (definition == null) { + throw new IllegalStateException("Field 'definition' must not be null."); + } + return definition; + } + + + /** + * Creates the {@link File} object for the breakpoints file. + * + * @param files + * The assembler editor file containing the path to the output + * folder and file, not <code>null</code>. + * + * @return The file to created (if there are breakpoints) or deleted (if + * there are not), or <code>null</code> to indicate that the runner + * does no support breakpoints. + * + * @since 1.6.1 + */ + public File createBreakpointsFile(CompilerFiles files) { + if (files == null) { + throw new IllegalArgumentException("Parameter 'files' must not be null."); + } + return null; + } + + /** + * Creates the content for the breakpoints file. + * + * @param breakpoints + * The array of defined (possibly disabled) breakpoints, may be + * empty, not <code>null</code>. + * @param breakpointBuilder + * The sting builder for create the actual file content. @ + * + * @return The number of active breakpoints or <code>0</code> if no + * breakpoints are active. + * @since 1.6.1 + */ + public int createBreakpointsFileContent(AssemblerBreakpoint[] breakpoints, StringBuilder breakpointBuilder) { + if (breakpoints == null) { + throw new IllegalArgumentException("Parameter 'breakpoints' must not be null."); + } + if (breakpointBuilder == null) { + throw new IllegalArgumentException("Parameter 'breakpointBuilder' must not be null."); + } + return 0; + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/runner/RunnerDefinition.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/runner/RunnerDefinition.java new file mode 100644 index 00000000..363ae72a --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/runner/RunnerDefinition.java @@ -0,0 +1,245 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.runner; + +import com.wudsn.ide.asm.Hardware; + +/** + * Definition of a runner. The definition contains all static meta information + * about the runner. It is normally defined via an extension. + * + * + * For launching application under MacOS see + * https://bugs.eclipse.org/bugs/show_bug.cgi?id=82155 and + * http://www.coderanch.com/t/111494/Mac-OS/launching-Safari-from-Java-App. + * + * @author Peter Dell + */ +public final class RunnerDefinition implements Comparable<RunnerDefinition> { + + public static final String RUNNER_EXECUTABLE_PATH = "${runnerExecutablePath}"; + public static final String OUTPUT_FILE_PATH = "${outputFilePath}"; + + // Id + private Hardware hardware; + private String id; + private String name; + + // Installation and use. + private String homePageURL; + + // Compiling. + private String defaultCommandLine; + + /** + * Creation is package local. Called by {@link RunnerRegistry} only. + */ + RunnerDefinition() { + + } + + /** + * Sets the hardware of the runner. Called by {@link RunnerRegistry} only. + * + * @param hardware + * The hardware, not <code>null</code>. + */ + final void setHardware(Hardware hardware) { + if (hardware == null) { + throw new IllegalArgumentException( + "Parameter 'hardware' must not be null."); + } + this.hardware = hardware; + } + + /** + * Gets the hardware of the runner. + * + * @return The hardware of the runner, not empty and not <code>null</code>. + */ + public final Hardware getHardware() { + if (hardware == null) { + throw new IllegalStateException( + "Field 'hardware' must not be null."); + } + return hardware; + } + + /** + * Sets the id of the runner. Called by {@link RunnerRegistry} only. The id + * is only unique together with the hardware returned by + * {@link #getHardware()}. + * + * @param id + * The id of the runner, not empty and not <code>null</code>. + */ + final void setId(String id) { + if (id == null) { + throw new IllegalArgumentException( + "Parameter 'id' must not be null."); + } + this.id = id; + } + + /** + * Gets the id of the runner. + * + * @return The id of the runner, not empty and not <code>null</code>. + */ + public final String getId() { + if (id == null) { + throw new IllegalStateException("Field 'id' must not be null."); + } + return id; + } + + /** + * Sets the name of the runner. Called by {@link RunnerRegistry} only. + * + * @param name + * The name of the runner, not empty and not <code>null</code>. + */ + final void setName(String name) { + if (name == null) { + throw new IllegalArgumentException( + "Parameter 'name' must not be null."); + } + this.name = name; + } + + /** + * Gets the name of the runner. + * + * @return The name of the runner, not empty and not <code>null</code>. + */ + public final String getName() { + if (name == null) { + throw new IllegalStateException("Field 'name' must not be null."); + } + return name; + } + + /** + * Determines if the runner allows a runner executable path to be + * configured. + * + * @return <code>true</code> if the runner allows a runner executable path + * to be configured, <code>false</code> otherwise. + */ + public final boolean isRunnerExecutablePathPossible() { + boolean result; + + result = id.equals(RunnerId.USER_DEFINED_APPLICATION) + || defaultCommandLine.contains(RUNNER_EXECUTABLE_PATH); + return result; + } + + /** + * Sets the absolute URL of the home page where the runner can be + * downloaded. Called by {@link RunnerRegistry} only. + * + * @param homePageURL + * The absolute URL of the home page where the runner can be + * downloaded. May be empty or <code>null</code>. + */ + final void setHomePageURL(String homePageURL) { + if (homePageURL == null) { + homePageURL = ""; + } + this.homePageURL = homePageURL; + } + + /** + * Gets the absolute URL of the home page where the runner can be + * downloaded. + * + * @return The absolute URL of the home page where the runner can be + * downloaded. The result may be empty, not <code>null</code>. + */ + public final String getHomePageURL() { + if (homePageURL == null) { + throw new IllegalStateException( + "Field 'homePageURL' must not be null."); + } + return homePageURL; + } + + /** + * Sets the runner default command line. Called by {@link RunnerRegistry} + * only. + * + * @param defaultCommandLine + * The runner default parameters, may be empty or + * <code>null</code>. + */ + final void setDefaultCommandLine(String defaultCommandLine) { + if (defaultCommandLine == null) { + defaultCommandLine = ""; + } + this.defaultCommandLine = defaultCommandLine; + } + + /** + * Gets the runner default command line. + * + * @return The runner default parameters, not <code>null</code>. + */ + public final String getDefaultCommandLine() { + if (defaultCommandLine == null) { + throw new IllegalStateException( + "Field 'defaultCommandLine' must not be null."); + } + return defaultCommandLine; + } + + /** + * Compare instances of this class based on their name. + * + * @param o + * The object to compare this one with, not <code>null</code>. + */ + @Override + public final int compareTo(RunnerDefinition o) { + if (o == null) { + throw new IllegalArgumentException( + "Parameter 'o' must not be null."); + } + if (name == null || o.name == null) { + throw new IllegalStateException( + "Field 'name' must not be null for this or for argument."); + + } + int result = id.compareTo(o.id); + if (result == 0) { + return 0; + } + if (id.equals(RunnerId.DEFAULT_APPLICATION)) { + return -1; + } else if (id.equals(RunnerId.USER_DEFINED_APPLICATION)) { + return +1; + } + return name.compareTo(o.name); + } + + @Override + public final String toString() { + return hardware.toString().toLowerCase() + "." + id; + } +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/runner/RunnerId.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/runner/RunnerId.java new file mode 100644 index 00000000..c6fedd38 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/runner/RunnerId.java @@ -0,0 +1,40 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.runner; + +/** + * Predefined runner ids which are valid for all hardwares. + * + * @author Peter Dell + */ +public final class RunnerId { + + /** + * Creation is private. + */ + private RunnerId() { + } + + // The ID of the system default application for a file extension + public final static String DEFAULT_APPLICATION = "default_application"; + // The ID of the user defined application + public final static String USER_DEFINED_APPLICATION = "user_defined_application"; + +} diff --git a/com.wudsn.ide.asm/src/com/wudsn/ide/asm/runner/RunnerRegistry.java b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/runner/RunnerRegistry.java new file mode 100644 index 00000000..9e490345 --- /dev/null +++ b/com.wudsn.ide.asm/src/com/wudsn/ide/asm/runner/RunnerRegistry.java @@ -0,0 +1,233 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.asm.runner; + +import java.util.ArrayList; +import java.util.Collections; +import java.util.List; +import java.util.Map; +import java.util.TreeMap; + +import org.eclipse.core.runtime.CoreException; +import org.eclipse.core.runtime.IConfigurationElement; +import org.eclipse.core.runtime.IExtension; +import org.eclipse.core.runtime.IExtensionPoint; +import org.eclipse.core.runtime.IExtensionRegistry; +import org.eclipse.core.runtime.Platform; + +import com.wudsn.ide.asm.Hardware; + +/** + * Registry for runners, based on the extension points + * {@value RunnerRegistry#RUNNERS}. + * + * @author Peter Dell + * + */ +public final class RunnerRegistry { + + /** + * The id of the extension point which provides the runners. + */ + private static final String RUNNERS = "com.wudsn.ide.asm.runners"; + + /** + * The registered runner definition. + */ + private List<RunnerDefinition> runnerDefinitionList; + + /** + * The cached map of runner instances. + */ + private Map<String, Runner> runnerMap; + + /** + * Creation is public. + */ + public RunnerRegistry() { + runnerDefinitionList = Collections.emptyList(); + runnerMap = Collections.emptyMap(); + + } + + /** + * Initializes the list of available runners. + */ + public void init() { + + runnerDefinitionList = new ArrayList<RunnerDefinition>(); + runnerMap = new TreeMap<String, Runner>(); + + IExtensionRegistry extensionRegistry = Platform.getExtensionRegistry(); + IExtensionPoint extensionPoint = extensionRegistry + .getExtensionPoint(RUNNERS); + IExtension[] extensions = extensionPoint.getExtensions(); + + for (IExtension extension : extensions) { + IConfigurationElement[] configurationElements = extension + .getConfigurationElements(); + for (IConfigurationElement configurationElement : configurationElements) { + + RunnerDefinition runnerDefinition; + runnerDefinition = new RunnerDefinition(); + runnerDefinition.setId(configurationElement.getAttribute("id")); + runnerDefinition + .setHardware(Hardware.valueOf(configurationElement + .getAttribute("hardware"))); + runnerDefinition.setName(configurationElement + .getAttribute("name")); + runnerDefinition.setHomePageURL(configurationElement + .getAttribute("homePageURL")); + runnerDefinition.setDefaultCommandLine(configurationElement + .getAttribute("defaultCommandLine")); + + runnerDefinitionList.add(runnerDefinition); + + addRunner(configurationElement, runnerDefinition); + } + } + + runnerDefinitionList = new ArrayList<RunnerDefinition>( + runnerDefinitionList); + Collections.sort(runnerDefinitionList); + runnerDefinitionList = Collections + .unmodifiableList(runnerDefinitionList); + runnerMap = Collections.unmodifiableMap(runnerMap); + } + + /** + * Adds a new runner. + * + * @param configurationElement + * The configuration element used as class instance factory, not + * <code>null</code>. + * + * @param runnerDefinition + * The runner definition, not <code>null</code>. + */ + private void addRunner(IConfigurationElement configurationElement, + RunnerDefinition runnerDefinition) { + if (configurationElement == null) { + throw new IllegalArgumentException( + "Parameter 'configurationElement' must not be null."); + } + if (runnerDefinition == null) { + throw new IllegalArgumentException( + "Parameter 'runnerDefinition' must not be null."); + } + + String id = runnerDefinition.getHardware().toString().toLowerCase() + + "." + runnerDefinition.getId(); + + // Optionally use a specific runner implementation class. + Runner runner; + if (configurationElement.getAttribute("class") != null) { + try { + // The class loading must be delegated to the framework. + runner = (Runner) configurationElement + .createExecutableExtension("class"); + } catch (CoreException ex) { + throw new RuntimeException( + "Cannot create runner instance for id '" + id + "'.", + ex); + } + } else { + runner = new Runner(); + } + + runner.setDefinition(runnerDefinition); + runner = runnerMap.put(id, runner); + if (runner != null) { + throw new RuntimeException("Runner with id '" + + runnerDefinition.getId() + "' for hardware '" + + runnerDefinition.getHardware().toString() + + "' is already registered to class '" + + runner.getClass().getName() + "'."); + } + + } + + /** + * Gets the unmodifiable list of runner definitions, sorted by their name. + * + * @param hardware + * The hardware used for filtering, not <code>null</code>. + * + * @return The unmodifiable list of runner definitions which have the + * matching hardware or {@link Hardware#GENERIC}, sorted by their + * id, may be empty, not <code>null</code>. + */ + public List<RunnerDefinition> getDefinitions(Hardware hardware) { + if (hardware == null) { + throw new IllegalArgumentException( + "Parameter 'hardware' must not be null."); + } + + List<RunnerDefinition> result = new ArrayList<RunnerDefinition>( + runnerDefinitionList.size()); + for (RunnerDefinition runnerDefinition : runnerDefinitionList) { + if (runnerDefinition.getHardware().equals(hardware) + || runnerDefinition.getHardware().equals(Hardware.GENERIC)) { + result.add(runnerDefinition); + } + } + result = Collections.unmodifiableList(result); + return result; + } + + /** + * Gets the runner for a given id. Instances of runner are stateless + * singletons within the plugin. + * + * @param hardware + * The hardware, not <code>null</code>. + * @param runnerId + * The runner type, see {@link RunnerId}, not <code>null</code>. + * + * @return The runner, not <code>null</code>. + */ + public Runner getRunner(Hardware hardware, String runnerId) { + if (hardware == null) { + throw new IllegalArgumentException( + "Parameter 'hardware' must not be null."); + } + if (runnerId == null) { + throw new IllegalArgumentException( + "Parameter 'runnerId' must not be null."); + } + Runner result; + synchronized (runnerMap) { + + result = runnerMap.get(hardware.toString().toLowerCase() + "." + + runnerId); + if (result == null) { + result = runnerMap.get(Hardware.GENERIC.toString() + .toLowerCase() + "." + runnerId); + } + } + if (result == null) { + + throw new IllegalArgumentException("Unknown runner id '" + runnerId + + "' for hardware '" + hardware.toString() + "'."); + } + + return result; + } +} diff --git a/com.wudsn.ide.base.feature/.project b/com.wudsn.ide.base.feature/.project new file mode 100644 index 00000000..8e536f22 --- /dev/null +++ b/com.wudsn.ide.base.feature/.project @@ -0,0 +1,17 @@ +<?xml version="1.0" encoding="UTF-8"?> +<projectDescription> + <name>com.wudsn.ide.base.feature</name> + <comment></comment> + <projects> + </projects> + <buildSpec> + <buildCommand> + <name>org.eclipse.pde.FeatureBuilder</name> + <arguments> + </arguments> + </buildCommand> + </buildSpec> + <natures> + <nature>org.eclipse.pde.FeatureNature</nature> + </natures> +</projectDescription> diff --git a/com.wudsn.ide.base.feature/build.properties b/com.wudsn.ide.base.feature/build.properties new file mode 100644 index 00000000..5f6dacff --- /dev/null +++ b/com.wudsn.ide.base.feature/build.properties @@ -0,0 +1,3 @@ +bin.includes = feature.xml,\ + build.properties,\ + .project diff --git a/com.wudsn.ide.base.feature/feature.xml b/com.wudsn.ide.base.feature/feature.xml new file mode 100644 index 00000000..5c926ec5 --- /dev/null +++ b/com.wudsn.ide.base.feature/feature.xml @@ -0,0 +1,321 @@ +<?xml version="1.0" encoding="UTF-8"?> +<feature + id="com.wudsn.ide.base.feature" + label="WUDSN Base Feature" + version="1.7.0.qualifier" + provider-name="WUDSN"> + + <description url="http:://www.wudsn.com"> + General IDE Enhancements for Eclipse like "Open Folder" context +menu in package explorer and "Sort" context menu in text editors +which are normally part of WUDSN IDE. This feature can be installed +in an Eclipse instance even if this instance is used to develop +WUDSN IDE or other Java code. Do not install this feature if +you plan to install WUDSN IDE itself. + </description> + + <copyright> + WUDSN IDE Copyright (C) 2009 - 2014 Peter Dell +66822 Lebach, Germany +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, version 2 of the License. +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, see http://www.gnu.org/licenses. + </copyright> + + <license url="http://www.gnu.org/licenses"> + GNU GENERAL PUBLIC LICENSE + Version 2, June 1991 + + Copyright (C) 1989, 1991 Free Software Foundation, Inc., + 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + + Preamble + + The licenses for most software are designed to take away your +freedom to share and change it. By contrast, the GNU General Public +License is intended to guarantee your freedom to share and change free +software--to make sure the software is free for all its users. This +General Public License applies to most of the Free Software +Foundation's software and to any other program whose authors commit to +using it. (Some other Free Software Foundation software is covered by +the GNU Lesser General Public License instead.) You can apply it to +your programs, too. + + When we speak of free software, we are referring to freedom, not +price. Our General Public Licenses are designed to make sure that you +have the freedom to distribute copies of free software (and charge for +this service if you wish), that you receive source code or can get it +if you want it, that you can change the software or use pieces of it +in new free programs; and that you know you can do these things. + + To protect your rights, we need to make restrictions that forbid +anyone to deny you these rights or to ask you to surrender the rights. +These restrictions translate to certain responsibilities for you if you +distribute copies of the software, or if you modify it. + + For example, if you distribute copies of such a program, whether +gratis or for a fee, you must give the recipients all the rights that +you have. You must make sure that they, too, receive or can get the +source code. And you must show them these terms so they know their +rights. + + We protect your rights with two steps: (1) copyright the software, and +(2) offer you this license which gives you legal permission to copy, +distribute and/or modify the software. + + Also, for each author's protection and ours, we want to make certain +that everyone understands that there is no warranty for this free +software. If the software is modified by someone else and passed on, we +want its recipients to know that what they have is not the original, so +that any problems introduced by others will not reflect on the original +authors' reputations. + + Finally, any free program is threatened constantly by software +patents. We wish to avoid the danger that redistributors of a free +program will individually obtain patent licenses, in effect making the +program proprietary. To prevent this, we have made it clear that any +patent must be licensed for everyone's free use or not licensed at all. + + The precise terms and conditions for copying, distribution and +modification follow. + + GNU GENERAL PUBLIC LICENSE + TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION + + 0. This License applies to any program or other work which contains +a notice placed by the copyright holder saying it may be distributed +under the terms of this General Public License. The "Program", below, +refers to any such program or work, and a "work based on the Program" +means either the Program or any derivative work under copyright law: +that is to say, a work containing the Program or a portion of it, +either verbatim or with modifications and/or translated into another +language. (Hereinafter, translation is included without limitation in +the term "modification".) Each licensee is addressed as "you". + +Activities other than copying, distribution and modification are not +covered by this License; they are outside its scope. The act of +running the Program is not restricted, and the output from the Program +is covered only if its contents constitute a work based on the +Program (independent of having been made by running the Program). +Whether that is true depends on what the Program does. + + 1. You may copy and distribute verbatim copies of the Program's +source code as you receive it, in any medium, provided that you +conspicuously and appropriately publish on each copy an appropriate +copyright notice and disclaimer of warranty; keep intact all the +notices that refer to this License and to the absence of any warranty; +and give any other recipients of the Program a copy of this License +along with the Program. + +You may charge a fee for the physical act of transferring a copy, and +you may at your option offer warranty protection in exchange for a fee. + + 2. You may modify your copy or copies of the Program or any portion +of it, thus forming a work based on the Program, and copy and +distribute such modifications or work under the terms of Section 1 +above, provided that you also meet all of these conditions: + + a) You must cause the modified files to carry prominent notices + stating that you changed the files and the date of any change. + + b) You must cause any work that you distribute or publish, that in + whole or in part contains or is derived from the Program or any + part thereof, to be licensed as a whole at no charge to all third + parties under the terms of this License. + + c) If the modified program normally reads commands interactively + when run, you must cause it, when started running for such + interactive use in the most ordinary way, to print or display an + announcement including an appropriate copyright notice and a + notice that there is no warranty (or else, saying that you provide + a warranty) and that users may redistribute the program under + these conditions, and telling the user how to view a copy of this + License. (Exception: if the Program itself is interactive but + does not normally print such an announcement, your work based on + the Program is not required to print an announcement.) + +These requirements apply to the modified work as a whole. If +identifiable sections of that work are not derived from the Program, +and can be reasonably considered independent and separate works in +themselves, then this License, and its terms, do not apply to those +sections when you distribute them as separate works. But when you +distribute the same sections as part of a whole which is a work based +on the Program, the distribution of the whole must be on the terms of +this License, whose permissions for other licensees extend to the +entire whole, and thus to each and every part regardless of who wrote it. + +Thus, it is not the intent of this section to claim rights or contest +your rights to work written entirely by you; rather, the intent is to +exercise the right to control the distribution of derivative or +collective works based on the Program. + +In addition, mere aggregation of another work not based on the Program +with the Program (or with a work based on the Program) on a volume of +a storage or distribution medium does not bring the other work under +the scope of this License. + + 3. You may copy and distribute the Program (or a work based on it, +under Section 2) in object code or executable form under the terms of +Sections 1 and 2 above provided that you also do one of the following: + + a) Accompany it with the complete corresponding machine-readable + source code, which must be distributed under the terms of Sections + 1 and 2 above on a medium customarily used for software interchange; or, + + b) Accompany it with a written offer, valid for at least three + years, to give any third party, for a charge no more than your + cost of physically performing source distribution, a complete + machine-readable copy of the corresponding source code, to be + distributed under the terms of Sections 1 and 2 above on a medium + customarily used for software interchange; or, + + c) Accompany it with the information you received as to the offer + to distribute corresponding source code. (This alternative is + allowed only for noncommercial distribution and only if you + received the program in object code or executable form with such + an offer, in accord with Subsection b above.) + +The source code for a work means the preferred form of the work for +making modifications to it. For an executable work, complete source +code means all the source code for all modules it contains, plus any +associated interface definition files, plus the scripts used to +control compilation and installation of the executable. However, as a +special exception, the source code distributed need not include +anything that is normally distributed (in either source or binary +form) with the major components (compiler, kernel, and so on) of the +operating system on which the executable runs, unless that component +itself accompanies the executable. + +If distribution of executable or object code is made by offering +access to copy from a designated place, then offering equivalent +access to copy the source code from the same place counts as +distribution of the source code, even though third parties are not +compelled to copy the source along with the object code. + + 4. You may not copy, modify, sublicense, or distribute the Program +except as expressly provided under this License. Any attempt +otherwise to copy, modify, sublicense or distribute the Program is +void, and will automatically terminate your rights under this License. +However, parties who have received copies, or rights, from you under +this License will not have their licenses terminated so long as such +parties remain in full compliance. + + 5. You are not required to accept this License, since you have not +signed it. However, nothing else grants you permission to modify or +distribute the Program or its derivative works. These actions are +prohibited by law if you do not accept this License. Therefore, by +modifying or distributing the Program (or any work based on the +Program), you indicate your acceptance of this License to do so, and +all its terms and conditions for copying, distributing or modifying +the Program or works based on it. + + 6. Each time you redistribute the Program (or any work based on the +Program), the recipient automatically receives a license from the +original licensor to copy, distribute or modify the Program subject to +these terms and conditions. You may not impose any further +restrictions on the recipients' exercise of the rights granted herein. +You are not responsible for enforcing compliance by third parties to +this License. + + 7. If, as a consequence of a court judgment or allegation of patent +infringement or for any other reason (not limited to patent issues), +conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License. If you cannot +distribute so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you +may not distribute the Program at all. For example, if a patent +license would not permit royalty-free redistribution of the Program by +all those who receive copies directly or indirectly through you, then +the only way you could satisfy both it and this License would be to +refrain entirely from distribution of the Program. + +If any portion of this section is held invalid or unenforceable under +any particular circumstance, the balance of the section is intended to +apply and the section as a whole is intended to apply in other +circumstances. + +It is not the purpose of this section to induce you to infringe any +patents or other property right claims or to contest validity of any +such claims; this section has the sole purpose of protecting the +integrity of the free software distribution system, which is +implemented by public license practices. Many people have made +generous contributions to the wide range of software distributed +through that system in reliance on consistent application of that +system; it is up to the author/donor to decide if he or she is willing +to distribute software through any other system and a licensee cannot +impose that choice. + +This section is intended to make thoroughly clear what is believed to +be a consequence of the rest of this License. + + 8. If the distribution and/or use of the Program is restricted in +certain countries either by patents or by copyrighted interfaces, the +original copyright holder who places the Program under this License +may add an explicit geographical distribution limitation excluding +those countries, so that distribution is permitted only in or among +countries not thus excluded. In such case, this License incorporates +the limitation as if written in the body of this License. + + 9. The Free Software Foundation may publish revised and/or new versions +of the General Public License from time to time. Such new versions will +be similar in spirit to the present version, but may differ in detail to +address new problems or concerns. + +Each version is given a distinguishing version number. If the Program +specifies a version number of this License which applies to it and "any +later version", you have the option of following the terms and conditions +either of that version or of any later version published by the Free +Software Foundation. If the Program does not specify a version number of +this License, you may choose any version ever published by the Free Software +Foundation. + + 10. If you wish to incorporate parts of the Program into other free +programs whose distribution conditions are different, write to the author +to ask for permission. For software which is copyrighted by the Free +Software Foundation, write to the Free Software Foundation; we sometimes +make exceptions for this. Our decision will be guided by the two goals +of preserving the free status of all derivatives of our free software and +of promoting the sharing and reuse of software generally. + + NO WARRANTY + + 11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY +FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN +OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES +PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED +OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF +MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS +TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE +PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING, +REPAIR OR CORRECTION. + + 12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING +WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR +REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, +INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING +OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED +TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY +YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER +PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE +POSSIBILITY OF SUCH DAMAGES. + + END OF TERMS AND CONDITIONS + </license> + + <plugin + id="com.wudsn.ide.base" + download-size="0" + install-size="0" + version="0.0.0" + unpack="false"/> + +</feature> diff --git a/com.wudsn.ide.base/.classpath b/com.wudsn.ide.base/.classpath new file mode 100644 index 00000000..23e0cc4a --- /dev/null +++ b/com.wudsn.ide.base/.classpath @@ -0,0 +1,7 @@ +<?xml version="1.0" encoding="UTF-8"?> +<classpath> + <classpathentry kind="con" path="org.eclipse.jdt.launching.JRE_CONTAINER/org.eclipse.jdt.internal.debug.ui.launcher.StandardVMType/JavaSE-1.6"/> + <classpathentry kind="src" path="src"/> + <classpathentry kind="con" path="org.eclipse.pde.core.requiredPlugins"/> + <classpathentry kind="output" path="bin"/> +</classpath> diff --git a/com.wudsn.ide.base/.project b/com.wudsn.ide.base/.project new file mode 100644 index 00000000..695525d6 --- /dev/null +++ b/com.wudsn.ide.base/.project @@ -0,0 +1,28 @@ +<?xml version="1.0" encoding="UTF-8"?> +<projectDescription> + <name>com.wudsn.ide.base</name> + <comment></comment> + <projects> + </projects> + <buildSpec> + <buildCommand> + <name>org.eclipse.jdt.core.javabuilder</name> + <arguments> + </arguments> + </buildCommand> + <buildCommand> + <name>org.eclipse.pde.ManifestBuilder</name> + <arguments> + </arguments> + </buildCommand> + <buildCommand> + <name>org.eclipse.pde.SchemaBuilder</name> + <arguments> + </arguments> + </buildCommand> + </buildSpec> + <natures> + <nature>org.eclipse.pde.PluginNature</nature> + <nature>org.eclipse.jdt.core.javanature</nature> + </natures> +</projectDescription> diff --git a/com.wudsn.ide.base/.settings/org.eclipse.jdt.core.prefs b/com.wudsn.ide.base/.settings/org.eclipse.jdt.core.prefs new file mode 100644 index 00000000..54e493c0 --- /dev/null +++ b/com.wudsn.ide.base/.settings/org.eclipse.jdt.core.prefs @@ -0,0 +1,11 @@ +eclipse.preferences.version=1 +org.eclipse.jdt.core.compiler.codegen.inlineJsrBytecode=enabled +org.eclipse.jdt.core.compiler.codegen.targetPlatform=1.6 +org.eclipse.jdt.core.compiler.codegen.unusedLocal=preserve +org.eclipse.jdt.core.compiler.compliance=1.6 +org.eclipse.jdt.core.compiler.debug.lineNumber=generate +org.eclipse.jdt.core.compiler.debug.localVariable=generate +org.eclipse.jdt.core.compiler.debug.sourceFile=generate +org.eclipse.jdt.core.compiler.problem.assertIdentifier=error +org.eclipse.jdt.core.compiler.problem.enumIdentifier=error +org.eclipse.jdt.core.compiler.source=1.6 diff --git a/com.wudsn.ide.base/META-INF/MANIFEST.MF b/com.wudsn.ide.base/META-INF/MANIFEST.MF new file mode 100644 index 00000000..2419ae4b --- /dev/null +++ b/com.wudsn.ide.base/META-INF/MANIFEST.MF @@ -0,0 +1,26 @@ +Manifest-Version: 1.0 +Bundle-ManifestVersion: 2 +Bundle-Name: WUDSN IDE Base Plug-in +Bundle-SymbolicName: com.wudsn.ide.base;singleton:=true +Bundle-Version: 1.7.0.qualifier +Bundle-Activator: com.wudsn.ide.base.BasePlugin +Bundle-Localization: plugin +Bundle-Vendor: Peter Dell +Require-Bundle: org.eclipse.core.runtime, + org.eclipse.core.resources, + org.eclipse.core.expressions, + org.eclipse.core.filesystem, + org.eclipse.jface.text, + org.eclipse.ui, + org.eclipse.ui.ide, + org.eclipse.ui.views, + org.eclipse.ui.workbench.texteditor, + org.eclipse.compare +Bundle-RequiredExecutionEnvironment: JavaSE-1.6 +Bundle-ActivationPolicy: lazy +Export-Package: com.wudsn.ide.base, + com.wudsn.ide.base.common, + com.wudsn.ide.base.editor, + com.wudsn.ide.base.editor.hex, + com.wudsn.ide.base.gui +Import-Package: org.eclipse.ui.internal.editors.text diff --git a/com.wudsn.ide.base/bin/.gitignore b/com.wudsn.ide.base/bin/.gitignore new file mode 100644 index 00000000..43e58b99 --- /dev/null +++ b/com.wudsn.ide.base/bin/.gitignore @@ -0,0 +1 @@ +/com diff --git a/com.wudsn.ide.base/build.properties b/com.wudsn.ide.base/build.properties new file mode 100644 index 00000000..0720079e --- /dev/null +++ b/com.wudsn.ide.base/build.properties @@ -0,0 +1,14 @@ +source.. = src/ +output.. = bin/ +bin.includes = META-INF/,\ + .,\ + icons/,\ + plugin.properties,\ + plugin_de_DE.properties,\ + plugin.xml,\ + fonts/,\ + src/,\ + build.properties,\ + .project,\ + .classpath + diff --git a/com.wudsn.ide.base/fonts/atari8/AtariClassic-Regular.ttf b/com.wudsn.ide.base/fonts/atari8/AtariClassic-Regular.ttf new file mode 100644 index 0000000000000000000000000000000000000000..073c065be591fd73b1c53164a2fda0c12b37985a GIT binary patch literal 29096 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 HOZ>k8*h82v literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.base/fonts/atari8/AtariClassicInt-Regular.ttf b/com.wudsn.ide.base/fonts/atari8/AtariClassicInt-Regular.ttf new file mode 100644 index 0000000000000000000000000000000000000000..3c5b788fbeef43706994a11ecf5419dbd1b25576 GIT binary patch literal 30944 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 a=yw_YE~DSog|0TK|4Y+2{C`Y2N&W{a0Tm|z literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.base/fonts/atari8/ReadMe.rtf b/com.wudsn.ide.base/fonts/atari8/ReadMe.rtf new file mode 100644 index 00000000..a36354c3 --- /dev/null +++ b/com.wudsn.ide.base/fonts/atari8/ReadMe.rtf @@ -0,0 +1,317 @@ +{\rtf1\ansi\deff0\deftab720{\fonttbl{\f0\fswiss MS Sans Serif;}{\f1\froman\fcharset2 Symbol;}{\f2\fmodern\fprq1 Atari Classic Chunky;}{\f3\froman Times New Roman;}{\f4\fswiss\fprq2 Arial;}} +{\colortbl\red0\green0\blue0;\red0\green0\blue255;} +\deflang1033\pard\plain\f2\fs24\cf1 Atari Classic TrueType Fonts \plain\f2\fs24\cf0 +\par \plain\f4\fs16\cf0 (Windows Version 1.1) +\par Created by Mark Simonson (v.1.0-1998, v.1.1-2001) +\par marksim@bitstream.net +\par Website: Mac/Atari Fusion--Atari Home Computer Resources for Mac Users +\par http://www2.bitstream.net/~marksim/atarimac/ +\par Macintosh version also available. +\par +\par With these fonts installed, you can view and print Atari text files in any text editor that allows you to change fonts (WordPad, for example). Tip: In order to get the correct line breaks, you will need to change the ATASCII return character (155) to the DOS LF character. (In the Character Map accessory, the ATASCII return is the blank character that comes just before the inverse up-arrow.) +\par +\par There are three different fonts. \plain\f4\fs16\cf0\b Atari Classic Chunky \plain\f4\fs16\cf0 is a pixel-for-pixel copy of the original ATASCII character set. \plain\f4\fs16\cf0\b Atari Classic Smooth \plain\f4\fs16\cf0 interprets the pixel aliasing (stair steps) as diagonal lines. \plain\f4\fs16\cf0\b Atari Classic Extrasmooth \plain\f4\fs16\cf0 refines this idea further with the addition of curves. \plain\f4\fs16\cf0\b Smooth\plain\f4\fs16\cf0 and \plain\f4\fs16\cf0\b Extrasmooth\plain\f4\fs16\cf0 were designed for better appearance and legibility at larger sizes and on print-outs. Use the one that looks best to you. +\par +\par These fonts will tend to look uneven at font sizes that do not correspond to the 8-by-8 pixel grid that the characters are based on. Because Windows assumes 96ppi screen resolution, they will look best in a font size that is a multiple of 6 (i.e., 6pt, 12pt, 18pt, etc.). (In Windows, 6 points = 8 pixels.) +\par +\par The Atari Classic TrueType fonts duplicate the ATASCII character set on a low-level basis. Unlike a normal Windows font, ATASCII utilizes all character codes from $00 to $FF (0 to 255). The lower half are normal characters; the upper half are inverse versions of the lower half. The basic ASCII characters ($00 to $7F) correspond fairly closely except for the first 32, which don't normally contain characters in a Windows font. +\par +\par Due to differences between the way Windows and the Atari use character codes, not all characters will display properly in Windows. In fact, some characters will not display at all (though they do exist in the font). Unfortunately, this is due to certain character codes being reserved in Windows and there doesn't appear to be any way to work around it. The character codes affected are: $00-$1F (0-31), $7F-$81 (127-129), $8D-$90 (141-144), $9D (157), and $9F (158). +\par +\par Not all characters can be typed from the keyboard. You can however copy characters as needed from this document (see tables below). The Character Map desk accessory can help also. +\par +\par \plain\f4\fs16\cf0\b ATASCII CHARACTER SET TABLES +\par \plain\f4\fs16\cf0 +\par In order to see the ATASCII character set with these tables, the Atari Classic TrueType fonts must be installed. Characters that are not displayed properly are due to character code usage differences between ATASCII and Windows (see above). +\par +\par +\par \plain\f4\fs16\cf0\b TABLE 1: ATASCII Character Dump Block +\par \plain\f4\fs16\cf0 +\par All characters (ATASCII $00 thru $FF) 16 characters per +\par line. +\par +\par +\par \plain\f2\fs12\cf0 \'01\'02\'03\'04\'05\'06\'07\'08\tab +\par \'0b\'0c +\par \'0e\'0f +\par \'10\'11\'12\'13\'14\'15\'16\'17\'18\'19\'1a\'1b\'1c\'1d\'1e\'1f +\par !"#$%&'()*+,-./ +\par 0123456789:;<=>? +\par @ABCDEFGHIJKLMNO +\par PQRSTUVWXYZ[\\]^_ +\par `abcdefghijklmno +\par pqrstuvwxyz\{|\}~ +\par \'80\'81\'82\'83\'84\'85\'86\'87\'88\'89\'8a\'8b\'8c\'8d\'8e\'8f +\par \'90''""\bullet \endash \emdash \'98\'99\'9a \'9c\'9d\'9e\'9f +\par \~\'a1\'a2\'a3\'a4\'a5\'a6\'a7\'a8\'a9\'aa\'ab\'ac\'ad\'ae\'af +\par \'b0\'b1\'b2\'b3\'b4\'b5\'b6\'b7\'b8\'b9\'ba\'bb\'bc\'bd\'be\'bf +\par \'c0\'c1\'c2\'c3\'c4\'c5\'c6\'c7\'c8\'c9\'ca\'cb\'cc\'cd\'ce\'cf +\par \'d0\'d1\'d2\'d3\'d4\'d5\'d6\'d7\'d8\'d9\'da\'db\'dc\'dd\'de\'df +\par \'e0\'e1\'e2\'e3\'e4\'e5\'e6\'e7\'e8\'e9\'ea\'eb\'ec\'ed\'ee\'ef +\par \'f0\'f1\'f2\'f3\'f4\'f5\'f6\'f7\'f8\'f9\'fa\'fb\'fc\'fd\'fe\'ff +\par \plain\f4\fs16\cf0 +\par +\par \plain\f4\fs16\cf0\b TABLE 2: ATASCII Character Dump List +\par \plain\f4\fs16\cf0 +\par All characters (ATASCII $00 thru $FF) one character per +\par line with hexadecimal value indicated on the left. +\par +\par \plain\f2\fs12\cf0 00= +\par 01=\'01 +\par 02=\'02 +\par 03=\'03 +\par 04=\'04 +\par 05=\'05 +\par 06=\'06 +\par 07=\'07 +\par 08=\'08 +\par 09=\tab +\par 0A= +\par +\par 0B=\'0b +\par 0C=\'0c +\par 0D= +\par 0E=\'0e +\par 0F=\'0f +\par 10=\'10 +\par 11=\'11 +\par 12=\'12 +\par 13=\'13 +\par 14=\'14 +\par 15=\'15 +\par 16=\'16 +\par 17=\'17 +\par 18=\'18 +\par 19=\'19 +\par 1A=\'1a +\par 1B=\'1b +\par 1C=\'1c +\par 1D=\'1d +\par 1E=\'1e +\par 1F=\'1f +\par 20= +\par 21=! +\par 22=" +\par 23=# +\par 24=$ +\par 25=% +\par 26=& +\par 27=' +\par 28=( +\par 29=) +\par 2A=* +\par 2B=+ +\par 2C=, +\par 2D=- +\par 2E=. +\par 2F=/ +\par 30=0 +\par 31=1 +\par 32=2 +\par 33=3 +\par 34=4 +\par 35=5 +\par 36=6 +\par 37=7 +\par 38=8 +\par 39=9 +\par 3A=: +\par 3B=; +\par 3C=< +\par 3D== +\par 3E=> +\par 3F=? +\par 40=@ +\par 41=A +\par 42=B +\par 43=C +\par 44=D +\par 45=E +\par 46=F +\par 47=G +\par 48=H +\par 49=I +\par 4A=J +\par 4B=K +\par 4C=L +\par 4D=M +\par 4E=N +\par 4F=O +\par 50=P +\par 51=Q +\par 52=R +\par 53=S +\par 54=T +\par 55=U +\par 56=V +\par 57=W +\par 58=X +\par 59=Y +\par 5A=Z +\par 5B=[ +\par 5C=\\ +\par 5D=] +\par 5E=^ +\par 5F=_ +\par 60=` +\par 61=a +\par 62=b +\par 63=c +\par 64=d +\par 65=e +\par 66=f +\par 67=g +\par 68=h +\par 69=i +\par 6A=j +\par 6B=k +\par 6C=l +\par 6D=m +\par 6E=n +\par 6F=o +\par 70=p +\par 71=q +\par 72=r +\par 73=s +\par 74=t +\par 75=u +\par 76=v +\par 77=w +\par 78=x +\par 79=y +\par 7A=z +\par 7B=\{ +\par 7C=| +\par 7D=\} +\par 7E=~ +\par 7F= +\par 80=\'80 +\par 81=\'81 +\par 82=\'82 +\par 83=\'83 +\par 84=\'84 +\par 85=\'85 +\par 86=\'86 +\par 87=\'87 +\par 88=\'88 +\par 89=\'89 +\par 8A=\'8a +\par 8B=\'8b +\par 8C=\'8c +\par 8D=\'8d +\par 8E=\'8e +\par 8F=\'8f +\par 90=\'90 +\par 91=' +\par 92=' +\par 93=" +\par 94=" +\par 95=\bullet +\par 96=\endash +\par 97=\emdash +\par 98=\'98 +\par 99=\'99 +\par 9A=\'9a +\par 9B= +\par 9C=\'9c +\par 9D=\'9d +\par 9E=\'9e +\par 9F=\'9f +\par A0=\~ +\par A1=\'a1 +\par A2=\'a2 +\par A3=\'a3 +\par A4=\'a4 +\par A5=\'a5 +\par A6=\'a6 +\par A7=\'a7 +\par A8=\'a8 +\par A9=\'a9 +\par AA=\'aa +\par AB=\'ab +\par AC=\'ac +\par AD=\'ad +\par AE=\'ae +\par AF=\'af +\par B0=\'b0 +\par B1=\'b1 +\par B2=\'b2 +\par B3=\'b3 +\par B4=\'b4 +\par B5=\'b5 +\par B6=\'b6 +\par B7=\'b7 +\par B8=\'b8 +\par B9=\'b9 +\par BA=\'ba +\par BB=\'bb +\par BC=\'bc +\par BD=\'bd +\par BE=\'be +\par BF=\'bf +\par C0=\'c0 +\par C1=\'c1 +\par C2=\'c2 +\par C3=\'c3 +\par C4=\'c4 +\par C5=\'c5 +\par C6=\'c6 +\par C7=\'c7 +\par C8=\'c8 +\par C9=\'c9 +\par CA=\'ca +\par CB=\'cb +\par CC=\'cc +\par CD=\'cd +\par CE=\'ce +\par CF=\'cf +\par D0=\'d0 +\par D1=\'d1 +\par D2=\'d2 +\par D3=\'d3 +\par D4=\'d4 +\par D5=\'d5 +\par D6=\'d6 +\par D7=\'d7 +\par D8=\'d8 +\par D9=\'d9 +\par DA=\'da +\par DB=\'db +\par DC=\'dc +\par DD=\'dd +\par DE=\'de +\par DF=\'df +\par E0=\'e0 +\par E1=\'e1 +\par E2=\'e2 +\par E3=\'e3 +\par E4=\'e4 +\par E5=\'e5 +\par E6=\'e6 +\par E7=\'e7 +\par E8=\'e8 +\par E9=\'e9 +\par EA=\'ea +\par EB=\'eb +\par EC=\'ec +\par ED=\'ed +\par EE=\'ee +\par EF=\'ef +\par F0=\'f0 +\par F1=\'f1 +\par F2=\'f2 +\par F3=\'f3 +\par F4=\'f4 +\par F5=\'f5 +\par F6=\'f6 +\par F7=\'f7 +\par F8=\'f8 +\par F9=\'f9 +\par FA=\'fa +\par FB=\'fb +\par FC=\'fc +\par FD=\'fd +\par FE=\'fe +\par FF=\'ff +\par } +� \ No newline at end of file diff --git a/com.wudsn.ide.base/fonts/c64/C64Classic-Regular.ttf b/com.wudsn.ide.base/fonts/c64/C64Classic-Regular.ttf new file mode 100644 index 0000000000000000000000000000000000000000..1583ded86be9b68f7ed591b4e7263805f6739a8f GIT binary patch literal 35036 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 literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.base/icons/copy-disabled.gif b/com.wudsn.ide.base/icons/copy-disabled.gif new file mode 100644 index 0000000000000000000000000000000000000000..98626a819b92fb36008c642fbc659c735a96671a GIT binary patch literal 366 zcmZ?wbhEHb6krfwxXQrr`}dz;zkdJx`Rm7zp9c>eJ#gswx{cdUow@Mo^Vg4`zI^!j z`Ox8`8@KG-wfE4@JqNe#+_!bd-p$)~?>~5C-@&8DPn|n@;>_V=rw$%D@%_iouiw6Z z`TFh9;Uh<m9zAj5=<(x6j~zRD=Je51Cy$&udGzFoqi0U+KYe`P&1>f`ojZ2u;K4tC z{`~*{pMf%<_>+Z^fkBHw2V@M$PYi6`4pR#}bhrXfs9dV>v2+ZWFsXydX6}s;PE!|F zzUdQ8P3<P`(3+y4AW*B%&wQZd)-qQ1QyF{te*BHmkdR=Otq>6t<`v*!YLu*$Yi4eq zD$yt<DJegpqpL@9My1TmS<7ZNa;%b;s_g9Mky^<v%Drm)jz%snP8N<WtbBr;M;bka Sc6098b1Ea&>w<<OgEau}EuHlM literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.base/icons/copy-enabled.gif b/com.wudsn.ide.base/icons/copy-enabled.gif new file mode 100644 index 0000000000000000000000000000000000000000..71d7c95aa8caa2e8f139d262a4e5e08aba2b88e5 GIT binary patch literal 594 zcmZ?wbhEHb6krfwc*el+`Rnf=KmR0GEr0jv*Qc+)fByRO<LB?Vvc+k2E3%r_<hHCW z?ATb`xv6H-j=IUadKMi1@a1>o^gS)J_IJ)ZIC1ImdoO<6eg6Iai|<d~{CxiI=i5)e zK7adt2PpFL$GsQdAHMqW>ch`(KYl-Z_w&Qo-=Dty{`%wh_n*IK&6+S{#>BaEdS}g? zuy9`2swFMU7Pc>%*EMf;?~yIV2RD`OU0=RsRqfVQbsLs9u3Fr>Y(e|+t%XN67a!VG za&TkmzV+q%)>rIYQ?+Yt^@e4Qs~5LCIMIJ)Q^u7|nYZ@VT-}uU|Nno6Ap#VCvM@3* zxHIU0+zW~m2KMz0?oG`ttxXmkot<s&7OeeUT`iVsjBI`VGhHm2tyDz>4W_b9Wps5j zXfk4vRxz9}$i=Sc<mzngZm%rDZ@WfCnnC)ow2ixoloYewZhj$N9!@3!J9krsJ;o=Q zPu`U>ao16jlytap?!qMnQ+Hj7J!X%ee17cct|$M~{QZlIOcJ_%OI|o69_p&$5Z-06 psEL`Ko0;1xqTxYPJC}anf*T!+9v<!&(<!hB>{#qMxsidv8UQV$-}(Rm literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.base/icons/cut-disabled.gif b/com.wudsn.ide.base/icons/cut-disabled.gif new file mode 100644 index 0000000000000000000000000000000000000000..bd43144729af8d6a70cdf48ff289542be9a133ed GIT binary patch literal 138 zcmZ?wbhEHb6krfw*v!Dt(%yUe%(<28x9;A5c>bcLH}BlvzH|4Xg9rcr|IdI8DE?$& zWMJT6&;dz;%wS+K7dYv;dhPR=>W+IMY)SG#Me{VhB5$(3RX@itv29I(1k0J!R-Xso YCeL+iKghK*GOtJZs-ncyFeL_S0C&JKA^-pY literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.base/icons/cut-enabled.gif b/com.wudsn.ide.base/icons/cut-enabled.gif new file mode 100644 index 0000000000000000000000000000000000000000..d044e59777b787792b58cd20446f3c9dc6e6194c GIT binary patch literal 212 zcmZ?wbhEHb6krfwIKsei<My+ChtJMkzH{5YGvY?EmF;tV6RMNCHy2OcvF_;I#O}@7 zxr^OvS9{d17L1!M6hB8Oe)j+W{~3q|ia%Kx85jf^bU->lb~3OACaCnKWKMG`nAKpo zCN<zds$uv0TU-|dB$QhUR18G8nHxMNcBt@8WIFgj%h08>?d%F2U7-g@7tAoWRp67E bU?#;O_=e*$=hE1@yx+`a?3a5gGFSrujh8`r literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.base/icons/delete-disabled.gif b/com.wudsn.ide.base/icons/delete-disabled.gif new file mode 100644 index 0000000000000000000000000000000000000000..9e142981b2330ee95980dbfaf14f8b6c93f3b563 GIT binary patch literal 221 zcmV<303!cKNk%w1VGsZi0K@<Q!K<vsu(8ChufD0OwV|T4prOUEvC+Q2)x*Tq#Kp0j zoy@klv!0-`ouAIRyV1VDzp<~ru&%wWth=kKzOJpzy}Y}ss>-;y*~!V=&CT4-&ceaL z|Ns900000000000A^8LW000~SEC2ui01yBW000GAASaGwX`bh%mg+`s&;_|7EYsW; zX<5|nJzP$L!BixbB1KrUBoI)<ggM$64u?m8S`|3H0&b45D3=S=4Wm4049X1QMp0I@ X=nH#Q0npd@el;sEgD8fFB_RMi3At)$ literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.base/icons/hex-editor-16x16.gif b/com.wudsn.ide.base/icons/hex-editor-16x16.gif new file mode 100644 index 0000000000000000000000000000000000000000..a8c58eba18359f2831a2f3cb9f2e164fd0303a32 GIT binary patch literal 365 zcmZ?wbhEHb6krfwxXQrr{l}j#-~OC9cX$2vliz;)`T6V5#Ccmc?z!;r<@cv=eu^8# zK6v@#+1sBl-~W93>DT9Pzwf>H{`Bq7S08@9`~2(Cn;##)0?F^s-~SL(Q@nQTT6Ml% zYpHcZp>1D<<>GFaO*4Gvw!1BBF`nOOI<3b1$O4Zei~Tos8LVwLJT%L8*Hrt{i`}kl z%DlC==Kufy3}gYtpDc_F47v<DAl)E8F|c(Vm{yRXCFOf~$%T*%vv7~40<0^-A_FqL zMPF1gFMql*LUXpW_w&n7GPb*BnqIrI!rbkUaN>FOIT8&#vQCQpZSDM;BHbc#PD;WP zCkd-?Pvcf~QZ8lYViRJMU}B%k<|NC>$;HGazJ^J6HJ6h#2gfGC^&8hX$p{E+65F-O VNlHPUZJy+=lgpyc>pC)60|1relxzS1 literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.base/icons/hex-editor-segment-16x16.gif b/com.wudsn.ide.base/icons/hex-editor-segment-16x16.gif new file mode 100644 index 0000000000000000000000000000000000000000..298871653f056a8123af853bd587a3a394a208f0 GIT binary patch literal 183 zcmZ?wbhEHb6krfwIKsg2JR$LfrojmVlM|YTCuDU#`Nn+ojrm&Ea6(e+`Q+}C&Ne6A z?M~WReVf$xJR#w0P0icp=KufyGY}0Ff3h$#FbFc}fOLTDWMDN(Q0Ys_oR_g`UCwH~ wf;~&Tl5cyRGV#A}pn33MfPoClO2#9M3=T}1PDxg(O9XD{96I`To)d#L0F!by`Tzg` literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.base/icons/import-disabled.gif b/com.wudsn.ide.base/icons/import-disabled.gif new file mode 100644 index 0000000000000000000000000000000000000000..cd4474182f048ef617d0355fbee88208910cfc3c GIT binary patch literal 143 zcmZ?wbhEHb6krfw*v!Dt(lX`Nod?%$+&Odp%GRCx_Z&EO@yd<kr!O2kb^hATJBJP) z{Qv(y0}@dD$->CMz|Eio5(AmRz+zKy(sT9Ru*5ax-{d_?@2;40*>jbbz!WZrUFiX9 oUMvz>uQWy6^-&5#T2~ii<IxZSw&l$wR;~fFBc^o-F)~;K0Q*NfL;wH) literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.base/icons/import-enabled.gif b/com.wudsn.ide.base/icons/import-enabled.gif new file mode 100644 index 0000000000000000000000000000000000000000..d38085ad9c273000d1c7ef3ea0144de87b776e46 GIT binary patch literal 327 zcmZ?wbhEHb6krfwxXQp_Z$IBYetux?yuh5fA^8gfa_0sYEDX$@8&R?(qGWMI>Eg)J z#W5Aj<7-wY)UHmhUz<|DHmzxW>%y~b3(wBod@a6qb$sor<c77ai_W%z(3uIVE-l}E zYx$mA>kr;FE?8n*u-Kt|Id9CY|Ns9p&;Tg@WMO1rP-f5pnGNz216#_0i3J`yQvJsm zgg7JQrk+bO_7dTAKGtDoB+_bkXzjJf5992w1-8iStkCP%_Lx#<V>5y6_`@wFT09~= wevLfb(wrPzoISn09c+@45|YAGBqXFHg(kDGv$8VHpTB^88S}Cb7e@wb0E(nwr~m)} literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.base/icons/paste-disabled.gif b/com.wudsn.ide.base/icons/paste-disabled.gif new file mode 100644 index 0000000000000000000000000000000000000000..1c7668db98ee3b2bc43b84b274b7f813e320f9f6 GIT binary patch literal 361 zcmZ?wbhEHb6krfwxXQrr>(}p}KY#uB@$>ulAL}-5-@1MG+Vz`How>08;L*=tzJ2=q z_2Z{6*RJ2(wsYU9Gw0W=-+b`MiGzoaUcG+v>$mS;zJB}s<?G=iM}PhP{qxtaBS()O zKYsMsv7@I>9yxgeh>pH~`Q*ms6L)T$fBNY5&1>gxT|Is2+_C$&FTa2L^6l&AZ(cpS zd*l40d)IGVI&tpg!8_N_zj*rK*`qtJUpzT<@Zg_6e}4b|{r~@e2E0J=CkrD3gDHa! zNEOIW3~X%&W)*noNM&zY^2TF|rwY%>ODl9N*7Y@Qsx(sA_uzP@)|9#DUYJOBIxP*D zv!h1$cTJM~xg2}#i}4Nmvf}J4CdzfrMg~f~atw?TjFVH0w5MxJF|iBqvM^6oTcjo} wylnYYJw+`oNs&48M5d~2Q4!-3<>chzoT{s%pdiT2&2@x(s@JL0rj88O0615?kN^Mx literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.base/icons/paste-enabled.gif b/com.wudsn.ide.base/icons/paste-enabled.gif new file mode 100644 index 0000000000000000000000000000000000000000..39dd4d9d38aa85f9c4c9ff652c453c555b850980 GIT binary patch literal 605 zcmZ?wbhEHb6krfwc*el+?fb9KUw^evS=T;g?WeE5fBgI%SGG8}Wo>-%w1nbmg-r|E zr>?vI;>V*`Ki+@-_3qQp#_4++rfg}PvSs4Z<4qG+wNG6;effbI%MbL;-q1T^)%sm$ z?!EZ&?Cp=w-+uSZShWts{r>RPk4LY6y!!C-{>$%=UVncIlKnMl_Og2~zfYR8?9;bj zs}ElJ`s3HE`3q+*Sh#S>oE0mkuU|W9_m-}08~Qe^ne^a9|E;|>pN<uNJXZMWWaY;b zm7mTue>&Uz=|bP9i+x|N&is6J)|Z=0zuevM_2I6sPY!)NTKMTi<>xcapD*-%xia(f zm04eJEd6q0>DRj(zCGOa?eU>+PY!*1cH+jmoHHv*FRZD!vZ4Cin#!ZI5)aQxIk&Rx z_@bgyOG_@Vt2#J4=g7SLeKWGxHU_P43ES8iv$;2U&$O(q6VjhtJ9g#Z`v3p`Gf)f^ zf3h$#Fk~|5fJ_3#2?P7ChRmksme#gR2^}4Y_7)L&c`133_VkDdE^#q2ajwLO^p>z7 zP9Y(|WkMW5;fsRSZ`c^p5*rxc8fd(0cVtVbi;KUDG3!xQbCZ@RXTLKhy25M>W+$3s ze0-gpOl4WL)zuk|nxh@P9lQ+rgaz2y_zj!m?d?204U~WWGH8yovGtg+=fQz<b2-JE z54^ch=+fRLpeohDagpKZ0;Axb2PYmTGIO$N1~r^8;OLb!N<P975s-MGVH)!#7A6L3 E08U&i@c;k- literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.base/icons/save-as-disabled.gif b/com.wudsn.ide.base/icons/save-as-disabled.gif new file mode 100644 index 0000000000000000000000000000000000000000..df83a548ec27243d6dadb2b46e68c11d39c4e59f GIT binary patch literal 232 zcmZ?wbhEHb6krfwIKsg2<Hyf$-@bqO^7Yl553k?6d-3x1rmedkJ$e4%(bHRZ?yp$A z{^|3VkDosO^!dyC4<A2%{&M~1&3y+B{P_9v`;Q-I&Yit?`|6>C2mk*4`{&P}pFe;8 z|Nox>6)65>VPs&CV$cBzgY0BrO<kbYm!cV!v1-whsgB_~&NHvFO;lmL#Nxtu`-D=5 zP2Y(Qhl(s0tqU1T4ql0n>Fg+*T@a$%vZ3OS?hM|H=A9p6b$Dh}9DVRlj<2n8UXq2u cJ`N7H0IuT3c<oNL?jALr{!m@B35pEX0NYx3OaK4? literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.base/icons/save-as-enabled.gif b/com.wudsn.ide.base/icons/save-as-enabled.gif new file mode 100644 index 0000000000000000000000000000000000000000..466bfb11281982ab51b763922eb51bc07e6e6960 GIT binary patch literal 583 zcmZ?wbhEHb6krfwc*el+<Hw)owI5=uSI1Saj;&r5SG_7Uf0jdVaboSNq}r7yZ~xkT z`pJ`*A5$AwrZlWLap%{`J3kMce|GS~vtw6Y9KZ77?9ErHjVrQSmTx+Id-s_q2QEB2 zck9*F`){w_f0Na^yr5(0*;}vfK7D`w@{L#TKIeBV-Ffu#lUE;0dzO^;EdKEI*ZWUj zBT6S$O;}t#VbQ(U-`;=yb^pz`4_|-QPhNQM&A0n+zBW%=@ZsCf&tJc_OrQV$+s}@f z^E+nF+kW_1VRKv8?0Fma?q9lj-M_!To}8ci|Nno6AqW(IvM@3*I5X&g+zyHp2KFTl z&P~lNtxZaHcDD9C_O_i$Eq3NMj?<>wnA^2jY1!!;YMNPU+G$z0SSZ^W=ouIoo7gFv zwixl)IqlkQr)Jcmt0>WV#7;s{r$t>>M)sV{*$WyiD$)uHd^h<-rA6gi#6^Vp7@j)u z35&?JhzUyZ@e6P{@Cgd>ws1&t^X<}pE@`Bq5VDA&k%@^-Btj!$;lg$SUP%svlqA<K QetrpwMH3e{IxtuR04B%A3jhEB literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.base/plugin.properties b/com.wudsn.ide.base/plugin.properties new file mode 100644 index 00000000..e4dc0b6f --- /dev/null +++ b/com.wudsn.ide.base/plugin.properties @@ -0,0 +1,58 @@ +com.wudsn.ide.base.editor.CommonOpenFolderCommand.name=Open Folder + +com.wudsn.ide.base.editor.text.TextEditorConvertNumbersMenu.label=Convert +com.wudsn.ide.base.editor.text.TextEditorConvertNumbersMenu.mnemonic=C +com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToDecimalCommand.name=To Decimal Values +com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToDecimalCommand.mnemonic=D +com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToHexaDecimalCommand.name=To Hexadecimal Values +com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToHexaDecimalCommand.mnemonic=H +com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToBinaryCommand.name=To Binary Values +com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToBinaryCommand.mnemonic=B + +com.wudsn.ide.base.editor.text.TextEditorSortMenu.label=Sort +com.wudsn.ide.base.editor.text.TextEditorSortMenu.mnemonic=S +com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveCommand.name=Case Sensitively +com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveCommand.mnemonic=C +com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveWithoutDuplicatesCommand.name=Case Sensitively Without Duplicates +com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveWithoutDuplicatesCommand.mnemonic=D +com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveCommand.name=Case Insensitively +com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveCommand.mnemonic=I +com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveWithoutDuplicatesCommand.name=Case Insensitively Without Duplicates +com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveWithoutDuplicatesCommand.mnemonic=W +com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericCommand.name=Numerically +com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericCommand.mnemonic=N +com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericWithoutDuplicatesCommand.name=Numerically Without Duplicates +com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericWithoutDuplicatesCommand.mnemonic=u +com.wudsn.ide.base.editor.text.TextEditorReverseLinesCommand.name=Reverse +com.wudsn.ide.base.editor.text.TextEditorReverseLinesCommand.mnemonic=R + +com.wudsn.ide.base.editor.BinaryFile.name=Binary File + +com.wudsn.ide.base.editor.hex.HexEditor.name=Hex Editor +com.wudsn.ide.base.editor.hex.HexEditorOpenCommand.name=Open With Hex Editor +com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardCommand.name=Copy +com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsMenu.name=Copy as +com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsDecimalValuesCommand.name=Copy as Decimal Values +com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsDecimalValuesBlockCommand.name=Copy as Decimal Values in Block Format +com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsHexValuesCommand.name=Copy as Hexadecimal Values +com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsASCIIStringCommand.name=Copy as ASCII String +com.wudsn.ide.base.editor.hex.HexEditorPasteFromClipboardCommand.name=Paste +com.wudsn.ide.base.editor.hex.HexEditorSaveSelectionAsCommand.name=Save Selection as... + +com.wudsn.ide.base.editor.hex.HexEditorFileContentMode.BINARY=Binary +com.wudsn.ide.base.editor.hex.HexEditorFileContentMode.ATARI_COM_FILE=Atari COM File +com.wudsn.ide.base.editor.hex.HexEditorFileContentMode.ATARI_DISK_IMAGE=Atari Disk Image +com.wudsn.ide.base.editor.hex.HexEditorFileContentMode.ATARI_DISK_IMAGE_K_FILE=Atari Disk Image (k-File) +com.wudsn.ide.base.editor.hex.HexEditorFileContentMode.ATARI_MADS_FILE=Atari MADS File +com.wudsn.ide.base.editor.hex.HexEditorFileContentMode.ATARI_SDX_FILE=Atari SpartaDOS X File +com.wudsn.ide.base.editor.hex.HexEditorFileContentMode.ATARI_SAP_FILE=Atari SAP File +com.wudsn.ide.base.editor.hex.HexEditorFileContentMode.C64_PRG_FILE=C64 PRG File + +com.wudsn.ide.base.editor.hex.HexEditorCharacterSet.ASCII=ASCII +com.wudsn.ide.base.editor.hex.HexEditorCharacterSet.ATARI_ATASCII=Atari ATASCII +com.wudsn.ide.base.editor.hex.HexEditorCharacterSet.ATARI_ATASCII_SCREEN_CODE=Atari ATASCII Screen Code +com.wudsn.ide.base.editor.hex.HexEditorCharacterSet.ATARI_INTERNATIONAL=Atari International +com.wudsn.ide.base.editor.hex.HexEditorCharacterSet.ATARI_INTERNATIONAL_SCREEN_CODE=Atari International Screen Code +com.wudsn.ide.base.editor.hex.HexEditorCharacterSet.C64_PETSCII_UPPER_CASE=C64 PETSCII Upper Case +com.wudsn.ide.base.editor.hex.HexEditorCharacterSet.C64_PETSCII_LOWER_CASE=C64 PETSCII Lower Case + diff --git a/com.wudsn.ide.base/plugin.xml b/com.wudsn.ide.base/plugin.xml new file mode 100644 index 00000000..2d6cb3e8 --- /dev/null +++ b/com.wudsn.ide.base/plugin.xml @@ -0,0 +1,632 @@ +<?xml version="1.0" encoding="UTF-8"?> +<?eclipse version="3.0"?> +<plugin> + <extension + name="CommonCommands" + point="org.eclipse.ui.commands"> + <command + id="com.wudsn.ide.base.editor.CommonOpenFolderCommand" + name="%com.wudsn.ide.base.editor.CommonOpenFolderCommand.name"> + </command> + </extension> + <extension + name="CommonHandlers" + point="org.eclipse.ui.handlers"> + <handler + class="com.wudsn.ide.base.editor.CommonOpenFolderCommandHandler" + commandId="com.wudsn.ide.base.editor.CommonOpenFolderCommand"> + </handler> + </extension> + <extension + point="org.eclipse.core.expressions.propertyTesters"> + <propertyTester + class="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersCommand$EnabledPropertyTester" + id="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersCommand.EnabledPropertyTester" + namespace="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersCommand" + properties="isEnabled" + type="org.eclipse.jface.viewers.ISelection"> + </propertyTester> + </extension> + <extension + point="org.eclipse.core.expressions.definitions"> + <definition + id="com.wudsn.ide.base.text.TextEditorActive"> + <with + variable="activeEditor"> + <instanceof + value="org.eclipse.ui.texteditor.ITextEditor"> + </instanceof> + </with> + </definition> + + </extension> + <extension + name="TextEditorCommands" + point="org.eclipse.ui.commands"> + <command + categoryId="org.eclipse.ui.category.edit" + id="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToDecimalCommand" + name="%com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToDecimalCommand.name"> + </command> + <command + categoryId="org.eclipse.ui.category.edit" + id="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToHexaDecimalCommand" + name="%com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToHexaDecimalCommand.name"> + </command> + <command + categoryId="org.eclipse.ui.category.edit" + id="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToBinaryCommand" + name="%com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToBinaryCommand.name"> + </command> + <command + categoryId="org.eclipse.ui.category.edit" + id="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveCommand" + name="%com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveCommand.name"> + </command> + <command + categoryId="org.eclipse.ui.category.edit" + id="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveWithoutDuplicatesCommand" + name="%com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveWithoutDuplicatesCommand.name"> + </command> + <command + categoryId="org.eclipse.ui.category.edit" + id="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveCommand" + name="%com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveCommand.name"> + </command> + <command + id="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveWithoutDuplicatesCommand" + name="%com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveWithoutDuplicatesCommand.name"> + </command> + <command + categoryId="org.eclipse.ui.category.edit" + id="com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericCommand" + name="%com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericCommand.name"> + </command> + <command + categoryId="org.eclipse.ui.category.edit" + id="com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericWithoutDuplicatesCommand" + name="%com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericWithoutDuplicatesCommand.name"> + </command> + <command + categoryId="org.eclipse.ui.category.edit" + id="com.wudsn.ide.base.editor.text.TextEditorReverseLinesCommand" + name="%com.wudsn.ide.base.editor.text.TextEditorReverseLinesCommand.name"> + </command> + </extension> + <extension + name="TextEditorHandlers" + point="org.eclipse.ui.handlers"> + <handler + class="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersCommand$Handler" + commandId="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToDecimalCommand"> + <activeWhen> + <and> + <with + variable="selection"> + <test + forcePluginActivation="false" + property="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersCommand.isEnabled" + value="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToDecimalCommand"> + </test> + </with> + </and> + </activeWhen> + </handler> + <handler + class="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersCommand$Handler" + commandId="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToHexaDecimalCommand"> + <activeWhen> + <with + variable="selection"> + <test + forcePluginActivation="false" + property="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersCommand.isEnabled" + value="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToHexaDecimalCommand"> + </test> + </with> + </activeWhen> + </handler> + <handler + class="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersCommand$Handler" + commandId="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToBinaryCommand"> + <activeWhen> + <with + variable="selection"> + <test + forcePluginActivation="false" + property="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersCommand.isEnabled" + value="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToBinaryCommand"> + </test> + </with> + </activeWhen> + </handler> + <handler + class="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveCommandHandler" + commandId="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveCommand"> + <activeWhen> + <reference + definitionId="com.wudsn.ide.base.text.TextEditorActive"> + </reference> + </activeWhen> + </handler> + <handler + class="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveCommandHandler" + commandId="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveWithoutDuplicatesCommand"> + <activeWhen> + <reference + definitionId="com.wudsn.ide.base.text.TextEditorActive"> + </reference> + </activeWhen> + </handler> + <handler + class="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveCommandHandler" + commandId="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveCommand"> + <activeWhen> + <reference + definitionId="com.wudsn.ide.base.text.TextEditorActive"> + </reference> + </activeWhen> + </handler> + <handler + class="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveCommandHandler" + commandId="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveWithoutDuplicatesCommand"> + <activeWhen> + <reference + definitionId="com.wudsn.ide.base.text.TextEditorActive"> + </reference> + </activeWhen> + </handler> + <handler + class="com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericCommandHandler" + commandId="com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericCommand"> + <activeWhen> + <reference + definitionId="com.wudsn.ide.base.text.TextEditorActive"> + </reference> + </activeWhen> + </handler> + <handler + class="com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericCommandHandler" + commandId="com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericWithoutDuplicatesCommand"> + <activeWhen> + <reference + definitionId="com.wudsn.ide.base.text.TextEditorActive"> + </reference> + </activeWhen> + </handler> + <handler + class="com.wudsn.ide.base.editor.text.TextEditorReverseLinesCommandHandler" + commandId="com.wudsn.ide.base.editor.text.TextEditorReverseLinesCommand"> + <activeWhen> + <reference + definitionId="com.wudsn.ide.base.text.TextEditorActive"> + </reference> + </activeWhen> + </handler> + </extension> + <extension + name="TextEditorMenus" + point="org.eclipse.ui.menus"> + <menuContribution + locationURI="popup:#TextEditorContext?after=additions"> + <menu + id="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersMenu" + label="%com.wudsn.ide.base.editor.text.TextEditorConvertNumbersMenu.label"> + <command + commandId="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToDecimalCommand" + id="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToDecimalCommand" + style="push"> + </command> + <command + commandId="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToHexaDecimalCommand" + id="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToHexaDecimalCommand" + style="push"> + </command> + <command + commandId="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToBinaryCommand" + id="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToBinaryCommand" + style="push"> + </command> + <visibleWhen + checkEnabled="false"> + <reference + definitionId="com.wudsn.ide.base.text.TextEditorActive"> + </reference> + </visibleWhen> + </menu> + </menuContribution> + <menuContribution + locationURI="menu:edit?after=additions"> + <menu + label="%com.wudsn.ide.base.editor.text.TextEditorConvertNumbersMenu.label" + mnemonic="%com.wudsn.ide.base.editor.text.TextEditorConvertNumbersMenu.mnemonic"> + <command + commandId="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToDecimalCommand" + id="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToDecimalCommand" + mnemonic="%com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToDecimalCommand.mnemonic" + style="push"> + </command> + <command + commandId="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToHexaDecimalCommand" + id="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToHexaDecimalCommand" + mnemonic="%com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToHexaDecimalCommand.mnemonic" + style="push"> + </command> + <command + commandId="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToBinaryCommand" + id="com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToBinaryCommand" + mnemonic="%com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToBinaryCommand.mnemonic" + style="push"> + </command> + <visibleWhen + checkEnabled="false"> + <reference + definitionId="com.wudsn.ide.base.text.TextEditorActive"> + </reference> + </visibleWhen> + </menu> + </menuContribution> + <menuContribution + locationURI="popup:#TextEditorContext?after=additions"> + <menu + id="com.wudsn.ide.base.editor.text.TextEditorSortMenu" + label="%com.wudsn.ide.base.editor.text.TextEditorSortMenu.label"> + <command + commandId="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveCommand" + id="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveCommand" + style="push"> + </command> + <command + commandId="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveWithoutDuplicatesCommand" + id="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveWithoutDuplicatesCommand" + style="push"> + </command> + <command + commandId="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveCommand" + id="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveCommandHandler" + style="push"> + </command> + <command + commandId="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveWithoutDuplicatesCommand" + id="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveWithoutDuplicates" + style="push"> + </command> + <command + commandId="com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericCommand" + id="com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericCommand" + style="push"> + </command> + <command + commandId="com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericWithoutDuplicatesCommand" + id="com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericWithoutDuplicatesCommand" + style="push"> + </command> + <command + commandId="com.wudsn.ide.base.editor.text.TextEditorReverseLinesCommand" + id="com.wudsn.ide.base.editor.text.TextEditorReverseLinesCommand" + style="push"> + </command> + <visibleWhen + checkEnabled="false"> + <with + variable="activeEditor"> + <instanceof + value="org.eclipse.ui.texteditor.ITextEditor"> + </instanceof> + </with> + </visibleWhen> + </menu> + </menuContribution> + <menuContribution + locationURI="menu:edit?after=additions"> + <menu + label="%com.wudsn.ide.base.editor.text.TextEditorSortMenu.label" + mnemonic="%com.wudsn.ide.base.editor.text.TextEditorSortMenu.mnemonic"> + <command + commandId="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveCommand" + id="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveCommand" + mnemonic="%com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveCommand.mnemonic" + style="push"> + </command> + <command + commandId="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveWithoutDuplicatesCommand" + id="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveWithoutDuplicatesCommand" + mnemonic="%com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveWithoutDuplicatesCommand.mnemonic" + style="push"> + </command> + <command + commandId="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveCommand" + id="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveCommand" + mnemonic="%com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveCommand.mnemonic" + style="push"> + </command> + <command + commandId="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveWithoutDuplicatesCommand" + id="com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveWithoutDuplicatesCommand" + mnemonic="%com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveWithoutDuplicatesCommand.mnemonic" + style="push"> + </command> + <command + commandId="com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericCommand" + id="com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericCommand" + mnemonic="%com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericCommand.mnemonic" + style="push"> + </command> + <command + commandId="com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericWithoutDuplicatesCommand" + id="com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericWithoutDuplicatesCommand" + mnemonic="%com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericWithoutDuplicatesCommand.mnemonic" + style="push"> + </command> + <command + commandId="com.wudsn.ide.base.editor.text.TextEditorReverseLinesCommand" + id="com.wudsn.ide.base.editor.text.TextEditorReverseLinesCommand" + mnemonic="%com.wudsn.ide.base.editor.text.TextEditorReverseLinesCommand.mnemonic" + style="push"> + </command> + <visibleWhen + checkEnabled="false"> + <with + variable="activeEditor"> + <instanceof + value="org.eclipse.ui.texteditor.ITextEditor"> + </instanceof> + </with> + </visibleWhen> + </menu> + </menuContribution> + </extension> + <extension + name="HexEditorCommands" + point="org.eclipse.ui.commands"> + <command + id="com.wudsn.ide.base.editor.hex.HexEditorOpenCommand" + name="%com.wudsn.ide.base.editor.hex.HexEditorOpenCommand.name"> + </command> + <command + id="com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardCommand" + name="%com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardCommand.name"> + </command> + <command + id="com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsHexValuesCommand" + name="%com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsHexValuesCommand.name"> + </command> + <command + id="com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsDecimalValuesCommand" + name="%com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsDecimalValuesCommand.name"> + </command> + <command + id="com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsDecimalValuesBlockCommand" + name="%com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsDecimalValuesBlockCommand.name"> + </command> + <command + id="com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsASCIIStringCommand" + name="%com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsASCIIStringCommand.name"> + </command> + <command + id="com.wudsn.ide.base.editor.hex.HexEditorPasteFromClipboardCommand" + name="%com.wudsn.ide.base.editor.hex.HexEditorPasteFromClipboardCommand.name"> + </command> + <command + id="com.wudsn.ide.base.editor.hex.HexEditorSaveSelectionAsCommand" + name="%com.wudsn.ide.base.editor.hex.HexEditorSaveSelectionAsCommand.name"> + </command> + </extension> + <extension + name="HexEditorHandlers" + point="org.eclipse.ui.handlers"> + <handler + class="com.wudsn.ide.base.editor.hex.HexEditorOpenCommandHandler" + commandId="com.wudsn.ide.base.editor.hex.HexEditorOpenCommand"> + </handler> + <handler + class="com.wudsn.ide.base.editor.hex.HexEditorClipboardCommandHandler" + commandId="com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardCommand"> + <activeWhen> + <with + variable="activeMenuSelection"> + <instanceof + value="com.wudsn.ide.base.editor.hex.HexEditorSelection"> + </instanceof> + </with> + </activeWhen> + </handler> + <handler + class="com.wudsn.ide.base.editor.hex.HexEditorClipboardCommandHandler" + commandId="com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsHexValuesCommand"> + <activeWhen> + <with + variable="activeMenuSelection"> + <instanceof + value="com.wudsn.ide.base.editor.hex.HexEditorSelection"> + </instanceof> + </with> + </activeWhen> + </handler> + <handler + class="com.wudsn.ide.base.editor.hex.HexEditorClipboardCommandHandler" + commandId="com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsDecimalValuesCommand"> + <activeWhen> + <with + variable="activeMenuSelection"> + <instanceof + value="com.wudsn.ide.base.editor.hex.HexEditorSelection"> + </instanceof> + </with> + </activeWhen> + </handler> + <handler + class="com.wudsn.ide.base.editor.hex.HexEditorClipboardCommandHandler" + commandId="com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsDecimalValuesBlockCommand"> + <activeWhen> + <with + variable="activeMenuSelection"> + <instanceof + value="com.wudsn.ide.base.editor.hex.HexEditorSelection"> + </instanceof> + </with> + </activeWhen> + </handler> + <handler + class="com.wudsn.ide.base.editor.hex.HexEditorClipboardCommandHandler" + commandId="com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsASCIIStringCommand"> + <activeWhen> + <with + variable="activeMenuSelection"> + <instanceof + value="com.wudsn.ide.base.editor.hex.HexEditorSelection"> + </instanceof> + </with> + </activeWhen> + </handler> + <handler + class="com.wudsn.ide.base.editor.hex.HexEditorClipboardCommandHandler" + commandId="com.wudsn.ide.base.editor.hex.HexEditorPasteFromClipboardCommand"> + </handler> + <handler + class="com.wudsn.ide.base.editor.hex.HexEditorSaveSelectionAsCommandHandler" + commandId="com.wudsn.ide.base.editor.hex.HexEditorSaveSelectionAsCommand"> + <activeWhen> + <with + variable="activeMenuSelection"> + <instanceof + value="com.wudsn.ide.base.editor.hex.HexEditorSelection"> + </instanceof> + </with> + </activeWhen> + </handler> + </extension> + <extension + name="HexEditorMenus" + point="org.eclipse.ui.menus"> + <menuContribution + locationURI="popup:org.eclipse.ui.popup.any?after=group.open"> + <command + commandId="com.wudsn.ide.base.editor.CommonOpenFolderCommand" + id="com.wudsn.ide.base.editor.CommonOpenFolderCommand" + style="push"> + <visibleWhen + checkEnabled="false"> + <or> + <with + variable="activeMenuSelection"> + <and> + <iterate + ifEmpty="false" + operator="and"> + </iterate> + <instanceof + value="org.eclipse.jface.viewers.IStructuredSelection"> + </instanceof> + </and> + </with> + <with + variable="activeMenuEditorInput"> + <and> + <iterate + ifEmpty="false" + operator="and"> + </iterate> + <instanceof + value="org.eclipse.jface.viewers.IStructuredSelection"> + </instanceof> + </and> + </with> + </or> + </visibleWhen> + </command> + </menuContribution> + <menuContribution + locationURI="popup:org.eclipse.ui.popup.any?before=group.openWith"> + <command + commandId="com.wudsn.ide.base.editor.hex.HexEditorOpenCommand" + id="com.wudsn.ide.base.editor.hex.HexEditorOpenCommand" + style="push"> + <visibleWhen + checkEnabled="false"> + <or> + <with + variable="activeMenuSelection"> + <iterate + ifEmpty="false" + operator="or"> + <instanceof + value="org.eclipse.core.resources.IFile"> + </instanceof> + </iterate> + </with> + <with + variable="activeMenuEditorInput"> + <iterate + ifEmpty="false" + operator="or"> + <instanceof + value="org.eclipse.ui.IFileEditorInput"> + </instanceof> + </iterate> + </with> + </or> + </visibleWhen> + </command> + </menuContribution> + <menuContribution + locationURI="popup:#HexEditorContext"> + <command + commandId="com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardCommand" + disabledIcon="icons/copy-disabled.gif" + icon="icons/copy-enabled.gif" + id="com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardCommand" + style="push"> + </command> + <menu + label="%com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsMenu.name"> + <command + commandId="com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsHexValuesCommand" + id="com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsHexValuesCommand" + style="push"> + </command> + <command + commandId="com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsDecimalValuesCommand" + id="com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsDecimalValuesCommand" + style="push"> + </command> + <command + commandId="com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsDecimalValuesBlockCommand" + id="com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsDecimalValuesBlockCommand" + style="push"> + </command> + <command + commandId="com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsASCIIStringCommand" + id="com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsASCIIStringCommand" + style="push"> + </command> + </menu> + <command + commandId="com.wudsn.ide.base.editor.hex.HexEditorPasteFromClipboardCommand" + icon="icons/paste-enabled.gif" + id="%com.wudsn.ide.base.editor.hex.HexEditorPasteFromClipboardCommand.name" + label="%com.wudsn.ide.base.editor.hex.HexEditorPasteFromClipboardCommand.name" + style="push"> + </command> + <command + commandId="com.wudsn.ide.base.editor.hex.HexEditorSaveSelectionAsCommand" + disabledIcon="icons/save-as-disabled.gif" + icon="icons/save-as-enabled.gif" + label="%com.wudsn.ide.base.editor.hex.HexEditorSaveSelectionAsCommand.name" + style="push"> + </command> + </menuContribution> + </extension> + + <extension + name="HexEditor" + point="org.eclipse.ui.editors"> + <editor + class="com.wudsn.ide.base.editor.hex.HexEditor" + default="true" + icon="icons/hex-editor-16x16.gif" + id="com.wudsn.ide.base.editor.hex.HexEditor" + name="%com.wudsn.ide.base.editor.hex.HexEditor.name"> + </editor> + </extension> + </plugin> diff --git a/com.wudsn.ide.base/plugin_de_DE.properties b/com.wudsn.ide.base/plugin_de_DE.properties new file mode 100644 index 00000000..8e0503b7 --- /dev/null +++ b/com.wudsn.ide.base/plugin_de_DE.properties @@ -0,0 +1,58 @@ +com.wudsn.ide.base.editor.CommonOpenFolderCommand.name=Ordner öffnen + +com.wudsn.ide.base.editor.text.TextEditorConvertNumbersMenu.label=Konvertieren +com.wudsn.ide.base.editor.text.TextEditorConvertNumbersMenu.mnemonic=K +com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToDecimalCommand.name=In dezimale Werte +com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToDecimalCommand.mnemonic=d +com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToHexaDecimalCommand.name=In hexadezimale Werte +com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToHexaDecimalCommand.mnemonic=h +com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToBinaryCommand.name=In binäre Werte +com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToBinaryCommand.mnemonic=b + +com.wudsn.ide.base.editor.text.TextEditorSortMenu.label=Sortieren +com.wudsn.ide.base.editor.text.TextEditorSortMenu.mnemonic=S +com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveCommand.name=Mit Groß-/Kleinschreibung +com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveCommand.mnemonic=M +com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveWithoutDuplicatesCommand.name=Mit Groß-/Kleinschreibung ohne Duplikate +com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseSensitiveWithoutDuplicatesCommand.mnemonic=D +com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveCommand.name=Ohne Groß-/Kleinschreibung +com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveCommand.mnemonic=O +com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveWithoutDuplicatesCommand.name=Ohne Groß-/Kleinschreibung ohne Duplikate +com.wudsn.ide.base.editor.text.TextEditorSortLinesCaseInsensitiveWithoutDuplicatesCommand.mnemonic=u +com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericCommand.name=Numerisch +com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericCommand.mnemonic=N +com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericWithoutDuplicatesCommand.name=Numerisch ohne Duplikate +com.wudsn.ide.base.editor.text.TextEditorSortLinesNumericWithoutDuplicatesCommand.mnemonic=m +com.wudsn.ide.base.editor.text.TextEditorReverseLinesCommand.name=Umkehren +com.wudsn.ide.base.editor.text.TextEditorReverseLinesCommand.mnemonic=U + +com.wudsn.ide.base.editor.BinaryFile.name=Binär-Datei + +com.wudsn.ide.base.editor.hex.HexEditor.name=Hex Editor +com.wudsn.ide.base.editor.hex.HexEditorOpenCommand.name=Öffnen mit Hex Editor +com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardCommand.name=Kopieren +com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsMenu.name=Kopieren als +com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsDecimalValuesCommand.name=Kopieren als dezimale Werte +com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsDecimalValuesBlockCommand.name=Kopieren als dezimale Werte im Blocksatz +com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsHexValuesCommand.name=Kopieren als hexadezimale Werte +com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsASCIIStringCommand.name=Kopieren als ASCII Text +com.wudsn.ide.base.editor.hex.HexEditorPasteFromClipboardCommand.name=Einfügen +com.wudsn.ide.base.editor.hex.HexEditorSaveSelectionAsCommand.name=Auswahl Speichern unter... + +com.wudsn.ide.base.editor.hex.HexEditorFileContentMode.BINARY=Binär +com.wudsn.ide.base.editor.hex.HexEditorFileContentMode.ATARI_COM_FILE=Atari COM-Datei +com.wudsn.ide.base.editor.hex.HexEditorFileContentMode.ATARI_DISK_IMAGE=Atari Disk Image +com.wudsn.ide.base.editor.hex.HexEditorFileContentMode.ATARI_DISK_IMAGE_K_FILE=Atari Disk Image (k-Datei) +com.wudsn.ide.base.editor.hex.HexEditorFileContentMode.ATARI_MADS_FILE=Atari MADS-Datei +com.wudsn.ide.base.editor.hex.HexEditorFileContentMode.ATARI_SAP_FILE=Atari SAP-Datei +com.wudsn.ide.base.editor.hex.HexEditorFileContentMode.ATARI_SDX_FILE=Atari SpartaDOS X-Datei +com.wudsn.ide.base.editor.hex.HexEditorFileContentMode.C64_PRG_FILE=C64 PRG-Datei + +com.wudsn.ide.base.editor.hex.HexEditorCharacterSet.ASCII=ASCII +com.wudsn.ide.base.editor.hex.HexEditorCharacterSet.ATARI_ATASCII=Atari ATASCII +com.wudsn.ide.base.editor.hex.HexEditorCharacterSet.ATARI_ATASCII_SCREEN_CODE=Atari ATASCII Bildschirmcode +com.wudsn.ide.base.editor.hex.HexEditorCharacterSet.ATARI_INTERNATIONAL=Atari International +com.wudsn.ide.base.editor.hex.HexEditorCharacterSet.ATARI_INTERNATIONAL_SCREEN_CODE=Atari International Bildschirmcode +com.wudsn.ide.base.editor.hex.HexEditorCharacterSet.C64_PETSCII_UPPER_CASE=C64 PETSCII Großbuchstaben +com.wudsn.ide.base.editor.hex.HexEditorCharacterSet.C64_PETSCII_LOWER_CASE=C64 PETSCII Kleinbuchstaben + diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/BasePlugin.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/BasePlugin.java new file mode 100644 index 00000000..cbbecb92 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/BasePlugin.java @@ -0,0 +1,87 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base; + +import org.osgi.framework.BundleContext; + +import com.wudsn.ide.base.common.AbstractIDEPlugin; + +/** + * The activator class controls the plug-in life cycle + */ +public final class BasePlugin extends AbstractIDEPlugin { + + /** + * The plugin id. + */ + public static final String ID = "com.wudsn.ide.base"; + + /** + * The shared instance. + */ + private static BasePlugin plugin; + + /** + * Creates a new instance. Must be public for dynamic instantiation. + */ + public BasePlugin() { + } + + /** + * {@inheritDoc} + */ + @Override + protected String getPluginId() { + return ID; + } + + /** + * {@inheritDoc} + */ + @Override + public void start(BundleContext context) throws Exception { + if (context != null) { + super.start(context); + } + plugin = this; + } + + /** + * {@inheritDoc} + */ + @Override + public void stop(BundleContext context) throws Exception { + plugin = null; + super.stop(context); + } + + /** + * Gets the shared plugin instance + * + * @return The plug-in, not <code>null</code>. + */ + public static BasePlugin getInstance() { + if (plugin == null) { + throw new IllegalStateException("Plugin not initialized or already stopped"); + } + return plugin; + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/Texts.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/Texts.java new file mode 100644 index 00000000..32df3a54 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/Texts.java @@ -0,0 +1,116 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base; + +import org.eclipse.osgi.util.NLS; + +import com.wudsn.ide.base.common.FileUtility; +import com.wudsn.ide.base.editor.hex.HexEditor; + +/** + * Class which holds the localized text constants. + * + * @author Peter Dell + */ +public final class Texts extends NLS { + + /** + * Common texts. + */ + public static String DIALOG_TITLE; + public static String FILE_PATH_FIELD_BROWSE_BUTTON_LABEL; + public static String FILE_PATH_FIELD_DIALOG_MESSAGE; + + /** + * Hex editor + */ + public static String HEX_EDITOR_FILE_SIZE; + + public static String HEX_EDITOR_ATARI_COM_BLOCK_HEADER; + public static String HEX_EDITOR_ATARI_COM_BLOCK_HEADER_PARAMETERS; + public static String HEX_EDITOR_ATARI_COM_BLOCK_ERROR; + + public static String HEX_EDITOR_ATARI_DISK_IMAGE_HEADER; + public static String HEX_EDITOR_ATARI_SECTOR_HEADER; + public static String HEX_EDITOR_ATARI_SECTOR_HEADER_PARAMETERS; + public static String HEX_EDITOR_ATARI_SECTOR_ERROR; + + public static String HEX_EDITOR_ATARI_MADS_RELOC_BLOCK_HEADER; + public static String HEX_EDITOR_ATARI_MADS_UPDATE_RELOC_BLOCK_HEADER; + public static String HEX_EDITOR_ATARI_MADS_UPDATE_SYMBOLS_BLOCK_HEADER; + public static String HEX_EDITOR_ATARI_MADS_DEFINE_SYMBOLS_BLOCK_HEADER; + public static String HEX_EDITOR_ATARI_MADS_DEFINE_SYMBOL_HEADER; + public static String HEX_EDITOR_ATARI_MADS_BLOCK_ERROR; + + public static String HEX_EDITOR_ATARI_SAP_FILE_HEADER; + + public static String HEX_EDITOR_ATARI_SDX_NON_RELOC_BLOCK_HEADER; + public static String HEX_EDITOR_ATARI_SDX_NON_RELOC_BLOCK_HEADER_PARAMETERS; + public static String HEX_EDITOR_ATARI_SDX_RELOC_BLOCK_HEADER; + public static String HEX_EDITOR_ATARI_SDX_UPDATE_RELOC_BLOCK_HEADER; + public static String HEX_EDITOR_ATARI_SDX_UPDATE_SYMBOLS_BLOCK_HEADER; + public static String HEX_EDITOR_ATARI_SDX_DEFINE_SYMBOLS_BLOCK_HEADER; + public static String HEX_EDITOR_ATARI_SDX_BLOCK_ERROR; + + public static String HEX_EDITOR_C64_PRG_HEADER; + public static String HEX_EDITOR_C64_PRG_HEADER_PARAMETERS; + public static String HEX_EDITOR_C64_PRG_ERROR; + + public static String HEX_EDITOR_FILE_CONTENT_SIZE_FIELD_LABEL; + public static String HEX_EDITOR_FILE_CONTENT_SIZE_FIELD_TEXT; + public static String HEX_EDITOR_FILE_CONTENT_MODE_FIELD_LABEL; + public static String HEX_EDITOR_CHARACTER_SET_TYPE_FIELD_LABEL; + public static String HEX_EDITOR_BYTES_PER_ROW_FIELD_LABEL; + + public static String HEX_EDITOR_SAVE_SELECTION_AS_DIALOG_TITLE; + + /** + * Messages for {@link FileUtility}. + */ + public static String MESSAGE_E200; + public static String MESSAGE_E201; + public static String MESSAGE_E202; + public static String MESSAGE_E203; + public static String MESSAGE_E204; + public static String MESSAGE_E205; + public static String MESSAGE_E206; + public static String MESSAGE_E207; + public static String MESSAGE_E208; + public static String MESSAGE_E209; + public static String MESSAGE_E210; + public static String MESSAGE_E211; + public static String MESSAGE_E212; + public static String MESSAGE_E213; + + /** + * Message for the {@link HexEditor} + */ + public static String MESSAGE_E300; + public static String MESSAGE_E301; + public static String MESSAGE_I302; + public static String MESSAGE_I303; + + /** + * Initializes the constants. + */ + static { + NLS.initializeMessages(Texts.class.getName(), Texts.class); + } +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/Texts.properties b/com.wudsn.ide.base/src/com/wudsn/ide/base/Texts.properties new file mode 100644 index 00000000..3e48edc4 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/Texts.properties @@ -0,0 +1,63 @@ +DIALOG_TITLE=WUDSN IDE Dialog +FILE_PATH_FIELD_BROWSE_BUTTON_LABEL=Browse... +FILE_PATH_FIELD_DIALOG_MESSAGE=Select {0} + +HEX_EDITOR_FILE_SIZE=File contains ${0} ({1}) bytes. + +HEX_EDITOR_ATARI_COM_BLOCK_HEADER=COM block +HEX_EDITOR_ATARI_COM_BLOCK_HEADER_PARAMETERS={0}-{1} ({2}) +HEX_EDITOR_ATARI_COM_BLOCK_ERROR=COM file structure error + +HEX_EDITOR_ATARI_DISK_IMAGE_HEADER=ATR header +HEX_EDITOR_ATARI_SECTOR_HEADER=Sector +HEX_EDITOR_ATARI_SECTOR_HEADER_PARAMETERS=({2}) +HEX_EDITOR_ATARI_SECTOR_ERROR=Sector structure error + +HEX_EDITOR_ATARI_MADS_RELOC_BLOCK_HEADER=MADS RELOC block {0}-{1} {2} +HEX_EDITOR_ATARI_MADS_UPDATE_RELOC_BLOCK_HEADER=MADS UPDATE RELOC block {0} {1} +HEX_EDITOR_ATARI_MADS_UPDATE_SYMBOLS_BLOCK_HEADER=MADS UPDATE SYMBOLS block {0} {1} +HEX_EDITOR_ATARI_MADS_DEFINE_SYMBOLS_BLOCK_HEADER=MADS DEFINE SYMBOLS block {0} +HEX_EDITOR_ATARI_MADS_DEFINE_SYMBOL_HEADER=MADS DEFINE SYMBOL {0} {1} {2} {3} +HEX_EDITOR_ATARI_MADS_BLOCK_ERROR=MADS file structure error + +HEX_EDITOR_ATARI_SAP_FILE_HEADER=SAP Header + +HEX_EDITOR_ATARI_SDX_NON_RELOC_BLOCK_HEADER=SDX NON-RELOC block +HEX_EDITOR_ATARI_SDX_NON_RELOC_BLOCK_HEADER_PARAMETERS={0}-{1} ({2}) +HEX_EDITOR_ATARI_SDX_RELOC_BLOCK_HEADER=SDX RELOC block {0} {1} {2} {3} +HEX_EDITOR_ATARI_SDX_UPDATE_RELOC_BLOCK_HEADER=SDX UPDATE RELOC block {0} {1} +HEX_EDITOR_ATARI_SDX_UPDATE_SYMBOLS_BLOCK_HEADER=SDX UPDATE SYMBOLS block {0} {1} +HEX_EDITOR_ATARI_SDX_DEFINE_SYMBOLS_BLOCK_HEADER=SDX DEFINE SYMBOLS block {0} {1} {2} +HEX_EDITOR_ATARI_SDX_BLOCK_ERROR=SDX file structure error + +HEX_EDITOR_C64_PRG_HEADER=Program +HEX_EDITOR_C64_PRG_HEADER_PARAMETERS={0}-{1} ({2}) +HEX_EDITOR_C64_PRG_ERROR=PRG file structure error + +HEX_EDITOR_FILE_CONTENT_SIZE_FIELD_LABEL=File Size +HEX_EDITOR_FILE_CONTENT_SIZE_FIELD_TEXT=${0} ({1}) bytes +HEX_EDITOR_FILE_CONTENT_MODE_FIELD_LABEL=File Mode +HEX_EDITOR_CHARACTER_SET_TYPE_FIELD_LABEL=Character Set +HEX_EDITOR_BYTES_PER_ROW_FIELD_LABEL=Bytes per Row + +HEX_EDITOR_SAVE_SELECTION_AS_DIALOG_TITLE=Save ${0} ({1}) bytes as... + +MESSAGE_E200=Folder '{0}' does not exist. +MESSAGE_E201='{0}' is no folder but a file. +MESSAGE_E202=Cannot create folder '{0}'. +MESSAGE_E203=File '{0}' does not exist. +MESSAGE_E204='{0}' is no file but a folder. +MESSAGE_E205=Cannot open file '{0}' for reading. {1} +MESSAGE_E206=Cannot read content of file '{0}'. +MESSAGE_E207=Content of file '{0}' exceeds the specified maximum size of {1} bytes. +MESSAGE_E208=Content of file '{0}' exceeds the specified maximum size of {1} characters. +MESSAGE_E209=Cannot close input stream of file '{0}'. +MESSAGE_E210=Cannot create file '{0}'. +MESSAGE_E211=Cannot open file '{0}' for writing. +MESSAGE_E212=Cannot write content of file '{0}'. +MESSAGE_E213=Cannot close output stream of file '{0}'. + +MESSAGE_E300=File content cannot be interpreted as '{0}'. +MESSAGE_E301=File of type '{0}' is corrupted; check the last section of the file. +MESSAGE_I302=${0} ({1}) bytes copied to clipboard. +MESSAGE_I303=${0} ({1}) bytes saved as '{2}'. \ No newline at end of file diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/Texts_de_DE.properties b/com.wudsn.ide.base/src/com/wudsn/ide/base/Texts_de_DE.properties new file mode 100644 index 00000000..9362106d --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/Texts_de_DE.properties @@ -0,0 +1,63 @@ +DIALOG_TITLE=WUDSN IDE Dialog +FILE_PATH_FIELD_BROWSE_BUTTON_LABEL=Durchsuchen... +FILE_PATH_FIELD_DIALOG_MESSAGE={0} auswählen + +HEX_EDITOR_FILE_SIZE=Datei enhält {0} ({1}) Bytes. + +HEX_EDITOR_ATARI_COM_BLOCK_HEADER=COM Block +HEX_EDITOR_ATARI_COM_BLOCK_HEADER_PARAMETERS={0}-{1} ({2}) +HEX_EDITOR_ATARI_COM_BLOCK_ERROR=COM Dateistruktur defekt + +HEX_EDITOR_ATARI_DISK_IMAGE_HEADER=ATR Header +HEX_EDITOR_ATARI_SECTOR_HEADER=Sektor +HEX_EDITOR_ATARI_SECTOR_HEADER_PARAMETERS=({2}) +HEX_EDITOR_ATARI_SECTOR_ERROR=Sektorfehler + +HEX_EDITOR_ATARI_MADS_RELOC_BLOCK_HEADER=MADS RELOC Block {0}-{1} {2} +HEX_EDITOR_ATARI_MADS_UPDATE_RELOC_BLOCK_HEADER=MADS UPDATE RELOC Block {0} {1} +HEX_EDITOR_ATARI_MADS_UPDATE_SYMBOLS_BLOCK_HEADER=MADS UPDATE SYMBOLS Block {0} {1} +HEX_EDITOR_ATARI_MADS_DEFINE_SYMBOLS_BLOCK_HEADER=MADS DEFINE SYMBOLS Block {0} +HEX_EDITOR_ATARI_MADS_DEFINE_SYMBOL_HEADER=MADS DEFINE SYMBOL {0} {1} {2} {3} +HEX_EDITOR_ATARI_MADS_BLOCK_ERROR=MADS Dateistruktur defekt + +HEX_EDITOR_ATARI_SAP_FILE_HEADER=SAP Header + +HEX_EDITOR_ATARI_SDX_NON_RELOC_BLOCK_HEADER=SDX NON-RELOC block +HEX_EDITOR_ATARI_SDX_NON_RELOC_BLOCK_HEADER_PARAMETERS={0}-{1} ({2}) +HEX_EDITOR_ATARI_SDX_RELOC_BLOCK_HEADER=SDX NON-RELOC Block {0} {1} {2} {3} +HEX_EDITOR_ATARI_SDX_UPDATE_RELOC_BLOCK_HEADER=SDX UPDATE RELOC Block {0} {1} +HEX_EDITOR_ATARI_SDX_UPDATE_SYMBOLS_BLOCK_HEADER=SDX UPDATE SYMBOLS Block {0} {1} +HEX_EDITOR_ATARI_SDX_DEFINE_SYMBOLS_BLOCK_HEADER=SDX DEFINE SYMBOLS Block {0} {1} {2} +HEX_EDITOR_ATARI_SDX_BLOCK_ERROR=SDX Dateistruktur defekt + +HEX_EDITOR_C64_PRG_HEADER=Programm +HEX_EDITOR_C64_PRG_HEADER_PARAMETERS={0}-{1} ({2}) +HEX_EDITOR_C64_PRG_ERROR=PRG Dateistruktur defekt + +HEX_EDITOR_FILE_CONTENT_SIZE_FIELD_LABEL=Dateigröße +HEX_EDITOR_FILE_CONTENT_SIZE_FIELD_TEXT=${0} ({1}) Bytes +HEX_EDITOR_FILE_CONTENT_MODE_FIELD_LABEL=Datei-Modus +HEX_EDITOR_CHARACTER_SET_TYPE_FIELD_LABEL=Zeichensatz +HEX_EDITOR_BYTES_PER_ROW_FIELD_LABEL=Bytes pro Zeile + +HEX_EDITOR_SAVE_SELECTION_AS_DIALOG_TITLE=Speichere ${0} ({1}) Bytes unter... + +MESSAGE_E200=Ordner '{0}' existiert nicht. +MESSAGE_E201='{0}' ist kein Ordner sondern eine Datei. +MESSAGE_E202=Ordner '{0}' kann nicht erstellt werden. +MESSAGE_E203=Datei '{0}' existiert nicht. +MESSAGE_E204='{0}' ist keine Datei sondern ein Ordner. +MESSAGE_E205=Datei '{0}' kann nicht zum Lesen geöffnet werden. +MESSAGE_E206=Inhalt der Datei '{0}' kann nicht gelesen werden. +MESSAGE_E207=Inhalt der Datei '{0}' überschreitet die angegebene Maximalgröße von {1} Bytes. +MESSAGE_E208=Inhalt der Datei '{0}' überschreitet die angegebene Maximalgröße von {1} Zeichen. +MESSAGE_E209=Eingabendatenstrom der Datei '{0}' kann geschlossen werden. +MESSAGE_E210=Datei '{0}' kann nicht erstellt werden. +MESSAGE_E211=Datei '{0}' kann nicht zum Schreiben geöffnet werden. +MESSAGE_E212=Inhalt der Datei '{0}' kann nicht geschrieben werden. +MESSAGE_E213=Ausgabedatenstrom der Datei '{0}' kann geschlossen werden. + +MESSAGE_E300=Inhalt der Datei kann nicht als '{0}' interpretiert werden. +MESSAGE_E301=Datei vom Typ '{0}' ist korrupt; überprüfen Sie den letzten Abschnitt der Datei +MESSAGE_I302=${0} ({1}) Bytes in die Zwischenablage kopiert. +MESSAGE_I303=${0} ({1}) Bytes gespeichert unter '{2}'. \ No newline at end of file diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/common/AbstractIDEPlugin.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/AbstractIDEPlugin.java new file mode 100644 index 00000000..c8f7eea5 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/AbstractIDEPlugin.java @@ -0,0 +1,293 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.common; + +import java.net.URL; +import java.util.Enumeration; +import java.util.HashMap; +import java.util.Map; + +import org.eclipse.core.runtime.FileLocator; +import org.eclipse.core.runtime.IPath; +import org.eclipse.core.runtime.IStatus; +import org.eclipse.core.runtime.Path; +import org.eclipse.core.runtime.Status; +import org.eclipse.core.runtime.preferences.InstanceScope; +import org.eclipse.jface.dialogs.ErrorDialog; +import org.eclipse.jface.resource.ImageDescriptor; +import org.eclipse.swt.graphics.Image; +import org.eclipse.swt.widgets.Shell; +import org.eclipse.ui.plugin.AbstractUIPlugin; +import org.osgi.framework.BundleContext; +import org.osgi.service.prefs.BackingStoreException; + +import com.wudsn.ide.base.Texts; + +/** + * The plugin base class for IDE plugins. + * + * @author Peter Dell + */ +public abstract class AbstractIDEPlugin extends AbstractUIPlugin { + + private Map<String, Image> images; + + /** + * Creates a new instance. Must be public for dynamic instantiation. + */ + protected AbstractIDEPlugin() { + images = new HashMap<String, Image>(); + + } + + /** + * Gets the plugin id. + * + * @return The plugin id, not empty and not <code>null</code>. + */ + protected abstract String getPluginId(); + + /** + * {@inheritDoc} + */ + @Override + public void start(BundleContext context) throws Exception { + super.start(context); + } + + /** + * {@inheritDoc} + */ + @Override + public void stop(BundleContext context) throws Exception { + super.stop(context); + } + + /** + * Logs an info message to the plugin log and the standard output stream. + * + * @param message + * The message, not <code>null</code>. + * @param parameters + * The message parameters, may be empty or <code>null</code>. + */ + public final void log(String message, Object[] parameters) { + if (message == null) { + throw new IllegalArgumentException( + "Parameter 'message' must not be null."); + } + message = format(message, parameters); + getLog().log( + new Status(IStatus.INFO, getPluginId(), IStatus.OK, message, + null)); +// System.out.println(message); + } + + /** + * Logs an error message and an exception to the plugin log and the standard + * error stream. + * + * @param message + * The message, not <code>null</code>. + * @param parameters + * The message parameters, may be empty or <code>null</code>. + * @param th + * The throwable or <code>null</code>. + */ + public final void logError(String message, Object[] parameters, Throwable th) { + if (message == null) { + throw new IllegalArgumentException( + "Parameter 'message' must not be null."); + } + + message = format(message, parameters); + if (th != null) { + message = message + "\n" + th.getMessage(); + } + getLog().log( + new Status(IStatus.ERROR, getPluginId(), IStatus.ERROR, + message, th)); + // System.err.println(message); + // if (th != null) { + // th.printStackTrace(System.err); + // } + // System.err.flush(); + + } + + private String format(String message, Object... parameters) { + if (parameters == null) { + parameters = new String[0]; + } + String[] stringParameters = new String[parameters.length]; + for (int i = 0; i < parameters.length; i++) { + Object parameter = parameters[i]; + String stringParameter; + if (parameter == null) { + stringParameter = "null"; + } else { + stringParameter = parameter.toString(); + } + stringParameters[i] = stringParameter; + } + message = TextUtility.format(message, stringParameters); + return message; + } + + /** + * Shows error message and an exception as error dialog and logs them to the + * plugin log and the standard error stream. + * + * @param shell + * The shell, not <code>null</code>. + * + * @param message + * The message, not <code>null</code>. + * @param th + * The throwable, not <code>null</code>. + */ + public final void showError(Shell shell, String message, Throwable th) { + if (shell == null) { + throw new IllegalArgumentException( + "Parameter 'shell' must not be null."); + } + if (message == null) { + throw new IllegalArgumentException( + "Parameter 'message' must not be null."); + } + if (th == null) { + throw new IllegalArgumentException( + "Parameter 'th' must not be null."); + } + + Status status = new Status(IStatus.ERROR, getPluginId(), message); + + ErrorDialog + .openError( + shell, + Texts.DIALOG_TITLE, + th.getClass().getName() + + ": " + + th.getMessage() + + "\nCheck the .log file in the .metadata folder of the workspace for details.", + status); + logError(message, null, th); + } + + public final void savePreferences() { + String pluginId; + pluginId = getPluginId(); + try { + InstanceScope.INSTANCE.getNode(pluginId).flush(); + } catch (BackingStoreException ex) { + + throw new RuntimeException("Cannot store preferences for plugin '" + + pluginId + "'", ex); + } + } + + /** + * Gets the image for the specified plug-in relative path. + * + * @param path + * The plug-in relative path, not <code>null</code>. + * + * @return The image, not <code>null</code>. + */ + public final Image getImage(String path) { + if (path == null) { + throw new IllegalArgumentException( + "Parameter 'path' must not be null."); + } + Image result; + synchronized (images) { + result = images.get(path); + if (result == null) { + + ImageDescriptor imageDescriptor; + imageDescriptor = getImageDescriptor(path); + if (imageDescriptor == null) { + throw new RuntimeException("Image '" + path + + "' not found."); + } + result = imageDescriptor.createImage(); + images.put(path, result); + } + } + + return result; + + } + + /** + * Gets the image for the specified plug-in relative path. + * + * @param path + * The plug-in relative path, not <code>null</code>. + * + * @return The image descriptor or <code>null</code> if no image resource + * was found. + */ + public final ImageDescriptor getImageDescriptor(String path) { + ImageDescriptor imageDescriptor; + imageDescriptor = AbstractUIPlugin.imageDescriptorFromPlugin( + getPluginId(), "icons/" + path); + return imageDescriptor; + } + + /** + * Gets the absolute path of a entry from the plugin's directory. + * + * @param fileName + * The name of a file or directory, no compound name like + * "dir1\dir2" or "dir1\file1", not <code>null</code>. + * + * @return The absolute path of the file or directory in the plugin's + * directory, may be empty, not <code>null</code>. + */ + public final String getFilePathFromPlugin(String fileName) { + if (fileName == null) { + throw new IllegalArgumentException( + "Parameter 'entry' must not be null."); + } + URL url = null; + IPath path = null; + String result = ""; + + Enumeration<URL> enu = getBundle().findEntries("/", fileName, true); + if (enu != null && enu.hasMoreElements()) { + url = enu.nextElement(); + } + + if (url == null) { + return ""; + } + + try { + path = new Path(FileLocator.toFileURL(url).getPath()); + result = path.makeAbsolute().toOSString(); + } catch (Exception ex) { + result = ""; + } + + return result; + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/common/ByteArrayUtility.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/ByteArrayUtility.java new file mode 100644 index 00000000..50438e95 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/ByteArrayUtility.java @@ -0,0 +1,31 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.common; + +/** + * Utility class to handle byte arrays. + */ +public final class ByteArrayUtility { + + public static final int KB = 1024; + public static final int MB = KB * KB; + + public static final int MASK_FF = 0xff; +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/common/ClassPathUtility.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/ClassPathUtility.java new file mode 100644 index 00000000..261c7305 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/ClassPathUtility.java @@ -0,0 +1,97 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.common; + +import java.io.File; +import java.net.URI; +import java.net.URISyntaxException; +import java.net.URL; + +public final class ClassPathUtility { + + /** + * Find the URI of the ".jar" file containing the resource files. + * + * @return The URI of the ".jar" file containing the resource files or + * <code>null</code>. + */ + public static URI getJarURI() { + // Try to load simple menu from folder where the .jar file is located + ClassLoader classLoader = ClassPathUtility.class.getClassLoader(); + String jarPath = ClassPathUtility.class.getName().replace('.', '/') + ".class"; + URL url = classLoader.getResource(jarPath); + if (url == null) { + url = ClassLoader.getSystemResource(jarPath); + } + if (url == null) { + return null; + } + try { + URI uri = url.toURI(); + + // Convert "jar:file:/..." to file URI. + if (uri.getScheme().equals("jar")) { + String uriString = uri.getRawSchemeSpecificPart(); + int index = uriString.lastIndexOf("!"); + if (index > 0) { + uriString = uriString.substring(0, index); + uri = new URI(uriString); + return uri; + } + } + + } catch (URISyntaxException ex) { + throw new RuntimeException("Error when resolving URL '" + url + "'."); + } + return null; + + } + + /** + * The ".jar" file containing the resource files. + * + * @return The ".jar" file containing the resource files or + * <code>null</code>. + */ + public static File getJarFile() { + URI uri = getJarURI(); + + File result = null; + if (uri != null) { + result = new File(uri); + } + return result; + } + + /** + * The folder of the ".jar" file containing the resource files. + * + * @return The folder of the ".jar" file containing the resource files or + * <code>null</code>. + */ + public static File getJarFolder() { + + File result = getJarFile(); + if (result != null) { + result = result.getParentFile(); + } + return result; + } +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/common/EnumUtility.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/EnumUtility.java new file mode 100644 index 00000000..2e44efaa --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/EnumUtility.java @@ -0,0 +1,69 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.common; + +import java.util.Locale; +import java.util.MissingResourceException; +import java.util.ResourceBundle; + +import com.wudsn.ide.base.BasePlugin; + +/** + * Utility class for enums. + * + * @author Peter Dell + */ + +public final class EnumUtility { + + /** + * Gets the localized text for an enum value. + * + * @param enumValue + * The enum value, not <code>null</code>. + * @return The localized text, may be empty, not <code>null</code>. + */ + public static String getText(Enum<?> enumValue) { + if (enumValue == null) { + throw new IllegalArgumentException( + "Parameter 'enumValue' must not be null."); + } + + String result; + + Class<?> enumClass = enumValue.getClass(); + + String key = enumClass.getName() + "." + enumValue.name(); + try { + ResourceBundle resourceBundle; + + resourceBundle = ResourceBundle.getBundle("plugin", + Locale.getDefault(), enumClass.getClassLoader()); + result = resourceBundle.getString(key); + } catch (MissingResourceException ex) { + result = enumValue.name() + " - Text missing"; + BasePlugin.getInstance().logError( + "Resource for enum value {0} is missing.", + new Object[] { key }, ex); + } + return result; + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/common/FileUtility.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/FileUtility.java new file mode 100644 index 00000000..83e9ed9d --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/FileUtility.java @@ -0,0 +1,544 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.common; + +import java.io.ByteArrayOutputStream; +import java.io.File; +import java.io.FileInputStream; +import java.io.FileNotFoundException; +import java.io.FileOutputStream; +import java.io.IOException; +import java.io.InputStream; +import java.io.InputStreamReader; +import java.io.OutputStream; +import java.io.OutputStreamWriter; + +import org.eclipse.core.resources.IFile; +import org.eclipse.core.runtime.CoreException; +import org.eclipse.core.runtime.IStatus; +import org.eclipse.core.runtime.Status; + +import com.wudsn.ide.base.BasePlugin; +import com.wudsn.ide.base.Texts; + +/** + * Utility class to access files and their content. + * + * @author Peter Dell + */ +public final class FileUtility { + + /** + * Intentionally read an unlimited amount of bytes. + */ + public static final int MAX_SIZE_UNLIMITED = -1; + + /** + * Intentionally read at most 1MB of bytes or chars. + */ + public static final int MAX_SIZE_1MB = 1024 * 1024; + + /** + * Buffer size for the bytes or chars. + */ + private static final int BUFFER_SIZE = 8192; + + /** + * Creation is private, + */ + private FileUtility() { + } + + /** + * Reads the content of a file as byte array. + * + * @param ioFile + * The file, not <code>null</code>. + * @param maxSize + * The maximum number of character to read or + * {@link #MAX_SIZE_UNLIMITED}. + * @param errorOnMaxSizeExceeded + * If <code>true</code>, an error will be thrown in case the + * specified maximum number of bytes is exceeded. If + * <code>false</code>, the content will be truncated to the + * specified maximum size (plus the internal buffer size). + * + * @return The content of the file, may be empty, not <code>null</code>. + * + * @throws CoreException + * If the file does not exist or cannot be read. + */ + public static byte[] readBytes(File ioFile, long maxSize, boolean errorOnMaxSizeExceeded) throws CoreException { + InputStream inputStream; + String filePath; + + if (ioFile == null) { + throw new IllegalArgumentException("Parameter 'ioFile' must not be null."); + } + + filePath = ioFile.getAbsolutePath(); + try { + inputStream = new FileInputStream(ioFile); + } catch (FileNotFoundException ex) { + // ERROR: Cannot open file '{0}' for reading. + throw new CoreException(new Status(IStatus.ERROR, BasePlugin.ID, TextUtility.format(Texts.MESSAGE_E205, + filePath, ex.getMessage()), ex)); + } + return readBytes(filePath, inputStream, maxSize, errorOnMaxSizeExceeded); + + } + + /** + * Reads the content of a file as byte array. + * + * @param iFile + * The file, not <code>null</code>. + * @param maxSize + * The maximum number of character to read or + * {@link #MAX_SIZE_UNLIMITED}. + * @param errorOnMaxSizeExceeded + * If <code>true</code>, an error will be thrown in case the + * specified maximum number of bytes is exceeded. If + * <code>false</code>, the content will be truncated to the + * specified maximum size (plus the internal buffer size). + * + * @return The content of the file, may be empty, not <code>null</code>. + * + * @throws CoreException + * If the file does not exist or cannot be read. + */ + public static byte[] readBytes(IFile iFile, long maxSize, boolean errorOnMaxSizeExceeded) throws CoreException { + InputStream inputStream; + String filePath; + + if (iFile == null) { + throw new IllegalArgumentException("Parameter 'iFile' must not be null."); + } + + filePath = iFile.getFullPath().toString(); + try { + inputStream = iFile.getContents(); + } catch (CoreException ex) { + // ERROR: Cannot open file '{0}' for reading. {1} + throw new CoreException(new Status(IStatus.ERROR, BasePlugin.ID, TextUtility.format(Texts.MESSAGE_E205, + filePath, ex.getMessage()), ex)); + } + + return readBytes(filePath, inputStream, maxSize, errorOnMaxSizeExceeded); + + } + + private static byte[] readBytes(String filePath, InputStream inputStream, long maxSize, + boolean errrorOnMaxSizeExceeded) throws CoreException { + + if (filePath == null) { + throw new IllegalArgumentException("Parameter 'filePath' must not be null."); + } + if (inputStream == null) { + throw new IllegalArgumentException("Parameter 'inputStream' must not be null."); + } + if (maxSize < MAX_SIZE_UNLIMITED) { + throw new IllegalArgumentException("Parameter 'maxSize' must not be less than " + MAX_SIZE_UNLIMITED + "."); + } + + byte[] result; + try { + ByteArrayOutputStream bos = new ByteArrayOutputStream(); + + int size = 0; + byte[] buffer = new byte[BUFFER_SIZE]; + int count = 0; + + do { + count = inputStream.read(buffer); + + if (count > 0) { + bos.write(buffer, 0, count); + + size += count; + } + } while ((count > -1) && (maxSize == MAX_SIZE_UNLIMITED || size <= maxSize)); + + // Specified maximum size exceeded? + if (maxSize != MAX_SIZE_UNLIMITED && size > maxSize && errrorOnMaxSizeExceeded) { + // ERROR: Content of file '{0}' exceeds the specified maximum + // size of {1} bytes. + throw new CoreException(new Status(IStatus.ERROR, BasePlugin.ID, TextUtility.format(Texts.MESSAGE_E207, + filePath, String.valueOf(maxSize)))); + } + bos.close(); + + result = bos.toByteArray(); + + } catch (IOException ex) { + // ERROR: Cannot read content of file '{0}'. + throw new CoreException(new Status(IStatus.ERROR, BasePlugin.ID, TextUtility.format(Texts.MESSAGE_E206, + filePath), ex)); + + } finally { + try { + inputStream.close(); + } catch (IOException ex) { + // ERROR: Cannot close input stream of file'{0}'. + throw new CoreException(new Status(IStatus.ERROR, BasePlugin.ID, TextUtility.format(Texts.MESSAGE_E209, + filePath), ex)); + } + } + return result; + } + + /** + * Write a byte array to a file. + * + * @param ioFile + * The file, not <code>null</code>. + * @param content + * The content of the file, may be empty, not <code>null</code>. + * @throws CoreException + * If the file does not exist or cannot be read. + */ + public static void writeBytes(File ioFile, byte[] content) throws CoreException { + OutputStream outputStream; + String filePath; + + if (ioFile == null) { + throw new IllegalArgumentException("Parameter 'ioFile' must not be null."); + } + if (content == null) { + throw new IllegalArgumentException("Parameter 'content' must not be null."); + } + + filePath = ioFile.getAbsolutePath(); + + if (!ioFile.exists()) { + boolean result; + try { + result = ioFile.createNewFile(); + + } catch (IOException ex) { + // ERROR: Cannot create file '{0}'. + throw new CoreException(new Status(IStatus.ERROR, BasePlugin.ID, TextUtility.format(Texts.MESSAGE_E210, + filePath), ex)); + } + if (!result) { + // ERROR: Cannot create file '{0}'. + throw new CoreException(new Status(IStatus.ERROR, BasePlugin.ID, TextUtility.format(Texts.MESSAGE_E210, + filePath))); + } + } + + if (!ioFile.isFile()) { + // ERROR: '{0}' is no file but a folder. + throw new CoreException(new Status(IStatus.ERROR, BasePlugin.ID, TextUtility.format(Texts.MESSAGE_E204, + filePath))); + } + try { + outputStream = new FileOutputStream(ioFile); + } catch (FileNotFoundException ex) { + // ERROR: Cannot open file '{0}' for writing. + throw new CoreException(new Status(IStatus.ERROR, BasePlugin.ID, TextUtility.format(Texts.MESSAGE_E211, + filePath), ex)); + } + writeBytes(filePath, outputStream, content); + + } + + private static void writeBytes(String filePath, OutputStream outputStream, byte[] content) throws CoreException { + + if (filePath == null) { + throw new IllegalArgumentException("Parameter 'filePath' must not be null."); + } + if (outputStream == null) { + throw new IllegalArgumentException("Parameter 'outputStream' must not be null."); + } + if (content == null) { + throw new IllegalArgumentException("Parameter 'content' must not be null."); + } + + try { + outputStream.write(content); + } catch (IOException ex) { + // ERROR: Cannot write content of file '{0}'. + throw new CoreException(new Status(IStatus.ERROR, BasePlugin.ID, TextUtility.format(Texts.MESSAGE_E212, + filePath), ex)); + } finally { + try { + outputStream.close(); + } catch (IOException ex) { + // ERROR: Cannot close output stream of file'{0}'. + throw new CoreException(new Status(IStatus.ERROR, BasePlugin.ID, TextUtility.format(Texts.MESSAGE_E213, + filePath), ex)); + } + } + return; + } + + /** + * Reads the content of a file as string. + * + * @param ioFile + * The file, not <code>null</code>. + * @param maxSize + * The maximum number of character to read or + * {@link #MAX_SIZE_UNLIMITED}. + * @return The content of the file, may be empty, not <code>null</code>. + * @throws CoreException + * If the file does not exist or cannot be read. + */ + public static String readString(File ioFile, long maxSize) throws CoreException { + InputStream inputStream; + String filePath; + + if (ioFile == null) { + throw new IllegalArgumentException("Parameter 'ioFile' must not be null."); + } + + filePath = ioFile.getAbsolutePath(); + + if (!ioFile.exists()) { + // ERROR: File '{0}' does not exist. + throw new CoreException(new Status(IStatus.ERROR, BasePlugin.ID, TextUtility.format(Texts.MESSAGE_E203, + filePath))); + } + if (!ioFile.isFile()) { + // ERROR: '{0}' is no file but a folder. + throw new CoreException(new Status(IStatus.ERROR, BasePlugin.ID, TextUtility.format(Texts.MESSAGE_E204, + filePath))); + } + try { + inputStream = new FileInputStream(ioFile); + } catch (FileNotFoundException ex) { + // ERROR: Cannot open file '{0}' for reading. + throw new CoreException(new Status(IStatus.ERROR, BasePlugin.ID, TextUtility.format(Texts.MESSAGE_E205, + filePath), ex)); + } + return readString(filePath, inputStream, maxSize); + + } + + /** + * Reads a string from an input stream. + * + * @param filePath + * The file path to be used in error messages. + * @param inputStream + * The input stream, not <code>null</code>. + * @param maxSize + * The maximum number of bytes to read, a non-negative integer or @link + * #MAX_SIZE_UNLIMITED}, see also {@link #MAX_SIZE_1MB}. + * @return The string, may be empty, not <code>null</code>. + * @throws CoreException + * in case the file cannot be read. the iNput stream has been + * closed in this case. + */ + public static String readString(String filePath, InputStream inputStream, long maxSize) throws CoreException { + + if (filePath == null) { + throw new IllegalArgumentException("Parameter 'filePath' must not be null."); + } + if (inputStream == null) { + throw new IllegalArgumentException("Parameter 'inputStream' must not be null."); + } + if (maxSize < MAX_SIZE_UNLIMITED) { + throw new IllegalArgumentException("Parameter 'maxSize' must not be less than " + MAX_SIZE_UNLIMITED + "."); + } + + StringBuilder result; + result = new StringBuilder(); + try { + InputStreamReader reader = new InputStreamReader(inputStream); + + int size = 0; + char[] buffer = new char[BUFFER_SIZE]; + int count = 0; + + do { + count = reader.read(buffer); + + if (count > 0) { + result.append(buffer, 0, count); + + size += count; + } + } while ((count > -1) && (maxSize == MAX_SIZE_UNLIMITED || size <= maxSize)); + + // Specified maximum size exceeded? + if (maxSize != MAX_SIZE_UNLIMITED && size > maxSize) { + // ERROR: Content of file '{0}' exceeds the specified maximum + // size of {1} characters. + throw new CoreException(new Status(IStatus.ERROR, BasePlugin.ID, TextUtility.format(Texts.MESSAGE_E208, + filePath, String.valueOf(maxSize)))); + } + + reader.close(); + + } catch (IOException ex) { + // ERROR: Cannot read content of file '{0}'. + throw new CoreException(new Status(IStatus.ERROR, BasePlugin.ID, TextUtility.format(Texts.MESSAGE_E206, + filePath), ex)); + } finally { + try { + inputStream.close(); + } catch (IOException ex) { + // ERROR: Cannot close input stream of file'{0}'. + throw new CoreException(new Status(IStatus.ERROR, BasePlugin.ID, TextUtility.format(Texts.MESSAGE_E209, + filePath), ex)); + } + } + return result.toString(); + } + + /** + * Write a string to a file. + * + * @param ioFile + * The file, not <code>null</code>. + * @param content + * The content of the file, may be empty, not <code>null</code>. + * @throws CoreException + * If the file does not exist or cannot be read. + */ + public static void writeString(File ioFile, String content) throws CoreException { + OutputStream outputStream; + String filePath; + + if (ioFile == null) { + throw new IllegalArgumentException("Parameter 'ioFile' must not be null."); + } + if (content == null) { + throw new IllegalArgumentException("Parameter 'content' must not be null."); + } + + filePath = ioFile.getAbsolutePath(); + + if (!ioFile.exists()) { + boolean result; + try { + result = ioFile.createNewFile(); + + } catch (IOException ex) { + // ERROR: Cannot create file '{0}'. + throw new CoreException(new Status(IStatus.ERROR, BasePlugin.ID, TextUtility.format(Texts.MESSAGE_E210, + filePath), ex)); + } + if (!result) { + // ERROR: Cannot create file '{0}'. + throw new CoreException(new Status(IStatus.ERROR, BasePlugin.ID, TextUtility.format(Texts.MESSAGE_E210, + filePath))); + } + } + + if (!ioFile.isFile()) { + // ERROR: '{0}' is no file but a folder. + throw new CoreException(new Status(IStatus.ERROR, BasePlugin.ID, TextUtility.format(Texts.MESSAGE_E204, + filePath))); + } + try { + outputStream = new FileOutputStream(ioFile); + } catch (FileNotFoundException ex) { + // ERROR: Cannot open file '{0}' for writing. + throw new CoreException(new Status(IStatus.ERROR, BasePlugin.ID, TextUtility.format(Texts.MESSAGE_E211, + filePath), ex)); + } + writeString(filePath, outputStream, content); + + } + + private static void writeString(String filePath, OutputStream outputStream, String content) throws CoreException { + + if (filePath == null) { + throw new IllegalArgumentException("Parameter 'filePath' must not be null."); + } + if (outputStream == null) { + throw new IllegalArgumentException("Parameter 'outputStream' must not be null."); + } + if (content == null) { + throw new IllegalArgumentException("Parameter 'content' must not be null."); + } + + try { + OutputStreamWriter writer = new OutputStreamWriter(outputStream); + writer.write(content); + + writer.close(); + + } catch (IOException ex) { + // ERROR: Cannot write content of file '{0}'. + throw new CoreException(new Status(IStatus.ERROR, BasePlugin.ID, TextUtility.format(Texts.MESSAGE_E212, + filePath), ex)); + } finally { + try { + outputStream.close(); + } catch (IOException ex) { + // ERROR: Cannot close output stream of file'{0}'. + throw new CoreException(new Status(IStatus.ERROR, BasePlugin.ID, TextUtility.format(Texts.MESSAGE_E213, + filePath), ex)); + } + } + + return; + } + + /** + * Gets the canonical file of a file. If the canonical path cannot be + * determined, the absolute path is returned. + * + * @param file + * The file, not <code>null</code>. + * @return The canonical file of the file,not <code>null</code>. + * + * @since 1.7.0 + */ + public static File getCanonicalFile(File file) { + if (file == null) { + throw new IllegalArgumentException("Parameter 'result' must not be null."); + } + File result; + try { + result = file.getCanonicalFile(); + } catch (IOException ex) { + result = file.getAbsoluteFile(); + } + return result; + + } + + /** + * Gets the file extension from a file path. + * + * @param path + * The file path, may be empty, not <code>null</code>. + * @return The file extension, may be empty, not <code>null</code>. + * + * @since 1.7.0 + */ + public static final String getFileExtension(String path) { + if (path == null) { + throw new IllegalArgumentException("Parameter 'path' must not be null."); + } + String extension = ""; + int index = path.lastIndexOf('.'); + if (index >= 0) { + extension = path.substring(index); + } + return extension; + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/common/HexUtility.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/HexUtility.java new file mode 100644 index 00000000..e555e086 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/HexUtility.java @@ -0,0 +1,159 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.common; + +/** + * Utility class to handle hex numbers. + */ +public final class HexUtility { + + /** + * Automatic length indicator for {@link #getLongValueHexString(long, int)}, + */ + public static final int AUTOMATIC_LENGTH = 0; + + /** + * Static array of all one-byte upper case hex numbers (00...FF) + */ + private static final String[] hexStrings; + + /** + * Static initialization. + */ + static { + + // Fill in the array of hex strings + hexStrings = new String[256]; + for (int i = 0; i < 256; i++) { + String s = Integer.toHexString(i).toUpperCase(); + hexStrings[i] = (s.length() < 2) ? ("0" + s) : s; + } + } + + /** + * Creation is private. + */ + private HexUtility() { + + } + + /** + * Gets the hex string for a single byte value. + * + * @param byteValue + * The byte value. + * @return The string, not empty and not <code>null</code>. + */ + public static String getByteValueHexString(int byteValue) { + if (byteValue < 0 || byteValue > 255) { + throw new IllegalArgumentException("Integer value " + byteValue + + " is no byte value."); + } + return hexStrings[byteValue]; + } + + /** + * Gets the minimum length of the hex string for a long value. + * + * @param longValue + * The non-negative long value. + * + * @return The minimum length of the hex string for the long value, an even + * positive integer. + */ + public static int getLongValueHexLength(long longValue) { + if (longValue < 0) { + throw new RuntimeException( + "Parameter 'longValue' must not be negative. Specified value is " + + longValue + "."); + } + int result = Long.toHexString(longValue).length(); + if ((result & 1) == 1) { + result++; + } + return result; + } + + /** + * Gets the hex string for a long value. + * + * @param longValue + * The non-negative long value. + * @return The string, not empty and not <code>null</code>. + */ + public static String getLongValueHexString(long longValue) { + return getLongValueHexString(longValue, + getLongValueHexLength(longValue)); + } + + /** + * Gets the hex string for a long value. + * + * @param longValue + * The non-negative long value. + * @param length + * The minimum number of characters to be used. If the result + * string is shorter, spaces will be prepended in case a length + * other then {@link #AUTOMATIC_LENGTH} is specified. + * @return The string, not empty and not <code>null</code>. + */ + public static String getLongValueHexString(long longValue, int length) { + if (longValue < 0) { + throw new RuntimeException( + "Parameter 'longValue' must not be negative. Specified value is " + + longValue + "."); + } + if (length < 0) { + throw new IllegalArgumentException( + "Parameter 'length' must not be negative. Specified value is " + + length + "."); + } + String result = Long.toHexString(longValue).toUpperCase(); + if (length > AUTOMATIC_LENGTH) { + int difference = length - result.length(); + if (difference > 0) { + StringBuilder builder = new StringBuilder(length); + for (int i = 0; i < difference; i++) { + builder.append('0'); + } + builder.insert(difference, result); + result = builder.toString(); + } + } + return result; + } + + /** + * Gets the ASCII char for a single byte value. + * + * @param byteValue + * The byte value. + * @return The string, not empty and not <code>null</code>. + */ + public static char getChar(int byteValue) { + char result; + if (byteValue >= 32 && byteValue <= 127) { + result = (char) (byteValue & 0xff); + } else { + result = ' '; + } + return result; + } +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/common/IPathUtility.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/IPathUtility.java new file mode 100644 index 00000000..1515c2cd --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/IPathUtility.java @@ -0,0 +1,77 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.common; + +import org.eclipse.core.runtime.IPath; +import org.eclipse.core.runtime.Path; + +public final class IPathUtility { + + /** + * Creation is private. + */ + private IPathUtility() { + } + + public static IPath createEmptyPath() { + return new Path(""); + } + + public static IPath makeRelative(IPath filePath, IPath filePathPrefix) { + if (filePath == null) { + throw new IllegalArgumentException( + "Parameter 'filePath' must not be null."); + } + if (filePathPrefix == null) { + throw new IllegalArgumentException( + "Parameter 'filePathPrefix' must not be null."); + } + if (filePath.isAbsolute() && !filePathPrefix.isEmpty()) { + if (filePathPrefix.isPrefixOf(filePath)) { + filePath = filePath.removeFirstSegments(filePathPrefix + .segmentCount()); + } + } + return filePath; + } + + public static IPath makeAbsolute(IPath filePath, IPath filePathPrefix, + boolean forcePrefix) { + if (filePath == null) { + throw new IllegalArgumentException( + "Parameter 'filePath' must not be null."); + } + if (filePathPrefix == null) { + throw new IllegalArgumentException( + "Parameter 'filePathPrefix' must not be null."); + } + + // If the file path is empty, the prefix is omitted by default. + // Only if forcePrefix is true, it is added. + if (!filePath.isEmpty() || forcePrefix) { + if (!filePath.isAbsolute() && !filePathPrefix.isEmpty()) { + if (!filePathPrefix.isPrefixOf(filePath)) { + filePath = filePathPrefix.append(filePath); + } + } + } + return filePath; + } +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/common/InputStreamMonitor.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/InputStreamMonitor.java new file mode 100644 index 00000000..892b4019 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/InputStreamMonitor.java @@ -0,0 +1,176 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.common; + +import java.io.IOException; +import java.io.OutputStream; +import java.util.*; + +import com.wudsn.ide.base.BasePlugin; + +/** + * Writes to the input stream of a system process, queueing output if the stream + * is blocked. + * + * The input stream monitor writes to system in via an output stream. + * + * @author Peter Dell + */ +final class InputStreamMonitor { + + /** + * The stream which is being written to (connected to system in). + */ + private OutputStream outputStream; + + /** + * Whether the underlying output stream has been closed + */ + private boolean outputStreamClosed; + + /** + * The queue of output. + */ + private List<String> outputQueue; + /** + * A lock for ensuring that writes to the queue are contiguous + */ + private Object outputQueueLock; + + /** + * The thread which writes to the stream. + */ + private Thread thread; + + /** + * Creates an input stream monitor which writes to system in via the given + * output stream. + * + * @param outputStream + * output stream + */ + public InputStreamMonitor(OutputStream outputStream) { + if (outputStream == null) { + throw new IllegalArgumentException( + "Parameter 'outputStream' must not be null."); + } + this.outputStream = outputStream; + outputQueue = new ArrayList<String>(); + outputQueueLock = new Object(); + } + + /** + * Appends the given text to the stream, or queues the text to be written at + * a later time if the stream is blocked. + * + * @param text + * text to append + */ + public void write(String text) { + synchronized (outputQueueLock) { + outputQueue.add(text); + outputQueueLock.notifyAll(); + } + } + + /** + * Starts a thread which writes the stream. + */ + public void startMonitoring() { + if (thread == null) { + thread = new Thread(new RunnableWithLogging() { + @Override + public void runWithLogging() { + write(); + } + }, "InputStreamMonitor"); + thread.setDaemon(true); + thread.start(); + } + } + + /** + * Close all communications between this monitor and the underlying stream. + */ + public void close() { + if (thread != null) { + Thread thread = this.thread; + this.thread = null; + thread.interrupt(); + } + } + + /** + * Continuously writes to the stream. + */ + protected void write() { + while (thread != null) { + writeNext(); + } + if (!outputStreamClosed) { + try { + outputStream.close(); + } catch (IOException ex) { + BasePlugin.getInstance().logError("IOException during write()", + null, ex); + } + } + } + + /** + * Write the text in the queue to the stream. + */ + protected void writeNext() { + while (!outputQueue.isEmpty() && !outputStreamClosed) { + String text = outputQueue.get(0); + outputQueue.remove(0); + try { + outputStream.write(text.getBytes()); + outputStream.flush(); + } catch (IOException ex) { + BasePlugin.getInstance().logError( + "IOException during writeNext()", null, ex); + } + } + try { + synchronized (outputQueueLock) { + outputQueueLock.wait(); + } + } catch (InterruptedException e) { + } + } + + /** + * Closes the output stream attached to the standard input stream of this + * monitor's process. + * + * @exception IOException + * if an exception occurs closing the input stream + */ + public void closeInputStream() throws IOException { + if (!outputStreamClosed) { + outputStreamClosed = true; + outputStream.close(); + } else { + throw new IOException(); + } + + } +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/common/MarkerUtility.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/MarkerUtility.java new file mode 100644 index 00000000..c0d9826e --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/MarkerUtility.java @@ -0,0 +1,125 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.common; + +import org.eclipse.core.resources.IFile; +import org.eclipse.core.resources.IMarker; +import org.eclipse.core.runtime.CoreException; +import org.eclipse.ui.IEditorPart; +import org.eclipse.ui.IWorkbenchPage; +import org.eclipse.ui.PartInitException; +import org.eclipse.ui.ide.IDE; + +/** + * Utility class for processing markers. + * + * @author Peter Dell + * @since 1.6.1 + */ +public final class MarkerUtility { + + /** + * Creation is private. + */ + private MarkerUtility() { + + } + + /** + * Creates a marker associated with an {@link IFile} resource. + * + * @param file + * The {@link IFile} resource to which this message shall be + * attached, not <code>null</code>. + * @param lineNumber + * An positive integer value indicating the line number for a + * text marker. 0 to indicate that the line number is unknown. + * @param severity + * The message severity, see {@link IMarker#SEVERITY} + * @param message + * The message, may contain parameter "{0}" to "{9}". May be + * empty, not <code>null</code>. + * @param parameters + * The format parameters for the message, may be empty, not + * <code>null</code>. + * + * @return The marker representing the message, not <code>null</code>. + */ + public static IMarker createMarker(IFile file, int lineNumber, + int severity, String message, String... parameters) { + if (file == null) { + throw new IllegalArgumentException( + "Parameter 'file' must not be null."); + } + if (message == null) { + throw new IllegalArgumentException( + "Parameter 'message' must not be null."); + } + if (parameters == null) { + throw new IllegalArgumentException( + "Parameter 'parameters' must not be null."); + } + + message = TextUtility.format(message, parameters); + try { + + IMarker marker = file.createMarker(IMarker.PROBLEM); + if (lineNumber > 0) { + marker.setAttribute(IMarker.LINE_NUMBER, lineNumber); + } + marker.setAttribute(IMarker.SEVERITY, severity); + marker.setAttribute(IMarker.MESSAGE, message); + marker.setAttribute(IMarker.TRANSIENT, true); + return marker; + } catch (CoreException ex) { + throw new RuntimeException(ex); + } + } + + /** + * Navigates to the file and line number defined by the marker. + * + * @param editor + * The editor part to start from, not <code>null</code>. If + * required an additional editor will be opened in the same + * workbench page. + * + * @param marker + * The marker, not <code>null</code>. The resource of the marker + * must be an {@link IFile}. + */ + public static void gotoMarker(IEditorPart editor, IMarker marker) { + if (marker == null) { + throw new IllegalArgumentException( + "Parameter 'marker' must not be null."); + } + try { + IFile iFile = (IFile) marker.getResource(); + IWorkbenchPage page = editor.getSite().getPage(); + IEditorPart otherEditor = IDE.openEditor(page, iFile, true); + if (otherEditor != null) { + otherEditor.setFocus(); + IDE.gotoMarker(otherEditor, marker); + } + } catch (PartInitException ex) { + throw new RuntimeException(ex); + } + } +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/common/NumberFactory.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/NumberFactory.java new file mode 100644 index 00000000..808d9b2f --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/NumberFactory.java @@ -0,0 +1,57 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.common; + +/** + * Factory class to obtain instances of elementary number type wrappers without + * creating new instances. + * + * @author Peter Dell + * + */ +public final class NumberFactory { + + private static final int MAX_INTEGERS = 2048; + private static final Integer[] INTEGERS; + + /** + * Static initialization. + */ + static { + INTEGERS = new Integer[MAX_INTEGERS]; + for (int i = 0; i < MAX_INTEGERS; i++) { + INTEGERS[i] = Integer.valueOf(i); + } + } + + /** + * Gets the @link Integer} instance for an int value. + * + * @param value + * The int value. + * @return The {@link Integer} instance, not <code>null</code>. + */ + public static final Integer getInteger(int value) { + if (0 <= value && value <= MAX_INTEGERS) { + + } + return Integer.valueOf(value); + } +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/common/NumberUtility.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/NumberUtility.java new file mode 100644 index 00000000..ed384322 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/NumberUtility.java @@ -0,0 +1,89 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.common; + +/** + * Utility class to handle decimal numbers. + */ +public final class NumberUtility { + + /** + * Automatic length indicator for + * {@link #getLongValueDecimalString(long, int)}, + */ + public static final int AUTOMATIC_LENGTH = 0; + + /** + * Creation is private. + */ + private NumberUtility() { + } + + /** + * Gets the minimum length of the decimal string for a long value. + * + * @param longValue + * The non-negative long value. + * + * @return The minimum length of the decimal string for the long value, a + * positive integer. + */ + public static int getLongValueDecimalLength(int longValue) { + return Long.toString(longValue).length(); + } + + /** + * Gets the decimal string for a long value. + * + * @param longValue + * The non-negative long value. + * + * @return The string, not empty and not <code>null</code>. + */ + public static String getLongValueDecimalString(long longValue) { + return getLongValueDecimalString(longValue, AUTOMATIC_LENGTH); + } + + /** + * Gets the decimal string for a long value. + * + * @param longValue + * The non-negative long value. + * @param length + * The minimum number of characters to be used. If the result + * string is shorter, spaces will be prepended in case a length + * other then {@link #AUTOMATIC_LENGTH} is specified. + * @return The string, not empty and not <code>null</code>. + */ + public static String getLongValueDecimalString(long longValue, int length) { + + String result = Long.toString(longValue); + if (length > AUTOMATIC_LENGTH) { + int difference = length - result.length(); + if (difference > 0) { + StringBuilder builder = new StringBuilder(length); + builder.insert(difference, result); + result = builder.toString(); + } + } + return result; + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/common/OutputStreamMonitor.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/OutputStreamMonitor.java new file mode 100644 index 00000000..7ff2fc9b --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/OutputStreamMonitor.java @@ -0,0 +1,206 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.common; + +import java.io.BufferedInputStream; +import java.io.IOException; +import java.io.InputStream; +import java.io.PrintStream; + +import com.wudsn.ide.base.BasePlugin; + +/** + * Monitors the output stream of a system process and notifies listeners of + * additions to the stream. + * + * The output stream monitor reads system out (or err) via and input stream. + * + * @author Peter Dell + */ +final class OutputStreamMonitor { + + /** + * The stream being monitored (connected system out or err). + */ + private InputStream inputStream; + + /** + * The stream to which the output is sent. + */ + private PrintStream outputStream; + + /** + * The encoding of the stream to which the output is sent. + */ + private String outputStreamEncoding; + /** + * The local copy of the stream contents + */ + private StringBuilder bufferedContent; + + /** + * The thread which reads from the stream + */ + private Thread thread; + + /** + * The size of the read buffer + */ + private static final int BUFFER_SIZE = 8192; + + /** + * Whether or not this monitor has been killed. When the monitor is killed, + * it stops reading from the stream immediately. + */ + private boolean killed; + + /** + * Creates an output stream monitor on the given stream (connected to system + * out or err). + * + * @param inputStream + * The input stream to read from, not <code>null</code>. + * @param outputStreamEncoding + * The stream encoding or <code>null</code> for system default. + * @param outputStream + * The output stream, not <code>null</code>. + */ + public OutputStreamMonitor(InputStream inputStream, String outputStreamEncoding, PrintStream outputStream) { + if (inputStream == null) { + throw new IllegalArgumentException("Parameter 'stream' must not be null."); + } + if (outputStream == null) { + throw new IllegalArgumentException("Parameter 'outputStream' must not be null."); + } + this.inputStream = new BufferedInputStream(inputStream, BUFFER_SIZE); + this.outputStreamEncoding = outputStreamEncoding; + this.outputStream = outputStream; + bufferedContent = new StringBuilder(); + } + + /** + * Causes the monitor to close all communications between it and the + * underlying stream by waiting for the thread to terminate. + */ + protected void close() { + if (this.thread != null) { + Thread thread = this.thread; + this.thread = null; + try { + thread.join(); + } catch (InterruptedException ie) { + } + } + } + + /* + * (non-Javadoc) + * + * @see org.eclipse.debug.core.model.IStreamMonitor#getContents() + */ + public String getContents() { + synchronized (bufferedContent) { + return bufferedContent.toString(); + } + } + + /** + * Continually reads from the stream. + * <p> + * This method, along with the <code>startReading</code> method is used to + * allow <code>OutputStreamMonitor</code> to implement <code>Runnable</code> + * without publicly exposing a <code>run</code> method. + */ + void read() { + long lastSleep = System.currentTimeMillis(); + byte[] bytes = new byte[8192]; + int read = 0; + while (read >= 0) { + try { + if (killed) { + break; + } + read = inputStream.read(bytes); + if (read > 0) { + String text; + if (outputStreamEncoding != null) { + text = new String(bytes, 0, read, outputStreamEncoding); + } else { + text = new String(bytes, 0, read); + } + synchronized (bufferedContent) { + bufferedContent.append(text); + outputStream.print(text); + outputStream.flush(); + } + } + } catch (IOException ioe) { + if (!killed) { + BasePlugin.getInstance().logError("IOException occured", null, ioe); + } + return; + } catch (NullPointerException ex) { + // killing the stream monitor while reading can cause an NPE + // when reading from the stream + if (!killed && this.thread != null) { + BasePlugin.getInstance().logError("Cannot read from stream", null, ex); + } + return; + } + + long currentTime = System.currentTimeMillis(); + if (currentTime - lastSleep > 1000) { + lastSleep = currentTime; + try { + Thread.sleep(100); // just give up CPU to maintain UI + // responsiveness. + } catch (InterruptedException e) { + } + } + } + try { + inputStream.close(); + } catch (IOException ex) { + BasePlugin.getInstance().logError("Cannot close input stream.", null, ex); + } + + } + + protected void kill() { + killed = true; + } + + /** + * Starts a thread which reads from the stream + */ + protected void startMonitoring() { + if (this.thread == null) { + this.thread = new Thread(new RunnableWithLogging() { + @Override + public void runWithLogging() { + read(); + } + }, "OutputStreamMonitor"); + } + this.thread.setDaemon(true); + this.thread.setPriority(Thread.MIN_PRIORITY); + this.thread.start(); + } +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/common/ProcessWithLogs.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/ProcessWithLogs.java new file mode 100644 index 00000000..a6875538 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/ProcessWithLogs.java @@ -0,0 +1,227 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.common; + +import java.io.File; +import java.io.IOException; +import java.io.PrintStream; + +import org.eclipse.core.runtime.Platform; + +import com.wudsn.ide.base.BasePlugin; + +/** + * The process with logs is the inter-process interface to the executables. The + * {@link System#out} and the {@link System#err} streams are captured into + * strings. + * + * @author Peter Dell + */ +public final class ProcessWithLogs { + + private final String[] commandArray; + private final File workingDirectory; + private int exitValue; + private String outputLog; + private String errorLog; + + public static final String[] getExecutableExtensions() { + + String[] extensions; + String os = Platform.getOS(); + if (os.equals(Platform.OS_WIN32)) { + // Default to ".exe" for Windows + extensions = new String[] { "*.exe", "*.jar", "*.*" }; + } else if (os.equals(Platform.OS_MACOSX)) { + // No restrictions for MacOS X since "*.app" is not always a good + // choice. + extensions = new String[0]; + } else { + // No restrictions for all other operating systems. + extensions = new String[0]; + } + return extensions; + } + + /** + * Creates a new process. + * + * @param commandArray + * The command array, not empty and not <code>null</code>. + * @param workingDirectory + * The working directory, not <code>null</code>. + */ + public ProcessWithLogs(String[] commandArray, File workingDirectory) { + if (commandArray == null) { + throw new IllegalArgumentException( + "Parameter 'fullCommandArray' must not be null."); + } + if (commandArray.length == 0) { + throw new IllegalArgumentException( + "Parameter 'fullCommandArray.length' must not be empty."); + } + for (int i = 0; i < commandArray.length; i++) { + if (commandArray[i] == null) { + throw new IllegalArgumentException( + "Parameter 'commandArray' must contain null at positition " + + i + "."); + } + } + + if (workingDirectory == null) { + throw new IllegalArgumentException( + "Parameter 'workingDirectory' must not be null."); + } + + this.commandArray = commandArray; + this.workingDirectory = FileUtility.getCanonicalFile(workingDirectory); + exitValue = 0; + outputLog = ""; + errorLog = ""; + } + + /** + * Executes the compiler. + * + * @param out + * The print stream for the output output, see {@link System#out} + * . + * @param err + * The print stream for the error output, see {@link System#err}. + * @param wait + * <code>true</code> to wait for the process to terminate and + * collect the output. + * + * @throws IOException + * The the creation of the process fails. + */ + public void exec(PrintStream out, PrintStream err, boolean wait) + throws IOException { + + if (out == null) { + throw new IllegalArgumentException( + "Parameter 'out' must not be null."); + } + if (err == null) { + throw new IllegalArgumentException( + "Parameter 'err' must not be null."); + } + Process process = null; + exitValue = 0; + outputLog = ""; + errorLog = ""; + Profiler profiler = new Profiler(this); + profiler.begin("exec"); + try { + BasePlugin.getInstance().log( + "Executing process '{0}' in working directory '{1}'.", + new Object[] { getCommandArrayString(), + workingDirectory.getPath() }); + + process = Runtime.getRuntime().exec(commandArray, null, + workingDirectory); + } catch (IOException ex) { + BasePlugin.getInstance().logError( + "Cannot execute process '{0}' in working directory '{1}'.", + new Object[] { getCommandArrayString(), + workingDirectory.getPath() }, ex); + throw ex; + } finally { + profiler.end("exec"); + } + if (wait) { + String encoding = null; + StreamsProxy streamsProxy = new StreamsProxy(process, encoding, + out, err); + try { + profiler.begin("waitFor"); + process.waitFor(); + } catch (InterruptedException ex) { + BasePlugin.getInstance().logError("Process interrupted", null, + ex); + throw new IOException(ex.getMessage()); + } finally { + profiler.end("waitFor"); + process.destroy(); + } + + streamsProxy.close(); + exitValue = process.exitValue(); + outputLog = streamsProxy.getOutputStreamMonitor().getContents(); + errorLog = streamsProxy.getErrorStreamMonitor().getContents(); + } + + } + + /** + * Gets the command array to be as string. + * + * @return The command array as string, may be empty, not <code>null</code> + * .. + */ + public String getCommandArrayString() { + StringBuilder result; + result = new StringBuilder(); + for (int i = 0; i < commandArray.length; i++) { + result.append(commandArray[i]); + if (i < commandArray.length - 1) { + result.append(" "); + } + } + return result.toString(); + } + + /** + * Gets the working directory. + * + * @return The working directory, not <code>null</code>. + */ + public File getWorkingDirectory() { + return workingDirectory; + } + + /** + * Gets the exit value of the process. + * + * @return The exit value of the process. + */ + public int getExitValue() { + return exitValue; + } + + /** + * Gets the output log captured from {@link System#out}. + * + * @return The output log, maybe empty, not <code>null</code>. + */ + public String getOutputLog() { + return outputLog; + } + + /** + * Gets the error log captured from {@link System#out}. + * + * @return The output log, maybe empty, not <code>null</code>. + */ + public String getErrorLog() { + return errorLog; + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/common/Profiler.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/Profiler.java new file mode 100644 index 00000000..c0ee2047 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/Profiler.java @@ -0,0 +1,88 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.common; + +import java.util.HashMap; +import java.util.Map; + +import com.wudsn.ide.base.BasePlugin; + +public final class Profiler { + + public static final String PROPERTY_NAME = "com.wudsn.ide.common.base.Profiler"; + + private static final class Entry { + + public final long startTimeMillis; + public final String description; + + public Entry(String description) { + if (description == null) { + throw new IllegalArgumentException( + "Parameter 'description' must not be null."); + } + startTimeMillis = System.currentTimeMillis(); + this.description = description; + } + } + + private Object owner; + private Map<String, Entry> statistics; + + public Profiler(Object owner) { + if (owner == null) { + throw new IllegalArgumentException( + "Parameter 'owner' must not be null."); + } + this.owner = owner; + statistics = new HashMap<String, Entry>(); + } + + public void end(String key) { + Entry entry = statistics.get(key); + if (entry == null) { + throw new IllegalStateException("No begin for key '" + key + "'."); + } + + if (Boolean.getBoolean(PROPERTY_NAME)) { + Long duration = new Long( + (System.currentTimeMillis() - entry.startTimeMillis)); + if (entry.description.isEmpty()) { + BasePlugin.getInstance().log("Time for '{0}:{1}' is {2}ms", + new Object[] { owner, key, duration }); + } else { + BasePlugin.getInstance() + .log("Time for '{0}:{1}' of {2} is {3}ms", + new Object[] { owner, key, entry.description, + duration }); + } + } + + } + + public void begin(String key) { + begin(key, ""); + } + + public void begin(String key, String description) { + statistics.put(key, new Entry(description)); + + } +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/common/PropertiesSerializer.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/PropertiesSerializer.java new file mode 100644 index 00000000..292830f4 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/PropertiesSerializer.java @@ -0,0 +1,233 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.common; + +import java.util.Enumeration; +import java.util.Properties; + +/** + * Serializer which reads and writes simple and complex data types as + * properties. + * + * @author Peter Dell + */ +public class PropertiesSerializer { + + protected final SequencedProperties properties; + + /** + * Creation is public. + */ + public PropertiesSerializer() { + + properties = new SequencedProperties(); + } + + public final SequencedProperties getProperties() { + return properties; + } + + protected final void setProperty(String key, String value) { + if (key == null) { + throw new IllegalArgumentException( + "Parameter 'key' must not be null."); + } + if (value == null) { + throw new IllegalArgumentException( + "Parameter 'value' must not be null."); + } + + properties.put(key, value); + } + + public final String readString(String key, String defaultValue) { + + if (key == null) { + throw new IllegalArgumentException( + "Parameter 'key' must not be null."); + } + if (defaultValue == null) { + throw new IllegalArgumentException( + "Parameter 'defaultValue' must not be null."); + } + String result; + result = properties.getProperty(key); + if (result == null) { + result = defaultValue; + } + + return result; + } + + public final void writeString(String key, String value) { + + if (key == null) { + throw new IllegalArgumentException( + "Parameter 'key' must not be null."); + } + if (value == null) { + throw new IllegalArgumentException( + "Parameter 'value' must not be null."); + } + setProperty(key, value); + } + + public boolean readBoolean(String key, boolean defaultValue) { + if (key == null) { + throw new IllegalArgumentException( + "Parameter 'key' must not be null."); + + } + boolean result; + result = defaultValue; + String text = properties.getProperty(key); + if (text != null) { + result = Boolean.parseBoolean(text); + } + return result; + } + + public final void writeBoolean(String key, boolean value) { + if (key == null) { + throw new IllegalArgumentException( + "Parameter 'key' must not be null."); + } + properties.setProperty(key, Boolean.toString(value)); + } + + public final <T extends Enum<?>> T readEnum(String key, T defaultValue, + Class<T> enumClass) { + if (key == null) { + throw new IllegalArgumentException( + "Parameter 'key' must not be null."); + } + if (defaultValue == null) { + throw new IllegalArgumentException( + "Parameter 'defaultValue' must not be null."); + } + if (enumClass == null) { + throw new IllegalArgumentException( + "Parameter 'enumClass' must not be null."); + } + T result; + result = defaultValue; + String text = properties.getProperty(key); + if (text != null) { + T[] enumConstants = enumClass.getEnumConstants(); + + for (int i = 0; i < enumConstants.length; i++) { + T enumConstant = enumConstants[i]; + if (enumConstant.name().equals(text)) { + result = enumConstant; + break; + } + } + } + + return result; + } + + public final <T extends Enum<?>> void writeEnum(String key, T value) { + if (key == null) { + throw new IllegalArgumentException( + "Parameter 'key' must not be null."); + } + if (value == null) { + throw new IllegalArgumentException( + "Parameter 'value' must not be null."); + } + setProperty(key, value.name()); + } + + public final int readInteger(String key, int defaultValue) { + + if (key == null) { + throw new IllegalArgumentException( + "Parameter 'key' must not be null."); + } + + int result; + String text = properties.getProperty(key); + if (text == null) { + result = defaultValue; + } else { + try { + result = Integer.parseInt(text); + } catch (NumberFormatException ex) { + result = 0; + } + } + + return result; + } + + public final void writeInteger(String key, int value) { + + if (key == null) { + throw new IllegalArgumentException( + "Parameter 'key' must not be null."); + } + + setProperty(key, String.valueOf(value)); + } + + public final void readProperties(String key, PropertiesSerializer value) { + if (key == null) { + throw new IllegalArgumentException( + "Parameter 'key' must not be null."); + } + if (value == null) { + throw new IllegalArgumentException( + "Parameter 'value' must not be null."); + } + String prefix = key + "."; + Properties valueProperties = value.getProperties(); + valueProperties.clear(); + Enumeration<Object> i = properties.keys(); + while (i.hasMoreElements()) { + String valueKey = (String) i.nextElement(); + if (valueKey.startsWith(prefix)) { + valueProperties.setProperty( + valueKey.substring(prefix.length()), + properties.getProperty(valueKey)); + } + } + } + + public final void writeProperties(String key, PropertiesSerializer value) { + if (key == null) { + throw new IllegalArgumentException( + "Parameter 'key' must not be null."); + } + if (value == null) { + throw new IllegalArgumentException( + "Parameter 'value' must not be null."); + } + String prefix = key + "."; + Properties valueProperties = value.getProperties(); + Enumeration<Object> i = valueProperties.keys(); + while (i.hasMoreElements()) { + String valueKey = (String) i.nextElement(); + setProperty(prefix + valueKey, + valueProperties.getProperty(valueKey)); + } + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/common/ResourceUtility.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/ResourceUtility.java new file mode 100644 index 00000000..641f4698 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/ResourceUtility.java @@ -0,0 +1,317 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ +package com.wudsn.ide.base.common; + +import java.io.ByteArrayInputStream; +import java.io.ByteArrayOutputStream; +import java.io.File; +import java.io.IOException; +import java.io.InputStream; +import java.io.InputStreamReader; +import java.net.URI; +import java.net.URISyntaxException; +import java.net.URL; +import java.net.URLConnection; +import java.net.URLStreamHandler; +import java.util.jar.JarFile; +import java.util.zip.ZipEntry; + +import org.eclipse.core.runtime.CoreException; + +import com.wudsn.ide.base.BasePlugin; + +/** + * Utility class to access resources in the class path. + * + * @author Peter Dell + */ +public final class ResourceUtility { + + public interface ResourceModifier { + public byte[] modifyResource(URL url, byte[] data); + } + + private final static class ResourceURLStreamHandler extends URLStreamHandler { + + ResourceModifier resourceModifier; + + /** + * Creates a new handler. + * + * @param resourceModifier + * The resource modifier or <code>null</code>. + */ + public ResourceURLStreamHandler(ResourceModifier resourceModifier) { + this.resourceModifier = resourceModifier; + } + + @Override + protected URLConnection openConnection(URL url) throws IOException { + + return new URLConnection(url) { + + @Override + public void connect() throws IOException { + } + + @Override + public String getContentType() { + return "text/html"; + } + + @Override + public InputStream getInputStream() throws IOException { + byte[] data; + data = null; + + // Handle "jar" protocol. + if (url.getProtocol().equals("jar")) { + + // Extract the "file:/... " part from the + // "jar:file:/..." + // URL. + String path = url.getPath(); + + // Check if there's a ".jar" available. + URI jarURI = ClassPathUtility.getJarURI(); + if (jarURI != null) { + + // If yes, strip the path prefix to determine the + // relative path. + String jarURIString = jarURI.toString(); + if (path.startsWith(jarURIString)) { + path = path.substring(jarURIString.length() + 2); + } + } + data = ResourceUtility.loadResourceAsByteArray(path); + if (data == null) { + throw new IOException("No resource found with path '" + path + "' found."); + } + + } // Handle "file:" protocol. + else if (url.getProtocol().equals("file")) { + File file; + try { + file = new File(url.toURI()); + data = FileUtility.readBytes(file, FileUtility.MAX_SIZE_UNLIMITED, false); + + } catch (URISyntaxException ex) { + // ignore, not found + } catch (CoreException ex) { + throw new IOException(ex.getMessage()); + } + } + + if (data == null) { + data = ("Invalid URL: " + url).getBytes(); + } else { + if (resourceModifier != null) { + data = resourceModifier.modifyResource(url, data); + if (data == null) { + data = ("Resource modified returned null for URL: " + url).getBytes(); + } + + } + } + return new ByteArrayInputStream(data); + + } + }; + + } + } + + private final static class JarEntryInputStream extends InputStream { + private JarFile jar; + private InputStream zipEntryInputStream; + + public JarEntryInputStream(File jarFile, String path) throws IOException { + if (jarFile == null) { + throw new IllegalArgumentException("Parameter 'jarFile' must not be null."); + } + if (path == null) { + throw new IllegalArgumentException("Parameter 'path' must not be null."); + } + + jar = new JarFile(jarFile); + ZipEntry zipEntry = jar.getEntry(path); + zipEntryInputStream = jar.getInputStream(zipEntry); + } + + @Override + public int read() throws IOException { + return zipEntryInputStream.read(); + } + + @Override + public void close() throws IOException { + try { + zipEntryInputStream.close(); + } catch (IOException ex) { + throw (ex); + } finally { + try { + jar.close(); + } catch (IOException ex) { + throw (ex); + } + } + } + } + + /** + * Creates a new handler. + * + * @param resourceModifier + * The resource modifier or <code>null</code>. + * @return The handler, not <code>null</code>. + */ + public static URLStreamHandler createStreamHandler(ResourceModifier resourceModifier) { + return new ResourceURLStreamHandler(resourceModifier); + + } + + /** + * Creation is private, + */ + private ResourceUtility() { + + } + + /** + * Self implemented logic to bypass the bug described in <a + * href="http://bugs.sun.com/view_bug.do?bug_id=4523159">JDK-4523159 : + * getResourceAsStream on jars in path with "!"</a>. + * + * @param path + * The path of the resource to load, not <code>null</code>. + * @return The input stream or <code>null</code> if the source was not + * found. + */ + private static InputStream getInputStream(String path) { + if (path == null) { + throw new IllegalArgumentException("Parameter 'path' must not be null."); + } + // If there is no loader, the program was launched using the Java + // boot class path and the system class loader must be used. + ClassLoader loader = ResourceUtility.class.getClassLoader(); + URL url = (loader == null) ? ClassLoader.getSystemResource(path) : loader.getResource(path); + InputStream result = null; + try { + if (url != null) { + try { + result = url.openStream(); + } catch (IOException ignore) { + } + if (result == null) { + File jarFile = ClassPathUtility.getJarFile(); + + if (jarFile != null) { + result = new JarEntryInputStream(jarFile, path); + + } + } + } + } catch (IOException ex) { + BasePlugin.getInstance().logError("Cannot get input stream for path '{0}'", new Object[] { path }, ex); + } + return result; + } + + /** + * Loads a resource as byte array. + * + * @param path + * The resource path, not empty, not <code>null</code>. + * @return The binary resource content or <code>null</code> if the resource + * was not found. + */ + public static byte[] loadResourceAsByteArray(String path) { + if (path == null) { + throw new IllegalArgumentException("Parameter 'path' must not be null."); + } + if (StringUtility.isEmpty(path)) { + throw new IllegalArgumentException("Parameter 'path' must not be empty."); + } + InputStream inputStream = getInputStream(path); + if (inputStream == null) { + return null; + } + ByteArrayOutputStream outputStream = new ByteArrayOutputStream(); + try { + byte[] buffer = new byte[8192]; + int actualLength; + while ((actualLength = inputStream.read(buffer, 0, buffer.length)) != -1) { + outputStream.write(buffer, 0, actualLength); + } + + } catch (IOException ex) { + BasePlugin.getInstance().logError("Cannot load resource '{0}'.", new Object[] { path }, ex); + } finally { + + try { + inputStream.close(); + } catch (IOException ignore) { + } + } + return outputStream.toByteArray(); + } + + /** + * Loads a resource as string. + * + * @param path + * The resource path, not empty, not <code>null</code>. + * @return The resource content or <code>null</code> if the resource was not + * found. + */ + public static String loadResourceAsString(String path) { + if (path == null) { + throw new IllegalArgumentException("Parameter 'path' must not be null."); + } + if (StringUtility.isEmpty(path)) { + throw new IllegalArgumentException("Parameter 'path' must not be empty."); + } + final InputStream inputStream = getInputStream(path); + if (inputStream == null) { + return null; + } + StringBuilder builder = new StringBuilder(); + try { + InputStreamReader reader = new InputStreamReader(inputStream); + char[] buffer = new char[8192]; + int actualLength; + while ((actualLength = reader.read(buffer, 0, buffer.length)) != -1) { + builder.append(buffer, 0, actualLength); + } + reader.close(); + + } catch (IOException ex) { + BasePlugin.getInstance().logError("Cannot load resource '{0}'.", new Object[] { path }, ex); + } finally { + + try { + inputStream.close(); + } catch (IOException ignore) { + } + } + return builder.toString(); + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/common/RunnableWithLogging.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/RunnableWithLogging.java new file mode 100644 index 00000000..b333b100 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/RunnableWithLogging.java @@ -0,0 +1,45 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.common; + +import com.wudsn.ide.base.BasePlugin; + +/** + * Wrapper class which ensures that all exceptions in runnables are logged. + * + * @author Peter Dell + */ +public abstract class RunnableWithLogging implements Runnable { + + @Override + public final void run() { + try { + runWithLogging(); + + } catch (Throwable th) { + BasePlugin.getInstance().logError( + "Runnable '{0}' ended with exception.", + new Object[] { getClass().getName() }, th); + } + } + + protected abstract void runWithLogging(); + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/common/SequencedProperties.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/SequencedProperties.java new file mode 100644 index 00000000..c97ed890 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/SequencedProperties.java @@ -0,0 +1,80 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ +package com.wudsn.ide.base.common; + +import java.util.ArrayList; +import java.util.Collections; +import java.util.Enumeration; +import java.util.List; +import java.util.Map; +import java.util.Properties; + +/** + * Properties which keep their sequence of keys. + * + * @author Peter Dell + */ +public final class SequencedProperties extends Properties { + /** + * Not used. + */ + private static final long serialVersionUID = 1L; + + private List<Object> propertyNames; + + public SequencedProperties() { + propertyNames = new ArrayList<Object>(); + } + + @Override + public synchronized Object put(Object key, Object value) { + if (propertyNames.contains(key)) { + throw new IllegalArgumentException("Value for key '" + key + + "' already added."); + } + propertyNames.add(key); + return super.put(key, value); + } + + /** + * Returns an enumeration of the keys in this hashtable. + * + * @return an enumeration of the keys in this hashtable. + * @see Enumeration + * @see #elements() + * @see #keySet() + * @see Map + */ + @Override + public synchronized Enumeration<Object> keys() { + return Collections.enumeration(propertyNames); + } + + @Override + public synchronized String toString() { + StringBuilder builder = new StringBuilder(); + for (Object key : propertyNames) { + + builder.append(key).append("=").append(get(key)).append("\n"); + } + return builder.toString(); + + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/common/StreamsProxy.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/StreamsProxy.java new file mode 100644 index 00000000..e6dd8881 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/StreamsProxy.java @@ -0,0 +1,100 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ +package com.wudsn.ide.base.common; + +import java.io.IOException; +import java.io.PrintStream; + +/** + * Wrapper around the input and output stream of a process. The stream a + * redirected to the corresponding {@link InputStreamMonitor} and + * {@link OutputStreamMonitor} + * + * @author Peter Dell + */ +final class StreamsProxy { + + private OutputStreamMonitor outputMonitor; + private OutputStreamMonitor errorMonitor; + private InputStreamMonitor inputMonitor; + private boolean closed; + + public StreamsProxy(Process process, String encoding, PrintStream out, PrintStream err) { + if (process == null) { + throw new IllegalArgumentException( + "Parameter 'process' must not be null"); + } + if (out == null) { + throw new IllegalArgumentException("Parameter 'out' must not be null."); + } + if (err == null) { + throw new IllegalArgumentException("Parameter 'err' must not be null."); + } + outputMonitor = new OutputStreamMonitor(process.getInputStream(), + encoding, out); + errorMonitor = new OutputStreamMonitor(process.getErrorStream(), + encoding, err); + inputMonitor = new InputStreamMonitor(process.getOutputStream()); + outputMonitor.startMonitoring(); + errorMonitor.startMonitoring(); + inputMonitor.startMonitoring(); + + } + + public void close() { + if (!closed) { + closed = true; + outputMonitor.close(); + errorMonitor.close(); + inputMonitor.close(); + } + } + + public void kill() { + closed = true; + outputMonitor.kill(); + errorMonitor.kill(); + inputMonitor.close(); + } + + public OutputStreamMonitor getErrorStreamMonitor() { + return errorMonitor; + } + + public OutputStreamMonitor getOutputStreamMonitor() { + return outputMonitor; + } + + public void write(String input) throws IOException { + if (!closed) { + inputMonitor.write(input); + } else { + throw new IOException("StreamsProxy alsready closed"); + } + } + + public void closeInputStream() throws IOException { + if (!closed) { + inputMonitor.closeInputStream(); + } else { + throw new IOException("StreamsProxy alsready closed"); + } + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/common/StringUtility.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/StringUtility.java new file mode 100644 index 00000000..bbaed395 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/StringUtility.java @@ -0,0 +1,70 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.common; + +/** + * Utility class to handle strings. + * + * @author Peter Dell + * + */ +public final class StringUtility { + + /** + * Creation is private. + */ + private StringUtility() { + + } + + /** + * Determines if a string value is empty, i.e. has zero length or is only + * containing white spaces. + * + * @param value + * The string value, not <code>null</code>. + * @return <code>true</code> if the value is empty or only containing of + * white spaces, <code>false</code> otherwise. + */ + public static boolean isEmpty(String value) { + if (value == null) { + throw new IllegalArgumentException( + "Parameter 'value' must not be null."); + } + return value.trim().length() == 0; + } + + /** + * Determines if a string value is specified, i.e. not empty and not only + * containing white spaces. + * + * @param value + * The string value, not <code>null</code>. + * @return <code>true</code> if the value is not empty and not only + * containing of white spaces, <code>false</code> otherwise. + */ + public static boolean isSpecified(String value) { + if (value == null) { + throw new IllegalArgumentException( + "Parameter 'value' must not be null."); + } + return value.trim().length() > 0; + } +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/common/TextUtility.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/TextUtility.java new file mode 100644 index 00000000..c8249e18 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/common/TextUtility.java @@ -0,0 +1,71 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.common; + + +/** + * Utility class for processing text. + * + * @author Peter Dell + */ +public final class TextUtility { + + /** + * Parameter variable tokens. + */ + private static final String[] PARAMETERS = { "{0}", "{1}", "{2}", "{3}", + "{4}", "{5}", "{6}", "{7}", "{8}", "{9}" }; + + /** + * Creation is private. + */ + private TextUtility() { + + } + + /** + * Formats a text with parameters "{0}" to "{9}". + * + * @param text + * The text with the parameters "{0}" to "{9}", may be empty, not + * <code>null</code>. + * @param parameters + * The parameters, may be empty or <code>null</code>. + * + * @return The formatted text, may be empty, not <code>null</code>. + */ + public static String format(String text, String... parameters) { + if (text == null) { + throw new IllegalArgumentException( + "Parameter 'text' must not be null."); + } + if (parameters == null) { + parameters = new String[0]; + } + for (int i = 0; i < parameters.length; i++) { + String parameter = parameters[i]; + if (parameter == null) { + parameter = ""; + } + text = text.replace(PARAMETERS[i], parameter); + } + return text; + } +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/CommonOpenEditorCommandHandler.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/CommonOpenEditorCommandHandler.java new file mode 100644 index 00000000..16547f8c --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/CommonOpenEditorCommandHandler.java @@ -0,0 +1,151 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor; + +import java.util.ArrayList; +import java.util.Iterator; +import java.util.List; + +import org.eclipse.core.commands.AbstractHandler; +import org.eclipse.core.commands.ExecutionEvent; +import org.eclipse.core.commands.ExecutionException; +import org.eclipse.core.resources.IFile; +import org.eclipse.jface.dialogs.MessageDialog; +import org.eclipse.jface.viewers.ISelection; +import org.eclipse.jface.viewers.IStructuredSelection; +import org.eclipse.ui.IEditorPart; +import org.eclipse.ui.IFileEditorInput; +import org.eclipse.ui.IWorkbenchPage; +import org.eclipse.ui.IWorkbenchWindow; +import org.eclipse.ui.PartInitException; +import org.eclipse.ui.PlatformUI; +import org.eclipse.ui.handlers.HandlerUtil; +import org.eclipse.ui.ide.IDE; +import org.eclipse.ui.part.FileEditorInput; + +import com.wudsn.ide.base.Texts; +import com.wudsn.ide.base.common.StringUtility; + +/** + * The action to open an editor from the context menu. The editor will become + * the new default editor for the resources on which the action is executed + * successfully. + * + * @author Peter Dell + */ +public abstract class CommonOpenEditorCommandHandler extends AbstractHandler { + + private final String editorId; + + /** + * Creation is protected, is only call by sub-classes. + * + * @param editorId + * The editor id, not empty and not <code>null</code>. + */ + protected CommonOpenEditorCommandHandler(String editorId) { + if (editorId == null) { + throw new IllegalArgumentException( + "Parameter 'editorId' must not be null."); + } + if (StringUtility.isEmpty(editorId)) { + throw new IllegalArgumentException( + "Parameter 'editorId' must not be empty."); + } + this.editorId = editorId; + } + + @Override + public final Object execute(ExecutionEvent event) throws ExecutionException { + List<IFile> files = new ArrayList<IFile>(3); + ISelection menuSelection; + menuSelection = HandlerUtil.getActiveMenuSelection(event); + ISelection menuEditorInputSelection; + menuEditorInputSelection = HandlerUtil.getActiveMenuEditorInput(event); + + if (menuSelection instanceof IStructuredSelection) { + Iterator<?> i = ((IStructuredSelection) menuSelection).iterator(); + while (i.hasNext()) { + Object object = i.next(); + openFile(files, object); + + } + } else if (menuEditorInputSelection instanceof IStructuredSelection) { + Iterator<?> i = ((IStructuredSelection) menuEditorInputSelection) + .iterator(); + while (i.hasNext()) { + Object object = i.next(); + if (object instanceof IFileEditorInput) { + IFileEditorInput fileEditorInput = (IFileEditorInput) object; + openFile(files, fileEditorInput.getFile()); + } + } + } + + return null; + } + + /** + * Opens the specified file on the editor. + * + * @param files + * The modifiable list of files already opened, not + * <code>null</code>. + * @param object + * The IFile object or <code>null</code>. + */ + private void openFile(List<IFile> files, Object object) { + + if (files == null) { + throw new IllegalArgumentException( + "Parameter 'files' must not be null."); + } + if (object instanceof IFile) { + + IFile file = (IFile) object; + if (!files.contains(file)) { + files.add(file); + + IWorkbenchWindow workbenchWindow = PlatformUI.getWorkbench() + .getActiveWorkbenchWindow(); + IWorkbenchPage page = workbenchWindow.getActivePage(); + + // Open an editor on the new file. + try { + FileEditorInput editorInput = new FileEditorInput(file); + IEditorPart editor = page.findEditor(editorInput); + if (editor != null) { + String id = editor.getEditorSite().getId(); + if (!id.equals(editorId)) { + if (!page.closeEditor(editor, true)) { + return; + } + } + } + IDE.setDefaultEditor(file, editorId); + IDE.openEditor(page, editorInput, editorId); + } catch (PartInitException exception) { + MessageDialog.openError(workbenchWindow.getShell(), + Texts.DIALOG_TITLE, exception.getMessage()); + } + } + } + } +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/CommonOpenFolderCommandHandler.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/CommonOpenFolderCommandHandler.java new file mode 100644 index 00000000..d4da9417 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/CommonOpenFolderCommandHandler.java @@ -0,0 +1,154 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor; + +import java.io.File; +import java.util.ArrayList; +import java.util.Iterator; +import java.util.List; + +import org.eclipse.core.commands.AbstractHandler; +import org.eclipse.core.commands.ExecutionEvent; +import org.eclipse.core.commands.ExecutionException; +import org.eclipse.core.resources.IResource; +import org.eclipse.core.runtime.IAdaptable; +import org.eclipse.core.runtime.IPath; +import org.eclipse.jface.viewers.ISelection; +import org.eclipse.jface.viewers.IStructuredSelection; +import org.eclipse.swt.program.Program; +import org.eclipse.ui.IFileEditorInput; +import org.eclipse.ui.handlers.HandlerUtil; + +import com.wudsn.ide.base.common.FileUtility; + +/** + * Event handler for the "Open Folder" context menu command to start a given + * editor. + * + * @author Peter Dell + */ +public final class CommonOpenFolderCommandHandler extends AbstractHandler { + + @Override + public Object execute(ExecutionEvent event) throws ExecutionException { + List<String> folderPaths = new ArrayList<String>(3); + + ISelection menuSelection; + menuSelection = HandlerUtil.getActiveMenuSelection(event); + ISelection menuEditorInputSelection; + menuEditorInputSelection = HandlerUtil.getActiveMenuEditorInput(event); + + if (menuSelection instanceof IStructuredSelection) { + Iterator<?> i = ((IStructuredSelection) menuSelection).iterator(); + while (i.hasNext()) { + Object object = i.next(); + openFolder(folderPaths, object); + + } + } else if (menuEditorInputSelection instanceof IStructuredSelection) { + Iterator<?> i = ((IStructuredSelection) menuEditorInputSelection).iterator(); + while (i.hasNext()) { + Object object = i.next(); + if (object instanceof IFileEditorInput) { + IFileEditorInput fileEditorInput = (IFileEditorInput) object; + openFolder(folderPaths, fileEditorInput.getFile()); + } + } + } + + return null; + } + + /** + * Open the folder for an object (resource). + * + * @param folderPaths + * The modifiable list of folders already opened, not + * <code>null</code>. + * @param object + * The object of which the folder shall be opened or + * <code>null</code>. + */ + private void openFolder(List<String> folderPaths, Object object) { + if (folderPaths == null) { + throw new IllegalArgumentException("Parameter 'folderPaths' must not be null."); + } + String folderPath; + + if (object instanceof IAdaptable && !(object instanceof IResource || object instanceof File)) { + IAdaptable adapter = (IAdaptable) object; + object = adapter.getAdapter(IResource.class); + if (object == null) { + object = adapter.getAdapter(File.class); + } + } + + if (object instanceof IResource) { + IResource resource = (IResource) object; + + IPath path = resource.getRawLocation(); + if (path == null) { + path = resource.getLocation(); + } + folderPath = getFolderPath(path); + + } else if (object instanceof File) { + File file = (File) object; + + if (file.isDirectory()) { + folderPath = FileUtility.getCanonicalFile(file).getPath(); + } else { + folderPath = FileUtility.getCanonicalFile(file.getParentFile()).getPath(); + } + + } else { + folderPath = null; + } + + if (folderPath != null && !folderPaths.contains(folderPath)) { + folderPaths.add(folderPath); + Program.launch(folderPath); + } + } + + /** + * Converts an IPath to a real OS path. + * + * @param path + * The path or <code>null</code>. + * @return The folder path, may be empty or <code>null</code>. + */ + private String getFolderPath(IPath path) { + String folderPath; + if (path != null) { + String resourcePath = path.toOSString(); + File file = FileUtility.getCanonicalFile(new File(resourcePath)); + if (file.isDirectory()) { + folderPath = file.getPath(); + } else { + folderPath = file.getParentFile().getPath(); + } + } else { + folderPath = null; + } + return folderPath; + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/Hardware.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/Hardware.java new file mode 100644 index 00000000..be17c367 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/Hardware.java @@ -0,0 +1,31 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ +package com.wudsn.ide.base.editor.hex; + +/** + * Enum for the supported hardware platforms. Used for file content modes and + * character sets. + * + * @author Peter Dell + * + */ +public enum Hardware { + + GENERIC, ATARI8BIT, C64 +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditor.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditor.java new file mode 100644 index 00000000..78f47a71 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditor.java @@ -0,0 +1,648 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.hex; + +import java.io.File; +import java.util.Iterator; + +import org.eclipse.core.resources.IFile; +import org.eclipse.core.runtime.CoreException; +import org.eclipse.core.runtime.IPath; +import org.eclipse.core.runtime.IProgressMonitor; +import org.eclipse.core.runtime.IStatus; +import org.eclipse.jface.action.MenuManager; +import org.eclipse.jface.viewers.ISelection; +import org.eclipse.jface.viewers.ISelectionChangedListener; +import org.eclipse.jface.viewers.ISelectionProvider; +import org.eclipse.jface.viewers.IStructuredSelection; +import org.eclipse.jface.viewers.StyledString; +import org.eclipse.swt.SWT; +import org.eclipse.swt.custom.StyledText; +import org.eclipse.swt.events.FocusEvent; +import org.eclipse.swt.events.FocusListener; +import org.eclipse.swt.events.KeyEvent; +import org.eclipse.swt.events.KeyListener; +import org.eclipse.swt.graphics.Point; +import org.eclipse.swt.layout.FillLayout; +import org.eclipse.swt.layout.GridData; +import org.eclipse.swt.layout.GridLayout; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Menu; +import org.eclipse.ui.IEditorInput; +import org.eclipse.ui.IEditorSite; +import org.eclipse.ui.IFileEditorInput; +import org.eclipse.ui.IPathEditorInput; +import org.eclipse.ui.PartInitException; +import org.eclipse.ui.part.EditorPart; +import org.eclipse.ui.views.contentoutline.IContentOutlinePage; + +import com.wudsn.ide.base.BasePlugin; +import com.wudsn.ide.base.Texts; +import com.wudsn.ide.base.common.ByteArrayUtility; +import com.wudsn.ide.base.common.FileUtility; +import com.wudsn.ide.base.common.HexUtility; +import com.wudsn.ide.base.common.NumberUtility; +import com.wudsn.ide.base.common.Profiler; +import com.wudsn.ide.base.common.TextUtility; +import com.wudsn.ide.base.gui.Action; +import com.wudsn.ide.base.gui.Application; +import com.wudsn.ide.base.gui.EnumField; +import com.wudsn.ide.base.gui.IntegerField; +import com.wudsn.ide.base.gui.MessageManager; +import com.wudsn.ide.base.gui.SWTFactory; +import com.wudsn.ide.base.gui.TextField; + +/** + * The Hex Editor. This editor offset and outline view for the block of the file + * and a context menu for copying the current selection clip board in different + * formats. + * + * TODO Complete copy & paste, complete documentation + * + * @author Peter Dell + */ +public final class HexEditor extends EditorPart implements ISelectionProvider, + Application { + + private static final String LABEL_SUFFIX = ": "; + + public final class MessageIds { + + /** + * Creation is private. + */ + private MessageIds() { + } + + public static final int FILE_CONTENT_MODE = 1; + public static final int CHARACTER_SET = 2; + public static final int BYTES_PER_LINE = 3; + } + + public final class Actions { + + /** + * Creation is private. + */ + private Actions() { + } + + public static final int FILE_CONTENT_MODE_CHANGED = 1000; + public static final int CHARACTER_SET_TYPE_CHANGED = 1001; + public static final int BYTES_PER_ROW_CHANGED = 1002; + } + + public static final String ID = "com.wudsn.ide.base.editor.hex.HexEditor"; + + private static final String CONTEXT_MENU_ID = "#HexEditorContext"; + private static final long MAX_FILE_SIZE = 8 * ByteArrayUtility.MB; + + private MessageManager messageManager; + private HexEditorParserComponent parserComponent; + + // Editor content outline page. + private HexEditorContentOutlinePage contentOutlinePage; + + // Editor header area. + private TextField fileContentSizeField; + private EnumField<HexEditorFileContentMode> fileContentModeField; + private EnumField<HexEditorCharacterSet> characterSetField; + private IntegerField bytesPerRowField; + + private StyledText textField; + + // File source. + private IFile iFile; + private File ioFile; + + /** + * This main method is for testing the speed of the source file parser + * component only. + * + * @param args + * Not used, not <code>null</code>. + * @throws Exception + * If anything goes terribly wrong. + */ + public static void main(String[] args) throws Exception { + + // Initialize for stand alone usage. + new BasePlugin().start(null); + + HexEditorParserComponent parser = new HexEditorParserComponent( + new MessageManager(new HexEditor())); + parser.setFileContent(new byte[100000]); + parser.determinePossibleFileContentModes(); + + long startTimeMillis = System.currentTimeMillis(); + parser.setFileContentMode(HexEditorFileContentMode.BINARY); + parser.parseFileContent(); + long duration = System.currentTimeMillis() - startTimeMillis; + System.out.println(duration); + System.exit(0); + + } + + /** + * Creation is public. Called by extension point + * "org.eclipse.ui.popupMenus". + */ + public HexEditor() { + super(); + + messageManager = new MessageManager(this); + + parserComponent = new HexEditorParserComponent(messageManager); + + } + + /** + * @see org.eclipse.ui.IEditorPart#init(IEditorSite, IEditorInput) + */ + @Override + public void init(IEditorSite site, IEditorInput input) + throws PartInitException { + setSite(site); + setInput(input); + + try { + load(); + } catch (CoreException ex) { + BasePlugin.getInstance().showError(site.getShell(), + ex.getMessage(), ex); + } + } + + /** + * @see org.eclipse.ui.IWorkbenchPart#createPartControl(Composite) + */ + @Override + public void createPartControl(Composite parent) { + + getSite().setSelectionProvider(this); + + GridLayout gridLayout = new GridLayout(1, true); + gridLayout.marginWidth = 0; + parent.setLayout(gridLayout); + GridData gd = new GridData(GridData.VERTICAL_ALIGN_BEGINNING); + gd.horizontalSpan = 1; + parent.setLayoutData(gd); + + Composite header = SWTFactory.createComposite(parent, 8, 1, + GridData.FILL_HORIZONTAL); + FillLayout fillLayout = new FillLayout(SWT.HORIZONTAL); + fillLayout.marginWidth = 10; + header.setLayout(fillLayout); + + fileContentSizeField = new TextField(header, + Texts.HEX_EDITOR_FILE_CONTENT_SIZE_FIELD_LABEL + LABEL_SUFFIX, + SWT.READ_ONLY); + fileContentSizeField.getLabel().setAlignment(SWT.RIGHT); + + fileContentModeField = new EnumField<HexEditorFileContentMode>(header, + Texts.HEX_EDITOR_FILE_CONTENT_MODE_FIELD_LABEL + LABEL_SUFFIX, + HexEditorFileContentMode.class, null); + fileContentModeField.getLabel().setAlignment(SWT.RIGHT); + + messageManager.registerField(fileContentModeField, + MessageIds.FILE_CONTENT_MODE); + fileContentModeField.addSelectionAction(new Action( + Actions.FILE_CONTENT_MODE_CHANGED, this)); + + characterSetField = new EnumField<HexEditorCharacterSet>(header, + Texts.HEX_EDITOR_CHARACTER_SET_TYPE_FIELD_LABEL + LABEL_SUFFIX, + HexEditorCharacterSet.class, null); + characterSetField.getLabel().setAlignment(SWT.RIGHT); + messageManager.registerField(characterSetField, + MessageIds.CHARACTER_SET); + characterSetField.addSelectionAction(new Action( + Actions.CHARACTER_SET_TYPE_CHANGED, this)); + + bytesPerRowField = new IntegerField(header, + Texts.HEX_EDITOR_BYTES_PER_ROW_FIELD_LABEL + LABEL_SUFFIX, + null, false, 1, SWT.NONE); + bytesPerRowField.getLabel().setAlignment(SWT.RIGHT); + messageManager.registerField(characterSetField, + MessageIds.BYTES_PER_LINE); + bytesPerRowField.getControl().addKeyListener(new KeyListener() { + + @Override + public void keyReleased(KeyEvent e) { + if (e.keyCode == '\r') { + performAction(new Action(Actions.BYTES_PER_ROW_CHANGED, + HexEditor.this)); + } + + } + + @Override + public void keyPressed(KeyEvent e) { + + } + }); + bytesPerRowField.getControl().addFocusListener(new FocusListener() { + + @Override + public void focusLost(FocusEvent e) { + performAction(new Action(Actions.BYTES_PER_ROW_CHANGED, + HexEditor.this)); + + } + + @Override + public void focusGained(FocusEvent e) { + } + }); + + // SWT.WRAP is very slow, so it's not used. + textField = new StyledText(parent, SWT.SCROLL_LINE | SWT.V_SCROLL + | SWT.H_SCROLL | SWT.READ_ONLY); + gd = new GridData(GridData.FILL_VERTICAL | GridData.FILL_HORIZONTAL); + gd.horizontalIndent = 0; + textField.setLayoutData(gd); + textField.setIndent(10); + textField.setLineSpacing(0); + + // Create a menu manager for the context menu. + MenuManager manager = new MenuManager(CONTEXT_MENU_ID, CONTEXT_MENU_ID); + manager.setRemoveAllWhenShown(true); + + // Create menu and link to the field. + Menu textContextMenu = manager.createContextMenu(textField); + textField.setMenu(textContextMenu); + + getEditorSite().registerContextMenu(CONTEXT_MENU_ID, manager, this, + false); + messageManager.clearMessages(); + dataToUi(); + } + + @Override + public Object getAdapter(@SuppressWarnings("rawtypes") Class adapter) { + if (adapter != null && IContentOutlinePage.class.equals(adapter)) { + if (contentOutlinePage == null) { + contentOutlinePage = new HexEditorContentOutlinePage(this); + + contentOutlinePage.setInput(parserComponent.getOutlineBlocks()); + } + + return contentOutlinePage; + } + return super.getAdapter(adapter); + } + + /** + * @see org.eclipse.ui.IWorkbenchPart#setFocus() + */ + @Override + public void setFocus() { + textField.setFocus(); + } + + private void load() throws CoreException { + + // Clear fields. + ioFile = null; + iFile = null; + + String fileName = ""; + IEditorInput input = getEditorInput(); + if (input instanceof IFileEditorInput) { + // Input file found in Eclipse Workspace. + iFile = ((IFileEditorInput) input).getFile(); + ioFile = iFile.getRawLocation().toFile(); + fileName = iFile.getName(); + } else if (input instanceof IPathEditorInput) { + // Input file is outside the Eclipse Workspace + IPathEditorInput pathEditorInput = (IPathEditorInput) input; + IPath path = pathEditorInput.getPath(); + ioFile = path.toFile(); + fileName = ioFile.getName(); + + } else { + // Not supported. + } + + byte[] fileContent; + Profiler profiler = new Profiler(this); + profiler.begin("readBytes", fileName); + if (ioFile != null) { + fileContent = FileUtility.readBytes(ioFile, MAX_FILE_SIZE, false); + } else if (iFile != null) { + fileContent = FileUtility.readBytes(iFile, MAX_FILE_SIZE, false); + } else { + fileContent = new byte[0]; + } + profiler.end("readBytes"); + + // Set the content, determine the default file content mode and + // character set. + parserComponent.setFileContent(fileContent); + HexEditorFileContentMode defaultFileContentMode = parserComponent + .determinePossibleFileContentModes(); + HexEditorCharacterSet defaultCharacterSet = HexEditorCharacterSet + .getDefaultCharacterSet(defaultFileContentMode); + parserComponent.setFileContentMode(defaultFileContentMode); + parserComponent.setCharacterSet(defaultCharacterSet); + + setPartName(fileName); + + } + + public int getBytesPerRow() { + return parserComponent.getBytesPerRow(); + } + + @Override + public void addSelectionChangedListener(ISelectionChangedListener listener) { + // Nothing. + } + + @Override + public void removeSelectionChangedListener( + ISelectionChangedListener listener) { + // Nothing. + } + + @Override + public HexEditorSelection getSelection() { + + Point selection = textField.getSelection(); + + // if (selection.x == selection.y) { + // return null; + // } + + // BasePlugin.getInstance().log( + // "HexEditor selection.x={0} selection.y={1}", + // new Object[] { String.valueOf(selection.x), + // String.valueOf(selection.y) }); + + return parserComponent.getSelection(selection.x, selection.y); + } + + @Override + public void setSelection(ISelection selection) { + // Single range selection? + if (selection instanceof HexEditorSelection) { + HexEditorSelection hexEditorSelection = (HexEditorSelection) selection; + int textStartOffset = parserComponent + .getByteTextOffset(hexEditorSelection.getStartOffset()); + int textEndOffset = parserComponent + .getByteTextOffset(hexEditorSelection.getEndOffset()); + setSelectionOffsets(textStartOffset, textEndOffset); + // Range of outline tree objects? + } else if (selection instanceof IStructuredSelection) { + IStructuredSelection structuredSelection = (IStructuredSelection) selection; + + if (structuredSelection.getFirstElement() instanceof HexEditorContentOutlineTreeObject) { + Iterator<?> i = ((IStructuredSelection) selection).iterator(); + + int textStartOffset = Integer.MAX_VALUE; + int textEndOffset = Integer.MIN_VALUE; + while (i.hasNext()) { + HexEditorContentOutlineTreeObject treeObject = (HexEditorContentOutlineTreeObject) i + .next(); + textStartOffset = Math.min(treeObject.getTextStartOffset(), + textStartOffset); + textEndOffset = Math.max(treeObject.getTextEndOffset(), + textEndOffset); + } + setSelectionOffsets(textStartOffset, textEndOffset); + } + + } + setFocus(); + } + + private void setSelectionOffsets(int textStartOffset, int textEndOffset) { + try { + // Mark complete selection area. This also scrolls to + // the end of the area. + textField.setSelection(new Point(textStartOffset, textEndOffset)); + // + // // But we want to see start of the selection are, so + // // position explicitly. + textField.setTopIndex(textField.getContent().getLineAtOffset( + textStartOffset)); + } catch (IllegalArgumentException x) { + // Ignore + } + } + + /** + * Gets the file path for saving the current selection. Called by + * {@link HexEditorSaveSelectionAsCommandHandler }. + * + * @return The file path, not <code>null</code>. + */ + public String getSelectionSaveFilePath() { + String result = "Selection"; + String extension = ".bin"; + if (parserComponent.getFileContentMode().equals( + HexEditorFileContentMode.ATARI_DISK_IMAGE_K_FILE)) { + extension = ".xex"; + } + + if (ioFile != null) { + result = ioFile.getAbsolutePath(); + int index = result.lastIndexOf('.'); + if (index >= 0) { + result = result.substring(0, index); + } + } + + result += extension; + + return result; + } + + @Override + public boolean isDirty() { + return false; + } + + @Override + public void doSave(IProgressMonitor monitor) { + // Nothing. + + } + + @Override + public boolean isSaveAsAllowed() { + return false; + } + + @Override + public void doSaveAs() { + // Nothing. + } + + /** + * {@inheritDoc} + */ + @Override + public MessageManager getMessageManager() { + return messageManager; + } + + private void dataFromUi() { + messageManager.clearMessages(); + parserComponent.setFileContentMode(fileContentModeField.getValue()); + parserComponent.setCharacterSet(characterSetField.getValue()); + int bytesPerRow = bytesPerRowField.getValue(); + if (bytesPerRow < 1) { + bytesPerRow = 16; + } else if (bytesPerRow > 256) { + bytesPerRow = 256; + } + parserComponent.setBytesPerRow(bytesPerRow); + } + + private void dataToUi() { + + // File content size. + String text = TextUtility.format( + Texts.HEX_EDITOR_FILE_CONTENT_SIZE_FIELD_TEXT, + HexUtility.getLongValueHexString(parserComponent + .getFileContent().length), NumberUtility + .getLongValueDecimalString(parserComponent + .getFileContent().length)); + fileContentSizeField.setValue(text); + + // File content mode. + fileContentModeField.setValue(parserComponent.getFileContentMode()); + + // Character set. + HexEditorCharacterSet characterSet = parserComponent.getCharacterSet(); + characterSetField.setValue(characterSet); + if (!textField.getFont().equals(characterSet.getFont())) { + textField.setFont(characterSet.getFont()); + } + + // Bytes per Row + bytesPerRowField.setValue(parserComponent.getBytesPerRow()); + + if (parserComponent.isParsingFileContentRequired()) { + StyledString styledString = parserComponent.parseFileContent(); + textField.setText(styledString.getString()); + textField.setStyleRanges(styledString.getStyleRanges()); + + if (contentOutlinePage != null) { + contentOutlinePage.setInput(parserComponent.getOutlineBlocks()); + } + } + + messageManager.displayMessages(); + } + + /** + * {@inheritDoc} + */ + @Override + public void performAction(Action action) { + try { + + ISelection oldSelection = null; + dataFromUi(); + + switch (action.getId()) { + case Actions.FILE_CONTENT_MODE_CHANGED: + case Actions.CHARACTER_SET_TYPE_CHANGED: + case Actions.BYTES_PER_ROW_CHANGED: + oldSelection = getSelection(); + break; + } + + dataToUi(); + if (oldSelection != null) { + setSelection(oldSelection); + } + } catch (Exception ex) { + BasePlugin.getInstance().showError(getSite().getShell(), + "Error in update()", ex); + } + + } + + /** + * Called by {@link HexEditorClipboardCommandHandler}. + * + * @param bytes + * The byte array to be pasted, may be empty, not + * <code>null</code>. + * + * TODO Hex paste is not working yet + */ + final void pasteFromClipboard(byte[] bytes) { + if (bytes == null) { + throw new IllegalArgumentException( + "Parameter 'bytes' must not be null."); + } + HexEditorSelection selection = getSelection(); + byte[] newFileContent; + + // If there is no end offset, we insert the new bytes. + if (selection.getEndOffset() != HexEditorParserComponent.UNDEFINED_OFFSET) { + int selectionLength = selection.getEndOffset() + - selection.getStartOffset() + 1; + newFileContent = new byte[parserComponent.getFileContent().length + - selectionLength + bytes.length]; + System.arraycopy(parserComponent.getFileContent(), 0, + newFileContent, 0, selection.getStartOffset()); + System.arraycopy(bytes, 0, newFileContent, + selection.getStartOffset(), bytes.length); + int length = parserComponent.getFileContent().length + - selection.getEndOffset() - 1; + if (length > 0) { + // TODO Hex paste is not working yet + System.arraycopy(parserComponent.getFileContent(), + selection.getEndOffset(), + + newFileContent, selection.getStartOffset() + + bytes.length, length); + } + messageManager + .sendMessage( + 0, + IStatus.OK, + "${0} ({1}) bytes pasted from clipboard to replace ${2} ({3}) bytes ", + HexUtility.getLongValueHexString(bytes.length), + NumberUtility + .getLongValueDecimalString(bytes.length), + HexUtility.getLongValueHexString(selectionLength), + NumberUtility + .getLongValueDecimalString(selectionLength)); + } else { + // If there is and end offset, we replace the selection with the new + // bytes. + newFileContent = parserComponent.getFileContent(); + messageManager.sendMessage(0, IStatus.OK, + "${0} ({1}) bytes inserted from clipboard", + HexUtility.getLongValueHexString(bytes.length), + NumberUtility.getLongValueDecimalString(bytes.length)); + } + + parserComponent.setFileContent(newFileContent); + dataToUi(); + + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorCharacterSet.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorCharacterSet.java new file mode 100644 index 00000000..3b166a6c --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorCharacterSet.java @@ -0,0 +1,270 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.hex; + +import java.io.File; +import java.util.Map; +import java.util.TreeMap; + +import org.eclipse.jface.resource.JFaceResources; +import org.eclipse.swt.SWT; +import org.eclipse.swt.graphics.Device; +import org.eclipse.swt.graphics.Font; +import org.eclipse.swt.widgets.Display; + +import com.wudsn.ide.base.BasePlugin; +import com.wudsn.ide.base.common.FileUtility; +import com.wudsn.ide.base.common.ResourceUtility; + +/** + * Logical character set with physical font and character mapping. + * + * @since 1.7.0 + */ +enum HexEditorCharacterSet { + ASCII, ATARI_ATASCII, ATARI_ATASCII_SCREEN_CODE, ATARI_INTERNATIONAL, ATARI_INTERNATIONAL_SCREEN_CODE, C64_PETSCII_UPPER_CASE, C64_PETSCII_LOWER_CASE; + + /** + * Data class to encapsulate lazy loading and reuse of SWT fonts. + */ + private final static class Data { + + // Atari and C64 font by Mark Simonson, marksim@bitstream.net, + // http://www2.bitstream.net/~marksim/atarimac + private final static String ATARI_FONT_PATH = "fonts/atari8/AtariClassic-Regular.ttf"; + private final static String ATARI_FONT_NAME = "Atari Classic"; + private final static String ATARI_INT_FONT_PATH = "fonts/atari8/AtariClassicInt-Regular.ttf"; + private final static String ATARI_INT_FONT_NAME = "Atari Classic Int"; + private final static int ATARI_FONT_SIZE = 6; + + private final static String C64_FONT_PATH = "fonts/c64/C64Classic-Regular.ttf"; + private final static String C64_FONT_NAME = "C64 Classic"; + private final static int C64_FONT_SIZE = 6; + + private static Map<HexEditorCharacterSet, Data> instanceMap; + private static Map<String, Font> fontMap; + + Font font; + char[] characterMapping; + + static { + instanceMap = new TreeMap<HexEditorCharacterSet, Data>(); + fontMap = new TreeMap<String, Font>(); + } + + /** + * Gets a data instance base on the font type. + * + * @param type + * The font type, not <code>null</code>. + * + * @return The instance, not <code>null</code>. + */ + public static Data getInstance(HexEditorCharacterSet type) { + if (type == null) { + throw new IllegalArgumentException("Parameter 'type' must not be null."); + } + Data result; + synchronized (instanceMap) { + result = instanceMap.get(type); + // Add "|| true" below to disable caching for debugging + // purposes. + if (result == null) { + result = new Data(); + result.font = null; + result.characterMapping = new char[256]; + String fontPath = null; + String fontName = ""; + int fontSize = -1; + switch (type) { + case ASCII: + for (int i = 0; i < 256; i++) { + // 7-bit ASCII + int charValue = i & 0x7f; + // Convert control characters to "." + if (charValue < 0x20) { + charValue = '.'; + } + result.characterMapping[i] = (char) charValue; + } + break; + case ATARI_ATASCII: + fontPath = ATARI_FONT_PATH; + fontName = ATARI_FONT_NAME; + fontSize = ATARI_FONT_SIZE; + result.setIdentityMapping(); + break; + case ATARI_ATASCII_SCREEN_CODE: + fontPath = ATARI_FONT_PATH; + fontName = ATARI_FONT_NAME; + fontSize = ATARI_FONT_SIZE; + result.setAtariScreenCodeMapping(); + break; + case ATARI_INTERNATIONAL: + fontPath = ATARI_INT_FONT_PATH; + fontName = ATARI_INT_FONT_NAME; + fontSize = ATARI_FONT_SIZE; + result.setIdentityMapping(); + break; + case ATARI_INTERNATIONAL_SCREEN_CODE: + fontPath = ATARI_INT_FONT_PATH; + fontName = ATARI_INT_FONT_NAME; + fontSize = ATARI_FONT_SIZE; + result.setAtariScreenCodeMapping(); + break; + case C64_PETSCII_UPPER_CASE: + fontPath = C64_FONT_PATH; + fontName = C64_FONT_NAME; + fontSize = C64_FONT_SIZE; + result.setIdentityMapping(); + break; + case C64_PETSCII_LOWER_CASE: + fontPath = C64_FONT_PATH; + fontName = C64_FONT_NAME; + fontSize = C64_FONT_SIZE; + for (int i = 0; i < 256; i++) { + result.characterMapping[i] = (char) (0x200 + i); + } + break; + default: + throw new IllegalArgumentException("Unsupported font type " + type + "."); + } + + if (fontPath != null) { + // Check if temp file is already cached? + result.font = fontMap.get(fontPath); + if (result.font == null) { + File file = null; + try { + file = File.createTempFile("Data", null); + byte[] content = ResourceUtility.loadResourceAsByteArray(fontPath); + FileUtility.writeBytes(file, content); + } catch (Exception ex) { + BasePlugin.getInstance().logError( + "Error while copying font data of font '{0}' to temporary file.", + new Object[] { fontPath }, ex); + if (file != null) { + file.delete(); + } + file = null; + } + + // If temp file is present, + if (file != null) { + Device device = Display.getDefault(); + String absolutePath = file.getAbsolutePath(); + if (device.loadFont(absolutePath)) { + result.font = new Font(device, fontName, fontSize, SWT.NORMAL); + // Make sure the file is kept until the + // process + // ends. + file.deleteOnExit(); + fontMap.put(fontPath, result.font); + + } else { + // Loading failed, so no need to keep the + // file. + file.delete(); + } + } + } + } + + if (result.font == null) { + result.font = JFaceResources.getTextFont(); + } + instanceMap.put(type, result); + } + } + return result; + } + + private void setIdentityMapping() { + for (int i = 0; i < 256; i++) { + characterMapping[i] = (char) (0x100 + i); + } + } + + private void setAtariScreenCodeMapping() { + for (int i = 0; i < 256; i++) { + int charValue = i & 0x7f; + if (charValue < 0x40) { + charValue = charValue + 0x20; + } else if (charValue < 0x60) { + charValue = charValue - 0x40; + } + if (i > 0x80) { + charValue |= 0x80; + } + characterMapping[i] = (char) (0x100 + charValue); + } + } + + /** + * Creation is private. + */ + private Data() { + } + + } + + /** + * Determines the default character set for a given file content mode. + * + * @param fileContentMode + * The file content mode, not <code>null</code>. + * @return The default character set, not <code>null</code>. + */ + public static HexEditorCharacterSet getDefaultCharacterSet(HexEditorFileContentMode fileContentMode) { + if (fileContentMode == null) { + throw new IllegalArgumentException("Parameter 'fileContentMode' must not be null."); + } + switch (fileContentMode.getHardware()) { + case GENERIC: + return ASCII; + case ATARI8BIT: + return ATARI_ATASCII; + case C64: + return C64_PETSCII_UPPER_CASE; + } + throw new IllegalArgumentException("File content mode " + fileContentMode + " has an unknown hardware " + + fileContentMode.getHardware()); + } + + /** + * Gets the SWT font. + * + * @return The SWT font, not <code>null</code>. + */ + public Font getFont() { + return Data.getInstance(this).font; + } + + /** + * Gets the character mapping. + * + * @return The character mapping as an array of 256 char values, not + * <code>null</code>. + */ + public char[] getCharacterMapping() { + return Data.getInstance(this).characterMapping; + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorClipboardCommandHandler.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorClipboardCommandHandler.java new file mode 100644 index 00000000..a9dd3e44 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorClipboardCommandHandler.java @@ -0,0 +1,184 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.hex; + +import org.eclipse.core.commands.ExecutionException; +import org.eclipse.core.runtime.IStatus; +import org.eclipse.swt.dnd.Clipboard; +import org.eclipse.swt.dnd.TextTransfer; +import org.eclipse.swt.dnd.Transfer; +import org.eclipse.ui.IWorkbenchSite; +import org.eclipse.ui.handlers.HandlerUtil; + +import com.wudsn.ide.base.Texts; +import com.wudsn.ide.base.common.HexUtility; +import com.wudsn.ide.base.common.NumberUtility; + +/** + * The handler for the commands to copy the selection to the clipboard. + * + * @author Peter Dell + */ +public final class HexEditorClipboardCommandHandler extends HexEditorSelectionCommandHandler { + + public static final class CommandIds { + + private CommandIds() { + } + + public static final String COPY = "com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardCommand"; + public static final String COPY_AS_HEX_VALUES = "com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsHexValuesCommand"; + public static final String COPY_AS_DECIMAL_VALUES = "com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsDecimalValuesCommand"; + public static final String COPY_AS_DECIMAL_VALUES_BLOCK = "com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsDecimalValuesBlockCommand"; + public static final String COPY_AS_ASCII_STRING = "com.wudsn.ide.base.editor.hex.HexEditorCopyToClipboardAsASCIIStringCommand"; + public static final String PASTE = "com.wudsn.ide.base.editor.hex.HexEditorPasteFromClipboardCommand"; + } + + /** + * Creation is public. Called by extension point "org.eclipse.ui.handlers". + */ + public HexEditorClipboardCommandHandler() { + super(); + } + + @Override + protected void performAction() throws ExecutionException { + + byte[] bytes; + bytes = hexEditorSelection.getBytes(); + StringBuilder builder = new StringBuilder(5 * bytes.length); + String lineSeparator = System.getProperty("line.separator"); + + Object[] data; + Transfer[] transfers; + + int bytesPerRow = hexEditor.getBytesPerRow(); + if (commandId.equals(CommandIds.COPY) && !hexEditorSelection.isEmpty()) { + data = new Object[] { hexEditorSelection }; + transfers = new Transfer[] { HexEditorSelectionTransfer.getInstance() }; + copyToClipboard(bytes, data, transfers); + + } else if ((commandId.equals(CommandIds.COPY_AS_HEX_VALUES) + || commandId.equals(CommandIds.COPY_AS_DECIMAL_VALUES) + || commandId.equals(CommandIds.COPY_AS_DECIMAL_VALUES_BLOCK) || commandId + .equals(CommandIds.COPY_AS_ASCII_STRING)) && !hexEditorSelection.isEmpty()) { + if (commandId.equals(CommandIds.COPY_AS_HEX_VALUES)) { + builder.append(".byte "); + for (int i = 0; i < bytes.length; i++) { + builder.append("$"); + builder.append(HexUtility.getByteValueHexString(bytes[i] & 0xff)); + if ((i + 1) % bytesPerRow == 0) { + builder.append(lineSeparator); + if (i < bytes.length - 1) { + builder.append(".byte "); + } + } else { + if (i < bytes.length - 1) { + builder.append(','); + } + } + } + } else if (commandId.equals(CommandIds.COPY_AS_DECIMAL_VALUES) + || commandId.equals(CommandIds.COPY_AS_DECIMAL_VALUES_BLOCK)) { + // In block mode, decimals are aligned to 3 digits. + boolean block = commandId.equals(CommandIds.COPY_AS_DECIMAL_VALUES_BLOCK); + builder.append(".byte "); + for (int i = 0; i < bytes.length; i++) { + int b = bytes[i] & 0xff; + if (block) { + if (b < 10) { + builder.append(" "); + } else if (b < 100) { + builder.append(' '); + } + } + builder.append(Integer.toString(b)); + if ((i + 1) % bytesPerRow == 0) { + builder.append(lineSeparator); + if (i < bytes.length - 1) { + builder.append(".byte "); + } + } else { + if (i < bytes.length - 1) { + builder.append(','); + } + } + } + } else if (commandId.equals(CommandIds.COPY_AS_ASCII_STRING)) { + for (int i = 0; i < bytes.length; i++) { + char c = (char) (bytes[i] & 0xff); + builder.append(c); + } + } else { + throw new IllegalArgumentException("Unknown command '" + commandId + "'."); + } + data = new Object[] { builder.toString(), hexEditorSelection }; + transfers = new Transfer[] { TextTransfer.getInstance(), HexEditorSelectionTransfer.getInstance() }; + copyToClipboard(bytes, data, transfers); + + } else if (commandId.equals(CommandIds.PASTE)) { + pasteFromClipboard(); + } + + } + + private void copyToClipboard(byte[] bytes, Object[] data, Transfer[] transfers) throws ExecutionException { + if (bytes == null) { + throw new IllegalArgumentException("Parameter 'bytes' must not be null."); + } + if (data == null) { + throw new IllegalArgumentException("Parameter 'data' must not be null."); + } + if (transfers == null) { + throw new IllegalArgumentException("Parameter 'transfers' must not be null."); + } + IWorkbenchSite site = HandlerUtil.getActiveSiteChecked(event); + Clipboard clipboard = new Clipboard(site.getShell().getDisplay()); + try { + + clipboard.setContents(data, transfers); + + } finally { + clipboard.dispose(); + } + + // INFO: ${0} ({1}) bytes copied to clipboard. + messageManager.sendMessage(0, IStatus.OK, Texts.MESSAGE_I302, HexUtility.getLongValueHexString(bytes.length), + NumberUtility.getLongValueDecimalString(bytes.length)); + } + + private void pasteFromClipboard() throws ExecutionException { + + IWorkbenchSite site = HandlerUtil.getActiveSiteChecked(event); + Clipboard clipboard = new Clipboard(site.getShell().getDisplay()); + try { + + Object data = clipboard.getContents(HexEditorSelectionTransfer.getInstance()); + if (data != null) { + byte[] bytes = ((HexEditorSelection) data).getBytes(); + hexEditor.pasteFromClipboard(bytes); + + } + } finally { + clipboard.dispose(); + } + + } +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorContentOutlineLabelProvider.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorContentOutlineLabelProvider.java new file mode 100644 index 00000000..6b0feec7 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorContentOutlineLabelProvider.java @@ -0,0 +1,86 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.hex; + +import org.eclipse.jface.viewers.DelegatingStyledCellLabelProvider; +import org.eclipse.jface.viewers.LabelProvider; +import org.eclipse.jface.viewers.StyledString; +import org.eclipse.swt.graphics.Image; + +import com.wudsn.ide.base.BasePlugin; + +/** + * Label provider for the tree objects in the outline page. + * + * @author Peter Dell + */ +final class HexEditorContentOutlineLabelProvider extends + DelegatingStyledCellLabelProvider { + + /** Outline segment image */ + private final Image segmentImage; + + private static class HexEditorStyledLabelProvider extends LabelProvider + implements IStyledLabelProvider { + + /** + * Creation is local. + */ + HexEditorStyledLabelProvider() { + + } + + @Override + public StyledString getStyledText(Object element) { + if (element == null) { + throw new IllegalArgumentException( + "Parameter 'element' must not be null."); + } + HexEditorContentOutlineTreeObject treeObject; + treeObject = (HexEditorContentOutlineTreeObject) element; + return treeObject.getStyledString(); + } + } + + /** + * Creates a new instance. + * + * Called by + * {@link HexEditorContentOutlinePage#createControl(org.eclipse.swt.widgets.Composite)} + * . + */ + HexEditorContentOutlineLabelProvider() { + super(new HexEditorStyledLabelProvider()); + BasePlugin plugin; + plugin = BasePlugin.getInstance(); + segmentImage = plugin + .getImage("hex-editor-segment-16x16.gif"); + } + + @Override + public Image getImage(Object element) { + Image result; + + result = segmentImage; + + return result; + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorContentOutlinePage.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorContentOutlinePage.java new file mode 100644 index 00000000..0fd57cb8 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorContentOutlinePage.java @@ -0,0 +1,88 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.hex; + +import org.eclipse.jface.resource.JFaceResources; +import org.eclipse.jface.viewers.AbstractTreeViewer; +import org.eclipse.jface.viewers.ISelection; +import org.eclipse.jface.viewers.SelectionChangedEvent; +import org.eclipse.jface.viewers.TreeViewer; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.ui.views.contentoutline.ContentOutlinePage; + +/** + * Content outline page for the hex editor. + * + * @author Peter Dell + */ +final class HexEditorContentOutlinePage extends ContentOutlinePage { + + private HexEditor editor; + private Object input; + + HexEditorContentOutlinePage(HexEditor editor) { + if (editor == null) { + throw new IllegalArgumentException( + "Parameter 'editor' must not be null."); + } + this.editor = editor; + + } + + @Override + public void createControl(Composite parent) { + super.createControl(parent); + + TreeViewer viewer = getTreeViewer(); + viewer.getControl().setFont(JFaceResources.getTextFont()); + viewer + .setContentProvider(new HexEditorContentOutlineTreeContentProvider()); + viewer.setLabelProvider(new HexEditorContentOutlineLabelProvider()); + viewer.addSelectionChangedListener(this); + viewer.setAutoExpandLevel(AbstractTreeViewer.ALL_LEVELS); + + updateTreeView(); + } + + @Override + public void selectionChanged(SelectionChangedEvent event) { + super.selectionChanged(event); + + ISelection selection = event.getSelection(); + editor.setSelection(selection); + } + + void setInput(Object input) { + if (input == null) { + throw new IllegalArgumentException( + "Parameter 'input' must not be null."); + } + this.input = input; + updateTreeView(); + } + + private void updateTreeView() { + TreeViewer viewer = getTreeViewer(); + if (viewer != null) { + viewer.setInput(input); + } + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorContentOutlineTreeContentProvider.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorContentOutlineTreeContentProvider.java new file mode 100644 index 00000000..b2182a6c --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorContentOutlineTreeContentProvider.java @@ -0,0 +1,94 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.hex; + +import java.util.List; + +import org.eclipse.jface.viewers.ITreeContentProvider; +import org.eclipse.jface.viewers.Viewer; + +/** + * Tree content provider to {@link HexEditorContentOutlinePage}. + * + * @author Peter Dell + */ +final class HexEditorContentOutlineTreeContentProvider implements + ITreeContentProvider { + + HexEditorContentOutlineTreeContentProvider() { + } + + /** + * {@inheritDoc} + */ + @Override + public Object[] getChildren(Object parentElement) { + return null; + } + + /** + * {@inheritDoc} + */ + @Override + public Object getParent(Object element) { + + return null; + } + + /** + * {@inheritDoc} + */ + @Override + public boolean hasChildren(Object element) { + return false; + } + + /** + * {@inheritDoc} + */ + @Override + public Object[] getElements(Object inputElement) { + Object[] result; + if (inputElement instanceof List<?>) { + result = ((List<?>) inputElement).toArray(); + } else { + result = null; + } + + return result; + } + + /** + * {@inheritDoc} + */ + @Override + public void dispose() { + + } + + /** + * {@inheritDoc} + */ + @Override + public void inputChanged(Viewer viewer, Object oldInput, Object newInput) { + + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorContentOutlineTreeObject.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorContentOutlineTreeObject.java new file mode 100644 index 00000000..65842ab2 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorContentOutlineTreeObject.java @@ -0,0 +1,148 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.hex; + +import org.eclipse.jface.viewers.StyledString; + +/** + * The object representing a node in the hex editor content outline tree. + * + * @author Peter Dell + */ +public final class HexEditorContentOutlineTreeObject { + + private final StyledString styledString; + private int fileStartOffset; + private int textStartOffset; + private int fileEndOffset; + private int textEndOffset; + + /** + * Create a new instance. + * + * @param styledString + * The styled string of the instance, may be empty not + * <code>null</code>. + */ + public HexEditorContentOutlineTreeObject(StyledString styledString) { + if (styledString == null) { + throw new IllegalArgumentException( + "Parameter 'styledString' must not be null."); + } + this.styledString = new StyledString().append(styledString); + } + + /** + * Gets the styled string of the object. + * + * @return The styled string, not <code>null</code>. + */ + public StyledString getStyledString() { + return styledString; + } + + /** + * Gets the start offset of the tree object in the file. + * + * @return The start offset, a non-negative integer. + */ + public int getFileStartOffset() { + return fileStartOffset; + } + + /** + * Sets the start offset of the tree object in the file. + * + * @param fileOffset + * The start offset, a non-negative integer or <code>-1</code> if the offset is not defined. + */ + public void setFileStartOffset(int fileOffset) { + + this.fileStartOffset = fileOffset; + } + + /** + * Gets the end offset of the tree object in the file. + * + * @return The end offset, a non-negative integer or <code>-1</code> if the offset is not defined. + */ + public int getFileEndOffset() { + return fileEndOffset; + } + + /** + * Sets the end offset of the tree object in the file or <code>-1</code> if the offset is not defined. + * + * @param fileOffset + * The end offset, a non-negative integer or <code>-1</code> if the offset is not defined. + */ + public void setFileEndOffset(int fileOffset) { + + this.fileEndOffset = fileOffset; + } + + /** + * Gets the start offset of the tree object in the text. + * + * @return The offset, a non-negative integer. + */ + public int getTextStartOffset() { + return textStartOffset; + } + + /** + * Sets text start offset of the tree object in the text + * + * @param textOffset + * The offset, a non-negative integer. + */ + public void setTextStartOffset(int textOffset) { + if (textOffset < 0) { + throw new IllegalArgumentException( + "Parameter 'textOffset' must not be negative. Specified value is " + + textOffset + "."); + } + this.textStartOffset = textOffset; + } + + /** + * Gets the end offset of the tree object in the text. + * + * @return The offset, a non-negative integer. + */ + public int getTextEndOffset() { + return textEndOffset; + } + + /** + * Sets text end offset of the tree object in the text + * + * @param textOffset + * The offset, a non-negative integer. + */ + public void setTextEndOffset(int textOffset) { + if (textOffset < 0) { + throw new IllegalArgumentException( + "Parameter 'textOffset' must not be negative. Specified value is " + + textOffset + "."); + } + this.textEndOffset = textOffset; + } +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorFileContentMode.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorFileContentMode.java new file mode 100644 index 00000000..6cdba4ca --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorFileContentMode.java @@ -0,0 +1,40 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.hex; + +enum HexEditorFileContentMode { + + BINARY(Hardware.GENERIC), ATARI_COM_FILE(Hardware.ATARI8BIT), ATARI_DISK_IMAGE(Hardware.ATARI8BIT), ATARI_DISK_IMAGE_K_FILE( + Hardware.ATARI8BIT), ATARI_MADS_FILE(Hardware.ATARI8BIT), ATARI_SDX_FILE(Hardware.ATARI8BIT), ATARI_SAP_FILE( + Hardware.ATARI8BIT), C64_PRG_FILE(Hardware.C64); + + private Hardware hardware; + + private HexEditorFileContentMode(Hardware hardware) { + if (hardware == null) { + throw new IllegalArgumentException("Parameter 'hardware' must not be null."); + } + this.hardware = hardware; + } + + public Hardware getHardware() { + return hardware; + } +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorOpenCommandHandler.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorOpenCommandHandler.java new file mode 100644 index 00000000..c7e834bb --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorOpenCommandHandler.java @@ -0,0 +1,39 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.hex; + +import com.wudsn.ide.base.editor.CommonOpenEditorCommandHandler; + + +/** + * The action to open the hex editor from the context menu. + * + * @author Peter Dell + */ +public final class HexEditorOpenCommandHandler extends + CommonOpenEditorCommandHandler { + + /** + * Creation is public. Called by extension "org.eclipse.ui.popupMenus". + */ + public HexEditorOpenCommandHandler() { + super(HexEditor.ID); + } +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorParser.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorParser.java new file mode 100644 index 00000000..1d2314ca --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorParser.java @@ -0,0 +1,226 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ +package com.wudsn.ide.base.editor.hex; + +import org.eclipse.jface.viewers.StyledString; +import org.eclipse.jface.viewers.StyledString.Styler; + +import com.wudsn.ide.base.common.HexUtility; +import com.wudsn.ide.base.common.NumberUtility; +import com.wudsn.ide.base.common.StringUtility; +import com.wudsn.ide.base.common.TextUtility; + +public abstract class HexEditorParser { + + private HexEditorParserComponent owner; + protected Styler offsetStyler; + protected Styler addressStyler; + + /** + * Creation is protected. + */ + protected HexEditorParser() { + + } + + /** + * Initialized by owner. + * + * @param owner + * The owner, not <code>null</code>. + * @param offsetStyler + * The offset styler, not <code>null</code>. + * @param addressStyler + * The address styler, not <code>null</code>. + */ + void init(HexEditorParserComponent owner, Styler offsetStyler, Styler addressStyler) { + if (owner == null) { + throw new IllegalArgumentException("Parameter 'owner' must not be null."); + } + if (offsetStyler == null) { + throw new IllegalArgumentException("Parameter 'offsetStyler' must not be null."); + } + if (addressStyler == null) { + throw new IllegalArgumentException("Parameter 'offsetStyler' must not be null."); + } + this.owner = owner; + this.offsetStyler = offsetStyler; + this.addressStyler = addressStyler; + } + + /** + * Public API for parsing. + * + * @param contentBuilder + * The content builder, not <code>null</code>. + * @return <code>true</code> if parsing was OK, <code>false</code>otherwise. + */ + public abstract boolean parse(StyledString contentBuilder); + + /** + * Gets the length of the file content. + * + * @return The length of the file content, a non-negative integer. + */ + protected final int getFileContentLength() { + return owner.getFileContent().length; + } + + /** + * Gets a byte from the file content. + * + * @param offset + * The offset, a non-negative integer. + * @return The byte from the file content. + */ + protected final int getFileContentByte(int offset) { + return owner.getFileContentByte(offset); + } + + /** + * Gets a word from the file content. + * + * @param offset + * The offset, a non-negative integer. + * @return The word from the file content. + */ + protected final int getFileContentWord(int offset) { + return owner.getFileContentWord(offset); + } + + /** + * Prints a block header in the context area and adds a block to the + * outline. + * + * @param contentBuilder + * The content builder, not <code>null</code>. + * @param blockHeaderText + * The header text for the block, may be empty, not + * <code>null</code>. + * @param blockHeaderNumber + * The block count or <code>-1</code> if count shall not be + * displayed. + * @param blockHeaderParameterText + * The pattern text of the form "{0}-{1} ({2})" + * @param offset + * The start offset, a non-negative integer. + * @param startAddress + * The start address, a non-negative integer. + * @param endAddress + * The end address, a non-negative integer. + * + * @return The tree object representing the block. + */ + protected final HexEditorContentOutlineTreeObject printBlockHeader(StyledString contentBuilder, + String blockHeaderText, int blockHeaderNumber, String blockHeaderParameterText, int offset, + int startAddress, int endAddress) { + + if (contentBuilder == null) { + throw new IllegalArgumentException("Parameter 'contentBuilder' must not be null."); + } + int blockLength = endAddress - startAddress + 1; + String blockHeaderNumberText; + if (blockHeaderNumber >= 0) { + blockHeaderNumberText = NumberUtility.getLongValueDecimalString(blockHeaderNumber); + } else { + blockHeaderNumberText = ""; + } + int length = Math.max(4, HexUtility.getLongValueHexLength(endAddress)); + String hexText = TextUtility.format(blockHeaderParameterText, + HexUtility.getLongValueHexString(startAddress, length), + HexUtility.getLongValueHexString(endAddress, length), + HexUtility.getLongValueHexString(blockLength, length)); + + String decimalText = TextUtility.format(blockHeaderParameterText, + NumberUtility.getLongValueDecimalString(startAddress), + NumberUtility.getLongValueDecimalString(endAddress), + NumberUtility.getLongValueDecimalString(blockLength)); + + StyledString styledString; + styledString = new StyledString(); + styledString.append(blockHeaderText, offsetStyler); + if (blockHeaderNumber >= 0) { + styledString.append(" "); + styledString.append(blockHeaderNumberText, offsetStyler); + + } + if (StringUtility.isSpecified(blockHeaderParameterText)) { + styledString.append(" : "); + styledString.append(hexText, addressStyler); + styledString.append(" : "); + styledString.append(decimalText); + } + + contentBuilder.append(blockHeaderText, offsetStyler); + if (blockHeaderNumber >= 0) { + contentBuilder.append(" "); + contentBuilder.append(blockHeaderNumberText, offsetStyler); + } + contentBuilder.append("\n"); + return owner.printBlockHeader( contentBuilder, styledString,offset); + } + + /** + * Prints a block header in the context area and adds a block to the + * outline. + * + * @param contentBuilder + * The content builder, not <code>null</code>. + * @param headerStyledString + * The style string for the block header in the outline, not + * <code>null</code>. + * @param offset + * The start offset, a non-negative integer. + * + * @return The tree object representing the block. + */ + protected final HexEditorContentOutlineTreeObject printBlockHeader(StyledString contentBuilder, + StyledString headerStyledString, int offset) { + return owner.printBlockHeader(contentBuilder, headerStyledString, offset); + } + + /** + * Prints the last block in case if contains an error like the wrong number + * of bytes. + * + * @param contentBuilder + * The content builder, not <code>null</code>. + * @param errorText + * The error text, not empty and not <code>null</code>. + * @param length + * The length of the last block, a non-negative integer. + * @param offset + * The offset of the last block, a non-negative integer. + */ + protected final void printBlockWithError(StyledString contentBuilder, String errorText, int length, int offset) { + owner.printBlockWithError(contentBuilder, errorText, length, offset); + } + + protected final void skipByteTextIndex(int offset) { + owner.skipByteTextIndex(offset); + + } + + protected final int printBytes(HexEditorContentOutlineTreeObject treeObject, StyledString contentBuilder, + int offset, int maxOffset, boolean withStartAddress, int startAddress) { + return owner.printBytes(treeObject, contentBuilder, offset, maxOffset, withStartAddress, startAddress); + + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorParserComponent.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorParserComponent.java new file mode 100644 index 00000000..4b089099 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorParserComponent.java @@ -0,0 +1,727 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ +package com.wudsn.ide.base.editor.hex; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; + +import org.eclipse.core.runtime.IStatus; +import org.eclipse.jface.preference.JFacePreferences; +import org.eclipse.jface.viewers.StyledString; +import org.eclipse.jface.viewers.StyledString.Styler; +import org.eclipse.swt.graphics.TextStyle; + +import com.wudsn.ide.base.Texts; +import com.wudsn.ide.base.common.EnumUtility; +import com.wudsn.ide.base.common.HexUtility; +import com.wudsn.ide.base.common.NumberUtility; +import com.wudsn.ide.base.common.Profiler; +import com.wudsn.ide.base.common.TextUtility; +import com.wudsn.ide.base.editor.hex.HexEditor.MessageIds; +import com.wudsn.ide.base.editor.hex.parser.AtariCOMParser; +import com.wudsn.ide.base.editor.hex.parser.AtariDiskImageKFileParser; +import com.wudsn.ide.base.editor.hex.parser.AtariDiskImageParser; +import com.wudsn.ide.base.editor.hex.parser.AtariMADSParser; +import com.wudsn.ide.base.editor.hex.parser.AtariParser; +import com.wudsn.ide.base.editor.hex.parser.AtariSAPParser; +import com.wudsn.ide.base.editor.hex.parser.AtariSDXParser; +import com.wudsn.ide.base.editor.hex.parser.BinaryParser; +import com.wudsn.ide.base.editor.hex.parser.C64PRGParser; +import com.wudsn.ide.base.gui.MessageManager; + +final class HexEditorParserComponent { + + public static final int UNDEFINED_OFFSET = -1; + private final static int BYTES_PER_ROW = 16; + private static final int INT_FF = 0xff; + + // Callback API. + private MessageManager messageManager; + + // Style components. + private Styler offsetStyler; + private Styler addressStyler; + private Styler charStyler; + private Styler errorStyler; + + // File content and state. + private boolean fileContentParsed; + private HexEditorFileContentMode fileContentMode; + private byte[] fileContent; + private int bytesPerRow; + private HexEditorCharacterSet characterSet; + + // Previous state with regards to parsing. + private HexEditorFileContentMode oldFileContentMode; + private byte[] oldFileContent; + private int oldBytesPerRow; + private HexEditorCharacterSet oldCharacterSet; + + // Parsing state. + private List<HexEditorFileContentMode> possibleFileContentModes; + private List<HexEditorContentOutlineTreeObject> outlineBlocks; + private int[] byteTextOffsets; + private int byteTextIndex; + + // Line buffers for binary to hex and char conversion. + private char[] hexChars; + private char[] hexBuffer; + private char[] charBuffer; + + public HexEditorParserComponent(MessageManager messageManager) { + if (messageManager == null) { + throw new IllegalArgumentException( + "Parameter 'messageManager' must not be null."); + } + this.messageManager = messageManager; + + // Get static stylers for the styled string. + offsetStyler = StyledString.createColorRegistryStyler( + JFacePreferences.COUNTER_COLOR, null); + addressStyler = StyledString.createColorRegistryStyler( + JFacePreferences.QUALIFIER_COLOR, null); + charStyler = StyledString.createColorRegistryStyler( + JFacePreferences.HYPERLINK_COLOR, null); + + charStyler = new Styler() { + + @Override + public void applyStyles(TextStyle textStyle) { + textStyle.font = null; + + } + + }; + + errorStyler = StyledString.createColorRegistryStyler( + JFacePreferences.ERROR_COLOR, null); + + // Initialize hex chars and normal character set type. + hexChars = new char[] { '0', '1', '2', '3', '4', '5', '6', '7', '8', + '9', 'A', 'B', 'C', 'D', 'E', 'F' }; + characterSet = HexEditorCharacterSet.ASCII; + + clear(); + + } + + private void clear() { + // Initialize with empty file. + fileContentParsed = false; + fileContentMode = HexEditorFileContentMode.BINARY; + setFileContent(new byte[0]); + characterSet = HexEditorCharacterSet.ASCII; + bytesPerRow = BYTES_PER_ROW; + + oldFileContentMode = null; + oldFileContent = null; + oldCharacterSet = null; + oldBytesPerRow = 0; + + possibleFileContentModes = new ArrayList<HexEditorFileContentMode>(); + outlineBlocks = new ArrayList<HexEditorContentOutlineTreeObject>(); + } + + public void setFileContent(byte[] fileContent) { + if (fileContent == null) { + throw new IllegalArgumentException( + "Parameter 'fileContent' must not be null."); + } + this.fileContent = fileContent; + initByteTextOffsets(); + } + + /** + * Reserve enough space for the lookup table that maps text offsets to file + * offsets. + */ + private void initByteTextOffsets() { + // Twice the space, because some format display the content twice, for + // example ATARI_DISK_IMAGE_K_FILE. + byteTextOffsets = new int[fileContent.length * 2]; + byteTextIndex = 0; + } + + /** + * Gets the current file content. + * + * @return The file content, may be empty or <code>null</code>. + */ + public byte[] getFileContent() { + return fileContent; + } + + /** + * Determines the possible file content modes based on the file content. + * + * @return The suggested default file content mode, not <code>null</code>. + */ + public HexEditorFileContentMode determinePossibleFileContentModes() { + + HexEditorFileContentMode result = HexEditorFileContentMode.BINARY; + possibleFileContentModes.clear(); + possibleFileContentModes.add(fileContentMode); + + HexEditorFileContentMode defaultMode = result; + + // COM header present? + if (fileContent.length > 6) { + // AtariDOS COM file? + if (getFileContentWord(0) == AtariParser.COM_HEADER) { + int startAddress = getFileContentWord(2); + int endAddress = getFileContentWord(4); + if (startAddress >= 0 && endAddress >= startAddress) { + defaultMode = HexEditorFileContentMode.ATARI_COM_FILE; + possibleFileContentModes.add(defaultMode); + + if (fileContent.length > 16) { + if (getFileContentWord(6) == AtariMADSParser.RELOC_HEADER) { + defaultMode = HexEditorFileContentMode.ATARI_MADS_FILE; + possibleFileContentModes.add(defaultMode); + } + } + // New default? + if (result.equals(HexEditorFileContentMode.BINARY)) { + result = defaultMode; + } + } + } // SpartaDOS X non relocatable file? + else if (getFileContentWord(0) == AtariSDXParser.NON_RELOC_HEADER) { + int startAddress = getFileContentWord(2); + int endAddress = getFileContentWord(4); + if (startAddress > 0 && endAddress >= startAddress) { + defaultMode = HexEditorFileContentMode.ATARI_SDX_FILE; + possibleFileContentModes.add(defaultMode); + // New default? + if (result.equals(HexEditorFileContentMode.BINARY)) { + result = defaultMode; + } + } + } // SpartaDOS X relocatable file? + else if (getFileContentWord(0) == AtariSDXParser.RELOC_HEADER + && fileContent.length > 8) { + int blockNumber = getFileContentByte(2); + if (blockNumber > 0) { + defaultMode = HexEditorFileContentMode.ATARI_SDX_FILE; + possibleFileContentModes.add(defaultMode); + // New default? + if (result.equals(HexEditorFileContentMode.BINARY)) { + result = defaultMode; + } + } + } + } + + // ATR header present? + if ((fileContent.length > 16 && getFileContentByte(0) == 0x96 && getFileContentByte(1) == 0x02)) { + defaultMode = HexEditorFileContentMode.ATARI_DISK_IMAGE; + possibleFileContentModes.add(defaultMode); + + // Special case of k-file (converted COM file) + int offset = AtariDiskImageKFileParser.ATARI_DISK_IMAGE_K_FILE_COM_FILE_OFFSET; + if (fileContent.length > offset + 2 + && getFileContentWord(offset) == 0xffff) { + final int[] kFileBootHeader = new int[] { 0x00, 0x03, 0x00, + 0x07, 0x14, 0x07, 0x4C, 0x14, 0x07 }; + boolean kFileBootHeaderFound = true; + for (int i = 0; i < kFileBootHeader.length; i++) { + if (getFileContentByte(16 + i) != kFileBootHeader[i]) { + kFileBootHeaderFound = false; + } + } + if (kFileBootHeaderFound) { + defaultMode = HexEditorFileContentMode.ATARI_DISK_IMAGE_K_FILE; + possibleFileContentModes.add(defaultMode); + } + } + + // New default? + if (result.equals(HexEditorFileContentMode.BINARY)) { + result = defaultMode; + } + } + + // SAP header present? + if ((fileContent.length > 11 && getFileContentByte(0) == 0x53 && getFileContentByte(1) == 0x41) + && getFileContentByte(2) == 0x50) { + possibleFileContentModes + .add(HexEditorFileContentMode.ATARI_SAP_FILE); + // New default? + if (result.equals(HexEditorFileContentMode.BINARY)) { + result = HexEditorFileContentMode.ATARI_SAP_FILE; + } + } + + // PRG header present? + if ((fileContent.length > 2 && getFileContentWord(0) + + getFileContent().length - 2 < 0x10000)) { + possibleFileContentModes.add(HexEditorFileContentMode.C64_PRG_FILE); + int loadAddress = getFileContentWord(0); + if (result.equals(HexEditorFileContentMode.BINARY) + && loadAddress >= 0x800 && loadAddress < 0x2000) { + result = HexEditorFileContentMode.C64_PRG_FILE; + } + } + return result; + } + + /** + * Sets the file content for {@link #parseFileContent()}. + * + * @param fileContentMode + * The file content mode, not <code>null</code>. + */ + public void setFileContentMode(HexEditorFileContentMode fileContentMode) { + if (fileContentMode == null) { + throw new IllegalArgumentException( + "Parameter 'fileContentMode' must not be null."); + } + this.fileContentMode = fileContentMode; + } + + /** + * Gets the file content for {@link #parseFileContent()}. + * + * @return fileContentMode The file content mode, not <code>null</code>. + */ + public HexEditorFileContentMode getFileContentMode() { + return fileContentMode; + } + + /** + * Sets the character set type. + * + * @param characterSet + * The character set type, not <code>null</code>. + */ + public void setCharacterSet(HexEditorCharacterSet characterSet) { + if (characterSet == null) { + throw new IllegalArgumentException( + "Parameter 'characterSet' must not be null."); + } + + this.characterSet = characterSet; + } + + /** + * Gets the character set type. + * + * @return characterSet The character set type, not <code>null</code>. + */ + public HexEditorCharacterSet getCharacterSet() { + return characterSet; + } + + /** + * Sets the number of bytes per row for {@link #parseFileContent()}. + * + * @param bytesPerRow + * The number of bytes per row, a positive integer. + */ + public void setBytesPerRow(int bytesPerRow) { + if (bytesPerRow < 1) { + throw new IllegalArgumentException( + "Parameter 'bytesPerRow' must not be positive. Specified valie was " + + bytesPerRow + "."); + } + this.bytesPerRow = bytesPerRow; + } + + /** + * Gets the number of bytes per row for {@link #parseFileContent()}. + * + * @return The number of bytes per row, a positive integer. + */ + public int getBytesPerRow() { + return bytesPerRow; + } + + /** + * Determines if parsing is required. + * + * @return <code>true</code> if parsing is required, <code>false</code> + * otherwise. + */ + public boolean isParsingFileContentRequired() { + return !fileContentParsed + || !Arrays.equals(fileContent, oldFileContent) + || !fileContentMode.equals(oldFileContentMode) + || !characterSet.equals(oldCharacterSet) + || bytesPerRow != oldBytesPerRow; + } + + /** + * Parse the file content set with {@link #setFileContent(byte[])} according + * to the parameters set with + * {@link #setFileContentMode(HexEditorFileContentMode)}, + * {@link #setBytesPerRow(int)} and + * {@link #setCharacterSet(HexEditorCharacterSet)}. + * + * @return The styles string representing the content. + */ + public StyledString parseFileContent() { + + Profiler profiler = new Profiler(this); + profiler.begin("parseFileContent", fileContent.length + " bytes"); + + outlineBlocks.clear(); + initByteTextOffsets(); + + StyledString contentBuilder = new StyledString(); + HexEditorContentOutlineTreeObject treeObject; + String text = TextUtility.format(Texts.HEX_EDITOR_FILE_SIZE, + HexUtility.getLongValueHexString(fileContent.length), + NumberUtility.getLongValueDecimalString(fileContent.length)); + contentBuilder.append(text); + treeObject = new HexEditorContentOutlineTreeObject(contentBuilder); + treeObject.setFileStartOffset(0); + treeObject.setTextStartOffset(contentBuilder.length()); + outlineBlocks.add(treeObject); + + contentBuilder = new StyledString(); + if (!possibleFileContentModes.contains(fileContentMode)) { + messageManager.sendMessage(MessageIds.FILE_CONTENT_MODE, + IStatus.ERROR, Texts.MESSAGE_E300, + EnumUtility.getText(fileContentMode)); + return contentBuilder; + } + + if (fileContent.length > 0) { + boolean error; + HexEditorParser parser; + if (fileContentMode.equals(HexEditorFileContentMode.ATARI_COM_FILE)) { + parser = new AtariCOMParser(); + } else if (fileContentMode + .equals(HexEditorFileContentMode.ATARI_DISK_IMAGE)) { + parser = new AtariDiskImageParser(); + } else if (fileContentMode + .equals(HexEditorFileContentMode.ATARI_DISK_IMAGE_K_FILE)) { + parser = new AtariDiskImageKFileParser(); + } else if (fileContentMode + .equals(HexEditorFileContentMode.ATARI_MADS_FILE)) { + parser = new AtariMADSParser(); + } else if (fileContentMode + .equals(HexEditorFileContentMode.ATARI_SDX_FILE)) { + parser = new AtariSDXParser(); + } else if (fileContentMode + .equals(HexEditorFileContentMode.ATARI_SAP_FILE)) { + parser = new AtariSAPParser(); + } else if (fileContentMode + .equals(HexEditorFileContentMode.C64_PRG_FILE)) { + parser = new C64PRGParser(); + } else { + parser = new BinaryParser(); + } + + // Initialize the buffers for the hex and char conversion. + hexBuffer = new char[3 + bytesPerRow * 3 + 2]; + for (int i = 0; i < hexBuffer.length; i++) { + hexBuffer[i] = ' '; + } + hexBuffer[1] = ':'; + hexBuffer[hexBuffer.length - 2] = '|'; + charBuffer = new char[bytesPerRow + 1]; + charBuffer[charBuffer.length - 1] = '\n'; + + parser.init(this, offsetStyler, addressStyler); + error = parser.parse(contentBuilder); + if (error) { + messageManager.sendMessage(MessageIds.FILE_CONTENT_MODE, + IStatus.ERROR, Texts.MESSAGE_E301, + EnumUtility.getText(fileContentMode)); + } + } + + profiler.end("parseFileContent"); + + // Copy current state to state backup for change detection in {@link + // #isParsingFileContentRequired}, + fileContentParsed = true; + oldFileContentMode = fileContentMode; + oldFileContent = fileContent; + oldCharacterSet = characterSet; + oldBytesPerRow = bytesPerRow; + + return contentBuilder; + } + + /** + * Gets a byte from the file content. + * + * @param offset + * The offset, a non-negative integer. + * @return The byte from the file content. + */ + final int getFileContentByte(int offset) { + return fileContent[offset] & INT_FF; + } + + /** + * Gets a word from the file content. + * + * @param offset + * The offset, a non-negative integer. + * @return The word from the file content. + */ + final int getFileContentWord(int offset) { + return getFileContentByte(offset) + 256 + * getFileContentByte(offset + 1); + } + + /** + * Prints a block header in the context area and adds a block to the + * outline. + * + * @param contentBuilder + * The content builder, not <code>null</code>. + * @param headerStyledString + * The style string for the block header in the outline, not + * <code>null</code>. + * @param offset + * The start offset, a non-negative integer. + * + * @return The tree object representing the block. + */ + final HexEditorContentOutlineTreeObject printBlockHeader( + StyledString contentBuilder, StyledString headerStyledString, + int offset) { + + if (contentBuilder == null) { + throw new IllegalArgumentException( + "Parameter 'contentBuilder' must not be null."); + } + if (headerStyledString == null) { + throw new IllegalArgumentException( + "Parameter 'styledString' must not be null."); + } + HexEditorContentOutlineTreeObject treeObject; + + treeObject = new HexEditorContentOutlineTreeObject(headerStyledString); + treeObject.setFileStartOffset(offset); + treeObject.setTextStartOffset(contentBuilder.length()); + outlineBlocks.add(treeObject); + return treeObject; + } + + /** + * Prints the last block in case if contains an error like the wrong number + * of bytes. + * + * @param contentBuilder + * The content builder, not <code>null</code>. + * @param errorText + * The error text, not empty and not <code>null</code>. + * @param length + * The length of the last block, a non-negative integer. + * @param offset + * The offset of the last block, a non-negative integer. + */ + final void printBlockWithError(StyledString contentBuilder, + String errorText, int length, int offset) { + if (contentBuilder == null) { + throw new IllegalArgumentException( + "Parameter 'contentBuilder' must not be null."); + } + + if (errorText == null) { + throw new IllegalArgumentException( + "Parameter 'errorText' must not be null."); + } + HexEditorContentOutlineTreeObject treeObject; + StyledString styledString = new StyledString(errorText, errorStyler); + treeObject = new HexEditorContentOutlineTreeObject(styledString); + treeObject.setFileStartOffset(UNDEFINED_OFFSET); + treeObject.setFileEndOffset(UNDEFINED_OFFSET); + treeObject.setTextStartOffset(contentBuilder.length()); + treeObject.setTextEndOffset(contentBuilder.length()); + + outlineBlocks.add(treeObject); + contentBuilder.append(styledString); + contentBuilder.append("\n"); + offset = printBytes(treeObject, contentBuilder, offset, length - 1, + true, 0); + } + + final void skipByteTextIndex(int offset) { + byteTextIndex += offset; + } + + final int printBytes(HexEditorContentOutlineTreeObject treeObject, + StyledString contentBuilder, int offset, int maxOffset, + boolean withStartAddress, int startAddress) { + + if (offset < 0) { + throw new IllegalArgumentException( + "Parameter 'offset' must not be negative, specified value is " + + offset + "."); + } + if (maxOffset < 0) { + throw new IllegalArgumentException( + "Parameter 'offset' must not be negative, specified value is " + + maxOffset + "."); + } + int length = Math.max(4, HexUtility.getLongValueHexLength(maxOffset)); + char[] characterMapping = characterSet.getCharacterMapping(); + while (offset <= maxOffset) { + int contentBuilderLineStartOffset = contentBuilder.length(); + + contentBuilder.append( + HexUtility.getLongValueHexString(offset, length), + offsetStyler); + + if (withStartAddress) { + contentBuilder.append(" : "); + contentBuilder.append( + HexUtility.getLongValueHexString(startAddress, length), + addressStyler); + + } + + // Remember byte offset where the new line starts. + int contentBuilderStartOffset = contentBuilder.length(); + int h = 3; + for (int b = 0; b < bytesPerRow; b++) { + char highChar; + char lowChar; + char charValue; + if (offset > maxOffset) { + highChar = ' '; + lowChar = ' '; + charValue = ' '; + } else { + int byteValue = getFileContentByte(offset); + highChar = hexChars[byteValue >> 4]; + lowChar = hexChars[byteValue & 0xf]; + charValue = characterMapping[byteValue]; + byteTextOffsets[byteTextIndex++] = (b == 0 ? contentBuilderLineStartOffset + : contentBuilderStartOffset + h); + offset++; + startAddress++; + } + hexBuffer[h++] = highChar; + hexBuffer[h++] = lowChar; + h++; + charBuffer[b] = charValue; + } + contentBuilder.append(hexBuffer); + contentBuilder.append(charBuffer, charStyler); + } + treeObject.setFileEndOffset(offset); + treeObject.setTextEndOffset(contentBuilder.length()); + return offset; + } + + /** + * Gets the list of outline blocks determined by {@link #parseFileContent()} + * . + * + * @return The list of outline blocks, may be empty, not <code>null</code>. + */ + public List<HexEditorContentOutlineTreeObject> getOutlineBlocks() { + return outlineBlocks; + } + + /** + * Gets the selection represented by the start and end offset in the text + * field. + * + * @param x + * is the offset of the first selected character + * @param y + * is the offset after the last selected character. + * @return The selection or <code>null</code>. + */ + public HexEditorSelection getSelection(int x, int y) { + + int startOffset = UNDEFINED_OFFSET; + int endOffset = UNDEFINED_OFFSET; + int textOffset = 0; + for (int i = 0; i < byteTextIndex + && (startOffset == UNDEFINED_OFFSET || endOffset == UNDEFINED_OFFSET); i++) { + int nextTextOffset; + if (i < byteTextIndex - 1) { + nextTextOffset = byteTextOffsets[i + 1]; + } else { + nextTextOffset = Integer.MAX_VALUE; + } + if (startOffset == UNDEFINED_OFFSET && textOffset - 1 <= x + && x < nextTextOffset - 1) { + startOffset = i; + } + if (startOffset != UNDEFINED_OFFSET + && endOffset == UNDEFINED_OFFSET && textOffset < y + && y <= nextTextOffset) { + endOffset = i; + } + textOffset = nextTextOffset; + + } + + if (startOffset == UNDEFINED_OFFSET) { + return null; + } + int length; + byte[] bytes; + + length = endOffset - startOffset + 1; + // BasePlugin.getInstance().log("HexEditor.getSelection(): startOffset={0} endoffset={1} length={2}", + // new Object[] { String.valueOf(startOffset), + // String.valueOf(endOffset), String.valueOf(length) }); + // Length not empty and selection does not cross file end boundary. + if (length > 0 && endOffset < fileContent.length) { + // Reposition into first occurrence of in the file. + // This is relevant for the format that display the content more + // than once. + bytes = new byte[length]; + // startOffset = startOffset % fileContent.length; + // endOffset = endOffset % fileContent.length; + // length = endOffset - startOffset + 1; + System.arraycopy(fileContent, startOffset, bytes, 0, length); + + // BasePlugin.getInstance().log( + // "HexEditor startOffset={0} endoffset={1} length={2}", + // new Object[] { String.valueOf(startOffset), + // String.valueOf(endOffset), String.valueOf(length) }); + + } else { + endOffset = startOffset; + bytes = new byte[0]; + } + HexEditorSelection hexEditorSelection = new HexEditorSelection( + startOffset, endOffset, bytes); + return hexEditorSelection; + } + + /** + * Gets the text offset for a byte offset. + * + * @param byteOffset + * The byte offset in the original byte array. + * @return The text offset where the byte is represented or <code>-1</code> + * if there is no such text offset. + */ + public int getByteTextOffset(int byteOffset) { + if (byteOffset < byteTextOffsets.length) { + return byteTextOffsets[byteOffset]; + } + return -1; + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorSaveSelectionAsCommandHandler.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorSaveSelectionAsCommandHandler.java new file mode 100644 index 00000000..94d68e08 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorSaveSelectionAsCommandHandler.java @@ -0,0 +1,79 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.hex; + +import java.io.File; + +import org.eclipse.core.commands.ExecutionException; +import org.eclipse.core.runtime.CoreException; +import org.eclipse.core.runtime.IStatus; +import org.eclipse.swt.SWT; +import org.eclipse.swt.widgets.FileDialog; +import org.eclipse.swt.widgets.Shell; +import org.eclipse.ui.handlers.HandlerUtil; + +import com.wudsn.ide.base.Texts; +import com.wudsn.ide.base.common.FileUtility; +import com.wudsn.ide.base.common.HexUtility; +import com.wudsn.ide.base.common.NumberUtility; +import com.wudsn.ide.base.common.TextUtility; + +public final class HexEditorSaveSelectionAsCommandHandler extends HexEditorSelectionCommandHandler { + + public static final class CommandIds { + + private CommandIds() { + } + + public static final String SAVE_SELECTION_AS = "com.wudsn.ide.base.editor.hex.HexEditorSaveSelectionAsCommand"; + } + + @Override + protected void performAction() throws ExecutionException { + if (commandId.equals(CommandIds.SAVE_SELECTION_AS) && !hexEditorSelection.isEmpty()) { + Shell shell = HandlerUtil.getActiveShell(event); + if (shell == null) { + return; + } + byte[] content = hexEditorSelection.getBytes(); + + FileDialog dialog = new FileDialog(shell, SWT.SAVE); + int length = content.length; + String hexLength = HexUtility.getLongValueHexString(length); + String decimalLength = NumberUtility.getLongValueDecimalString(length); + dialog.setText(TextUtility + .format(Texts.HEX_EDITOR_SAVE_SELECTION_AS_DIALOG_TITLE, hexLength, decimalLength)); + dialog.setFileName(hexEditor.getSelectionSaveFilePath()); + String filePath = dialog.open(); + if (filePath != null) { + try { + FileUtility.writeBytes(new File(filePath), content); + // INFO: ${0} ({1}) bytes saved as '{2}'. + hexEditor.getMessageManager().sendMessage(0, IStatus.OK, Texts.MESSAGE_I303, hexLength, + decimalLength, filePath); + } catch (CoreException ex) { + throw new ExecutionException(ex.getMessage()); + } + } + } + + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorSelection.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorSelection.java new file mode 100644 index 00000000..94ff11c2 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorSelection.java @@ -0,0 +1,106 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.hex; + +import org.eclipse.jface.viewers.ISelection; + +/** + * Container class for selections in the hex editor. + * + * @author Peter Dell + * + */ +final class HexEditorSelection implements ISelection { + + private int startOffset; + private int endOffset; + private byte[] bytes; + + /** + * Creates a new selection. + * + * @param startOffset + * The start offset in the original array, a non-negative number. + * @param endOffset + * The end offset in the original array, a non-negative number + * greater or equal to the start offset. + * @param bytes + * The content of the selection, may be empty, not + * <code>null</code>. + */ + public HexEditorSelection(int startOffset, int endOffset, byte[] bytes) { + + if (startOffset < 0) { + throw new IllegalArgumentException("Parameter 'startOffset' must not be negative, specified value is " + + startOffset + "."); + } + if (endOffset < startOffset) { + throw new IllegalArgumentException("Parameter 'endOffset' must not be smaller than startOffset " + + startOffset + ", specified value is " + endOffset + "."); + } + if (bytes == null) { + throw new IllegalArgumentException("Parameter 'bytes' must not be null."); + } + this.startOffset = startOffset; + this.endOffset = endOffset; + this.bytes = bytes; + } + + @Override + public boolean isEmpty() { + return bytes.length == 0; + } + + /** + * Gets the start offset of the selection in the original array. + * + * @return The start offset in the original array, a non-negative number. + * + */ + public int getStartOffset() { + return startOffset; + } + + /** + * Gets the end offset in the original array. + * + * @return The end offset in the original array, a non-negative number + * greater or equal to the start offset. + */ + public int getEndOffset() { + return endOffset; + } + + /** + * Gets the content of the selection. + * + * @return The content of the selection, may be empty, not <code>null</code> + * . + */ + public byte[] getBytes() { + return bytes; + } + + @Override + public String toString() { + return "HexEditorSelection from " + startOffset + " to " + endOffset + ": " + bytes.length + " bytes"; + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorSelectionCommandHandler.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorSelectionCommandHandler.java new file mode 100644 index 00000000..f5a34bb7 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorSelectionCommandHandler.java @@ -0,0 +1,77 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.hex; + +import org.eclipse.core.commands.AbstractHandler; +import org.eclipse.core.commands.ExecutionEvent; +import org.eclipse.core.commands.ExecutionException; +import org.eclipse.jface.viewers.ISelection; +import org.eclipse.ui.IEditorPart; +import org.eclipse.ui.handlers.HandlerUtil; + +import com.wudsn.ide.base.gui.MessageManager; + +/** + * The handler based class for commands based on the {@link HexEditorSelection}. + * + * @author Peter Dell + */ +public abstract class HexEditorSelectionCommandHandler extends AbstractHandler { + + protected ExecutionEvent event; + protected String commandId; + protected HexEditorSelection hexEditorSelection; + protected HexEditor hexEditor; + protected MessageManager messageManager; + + /** + * Creation is protected. + */ + protected HexEditorSelectionCommandHandler() { + super(); + } + + @Override + public final Object execute(ExecutionEvent event) throws ExecutionException { + IEditorPart editorPart; + ISelection menuEditorInputSelection; + editorPart = HandlerUtil.getActiveEditor(event); + menuEditorInputSelection = HandlerUtil.getActiveMenuSelection(event); + + if (editorPart instanceof HexEditor && menuEditorInputSelection instanceof HexEditorSelection) { + + this.event = event; + this.commandId = event.getCommand().getId(); + this.hexEditorSelection = (HexEditorSelection) menuEditorInputSelection; + this.hexEditor = ((HexEditor) editorPart); + this.messageManager = hexEditor.getMessageManager(); + messageManager.clearMessages(); + + performAction(); + + messageManager.displayMessages(); + + } + + return null; + } + + protected abstract void performAction() throws ExecutionException; +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorSelectionTransfer.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorSelectionTransfer.java new file mode 100644 index 00000000..e01cc6a4 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/HexEditorSelectionTransfer.java @@ -0,0 +1,96 @@ +package com.wudsn.ide.base.editor.hex; + +import java.io.ByteArrayInputStream; +import java.io.ByteArrayOutputStream; +import java.io.DataInputStream; +import java.io.DataOutputStream; +import java.io.IOException; + +import org.eclipse.swt.dnd.ByteArrayTransfer; +import org.eclipse.swt.dnd.TransferData; + +public final class HexEditorSelectionTransfer extends ByteArrayTransfer { + + private static final String HEX_EDITOR_SELECTION_NAME = "HexEditorSelection"; + private static final int HEX_EDITOR_SELECTION_ID = registerType(HEX_EDITOR_SELECTION_NAME); + private static HexEditorSelectionTransfer instance = new HexEditorSelectionTransfer(); + + private HexEditorSelectionTransfer() { + } + + public static HexEditorSelectionTransfer getInstance() { + return instance; + } + + @Override + public void javaToNative(Object object, TransferData transferData) { + if (object == null || !(object instanceof HexEditorSelection)) + return; + + if (isSupportedType(transferData)) { + HexEditorSelection hexEditorSelection = (HexEditorSelection) object; + try { + // write data to a byte array and then ask super to convert to + // pMedium + ByteArrayOutputStream out = new ByteArrayOutputStream(); + DataOutputStream writeOut = new DataOutputStream(out); + byte[] bytes = hexEditorSelection.getBytes(); + writeOut.writeInt(hexEditorSelection.getStartOffset()); + writeOut.writeInt(hexEditorSelection.getEndOffset()); + writeOut.writeInt(bytes.length); + writeOut.write(bytes); + + byte[] buffer = out.toByteArray(); + writeOut.close(); + + super.javaToNative(buffer, transferData); + + } catch (IOException e) { + } + } + } + + @Override + public Object nativeToJava(TransferData transferData) { + + if (isSupportedType(transferData)) { + + byte[] buffer = (byte[]) super.nativeToJava(transferData); + if (buffer == null) { + return null; + } + + HexEditorSelection hexEditorSelection; + hexEditorSelection = null; + try { + ByteArrayInputStream in = new ByteArrayInputStream(buffer); + DataInputStream readIn = new DataInputStream(in); + while (readIn.available() > 0) { + int startOffset=readIn.readInt(); + int endOffset=readIn.readInt(); + int size = readIn.readInt(); + byte[] bytes = new byte[size]; + readIn.read(bytes); + hexEditorSelection = new HexEditorSelection(startOffset, endOffset, bytes); + + } + readIn.close(); + } catch (IOException ex) { + hexEditorSelection = null; + } + return hexEditorSelection; + } + + return null; + } + + @Override + protected String[] getTypeNames() { + return new String[] { HEX_EDITOR_SELECTION_NAME }; + } + + @Override + protected int[] getTypeIds() { + return new int[] { HEX_EDITOR_SELECTION_ID }; + } +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariCOMParser.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariCOMParser.java new file mode 100644 index 00000000..c437b998 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariCOMParser.java @@ -0,0 +1,36 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.hex.parser; + +import org.eclipse.jface.viewers.StyledString; + +public final class AtariCOMParser extends AtariParser { + + @Override + public boolean parse(StyledString contentBuilder) { + if (contentBuilder == null) { + throw new IllegalArgumentException("Parameter 'contentBuilder' must not be null."); + } + int offset = 0; + int fileContentLenght = getFileContentLength(); + return parseAtariCOMFile(contentBuilder, offset, fileContentLenght); + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariDiskImageKFileParser.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariDiskImageKFileParser.java new file mode 100644 index 00000000..b9056a0a --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariDiskImageKFileParser.java @@ -0,0 +1,47 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.hex.parser; + +import org.eclipse.jface.viewers.StyledString; + +public final class AtariDiskImageKFileParser extends AtariDiskImageParser { + // The offset where the COM file starts in an Atari Disk Image (k-file). + public static final int ATARI_DISK_IMAGE_K_FILE_COM_FILE_OFFSET = 16 + 3 * 128; + + @Override + public boolean parse(StyledString contentBuilder) { + + if (contentBuilder == null) { + throw new IllegalArgumentException("Parameter 'contentBuilder' must not be null."); + } + + boolean error = super.parse(contentBuilder); + + // If the disk image is a k-file image, the contained COM file is parsed + // as well. + if (!error) { + // The length of the k-file is stored in $709/$70a. + int length = ATARI_DISK_IMAGE_K_FILE_COM_FILE_OFFSET + getFileContentByte(0x19) + 256 * getFileContentByte(0x1a); + error = parseAtariCOMFile(contentBuilder, ATARI_DISK_IMAGE_K_FILE_COM_FILE_OFFSET, length); + } + return error; + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariDiskImageParser.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariDiskImageParser.java new file mode 100644 index 00000000..7781ad2d --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariDiskImageParser.java @@ -0,0 +1,89 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.hex.parser; + +import org.eclipse.jface.viewers.StyledString; + +import com.wudsn.ide.base.Texts; +import com.wudsn.ide.base.editor.hex.HexEditorContentOutlineTreeObject; + +public class AtariDiskImageParser extends AtariParser { + + @Override + public boolean parse(StyledString contentBuilder) { + + if (contentBuilder == null) { + throw new IllegalArgumentException("Parameter 'contentBuilder' must not be null."); + } + + boolean error = false; + int length = getFileContentLength(); + int offset = 0; + + HexEditorContentOutlineTreeObject treeObject; + treeObject = printBlockHeader(contentBuilder, Texts.HEX_EDITOR_ATARI_DISK_IMAGE_HEADER, -1, "", offset, offset, + offset + 15); + offset = printBytes(treeObject, contentBuilder, offset, offset + 15, true, 0); + contentBuilder.append("\n"); + + boolean blockMode; + + blockMode = true; + + int mainSectorSize = getFileContentByte(4) + 256 * getFileContentByte(5); + int bootSectorSize = mainSectorSize; + + if (bootSectorSize == 256 && (length % 256) == 128 + 16) { + bootSectorSize = 128; + } + int startAddress = 0; + int sectorCount; + int sectorSize; + + sectorCount = 1; + sectorSize = bootSectorSize; + try { + while (blockMode && !error) { + treeObject = printBlockHeader(contentBuilder, Texts.HEX_EDITOR_ATARI_SECTOR_HEADER, sectorCount, + + Texts.HEX_EDITOR_ATARI_SECTOR_HEADER_PARAMETERS, offset, startAddress, startAddress + sectorSize - 1); + offset = printBytes(treeObject, contentBuilder, offset, offset + sectorSize - 1, true, startAddress); + contentBuilder.append("\n"); + + if (offset >= length) { + blockMode = false; + } else if (length - offset < sectorSize) { + error = true; + } + sectorCount++; + if (sectorCount > 3) { + sectorSize = mainSectorSize; + } + } + } catch (RuntimeException ex) { + contentBuilder.append(ex.toString()); + } + if (error) { + printBlockWithError(contentBuilder, Texts.HEX_EDITOR_ATARI_SECTOR_ERROR, length, offset); + } + return error; + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariMADSParser.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariMADSParser.java new file mode 100644 index 00000000..9f9dfd83 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariMADSParser.java @@ -0,0 +1,221 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.hex.parser; + +import org.eclipse.jface.viewers.StyledString; + +import com.wudsn.ide.base.Texts; +import com.wudsn.ide.base.common.HexUtility; +import com.wudsn.ide.base.common.TextUtility; +import com.wudsn.ide.base.editor.hex.HexEditorContentOutlineTreeObject; +import com.wudsn.ide.base.editor.hex.HexEditorParser; + +/** + * All parsing in here is based on the format description in the MADS online documentation. + * + * @author Peter Dell + */ +public class AtariMADSParser extends HexEditorParser { + + public static final int COM_HEADER = 0xffff; + public static final int RELOC_HEADER = 0x524d; + public static final int UPDATE_RELOC_HEADER = 0xffef; + public static final int UPDATE_SYMBOLS_HEADER = 0xffee; + public static final int DEFINE_SYMBOLS_HEADER = 0xffed; + + @Override + public boolean parse(StyledString contentBuilder) { + if (contentBuilder == null) { + throw new IllegalArgumentException("Parameter 'contentBuilder' must not be null."); + } + + boolean error; + int offset = 0; + + // Skip offset bytes in lookup array. + skipByteTextIndex(offset); + + HexEditorContentOutlineTreeObject treeObject; + int fileContentLength = getFileContentLength(); + + error = (fileContentLength - offset) < 17; + boolean first = true; + boolean more = true; + try { + while (more && !error) { + if (!first) { + contentBuilder.append("\n"); + } + first = false; + if (offset == fileContentLength) { + more = false; + } else { + int header = getFileContentWord(offset); + if (header == COM_HEADER && getFileContentWord(offset + 6) == RELOC_HEADER) { + int startAddress = getFileContentWord(offset + 2); + int endAddress = getFileContentWord(offset + 4); + int config = getFileContentByte(offset + 9); + + String headerText = TextUtility.format(Texts.HEX_EDITOR_ATARI_MADS_RELOC_BLOCK_HEADER, + HexUtility.getLongValueHexString(startAddress, 4), + HexUtility.getLongValueHexString(endAddress, 4), + HexUtility.getByteValueHexString(config)); + + treeObject = printHeader(contentBuilder, offset, headerText); + offset = printBytes(treeObject, contentBuilder, offset, offset + 15, true, 0); + + int blockEnd = offset + endAddress - startAddress; + + offset = printBytes(treeObject, contentBuilder, offset, blockEnd, true, startAddress); + + } else if (header == UPDATE_RELOC_HEADER) { + + int type = getFileContentByte(offset + 2); + int dataLength = getFileContentWord(offset + 3); + + treeObject = printTypedHeader(contentBuilder, offset, + Texts.HEX_EDITOR_ATARI_MADS_UPDATE_RELOC_BLOCK_HEADER, type, dataLength); + offset = printBytes(treeObject, contentBuilder, offset, offset + 4, false, 0); + + // The exception is an update block for address high + // bytes ">", where for such a block an extra BYTE is + // stored for each address (low byte of address being + // modified). + int dataSize = (type == '>' ? 3 : 2); + int blockEnd = offset + (dataLength * dataSize) - 1; + + offset = printBytes(treeObject, contentBuilder, offset, blockEnd, false, 0); + + } else if (header == UPDATE_SYMBOLS_HEADER) { + /** + * <pre> + * HEADER WORD ($FFEE) + * TYPE CHAR (B-YTE, W-ORD, L-ONG, D-WORD, <, >) + * DATA_LENGTH WORD + * LABEL_LENGTH WORD + * LABEL_NAME ATASCII + * DATA WORD .. .. .. (DATA_LENGTH words) + * </pre> + */ + int type = getFileContentByte(offset + 2); + int dataLength = getFileContentWord(offset + 3); + int labelLength = getFileContentWord(offset + 5); + + treeObject = printTypedHeader(contentBuilder, offset, + Texts.HEX_EDITOR_ATARI_MADS_UPDATE_SYMBOLS_BLOCK_HEADER, type, dataLength); + offset = printBytes(treeObject, contentBuilder, offset, offset + 6, false, 0); + offset = printBytes(treeObject, contentBuilder, offset, offset + labelLength - 1, false, 0); + offset = printBytes(treeObject, contentBuilder, offset, offset + dataLength * 2 - 1, false, 0); + + } else if (header == DEFINE_SYMBOLS_HEADER) { + + int length = getFileContentWord(offset + 2); + String headerText = TextUtility.format(Texts.HEX_EDITOR_ATARI_MADS_DEFINE_SYMBOLS_BLOCK_HEADER, + HexUtility.getLongValueHexString(length, 4)); + StyledString headerStyledString = new StyledString(headerText, offsetStyler); + contentBuilder.append(headerStyledString).append("\n\n"); + treeObject = printBlockHeader(contentBuilder, headerStyledString, offset); + + offset += 4; + for (int i = 0; i < length; i++) { + int type = getFileContentByte(offset); + int labelType = getFileContentByte(offset + 1); + int labelLength = getFileContentWord(offset + 2); + String labelName = getLabelName(offset + 4, labelLength); + int address = getFileContentWord(offset + 4 + labelLength); + int headerEnd = offset + 6 + labelLength - 1; + switch (labelType) { + case 'P': + int procType = getFileContentByte(headerEnd + 1); + int paramCount = getFileContentWord(headerEnd + 2); + headerEnd += 4; + for (int j = 0; j < paramCount; j++) { + switch (procType) { + case 'D': + break; + case 'R': + headerEnd += 1; + break; + case 'V': + headerEnd += 1; + int paramLenght = getFileContentWord(headerEnd); + headerEnd += 2 + paramLenght; + break; + } + } + break; + case 'A': + break; + case 'S': + break; + } + + headerText = TextUtility.format(Texts.HEX_EDITOR_ATARI_MADS_DEFINE_SYMBOL_HEADER, + String.valueOf((char) type), String.valueOf((char) labelType), labelName, + HexUtility.getLongValueHexString(address, 4)); + headerStyledString = new StyledString(headerText, offsetStyler); + contentBuilder.append(headerStyledString).append("\n"); + treeObject = printBlockHeader(contentBuilder, headerStyledString, offset); + offset = printBytes(treeObject, contentBuilder, offset, headerEnd, false, 0); + + } + + } else { + error = true; + } + } + } + } catch (RuntimeException ex) { + contentBuilder.append(ex.toString()); + error = true; + } + + if (error) { + printBlockWithError(contentBuilder, Texts.HEX_EDITOR_ATARI_MADS_BLOCK_ERROR, fileContentLength, offset); + } + return error; + } + + private HexEditorContentOutlineTreeObject printHeader(StyledString contentBuilder, int offset, String headerText) { + HexEditorContentOutlineTreeObject treeObject; + StyledString headerStyledString = new StyledString(headerText, offsetStyler); + contentBuilder.append(headerStyledString).append("\n"); + treeObject = printBlockHeader(contentBuilder, headerStyledString, offset); + return treeObject; + } + + private HexEditorContentOutlineTreeObject printTypedHeader(StyledString contentBuilder, int offset, String text, + int type, int dataLength) { + HexEditorContentOutlineTreeObject treeObject; + String headerText = TextUtility.format(text, String.valueOf((char) type), + HexUtility.getLongValueHexString(dataLength, 4)); + treeObject = printHeader(contentBuilder, offset, headerText); + return treeObject; + } + + private String getLabelName(int offset, int length) { + StringBuffer buffer = new StringBuffer(8); + for (int i = 0; i < length; i++) { + buffer.append((char) getFileContentByte(offset + i)); + } + return buffer.toString(); + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariParser.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariParser.java new file mode 100644 index 00000000..01da078f --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariParser.java @@ -0,0 +1,127 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.hex.parser; + +import org.eclipse.jface.viewers.StyledString; + +import com.wudsn.ide.base.Texts; +import com.wudsn.ide.base.editor.hex.HexEditorContentOutlineTreeObject; +import com.wudsn.ide.base.editor.hex.HexEditorParser; + +public abstract class AtariParser extends HexEditorParser { + + public final static int COM_HEADER = 0xffff; + + protected final boolean parseAtariCOMFile(StyledString contentBuilder, int offset, int fileContentLength) { + if (contentBuilder == null) { + throw new IllegalArgumentException("Parameter 'contentBuilder' must not be null."); + } + boolean error; + int startAddress; + int endAddress; + + int blockCount; + int blockEnd; + + // Skip offset bytes in lookup array. + skipByteTextIndex(offset); + + HexEditorContentOutlineTreeObject treeObject; + + error = (fileContentLength - offset) < 7; + if (!error) { + startAddress = getFileContentWord(offset + 2); + endAddress = getFileContentWord(offset + 4); + + blockCount = 1; + blockEnd = offset + endAddress - startAddress + 6; + treeObject = printBlockHeader(contentBuilder, Texts.HEX_EDITOR_ATARI_COM_BLOCK_HEADER, blockCount, + Texts.HEX_EDITOR_ATARI_COM_BLOCK_HEADER_PARAMETERS, offset, startAddress, endAddress); + offset = printBytes(treeObject, contentBuilder, offset, offset + 5, true, 0); + + boolean blockMode; + blockMode = true; + error = blockEnd >= fileContentLength; + try { + while (blockMode && !error) { + offset = printBytes(treeObject, contentBuilder, offset, blockEnd, true, startAddress); + + int headerLength = -1; + // No more bytes left? + if (offset == fileContentLength ) { + blockMode = false; + } else + // At least 5 bytes available? (4 header bytes and 1 data + // byte) + if (fileContentLength - offset < 5) { + error = true; + } else { + boolean comHeader; + comHeader = getFileContentWord(offset) == COM_HEADER; + if (comHeader) { + // At least 7 bytes available? (6 header bytes and 1 + // data byte) + if (fileContentLength - offset < 7) { + error = true; + } else { + // Inner COM header found + headerLength = 6; + startAddress = getFileContentByte(offset + 2) + 256 * getFileContentByte(offset + 3); + endAddress = getFileContentByte(offset + 4) + 256 * getFileContentByte(offset + 5); + } + } else { + // No inner COM header found + headerLength = 4; + startAddress = getFileContentByte(offset + 0) + 256 * getFileContentByte(offset + 1); + endAddress = getFileContentByte(offset + 2) + 256 * getFileContentByte(offset + 3); + } + error = endAddress < startAddress; + } + + if (blockMode) { + contentBuilder.append("\n"); + } + + if (blockMode && !error) { + blockCount++; + blockEnd = offset + endAddress - startAddress + headerLength; + if (blockEnd < fileContentLength) { + + treeObject = printBlockHeader(contentBuilder, Texts.HEX_EDITOR_ATARI_COM_BLOCK_HEADER, + blockCount, Texts.HEX_EDITOR_ATARI_COM_BLOCK_HEADER_PARAMETERS, offset, + startAddress, endAddress); + offset = printBytes(treeObject, contentBuilder, offset, offset + headerLength - 1, true, 0); + } else { + error = true; + } + } + } + } catch (RuntimeException ex) { + contentBuilder.append(ex.toString()); + error = true; + } + } + if (error) { + printBlockWithError(contentBuilder, Texts.HEX_EDITOR_ATARI_COM_BLOCK_ERROR, fileContentLength, offset); + } + return error; + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariSAPParser.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariSAPParser.java new file mode 100644 index 00000000..d62131a6 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariSAPParser.java @@ -0,0 +1,51 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.hex.parser; + +import org.eclipse.jface.viewers.StyledString; + +import com.wudsn.ide.base.Texts; +import com.wudsn.ide.base.editor.hex.HexEditorContentOutlineTreeObject; + +public final class AtariSAPParser extends AtariParser { + + @Override + public boolean parse(StyledString contentBuilder) { + if (contentBuilder == null) { + throw new IllegalArgumentException("Parameter 'contentBuilder' must not be null."); + } + int offset = 0; + int fileContentLenght = getFileContentLength(); + int maxOffset = fileContentLenght - 2; + while (offset < maxOffset && getFileContentByte(offset) != 0xff && getFileContentByte(offset) != 0xff) { + offset++; + } + if (offset == maxOffset) { + return false; + } + HexEditorContentOutlineTreeObject treeObject = printBlockHeader(contentBuilder, + Texts.HEX_EDITOR_ATARI_SAP_FILE_HEADER, -1, "", 0, 0, 0); + printBytes(treeObject, contentBuilder, 0, offset - 1, false, 0); + contentBuilder.append("\n"); + + return parseAtariCOMFile(contentBuilder, offset, fileContentLenght); + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariSDXParser.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariSDXParser.java new file mode 100644 index 00000000..d6697737 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/AtariSDXParser.java @@ -0,0 +1,195 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.hex.parser; + +import org.eclipse.jface.viewers.StyledString; + +import com.wudsn.ide.base.Texts; +import com.wudsn.ide.base.common.HexUtility; +import com.wudsn.ide.base.common.NumberUtility; +import com.wudsn.ide.base.common.TextUtility; +import com.wudsn.ide.base.editor.hex.HexEditorContentOutlineTreeObject; +import com.wudsn.ide.base.editor.hex.HexEditorParser; + +public class AtariSDXParser extends HexEditorParser { + + public static final int NON_RELOC_HEADER = 0xfffa; + public static final int RELOC_HEADER = 0xfffe; + public static final int UPDATE_RELOC_HEADER = 0xfffd; + public static final int UPDATE_SYMBOLS_HEADER = 0xfffb; + public static final int DEFINE_SYMBOLS_HEADER = 0xfffc; + + @Override + public boolean parse(StyledString contentBuilder) { + if (contentBuilder == null) { + throw new IllegalArgumentException("Parameter 'contentBuilder' must not be null."); + } + + boolean error; + int offset = 0; + + // Skip offset bytes in lookup array. + skipByteTextIndex(offset); + + HexEditorContentOutlineTreeObject treeObject; + int fileContentLength = getFileContentLength(); + + error = (fileContentLength - offset) < 7; + boolean first = true; + boolean more = true; + try { + while (more && !error) { + if (!first) { + contentBuilder.append("\n"); + } + first = false; + if (offset == fileContentLength) { + more = false; + } else { + int header = getFileContentWord(offset); + if (header == NON_RELOC_HEADER) { + int startAddress = getFileContentWord(offset + 2); + int endAddress = getFileContentWord(offset + 4); + + treeObject = printBlockHeader(contentBuilder, + Texts.HEX_EDITOR_ATARI_SDX_NON_RELOC_BLOCK_HEADER, -1, + Texts.HEX_EDITOR_ATARI_SDX_NON_RELOC_BLOCK_HEADER_PARAMETERS, offset, startAddress, + endAddress); + offset = printBytes(treeObject, contentBuilder, offset, offset + 5, true, 0); + + int blockEnd = offset + endAddress - startAddress; + + offset = printBytes(treeObject, contentBuilder, offset, blockEnd, true, startAddress); + + } else if (header == RELOC_HEADER) { + int blockNumber = getFileContentByte(offset + 2); + int blockId = getFileContentByte(offset + 3); + int blockOffset = getFileContentWord(offset + 4); + int blockLength = getFileContentWord(offset + 6); + + String headerText = TextUtility.format(Texts.HEX_EDITOR_ATARI_SDX_RELOC_BLOCK_HEADER, + NumberUtility.getLongValueDecimalString(blockNumber), + HexUtility.getByteValueHexString(blockId), + HexUtility.getLongValueHexString(blockOffset, 4), + HexUtility.getLongValueHexString(blockLength, 4)); + StyledString headerStyledString = new StyledString(headerText, offsetStyler); + contentBuilder.append(headerStyledString).append("\n"); + treeObject = printBlockHeader(contentBuilder, headerStyledString, offset); + offset = printBytes(treeObject, contentBuilder, offset, offset + 7, true, 0); + + int blockEnd = offset + blockLength - 1; + + // Print bytes only of the block is not marked as EMPTY + if ((blockId & 0x80) != 0x80) { + offset = printBytes(treeObject, contentBuilder, offset, blockEnd, true, blockOffset); + } + } else if (header == UPDATE_RELOC_HEADER) { + int blockNumber = getFileContentByte(offset + 2); + int blockLength = getFileContentWord(offset + 3); + + String headerText = TextUtility.format(Texts.HEX_EDITOR_ATARI_SDX_UPDATE_RELOC_BLOCK_HEADER, + NumberUtility.getLongValueDecimalString(blockNumber), + HexUtility.getLongValueHexString(blockLength, 4)); + StyledString headerStyledString = new StyledString(headerText, offsetStyler); + contentBuilder.append(headerStyledString).append("\n"); + treeObject = printBlockHeader(contentBuilder, headerStyledString, offset); + offset = printBytes(treeObject, contentBuilder, offset, offset + 4, true, 0); + int blockEnd = getBlockEnd(offset); + offset = printBytes(treeObject, contentBuilder, offset, blockEnd, true, 0); + } else if (header == UPDATE_SYMBOLS_HEADER) { + String symbolName=getSymbolName(offset+2); + int blockLength = getFileContentWord(offset + 10); + + String headerText = TextUtility.format(Texts.HEX_EDITOR_ATARI_SDX_UPDATE_SYMBOLS_BLOCK_HEADER, + symbolName, HexUtility.getLongValueHexString(blockLength, 4)); + StyledString headerStyledString = new StyledString(headerText, offsetStyler); + contentBuilder.append(headerStyledString).append("\n"); + treeObject = printBlockHeader(contentBuilder, headerStyledString, offset); + offset = printBytes(treeObject, contentBuilder, offset, offset + 11, true, 0); + int blockEnd = getBlockEnd(offset); + offset = printBytes(treeObject, contentBuilder, offset, blockEnd, true, 0); + } else if (header == DEFINE_SYMBOLS_HEADER) { + int blockNumber = getFileContentByte(offset + 2); + int blockOffset = getFileContentWord(offset + 3); + String symbolName=getSymbolName(offset+5); + + String headerText = TextUtility.format(Texts.HEX_EDITOR_ATARI_SDX_DEFINE_SYMBOLS_BLOCK_HEADER, + NumberUtility.getLongValueDecimalString(blockNumber), + HexUtility.getLongValueHexString(blockOffset), symbolName); + StyledString headerStyledString = new StyledString(headerText, offsetStyler); + contentBuilder.append(headerStyledString).append("\n"); + treeObject = printBlockHeader(contentBuilder, headerStyledString, offset); + offset = printBytes(treeObject, contentBuilder, offset, offset + 12, true, 0); + } else { + error = true; + } + } + } + } catch (RuntimeException ex) { + contentBuilder.append(ex.toString()); + error = true; + } + + if (error) { + printBlockWithError(contentBuilder, Texts.HEX_EDITOR_ATARI_SDX_BLOCK_ERROR, fileContentLength, offset); + } + return error; + } + + /** + * Gets end offset for UPDATE_RELOC_HEADER and UPDATE_SYMBOLS_HEADER. + * + * @param offset + * The start offset, a non-negative integer. + * @return The end offset, a non-negative integer. + */ + private int getBlockEnd(int offset) { + int fileContentLength = getFileContentLength(); + int i = offset; + int blockEnd = -1; + while (blockEnd < 0 && i < fileContentLength) { + int location = getFileContentByte(i); + switch (location) { + case 0xfc: + blockEnd = i; + break; + case 0xfd: + i += 3; + break; + case 0xfe: + i += 3; + break; + default: + i++; + break; + } + } + return blockEnd; + } + + private String getSymbolName(int offset) { + StringBuffer buffer = new StringBuffer(8); + for (int i = 0; i < 8; i++) { + buffer.append((char) getFileContentByte(offset + i)); + } + return buffer.toString(); + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/BinaryParser.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/BinaryParser.java new file mode 100644 index 00000000..0f4fe8dd --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/BinaryParser.java @@ -0,0 +1,39 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.hex.parser; + +import org.eclipse.jface.viewers.StyledString; + +import com.wudsn.ide.base.editor.hex.HexEditorContentOutlineTreeObject; +import com.wudsn.ide.base.editor.hex.HexEditorParser; + +public class BinaryParser extends HexEditorParser { + + @Override + public boolean parse(StyledString contentBuilder) { + if (contentBuilder == null) { + throw new IllegalArgumentException("Parameter 'contentBuilder' must not be null."); + } + HexEditorContentOutlineTreeObject treeObject = new HexEditorContentOutlineTreeObject(contentBuilder); + printBytes(treeObject, contentBuilder, 0, getFileContentLength() - 1, false, 0); + return false; + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/C64PRGParser.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/C64PRGParser.java new file mode 100644 index 00000000..cd0cb187 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/hex/parser/C64PRGParser.java @@ -0,0 +1,63 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.hex.parser; + +import org.eclipse.jface.viewers.StyledString; + +import com.wudsn.ide.base.Texts; +import com.wudsn.ide.base.editor.hex.HexEditorContentOutlineTreeObject; +import com.wudsn.ide.base.editor.hex.HexEditorParser; + +public class C64PRGParser extends HexEditorParser { + + @Override + public boolean parse(StyledString contentBuilder) { + if (contentBuilder == null) { + throw new IllegalArgumentException("Parameter 'contentBuilder' must not be null."); + } + boolean error; + int startAddress; + int endAddress; + + int length = getFileContentLength(); + int offset = 0; + + error = (length < 2); + if (!error) { + startAddress = getFileContentByte(offset + 0) + 256 * getFileContentByte(offset + 1); + endAddress = startAddress + length - 3; + + HexEditorContentOutlineTreeObject treeObject; + treeObject = printBlockHeader(contentBuilder, Texts.HEX_EDITOR_C64_PRG_HEADER, -1, + Texts.HEX_EDITOR_C64_PRG_HEADER_PARAMETERS, offset, startAddress, endAddress); + offset = printBytes(treeObject, contentBuilder, offset, offset + 1, true, 0); + + error = endAddress > 0xffff; + if (!error) { + printBytes(treeObject, contentBuilder, offset, length - 1, true, startAddress); + } + } + if (error) { + printBlockWithError(contentBuilder, Texts.HEX_EDITOR_C64_PRG_ERROR, length, offset); + } + return error; + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorConvertNumbersCommand.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorConvertNumbersCommand.java new file mode 100644 index 00000000..7319298c --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorConvertNumbersCommand.java @@ -0,0 +1,296 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.text; + +import org.eclipse.core.commands.AbstractHandler; +import org.eclipse.core.commands.ExecutionEvent; +import org.eclipse.core.commands.ExecutionException; +import org.eclipse.core.expressions.PropertyTester; +import org.eclipse.jface.text.BadLocationException; +import org.eclipse.jface.text.IDocument; +import org.eclipse.jface.text.ITextSelection; +import org.eclipse.text.edits.ReplaceEdit; +import org.eclipse.ui.IEditorPart; +import org.eclipse.ui.handlers.HandlerUtil; +import org.eclipse.ui.texteditor.ITextEditor; +import org.eclipse.ui.texteditor.ITextEditorExtension2; + +/** + * Command handler for converting number within a text selection.<br/> + * TODO Check block selection mode and read-only mode for multi line conversion, + * see https://bugs.eclipse.org/bugs/show_bug.cgi?id=382707<br/> + * + * @author Peter Dell + * + * @since 1.6.3 + */ +public final class TextEditorConvertNumbersCommand { + + public static final class Id { + public static final String TO_DECIMAL = "com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToDecimalCommand"; + public static final String TO_HEXA_DECIMAL = "com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToHexaDecimalCommand"; + public static final String TO_BINARY = "com.wudsn.ide.base.editor.text.TextEditorConvertNumbersToBinaryCommand"; + } + + private static final class Mode { + public final static int DECIMAL = 1; + public final static int HEXA_DECIMAL = 2; + public final static int BINARY = 3; + } + + /** + * In the productive code, this field must be <code>false</code>, to enable + * the property checks when the menu is created. To debug the conversion, + * this field must be set to <code>true</code>, to disable the property + * checks when the menu is created. + */ + + private static final boolean DEBUG = false; + + public static final class EnabledPropertyTester extends PropertyTester { + + public EnabledPropertyTester() { + } + + @Override + public boolean test(final Object receiver, final String property, final Object[] args, + final Object expectedValue) { + + if (property.equals("isEnabled") && receiver instanceof ITextSelection && expectedValue instanceof String) { + ITextSelection selection = (ITextSelection) receiver; + String commandId = (String) expectedValue; + int length = selection.getLength(); + boolean enabled; + // For performance reasons, the test is skipped of the selected + // block is large and we simply assume there are numbers in. + if (length <= 0) { + enabled = false; + } else if (length > 1024 || DEBUG) { + enabled = true; + } else { + StringBuilder result = new StringBuilder(selection.getLength()); + enabled = convertNumberValues(selection, commandId, result); + } + return enabled; + + } + return false; + } + } + + public static final class Handler extends AbstractHandler { + + public Handler() { + + } + + @Override + public Object execute(ExecutionEvent event) throws ExecutionException { + IEditorPart editor; + String commandId; + + commandId = event.getCommand().getId(); + editor = HandlerUtil.getActiveEditorChecked(event); + + if (!(editor instanceof ITextEditor)) { + return null; + } + + // Find editor, document, selection, start and end line. + ITextEditor textEditor; + IDocument document; + ITextSelection selection; + + textEditor = (ITextEditor) editor; + + document = textEditor.getDocumentProvider().getDocument(textEditor.getEditorInput()); + selection = (ITextSelection) textEditor.getSelectionProvider().getSelection(); + StringBuilder result = new StringBuilder(selection.getLength()); + if (convertNumberValues(selection, commandId, result)) { + try { + + if (editor instanceof ITextEditorExtension2) { + ITextEditorExtension2 extension = (ITextEditorExtension2) editor; + if (!extension.validateEditorInputState()) { + return null; + } + } + // TODO: Multiple replace edits in case of block selection. + ReplaceEdit replaceEdit = new ReplaceEdit(selection.getOffset(), selection.getLength(), + result.toString()); + replaceEdit.apply(document); + + } catch (BadLocationException ex) { + throw new RuntimeException(ex); + } + } + + return null; + } + + } + + final static boolean convertNumberValues(ITextSelection selection, String commandId, StringBuilder result) { + + if (selection == null) { + throw new IllegalArgumentException("Parameter 'selection' must not be null."); + } + + int mode; + if (commandId.equals(Id.TO_DECIMAL)) { + mode = Mode.DECIMAL; + } else if (commandId.equals(Id.TO_HEXA_DECIMAL)) { + mode = Mode.HEXA_DECIMAL; + } else if (commandId.equals(Id.TO_BINARY)) { + mode = Mode.BINARY; + } else { + throw new IllegalArgumentException("Unsupported command '" + commandId + "'."); + } + if (result == null) { + throw new IllegalArgumentException("Parameter 'result' must not be null."); + } + + // if (selection instanceof IBlockTextSelection) { + // return false; + // } + + int length = selection.getLength(); + if (length == 0) { + return false; + } + + String text = selection.getText(); + length = text.length(); // Because it might be a block selection + boolean numbersFound = false; + + StringBuilder number = new StringBuilder(); + int offset = 0; + int endOffset = length; + + try { + while (offset < endOffset) { + char c1; + char c2; + long value; + c1 = text.charAt(offset); + + if (offset < endOffset - 1) { + c2 = text.charAt(offset + 1); + } else { + c2 = ' '; + } + + if (c1 >= '0' && c1 <= '9') { + number.setLength(0); + while (offset < endOffset && c1 >= '0' && c1 <= '9') { + number.append(c1); + offset++; + if (offset < endOffset) { + c1 = text.charAt(offset); + } + } + value = Long.parseLong(number.toString()); + appendNumberValue(result, mode, value); + numbersFound = true; + + } else if (c1 == '$' + && ((c2 >= '0' && c2 <= '9') || (c2 >= 'A' && c2 <= 'F') || (c2 >= 'a' && c2 <= 'f'))) { + number.setLength(0); + offset++; + c1 = c2; + while (offset < endOffset + && ((c1 >= '0' && c1 <= '9') || (c1 >= 'A' && c1 <= 'F') || (c1 >= 'a' && c1 <= 'f'))) { + number.append(c1); + offset++; + if (offset < endOffset) { + c1 = text.charAt(offset); + } + } + + value = Long.parseLong(number.toString(), 16); + appendNumberValue(result, mode, value); + numbersFound = true; + + } else if (c1 == '%' && (c2 >= '0' && c2 <= '1')) { + number.setLength(0); + offset++; + c1 = c2; + while (offset < endOffset && ((c1 >= '0' && c1 <= '1'))) { + number.append(c1); + offset++; + if (offset < endOffset) { + c1 = text.charAt(offset); + } + } + value = Long.parseLong(number.toString(), 2); + appendNumberValue(result, mode, value); + numbersFound = true; + } else { + result.append(c1); + offset++; + } + } + } catch (NumberFormatException ex) { + return false; // For example if the number becomes too large + } + + // Do nothing if the text is already the same. + if (result.toString().equals(selection.getText())) { + numbersFound = false; + } + return numbersFound; + } + + private static void appendNumberValue(StringBuilder result, int mode, long value) { + if (result == null) { + throw new IllegalArgumentException("Parameter 'result' must not be null."); + } + + switch (mode) { + case Mode.DECIMAL: + result.append(Long.toString(value)); + break; + case Mode.HEXA_DECIMAL: + result.append("$"); + String hexValue = Long.toHexString(value); + int hexLength = hexValue.length(); + while ((hexLength & 1) != 0) { + result.append('0'); + hexLength++; + } + result.append(hexValue); + break; + case Mode.BINARY: + result.append("%"); + String binaryValue = Long.toBinaryString(value); + int binarLength = binaryValue.length(); + while ((binarLength & 7) != 0) { + result.append('0'); + binarLength++; + } + result.append(binaryValue); + break; + + default: + throw new IllegalArgumentException("Unsupported mode " + mode + "."); + } + + } +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorLinesCommandHandler.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorLinesCommandHandler.java new file mode 100644 index 00000000..4ab04468 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorLinesCommandHandler.java @@ -0,0 +1,139 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.text; + +import java.util.ArrayList; +import java.util.List; + +import org.eclipse.core.commands.AbstractHandler; +import org.eclipse.core.commands.Command; +import org.eclipse.core.commands.ExecutionEvent; +import org.eclipse.core.commands.ExecutionException; +import org.eclipse.jface.text.BadLocationException; +import org.eclipse.jface.text.IDocument; +import org.eclipse.jface.text.ITextSelection; +import org.eclipse.text.edits.ReplaceEdit; +import org.eclipse.ui.IEditorPart; +import org.eclipse.ui.handlers.HandlerUtil; +import org.eclipse.ui.texteditor.ITextEditor; +import org.eclipse.ui.texteditor.ITextEditorExtension2; + +/** + * Base class for text editor actions operating on the selected lines. Inspired + * by http://www.stateofflow.com/projects/2/sortit. + * + * @author Peter Dell + */ +abstract class TextEditorLinesCommandHandler extends AbstractHandler { + + @Override + public Object execute(ExecutionEvent event) throws ExecutionException { + IEditorPart editor; + editor = HandlerUtil.getActiveEditorChecked(event); + + if (!(editor instanceof ITextEditor)) { + return null; + } + + // Find editor, document, selection, start and end line. + ITextEditor textEditor; + IDocument document; + ITextSelection selection; + int endLineIndex; + int startLineIndex; + + textEditor = (ITextEditor) editor; + + document = textEditor.getDocumentProvider().getDocument( + textEditor.getEditorInput()); + selection = (ITextSelection) textEditor.getSelectionProvider() + .getSelection(); + endLineIndex = selection.getEndLine(); + startLineIndex = selection.getStartLine(); + if (startLineIndex == endLineIndex) { + startLineIndex = 0; + endLineIndex = document.getNumberOfLines() - 1; + } + + try { + // Collect lines. + int startOffset; + int length; + + startOffset = document.getLineOffset(startLineIndex); + length = 0; + + List<String> lines; + lines = new ArrayList<String>(); + for (int line = startLineIndex; line <= endLineIndex; line++) { + int delimiterLength = document.getLineDelimiter(line) == null ? 0 + : document.getLineDelimiter(line).length(); + String lineText = document.get(document.getLineOffset(line), + document.getLineLength(line) - delimiterLength); + lines.add(lineText); + length = length + lineText.length() + delimiterLength; + } + + process(event.getCommand(), lines); + + int lineCount = lines.size(); + String lineDelimiter = document.getLineDelimiter(startLineIndex); + StringBuilder replacementText = new StringBuilder(); + for (int i = 0; i < lineCount; i++) { + replacementText.append(lines.get(i)); + if ((i < lineCount - 1) || (i == lineCount - 1) + && (document.getLineDelimiter(endLineIndex) != null)) { + replacementText.append(lineDelimiter); + } + } + + if (editor instanceof ITextEditorExtension2) { + ITextEditorExtension2 extension = (ITextEditorExtension2) editor; + if (!extension.validateEditorInputState()) { + return null; + } + } + + ReplaceEdit replaceEdit = new ReplaceEdit(startOffset, length, + replacementText.toString()); + replaceEdit.apply(document); + // re-select the lines that have been processed + textEditor.getSelectionProvider().setSelection(selection); + } catch (BadLocationException ex) { + throw new ExecutionException("Error during handler execution.", ex); + } + return null; + } + + /** + * Subclasses must process the lines given in the list. The processed list + * will be used to rebuild the lines in the editor. + * + * @param command + * The command which is currently executed, not <code>null</code> + * . + * + * @param lines + * The modifiable list of lines, may be empty, not + * <code>null</code>. + */ + protected abstract void process(Command command, List<String> lines); + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorReverseLinesCommandHandler.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorReverseLinesCommandHandler.java new file mode 100644 index 00000000..bd2cbf9a --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorReverseLinesCommandHandler.java @@ -0,0 +1,51 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.text; + +import java.util.Collections; +import java.util.List; + +import org.eclipse.core.commands.Command; + +/** + * Action to reverse the sequence of lines. + * + * @author Peter Dell + */ +public final class TextEditorReverseLinesCommandHandler extends + TextEditorLinesCommandHandler { + + @Override + protected void process(Command command, List<String> lines) { + if (command == null) { + throw new IllegalArgumentException( + "Parameter 'command' must not be null."); + } + if (lines == null) { + throw new IllegalArgumentException( + "Parameter 'lines' must not be null."); + } + for (int i = 0; i < lines.size() / 2; i++) { + int j = lines.size() - i - 1; + Collections.swap(lines, i, j); + + } + } +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorSortLinesCaseInsensitiveCommandHandler.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorSortLinesCaseInsensitiveCommandHandler.java new file mode 100644 index 00000000..b2dff158 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorSortLinesCaseInsensitiveCommandHandler.java @@ -0,0 +1,37 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.text; + +import java.util.Comparator; + +/** + * Action to sort lines case insensitively. + * + * @author Peter Dell + */ +public final class TextEditorSortLinesCaseInsensitiveCommandHandler extends + TextEditorSortLinesCommandHandler { + + @Override + protected Comparator<String> getComparator() { + return String.CASE_INSENSITIVE_ORDER; + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorSortLinesCaseSensitiveCommandHandler.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorSortLinesCaseSensitiveCommandHandler.java new file mode 100644 index 00000000..ee4cd981 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorSortLinesCaseSensitiveCommandHandler.java @@ -0,0 +1,50 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.text; + +import java.util.Comparator; + +/** + * Action to sort lines case sensitively. + * + * @author Peter Dell + */ +public final class TextEditorSortLinesCaseSensitiveCommandHandler extends + TextEditorSortLinesCommandHandler { + + private static final Comparator<String> CASE_SENSITIVE_COMPARATOR = new CaseSensitiveComparator(); + + private static class CaseSensitiveComparator implements Comparator<String> { + + public CaseSensitiveComparator() { + } + + @Override + public int compare(String o1, String o2) { + return o1.compareTo(o2); + } + } + + @Override + protected Comparator<String> getComparator() { + return CASE_SENSITIVE_COMPARATOR; + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorSortLinesCommandHandler.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorSortLinesCommandHandler.java new file mode 100644 index 00000000..2c3bf79b --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorSortLinesCommandHandler.java @@ -0,0 +1,71 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.text; + +import java.util.Collections; +import java.util.Comparator; +import java.util.List; + +import org.eclipse.core.commands.Command; + +/** + * Base class for text editor actions sorting on the selected lines. + * + * @author Peter Dell + */ +abstract class TextEditorSortLinesCommandHandler extends + TextEditorLinesCommandHandler { + + @Override + protected void process(Command command, List<String> lines) { + if (command == null) { + throw new IllegalArgumentException( + "Parameter 'command' must not be null."); + } + if (lines == null) { + throw new IllegalArgumentException( + "Parameter 'lines' must not be null."); + } + Comparator<String> comparator = getComparator(); + Collections.sort(lines, comparator); + + if (command.getId().endsWith("WithoutDuplicatesCommand")) { + int i = 0; + while (lines.size() > 1 && i < lines.size() - 1) { + String line1 = lines.get(i); + String line2 = lines.get(i + 1); + int result = comparator.compare(line1, line2); + if (result == 0) { + lines.remove(i + 1); + } else { + i++; + } + } + } + } + + /** + * Sub classes must provide a comparator for strings. + * + * @return The comparator, not <code>null</code>. + */ + protected abstract Comparator<String> getComparator(); + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorSortLinesNumericCommandHandler.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorSortLinesNumericCommandHandler.java new file mode 100644 index 00000000..26bc7a0d --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/editor/text/TextEditorSortLinesNumericCommandHandler.java @@ -0,0 +1,81 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.editor.text; + +import java.util.Comparator; +import java.util.regex.Matcher; +import java.util.regex.Pattern; + +import com.wudsn.ide.base.common.StringUtility; + +/** + * Action to sort lines by the first decimal number found starting at first + * position. + * + * @author Peter Dell + */ +public final class TextEditorSortLinesNumericCommandHandler extends + TextEditorSortLinesCommandHandler { + + private static final class NumericComparator implements Comparator<String> { + + private static final Double MAXIMUM = new Double( + Double.POSITIVE_INFINITY); + private static final Pattern numericPattern = Pattern + .compile("[-+]?([0-9]*\\.)?[0-9]+([eE][-+]?[0-9]+)?"); //$NON-NLS-1$ + + public NumericComparator() { + + } + + @Override + public int compare(String o1, String o2) { + return getNumber(o1).compareTo(getNumber(o2)); + } + + private Double getNumber(String text) { + Double result; + + result = MAXIMUM; + + if (text != null && StringUtility.isSpecified(text)) { + try { + Matcher m = numericPattern.matcher(text); + if (m.find()) { + text = text.substring(m.start(), m.end()); + result = Double.valueOf(text); + } + + } catch (NumberFormatException ignore) { + + } + } + return result; + } + } + + private static final NumericComparator COMPARATOR = new NumericComparator(); + + @Override + protected Comparator<String> getComparator() { + return COMPARATOR; + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/Action.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/Action.java new file mode 100644 index 00000000..e600ce86 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/Action.java @@ -0,0 +1,109 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.gui; + +import org.eclipse.swt.events.SelectionEvent; +import org.eclipse.swt.events.SelectionListener; + +/** + * Action class which abstracts from SWT. + * + * @author Peter Dell + * + */ +public final class Action implements SelectionListener { + + private int id; + private ActionListener actionListener; + private boolean enabled; + + /** + * Creates a new action. + * + * @param id + * The action is, a non-negative integer. + * @param actionListener + * The action listener, not <code>null</code>. + */ + public Action(int id, ActionListener actionListener) { + if (id < 0) { + throw new IllegalArgumentException( + "Parameter 'id' must not be negative. Specified value is " + + id + "."); + } + if (actionListener == null) { + throw new IllegalArgumentException( + "Parameter 'actionListener' must not be null."); + } + this.id = id; + this.actionListener = actionListener; + this.enabled = true; + } + + /** + * Gets the id of the action. + * + * @return The id of the action, a non-negative integer. + */ + public int getId() { + return id; + } + + /** + * Sets the enabled state of the action. Disabled actions do not fire. + * + * @param enabled + * <code>true</code> to enable the action, <code>false</code> to + * disable the action. + */ + public void setEnabled(boolean enabled) { + this.enabled = enabled; + + } + + /** + * Callback from SWT, do not use. + */ + @Override + public void widgetDefaultSelected(SelectionEvent e) { + performAction(); + } + + /** + * Callback from SWT, do not use. + */ + @Override + public void widgetSelected(SelectionEvent e) { + performAction(); + + } + + private void performAction() { + if (enabled) { + actionListener.performAction(this); + } + } + + @Override + public String toString() { + return "actionId=" + id; + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/ActionListener.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/ActionListener.java new file mode 100644 index 00000000..da405cf4 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/ActionListener.java @@ -0,0 +1,35 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.gui; + +/** + * Consumers of {@link Action} have to implement this interface. + * @author Peter Dell + * + */ +public interface ActionListener { + + /** + * Perform the operation associated with an action. + * + * @param action The action, not <code>null</code>. + */ + public void performAction(Action action); +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/Application.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/Application.java new file mode 100644 index 00000000..f2a5a5da --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/Application.java @@ -0,0 +1,33 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.gui; + +import com.wudsn.ide.base.gui.MessageManager; + +/** + * Interface to be implemented by applications. + * + * @author Peter Dell + * + */ +public interface Application extends ActionListener{ + + public MessageManager getMessageManager(); +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/CheckBoxField.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/CheckBoxField.java new file mode 100644 index 00000000..9f323dab --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/CheckBoxField.java @@ -0,0 +1,133 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.gui; + +import java.util.ArrayList; +import java.util.List; + +import org.eclipse.swt.SWT; +import org.eclipse.swt.widgets.Button; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Control; +import org.eclipse.swt.widgets.Label; + +/** + * Check box field for boolean value. + * + * @author Peter Dell + * + * @since 1.6.0 + */ +public final class CheckBoxField extends Field { + + private Button button; + private List<Action> selectionActions; + + /** + * Creates a check box field. + * + * @param parent + * The parent composite, not <code>null</code>. + * @param labelText + * The label text, not <code>null</code>. + * @param style + * The SWT style. + */ + public CheckBoxField(Composite parent, String labelText, int style) { + if (parent == null) { + throw new IllegalArgumentException( + "Parameter 'parent' must not be null."); + } + if (labelText == null) { + throw new IllegalArgumentException( + "Parameter 'labelText' must not be null."); + } + + label = new Label(parent, SWT.NONE); + label.setText(labelText); + button = new Button(parent, SWT.CHECK | style); + selectionActions = new ArrayList<Action>(1); + } + + /** + * {@inheritDoc} + */ + @Override + public Control getControl() { + return button; + } + + public void setVisible(boolean visible) { + label.setVisible(visible); + button.setVisible(visible); + } + + /** + * {@inheritDoc} + */ + @Override + public void setEnabled(boolean enabled) { + label.setEnabled(enabled); + } + + /** + * {@inheritDoc} + */ + @Override + public void setEditable(boolean editable) { + button.setEnabled(editable); + } + + /** + * Sets the value. + * + * @param value + * The value. + */ + public void setValue(boolean value) { + button.setSelection(value); + } + + /** + * Gets the value. + * + * @return The value. + */ + public boolean getValue() { + + return button.getSelection(); + } + + /** + * Adds a selection action which is fire when the field content changes. + * + * @param action + * The selection action, not <code>null</code>. + */ + public void addSelectionAction(Action action) { + if (action == null) { + throw new IllegalArgumentException( + "Parameter 'action' must not be null."); + } + selectionActions.add(action); + button.addSelectionListener(action); + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/EnumField.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/EnumField.java new file mode 100644 index 00000000..f5944731 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/EnumField.java @@ -0,0 +1,226 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ +package com.wudsn.ide.base.gui; + +import java.util.ArrayList; +import java.util.List; +import java.util.Locale; +import java.util.MissingResourceException; +import java.util.ResourceBundle; + +import org.eclipse.swt.SWT; +import org.eclipse.swt.widgets.Combo; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Control; +import org.eclipse.swt.widgets.Label; + +import com.wudsn.ide.base.BasePlugin; + +/** + * Combo field offering the elements of an enum class in generic and type safe + * manner. + * + * @author Peter Dell + * + * @param <T> + * The enum class. + */ +public final class EnumField<T extends Enum<?>> extends Field { + + private static final class Entry<T> { + public T constant; + public String name; + public String text; + + public Entry() { + } + + } + + private Combo combo; + private List<Entry<T>> entries; + + /** + * Creates a new enum field. + * + * @param parent + * The parent composite, not <code>null</code>. + * @param labelText + * The label text, not <code>null</code>. + * @param enumClass + * The enum class, not <code>null</code>. + * @param visibleValues + * The non-empty array containing the subset of enum values to be + * displayed or <code>null</code> to display all values. + */ + public EnumField(Composite parent, String labelText, + Class<? extends T> enumClass, T visibleValues[]) { + if (parent == null) { + throw new IllegalArgumentException( + "Parameter 'parent' must not be null."); + } + if (labelText == null) { + throw new IllegalArgumentException( + "Parameter 'labelText' must not be null."); + } + if (enumClass == null) { + throw new IllegalArgumentException( + "Parameter 'enumClass' must not be null."); + } + if (visibleValues != null && visibleValues.length == 0) { + throw new IllegalArgumentException( + "Parameter 'visibleValues' must not be empty"); + } + label = new Label(parent, SWT.NONE); + label.setText(labelText); + combo = new Combo(parent, SWT.DROP_DOWN); + + ResourceBundle resourceBundle; + resourceBundle = ResourceBundle.getBundle("plugin", + Locale.getDefault(), enumClass.getClassLoader()); + + T[] constants = enumClass.getEnumConstants(); + + if (constants.length == 0) { + throw new IllegalArgumentException("Enum class '" + enumClass + + "' has no constants."); + } + + // Ensure all visible values are contained in the set of constants. + if (visibleValues != null) { + for (int i = 0; i < visibleValues.length; i++) { + T visibleValue = visibleValues[i]; + boolean found = false; + for (int j = 0; j < constants.length; j++) { + if (constants[j] == visibleValue) { + found = true; + continue; + } + } + if (!found) { + throw new IllegalArgumentException( + "Parameter 'visibleValues' contain the undefined value '" + + visibleValue + "'."); + } + } + constants = visibleValues; + } + + // Read the localized texts and create the entries. + entries = new ArrayList<Entry<T>>(constants.length); + for (int i = 0; i < constants.length; i++) { + Entry<T> entry = new Entry<T>(); + entry.constant = constants[i]; + entry.name = constants[i].name(); + String key = enumClass.getName() + "." + entry.name; + try { + entry.text = resourceBundle.getString(key); + } catch (MissingResourceException ex) { + entry.text = entry.name + " - Text missing"; + BasePlugin.getInstance().logError( + "Resource for enum value {0} is missing.", + new Object[] { key }, ex); + } + + entries.add(entry); + } + + for (Entry<T> entry : entries) { + combo.add(entry.text); + } + combo.select(0); + + } + + /** + * {@inheritDoc} + */ + @Override + public Control getControl() { + return combo; + } + + /** + * {@inheritDoc} + */ + @Override + public void setEnabled(boolean enabled) { + label.setEnabled(enabled); + combo.setEnabled(enabled); + + } + + /** + * {@inheritDoc} + */ + @Override + public void setEditable(boolean editable) { + // There is only a style SWT#READ_ONLY, but no changeable property + } + + /** + * Sets the value. + * + * @param value + * The value, not <code>null</code>. + */ + public void setValue(T value) { + if (value == null) { + throw new IllegalArgumentException( + "Parameter 'value' must not be null."); + } + for (int i = 0; i < entries.size(); i++) { + if (value.name().equals(entries.get(i).name)) { + combo.select(i); + break; + } + } + } + + /** + * Gets the value. + * + * @return The value, not <code>null</code>. + */ + public T getValue() { + int index; + index = combo.getSelectionIndex(); + if (index == -1) { + throw new IllegalStateException("No item selected"); + } + T result = entries.get(index).constant; + return result; + } + + /** + * Adds a selection action. + * + * @param action + * The selection action, not <code>null</code>. + */ + public void addSelectionAction(Action action) { + if (action == null) { + throw new IllegalArgumentException( + "Parameter 'action' must not be null."); + } + combo.addSelectionListener(action); + + } + +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/Field.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/Field.java new file mode 100644 index 00000000..babffef2 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/Field.java @@ -0,0 +1,100 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.gui; + +import javax.swing.event.ChangeEvent; +import javax.swing.event.ChangeListener; + +import org.eclipse.swt.widgets.Control; +import org.eclipse.swt.widgets.Label; + +/** + * Base class for all fields. + * + * @author Peter Dell + * + */ +public abstract class Field { + + protected Label label; + private ChangeListener changeListener; + + /** + * Creation is protected. + */ + protected Field() { + + } + + /** + * Gets the label for the field. + * + * @return The label, not <code>null</code>. + */ + public final Label getLabel() { + if (label == null) { + throw new IllegalStateException("Label not yet created."); + } + return label; + } + + /** + * Gets the control relevant for messages decorations. + * + * @return The control relevant for messages decorations, not + * <code>null</code>. + */ + public abstract Control getControl(); + + /** + * Sets the enabled state of the field. + * + * @param enabled + * The enabled state of the field. + */ + public abstract void setEnabled(boolean enabled); + + /** + * Sets the editable state of the field. + * + * @param editable + * The editable state of the field. + */ + public abstract void setEditable(boolean editable); + + /** + * Adds a change listener to this field. + * + * @param changeListener + * The change listener, not <code>null</code>. + */ + public final void addChangeListener(ChangeListener changeListener) { + if (changeListener == null) { + throw new IllegalArgumentException("Parameter 'changeListener' must not be null."); + } + this.changeListener = changeListener; + } + + protected final void notifyChangeListenner() { + if (changeListener != null) { + changeListener.stateChanged(new ChangeEvent(this)); + } + } +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/FilePathField.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/FilePathField.java new file mode 100644 index 00000000..e16e58e5 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/FilePathField.java @@ -0,0 +1,277 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.gui; + +import java.util.ArrayList; +import java.util.List; + +import org.eclipse.core.resources.IFile; +import org.eclipse.core.resources.IResource; +import org.eclipse.core.resources.IWorkspaceRoot; +import org.eclipse.core.resources.ResourcesPlugin; +import org.eclipse.core.runtime.IPath; +import org.eclipse.core.runtime.Path; +import org.eclipse.jface.viewers.TreePath; +import org.eclipse.jface.window.Window; +import org.eclipse.swt.SWT; +import org.eclipse.swt.events.DisposeEvent; +import org.eclipse.swt.events.DisposeListener; +import org.eclipse.swt.events.ModifyEvent; +import org.eclipse.swt.events.ModifyListener; +import org.eclipse.swt.events.SelectionAdapter; +import org.eclipse.swt.events.SelectionEvent; +import org.eclipse.swt.layout.GridData; +import org.eclipse.swt.widgets.Button; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Control; +import org.eclipse.swt.widgets.Label; +import org.eclipse.swt.widgets.Text; +import org.eclipse.ui.dialogs.ElementTreeSelectionDialog; +import org.eclipse.ui.model.BaseWorkbenchContentProvider; +import org.eclipse.ui.model.WorkbenchLabelProvider; + +import com.wudsn.ide.base.Texts; +import com.wudsn.ide.base.common.IPathUtility; +import com.wudsn.ide.base.common.TextUtility; + +/** + * Text field with a build in file browser button. + * + * @author Peter Dell + * + */ +public final class FilePathField extends Field { + + /** + * Inner class to handle the file selection dialog. + */ + private final class BrowseButtonSelectionAdapter extends SelectionAdapter { + + public BrowseButtonSelectionAdapter() { + } + + @Override + public void widgetSelected(SelectionEvent evt) { + + IPath newValue = getResourcePath(); + if (newValue != null) { + newValue = IPathUtility.makeRelative(newValue, filePathPrefix); + // Set field content and notify change listeners. + filePathField.setText(newValue.toPortableString()); + } + } + + /** + * Helper to open the resource selection dialog. + * + * @return The file name the user selected or <code>null</code> if not. + */ + private IPath getResourcePath() { + + IWorkspaceRoot workspaceRoot = ResourcesPlugin.getWorkspace().getRoot(); + + ElementTreeSelectionDialog dialog = new ElementTreeSelectionDialog(browseButton.getShell(), + new WorkbenchLabelProvider(), new BaseWorkbenchContentProvider()); + + dialog.setTitle(TextUtility.format(Texts.FILE_PATH_FIELD_DIALOG_MESSAGE, label.getText())); + dialog.setInput(workspaceRoot); + IPath filePath = new Path(filePathField.getText()); + filePath = IPathUtility.makeAbsolute(filePath, filePathPrefix, true); + IFile ifile = workspaceRoot.getFile(filePath); + // If file is not there, default to its parent (folder). + Object selection; + if (ifile != null && ifile.exists()) { + selection = ifile; + } else { + int size = filePath.segmentCount(); + List<IResource> resources = new ArrayList<IResource>(size); + IResource resource = ifile; + while (resource != null) { + resource = resource.getParent(); + if (resource != null) { + resources.add(resource); + } + } + selection = new TreePath(resources.toArray(new IResource[resources.size()])); + } + dialog.setInitialSelection(selection); + + if (dialog.open() == Window.OK) { + Object[] result = dialog.getResult(); + if (result != null && result.length > 0) { + ifile = (IFile) result[0]; + return ifile.getFullPath(); + } + + } + return null; + } + } + + /** + * Constructor fields. + */ + final int dialogMode; + final Text filePathField; + final ModifyListener filePathFieldModifyListener; + + /** + * State fields. + */ + private boolean enabled; + private boolean editable; + + /** + * Runtime fields. + */ + Button browseButton; + IPath filePathPrefix; + + /** + * Creates a new file path field. + * + * @param parent + * The parent composite, not <code>null</code>. + * @param labelText + * The label text, not <code>null</code>. + * @param dialogMode + * {@link SWT#OPEN} or e {@link SWT#SAVE}. + */ + public FilePathField(Composite parent, String labelText, int dialogMode) { + if (parent == null) { + throw new IllegalArgumentException("Parameter 'parent' must not be null."); + } + if (dialogMode != SWT.OPEN && dialogMode != SWT.SAVE) { + throw new IllegalArgumentException( + "Parameter 'dialogMode' must be 'SWT.OPEN' or 'SWT.SAVE'. Specified value is " + dialogMode + "."); + } + this.dialogMode = dialogMode; + + label = new Label(parent, SWT.NONE); + label.setText(labelText); + + filePathField = new Text(parent, SWT.SINGLE | SWT.BORDER); + filePathField.setLayoutData(new GridData(GridData.FILL_HORIZONTAL)); + filePathFieldModifyListener = new ModifyListener() { + + @Override + public void modifyText(ModifyEvent e) { + FilePathField.this.notifyChangeListenner(); + + } + }; + enabled = true; + editable = true; + + browseButton = new Button(parent, SWT.PUSH); + browseButton.setText(Texts.FILE_PATH_FIELD_BROWSE_BUTTON_LABEL); + + browseButton.addSelectionListener(new BrowseButtonSelectionAdapter()); + browseButton.addDisposeListener(new DisposeListener() { + @Override + public void widgetDisposed(DisposeEvent event) { + browseButton = null; + } + }); + + filePathPrefix = new Path(""); + } + + /** + * {@inheritDoc} + */ + @Override + public Control getControl() { + return filePathField; + } + + public void setLabelText(String labelText) { + label.setText(labelText); + + } + + public void setVisible(boolean visible) { + label.setVisible(visible); + filePathField.setVisible(visible); + browseButton.setVisible(visible); + } + + /** + * {@inheritDoc} + */ + @Override + public void setEnabled(boolean enabled) { + this.enabled = enabled; + label.setEnabled(enabled); + filePathField.setEnabled(enabled); + filePathField.setEditable(enabled & editable); + browseButton.setEnabled(enabled); + } + + /** + * {@inheritDoc} + */ + @Override + public void setEditable(boolean editable) { + this.editable = editable; + filePathField.setEditable(enabled & editable); + } + + /** + * Sets the path prefix which will be stripped automatically if it is the + * prefix of the user input. + * + * @param filePathPrefix + * The path prefix, may be empty, not <code>null</code>. + */ + public void setFilePathPrefix(IPath filePathPrefix) { + if (filePathPrefix == null) { + throw new IllegalArgumentException("Parameter 'filePathPrefix' must not be null."); + } + this.filePathPrefix = filePathPrefix; + } + + /** + * Sets the value. + * + * @param value + * The value, not <code>null</code>. + */ + public void setValue(String value) { + if (value == null) { + throw new IllegalArgumentException("Parameter 'value' must not be null."); + } + filePathField.removeModifyListener(filePathFieldModifyListener); + filePathField.setText(value); + filePathField.addModifyListener(filePathFieldModifyListener); + } + + /** + * Gets the value. + * + * @return The value, not <code>null</code>. + */ + public String getValue() { + String result = filePathField.getText(); + result = result.trim(); + return result; + + } +} \ No newline at end of file diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/IntegerField.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/IntegerField.java new file mode 100644 index 00000000..46b48de0 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/IntegerField.java @@ -0,0 +1,291 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.gui; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; + +import org.eclipse.swt.SWT; +import org.eclipse.swt.events.ModifyEvent; +import org.eclipse.swt.events.ModifyListener; +import org.eclipse.swt.widgets.Combo; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Control; +import org.eclipse.swt.widgets.Label; +import org.eclipse.swt.widgets.Text; + +import com.wudsn.ide.base.common.HexUtility; +import com.wudsn.ide.base.common.NumberUtility; + +/** + * Numeric field for integers. + * + * TODO Have explicit Hex Mode/sub class with minimum digi number for 16 bit + * addresses + * + * @author Peter Dell + * + */ +public final class IntegerField extends Field { + + private int[] defaultValues; + private boolean hexMode; + private int digitLength; + + private Combo combo; + private Text text; + private List<Action> selectionActions; + + /** + * Creates a integer field. + * + * @param parent + * The parent composite, not <code>null</code>. + * @param labelText + * The label text, not <code>null</code>. + * @param defaultValues + * The array of default values or <code>null</code>. + * @param digitLength + * The minimum digit length, see {@link NumberUtility} and + * {@link HexUtility}. + * @param hexMode + * <code>true</code> if display and value help shall be in hex + * mode + * @param style + * The SWT style. + */ + public IntegerField(Composite parent, String labelText, + int[] defaultValues, boolean hexMode, int digitLength, int style) { + if (parent == null) { + throw new IllegalArgumentException( + "Parameter 'parent' must not be null."); + } + if (labelText == null) { + throw new IllegalArgumentException( + "Parameter 'labelText' must not be null."); + } + this.hexMode = hexMode; + this.digitLength = digitLength; + + label = new Label(parent, SWT.NONE); + label.setText(labelText); + + ModifyListener modifyListener= new ModifyListener() { + + @Override + public void modifyText(ModifyEvent e) { + IntegerField.this.notifyChangeListenner(); + + } + }; + + if (defaultValues != null) { + if (defaultValues.length == 0) { + throw new IllegalArgumentException( + "Parameter 'defaultValues0' must not be empty."); + } + combo = new Combo(parent, SWT.DROP_DOWN | style); + + setDefaultValues(defaultValues); + combo.select(0); + combo.addModifyListener(modifyListener); + + } else { + text = new Text(parent, style); + text.addModifyListener(modifyListener); + } + + + selectionActions = new ArrayList<Action>(1); + } + + /** + * {@inheritDoc} + */ + @Override + public Control getControl() { + if (combo != null) { + return combo; + } + return text; + } + + public void setVisible(boolean visible) { + label.setVisible(visible); + if (combo != null) { + combo.setVisible(visible); + } else { + text.setVisible(visible); + } + } + + /** + * {@inheritDoc} + */ + @Override + public void setEnabled(boolean enabled) { + label.setEnabled(enabled); + if (combo != null) { + combo.setEnabled(enabled); + } else { + text.setEnabled(enabled); + } + + } + + /** + * {@inheritDoc} + */ + @Override + public void setEditable(boolean editable) { + if (combo != null) { + // There is only an SWT.READ_ONLY style but not property + } else { + text.setEditable(editable); + } + + } + + public void setDefaultValues(int[] defaultValues) { + if (defaultValues == null) { + throw new IllegalArgumentException( + "Parameter 'defaultValues' must not be null."); + } + + if (defaultValues.length == 0) { + throw new IllegalArgumentException( + "Parameter 'defaultValues' must not be empty."); + } + + if (this.defaultValues != null) { + if (Arrays.equals(defaultValues, this.defaultValues)) { + return; + } + } + combo.removeAll(); + + // Compute maxmimum. + int max = 0; + for (int defaultValue : defaultValues) { + if (defaultValue > max) { + max = defaultValue; + } + } + + for (int defaultValue : defaultValues) { + String textValue; + if (hexMode) { + textValue = HexUtility.getLongValueHexString(defaultValue, + digitLength); + } else { + textValue = NumberUtility.getLongValueDecimalString( + defaultValue, digitLength); + } + combo.add(textValue); + } + this.defaultValues = defaultValues; + } + + /** + * Sets the value. + * + * @param value + * The value. + */ + public void setValue(int value) { + String textValue; + if (hexMode) { + textValue = HexUtility.getLongValueHexString(value, digitLength); + } else { + textValue = NumberUtility.getLongValueDecimalString(value, + digitLength); + } + for (Action action : selectionActions) { + action.setEnabled(false); + } + setText(textValue); + for (Action action : selectionActions) { + action.setEnabled(true); + } + } + + private void setText(String value) { + if (value == null) { + throw new IllegalArgumentException( + "Parameter 'value' must not be null."); + } + + if (combo != null) { + combo.setText(value); + } else { + text.setText(value); + } + } + + /** + * Gets the value. + * + * @return The value. + */ + public int getValue() { + int result; + String textValue = getText().toLowerCase(); + + try { + if (hexMode) { + result = Integer.parseInt(textValue, 16); + } else { + result = Integer.parseInt(textValue); + + } + } catch (NumberFormatException ex) { + result = 0; + } + + return result; + } + + private String getText() { + String result; + if (combo != null) { + result = combo.getText(); + } else { + result = text.getText(); + } + return result; + } + + /** + * Adds a selection action which is fired when the field content changes. + * + * @param action + * The selection action, not <code>null</code>. + */ + public void addSelectionAction(Action action) { + if (action == null) { + throw new IllegalArgumentException( + "Parameter 'action' must not be null."); + } + selectionActions.add(action); + combo.addSelectionListener(action); + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/MessageManager.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/MessageManager.java new file mode 100644 index 00000000..491323a8 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/MessageManager.java @@ -0,0 +1,370 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.gui; + +import java.util.ArrayList; +import java.util.List; + +import org.eclipse.core.runtime.CoreException; +import org.eclipse.core.runtime.IStatus; +import org.eclipse.jface.action.IStatusLineManager; +import org.eclipse.jface.fieldassist.ControlDecoration; +import org.eclipse.jface.fieldassist.FieldDecorationRegistry; +import org.eclipse.swt.SWT; +import org.eclipse.swt.graphics.Color; +import org.eclipse.swt.graphics.Image; +import org.eclipse.swt.widgets.Control; +import org.eclipse.swt.widgets.Display; +import org.eclipse.ui.IEditorPart; +import org.eclipse.ui.IViewPart; +import org.eclipse.ui.IWorkbenchPart; + +import com.wudsn.ide.base.BasePlugin; +import com.wudsn.ide.base.common.TextUtility; + +/** + * Message manager for an work benach part of type {@link IEditorPart} or + * {@link IViewPart}. + * + * @author Peter Dell + * + */ +public final class MessageManager { + + private final static class FieldRegistryEntry { + + private Field field; + private int messageId; + + private ControlDecoration controlDecoration; + + public FieldRegistryEntry(Field field, int messageId) { + if (field == null) { + throw new IllegalArgumentException( + "Parameter 'field' must not be null."); + } + if (messageId < 0) { + throw new IllegalArgumentException( + "Parameter 'messageId' must not be negative."); + } + this.field = field; + this.messageId = messageId; + + Control control = field.getControl(); + + controlDecoration = new ControlDecoration(control, SWT.RIGHT + | SWT.TOP); + controlDecoration.setShowHover(true); + } + + public Field getField() { + return field; + } + + public int getMessageId() { + return messageId; + } + + public ControlDecoration getControlDecoration() { + return controlDecoration; + + } + } + + private static final class MessageQueueEntry { + + private int messageId; + private int severity; + private String message; + private String[] parameters; + private Throwable throwable; + + public MessageQueueEntry(int messageId, int severity, String message, + String[] parameters, Throwable throwable) { + this.messageId = messageId; + this.severity = severity; + this.message = message; + this.parameters = parameters; + this.throwable = throwable; + } + + public int getMessageId() { + return messageId; + } + + public int getSeverity() { + return severity; + } + + public String getMessage() { + return message; + } + + public String[] getParameters() { + return parameters; + } + + public Throwable getThrowable() { + return throwable; + } + + } + + private IWorkbenchPart workbenchPart; + + private IStatusLineManager statusLineManager; + + private List<FieldRegistryEntry> fieldRegistryEntries; + + private List<MessageQueueEntry> messageQueueEntries; + private boolean messageQueueError; + + private Color yellow; + private Color red; + + public MessageManager(IWorkbenchPart workbenchPart) { + if (workbenchPart == null) { + throw new IllegalArgumentException( + "Parameter 'workbenchPart' must not be null."); + } + this.workbenchPart = workbenchPart; + + fieldRegistryEntries = new ArrayList<FieldRegistryEntry>(); + messageQueueEntries = new ArrayList<MessageQueueEntry>(); + messageQueueError = false; + + yellow = new Color(Display.getDefault(), 0, 255, 255); + red = new Color(Display.getDefault(), 255, 0, 0); + + } + + public void dispose() { + for (FieldRegistryEntry fieldRegistryEntry : fieldRegistryEntries) { + ControlDecoration controlDecoration; + controlDecoration = fieldRegistryEntry.getControlDecoration(); + controlDecoration.hide(); + controlDecoration.dispose(); + } + fieldRegistryEntries.clear(); + + yellow.dispose(); + red.dispose(); + } + + private void initStatusLineManager() { + if (statusLineManager == null) { + if (workbenchPart instanceof IEditorPart) { + statusLineManager = ((IEditorPart) workbenchPart) + .getEditorSite().getActionBars().getStatusLineManager(); + } else if (workbenchPart instanceof IViewPart) { + statusLineManager = ((IViewPart) workbenchPart).getViewSite() + .getActionBars().getStatusLineManager(); + } else { + throw new IllegalStateException("Workbench part " + + workbenchPart + " has an unsupported type."); + } + } + } + + public void registerField(Field field, int messageId) { + if (field == null) { + throw new IllegalArgumentException( + "Parameter 'field' must not be null."); + } + if (messageId < 0) { + throw new IllegalArgumentException( + "Parameter 'messageId' must not be negative."); + } + FieldRegistryEntry fieldRegistryEntry = new FieldRegistryEntry(field, + messageId); + fieldRegistryEntries.add(fieldRegistryEntry); + } + + public void clearMessages() { + messageQueueEntries.clear(); + messageQueueError = false; + + for (FieldRegistryEntry fieldRegistryEntry : fieldRegistryEntries) { + + Control control = fieldRegistryEntry.getField().getControl(); + control.setForeground(control.getParent().getForeground()); + ControlDecoration controlDecoration; + controlDecoration = fieldRegistryEntry.getControlDecoration(); + controlDecoration.hide(); + } + + initStatusLineManager(); + statusLineManager.setMessage(null); + statusLineManager.setErrorMessage(null); + + } + + /** + * Sends a message to the message queue. + * + * @param messageId + * The message id identifying the target UI element of the + * message. + * @param severity + * The severity, see {@link IStatus#INFO},{@link IStatus#WARNING} + * , {@link IStatus#ERROR}. + * @param message + * The message text, not <code>null</code>. + * @param parameters + * The message parameters, may be empty or null. + */ + public void sendMessage(int messageId, int severity, String message, + String... parameters) { + if (message == null) { + throw new IllegalArgumentException( + "Parameter 'message' must not be null."); + } + MessageQueueEntry messageQueueEntry; + messageQueueEntry = new MessageQueueEntry(messageId, severity, message, + parameters, null); + addMessageQueueEntry(messageQueueEntry); + } + + /** + * Sends a message to the message queue based on a core exception. + * + * @param messageId + * The message id identifying the target UI element of the + * message. + * @param coreException + * The core exception with the status and text information, not + * <code>null</code>. + */ + public void sendMessage(int messageId, CoreException coreException) { + if (coreException == null) { + throw new IllegalArgumentException( + "Parameter 'coreException' must not be null."); + } + MessageQueueEntry messageQueueEntry; + messageQueueEntry = new MessageQueueEntry(messageId, coreException + .getStatus().getSeverity(), coreException.getStatus() + .getMessage(), null, coreException); + addMessageQueueEntry(messageQueueEntry); + + } + + private void addMessageQueueEntry(MessageQueueEntry messageQueueEntry) { + if (messageQueueEntry == null) { + throw new IllegalArgumentException( + "Parameter 'messageQueueEntry' must not be null."); + } + messageQueueEntries.add(messageQueueEntry); + if (messageQueueEntry.getSeverity() == IStatus.ERROR) { + messageQueueError = true; + } + } + + public boolean containsError() { + return messageQueueError; + } + + public void displayMessages() { + + initStatusLineManager(); + + FieldDecorationRegistry fieldDecorationRegistry = FieldDecorationRegistry + .getDefault(); + Image informationImage = fieldDecorationRegistry.getFieldDecoration( + FieldDecorationRegistry.DEC_INFORMATION).getImage(); + Image warningImage = fieldDecorationRegistry.getFieldDecoration( + FieldDecorationRegistry.DEC_INFORMATION).getImage(); + Image errorImage = fieldDecorationRegistry.getFieldDecoration( + FieldDecorationRegistry.DEC_ERROR).getImage(); + + for (MessageQueueEntry messageQueueEntry : messageQueueEntries) { + + String messageText = TextUtility.format( + messageQueueEntry.getMessage(), + messageQueueEntry.getParameters()); + + BasePlugin plugin = BasePlugin.getInstance(); + + switch (messageQueueEntry.getSeverity()) { + case IStatus.OK: + statusLineManager.setMessage(messageText); + + plugin.log(messageQueueEntry.getMessage(), + messageQueueEntry.getParameters()); + break; + + case IStatus.INFO: + case IStatus.WARNING: + case IStatus.ERROR: + Image image; + switch (messageQueueEntry.getSeverity()) { + case IStatus.INFO: + image = informationImage; + break; + case IStatus.WARNING: + image = warningImage; + break; + case IStatus.ERROR: + image = errorImage; + break; + default: + image = null; + } + for (FieldRegistryEntry fieldRegistryEntry : fieldRegistryEntries) { + if (fieldRegistryEntry.getMessageId() == messageQueueEntry + .getMessageId()) { + + ControlDecoration controlDecoration; + Control control; + control = fieldRegistryEntry.getField().getControl(); + + if (messageQueueEntry.getSeverity() == IStatus.ERROR) { + control.setForeground(red); + } else { + control.setForeground(control.getParent() + .getForeground()); + } + + controlDecoration = fieldRegistryEntry + .getControlDecoration(); + + controlDecoration.setImage(image); + controlDecoration.setDescriptionText(messageText); + controlDecoration.show(); + } + } + if (messageQueueEntry.getSeverity() == IStatus.ERROR) { + statusLineManager.setErrorMessage(messageText); + } else { + statusLineManager.setMessage(messageText); + + } + plugin.logError(messageQueueEntry.getMessage(), + messageQueueEntry.getParameters(), + messageQueueEntry.getThrowable()); + break; + default: + throw new IllegalStateException("Severity " + + messageQueueEntry.getSeverity() + + " is not supported."); + } + + } + } +} diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/MultiLineTextField.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/MultiLineTextField.java new file mode 100644 index 00000000..9ffcc6b8 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/MultiLineTextField.java @@ -0,0 +1,210 @@ +/** + * Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.gui; + +import org.eclipse.swt.SWT; +import org.eclipse.swt.custom.Bullet; +import org.eclipse.swt.custom.CaretEvent; +import org.eclipse.swt.custom.CaretListener; +import org.eclipse.swt.custom.LineBackgroundEvent; +import org.eclipse.swt.custom.LineBackgroundListener; +import org.eclipse.swt.custom.LineStyleEvent; +import org.eclipse.swt.custom.LineStyleListener; +import org.eclipse.swt.custom.ST; +import org.eclipse.swt.custom.StyleRange; +import org.eclipse.swt.custom.StyledText; +import org.eclipse.swt.events.ModifyEvent; +import org.eclipse.swt.events.ModifyListener; +import org.eclipse.swt.graphics.Color; +import org.eclipse.swt.graphics.GlyphMetrics; +import org.eclipse.swt.graphics.RGB; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Control; +import org.eclipse.swt.widgets.Display; + +/** + * Multi line text field without label. + * + * @author Peter Dell + * + * @since 1.6.0 + */ +public final class MultiLineTextField extends Field { + + static final Color LINE_NUMBER_COLOR = new Color(Display.getCurrent(), new RGB(0, 0, 255)); + static final Color LINE_HIGHLIGHT_COLOR = new Color(Display.getCurrent(), new RGB(220, 220, 255)); + + Color parentBackground; + Color ownBackground; + private StyledText text; + private ModifyListener textModifyListener; + + /** + * Creates a text field. + * + * @param parent + * The parent composite, not <code>null</code>. + * @param style + * The SWT style. + */ + public MultiLineTextField(final Composite parent, int style) { + if (parent == null) { + throw new IllegalArgumentException("Parameter 'parent' must not be null."); + } + + parentBackground = parent.getBackground(); + text = new StyledText(parent, style | SWT.MULTI); + ownBackground = text.getBackground(); + + text.addCaretListener(new CaretListener() { + + @Override + public void caretMoved(CaretEvent event) { + StyledText text = (StyledText) event.widget; + text.redraw(); + } + }); + text.addLineStyleListener(new LineStyleListener() { + @Override + public void lineGetStyle(LineStyleEvent event) { + // Set the line number + StyledText text = (StyledText) event.widget; + event.bulletIndex = text.getLineAtOffset(event.lineOffset); + + // Set the style, 12 pixels wide for each digit + StyleRange style = new StyleRange(); + style.metrics = new GlyphMetrics(0, 0, Integer.toString(text.getLineCount() + 1).length() * 12); + style.foreground = LINE_NUMBER_COLOR; + + // Create and set the bullet + event.bullet = new Bullet(ST.BULLET_NUMBER, style); + } + }); + + text.addLineBackgroundListener(new LineBackgroundListener() { + + @Override + public void lineGetBackground(LineBackgroundEvent event) { + StyledText text = (StyledText) event.widget; + if (text.getEditable()) { + if (text.getLineAtOffset(event.lineOffset) == text.getLineAtOffset(text.getCaretOffset())) { + event.lineBackground = LINE_HIGHLIGHT_COLOR; + } else { + event.lineBackground = ownBackground; + } + } else { + event.lineBackground = parentBackground; + } + } + }); + + textModifyListener = new ModifyListener() { + @Override + public void modifyText(ModifyEvent e) { + MultiLineTextField.this.notifyChangeListenner(); + } + }; + } + + /** + * Gets the text control representing this text field. + * + * @return The text control, not <code>null</code>. + */ + public StyledText getText() { + return text; + } + + /** + * {@inheritDoc} + */ + @Override + public Control getControl() { + return text; + } + + public void setVisible(boolean visible) { + text.setVisible(visible); + + } + + /** + * {@inheritDoc} + */ + @Override + public void setEnabled(boolean enabled) { + text.setEnabled(enabled); + } + + /** + * {@inheritDoc} + */ + @Override + public void setEditable(boolean editable) { + text.setEditable(editable); + text.setBackground(editable ? ownBackground : parentBackground); + } + + /** + * Sets the value. + * + * @param value + * The value, not <code>null</code>. + */ + public void setValue(String value) { + if (value == null) { + throw new IllegalArgumentException("Parameter 'value' must not be null."); + } + if (!value.equals(text.getText())) { + text.removeModifyListener(textModifyListener); + text.setText(value); + text.addModifyListener(textModifyListener); + } + } + + /** + * Gets the value. + * + * @return The value, not <code>null</code>. + */ + public String getValue() { + String result; + result = text.getText(); + result = result.trim(); + return result; + + } + + /** + * Sets the selection. + * <p> + * Indexing is zero based. The range of a selection is from 0..N where N is + * the number of characters in the widget. + * + * @param start + * new caret position. + */ + + public void setSelection(int start) { + text.setSelection(start); + text.setFocus(); + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/SWTFactory.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/SWTFactory.java new file mode 100644 index 00000000..dbb5507e --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/SWTFactory.java @@ -0,0 +1,133 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.gui; + +import org.eclipse.swt.SWT; +import org.eclipse.swt.layout.GridData; +import org.eclipse.swt.layout.GridLayout; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Group; +import org.eclipse.swt.widgets.Label; + +/** + * Utility class for creating SWT components. Based on the SWT factory of the + * JDT. + * + * @author Peter Dell + */ +public final class SWTFactory { + private SWTFactory() { + } + + /** + * Creates a composite that uses the parent's font and has a grid layout + * + * @param parent + * The parent to add the composite to, not <code>null</code>. + * @param columns + * The number of columns the composite should have. + * @param hspan + * The horizontal span the new composite should take up in the + * parent. + * @param style + * The fill style of the composite {@link GridData#GridData(int)} + * . + * @return The new composite with a grid layout, not <code>null</code>. + */ + public static Composite createComposite(Composite parent, int columns, + int hspan, int style) { + if (parent == null) { + throw new IllegalArgumentException( + "Parameter 'parent' must not be null."); + } + Composite composite = new Composite(parent, SWT.NONE); + GridLayout gridLayout = new GridLayout(columns, false); + gridLayout.marginWidth = 0; + composite.setLayout(gridLayout); + composite.setFont(parent.getFont()); + GridData gd = new GridData(style); + gd.horizontalSpan = hspan; + composite.setLayoutData(gd); + return composite; + } + + /** + * Creates a group that uses the parent's font and has a grid layout + * + * @param parent + * The parent to add the composite to, not <code>null</code>. + * @param text + * The group title, not <code>null</code>. + * @param columns + * The number of columns the composite should have. + * @param hspan + * The horizontal span the new composite should take up in the + * parent. + * @param fill + * The fill style of the composite {@link GridData}. + * @return The new composite with a grid layout, not <code>null</code>. + */ + public static Group createGroup(Composite parent, String text, int columns, + int hspan, int fill) { + if (parent == null) { + throw new IllegalArgumentException( + "Parameter 'parent' must not be null."); + } + if (text == null) { + throw new IllegalArgumentException( + "Parameter 'text' must not be null."); + } + Group group = new Group(parent, SWT.NONE); + GridLayout layout = new GridLayout(columns, true); + group.setLayout(layout); + group.setText(text); + group.setFont(parent.getFont()); + GridData gd = new GridData(fill); + gd.horizontalSpan = hspan; + group.setLayoutData(gd); + return group; + } + + /** + * Create a number of labels to fill up the layout. + * + * @param composite + * The composite which is used as parent for the label, not + * <code>null</code>. + * @param number + * The number of labels to be created, a positive integer. + */ + @SuppressWarnings("unused") + public static void createLabels(Composite composite, int number) { + if (composite == null) { + throw new IllegalArgumentException( + "Parameter 'composite' must not be null."); + } + if (number < 1) { + throw new IllegalArgumentException( + "Parameter 'number' must not be positive. Specified value is " + + number + "."); + } + for (int i = 0; i < number; i++) { + new Label(composite, SWT.NONE); + } + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/TextField.java b/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/TextField.java new file mode 100644 index 00000000..e018cb07 --- /dev/null +++ b/com.wudsn.ide.base/src/com/wudsn/ide/base/gui/TextField.java @@ -0,0 +1,133 @@ +/** +* Copyright (C) 2009 - 2014 <a href="http://www.wudsn.com" target="_top">Peter Dell</a> + * + * This file is part of WUDSN IDE. + * + * WUDSN IDE 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. + * + * WUDSN IDE 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 WUDSN IDE. If not, see <http://www.gnu.org/licenses/>. + */ + +package com.wudsn.ide.base.gui; + +import org.eclipse.swt.SWT; +import org.eclipse.swt.widgets.Composite; +import org.eclipse.swt.widgets.Control; +import org.eclipse.swt.widgets.Label; +import org.eclipse.swt.widgets.Text; + +/** + * Simple text field. + * + * @author Peter Dell + * + */ +public final class TextField extends Field { + + private Text text; + + /** + * Creates a text field. + * + * @param parent + * The parent composite, not <code>null</code>. + * @param labelText + * The label text, not <code>null</code>. + * @param style + * The SWT style. + */ + public TextField(Composite parent, String labelText, + int style) { + if (parent == null) { + throw new IllegalArgumentException( + "Parameter 'parent' must not be null."); + } + if (labelText == null) { + throw new IllegalArgumentException( + "Parameter 'labelText' must not be null."); + } + + label = new Label(parent, SWT.RIGHT); + label.setText(labelText); + + text = new Text(parent, style); + + } + + /** + * Gets the text control representing this text field. + * + * @return The text control, not <code>null</code>. + */ + public Text getText() { + return text; + } + + /** + * {@inheritDoc} + */ + @Override + public Control getControl() { + return text; + } + + public void setVisible(boolean visible) { + label.setVisible(visible); + text.setVisible(visible); + + } + + /** + * {@inheritDoc} + */ + @Override + public void setEnabled(boolean enabled) { + label.setEnabled(enabled); + text.setEnabled(enabled); + } + + /** + * {@inheritDoc} + */ + @Override + public void setEditable(boolean editable) { + text.setEditable(editable); + } + + /** + * Sets the value. + * + * @param value + * The value, not <code>null</code>. + */ + public void setValue(String value) { + if (value == null) { + throw new IllegalArgumentException( + "Parameter 'value' must not be null."); + } + text.setText(value); + } + + /** + * Gets the value. + * + * @return The value, not <code>null</code>. + */ + public String getValue() { + String result; + result = text.getText(); + result = result.trim(); + return result; + + } + +} \ No newline at end of file diff --git a/com.wudsn.ide.feature/.project b/com.wudsn.ide.feature/.project new file mode 100644 index 00000000..b9470f0d --- /dev/null +++ b/com.wudsn.ide.feature/.project @@ -0,0 +1,17 @@ +<?xml version="1.0" encoding="UTF-8"?> +<projectDescription> + <name>com.wudsn.ide.feature</name> + <comment></comment> + <projects> + </projects> + <buildSpec> + <buildCommand> + <name>org.eclipse.pde.FeatureBuilder</name> + <arguments> + </arguments> + </buildCommand> + </buildSpec> + <natures> + <nature>org.eclipse.pde.FeatureNature</nature> + </natures> +</projectDescription> diff --git a/com.wudsn.ide.feature/build.properties b/com.wudsn.ide.feature/build.properties new file mode 100644 index 00000000..2b07cd1a --- /dev/null +++ b/com.wudsn.ide.feature/build.properties @@ -0,0 +1,4 @@ +bin.includes = feature.xml,\ + gpl-2_0.txt,\ + build.properties,\ + .project diff --git a/com.wudsn.ide.feature/feature.xml b/com.wudsn.ide.feature/feature.xml new file mode 100644 index 00000000..62a96009 --- /dev/null +++ b/com.wudsn.ide.feature/feature.xml @@ -0,0 +1,378 @@ +<?xml version="1.0" encoding="UTF-8"?> +<feature + id="com.wudsn.ide.feature" + label="WUDSN IDE" + version="1.7.0.qualifier" + provider-name="WUDSN"> + + <description url="http://www.wudsn.com"> + The WUDSN IDE is an Eclipse feature which provides integrated +an assembler development environment for Apple II, Atari 2600, +Atari 7800, Atari 8-bit, Commodore 8-bit and Nintendo 8-bit +computers. + </description> + + <copyright> + WUDSN IDE Copyright (C) 2009 - 2018 Peter Dell +66822 Lebach, Germany +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, version 2 of the License. +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, see http://www.gnu.org/licenses. +This software includes the runtime of the Rhino JavaScript engine +published under the Mozialla Public Licence (MPL) as part of +the graphics editor. +See http://www.mozilla.org/rhino for details. +This software includes algorithms and code ported from the First +Atari Image Library (FAIL) as part of the graphics editor. +See http://http://fail.sourceforge.net/ for details. +This software uses the libraries from Another Slight Atari Player +(ASAP) for replaying Atari 8-bit sound modules. +See http://asap.sourceforge.net/ for details. +This software uses the libraries from JSIDPlay2 for replaying +C64 sound modules. +See http://jsidplay2.sourceforge.net/ for details. + </copyright> + + <license url="http://www.gnu.org/licenses"> + GNU GENERAL PUBLIC LICENSE + Version 2, June 1991 + + Copyright (C) 1989, 1991 Free Software Foundation, Inc., + 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + + Preamble + + The licenses for most software are designed to take away your +freedom to share and change it. By contrast, the GNU General Public +License is intended to guarantee your freedom to share and change free +software--to make sure the software is free for all its users. This +General Public License applies to most of the Free Software +Foundation's software and to any other program whose authors commit to +using it. (Some other Free Software Foundation software is covered by +the GNU Lesser General Public License instead.) You can apply it to +your programs, too. + + When we speak of free software, we are referring to freedom, not +price. Our General Public Licenses are designed to make sure that you +have the freedom to distribute copies of free software (and charge for +this service if you wish), that you receive source code or can get it +if you want it, that you can change the software or use pieces of it +in new free programs; and that you know you can do these things. + + To protect your rights, we need to make restrictions that forbid +anyone to deny you these rights or to ask you to surrender the rights. +These restrictions translate to certain responsibilities for you if you +distribute copies of the software, or if you modify it. + + For example, if you distribute copies of such a program, whether +gratis or for a fee, you must give the recipients all the rights that +you have. You must make sure that they, too, receive or can get the +source code. And you must show them these terms so they know their +rights. + + We protect your rights with two steps: (1) copyright the software, and +(2) offer you this license which gives you legal permission to copy, +distribute and/or modify the software. + + Also, for each author's protection and ours, we want to make certain +that everyone understands that there is no warranty for this free +software. If the software is modified by someone else and passed on, we +want its recipients to know that what they have is not the original, so +that any problems introduced by others will not reflect on the original +authors' reputations. + + Finally, any free program is threatened constantly by software +patents. We wish to avoid the danger that redistributors of a free +program will individually obtain patent licenses, in effect making the +program proprietary. To prevent this, we have made it clear that any +patent must be licensed for everyone's free use or not licensed at all. + + The precise terms and conditions for copying, distribution and +modification follow. + + GNU GENERAL PUBLIC LICENSE + TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION + + 0. This License applies to any program or other work which contains +a notice placed by the copyright holder saying it may be distributed +under the terms of this General Public License. The "Program", below, +refers to any such program or work, and a "work based on the Program" +means either the Program or any derivative work under copyright law: +that is to say, a work containing the Program or a portion of it, +either verbatim or with modifications and/or translated into another +language. (Hereinafter, translation is included without limitation in +the term "modification".) Each licensee is addressed as "you". + +Activities other than copying, distribution and modification are not +covered by this License; they are outside its scope. The act of +running the Program is not restricted, and the output from the Program +is covered only if its contents constitute a work based on the +Program (independent of having been made by running the Program). +Whether that is true depends on what the Program does. + + 1. You may copy and distribute verbatim copies of the Program's +source code as you receive it, in any medium, provided that you +conspicuously and appropriately publish on each copy an appropriate +copyright notice and disclaimer of warranty; keep intact all the +notices that refer to this License and to the absence of any warranty; +and give any other recipients of the Program a copy of this License +along with the Program. + +You may charge a fee for the physical act of transferring a copy, and +you may at your option offer warranty protection in exchange for a fee. + + 2. You may modify your copy or copies of the Program or any portion +of it, thus forming a work based on the Program, and copy and +distribute such modifications or work under the terms of Section 1 +above, provided that you also meet all of these conditions: + + a) You must cause the modified files to carry prominent notices + stating that you changed the files and the date of any change. + + b) You must cause any work that you distribute or publish, that in + whole or in part contains or is derived from the Program or any + part thereof, to be licensed as a whole at no charge to all third + parties under the terms of this License. + + c) If the modified program normally reads commands interactively + when run, you must cause it, when started running for such + interactive use in the most ordinary way, to print or display an + announcement including an appropriate copyright notice and a + notice that there is no warranty (or else, saying that you provide + a warranty) and that users may redistribute the program under + these conditions, and telling the user how to view a copy of this + License. (Exception: if the Program itself is interactive but + does not normally print such an announcement, your work based on + the Program is not required to print an announcement.) + +These requirements apply to the modified work as a whole. If +identifiable sections of that work are not derived from the Program, +and can be reasonably considered independent and separate works in +themselves, then this License, and its terms, do not apply to those +sections when you distribute them as separate works. But when you +distribute the same sections as part of a whole which is a work based +on the Program, the distribution of the whole must be on the terms of +this License, whose permissions for other licensees extend to the +entire whole, and thus to each and every part regardless of who wrote it. + +Thus, it is not the intent of this section to claim rights or contest +your rights to work written entirely by you; rather, the intent is to +exercise the right to control the distribution of derivative or +collective works based on the Program. + +In addition, mere aggregation of another work not based on the Program +with the Program (or with a work based on the Program) on a volume of +a storage or distribution medium does not bring the other work under +the scope of this License. + + 3. You may copy and distribute the Program (or a work based on it, +under Section 2) in object code or executable form under the terms of +Sections 1 and 2 above provided that you also do one of the following: + + a) Accompany it with the complete corresponding machine-readable + source code, which must be distributed under the terms of Sections + 1 and 2 above on a medium customarily used for software interchange; or, + + b) Accompany it with a written offer, valid for at least three + years, to give any third party, for a charge no more than your + cost of physically performing source distribution, a complete + machine-readable copy of the corresponding source code, to be + distributed under the terms of Sections 1 and 2 above on a medium + customarily used for software interchange; or, + + c) Accompany it with the information you received as to the offer + to distribute corresponding source code. (This alternative is + allowed only for noncommercial distribution and only if you + received the program in object code or executable form with such + an offer, in accord with Subsection b above.) + +The source code for a work means the preferred form of the work for +making modifications to it. For an executable work, complete source +code means all the source code for all modules it contains, plus any +associated interface definition files, plus the scripts used to +control compilation and installation of the executable. However, as a +special exception, the source code distributed need not include +anything that is normally distributed (in either source or binary +form) with the major components (compiler, kernel, and so on) of the +operating system on which the executable runs, unless that component +itself accompanies the executable. + +If distribution of executable or object code is made by offering +access to copy from a designated place, then offering equivalent +access to copy the source code from the same place counts as +distribution of the source code, even though third parties are not +compelled to copy the source along with the object code. + + 4. You may not copy, modify, sublicense, or distribute the Program +except as expressly provided under this License. Any attempt +otherwise to copy, modify, sublicense or distribute the Program is +void, and will automatically terminate your rights under this License. +However, parties who have received copies, or rights, from you under +this License will not have their licenses terminated so long as such +parties remain in full compliance. + + 5. You are not required to accept this License, since you have not +signed it. However, nothing else grants you permission to modify or +distribute the Program or its derivative works. These actions are +prohibited by law if you do not accept this License. Therefore, by +modifying or distributing the Program (or any work based on the +Program), you indicate your acceptance of this License to do so, and +all its terms and conditions for copying, distributing or modifying +the Program or works based on it. + + 6. Each time you redistribute the Program (or any work based on the +Program), the recipient automatically receives a license from the +original licensor to copy, distribute or modify the Program subject to +these terms and conditions. You may not impose any further +restrictions on the recipients' exercise of the rights granted herein. +You are not responsible for enforcing compliance by third parties to +this License. + + 7. If, as a consequence of a court judgment or allegation of patent +infringement or for any other reason (not limited to patent issues), +conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License. If you cannot +distribute so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you +may not distribute the Program at all. For example, if a patent +license would not permit royalty-free redistribution of the Program by +all those who receive copies directly or indirectly through you, then +the only way you could satisfy both it and this License would be to +refrain entirely from distribution of the Program. + +If any portion of this section is held invalid or unenforceable under +any particular circumstance, the balance of the section is intended to +apply and the section as a whole is intended to apply in other +circumstances. + +It is not the purpose of this section to induce you to infringe any +patents or other property right claims or to contest validity of any +such claims; this section has the sole purpose of protecting the +integrity of the free software distribution system, which is +implemented by public license practices. Many people have made +generous contributions to the wide range of software distributed +through that system in reliance on consistent application of that +system; it is up to the author/donor to decide if he or she is willing +to distribute software through any other system and a licensee cannot +impose that choice. + +This section is intended to make thoroughly clear what is believed to +be a consequence of the rest of this License. + + 8. If the distribution and/or use of the Program is restricted in +certain countries either by patents or by copyrighted interfaces, the +original copyright holder who places the Program under this License +may add an explicit geographical distribution limitation excluding +those countries, so that distribution is permitted only in or among +countries not thus excluded. In such case, this License incorporates +the limitation as if written in the body of this License. + + 9. The Free Software Foundation may publish revised and/or new versions +of the General Public License from time to time. Such new versions will +be similar in spirit to the present version, but may differ in detail to +address new problems or concerns. + +Each version is given a distinguishing version number. If the Program +specifies a version number of this License which applies to it and "any +later version", you have the option of following the terms and conditions +either of that version or of any later version published by the Free +Software Foundation. If the Program does not specify a version number of +this License, you may choose any version ever published by the Free Software +Foundation. + + 10. If you wish to incorporate parts of the Program into other free +programs whose distribution conditions are different, write to the author +to ask for permission. For software which is copyrighted by the Free +Software Foundation, write to the Free Software Foundation; we sometimes +make exceptions for this. Our decision will be guided by the two goals +of preserving the free status of all derivatives of our free software and +of promoting the sharing and reuse of software generally. + + NO WARRANTY + + 11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY +FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN +OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES +PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED +OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF +MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS +TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE +PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING, +REPAIR OR CORRECTION. + + 12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING +WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR +REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, +INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING +OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED +TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY +YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER +PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE +POSSIBILITY OF SUCH DAMAGES. + + END OF TERMS AND CONDITIONS + </license> + + <url> + <update label="WUDSN Eclipse Update Site" url="http://www.wudsn.com/update"/> + </url> + + <plugin + id="com.wudsn.ide.asm" + download-size="0" + install-size="0" + version="0.0.0" + unpack="false"/> + + <plugin + id="com.wudsn.ide.base" + download-size="0" + install-size="0" + version="0.0.0" + unpack="false"/> + + <plugin + id="com.wudsn.ide.gfx" + download-size="0" + install-size="0" + version="0.0.0" + unpack="false"/> + + <plugin + id="com.wudsn.ide.asm.compilers" + download-size="0" + install-size="0" + version="0.0.0" + unpack="false"/> + + <plugin + id="com.wudsn.ide.asm.compilers.test" + download-size="0" + install-size="0" + version="0.0.0" + unpack="false"/> + + <plugin + id="com.wudsn.ide.dsk" + download-size="0" + install-size="0" + version="0.0.0" + unpack="false"/> + + <plugin + id="com.wudsn.ide.snd" + download-size="0" + install-size="0" + version="0.0.0" + unpack="false"/> + +</feature> diff --git a/com.wudsn.ide.feature/gpl-2_0.txt b/com.wudsn.ide.feature/gpl-2_0.txt new file mode 100644 index 00000000..6e812ec1 --- /dev/null +++ b/com.wudsn.ide.feature/gpl-2_0.txt @@ -0,0 +1,280 @@ + GNU GENERAL PUBLIC LICENSE + Version 2, June 1991 + + Copyright (C) 1989, 1991 Free Software Foundation, Inc., + 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + + Preamble + + The licenses for most software are designed to take away your +freedom to share and change it. By contrast, the GNU General Public +License is intended to guarantee your freedom to share and change free +software--to make sure the software is free for all its users. This +General Public License applies to most of the Free Software +Foundation's software and to any other program whose authors commit to +using it. (Some other Free Software Foundation software is covered by +the GNU Lesser General Public License instead.) You can apply it to +your programs, too. + + When we speak of free software, we are referring to freedom, not +price. Our General Public Licenses are designed to make sure that you +have the freedom to distribute copies of free software (and charge for +this service if you wish), that you receive source code or can get it +if you want it, that you can change the software or use pieces of it +in new free programs; and that you know you can do these things. + + To protect your rights, we need to make restrictions that forbid +anyone to deny you these rights or to ask you to surrender the rights. +These restrictions translate to certain responsibilities for you if you +distribute copies of the software, or if you modify it. + + For example, if you distribute copies of such a program, whether +gratis or for a fee, you must give the recipients all the rights that +you have. You must make sure that they, too, receive or can get the +source code. And you must show them these terms so they know their +rights. + + We protect your rights with two steps: (1) copyright the software, and +(2) offer you this license which gives you legal permission to copy, +distribute and/or modify the software. + + Also, for each author's protection and ours, we want to make certain +that everyone understands that there is no warranty for this free +software. If the software is modified by someone else and passed on, we +want its recipients to know that what they have is not the original, so +that any problems introduced by others will not reflect on the original +authors' reputations. + + Finally, any free program is threatened constantly by software +patents. We wish to avoid the danger that redistributors of a free +program will individually obtain patent licenses, in effect making the +program proprietary. To prevent this, we have made it clear that any +patent must be licensed for everyone's free use or not licensed at all. + + The precise terms and conditions for copying, distribution and +modification follow. + + GNU GENERAL PUBLIC LICENSE + TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION + + 0. This License applies to any program or other work which contains +a notice placed by the copyright holder saying it may be distributed +under the terms of this General Public License. The "Program", below, +refers to any such program or work, and a "work based on the Program" +means either the Program or any derivative work under copyright law: +that is to say, a work containing the Program or a portion of it, +either verbatim or with modifications and/or translated into another +language. (Hereinafter, translation is included without limitation in +the term "modification".) Each licensee is addressed as "you". + +Activities other than copying, distribution and modification are not +covered by this License; they are outside its scope. The act of +running the Program is not restricted, and the output from the Program +is covered only if its contents constitute a work based on the +Program (independent of having been made by running the Program). +Whether that is true depends on what the Program does. + + 1. You may copy and distribute verbatim copies of the Program's +source code as you receive it, in any medium, provided that you +conspicuously and appropriately publish on each copy an appropriate +copyright notice and disclaimer of warranty; keep intact all the +notices that refer to this License and to the absence of any warranty; +and give any other recipients of the Program a copy of this License +along with the Program. + +You may charge a fee for the physical act of transferring a copy, and +you may at your option offer warranty protection in exchange for a fee. + + 2. You may modify your copy or copies of the Program or any portion +of it, thus forming a work based on the Program, and copy and +distribute such modifications or work under the terms of Section 1 +above, provided that you also meet all of these conditions: + + a) You must cause the modified files to carry prominent notices + stating that you changed the files and the date of any change. + + b) You must cause any work that you distribute or publish, that in + whole or in part contains or is derived from the Program or any + part thereof, to be licensed as a whole at no charge to all third + parties under the terms of this License. + + c) If the modified program normally reads commands interactively + when run, you must cause it, when started running for such + interactive use in the most ordinary way, to print or display an + announcement including an appropriate copyright notice and a + notice that there is no warranty (or else, saying that you provide + a warranty) and that users may redistribute the program under + these conditions, and telling the user how to view a copy of this + License. (Exception: if the Program itself is interactive but + does not normally print such an announcement, your work based on + the Program is not required to print an announcement.) + +These requirements apply to the modified work as a whole. If +identifiable sections of that work are not derived from the Program, +and can be reasonably considered independent and separate works in +themselves, then this License, and its terms, do not apply to those +sections when you distribute them as separate works. But when you +distribute the same sections as part of a whole which is a work based +on the Program, the distribution of the whole must be on the terms of +this License, whose permissions for other licensees extend to the +entire whole, and thus to each and every part regardless of who wrote it. + +Thus, it is not the intent of this section to claim rights or contest +your rights to work written entirely by you; rather, the intent is to +exercise the right to control the distribution of derivative or +collective works based on the Program. + +In addition, mere aggregation of another work not based on the Program +with the Program (or with a work based on the Program) on a volume of +a storage or distribution medium does not bring the other work under +the scope of this License. + + 3. You may copy and distribute the Program (or a work based on it, +under Section 2) in object code or executable form under the terms of +Sections 1 and 2 above provided that you also do one of the following: + + a) Accompany it with the complete corresponding machine-readable + source code, which must be distributed under the terms of Sections + 1 and 2 above on a medium customarily used for software interchange; or, + + b) Accompany it with a written offer, valid for at least three + years, to give any third party, for a charge no more than your + cost of physically performing source distribution, a complete + machine-readable copy of the corresponding source code, to be + distributed under the terms of Sections 1 and 2 above on a medium + customarily used for software interchange; or, + + c) Accompany it with the information you received as to the offer + to distribute corresponding source code. (This alternative is + allowed only for noncommercial distribution and only if you + received the program in object code or executable form with such + an offer, in accord with Subsection b above.) + +The source code for a work means the preferred form of the work for +making modifications to it. For an executable work, complete source +code means all the source code for all modules it contains, plus any +associated interface definition files, plus the scripts used to +control compilation and installation of the executable. However, as a +special exception, the source code distributed need not include +anything that is normally distributed (in either source or binary +form) with the major components (compiler, kernel, and so on) of the +operating system on which the executable runs, unless that component +itself accompanies the executable. + +If distribution of executable or object code is made by offering +access to copy from a designated place, then offering equivalent +access to copy the source code from the same place counts as +distribution of the source code, even though third parties are not +compelled to copy the source along with the object code. + + 4. You may not copy, modify, sublicense, or distribute the Program +except as expressly provided under this License. Any attempt +otherwise to copy, modify, sublicense or distribute the Program is +void, and will automatically terminate your rights under this License. +However, parties who have received copies, or rights, from you under +this License will not have their licenses terminated so long as such +parties remain in full compliance. + + 5. You are not required to accept this License, since you have not +signed it. However, nothing else grants you permission to modify or +distribute the Program or its derivative works. These actions are +prohibited by law if you do not accept this License. Therefore, by +modifying or distributing the Program (or any work based on the +Program), you indicate your acceptance of this License to do so, and +all its terms and conditions for copying, distributing or modifying +the Program or works based on it. + + 6. Each time you redistribute the Program (or any work based on the +Program), the recipient automatically receives a license from the +original licensor to copy, distribute or modify the Program subject to +these terms and conditions. You may not impose any further +restrictions on the recipients' exercise of the rights granted herein. +You are not responsible for enforcing compliance by third parties to +this License. + + 7. If, as a consequence of a court judgment or allegation of patent +infringement or for any other reason (not limited to patent issues), +conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License. If you cannot +distribute so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you +may not distribute the Program at all. For example, if a patent +license would not permit royalty-free redistribution of the Program by +all those who receive copies directly or indirectly through you, then +the only way you could satisfy both it and this License would be to +refrain entirely from distribution of the Program. + +If any portion of this section is held invalid or unenforceable under +any particular circumstance, the balance of the section is intended to +apply and the section as a whole is intended to apply in other +circumstances. + +It is not the purpose of this section to induce you to infringe any +patents or other property right claims or to contest validity of any +such claims; this section has the sole purpose of protecting the +integrity of the free software distribution system, which is +implemented by public license practices. Many people have made +generous contributions to the wide range of software distributed +through that system in reliance on consistent application of that +system; it is up to the author/donor to decide if he or she is willing +to distribute software through any other system and a licensee cannot +impose that choice. + +This section is intended to make thoroughly clear what is believed to +be a consequence of the rest of this License. + + 8. If the distribution and/or use of the Program is restricted in +certain countries either by patents or by copyrighted interfaces, the +original copyright holder who places the Program under this License +may add an explicit geographical distribution limitation excluding +those countries, so that distribution is permitted only in or among +countries not thus excluded. In such case, this License incorporates +the limitation as if written in the body of this License. + + 9. The Free Software Foundation may publish revised and/or new versions +of the General Public License from time to time. Such new versions will +be similar in spirit to the present version, but may differ in detail to +address new problems or concerns. + +Each version is given a distinguishing version number. If the Program +specifies a version number of this License which applies to it and "any +later version", you have the option of following the terms and conditions +either of that version or of any later version published by the Free +Software Foundation. If the Program does not specify a version number of +this License, you may choose any version ever published by the Free Software +Foundation. + + 10. If you wish to incorporate parts of the Program into other free +programs whose distribution conditions are different, write to the author +to ask for permission. For software which is copyrighted by the Free +Software Foundation, write to the Free Software Foundation; we sometimes +make exceptions for this. Our decision will be guided by the two goals +of preserving the free status of all derivatives of our free software and +of promoting the sharing and reuse of software generally. + + NO WARRANTY + + 11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY +FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN +OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES +PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED +OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF +MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS +TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE +PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING, +REPAIR OR CORRECTION. + + 12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING +WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR +REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, +INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING +OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED +TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY +YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER +PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE +POSSIBILITY OF SUCH DAMAGES. + + END OF TERMS AND CONDITIONS

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 Sfim-B`%+(f?X8fvss95YtCt@D literal 0 HcmV?d00001 diff --git a/com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/stella.txt b/com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/stella.txt new file mode 100644 index 00000000..e9eb92c9 --- /dev/null +++ b/com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/stella.txt @@ -0,0 +1,1938 @@ +TABLE OF CONTENTS 4 + + + + + + + STELLA + + PROGRAMMER'S + + GUIDE + + + + + + + + + + by + Steve Wright + 12/03/79 + + + + + + + + + + + + + + + + (Reconstructed by Charles Sinnett 6/11/93 Internet: + cas@mentor.cc.purdue.edu) + +Table of Contents + + +TELEVISION PROTOCOL 1 + Diagram 1 - Atari TV Frame 2 +The TIA (as seen by the programmer) 3 + 1.0 General Description 3 + 2.0 The Registers 3 + 3.0 Synchronization 4 + 3.1 Horizontal Timing 4 + 3.2 Microprocessor Synchronization 4 + 3.3 Vertical timing 4 + 4.0 Color and Luminosity 5 + 5.0 Playfield 5 + 6.0 The Moveable Objects Graphics 6 + 6.1 Missile Graphics (M0, M1) 6 + 6.2 Ball Graphics (BL) 6 + 6.3 Player Graphics (P0, P1) 7 + 7.0 Horizontal Positioning 8 + 8.0 Horizontal Motion 8 + 9.0 Object Priorities 9 + 10.0 Collisions 10 + 11.0 Sound 10 + 11.1 Tone 10 + 11.2 Frequency 10 + 11.3 Volume 10 + 12.0 Input Ports 11 + 12.1 Dumped Input Ports (INPT0 thru INPT3) 11 + 12.2 Latched Input Ports (INPT4, INPT5) 11 + +THE PIA (6532) 12 + 1.0 General 12 + 2.0 Interval timer 12 + 2.1 Setting the timer 12 + 2.2 Reading the timer 12 + 2.3 When the timer reaches zero 12 + 3.0 RAM 13 + 4.0 The I/O ports 13 + 4.1 Port B - Console Switches (read only) 13 + 5.0 Port A - Hand Controllers 13 + 5.1 Setting for input or output 13 + 5.2 Inputting and Outputting 14 + 5.3 Joystick Controllers 14 + 5.4 Paddle (pot) controllers 14 + 5.5 Keyboard controllers 15 + 6.0 Address summary table 15 + + +PAL/SECAM CONVERSIONS 16 + PAL 16 + SECAM 16 + + +TIA 1A - TELEVISION INTERFACE ADAPTOR (MODEL 1A) 17 + GENERAL DESCRIPTION 17 + DETAILED DESCRIPTION 18 + 1. Data and addressing 18 + 2. Synchronization 18 + A. Horizontal Timing 18 + B. Vertical Timing 18 + C. Composite Sync 18 + D. Microprocessor Synchronization 19 + 3. Playfield graphics Register 19 + A. Description 19 + B. Normal Serial Output 19 + C. Reflected Serial Output 19 + D. Timing Constraints 20 + 4. Horizontal Position Counters 20 + A. Description 20 + B. Ball position Counter 20 + C. Player Position Counters 20 + D. Missile Position Counters 21 + 5. Horizontal Motion Registers 21 + A. General Description 21 + B. Timing constraints 21 + + 6. Moving Object Graphics Registers 22 + A. General Description 22 + B. Missile Graphics 22 + C. Player Graphics 22 + D. Vertical Delay 23 + E. Ball Graphics 24 + 7. Collision Detection Latches 24 + A. Definitions 24 + B. Reading Collision 24 + C. Reset 24 + 8. Input ports 25 + A. General Description 25 + B. Dumped Input Ports (I0 through I3) 25 + C. Latched Input ports (I4, I5) 25 + 8.5 Priority Encoder 26 + A. Purpose 26 + B. Priority Assignment 26 + C. Priority Control 26 + 9 Color Luminance Registers 27 + A. Description 27 + B. Multiplexing 27 + 10. Color Phase Shifter 27 + 11. Audio Circuits 27 + A. Frequency Select 27 + B. Noise-Tone Generator 28 + C. Volume Select 28 + + Figure 1. Vertical Delay 29 + Figure 2. Synchronization 30 + Figure 3. Color-Luminance 30 + Figure 4. Typical Horizontal Motion Circuit 31 + Figure 5. Playfield Graphics 32 + Figure 6. Collision Detection 33 + Figure 7. Audio Circuit 34 + Figure 8. Input Ports 35 + Figure 9. Game System 36 + +Write Address Detailed Functions 37 + WSYNC (wait for sync) 37 + RSYNC (reset sync) 37 + VSYNC 37 + VBLANK 37 + PJ0 (PF1, PF2) 38 + PLAYFIELD REGISTERS SERIAL OUTPUT 38 + CTRLPF 38 + NUSIZ0 (NUSIZ1) 39 + RESP0 (RESP1, RESM0, RESM1, RESBL) 39 + RESMP0 (RESMP1) 39 + HMOVE 40 + HMCLR 40 + HMP0 (HMP1, HMM0, HMM1, HMBL) 40 + ENAM0 (ENAM1, ENABL) 41 + GRP0 (GRP1) 41 + REFP0 (REFP1) 41 + VDELP0 (VDELP1, VDELBL) 41 + CXCLR 41 + COLUP0 (COLUP1, COLUPF, COLUBK) 42 + AUDF0 (AUDF1) 42 + AUDC0 (AUDC1) 43 + AUDV0 (AUDV1) 43 + WRITE ADDRESS SUMMARY 44 + READ ADDRESS SUMMARY 45 + + TIA O0..02 AND LUM TIMING 46 + TIA WRITE TIMING CHARACTERISTICS 47 + TIA READ TIMING CHARACTERISTICS 48 + TIA COMP-SYN AND READY TIMING 49 + RSYNC, RES0O, H01, H02, SHB, 02, 0O 50 + TIA RSYNC AND BLANK AND READY TIMING 51 + + + +TELEVISION PROTOCOL + (The TV picture according to Atari) + + For the purposes of Stella programming, a single television + “frame” consists of 262 horizontal lines, and each line is + divided by 228 clock counts (3.58MHz). The actual TV + picture is drawn line by line from the top down 60 times a + second, and actaully consists of only a portion of the + entire “frame” (see diag. #1). A typical frame will + consists of 3 vertical sync (VSYNC) lines*, 37 vertical + blank (VBLANK) lines, 192 TV picture lines, and 30 overscan + lines. Atari’s research has shown that this pattern will + work on all types of TV sets. Each scan lines starts with + 68 clock counts of horizontal blank (not seen on the TV + screen) followed by 160 clock counts to fully scan one line + of TV picture. When the electron beam reaches the end of a + scan line, it returns to the left side of the screen, waits + for the 68 horizontal blank clock counts, and proceeds to + draw the next line below. + + All horizontal timing is taken care of by hardware, but the + microprocessor must “manually” control vertical timing to + signal the start of the next frame. When the last line of + the previous frame is detected, the microprocessor must + generate 3 lines of VSYNC, 37 lines of VBLANK, 192 lines of + actual TV picture, and 30 lines of overscan. Fortunately, + both VSYNC and VBLANK can simply be turned on and off at + the appropriate times, freeing the microprocessor for other + activities during their execution. + + * (to signal the TV set to start a new frame) + + The actual TV picture is drawn one line at a time by having + the microprocessor enter the data for that line into the + Television Interface Adaptor (TIA) chip, which then + converts the data into video signals. The TIA can only + have data in it that pertains to the line being currently + drawn, so the microprocessor must be “one step ahead” of + the electron beam on each line. Since one microprocessor + machine cycle occurs every 3 clock counts, the programmer + has only 76 machine cycles per line (228/3 = 76) to + construct the actual picture (actually less because the + microprocessor must be ahead of the raster). To allow more + time for the software, it is customary (but not required) + to update the TIA every two scan lines. The portion of the + program that constructs this TV picture is referred to as + the “Kernel”, as it is the essence or kernel of the game. + + In general, the remaining 70 scan lines (3 for VSYNC, 37 + for VBLANK, and 30 for overscan) will provides 5,320 + machine cycles (70 lines x 76 machine cycles) for + housekeeping and game logic. Such activities as + calculating the new position of a player, updating the + score, and checking for new inputs are typically done + during this time. +Diagram 1 - Atari TV Frame + + + + +The TIA (as seen by the programmer) +1.0 General Description + + The TIA is a custom IC designed to create the TV picture + and sound from the instructions sent to it by the + microprocessor. It converts the 8 bit parallel data from + the microprocessor into signals that are sent to video + modulation circuits which combine and shape those signals + to be compatible with ordinary TV reception. A “playfield” + and 5 moveable objects can be created and manipulated by + software. + + A playfield consisting of walls, clouds, barriers, and + other seldom moved objects can be created over a colored + background. The 5 moveable objects can be positioned + anywhere, and consists of 2 players, 2 missiles, and a + ball. The playfield, players, missiles, and ball are + created and manipulated by a series of registers in the TIA + that the microprocessor can address and write into. Each + type of object has certain defined capabilities. For + example, a player can be moved with one instruction, but + the playfield must be completely re-drawn in order to make + it “move”. + + Color and luminosity (brightness) can be assigned to the + background, playfield, and 5 moveable objects. Sound can + also be generated and controlled for volume, pitch, and + type of sound. Collisions between the various objects on + the TV screen are detected by the TIA and can be read by + the microprocessor . Input ports which can be read by the + microprocessor give the status of some of the various hand + held controllers. + +2.0 The Registers + + All instructions to the TIA are achieved by addressing and + writing to various registers in the chip. A key point to + remember is data written in a register is latched an + retained until altered by another write operation into that + register. For example, if the color register for a player + is set for red, that player will be red every time it is + drawn until that color register is changed. All of the + registers are addressed by the microprocessor as part of + the overall RAM/ROM memory space. + + All registers have fixed address locations and pre-assigned + address names for handy reference. Many registers do not + use all 8 data bits, and some registers are used to + “strobe” or trigger events. A “strobe” register executes + its function the instant it is written to (the data written + is ignored). The only registers the microprocessor can + read are the collision registers and input port registers. + These registers are conveniently arranged so that the data + bits of interest always appear as data bits 6 or 7 for easy + access. + +3.0 Synchronization + + 3.1 Horizontal Timing + When the electron beam scans across the TV screen and + reaches the right edge, it must be turned off and moved + back to the left edge of the screen to begin the next scan + line. The TIA takes care of this automatically, independent + of the microprocessor. A 3.58 MHz oscillator generates + clock pulses called “color clocks” which go into a pulse + counter in the TIA. This counter allows 160 color clocks + for the beam to reach the right edge, then generates a + horizontal sync signal (HSYNC) to return the beam to the + left edge. It also generates the signal to turn the beam + off (horizontal blanking) during its return time of 68 + color clocks. Total round trip for the electron beam is + 160 + 68 = 228 color clocks. Again, all the horizontal + timing is taken care of by the TIA without assistance from + the microprocessor. + + 3.2 Microprocessor Synchronization + The microprocessor’s clock is the 3.58 MHz oscillator + divided by 3, so one machine cycle is 3 color clocks. + Therefore, one complete scan line of 228 color clocks + allows only 76 machine cycles (228/3 = 76) per scan line. + The microprocessor must be synchronized with the TIA on a + line-by-line basis, but program loops and branches take + unpredictable lengths of time. To solve this software + sync. problem, the programmer can use the WSYNC (Wait for + SYNC) strobe register. Simply writing to the WSYNC causes + the microprocessor to halt until the electron beam reaches + the right edge of the screen, then the microprocessor + resumes operation at the beginning of the 68 color clocks + for horizontal blanking. Since the TIA latches all + instructions until altered by another write operation, it + could be updated every 2 or 3 lines. The advantage is the + programmer gains more time to execute software, but at a + price paid with lower vertical resolution in the graphics. + + NOTE: WSYNC and all the following addresses’ bit + structures are itemized in the TIA hardware manual. The + purpose of this document is to make them understandable. + + 3.3 Vertical timing + When the electron beam has scanned 262 lines, the TV set + must be signaled to blank the beam and position it at the + top of the screen to start a new frame. This signal is + called vertical sync, and the TIA must transmit this signal + for at least 3 scan lines. This is accomplished by writing + a “1” in D1 of VSYNC to turn it on, count at least 2 scan + lines, then write a “0” to D1 of VSYNC to turn it off. + + To physically turn the beam off during its repositioning + time, the TV set needs 37 scan lines of vertical blanks + signal from the TIA. This is accomplished by writing a “1” + in D1 of VBLANK to turn it on, count 37 lines, then write a + “0” to D1 of VBLANK to turn it off. The microprocessor is + of course free to execute other software during the + vertical timing commands, VSYNC and VBLANK. + +4.0 Color and Luminosity + + Color and luminosity can be assigned to the background + (BK), playfield (PF), ball (BL), player 0 (P0), player + 1(P1), missile 0 (M0), and missile 1 (M1). There are only + four color-lum registers for these 7 objects, so the + objects are paired to share the same register according to + the following list: + + color-lum register Objects colored + COLUMP0 P0, M0 (player 0, missile 0) + COLUMP1 P1, M1 (player 1, missile 1) + COLUMPF PF, BL (playfield, ball) + COLUMBK BK (background) + + For example, if the COLUMP0 register is set for light red, + both P0 and M0 will be light red when drawn. + + A color-lum register is set for both color and luminosity + by writing a single 7 bit instruction to that register. + Four of the bits select one of the 16 available colors, and + the other 3 bits select one of 8 levels of luminosity + (brightness). The specific codes required to create + specific color and lum are listed in the Detailed Address + List of the TIA hardware manual. As with all registers + (except the “strobe” registers), the data written to them + is latched until altered by another write operation. + +5.0 Playfield + + The PF register is used to create a playfield of walls, + clouds, barriers, etc., that are seldom moved. This low + resolution register is written into to draw the left half + of the TV screen only. The right half of the screen is + drawn by software selection of whether a duplication or a + reflection of the right half. + + The PF register is 20 bits wide, so the 20 bits are written + into 3 addresses: PF0, PF1, and PF2. PF0 is only 4 bits + wide and constructs the first 4 “bits” of the playfield, + starting at the left edge of the TV screen. PF1 constructs + the next 8 “bits”, and PF2 the last 8 bits” which end at + the center of the screen. The PF register is scanned from + left to right and where a “1” is found the PF color is + drawn, and where a “0” is found, the BK color is drawn. To + clear the playfield, obviously zeros must be written into + PF0, PF1, and PF2. + + To make the right half of the playfield into a duplication + or copy of the left half, a “0” is written to D0 of the + CTLPF (control playfield) register. Writing a “1” will + cause the reflection to be displayed. + +6.0 The Moveable Objects Graphics + + All 5 moveable objects (P0, M0, P1, M1, BL) can be assigned + a horizontal location on the screen and moved left or right + relative to their location. Vertical positions, however, + are treated in an entirely different manner. In principle, + these objects appear at whatever scan lines their graphics + registers are enabled. For example, let us assume the ball + is to be positioned vertically in the center of the screen. + The screen has 192 scan lines and we want the ball to be 2 + scan lines “thick”. The ball graphics would be disabled + until scan line 96, enabled for 2 scan lines, then disabled + for the rest of the frame. Each type of object (players, + missiles, and ball) has its’ own characteristics and + limitations. + + 6.1 Missile Graphics (M0, M1) + The two missile graphics registers will draw a missile on + any scan line by writing a “1” to the one bit enable + missile registers (ENAM0, ENAM1). Writing a “0” to these + registers will disable the graphics. The missiles’ left + edge is positioned by a horizontal position register, but + the right edge is a function of how wide the missile is + make. Width of a missile is controlled by writing into + bits D4 and D5 of the number-size registers (NUSIZ0, + NUSIZ1). This has the effect of “stretching” the missile + out over 1,2,4, or 8 color clock counts (a full scan line + is 160 color clocks). + + 6.2 Ball Graphics (BL) + The ball graphics register works just like the missile + registers. Writing a “1” to the enable ball register + (ENABL) enables the ball graphics until the register is + disabled. The ball can also be “stretched” to widths of 1, + 2, 4, or 8 color clock counts by writing to bits D4 and D5 + of the CTRLPF register. + + The ball can also be vertically delayed one can line. For + example, if the ball graphics were enabled on scan line 95, + it could be delayed to not display on the screen until scan + line 96 by writing a “1” to D0 of the vertical delay + (VDELBL) register. The reason for having a vertical delay + capability is because most programs will update the TIA + every 2 lines. This confines all vertical movements of + objects to 2 scan line “jumps”. The use of vertical delay + allows the objects to move one scan line at a time. + + 6.3 Player Graphics (P0, P1) + The player graphics are the most sophisticated of all the + moveable objects. They have all the capabilities of the + missile and ball graphics, plus three move capabilities. + Players can take on a “shape” such as a man or an airplane, + and the player can easily be flipped over horizontally to + display the mirror image (reflection) instead of the + original image, plus multiple copies of the players can be + created. + + The player graphics are drawn line-by-line like all other + graphics. The difference here is each scan line of the + player is 8 “bits” wide, whereas the missiles and ball are + one “bit” wide. Therefore, a player can be though of as + being drawn of graph paper 8 squares wide and as tall as + desired. To “color in the squares” of this imaginary graph + paper, 8 data bits are written into the players graphics + registers (GP0, GP1). This 8 bit register is scanned from + D7 to D0, and wherever a “1” is found that “square” gets + the players’ color (from the color-lum register) and where + a “0” is found that “square” gets the background color. To + position a player vertically, simply leave all “0’s” in the + graphics registers (GP0, GP1) until the electron beam is on + the scan line desired, write to the graphics register line- + by-line describing the player, then write all “0’s” to turn + off the players’ graphics until the end of that frame. + + To display a mirror image (reflection) instead of the + original figure, write a “1” to D3 of the one bit + reflection register (REFP0, REFP1). A “0” written to these + registers restores the original figure. + + Multiple copies of players as well as their size are + controlled by writing 3 bits (D0, D1, D2) into the number- + size registers (NUSIZ0, NUSIZ1). These three bits select + from 1 to 3 copies of the player, spacing of those copies, + as well as the size of the player (each “square” of the + player can be 1, 2, or 4 clocks wide). Whenever multiple + copies are selected, the TIA automatically creates the same + number of copies of the missile for that player. Again, + the specifics of all this are laid out in the TIA hardware + manual. + + Vertical delay for the players works exactly like the ball + by writing a “1” to D0 in the players’ vertical delay + registers (VDELP0, VDELP1). Writing a “0” to these + locations disables the vertical delay. + +7.0 Horizontal Positioning + + The horizontal position of each object is set by writing to + its’ associated reset register (RESP0, RESP1, RESM0, RESM1, + RESBL) which are all “strobe” registers (they trigger their + function as soon as they are addressed). That causes the + object to be positioned wherever the electron bean was in + its sweep across the screen when the register was reset. + for example, if the electron beam was 60 color clocks into + a scan line when RESP0 was written to, player 0 would be + positioned 60 color clocks "in” on the next scan line. + Whether or not P0 is actually drawn on the screen is a + function of the data in the GP0 register, but if it were + drawn, it would show up at 60. Resets to these registers + anywhere during horizontal blanking will position objects + at the left edge of the screen (color clock 0). Since + there are 3 color clocks per machine cycle, and it can take + up to 5 machine cycles to write the register, the + programmer is confined to positioning the objects at 15 + color clock intervals across the screen. This “course” + positioning is “fine tuned” by the Horizontal Motion, + explained in section 8.0. + + Missiles have an additional positioning command. Writing a + “1” to D1 of the reset missile-to-player register (RESMP0, + RESMP1) disables that missiles’ graphics (turns it off) and + repositions it horizontally to the center of its’ + associated player. Until a “0” is written to the register, + the missiles’ horizontal position is locked to the center + of its’ player in preparation to be fired again. + +8.0 Horizontal Motion + + Horizontal motion allows the programmer to move any of the + 5 graphics objects relative to their current horizontal + position. Each object has a 4 bit horizontal motion + register (HMP0, HMP1, HMM0, HMM1, HMBL) that can be loaded + with a value in the range of +7 to -8 (negative values are + expressed in two’s complement from). This motion is not + executed until the HMOVE register is written to, at which + time all motion registers move their respective objects. + Objects can be moved repeatedly by simply executing HMOVE. + Any object that is not to move must have a 0 in its motion + register. With the horizontal positioning command confined + to positioning objects at 15 color clock intervals, the + motion registers fills in the gaps by moving objects +7 to + -8 color clocks. Objects can not be placed at any color + clock position across the screen. All 5 motion registers + can be set to zero simultaneously by writing to the + horizontal motion clear register (HMCLR). + + There are timing constraints for the HMOVE command. The + HMOVE command must immediately follow a WSYNC (Wait for + SYNC) to insure the HMOVE operation occurs during + horizontal blanking. This is to allow sufficient time for + the motion registers to do their thing before the electron + beam starts drawing the next scan line. Also, for + mysterious internal hardware considerations, the motion + registers should not be modified for at least 24 machine + cycles after an HMOVE command. + +9.0 Object Priorities + + Each object is assigned a priority so when any two objects + overlap the one with the highest priority will appear to + move in front of the other. To simplify hardware logic, + the missiles have the same priority as their associated + player, and the ball has the same priority as the + playfield. The background, of course, has the lowest + priority. The following table illustrates the normal + (default) priority assignments. + + Priority Objects + 1 P0, M0 + 2 P1, M1 + 3 BL, PF + 4 BK + + This priority assignment means that players and missiles + will move in front of the playfield. To make the players + and missiles move behind the playfield, a "1" must be + written to D2 of the CTRLPF register. The following table + illustrates how the priorities are affected: + + Priority Objects + 1 PF, BL + 2 P0, M0 + 3 P1, M1 + 4 BK + + One more priority control is available to be used for + displaying the score. When a "1" is written to D1 of the + CTRLPF register, the left half of the playfield takes on + the color of player 0, and the right half the color of + player 1. The game score can now be displayed using the PF + graphics register, and the score will be in the same color + as its associated player. + +10.0 Collisions + + The TIA detects collisions between any of the 6 objects it + generates (the playfield and 5 moveable objects). There + are 15 possible two-object collisions which are stored in + 15 one bit latches. Each collision register contains two + of these latches which are read by the microprocessor on D6 + and D7 of the data bus for easy access. A "1" on the data + line indicates the collision it records has occurred. The + collision registers could be read at any time but is + usually done during vertical blank after all possible + collisions have occurred. The collision registers are all + reset simultaneously by writing to the collision reset + register (CXCLR). + +11.0 Sound + + There are two audio circuits for generating sound. They + are identical but completely independent and can be + operated simultaneously to produce sound effects through + the TV speaker. Each audio circuit has three registers + that control a noise-tone generator (what kind of sound), a + frequency selection (high or low pitch of the sound), and a + volume control. + + 11.1 Tone + The noise-tone generator is controlled by writing to the 4 + bit audio control registers (AUDC0, AUDC1). The values + written cause different kinds of sounds to be generated. + Some are pure tones like a flute, others have various + "noise" content like a rocket motor or explosion. Even + though the TIA hardware manual lists the sounds created by + each value, some experimentation will be necessary to find + "your sound". + + 11.2 Frequency + Frequency selection is controlled by writing to a 5 bit + audio frequency register (AUDF0, AUDF1). The value written + is used to divide a 30KHz reference frequency creating + higher or lower pitch of whatever type of sound is created + by the noise-tone generator. By combining the pure tones + available from the noise-tone generator with frequency + selection a wide range of tones can be generated. + + 11.3 Volume + Volume is controlled by writing to a 4 bit audio volume + register (AUDV0, AUDV1). Writing 0 to these registers + turns sound off completely, and writing any value up to 15 + increases the volume accordingly. + +12.0 Input Ports + + There are six input ports whose logic states can be read on + D7 by reading the input port addresses (INPT0, thru INPT5). + These six ports are divided into two types, "dumped" and + "latched". + + 12.1 Dumped Input Ports (INPT0 thru INPT3) + These four ports are used to read up to four paddle + controllers. Each paddle controller contains an adjustable + pot controlled by the knob on the controller. The output + of the pot is used to charge a capacitor in the console, + and when the capacitor is charged the input port goes HI. + The microprocessor discharges this capacitor by writing a + "1" to D7 of VBLANK then measures the time it takes to + detect a logic one at that port. This information can be + used to position objects on the screen based on the + position of the knob on the paddle controller. + + 12.2 Latched Input Ports (INPT4, INPT5) + These two ports have latches that are both enabled by + writing a "1" or disabled by writing a "0" to D6 of VBLANK. + When disabled the microprocessor reads the logic level of + the port directly. When enabled, the latch is set for + logic one and remains that way until its' port goes LO. + When the port goes LO the latch goes LO and remains that + way regardless of what the port does. The trigger buttons + of the joystick controllers connect to these ports. + +THE PIA (6532) + +1.0 General + + The PIA chip is an off-the-shelf 6532 Peripheral Interface + Adaptor which has three functions: a programmable timer, + 128 bytes of RAM, and two 8 bit parallel I/O ports. + +2.0 Interval timer + + The PIA uses the same clock as the microprocessor so that + one PIA cycle occurs for each machine cycle. The PIA can + be set for one of four different "intervals", where each + interval is some multiple of the clock (and therefore + machine cycles). A value from 1 to 255 is loaded into the + PIA which will be decremented by one at each interval. The + timer can now be read by the microprocessor to determine + elapsed time for timing various software operations and + keep them synchronized with the hardware (TIA chip). + + 2.1 Setting the timer + The timer is set by writing a value or count (from 1 to + 255) to the address of the desired interval setting + according to the following table : + + Hex Address Interval Mnemonic + 294 1 clock TIM1T + 295 8 clocks TIM8T + 296 64 clocks TIM64T + 297 1024 clocks T1024T + + For example, if the value of 100 were written to TIM64T + (HEX address 296) the timer would decrement to 0 in 6400 + clocks (64 clocks per interval x 100 intervals) which would + also be 6400 microprocessor machine cycles. + + 2.2 Reading the timer + The timer may be read any number of times after it is + loaded of course, but the programmer is usually interested + in whether or not the timer has reached 0. The timer is + read by reading INTIM at hex address 284. + + 2.3 When the timer reaches zero + The PIA decrements the value or count loaded into it once + each interval until it reaches 0. It holds that 0 counts + for one interval, then the counter flips to FF(HEX) and + decrements once each clock cycle, rather than once per + interval. The purpose of this feature is to allow the + programmer to determine how long ago the timer zeroed out + in the event the timer was read after it passed zero. + +3.0 RAM + + The PIA has 128 bytes of RAM located in the Stella memory + map from HEX address 80 to FF. The microprocessor stack is + normally located from FF on down, and variables are + normally located from 80 on up (hoping the two never meet). + +4.0 The I/O ports + + The two ports (Port A and Port B) are 8 bits wide and can + be set for either input or output. Port A is used to + interface to carious hand-held controllers but Port B is + dedicated to reading the status of the Stella console + switches. + + 4.1 Port B - Console Switches (read only) + Port B is hardwired to be an input port only that is read + by addressing SWCHB (HEX 282) to determine the status of + all the console switches according to the following table: + + Data Bit Switch Bit Meaning + D7 P1 difficulty 0 = amateur (B), 1 = pro (A) + D6 P0 difficulty 0 = amateur (B), 1 = pro + (A) + D5/D4 (not used) + D3 color - B/W 0 = B/W, 1 = color + D2 (not used) + D1 game select 0 = switch pressed + D0 game reset 0 = switch pressed + +5.0 Port A - Hand Controllers + + Port A is under full software control to be configured as + an input or an output port. It can then be used to read or + control various hand-head controllers with the data bits + defined differently depending on the type of controller + used. + + 5.1 Setting for input or output + Port A has an 8 bit wide Data Direction Register (DDR) that + is written to at SWACNT (HEX 281) to set each individual + pin of Port A to either input or output. The Port A pins + are labeled PA0 thru PA7, and writing a "0" to a pins' DDR + bit sets that pin as input, and a "1" sets it as an output. + For example, writing all 0's to SWACNT (the DDR for Port A) + sets PA0 thru PA7 (all 8 pins of Port A) as inputs. If F0 + (11110000) were written to SWACNT then PA7, PA6, PA5 & PA4 + would be outputs, and PA3, PA2, PA1 & PA0 would be inputs. + + 5.2 Inputting and Outputting + Once the DDR has set the pins of Port A for input or output + they may be read or written to by addressing SWCHA (HEX + 280). + + 5.3 Joystick Controllers + Two joysticks can be read by configuring the entire port as + input and reading the data at SWCHA according to the + following table: + + Data Bit Direction Player + D7 right P0 + D6 left P0 + D5 down P0 + D4 up P0 + D3 right P1 + D2 left P1 + D1 down P1 + D0 up P1 + + (P0 = left player, P1 = right player) + + A "0" in a data bit indicates the joystick has been moved + to close that switch. All "1's" in a player's nibble + indicates that joystick is not moving. + + 5.4 Paddle (pot) controllers + Only the paddle triggers are read from the PIA. The + paddles themselves are read at INP0 thru INPT3 of the TIA. + The paddle triggers can be read at SWCHA according to the + following table : + + Data Bit Paddle # + D7 P0 + D6 P1 + D5/D4 (not used) + D3 P2 + D2 P3 + D1/D0 (not used) + + 5.5 Keyboard controllers + The keyboard controller has 12 buttons arranged into 4 rows + and 3 columns. A signal is sent to a row, then the columns + are checked to see if a button is pushed, then the next row + is signaled and all columns sensed, etc. until the entire + keyboard is scanned and sensed. The PIA sends the signals + to the rows, and the columns are sensed by reading INPT0, + INPT1, and INPT4 of the TIA. With Port A configured as an + output port, the data bits will send a signal to the + keyboard controller rows according to the following table : + + Data Bit Keyboard Row Player + D7 bottom P0 + D6 third P0 + D5 second P0 + D4 top P0 + D3 bottom P1 + D2 third P1 + D1 second P1 + D0 top P1 + (P0 = left player, P1 = right player) + + NOTE : a delay of 400 microseconds is necessary between + writing to this port and reading the TIA input ports. + +6.0 Address summary table + + Hex Address Mnemonic Purpose + 280 SWCHA Port A; input or output (read or + write) + 281 SWACNT Port A DDR, 0= input, 1=output + 282 SWCHB Port B; console switches (read only) + 283 SWBCNT Port B DDR (hardwired as input) + 284 INTIM Timer output (read only) + 294 TIM1T set 1 clock interval (838 + nsec/interval) + 295 TIM8T set 8 clock interval (6.7 + usec/interval) + 296 TIM64T set 64 clock interval (53.6 + usec/interval) + 297 T1024T set 1024 clock interval (858.2 + usec/interval) + + NOTE: one clock is also one microprocessor machine cycle + +PAL/SECAM CONVERSIONS + +PAL + 1. The number of scan lines, and therefore the frame time + increases from NTSC to PAL according to the following + table: + + NTSC PAL + scan micro scan micro + lines seconds lines seconds + VBLANK 40 2548 48 3085 + KERNAL 192 12228 228 14656 + OVERSCAN 30 1910 36 2314 + FRAME 262 16686 312 20055 + + 2. Sounds will drop a little in pitch (frequency) because + of a slower crystal clock. Some sounds may need the + AUDF0/AUDF1 touched up. + + 3. PAL operates at 50 Hz compared to NTSC 60Hz, a 17% + reduction. If game play speed is based on frames per + second, it will slow down by 17%. This can be disastrous + for most skill/action carts. If the NTSC version is + designed with 2 byte fractional addition techniques (or + anything not based on frames per second) to move objects, + then PAL conversion can be as simple as changing the + fraction tables, avoiding major surgery on the program. + +SECAM + 1. SECAM is a little weird. It takes the PAL software, but + the console color/black & white switch is hardwired as + black & white. Therefore, it reads the PAL black & white + tables in software and assigns a fixed color to each lum of + black & white according to the following table: + + Lum Color + 0 Black + 2 Blue + 4 Red + 6 Magenta + 8 green + A cyan + C yellow + E white + + There is a trap here: the manual is the same for NTSC, + PAL, & SECAM. This means that the descriptions for black & + white must jive between NTSC & PAL. If you make major + changes to PAL black & white to achieve good SECAM color, + NTSC black & white must be made similar. + + 2. PAL sounds work fine on SECAM with one exception: when a + sound is to be turned off, it must be one by setting + AUDV0/AUDV1 to 0, not by setting AUDC0/AUDC1 to 0. + Otherwise, you get an obnoxious background sound. + + +TIA 1A - TELEVISION INTERFACE ADAPTOR (MODEL 1A) + +GENERAL DESCRIPTION + + The TIA is an MOS integrated circuit designed to interface + between an eight (8) bit microprocessor and a television + video modulator and to convert eight (8) bit parallel data + into serial outputs for the color, luminosity, and + composite sync required by a video modulator. + + This circuit operates on a line by line basis, always + outputting the same information every television line + unless new data is written into it by the microprocessor. + + A hardware sync counter produces horizontal sync timing + independent of the microprocessor. Vertical sync timing is + supplied to this circuit by the microprocessor and combined + into composite sync. + + Horizontal position counters are used to trigger the serial + output of five (5) horizontally movable objects; two + players, two missiles and a ball. The microprocessor can + add or subtract from these position counters to move these + objects right or left. + + The microprocessor determines all vertical position and + motion by writing zeros or ones into object registers + before each appropriate horizontal line. + + Walls, clouds and other seldom moved objects are produced + by a low resolution data register called the playfield + register. + + A fifteen (15) bit collision register detects all fifteen + possible two object collisions between these six (6) + objects (five moveable and one playfield). This collision + register can be read and reset by the microprocessor. Six + input ports are also provided on this chip that can be read + by the microprocessor. These input ports and the collision + register are the only chip addresses that can be read by + the microprocessor. All other addresses are write only. + + Color luminosity registers are included that can be + programmed by the microprocessor with eight (8) luminosity + and fifteen (15) color values. A digital phase shifter is + included on this chip to provide a single color output with + fifteen (15) phase angles. + + Two (2) independent audio generating circuits are included, + each with programmable frequency, noise content, and volume + control registers. + + +DETAILED DESCRIPTION + +1. Data and addressing + + Registers on this chip are addressed by the microprocessor + as part of its overall RAM-ROM memory space. The attached + table of read-write addresses summarizes the addressable + functions. There are no registers that are both read and + write. Some addresses however are both read and write, + with write data going into one register and read data + returning from a different register. + + If the read-write line is low, the data bits indicated in + this table will be written into the addressed write + location when the 02 clock goes from high to low. Some + registers are eight bits wide, some only one bit, and some + (strobes) have no bits, performing only control functions + (such as resets) when their address is written. + + If the read-write line is high, the addressed location can + be read by the microprocessor on data lines 6 and 7 while + the 02 clock is high. + + The addresses given in the table refer only to the six (6) + real address lines. If any of the four (4) chip select + lines are used for addressing, the addresses must be + modified accordingly. + +2. Synchronization + + A. Horizontal Timing + A hardware counter on this chip produces all horizontal + timing (such as sync, blank, burst) independent of the + microprocessor, This counter is driven from an external + 3.58 Mhz oscillator and has a total count of 228. Blank is + decoded as 68 counts and sync and color burst as 16 counts. + + B. Vertical Timing + There are one bit, addressable registers on this chip for + vertical sync and vertical blank. The timing for these + functions is established by the microprocessor by writing + zero or one into these bits. (VSYNC, VBLANK ) + + C. Composite Sync + Horizontal sync and the output of the vertical sync bit are + combined together to produce composite sync. This + composite sync signal drives a chip output pad to an + external composite video resistor network. + + D. Microprocessor Synchronization + The 3.58 MHz oscillator also clocks a divide by three + counter on this chip whose output (1.19 Mhz) is buffered to + drive an output pad called 00. This pad provides the input + phase zero clock to the microprocessor which then produces + the system 02 clock (1.19 Mhz). + Software program loops require different lengths of time to + run depending on branch decisions made within the program. + Additional synchronization between the software and + hardware. This is done with a one bit latch called WSYNC + (wait for sync). When the microprocessor finishes a + routine such as loading registers for a horizontal line, or + computing new vertical locations during vertical blank, it + can address WSYNC, setting this latch high. When this + latch is high, it drives an output pad to zero connected to + the microprocessor ready line (RDY). A zero on this line + causes the microprocessor to halt and wait. As shown in + figure 2, WSYNC latch is automatically reset to zero by the + leading edge of the next horizontal blank timing signal, + releasing the RDY line, allowing the microprocessor to + begin its computation and register writing for this + horizontal television line or line pair. + +3. Playfield graphics Register + + A. Description + Objects such as walls, clouds, and score) which are not + required to move, are written into a 20 bit register called + the playfield register. This register (Figure 5) is loaded + from the data bus by three separate write addresses (PF0, + PFl, PF2). Playfield may be loaded at any time. To clear + the playfield, zeros must be written into all three + addresses. + + B. Normal Serial Output + The playfield register is automatically scanned (and + converted to serial output) by a bi-directional shift + register clocked at a rate which spreads the twenty (20) + bits out over the left half of a horizontal line. This + scanning is initiated by the end of horizontal blank (left + edge of television screen). Normally the same scan is then + repeated, duplicating the same twenty (20) bit sequence + over the right half of the horizontal line. + + C. Reflected Serial Output + A reflected playfield may be requested by writing a one + into bit zero of the playfield control register (CTRLPF). + When this bit is true the scanning shift register will scan + the opposite direction during the right half of the + horizontal line, reversing the twenty (20) bit sequence. + + D. Timing Constraints + Even though the playfield bytes (PF0, PFl, PF2) may be + written to any time, if one of them is changed while being + serially scanned, part of the new value may both show up on + the television horizontal line. + +4. Horizontal Position Counters + + A. Description + The playfield is a fixed graphics register, always starting + its serial output when triggered by the beginning of each + television line. This chip also includes five "moveable" + graphics registers, whose serial outputs are triggered by + five separate horizontal position counters every time these + counters pass through zero count. These position counters + are clocked continuously during the unblanked portion of + every horizontal line and their count length is exactly + equal to the normal number of clocks supplied to them + during this time. They will therefore pass through zero at + the same time during each horizontal television line and + the triggered outputs will have no horizontal motion. A + typical horizontal counter is shown in figure 4. + + If extra clocks are supplied to these counters (or normal + clocks suppressed) the zero crossing time will shift and + the object will have moved left (extra clocks) or right + (fewer clocks). Some position counters have extra decoders + (in addition to a zero decode) to trigger multiple copies + of the same object across a horizontal line. + + All position counters can be reset to zero count by the + microprocessor at any time, by a write instruction to the + reset addresses (RESBL, RESM0, RESMl, RESP0, RESPl). If + reset occurs during horizontal blank, the object will + appear at the left side of the television screen. Properly + timed resets may position an object at any horizontal + location consistent with the microprocessor cycle time. + + B. Ball position Counter + The ball position counter has only the zero crossing decode + and therefore cannot trigger multiple copies of the ball + graphics. + + C. Player Position Counters + Each player position counter has three decodes in addition + to the zero crossing decode. These decodes are controlled + by bits 0,1,2 of the number size control registers (NUSIZ0, + NUSIZ1), and trigger 1,2, or 3 copies of the player (at + various spacings) across a horizontal line as shown on page + ___. These same control bits are used for the decodes on + the missile position counter, insuring an equal number of + players and missiles. + + D. Missile Position Counters + Missile position counters are identical to player position + counters except that they have another type of reset in + addition to the previously discussed horizontal position + reset. These extra reset addresses (RESMP0, RESMP1) write + data bit 1 into a one bit register whose output is used to + position the missile (horizontally) directly on top of its + corresponding player, and to disable the missile serial + output. + +5. Horizontal Motion Registers + + A. General Description + There are five write only registers on this chip that + contain the horizontal motion values for each of the five + moving objects. A typical horizontal motion register is + shown in figure 4 . The data bus (bits 4 through 7) is + written into these addresses (HMP0, HMPl, HMM0, HMMl, HMBL) + to load these registers with motion values. These registers + supply extra (or fewer) clocks to the horizontal position + counters only when commanded to do so by an HMOVE address + from the microprocessor. These registers may all be cleared + to zero (no motion) simultaneously by an HMCLR command from + the microprocessor, or individually by loading zeros into + each register. These registers are each four bits in + length and may be loaded with positive (left motion), + negative (right motion) or zero (no motion) values. + Negative values are represented in twos complement format. + + B. Timing constraints + These registers may be loaded or cleared at almost any + time. The motion values they contain will be used only when + an HMOVE command is addressed, and then all five motion + values will be used simultaneously into all five horizontal + position counters once. The only timing constraint on this + operation involves the HMOVE command. The HMOVE command + must be located in the microprocessor program immediately + after a wait for sync (WSYNC) command. This assures that + the HMOVE operation begins at the leading edge of + horizontal blank, and has the full blank time to supply + extra or fewer clocks to the horizontal position counters. + These registers should not be modified for at least 24 + Computer cycles after the HMOVE command. + +6. Moving Object Graphics Registers + + A. General Description + There are five graphics registers for moving objects on + this chip. These graphics registers are loaded (written) in + parallel by the microprocessor and like the playfield + register are scanned and converted to serial output. + Unlike the playfield register, which is always scanned + beginning at the left side of each horizontal line, moving + object graphics registers are scanned only when triggered + by a start decode from their horizontal position counter. + A typical graphics register is shown in figure 4 . + + B. Missile Graphics + The graphics registers for both missiles are identical and + very simple. They each consist of a one bit register called + missile enable (ENAM0, ENAM1). This graphics bit is scanned + (outputted) only when triggered by its corresponding + position counter. There are control bits (bits 4, 5, of + NUSIZ0, NUSIZ1) that can stretch this single graphics bit + out over widths of 1, 2, 4, or 8 clocks of horizontal line + time. (A full line is 160 clocks). + + C. Player Graphics + The graphics registers for both players are identical and + are rather complex. They each consist of eight bit parallel + registers (GRP0, GRP1) and a bi-directional parallel to + serial scan counter that converts the parallel data into + serial output. A one bit control register (REFP0, REFP1) + determines the direction (reflection) of the parallel to + serial scan, outputing either D7 through D0, or D0 though + D7. This allows reflection (horizontal flipping) of player + serial graphics data without having to flip the + microprocessor data. + + The clock into the scan counter can be controlled (three + bits of NUSIZ0 and NUSIZ1) to slow the scan rate and + stretch the eight bits of serial graphics out over widths + of 8, 16, or 32 clocks of horizontal line time. These same + control bits are used in the player-missile motion counters + to control multiple copies, so only three player widths ( + scan + rates) are available. + + D. Vertical Delay + Each of the player graphics registers actually consists of + two 8 bit parallel registers. The first (GRP0, GRP1) is + loaded (written) from the microprocessor 8 bit data bus. + The second is automatically loaded from the output of the + first. The reason for this is a complex subject called + vertical delay. A large amount of microprocessor time is + required to generate player, missile and playfield graphics + (table look up, masking, comparisons, etc.) and load these + into this chip's registers. For most game programs this + time is just too large to fit into one horizontal line + time. In fact for most games it will barely fit into two + line times (127 microseconds). Therefore, individual + graphics registers are loaded (written) every two lines, + and used twice for serial output between loads. This type + of programing will obviously limit the vertical height + resolution of objects to multiples of two lines. It also + will limit the resolution of vertical motion to two lines + jumps. Nothing can be done about the vertical height + resolution; however, vertical motion can be resolved to a + single line by addition of a second graphics register that + is automatically parallel loaded from the output of the + first, one line time after the first was loaded from the + data bus. This second graphics register output is + therefore always delayed vertically by one line. A control + bit called vertical delay (VDEL0, VDEL1) selects which of + these two registers is to be used for serial output. If + this control bit is set by the microprocessor between + picture frames, the object will be moved down (delayed) by + one line during the next frame. In most programming + applications player 0 graphics and player 1 graphics are + loaded (written) alternately, during the blank time just + prior to each line as shown in (figure 1). Since GRP0 and + GRP1 addresses from the microprocessor alternate, they are + delayed by one line from each other. The GRP0 address + decode can therefore be used to load the delayed graphics + register for player 1, and GRP1 likewise to load the + delayed graphics register for player 0. The two vertical + delay bits (VDEL0, VDELl) then select delayed or undelayed + registers for player 0 and player 1 as serial outputs. + + E. Ball Graphics + The ball graphics register is almost identical to the + missile graphics register. It also consists of a single + enable bit (ENABL) whose output is triggered by the ball + position counter. It also has two control bits (bits 4, 5 + of CTRLPF) that can stretch this single graphics bit out + over widths of 1, 2, 4, or 8 clocks of horizontal line + time. Unlike the missile graphics; however, the ball + graphics register has capability for vertical delay similar + to the player graphics. A second graphics (enable) bit is + alternately loaded from the output of the first, one line + after the first was loaded from the data bus. A ball + vertical delay bit (VDELBL) selects which of these two + graphics bits is used for the ball serial output. The first + graphics bit (ENABL) should be loaded during the same + horizontal blank time as player 0 (GRP0), because GRP1 is + used to load the second enable bit from the output of the + first on alternate lines. + +7. Collision Detection Latches + + A. Definitions + The serial outputs from all the graphics registers + represent real time horizontal location of objects on the + television screen. If any of these outputs occur at the + same time, they will overlap (collide) on the screen. + There are six objects generated on this chip (five moving + and playfield) allowing fifteen possible two object + collisions. These overlaps (collisions) are detected by + fifteen "and" gates whenever they occur, and are stored in + fifteen individual latch register bits, as shown in figure + 6. + + B. Reading Collision + The microprocessor can read these fifteen collision bits on + data lines 6 and 7 by addressing them two at a time. This + could be done at any time but is usually done between + frames (during vertical blank) after all possible + collisions have serially occurred. + + C. Reset + All collision bits are reset simultaneously by the + microprocessor using the reset address CXCLR. This is + usually done near the end of vertical blank, after + collisions have been tested. + +8. Input ports + + A. General Description + There are 6 input ports on this chip whose logic state may + be read on data line 7 with read addresses INPT0 through + INPT5. These 6 ports are divided into two types, "dumped" + and "latched". See Figure 8. + + B. Dumped Input Ports (I0 through I3) + These 4 input ports are normally used to read paddle + position from an external potentiometer-capacitor circuit. + In order to discharge these capacitors each of these input + ports has a large transistor, which may be turned on + (grounding the input ports) by writing into bit 7 of the + register VBLANK. When this control bit is cleared the + potentiometers begin to recharge the capacitors and the + microprocessor measures the time required to detect a logic + 1 at each input port. + + As long as bit 7 of register VBLANK is zero, these four + ports are general purpose high impedance input ports. When + this bit is a 1 these ports are grounded. + + C. Latched Input ports (I4, I5) + These two input ports have latches which can be enabled or + disabled by writing into bit 6 of register VBLANK. + + When disabled, these latches are removed from the circuit + completely and these ports become two general purpose input + ports, whose present logic state can be read directly by + the microprocessor. + + When enabled, these latches will store negative (zero logic + level) signals appearing on these two input ports, and the + input port addresses will read the latches instead of the + input ports. + + When first enabled these latches will remain positive as + long as the input ports remain positive (logic one). A zero + input port signal will clear a latch value to zero, where + it will remain (even after the port returns positive) until + disabled. Both latches may be simultaneously disabled by + writing a zero into bit 6 of register VBLANK. + +8.5 Priority Encoder + + A. Purpose + As discussed in the section on collisions, simultaneous + serial outputs from the graphics registers represent + overlap on the television screen. In order to have color- + luminosity values assigned to individual objects it is + necessary to establish priorities between objects when + overlapped. The priority encoder is shown in figure 3. + + B. Priority Assignment + The lack of any objects results in a color-lum value called + the background. The background (BK) has lowest priority and + only appears when no objects are outputing. In order to + simplify the logic, each missile is given the same color- + lum value and priority as it's corresponding player (P0, + M0) and the ball is given the same color-lum value and + priority as the playfield (PF, BL). + + The following table illustrates the normal priority + assignment: + + Highest Priority P0, M0 + Second Highest P1, M1 + Third Highest PF, BL + Lowest Priority BK + + Objects with higher priority will appear to move in front + of objects with lower priority. Players will therefore move + in front of playfield (clouds, walls, etc.). + + C. Priority Control + There are two playfield control bits that affect priority, + one called playfield priority (PFP) (bit 2 of CTRLPF) and + one called score (bit 1 of CTRLPF). When a one is written + into the PFP bit the priority assignment is modified as + shown below. + + Highest Priority PF, BL + Second Highest P0, M0 + Third Highest P1, M1 + Lowest Priority BK + + + Players will then move behind playfield (clouds, wall, + etc.). When a one is written into the score control bit, + the playfield is forced to take the color-lum of player 0 + in the left half of the screen and player 1 in the right + half of the screen. This is used when displaying score and + identifies the score with the correct player. The priority + encoder produces 4 register select lines shown in figure 3) + that are mutually exclusive. These 4 lines select either + background, player 0, player 1 or playfield, and only one + of them can be true at a time. + +9 Color Luminance Registers + + A. Description + There are four registers (shown in figure 3) that contain + color-lum codes. Four bits of color code and three its of + luminance code may be written into each of these registers + (COLUP0, COLUP1, COLUPF, COLUBK) by the microprocessor at + any time. These codes (representing 16 color values and 8 + luminance values) are given in the Detailed Address List. + + B. Multiplexing + The serial graphics output from all six objects is examined + by the priority encoder which activates one of the four + select lines into a 4 x 7 multiplexer. This multiplexer + (shown in figure 3) then selects one of the four color-lum + registers as a 7 line output. Three of these lines are + binary coded luminosity and go directly to chip output + pads. The other four lines go to the color phase shifter. + +10. Color Phase Shifter + + This portion of the chip (shown in figure 3) produces a + reference color output (color burst) during horizontal + blank and then during the unblanked portion of the line it + produces a color output shifted in phase with respect to + the color burst. The amount of phase shift determines the + color and is selected by the four color code lines from the + Color-lum multiplexer. Binary code 0 selects no color. + Code 1 selects gold (same phase as color burst). Codes 2 + (0010) through 15 (1111) shift the phase from zero through + almost 360 degrees allowing selection of 15 total colors + around the television color wheel. + +11. Audio Circuits + + Two audio circuits are incorporated on this chip. They are + identical and completely independent, although their + outputs could be combined externally into one speaker. + Each audio circuit consists of parts described below, and + in figure 7. + + A. Frequency Select + Clock pulses (at approximately 30 KHz) from the horizontal + sync counter pass through a divide by N circuit which is + controlled by the output code from a five bit frequency + register (AUDF). This register can be loaded (written) by + the microprocessor at any time, and causes the 30 KHz + clocks to be divided by 1 (code 00000) through 32 (code + 11111). This produces pulses that are digitally adjustable + from approximately 30 KHz to 1 KHz and are used to clock + the noise-tone generator. + + B. Noise-Tone Generator + This circuit contains a nine bit shift counter which may be + controlled by the output code from a four bit audio control + register(AUDC), and is clocked by the frequency select + circuit. The control register can be loaded by the + microprocessor at any time, and selects different shift + counter feedback taps and count lengths to produce a + variety of noise and tone qualities. + + C. Volume Select + The shift counter output is used to drive the audio output + pad through four driver transistors that are graduated in + size. Each transistor is twice as large as the previous one + and is enable by one bit from the audio volume register + (AUDV). This audio volume register may be loaded by the + microprocessor at any time. As binary codes 0 through 15 + are loaded, the pad drive transistors are enabled in a + binary sequence. The shift counter output therefore can + pull down on the audio output pad with 16 selectable + impedance levels. + + + + +Figure 1. Vertical Delay + + + + + + +Figure 2. Synchronization + + + +Figure 3. Color-Luminance + + + + +Figure 4. Typical Horizontal Motion Circuit + + + + + + + + +Figure 5. Playfield Graphics + + + + + +Figure 6. Collision Detection + + + + + + +Figure 7. Audio Circuit + + + + + +Figure 8. Input Ports + + + + + +Figure 9. Game System + + + +Write Address Detailed Functions + + +WSYNC (wait for sync) + This address halts microporcessor by clearing RDY latch to + zero. RDY is set true again by the leading edge of + horizontal blank. + + Data bits not used + + +RSYNC (reset sync) + This address resets the horizontal sync counter to define + the beginning of horizontal blank time, and is used in chip + testing. + + Data bits not used + + +VSYNC + This address controls vertical sync time by writing D1 into + the VSYNC latch + + D1 [1 = start vert sync, 0 = stop vertical sync] + +VBLANK + This address controls vertical blank and the latches and + dumping transistors on the input ports by writing into bits + D7, D6 and D1 of the VBLANK register. + + D1 [ 1 = start vert. blank, 0 = stop vert. blank] + D6 [ 1 = Enable I4 I5 latches, 0 = disable I4 I5 latches] + D7 [ 1 = dump I6I1I2I3 ports to ground, 0 = remove dump + path to ground] + Note : Disable latches (D6 = 0) also resets latches to + logic true + +PJ0 (PF1, PF2) + These addresses are used to write into playfield registers + + + PLAYFIELD REGISTERS SERIAL OUTPUT + + 1 horizontal line ( 160 clocks) + Playfield + Reflect Control + PF0 PF1 PF2 PF0 PF1 PF2 + + center + + PF0 PF1 PF2 PF2 PF1 + PF0 + + each bit = 4 clocks + +CTRLPF + This address is uded to write into the playfield control + register (a logic 1 causes action as described below) + + D0 = REF (reflect playfield) + D1 = SCORE (left half of playfield gets color of player 0, + right half gets color of player 1) + D2 = PFP (playfield gets priority over players so they can + move behind the playfield) + D4 & D5 = BALL SIZE + D5 D4 Width + 0 0 1 clock + 0 1 2 clocks + 1 0 4 clocks + 1 1 8 clocks + +NUSIZ0 (NUSIZ1) + These addresses control the number and size of players and + missiles. + + Missile Size D5 D4 Width + 0 0 1 clock + 0 1 2 clocks + 1 0 4 clocks + 1 1 8 clocks + + Player-Missile number & player size + + 1/2 television line (80 clocks) + 8 clocks per square + + +RESP0 (RESP1, RESM0, RESM1, RESBL) + These addresses are used to reset players, missiles and the + ball. The object will begin its serial graphics at the + time of a horizontal line at which the reset address + occurs. + + No data bits are used + + +RESMP0 (RESMP1) + These addresses are used to reset the hoiz. location of a + missile to the center of it’s corresponding player. As + long as this control bit is true (1) the missile will + remain locked to the center of it’s player and the missile + graphics will be siddabled. When a zero is written into + this location, the missile is enabled, and can be moved + independently from the player. + + D1 = RESMP (missile-player reset) +HMOVE + This address causes the horizontal motion register values + to be acted upon during the horizontal blank time in which + it occurs. It must occur at the beginning of horiz. blank + in order to allow time for generation of extra clock pulses + into the horizontal position counters if motion is desired + this command must immediately follow a WSYNC command in the + program. + + No data bits are used + + +HMCLR + This address clears all horizontal motion registers to zero + (no motion) + + No data bits are used + + +HMP0 (HMP1, HMM0, HMM1, HMBL) + These addresses write data (horizontal motion values) into + the horizontal motion registers. These registers will + cause horizontal motion only when commanded to do so by the + horiz. move command HMOVE. + The motion values are coded as shown below : + + D7 D6 D5 D4 + 0 1 1 1 +7 + 0 1 1 0 +6 + 0 1 0 1 +5 Move left + 0 1 0 0 +4 indicated number + 0 0 1 1 +3 of clocks + 0 0 1 0 +2 + 0 0 0 1 +1 + 0 0 0 0 0 No Motion + 1 1 1 1 -1 + 1 1 1 0 -2 + 1 1 0 1 -3 + 1 1 0 0 -4 move right + 1 0 1 1 -5 indicated number + 1 0 1 0 -6 of clocks + 1 0 0 1 -7 + 1 0 0 0 -8 + + WARNING : These motion registers should not be modified + during the 24 computer cycles immediately following an + HMOVE command. Unpredictable motion values may result. +ENAM0 (ENAM1, ENABL) + These addresses write D1 into the 1 bit missile or ball + graphics registers. + + D1 - [0 = dissables object, 1 = enables object] + + +GRP0 (GRP1) + These addresses write data into the player graphics + registers. + + Note: serial output begins with D7, unless REFP0 (REFP1) = + 1 + +REFP0 (REFP1) + These addesses write D3 into the 1 bit player reflect + registers + + D3 - [ 0 = no reflect, D7 of GRP on left, 1 = reflect, D0 + of GRP on left] + +VDELP0 (VDELP1, VDELBL) + These addresses write D0 into the 1 bit vertical delay + registers, to delay players or ball by one vertical line. + + D0 - [0 = no delay, 1 = delay] + +CXCLR + This adderess clears all collision latches to zero (no + collision) + + No data bits are used + +COLUP0 (COLUP1, COLUPF, COLUBK) + These addresses write data into the player, playfield, adn + background color-luminance registers + + COLOR D7 D6 D5 D4 D3 D2 D1 LUM + grey - gold 0 0 0 0 0 0 0 black + 0 0 0 1 0 0 1 dark grey + orange, brt-0 0 1 0 0 1 0 + org + 0 0 1 1 0 1 1 grey + pink - 0 1 0 0 1 0 0 + purple + 0 1 0 1 1 0 1 + purp-blue, 0 1 1 0 1 1 0 light gret + blue + 0 1 1 1 1 1 1 white + blue - lt. 1 0 0 0 + blue + 1 0 0 1 + torq. - 1 0 1 0 + grn. blue + 1 0 1 1 + grn. - yel. 1 1 0 0 + grn. + 1 1 0 1 + org. grn - 1 1 1 0 + lt org. + 1 1 1 1 + +AUDF0 (AUDF1) + These addresses write data into the audio frequency divider + registers. + + D4 D3 D2 D1 D0 30KHz divided + by + 0 0 0 0 0 no division + 0 0 0 0 1 divide by 2 + 0 0 0 1 0 divide by 3 + .. .. .. .. .. ... + . . . . . + 1 1 1 1 0 divide by 31 + 1 1 1 1 1 divide by 32 + +AUDC0 (AUDC1) + These addresses write data into the audio control registers + which control the noise content and additional division of + the audio output. + + D3 D2 D1 D0 Type of noise + or division + 0 0 0 0 set to 1 + 0 0 0 1 4 bit poly + 0 0 1 0 div 15 -> 4 + bit poly + 0 0 1 1 5 bit poly -> + 4 bit poly + 0 1 0 0 div 2 : pure + tone + 0 1 0 1 div 2 : pure + tone + 0 1 1 0 div 31 : pure + tone + 0 1 1 1 5 bit poly -> + div 2 + 1 0 0 0 9 bit poly + (white noise) + 1 0 0 1 5 bit poly + 1 0 1 0 div 31 : pure + tone + 1 0 1 1 set last 4 + bits to 1 + 1 1 0 0 div 6 : pure + tone + 1 1 0 1 div 6 : pure + tone + 1 1 1 0 div 93 : pure + tone + 1 1 1 1 5 bit poly div + 6 + + +AUDV0 (AUDV1) + These addresses write data into the audio volume registers + which set the pull down impedance driving the audio output + pads. + + D3 D2 D1 D0 Audio Output + Pull down + current + 0 0 0 0 No output + current + 0 0 0 1 lowest + 0 0 1 0 + .. .. .. .. + . . . . + 1 1 1 0 + 1 1 1 1 highest + + +WRITE ADDRESS SUMMARY + 6 bit Address 7 6 5 4 3 2 1 0 Function + addres Name + s + 00 VSYNC 1 vertical sync set-clear + 01 VBLANK 1 1 1 vertical blank set- + clear + 02 WSYNC s t r o b e wait for leading edge + of horizontal blank + 03 RSYNC s t r o b e reset horizontal sync + counter + 04 NUSIZ0 1 1 1 1 1 1 number-size player- + missile 0 + 05 NUSIZ1 1 1 1 1 1 1 number-size player- + missile 1 + 06 COLUP0 1 1 1 1 1 1 1 color-lum player 0 + 07 COLUP1 1 1 1 1 1 1 1 color-lum player 1 + 08 COLUPF 1 1 1 1 1 1 1 color-lum playfield + 09 COLUBK 1 1 1 1 1 1 1 color-lum background + 0A CTRLPF 1 1 1 1 1 control playfield ball + size & collisions + 0B REFP0 1 reflect player 0 + 0C REFP1 1 reflect player 1 + 0D PF0 1 1 1 1 playfield register byte + 0 + 0E PF1 1 1 1 1 1 1 1 1 playfield register byte + 1 + 0F PF2 1 1 1 1 1 1 1 1 playfield register byte + 2 + 10 RESP0 s t r o b e reset player 0 + 11 RESP1 s t r o b e reset player 1 + 12 RESM0 s t r o b e reset missile 0 + 13 RESM1 s t r o b e reset missile 1 + 14 RESBL s t r o b e reset ball + 15 AUDC0 1 1 1 1 audio control 0 + 16 AUDC1 1 1 1 1 1 audio control 1 + 17 AUDF0 1 1 1 1 1 audio frequency 0 + 18 AUDF1 1 1 1 1 audio frequency 1 + 19 AUDV0 1 1 1 1 audio volume 0 + 1A AUDV1 1 1 1 1 audio volume 1 + 1B GRP0 1 1 1 1 1 1 1 1 graphics player 0 + 1C GRP1 1 1 1 1 1 1 1 1 graphics player 1 + 1D ENAM0 1 graphics (enable) + missile 0 + 1E ENAM1 1 graphics (enable) + missile 1 + 1F ENABL 1 graphics (enable) ball + 20 HMP0 1 1 1 1 horizontal motion + player 0 + 21 HMP1 1 1 1 1 horizontal motion + player 1 + 22 HMM0 1 1 1 1 horizontal motion + missile 0 + 23 HMM1 1 1 1 1 horizontal motion + missile 1 + 24 HMBL 1 1 1 1 horizontal motion ball + 25 VDELP0 1 vertical delay player 0 + 26 VDEL01 1 vertical delay player 1 + 27 VDELBL 1 vertical delay ball + 28 RESMP0 1 reset missile 0 to + player 0 + 29 RESMP1 1 reset missile 1 to + player 1 + 2A HMOVE s t r o b e apply horizontal motion + 2B HMCLR s t r o b e clear horizontal motion + registers + 2C CXCLR s t r o b e clear collision latches + +READ ADDRESS SUMMARY + + 6 bit Address 7 6 5 4 3 2 1 0 Function + addres Name + s D7 D6 + 0 CXM0P 1 1 read MO P1 M0 P0 + collision + 1 CXM1P 1 1 read M1 P0 M1 P1 + collision + 2 CXP0FB 1 1 read P0 PF P0 BL + collision + 3 CXP1FB 1 1 read P1 PF P1 BL + collision + 4 CXM0FB 1 1 read M0 PF M0 BL + collision + 5 CXM1FB 1 1 read M1 PF M1 BL + collision + 6 CXBLPF 1 read BL PF unuse + collision d + 7 CXPPMM 1 1 read P0 P1 M0 M1 + collision + 8 INPT0 1 read pot port + 9 INPT1 1 read pot port + A INPT2 1 read pot port + B INPT3 1 read pot port + C INPT4 1 read input + D INPT5 1 read input + Note : I0, I2, I2, I3 + can be grounded under + software control. + I4, I5 can be converted + to latched inputs under + software control + +TIA O0..02 AND LUM TIMING +TIA WRITE TIMING CHARACTERISTICS +TIA READ TIMING CHARACTERISTICS + +TIA COMP-SYN AND READY TIMING + +RSYNC, RES0O, H01, H02, SHB, 02, 0O +TIA RSYNC AND BLANK AND READY TIMING + diff --git a/com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/tia_color.html b/com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/tia_color.html new file mode 100644 index 00000000..75bfc32a --- /dev/null +++ b/com.wudsn.ide.asm/help/www.qotile.net/minidig/docs/tia_color.html @@ -0,0 +1,898 @@ + +TIA Color Charts + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
NTSC TIA colors (128 unique colors)
HUE
L
U
M
0123456789ABCDEF
0-1                
2-3                
4-5                
6-7                
8-9                
A-B                
C-D                
E-F                
+