]> git.saurik.com Git - apple/xnu.git/blob - bsd/miscfs/synthfs/synthfs_vnops.c
xnu-123.5.tar.gz
[apple/xnu.git] / bsd / miscfs / synthfs / synthfs_vnops.c
1 /*
2 * Copyright (c) 2000 Apple Computer, Inc. All rights reserved.
3 *
4 * @APPLE_LICENSE_HEADER_START@
5 *
6 * The contents of this file constitute Original Code as defined in and
7 * are subject to the Apple Public Source License Version 1.1 (the
8 * "License"). You may not use this file except in compliance with the
9 * License. Please obtain a copy of the License at
10 * http://www.apple.com/publicsource and read it before using this file.
11 *
12 * This Original Code and all software distributed under the License are
13 * distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, EITHER
14 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
15 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
16 * FITNESS FOR A PARTICULAR PURPOSE OR NON-INFRINGEMENT. Please see the
17 * License for the specific language governing rights and limitations
18 * under the License.
19 *
20 * @APPLE_LICENSE_HEADER_END@
21 */
22 /*
23 * Copyright (c) 1998-1999 Apple Computer, Inc. All Rights Reserved.
24 *
25 * Modification History:
26 *
27 * 02-Feb-2000 Clark Warner Added copyfile to table
28 * 17-Aug-1999 Pat Dirks New today.
29 */
30
31 #include <mach/mach_types.h>
32 #include <mach/machine/boolean.h>
33 #include <sys/param.h>
34 #include <sys/systm.h>
35 #include <sys/kernel.h>
36 #include <sys/file.h>
37 #include <sys/stat.h>
38 #include <sys/buf.h>
39 #include <sys/proc.h>
40 #include <sys/conf.h>
41 #include <sys/mount.h>
42 #include <sys/vnode.h>
43 #include <sys/malloc.h>
44 #include <sys/dirent.h>
45 #include <sys/namei.h>
46 #include <sys/attr.h>
47
48 #include <sys/vm.h>
49 #include <sys/errno.h>
50 #include <vfs/vfs_support.h>
51
52 #include "synthfs.h"
53
54 #define RWSUPPORT 0
55
56 #if RWSUPPORT
57 #error NOT PORTED FOR UBC
58 /* when porting to UBC, do not just replace
59 * vnode_uncache by ubc_uncache - there's more
60 * to it than that!
61 */
62 #include <sys/ubc.h>
63 #endif
64
65 /* external routines defined in vfs_cache.c */
66 extern void cache_purge (struct vnode *vp);
67 extern int cache_lookup (struct vnode *dvp, struct vnode **vpp, struct componentname *cnp);
68 extern void cache_enter (struct vnode *dvp, struct vnode *vpp, struct componentname *cnp);
69
70 //extern void vnode_uncache(struct vnode *);
71
72 extern int groupmember(gid_t gid, struct ucred* cred);
73
74 #define VOPFUNC int (*)(void *)
75
76 /* Global vfs data structures for synthfs. */
77 int (**synthfs_vnodeop_p) (void *);
78 struct vnodeopv_entry_desc synthfs_vnodeop_entries[] = {
79 {&vop_default_desc, (VOPFUNC)vn_default_error},
80 {&vop_strategy_desc, (VOPFUNC)err_strategy}, /* strategy - not supported */
81 {&vop_bwrite_desc, (VOPFUNC)err_bwrite}, /* bwrite - not supported */
82 {&vop_lookup_desc, (VOPFUNC)synthfs_cached_lookup}, /* cached lookup */
83 {&vop_create_desc, (VOPFUNC)synthfs_create}, /* create - DEBUGGER */
84 {&vop_whiteout_desc, (VOPFUNC)err_whiteout}, /* whiteout - not supported */
85 {&vop_mknod_desc, (VOPFUNC)err_mknod}, /* mknod - not supported */
86 {&vop_mkcomplex_desc, (VOPFUNC)err_mkcomplex}, /* mkcomplex - not supported */
87 {&vop_open_desc, (VOPFUNC)synthfs_open}, /* open - DEBUGGER */
88 {&vop_close_desc, (VOPFUNC)nop_close}, /* close - NOP */
89 {&vop_access_desc, (VOPFUNC)synthfs_access}, /* access */
90 {&vop_getattr_desc, (VOPFUNC)synthfs_getattr}, /* getattr */
91 {&vop_setattr_desc, (VOPFUNC)synthfs_setattr}, /* setattr */
92 {&vop_getattrlist_desc, (VOPFUNC)err_getattrlist}, /* getattrlist - not supported */
93 {&vop_setattrlist_desc, (VOPFUNC)err_setattrlist}, /* setattrlist - not supported */
94 {&vop_read_desc, (VOPFUNC)err_read}, /* read - not supported */
95 {&vop_write_desc, (VOPFUNC)err_write}, /* write - not supported */
96 {&vop_lease_desc, (VOPFUNC)err_lease}, /* lease - not supported */
97 {&vop_ioctl_desc, (VOPFUNC)err_ioctl}, /* ioctl - not supported */
98 {&vop_select_desc, (VOPFUNC)synthfs_select}, /* select */
99 {&vop_exchange_desc, (VOPFUNC)err_exchange}, /* exchange - not supported */
100 {&vop_revoke_desc, (VOPFUNC)nop_revoke}, /* revoke - NOP */
101 {&vop_mmap_desc, (VOPFUNC)synthfs_mmap}, /* mmap - DEBUGGER */
102 {&vop_fsync_desc, (VOPFUNC)nop_fsync}, /* fsync - NOP */
103 {&vop_seek_desc, (VOPFUNC)nop_seek}, /* seek - NOP */
104 {&vop_remove_desc, (VOPFUNC)synthfs_remove}, /* remove */
105 {&vop_link_desc, (VOPFUNC)err_link}, /* link - not supported */
106 {&vop_rename_desc, (VOPFUNC)synthfs_rename}, /* rename */
107 {&vop_mkdir_desc, (VOPFUNC)synthfs_mkdir}, /* mkdir */
108 {&vop_rmdir_desc, (VOPFUNC)synthfs_rmdir}, /* rmdir */
109 {&vop_symlink_desc, (VOPFUNC)synthfs_symlink}, /* symlink */
110 {&vop_readdir_desc, (VOPFUNC)synthfs_readdir}, /* readdir */
111 {&vop_readdirattr_desc, (VOPFUNC)err_readdirattr}, /* readdirattr - not supported */
112 {&vop_readlink_desc, (VOPFUNC)synthfs_readlink}, /* readlink */
113 {&vop_abortop_desc, (VOPFUNC)nop_abortop}, /* abortop - NOP */
114 {&vop_inactive_desc, (VOPFUNC)synthfs_inactive}, /* inactive */
115 {&vop_reclaim_desc, (VOPFUNC)synthfs_reclaim}, /* reclaim */
116 {&vop_lock_desc, (VOPFUNC)synthfs_lock}, /* lock */
117 {&vop_unlock_desc, (VOPFUNC)synthfs_unlock}, /* unlock */
118 {&vop_bmap_desc, (VOPFUNC)err_bmap}, /* bmap - not supported */
119 {&vop_print_desc, (VOPFUNC)err_print}, /* print - not supported */
120 {&vop_islocked_desc, (VOPFUNC)synthfs_islocked}, /* islocked */
121 {&vop_pathconf_desc, (VOPFUNC)synthfs_pathconf}, /* pathconf */
122 {&vop_advlock_desc, (VOPFUNC)err_advlock}, /* advlock - not supported */
123 {&vop_blkatoff_desc, (VOPFUNC)err_blkatoff}, /* blkatoff - not supported */
124 {&vop_valloc_desc, (VOPFUNC)err_valloc}, /* valloc - not supported */
125 {&vop_reallocblks_desc, (VOPFUNC)err_reallocblks}, /* reallocblks - not supported */
126 {&vop_vfree_desc, (VOPFUNC)err_vfree}, /* vfree - not supported */
127 {&vop_truncate_desc, (VOPFUNC)err_truncate}, /* truncate - not supported */
128 {&vop_allocate_desc, (VOPFUNC)err_allocate}, /* allocate - not supported */
129 {&vop_update_desc, (VOPFUNC)synthfs_update}, /* update */
130 {&vop_pgrd_desc, (VOPFUNC)err_pgrd}, /* pgrd - not supported */
131 {&vop_pgwr_desc, (VOPFUNC)err_pgwr}, /* pgwr - not supported */
132 {&vop_pagein_desc, (VOPFUNC)err_pagein}, /* pagein - not supported */
133 {&vop_pageout_desc, (VOPFUNC)err_pageout}, /* pageout - not supported */
134 {&vop_devblocksize_desc, (VOPFUNC)err_devblocksize}, /* devblocksize - not supported */
135 {&vop_searchfs_desc, (VOPFUNC)err_searchfs}, /* searchfs - not supported */
136 {&vop_copyfile_desc, (VOPFUNC)err_copyfile}, /* copyfile - not supported */
137 { &vop_blktooff_desc, (VOPFUNC)err_blktooff }, /* blktooff not supported */
138 { &vop_offtoblk_desc, (VOPFUNC)err_offtoblk }, /* offtoblk not supported */
139 { &vop_cmap_desc, (VOPFUNC)err_cmap }, /* cmap not supported */
140 {(struct vnodeop_desc *) NULL, (int (*) ()) NULL}
141 };
142
143 /*
144 * Oh what a tangled web we weave. This structure will be used by
145 * bsd/vfs/vfs_conf.c to actually do the initialization of synthfs_vnodeop_p
146 */
147 struct vnodeopv_desc synthfs_vnodeop_opv_desc =
148 {&synthfs_vnodeop_p, synthfs_vnodeop_entries};
149
150
151
152 /*
153 * Create a regular file
154 #% create dvp L U U
155 #% create vpp - L -
156 #
157 vop_create {
158 IN WILLRELE struct vnode *dvp;
159 OUT struct vnode **vpp;
160 IN struct componentname *cnp;
161 IN struct vattr *vap;
162
163 We are responsible for freeing the namei buffer, it is done in hfs_makenode(), unless there is
164 a previous error.
165
166 */
167
168 int
169 synthfs_create(ap)
170 struct vop_create_args /* {
171 struct vnode *a_dvp;
172 struct vnode **a_vpp;
173 struct componentname *a_cnp;
174 struct vattr *a_vap;
175 } */ *ap;
176 {
177 #if DEBUG
178 struct vnode *dvp = ap->a_dvp;
179 char debugmsg[255];
180
181 sprintf(debugmsg, "synthfs_create: attempt to create '%s' in '%s' ?!", ap->a_cnp->cn_nameptr, VTOS(dvp)->s_name);
182 Debugger(debugmsg);
183 #endif
184
185 return EOPNOTSUPP;
186 }
187
188
189
190 /*
191 * Open called.
192 #% open vp L L L
193 #
194 vop_open {
195 IN struct vnode *vp;
196 IN int mode;
197 IN struct ucred *cred;
198 IN struct proc *p;
199 */
200
201 int
202 synthfs_open(ap)
203 struct vop_open_args /* {
204 struct vnode *a_vp;
205 int a_mode;
206 struct ucred *a_cred;
207 struct proc *a_p;
208 } */ *ap;
209 {
210 struct vnode *vp = ap->a_vp;
211
212 if (vp->v_type == VDIR) {
213 return 0;
214 } else {
215 #if DEBUG
216 struct synthfsnode *sp = VTOS(vp);
217 char debugmsg[255];
218
219 sprintf(debugmsg, "synthfs_open: attempt to open '/%s' ?!", sp->s_name);
220 Debugger(debugmsg);
221 #endif
222 };
223
224 return 0;
225 }
226
227
228
229 /*
230 * Mmap a file
231 *
232 * NB Currently unsupported.
233 # XXX - not used
234 #
235 vop_mmap {
236 IN struct vnode *vp;
237 IN int fflags;
238 IN struct ucred *cred;
239 IN struct proc *p;
240
241 */
242
243 /* ARGSUSED */
244
245 int
246 synthfs_mmap(ap)
247 struct vop_mmap_args /* {
248 struct vnode *a_vp;
249 int a_fflags;
250 struct ucred *a_cred;
251 struct proc *a_p;
252 } */ *ap;
253 {
254 #if DEBUG
255 struct vnode *vp = ap->a_vp;
256 char debugmsg[255];
257
258 sprintf(debugmsg, "synthfs_mmap: attempt to map '/%s' ?!", VTOS(vp)->s_name);
259 Debugger(debugmsg);
260 #endif
261
262 return EINVAL;
263 }
264
265
266
267 /*
268 #% access vp L L L
269 #
270 vop_access {
271 IN struct vnode *vp;
272 IN int mode;
273 IN struct ucred *cred;
274 IN struct proc *p;
275
276 */
277
278 int
279 synthfs_access(ap)
280 struct vop_access_args /* {
281 struct vnode *a_vp;
282 int a_mode;
283 struct ucred *a_cred;
284 struct proc *a_p;
285 } */ *ap;
286 {
287 struct vnode *vp = ap->a_vp;
288 mode_t mode = ap->a_mode;
289 struct ucred *cred = ap->a_cred;
290 struct synthfsnode *sp = VTOS(vp);
291 register gid_t *gp;
292 mode_t mask;
293 int retval = 0;
294 int i;
295
296 /*
297 * Disallow write attempts on read-only file systems;
298 * unless the file is a socket, fifo, or a block or
299 * character device resident on the file system.
300 */
301 if (mode & VWRITE) {
302 switch (vp->v_type) {
303 case VDIR:
304 case VLNK:
305 case VREG:
306 if (VTOVFS(vp)->mnt_flag & MNT_RDONLY)
307 return (EROFS);
308 break;
309 default:
310 break;
311 }
312 }
313
314 /* If immutable bit set, nobody gets to write it. */
315 if ((mode & VWRITE) && (sp->s_flags & IMMUTABLE))
316 return (EPERM);
317
318 /* Otherwise, user id 0 always gets access. */
319 if (ap->a_cred->cr_uid == 0) {
320 retval = 0;
321 goto Exit;
322 };
323
324 mask = 0;
325
326 /* Otherwise, check the owner. */
327 if (cred->cr_uid == sp->s_uid) {
328 if (mode & VEXEC)
329 mask |= S_IXUSR;
330 if (mode & VREAD)
331 mask |= S_IRUSR;
332 if (mode & VWRITE)
333 mask |= S_IWUSR;
334 retval = ((sp->s_mode & mask) == mask ? 0 : EACCES);
335 goto Exit;
336 }
337
338 /* Otherwise, check the groups. */
339 for (i = 0, gp = cred->cr_groups; i < cred->cr_ngroups; i++, gp++)
340 if (sp->s_gid == *gp) {
341 if (mode & VEXEC)
342 mask |= S_IXGRP;
343 if (mode & VREAD)
344 mask |= S_IRGRP;
345 if (mode & VWRITE)
346 mask |= S_IWGRP;
347 retval = ((sp->s_mode & mask) == mask ? 0 : EACCES);
348 goto Exit;
349 }
350
351 /* Otherwise, check everyone else. */
352 if (mode & VEXEC)
353 mask |= S_IXOTH;
354 if (mode & VREAD)
355 mask |= S_IROTH;
356 if (mode & VWRITE)
357 mask |= S_IWOTH;
358 retval = ((sp->s_mode & mask) == mask ? 0 : EACCES);
359
360 Exit:
361 return (retval);
362 }
363
364 /*
365 #% getattr vp = = =
366 #
367 vop_getattr {
368 IN struct vnode *vp;
369 IN struct vattr *vap;
370 IN struct ucred *cred;
371 IN struct proc *p;
372
373 */
374 int
375 synthfs_getattr(ap)
376 struct vop_getattr_args /* {
377 struct vnode *a_vp;
378 struct vattr *a_vap;
379 struct ucred *a_cred;
380 struct proc *a_p;
381 } */ *ap;
382 {
383 struct vnode *vp = ap->a_vp;
384 struct vattr *vap = ap->a_vap;
385 struct synthfsnode *sp = VTOS(vp);
386 struct synthfs_mntdata *smp = VTOSFS(vp);
387
388 vap->va_type = vp->v_type;
389 vap->va_mode = sp->s_mode;
390 vap->va_nlink = sp->s_linkcount;
391 vap->va_uid = sp->s_uid;
392 vap->va_gid = sp->s_gid;
393 vap->va_fsid = VTOVFS(vp)->mnt_stat.f_fsid.val[0];
394 vap->va_fileid = sp->s_nodeid;
395 switch (vp->v_type) {
396 case VDIR:
397 vap->va_size = (sp->s_u.d.d_entrycount + 2) * sizeof(struct dirent);
398 break;
399
400 case VREG:
401 vap->va_size = sp->s_u.f.f_size;
402 break;
403
404 case VLNK:
405 vap->va_size = sp->s_u.s.s_length;
406 break;
407
408 default:
409 vap->va_size = 0;
410 };
411 vap->va_blocksize = 512;
412 vap->va_atime.tv_sec = sp->s_accesstime.tv_sec;
413 vap->va_atime.tv_nsec = sp->s_accesstime.tv_usec * 1000;
414 vap->va_mtime.tv_sec = sp->s_modificationtime.tv_sec;
415 vap->va_mtime.tv_nsec = sp->s_modificationtime.tv_usec * 1000;
416 vap->va_ctime.tv_sec = sp->s_changetime.tv_sec;
417 vap->va_ctime.tv_nsec = sp->s_changetime.tv_usec * 1000;
418 vap->va_gen = sp->s_generation;
419 vap->va_flags = sp->s_flags;
420 vap->va_rdev = sp->s_rdev;
421 vap->va_bytes = vap->va_blocksize * ((vap->va_size + vap->va_blocksize - 1) / vap->va_blocksize);
422 vap->va_filerev = 0;
423 vap->va_vaflags = 0;
424
425 return (0);
426 }
427
428
429
430 /*
431 * Change the mode on a file or directory.
432 * vnode vp must be locked on entry.
433 */
434 int synthfs_chmod(struct vnode *vp, int mode, struct ucred *cred, struct proc *p)
435 {
436 struct synthfsnode *sp = VTOS(vp);
437 int result;
438
439 if ((cred->cr_uid != sp->s_uid) &&
440 (result = suser(cred, &p->p_acflag)))
441 return result;
442 if (cred->cr_uid) {
443 if (vp->v_type != VDIR && (mode & S_ISTXT))
444 return EFTYPE;
445 if (!groupmember(sp->s_gid, cred) && (mode & S_ISGID))
446 return (EPERM);
447 }
448 sp->s_mode &= ~ALLPERMS;
449 sp->s_mode |= (mode & ALLPERMS);
450 sp->s_nodeflags |= IN_CHANGE;
451 #if RWSUPPORT
452 if ((vp->v_flag & VTEXT) && (sp->s_mode & S_ISTXT) == 0) (void) vnode_uncache(vp);
453 #endif
454
455 return 0;
456 }
457
458
459
460 /*
461 * Change the flags on a file or directory.
462 * vnode vp must be locked on entry.
463 */
464 int synthfs_chflags(struct vnode *vp, u_long flags, struct ucred *cred, struct proc *p)
465 {
466 struct synthfsnode *sp = VTOS(vp);
467 int result;
468
469 if (cred->cr_uid != sp->s_uid &&
470 (result = suser(cred, &p->p_acflag)))
471 return result;
472
473 if (cred->cr_uid == 0) {
474 if ((sp->s_flags & (SF_IMMUTABLE | SF_APPEND)) &&
475 securelevel > 0) {
476 return EPERM;
477 };
478 sp->s_flags = flags;
479 } else {
480 if (sp->s_flags & (SF_IMMUTABLE | SF_APPEND) ||
481 (flags & UF_SETTABLE) != flags) {
482 return EPERM;
483 };
484 sp->s_flags &= SF_SETTABLE;
485 sp->s_flags |= (flags & UF_SETTABLE);
486 }
487 sp->s_nodeflags |= IN_CHANGE;
488
489 return 0;
490 }
491
492
493
494 /*
495 * Perform chown operation on vnode vp;
496 * vnode vp must be locked on entry.
497 */
498 int synthfs_chown(struct vnode *vp, uid_t uid, gid_t gid, struct ucred *cred, struct proc *p)
499 {
500 struct synthfsnode *sp = VTOS(vp);
501 uid_t ouid;
502 gid_t ogid;
503 int result = 0;
504
505 if (uid == (uid_t)VNOVAL) uid = sp->s_uid;
506 if (gid == (gid_t)VNOVAL) gid = sp->s_gid;
507
508 /*
509 * If we don't own the file, are trying to change the owner
510 * of the file, or are not a member of the target group,
511 * the caller must be superuser or the call fails.
512 */
513 if ((cred->cr_uid != sp->s_uid || uid != sp->s_uid ||
514 (gid != sp->s_gid && !groupmember((gid_t)gid, cred))) &&
515 (result = suser(cred, &p->p_acflag)))
516 return result;
517
518 ogid = sp->s_gid;
519 ouid = sp->s_uid;
520
521 sp->s_gid = gid;
522 sp->s_uid = uid;
523
524 if (ouid != uid || ogid != gid) sp->s_nodeflags |= IN_CHANGE;
525 if (ouid != uid && cred->cr_uid != 0) sp->s_mode &= ~S_ISUID;
526 if (ogid != gid && cred->cr_uid != 0) sp->s_mode &= ~S_ISGID;
527
528 return 0;
529 }
530
531
532
533 /*
534 * Set attribute vnode op. called from several syscalls
535 #% setattr vp L L L
536 #
537 vop_setattr {
538 IN struct vnode *vp;
539 IN struct vattr *vap;
540 IN struct ucred *cred;
541 IN struct proc *p;
542
543 */
544
545 int
546 synthfs_setattr(ap)
547 struct vop_setattr_args /* {
548 struct vnode *a_vp;
549 struct vattr *a_vap;
550 struct ucred *a_cred;
551 struct proc *a_p;
552 } */ *ap;
553 {
554 struct vnode *vp = ap->a_vp;
555 struct synthfsnode *sp = VTOS(vp);
556 struct vattr *vap = ap->a_vap;
557 struct ucred *cred = ap->a_cred;
558 struct proc *p = ap->a_p;
559 struct timeval atimeval, mtimeval;
560 int result;
561
562 /*
563 * Check for unsettable attributes.
564 */
565 if (((vap->va_type != VNON) && (vap->va_type != vp->v_type)) ||
566 (vap->va_nlink != VNOVAL) ||
567 (vap->va_fsid != VNOVAL) || (vap->va_fileid != VNOVAL) ||
568 (vap->va_blocksize != VNOVAL) || (vap->va_rdev != VNOVAL) ||
569 ((int)vap->va_bytes != VNOVAL) || (vap->va_gen != VNOVAL)) {
570 result = EINVAL;
571 goto Err_Exit;
572 }
573
574 if (vap->va_flags != VNOVAL) {
575 if (VTOVFS(vp)->mnt_flag & MNT_RDONLY) {
576 result = EROFS;
577 goto Err_Exit;
578 };
579 if ((result = synthfs_chflags(vp, vap->va_flags, cred, p))) {
580 goto Err_Exit;
581 };
582 if (vap->va_flags & (IMMUTABLE | APPEND)) {
583 result = 0;
584 goto Err_Exit;
585 };
586 }
587
588 if (sp->s_flags & (IMMUTABLE | APPEND)) {
589 result = EPERM;
590 goto Err_Exit;
591 };
592
593 /*
594 * Go through the fields and update iff not VNOVAL.
595 */
596 if (vap->va_uid != (uid_t)VNOVAL || vap->va_gid != (gid_t)VNOVAL) {
597 if (VTOVFS(vp)->mnt_flag & MNT_RDONLY) {
598 result = EROFS;
599 goto Err_Exit;
600 };
601 if ((result = synthfs_chown(vp, vap->va_uid, vap->va_gid, cred, p))) {
602 goto Err_Exit;
603 };
604 }
605 if (vap->va_size != VNOVAL) {
606 /*
607 * Disallow write attempts on read-only file systems;
608 * unless the file is a socket, fifo, or a block or
609 * character device resident on the file system.
610 */
611 switch (vp->v_type) {
612 case VDIR:
613 result = EISDIR;
614 goto Err_Exit;
615 case VLNK:
616 case VREG:
617 if (VTOVFS(vp)->mnt_flag & MNT_RDONLY) {
618 result = EROFS;
619 goto Err_Exit;
620 };
621 break;
622 default:
623 break;
624 }
625 #if RWSUPPORT
626 if ((result = VOP_TRUNCATE(vp, vap->va_size, 0, cred, p))) {
627 goto Err_Exit;
628 };
629 #else
630 result = EINVAL;
631 goto Err_Exit;
632 #endif
633 }
634
635 sp = VTOS(vp);
636 if (vap->va_atime.tv_sec != VNOVAL || vap->va_mtime.tv_sec != VNOVAL) {
637 if (VTOVFS(vp)->mnt_flag & MNT_RDONLY) {
638 result = EROFS;
639 goto Err_Exit;
640 };
641 if (cred->cr_uid != sp->s_uid &&
642 (result = suser(cred, &p->p_acflag)) &&
643 ((vap->va_vaflags & VA_UTIMES_NULL) == 0 ||
644 (result = VOP_ACCESS(vp, VWRITE, cred, p)))) {
645 goto Err_Exit;
646 };
647 if (vap->va_atime.tv_sec != VNOVAL)
648 sp->s_nodeflags |= IN_ACCESS;
649 if (vap->va_mtime.tv_sec != VNOVAL)
650 sp->s_nodeflags |= IN_CHANGE | IN_UPDATE;
651 atimeval.tv_sec = vap->va_atime.tv_sec;
652 atimeval.tv_usec = vap->va_atime.tv_nsec / 1000;
653 mtimeval.tv_sec = vap->va_mtime.tv_sec;
654 mtimeval.tv_usec = vap->va_mtime.tv_nsec / 1000;
655 if ((result = VOP_UPDATE(vp, &atimeval, &mtimeval, 1))) {
656 goto Err_Exit;
657 };
658 }
659
660 result = 0;
661 if (vap->va_mode != (mode_t)VNOVAL) {
662 if (VTOVFS(vp)->mnt_flag & MNT_RDONLY) {
663 result = EROFS;
664 goto Err_Exit;
665 };
666 result = synthfs_chmod(vp, (int)vap->va_mode, cred, p);
667 };
668
669 Err_Exit: ;
670
671 DBG_VOP(("synthfs_setattr: returning %d...\n", result));
672
673 return (result);
674 }
675
676
677
678 /*
679
680 #% rename sourcePar_vp U U U
681 #% rename source_vp U U U
682 #% rename targetPar_vp L U U
683 #% rename target_vp X U U
684 #
685 vop_rename {
686 IN WILLRELE struct vnode *sourcePar_vp;
687 IN WILLRELE struct vnode *source_vp;
688 IN struct componentname *source_cnp;
689 IN WILLRELE struct vnode *targetPar_vp;
690 IN WILLRELE struct vnode *target_vp;
691 IN struct componentname *target_cnp;
692
693
694 */
695
696 /*
697 * On entry:
698 * source's parent directory is unlocked
699 * source file or directory is unlocked
700 * destination's parent directory is locked
701 * destination file or directory is locked if it exists
702 *
703 * On exit:
704 * all denodes should be released
705 *
706 */
707
708 int
709 synthfs_rename(ap)
710 struct vop_rename_args /* {
711 struct vnode *a_fdvp;
712 struct vnode *a_fvp;
713 struct componentname *a_fcnp;
714 struct vnode *a_tdvp;
715 struct vnode *a_tvp;
716 struct componentname *a_tcnp;
717 } */ *ap;
718 {
719 struct vnode *target_vp = ap->a_tvp;
720 struct vnode *targetPar_vp = ap->a_tdvp;
721 struct vnode *source_vp = ap->a_fvp;
722 struct vnode *sourcePar_vp = ap->a_fdvp;
723 struct componentname *target_cnp = ap->a_tcnp;
724 struct componentname *source_cnp = ap->a_fcnp;
725 struct proc *p = source_cnp->cn_proc;
726 struct synthfsnode *target_sp, *targetPar_sp, *source_sp, *sourcePar_sp;
727 u_short doingdirectory = 0, oldparent = 0, newparent = 0;
728 int retval = 0;
729 struct timeval tv;
730
731 #if SYNTHFS_DIAGNOSTIC
732 if ((target_cnp->cn_flags & HASBUF) == 0 ||
733 (source_cnp->cn_flags & HASBUF) == 0)
734 panic("synthfs_rename: no name");
735 #endif
736
737 DBG_ASSERT((ap->a_fdvp->v_type == VDIR) && (ap->a_tdvp->v_type == VDIR));
738 target_sp = targetPar_sp = source_sp = sourcePar_sp = NULL;
739
740 /*
741 * Check for cross-device rename.
742 */
743 if ((source_vp->v_mount != targetPar_vp->v_mount) ||
744 (target_vp && (source_vp->v_mount != target_vp->v_mount))) {
745 retval = EXDEV;
746 goto abortit;
747 }
748
749 /*
750 * Check for access permissions
751 */
752 if (target_vp && ((VTOS(target_vp)->s_pflags & (IMMUTABLE | APPEND)) ||
753 (VTOS(targetPar_vp)->s_pflags & APPEND))) {
754 retval = EPERM;
755 goto abortit;
756 }
757
758 if ((retval = vn_lock(source_vp, LK_EXCLUSIVE, p)))
759 goto abortit;
760
761 sourcePar_sp = VTOS(sourcePar_vp);
762 source_sp = VTOS(source_vp);
763 oldparent = sourcePar_sp->s_nodeid;
764 if ((source_sp->s_pflags & (IMMUTABLE | APPEND)) || (sourcePar_sp->s_pflags & APPEND)) {
765 VOP_UNLOCK(source_vp, 0, p);
766 retval = EPERM;
767 goto abortit;
768 }
769
770 /*
771 * Be sure we are not renaming ".", "..", or an alias of ".". This
772 * leads to a crippled directory tree. It's pretty tough to do a
773 * "ls" or "pwd" with the "." directory entry missing, and "cd .."
774 * doesn't work if the ".." entry is missing.
775 */
776 if (source_sp->s_type == SYNTHFS_DIRECTORY) {
777 if ((source_cnp->cn_namelen == 1 && source_cnp->cn_nameptr[0] == '.')
778 || sourcePar_sp == source_sp
779 || (source_cnp->cn_flags & ISDOTDOT)
780 || (source_sp->s_nodeflags & IN_RENAME)) {
781 VOP_UNLOCK(source_vp, 0, p);
782 retval = EINVAL;
783 goto abortit;
784 }
785 source_sp->s_nodeflags |= IN_RENAME;
786 doingdirectory = TRUE;
787 }
788
789 /* Transit between abort and bad */
790
791 targetPar_sp = VTOS(targetPar_vp);
792 target_sp = target_vp ? VTOS(target_vp) : NULL;
793 newparent = targetPar_sp->s_nodeid;
794
795 retval = VOP_ACCESS(source_vp, VWRITE, target_cnp->cn_cred, target_cnp->cn_proc);
796 if (doingdirectory && (newparent != oldparent)) {
797 if (retval) /* write access check above */
798 goto bad;
799 }
800
801 /*
802 * If the destination exists, then be sure its type (file or dir)
803 * matches that of the source. And, if it is a directory make sure
804 * it is empty. Then delete the destination.
805 */
806 if (target_vp) {
807 /*
808 * If the parent directory is "sticky", then the user must
809 * own the parent directory, or the destination of the rename,
810 * otherwise the destination may not be changed (except by
811 * root). This implements append-only directories.
812 */
813 if ((targetPar_sp->s_mode & S_ISTXT) && target_cnp->cn_cred->cr_uid != 0 &&
814 target_cnp->cn_cred->cr_uid != targetPar_sp->s_uid &&
815 target_sp->s_uid != target_cnp->cn_cred->cr_uid) {
816 retval = EPERM;
817 goto bad;
818 }
819
820 /*
821 * VOP_REMOVE will vput targetPar_vp so we better bump
822 * its ref count and relockit, always set target_vp to
823 * NULL afterwards to indicate that were done with it.
824 */
825 VREF(targetPar_vp);
826 #if RWSUPPORT
827 if (target_vp->v_type == VREG) {
828 (void) vnode_uncache(target_vp);
829 };
830 #endif
831 cache_purge(target_vp);
832
833 target_cnp->cn_flags &= ~SAVENAME;
834 retval = VOP_REMOVE(targetPar_vp, target_vp, target_cnp);
835 (void) vn_lock(targetPar_vp, LK_EXCLUSIVE | LK_RETRY, p);
836
837 target_vp = NULL;
838 target_sp = NULL;
839
840 if (retval) goto bad;
841 };
842
843
844 if (newparent != oldparent)
845 vn_lock(sourcePar_vp, LK_EXCLUSIVE | LK_RETRY, p);
846
847 /* remove the existing entry from the namei cache: */
848 if (source_vp->v_type == VREG) cache_purge(source_vp);
849
850 retval = synthfs_move_rename_entry( source_vp, targetPar_vp, target_cnp->cn_nameptr);
851
852 if (newparent != oldparent)
853 VOP_UNLOCK(sourcePar_vp, 0, p);
854
855 if (retval) goto bad;
856
857 source_sp->s_nodeflags &= ~IN_RENAME;
858
859 /*
860 * Timestamp both parent directories.
861 * Note that if this is a rename within the same directory,
862 * (where targetPar_hp == sourcePar_hp)
863 * the code below is still safe and correct.
864 */
865 targetPar_sp->s_nodeflags |= IN_UPDATE;
866 sourcePar_sp->s_nodeflags |= IN_UPDATE;
867 tv = time;
868 SYNTHFSTIMES(targetPar_sp, &tv, &tv);
869 SYNTHFSTIMES(sourcePar_sp, &tv, &tv);
870
871 vput(targetPar_vp);
872 vrele(sourcePar_vp);
873 vput(source_vp);
874
875 return (retval);
876
877 bad:;
878 if (retval && doingdirectory)
879 source_sp->s_nodeflags &= ~IN_RENAME;
880
881 if (targetPar_vp == target_vp)
882 vrele(targetPar_vp);
883 else
884 vput(targetPar_vp);
885
886 if (target_vp)
887 vput(target_vp);
888
889 vrele(sourcePar_vp);
890
891 if (VOP_ISLOCKED(source_vp))
892 vput(source_vp);
893 else
894 vrele(source_vp);
895
896 return (retval);
897
898 abortit:;
899
900 VOP_ABORTOP(targetPar_vp, target_cnp); /* XXX, why not in NFS? */
901
902 if (targetPar_vp == target_vp)
903 vrele(targetPar_vp);
904 else
905 vput(targetPar_vp);
906
907 if (target_vp)
908 vput(target_vp);
909
910 VOP_ABORTOP(sourcePar_vp, source_cnp); /* XXX, why not in NFS? */
911
912 vrele(sourcePar_vp);
913 vrele(source_vp);
914
915 return (retval);
916 }
917
918
919
920 /*
921 * Mkdir system call
922
923 #% mkdir dvp L U U
924 #% mkdir vpp - L -
925 #
926 vop_mkdir {
927 IN WILLRELE struct vnode *dvp;
928 OUT struct vnode **vpp;
929 IN struct componentname *cnp;
930 IN struct vattr *vap;
931
932 We are responsible for freeing the namei buffer, it is done in synthfs_makenode(), unless there is
933 a previous error.
934
935 */
936
937 int
938 synthfs_mkdir(ap)
939 struct vop_mkdir_args /* {
940 struct vnode *a_dvp;
941 struct vnode **a_vpp;
942 struct componentname *a_cnp;
943 struct vattr *a_vap;
944 } */ *ap;
945 {
946 int retval;
947 struct vnode *dvp = ap->a_dvp;
948 struct componentname *cnp = ap->a_cnp;
949 int mode = MAKEIMODE(ap->a_vap->va_type, ap->a_vap->va_mode);
950 struct vnode *vp = NULL;
951
952 *ap->a_vpp = NULL;
953
954 retval = synthfs_new_directory(VTOVFS(dvp), dvp, cnp->cn_nameptr, VTOSFS(dvp)->synthfs_nextid++, mode, ap->a_cnp->cn_proc, &vp);
955 if (retval) goto Error_Exit;
956
957 retval = VOP_SETATTR(vp, ap->a_vap, cnp->cn_cred, cnp->cn_proc);
958 if (retval != 0) goto Error_Exit;
959
960 *ap->a_vpp = vp;
961
962 Error_Exit:;
963 if (retval != 0) {
964 if (vp) synthfs_remove_directory(vp);
965 VOP_ABORTOP(dvp, cnp);
966 }
967 vput(dvp);
968
969 return retval;
970 }
971
972
973
974 /*
975
976 #% remove dvp L U U
977 #% remove vp L U U
978 #
979 vop_remove {
980 IN WILLRELE struct vnode *dvp;
981 IN WILLRELE struct vnode *vp;
982 IN struct componentname *cnp;
983
984 */
985
986 int
987 synthfs_remove(ap)
988 struct vop_remove_args /* {
989 struct vnode *a_dvp;
990 struct vnode *a_vp;
991 struct componentname *a_cnp;
992 } */ *ap;
993 {
994 struct vnode *vp = ap->a_vp;
995 struct vnode *dvp = ap->a_dvp;
996 struct synthfsnode *sp = VTOS(vp);
997 struct timeval tv;
998 int retval = 0;
999
1000 if ((sp->s_flags & (IMMUTABLE | APPEND)) ||
1001 (VTOS(dvp)->s_flags & APPEND)) {
1002 retval = EPERM;
1003 goto out;
1004 };
1005
1006 /* This is sort of silly right now but someday it may make sense... */
1007 if (sp->s_nodeflags & IN_MODIFIED) {
1008 tv = time;
1009 VOP_UPDATE(vp, &tv, &tv, 0);
1010 };
1011
1012 /* remove the entry from the namei cache: */
1013 cache_purge(vp);
1014
1015 /* remove entry from tree and reclaim any resources consumed: */
1016 switch (sp->s_type) {
1017 case SYNTHFS_DIRECTORY:
1018 synthfs_remove_directory(vp);
1019 break;
1020
1021
1022 case SYNTHFS_SYMLINK:
1023 synthfs_remove_symlink(vp);
1024 break;
1025
1026 case SYNTHFS_FILE:
1027 /* Fall through to default case */
1028
1029 default:
1030 synthfs_remove_entry(vp);
1031 };
1032
1033 out:
1034
1035 if (! retval)
1036 VTOS(dvp)->s_nodeflags |= IN_CHANGE | IN_UPDATE;
1037
1038 if (dvp == vp) {
1039 vrele(vp);
1040 } else {
1041 vput(vp);
1042 };
1043
1044 vput(dvp);
1045 return (retval);
1046 }
1047
1048
1049
1050 /*
1051 #% rmdir dvp L U U
1052 #% rmdir vp L U U
1053 #
1054 vop_rmdir {
1055 IN WILLRELE struct vnode *dvp;
1056 IN WILLRELE struct vnode *vp;
1057 IN struct componentname *cnp;
1058
1059 */
1060
1061 int
1062 synthfs_rmdir(ap)
1063 struct vop_rmdir_args /* {
1064 struct vnode *a_dvp;
1065 struct vnode *a_vp;
1066 struct componentname *a_cnp;
1067 } */ *ap;
1068 {
1069 DBG_VOP(("synthfs_rmdir called\n"));
1070 return synthfs_remove((struct vop_remove_args *)ap);
1071 }
1072
1073
1074
1075 /*
1076 * synthfs_select - just say OK. Only possible op is readdir
1077 *
1078 * Locking policy: ignore
1079 */
1080 int
1081 synthfs_select(ap)
1082 struct vop_select_args /* {
1083 struct vnode *a_vp;
1084 int a_which;
1085 int a_fflags;
1086 struct ucred *a_cred;
1087 struct proc *a_p;
1088 } */ *ap;
1089 {
1090 DBG_VOP(("synthfs_select called\n"));
1091
1092 return (1);
1093 }
1094
1095 /*
1096 #
1097 #% symlink dvp L U U
1098 #% symlink vpp - U -
1099 #
1100 # XXX - note that the return vnode has already been vrele'ed
1101 # by the filesystem layer. To use it you must use vget,
1102 # possibly with a further namei.
1103 #
1104 vop_symlink {
1105 IN WILLRELE struct vnode *dvp;
1106 OUT WILLRELE struct vnode **vpp;
1107 IN struct componentname *cnp;
1108 IN struct vattr *vap;
1109 IN char *target;
1110
1111 We are responsible for freeing the namei buffer, it is done in synthfs_makenode(), unless there is
1112 a previous error.
1113
1114
1115 */
1116
1117 int
1118 synthfs_symlink(ap)
1119 struct vop_symlink_args /* {
1120 struct vnode *a_dvp;
1121 struct vnode **a_vpp;
1122 struct componentname *a_cnp;
1123 struct vattr *a_vap;
1124 char *a_target;
1125 } */ *ap;
1126 {
1127 struct vnode *dvp = ap->a_dvp;
1128 struct vnode **vpp = ap->a_vpp;
1129 struct componentname *cnp = ap->a_cnp;
1130 int retval;
1131
1132 *vpp = NULL;
1133
1134 retval = synthfs_new_symlink(VTOVFS(dvp), dvp, cnp->cn_nameptr, VTOSFS(dvp)->synthfs_nextid++, ap->a_target, ap->a_cnp->cn_proc, vpp);
1135 if (retval) goto Error_Exit;
1136
1137 VOP_UNLOCK(*vpp, 0, cnp->cn_proc);
1138
1139 Error_Exit:;
1140
1141 if (retval != 0) {
1142 VOP_ABORTOP(dvp, cnp);
1143 }
1144 vput(dvp);
1145
1146 return (retval);
1147 }
1148
1149
1150
1151 /*
1152 #
1153 #% readlink vp L L L
1154 #
1155 vop_readlink {
1156 IN struct vnode *vp;
1157 INOUT struct uio *uio;
1158 IN struct ucred *cred;
1159 */
1160
1161 int
1162 synthfs_readlink(ap)
1163 struct vop_readlink_args /* {
1164 struct vnode *a_vp;
1165 struct uio *a_uio;
1166 struct ucred *a_cred;
1167 } */ *ap;
1168 {
1169 struct vnode *vp = ap->a_vp;
1170 struct synthfsnode *sp = VTOS(vp);
1171 struct uio *uio = ap->a_uio;
1172 int retval;
1173 unsigned long count;
1174
1175 if (ap->a_uio->uio_offset > sp->s_u.s.s_length) {
1176 return 0;
1177 };
1178
1179 if (uio->uio_offset + uio->uio_resid <= sp->s_u.s.s_length) {
1180 count = uio->uio_resid;
1181 } else {
1182 count = sp->s_u.s.s_length - uio->uio_offset;
1183 };
1184 retval = uiomove((void *)((unsigned char *)sp->s_u.s.s_symlinktarget + uio->uio_offset), count, uio);
1185 return (retval);
1186
1187 }
1188
1189
1190
1191
1192
1193
1194 /*
1195 #% readdir vp L L L
1196 #
1197 vop_readdir {
1198 IN struct vnode *vp;
1199 INOUT struct uio *uio;
1200 IN struct ucred *cred;
1201 INOUT int *eofflag;
1202 OUT int *ncookies;
1203 INOUT u_long **cookies;
1204 */
1205
1206
1207 int
1208 synthfs_readdir(ap)
1209 struct vop_readdir_args /* {
1210 struct vnode *vp;
1211 struct uio *uio;
1212 struct ucred *cred;
1213 int *eofflag;
1214 int *ncookies;
1215 u_long **cookies;
1216 } */ *ap;
1217 {
1218 struct synthfsnode *sp = VTOS(ap->a_vp);
1219 register struct uio *uio = ap->a_uio;
1220 off_t diroffset; /* Offset into simulated directory file */
1221 struct synthfsnode *entry;
1222
1223 DBG_VOP(("\tuio_offset = %d, uio_resid = %d\n", (int) uio->uio_offset, uio->uio_resid));
1224
1225 /* We assume it's all one big buffer... */
1226 if (uio->uio_iovcnt > 1) {
1227 DBG_VOP(("\tuio->uio_iovcnt = %d?\n", uio->uio_iovcnt));
1228 return EINVAL;
1229 };
1230
1231 /*
1232 NFS cookies are not supported:
1233 */
1234 if ((ap->a_cookies != NULL) || (ap->a_ncookies != NULL)) {
1235 return EINVAL;
1236 };
1237
1238 diroffset = 0;
1239
1240 /*
1241 * We must synthesize . and ..
1242 */
1243 DBG_VOP(("\tstarting ... uio_offset = %d, uio_resid = %d\n", (int) uio->uio_offset, uio->uio_resid));
1244 if (uio->uio_offset == diroffset)
1245 {
1246 DBG_VOP(("\tAdding .\n"));
1247 diroffset += synthfs_adddirentry(sp->s_nodeid, DT_DIR, ".", uio);
1248 DBG_VOP(("\t after adding ., uio_offset = %d, uio_resid = %d\n", (int) uio->uio_offset, uio->uio_resid));
1249 }
1250 if ((uio->uio_resid > 0) && (diroffset > uio->uio_offset)) {
1251 /* Oops - we skipped over a partial entry: at best, diroffset should've just matched uio->uio_offset */
1252 return EINVAL;
1253 };
1254
1255 if (uio->uio_offset == diroffset)
1256 {
1257 DBG_VOP(("\tAdding ..\n"));
1258 if (sp->s_parent != NULL) {
1259 diroffset += synthfs_adddirentry(sp->s_parent->s_nodeid, DT_DIR, "..", uio);
1260 } else {
1261 diroffset += synthfs_adddirentry(sp->s_nodeid, DT_DIR, "..", uio);
1262 }
1263 DBG_VOP(("\t after adding .., uio_offset = %d, uio_resid = %d\n", (int) uio->uio_offset, uio->uio_resid));
1264 }
1265 if ((uio->uio_resid > 0) && (diroffset > uio->uio_offset)) {
1266 /* Oops - we skipped over a partial entry: at best, diroffset should've just matched uio->uio_offset */
1267 return EINVAL;
1268 };
1269
1270 /* OK, so much for the fakes. Now for the "real thing": */
1271 TAILQ_FOREACH(entry, &sp->s_u.d.d_subnodes, s_sibling) {
1272 if (diroffset == uio->uio_offset) {
1273 /* Return this entry */
1274 diroffset += synthfs_adddirentry(entry->s_nodeid, VTTOIF(STOV(entry)->v_type), entry->s_name, uio);
1275 };
1276 if ((uio->uio_resid > 0) && (diroffset > uio->uio_offset)) {
1277 /* Oops - we skipped over a partial entry: at best, diroffset should've just matched uio->uio_offset */
1278 return EINVAL;
1279 };
1280 };
1281
1282 if (ap->a_eofflag)
1283 *ap->a_eofflag = (entry == NULL); /* If we ran all the way through the list, there is no more */
1284
1285 return 0;
1286 }
1287
1288
1289
1290 /*
1291
1292 #% lookup dvp L ? ?
1293 #% lookup vpp - L -
1294
1295 */
1296
1297 int
1298 synthfs_cached_lookup(ap)
1299 struct vop_cachedlookup_args /* {
1300 struct vnode *a_dvp;
1301 struct vnode **a_vpp;
1302 struct componentname *a_cnp;
1303 } */ *ap;
1304 {
1305 struct vnode *dp = ap->a_dvp;
1306 struct componentname *cnp = ap->a_cnp;
1307 u_long nameiop = cnp->cn_nameiop;
1308 u_long flags = cnp->cn_flags;
1309 boolean_t lockparent = (flags & LOCKPARENT);
1310 struct proc *p = cnp->cn_proc;
1311 struct ucred *cred = cnp->cn_cred;
1312 struct vnode *target_vp = NULL;
1313 u_int32_t target_vnode_id; /* Capability ID of the target vnode for .. unlock/relock handling check */
1314 struct vnode **vpp = ap->a_vpp;
1315 int result = 0;
1316
1317 DBG_VOP(("synthfs_cached_lookup called, name = %s, namelen = %ld\n", ap->a_cnp->cn_nameptr, ap->a_cnp->cn_namelen));
1318 if (flags & LOCKPARENT) DBG_VOP(("\tLOCKPARENT is set\n"));
1319 if (flags & ISLASTCN) DBG_VOP(("\tISLASTCN is set\n"));
1320
1321 *vpp = NULL;
1322
1323 if (dp->v_type != VDIR) {
1324 result = ENOTDIR;
1325 goto Err_Exit;
1326 };
1327
1328 if ((flags & ISLASTCN) &&
1329 (VTOVFS(dp)->mnt_flag & MNT_RDONLY) &&
1330 ((nameiop == DELETE) || (nameiop == RENAME))) {
1331 result = EROFS;
1332 goto Err_Exit;
1333 };
1334
1335 result = VOP_ACCESS(dp, VEXEC, cred, cnp->cn_proc);
1336 if (result != 0) goto Err_Exit;
1337
1338 /*
1339 * Look up an entry in the namei cache
1340 */
1341
1342 result = cache_lookup(dp, vpp, cnp);
1343 if (result == 0) {
1344 /* There was no entry in the cache for this parent vnode/name pair:
1345 do the full-blown pathname lookup
1346 */
1347 return synthfs_lookup(ap);
1348 };
1349 if (result == ENOENT) return result;
1350
1351 /* An entry matching the parent vnode/name was found in the cache: */
1352
1353
1354 target_vp = *vpp;
1355 target_vnode_id = target_vp->v_id;
1356 if (target_vp == dp) {
1357 /* lookup on "." */
1358 VREF(target_vp);
1359 result = 0;
1360 } else if (flags & ISDOTDOT) {
1361 /*
1362 * Carefully now: trying to step from child to parent;
1363 * must release lock on child before trying to lock parent
1364 * vnode.
1365 */
1366 VOP_UNLOCK(dp, 0, p);
1367 result = vget(target_vp, LK_EXCLUSIVE, p);
1368 if ((result == 0) && lockparent && (flags & ISLASTCN)) {
1369 result = vn_lock(dp, LK_EXCLUSIVE, p);
1370 }
1371 } else {
1372 result = vget(target_vp, LK_EXCLUSIVE, p);
1373 if (!lockparent || (result != 0) || !(flags & ISLASTCN)) {
1374 VOP_UNLOCK(dp, 0, p);
1375 };
1376 };
1377
1378 /*
1379 Check to make sure the target vnode ID didn't change while we
1380 tried to lock it:
1381 */
1382 if (result == 0) {
1383 if (target_vnode_id == target_vp->v_id) {
1384 return 0; /* THIS IS THE NORMAL EXIT PATH */
1385 };
1386
1387 /* The vnode ID didn't match anymore: we've got another vnode! */
1388 vput(target_vp);
1389 /* Unlock the parent vnode in the cases where it should've been left locked: */
1390 if (lockparent && (dp != target_vp) && (flags & ISLASTCN)) {
1391 VOP_UNLOCK(dp, 0, p);
1392 };
1393 };
1394
1395 /* One last try for a successful lookup through the complete lookup path: */
1396 result = vn_lock(dp, LK_EXCLUSIVE, p);
1397 if (result == 0) {
1398 return synthfs_lookup(ap);
1399 };
1400
1401 Err_Exit:;
1402 return result;
1403 }
1404
1405
1406
1407 int
1408 synthfs_lookup(ap)
1409 struct vop_cachedlookup_args /* {
1410 struct vnode *a_dvp;
1411 struct vnode **a_vpp;
1412 struct componentname *a_cnp;
1413 } */ *ap;
1414 {
1415 struct vnode *dp = ap->a_dvp;
1416 struct synthfsnode *dsp = VTOS(dp);
1417 struct componentname *cnp = ap->a_cnp;
1418 u_long nameiop = cnp->cn_nameiop;
1419 // char *nameptr = cnp->cn_nameptr;
1420 u_long flags = cnp->cn_flags;
1421 long namelen = cnp->cn_namelen;
1422 struct proc *p = cnp->cn_proc;
1423 struct ucred *cred = cnp->cn_cred;
1424 struct synthfsnode *entry;
1425 struct vnode *target_vp = NULL;
1426 int result = 0;
1427 boolean_t found = FALSE;
1428 boolean_t isDot = FALSE;
1429 boolean_t isDotDot = FALSE;
1430 struct vnode *starting_parent = dp;
1431
1432 DBG_VOP(("synthfs_lookup called, name = %s, namelen = %ld\n", ap->a_cnp->cn_nameptr, ap->a_cnp->cn_namelen));
1433 if (flags & LOCKPARENT) DBG_VOP(("\tLOCKPARENT is set\n"));
1434 if (flags & ISLASTCN) DBG_VOP(("\tISLASTCN is set\n"));
1435
1436 *ap->a_vpp = NULL;
1437
1438 if (dp->v_type != VDIR) {
1439 result = ENOTDIR;
1440 goto Err_Exit;
1441 };
1442
1443 if ((flags & ISLASTCN) &&
1444 (VTOVFS(dp)->mnt_flag & MNT_RDONLY) &&
1445 ((nameiop == DELETE) || (nameiop == RENAME))) {
1446 result = EROFS;
1447 goto Err_Exit;
1448 };
1449
1450 result = VOP_ACCESS(dp, VEXEC, cred, cnp->cn_proc);
1451 if (result != 0) goto Err_Exit;
1452
1453 /* first check for "." and ".." */
1454 if (cnp->cn_nameptr[0] == '.') {
1455 if (namelen == 1) {
1456 /*
1457 "." requested
1458 */
1459 isDot = TRUE;
1460 found = TRUE;
1461
1462 target_vp = dp;
1463 VREF(target_vp);
1464
1465 result = 0;
1466
1467 goto Std_Exit;
1468 } else if ((namelen == 2) && (cnp->cn_nameptr[1] == '.')) {
1469 /*
1470 ".." requested
1471 */
1472 isDotDot = TRUE;
1473 found = TRUE;
1474
1475 if ((dsp->s_parent != NULL) && (dsp->s_parent != VTOS(dp))) {
1476 target_vp = STOV(dsp->s_parent);
1477 /*
1478 * Special case for ".." to prevent deadlock:
1479 * always release the parent vnode BEFORE trying to acquire
1480 * ITS parent. This avoids deadlocking with another lookup
1481 * starting from the target_vp trying to vget() this directory.
1482 */
1483 VOP_UNLOCK(dp, 0, p);
1484 result = vget(target_vp, LK_EXCLUSIVE | LK_RETRY, p);
1485 if (result != 0) {
1486 vn_lock(dp, LK_EXCLUSIVE | LK_RETRY, p);
1487 goto Err_Exit;
1488 }
1489 if ((flags & LOCKPARENT) && (flags & ISLASTCN)) {
1490 result = vn_lock(dp, LK_EXCLUSIVE, p);
1491 // vput(target_vp); /* XXX WHY WAS THIS HERE? */
1492 }
1493 } else {
1494 target_vp = dp;
1495 /* dp is alread locked and ref'ed */
1496 result = 0;
1497 }
1498
1499 goto Std_Exit;
1500 }
1501 }
1502
1503 /* finally, just look for entries by name (making sure the entry's length
1504 matches the cnp's namelen... */
1505 TAILQ_FOREACH(entry, &dsp->s_u.d.d_subnodes, s_sibling) {
1506 if ((bcmp(cnp->cn_nameptr, entry->s_name, (unsigned)namelen) == 0) &&
1507 (*(entry->s_name + namelen) == (char)0)) {
1508 found = TRUE;
1509 target_vp = STOV(entry);
1510 result = vget(target_vp, LK_EXCLUSIVE | LK_RETRY, p); /* vget is not really needed because refcount is always > 0... */
1511 if (result != 0) {
1512 vrele(target_vp);
1513 goto Err_Exit;
1514 };
1515
1516 /* The specified entry was found and successfully acquired: */
1517 goto Std_Exit;
1518 };
1519 };
1520
1521 found = FALSE;
1522
1523 Std_Exit:;
1524 if (found) {
1525 if ((nameiop == DELETE) && (flags & ISLASTCN)) {
1526 /*
1527 * Deleting entries requires write access:
1528 */
1529 result = VOP_ACCESS(dp, VWRITE, cred, p);
1530 if (result != 0) goto Err_Exit;
1531
1532 /*
1533 * If the parent directory is "sticky" then the user must own
1534 * the directory, or the file in it, in order to be allowed to
1535 * delete it (unless the user is root). This implements
1536 * append-only directories
1537 */
1538 if ((dsp->s_mode & S_ISVTX) &&
1539 (cred->cr_uid != 0) &&
1540 (cred->cr_uid != dsp->s_uid) &&
1541 (target_vp != NULL) &&
1542 (target_vp->v_type != VLNK) &&
1543 (VTOS(target_vp)->s_uid != cred->cr_uid)) {
1544 vput(target_vp);
1545 result = EPERM;
1546 goto Err_Exit;
1547 };
1548 };
1549
1550 if ((nameiop == RENAME) && (flags & WANTPARENT) && (flags * ISLASTCN)) {
1551 result = VOP_ACCESS(dp, VWRITE, cred, p);
1552 if (result != 0) goto Err_Exit;
1553
1554 if (isDot) {
1555 vrele(target_vp);
1556 result = EISDIR;
1557 goto Err_Exit;
1558 };
1559 };
1560 } else {
1561 /* The specified entry wasn't found: */
1562 result = ENOENT;
1563
1564 if ((flags & ISLASTCN) &&
1565 ((nameiop == CREATE) ||
1566 (nameiop == RENAME) ||
1567 ((nameiop == DELETE) && (flags & DOWHITEOUT) && (flags & ISWHITEOUT)))) {
1568 /* Write access is required to create entries in the directory: */
1569 result = VOP_ACCESS(dp, VWRITE, cred, p);
1570 if (result != 0) goto Err_Exit;
1571
1572 cnp->cn_flags |= SAVENAME;
1573
1574 result = EJUSTRETURN;
1575 }
1576 };
1577
1578 /* XXX PPD Should we do something special in case LOCKLEAF isn't set? */
1579 if (found && !isDot && !isDotDot && (!(flags & LOCKPARENT) || !(flags & ISLASTCN))) {
1580 VOP_UNLOCK(dp, 0, p);
1581 };
1582
1583 *ap->a_vpp = target_vp;
1584
1585 Err_Exit:;
1586 DBG_VOP(("synthfs_lookup: result = %d.\n", result));
1587 if (found) {
1588 if (target_vp) {
1589 if (VOP_ISLOCKED(target_vp)) {
1590 DBG_VOP(("synthfs_lookup: target_vp = 0x%08X (locked).\n", (u_long)target_vp));
1591 } else {
1592 DBG_VOP(("synthfs_lookup: target_vp = 0x%08X (NOT locked?).\n", (u_long)target_vp));
1593 };
1594 } else {
1595 DBG_VOP(("synthfs_lookup: found = true but target_vp = NULL?\n"));
1596 };
1597 } else {
1598 DBG_VOP(("synthf_lookup: target not found.\n"));
1599 };
1600 if (VOP_ISLOCKED(starting_parent)) {
1601 DBG_VOP(("synthfs_lookup: dp = %08X; starting_parent = 0x%08X (LOCKED).\n", (u_long)dp, (u_long)starting_parent));
1602 } else {
1603 DBG_VOP(("synthfs_lookup: dp = %08X; starting_parent = 0x%08X (UNLOCKED).\n", (u_long)dp, (u_long)starting_parent));
1604 };
1605
1606 return result;
1607 }
1608
1609
1610
1611 /*
1612
1613 #% pathconf vp L L L
1614 #
1615 vop_pathconf {
1616 IN struct vnode *vp;
1617 IN int name;
1618 OUT register_t *retval;
1619 */
1620 int
1621 synthfs_pathconf(ap)
1622 struct vop_pathconf_args /* {
1623 struct vnode *a_vp;
1624 int a_name;
1625 int *a_retval;
1626 } */ *ap;
1627 {
1628 DBG_VOP(("synthfs_pathconf called\n"));
1629
1630 switch (ap->a_name)
1631 {
1632 case _PC_LINK_MAX:
1633 *ap->a_retval = LINK_MAX;
1634 return (0);
1635 case _PC_NAME_MAX:
1636 *ap->a_retval = NAME_MAX;
1637 return (0);
1638 case _PC_PATH_MAX:
1639 *ap->a_retval = PATH_MAX;
1640 return (0);
1641 case _PC_PIPE_BUF:
1642 *ap->a_retval = PIPE_BUF;
1643 return (0);
1644 case _PC_CHOWN_RESTRICTED:
1645 *ap->a_retval = 1;
1646 return (0);
1647 case _PC_NO_TRUNC:
1648 *ap->a_retval = 1;
1649 return (0);
1650 default:
1651 return (EINVAL);
1652 }
1653 /* NOTREACHED */
1654 }
1655
1656
1657 /*
1658 * Update the access, modified, and node change times as specified by the
1659 * IACCESS, IUPDATE, and ICHANGE flags respectively. The IMODIFIED flag is
1660 * used to specify that the node needs to be updated but that the times have
1661 * already been set. The access and modified times are taken from the second
1662 * and third parameters; the node change time is always taken from the current
1663 * time. If waitfor is set, then wait for the disk write of the node to
1664 * complete.
1665 */
1666 /*
1667 #% update vp L L L
1668 IN struct vnode *vp;
1669 IN struct timeval *access;
1670 IN struct timeval *modify;
1671 IN int waitfor;
1672 */
1673
1674 int
1675 synthfs_update(ap)
1676 struct vop_update_args /* {
1677 struct vnode *a_vp;
1678 struct timeval *a_access;
1679 struct timeval *a_modify;
1680 int a_waitfor;
1681 } */ *ap;
1682 {
1683 struct vnode *vp = ap->a_vp;
1684 struct synthfsnode *sp = VTOS(vp);
1685
1686 DBG_ASSERT(sp != NULL);
1687 DBG_ASSERT(*((int*)&vp->v_interlock) == 0);
1688
1689 if (((sp->s_nodeflags & (IN_ACCESS | IN_CHANGE | IN_MODIFIED | IN_UPDATE)) != 0) &&
1690 !(VTOVFS(ap->a_vp)->mnt_flag & MNT_RDONLY)) {
1691 if (sp->s_nodeflags & IN_ACCESS) sp->s_accesstime = *ap->a_access;
1692 if (sp->s_nodeflags & IN_UPDATE) sp->s_modificationtime = *ap->a_modify;
1693 if (sp->s_nodeflags & IN_CHANGE) sp->s_changetime = time;
1694 };
1695
1696 /* After the updates are finished, clear the flags */
1697 sp->s_nodeflags &= ~(IN_ACCESS | IN_CHANGE | IN_MODIFIED | IN_UPDATE);
1698
1699 // DBG_ASSERT(*((int*)&ap->a_vp->v_interlock) == 0);
1700 return 0;
1701 }
1702
1703
1704
1705 /*******************************************************************************************
1706
1707 Utility/housekeeping vnode operations:
1708
1709 ******************************************************************************************/
1710
1711
1712 /*
1713 #% lock vp U L U
1714 #
1715 vop_lock {
1716 IN struct vnode *vp;
1717 IN int flags;
1718 IN struct proc *p;
1719 */
1720
1721 int
1722 synthfs_lock(ap)
1723 struct vop_lock_args /* {
1724 struct vnode *a_vp;
1725 int a_flags;
1726 struct proc *a_p;
1727 } */ *ap;
1728 {
1729 return lockmgr(&VTOS(ap->a_vp)->s_lock, ap->a_flags, &ap->a_vp->v_interlock, ap->a_p);
1730 }
1731
1732 /*
1733 * Unlock an synthfsnode.
1734 #% unlock vp L U L
1735 #
1736 vop_unlock {
1737 IN struct vnode *vp;
1738 IN int flags;
1739 IN struct proc *p;
1740
1741 */
1742 int
1743 synthfs_unlock(ap)
1744 struct vop_unlock_args /* {
1745 struct vnode *a_vp;
1746 int a_flags;
1747 struct proc *a_p;
1748 } */ *ap;
1749 {
1750 return lockmgr(&VTOS(ap->a_vp)->s_lock, ap->a_flags | LK_RELEASE, &ap->a_vp->v_interlock, ap->a_p);
1751 }
1752
1753 /*
1754 * Check for a locked synthfsnode.
1755 #% islocked vp = = =
1756 #
1757 vop_islocked {
1758 IN struct vnode *vp;
1759
1760 */
1761 int
1762 synthfs_islocked(ap)
1763 struct vop_islocked_args /* {
1764 struct vnode *a_vp;
1765 } */ *ap;
1766 {
1767 return lockstatus(&VTOS(ap->a_vp)->s_lock);
1768 }
1769
1770
1771
1772 /*
1773 #
1774 #% inactive vp L U U
1775 #
1776 vop_inactive {
1777 IN struct vnode *vp;
1778 IN struct proc *p;
1779
1780 */
1781
1782 int
1783 synthfs_inactive(ap)
1784 struct vop_inactive_args /* {
1785 struct vnode *a_vp;
1786 struct proc *a_p;
1787 } */ *ap;
1788 {
1789 struct vnode *vp = ap->a_vp;
1790 struct proc *p = ap->a_p;
1791 struct synthfsnode *sp = VTOS(vp);
1792 struct timeval tv;
1793
1794 if (vp->v_usecount != 0)
1795 DBG_VOP(("synthfs_inactive: bad usecount = %d\n", vp->v_usecount ));
1796
1797 /*
1798 * Ignore nodes related to stale file handles.
1799 */
1800 if (vp->v_type == VNON)
1801 goto out;
1802
1803 /* This is sort of silly but might make sense in the future: */
1804 if (sp->s_nodeflags & (IN_ACCESS | IN_CHANGE | IN_MODIFIED | IN_UPDATE)) {
1805 tv = time;
1806 VOP_UPDATE(vp, &tv, &tv, 0);
1807 }
1808
1809 out:
1810 VOP_UNLOCK(vp, 0, p);
1811 /*
1812 * If we are done with the inode, reclaim it
1813 * so that it can be reused immediately.
1814 */
1815 if (vp->v_type == VNON) {
1816 vrecycle(vp, (struct slock *)0, p);
1817 };
1818
1819 return 0;
1820 }
1821
1822
1823
1824 /*
1825 * synthfs_reclaim - Reclaim a vnode so that it can be used for other purposes.
1826 *
1827 * Locking policy: ignored
1828 */
1829 int
1830 synthfs_reclaim(ap)
1831 struct vop_reclaim_args /* { struct vnode *a_vp; struct proc *a_p; } */ *ap;
1832 {
1833 struct vnode *vp = ap->a_vp;
1834 struct synthfsnode *sp = VTOS(vp);
1835 void *name = sp->s_name;
1836
1837 sp->s_name = NULL;
1838 FREE(name, M_TEMP);
1839
1840 vp->v_data = NULL;
1841 FREE((void *)sp, M_SYNTHFS);
1842
1843 return (0);
1844 }