Newer
Older
/**
* This file is part of VisiCut.
* Copyright (C) 2011 Thomas Oster <thomas.oster@rwth-aachen.de>
* RWTH Aachen University - 52062 Aachen, Germany
*
* VisiCut 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.
*
* VisiCut is distributed in the hope that 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 VisiCut. If not, see <http://www.gnu.org/licenses/>.
**/
package com.t_oster.liblasercut.drivers;
import com.t_oster.liblasercut.IllegalJobException;
Thomas Oster
committed
import com.t_oster.liblasercut.JobPart;
import com.t_oster.liblasercut.LaserCutter;
import com.t_oster.liblasercut.LaserJob;
import com.t_oster.liblasercut.LaserProperty;
import com.t_oster.liblasercut.ProgressListener;
import com.t_oster.liblasercut.Raster3dPart;
import com.t_oster.liblasercut.RasterPart;
import com.t_oster.liblasercut.VectorCommand;
import com.t_oster.liblasercut.VectorPart;
import com.t_oster.liblasercut.platform.Point;
import com.t_oster.liblasercut.platform.Util;
Thomas Oster
committed
import java.io.*;
import java.net.InetSocketAddress;
import java.net.Socket;
Thomas Oster
committed
import java.util.ArrayList;
Thomas Oster
committed
import java.util.Collections;
import java.util.LinkedList;
import java.util.List;
import java.util.Locale;
import org.apache.commons.net.tftp.TFTP;
import org.apache.commons.net.tftp.TFTPClient;
/**
* This class implements a driver for the LAOS Lasercutter board.
* Currently it supports the simple code and the G-Code, which may be used in
* the future.
*
* @author Thomas Oster <thomas.oster@rwth-aachen.de>
*/
public class LaosCutter extends LaserCutter
{
private static final String SETTING_HOSTNAME = "Hostname / IP";
private static final String SETTING_PORT = "Port";
private static final String SETTING_BEDWIDTH = "Laserbed width";
private static final String SETTING_BEDHEIGHT = "Laserbed height";
private static final String SETTING_FLIPX = "X axis goes right to left (yes/no)";
private static final String SETTING_FLIPY = "Y axis goes bottom to top (yes/no)";
private static final String SETTING_MMPERSTEP = "mm per Step (for SimpleMode)";
private static final String SETTING_TFTP = "Use TFTP instead of TCP";
private static final String SETTING_RASTER_WHITESPACE = "Additional space per Raster line";
Thomas Oster
committed
private static final String SETTING_UNIDIR = "Engrave unidirectional";
private boolean unidir = false;
public LaosCutterProperty getLaserPropertyForVectorPart()
return new LaosCutterProperty();
public LaosCutterProperty getLaserPropertyForRasterPart()
return new LaosCutterProperty();
public LaosCutterProperty getLaserPropertyForRaster3dPart()
return new LaosCutterProperty();
Thomas Oster
committed
public void setEngraveUnidirectional(boolean uni)
{
this.unidir = uni;
}
public boolean isEngraveUnidirectional()
{
return this.unidir;
}
private double addSpacePerRasterLine = 5;
/**
* Get the value of addSpacePerRasterLine
*
* @return the value of addSpacePerRasterLine
*/
public double getAddSpacePerRasterLine()
{
return addSpacePerRasterLine;
}
/**
* Set the value of addSpacePerRasterLine
* This is a space (in mm) for the laserhead to gain
* speed before the first 'black' pixel in every line
*
* @param addSpacePerRasterLine new value of addSpacePerRasterLine
*/
public void setAddSpacePerRasterLine(double addSpacePerRasterLine)
{
this.addSpacePerRasterLine = addSpacePerRasterLine;
}
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
@Override
public String getModelName()
{
return "LAOS";
}
protected boolean useTftp = true;
/**
* Get the value of useTftp
*
* @return the value of useTftp
*/
public boolean isUseTftp()
{
return useTftp;
}
/**
* Set the value of useTftp
*
* @param useTftp new value of useTftp
*/
public void setUseTftp(boolean useTftp)
{
this.useTftp = useTftp;
}
protected boolean flipXaxis = false;
/**
* Get the value of flipXaxis
*
* @return the value of flipXaxis
*/
public boolean isFlipXaxis()
{
return flipXaxis;
}
/**
* Set the value of flipXaxis
*
* @param flipXaxis new value of flipXaxis
*/
public void setFlipXaxis(boolean flipXaxis)
{
this.flipXaxis = flipXaxis;
}
Thomas Oster
committed
protected boolean flipYaxis = true;
/**
* Get the value of flipYaxis
*
* @return the value of flipYaxis
*/
public boolean isFlipYaxis()
{
return flipYaxis;
}
/**
* Set the value of flipYaxis
*
* @param flipYaxis new value of flipYaxis
*/
public void setFlipYaxis(boolean flipYaxis)
{
this.flipYaxis = flipYaxis;
}
protected String hostname = "192.168.123.111";
/**
* Get the value of hostname
*
* @return the value of hostname
*/
public String getHostname()
{
return hostname;
}
/**
* Set the value of hostname
*
* @param hostname new value of hostname
*/
public void setHostname(String hostname)
{
this.hostname = hostname;
}
/**
* Get the value of port
*
* @return the value of port
*/
public int getPort()
{
return port;
}
/**
* Set the value of port
*
* @param port new value of port
*/
public void setPort(int port)
{
this.port = port;
}
/**
* Get the value of mmPerStep
*
* @return the value of mmPerStep
*/
public double getMmPerStep()
{
return mmPerStep;
}
/**
* Set the value of mmPerStep
*
* @param mmPerStep new value of mmPerStep
*/
public void setMmPerStep(double mmPerStep)
{
this.mmPerStep = mmPerStep;
}
}
private byte[] generateVectorGCode(VectorPart vp, int resolution) throws UnsupportedEncodingException
{
ByteArrayOutputStream result = new ByteArrayOutputStream();
PrintStream out = new PrintStream(result, true, "US-ASCII");
for (VectorCommand cmd : vp.getCommandList())
{
switch (cmd.getType())
{
case MOVETO:
move(out, cmd.getX(), cmd.getY(), resolution);
break;
case LINETO:
line(out, cmd.getX(), cmd.getY(), resolution);
this.setCurrentProperty(out, cmd.getProperty());
}
}
return result.toByteArray();
}
private void move(PrintStream out, float x, float y, int resolution)
out.printf("0 %d %d\n", px2steps(isFlipXaxis() ? Util.mm2px(bedWidth, resolution) - x : x, resolution), px2steps(isFlipYaxis() ? Util.mm2px(bedHeight, resolution) - y : y, resolution));
private void loadBitmapLine(PrintStream out, List<Long> dwords)
{
out.printf("9 %s %s ", "1", ""+(dwords.size()*32));
for(Long d:dwords)
{
out.printf(" "+d);
}
out.printf("\n");
}
private float currentPower = -1;
private void setPower(PrintStream out, float power)
{
if (currentPower != power)
{
out.printf("7 101 %d\n", (int) (power * 100));
currentPower = power;
}
}
private float currentSpeed = -1;
private void setSpeed(PrintStream out, float speed)
{
if (currentSpeed != speed)
{
out.printf("7 100 %d\n", (int) (speed * 100));
currentSpeed = speed;
}
}
private float currentFrequency = -1;
private void setFrequency(PrintStream out, int frequency)
{
if (currentFrequency != frequency)
{
out.printf("7 102 %d\n", frequency);
currentFrequency = frequency;
}
}
private float currentFocus = 0;
private void setFocus(PrintStream out, float focus)
{
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
if (currentFocus != focus)
{
out.printf(Locale.US, "2 %d\n", (int) (focus/this.mmPerStep));
currentFocus = focus;
}
}
private Boolean currentVentilation = null;
private void setVentilation(PrintStream out, boolean ventilation)
{
if (currentVentilation == null || !currentVentilation.equals(ventilation))
{
out.printf(Locale.US, "7 6 %d\n", ventilation ? 1 : 0);
currentVentilation = ventilation;
}
}
private Boolean currentPurge = null;
private void setPurge(PrintStream out, boolean purge)
{
if (currentPurge == null || !currentPurge.equals(purge))
{
out.printf(Locale.US, "7 7 %d\n", purge ? 1 : 0);
currentPurge = purge;
}
}
private void setCurrentProperty(PrintStream out, LaserProperty p)
{
if (p instanceof LaosCutterProperty)
{
LaosCutterProperty prop = (LaosCutterProperty) p;
setFocus(out, prop.getFocus());
setVentilation(out, prop.getVentilation());
setPurge(out, prop.getPurge());
setSpeed(out, prop.getSpeed());
setPower(out, prop.getPower());
setFrequency(out, prop.getFrequency());
throw new RuntimeException("The Laos driver only accepts LaosCutter properties (was "+p.getClass().toString()+")");
}
private void line(PrintStream out, float x, float y, int resolution)
out.printf("1 %d %d\n", px2steps(isFlipXaxis() ? Util.mm2px(bedWidth, resolution) - x : x, resolution), px2steps(isFlipYaxis() ? Util.mm2px(bedHeight, resolution) - y : y, resolution));
}
private byte[] generatePseudoRaster3dGCode(Raster3dPart rp, int resolution) throws UnsupportedEncodingException
{
ByteArrayOutputStream result = new ByteArrayOutputStream();
PrintStream out = new PrintStream(result, true, "US-ASCII");
boolean dirRight = true;
Point rasterStart = rp.getRasterStart();
this.setCurrentProperty(out, rp.getLaserProperty());
float maxPower = this.currentPower;
for (int line = 0; line < rp.getRasterHeight(); line++)
Point lineStart = rasterStart.clone();
lineStart.y += line;
List<Byte> bytes = rp.getRasterLine(line);
//remove heading zeroes
while (bytes.size() > 0 && bytes.get(0) == 0)
bytes.remove(0);
lineStart.x += 1;
}
//remove trailing zeroes
while (bytes.size() > 0 && bytes.get(bytes.size() - 1) == 0)
{
bytes.remove(bytes.size() - 1);
}
if (bytes.size() > 0)
{
if (dirRight)
//move to the first nonempyt point of the line
move(out, lineStart.x, lineStart.y, resolution);
byte old = bytes.get(0);
for (int pix = 0; pix < bytes.size(); pix++)
if (bytes.get(pix) != old)
move(out, lineStart.x + pix, lineStart.y, resolution);
else
{
setPower(out, maxPower * (0xFF & old) / 255);
line(out, lineStart.x + pix - 1, lineStart.y, resolution);
move(out, lineStart.x + pix, lineStart.y, resolution);
}
old = bytes.get(pix);
//last point is also not "white"
setPower(out, maxPower * (0xFF & bytes.get(bytes.size() - 1)) / 255);
line(out, lineStart.x + bytes.size() - 1, lineStart.y, resolution);
}
else
{
//move to the last nonempty point of the line
move(out, lineStart.x + bytes.size() - 1, lineStart.y, resolution);
byte old = bytes.get(bytes.size() - 1);
for (int pix = bytes.size() - 1; pix >= 0; pix--)
if (bytes.get(pix) != old || pix == 0)
move(out, lineStart.x + pix, lineStart.y, resolution);
else
{
setPower(out, maxPower * (0xFF & old) / 255);
line(out, lineStart.x + pix + 1, lineStart.y, resolution);
move(out, lineStart.x + pix, lineStart.y, resolution);
}
old = bytes.get(pix);
//last point is also not "white"
setPower(out, maxPower * (0xFF & bytes.get(0)) / 255);
line(out, lineStart.x, lineStart.y, resolution);
Thomas Oster
committed
}
if (!this.isEngraveUnidirectional())
{
dirRight = !dirRight;
}
Thomas Oster
committed
/**
* This Method takes a raster-line represented by a list of bytes,
* where: byte0 ist the left-most byte, in one byte, the MSB is the
Thomas Oster
committed
* left-most bit, 0 representing laser off, 1 representing laser on.
* The Output List of longs, where each value is the unsigned dword
* of 4 bytes of the input each, where the first dword is the leftmost
* dword and the LSB is the leftmost bit. If outputLeftToRight is false,
* the first dword is the rightmost dword and the LSB of each dword is the
* the Output is padded with zeroes on the right side, if leftToRight is true,
* on the left-side otherwise
* rightmost bit
* @param line
* @param outputLeftToRight
* @return
*/
public List<Long> byteLineToDwords(List<Byte> line, boolean outputLeftToRight)
Thomas Oster
committed
{
List<Long> result = new ArrayList<Long>();
int s = line.size();
for (int i=0;i<s;i++)
{
line.set(i, (byte) (Integer.reverse(0xFF&line.get(i))>>>24));
}
Thomas Oster
committed
for(int i=0; i<s; i+=4)
{
result.add(
(((long) (i+3 < s ? 0xFF&line.get(i+3) : 0))<<24)
+ (((long) (i+2 < s ? 0xFF&line.get(i+2) : 0))<<16)
+ (((long) (i+1 < s ? 0xFF&line.get(i+1) : 0))<<8)
+ ((long) (0xFF&line.get(i)))
Thomas Oster
committed
);
}
if (!outputLeftToRight)
{
Collections.reverse(result);
for(int i=0;i<result.size();i++)
{
result.set(i, Long.reverse(result.get(i)) >>> 32);
Thomas Oster
committed
}
}
return result;
}
private byte[] generateLaosRasterCode(RasterPart rp, int resolution) throws UnsupportedEncodingException, IOException
{
ByteArrayOutputStream result = new ByteArrayOutputStream();
PrintStream out = new PrintStream(result, true, "US-ASCII");
Thomas Oster
committed
boolean dirRight = true;
Point rasterStart = rp.getRasterStart();
this.setCurrentProperty(out, rp.getLaserProperty());
for (int line = 0; line < rp.getRasterHeight(); line++)
Point lineStart = rasterStart.clone();
lineStart.y += line;
List<Byte> bytes = rp.getRasterLine(line);
//remove heading zeroes
while (bytes.size() > 0 && bytes.get(0) == 0)
{
lineStart.x += 8;
bytes.remove(0);
}
//remove trailing zeroes
while (bytes.size() > 0 && bytes.get(bytes.size()-1) == 0)
{
bytes.remove(bytes.size()-1);
}
if (bytes.size() > 0)
Thomas Oster
committed
{
//add space on the left side
int space = (int) Util.mm2px(this.getAddSpacePerRasterLine(), resolution);
while (space > 0 && lineStart.x >= 8)
Thomas Oster
committed
{
bytes.add(0, (byte) 0);
space -= 8;
lineStart.x -=8;
Thomas Oster
committed
}
//add space on the right side
space = (int) Util.mm2px(this.getAddSpacePerRasterLine(), resolution);
int max = (int) Util.mm2px(this.getBedWidth(), resolution);
while (space > 0 && lineStart.x+(8*bytes.size()) < max-8)
Thomas Oster
committed
{
bytes.add((byte) 0);
space -= 8;
}
if (dirRight)
Thomas Oster
committed
{
//move to the first point of the line
move(out, lineStart.x, lineStart.y, resolution);
List<Long> dwords = this.byteLineToDwords(bytes, true);
loadBitmapLine(out, dwords);
line(out, lineStart.x + (dwords.size()*32), lineStart.y, resolution);
Thomas Oster
committed
}
Thomas Oster
committed
{
//move to the first point of the line
List<Long> dwords = this.byteLineToDwords(bytes, false);
move(out, lineStart.x+(dwords.size()*32), lineStart.y, resolution);
loadBitmapLine(out, dwords);
line(out, lineStart.x, lineStart.y, resolution);
Thomas Oster
committed
}
}
if (!this.isEngraveUnidirectional())
{
dirRight = !dirRight;
}
Thomas Oster
committed
private byte[] generateInitializationCode() throws UnsupportedEncodingException
{
ByteArrayOutputStream result = new ByteArrayOutputStream();
PrintStream out = new PrintStream(result, true, "US-ASCII");
return result.toByteArray();
}
private byte[] generateShutdownCode() throws UnsupportedEncodingException
{
ByteArrayOutputStream result = new ByteArrayOutputStream();
PrintStream out = new PrintStream(result, true, "US-ASCII");
this.setFocus(out, 0f);
this.setVentilation(out, false);
this.setPurge(out, false);
protected void writeJobCode(LaserJob job, OutputStream out, ProgressListener pl) throws UnsupportedEncodingException, IOException
{
out.write(this.generateInitializationCode());
pl.progressChanged(this, 20);
Thomas Oster
committed
int i = 0;
int max = job.getParts().size();
for (JobPart p : job.getParts())
Thomas Oster
committed
if (p instanceof Raster3dPart)
{
out.write(this.generatePseudoRaster3dGCode((Raster3dPart) p, job.getResolution()));
}
else if (p instanceof RasterPart)
{
out.write(this.generateLaosRasterCode((RasterPart) p, job.getResolution()));
}
else if (p instanceof VectorPart)
{
out.write(this.generateVectorGCode((VectorPart) p, job.getResolution()));
}
i++;
pl.progressChanged(this, 20 + (int) (i*(double) 60/max));
}
out.write(this.generateShutdownCode());
out.close();
}
@Override
public void sendJob(LaserJob job, ProgressListener pl) throws IllegalJobException, Exception
{
pl.progressChanged(this, 0);
BufferedOutputStream out;
ByteArrayOutputStream buffer = null;
pl.taskChanged(this, "checking job");
checkJob(job);
if (!useTftp)
{
pl.taskChanged(this, "connecting");
Socket connection = new Socket();
connection.connect(new InetSocketAddress(hostname, port), 3000);
out = new BufferedOutputStream(connection.getOutputStream());
pl.taskChanged(this, "sending");
}
else
{
buffer = new ByteArrayOutputStream();
out = new BufferedOutputStream(buffer);
pl.taskChanged(this, "buffering");
}
this.writeJobCode(job, out, pl);
pl.taskChanged(this, "connecting");
TFTPClient tftp = new TFTPClient();
//open a local UDP socket
tftp.open();
pl.taskChanged(this, "sending");
ByteArrayInputStream bain = new ByteArrayInputStream(buffer.toByteArray());
Thomas Oster
committed
tftp.sendFile(job.getName().replace(" ", "") +".lgc", TFTP.BINARY_MODE, bain, this.getHostname(), this.getPort());
pl.progressChanged(this, 100);
}
private List<Integer> resolutions;
@Override
public List<Integer> getResolutions()
{
if (resolutions == null)
{
//TODO: Calculate possible resolutions
//according to mm/step
resolutions = Arrays.asList(new Integer[]
{
Thomas Oster
committed
100,
200,
300,
500,
600,
1000,
1200
Thomas Oster
committed
protected double bedWidth = 300;
/**
* Get the value of bedWidth
*
* @return the value of bedWidth
*/
public double getBedWidth()
{
return bedWidth;
}
/**
* Set the value of bedWidth
*
* @param bedWidth new value of bedWidth
*/
public void setBedWidth(double bedWidth)
{
this.bedWidth = bedWidth;
}
Thomas Oster
committed
protected double bedHeight = 210;
/**
* Get the value of bedHeight
*
* @return the value of bedHeight
*/
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 List<String> settingAttributes;
@Override
public List<String> getSettingAttributes()
{
if (settingAttributes == null)
{
settingAttributes = new LinkedList<String>();
settingAttributes.add(SETTING_HOSTNAME);
settingAttributes.add(SETTING_PORT);
Thomas Oster
committed
settingAttributes.add(SETTING_UNIDIR);
settingAttributes.add(SETTING_BEDWIDTH);
settingAttributes.add(SETTING_BEDHEIGHT);
Thomas Oster
committed
//settingAttributes.add(SETTING_FLIPX);
//settingAttributes.add(SETTING_FLIPY);
//settingAttributes.add(SETTING_MMPERSTEP);
settingAttributes.add(SETTING_TFTP);
settingAttributes.add(SETTING_RASTER_WHITESPACE);
}
return settingAttributes;
}
@Override
public String getSettingValue(String attribute)
{
if (SETTING_RASTER_WHITESPACE.equals(attribute))
{
return "" + this.getAddSpacePerRasterLine();
}
Thomas Oster
committed
else if (SETTING_UNIDIR.equals(attribute))
{
return this.isEngraveUnidirectional() ? "yes" : "no";
}
else if (SETTING_HOSTNAME.equals(attribute))
{
return this.getHostname();
}
else if (SETTING_FLIPX.equals(attribute))
{
return this.isFlipXaxis() ? "yes" : "no";
}
else if (SETTING_FLIPY.equals(attribute))
{
return this.isFlipYaxis() ? "yes" : "no";
}
else if (SETTING_PORT.equals(attribute))
{
return "" + this.getPort();
}
else if (SETTING_BEDWIDTH.equals(attribute))
{
return "" + this.getBedWidth();
}
else if (SETTING_BEDHEIGHT.equals(attribute))
{
return "" + this.getBedHeight();
}
else if (SETTING_MMPERSTEP.equals(attribute))
{
return "" + this.getMmPerStep();
}
else if (SETTING_TFTP.equals(attribute))
{
return this.isUseTftp() ? "yes" : "no";
}
return null;
}
@Override
public void setSettingValue(String attribute, String value)
{
if (SETTING_RASTER_WHITESPACE.equals(attribute))
{
this.setAddSpacePerRasterLine(Double.parseDouble(value));
}
Thomas Oster
committed
else if (SETTING_UNIDIR.endsWith(attribute))
{
this.setEngraveUnidirectional("yes".equals(value));
}
else if (SETTING_HOSTNAME.equals(attribute))
{
this.setHostname(value);
}
else if (SETTING_PORT.equals(attribute))
{
this.setPort(Integer.parseInt(value));
}
else if (SETTING_FLIPX.equals(attribute))
{
this.setFlipXaxis("yes".equals(value));
}
else if (SETTING_FLIPY.equals(attribute))
{
this.setFlipYaxis("yes".equals(value));
}
else if (SETTING_BEDWIDTH.equals(attribute))
{
this.setBedWidth(Double.parseDouble(value));
}
else if (SETTING_BEDHEIGHT.equals(attribute))
{
this.setBedHeight(Double.parseDouble(value));
}
else if (SETTING_MMPERSTEP.equals(attribute))
{
this.setMmPerStep(Double.parseDouble(value));
}
else if (SETTING_TFTP.contains(attribute))
{
this.setUseTftp("yes".equals(value));
}
}
@Override
public LaserCutter clone()
{
LaosCutter clone = new LaosCutter();
clone.hostname = hostname;
clone.port = port;
clone.bedHeight = bedHeight;
clone.bedWidth = bedWidth;
clone.flipXaxis = flipXaxis;
clone.flipYaxis = flipYaxis;
clone.addSpacePerRasterLine = addSpacePerRasterLine;
Thomas Oster
committed
clone.unidir = unidir;