mirror of
				https://github.com/python/cpython.git
				synced 2025-10-26 19:24:34 +00:00 
			
		
		
		
	Issue #2489: Fix bug in _copy loop that could consume 100% cpu on EOF.
This commit is contained in:
		
							parent
							
								
									5c724a804a
								
							
						
					
					
						commit
						05f5953ab4
					
				
					 2 changed files with 101 additions and 6 deletions
				
			
		
							
								
								
									
										12
									
								
								Lib/pty.py
									
										
									
									
									
								
							
							
						
						
									
										12
									
								
								Lib/pty.py
									
										
									
									
									
								
							|  | @ -142,14 +142,20 @@ def _copy(master_fd, master_read=_read, stdin_read=_read): | ||||||
|     Copies |     Copies | ||||||
|             pty master -> standard output   (master_read) |             pty master -> standard output   (master_read) | ||||||
|             standard input -> pty master    (stdin_read)""" |             standard input -> pty master    (stdin_read)""" | ||||||
|     while 1: |     fds = [master_fd, STDIN_FILENO] | ||||||
|         rfds, wfds, xfds = select( |     while True: | ||||||
|                 [master_fd, STDIN_FILENO], [], []) |         rfds, wfds, xfds = select(fds, [], []) | ||||||
|         if master_fd in rfds: |         if master_fd in rfds: | ||||||
|             data = master_read(master_fd) |             data = master_read(master_fd) | ||||||
|  |             if not data:  # Reached EOF. | ||||||
|  |                 fds.remove(master_fd) | ||||||
|  |             else: | ||||||
|                 os.write(STDOUT_FILENO, data) |                 os.write(STDOUT_FILENO, data) | ||||||
|         if STDIN_FILENO in rfds: |         if STDIN_FILENO in rfds: | ||||||
|             data = stdin_read(STDIN_FILENO) |             data = stdin_read(STDIN_FILENO) | ||||||
|  |             if not data: | ||||||
|  |                 fds.remove(STDIN_FILENO) | ||||||
|  |             else: | ||||||
|                 _writen(master_fd, data) |                 _writen(master_fd, data) | ||||||
| 
 | 
 | ||||||
| def spawn(argv, master_read=_read, stdin_read=_read): | def spawn(argv, master_read=_read, stdin_read=_read): | ||||||
|  |  | ||||||
|  | @ -8,7 +8,9 @@ | ||||||
| import pty | import pty | ||||||
| import os | import os | ||||||
| import sys | import sys | ||||||
|  | import select | ||||||
| import signal | import signal | ||||||
|  | import socket | ||||||
| import unittest | import unittest | ||||||
| 
 | 
 | ||||||
| TEST_STRING_1 = b"I wish to buy a fish license.\n" | TEST_STRING_1 = b"I wish to buy a fish license.\n" | ||||||
|  | @ -194,9 +196,96 @@ def test_fork(self): | ||||||
| 
 | 
 | ||||||
|         # pty.fork() passed. |         # pty.fork() passed. | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  | class SmallPtyTests(unittest.TestCase): | ||||||
|  |     """These tests don't spawn children or hang.""" | ||||||
|  | 
 | ||||||
|  |     def setUp(self): | ||||||
|  |         self.orig_stdin_fileno = pty.STDIN_FILENO | ||||||
|  |         self.orig_stdout_fileno = pty.STDOUT_FILENO | ||||||
|  |         self.orig_pty_select = pty.select | ||||||
|  |         self.fds = []  # A list of file descriptors to close. | ||||||
|  |         self.select_rfds_lengths = [] | ||||||
|  |         self.select_rfds_results = [] | ||||||
|  | 
 | ||||||
|  |     def tearDown(self): | ||||||
|  |         pty.STDIN_FILENO = self.orig_stdin_fileno | ||||||
|  |         pty.STDOUT_FILENO = self.orig_stdout_fileno | ||||||
|  |         pty.select = self.orig_pty_select | ||||||
|  |         for fd in self.fds: | ||||||
|  |             try: | ||||||
|  |                 os.close(fd) | ||||||
|  |             except: | ||||||
|  |                 pass | ||||||
|  | 
 | ||||||
|  |     def _pipe(self): | ||||||
|  |         pipe_fds = os.pipe() | ||||||
|  |         self.fds.extend(pipe_fds) | ||||||
|  |         return pipe_fds | ||||||
|  | 
 | ||||||
|  |     def _mock_select(self, rfds, wfds, xfds): | ||||||
|  |         # This will raise IndexError when no more expected calls exist. | ||||||
|  |         self.assertEqual(self.select_rfds_lengths.pop(0), len(rfds)) | ||||||
|  |         return self.select_rfds_results.pop(0), [], [] | ||||||
|  | 
 | ||||||
|  |     def test__copy_to_each(self): | ||||||
|  |         """Test the normal data case on both master_fd and stdin.""" | ||||||
|  |         read_from_stdout_fd, mock_stdout_fd = self._pipe() | ||||||
|  |         pty.STDOUT_FILENO = mock_stdout_fd | ||||||
|  |         mock_stdin_fd, write_to_stdin_fd = self._pipe() | ||||||
|  |         pty.STDIN_FILENO = mock_stdin_fd | ||||||
|  |         socketpair = socket.socketpair() | ||||||
|  |         masters = [s.fileno() for s in socketpair] | ||||||
|  |         self.fds.extend(masters) | ||||||
|  | 
 | ||||||
|  |         # Feed data.  Smaller than PIPEBUF.  These writes will not block. | ||||||
|  |         os.write(masters[1], b'from master') | ||||||
|  |         os.write(write_to_stdin_fd, b'from stdin') | ||||||
|  | 
 | ||||||
|  |         # Expect two select calls, the last one will cause IndexError | ||||||
|  |         pty.select = self._mock_select | ||||||
|  |         self.select_rfds_lengths.append(2) | ||||||
|  |         self.select_rfds_results.append([mock_stdin_fd, masters[0]]) | ||||||
|  |         self.select_rfds_lengths.append(2) | ||||||
|  | 
 | ||||||
|  |         with self.assertRaises(IndexError): | ||||||
|  |             pty._copy(masters[0]) | ||||||
|  | 
 | ||||||
|  |         # Test that the right data went to the right places. | ||||||
|  |         rfds = select.select([read_from_stdout_fd, masters[1]], [], [], 0)[0] | ||||||
|  |         self.assertSameElements([read_from_stdout_fd, masters[1]], rfds) | ||||||
|  |         self.assertEqual(os.read(read_from_stdout_fd, 20), b'from master') | ||||||
|  |         self.assertEqual(os.read(masters[1], 20), b'from stdin') | ||||||
|  | 
 | ||||||
|  |     def test__copy_eof_on_all(self): | ||||||
|  |         """Test the empty read EOF case on both master_fd and stdin.""" | ||||||
|  |         read_from_stdout_fd, mock_stdout_fd = self._pipe() | ||||||
|  |         pty.STDOUT_FILENO = mock_stdout_fd | ||||||
|  |         mock_stdin_fd, write_to_stdin_fd = self._pipe() | ||||||
|  |         pty.STDIN_FILENO = mock_stdin_fd | ||||||
|  |         socketpair = socket.socketpair() | ||||||
|  |         masters = [s.fileno() for s in socketpair] | ||||||
|  |         self.fds.extend(masters) | ||||||
|  | 
 | ||||||
|  |         os.close(masters[1]) | ||||||
|  |         socketpair[1].close() | ||||||
|  |         os.close(write_to_stdin_fd) | ||||||
|  | 
 | ||||||
|  |         # Expect two select calls, the last one will cause IndexError | ||||||
|  |         pty.select = self._mock_select | ||||||
|  |         self.select_rfds_lengths.append(2) | ||||||
|  |         self.select_rfds_results.append([mock_stdin_fd, masters[0]]) | ||||||
|  |         # We expect that both fds were removed from the fds list as they | ||||||
|  |         # both encountered an EOF before the second select call. | ||||||
|  |         self.select_rfds_lengths.append(0) | ||||||
|  | 
 | ||||||
|  |         with self.assertRaises(IndexError): | ||||||
|  |             pty._copy(masters[0]) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| def test_main(verbose=None): | def test_main(verbose=None): | ||||||
|     try: |     try: | ||||||
|         run_unittest(PtyTest) |         run_unittest(SmallPtyTests, PtyTest) | ||||||
|     finally: |     finally: | ||||||
|         reap_children() |         reap_children() | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Gregory P. Smith
						Gregory P. Smith