158 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			158 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* This file contains file metadata retrieval and manipulation routines.
 | |
|  *
 | |
|  * The entry points into this file are:
 | |
|  *   get_mode		return a file's mode
 | |
|  *   do_stat		perform the STAT file system call
 | |
|  *   do_chmod		perform the CHMOD file system call
 | |
|  *   do_utime		perform the UTIME file system call
 | |
|  *
 | |
|  * Created:
 | |
|  *   April 2009 (D.C. van Moolenbroek)
 | |
|  */
 | |
| 
 | |
| #include "inc.h"
 | |
| 
 | |
| /*===========================================================================*
 | |
|  *				get_mode				     *
 | |
|  *===========================================================================*/
 | |
| PUBLIC mode_t get_mode(ino, mode)
 | |
| struct inode *ino;
 | |
| int mode;
 | |
| {
 | |
| /* Return the mode for an inode, given the inode and the HGFS retrieved mode.
 | |
|  */
 | |
| 
 | |
|   mode &= S_IRWXU;
 | |
|   mode = mode | (mode >> 3) | (mode >> 6);
 | |
| 
 | |
|   if (IS_DIR(ino))
 | |
| 	mode = S_IFDIR | (mode & opt.dir_mask);
 | |
|   else
 | |
| 	mode = S_IFREG | (mode & opt.file_mask);
 | |
| 
 | |
|   if (state.read_only)
 | |
| 	mode &= ~0222;
 | |
| 
 | |
|   return mode;
 | |
| }
 | |
| 
 | |
| /*===========================================================================*
 | |
|  *				do_stat					     *
 | |
|  *===========================================================================*/
 | |
| PUBLIC int do_stat()
 | |
| {
 | |
| /* Retrieve inode status.
 | |
|  */
 | |
|   struct inode *ino;
 | |
|   struct hgfs_attr attr;
 | |
|   struct stat stat;
 | |
|   char path[PATH_MAX];
 | |
|   ino_t ino_nr;
 | |
|   int r;
 | |
| 
 | |
|   ino_nr = m_in.REQ_INODE_NR;
 | |
| 
 | |
|   /* Don't increase the inode refcount: it's already open anyway */
 | |
|   if ((ino = find_inode(ino_nr)) == NULL)
 | |
| 	return EINVAL;
 | |
| 
 | |
|   attr.a_mask = HGFS_ATTR_MODE | HGFS_ATTR_SIZE | HGFS_ATTR_ATIME |
 | |
| 		HGFS_ATTR_MTIME | HGFS_ATTR_CTIME;
 | |
| 
 | |
|   if ((r = verify_inode(ino, path, &attr)) != OK)
 | |
| 	return r;
 | |
| 
 | |
|   stat.st_dev = state.dev;
 | |
|   stat.st_ino = ino_nr;
 | |
|   stat.st_mode = get_mode(ino, attr.a_mode);
 | |
|   stat.st_uid = opt.uid;
 | |
|   stat.st_gid = opt.gid;
 | |
|   stat.st_rdev = NO_DEV;
 | |
|   if (cmp64u(attr.a_size, LONG_MAX) > 0)
 | |
| 	stat.st_size = LONG_MAX;
 | |
|   else
 | |
| 	stat.st_size = ex64lo(attr.a_size);
 | |
|   stat.st_atime = attr.a_atime;
 | |
|   stat.st_mtime = attr.a_mtime;
 | |
|   stat.st_ctime = attr.a_ctime;
 | |
| 
 | |
|   /* We could make this more accurate by iterating over directory inodes'
 | |
|    * children, counting how many of those are directories as well.
 | |
|    * It's just not worth it.
 | |
|    */
 | |
|   stat.st_nlink = 0;
 | |
|   if (ino->i_parent != NULL) stat.st_nlink++;
 | |
|   if (IS_DIR(ino)) {
 | |
| 	stat.st_nlink++;
 | |
| 	if (HAS_CHILDREN(ino)) stat.st_nlink++;
 | |
|   }
 | |
| 
 | |
|   return sys_safecopyto(m_in.m_source, m_in.REQ_GRANT, 0,
 | |
| 	(vir_bytes) &stat, sizeof(stat), D);
 | |
| }
 | |
| 
 | |
| /*===========================================================================*
 | |
|  *				do_chmod				     *
 | |
|  *===========================================================================*/
 | |
| PUBLIC int do_chmod()
 | |
| {
 | |
| /* Change file mode.
 | |
|  */
 | |
|   struct inode *ino;
 | |
|   char path[PATH_MAX];
 | |
|   struct hgfs_attr attr;
 | |
|   int r;
 | |
| 
 | |
|   if (state.read_only)
 | |
| 	return EROFS;
 | |
| 
 | |
|   if ((ino = find_inode(m_in.REQ_INODE_NR)) == NULL)
 | |
| 	return EINVAL;
 | |
| 
 | |
|   if ((r = verify_inode(ino, path, NULL)) != OK)
 | |
| 	return r;
 | |
| 
 | |
|   /* Set the new file mode. */
 | |
|   attr.a_mask = HGFS_ATTR_MODE;
 | |
|   attr.a_mode = m_in.REQ_MODE; /* no need to convert in this direction */
 | |
| 
 | |
|   if ((r = hgfs_setattr(path, &attr)) != OK)
 | |
| 	return r;
 | |
| 
 | |
|   /* We have no idea what really happened. Query for the mode again. */
 | |
|   if ((r = verify_path(path, ino, &attr, NULL)) != OK)
 | |
| 	return r;
 | |
| 
 | |
|   m_out.RES_MODE = get_mode(ino, attr.a_mode);
 | |
| 
 | |
|   return OK;
 | |
| }
 | |
| 
 | |
| /*===========================================================================*
 | |
|  *				do_utime				     *
 | |
|  *===========================================================================*/
 | |
| PUBLIC int do_utime()
 | |
| {
 | |
| /* Set file times.
 | |
|  */
 | |
|   struct inode *ino;
 | |
|   char path[PATH_MAX];
 | |
|   struct hgfs_attr attr;
 | |
|   int r;
 | |
| 
 | |
|   if (state.read_only)
 | |
| 	return EROFS;
 | |
| 
 | |
|   if ((ino = find_inode(m_in.REQ_INODE_NR)) == NULL)
 | |
| 	return EINVAL;
 | |
| 
 | |
|   if ((r = verify_inode(ino, path, NULL)) != OK)
 | |
| 	return r;
 | |
| 
 | |
|   attr.a_mask = HGFS_ATTR_ATIME | HGFS_ATTR_MTIME;
 | |
|   attr.a_atime = m_in.REQ_ACTIME;
 | |
|   attr.a_mtime = m_in.REQ_MODTIME;
 | |
| 
 | |
|   return hgfs_setattr(path, &attr);
 | |
| }
 | 
