Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 1 | " Test the :disassemble command, and compilation as a side effect |
| 2 | |
Bram Moolenaar | f51cb4e | 2020-03-01 17:55:14 +0100 | [diff] [blame] | 3 | source check.vim |
| 4 | |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 5 | func NotCompiled() |
| 6 | echo "not" |
| 7 | endfunc |
| 8 | |
| 9 | let s:scriptvar = 4 |
| 10 | let g:globalvar = 'g' |
Bram Moolenaar | d3aac29 | 2020-04-19 14:32:17 +0200 | [diff] [blame] | 11 | let b:buffervar = 'b' |
| 12 | let w:windowvar = 'w' |
| 13 | let t:tabpagevar = 't' |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 14 | |
| 15 | def s:ScriptFuncLoad(arg: string) |
| 16 | let local = 1 |
| 17 | buffers |
| 18 | echo arg |
| 19 | echo local |
Bram Moolenaar | 8a1c101 | 2020-05-07 14:07:25 +0200 | [diff] [blame] | 20 | echo &lines |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 21 | echo v:version |
| 22 | echo s:scriptvar |
| 23 | echo g:globalvar |
Bram Moolenaar | d3aac29 | 2020-04-19 14:32:17 +0200 | [diff] [blame] | 24 | echo b:buffervar |
| 25 | echo w:windowvar |
| 26 | echo t:tabpagevar |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 27 | echo &tabstop |
| 28 | echo $ENVVAR |
| 29 | echo @z |
| 30 | enddef |
| 31 | |
Bram Moolenaar | f2460a3 | 2020-02-07 22:09:54 +0100 | [diff] [blame] | 32 | def Test_disassemble_load() |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 33 | assert_fails('disass NoFunc', 'E1061:') |
| 34 | assert_fails('disass NotCompiled', 'E1062:') |
Bram Moolenaar | 21456cd | 2020-02-13 21:29:32 +0100 | [diff] [blame] | 35 | assert_fails('disass', 'E471:') |
| 36 | assert_fails('disass [', 'E475:') |
| 37 | assert_fails('disass 234', 'E475:') |
| 38 | assert_fails('disass <XX>foo', 'E475:') |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 39 | |
| 40 | let res = execute('disass s:ScriptFuncLoad') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 41 | assert_match('<SNR>\d*_ScriptFuncLoad.*' .. |
| 42 | 'buffers.*' .. |
| 43 | ' EXEC \+buffers.*' .. |
| 44 | ' LOAD arg\[-1\].*' .. |
| 45 | ' LOAD $0.*' .. |
Bram Moolenaar | 8a1c101 | 2020-05-07 14:07:25 +0200 | [diff] [blame] | 46 | ' LOADOPT &lines.*' .. |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 47 | ' LOADV v:version.*' .. |
| 48 | ' LOADS s:scriptvar from .*test_vim9_disassemble.vim.*' .. |
| 49 | ' LOADG g:globalvar.*' .. |
Bram Moolenaar | d3aac29 | 2020-04-19 14:32:17 +0200 | [diff] [blame] | 50 | ' LOADB b:buffervar.*' .. |
| 51 | ' LOADW w:windowvar.*' .. |
| 52 | ' LOADT t:tabpagevar.*' .. |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 53 | ' LOADENV $ENVVAR.*' .. |
| 54 | ' LOADREG @z.*', |
| 55 | res) |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 56 | enddef |
| 57 | |
Bram Moolenaar | cfe435d | 2020-04-25 20:02:55 +0200 | [diff] [blame] | 58 | def s:EditExpand() |
| 59 | let filename = "file" |
| 60 | let filenr = 123 |
| 61 | edit the`=filename``=filenr`.txt |
| 62 | enddef |
| 63 | |
| 64 | def Test_disassemble_exec_expr() |
| 65 | let res = execute('disass s:EditExpand') |
| 66 | assert_match('<SNR>\d*_EditExpand.*' .. |
| 67 | ' let filename = "file".*' .. |
| 68 | '\d PUSHS "file".*' .. |
| 69 | '\d STORE $0.*' .. |
| 70 | ' let filenr = 123.*' .. |
| 71 | '\d STORE 123 in $1.*' .. |
| 72 | ' edit the`=filename``=filenr`.txt.*' .. |
| 73 | '\d PUSHS "edit the".*' .. |
| 74 | '\d LOAD $0.*' .. |
| 75 | '\d LOAD $1.*' .. |
| 76 | '\d 2STRING stack\[-1\].*' .. |
| 77 | '\d PUSHS ".txt".*' .. |
| 78 | '\d EXECCONCAT 4.*' .. |
| 79 | '\d PUSHNR 0.*' .. |
| 80 | '\d RETURN', |
| 81 | res) |
| 82 | enddef |
| 83 | |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 84 | def s:ScriptFuncPush() |
| 85 | let localbool = true |
| 86 | let localspec = v:none |
| 87 | let localblob = 0z1234 |
| 88 | if has('float') |
| 89 | let localfloat = 1.234 |
| 90 | endif |
| 91 | enddef |
| 92 | |
Bram Moolenaar | f2460a3 | 2020-02-07 22:09:54 +0100 | [diff] [blame] | 93 | def Test_disassemble_push() |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 94 | let res = execute('disass s:ScriptFuncPush') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 95 | assert_match('<SNR>\d*_ScriptFuncPush.*' .. |
| 96 | 'localbool = true.*' .. |
| 97 | ' PUSH v:true.*' .. |
| 98 | 'localspec = v:none.*' .. |
| 99 | ' PUSH v:none.*' .. |
| 100 | 'localblob = 0z1234.*' .. |
| 101 | ' PUSHBLOB 0z1234.*', |
| 102 | res) |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 103 | if has('float') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 104 | assert_match('<SNR>\d*_ScriptFuncPush.*' .. |
| 105 | 'localfloat = 1.234.*' .. |
| 106 | ' PUSHF 1.234.*', |
| 107 | res) |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 108 | endif |
| 109 | enddef |
| 110 | |
| 111 | def s:ScriptFuncStore() |
| 112 | let localnr = 1 |
| 113 | localnr = 2 |
| 114 | let localstr = 'abc' |
| 115 | localstr = 'xyz' |
| 116 | v:char = 'abc' |
| 117 | s:scriptvar = 'sv' |
| 118 | g:globalvar = 'gv' |
Bram Moolenaar | d3aac29 | 2020-04-19 14:32:17 +0200 | [diff] [blame] | 119 | b:buffervar = 'bv' |
| 120 | w:windowvar = 'wv' |
| 121 | t:tabpagevar = 'tv' |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 122 | &tabstop = 8 |
| 123 | $ENVVAR = 'ev' |
| 124 | @z = 'rv' |
| 125 | enddef |
| 126 | |
Bram Moolenaar | f2460a3 | 2020-02-07 22:09:54 +0100 | [diff] [blame] | 127 | def Test_disassemble_store() |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 128 | let res = execute('disass s:ScriptFuncStore') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 129 | assert_match('<SNR>\d*_ScriptFuncStore.*' .. |
| 130 | 'let localnr = 1.*' .. |
| 131 | 'localnr = 2.*' .. |
| 132 | ' STORE 2 in $0.*' .. |
| 133 | 'let localstr = ''abc''.*' .. |
| 134 | 'localstr = ''xyz''.*' .. |
| 135 | ' STORE $1.*' .. |
| 136 | 'v:char = ''abc''.*' .. |
| 137 | 'STOREV v:char.*' .. |
| 138 | 's:scriptvar = ''sv''.*' .. |
| 139 | ' STORES s:scriptvar in .*test_vim9_disassemble.vim.*' .. |
| 140 | 'g:globalvar = ''gv''.*' .. |
| 141 | ' STOREG g:globalvar.*' .. |
Bram Moolenaar | d3aac29 | 2020-04-19 14:32:17 +0200 | [diff] [blame] | 142 | 'b:buffervar = ''bv''.*' .. |
| 143 | ' STOREB b:buffervar.*' .. |
| 144 | 'w:windowvar = ''wv''.*' .. |
| 145 | ' STOREW w:windowvar.*' .. |
| 146 | 't:tabpagevar = ''tv''.*' .. |
| 147 | ' STORET t:tabpagevar.*' .. |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 148 | '&tabstop = 8.*' .. |
| 149 | ' STOREOPT &tabstop.*' .. |
| 150 | '$ENVVAR = ''ev''.*' .. |
| 151 | ' STOREENV $ENVVAR.*' .. |
| 152 | '@z = ''rv''.*' .. |
| 153 | ' STOREREG @z.*', |
| 154 | res) |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 155 | enddef |
| 156 | |
Bram Moolenaar | d72c1bf | 2020-04-19 16:28:59 +0200 | [diff] [blame] | 157 | def s:ScriptFuncUnlet() |
| 158 | g:somevar = "value" |
| 159 | unlet g:somevar |
| 160 | unlet! g:somevar |
Bram Moolenaar | 7bdaea6 | 2020-04-19 18:27:26 +0200 | [diff] [blame] | 161 | unlet $SOMEVAR |
Bram Moolenaar | d72c1bf | 2020-04-19 16:28:59 +0200 | [diff] [blame] | 162 | enddef |
| 163 | |
| 164 | def Test_disassemble_unlet() |
| 165 | let res = execute('disass s:ScriptFuncUnlet') |
| 166 | assert_match('<SNR>\d*_ScriptFuncUnlet.*' .. |
| 167 | 'g:somevar = "value".*' .. |
| 168 | '\d PUSHS "value".*' .. |
| 169 | '\d STOREG g:somevar.*' .. |
| 170 | 'unlet g:somevar.*' .. |
| 171 | '\d UNLET g:somevar.*' .. |
| 172 | 'unlet! g:somevar.*' .. |
Bram Moolenaar | 7bdaea6 | 2020-04-19 18:27:26 +0200 | [diff] [blame] | 173 | '\d UNLET! g:somevar.*' .. |
| 174 | 'unlet $SOMEVAR.*' .. |
| 175 | '\d UNLETENV $SOMEVAR.*', |
Bram Moolenaar | d72c1bf | 2020-04-19 16:28:59 +0200 | [diff] [blame] | 176 | res) |
| 177 | enddef |
| 178 | |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 179 | def s:ScriptFuncTry() |
| 180 | try |
| 181 | echo 'yes' |
| 182 | catch /fail/ |
| 183 | echo 'no' |
| 184 | finally |
Bram Moolenaar | c2a4b35 | 2020-02-06 22:41:16 +0100 | [diff] [blame] | 185 | throw 'end' |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 186 | endtry |
| 187 | enddef |
| 188 | |
Bram Moolenaar | f2460a3 | 2020-02-07 22:09:54 +0100 | [diff] [blame] | 189 | def Test_disassemble_try() |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 190 | let res = execute('disass s:ScriptFuncTry') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 191 | assert_match('<SNR>\d*_ScriptFuncTry.*' .. |
| 192 | 'try.*' .. |
| 193 | 'TRY catch -> \d\+, finally -> \d\+.*' .. |
| 194 | 'catch /fail/.*' .. |
| 195 | ' JUMP -> \d\+.*' .. |
| 196 | ' PUSH v:exception.*' .. |
| 197 | ' PUSHS "fail".*' .. |
| 198 | ' COMPARESTRING =\~.*' .. |
| 199 | ' JUMP_IF_FALSE -> \d\+.*' .. |
| 200 | ' CATCH.*' .. |
| 201 | 'finally.*' .. |
| 202 | ' PUSHS "end".*' .. |
| 203 | ' THROW.*' .. |
| 204 | 'endtry.*' .. |
| 205 | ' ENDTRY.*', |
| 206 | res) |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 207 | enddef |
| 208 | |
| 209 | def s:ScriptFuncNew() |
| 210 | let ll = [1, "two", 333] |
| 211 | let dd = #{one: 1, two: "val"} |
| 212 | enddef |
| 213 | |
Bram Moolenaar | f2460a3 | 2020-02-07 22:09:54 +0100 | [diff] [blame] | 214 | def Test_disassemble_new() |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 215 | let res = execute('disass s:ScriptFuncNew') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 216 | assert_match('<SNR>\d*_ScriptFuncNew.*' .. |
| 217 | 'let ll = \[1, "two", 333].*' .. |
| 218 | 'PUSHNR 1.*' .. |
| 219 | 'PUSHS "two".*' .. |
| 220 | 'PUSHNR 333.*' .. |
| 221 | 'NEWLIST size 3.*' .. |
| 222 | 'let dd = #{one: 1, two: "val"}.*' .. |
| 223 | 'PUSHS "one".*' .. |
| 224 | 'PUSHNR 1.*' .. |
| 225 | 'PUSHS "two".*' .. |
| 226 | 'PUSHS "val".*' .. |
| 227 | 'NEWDICT size 2.*', |
| 228 | res) |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 229 | enddef |
| 230 | |
Bram Moolenaar | 6e94978 | 2020-04-13 17:21:00 +0200 | [diff] [blame] | 231 | def FuncWithArg(arg: any) |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 232 | echo arg |
| 233 | enddef |
| 234 | |
| 235 | func UserFunc() |
| 236 | echo 'nothing' |
| 237 | endfunc |
| 238 | |
| 239 | func UserFuncWithArg(arg) |
| 240 | echo a:arg |
| 241 | endfunc |
| 242 | |
| 243 | def s:ScriptFuncCall(): string |
| 244 | changenr() |
| 245 | char2nr("abc") |
Bram Moolenaar | f2460a3 | 2020-02-07 22:09:54 +0100 | [diff] [blame] | 246 | Test_disassemble_new() |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 247 | FuncWithArg(343) |
| 248 | ScriptFuncNew() |
| 249 | s:ScriptFuncNew() |
| 250 | UserFunc() |
| 251 | UserFuncWithArg("foo") |
| 252 | let FuncRef = function("UserFunc") |
| 253 | FuncRef() |
| 254 | let FuncRefWithArg = function("UserFuncWithArg") |
| 255 | FuncRefWithArg("bar") |
| 256 | return "yes" |
| 257 | enddef |
| 258 | |
Bram Moolenaar | f2460a3 | 2020-02-07 22:09:54 +0100 | [diff] [blame] | 259 | def Test_disassemble_call() |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 260 | let res = execute('disass s:ScriptFuncCall') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 261 | assert_match('<SNR>\d\+_ScriptFuncCall.*' .. |
| 262 | 'changenr().*' .. |
| 263 | ' BCALL changenr(argc 0).*' .. |
| 264 | 'char2nr("abc").*' .. |
| 265 | ' PUSHS "abc".*' .. |
| 266 | ' BCALL char2nr(argc 1).*' .. |
| 267 | 'Test_disassemble_new().*' .. |
| 268 | ' DCALL Test_disassemble_new(argc 0).*' .. |
| 269 | 'FuncWithArg(343).*' .. |
| 270 | ' PUSHNR 343.*' .. |
| 271 | ' DCALL FuncWithArg(argc 1).*' .. |
| 272 | 'ScriptFuncNew().*' .. |
| 273 | ' DCALL <SNR>\d\+_ScriptFuncNew(argc 0).*' .. |
| 274 | 's:ScriptFuncNew().*' .. |
| 275 | ' DCALL <SNR>\d\+_ScriptFuncNew(argc 0).*' .. |
| 276 | 'UserFunc().*' .. |
| 277 | ' UCALL UserFunc(argc 0).*' .. |
| 278 | 'UserFuncWithArg("foo").*' .. |
| 279 | ' PUSHS "foo".*' .. |
| 280 | ' UCALL UserFuncWithArg(argc 1).*' .. |
| 281 | 'let FuncRef = function("UserFunc").*' .. |
| 282 | 'FuncRef().*' .. |
| 283 | ' LOAD $\d.*' .. |
| 284 | ' PCALL (argc 0).*' .. |
| 285 | 'let FuncRefWithArg = function("UserFuncWithArg").*' .. |
| 286 | 'FuncRefWithArg("bar").*' .. |
| 287 | ' PUSHS "bar".*' .. |
| 288 | ' LOAD $\d.*' .. |
| 289 | ' PCALL (argc 1).*' .. |
| 290 | 'return "yes".*' .. |
| 291 | ' PUSHS "yes".*' .. |
| 292 | ' RETURN.*', |
| 293 | res) |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 294 | enddef |
| 295 | |
Bram Moolenaar | b68b346 | 2020-05-06 21:06:30 +0200 | [diff] [blame] | 296 | def s:CreateRefs() |
| 297 | let local = 'a' |
| 298 | def Append(arg: string) |
| 299 | local ..= arg |
| 300 | enddef |
| 301 | g:Append = Append |
| 302 | def Get(): string |
| 303 | return local |
| 304 | enddef |
| 305 | g:Get = Get |
| 306 | enddef |
| 307 | |
| 308 | def Test_disassemble_closure() |
| 309 | CreateRefs() |
| 310 | let res = execute('disass g:Append') |
| 311 | assert_match('<lambda>\d.*' .. |
| 312 | 'local ..= arg.*' .. |
| 313 | '\d LOADOUTER $0.*' .. |
| 314 | '\d LOAD arg\[-1\].*' .. |
| 315 | '\d CONCAT.*' .. |
| 316 | '\d STOREOUTER $0.*' .. |
| 317 | '\d PUSHNR 0.*' .. |
| 318 | '\d RETURN.*', |
| 319 | res) |
| 320 | |
| 321 | res = execute('disass g:Get') |
| 322 | assert_match('<lambda>\d.*' .. |
| 323 | 'return local.*' .. |
| 324 | '\d LOADOUTER $0.*' .. |
| 325 | '\d RETURN.*', |
| 326 | res) |
| 327 | |
| 328 | unlet g:Append |
| 329 | unlet g:Get |
| 330 | enddef |
| 331 | |
Bram Moolenaar | 8ed0458 | 2020-02-22 19:07:28 +0100 | [diff] [blame] | 332 | |
Bram Moolenaar | bd5da37 | 2020-03-31 23:13:10 +0200 | [diff] [blame] | 333 | def EchoArg(arg: string): string |
| 334 | return arg |
| 335 | enddef |
| 336 | def RefThis(): func |
| 337 | return function('EchoArg') |
| 338 | enddef |
| 339 | def s:ScriptPCall() |
| 340 | RefThis()("text") |
| 341 | enddef |
| 342 | |
| 343 | def Test_disassemble_pcall() |
| 344 | let res = execute('disass s:ScriptPCall') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 345 | assert_match('<SNR>\d\+_ScriptPCall.*' .. |
| 346 | 'RefThis()("text").*' .. |
| 347 | '\d DCALL RefThis(argc 0).*' .. |
| 348 | '\d PUSHS "text".*' .. |
| 349 | '\d PCALL top (argc 1).*' .. |
| 350 | '\d PCALL end.*' .. |
| 351 | '\d DROP.*' .. |
| 352 | '\d PUSHNR 0.*' .. |
| 353 | '\d RETURN.*', |
| 354 | res) |
Bram Moolenaar | bd5da37 | 2020-03-31 23:13:10 +0200 | [diff] [blame] | 355 | enddef |
| 356 | |
| 357 | |
Bram Moolenaar | a26b970 | 2020-04-18 19:53:28 +0200 | [diff] [blame] | 358 | def s:FuncWithForwardCall(): string |
| 359 | return g:DefinedLater("yes") |
Bram Moolenaar | 7eeefd4 | 2020-02-26 21:24:23 +0100 | [diff] [blame] | 360 | enddef |
| 361 | |
| 362 | def DefinedLater(arg: string): string |
| 363 | return arg |
| 364 | enddef |
| 365 | |
| 366 | def Test_disassemble_update_instr() |
Bram Moolenaar | a26b970 | 2020-04-18 19:53:28 +0200 | [diff] [blame] | 367 | let res = execute('disass s:FuncWithForwardCall') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 368 | assert_match('FuncWithForwardCall.*' .. |
Bram Moolenaar | a26b970 | 2020-04-18 19:53:28 +0200 | [diff] [blame] | 369 | 'return g:DefinedLater("yes").*' .. |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 370 | '\d PUSHS "yes".*' .. |
Bram Moolenaar | a26b970 | 2020-04-18 19:53:28 +0200 | [diff] [blame] | 371 | '\d UCALL g:DefinedLater(argc 1).*' .. |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 372 | '\d CHECKTYPE string stack\[-1].*' .. |
| 373 | '\d RETURN.*', |
| 374 | res) |
Bram Moolenaar | 7eeefd4 | 2020-02-26 21:24:23 +0100 | [diff] [blame] | 375 | |
| 376 | " Calling the function will change UCALL into the faster DCALL |
| 377 | assert_equal('yes', FuncWithForwardCall()) |
| 378 | |
Bram Moolenaar | a26b970 | 2020-04-18 19:53:28 +0200 | [diff] [blame] | 379 | res = execute('disass s:FuncWithForwardCall') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 380 | assert_match('FuncWithForwardCall.*' .. |
Bram Moolenaar | a26b970 | 2020-04-18 19:53:28 +0200 | [diff] [blame] | 381 | 'return g:DefinedLater("yes").*' .. |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 382 | '\d PUSHS "yes".*' .. |
| 383 | '\d DCALL DefinedLater(argc 1).*' .. |
| 384 | '\d CHECKTYPE string stack\[-1].*' .. |
| 385 | '\d RETURN.*', |
| 386 | res) |
Bram Moolenaar | 7eeefd4 | 2020-02-26 21:24:23 +0100 | [diff] [blame] | 387 | enddef |
| 388 | |
| 389 | |
Bram Moolenaar | 8ed0458 | 2020-02-22 19:07:28 +0100 | [diff] [blame] | 390 | def FuncWithDefault(arg: string = 'default'): string |
| 391 | return arg |
| 392 | enddef |
| 393 | |
| 394 | def Test_disassemble_call_default() |
| 395 | let res = execute('disass FuncWithDefault') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 396 | assert_match('FuncWithDefault.*' .. |
| 397 | '\d PUSHS "default".*' .. |
| 398 | '\d STORE arg\[-1].*' .. |
| 399 | 'return arg.*' .. |
| 400 | '\d LOAD arg\[-1].*' .. |
| 401 | '\d RETURN.*', |
| 402 | res) |
Bram Moolenaar | 8ed0458 | 2020-02-22 19:07:28 +0100 | [diff] [blame] | 403 | enddef |
| 404 | |
| 405 | |
Bram Moolenaar | 158906c | 2020-02-06 20:39:45 +0100 | [diff] [blame] | 406 | def HasEval() |
| 407 | if has("eval") |
| 408 | echo "yes" |
| 409 | else |
| 410 | echo "no" |
| 411 | endif |
| 412 | enddef |
| 413 | |
| 414 | def HasNothing() |
| 415 | if has("nothing") |
| 416 | echo "yes" |
| 417 | else |
| 418 | echo "no" |
| 419 | endif |
| 420 | enddef |
| 421 | |
| 422 | def HasSomething() |
| 423 | if has("nothing") |
| 424 | echo "nothing" |
| 425 | elseif has("something") |
| 426 | echo "something" |
| 427 | elseif has("eval") |
| 428 | echo "eval" |
| 429 | elseif has("less") |
| 430 | echo "less" |
| 431 | endif |
| 432 | enddef |
| 433 | |
Bram Moolenaar | f2460a3 | 2020-02-07 22:09:54 +0100 | [diff] [blame] | 434 | def Test_disassemble_const_expr() |
Bram Moolenaar | 158906c | 2020-02-06 20:39:45 +0100 | [diff] [blame] | 435 | assert_equal("\nyes", execute('call HasEval()')) |
| 436 | let instr = execute('disassemble HasEval') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 437 | assert_match('HasEval.*' .. |
| 438 | 'if has("eval").*' .. |
| 439 | ' PUSHS "yes".*', |
| 440 | instr) |
Bram Moolenaar | 158906c | 2020-02-06 20:39:45 +0100 | [diff] [blame] | 441 | assert_notmatch('JUMP', instr) |
| 442 | |
| 443 | assert_equal("\nno", execute('call HasNothing()')) |
| 444 | instr = execute('disassemble HasNothing') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 445 | assert_match('HasNothing.*' .. |
| 446 | 'if has("nothing").*' .. |
| 447 | 'else.*' .. |
| 448 | ' PUSHS "no".*', |
| 449 | instr) |
Bram Moolenaar | 158906c | 2020-02-06 20:39:45 +0100 | [diff] [blame] | 450 | assert_notmatch('PUSHS "yes"', instr) |
| 451 | assert_notmatch('JUMP', instr) |
| 452 | |
| 453 | assert_equal("\neval", execute('call HasSomething()')) |
| 454 | instr = execute('disassemble HasSomething') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 455 | assert_match('HasSomething.*' .. |
| 456 | 'if has("nothing").*' .. |
| 457 | 'elseif has("something").*' .. |
| 458 | 'elseif has("eval").*' .. |
| 459 | ' PUSHS "eval".*' .. |
| 460 | 'elseif has("less").*', |
| 461 | instr) |
Bram Moolenaar | 158906c | 2020-02-06 20:39:45 +0100 | [diff] [blame] | 462 | assert_notmatch('PUSHS "nothing"', instr) |
| 463 | assert_notmatch('PUSHS "something"', instr) |
| 464 | assert_notmatch('PUSHS "less"', instr) |
| 465 | assert_notmatch('JUMP', instr) |
| 466 | enddef |
| 467 | |
Bram Moolenaar | f51cb4e | 2020-03-01 17:55:14 +0100 | [diff] [blame] | 468 | def WithFunc() |
Bram Moolenaar | 5deeb3f | 2020-04-05 17:08:17 +0200 | [diff] [blame] | 469 | let Funky1: func |
| 470 | let Funky2: func = function("len") |
| 471 | let Party2: func = funcref("UserFunc") |
Bram Moolenaar | f51cb4e | 2020-03-01 17:55:14 +0100 | [diff] [blame] | 472 | enddef |
| 473 | |
| 474 | def Test_disassemble_function() |
| 475 | let instr = execute('disassemble WithFunc') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 476 | assert_match('WithFunc.*' .. |
| 477 | 'let Funky1: func.*' .. |
| 478 | '0 PUSHFUNC "\[none]".*' .. |
| 479 | '1 STORE $0.*' .. |
| 480 | 'let Funky2: func = function("len").*' .. |
| 481 | '2 PUSHS "len".*' .. |
| 482 | '3 BCALL function(argc 1).*' .. |
| 483 | '4 STORE $1.*' .. |
| 484 | 'let Party2: func = funcref("UserFunc").*' .. |
| 485 | '\d PUSHS "UserFunc".*' .. |
| 486 | '\d BCALL funcref(argc 1).*' .. |
| 487 | '\d STORE $2.*' .. |
| 488 | '\d PUSHNR 0.*' .. |
| 489 | '\d RETURN.*', |
| 490 | instr) |
Bram Moolenaar | f51cb4e | 2020-03-01 17:55:14 +0100 | [diff] [blame] | 491 | enddef |
| 492 | |
| 493 | if has('channel') |
| 494 | def WithChannel() |
| 495 | let job1: job |
| 496 | let job2: job = job_start("donothing") |
| 497 | let chan1: channel |
| 498 | enddef |
| 499 | endif |
| 500 | |
| 501 | def Test_disassemble_channel() |
| 502 | CheckFeature channel |
| 503 | |
| 504 | let instr = execute('disassemble WithChannel') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 505 | assert_match('WithChannel.*' .. |
| 506 | 'let job1: job.*' .. |
| 507 | '\d PUSHJOB "no process".*' .. |
| 508 | '\d STORE $0.*' .. |
| 509 | 'let job2: job = job_start("donothing").*' .. |
| 510 | '\d PUSHS "donothing".*' .. |
| 511 | '\d BCALL job_start(argc 1).*' .. |
| 512 | '\d STORE $1.*' .. |
| 513 | 'let chan1: channel.*' .. |
| 514 | '\d PUSHCHANNEL 0.*' .. |
| 515 | '\d STORE $2.*' .. |
| 516 | '\d PUSHNR 0.*' .. |
| 517 | '\d RETURN.*', |
| 518 | instr) |
Bram Moolenaar | f51cb4e | 2020-03-01 17:55:14 +0100 | [diff] [blame] | 519 | enddef |
| 520 | |
Bram Moolenaar | 777770f | 2020-02-06 21:27:08 +0100 | [diff] [blame] | 521 | def WithLambda(): string |
| 522 | let F = {a -> "X" .. a .. "X"} |
| 523 | return F("x") |
| 524 | enddef |
| 525 | |
Bram Moolenaar | f2460a3 | 2020-02-07 22:09:54 +0100 | [diff] [blame] | 526 | def Test_disassemble_lambda() |
Bram Moolenaar | 777770f | 2020-02-06 21:27:08 +0100 | [diff] [blame] | 527 | assert_equal("XxX", WithLambda()) |
| 528 | let instr = execute('disassemble WithLambda') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 529 | assert_match('WithLambda.*' .. |
| 530 | 'let F = {a -> "X" .. a .. "X"}.*' .. |
| 531 | ' FUNCREF <lambda>\d\+.*' .. |
| 532 | 'PUSHS "x".*' .. |
| 533 | ' LOAD $0.*' .. |
| 534 | ' PCALL (argc 1).*' .. |
| 535 | ' CHECKTYPE string stack\[-1].*', |
| 536 | instr) |
Bram Moolenaar | 777770f | 2020-02-06 21:27:08 +0100 | [diff] [blame] | 537 | enddef |
| 538 | |
Bram Moolenaar | 6e94978 | 2020-04-13 17:21:00 +0200 | [diff] [blame] | 539 | def AndOr(arg: any): string |
Bram Moolenaar | 777770f | 2020-02-06 21:27:08 +0100 | [diff] [blame] | 540 | if arg == 1 && arg != 2 || arg == 4 |
| 541 | return 'yes' |
| 542 | endif |
| 543 | return 'no' |
| 544 | enddef |
| 545 | |
Bram Moolenaar | f2460a3 | 2020-02-07 22:09:54 +0100 | [diff] [blame] | 546 | def Test_disassemble_and_or() |
Bram Moolenaar | 777770f | 2020-02-06 21:27:08 +0100 | [diff] [blame] | 547 | assert_equal("yes", AndOr(1)) |
| 548 | assert_equal("no", AndOr(2)) |
| 549 | assert_equal("yes", AndOr(4)) |
| 550 | let instr = execute('disassemble AndOr') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 551 | assert_match('AndOr.*' .. |
| 552 | 'if arg == 1 && arg != 2 || arg == 4.*' .. |
| 553 | '\d LOAD arg\[-1].*' .. |
| 554 | '\d PUSHNR 1.*' .. |
| 555 | '\d COMPAREANY ==.*' .. |
| 556 | '\d JUMP_AND_KEEP_IF_FALSE -> \d\+.*' .. |
| 557 | '\d LOAD arg\[-1].*' .. |
| 558 | '\d PUSHNR 2.*' .. |
| 559 | '\d COMPAREANY !=.*' .. |
| 560 | '\d JUMP_AND_KEEP_IF_TRUE -> \d\+.*' .. |
| 561 | '\d LOAD arg\[-1].*' .. |
| 562 | '\d PUSHNR 4.*' .. |
| 563 | '\d COMPAREANY ==.*' .. |
| 564 | '\d JUMP_IF_FALSE -> \d\+.*', |
| 565 | instr) |
Bram Moolenaar | 777770f | 2020-02-06 21:27:08 +0100 | [diff] [blame] | 566 | enddef |
| 567 | |
Bram Moolenaar | 04d0522 | 2020-02-06 22:06:54 +0100 | [diff] [blame] | 568 | def ForLoop(): list<number> |
| 569 | let res: list<number> |
| 570 | for i in range(3) |
| 571 | res->add(i) |
| 572 | endfor |
| 573 | return res |
| 574 | enddef |
| 575 | |
Bram Moolenaar | f2460a3 | 2020-02-07 22:09:54 +0100 | [diff] [blame] | 576 | def Test_disassemble_for_loop() |
Bram Moolenaar | 04d0522 | 2020-02-06 22:06:54 +0100 | [diff] [blame] | 577 | assert_equal([0, 1, 2], ForLoop()) |
| 578 | let instr = execute('disassemble ForLoop') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 579 | assert_match('ForLoop.*' .. |
| 580 | 'let res: list<number>.*' .. |
| 581 | ' NEWLIST size 0.*' .. |
| 582 | '\d STORE $0.*' .. |
| 583 | 'for i in range(3).*' .. |
| 584 | '\d STORE -1 in $1.*' .. |
| 585 | '\d PUSHNR 3.*' .. |
| 586 | '\d BCALL range(argc 1).*' .. |
| 587 | '\d FOR $1 -> \d\+.*' .. |
| 588 | '\d STORE $2.*' .. |
| 589 | 'res->add(i).*' .. |
| 590 | '\d LOAD $0.*' .. |
| 591 | '\d LOAD $2.*' .. |
| 592 | '\d BCALL add(argc 2).*' .. |
| 593 | '\d DROP.*' .. |
| 594 | 'endfor.*' .. |
| 595 | '\d JUMP -> \d\+.*' .. |
| 596 | '\d DROP.*', |
| 597 | instr) |
Bram Moolenaar | 04d0522 | 2020-02-06 22:06:54 +0100 | [diff] [blame] | 598 | enddef |
| 599 | |
Bram Moolenaar | c2a4b35 | 2020-02-06 22:41:16 +0100 | [diff] [blame] | 600 | let g:number = 42 |
| 601 | |
| 602 | def Computing() |
| 603 | let nr = 3 |
| 604 | let nrres = nr + 7 |
| 605 | nrres = nr - 7 |
| 606 | nrres = nr * 7 |
| 607 | nrres = nr / 7 |
| 608 | nrres = nr % 7 |
| 609 | |
| 610 | let anyres = g:number + 7 |
| 611 | anyres = g:number - 7 |
| 612 | anyres = g:number * 7 |
| 613 | anyres = g:number / 7 |
| 614 | anyres = g:number % 7 |
| 615 | |
| 616 | if has('float') |
| 617 | let fl = 3.0 |
| 618 | let flres = fl + 7.0 |
| 619 | flres = fl - 7.0 |
| 620 | flres = fl * 7.0 |
| 621 | flres = fl / 7.0 |
| 622 | endif |
| 623 | enddef |
| 624 | |
Bram Moolenaar | f2460a3 | 2020-02-07 22:09:54 +0100 | [diff] [blame] | 625 | def Test_disassemble_computing() |
Bram Moolenaar | c2a4b35 | 2020-02-06 22:41:16 +0100 | [diff] [blame] | 626 | let instr = execute('disassemble Computing') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 627 | assert_match('Computing.*' .. |
| 628 | 'let nr = 3.*' .. |
| 629 | '\d STORE 3 in $0.*' .. |
| 630 | 'let nrres = nr + 7.*' .. |
| 631 | '\d LOAD $0.*' .. |
| 632 | '\d PUSHNR 7.*' .. |
| 633 | '\d OPNR +.*' .. |
| 634 | '\d STORE $1.*' .. |
| 635 | 'nrres = nr - 7.*' .. |
| 636 | '\d OPNR -.*' .. |
| 637 | 'nrres = nr \* 7.*' .. |
| 638 | '\d OPNR \*.*' .. |
| 639 | 'nrres = nr / 7.*' .. |
| 640 | '\d OPNR /.*' .. |
| 641 | 'nrres = nr % 7.*' .. |
| 642 | '\d OPNR %.*' .. |
| 643 | 'let anyres = g:number + 7.*' .. |
| 644 | '\d LOADG g:number.*' .. |
| 645 | '\d PUSHNR 7.*' .. |
| 646 | '\d OPANY +.*' .. |
| 647 | '\d STORE $2.*' .. |
| 648 | 'anyres = g:number - 7.*' .. |
| 649 | '\d OPANY -.*' .. |
| 650 | 'anyres = g:number \* 7.*' .. |
| 651 | '\d OPANY \*.*' .. |
| 652 | 'anyres = g:number / 7.*' .. |
| 653 | '\d OPANY /.*' .. |
| 654 | 'anyres = g:number % 7.*' .. |
| 655 | '\d OPANY %.*', |
| 656 | instr) |
Bram Moolenaar | c2a4b35 | 2020-02-06 22:41:16 +0100 | [diff] [blame] | 657 | if has('float') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 658 | assert_match('Computing.*' .. |
| 659 | 'let fl = 3.0.*' .. |
| 660 | '\d PUSHF 3.0.*' .. |
| 661 | '\d STORE $3.*' .. |
| 662 | 'let flres = fl + 7.0.*' .. |
| 663 | '\d LOAD $3.*' .. |
| 664 | '\d PUSHF 7.0.*' .. |
| 665 | '\d OPFLOAT +.*' .. |
| 666 | '\d STORE $4.*' .. |
| 667 | 'flres = fl - 7.0.*' .. |
| 668 | '\d OPFLOAT -.*' .. |
| 669 | 'flres = fl \* 7.0.*' .. |
| 670 | '\d OPFLOAT \*.*' .. |
| 671 | 'flres = fl / 7.0.*' .. |
| 672 | '\d OPFLOAT /.*', |
| 673 | instr) |
Bram Moolenaar | c2a4b35 | 2020-02-06 22:41:16 +0100 | [diff] [blame] | 674 | endif |
| 675 | enddef |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 676 | |
Bram Moolenaar | ee2e52a | 2020-02-19 14:17:18 +0100 | [diff] [blame] | 677 | def AddListBlob() |
| 678 | let reslist = [1, 2] + [3, 4] |
| 679 | let resblob = 0z1122 + 0z3344 |
| 680 | enddef |
| 681 | |
| 682 | def Test_disassemble_add_list_blob() |
| 683 | let instr = execute('disassemble AddListBlob') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 684 | assert_match('AddListBlob.*' .. |
| 685 | 'let reslist = \[1, 2] + \[3, 4].*' .. |
| 686 | '\d PUSHNR 1.*' .. |
| 687 | '\d PUSHNR 2.*' .. |
| 688 | '\d NEWLIST size 2.*' .. |
| 689 | '\d PUSHNR 3.*' .. |
| 690 | '\d PUSHNR 4.*' .. |
| 691 | '\d NEWLIST size 2.*' .. |
| 692 | '\d ADDLIST.*' .. |
| 693 | '\d STORE $.*.*' .. |
| 694 | 'let resblob = 0z1122 + 0z3344.*' .. |
| 695 | '\d PUSHBLOB 0z1122.*' .. |
| 696 | '\d PUSHBLOB 0z3344.*' .. |
| 697 | '\d ADDBLOB.*' .. |
| 698 | '\d STORE $.*', |
| 699 | instr) |
Bram Moolenaar | ee2e52a | 2020-02-19 14:17:18 +0100 | [diff] [blame] | 700 | enddef |
| 701 | |
| 702 | let g:aa = 'aa' |
| 703 | def ConcatString(): string |
| 704 | let res = g:aa .. "bb" |
| 705 | return res |
| 706 | enddef |
| 707 | |
| 708 | def Test_disassemble_concat() |
| 709 | let instr = execute('disassemble ConcatString') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 710 | assert_match('ConcatString.*' .. |
| 711 | 'let res = g:aa .. "bb".*' .. |
| 712 | '\d LOADG g:aa.*' .. |
| 713 | '\d PUSHS "bb".*' .. |
| 714 | '\d 2STRING stack\[-2].*' .. |
| 715 | '\d CONCAT.*' .. |
| 716 | '\d STORE $.*', |
| 717 | instr) |
Bram Moolenaar | ee2e52a | 2020-02-19 14:17:18 +0100 | [diff] [blame] | 718 | assert_equal('aabb', ConcatString()) |
| 719 | enddef |
| 720 | |
| 721 | def ListIndex(): number |
| 722 | let l = [1, 2, 3] |
| 723 | let res = l[1] |
| 724 | return res |
| 725 | enddef |
| 726 | |
| 727 | def Test_disassemble_list_index() |
| 728 | let instr = execute('disassemble ListIndex') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 729 | assert_match('ListIndex.*' .. |
| 730 | 'let l = \[1, 2, 3].*' .. |
| 731 | '\d PUSHNR 1.*' .. |
| 732 | '\d PUSHNR 2.*' .. |
| 733 | '\d PUSHNR 3.*' .. |
| 734 | '\d NEWLIST size 3.*' .. |
| 735 | '\d STORE $0.*' .. |
| 736 | 'let res = l\[1].*' .. |
| 737 | '\d LOAD $0.*' .. |
| 738 | '\d PUSHNR 1.*' .. |
| 739 | '\d INDEX.*' .. |
| 740 | '\d STORE $1.*', |
| 741 | instr) |
Bram Moolenaar | ee2e52a | 2020-02-19 14:17:18 +0100 | [diff] [blame] | 742 | assert_equal(2, ListIndex()) |
| 743 | enddef |
| 744 | |
| 745 | def DictMember(): number |
| 746 | let d = #{item: 1} |
| 747 | let res = d.item |
| 748 | return res |
| 749 | enddef |
| 750 | |
| 751 | def Test_disassemble_dict_member() |
| 752 | let instr = execute('disassemble DictMember') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 753 | assert_match('DictMember.*' .. |
| 754 | 'let d = #{item: 1}.*' .. |
| 755 | '\d PUSHS "item".*' .. |
| 756 | '\d PUSHNR 1.*' .. |
| 757 | '\d NEWDICT size 1.*' .. |
| 758 | '\d STORE $0.*' .. |
| 759 | 'let res = d.item.*' .. |
| 760 | '\d LOAD $0.*' .. |
| 761 | '\d MEMBER item.*' .. |
| 762 | '\d STORE $1.*', |
| 763 | instr) |
Bram Moolenaar | ee2e52a | 2020-02-19 14:17:18 +0100 | [diff] [blame] | 764 | call assert_equal(1, DictMember()) |
| 765 | enddef |
| 766 | |
| 767 | def NegateNumber(): number |
| 768 | let nr = 9 |
| 769 | let plus = +nr |
| 770 | let res = -nr |
| 771 | return res |
| 772 | enddef |
| 773 | |
| 774 | def Test_disassemble_negate_number() |
| 775 | let instr = execute('disassemble NegateNumber') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 776 | assert_match('NegateNumber.*' .. |
| 777 | 'let nr = 9.*' .. |
| 778 | '\d STORE 9 in $0.*' .. |
| 779 | 'let plus = +nr.*' .. |
| 780 | '\d LOAD $0.*' .. |
| 781 | '\d CHECKNR.*' .. |
| 782 | '\d STORE $1.*' .. |
| 783 | 'let res = -nr.*' .. |
| 784 | '\d LOAD $0.*' .. |
| 785 | '\d NEGATENR.*' .. |
| 786 | '\d STORE $2.*', |
| 787 | instr) |
Bram Moolenaar | ee2e52a | 2020-02-19 14:17:18 +0100 | [diff] [blame] | 788 | call assert_equal(-9, NegateNumber()) |
| 789 | enddef |
| 790 | |
| 791 | def InvertBool(): bool |
| 792 | let flag = true |
| 793 | let invert = !flag |
| 794 | let res = !!flag |
| 795 | return res |
| 796 | enddef |
| 797 | |
| 798 | def Test_disassemble_invert_bool() |
| 799 | let instr = execute('disassemble InvertBool') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 800 | assert_match('InvertBool.*' .. |
| 801 | 'let flag = true.*' .. |
| 802 | '\d PUSH v:true.*' .. |
| 803 | '\d STORE $0.*' .. |
| 804 | 'let invert = !flag.*' .. |
| 805 | '\d LOAD $0.*' .. |
| 806 | '\d INVERT (!val).*' .. |
| 807 | '\d STORE $1.*' .. |
| 808 | 'let res = !!flag.*' .. |
| 809 | '\d LOAD $0.*' .. |
| 810 | '\d 2BOOL (!!val).*' .. |
| 811 | '\d STORE $2.*', |
| 812 | instr) |
Bram Moolenaar | ee2e52a | 2020-02-19 14:17:18 +0100 | [diff] [blame] | 813 | call assert_equal(true, InvertBool()) |
| 814 | enddef |
| 815 | |
Bram Moolenaar | f2460a3 | 2020-02-07 22:09:54 +0100 | [diff] [blame] | 816 | def Test_disassemble_compare() |
| 817 | " TODO: COMPAREFUNC |
| 818 | let cases = [ |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 819 | ['true == false', 'COMPAREBOOL =='], |
| 820 | ['true != false', 'COMPAREBOOL !='], |
| 821 | ['v:none == v:null', 'COMPARESPECIAL =='], |
| 822 | ['v:none != v:null', 'COMPARESPECIAL !='], |
| 823 | |
| 824 | ['111 == 222', 'COMPARENR =='], |
| 825 | ['111 != 222', 'COMPARENR !='], |
| 826 | ['111 > 222', 'COMPARENR >'], |
| 827 | ['111 < 222', 'COMPARENR <'], |
| 828 | ['111 >= 222', 'COMPARENR >='], |
| 829 | ['111 <= 222', 'COMPARENR <='], |
| 830 | ['111 =~ 222', 'COMPARENR =\~'], |
| 831 | ['111 !~ 222', 'COMPARENR !\~'], |
| 832 | |
| 833 | ['"xx" != "yy"', 'COMPARESTRING !='], |
| 834 | ['"xx" > "yy"', 'COMPARESTRING >'], |
| 835 | ['"xx" < "yy"', 'COMPARESTRING <'], |
| 836 | ['"xx" >= "yy"', 'COMPARESTRING >='], |
| 837 | ['"xx" <= "yy"', 'COMPARESTRING <='], |
| 838 | ['"xx" =~ "yy"', 'COMPARESTRING =\~'], |
| 839 | ['"xx" !~ "yy"', 'COMPARESTRING !\~'], |
| 840 | ['"xx" is "yy"', 'COMPARESTRING is'], |
| 841 | ['"xx" isnot "yy"', 'COMPARESTRING isnot'], |
| 842 | |
| 843 | ['0z11 == 0z22', 'COMPAREBLOB =='], |
| 844 | ['0z11 != 0z22', 'COMPAREBLOB !='], |
| 845 | ['0z11 is 0z22', 'COMPAREBLOB is'], |
| 846 | ['0z11 isnot 0z22', 'COMPAREBLOB isnot'], |
| 847 | |
| 848 | ['[1,2] == [3,4]', 'COMPARELIST =='], |
| 849 | ['[1,2] != [3,4]', 'COMPARELIST !='], |
| 850 | ['[1,2] is [3,4]', 'COMPARELIST is'], |
| 851 | ['[1,2] isnot [3,4]', 'COMPARELIST isnot'], |
| 852 | |
| 853 | ['#{a:1} == #{x:2}', 'COMPAREDICT =='], |
| 854 | ['#{a:1} != #{x:2}', 'COMPAREDICT !='], |
| 855 | ['#{a:1} is #{x:2}', 'COMPAREDICT is'], |
| 856 | ['#{a:1} isnot #{x:2}', 'COMPAREDICT isnot'], |
| 857 | |
| 858 | ['{->33} == {->44}', 'COMPAREFUNC =='], |
| 859 | ['{->33} != {->44}', 'COMPAREFUNC !='], |
| 860 | ['{->33} is {->44}', 'COMPAREFUNC is'], |
| 861 | ['{->33} isnot {->44}', 'COMPAREFUNC isnot'], |
| 862 | |
| 863 | ['77 == g:xx', 'COMPAREANY =='], |
| 864 | ['77 != g:xx', 'COMPAREANY !='], |
| 865 | ['77 > g:xx', 'COMPAREANY >'], |
| 866 | ['77 < g:xx', 'COMPAREANY <'], |
| 867 | ['77 >= g:xx', 'COMPAREANY >='], |
| 868 | ['77 <= g:xx', 'COMPAREANY <='], |
| 869 | ['77 =~ g:xx', 'COMPAREANY =\~'], |
| 870 | ['77 !~ g:xx', 'COMPAREANY !\~'], |
| 871 | ['77 is g:xx', 'COMPAREANY is'], |
| 872 | ['77 isnot g:xx', 'COMPAREANY isnot'], |
| 873 | ] |
Bram Moolenaar | f2460a3 | 2020-02-07 22:09:54 +0100 | [diff] [blame] | 874 | if has('float') |
| 875 | cases->extend([ |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 876 | ['1.1 == 2.2', 'COMPAREFLOAT =='], |
| 877 | ['1.1 != 2.2', 'COMPAREFLOAT !='], |
| 878 | ['1.1 > 2.2', 'COMPAREFLOAT >'], |
| 879 | ['1.1 < 2.2', 'COMPAREFLOAT <'], |
| 880 | ['1.1 >= 2.2', 'COMPAREFLOAT >='], |
| 881 | ['1.1 <= 2.2', 'COMPAREFLOAT <='], |
| 882 | ['1.1 =~ 2.2', 'COMPAREFLOAT =\~'], |
| 883 | ['1.1 !~ 2.2', 'COMPAREFLOAT !\~'], |
| 884 | ]) |
Bram Moolenaar | f2460a3 | 2020-02-07 22:09:54 +0100 | [diff] [blame] | 885 | endif |
| 886 | |
| 887 | let nr = 1 |
| 888 | for case in cases |
| 889 | writefile(['def TestCase' .. nr .. '()', |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 890 | ' if ' .. case[0], |
| 891 | ' echo 42' |
| 892 | ' endif', |
| 893 | 'enddef'], 'Xdisassemble') |
Bram Moolenaar | f2460a3 | 2020-02-07 22:09:54 +0100 | [diff] [blame] | 894 | source Xdisassemble |
| 895 | let instr = execute('disassemble TestCase' .. nr) |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 896 | assert_match('TestCase' .. nr .. '.*' .. |
| 897 | 'if ' .. substitute(case[0], '[[~]', '\\\0', 'g') .. '.*' .. |
| 898 | '\d \(PUSH\|FUNCREF\).*' .. |
| 899 | '\d \(PUSH\|FUNCREF\|LOADG\).*' .. |
| 900 | '\d ' .. case[1] .. '.*' .. |
| 901 | '\d JUMP_IF_FALSE -> \d\+.*', |
| 902 | instr) |
Bram Moolenaar | f2460a3 | 2020-02-07 22:09:54 +0100 | [diff] [blame] | 903 | |
| 904 | nr += 1 |
| 905 | endfor |
| 906 | |
Bram Moolenaar | 22da559 | 2020-03-19 14:52:20 +0100 | [diff] [blame] | 907 | delete('Xdisassemble') |
Bram Moolenaar | f2460a3 | 2020-02-07 22:09:54 +0100 | [diff] [blame] | 908 | enddef |
| 909 | |
Bram Moolenaar | a4d4cf4 | 2020-04-02 13:50:27 +0200 | [diff] [blame] | 910 | def Test_disassemble_compare_const() |
| 911 | let cases = [ |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 912 | ['"xx" == "yy"', false], |
| 913 | ['"aa" == "aa"', true], |
| 914 | ['has("eval") ? true : false', true], |
| 915 | ['has("asdf") ? true : false', false], |
| 916 | ] |
Bram Moolenaar | a4d4cf4 | 2020-04-02 13:50:27 +0200 | [diff] [blame] | 917 | |
| 918 | let nr = 1 |
| 919 | for case in cases |
| 920 | writefile(['def TestCase' .. nr .. '()', |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 921 | ' if ' .. case[0], |
| 922 | ' echo 42' |
| 923 | ' endif', |
| 924 | 'enddef'], 'Xdisassemble') |
Bram Moolenaar | a4d4cf4 | 2020-04-02 13:50:27 +0200 | [diff] [blame] | 925 | source Xdisassemble |
| 926 | let instr = execute('disassemble TestCase' .. nr) |
| 927 | if case[1] |
| 928 | " condition true, "echo 42" executed |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 929 | assert_match('TestCase' .. nr .. '.*' .. |
| 930 | 'if ' .. substitute(case[0], '[[~]', '\\\0', 'g') .. '.*' .. |
| 931 | '\d PUSHNR 42.*' .. |
| 932 | '\d ECHO 1.*' .. |
| 933 | '\d PUSHNR 0.*' .. |
| 934 | '\d RETURN.*', |
| 935 | instr) |
Bram Moolenaar | a4d4cf4 | 2020-04-02 13:50:27 +0200 | [diff] [blame] | 936 | else |
| 937 | " condition false, function just returns |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 938 | assert_match('TestCase' .. nr .. '.*' .. |
| 939 | 'if ' .. substitute(case[0], '[[~]', '\\\0', 'g') .. '[ \n]*' .. |
| 940 | 'echo 42[ \n]*' .. |
| 941 | 'endif[ \n]*' .. |
| 942 | '\s*\d PUSHNR 0.*' .. |
| 943 | '\d RETURN.*', |
| 944 | instr) |
Bram Moolenaar | a4d4cf4 | 2020-04-02 13:50:27 +0200 | [diff] [blame] | 945 | endif |
| 946 | |
| 947 | nr += 1 |
| 948 | endfor |
| 949 | |
| 950 | delete('Xdisassemble') |
| 951 | enddef |
| 952 | |
Bram Moolenaar | ad39c09 | 2020-02-26 18:23:43 +0100 | [diff] [blame] | 953 | def s:Execute() |
| 954 | execute 'help vim9.txt' |
| 955 | let cmd = 'help vim9.txt' |
| 956 | execute cmd |
| 957 | let tag = 'vim9.txt' |
| 958 | execute 'help ' .. tag |
| 959 | enddef |
| 960 | |
| 961 | def Test_disassemble_execute() |
| 962 | let res = execute('disass s:Execute') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 963 | assert_match('\<SNR>\d*_Execute.*' .. |
| 964 | "execute 'help vim9.txt'.*" .. |
| 965 | '\d PUSHS "help vim9.txt".*' .. |
| 966 | '\d EXECUTE 1.*' .. |
| 967 | "let cmd = 'help vim9.txt'.*" .. |
| 968 | '\d PUSHS "help vim9.txt".*' .. |
| 969 | '\d STORE $0.*' .. |
| 970 | 'execute cmd.*' .. |
| 971 | '\d LOAD $0.*' .. |
| 972 | '\d EXECUTE 1.*' .. |
| 973 | "let tag = 'vim9.txt'.*" .. |
| 974 | '\d PUSHS "vim9.txt".*' .. |
| 975 | '\d STORE $1.*' .. |
| 976 | "execute 'help ' .. tag.*" .. |
| 977 | '\d PUSHS "help ".*' .. |
| 978 | '\d LOAD $1.*' .. |
| 979 | '\d CONCAT.*' .. |
| 980 | '\d EXECUTE 1.*' .. |
| 981 | '\d PUSHNR 0.*' .. |
| 982 | '\d RETURN', |
| 983 | res) |
Bram Moolenaar | ad39c09 | 2020-02-26 18:23:43 +0100 | [diff] [blame] | 984 | enddef |
| 985 | |
Bram Moolenaar | f93c7fe | 2020-04-23 22:16:53 +0200 | [diff] [blame] | 986 | def s:Echomsg() |
| 987 | echomsg 'some' 'message' |
| 988 | echoerr 'went' .. 'wrong' |
| 989 | enddef |
| 990 | |
| 991 | def Test_disassemble_echomsg() |
| 992 | let res = execute('disass s:Echomsg') |
| 993 | assert_match('\<SNR>\d*_Echomsg.*' .. |
| 994 | "echomsg 'some' 'message'.*" .. |
| 995 | '\d PUSHS "some".*' .. |
| 996 | '\d PUSHS "message".*' .. |
| 997 | '\d ECHOMSG 2.*' .. |
| 998 | "echoerr 'went' .. 'wrong'.*" .. |
Bram Moolenaar | 61a8981 | 2020-05-07 16:58:17 +0200 | [diff] [blame] | 999 | '\d PUSHS "wentwrong".*' .. |
Bram Moolenaar | f93c7fe | 2020-04-23 22:16:53 +0200 | [diff] [blame] | 1000 | '\d ECHOERR 1.*' .. |
| 1001 | '\d PUSHNR 0.*' .. |
| 1002 | '\d RETURN', |
| 1003 | res) |
| 1004 | enddef |
| 1005 | |
Bram Moolenaar | 61a6d4e | 2020-03-01 23:32:25 +0100 | [diff] [blame] | 1006 | def SomeStringArg(arg: string) |
| 1007 | echo arg |
| 1008 | enddef |
| 1009 | |
| 1010 | def SomeAnyArg(arg: any) |
| 1011 | echo arg |
| 1012 | enddef |
| 1013 | |
| 1014 | def SomeStringArgAndReturn(arg: string): string |
| 1015 | return arg |
| 1016 | enddef |
| 1017 | |
| 1018 | def Test_display_func() |
| 1019 | let res1 = execute('function SomeStringArg') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 1020 | assert_match('.* def SomeStringArg(arg: string).*' .. |
| 1021 | ' echo arg.*' .. |
| 1022 | ' enddef', |
| 1023 | res1) |
Bram Moolenaar | 61a6d4e | 2020-03-01 23:32:25 +0100 | [diff] [blame] | 1024 | |
| 1025 | let res2 = execute('function SomeAnyArg') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 1026 | assert_match('.* def SomeAnyArg(arg: any).*' .. |
| 1027 | ' echo arg.*' .. |
| 1028 | ' enddef', |
| 1029 | res2) |
Bram Moolenaar | 61a6d4e | 2020-03-01 23:32:25 +0100 | [diff] [blame] | 1030 | |
| 1031 | let res3 = execute('function SomeStringArgAndReturn') |
Bram Moolenaar | 675f716 | 2020-04-12 22:53:54 +0200 | [diff] [blame] | 1032 | assert_match('.* def SomeStringArgAndReturn(arg: string): string.*' .. |
| 1033 | ' return arg.*' .. |
| 1034 | ' enddef', |
| 1035 | res3) |
Bram Moolenaar | 61a6d4e | 2020-03-01 23:32:25 +0100 | [diff] [blame] | 1036 | enddef |
| 1037 | |
Bram Moolenaar | 61a8981 | 2020-05-07 16:58:17 +0200 | [diff] [blame] | 1038 | def s:ConcatStrings(): string |
| 1039 | return 'one' .. 'two' .. 'three' |
| 1040 | enddef |
| 1041 | |
Bram Moolenaar | 7d131b0 | 2020-05-08 19:10:34 +0200 | [diff] [blame] | 1042 | def s:ComputeConst(): number |
| 1043 | return 2 + 3 * 4 / 6 + 7 |
| 1044 | enddef |
| 1045 | |
Bram Moolenaar | 61a8981 | 2020-05-07 16:58:17 +0200 | [diff] [blame] | 1046 | def Test_simplify_const_expr() |
| 1047 | let res = execute('disass s:ConcatStrings') |
| 1048 | assert_match('\<SNR>\d*_ConcatStrings.*' .. |
| 1049 | '\d PUSHS "onetwothree".*' .. |
| 1050 | '\d RETURN', |
| 1051 | res) |
Bram Moolenaar | 7d131b0 | 2020-05-08 19:10:34 +0200 | [diff] [blame] | 1052 | |
| 1053 | res = execute('disass s:ComputeConst') |
| 1054 | assert_match('\<SNR>\d*_ComputeConst.*' .. |
| 1055 | '\d PUSHNR 11.*' .. |
| 1056 | '\d RETURN', |
| 1057 | res) |
Bram Moolenaar | 61a8981 | 2020-05-07 16:58:17 +0200 | [diff] [blame] | 1058 | enddef |
| 1059 | |
Bram Moolenaar | 5cab73f | 2020-02-06 19:25:19 +0100 | [diff] [blame] | 1060 | " vim: ts=8 sw=2 sts=2 expandtab tw=80 fdm=marker |