mirror of
https://github.com/vim/vim.git
synced 2025-09-24 03:44:06 -04:00
patch 8.2.2777: Vim9: blob operations not tested in all ways
Problem: Vim9: blob operations not tested in all ways. Solution: Run tests with CheckLegacyAndVim9Success(). Make blob assign with index work.
This commit is contained in:
26
src/blob.c
26
src/blob.c
@@ -125,13 +125,33 @@ blob_get(blob_T *b, int idx)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Store one byte "c" in blob "b" at "idx".
|
* Store one byte "byte" in blob "blob" at "idx".
|
||||||
* Caller must make sure that "idx" is valid.
|
* Caller must make sure that "idx" is valid.
|
||||||
*/
|
*/
|
||||||
void
|
void
|
||||||
blob_set(blob_T *b, int idx, char_u c)
|
blob_set(blob_T *blob, int idx, int byte)
|
||||||
{
|
{
|
||||||
((char_u*)b->bv_ga.ga_data)[idx] = c;
|
((char_u*)blob->bv_ga.ga_data)[idx] = byte;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Store one byte "byte" in blob "blob" at "idx".
|
||||||
|
* Append one byte if needed.
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
blob_set_append(blob_T *blob, int idx, int byte)
|
||||||
|
{
|
||||||
|
garray_T *gap = &blob->bv_ga;
|
||||||
|
|
||||||
|
// Allow for appending a byte. Setting a byte beyond
|
||||||
|
// the end is an error otherwise.
|
||||||
|
if (idx < gap->ga_len
|
||||||
|
|| (idx == gap->ga_len && ga_grow(gap, 1) == OK))
|
||||||
|
{
|
||||||
|
blob_set(blob, idx, byte);
|
||||||
|
if (idx == gap->ga_len)
|
||||||
|
++gap->ga_len;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@@ -403,3 +403,5 @@ EXTERN char e_blob_required[]
|
|||||||
INIT(= N_("E1182: Blob required"));
|
INIT(= N_("E1182: Blob required"));
|
||||||
EXTERN char e_cannot_use_range_with_assignment_operator_str[]
|
EXTERN char e_cannot_use_range_with_assignment_operator_str[]
|
||||||
INIT(= N_("E1183: Cannot use a range with an assignment operator: %s"));
|
INIT(= N_("E1183: Cannot use a range with an assignment operator: %s"));
|
||||||
|
EXTERN char e_blob_not_set[]
|
||||||
|
INIT(= N_("E1184: Blob not set"));
|
||||||
|
@@ -7,7 +7,8 @@ void blob_free(blob_T *b);
|
|||||||
void blob_unref(blob_T *b);
|
void blob_unref(blob_T *b);
|
||||||
long blob_len(blob_T *b);
|
long blob_len(blob_T *b);
|
||||||
int blob_get(blob_T *b, int idx);
|
int blob_get(blob_T *b, int idx);
|
||||||
void blob_set(blob_T *b, int idx, char_u c);
|
void blob_set(blob_T *blob, int idx, int byte);
|
||||||
|
void blob_set_append(blob_T *blob, int idx, int byte);
|
||||||
int blob_equal(blob_T *b1, blob_T *b2);
|
int blob_equal(blob_T *b1, blob_T *b2);
|
||||||
int read_blob(FILE *fd, blob_T *blob);
|
int read_blob(FILE *fd, blob_T *blob);
|
||||||
int write_blob(FILE *fd, blob_T *blob);
|
int write_blob(FILE *fd, blob_T *blob);
|
||||||
|
@@ -120,7 +120,8 @@ func Test_blob_assign()
|
|||||||
endfunc
|
endfunc
|
||||||
|
|
||||||
func Test_blob_get_range()
|
func Test_blob_get_range()
|
||||||
let b = 0z0011223344
|
let lines =<< trim END
|
||||||
|
VAR b = 0z0011223344
|
||||||
call assert_equal(0z2233, b[2 : 3])
|
call assert_equal(0z2233, b[2 : 3])
|
||||||
call assert_equal(0z223344, b[2 : -1])
|
call assert_equal(0z223344, b[2 : -1])
|
||||||
call assert_equal(0z00, b[0 : -5])
|
call assert_equal(0z00, b[0 : -5])
|
||||||
@@ -130,10 +131,17 @@ func Test_blob_get_range()
|
|||||||
call assert_equal(0z0011223344, b[: -1])
|
call assert_equal(0z0011223344, b[: -1])
|
||||||
call assert_equal(0z, b[5 : 6])
|
call assert_equal(0z, b[5 : 6])
|
||||||
call assert_equal(0z0011, b[-10 : 1])
|
call assert_equal(0z0011, b[-10 : 1])
|
||||||
|
END
|
||||||
|
call CheckLegacyAndVim9Success(lines)
|
||||||
|
|
||||||
|
" legacy script white space
|
||||||
|
let b = 0z0011223344
|
||||||
|
call assert_equal(0z2233, b[2:3])
|
||||||
endfunc
|
endfunc
|
||||||
|
|
||||||
func Test_blob_get()
|
func Test_blob_get()
|
||||||
let b = 0z0011223344
|
let lines =<< trim END
|
||||||
|
VAR b = 0z0011223344
|
||||||
call assert_equal(0x00, get(b, 0))
|
call assert_equal(0x00, get(b, 0))
|
||||||
call assert_equal(0x22, get(b, 2, 999))
|
call assert_equal(0x22, get(b, 2, 999))
|
||||||
call assert_equal(0x44, get(b, 4))
|
call assert_equal(0x44, get(b, 4))
|
||||||
@@ -148,12 +156,25 @@ func Test_blob_get()
|
|||||||
call assert_equal(0x22, b[2])
|
call assert_equal(0x22, b[2])
|
||||||
call assert_equal(0x44, b[4])
|
call assert_equal(0x44, b[4])
|
||||||
call assert_equal(0x44, b[-1])
|
call assert_equal(0x44, b[-1])
|
||||||
call assert_fails('echo b[5]', 'E979:')
|
END
|
||||||
call assert_fails('echo b[-8]', 'E979:')
|
call CheckLegacyAndVim9Success(lines)
|
||||||
|
|
||||||
|
let lines =<< trim END
|
||||||
|
VAR b = 0z0011223344
|
||||||
|
echo b[5]
|
||||||
|
END
|
||||||
|
call CheckLegacyAndVim9Failure(lines, 'E979:')
|
||||||
|
|
||||||
|
let lines =<< trim END
|
||||||
|
VAR b = 0z0011223344
|
||||||
|
echo b[-8]
|
||||||
|
END
|
||||||
|
call CheckLegacyAndVim9Failure(lines, 'E979:')
|
||||||
endfunc
|
endfunc
|
||||||
|
|
||||||
func Test_blob_to_string()
|
func Test_blob_to_string()
|
||||||
let b = 0z00112233445566778899aabbccdd
|
let lines =<< trim END
|
||||||
|
VAR b = 0z00112233445566778899aabbccdd
|
||||||
call assert_equal('0z00112233.44556677.8899AABB.CCDD', string(b))
|
call assert_equal('0z00112233.44556677.8899AABB.CCDD', string(b))
|
||||||
call assert_equal(b, eval(string(b)))
|
call assert_equal(b, eval(string(b)))
|
||||||
call remove(b, 4, -1)
|
call remove(b, 4, -1)
|
||||||
@@ -161,47 +182,104 @@ func Test_blob_to_string()
|
|||||||
call remove(b, 0, 3)
|
call remove(b, 0, 3)
|
||||||
call assert_equal('0z', string(b))
|
call assert_equal('0z', string(b))
|
||||||
call assert_equal('0z', string(test_null_blob()))
|
call assert_equal('0z', string(test_null_blob()))
|
||||||
|
END
|
||||||
|
call CheckLegacyAndVim9Success(lines)
|
||||||
endfunc
|
endfunc
|
||||||
|
|
||||||
func Test_blob_compare()
|
func Test_blob_compare()
|
||||||
let b1 = 0z0011
|
let lines =<< trim END
|
||||||
let b2 = 0z1100
|
VAR b1 = 0z0011
|
||||||
let b3 = 0z001122
|
VAR b2 = 0z1100
|
||||||
|
VAR b3 = 0z001122
|
||||||
call assert_true(b1 == b1)
|
call assert_true(b1 == b1)
|
||||||
call assert_false(b1 == b2)
|
call assert_false(b1 == b2)
|
||||||
call assert_false(b1 == b3)
|
call assert_false(b1 == b3)
|
||||||
call assert_true(b1 != b2)
|
call assert_true(b1 != b2)
|
||||||
call assert_true(b1 != b3)
|
call assert_true(b1 != b3)
|
||||||
call assert_true(b1 == 0z0011)
|
call assert_true(b1 == 0z0011)
|
||||||
call assert_fails('echo b1 == 9', 'E977:')
|
|
||||||
call assert_fails('echo b1 != 9', 'E977:')
|
|
||||||
|
|
||||||
call assert_false(b1 is b2)
|
call assert_false(b1 is b2)
|
||||||
let b2 = b1
|
LET b2 = b1
|
||||||
call assert_true(b1 == b2)
|
call assert_true(b1 == b2)
|
||||||
call assert_true(b1 is b2)
|
call assert_true(b1 is b2)
|
||||||
let b2 = copy(b1)
|
LET b2 = copy(b1)
|
||||||
call assert_true(b1 == b2)
|
call assert_true(b1 == b2)
|
||||||
call assert_false(b1 is b2)
|
call assert_false(b1 is b2)
|
||||||
let b2 = b1[:]
|
LET b2 = b1[:]
|
||||||
call assert_true(b1 == b2)
|
call assert_true(b1 == b2)
|
||||||
call assert_false(b1 is b2)
|
call assert_false(b1 is b2)
|
||||||
call assert_true(b1 isnot b2)
|
call assert_true(b1 isnot b2)
|
||||||
|
END
|
||||||
|
call CheckLegacyAndVim9Success(lines)
|
||||||
|
|
||||||
call assert_fails('let x = b1 > b2')
|
let lines =<< trim END
|
||||||
call assert_fails('let x = b1 < b2')
|
VAR b1 = 0z0011
|
||||||
call assert_fails('let x = b1 - b2')
|
echo b1 == 9
|
||||||
call assert_fails('let x = b1 / b2')
|
END
|
||||||
call assert_fails('let x = b1 * b2')
|
call CheckLegacyAndVim9Failure(lines, ['E977:', 'E1072', 'E1072'])
|
||||||
|
|
||||||
|
let lines =<< trim END
|
||||||
|
VAR b1 = 0z0011
|
||||||
|
echo b1 != 9
|
||||||
|
END
|
||||||
|
call CheckLegacyAndVim9Failure(lines, ['E977:', 'E1072', 'E1072'])
|
||||||
|
|
||||||
|
let lines =<< trim END
|
||||||
|
VAR b1 = 0z0011
|
||||||
|
VAR b2 = 0z1100
|
||||||
|
VAR x = b1 > b2
|
||||||
|
END
|
||||||
|
call CheckLegacyAndVim9Failure(lines, ['E978:', 'E1072:', 'E1072:'])
|
||||||
|
|
||||||
|
let lines =<< trim END
|
||||||
|
VAR b1 = 0z0011
|
||||||
|
VAR b2 = 0z1100
|
||||||
|
VAR x = b1 < b2
|
||||||
|
END
|
||||||
|
call CheckLegacyAndVim9Failure(lines, ['E978:', 'E1072:', 'E1072:'])
|
||||||
|
|
||||||
|
let lines =<< trim END
|
||||||
|
VAR b1 = 0z0011
|
||||||
|
VAR b2 = 0z1100
|
||||||
|
VAR x = b1 - b2
|
||||||
|
END
|
||||||
|
call CheckLegacyAndVim9Failure(lines, ['E974:', 'E1036:', 'E974:'])
|
||||||
|
|
||||||
|
let lines =<< trim END
|
||||||
|
VAR b1 = 0z0011
|
||||||
|
VAR b2 = 0z1100
|
||||||
|
VAR x = b1 / b2
|
||||||
|
END
|
||||||
|
call CheckLegacyAndVim9Failure(lines, ['E974:', 'E1036:', 'E974:'])
|
||||||
|
|
||||||
|
let lines =<< trim END
|
||||||
|
VAR b1 = 0z0011
|
||||||
|
VAR b2 = 0z1100
|
||||||
|
VAR x = b1 * b2
|
||||||
|
END
|
||||||
|
call CheckLegacyAndVim9Failure(lines, ['E974:', 'E1036:', 'E974:'])
|
||||||
endfunc
|
endfunc
|
||||||
|
|
||||||
" test for range assign
|
func Test_blob_index_assign()
|
||||||
func Test_blob_range_assign()
|
let lines =<< trim END
|
||||||
let b = 0z00
|
VAR b = 0z00
|
||||||
let b[1] = 0x11
|
LET b[1] = 0x11
|
||||||
let b[2] = 0x22
|
LET b[2] = 0x22
|
||||||
call assert_equal(0z001122, b)
|
call assert_equal(0z001122, b)
|
||||||
call assert_fails('let b[4] = 0x33', 'E979:')
|
END
|
||||||
|
call CheckLegacyAndVim9Success(lines)
|
||||||
|
|
||||||
|
let lines =<< trim END
|
||||||
|
VAR b = 0z00
|
||||||
|
LET b[2] = 0x33
|
||||||
|
END
|
||||||
|
call CheckLegacyAndVim9Failure(lines, 'E979:')
|
||||||
|
|
||||||
|
let lines =<< trim END
|
||||||
|
VAR b = 0z00
|
||||||
|
LET b[-2] = 0x33
|
||||||
|
END
|
||||||
|
call CheckLegacyAndVim9Failure(lines, 'E979:')
|
||||||
endfunc
|
endfunc
|
||||||
|
|
||||||
func Test_blob_for_loop()
|
func Test_blob_for_loop()
|
||||||
|
@@ -246,6 +246,8 @@ def s:ScriptFuncStoreMember()
|
|||||||
locallist[0] = 123
|
locallist[0] = 123
|
||||||
var localdict: dict<number> = {}
|
var localdict: dict<number> = {}
|
||||||
localdict["a"] = 456
|
localdict["a"] = 456
|
||||||
|
var localblob: blob = 0z1122
|
||||||
|
localblob[1] = 33
|
||||||
enddef
|
enddef
|
||||||
|
|
||||||
def Test_disassemble_store_member()
|
def Test_disassemble_store_member()
|
||||||
@@ -259,7 +261,7 @@ def Test_disassemble_store_member()
|
|||||||
'\d PUSHNR 123\_s*' ..
|
'\d PUSHNR 123\_s*' ..
|
||||||
'\d PUSHNR 0\_s*' ..
|
'\d PUSHNR 0\_s*' ..
|
||||||
'\d LOAD $0\_s*' ..
|
'\d LOAD $0\_s*' ..
|
||||||
'\d STORELIST\_s*' ..
|
'\d STOREINDEX list\_s*' ..
|
||||||
'var localdict: dict<number> = {}\_s*' ..
|
'var localdict: dict<number> = {}\_s*' ..
|
||||||
'\d NEWDICT size 0\_s*' ..
|
'\d NEWDICT size 0\_s*' ..
|
||||||
'\d SETTYPE dict<number>\_s*' ..
|
'\d SETTYPE dict<number>\_s*' ..
|
||||||
@@ -268,7 +270,15 @@ def Test_disassemble_store_member()
|
|||||||
'\d\+ PUSHNR 456\_s*' ..
|
'\d\+ PUSHNR 456\_s*' ..
|
||||||
'\d\+ PUSHS "a"\_s*' ..
|
'\d\+ PUSHS "a"\_s*' ..
|
||||||
'\d\+ LOAD $1\_s*' ..
|
'\d\+ LOAD $1\_s*' ..
|
||||||
'\d\+ STOREDICT\_s*' ..
|
'\d\+ STOREINDEX dict\_s*' ..
|
||||||
|
'var localblob: blob = 0z1122\_s*' ..
|
||||||
|
'\d\+ PUSHBLOB 0z1122\_s*' ..
|
||||||
|
'\d\+ STORE $2\_s*' ..
|
||||||
|
'localblob\[1\] = 33\_s*' ..
|
||||||
|
'\d\+ PUSHNR 33\_s*' ..
|
||||||
|
'\d\+ PUSHNR 1\_s*' ..
|
||||||
|
'\d\+ LOAD $2\_s*' ..
|
||||||
|
'\d\+ STOREINDEX blob\_s*' ..
|
||||||
'\d\+ RETURN 0',
|
'\d\+ RETURN 0',
|
||||||
res)
|
res)
|
||||||
enddef
|
enddef
|
||||||
@@ -291,7 +301,7 @@ def Test_disassemble_store_index()
|
|||||||
'\d PUSHNR 0\_s*' ..
|
'\d PUSHNR 0\_s*' ..
|
||||||
'\d LOAD $0\_s*' ..
|
'\d LOAD $0\_s*' ..
|
||||||
'\d MEMBER dd\_s*' ..
|
'\d MEMBER dd\_s*' ..
|
||||||
'\d STOREINDEX\_s*' ..
|
'\d STOREINDEX any\_s*' ..
|
||||||
'\d\+ RETURN 0',
|
'\d\+ RETURN 0',
|
||||||
res)
|
res)
|
||||||
enddef
|
enddef
|
||||||
|
@@ -750,6 +750,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 */
|
||||||
|
/**/
|
||||||
|
2777,
|
||||||
/**/
|
/**/
|
||||||
2776,
|
2776,
|
||||||
/**/
|
/**/
|
||||||
|
@@ -6209,14 +6209,21 @@ compile_assign_unlet(
|
|||||||
}
|
}
|
||||||
if (dest_type == VAR_DICT && may_generate_2STRING(-1, cctx) == FAIL)
|
if (dest_type == VAR_DICT && may_generate_2STRING(-1, cctx) == FAIL)
|
||||||
return FAIL;
|
return FAIL;
|
||||||
if (dest_type == VAR_LIST)
|
if (dest_type == VAR_LIST || dest_type == VAR_BLOB)
|
||||||
{
|
{
|
||||||
if (range
|
type_T *type;
|
||||||
&& need_type(((type_T **)stack->ga_data)[stack->ga_len - 2],
|
|
||||||
&t_number, -1, 0, cctx, FALSE, FALSE) == FAIL)
|
if (range)
|
||||||
|
{
|
||||||
|
type = ((type_T **)stack->ga_data)[stack->ga_len - 2];
|
||||||
|
if (need_type(type, &t_number,
|
||||||
|
-1, 0, cctx, FALSE, FALSE) == FAIL)
|
||||||
return FAIL;
|
return FAIL;
|
||||||
if (need_type(((type_T **)stack->ga_data)[stack->ga_len - 1],
|
}
|
||||||
&t_number, -1, 0, cctx, FALSE, FALSE) == FAIL)
|
type = ((type_T **)stack->ga_data)[stack->ga_len - 1];
|
||||||
|
if ((dest_type != VAR_BLOB || type != &t_special)
|
||||||
|
&& need_type(type, &t_number,
|
||||||
|
-1, 0, cctx, FALSE, FALSE) == FAIL)
|
||||||
return FAIL;
|
return FAIL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -2221,7 +2221,35 @@ call_def_function(
|
|||||||
}
|
}
|
||||||
else if (status == OK && dest_type == VAR_BLOB)
|
else if (status == OK && dest_type == VAR_BLOB)
|
||||||
{
|
{
|
||||||
// TODO
|
long lidx = (long)tv_idx->vval.v_number;
|
||||||
|
blob_T *blob = tv_dest->vval.v_blob;
|
||||||
|
varnumber_T nr;
|
||||||
|
int error = FALSE;
|
||||||
|
int len;
|
||||||
|
|
||||||
|
if (blob == NULL)
|
||||||
|
{
|
||||||
|
emsg(_(e_blob_not_set));
|
||||||
|
goto on_error;
|
||||||
|
}
|
||||||
|
len = blob_len(blob);
|
||||||
|
if (lidx < 0 && len + lidx >= 0)
|
||||||
|
// negative index is relative to the end
|
||||||
|
lidx = len + lidx;
|
||||||
|
|
||||||
|
// Can add one byte at the end.
|
||||||
|
if (lidx < 0 || lidx > len)
|
||||||
|
{
|
||||||
|
semsg(_(e_blobidx), lidx);
|
||||||
|
goto on_error;
|
||||||
|
}
|
||||||
|
if (value_check_lock(blob->bv_lock,
|
||||||
|
(char_u *)"blob", FALSE))
|
||||||
|
goto on_error;
|
||||||
|
nr = tv_get_number_chk(tv, &error);
|
||||||
|
if (error)
|
||||||
|
goto on_error;
|
||||||
|
blob_set_append(blob, lidx, nr);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@@ -4415,19 +4443,8 @@ ex_disassemble(exarg_T *eap)
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case ISN_STOREINDEX:
|
case ISN_STOREINDEX:
|
||||||
switch (iptr->isn_arg.vartype)
|
smsg("%4d STOREINDEX %s", current,
|
||||||
{
|
vartype_name(iptr->isn_arg.vartype));
|
||||||
case VAR_LIST:
|
|
||||||
smsg("%4d STORELIST", current);
|
|
||||||
break;
|
|
||||||
case VAR_DICT:
|
|
||||||
smsg("%4d STOREDICT", current);
|
|
||||||
break;
|
|
||||||
case VAR_ANY:
|
|
||||||
smsg("%4d STOREINDEX", current);
|
|
||||||
break;
|
|
||||||
default: break;
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case ISN_STORERANGE:
|
case ISN_STORERANGE:
|
||||||
|
Reference in New Issue
Block a user