For some of these events, I'm trying to use the threading features and minimize busy delays, and the os_thread_delay_until() function is suggested and works great.
For some threads, the delay may be quite long, 5-10 minutes; is there a way to force the os_thread_delay() function to return early? With my "normal" code I can manually reset last_thread_time = 0 from another function/thread and this will cause it to run do_thing(). I tried doing a similar behavior to the above and it's not behaving as I expect. Is this normal? Any way around it?
I think the answer is probably no, but I also noticed something:
The threading explainer is wrong about the first parameter to os_thread_delay_until() which is a thin layer over vTaskDelayUntil(). It should not be zero on first invocation, but rather xTaskGetTickCount() according to the FreeRTOS docs. I think it works because on first invocation 0 will cause the wake time to be in the past in most cases, which causes it to not block but still set the last wake time, so your code will execute twice the first time but then function normally.
I suspect you can't just change the last wake time parameter and have the change take effect, and there does not appear to be call to terminate the wait.
However, I the overhead for swapping a thread to check for wait is so small that it's not really worth optimizing for, as long as you yield CPU using delay(1) or similar. This is probably the best solution if you have a variable wait.
It's not that much overhead because you're just doing a test and a delay on each millisecond timeslice, 1000 times per second. Presumably you don't have that mans threads, so the overhead should be small.
However, presumably you don't need your send interval accurate to the nearest millisecond, so you could delay for a longer period, which would lower the overhead more, and still provide a way to wait until the send interval is complete, or force sending early. That's probably what I would do.
Another way to do it would be to halt the uploader_thread_function with a mutex. You then release the mutex either from the force function, or from some other timer. You could use a software timer to trigger the mutex to release, for example.