mirror of
				https://github.com/ARM-software/devlib.git
				synced 2025-10-26 03:23:19 +00:00 
			
		
		
		
	cgroups: big refactoring to make it more generic
This patch refactors the CGroup module to support multiple controllers.
The new interface is based on two main classes:
.:: Controller(kind)
    Provides a wrapper class for a CGgroup controller of "kind".
    At module initialization time, all the controller available in the
    system are enumerated and a corresponding controller class created.
    A valid reference to the Controller class for a specific CGroup
    controller can be obtained using the controller(kind) method exposed
    by the module, e.g. to get a reference to the CPUSET controller:
	cpuset = target.cgroups.controller('cpuset')
.:: CGroup(name)
    Provides a wrapper class for a CGroup of "name", i.e. a folder which
    path is "name" relative to the controller mount point.
    At module initialization time, the root control group for each
    controller is created by the controller class.
    A valid reference to a CGroup for a specific controller can be
    obtained using the cgroup(name) method exposed by the controller
    object, e.g. to get a reference to the "/LITTLE" cgroup of the
    cpuset controller:
        cpuset_littles = cpuset.cgroup('/LITTLE')
The CGroup object exposes simple get()/set() methods which allows to
read and configure all the supported attributes of a controller.
For example, to set the "cpus" of a cpuset control group:
       cpuset_littles.set(cpus=[0,1])
NOTE: the set() method accepts a variable list of parameters which name
must match a valid (writable) attribute for that controller.
Signed-off-by: Patrick Bellasi <patrick.bellasi@arm.com>
			
			
This commit is contained in:
		| @@ -22,9 +22,7 @@ from devlib.utils.misc import list_to_ranges, isiterable | |||||||
| from devlib.utils.types import boolean | from devlib.utils.types import boolean | ||||||
|  |  | ||||||
|  |  | ||||||
| class CgroupController(object): | class Controller(object): | ||||||
|  |  | ||||||
|     kind = 'cpuset' |  | ||||||
|  |  | ||||||
|     def __new__(cls, arg): |     def __new__(cls, arg): | ||||||
|         if isinstance(arg, cls): |         if isinstance(arg, cls): | ||||||
| @@ -32,98 +30,90 @@ class CgroupController(object): | |||||||
|         else: |         else: | ||||||
|             return object.__new__(cls, arg) |             return object.__new__(cls, arg) | ||||||
|  |  | ||||||
|     def __init__(self, mount_name): |     def __init__(self, kind): | ||||||
|  |         self.mount_name = 'devlib_'+kind | ||||||
|  |         self.kind = kind | ||||||
|  |         self.target = None | ||||||
|  |  | ||||||
|  |         self.logger = logging.getLogger('cgroups.'+self.kind) | ||||||
|         self.mount_point = None |         self.mount_point = None | ||||||
|         self.mount_name = mount_name |         self._cgroups = {} | ||||||
|         self.logger = logging.getLogger(self.kind) |  | ||||||
|  |  | ||||||
|     def probe(self, target): |     def probe(self, target): | ||||||
|         raise NotImplementedError() |         try: | ||||||
|  |             exists = target.execute('{} grep {} /proc/cgroups'\ | ||||||
|  |                     .format(target.busybox, self.kind)) | ||||||
|  |         except TargetError: | ||||||
|  |             return False | ||||||
|  |         return True | ||||||
|  |  | ||||||
|     def mount(self, device, mount_root): |     def mount(self, target, mount_root): | ||||||
|         self.target = device |  | ||||||
|         self.mount_point = device.path.join(mount_root, self.mount_name) |         mounted = target.list_file_systems() | ||||||
|         mounted = self.target.list_file_systems() |         if self.mount_name in [e.device for e in mounted]: | ||||||
|         if self.mount_point in [e.mount_point for e in mounted]: |             # Identify mount point if controller is already in use | ||||||
|             self.logger.debug('controller is already mounted.') |             self.mount_point = [ | ||||||
|  |                     fs.mount_point | ||||||
|  |                     for fs in mounted | ||||||
|  |                     if fs.device == self.mount_name | ||||||
|  |                 ][0] | ||||||
|         else: |         else: | ||||||
|             self.target.execute('mkdir -p {} 2>/dev/null'.format(self.mount_point), |             # Mount the controller if not already in use | ||||||
|                                 as_root=True) |             self.mount_point = target.path.join(mount_root, self.mount_name) | ||||||
|             self.target.execute('mount -t cgroup -o {} {} {}'.format(self.kind, |             target.execute('mkdir -p {} 2>/dev/null'\ | ||||||
|                                                                      self.mount_name, |                     .format(self.mount_point), as_root=True) | ||||||
|                                                                      self.mount_point), |             target.execute('mount -t cgroup -o {} {} {}'\ | ||||||
|                                 as_root=True) |                     .format(self.kind, | ||||||
|  |                             self.mount_name, | ||||||
|  |                             self.mount_point), | ||||||
|  |                             as_root=True) | ||||||
|  |  | ||||||
|  |         self.logger.info('Controller %s mounted under: %s', | ||||||
|  |             self.kind, self.mount_point) | ||||||
|  |  | ||||||
| class CpusetGroup(object): |         # Mark this contoller as available | ||||||
|  |         self.target = target | ||||||
|  |  | ||||||
|     def __init__(self, controller, name, cpus, mems): |         # Create root control group | ||||||
|         self.controller = controller |         self.cgroup('/') | ||||||
|         self.target = controller.target |  | ||||||
|         self.name = name |  | ||||||
|         if name == 'root': |  | ||||||
|             self.directory = controller.mount_point |  | ||||||
|         else: |  | ||||||
|             self.directory = self.target.path.join(controller.mount_point, name) |  | ||||||
|             self.target.execute('mkdir -p {}'.format(self.directory), as_root=True) |  | ||||||
|         self.cpus_file = self.target.path.join(self.directory, 'cpuset.cpus') |  | ||||||
|         self.mems_file = self.target.path.join(self.directory, 'cpuset.mems') |  | ||||||
|         self.tasks_file = self.target.path.join(self.directory, 'tasks') |  | ||||||
|         self.set(cpus, mems) |  | ||||||
|  |  | ||||||
|     def set(self, cpus, mems): |     def cgroup(self, name): | ||||||
|         if isiterable(cpus): |         if not self.target: | ||||||
|             cpus = list_to_ranges(cpus) |             raise RuntimeError('CGroup creation failed: {} controller not mounted'\ | ||||||
|         if isiterable(mems): |                     .format(self.kind)) | ||||||
|             mems = list_to_ranges(mems) |         if name not in self._cgroups: | ||||||
|         self.target.write_value(self.cpus_file, cpus) |             self._cgroups[name] = CGroup(self, name) | ||||||
|         self.target.write_value(self.mems_file, mems) |         return self._cgroups[name] | ||||||
|  |  | ||||||
|     def get(self): |     def exists(self, name): | ||||||
|         cpus = self.target.read_value(self.cpus_file) |         if not self.target: | ||||||
|         mems = self.target.read_value(self.mems_file) |             raise RuntimeError('CGroup creation failed: {} controller not mounted'\ | ||||||
|         return (cpus, mems) |                     .format(self.kind)) | ||||||
|  |         if name not in self._cgroups: | ||||||
|  |             self._cgroups[name] = CGroup(self, name, create=False) | ||||||
|  |         return self._cgroups[name].existe() | ||||||
|  |  | ||||||
|     def get_tasks(self): |     def list_all(self): | ||||||
|         task_ids = self.target.read_value(self.tasks_file).split() |         self.logger.debug('Listing groups for %s controller', self.kind) | ||||||
|         return map(int, task_ids) |         output = self.target.execute('{} find {} -type d'\ | ||||||
|  |                 .format(self.target.busybox, self.mount_point)) | ||||||
|     def add_tasks(self, tasks): |         cgroups = [] | ||||||
|         for tid in tasks: |         for cg in output.split('\n'): | ||||||
|             self.add_task(tid) |             cg = cg.replace(self.mount_point + '/', '/') | ||||||
|  |             cg = cg.replace(self.mount_point, '/') | ||||||
|     def add_task(self, tid): |             cg = cg.strip() | ||||||
|         self.target.write_value(self.tasks_file, tid, verify=False) |             if cg == '': | ||||||
|  |                 continue | ||||||
|  |             self.logger.debug('Populate %s cgroup: %s', self.kind, cg) | ||||||
| class CpusetController(CgroupController): |             cgroups.append(cg) | ||||||
|  |         return cgroups | ||||||
|     name = 'cpuset' |  | ||||||
|  |  | ||||||
|     def __init__(self, *args, **kwargs): |  | ||||||
|         super(CpusetController, self).__init__(*args, **kwargs) |  | ||||||
|         self.groups = {} |  | ||||||
|  |  | ||||||
|     def probe(self, target): |  | ||||||
|         return target.config.is_enabled('cpusets') |  | ||||||
|  |  | ||||||
|     def mount(self, device, mount_root): |  | ||||||
|         super(CpusetController, self).mount(device, mount_root) |  | ||||||
|         self.create_group('root', self.target.list_online_cpus(), 0) |  | ||||||
|  |  | ||||||
|     def create_group(self, name, cpus, mems): |  | ||||||
|         if not hasattr(self, 'target'): |  | ||||||
|             raise RuntimeError('Attempting to create group for unmounted controller {}'.format(self.kind)) |  | ||||||
|         if name in self.groups: |  | ||||||
|             raise ValueError('Group {} already exists'.format(name)) |  | ||||||
|         self.groups[name] = CpusetGroup(self, name, cpus, mems) |  | ||||||
|  |  | ||||||
|     def move_tasks(self, source, dest): |     def move_tasks(self, source, dest): | ||||||
|         try: |         try: | ||||||
|             source_group = self.groups[source] |             srcg = self._cgroups[source] | ||||||
|             dest_group = self.groups[dest] |             dstg = self._cgroups[dest] | ||||||
|             command = 'for task in $(cat {}); do echo $task>{}; done' |             command = 'for task in $(cat {}); do echo $task>{}; done' | ||||||
|             self.target.execute(command.format(source_group.tasks_file, dest_group.tasks_file), |             self.target.execute(command.format(srcg.tasks_file, dstg.tasks_file), | ||||||
|                                 # this will always fail as some of the tasks |                                 # this will always fail as some of the tasks | ||||||
|                                 # are kthreads that cannot be migrated, but we |                                 # are kthreads that cannot be migrated, but we | ||||||
|                                 # don't care about those, so don't check exit |                                 # don't care about those, so don't check exit | ||||||
| @@ -132,61 +122,146 @@ class CpusetController(CgroupController): | |||||||
|         except KeyError as e: |         except KeyError as e: | ||||||
|             raise ValueError('Unkown group: {}'.format(e)) |             raise ValueError('Unkown group: {}'.format(e)) | ||||||
|  |  | ||||||
|     def move_all_tasks_to(self, target_group): |     def move_all_tasks_to(self, dest): | ||||||
|         for group in self.groups: |         for cgroup in self._cgroups: | ||||||
|             if group != target_group: |             if cgroup != dest: | ||||||
|                 self.move_tasks(group, target_group) |                 self.move_tasks(cgroup, dest) | ||||||
|  |  | ||||||
|     def __getattr__(self, name): | class CGroup(object): | ||||||
|  |  | ||||||
|  |     def __init__(self, controller, name, create=True): | ||||||
|  |         self.logger = logging.getLogger('cgroups.' + controller.kind) | ||||||
|  |         self.target = controller.target | ||||||
|  |         self.controller = controller | ||||||
|  |         self.name = name | ||||||
|  |  | ||||||
|  |         # Control cgroup path | ||||||
|  |         self.directory = controller.mount_point | ||||||
|  |         if name != '/': | ||||||
|  |             self.directory = self.target.path.join(controller.mount_point, name[1:]) | ||||||
|  |  | ||||||
|  |         # Setup path for tasks file | ||||||
|  |         self.tasks_file = self.target.path.join(self.directory, 'tasks') | ||||||
|  |         self.procs_file = self.target.path.join(self.directory, 'cgroup.procs') | ||||||
|  |  | ||||||
|  |         if not create: | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         self.logger.info('Creating cgroup %s', self.directory) | ||||||
|  |         self.target.execute('[ -d {0} ] || mkdir -p {0}'\ | ||||||
|  |                 .format(self.directory), as_root=True) | ||||||
|  |  | ||||||
|  |     def exists(self): | ||||||
|         try: |         try: | ||||||
|             return self.groups[name] |             self.target.execute('[ -d {0} ]'\ | ||||||
|         except KeyError: |                 .format(self.directory)) | ||||||
|             raise AttributeError(name) |             return True | ||||||
|  |         except TargetError: | ||||||
|  |             return False | ||||||
|  |  | ||||||
|  |     def get(self): | ||||||
|  |         conf = {} | ||||||
|  |  | ||||||
|  |         logging.debug('Reading %s attributes from:', | ||||||
|  |                 self.controller.kind) | ||||||
|  |         logging.debug('  %s', | ||||||
|  |                 self.directory) | ||||||
|  |         output = self.target.execute('{} grep \'\' {}/{}.*'.format( | ||||||
|  |                     self.target.busybox, | ||||||
|  |                     self.directory, | ||||||
|  |                     self.controller.kind)) | ||||||
|  |         for res in output.split('\n'): | ||||||
|  |             if res.find(self.controller.kind) < 0: | ||||||
|  |                 continue | ||||||
|  |             res = res.split('.')[1] | ||||||
|  |             attr = res.split(':')[0] | ||||||
|  |             value = res.split(':')[1] | ||||||
|  |             conf[attr] = value | ||||||
|  |  | ||||||
|  |         return conf | ||||||
|  |  | ||||||
|  |     def set(self, **attrs): | ||||||
|  |         for idx in attrs: | ||||||
|  |             if isiterable(attrs[idx]): | ||||||
|  |                 attrs[idx] = list_to_ranges(attrs[idx]) | ||||||
|  |             # Build attribute path | ||||||
|  |             path = '{}.{}'.format(self.controller.kind, idx) | ||||||
|  |             path = self.target.path.join(self.directory, path) | ||||||
|  |  | ||||||
|  |             self.logger.debug('Set attribute [%s] to: %s"', | ||||||
|  |                     path, attrs[idx]) | ||||||
|  |  | ||||||
|  |             # Set the attribute value | ||||||
|  |             self.target.write_value(path, attrs[idx]) | ||||||
|  |  | ||||||
|  |     def get_tasks(self): | ||||||
|  |         task_ids = self.target.read_value(self.tasks_file).split() | ||||||
|  |         logging.debug('Tasks: %s', task_ids) | ||||||
|  |         return map(int, task_ids) | ||||||
|  |  | ||||||
|  |     def add_task(self, tid): | ||||||
|  |         self.target.write_value(self.tasks_file, tid, verify=False) | ||||||
|  |  | ||||||
|  |     def add_tasks(self, tasks): | ||||||
|  |         for tid in tasks: | ||||||
|  |             self.add_task(tid) | ||||||
|  |  | ||||||
|  |     def add_proc(self, pid): | ||||||
|  |         self.target.write_value(self.procs_file, pid, verify=False) | ||||||
|  |  | ||||||
| CgroupSubsystemEntry = namedtuple('CgroupSubsystemEntry', 'name hierarchy num_cgroups enabled') | CgroupSubsystemEntry = namedtuple('CgroupSubsystemEntry', 'name hierarchy num_cgroups enabled') | ||||||
|  |  | ||||||
|  |  | ||||||
| class CgroupsModule(Module): | class CgroupsModule(Module): | ||||||
|  |  | ||||||
|     name = 'cgroups' |     name = 'cgroups' | ||||||
|     controller_cls = [ |  | ||||||
|         CpusetController, |  | ||||||
|     ] |  | ||||||
|  |  | ||||||
|     cgroup_root = '/sys/fs/cgroup' |     cgroup_root = '/sys/fs/cgroup' | ||||||
|  |  | ||||||
|     @staticmethod |     @staticmethod | ||||||
|     def probe(target): |     def probe(target): | ||||||
|         return target.config.has('cgroups') and target.is_rooted |         return target.config.has('cgroups') and target.is_rooted | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|     def __init__(self, target): |     def __init__(self, target): | ||||||
|         super(CgroupsModule, self).__init__(target) |         super(CgroupsModule, self).__init__(target) | ||||||
|  |  | ||||||
|  |         self.logger = logging.getLogger('CGroups') | ||||||
|  |  | ||||||
|  |         # Initialize controllers mount point | ||||||
|         mounted = self.target.list_file_systems() |         mounted = self.target.list_file_systems() | ||||||
|         if self.cgroup_root not in [e.mount_point for e in mounted]: |         if self.cgroup_root not in [e.mount_point for e in mounted]: | ||||||
|             self.target.execute('mount -t tmpfs {} {}'.format('cgroup_root', self.cgroup_root), |             self.target.execute('mount -t tmpfs {} {}'\ | ||||||
|                                 as_root=True) |                     .format('cgroup_root', | ||||||
|  |                             self.cgroup_root), | ||||||
|  |                             as_root=True) | ||||||
|         else: |         else: | ||||||
|             self.logger.debug('cgroup_root already mounted at {}'.format(self.cgroup_root)) |             self.logger.debug('cgroup_root already mounted at %s', | ||||||
|         self.controllers = [] |                     self.cgroup_root) | ||||||
|         for cls in self.controller_cls: |  | ||||||
|             controller = cls('devlib_{}'.format(cls.name)) |         # Load list of available controllers | ||||||
|             if controller.probe(self.target): |         controllers = [] | ||||||
|                 if controller.mount_name in [e.device for e in mounted]: |         subsys = self.list_subsystems() | ||||||
|                     self.logger.debug('controller {} is already mounted.'.format(controller.kind)) |         for (n, h, c, e) in subsys: | ||||||
|                 else: |             controllers.append(n) | ||||||
|                     try: |         self.logger.info('Available controllers: %s', controllers) | ||||||
|                         controller.mount(self.target, self.cgroup_root) |  | ||||||
|                     except TargetError: |         # Initialize controllers | ||||||
|                         message = 'cgroups {} controller is not supported by the target' |         self.controllers = {} | ||||||
|                         raise TargetError(message.format(controller.kind)) |         for idx in controllers: | ||||||
|  |             controller = Controller(idx) | ||||||
|  |             self.logger.debug('Init %s controller...', controller.kind) | ||||||
|  |             if not controller.probe(self.target): | ||||||
|  |                 continue | ||||||
|  |             try: | ||||||
|  |                 controller.mount(self.target, self.cgroup_root) | ||||||
|  |             except TargetError: | ||||||
|  |                 message = 'cgroups {} controller is not supported by the target' | ||||||
|  |                 raise TargetError(message.format(controller.kind)) | ||||||
|  |             self.logger.debug('Controller %s enabled', controller.kind) | ||||||
|  |             self.controllers[idx] = controller | ||||||
|  |  | ||||||
|     def list_subsystems(self): |     def list_subsystems(self): | ||||||
|         subsystems = [] |         subsystems = [] | ||||||
|         for line in self.target.execute('cat /proc/cgroups').split('\n')[1:]: |         for line in self.target.execute('{} cat /proc/cgroups'\ | ||||||
|  |                 .format(self.target.busybox)).split('\n')[1:]: | ||||||
|             line = line.strip() |             line = line.strip() | ||||||
|             if not line or line.startswith('#'): |             if not line or line.startswith('#'): | ||||||
|                 continue |                 continue | ||||||
| @@ -198,9 +273,9 @@ class CgroupsModule(Module): | |||||||
|         return subsystems |         return subsystems | ||||||
|  |  | ||||||
|  |  | ||||||
|     def get_cgroup_controller(self, kind): |     def controller(self, kind): | ||||||
|         for controller in self.controllers: |         if kind not in self.controllers: | ||||||
|             if controller.kind == kind: |             self.logger.warning('Controller %s not available', kind) | ||||||
|                 return controller |             return None | ||||||
|         raise ValueError(kind) |         return self.controllers[kind] | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user