Compare commits

...

3 Commits

Author SHA1 Message Date
Matt
d9c2a382f1 Update build.yml (#1276) 2024-03-14 00:32:08 -05:00
Matt
e125632960 Free native resources in apriltag pipelines (#1272)
Addresses memory leak when switching between apriltag/aruco pipelines
2024-03-14 01:22:32 -04:00
Matt
d8f82bf9ee Opencv cal: CALIB_USE_LU and use camera focal length guess (#1268) 2024-03-09 08:31:54 -05:00
16 changed files with 191 additions and 45 deletions

View File

@@ -367,13 +367,13 @@ jobs:
- os: ubuntu-latest - os: ubuntu-latest
artifact-name: LinuxArm64 artifact-name: LinuxArm64
image_suffix: orangepi5 image_suffix: orangepi5
image_url: https://github.com/PhotonVision/photon-image-modifier/releases/download/v2024.0.9/photonvision_opi5.img.xz image_url: https://github.com/PhotonVision/photon-image-modifier/releases/download/v2024.0.10/photonvision_opi5.img.xz
cpu: cortex-a8 cpu: cortex-a8
image_additional_mb: 4096 image_additional_mb: 4096
- os: ubuntu-latest - os: ubuntu-latest
artifact-name: LinuxArm64 artifact-name: LinuxArm64
image_suffix: orangepi5plus image_suffix: orangepi5plus
image_url: https://github.com/PhotonVision/photon-image-modifier/releases/download/v2024.0.9/photonvision_opi5plus.img.xz image_url: https://github.com/PhotonVision/photon-image-modifier/releases/download/v2024.0.10/photonvision_opi5plus.img.xz
cpu: cortex-a8 cpu: cortex-a8
image_additional_mb: 4096 image_additional_mb: 4096

View File

@@ -23,16 +23,37 @@ import java.util.Comparator;
import org.opencv.core.Mat; import org.opencv.core.Mat;
import org.opencv.objdetect.ArucoDetector; import org.opencv.objdetect.ArucoDetector;
import org.opencv.objdetect.DetectorParameters; import org.opencv.objdetect.DetectorParameters;
import org.opencv.objdetect.Dictionary;
import org.opencv.objdetect.Objdetect; import org.opencv.objdetect.Objdetect;
import org.photonvision.common.logging.LogGroup; import org.photonvision.common.logging.LogGroup;
import org.photonvision.common.logging.Logger; import org.photonvision.common.logging.Logger;
import org.photonvision.vision.opencv.Releasable;
/** This class wraps an {@link ArucoDetector} for convenience. */ /** This class wraps an {@link ArucoDetector} for convenience. */
public class PhotonArucoDetector { public class PhotonArucoDetector implements Releasable {
private static final Logger logger = new Logger(PhotonArucoDetector.class, LogGroup.VisionModule); private static final Logger logger = new Logger(PhotonArucoDetector.class, LogGroup.VisionModule);
private final ArucoDetector detector = private static class ArucoDetectorHack extends ArucoDetector {
new ArucoDetector(Objdetect.getPredefinedDictionary(Objdetect.DICT_APRILTAG_16h5)); public ArucoDetectorHack(Dictionary predefinedDictionary) {
super(predefinedDictionary);
}
// avoid double-free by keeping track of this ourselves (ew)
private boolean freed = false;
@Override
public void finalize() throws Throwable {
if (freed) {
return;
}
super.finalize();
freed = true;
}
}
private final ArucoDetectorHack detector =
new ArucoDetectorHack(Objdetect.getPredefinedDictionary(Objdetect.DICT_APRILTAG_16h5));
private final Mat ids = new Mat(); private final Mat ids = new Mat();
private final ArrayList<Mat> cornerMats = new ArrayList<>(); private final ArrayList<Mat> cornerMats = new ArrayList<>();
@@ -95,4 +116,16 @@ public class PhotonArucoDetector {
return results; return results;
} }
@Override
public void release() {
try {
detector.finalize();
} catch (Throwable e) {
logger.error("Exception destroying PhotonArucoDetector", e);
}
ids.release();
for (var m : cornerMats) m.release();
cornerMats.clear();
}
} }

View File

@@ -33,7 +33,7 @@ import org.photonvision.vision.opencv.CVMat;
* path}. * path}.
*/ */
public class FileFrameProvider extends CpuImageProcessor { public class FileFrameProvider extends CpuImageProcessor {
public static final int MAX_FPS = 5; public static final int MAX_FPS = 10;
private static int count = 0; private static int count = 0;
private final int thisIndex = count++; private final int thisIndex = count++;

View File

@@ -21,11 +21,13 @@ import edu.wpi.first.apriltag.AprilTagDetection;
import edu.wpi.first.apriltag.AprilTagDetector; import edu.wpi.first.apriltag.AprilTagDetector;
import java.util.List; import java.util.List;
import org.photonvision.vision.opencv.CVMat; import org.photonvision.vision.opencv.CVMat;
import org.photonvision.vision.opencv.Releasable;
import org.photonvision.vision.pipe.CVPipe; import org.photonvision.vision.pipe.CVPipe;
public class AprilTagDetectionPipe public class AprilTagDetectionPipe
extends CVPipe<CVMat, List<AprilTagDetection>, AprilTagDetectionPipeParams> { extends CVPipe<CVMat, List<AprilTagDetection>, AprilTagDetectionPipeParams>
private final AprilTagDetector m_detector = new AprilTagDetector(); implements Releasable {
private AprilTagDetector m_detector = new AprilTagDetector();
public AprilTagDetectionPipe() { public AprilTagDetectionPipe() {
super(); super();
@@ -40,6 +42,10 @@ public class AprilTagDetectionPipe
return List.of(); return List.of();
} }
if (m_detector == null) {
throw new RuntimeException("Apriltag detector was released!");
}
var ret = m_detector.detect(in.getMat()); var ret = m_detector.detect(in.getMat());
if (ret == null) { if (ret == null) {
@@ -60,4 +66,10 @@ public class AprilTagDetectionPipe
super.setParams(newParams); super.setParams(newParams);
} }
@Override
public void release() {
m_detector.close();
m_detector = null;
}
} }

View File

@@ -25,13 +25,15 @@ import org.opencv.calib3d.Calib3d;
import org.opencv.core.MatOfPoint2f; import org.opencv.core.MatOfPoint2f;
import org.opencv.core.Point; import org.opencv.core.Point;
import org.photonvision.vision.calibration.CameraCalibrationCoefficients; import org.photonvision.vision.calibration.CameraCalibrationCoefficients;
import org.photonvision.vision.opencv.Releasable;
import org.photonvision.vision.pipe.CVPipe; import org.photonvision.vision.pipe.CVPipe;
public class AprilTagPoseEstimatorPipe public class AprilTagPoseEstimatorPipe
extends CVPipe< extends CVPipe<
AprilTagDetection, AprilTagDetection,
AprilTagPoseEstimate, AprilTagPoseEstimate,
AprilTagPoseEstimatorPipe.AprilTagPoseEstimatorPipeParams> { AprilTagPoseEstimatorPipe.AprilTagPoseEstimatorPipeParams>
implements Releasable {
private final AprilTagPoseEstimator m_poseEstimator = private final AprilTagPoseEstimator m_poseEstimator =
new AprilTagPoseEstimator(new AprilTagPoseEstimator.Config(0, 0, 0, 0, 0)); new AprilTagPoseEstimator(new AprilTagPoseEstimator.Config(0, 0, 0, 0, 0));
@@ -92,6 +94,11 @@ public class AprilTagPoseEstimatorPipe
super.setParams(newParams); super.setParams(newParams);
} }
@Override
public void release() {
temp.release();
}
public static class AprilTagPoseEstimatorPipeParams { public static class AprilTagPoseEstimatorPipeParams {
final AprilTagPoseEstimator.Config config; final AprilTagPoseEstimator.Config config;
final CameraCalibrationCoefficients calibration; final CameraCalibrationCoefficients calibration;

View File

@@ -29,10 +29,12 @@ import org.opencv.objdetect.Objdetect;
import org.photonvision.vision.aruco.ArucoDetectionResult; import org.photonvision.vision.aruco.ArucoDetectionResult;
import org.photonvision.vision.aruco.PhotonArucoDetector; import org.photonvision.vision.aruco.PhotonArucoDetector;
import org.photonvision.vision.opencv.CVMat; import org.photonvision.vision.opencv.CVMat;
import org.photonvision.vision.opencv.Releasable;
import org.photonvision.vision.pipe.CVPipe; import org.photonvision.vision.pipe.CVPipe;
public class ArucoDetectionPipe public class ArucoDetectionPipe
extends CVPipe<CVMat, List<ArucoDetectionResult>, ArucoDetectionPipeParams> { extends CVPipe<CVMat, List<ArucoDetectionResult>, ArucoDetectionPipeParams>
implements Releasable {
// ArucoDetector wrapper class // ArucoDetector wrapper class
private final PhotonArucoDetector photonDetector = new PhotonArucoDetector(); private final PhotonArucoDetector photonDetector = new PhotonArucoDetector();
@@ -131,4 +133,9 @@ public class ArucoDetectionPipe
var pt2 = new Point(corner.x + windowSize, corner.y + windowSize); var pt2 = new Point(corner.x + windowSize, corner.y + windowSize);
Imgproc.rectangle(outputMat, pt1, pt2, new Scalar(0, 0, 255), thickness); Imgproc.rectangle(outputMat, pt1, pt2, new Scalar(0, 0, 255), thickness);
} }
@Override
public void release() {
photonDetector.release();
}
} }

View File

@@ -32,13 +32,15 @@ import org.opencv.core.MatOfPoint3f;
import org.opencv.core.Point3; import org.opencv.core.Point3;
import org.photonvision.vision.aruco.ArucoDetectionResult; import org.photonvision.vision.aruco.ArucoDetectionResult;
import org.photonvision.vision.calibration.CameraCalibrationCoefficients; import org.photonvision.vision.calibration.CameraCalibrationCoefficients;
import org.photonvision.vision.opencv.Releasable;
import org.photonvision.vision.pipe.CVPipe; import org.photonvision.vision.pipe.CVPipe;
public class ArucoPoseEstimatorPipe public class ArucoPoseEstimatorPipe
extends CVPipe< extends CVPipe<
ArucoDetectionResult, ArucoDetectionResult,
AprilTagPoseEstimate, AprilTagPoseEstimate,
ArucoPoseEstimatorPipe.ArucoPoseEstimatorPipeParams> { ArucoPoseEstimatorPipe.ArucoPoseEstimatorPipeParams>
implements Releasable {
// image points of marker corners // image points of marker corners
private final MatOfPoint2f imagePoints = new MatOfPoint2f(Mat.zeros(4, 1, CvType.CV_32FC2)); private final MatOfPoint2f imagePoints = new MatOfPoint2f(Mat.zeros(4, 1, CvType.CV_32FC2));
// rvec/tvec estimations from solvepnp // rvec/tvec estimations from solvepnp
@@ -117,6 +119,18 @@ public class ArucoPoseEstimatorPipe
super.setParams(newParams); super.setParams(newParams);
} }
@Override
public void release() {
imagePoints.release();
for (var m : rvecs) m.release();
rvecs.clear();
for (var m : tvecs) m.release();
tvecs.clear();
rvec.release();
tvec.release();
reprojectionErrors.release();
}
public static class ArucoPoseEstimatorPipeParams { public static class ArucoPoseEstimatorPipeParams {
final CameraCalibrationCoefficients calibration; final CameraCalibrationCoefficients calibration;
final double tagSize; final double tagSize;

View File

@@ -38,13 +38,26 @@ import org.photonvision.vision.calibration.CameraCalibrationCoefficients;
import org.photonvision.vision.calibration.CameraLensModel; import org.photonvision.vision.calibration.CameraLensModel;
import org.photonvision.vision.calibration.JsonImageMat; import org.photonvision.vision.calibration.JsonImageMat;
import org.photonvision.vision.calibration.JsonMatOfDouble; import org.photonvision.vision.calibration.JsonMatOfDouble;
import org.photonvision.vision.frame.FrameStaticProperties;
import org.photonvision.vision.pipe.CVPipe; import org.photonvision.vision.pipe.CVPipe;
import org.photonvision.vision.pipe.impl.FindBoardCornersPipe.FindBoardCornersPipeResult;
public class Calibrate3dPipe public class Calibrate3dPipe
extends CVPipe< extends CVPipe<
List<FindBoardCornersPipe.FindBoardCornersPipeResult>, Calibrate3dPipe.CalibrationInput,
CameraCalibrationCoefficients, CameraCalibrationCoefficients,
Calibrate3dPipe.CalibratePipeParams> { Calibrate3dPipe.CalibratePipeParams> {
public static class CalibrationInput {
final List<FindBoardCornersPipe.FindBoardCornersPipeResult> observations;
final FrameStaticProperties imageProps;
public CalibrationInput(
List<FindBoardCornersPipeResult> observations, FrameStaticProperties imageProps) {
this.observations = observations;
this.imageProps = imageProps;
}
}
// For logging // For logging
private static final Logger logger = new Logger(Calibrate3dPipe.class, LogGroup.General); private static final Logger logger = new Logger(Calibrate3dPipe.class, LogGroup.General);
@@ -63,10 +76,9 @@ public class Calibrate3dPipe
* @return Result of processing. * @return Result of processing.
*/ */
@Override @Override
protected CameraCalibrationCoefficients process( protected CameraCalibrationCoefficients process(CalibrationInput in) {
List<FindBoardCornersPipe.FindBoardCornersPipeResult> in) { var filteredIn =
in = in.observations.stream()
in.stream()
.filter( .filter(
it -> it ->
it != null it != null
@@ -79,17 +91,21 @@ public class Calibrate3dPipe
var start = System.nanoTime(); var start = System.nanoTime();
if (MrCalJNILoader.getInstance().isLoaded() && params.useMrCal) { if (MrCalJNILoader.getInstance().isLoaded() && params.useMrCal) {
logger.debug("Calibrating with mrcal!"); logger.debug("Calibrating with mrcal!");
ret = calibrateMrcal(in); ret =
calibrateMrcal(
filteredIn, in.imageProps.horizontalFocalLength, in.imageProps.verticalFocalLength);
} else { } else {
logger.debug("Calibrating with opencv!"); logger.debug("Calibrating with opencv!");
ret = calibrateOpenCV(in); ret =
calibrateOpenCV(
filteredIn, in.imageProps.horizontalFocalLength, in.imageProps.verticalFocalLength);
} }
var dt = System.nanoTime() - start; var dt = System.nanoTime() - start;
if (ret != null) if (ret != null)
logger.info( logger.info(
"CALIBRATION SUCCESS for res " "CALIBRATION SUCCESS for res "
+ in.get(0).size + in.observations.get(0).size
+ " in " + " in "
+ dt / 1e6 + dt / 1e6
+ "ms! camMatrix: \n" + "ms! camMatrix: \n"
@@ -103,7 +119,7 @@ public class Calibrate3dPipe
} }
protected CameraCalibrationCoefficients calibrateOpenCV( protected CameraCalibrationCoefficients calibrateOpenCV(
List<FindBoardCornersPipe.FindBoardCornersPipeResult> in) { List<FindBoardCornersPipe.FindBoardCornersPipeResult> in, double fxGuess, double fyGuess) {
List<Mat> objPoints = in.stream().map(it -> it.objectPoints).collect(Collectors.toList()); List<Mat> objPoints = in.stream().map(it -> it.objectPoints).collect(Collectors.toList());
List<Mat> imgPts = in.stream().map(it -> it.imagePoints).collect(Collectors.toList()); List<Mat> imgPts = in.stream().map(it -> it.imagePoints).collect(Collectors.toList());
if (objPoints.size() != imgPts.size()) { if (objPoints.size() != imgPts.size()) {
@@ -111,30 +127,32 @@ public class Calibrate3dPipe
return null; return null;
} }
Mat cameraMatrix = new Mat(); Mat cameraMatrix = new Mat(3, 3, CvType.CV_64F);
MatOfDouble distortionCoefficients = new MatOfDouble(); MatOfDouble distortionCoefficients = new MatOfDouble();
List<Mat> rvecs = new ArrayList<>(); List<Mat> rvecs = new ArrayList<>();
List<Mat> tvecs = new ArrayList<>(); List<Mat> tvecs = new ArrayList<>();
// RMS of the calibration // initial camera matrix guess
double calibrationAccuracy; double cx = (in.get(0).size.width / 2.0) - 0.5;
double cy = (in.get(0).size.width / 2.0) - 0.5;
cameraMatrix.put(0, 0, new double[] {fxGuess, 0, cx, 0, fyGuess, cy, 0, 0, 1});
try { try {
// FindBoardCorners pipe outputs all the image points, object points, and frames to calculate // FindBoardCorners pipe outputs all the image points, object points, and frames to calculate
// imageSize from, other parameters are output Mats // imageSize from, other parameters are output Mats
calibrationAccuracy = Calib3d.calibrateCameraExtended(
Calib3d.calibrateCameraExtended( objPoints,
objPoints, imgPts,
imgPts, new Size(in.get(0).size.width, in.get(0).size.height),
new Size(in.get(0).size.width, in.get(0).size.height), cameraMatrix,
cameraMatrix, distortionCoefficients,
distortionCoefficients, rvecs,
rvecs, tvecs,
tvecs, stdDeviationsIntrinsics,
stdDeviationsIntrinsics, stdDeviationsExtrinsics,
stdDeviationsExtrinsics, perViewErrors,
perViewErrors); Calib3d.CALIB_USE_LU + Calib3d.CALIB_USE_INTRINSIC_GUESS);
} catch (Exception e) { } catch (Exception e) {
logger.error("Calibration failed!", e); logger.error("Calibration failed!", e);
e.printStackTrace(); e.printStackTrace();
@@ -164,13 +182,12 @@ public class Calibrate3dPipe
} }
protected CameraCalibrationCoefficients calibrateMrcal( protected CameraCalibrationCoefficients calibrateMrcal(
List<FindBoardCornersPipe.FindBoardCornersPipeResult> in) { List<FindBoardCornersPipe.FindBoardCornersPipeResult> in, double fxGuess, double fyGuess) {
List<MatOfPoint2f> corner_locations = List<MatOfPoint2f> corner_locations =
in.stream().map(it -> it.imagePoints).map(MatOfPoint2f::new).collect(Collectors.toList()); in.stream().map(it -> it.imagePoints).map(MatOfPoint2f::new).collect(Collectors.toList());
int imageWidth = (int) in.get(0).size.width; int imageWidth = (int) in.get(0).size.width;
int imageHeight = (int) in.get(0).size.height; int imageHeight = (int) in.get(0).size.height;
final double FOCAL_LENGTH_GUESS = 1200;
MrCalResult result = MrCalResult result =
MrCalJNI.calibrateCamera( MrCalJNI.calibrateCamera(
@@ -180,7 +197,7 @@ public class Calibrate3dPipe
params.squareSize, params.squareSize,
imageWidth, imageWidth,
imageHeight, imageHeight,
FOCAL_LENGTH_GUESS); (fxGuess + fyGuess) / 2.0);
// intrinsics are fx fy cx cy from mrcal // intrinsics are fx fy cx cy from mrcal
JsonMatOfDouble cameraMatrixMat = JsonMatOfDouble cameraMatrixMat =

View File

@@ -36,6 +36,7 @@ import org.photonvision.vision.frame.FrameThresholdType;
import org.photonvision.vision.opencv.CVMat; import org.photonvision.vision.opencv.CVMat;
import org.photonvision.vision.opencv.ImageRotationMode; import org.photonvision.vision.opencv.ImageRotationMode;
import org.photonvision.vision.pipe.CVPipe.CVPipeResult; import org.photonvision.vision.pipe.CVPipe.CVPipeResult;
import org.photonvision.vision.pipe.impl.Calibrate3dPipe.CalibrationInput;
import org.photonvision.vision.pipe.impl.FindBoardCornersPipe.FindBoardCornersPipeResult; import org.photonvision.vision.pipe.impl.FindBoardCornersPipe.FindBoardCornersPipeResult;
import org.photonvision.vision.pipeline.CVPipeline; import org.photonvision.vision.pipeline.CVPipeline;
import org.photonvision.vision.pipeline.Calibration3dPipelineSettings; import org.photonvision.vision.pipeline.Calibration3dPipelineSettings;
@@ -176,7 +177,8 @@ public class Calibrate3dPipeline
/*Pass the board corners to the pipe, which will check again to see if all boards are valid /*Pass the board corners to the pipe, which will check again to see if all boards are valid
and returns the corresponding image and object points*/ and returns the corresponding image and object points*/
calibrationOutput = calibrate3dPipe.run(foundCornersList); calibrationOutput =
calibrate3dPipe.run(new CalibrationInput(foundCornersList, frameStaticProperties));
this.calibrating = false; this.calibrating = false;
@@ -229,4 +231,9 @@ public class Calibrate3dPipeline
public CameraCalibrationCoefficients cameraCalibrationCoefficients() { public CameraCalibrationCoefficients cameraCalibrationCoefficients() {
return calibrationOutput.output; return calibrationOutput.output;
} }
@Override
public void release() {
// we never actually need to give resources up since pipelinemanager only makes one of us
}
} }

View File

@@ -221,4 +221,11 @@ public class AprilTagPipeline extends CVPipeline<CVPipelineResult, AprilTagPipel
return new CVPipelineResult(sumPipeNanosElapsed, fps, targetList, multiTagResult, frame); return new CVPipelineResult(sumPipeNanosElapsed, fps, targetList, multiTagResult, frame);
} }
@Override
public void release() {
aprilTagDetectionPipe.release();
singleTagPoseEstimatorPipe.release();
super.release();
}
} }

View File

@@ -60,8 +60,8 @@ import org.photonvision.vision.target.TrackedTarget;
import org.photonvision.vision.target.TrackedTarget.TargetCalculationParameters; import org.photonvision.vision.target.TrackedTarget.TargetCalculationParameters;
public class ArucoPipeline extends CVPipeline<CVPipelineResult, ArucoPipelineSettings> { public class ArucoPipeline extends CVPipeline<CVPipelineResult, ArucoPipelineSettings> {
private final ArucoDetectionPipe arucoDetectionPipe = new ArucoDetectionPipe(); private ArucoDetectionPipe arucoDetectionPipe = new ArucoDetectionPipe();
private final ArucoPoseEstimatorPipe singleTagPoseEstimatorPipe = new ArucoPoseEstimatorPipe(); private ArucoPoseEstimatorPipe singleTagPoseEstimatorPipe = new ArucoPoseEstimatorPipe();
private final MultiTargetPNPPipe multiTagPNPPipe = new MultiTargetPNPPipe(); private final MultiTargetPNPPipe multiTagPNPPipe = new MultiTargetPNPPipe();
private final CalculateFPSPipe calculateFPSPipe = new CalculateFPSPipe(); private final CalculateFPSPipe calculateFPSPipe = new CalculateFPSPipe();
@@ -250,4 +250,13 @@ public class ArucoPipeline extends CVPipeline<CVPipelineResult, ArucoPipelineSet
windowSize, windowSize,
constant); constant);
} }
@Override
public void release() {
arucoDetectionPipe.release();
singleTagPoseEstimatorPipe.release();
arucoDetectionPipe = null;
singleTagPoseEstimatorPipe = null;
super.release();
}
} }

View File

@@ -34,6 +34,9 @@ public abstract class CVPipeline<R extends CVPipelineResult, S extends CVPipelin
private final FrameThresholdType thresholdType; private final FrameThresholdType thresholdType;
// So releaseable doesn't keep track of if we double-free something. so (ew) remember that here
protected volatile boolean released = false;
public CVPipeline(FrameThresholdType thresholdType) { public CVPipeline(FrameThresholdType thresholdType) {
this.thresholdType = thresholdType; this.thresholdType = thresholdType;
} }
@@ -64,6 +67,9 @@ public abstract class CVPipeline<R extends CVPipelineResult, S extends CVPipelin
} }
public R run(Frame frame, QuirkyCamera cameraQuirks) { public R run(Frame frame, QuirkyCamera cameraQuirks) {
if (released) {
throw new RuntimeException("Pipeline use-after-free!");
}
if (settings == null) { if (settings == null) {
throw new RuntimeException("No settings provided for pipeline!"); throw new RuntimeException("No settings provided for pipeline!");
} }
@@ -85,5 +91,7 @@ public abstract class CVPipeline<R extends CVPipelineResult, S extends CVPipelin
* switch. Stubbed out, but override if needed. * switch. Stubbed out, but override if needed.
*/ */
@Override @Override
public void release() {} public void release() {
released = true;
}
} }

View File

@@ -88,4 +88,9 @@ public class DriverModePipeline
fps, fps,
new Frame(frame.processedImage, frame.colorImage, frame.type, frame.frameStaticProperties)); new Frame(frame.processedImage, frame.colorImage, frame.type, frame.frameStaticProperties));
} }
@Override
public void release() {
// we never actually need to give resources up since pipelinemanager only makes one of us
}
} }

View File

@@ -131,5 +131,6 @@ public class ObjectDetectionPipeline
@Override @Override
public void release() { public void release() {
rknnPipe.release(); rknnPipe.release();
super.release();
} }
} }

View File

@@ -148,6 +148,7 @@ public class PipelineManager {
* @return The currently active pipeline. * @return The currently active pipeline.
*/ */
public CVPipeline getCurrentPipeline() { public CVPipeline getCurrentPipeline() {
updatePipelineFromRequested();
if (currentPipelineIndex < 0) { if (currentPipelineIndex < 0) {
switch (currentPipelineIndex) { switch (currentPipelineIndex) {
case CAL_3D_INDEX: case CAL_3D_INDEX:
@@ -170,6 +171,8 @@ public class PipelineManager {
return getPipelineSettings(currentPipelineIndex); return getPipelineSettings(currentPipelineIndex);
} }
private volatile int requestedIndex = 0;
/** /**
* Internal method for setting the active pipeline. <br> * Internal method for setting the active pipeline. <br>
* <br> * <br>
@@ -179,6 +182,22 @@ public class PipelineManager {
* @param newIndex Index of pipeline to be active * @param newIndex Index of pipeline to be active
*/ */
private void setPipelineInternal(int newIndex) { private void setPipelineInternal(int newIndex) {
requestedIndex = newIndex;
}
/**
* Based on a requested pipeline index, create/destroy pipelines as necessary. We do this as a
* side effect of the main thread that calls getCurrentPipeline to avoid race conditions between
* server threads and the VisionRunner TODO: this should be refactored. Shame Java doesn't have
* RAII
*/
private void updatePipelineFromRequested() {
int newIndex = requestedIndex;
if (newIndex == currentPipelineIndex) {
// nothing to do, probably no change -- give up
return;
}
if (newIndex < 0 && currentPipelineIndex >= 0) { if (newIndex < 0 && currentPipelineIndex >= 0) {
// Transitioning to a built-in pipe, save off the current user one // Transitioning to a built-in pipe, save off the current user one
lastUserPipelineIdx = currentPipelineIndex; lastUserPipelineIdx = currentPipelineIndex;
@@ -189,8 +208,8 @@ public class PipelineManager {
return; return;
} }
// Cleanup potential old native resources before swapping over // Cleanup potential old native resources before swapping over for user pipelines
if (currentUserPipeline != null) { if (currentUserPipeline != null && !(newIndex < 0)) {
currentUserPipeline.release(); currentUserPipeline.release();
} }

View File

@@ -277,7 +277,7 @@ public class Main {
CameraConfiguration camConf2024 = CameraConfiguration camConf2024 =
ConfigManager.getInstance().getConfig().getCameraConfigurations().get("WPI2024"); ConfigManager.getInstance().getConfig().getCameraConfigurations().get("WPI2024");
if (camConf2024 == null || true) { if (camConf2024 == null) {
camConf2024 = camConf2024 =
new CameraConfiguration( new CameraConfiguration(
"WPI2024", "WPI2024",