2012-10-23 06:31:11 +02: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.
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include "go-assert.h"
|
2014-05-29 01:10:47 +02:00
|
|
|
#include <complex.h>
|
2010-12-03 05:34:57 +01:00
|
|
|
#include <signal.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2011-01-29 08:16:20 +01:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <fcntl.h>
|
2012-05-24 22:44:34 +02:00
|
|
|
#include <unistd.h>
|
2010-12-03 05:34:57 +01:00
|
|
|
#include <pthread.h>
|
2010-12-17 07:42:06 +01:00
|
|
|
#include <semaphore.h>
|
2011-11-28 06:45:49 +01:00
|
|
|
#include <ucontext.h>
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
#ifdef HAVE_SYS_MMAN_H
|
|
|
|
#include <sys/mman.h>
|
|
|
|
#endif
|
|
|
|
|
2012-11-01 04:02:13 +01:00
|
|
|
#include "interface.h"
|
2010-12-03 05:34:57 +01:00
|
|
|
#include "go-alloc.h"
|
|
|
|
|
2013-01-24 20:44:23 +01:00
|
|
|
#define _STRINGIFY2_(x) #x
|
|
|
|
#define _STRINGIFY_(x) _STRINGIFY2_(x)
|
|
|
|
#define GOSYM_PREFIX _STRINGIFY_(__USER_LABEL_PREFIX__)
|
|
|
|
|
2010-12-03 05:34:57 +01:00
|
|
|
/* This file supports C files copied from the 6g runtime library.
|
|
|
|
This is a version of the 6g runtime.h rewritten for gccgo's version
|
|
|
|
of the code. */
|
|
|
|
|
|
|
|
typedef signed int int8 __attribute__ ((mode (QI)));
|
|
|
|
typedef unsigned int uint8 __attribute__ ((mode (QI)));
|
|
|
|
typedef signed int int16 __attribute__ ((mode (HI)));
|
|
|
|
typedef unsigned int uint16 __attribute__ ((mode (HI)));
|
|
|
|
typedef signed int int32 __attribute__ ((mode (SI)));
|
|
|
|
typedef unsigned int uint32 __attribute__ ((mode (SI)));
|
|
|
|
typedef signed int int64 __attribute__ ((mode (DI)));
|
|
|
|
typedef unsigned int uint64 __attribute__ ((mode (DI)));
|
|
|
|
typedef float float32 __attribute__ ((mode (SF)));
|
|
|
|
typedef double float64 __attribute__ ((mode (DF)));
|
2012-10-23 06:31:11 +02:00
|
|
|
typedef signed int intptr __attribute__ ((mode (pointer)));
|
2010-12-03 05:34:57 +01:00
|
|
|
typedef unsigned int uintptr __attribute__ ((mode (pointer)));
|
|
|
|
|
2012-11-06 19:46:38 +01:00
|
|
|
typedef intptr intgo; // Go's int
|
|
|
|
typedef uintptr uintgo; // Go's uint
|
2012-10-23 06:31:11 +02:00
|
|
|
|
2014-07-19 10:53:52 +02:00
|
|
|
typedef uintptr uintreg;
|
|
|
|
|
2010-12-03 05:34:57 +01:00
|
|
|
/* Defined types. */
|
|
|
|
|
|
|
|
typedef uint8 bool;
|
|
|
|
typedef uint8 byte;
|
2012-03-07 02:16:20 +01:00
|
|
|
typedef struct Func Func;
|
2016-08-30 23:07:47 +02:00
|
|
|
typedef struct g G;
|
|
|
|
typedef struct mutex Lock;
|
|
|
|
typedef struct m M;
|
|
|
|
typedef struct p P;
|
|
|
|
typedef struct note Note;
|
2013-07-16 08:54:42 +02:00
|
|
|
typedef struct String String;
|
2013-06-19 01:49:49 +02:00
|
|
|
typedef struct FuncVal FuncVal;
|
2011-12-21 23:24:47 +01:00
|
|
|
typedef struct SigTab SigTab;
|
2016-08-30 23:07:47 +02:00
|
|
|
typedef struct mcache MCache;
|
2011-09-16 17:47:21 +02:00
|
|
|
typedef struct FixAlloc FixAlloc;
|
2016-10-10 18:52:09 +02:00
|
|
|
typedef struct hchan Hchan;
|
2011-12-13 00:40:51 +01:00
|
|
|
typedef struct Timers Timers;
|
|
|
|
typedef struct Timer Timer;
|
2016-08-30 23:07:47 +02:00
|
|
|
typedef struct gcstats GCStats;
|
2013-07-16 08:54:42 +02:00
|
|
|
typedef struct LFNode LFNode;
|
|
|
|
typedef struct ParFor ParFor;
|
|
|
|
typedef struct ParForThread ParForThread;
|
2016-08-30 23:07:47 +02:00
|
|
|
typedef struct cgoMal CgoMal;
|
2013-07-16 08:54:42 +02:00
|
|
|
typedef struct PollDesc PollDesc;
|
2016-10-10 18:52:09 +02:00
|
|
|
typedef struct sudog SudoG;
|
2010-12-03 05:34:57 +01:00
|
|
|
|
2011-12-01 09:06:16 +01:00
|
|
|
typedef struct __go_open_array Slice;
|
2011-11-30 01:21:52 +01:00
|
|
|
typedef struct __go_interface Iface;
|
|
|
|
typedef struct __go_empty_interface Eface;
|
|
|
|
typedef struct __go_type_descriptor Type;
|
2016-08-30 23:07:47 +02:00
|
|
|
typedef struct _defer Defer;
|
|
|
|
typedef struct _panic Panic;
|
2011-11-11 22:02:48 +01:00
|
|
|
|
2012-11-21 08:03:38 +01:00
|
|
|
typedef struct __go_ptr_type PtrType;
|
2011-11-30 01:21:52 +01:00
|
|
|
typedef struct __go_func_type FuncType;
|
2013-11-06 20:49:01 +01:00
|
|
|
typedef struct __go_interface_type InterfaceType;
|
2011-11-30 01:21:52 +01:00
|
|
|
typedef struct __go_map_type MapType;
|
2013-07-16 08:54:42 +02:00
|
|
|
typedef struct __go_channel_type ChanType;
|
2011-11-30 01:21:52 +01:00
|
|
|
|
2016-08-30 23:07:47 +02:00
|
|
|
typedef struct traceback Traceback;
|
2012-05-24 23:07:18 +02:00
|
|
|
|
2016-08-30 23:07:47 +02:00
|
|
|
typedef struct location Location;
|
|
|
|
|
|
|
|
struct String
|
|
|
|
{
|
|
|
|
const byte* str;
|
|
|
|
intgo len;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct FuncVal
|
|
|
|
{
|
|
|
|
void (*fn)(void);
|
|
|
|
// variable-size, fn-specific data here
|
|
|
|
};
|
|
|
|
|
|
|
|
#include "array.h"
|
|
|
|
#include "interface.h"
|
|
|
|
|
2016-09-30 15:45:08 +02:00
|
|
|
// Rename Go types generated by mkrsysinfo.sh from C types, to avoid
|
|
|
|
// the name conflict.
|
|
|
|
#define timeval go_timeval
|
|
|
|
#define timespec go_timespec
|
|
|
|
|
2016-08-30 23:07:47 +02:00
|
|
|
#include "runtime.inc"
|
2013-01-30 23:24:40 +01:00
|
|
|
|
2016-09-30 15:45:08 +02:00
|
|
|
#undef timeval
|
|
|
|
#undef timespec
|
|
|
|
|
2011-11-28 06:45:49 +01:00
|
|
|
/*
|
2012-11-21 08:03:38 +01:00
|
|
|
* Per-CPU declaration.
|
2011-11-28 06:45:49 +01:00
|
|
|
*/
|
|
|
|
extern M* runtime_m(void);
|
2016-09-29 02:56:44 +02:00
|
|
|
extern G* runtime_g(void)
|
|
|
|
__asm__(GOSYM_PREFIX "runtime.getg");
|
2010-12-03 05:34:57 +01:00
|
|
|
|
2011-11-11 22:02:48 +01:00
|
|
|
extern M runtime_m0;
|
|
|
|
extern G runtime_g0;
|
2010-12-03 05:34:57 +01:00
|
|
|
|
2013-07-16 08:54:42 +02:00
|
|
|
enum
|
2010-12-03 05:34:57 +01:00
|
|
|
{
|
|
|
|
true = 1,
|
|
|
|
false = 0,
|
|
|
|
};
|
2012-10-23 06:31:11 +02:00
|
|
|
enum
|
|
|
|
{
|
|
|
|
PtrSize = sizeof(void*),
|
|
|
|
};
|
2013-01-29 21:52:43 +01:00
|
|
|
enum
|
|
|
|
{
|
|
|
|
// Per-M stack segment cache size.
|
|
|
|
StackCacheSize = 32,
|
|
|
|
// Global <-> per-M stack segment cache transfer batch size.
|
|
|
|
StackCacheBatch = 16,
|
|
|
|
};
|
2013-07-16 08:54:42 +02:00
|
|
|
|
2011-12-21 23:24:47 +01:00
|
|
|
struct SigTab
|
|
|
|
{
|
|
|
|
int32 sig;
|
|
|
|
int32 flags;
|
2016-02-03 22:58:02 +01:00
|
|
|
void* fwdsig;
|
2011-12-21 23:24:47 +01:00
|
|
|
};
|
|
|
|
|
2013-11-06 20:49:01 +01:00
|
|
|
// Layout of in-memory per-function information prepared by linker
|
|
|
|
// See http://golang.org/s/go12symtab.
|
|
|
|
// Keep in sync with linker and with ../../libmach/sym.c
|
|
|
|
// and with package debug/gosym.
|
2012-03-07 02:16:20 +01:00
|
|
|
struct Func
|
|
|
|
{
|
|
|
|
String name;
|
|
|
|
uintptr entry; // entry pc
|
|
|
|
};
|
|
|
|
|
2014-07-19 10:53:52 +02:00
|
|
|
#ifdef GOOS_nacl
|
|
|
|
enum {
|
|
|
|
NaCl = 1,
|
|
|
|
};
|
|
|
|
#else
|
|
|
|
enum {
|
|
|
|
NaCl = 0,
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
2012-01-13 06:11:45 +01:00
|
|
|
#ifdef GOOS_windows
|
2011-11-14 23:26:45 +01:00
|
|
|
enum {
|
|
|
|
Windows = 1
|
|
|
|
};
|
|
|
|
#else
|
|
|
|
enum {
|
|
|
|
Windows = 0
|
|
|
|
};
|
|
|
|
#endif
|
2014-06-07 00:37:27 +02:00
|
|
|
#ifdef GOOS_solaris
|
|
|
|
enum {
|
|
|
|
Solaris = 1
|
|
|
|
};
|
|
|
|
#else
|
|
|
|
enum {
|
|
|
|
Solaris = 0
|
|
|
|
};
|
|
|
|
#endif
|
2011-11-14 23:26:45 +01:00
|
|
|
|
2011-12-13 00:40:51 +01:00
|
|
|
struct Timers
|
|
|
|
{
|
|
|
|
Lock;
|
|
|
|
G *timerproc;
|
|
|
|
bool sleeping;
|
|
|
|
bool rescheduling;
|
|
|
|
Note waitnote;
|
|
|
|
Timer **t;
|
|
|
|
int32 len;
|
|
|
|
int32 cap;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Package time knows the layout of this structure.
|
|
|
|
// If this struct changes, adjust ../time/sleep.go:/runtimeTimer.
|
2014-07-19 10:53:52 +02:00
|
|
|
// For GOOS=nacl, package syscall knows the layout of this structure.
|
|
|
|
// If this struct changes, adjust ../syscall/net_nacl.go:/runtimeTimer.
|
2011-12-13 00:40:51 +01:00
|
|
|
struct Timer
|
|
|
|
{
|
2015-01-15 01:27:56 +01:00
|
|
|
intgo i; // heap index
|
2011-12-13 00:40:51 +01:00
|
|
|
|
|
|
|
// Timer wakes up at when, and then at when+period, ... (period > 0 only)
|
|
|
|
// each time calling f(now, arg) in the timer goroutine, so f must be
|
|
|
|
// a well-behaved function and not block.
|
|
|
|
int64 when;
|
|
|
|
int64 period;
|
2013-06-19 01:49:49 +02:00
|
|
|
FuncVal *fv;
|
2011-12-13 00:40:51 +01:00
|
|
|
Eface arg;
|
2015-01-15 01:27:56 +01:00
|
|
|
uintptr seq;
|
2011-12-13 00:40:51 +01:00
|
|
|
};
|
|
|
|
|
2012-10-23 06:31:11 +02:00
|
|
|
// Lock-free stack node.
|
|
|
|
struct LFNode
|
|
|
|
{
|
|
|
|
LFNode *next;
|
|
|
|
uintptr pushcnt;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Parallel for descriptor.
|
|
|
|
struct ParFor
|
|
|
|
{
|
2015-10-31 01:59:47 +01:00
|
|
|
const FuncVal *body; // executed for each element
|
2012-10-23 06:31:11 +02:00
|
|
|
uint32 done; // number of idle threads
|
|
|
|
uint32 nthr; // total number of threads
|
|
|
|
uint32 nthrmax; // maximum number of threads
|
|
|
|
uint32 thrseq; // thread id sequencer
|
|
|
|
uint32 cnt; // iteration space [0, cnt)
|
|
|
|
bool wait; // if true, wait while all threads finish processing,
|
|
|
|
// otherwise parfor may return while other threads are still working
|
|
|
|
ParForThread *thr; // array of thread descriptors
|
|
|
|
// stats
|
|
|
|
uint64 nsteal;
|
|
|
|
uint64 nstealcnt;
|
|
|
|
uint64 nprocyield;
|
|
|
|
uint64 nosyield;
|
|
|
|
uint64 nsleep;
|
|
|
|
};
|
|
|
|
|
2013-11-06 20:49:01 +01:00
|
|
|
extern bool runtime_precisestack;
|
2014-07-19 10:53:52 +02:00
|
|
|
extern bool runtime_copystack;
|
2013-11-06 20:49:01 +01:00
|
|
|
|
2011-12-13 00:40:51 +01:00
|
|
|
/*
|
|
|
|
* defined macros
|
|
|
|
* you need super-gopher-guru privilege
|
|
|
|
* to add this list.
|
|
|
|
*/
|
2010-12-03 05:34:57 +01:00
|
|
|
#define nelem(x) (sizeof(x)/sizeof((x)[0]))
|
|
|
|
#define nil ((void*)0)
|
|
|
|
#define USED(v) ((void) v)
|
2014-07-12 02:01:09 +02:00
|
|
|
#define ROUND(x, n) (((x)+(n)-1)&~(uintptr)((n)-1)) /* all-caps to mark as macro: it evaluates n twice */
|
2010-12-03 05:34:57 +01:00
|
|
|
|
2013-07-16 08:54:42 +02:00
|
|
|
byte* runtime_startup_random_data;
|
|
|
|
uint32 runtime_startup_random_data_len;
|
|
|
|
void runtime_get_random_data(byte**, int32*);
|
|
|
|
|
|
|
|
enum {
|
|
|
|
// hashinit wants this many random bytes
|
|
|
|
HashRandomBytes = 32
|
|
|
|
};
|
|
|
|
void runtime_hashinit(void);
|
|
|
|
|
2013-12-03 15:51:07 +01:00
|
|
|
void runtime_traceback(void);
|
2013-07-16 08:54:42 +02:00
|
|
|
void runtime_tracebackothers(G*);
|
2014-06-07 00:37:27 +02:00
|
|
|
enum
|
|
|
|
{
|
|
|
|
// The maximum number of frames we print for a traceback
|
|
|
|
TracebackMaxFrames = 100,
|
|
|
|
};
|
2013-07-16 08:54:42 +02:00
|
|
|
|
2011-11-14 23:26:45 +01:00
|
|
|
/*
|
|
|
|
* external data
|
|
|
|
*/
|
2012-11-21 08:03:38 +01:00
|
|
|
extern uintptr runtime_zerobase;
|
2014-06-07 00:37:27 +02:00
|
|
|
extern G** runtime_allg;
|
|
|
|
extern uintptr runtime_allglen;
|
2013-01-29 21:52:43 +01:00
|
|
|
extern G* runtime_lastg;
|
|
|
|
extern M* runtime_allm;
|
2013-07-16 08:54:42 +02:00
|
|
|
extern P** runtime_allp;
|
2011-11-28 06:45:49 +01:00
|
|
|
extern int32 runtime_gomaxprocs;
|
2013-07-23 22:26:09 +02:00
|
|
|
extern uint32 runtime_needextram;
|
2011-11-14 23:26:45 +01:00
|
|
|
extern uint32 runtime_panicking;
|
2013-07-16 08:54:42 +02:00
|
|
|
extern int8* runtime_goos;
|
2013-01-29 21:52:43 +01:00
|
|
|
extern int32 runtime_ncpu;
|
2013-07-16 08:54:42 +02:00
|
|
|
extern void (*runtime_sysargs)(int32, uint8**);
|
2016-09-29 02:56:44 +02:00
|
|
|
extern struct debugVars runtime_debug;
|
2014-07-12 02:01:09 +02:00
|
|
|
extern uintptr runtime_maxstacksize;
|
2011-11-14 23:26:45 +01:00
|
|
|
|
2015-04-29 23:31:53 +02:00
|
|
|
extern bool runtime_isstarted;
|
|
|
|
extern bool runtime_isarchive;
|
|
|
|
|
2011-11-14 23:26:45 +01:00
|
|
|
/*
|
|
|
|
* common functions and data
|
|
|
|
*/
|
2013-07-16 08:54:42 +02:00
|
|
|
#define runtime_strcmp(s1, s2) __builtin_strcmp((s1), (s2))
|
2014-07-19 10:53:52 +02:00
|
|
|
#define runtime_strncmp(s1, s2, n) __builtin_strncmp((s1), (s2), (n))
|
2013-07-16 08:54:42 +02:00
|
|
|
#define runtime_strstr(s1, s2) __builtin_strstr((s1), (s2))
|
2012-11-06 19:28:21 +01:00
|
|
|
intgo runtime_findnull(const byte*);
|
2013-11-06 20:49:01 +01:00
|
|
|
intgo runtime_findnullw(const uint16*);
|
2010-12-03 05:34:57 +01:00
|
|
|
|
2013-11-06 20:49:01 +01:00
|
|
|
void runtime_gogo(G*);
|
2013-07-16 08:54:42 +02:00
|
|
|
struct __go_func_type;
|
2016-09-29 02:56:44 +02:00
|
|
|
void runtime_args(int32, byte**)
|
|
|
|
__asm__ (GOSYM_PREFIX "runtime.args");
|
2011-11-22 21:24:44 +01:00
|
|
|
void runtime_osinit();
|
2016-09-29 02:56:44 +02:00
|
|
|
void runtime_goargs(void)
|
|
|
|
__asm__ (GOSYM_PREFIX "runtime.goargs");
|
2011-11-14 23:26:45 +01:00
|
|
|
void runtime_goenvs(void);
|
2016-09-29 02:56:44 +02:00
|
|
|
void runtime_goenvs_unix(void)
|
|
|
|
__asm__ (GOSYM_PREFIX "runtime.goenvs_unix");
|
2011-12-21 23:24:47 +01:00
|
|
|
void runtime_throw(const char*) __attribute__ ((noreturn));
|
2011-11-30 01:21:52 +01:00
|
|
|
void runtime_panicstring(const char*) __attribute__ ((noreturn));
|
2014-07-19 10:53:52 +02:00
|
|
|
bool runtime_canpanic(G*);
|
2012-05-24 22:44:34 +02:00
|
|
|
void runtime_printf(const char*, ...);
|
2014-07-19 10:53:52 +02:00
|
|
|
int32 runtime_snprintf(byte*, int32, const char*, ...);
|
2013-07-16 08:54:42 +02:00
|
|
|
#define runtime_mcmp(a, b, s) __builtin_memcmp((a), (b), (s))
|
|
|
|
#define runtime_memmove(a, b, s) __builtin_memmove((a), (b), (s))
|
2010-12-03 05:34:57 +01:00
|
|
|
void* runtime_mal(uintptr);
|
2013-07-16 08:54:42 +02:00
|
|
|
String runtime_gostring(const byte*);
|
|
|
|
String runtime_gostringnocopy(const byte*);
|
2011-11-28 06:45:49 +01:00
|
|
|
void runtime_schedinit(void);
|
2016-02-12 23:10:09 +01:00
|
|
|
void runtime_initsig(bool);
|
2012-03-02 21:01:37 +01:00
|
|
|
void runtime_sigenable(uint32 sig);
|
2013-07-16 08:54:42 +02:00
|
|
|
void runtime_sigdisable(uint32 sig);
|
2015-10-31 01:59:47 +01:00
|
|
|
void runtime_sigignore(uint32 sig);
|
2013-07-16 08:54:42 +02:00
|
|
|
int32 runtime_gotraceback(bool *crash);
|
2012-05-17 07:30:25 +02:00
|
|
|
void runtime_goroutineheader(G*);
|
2013-01-30 23:24:40 +01:00
|
|
|
void runtime_printtrace(Location*, int32, bool);
|
2013-07-16 08:54:42 +02:00
|
|
|
#define runtime_open(p, f, m) open((p), (f), (m))
|
|
|
|
#define runtime_read(d, v, n) read((d), (v), (n))
|
|
|
|
#define runtime_write(d, v, n) write((d), (v), (n))
|
|
|
|
#define runtime_close(d) close(d)
|
|
|
|
void runtime_ready(G*);
|
2015-10-31 01:59:47 +01:00
|
|
|
String runtime_getenv(const char*);
|
|
|
|
int32 runtime_atoi(const byte*, intgo);
|
2011-11-28 06:45:49 +01:00
|
|
|
void* runtime_mstart(void*);
|
2016-08-30 23:07:47 +02:00
|
|
|
G* runtime_malg(int32, byte**, uintptr*);
|
2013-07-16 08:54:42 +02:00
|
|
|
void runtime_mpreinit(M*);
|
2011-11-28 06:45:49 +01:00
|
|
|
void runtime_minit(void);
|
2013-07-16 08:54:42 +02:00
|
|
|
void runtime_unminit(void);
|
2013-07-23 22:26:09 +02:00
|
|
|
void runtime_needm(void);
|
|
|
|
void runtime_dropm(void);
|
2013-07-16 08:54:42 +02:00
|
|
|
void runtime_signalstack(byte*, int32);
|
|
|
|
MCache* runtime_allocmcache(void);
|
|
|
|
void runtime_freemcache(MCache*);
|
2010-12-03 05:34:57 +01:00
|
|
|
void runtime_mallocinit(void);
|
2013-07-16 08:54:42 +02:00
|
|
|
void runtime_mprofinit(void);
|
|
|
|
#define runtime_malloc(s) __go_alloc(s)
|
|
|
|
#define runtime_free(p) __go_free(p)
|
|
|
|
#define runtime_getcallersp(p) __builtin_frame_address(1)
|
|
|
|
int32 runtime_mcount(void);
|
|
|
|
int32 runtime_gcount(void);
|
2013-11-06 20:49:01 +01:00
|
|
|
void runtime_mcall(void(*)(G*));
|
compiler, runtime: replace hashmap code with Go 1.7 hashmap
This change removes the gccgo-specific hashmap code and replaces it with
the hashmap code from the Go 1.7 runtime. The Go 1.7 hashmap code is
more efficient, does a better job on details like when to update a key,
and provides some support against denial-of-service attacks.
The compiler is changed to call the new hashmap functions instead of the
old ones.
The compiler now tracks which types are reflexive and which require
updating when used as a map key, and records the information in map type
descriptors.
Map_index_expression is simplified. The special case for a map index on
the right hand side of a tuple expression has been unnecessary for some
time, and is removed. The support for specially marking a map index as
an lvalue is removed, in favor of lowering an assignment to a map index
into a function call. The long-obsolete support for a map index of a
pointer to a map is removed.
The __go_new_map_big function (known to the compiler as
Runtime::MAKEMAPBIG) is no longer needed, as the new runtime.makemap
function takes an int64 hint argument.
The old map descriptor type and supporting expression is removed.
The compiler was still supporting the long-obsolete syntax `m[k] = 0,
false` to delete a value from a map. That is now removed, requiring a
change to one of the gccgo-specific tests.
The builtin len function applied to a map or channel p is now compiled
as `p == nil ? 0 : *(*int)(p)`. The __go_chan_len function (known to
the compiler as Runtime::CHAN_LEN) is removed.
Support for a shared zero value for maps to large value types is
introduced, along the lines of the gc compiler. The zero value is
handled as a common variable.
The hash function is changed to take a seed argument, changing the
runtime hash functions and the compiler-generated hash functions.
Unlike the gc compiler, both the hash and equal functions continue to
take the type length.
Types that can not be compared now store nil for the hash and equal
functions, rather than pointing to functions that throw. Interface hash
and comparison functions now check explicitly for nil. This matches the
gc compiler and permits a simple implementation for ismapkey.
The compiler is changed to permit marking struct and array types as
incomparable, meaning that they have no hash or equal function. We use
this for thunk types, removing the existing special code to avoid
generating hash/equal functions for them.
The C runtime code adds memclr, memequal, and memmove functions.
The hashmap code uses go:linkname comments to make the functions
visible, as otherwise the compiler would discard them.
The hashmap code comments out the unused reference to the address of the
first parameter in the race code, as otherwise the compiler thinks that
the parameter escapes and copies it onto the heap. This is probably not
needed when we enable escape analysis.
Several runtime map tests that ere previously skipped for gccgo are now
run.
The Go runtime picks up type kind information and stubs. The type kind
information causes the generated runtime header file to define some
constants, including `empty`, and the C code is adjusted accordingly.
A Go-callable version of runtime.throw, that takes a Go string, is
added to be called from the hashmap code.
Reviewed-on: https://go-review.googlesource.com/29447
* go.go-torture/execute/map-1.go: Replace old map deletion syntax
with call to builtin delete function.
From-SVN: r240334
2016-09-21 22:58:51 +02:00
|
|
|
uint32 runtime_fastrand1(void) __asm__ (GOSYM_PREFIX "runtime.fastrand1");
|
2016-09-29 02:56:44 +02:00
|
|
|
int32 runtime_timediv(int64, int32, int32*)
|
|
|
|
__asm__ (GOSYM_PREFIX "runtime.timediv");
|
2014-07-19 10:53:52 +02:00
|
|
|
int32 runtime_round2(int32 x); // round x up to a power of 2.
|
2013-07-16 08:54:42 +02:00
|
|
|
|
2014-06-07 00:37:27 +02:00
|
|
|
// atomic operations
|
|
|
|
#define runtime_cas(pval, old, new) __sync_bool_compare_and_swap (pval, old, new)
|
|
|
|
#define runtime_cas64(pval, old, new) __sync_bool_compare_and_swap (pval, old, new)
|
|
|
|
#define runtime_casp(pval, old, new) __sync_bool_compare_and_swap (pval, old, new)
|
|
|
|
// Don't confuse with XADD x86 instruction,
|
|
|
|
// this one is actually 'addx', that is, add-and-fetch.
|
|
|
|
#define runtime_xadd(p, v) __sync_add_and_fetch (p, v)
|
|
|
|
#define runtime_xadd64(p, v) __sync_add_and_fetch (p, v)
|
|
|
|
#define runtime_xchg(p, v) __atomic_exchange_n (p, v, __ATOMIC_SEQ_CST)
|
|
|
|
#define runtime_xchg64(p, v) __atomic_exchange_n (p, v, __ATOMIC_SEQ_CST)
|
|
|
|
#define runtime_xchgp(p, v) __atomic_exchange_n (p, v, __ATOMIC_SEQ_CST)
|
|
|
|
#define runtime_atomicload(p) __atomic_load_n (p, __ATOMIC_SEQ_CST)
|
|
|
|
#define runtime_atomicstore(p, v) __atomic_store_n (p, v, __ATOMIC_SEQ_CST)
|
|
|
|
#define runtime_atomicstore64(p, v) __atomic_store_n (p, v, __ATOMIC_SEQ_CST)
|
|
|
|
#define runtime_atomicload64(p) __atomic_load_n (p, __ATOMIC_SEQ_CST)
|
|
|
|
#define runtime_atomicloadp(p) __atomic_load_n (p, __ATOMIC_SEQ_CST)
|
|
|
|
#define runtime_atomicstorep(p, v) __atomic_store_n (p, v, __ATOMIC_SEQ_CST)
|
|
|
|
|
2016-08-30 23:07:47 +02:00
|
|
|
void runtime_setg(G*);
|
2013-07-16 08:54:42 +02:00
|
|
|
void runtime_newextram(void);
|
|
|
|
#define runtime_exit(s) exit(s)
|
|
|
|
#define runtime_breakpoint() __builtin_trap()
|
2011-11-28 06:45:49 +01:00
|
|
|
void runtime_gosched(void);
|
2013-11-06 20:49:01 +01:00
|
|
|
void runtime_gosched0(G*);
|
|
|
|
void runtime_schedtrace(bool);
|
2014-06-07 00:37:27 +02:00
|
|
|
void runtime_park(bool(*)(G*, void*), void*, const char*);
|
|
|
|
void runtime_parkunlock(Lock*, const char*);
|
2012-10-23 06:31:11 +02:00
|
|
|
void runtime_tsleep(int64, const char*);
|
2011-12-13 00:40:51 +01:00
|
|
|
M* runtime_newm(void);
|
2011-11-28 06:45:49 +01:00
|
|
|
void runtime_goexit(void);
|
2016-09-30 15:45:08 +02:00
|
|
|
void runtime_entersyscall(int32)
|
|
|
|
__asm__ (GOSYM_PREFIX "runtime.entersyscall");
|
|
|
|
void runtime_entersyscallblock(int32)
|
|
|
|
__asm__ (GOSYM_PREFIX "runtime.entersyscallblock");
|
|
|
|
void runtime_exitsyscall(int32)
|
|
|
|
__asm__ (GOSYM_PREFIX "runtime.exitsyscall");
|
2013-07-16 08:54:42 +02:00
|
|
|
G* __go_go(void (*pfn)(void*), void*);
|
2010-12-03 05:34:57 +01:00
|
|
|
void siginit(void);
|
|
|
|
bool __go_sigsend(int32 sig);
|
2014-07-19 23:36:26 +02:00
|
|
|
int32 runtime_callers(int32, Location*, int32, bool keep_callers);
|
2016-09-29 02:56:44 +02:00
|
|
|
int64 runtime_nanotime(void) // monotonic time
|
|
|
|
__asm__(GOSYM_PREFIX "runtime.nanotime");
|
2014-07-19 10:53:52 +02:00
|
|
|
int64 runtime_unixnanotime(void); // real time, can skip
|
2013-07-16 08:54:42 +02:00
|
|
|
void runtime_dopanic(int32) __attribute__ ((noreturn));
|
|
|
|
void runtime_startpanic(void);
|
2013-11-06 20:49:01 +01:00
|
|
|
void runtime_freezetheworld(void);
|
|
|
|
void runtime_unwindstack(G*, byte*);
|
2013-07-16 08:54:42 +02:00
|
|
|
void runtime_sigprof();
|
|
|
|
void runtime_resetcpuprofiler(int32);
|
|
|
|
void runtime_setcpuprofilerate(void(*)(uintptr*, int32), int32);
|
2016-09-29 02:56:44 +02:00
|
|
|
void runtime_usleep(uint32)
|
|
|
|
__asm__ (GOSYM_PREFIX "runtime.usleep");
|
|
|
|
int64 runtime_cputicks(void)
|
|
|
|
__asm__ (GOSYM_PREFIX "runtime.cputicks");
|
|
|
|
int64 runtime_tickspersecond(void)
|
|
|
|
__asm__ (GOSYM_PREFIX "runtime.tickspersecond");
|
2012-10-23 06:31:11 +02:00
|
|
|
void runtime_blockevent(int64, int32);
|
|
|
|
extern int64 runtime_blockprofilerate;
|
2013-07-16 08:54:42 +02:00
|
|
|
void runtime_addtimer(Timer*);
|
|
|
|
bool runtime_deltimer(Timer*);
|
|
|
|
G* runtime_netpoll(bool);
|
|
|
|
void runtime_netpollinit(void);
|
2013-11-06 20:49:01 +01:00
|
|
|
int32 runtime_netpollopen(uintptr, PollDesc*);
|
|
|
|
int32 runtime_netpollclose(uintptr);
|
2013-07-16 08:54:42 +02:00
|
|
|
void runtime_netpollready(G**, PollDesc*, int32);
|
2013-11-06 20:49:01 +01:00
|
|
|
uintptr runtime_netpollfd(PollDesc*);
|
2014-07-19 10:53:52 +02:00
|
|
|
void runtime_netpollarm(PollDesc*, int32);
|
|
|
|
void** runtime_netpolluser(PollDesc*);
|
|
|
|
bool runtime_netpollclosing(PollDesc*);
|
|
|
|
void runtime_netpolllock(PollDesc*);
|
|
|
|
void runtime_netpollunlock(PollDesc*);
|
2013-07-16 08:54:42 +02:00
|
|
|
void runtime_crash(void);
|
2016-09-29 02:56:44 +02:00
|
|
|
void runtime_parsedebugvars(void)
|
|
|
|
__asm__(GOSYM_PREFIX "runtime.parsedebugvars");
|
2013-11-06 20:49:01 +01:00
|
|
|
void _rt0_go(void);
|
|
|
|
void* runtime_funcdata(Func*, int32);
|
2014-07-12 02:01:09 +02:00
|
|
|
int32 runtime_setmaxthreads(int32);
|
2014-07-19 10:53:52 +02:00
|
|
|
G* runtime_timejump(void);
|
|
|
|
void runtime_iterate_finq(void (*callback)(FuncVal*, void*, const FuncType*, const PtrType*));
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
void runtime_stoptheworld(void);
|
2012-10-23 06:31:11 +02:00
|
|
|
void runtime_starttheworld(void);
|
2012-03-02 21:01:37 +01:00
|
|
|
extern uint32 runtime_worldsema;
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
* mutual exclusion locks. in the uncontended case,
|
|
|
|
* as fast as spin locks (just a few user-level instructions),
|
|
|
|
* but on the contention path they sleep in the kernel.
|
2011-11-22 21:24:44 +01:00
|
|
|
* a zeroed Lock is unlocked (no need to initialize each lock).
|
2010-12-03 05:34:57 +01:00
|
|
|
*/
|
2016-09-29 02:56:44 +02:00
|
|
|
void runtime_lock(Lock*)
|
|
|
|
__asm__(GOSYM_PREFIX "runtime.lock");
|
|
|
|
void runtime_unlock(Lock*)
|
|
|
|
__asm__(GOSYM_PREFIX "runtime.unlock");
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
* sleep and wakeup on one-time events.
|
|
|
|
* before any calls to notesleep or notewakeup,
|
|
|
|
* must call noteclear to initialize the Note.
|
2011-11-22 21:24:44 +01:00
|
|
|
* then, exactly one thread can call notesleep
|
2010-12-03 05:34:57 +01:00
|
|
|
* and exactly one thread can call notewakeup (once).
|
2011-11-22 21:24:44 +01:00
|
|
|
* once notewakeup has been called, the notesleep
|
|
|
|
* will return. future notesleep will return immediately.
|
|
|
|
* subsequent noteclear must be called only after
|
|
|
|
* previous notesleep has returned, e.g. it's disallowed
|
|
|
|
* to call noteclear straight after notewakeup.
|
|
|
|
*
|
|
|
|
* notetsleep is like notesleep but wakes up after
|
|
|
|
* a given number of nanoseconds even if the event
|
|
|
|
* has not yet happened. if a goroutine uses notetsleep to
|
|
|
|
* wake up early, it must wait to call noteclear until it
|
|
|
|
* can be sure that no other goroutine is calling
|
|
|
|
* notewakeup.
|
2013-11-06 20:49:01 +01:00
|
|
|
*
|
|
|
|
* notesleep/notetsleep are generally called on g0,
|
|
|
|
* notetsleepg is similar to notetsleep but is called on user g.
|
2010-12-03 05:34:57 +01:00
|
|
|
*/
|
2016-09-30 15:45:08 +02:00
|
|
|
void runtime_noteclear(Note*)
|
|
|
|
__asm__ (GOSYM_PREFIX "runtime.noteclear");
|
|
|
|
void runtime_notesleep(Note*)
|
|
|
|
__asm__ (GOSYM_PREFIX "runtime.notesleep");
|
|
|
|
void runtime_notewakeup(Note*)
|
|
|
|
__asm__ (GOSYM_PREFIX "runtime.notewakeup");
|
|
|
|
bool runtime_notetsleep(Note*, int64) // false - timeout
|
|
|
|
__asm__ (GOSYM_PREFIX "runtime.notetsleep");
|
|
|
|
bool runtime_notetsleepg(Note*, int64) // false - timeout
|
|
|
|
__asm__ (GOSYM_PREFIX "runtime.notetsleepg");
|
2011-11-22 21:24:44 +01:00
|
|
|
|
2012-10-23 06:31:11 +02:00
|
|
|
/*
|
|
|
|
* Lock-free stack.
|
|
|
|
* Initialize uint64 head to 0, compare with 0 to test for emptiness.
|
|
|
|
* The stack does not keep pointers to nodes,
|
|
|
|
* so they can be garbage collected if there are no other pointers to nodes.
|
|
|
|
*/
|
|
|
|
void runtime_lfstackpush(uint64 *head, LFNode *node)
|
2013-01-24 20:44:23 +01:00
|
|
|
__asm__ (GOSYM_PREFIX "runtime.lfstackpush");
|
2012-10-23 06:31:11 +02:00
|
|
|
LFNode* runtime_lfstackpop(uint64 *head);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Parallel for over [0, n).
|
|
|
|
* body() is executed for each iteration.
|
|
|
|
* nthr - total number of worker threads.
|
|
|
|
* if wait=true, threads return from parfor() when all work is done;
|
|
|
|
* otherwise, threads can return while other threads are still finishing processing.
|
|
|
|
*/
|
|
|
|
ParFor* runtime_parforalloc(uint32 nthrmax);
|
2015-10-31 01:59:47 +01:00
|
|
|
void runtime_parforsetup(ParFor *desc, uint32 nthr, uint32 n, bool wait, const FuncVal *body);
|
2014-07-12 02:01:09 +02:00
|
|
|
void runtime_parfordo(ParFor *desc);
|
|
|
|
void runtime_parforiters(ParFor*, uintptr, uintptr*, uintptr*);
|
2012-10-23 06:31:11 +02:00
|
|
|
|
2012-03-02 17:38:43 +01:00
|
|
|
/*
|
|
|
|
* low level C-called
|
|
|
|
*/
|
|
|
|
#define runtime_mmap mmap
|
|
|
|
#define runtime_munmap munmap
|
|
|
|
#define runtime_madvise madvise
|
|
|
|
#define runtime_memclr(buf, size) __builtin_memset((buf), 0, (size))
|
2012-03-02 21:01:37 +01:00
|
|
|
#define runtime_getcallerpc(p) __builtin_return_address(0)
|
2012-03-02 17:38:43 +01:00
|
|
|
|
|
|
|
#ifdef __rtems__
|
|
|
|
void __wrap_rtems_task_variable_add(void **);
|
|
|
|
#endif
|
|
|
|
|
2011-11-30 01:21:52 +01:00
|
|
|
/*
|
|
|
|
* runtime go-called
|
|
|
|
*/
|
2013-06-19 01:49:49 +02:00
|
|
|
void reflect_call(const struct __go_func_type *, FuncVal *, _Bool, _Bool,
|
2012-03-02 17:38:43 +01:00
|
|
|
void **, void **)
|
2013-01-24 20:44:23 +01:00
|
|
|
__asm__ (GOSYM_PREFIX "reflect.call");
|
2011-11-30 01:21:52 +01:00
|
|
|
#define runtime_panic __go_panic
|
2011-12-02 20:34:41 +01:00
|
|
|
|
2012-03-02 17:38:43 +01:00
|
|
|
/*
|
|
|
|
* runtime c-called (but written in Go)
|
|
|
|
*/
|
|
|
|
void runtime_printany(Eface)
|
2013-01-24 20:44:23 +01:00
|
|
|
__asm__ (GOSYM_PREFIX "runtime.Printany");
|
2012-03-02 17:38:43 +01:00
|
|
|
void runtime_newTypeAssertionError(const String*, const String*, const String*, const String*, Eface*)
|
2013-01-24 20:44:23 +01:00
|
|
|
__asm__ (GOSYM_PREFIX "runtime.NewTypeAssertionError");
|
2013-11-06 20:49:01 +01:00
|
|
|
void runtime_newErrorCString(const char*, Eface*)
|
|
|
|
__asm__ (GOSYM_PREFIX "runtime.NewErrorCString");
|
2012-03-02 17:38:43 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
* wrapped for go users
|
|
|
|
*/
|
2013-11-06 20:49:01 +01:00
|
|
|
void runtime_semacquire(uint32 volatile *, bool);
|
2011-11-28 06:45:49 +01:00
|
|
|
void runtime_semrelease(uint32 volatile *);
|
|
|
|
int32 runtime_gomaxprocsfunc(int32 n);
|
2016-09-30 15:45:08 +02:00
|
|
|
void runtime_procyield(uint32)
|
|
|
|
__asm__(GOSYM_PREFIX "runtime.procyield");
|
|
|
|
void runtime_osyield(void)
|
|
|
|
__asm__(GOSYM_PREFIX "runtime.osyield");
|
2013-07-16 08:54:42 +02:00
|
|
|
void runtime_lockOSThread(void);
|
|
|
|
void runtime_unlockOSThread(void);
|
2014-07-12 02:01:09 +02:00
|
|
|
bool runtime_lockedOSThread(void);
|
2011-03-27 21:14:55 +02:00
|
|
|
|
2013-01-30 02:37:13 +01:00
|
|
|
bool runtime_showframe(String, bool);
|
2013-11-06 20:49:01 +01:00
|
|
|
void runtime_printcreatedby(G*);
|
2012-05-17 07:30:25 +02:00
|
|
|
|
2012-03-06 18:57:23 +01:00
|
|
|
uintptr runtime_memlimit(void);
|
|
|
|
|
2013-07-16 08:54:42 +02:00
|
|
|
#define ISNAN(f) __builtin_isnan(f)
|
|
|
|
|
2012-10-23 06:31:11 +02:00
|
|
|
enum
|
|
|
|
{
|
2014-09-04 00:56:09 +02:00
|
|
|
UseSpanType = 1,
|
2012-10-23 06:31:11 +02:00
|
|
|
};
|
|
|
|
|
2012-03-02 17:38:43 +01:00
|
|
|
#define runtime_setitimer setitimer
|
2012-03-06 18:57:23 +01:00
|
|
|
|
2016-09-29 02:56:44 +02:00
|
|
|
void runtime_check(void)
|
|
|
|
__asm__ (GOSYM_PREFIX "runtime.check");
|
2012-03-06 18:57:23 +01:00
|
|
|
|
|
|
|
// A list of global variables that the garbage collector must scan.
|
|
|
|
struct root_list {
|
|
|
|
struct root_list *next;
|
|
|
|
struct root {
|
|
|
|
void *decl;
|
|
|
|
size_t size;
|
|
|
|
} roots[];
|
|
|
|
};
|
|
|
|
|
|
|
|
void __go_register_gc_roots(struct root_list*);
|
2012-04-20 06:58:26 +02:00
|
|
|
|
|
|
|
// Size of stack space allocated using Go's allocator.
|
|
|
|
// This will be 0 when using split stacks, as in that case
|
|
|
|
// the stacks are allocated by the splitstack library.
|
|
|
|
extern uintptr runtime_stacks_sys;
|
2012-05-17 07:30:25 +02:00
|
|
|
|
2012-09-28 16:48:30 +02:00
|
|
|
struct backtrace_state;
|
|
|
|
extern struct backtrace_state *__go_get_backtrace_state(void);
|
2016-07-22 20:15:38 +02:00
|
|
|
extern _Bool __go_file_line(uintptr, int, String*, String*, intgo *);
|
2013-07-16 08:54:42 +02:00
|
|
|
extern void runtime_main(void*);
|
2013-12-01 02:40:16 +01:00
|
|
|
extern uint32 runtime_in_callers;
|
2012-08-07 06:42:49 +02:00
|
|
|
|
|
|
|
int32 getproccount(void);
|
2013-07-16 08:54:42 +02:00
|
|
|
|
|
|
|
#define PREFETCH(p) __builtin_prefetch(p)
|
2013-09-03 23:52:37 +02:00
|
|
|
|
2013-11-06 20:49:01 +01:00
|
|
|
bool runtime_gcwaiting(void);
|
|
|
|
void runtime_badsignal(int);
|
2014-06-07 00:37:27 +02:00
|
|
|
Defer* runtime_newdefer(void);
|
|
|
|
void runtime_freedefer(Defer*);
|
2014-07-19 10:53:52 +02:00
|
|
|
|
|
|
|
struct time_now_ret
|
|
|
|
{
|
|
|
|
int64_t sec;
|
|
|
|
int32_t nsec;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct time_now_ret now() __asm__ (GOSYM_PREFIX "time.now")
|
|
|
|
__attribute__ ((no_split_stack));
|
2015-04-29 23:31:53 +02:00
|
|
|
|
|
|
|
extern void _cgo_wait_runtime_init_done (void);
|
|
|
|
extern void _cgo_notify_runtime_init_done (void);
|
|
|
|
extern _Bool runtime_iscgo;
|
|
|
|
extern _Bool runtime_cgoHasExtraM;
|
|
|
|
extern Hchan *runtime_main_init_done;
|
2016-02-09 01:34:55 +01:00
|
|
|
extern uintptr __go_end __attribute__ ((weak));
|