| // 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 <errno.h> |
| #include <signal.h> |
| #include <unistd.h> |
| |
| #if defined(__i386__) || defined(__x86_64__) |
| #include <cpuid.h> |
| #endif |
| |
| #ifdef __linux__ |
| #include <syscall.h> |
| #endif |
| |
| #include "config.h" |
| |
| #include "runtime.h" |
| #include "arch.h" |
| #include "array.h" |
| |
| int32 |
| runtime_atoi(const byte *p, intgo len) |
| { |
| int32 n; |
| |
| n = 0; |
| while(len > 0 && '0' <= *p && *p <= '9') { |
| n = n*10 + *p++ - '0'; |
| len--; |
| } |
| return n; |
| } |
| |
| #if defined(__i386__) || defined(__x86_64__) || defined (__s390__) || defined (__s390x__) |
| |
| // When cputicks is just asm instructions, skip the split stack |
| // prologue for speed. |
| |
| int64 runtime_cputicks(void) __attribute__((no_split_stack)); |
| |
| #endif |
| |
| // Whether the processor supports SSE2. |
| #if defined (__i386__) |
| static _Bool hasSSE2; |
| |
| // Force appropriate CPU level so that we can call the lfence/mfence |
| // builtins. |
| |
| #pragma GCC push_options |
| #pragma GCC target("sse2") |
| |
| #elif defined(__x86_64__) |
| #define hasSSE2 true |
| #endif |
| |
| #if defined(__i386__) || defined(__x86_64__) |
| // Whether to use lfence, as opposed to mfence. |
| // Set based on cpuid. |
| static _Bool lfenceBeforeRdtsc; |
| #endif // defined(__i386__) || defined(__x86_64__) |
| |
| int64 |
| runtime_cputicks(void) |
| { |
| #if defined(__i386__) || defined(__x86_64__) |
| if (hasSSE2) { |
| if (lfenceBeforeRdtsc) { |
| __builtin_ia32_lfence(); |
| } else { |
| __builtin_ia32_mfence(); |
| } |
| } |
| return __builtin_ia32_rdtsc(); |
| #elif defined (__s390__) || defined (__s390x__) |
| uint64 clock = 0; |
| /* stckf may not write the return variable in case of a clock error, so make |
| it read-write to prevent that the initialisation is optimised out. |
| Note: Targets below z9-109 will crash when executing store clock fast, i.e. |
| we don't support Go for machines older than that. */ |
| asm volatile(".insn s,0xb27c0000,%0" /* stckf */ : "+Q" (clock) : : "cc" ); |
| return (int64)clock; |
| #else |
| // Currently cputicks() is used in blocking profiler and to seed runtime·fastrand(). |
| // runtime·nanotime() is a poor approximation of CPU ticks that is enough for the profiler. |
| // TODO: need more entropy to better seed fastrand. |
| return runtime_nanotime(); |
| #endif |
| } |
| |
| #if defined(__i386__) |
| #pragma GCC pop_options |
| #endif |
| |
| void |
| runtime_signalstack(byte *p, uintptr n) |
| { |
| stack_t st; |
| |
| st.ss_sp = p; |
| st.ss_size = n; |
| st.ss_flags = 0; |
| if(p == nil) |
| st.ss_flags = SS_DISABLE; |
| if(sigaltstack(&st, nil) < 0) |
| *(int *)0xf1 = 0xf1; |
| } |
| |
| int32 go_open(char *, int32, int32) |
| __asm__ (GOSYM_PREFIX "runtime.open"); |
| |
| int32 |
| go_open(char *name, int32 mode, int32 perm) |
| { |
| return runtime_open(name, mode, perm); |
| } |
| |
| int32 go_read(int32, void *, int32) |
| __asm__ (GOSYM_PREFIX "runtime.read"); |
| |
| int32 |
| go_read(int32 fd, void *p, int32 n) |
| { |
| return runtime_read(fd, p, n); |
| } |
| |
| int32 go_write(uintptr, void *, int32) |
| __asm__ (GOSYM_PREFIX "runtime.write"); |
| |
| int32 |
| go_write(uintptr fd, void *p, int32 n) |
| { |
| return runtime_write(fd, p, n); |
| } |
| |
| int32 go_closefd(int32) |
| __asm__ (GOSYM_PREFIX "runtime.closefd"); |
| |
| int32 |
| go_closefd(int32 fd) |
| { |
| return runtime_close(fd); |
| } |
| |
| intgo go_errno(void) |
| __asm__ (GOSYM_PREFIX "runtime.errno"); |
| |
| intgo |
| go_errno() |
| { |
| return (intgo)errno; |
| } |
| |
| uintptr getEnd(void) |
| __asm__ (GOSYM_PREFIX "runtime.getEnd"); |
| |
| uintptr |
| getEnd() |
| { |
| #ifdef _AIX |
| // mmap adresses range start at 0x30000000 on AIX for 32 bits processes |
| uintptr end = 0x30000000U; |
| #else |
| uintptr end = 0; |
| uintptr *pend; |
| |
| pend = &__go_end; |
| if (pend != nil) { |
| end = *pend; |
| } |
| #endif |
| |
| return end; |
| } |
| |
| // CPU-specific initialization. |
| // Fetch CPUID info on x86. |
| |
| void |
| runtime_cpuinit() |
| { |
| #if defined(__i386__) || defined(__x86_64__) |
| unsigned int eax, ebx, ecx, edx; |
| |
| if (__get_cpuid(0, &eax, &ebx, &ecx, &edx)) { |
| if (eax != 0 |
| && ebx == 0x756E6547 // "Genu" |
| && edx == 0x49656E69 // "ineI" |
| && ecx == 0x6C65746E) { // "ntel" |
| lfenceBeforeRdtsc = true; |
| } |
| } |
| if (__get_cpuid(1, &eax, &ebx, &ecx, &edx)) { |
| setCpuidECX(ecx); |
| #if defined(__i386__) |
| if ((edx & bit_SSE2) != 0) { |
| hasSSE2 = true; |
| } |
| #endif |
| } |
| |
| #if defined(HAVE_AS_X86_AES) |
| setSupportAES(true); |
| #endif |
| #endif |
| } |
| |
| // A publication barrier: a store/store barrier. |
| |
| void publicationBarrier(void) |
| __asm__ (GOSYM_PREFIX "runtime.publicationBarrier"); |
| |
| void |
| publicationBarrier() |
| { |
| __atomic_thread_fence(__ATOMIC_RELEASE); |
| } |
| |
| #ifdef __linux__ |
| |
| /* Currently sbrk0 is only called on GNU/Linux. */ |
| |
| uintptr sbrk0(void) |
| __asm__ (GOSYM_PREFIX "runtime.sbrk0"); |
| |
| uintptr |
| sbrk0() |
| { |
| return syscall(SYS_brk, (uintptr)(0)); |
| } |
| |
| #endif /* __linux__ */ |