2010-01-26 00:18:02 +01:00
|
|
|
/* 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 *
|
|
|
|
*===========================================================================*/
|
2012-03-25 20:25:53 +02:00
|
|
|
mode_t get_mode(ino, mode)
|
2010-01-26 00:18:02 +01:00
|
|
|
struct inode *ino;
|
|
|
|
int mode;
|
|
|
|
{
|
2012-04-09 17:17:42 +02:00
|
|
|
/* Return the mode for an inode, given the inode and the retrieved mode.
|
2010-01-26 00:18:02 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
mode &= S_IRWXU;
|
|
|
|
mode = mode | (mode >> 3) | (mode >> 6);
|
|
|
|
|
|
|
|
if (IS_DIR(ino))
|
2012-04-09 18:08:26 +02:00
|
|
|
mode = S_IFDIR | (mode & sffs_params->p_dir_mask);
|
2010-01-26 00:18:02 +01:00
|
|
|
else
|
2012-04-09 18:08:26 +02:00
|
|
|
mode = S_IFREG | (mode & sffs_params->p_file_mask);
|
2010-01-26 00:18:02 +01:00
|
|
|
|
2012-04-09 18:08:26 +02:00
|
|
|
if (state.s_read_only)
|
2012-04-09 17:17:42 +02:00
|
|
|
mode &= ~(S_IWUSR | S_IWGRP | S_IWOTH);
|
2010-01-26 00:18:02 +01:00
|
|
|
|
|
|
|
return mode;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* do_stat *
|
|
|
|
*===========================================================================*/
|
2012-03-25 20:25:53 +02:00
|
|
|
int do_stat()
|
2010-01-26 00:18:02 +01:00
|
|
|
{
|
2010-08-07 13:50:15 +02:00
|
|
|
/* Retrieve inode status.
|
2010-01-26 00:18:02 +01:00
|
|
|
*/
|
|
|
|
struct inode *ino;
|
2012-04-09 18:08:26 +02:00
|
|
|
struct sffs_attr attr;
|
2010-01-26 00:18:02 +01:00
|
|
|
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 */
|
2010-05-10 15:26:00 +02:00
|
|
|
if ((ino = find_inode(ino_nr)) == NULL)
|
2010-01-26 00:18:02 +01:00
|
|
|
return EINVAL;
|
|
|
|
|
2012-04-09 18:08:26 +02:00
|
|
|
attr.a_mask = SFFS_ATTR_MODE | SFFS_ATTR_SIZE | SFFS_ATTR_CRTIME |
|
|
|
|
SFFS_ATTR_ATIME | SFFS_ATTR_MTIME | SFFS_ATTR_CTIME;
|
2010-01-26 00:18:02 +01:00
|
|
|
|
|
|
|
if ((r = verify_inode(ino, path, &attr)) != OK)
|
|
|
|
return r;
|
|
|
|
|
2011-07-01 21:35:54 +02:00
|
|
|
memset(&stat, 0, sizeof(struct stat));
|
|
|
|
|
2012-04-09 18:08:26 +02:00
|
|
|
stat.st_dev = state.s_dev;
|
2010-01-26 00:18:02 +01:00
|
|
|
stat.st_ino = ino_nr;
|
|
|
|
stat.st_mode = get_mode(ino, attr.a_mode);
|
2012-04-09 18:08:26 +02:00
|
|
|
stat.st_uid = sffs_params->p_uid;
|
|
|
|
stat.st_gid = sffs_params->p_gid;
|
2010-01-26 00:18:02 +01:00
|
|
|
stat.st_rdev = NO_DEV;
|
2010-09-27 15:19:25 +02:00
|
|
|
if (cmp64u(attr.a_size, LONG_MAX) > 0)
|
|
|
|
stat.st_size = LONG_MAX;
|
|
|
|
else
|
|
|
|
stat.st_size = ex64lo(attr.a_size);
|
2012-03-30 01:39:12 +02:00
|
|
|
stat.st_atimespec = attr.a_atime;
|
|
|
|
stat.st_mtimespec = attr.a_mtime;
|
|
|
|
stat.st_ctimespec = attr.a_ctime;
|
|
|
|
stat.st_birthtimespec = attr.a_crtime;
|
2010-01-26 00:18:02 +01:00
|
|
|
|
2011-07-01 21:35:54 +02:00
|
|
|
stat.st_blocks = stat.st_size / S_BLKSIZE;
|
|
|
|
if (stat.st_size % S_BLKSIZE != 0)
|
|
|
|
stat.st_blocks += 1;
|
|
|
|
|
|
|
|
stat.st_blksize = BLOCK_SIZE;
|
|
|
|
|
2010-01-26 00:18:02 +01:00
|
|
|
/* 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;
|
2010-05-10 15:26:00 +02:00
|
|
|
if (ino->i_parent != NULL) stat.st_nlink++;
|
2010-01-26 00:18:02 +01:00
|
|
|
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,
|
2012-06-16 03:46:15 +02:00
|
|
|
(vir_bytes) &stat, sizeof(stat));
|
2010-01-26 00:18:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* do_chmod *
|
|
|
|
*===========================================================================*/
|
2012-03-25 20:25:53 +02:00
|
|
|
int do_chmod()
|
2010-01-26 00:18:02 +01:00
|
|
|
{
|
|
|
|
/* Change file mode.
|
|
|
|
*/
|
|
|
|
struct inode *ino;
|
|
|
|
char path[PATH_MAX];
|
2012-04-09 18:08:26 +02:00
|
|
|
struct sffs_attr attr;
|
2010-01-26 00:18:02 +01:00
|
|
|
int r;
|
|
|
|
|
2012-04-09 18:08:26 +02:00
|
|
|
if (state.s_read_only)
|
2010-01-26 00:18:02 +01:00
|
|
|
return EROFS;
|
|
|
|
|
2010-05-10 15:26:00 +02:00
|
|
|
if ((ino = find_inode(m_in.REQ_INODE_NR)) == NULL)
|
2010-01-26 00:18:02 +01:00
|
|
|
return EINVAL;
|
|
|
|
|
|
|
|
if ((r = verify_inode(ino, path, NULL)) != OK)
|
|
|
|
return r;
|
|
|
|
|
|
|
|
/* Set the new file mode. */
|
2012-04-09 18:08:26 +02:00
|
|
|
attr.a_mask = SFFS_ATTR_MODE;
|
2010-01-26 00:18:02 +01:00
|
|
|
attr.a_mode = m_in.REQ_MODE; /* no need to convert in this direction */
|
|
|
|
|
2012-04-09 18:08:26 +02:00
|
|
|
if ((r = sffs_table->t_setattr(path, &attr)) != OK)
|
2010-01-26 00:18:02 +01:00
|
|
|
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 *
|
|
|
|
*===========================================================================*/
|
2012-03-25 20:25:53 +02:00
|
|
|
int do_utime()
|
2010-01-26 00:18:02 +01:00
|
|
|
{
|
|
|
|
/* Set file times.
|
|
|
|
*/
|
|
|
|
struct inode *ino;
|
|
|
|
char path[PATH_MAX];
|
2012-04-09 18:08:26 +02:00
|
|
|
struct sffs_attr attr;
|
2010-01-26 00:18:02 +01:00
|
|
|
int r;
|
|
|
|
|
2012-04-09 18:08:26 +02:00
|
|
|
if (state.s_read_only)
|
2010-01-26 00:18:02 +01:00
|
|
|
return EROFS;
|
|
|
|
|
2010-05-10 15:26:00 +02:00
|
|
|
if ((ino = find_inode(m_in.REQ_INODE_NR)) == NULL)
|
2010-01-26 00:18:02 +01:00
|
|
|
return EINVAL;
|
|
|
|
|
|
|
|
if ((r = verify_inode(ino, path, NULL)) != OK)
|
|
|
|
return r;
|
|
|
|
|
2011-12-21 23:29:29 +01:00
|
|
|
attr.a_mask = 0;
|
|
|
|
|
|
|
|
switch(m_in.REQ_ACNSEC) {
|
|
|
|
case UTIME_OMIT: /* do not touch */
|
|
|
|
break;
|
|
|
|
case UTIME_NOW:
|
|
|
|
/* XXX VFS should have time() into ACTIME, for compat; we trust it! */
|
|
|
|
m_in.REQ_ACNSEC = 0;
|
|
|
|
/*FALLTHROUGH*/
|
|
|
|
default:
|
|
|
|
/* cases m_in.REQ_ACNSEC < 0 || m_in.REQ_ACNSEC >= 1E9
|
|
|
|
* are caught by VFS to cooperate with old instances of EXT2
|
|
|
|
*/
|
|
|
|
attr.a_atime.tv_sec = m_in.REQ_ACTIME;
|
|
|
|
attr.a_atime.tv_nsec = m_in.REQ_ACNSEC;
|
|
|
|
attr.a_mask |= SFFS_ATTR_ATIME;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
switch(m_in.REQ_MODNSEC) {
|
|
|
|
case UTIME_OMIT: /* do not touch */
|
|
|
|
break;
|
|
|
|
case UTIME_NOW:
|
|
|
|
/* XXX VFS should have time() into MODTIME, for compat; we trust it! */
|
|
|
|
m_in.REQ_MODNSEC = 0;
|
|
|
|
/*FALLTHROUGH*/
|
|
|
|
default:
|
|
|
|
/* cases m_in.REQ_MODNSEC < 0 || m_in.REQ_MODNSEC >= 1E9
|
|
|
|
* are caught by VFS to cooperate with old instances
|
|
|
|
*/
|
|
|
|
attr.a_mtime.tv_sec = m_in.REQ_MODTIME;
|
|
|
|
attr.a_mtime.tv_nsec = m_in.REQ_MODNSEC;
|
|
|
|
attr.a_mask |= SFFS_ATTR_MTIME;
|
|
|
|
break;
|
|
|
|
}
|
2012-04-09 18:08:26 +02:00
|
|
|
return sffs_table->t_setattr(path, &attr);
|
2010-01-26 00:18:02 +01:00
|
|
|
}
|