xref: /onnv-gate/usr/src/lib/libc/port/sys/open.c (revision 11798:1e7f1f154004)
1*11798SRoger.Faulkner@Sun.COM /*
2*11798SRoger.Faulkner@Sun.COM  * CDDL HEADER START
3*11798SRoger.Faulkner@Sun.COM  *
4*11798SRoger.Faulkner@Sun.COM  * The contents of this file are subject to the terms of the
5*11798SRoger.Faulkner@Sun.COM  * Common Development and Distribution License (the "License").
6*11798SRoger.Faulkner@Sun.COM  * You may not use this file except in compliance with the License.
7*11798SRoger.Faulkner@Sun.COM  *
8*11798SRoger.Faulkner@Sun.COM  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9*11798SRoger.Faulkner@Sun.COM  * or http://www.opensolaris.org/os/licensing.
10*11798SRoger.Faulkner@Sun.COM  * See the License for the specific language governing permissions
11*11798SRoger.Faulkner@Sun.COM  * and limitations under the License.
12*11798SRoger.Faulkner@Sun.COM  *
13*11798SRoger.Faulkner@Sun.COM  * When distributing Covered Code, include this CDDL HEADER in each
14*11798SRoger.Faulkner@Sun.COM  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15*11798SRoger.Faulkner@Sun.COM  * If applicable, add the following below this CDDL HEADER, with the
16*11798SRoger.Faulkner@Sun.COM  * fields enclosed by brackets "[]" replaced with your own identifying
17*11798SRoger.Faulkner@Sun.COM  * information: Portions Copyright [yyyy] [name of copyright owner]
18*11798SRoger.Faulkner@Sun.COM  *
19*11798SRoger.Faulkner@Sun.COM  * CDDL HEADER END
20*11798SRoger.Faulkner@Sun.COM  */
21*11798SRoger.Faulkner@Sun.COM 
22*11798SRoger.Faulkner@Sun.COM /*
23*11798SRoger.Faulkner@Sun.COM  * Copyright 2010 Sun Microsystems, Inc.  All rights reserved.
24*11798SRoger.Faulkner@Sun.COM  * Use is subject to license terms.
25*11798SRoger.Faulkner@Sun.COM  */
26*11798SRoger.Faulkner@Sun.COM 
27*11798SRoger.Faulkner@Sun.COM /*	Copyright (c) 1988 AT&T	*/
28*11798SRoger.Faulkner@Sun.COM /*	  All Rights Reserved  	*/
29*11798SRoger.Faulkner@Sun.COM 
30*11798SRoger.Faulkner@Sun.COM #include "lint.h"
31*11798SRoger.Faulkner@Sun.COM #include <sys/mkdev.h>
32*11798SRoger.Faulkner@Sun.COM #include <limits.h>
33*11798SRoger.Faulkner@Sun.COM #include <stdarg.h>
34*11798SRoger.Faulkner@Sun.COM #include <unistd.h>
35*11798SRoger.Faulkner@Sun.COM #include <strings.h>
36*11798SRoger.Faulkner@Sun.COM #include <errno.h>
37*11798SRoger.Faulkner@Sun.COM #include <sys/stat.h>
38*11798SRoger.Faulkner@Sun.COM #include <sys/fcntl.h>
39*11798SRoger.Faulkner@Sun.COM #include <sys/stropts.h>
40*11798SRoger.Faulkner@Sun.COM #include <sys/stream.h>
41*11798SRoger.Faulkner@Sun.COM #include <sys/ptms.h>
42*11798SRoger.Faulkner@Sun.COM #include <sys/syscall.h>
43*11798SRoger.Faulkner@Sun.COM #include "libc.h"
44*11798SRoger.Faulkner@Sun.COM 
45*11798SRoger.Faulkner@Sun.COM static int xpg4_fixup(int fd);
46*11798SRoger.Faulkner@Sun.COM static void push_module(int fd);
47*11798SRoger.Faulkner@Sun.COM static int isptsfd(int fd);
48*11798SRoger.Faulkner@Sun.COM static void itoa(int i, char *ptr);
49*11798SRoger.Faulkner@Sun.COM 
50*11798SRoger.Faulkner@Sun.COM int
__openat(int dfd,const char * path,int oflag,mode_t mode)51*11798SRoger.Faulkner@Sun.COM __openat(int dfd, const char *path, int oflag, mode_t mode)
52*11798SRoger.Faulkner@Sun.COM {
53*11798SRoger.Faulkner@Sun.COM 	int fd = syscall(SYS_openat, dfd, path, oflag, mode);
54*11798SRoger.Faulkner@Sun.COM 	return (xpg4_fixup(fd));
55*11798SRoger.Faulkner@Sun.COM }
56*11798SRoger.Faulkner@Sun.COM 
57*11798SRoger.Faulkner@Sun.COM int
__open(const char * path,int oflag,mode_t mode)58*11798SRoger.Faulkner@Sun.COM __open(const char *path, int oflag, mode_t mode)
59*11798SRoger.Faulkner@Sun.COM {
60*11798SRoger.Faulkner@Sun.COM #if defined(_RETAIN_OLD_SYSCALLS)
61*11798SRoger.Faulkner@Sun.COM 	int fd = syscall(SYS_open, path, oflag, mode);
62*11798SRoger.Faulkner@Sun.COM 	return (xpg4_fixup(fd));
63*11798SRoger.Faulkner@Sun.COM #else
64*11798SRoger.Faulkner@Sun.COM 	return (__openat(AT_FDCWD, path, oflag, mode));
65*11798SRoger.Faulkner@Sun.COM #endif
66*11798SRoger.Faulkner@Sun.COM }
67*11798SRoger.Faulkner@Sun.COM 
68*11798SRoger.Faulkner@Sun.COM #if !defined(_LP64)
69*11798SRoger.Faulkner@Sun.COM 
70*11798SRoger.Faulkner@Sun.COM int
__openat64(int dfd,const char * path,int oflag,mode_t mode)71*11798SRoger.Faulkner@Sun.COM __openat64(int dfd, const char *path, int oflag, mode_t mode)
72*11798SRoger.Faulkner@Sun.COM {
73*11798SRoger.Faulkner@Sun.COM 	int fd = syscall(SYS_openat64, dfd, path, oflag, mode);
74*11798SRoger.Faulkner@Sun.COM 	return (xpg4_fixup(fd));
75*11798SRoger.Faulkner@Sun.COM }
76*11798SRoger.Faulkner@Sun.COM 
77*11798SRoger.Faulkner@Sun.COM int
__open64(const char * path,int oflag,mode_t mode)78*11798SRoger.Faulkner@Sun.COM __open64(const char *path, int oflag, mode_t mode)
79*11798SRoger.Faulkner@Sun.COM {
80*11798SRoger.Faulkner@Sun.COM #if defined(_RETAIN_OLD_SYSCALLS)
81*11798SRoger.Faulkner@Sun.COM 	int fd = syscall(SYS_open64, path, oflag, mode);
82*11798SRoger.Faulkner@Sun.COM 	return (xpg4_fixup(fd));
83*11798SRoger.Faulkner@Sun.COM #else
84*11798SRoger.Faulkner@Sun.COM 	return (__openat64(AT_FDCWD, path, oflag, mode));
85*11798SRoger.Faulkner@Sun.COM #endif
86*11798SRoger.Faulkner@Sun.COM }
87*11798SRoger.Faulkner@Sun.COM 
88*11798SRoger.Faulkner@Sun.COM #endif	/* !_LP64 */
89*11798SRoger.Faulkner@Sun.COM 
90*11798SRoger.Faulkner@Sun.COM /*
91*11798SRoger.Faulkner@Sun.COM  * XPG4v2 requires that open of a slave pseudo terminal device
92*11798SRoger.Faulkner@Sun.COM  * provides the process with an interface that is identical to
93*11798SRoger.Faulkner@Sun.COM  * the terminal interface. For a more detailed discussion,
94*11798SRoger.Faulkner@Sun.COM  * see bugid 4025044.
95*11798SRoger.Faulkner@Sun.COM  */
96*11798SRoger.Faulkner@Sun.COM static int
xpg4_fixup(int fd)97*11798SRoger.Faulkner@Sun.COM xpg4_fixup(int fd)
98*11798SRoger.Faulkner@Sun.COM {
99*11798SRoger.Faulkner@Sun.COM 	if (libc__xpg4 != 0 && fd >= 0 && isptsfd(fd))
100*11798SRoger.Faulkner@Sun.COM 		push_module(fd);
101*11798SRoger.Faulkner@Sun.COM 	return (fd);
102*11798SRoger.Faulkner@Sun.COM }
103*11798SRoger.Faulkner@Sun.COM 
104*11798SRoger.Faulkner@Sun.COM /*
105*11798SRoger.Faulkner@Sun.COM  * Check if the file matches an entry in the /dev/pts directory.
106*11798SRoger.Faulkner@Sun.COM  * Be careful to preserve errno.
107*11798SRoger.Faulkner@Sun.COM  */
108*11798SRoger.Faulkner@Sun.COM static int
isptsfd(int fd)109*11798SRoger.Faulkner@Sun.COM isptsfd(int fd)
110*11798SRoger.Faulkner@Sun.COM {
111*11798SRoger.Faulkner@Sun.COM 	char buf[TTYNAME_MAX];
112*11798SRoger.Faulkner@Sun.COM 	char *str1 = buf;
113*11798SRoger.Faulkner@Sun.COM 	const char *str2 = "/dev/pts/";
114*11798SRoger.Faulkner@Sun.COM 	struct stat64 fsb, stb;
115*11798SRoger.Faulkner@Sun.COM 	int oerrno = errno;
116*11798SRoger.Faulkner@Sun.COM 	int rval = 0;
117*11798SRoger.Faulkner@Sun.COM 
118*11798SRoger.Faulkner@Sun.COM 	if (fstat64(fd, &fsb) == 0 && S_ISCHR(fsb.st_mode)) {
119*11798SRoger.Faulkner@Sun.COM 		/*
120*11798SRoger.Faulkner@Sun.COM 		 * Do this without strcpy() or strlen(),
121*11798SRoger.Faulkner@Sun.COM 		 * to avoid invoking the dynamic linker.
122*11798SRoger.Faulkner@Sun.COM 		 */
123*11798SRoger.Faulkner@Sun.COM 		while (*str2 != '\0')
124*11798SRoger.Faulkner@Sun.COM 			*str1++ = *str2++;
125*11798SRoger.Faulkner@Sun.COM 		/*
126*11798SRoger.Faulkner@Sun.COM 		 * Inline version of minor(dev), to avoid the dynamic linker.
127*11798SRoger.Faulkner@Sun.COM 		 */
128*11798SRoger.Faulkner@Sun.COM 		itoa(fsb.st_rdev & MAXMIN, str1);
129*11798SRoger.Faulkner@Sun.COM 		if (stat64(buf, &stb) == 0)
130*11798SRoger.Faulkner@Sun.COM 			rval = (stb.st_rdev == fsb.st_rdev);
131*11798SRoger.Faulkner@Sun.COM 	}
132*11798SRoger.Faulkner@Sun.COM 	errno = oerrno;
133*11798SRoger.Faulkner@Sun.COM 	return (rval);
134*11798SRoger.Faulkner@Sun.COM }
135*11798SRoger.Faulkner@Sun.COM 
136*11798SRoger.Faulkner@Sun.COM /*
137*11798SRoger.Faulkner@Sun.COM  * Converts a number to a string (null terminated).
138*11798SRoger.Faulkner@Sun.COM  */
139*11798SRoger.Faulkner@Sun.COM static void
itoa(int i,char * ptr)140*11798SRoger.Faulkner@Sun.COM itoa(int i, char *ptr)
141*11798SRoger.Faulkner@Sun.COM {
142*11798SRoger.Faulkner@Sun.COM 	int dig = 0;
143*11798SRoger.Faulkner@Sun.COM 	int tempi;
144*11798SRoger.Faulkner@Sun.COM 
145*11798SRoger.Faulkner@Sun.COM 	tempi = i;
146*11798SRoger.Faulkner@Sun.COM 	do {
147*11798SRoger.Faulkner@Sun.COM 		dig++;
148*11798SRoger.Faulkner@Sun.COM 		tempi /= 10;
149*11798SRoger.Faulkner@Sun.COM 	} while (tempi);
150*11798SRoger.Faulkner@Sun.COM 
151*11798SRoger.Faulkner@Sun.COM 	ptr += dig;
152*11798SRoger.Faulkner@Sun.COM 	*ptr = '\0';
153*11798SRoger.Faulkner@Sun.COM 	while (--dig >= 0) {
154*11798SRoger.Faulkner@Sun.COM 		*(--ptr) = i % 10 + '0';
155*11798SRoger.Faulkner@Sun.COM 		i /= 10;
156*11798SRoger.Faulkner@Sun.COM 	}
157*11798SRoger.Faulkner@Sun.COM }
158*11798SRoger.Faulkner@Sun.COM 
159*11798SRoger.Faulkner@Sun.COM /*
160*11798SRoger.Faulkner@Sun.COM  * Push modules to provide tty semantics
161*11798SRoger.Faulkner@Sun.COM  */
162*11798SRoger.Faulkner@Sun.COM static void
push_module(int fd)163*11798SRoger.Faulkner@Sun.COM push_module(int fd)
164*11798SRoger.Faulkner@Sun.COM {
165*11798SRoger.Faulkner@Sun.COM 	struct strioctl istr;
166*11798SRoger.Faulkner@Sun.COM 	int oerrno = errno;
167*11798SRoger.Faulkner@Sun.COM 
168*11798SRoger.Faulkner@Sun.COM 	istr.ic_cmd = PTSSTTY;
169*11798SRoger.Faulkner@Sun.COM 	istr.ic_len = 0;
170*11798SRoger.Faulkner@Sun.COM 	istr.ic_timout = 0;
171*11798SRoger.Faulkner@Sun.COM 	istr.ic_dp = NULL;
172*11798SRoger.Faulkner@Sun.COM 	if (ioctl(fd, I_STR, &istr) != -1) {
173*11798SRoger.Faulkner@Sun.COM 		(void) ioctl(fd, __I_PUSH_NOCTTY, "ptem");
174*11798SRoger.Faulkner@Sun.COM 		(void) ioctl(fd, __I_PUSH_NOCTTY, "ldterm");
175*11798SRoger.Faulkner@Sun.COM 		(void) ioctl(fd, __I_PUSH_NOCTTY, "ttcompat");
176*11798SRoger.Faulkner@Sun.COM 		istr.ic_cmd = PTSSTTY;
177*11798SRoger.Faulkner@Sun.COM 		istr.ic_len = 0;
178*11798SRoger.Faulkner@Sun.COM 		istr.ic_timout = 0;
179*11798SRoger.Faulkner@Sun.COM 		istr.ic_dp = NULL;
180*11798SRoger.Faulkner@Sun.COM 		(void) ioctl(fd, I_STR, &istr);
181*11798SRoger.Faulkner@Sun.COM 	}
182*11798SRoger.Faulkner@Sun.COM 	errno = oerrno;
183*11798SRoger.Faulkner@Sun.COM }
184