18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 28c2ecf20Sopenharmony_ci#include <linux/stringify.h> 38c2ecf20Sopenharmony_ci#include <sys/types.h> 48c2ecf20Sopenharmony_ci#include <sys/stat.h> 58c2ecf20Sopenharmony_ci#include <fcntl.h> 68c2ecf20Sopenharmony_ci#include <stdio.h> 78c2ecf20Sopenharmony_ci#include <stdlib.h> 88c2ecf20Sopenharmony_ci#include <string.h> 98c2ecf20Sopenharmony_ci#include "fs.h" 108c2ecf20Sopenharmony_ci 118c2ecf20Sopenharmony_ciint cgroupfs_find_mountpoint(char *buf, size_t maxlen, const char *subsys) 128c2ecf20Sopenharmony_ci{ 138c2ecf20Sopenharmony_ci FILE *fp; 148c2ecf20Sopenharmony_ci char mountpoint[PATH_MAX + 1], tokens[PATH_MAX + 1], type[PATH_MAX + 1]; 158c2ecf20Sopenharmony_ci char path_v1[PATH_MAX + 1], path_v2[PATH_MAX + 2], *path; 168c2ecf20Sopenharmony_ci char *token, *saved_ptr = NULL; 178c2ecf20Sopenharmony_ci 188c2ecf20Sopenharmony_ci fp = fopen("/proc/mounts", "r"); 198c2ecf20Sopenharmony_ci if (!fp) 208c2ecf20Sopenharmony_ci return -1; 218c2ecf20Sopenharmony_ci 228c2ecf20Sopenharmony_ci /* 238c2ecf20Sopenharmony_ci * in order to handle split hierarchy, we need to scan /proc/mounts 248c2ecf20Sopenharmony_ci * and inspect every cgroupfs mount point to find one that has 258c2ecf20Sopenharmony_ci * perf_event subsystem 268c2ecf20Sopenharmony_ci */ 278c2ecf20Sopenharmony_ci path_v1[0] = '\0'; 288c2ecf20Sopenharmony_ci path_v2[0] = '\0'; 298c2ecf20Sopenharmony_ci 308c2ecf20Sopenharmony_ci while (fscanf(fp, "%*s %"__stringify(PATH_MAX)"s %"__stringify(PATH_MAX)"s %" 318c2ecf20Sopenharmony_ci __stringify(PATH_MAX)"s %*d %*d\n", 328c2ecf20Sopenharmony_ci mountpoint, type, tokens) == 3) { 338c2ecf20Sopenharmony_ci 348c2ecf20Sopenharmony_ci if (!path_v1[0] && !strcmp(type, "cgroup")) { 358c2ecf20Sopenharmony_ci 368c2ecf20Sopenharmony_ci token = strtok_r(tokens, ",", &saved_ptr); 378c2ecf20Sopenharmony_ci 388c2ecf20Sopenharmony_ci while (token != NULL) { 398c2ecf20Sopenharmony_ci if (subsys && !strcmp(token, subsys)) { 408c2ecf20Sopenharmony_ci strcpy(path_v1, mountpoint); 418c2ecf20Sopenharmony_ci break; 428c2ecf20Sopenharmony_ci } 438c2ecf20Sopenharmony_ci token = strtok_r(NULL, ",", &saved_ptr); 448c2ecf20Sopenharmony_ci } 458c2ecf20Sopenharmony_ci } 468c2ecf20Sopenharmony_ci 478c2ecf20Sopenharmony_ci if (!path_v2[0] && !strcmp(type, "cgroup2")) 488c2ecf20Sopenharmony_ci strcpy(path_v2, mountpoint); 498c2ecf20Sopenharmony_ci 508c2ecf20Sopenharmony_ci if (path_v1[0] && path_v2[0]) 518c2ecf20Sopenharmony_ci break; 528c2ecf20Sopenharmony_ci } 538c2ecf20Sopenharmony_ci fclose(fp); 548c2ecf20Sopenharmony_ci 558c2ecf20Sopenharmony_ci if (path_v1[0]) 568c2ecf20Sopenharmony_ci path = path_v1; 578c2ecf20Sopenharmony_ci else if (path_v2[0]) 588c2ecf20Sopenharmony_ci path = path_v2; 598c2ecf20Sopenharmony_ci else 608c2ecf20Sopenharmony_ci return -1; 618c2ecf20Sopenharmony_ci 628c2ecf20Sopenharmony_ci if (strlen(path) < maxlen) { 638c2ecf20Sopenharmony_ci strcpy(buf, path); 648c2ecf20Sopenharmony_ci return 0; 658c2ecf20Sopenharmony_ci } 668c2ecf20Sopenharmony_ci return -1; 678c2ecf20Sopenharmony_ci} 68