Elliott Hughes | 5001206 | 2015-03-10 22:22:24 -0700 | [diff] [blame] | 1 | /* $OpenBSD: expr.c,v 1.24 2014/12/08 14:26:31 otto Exp $ */ |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 2 | |
| 3 | /*- |
Thorsten Glaser | c2dc5de | 2013-02-18 23:02:51 +0000 | [diff] [blame] | 4 | * Copyright (c) 2003, 2004, 2005, 2006, 2007, 2008, 2009, 2010, |
Elliott Hughes | a3c3f96 | 2017-04-12 16:52:30 -0700 | [diff] [blame] | 5 | * 2011, 2012, 2013, 2014, 2016, 2017 |
Elliott Hughes | fc0307d | 2016-02-02 15:26:47 -0800 | [diff] [blame] | 6 | * mirabilos <m@mirbsd.org> |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 7 | * |
| 8 | * Provided that these terms and disclaimer and all copyright notices |
| 9 | * are retained or reproduced in an accompanying document, permission |
| 10 | * is granted to deal in this work without restriction, including un- |
| 11 | * limited rights to use, publicly perform, distribute, sell, modify, |
| 12 | * merge, give away, or sublicence. |
| 13 | * |
| 14 | * This work is provided "AS IS" and WITHOUT WARRANTY of any kind, to |
| 15 | * the utmost extent permitted by applicable law, neither express nor |
| 16 | * implied; without malicious intent or gross negligence. In no event |
| 17 | * may a licensor, author or contributor be held liable for indirect, |
| 18 | * direct, other damage, loss, or other issues arising in any way out |
| 19 | * of dealing in the work, even if advised of the possibility of such |
| 20 | * damage or existence of a defect, except proven that it results out |
| 21 | * of said person's immediate fault when using the work as intended. |
| 22 | */ |
| 23 | |
| 24 | #include "sh.h" |
| 25 | |
Elliott Hughes | a3c3f96 | 2017-04-12 16:52:30 -0700 | [diff] [blame] | 26 | __RCSID("$MirOS: src/bin/mksh/expr.c,v 1.93 2017/04/02 16:47:41 tg Exp $"); |
Thorsten Glaser | c2dc5de | 2013-02-18 23:02:51 +0000 | [diff] [blame] | 27 | |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 28 | #define EXPRTOK_DEFNS |
| 29 | #include "exprtok.h" |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 30 | |
| 31 | /* precisions; used to be enum prec but we do arithmetics on it */ |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 32 | #define P_PRIMARY 0 /* VAR, LIT, (), ! ~ ++ -- */ |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 33 | #define P_MULT 1 /* * / % */ |
| 34 | #define P_ADD 2 /* + - */ |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 35 | #define P_SHIFT 3 /* ^< ^> << >> */ |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 36 | #define P_RELATION 4 /* < <= > >= */ |
| 37 | #define P_EQUALITY 5 /* == != */ |
| 38 | #define P_BAND 6 /* & */ |
| 39 | #define P_BXOR 7 /* ^ */ |
| 40 | #define P_BOR 8 /* | */ |
| 41 | #define P_LAND 9 /* && */ |
| 42 | #define P_LOR 10 /* || */ |
| 43 | #define P_TERN 11 /* ?: */ |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 44 | /* = += -= *= /= %= ^<= ^>= <<= >>= &= ^= |= */ |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 45 | #define P_ASSIGN 12 |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 46 | #define P_COMMA 13 /* , */ |
| 47 | #define MAX_PREC P_COMMA |
| 48 | |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 49 | enum token { |
| 50 | #define EXPRTOK_ENUM |
| 51 | #include "exprtok.h" |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 52 | }; |
| 53 | |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 54 | static const char opname[][4] = { |
| 55 | #define EXPRTOK_NAME |
| 56 | #include "exprtok.h" |
| 57 | }; |
| 58 | |
| 59 | static const uint8_t oplen[] = { |
| 60 | #define EXPRTOK_LEN |
| 61 | #include "exprtok.h" |
| 62 | }; |
| 63 | |
| 64 | static const uint8_t opprec[] = { |
| 65 | #define EXPRTOK_PREC |
| 66 | #include "exprtok.h" |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 67 | }; |
| 68 | |
Thorsten Glaser | c2dc5de | 2013-02-18 23:02:51 +0000 | [diff] [blame] | 69 | typedef struct expr_state { |
| 70 | /* expression being evaluated */ |
| 71 | const char *expression; |
| 72 | /* lexical position */ |
| 73 | const char *tokp; |
| 74 | /* value from token() */ |
| 75 | struct tbl *val; |
| 76 | /* variable that is being recursively expanded (EXPRINEVAL flag set) */ |
| 77 | struct tbl *evaling; |
| 78 | /* token from token() */ |
| 79 | enum token tok; |
| 80 | /* don't do assignments (for ?:, &&, ||) */ |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 81 | uint8_t noassign; |
Thorsten Glaser | c2dc5de | 2013-02-18 23:02:51 +0000 | [diff] [blame] | 82 | /* evaluating an $(()) expression? */ |
| 83 | bool arith; |
| 84 | /* unsigned arithmetic calculation */ |
| 85 | bool natural; |
| 86 | } Expr_state; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 87 | |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 88 | enum error_type { |
| 89 | ET_UNEXPECTED, ET_BADLIT, ET_RECURSIVE, |
| 90 | ET_LVALUE, ET_RDONLY, ET_STR |
| 91 | }; |
| 92 | |
| 93 | static void evalerr(Expr_state *, enum error_type, const char *) |
| 94 | MKSH_A_NORETURN; |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 95 | static struct tbl *evalexpr(Expr_state *, unsigned int); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 96 | static void exprtoken(Expr_state *); |
| 97 | static struct tbl *do_ppmm(Expr_state *, enum token, struct tbl *, bool); |
| 98 | static void assign_check(Expr_state *, enum token, struct tbl *); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 99 | static struct tbl *intvar(Expr_state *, struct tbl *); |
| 100 | |
| 101 | /* |
| 102 | * parse and evaluate expression |
| 103 | */ |
| 104 | int |
| 105 | evaluate(const char *expr, mksh_ari_t *rval, int error_ok, bool arith) |
| 106 | { |
| 107 | struct tbl v; |
| 108 | int ret; |
| 109 | |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 110 | v.flag = DEFINED | INTEGER; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 111 | v.type = 0; |
| 112 | ret = v_evaluate(&v, expr, error_ok, arith); |
| 113 | *rval = v.val.i; |
| 114 | return (ret); |
| 115 | } |
| 116 | |
| 117 | /* |
| 118 | * parse and evaluate expression, storing result in vp. |
| 119 | */ |
| 120 | int |
| 121 | v_evaluate(struct tbl *vp, const char *expr, volatile int error_ok, |
| 122 | bool arith) |
| 123 | { |
| 124 | struct tbl *v; |
| 125 | Expr_state curstate; |
| 126 | Expr_state * const es = &curstate; |
| 127 | int i; |
| 128 | |
| 129 | /* save state to allow recursive calls */ |
Thorsten Glaser | c2dc5de | 2013-02-18 23:02:51 +0000 | [diff] [blame] | 130 | memset(&curstate, 0, sizeof(curstate)); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 131 | curstate.expression = curstate.tokp = expr; |
Thorsten Glaser | c2dc5de | 2013-02-18 23:02:51 +0000 | [diff] [blame] | 132 | curstate.tok = BAD; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 133 | curstate.arith = arith; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 134 | |
| 135 | newenv(E_ERRH); |
Thorsten Glaser | c2dc5de | 2013-02-18 23:02:51 +0000 | [diff] [blame] | 136 | if ((i = kshsetjmp(e->jbuf))) { |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 137 | /* Clear EXPRINEVAL in of any variables we were playing with */ |
| 138 | if (curstate.evaling) |
| 139 | curstate.evaling->flag &= ~EXPRINEVAL; |
| 140 | quitenv(NULL); |
| 141 | if (i == LAEXPR) { |
| 142 | if (error_ok == KSH_RETURN_ERROR) |
| 143 | return (0); |
| 144 | errorfz(); |
| 145 | } |
| 146 | unwind(i); |
| 147 | /* NOTREACHED */ |
| 148 | } |
| 149 | |
| 150 | exprtoken(es); |
| 151 | if (es->tok == END) { |
| 152 | es->tok = LIT; |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 153 | es->val = tempvar(""); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 154 | } |
| 155 | v = intvar(es, evalexpr(es, MAX_PREC)); |
| 156 | |
| 157 | if (es->tok != END) |
| 158 | evalerr(es, ET_UNEXPECTED, NULL); |
| 159 | |
| 160 | if (es->arith && es->natural) |
| 161 | vp->flag |= INT_U; |
| 162 | if (vp->flag & INTEGER) |
| 163 | setint_v(vp, v, es->arith); |
| 164 | else |
| 165 | /* can fail if readonly */ |
| 166 | setstr(vp, str_val(v), error_ok); |
| 167 | |
| 168 | quitenv(NULL); |
| 169 | |
| 170 | return (1); |
| 171 | } |
| 172 | |
| 173 | static void |
| 174 | evalerr(Expr_state *es, enum error_type type, const char *str) |
| 175 | { |
| 176 | char tbuf[2]; |
| 177 | const char *s; |
| 178 | |
| 179 | es->arith = false; |
| 180 | switch (type) { |
| 181 | case ET_UNEXPECTED: |
| 182 | switch (es->tok) { |
| 183 | case VAR: |
| 184 | s = es->val->name; |
| 185 | break; |
| 186 | case LIT: |
| 187 | s = str_val(es->val); |
| 188 | break; |
| 189 | case END: |
| 190 | s = "end of expression"; |
| 191 | break; |
| 192 | case BAD: |
| 193 | tbuf[0] = *es->tokp; |
| 194 | tbuf[1] = '\0'; |
| 195 | s = tbuf; |
| 196 | break; |
| 197 | default: |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 198 | s = opname[(int)es->tok]; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 199 | } |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 200 | warningf(true, Tf_sD_s_qs, es->expression, |
| 201 | Tunexpected, s); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 202 | break; |
| 203 | |
| 204 | case ET_BADLIT: |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 205 | warningf(true, Tf_sD_s_qs, es->expression, |
Elliott Hughes | a3c3f96 | 2017-04-12 16:52:30 -0700 | [diff] [blame] | 206 | Tbadnum, str); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 207 | break; |
| 208 | |
| 209 | case ET_RECURSIVE: |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 210 | warningf(true, Tf_sD_s_qs, es->expression, |
Geremy Condra | 03ebf06 | 2011-10-12 18:17:24 -0700 | [diff] [blame] | 211 | "expression recurses on parameter", str); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 212 | break; |
| 213 | |
| 214 | case ET_LVALUE: |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 215 | warningf(true, Tf_sD_s_s, |
Geremy Condra | 03ebf06 | 2011-10-12 18:17:24 -0700 | [diff] [blame] | 216 | es->expression, str, "requires lvalue"); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 217 | break; |
| 218 | |
| 219 | case ET_RDONLY: |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 220 | warningf(true, Tf_sD_s_s, |
Thorsten Glaser | c2dc5de | 2013-02-18 23:02:51 +0000 | [diff] [blame] | 221 | es->expression, str, "applied to read-only variable"); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 222 | break; |
| 223 | |
| 224 | default: /* keep gcc happy */ |
| 225 | case ET_STR: |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 226 | warningf(true, Tf_sD_s, es->expression, str); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 227 | break; |
| 228 | } |
| 229 | unwind(LAEXPR); |
| 230 | } |
| 231 | |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 232 | /* do a ++ or -- operation */ |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 233 | static struct tbl * |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 234 | do_ppmm(Expr_state *es, enum token op, struct tbl *vasn, bool is_prefix) |
| 235 | { |
| 236 | struct tbl *vl; |
| 237 | mksh_uari_t oval; |
| 238 | |
| 239 | assign_check(es, op, vasn); |
| 240 | |
| 241 | vl = intvar(es, vasn); |
| 242 | oval = vl->val.u; |
| 243 | if (op == O_PLUSPLUS) |
| 244 | ++vl->val.u; |
| 245 | else |
| 246 | --vl->val.u; |
| 247 | if (!es->noassign) { |
| 248 | if (vasn->flag & INTEGER) |
| 249 | setint_v(vasn, vl, es->arith); |
| 250 | else |
| 251 | setint(vasn, vl->val.i); |
| 252 | } |
| 253 | if (!is_prefix) |
| 254 | /* undo the increment/decrement */ |
| 255 | vl->val.u = oval; |
| 256 | |
| 257 | return (vl); |
| 258 | } |
| 259 | |
| 260 | static struct tbl * |
| 261 | evalexpr(Expr_state *es, unsigned int prec) |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 262 | { |
| 263 | struct tbl *vl, *vr = NULL, *vasn; |
| 264 | enum token op; |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 265 | mksh_uari_t res = 0, t1, t2, t3; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 266 | |
| 267 | if (prec == P_PRIMARY) { |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 268 | switch ((int)(op = es->tok)) { |
| 269 | case O_BNOT: |
| 270 | case O_LNOT: |
| 271 | case O_MINUS: |
| 272 | case O_PLUS: |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 273 | exprtoken(es); |
| 274 | vl = intvar(es, evalexpr(es, P_PRIMARY)); |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 275 | switch ((int)op) { |
| 276 | case O_BNOT: |
| 277 | vl->val.u = ~vl->val.u; |
| 278 | break; |
| 279 | case O_LNOT: |
| 280 | vl->val.u = !vl->val.u; |
| 281 | break; |
| 282 | case O_MINUS: |
| 283 | vl->val.u = -vl->val.u; |
| 284 | break; |
| 285 | case O_PLUS: |
| 286 | /* nop */ |
| 287 | break; |
| 288 | } |
| 289 | break; |
| 290 | |
| 291 | case OPEN_PAREN: |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 292 | exprtoken(es); |
| 293 | vl = evalexpr(es, MAX_PREC); |
| 294 | if (es->tok != CLOSE_PAREN) |
| 295 | evalerr(es, ET_STR, "missing )"); |
| 296 | exprtoken(es); |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 297 | break; |
| 298 | |
| 299 | case O_PLUSPLUS: |
| 300 | case O_MINUSMINUS: |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 301 | exprtoken(es); |
| 302 | vl = do_ppmm(es, op, es->val, true); |
| 303 | exprtoken(es); |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 304 | break; |
| 305 | |
| 306 | case VAR: |
| 307 | case LIT: |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 308 | vl = es->val; |
| 309 | exprtoken(es); |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 310 | break; |
| 311 | |
| 312 | default: |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 313 | evalerr(es, ET_UNEXPECTED, NULL); |
| 314 | /* NOTREACHED */ |
| 315 | } |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 316 | |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 317 | if (es->tok == O_PLUSPLUS || es->tok == O_MINUSMINUS) { |
| 318 | vl = do_ppmm(es, es->tok, vl, false); |
| 319 | exprtoken(es); |
| 320 | } |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 321 | |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 322 | return (vl); |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 323 | /* prec == P_PRIMARY */ |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 324 | } |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 325 | |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 326 | vl = evalexpr(es, prec - 1); |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 327 | while ((int)(op = es->tok) >= (int)O_EQ && (int)op <= (int)O_COMMA && |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 328 | opprec[(int)op] == prec) { |
Elliott Hughes | 966dd55 | 2016-12-08 15:56:04 -0800 | [diff] [blame] | 329 | switch ((int)op) { |
| 330 | case O_TERN: |
| 331 | case O_LAND: |
| 332 | case O_LOR: |
| 333 | break; |
| 334 | default: |
| 335 | exprtoken(es); |
| 336 | } |
| 337 | |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 338 | vasn = vl; |
Thorsten Glaser | c2dc5de | 2013-02-18 23:02:51 +0000 | [diff] [blame] | 339 | if (op != O_ASN) |
| 340 | /* vl may not have a value yet */ |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 341 | vl = intvar(es, vl); |
| 342 | if (IS_ASSIGNOP(op)) { |
Thorsten Glaser | c2dc5de | 2013-02-18 23:02:51 +0000 | [diff] [blame] | 343 | if (!es->noassign) |
| 344 | assign_check(es, op, vasn); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 345 | vr = intvar(es, evalexpr(es, P_ASSIGN)); |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 346 | } else if (op == O_TERN) { |
| 347 | bool ev = vl->val.u != 0; |
| 348 | |
| 349 | if (!ev) |
| 350 | es->noassign++; |
Elliott Hughes | 966dd55 | 2016-12-08 15:56:04 -0800 | [diff] [blame] | 351 | exprtoken(es); |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 352 | vl = evalexpr(es, MAX_PREC); |
| 353 | if (!ev) |
| 354 | es->noassign--; |
| 355 | if (es->tok != CTERN) |
| 356 | evalerr(es, ET_STR, "missing :"); |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 357 | if (ev) |
| 358 | es->noassign++; |
Elliott Hughes | 966dd55 | 2016-12-08 15:56:04 -0800 | [diff] [blame] | 359 | exprtoken(es); |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 360 | vr = evalexpr(es, P_TERN); |
| 361 | if (ev) |
| 362 | es->noassign--; |
| 363 | vl = ev ? vl : vr; |
| 364 | continue; |
| 365 | } else if (op != O_LAND && op != O_LOR) |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 366 | vr = intvar(es, evalexpr(es, prec - 1)); |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 367 | |
| 368 | /* common ops setup */ |
| 369 | switch ((int)op) { |
| 370 | case O_DIV: |
| 371 | case O_DIVASN: |
| 372 | case O_MOD: |
| 373 | case O_MODASN: |
| 374 | if (vr->val.u == 0) { |
| 375 | if (!es->noassign) |
| 376 | evalerr(es, ET_STR, "zero divisor"); |
| 377 | vr->val.u = 1; |
| 378 | } |
| 379 | /* calculate the absolute values */ |
| 380 | t1 = vl->val.i < 0 ? -vl->val.u : vl->val.u; |
| 381 | t2 = vr->val.i < 0 ? -vr->val.u : vr->val.u; |
| 382 | break; |
| 383 | #ifndef MKSH_LEGACY_MODE |
| 384 | case O_LSHIFT: |
| 385 | case O_LSHIFTASN: |
| 386 | case O_RSHIFT: |
| 387 | case O_RSHIFTASN: |
| 388 | case O_ROL: |
| 389 | case O_ROLASN: |
| 390 | case O_ROR: |
| 391 | case O_RORASN: |
| 392 | t1 = vl->val.u; |
| 393 | t2 = vr->val.u & 31; |
| 394 | break; |
| 395 | #endif |
| 396 | case O_LAND: |
| 397 | case O_LOR: |
| 398 | t1 = vl->val.u; |
| 399 | t2 = 0; /* gcc */ |
| 400 | break; |
| 401 | default: |
| 402 | t1 = vl->val.u; |
| 403 | t2 = vr->val.u; |
| 404 | break; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 405 | } |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 406 | |
| 407 | #define cmpop(op) (es->natural ? \ |
| 408 | (mksh_uari_t)(vl->val.u op vr->val.u) : \ |
| 409 | (mksh_uari_t)(vl->val.i op vr->val.i) \ |
| 410 | ) |
| 411 | |
| 412 | /* op calculation */ |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 413 | switch ((int)op) { |
| 414 | case O_TIMES: |
| 415 | case O_TIMESASN: |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 416 | res = t1 * t2; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 417 | break; |
| 418 | case O_MOD: |
| 419 | case O_MODASN: |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 420 | if (es->natural) { |
| 421 | res = vl->val.u % vr->val.u; |
| 422 | break; |
| 423 | } |
| 424 | goto signed_division; |
| 425 | case O_DIV: |
| 426 | case O_DIVASN: |
| 427 | if (es->natural) { |
| 428 | res = vl->val.u / vr->val.u; |
| 429 | break; |
| 430 | } |
| 431 | signed_division: |
| 432 | /* |
| 433 | * a / b = abs(a) / abs(b) * sgn((u)a^(u)b) |
| 434 | */ |
| 435 | t3 = t1 / t2; |
| 436 | #ifndef MKSH_LEGACY_MODE |
| 437 | res = ((vl->val.u ^ vr->val.u) & 0x80000000) ? -t3 : t3; |
| 438 | #else |
| 439 | res = ((t1 == vl->val.u ? 0 : 1) ^ |
| 440 | (t2 == vr->val.u ? 0 : 1)) ? -t3 : t3; |
Thorsten Glaser | c2dc5de | 2013-02-18 23:02:51 +0000 | [diff] [blame] | 441 | #endif |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 442 | if (op == O_MOD || op == O_MODASN) { |
| 443 | /* |
| 444 | * primitive modulo, to get the sign of |
| 445 | * the result correct: |
| 446 | * (a % b) = a - ((a / b) * b) |
| 447 | * the subtraction and multiplication |
| 448 | * are, amazingly enough, sign ignorant |
| 449 | */ |
| 450 | res = vl->val.u - (res * vr->val.u); |
| 451 | } |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 452 | break; |
| 453 | case O_PLUS: |
| 454 | case O_PLUSASN: |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 455 | res = t1 + t2; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 456 | break; |
| 457 | case O_MINUS: |
| 458 | case O_MINUSASN: |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 459 | res = t1 - t2; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 460 | break; |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 461 | #ifndef MKSH_LEGACY_MODE |
| 462 | case O_ROL: |
| 463 | case O_ROLASN: |
| 464 | res = (t1 << t2) | (t1 >> (32 - t2)); |
| 465 | break; |
| 466 | case O_ROR: |
| 467 | case O_RORASN: |
| 468 | res = (t1 >> t2) | (t1 << (32 - t2)); |
| 469 | break; |
| 470 | #endif |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 471 | case O_LSHIFT: |
| 472 | case O_LSHIFTASN: |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 473 | res = t1 << t2; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 474 | break; |
| 475 | case O_RSHIFT: |
| 476 | case O_RSHIFTASN: |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 477 | res = es->natural || vl->val.i >= 0 ? |
| 478 | t1 >> t2 : |
| 479 | ~(~t1 >> t2); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 480 | break; |
| 481 | case O_LT: |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 482 | res = cmpop(<); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 483 | break; |
| 484 | case O_LE: |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 485 | res = cmpop(<=); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 486 | break; |
| 487 | case O_GT: |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 488 | res = cmpop(>); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 489 | break; |
| 490 | case O_GE: |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 491 | res = cmpop(>=); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 492 | break; |
| 493 | case O_EQ: |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 494 | res = t1 == t2; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 495 | break; |
| 496 | case O_NE: |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 497 | res = t1 != t2; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 498 | break; |
| 499 | case O_BAND: |
| 500 | case O_BANDASN: |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 501 | res = t1 & t2; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 502 | break; |
| 503 | case O_BXOR: |
| 504 | case O_BXORASN: |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 505 | res = t1 ^ t2; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 506 | break; |
| 507 | case O_BOR: |
| 508 | case O_BORASN: |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 509 | res = t1 | t2; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 510 | break; |
| 511 | case O_LAND: |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 512 | if (!t1) |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 513 | es->noassign++; |
Elliott Hughes | 966dd55 | 2016-12-08 15:56:04 -0800 | [diff] [blame] | 514 | exprtoken(es); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 515 | vr = intvar(es, evalexpr(es, prec - 1)); |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 516 | res = t1 && vr->val.u; |
| 517 | if (!t1) |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 518 | es->noassign--; |
| 519 | break; |
| 520 | case O_LOR: |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 521 | if (t1) |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 522 | es->noassign++; |
Elliott Hughes | 966dd55 | 2016-12-08 15:56:04 -0800 | [diff] [blame] | 523 | exprtoken(es); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 524 | vr = intvar(es, evalexpr(es, prec - 1)); |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 525 | res = t1 || vr->val.u; |
| 526 | if (t1) |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 527 | es->noassign--; |
| 528 | break; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 529 | case O_ASN: |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 530 | case O_COMMA: |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 531 | res = t2; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 532 | break; |
| 533 | } |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 534 | |
| 535 | #undef cmpop |
| 536 | |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 537 | if (IS_ASSIGNOP(op)) { |
Thorsten Glaser | c2dc5de | 2013-02-18 23:02:51 +0000 | [diff] [blame] | 538 | vr->val.u = res; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 539 | if (!es->noassign) { |
| 540 | if (vasn->flag & INTEGER) |
| 541 | setint_v(vasn, vr, es->arith); |
| 542 | else |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 543 | setint(vasn, vr->val.i); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 544 | } |
| 545 | vl = vr; |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 546 | } else |
Thorsten Glaser | c2dc5de | 2013-02-18 23:02:51 +0000 | [diff] [blame] | 547 | vl->val.u = res; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 548 | } |
| 549 | return (vl); |
| 550 | } |
| 551 | |
| 552 | static void |
| 553 | exprtoken(Expr_state *es) |
| 554 | { |
| 555 | const char *cp = es->tokp; |
| 556 | int c; |
| 557 | char *tvar; |
| 558 | |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 559 | /* skip whitespace */ |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 560 | skip_spaces: |
| 561 | while ((c = *cp), ksh_isspace(c)) |
| 562 | ++cp; |
| 563 | if (es->tokp == es->expression && c == '#') { |
| 564 | /* expression begins with # */ |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 565 | /* switch to unsigned */ |
| 566 | es->natural = true; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 567 | ++cp; |
| 568 | goto skip_spaces; |
| 569 | } |
| 570 | es->tokp = cp; |
| 571 | |
| 572 | if (c == '\0') |
| 573 | es->tok = END; |
| 574 | else if (ksh_isalphx(c)) { |
Elliott Hughes | a3c3f96 | 2017-04-12 16:52:30 -0700 | [diff] [blame] | 575 | do { |
| 576 | c = *++cp; |
| 577 | } while (ksh_isalnux(c)); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 578 | if (c == '[') { |
Geremy Condra | 03ebf06 | 2011-10-12 18:17:24 -0700 | [diff] [blame] | 579 | size_t len; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 580 | |
| 581 | len = array_ref_len(cp); |
| 582 | if (len == 0) |
| 583 | evalerr(es, ET_STR, "missing ]"); |
| 584 | cp += len; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 585 | } |
| 586 | if (es->noassign) { |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 587 | es->val = tempvar(""); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 588 | es->val->flag |= EXPRLVALUE; |
| 589 | } else { |
| 590 | strndupx(tvar, es->tokp, cp - es->tokp, ATEMP); |
| 591 | es->val = global(tvar); |
| 592 | afree(tvar, ATEMP); |
| 593 | } |
| 594 | es->tok = VAR; |
| 595 | } else if (c == '1' && cp[1] == '#') { |
| 596 | cp += 2; |
Elliott Hughes | fc0307d | 2016-02-02 15:26:47 -0800 | [diff] [blame] | 597 | if (*cp) |
| 598 | cp += utf_ptradj(cp); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 599 | strndupx(tvar, es->tokp, cp - es->tokp, ATEMP); |
| 600 | goto process_tvar; |
| 601 | #ifndef MKSH_SMALL |
| 602 | } else if (c == '\'') { |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 603 | if (*++cp == '\0') { |
| 604 | es->tok = END; |
| 605 | evalerr(es, ET_UNEXPECTED, NULL); |
| 606 | } |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 607 | cp += utf_ptradj(cp); |
| 608 | if (*cp++ != '\'') |
| 609 | evalerr(es, ET_STR, |
| 610 | "multi-character character constant"); |
| 611 | /* 'x' -> 1#x (x = one multibyte character) */ |
| 612 | c = cp - es->tokp; |
| 613 | tvar = alloc(c + /* NUL */ 1, ATEMP); |
| 614 | tvar[0] = '1'; |
| 615 | tvar[1] = '#'; |
| 616 | memcpy(tvar + 2, es->tokp + 1, c - 2); |
| 617 | tvar[c] = '\0'; |
| 618 | goto process_tvar; |
| 619 | #endif |
| 620 | } else if (ksh_isdigit(c)) { |
| 621 | while (c != '_' && (ksh_isalnux(c) || c == '#')) |
| 622 | c = *cp++; |
| 623 | strndupx(tvar, es->tokp, --cp - es->tokp, ATEMP); |
| 624 | process_tvar: |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 625 | es->val = tempvar(""); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 626 | es->val->flag &= ~INTEGER; |
| 627 | es->val->type = 0; |
| 628 | es->val->val.s = tvar; |
| 629 | if (setint_v(es->val, es->val, es->arith) == NULL) |
| 630 | evalerr(es, ET_BADLIT, tvar); |
| 631 | afree(tvar, ATEMP); |
| 632 | es->tok = LIT; |
| 633 | } else { |
| 634 | int i, n0; |
| 635 | |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 636 | for (i = 0; (n0 = opname[i][0]); i++) |
| 637 | if (c == n0 && strncmp(cp, opname[i], |
| 638 | (size_t)oplen[i]) == 0) { |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 639 | es->tok = (enum token)i; |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 640 | cp += oplen[i]; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 641 | break; |
| 642 | } |
| 643 | if (!n0) |
| 644 | es->tok = BAD; |
| 645 | } |
| 646 | es->tokp = cp; |
| 647 | } |
| 648 | |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 649 | static void |
| 650 | assign_check(Expr_state *es, enum token op, struct tbl *vasn) |
| 651 | { |
Thorsten Glaser | c2dc5de | 2013-02-18 23:02:51 +0000 | [diff] [blame] | 652 | if (es->tok == END || !vasn || |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 653 | (vasn->name[0] == '\0' && !(vasn->flag & EXPRLVALUE))) |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 654 | evalerr(es, ET_LVALUE, opname[(int)op]); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 655 | else if (vasn->flag & RDONLY) |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 656 | evalerr(es, ET_RDONLY, opname[(int)op]); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 657 | } |
| 658 | |
Thorsten Glaser | c2dc5de | 2013-02-18 23:02:51 +0000 | [diff] [blame] | 659 | struct tbl * |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 660 | tempvar(const char *vname) |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 661 | { |
| 662 | struct tbl *vp; |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 663 | size_t vsize; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 664 | |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 665 | vsize = strlen(vname) + 1; |
| 666 | vp = alloc(offsetof(struct tbl, name[0]) + vsize, ATEMP); |
| 667 | memcpy(vp->name, vname, vsize); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 668 | vp->flag = ISSET|INTEGER; |
| 669 | vp->type = 0; |
| 670 | vp->areap = ATEMP; |
| 671 | vp->ua.hval = 0; |
| 672 | vp->val.i = 0; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 673 | return (vp); |
| 674 | } |
| 675 | |
| 676 | /* cast (string) variable to temporary integer variable */ |
| 677 | static struct tbl * |
| 678 | intvar(Expr_state *es, struct tbl *vp) |
| 679 | { |
| 680 | struct tbl *vq; |
| 681 | |
| 682 | /* try to avoid replacing a temp var with another temp var */ |
| 683 | if (vp->name[0] == '\0' && |
| 684 | (vp->flag & (ISSET|INTEGER|EXPRLVALUE)) == (ISSET|INTEGER)) |
| 685 | return (vp); |
| 686 | |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 687 | vq = tempvar(""); |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 688 | if (setint_v(vq, vp, es->arith) == NULL) { |
| 689 | if (vp->flag & EXPRINEVAL) |
| 690 | evalerr(es, ET_RECURSIVE, vp->name); |
| 691 | es->evaling = vp; |
| 692 | vp->flag |= EXPRINEVAL; |
| 693 | v_evaluate(vq, str_val(vp), KSH_UNWIND_ERROR, es->arith); |
| 694 | vp->flag &= ~EXPRINEVAL; |
| 695 | es->evaling = NULL; |
| 696 | } |
| 697 | return (vq); |
| 698 | } |
| 699 | |
| 700 | |
| 701 | /* |
| 702 | * UTF-8 support code: high-level functions |
| 703 | */ |
| 704 | |
| 705 | int |
| 706 | utf_widthadj(const char *src, const char **dst) |
| 707 | { |
| 708 | size_t len; |
| 709 | unsigned int wc; |
| 710 | int width; |
| 711 | |
| 712 | if (!UTFMODE || (len = utf_mbtowc(&wc, src)) == (size_t)-1 || |
| 713 | wc == 0) |
| 714 | len = width = 1; |
| 715 | else if ((width = utf_wcwidth(wc)) < 0) |
| 716 | /* XXX use 2 for x_zotc3 here? */ |
| 717 | width = 1; |
| 718 | |
| 719 | if (dst) |
| 720 | *dst = src + len; |
| 721 | return (width); |
| 722 | } |
| 723 | |
Geremy Condra | 03ebf06 | 2011-10-12 18:17:24 -0700 | [diff] [blame] | 724 | size_t |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 725 | utf_mbswidth(const char *s) |
| 726 | { |
Geremy Condra | 03ebf06 | 2011-10-12 18:17:24 -0700 | [diff] [blame] | 727 | size_t len, width = 0; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 728 | unsigned int wc; |
Geremy Condra | 03ebf06 | 2011-10-12 18:17:24 -0700 | [diff] [blame] | 729 | int cw; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 730 | |
| 731 | if (!UTFMODE) |
| 732 | return (strlen(s)); |
| 733 | |
| 734 | while (*s) |
| 735 | if (((len = utf_mbtowc(&wc, s)) == (size_t)-1) || |
| 736 | ((cw = utf_wcwidth(wc)) == -1)) { |
| 737 | s++; |
| 738 | width += 1; |
| 739 | } else { |
| 740 | s += len; |
| 741 | width += cw; |
| 742 | } |
| 743 | return (width); |
| 744 | } |
| 745 | |
| 746 | const char * |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 747 | utf_skipcols(const char *p, int cols, int *colp) |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 748 | { |
| 749 | int c = 0; |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 750 | const char *q; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 751 | |
| 752 | while (c < cols) { |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 753 | if (!*p) { |
| 754 | /* end of input; special handling for edit.c */ |
| 755 | if (!colp) |
| 756 | return (p + cols - c); |
| 757 | *colp = c; |
| 758 | return (p); |
| 759 | } |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 760 | c += utf_widthadj(p, &p); |
| 761 | } |
Elliott Hughes | 77740fc | 2016-08-12 15:06:53 -0700 | [diff] [blame] | 762 | if (UTFMODE) |
| 763 | while (utf_widthadj(p, &q) == 0) |
| 764 | p = q; |
| 765 | if (colp) |
| 766 | *colp = c; |
Jean-Baptiste Queru | 5155f1c | 2011-06-16 10:05:28 -0700 | [diff] [blame] | 767 | return (p); |
| 768 | } |
| 769 | |
| 770 | size_t |
| 771 | utf_ptradj(const char *src) |
| 772 | { |
| 773 | register size_t n; |
| 774 | |
| 775 | if (!UTFMODE || |
| 776 | *(const unsigned char *)(src) < 0xC2 || |
| 777 | (n = utf_mbtowc(NULL, src)) == (size_t)-1) |
| 778 | n = 1; |
| 779 | return (n); |
| 780 | } |
| 781 | |
| 782 | /* |
| 783 | * UTF-8 support code: low-level functions |
| 784 | */ |
| 785 | |
| 786 | /* CESU-8 multibyte and wide character conversion crafted for mksh */ |
| 787 | |
| 788 | size_t |
| 789 | utf_mbtowc(unsigned int *dst, const char *src) |
| 790 | { |
| 791 | const unsigned char *s = (const unsigned char *)src; |
| 792 | unsigned int c, wc; |
| 793 | |
| 794 | if ((wc = *s++) < 0x80) { |
| 795 | out: |
| 796 | if (dst != NULL) |
| 797 | *dst = wc; |
| 798 | return (wc ? ((const char *)s - src) : 0); |
| 799 | } |
| 800 | if (wc < 0xC2 || wc >= 0xF0) |
| 801 | /* < 0xC0: spurious second byte */ |
| 802 | /* < 0xC2: non-minimalistic mapping error in 2-byte seqs */ |
| 803 | /* > 0xEF: beyond BMP */ |
| 804 | goto ilseq; |
| 805 | |
| 806 | if (wc < 0xE0) { |
| 807 | wc = (wc & 0x1F) << 6; |
| 808 | if (((c = *s++) & 0xC0) != 0x80) |
| 809 | goto ilseq; |
| 810 | wc |= c & 0x3F; |
| 811 | goto out; |
| 812 | } |
| 813 | |
| 814 | wc = (wc & 0x0F) << 12; |
| 815 | |
| 816 | if (((c = *s++) & 0xC0) != 0x80) |
| 817 | goto ilseq; |
| 818 | wc |= (c & 0x3F) << 6; |
| 819 | |
| 820 | if (((c = *s++) & 0xC0) != 0x80) |
| 821 | goto ilseq; |
| 822 | wc |= c & 0x3F; |
| 823 | |
| 824 | /* Check for non-minimalistic mapping error in 3-byte seqs */ |
| 825 | if (wc >= 0x0800 && wc <= 0xFFFD) |
| 826 | goto out; |
| 827 | ilseq: |
| 828 | return ((size_t)(-1)); |
| 829 | } |
| 830 | |
| 831 | size_t |
| 832 | utf_wctomb(char *dst, unsigned int wc) |
| 833 | { |
| 834 | unsigned char *d; |
| 835 | |
| 836 | if (wc < 0x80) { |
| 837 | *dst = wc; |
| 838 | return (1); |
| 839 | } |
| 840 | |
| 841 | d = (unsigned char *)dst; |
| 842 | if (wc < 0x0800) |
| 843 | *d++ = (wc >> 6) | 0xC0; |
| 844 | else { |
| 845 | *d++ = ((wc = wc > 0xFFFD ? 0xFFFD : wc) >> 12) | 0xE0; |
| 846 | *d++ = ((wc >> 6) & 0x3F) | 0x80; |
| 847 | } |
| 848 | *d++ = (wc & 0x3F) | 0x80; |
| 849 | return ((char *)d - dst); |
| 850 | } |
| 851 | |
Geremy Condra | 03ebf06 | 2011-10-12 18:17:24 -0700 | [diff] [blame] | 852 | /* |
| 853 | * Wrapper around access(2) because it says root can execute everything |
| 854 | * on some operating systems. Does not set errno, no user needs it. Use |
| 855 | * this iff mode can have the X_OK bit set, access otherwise. |
| 856 | */ |
| 857 | int |
| 858 | ksh_access(const char *fn, int mode) |
| 859 | { |
Elliott Hughes | a3c3f96 | 2017-04-12 16:52:30 -0700 | [diff] [blame] | 860 | #ifdef __OS2__ |
| 861 | return (access_ex(access, fn, mode)); |
| 862 | #else |
Geremy Condra | 03ebf06 | 2011-10-12 18:17:24 -0700 | [diff] [blame] | 863 | int rv; |
| 864 | struct stat sb; |
| 865 | |
| 866 | if ((rv = access(fn, mode)) == 0 && kshuid == 0 && (mode & X_OK) && |
| 867 | (rv = stat(fn, &sb)) == 0 && !S_ISDIR(sb.st_mode) && |
| 868 | (sb.st_mode & (S_IXUSR|S_IXGRP|S_IXOTH)) == 0) |
| 869 | rv = -1; |
| 870 | |
| 871 | return (rv); |
Elliott Hughes | a3c3f96 | 2017-04-12 16:52:30 -0700 | [diff] [blame] | 872 | #endif |
Geremy Condra | 03ebf06 | 2011-10-12 18:17:24 -0700 | [diff] [blame] | 873 | } |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 874 | |
Elliott Hughes | fc0307d | 2016-02-02 15:26:47 -0800 | [diff] [blame] | 875 | #ifndef MIRBSD_BOOTFLOPPY |
Elliott Hughes | 966dd55 | 2016-12-08 15:56:04 -0800 | [diff] [blame] | 876 | /* From: X11/xc/programs/xterm/wcwidth.c,v 1.9 */ |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 877 | |
| 878 | struct mb_ucsrange { |
| 879 | unsigned short beg; |
| 880 | unsigned short end; |
| 881 | }; |
| 882 | |
| 883 | static int mb_ucsbsearch(const struct mb_ucsrange arr[], size_t elems, |
Elliott Hughes | 737fdce | 2014-08-07 12:59:26 -0700 | [diff] [blame] | 884 | unsigned int val) MKSH_A_PURE; |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 885 | |
| 886 | /* |
Elliott Hughes | 966dd55 | 2016-12-08 15:56:04 -0800 | [diff] [blame] | 887 | * Generated from the Unicode Character Database, Version 9.0.0, by |
| 888 | * MirOS: contrib/code/Snippets/eawparse,v 1.3 2014/11/16 12:16:24 tg Exp $ |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 889 | */ |
| 890 | |
| 891 | static const struct mb_ucsrange mb_ucs_combining[] = { |
| 892 | { 0x0300, 0x036F }, |
| 893 | { 0x0483, 0x0489 }, |
| 894 | { 0x0591, 0x05BD }, |
| 895 | { 0x05BF, 0x05BF }, |
| 896 | { 0x05C1, 0x05C2 }, |
| 897 | { 0x05C4, 0x05C5 }, |
| 898 | { 0x05C7, 0x05C7 }, |
Elliott Hughes | 737fdce | 2014-08-07 12:59:26 -0700 | [diff] [blame] | 899 | { 0x0600, 0x0605 }, |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 900 | { 0x0610, 0x061A }, |
Elliott Hughes | 737fdce | 2014-08-07 12:59:26 -0700 | [diff] [blame] | 901 | { 0x061C, 0x061C }, |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 902 | { 0x064B, 0x065F }, |
| 903 | { 0x0670, 0x0670 }, |
| 904 | { 0x06D6, 0x06DD }, |
| 905 | { 0x06DF, 0x06E4 }, |
| 906 | { 0x06E7, 0x06E8 }, |
| 907 | { 0x06EA, 0x06ED }, |
| 908 | { 0x070F, 0x070F }, |
| 909 | { 0x0711, 0x0711 }, |
| 910 | { 0x0730, 0x074A }, |
| 911 | { 0x07A6, 0x07B0 }, |
| 912 | { 0x07EB, 0x07F3 }, |
| 913 | { 0x0816, 0x0819 }, |
| 914 | { 0x081B, 0x0823 }, |
| 915 | { 0x0825, 0x0827 }, |
| 916 | { 0x0829, 0x082D }, |
| 917 | { 0x0859, 0x085B }, |
Elliott Hughes | 966dd55 | 2016-12-08 15:56:04 -0800 | [diff] [blame] | 918 | { 0x08D4, 0x0902 }, |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 919 | { 0x093A, 0x093A }, |
| 920 | { 0x093C, 0x093C }, |
| 921 | { 0x0941, 0x0948 }, |
| 922 | { 0x094D, 0x094D }, |
| 923 | { 0x0951, 0x0957 }, |
| 924 | { 0x0962, 0x0963 }, |
| 925 | { 0x0981, 0x0981 }, |
| 926 | { 0x09BC, 0x09BC }, |
| 927 | { 0x09C1, 0x09C4 }, |
| 928 | { 0x09CD, 0x09CD }, |
| 929 | { 0x09E2, 0x09E3 }, |
| 930 | { 0x0A01, 0x0A02 }, |
| 931 | { 0x0A3C, 0x0A3C }, |
| 932 | { 0x0A41, 0x0A42 }, |
| 933 | { 0x0A47, 0x0A48 }, |
| 934 | { 0x0A4B, 0x0A4D }, |
| 935 | { 0x0A51, 0x0A51 }, |
| 936 | { 0x0A70, 0x0A71 }, |
| 937 | { 0x0A75, 0x0A75 }, |
| 938 | { 0x0A81, 0x0A82 }, |
| 939 | { 0x0ABC, 0x0ABC }, |
| 940 | { 0x0AC1, 0x0AC5 }, |
| 941 | { 0x0AC7, 0x0AC8 }, |
| 942 | { 0x0ACD, 0x0ACD }, |
| 943 | { 0x0AE2, 0x0AE3 }, |
| 944 | { 0x0B01, 0x0B01 }, |
| 945 | { 0x0B3C, 0x0B3C }, |
| 946 | { 0x0B3F, 0x0B3F }, |
| 947 | { 0x0B41, 0x0B44 }, |
| 948 | { 0x0B4D, 0x0B4D }, |
| 949 | { 0x0B56, 0x0B56 }, |
| 950 | { 0x0B62, 0x0B63 }, |
| 951 | { 0x0B82, 0x0B82 }, |
| 952 | { 0x0BC0, 0x0BC0 }, |
| 953 | { 0x0BCD, 0x0BCD }, |
Elliott Hughes | 737fdce | 2014-08-07 12:59:26 -0700 | [diff] [blame] | 954 | { 0x0C00, 0x0C00 }, |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 955 | { 0x0C3E, 0x0C40 }, |
| 956 | { 0x0C46, 0x0C48 }, |
| 957 | { 0x0C4A, 0x0C4D }, |
| 958 | { 0x0C55, 0x0C56 }, |
| 959 | { 0x0C62, 0x0C63 }, |
Elliott Hughes | 737fdce | 2014-08-07 12:59:26 -0700 | [diff] [blame] | 960 | { 0x0C81, 0x0C81 }, |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 961 | { 0x0CBC, 0x0CBC }, |
| 962 | { 0x0CBF, 0x0CBF }, |
| 963 | { 0x0CC6, 0x0CC6 }, |
| 964 | { 0x0CCC, 0x0CCD }, |
| 965 | { 0x0CE2, 0x0CE3 }, |
Elliott Hughes | 737fdce | 2014-08-07 12:59:26 -0700 | [diff] [blame] | 966 | { 0x0D01, 0x0D01 }, |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 967 | { 0x0D41, 0x0D44 }, |
| 968 | { 0x0D4D, 0x0D4D }, |
| 969 | { 0x0D62, 0x0D63 }, |
| 970 | { 0x0DCA, 0x0DCA }, |
| 971 | { 0x0DD2, 0x0DD4 }, |
| 972 | { 0x0DD6, 0x0DD6 }, |
| 973 | { 0x0E31, 0x0E31 }, |
| 974 | { 0x0E34, 0x0E3A }, |
| 975 | { 0x0E47, 0x0E4E }, |
| 976 | { 0x0EB1, 0x0EB1 }, |
| 977 | { 0x0EB4, 0x0EB9 }, |
| 978 | { 0x0EBB, 0x0EBC }, |
| 979 | { 0x0EC8, 0x0ECD }, |
| 980 | { 0x0F18, 0x0F19 }, |
| 981 | { 0x0F35, 0x0F35 }, |
| 982 | { 0x0F37, 0x0F37 }, |
| 983 | { 0x0F39, 0x0F39 }, |
| 984 | { 0x0F71, 0x0F7E }, |
| 985 | { 0x0F80, 0x0F84 }, |
| 986 | { 0x0F86, 0x0F87 }, |
| 987 | { 0x0F8D, 0x0F97 }, |
| 988 | { 0x0F99, 0x0FBC }, |
| 989 | { 0x0FC6, 0x0FC6 }, |
| 990 | { 0x102D, 0x1030 }, |
| 991 | { 0x1032, 0x1037 }, |
| 992 | { 0x1039, 0x103A }, |
| 993 | { 0x103D, 0x103E }, |
| 994 | { 0x1058, 0x1059 }, |
| 995 | { 0x105E, 0x1060 }, |
| 996 | { 0x1071, 0x1074 }, |
| 997 | { 0x1082, 0x1082 }, |
| 998 | { 0x1085, 0x1086 }, |
| 999 | { 0x108D, 0x108D }, |
| 1000 | { 0x109D, 0x109D }, |
| 1001 | { 0x1160, 0x11FF }, |
| 1002 | { 0x135D, 0x135F }, |
| 1003 | { 0x1712, 0x1714 }, |
| 1004 | { 0x1732, 0x1734 }, |
| 1005 | { 0x1752, 0x1753 }, |
| 1006 | { 0x1772, 0x1773 }, |
| 1007 | { 0x17B4, 0x17B5 }, |
| 1008 | { 0x17B7, 0x17BD }, |
| 1009 | { 0x17C6, 0x17C6 }, |
| 1010 | { 0x17C9, 0x17D3 }, |
| 1011 | { 0x17DD, 0x17DD }, |
Elliott Hughes | 737fdce | 2014-08-07 12:59:26 -0700 | [diff] [blame] | 1012 | { 0x180B, 0x180E }, |
Elliott Hughes | 966dd55 | 2016-12-08 15:56:04 -0800 | [diff] [blame] | 1013 | { 0x1885, 0x1886 }, |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 1014 | { 0x18A9, 0x18A9 }, |
| 1015 | { 0x1920, 0x1922 }, |
| 1016 | { 0x1927, 0x1928 }, |
| 1017 | { 0x1932, 0x1932 }, |
| 1018 | { 0x1939, 0x193B }, |
| 1019 | { 0x1A17, 0x1A18 }, |
Elliott Hughes | 737fdce | 2014-08-07 12:59:26 -0700 | [diff] [blame] | 1020 | { 0x1A1B, 0x1A1B }, |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 1021 | { 0x1A56, 0x1A56 }, |
| 1022 | { 0x1A58, 0x1A5E }, |
| 1023 | { 0x1A60, 0x1A60 }, |
| 1024 | { 0x1A62, 0x1A62 }, |
| 1025 | { 0x1A65, 0x1A6C }, |
| 1026 | { 0x1A73, 0x1A7C }, |
| 1027 | { 0x1A7F, 0x1A7F }, |
Elliott Hughes | 737fdce | 2014-08-07 12:59:26 -0700 | [diff] [blame] | 1028 | { 0x1AB0, 0x1ABE }, |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 1029 | { 0x1B00, 0x1B03 }, |
| 1030 | { 0x1B34, 0x1B34 }, |
| 1031 | { 0x1B36, 0x1B3A }, |
| 1032 | { 0x1B3C, 0x1B3C }, |
| 1033 | { 0x1B42, 0x1B42 }, |
| 1034 | { 0x1B6B, 0x1B73 }, |
| 1035 | { 0x1B80, 0x1B81 }, |
| 1036 | { 0x1BA2, 0x1BA5 }, |
| 1037 | { 0x1BA8, 0x1BA9 }, |
Elliott Hughes | 737fdce | 2014-08-07 12:59:26 -0700 | [diff] [blame] | 1038 | { 0x1BAB, 0x1BAD }, |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 1039 | { 0x1BE6, 0x1BE6 }, |
| 1040 | { 0x1BE8, 0x1BE9 }, |
| 1041 | { 0x1BED, 0x1BED }, |
| 1042 | { 0x1BEF, 0x1BF1 }, |
| 1043 | { 0x1C2C, 0x1C33 }, |
| 1044 | { 0x1C36, 0x1C37 }, |
| 1045 | { 0x1CD0, 0x1CD2 }, |
| 1046 | { 0x1CD4, 0x1CE0 }, |
| 1047 | { 0x1CE2, 0x1CE8 }, |
| 1048 | { 0x1CED, 0x1CED }, |
| 1049 | { 0x1CF4, 0x1CF4 }, |
Elliott Hughes | 737fdce | 2014-08-07 12:59:26 -0700 | [diff] [blame] | 1050 | { 0x1CF8, 0x1CF9 }, |
| 1051 | { 0x1DC0, 0x1DF5 }, |
Elliott Hughes | 966dd55 | 2016-12-08 15:56:04 -0800 | [diff] [blame] | 1052 | { 0x1DFB, 0x1DFF }, |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 1053 | { 0x200B, 0x200F }, |
| 1054 | { 0x202A, 0x202E }, |
| 1055 | { 0x2060, 0x2064 }, |
Elliott Hughes | 737fdce | 2014-08-07 12:59:26 -0700 | [diff] [blame] | 1056 | { 0x2066, 0x206F }, |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 1057 | { 0x20D0, 0x20F0 }, |
| 1058 | { 0x2CEF, 0x2CF1 }, |
| 1059 | { 0x2D7F, 0x2D7F }, |
| 1060 | { 0x2DE0, 0x2DFF }, |
| 1061 | { 0x302A, 0x302D }, |
| 1062 | { 0x3099, 0x309A }, |
| 1063 | { 0xA66F, 0xA672 }, |
| 1064 | { 0xA674, 0xA67D }, |
Elliott Hughes | 966dd55 | 2016-12-08 15:56:04 -0800 | [diff] [blame] | 1065 | { 0xA69E, 0xA69F }, |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 1066 | { 0xA6F0, 0xA6F1 }, |
| 1067 | { 0xA802, 0xA802 }, |
| 1068 | { 0xA806, 0xA806 }, |
| 1069 | { 0xA80B, 0xA80B }, |
| 1070 | { 0xA825, 0xA826 }, |
Elliott Hughes | 966dd55 | 2016-12-08 15:56:04 -0800 | [diff] [blame] | 1071 | { 0xA8C4, 0xA8C5 }, |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 1072 | { 0xA8E0, 0xA8F1 }, |
| 1073 | { 0xA926, 0xA92D }, |
| 1074 | { 0xA947, 0xA951 }, |
| 1075 | { 0xA980, 0xA982 }, |
| 1076 | { 0xA9B3, 0xA9B3 }, |
| 1077 | { 0xA9B6, 0xA9B9 }, |
| 1078 | { 0xA9BC, 0xA9BC }, |
Elliott Hughes | 737fdce | 2014-08-07 12:59:26 -0700 | [diff] [blame] | 1079 | { 0xA9E5, 0xA9E5 }, |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 1080 | { 0xAA29, 0xAA2E }, |
| 1081 | { 0xAA31, 0xAA32 }, |
| 1082 | { 0xAA35, 0xAA36 }, |
| 1083 | { 0xAA43, 0xAA43 }, |
| 1084 | { 0xAA4C, 0xAA4C }, |
Elliott Hughes | 737fdce | 2014-08-07 12:59:26 -0700 | [diff] [blame] | 1085 | { 0xAA7C, 0xAA7C }, |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 1086 | { 0xAAB0, 0xAAB0 }, |
| 1087 | { 0xAAB2, 0xAAB4 }, |
| 1088 | { 0xAAB7, 0xAAB8 }, |
| 1089 | { 0xAABE, 0xAABF }, |
| 1090 | { 0xAAC1, 0xAAC1 }, |
| 1091 | { 0xAAEC, 0xAAED }, |
| 1092 | { 0xAAF6, 0xAAF6 }, |
| 1093 | { 0xABE5, 0xABE5 }, |
| 1094 | { 0xABE8, 0xABE8 }, |
| 1095 | { 0xABED, 0xABED }, |
| 1096 | { 0xFB1E, 0xFB1E }, |
| 1097 | { 0xFE00, 0xFE0F }, |
Elliott Hughes | 966dd55 | 2016-12-08 15:56:04 -0800 | [diff] [blame] | 1098 | { 0xFE20, 0xFE2F }, |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 1099 | { 0xFEFF, 0xFEFF }, |
| 1100 | { 0xFFF9, 0xFFFB } |
| 1101 | }; |
| 1102 | |
| 1103 | static const struct mb_ucsrange mb_ucs_fullwidth[] = { |
| 1104 | { 0x1100, 0x115F }, |
Elliott Hughes | 966dd55 | 2016-12-08 15:56:04 -0800 | [diff] [blame] | 1105 | { 0x231A, 0x231B }, |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 1106 | { 0x2329, 0x232A }, |
Elliott Hughes | 966dd55 | 2016-12-08 15:56:04 -0800 | [diff] [blame] | 1107 | { 0x23E9, 0x23EC }, |
| 1108 | { 0x23F0, 0x23F0 }, |
| 1109 | { 0x23F3, 0x23F3 }, |
| 1110 | { 0x25FD, 0x25FE }, |
| 1111 | { 0x2614, 0x2615 }, |
| 1112 | { 0x2648, 0x2653 }, |
| 1113 | { 0x267F, 0x267F }, |
| 1114 | { 0x2693, 0x2693 }, |
| 1115 | { 0x26A1, 0x26A1 }, |
| 1116 | { 0x26AA, 0x26AB }, |
| 1117 | { 0x26BD, 0x26BE }, |
| 1118 | { 0x26C4, 0x26C5 }, |
| 1119 | { 0x26CE, 0x26CE }, |
| 1120 | { 0x26D4, 0x26D4 }, |
| 1121 | { 0x26EA, 0x26EA }, |
| 1122 | { 0x26F2, 0x26F3 }, |
| 1123 | { 0x26F5, 0x26F5 }, |
| 1124 | { 0x26FA, 0x26FA }, |
| 1125 | { 0x26FD, 0x26FD }, |
| 1126 | { 0x2705, 0x2705 }, |
| 1127 | { 0x270A, 0x270B }, |
| 1128 | { 0x2728, 0x2728 }, |
| 1129 | { 0x274C, 0x274C }, |
| 1130 | { 0x274E, 0x274E }, |
| 1131 | { 0x2753, 0x2755 }, |
| 1132 | { 0x2757, 0x2757 }, |
| 1133 | { 0x2795, 0x2797 }, |
| 1134 | { 0x27B0, 0x27B0 }, |
| 1135 | { 0x27BF, 0x27BF }, |
| 1136 | { 0x2B1B, 0x2B1C }, |
| 1137 | { 0x2B50, 0x2B50 }, |
| 1138 | { 0x2B55, 0x2B55 }, |
Thorsten Glaser | 811a575 | 2013-07-25 14:24:45 +0000 | [diff] [blame] | 1139 | { 0x2E80, 0x303E }, |
| 1140 | { 0x3040, 0xA4CF }, |
| 1141 | { 0xA960, 0xA97F }, |
| 1142 | { 0xAC00, 0xD7A3 }, |
| 1143 | { 0xF900, 0xFAFF }, |
| 1144 | { 0xFE10, 0xFE19 }, |
| 1145 | { 0xFE30, 0xFE6F }, |
| 1146 | { 0xFF00, 0xFF60 }, |
| 1147 | { 0xFFE0, 0xFFE6 } |
| 1148 | }; |
| 1149 | |
| 1150 | /* simple binary search in ranges, with bounds optimisation */ |
| 1151 | static int |
| 1152 | mb_ucsbsearch(const struct mb_ucsrange arr[], size_t elems, unsigned int val) |
| 1153 | { |
| 1154 | size_t min = 0, mid, max = elems; |
| 1155 | |
| 1156 | if (val < arr[min].beg || val > arr[max - 1].end) |
| 1157 | return (0); |
| 1158 | |
| 1159 | while (min < max) { |
| 1160 | mid = (min + max) / 2; |
| 1161 | |
| 1162 | if (val < arr[mid].beg) |
| 1163 | max = mid; |
| 1164 | else if (val > arr[mid].end) |
| 1165 | min = mid + 1; |
| 1166 | else |
| 1167 | return (1); |
| 1168 | } |
| 1169 | return (0); |
| 1170 | } |
| 1171 | |
| 1172 | /* Unix column width of a wide character (Unicode code point, really) */ |
| 1173 | int |
| 1174 | utf_wcwidth(unsigned int wc) |
| 1175 | { |
| 1176 | /* except NUL, C0/C1 control characters and DEL yield -1 */ |
| 1177 | if (wc < 0x20 || (wc >= 0x7F && wc < 0xA0)) |
| 1178 | return (wc ? -1 : 0); |
| 1179 | |
| 1180 | /* combining characters use 0 screen columns */ |
| 1181 | if (mb_ucsbsearch(mb_ucs_combining, NELEM(mb_ucs_combining), wc)) |
| 1182 | return (0); |
| 1183 | |
| 1184 | /* all others use 1 or 2 screen columns */ |
| 1185 | if (mb_ucsbsearch(mb_ucs_fullwidth, NELEM(mb_ucs_fullwidth), wc)) |
| 1186 | return (2); |
| 1187 | return (1); |
| 1188 | } |
Elliott Hughes | fc0307d | 2016-02-02 15:26:47 -0800 | [diff] [blame] | 1189 | #endif |