 MaSta
		
			MaSta
		
		
		
		
		
		
		
		
	
			11-15-2022 07:25 AM
Hello!
I searched here before asking this and similar topics were found, but they're only similar.
Problem: I start a sub VI from within my main VI. The sub VI runs a loop, controlled by buttons on the main VI front panel, which are referenced into the sub VI. When pressing STOP, the sub VI would still continue the code in the loop until the next iteration. This is not wanted, so my idea was to abort the sub VI via a method and its VI reference. LV denies this by saying like "The VI you are trying to stop is in an illegal state" and of course doesn't stop the VI and its loop immediately.
Do I need to set up something special in the VI properties? There are different options under "Execution", but I don't see the connection.
 JÞB
		
			JÞB
		
		
		
		
		
		
		
		
	
			11-15-2022 07:45 AM
Your consumer states are improperly defined.
Read about a trip to grandma's house here! Every state can be defined with a finite time period <200msec.
 paul_cardinale
		
			paul_cardinale
		
		
		
		
		
		
		
		
	
			11-15-2022 08:07 AM
If you use the "Call By Reference" function instead of a regular subVI call, then you can get away with an abort. But use caution.
 altenbach
		
			altenbach
		
		
		 
		
		
		
		
		
	
			11-15-2022 09:18 AM - edited 11-16-2022 10:51 AM
It is difficult to give specific advice without seeing the code, but your stop button will only get read once per while loop iteration and the loop cannot complete until everything in it has completed.
Can you explain what the loop is doing that prevents is from reading the stop condition at regular intervals? For example if there is a lengthy inner loop (FOR or while), the terminal would belong there. If it is waiting forever for some timeout, that needs to be broken. What does the subVI actually do? Does it really need to stop or would it be sufficient to just hide the panel?
I would guess that you have a relatively poor architecture and whatever you need the program to do could be done in much better and different ways.
Once you show us a simplified version of your code and tell us how the program should function as it is operated (step-by-step details!), I am sure we we can point you to a much better solution.
11-16-2022 04:32 AM
I made a simplified VI. example1 is the one that works, but doesn't exit the sub VI immediately, depending on where the code in the FOR loop currently is. example2 is supposed to exit the subVI anytime, but doesn't do it.
 Bob_Schor
		
			Bob_Schor
		
		
		 
		
		
		
		
		
	
			11-16-2022 10:26 AM
When you want to stop parallel loops, you need a way to "get inside the loop". In the "old days", there were "References" that could be passed, but they were clumsy, and did not maintain "data flow" (you had no "wire" for the reference). In LabVIEW 2016, the Asynchronous Channel Wire was introduced, and makes a very intuitive way to pass things "asynchronously" (here meaning "instantaneously") from one loop to another. Here's a simple example with a Stream Channel wire:
The Stream Channel Wire is created by a "Channel Writer", the rectangular item in the first Loop that takes the Boolean "Stop" control in and exports a "Green Pipe" -- green for Boolean, pipe as a metaphor for Asynchronous Channel Wire which passes over (not tunneling through) the "wall" of the While Loop. The second (receiving) Loop has a corresponding Channel Reader that takes the Boolean out of the Stream and sends it to the Stop Control of the second loop.
The Stream Channel Wire is like a Queue -- everything put in one end is sequentially read at the other end. If you run this code, you may be surprised at the results -- the second loop will not stop until it reads all the commands from the first Loop, namely "Go, Go, Go, ..., Stop", and as it is the slower loop, it finishes later, and shows it looped as many times as the first loop.
To get the two loops to stop at the same time, replace the Stream Channel Wire with a "Tag" Channel Wire, where the Reader only reads the most recent item (if any) that is passed in:
This behaves as you would expect -- the two loops stop at the same time (with the second loop showing fewer iterations).
I've put both VIs right here on the diagram. What if the second Loop was a sub-VI and I wanted to pass it the Channel Wire? Simple -- select the second Loop (including the Tag Channel Wire), go to the Edit Menu and choose "Create Sub-VI". Try that. You'll see that the sub-VI has a Channel Wire Reference as an input, so you can pass Channel Wires into and out of sub-VIs.
See if this works for you.
Bob Schor
11-16-2022 10:56 AM - edited 11-16-2022 10:59 AM
@Bob: Very clever way. Didn't know of the pipe. However, it doesn't help to end the sub VI. I also opened another thread where the topic was to end two loops at the same time. Did you perhaps want to answer there?
 altenbach
		
			altenbach
		
		
		 
		
		
		
		
		
	
			11-16-2022 12:28 PM
@MaSta wrote:
I made a simplified VI. example1 is the one that works, but doesn't exit the sub VI immediately, depending on where the code in the FOR loop currently is. example2 is supposed to exit the subVI anytime, but doesn't do it.
This code is too simple! What does the FOR loop actually do? If it does a hard calculation that takes 10 seconds, it needs to complete. If it just does something quick every 10 seconds, use an inner loop to poll the button. And why is is a FOR loop instead of a while loop? I really doubt that waiting for 100000x 10 second iterations (11+ days!!!!) for a natural termination of the loop is reasonable in any scenario.
Here's one possibility (might need some tweaks, depending on if you want the first task a 0s or 10s, etc.):
 paul_cardinale
		
			paul_cardinale
		
		
		
		
		
		
		
		
	
			11-16-2022 03:20 PM
How about this.
11-17-2022 02:23 AM
@altenbach wrote:
This code is too simple! What does the FOR loop actually do?
That's the point. It shouldn't matter what the FOR loop does. I know, I could also use a 5 ms loop timer, count 10 iterations to achieve a 50 ms timing and then execute the actual code which, perhaps, takes 5 ms to do everything, so that the FOR loop could end with a max. of 5 ms delay.
In my imagination, even if LV is dataflow based, I think that it would create a process, a task for a sub VI. Tasks can be killed programmatically.
I learned from the LV help that the method to abort the VI isn't for sub VIs, but for the main VI. So it's basically the same as the Stop VI.
The conclusion now is to do the counter method.