1 /* pt_chmod - helper program for `grantpt'.
2 Copyright (C) 1998-2016 Free Software Foundation, Inc.
3 This file is part of the GNU C Library.
4 Contributed by C. Scott Ananian <cananian@alumni.princeton.edu>, 1998.
6 The GNU C Library is free software; you can redistribute it and/or
7 modify it under the terms of the GNU Lesser General Public
8 License as published by the Free Software Foundation; either
9 version 2.1 of the License, or (at your option) any later version.
11 The GNU C Library 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 GNU
14 Lesser General Public License for more details.
16 You should have received a copy of the GNU Lesser General Public
17 License along with the GNU C Library; if not, see
18 <http://www.gnu.org/licenses/>. */
33 # include <sys/capability.h>
34 # include <sys/prctl.h>
37 #include "pty-private.h"
39 /* Get libc version number. */
40 #include "../version.h"
42 #define PACKAGE _libc_intl_domainname
44 /* Name and version of program. */
45 static void print_version (FILE *stream
, struct argp_state
*state
);
46 void (*argp_program_version_hook
) (FILE *, struct argp_state
*) = print_version
;
48 /* Function to print some extra text in the help message. */
49 static char *more_help (int key
, const char *text
, void *input
);
51 /* Data structure to communicate with argp functions. */
52 static struct argp argp
=
54 NULL
, NULL
, NULL
, NULL
, NULL
, more_help
58 /* Print the version information. */
60 print_version (FILE *stream
, struct argp_state
*state
)
62 fprintf (stream
, "pt_chown %s%s\n", PKGVERSION
, VERSION
);
63 fprintf (stream
, gettext ("\
64 Copyright (C) %s Free Software Foundation, Inc.\n\
65 This is free software; see the source for copying conditions. There is NO\n\
66 warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.\n\
71 more_help (int key
, const char *text
, void *input
)
78 case ARGP_KEY_HELP_PRE_DOC
:
79 asprintf (&cp
, gettext ("\
80 Set the owner, group and access permission of the slave pseudo\
81 terminal corresponding to the master pseudo terminal passed on\
82 file descriptor `%d'. This is the helper program for the\
83 `grantpt' function. It is not intended to be run directly from\
84 the command line.\n"),
87 case ARGP_KEY_HELP_EXTRA
:
88 /* We print some extra information. */
89 if (asprintf (&tp
, gettext ("\
90 For bug reporting instructions, please see:\n\
91 %s.\n"), REPORT_BUGS_TO
) < 0)
93 if (asprintf (&cp
, gettext ("\
94 The owner is set to the current user, the group is set to `%s',\
95 and the access permission is set to `%o'.\n\n\
97 TTY_GROUP
, S_IRUSR
|S_IWUSR
|S_IWGRP
, tp
) < 0)
106 return (char *) text
;
117 /* Check that PTY_FILENO is a valid master pseudo terminal. */
118 pty
= ptsname (PTY_FILENO
);
120 return errno
== EBADF
? FAIL_EBADF
: FAIL_EINVAL
;
122 /* Check that the returned slave pseudo terminal is a
124 if (stat64 (pty
, &st
) < 0 || !S_ISCHR (st
.st_mode
))
127 /* Get the group ID of the special `tty' group. */
128 p
= getgrnam (TTY_GROUP
);
129 gid
= p
? p
->gr_gid
: getgid ();
131 /* Set the owner to the real user ID, and the group to that special
133 if (chown (pty
, getuid (), gid
) < 0)
136 /* Set the permission mode to readable and writable by the owner,
137 and writable by the group. */
138 if ((st
.st_mode
& ACCESSPERMS
) != (S_IRUSR
|S_IWUSR
|S_IWGRP
)
139 && chmod (pty
, S_IRUSR
|S_IWUSR
|S_IWGRP
) < 0)
147 main (int argc
, char *argv
[])
149 uid_t euid
= geteuid ();
150 uid_t uid
= getuid ();
154 /* Clear any signal mask from the parent process. */
155 sigemptyset (&signalset
);
156 sigprocmask (SIG_SETMASK
, &signalset
, NULL
);
158 if (argc
== 1 && euid
== 0)
161 /* Drop privileges. */
164 static const cap_value_t cap_list
[] =
165 { CAP_CHOWN
, CAP_FOWNER
};
166 # define ncap_list (sizeof (cap_list) / sizeof (cap_list[0]))
167 cap_t caps
= cap_init ();
171 /* There is no reason why these should not work. */
172 cap_set_flag (caps
, CAP_PERMITTED
, ncap_list
, cap_list
, CAP_SET
);
173 cap_set_flag (caps
, CAP_EFFECTIVE
, ncap_list
, cap_list
, CAP_SET
);
175 int res
= cap_set_proc (caps
);
179 if (__glibc_unlikely (res
!= 0))
184 /* Normal invocation of this program is with no arguments and
186 return do_pt_chown ();
189 /* We aren't going to be using privileges, so drop them right now. */
192 /* Set locale via LC_ALL. */
193 setlocale (LC_ALL
, "");
195 /* Set the text message domain. */
196 textdomain (PACKAGE
);
198 /* parse and process arguments. */
199 argp_parse (&argp
, argc
, argv
, 0, &remaining
, NULL
);
201 if (remaining
< argc
)
203 /* We should not be called with any non-option parameters. */
204 error (0, 0, gettext ("too many arguments"));
205 argp_help (&argp
, stdout
, ARGP_HELP_SEE
| ARGP_HELP_EXIT_ERR
,
206 program_invocation_short_name
);
210 /* Check if we are properly installed. */
212 error (FAIL_EXEC
, 0, gettext ("needs to be installed setuid `root'"));