% texlua.w % % Copyright (C) 1994-2007 Lua.org, PUC-Rio. All rights reserved. % Copyright 2006-2010 Taco Hoekwater <taco@@luatex.org> % % Permission is hereby granted, free of charge, to any person obtaining % a copy of this software and associated documentation files (the % "Software"), to deal in the Software without restriction, including % without limitation the rights to use, copy, modify, merge, publish, % distribute, sublicense, and/or sell copies of the Software, and to % permit persons to whom the Software is furnished to do so, subject to % the following conditions: * % The above copyright notice and this permission notice shall be % included in all copies or substantial portions of the Software. * % THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, % EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF % MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. % IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY % CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, % TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE % SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. % This file is part of LuaTeX. @ @c #include <errno.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #define luac_c #define LUA_CORE #include "lua51/lua.h" #include "lua51/lauxlib.h" #include "lua51/ldo.h" #include "lua51/lfunc.h" #include "lua51/lmem.h" #include "lua51/lobject.h" #include "lua51/lopcodes.h" #include "lua51/lstring.h" #include "lua51/lundump.h" #include "lua/luatex-api.h" @ @c static const char _svn_version[] = "$Id: texluac.w 3707 2010-05-28 13:24:16Z oneiros $ $URL: http://foundry.supelec.fr/svn/luatex/tags/beta-0.70.1/source/texk/web2c/luatexdir/lua/texluac.w $"; /* fix for non-gcc compilation: */ #if !defined(__GNUC__) || (__GNUC__ < 2) # define __attribute__(x) #endif /* !defined(__GNUC__) || (__GNUC__ < 2) */ @ @c #define PROGNAME "texluac" /* default program name */ #define OUTPUT PROGNAME ".out" /* default output file */ static int dumping = 1; /* dump bytecodes? */ static int stripping = 0; /* strip debug information? */ static char Output[] = { OUTPUT }; /* default output file name */ static const char *output = Output; /* actual output file name */ static const char *progname = PROGNAME; /* actual program name */ @ @c __attribute__ ((noreturn)) static void fatal(const char *message) { fprintf(stderr, "%s: %s\n", progname, message); exit(EXIT_FAILURE); } @ @c __attribute__ ((noreturn)) static void cannot(const char *what) { fprintf(stderr, "%s: cannot %s %s: %s\n", progname, what, output, strerror(errno)); exit(EXIT_FAILURE); } @ @c __attribute__ ((noreturn)) static void usage(const char *message) { if (*message == '-') fprintf(stderr, "%s: unrecognized option " LUA_QS "\n", progname, message); else fprintf(stderr, "%s: %s\n", progname, message); fprintf(stderr, "usage: %s [options] [filenames].\n" "Available options are:\n" " - process stdin\n" " -o name output to file " LUA_QL("name") " (default is \"%s\")\n" " -p parse only\n" " -s strip debug information\n" " -v show version information\n" " -- stop handling options\n", progname, Output); exit(EXIT_FAILURE); } @ @c #define IS(s) (strcmp(argv[i],s)==0) static int doargs(int argc, char *argv[]) { int i; int version = 0; if (argv[0] != NULL && *argv[0] != 0) progname = argv[0]; for (i = 1; i < argc; i++) { if (*argv[i] != '-') /* end of options; keep it */ break; else if (IS("--")) { /* end of options; skip it */ ++i; if (version) ++version; break; } else if (IS("-")) /* end of options; use stdin */ break; else if (IS("-o")) { /* output file */ output = argv[++i]; if (output == NULL || *output == 0) usage(LUA_QL("-o") " needs argument"); if (IS("-")) output = NULL; } else if (IS("-p")) /* parse only */ dumping = 0; else if (IS("-s")) /* strip debug information */ stripping = 1; else if (IS("-v")) /* show version */ ++version; else if (IS("--luaconly")) /* ignore */ ; else if (IS("--luac")) /* ignore */ ; else /* unknown option */ usage(argv[i]); } if (i == argc && (!dumping)) { dumping = 0; argv[--i] = Output; } if (version) { printf("%s %s\n", LUA_RELEASE, LUA_COPYRIGHT); if (version == argc - 1) exit(EXIT_SUCCESS); } return i; } @ @c #define toproto(L,i) (clvalue(L->top+(i))->l.p) static const Proto *combine(lua_State * L, int n) { if (n == 1) return toproto(L, -1); else { int i, pc; Proto *f = luaF_newproto(L); setptvalue2s(L, L->top, f); incr_top(L); f->source = luaS_newliteral(L, "=(" PROGNAME ")"); f->maxstacksize = 1; pc = 2 * n + 1; f->code = luaM_newvector(L, (unsigned long) pc, Instruction); f->sizecode = pc; f->p = luaM_newvector(L, (unsigned long) n, Proto *); f->sizep = n; pc = 0; for (i = 0; i < n; i++) { f->p[i] = toproto(L, i - n - 1); f->code[pc++] = CREATE_ABx(OP_CLOSURE, 0, i); f->code[pc++] = CREATE_ABC(OP_CALL, 0, 1, 1); } f->code[pc++] = CREATE_ABC(OP_RETURN, 0, 1, 0); return f; } } @ @c static int writer(lua_State * L, const void *p, size_t size, void *u) { UNUSED(L); return (fwrite(p, size, 1, (FILE *) u) != 1) && (size != 0); } @ @c struct Smain { int argc; char **argv; }; @ @c static int pmain(lua_State * L) { struct Smain *s = (struct Smain *) lua_touserdata(L, 1); int argc = s->argc; char **argv = s->argv; const Proto *f; int i; if (!lua_checkstack(L, argc)) fatal("too many input files"); for (i = 0; i < argc; i++) { const char *filename = IS("-") ? NULL : argv[i]; if (luaL_loadfile(L, filename) != 0) fatal(lua_tostring(L, -1)); } f = combine(L, argc); if (dumping) { FILE *D = (output == NULL) ? stdout : fopen(output, "wb"); if (D == NULL) cannot("open"); lua_lock(L); luaU_dump(L, f, writer, D, stripping); lua_unlock(L); if (ferror(D)) cannot("write"); if (fclose(D)) cannot("close"); } return 0; } @ @c int luac_main(int argc, char *argv[]) { lua_State *L; struct Smain s; int i = doargs(argc, argv); argc -= i; argv += i; if (argc <= 0) usage("no input files given"); L = lua_open(); if (L == NULL) fatal("not enough memory for state"); s.argc = argc; s.argv = argv; if (lua_cpcall(L, pmain, &s) != 0) fatal(lua_tostring(L, -1)); lua_close(L); return EXIT_SUCCESS; }