-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathJeVoisInterface.java
654 lines (566 loc) · 22.7 KB
/
JeVoisInterface.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
package org.team639.subsystems;
import edu.wpi.first.cscore.MjpegServer;
import edu.wpi.first.cscore.UsbCamera;
import edu.wpi.first.cscore.VideoMode.PixelFormat;
import edu.wpi.first.wpilibj.DriverStation;
import edu.wpi.first.wpilibj.SerialPort;
import edu.wpi.first.wpilibj.Timer;
import edu.wpi.first.networktables.NetworkTable;
import edu.wpi.first.networktables.NetworkTableEntry;
import edu.wpi.first.networktables.NetworkTableInstance;
public class JeVoisInterface
{
// Serial Port Constants
private static final int BAUD_RATE = 38400;
// MJPG Streaming Constants
private static final int MJPG_STREAM_PORT = 1180;
// Packet format constants
private static final String PKT_START = "{";
private static final String PKT_END = "}";
private static final String PKT_SEP = ",";
// Confgure the camera to stream debug images or not.
private boolean broadcastUSBCam = false;
// When not streaming, use this mapping
private static final int NO_STREAM_MAPPING = 0;
// When streaming, use this set of configuration
private static final int STREAM_WIDTH_PX = 320;
private static final int STREAM_HEIGHT_PX = 240;
private static final int STREAM_RATE_FPS = 30;
// Serial port used for getting target data from JeVois
private SerialPort visionPort = null;
// USBCam and server used for broadcasting a webstream of what is seen
private UsbCamera visionCam = null;
private MjpegServer camServer = null;
// Status variables
private boolean dataStreamRunning = false;
private boolean camStreamRunning = false;
private boolean visionOnline = false;
// Packet rate performace tracking
private double packetRxTime = 0;
private double prevPacketRxTime = 0;
private double packetRate_PPS = 0;
// Most recently seen target information
// private boolean tgtVisible = false;
private double[] positionHub;
private double[] positionRed;
private double[] positionBlue;
// private double tgtRange = 0;
private double tgtTime = 0;
// Info about the JeVois performace & status
private double jeVoisCpuTempC = 0;
private double jeVoisCpuLoadPct = 0;
private double jeVoisFramerateFPS = 0;
private double packetRxRatePPS = 0;
//=======================================================
//== BEGIN PUBLIC INTERFACE
//=======================================================
/**
* Constructor (simple). Opens a USB serial port to the JeVois camera, sends a few test commands checking for error,
* then fires up the user's program and begins listening for target info packets in the background
*/
public JeVoisInterface() {
this(false); //Default - stream disabled, just run serial.
positionHub = new double[2];
positionRed = new double[2];
positionBlue = new double[2];
}
/**
* Constructor (more complex). Opens a USB serial port to the JeVois camera, sends a few test commands checking for error,
* then fires up the user's program and begins listening for target info packets in the background.
* Pass TRUE to additionaly enable a USB camera stream of what the vision camera is seeing.
*/
public JeVoisInterface(boolean useUSBStream)
{
int retry_counter = 0;
//Retry strategy to get this serial port open.
//I have yet to see a single retry used assuming the camera is plugged in
// but you never know.
while(visionPort == null && retry_counter++ < 10){
try
{
System.out.print("Creating JeVois SerialPort...");
visionPort = new SerialPort(BAUD_RATE, SerialPort.Port.kMXP);
System.out.println("SUCCESS!!");
}
catch (Exception e)
{
System.out.println("FAILED!!");
e.printStackTrace();
sleep(500);
System.out.println("Retry " + Integer.toString(retry_counter));
}
}
//Report an error if we didn't get to open the serial port
if(visionPort == null)
{
DriverStation.reportError("Cannot open serial port to JeVois. Not starting vision system.", false);
return;
}
//Test to make sure we are actually talking to the JeVois
if(sendPing() != 0)
{
DriverStation.reportError("JeVois ping test failed. Continuing anyway.", false);
//return;
}
//Ensure the JeVois is starting with the stream off.
stopDataOnlyStream();
setCameraStreamActive(useUSBStream);
start();
//Start listening for packets
packetListenerThread.setDaemon(true);
packetListenerThread.start();
}
public void start(){
if(broadcastUSBCam){
//Start streaming the JeVois via webcam
//This auto-starts the serial stream
startCameraStream();
System.out.println("Started camera stream\n");
} else {
startDataOnlyStream();
System.out.println("Started HEADLESS stream\n");
}
}
public void stop(){
if(broadcastUSBCam){
//Start streaming the JeVois via webcam
//This auto-starts the serial stream
stopCameraStream();
} else {
stopDataOnlyStream();
}
}
/**
* Send commands to the JeVois to configure it for image-processing friendly parameters
*/
public void setCamVisionProcMode() {
if (visionPort != null){
sendCmdAndCheck("setcam autoexp 1"); //Disable auto exposure
sendCmdAndCheck("setcam absexp 75"); //Force exposure to a low value for vision processing
}
}
/**
* Send parameters to the camera to configure it for a human-readable image
*/
public void setCamHumanDriverMode() {
if (visionPort != null){
sendCmdAndCheck("setcam autoexp 0"); //Enable AutoExposure
}
}
/*
* Main getters/setters
*/
/**
* Set to true to enable the camera stream, or set to false to stream serial-packets only.
* Note this cannot be changed at runtime due to jevois constraints. You must stop whatatever processing
* is going on first.
*/
public void setCameraStreamActive(boolean active){
if(dataStreamRunning == false){
broadcastUSBCam = active;
} else {
DriverStation.reportError("Attempt to change cal stream mode while JeVois is still running. This is disallowed.", false);
}
}
// /**
// * Returns the most recently seen target's angle relative to the camera in degrees
// * Positive means to the Right of center, negative means to the left
// */
// public double getTgtAngle_Deg() {
// return tgtAngleDeg;
// }
// /**
// * Returns the most recently seen target's range from the camera in inches
// * Range means distance along the ground from camera mount point to observed target
// * Return values should only be positive
// */
// public double getTgtRange_in() {
// return tgtRange;
// }
// /**
// * Get the estimated timestamp of the most recent target observation.
// * This is calculated based on the FPGA timestamp at packet RX time, minus the reportetd vision pipeline delay.
// * It will not currently account for serial hardware or other delays.
// */
// public double getTgtTime() {
// return tgtTime;
// }
// /**
// * Returns true when the roboRIO is recieving packets from the JeVois, false if no packets have been recieved.
// * Other modules should not use the vision processing results if this returns false.
// */
// public boolean isVisionOnline() {
// return visionOnline;
// }
// /**
// * Returns true when the JeVois sees a target and is tracking it, false otherwise.
// */
// public boolean isTgtVisible() {
// return tgtVisible;
// }
/**
* Returns the JeVois's most recently reported CPU Temperature in deg C
*/
public double getJeVoisCPUTemp_C(){
return jeVoisCpuTempC;
}
/**
* Returns the JeVois's most recently reported CPU Load in percent of max
*/
public double getJeVoisCpuLoad_pct(){
return jeVoisCpuLoadPct;
}
/**
* Returns the JeVois's most recently reported pipline framerate in Frames per second
*/
public double getJeVoisFramerate_FPS(){
return jeVoisFramerateFPS;
}
/**
* Returns the roboRIO measured serial packet recieve rate in packets per second
*/
public int getPacketRxRate_PPS(){
if(visionOnline){
return (int)Math.round(packetRxRatePPS);
} else {
return 0;
}
}
//=======================================================
//== END PUBLIC INTERFACE
//=======================================================
/**
* This is the main perodic update function for the Listener. It is intended
* to be run in a background task, as it will block until it gets packets.
*/
private void backgroundUpdate()
{
// Grab packets and parse them.
String packet;
prevPacketRxTime = packetRxTime;
packet = blockAndGetPacket(2.0);
if (packet != null)
{
packetRxTime = Timer.getFPGATimestamp();
if (parsePacket(packet, packetRxTime) == 0)
{
visionOnline = true;
packetRxRatePPS = 1.0/(packetRxTime - prevPacketRxTime);
}
else
{
visionOnline = false;
}
} else
{
visionOnline = false;
DriverStation.reportWarning("No packet received (bg)", false);
}
}
/**
* Send the ping command to the JeVois to verify it is connected
* @return 0 on success, -1 on unexpected response, -2 on timeout
*/
private int sendPing() {
int retval = -1;
if (visionPort != null){
System.out.print("Sending PING\n");
retval = sendCmdAndCheck("ping");
} else {
System.out.print("Serial port NULL; not sending\n");
}
return retval;
}
private void startDataOnlyStream(){
//Send serial commands to start the streaming of target info
sendCmdAndCheck("setmapping " + Integer.toString(NO_STREAM_MAPPING));
sendCmdAndCheck("streamon");
dataStreamRunning = true;
}
private void stopDataOnlyStream(){
//Send serial commands to stop the streaming of target info
sendCmdAndCheck("streamoff");
dataStreamRunning = false;
}
/**
* Open an Mjpeg streamer from the JeVois camera
*/
private void startCameraStream(){
try{
System.out.print("Starting JeVois Cam Stream...");
visionCam = new UsbCamera("VisionProcCam", 0);
visionCam.setVideoMode(PixelFormat.kBGR, STREAM_WIDTH_PX, STREAM_HEIGHT_PX, STREAM_RATE_FPS);
camServer = new MjpegServer("VisionCamServer", MJPG_STREAM_PORT);
camServer.setSource(visionCam);
camStreamRunning = true;
dataStreamRunning = true;
System.out.println("SUCCESS!!");
} catch (Exception e) {
DriverStation.reportError("Cannot start camera stream from JeVois", false);
e.printStackTrace();
}
}
/**
* Cease the operation of the camera stream. Unknown if needed.
*/
private void stopCameraStream(){
if(camStreamRunning){
//camServer.free();
//visionCam.free();
camStreamRunning = false;
dataStreamRunning = false;
}
}
/**
* Sends a command over serial to JeVois and returns immediately.
* @param cmd String of the command to send (ex: "ping")
* @return number of bytes written
*/
public int sendCmd(String cmd){
int bytes;
bytes = visionPort.writeString(cmd + "\n");
System.out.println("wrote " + bytes + "/" + (cmd.length()+1) + " bytes, cmd: " + cmd);
return bytes;
};
/**
* Sends a command over serial to the JeVois, waits for a response, and checks that response
* Automatically ends the line termination character.
* @param cmd String of the command to send (ex: "ping")
* @return 0 if OK detected, -1 if ERR detected, -2 if timeout waiting for response
*/
public int sendCmdAndCheck(String cmd){
int retval = 0;
sendCmd(cmd);
retval = blockAndCheckForOK(1.0);
if(retval == -1){
System.out.println(cmd + " Produced an error");
} else if (retval == -2) {
System.out.println(cmd + " timed out");
}
return retval;
};
//Persistent but "local" variables for getBytesPeriodic()
private String getBytesWork = "";
private int loopCount = 0;
/**
* Read bytes from the serial port in a non-blocking fashion
* Will return the whole thing once the first "OK" or "ERR" is seen in the stream.
* Returns null if no string read back yet.
*/
private String getCmdResponseNonBlock() {
String retval = null;
if (visionPort != null){
if (visionPort.getBytesReceived() > 0) {
String rxString = visionPort.readString();
System.out.println("Waited: " + loopCount + " loops, Rcv'd: " + rxString);
getBytesWork += rxString;
if(getBytesWork.contains("OK") || getBytesWork.contains("ERR")){
retval = getBytesWork;
getBytesWork = "";
System.out.println(retval);
}
loopCount = 0;
} else {
++loopCount;
}
}
return retval;
}
/**
* Blocks thread execution till we get a response from the serial line
* or timeout.
* Return values:
* 0 = OK in response
* -1 = ERR in response
* -2 = No token found before timeout_s
*/
private int blockAndCheckForOK(double timeout_s){
int retval = -2;
double startTime = Timer.getFPGATimestamp();
String testStr = "";
if (visionPort != null){
while(Timer.getFPGATimestamp() - startTime < timeout_s){
if (visionPort.getBytesReceived() > 0) {
testStr += visionPort.readString();
if(testStr.contains("OK")){
retval = 0;
break;
}else if(testStr.contains("ERR")){
DriverStation.reportError("JeVois reported error:\n" + testStr, false);
retval = -1;
break;
}
} else {
sleep(10);
}
}
}
return retval;
}
// buffer to contain data from the port while we gather full packets
private StringBuffer packetBuffer = new StringBuffer(100);
/**
* Blocks thread execution till we get a valid packet from the serial line
* or timeout.
* Return values:
* String = the packet
* null = No full packet found before timeout_s
*/
private String blockAndGetPacket(double timeout_s){
String retval = null;
double startTime = Timer.getFPGATimestamp();
int endIdx = -1;
int startIdx = -1;
if (visionPort != null){
while(Timer.getFPGATimestamp() - startTime < timeout_s){
// Keep trying to get bytes from the serial port until the timeout expires.
if (visionPort.getBytesReceived() > 0) {
// If there are any bytes available, read them in and
// append them to the buffer.
packetBuffer = packetBuffer.append(visionPort.readString());
// Attempt to detect if the buffer currently contains a complete packet
if(packetBuffer.indexOf(PKT_START) != -1){
endIdx = packetBuffer.lastIndexOf(PKT_END);
if(endIdx != -1){
// Buffer also contains at least one start & end character.
// But we don't know if they're in the right order yet.
// Start by getting the most-recent packet end character's index
// Look for the index of the start character for the packet
// described by endIdx. Note this line of code assumes the
// start character for the packet must come _before_ the
// end character.
startIdx = packetBuffer.lastIndexOf(PKT_START, endIdx);
if(startIdx == -1){
// If there was no start character before the end character,
// we can assume that we have something a bit wacky in our
// buffer. For example: ",abc}garbage{1,2".
// Since we've started to receive a good packet, discard
// everything prior to the start character.
startIdx = packetBuffer.lastIndexOf(PKT_START);
packetBuffer.delete(0, startIdx);
} else {
// Buffer contains a full packet. Extract it and clean up buffer
retval = packetBuffer.substring(startIdx+1, endIdx-1);
packetBuffer.delete(0, endIdx+1);
break;
}
} else {
// In this case, we have a start character, but no end to the buffer yet.
// Do nothing, just wait for more characters to come in.
sleep(5);
}
} else {
// Buffer contains no start characters. None of the current buffer contents can
// be meaningful. Discard the whole thing.
packetBuffer.delete(0, packetBuffer.length());
sleep(5);
}
} else {
sleep(5);
}
}
}
return retval;
}
/**
* Private wrapper around the Thread.sleep method, to catch that interrupted error.
* @param time_ms
*/
private void sleep(int time_ms){
try {
Thread.sleep(time_ms);
} catch (InterruptedException e) {
System.out.println("DO NOT WAKE THE SLEEPY BEAST");
e.printStackTrace();
}
}
/**
* Mostly for debugging. Blocks execution forever and just prints all serial
* characters to the console. It might print a different message too if nothing
* comes in.
*/
public void blockAndPrintAllSerial(){
if (visionPort != null){
while(!Thread.interrupted()){
if (visionPort.getBytesReceived() > 0) {
System.out.print(visionPort.readString());
} else {
System.out.println("Nothing Rx'ed");
sleep(100);
}
}
}
}
private void parsePosition(String[] src, int start, double[] dst)
{
dst[0] = Double.parseDouble(src[start]);
dst[1] = Double.parseDouble(src[start+1]);
}
/**
* Parse individual numbers from a packet
* @param pkt
*/
public int parsePacket(String pkt, double rx_Time)
{
//Parsing constants. These must be aligned with JeVois code.
// Message is space-separated floating point numbers
// First three pairs of numbers are:
// d: distance (inches) -1 if target is not found
// phi: angle (radians)
// Order is: HUB REDBALL BLUEBALL
// Four performance numbers are:
// JV_framerate, JV_cpuload, JV_cputemp, JV_pipelinedelay
final int NTOK = 11;
final int TOK_FRAME_CNT = 0;
final int TOK_HUB = 1; // 1 and 2 are hub d and phi
final int TOK_RED = 3;
final int TOK_BLUE = 5;
final int TOK_JV_LOAD = 7;
final int TOK_JV_CPUTEMP = 8;
final int TOK_JV_FRATE = 9;
final int TOK_JV_PIPEDELAY = 10;
//Split string into many substrings, presuming those strings are separated by commas
String[] tokens = pkt.split(PKT_SEP);
System.out.println(tokens);
//Check there were enough substrings found
if (tokens.length < NTOK) {
DriverStation.reportError("Malformed vision packet. Expected " + Integer.toString(NTOK) +
", received " + Integer.toString(tokens.length), false);
return -1;
}
//Convert each string into the proper internal value
try
{
//Use Java built-in double to string conversion on most of the rest
parsePosition(tokens, TOK_HUB, positionHub);
parsePosition(tokens, TOK_RED, positionRed);
parsePosition(tokens, TOK_BLUE, positionBlue);
tgtTime = rx_Time - Double.parseDouble(tokens[TOK_JV_PIPEDELAY])/1000000.0;
jeVoisCpuTempC = Double.parseDouble(tokens[TOK_JV_CPUTEMP]);
jeVoisCpuLoadPct = Double.parseDouble(tokens[TOK_JV_LOAD]);
// sd.putNumberArray("positionHub", positionHub);
// sd.putNumberArray("positionRed", positionRed);
// sd.putNumberArray("positionBlue", positionBlue);
} catch (Exception e) {
DriverStation.reportError("Unhandled exception while parsing Vision packet: " + e.getMessage() + "\n" + e.getStackTrace(), false);
return -1;
}
return 0;
}
/**
* This thread runs a periodic task in the background to listen for vision camera packets.
*/
Thread packetListenerThread = new Thread(new Runnable()
{
public void run()
{
// NetworkTableInstance inst = NetworkTableInstance.getDefault();
// NetworkTable sd = inst.getTable("Visions");
// NetworkTableEntry positionHub = sd.getEntry("positionHub");
while(!Thread.interrupted())
{
backgroundUpdate();
}
}
});
}