Blame nptl/pt-cleanup.c

Packit 6c4009
/* Copyright (C) 2002-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
   Contributed by Ulrich Drepper <drepper@redhat.com>, 2002.
Packit 6c4009
Packit 6c4009
   The GNU C Library is free software; you can redistribute it and/or
Packit 6c4009
   modify it under the terms of the GNU Lesser General Public
Packit 6c4009
   License as published by the Free Software Foundation; either
Packit 6c4009
   version 2.1 of the License, or (at your option) any later version.
Packit 6c4009
Packit 6c4009
   The GNU C Library is distributed in the hope that it will be useful,
Packit 6c4009
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 6c4009
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 6c4009
   Lesser General Public License for more details.
Packit 6c4009
Packit 6c4009
   You should have received a copy of the GNU Lesser General Public
Packit 6c4009
   License along with the GNU C Library; if not, see
Packit 6c4009
   <http://www.gnu.org/licenses/>.  */
Packit 6c4009
Packit 6c4009
#include <setjmp.h>
Packit 6c4009
#include <stdlib.h>
Packit 6c4009
#include "pthreadP.h"
Packit 6c4009
#include <jmpbuf-unwind.h>
Packit 6c4009
Packit 6c4009
void
Packit 6c4009
__pthread_cleanup_upto (__jmp_buf target, char *targetframe)
Packit 6c4009
{
Packit 6c4009
  struct pthread *self = THREAD_SELF;
Packit 6c4009
  struct _pthread_cleanup_buffer *cbuf;
Packit 6c4009
Packit 6c4009
  /* Adjust all pointers used in comparisons, so that top of thread's
Packit 6c4009
     stack is at the top of address space.  Without that, things break
Packit 6c4009
     if stack is allocated above the main stack.  */
Packit 6c4009
  uintptr_t adj = (uintptr_t) self->stackblock + self->stackblock_size;
Packit 6c4009
  uintptr_t targetframe_adj = (uintptr_t) targetframe - adj;
Packit 6c4009
Packit 6c4009
  for (cbuf = THREAD_GETMEM (self, cleanup);
Packit 6c4009
       cbuf != NULL && _JMPBUF_UNWINDS_ADJ (target, cbuf, adj);
Packit 6c4009
       cbuf = cbuf->__prev)
Packit 6c4009
    {
Packit 6c4009
#if _STACK_GROWS_DOWN
Packit 6c4009
      if ((uintptr_t) cbuf - adj <= targetframe_adj)
Packit 6c4009
        {
Packit 6c4009
          cbuf = NULL;
Packit 6c4009
          break;
Packit 6c4009
        }
Packit 6c4009
#elif _STACK_GROWS_UP
Packit 6c4009
      if ((uintptr_t) cbuf - adj >= targetframe_adj)
Packit 6c4009
        {
Packit 6c4009
          cbuf = NULL;
Packit 6c4009
          break;
Packit 6c4009
        }
Packit 6c4009
#else
Packit 6c4009
# error "Define either _STACK_GROWS_DOWN or _STACK_GROWS_UP"
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
      /* Call the cleanup code.  */
Packit 6c4009
      cbuf->__routine (cbuf->__arg);
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  THREAD_SETMEM (self, cleanup, cbuf);
Packit 6c4009
}
Packit 6c4009
hidden_def (__pthread_cleanup_upto)