156 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			156 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* This file takes care of those system calls that deal with time.
 | |
|  *
 | |
|  * The entry points into this file are
 | |
|  *   do_utimens:	perform the UTIMENS system call
 | |
|  */
 | |
| 
 | |
| #include "fs.h"
 | |
| #include <minix/callnr.h>
 | |
| #include <minix/com.h>
 | |
| #include <time.h>
 | |
| #include <string.h>
 | |
| #include <sys/stat.h>
 | |
| #include <fcntl.h>
 | |
| #include "file.h"
 | |
| #include "path.h"
 | |
| #include "vnode.h"
 | |
| #include <minix/vfsif.h>
 | |
| #include "vmnt.h"
 | |
| 
 | |
| #define	UTIMENS_STYLE	0	/* utimes(2)/utimensat(2) style, named file */
 | |
| #define	FUTIMENS_STYLE	1	/* futimens(2)/futimes(2) style, file desc. */
 | |
| 
 | |
| /*===========================================================================*
 | |
|  *				do_utimens				     *
 | |
|  *===========================================================================*/
 | |
| int do_utimens(void)
 | |
| {
 | |
| /* Perform the utimens(name, times, flag) system call, and its friends.
 | |
|  * Implement a very large but not complete subset of the utimensat()
 | |
|  * Posix:2008/XOpen-7 function.
 | |
|  * Are handled all the following cases:
 | |
|  * . utimensat(AT_FDCWD, "/some/absolute/path", , )
 | |
|  * . utimensat(AT_FDCWD, "some/path", , )
 | |
|  * . utimens("anything", ) really special case of the above two
 | |
|  * . lutimens("anything", ) also really special case of the above
 | |
|  * . utimensat(fd, "/some/absolute/path", , ) although fd is useless here
 | |
|  * . futimens(fd, )
 | |
|  * Are not handled the following cases:
 | |
|  * . utimensat(fd, "some/path", , ) path to a file relative to some open fd
 | |
|  */
 | |
|   int r, kind, lookup_flags;
 | |
|   struct vnode *vp;
 | |
|   struct filp *filp = NULL; /* initialization required by clueless GCC */
 | |
|   struct vmnt *vmp;
 | |
|   struct timespec actim, modtim, now, newactim, newmodtim;
 | |
|   char fullpath[PATH_MAX];
 | |
|   struct lookup resolve;
 | |
|   vir_bytes vname;
 | |
|   size_t vname_length;
 | |
| 
 | |
|   memset(&now, 0, sizeof(now));
 | |
| 
 | |
|   /* The case times==NULL is handled by the caller, replaced with UTIME_NOW */
 | |
|   actim.tv_sec = job_m_in.m_vfs_utimens.atime;
 | |
|   actim.tv_nsec = job_m_in.m_vfs_utimens.ansec;
 | |
|   modtim.tv_sec = job_m_in.m_vfs_utimens.mtime;
 | |
|   modtim.tv_nsec = job_m_in.m_vfs_utimens.mnsec;
 | |
| 
 | |
|   if (job_m_in.m_vfs_utimens.name != NULL) {
 | |
| 	kind = UTIMENS_STYLE;
 | |
| 	if (job_m_in.m_vfs_utimens.flags & ~AT_SYMLINK_NOFOLLOW)
 | |
| 		return EINVAL; /* unknown flag */
 | |
| 	/* Temporarily open the file */
 | |
| 	vname = (vir_bytes) job_m_in.m_vfs_utimens.name;
 | |
| 	vname_length = (size_t) job_m_in.m_vfs_utimens.len;
 | |
| 	if (job_m_in.m_vfs_utimens.flags & AT_SYMLINK_NOFOLLOW)
 | |
| 		lookup_flags = PATH_RET_SYMLINK;
 | |
| 	else
 | |
| 		lookup_flags = PATH_NOFLAGS;
 | |
| 	lookup_init(&resolve, fullpath, lookup_flags, &vmp, &vp);
 | |
| 	resolve.l_vmnt_lock = VMNT_READ;
 | |
| 	resolve.l_vnode_lock = VNODE_READ;
 | |
| 	/* Temporarily open the file */
 | |
| 	if (fetch_name(vname, vname_length, fullpath) != OK) return(err_code);
 | |
| 	if ((vp = eat_path(&resolve, fp)) == NULL) return(err_code);
 | |
|   }
 | |
|   else {
 | |
| 	kind = FUTIMENS_STYLE;
 | |
| 	/* Change timestamps on already-opened fd. Is it valid? */
 | |
| 	if (job_m_in.m_vfs_utimens.flags != 0)
 | |
| 		return EINVAL; /* unknown flag */
 | |
| 	if ((filp = get_filp(job_m_in.m_vfs_utimens.fd, VNODE_READ)) == NULL)
 | |
| 		return err_code;
 | |
| 	vp = filp->filp_vno;
 | |
|   }
 | |
| 
 | |
|   r = OK;
 | |
|   /* Only the owner of a file or the super user can change timestamps. */
 | |
|   if (vp->v_uid != fp->fp_effuid && fp->fp_effuid != SU_UID) r = EPERM;
 | |
|   /* Need write permission (or super user) to 'touch' the file */
 | |
|   if (r != OK && actim.tv_nsec == UTIME_NOW
 | |
|               && modtim.tv_nsec == UTIME_NOW) r = forbidden(fp, vp, W_BIT);
 | |
|   if (read_only(vp) != OK) r = EROFS; /* Not even su can touch if R/O */
 | |
| 
 | |
|   if (r == OK) {
 | |
| 	/* Do we need to ask for current time? */
 | |
| 	if (actim.tv_nsec == UTIME_NOW
 | |
| 	 || actim.tv_nsec == UTIME_OMIT
 | |
| 	 || modtim.tv_nsec == UTIME_NOW
 | |
| 	 || modtim.tv_nsec == UTIME_OMIT) {
 | |
| 		now = clock_timespec();
 | |
| 	}
 | |
| 
 | |
| 	/* Build the request */
 | |
| 	switch (actim.tv_nsec) {
 | |
| 	case UTIME_NOW:
 | |
| 		newactim = now;
 | |
| 		break;
 | |
| 	case UTIME_OMIT:
 | |
| 		newactim.tv_nsec = UTIME_OMIT;
 | |
| 		/* Be nice with old FS, put a sensible value in
 | |
| 		 * otherwise not used field for seconds
 | |
| 		 */
 | |
| 		newactim.tv_sec = now.tv_sec;
 | |
| 		break;
 | |
| 	default:
 | |
| 		if ( (unsigned)actim.tv_nsec >= 1000000000)
 | |
| 			r = EINVAL;
 | |
| 		else
 | |
| 			newactim = actim;
 | |
| 		break;
 | |
| 	}
 | |
| 	switch (modtim.tv_nsec) {
 | |
| 	case UTIME_NOW:
 | |
| 		newmodtim = now;
 | |
| 		break;
 | |
| 	case UTIME_OMIT:
 | |
| 		newmodtim.tv_nsec = UTIME_OMIT;
 | |
| 		/* Be nice with old FS, put a sensible value */
 | |
| 		newmodtim.tv_sec = now.tv_sec;
 | |
| 		break;
 | |
| 	default:
 | |
| 		if ( (unsigned)modtim.tv_nsec >= 1000000000)
 | |
| 			r = EINVAL;
 | |
| 		else
 | |
| 			newmodtim = modtim;
 | |
| 		break;
 | |
| 	}
 | |
|   }
 | |
| 
 | |
|   if (r == OK)
 | |
| 	/* Issue request */
 | |
| 	r = req_utime(vp->v_fs_e, vp->v_inode_nr, &newactim, &newmodtim);
 | |
| 
 | |
|   if (kind == UTIMENS_STYLE) {
 | |
| 	/* Close the temporary */
 | |
| 	unlock_vnode(vp);
 | |
| 	unlock_vmnt(vmp);
 | |
| 	put_vnode(vp);
 | |
|   }
 | |
|   else { /* Change timestamps on opened fd. */
 | |
| 	unlock_filp(filp);
 | |
|   }
 | |
|   return r;
 | |
| }
 | 
