This patch adds cgcreate tool, which creates cgroups based on input
parameters - the syntax is:

cgcreate -t <tuid>:<tgid> -a <agid>:<auid> -g <list of controllers>:<relative 
path to cgroup>

where:
-a enables user to define admin gid and uid
-t enables user to define task gid and uid
-g sets pairs list of controllers-relative path to cgroup

------------------------------
fix changelog:

fix memory leaks which dhaval find:
adds cgroup_free to two places:
                                                if (!cgc) {
                                                                ret = ECGINVAL;
                                                                fprintf(stderr, 
"%s: "
                                                                                
"controller %s can't be add\n",
                                                                                
argv[0], cgroup_list[i]->controllers[j]);
+                                                               
cgroup_free(&cgroup);
                                                                goto err;
                                                }
and
                                if (ret) {
                                                fprintf(stderr, "%s: "
                                                                "can't create 
cgroup %s: %s\n",
                                                                argv[0], 
cgroup->name, cgroup_strerror(ret));
+                                               cgroup_free(&cgroup);
                                                goto err;
                                }

remove needless condition (thanks dhaval)
-               if (!cgroup->index) {
-                               ret = ECGROUPNOTEXIST;
-                               goto err;
-               }
the test is redundant.


and change cgcreate to argv[0]


regarding Dhaval comments:

>> -bin_PROGRAMS = cgexec cgclassify
>> +bin_PROGRAMS = cgexec cgclassify cgcreate
>
>I wonder. Should this come in sbin if you are going to give absolute
>path. Would it be a good idea to look at classifying it wrt the user's
>cgroup? (the one where cgclassify/cgrulesengd classifies a task to)

I prefer to have the tool in bin directory, just for the case in which there is 
a cgroup in which the user have the right to create group there.


>> + fprintf(stderr, "cgcreate: "
>> + "controller %s can't be add\n",
>> + cgroup_list[i]->controllers[j]);
>> + goto err;
>
>argv[0] instead of cgcreate ? (I'm not sure if that is the correct
>programming practice, so please take this with a grain of salt)

I found both cases - some programs uses name (gcc), some uses argv[0] (ls)
So if you preffer it I can change this.


>> + goto err;
>> + }
>> + }
>
>cgroup_free() ?

Thanks, changed

And fix the problem with cgroup_free.

Signed-off-by: Ivana Varekova <[email protected]>
---

 src/tools/Makefile.am |    4 +
 src/tools/cgcreate.c  |  202 +++++++++++++++++++++++++++++++++++++++++++++++++
 2 files changed, 205 insertions(+), 1 deletions(-)
 create mode 100644 src/tools/cgcreate.c

diff --git a/src/tools/Makefile.am b/src/tools/Makefile.am
index e1cace3..e3a3add 100644
--- a/src/tools/Makefile.am
+++ b/src/tools/Makefile.am
@@ -1,12 +1,14 @@
 INCLUDES = -I$(top_srcdir)/src -I$(top_srcdir)/include
 LDADD = $(top_srcdir)/src/.libs/libcgroup.la
 
-bin_PROGRAMS = cgexec cgclassify
+bin_PROGRAMS = cgexec cgclassify cgcreate
 sbin_PROGRAMS = cgconfigparser
 
 cgexec_SOURCES = cgexec.c tools-common.c tools-common.h
 
 cgclassify_SOURCES = cgclassify.c tools-common.c tools-common.h
 
+cgcreate_SOURCES = cgcreate.c tools-common.c tools-common.h
+
 cgconfigparser_SOURCES = cgconfig.c
 
diff --git a/src/tools/cgcreate.c b/src/tools/cgcreate.c
new file mode 100644
index 0000000..38c767e
--- /dev/null
+++ b/src/tools/cgcreate.c
@@ -0,0 +1,202 @@
+#include <libcgroup.h>
+#include <libcgroup-internal.h>
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <pwd.h>
+#include <sys/types.h>
+#include <errno.h>
+
+#include "tools-common.h"
+
+int main(int argc, char *argv[])
+{
+       int ret = 0;
+       int i, j;
+       char c;
+
+       /* Structure to get GID from group name */
+       struct group *grp = NULL;
+       char *grp_string = NULL;
+
+       /* Structure to get UID from user name */
+       struct passwd *pwd = NULL;
+       char *pwd_string = NULL;
+
+       uid_t tuid = CGRULE_INVALID, auid = CGRULE_INVALID;
+       gid_t tgid = CGRULE_INVALID, agid = CGRULE_INVALID;
+
+       struct cgroup_group_spec *cgroup_list[CG_HIER_MAX];
+       struct cgroup *cgroup;
+       struct cgroup_controller *cgc;
+
+       /* no parametr on input */
+       if (argc < 2) {
+               fprintf(stderr, "Usage is %s "
+                       "-t <tuid>:<tgid> -a <agid>:<auid> "
+                       "-g <list of controllers>:<relative path to cgroup>\n",
+                       argv[0]);
+               return -1;
+       }
+
+       memset(cgroup_list, 0, sizeof(cgroup_list));
+       /* parse arguments */
+       while ((c = getopt(argc, argv, "a:t:g:")) > 0) {
+               switch (c) {
+               case 'a':
+                       /* set admin uid/gid */
+                       if (optarg[0] == ':')
+                               grp_string = strtok(optarg, ":");
+                       else {
+                               pwd_string = strtok(optarg, ":");
+                               if (pwd_string != NULL)
+                                       grp_string = strtok(NULL, ":");
+                       }
+
+                       if (pwd_string != NULL) {
+                               pwd = getpwnam(pwd_string);
+                               if (pwd != NULL) {
+                                       auid = pwd->pw_uid;
+                               } else {
+                                       fprintf(stderr, "%s: "
+                                               "can't find uid of user %s.\n",
+                                               argv[0], pwd_string);
+                                       return -1;
+                               }
+                       }
+                       if (grp_string != NULL) {
+                               grp = getgrnam(grp_string);
+                               if (grp != NULL)
+                                       agid = grp->gr_gid;
+                               else {
+                                       fprintf(stderr, "%s: "
+                                               "can't find gid of group %s.\n",
+                                               argv[0], grp_string);
+                                       return -1;
+                               }
+                       }
+
+                       break;
+               case 't':
+                       /* set task uid/gid */
+                       if (optarg[0] == ':')
+                               grp_string = strtok(optarg, ":");
+                       else {
+                               pwd_string = strtok(optarg, ":");
+                               if (pwd_string != NULL)
+                                       grp_string = strtok(NULL, ":");
+                       }
+
+                       if (pwd_string != NULL) {
+                               pwd = getpwnam(pwd_string);
+                               if (pwd != NULL) {
+                                       tuid = pwd->pw_uid;
+                               } else {
+                                       fprintf(stderr, "%s: "
+                                               "can't find uid of user %s.\n",
+                                               argv[0], pwd_string);
+                                       return -1;
+                               }
+                       }
+                       if (grp_string != NULL) {
+                               grp = getgrnam(grp_string);
+                               if (grp != NULL)
+                                       tgid = grp->gr_gid;
+                               else {
+                                       fprintf(stderr, "%s: "
+                                               "can't find gid of group %s.\n",
+                                               argv[0], grp_string);
+                                       return -1;
+                               }
+                       }
+                       break;
+               case 'g':
+                       if (parse_cgroup_spec(cgroup_list, optarg)) {
+                               fprintf(stderr, "%s: "
+                                       "cgroup controller and path"
+                                       "parsing failed (%s)\n",
+                                       argv[0], argv[optind]);
+                               return -1;
+                       }
+                       break;
+               default:
+                       fprintf(stderr, "%s: "
+                               "invalid command line option\n",
+                               argv[0]);
+                       return -1;
+                       break;
+               }
+       }
+
+       /* no cgroup name */
+       if (argv[optind]) {
+               fprintf(stderr, "%s: "
+                       "wrong arguments (%s)\n",
+                       argv[0], argv[optind]);
+               ret = -1;
+               goto err;
+       }
+
+       /* initialize libcg */
+       ret = cgroup_init();
+       if (ret) {
+               fprintf(stderr, "%s: "
+                       "libcgroup initialization failed: %s\n",
+                       argv[0], cgroup_strerror(ret));
+               goto err;
+       }
+
+       /* for each new cgroup */
+       for (i = 0; i < CG_HIER_MAX; i++) {
+               if (!cgroup_list[i])
+                       break;
+
+               /* create the new cgroup structure */
+               cgroup = cgroup_new_cgroup(cgroup_list[i]->path);
+               if (!cgroup) {
+                       ret = ECGFAIL;
+                       fprintf(stderr, "%s: can't add new cgroup: %s\n",
+                               argv[0], cgroup_strerror(ret));
+                       goto err;
+               }
+
+               /* set uid and gid for the new cgroup based on input options */
+               ret = cgroup_set_uid_gid(cgroup, tuid, tgid, auid, agid);
+               if (ret)
+                       goto err;
+
+               /* add controllers to the new cgroup */
+               j = 0;
+               while (cgroup_list[i]->controllers[j]) {
+                       cgc = cgroup_add_controller(cgroup,
+                               cgroup_list[i]->controllers[j]);
+                       if (!cgc) {
+                               ret = ECGINVAL;
+                               fprintf(stderr, "%s: "
+                                       "controller %s can't be add\n",
+                                       argv[0], 
cgroup_list[i]->controllers[j]);
+                               cgroup_free(&cgroup);
+                               goto err;
+                       }
+                       j++;
+               }
+
+               /* all variables set so create cgroup */
+               ret = cgroup_create_cgroup(cgroup, 0);
+               if (ret) {
+                       fprintf(stderr, "%s: "
+                               "can't create cgroup %s: %s\n",
+                               argv[0], cgroup->name, cgroup_strerror(ret));
+                       cgroup_free(&cgroup);
+                       goto err;
+               }
+               cgroup_free(&cgroup);
+       }
+err:
+       for (i = 0; i < CG_HIER_MAX; i++) {
+               if (cgroup_list[i])
+                       cgroup_free_group_spec(cgroup_list[i]);
+       }
+       return ret;
+}


------------------------------------------------------------------------------
Register Now for Creativity and Technology (CaT), June 3rd, NYC. CaT 
is a gathering of tech-side developers & brand creativity professionals. Meet
the minds behind Google Creative Lab, Visual Complexity, Processing, & 
iPhoneDevCamp as they present alongside digital heavyweights like Barbarian 
Group, R/GA, & Big Spaceship. http://p.sf.net/sfu/creativitycat-com 
_______________________________________________
Libcg-devel mailing list
[email protected]
https://lists.sourceforge.net/lists/listinfo/libcg-devel

Reply via email to