1 from __future__ import absolute_import
4 import os, glob, sys, time, math, re, traceback, threading
7 from serial import Serial
9 from avr_isp import stk500v2
10 from avr_isp import ispBase
11 from avr_isp import intelHex
13 from util import profile
14 from util import version
25 key=_winreg.OpenKey(_winreg.HKEY_LOCAL_MACHINE,"HARDWARE\\DEVICEMAP\\SERIALCOMM")
28 baselist+=[_winreg.EnumValue(key,i)[1]]
32 baselist = baselist + glob.glob('/dev/ttyUSB*') + glob.glob('/dev/ttyACM*') + glob.glob("/dev/tty.usb*") + glob.glob("/dev/cu.*") + glob.glob("/dev/rfcomm*")
33 prev = profile.getPreference('serial_port_auto')
36 baselist.insert(0, prev)
37 if version.isDevVersion():
38 baselist.append('VIRTUAL')
42 ret = [250000, 230400, 115200, 57600, 38400, 19200, 9600]
43 if profile.getPreference('serial_baud_auto') != '':
44 prev = int(profile.getPreference('serial_baud_auto'))
50 class VirtualPrinter():
52 self.readList = ['start\n', 'Marlin: Virtual Marlin!\n', '\x80\n']
55 self.lastTempAt = time.time()
57 self.bedTargetTemp = 1.0
59 def write(self, data):
60 if self.readList == None:
62 #print "Send: %s" % (data.rstrip())
63 if 'M104' in data or 'M109' in data:
65 self.targetTemp = float(data[data.find('S')+1:])
68 if 'M140' in data or 'M190' in data:
70 self.bedTargetTemp = float(data[data.find('S')+1:])
74 self.readList.append("ok T:%.2f /%.2f B:%.2f /%.2f @:64\n" % (self.temp, self.targetTemp, self.bedTemp, self.bedTargetTemp))
75 elif len(data.strip()) > 0:
76 self.readList.append("ok\n")
79 if self.readList == None:
82 timeDiff = self.lastTempAt - time.time()
83 self.lastTempAt = time.time()
84 if abs(self.temp - self.targetTemp) > 1:
85 self.temp += math.copysign(timeDiff, self.targetTemp - self.temp)
86 if abs(self.bedTemp - self.bedTargetTemp) > 1:
87 self.bedTemp += math.copysign(timeDiff, self.bedTargetTemp - self.bedTemp)
88 while len(self.readList) < 1:
93 if self.readList == None:
96 #print "Recv: %s" % (self.readList[0].rstrip())
97 return self.readList.pop(0)
102 class MachineComPrintCallback(object):
103 def mcLog(self, message):
106 def mcTempUpdate(self, temp, bedTemp):
109 def mcStateChange(self, state):
112 def mcMessage(self, message):
115 def mcProgress(self, lineNr):
118 def mcZChange(self, newZ):
121 class MachineCom(object):
123 STATE_DETECT_BAUDRATE = 1
125 STATE_OPERATIONAL = 3
130 STATE_CLOSED_WITH_ERROR = 8
132 def __init__(self, port = None, baudrate = None, callbackObject = None):
134 port = profile.getPreference('serial_port')
136 if profile.getPreference('serial_baud') == 'AUTO':
139 baudrate = int(profile.getPreference('serial_baud'))
140 if callbackObject == None:
141 callbackObject = MachineComPrintCallback()
143 self._callback = callbackObject
144 self._state = self.STATE_NONE
146 self._baudrateDetectList = baudrateList()
147 self._baudrateDetectRetry = 0
150 self._gcodeList = None
152 self._commandQueue = queue.Queue()
153 self._logQueue = queue.Queue(256)
154 self._feedRateModifier = {}
158 programmer = stk500v2.Stk500v2()
159 self._log("Serial port list: %s" % (str(serialList())))
160 for p in serialList():
162 self._log("Connecting to: %s" % (p))
163 programmer.connect(p)
164 self._serial = programmer.leaveISP()
165 profile.putPreference('serial_port_auto', p)
167 except ispBase.IspError as (e):
168 self._log("Error while connecting to %s: %s" % (p, str(e)))
171 self._log("Unexpected error while connecting to serial port: %s %s" % (p, getExceptionString()))
173 elif port == 'VIRTUAL':
174 self._serial = VirtualPrinter()
177 self._log("Connecting to: %s" % (port))
179 self._serial = Serial(port, 115200, timeout=0.1)
181 self._serial = Serial(port, baudrate, timeout=2)
183 self._log("Unexpected error while connecting to serial port: %s %s" % (port, getExceptionString()))
184 if self._serial == None:
185 self._log("Failed to open serial port (%s)" % (port))
186 self._errorValue = 'Failed to autodetect serial port.'
187 self._changeState(self.STATE_ERROR)
189 self._log("Connected to: %s, starting monitor" % (self._serial))
191 self._changeState(self.STATE_DETECT_BAUDRATE)
193 self._changeState(self.STATE_CONNECTING)
194 self.thread = threading.Thread(target=self._monitor)
195 self.thread.daemon = True
198 def _changeState(self, newState):
199 if self._state == newState:
201 oldState = self.getStateString()
202 self._state = newState
203 self._log('Changing monitoring state from \'%s\' to \'%s\'' % (oldState, self.getStateString()))
204 self._callback.mcStateChange(newState)
209 def getStateString(self):
210 if self._state == self.STATE_NONE:
212 if self._state == self.STATE_DETECT_BAUDRATE:
213 return "Detect baudrate"
214 if self._state == self.STATE_CONNECTING:
216 if self._state == self.STATE_OPERATIONAL:
218 if self._state == self.STATE_PRINTING:
220 if self._state == self.STATE_PAUSED:
222 if self._state == self.STATE_CLOSED:
224 if self._state == self.STATE_ERROR:
225 return "Error: %s" % (self._errorValue)
226 if self._state == self.STATE_CLOSED_WITH_ERROR:
227 return "Error: %s" % (self._errorValue)
228 return "?%d?" % (self._state)
230 def isClosedOrError(self):
231 return self._state == self.STATE_ERROR or self._state == self.STATE_CLOSED_WITH_ERROR or self._state == self.STATE_CLOSED
234 return self._state == self.STATE_ERROR or self._state == self.STATE_CLOSED_WITH_ERROR
236 def isOperational(self):
237 return self._state == self.STATE_OPERATIONAL or self._state == self.STATE_PRINTING or self._state == self.STATE_PAUSED
239 def isPrinting(self):
240 return self._state == self.STATE_PRINTING
242 def getPrintPos(self):
243 return self._gcodePos
245 def getPrintTime(self):
246 return time.time() - self._printStartTime
249 return self._state == self.STATE_PAUSED
254 def getBedTemp(self):
259 while not self._logQueue.empty():
260 ret.append(self._logQueue.get())
262 self._logQueue.put(line, False)
266 timeout = time.time() + 5
267 tempRequestTimeout = timeout
269 line = self._readline()
273 #No matter the state, if we see an error, goto the error state and store the error for reference.
274 if line.startswith('Error: '):
275 #Oh YEAH, consistency.
276 # Marlin reports an MIN/MAX temp error as "Error: x\n: Extruder switched off. MAXTEMP triggered !\n"
277 # But a bed temp error is reported as "Error: Temperature heated bed switched off. MAXTEMP triggered !!"
278 # So we can have an extra newline in the most common case. Awesome work people.
279 if re.match('Error: [0-9]\n', line):
280 line = line.rstrip() + self._readline()
281 self._errorValue = line
282 self._changeState(self.STATE_ERROR)
284 self._temp = float(re.search("[0-9\.]*", line.split(' T:')[1]).group(0))
286 self._bedTemp = float(re.search("[0-9\.]*", line.split(' B:')[1]).group(0))
287 self._callback.mcTempUpdate(self._temp, self._bedTemp)
288 elif line.strip() != 'ok':
289 self._callback.mcMessage(line)
291 if self._state == self.STATE_DETECT_BAUDRATE:
292 if line == '' or time.time() > timeout:
293 if len(self._baudrateDetectList) < 1:
294 self._log("No more baudrates to test, and no suitable baudrate found.")
296 elif self._baudrateDetectRetry > 0:
297 self._baudrateDetectRetry -= 1
298 self._serial.write('\n')
299 self._sendCommand("M105")
301 baudrate = self._baudrateDetectList.pop(0)
303 self._serial.baudrate = baudrate
304 self._serial.timeout = 0.5
305 self._log("Trying baudrate: %d" % (baudrate))
306 self._baudrateDetectRetry = 5
307 timeout = time.time() + 5
308 self._serial.write('\n')
309 self._sendCommand("M105")
311 self._log("Unexpected error while setting baudrate: %d %s" % (baudrate, getExceptionString()))
313 self._serial.timeout = 2
314 profile.putPreference('serial_baud_auto', self._serial.baudrate)
315 self._changeState(self.STATE_OPERATIONAL)
316 elif self._state == self.STATE_CONNECTING:
318 self._sendCommand("M105")
320 self._changeState(self.STATE_OPERATIONAL)
321 if time.time() > timeout:
323 elif self._state == self.STATE_OPERATIONAL:
324 #Request the temperature on comm timeout (every 2 seconds) when we are not printing.
326 self._sendCommand("M105")
327 tempRequestTimeout = time.time() + 5
328 elif self._state == self.STATE_PRINTING:
329 if line == '' and time.time() > timeout:
330 self._log("Communication timeout during printing, forcing a line")
332 #Even when printing request the temperture every 5 seconds.
333 if time.time() > tempRequestTimeout:
334 self._commandQueue.put("M105")
335 tempRequestTimeout = time.time() + 5
337 timeout = time.time() + 5
338 if not self._commandQueue.empty():
339 self._sendCommand(self._commandQueue.get())
342 elif "resend" in line.lower() or "rs" in line:
344 self._gcodePos = int(line.replace("N:"," ").replace("N"," ").replace(":"," ").split()[-1])
347 self._gcodePos = int(line.split()[1])
348 self._log("Connection closed, closing down monitor")
350 def _log(self, message):
351 self._callback.mcLog(message)
353 self._logQueue.put(message, False)
355 #If the log queue is full, remove the first message and append the new message again
357 self._logQueue.put(message, False)
360 if self._serial == None:
363 ret = self._serial.readline()
365 self._log("Unexpected error while reading serial port: %s" % (getExceptionString()))
366 self._errorValue = getExceptionString()
370 #self._log("Recv: TIMEOUT")
372 self._log("Recv: %s" % (unicode(ret, 'ascii', 'replace').encode('ascii', 'replace').rstrip()))
375 def close(self, isError = False):
376 if self._serial != None:
379 self._changeState(self.STATE_CLOSED_WITH_ERROR)
381 self._changeState(self.STATE_CLOSED)
387 def _sendCommand(self, cmd):
388 if self._serial == None:
390 self._log('Send: %s' % (cmd))
392 #TODO: This can throw a write timeout exception, but we do not want timeout on writes. Find a fix for this.
393 # Oddly enough, the write timeout is not even set and thus we should not get a write timeout.
394 self._serial.write(cmd + '\n')
396 self._log("Unexpected error while writing serial port: %s" % (getExceptionString()))
397 self._errorValue = getExceptionString()
401 if self._gcodePos >= len(self._gcodeList):
402 self._changeState(self.STATE_OPERATIONAL)
404 line = self._gcodeList[self._gcodePos]
405 if type(line) is tuple:
406 self._printSection = line[1]
409 if line == 'M0' or line == 'M1':
411 line = 'M105' #Don't send the M0 or M1 to the machine, as M0 and M1 are handled as an LCD menu pause.
412 if self._printSection in self._feedRateModifier:
413 line = re.sub('F([0-9]*)', lambda m: 'F' + str(int(int(m.group(1)) * self._feedRateModifier[self._printSection])), line)
414 if ('G0' in line or 'G1' in line) and 'Z' in line:
415 z = float(re.search('Z([0-9\.]*)', line).group(1))
416 if self._currentZ != z:
418 self._callback.mcZChange(z)
420 self._log("Unexpected error: %s" % (getExceptionString()))
421 checksum = reduce(lambda x,y:x^y, map(ord, "N%d%s" % (self._gcodePos, line)))
422 self._sendCommand("N%d%s*%d" % (self._gcodePos, line, checksum))
424 self._callback.mcProgress(self._gcodePos)
426 def sendCommand(self, cmd):
427 cmd = cmd.encode('ascii', 'replace')
428 if self.isPrinting():
429 self._commandQueue.put(cmd)
430 elif self.isOperational():
431 self._sendCommand(cmd)
433 def printGCode(self, gcodeList):
434 if not self.isOperational() or self.isPrinting():
436 self._gcodeList = gcodeList
438 self._printSection = 'CUSTOM'
439 self._changeState(self.STATE_PRINTING)
440 self._printStartTime = time.time()
441 for i in xrange(0, 6):
444 def cancelPrint(self):
445 if self.isOperational():
446 self._changeState(self.STATE_OPERATIONAL)
448 def setPause(self, pause):
449 if not pause and self.isPaused():
450 self._changeState(self.STATE_PRINTING)
451 for i in xrange(0, 6):
453 if pause and self.isPrinting():
454 self._changeState(self.STATE_PAUSED)
456 def setFeedrateModifier(self, type, value):
457 self._feedRateModifier[type] = value
459 def getExceptionString():
460 locationInfo = traceback.extract_tb(sys.exc_info()[2])[0]
461 return "%s: '%s' @ %s:%s:%d" % (str(sys.exc_info()[0].__name__), str(sys.exc_info()[1]), os.path.basename(locationInfo[0]), locationInfo[2], locationInfo[1])