mirror of
				https://github.com/jart/cosmopolitan.git
				synced 2025-10-25 02:30:57 +00:00 
			
		
		
		
	Add antirez picol (tiny Tcl clone) (#340)
This commit is contained in:
		
							parent
							
								
									9f8e6c10dd
								
							
						
					
					
						commit
						2eebb198a9
					
				
					 2 changed files with 661 additions and 0 deletions
				
			
		|  | @ -143,6 +143,18 @@ o/$(MODE)/examples/nesemu1.com.dbg:						\ | |||
| 		$(EXAMPLES_BOOTLOADER) | ||||
| 	@$(APELINK) | ||||
| 
 | ||||
| o/$(MODE)/examples/picol.o:					\ | ||||
| 		OVERRIDE_CPPFLAGS +=				\
 | ||||
| 			-DSTACK_FRAME_UNLIMITED | ||||
| 
 | ||||
| o/$(MODE)/examples/picol.com.dbg:				\ | ||||
| 		$(EXAMPLES_DEPS)				\
 | ||||
| 		o/$(MODE)/examples/picol.o			\
 | ||||
| 		o/$(MODE)/examples/examples.pkg			\
 | ||||
| 		$(CRT)						\
 | ||||
| 		$(APE_NO_MODIFY_SELF) | ||||
| 	@$(APELINK) | ||||
| 
 | ||||
| o/$(MODE)/examples/nesemu1.com:							\ | ||||
| 		o/$(MODE)/examples/nesemu1.com.dbg				\
 | ||||
| 		o/$(MODE)/third_party/zip/zip.com				\
 | ||||
|  |  | |||
							
								
								
									
										649
									
								
								examples/picol.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										649
									
								
								examples/picol.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,649 @@ | |||
| /* Tcl in ~ 500 lines of code.
 | ||||
|  * | ||||
|  * Copyright (c) 2007-2016, Salvatore Sanfilippo <antirez at gmail dot com> | ||||
|  * All rights reserved. | ||||
|  * | ||||
|  * Redistribution and use in source and binary forms, with or without | ||||
|  * modification, are permitted provided that the following conditions are met: | ||||
|  * | ||||
|  *   * Redistributions of source code must retain the above copyright notice, | ||||
|  *     this list of conditions and the following disclaimer. | ||||
|  *   * Redistributions in binary form must reproduce the above copyright | ||||
|  *     notice, this list of conditions and the following disclaimer in the | ||||
|  *     documentation and/or other materials provided with the distribution. | ||||
|  * | ||||
|  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | ||||
|  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | ||||
|  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE | ||||
|  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE | ||||
|  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR | ||||
|  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF | ||||
|  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS | ||||
|  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN | ||||
|  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) | ||||
|  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE | ||||
|  * POSSIBILITY OF SUCH DAMAGE. | ||||
|  */ | ||||
| 
 | ||||
| /*
 | ||||
|  * Original from http://oldblog.antirez.com/page/picol.html
 | ||||
|  * Changes on 2021-12-27 by octetta : | ||||
|  * . Use Cosmopolitan's headers. | ||||
|  * . Formatted as per Cosmopolitan's standards. | ||||
|  */ | ||||
| 
 | ||||
| #include <libc/log/log.h> | ||||
| #include <libc/stdio/stdio.h> | ||||
| 
 | ||||
| #include "libc/fmt/conv.h" | ||||
| #include "libc/fmt/fmt.h" | ||||
| 
 | ||||
| enum { PICOL_OK, PICOL_ERR, PICOL_RETURN, PICOL_BREAK, PICOL_CONTINUE }; | ||||
| enum { PT_ESC, PT_STR, PT_CMD, PT_VAR, PT_SEP, PT_EOL, PT_EOF }; | ||||
| 
 | ||||
| struct picolParser { | ||||
|   char *text; | ||||
|   char *p;         /* current text position */ | ||||
|   int len;         /* remaining length */ | ||||
|   char *start;     /* token start */ | ||||
|   char *end;       /* token end */ | ||||
|   int type;        /* token type, PT_... */ | ||||
|   int insidequote; /* True if inside " " */ | ||||
| }; | ||||
| 
 | ||||
| struct picolVar { | ||||
|   char *name, *val; | ||||
|   struct picolVar *next; | ||||
| }; | ||||
| 
 | ||||
| struct picolInterp; /* forward declaration */ | ||||
| typedef int (*picolCmdFunc)(struct picolInterp *i, int argc, char **argv, | ||||
|                             void *privdata); | ||||
| 
 | ||||
| struct picolCmd { | ||||
|   char *name; | ||||
|   picolCmdFunc func; | ||||
|   void *privdata; | ||||
|   struct picolCmd *next; | ||||
| }; | ||||
| 
 | ||||
| struct picolCallFrame { | ||||
|   struct picolVar *vars; | ||||
|   struct picolCallFrame *parent; /* parent is NULL at top level */ | ||||
| }; | ||||
| 
 | ||||
| struct picolInterp { | ||||
|   int level; /* Level of nesting */ | ||||
|   struct picolCallFrame *callframe; | ||||
|   struct picolCmd *commands; | ||||
|   char *result; | ||||
| }; | ||||
| 
 | ||||
| void picolInitParser(struct picolParser *p, char *text) { | ||||
|   p->text = p->p = text; | ||||
|   p->len = strlen(text); | ||||
|   p->start = 0; | ||||
|   p->end = 0; | ||||
|   p->insidequote = 0; | ||||
|   p->type = PT_EOL; | ||||
| } | ||||
| 
 | ||||
| int picolParseSep(struct picolParser *p) { | ||||
|   p->start = p->p; | ||||
|   while (*p->p == ' ' || *p->p == '\t' || *p->p == '\n' || *p->p == '\r') { | ||||
|     p->p++; | ||||
|     p->len--; | ||||
|   } | ||||
|   p->end = p->p - 1; | ||||
|   p->type = PT_SEP; | ||||
|   return PICOL_OK; | ||||
| } | ||||
| 
 | ||||
| int picolParseEol(struct picolParser *p) { | ||||
|   p->start = p->p; | ||||
|   while (*p->p == ' ' || *p->p == '\t' || *p->p == '\n' || *p->p == '\r' || | ||||
|          *p->p == ';') { | ||||
|     p->p++; | ||||
|     p->len--; | ||||
|   } | ||||
|   p->end = p->p - 1; | ||||
|   p->type = PT_EOL; | ||||
|   return PICOL_OK; | ||||
| } | ||||
| 
 | ||||
| int picolParseCommand(struct picolParser *p) { | ||||
|   int level = 1; | ||||
|   int blevel = 0; | ||||
|   p->start = ++p->p; | ||||
|   p->len--; | ||||
|   while (1) { | ||||
|     if (p->len == 0) { | ||||
|       break; | ||||
|     } else if (*p->p == '[' && blevel == 0) { | ||||
|       level++; | ||||
|     } else if (*p->p == ']' && blevel == 0) { | ||||
|       if (!--level) break; | ||||
|     } else if (*p->p == '\\') { | ||||
|       p->p++; | ||||
|       p->len--; | ||||
|     } else if (*p->p == '{') { | ||||
|       blevel++; | ||||
|     } else if (*p->p == '}') { | ||||
|       if (blevel != 0) blevel--; | ||||
|     } | ||||
|     p->p++; | ||||
|     p->len--; | ||||
|   } | ||||
|   p->end = p->p - 1; | ||||
|   p->type = PT_CMD; | ||||
|   if (*p->p == ']') { | ||||
|     p->p++; | ||||
|     p->len--; | ||||
|   } | ||||
|   return PICOL_OK; | ||||
| } | ||||
| 
 | ||||
| int picolParseVar(struct picolParser *p) { | ||||
|   p->start = ++p->p; | ||||
|   p->len--; /* skip the $ */ | ||||
|   while (1) { | ||||
|     if ((*p->p >= 'a' && *p->p <= 'z') || (*p->p >= 'A' && *p->p <= 'Z') || | ||||
|         (*p->p >= '0' && *p->p <= '9') || *p->p == '_') { | ||||
|       p->p++; | ||||
|       p->len--; | ||||
|       continue; | ||||
|     } | ||||
|     break; | ||||
|   } | ||||
|   if (p->start == p->p) { /* It's just a single char string "$" */ | ||||
|     p->start = p->end = p->p - 1; | ||||
|     p->type = PT_STR; | ||||
|   } else { | ||||
|     p->end = p->p - 1; | ||||
|     p->type = PT_VAR; | ||||
|   } | ||||
|   return PICOL_OK; | ||||
| } | ||||
| 
 | ||||
| int picolParseBrace(struct picolParser *p) { | ||||
|   int level = 1; | ||||
|   p->start = ++p->p; | ||||
|   p->len--; | ||||
|   while (1) { | ||||
|     if (p->len >= 2 && *p->p == '\\') { | ||||
|       p->p++; | ||||
|       p->len--; | ||||
|     } else if (p->len == 0 || *p->p == '}') { | ||||
|       level--; | ||||
|       if (level == 0 || p->len == 0) { | ||||
|         p->end = p->p - 1; | ||||
|         if (p->len) { | ||||
|           p->p++; | ||||
|           p->len--; /* Skip final closed brace */ | ||||
|         } | ||||
|         p->type = PT_STR; | ||||
|         return PICOL_OK; | ||||
|       } | ||||
|     } else if (*p->p == '{') | ||||
|       level++; | ||||
|     p->p++; | ||||
|     p->len--; | ||||
|   } | ||||
|   return PICOL_OK; /* unreached */ | ||||
| } | ||||
| 
 | ||||
| int picolParseString(struct picolParser *p) { | ||||
|   int newword = (p->type == PT_SEP || p->type == PT_EOL || p->type == PT_STR); | ||||
|   if (newword && *p->p == '{') | ||||
|     return picolParseBrace(p); | ||||
|   else if (newword && *p->p == '"') { | ||||
|     p->insidequote = 1; | ||||
|     p->p++; | ||||
|     p->len--; | ||||
|   } | ||||
|   p->start = p->p; | ||||
|   while (1) { | ||||
|     if (p->len == 0) { | ||||
|       p->end = p->p - 1; | ||||
|       p->type = PT_ESC; | ||||
|       return PICOL_OK; | ||||
|     } | ||||
|     switch (*p->p) { | ||||
|       case '\\': | ||||
|         if (p->len >= 2) { | ||||
|           p->p++; | ||||
|           p->len--; | ||||
|         } | ||||
|         break; | ||||
|       case '$': | ||||
|       case '[': | ||||
|         p->end = p->p - 1; | ||||
|         p->type = PT_ESC; | ||||
|         return PICOL_OK; | ||||
|       case ' ': | ||||
|       case '\t': | ||||
|       case '\n': | ||||
|       case '\r': | ||||
|       case ';': | ||||
|         if (!p->insidequote) { | ||||
|           p->end = p->p - 1; | ||||
|           p->type = PT_ESC; | ||||
|           return PICOL_OK; | ||||
|         } | ||||
|         break; | ||||
|       case '"': | ||||
|         if (p->insidequote) { | ||||
|           p->end = p->p - 1; | ||||
|           p->type = PT_ESC; | ||||
|           p->p++; | ||||
|           p->len--; | ||||
|           p->insidequote = 0; | ||||
|           return PICOL_OK; | ||||
|         } | ||||
|         break; | ||||
|     } | ||||
|     p->p++; | ||||
|     p->len--; | ||||
|   } | ||||
|   return PICOL_OK; /* unreached */ | ||||
| } | ||||
| 
 | ||||
| int picolParseComment(struct picolParser *p) { | ||||
|   while (p->len && *p->p != '\n') { | ||||
|     p->p++; | ||||
|     p->len--; | ||||
|   } | ||||
|   return PICOL_OK; | ||||
| } | ||||
| 
 | ||||
| int picolGetToken(struct picolParser *p) { | ||||
|   while (1) { | ||||
|     if (!p->len) { | ||||
|       if (p->type != PT_EOL && p->type != PT_EOF) | ||||
|         p->type = PT_EOL; | ||||
|       else | ||||
|         p->type = PT_EOF; | ||||
|       return PICOL_OK; | ||||
|     } | ||||
|     switch (*p->p) { | ||||
|       case ' ': | ||||
|       case '\t': | ||||
|       case '\r': | ||||
|         if (p->insidequote) return picolParseString(p); | ||||
|         return picolParseSep(p); | ||||
|       case '\n': | ||||
|       case ';': | ||||
|         if (p->insidequote) return picolParseString(p); | ||||
|         return picolParseEol(p); | ||||
|       case '[': | ||||
|         return picolParseCommand(p); | ||||
|       case '$': | ||||
|         return picolParseVar(p); | ||||
|       case '#': | ||||
|         if (p->type == PT_EOL) { | ||||
|           picolParseComment(p); | ||||
|           continue; | ||||
|         } | ||||
|         return picolParseString(p); | ||||
|       default: | ||||
|         return picolParseString(p); | ||||
|     } | ||||
|   } | ||||
|   return PICOL_OK; /* unreached */ | ||||
| } | ||||
| 
 | ||||
| void picolInitInterp(struct picolInterp *i) { | ||||
|   i->level = 0; | ||||
|   i->callframe = malloc(sizeof(struct picolCallFrame)); | ||||
|   i->callframe->vars = NULL; | ||||
|   i->callframe->parent = NULL; | ||||
|   i->commands = NULL; | ||||
|   i->result = strdup(""); | ||||
| } | ||||
| 
 | ||||
| void picolSetResult(struct picolInterp *i, char *s) { | ||||
|   free(i->result); | ||||
|   i->result = strdup(s); | ||||
| } | ||||
| 
 | ||||
| struct picolVar *picolGetVar(struct picolInterp *i, char *name) { | ||||
|   struct picolVar *v = i->callframe->vars; | ||||
|   while (v) { | ||||
|     if (strcmp(v->name, name) == 0) return v; | ||||
|     v = v->next; | ||||
|   } | ||||
|   return NULL; | ||||
| } | ||||
| 
 | ||||
| int picolSetVar(struct picolInterp *i, char *name, char *val) { | ||||
|   struct picolVar *v = picolGetVar(i, name); | ||||
|   if (v) { | ||||
|     free(v->val); | ||||
|     v->val = strdup(val); | ||||
|   } else { | ||||
|     v = malloc(sizeof(*v)); | ||||
|     v->name = strdup(name); | ||||
|     v->val = strdup(val); | ||||
|     v->next = i->callframe->vars; | ||||
|     i->callframe->vars = v; | ||||
|   } | ||||
|   return PICOL_OK; | ||||
| } | ||||
| 
 | ||||
| struct picolCmd *picolGetCommand(struct picolInterp *i, char *name) { | ||||
|   struct picolCmd *c = i->commands; | ||||
|   while (c) { | ||||
|     if (strcmp(c->name, name) == 0) return c; | ||||
|     c = c->next; | ||||
|   } | ||||
|   return NULL; | ||||
| } | ||||
| 
 | ||||
| int picolRegisterCommand(struct picolInterp *i, char *name, picolCmdFunc f, | ||||
|                          void *privdata) { | ||||
|   struct picolCmd *c = picolGetCommand(i, name); | ||||
|   char errbuf[1024]; | ||||
|   if (c) { | ||||
|     snprintf(errbuf, 1024, "Command '%s' already defined", name); | ||||
|     picolSetResult(i, errbuf); | ||||
|     return PICOL_ERR; | ||||
|   } | ||||
|   c = malloc(sizeof(*c)); | ||||
|   c->name = strdup(name); | ||||
|   c->func = f; | ||||
|   c->privdata = privdata; | ||||
|   c->next = i->commands; | ||||
|   i->commands = c; | ||||
|   return PICOL_OK; | ||||
| } | ||||
| 
 | ||||
| /* EVAL! */ | ||||
| int picolEval(struct picolInterp *i, char *t) { | ||||
|   struct picolParser p; | ||||
|   int argc = 0, j; | ||||
|   char **argv = NULL; | ||||
|   char errbuf[1024]; | ||||
|   int retcode = PICOL_OK; | ||||
|   picolSetResult(i, ""); | ||||
|   picolInitParser(&p, t); | ||||
|   while (1) { | ||||
|     char *t; | ||||
|     int tlen; | ||||
|     int prevtype = p.type; | ||||
|     picolGetToken(&p); | ||||
|     if (p.type == PT_EOF) break; | ||||
|     tlen = p.end - p.start + 1; | ||||
|     if (tlen < 0) tlen = 0; | ||||
|     t = malloc(tlen + 1); | ||||
|     memcpy(t, p.start, tlen); | ||||
|     t[tlen] = '\0'; | ||||
|     if (p.type == PT_VAR) { | ||||
|       struct picolVar *v = picolGetVar(i, t); | ||||
|       if (!v) { | ||||
|         snprintf(errbuf, 1024, "No such variable '%s'", t); | ||||
|         free(t); | ||||
|         picolSetResult(i, errbuf); | ||||
|         retcode = PICOL_ERR; | ||||
|         goto err; | ||||
|       } | ||||
|       free(t); | ||||
|       t = strdup(v->val); | ||||
|     } else if (p.type == PT_CMD) { | ||||
|       retcode = picolEval(i, t); | ||||
|       free(t); | ||||
|       if (retcode != PICOL_OK) goto err; | ||||
|       t = strdup(i->result); | ||||
|     } else if (p.type == PT_ESC) { | ||||
|       /* XXX: escape handling missing! */ | ||||
|     } else if (p.type == PT_SEP) { | ||||
|       prevtype = p.type; | ||||
|       free(t); | ||||
|       continue; | ||||
|     } | ||||
|     /* We have a complete command + args. Call it! */ | ||||
|     if (p.type == PT_EOL) { | ||||
|       struct picolCmd *c; | ||||
|       free(t); | ||||
|       prevtype = p.type; | ||||
|       if (argc) { | ||||
|         if ((c = picolGetCommand(i, argv[0])) == NULL) { | ||||
|           snprintf(errbuf, 1024, "No such command '%s'", argv[0]); | ||||
|           picolSetResult(i, errbuf); | ||||
|           retcode = PICOL_ERR; | ||||
|           goto err; | ||||
|         } | ||||
|         retcode = c->func(i, argc, argv, c->privdata); | ||||
|         if (retcode != PICOL_OK) goto err; | ||||
|       } | ||||
|       /* Prepare for the next command */ | ||||
|       for (j = 0; j < argc; j++) free(argv[j]); | ||||
|       free(argv); | ||||
|       argv = NULL; | ||||
|       argc = 0; | ||||
|       continue; | ||||
|     } | ||||
|     /* We have a new token, append to the previous or as new arg? */ | ||||
|     if (prevtype == PT_SEP || prevtype == PT_EOL) { | ||||
|       argv = realloc(argv, sizeof(char *) * (argc + 1)); | ||||
|       argv[argc] = t; | ||||
|       argc++; | ||||
|     } else { /* Interpolation */ | ||||
|       int oldlen = strlen(argv[argc - 1]), tlen = strlen(t); | ||||
|       argv[argc - 1] = realloc(argv[argc - 1], oldlen + tlen + 1); | ||||
|       memcpy(argv[argc - 1] + oldlen, t, tlen); | ||||
|       argv[argc - 1][oldlen + tlen] = '\0'; | ||||
|       free(t); | ||||
|     } | ||||
|     prevtype = p.type; | ||||
|   } | ||||
| err: | ||||
|   for (j = 0; j < argc; j++) free(argv[j]); | ||||
|   free(argv); | ||||
|   return retcode; | ||||
| } | ||||
| 
 | ||||
| /* ACTUAL COMMANDS! */ | ||||
| int picolArityErr(struct picolInterp *i, char *name) { | ||||
|   char buf[1024]; | ||||
|   snprintf(buf, 1024, "Wrong number of args for %s", name); | ||||
|   picolSetResult(i, buf); | ||||
|   return PICOL_ERR; | ||||
| } | ||||
| 
 | ||||
| int picolCommandMath(struct picolInterp *i, int argc, char **argv, void *pd) { | ||||
|   char buf[64]; | ||||
|   int a, b, c; | ||||
|   if (argc != 3) return picolArityErr(i, argv[0]); | ||||
|   a = atoi(argv[1]); | ||||
|   b = atoi(argv[2]); | ||||
|   if (argv[0][0] == '+') | ||||
|     c = a + b; | ||||
|   else if (argv[0][0] == '-') | ||||
|     c = a - b; | ||||
|   else if (argv[0][0] == '*') | ||||
|     c = a * b; | ||||
|   else if (argv[0][0] == '/') | ||||
|     c = a / b; | ||||
|   else if (argv[0][0] == '>' && argv[0][1] == '\0') | ||||
|     c = a > b; | ||||
|   else if (argv[0][0] == '>' && argv[0][1] == '=') | ||||
|     c = a >= b; | ||||
|   else if (argv[0][0] == '<' && argv[0][1] == '\0') | ||||
|     c = a < b; | ||||
|   else if (argv[0][0] == '<' && argv[0][1] == '=') | ||||
|     c = a <= b; | ||||
|   else if (argv[0][0] == '=' && argv[0][1] == '=') | ||||
|     c = a == b; | ||||
|   else if (argv[0][0] == '!' && argv[0][1] == '=') | ||||
|     c = a != b; | ||||
|   else | ||||
|     c = 0; /* I hate warnings */ | ||||
|   snprintf(buf, 64, "%d", c); | ||||
|   picolSetResult(i, buf); | ||||
|   return PICOL_OK; | ||||
| } | ||||
| 
 | ||||
| int picolCommandSet(struct picolInterp *i, int argc, char **argv, void *pd) { | ||||
|   if (argc != 3) return picolArityErr(i, argv[0]); | ||||
|   picolSetVar(i, argv[1], argv[2]); | ||||
|   picolSetResult(i, argv[2]); | ||||
|   return PICOL_OK; | ||||
| } | ||||
| 
 | ||||
| int picolCommandPuts(struct picolInterp *i, int argc, char **argv, void *pd) { | ||||
|   if (argc != 2) return picolArityErr(i, argv[0]); | ||||
|   printf("%s\n", argv[1]); | ||||
|   return PICOL_OK; | ||||
| } | ||||
| 
 | ||||
| int picolCommandIf(struct picolInterp *i, int argc, char **argv, void *pd) { | ||||
|   int retcode; | ||||
|   if (argc != 3 && argc != 5) return picolArityErr(i, argv[0]); | ||||
|   if ((retcode = picolEval(i, argv[1])) != PICOL_OK) return retcode; | ||||
|   if (atoi(i->result)) | ||||
|     return picolEval(i, argv[2]); | ||||
|   else if (argc == 5) | ||||
|     return picolEval(i, argv[4]); | ||||
|   return PICOL_OK; | ||||
| } | ||||
| 
 | ||||
| int picolCommandWhile(struct picolInterp *i, int argc, char **argv, void *pd) { | ||||
|   if (argc != 3) return picolArityErr(i, argv[0]); | ||||
|   while (1) { | ||||
|     int retcode = picolEval(i, argv[1]); | ||||
|     if (retcode != PICOL_OK) return retcode; | ||||
|     if (atoi(i->result)) { | ||||
|       if ((retcode = picolEval(i, argv[2])) == PICOL_CONTINUE) | ||||
|         continue; | ||||
|       else if (retcode == PICOL_OK) | ||||
|         continue; | ||||
|       else if (retcode == PICOL_BREAK) | ||||
|         return PICOL_OK; | ||||
|       else | ||||
|         return retcode; | ||||
|     } else { | ||||
|       return PICOL_OK; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| int picolCommandRetCodes(struct picolInterp *i, int argc, char **argv, | ||||
|                          void *pd) { | ||||
|   if (argc != 1) return picolArityErr(i, argv[0]); | ||||
|   if (strcmp(argv[0], "break") == 0) | ||||
|     return PICOL_BREAK; | ||||
|   else if (strcmp(argv[0], "continue") == 0) | ||||
|     return PICOL_CONTINUE; | ||||
|   return PICOL_OK; | ||||
| } | ||||
| 
 | ||||
| void picolDropCallFrame(struct picolInterp *i) { | ||||
|   struct picolCallFrame *cf = i->callframe; | ||||
|   struct picolVar *v = cf->vars, *t; | ||||
|   while (v) { | ||||
|     t = v->next; | ||||
|     free(v->name); | ||||
|     free(v->val); | ||||
|     free(v); | ||||
|     v = t; | ||||
|   } | ||||
|   i->callframe = cf->parent; | ||||
|   free(cf); | ||||
| } | ||||
| 
 | ||||
| int picolCommandCallProc(struct picolInterp *i, int argc, char **argv, | ||||
|                          void *pd) { | ||||
|   char **x = pd, *alist = x[0], *body = x[1], *p = strdup(alist), *tofree; | ||||
|   struct picolCallFrame *cf = malloc(sizeof(*cf)); | ||||
|   int arity = 0, done = 0, errcode = PICOL_OK; | ||||
|   char errbuf[1024]; | ||||
|   cf->vars = NULL; | ||||
|   cf->parent = i->callframe; | ||||
|   i->callframe = cf; | ||||
|   tofree = p; | ||||
|   while (1) { | ||||
|     char *start = p; | ||||
|     while (*p != ' ' && *p != '\0') p++; | ||||
|     if (*p != '\0' && p == start) { | ||||
|       p++; | ||||
|       continue; | ||||
|     } | ||||
|     if (p == start) break; | ||||
|     if (*p == '\0') | ||||
|       done = 1; | ||||
|     else | ||||
|       *p = '\0'; | ||||
|     if (++arity > argc - 1) goto arityerr; | ||||
|     picolSetVar(i, start, argv[arity]); | ||||
|     p++; | ||||
|     if (done) break; | ||||
|   } | ||||
|   free(tofree); | ||||
|   if (arity != argc - 1) goto arityerr; | ||||
|   errcode = picolEval(i, body); | ||||
|   if (errcode == PICOL_RETURN) errcode = PICOL_OK; | ||||
|   picolDropCallFrame(i); /* remove the called proc callframe */ | ||||
|   return errcode; | ||||
| arityerr: | ||||
|   snprintf(errbuf, 1024, "Proc '%s' called with wrong arg num", argv[0]); | ||||
|   picolSetResult(i, errbuf); | ||||
|   picolDropCallFrame(i); /* remove the called proc callframe */ | ||||
|   return PICOL_ERR; | ||||
| } | ||||
| 
 | ||||
| int picolCommandProc(struct picolInterp *i, int argc, char **argv, void *pd) { | ||||
|   char **procdata = malloc(sizeof(char *) * 2); | ||||
|   if (argc != 4) return picolArityErr(i, argv[0]); | ||||
|   procdata[0] = strdup(argv[2]); /* arguments list */ | ||||
|   procdata[1] = strdup(argv[3]); /* procedure body */ | ||||
|   return picolRegisterCommand(i, argv[1], picolCommandCallProc, procdata); | ||||
| } | ||||
| 
 | ||||
| int picolCommandReturn(struct picolInterp *i, int argc, char **argv, void *pd) { | ||||
|   if (argc != 1 && argc != 2) return picolArityErr(i, argv[0]); | ||||
|   picolSetResult(i, (argc == 2) ? argv[1] : ""); | ||||
|   return PICOL_RETURN; | ||||
| } | ||||
| 
 | ||||
| void picolRegisterCoreCommands(struct picolInterp *i) { | ||||
|   int j; | ||||
|   char *name[] = {"+", "-", "*", "/", ">", ">=", "<", "<=", "==", "!="}; | ||||
|   for (j = 0; j < (int)(sizeof(name) / sizeof(char *)); j++) | ||||
|     picolRegisterCommand(i, name[j], picolCommandMath, NULL); | ||||
|   picolRegisterCommand(i, "set", picolCommandSet, NULL); | ||||
|   picolRegisterCommand(i, "puts", picolCommandPuts, NULL); | ||||
|   picolRegisterCommand(i, "if", picolCommandIf, NULL); | ||||
|   picolRegisterCommand(i, "while", picolCommandWhile, NULL); | ||||
|   picolRegisterCommand(i, "break", picolCommandRetCodes, NULL); | ||||
|   picolRegisterCommand(i, "continue", picolCommandRetCodes, NULL); | ||||
|   picolRegisterCommand(i, "proc", picolCommandProc, NULL); | ||||
|   picolRegisterCommand(i, "return", picolCommandReturn, NULL); | ||||
| } | ||||
| 
 | ||||
| int main(int argc, char **argv) { | ||||
|   struct picolInterp interp; | ||||
|   picolInitInterp(&interp); | ||||
|   picolRegisterCoreCommands(&interp); | ||||
|   if (argc == 1) { | ||||
|     while (1) { | ||||
|       char clibuf[1024]; | ||||
|       int retcode; | ||||
|       printf("picol> "); | ||||
|       fflush(stdout); | ||||
|       if (fgets(clibuf, 1024, stdin) == NULL) return 0; | ||||
|       retcode = picolEval(&interp, clibuf); | ||||
|       if (interp.result[0] != '\0') printf("[%d] %s\n", retcode, interp.result); | ||||
|     } | ||||
|   } else if (argc == 2) { | ||||
|     char buf[1024 * 16]; | ||||
|     FILE *fp = fopen(argv[1], "r"); | ||||
|     if (!fp) { | ||||
|       perror("open"); | ||||
|       exit(1); | ||||
|     } | ||||
|     buf[fread(buf, 1, 1024 * 16, fp)] = '\0'; | ||||
|     fclose(fp); | ||||
|     if (picolEval(&interp, buf) != PICOL_OK) printf("%s\n", interp.result); | ||||
|   } | ||||
|   return 0; | ||||
| } | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue