1
0
Fork 0
mirror of https://github.com/ruby/ruby.git synced 2022-11-09 12:17:21 -05:00
ruby--ruby/ruby.c
matz 7ea2ceddb8 This commit was generated by cvs2svn to compensate for changes in r11,
which included commits to RCS files with non-trunk default branches.


git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@12 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
1998-01-16 12:19:22 +00:00

751 lines
13 KiB
C

/************************************************
ruby.c -
$Author$
$Date$
created at: Tue Aug 10 12:47:31 JST 1993
Copyright (C) 1993-1996 Yukihiro Matsumoto
************************************************/
#ifdef _WIN32
#include <windows.h>
#endif
#include "ruby.h"
#include "re.h"
#include "dln.h"
#include <stdio.h>
#include <ctype.h>
#include <sys/types.h>
#include <fcntl.h>
#ifdef HAVE_UNISTD_H
#include <unistd.h>
#endif
#ifndef HAVE_STRING_H
char *strchr();
char *strrchr();
char *strstr();
#endif
char *getenv();
static int version, copyright;
VALUE debug = FALSE;
VALUE verbose = FALSE;
int tainting = FALSE;
static int sflag = FALSE;
char *inplace = FALSE;
char *strdup();
extern char *sourcefile;
extern int yydebug;
extern int nerrs;
static int xflag = FALSE;
extern VALUE RS, RS_default, ORS, FS;
static void load_stdin();
static void load_file _((char *, int));
static void forbid_setid _((char *));
static VALUE do_loop = FALSE, do_print = FALSE;
static VALUE do_check = FALSE, do_line = FALSE;
static VALUE do_split = FALSE;
static char *script;
static int origargc;
static char **origargv;
extern int sourceline;
extern char *sourcefile;
#ifndef RUBY_LIB
#define RUBY_LIB "/usr/local/lib/ruby"
#endif
#if defined(MSDOS) || defined(NT)
#define RUBY_LIB_SEP ';'
#else
#define RUBY_LIB_SEP ':'
#endif
extern VALUE rb_load_path;
VALUE Frequire();
static void
addpath(path)
char *path;
{
if (path == 0) return;
if (strchr(path, RUBY_LIB_SEP)) {
char *p, *s;
VALUE ary = ary_new();
p = path;
while (*p) {
while (*p == RUBY_LIB_SEP) p++;
if (s = strchr(p, RUBY_LIB_SEP)) {
ary_push(ary, str_new(p, (int)(s-p)));
p = s + 1;
}
else {
ary_push(ary, str_new2(p));
break;
}
}
rb_load_path = ary_plus(ary, rb_load_path);
}
else {
ary_unshift(rb_load_path, str_new2(path));
}
}
struct req_list {
char *name;
struct req_list *next;
} *req_list;
static void
add_modules(mod)
char *mod;
{
struct req_list *list;
list = ALLOC(struct req_list);
list->name = mod;
list->next = req_list;
req_list = list;
}
void
rb_require_modules()
{
struct req_list *list = req_list;
struct req_list *tmp;
req_list = 0;
while (list) {
f_require(Qnil, str_new2(list->name));
tmp = list->next;
free(list);
list = tmp;
}
}
static void
proc_options(argcp, argvp)
int *argcp;
char ***argvp;
{
int argc = *argcp;
char **argv = *argvp;
int script_given, do_search;
char *s;
if (argc == 0) return;
version = FALSE;
do_search = FALSE;
script_given = 0;
for (argc--,argv++; argc > 0; argc--,argv++) {
if (argv[0][0] != '-' || !argv[0][1]) break;
s = argv[0]+1;
reswitch:
switch (*s) {
case 'a':
do_split = TRUE;
s++;
goto reswitch;
case 'p':
do_print = TRUE;
/* through */
case 'n':
do_loop = TRUE;
s++;
goto reswitch;
case 'd':
debug = TRUE;
s++;
goto reswitch;
case 'y':
yydebug = 1;
s++;
goto reswitch;
case 'v':
show_version();
verbose = 2;
case 'w':
verbose |= 1;
s++;
goto reswitch;
case 'c':
do_check = TRUE;
s++;
goto reswitch;
case 's':
forbid_setid("-s");
sflag = TRUE;
s++;
goto reswitch;
case 'l':
do_line = TRUE;
ORS = RS;
s++;
goto reswitch;
case 'S':
forbid_setid("-S");
do_search = TRUE;
s++;
goto reswitch;
case 'e':
forbid_setid("-e");
script_given++;
if (script == 0) script = "-e";
if (argv[1]) {
compile_string("-e", argv[1], strlen(argv[1]));
argc--,argv++;
}
else {
compile_string("-e", "", 0);
}
break;
case 'r':
forbid_setid("-r");
if (*++s) {
add_modules(s);
}
else if (argv[1]) {
add_modules(argv[1]);
argc--,argv++;
}
break;
case 'i':
forbid_setid("-i");
if (inplace) free(inplace);
inplace = strdup(s+1);
break;
case 'x':
xflag = TRUE;
s++;
if (*s && chdir(s) < 0) {
Fatal("Can't chdir to %s", s);
}
break;
case 'X':
s++;
if (!*s) {
s = argv[1];
argc--,argv++;
}
if (*s && chdir(s) < 0) {
Fatal("Can't chdir to %s", s);
}
break;
case 'F':
FS = str_new2(s+1);
break;
case 'K':
s++;
rb_set_kcode(s);
s++;
goto reswitch;
case 'T':
{
int numlen;
int v = 1;
if (*++s) {
v = scan_oct(s, 2, &numlen);
if (numlen == 0) v = 1;
}
rb_set_safe_level(v);
tainting = TRUE;
}
break;
case 'I':
forbid_setid("-I");
if (*++s)
addpath(s);
else if (argv[1]) {
addpath(argv[1]);
argc--,argv++;
}
break;
case '0':
{
int numlen;
int v;
char c;
v = scan_oct(s, 4, &numlen);
s += numlen;
if (v > 0377) RS = Qnil;
else if (v == 0 && numlen >= 2) {
RS = str_new2("\n\n");
}
else {
c = v & 0xff;
RS = str_new(&c, 1);
}
}
goto reswitch;
case '-':
if (!s[1]) {
argc--,argv++;
goto switch_end;
}
s++;
if (strcmp("copyright", s) == 0)
copyright = 1;
else if (strcmp("debug", s) == 0)
debug = 1;
else if (strcmp("version", s) == 0)
version = 1;
else if (strcmp("verbose", s) == 0)
verbose = 2;
else if (strcmp("yydebug", s) == 0)
yydebug = 1;
else {
Fatal("Unrecognized long option: --%s",s);
}
break;
case '*':
case ' ':
if (s[1] == '-') s+=2;
break;
default:
Fatal("Unrecognized switch: -%s",s);
case 0:
break;
}
}
switch_end:
if (*argvp[0] == 0) return;
if (version) {
show_version();
exit(0);
}
if (copyright) {
show_copyright();
}
if (script_given == FALSE) {
if (argc == 0) { /* no more args */
if (verbose == 3) exit(0);
script = "-";
load_stdin();
}
else {
script = argv[0];
if (script[0] == '\0') {
script = "-";
load_stdin();
}
else {
if (do_search) {
char *path = getenv("RUBYPATH");
script = 0;
if (path) {
script = dln_find_file(argv[0], path);
}
if (!script) {
script = dln_find_file(argv[0], getenv("PATH"));
}
if (!script) script = argv[0];
}
load_file(script, 1);
}
argc--; argv++;
}
}
if (verbose) verbose = TRUE;
xflag = FALSE;
*argvp = argv;
*argcp = argc;
if (sflag) {
char *s;
argc = *argcp; argv = *argvp;
for (; argc > 0 && argv[0][0] == '-'; argc--,argv++) {
if (argv[0][1] == '-') {
argc--,argv++;
break;
}
argv[0][0] = '$';
if (s = strchr(argv[0], '=')) {
*s++ = '\0';
rb_gvar_set2(argv[0], str_new2(s));
}
else {
rb_gvar_set2(argv[0], TRUE);
}
}
*argcp = argc; *argvp = argv;
}
}
static void
load_file(fname, script)
char *fname;
int script;
{
extern VALUE rb_stdin;
VALUE f;
int line_start = 1;
if (strcmp(fname, "-") == 0) {
f = rb_stdin;
}
else {
f = file_open(fname, "r");
}
if (script) {
VALUE c;
VALUE line;
VALUE rs = RS;
RS = RS_default;
if (xflag) {
forbid_setid("-x");
xflag = FALSE;
while (!NIL_P(line = io_gets(f))) {
line_start++;
if (RSTRING(line)->len > 2
&& RSTRING(line)->ptr[0] == '#'
&& RSTRING(line)->ptr[1] == '!') {
if (strstr(RSTRING(line)->ptr, "ruby")) {
goto start_read;
}
}
}
RS = rs;
LoadError("No Ruby script found in input");
}
c = io_getc(f);
if (c == INT2FIX('#')) {
line = io_gets(f);
line_start++;
if (RSTRING(line)->len > 2
&& RSTRING(line)->ptr[0] == '!') {
char *p;
if ((p = strstr(RSTRING(line)->ptr, "ruby")) == 0) {
/* not ruby script, kick the program */
char **argv;
char *path;
char *pend = RSTRING(line)->ptr + RSTRING(line)->len;
p = RSTRING(line)->ptr + 2; /* skip `#!' */
if (pend[-1] == '\n') pend--; /* chomp line */
if (pend[-1] == '\r') pend--;
*pend = '\0';
while (p < pend && isspace(*p))
p++;
path = p; /* interpreter path */
while (p < pend && !isspace(*p))
p++;
*p++ = '\0';
if (p < pend) {
argv = ALLOCA_N(char*, origargc+3);
argv[1] = p;
MEMCPY(argv+2, origargv+1, char*, origargc);
}
else {
argv = origargv;
}
argv[0] = path;
execv(path, argv);
sourcefile = fname;
sourceline = 1;
Fatal("Can't exec %s", path);
}
start_read:
if (p = strstr(RSTRING(line)->ptr, "ruby -")) {
int argc; char *argv[2]; char **argvp = argv;
UCHAR *s;
s = RSTRING(line)->ptr;
while (isspace(*s))
s++;
*s = '\0';
RSTRING(line)->ptr[RSTRING(line)->len-1] = '\0';
if (RSTRING(line)->ptr[RSTRING(line)->len-2] == '\r')
RSTRING(line)->ptr[RSTRING(line)->len-2] = '\0';
argc = 2; argv[0] = 0; argv[1] = p + 5;
proc_options(&argc, &argvp);
}
}
}
else if (!NIL_P(c)) {
io_ungetc(f, c);
}
RS = rs;
}
compile_file(fname, f, line_start);
if (f != rb_stdin) io_close(f);
}
void
rb_load_file(fname)
char *fname;
{
load_file(fname, 0);
}
static void
load_stdin()
{
forbid_setid("program input from stdin");
load_file("-", 1);
}
VALUE rb_progname;
VALUE rb_argv;
VALUE rb_argv0;
static void
set_arg0(val, id)
VALUE val;
ID id;
{
char *s;
int i;
static int len;
if (origargv == 0) Fail("$0 not initialized");
Check_Type(val, T_STRING);
if (len == 0) {
s = origargv[0];
s += strlen(s);
/* See if all the arguments are contiguous in memory */
for (i = 1; i < origargc; i++) {
if (origargv[i] == s + 1)
s += strlen(++s); /* this one is ok too */
}
len = s - origargv[0];
}
s = RSTRING(val)->ptr;
i = RSTRING(val)->len;
if (i > len) {
memcpy(origargv[0], s, len);
origargv[0][len] = '\0';
}
else {
memcpy(origargv[0], s, i);
s = origargv[0]+i;
*s++ = '\0';
while (++i < len)
*s++ = ' ';
}
rb_progname = str_taint(str_new2(origargv[0]));
}
void
ruby_script(name)
char *name;
{
if (name) {
rb_progname = str_taint(str_new2(name));
sourcefile = name;
}
}
static int uid, euid, gid, egid;
static void
init_ids()
{
uid = (int)getuid();
euid = (int)geteuid();
gid = (int)getgid();
egid = (int)getegid();
#ifdef VMS
uid |= gid << 16;
euid |= egid << 16;
#endif
if (uid && (euid != uid || egid != gid)) {
rb_set_safe_level(1);
}
}
static void
forbid_setid(s)
char *s;
{
if (euid != uid)
Fatal("No %s allowed while running setuid", s);
if (egid != gid)
Fatal("No %s allowed while running setgid", s);
}
#if defined(_WIN32) || defined(DJGPP)
static char *
ruby_libpath()
{
static char libpath[FILENAME_MAX+1];
char *p;
#if defined(_WIN32)
GetModuleFileName(NULL, libpath, sizeof libpath);
#elif defined(DJGPP)
extern char *__dos_argv0;
strcpy(libpath, __dos_argv0);
#endif
p = strrchr(libpath, '\\');
if (p)
*p = 0;
if (!strcasecmp(p-4, "\\bin"))
p -= 4;
strcpy(p, "\\lib");
#if defined(__CYGWIN32__)
p = (char *)malloc(strlen(libpath)+10);
if (!p)
return 0;
cygwin32_conv_to_posix_path(libpath, p);
strcpy(libpath, p);
free(p);
#else
for (p = libpath; *p; p++)
if (*p == '\\')
*p = '/';
#endif
return libpath;
}
#endif
void
ruby_prog_init()
{
init_ids();
sourcefile = "ruby";
rb_define_variable("$VERBOSE", &verbose);
rb_define_variable("$-v", &verbose);
rb_define_variable("$DEBUG", &debug);
rb_define_variable("$-d", &debug);
rb_define_readonly_variable("$-p", &do_print);
rb_define_readonly_variable("$-l", &do_line);
#if defined(_WIN32) || defined(DJGPP)
addpath(ruby_libpath());
#endif
if (rb_safe_level() == 0) {
addpath(getenv("RUBYLIB"));
}
#ifdef RUBY_THIN_ARCHLIB
addpath(RUBY_THIN_ARCHLIB);
#endif
#ifdef RUBY_ARCHLIB
addpath(RUBY_ARCHLIB);
#endif
addpath(RUBY_LIB);
if (rb_safe_level() == 0) {
addpath(".");
}
rb_define_hooked_variable("$0", &rb_progname, 0, set_arg0);
rb_argv = ary_new();
rb_define_readonly_variable("$*", &rb_argv);
rb_define_global_const("ARGV", rb_argv);
rb_define_readonly_variable("$-a", &do_split);
rb_global_variable(&rb_argv0);
#ifdef MSDOS
/*
* There is no way we can refer to them from ruby, so close them to save
* space.
*/
(void)fclose(stdaux);
(void)fclose(stdprn);
#endif
}
void
ruby_set_argv(argc, argv)
int argc;
char **argv;
{
int i;
#if defined(USE_DLN_A_OUT)
if (origargv) dln_argv0 = origargv[0];
else dln_argv0 = argv[0];
#endif
for (i=0; i < argc; i++) {
ary_push(rb_argv, str_taint(str_new2(argv[i])));
}
}
void
ruby_process_options(argc, argv)
int argc;
char **argv;
{
extern VALUE errat;
int i;
origargc = argc; origargv = argv;
ruby_script(argv[0]); /* for the time being */
rb_argv0 = str_taint(str_new2(argv[0]));
#if defined(USE_DLN_A_OUT)
dln_argv0 = argv[0];
#endif
proc_options(&argc, &argv);
ruby_script(script);
ruby_set_argv(argc, argv);
if (do_check && nerrs == 0) {
printf("Syntax OK\n");
exit(0);
}
if (do_print) {
yyappend_print();
}
if (do_loop) {
yywhile_loop(do_line, do_split);
}
}