162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
262306a36Sopenharmony_ci#ifndef _GNU_SOURCE
362306a36Sopenharmony_ci# define _GNU_SOURCE
462306a36Sopenharmony_ci#endif
562306a36Sopenharmony_ci
662306a36Sopenharmony_ci#include <stdio.h>
762306a36Sopenharmony_ci#include <stdlib.h>
862306a36Sopenharmony_ci#include <string.h>
962306a36Sopenharmony_ci#include <linux/string.h>
1062306a36Sopenharmony_ci#include <errno.h>
1162306a36Sopenharmony_ci#include <unistd.h>
1262306a36Sopenharmony_ci#include "fs.h"
1362306a36Sopenharmony_ci
1462306a36Sopenharmony_ci#include "tracing_path.h"
1562306a36Sopenharmony_ci
1662306a36Sopenharmony_cistatic char tracing_path[PATH_MAX]        = "/sys/kernel/tracing";
1762306a36Sopenharmony_ci
1862306a36Sopenharmony_cistatic void __tracing_path_set(const char *tracing, const char *mountpoint)
1962306a36Sopenharmony_ci{
2062306a36Sopenharmony_ci	snprintf(tracing_path, sizeof(tracing_path), "%s/%s",
2162306a36Sopenharmony_ci		 mountpoint, tracing);
2262306a36Sopenharmony_ci}
2362306a36Sopenharmony_ci
2462306a36Sopenharmony_cistatic const char *tracing_path_tracefs_mount(void)
2562306a36Sopenharmony_ci{
2662306a36Sopenharmony_ci	const char *mnt;
2762306a36Sopenharmony_ci
2862306a36Sopenharmony_ci	mnt = tracefs__mount();
2962306a36Sopenharmony_ci	if (!mnt)
3062306a36Sopenharmony_ci		return NULL;
3162306a36Sopenharmony_ci
3262306a36Sopenharmony_ci	__tracing_path_set("", mnt);
3362306a36Sopenharmony_ci
3462306a36Sopenharmony_ci	return tracing_path;
3562306a36Sopenharmony_ci}
3662306a36Sopenharmony_ci
3762306a36Sopenharmony_cistatic const char *tracing_path_debugfs_mount(void)
3862306a36Sopenharmony_ci{
3962306a36Sopenharmony_ci	const char *mnt;
4062306a36Sopenharmony_ci
4162306a36Sopenharmony_ci	mnt = debugfs__mount();
4262306a36Sopenharmony_ci	if (!mnt)
4362306a36Sopenharmony_ci		return NULL;
4462306a36Sopenharmony_ci
4562306a36Sopenharmony_ci	__tracing_path_set("tracing/", mnt);
4662306a36Sopenharmony_ci
4762306a36Sopenharmony_ci	return tracing_path;
4862306a36Sopenharmony_ci}
4962306a36Sopenharmony_ci
5062306a36Sopenharmony_ciconst char *tracing_path_mount(void)
5162306a36Sopenharmony_ci{
5262306a36Sopenharmony_ci	const char *mnt;
5362306a36Sopenharmony_ci
5462306a36Sopenharmony_ci	mnt = tracing_path_tracefs_mount();
5562306a36Sopenharmony_ci	if (mnt)
5662306a36Sopenharmony_ci		return mnt;
5762306a36Sopenharmony_ci
5862306a36Sopenharmony_ci	mnt = tracing_path_debugfs_mount();
5962306a36Sopenharmony_ci
6062306a36Sopenharmony_ci	return mnt;
6162306a36Sopenharmony_ci}
6262306a36Sopenharmony_ci
6362306a36Sopenharmony_civoid tracing_path_set(const char *mntpt)
6462306a36Sopenharmony_ci{
6562306a36Sopenharmony_ci	__tracing_path_set("tracing/", mntpt);
6662306a36Sopenharmony_ci}
6762306a36Sopenharmony_ci
6862306a36Sopenharmony_cichar *get_tracing_file(const char *name)
6962306a36Sopenharmony_ci{
7062306a36Sopenharmony_ci	char *file;
7162306a36Sopenharmony_ci
7262306a36Sopenharmony_ci	if (asprintf(&file, "%s/%s", tracing_path_mount(), name) < 0)
7362306a36Sopenharmony_ci		return NULL;
7462306a36Sopenharmony_ci
7562306a36Sopenharmony_ci	return file;
7662306a36Sopenharmony_ci}
7762306a36Sopenharmony_ci
7862306a36Sopenharmony_civoid put_tracing_file(char *file)
7962306a36Sopenharmony_ci{
8062306a36Sopenharmony_ci	free(file);
8162306a36Sopenharmony_ci}
8262306a36Sopenharmony_ci
8362306a36Sopenharmony_cichar *get_events_file(const char *name)
8462306a36Sopenharmony_ci{
8562306a36Sopenharmony_ci	char *file;
8662306a36Sopenharmony_ci
8762306a36Sopenharmony_ci	if (asprintf(&file, "%s/events/%s", tracing_path_mount(), name) < 0)
8862306a36Sopenharmony_ci		return NULL;
8962306a36Sopenharmony_ci
9062306a36Sopenharmony_ci	return file;
9162306a36Sopenharmony_ci}
9262306a36Sopenharmony_ci
9362306a36Sopenharmony_civoid put_events_file(char *file)
9462306a36Sopenharmony_ci{
9562306a36Sopenharmony_ci	free(file);
9662306a36Sopenharmony_ci}
9762306a36Sopenharmony_ci
9862306a36Sopenharmony_ciDIR *tracing_events__opendir(void)
9962306a36Sopenharmony_ci{
10062306a36Sopenharmony_ci	DIR *dir = NULL;
10162306a36Sopenharmony_ci	char *path = get_tracing_file("events");
10262306a36Sopenharmony_ci
10362306a36Sopenharmony_ci	if (path) {
10462306a36Sopenharmony_ci		dir = opendir(path);
10562306a36Sopenharmony_ci		put_events_file(path);
10662306a36Sopenharmony_ci	}
10762306a36Sopenharmony_ci
10862306a36Sopenharmony_ci	return dir;
10962306a36Sopenharmony_ci}
11062306a36Sopenharmony_ci
11162306a36Sopenharmony_ciint tracing_events__scandir_alphasort(struct dirent ***namelist)
11262306a36Sopenharmony_ci{
11362306a36Sopenharmony_ci	char *path = get_tracing_file("events");
11462306a36Sopenharmony_ci	int ret;
11562306a36Sopenharmony_ci
11662306a36Sopenharmony_ci	if (!path) {
11762306a36Sopenharmony_ci		*namelist = NULL;
11862306a36Sopenharmony_ci		return 0;
11962306a36Sopenharmony_ci	}
12062306a36Sopenharmony_ci
12162306a36Sopenharmony_ci	ret = scandir(path, namelist, NULL, alphasort);
12262306a36Sopenharmony_ci	put_events_file(path);
12362306a36Sopenharmony_ci
12462306a36Sopenharmony_ci	return ret;
12562306a36Sopenharmony_ci}
12662306a36Sopenharmony_ci
12762306a36Sopenharmony_ciint tracing_path__strerror_open_tp(int err, char *buf, size_t size,
12862306a36Sopenharmony_ci				   const char *sys, const char *name)
12962306a36Sopenharmony_ci{
13062306a36Sopenharmony_ci	char sbuf[128];
13162306a36Sopenharmony_ci	char filename[PATH_MAX];
13262306a36Sopenharmony_ci
13362306a36Sopenharmony_ci	snprintf(filename, PATH_MAX, "%s/%s", sys, name ?: "*");
13462306a36Sopenharmony_ci
13562306a36Sopenharmony_ci	switch (err) {
13662306a36Sopenharmony_ci	case ENOENT:
13762306a36Sopenharmony_ci		/*
13862306a36Sopenharmony_ci		 * We will get here if we can't find the tracepoint, but one of
13962306a36Sopenharmony_ci		 * debugfs or tracefs is configured, which means you probably
14062306a36Sopenharmony_ci		 * want some tracepoint which wasn't compiled in your kernel.
14162306a36Sopenharmony_ci		 * - jirka
14262306a36Sopenharmony_ci		 */
14362306a36Sopenharmony_ci		if (debugfs__configured() || tracefs__configured()) {
14462306a36Sopenharmony_ci			/* sdt markers */
14562306a36Sopenharmony_ci			if (!strncmp(filename, "sdt_", 4)) {
14662306a36Sopenharmony_ci				snprintf(buf, size,
14762306a36Sopenharmony_ci					"Error:\tFile %s/events/%s not found.\n"
14862306a36Sopenharmony_ci					"Hint:\tSDT event cannot be directly recorded on.\n"
14962306a36Sopenharmony_ci					"\tPlease first use 'perf probe %s:%s' before recording it.\n",
15062306a36Sopenharmony_ci					tracing_path, filename, sys, name);
15162306a36Sopenharmony_ci			} else {
15262306a36Sopenharmony_ci				snprintf(buf, size,
15362306a36Sopenharmony_ci					 "Error:\tFile %s/events/%s not found.\n"
15462306a36Sopenharmony_ci					 "Hint:\tPerhaps this kernel misses some CONFIG_ setting to enable this feature?.\n",
15562306a36Sopenharmony_ci					 tracing_path, filename);
15662306a36Sopenharmony_ci			}
15762306a36Sopenharmony_ci			break;
15862306a36Sopenharmony_ci		}
15962306a36Sopenharmony_ci		snprintf(buf, size, "%s",
16062306a36Sopenharmony_ci			 "Error:\tUnable to find debugfs/tracefs\n"
16162306a36Sopenharmony_ci			 "Hint:\tWas your kernel compiled with debugfs/tracefs support?\n"
16262306a36Sopenharmony_ci			 "Hint:\tIs the debugfs/tracefs filesystem mounted?\n"
16362306a36Sopenharmony_ci			 "Hint:\tTry 'sudo mount -t debugfs nodev /sys/kernel/debug'");
16462306a36Sopenharmony_ci		break;
16562306a36Sopenharmony_ci	case EACCES: {
16662306a36Sopenharmony_ci		snprintf(buf, size,
16762306a36Sopenharmony_ci			 "Error:\tNo permissions to read %s/events/%s\n"
16862306a36Sopenharmony_ci			 "Hint:\tTry 'sudo mount -o remount,mode=755 %s'\n",
16962306a36Sopenharmony_ci			 tracing_path, filename, tracing_path_mount());
17062306a36Sopenharmony_ci	}
17162306a36Sopenharmony_ci		break;
17262306a36Sopenharmony_ci	default:
17362306a36Sopenharmony_ci		snprintf(buf, size, "%s", str_error_r(err, sbuf, sizeof(sbuf)));
17462306a36Sopenharmony_ci		break;
17562306a36Sopenharmony_ci	}
17662306a36Sopenharmony_ci
17762306a36Sopenharmony_ci	return 0;
17862306a36Sopenharmony_ci}
179