2022-04-22 10:33:49 -07:00
// SPDX-License-Identifier: GPL-2.0
# define _GNU_SOURCE
# include <linux/limits.h>
2022-04-22 10:33:52 -07:00
# include <sys/sysinfo.h>
# include <sys/wait.h>
2022-04-22 10:33:51 -07:00
# include <errno.h>
# include <pthread.h>
2022-04-22 10:33:49 -07:00
# include <stdio.h>
2022-04-22 10:33:51 -07:00
# include <time.h>
2022-04-22 10:33:49 -07:00
# include "../kselftest.h"
# include "cgroup_util.h"
2022-04-22 10:33:52 -07:00
enum hog_clock_type {
// Count elapsed time using the CLOCK_PROCESS_CPUTIME_ID clock.
CPU_HOG_CLOCK_PROCESS ,
// Count elapsed time using system wallclock time.
CPU_HOG_CLOCK_WALL ,
} ;
2022-04-22 10:33:51 -07:00
struct cpu_hog_func_param {
int nprocs ;
struct timespec ts ;
2022-04-22 10:33:52 -07:00
enum hog_clock_type clock_type ;
2022-04-22 10:33:51 -07:00
} ;
2022-04-22 10:33:49 -07:00
/*
* This test creates two nested cgroups with and without enabling
* the cpu controller .
*/
static int test_cpucg_subtree_control ( const char * root )
{
char * parent = NULL , * child = NULL , * parent2 = NULL , * child2 = NULL ;
int ret = KSFT_FAIL ;
// Create two nested cgroups with the cpu controller enabled.
parent = cg_name ( root , " cpucg_test_0 " ) ;
if ( ! parent )
goto cleanup ;
if ( cg_create ( parent ) )
goto cleanup ;
if ( cg_write ( parent , " cgroup.subtree_control " , " +cpu " ) )
goto cleanup ;
child = cg_name ( parent , " cpucg_test_child " ) ;
if ( ! child )
goto cleanup ;
if ( cg_create ( child ) )
goto cleanup ;
if ( cg_read_strstr ( child , " cgroup.controllers " , " cpu " ) )
goto cleanup ;
// Create two nested cgroups without enabling the cpu controller.
parent2 = cg_name ( root , " cpucg_test_1 " ) ;
if ( ! parent2 )
goto cleanup ;
if ( cg_create ( parent2 ) )
goto cleanup ;
child2 = cg_name ( parent2 , " cpucg_test_child " ) ;
if ( ! child2 )
goto cleanup ;
if ( cg_create ( child2 ) )
goto cleanup ;
if ( ! cg_read_strstr ( child2 , " cgroup.controllers " , " cpu " ) )
goto cleanup ;
ret = KSFT_PASS ;
cleanup :
cg_destroy ( child ) ;
free ( child ) ;
cg_destroy ( child2 ) ;
free ( child2 ) ;
cg_destroy ( parent ) ;
free ( parent ) ;
cg_destroy ( parent2 ) ;
free ( parent2 ) ;
return ret ;
}
2022-04-22 10:33:51 -07:00
static void * hog_cpu_thread_func ( void * arg )
{
while ( 1 )
;
return NULL ;
}
static struct timespec
timespec_sub ( const struct timespec * lhs , const struct timespec * rhs )
{
struct timespec zero = {
. tv_sec = 0 ,
. tv_nsec = 0 ,
} ;
struct timespec ret ;
if ( lhs - > tv_sec < rhs - > tv_sec )
return zero ;
ret . tv_sec = lhs - > tv_sec - rhs - > tv_sec ;
if ( lhs - > tv_nsec < rhs - > tv_nsec ) {
if ( ret . tv_sec = = 0 )
return zero ;
ret . tv_sec - - ;
ret . tv_nsec = NSEC_PER_SEC - rhs - > tv_nsec + lhs - > tv_nsec ;
} else
ret . tv_nsec = lhs - > tv_nsec - rhs - > tv_nsec ;
return ret ;
}
static int hog_cpus_timed ( const char * cgroup , void * arg )
{
const struct cpu_hog_func_param * param =
( struct cpu_hog_func_param * ) arg ;
struct timespec ts_run = param - > ts ;
struct timespec ts_remaining = ts_run ;
2022-04-22 10:33:52 -07:00
struct timespec ts_start ;
2022-04-22 10:33:51 -07:00
int i , ret ;
2022-04-22 10:33:52 -07:00
ret = clock_gettime ( CLOCK_MONOTONIC , & ts_start ) ;
if ( ret ! = 0 )
return ret ;
2022-04-22 10:33:51 -07:00
for ( i = 0 ; i < param - > nprocs ; i + + ) {
pthread_t tid ;
ret = pthread_create ( & tid , NULL , & hog_cpu_thread_func , NULL ) ;
if ( ret ! = 0 )
return ret ;
}
while ( ts_remaining . tv_sec > 0 | | ts_remaining . tv_nsec > 0 ) {
struct timespec ts_total ;
ret = nanosleep ( & ts_remaining , NULL ) ;
if ( ret & & errno ! = EINTR )
return ret ;
2022-04-22 10:33:52 -07:00
if ( param - > clock_type = = CPU_HOG_CLOCK_PROCESS ) {
ret = clock_gettime ( CLOCK_PROCESS_CPUTIME_ID , & ts_total ) ;
if ( ret ! = 0 )
return ret ;
} else {
struct timespec ts_current ;
ret = clock_gettime ( CLOCK_MONOTONIC , & ts_current ) ;
if ( ret ! = 0 )
return ret ;
ts_total = timespec_sub ( & ts_current , & ts_start ) ;
}
2022-04-22 10:33:51 -07:00
ts_remaining = timespec_sub ( & ts_run , & ts_total ) ;
}
return 0 ;
}
/*
* Creates a cpu cgroup , burns a CPU for a few quanta , and verifies that
* cpu . stat shows the expected output .
*/
static int test_cpucg_stats ( const char * root )
{
int ret = KSFT_FAIL ;
long usage_usec , user_usec , system_usec ;
long usage_seconds = 2 ;
long expected_usage_usec = usage_seconds * USEC_PER_SEC ;
char * cpucg ;
cpucg = cg_name ( root , " cpucg_test " ) ;
if ( ! cpucg )
goto cleanup ;
if ( cg_create ( cpucg ) )
goto cleanup ;
usage_usec = cg_read_key_long ( cpucg , " cpu.stat " , " usage_usec " ) ;
user_usec = cg_read_key_long ( cpucg , " cpu.stat " , " user_usec " ) ;
system_usec = cg_read_key_long ( cpucg , " cpu.stat " , " system_usec " ) ;
if ( usage_usec ! = 0 | | user_usec ! = 0 | | system_usec ! = 0 )
goto cleanup ;
struct cpu_hog_func_param param = {
. nprocs = 1 ,
. ts = {
. tv_sec = usage_seconds ,
. tv_nsec = 0 ,
} ,
2022-04-22 10:33:52 -07:00
. clock_type = CPU_HOG_CLOCK_PROCESS ,
2022-04-22 10:33:51 -07:00
} ;
if ( cg_run ( cpucg , hog_cpus_timed , ( void * ) & param ) )
goto cleanup ;
usage_usec = cg_read_key_long ( cpucg , " cpu.stat " , " usage_usec " ) ;
user_usec = cg_read_key_long ( cpucg , " cpu.stat " , " user_usec " ) ;
if ( user_usec < = 0 )
goto cleanup ;
if ( ! values_close ( usage_usec , expected_usage_usec , 1 ) )
goto cleanup ;
ret = KSFT_PASS ;
cleanup :
cg_destroy ( cpucg ) ;
free ( cpucg ) ;
return ret ;
}
2022-04-22 10:33:52 -07:00
/*
* First , this test creates the following hierarchy :
* A
* A / B cpu . weight = 50
* A / C cpu . weight = 100
* A / D cpu . weight = 150
*
* A separate process is then created for each child cgroup which spawns as
* many threads as there are cores , and hogs each CPU as much as possible
* for some time interval .
*
* Once all of the children have exited , we verify that each child cgroup
* was given proportional runtime as informed by their cpu . weight .
*/
static int test_cpucg_weight_overprovisioned ( const char * root )
{
struct child {
char * cgroup ;
pid_t pid ;
long usage ;
} ;
int ret = KSFT_FAIL , i ;
char * parent = NULL ;
struct child children [ 3 ] = { NULL } ;
long usage_seconds = 10 ;
parent = cg_name ( root , " cpucg_test_0 " ) ;
if ( ! parent )
goto cleanup ;
if ( cg_create ( parent ) )
goto cleanup ;
if ( cg_write ( parent , " cgroup.subtree_control " , " +cpu " ) )
goto cleanup ;
for ( i = 0 ; i < ARRAY_SIZE ( children ) ; i + + ) {
children [ i ] . cgroup = cg_name_indexed ( parent , " cpucg_child " , i ) ;
if ( ! children [ i ] . cgroup )
goto cleanup ;
if ( cg_create ( children [ i ] . cgroup ) )
goto cleanup ;
if ( cg_write_numeric ( children [ i ] . cgroup , " cpu.weight " ,
50 * ( i + 1 ) ) )
goto cleanup ;
}
for ( i = 0 ; i < ARRAY_SIZE ( children ) ; i + + ) {
struct cpu_hog_func_param param = {
. nprocs = get_nprocs ( ) ,
. ts = {
. tv_sec = usage_seconds ,
. tv_nsec = 0 ,
} ,
. clock_type = CPU_HOG_CLOCK_WALL ,
} ;
pid_t pid = cg_run_nowait ( children [ i ] . cgroup , hog_cpus_timed ,
( void * ) & param ) ;
if ( pid < = 0 )
goto cleanup ;
children [ i ] . pid = pid ;
}
for ( i = 0 ; i < ARRAY_SIZE ( children ) ; i + + ) {
int retcode ;
waitpid ( children [ i ] . pid , & retcode , 0 ) ;
if ( ! WIFEXITED ( retcode ) )
goto cleanup ;
if ( WEXITSTATUS ( retcode ) )
goto cleanup ;
}
for ( i = 0 ; i < ARRAY_SIZE ( children ) ; i + + )
children [ i ] . usage = cg_read_key_long ( children [ i ] . cgroup ,
" cpu.stat " , " usage_usec " ) ;
for ( i = 0 ; i < ARRAY_SIZE ( children ) - 1 ; i + + ) {
long delta ;
if ( children [ i + 1 ] . usage < = children [ i ] . usage )
goto cleanup ;
delta = children [ i + 1 ] . usage - children [ i ] . usage ;
if ( ! values_close ( delta , children [ 0 ] . usage , 35 ) )
goto cleanup ;
}
ret = KSFT_PASS ;
cleanup :
for ( i = 0 ; i < ARRAY_SIZE ( children ) ; i + + ) {
cg_destroy ( children [ i ] . cgroup ) ;
free ( children [ i ] . cgroup ) ;
}
cg_destroy ( parent ) ;
free ( parent ) ;
return ret ;
}
2022-04-22 10:33:49 -07:00
# define T(x) { x, #x }
struct cpucg_test {
int ( * fn ) ( const char * root ) ;
const char * name ;
} tests [ ] = {
T ( test_cpucg_subtree_control ) ,
2022-04-22 10:33:51 -07:00
T ( test_cpucg_stats ) ,
2022-04-22 10:33:52 -07:00
T ( test_cpucg_weight_overprovisioned ) ,
2022-04-22 10:33:49 -07:00
} ;
# undef T
int main ( int argc , char * argv [ ] )
{
char root [ PATH_MAX ] ;
int i , ret = EXIT_SUCCESS ;
if ( cg_find_unified_root ( root , sizeof ( root ) ) )
ksft_exit_skip ( " cgroup v2 isn't mounted \n " ) ;
if ( cg_read_strstr ( root , " cgroup.subtree_control " , " cpu " ) )
if ( cg_write ( root , " cgroup.subtree_control " , " +cpu " ) )
ksft_exit_skip ( " Failed to set cpu controller \n " ) ;
for ( i = 0 ; i < ARRAY_SIZE ( tests ) ; i + + ) {
switch ( tests [ i ] . fn ( root ) ) {
case KSFT_PASS :
ksft_test_result_pass ( " %s \n " , tests [ i ] . name ) ;
break ;
case KSFT_SKIP :
ksft_test_result_skip ( " %s \n " , tests [ i ] . name ) ;
break ;
default :
ret = EXIT_FAILURE ;
ksft_test_result_fail ( " %s \n " , tests [ i ] . name ) ;
break ;
}
}
return ret ;
}