Blame sysdeps/mach/htl/pt-stack-alloc.c

Packit 6c4009
/* Allocate a new stack.  Mach version.
Packit 6c4009
   Copyright (C) 2000-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
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 <errno.h>
Packit 6c4009
Packit 6c4009
#include <mach.h>
Packit 6c4009
#include <mach/machine/vm_param.h>
Packit 6c4009
Packit 6c4009
#include <pt-internal.h>
Packit 6c4009
Packit 6c4009
/* The next address to use for stack allocation.  */
Packit 6c4009
static vm_address_t next_stack_base = VM_MIN_ADDRESS;
Packit 6c4009
Packit 6c4009
Packit 6c4009
/* Allocate a new stack of size STACKSIZE.  If successful, store the
Packit 6c4009
   address of the newly allocated stack in *STACKADDR and return 0.
Packit 6c4009
   Otherwise return an error code (EINVAL for an invalid stack size,
Packit 6c4009
   EAGAIN if the system lacked the necessary resources to allocate a
Packit 6c4009
   new stack).  */
Packit 6c4009
int
Packit 6c4009
__pthread_stack_alloc (void **stackaddr, size_t stacksize)
Packit 6c4009
{
Packit 6c4009
  vm_offset_t base;
Packit 6c4009
  int i = 0;
Packit 6c4009
Packit 6c4009
get_stack:
Packit 6c4009
  i++;
Packit 6c4009
  for (base = next_stack_base;
Packit 6c4009
       base < VM_MAX_ADDRESS
Packit 6c4009
       && __vm_allocate (__mach_task_self (), &base,
Packit 6c4009
			 stacksize, FALSE) != KERN_SUCCESS; base += stacksize)
Packit 6c4009
    ;
Packit 6c4009
Packit 6c4009
  if (base >= VM_MAX_ADDRESS)
Packit 6c4009
    {
Packit 6c4009
      if (i == 1)
Packit 6c4009
	{
Packit 6c4009
	  next_stack_base = VM_MIN_ADDRESS;
Packit 6c4009
	  goto get_stack;
Packit 6c4009
	}
Packit 6c4009
      else
Packit 6c4009
	return EAGAIN;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  if (base >= VM_MAX_ADDRESS)
Packit 6c4009
    return EAGAIN;
Packit 6c4009
Packit 6c4009
  next_stack_base = base + stacksize;
Packit 6c4009
Packit 6c4009
  (*stackaddr) = (void *) base;
Packit 6c4009
  return 0;
Packit 6c4009
}