/* ====================================================================
* The Kannel Software License, Version 1.0
*
* Copyright (c) 2001-2005 Kannel Group
* Copyright (c) 1998-2001 WapIT Ltd.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
*
* 2. 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.
*
* 3. The end-user documentation included with the redistribution,
* if any, must include the following acknowledgment:
* "This product includes software developed by the
* Kannel Group (http://www.kannel.org/)."
* Alternately, this acknowledgment may appear in the software itself,
* if and wherever such third-party acknowledgments normally appear.
*
* 4. The names "Kannel" and "Kannel Group" must not be used to
* endorse or promote products derived from this software without
* prior written permission. For written permission, please
* contact org@kannel.org.
*
* 5. Products derived from this software may not be called "Kannel",
* nor may "Kannel" appear in their name, without prior written
* permission of the Kannel Group.
*
* THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED 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 KANNEL GROUP OR ITS 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.
* ====================================================================
*
* This software consists of voluntary contributions made by many
* individuals on behalf of the Kannel Group. For more information on
* the Kannel Group, please see .
*
* Portions of this software are based upon software originally written at
* WapIT Ltd., Helsinki, Finland for the Kannel project.
*/
/*
*
* wmlsc.c
*
* Author: Markku Rossi
*
* Copyright (c) 1999-2000 WAPIT OY LTD.
* All rights reserved.
*
* Main for the WMLScript compiler.
*
*/
#include
#include
#include
#include
#if HAVE_UNISTD_H
#include
#endif /* HAVE_UNISTD_H */
#include
#include
#include "gwlib/gwlib.h"
#include "ws.h"
/* XXX This module, as well, should use the gwmem wrappers. We'll change
this later. --liw */
#undef malloc
#undef realloc
#undef free
/********************* Prototypes for static functions ******************/
/* Print usage message to the stdout. */
static void usage(void);
/* A callback function to receive the meta-pragmas. */
static void pragma_meta(const WsUtf8String *property_name,
const WsUtf8String *content,
const WsUtf8String *scheme,
void *context);
/********************* Static variables *********************************/
/* The name of the compiler program. */
static char *program;
/* Use ws_compile_data() instead of ws_compile_file(). */
static int eval_data = 0;
/********************* Global functions *********************************/
int main(int argc, char *argv[])
{
int i;
WsCompilerParams params;
WsCompilerPtr compiler;
WsResult result;
int opt;
program = strrchr(argv[0], '/');
if (program)
program++;
else
program = argv[0];
/* Initialize the parameters structure. The command line options
modify this directly. */
memset(¶ms, 0, sizeof(params));
/* Process command line arguments. */
while ((opt = getopt(argc, argv, "adhsv")) != EOF) {
switch (opt) {
case 'a':
params.print_assembler = 1;
break;
case 'd':
eval_data = 1;
break;
case 'h':
usage();
exit(0);
break;
case 'l':
params.use_latin1_strings = 1;
break;
case 'p':
params.meta_name_cb = pragma_meta;
params.meta_name_cb_context = "meta name";
params.meta_http_equiv_cb = pragma_meta;
params.meta_http_equiv_cb_context = "meta http equiv";
break;
case 's':
params.print_symbolic_assembler = 1;
break;
case 'v':
params.verbose = 1;
break;
case '?':
printf("Try `%s -h' for a complete list of options.\n",
program);
exit(1);
}
}
/* Create compiler. */
compiler = ws_create(¶ms);
if (compiler == NULL) {
fprintf(stderr, "wsc: could not create compiler\n");
exit(1);
}
for (i = optind; i < argc; i++) {
FILE *ifp, *ofp;
char *outname;
ifp = fopen(argv[i], "rb");
if (ifp == NULL) {
fprintf(stderr, "wsc: could not open input file `%s': %s'\n",
argv[i], strerror(errno));
exit(1);
}
/* Create the output name. */
outname = malloc(strlen(argv[i]) + 1 + 1);
if (outname == NULL) {
fprintf(stderr, "wmlsc: could not create output file name: %s\n",
strerror(errno));
exit(1);
}
strcpy(outname, argv[i]);
strcat(outname, "c");
ofp = fopen(outname, "wb");
if (ofp == NULL) {
fprintf(stderr, "wsc: could not create output file `%s': %s\n",
outname, strerror(errno));
exit(1);
}
if (eval_data) {
/* Use the ws_compile_data() interface. */
struct stat stat_st;
unsigned char *data;
unsigned char *output;
size_t output_len;
if (stat(argv[i], &stat_st) == -1) {
fprintf(stderr, "wsc: could not stat input file `%s': %s\n",
argv[i], strerror(errno));
exit(1);
}
/* Allocate the input buffer. */
data = malloc(stat_st.st_size);
if (data == NULL) {
fprintf(stderr, "wsc: could not allocate input buffer: %s\n",
strerror(errno));
exit(1);
}
if (fread(data, 1, stat_st.st_size, ifp) < (size_t) stat_st.st_size) {
fprintf(stderr, "wsc: could not read data: %s\n",
strerror(errno));
exit(1);
}
result = ws_compile_data(compiler, argv[i], data, stat_st.st_size,
&output, &output_len);
if (result == WS_OK) {
/* Save the output to `ofp'. */
if (fwrite(output, 1, output_len, ofp) != output_len) {
fprintf(stderr,
"wsc: could not save output to file `%s': %s\n",
outname, strerror(errno));
exit(1);
}
}
free(data);
ws_free_byte_code(output);
} else {
/* Use the ws_compile_file() interface. */
result = ws_compile_file(compiler, argv[i], ifp, ofp);
}
/* Common cleanup. */
fclose(ifp);
fclose(ofp);
if (result != WS_OK) {
remove(outname);
fprintf(stderr, "wsc: compilation failed: %s\n",
ws_result_to_string(result));
exit(1);
}
free(outname);
}
ws_destroy(compiler);
return 0;
}
/********************* Static functions *********************************/
static void usage(void)
{
printf("Usage: %s OPTION... FILE...\n\
\n\
-a disassemble resulting byte-code and print it to the\n\
standard output\n\
-d use ws_eval_data() function instead of ws_eval_file()\n\
-h print this help message and exit successfully\n\
-l encode strings in ISO-8859/1 (ISO latin1) instead of using\n\
UTF-8\n\
-p print pragmas\n\
-s print symbolic byte-code assembler to the standard output\n\
-v print verbose progress messages\n\
\n",
program);
}
static void pragma_meta(const WsUtf8String *property_name,
const WsUtf8String *content,
const WsUtf8String *scheme, void *context)
{
FILE *fp = stdout;
char *what = (char *) context;
char *property_name_l = (char *) ws_utf8_to_latin1(property_name, '?', NULL);
char *content_l = (char *) ws_utf8_to_latin1(content, '?', NULL);
char *scheme_l = (char *) ws_utf8_to_latin1(scheme, '?', NULL);
fprintf(fp, "%s: name=\"%s\", content=\"%s\",",
what,
property_name_l ? property_name_l : "",
content_l ? content_l : "");
if (scheme)
fprintf(fp, ", scheme=\"%s\"",
scheme_l ? scheme_l : "");
fprintf(fp, "\n");
ws_utf8_free_data((unsigned char *) property_name_l);
ws_utf8_free_data((unsigned char *) content_l);
ws_utf8_free_data((unsigned char *) scheme_l);
}