02-06-2014 10:47 AM
After switching from TestStand 4.1 to 2012 I see an interesting problem.
I start a MainSequence through the SinglePass execution entry point using the parallel model. In ProcessSetup (in execution N) I start a background thread that does some visualization tasks. Then the model goes through "Initialize TestSockets" and starts my MainSequence (in execution N+1).
While the MainSequence is running, execution N hangs in ParallelModel.seq>Single Pass, Step "Wait for TestSockets", as it should be. Usually, when the MainSequence ends, the execution terminates and the execution N advances to the next step "Check for Terminate and Abort" and some time later it runs through ProcessCleanup -- where I would send my background thread a notification to terminate itself.
That works as long as I do not start that background thread. But when this thread is running, execution N+1 never leaves the MainSequence. I reach a break point at the end of MainSequence, advance one step, and then all executions are happily displaying a green light and keep on running. So execution N never leaves "Wait for TestSockets" and never reaches ProcessCleanup, so my thread does not get the termination signal etc.
But I remember distinctly that this worked in TestStand 4.1 and anyway, I do not understand this. Why does a background thread, started in execution N, prevent execution N+1 from terminating?
Regards
Peter
Solved! Go to Solution.
02-06-2014 11:53 AM
In your Sequence Call Advanced Settings for the sequence call that is spawning the thread, what is Automatically Wait for the Thread to Complete at the End of the Current Sequence set to?
Maybe?
02-06-2014 11:58 AM
I rephrase my question:
When an execution starts a sequence in a new thread (not waiting for the thread to complete at the end of the sequence), does it have to take care at the end of its MainSequence that the thread is ordered to end itself in order for the execution to terminate?
The reason to ask the question thus is that after two days of puzzling I just now stumbled over the solution and do not quite understand what is happening. Therefore, some more background detail:
The situation is as follows:
At that point, execution N+1 (Worker) has ended, execution N (SinglePass) and N+1 (Socket) run on, never terminating. Both HMI_Loop-Threads are still running and - which may be crucial - they are using the same server DLL (not quite OPC, but similar).
If I introduce a signal that MainSequence sets when it is about to end and which HMI_Loop2 listens to to end its loop, then everything works. HMI_Loop2 ends, MainSequence ends, thereby ending execution N+2 and finally HMI_Loop1 receives its stop signal from ProcessCleanup and all is well.
What I do not understand is: HMI_Loop2 is started with ThreadTerminationOption_Normal, so I assumed, when MainSequence reaches its end, it would terminate its execution and thereby the HMI_Loop2-Thread. This never happened.
It was always possible, though, to stop everything by "Debug > TerminateAll", so I assume that MainSequence reaching its end did not terminate the Socket execution by itself, because it "knew" that there was still a thread running under its control. And therefore my question on top.
Regards
Peter
02-06-2014 11:59 AM
It is set to "not wait" because this is started in an underlying sequence call which would otherwise block the rest of the execution.
Thanks
Peter
02-07-2014 10:00 AM - edited 02-07-2014 10:05 AM
"When an execution starts a sequence in a new thread (not waiting for the thread to complete at the end of the sequence), does it have to take care at the end of its MainSequence that the thread is ordered to end itself in order for the execution to terminate?"
What do you mean by terminate? The process models do not normally terminate any executions. Do you mean, "Do all threads in an execution have to complete before an execution completes?"? If so then the answer is yes.
I don't completely follow your explanation above. I'm not sure what you mean by signalling, but I think you are misunderstanding what terminate means in TestStand. Termination only happens when either a user explicitly requests an execution to terminate (e.g. terminate all), or your sequence has a post action to terminate or programmatically initiates a terminate. Without termination explicitly happening, executions normally complete when all of their threads are finished executing. If you are spawning new threads and executions, you need to come up with a mechanism for them to know when they should exit. I do NOT recommend relying on or using termination for that. Termination is like aborting (but with cleanup), it's not intended to be something that happens as part of the normal flow of execution. There are many ways to tell your worker threads when to complete. Perhaps this signaling your are referring to is one way. You could also use a teststand notification step, or a boolean by reference parameter.
Hope this helps clarify things,
-Doug
02-10-2014 06:37 AM
I was probably not quite clear in my explanation. You understood my question correctly, and so the answer is, as I assumed, "yes".
I did indeed use the "terminate" incorrectly, referring to a termination by the user or an API call as well as to the natural completion of an execution when all its sequences / threads end. And by signalling I meant exactly that, using a notification or a flag monitored by the thread to tell it, when it should end its operation.
So thank you, that clarified things and helped me find a correct solution.
Peter