(__getlogin_r_loginuid): Also fail if tpwd after pwuid call is NULL.
[glibc.git] / gshadow / sgetsgent.c
blob94e131566bfc277cfb89bc1fff1d8d4af67f5ad0
1 /* Copyright (C) 2009 Free Software Foundation, Inc.
2 This file is part of the GNU C Library.
4 The GNU C Library is free software; you can redistribute it and/or
5 modify it under the terms of the GNU Lesser General Public
6 License as published by the Free Software Foundation; either
7 version 2.1 of the License, or (at your option) any later version.
9 The GNU C Library is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 Lesser General Public License for more details.
14 You should have received a copy of the GNU Lesser General Public
15 License along with the GNU C Library; if not, write to the Free
16 Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
17 02111-1307 USA. */
19 #include <errno.h>
20 #include <bits/libc-lock.h>
21 #include <gshadow.h>
22 #include <stdlib.h>
25 /* A reasonable size for a buffer to start with. */
26 #define BUFLEN_SPWD 1024
28 /* We need to protect the dynamic buffer handling. */
29 __libc_lock_define_initialized (static, lock);
31 /* Read one shadow entry from the given stream. */
32 struct sgrp *
33 sgetsgent (const char *string)
35 static char *buffer;
36 static size_t buffer_size;
37 static struct sgrp resbuf;
38 struct sgrp *result;
39 int save;
41 /* Get lock. */
42 __libc_lock_lock (lock);
44 /* Allocate buffer if not yet available. */
45 if (buffer == NULL)
47 buffer_size = BUFLEN_SPWD;
48 buffer = malloc (buffer_size);
51 while (buffer != NULL
52 && __sgetsgent_r (string, &resbuf, buffer, buffer_size, &result) != 0
53 && errno == ERANGE)
55 char *new_buf;
56 buffer_size += BUFLEN_SPWD;
57 new_buf = realloc (buffer, buffer_size);
58 if (new_buf == NULL)
60 /* We are out of memory. Free the current buffer so that the
61 process gets a chance for a normal termination. */
62 save = errno;
63 free (buffer);
64 __set_errno (save);
66 buffer = new_buf;
69 if (buffer == NULL)
70 result = NULL;
72 /* Release lock. Preserve error value. */
73 save = errno;
74 __libc_lock_unlock (lock);
75 __set_errno (save);
77 return result;