blob: 27b71feb2a0b5b41e928bdf395bab995771d3bb1 [file] [log] [blame]
Bram Moolenaar5cab73f2020-02-06 19:25:19 +01001" Test the :disassemble command, and compilation as a side effect
2
Bram Moolenaarf51cb4e2020-03-01 17:55:14 +01003source check.vim
Bram Moolenaar62aec932022-01-29 21:45:34 +00004import './vim9.vim' as v9
Bram Moolenaarf51cb4e2020-03-01 17:55:14 +01005
Bram Moolenaar62aec932022-01-29 21:45:34 +00006func s:NotCompiled()
Bram Moolenaar5cab73f2020-02-06 19:25:19 +01007 echo "not"
8endfunc
9
10let s:scriptvar = 4
11let g:globalvar = 'g'
Bram Moolenaard3aac292020-04-19 14:32:17 +020012let b:buffervar = 'b'
13let w:windowvar = 'w'
14let t:tabpagevar = 't'
Bram Moolenaar5cab73f2020-02-06 19:25:19 +010015
16def s:ScriptFuncLoad(arg: string)
Bram Moolenaarac564082020-09-27 19:05:33 +020017 var local = 1
Bram Moolenaar5cab73f2020-02-06 19:25:19 +010018 buffers
Bram Moolenaare4984292020-12-13 14:19:25 +010019 echo
Bram Moolenaar5cab73f2020-02-06 19:25:19 +010020 echo arg
21 echo local
Bram Moolenaar8a1c1012020-05-07 14:07:25 +020022 echo &lines
Bram Moolenaar5cab73f2020-02-06 19:25:19 +010023 echo v:version
24 echo s:scriptvar
25 echo g:globalvar
Bram Moolenaar2f8ce0a2020-07-19 19:47:35 +020026 echo get(g:, "global")
Bram Moolenaar03290b82020-12-19 16:30:44 +010027 echo g:auto#var
Bram Moolenaard3aac292020-04-19 14:32:17 +020028 echo b:buffervar
Bram Moolenaar2f8ce0a2020-07-19 19:47:35 +020029 echo get(b:, "buffer")
Bram Moolenaard3aac292020-04-19 14:32:17 +020030 echo w:windowvar
Bram Moolenaar2f8ce0a2020-07-19 19:47:35 +020031 echo get(w:, "window")
Bram Moolenaard3aac292020-04-19 14:32:17 +020032 echo t:tabpagevar
Bram Moolenaar2f8ce0a2020-07-19 19:47:35 +020033 echo get(t:, "tab")
Bram Moolenaar5cab73f2020-02-06 19:25:19 +010034 echo &tabstop
35 echo $ENVVAR
36 echo @z
37enddef
38
Bram Moolenaarf2460a32020-02-07 22:09:54 +010039def Test_disassemble_load()
Bram Moolenaar5cab73f2020-02-06 19:25:19 +010040 assert_fails('disass NoFunc', 'E1061:')
Bram Moolenaar451c2e32020-08-15 16:33:28 +020041 assert_fails('disass NotCompiled', 'E1091:')
Bram Moolenaar21456cd2020-02-13 21:29:32 +010042 assert_fails('disass', 'E471:')
43 assert_fails('disass [', 'E475:')
Bram Moolenaar9b7bf9e2020-07-11 22:14:59 +020044 assert_fails('disass 234', 'E129:')
45 assert_fails('disass <XX>foo', 'E129:')
Bram Moolenaarf79d9dd2022-05-21 15:39:02 +010046 assert_fails('disass Test_disassemble_load burp', 'E488:')
47 assert_fails('disass debug debug Test_disassemble_load', 'E488:')
48 assert_fails('disass profile profile Test_disassemble_load', 'E488:')
Bram Moolenaar5cab73f2020-02-06 19:25:19 +010049
Bram Moolenaarac564082020-09-27 19:05:33 +020050 var res = execute('disass s:ScriptFuncLoad')
Bram Moolenaar675f7162020-04-12 22:53:54 +020051 assert_match('<SNR>\d*_ScriptFuncLoad.*' ..
Bram Moolenaare4984292020-12-13 14:19:25 +010052 'buffers\_s*' ..
53 '\d\+ EXEC \+buffers\_s*' ..
54 'echo\_s*' ..
55 'echo arg\_s*' ..
56 '\d\+ LOAD arg\[-1\]\_s*' ..
57 '\d\+ ECHO 1\_s*' ..
58 'echo local\_s*' ..
59 '\d\+ LOAD $0\_s*' ..
60 '\d\+ ECHO 1\_s*' ..
61 'echo &lines\_s*' ..
62 '\d\+ LOADOPT &lines\_s*' ..
63 '\d\+ ECHO 1\_s*' ..
64 'echo v:version\_s*' ..
65 '\d\+ LOADV v:version\_s*' ..
66 '\d\+ ECHO 1\_s*' ..
67 'echo s:scriptvar\_s*' ..
68 '\d\+ LOADS s:scriptvar from .*test_vim9_disassemble.vim\_s*' ..
69 '\d\+ ECHO 1\_s*' ..
70 'echo g:globalvar\_s*' ..
71 '\d\+ LOADG g:globalvar\_s*' ..
72 '\d\+ ECHO 1\_s*' ..
Bram Moolenaar2f8ce0a2020-07-19 19:47:35 +020073 'echo get(g:, "global")\_s*' ..
74 '\d\+ LOAD g:\_s*' ..
75 '\d\+ PUSHS "global"\_s*' ..
Bram Moolenaar03290b82020-12-19 16:30:44 +010076 '\d\+ BCALL get(argc 2)\_s*' ..
77 '\d\+ ECHO 1\_s*' ..
78 'echo g:auto#var\_s*' ..
79 '\d\+ LOADAUTO g:auto#var\_s*' ..
80 '\d\+ ECHO 1\_s*' ..
81 'echo b:buffervar\_s*' ..
82 '\d\+ LOADB b:buffervar\_s*' ..
83 '\d\+ ECHO 1\_s*' ..
Bram Moolenaar2f8ce0a2020-07-19 19:47:35 +020084 'echo get(b:, "buffer")\_s*' ..
85 '\d\+ LOAD b:\_s*' ..
86 '\d\+ PUSHS "buffer"\_s*' ..
87 '\d\+ BCALL get(argc 2).*' ..
Bram Moolenaard3aac292020-04-19 14:32:17 +020088 ' LOADW w:windowvar.*' ..
Bram Moolenaar2f8ce0a2020-07-19 19:47:35 +020089 'echo get(w:, "window")\_s*' ..
90 '\d\+ LOAD w:\_s*' ..
91 '\d\+ PUSHS "window"\_s*' ..
92 '\d\+ BCALL get(argc 2).*' ..
Bram Moolenaard3aac292020-04-19 14:32:17 +020093 ' LOADT t:tabpagevar.*' ..
Bram Moolenaar2f8ce0a2020-07-19 19:47:35 +020094 'echo get(t:, "tab")\_s*' ..
95 '\d\+ LOAD t:\_s*' ..
96 '\d\+ PUSHS "tab"\_s*' ..
97 '\d\+ BCALL get(argc 2).*' ..
Bram Moolenaar675f7162020-04-12 22:53:54 +020098 ' LOADENV $ENVVAR.*' ..
99 ' LOADREG @z.*',
100 res)
Bram Moolenaar5cab73f2020-02-06 19:25:19 +0100101enddef
102
Bram Moolenaarcfe435d2020-04-25 20:02:55 +0200103def s:EditExpand()
Bram Moolenaarac564082020-09-27 19:05:33 +0200104 var filename = "file"
105 var filenr = 123
Bram Moolenaarcfe435d2020-04-25 20:02:55 +0200106 edit the`=filename``=filenr`.txt
107enddef
108
109def Test_disassemble_exec_expr()
Bram Moolenaarac564082020-09-27 19:05:33 +0200110 var res = execute('disass s:EditExpand')
Bram Moolenaar7c5ad342020-08-12 15:48:55 +0200111 assert_match('<SNR>\d*_EditExpand\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +0200112 ' var filename = "file"\_s*' ..
Bram Moolenaar7c5ad342020-08-12 15:48:55 +0200113 '\d PUSHS "file"\_s*' ..
114 '\d STORE $0\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +0200115 ' var filenr = 123\_s*' ..
Bram Moolenaar7c5ad342020-08-12 15:48:55 +0200116 '\d STORE 123 in $1\_s*' ..
117 ' edit the`=filename``=filenr`.txt\_s*' ..
118 '\d PUSHS "edit the"\_s*' ..
119 '\d LOAD $0\_s*' ..
120 '\d LOAD $1\_s*' ..
121 '\d 2STRING stack\[-1\]\_s*' ..
122 '\d\+ PUSHS ".txt"\_s*' ..
123 '\d\+ EXECCONCAT 4\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +0200124 '\d\+ RETURN void',
Bram Moolenaar7c5ad342020-08-12 15:48:55 +0200125 res)
126enddef
127
Bram Moolenaar20677332021-06-06 17:02:53 +0200128if has('python3')
129 def s:PyHeredoc()
130 python3 << EOF
131 print('hello')
132EOF
133 enddef
134
135 def Test_disassemble_python_heredoc()
136 var res = execute('disass s:PyHeredoc')
137 assert_match('<SNR>\d*_PyHeredoc.*' ..
138 " python3 << EOF^@ print('hello')^@EOF\\_s*" ..
139 '\d EXEC_SPLIT python3 << EOF^@ print(''hello'')^@EOF\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +0200140 '\d RETURN void',
Bram Moolenaar20677332021-06-06 17:02:53 +0200141 res)
142 enddef
143endif
144
Bram Moolenaar4c137212021-04-19 16:48:48 +0200145def s:Substitute()
146 var expr = "abc"
147 :%s/a/\=expr/&g#c
148enddef
149
150def Test_disassemble_substitute()
151 var res = execute('disass s:Substitute')
152 assert_match('<SNR>\d*_Substitute.*' ..
153 ' var expr = "abc"\_s*' ..
154 '\d PUSHS "abc"\_s*' ..
155 '\d STORE $0\_s*' ..
156 ' :%s/a/\\=expr/&g#c\_s*' ..
157 '\d SUBSTITUTE :%s/a/\\=expr/&g#c\_s*' ..
158 ' 0 LOAD $0\_s*' ..
159 ' -------------\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +0200160 '\d RETURN void',
Bram Moolenaar4c137212021-04-19 16:48:48 +0200161 res)
162enddef
163
Bram Moolenaarf18332f2021-05-07 17:55:55 +0200164
165def s:SearchPair()
166 var col = 8
167 searchpair("{", "", "}", "", "col('.') > col")
168enddef
169
170def Test_disassemble_seachpair()
171 var res = execute('disass s:SearchPair')
172 assert_match('<SNR>\d*_SearchPair.*' ..
173 ' var col = 8\_s*' ..
174 '\d STORE 8 in $0\_s*' ..
175 ' searchpair("{", "", "}", "", "col(''.'') > col")\_s*' ..
176 '\d PUSHS "{"\_s*' ..
177 '\d PUSHS ""\_s*' ..
178 '\d PUSHS "}"\_s*' ..
179 '\d PUSHS ""\_s*' ..
180 '\d INSTR\_s*' ..
181 ' 0 PUSHS "."\_s*' ..
182 ' 1 BCALL col(argc 1)\_s*' ..
183 ' 2 LOAD $0\_s*' ..
184 ' 3 COMPARENR >\_s*' ..
185 ' -------------\_s*' ..
186 '\d BCALL searchpair(argc 5)\_s*' ..
187 '\d DROP\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +0200188 '\d RETURN void',
Bram Moolenaarf18332f2021-05-07 17:55:55 +0200189 res)
190enddef
191
192
LemonBoyf3b48952022-05-05 13:53:03 +0100193def s:SubstituteExpr()
194 substitute('a', 'b', '\=123', 'g')
195enddef
196
197def Test_disassemble_substitute_expr()
198 var res = execute('disass s:SubstituteExpr')
199 assert_match('<SNR>\d*_SubstituteExpr.*' ..
200 'substitute(''a'', ''b'', ''\\=123'', ''g'')\_s*' ..
201 '\d PUSHS "a"\_s*' ..
202 '\d PUSHS "b"\_s*' ..
203 '\d INSTR\_s*' ..
204 ' 0 PUSHNR 123\_s*' ..
205 ' -------------\_s*' ..
206 '\d PUSHS "g"\_s*' ..
207 '\d BCALL substitute(argc 4)\_s*' ..
208 '\d DROP\_s*' ..
209 '\d RETURN void',
210 res)
211enddef
212
Bram Moolenaar2d1c57e2021-04-19 20:50:03 +0200213def s:RedirVar()
214 var result: string
215 redir =>> result
216 echo "text"
217 redir END
218enddef
219
220def Test_disassemble_redir_var()
221 var res = execute('disass s:RedirVar')
222 assert_match('<SNR>\d*_RedirVar.*' ..
223 ' var result: string\_s*' ..
224 '\d PUSHS "\[NULL\]"\_s*' ..
225 '\d STORE $0\_s*' ..
226 ' redir =>> result\_s*' ..
227 '\d REDIR\_s*' ..
228 ' echo "text"\_s*' ..
229 '\d PUSHS "text"\_s*' ..
230 '\d ECHO 1\_s*' ..
231 ' redir END\_s*' ..
232 '\d LOAD $0\_s*' ..
233 '\d REDIR END\_s*' ..
LemonBoy372bcce2022-04-25 12:43:20 +0100234 '\d CONCAT size 2\_s*' ..
Bram Moolenaar2d1c57e2021-04-19 20:50:03 +0200235 '\d STORE $0\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +0200236 '\d RETURN void',
Bram Moolenaar2d1c57e2021-04-19 20:50:03 +0200237 res)
238enddef
239
Bram Moolenaar5f7d4c02021-05-05 21:31:39 +0200240def s:Cexpr()
241 var errors = "list of errors"
242 cexpr errors
243enddef
244
245def Test_disassemble_cexpr()
246 var res = execute('disass s:Cexpr')
247 assert_match('<SNR>\d*_Cexpr.*' ..
248 ' var errors = "list of errors"\_s*' ..
249 '\d PUSHS "list of errors"\_s*' ..
250 '\d STORE $0\_s*' ..
251 ' cexpr errors\_s*' ..
252 '\d CEXPR pre cexpr\_s*' ..
253 '\d LOAD $0\_s*' ..
254 '\d CEXPR core cexpr "cexpr errors"\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +0200255 '\d RETURN void',
Bram Moolenaar5f7d4c02021-05-05 21:31:39 +0200256 res)
257enddef
258
Bram Moolenaar7c5ad342020-08-12 15:48:55 +0200259def s:YankRange()
260 norm! m[jjm]
261 :'[,']yank
262enddef
263
264def Test_disassemble_yank_range()
Bram Moolenaarac564082020-09-27 19:05:33 +0200265 var res = execute('disass s:YankRange')
Bram Moolenaar7c5ad342020-08-12 15:48:55 +0200266 assert_match('<SNR>\d*_YankRange.*' ..
267 ' norm! m\[jjm\]\_s*' ..
268 '\d EXEC norm! m\[jjm\]\_s*' ..
269 ' :''\[,''\]yank\_s*' ..
270 '\d EXEC :''\[,''\]yank\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +0200271 '\d RETURN void',
Bram Moolenaarcfe435d2020-04-25 20:02:55 +0200272 res)
273enddef
274
Bram Moolenaarc3516f72020-09-08 22:45:35 +0200275def s:PutExpr()
276 :3put ="text"
277enddef
278
279def Test_disassemble_put_expr()
Bram Moolenaarac564082020-09-27 19:05:33 +0200280 var res = execute('disass s:PutExpr')
Bram Moolenaarc3516f72020-09-08 22:45:35 +0200281 assert_match('<SNR>\d*_PutExpr.*' ..
282 ' :3put ="text"\_s*' ..
283 '\d PUSHS "text"\_s*' ..
284 '\d PUT = 3\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +0200285 '\d RETURN void',
Bram Moolenaarc3516f72020-09-08 22:45:35 +0200286 res)
287enddef
288
64-bitmane08f10a2025-03-18 22:14:34 +0100289def s:IputExpr()
290 :3iput ="text"
291enddef
292
293def Test_disassemble_iput_expr()
294 var res = execute('disass s:IputExpr')
295 assert_match('<SNR>\d*_IputExpr.*' ..
296 ' :3iput ="text"\_s*' ..
297 '\d PUSHS "text"\_s*' ..
298 '\d IPUT = 3\_s*' ..
299 '\d RETURN void',
300 res)
301enddef
302
Bram Moolenaar08597872020-12-10 19:43:40 +0100303def s:PutRange()
304 :$-2put a
Bram Moolenaarf6ced982022-04-28 12:00:49 +0100305 :$-3put! b
Bram Moolenaar08597872020-12-10 19:43:40 +0100306enddef
307
308def Test_disassemble_put_range()
309 var res = execute('disass s:PutRange')
310 assert_match('<SNR>\d*_PutRange.*' ..
311 ' :$-2put a\_s*' ..
312 '\d RANGE $-2\_s*' ..
313 '\d PUT a range\_s*' ..
Bram Moolenaarf6ced982022-04-28 12:00:49 +0100314
315 ' :$-3put! b\_s*' ..
316 '\d RANGE $-3\_s*' ..
317 '\d PUT b above range\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +0200318 '\d RETURN void',
Bram Moolenaar08597872020-12-10 19:43:40 +0100319 res)
320enddef
321
64-bitmane08f10a2025-03-18 22:14:34 +0100322def s:IputRange()
323 :$-2iput a
324 :$-3iput! b
325enddef
326
327def Test_disassemble_iput_range()
328 var res = execute('disass s:IputRange')
329 assert_match('<SNR>\d*_IputRange.*' ..
330 ' :$-2iput a\_s*' ..
331 '\d RANGE $-2\_s*' ..
332 '\d IPUT a range\_s*' ..
333
334 ' :$-3iput! b\_s*' ..
335 '\d RANGE $-3\_s*' ..
336 '\d IPUT b above range\_s*' ..
337 '\d RETURN void',
338 res)
339enddef
340
Bram Moolenaar5cab73f2020-02-06 19:25:19 +0100341def s:ScriptFuncPush()
Bram Moolenaarac564082020-09-27 19:05:33 +0200342 var localbool = true
343 var localspec = v:none
344 var localblob = 0z1234
Bram Moolenaar73e28dc2022-09-17 21:08:33 +0100345 var localfloat = 1.234
Bram Moolenaar5cab73f2020-02-06 19:25:19 +0100346enddef
347
Bram Moolenaarf2460a32020-02-07 22:09:54 +0100348def Test_disassemble_push()
Bram Moolenaar0e9bdad2022-10-15 20:06:33 +0100349 mkdir('Xdisdir/autoload', 'pR')
Bram Moolenaar06b77222022-01-25 15:51:56 +0000350 var save_rtp = &rtp
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +0100351 exe 'set rtp^=' .. getcwd() .. '/Xdisdir'
Bram Moolenaar06b77222022-01-25 15:51:56 +0000352
353 var lines =<< trim END
354 vim9script
355 END
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +0100356 writefile(lines, 'Xdisdir/autoload/autoscript.vim')
Bram Moolenaar06b77222022-01-25 15:51:56 +0000357
358 lines =<< trim END
359 vim9script
360 import autoload 'autoscript.vim'
361
Bram Moolenaara749a422022-02-12 19:52:25 +0000362 def AutoloadFunc()
Bram Moolenaar06b77222022-01-25 15:51:56 +0000363 &operatorfunc = autoscript.Opfunc
364 enddef
365
Bram Moolenaara749a422022-02-12 19:52:25 +0000366 var res = execute('disass AutoloadFunc')
Bram Moolenaar06b77222022-01-25 15:51:56 +0000367 assert_match('<SNR>\d*_AutoloadFunc.*' ..
368 '&operatorfunc = autoscript.Opfunc\_s*' ..
369 '0 AUTOLOAD autoscript#Opfunc\_s*' ..
370 '1 STOREFUNCOPT &operatorfunc\_s*' ..
371 '2 RETURN void',
372 res)
373 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000374 v9.CheckScriptSuccess(lines)
Bram Moolenaar06b77222022-01-25 15:51:56 +0000375
Bram Moolenaar06b77222022-01-25 15:51:56 +0000376 &rtp = save_rtp
Bram Moolenaar5cab73f2020-02-06 19:25:19 +0100377enddef
378
Bram Moolenaarc0ceeeb2022-03-30 21:12:27 +0100379def Test_disassemble_import_autoload()
Bram Moolenaar0e9bdad2022-10-15 20:06:33 +0100380 writefile(['vim9script'], 'XimportAL.vim', 'D')
Bram Moolenaarc0ceeeb2022-03-30 21:12:27 +0100381
382 var lines =<< trim END
383 vim9script
384 import autoload './XimportAL.vim'
385
386 def AutoloadFunc()
387 echo XimportAL.SomeFunc()
388 echo XimportAL.someVar
389 XimportAL.someVar = "yes"
390 enddef
391
392 var res = execute('disass AutoloadFunc')
393 assert_match('<SNR>\d*_AutoloadFunc.*' ..
394 'echo XimportAL.SomeFunc()\_s*' ..
Bram Moolenaar17125182022-03-30 21:57:50 +0100395 '\d SOURCE .*/testdir/XimportAL.vim\_s*' ..
Bram Moolenaarc0ceeeb2022-03-30 21:12:27 +0100396 '\d PUSHFUNC "<80><fd>R\d\+_SomeFunc"\_s*' ..
397 '\d PCALL top (argc 0)\_s*' ..
398 '\d PCALL end\_s*' ..
399 '\d ECHO 1\_s*' ..
400
401 'echo XimportAL.someVar\_s*' ..
402 '\d SOURCE .*/testdir/XimportAL.vim\_s*' ..
403 '\d LOADEXPORT s:someVar from .*/testdir/XimportAL.vim\_s*' ..
404 '\d ECHO 1\_s*' ..
405
406 'XimportAL.someVar = "yes"\_s*' ..
407 '\d\+ PUSHS "yes"\_s*' ..
408 '\d\+ SOURCE .*/testdir/XimportAL.vim\_s*' ..
409 '\d\+ STOREEXPORT someVar in .*/testdir/XimportAL.vim\_s*' ..
410
411 '\d\+ RETURN void',
412 res)
413 END
414 v9.CheckScriptSuccess(lines)
Bram Moolenaarc0ceeeb2022-03-30 21:12:27 +0100415enddef
416
Ernie Rael3f821d62024-04-24 20:07:50 +0200417def Test_disassemble_import_autoload_autoload()
418 mkdir('Xauto_auto/autoload', 'pR')
419 var lines =<< trim END
420 vim9script
421 export const val = 11
422 END
423 writefile(lines, 'Xauto_auto/autoload/Xauto_vars_f1.vim')
424
425 lines =<< trim END
426 vim9script
427
428 import autoload './Xauto_auto/autoload/Xauto_vars_f1.vim' as f1
429 def F()
430 f1.val = 13
431 enddef
432 var res = execute('disass F')
433
434 assert_match('<SNR>\d*_F.*' ..
435 'f1.val = 13\_s*' ..
436 '\d PUSHNR 13\_s*' ..
437 '\d SOURCE .*/Xauto_auto/autoload/Xauto_vars_f1.vim\_s*' ..
438 '\d STOREEXPORT val in .*/Xauto_auto/autoload/Xauto_vars_f1.vim\_s*' ..
439 '\d RETURN void',
440 res)
441 END
442 v9.CheckScriptSuccess(lines)
443enddef
444
Bram Moolenaar5cab73f2020-02-06 19:25:19 +0100445def s:ScriptFuncStore()
Bram Moolenaarac564082020-09-27 19:05:33 +0200446 var localnr = 1
Bram Moolenaar5cab73f2020-02-06 19:25:19 +0100447 localnr = 2
Bram Moolenaarac564082020-09-27 19:05:33 +0200448 var localstr = 'abc'
Bram Moolenaar5cab73f2020-02-06 19:25:19 +0100449 localstr = 'xyz'
450 v:char = 'abc'
451 s:scriptvar = 'sv'
452 g:globalvar = 'gv'
Bram Moolenaar03290b82020-12-19 16:30:44 +0100453 g:auto#var = 'av'
Bram Moolenaard3aac292020-04-19 14:32:17 +0200454 b:buffervar = 'bv'
455 w:windowvar = 'wv'
456 t:tabpagevar = 'tv'
Bram Moolenaar5cab73f2020-02-06 19:25:19 +0100457 &tabstop = 8
Bram Moolenaardcb53be2021-12-09 14:23:43 +0000458 &opfunc = (t) => len(t)
Bram Moolenaar5cab73f2020-02-06 19:25:19 +0100459 $ENVVAR = 'ev'
460 @z = 'rv'
461enddef
462
Bram Moolenaarf2460a32020-02-07 22:09:54 +0100463def Test_disassemble_store()
Bram Moolenaarac564082020-09-27 19:05:33 +0200464 var res = execute('disass s:ScriptFuncStore')
Bram Moolenaar675f7162020-04-12 22:53:54 +0200465 assert_match('<SNR>\d*_ScriptFuncStore.*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +0200466 'var localnr = 1.*' ..
Bram Moolenaar675f7162020-04-12 22:53:54 +0200467 'localnr = 2.*' ..
468 ' STORE 2 in $0.*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +0200469 'var localstr = ''abc''.*' ..
Bram Moolenaar675f7162020-04-12 22:53:54 +0200470 'localstr = ''xyz''.*' ..
471 ' STORE $1.*' ..
472 'v:char = ''abc''.*' ..
473 'STOREV v:char.*' ..
474 's:scriptvar = ''sv''.*' ..
475 ' STORES s:scriptvar in .*test_vim9_disassemble.vim.*' ..
476 'g:globalvar = ''gv''.*' ..
477 ' STOREG g:globalvar.*' ..
Bram Moolenaar03290b82020-12-19 16:30:44 +0100478 'g:auto#var = ''av''.*' ..
479 ' STOREAUTO g:auto#var.*' ..
Bram Moolenaard3aac292020-04-19 14:32:17 +0200480 'b:buffervar = ''bv''.*' ..
481 ' STOREB b:buffervar.*' ..
482 'w:windowvar = ''wv''.*' ..
483 ' STOREW w:windowvar.*' ..
484 't:tabpagevar = ''tv''.*' ..
485 ' STORET t:tabpagevar.*' ..
Bram Moolenaardcb53be2021-12-09 14:23:43 +0000486 '&tabstop = 8\_s*' ..
487 '\d\+ PUSHNR 8\_s*' ..
488 '\d\+ STOREOPT &tabstop\_s*' ..
489 '&opfunc = (t) => len(t)\_s*' ..
490 '\d\+ FUNCREF <lambda>\d\+\_s*' ..
491 '\d\+ STOREFUNCOPT &opfunc\_s*' ..
492 '$ENVVAR = ''ev''\_s*' ..
493 '\d\+ PUSHS "ev"\_s*' ..
494 '\d\+ STOREENV $ENVVAR\_s*' ..
Bram Moolenaar675f7162020-04-12 22:53:54 +0200495 '@z = ''rv''.*' ..
Bram Moolenaardcb53be2021-12-09 14:23:43 +0000496 '\d\+ STOREREG @z.*',
Bram Moolenaar675f7162020-04-12 22:53:54 +0200497 res)
Bram Moolenaar5cab73f2020-02-06 19:25:19 +0100498enddef
499
Bram Moolenaarcb790402020-05-15 20:53:00 +0200500def s:ScriptFuncStoreMember()
Bram Moolenaarac564082020-09-27 19:05:33 +0200501 var locallist: list<number> = []
Bram Moolenaarcb790402020-05-15 20:53:00 +0200502 locallist[0] = 123
Bram Moolenaarac564082020-09-27 19:05:33 +0200503 var localdict: dict<number> = {}
Bram Moolenaarcb790402020-05-15 20:53:00 +0200504 localdict["a"] = 456
Bram Moolenaar51e93322021-04-17 20:44:56 +0200505 var localblob: blob = 0z1122
506 localblob[1] = 33
Bram Moolenaarcb790402020-05-15 20:53:00 +0200507enddef
508
509def Test_disassemble_store_member()
Bram Moolenaarac564082020-09-27 19:05:33 +0200510 var res = execute('disass s:ScriptFuncStoreMember')
Bram Moolenaarcb790402020-05-15 20:53:00 +0200511 assert_match('<SNR>\d*_ScriptFuncStoreMember\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +0200512 'var locallist: list<number> = []\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +0200513 '\d NEWLIST size 0\_s*' ..
Bram Moolenaaraa210a32021-01-02 15:41:03 +0100514 '\d SETTYPE list<number>\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +0200515 '\d STORE $0\_s*' ..
516 'locallist\[0\] = 123\_s*' ..
517 '\d PUSHNR 123\_s*' ..
518 '\d PUSHNR 0\_s*' ..
519 '\d LOAD $0\_s*' ..
Bram Moolenaar51e93322021-04-17 20:44:56 +0200520 '\d STOREINDEX list\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +0200521 'var localdict: dict<number> = {}\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +0200522 '\d NEWDICT size 0\_s*' ..
Bram Moolenaaraa210a32021-01-02 15:41:03 +0100523 '\d SETTYPE dict<number>\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +0200524 '\d STORE $1\_s*' ..
525 'localdict\["a"\] = 456\_s*' ..
526 '\d\+ PUSHNR 456\_s*' ..
527 '\d\+ PUSHS "a"\_s*' ..
528 '\d\+ LOAD $1\_s*' ..
Bram Moolenaar51e93322021-04-17 20:44:56 +0200529 '\d\+ STOREINDEX dict\_s*' ..
530 'var localblob: blob = 0z1122\_s*' ..
531 '\d\+ PUSHBLOB 0z1122\_s*' ..
532 '\d\+ STORE $2\_s*' ..
533 'localblob\[1\] = 33\_s*' ..
534 '\d\+ PUSHNR 33\_s*' ..
535 '\d\+ PUSHNR 1\_s*' ..
536 '\d\+ LOAD $2\_s*' ..
537 '\d\+ STOREINDEX blob\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +0200538 '\d\+ RETURN void',
Bram Moolenaarcb790402020-05-15 20:53:00 +0200539 res)
540enddef
541
Bram Moolenaar8acb9cc2022-03-08 13:18:55 +0000542if has('job')
543 def s:StoreNull()
544 var ss = null_string
545 var bb = null_blob
546 var dd = null_dict
547 var ll = null_list
548 var Ff = null_function
549 var Pp = null_partial
550 var jj = null_job
551 var cc = null_channel
Yegappan Lakshmananf4ee1cb2023-10-09 17:57:27 +0200552 var oo = null_object
553 var nc = null_class
Bram Moolenaar8acb9cc2022-03-08 13:18:55 +0000554 enddef
555
556 def Test_disassemble_assign_null()
557 var res = execute('disass s:StoreNull')
558 assert_match('<SNR>\d*_StoreNull\_s*' ..
559 'var ss = null_string\_s*' ..
560 '\d\+ PUSHS "\[NULL\]"\_s*' ..
561 '\d\+ STORE $\d\_s*' ..
562
563 'var bb = null_blob\_s*' ..
564 '\d\+ PUSHBLOB 0z\_s*' ..
565 '\d\+ STORE $\d\_s*' ..
566
567 'var dd = null_dict\_s*' ..
Bram Moolenaarec15b1c2022-03-27 16:29:53 +0100568 '\d\+ NEWDICT size -1\_s*' ..
Bram Moolenaar8acb9cc2022-03-08 13:18:55 +0000569 '\d\+ STORE $\d\_s*' ..
570
571 'var ll = null_list\_s*' ..
Bram Moolenaarec15b1c2022-03-27 16:29:53 +0100572 '\d\+ NEWLIST size -1\_s*' ..
Bram Moolenaar8acb9cc2022-03-08 13:18:55 +0000573 '\d\+ STORE $\d\_s*' ..
574
575 'var Ff = null_function\_s*' ..
576 '\d\+ PUSHFUNC "\[none\]"\_s*' ..
577 '\d\+ STORE $\d\_s*' ..
578
579 'var Pp = null_partial\_s*' ..
580 '\d\+ NEWPARTIAL\_s*' ..
581 '\d\+ STORE $\d\_s*' ..
582
583 'var jj = null_job\_s*' ..
584 '\d\+ PUSHJOB "no process"\_s*' ..
585 '\d\+ STORE $\d\_s*' ..
586
587 'var cc = null_channel\_s*' ..
588 '\d\+ PUSHCHANNEL 0\_s*' ..
589 '\d\+ STORE $\d\_s*' ..
590
Yegappan Lakshmananf4ee1cb2023-10-09 17:57:27 +0200591 'var oo = null_object\_s*' ..
592 '\d\+ PUSHOBJ null\_s*' ..
593 '\d\+ STORE $\d\_s*' ..
594
595 'var nc = null_class\_s*' ..
596 '\d\+ PUSHCLASS null\_s*' ..
597 '\d\+ STORE $\d\_s*' ..
598
Bram Moolenaar8acb9cc2022-03-08 13:18:55 +0000599 '\d\+ RETURN void',
600 res)
601 enddef
602endif
603
Bram Moolenaar4f5e3972020-12-21 17:30:50 +0100604def s:ScriptFuncStoreIndex()
605 var d = {dd: {}}
606 d.dd[0] = 0
607enddef
608
609def Test_disassemble_store_index()
610 var res = execute('disass s:ScriptFuncStoreIndex')
611 assert_match('<SNR>\d*_ScriptFuncStoreIndex\_s*' ..
612 'var d = {dd: {}}\_s*' ..
613 '\d PUSHS "dd"\_s*' ..
614 '\d NEWDICT size 0\_s*' ..
615 '\d NEWDICT size 1\_s*' ..
Yegappan Lakshmanan66897192023-12-05 15:51:50 +0100616 '\d SETTYPE dict<dict<any>>\_s*' ..
Bram Moolenaar4f5e3972020-12-21 17:30:50 +0100617 '\d STORE $0\_s*' ..
618 'd.dd\[0\] = 0\_s*' ..
619 '\d PUSHNR 0\_s*' ..
620 '\d PUSHNR 0\_s*' ..
621 '\d LOAD $0\_s*' ..
622 '\d MEMBER dd\_s*' ..
Bram Moolenaarb1b6f4d2021-09-13 18:25:54 +0200623 '\d\+ USEDICT\_s*' ..
624 '\d\+ STOREINDEX any\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +0200625 '\d\+ RETURN void',
Bram Moolenaar4f5e3972020-12-21 17:30:50 +0100626 res)
627enddef
628
Bram Moolenaar0779fab2020-06-18 22:18:18 +0200629def s:ListAssign()
Bram Moolenaarac564082020-09-27 19:05:33 +0200630 var x: string
631 var y: string
632 var l: list<any>
Bram Moolenaar0779fab2020-06-18 22:18:18 +0200633 [x, y; l] = g:stringlist
634enddef
635
636def Test_disassemble_list_assign()
Bram Moolenaarac564082020-09-27 19:05:33 +0200637 var res = execute('disass s:ListAssign')
Bram Moolenaar0779fab2020-06-18 22:18:18 +0200638 assert_match('<SNR>\d*_ListAssign\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +0200639 'var x: string\_s*' ..
Bram Moolenaar0779fab2020-06-18 22:18:18 +0200640 '\d PUSHS "\[NULL\]"\_s*' ..
641 '\d STORE $0\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +0200642 'var y: string\_s*' ..
Bram Moolenaar0779fab2020-06-18 22:18:18 +0200643 '\d PUSHS "\[NULL\]"\_s*' ..
644 '\d STORE $1\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +0200645 'var l: list<any>\_s*' ..
Bram Moolenaar0779fab2020-06-18 22:18:18 +0200646 '\d NEWLIST size 0\_s*' ..
647 '\d STORE $2\_s*' ..
648 '\[x, y; l\] = g:stringlist\_s*' ..
649 '\d LOADG g:stringlist\_s*' ..
Bram Moolenaar5e654232020-09-16 15:22:00 +0200650 '\d CHECKTYPE list<any> stack\[-1\]\_s*' ..
Bram Moolenaar0779fab2020-06-18 22:18:18 +0200651 '\d CHECKLEN >= 2\_s*' ..
652 '\d\+ ITEM 0\_s*' ..
Bram Moolenaarbd3a9d22022-05-17 16:12:39 +0100653 '\d\+ CHECKTYPE string stack\[-1\] var 1\_s*' ..
Bram Moolenaar0779fab2020-06-18 22:18:18 +0200654 '\d\+ STORE $0\_s*' ..
655 '\d\+ ITEM 1\_s*' ..
Bram Moolenaarbd3a9d22022-05-17 16:12:39 +0100656 '\d\+ CHECKTYPE string stack\[-1\] var 2\_s*' ..
Bram Moolenaar0779fab2020-06-18 22:18:18 +0200657 '\d\+ STORE $1\_s*' ..
658 '\d\+ SLICE 2\_s*' ..
659 '\d\+ STORE $2\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +0200660 '\d\+ RETURN void',
Bram Moolenaar0779fab2020-06-18 22:18:18 +0200661 res)
662enddef
663
Bram Moolenaar035bd1c2021-06-21 19:44:11 +0200664def s:ListAssignWithOp()
665 var a = 2
666 var b = 3
667 [a, b] += [4, 5]
668enddef
669
670def Test_disassemble_list_assign_with_op()
671 var res = execute('disass s:ListAssignWithOp')
672 assert_match('<SNR>\d*_ListAssignWithOp\_s*' ..
673 'var a = 2\_s*' ..
674 '\d STORE 2 in $0\_s*' ..
675 'var b = 3\_s*' ..
676 '\d STORE 3 in $1\_s*' ..
677 '\[a, b\] += \[4, 5\]\_s*' ..
678 '\d\+ PUSHNR 4\_s*' ..
679 '\d\+ PUSHNR 5\_s*' ..
680 '\d\+ NEWLIST size 2\_s*' ..
Bram Moolenaar035bd1c2021-06-21 19:44:11 +0200681 '\d\+ LOAD $0\_s*' ..
682 '\d\+ ITEM 0 with op\_s*' ..
683 '\d\+ OPNR +\_s*' ..
684 '\d\+ STORE $0\_s*' ..
685 '\d\+ LOAD $1\_s*' ..
686 '\d\+ ITEM 1 with op\_s*' ..
687 '\d\+ OPNR +\_s*' ..
688 '\d\+ STORE $1\_s*' ..
689 '\d\+ DROP\_s*' ..
690 '\d\+ RETURN void',
691 res)
692enddef
693
Bram Moolenaar1dcae592020-10-19 19:02:42 +0200694def s:ListAdd()
695 var l: list<number> = []
696 add(l, 123)
697 add(l, g:aNumber)
698enddef
699
700def Test_disassemble_list_add()
701 var res = execute('disass s:ListAdd')
702 assert_match('<SNR>\d*_ListAdd\_s*' ..
703 'var l: list<number> = []\_s*' ..
704 '\d NEWLIST size 0\_s*' ..
Bram Moolenaaraa210a32021-01-02 15:41:03 +0100705 '\d SETTYPE list<number>\_s*' ..
Bram Moolenaar1dcae592020-10-19 19:02:42 +0200706 '\d STORE $0\_s*' ..
707 'add(l, 123)\_s*' ..
708 '\d LOAD $0\_s*' ..
709 '\d PUSHNR 123\_s*' ..
710 '\d LISTAPPEND\_s*' ..
711 '\d DROP\_s*' ..
712 'add(l, g:aNumber)\_s*' ..
713 '\d LOAD $0\_s*' ..
714 '\d\+ LOADG g:aNumber\_s*' ..
715 '\d\+ CHECKTYPE number stack\[-1\]\_s*' ..
716 '\d\+ LISTAPPEND\_s*' ..
717 '\d\+ DROP\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +0200718 '\d\+ RETURN void',
Bram Moolenaar1dcae592020-10-19 19:02:42 +0200719 res)
720enddef
721
Bram Moolenaar80b0e5e2020-10-19 20:45:36 +0200722def s:BlobAdd()
723 var b: blob = 0z
724 add(b, 123)
725 add(b, g:aNumber)
726enddef
727
728def Test_disassemble_blob_add()
729 var res = execute('disass s:BlobAdd')
730 assert_match('<SNR>\d*_BlobAdd\_s*' ..
731 'var b: blob = 0z\_s*' ..
732 '\d PUSHBLOB 0z\_s*' ..
733 '\d STORE $0\_s*' ..
734 'add(b, 123)\_s*' ..
735 '\d LOAD $0\_s*' ..
736 '\d PUSHNR 123\_s*' ..
737 '\d BLOBAPPEND\_s*' ..
738 '\d DROP\_s*' ..
739 'add(b, g:aNumber)\_s*' ..
740 '\d LOAD $0\_s*' ..
741 '\d\+ LOADG g:aNumber\_s*' ..
742 '\d\+ CHECKTYPE number stack\[-1\]\_s*' ..
743 '\d\+ BLOBAPPEND\_s*' ..
744 '\d\+ DROP\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +0200745 '\d\+ RETURN void',
Bram Moolenaar80b0e5e2020-10-19 20:45:36 +0200746 res)
747enddef
748
Bram Moolenaarf62d7392021-04-14 12:40:00 +0200749def s:BlobIndexSlice()
750 var b: blob = 0z112233
751 echo b[1]
752 echo b[1 : 2]
753enddef
754
755def Test_disassemble_blob_index_slice()
756 var res = execute('disass s:BlobIndexSlice')
757 assert_match('<SNR>\d*_BlobIndexSlice\_s*' ..
758 'var b: blob = 0z112233\_s*' ..
759 '\d PUSHBLOB 0z112233\_s*' ..
760 '\d STORE $0\_s*' ..
761 'echo b\[1\]\_s*' ..
762 '\d LOAD $0\_s*' ..
763 '\d PUSHNR 1\_s*' ..
764 '\d BLOBINDEX\_s*' ..
765 '\d ECHO 1\_s*' ..
766 'echo b\[1 : 2\]\_s*' ..
767 '\d LOAD $0\_s*' ..
768 '\d PUSHNR 1\_s*' ..
769 '\d\+ PUSHNR 2\_s*' ..
770 '\d\+ BLOBSLICE\_s*' ..
771 '\d\+ ECHO 1\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +0200772 '\d\+ RETURN void',
Bram Moolenaarf62d7392021-04-14 12:40:00 +0200773 res)
774enddef
775
Bram Moolenaard72c1bf2020-04-19 16:28:59 +0200776def s:ScriptFuncUnlet()
777 g:somevar = "value"
778 unlet g:somevar
779 unlet! g:somevar
Bram Moolenaar7bdaea62020-04-19 18:27:26 +0200780 unlet $SOMEVAR
Bram Moolenaarf6ced982022-04-28 12:00:49 +0100781
782 var l = [1, 2, 3]
783 unlet l[2]
784 unlet l[0 : 1]
Bram Moolenaard72c1bf2020-04-19 16:28:59 +0200785enddef
786
787def Test_disassemble_unlet()
Bram Moolenaarac564082020-09-27 19:05:33 +0200788 var res = execute('disass s:ScriptFuncUnlet')
Bram Moolenaarcb790402020-05-15 20:53:00 +0200789 assert_match('<SNR>\d*_ScriptFuncUnlet\_s*' ..
790 'g:somevar = "value"\_s*' ..
791 '\d PUSHS "value"\_s*' ..
792 '\d STOREG g:somevar\_s*' ..
793 'unlet g:somevar\_s*' ..
794 '\d UNLET g:somevar\_s*' ..
795 'unlet! g:somevar\_s*' ..
796 '\d UNLET! g:somevar\_s*' ..
797 'unlet $SOMEVAR\_s*' ..
Bram Moolenaarf6ced982022-04-28 12:00:49 +0100798 '\d UNLETENV $SOMEVAR\_s*' ..
799
800 'var l = \[1, 2, 3]\_s*' ..
801 '\d\+ PUSHNR 1\_s*' ..
802 '\d\+ PUSHNR 2\_s*' ..
803 '\d\+ PUSHNR 3\_s*' ..
804 '\d\+ NEWLIST size 3\_s*' ..
805 '\d\+ SETTYPE list<number>\_s*' ..
806 '\d\+ STORE $0\_s*' ..
807
808 'unlet l\[2]\_s*' ..
809 '\d\+ PUSHNR 2\_s*' ..
810 '\d\+ LOAD $0\_s*' ..
811 '\d\+ UNLETINDEX\_s*' ..
812
813 'unlet l\[0 : 1]\_s*' ..
814 '\d\+ PUSHNR 0\_s*' ..
815 '\d\+ PUSHNR 1\_s*' ..
816 '\d\+ LOAD $0\_s*' ..
817 '\d\+ UNLETRANGE\_s*',
Bram Moolenaard72c1bf2020-04-19 16:28:59 +0200818 res)
819enddef
820
Bram Moolenaaraacc9662021-08-13 19:40:51 +0200821def s:LockLocal()
822 var d = {a: 1}
823 lockvar d.a
Bram Moolenaarf6ced982022-04-28 12:00:49 +0100824 const nr = 22
Bram Moolenaaraacc9662021-08-13 19:40:51 +0200825enddef
826
Bram Moolenaare88c6b72022-02-15 15:37:11 +0000827def Test_disassemble_lock_local()
Bram Moolenaaraacc9662021-08-13 19:40:51 +0200828 var res = execute('disass s:LockLocal')
829 assert_match('<SNR>\d*_LockLocal\_s*' ..
830 'var d = {a: 1}\_s*' ..
831 '\d PUSHS "a"\_s*' ..
832 '\d PUSHNR 1\_s*' ..
833 '\d NEWDICT size 1\_s*' ..
Bram Moolenaare88c6b72022-02-15 15:37:11 +0000834 '\d SETTYPE dict<number>\_s*' ..
Bram Moolenaaraacc9662021-08-13 19:40:51 +0200835 '\d STORE $0\_s*' ..
836 'lockvar d.a\_s*' ..
837 '\d LOAD $0\_s*' ..
Bram Moolenaarf6ced982022-04-28 12:00:49 +0100838 '\d LOCKUNLOCK lockvar 2 d.a\_s*' ..
839
840 'const nr = 22\_s*' ..
841 '\d\+ PUSHNR 22\_s*' ..
842 '\d\+ LOCKCONST\_s*' ..
843 '\d\+ STORE $1',
Bram Moolenaaraacc9662021-08-13 19:40:51 +0200844 res)
845enddef
846
Bram Moolenaar5cab73f2020-02-06 19:25:19 +0100847def s:ScriptFuncTry()
848 try
Bram Moolenaarcb790402020-05-15 20:53:00 +0200849 echo "yes"
Bram Moolenaar5cab73f2020-02-06 19:25:19 +0100850 catch /fail/
Bram Moolenaarcb790402020-05-15 20:53:00 +0200851 echo "no"
Bram Moolenaar5cab73f2020-02-06 19:25:19 +0100852 finally
Bram Moolenaarcb790402020-05-15 20:53:00 +0200853 throw "end"
Bram Moolenaar5cab73f2020-02-06 19:25:19 +0100854 endtry
855enddef
856
Bram Moolenaarf2460a32020-02-07 22:09:54 +0100857def Test_disassemble_try()
Bram Moolenaarac564082020-09-27 19:05:33 +0200858 var res = execute('disass s:ScriptFuncTry')
Bram Moolenaarcb790402020-05-15 20:53:00 +0200859 assert_match('<SNR>\d*_ScriptFuncTry\_s*' ..
860 'try\_s*' ..
Bram Moolenaar7e82c5f2021-02-21 21:32:45 +0100861 '\d TRY catch -> \d\+, finally -> \d\+, endtry -> \d\+\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +0200862 'echo "yes"\_s*' ..
863 '\d PUSHS "yes"\_s*' ..
864 '\d ECHO 1\_s*' ..
865 'catch /fail/\_s*' ..
866 '\d JUMP -> \d\+\_s*' ..
867 '\d PUSH v:exception\_s*' ..
868 '\d PUSHS "fail"\_s*' ..
869 '\d COMPARESTRING =\~\_s*' ..
870 '\d JUMP_IF_FALSE -> \d\+\_s*' ..
871 '\d CATCH\_s*' ..
872 'echo "no"\_s*' ..
873 '\d\+ PUSHS "no"\_s*' ..
874 '\d\+ ECHO 1\_s*' ..
875 'finally\_s*' ..
Bram Moolenaar7e82c5f2021-02-21 21:32:45 +0100876 '\d\+ FINALLY\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +0200877 'throw "end"\_s*' ..
878 '\d\+ PUSHS "end"\_s*' ..
879 '\d\+ THROW\_s*' ..
880 'endtry\_s*' ..
881 '\d\+ ENDTRY',
Bram Moolenaar675f7162020-04-12 22:53:54 +0200882 res)
Bram Moolenaar5cab73f2020-02-06 19:25:19 +0100883enddef
884
885def s:ScriptFuncNew()
Bram Moolenaarac564082020-09-27 19:05:33 +0200886 var ll = [1, "two", 333]
Bram Moolenaare0de1712020-12-02 17:36:54 +0100887 var dd = {one: 1, two: "val"}
Bram Moolenaar5cab73f2020-02-06 19:25:19 +0100888enddef
889
Bram Moolenaarf2460a32020-02-07 22:09:54 +0100890def Test_disassemble_new()
Bram Moolenaarac564082020-09-27 19:05:33 +0200891 var res = execute('disass s:ScriptFuncNew')
Bram Moolenaarcb790402020-05-15 20:53:00 +0200892 assert_match('<SNR>\d*_ScriptFuncNew\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +0200893 'var ll = \[1, "two", 333\]\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +0200894 '\d PUSHNR 1\_s*' ..
895 '\d PUSHS "two"\_s*' ..
896 '\d PUSHNR 333\_s*' ..
897 '\d NEWLIST size 3\_s*' ..
898 '\d STORE $0\_s*' ..
Bram Moolenaare0de1712020-12-02 17:36:54 +0100899 'var dd = {one: 1, two: "val"}\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +0200900 '\d PUSHS "one"\_s*' ..
901 '\d PUSHNR 1\_s*' ..
902 '\d PUSHS "two"\_s*' ..
903 '\d PUSHS "val"\_s*' ..
904 '\d NEWDICT size 2\_s*',
Bram Moolenaar675f7162020-04-12 22:53:54 +0200905 res)
Bram Moolenaar5cab73f2020-02-06 19:25:19 +0100906enddef
907
Bram Moolenaar62aec932022-01-29 21:45:34 +0000908def s:FuncWithArg(arg: any)
Bram Moolenaar5cab73f2020-02-06 19:25:19 +0100909 echo arg
910enddef
911
Bram Moolenaar62aec932022-01-29 21:45:34 +0000912func s:UserFunc()
Bram Moolenaar5cab73f2020-02-06 19:25:19 +0100913 echo 'nothing'
914endfunc
915
Bram Moolenaar62aec932022-01-29 21:45:34 +0000916func s:UserFuncWithArg(arg)
Bram Moolenaar5cab73f2020-02-06 19:25:19 +0100917 echo a:arg
918endfunc
919
920def s:ScriptFuncCall(): string
921 changenr()
922 char2nr("abc")
Bram Moolenaar62aec932022-01-29 21:45:34 +0000923 g:Test_disassemble_new()
Bram Moolenaar5cab73f2020-02-06 19:25:19 +0100924 FuncWithArg(343)
925 ScriptFuncNew()
926 s:ScriptFuncNew()
927 UserFunc()
928 UserFuncWithArg("foo")
Bram Moolenaarac564082020-09-27 19:05:33 +0200929 var FuncRef = function("UserFunc")
Bram Moolenaar5cab73f2020-02-06 19:25:19 +0100930 FuncRef()
Bram Moolenaarac564082020-09-27 19:05:33 +0200931 var FuncRefWithArg = function("UserFuncWithArg")
Bram Moolenaar5cab73f2020-02-06 19:25:19 +0100932 FuncRefWithArg("bar")
933 return "yes"
934enddef
935
Bram Moolenaarf2460a32020-02-07 22:09:54 +0100936def Test_disassemble_call()
Bram Moolenaarac564082020-09-27 19:05:33 +0200937 var res = execute('disass s:ScriptFuncCall')
Bram Moolenaarcb790402020-05-15 20:53:00 +0200938 assert_match('<SNR>\d\+_ScriptFuncCall\_s*' ..
939 'changenr()\_s*' ..
940 '\d BCALL changenr(argc 0)\_s*' ..
941 '\d DROP\_s*' ..
942 'char2nr("abc")\_s*' ..
943 '\d PUSHS "abc"\_s*' ..
944 '\d BCALL char2nr(argc 1)\_s*' ..
945 '\d DROP\_s*' ..
Bram Moolenaar62aec932022-01-29 21:45:34 +0000946 'g:Test_disassemble_new()\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +0200947 '\d DCALL Test_disassemble_new(argc 0)\_s*' ..
948 '\d DROP\_s*' ..
949 'FuncWithArg(343)\_s*' ..
950 '\d\+ PUSHNR 343\_s*' ..
Bram Moolenaar62aec932022-01-29 21:45:34 +0000951 '\d\+ DCALL <SNR>\d\+_FuncWithArg(argc 1)\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +0200952 '\d\+ DROP\_s*' ..
953 'ScriptFuncNew()\_s*' ..
954 '\d\+ DCALL <SNR>\d\+_ScriptFuncNew(argc 0)\_s*' ..
955 '\d\+ DROP\_s*' ..
956 's:ScriptFuncNew()\_s*' ..
957 '\d\+ DCALL <SNR>\d\+_ScriptFuncNew(argc 0)\_s*' ..
958 '\d\+ DROP\_s*' ..
959 'UserFunc()\_s*' ..
Bram Moolenaar62aec932022-01-29 21:45:34 +0000960 '\d\+ UCALL <80><fd>R\d\+_UserFunc(argc 0)\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +0200961 '\d\+ DROP\_s*' ..
962 'UserFuncWithArg("foo")\_s*' ..
963 '\d\+ PUSHS "foo"\_s*' ..
Bram Moolenaar62aec932022-01-29 21:45:34 +0000964 '\d\+ UCALL <80><fd>R\d\+_UserFuncWithArg(argc 1)\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +0200965 '\d\+ DROP\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +0200966 'var FuncRef = function("UserFunc")\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +0200967 '\d\+ PUSHS "UserFunc"\_s*' ..
968 '\d\+ BCALL function(argc 1)\_s*' ..
969 '\d\+ STORE $0\_s*' ..
970 'FuncRef()\_s*' ..
971 '\d\+ LOAD $\d\_s*' ..
972 '\d\+ PCALL (argc 0)\_s*' ..
973 '\d\+ DROP\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +0200974 'var FuncRefWithArg = function("UserFuncWithArg")\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +0200975 '\d\+ PUSHS "UserFuncWithArg"\_s*' ..
976 '\d\+ BCALL function(argc 1)\_s*' ..
977 '\d\+ STORE $1\_s*' ..
978 'FuncRefWithArg("bar")\_s*' ..
979 '\d\+ PUSHS "bar"\_s*' ..
980 '\d\+ LOAD $\d\_s*' ..
981 '\d\+ PCALL (argc 1)\_s*' ..
982 '\d\+ DROP\_s*' ..
983 'return "yes"\_s*' ..
984 '\d\+ PUSHS "yes"\_s*' ..
985 '\d\+ RETURN',
Bram Moolenaar675f7162020-04-12 22:53:54 +0200986 res)
Bram Moolenaar5cab73f2020-02-06 19:25:19 +0100987enddef
988
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +0200989
990def s:CreateRefs()
991 var local = 'a'
992 def Append(arg: string)
993 local ..= arg
994 enddef
995 g:Append = Append
996 def Get(): string
997 return local
998 enddef
999 g:Get = Get
1000enddef
1001
1002def Test_disassemble_closure()
1003 CreateRefs()
1004 var res = execute('disass g:Append')
1005 assert_match('<lambda>\d\_s*' ..
1006 'local ..= arg\_s*' ..
Bram Moolenaarab360522021-01-10 14:02:28 +01001007 '\d LOADOUTER level 1 $0\_s*' ..
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02001008 '\d LOAD arg\[-1\]\_s*' ..
LemonBoy372bcce2022-04-25 12:43:20 +01001009 '\d CONCAT size 2\_s*' ..
Bram Moolenaarab360522021-01-10 14:02:28 +01001010 '\d STOREOUTER level 1 $0\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +02001011 '\d RETURN void',
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02001012 res)
1013
1014 res = execute('disass g:Get')
1015 assert_match('<lambda>\d\_s*' ..
1016 'return local\_s*' ..
Bram Moolenaarab360522021-01-10 14:02:28 +01001017 '\d LOADOUTER level 1 $0\_s*' ..
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02001018 '\d RETURN',
1019 res)
1020
1021 unlet g:Append
1022 unlet g:Get
1023enddef
Bram Moolenaarb68b3462020-05-06 21:06:30 +02001024
Bram Moolenaar95e4dd82022-04-27 22:15:40 +01001025def s:ClosureArg(arg: string)
1026 var Ref = () => arg .. "x"
1027enddef
1028
1029def Test_disassemble_closure_arg()
1030 var res = execute('disass s:ClosureArg')
1031 assert_match('<SNR>\d\+_ClosureArg\_s*' ..
1032 'var Ref = () => arg .. "x"\_s*' ..
1033 '\d FUNCREF <lambda>\d\+',
1034 res)
1035 var lres = execute('disass ' .. matchstr(res, '<lambda>\d\+'))
1036 assert_match('<lambda>\d\+\_s*' ..
1037 'return arg .. "x"\_s*' ..
1038 '\d LOADOUTER level 1 arg\[-1]\_s*' ..
1039 '\d PUSHS "x"\_s*' ..
1040 '\d CONCAT size 2\_s*' ..
1041 '\d RETURN',
1042 lres)
1043enddef
Bram Moolenaar8ed04582020-02-22 19:07:28 +01001044
Bram Moolenaar8abb5842022-09-17 12:39:58 +01001045def s:ClosureInLoop()
1046 for i in range(5)
1047 var ii = i
1048 continue
1049 break
1050 if g:val
1051 return
1052 endif
1053 g:Ref = () => ii
1054 continue
1055 break
1056 if g:val
1057 return
1058 endif
1059 endfor
1060enddef
1061
1062" Mainly check that ENDLOOP is only produced after a closure was created.
1063def Test_disassemble_closure_in_loop()
1064 var res = execute('disass s:ClosureInLoop')
1065 assert_match('<SNR>\d\+_ClosureInLoop\_s*' ..
1066 'for i in range(5)\_s*' ..
1067 '\d\+ STORE -1 in $0\_s*' ..
1068 '\d\+ PUSHNR 5\_s*' ..
1069 '\d\+ BCALL range(argc 1)\_s*' ..
1070 '\d\+ FOR $0 -> \d\+\_s*' ..
1071 '\d\+ STORE $2\_s*' ..
1072
1073 'var ii = i\_s*' ..
1074 '\d\+ LOAD $2\_s*' ..
1075 '\d\+ STORE $3\_s*' ..
1076
1077 'continue\_s*' ..
1078 '\d\+ JUMP -> \d\+\_s*' ..
1079
1080 'break\_s*' ..
1081 '\d\+ JUMP -> \d\+\_s*' ..
1082
1083 'if g:val\_s*' ..
1084 '\d\+ LOADG g:val\_s*' ..
1085 '\d\+ COND2BOOL\_s*' ..
1086 '\d\+ JUMP_IF_FALSE -> \d\+\_s*' ..
1087
1088 ' return\_s*' ..
1089 '\d\+ PUSHNR 0\_s*' ..
1090 '\d\+ RETURN\_s*' ..
1091
1092 'endif\_s*' ..
1093 'g:Ref = () => ii\_s*' ..
Julio B1fa22e32024-04-18 22:05:12 +02001094 '\d\+ FUNCREF <lambda>\d\+ vars $3-$3\_s*' ..
Bram Moolenaar8abb5842022-09-17 12:39:58 +01001095 '\d\+ STOREG g:Ref\_s*' ..
1096
1097 'continue\_s*' ..
Bram Moolenaarcc341812022-09-19 15:54:34 +01001098 '\d\+ ENDLOOP ref $1 save $3-$3 depth 0\_s*' ..
Bram Moolenaar8abb5842022-09-17 12:39:58 +01001099 '\d\+ JUMP -> \d\+\_s*' ..
1100
1101 'break\_s*' ..
Bram Moolenaarcc341812022-09-19 15:54:34 +01001102 '\d\+ ENDLOOP ref $1 save $3-$3 depth 0\_s*' ..
Bram Moolenaar8abb5842022-09-17 12:39:58 +01001103 '\d\+ JUMP -> \d\+\_s*' ..
1104
1105 'if g:val\_s*' ..
1106 '\d\+ LOADG g:val\_s*' ..
1107 '\d\+ COND2BOOL\_s*' ..
1108 '\d\+ JUMP_IF_FALSE -> \d\+\_s*' ..
1109
1110 ' return\_s*' ..
1111 '\d\+ PUSHNR 0\_s*' ..
Bram Moolenaarcc341812022-09-19 15:54:34 +01001112 '\d\+ ENDLOOP ref $1 save $3-$3 depth 0\_s*' ..
Bram Moolenaar8abb5842022-09-17 12:39:58 +01001113 '\d\+ RETURN\_s*' ..
1114
1115 'endif\_s*' ..
1116 'endfor\_s*' ..
Bram Moolenaarcc341812022-09-19 15:54:34 +01001117 '\d\+ ENDLOOP ref $1 save $3-$3 depth 0\_s*' ..
Bram Moolenaar8abb5842022-09-17 12:39:58 +01001118 '\d\+ JUMP -> \d\+\_s*' ..
1119 '\d\+ DROP\_s*' ..
1120 '\d\+ RETURN void',
1121 res)
1122enddef
1123
Bram Moolenaarbd5da372020-03-31 23:13:10 +02001124def EchoArg(arg: string): string
1125 return arg
1126enddef
Bram Moolenaar62aec932022-01-29 21:45:34 +00001127def s:RefThis(): func
Bram Moolenaarbd5da372020-03-31 23:13:10 +02001128 return function('EchoArg')
1129enddef
1130def s:ScriptPCall()
1131 RefThis()("text")
1132enddef
1133
1134def Test_disassemble_pcall()
Bram Moolenaarac564082020-09-27 19:05:33 +02001135 var res = execute('disass s:ScriptPCall')
Bram Moolenaarcb790402020-05-15 20:53:00 +02001136 assert_match('<SNR>\d\+_ScriptPCall\_s*' ..
1137 'RefThis()("text")\_s*' ..
Bram Moolenaar62aec932022-01-29 21:45:34 +00001138 '\d DCALL <SNR>\d\+_RefThis(argc 0)\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +02001139 '\d PUSHS "text"\_s*' ..
1140 '\d PCALL top (argc 1)\_s*' ..
1141 '\d PCALL end\_s*' ..
1142 '\d DROP\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +02001143 '\d RETURN void',
Bram Moolenaar675f7162020-04-12 22:53:54 +02001144 res)
Bram Moolenaarbd5da372020-03-31 23:13:10 +02001145enddef
1146
1147
Bram Moolenaara26b9702020-04-18 19:53:28 +02001148def s:FuncWithForwardCall(): string
1149 return g:DefinedLater("yes")
Bram Moolenaar7eeefd42020-02-26 21:24:23 +01001150enddef
1151
1152def DefinedLater(arg: string): string
1153 return arg
1154enddef
1155
1156def Test_disassemble_update_instr()
Bram Moolenaarac564082020-09-27 19:05:33 +02001157 var res = execute('disass s:FuncWithForwardCall')
Bram Moolenaarcb790402020-05-15 20:53:00 +02001158 assert_match('FuncWithForwardCall\_s*' ..
1159 'return g:DefinedLater("yes")\_s*' ..
1160 '\d PUSHS "yes"\_s*' ..
Bram Moolenaar822ba242020-05-24 23:00:18 +02001161 '\d DCALL DefinedLater(argc 1)\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +02001162 '\d RETURN',
Bram Moolenaar675f7162020-04-12 22:53:54 +02001163 res)
Bram Moolenaar7eeefd42020-02-26 21:24:23 +01001164
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001165 # Calling the function will change UCALL into the faster DCALL
Bram Moolenaar7eeefd42020-02-26 21:24:23 +01001166 assert_equal('yes', FuncWithForwardCall())
1167
Bram Moolenaara26b9702020-04-18 19:53:28 +02001168 res = execute('disass s:FuncWithForwardCall')
Bram Moolenaarcb790402020-05-15 20:53:00 +02001169 assert_match('FuncWithForwardCall\_s*' ..
1170 'return g:DefinedLater("yes")\_s*' ..
1171 '\d PUSHS "yes"\_s*' ..
1172 '\d DCALL DefinedLater(argc 1)\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +02001173 '\d RETURN',
Bram Moolenaar675f7162020-04-12 22:53:54 +02001174 res)
Bram Moolenaar7eeefd42020-02-26 21:24:23 +01001175enddef
1176
1177
Bram Moolenaar9ce47ec2021-04-20 22:16:39 +02001178def FuncWithDefault(l: number, arg: string = "default", nr = 77): string
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +02001179 return arg .. nr
Bram Moolenaar8ed04582020-02-22 19:07:28 +01001180enddef
1181
1182def Test_disassemble_call_default()
Bram Moolenaarac564082020-09-27 19:05:33 +02001183 var res = execute('disass FuncWithDefault')
Bram Moolenaarcb790402020-05-15 20:53:00 +02001184 assert_match('FuncWithDefault\_s*' ..
Bram Moolenaar9ce47ec2021-04-20 22:16:39 +02001185 ' arg = "default"\_s*' ..
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +02001186 '\d JUMP_IF_ARG_SET arg\[-2\] -> 3\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +02001187 '\d PUSHS "default"\_s*' ..
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +02001188 '\d STORE arg\[-2]\_s*' ..
Bram Moolenaar9ce47ec2021-04-20 22:16:39 +02001189 ' nr = 77\_s*' ..
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +02001190 '3 JUMP_IF_ARG_SET arg\[-1\] -> 6\_s*' ..
1191 '\d PUSHNR 77\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +02001192 '\d STORE arg\[-1]\_s*' ..
Bram Moolenaar9ce47ec2021-04-20 22:16:39 +02001193 ' return arg .. nr\_s*' ..
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +02001194 '6 LOAD arg\[-2]\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +02001195 '\d LOAD arg\[-1]\_s*' ..
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +02001196 '\d 2STRING stack\[-1]\_s*' ..
LemonBoy372bcce2022-04-25 12:43:20 +01001197 '\d\+ CONCAT size 2\_s*' ..
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +02001198 '\d\+ RETURN',
Bram Moolenaar675f7162020-04-12 22:53:54 +02001199 res)
Bram Moolenaar8ed04582020-02-22 19:07:28 +01001200enddef
1201
1202
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001203def s:HasEval()
Bram Moolenaar158906c2020-02-06 20:39:45 +01001204 if has("eval")
1205 echo "yes"
1206 else
1207 echo "no"
1208 endif
1209enddef
1210
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001211def s:HasNothing()
Bram Moolenaar158906c2020-02-06 20:39:45 +01001212 if has("nothing")
1213 echo "yes"
1214 else
1215 echo "no"
1216 endif
1217enddef
1218
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001219def s:HasSomething()
Bram Moolenaar158906c2020-02-06 20:39:45 +01001220 if has("nothing")
1221 echo "nothing"
1222 elseif has("something")
1223 echo "something"
1224 elseif has("eval")
1225 echo "eval"
1226 elseif has("less")
1227 echo "less"
1228 endif
1229enddef
1230
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001231def s:HasGuiRunning()
Bram Moolenaar8cebd432020-11-08 12:49:47 +01001232 if has("gui_running")
1233 echo "yes"
1234 else
1235 echo "no"
1236 endif
1237enddef
1238
LemonBoy58f331a2022-04-02 21:59:06 +01001239def s:LenConstant(): number
1240 return len("foo") + len("fighters")
1241enddef
1242
Bram Moolenaarf2460a32020-02-07 22:09:54 +01001243def Test_disassemble_const_expr()
LemonBoy58f331a2022-04-02 21:59:06 +01001244 var instr = execute('disassemble LenConstant')
1245 assert_match('LenConstant\_s*' ..
1246 'return len("foo") + len("fighters")\_s*' ..
1247 '\d PUSHNR 11\_s*',
1248 instr)
1249 assert_notmatch('BCALL len', instr)
1250
Bram Moolenaard2c61702020-09-06 15:58:36 +02001251 assert_equal("\nyes", execute('HasEval()'))
LemonBoy58f331a2022-04-02 21:59:06 +01001252 instr = execute('disassemble HasEval')
Bram Moolenaarcb790402020-05-15 20:53:00 +02001253 assert_match('HasEval\_s*' ..
1254 'if has("eval")\_s*' ..
1255 'echo "yes"\_s*' ..
1256 '\d PUSHS "yes"\_s*' ..
1257 '\d ECHO 1\_s*' ..
1258 'else\_s*' ..
1259 'echo "no"\_s*' ..
1260 'endif\_s*',
Bram Moolenaar675f7162020-04-12 22:53:54 +02001261 instr)
Bram Moolenaar158906c2020-02-06 20:39:45 +01001262 assert_notmatch('JUMP', instr)
1263
Bram Moolenaard2c61702020-09-06 15:58:36 +02001264 assert_equal("\nno", execute('HasNothing()'))
Bram Moolenaar158906c2020-02-06 20:39:45 +01001265 instr = execute('disassemble HasNothing')
Bram Moolenaarcb790402020-05-15 20:53:00 +02001266 assert_match('HasNothing\_s*' ..
1267 'if has("nothing")\_s*' ..
1268 'echo "yes"\_s*' ..
1269 'else\_s*' ..
1270 'echo "no"\_s*' ..
1271 '\d PUSHS "no"\_s*' ..
1272 '\d ECHO 1\_s*' ..
1273 'endif',
Bram Moolenaar675f7162020-04-12 22:53:54 +02001274 instr)
Bram Moolenaar158906c2020-02-06 20:39:45 +01001275 assert_notmatch('PUSHS "yes"', instr)
1276 assert_notmatch('JUMP', instr)
1277
Bram Moolenaard2c61702020-09-06 15:58:36 +02001278 assert_equal("\neval", execute('HasSomething()'))
Bram Moolenaar158906c2020-02-06 20:39:45 +01001279 instr = execute('disassemble HasSomething')
Bram Moolenaar675f7162020-04-12 22:53:54 +02001280 assert_match('HasSomething.*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +02001281 'if has("nothing")\_s*' ..
1282 'echo "nothing"\_s*' ..
1283 'elseif has("something")\_s*' ..
1284 'echo "something"\_s*' ..
1285 'elseif has("eval")\_s*' ..
1286 'echo "eval"\_s*' ..
1287 '\d PUSHS "eval"\_s*' ..
1288 '\d ECHO 1\_s*' ..
1289 'elseif has("less").*' ..
1290 'echo "less"\_s*' ..
1291 'endif',
Bram Moolenaar675f7162020-04-12 22:53:54 +02001292 instr)
Bram Moolenaar158906c2020-02-06 20:39:45 +01001293 assert_notmatch('PUSHS "nothing"', instr)
1294 assert_notmatch('PUSHS "something"', instr)
1295 assert_notmatch('PUSHS "less"', instr)
1296 assert_notmatch('JUMP', instr)
Bram Moolenaar8cebd432020-11-08 12:49:47 +01001297
1298 var result: string
1299 var instr_expected: string
1300 if has('gui')
1301 if has('gui_running')
1302 # GUI already running, always returns "yes"
1303 result = "\nyes"
1304 instr_expected = 'HasGuiRunning.*' ..
1305 'if has("gui_running")\_s*' ..
1306 ' echo "yes"\_s*' ..
1307 '\d PUSHS "yes"\_s*' ..
1308 '\d ECHO 1\_s*' ..
1309 'else\_s*' ..
1310 ' echo "no"\_s*' ..
1311 'endif'
1312 else
1313 result = "\nno"
1314 if has('unix')
1315 # GUI not running but can start later, call has()
1316 instr_expected = 'HasGuiRunning.*' ..
1317 'if has("gui_running")\_s*' ..
1318 '\d PUSHS "gui_running"\_s*' ..
1319 '\d BCALL has(argc 1)\_s*' ..
Bram Moolenaaraf8ea0d2021-04-11 18:24:46 +02001320 '\d COND2BOOL\_s*' ..
Bram Moolenaar8cebd432020-11-08 12:49:47 +01001321 '\d JUMP_IF_FALSE -> \d\_s*' ..
1322 ' echo "yes"\_s*' ..
1323 '\d PUSHS "yes"\_s*' ..
1324 '\d ECHO 1\_s*' ..
1325 'else\_s*' ..
1326 '\d JUMP -> \d\_s*' ..
1327 ' echo "no"\_s*' ..
1328 '\d PUSHS "no"\_s*' ..
1329 '\d ECHO 1\_s*' ..
1330 'endif'
1331 else
1332 # GUI not running, always return "no"
1333 instr_expected = 'HasGuiRunning.*' ..
1334 'if has("gui_running")\_s*' ..
1335 ' echo "yes"\_s*' ..
1336 'else\_s*' ..
1337 ' echo "no"\_s*' ..
1338 '\d PUSHS "no"\_s*' ..
1339 '\d ECHO 1\_s*' ..
1340 'endif'
1341 endif
1342 endif
1343 else
1344 # GUI not supported, always return "no"
1345 result = "\nno"
1346 instr_expected = 'HasGuiRunning.*' ..
1347 'if has("gui_running")\_s*' ..
1348 ' echo "yes"\_s*' ..
1349 'else\_s*' ..
1350 ' echo "no"\_s*' ..
1351 '\d PUSHS "no"\_s*' ..
1352 '\d ECHO 1\_s*' ..
1353 'endif'
1354 endif
1355
1356 assert_equal(result, execute('HasGuiRunning()'))
1357 instr = execute('disassemble HasGuiRunning')
1358 assert_match(instr_expected, instr)
Bram Moolenaar158906c2020-02-06 20:39:45 +01001359enddef
1360
Bram Moolenaarefd88552020-06-18 20:50:10 +02001361def ReturnInIf(): string
Bram Moolenaar8e02faf2020-11-18 16:35:02 +01001362 if 1 < 0
1363 return "maybe"
1364 endif
Bram Moolenaarefd88552020-06-18 20:50:10 +02001365 if g:cond
1366 return "yes"
1367 else
1368 return "no"
1369 endif
1370enddef
1371
1372def Test_disassemble_return_in_if()
Bram Moolenaarac564082020-09-27 19:05:33 +02001373 var instr = execute('disassemble ReturnInIf')
Bram Moolenaarefd88552020-06-18 20:50:10 +02001374 assert_match('ReturnInIf\_s*' ..
Bram Moolenaar8e02faf2020-11-18 16:35:02 +01001375 'if 1 < 0\_s*' ..
1376 ' return "maybe"\_s*' ..
1377 'endif\_s*' ..
Bram Moolenaarefd88552020-06-18 20:50:10 +02001378 'if g:cond\_s*' ..
1379 '0 LOADG g:cond\_s*' ..
Bram Moolenaarea2d4072020-11-12 12:08:51 +01001380 '1 COND2BOOL\_s*' ..
1381 '2 JUMP_IF_FALSE -> 5\_s*' ..
Bram Moolenaarefd88552020-06-18 20:50:10 +02001382 'return "yes"\_s*' ..
Bram Moolenaarea2d4072020-11-12 12:08:51 +01001383 '3 PUSHS "yes"\_s*' ..
1384 '4 RETURN\_s*' ..
Bram Moolenaarefd88552020-06-18 20:50:10 +02001385 'else\_s*' ..
1386 ' return "no"\_s*' ..
Bram Moolenaarea2d4072020-11-12 12:08:51 +01001387 '5 PUSHS "no"\_s*' ..
1388 '6 RETURN$',
Bram Moolenaarefd88552020-06-18 20:50:10 +02001389 instr)
1390enddef
1391
Bram Moolenaarf51cb4e2020-03-01 17:55:14 +01001392def WithFunc()
Bram Moolenaarac564082020-09-27 19:05:33 +02001393 var Funky1: func
1394 var Funky2: func = function("len")
1395 var Party2: func = funcref("UserFunc")
Bram Moolenaarf51cb4e2020-03-01 17:55:14 +01001396enddef
1397
1398def Test_disassemble_function()
Bram Moolenaarac564082020-09-27 19:05:33 +02001399 var instr = execute('disassemble WithFunc')
Bram Moolenaarcb790402020-05-15 20:53:00 +02001400 assert_match('WithFunc\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02001401 'var Funky1: func\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +02001402 '0 PUSHFUNC "\[none]"\_s*' ..
1403 '1 STORE $0\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02001404 'var Funky2: func = function("len")\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +02001405 '2 PUSHS "len"\_s*' ..
1406 '3 BCALL function(argc 1)\_s*' ..
1407 '4 STORE $1\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02001408 'var Party2: func = funcref("UserFunc")\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +02001409 '\d PUSHS "UserFunc"\_s*' ..
1410 '\d BCALL funcref(argc 1)\_s*' ..
1411 '\d STORE $2\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +02001412 '\d RETURN void',
Bram Moolenaar675f7162020-04-12 22:53:54 +02001413 instr)
Bram Moolenaarf51cb4e2020-03-01 17:55:14 +01001414enddef
1415
1416if has('channel')
1417 def WithChannel()
Bram Moolenaarac564082020-09-27 19:05:33 +02001418 var job1: job
1419 var job2: job = job_start("donothing")
1420 var chan1: channel
Bram Moolenaarf51cb4e2020-03-01 17:55:14 +01001421 enddef
1422endif
1423
1424def Test_disassemble_channel()
1425 CheckFeature channel
1426
Bram Moolenaarac564082020-09-27 19:05:33 +02001427 var instr = execute('disassemble WithChannel')
Bram Moolenaarcb790402020-05-15 20:53:00 +02001428 assert_match('WithChannel\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02001429 'var job1: job\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +02001430 '\d PUSHJOB "no process"\_s*' ..
1431 '\d STORE $0\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02001432 'var job2: job = job_start("donothing")\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +02001433 '\d PUSHS "donothing"\_s*' ..
1434 '\d BCALL job_start(argc 1)\_s*' ..
1435 '\d STORE $1\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02001436 'var chan1: channel\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +02001437 '\d PUSHCHANNEL 0\_s*' ..
1438 '\d STORE $2\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +02001439 '\d RETURN void',
Bram Moolenaar675f7162020-04-12 22:53:54 +02001440 instr)
Bram Moolenaarf51cb4e2020-03-01 17:55:14 +01001441enddef
1442
Bram Moolenaar62aec932022-01-29 21:45:34 +00001443def s:WithLambda(): string
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001444 var F = (a) => "X" .. a .. "X"
Bram Moolenaar777770f2020-02-06 21:27:08 +01001445 return F("x")
1446enddef
1447
Bram Moolenaarf2460a32020-02-07 22:09:54 +01001448def Test_disassemble_lambda()
Bram Moolenaar777770f2020-02-06 21:27:08 +01001449 assert_equal("XxX", WithLambda())
Bram Moolenaarac564082020-09-27 19:05:33 +02001450 var instr = execute('disassemble WithLambda')
Bram Moolenaarcb790402020-05-15 20:53:00 +02001451 assert_match('WithLambda\_s*' ..
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001452 'var F = (a) => "X" .. a .. "X"\_s*' ..
Bram Moolenaar148ce7a2020-09-23 21:57:23 +02001453 '\d FUNCREF <lambda>\d\+\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +02001454 '\d STORE $0\_s*' ..
1455 'return F("x")\_s*' ..
1456 '\d PUSHS "x"\_s*' ..
1457 '\d LOAD $0\_s*' ..
1458 '\d PCALL (argc 1)\_s*' ..
Bram Moolenaar822ba242020-05-24 23:00:18 +02001459 '\d RETURN',
Bram Moolenaar675f7162020-04-12 22:53:54 +02001460 instr)
Bram Moolenaarbfd65582020-07-13 18:18:00 +02001461
Bram Moolenaarac564082020-09-27 19:05:33 +02001462 var name = substitute(instr, '.*\(<lambda>\d\+\).*', '\1', '')
Bram Moolenaarbfd65582020-07-13 18:18:00 +02001463 instr = execute('disassemble ' .. name)
1464 assert_match('<lambda>\d\+\_s*' ..
1465 'return "X" .. a .. "X"\_s*' ..
1466 '\d PUSHS "X"\_s*' ..
1467 '\d LOAD arg\[-1\]\_s*' ..
Bram Moolenaar418f1df2020-08-12 21:34:49 +02001468 '\d 2STRING_ANY stack\[-1\]\_s*' ..
LemonBoy372bcce2022-04-25 12:43:20 +01001469 '\d CONCAT size 2\_s*' ..
Bram Moolenaarbfd65582020-07-13 18:18:00 +02001470 '\d PUSHS "X"\_s*' ..
LemonBoy372bcce2022-04-25 12:43:20 +01001471 '\d CONCAT size 2\_s*' ..
Bram Moolenaarbfd65582020-07-13 18:18:00 +02001472 '\d RETURN',
1473 instr)
Bram Moolenaar777770f2020-02-06 21:27:08 +01001474enddef
1475
Bram Moolenaar62aec932022-01-29 21:45:34 +00001476def s:LambdaWithType(): number
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001477 var Ref = (a: number) => a + 10
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001478 return Ref(g:value)
1479enddef
1480
1481def Test_disassemble_lambda_with_type()
1482 g:value = 5
1483 assert_equal(15, LambdaWithType())
1484 var instr = execute('disassemble LambdaWithType')
1485 assert_match('LambdaWithType\_s*' ..
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001486 'var Ref = (a: number) => a + 10\_s*' ..
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001487 '\d FUNCREF <lambda>\d\+\_s*' ..
1488 '\d STORE $0\_s*' ..
1489 'return Ref(g:value)\_s*' ..
1490 '\d LOADG g:value\_s*' ..
1491 '\d LOAD $0\_s*' ..
Bram Moolenaare32e5162021-01-21 20:21:29 +01001492 '\d CHECKTYPE number stack\[-2\] arg 1\_s*' ..
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001493 '\d PCALL (argc 1)\_s*' ..
1494 '\d RETURN',
1495 instr)
1496enddef
1497
Bram Moolenaar38ddf332020-07-31 22:05:04 +02001498def NestedOuter()
1499 def g:Inner()
1500 echomsg "inner"
1501 enddef
1502enddef
1503
Bram Moolenaar8863bda2021-03-17 18:42:08 +01001504def Test_disassemble_nested_func()
Bram Moolenaarac564082020-09-27 19:05:33 +02001505 var instr = execute('disassemble NestedOuter')
Bram Moolenaar38ddf332020-07-31 22:05:04 +02001506 assert_match('NestedOuter\_s*' ..
1507 'def g:Inner()\_s*' ..
1508 'echomsg "inner"\_s*' ..
1509 'enddef\_s*' ..
1510 '\d NEWFUNC <lambda>\d\+ Inner\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +02001511 '\d RETURN void',
Bram Moolenaar38ddf332020-07-31 22:05:04 +02001512 instr)
1513enddef
1514
Bram Moolenaar6abdcf82020-11-22 18:15:44 +01001515def NestedDefList()
1516 def
1517 def Info
1518 def /Info
1519 def /Info/
1520enddef
1521
Bram Moolenaar8863bda2021-03-17 18:42:08 +01001522def Test_disassemble_nested_def_list()
Bram Moolenaar6abdcf82020-11-22 18:15:44 +01001523 var instr = execute('disassemble NestedDefList')
1524 assert_match('NestedDefList\_s*' ..
1525 'def\_s*' ..
1526 '\d DEF \_s*' ..
1527 'def Info\_s*' ..
1528 '\d DEF Info\_s*' ..
1529 'def /Info\_s*' ..
1530 '\d DEF /Info\_s*' ..
1531 'def /Info/\_s*' ..
1532 '\d DEF /Info/\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +02001533 '\d RETURN void',
Bram Moolenaar6abdcf82020-11-22 18:15:44 +01001534 instr)
1535enddef
1536
Bram Moolenaar62aec932022-01-29 21:45:34 +00001537def s:AndOr(arg: any): string
Bram Moolenaar777770f2020-02-06 21:27:08 +01001538 if arg == 1 && arg != 2 || arg == 4
1539 return 'yes'
1540 endif
1541 return 'no'
1542enddef
1543
Bram Moolenaarf2460a32020-02-07 22:09:54 +01001544def Test_disassemble_and_or()
Bram Moolenaar777770f2020-02-06 21:27:08 +01001545 assert_equal("yes", AndOr(1))
1546 assert_equal("no", AndOr(2))
1547 assert_equal("yes", AndOr(4))
Bram Moolenaarac564082020-09-27 19:05:33 +02001548 var instr = execute('disassemble AndOr')
Bram Moolenaarcb790402020-05-15 20:53:00 +02001549 assert_match('AndOr\_s*' ..
1550 'if arg == 1 && arg != 2 || arg == 4\_s*' ..
1551 '\d LOAD arg\[-1]\_s*' ..
1552 '\d PUSHNR 1\_s*' ..
1553 '\d COMPAREANY ==\_s*' ..
Bram Moolenaar2bb26582020-10-03 22:52:39 +02001554 '\d JUMP_IF_COND_FALSE -> \d\+\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +02001555 '\d LOAD arg\[-1]\_s*' ..
1556 '\d PUSHNR 2\_s*' ..
1557 '\d COMPAREANY !=\_s*' ..
Bram Moolenaar2bb26582020-10-03 22:52:39 +02001558 '\d JUMP_IF_COND_TRUE -> \d\+\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +02001559 '\d LOAD arg\[-1]\_s*' ..
1560 '\d\+ PUSHNR 4\_s*' ..
1561 '\d\+ COMPAREANY ==\_s*' ..
1562 '\d\+ JUMP_IF_FALSE -> \d\+',
Bram Moolenaar675f7162020-04-12 22:53:54 +02001563 instr)
Bram Moolenaar777770f2020-02-06 21:27:08 +01001564enddef
1565
Bram Moolenaar62aec932022-01-29 21:45:34 +00001566def s:AndConstant(arg: any): string
Bram Moolenaar1a7ee4d2021-09-16 16:15:07 +02001567 if true && arg
1568 return "yes"
1569 endif
1570 if false && arg
1571 return "never"
1572 endif
1573 return "no"
1574enddef
1575
1576def Test_disassemble_and_constant()
1577 assert_equal("yes", AndConstant(1))
1578 assert_equal("no", AndConstant(false))
1579 var instr = execute('disassemble AndConstant')
1580 assert_match('AndConstant\_s*' ..
1581 'if true && arg\_s*' ..
1582 '0 LOAD arg\[-1\]\_s*' ..
1583 '1 COND2BOOL\_s*' ..
1584 '2 JUMP_IF_FALSE -> 5\_s*' ..
1585 'return "yes"\_s*' ..
1586 '3 PUSHS "yes"\_s*' ..
1587 '4 RETURN\_s*' ..
1588 'endif\_s*' ..
1589 'if false && arg\_s*' ..
1590 'return "never"\_s*' ..
1591 'endif\_s*' ..
1592 'return "no"\_s*' ..
1593 '5 PUSHS "no"\_s*' ..
1594 '6 RETURN',
1595 instr)
1596enddef
1597
Bram Moolenaar62aec932022-01-29 21:45:34 +00001598def s:ForLoop(): list<number>
Bram Moolenaarac564082020-09-27 19:05:33 +02001599 var res: list<number>
Bram Moolenaar04d05222020-02-06 22:06:54 +01001600 for i in range(3)
1601 res->add(i)
1602 endfor
1603 return res
1604enddef
1605
Bram Moolenaarf2460a32020-02-07 22:09:54 +01001606def Test_disassemble_for_loop()
Bram Moolenaar04d05222020-02-06 22:06:54 +01001607 assert_equal([0, 1, 2], ForLoop())
Bram Moolenaarac564082020-09-27 19:05:33 +02001608 var instr = execute('disassemble ForLoop')
Bram Moolenaarcb790402020-05-15 20:53:00 +02001609 assert_match('ForLoop\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02001610 'var res: list<number>\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +02001611 '\d NEWLIST size 0\_s*' ..
Bram Moolenaaraa210a32021-01-02 15:41:03 +01001612 '\d SETTYPE list<number>\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +02001613 '\d STORE $0\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01001614
Bram Moolenaarcb790402020-05-15 20:53:00 +02001615 'for i in range(3)\_s*' ..
1616 '\d STORE -1 in $1\_s*' ..
1617 '\d PUSHNR 3\_s*' ..
1618 '\d BCALL range(argc 1)\_s*' ..
1619 '\d FOR $1 -> \d\+\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01001620 '\d STORE $3\_s*' ..
1621
Bram Moolenaarcb790402020-05-15 20:53:00 +02001622 'res->add(i)\_s*' ..
1623 '\d LOAD $0\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01001624 '\d LOAD $3\_s*' ..
Bram Moolenaar1dcae592020-10-19 19:02:42 +02001625 '\d\+ LISTAPPEND\_s*' ..
Bram Moolenaarcb790402020-05-15 20:53:00 +02001626 '\d\+ DROP\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01001627
Bram Moolenaarcb790402020-05-15 20:53:00 +02001628 'endfor\_s*' ..
1629 '\d\+ JUMP -> \d\+\_s*' ..
1630 '\d\+ DROP',
Bram Moolenaar675f7162020-04-12 22:53:54 +02001631 instr)
Bram Moolenaar04d05222020-02-06 22:06:54 +01001632enddef
1633
Bram Moolenaar62aec932022-01-29 21:45:34 +00001634def s:ForLoopEval(): string
Bram Moolenaarac564082020-09-27 19:05:33 +02001635 var res = ""
Bram Moolenaar0ad3e892020-07-05 21:38:11 +02001636 for str in eval('["one", "two"]')
1637 res ..= str
1638 endfor
1639 return res
1640enddef
1641
1642def Test_disassemble_for_loop_eval()
1643 assert_equal('onetwo', ForLoopEval())
Bram Moolenaarac564082020-09-27 19:05:33 +02001644 var instr = execute('disassemble ForLoopEval')
Bram Moolenaar0ad3e892020-07-05 21:38:11 +02001645 assert_match('ForLoopEval\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02001646 'var res = ""\_s*' ..
Bram Moolenaar0ad3e892020-07-05 21:38:11 +02001647 '\d PUSHS ""\_s*' ..
1648 '\d STORE $0\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01001649
Bram Moolenaar0ad3e892020-07-05 21:38:11 +02001650 'for str in eval(''\["one", "two"\]'')\_s*' ..
1651 '\d STORE -1 in $1\_s*' ..
1652 '\d PUSHS "\["one", "two"\]"\_s*' ..
1653 '\d BCALL eval(argc 1)\_s*' ..
Bram Moolenaar0ad3e892020-07-05 21:38:11 +02001654 '\d FOR $1 -> \d\+\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01001655 '\d STORE $3\_s*' ..
1656
Bram Moolenaar0ad3e892020-07-05 21:38:11 +02001657 'res ..= str\_s*' ..
1658 '\d\+ LOAD $0\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01001659 '\d\+ LOAD $3\_s*' ..
Bram Moolenaarf5d52c92021-07-31 22:51:10 +02001660 '\d 2STRING_ANY stack\[-1\]\_s*' ..
LemonBoy372bcce2022-04-25 12:43:20 +01001661 '\d\+ CONCAT size 2\_s*' ..
Bram Moolenaar0ad3e892020-07-05 21:38:11 +02001662 '\d\+ STORE $0\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01001663
Bram Moolenaar0ad3e892020-07-05 21:38:11 +02001664 'endfor\_s*' ..
Bram Moolenaar74e54fc2021-03-26 20:41:29 +01001665 '\d\+ JUMP -> 5\_s*' ..
Bram Moolenaar0ad3e892020-07-05 21:38:11 +02001666 '\d\+ DROP\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01001667
Bram Moolenaar0ad3e892020-07-05 21:38:11 +02001668 'return res\_s*' ..
1669 '\d\+ LOAD $0\_s*' ..
1670 '\d\+ RETURN',
1671 instr)
1672enddef
1673
Bram Moolenaar62aec932022-01-29 21:45:34 +00001674def s:ForLoopUnpack()
Bram Moolenaar792f7862020-11-23 08:31:18 +01001675 for [x1, x2] in [[1, 2], [3, 4]]
1676 echo x1 x2
1677 endfor
1678enddef
1679
1680def Test_disassemble_for_loop_unpack()
1681 var instr = execute('disassemble ForLoopUnpack')
1682 assert_match('ForLoopUnpack\_s*' ..
1683 'for \[x1, x2\] in \[\[1, 2\], \[3, 4\]\]\_s*' ..
1684 '\d\+ STORE -1 in $0\_s*' ..
1685 '\d\+ PUSHNR 1\_s*' ..
1686 '\d\+ PUSHNR 2\_s*' ..
1687 '\d\+ NEWLIST size 2\_s*' ..
1688 '\d\+ PUSHNR 3\_s*' ..
1689 '\d\+ PUSHNR 4\_s*' ..
1690 '\d\+ NEWLIST size 2\_s*' ..
1691 '\d\+ NEWLIST size 2\_s*' ..
1692 '\d\+ FOR $0 -> 16\_s*' ..
1693 '\d\+ UNPACK 2\_s*' ..
Bram Moolenaar792f7862020-11-23 08:31:18 +01001694 '\d\+ STORE $2\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01001695 '\d\+ STORE $3\_s*' ..
1696
Bram Moolenaar792f7862020-11-23 08:31:18 +01001697 'echo x1 x2\_s*' ..
Bram Moolenaar792f7862020-11-23 08:31:18 +01001698 '\d\+ LOAD $2\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01001699 '\d\+ LOAD $3\_s*' ..
Bram Moolenaar792f7862020-11-23 08:31:18 +01001700 '\d\+ ECHO 2\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01001701
Bram Moolenaar792f7862020-11-23 08:31:18 +01001702 'endfor\_s*' ..
1703 '\d\+ JUMP -> 8\_s*' ..
1704 '\d\+ DROP\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +02001705 '\d\+ RETURN void',
Bram Moolenaar792f7862020-11-23 08:31:18 +01001706 instr)
1707enddef
1708
Bram Moolenaar62aec932022-01-29 21:45:34 +00001709def s:ForLoopContinue()
Bram Moolenaarc150c092021-02-13 15:02:46 +01001710 for nr in [1, 2]
1711 try
1712 echo "ok"
1713 try
1714 echo "deeper"
1715 catch
1716 continue
1717 endtry
1718 catch
1719 echo "not ok"
1720 endtry
1721 endfor
1722enddef
1723
1724def Test_disassemble_for_loop_continue()
1725 var instr = execute('disassemble ForLoopContinue')
1726 assert_match('ForLoopContinue\_s*' ..
1727 'for nr in \[1, 2]\_s*' ..
1728 '0 STORE -1 in $0\_s*' ..
1729 '1 PUSHNR 1\_s*' ..
1730 '2 PUSHNR 2\_s*' ..
1731 '3 NEWLIST size 2\_s*' ..
1732 '4 FOR $0 -> 22\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01001733 '5 STORE $2\_s*' ..
1734
Bram Moolenaarc150c092021-02-13 15:02:46 +01001735 'try\_s*' ..
Bram Moolenaar7e82c5f2021-02-21 21:32:45 +01001736 '6 TRY catch -> 17, endtry -> 20\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01001737
Bram Moolenaarc150c092021-02-13 15:02:46 +01001738 'echo "ok"\_s*' ..
1739 '7 PUSHS "ok"\_s*' ..
1740 '8 ECHO 1\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01001741
Bram Moolenaarc150c092021-02-13 15:02:46 +01001742 'try\_s*' ..
Bram Moolenaar7e82c5f2021-02-21 21:32:45 +01001743 '9 TRY catch -> 13, endtry -> 15\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01001744
Bram Moolenaarc150c092021-02-13 15:02:46 +01001745 'echo "deeper"\_s*' ..
1746 '10 PUSHS "deeper"\_s*' ..
1747 '11 ECHO 1\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01001748
Bram Moolenaarc150c092021-02-13 15:02:46 +01001749 'catch\_s*' ..
1750 '12 JUMP -> 15\_s*' ..
1751 '13 CATCH\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01001752
Bram Moolenaarc150c092021-02-13 15:02:46 +01001753 'continue\_s*' ..
1754 '14 TRY-CONTINUE 2 levels -> 4\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01001755
Bram Moolenaarc150c092021-02-13 15:02:46 +01001756 'endtry\_s*' ..
1757 '15 ENDTRY\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01001758
Bram Moolenaarc150c092021-02-13 15:02:46 +01001759 'catch\_s*' ..
1760 '16 JUMP -> 20\_s*' ..
1761 '17 CATCH\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01001762
Bram Moolenaarc150c092021-02-13 15:02:46 +01001763 'echo "not ok"\_s*' ..
1764 '18 PUSHS "not ok"\_s*' ..
1765 '19 ECHO 1\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01001766
Bram Moolenaarc150c092021-02-13 15:02:46 +01001767 'endtry\_s*' ..
1768 '20 ENDTRY\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01001769
Bram Moolenaarc150c092021-02-13 15:02:46 +01001770 'endfor\_s*' ..
1771 '21 JUMP -> 4\_s*' ..
1772 '\d\+ DROP\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +02001773 '\d\+ RETURN void',
Bram Moolenaarc150c092021-02-13 15:02:46 +01001774 instr)
1775enddef
1776
Bram Moolenaarc2a4b352020-02-06 22:41:16 +01001777let g:number = 42
1778
Bram Moolenaar62aec932022-01-29 21:45:34 +00001779def s:TypeCast()
Bram Moolenaarac564082020-09-27 19:05:33 +02001780 var l: list<number> = [23, <number>g:number]
Bram Moolenaar64d662d2020-08-09 19:02:50 +02001781enddef
1782
1783def Test_disassemble_typecast()
Bram Moolenaarac564082020-09-27 19:05:33 +02001784 var instr = execute('disassemble TypeCast')
Bram Moolenaar64d662d2020-08-09 19:02:50 +02001785 assert_match('TypeCast.*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02001786 'var l: list<number> = \[23, <number>g:number\].*' ..
Bram Moolenaar64d662d2020-08-09 19:02:50 +02001787 '\d PUSHNR 23\_s*' ..
1788 '\d LOADG g:number\_s*' ..
1789 '\d CHECKTYPE number stack\[-1\]\_s*' ..
1790 '\d NEWLIST size 2\_s*' ..
Bram Moolenaaraa210a32021-01-02 15:41:03 +01001791 '\d SETTYPE list<number>\_s*' ..
Bram Moolenaar64d662d2020-08-09 19:02:50 +02001792 '\d STORE $0\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +02001793 '\d RETURN void\_s*',
Bram Moolenaar64d662d2020-08-09 19:02:50 +02001794 instr)
1795enddef
1796
LemonBoy50d48542024-07-04 17:03:17 +02001797def Test_disassemble_object_cast()
1798 # Downcasting.
1799 var lines =<< trim END
1800 vim9script
1801 class A
1802 endclass
1803 class B extends A
1804 var mylen: number
1805 endclass
1806 def F(o: A): number
1807 return (<B>o).mylen
1808 enddef
1809
1810 g:instr = execute('disassemble F')
1811 END
1812 v9.CheckScriptSuccess(lines)
1813 assert_match('\<SNR>\d*_F\_s*' ..
1814 'return (<B>o).mylen\_s*' ..
1815 '0 LOAD arg\[-1\]\_s*' ..
1816 '1 CHECKTYPE object<B> stack\[-1\]\_s*' ..
1817 '2 OBJ_MEMBER 0\_s*' ..
1818 '3 RETURN\_s*',
1819 g:instr)
1820
1821 # Upcasting.
1822 lines =<< trim END
1823 vim9script
1824 class A
1825 var mylen: number
1826 endclass
1827 class B extends A
1828 endclass
1829 def F(o: B): number
1830 return (<A>o).mylen
1831 enddef
1832
1833 g:instr = execute('disassemble F')
1834 END
1835 v9.CheckScriptSuccess(lines)
1836 assert_match('\<SNR>\d*_F\_s*' ..
1837 'return (<A>o).mylen\_s*' ..
1838 '0 LOAD arg\[-1\]\_s*' ..
1839 '1 OBJ_MEMBER 0\_s*' ..
1840 '2 RETURN\_s*',
1841 g:instr)
1842
1843 # Casting, type is not statically known.
1844 lines =<< trim END
1845 vim9script
1846 class A
1847 endclass
1848 class B extends A
1849 endclass
1850 def F(o: any): any
1851 return <A>o
1852 enddef
1853
1854 g:instr = execute('disassemble F')
1855 END
1856 v9.CheckScriptSuccess(lines)
1857 assert_match('\<SNR>\d*_F\_s*' ..
1858 'return <A>o\_s*' ..
1859 '0 LOAD arg\[-1\]\_s*' ..
1860 '1 CHECKTYPE object<A> stack\[-1\]\_s*' ..
1861 '2 RETURN\_s*',
1862 g:instr)
1863enddef
1864
Bram Moolenaar62aec932022-01-29 21:45:34 +00001865def s:Computing()
Bram Moolenaarac564082020-09-27 19:05:33 +02001866 var nr = 3
1867 var nrres = nr + 7
Bram Moolenaarc2a4b352020-02-06 22:41:16 +01001868 nrres = nr - 7
1869 nrres = nr * 7
1870 nrres = nr / 7
1871 nrres = nr % 7
1872
Bram Moolenaarac564082020-09-27 19:05:33 +02001873 var anyres = g:number + 7
Bram Moolenaarc2a4b352020-02-06 22:41:16 +01001874 anyres = g:number - 7
1875 anyres = g:number * 7
1876 anyres = g:number / 7
1877 anyres = g:number % 7
1878
Bram Moolenaar73e28dc2022-09-17 21:08:33 +01001879 var fl = 3.0
1880 var flres = fl + 7.0
1881 flres = fl - 7.0
1882 flres = fl * 7.0
1883 flres = fl / 7.0
Bram Moolenaarc2a4b352020-02-06 22:41:16 +01001884enddef
1885
Bram Moolenaarf2460a32020-02-07 22:09:54 +01001886def Test_disassemble_computing()
Bram Moolenaarac564082020-09-27 19:05:33 +02001887 var instr = execute('disassemble Computing')
Bram Moolenaar675f7162020-04-12 22:53:54 +02001888 assert_match('Computing.*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02001889 'var nr = 3.*' ..
Bram Moolenaar675f7162020-04-12 22:53:54 +02001890 '\d STORE 3 in $0.*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02001891 'var nrres = nr + 7.*' ..
Bram Moolenaar675f7162020-04-12 22:53:54 +02001892 '\d LOAD $0.*' ..
1893 '\d PUSHNR 7.*' ..
1894 '\d OPNR +.*' ..
1895 '\d STORE $1.*' ..
1896 'nrres = nr - 7.*' ..
1897 '\d OPNR -.*' ..
1898 'nrres = nr \* 7.*' ..
1899 '\d OPNR \*.*' ..
1900 'nrres = nr / 7.*' ..
1901 '\d OPNR /.*' ..
1902 'nrres = nr % 7.*' ..
1903 '\d OPNR %.*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02001904 'var anyres = g:number + 7.*' ..
Bram Moolenaar675f7162020-04-12 22:53:54 +02001905 '\d LOADG g:number.*' ..
1906 '\d PUSHNR 7.*' ..
1907 '\d OPANY +.*' ..
1908 '\d STORE $2.*' ..
1909 'anyres = g:number - 7.*' ..
1910 '\d OPANY -.*' ..
1911 'anyres = g:number \* 7.*' ..
1912 '\d OPANY \*.*' ..
1913 'anyres = g:number / 7.*' ..
1914 '\d OPANY /.*' ..
1915 'anyres = g:number % 7.*' ..
1916 '\d OPANY %.*',
1917 instr)
Bram Moolenaar73e28dc2022-09-17 21:08:33 +01001918 assert_match('Computing.*' ..
1919 'var fl = 3.0.*' ..
1920 '\d PUSHF 3.0.*' ..
1921 '\d STORE $3.*' ..
1922 'var flres = fl + 7.0.*' ..
1923 '\d LOAD $3.*' ..
1924 '\d PUSHF 7.0.*' ..
1925 '\d OPFLOAT +.*' ..
1926 '\d STORE $4.*' ..
1927 'flres = fl - 7.0.*' ..
1928 '\d OPFLOAT -.*' ..
1929 'flres = fl \* 7.0.*' ..
1930 '\d OPFLOAT \*.*' ..
1931 'flres = fl / 7.0.*' ..
1932 '\d OPFLOAT /.*',
1933 instr)
Bram Moolenaarc2a4b352020-02-06 22:41:16 +01001934enddef
Bram Moolenaar5cab73f2020-02-06 19:25:19 +01001935
Bram Moolenaar62aec932022-01-29 21:45:34 +00001936def s:AddListBlob()
Bram Moolenaarac564082020-09-27 19:05:33 +02001937 var reslist = [1, 2] + [3, 4]
1938 var resblob = 0z1122 + 0z3344
Bram Moolenaaree2e52a2020-02-19 14:17:18 +01001939enddef
1940
1941def Test_disassemble_add_list_blob()
Bram Moolenaarac564082020-09-27 19:05:33 +02001942 var instr = execute('disassemble AddListBlob')
Bram Moolenaar675f7162020-04-12 22:53:54 +02001943 assert_match('AddListBlob.*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02001944 'var reslist = \[1, 2] + \[3, 4].*' ..
Bram Moolenaar675f7162020-04-12 22:53:54 +02001945 '\d PUSHNR 1.*' ..
1946 '\d PUSHNR 2.*' ..
1947 '\d NEWLIST size 2.*' ..
1948 '\d PUSHNR 3.*' ..
1949 '\d PUSHNR 4.*' ..
1950 '\d NEWLIST size 2.*' ..
1951 '\d ADDLIST.*' ..
1952 '\d STORE $.*.*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02001953 'var resblob = 0z1122 + 0z3344.*' ..
Bram Moolenaar675f7162020-04-12 22:53:54 +02001954 '\d PUSHBLOB 0z1122.*' ..
1955 '\d PUSHBLOB 0z3344.*' ..
1956 '\d ADDBLOB.*' ..
1957 '\d STORE $.*',
1958 instr)
Bram Moolenaaree2e52a2020-02-19 14:17:18 +01001959enddef
1960
1961let g:aa = 'aa'
Bram Moolenaar62aec932022-01-29 21:45:34 +00001962def s:ConcatString(): string
Bram Moolenaarac564082020-09-27 19:05:33 +02001963 var res = g:aa .. "bb"
Bram Moolenaaree2e52a2020-02-19 14:17:18 +01001964 return res
1965enddef
1966
1967def Test_disassemble_concat()
Bram Moolenaarac564082020-09-27 19:05:33 +02001968 var instr = execute('disassemble ConcatString')
Bram Moolenaar675f7162020-04-12 22:53:54 +02001969 assert_match('ConcatString.*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02001970 'var res = g:aa .. "bb".*' ..
Bram Moolenaar675f7162020-04-12 22:53:54 +02001971 '\d LOADG g:aa.*' ..
1972 '\d PUSHS "bb".*' ..
Bram Moolenaar418f1df2020-08-12 21:34:49 +02001973 '\d 2STRING_ANY stack\[-2].*' ..
Bram Moolenaar675f7162020-04-12 22:53:54 +02001974 '\d CONCAT.*' ..
1975 '\d STORE $.*',
1976 instr)
Bram Moolenaaree2e52a2020-02-19 14:17:18 +01001977 assert_equal('aabb', ConcatString())
1978enddef
1979
Bram Moolenaar62aec932022-01-29 21:45:34 +00001980def s:StringIndex(): string
Bram Moolenaarac564082020-09-27 19:05:33 +02001981 var s = "abcd"
1982 var res = s[1]
Bram Moolenaar747f11a2020-07-19 18:38:37 +02001983 return res
1984enddef
1985
1986def Test_disassemble_string_index()
Bram Moolenaarac564082020-09-27 19:05:33 +02001987 var instr = execute('disassemble StringIndex')
Bram Moolenaar747f11a2020-07-19 18:38:37 +02001988 assert_match('StringIndex\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02001989 'var s = "abcd"\_s*' ..
Bram Moolenaar747f11a2020-07-19 18:38:37 +02001990 '\d PUSHS "abcd"\_s*' ..
1991 '\d STORE $0\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02001992 'var res = s\[1]\_s*' ..
Bram Moolenaar747f11a2020-07-19 18:38:37 +02001993 '\d LOAD $0\_s*' ..
1994 '\d PUSHNR 1\_s*' ..
1995 '\d STRINDEX\_s*' ..
1996 '\d STORE $1\_s*',
1997 instr)
1998 assert_equal('b', StringIndex())
1999enddef
2000
Bram Moolenaar62aec932022-01-29 21:45:34 +00002001def s:StringSlice(): string
Bram Moolenaarac564082020-09-27 19:05:33 +02002002 var s = "abcd"
Bram Moolenaarde4f95b2020-12-30 20:39:21 +01002003 var res = s[1 : 8]
Bram Moolenaared591872020-08-15 22:14:53 +02002004 return res
2005enddef
2006
2007def Test_disassemble_string_slice()
Bram Moolenaarac564082020-09-27 19:05:33 +02002008 var instr = execute('disassemble StringSlice')
Bram Moolenaared591872020-08-15 22:14:53 +02002009 assert_match('StringSlice\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02002010 'var s = "abcd"\_s*' ..
Bram Moolenaared591872020-08-15 22:14:53 +02002011 '\d PUSHS "abcd"\_s*' ..
2012 '\d STORE $0\_s*' ..
Bram Moolenaarde4f95b2020-12-30 20:39:21 +01002013 'var res = s\[1 : 8]\_s*' ..
Bram Moolenaared591872020-08-15 22:14:53 +02002014 '\d LOAD $0\_s*' ..
2015 '\d PUSHNR 1\_s*' ..
2016 '\d PUSHNR 8\_s*' ..
2017 '\d STRSLICE\_s*' ..
2018 '\d STORE $1\_s*',
2019 instr)
2020 assert_equal('bcd', StringSlice())
2021enddef
2022
Bram Moolenaar62aec932022-01-29 21:45:34 +00002023def s:ListIndex(): number
Bram Moolenaarac564082020-09-27 19:05:33 +02002024 var l = [1, 2, 3]
2025 var res = l[1]
Bram Moolenaaree2e52a2020-02-19 14:17:18 +01002026 return res
2027enddef
2028
2029def Test_disassemble_list_index()
Bram Moolenaarac564082020-09-27 19:05:33 +02002030 var instr = execute('disassemble ListIndex')
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002031 assert_match('ListIndex\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02002032 'var l = \[1, 2, 3]\_s*' ..
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002033 '\d PUSHNR 1\_s*' ..
2034 '\d PUSHNR 2\_s*' ..
2035 '\d PUSHNR 3\_s*' ..
2036 '\d NEWLIST size 3\_s*' ..
Bram Moolenaare88c6b72022-02-15 15:37:11 +00002037 '\d SETTYPE list<number>\_s*' ..
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002038 '\d STORE $0\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02002039 'var res = l\[1]\_s*' ..
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002040 '\d LOAD $0\_s*' ..
2041 '\d PUSHNR 1\_s*' ..
Bram Moolenaar747f11a2020-07-19 18:38:37 +02002042 '\d LISTINDEX\_s*' ..
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002043 '\d STORE $1\_s*',
Bram Moolenaar675f7162020-04-12 22:53:54 +02002044 instr)
Bram Moolenaaree2e52a2020-02-19 14:17:18 +01002045 assert_equal(2, ListIndex())
2046enddef
2047
Bram Moolenaar62aec932022-01-29 21:45:34 +00002048def s:ListSlice(): list<number>
Bram Moolenaarac564082020-09-27 19:05:33 +02002049 var l = [1, 2, 3]
Bram Moolenaarde4f95b2020-12-30 20:39:21 +01002050 var res = l[1 : 8]
Bram Moolenaared591872020-08-15 22:14:53 +02002051 return res
2052enddef
2053
2054def Test_disassemble_list_slice()
Bram Moolenaarac564082020-09-27 19:05:33 +02002055 var instr = execute('disassemble ListSlice')
Bram Moolenaared591872020-08-15 22:14:53 +02002056 assert_match('ListSlice\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02002057 'var l = \[1, 2, 3]\_s*' ..
Bram Moolenaared591872020-08-15 22:14:53 +02002058 '\d PUSHNR 1\_s*' ..
2059 '\d PUSHNR 2\_s*' ..
2060 '\d PUSHNR 3\_s*' ..
2061 '\d NEWLIST size 3\_s*' ..
Bram Moolenaare88c6b72022-02-15 15:37:11 +00002062 '\d SETTYPE list<number>\_s*' ..
Bram Moolenaared591872020-08-15 22:14:53 +02002063 '\d STORE $0\_s*' ..
Bram Moolenaarde4f95b2020-12-30 20:39:21 +01002064 'var res = l\[1 : 8]\_s*' ..
Bram Moolenaared591872020-08-15 22:14:53 +02002065 '\d LOAD $0\_s*' ..
2066 '\d PUSHNR 1\_s*' ..
2067 '\d PUSHNR 8\_s*' ..
Bram Moolenaare88c6b72022-02-15 15:37:11 +00002068 '\d\+ LISTSLICE\_s*' ..
2069 '\d\+ SETTYPE list<number>\_s*' ..
2070 '\d\+ STORE $1\_s*',
Bram Moolenaared591872020-08-15 22:14:53 +02002071 instr)
2072 assert_equal([2, 3], ListSlice())
2073enddef
2074
Bram Moolenaar62aec932022-01-29 21:45:34 +00002075def s:DictMember(): number
Bram Moolenaare0de1712020-12-02 17:36:54 +01002076 var d = {item: 1}
Bram Moolenaarac564082020-09-27 19:05:33 +02002077 var res = d.item
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002078 res = d["item"]
Bram Moolenaaree2e52a2020-02-19 14:17:18 +01002079 return res
2080enddef
2081
2082def Test_disassemble_dict_member()
Bram Moolenaarac564082020-09-27 19:05:33 +02002083 var instr = execute('disassemble DictMember')
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002084 assert_match('DictMember\_s*' ..
Bram Moolenaare0de1712020-12-02 17:36:54 +01002085 'var d = {item: 1}\_s*' ..
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002086 '\d PUSHS "item"\_s*' ..
2087 '\d PUSHNR 1\_s*' ..
2088 '\d NEWDICT size 1\_s*' ..
Bram Moolenaare88c6b72022-02-15 15:37:11 +00002089 '\d SETTYPE dict<number>\_s*' ..
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002090 '\d STORE $0\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02002091 'var res = d.item\_s*' ..
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002092 '\d\+ LOAD $0\_s*' ..
2093 '\d\+ MEMBER item\_s*' ..
Bram Moolenaarb1b6f4d2021-09-13 18:25:54 +02002094 '\d\+ USEDICT\_s*' ..
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002095 '\d\+ STORE $1\_s*' ..
2096 'res = d\["item"\]\_s*' ..
2097 '\d\+ LOAD $0\_s*' ..
2098 '\d\+ PUSHS "item"\_s*' ..
2099 '\d\+ MEMBER\_s*' ..
Bram Moolenaarb1b6f4d2021-09-13 18:25:54 +02002100 '\d\+ USEDICT\_s*' ..
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002101 '\d\+ STORE $1\_s*',
Bram Moolenaar675f7162020-04-12 22:53:54 +02002102 instr)
Bram Moolenaard2c61702020-09-06 15:58:36 +02002103 assert_equal(1, DictMember())
Bram Moolenaaree2e52a2020-02-19 14:17:18 +01002104enddef
2105
Bram Moolenaarcc673e72020-08-16 17:33:35 +02002106let somelist = [1, 2, 3, 4, 5]
Bram Moolenaar62aec932022-01-29 21:45:34 +00002107def s:AnyIndex(): number
Bram Moolenaarac564082020-09-27 19:05:33 +02002108 var res = g:somelist[2]
Bram Moolenaarcc673e72020-08-16 17:33:35 +02002109 return res
2110enddef
2111
2112def Test_disassemble_any_index()
Bram Moolenaarac564082020-09-27 19:05:33 +02002113 var instr = execute('disassemble AnyIndex')
Bram Moolenaarcc673e72020-08-16 17:33:35 +02002114 assert_match('AnyIndex\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02002115 'var res = g:somelist\[2\]\_s*' ..
Bram Moolenaarcc673e72020-08-16 17:33:35 +02002116 '\d LOADG g:somelist\_s*' ..
2117 '\d PUSHNR 2\_s*' ..
2118 '\d ANYINDEX\_s*' ..
2119 '\d STORE $0\_s*' ..
2120 'return res\_s*' ..
2121 '\d LOAD $0\_s*' ..
2122 '\d CHECKTYPE number stack\[-1\]\_s*' ..
2123 '\d RETURN',
2124 instr)
2125 assert_equal(3, AnyIndex())
2126enddef
2127
Bram Moolenaar62aec932022-01-29 21:45:34 +00002128def s:AnySlice(): list<number>
Bram Moolenaarde4f95b2020-12-30 20:39:21 +01002129 var res = g:somelist[1 : 3]
Bram Moolenaarcc673e72020-08-16 17:33:35 +02002130 return res
2131enddef
2132
2133def Test_disassemble_any_slice()
Bram Moolenaarac564082020-09-27 19:05:33 +02002134 var instr = execute('disassemble AnySlice')
Bram Moolenaarcc673e72020-08-16 17:33:35 +02002135 assert_match('AnySlice\_s*' ..
Bram Moolenaarde4f95b2020-12-30 20:39:21 +01002136 'var res = g:somelist\[1 : 3\]\_s*' ..
Bram Moolenaarcc673e72020-08-16 17:33:35 +02002137 '\d LOADG g:somelist\_s*' ..
2138 '\d PUSHNR 1\_s*' ..
2139 '\d PUSHNR 3\_s*' ..
2140 '\d ANYSLICE\_s*' ..
2141 '\d STORE $0\_s*' ..
2142 'return res\_s*' ..
2143 '\d LOAD $0\_s*' ..
Bram Moolenaar5e654232020-09-16 15:22:00 +02002144 '\d CHECKTYPE list<number> stack\[-1\]\_s*' ..
Bram Moolenaarcc673e72020-08-16 17:33:35 +02002145 '\d RETURN',
2146 instr)
2147 assert_equal([2, 3, 4], AnySlice())
2148enddef
2149
Bram Moolenaar62aec932022-01-29 21:45:34 +00002150def s:NegateNumber(): number
Bram Moolenaarcd6b4f32021-08-15 20:36:28 +02002151 g:nr = 9
2152 var plus = +g:nr
2153 var minus = -g:nr
2154 return minus
Bram Moolenaaree2e52a2020-02-19 14:17:18 +01002155enddef
2156
2157def Test_disassemble_negate_number()
Bram Moolenaarac564082020-09-27 19:05:33 +02002158 var instr = execute('disassemble NegateNumber')
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002159 assert_match('NegateNumber\_s*' ..
Bram Moolenaarcd6b4f32021-08-15 20:36:28 +02002160 'g:nr = 9\_s*' ..
2161 '\d PUSHNR 9\_s*' ..
2162 '\d STOREG g:nr\_s*' ..
2163 'var plus = +g:nr\_s*' ..
2164 '\d LOADG g:nr\_s*' ..
2165 '\d CHECKTYPE number stack\[-1\]\_s*' ..
2166 '\d STORE $0\_s*' ..
2167 'var minus = -g:nr\_s*' ..
2168 '\d LOADG g:nr\_s*' ..
2169 '\d CHECKTYPE number stack\[-1\]\_s*' ..
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002170 '\d NEGATENR\_s*' ..
Bram Moolenaarcd6b4f32021-08-15 20:36:28 +02002171 '\d STORE $1\_s*',
Bram Moolenaar675f7162020-04-12 22:53:54 +02002172 instr)
Bram Moolenaard2c61702020-09-06 15:58:36 +02002173 assert_equal(-9, NegateNumber())
Bram Moolenaaree2e52a2020-02-19 14:17:18 +01002174enddef
2175
Bram Moolenaar62aec932022-01-29 21:45:34 +00002176def s:InvertBool(): bool
Bram Moolenaarac564082020-09-27 19:05:33 +02002177 var flag = true
2178 var invert = !flag
2179 var res = !!flag
Bram Moolenaaree2e52a2020-02-19 14:17:18 +01002180 return res
2181enddef
2182
2183def Test_disassemble_invert_bool()
Bram Moolenaarac564082020-09-27 19:05:33 +02002184 var instr = execute('disassemble InvertBool')
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002185 assert_match('InvertBool\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02002186 'var flag = true\_s*' ..
Bram Moolenaara8b8af12021-01-01 15:11:04 +01002187 '\d PUSH true\_s*' ..
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002188 '\d STORE $0\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02002189 'var invert = !flag\_s*' ..
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002190 '\d LOAD $0\_s*' ..
Bram Moolenaar5fa9b242021-06-04 21:00:32 +02002191 '\d INVERT -1 (!val)\_s*' ..
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002192 '\d STORE $1\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02002193 'var res = !!flag\_s*' ..
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002194 '\d LOAD $0\_s*' ..
Bram Moolenaar5fa9b242021-06-04 21:00:32 +02002195 '\d 2BOOL -1 (!!val)\_s*' ..
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002196 '\d STORE $2\_s*',
Bram Moolenaar675f7162020-04-12 22:53:54 +02002197 instr)
Bram Moolenaard2c61702020-09-06 15:58:36 +02002198 assert_equal(true, InvertBool())
Bram Moolenaaree2e52a2020-02-19 14:17:18 +01002199enddef
2200
Bram Moolenaar62aec932022-01-29 21:45:34 +00002201def s:ReturnBool(): bool
Bram Moolenaar1a7ee4d2021-09-16 16:15:07 +02002202 var one = 1
2203 var zero = 0
Bram Moolenaar5cd64792021-12-25 18:23:24 +00002204 var none: number
Bram Moolenaar1a7ee4d2021-09-16 16:15:07 +02002205 var name: bool = one && zero || one
Bram Moolenaar2bb26582020-10-03 22:52:39 +02002206 return name
Bram Moolenaar4ed124c2020-09-09 20:03:46 +02002207enddef
2208
2209def Test_disassemble_return_bool()
Bram Moolenaarac564082020-09-27 19:05:33 +02002210 var instr = execute('disassemble ReturnBool')
Bram Moolenaar4ed124c2020-09-09 20:03:46 +02002211 assert_match('ReturnBool\_s*' ..
Bram Moolenaar1a7ee4d2021-09-16 16:15:07 +02002212 'var one = 1\_s*' ..
2213 '0 STORE 1 in $0\_s*' ..
2214 'var zero = 0\_s*' ..
Bram Moolenaar5cd64792021-12-25 18:23:24 +00002215 'var none: number\_s*' ..
Bram Moolenaar1a7ee4d2021-09-16 16:15:07 +02002216 'var name: bool = one && zero || one\_s*' ..
Bram Moolenaar5cd64792021-12-25 18:23:24 +00002217 '1 LOAD $0\_s*' ..
2218 '2 COND2BOOL\_s*' ..
2219 '3 JUMP_IF_COND_FALSE -> 6\_s*' ..
2220 '4 LOAD $1\_s*' ..
2221 '5 COND2BOOL\_s*' ..
2222 '6 JUMP_IF_COND_TRUE -> 9\_s*' ..
2223 '7 LOAD $0\_s*' ..
2224 '8 COND2BOOL\_s*' ..
2225 '9 STORE $3\_s*' ..
Bram Moolenaar2bb26582020-10-03 22:52:39 +02002226 'return name\_s*' ..
Bram Moolenaar94722c52023-01-28 19:19:03 +00002227 '\d\+ LOAD $3\_s*' ..
Bram Moolenaarea2d4072020-11-12 12:08:51 +01002228 '\d\+ RETURN',
Bram Moolenaar4ed124c2020-09-09 20:03:46 +02002229 instr)
2230 assert_equal(true, InvertBool())
2231enddef
2232
Bram Moolenaar62aec932022-01-29 21:45:34 +00002233def s:AutoInit()
Bram Moolenaarfb9dcb02021-12-25 22:00:49 +00002234 var t: number
2235 t = 1
2236 t = 0
2237enddef
2238
2239def Test_disassemble_auto_init()
2240 var instr = execute('disassemble AutoInit')
2241 assert_match('AutoInit\_s*' ..
2242 'var t: number\_s*' ..
2243 't = 1\_s*' ..
2244 '\d STORE 1 in $0\_s*' ..
2245 't = 0\_s*' ..
2246 '\d STORE 0 in $0\_s*' ..
2247 '\d\+ RETURN void',
2248 instr)
2249enddef
2250
Bram Moolenaarf2460a32020-02-07 22:09:54 +01002251def Test_disassemble_compare()
Bram Moolenaarac564082020-09-27 19:05:33 +02002252 var cases = [
Bram Moolenaara5565e42020-05-09 15:44:01 +02002253 ['true == isFalse', 'COMPAREBOOL =='],
2254 ['true != isFalse', 'COMPAREBOOL !='],
2255 ['v:none == isNull', 'COMPARESPECIAL =='],
2256 ['v:none != isNull', 'COMPARESPECIAL !='],
Bram Moolenaar7a222242022-03-01 19:23:24 +00002257 ['"text" == isNull', 'COMPARENULL =='],
2258 ['"text" != isNull', 'COMPARENULL !='],
Bram Moolenaar675f7162020-04-12 22:53:54 +02002259
Bram Moolenaara5565e42020-05-09 15:44:01 +02002260 ['111 == aNumber', 'COMPARENR =='],
2261 ['111 != aNumber', 'COMPARENR !='],
2262 ['111 > aNumber', 'COMPARENR >'],
2263 ['111 < aNumber', 'COMPARENR <'],
2264 ['111 >= aNumber', 'COMPARENR >='],
2265 ['111 <= aNumber', 'COMPARENR <='],
2266 ['111 =~ aNumber', 'COMPARENR =\~'],
2267 ['111 !~ aNumber', 'COMPARENR !\~'],
Bram Moolenaar675f7162020-04-12 22:53:54 +02002268
Bram Moolenaara5565e42020-05-09 15:44:01 +02002269 ['"xx" != aString', 'COMPARESTRING !='],
2270 ['"xx" > aString', 'COMPARESTRING >'],
2271 ['"xx" < aString', 'COMPARESTRING <'],
2272 ['"xx" >= aString', 'COMPARESTRING >='],
2273 ['"xx" <= aString', 'COMPARESTRING <='],
2274 ['"xx" =~ aString', 'COMPARESTRING =\~'],
2275 ['"xx" !~ aString', 'COMPARESTRING !\~'],
2276 ['"xx" is aString', 'COMPARESTRING is'],
2277 ['"xx" isnot aString', 'COMPARESTRING isnot'],
Bram Moolenaar675f7162020-04-12 22:53:54 +02002278
Bram Moolenaara5565e42020-05-09 15:44:01 +02002279 ['0z11 == aBlob', 'COMPAREBLOB =='],
2280 ['0z11 != aBlob', 'COMPAREBLOB !='],
2281 ['0z11 is aBlob', 'COMPAREBLOB is'],
2282 ['0z11 isnot aBlob', 'COMPAREBLOB isnot'],
Bram Moolenaar675f7162020-04-12 22:53:54 +02002283
Bram Moolenaara5565e42020-05-09 15:44:01 +02002284 ['[1, 2] == aList', 'COMPARELIST =='],
2285 ['[1, 2] != aList', 'COMPARELIST !='],
2286 ['[1, 2] is aList', 'COMPARELIST is'],
2287 ['[1, 2] isnot aList', 'COMPARELIST isnot'],
Bram Moolenaar675f7162020-04-12 22:53:54 +02002288
Bram Moolenaare0de1712020-12-02 17:36:54 +01002289 ['{a: 1} == aDict', 'COMPAREDICT =='],
2290 ['{a: 1} != aDict', 'COMPAREDICT !='],
2291 ['{a: 1} is aDict', 'COMPAREDICT is'],
2292 ['{a: 1} isnot aDict', 'COMPAREDICT isnot'],
Bram Moolenaar675f7162020-04-12 22:53:54 +02002293
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002294 ['(() => 33) == (() => 44)', 'COMPAREFUNC =='],
2295 ['(() => 33) != (() => 44)', 'COMPAREFUNC !='],
2296 ['(() => 33) is (() => 44)', 'COMPAREFUNC is'],
2297 ['(() => 33) isnot (() => 44)', 'COMPAREFUNC isnot'],
Bram Moolenaar675f7162020-04-12 22:53:54 +02002298
2299 ['77 == g:xx', 'COMPAREANY =='],
2300 ['77 != g:xx', 'COMPAREANY !='],
2301 ['77 > g:xx', 'COMPAREANY >'],
2302 ['77 < g:xx', 'COMPAREANY <'],
2303 ['77 >= g:xx', 'COMPAREANY >='],
2304 ['77 <= g:xx', 'COMPAREANY <='],
2305 ['77 =~ g:xx', 'COMPAREANY =\~'],
2306 ['77 !~ g:xx', 'COMPAREANY !\~'],
2307 ['77 is g:xx', 'COMPAREANY is'],
2308 ['77 isnot g:xx', 'COMPAREANY isnot'],
2309 ]
Bram Moolenaarac564082020-09-27 19:05:33 +02002310 var floatDecl = ''
Bram Moolenaar73e28dc2022-09-17 21:08:33 +01002311 cases->extend([
2312 ['1.1 == aFloat', 'COMPAREFLOAT =='],
2313 ['1.1 != aFloat', 'COMPAREFLOAT !='],
2314 ['1.1 > aFloat', 'COMPAREFLOAT >'],
2315 ['1.1 < aFloat', 'COMPAREFLOAT <'],
2316 ['1.1 >= aFloat', 'COMPAREFLOAT >='],
2317 ['1.1 <= aFloat', 'COMPAREFLOAT <='],
2318 ['1.1 =~ aFloat', 'COMPAREFLOAT =\~'],
2319 ['1.1 !~ aFloat', 'COMPAREFLOAT !\~'],
2320 ])
2321 floatDecl = 'var aFloat = 2.2'
Bram Moolenaarf2460a32020-02-07 22:09:54 +01002322
Bram Moolenaarac564082020-09-27 19:05:33 +02002323 var nr = 1
Bram Moolenaarf2460a32020-02-07 22:09:54 +01002324 for case in cases
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02002325 # declare local variables to get a non-constant with the right type
Bram Moolenaarf2460a32020-02-07 22:09:54 +01002326 writefile(['def TestCase' .. nr .. '()',
Bram Moolenaarac564082020-09-27 19:05:33 +02002327 ' var isFalse = false',
2328 ' var isNull = v:null',
2329 ' var aNumber = 222',
2330 ' var aString = "yy"',
2331 ' var aBlob = 0z22',
2332 ' var aList = [3, 4]',
Bram Moolenaare0de1712020-12-02 17:36:54 +01002333 ' var aDict = {x: 2}',
Bram Moolenaara5565e42020-05-09 15:44:01 +02002334 floatDecl,
Bram Moolenaar675f7162020-04-12 22:53:54 +02002335 ' if ' .. case[0],
Bram Moolenaar2984ed32022-08-20 14:51:17 +01002336 ' echo 42',
Bram Moolenaar675f7162020-04-12 22:53:54 +02002337 ' endif',
2338 'enddef'], 'Xdisassemble')
Bram Moolenaarf2460a32020-02-07 22:09:54 +01002339 source Xdisassemble
Bram Moolenaarac564082020-09-27 19:05:33 +02002340 var instr = execute('disassemble TestCase' .. nr)
Bram Moolenaar675f7162020-04-12 22:53:54 +02002341 assert_match('TestCase' .. nr .. '.*' ..
2342 'if ' .. substitute(case[0], '[[~]', '\\\0', 'g') .. '.*' ..
2343 '\d \(PUSH\|FUNCREF\).*' ..
Bram Moolenaara5565e42020-05-09 15:44:01 +02002344 '\d \(PUSH\|FUNCREF\|LOAD\).*' ..
Bram Moolenaar675f7162020-04-12 22:53:54 +02002345 '\d ' .. case[1] .. '.*' ..
2346 '\d JUMP_IF_FALSE -> \d\+.*',
2347 instr)
Bram Moolenaarf2460a32020-02-07 22:09:54 +01002348
2349 nr += 1
2350 endfor
2351
Bram Moolenaar22da5592020-03-19 14:52:20 +01002352 delete('Xdisassemble')
Bram Moolenaarf2460a32020-02-07 22:09:54 +01002353enddef
2354
Bram Moolenaar92f26c22020-10-03 20:17:30 +02002355def s:FalsyOp()
2356 echo g:flag ?? "yes"
2357 echo [] ?? "empty list"
2358 echo "" ?? "empty string"
2359enddef
2360
Dominique Pelle81b573d2022-03-22 21:14:55 +00002361def Test_disassemble_falsy_op()
Bram Moolenaar92f26c22020-10-03 20:17:30 +02002362 var res = execute('disass s:FalsyOp')
2363 assert_match('\<SNR>\d*_FalsyOp\_s*' ..
2364 'echo g:flag ?? "yes"\_s*' ..
2365 '0 LOADG g:flag\_s*' ..
2366 '1 JUMP_AND_KEEP_IF_TRUE -> 3\_s*' ..
2367 '2 PUSHS "yes"\_s*' ..
2368 '3 ECHO 1\_s*' ..
2369 'echo \[\] ?? "empty list"\_s*' ..
2370 '4 NEWLIST size 0\_s*' ..
2371 '5 JUMP_AND_KEEP_IF_TRUE -> 7\_s*' ..
2372 '6 PUSHS "empty list"\_s*' ..
2373 '7 ECHO 1\_s*' ..
2374 'echo "" ?? "empty string"\_s*' ..
2375 '\d\+ PUSHS "empty string"\_s*' ..
2376 '\d\+ ECHO 1\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +02002377 '\d\+ RETURN void',
Bram Moolenaar92f26c22020-10-03 20:17:30 +02002378 res)
2379enddef
2380
Bram Moolenaara4d4cf42020-04-02 13:50:27 +02002381def Test_disassemble_compare_const()
Bram Moolenaarac564082020-09-27 19:05:33 +02002382 var cases = [
Bram Moolenaar675f7162020-04-12 22:53:54 +02002383 ['"xx" == "yy"', false],
2384 ['"aa" == "aa"', true],
2385 ['has("eval") ? true : false', true],
2386 ['has("asdf") ? true : false', false],
2387 ]
Bram Moolenaara4d4cf42020-04-02 13:50:27 +02002388
Bram Moolenaarac564082020-09-27 19:05:33 +02002389 var nr = 1
Bram Moolenaara4d4cf42020-04-02 13:50:27 +02002390 for case in cases
2391 writefile(['def TestCase' .. nr .. '()',
Bram Moolenaar675f7162020-04-12 22:53:54 +02002392 ' if ' .. case[0],
Bram Moolenaar2984ed32022-08-20 14:51:17 +01002393 ' echo 42',
Bram Moolenaar675f7162020-04-12 22:53:54 +02002394 ' endif',
2395 'enddef'], 'Xdisassemble')
Bram Moolenaara4d4cf42020-04-02 13:50:27 +02002396 source Xdisassemble
Bram Moolenaarac564082020-09-27 19:05:33 +02002397 var instr = execute('disassemble TestCase' .. nr)
Bram Moolenaara4d4cf42020-04-02 13:50:27 +02002398 if case[1]
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02002399 # condition true, "echo 42" executed
Bram Moolenaar675f7162020-04-12 22:53:54 +02002400 assert_match('TestCase' .. nr .. '.*' ..
2401 'if ' .. substitute(case[0], '[[~]', '\\\0', 'g') .. '.*' ..
2402 '\d PUSHNR 42.*' ..
2403 '\d ECHO 1.*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +02002404 '\d RETURN void',
Bram Moolenaar675f7162020-04-12 22:53:54 +02002405 instr)
Bram Moolenaara4d4cf42020-04-02 13:50:27 +02002406 else
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02002407 # condition false, function just returns
Bram Moolenaar675f7162020-04-12 22:53:54 +02002408 assert_match('TestCase' .. nr .. '.*' ..
2409 'if ' .. substitute(case[0], '[[~]', '\\\0', 'g') .. '[ \n]*' ..
2410 'echo 42[ \n]*' ..
2411 'endif[ \n]*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +02002412 '\d RETURN void',
Bram Moolenaar675f7162020-04-12 22:53:54 +02002413 instr)
Bram Moolenaara4d4cf42020-04-02 13:50:27 +02002414 endif
2415
2416 nr += 1
2417 endfor
2418
2419 delete('Xdisassemble')
2420enddef
2421
Bram Moolenaarad39c092020-02-26 18:23:43 +01002422def s:Execute()
2423 execute 'help vim9.txt'
Bram Moolenaarac564082020-09-27 19:05:33 +02002424 var cmd = 'help vim9.txt'
Bram Moolenaarad39c092020-02-26 18:23:43 +01002425 execute cmd
Bram Moolenaarac564082020-09-27 19:05:33 +02002426 var tag = 'vim9.txt'
Bram Moolenaarad39c092020-02-26 18:23:43 +01002427 execute 'help ' .. tag
2428enddef
2429
2430def Test_disassemble_execute()
Bram Moolenaarac564082020-09-27 19:05:33 +02002431 var res = execute('disass s:Execute')
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002432 assert_match('\<SNR>\d*_Execute\_s*' ..
2433 "execute 'help vim9.txt'\\_s*" ..
2434 '\d PUSHS "help vim9.txt"\_s*' ..
2435 '\d EXECUTE 1\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02002436 "var cmd = 'help vim9.txt'\\_s*" ..
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002437 '\d PUSHS "help vim9.txt"\_s*' ..
2438 '\d STORE $0\_s*' ..
2439 'execute cmd\_s*' ..
2440 '\d LOAD $0\_s*' ..
2441 '\d EXECUTE 1\_s*' ..
Bram Moolenaarac564082020-09-27 19:05:33 +02002442 "var tag = 'vim9.txt'\\_s*" ..
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002443 '\d PUSHS "vim9.txt"\_s*' ..
2444 '\d STORE $1\_s*' ..
2445 "execute 'help ' .. tag\\_s*" ..
2446 '\d\+ PUSHS "help "\_s*' ..
2447 '\d\+ LOAD $1\_s*' ..
LemonBoy372bcce2022-04-25 12:43:20 +01002448 '\d\+ CONCAT size 2\_s*' ..
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002449 '\d\+ EXECUTE 1\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +02002450 '\d\+ RETURN void',
Bram Moolenaar675f7162020-04-12 22:53:54 +02002451 res)
Bram Moolenaarad39c092020-02-26 18:23:43 +01002452enddef
2453
Bram Moolenaare4eed8c2021-12-01 15:22:56 +00002454def s:OnlyRange()
2455 :$
2456 :123
2457 :'m
2458enddef
2459
2460def Test_disassemble_range_only()
2461 var res = execute('disass s:OnlyRange')
2462 assert_match('\<SNR>\d*_OnlyRange\_s*' ..
2463 ':$\_s*' ..
2464 '\d EXECRANGE $\_s*' ..
2465 ':123\_s*' ..
2466 '\d EXECRANGE 123\_s*' ..
2467 ':''m\_s*' ..
2468 '\d EXECRANGE ''m\_s*' ..
2469 '\d\+ RETURN void',
2470 res)
2471enddef
2472
Bram Moolenaarf6ced982022-04-28 12:00:49 +01002473def s:StoreRange()
2474 var l = [1, 2]
2475 l[0 : 1] = [7, 8]
2476enddef
2477
2478def Test_disassemble_store_range()
2479 var res = execute('disass s:StoreRange')
2480 assert_match('\<SNR>\d*_StoreRange\_s*' ..
2481 'var l = \[1, 2]\_s*' ..
2482 '\d PUSHNR 1\_s*' ..
2483 '\d PUSHNR 2\_s*' ..
2484 '\d NEWLIST size 2\_s*' ..
2485 '\d SETTYPE list<number>\_s*' ..
2486 '\d STORE $0\_s*' ..
2487
2488 'l\[0 : 1] = \[7, 8]\_s*' ..
2489 '\d\+ PUSHNR 7\_s*' ..
2490 '\d\+ PUSHNR 8\_s*' ..
2491 '\d\+ NEWLIST size 2\_s*' ..
2492 '\d\+ PUSHNR 0\_s*' ..
2493 '\d\+ PUSHNR 1\_s*' ..
2494 '\d\+ LOAD $0\_s*' ..
2495 '\d\+ STORERANGE\_s*' ..
2496 '\d\+ RETURN void',
2497 res)
2498enddef
2499
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02002500def s:Echomsg()
2501 echomsg 'some' 'message'
Bram Moolenaar7de62622021-08-07 15:05:47 +02002502 echoconsole 'nothing'
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02002503 echoerr 'went' .. 'wrong'
Bram Moolenaar7d7ad7b2022-09-01 16:00:53 +01002504 var local = 'window'
2505 echowin 'in' local
Bram Moolenaarbdc09a12022-10-07 14:31:45 +01002506 :5echowin 'five'
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02002507enddef
2508
2509def Test_disassemble_echomsg()
Bram Moolenaarac564082020-09-27 19:05:33 +02002510 var res = execute('disass s:Echomsg')
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002511 assert_match('\<SNR>\d*_Echomsg\_s*' ..
2512 "echomsg 'some' 'message'\\_s*" ..
2513 '\d PUSHS "some"\_s*' ..
2514 '\d PUSHS "message"\_s*' ..
2515 '\d ECHOMSG 2\_s*' ..
Bram Moolenaar7de62622021-08-07 15:05:47 +02002516 "echoconsole 'nothing'\\_s*" ..
2517 '\d PUSHS "nothing"\_s*' ..
2518 '\d ECHOCONSOLE 1\_s*' ..
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002519 "echoerr 'went' .. 'wrong'\\_s*" ..
2520 '\d PUSHS "wentwrong"\_s*' ..
2521 '\d ECHOERR 1\_s*' ..
Bram Moolenaar7d7ad7b2022-09-01 16:00:53 +01002522 "var local = 'window'\\_s*" ..
2523 '\d\+ PUSHS "window"\_s*' ..
2524 '\d\+ STORE $0\_s*' ..
2525 "echowin 'in' local\\_s*" ..
2526 '\d\+ PUSHS "in"\_s*' ..
2527 '\d\+ LOAD $0\_s*' ..
2528 '\d\+ ECHOWINDOW 2\_s*' ..
Bram Moolenaarbdc09a12022-10-07 14:31:45 +01002529 ":5echowin 'five'\\_s*" ..
2530 '\d\+ PUSHS "five"\_s*' ..
2531 '\d\+ ECHOWINDOW 1 (5 sec)\_s*' ..
Bram Moolenaar7d7ad7b2022-09-01 16:00:53 +01002532 '\d\+ RETURN void',
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02002533 res)
2534enddef
2535
Bram Moolenaar61a6d4e2020-03-01 23:32:25 +01002536def SomeStringArg(arg: string)
2537 echo arg
2538enddef
2539
2540def SomeAnyArg(arg: any)
2541 echo arg
2542enddef
2543
2544def SomeStringArgAndReturn(arg: string): string
2545 return arg
2546enddef
2547
2548def Test_display_func()
Bram Moolenaarac564082020-09-27 19:05:33 +02002549 var res1 = execute('function SomeStringArg')
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002550 assert_match('.* def SomeStringArg(arg: string)\_s*' ..
2551 '\d *echo arg.*' ..
2552 ' *enddef',
Bram Moolenaar675f7162020-04-12 22:53:54 +02002553 res1)
Bram Moolenaar61a6d4e2020-03-01 23:32:25 +01002554
Bram Moolenaarac564082020-09-27 19:05:33 +02002555 var res2 = execute('function SomeAnyArg')
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002556 assert_match('.* def SomeAnyArg(arg: any)\_s*' ..
2557 '\d *echo arg\_s*' ..
2558 ' *enddef',
Bram Moolenaar675f7162020-04-12 22:53:54 +02002559 res2)
Bram Moolenaar61a6d4e2020-03-01 23:32:25 +01002560
Bram Moolenaarac564082020-09-27 19:05:33 +02002561 var res3 = execute('function SomeStringArgAndReturn')
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002562 assert_match('.* def SomeStringArgAndReturn(arg: string): string\_s*' ..
2563 '\d *return arg\_s*' ..
2564 ' *enddef',
Bram Moolenaar675f7162020-04-12 22:53:54 +02002565 res3)
Bram Moolenaar61a6d4e2020-03-01 23:32:25 +01002566enddef
2567
Bram Moolenaar09689a02020-05-09 22:50:08 +02002568def Test_vim9script_forward_func()
Bram Moolenaarac564082020-09-27 19:05:33 +02002569 var lines =<< trim END
Bram Moolenaar09689a02020-05-09 22:50:08 +02002570 vim9script
2571 def FuncOne(): string
2572 return FuncTwo()
2573 enddef
2574 def FuncTwo(): string
2575 return 'two'
2576 enddef
Bram Moolenaar67979662020-06-20 22:50:47 +02002577 g:res_FuncOne = execute('disass FuncOne')
Bram Moolenaar09689a02020-05-09 22:50:08 +02002578 END
Bram Moolenaar0e9bdad2022-10-15 20:06:33 +01002579 writefile(lines, 'Xdisassemble', 'D')
Bram Moolenaar09689a02020-05-09 22:50:08 +02002580 source Xdisassemble
2581
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02002582 # check that the first function calls the second with DCALL
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002583 assert_match('\<SNR>\d*_FuncOne\_s*' ..
2584 'return FuncTwo()\_s*' ..
2585 '\d DCALL <SNR>\d\+_FuncTwo(argc 0)\_s*' ..
Bram Moolenaar09689a02020-05-09 22:50:08 +02002586 '\d RETURN',
2587 g:res_FuncOne)
2588
Bram Moolenaar09689a02020-05-09 22:50:08 +02002589 unlet g:res_FuncOne
2590enddef
2591
Bram Moolenaar61a89812020-05-07 16:58:17 +02002592def s:ConcatStrings(): string
2593 return 'one' .. 'two' .. 'three'
2594enddef
2595
Bram Moolenaar7d131b02020-05-08 19:10:34 +02002596def s:ComputeConst(): number
2597 return 2 + 3 * 4 / 6 + 7
2598enddef
2599
Bram Moolenaar1c747212020-05-09 18:28:34 +02002600def s:ComputeConstParen(): number
2601 return ((2 + 4) * (8 / 2)) / (3 + 4)
2602enddef
2603
Bram Moolenaar61a89812020-05-07 16:58:17 +02002604def Test_simplify_const_expr()
Bram Moolenaarac564082020-09-27 19:05:33 +02002605 var res = execute('disass s:ConcatStrings')
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002606 assert_match('<SNR>\d*_ConcatStrings\_s*' ..
2607 "return 'one' .. 'two' .. 'three'\\_s*" ..
2608 '\d PUSHS "onetwothree"\_s*' ..
Bram Moolenaar61a89812020-05-07 16:58:17 +02002609 '\d RETURN',
2610 res)
Bram Moolenaar7d131b02020-05-08 19:10:34 +02002611
2612 res = execute('disass s:ComputeConst')
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002613 assert_match('<SNR>\d*_ComputeConst\_s*' ..
2614 'return 2 + 3 \* 4 / 6 + 7\_s*' ..
2615 '\d PUSHNR 11\_s*' ..
Bram Moolenaar7d131b02020-05-08 19:10:34 +02002616 '\d RETURN',
2617 res)
Bram Moolenaar1c747212020-05-09 18:28:34 +02002618
2619 res = execute('disass s:ComputeConstParen')
Bram Moolenaar4902ab12020-05-15 19:21:31 +02002620 assert_match('<SNR>\d*_ComputeConstParen\_s*' ..
2621 'return ((2 + 4) \* (8 / 2)) / (3 + 4)\_s*' ..
2622 '\d PUSHNR 3\>\_s*' ..
Bram Moolenaar1c747212020-05-09 18:28:34 +02002623 '\d RETURN',
2624 res)
Bram Moolenaar61a89812020-05-07 16:58:17 +02002625enddef
2626
Bram Moolenaar389df252020-07-09 21:20:47 +02002627def s:CallAppend()
2628 eval "some text"->append(2)
2629enddef
2630
2631def Test_shuffle()
Bram Moolenaarac564082020-09-27 19:05:33 +02002632 var res = execute('disass s:CallAppend')
Bram Moolenaar389df252020-07-09 21:20:47 +02002633 assert_match('<SNR>\d*_CallAppend\_s*' ..
2634 'eval "some text"->append(2)\_s*' ..
2635 '\d PUSHS "some text"\_s*' ..
2636 '\d PUSHNR 2\_s*' ..
2637 '\d SHUFFLE 2 up 1\_s*' ..
2638 '\d BCALL append(argc 2)\_s*' ..
2639 '\d DROP\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +02002640 '\d RETURN void',
Bram Moolenaar389df252020-07-09 21:20:47 +02002641 res)
2642enddef
2643
Bram Moolenaarf4c6e1e2020-10-23 18:02:32 +02002644
2645def s:SilentMessage()
2646 silent echomsg "text"
2647 silent! echoerr "error"
2648enddef
2649
2650def Test_silent()
2651 var res = execute('disass s:SilentMessage')
2652 assert_match('<SNR>\d*_SilentMessage\_s*' ..
2653 'silent echomsg "text"\_s*' ..
Bram Moolenaar02194d22020-10-24 23:08:38 +02002654 '\d CMDMOD silent\_s*' ..
Bram Moolenaarf4c6e1e2020-10-23 18:02:32 +02002655 '\d PUSHS "text"\_s*' ..
2656 '\d ECHOMSG 1\_s*' ..
Bram Moolenaar02194d22020-10-24 23:08:38 +02002657 '\d CMDMOD_REV\_s*' ..
Bram Moolenaarf4c6e1e2020-10-23 18:02:32 +02002658 'silent! echoerr "error"\_s*' ..
Bram Moolenaar02194d22020-10-24 23:08:38 +02002659 '\d CMDMOD silent!\_s*' ..
Bram Moolenaarf4c6e1e2020-10-23 18:02:32 +02002660 '\d PUSHS "error"\_s*' ..
2661 '\d ECHOERR 1\_s*' ..
Bram Moolenaar02194d22020-10-24 23:08:38 +02002662 '\d CMDMOD_REV\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +02002663 '\d\+ RETURN void',
Bram Moolenaara91a7132021-03-25 21:12:15 +01002664 res)
2665enddef
2666
2667def s:SilentIf()
2668 silent if 4 == g:five
2669 silent elseif 4 == g:five
Bram Moolenaarfa984412021-03-25 22:15:28 +01002670 endif
Bram Moolenaara91a7132021-03-25 21:12:15 +01002671enddef
2672
2673def Test_silent_if()
2674 var res = execute('disass s:SilentIf')
2675 assert_match('<SNR>\d*_SilentIf\_s*' ..
2676 'silent if 4 == g:five\_s*' ..
2677 '\d\+ CMDMOD silent\_s*' ..
2678 '\d\+ PUSHNR 4\_s*' ..
2679 '\d\+ LOADG g:five\_s*' ..
2680 '\d\+ COMPAREANY ==\_s*' ..
2681 '\d\+ CMDMOD_REV\_s*' ..
2682 '\d\+ JUMP_IF_FALSE -> \d\+\_s*' ..
2683 'silent elseif 4 == g:five\_s*' ..
2684 '\d\+ JUMP -> \d\+\_s*' ..
2685 '\d\+ CMDMOD silent\_s*' ..
2686 '\d\+ PUSHNR 4\_s*' ..
2687 '\d\+ LOADG g:five\_s*' ..
2688 '\d\+ COMPAREANY ==\_s*' ..
2689 '\d\+ CMDMOD_REV\_s*' ..
2690 '\d\+ JUMP_IF_FALSE -> \d\+\_s*' ..
Bram Moolenaarfa984412021-03-25 22:15:28 +01002691 'endif\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +02002692 '\d\+ RETURN void',
Bram Moolenaara91a7132021-03-25 21:12:15 +01002693 res)
2694enddef
2695
2696def s:SilentFor()
2697 silent for i in [0]
Bram Moolenaarfa984412021-03-25 22:15:28 +01002698 endfor
Bram Moolenaara91a7132021-03-25 21:12:15 +01002699enddef
2700
2701def Test_silent_for()
2702 var res = execute('disass s:SilentFor')
2703 assert_match('<SNR>\d*_SilentFor\_s*' ..
2704 'silent for i in \[0\]\_s*' ..
2705 '\d CMDMOD silent\_s*' ..
2706 '\d STORE -1 in $0\_s*' ..
2707 '\d PUSHNR 0\_s*' ..
2708 '\d NEWLIST size 1\_s*' ..
2709 '\d CMDMOD_REV\_s*' ..
2710 '5 FOR $0 -> 8\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01002711 '\d STORE $2\_s*' ..
2712
Bram Moolenaarfa984412021-03-25 22:15:28 +01002713 'endfor\_s*' ..
Bram Moolenaara91a7132021-03-25 21:12:15 +01002714 '\d JUMP -> 5\_s*' ..
2715 '8 DROP\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +02002716 '\d RETURN void\_s*',
Bram Moolenaara91a7132021-03-25 21:12:15 +01002717 res)
2718enddef
2719
2720def s:SilentWhile()
2721 silent while g:not
Bram Moolenaarfa984412021-03-25 22:15:28 +01002722 endwhile
Bram Moolenaara91a7132021-03-25 21:12:15 +01002723enddef
2724
2725def Test_silent_while()
2726 var res = execute('disass s:SilentWhile')
2727 assert_match('<SNR>\d*_SilentWhile\_s*' ..
2728 'silent while g:not\_s*' ..
2729 '0 CMDMOD silent\_s*' ..
2730 '\d LOADG g:not\_s*' ..
2731 '\d COND2BOOL\_s*' ..
2732 '\d CMDMOD_REV\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01002733 '\d WHILE $0 -> 6\_s*' ..
Bram Moolenaara91a7132021-03-25 21:12:15 +01002734
Bram Moolenaarfa984412021-03-25 22:15:28 +01002735 'endwhile\_s*' ..
Bram Moolenaara91a7132021-03-25 21:12:15 +01002736 '\d JUMP -> 0\_s*' ..
Bram Moolenaarf57b43c2021-06-15 22:13:27 +02002737 '6 RETURN void\_s*',
Bram Moolenaara91a7132021-03-25 21:12:15 +01002738 res)
2739enddef
2740
2741def s:SilentReturn(): string
2742 silent return "done"
2743enddef
2744
2745def Test_silent_return()
2746 var res = execute('disass s:SilentReturn')
2747 assert_match('<SNR>\d*_SilentReturn\_s*' ..
2748 'silent return "done"\_s*' ..
2749 '\d CMDMOD silent\_s*' ..
2750 '\d PUSHS "done"\_s*' ..
2751 '\d CMDMOD_REV\_s*' ..
2752 '\d RETURN',
Bram Moolenaarf4c6e1e2020-10-23 18:02:32 +02002753 res)
2754enddef
2755
Bram Moolenaarb2049902021-01-24 12:53:53 +01002756def s:Profiled(): string
Bram Moolenaar8cec9272021-06-23 20:20:53 +02002757 # comment
Bram Moolenaarb2049902021-01-24 12:53:53 +01002758 echo "profiled"
Bram Moolenaar8cec9272021-06-23 20:20:53 +02002759 # comment
Bram Moolenaar26d71162021-06-14 21:08:56 +02002760 var some = "some text"
Bram Moolenaarb2049902021-01-24 12:53:53 +01002761 return "done"
2762enddef
2763
2764def Test_profiled()
Bram Moolenaarf002a412021-01-24 13:34:18 +01002765 if !has('profile')
2766 MissingFeature 'profile'
2767 endif
Bram Moolenaare99d4222021-06-13 14:01:26 +02002768 var res = execute('disass profile s:Profiled')
Bram Moolenaarb2049902021-01-24 12:53:53 +01002769 assert_match('<SNR>\d*_Profiled\_s*' ..
Bram Moolenaar8cec9272021-06-23 20:20:53 +02002770 '# comment\_s*' ..
Bram Moolenaarb2049902021-01-24 12:53:53 +01002771 'echo "profiled"\_s*' ..
Bram Moolenaar8cec9272021-06-23 20:20:53 +02002772 '\d PROFILE START line 2\_s*' ..
Bram Moolenaarb2049902021-01-24 12:53:53 +01002773 '\d PUSHS "profiled"\_s*' ..
2774 '\d ECHO 1\_s*' ..
Bram Moolenaar8cec9272021-06-23 20:20:53 +02002775 '# comment\_s*' ..
Bram Moolenaar26d71162021-06-14 21:08:56 +02002776 'var some = "some text"\_s*' ..
Bram Moolenaarced68a02021-01-24 17:53:47 +01002777 '\d PROFILE END\_s*' ..
Bram Moolenaar8cec9272021-06-23 20:20:53 +02002778 '\d PROFILE START line 4\_s*' ..
Bram Moolenaar26d71162021-06-14 21:08:56 +02002779 '\d PUSHS "some text"\_s*' ..
2780 '\d STORE $0\_s*' ..
2781 'return "done"\_s*' ..
2782 '\d PROFILE END\_s*' ..
Bram Moolenaar8cec9272021-06-23 20:20:53 +02002783 '\d PROFILE START line 5\_s*' ..
Bram Moolenaarb2049902021-01-24 12:53:53 +01002784 '\d PUSHS "done"\_s*' ..
Bram Moolenaar26d71162021-06-14 21:08:56 +02002785 '\d\+ RETURN\_s*' ..
2786 '\d\+ PROFILE END',
Bram Moolenaarb2049902021-01-24 12:53:53 +01002787 res)
2788enddef
2789
Bram Moolenaare99d4222021-06-13 14:01:26 +02002790def Test_debugged()
2791 var res = execute('disass debug s:Profiled')
2792 assert_match('<SNR>\d*_Profiled\_s*' ..
Bram Moolenaar8cec9272021-06-23 20:20:53 +02002793 '# comment\_s*' ..
Bram Moolenaare99d4222021-06-13 14:01:26 +02002794 'echo "profiled"\_s*' ..
Bram Moolenaar8cec9272021-06-23 20:20:53 +02002795 '\d DEBUG line 1-2 varcount 0\_s*' ..
Bram Moolenaare99d4222021-06-13 14:01:26 +02002796 '\d PUSHS "profiled"\_s*' ..
2797 '\d ECHO 1\_s*' ..
Bram Moolenaar8cec9272021-06-23 20:20:53 +02002798 '# comment\_s*' ..
Bram Moolenaar26d71162021-06-14 21:08:56 +02002799 'var some = "some text"\_s*' ..
Bram Moolenaar8cec9272021-06-23 20:20:53 +02002800 '\d DEBUG line 3-4 varcount 0\_s*' ..
Bram Moolenaar26d71162021-06-14 21:08:56 +02002801 '\d PUSHS "some text"\_s*' ..
2802 '\d STORE $0\_s*' ..
Bram Moolenaare99d4222021-06-13 14:01:26 +02002803 'return "done"\_s*' ..
Bram Moolenaar8cec9272021-06-23 20:20:53 +02002804 '\d DEBUG line 5-5 varcount 1\_s*' ..
Bram Moolenaare99d4222021-06-13 14:01:26 +02002805 '\d PUSHS "done"\_s*' ..
2806 '\d RETURN\_s*',
2807 res)
2808enddef
2809
Bram Moolenaar90770b72021-11-30 20:57:38 +00002810def s:ElseifConstant()
2811 if g:value
2812 echo "one"
2813 elseif true
2814 echo "true"
2815 elseif false
2816 echo "false"
2817 endif
Bram Moolenaar1b5f7a62021-12-21 13:30:42 +00002818 if 0
2819 echo "yes"
2820 elseif 0
2821 echo "no"
2822 endif
Bram Moolenaar90770b72021-11-30 20:57:38 +00002823enddef
2824
2825def Test_debug_elseif_constant()
Bram Moolenaar1b5f7a62021-12-21 13:30:42 +00002826 var res = execute('disass debug s:ElseifConstant')
Bram Moolenaar90770b72021-11-30 20:57:38 +00002827 assert_match('<SNR>\d*_ElseifConstant\_s*' ..
2828 'if g:value\_s*' ..
Bram Moolenaar1b5f7a62021-12-21 13:30:42 +00002829 '0 DEBUG line 1-1 varcount 0\_s*' ..
2830 '1 LOADG g:value\_s*' ..
2831 '2 COND2BOOL\_s*' ..
2832 '3 JUMP_IF_FALSE -> 8\_s*' ..
Bram Moolenaar90770b72021-11-30 20:57:38 +00002833 'echo "one"\_s*' ..
Bram Moolenaar1b5f7a62021-12-21 13:30:42 +00002834 '4 DEBUG line 2-2 varcount 0\_s*' ..
2835 '5 PUSHS "one"\_s*' ..
2836 '6 ECHO 1\_s*' ..
Bram Moolenaar90770b72021-11-30 20:57:38 +00002837 'elseif true\_s*' ..
Bram Moolenaar1b5f7a62021-12-21 13:30:42 +00002838 '7 JUMP -> 12\_s*' ..
2839 '8 DEBUG line 3-3 varcount 0\_s*' ..
Bram Moolenaar90770b72021-11-30 20:57:38 +00002840 'echo "true"\_s*' ..
Bram Moolenaar1b5f7a62021-12-21 13:30:42 +00002841 '9 DEBUG line 4-4 varcount 0\_s*' ..
2842 '10 PUSHS "true"\_s*' ..
2843 '11 ECHO 1\_s*' ..
Bram Moolenaar90770b72021-11-30 20:57:38 +00002844 'elseif false\_s*' ..
2845 'echo "false"\_s*' ..
2846 'endif\_s*' ..
Bram Moolenaar1b5f7a62021-12-21 13:30:42 +00002847 'if 0\_s*' ..
2848 '12 DEBUG line 8-8 varcount 0\_s*' ..
2849 'echo "yes"\_s*' ..
2850 'elseif 0\_s*' ..
2851 '13 DEBUG line 11-10 varcount 0\_s*' ..
2852 'echo "no"\_s*' ..
2853 'endif\_s*' ..
2854 '14 RETURN void*',
Bram Moolenaar90770b72021-11-30 20:57:38 +00002855 res)
2856enddef
2857
Bram Moolenaar093165c2021-08-22 13:35:31 +02002858def s:DebugElseif()
2859 var b = false
2860 if b
2861 eval 1 + 0
2862 silent elseif !b
2863 eval 2 + 0
2864 endif
2865enddef
2866
2867def Test_debug_elseif()
2868 var res = execute('disass debug s:DebugElseif')
2869 assert_match('<SNR>\d*_DebugElseif\_s*' ..
2870 'var b = false\_s*' ..
2871 '0 DEBUG line 1-1 varcount 0\_s*' ..
2872 '1 PUSH false\_s*' ..
2873 '2 STORE $0\_s*' ..
2874
2875 'if b\_s*' ..
2876 '3 DEBUG line 2-2 varcount 1\_s*' ..
2877 '4 LOAD $0\_s*' ..
2878 '5 JUMP_IF_FALSE -> 10\_s*' ..
2879
2880 'eval 1 + 0\_s*' ..
2881 '6 DEBUG line 3-3 varcount 1\_s*' ..
2882 '7 PUSHNR 1\_s*' ..
2883 '8 DROP\_s*' ..
2884
2885 'silent elseif !b\_s*' ..
2886 '9 JUMP -> 20\_s*' ..
2887 '10 CMDMOD silent\_s*' ..
2888 '11 DEBUG line 4-4 varcount 1\_s*' ..
2889 '12 LOAD $0\_s*' ..
2890 '13 INVERT -1 (!val)\_s*' ..
2891 '14 CMDMOD_REV\_s*' ..
2892 '15 JUMP_IF_FALSE -> 20\_s*' ..
2893
2894 'eval 2 + 0\_s*' ..
2895 '16 DEBUG line 5-5 varcount 1\_s*' ..
2896 '17 PUSHNR 2\_s*' ..
2897 '18 DROP\_s*' ..
2898
2899 'endif\_s*' ..
2900 '19 DEBUG line 6-6 varcount 1\_s*' ..
2901 '20 RETURN void*',
2902 res)
2903enddef
2904
Bram Moolenaar2b4ecc22022-01-02 14:08:18 +00002905def s:DebugFor()
2906 echo "hello"
2907 for a in [0]
2908 echo a
2909 endfor
2910enddef
2911
2912def Test_debug_for()
2913 var res = execute('disass debug s:DebugFor')
2914 assert_match('<SNR>\d*_DebugFor\_s*' ..
2915 'echo "hello"\_s*' ..
2916 '0 DEBUG line 1-1 varcount 0\_s*' ..
2917 '1 PUSHS "hello"\_s*' ..
2918 '2 ECHO 1\_s*' ..
2919
2920 'for a in \[0\]\_s*' ..
2921 '3 DEBUG line 2-2 varcount 0\_s*' ..
2922 '4 STORE -1 in $0\_s*' ..
2923 '5 PUSHNR 0\_s*' ..
2924 '6 NEWLIST size 1\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01002925 '7 DEBUG line 2-2 varcount 3\_s*' ..
Bram Moolenaar2b4ecc22022-01-02 14:08:18 +00002926 '8 FOR $0 -> 15\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01002927 '9 STORE $2\_s*' ..
Bram Moolenaar2b4ecc22022-01-02 14:08:18 +00002928
2929 'echo a\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01002930 '10 DEBUG line 3-3 varcount 3\_s*' ..
2931 '11 LOAD $2\_s*' ..
Bram Moolenaar2b4ecc22022-01-02 14:08:18 +00002932 '12 ECHO 1\_s*' ..
2933
2934 'endfor\_s*' ..
Bram Moolenaarb46c0832022-09-15 17:19:37 +01002935 '13 DEBUG line 4-4 varcount 3\_s*' ..
Bram Moolenaar2b4ecc22022-01-02 14:08:18 +00002936 '14 JUMP -> 7\_s*' ..
2937 '15 DROP\_s*' ..
2938 '16 RETURN void*',
2939 res)
2940enddef
2941
Bram Moolenaar21ebb082022-02-04 21:58:58 +00002942def s:TryCatch()
2943 try
2944 echo "try"
2945 catch /error/
2946 echo "caught"
2947 endtry
2948enddef
2949
2950def Test_debug_try_catch()
2951 var res = execute('disass debug s:TryCatch')
2952 assert_match('<SNR>\d*_TryCatch\_s*' ..
2953 'try\_s*' ..
2954 '0 DEBUG line 1-1 varcount 0\_s*' ..
2955 '1 TRY catch -> 7, endtry -> 17\_s*' ..
2956 'echo "try"\_s*' ..
2957 '2 DEBUG line 2-2 varcount 0\_s*' ..
2958 '3 PUSHS "try"\_s*' ..
2959 '4 ECHO 1\_s*' ..
2960 'catch /error/\_s*' ..
2961 '5 DEBUG line 3-3 varcount 0\_s*' ..
2962 '6 JUMP -> 17\_s*' ..
2963 '7 DEBUG line 4-3 varcount 0\_s*' ..
2964 '8 PUSH v:exception\_s*' ..
2965 '9 PUSHS "error"\_s*' ..
2966 '10 COMPARESTRING =\~\_s*' ..
2967 '11 JUMP_IF_FALSE -> 17\_s*' ..
2968 '12 CATCH\_s*' ..
2969 'echo "caught"\_s*' ..
2970 '13 DEBUG line 4-4 varcount 0\_s*' ..
2971 '14 PUSHS "caught"\_s*' ..
2972 '15 ECHO 1\_s*' ..
2973 'endtry\_s*' ..
2974 '16 DEBUG line 5-5 varcount 0\_s*' ..
2975 '17 ENDTRY\_s*' ..
2976 '\d\+ RETURN void',
2977 res)
2978enddef
2979
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002980func s:Legacy() dict
Bram Moolenaarb1b6f4d2021-09-13 18:25:54 +02002981 echo 'legacy'
2982endfunc
2983
2984def s:UseMember()
2985 var d = {func: Legacy}
2986 var v = d.func()
2987enddef
2988
2989def Test_disassemble_dict_stack()
2990 var res = execute('disass s:UseMember')
2991 assert_match('<SNR>\d*_UseMember\_s*' ..
2992 'var d = {func: Legacy}\_s*' ..
2993 '\d PUSHS "func"\_s*' ..
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002994 '\d PUSHFUNC "<80><fd>R\d\+_Legacy"\_s*' ..
Bram Moolenaarb1b6f4d2021-09-13 18:25:54 +02002995 '\d NEWDICT size 1\_s*' ..
Bram Moolenaare88c6b72022-02-15 15:37:11 +00002996 '\d SETTYPE dict<func(...): any>\_s*' ..
Bram Moolenaarb1b6f4d2021-09-13 18:25:54 +02002997 '\d STORE $0\_s*' ..
2998
2999 'var v = d.func()\_s*' ..
3000 '\d LOAD $0\_s*' ..
3001 '\d MEMBER func\_s*' ..
3002 '\d PCALL top (argc 0)\_s*' ..
3003 '\d PCALL end\_s*' ..
3004 '\d CLEARDICT\_s*' ..
3005 '\d\+ STORE $1\_s*' ..
3006 '\d\+ RETURN void*',
3007 res)
3008enddef
3009
Bram Moolenaar95e4dd82022-04-27 22:15:40 +01003010def s:RetLegacy(): string
3011 legacy return "yes"
3012enddef
3013
3014def Test_disassemble_return_legacy()
3015 var res = execute('disass s:RetLegacy')
3016 assert_match('<SNR>\d*_RetLegacy\_s*' ..
3017 'legacy return "yes"\_s*' ..
3018 '\d CMDMOD legacy\_s*' ..
3019 '\d EVAL legacy "yes"\_s*' ..
3020 '\d CHECKTYPE string stack\[-1]\_s*' ..
3021 '\d CMDMOD_REV\_s*' ..
3022 '\d RETURN',
3023 res)
3024enddef
3025
Bram Moolenaarf62d7392021-04-14 12:40:00 +02003026def s:EchoMessages()
3027 echohl ErrorMsg | echom v:exception | echohl NONE
3028enddef
3029
3030def Test_disassemble_nextcmd()
3031 # splitting commands and removing trailing blanks should not change the line
3032 var res = execute('disass s:EchoMessages')
3033 assert_match('<SNR>\d*_EchoMessages\_s*' ..
3034 'echohl ErrorMsg | echom v:exception | echohl NONE',
3035 res)
3036enddef
3037
Bram Moolenaar6db660b2021-08-01 14:08:54 +02003038def Test_disassemble_after_reload()
Bram Moolenaar0e9bdad2022-10-15 20:06:33 +01003039 var lines =<< trim END
3040 vim9script
3041 if exists('g:ThisFunc')
3042 finish
3043 endif
3044 var name: any
3045 def g:ThisFunc(): number
3046 g:name = name
3047 return 0
3048 enddef
3049 def g:ThatFunc(): number
3050 name = g:name
3051 return 0
3052 enddef
3053 END
3054 lines->writefile('Xreload.vim', 'D')
Bram Moolenaar6db660b2021-08-01 14:08:54 +02003055
Bram Moolenaar0e9bdad2022-10-15 20:06:33 +01003056 source Xreload.vim
3057 g:ThisFunc()
3058 g:ThatFunc()
Bram Moolenaar6db660b2021-08-01 14:08:54 +02003059
Bram Moolenaar0e9bdad2022-10-15 20:06:33 +01003060 source Xreload.vim
3061 var res = execute('disass g:ThisFunc')
3062 assert_match('ThisFunc\_s*' ..
3063 'g:name = name\_s*' ..
3064 '\d LOADSCRIPT \[deleted\] from .*/Xreload.vim\_s*' ..
3065 '\d STOREG g:name\_s*' ..
3066 'return 0\_s*' ..
3067 '\d PUSHNR 0\_s*' ..
3068 '\d RETURN\_s*',
3069 res)
Bram Moolenaar6db660b2021-08-01 14:08:54 +02003070
Bram Moolenaar0e9bdad2022-10-15 20:06:33 +01003071 res = execute('disass g:ThatFunc')
3072 assert_match('ThatFunc\_s*' ..
3073 'name = g:name\_s*' ..
3074 '\d LOADG g:name\_s*' ..
3075 '\d STORESCRIPT \[deleted\] in .*/Xreload.vim\_s*' ..
3076 'return 0\_s*' ..
3077 '\d PUSHNR 0\_s*' ..
3078 '\d RETURN\_s*',
3079 res)
Bram Moolenaar6db660b2021-08-01 14:08:54 +02003080
Bram Moolenaar0e9bdad2022-10-15 20:06:33 +01003081 delfunc g:ThisFunc
3082 delfunc g:ThatFunc
Bram Moolenaar6db660b2021-08-01 14:08:54 +02003083enddef
3084
LemonBoy2eaef102022-05-06 13:14:50 +01003085def s:MakeString(x: number): string
3086 return $"x={x} x^2={x * x}"
3087enddef
Bram Moolenaar7cd24222021-01-12 18:58:39 +01003088
LemonBoy2eaef102022-05-06 13:14:50 +01003089def Test_disassemble_string_interp()
3090 var instr = execute('disassemble s:MakeString')
3091 assert_match('MakeString\_s*' ..
3092 'return $"x={x} x^2={x \* x}"\_s*' ..
3093 '0 PUSHS "x="\_s*' ..
3094 '1 LOAD arg\[-1\]\_s*' ..
3095 '2 2STRING stack\[-1\]\_s*' ..
3096 '3 PUSHS " x^2="\_s*' ..
3097 '4 LOAD arg\[-1\]\_s*' ..
3098 '5 LOAD arg\[-1\]\_s*' ..
3099 '6 OPNR \*\_s*' ..
3100 '7 2STRING stack\[-1\]\_s*' ..
3101 '8 CONCAT size 4\_s*' ..
3102 '9 RETURN\_s*',
3103 instr)
3104enddef
Bram Moolenaarb1b6f4d2021-09-13 18:25:54 +02003105
Yegappan Lakshmanana061f342022-05-22 19:13:49 +01003106def BitShift()
3107 var a = 1 << 2
3108 var b = 8 >> 1
3109 var c = a << b
Yegappan Lakshmananf4ee1cb2023-10-09 17:57:27 +02003110 var d = b >> a
Yegappan Lakshmanana061f342022-05-22 19:13:49 +01003111enddef
3112
3113def Test_disassemble_bitshift()
3114 var instr = execute('disassemble BitShift')
3115 assert_match('BitShift\_s*' ..
3116 'var a = 1 << 2\_s*' ..
3117 '0 STORE 4 in $0\_s*' ..
3118 'var b = 8 >> 1\_s*' ..
3119 '1 STORE 4 in $1\_s*' ..
3120 'var c = a << b\_s*' ..
3121 '2 LOAD $0\_s*' ..
3122 '3 LOAD $1\_s*' ..
3123 '4 OPNR <<\_s*' ..
3124 '5 STORE $2\_s*' ..
Yegappan Lakshmananf4ee1cb2023-10-09 17:57:27 +02003125 'var d = b >> a\_s*' ..
Yegappan Lakshmanana061f342022-05-22 19:13:49 +01003126 '6 LOAD $1\_s*' ..
3127 '7 LOAD $0\_s*' ..
Yegappan Lakshmananf4ee1cb2023-10-09 17:57:27 +02003128 '8 OPNR >>\_s*' ..
Yegappan Lakshmanana061f342022-05-22 19:13:49 +01003129 '9 STORE $3\_s*' ..
3130 '10 RETURN void', instr)
3131enddef
3132
Bram Moolenaar1d84f762022-09-03 21:35:53 +01003133def s:OneDefer()
3134 defer delete("file")
3135enddef
3136
3137def Test_disassemble_defer()
3138 var instr = execute('disassemble s:OneDefer')
3139 assert_match('OneDefer\_s*' ..
3140 'defer delete("file")\_s*' ..
3141 '\d PUSHFUNC "delete"\_s*' ..
3142 '\d PUSHS "file"\_s*' ..
3143 '\d DEFER 1 args\_s*' ..
3144 '\d RETURN\_s*',
3145 instr)
3146enddef
3147
Bram Moolenaar38f1ab32023-02-21 20:09:46 +00003148def Test_disassemble_class_function()
3149 var lines =<< trim END
3150 vim9script
3151
3152 class Cl
3153 static def Fc(): string
3154 return "x"
3155 enddef
3156 endclass
3157
3158 g:instr = execute('disassemble Cl.Fc')
3159 END
3160 v9.CheckScriptSuccess(lines)
3161 assert_match('Fc\_s*' ..
3162 'return "x"\_s*' ..
3163 '\d PUSHS "x"\_s*' ..
3164 '\d RETURN\_s*',
3165 g:instr)
3166
3167 lines =<< trim END
3168 vim9script
3169
3170 class Cl
3171 def Fo(): string
3172 return "y"
3173 enddef
3174 endclass
3175
3176 g:instr = execute('disassemble Cl.Fo')
3177 END
3178 v9.CheckScriptSuccess(lines)
3179 assert_match('Fo\_s*' ..
3180 'return "y"\_s*' ..
3181 '\d PUSHS "y"\_s*' ..
3182 '\d RETURN\_s*',
3183 g:instr)
3184
3185 unlet g:instr
3186enddef
3187
Yegappan Lakshmanan28a60f82023-09-05 20:42:18 +02003188" Disassemble instructions for using an interface with static and regular member
3189" variables.
3190def Test_disassemble_interface_static_member()
3191 var lines =<< trim END
3192 vim9script
3193 interface I
Doug Kearns74da0ee2023-12-14 20:26:26 +01003194 var o_var: number
3195 var o_var2: number
Yegappan Lakshmanan28a60f82023-09-05 20:42:18 +02003196 endinterface
3197
3198 class C implements I
Doug Kearns74da0ee2023-12-14 20:26:26 +01003199 public static var s_var: number
3200 var o_var: number
3201 public static var s_var2: number
3202 var o_var2: number
Yegappan Lakshmanan28a60f82023-09-05 20:42:18 +02003203 endclass
3204
3205 def F1(i: I)
3206 var x: number
Yegappan Lakshmanan28a60f82023-09-05 20:42:18 +02003207 x = i.o_var
Yegappan Lakshmanan28a60f82023-09-05 20:42:18 +02003208 x = i.o_var2
3209 enddef
3210
3211 def F2(o: C)
3212 var x: number
Yegappan Lakshmanan28a60f82023-09-05 20:42:18 +02003213 x = o.o_var
Yegappan Lakshmanan28a60f82023-09-05 20:42:18 +02003214 x = o.o_var2
3215 enddef
3216
3217 g:instr1 = execute('disassemble F1')
3218 g:instr2 = execute('disassemble F2')
3219 END
3220 v9.CheckScriptSuccess(lines)
3221 assert_match('<SNR>\d*_F1\_s*' ..
3222 'var x: number\_s*' ..
Yegappan Lakshmanan28a60f82023-09-05 20:42:18 +02003223 'x = i.o_var\_s*' ..
Yegappan Lakshmanan04054052023-09-10 18:12:56 +02003224 '0 LOAD arg\[-1\]\_s*' ..
3225 '1 ITF_MEMBER 0 on I\_s*' ..
3226 '2 STORE $0\_s*' ..
Yegappan Lakshmanan28a60f82023-09-05 20:42:18 +02003227 'x = i.o_var2\_s*' ..
Yegappan Lakshmanan04054052023-09-10 18:12:56 +02003228 '3 LOAD arg\[-1\]\_s*' ..
3229 '4 ITF_MEMBER 1 on I\_s*' ..
3230 '5 STORE $0\_s*' ..
3231 '6 RETURN void\_s*',
Yegappan Lakshmanan28a60f82023-09-05 20:42:18 +02003232 g:instr1)
3233 assert_match('<SNR>\d*_F2\_s*' ..
3234 'var x: number\_s*' ..
Yegappan Lakshmanan28a60f82023-09-05 20:42:18 +02003235 'x = o.o_var\_s*' ..
Yegappan Lakshmanan04054052023-09-10 18:12:56 +02003236 '0 LOAD arg\[-1\]\_s*' ..
3237 '1 OBJ_MEMBER 0\_s*' ..
3238 '2 STORE $0\_s*' ..
Yegappan Lakshmanan28a60f82023-09-05 20:42:18 +02003239 'x = o.o_var2\_s*' ..
Yegappan Lakshmanan04054052023-09-10 18:12:56 +02003240 '3 LOAD arg\[-1\]\_s*' ..
3241 '4 OBJ_MEMBER 1\_s*' ..
3242 '5 STORE $0\_s*' ..
3243 '6 RETURN void',
Yegappan Lakshmanan28a60f82023-09-05 20:42:18 +02003244 g:instr2)
3245
3246 unlet g:instr1
3247 unlet g:instr2
3248enddef
3249
Yegappan Lakshmananf4ee1cb2023-10-09 17:57:27 +02003250" Disassemble instructions for loading and storing class variables
3251def Test_disassemble_class_variable()
3252 var lines =<< trim END
3253 vim9script
3254
3255 class A
Doug Kearns74da0ee2023-12-14 20:26:26 +01003256 public static var val = 10
Yegappan Lakshmananf4ee1cb2023-10-09 17:57:27 +02003257 def Foo(): number
3258 val = 20
3259 return val
3260 enddef
3261 endclass
3262
3263 g:instr = execute('disassemble A.Foo')
3264 END
3265 v9.CheckScriptSuccess(lines)
3266 assert_match('Foo\_s*' ..
3267 'val = 20\_s*' ..
3268 '0 PUSHNR 20\_s*' ..
3269 '1 STORE CLASSMEMBER A.val\_s*' ..
3270 'return val\_s*' ..
3271 '2 LOAD CLASSMEMBER A.val\_s*' ..
3272 '3 RETURN', g:instr)
3273
3274 unlet g:instr
3275enddef
3276
3277" Disassemble instructions for METHODCALL
3278def Test_disassemble_methodcall()
3279 var lines =<< trim END
3280 vim9script
3281 interface A
3282 def Foo()
3283 endinterface
3284 def Bar(a: A)
3285 a.Foo()
3286 enddef
3287 g:instr = execute('disassemble Bar')
3288 END
3289 v9.CheckScriptSuccess(lines)
3290 assert_match('<SNR>\d*_Bar\_s*' ..
3291 'a.Foo()\_s*' ..
3292 '0 LOAD arg\[-1\]\_s*' ..
3293 '1 METHODCALL A.Foo(argc 0)\_s*' ..
3294 '2 DROP\_s*' ..
3295 '3 RETURN void', g:instr)
3296
3297 unlet g:instr
3298enddef
3299
3300" Disassemble instructions for ISN_JUMP_IF_ARG_NOT_SET
3301def Test_disassemble_ifargnotset()
3302 var lines =<< trim END
3303 vim9script
3304 class A
Doug Kearns74da0ee2023-12-14 20:26:26 +01003305 var val: number = 10
Yegappan Lakshmananf4ee1cb2023-10-09 17:57:27 +02003306 endclass
3307 g:instr = execute('disassemble A.new')
3308 END
3309 v9.CheckScriptSuccess(lines)
3310 assert_match('new\_s*' ..
3311 '0 NEW A size \d\+\_s*' ..
3312 '1 PUSHNR 10\_s*' ..
3313 '2 STORE_THIS 0\_s*' ..
3314 'ifargisset 0 this.val = val\_s*' ..
3315 '3 JUMP_IF_ARG_NOT_SET arg\[-1\] -> 8\_s*' ..
3316 '4 LOAD arg\[-1\]\_s*' ..
3317 '5 PUSHNR 0\_s*' ..
3318 '6 LOAD $0\_s*' ..
3319 '7 STOREINDEX object\_s*' ..
3320 '8 RETURN object', g:instr)
3321
3322 unlet g:instr
3323enddef
3324
Ernie Raele75fde62023-12-21 17:18:54 +01003325" Disassemble instructions for ISN_COMPAREOBJECT
Yegappan Lakshmananf4ee1cb2023-10-09 17:57:27 +02003326def Test_disassemble_compare_class_object()
3327 var lines =<< trim END
3328 vim9script
3329 class A
3330 endclass
3331 class B
3332 endclass
3333 def Foo(a: A, b: B)
Yegappan Lakshmananf4ee1cb2023-10-09 17:57:27 +02003334 if a == b
3335 endif
3336 enddef
3337 g:instr = execute('disassemble Foo')
3338 END
3339 v9.CheckScriptSuccess(lines)
3340 assert_match('<SNR>\d*_Foo\_s*' ..
Ernie Raele75fde62023-12-21 17:18:54 +01003341 'if a == b\_s*' ..
3342 '0 LOAD arg\[-2\]\_s*' ..
3343 '1 LOAD arg\[-1\]\_s*' ..
3344 '2 COMPAREOBJECT ==\_s*' ..
Yegappan Lakshmananf4ee1cb2023-10-09 17:57:27 +02003345 '3 JUMP_IF_FALSE -> 4\_s*' ..
3346 'endif\_s*' ..
Ernie Raele75fde62023-12-21 17:18:54 +01003347 '4 RETURN void', g:instr)
Yegappan Lakshmananf4ee1cb2023-10-09 17:57:27 +02003348 unlet g:instr
3349enddef
3350
3351" Disassemble instructions for ISN_CHECKTYPE with a float|number
3352def Test_checktype_float()
3353 var lines =<< trim END
3354 vim9script
3355 def Foo()
3356 var f: float = 0.0
3357 var a: any
3358 f += a
3359 enddef
3360 g:instr = execute('disassemble Foo')
3361 END
3362 v9.CheckScriptSuccess(lines)
3363 assert_match('<SNR>\d*_Foo\_s*' ..
3364 'var f: float = 0.0\_s*' ..
3365 '0 PUSHF 0.0\_s*' ..
3366 '1 STORE $0\_s*' ..
3367 'var a: any\_s*' ..
3368 'f += a\_s*' ..
3369 '2 LOAD $0\_s*' ..
3370 '3 LOAD $1\_s*' ..
3371 '4 CHECKTYPE float|number stack\[-1\]\_s*' ..
3372 '5 OPANY +\_s*' ..
3373 '6 STORE $0\_s*' ..
3374 '7 RETURN void', g:instr)
3375 unlet g:instr
3376enddef
3377
3378" Disassemble instructions for ISN_FUNCREF with a class
3379def Test_funcref_with_class()
3380 var lines =<< trim END
3381 vim9script
3382 class A
3383 def Foo()
3384 enddef
3385 endclass
3386 class B extends A
3387 def Foo()
3388 enddef
3389 endclass
3390 def Bar(a: A)
3391 defer a.Foo()
3392 enddef
3393 g:instr = execute('disassemble Bar')
3394 END
3395 v9.CheckScriptSuccess(lines)
3396 assert_match('<SNR>\d*_Bar\_s*' ..
3397 'defer a.Foo()\_s*' ..
3398 '0 LOAD arg\[-1\]\_s*' ..
3399 '1 FUNCREF A.Foo\_s*' ..
Yegappan Lakshmananf3eac692023-10-17 11:00:45 +02003400 '2 DEFER 0 args\_s*' ..
Yegappan Lakshmananf4ee1cb2023-10-09 17:57:27 +02003401 '3 RETURN void', g:instr)
3402 unlet g:instr
3403enddef
3404
Yegappan Lakshmanand3eae7b2024-03-03 16:26:58 +01003405" Disassemble instructions for calls to a string() function in an object
3406def Test_disassemble_object_string()
3407 var lines =<< trim END
3408 vim9script
3409 class A
3410 def string(): string
3411 return 'A'
3412 enddef
3413 endclass
3414 def Bar()
3415 var a = A.new()
3416 var s = string(a)
3417 s = string(A)
3418 enddef
3419 g:instr = execute('disassemble Bar')
3420 END
3421 v9.CheckScriptSuccess(lines)
3422 assert_match('<SNR>\d*_Bar\_s*' ..
3423 'var a = A.new()\_s*' ..
3424 '0 DCALL new(argc 0)\_s*' ..
3425 '1 STORE $0\_s*' ..
3426 'var s = string(a)\_s*' ..
3427 '2 LOAD $0\_s*' ..
3428 '3 METHODCALL A.string(argc 0)\_s*' ..
3429 '4 STORE $1\_s*' ..
3430 's = string(A)\_s*' ..
3431 '5 LOADSCRIPT A-0 from .*\_s*' ..
3432 '6 BCALL string(argc 1)\_s*' ..
3433 '7 STORE $1\_s*' ..
3434 '8 RETURN void', g:instr)
3435 unlet g:instr
3436
3437 # Use the default string() function for a class
3438 lines =<< trim END
3439 vim9script
3440 class A
3441 endclass
3442 def Bar()
3443 var a = A.new()
3444 var s = string(a)
3445 s = string(A)
3446 enddef
3447 g:instr = execute('disassemble Bar')
3448 END
3449 v9.CheckScriptSuccess(lines)
3450 assert_match('<SNR>\d*_Bar\_s*' ..
3451 'var a = A.new()\_s*' ..
3452 '0 DCALL new(argc 0)\_s*' ..
3453 '1 STORE $0\_s*' ..
3454 'var s = string(a)\_s*' ..
3455 '2 LOAD $0\_s*' ..
3456 '3 BCALL string(argc 1)\_s*' ..
3457 '4 STORE $1\_s*' ..
3458 's = string(A)\_s*' ..
3459 '5 LOADSCRIPT A-0 from .*\_s*' ..
3460 '6 BCALL string(argc 1)\_s*' ..
3461 '7 STORE $1\_s*' ..
3462 '8 RETURN void', g:instr)
3463 unlet g:instr
3464enddef
3465
3466" Disassemble instructions for calls to a empty() function in an object
3467def Test_disassemble_object_empty()
3468 var lines =<< trim END
3469 vim9script
3470 class A
3471 def empty(): bool
3472 return true
3473 enddef
3474 endclass
3475 def Bar()
3476 var a = A.new()
3477 var s = empty(a)
3478 enddef
3479 g:instr = execute('disassemble Bar')
3480 END
3481 v9.CheckScriptSuccess(lines)
3482 assert_match('<SNR>\d*_Bar\_s*' ..
3483 'var a = A.new()\_s*' ..
3484 '0 DCALL new(argc 0)\_s*' ..
3485 '1 STORE $0\_s*' ..
3486 'var s = empty(a)\_s*' ..
3487 '2 LOAD $0\_s*' ..
3488 '3 METHODCALL A.empty(argc 0)\_s*' ..
3489 '4 STORE $1\_s*' ..
3490 '5 RETURN void', g:instr)
3491 unlet g:instr
3492
3493 # Use the default empty() function for a class
3494 lines =<< trim END
3495 vim9script
3496 class A
3497 endclass
3498 def Bar()
3499 var a = A.new()
3500 var s = empty(a)
3501 enddef
3502 g:instr = execute('disassemble Bar')
3503 END
3504 v9.CheckScriptSuccess(lines)
3505 assert_match('<SNR>\d*_Bar\_s*' ..
3506 'var a = A.new()\_s*' ..
3507 '0 DCALL new(argc 0)\_s*' ..
3508 '1 STORE $0\_s*' ..
3509 'var s = empty(a)\_s*' ..
3510 '2 LOAD $0\_s*' ..
3511 '3 BCALL empty(argc 1)\_s*' ..
3512 '4 STORE $1\_s*' ..
3513 '5 RETURN void', g:instr)
3514 unlet g:instr
3515enddef
3516
3517" Disassemble instructions for calls to a len() function in an object
3518def Test_disassemble_object_len()
3519 var lines =<< trim END
3520 vim9script
3521 class A
3522 def len(): number
3523 return 10
3524 enddef
3525 endclass
3526 def Bar()
3527 var a = A.new()
3528 var s = len(a)
3529 enddef
3530 g:instr = execute('disassemble Bar')
3531 END
3532 v9.CheckScriptSuccess(lines)
3533 assert_match('<SNR>\d*_Bar\_s*' ..
3534 'var a = A.new()\_s*' ..
3535 '0 DCALL new(argc 0)\_s*' ..
3536 '1 STORE $0\_s*' ..
3537 'var s = len(a)\_s*' ..
3538 '2 LOAD $0\_s*' ..
3539 '3 METHODCALL A.len(argc 0)\_s*' ..
3540 '4 STORE $1\_s*' ..
3541 '5 RETURN void', g:instr)
3542 unlet g:instr
3543
3544 # Use the default len() function for a class
3545 lines =<< trim END
3546 vim9script
3547 class A
3548 endclass
3549 def Bar()
3550 var a = A.new()
3551 var s = len(a)
3552 enddef
3553 g:instr = execute('disassemble Bar')
3554 END
3555 v9.CheckScriptSuccess(lines)
3556 assert_match('<SNR>\d*_Bar\_s*' ..
3557 'var a = A.new()\_s*' ..
3558 '0 DCALL new(argc 0)\_s*' ..
3559 '1 STORE $0\_s*' ..
3560 'var s = len(a)\_s*' ..
3561 '2 LOAD $0\_s*' ..
3562 '3 BCALL len(argc 1)\_s*' ..
3563 '4 STORE $1\_s*' ..
3564 '5 RETURN void', g:instr)
3565 unlet g:instr
3566enddef
3567
Yegappan Lakshmanand990bf02024-03-22 19:56:17 +01003568" Disassemble instructions for using a compound operator in a closure
3569def Test_disassemble_compound_op_in_closure()
3570 var lines =<< trim END
3571 vim9script
3572 class A
3573 var foo: number = 1
3574 def Foo(): func
3575 var Fn = () => {
3576 this.foo += 1
3577 }
3578 return Fn
3579 enddef
3580 endclass
3581 var a = A.new()
3582 var Lambda = a.Foo()
3583 var num = matchstr(string(Lambda), '\d\+')
3584 g:instr = execute($'disassemble <lambda>{num}')
3585 END
3586 v9.CheckScriptSuccess(lines)
3587 assert_match('<lambda>\d\+\_s*' ..
3588 'this.foo += 1\_s*' ..
3589 '0 LOADOUTER level 0 $0\_s*' ..
3590 '1 OBJ_MEMBER 0\_s*' ..
3591 '2 PUSHNR 1\_s*' ..
3592 '3 OPNR +\_s*' ..
3593 '4 PUSHNR 0\_s*' ..
3594 '5 LOADOUTER level 0 $0\_s*' ..
3595 '6 STOREINDEX object\_s*' ..
3596 '7 RETURN void', g:instr)
3597 unlet g:instr
3598enddef
3599
LemonBoyf4af3312024-07-04 13:43:12 +02003600def Test_disassemble_member_initializer()
3601 var lines =<< trim END
3602 vim9script
3603 class A
3604 var l: list<string> = []
3605 var d: dict<string> = {}
3606 endclass
3607 g:instr = execute('disassemble A.new')
3608 END
3609 v9.CheckScriptSuccess(lines)
3610 # Ensure SETTYPE is emitted and that matches the declared type.
3611 assert_match('new\_s*' ..
3612 '0 NEW A size \d\+\_s*' ..
3613 '1 NEWLIST size 0\_s*' ..
3614 '2 SETTYPE list<string>\_s*' ..
3615 '3 STORE_THIS 0\_s*' ..
3616 '4 NEWDICT size 0\_s*' ..
3617 '5 SETTYPE dict<string>\_s*' ..
3618 '6 STORE_THIS 1', g:instr)
3619 unlet g:instr
3620enddef
3621
Yegappan Lakshmananc10342d2025-01-11 09:39:01 +01003622" Disassemble the code generated for accessing a interface member variable
3623def Test_disassemble_interface_variable_access()
3624 var lines =<< trim END
3625 vim9script
3626 interface IX
3627 var F: func(): string
3628 endinterface
3629
3630 def Foo(ix: IX): string
3631 return ix.F()
3632 enddef
3633
3634 g:instr = execute('disassemble Foo')
3635 END
3636 v9.CheckScriptSuccess(lines)
3637 assert_match('<SNR>\d\+_Foo\_s*' ..
3638 'return ix.F()\_s*' ..
3639 '0 LOAD arg\[-1\]\_s*' ..
3640 '1 ITF_MEMBER 0 on IX\_s*' ..
3641 '2 PCALL top (argc 0)\_s*' ..
3642 '3 PCALL end\_s*' ..
3643 '4 RETURN', g:instr)
3644 unlet g:instr
3645enddef
3646
Yegappan Lakshmanan6289f912025-01-14 17:13:36 +01003647" Disassemble the code generated for accessing a script-local funcref
3648def Test_disassemble_using_script_local_funcref()
3649 var lines =<< trim END
3650 vim9script
3651 def Noop()
3652 enddef
3653 export var Setup = Noop
3654 export def Run()
3655 Setup()
3656 enddef
3657 g:instr = execute('disassemble Run')
3658 END
3659 v9.CheckScriptSuccess(lines)
3660 assert_match('<SNR>\d\+_Run\_s*' ..
3661 'Setup()\_s*' ..
3662 '0 LOADSCRIPT Setup-0 from .*\_s*' ..
3663 '1 PCALL (argc 0)\_s*' ..
3664 '2 DROP\_s*' ..
3665 '3 RETURN void\_s*', g:instr)
3666 unlet g:instr
3667enddef
3668
Yegappan Lakshmanan16f2d3a2025-02-24 19:23:43 +01003669" Disassemble the code generated for using a script local variable
3670" in an instance variable initialization expression
3671def Test_disassemble_using_script_local_var_in_obj_init()
3672 var lines =<< trim END
3673 vim9script
3674 const DEFAULT = 'default-obj_key'
3675 export class ObjKey
3676 const unique_object_id3 = DEFAULT
3677 endclass
3678 END
3679 writefile(lines, 'Xscriptlocalobjinit.vim', 'D')
3680 lines =<< trim END
3681 vim9script
3682 import './Xscriptlocalobjinit.vim' as obj_key
3683
3684 class C1 extends obj_key.ObjKey
3685 endclass
3686 g:instr = execute('disassemble C1.new')
3687 END
3688 v9.CheckScriptSuccess(lines)
3689 assert_match('new\_s*' ..
3690 '0 NEW C1 size \d\+\_s*' ..
3691 '1 SCRIPTCTX_SET .*/Xscriptlocalobjinit.vim\_s*' ..
3692 '2 LOADSCRIPT DEFAULT-0 from .*/Xscriptlocalobjinit.vim\_s*' ..
3693 '3 SCRIPTCTX_SET .*\_s*' ..
3694 '4 STORE_THIS 0', g:instr)
3695 unlet g:instr
3696enddef
3697
Bram Moolenaar5cab73f2020-02-06 19:25:19 +01003698" vim: ts=8 sw=2 sts=2 expandtab tw=80 fdm=marker