Changeset 1707
- Timestamp:
- Sep 2, 2008, 6:06:42 AM (17 years ago)
- Location:
- trunk/src/kmk
- Files:
-
- 6 edited
Legend:
- Unmodified
- Added
- Removed
-
TabularUnified trunk/src/kmk/Makefile.am ¶
r1705 r1707 61 61 kmkbuiltin/cp_utils.c \ 62 62 kmkbuiltin/echo.c \ 63 kmkbuiltin/expr.c \ 63 64 kmkbuiltin/install.c \ 64 65 kmkbuiltin/kDepIDB.c \ -
TabularUnified trunk/src/kmk/Makefile.kmk ¶
r1705 r1707 213 213 kmkbuiltin/cp_utils.c \ 214 214 kmkbuiltin/echo.c \ 215 kmkbuiltin/expr.c \ 215 216 kmkbuiltin/install.c \ 216 217 kmkbuiltin/kDepIDB.c \ … … 242 243 kmk_cmp \ 243 244 kmk_echo \ 245 kmk_expr \ 244 246 kmk_md5sum \ 245 247 kmk_mkdir \ … … 287 289 kmk_echo_SOURCES = \ 288 290 kmkbuiltin/echo.c 291 292 kmk_expr_TEMPLATE = BIN-KMK 293 kmk_expr_DEFS = kmk_builtin_expr=main 294 kmk_expr_SOURCES = \ 295 kmkbuiltin/expr.c 289 296 290 297 kmk_install_TEMPLATE = BIN-KMK -
TabularUnified trunk/src/kmk/kmkbuiltin.c ¶
r1705 r1707 212 212 else if (!strcmp(pszCmd, "cp")) 213 213 rc = kmk_builtin_cp(argc, argv, environ); 214 else if (!strcmp(pszCmd, "expr")) 215 rc = kmk_builtin_expr(argc, argv, environ); 214 216 else if (!strcmp(pszCmd, "ln")) 215 217 rc = kmk_builtin_ln(argc, argv, environ); -
TabularUnified trunk/src/kmk/kmkbuiltin.h ¶
r1705 r1707 42 42 extern int kmk_builtin_cmp(int argc, char **argv, char **envp); 43 43 extern int kmk_builtin_echo(int argc, char **argv, char **envp); 44 extern int kmk_builtin_expr(int argc, char **argv, char **envp); 44 45 extern int kmk_builtin_install(int argc, char **argv, char **envp); 45 46 extern int kmk_builtin_ln(int argc, char **argv, char **envp); -
TabularUnified trunk/src/kmk/kmkbuiltin/expr.c ¶
r1703 r1707 12 12 #include <locale.h> 13 13 #include <ctype.h> 14 #ifdef KMK_WITH_REGEX 14 15 #include <regex.h> 15 #include <err.h> 16 17 struct val *make_int(int); 18 struct val *make_str(char *); 19 void free_value(struct val *); 20 int is_integer(struct val *, int *); 21 int to_integer(struct val *); 22 void to_string(struct val *); 23 int is_zero_or_null(struct val *); 24 void nexttoken(int); 25 __dead void error(void); 26 struct val *eval6(void); 27 struct val *eval5(void); 28 struct val *eval4(void); 29 struct val *eval3(void); 30 struct val *eval2(void); 31 struct val *eval1(void); 32 struct val *eval0(void); 16 #endif 17 #include <setjmp.h> 18 #include <assert.h> 19 #include "err.h" 20 #include "getopt.h" 21 #include "kmkbuiltin.h" 22 23 static struct val *make_int(int); 24 static struct val *make_str(char *); 25 static void free_value(struct val *); 26 static int is_integer(struct val *, int *); 27 static int to_integer(struct val *); 28 static void to_string(struct val *); 29 static int is_zero_or_null(struct val *); 30 static void nexttoken(int); 31 static __dead void error(void); 32 static struct val *eval6(void); 33 static struct val *eval5(void); 34 static struct val *eval4(void); 35 static struct val *eval3(void); 36 static struct val *eval2(void); 37 static struct val *eval1(void); 38 static struct val *eval0(void); 33 39 34 40 enum token { … … 49 55 }; 50 56 51 enum token token; 52 struct val *tokval; 53 char **av; 54 55 struct val * 57 static enum token token; 58 static struct val *tokval; 59 static char **av; 60 static jmp_buf g_expr_jmp; 61 static void **recorded_allocations; 62 static int num_recorded_allocations; 63 64 65 static void expr_mem_record_alloc(void *ptr) 66 { 67 if (!(num_recorded_allocations & 31)) { 68 void *newtab = realloc(recorded_allocations, (num_recorded_allocations + 33) * sizeof(void *)); 69 if (!newtab) 70 longjmp(g_expr_jmp, err(3, NULL)); 71 recorded_allocations = (void **)newtab; 72 } 73 recorded_allocations[num_recorded_allocations++] = ptr; 74 } 75 76 77 static void expr_mem_record_free(void *ptr) 78 { 79 int i = num_recorded_allocations; 80 while (i-- > 0) 81 if (recorded_allocations[i] == ptr) { 82 num_recorded_allocations--; 83 recorded_allocations[i] = recorded_allocations[num_recorded_allocations]; 84 return; 85 } 86 assert(i >= 0); 87 } 88 89 static void expr_mem_init(void) 90 { 91 num_recorded_allocations = 0; 92 recorded_allocations = NULL; 93 } 94 95 static void expr_mem_cleanup(void) 96 { 97 if (recorded_allocations) { 98 while (num_recorded_allocations-- > 0) 99 free(recorded_allocations[num_recorded_allocations]); 100 free(recorded_allocations); 101 recorded_allocations = NULL; 102 } 103 } 104 105 106 static struct val * 56 107 make_int(int i) 57 108 { … … 59 110 60 111 vp = (struct val *) malloc(sizeof(*vp)); 61 if (vp == NULL) {62 err(3, NULL);63 }112 if (vp == NULL) 113 longjmp(g_expr_jmp, err(3, NULL)); 114 expr_mem_record_alloc(vp); 64 115 vp->type = integer; 65 116 vp->u.i = i; … … 68 119 69 120 70 st ruct val *121 static struct val * 71 122 make_str(char *s) 72 123 { … … 74 125 75 126 vp = (struct val *) malloc(sizeof(*vp)); 76 if (vp == NULL || ((vp->u.s = strdup(s)) == NULL)) { 77 err(3, NULL); 78 } 127 if (vp == NULL || ((vp->u.s = strdup(s)) == NULL)) 128 longjmp(g_expr_jmp, err(3, NULL)); 129 expr_mem_record_alloc(vp->u.s); 130 expr_mem_record_alloc(vp); 79 131 vp->type = string; 80 132 return vp; … … 82 134 83 135 84 void136 static void 85 137 free_value(struct val *vp) 86 138 { 87 if (vp->type == string) 139 if (vp->type == string) { 140 expr_mem_record_free(vp->u.s); 88 141 free(vp->u.s); 142 } 89 143 free(vp); 144 expr_mem_record_free(vp); 90 145 } 91 146 92 147 93 148 /* determine if vp is an integer; if so, return it's value in *r */ 94 int149 static int 95 150 is_integer(struct val *vp, int *r) 96 151 { … … 134 189 135 190 /* coerce to vp to an integer */ 136 int191 static int 137 192 to_integer(struct val *vp) 138 193 { … … 143 198 144 199 if (is_integer(vp, &r)) { 200 expr_mem_record_free(vp->u.s); 145 201 free(vp->u.s); 146 202 vp->u.i = r; … … 154 210 155 211 /* coerce to vp to an string */ 156 void212 static void 157 213 to_string(struct val *vp) 158 214 { … … 163 219 164 220 if (asprintf(&tmp, "%d", vp->u.i) == -1) 165 err(3, NULL); 221 longjmp(g_expr_jmp, err(3, NULL)); 222 expr_mem_record_alloc(tmp); 166 223 167 224 vp->type = string; … … 169 226 } 170 227 171 int228 static int 172 229 is_zero_or_null(struct val *vp) 173 230 { … … 180 237 } 181 238 182 void239 static void 183 240 nexttoken(int pat) 184 241 { … … 220 277 } 221 278 222 __deadvoid279 static void 223 280 error(void) 224 281 { 225 errx(2, "syntax error");282 longjmp(g_expr_jmp, errx(2, "syntax error")); 226 283 /* NOTREACHED */ 227 284 } 228 285 229 st ruct val *286 static struct val * 230 287 eval6(void) 231 288 { … … 253 310 254 311 /* Parse and evaluate match (regex) expressions */ 255 st ruct val *312 static struct val * 256 313 eval5(void) 257 314 { 315 #ifdef KMK_WITH_REGEX 258 316 regex_t rp; 259 317 regmatch_t rm[2]; … … 275 333 if ((eval = regcomp(&rp, r->u.s, 0)) != 0) { 276 334 regerror(eval, &rp, errbuf, sizeof(errbuf)); 277 errx(2, "%s", errbuf);335 longjmp(g_expr_jmp, errx(2, "%s", errbuf)); 278 336 } 279 337 … … 305 363 306 364 return l; 365 #else 366 longjmp(g_expr_jmp, errx(2, "regex not supported, sorry.")); 367 #endif 307 368 } 308 369 309 370 /* Parse and evaluate multiplication and division expressions */ 310 st ruct val *371 static struct val * 311 372 eval4(void) 312 373 { … … 320 381 321 382 if (!to_integer(l) || !to_integer(r)) { 322 errx(2, "non-numeric argument");383 longjmp(g_expr_jmp, errx(2, "non-numeric argument")); 323 384 } 324 385 … … 327 388 } else { 328 389 if (r->u.i == 0) { 329 errx(2, "division by zero");390 longjmp(g_expr_jmp, errx(2, "division by zero")); 330 391 } 331 392 if (op == DIV) { … … 343 404 344 405 /* Parse and evaluate addition and subtraction expressions */ 345 st ruct val *406 static struct val * 346 407 eval3(void) 347 408 { … … 355 416 356 417 if (!to_integer(l) || !to_integer(r)) { 357 errx(2, "non-numeric argument");418 longjmp(g_expr_jmp, errx(2, "non-numeric argument")); 358 419 } 359 420 … … 371 432 372 433 /* Parse and evaluate comparison expressions */ 373 st ruct val *434 static struct val * 374 435 eval2(void) 375 436 { … … 444 505 445 506 /* Parse and evaluate & expressions */ 446 st ruct val *507 static struct val * 447 508 eval1(void) 448 509 { … … 467 528 468 529 /* Parse and evaluate | expressions */ 469 st ruct val *530 static struct val * 470 531 eval0(void) 471 532 { … … 490 551 491 552 int 492 main(int argc, char *argv[])553 kmk_builtin_expr(int argc, char *argv[], char **envp) 493 554 { 494 555 struct val *vp; 495 556 int rval; 557 558 /* re-init globals */ 559 token = 0; 560 tokval = 0; 561 av = 0; 562 expr_mem_init(); 563 564 #ifdef kmk_builtin_expr /* kmk already does this. */ 496 565 (void) setlocale(LC_ALL, ""); 566 #endif 497 567 498 568 if (argc > 1 && !strcmp(argv[1], "--")) … … 501 571 av = argv + 1; 502 572 503 nexttoken(0); 504 vp = eval0(); 505 506 if (token != EOI) { 507 error(); 508 /* NOTREACHED */ 509 } 510 511 if (vp->type == integer) 512 printf("%d\n", vp->u.i); 513 else 514 printf("%s\n", vp->u.s); 515 516 exit(is_zero_or_null(vp)); 517 } 573 rval = setjmp(g_expr_jmp); 574 if (!rval) { 575 nexttoken(0); 576 vp = eval0(); 577 578 if (token != EOI) { 579 error(); 580 /* NOTREACHED */ 581 } 582 583 if (vp->type == integer) 584 printf("%d\n", vp->u.i); 585 else 586 printf("%s\n", vp->u.s); 587 588 rval = is_zero_or_null(vp); 589 } 590 /* else: longjmp */ 591 592 /* cleanup */ 593 expr_mem_cleanup(); 594 return rval; 595 } -
TabularUnified trunk/src/kmk/variable.c ¶
r1706 r1707 1311 1311 #ifdef CONFIG_WITH_KMK_BUILTIN 1312 1312 /* The supported kMk Builtin commands. */ 1313 (void) define_variable ("KMK_BUILTIN", 11, "append cat chmod cp cmp echo install kDepIDB ln md5sum mkdir mv printf rm rmdir test", o_default, 0);1313 (void) define_variable ("KMK_BUILTIN", 11, "append cat chmod cp cmp echo expr install kDepIDB ln md5sum mkdir mv printf rm rmdir test", o_default, 0); 1314 1314 #endif 1315 1315
Note:
See TracChangeset
for help on using the changeset viewer.