blob: ae965d8242ba9fb59dfdc816a779b476cb1f9b99 [file] [log] [blame]
Antoine Pitrouf4f61712010-09-07 16:34:47 +00001import array
Michael W. Hudsonf0089982003-03-03 12:29:42 +00002import unittest
R. David Murray3db8a342009-03-30 23:05:48 +00003from test.test_support import run_unittest, import_module, get_attribute
Michael W. Hudsonf0089982003-03-03 12:29:42 +00004import os, struct
R. David Murray3db8a342009-03-30 23:05:48 +00005fcntl = import_module('fcntl')
6termios = import_module('termios')
7get_attribute(termios, 'TIOCGPGRP') #Can't run tests without this feature
Michael W. Hudsonf0089982003-03-03 12:29:42 +00008
Neal Norwitz26f42f62003-03-20 04:33:16 +00009try:
10 tty = open("/dev/tty", "r")
Neal Norwitz26f42f62003-03-20 04:33:16 +000011except IOError:
Benjamin Petersonbec087f2009-03-26 21:10:30 +000012 raise unittest.SkipTest("Unable to open /dev/tty")
Florent Xicluna172e15f2010-08-09 20:02:00 +000013else:
14 # Skip if another process is in foreground
15 r = fcntl.ioctl(tty, termios.TIOCGPGRP, " ")
16 tty.close()
17 rpgrp = struct.unpack("i", r)[0]
18 if rpgrp not in (os.getpgrp(), os.getsid(0)):
19 raise unittest.SkipTest("Neither the process group nor the session "
20 "are attached to /dev/tty")
21 del tty, r, rpgrp
Neal Norwitz26f42f62003-03-20 04:33:16 +000022
Gregory P. Smith6af3db82008-03-20 05:41:53 +000023try:
24 import pty
25except ImportError:
26 pty = None
27
Michael W. Hudsonf0089982003-03-03 12:29:42 +000028class IoctlTests(unittest.TestCase):
29 def test_ioctl(self):
Guido van Rossum8ee3e5a2005-09-14 18:09:42 +000030 # If this process has been put into the background, TIOCGPGRP returns
31 # the session ID instead of the process group id.
32 ids = (os.getpgrp(), os.getsid(0))
Michael W. Hudsonf0089982003-03-03 12:29:42 +000033 tty = open("/dev/tty", "r")
34 r = fcntl.ioctl(tty, termios.TIOCGPGRP, " ")
Guido van Rossum8ee3e5a2005-09-14 18:09:42 +000035 rpgrp = struct.unpack("i", r)[0]
Ezio Melottiaa980582010-01-23 23:04:36 +000036 self.assertIn(rpgrp, ids)
Michael W. Hudsonf0089982003-03-03 12:29:42 +000037
Antoine Pitrouf4f61712010-09-07 16:34:47 +000038 def _check_ioctl_mutate_len(self, nbytes=None):
39 buf = array.array('i')
40 intsize = buf.itemsize
Guido van Rossum8ee3e5a2005-09-14 18:09:42 +000041 ids = (os.getpgrp(), os.getsid(0))
Antoine Pitrouf4f61712010-09-07 16:34:47 +000042 # A fill value unlikely to be in `ids`
43 fill = -12345
44 if nbytes is not None:
45 # Extend the buffer so that it is exactly `nbytes` bytes long
46 buf.extend([fill] * (nbytes // intsize))
47 self.assertEqual(len(buf) * intsize, nbytes) # sanity check
48 else:
49 buf.append(fill)
50 with open("/dev/tty", "r") as tty:
51 r = fcntl.ioctl(tty, termios.TIOCGPGRP, buf, 1)
Guido van Rossum8ee3e5a2005-09-14 18:09:42 +000052 rpgrp = buf[0]
Ezio Melotti2623a372010-11-21 13:34:58 +000053 self.assertEqual(r, 0)
Ezio Melottiaa980582010-01-23 23:04:36 +000054 self.assertIn(rpgrp, ids)
Michael W. Hudsonf0089982003-03-03 12:29:42 +000055
Antoine Pitrouf4f61712010-09-07 16:34:47 +000056 def test_ioctl_mutate(self):
57 self._check_ioctl_mutate_len()
58
59 def test_ioctl_mutate_1024(self):
60 # Issue #9758: a mutable buffer of exactly 1024 bytes wouldn't be
61 # copied back after the system call.
62 self._check_ioctl_mutate_len(1024)
63
64 def test_ioctl_mutate_2048(self):
65 # Test with a larger buffer, just for the record.
66 self._check_ioctl_mutate_len(2048)
67
Gregory P. Smith6af3db82008-03-20 05:41:53 +000068 def test_ioctl_signed_unsigned_code_param(self):
69 if not pty:
Benjamin Petersonbec087f2009-03-26 21:10:30 +000070 raise unittest.SkipTest('pty module required')
Gregory P. Smith6af3db82008-03-20 05:41:53 +000071 mfd, sfd = pty.openpty()
72 try:
73 if termios.TIOCSWINSZ < 0:
74 set_winsz_opcode_maybe_neg = termios.TIOCSWINSZ
75 set_winsz_opcode_pos = termios.TIOCSWINSZ & 0xffffffffL
76 else:
77 set_winsz_opcode_pos = termios.TIOCSWINSZ
78 set_winsz_opcode_maybe_neg, = struct.unpack("i",
79 struct.pack("I", termios.TIOCSWINSZ))
80
Gregory P. Smith6af3db82008-03-20 05:41:53 +000081 our_winsz = struct.pack("HHHH",80,25,0,0)
82 # test both with a positive and potentially negative ioctl code
83 new_winsz = fcntl.ioctl(mfd, set_winsz_opcode_pos, our_winsz)
84 new_winsz = fcntl.ioctl(mfd, set_winsz_opcode_maybe_neg, our_winsz)
Gregory P. Smith6af3db82008-03-20 05:41:53 +000085 finally:
86 os.close(mfd)
87 os.close(sfd)
88
Michael W. Hudsonf0089982003-03-03 12:29:42 +000089def test_main():
90 run_unittest(IoctlTests)
91
92if __name__ == "__main__":
93 test_main()