blob: d2e1a8270d67976e52d705eb3f990673469c11e3 [file] [log] [blame]
/* { dg-do compile } */
/* { dg-require-effective-target arm_hard_vfp_ok } */
/* { dg-options "-O2 -mapcs -march=armv7-a -mfloat-abi=hard -mfpu=vfpv3-d16" } */
int inode_permission (), try_break_deleg ();
int mutex_lock (), mutex_unlock ();
struct mutex
{
};
struct dentry
{
struct inode *d_inode;
};
struct inode
{
const struct inode_operations *i_op;
struct super_block *i_sb;
union
{
const unsigned int i_nlink;
};
unsigned long i_state;
struct mutex i_mutex;
};
struct super_block
{
unsigned int s_max_links;
};
struct inode_operations
{
int (*link) (struct dentry *, struct inode *, struct dentry *);
} __attribute__ ((__aligned__ ((1 << 6))));
static inline __attribute__ ((always_inline))
__attribute__ ((no_instrument_function))
int may_create (struct inode *dir, struct dentry *child)
{
if (child->d_inode)
return -17;
return inode_permission (dir, 0x00000002 | 0x00000001);
}
int
vfs_link (struct dentry *old_dentry, struct inode *dir,
struct dentry *new_dentry, struct inode **delegated_inode)
{
struct inode *inode = old_dentry->d_inode;
unsigned max_links = dir->i_sb->s_max_links;
int error;
error = may_create (dir, new_dentry);
if (error)
return error;
mutex_lock (&inode->i_mutex);
if (inode->i_nlink == 0 && !(inode->i_state & (1 << 10)))
error = -2;
else if (max_links && inode->i_nlink >= max_links)
error = -31;
else
{
error = try_break_deleg (inode, delegated_inode);
error = dir->i_op->link (old_dentry, dir, new_dentry);
}
mutex_unlock (&inode->i_mutex);
return error;
}