Well now I have just put my thread code into a critical section and then in the CellTextPaint event of the listbox, I try to enter that CriticalSection.
So the code in the CellTextPaint event should not run if the thread is running…
So more crash reports. Now with the CellTextPaint Event being restricted from painting while the thread is running (and I can see the effect of this in the listbox FYI), I am still crashing. Here’s where it gets interesting…
Here’s the first part of the first crash report I had:
Look where it crashed - at a wait. Looks like the thread was wanting to start its thing, the listbox was doing its stuff and so then the thread was asked to wait - it crashed. This was using a critical section and yes, the type was set to preemptive.
Just had another crash. This one I could actually catch and get a stack dump. Same thing. But it always seems to happen during an unlock. Maybe that is the culprit…
I think I may have multiple issues here. The ThreadAccessingUI exception part seems to have been resolved by making a copy of the image created in the threads before painting it to the canvases in the main thread.
But then I still keep getting random crashes and I’m not entirely certain. A number of the crash reports show a crash in com.apple.CFNetwork.Connection
I think this is happening due to some sort of collision or something if I reset the URLConnection or TCPSocket. When I make the send request on the network object I start a timer. In the received event I turn the timer off. If the timer ends up running it resets the socket. I think I may not be letting the framework have enough time to close and destroy the socket before recreating it. Hard to tell but this is my best guess. I can do this reset manually from my UI and I have never seen it crash. However, if there are multiple sockets at once that need to be reset, I can see something like this possibly happening. And this is all on the main thread by the way, so it’s not a thread issue.
It’s just really hard to debug and figure out. My app ran all night until 6:45 AM this morning when it crashed. Makes debugging difficult when something has to run for hours before you have an event and then it’s still hard to tell. This morning’s crash was on the main thread during a canvas draw. No network thread crash. Wish we had a way to get better description of what’s happening in these crash reports…
Out of curiosity, for those of us on the sidelines wondering about the value (time overhead + debugging headaches vs doing things the old-fashioned way (single core), what is your expected performance bump?
I understand you can run an A/B test until you get the “B” part (pre-emptive routines) working. Did you have an expected performance return before embarking on your journey?
I think the performance improvement is significant. Previously with capturing too many images, the UI would just start lagging. Things just are smoother. If I can figure out some of these issues, then yes, it will be well worth it.
It seems like the issue I am having right now is not even related to preemptive threads as it is an OS X framework thread that is crashing. I’m trying to figure out if there is anything in my code or network objects that could be causing this.
And truthfully, this sort of issue could have happened previously before I went to preemptive threading. I don’t really know if I ever ran the routines as long as I have been running them now in testing. It is a major challenge trying to figure out what you are doing wrong by reading crash reports and having to let you app run for hours and hours before it crashes.
Potentially very significant, but there are some issues to be worked out - you can read the bad news (very slow in 3.1) and good news (possibly has already been fixed for next release) here: Xojo1BRC (Xojo One Billion Row Challenge) - #38 by Kem_Tekinay
Yes, this is a work in progress. Step one was getting preemptive threads to work at all in a usable way, and that went way quicker than (I think) anyone imagined.
Now it’s dealing with the inevitable issues that come along with such a huge change, including performance bottlenecks.
If/when that’s addressed, I’d expect a dividable task that takes n time would go take something more than n/c when using c cores. (It likely won’t be n/c because of the processing overhead.)
You guys make great points in terms of raw data processing power. But I think even now preemptive threads work much better than cooperative threading.
In my app here, prior to using preemptive threads the maximum data rate I would see when pulling images was maybe about 60 Mb/s across the network. Now I am getting nearly 10x that rate.
I should try doing the same thing but using cooperative threads. My original code was originally running all on the main thread but using timers.
Sorry, should have been more explicit. Start and Stopping of timers, resetting the period etc. If a lot of things are using the same timer then with preemptive threads can end up with the timers in odd states. One item could have something like:
MyTimer.Interval = 300
MyTimer.Mode = Multiple
Another part of the code could:
MyTimer.Interval = 200
MyTimer.Mode = Stop
In a permeative world you could end up with the following sequence:
For example, down to threads executing in a meshed sequence. The multiple properties need to be an atomic operation. So you either end up with the first pair running to completion or the second pair running to completion.
Why would you only have one timer? My app uses sockets, and I subclass SSL Socket for that. One of the subclass properties is a Timer, another is the thread that will own the socket. Thus everything is nicely encpsulated and the thread is nor going to be interfering with properties used by another thread.