| """This test case provides support for checking forking and wait behavior. |
| |
| To test different wait behavior, override the wait_impl method. |
| |
| We want fork1() semantics -- only the forking thread survives in the |
| child after a fork(). |
| |
| On some systems (e.g. Solaris without posix threads) we find that all |
| active threads survive in the child after a fork(); this is an error. |
| |
| While BeOS doesn't officially support fork and native threading in |
| the same application, the present example should work just fine. DC |
| """ |
| |
| import os, sys, time, unittest |
| import test.test_support as test_support |
| thread = test_support.import_module('thread') |
| |
| LONGSLEEP = 2 |
| SHORTSLEEP = 0.5 |
| NUM_THREADS = 4 |
| |
| class ForkWait(unittest.TestCase): |
| |
| def setUp(self): |
| self.alive = {} |
| self.stop = 0 |
| |
| def f(self, id): |
| while not self.stop: |
| self.alive[id] = os.getpid() |
| try: |
| time.sleep(SHORTSLEEP) |
| except IOError: |
| pass |
| |
| def wait_impl(self, cpid): |
| for i in range(10): |
| # waitpid() shouldn't hang, but some of the buildbots seem to hang |
| # in the forking tests. This is an attempt to fix the problem. |
| spid, status = os.waitpid(cpid, os.WNOHANG) |
| if spid == cpid: |
| break |
| time.sleep(2 * SHORTSLEEP) |
| |
| self.assertEqual(spid, cpid) |
| self.assertEqual(status, 0, "cause = %d, exit = %d" % (status&0xff, status>>8)) |
| |
| def test_wait(self): |
| for i in range(NUM_THREADS): |
| thread.start_new(self.f, (i,)) |
| |
| time.sleep(LONGSLEEP) |
| |
| a = self.alive.keys() |
| a.sort() |
| self.assertEqual(a, range(NUM_THREADS)) |
| |
| prefork_lives = self.alive.copy() |
| |
| if sys.platform in ['unixware7']: |
| cpid = os.fork1() |
| else: |
| cpid = os.fork() |
| |
| if cpid == 0: |
| # Child |
| time.sleep(LONGSLEEP) |
| n = 0 |
| for key in self.alive: |
| if self.alive[key] != prefork_lives[key]: |
| n += 1 |
| os._exit(n) |
| else: |
| # Parent |
| self.wait_impl(cpid) |
| # Tell threads to die |
| self.stop = 1 |
| time.sleep(2*SHORTSLEEP) # Wait for threads to die |