blob: 351560024e8d7305f5c865de11534cf66d2bc8e7 [file] [log] [blame]
Bram Moolenaardb77b842019-03-24 14:58:31 +01001" Test signal handling.
2
Bram Moolenaar8c5a2782019-08-07 23:07:07 +02003CheckUnix
Bram Moolenaardb77b842019-03-24 14:58:31 +01004
Bram Moolenaar520e2452019-04-27 17:32:40 +02005" Check whether a signal is available on this system.
6func HasSignal(signal)
7 let signals = system('kill -l')
8 return signals =~# '\<' .. a:signal .. '\>'
9endfunc
10
Bram Moolenaardb77b842019-03-24 14:58:31 +010011" Test signal WINCH (window resize signal)
12func Test_signal_WINCH()
Bram Moolenaar8c5a2782019-08-07 23:07:07 +020013 CheckNotGui
14 if !HasSignal('WINCH')
15 throw 'Skipped: WINCH signal not supported'
Bram Moolenaardb77b842019-03-24 14:58:31 +010016 endif
17
18 " We do not actually want to change the size of the terminal.
19 let old_WS = ''
20 if exists('&t_WS')
21 let old_WS = &t_WS
22 let &t_WS = ''
23 endif
24
25 let old_lines = &lines
26 let old_columns = &columns
27 let new_lines = &lines - 2
28 let new_columns = &columns - 2
29
Bram Moolenaar520e2452019-04-27 17:32:40 +020030 exe 'set lines=' .. new_lines
31 exe 'set columns=' .. new_columns
Bram Moolenaardb77b842019-03-24 14:58:31 +010032 call assert_equal(new_lines, &lines)
33 call assert_equal(new_columns, &columns)
34
35 " Send signal and wait for signal to be processed.
36 " 'lines' and 'columns' should have been restored
37 " after handing signal WINCH.
Bram Moolenaar520e2452019-04-27 17:32:40 +020038 exe 'silent !kill -s WINCH ' .. getpid()
Bram Moolenaardb77b842019-03-24 14:58:31 +010039 call WaitForAssert({-> assert_equal(old_lines, &lines)})
40 call assert_equal(old_columns, &columns)
41
42 if old_WS != ''
43 let &t_WS = old_WS
44 endif
45endfunc
Bram Moolenaar520e2452019-04-27 17:32:40 +020046
47" Test signal PWR, which should update the swap file.
48func Test_signal_PWR()
49 if !HasSignal('PWR')
Bram Moolenaarbad88042020-03-23 20:54:32 +010050 throw 'Skipped: PWR signal not supported'
Bram Moolenaar520e2452019-04-27 17:32:40 +020051 endif
52
53 " Set a very large 'updatetime' and 'updatecount', so that we can be sure
54 " that swap file is updated as a result of sending PWR signal, and not
55 " because of exceeding 'updatetime' or 'updatecount' when changing buffer.
56 set updatetime=100000 updatecount=100000
57 new Xtest_signal_PWR
58 let swap_name = swapname('%')
59 call setline(1, '123')
60 preserve
61 let swap_content = readfile(swap_name, 'b')
62
63 " Update the buffer and check that the swap file is not yet updated,
64 " since we set 'updatetime' and 'updatecount' to large values.
65 call setline(1, 'abc')
66 call assert_equal(swap_content, readfile(swap_name, 'b'))
67
68 " Sending PWR signal should update the swap file.
69 exe 'silent !kill -s PWR ' .. getpid()
70 call WaitForAssert({-> assert_notequal(swap_content, readfile(swap_name, 'b'))})
71
72 bwipe!
73 set updatetime& updatecount&
74endfunc
Bram Moolenaarbad88042020-03-23 20:54:32 +010075
76" Test signal INT. Handler sets got_int. It should be like typing CTRL-C.
77func Test_signal_INT()
Bram Moolenaar494e9062020-05-31 21:28:02 +020078 CheckRunVimInTerminal
Bram Moolenaarbad88042020-03-23 20:54:32 +010079 if !HasSignal('INT')
80 throw 'Skipped: INT signal not supported'
81 endif
82
Bram Moolenaarbad88042020-03-23 20:54:32 +010083 let buf = RunVimInTerminal('', {'rows': 6})
84 let pid_vim = term_getjob(buf)->job_info().process
85
86 " Check that an endless loop in Vim is interrupted by signal INT.
Bram Moolenaar61e37842022-06-25 12:13:28 +010087 call term_sendkeys(buf, ":call setline(1, 'running')\n")
Bram Moolenaarbad88042020-03-23 20:54:32 +010088 call term_sendkeys(buf, ":while 1 | endwhile\n")
89 call WaitForAssert({-> assert_equal(':while 1 | endwhile', term_getline(buf, 6))})
90 exe 'silent !kill -s INT ' .. pid_vim
Bram Moolenaar61e37842022-06-25 12:13:28 +010091 sleep 50m
Dominique Pelle81b573d2022-03-22 21:14:55 +000092 call term_sendkeys(buf, ":call setline(1, 'INTERRUPTED')\n")
93 call WaitForAssert({-> assert_equal('INTERRUPTED', term_getline(buf, 1))})
Bram Moolenaarbad88042020-03-23 20:54:32 +010094
95 call StopVimInTerminal(buf)
96endfunc
Bram Moolenaar48a68712020-05-12 14:42:02 +020097
dbivolaruab16ad32021-12-29 19:41:47 +000098" Test signal TSTP. Handler sets got_tstp.
99func Test_signal_TSTP()
100 CheckRunVimInTerminal
101 if !HasSignal('TSTP')
102 throw 'Skipped: TSTP signal not supported'
103 endif
104
dbivolaruab16ad32021-12-29 19:41:47 +0000105 " If test fails once, it can leave temporary files and trying to rerun
106 " the test would then fail again if they are not deleted first.
107 call delete('.Xsig_TERM.swp')
108 call delete('XsetupAucmd')
K.Takataa9480db2022-09-06 10:56:19 +0100109 call delete('XautoOut1')
110 call delete('XautoOut2')
dbivolaruab16ad32021-12-29 19:41:47 +0000111 let lines =<< trim END
K.Takataa9480db2022-09-06 10:56:19 +0100112 au VimSuspend * call writefile(["VimSuspend triggered"], "XautoOut1", "as")
113 au VimResume * call writefile(["VimResume triggered"], "XautoOut2", "as")
dbivolaruab16ad32021-12-29 19:41:47 +0000114 END
Bram Moolenaar56564962022-10-10 22:39:42 +0100115 call writefile(lines, 'XsetupAucmd', 'D')
dbivolaruab16ad32021-12-29 19:41:47 +0000116
117 let buf = RunVimInTerminal('-S XsetupAucmd Xsig_TERM', {'rows': 6})
118 let pid_vim = term_getjob(buf)->job_info().process
119
120 call term_sendkeys(buf, ":call setline(1, 'foo')\n")
121 call WaitForAssert({-> assert_equal('foo', term_getline(buf, 1))})
122
123 call assert_false(filereadable('Xsig_TERM'))
124
125 " After TSTP the file is not saved (same function as ^Z)
126 exe 'silent !kill -s TSTP ' .. pid_vim
127 call WaitForAssert({-> assert_true(filereadable('.Xsig_TERM.swp'))})
K.Takataa9480db2022-09-06 10:56:19 +0100128 sleep 100m
dbivolaruab16ad32021-12-29 19:41:47 +0000129
Bram Moolenaar61e37842022-06-25 12:13:28 +0100130 " We resume after the suspend. Sleep a bit for the signal to take effect,
Bram Moolenaar94722c52023-01-28 19:19:03 +0000131 " also when running under valgrind.
dbivolaruab16ad32021-12-29 19:41:47 +0000132 exe 'silent !kill -s CONT ' .. pid_vim
K.Takataa9480db2022-09-06 10:56:19 +0100133 call WaitForAssert({-> assert_true(filereadable('XautoOut2'))})
134 sleep 10m
dbivolaruab16ad32021-12-29 19:41:47 +0000135
136 call StopVimInTerminal(buf)
137
K.Takataa9480db2022-09-06 10:56:19 +0100138 let result = readfile('XautoOut1')
139 call assert_equal(["VimSuspend triggered"], result)
140 let result = readfile('XautoOut2')
141 call assert_equal(["VimResume triggered"], result)
dbivolaruab16ad32021-12-29 19:41:47 +0000142
143 %bwipe!
144 call delete('.Xsig_TERM.swp')
K.Takataa9480db2022-09-06 10:56:19 +0100145 call delete('XautoOut1')
146 call delete('XautoOut2')
dbivolaruab16ad32021-12-29 19:41:47 +0000147endfunc
148
Bram Moolenaar48a68712020-05-12 14:42:02 +0200149" Test a deadly signal.
150"
151" There are several deadly signals: SISEGV, SIBUS, SIGTERM...
152" Test uses signal SIGTERM as it does not create a core
153" dump file unlike SIGSEGV, SIGBUS, etc. See "man 7 signals.
154"
155" Vim should exit with a deadly signal and unsaved changes
156" should be recoverable from the swap file preserved as a
157" result of the deadly signal handler.
158func Test_deadly_signal_TERM()
159 if !HasSignal('TERM')
160 throw 'Skipped: TERM signal not supported'
161 endif
Bram Moolenaar494e9062020-05-31 21:28:02 +0200162 CheckRunVimInTerminal
Bram Moolenaard14fd522020-06-01 15:05:19 +0200163
164 " If test fails once, it can leave temporary files and trying to rerun
165 " the test would then fail again if they are not deleted first.
166 call delete('.Xsig_TERM.swp')
167 call delete('XsetupAucmd')
168 call delete('XautoOut')
Bram Moolenaar129d6bf2020-05-16 16:08:35 +0200169 let lines =<< trim END
Bram Moolenaard14fd522020-06-01 15:05:19 +0200170 au VimLeave * call writefile(["VimLeave triggered"], "XautoOut", "as")
171 au VimLeavePre * call writefile(["VimLeavePre triggered"], "XautoOut", "as")
Bram Moolenaar129d6bf2020-05-16 16:08:35 +0200172 END
Bram Moolenaar56564962022-10-10 22:39:42 +0100173 call writefile(lines, 'XsetupAucmd', 'D')
Bram Moolenaar48a68712020-05-12 14:42:02 +0200174
Bram Moolenaar129d6bf2020-05-16 16:08:35 +0200175 let buf = RunVimInTerminal('-S XsetupAucmd Xsig_TERM', {'rows': 6})
Bram Moolenaar48a68712020-05-12 14:42:02 +0200176 let pid_vim = term_getjob(buf)->job_info().process
177
178 call term_sendkeys(buf, ":call setline(1, 'foo')\n")
179 call WaitForAssert({-> assert_equal('foo', term_getline(buf, 1))})
180
181 call assert_false(filereadable('Xsig_TERM'))
182 exe 'silent !kill -s TERM ' .. pid_vim
Bram Moolenaar48a68712020-05-12 14:42:02 +0200183 call WaitForAssert({-> assert_true(filereadable('.Xsig_TERM.swp'))})
184
185 " Don't call StopVimInTerminal() as it expects job to be still running.
186 call WaitForAssert({-> assert_equal("finished", term_getstatus(buf))})
187
188 new
189 silent recover .Xsig_TERM.swp
190 call assert_equal(['foo'], getline(1, '$'))
191
Bram Moolenaar129d6bf2020-05-16 16:08:35 +0200192 let result = readfile('XautoOut')
Bram Moolenaar67322bf2020-12-06 15:03:19 +0100193 call assert_equal(["VimLeavePre triggered", "VimLeave triggered"], result)
Bram Moolenaar129d6bf2020-05-16 16:08:35 +0200194
Bram Moolenaar48a68712020-05-12 14:42:02 +0200195 %bwipe!
196 call delete('.Xsig_TERM.swp')
Bram Moolenaar129d6bf2020-05-16 16:08:35 +0200197 call delete('XautoOut')
Bram Moolenaar48a68712020-05-12 14:42:02 +0200198endfunc
199
200" vim: ts=8 sw=2 sts=2 tw=80 fdm=marker