diff options
Diffstat (limited to 'editors')
-rw-r--r-- | editors/awk.c | 26 |
1 files changed, 20 insertions, 6 deletions
diff --git a/editors/awk.c b/editors/awk.c index 728ee8685..2af823808 100644 --- a/editors/awk.c +++ b/editors/awk.c | |||
@@ -555,7 +555,7 @@ struct globals { | |||
555 | const char *g_progname; | 555 | const char *g_progname; |
556 | int g_lineno; | 556 | int g_lineno; |
557 | int nfields; | 557 | int nfields; |
558 | int maxfields; /* used in fsrealloc() only */ | 558 | unsigned maxfields; |
559 | var *Fields; | 559 | var *Fields; |
560 | char *g_pos; | 560 | char *g_pos; |
561 | char g_saved_ch; | 561 | char g_saved_ch; |
@@ -1931,9 +1931,9 @@ static void fsrealloc(int size) | |||
1931 | { | 1931 | { |
1932 | int i, newsize; | 1932 | int i, newsize; |
1933 | 1933 | ||
1934 | if (size >= maxfields) { | 1934 | if ((unsigned)size >= maxfields) { |
1935 | /* Sanity cap, easier than catering for overflows */ | 1935 | /* Sanity cap, easier than catering for over/underflows */ |
1936 | if (size > 0xffffff) | 1936 | if ((unsigned)size > 0xffffff) |
1937 | bb_die_memory_exhausted(); | 1937 | bb_die_memory_exhausted(); |
1938 | 1938 | ||
1939 | i = maxfields; | 1939 | i = maxfields; |
@@ -2891,6 +2891,7 @@ static var *evaluate(node *op, var *res) | |||
2891 | uint32_t opinfo; | 2891 | uint32_t opinfo; |
2892 | int opn; | 2892 | int opn; |
2893 | node *op1; | 2893 | node *op1; |
2894 | var *old_Fields_ptr; | ||
2894 | 2895 | ||
2895 | opinfo = op->info; | 2896 | opinfo = op->info; |
2896 | opn = (opinfo & OPNMASK); | 2897 | opn = (opinfo & OPNMASK); |
@@ -2899,10 +2900,16 @@ static var *evaluate(node *op, var *res) | |||
2899 | debug_printf_eval("opinfo:%08x opn:%08x\n", opinfo, opn); | 2900 | debug_printf_eval("opinfo:%08x opn:%08x\n", opinfo, opn); |
2900 | 2901 | ||
2901 | /* execute inevitable things */ | 2902 | /* execute inevitable things */ |
2903 | old_Fields_ptr = NULL; | ||
2902 | if (opinfo & OF_RES1) { | 2904 | if (opinfo & OF_RES1) { |
2903 | if ((opinfo & OF_REQUIRED) && !op1) | 2905 | if ((opinfo & OF_REQUIRED) && !op1) |
2904 | syntax_error(EMSG_TOO_FEW_ARGS); | 2906 | syntax_error(EMSG_TOO_FEW_ARGS); |
2905 | L.v = evaluate(op1, TMPVAR0); | 2907 | L.v = evaluate(op1, TMPVAR0); |
2908 | /* Does L.v point to $n variable? */ | ||
2909 | if ((size_t)(L.v - Fields) < maxfields) { | ||
2910 | /* yes, remember where Fields[] is */ | ||
2911 | old_Fields_ptr = Fields; | ||
2912 | } | ||
2906 | if (opinfo & OF_STR1) { | 2913 | if (opinfo & OF_STR1) { |
2907 | L.s = getvar_s(L.v); | 2914 | L.s = getvar_s(L.v); |
2908 | debug_printf_eval("L.s:'%s'\n", L.s); | 2915 | debug_printf_eval("L.s:'%s'\n", L.s); |
@@ -2921,8 +2928,15 @@ static var *evaluate(node *op, var *res) | |||
2921 | */ | 2928 | */ |
2922 | if (opinfo & OF_RES2) { | 2929 | if (opinfo & OF_RES2) { |
2923 | R.v = evaluate(op->r.n, TMPVAR1); | 2930 | R.v = evaluate(op->r.n, TMPVAR1); |
2924 | //TODO: L.v may be invalid now, set L.v to NULL to catch bugs? | 2931 | /* Seen in $5=$$5=$0: |
2925 | //L.v = NULL; | 2932 | * Evaluation of R.v ($$5=$0 expression) |
2933 | * made L.v ($5) invalid. It's detected here. | ||
2934 | */ | ||
2935 | if (old_Fields_ptr) { | ||
2936 | //if (old_Fields_ptr != Fields) | ||
2937 | // debug_printf_eval("L.v moved\n"); | ||
2938 | L.v += Fields - old_Fields_ptr; | ||
2939 | } | ||
2926 | if (opinfo & OF_STR2) { | 2940 | if (opinfo & OF_STR2) { |
2927 | R.s = getvar_s(R.v); | 2941 | R.s = getvar_s(R.v); |
2928 | debug_printf_eval("R.s:'%s'\n", R.s); | 2942 | debug_printf_eval("R.s:'%s'\n", R.s); |