Skip to content

Commit 86829a7

Browse files
committed
Updates from review. Add lots of documentation, fix hostname on windows
Set a flag when cscore can't be loaded to make the operation perform() fail, instead of crashing to desktop Still need to do mac hostname resolution
1 parent 1691326 commit 86829a7

File tree

1 file changed

+108
-33
lines changed

1 file changed

+108
-33
lines changed

core/src/main/java/edu/wpi/grip/core/operations/composite/PublishVideoOperation.java

+108-33
Original file line numberDiff line numberDiff line change
@@ -17,10 +17,15 @@
1717
import edu.wpi.first.wpilibj.networktables.NetworkTable;
1818
import edu.wpi.first.wpilibj.tables.ITable;
1919

20+
import org.apache.commons.lang.SystemUtils;
2021
import org.bytedeco.javacpp.opencv_core;
2122
import org.opencv.core.Mat;
2223

24+
import java.io.BufferedReader;
25+
import java.io.IOException;
26+
import java.io.InputStreamReader;
2327
import java.lang.reflect.Field;
28+
import java.util.Arrays;
2429
import java.util.Deque;
2530
import java.util.LinkedList;
2631
import java.util.List;
@@ -29,9 +34,6 @@
2934
import java.util.stream.Collectors;
3035
import java.util.stream.Stream;
3136

32-
import static org.bytedeco.javacpp.opencv_core.CV_8S;
33-
import static org.bytedeco.javacpp.opencv_core.CV_8U;
34-
3537
/**
3638
* Publish an M-JPEG stream with the protocol used by SmartDashboard and the FRC Dashboard. This
3739
* allows FRC teams to view video streams on their dashboard during competition even when GRIP has
@@ -41,14 +43,23 @@ public class PublishVideoOperation implements Operation {
4143

4244
private static final Logger logger = Logger.getLogger(PublishVideoOperation.class.getName());
4345

46+
/**
47+
* Flags whether or not cscore was loaded. If it could not be loaded, the MJPEG streaming server
48+
* can't be started, preventing this operation from running.
49+
*/
50+
private static final boolean cscoreLoaded;
51+
4452
static {
53+
boolean loaded;
4554
try {
4655
// Loading the CameraServerJNI class will load the appropriate platform-specific OpenCV JNI
4756
CameraServerJNI.getHostname();
57+
loaded = true;
4858
} catch (Throwable e) {
49-
logger.log(Level.SEVERE, "CameraServerJNI load failed! Exiting", e);
50-
System.exit(31);
59+
logger.log(Level.SEVERE, "CameraServerJNI load failed!", e);
60+
loaded = false;
5161
}
62+
cscoreLoaded = loaded;
5263
}
5364

5465
public static final OperationDescription DESCRIPTION =
@@ -58,15 +69,15 @@ public class PublishVideoOperation implements Operation {
5869
.category(OperationDescription.Category.NETWORK)
5970
.icon(Icon.iconStream("publish-video"))
6071
.build();
61-
private static final int PORT = 1180;
72+
private static final int INITIAL_PORT = 1180;
73+
private static final int MAX_STEP_COUNT = 10; // limit ports to 1180-1189
6274

6375
@SuppressWarnings("PMD.AssignmentToNonFinalStatic")
6476
private static int totalStepCount;
6577
@SuppressWarnings("PMD.AssignmentToNonFinalStatic")
6678
private static int numSteps;
67-
private static final int MAX_STEP_COUNT = 10; // limit ports to 1180-1189
6879
private static final Deque<Integer> availablePorts =
69-
Stream.iterate(PORT, i -> i + 1)
80+
Stream.iterate(INITIAL_PORT, i -> i + 1)
7081
.limit(MAX_STEP_COUNT)
7182
.collect(Collectors.toCollection(LinkedList::new));
7283

@@ -77,7 +88,7 @@ public class PublishVideoOperation implements Operation {
7788

7889
// Write to the /CameraPublisher table so the MJPEG streams are discoverable by other
7990
// applications connected to the same NetworkTable server (eg Shuffleboard)
80-
private static final ITable cameraPublisherTable = NetworkTable.getTable("/CameraPublisher");
91+
private final ITable cameraPublisherTable = NetworkTable.getTable("/CameraPublisher"); // NOPMD
8192
private final ITable ourTable;
8293
private final Mat publishMat = new Mat();
8394
private long lastFrame = -1;
@@ -95,16 +106,22 @@ public PublishVideoOperation(InputSocket.Factory inputSocketFactory) {
95106
this.qualitySocket = inputSocketFactory.create(SocketHints.Inputs
96107
.createNumberSliderSocketHint("Quality", 80, 0, 100));
97108

98-
int ourPort = availablePorts.removeFirst();
99-
100-
server = new MjpegServer("GRIP video publishing server " + totalStepCount, ourPort);
101-
serverSource = new CvSource("GRIP CvSource " + totalStepCount,
102-
VideoMode.PixelFormat.kMJPEG, 0, 0, 0);
103-
server.setSource(serverSource);
104-
105-
ourTable = cameraPublisherTable.getSubTable("GRIP-" + totalStepCount);
106-
ourTable.putStringArray("streams",
107-
new String[]{CameraServerJNI.getHostname() + ":" + ourPort + "/?action=stream"});
109+
if (cscoreLoaded) {
110+
int ourPort = availablePorts.removeFirst();
111+
112+
server = new MjpegServer("GRIP video publishing server " + totalStepCount, ourPort);
113+
serverSource = new CvSource("GRIP CvSource " + totalStepCount,
114+
VideoMode.PixelFormat.kMJPEG, 0, 0, 0);
115+
server.setSource(serverSource);
116+
117+
ourTable = cameraPublisherTable.getSubTable("GRIP-" + totalStepCount);
118+
ourTable.putStringArray("streams",
119+
new String[]{"mjpeg:http://" + getHostName() + ":" + ourPort + "/?action=stream"});
120+
} else {
121+
server = null;
122+
serverSource = null;
123+
ourTable = null;
124+
}
108125

109126
numSteps++;
110127
totalStepCount++;
@@ -126,39 +143,60 @@ public List<OutputSocket> getOutputSockets() {
126143
@Override
127144
public void perform() {
128145
final long now = System.nanoTime(); // NOPMD
146+
147+
if (!cscoreLoaded) {
148+
throw new IllegalStateException(
149+
"cscore could not be loaded. The image streaming server cannot be started.");
150+
}
151+
129152
opencv_core.Mat input = inputSocket.getValue().get();
130153
if (input.empty() || input.isNull()) {
131154
throw new IllegalArgumentException("Input image must not be empty");
132155
}
133156

134157
copyJavaCvToOpenCvMat(input, publishMat);
158+
// Make sure the output resolution is up to date. Might not be needed, depends on cscore updates
159+
serverSource.setResolution(input.size().width(), input.size().height());
135160
serverSource.putFrame(publishMat);
136161
if (lastFrame != -1) {
137162
long dt = now - lastFrame;
138163
serverSource.setFPS((int) (1e9 / dt));
139164
}
140165
lastFrame = now;
141-
server.setSource(serverSource);
142166
}
143167

144168
@Override
145169
public synchronized void cleanUp() {
146-
// Stop the video server if there are no Publish Video steps left
147170
numSteps--;
148-
availablePorts.addFirst(server.getPort());
149-
ourTable.getKeys().forEach(ourTable::delete);
150-
}
151-
152-
private void copyJavaCvToOpenCvMat(opencv_core.Mat javaCvMat, Mat openCvMat) {
153-
if (javaCvMat.depth() != CV_8U && javaCvMat.depth() != CV_8S) {
154-
throw new IllegalArgumentException("Only 8-bit depth images are supported");
171+
if (cscoreLoaded) {
172+
availablePorts.addFirst(server.getPort());
173+
ourTable.getKeys().forEach(ourTable::delete);
174+
serverSource.setConnected(false);
175+
serverSource.free();
176+
server.free();
155177
}
178+
}
156179

157-
final opencv_core.Size size = javaCvMat.size();
158-
159-
// Make sure the output resolution is up to date
160-
serverSource.setResolution(size.width(), size.height());
161-
180+
/**
181+
* Copies the data from a JavaCV Mat wrapper object into an OpenCV Mat wrapper object so it's
182+
* usable by the {@link CvSource} for this operation.
183+
*
184+
* <p>Since the JavaCV and OpenCV bindings both target the same native version of OpenCV, this is
185+
* implemented by simply changing the OpenCV Mat's native pointer to be the same as the one for
186+
* the JavaCV Mat. This prevents memory copies and resizing/reallocating memory for the OpenCV
187+
* wrapper to fit the source image. Updating the pointer is a simple field write (albeit via
188+
* reflection), which is much faster and easier than allocating and copying byte buffers.</p>
189+
*
190+
* <p>A caveat to this approach is that the memory layout used by the OpenCV binaries bundled with
191+
* both wrapper libraries <i>must</i> be identical. Using the same OpenCV version for both
192+
* libraries should be enough.</p>
193+
*
194+
* @param javaCvMat the JavaCV Mat wrapper object to copy from
195+
* @param openCvMat the OpenCV Mat wrapper object to copy into
196+
* @throws RuntimeException if the OpenCV native pointer could not be set
197+
*/
198+
private static void copyJavaCvToOpenCvMat(opencv_core.Mat javaCvMat, Mat openCvMat)
199+
throws RuntimeException {
162200
// Make the OpenCV Mat object point to the same block of memory as the JavaCV object.
163201
// This requires no data transfers or copies and is O(1) instead of O(n)
164202
if (javaCvMat.address() != openCvMat.nativeObj) {
@@ -168,7 +206,44 @@ private void copyJavaCvToOpenCvMat(opencv_core.Mat javaCvMat, Mat openCvMat) {
168206
nativeObjField.setLong(openCvMat, javaCvMat.address());
169207
} catch (ReflectiveOperationException e) {
170208
logger.log(Level.WARNING, "Could not set native object pointer", e);
209+
throw new RuntimeException("Could not copy the image", e);
210+
}
211+
}
212+
}
213+
214+
/**
215+
* Multi platform method for getting the hostname of the local computer. cscore's
216+
* {@link CameraServerJNI#getHostname() getHostName() function} only works on Linux, so we need to
217+
* implement the method for Windows and Mac ourselves.
218+
*/
219+
private static String getHostName() {
220+
if (SystemUtils.IS_OS_WINDOWS) {
221+
// Use the Windows `hostname` command-line utility
222+
// This will return a single line of text containing the hostname, no parsing required
223+
ProcessBuilder builder = new ProcessBuilder("hostname");
224+
Process hostname;
225+
try {
226+
hostname = builder.start();
227+
} catch (IOException e) {
228+
logger.log(Level.WARNING, "Could not start hostname process", e);
229+
return "";
230+
}
231+
try (BufferedReader in =
232+
new BufferedReader(new InputStreamReader(hostname.getInputStream()))) {
233+
return in.readLine() + ".local";
234+
} catch (IOException e) {
235+
logger.log(Level.WARNING, "Could not read the hostname process output", e);
236+
return "";
171237
}
238+
} else if (SystemUtils.IS_OS_LINUX) {
239+
// cscore already defines it for linux
240+
return CameraServerJNI.getHostname();
241+
} else if (SystemUtils.IS_OS_MAC) {
242+
// todo
243+
return "TODO-MAC";
244+
} else {
245+
throw new UnsupportedOperationException(
246+
"Unsupported operating system " + System.getProperty("os.name"));
172247
}
173248
}
174249

0 commit comments

Comments
 (0)