| /* Copyright (C) 1993-2016 Free Software Foundation, Inc. | 
 |    This file is part of the GNU C Library. | 
 |  | 
 |    The GNU C Library is free software; you can redistribute it and/or | 
 |    modify it under the terms of the GNU Lesser General Public | 
 |    License as published by the Free Software Foundation; either | 
 |    version 2.1 of the License, or (at your option) any later version. | 
 |  | 
 |    The GNU C Library is distributed in the hope that it will be useful, | 
 |    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
 |    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
 |    Lesser General Public License for more details. | 
 |  | 
 |    You should have received a copy of the GNU Lesser General Public | 
 |    License along with the GNU C Library; if not, see | 
 |    <http://www.gnu.org/licenses/>.  */ | 
 |  | 
 | #include <hurd.h> | 
 | #include <hurd/id.h> | 
 |  | 
 | struct hurd_id_data _hurd_id; | 
 |  | 
 |  | 
 | /* Check that _hurd_id.{gen,aux} are valid and update them if not. | 
 |    Expects _hurd_id.lock to be held and does not release it.  */ | 
 |  | 
 | error_t | 
 | _hurd_check_ids (void) | 
 | { | 
 |   if (! _hurd_id.valid) | 
 |     { | 
 |       inline void dealloc (__typeof (_hurd_id.gen) *p) | 
 | 	{ | 
 | 	  if (p->uids) | 
 | 	    { | 
 | 	      __vm_deallocate (__mach_task_self (), | 
 | 			       (vm_address_t) p->uids, | 
 | 			       p->nuids * sizeof (uid_t)); | 
 | 	      p->uids = NULL; | 
 | 	    } | 
 | 	  p->nuids = 0; | 
 | 	  if (p->gids) | 
 | 	    { | 
 | 	      __vm_deallocate (__mach_task_self (), | 
 | 			       (vm_address_t) p->gids, | 
 | 			       p->ngids * sizeof (gid_t)); | 
 | 	      p->gids = NULL; | 
 | 	    } | 
 | 	  p->ngids = 0; | 
 | 	} | 
 |  | 
 |       error_t err; | 
 |  | 
 |       dealloc (&_hurd_id.gen); | 
 |       dealloc (&_hurd_id.aux); | 
 |  | 
 |       if (_hurd_id.rid_auth != MACH_PORT_NULL) | 
 | 	{ | 
 | 	  __mach_port_deallocate (__mach_task_self (), _hurd_id.rid_auth); | 
 | 	  _hurd_id.rid_auth = MACH_PORT_NULL; | 
 | 	} | 
 |  | 
 |       if (err = __USEPORT (AUTH, __auth_getids | 
 | 			   (port, | 
 | 			    &_hurd_id.gen.uids, &_hurd_id.gen.nuids, | 
 | 			    &_hurd_id.aux.uids, &_hurd_id.aux.nuids, | 
 | 			    &_hurd_id.gen.gids, &_hurd_id.gen.ngids, | 
 | 			    &_hurd_id.aux.gids, &_hurd_id.aux.ngids))) | 
 | 	return err; | 
 |  | 
 |       _hurd_id.valid = 1; | 
 |     } | 
 |  | 
 |   return 0; | 
 | } | 
 |  | 
 | static void | 
 | init_id (void) | 
 | { | 
 |   __mutex_init (&_hurd_id.lock); | 
 |   _hurd_id.valid = 0; | 
 |   _hurd_id.rid_auth = MACH_PORT_NULL; | 
 |   _hurd_id.gen.uids = _hurd_id.aux.uids = NULL; | 
 |   _hurd_id.gen.nuids = _hurd_id.aux.nuids = 0; | 
 |   _hurd_id.gen.gids = _hurd_id.aux.gids = NULL; | 
 |   _hurd_id.gen.ngids = _hurd_id.aux.ngids = 0; | 
 |  | 
 |   (void) &init_id;		/* Avoid "defined but not used" warning.  */ | 
 | } | 
 | text_set_element (_hurd_preinit_hook, init_id); |