/** * 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 (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 { 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"); } } 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; } }