LCOV - code coverage report
Current view: top level - js/src - jsnativestack.cpp (source / functions) Hit Total Coverage
Test: output.info Lines: 11 12 91.7 %
Date: 2017-07-14 16:53:18 Functions: 1 1 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
       2             :  * vim: set ts=8 sts=4 et sw=4 tw=99:
       3             :  * This Source Code Form is subject to the terms of the Mozilla Public
       4             :  * License, v. 2.0. If a copy of the MPL was not distributed with this
       5             :  * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
       6             : 
       7             : #include "jsnativestack.h"
       8             : 
       9             : #ifdef XP_WIN
      10             : # include "jswin.h"
      11             : 
      12             : #elif defined(XP_DARWIN) || defined(DARWIN) || defined(XP_UNIX)
      13             : # include <pthread.h>
      14             : 
      15             : # if defined(__FreeBSD__) || defined(__OpenBSD__) || defined(__DragonFly__)
      16             : #  include <pthread_np.h>
      17             : # endif
      18             : 
      19             : # if defined(ANDROID) && !defined(__aarch64__)
      20             : #  include <sys/types.h>
      21             : #  include <unistd.h>
      22             : # endif
      23             : 
      24             : #else
      25             : # error "Unsupported platform"
      26             : 
      27             : #endif
      28             : 
      29             : #if defined(XP_WIN)
      30             : 
      31             : void*
      32             : js::GetNativeStackBaseImpl()
      33             : {
      34             : # if defined(_M_IX86) && defined(_MSC_VER)
      35             :     /*
      36             :      * offset 0x18 from the FS segment register gives a pointer to
      37             :      * the thread information block for the current thread
      38             :      */
      39             :     NT_TIB* pTib;
      40             :     __asm {
      41             :         MOV EAX, FS:[18h]
      42             :         MOV pTib, EAX
      43             :     }
      44             :     return static_cast<void*>(pTib->StackBase);
      45             : 
      46             : # elif defined(_M_X64)
      47             :     PNT_TIB64 pTib = reinterpret_cast<PNT_TIB64>(NtCurrentTeb());
      48             :     return reinterpret_cast<void*>(pTib->StackBase);
      49             : 
      50             : # elif defined(_M_ARM)
      51             :     PNT_TIB pTib = reinterpret_cast<PNT_TIB>(NtCurrentTeb());
      52             :     return static_cast<void*>(pTib->StackBase);
      53             : 
      54             : # elif defined(_WIN32) && defined(__GNUC__)
      55             :     NT_TIB* pTib;
      56             :     asm ("movl %%fs:0x18, %0\n" : "=r" (pTib));
      57             :     return static_cast<void*>(pTib->StackBase);
      58             : 
      59             : # endif
      60             : }
      61             : 
      62             : #elif defined(SOLARIS)
      63             : 
      64             : #include <ucontext.h>
      65             : 
      66             : JS_STATIC_ASSERT(JS_STACK_GROWTH_DIRECTION < 0);
      67             : 
      68             : void*
      69             : js::GetNativeStackBaseImpl()
      70             : {
      71             :     stack_t st;
      72             :     stack_getbounds(&st);
      73             :     return static_cast<char*>(st.ss_sp) + st.ss_size;
      74             : }
      75             : 
      76             : #elif defined(AIX)
      77             : 
      78             : #include <ucontext.h>
      79             : 
      80             : JS_STATIC_ASSERT(JS_STACK_GROWTH_DIRECTION < 0);
      81             : 
      82             : void*
      83             : js::GetNativeStackBaseImpl()
      84             : {
      85             :     ucontext_t context;
      86             :     getcontext(&context);
      87             :     return static_cast<char*>(context.uc_stack.ss_sp) +
      88             :         context.uc_stack.ss_size;
      89             : }
      90             : 
      91             : #else /* XP_UNIX */
      92             : 
      93             : void*
      94          40 : js::GetNativeStackBaseImpl()
      95             : {
      96          40 :     pthread_t thread = pthread_self();
      97             : # if defined(XP_DARWIN) || defined(DARWIN)
      98             :     return pthread_get_stackaddr_np(thread);
      99             : 
     100             : # else
     101             :     pthread_attr_t sattr;
     102          40 :     pthread_attr_init(&sattr);
     103             : #  if defined(__OpenBSD__)
     104             :     stack_t ss;
     105             : #  elif defined(PTHREAD_NP_H) || defined(_PTHREAD_NP_H_) || defined(NETBSD)
     106             :     /* e.g. on FreeBSD 4.8 or newer, neundorf@kde.org */
     107             :     pthread_attr_get_np(thread, &sattr);
     108             : #  else
     109             :     /*
     110             :      * FIXME: this function is non-portable;
     111             :      * other POSIX systems may have different np alternatives
     112             :      */
     113          40 :     pthread_getattr_np(thread, &sattr);
     114             : #  endif
     115             : 
     116          40 :     void* stackBase = 0;
     117          40 :     size_t stackSize = 0;
     118             :     int rc;
     119             : # if defined(__OpenBSD__)
     120             :     rc = pthread_stackseg_np(pthread_self(), &ss);
     121             :     stackBase = (void*)((size_t) ss.ss_sp - ss.ss_size);
     122             :     stackSize = ss.ss_size;
     123             : # elif defined(ANDROID) && !defined(__aarch64__)
     124             :     if (gettid() == getpid()) {
     125             :         // bionic's pthread_attr_getstack prior to API 21 doesn't tell the truth
     126             :         // for the main thread (see bug 846670). So we scan /proc/self/maps to
     127             :         // find the segment which contains the stack.
     128             :         rc = -1;
     129             : 
     130             :         // Put the string on the stack, otherwise there is the danger that it
     131             :         // has not been decompressed by the the on-demand linker. Bug 1165460.
     132             :         //
     133             :         // The volatile keyword should stop the compiler from trying to omit
     134             :         // the stack copy in the future (hopefully).
     135             :         volatile char path[] = "/proc/self/maps";
     136             :         FILE* fs = fopen((const char*)path, "r");
     137             : 
     138             :         if (fs) {
     139             :             char line[100];
     140             :             unsigned long stackAddr = (unsigned long)&sattr;
     141             :             while (fgets(line, sizeof(line), fs) != nullptr) {
     142             :                 unsigned long stackStart;
     143             :                 unsigned long stackEnd;
     144             :                 if (sscanf(line, "%lx-%lx ", &stackStart, &stackEnd) == 2 &&
     145             :                     stackAddr >= stackStart && stackAddr < stackEnd) {
     146             :                     stackBase = (void*)stackStart;
     147             :                     stackSize = stackEnd - stackStart;
     148             :                     rc = 0;
     149             :                     break;
     150             :                 }
     151             :             }
     152             :             fclose(fs);
     153             :         }
     154             :     } else
     155             :         // For non main-threads pthread allocates the stack itself so it tells
     156             :         // the truth.
     157             :         rc = pthread_attr_getstack(&sattr, &stackBase, &stackSize);
     158             : # else
     159          40 :     rc = pthread_attr_getstack(&sattr, &stackBase, &stackSize);
     160             : # endif
     161          40 :     if (rc)
     162           0 :         MOZ_CRASH();
     163          40 :     MOZ_ASSERT(stackBase);
     164          40 :     pthread_attr_destroy(&sattr);
     165             : 
     166             : #  if JS_STACK_GROWTH_DIRECTION > 0
     167             :     return stackBase;
     168             : #  else
     169          40 :     return static_cast<char*>(stackBase) + stackSize;
     170             : #  endif
     171             : # endif
     172             : }
     173             : 
     174             : #endif /* !XP_WIN */

Generated by: LCOV version 1.13