199451b44SJordan Rupprecht"""Test that we handle inferiors which change their process group""" 299451b44SJordan Rupprecht 399451b44SJordan Rupprecht 499451b44SJordan Rupprecht 599451b44SJordan Rupprechtimport os 699451b44SJordan Rupprechtimport lldb 799451b44SJordan Rupprechtfrom lldbsuite.test.decorators import * 899451b44SJordan Rupprechtfrom lldbsuite.test.lldbtest import * 999451b44SJordan Rupprechtfrom lldbsuite.test import lldbutil 1099451b44SJordan Rupprecht 1199451b44SJordan Rupprecht 1299451b44SJordan Rupprechtclass ChangeProcessGroupTestCase(TestBase): 1399451b44SJordan Rupprecht 1499451b44SJordan Rupprecht mydir = TestBase.compute_mydir(__file__) 1599451b44SJordan Rupprecht NO_DEBUG_INFO_TESTCASE = True 1699451b44SJordan Rupprecht 1799451b44SJordan Rupprecht def setUp(self): 1899451b44SJordan Rupprecht # Call super's setUp(). 1999451b44SJordan Rupprecht TestBase.setUp(self) 2099451b44SJordan Rupprecht # Find the line number to break for main.c. 2199451b44SJordan Rupprecht self.line = line_number('main.c', '// Set breakpoint here') 2299451b44SJordan Rupprecht 2399451b44SJordan Rupprecht @skipIfFreeBSD # Times out on FreeBSD llvm.org/pr23731 2499451b44SJordan Rupprecht @skipIfWindows # setpgid call does not exist on Windows 2599451b44SJordan Rupprecht @expectedFailureAndroid("http://llvm.org/pr23762", api_levels=[16]) 2699451b44SJordan Rupprecht @expectedFailureNetBSD 275238b800SJonas Devlieghere @skipIfReproducer # File synchronization is not supported during replay. 28*239f488fSRaphael Isemann @skipIftvOS # fork not available on tvOS. 29*239f488fSRaphael Isemann @skipIfwatchOS # fork not available on watchOS. 3099451b44SJordan Rupprecht def test_setpgid(self): 3199451b44SJordan Rupprecht self.build() 3299451b44SJordan Rupprecht exe = self.getBuildArtifact("a.out") 3399451b44SJordan Rupprecht 3499451b44SJordan Rupprecht # Use a file as a synchronization point between test and inferior. 3599451b44SJordan Rupprecht pid_file_path = lldbutil.append_to_process_working_directory(self, 3699451b44SJordan Rupprecht "pid_file_%d" % (int(time.time()))) 3799451b44SJordan Rupprecht self.addTearDownHook( 3899451b44SJordan Rupprecht lambda: self.run_platform_command( 3999451b44SJordan Rupprecht "rm %s" % 4099451b44SJordan Rupprecht (pid_file_path))) 4199451b44SJordan Rupprecht 4299451b44SJordan Rupprecht popen = self.spawnSubprocess(exe, [pid_file_path]) 4399451b44SJordan Rupprecht 4499451b44SJordan Rupprecht pid = lldbutil.wait_for_file_on_target(self, pid_file_path) 4599451b44SJordan Rupprecht 4699451b44SJordan Rupprecht # make sure we cleanup the forked child also 4799451b44SJordan Rupprecht def cleanupChild(): 4899451b44SJordan Rupprecht if lldb.remote_platform: 4999451b44SJordan Rupprecht lldb.remote_platform.Kill(int(pid)) 5099451b44SJordan Rupprecht else: 5199451b44SJordan Rupprecht if os.path.exists("/proc/" + pid): 5299451b44SJordan Rupprecht os.kill(int(pid), signal.SIGKILL) 5399451b44SJordan Rupprecht self.addTearDownHook(cleanupChild) 5499451b44SJordan Rupprecht 5599451b44SJordan Rupprecht # Create a target by the debugger. 5699451b44SJordan Rupprecht target = self.dbg.CreateTarget(exe) 5799451b44SJordan Rupprecht self.assertTrue(target, VALID_TARGET) 5899451b44SJordan Rupprecht 5999451b44SJordan Rupprecht listener = lldb.SBListener("my.attach.listener") 6099451b44SJordan Rupprecht error = lldb.SBError() 6199451b44SJordan Rupprecht process = target.AttachToProcessWithID(listener, int(pid), error) 6299451b44SJordan Rupprecht self.assertTrue(error.Success() and process, PROCESS_IS_VALID) 6399451b44SJordan Rupprecht 6499451b44SJordan Rupprecht # set a breakpoint just before the setpgid() call 6599451b44SJordan Rupprecht lldbutil.run_break_set_by_file_and_line( 6699451b44SJordan Rupprecht self, 'main.c', self.line, num_expected_locations=-1) 6799451b44SJordan Rupprecht 6899451b44SJordan Rupprecht thread = process.GetSelectedThread() 6999451b44SJordan Rupprecht 7099451b44SJordan Rupprecht # release the child from its loop 7199451b44SJordan Rupprecht value = thread.GetSelectedFrame().EvaluateExpression("release_child_flag = 1") 72b3a0c4d7SRaphael Isemann self.assertTrue(value.IsValid()) 73b3a0c4d7SRaphael Isemann self.assertEquals(value.GetValueAsUnsigned(0), 1) 7499451b44SJordan Rupprecht process.Continue() 7599451b44SJordan Rupprecht 7699451b44SJordan Rupprecht # make sure the child's process group id is different from its pid 7799451b44SJordan Rupprecht value = thread.GetSelectedFrame().EvaluateExpression("(int)getpgid(0)") 7899451b44SJordan Rupprecht self.assertTrue(value.IsValid()) 7999451b44SJordan Rupprecht self.assertNotEqual(value.GetValueAsUnsigned(0), int(pid)) 8099451b44SJordan Rupprecht 8199451b44SJordan Rupprecht # step over the setpgid() call 8299451b44SJordan Rupprecht thread.StepOver() 8399451b44SJordan Rupprecht self.assertEqual(thread.GetStopReason(), lldb.eStopReasonPlanComplete) 8499451b44SJordan Rupprecht 8599451b44SJordan Rupprecht # verify that the process group has been set correctly 8699451b44SJordan Rupprecht # this also checks that we are still in full control of the child 8799451b44SJordan Rupprecht value = thread.GetSelectedFrame().EvaluateExpression("(int)getpgid(0)") 8899451b44SJordan Rupprecht self.assertTrue(value.IsValid()) 8999451b44SJordan Rupprecht self.assertEqual(value.GetValueAsUnsigned(0), int(pid)) 9099451b44SJordan Rupprecht 9199451b44SJordan Rupprecht # run to completion 9299451b44SJordan Rupprecht process.Continue() 9399451b44SJordan Rupprecht self.assertEqual(process.GetState(), lldb.eStateExited) 94