2007-10-25 16:27:34 -04:00
|
|
|
/*
|
|
|
|
* Ikarus Scheme -- A compiler for R6RS Scheme.
|
|
|
|
* Copyright (C) 2006,2007 Abdulaziz Ghuloum
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License version 3 as
|
|
|
|
* published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful, but
|
|
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2006-11-23 19:38:26 -05:00
|
|
|
|
|
|
|
|
2007-10-17 09:22:47 -04:00
|
|
|
#include "ikarus-data.h"
|
2006-11-23 19:38:26 -05:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <errno.h>
|
2006-11-23 19:48:14 -05:00
|
|
|
#include <gmp.h>
|
2006-12-24 01:43:20 -05:00
|
|
|
#include <signal.h>
|
2007-10-15 17:58:03 -04:00
|
|
|
#include <sys/mman.h>
|
2006-11-23 19:38:26 -05:00
|
|
|
|
2006-12-24 01:43:20 -05:00
|
|
|
void register_handlers();
|
|
|
|
void register_alt_stack();
|
|
|
|
|
2007-10-17 09:22:47 -04:00
|
|
|
void ikarus_usage_short(){
|
|
|
|
fprintf(stderr, "ikarus -h for more help\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
void ikarus_usage(){
|
|
|
|
static char* helpstring =
|
|
|
|
"\n\
|
|
|
|
Options for running ikarus scheme:\n\
|
|
|
|
\n ikarus -h\n\
|
|
|
|
Prints this help message then exits.\n\
|
2007-11-03 17:31:18 -04:00
|
|
|
\n ikarus [-b <bootfile>] --r6rs-script <scriptfile> opts ...\n\
|
2007-10-17 09:22:47 -04:00
|
|
|
Starts ikarus in r6rs-script mode. The script file is treated\n\
|
|
|
|
as an R6RS-script. The options opts ... can be obtained using\n\
|
|
|
|
the \"command-line\" procedure in the (rnrs programs) library.\n\
|
|
|
|
\n ikarus [-b <bootfile>] <file> ... [-- opts ...]\n\
|
|
|
|
Starts ikarus in interactive mode. Each of the files is first\n\
|
|
|
|
loaded into the interaction environment before the interactive\n\
|
|
|
|
repl is started. The options opts can be obtained using the\n\
|
|
|
|
\"command-line\" procedure.\n\
|
|
|
|
\n\
|
|
|
|
If the option [-b <bootfile>] is provided, the bootfile is used\n\
|
|
|
|
as the system's initial boot file from which the environment is\n\
|
|
|
|
initialized. If the -b option is not supplied, the default boot\n\
|
2007-11-10 08:28:19 -05:00
|
|
|
file is used. The current default boot file location is\n\
|
|
|
|
\"%s\".\n\
|
|
|
|
Consult the Ikarus Scheme User's Guide for more details.\n\n";
|
|
|
|
fprintf(stderr, helpstring, BOOTFILE);
|
2007-10-17 09:22:47 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-12-24 01:43:20 -05:00
|
|
|
ikpcb* the_pcb;
|
2006-12-07 01:38:04 -05:00
|
|
|
|
2006-12-01 09:23:37 -05:00
|
|
|
/* get_option
|
|
|
|
|
|
|
|
takes pointers to argc and argv and looks for the first
|
|
|
|
option matching opt. If one exists, it removes it from the argv
|
|
|
|
list, updates argc, and returns a pointer to the option value.
|
|
|
|
returns null if option is not found.
|
|
|
|
*/
|
|
|
|
char*
|
|
|
|
get_option(char* opt, int argc, char** argv){
|
|
|
|
int i;
|
|
|
|
for(i=1; i<argc; i++){
|
|
|
|
if(strcmp(opt, argv[i]) == 0){
|
|
|
|
if((i+1) < argc){
|
|
|
|
char* rv = argv[i+1];
|
|
|
|
int j;
|
|
|
|
for(j=i+2; j<argc; j++, i++){
|
|
|
|
argv[i] = argv[j];
|
|
|
|
}
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
else {
|
2007-10-17 09:22:47 -04:00
|
|
|
fprintf(stderr,
|
|
|
|
"ikarus error: option %s requires a value, none provided\n",
|
|
|
|
opt);
|
|
|
|
ikarus_usage_short();
|
2006-12-01 09:23:37 -05:00
|
|
|
exit(-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if(strcmp("--", argv[i]) == 0){
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-10-17 09:22:47 -04:00
|
|
|
int
|
|
|
|
get_option0(char* opt, int argc, char** argv){
|
|
|
|
int i;
|
|
|
|
for(i=1; i<argc; i++){
|
|
|
|
if(strcmp(opt, argv[i]) == 0){
|
|
|
|
int j;
|
|
|
|
for(j=i+1; j<argc; j++, i++){
|
|
|
|
argv[i] = argv[j];
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else if(strcmp("--", argv[i]) == 0){
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-12-01 11:18:01 -05:00
|
|
|
int
|
|
|
|
file_exists(char* filename){
|
|
|
|
struct stat sb;
|
|
|
|
int s = stat(filename, &sb);
|
|
|
|
return (s == 0);
|
|
|
|
}
|
|
|
|
|
2006-11-23 19:38:26 -05:00
|
|
|
int main(int argc, char** argv){
|
2007-10-17 09:22:47 -04:00
|
|
|
if(get_option0("-h", argc, argv)){
|
|
|
|
ikarus_usage();
|
|
|
|
exit(0);
|
|
|
|
}
|
2006-12-01 11:18:01 -05:00
|
|
|
char buff[FILENAME_MAX];
|
2006-12-01 09:23:37 -05:00
|
|
|
char* boot_file = get_option("-b", argc, argv);
|
|
|
|
if(boot_file){
|
|
|
|
argc -= 2;
|
2007-11-10 08:28:19 -05:00
|
|
|
} else {
|
|
|
|
boot_file = BOOTFILE;
|
2006-12-01 11:18:01 -05:00
|
|
|
}
|
2006-11-23 19:48:14 -05:00
|
|
|
if(sizeof(mp_limb_t) != sizeof(int)){
|
2007-11-10 08:28:19 -05:00
|
|
|
fprintf(stderr, "ERROR: limb size does not match\n");
|
|
|
|
exit(-1);
|
2006-11-23 19:48:14 -05:00
|
|
|
}
|
|
|
|
if(mp_bits_per_limb != (8*sizeof(int))){
|
2007-11-10 08:28:19 -05:00
|
|
|
fprintf(stderr, "ERROR: invalid bits_per_limb=%d\n", mp_bits_per_limb);
|
|
|
|
exit(-1);
|
2006-11-23 19:48:14 -05:00
|
|
|
}
|
2006-11-23 19:38:26 -05:00
|
|
|
ikpcb* pcb = ik_make_pcb();
|
2006-12-24 01:43:20 -05:00
|
|
|
the_pcb = pcb;
|
2006-12-01 09:52:12 -05:00
|
|
|
{ /* set up arg_list */
|
|
|
|
ikp arg_list = null_object;
|
|
|
|
int i = argc-1;
|
|
|
|
while(i > 0){
|
|
|
|
char* s = argv[i];
|
|
|
|
int n = strlen(s);
|
2007-05-19 14:13:51 -04:00
|
|
|
ikp str = ik_alloc(pcb, align(n*string_char_size+disp_string_data+1))
|
|
|
|
+ string_tag;
|
|
|
|
ref(str, off_string_length) = fix(n);
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for(i=0; i<n; i++){
|
|
|
|
string_set(str, i, integer_to_char(s[i]));
|
|
|
|
}
|
|
|
|
}
|
2006-12-01 09:52:12 -05:00
|
|
|
ikp p = ik_alloc(pcb, pair_size);
|
2007-05-19 14:13:51 -04:00
|
|
|
ref(p, disp_car) = str;
|
2006-12-01 09:52:12 -05:00
|
|
|
ref(p, disp_cdr) = arg_list;
|
|
|
|
arg_list = p+pair_tag;
|
|
|
|
i--;
|
|
|
|
}
|
|
|
|
pcb->arg_list = arg_list;
|
|
|
|
}
|
2006-12-24 01:43:20 -05:00
|
|
|
register_handlers();
|
|
|
|
register_alt_stack();
|
2006-12-01 09:23:37 -05:00
|
|
|
ik_fasl_load(pcb, boot_file);
|
2006-11-29 18:45:13 -05:00
|
|
|
/*
|
2006-11-23 19:44:29 -05:00
|
|
|
fprintf(stderr, "collect time: %d.%03d utime, %d.%03d stime (%d collections)\n",
|
2006-11-29 18:45:13 -05:00
|
|
|
pcb->collect_utime.tv_sec,
|
|
|
|
pcb->collect_utime.tv_usec/1000,
|
|
|
|
pcb->collect_stime.tv_sec,
|
|
|
|
pcb->collect_stime.tv_usec/1000,
|
|
|
|
pcb->collection_id );
|
|
|
|
*/
|
2006-11-23 19:38:26 -05:00
|
|
|
ik_delete_pcb(pcb);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2006-12-24 01:43:20 -05:00
|
|
|
#if 0
|
2007-09-09 20:58:47 -04:00
|
|
|
Notice how the bsd manpages have incorrect type for the handler.
|
|
|
|
|
2006-12-24 01:43:20 -05:00
|
|
|
#include <signal.h>
|
|
|
|
|
|
|
|
struct sigaction {
|
|
|
|
union {
|
|
|
|
void (*__sa_handler)(int);
|
|
|
|
void (*__sa_sigaction)(int, struct __siginfo *, void *);
|
|
|
|
} __sigaction_u; /* signal handler */
|
|
|
|
int sa_flags; /* see signal options below */
|
|
|
|
sigset_t sa_mask; /* signal mask to apply */
|
|
|
|
};
|
|
|
|
|
|
|
|
#define sa_handler __sigaction_u.__sa_handler
|
|
|
|
#define sa_sigaction __sigaction_u.__sa_sigaction
|
|
|
|
|
|
|
|
int
|
|
|
|
sigaction(int sig, const struct sigaction * restrict act,
|
|
|
|
struct sigaction * restrict oact);
|
|
|
|
#endif
|
2006-11-23 19:38:26 -05:00
|
|
|
|
2007-09-09 20:58:47 -04:00
|
|
|
void handler(int signo, siginfo_t* info, void* uap){
|
2006-12-24 03:24:53 -05:00
|
|
|
the_pcb->engine_counter = -1;
|
2006-12-24 01:43:20 -05:00
|
|
|
the_pcb->interrupted = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
register_handlers(){
|
|
|
|
struct sigaction sa;
|
2007-08-28 21:27:37 -04:00
|
|
|
sa.sa_sigaction = handler;
|
2007-10-16 02:10:51 -04:00
|
|
|
#ifdef __CYGWIN__
|
|
|
|
sa.sa_flags = SA_SIGINFO;
|
|
|
|
#else
|
2006-12-24 01:43:20 -05:00
|
|
|
sa.sa_flags = SA_SIGINFO | SA_ONSTACK;
|
2007-10-16 02:10:51 -04:00
|
|
|
#endif
|
2007-08-29 01:45:10 -04:00
|
|
|
sigemptyset(&sa.sa_mask);
|
2006-12-24 01:43:20 -05:00
|
|
|
int err = sigaction(SIGINT, &sa, 0);
|
|
|
|
if(err){
|
|
|
|
fprintf(stderr, "Sigaction Failed: %s\n", strerror(errno));
|
|
|
|
exit(-1);
|
|
|
|
}
|
|
|
|
}
|
2006-11-23 19:38:26 -05:00
|
|
|
|
2006-12-24 01:43:20 -05:00
|
|
|
|
|
|
|
#if 0
|
|
|
|
SYNOPSIS
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <signal.h>
|
|
|
|
|
|
|
|
struct sigaltstack {
|
|
|
|
char *ss_sp;
|
|
|
|
int ss_size;
|
|
|
|
int ss_flags;
|
|
|
|
};
|
|
|
|
|
|
|
|
int
|
|
|
|
sigaltstack(const struct sigaltstack *ss, struct sigaltstack *oss);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
void
|
|
|
|
register_alt_stack(){
|
2007-10-16 02:10:51 -04:00
|
|
|
#ifndef __CYGWIN__
|
2007-10-18 00:16:53 -04:00
|
|
|
char* stk = mmap(0, SIGSTKSZ, PROT_READ|PROT_WRITE|PROT_EXEC,
|
|
|
|
MAP_PRIVATE|MAP_ANON, -1, 0);
|
2007-10-15 17:58:03 -04:00
|
|
|
// char* stk = ik_mmap(SIGSTKSZ);
|
|
|
|
if(stk == (char*)-1){
|
2006-12-24 01:43:20 -05:00
|
|
|
fprintf(stderr, "Cannot maloc an alt stack\n");
|
|
|
|
exit(-1);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct sigaltstack sa;
|
|
|
|
sa.ss_sp = stk;
|
|
|
|
sa.ss_size = SIGSTKSZ;
|
|
|
|
sa.ss_flags = 0;
|
|
|
|
int err = sigaltstack(&sa, 0);
|
|
|
|
if(err){
|
|
|
|
fprintf(stderr, "Cannot set alt stack: %s\n", strerror(errno));
|
|
|
|
exit(-1);
|
|
|
|
}
|
2007-10-15 17:58:03 -04:00
|
|
|
#endif
|
2006-12-24 01:43:20 -05:00
|
|
|
}
|