2011-11-14 23:26:45 +01:00
|
|
|
// Copyright 2009 The Go Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#include "runtime.h"
|
|
|
|
#include "array.h"
|
|
|
|
#include "go-panic.h"
|
|
|
|
#include "go-string.h"
|
|
|
|
|
|
|
|
uint32 runtime_panicking;
|
|
|
|
|
|
|
|
static Lock paniclk;
|
|
|
|
|
|
|
|
void
|
|
|
|
runtime_startpanic(void)
|
|
|
|
{
|
2011-11-28 06:45:49 +01:00
|
|
|
M *m;
|
|
|
|
|
|
|
|
m = runtime_m();
|
2011-11-14 23:26:45 +01:00
|
|
|
if(m->dying) {
|
|
|
|
runtime_printf("panic during panic\n");
|
|
|
|
runtime_exit(3);
|
|
|
|
}
|
|
|
|
m->dying = 1;
|
|
|
|
runtime_xadd(&runtime_panicking, 1);
|
|
|
|
runtime_lock(&paniclk);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
runtime_dopanic(int32 unused __attribute__ ((unused)))
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
static bool didothers;
|
|
|
|
|
|
|
|
if(g->sig != 0)
|
|
|
|
runtime_printf("[signal %x code=%p addr=%p pc=%p]\n",
|
|
|
|
g->sig, g->sigcode0, g->sigcode1, g->sigpc);
|
|
|
|
|
|
|
|
if(runtime_gotraceback()){
|
|
|
|
if(!didothers) {
|
|
|
|
didothers = true;
|
|
|
|
runtime_tracebackothers(g);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
|
|
|
|
runtime_unlock(&paniclk);
|
|
|
|
if(runtime_xadd(&runtime_panicking, -1) != 0) {
|
|
|
|
// Some other m is panicking too.
|
|
|
|
// Let it print what it needs to print.
|
|
|
|
// Wait forever without chewing up cpu.
|
|
|
|
// It will exit when it's done.
|
|
|
|
static Lock deadlock;
|
|
|
|
runtime_lock(&deadlock);
|
|
|
|
runtime_lock(&deadlock);
|
|
|
|
}
|
|
|
|
|
|
|
|
runtime_exit(2);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
runtime_throw(const char *s)
|
|
|
|
{
|
|
|
|
runtime_startpanic();
|
|
|
|
runtime_printf("throw: %s\n", s);
|
|
|
|
runtime_dopanic(0);
|
|
|
|
*(int32*)0 = 0; // not reached
|
|
|
|
runtime_exit(1); // even more not reached
|
|
|
|
}
|
|
|
|
|
2011-11-30 01:21:52 +01:00
|
|
|
void
|
|
|
|
runtime_panicstring(const char *s)
|
|
|
|
{
|
|
|
|
Eface err;
|
|
|
|
|
|
|
|
if(runtime_m()->gcing) {
|
|
|
|
runtime_printf("panic: %s\n", s);
|
|
|
|
runtime_throw("panic during gc");
|
|
|
|
}
|
|
|
|
runtime_newErrorString(runtime_gostringnocopy((const byte*)s), &err);
|
|
|
|
runtime_panic(err);
|
|
|
|
}
|
|
|
|
|
2011-11-14 23:26:45 +01:00
|
|
|
static int32 argc;
|
|
|
|
static byte** argv;
|
|
|
|
|
|
|
|
extern Slice os_Args asm ("libgo_os.os.Args");
|
2011-12-13 00:40:51 +01:00
|
|
|
extern Slice syscall_Envs asm ("libgo_syscall.syscall.Envs");
|
2011-11-14 23:26:45 +01:00
|
|
|
|
|
|
|
void
|
|
|
|
runtime_args(int32 c, byte **v)
|
|
|
|
{
|
|
|
|
argc = c;
|
|
|
|
argv = v;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
runtime_goargs(void)
|
|
|
|
{
|
|
|
|
String *s;
|
|
|
|
int32 i;
|
|
|
|
|
|
|
|
// for windows implementation see "os" package
|
|
|
|
if(Windows)
|
|
|
|
return;
|
|
|
|
|
|
|
|
s = runtime_malloc(argc*sizeof s[0]);
|
|
|
|
for(i=0; i<argc; i++)
|
2011-11-30 01:21:52 +01:00
|
|
|
s[i] = runtime_gostringnocopy((const byte*)argv[i]);
|
2011-11-14 23:26:45 +01:00
|
|
|
os_Args.__values = (void*)s;
|
|
|
|
os_Args.__count = argc;
|
|
|
|
os_Args.__capacity = argc;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
runtime_goenvs(void)
|
|
|
|
{
|
|
|
|
String *s;
|
|
|
|
int32 i, n;
|
|
|
|
|
|
|
|
for(n=0; argv[argc+1+n] != 0; n++)
|
|
|
|
;
|
|
|
|
|
|
|
|
s = runtime_malloc(n*sizeof s[0]);
|
|
|
|
for(i=0; i<n; i++)
|
|
|
|
s[i] = runtime_gostringnocopy(argv[argc+1+i]);
|
2011-12-13 00:40:51 +01:00
|
|
|
syscall_Envs.__values = (void*)s;
|
|
|
|
syscall_Envs.__count = n;
|
|
|
|
syscall_Envs.__capacity = n;
|
2011-11-14 23:26:45 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
const byte*
|
|
|
|
runtime_getenv(const char *s)
|
|
|
|
{
|
|
|
|
int32 i, j, len;
|
|
|
|
const byte *v, *bs;
|
|
|
|
String* envv;
|
|
|
|
int32 envc;
|
|
|
|
|
|
|
|
bs = (const byte*)s;
|
|
|
|
len = runtime_findnull(bs);
|
2011-12-13 00:40:51 +01:00
|
|
|
envv = (String*)syscall_Envs.__values;
|
|
|
|
envc = syscall_Envs.__count;
|
2011-11-14 23:26:45 +01:00
|
|
|
for(i=0; i<envc; i++){
|
|
|
|
if(envv[i].__length <= len)
|
|
|
|
continue;
|
|
|
|
v = (const byte*)envv[i].__data;
|
|
|
|
for(j=0; j<len; j++)
|
|
|
|
if(bs[j] != v[j])
|
|
|
|
goto nomatch;
|
|
|
|
if(v[len] != '=')
|
|
|
|
goto nomatch;
|
|
|
|
return v+len+1;
|
|
|
|
nomatch:;
|
|
|
|
}
|
|
|
|
return nil;
|
|
|
|
}
|
|
|
|
|
|
|
|
int32
|
|
|
|
runtime_atoi(const byte *p)
|
|
|
|
{
|
|
|
|
int32 n;
|
|
|
|
|
|
|
|
n = 0;
|
|
|
|
while('0' <= *p && *p <= '9')
|
|
|
|
n = n*10 + *p++ - '0';
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32
|
|
|
|
runtime_fastrand1(void)
|
|
|
|
{
|
2011-11-28 06:45:49 +01:00
|
|
|
M *m;
|
2011-11-14 23:26:45 +01:00
|
|
|
uint32 x;
|
|
|
|
|
2011-11-28 06:45:49 +01:00
|
|
|
m = runtime_m();
|
2011-11-14 23:26:45 +01:00
|
|
|
x = m->fastrand;
|
|
|
|
x += x;
|
|
|
|
if(x & 0x80000000L)
|
|
|
|
x ^= 0x88888eefUL;
|
|
|
|
m->fastrand = x;
|
|
|
|
return x;
|
|
|
|
}
|