|  | /* | 
|  | * Copyright (C) Sistina Software, Inc.  1997-2003 All rights reserved. | 
|  | * Copyright (C) 2004-2006 Red Hat, Inc.  All rights reserved. | 
|  | * | 
|  | * This copyrighted material is made available to anyone wishing to use, | 
|  | * modify, copy, or redistribute it subject to the terms and conditions | 
|  | * of the GNU General Public License version 2. | 
|  | */ | 
|  |  | 
|  | #include <linux/slab.h> | 
|  | #include <linux/spinlock.h> | 
|  | #include <linux/completion.h> | 
|  | #include <linux/buffer_head.h> | 
|  | #include <linux/capability.h> | 
|  | #include <linux/xattr.h> | 
|  | #include <linux/gfs2_ondisk.h> | 
|  | #include <linux/lm_interface.h> | 
|  | #include <asm/uaccess.h> | 
|  |  | 
|  | #include "gfs2.h" | 
|  | #include "incore.h" | 
|  | #include "acl.h" | 
|  | #include "eaops.h" | 
|  | #include "eattr.h" | 
|  | #include "util.h" | 
|  |  | 
|  | /** | 
|  | * gfs2_ea_name2type - get the type of the ea, and truncate type from the name | 
|  | * @namep: ea name, possibly with type appended | 
|  | * | 
|  | * Returns: GFS2_EATYPE_XXX | 
|  | */ | 
|  |  | 
|  | unsigned int gfs2_ea_name2type(const char *name, const char **truncated_name) | 
|  | { | 
|  | unsigned int type; | 
|  |  | 
|  | if (strncmp(name, "system.", 7) == 0) { | 
|  | type = GFS2_EATYPE_SYS; | 
|  | if (truncated_name) | 
|  | *truncated_name = name + sizeof("system.") - 1; | 
|  | } else if (strncmp(name, "user.", 5) == 0) { | 
|  | type = GFS2_EATYPE_USR; | 
|  | if (truncated_name) | 
|  | *truncated_name = name + sizeof("user.") - 1; | 
|  | } else if (strncmp(name, "security.", 9) == 0) { | 
|  | type = GFS2_EATYPE_SECURITY; | 
|  | if (truncated_name) | 
|  | *truncated_name = name + sizeof("security.") - 1; | 
|  | } else { | 
|  | type = GFS2_EATYPE_UNUSED; | 
|  | if (truncated_name) | 
|  | *truncated_name = NULL; | 
|  | } | 
|  |  | 
|  | return type; | 
|  | } | 
|  |  | 
|  | static int system_eo_get(struct gfs2_inode *ip, struct gfs2_ea_request *er) | 
|  | { | 
|  | if (!GFS2_ACL_IS_ACCESS(er->er_name, er->er_name_len) && | 
|  | !GFS2_ACL_IS_DEFAULT(er->er_name, er->er_name_len) && | 
|  | !capable(CAP_SYS_ADMIN)) | 
|  | return -EPERM; | 
|  |  | 
|  | if (GFS2_SB(&ip->i_inode)->sd_args.ar_posix_acl == 0 && | 
|  | (GFS2_ACL_IS_ACCESS(er->er_name, er->er_name_len) || | 
|  | GFS2_ACL_IS_DEFAULT(er->er_name, er->er_name_len))) | 
|  | return -EOPNOTSUPP; | 
|  |  | 
|  | return gfs2_ea_get_i(ip, er); | 
|  | } | 
|  |  | 
|  | static int system_eo_set(struct gfs2_inode *ip, struct gfs2_ea_request *er) | 
|  | { | 
|  | int remove = 0; | 
|  | int error; | 
|  |  | 
|  | if (GFS2_ACL_IS_ACCESS(er->er_name, er->er_name_len)) { | 
|  | if (!(er->er_flags & GFS2_ERF_MODE)) { | 
|  | er->er_mode = ip->i_inode.i_mode; | 
|  | er->er_flags |= GFS2_ERF_MODE; | 
|  | } | 
|  | error = gfs2_acl_validate_set(ip, 1, er, | 
|  | &remove, &er->er_mode); | 
|  | if (error) | 
|  | return error; | 
|  | error = gfs2_ea_set_i(ip, er); | 
|  | if (error) | 
|  | return error; | 
|  | if (remove) | 
|  | gfs2_ea_remove_i(ip, er); | 
|  | return 0; | 
|  |  | 
|  | } else if (GFS2_ACL_IS_DEFAULT(er->er_name, er->er_name_len)) { | 
|  | error = gfs2_acl_validate_set(ip, 0, er, | 
|  | &remove, NULL); | 
|  | if (error) | 
|  | return error; | 
|  | if (!remove) | 
|  | error = gfs2_ea_set_i(ip, er); | 
|  | else { | 
|  | error = gfs2_ea_remove_i(ip, er); | 
|  | if (error == -ENODATA) | 
|  | error = 0; | 
|  | } | 
|  | return error; | 
|  | } | 
|  |  | 
|  | return -EPERM; | 
|  | } | 
|  |  | 
|  | static int system_eo_remove(struct gfs2_inode *ip, struct gfs2_ea_request *er) | 
|  | { | 
|  | if (GFS2_ACL_IS_ACCESS(er->er_name, er->er_name_len)) { | 
|  | int error = gfs2_acl_validate_remove(ip, 1); | 
|  | if (error) | 
|  | return error; | 
|  |  | 
|  | } else if (GFS2_ACL_IS_DEFAULT(er->er_name, er->er_name_len)) { | 
|  | int error = gfs2_acl_validate_remove(ip, 0); | 
|  | if (error) | 
|  | return error; | 
|  |  | 
|  | } else | 
|  | return -EPERM; | 
|  |  | 
|  | return gfs2_ea_remove_i(ip, er); | 
|  | } | 
|  |  | 
|  | static const struct gfs2_eattr_operations gfs2_user_eaops = { | 
|  | .eo_get = gfs2_ea_get_i, | 
|  | .eo_set = gfs2_ea_set_i, | 
|  | .eo_remove = gfs2_ea_remove_i, | 
|  | .eo_name = "user", | 
|  | }; | 
|  |  | 
|  | const struct gfs2_eattr_operations gfs2_system_eaops = { | 
|  | .eo_get = system_eo_get, | 
|  | .eo_set = system_eo_set, | 
|  | .eo_remove = system_eo_remove, | 
|  | .eo_name = "system", | 
|  | }; | 
|  |  | 
|  | static const struct gfs2_eattr_operations gfs2_security_eaops = { | 
|  | .eo_get = gfs2_ea_get_i, | 
|  | .eo_set = gfs2_ea_set_i, | 
|  | .eo_remove = gfs2_ea_remove_i, | 
|  | .eo_name = "security", | 
|  | }; | 
|  |  | 
|  | const struct gfs2_eattr_operations *gfs2_ea_ops[] = { | 
|  | NULL, | 
|  | &gfs2_user_eaops, | 
|  | &gfs2_system_eaops, | 
|  | &gfs2_security_eaops, | 
|  | }; | 
|  |  |