Annotation of embedaddon/libiconv/srclib/areadlink.c, revision 1.1.1.1

1.1       misho       1: /* areadlink.c -- readlink wrapper to return the link name in malloc'd storage
                      2:    Unlike xreadlink and xreadlink_with_size, don't ever call exit.
                      3: 
                      4:    Copyright (C) 2001, 2003-2007 Free Software Foundation, Inc.
                      5: 
                      6:    This program is free software: you can redistribute it and/or modify
                      7:    it under the terms of the GNU General Public License as published by
                      8:    the Free Software Foundation; either version 3 of the License, or
                      9:    (at your option) any later version.
                     10: 
                     11:    This program is distributed in the hope that it will be useful,
                     12:    but WITHOUT ANY WARRANTY; without even the implied warranty of
                     13:    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
                     14:    GNU General Public License for more details.
                     15: 
                     16:    You should have received a copy of the GNU General Public License
                     17:    along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
                     18: 
                     19: /* Written by Jim Meyering <jim@meyering.net>
                     20:    and Bruno Haible <bruno@clisp.org>.  */
                     21: 
                     22: #include <config.h>
                     23: 
                     24: /* Specification.  */
                     25: #include "areadlink.h"
                     26: 
                     27: #include <string.h>
                     28: #include <errno.h>
                     29: #include <limits.h>
                     30: #include <sys/types.h>
                     31: #include <stdlib.h>
                     32: #include <unistd.h>
                     33: 
                     34: #ifndef SIZE_MAX
                     35: # define SIZE_MAX ((size_t) -1)
                     36: #endif
                     37: #ifndef SSIZE_MAX
                     38: # define SSIZE_MAX ((ssize_t) (SIZE_MAX / 2))
                     39: #endif
                     40: 
                     41: /* Call readlink to get the symbolic link value of FILENAME.
                     42:    Return a pointer to that NUL-terminated string in malloc'd storage.
                     43:    If readlink fails, return NULL and set errno.
                     44:    If realloc fails, or if the link value is longer than SIZE_MAX :-),
                     45:    return NULL and set errno to ENOMEM.  */
                     46: 
                     47: char *
                     48: areadlink (char const *filename)
                     49: {
                     50:   /* The initial buffer size for the link value.  A power of 2
                     51:      detects arithmetic overflow earlier, but is not required.  */
                     52: #define INITIAL_BUF_SIZE 1024
                     53: 
                     54:   /* Allocate the initial buffer on the stack.  This way, in the common
                     55:      case of a symlink of small size, we get away with a single small malloc()
                     56:      instead of a big malloc() followed by a shrinking realloc().  */
                     57:   char initial_buf[INITIAL_BUF_SIZE];
                     58: 
                     59:   char *buffer = initial_buf;
                     60:   size_t buf_size = sizeof (initial_buf);
                     61: 
                     62:   while (1)
                     63:     {
                     64:       /* Attempt to read the link into the current buffer.  */
                     65:       ssize_t link_length = readlink (filename, buffer, buf_size);
                     66: 
                     67:       /* On AIX 5L v5.3 and HP-UX 11i v2 04/09, readlink returns -1
                     68:         with errno == ERANGE if the buffer is too small.  */
                     69:       if (link_length < 0 && errno != ERANGE)
                     70:        {
                     71:          if (buffer != initial_buf)
                     72:            {
                     73:              int saved_errno = errno;
                     74:              free (buffer);
                     75:              errno = saved_errno;
                     76:            }
                     77:          return NULL;
                     78:        }
                     79: 
                     80:       if ((size_t) link_length < buf_size)
                     81:        {
                     82:          buffer[link_length++] = '\0';
                     83: 
                     84:          /* Return it in a chunk of memory as small as possible.  */
                     85:          if (buffer == initial_buf)
                     86:            {
                     87:              buffer = (char *) malloc (link_length);
                     88:              if (buffer == NULL)
                     89:                /* errno is ENOMEM.  */
                     90:                return NULL;
                     91:              memcpy (buffer, initial_buf, link_length);
                     92:            }
                     93:          else
                     94:            {
                     95:              /* Shrink buffer before returning it.  */
                     96:              if ((size_t) link_length < buf_size)
                     97:                {
                     98:                  char *smaller_buffer = (char *) realloc (buffer, link_length);
                     99: 
                    100:                  if (smaller_buffer != NULL)
                    101:                    buffer = smaller_buffer;
                    102:                }
                    103:            }
                    104:          return buffer;
                    105:        }
                    106: 
                    107:       if (buffer != initial_buf)
                    108:        free (buffer);
                    109:       buf_size *= 2;
                    110:       if (SSIZE_MAX < buf_size || (SIZE_MAX / 2 < SSIZE_MAX && buf_size == 0))
                    111:        {
                    112:          errno = ENOMEM;
                    113:          return NULL;
                    114:        }
                    115:       buffer = (char *) malloc (buf_size);
                    116:       if (buffer == NULL)
                    117:        /* errno is ENOMEM.  */
                    118:        return NULL;
                    119:     }
                    120: }

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>