0
0
mirror of https://github.com/vim/vim.git synced 2025-07-04 23:07:33 -04:00

patch 9.1.0198: Vim9: compound operators broken for lambdas in an object

Problem:  Vim9: compound operators broken for lambdas in an object
          (girishji)
Solution: When using an object from the outer scope, use the LOADOUTER
          instruction to load the object (Yegappan Lakshmanan).

fixes: #14236
closes: #14266

Signed-off-by: Yegappan Lakshmanan <yegappan@yahoo.com>
Signed-off-by: Christian Brabandt <cb@256bit.org>
This commit is contained in:
Yegappan Lakshmanan 2024-03-22 19:56:17 +01:00 committed by Christian Brabandt
parent 76d62985c1
commit d990bf08d8
No known key found for this signature in database
GPG Key ID: F3F92DA383FDDE09
4 changed files with 116 additions and 1 deletions

View File

@ -10349,4 +10349,75 @@ def Test_Ref_Class_Within_Same_Class()
v9.CheckScriptFailure(lines, 'E1347: Not a valid interface: A', 3) v9.CheckScriptFailure(lines, 'E1347: Not a valid interface: A', 3)
enddef enddef
" Test for using a compound operator from a lambda function in an object method
def Test_compound_op_in_objmethod_lambda()
# Test using the "+=" operator
var lines =<< trim END
vim9script
class A
var n: number = 10
def Foo()
var Fn = () => {
this.n += 1
}
Fn()
enddef
endclass
var a = A.new()
a.Foo()
assert_equal(11, a.n)
END
v9.CheckScriptSuccess(lines)
# Test using the "..=" operator
lines =<< trim END
vim9script
class A
var s: string = "a"
def Foo()
var Fn = () => {
this.s ..= "a"
}
Fn()
enddef
endclass
var a = A.new()
a.Foo()
a.Foo()
assert_equal("aaa", a.s)
END
v9.CheckScriptSuccess(lines)
enddef
" call a lambda function in one object from another object
def Test_lambda_invocation_across_classes()
var lines =<< trim END
vim9script
class A
var s: string = "foo"
def GetFn(): func
var Fn = (): string => {
return this.s
}
return Fn
enddef
endclass
class B
var s: string = "bar"
def GetFn(): func
var a = A.new()
return a.GetFn()
enddef
endclass
var b = B.new()
var Fn = b.GetFn()
assert_equal("foo", Fn())
END
v9.CheckScriptSuccess(lines)
enddef
" vim: ts=8 sw=2 sts=2 expandtab tw=80 fdm=marker " vim: ts=8 sw=2 sts=2 expandtab tw=80 fdm=marker

View File

@ -3436,4 +3436,36 @@ def Test_disassemble_object_len()
unlet g:instr unlet g:instr
enddef enddef
" Disassemble instructions for using a compound operator in a closure
def Test_disassemble_compound_op_in_closure()
var lines =<< trim END
vim9script
class A
var foo: number = 1
def Foo(): func
var Fn = () => {
this.foo += 1
}
return Fn
enddef
endclass
var a = A.new()
var Lambda = a.Foo()
var num = matchstr(string(Lambda), '\d\+')
g:instr = execute($'disassemble <lambda>{num}')
END
v9.CheckScriptSuccess(lines)
assert_match('<lambda>\d\+\_s*' ..
'this.foo += 1\_s*' ..
'0 LOADOUTER level 0 $0\_s*' ..
'1 OBJ_MEMBER 0\_s*' ..
'2 PUSHNR 1\_s*' ..
'3 OPNR +\_s*' ..
'4 PUSHNR 0\_s*' ..
'5 LOADOUTER level 0 $0\_s*' ..
'6 STOREINDEX object\_s*' ..
'7 RETURN void', g:instr)
unlet g:instr
enddef
" vim: ts=8 sw=2 sts=2 expandtab tw=80 fdm=marker " vim: ts=8 sw=2 sts=2 expandtab tw=80 fdm=marker

View File

@ -704,6 +704,8 @@ static char *(features[]) =
static int included_patches[] = static int included_patches[] =
{ /* Add new patch number below this line */ { /* Add new patch number below this line */
/**/
198,
/**/ /**/
197, 197,
/**/ /**/

View File

@ -2285,7 +2285,17 @@ compile_load_lhs_with_index(lhs_T *lhs, char_u *var_start, cctx_T *cctx)
if (dot - var_start == 4 && STRNCMP(var_start, "this", 4) == 0) if (dot - var_start == 4 && STRNCMP(var_start, "this", 4) == 0)
{ {
// load "this" // load "this"
if (generate_LOAD(cctx, ISN_LOAD, 0, NULL, lhs->lhs_type) == FAIL) lvar_T *lvar = lhs->lhs_lvar;
int rc;
if (lvar->lv_from_outer > 0)
rc = generate_LOADOUTER(cctx, lvar->lv_idx,
lvar->lv_from_outer, lvar->lv_loop_depth,
lvar->lv_loop_idx, type);
else
rc = generate_LOAD(cctx, ISN_LOAD, lvar->lv_idx, NULL, type);
if (rc == FAIL)
return FAIL; return FAIL;
} }
else else