Skip to content
Snippets Groups Projects
GenericGcodeDriver.java 14.5 KiB
Newer Older
  • Learn to ignore specific revisions
  • /**
     * This file is part of LibLaserCut.
     * Copyright (C) 2011 - 2014 Thomas Oster <mail@thomas-oster.de>
     *
     * LibLaserCut is free software: you can redistribute it and/or modify
     * it under the terms of the GNU Lesser General Public License as published by
     * the Free Software Foundation, either version 3 of the License, or
     * (at your option) any later version.
     *
     * LibLaserCut is distributed in the hope that it will be useful,
     * but WITHOUT ANY WARRANTY; without even the implied warranty of
     * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
     * GNU Lesser General Public License for more details.
     *
     * You should have received a copy of the GNU Lesser General Public License
     * along with LibLaserCut. If not, see <http://www.gnu.org/licenses/>.
     *
     **/
    package com.t_oster.liblasercut.drivers;
    
    import com.t_oster.liblasercut.*;
    import com.t_oster.liblasercut.platform.Util;
    import java.io.BufferedReader;
    import java.io.IOException;
    import java.io.InputStreamReader;
    import java.io.PrintStream;
    import java.io.UnsupportedEncodingException;
    import java.net.InetSocketAddress;
    import java.net.Socket;
    import purejavacomm.*;
    import java.util.*;
    
    /**
     * This class implements a driver for a generic GRBL GCode Lasercutter.
     * It should contain all possible options and is inteded to be the superclass
     * for e.g. the SmoothieBoard and the Lasersaur driver.
     *
     * @author Thomas Oster <thomas.oster@rwth-aachen.de>
     */
    public class GenericGcodeDriver extends LaserCutter {
    
      protected static final String SETTING_HOST = "IP/Hostname";
      protected static final String SETTING_COMPORT = "COM Port";
      protected static final String SETTING_BEDWIDTH = "Laserbed width";
      protected static final String SETTING_BEDHEIGHT = "Laserbed height";
      protected static final String SETTING_MAX_SPEED = "Max speed (in mm/min)";
      protected static final String SETTING_PRE_JOB_GCODE = "Pre-Job GCode (comma separated)";
      protected static final String SETTING_POST_JOB_GCODE = "Post-Job GCode (comma separated)";
      protected static final String SETTING_IDENTIFICATION_STRING = "Board Identification String";
      protected static final String SETTING_WAIT_FOR_OK = "Wait for OK after each line (interactive mode)";
      protected static final String LINEEND = "\r\n";
      
      protected boolean waitForOKafterEachLine = true;
    
      public boolean isWaitForOKafterEachLine()
      {
        return waitForOKafterEachLine;
      }
    
      public void setWaitForOKafterEachLine(boolean waitForOKafterEachLine)
      {
        this.waitForOKafterEachLine = waitForOKafterEachLine;
      }
    
      public String getIdentificationLine()
      {
        return identificationLine;
      }
    
      public void setIdentificationLine(String identificationLine)
      {
        this.identificationLine = identificationLine;
      }
      
      protected String preJobGcode = "G21,G90";
    
      public String getPreJobGcode()
      {
        return preJobGcode;
      }
    
      public void setPreJobGcode(String preJobGcode)
      {
        this.preJobGcode = preJobGcode;
      }
      
      protected String postJobGcode = "G0 X0 Y0";
    
      public String getPostJobGcode()
      {
        return postJobGcode;
      }
    
      public void setPostJobGcode(String postJobGcode)
      {
        this.postJobGcode = postJobGcode;
      }
      
      /**
       * What is expected to be received after serial/telnet connection
       * Used e.g. for auto-detecting the serial port.
       */
      protected String identificationLine = "Grbl";
      
      @Override
      public String getModelName() {
        return "Generic GRBL GCode Driver";
      }
      
      protected String host = "10.10.10.222";
    
      public String getHost()
      {
        return host;
      }
    
      public void setHost(String host)
      {
        this.host = host;
      }
      
      protected String comport = "auto";
    
      public String getComport()
      {
        return comport;
      }
    
      public void setComport(String comport)
      {
        this.comport = comport;
      }
      
      protected double max_speed = 20*60;
    
      public double getMax_speed()
      {
        return max_speed;
      }
    
      public void setMax_speed(double max_speed)
      {
        this.max_speed = max_speed;
      }
      
      @Override
      /**
       * We do not support Frequency atm, so we return power,speed and focus
       */
      public LaserProperty getLaserPropertyForVectorPart() {
        return new PowerSpeedFocusProperty();
      }
    
      protected void writeVectorGCode(VectorPart vp, double resolution) throws UnsupportedEncodingException, IOException {
        for (VectorCommand cmd : vp.getCommandList()) {
          switch (cmd.getType()) {
            case MOVETO:
              int x = cmd.getX();
              int y = cmd.getY();
              move(out, x, y, resolution);
              break;
            case LINETO:
              x = cmd.getX();
              y = cmd.getY();
              line(out, x, y, resolution);
              break;
            case SETPROPERTY:
              PowerSpeedFocusProperty p = (PowerSpeedFocusProperty) cmd.getProperty();
              setPower(p.getPower());
              setSpeed(p.getSpeed());
              setFocus(out, p.getFocus(), resolution);
              break;
          }
        }
      }
      private double currentPower = -1;
      private double currentSpeed = -1;
      private double nextPower = -1;
      private double nextSpeed = -1;
      private double currentFocus = 0;
    
      protected void setSpeed(double speedInPercent) {
        nextSpeed = speedInPercent;
      }
    
      protected void setPower(double powerInPercent) {
        nextPower = powerInPercent;
      }
      
      protected void setFocus(PrintStream out, double focus, double resolution) throws IOException {
        if (currentFocus != focus)
        {
          sendLine("G0 Z%f", Util.px2mm(focus, resolution));
          currentFocus = focus;
        }
      }
    
      protected void move(PrintStream out, int x, int y, double resolution) throws IOException {
        sendLine("G0 X%f Y%f", Util.px2mm(x, resolution), Util.px2mm(y, resolution));
      }
    
      protected void line(PrintStream out, int x, int y, double resolution) throws IOException {
        String append = "";
        if (nextPower != currentPower)
        {
          append += String.format(Locale.US, " S%f", nextPower);
          currentPower = nextPower;
        }
        if (nextSpeed != currentSpeed)
        {
          append += String.format(Locale.US, " F%d", (int) (max_speed*nextSpeed/100.0));
          currentSpeed = nextSpeed;
        }
        sendLine("G1 X%f Y%f"+append, Util.px2mm(x, resolution), Util.px2mm(y, resolution));
      }
    
      private void writeInitializationCode() throws IOException {
        if (preJobGcode != null)
        {
          for (String line : preJobGcode.split(","))
          {
            sendLine(line);
          }
        }
      }
    
      private void writeShutdownCode() throws IOException {
        if (postJobGcode != null)
        {
          for (String line : postJobGcode.split(","))
          {
            sendLine(line);
          }
        }
      }
    
      private BufferedReader in;
      private PrintStream out;
      private Socket socket;
      private CommPort port;
      private CommPortIdentifier portIdentifier;
      
      protected void sendLine(String text, Object... parameters) throws IOException
      {
        out.format(text+LINEEND, parameters);
        if (isWaitForOKafterEachLine())
        {
          if (!"ok".equals(in.readLine()))
          {
            throw new IOException("Smoothieboard did not respond 'ok'");
          }
        }
      }
      
      protected String connect_serial(CommPortIdentifier i, ProgressListener pl) throws PortInUseException, IOException
      {
        pl.taskChanged(this, "opening '"+i.getName()+"'");
        if (i.getPortType() == CommPortIdentifier.PORT_SERIAL)
        {
          try
          {
            port = i.open("VisiCut", 1000);
            out = new PrintStream(port.getOutputStream(), true, "US-ASCII");
            in = new BufferedReader(new InputStreamReader(port.getInputStream()));
            if (getIdentificationLine() != null && getIdentificationLine().length() > 0)
            {
              String line = in.readLine();
              if (!getIdentificationLine().equals(line))
              {
                in.close();
                out.close();
                port.close();
                return ("Does not seem to be a smoothieboard on "+i.getName());
              }
            }
            portIdentifier = i;
            return null;
          }
          catch (PortInUseException e)
          {
            return "Port in use "+i.getName();
          }
          catch (IOException e)
          {
            return "IO Error "+i.getName();
          }
          catch (PureJavaIllegalStateException e)
          {
            return "Could not open "+i.getName();
          }
        }
        else
        {
          return "Not a serial Port "+comport;
        }
      }
      
      protected void connect(ProgressListener pl) throws IOException, PortInUseException, NoSuchPortException
      {
    
        if (getHost() != null && getHost().length() > 0)
        {
          socket = new Socket();
          socket.connect(new InetSocketAddress(getHost(), 23), 1000);
          in = new BufferedReader(new InputStreamReader(socket.getInputStream()));
          out = new PrintStream(socket.getOutputStream(), true, "US-ASCII");
        }
        else if (getComport() != null && !getComport().equals(""))
    
        {
          String error = "No serial port found";
          if (portIdentifier == null && !getComport().equals("auto"))
          {
            portIdentifier = CommPortIdentifier.getPortIdentifier(getComport());
          }
          
          if (portIdentifier != null)
          {//use port identifier we had last time
            error = connect_serial(portIdentifier, pl);
          }
          else
          {
            Enumeration<CommPortIdentifier> e = CommPortIdentifier.getPortIdentifiers();
            while (e.hasMoreElements())
            {
              CommPortIdentifier i = e.nextElement();
              if (i.getPortType() == CommPortIdentifier.PORT_SERIAL)
              {
                error = connect_serial(i, pl);
                if (error == null)
                {
                  break;
                }
              }
            }
          }
          if (error != null)
          {
            throw new IOException(error);
          }
        }
        else
        {
    
          throw new IOException("Either COM Port or IP/Host has to be set");
    
        }
      }
      
      protected void disconnect() throws IOException
      {
        in.close();
        out.close();
        if (this.socket != null)
        {
          socket.close();
          socket = null;
        }
        else if (this.port != null)
        {
          this.port.close();
          this.port = null;
        }
      }
      
      @Override
      public void sendJob(LaserJob job, ProgressListener pl, List<String> warnings) throws IllegalJobException, Exception {
        pl.progressChanged(this, 0);
        this.currentPower = -1;
        this.currentSpeed = -1;
    
        pl.taskChanged(this, "checking job");
        checkJob(job);
        job.applyStartPoint();
        pl.taskChanged(this, "connecting...");
        connect(pl);
        pl.taskChanged(this, "sending");
        writeInitializationCode();
        pl.progressChanged(this, 20);
        int i = 0;
        int max = job.getParts().size();
        for (JobPart p : job.getParts())
        {
          if (p instanceof RasterPart)
          {
            RasterPart rp = (RasterPart) p;
            LaserProperty black = rp.getLaserProperty();
            LaserProperty white = black.clone();
            white.setProperty("power", 0);
            p = convertRasterToVectorPart((RasterPart) p, black, white,  p.getDPI(), false);
          }
          if (p instanceof VectorPart)
          {
            //TODO: in direct mode use progress listener to indicate progress 
            //of individual job
            writeVectorGCode((VectorPart) p, p.getDPI());
          }
          i++;
          pl.progressChanged(this, 20 + (int) (i*(double) 60/max));
        }
        writeShutdownCode();
        disconnect();
        pl.taskChanged(this, "sent.");
        pl.progressChanged(this, 100);
      }
      private List<Double> resolutions;
    
      @Override
      public List<Double> getResolutions() {
        if (resolutions == null) {
          resolutions = Arrays.asList(new Double[]{
                    500d
                  });
        }
        return resolutions;
      }
      protected double bedWidth = 250;
    
      /**
       * Get the value of bedWidth
       *
       * @return the value of bedWidth
       */
      @Override
      public double getBedWidth() {
        return bedWidth;
      }
    
      /**
       * Set the value of bedWidth
       *
       * @param bedWidth new value of bedWidth
       */
      public void setBedWidth(double bedWidth) {
        this.bedWidth = bedWidth;
      }
      protected double bedHeight = 280;
    
      /**
       * Get the value of bedHeight
       *
       * @return the value of bedHeight
       */
      @Override
      public double getBedHeight() {
        return bedHeight;
      }
    
      /**
       * Set the value of bedHeight
       *
       * @param bedHeight new value of bedHeight
       */
      public void setBedHeight(double bedHeight) {
        this.bedHeight = bedHeight;
      }
      
      private static String[] settingAttributes = new String[]{
        SETTING_BEDWIDTH,
        SETTING_BEDHEIGHT,
        SETTING_COMPORT,
        SETTING_HOST,
        SETTING_IDENTIFICATION_STRING,
        SETTING_MAX_SPEED,
        SETTING_PRE_JOB_GCODE,
        SETTING_POST_JOB_GCODE,
        SETTING_WAIT_FOR_OK
      };
    
      @Override
      public String[] getPropertyKeys() {
        return settingAttributes;
      }
    
      @Override
      public Object getProperty(String attribute) {
        if (SETTING_HOST.equals(attribute)) {
          return this.getHost();
        } else if (SETTING_BEDWIDTH.equals(attribute)) {
          return this.getBedWidth();
        } else if (SETTING_BEDHEIGHT.equals(attribute)) {
          return this.getBedHeight();
        } else if (SETTING_COMPORT.equals(attribute)) {
          return this.getComport();
        } else if (SETTING_HOST.equals(attribute)) {
          return this.getHost();
        } else if (SETTING_IDENTIFICATION_STRING.equals(attribute)) {
          return this.getIdentificationLine();
        } else if (SETTING_MAX_SPEED.equals(attribute)) {
          return this.getMax_speed();
        } else if (SETTING_PRE_JOB_GCODE.equals(attribute)) {
          return this.getPreJobGcode();
        } else if (SETTING_POST_JOB_GCODE.equals(attribute)) {
          return this.getPostJobGcode();
        } else if (SETTING_WAIT_FOR_OK.equals(attribute)) {
          return this.isWaitForOKafterEachLine();
        }
        
        return null;
      }
    
      @Override
      public void setProperty(String attribute, Object value) {
        if (SETTING_HOST.equals(attribute)) {
          this.setHost((String) value);
        } else if (SETTING_BEDWIDTH.equals(attribute)) {
          this.setBedWidth((Double) value);
        } else if (SETTING_BEDHEIGHT.equals(attribute)) {
          this.setBedHeight((Double) value);
        } else if (SETTING_COMPORT.equals(attribute)) {
          this.setComport((String) value);
        } else if (SETTING_HOST.equals(attribute)) {
          this.setHost((String) value);
        } else if (SETTING_IDENTIFICATION_STRING.equals(attribute)) {
          this.setIdentificationLine((String) value);
        } else if (SETTING_MAX_SPEED.equals(attribute)) {
          this.setMax_speed((Double) max_speed);
        } else if (SETTING_PRE_JOB_GCODE.equals(attribute)) {
          this.setPreJobGcode((String) value);
        } else if (SETTING_POST_JOB_GCODE.equals(attribute)) {
          this.setPostJobGcode((String) value);
        } else if (SETTING_WAIT_FOR_OK.equals(attribute)) {
          this.setWaitForOKafterEachLine((Boolean) value);
        }
      }
    
      @Override
      public GenericGcodeDriver clone() {
        GenericGcodeDriver clone = new GenericGcodeDriver();
        clone.copyProperties(this);
        return clone;
      }
    }