i386 log2 implementation.
[kopensolaris-gnu/glibc.git] / sysdeps / unix / grantpt.c
index d216baa..71f73bf 100644 (file)
@@ -1,4 +1,4 @@
-/* Copyright (C) 1998 Free Software Foundation, Inc.
+/* Copyright (C) 1998, 2000 Free Software Foundation, Inc.
    This file is part of the GNU C Library.
    Contributed by Zack Weinberg <zack@rabi.phys.columbia.edu>, 1998.
 
@@ -48,7 +48,7 @@ pts_name (int fd, char **pts, size_t buf_len)
 
       if (buf_len)
        {
-         rv = ptsname_r (fd, buf, buf_len);
+         rv = __ptsname_r (fd, buf, buf_len);
 
          if (rv != 0 || memchr (buf, '\0', buf_len))
            /* We either got an error, or we succeeded and the
@@ -90,13 +90,14 @@ pts_name (int fd, char **pts, size_t buf_len)
 int
 grantpt (int fd)
 {
+  int retval = -1;
 #ifdef PATH_MAX
   char _buf[PATH_MAX];
 #else
   char _buf[512];
 #endif
   char *buf = _buf;
-  struct stat st;
+  struct stat64 st;
   char *grtmpbuf;
   struct group grbuf;
   size_t grbuflen = __sysconf (_SC_GETGR_R_SIZE_MAX);
@@ -107,9 +108,9 @@ grantpt (int fd)
 
   if (pts_name (fd, &buf, sizeof (_buf)))
     return -1;
-  
-  if (__stat (buf, &st) < 0)
-    return -1;
+
+  if (__xstat64 (_STAT_VER, buf, &st) < 0)
+    goto cleanup;
 
   /* Make sure that we own the device.  */
   uid = __getuid ();
@@ -125,7 +126,7 @@ grantpt (int fd)
        Try a moderate value.  */
     grbuflen = 1024;
   grtmpbuf = (char *) __alloca (grbuflen);
-  getgrnam_r (TTY_GROUP, &grbuf, grtmpbuf, grbuflen, &p);
+  __getgrnam_r (TTY_GROUP, &grbuf, grtmpbuf, grbuflen, &p);
   gid = p ? p->gr_gid : __getgid ();
 
   /* Make sure the group of the device is that special group.  */
@@ -143,19 +144,20 @@ grantpt (int fd)
        goto helper;
     }
 
-  return 0;
+  retval = 0;
+  goto cleanup;
 
   /* We have to use the helper program.  */
  helper:
 
   pid = __fork ();
   if (pid == -1)
-    return -1;
+    goto cleanup;
   else if (pid == 0)
     {
       /* Disable core dumps.  */
       struct rlimit rl = { 0, 0 };
-      setrlimit (RLIMIT_CORE, &rl);
+      __setrlimit (RLIMIT_CORE, &rl);
 
       /* We pase the master pseudo terminal as file descriptor PTY_FILENO.  */
       if (fd != PTY_FILENO)
@@ -168,36 +170,38 @@ grantpt (int fd)
   else
     {
       int w;
-      
+
       if (__waitpid (pid, &w, 0) == -1)
-       return -1;
+       goto cleanup;
       if (!WIFEXITED (w))
-       {
-         __set_errno (ENOEXEC);
-         return -1;
-       }
+       __set_errno (ENOEXEC);
       else
        switch (WEXITSTATUS(w))
          {
          case 0:
+           retval = 0;
            break;
          case FAIL_EBADF:
            __set_errno (EBADF);
-           return -1;
+           break;
          case FAIL_EINVAL:
            __set_errno (EINVAL);
-           return -1;
+           break;
          case FAIL_EACCES:
            __set_errno (EACCES);
-           return -1;
+           break;
          case FAIL_EXEC:
            __set_errno (ENOEXEC);
-           return -1;
+           break;
 
          default:
            assert(! "getpt: internal error: invalid exit code from pt_chown");
          }
     }
 
-  return 0;
+ cleanup:
+  if (buf != _buf)
+    free (buf);
+
+  return retval;
 }