| 
									
										
										
										
											2017-06-27 20:23:30 -04:00
										 |  |  | #!/usr/bin/env python2 | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | # Main code for host side printer firmware | 
					
						
							|  |  |  | # | 
					
						
							| 
									
										
										
										
											2017-08-21 11:25:26 -04:00
										 |  |  | # Copyright (C) 2016,2017  Kevin O'Connor <kevin@koconnor.net> | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | # | 
					
						
							|  |  |  | # This file may be distributed under the terms of the GNU GPLv3 license. | 
					
						
							|  |  |  | import sys, optparse, ConfigParser, logging, time, threading | 
					
						
							| 
									
										
										
										
											2017-08-21 11:25:26 -04:00
										 |  |  | import util, reactor, queuelogger, msgproto, gcode | 
					
						
							| 
									
										
										
										
											2017-09-06 14:51:47 -04:00
										 |  |  | import pins, mcu, chipmisc, toolhead, extruder, heater, fan | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-11 22:21:55 -05:00
										 |  |  | message_ready = "Printer is ready" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-30 14:30:45 -05:00
										 |  |  | message_startup = """
 | 
					
						
							|  |  |  | The klippy host software is attempting to connect.  Please | 
					
						
							|  |  |  | retry in a few moments. | 
					
						
							|  |  |  | Printer is not ready | 
					
						
							|  |  |  | """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | message_restart = """
 | 
					
						
							| 
									
										
										
										
											2016-11-30 23:47:40 -05:00
										 |  |  | Once the underlying issue is corrected, use the "RESTART" | 
					
						
							|  |  |  | command to reload the config and restart the host software. | 
					
						
							| 
									
										
										
										
											2016-11-30 14:30:45 -05:00
										 |  |  | Printer is halted | 
					
						
							|  |  |  | """
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-09 23:50:13 -05:00
										 |  |  | message_protocol_error = """
 | 
					
						
							|  |  |  | This type of error is frequently caused by running an older | 
					
						
							|  |  |  | version of the firmware on the micro-controller (fix by | 
					
						
							|  |  |  | recompiling and flashing the firmware). | 
					
						
							|  |  |  | Once the underlying issue is corrected, use the "RESTART" | 
					
						
							|  |  |  | command to reload the config and restart the host software. | 
					
						
							|  |  |  | Protocol error connecting to printer | 
					
						
							|  |  |  | """
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-30 14:30:45 -05:00
										 |  |  | message_mcu_connect_error = """
 | 
					
						
							| 
									
										
										
										
											2017-03-08 22:26:10 -05:00
										 |  |  | Once the underlying issue is corrected, use the | 
					
						
							|  |  |  | "FIRMWARE_RESTART" command to reset the firmware, reload the | 
					
						
							|  |  |  | config, and restart the host software. | 
					
						
							| 
									
										
										
										
											2016-11-30 14:30:45 -05:00
										 |  |  | Error configuring printer | 
					
						
							|  |  |  | """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | message_shutdown = """
 | 
					
						
							| 
									
										
										
										
											2017-04-13 14:53:41 -04:00
										 |  |  | Once the underlying issue is corrected, use the | 
					
						
							|  |  |  | "FIRMWARE_RESTART" command to reset the firmware, reload the | 
					
						
							|  |  |  | config, and restart the host software. | 
					
						
							| 
									
										
										
										
											2016-11-30 14:30:45 -05:00
										 |  |  | Printer is shutdown | 
					
						
							|  |  |  | """
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | class ConfigWrapper: | 
					
						
							| 
									
										
										
										
											2016-11-30 15:05:26 -05:00
										 |  |  |     error = ConfigParser.Error | 
					
						
							| 
									
										
										
										
											2016-11-30 15:39:36 -05:00
										 |  |  |     class sentinel: | 
					
						
							|  |  |  |         pass | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     def __init__(self, printer, section): | 
					
						
							|  |  |  |         self.printer = printer | 
					
						
							|  |  |  |         self.section = section | 
					
						
							| 
									
										
										
										
											2017-04-11 11:37:09 -04:00
										 |  |  |     def get_wrapper(self, parser, option, default | 
					
						
							|  |  |  |                     , minval=None, maxval=None, above=None, below=None): | 
					
						
							| 
									
										
										
										
											2016-11-30 15:39:36 -05:00
										 |  |  |         if (default is not self.sentinel | 
					
						
							|  |  |  |             and not self.printer.fileconfig.has_option(self.section, option)): | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |             return default | 
					
						
							| 
									
										
										
										
											2016-11-30 16:07:17 -05:00
										 |  |  |         self.printer.all_config_options[ | 
					
						
							|  |  |  |             (self.section.lower(), option.lower())] = 1 | 
					
						
							| 
									
										
										
										
											2016-11-30 15:39:36 -05:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2017-04-11 11:37:09 -04:00
										 |  |  |             v = parser(self.section, option) | 
					
						
							| 
									
										
										
										
											2017-06-09 23:32:49 -04:00
										 |  |  |         except self.error as e: | 
					
						
							| 
									
										
										
										
											2016-11-30 15:39:36 -05:00
										 |  |  |             raise | 
					
						
							|  |  |  |         except: | 
					
						
							|  |  |  |             raise self.error("Unable to parse option '%s' in section '%s'" % ( | 
					
						
							|  |  |  |                 option, self.section)) | 
					
						
							| 
									
										
										
										
											2017-04-11 11:37:09 -04:00
										 |  |  |         if minval is not None and v < minval: | 
					
						
							|  |  |  |             raise self.error( | 
					
						
							|  |  |  |                 "Option '%s' in section '%s' must have minimum of %s" % ( | 
					
						
							|  |  |  |                     option, self.section, minval)) | 
					
						
							|  |  |  |         if maxval is not None and v > maxval: | 
					
						
							|  |  |  |             raise self.error( | 
					
						
							|  |  |  |                 "Option '%s' in section '%s' must have maximum of %s" % ( | 
					
						
							|  |  |  |                     option, self.section, maxval)) | 
					
						
							|  |  |  |         if above is not None and v <= above: | 
					
						
							|  |  |  |             raise self.error( | 
					
						
							|  |  |  |                 "Option '%s' in section '%s' must be above %s" % ( | 
					
						
							|  |  |  |                     option, self.section, above)) | 
					
						
							|  |  |  |         if below is not None and v >= below: | 
					
						
							|  |  |  |             raise self.error( | 
					
						
							|  |  |  |                 "Option '%s' in section '%s' must be below %s" % ( | 
					
						
							|  |  |  |                     option, self.section, below)) | 
					
						
							|  |  |  |         return v | 
					
						
							| 
									
										
										
										
											2016-11-30 15:39:36 -05:00
										 |  |  |     def get(self, option, default=sentinel): | 
					
						
							|  |  |  |         return self.get_wrapper(self.printer.fileconfig.get, option, default) | 
					
						
							| 
									
										
										
										
											2017-04-11 11:37:09 -04:00
										 |  |  |     def getint(self, option, default=sentinel, minval=None, maxval=None): | 
					
						
							|  |  |  |         return self.get_wrapper( | 
					
						
							|  |  |  |             self.printer.fileconfig.getint, option, default, minval, maxval) | 
					
						
							|  |  |  |     def getfloat(self, option, default=sentinel | 
					
						
							|  |  |  |                  , minval=None, maxval=None, above=None, below=None): | 
					
						
							| 
									
										
										
										
											2016-11-30 15:39:36 -05:00
										 |  |  |         return self.get_wrapper( | 
					
						
							| 
									
										
										
										
											2017-04-11 11:37:09 -04:00
										 |  |  |             self.printer.fileconfig.getfloat, option, default | 
					
						
							|  |  |  |             , minval, maxval, above, below) | 
					
						
							| 
									
										
										
										
											2016-11-30 15:39:36 -05:00
										 |  |  |     def getboolean(self, option, default=sentinel): | 
					
						
							|  |  |  |         return self.get_wrapper( | 
					
						
							|  |  |  |             self.printer.fileconfig.getboolean, option, default) | 
					
						
							| 
									
										
										
										
											2016-11-30 15:50:28 -05:00
										 |  |  |     def getchoice(self, option, choices, default=sentinel): | 
					
						
							|  |  |  |         c = self.get(option, default) | 
					
						
							|  |  |  |         if c not in choices: | 
					
						
							|  |  |  |             raise self.error( | 
					
						
							|  |  |  |                 "Option '%s' in section '%s' is not a valid choice" % ( | 
					
						
							|  |  |  |                     option, self.section)) | 
					
						
							|  |  |  |         return choices[c] | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     def getsection(self, section): | 
					
						
							|  |  |  |         return ConfigWrapper(self.printer, section) | 
					
						
							| 
									
										
										
										
											2017-04-29 13:57:02 -04:00
										 |  |  |     def has_section(self, section): | 
					
						
							|  |  |  |         return self.printer.fileconfig.has_section(section) | 
					
						
							| 
									
										
										
										
											2017-07-04 12:24:11 -04:00
										 |  |  |     def get_prefix_sections(self, prefix): | 
					
						
							|  |  |  |         return [self.getsection(s) for s in self.printer.fileconfig.sections() | 
					
						
							|  |  |  |                 if s.startswith(prefix)] | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-24 10:17:12 -05:00
										 |  |  | class ConfigLogger(): | 
					
						
							| 
									
										
										
										
											2017-05-01 13:44:06 -04:00
										 |  |  |     def __init__(self, cfg, bglogger): | 
					
						
							|  |  |  |         self.lines = ["===== Config file ====="] | 
					
						
							| 
									
										
										
										
											2016-12-24 10:17:12 -05:00
										 |  |  |         cfg.write(self) | 
					
						
							| 
									
										
										
										
											2017-05-01 13:44:06 -04:00
										 |  |  |         self.lines.append("=======================") | 
					
						
							|  |  |  |         data = "\n".join(self.lines) | 
					
						
							|  |  |  |         logging.info(data) | 
					
						
							|  |  |  |         bglogger.set_rollover_info("config", data) | 
					
						
							| 
									
										
										
										
											2016-12-24 10:17:12 -05:00
										 |  |  |     def write(self, data): | 
					
						
							| 
									
										
										
										
											2017-05-01 13:44:06 -04:00
										 |  |  |         self.lines.append(data.strip()) | 
					
						
							| 
									
										
										
										
											2016-12-24 10:17:12 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | class Printer: | 
					
						
							| 
									
										
										
										
											2017-09-06 14:51:47 -04:00
										 |  |  |     config_error = ConfigParser.Error | 
					
						
							| 
									
										
										
										
											2017-08-21 17:19:43 -04:00
										 |  |  |     def __init__(self, input_fd, bglogger, start_args): | 
					
						
							| 
									
										
										
										
											2017-05-01 13:44:06 -04:00
										 |  |  |         self.bglogger = bglogger | 
					
						
							| 
									
										
										
										
											2017-08-21 17:19:43 -04:00
										 |  |  |         self.start_args = start_args | 
					
						
							| 
									
										
										
										
											2017-05-01 13:44:06 -04:00
										 |  |  |         if bglogger is not None: | 
					
						
							|  |  |  |             bglogger.set_rollover_info("config", None) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |         self.reactor = reactor.Reactor() | 
					
						
							| 
									
										
										
										
											2017-03-12 22:02:32 -04:00
										 |  |  |         self.objects = {} | 
					
						
							| 
									
										
										
										
											2017-08-21 17:19:43 -04:00
										 |  |  |         self.gcode = gcode.GCodeParser(self, input_fd) | 
					
						
							| 
									
										
										
										
											2017-08-26 00:13:36 -04:00
										 |  |  |         self.stats_timer = self.reactor.register_timer(self._stats) | 
					
						
							| 
									
										
										
										
											2016-11-28 13:14:56 -05:00
										 |  |  |         self.connect_timer = self.reactor.register_timer( | 
					
						
							| 
									
										
										
										
											2017-08-26 00:13:36 -04:00
										 |  |  |             self._connect, self.reactor.NOW) | 
					
						
							| 
									
										
										
										
											2016-11-30 16:07:17 -05:00
										 |  |  |         self.all_config_options = {} | 
					
						
							| 
									
										
										
										
											2016-11-30 19:05:25 -05:00
										 |  |  |         self.need_dump_debug = False | 
					
						
							| 
									
										
										
										
											2016-11-30 14:30:45 -05:00
										 |  |  |         self.state_message = message_startup | 
					
						
							| 
									
										
										
										
											2016-11-30 23:47:40 -05:00
										 |  |  |         self.run_result = None | 
					
						
							| 
									
										
										
										
											2016-11-22 12:14:25 -05:00
										 |  |  |         self.fileconfig = None | 
					
						
							| 
									
										
										
										
											2017-08-14 11:46:35 -04:00
										 |  |  |         self.mcus = [] | 
					
						
							| 
									
										
										
										
											2017-08-21 17:19:43 -04:00
										 |  |  |     def get_start_args(self): | 
					
						
							|  |  |  |         return self.start_args | 
					
						
							| 
									
										
										
										
											2017-08-26 00:13:36 -04:00
										 |  |  |     def _stats(self, eventtime, force_output=False): | 
					
						
							| 
									
										
										
										
											2016-11-30 19:05:25 -05:00
										 |  |  |         if self.need_dump_debug: | 
					
						
							|  |  |  |             # Call dump_debug here so it is executed in the main thread | 
					
						
							|  |  |  |             self.gcode.dump_debug() | 
					
						
							|  |  |  |             self.need_dump_debug = False | 
					
						
							| 
									
										
										
										
											2017-04-14 10:29:39 -04:00
										 |  |  |         toolhead = self.objects.get('toolhead') | 
					
						
							| 
									
										
										
										
											2017-08-14 11:46:35 -04:00
										 |  |  |         if toolhead is None: | 
					
						
							| 
									
										
										
										
											2017-06-15 09:37:57 -04:00
										 |  |  |             return eventtime + 1. | 
					
						
							| 
									
										
										
										
											2017-09-18 21:11:11 -04:00
										 |  |  |         is_active = toolhead.check_active(eventtime) | 
					
						
							| 
									
										
										
										
											2017-04-14 10:29:39 -04:00
										 |  |  |         if not is_active and not force_output: | 
					
						
							| 
									
										
										
										
											2017-06-15 09:37:57 -04:00
										 |  |  |             return eventtime + 1. | 
					
						
							| 
									
										
										
										
											2016-11-22 12:14:25 -05:00
										 |  |  |         out = [] | 
					
						
							|  |  |  |         out.append(self.gcode.stats(eventtime)) | 
					
						
							| 
									
										
										
										
											2017-09-18 21:11:11 -04:00
										 |  |  |         out.append(toolhead.stats(eventtime)) | 
					
						
							| 
									
										
										
										
											2017-08-14 11:46:35 -04:00
										 |  |  |         for m in self.mcus: | 
					
						
							|  |  |  |             out.append(m.stats(eventtime)) | 
					
						
							| 
									
										
										
										
											2017-01-10 12:29:52 -05:00
										 |  |  |         logging.info("Stats %.1f: %s" % (eventtime, ' '.join(out))) | 
					
						
							| 
									
										
										
										
											2016-11-22 12:14:25 -05:00
										 |  |  |         return eventtime + 1. | 
					
						
							| 
									
										
										
										
											2017-04-29 13:57:02 -04:00
										 |  |  |     def add_object(self, name, obj): | 
					
						
							|  |  |  |         self.objects[name] = obj | 
					
						
							| 
									
										
										
										
											2017-08-26 00:13:36 -04:00
										 |  |  |     def _load_config(self): | 
					
						
							| 
									
										
										
										
											2016-11-22 12:14:25 -05:00
										 |  |  |         self.fileconfig = ConfigParser.RawConfigParser() | 
					
						
							| 
									
										
										
										
											2017-08-21 17:19:43 -04:00
										 |  |  |         config_file = self.start_args['config_file'] | 
					
						
							|  |  |  |         res = self.fileconfig.read(config_file) | 
					
						
							| 
									
										
										
										
											2016-11-30 14:57:18 -05:00
										 |  |  |         if not res: | 
					
						
							| 
									
										
										
										
											2017-09-06 14:51:47 -04:00
										 |  |  |             raise self.config_error("Unable to open config file %s" % ( | 
					
						
							| 
									
										
										
										
											2017-08-21 17:19:43 -04:00
										 |  |  |                 config_file,)) | 
					
						
							| 
									
										
										
										
											2017-05-01 13:44:06 -04:00
										 |  |  |         if self.bglogger is not None: | 
					
						
							|  |  |  |             ConfigLogger(self.fileconfig, self.bglogger) | 
					
						
							| 
									
										
										
										
											2017-04-29 13:57:02 -04:00
										 |  |  |         # Create printer components | 
					
						
							|  |  |  |         config = ConfigWrapper(self, 'printer') | 
					
						
							| 
									
										
										
										
											2017-09-06 14:51:47 -04:00
										 |  |  |         for m in [pins, mcu, chipmisc, toolhead, extruder, heater, fan]: | 
					
						
							| 
									
										
										
										
											2017-04-29 13:57:02 -04:00
										 |  |  |             m.add_printer_objects(self, config) | 
					
						
							| 
									
										
										
										
											2017-08-14 11:46:35 -04:00
										 |  |  |         self.mcus = mcu.get_printer_mcus(self) | 
					
						
							| 
									
										
										
										
											2017-03-12 22:43:05 -04:00
										 |  |  |         # Validate that there are no undefined parameters in the config file | 
					
						
							| 
									
										
										
										
											2017-06-06 12:35:13 -04:00
										 |  |  |         valid_sections = { s: 1 for s, o in self.all_config_options } | 
					
						
							| 
									
										
										
										
											2016-11-30 16:07:17 -05:00
										 |  |  |         for section in self.fileconfig.sections(): | 
					
						
							|  |  |  |             section = section.lower() | 
					
						
							|  |  |  |             if section not in valid_sections: | 
					
						
							| 
									
										
										
										
											2017-09-06 14:51:47 -04:00
										 |  |  |                 raise self.config_error("Unknown config file section '%s'" % ( | 
					
						
							| 
									
										
										
										
											2016-11-30 16:07:17 -05:00
										 |  |  |                     section,)) | 
					
						
							|  |  |  |             for option in self.fileconfig.options(section): | 
					
						
							|  |  |  |                 option = option.lower() | 
					
						
							|  |  |  |                 if (section, option) not in self.all_config_options: | 
					
						
							| 
									
										
										
										
											2017-09-06 14:51:47 -04:00
										 |  |  |                     raise self.config_error( | 
					
						
							| 
									
										
										
										
											2016-11-30 16:07:17 -05:00
										 |  |  |                         "Unknown option '%s' in section '%s'" % ( | 
					
						
							|  |  |  |                             option, section)) | 
					
						
							| 
									
										
										
										
											2017-08-26 00:13:36 -04:00
										 |  |  |     def _connect(self, eventtime): | 
					
						
							| 
									
										
										
										
											2017-09-19 17:12:04 -04:00
										 |  |  |         self.reactor.unregister_timer(self.connect_timer) | 
					
						
							| 
									
										
										
										
											2016-11-30 14:30:45 -05:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2017-08-26 00:13:36 -04:00
										 |  |  |             self._load_config() | 
					
						
							| 
									
										
										
										
											2017-08-14 11:46:35 -04:00
										 |  |  |             for m in self.mcus: | 
					
						
							|  |  |  |                 m.connect() | 
					
						
							| 
									
										
										
										
											2016-11-30 14:30:45 -05:00
										 |  |  |             self.gcode.set_printer_ready(True) | 
					
						
							| 
									
										
										
										
											2017-02-11 22:21:55 -05:00
										 |  |  |             self.state_message = message_ready | 
					
						
							| 
									
										
										
										
											2017-09-19 17:12:04 -04:00
										 |  |  |             if self.start_args.get('debugoutput') is None: | 
					
						
							|  |  |  |                 self.reactor.update_timer(self.stats_timer, self.reactor.NOW) | 
					
						
							| 
									
										
										
										
											2017-09-06 14:51:47 -04:00
										 |  |  |         except (self.config_error, pins.error) as e: | 
					
						
							| 
									
										
										
										
											2016-11-30 14:57:18 -05:00
										 |  |  |             logging.exception("Config error") | 
					
						
							|  |  |  |             self.state_message = "%s%s" % (str(e), message_restart) | 
					
						
							| 
									
										
										
										
											2017-06-09 23:32:49 -04:00
										 |  |  |         except msgproto.error as e: | 
					
						
							| 
									
										
										
										
											2017-01-09 23:50:13 -05:00
										 |  |  |             logging.exception("Protocol error") | 
					
						
							|  |  |  |             self.state_message = "%s%s" % (str(e), message_protocol_error) | 
					
						
							| 
									
										
										
										
											2017-06-09 23:32:49 -04:00
										 |  |  |         except mcu.error as e: | 
					
						
							| 
									
										
										
										
											2016-11-30 14:30:45 -05:00
										 |  |  |             logging.exception("MCU error during connect") | 
					
						
							|  |  |  |             self.state_message = "%s%s" % (str(e), message_mcu_connect_error) | 
					
						
							|  |  |  |         except: | 
					
						
							|  |  |  |             logging.exception("Unhandled exception during connect") | 
					
						
							|  |  |  |             self.state_message = "Internal error during connect.%s" % ( | 
					
						
							| 
									
										
										
										
											2017-08-26 13:20:16 -04:00
										 |  |  |                 message_restart,) | 
					
						
							| 
									
										
										
										
											2016-11-28 13:14:56 -05:00
										 |  |  |         return self.reactor.NEVER | 
					
						
							|  |  |  |     def run(self): | 
					
						
							| 
									
										
										
										
											2017-02-06 13:31:34 -05:00
										 |  |  |         systime = time.time() | 
					
						
							|  |  |  |         monotime = self.reactor.monotonic() | 
					
						
							|  |  |  |         logging.info("Start printer at %s (%.1f %.1f)" % ( | 
					
						
							|  |  |  |             time.asctime(time.localtime(systime)), systime, monotime)) | 
					
						
							| 
									
										
										
										
											2017-08-26 13:20:16 -04:00
										 |  |  |         # Enter main reactor loop | 
					
						
							| 
									
										
										
										
											2016-11-30 14:30:45 -05:00
										 |  |  |         try: | 
					
						
							|  |  |  |             self.reactor.run() | 
					
						
							|  |  |  |         except: | 
					
						
							|  |  |  |             logging.exception("Unhandled exception during run") | 
					
						
							| 
									
										
										
										
											2017-08-26 13:20:16 -04:00
										 |  |  |             return "exit" | 
					
						
							|  |  |  |         # Check restart flags | 
					
						
							|  |  |  |         run_result = self.run_result | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             self._stats(self.reactor.monotonic(), force_output=True) | 
					
						
							| 
									
										
										
										
											2017-08-14 11:46:35 -04:00
										 |  |  |             for m in self.mcus: | 
					
						
							| 
									
										
										
										
											2017-08-26 13:20:16 -04:00
										 |  |  |                 if run_result == 'firmware_restart': | 
					
						
							| 
									
										
										
										
											2017-08-14 11:46:35 -04:00
										 |  |  |                     m.microcontroller_restart() | 
					
						
							|  |  |  |                 m.disconnect() | 
					
						
							| 
									
										
										
										
											2017-08-26 13:20:16 -04:00
										 |  |  |         except: | 
					
						
							|  |  |  |             logging.exception("Unhandled exception during post run") | 
					
						
							|  |  |  |         return run_result | 
					
						
							| 
									
										
										
										
											2016-11-30 14:30:45 -05:00
										 |  |  |     def get_state_message(self): | 
					
						
							|  |  |  |         return self.state_message | 
					
						
							|  |  |  |     def note_shutdown(self, msg): | 
					
						
							| 
									
										
										
										
											2017-02-11 22:21:55 -05:00
										 |  |  |         if self.state_message == message_ready: | 
					
						
							| 
									
										
										
										
											2016-11-30 19:05:25 -05:00
										 |  |  |             self.need_dump_debug = True | 
					
						
							| 
									
										
										
										
											2017-09-05 21:15:24 -04:00
										 |  |  |         self.state_message = "%s%s" % (msg, message_shutdown) | 
					
						
							| 
									
										
										
										
											2016-11-22 19:38:51 -05:00
										 |  |  |         self.gcode.set_printer_ready(False) | 
					
						
							| 
									
										
										
										
											2016-11-30 19:31:46 -05:00
										 |  |  |     def note_mcu_error(self, msg): | 
					
						
							|  |  |  |         self.state_message = "%s%s" % (msg, message_restart) | 
					
						
							|  |  |  |         self.gcode.set_printer_ready(False) | 
					
						
							| 
									
										
										
										
											2016-11-30 23:47:40 -05:00
										 |  |  |         self.gcode.motor_heater_off() | 
					
						
							| 
									
										
										
										
											2017-03-08 22:26:10 -05:00
										 |  |  |     def request_exit(self, result="exit"): | 
					
						
							|  |  |  |         self.run_result = result | 
					
						
							| 
									
										
										
										
											2016-11-30 23:47:40 -05:00
										 |  |  |         self.reactor.end() | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ###################################################################### | 
					
						
							|  |  |  | # Startup | 
					
						
							|  |  |  | ###################################################################### | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-14 11:46:35 -04:00
										 |  |  | def arg_dictionary(option, opt_str, value, parser): | 
					
						
							|  |  |  |     key, fname = "dictionary", value | 
					
						
							|  |  |  |     if '=' in value: | 
					
						
							|  |  |  |         mcu_name, fname = value.split('=', 1) | 
					
						
							|  |  |  |         key = "dictionary_" + mcu_name | 
					
						
							|  |  |  |     if parser.values.dictionary is None: | 
					
						
							|  |  |  |         parser.values.dictionary = {} | 
					
						
							|  |  |  |     parser.values.dictionary[key] = fname | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | def main(): | 
					
						
							|  |  |  |     usage = "%prog [options] <config file>" | 
					
						
							|  |  |  |     opts = optparse.OptionParser(usage) | 
					
						
							| 
									
										
										
										
											2017-08-21 17:19:43 -04:00
										 |  |  |     opts.add_option("-i", "--debuginput", dest="debuginput", | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |                     help="read commands from file instead of from tty port") | 
					
						
							| 
									
										
										
										
											2016-11-20 20:40:31 -05:00
										 |  |  |     opts.add_option("-I", "--input-tty", dest="inputtty", default='/tmp/printer', | 
					
						
							|  |  |  |                     help="input tty name (default is /tmp/printer)") | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     opts.add_option("-l", "--logfile", dest="logfile", | 
					
						
							|  |  |  |                     help="write log to file instead of stderr") | 
					
						
							|  |  |  |     opts.add_option("-v", action="store_true", dest="verbose", | 
					
						
							|  |  |  |                     help="enable debug messages") | 
					
						
							| 
									
										
										
										
											2017-08-21 17:19:43 -04:00
										 |  |  |     opts.add_option("-o", "--debugoutput", dest="debugoutput", | 
					
						
							|  |  |  |                     help="write output to file instead of to serial port") | 
					
						
							| 
									
										
										
										
											2017-08-14 11:46:35 -04:00
										 |  |  |     opts.add_option("-d", "--dictionary", dest="dictionary", type="string", | 
					
						
							|  |  |  |                     action="callback", callback=arg_dictionary, | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |                     help="file to read for mcu protocol dictionary") | 
					
						
							|  |  |  |     options, args = opts.parse_args() | 
					
						
							|  |  |  |     if len(args) != 1: | 
					
						
							|  |  |  |         opts.error("Incorrect number of arguments") | 
					
						
							| 
									
										
										
										
											2017-08-21 17:19:43 -04:00
										 |  |  |     start_args = {'config_file': args[0], 'start_reason': 'startup'} | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 17:19:43 -04:00
										 |  |  |     input_fd = bglogger = None | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     debuglevel = logging.INFO | 
					
						
							|  |  |  |     if options.verbose: | 
					
						
							|  |  |  |         debuglevel = logging.DEBUG | 
					
						
							| 
									
										
										
										
											2017-08-21 17:19:43 -04:00
										 |  |  |     if options.debuginput: | 
					
						
							|  |  |  |         start_args['debuginput'] = options.debuginput | 
					
						
							|  |  |  |         debuginput = open(options.debuginput, 'rb') | 
					
						
							| 
									
										
										
										
											2016-11-20 20:40:31 -05:00
										 |  |  |         input_fd = debuginput.fileno() | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         input_fd = util.create_pty(options.inputtty) | 
					
						
							| 
									
										
										
										
											2017-08-21 17:19:43 -04:00
										 |  |  |     if options.debugoutput: | 
					
						
							|  |  |  |         start_args['debugoutput'] = options.debugoutput | 
					
						
							| 
									
										
										
										
											2017-08-14 11:46:35 -04:00
										 |  |  |         start_args.update(options.dictionary) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     if options.logfile: | 
					
						
							| 
									
										
										
										
											2016-11-11 20:22:39 -05:00
										 |  |  |         bglogger = queuelogger.setup_bg_logging(options.logfile, debuglevel) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     else: | 
					
						
							|  |  |  |         logging.basicConfig(level=debuglevel) | 
					
						
							|  |  |  |     logging.info("Starting Klippy...") | 
					
						
							| 
									
										
										
										
											2017-08-21 17:19:43 -04:00
										 |  |  |     start_args['software_version'] = util.get_git_version() | 
					
						
							| 
									
										
										
										
											2017-05-01 13:44:06 -04:00
										 |  |  |     if bglogger is not None: | 
					
						
							|  |  |  |         lines = ["Args: %s" % (sys.argv,), | 
					
						
							| 
									
										
										
										
											2017-08-21 17:19:43 -04:00
										 |  |  |                  "Git version: %s" % (repr(start_args['software_version']),), | 
					
						
							| 
									
										
										
										
											2017-05-01 13:44:06 -04:00
										 |  |  |                  "CPU: %s" % (util.get_cpu_info(),), | 
					
						
							|  |  |  |                  "Python: %s" % (repr(sys.version),)] | 
					
						
							|  |  |  |         lines = "\n".join(lines) | 
					
						
							|  |  |  |         logging.info(lines) | 
					
						
							|  |  |  |         bglogger.set_rollover_info('versions', lines) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 17:19:43 -04:00
										 |  |  |     # Start Printer() class | 
					
						
							| 
									
										
										
										
											2016-11-30 23:47:40 -05:00
										 |  |  |     while 1: | 
					
						
							| 
									
										
										
										
											2017-08-21 17:19:43 -04:00
										 |  |  |         printer = Printer(input_fd, bglogger, start_args) | 
					
						
							| 
									
										
										
										
											2016-11-30 23:47:40 -05:00
										 |  |  |         res = printer.run() | 
					
						
							| 
									
										
										
										
											2017-08-26 13:20:16 -04:00
										 |  |  |         if res == 'exit': | 
					
						
							| 
									
										
										
										
											2017-08-21 17:19:43 -04:00
										 |  |  |             break | 
					
						
							|  |  |  |         time.sleep(1.) | 
					
						
							|  |  |  |         logging.info("Restarting printer") | 
					
						
							|  |  |  |         start_args['start_reason'] = res | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-11 20:22:39 -05:00
										 |  |  |     if bglogger is not None: | 
					
						
							|  |  |  |         bglogger.stop() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | if __name__ == '__main__': | 
					
						
							|  |  |  |     main() |