char path[FILENAME_MAX];
};
+/*
+ * Detailed information about available controller.
+ */
+
+struct controller_data {
+/** Controller name. */
+ char name[FILENAME_MAX];
+/**
+ * Hierarchy ID. Controllers with the same hierarchy ID
+ * are mounted together as one hierarchy. Controllers with
+ * ID 0 are not currently used.
+ */
+ int hierarchy;
+/** Number of groups. */
+ int num_cgroups;
+/** Enabled flag */
+ int enabled;
+};
+
/* Functions and structures that can be used by the application*/
struct cgroup;
struct cgroup_controller;
int cgroup_get_controller_next(void **handle, struct cgroup_mount_point *info);
int cgroup_get_controller_end(void **handle);
+/**
+ * Read the list of controllers from /proc/cgroups (not mounted included)
+ * @param handle: Handle to be used for iteration.
+ * @param info: The structure which contains all controller data
+ */
+int cgroup_get_all_controller_begin(void **handle,
+ struct controller_data *info);
+/*
+ * While walking through the mount table, the controllers will be
+ * returned in the same order as is in /proc/cgroups file
+ */
+int cgroup_get_all_controller_next(void **handle, struct controller_data *info);
+int cgroup_get_all_controller_end(void **handle);
+
/*
* Reads the mount to table to give the mount point of a controller
* @controller: Name of the controller
pthread_rwlock_unlock(&cg_mount_table_lock);
return ret;
}
+
+
+int cgroup_get_all_controller_end(void **handle)
+{
+ FILE *proc_cgroup = (FILE *) *handle;
+
+ if (!cgroup_initialized)
+ return ECGROUPNOTINITIALIZED;
+
+ if (!proc_cgroup)
+ return ECGINVAL;
+
+ fclose(proc_cgroup);
+ *handle = NULL;
+
+ return 0;
+}
+
+
+int cgroup_get_all_controller_next(void **handle, struct controller_data *info)
+{
+ FILE *proc_cgroup = (FILE *) *handle;
+ int err = 0;
+ int hierarchy, num_cgroups, enabled;
+ char subsys_name[FILENAME_MAX];
+
+ if (!cgroup_initialized)
+ return ECGROUPNOTINITIALIZED;
+
+ if (!proc_cgroup)
+ return ECGINVAL;
+
+ if (!info)
+ return ECGINVAL;
+
+ err = fscanf(proc_cgroup, "%s %d %d %d\n", subsys_name,
+ &hierarchy, &num_cgroups, &enabled);
+
+ if (err != 4)
+ return ECGEOF;
+
+ strncpy(info->name, subsys_name, FILENAME_MAX);
+ info->name[FILENAME_MAX-1] = '\0';
+ info->hierarchy = hierarchy;
+ info->num_cgroups = num_cgroups;
+ info->enabled = enabled;
+
+ return 0;
+}
+
+
+int cgroup_get_all_controller_begin(void **handle, struct controller_data *info)
+{
+ FILE *proc_cgroup = NULL;
+ char buf[FILENAME_MAX];
+
+ if (!cgroup_initialized)
+ return ECGROUPNOTINITIALIZED;
+
+ if (!info)
+ return ECGINVAL;
+
+ proc_cgroup = fopen("/proc/cgroups", "r");
+ if (!proc_cgroup) {
+ last_errno = errno;
+ return ECGOTHER;
+ }
+
+ if (!fgets(buf, FILENAME_MAX, proc_cgroup)) {
+ last_errno = errno;
+ return ECGOTHER;
+ }
+ *handle = proc_cgroup;
+
+ return cgroup_get_all_controller_next(handle, info);
+}