mirror of
https://github.com/esp8266/Arduino.git
synced 2025-12-19 22:42:48 +03:00
148 lines
4.9 KiB
Java
148 lines
4.9 KiB
Java
/* -*- mode: java; c-basic-offset: 2; indent-tabs-mode: nil -*- */
|
|
|
|
/*
|
|
Part of the Processing project - http://processing.org
|
|
|
|
Copyright (c) 2004-06 Ben Fry and Casey Reas
|
|
Copyright (c) 2001-04 Massachusetts Institute of Technology
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program 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 General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software Foundation,
|
|
Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
package processing.app.debug;
|
|
|
|
import java.io.InputStream;
|
|
import java.io.InputStreamReader;
|
|
import java.io.Reader;
|
|
import java.net.SocketException;
|
|
|
|
/**
|
|
* Slurps up messages from compiler.
|
|
*/
|
|
public class MessageSiphon implements Runnable {
|
|
|
|
private final Reader streamReader;
|
|
private final MessageConsumer consumer;
|
|
|
|
private Thread thread;
|
|
private boolean canRun;
|
|
// Data is processed line-by-line if possible, but if this is non-zero
|
|
// then a partial line is also processed if no line end is received
|
|
// within this many milliseconds.
|
|
private int lineTimeout;
|
|
|
|
public MessageSiphon(InputStream stream, MessageConsumer consumer) {
|
|
this(stream, consumer, 0);
|
|
}
|
|
|
|
public MessageSiphon(InputStream stream, MessageConsumer consumer, int lineTimeout) {
|
|
this.streamReader = new InputStreamReader(stream);
|
|
this.consumer = consumer;
|
|
this.canRun = true;
|
|
this.lineTimeout = lineTimeout;
|
|
|
|
thread = new Thread(this);
|
|
// don't set priority too low, otherwise exceptions won't
|
|
// bubble up in time (i.e. compile errors have a weird delay)
|
|
//thread.setPriority(Thread.MIN_PRIORITY);
|
|
thread.setPriority(Thread.MAX_PRIORITY - 1);
|
|
thread.start();
|
|
}
|
|
|
|
|
|
public void run() {
|
|
try {
|
|
// process data until we hit EOF; this will happily block
|
|
// (effectively sleeping the thread) until new data comes in.
|
|
// when the program is finally done, null will come through.
|
|
//
|
|
StringBuilder currentLine = new StringBuilder();
|
|
long lineStartTime = 0;
|
|
while (canRun) {
|
|
// First, try to read as many characters as possible. Take care
|
|
// not to block when:
|
|
// 1. lineTimeout is nonzero, and
|
|
// 2. we have some characters buffered already
|
|
while (lineTimeout == 0 || currentLine.length() == 0 || streamReader.ready()) {
|
|
int c = streamReader.read();
|
|
if (c == -1)
|
|
return; // EOF
|
|
if (!canRun)
|
|
return;
|
|
|
|
// Keep track of the line start time
|
|
if (currentLine.length() == 0)
|
|
lineStartTime = System.nanoTime();
|
|
|
|
// Store the character line
|
|
currentLine.append((char)c);
|
|
|
|
if (c == '\n') {
|
|
// We read a full line, pass it on
|
|
consumer.message(currentLine.toString());
|
|
currentLine.setLength(0);
|
|
}
|
|
}
|
|
|
|
// No more characters available. Wait until lineTimeout
|
|
// milliseconds have passed since the start of the line and then
|
|
// try reading again. If the time has already passed, then just
|
|
// pass on the characters read so far.
|
|
long passed = (System.nanoTime() - lineStartTime) / 1000;
|
|
if (passed < this.lineTimeout) {
|
|
Thread.sleep(this.lineTimeout - passed);
|
|
continue;
|
|
}
|
|
|
|
consumer.message(currentLine.toString());
|
|
currentLine.setLength(0);
|
|
}
|
|
//EditorConsole.systemOut.println("messaging thread done");
|
|
} catch (NullPointerException npe) {
|
|
// Fairly common exception during shutdown
|
|
} catch (SocketException e) {
|
|
// socket has been close while we were wainting for data. nothing to see here, move along
|
|
} catch (Exception e) {
|
|
// On Linux and sometimes on Mac OS X, a "bad file descriptor"
|
|
// message comes up when closing an applet that's run externally.
|
|
// That message just gets supressed here..
|
|
String mess = e.getMessage();
|
|
if ((mess != null) &&
|
|
(mess.indexOf("Bad file descriptor") != -1)) {
|
|
//if (e.getMessage().indexOf("Bad file descriptor") == -1) {
|
|
//System.err.println("MessageSiphon err " + e);
|
|
//e.printStackTrace();
|
|
} else {
|
|
e.printStackTrace();
|
|
}
|
|
} finally {
|
|
thread = null;
|
|
}
|
|
}
|
|
|
|
// Wait until the MessageSiphon thread is complete.
|
|
public void join() throws java.lang.InterruptedException {
|
|
// Grab a temp copy in case another thread nulls the "thread"
|
|
// member variable
|
|
Thread t = thread;
|
|
if (t != null) t.join();
|
|
}
|
|
|
|
public void stop() {
|
|
this.canRun = false;
|
|
}
|
|
|
|
}
|