#include <xmake/variable.h>
#include <xmake/xmake.h>
+struct linebuf {
+
+ char *start;
+ unsigned long size;
+
+ FILE *f;
+
+};
+
struct builtin_function {
const char *name;
}
+static void read_line (struct linebuf *lbuf) {
+
+ char *end = lbuf->start + lbuf->size;
+ char *p = lbuf->start;
+
+ while (fgets (p, end - p, lbuf->f)) {
+
+ size_t offset;
+
+ p += strlen (p);
+ offset = p - lbuf->start;
+
+ if (p[-1] == '\n') {
+
+ p--;
+
+ if (p[-1] == '\r') {
+ p[-1] = ' ';
+ }
+
+ }
+
+ if (end - p >= 80) {
+ continue;
+ }
+
+ lbuf->size *= 2;
+ lbuf->start = xrealloc (lbuf->start, lbuf->size);
+
+ p = lbuf->start + offset;
+ end = lbuf->start + lbuf->size;
+
+ }
+
+}
+
static char *func_error (const char *filename, unsigned long line_no, char *input) {
- fprintf (stderr, "%s: %s: %lu: %s\n", program_name, filename, line_no, input);
+ fprintf (stderr, "%s: %s: %lu: *** %s. Stop.\n", program_name, filename, line_no, input);
exit (EXIT_FAILURE);
}
+static char *func_shell (const char *filename, unsigned long line_no, char *input) {
+
+ struct linebuf lbuf;
+ FILE *fp;
+
+ if (!*input) {
+
+ fprintf (stderr, "%s: %s: %lu: empty call\n", program_name, filename, line_no);
+ return 0;
+
+ }
+
+ if (!(fp = popen (input, "rb"))) {
+ return 0;
+ }
+
+ lbuf.size = 256;
+
+ lbuf.start = xmalloc (lbuf.size);
+ lbuf.f = fp;
+
+ read_line (&lbuf);
+ pclose(fp);
+
+ return lbuf.start;
+
+}
+
static struct builtin_function builtin_functions[] ={
- { "eval", &func_eval },
{ "error", &func_error },
+ { "shell", &func_shell },
+ { "eval", &func_eval },
{ 0, 0 }
};
q++;
}
- alloc_replacement = func->func (filename, line_no, q);
+ if ((alloc_replacement = func->func (filename, line_no, q))) {
+
+ new = xmalloc (pos + strlen (alloc_replacement) + 1);
+
+ memcpy (new, line, pos);
+ memcpy (new + pos, alloc_replacement, strlen (alloc_replacement));
+
+ free (alloc_replacement);
+ return new;
+
+ }
} else if ((q = strchr (content, '='))) {