netdev.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
From: Benjamin Thery <benjamin.thery@bull.net>
To: Greg KH <greg@kroah.com>
Cc: Mark Ryden <markryde@gmail.com>,
	"Eric W. Biederman" <ebiederm@xmission.com>,
	Greg Kroah-Hartman <gregkh@suse.de>,
	Andrew Morton <akpm@linux-foundation.org>,
	Tejun Heo <htejun@gmail.com>,
	Daniel Lezcano <dlezcano@fr.ibm.com>,
	linux-kernel@vger.kernel.org, Al Viro <viro@ftp.linux.org.uk>,
	Linux Containers <containers@lists.osdl.org>,
	netdev@vger.kernel.org
Subject: Re: [PATCH 1/8] sysfs: Implement sysfs tagged directory support.
Date: Tue, 02 Sep 2008 16:03:20 +0200	[thread overview]
Message-ID: <48BD47A8.1080909@bull.net> (raw)
In-Reply-To: <dac45060809020654g62afb3c9yd104682b93b43bee@mail.gmail.com>

Mark Ryden wrote:
> Hello,
>  I hope that this patch (from 4.7.08) was not forgetten... I don't see
> it for example in linux-net (I have an up-to-date linux-next git
> tree).

I was about to ask the same thing.

Greg, what is the plan with these remaining patches for sysfs tagged
dirs? Will you have some time to merge them in your tree soon?

I ran my network namespace tests last week with this latest patchset
from Eric applied on top of your tree (which already contains the first
batch of patches). Everything looked good.

Regards,
Benjamin


> 
> Regards,
> Mark
> 
> On Wed, Aug 27, 2008 at 6:18 PM, Benjamin Thery <benjamin.thery@bull.net> wrote:
>> Eric W. Biederman wrote:
>>> The problem.  When implementing a network namespace I need to be able
>>> to have multiple network devices with the same name.  Currently this
>>> is a problem for /sys/class/net/*, /sys/devices/virtual/net/*, and
>>> potentially a few other directories of the form /sys/ ... /net/*.
>>>
>>> What this patch does is to add an additional tag field to the
>>> sysfs dirent structure.  For directories that should show different
>>> contents depending on the context such as /sys/class/net/, and
>>> /sys/devices/virtual/net/ this tag field is used to specify the
>>> context in which those directories should be visible.  Effectively
>>> this is the same as creating multiple distinct directories with
>>> the same name but internally to sysfs the result is nicer.
>>>
>>> I am calling the concept of a single directory that looks like multiple
>>> directories all at the same path in the filesystem tagged directories.
>>>
>>> For the networking namespace the set of directories whose contents I need
>>> to filter with tags can depend on the presence or absence of hotplug
>>> hardware or which modules are currently loaded.  Which means I need
>>> a simple race free way to setup those directories as tagged.
>>>
>>> To achieve a reace free design all tagged directories are created
>>> and managed by sysfs itself.
>>>
>>> Users of this interface:
>>> - define a type in the sysfs_tag_type enumeration.
>>> - call sysfs_register_tag_types with the type and it's operations
>>> - call sysfs_make_tagged_dir with the tag type on directories
>>>  to be managed by this tag type
>>> - sysfs_exit_tag when an individual tag is no longer valid
>>>
>>> - Implement mount_tag() which returns the tag of the calling process
>>>  so we can attach it to a sysfs superblock.
>>> - Implement ktype.sysfs_tag() which returns the tag of a syfs kobject.
>>>
>>> Everything else is left up to sysfs and the driver layer.
>>>
>>> For the network namespace mount_tag and sysfs_tag are essentially
>>> one line functions, and look to remain that.
>>>
>>> Tags are currently represented a const void * pointers as that is
>>> both generic, prevides enough information for equality comparisons,
>>> and is trivial to create for current users, as it is just the
>>> existing namespace pointer.
>>>
>>> The work needed in sysfs is more extensive.  At each directory
>>> or symlink creating I need to check if the directory it is being
>>> created in is a tagged directory and if so generate the appropriate
>>> tag to place on the sysfs_dirent.  Likewise at each symlink or
>>> directory removal I need to check if the sysfs directory it is
>>> being removed from is a tagged directory and if so figure out
>>> which tag goes along with the name I am deleting.
>>>
>>> Currently only directories which hold kobjects, and
>>> symlinks are supported.  There is not enough information
>>> in the current file attribute interfaces to give us anything
>>> to discriminate on which makes it useless, and there are
>>> no potential users which makes it an uninteresting problem
>>> to solve.
>>>
>>> Signed-off-by: Eric W. Biederman <ebiederm@xmission.com>
>>> Signed-off-by: Benjamin Thery <benjamin.thery@bull.net>
>>> ---
>>>  fs/sysfs/bin.c          |    2 +-
>>>  fs/sysfs/dir.c          |  139
>>> ++++++++++++++++++++++++++++++++++++++++++-----
>>>  fs/sysfs/file.c         |   11 +++--
>>>  fs/sysfs/group.c        |    4 +-
>>>  fs/sysfs/inode.c        |    7 ++-
>>>  fs/sysfs/mount.c        |  115 +++++++++++++++++++++++++++++++++++++--
>>>  fs/sysfs/symlink.c      |    2 +-
>>>  fs/sysfs/sysfs.h        |   19 ++++++-
>>>  include/linux/kobject.h |    1 +
>>>  include/linux/sysfs.h   |   31 +++++++++++
>>>  10 files changed, 298 insertions(+), 33 deletions(-)
>>>
>>> diff --git a/fs/sysfs/bin.c b/fs/sysfs/bin.c
>>> index 006fc64..86e1128 100644
>>> --- a/fs/sysfs/bin.c
>>> +++ b/fs/sysfs/bin.c
>>> @@ -252,7 +252,7 @@ int sysfs_create_bin_file(struct kobject * kobj,
>>> struct bin_attribute * attr)
>>>   void sysfs_remove_bin_file(struct kobject * kobj, struct bin_attribute *
>>> attr)
>>>  {
>>> -       sysfs_hash_and_remove(kobj->sd, attr->attr.name);
>>> +       sysfs_hash_and_remove(kobj, kobj->sd, attr->attr.name);
>>>  }
>>>   EXPORT_SYMBOL_GPL(sysfs_create_bin_file);
>>> diff --git a/fs/sysfs/dir.c b/fs/sysfs/dir.c
>>> index 4ffcfd2..dec7586 100644
>>> --- a/fs/sysfs/dir.c
>>> +++ b/fs/sysfs/dir.c
>>> @@ -30,6 +30,30 @@ DEFINE_SPINLOCK(sysfs_assoc_lock);
>>>  static DEFINE_SPINLOCK(sysfs_ino_lock);
>>>  static DEFINE_IDA(sysfs_ino_ida);
>>>  +static const void *sysfs_creation_tag(struct sysfs_dirent *parent_sd,
>>> +                                     struct sysfs_dirent *sd)
>>> +{
>>> +       const void *tag = NULL;
>>> +
>>> +       if (sysfs_tag_type(parent_sd)) {
>>> +               struct kobject *kobj;
>>> +               switch (sysfs_type(sd)) {
>>> +               case SYSFS_DIR:
>>> +                       kobj = sd->s_dir.kobj;
>>> +                       break;
>>> +               case SYSFS_KOBJ_LINK:
>>> +                       kobj = sd->s_symlink.target_sd->s_dir.kobj;
>>> +                       break;
>>> +               default:
>>> +                       BUG();
>>> +               }
>>> +               tag = kobj->ktype->sysfs_tag(kobj);
>>> +               /* NULL tags are reserved for internal use */
>>> +               BUG_ON(tag == NULL);
>>> +       }
>>> +       return tag;
>>> +}
>>> +
>>>  /**
>>>  *     sysfs_link_sibling - link sysfs_dirent into sibling list
>>>  *     @sd: sysfs_dirent of interest
>>> @@ -101,8 +125,19 @@ static void sysfs_unlink_sibling(struct sysfs_dirent
>>> *sd)
>>>  struct dentry *sysfs_get_dentry(struct super_block *sb,
>>>                                struct sysfs_dirent *sd)
>>>  {
>>> -       struct dentry *dentry = dget(sb->s_root);
>>> +       struct dentry *dentry;
>>> +
>>> +       /* Bail if this sd won't show up in this superblock */
>>> +       if (sd->s_parent) {
>>> +               enum sysfs_tag_type type;
>>> +               const void *tag;
>>> +               type = sysfs_tag_type(sd->s_parent);
>>> +               tag = sysfs_info(sb)->tag[type];
>>> +               if (sd->s_tag != tag)
>>> +                       return ERR_PTR(-EXDEV);
>>> +       }
>>>  +       dentry = dget(sb->s_root);
>>>        while (dentry->d_fsdata != sd) {
>>>                struct sysfs_dirent *cur;
>>>                struct dentry *parent;
>>> @@ -421,10 +456,15 @@ void sysfs_addrm_start(struct sysfs_addrm_cxt *acxt,
>>>  */
>>>  int __sysfs_add_one(struct sysfs_addrm_cxt *acxt, struct sysfs_dirent
>>> *sd)
>>>  {
>>> -       if (sysfs_find_dirent(acxt->parent_sd, sd->s_name))
>>> +       const void *tag = NULL;
>>> +
>>> +       tag = sysfs_creation_tag(acxt->parent_sd, sd);
>>> +
>>> +       if (sysfs_find_dirent(acxt->parent_sd, tag, sd->s_name))
>>>                return -EEXIST;
>>>          sd->s_parent = sysfs_get(acxt->parent_sd);
>>> +       sd->s_tag = tag;
>>>          if (sysfs_type(sd) == SYSFS_DIR && acxt->parent_inode)
>>>                inc_nlink(acxt->parent_inode);
>>> @@ -602,13 +642,17 @@ void sysfs_addrm_finish(struct sysfs_addrm_cxt
>>> *acxt)
>>>  *     Pointer to sysfs_dirent if found, NULL if not.
>>>  */
>>>  struct sysfs_dirent *sysfs_find_dirent(struct sysfs_dirent *parent_sd,
>>> +                                      const void *tag,
>>>                                       const unsigned char *name)
>>>  {
>>>        struct sysfs_dirent *sd;
>>>  -       for (sd = parent_sd->s_dir.children; sd; sd = sd->s_sibling)
>>> +       for (sd = parent_sd->s_dir.children; sd; sd = sd->s_sibling) {
>>> +               if (sd->s_tag != tag)
>>> +                       continue;
>>>                if (!strcmp(sd->s_name, name))
>>>                        return sd;
>>> +       }
>>>        return NULL;
>>>  }
>>>  @@ -632,7 +676,7 @@ struct sysfs_dirent *sysfs_get_dirent(struct
>>> sysfs_dirent *parent_sd,
>>>        struct sysfs_dirent *sd;
>>>          mutex_lock(&sysfs_mutex);
>>> -       sd = sysfs_find_dirent(parent_sd, name);
>>> +       sd = sysfs_find_dirent(parent_sd, NULL, name);
>>>        sysfs_get(sd);
>>>        mutex_unlock(&sysfs_mutex);
>>>  @@ -699,13 +743,18 @@ static struct dentry * sysfs_lookup(struct inode
>>> *dir, struct dentry *dentry,
>>>                                struct nameidata *nd)
>>>  {
>>>        struct dentry *ret = NULL;
>>> -       struct sysfs_dirent *parent_sd = dentry->d_parent->d_fsdata;
>>> +       struct dentry *parent = dentry->d_parent;
>>> +       struct sysfs_dirent *parent_sd = parent->d_fsdata;
>>>        struct sysfs_dirent *sd;
>>>        struct inode *inode;
>>> +       enum sysfs_tag_type type;
>>> +       const void *tag;
>>>          mutex_lock(&sysfs_mutex);
>>>  -       sd = sysfs_find_dirent(parent_sd, dentry->d_name.name);
>>> +       type = sysfs_tag_type(parent_sd);
>>> +       tag = sysfs_info(parent->d_sb)->tag[type];
>>> +       sd = sysfs_find_dirent(parent_sd, tag, dentry->d_name.name);
>>>          /* no such entry */
>>>        if (!sd) {
>>> @@ -913,19 +962,24 @@ int sysfs_rename_dir(struct kobject * kobj, const
>>> char *new_name)
>>>        struct sysfs_rename_struct *srs;
>>>        struct inode *parent_inode = NULL;
>>>        const char *dup_name = NULL;
>>> +       const void *old_tag, *tag;
>>>        int error;
>>>          INIT_LIST_HEAD(&todo);
>>>        mutex_lock(&sysfs_rename_mutex);
>>> +       old_tag = sd->s_tag;
>>> +       tag = sysfs_creation_tag(sd->s_parent, sd);
>>>          error = 0;
>>> -       if (strcmp(sd->s_name, new_name) == 0)
>>> +       if ((old_tag == tag) && (strcmp(sd->s_name, new_name) == 0))
>>>                goto out;       /* nothing to rename */
>>>          sysfs_grab_supers();
>>> -       error = prep_rename(&todo, sd, sd->s_parent, new_name);
>>> -       if (error)
>>> -               goto out_release;
>>> +       if (old_tag == tag) {
>>> +               error = prep_rename(&todo, sd, sd->s_parent, new_name);
>>> +               if (error)
>>> +                       goto out_release;
>>> +       }
>>>          error = -ENOMEM;
>>>        mutex_lock(&sysfs_mutex);
>>> @@ -938,7 +992,7 @@ int sysfs_rename_dir(struct kobject * kobj, const char
>>> *new_name)
>>>        mutex_lock(&sysfs_mutex);
>>>          error = -EEXIST;
>>> -       if (sysfs_find_dirent(sd->s_parent, new_name))
>>> +       if (sysfs_find_dirent(sd->s_parent, tag, new_name))
>>>                goto out_unlock;
>>>          /* rename sysfs_dirent */
>>> @@ -949,6 +1003,7 @@ int sysfs_rename_dir(struct kobject * kobj, const
>>> char *new_name)
>>>          dup_name = sd->s_name;
>>>        sd->s_name = new_name;
>>> +       sd->s_tag = tag;
>>>          /* rename */
>>>        list_for_each_entry(srs, &todo, list) {
>>> @@ -956,6 +1011,20 @@ int sysfs_rename_dir(struct kobject * kobj, const
>>> char *new_name)
>>>                d_move(srs->old_dentry, srs->new_dentry);
>>>        }
>>>  +       /* If we are moving across superblocks drop the dcache entries */
>>> +       if (old_tag != tag) {
>>> +               struct super_block *sb;
>>> +               struct dentry *dentry;
>>> +               list_for_each_entry(sb, &sysfs_fs_type.fs_supers,
>>> s_instances) {
>>> +                       dentry = __sysfs_get_dentry(sb, sd);
>>> +                       if (!dentry)
>>> +                               continue;
>>> +                       shrink_dcache_parent(dentry);
>>> +                       d_drop(dentry);
>>> +                       dput(dentry);
>>> +               }
>>> +       }
>>> +
>>>        error = 0;
>>>  out_unlock:
>>>        mutex_unlock(&sysfs_mutex);
>>> @@ -978,11 +1047,13 @@ int sysfs_move_dir(struct kobject *kobj, struct
>>> kobject *new_parent_kobj)
>>>        struct sysfs_rename_struct *srs;
>>>        struct inode *old_parent_inode = NULL, *new_parent_inode = NULL;
>>>        int error;
>>> +       const void *tag;
>>>          INIT_LIST_HEAD(&todo);
>>>        mutex_lock(&sysfs_rename_mutex);
>>>        BUG_ON(!sd->s_parent);
>>>        new_parent_sd = new_parent_kobj->sd ? new_parent_kobj->sd :
>>> &sysfs_root;
>>> +       tag = sd->s_tag;
>>>          error = 0;
>>>        if (sd->s_parent == new_parent_sd)
>>> @@ -1016,7 +1087,7 @@ again:
>>>        mutex_lock(&sysfs_mutex);
>>>          error = -EEXIST;
>>> -       if (sysfs_find_dirent(new_parent_sd, sd->s_name))
>>> +       if (sysfs_find_dirent(new_parent_sd, tag, sd->s_name))
>>>                goto out_unlock;
>>>          error = 0;
>>> @@ -1055,10 +1126,12 @@ static inline unsigned char dt_type(struct
>>> sysfs_dirent *sd)
>>>   static int sysfs_readdir(struct file * filp, void * dirent, filldir_t
>>> filldir)
>>>  {
>>> -       struct dentry *dentry = filp->f_path.dentry;
>>> -       struct sysfs_dirent * parent_sd = dentry->d_fsdata;
>>> +       struct dentry *parent = filp->f_path.dentry;
>>> +       struct sysfs_dirent *parent_sd = parent->d_fsdata;
>>>        struct sysfs_dirent *pos;
>>>        ino_t ino;
>>> +       enum sysfs_tag_type type;
>>> +       const void *tag;
>>>          if (filp->f_pos == 0) {
>>>                ino = parent_sd->s_ino;
>>> @@ -1076,6 +1149,9 @@ static int sysfs_readdir(struct file * filp, void *
>>> dirent, filldir_t filldir)
>>>        if ((filp->f_pos > 1) && (filp->f_pos < INT_MAX)) {
>>>                mutex_lock(&sysfs_mutex);
>>>  +               type = sysfs_tag_type(parent_sd);
>>> +               tag = sysfs_info(parent->d_sb)->tag[type];
>>> +
>>>                /* Skip the dentries we have already reported */
>>>                pos = parent_sd->s_dir.children;
>>>                while (pos && (filp->f_pos > pos->s_ino))
>>> @@ -1085,6 +1161,9 @@ static int sysfs_readdir(struct file * filp, void *
>>> dirent, filldir_t filldir)
>>>                        const char * name;
>>>                        int len;
>>>  +                       if (pos->s_tag != tag)
>>> +                               continue;
>>> +
>>>                        name = pos->s_name;
>>>                        len = strlen(name);
>>>                        filp->f_pos = ino = pos->s_ino;
>>> @@ -1105,3 +1184,35 @@ const struct file_operations sysfs_dir_operations =
>>> {
>>>        .read           = generic_read_dir,
>>>        .readdir        = sysfs_readdir,
>>>  };
>>> +
>>> +/**
>>> + *     sysfs_make_tagged_dir - Require tags of all the entries in a
>>> directory.
>>> + *     @kobj:  object whose children should be filtered by tags
>>> + *
>>> + *     Once tagging has been enabled on a directory the contents
>>> + *     of the directory become dependent upon context captured when
>>> + *     sysfs was mounted.
>>> + */
>>> +int sysfs_make_tagged_dir(struct kobject *kobj, enum sysfs_tag_type type)
>>> +{
>>> +       struct sysfs_dirent *sd;
>>> +       int err;
>>> +
>>> +       err = -ENOENT;
>>> +       sd = kobj->sd;
>>> +
>>> +       mutex_lock(&sysfs_mutex);
>>> +       err = -EINVAL;
>>> +       /* We can only enable tagging when we have a valid tag type
>>> +        * on empty directories where taggint has not already been
>>> +        * enabled.
>>> +        */
>>> +       if ((type > SYSFS_TAG_TYPE_NONE) && (type < SYSFS_TAG_TYPES) &&
>>> +           tag_ops[type] && !sysfs_tag_type(sd) &&
>>> +           (sysfs_type(sd) == SYSFS_DIR) && !sd->s_dir.children) {
>>> +               err = 0;
>>> +               sd->s_flags |= (type << SYSFS_TAG_TYPE_SHIFT);
>>> +       }
>>> +       mutex_unlock(&sysfs_mutex);
>>> +       return err;
>>> +}
>>> diff --git a/fs/sysfs/file.c b/fs/sysfs/file.c
>>> index 61c3476..091c0de 100644
>>> --- a/fs/sysfs/file.c
>>> +++ b/fs/sysfs/file.c
>>> @@ -476,9 +476,12 @@ void sysfs_notify(struct kobject *k, char *dir, char
>>> *attr)
>>>        mutex_lock(&sysfs_mutex);
>>>          if (sd && dir)
>>> -               sd = sysfs_find_dirent(sd, dir);
>>> +               /* Only directories are tagged, so no need to pass
>>> +                * a tag explicitly.
>>> +                */
>>> +               sd = sysfs_find_dirent(sd, NULL, dir);
>>>        if (sd && attr)
>>> -               sd = sysfs_find_dirent(sd, attr);
>>> +               sd = sysfs_find_dirent(sd, NULL, attr);
>>>        if (sd)
>>>                sysfs_notify_dirent(sd);
>>>  @@ -640,7 +643,7 @@ EXPORT_SYMBOL_GPL(sysfs_chmod_file);
>>>   void sysfs_remove_file(struct kobject * kobj, const struct attribute *
>>> attr)
>>>  {
>>> -       sysfs_hash_and_remove(kobj->sd, attr->name);
>>> +       sysfs_hash_and_remove(kobj, kobj->sd, attr->name);
>>>  }
>>>   @@ -660,7 +663,7 @@ void sysfs_remove_file_from_group(struct kobject
>>> *kobj,
>>>        else
>>>                dir_sd = sysfs_get(kobj->sd);
>>>        if (dir_sd) {
>>> -               sysfs_hash_and_remove(dir_sd, attr->name);
>>> +               sysfs_hash_and_remove(kobj, dir_sd, attr->name);
>>>                sysfs_put(dir_sd);
>>>        }
>>>  }
>>> diff --git a/fs/sysfs/group.c b/fs/sysfs/group.c
>>> index fe61194..5fba6f2 100644
>>> --- a/fs/sysfs/group.c
>>> +++ b/fs/sysfs/group.c
>>> @@ -23,7 +23,7 @@ static void remove_files(struct sysfs_dirent *dir_sd,
>>> struct kobject *kobj,
>>>        int i;
>>>          for (i = 0, attr = grp->attrs; *attr; i++, attr++)
>>> -               sysfs_hash_and_remove(dir_sd, (*attr)->name);
>>> +               sysfs_hash_and_remove(kobj, dir_sd, (*attr)->name);
>>>  }
>>>   static int create_files(struct sysfs_dirent *dir_sd, struct kobject
>>> *kobj,
>>> @@ -39,7 +39,7 @@ static int create_files(struct sysfs_dirent *dir_sd,
>>> struct kobject *kobj,
>>>                 * visibility.  Do this by first removing then
>>>                 * re-adding (if required) the file */
>>>                if (update)
>>> -                       sysfs_hash_and_remove(dir_sd, (*attr)->name);
>>> +                       sysfs_hash_and_remove(kobj, dir_sd,
>>> (*attr)->name);
>>>                if (grp->is_visible) {
>>>                        mode = grp->is_visible(kobj, *attr, i);
>>>                        if (!mode)
>>> diff --git a/fs/sysfs/inode.c b/fs/sysfs/inode.c
>>> index 80f8fd4..b5fc78a 100644
>>> --- a/fs/sysfs/inode.c
>>> +++ b/fs/sysfs/inode.c
>>> @@ -226,17 +226,20 @@ struct inode * sysfs_get_inode(struct sysfs_dirent
>>> *sd)
>>>        return inode;
>>>  }
>>>  -int sysfs_hash_and_remove(struct sysfs_dirent *dir_sd, const char *name)
>>> +int sysfs_hash_and_remove(struct kobject *kobj, struct sysfs_dirent
>>> *dir_sd,
>>> +                         const char *name)
>>>  {
>>>        struct sysfs_addrm_cxt acxt;
>>>        struct sysfs_dirent *sd;
>>> +       const void *tag;
>>>          if (!dir_sd)
>>>                return -ENOENT;
>>>          sysfs_addrm_start(&acxt, dir_sd);
>>> +       tag = kobj->sd->s_tag;
>>>  -       sd = sysfs_find_dirent(dir_sd, name);
>>> +       sd = sysfs_find_dirent(dir_sd, tag, name);
>>>        if (sd)
>>>                sysfs_remove_one(&acxt, sd);
>>>  diff --git a/fs/sysfs/mount.c b/fs/sysfs/mount.c
>>> index 6ebda1a..8f2237a 100644
>>> --- a/fs/sysfs/mount.c
>>> +++ b/fs/sysfs/mount.c
>>> @@ -35,12 +35,15 @@ static const struct super_operations sysfs_ops = {
>>>  struct sysfs_dirent sysfs_root = {
>>>        .s_name         = "",
>>>        .s_count        = ATOMIC_INIT(1),
>>> -       .s_flags        = SYSFS_DIR,
>>> +       .s_flags        = SYSFS_DIR | (SYSFS_TAG_TYPE_NONE <<
>>> SYSFS_TAG_TYPE_SHIFT),
>>>        .s_mode         = S_IFDIR | S_IRWXU | S_IRUGO | S_IXUGO,
>>>        .s_ino          = 1,
>>>  };
>>>  -static int sysfs_fill_super(struct super_block *sb, void *data, int
>>> silent)
>>> +struct sysfs_tag_type_operations *tag_ops[SYSFS_TAG_TYPES];
>>> +
>>> +static int sysfs_fill_super(struct super_block *sb, void *data, int
>>> silent,
>>> +       const void *tags[SYSFS_TAG_TYPES])
>>>  {
>>>        struct sysfs_super_info *info = NULL;
>>>        struct inode *inode = NULL;
>>> @@ -76,8 +79,10 @@ static int sysfs_fill_super(struct super_block *sb,
>>> void *data, int silent)
>>>                goto out_err;
>>>        }
>>>        root->d_fsdata = &sysfs_root;
>>> +       root->d_sb = sb;
>>>        sb->s_root = root;
>>>        sb->s_fs_info = info;
>>> +       memcpy(info->tag, tags, sizeof(info->tag[0])*SYSFS_TAG_TYPES);
>>>        return 0;
>>>   out_err:
>>> @@ -89,20 +94,74 @@ out_err:
>>>        return error;
>>>  }
>>>  +static int sysfs_test_super(struct super_block *sb, void *ptr)
>>> +{
>>> +       const void **tag = ptr;
>>> +       struct sysfs_super_info *info = sysfs_info(sb);
>>> +       enum sysfs_tag_type type;
>>> +       int found = 1;
>>> +
>>> +       for (type = SYSFS_TAG_TYPE_NONE; type < SYSFS_TAG_TYPES; type++) {
>>> +               if (info->tag[type] != tag[type]) {
>>> +                       found = 0;
>>> +                       break;
>>> +               }
>>> +       }
>>> +
>>> +       return found;
>>> +}
>>> +
>>>  static int sysfs_get_sb(struct file_system_type *fs_type,
>>>        int flags, const char *dev_name, void *data, struct vfsmount *mnt)
>>>  {
>>> -       int rc;
>>> +       const void *tag[SYSFS_TAG_TYPES];
>>> +       struct super_block *sb;
>>> +       int error;
>>> +       enum sysfs_tag_type type;
>>> +
>>> +       for (type = SYSFS_TAG_TYPE_NONE; type < SYSFS_TAG_TYPES; type++) {
>>> +               tag[type] = NULL;
>>> +               if (!tag_ops[type])
>>> +                       continue;
>>> +               tag[type] = tag_ops[type]->mount_tag();
>>> +       }
>>> +
>>>        mutex_lock(&sysfs_rename_mutex);
>>> -       rc = get_sb_single(fs_type, flags, data, sysfs_fill_super, mnt);
>>> +       sb = sget(fs_type, sysfs_test_super, set_anon_super, tag);
>>> +       if (IS_ERR(sb)) {
>>> +               error = PTR_ERR(sb);
>>> +               goto out;
>>> +       }
>>> +       if (!sb->s_root) {
>>> +               sb->s_flags = flags;
>>> +               error = sysfs_fill_super(sb, data, flags & MS_SILENT ? 1 :
>>> 0,
>>> +                                       tag);
>>> +               if (error) {
>>> +                       up_write(&sb->s_umount);
>>> +                       deactivate_super(sb);
>>> +                       goto out;
>>> +               }
>>> +               sb->s_flags |= MS_ACTIVE;
>>> +       }
>>> +       do_remount_sb(sb, flags, data, 0);
>>> +       error = simple_set_mnt(mnt, sb);
>>> +out:
>>>        mutex_unlock(&sysfs_rename_mutex);
>>> -       return rc;
>>> +       return error;
>>> +}
>>> +
>>> +static void sysfs_kill_sb(struct super_block *sb)
>>> +{
>>> +       struct sysfs_super_info *info = sysfs_info(sb);
>>> +
>>> +       kill_anon_super(sb);
>>> +       kfree(info);
>>>  }
>>>   struct file_system_type sysfs_fs_type = {
>>>        .name           = "sysfs",
>>>        .get_sb         = sysfs_get_sb,
>>> -       .kill_sb        = kill_anon_super,
>>> +       .kill_sb        = sysfs_kill_sb,
>>>  };
>>>   void sysfs_grab_supers(void)
>>> @@ -146,6 +205,50 @@ restart:
>>>        spin_unlock(&sb_lock);
>>>  }
>>>  +int sysfs_register_tag_type(enum sysfs_tag_type type, struct
>>> sysfs_tag_type_operations *ops)
>>> +{
>>> +       int error;
>>> +
>>> +       mutex_lock(&sysfs_rename_mutex);
>>> +
>>> +       error = -EINVAL;
>>> +       if (type >= SYSFS_TAG_TYPES)
>>> +               goto out;
>>> +
>>> +       error = -EINVAL;
>>> +       if (type <= SYSFS_TAG_TYPE_NONE)
>>> +               goto out;
>>> +
>>> +       error = -EBUSY;
>>> +       if (tag_ops[type])
>>> +               goto out;
>>> +
>>> +       error = 0;
>>> +       tag_ops[type] = ops;
>>> +
>>> +out:
>>> +       mutex_unlock(&sysfs_rename_mutex);
>>> +       return error;
>>> +}
>>> +
>>> +void sysfs_exit_tag(enum sysfs_tag_type type, const void *tag)
>>> +{
>>> +       /* Allow the tag to go away while sysfs is still mounted. */
>>> +       struct super_block *sb;
>>> +       mutex_lock(&sysfs_rename_mutex);
>>> +       sysfs_grab_supers();
>>> +       mutex_lock(&sysfs_mutex);
>>> +       list_for_each_entry(sb, &sysfs_fs_type.fs_supers, s_instances) {
>>> +               struct sysfs_super_info *info = sysfs_info(sb);
>>> +               if (info->tag[type] != tag)
>>> +                       continue;
>>> +               info->tag[type] = NULL;
>>> +       }
>>> +       mutex_unlock(&sysfs_mutex);
>>> +       sysfs_release_supers();
>>> +       mutex_unlock(&sysfs_rename_mutex);
>>> +}
>>> +
>>>  int __init sysfs_init(void)
>>>  {
>>>        int err = -ENOMEM;
>>> diff --git a/fs/sysfs/symlink.c b/fs/sysfs/symlink.c
>>> index a3ba217..54b2e5f 100644
>>> --- a/fs/sysfs/symlink.c
>>> +++ b/fs/sysfs/symlink.c
>>> @@ -119,7 +119,7 @@ void sysfs_remove_link(struct kobject * kobj, const
>>> char * name)
>>>        else
>>>                parent_sd = kobj->sd;
>>>  -       sysfs_hash_and_remove(parent_sd, name);
>>> +       sysfs_hash_and_remove(kobj, parent_sd, name);
>>>  }
>>>   static int sysfs_get_target_path(struct sysfs_dirent *parent_sd,
>>> diff --git a/fs/sysfs/sysfs.h b/fs/sysfs/sysfs.h
>>> index f0e5ecb..67115ec 100644
>>> --- a/fs/sysfs/sysfs.h
>>> +++ b/fs/sysfs/sysfs.h
>>> @@ -45,6 +45,7 @@ struct sysfs_dirent {
>>>        struct sysfs_dirent     *s_sibling;
>>>        const char              *s_name;
>>>  +       const void              *s_tag;
>>>        union {
>>>                struct sysfs_elem_dir           s_dir;
>>>                struct sysfs_elem_symlink       s_symlink;
>>> @@ -67,14 +68,22 @@ struct sysfs_dirent {
>>>  #define SYSFS_KOBJ_LINK                        0x0008
>>>  #define SYSFS_COPY_NAME                        (SYSFS_DIR |
>>> SYSFS_KOBJ_LINK)
>>>  -#define SYSFS_FLAG_MASK                        ~SYSFS_TYPE_MASK
>>> -#define SYSFS_FLAG_REMOVED             0x0200
>>> +#define SYSFS_TAG_TYPE_MASK            0xff00
>>> +#define SYSFS_TAG_TYPE_SHIFT           8
>>> +
>>> +#define SYSFS_FLAG_MASK                        ~(SYSFS_TYPE_MASK |
>>> SYSFS_TAG_TYPE_MASK)
>>> +#define SYSFS_FLAG_REMOVED             0x020000
>>>   static inline unsigned int sysfs_type(struct sysfs_dirent *sd)
>>>  {
>>>        return sd->s_flags & SYSFS_TYPE_MASK;
>>>  }
>>>  +static inline enum sysfs_tag_type sysfs_tag_type(struct sysfs_dirent
>>> *sd)
>>> +{
>>> +       return (sd->s_flags & SYSFS_TAG_TYPE_MASK) >>
>>> SYSFS_TAG_TYPE_SHIFT;
>>> +}
>>> +
>>>  /*
>>>  * Context structure to be used while adding/removing nodes.
>>>  */
>>> @@ -87,6 +96,7 @@ struct sysfs_addrm_cxt {
>>>   struct sysfs_super_info {
>>>        int     grabbed;
>>> +       const void *tag[SYSFS_TAG_TYPES];
>>>  };
>>>   #define sysfs_info(SB) ((struct sysfs_super_info *)(SB)->s_fs_info)
>>> @@ -98,6 +108,7 @@ extern struct sysfs_dirent sysfs_root;
>>>  extern struct super_block *sysfs_sb;
>>>  extern struct kmem_cache *sysfs_dir_cachep;
>>>  extern struct file_system_type sysfs_fs_type;
>>> +extern struct sysfs_tag_type_operations *tag_ops[SYSFS_TAG_TYPES];
>>>   void sysfs_grab_supers(void);
>>>  void sysfs_release_supers(void);
>>> @@ -124,6 +135,7 @@ void sysfs_remove_one(struct sysfs_addrm_cxt *acxt,
>>> struct sysfs_dirent *sd);
>>>  void sysfs_addrm_finish(struct sysfs_addrm_cxt *acxt);
>>>   struct sysfs_dirent *sysfs_find_dirent(struct sysfs_dirent *parent_sd,
>>> +                                      const void *tag,
>>>                                       const unsigned char *name);
>>>  struct sysfs_dirent *sysfs_get_dirent(struct sysfs_dirent *parent_sd,
>>>                                      const unsigned char *name);
>>> @@ -158,7 +170,8 @@ static inline void __sysfs_put(struct sysfs_dirent
>>> *sd)
>>>  struct inode *sysfs_get_inode(struct sysfs_dirent *sd);
>>>  int sysfs_sd_setattr(struct sysfs_dirent *sd, struct inode *inode, struct
>>> iattr *iattr);
>>>  int sysfs_setattr(struct dentry *dentry, struct iattr *iattr);
>>> -int sysfs_hash_and_remove(struct sysfs_dirent *dir_sd, const char *name);
>>> +int sysfs_hash_and_remove(struct kobject *kobj, struct sysfs_dirent
>>> *dir_sd,
>>> +                         const char *name);
>>>  int sysfs_inode_init(void);
>>>   /*
>>> diff --git a/include/linux/kobject.h b/include/linux/kobject.h
>>> index 5437ac0..beb3573 100644
>>> --- a/include/linux/kobject.h
>>> +++ b/include/linux/kobject.h
>>> @@ -105,6 +105,7 @@ struct kobj_type {
>>>        void (*release)(struct kobject *kobj);
>>>        struct sysfs_ops *sysfs_ops;
>>>        struct attribute **default_attrs;
>>> +       const void *(*sysfs_tag)(struct kobject *kobj);
>>>  };
>>>   struct kobj_uevent_env {
>>> diff --git a/include/linux/sysfs.h b/include/linux/sysfs.h
>>> index d8e0230..ba68829 100644
>>> --- a/include/linux/sysfs.h
>>> +++ b/include/linux/sysfs.h
>>> @@ -80,6 +80,15 @@ struct sysfs_ops {
>>>   struct sysfs_dirent;
>>>  +enum sysfs_tag_type {
>>> +       SYSFS_TAG_TYPE_NONE = 0,
>>> +       SYSFS_TAG_TYPES
>>> +};
>>> +
>>> +struct sysfs_tag_type_operations {
>>> +       const void *(*mount_tag)(void);
>>> +};
>>> +
>>>  #ifdef CONFIG_SYSFS
>>>   int sysfs_schedule_callback(struct kobject *kobj, void (*func)(void *),
>>> @@ -126,6 +135,12 @@ struct sysfs_dirent *sysfs_get_dirent(struct
>>> sysfs_dirent *parent_sd,
>>>  struct sysfs_dirent *sysfs_get(struct sysfs_dirent *sd);
>>>  void sysfs_put(struct sysfs_dirent *sd);
>>>  void sysfs_printk_last_file(void);
>>> +
>>> +int sysfs_make_tagged_dir(struct kobject *, enum sysfs_tag_type
>>> tag_type);
>>> +int sysfs_register_tag_type(enum sysfs_tag_type type,
>>> +                           struct sysfs_tag_type_operations *ops);
>>> +void sysfs_exit_tag(enum sysfs_tag_type type, const void *tag);
>>> +
>>>  int __must_check sysfs_init(void);
>>>   #else /* CONFIG_SYSFS */
>>> @@ -249,6 +264,22 @@ static inline void sysfs_put(struct sysfs_dirent *sd)
>>>  {
>>>  }
>>>  +staticn inline int sysfs_make_tagged_dir(struct kobject *kobj,
>>   ______^
>>
>> This typo is still present in your patch in the CONFIG_SYSFS=n case.
>>
>> Otherwise the patchset, combined with the patches Greg has already
>> merged in his tree, still works great for me with network namespaces.
>>
>> --Benjamin
>>
>>> +                                               enum sysfs_tag_type
>>> tag_type)
>>> +{
>>> +       return 0;
>>> +}
>>> +
>>> +static inline int sysfs_register_tag_type(enum sysfs_tag_type type,
>>> +                                       struct sysfs_tag_type_operations
>>> *ops)
>>> +{
>>> +       return 0;
>>> +}
>>> +
>>> +static inline void sysfs_exit_tag(enum sysfs_tag_type type, const void
>>> *tag)
>>> +{
>>> +}
>>> +
>>>  static inline int __must_check sysfs_init(void)
>>>  {
>>>        return 0;
>>
>> --
>> B e n j a m i n   T h e r y  - BULL/DT/Open Software R&D
>>
>>   http://www.bull.com
>> --
>> To unsubscribe from this list: send the line "unsubscribe netdev" in
>> the body of a message to majordomo@vger.kernel.org
>> More majordomo info at  http://vger.kernel.org/majordomo-info.html
>>
> 
> 


-- 
B e n j a m i n   T h e r y  - BULL/DT/Open Software R&D

    http://www.bull.com

  reply	other threads:[~2008-09-02 14:03 UTC|newest]

Thread overview: 80+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
     [not found] <20080618170729.808539948@theryb.frec.bull.fr>
     [not found] ` <20080618170731.002784342@theryb.frec.bull.fr>
     [not found]   ` <485F04E1.70204@gmail.com>
     [not found]     ` <m1y74svtff.fsf@frodo.ebiederm.org>
     [not found]       ` <486706C9.9040303@gmail.com>
     [not found]         ` <m18wwmsqdv.fsf@frodo.ebiederm.org>
     [not found]           ` <4869D314.5030403@gmail.com>
     [not found]             ` <m1ej6enep7.fsf@frodo.ebiederm.org>
     [not found]               ` <486A0751.9080602@gmail.com>
     [not found]                 ` <m1wsk5hjm0.fsf@frodo.ebiederm.org>
     [not found]                   ` <486AF4FA.8020805@gmail.com>
     [not found]                     ` <m1ej6d9c2r.fsf@frodo.ebiederm.org>
     [not found]                       ` <486B060C.7030607@gmail.com>
     [not found]                         ` <m14p78s02q.fsf@frodo.ebiederm.org>
     [not found]                           ` <486C4515.1070007@gmail.com>
     [not found]                             ` <m1hcb7o8lv.fsf@frodo.ebiederm.org>
     [not found]                               ` <486CB051.5000507@fr.ibm.com>
     [not found]                                 ` <m14p77m9uy.fsf@frodo.ebiederm.org>
     [not found]                                   ` <486CF71F.5090405@gmail.com>
2008-07-04  0:48                                     ` [PATCH 00/15] sysfs support for namespaces Eric W. Biederman
2008-07-04  1:05                                       ` [PATCH 01/15] kobject: Cleanup kobject_rename and !CONFIG_SYSFS Eric W. Biederman
2008-07-04  1:07                                         ` [PATCH 02/15] sysfs: Support for preventing unmounts Eric W. Biederman
2008-07-04  1:08                                           ` [PATCH 03/15] sysfs: sysfs_get_dentry add a sb parameter Eric W. Biederman
2008-07-04  1:09                                             ` [PATCH 04/15] sysfs: Implement __sysfs_get_dentry Eric W. Biederman
2008-07-04  1:10                                               ` [PATCH 05/15] sysfs: Rename Support multiple superblocks Eric W. Biederman
2008-07-04  1:11                                                 ` [PATCH 06/15] Introduce sysfs_sd_setattr and fix sysfs_chmod Eric W. Biederman
2008-07-04  1:13                                                   ` [PATCH 07/15] sysfs: sysfs_chmod_file handle multiple superblocks Eric W. Biederman
2008-07-04  1:14                                                     ` [PATCH 08/15] sysfs: Make sysfs_mount static once again Eric W. Biederman
2008-07-04  1:16                                                       ` [PATCH 09/15] sysfs: Implement sysfs tagged directory support Eric W. Biederman
2008-07-04  1:17                                                         ` [PATCH 10/15] sysfs: Merge sysfs_rename_dir and sysfs_move_dir Eric W. Biederman
2008-07-04  1:18                                                           ` [PATCH 11/15] sysfs: Implement sysfs_delete_link and sysfs_rename_link Eric W. Biederman
2008-07-04  1:20                                                             ` [PATCH 12/15] driver core: Implement tagged directory support for device classes Eric W. Biederman
2008-07-04  1:21                                                               ` [PATCH 13/15] Revert "netns: Fix device renaming for sysfs" Eric W. Biederman
2008-07-04  1:22                                                                 ` [PATCH 14/15] netns: Enable tagging for net_class directories in sysfs Eric W. Biederman
2008-07-04  1:23                                                                   ` [PATCH 15/15] sysfs: user namespaces: fix bug with clone(CLONE_NEWUSER) with fairsched Eric W. Biederman
2008-07-04  7:50                                                               ` [PATCH 12/15] driver core: Implement tagged directory support for device classes Tejun Heo
2008-07-04 13:31                                                                 ` Eric W. Biederman
2008-07-04 13:57                                                                   ` Tejun Heo
2008-07-04 16:12                                                                     ` Greg KH
2008-07-04 21:49                                                                       ` Eric W. Biederman
2008-07-14  1:54                                                                       ` Eric W. Biederman
2008-07-16  3:25                                                                         ` Tejun Heo
2008-07-16  5:41                                                                           ` Eric W. Biederman
2008-07-16  5:50                                                                             ` Tejun Heo
2008-07-16  6:32                                                                               ` Eric W. Biederman
2008-07-16  6:48                                                                                 ` Tejun Heo
2008-07-16  7:02                                                                                   ` Tejun Heo
2008-07-16 19:07                                                                                     ` Eric W. Biederman
2008-07-16 21:09                                                                                   ` Eric W. Biederman
2008-07-17 23:08                                                                           ` Greg KH
2008-07-18 12:41                                                                             ` Tejun Heo
2008-07-18 18:49                                                                               ` Greg KH
2008-07-18 20:19                                                                                 ` Eric W. Biederman
2008-07-19  1:07                                                                                 ` Tejun Heo
2008-08-03  6:59                                                                                 ` Eric W. Biederman
2008-07-04 22:00                                                                     ` Eric W. Biederman
2008-08-20  2:17                                                         ` [PATCH 09/15] sysfs: Implement sysfs tagged directory support Greg KH
2008-08-20  6:58                                                           ` Eric W. Biederman
2008-08-21  6:31                                                           ` [PATCH 0/8] sysfs namespace support Eric W. Biederman
2008-08-21  6:33                                                             ` [PATCH 1/8] sysfs: Implement sysfs tagged directory support Eric W. Biederman
2008-08-21  6:34                                                               ` [PATCH 2/8] sysfs: Merge sysfs_rename_dir and sysfs_move_dir Eric W. Biederman
2008-08-21  6:35                                                                 ` [PATCH 3/8] sysfs: Implement sysfs_delete_link and sysfs_rename_link Eric W. Biederman
2008-08-21  6:36                                                                   ` [PATCH 5/8] sysfs: Remove sysfs_create_link_nowarn Eric W. Biederman
2008-08-21  6:38                                                                     ` [PATCH 6/8] Revert "netns: Fix device renaming for sysfs" Eric W. Biederman
2008-08-21  6:39                                                                       ` [PATCH 7/8] netns: Enable tagging for net_class directories in sysfs Eric W. Biederman
2008-08-21  6:40                                                                         ` [PATCH 8/8] sysfs: user namespaces: fix bug with clone(CLONE_NEWUSER) with fairsched Eric W. Biederman
2008-08-21  6:47                                                                         ` [PATCH 7/8] netns: Enable tagging for net_class directories in sysfs David Miller
2008-08-21  6:47                                                                       ` [PATCH 6/8] Revert "netns: Fix device renaming for sysfs" David Miller
2008-08-21  6:37                                                                   ` [PATCH 4/8] driver core: Implement tagged directory support for device classes Eric W. Biederman
2008-08-27 15:18                                                               ` [PATCH 1/8] sysfs: Implement sysfs tagged directory support Benjamin Thery
2008-09-02 13:54                                                                 ` Mark Ryden
2008-09-02 14:03                                                                   ` Benjamin Thery [this message]
2008-09-02 17:01                                                                     ` Greg KH
2008-09-04  5:33                                                                       ` David Shwatrz
2008-09-04  6:44                                                                         ` Benjamin Thery
2008-09-08 18:39                                                                           ` Mark Ryden
2008-10-07 16:39                                                                           ` Mark Ryden
2008-10-07 16:48                                                                             ` Greg KH
2008-10-07 20:31                                                                               ` Eric W. Biederman
2008-10-07 21:09                                                                                 ` Greg KH
2008-10-07 22:27                                                                                   ` Eric W. Biederman
2008-10-08 13:00                                                                                     ` Christoph Hellwig
2008-10-14  3:20                                                                                       ` Eric W. Biederman
2008-10-07 16:52                                                                             ` Daniel Lezcano
2008-08-21  6:37                                                             ` [PATCH 0/8] sysfs namespace support David Miller
2008-07-04  6:44                                                       ` [PATCH 08/15] sysfs: Make sysfs_mount static once again Tejun Heo
2008-07-04  6:44                                                     ` [PATCH 07/15] sysfs: sysfs_chmod_file handle multiple superblocks Tejun Heo
2008-08-20  2:16                                                     ` patch sysfs-sysfs_chmod_file-handle-multiple-superblocks.patch added to gregkh-2.6 tree gregkh
2008-07-04  6:40                                                   ` [PATCH 06/15] Introduce sysfs_sd_setattr and fix sysfs_chmod Tejun Heo
2008-08-20  2:16                                                   ` patch sysfs-introduce-sysfs_sd_setattr-and-fix-sysfs_chmod.patch added to gregkh-2.6 tree gregkh
2008-08-20  2:16                                                 ` patch sysfs-rename-support-multiple-superblocks.patch " gregkh
2008-08-20  2:16                                               ` patch sysfs-implement-__sysfs_get_dentry.patch " gregkh
2008-08-20  2:16                                             ` patch sysfs-sysfs_get_dentry-add-a-sb-parameter.patch " gregkh
2008-07-04  6:33                                         ` [PATCH 01/15] kobject: Cleanup kobject_rename and !CONFIG_SYSFS Tejun Heo
2008-08-20  1:48                                         ` patch kobject-cleanup-kobject_rename-and-config_sysfs.patch added to gregkh-2.6 tree gregkh
2008-07-04  1:27                                       ` [PATCH 00/15] sysfs support for namespaces Eric W. Biederman
2008-07-06  4:42                                       ` Eric W. Biederman
2008-07-07 11:41                                         ` Cornelia Huck
2008-07-07 12:22                                           ` Eric W. Biederman

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=48BD47A8.1080909@bull.net \
    --to=benjamin.thery@bull.net \
    --cc=akpm@linux-foundation.org \
    --cc=containers@lists.osdl.org \
    --cc=dlezcano@fr.ibm.com \
    --cc=ebiederm@xmission.com \
    --cc=greg@kroah.com \
    --cc=gregkh@suse.de \
    --cc=htejun@gmail.com \
    --cc=linux-kernel@vger.kernel.org \
    --cc=markryde@gmail.com \
    --cc=netdev@vger.kernel.org \
    --cc=viro@ftp.linux.org.uk \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).