]> granicus.if.org Git - vim/commitdiff
patch 8.2.2765: Vim9: not all blob operations work v8.2.2765
authorBram Moolenaar <Bram@vim.org>
Wed, 14 Apr 2021 18:35:23 +0000 (20:35 +0200)
committerBram Moolenaar <Bram@vim.org>
Wed, 14 Apr 2021 18:35:23 +0000 (20:35 +0200)
Problem:    Vim9: not all blob operations work.
Solution:   Run more tests also with Vim9 script and :def functions.  Fix what
            doesn't work.

src/blob.c
src/errors.h
src/eval.c
src/proto/blob.pro
src/testdir/test_blob.vim
src/testdir/vim9.vim
src/version.c
src/vim9execute.c

index d758beb19db2a7de875d2b83b527dc6361ff0fcf..f020966ae260555e625fc4f8a536b53c5e794d2d 100644 (file)
@@ -336,6 +336,36 @@ blob_slice_or_index(
     return OK;
 }
 
+/*
+ * Check if "n1"- is a valid index for a blobl with length "bloblen".
+ */
+    int
+check_blob_index(long bloblen, varnumber_T n1, int is_range, int quiet)
+{
+    if (n1 < 0 || n1 > bloblen)
+    {
+       if (!quiet)
+           semsg(_(e_blobidx), n1);
+       return FAIL;
+    }
+    return OK;
+}
+
+/*
+ * Check if "n1"-"n2" is a valid range for a blob with length "bloblen".
+ */
+    int
+check_blob_range(long bloblen, varnumber_T n1, varnumber_T n2, int quiet)
+{
+    if (n2 < 0 || n2 >= bloblen || n2 < n1)
+    {
+       if (!quiet)
+           semsg(_(e_blobidx), n2);
+       return FAIL;
+    }
+    return OK;
+}
+
 /*
  * Set bytes "n1" to "n2" (inclusive) in "dest" to the value of "src".
  * Caller must make sure "src" is a blob.
index d38562cfab7e742871cdf0ee84e0b0271a0b7553..99f3aaff5a551d208784872ee9297105adec2218 100644 (file)
@@ -401,3 +401,5 @@ EXTERN char e_cannot_use_underscore_here[]
        INIT(= N_("E1181: Cannot use an underscore here"));
 EXTERN char e_blob_required[]
        INIT(= N_("E1182: Blob required"));
+EXTERN char e_cannot_use_range_with_assignment_operator_str[]
+       INIT(= N_("E1183: Cannot use a range with an assignment operator: %s"));
index 99768ee280cda0c5a7ddb6be1e23ceca9d8b537d..d8607f2be6ae4938c6044ec62a0b8bdd3a4bc7eb 100644 (file)
@@ -1175,12 +1175,9 @@ get_lval(
                lp->ll_n1 = (long)tv_get_number(&var1);
            clear_tv(&var1);
 
-           if (lp->ll_n1 < 0
-                   || lp->ll_n1 > bloblen
-                   || (lp->ll_range && lp->ll_n1 == bloblen))
+           if (check_blob_index(bloblen, lp->ll_n1, lp->ll_range, quiet)
+                                                                      == FAIL)
            {
-               if (!quiet)
-                   semsg(_(e_blobidx), lp->ll_n1);
                clear_tv(&var2);
                return NULL;
            }
@@ -1188,14 +1185,9 @@ get_lval(
            {
                lp->ll_n2 = (long)tv_get_number(&var2);
                clear_tv(&var2);
-               if (lp->ll_n2 < 0
-                       || lp->ll_n2 >= bloblen
-                       || lp->ll_n2 < lp->ll_n1)
-               {
-                   if (!quiet)
-                       semsg(_(e_blobidx), lp->ll_n2);
+               if (check_blob_range(bloblen, lp->ll_n1, lp->ll_n2, quiet)
+                                                                      == FAIL)
                    return NULL;
-               }
            }
            lp->ll_blob = lp->ll_tv->vval.v_blob;
            lp->ll_tv = NULL;
index 7da269c35f043072fafa477b1be7288c97026b50..0b7cbd3a86fc1238edb52dfcec5cd5ce80edd588 100644 (file)
@@ -14,6 +14,8 @@ int write_blob(FILE *fd, blob_T *blob);
 char_u *blob2string(blob_T *blob, char_u **tofree, char_u *numbuf);
 blob_T *string2blob(char_u *str);
 int blob_slice_or_index(blob_T *blob, int is_range, varnumber_T n1, varnumber_T n2, int exclusive, typval_T *rettv);
+int check_blob_index(long bloblen, varnumber_T n1, int is_range, int quiet);
+int check_blob_range(long bloblen, varnumber_T n1, varnumber_T n2, int quiet);
 int blob_set_range(blob_T *dest, long n1, long n2, typval_T *src);
 void blob_remove(typval_T *argvars, typval_T *rettv);
 /* vim: set ft=c : */
index 34ed0caf2fc771873198a6a8c9c76d6aa49917bd..3699f3bb194abf2269e0396d3849009b9366a89d 100644 (file)
@@ -76,16 +76,47 @@ func Test_blob_assign()
   END
   call CheckLegacyAndVim9Success(lines)
 
-  " TODO: move to above once it works
-  let b = 0zDEADBEEF
-  call assert_fails('let b[2 : 3] = 0z112233', 'E972:')
-  call assert_fails('let b[2 : 3] = 0z11', 'E972:')
-  call assert_fails('let b[3 : 2] = 0z', 'E979:')
-
-  call assert_fails('let b ..= 0z33', 'E734:')
-  call assert_fails('let b ..= "xx"', 'E734:')
-  call assert_fails('let b += "xx"', 'E734:')
-  call assert_fails('let b[1 : 1] ..= 0z55', 'E734:')
+  let lines =<< trim END
+      VAR b = 0zDEADBEEF
+      LET b[2 : 3] = 0z112233
+  END
+  call CheckLegacyAndVim9Failure(lines, 'E972:')
+
+  let lines =<< trim END
+      VAR b = 0zDEADBEEF
+      LET b[2 : 3] = 0z11
+  END
+  call CheckLegacyAndVim9Failure(lines, 'E972:')
+
+  let lines =<< trim END
+      VAR b = 0zDEADBEEF
+      LET b[3 : 2] = 0z
+  END
+  call CheckLegacyAndVim9Failure(lines, 'E979:')
+
+  let lines =<< trim END
+      VAR b = 0zDEADBEEF
+      LET b ..= 0z33
+  END
+  call CheckLegacyAndVim9Failure(lines, ['E734:', 'E1019:', 'E734:'])
+
+  let lines =<< trim END
+      VAR b = 0zDEADBEEF
+      LET b ..= "xx"
+  END
+  call CheckLegacyAndVim9Failure(lines, ['E734:', 'E1019:', 'E734:'])
+
+  let lines =<< trim END
+      VAR b = 0zDEADBEEF
+      LET b += "xx"
+  END
+  call CheckLegacyAndVim9Failure(lines, ['E734:', 'E1012:', 'E734:'])
+
+  let lines =<< trim END
+      VAR b = 0zDEADBEEF
+      LET b[1 : 1] ..= 0z55
+  END
+  call CheckLegacyAndVim9Failure(lines, ['E734:', 'E1183:', 'E734:'])
 endfunc
 
 func Test_blob_get_range()
index e05f8590bbbe85940d3c7ffa79bb1676e8e23f06..9efa65552ee64dcb69a6fce2b78c7cba2bc718d9 100644 (file)
@@ -144,8 +144,23 @@ func CheckLegacySuccess(lines)
   try
     exe 'so ' .. fname
     call Func()
+  finally
     delfunc! Func
+    call chdir(cwd)
+    call delete(fname)
+  endtry
+endfunc
+
+" Check that "lines" inside a legacy function results in the expected error
+func CheckLegacyFailure(lines, error)
+  let cwd = getcwd()
+  let fname = 'XlegacyFails' .. s:sequence
+  let s:sequence += 1
+  call writefile(['func Func()'] + a:lines + ['endfunc', 'call Func()'], fname)
+  try
+    call assert_fails('so ' .. fname, a:error)
   finally
+    delfunc! Func
     call chdir(cwd)
     call delete(fname)
   endtry
@@ -168,3 +183,35 @@ def CheckLegacyAndVim9Success(lines: list<string>)
   CheckDefSuccess(vim9lines)
   CheckScriptSuccess(['vim9script'] + vim9lines)
 enddef
+
+" Execute "lines" in a legacy function, :def function and Vim9 script.
+" Use 'VAR' for a declaration.
+" Use 'LET' for an assignment
+" Use ' #"' for a comment
+def CheckLegacyAndVim9Failure(lines: list<string>, error: any)
+  var legacyError: string
+  var defError: string
+  var scriptError: string
+
+  if type(error) == type('string')
+    legacyError = error
+    defError = error
+    scriptError = error
+  else
+    legacyError = error[0]
+    defError = error[1]
+    scriptError = error[2]
+  endif
+
+  var legacylines = lines->mapnew((_, v) =>
+                               v->substitute('\<VAR\>', 'let', 'g')
+                                ->substitute('\<LET\>', 'let', 'g')
+                                ->substitute('#"', ' "', 'g'))
+  CheckLegacyFailure(legacylines, legacyError)
+
+  var vim9lines = lines->mapnew((_, v) =>
+                               v->substitute('\<VAR\>', 'var', 'g')
+                                ->substitute('\<LET ', '', 'g'))
+  CheckDefExecFailure(vim9lines, defError)
+  CheckScriptFailure(['vim9script'] + vim9lines, scriptError)
+enddef
index 6ad07f7d76fb6028d406adbe55ea2eb9ade67f46..0127c2c51c52140a545e31555059bef8e9933345 100644 (file)
@@ -750,6 +750,8 @@ static char *(features[]) =
 
 static int included_patches[] =
 {   /* Add new patch number below this line */
+/**/
+    2765,
 /**/
     2764,
 /**/
index 026a9ee45d88a0adeb6f727e3f86c219d35bc533..bdf4a02b7195e83a730e15a660b1b02cb81e1f3a 100644 (file)
@@ -2278,8 +2278,18 @@ call_def_function(
                            if (error)
                                status = FAIL;
                            else
-                               status = blob_set_range(tv_dest->vval.v_blob,
-                                                                  n1, n2, tv);
+                           {
+                               long    bloblen = blob_len(tv_dest->vval.v_blob);
+
+                               if (check_blob_index(bloblen,
+                                                      n1, TRUE, FALSE) == FAIL
+                                       || check_blob_range(bloblen,
+                                                       n1, n2, FALSE) == FAIL)
+                                   status = FAIL;
+                               else
+                                   status = blob_set_range(
+                                            tv_dest->vval.v_blob, n1, n2, tv);
+                           }
                        }
                    }