blob: fc8c39365f12b795bbc20be50a398e8a2cb02247 [file] [log] [blame]
Roger E. Massefb01d4b1996-12-17 17:41:09 +00001"""Test program for the fcntl C module.
Roger E. Massefb01d4b1996-12-17 17:41:09 +00002"""
Gregory P. Smithe5aefa42013-03-31 10:10:50 -07003import platform
Christian Heimese25f35e2008-03-20 10:49:03 +00004import os
5import struct
6import sys
Christian Heimesdd15f6c2008-03-16 00:07:10 +00007import unittest
Dong-hee Nabefa0322019-11-08 05:31:41 +09008from multiprocessing import Process
Serhiy Storchakabedce352021-09-19 22:36:03 +03009from test.support import verbose, cpython_only
Hai Shia089d212020-07-06 17:15:08 +080010from test.support.import_helper import import_module
11from test.support.os_helper import TESTFN, unlink
12
R. David Murraya21e4ca2009-03-31 23:16:50 +000013
Ezio Melotti78ede7c2013-08-23 23:06:31 +030014# Skip test if no fcntl module.
R. David Murraya21e4ca2009-03-31 23:16:50 +000015fcntl = import_module('fcntl')
16
Roger E. Massefb01d4b1996-12-17 17:41:09 +000017
Martin v. Löwisa660a342001-10-18 22:07:48 +000018
Christian Heimesdd15f6c2008-03-16 00:07:10 +000019def get_lockdata():
Antoine Pitrou6103ab12009-10-24 20:11:21 +000020 try:
21 os.O_LARGEFILE
22 except AttributeError:
23 start_len = "ll"
Hye-Shik Changac89f6e2005-04-04 15:21:04 +000024 else:
Antoine Pitrou6103ab12009-10-24 20:11:21 +000025 start_len = "qq"
Fred Drakebc7809b2001-05-09 21:11:59 +000026
Benjamin Peterson288d1da2017-09-28 22:44:27 -070027 if (sys.platform.startswith(('netbsd', 'freebsd', 'openbsd'))
Victor Stinnere6747472011-08-21 00:39:18 +020028 or sys.platform == 'darwin'):
Christian Heimesdd15f6c2008-03-16 00:07:10 +000029 if struct.calcsize('l') == 8:
30 off_t = 'l'
31 pid_t = 'i'
32 else:
33 off_t = 'lxxxx'
34 pid_t = 'l'
35 lockdata = struct.pack(off_t + off_t + pid_t + 'hh', 0, 0, 0,
36 fcntl.F_WRLCK, 0)
doko@ubuntu.com1dfb9182013-08-03 16:12:33 +020037 elif sys.platform.startswith('gnukfreebsd'):
38 lockdata = struct.pack('qqihhi', 0, 0, 0, fcntl.F_WRLCK, 0, 0)
Michael Feltb7eec942019-04-08 02:51:33 +020039 elif sys.platform in ['hp-uxB', 'unixware7']:
Christian Heimesdd15f6c2008-03-16 00:07:10 +000040 lockdata = struct.pack('hhlllii', fcntl.F_WRLCK, 0, 0, 0, 0, 0, 0)
Christian Heimesdd15f6c2008-03-16 00:07:10 +000041 else:
42 lockdata = struct.pack('hh'+start_len+'hh', fcntl.F_WRLCK, 0, 0, 0, 0, 0)
43 if lockdata:
44 if verbose:
45 print('struct.pack: ', repr(lockdata))
46 return lockdata
Fred Drakebc7809b2001-05-09 21:11:59 +000047
Christian Heimesdd15f6c2008-03-16 00:07:10 +000048lockdata = get_lockdata()
Roger E. Massefb01d4b1996-12-17 17:41:09 +000049
Serhiy Storchaka5cfc79d2014-02-07 10:06:39 +020050class BadFile:
51 def __init__(self, fn):
52 self.fn = fn
53 def fileno(self):
54 return self.fn
55
Dong-hee Na99602302019-11-19 17:12:42 +090056def try_lockf_on_other_process_fail(fname, cmd):
57 f = open(fname, 'wb+')
58 try:
59 fcntl.lockf(f, cmd)
60 except BlockingIOError:
61 pass
62 finally:
63 f.close()
64
65def try_lockf_on_other_process(fname, cmd):
66 f = open(fname, 'wb+')
67 fcntl.lockf(f, cmd)
68 fcntl.lockf(f, fcntl.LOCK_UN)
69 f.close()
70
Christian Heimesdd15f6c2008-03-16 00:07:10 +000071class TestFcntl(unittest.TestCase):
72
73 def setUp(self):
74 self.f = None
75
76 def tearDown(self):
Antoine Pitrou61f77b52009-05-24 15:46:13 +000077 if self.f and not self.f.closed:
Christian Heimesdd15f6c2008-03-16 00:07:10 +000078 self.f.close()
79 unlink(TESTFN)
80
81 def test_fcntl_fileno(self):
82 # the example from the library docs
Victor Stinnera6d2c762011-06-30 18:20:11 +020083 self.f = open(TESTFN, 'wb')
Christian Heimesdd15f6c2008-03-16 00:07:10 +000084 rv = fcntl.fcntl(self.f.fileno(), fcntl.F_SETFL, os.O_NONBLOCK)
85 if verbose:
86 print('Status from fcntl with O_NONBLOCK: ', rv)
Jesus Ceaf1af7052012-10-05 02:48:46 +020087 rv = fcntl.fcntl(self.f.fileno(), fcntl.F_SETLKW, lockdata)
88 if verbose:
89 print('String from fcntl with F_SETLKW: ', repr(rv))
Christian Heimesdd15f6c2008-03-16 00:07:10 +000090 self.f.close()
91
92 def test_fcntl_file_descriptor(self):
93 # again, but pass the file rather than numeric descriptor
Victor Stinnera6d2c762011-06-30 18:20:11 +020094 self.f = open(TESTFN, 'wb')
Christian Heimesdd15f6c2008-03-16 00:07:10 +000095 rv = fcntl.fcntl(self.f, fcntl.F_SETFL, os.O_NONBLOCK)
Ezio Melotti892584e2013-08-23 23:09:32 +030096 if verbose:
97 print('Status from fcntl with O_NONBLOCK: ', rv)
Jesus Ceaf1af7052012-10-05 02:48:46 +020098 rv = fcntl.fcntl(self.f, fcntl.F_SETLKW, lockdata)
Ezio Melotti892584e2013-08-23 23:09:32 +030099 if verbose:
100 print('String from fcntl with F_SETLKW: ', repr(rv))
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000101 self.f.close()
Fred Drakebc7809b2001-05-09 21:11:59 +0000102
Serhiy Storchaka78980432013-01-15 01:12:17 +0200103 def test_fcntl_bad_file(self):
Serhiy Storchaka5cfc79d2014-02-07 10:06:39 +0200104 with self.assertRaises(ValueError):
105 fcntl.fcntl(-1, fcntl.F_SETFL, os.O_NONBLOCK)
106 with self.assertRaises(ValueError):
107 fcntl.fcntl(BadFile(-1), fcntl.F_SETFL, os.O_NONBLOCK)
108 with self.assertRaises(TypeError):
109 fcntl.fcntl('spam', fcntl.F_SETFL, os.O_NONBLOCK)
110 with self.assertRaises(TypeError):
111 fcntl.fcntl(BadFile('spam'), fcntl.F_SETFL, os.O_NONBLOCK)
112
113 @cpython_only
114 def test_fcntl_bad_file_overflow(self):
115 from _testcapi import INT_MAX, INT_MIN
Serhiy Storchaka78980432013-01-15 01:12:17 +0200116 # Issue 15989
Serhiy Storchaka5cfc79d2014-02-07 10:06:39 +0200117 with self.assertRaises(OverflowError):
118 fcntl.fcntl(INT_MAX + 1, fcntl.F_SETFL, os.O_NONBLOCK)
119 with self.assertRaises(OverflowError):
120 fcntl.fcntl(BadFile(INT_MAX + 1), fcntl.F_SETFL, os.O_NONBLOCK)
121 with self.assertRaises(OverflowError):
122 fcntl.fcntl(INT_MIN - 1, fcntl.F_SETFL, os.O_NONBLOCK)
123 with self.assertRaises(OverflowError):
124 fcntl.fcntl(BadFile(INT_MIN - 1), fcntl.F_SETFL, os.O_NONBLOCK)
Serhiy Storchaka78980432013-01-15 01:12:17 +0200125
Gregory P. Smithe5aefa42013-03-31 10:10:50 -0700126 @unittest.skipIf(
127 platform.machine().startswith('arm') and platform.system() == 'Linux',
128 "ARM Linux returns EINVAL for F_NOTIFY DN_MULTISHOT")
Antoine Pitrou61f77b52009-05-24 15:46:13 +0000129 def test_fcntl_64_bit(self):
130 # Issue #1309352: fcntl shouldn't fail when the third arg fits in a
131 # C 'long' but not in a C 'int'.
132 try:
133 cmd = fcntl.F_NOTIFY
134 # This flag is larger than 2**31 in 64-bit builds
135 flags = fcntl.DN_MULTISHOT
136 except AttributeError:
137 self.skipTest("F_NOTIFY or DN_MULTISHOT unavailable")
138 fd = os.open(os.path.dirname(os.path.abspath(TESTFN)), os.O_RDONLY)
139 try:
140 fcntl.fcntl(fd, cmd, flags)
141 finally:
142 os.close(fd)
143
Christian Heimes0a956f12013-12-05 16:13:03 +0100144 def test_flock(self):
Christian Heimes2e7d4f02013-12-07 18:19:21 +0100145 # Solaris needs readable file for shared lock
146 self.f = open(TESTFN, 'wb+')
Christian Heimes0a956f12013-12-05 16:13:03 +0100147 fileno = self.f.fileno()
148 fcntl.flock(fileno, fcntl.LOCK_SH)
149 fcntl.flock(fileno, fcntl.LOCK_UN)
150 fcntl.flock(self.f, fcntl.LOCK_SH | fcntl.LOCK_NB)
151 fcntl.flock(self.f, fcntl.LOCK_UN)
152 fcntl.flock(fileno, fcntl.LOCK_EX)
153 fcntl.flock(fileno, fcntl.LOCK_UN)
154
155 self.assertRaises(ValueError, fcntl.flock, -1, fcntl.LOCK_SH)
156 self.assertRaises(TypeError, fcntl.flock, 'spam', fcntl.LOCK_SH)
Serhiy Storchakaf28ba362014-02-07 10:10:55 +0200157
Dong-hee Na99602302019-11-19 17:12:42 +0900158 @unittest.skipIf(platform.system() == "AIX", "AIX returns PermissionError")
Dong-hee Nabefa0322019-11-08 05:31:41 +0900159 def test_lockf_exclusive(self):
160 self.f = open(TESTFN, 'wb+')
161 cmd = fcntl.LOCK_EX | fcntl.LOCK_NB
Dong-hee Nabefa0322019-11-08 05:31:41 +0900162 fcntl.lockf(self.f, cmd)
Dong-hee Na99602302019-11-19 17:12:42 +0900163 p = Process(target=try_lockf_on_other_process_fail, args=(TESTFN, cmd))
Dong-hee Nabefa0322019-11-08 05:31:41 +0900164 p.start()
165 p.join()
166 fcntl.lockf(self.f, fcntl.LOCK_UN)
167 self.assertEqual(p.exitcode, 0)
168
Dong-hee Na99602302019-11-19 17:12:42 +0900169 @unittest.skipIf(platform.system() == "AIX", "AIX returns PermissionError")
Dong-hee Nabefa0322019-11-08 05:31:41 +0900170 def test_lockf_share(self):
171 self.f = open(TESTFN, 'wb+')
172 cmd = fcntl.LOCK_SH | fcntl.LOCK_NB
Dong-hee Nabefa0322019-11-08 05:31:41 +0900173 fcntl.lockf(self.f, cmd)
Dong-hee Na99602302019-11-19 17:12:42 +0900174 p = Process(target=try_lockf_on_other_process, args=(TESTFN, cmd))
Dong-hee Nabefa0322019-11-08 05:31:41 +0900175 p.start()
176 p.join()
177 fcntl.lockf(self.f, fcntl.LOCK_UN)
178 self.assertEqual(p.exitcode, 0)
179
Serhiy Storchakaf28ba362014-02-07 10:10:55 +0200180 @cpython_only
181 def test_flock_overflow(self):
182 import _testcapi
Christian Heimes0a956f12013-12-05 16:13:03 +0100183 self.assertRaises(OverflowError, fcntl.flock, _testcapi.INT_MAX+1,
184 fcntl.LOCK_SH)
185
Vinay Sharma13f37f22019-08-29 07:26:17 +0530186 @unittest.skipIf(sys.platform != 'darwin', "F_GETPATH is only available on macos")
187 def test_fcntl_f_getpath(self):
188 self.f = open(TESTFN, 'wb')
Benjamin Peterson465e5d52019-08-28 22:06:49 -0700189 expected = os.path.abspath(TESTFN).encode('utf-8')
190 res = fcntl.fcntl(self.f.fileno(), fcntl.F_GETPATH, bytes(len(expected)))
191 self.assertEqual(expected, res)
Fred Drakebc7809b2001-05-09 21:11:59 +0000192
Gregory P. Smith786addd2020-10-20 17:37:20 -0700193 @unittest.skipUnless(
194 hasattr(fcntl, "F_SETPIPE_SZ") and hasattr(fcntl, "F_GETPIPE_SZ"),
195 "F_SETPIPE_SZ and F_GETPIPE_SZ are not available on all platforms.")
Ruben Vorderman23c0fb82020-10-20 01:30:02 +0200196 def test_fcntl_f_pipesize(self):
197 test_pipe_r, test_pipe_w = os.pipe()
Gregory P. Smith786addd2020-10-20 17:37:20 -0700198 try:
199 # Get the default pipesize with F_GETPIPE_SZ
200 pipesize_default = fcntl.fcntl(test_pipe_w, fcntl.F_GETPIPE_SZ)
201 pipesize = pipesize_default // 2 # A new value to detect change.
202 if pipesize < 512: # the POSIX minimum
203 raise unittest.SkitTest(
204 'default pipesize too small to perform test.')
205 fcntl.fcntl(test_pipe_w, fcntl.F_SETPIPE_SZ, pipesize)
206 self.assertEqual(fcntl.fcntl(test_pipe_w, fcntl.F_GETPIPE_SZ),
207 pipesize)
208 finally:
209 os.close(test_pipe_r)
210 os.close(test_pipe_w)
Ruben Vorderman23c0fb82020-10-20 01:30:02 +0200211
212
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000213if __name__ == '__main__':
Serhiy Storchakabedce352021-09-19 22:36:03 +0300214 unittest.main()