NI Vision Functions

I think this is best answered by one of the NI guys/gals, but anyone is more than welcome to step in (especially if you know what to do :P).

I was looking through the NIVision.java file, and saw many, many functions for lots of functionality. They look like C++ functions. Some of them are, from what I can tell, implemented in Java as blocking functions. However, the vast majority of them are not. I would like to call some of these functions. Before I start digging into implementing them myself, is there a reason they are not implemented?

My plan is to make my own task executer, and implement my own blocking function in the same style as the ones already implemented. They seem to follow a simple pattern of name and parameters to get from C++ to Java. Does anyone have experience doing this?

Thanks a bunch!

Those NI Vision functions are implemented by calling the NI C library. That way you get maximum performance on image transformations. Just be careful, it is allocating C memory for the various image classes and they have to be explicitly deallocated. Look at the documentation or sample programs to see how to do that.

Brad

Okay, just to clarify, the structure for calling them would be the same (generally speaking) as the functions that are already implemented? Make a blocking function, assign it to a task executer, then call it?

That is correct.

Alright, I got my first one working! I did the convex hull, since that will be useful to have, and didn’t require any new structures.

private static final BlockingFunction imaqConvexHullFn = NativeLibrary.getDefaultInstance().getBlockingFunction("imaqConvexHull");
    static { imaqConvexHullFn.setTaskExecutor(taskExecutor); }
    
    public static void convexHull(Pointer source, Pointer dest, int connectivity8)  throws NIVisionException{
        assertCleanStatus(imaqConvexHullFn.call3(source.address().toUWord().toPrimitive(), dest.address().toUWord().toPrimitive(), connectivity8));
    }

Only thing I am unsure of it that according to the NI docs, connectivity8 is a boolean, so I assume 0 for false (meaning use connectivity to the 4 adjacent pixels) and 1 for true (meaning use connectivity to the 8 adjacent pixels).

That should be correct.

Where would this code go? In NIVision.java or in your main robot class file?

If you want to modify the WPILib source code, it can go in the NIVision.java. However, each time the plugins get updated, I believe your changed will get removed. I just put it in my robot class, but for the final code it will go in its own class. To answer your question, it can go anywhere.

We try adding it to our own class and tried extending it. We are getting the following error. cannot find symbol variable imaqConvexHullFn

We imported import com.sun.cldc.jna.*; is there anything else we need to import or do we need to extend the class or something.

So you’ll notice that the first line declares the function “imaqConvexHullFn” as a function to access the C code. Do you have this line? Copy and pasting the relevant sections of code would make it easier.

We’ve been starting to work on camera-tracking code, and we ran into some issues. We thought that it might be a good idea to try the code that was posted here first, and make sure we could get that much to work. This is what we have so far:


public class Camera2012 {

    public AxisCamera camera = AxisCamera.getInstance();
    
    private static final BlockingFunction imaqConvexHullFn = NativeLibrary.getDefaultInstance().getBlockingFunction("imaqConvexHull");
        static { imaqConvexHullFn.setTaskExecutor(taskExecutor); }
    
    public static void convexHull(Pointer source, Pointer dest, int connectivity8) throws NIVisionException {
        assertCleanStatus(imaqConvexHullFn.call3(source.address().toUWord().toPrimitive(), dest.address().toUWord().toPrimitive(), connectivity8));
    }
}

We get the following errors:

taskExecutor: cannot find symbol
assertCleanStatus: cannot find symbol

My own experience with Java is limited; could you help us out with this? If you happen to notice anything that might be helpful, we’ll take all suggestions.

Thanks so much!

Essentially, this means you never created the taskExecutor object. The simplest way (I think) to use that code is to copy NIVision.java to your project, and add Dan’s code into that.

You need to instantiate your own TaskExecutor and pass it to setTaskExecutor. assertCleanStatus is a protected method of the NIVision class, which checks if its argument is nonzero and if so throws an NIVisionException. Why it is protected, I don’t know; you can’t extend the classs because the only constructor is private. Just make your own, or copy+paste the source code from NIVision.java.

Is there any documentation for the imaq functions C-side? Specifically, I would like to use imaqDetectRectangles and need the fields of the RectangleMatch struct.

Look in C:\Program Files\National Instruments\Vision\Documentation\NIVisionCVI.chm

-Joe

The way I did it was to make a new executer, and copy and past the assertCleanStatus function from the NIVision.java.

Thanks so much for the tips - everything runs error-free now.

Would it be possible to see an example of convexHull() being properly called (for an image which I’ve already used an HSL threshold on)?

Sure, here is an example of how I call it:

convexHull(binaryImage.image,binaryImage.image,1);

That will find the convex hull of binaryImage and put it in binaryImage

You may want to check out this post. They have wrapped it into the library as of today. http://www.chiefdelphi.com/forums/showthread.php?t=100792

Yeah, but there are still many many functions upwrapped :stuck_out_tongue:

What would you like to see? Please be specific.

Brad