cliechti | e8f75f9 | 2002-02-14 01:33:33 +0000 | [diff] [blame] | 1 | #!jython |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 2 | # |
cliechti | 809b82e | 2008-06-22 22:45:04 +0000 | [diff] [blame] | 3 | # Python Serial Port Extension for Win32, Linux, BSD, Jython |
| 4 | # module for serial IO for Jython and JavaComm |
| 5 | # see __init__.py |
| 6 | # |
| 7 | # (C) 2002-2008 Chris Liechti <cliechti@gmx.net> |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 8 | # this is distributed under a free software license, see license.txt |
| 9 | |
cliechti | 39cfb7b | 2011-08-22 00:30:07 +0000 | [diff] [blame] | 10 | from serial.serialutil import * |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 11 | |
cliechti | 809b82e | 2008-06-22 22:45:04 +0000 | [diff] [blame] | 12 | def my_import(name): |
| 13 | mod = __import__(name) |
| 14 | components = name.split('.') |
| 15 | for comp in components[1:]: |
| 16 | mod = getattr(mod, comp) |
| 17 | return mod |
| 18 | |
cliechti | 4a567a0 | 2009-07-27 22:09:31 +0000 | [diff] [blame] | 19 | |
cliechti | 809b82e | 2008-06-22 22:45:04 +0000 | [diff] [blame] | 20 | def detect_java_comm(names): |
| 21 | """try given list of modules and return that imports""" |
| 22 | for name in names: |
| 23 | try: |
| 24 | mod = my_import(name) |
| 25 | mod.SerialPort |
| 26 | return mod |
| 27 | except (ImportError, AttributeError): |
| 28 | pass |
| 29 | raise ImportError("No Java Communications API implementation found") |
| 30 | |
cliechti | 4a567a0 | 2009-07-27 22:09:31 +0000 | [diff] [blame] | 31 | |
cliechti | 809b82e | 2008-06-22 22:45:04 +0000 | [diff] [blame] | 32 | # Java Communications API implementations |
| 33 | # http://mho.republika.pl/java/comm/ |
| 34 | |
| 35 | comm = detect_java_comm([ |
| 36 | 'javax.comm', # Sun/IBM |
| 37 | 'gnu.io', # RXTX |
| 38 | ]) |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 39 | |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 40 | |
| 41 | def device(portnumber): |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 42 | """Turn a port number into a device name""" |
cliechti | 809b82e | 2008-06-22 22:45:04 +0000 | [diff] [blame] | 43 | enum = comm.CommPortIdentifier.getPortIdentifiers() |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 44 | ports = [] |
| 45 | while enum.hasMoreElements(): |
| 46 | el = enum.nextElement() |
cliechti | 809b82e | 2008-06-22 22:45:04 +0000 | [diff] [blame] | 47 | if el.getPortType() == comm.CommPortIdentifier.PORT_SERIAL: |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 48 | ports.append(el) |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 49 | return ports[portnumber].getName() |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 50 | |
cliechti | f81362e | 2009-07-25 03:44:33 +0000 | [diff] [blame] | 51 | |
Chris Liechti | ef6b7b4 | 2015-08-06 22:19:26 +0200 | [diff] [blame^] | 52 | class Serial(SerialBase): |
cliechti | 7d44856 | 2014-08-03 21:57:45 +0000 | [diff] [blame] | 53 | """\ |
| 54 | Serial port class, implemented with Java Communications API and |
| 55 | thus usable with jython and the appropriate java extension. |
| 56 | """ |
cliechti | 58b481c | 2009-02-16 20:42:32 +0000 | [diff] [blame] | 57 | |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 58 | def open(self): |
cliechti | 7d44856 | 2014-08-03 21:57:45 +0000 | [diff] [blame] | 59 | """\ |
| 60 | Open port with current settings. This may throw a SerialException |
| 61 | if the port cannot be opened. |
| 62 | """ |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 63 | if self._port is None: |
| 64 | raise SerialException("Port must be configured before it can be used.") |
cliechti | 8f69e70 | 2011-03-19 00:22:32 +0000 | [diff] [blame] | 65 | if self._isOpen: |
| 66 | raise SerialException("Port is already open.") |
cliechti | 4a567a0 | 2009-07-27 22:09:31 +0000 | [diff] [blame] | 67 | if type(self._port) == type(''): # strings are taken directly |
cliechti | 809b82e | 2008-06-22 22:45:04 +0000 | [diff] [blame] | 68 | portId = comm.CommPortIdentifier.getPortIdentifier(self._port) |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 69 | else: |
cliechti | 4a567a0 | 2009-07-27 22:09:31 +0000 | [diff] [blame] | 70 | portId = comm.CommPortIdentifier.getPortIdentifier(device(self._port)) # numbers are transformed to a comport id obj |
cliechti | 4616bf1 | 2002-04-08 23:13:14 +0000 | [diff] [blame] | 71 | try: |
| 72 | self.sPort = portId.open("python serial module", 10) |
| 73 | except Exception, msg: |
| 74 | self.sPort = None |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 75 | raise SerialException("Could not open port: %s" % msg) |
| 76 | self._reconfigurePort() |
| 77 | self._instream = self.sPort.getInputStream() |
| 78 | self._outstream = self.sPort.getOutputStream() |
| 79 | self._isOpen = True |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 80 | |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 81 | def _reconfigurePort(self): |
cliechti | 4a567a0 | 2009-07-27 22:09:31 +0000 | [diff] [blame] | 82 | """Set communication parameters on opened port.""" |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 83 | if not self.sPort: |
| 84 | raise SerialException("Can only operate on a valid port handle") |
cliechti | 58b481c | 2009-02-16 20:42:32 +0000 | [diff] [blame] | 85 | |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 86 | self.sPort.enableReceiveTimeout(30) |
| 87 | if self._bytesize == FIVEBITS: |
cliechti | 809b82e | 2008-06-22 22:45:04 +0000 | [diff] [blame] | 88 | jdatabits = comm.SerialPort.DATABITS_5 |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 89 | elif self._bytesize == SIXBITS: |
cliechti | 809b82e | 2008-06-22 22:45:04 +0000 | [diff] [blame] | 90 | jdatabits = comm.SerialPort.DATABITS_6 |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 91 | elif self._bytesize == SEVENBITS: |
cliechti | 809b82e | 2008-06-22 22:45:04 +0000 | [diff] [blame] | 92 | jdatabits = comm.SerialPort.DATABITS_7 |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 93 | elif self._bytesize == EIGHTBITS: |
cliechti | 809b82e | 2008-06-22 22:45:04 +0000 | [diff] [blame] | 94 | jdatabits = comm.SerialPort.DATABITS_8 |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 95 | else: |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 96 | raise ValueError("unsupported bytesize: %r" % self._bytesize) |
cliechti | 58b481c | 2009-02-16 20:42:32 +0000 | [diff] [blame] | 97 | |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 98 | if self._stopbits == STOPBITS_ONE: |
cliechti | 809b82e | 2008-06-22 22:45:04 +0000 | [diff] [blame] | 99 | jstopbits = comm.SerialPort.STOPBITS_1 |
cliechti | c5aed01 | 2014-08-01 02:34:22 +0000 | [diff] [blame] | 100 | elif self._stopbits == STOPBITS_ONE_POINT_FIVE: |
| 101 | jstopbits = comm.SerialPort.STOPBITS_1_5 |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 102 | elif self._stopbits == STOPBITS_TWO: |
cliechti | 809b82e | 2008-06-22 22:45:04 +0000 | [diff] [blame] | 103 | jstopbits = comm.SerialPort.STOPBITS_2 |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 104 | else: |
| 105 | raise ValueError("unsupported number of stopbits: %r" % self._stopbits) |
| 106 | |
| 107 | if self._parity == PARITY_NONE: |
cliechti | 809b82e | 2008-06-22 22:45:04 +0000 | [diff] [blame] | 108 | jparity = comm.SerialPort.PARITY_NONE |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 109 | elif self._parity == PARITY_EVEN: |
cliechti | 809b82e | 2008-06-22 22:45:04 +0000 | [diff] [blame] | 110 | jparity = comm.SerialPort.PARITY_EVEN |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 111 | elif self._parity == PARITY_ODD: |
cliechti | 809b82e | 2008-06-22 22:45:04 +0000 | [diff] [blame] | 112 | jparity = comm.SerialPort.PARITY_ODD |
| 113 | elif self._parity == PARITY_MARK: |
| 114 | jparity = comm.SerialPort.PARITY_MARK |
| 115 | elif self._parity == PARITY_SPACE: |
| 116 | jparity = comm.SerialPort.PARITY_SPACE |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 117 | else: |
| 118 | raise ValueError("unsupported parity type: %r" % self._parity) |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 119 | |
| 120 | jflowin = jflowout = 0 |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 121 | if self._rtscts: |
cliechti | 809b82e | 2008-06-22 22:45:04 +0000 | [diff] [blame] | 122 | jflowin |= comm.SerialPort.FLOWCONTROL_RTSCTS_IN |
| 123 | jflowout |= comm.SerialPort.FLOWCONTROL_RTSCTS_OUT |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 124 | if self._xonxoff: |
cliechti | 809b82e | 2008-06-22 22:45:04 +0000 | [diff] [blame] | 125 | jflowin |= comm.SerialPort.FLOWCONTROL_XONXOFF_IN |
| 126 | jflowout |= comm.SerialPort.FLOWCONTROL_XONXOFF_OUT |
cliechti | 58b481c | 2009-02-16 20:42:32 +0000 | [diff] [blame] | 127 | |
cliechti | 6670f73 | 2009-07-25 23:49:26 +0000 | [diff] [blame] | 128 | self.sPort.setSerialPortParams(self._baudrate, jdatabits, jstopbits, jparity) |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 129 | self.sPort.setFlowControlMode(jflowin | jflowout) |
cliechti | 58b481c | 2009-02-16 20:42:32 +0000 | [diff] [blame] | 130 | |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 131 | if self._timeout >= 0: |
cliechti | c5aed01 | 2014-08-01 02:34:22 +0000 | [diff] [blame] | 132 | self.sPort.enableReceiveTimeout(int(self._timeout*1000)) |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 133 | else: |
| 134 | self.sPort.disableReceiveTimeout() |
| 135 | |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 136 | def close(self): |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 137 | """Close port""" |
| 138 | if self._isOpen: |
| 139 | if self.sPort: |
| 140 | self._instream.close() |
| 141 | self._outstream.close() |
| 142 | self.sPort.close() |
| 143 | self.sPort = None |
| 144 | self._isOpen = False |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 145 | |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 146 | def makeDeviceName(self, port): |
| 147 | return device(port) |
cliechti | 95c6221 | 2002-03-04 22:17:53 +0000 | [diff] [blame] | 148 | |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 149 | # - - - - - - - - - - - - - - - - - - - - - - - - |
cliechti | 95c6221 | 2002-03-04 22:17:53 +0000 | [diff] [blame] | 150 | |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 151 | def inWaiting(self): |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 152 | """Return the number of characters currently in the input buffer.""" |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 153 | if not self.sPort: raise portNotOpenError |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 154 | return self._instream.available() |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 155 | |
cliechti | 4a567a0 | 2009-07-27 22:09:31 +0000 | [diff] [blame] | 156 | def read(self, size=1): |
cliechti | 7d44856 | 2014-08-03 21:57:45 +0000 | [diff] [blame] | 157 | """\ |
| 158 | Read size bytes from the serial port. If a timeout is set it may |
| 159 | return less characters as requested. With no timeout it will block |
| 160 | until the requested number of bytes is read. |
| 161 | """ |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 162 | if not self.sPort: raise portNotOpenError |
cliechti | 4a567a0 | 2009-07-27 22:09:31 +0000 | [diff] [blame] | 163 | read = bytearray() |
cliechti | a9e4e95 | 2002-05-26 01:20:22 +0000 | [diff] [blame] | 164 | if size > 0: |
| 165 | while len(read) < size: |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 166 | x = self._instream.read() |
cliechti | a9e4e95 | 2002-05-26 01:20:22 +0000 | [diff] [blame] | 167 | if x == -1: |
| 168 | if self.timeout >= 0: |
| 169 | break |
| 170 | else: |
cliechti | 4a567a0 | 2009-07-27 22:09:31 +0000 | [diff] [blame] | 171 | read.append(x) |
| 172 | return bytes(read) |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 173 | |
cliechti | 4a567a0 | 2009-07-27 22:09:31 +0000 | [diff] [blame] | 174 | def write(self, data): |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 175 | """Output the given string over the serial port.""" |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 176 | if not self.sPort: raise portNotOpenError |
cliechti | ddd7813 | 2009-07-28 01:13:28 +0000 | [diff] [blame] | 177 | if not isinstance(data, (bytes, bytearray)): |
| 178 | raise TypeError('expected %s or bytearray, got %s' % (bytes, type(data))) |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 179 | self._outstream.write(data) |
cliechti | f81362e | 2009-07-25 03:44:33 +0000 | [diff] [blame] | 180 | return len(data) |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 181 | |
| 182 | def flushInput(self): |
| 183 | """Clear input buffer, discarding all that is in the buffer.""" |
| 184 | if not self.sPort: raise portNotOpenError |
| 185 | self._instream.skip(self._instream.available()) |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 186 | |
| 187 | def flushOutput(self): |
cliechti | 7d44856 | 2014-08-03 21:57:45 +0000 | [diff] [blame] | 188 | """\ |
| 189 | Clear output buffer, aborting the current output and |
| 190 | discarding all that is in the buffer. |
| 191 | """ |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 192 | if not self.sPort: raise portNotOpenError |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 193 | self._outstream.flush() |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 194 | |
cliechti | aaa0460 | 2006-02-05 23:02:46 +0000 | [diff] [blame] | 195 | def sendBreak(self, duration=0.25): |
cliechti | 997b63c | 2008-06-21 00:09:31 +0000 | [diff] [blame] | 196 | """Send break condition. Timed, returns to idle state after given duration.""" |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 197 | if not self.sPort: raise portNotOpenError |
cliechti | aaa0460 | 2006-02-05 23:02:46 +0000 | [diff] [blame] | 198 | self.sPort.sendBreak(duration*1000.0) |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 199 | |
cliechti | 997b63c | 2008-06-21 00:09:31 +0000 | [diff] [blame] | 200 | def setBreak(self, level=1): |
| 201 | """Set break: Controls TXD. When active, to transmitting is possible.""" |
| 202 | if self.fd is None: raise portNotOpenError |
| 203 | raise SerialException("The setBreak function is not implemented in java.") |
| 204 | |
cliechti | 4569bac | 2007-11-08 21:57:19 +0000 | [diff] [blame] | 205 | def setRTS(self, level=1): |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 206 | """Set terminal status line: Request To Send""" |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 207 | if not self.sPort: raise portNotOpenError |
cliechti | 4569bac | 2007-11-08 21:57:19 +0000 | [diff] [blame] | 208 | self.sPort.setRTS(level) |
cliechti | f81362e | 2009-07-25 03:44:33 +0000 | [diff] [blame] | 209 | |
cliechti | 4569bac | 2007-11-08 21:57:19 +0000 | [diff] [blame] | 210 | def setDTR(self, level=1): |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 211 | """Set terminal status line: Data Terminal Ready""" |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 212 | if not self.sPort: raise portNotOpenError |
cliechti | 4569bac | 2007-11-08 21:57:19 +0000 | [diff] [blame] | 213 | self.sPort.setDTR(level) |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 214 | |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 215 | def getCTS(self): |
| 216 | """Read terminal status line: Clear To Send""" |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 217 | if not self.sPort: raise portNotOpenError |
cliechti | d6bf52c | 2003-10-01 02:28:12 +0000 | [diff] [blame] | 218 | self.sPort.isCTS() |
| 219 | |
| 220 | def getDSR(self): |
| 221 | """Read terminal status line: Data Set Ready""" |
| 222 | if not self.sPort: raise portNotOpenError |
| 223 | self.sPort.isDSR() |
| 224 | |
| 225 | def getRI(self): |
| 226 | """Read terminal status line: Ring Indicator""" |
| 227 | if not self.sPort: raise portNotOpenError |
| 228 | self.sPort.isRI() |
| 229 | |
| 230 | def getCD(self): |
| 231 | """Read terminal status line: Carrier Detect""" |
| 232 | if not self.sPort: raise portNotOpenError |
| 233 | self.sPort.isCD() |
| 234 | |
| 235 | |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 236 | if __name__ == '__main__': |
| 237 | s = Serial(0, |
cliechti | 4a567a0 | 2009-07-27 22:09:31 +0000 | [diff] [blame] | 238 | baudrate=19200, # baudrate |
| 239 | bytesize=EIGHTBITS, # number of databits |
| 240 | parity=PARITY_EVEN, # enable parity checking |
| 241 | stopbits=STOPBITS_ONE, # number of stopbits |
| 242 | timeout=3, # set a timeout value, None for waiting forever |
| 243 | xonxoff=0, # enable software flow control |
| 244 | rtscts=0, # enable RTS/CTS flow control |
| 245 | ) |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 246 | s.setRTS(1) |
| 247 | s.setDTR(1) |
| 248 | s.flushInput() |
| 249 | s.flushOutput() |
| 250 | s.write('hello') |
cliechti | 7aaead3 | 2009-07-23 14:02:41 +0000 | [diff] [blame] | 251 | sys.stdio.write('%r\n' % s.read(5)) |
| 252 | sys.stdio.write('%s\n' % s.inWaiting()) |
cliechti | 89b4af1 | 2002-02-12 23:24:41 +0000 | [diff] [blame] | 253 | del s |
| 254 | |
| 255 | |