xref: /freebsd/stand/libsa/open.c (revision ca987d4641cdcd7f27e153db17c5bf064934faf5)
1*ca987d46SWarner Losh /*	$NetBSD: open.c,v 1.16 1997/01/28 09:41:03 pk Exp $	*/
2*ca987d46SWarner Losh 
3*ca987d46SWarner Losh /*-
4*ca987d46SWarner Losh  * Copyright (c) 1993
5*ca987d46SWarner Losh  *	The Regents of the University of California.  All rights reserved.
6*ca987d46SWarner Losh  *
7*ca987d46SWarner Losh  * This code is derived from software contributed to Berkeley by
8*ca987d46SWarner Losh  * The Mach Operating System project at Carnegie-Mellon University.
9*ca987d46SWarner Losh  *
10*ca987d46SWarner Losh  * Redistribution and use in source and binary forms, with or without
11*ca987d46SWarner Losh  * modification, are permitted provided that the following conditions
12*ca987d46SWarner Losh  * are met:
13*ca987d46SWarner Losh  * 1. Redistributions of source code must retain the above copyright
14*ca987d46SWarner Losh  *    notice, this list of conditions and the following disclaimer.
15*ca987d46SWarner Losh  * 2. Redistributions in binary form must reproduce the above copyright
16*ca987d46SWarner Losh  *    notice, this list of conditions and the following disclaimer in the
17*ca987d46SWarner Losh  *    documentation and/or other materials provided with the distribution.
18*ca987d46SWarner Losh  * 3. Neither the name of the University nor the names of its contributors
19*ca987d46SWarner Losh  *    may be used to endorse or promote products derived from this software
20*ca987d46SWarner Losh  *    without specific prior written permission.
21*ca987d46SWarner Losh  *
22*ca987d46SWarner Losh  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
23*ca987d46SWarner Losh  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
24*ca987d46SWarner Losh  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
25*ca987d46SWarner Losh  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
26*ca987d46SWarner Losh  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
27*ca987d46SWarner Losh  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
28*ca987d46SWarner Losh  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
29*ca987d46SWarner Losh  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
30*ca987d46SWarner Losh  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
31*ca987d46SWarner Losh  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
32*ca987d46SWarner Losh  * SUCH DAMAGE.
33*ca987d46SWarner Losh  *
34*ca987d46SWarner Losh  *	@(#)open.c	8.1 (Berkeley) 6/11/93
35*ca987d46SWarner Losh  *
36*ca987d46SWarner Losh  *
37*ca987d46SWarner Losh  * Copyright (c) 1989, 1990, 1991 Carnegie Mellon University
38*ca987d46SWarner Losh  * All Rights Reserved.
39*ca987d46SWarner Losh  *
40*ca987d46SWarner Losh  * Author: Alessandro Forin
41*ca987d46SWarner Losh  *
42*ca987d46SWarner Losh  * Permission to use, copy, modify and distribute this software and its
43*ca987d46SWarner Losh  * documentation is hereby granted, provided that both the copyright
44*ca987d46SWarner Losh  * notice and this permission notice appear in all copies of the
45*ca987d46SWarner Losh  * software, derivative works or modified versions, and any portions
46*ca987d46SWarner Losh  * thereof, and that both notices appear in supporting documentation.
47*ca987d46SWarner Losh  *
48*ca987d46SWarner Losh  * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS"
49*ca987d46SWarner Losh  * CONDITION.  CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR
50*ca987d46SWarner Losh  * ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
51*ca987d46SWarner Losh  *
52*ca987d46SWarner Losh  * Carnegie Mellon requests users of this software to return to
53*ca987d46SWarner Losh  *
54*ca987d46SWarner Losh  *  Software Distribution Coordinator  or  Software.Distribution@CS.CMU.EDU
55*ca987d46SWarner Losh  *  School of Computer Science
56*ca987d46SWarner Losh  *  Carnegie Mellon University
57*ca987d46SWarner Losh  *  Pittsburgh PA 15213-3890
58*ca987d46SWarner Losh  *
59*ca987d46SWarner Losh  * any improvements or extensions that they make and grant Carnegie the
60*ca987d46SWarner Losh  * rights to redistribute these changes.
61*ca987d46SWarner Losh  */
62*ca987d46SWarner Losh 
63*ca987d46SWarner Losh #include <sys/cdefs.h>
64*ca987d46SWarner Losh __FBSDID("$FreeBSD$");
65*ca987d46SWarner Losh 
66*ca987d46SWarner Losh #include "stand.h"
67*ca987d46SWarner Losh 
68*ca987d46SWarner Losh struct fs_ops *exclusive_file_system;
69*ca987d46SWarner Losh 
70*ca987d46SWarner Losh struct open_file files[SOPEN_MAX];
71*ca987d46SWarner Losh 
72*ca987d46SWarner Losh static int
73*ca987d46SWarner Losh o_gethandle(void)
74*ca987d46SWarner Losh {
75*ca987d46SWarner Losh     int		fd;
76*ca987d46SWarner Losh 
77*ca987d46SWarner Losh     for (fd = 0; fd < SOPEN_MAX; fd++)
78*ca987d46SWarner Losh 	if (files[fd].f_flags == 0)
79*ca987d46SWarner Losh 	    return(fd);
80*ca987d46SWarner Losh     return(-1);
81*ca987d46SWarner Losh }
82*ca987d46SWarner Losh 
83*ca987d46SWarner Losh static void
84*ca987d46SWarner Losh o_rainit(struct open_file *f)
85*ca987d46SWarner Losh {
86*ca987d46SWarner Losh     f->f_rabuf = malloc(SOPEN_RASIZE);
87*ca987d46SWarner Losh     f->f_ralen = 0;
88*ca987d46SWarner Losh     f->f_raoffset = 0;
89*ca987d46SWarner Losh }
90*ca987d46SWarner Losh 
91*ca987d46SWarner Losh int
92*ca987d46SWarner Losh open(const char *fname, int mode)
93*ca987d46SWarner Losh {
94*ca987d46SWarner Losh     struct fs_ops	*fs;
95*ca987d46SWarner Losh     struct open_file	*f;
96*ca987d46SWarner Losh     int			fd, i, error, besterror;
97*ca987d46SWarner Losh     const char		*file;
98*ca987d46SWarner Losh 
99*ca987d46SWarner Losh     if ((fd = o_gethandle()) == -1) {
100*ca987d46SWarner Losh 	errno = EMFILE;
101*ca987d46SWarner Losh 	return(-1);
102*ca987d46SWarner Losh     }
103*ca987d46SWarner Losh 
104*ca987d46SWarner Losh     f = &files[fd];
105*ca987d46SWarner Losh     f->f_flags = mode + 1;
106*ca987d46SWarner Losh     f->f_dev = (struct devsw *)0;
107*ca987d46SWarner Losh     f->f_ops = (struct fs_ops *)0;
108*ca987d46SWarner Losh     f->f_offset = 0;
109*ca987d46SWarner Losh     f->f_devdata = NULL;
110*ca987d46SWarner Losh     file = (char *)0;
111*ca987d46SWarner Losh 
112*ca987d46SWarner Losh     if (exclusive_file_system != NULL) {
113*ca987d46SWarner Losh 	fs = exclusive_file_system;
114*ca987d46SWarner Losh 	error = (fs->fo_open)(fname, f);
115*ca987d46SWarner Losh 	if (error == 0)
116*ca987d46SWarner Losh 	    goto ok;
117*ca987d46SWarner Losh 	goto err;
118*ca987d46SWarner Losh     }
119*ca987d46SWarner Losh 
120*ca987d46SWarner Losh     error = devopen(f, fname, &file);
121*ca987d46SWarner Losh     if (error ||
122*ca987d46SWarner Losh 	(((f->f_flags & F_NODEV) == 0) && f->f_dev == (struct devsw *)0))
123*ca987d46SWarner Losh 	goto err;
124*ca987d46SWarner Losh 
125*ca987d46SWarner Losh     /* see if we opened a raw device; otherwise, 'file' is the file name. */
126*ca987d46SWarner Losh     if (file == (char *)0 || *file == '\0') {
127*ca987d46SWarner Losh 	f->f_flags |= F_RAW;
128*ca987d46SWarner Losh 	f->f_rabuf = NULL;
129*ca987d46SWarner Losh 	return (fd);
130*ca987d46SWarner Losh     }
131*ca987d46SWarner Losh 
132*ca987d46SWarner Losh     /* pass file name to the different filesystem open routines */
133*ca987d46SWarner Losh     besterror = ENOENT;
134*ca987d46SWarner Losh     for (i = 0; file_system[i] != NULL; i++) {
135*ca987d46SWarner Losh 	fs = file_system[i];
136*ca987d46SWarner Losh 	error = (fs->fo_open)(file, f);
137*ca987d46SWarner Losh 	if (error == 0)
138*ca987d46SWarner Losh 	    goto ok;
139*ca987d46SWarner Losh 	if (error != EINVAL)
140*ca987d46SWarner Losh 	    besterror = error;
141*ca987d46SWarner Losh     }
142*ca987d46SWarner Losh     error = besterror;
143*ca987d46SWarner Losh 
144*ca987d46SWarner Losh  fail:
145*ca987d46SWarner Losh     if ((f->f_flags & F_NODEV) == 0 && f->f_dev != NULL)
146*ca987d46SWarner Losh 	f->f_dev->dv_close(f);
147*ca987d46SWarner Losh     if (error)
148*ca987d46SWarner Losh 	devclose(f);
149*ca987d46SWarner Losh 
150*ca987d46SWarner Losh  err:
151*ca987d46SWarner Losh     f->f_flags = 0;
152*ca987d46SWarner Losh     errno = error;
153*ca987d46SWarner Losh     return (-1);
154*ca987d46SWarner Losh 
155*ca987d46SWarner Losh  ok:
156*ca987d46SWarner Losh     f->f_ops = fs;
157*ca987d46SWarner Losh     o_rainit(f);
158*ca987d46SWarner Losh     return (fd);
159*ca987d46SWarner Losh }
160