Finished MPROM support. Fixes #10
[Mograsim.git] / plugins / net.mograsim.plugin.core / src / net / mograsim / plugin / launch / MachineLaunchConfigType.java
index 32ea832..a09ff23 100644 (file)
@@ -6,23 +6,27 @@ import java.util.Optional;
 
 import org.eclipse.core.resources.IFile;
 import org.eclipse.core.resources.IProject;
-import org.eclipse.core.resources.IResourceChangeEvent;
-import org.eclipse.core.resources.IResourceChangeListener;
 import org.eclipse.core.resources.ResourcesPlugin;
 import org.eclipse.core.runtime.CoreException;
 import org.eclipse.core.runtime.IProgressMonitor;
 import org.eclipse.core.runtime.IStatus;
 import org.eclipse.core.runtime.Status;
+import org.eclipse.debug.core.DebugPlugin;
 import org.eclipse.debug.core.ILaunch;
 import org.eclipse.debug.core.ILaunchConfiguration;
+import org.eclipse.debug.core.model.IMemoryBlock;
 import org.eclipse.debug.core.model.LaunchConfigurationDelegate;
 import org.eclipse.ui.statushandlers.StatusManager;
 
+import net.mograsim.machine.BitVectorMemoryDefinition;
 import net.mograsim.machine.Machine;
 import net.mograsim.machine.MachineDefinition;
-import net.mograsim.machine.mi.MicroInstructionMemory;
+import net.mograsim.machine.MainMemoryDefinition;
+import net.mograsim.machine.mi.MPROMDefinition;
 import net.mograsim.machine.mi.MicroInstructionMemoryDefinition;
 import net.mograsim.machine.mi.MicroInstructionMemoryParser;
+import net.mograsim.machine.standard.memory.BitVectorBasedMemoryParser;
+import net.mograsim.machine.standard.memory.StandardBitVectorMemory;
 import net.mograsim.plugin.MograsimActivator;
 import net.mograsim.plugin.nature.MachineContext;
 import net.mograsim.plugin.nature.MograsimNature;
@@ -30,23 +34,10 @@ import net.mograsim.plugin.nature.ProjectMachineContext;
 
 public class MachineLaunchConfigType extends LaunchConfigurationDelegate
 {
-       public static final String PROJECT_ATTR = MograsimActivator.PLUGIN_ID + "project";
-       public static final String MPM_FILE_ATTR = MograsimActivator.PLUGIN_ID + "mpm";
-
-       private final IResourceChangeListener resChangedListener;
-
-       public MachineLaunchConfigType()
-       {
-               this.resChangedListener = this::resourceChanged;
-               ResourcesPlugin.getWorkspace().addResourceChangeListener(resChangedListener,
-                               // IResourceChangeEvent.POST_BUILD |
-                               IResourceChangeEvent.POST_CHANGE |
-                               // IResourceChangeEvent.PRE_BUILD |
-                               // IResourceChangeEvent.PRE_CLOSE |
-                               // IResourceChangeEvent.PRE_DELETE |
-                               // IResourceChangeEvent.PRE_REFRESH |
-                                               0);
-       }
+       public static final String PROJECT_ATTR = MograsimActivator.PLUGIN_ID + ".project";
+       public static final String MPM_FILE_ATTR = MograsimActivator.PLUGIN_ID + ".mpm";
+       public static final String MPROM_FILE_ATTR = MograsimActivator.PLUGIN_ID + ".mprom";
+       public static final String INITIAL_RAM_FILE_ATTR = MograsimActivator.PLUGIN_ID + ".initialram";
 
        @Override
        public boolean preLaunchCheck(ILaunchConfiguration configuration, String mode, IProgressMonitor monitor) throws CoreException
@@ -68,13 +59,15 @@ public class MachineLaunchConfigType extends LaunchConfigurationDelegate
 
                MachineDefinition machineDefinition = machDefOptional.orElseThrow();
                MicroInstructionMemoryDefinition miMemDef = machineDefinition.getMicroInstructionMemoryDefinition();
+               MainMemoryDefinition mainMemDef = machineDefinition.getMainMemoryDefinition();
+               MPROMDefinition mpromDef = machineDefinition.getMPROMDefinition();
 
                String mpmFileName = configuration.getAttribute(MPM_FILE_ATTR, "");
                if ("".equals(mpmFileName))
                        return showErrorAndReturnFalse("No MPM file specified");
 
                IFile mpmFile = project.getFile(mpmFileName);
-               if (!mpmFile.isAccessible())
+               if (mpmFile == null || !mpmFile.isAccessible())
                        return showErrorAndReturnFalse("MPM file not accessible");
 
                try (InputStream mpmStream = mpmFile.getContents())
@@ -86,9 +79,39 @@ public class MachineLaunchConfigType extends LaunchConfigurationDelegate
                        throw new CoreException(new Status(IStatus.ERROR, MograsimActivator.PLUGIN_ID, "Unexpected IO exception reading MPM file", e));
                }
 
+               String mpromFileName = configuration.getAttribute(MPROM_FILE_ATTR, "");
+               if (!checkMemoryFileReadable(project, mpromDef, mpromFileName, "MPROM"))
+                       return false;
+
+               String initialRAMFileName = configuration.getAttribute(INITIAL_RAM_FILE_ATTR, "");
+               if (!checkMemoryFileReadable(project, mainMemDef, initialRAMFileName, "initial RAM"))
+                       return false;
+
                return super.preLaunchCheck(configuration, mode, monitor);
        }
 
+       private static boolean checkMemoryFileReadable(IProject project, BitVectorMemoryDefinition mainMemDef, String fileName, String fileType)
+                       throws CoreException
+       {
+               if (!"".equals(fileName))
+               {
+                       IFile initialRAMFile = project.getFile(fileName);
+                       if (initialRAMFile == null || !initialRAMFile.isAccessible())
+                               return showErrorAndReturnFalse(fileType + " file not accessible");
+
+                       try (InputStream initialRAMStream = initialRAMFile.getContents())
+                       {
+                               BitVectorBasedMemoryParser.parseMemory(new StandardBitVectorMemory<>(mainMemDef), initialRAMStream);
+                       }
+                       catch (IOException e)
+                       {
+                               throw new CoreException(
+                                               new Status(IStatus.ERROR, MograsimActivator.PLUGIN_ID, "Unexpected IO exception reading " + fileType + " file", e));
+                       }
+               }
+               return true;
+       }
+
        private static boolean showErrorAndReturnFalse(String message)
        {
                StatusManager.getManager().handle(new Status(IStatus.ERROR, MograsimActivator.PLUGIN_ID, message, null), StatusManager.SHOW);
@@ -103,55 +126,27 @@ public class MachineLaunchConfigType extends LaunchConfigurationDelegate
 
                MachineContext machineContext = ProjectMachineContext.getMachineContextOf(project);
                MachineDefinition machineDefinition = machineContext.getMachineDefinition().orElseThrow();
-               MicroInstructionMemoryDefinition miMemDef = machineDefinition.getMicroInstructionMemoryDefinition();
 
                IFile mpmFile = project.getFile(configuration.getAttribute(MPM_FILE_ATTR, ""));
-
-               MicroInstructionMemory mpm;
-               try (InputStream mpmStream = mpmFile.getContents())
-               {
-                       mpm = MicroInstructionMemoryParser.parseMemory(miMemDef, mpmStream);
-               }
-               catch (IOException e)
-               {
-                       throw new CoreException(new Status(IStatus.ERROR, MograsimActivator.PLUGIN_ID, "Unexpected IO exception reading MPM file", e));
-               }
-
-               MachineDebugTarget debugTarget = new MachineDebugTarget(new MachineProcess(launch, machineDefinition));
+               Optional<IFile> mpromFile = fileOptional(project, configuration.getAttribute(MPROM_FILE_ATTR, ""));
+               Optional<IFile> memFile = fileOptional(project, configuration.getAttribute(INITIAL_RAM_FILE_ATTR, ""));
+               MachineDebugTarget debugTarget = new MachineDebugTarget(launch, mpmFile, mpromFile, memFile, machineDefinition);
+               // TODO make selectable whether the machine starts paused or not
+               debugTarget.suspend();
                debugTarget.setExecutionSpeed(1);
                Machine machine = debugTarget.getMachine();
-               machine.getMicroInstructionMemory().bind(mpm);
                machine.reset();
+
+               // Add the default Mograsim memory block to make it less confusing and more comfortable.
+               DebugPlugin.getDefault().getMemoryBlockManager()
+                               .addMemoryBlocks(new IMemoryBlock[] { new MainMemoryBlockExtension(debugTarget, "0", null) });
        }
 
-       private void resourceChanged(IResourceChangeEvent event)
+       private static Optional<IFile> fileOptional(IProject project, String filename)
        {
-               // TODO react to MPM changes
-               int type = event.getType();
-               String typeStr;
-               switch (type)
-               {
-               case IResourceChangeEvent.POST_BUILD:
-                       typeStr = "POST_BUILD";
-                       break;
-               case IResourceChangeEvent.POST_CHANGE:
-                       typeStr = "POST_CHANGE";
-                       break;
-               case IResourceChangeEvent.PRE_BUILD:
-                       typeStr = "PRE_BUILD";
-                       break;
-               case IResourceChangeEvent.PRE_CLOSE:
-                       typeStr = "PRE_CLOSE";
-                       break;
-               case IResourceChangeEvent.PRE_DELETE:
-                       typeStr = "PRE_DELETE";
-                       break;
-               case IResourceChangeEvent.PRE_REFRESH:
-                       typeStr = "PRE_REFRESH";
-                       break;
-               default:
-                       typeStr = "<unknown: " + type + ">";
-               }
-               System.out.println(typeStr + ": " + event);
+               if ("".equals(filename))
+                       return Optional.empty();
+               return Optional.of(project.getFile(filename));
        }
+
 }
\ No newline at end of file