d8f9da2404
In places where the equivalent was already being done, i.e.: free(a); a = NULL; And in placs where struct members are being freed so that if we have some erroneous reference to its struct, then accesses to freed members will result in segfaults, which we can detect faster than use after free to areas that may still have something seemingly valid. Cc: Adrian Hunter <adrian.hunter@intel.com> Cc: Jiri Olsa <jolsa@kernel.org> Cc: Namhyung Kim <namhyung@kernel.org> Link: https://lkml.kernel.org/n/tip-jatyoofo5boc1bsvoig6bb6i@git.kernel.org Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
142 lines
3.8 KiB
C
142 lines
3.8 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
#include <stdlib.h>
|
|
#include <sys/types.h>
|
|
#include <unistd.h>
|
|
#include <sys/prctl.h>
|
|
#include "tests.h"
|
|
#include "thread_map.h"
|
|
#include "debug.h"
|
|
#include <linux/zalloc.h>
|
|
|
|
#define NAME (const char *) "perf"
|
|
#define NAMEUL (unsigned long) NAME
|
|
|
|
int test__thread_map(struct test *test __maybe_unused, int subtest __maybe_unused)
|
|
{
|
|
struct thread_map *map;
|
|
|
|
TEST_ASSERT_VAL("failed to set process name",
|
|
!prctl(PR_SET_NAME, NAMEUL, 0, 0, 0));
|
|
|
|
/* test map on current pid */
|
|
map = thread_map__new_by_pid(getpid());
|
|
TEST_ASSERT_VAL("failed to alloc map", map);
|
|
|
|
thread_map__read_comms(map);
|
|
|
|
TEST_ASSERT_VAL("wrong nr", map->nr == 1);
|
|
TEST_ASSERT_VAL("wrong pid",
|
|
thread_map__pid(map, 0) == getpid());
|
|
TEST_ASSERT_VAL("wrong comm",
|
|
thread_map__comm(map, 0) &&
|
|
!strcmp(thread_map__comm(map, 0), NAME));
|
|
TEST_ASSERT_VAL("wrong refcnt",
|
|
refcount_read(&map->refcnt) == 1);
|
|
thread_map__put(map);
|
|
|
|
/* test dummy pid */
|
|
map = thread_map__new_dummy();
|
|
TEST_ASSERT_VAL("failed to alloc map", map);
|
|
|
|
thread_map__read_comms(map);
|
|
|
|
TEST_ASSERT_VAL("wrong nr", map->nr == 1);
|
|
TEST_ASSERT_VAL("wrong pid", thread_map__pid(map, 0) == -1);
|
|
TEST_ASSERT_VAL("wrong comm",
|
|
thread_map__comm(map, 0) &&
|
|
!strcmp(thread_map__comm(map, 0), "dummy"));
|
|
TEST_ASSERT_VAL("wrong refcnt",
|
|
refcount_read(&map->refcnt) == 1);
|
|
thread_map__put(map);
|
|
return 0;
|
|
}
|
|
|
|
static int process_event(struct perf_tool *tool __maybe_unused,
|
|
union perf_event *event,
|
|
struct perf_sample *sample __maybe_unused,
|
|
struct machine *machine __maybe_unused)
|
|
{
|
|
struct thread_map_event *map = &event->thread_map;
|
|
struct thread_map *threads;
|
|
|
|
TEST_ASSERT_VAL("wrong nr", map->nr == 1);
|
|
TEST_ASSERT_VAL("wrong pid", map->entries[0].pid == (u64) getpid());
|
|
TEST_ASSERT_VAL("wrong comm", !strcmp(map->entries[0].comm, NAME));
|
|
|
|
threads = thread_map__new_event(&event->thread_map);
|
|
TEST_ASSERT_VAL("failed to alloc map", threads);
|
|
|
|
TEST_ASSERT_VAL("wrong nr", threads->nr == 1);
|
|
TEST_ASSERT_VAL("wrong pid",
|
|
thread_map__pid(threads, 0) == getpid());
|
|
TEST_ASSERT_VAL("wrong comm",
|
|
thread_map__comm(threads, 0) &&
|
|
!strcmp(thread_map__comm(threads, 0), NAME));
|
|
TEST_ASSERT_VAL("wrong refcnt",
|
|
refcount_read(&threads->refcnt) == 1);
|
|
thread_map__put(threads);
|
|
return 0;
|
|
}
|
|
|
|
int test__thread_map_synthesize(struct test *test __maybe_unused, int subtest __maybe_unused)
|
|
{
|
|
struct thread_map *threads;
|
|
|
|
TEST_ASSERT_VAL("failed to set process name",
|
|
!prctl(PR_SET_NAME, NAMEUL, 0, 0, 0));
|
|
|
|
/* test map on current pid */
|
|
threads = thread_map__new_by_pid(getpid());
|
|
TEST_ASSERT_VAL("failed to alloc map", threads);
|
|
|
|
thread_map__read_comms(threads);
|
|
|
|
TEST_ASSERT_VAL("failed to synthesize map",
|
|
!perf_event__synthesize_thread_map2(NULL, threads, process_event, NULL));
|
|
|
|
return 0;
|
|
}
|
|
|
|
int test__thread_map_remove(struct test *test __maybe_unused, int subtest __maybe_unused)
|
|
{
|
|
struct thread_map *threads;
|
|
char *str;
|
|
int i;
|
|
|
|
TEST_ASSERT_VAL("failed to allocate map string",
|
|
asprintf(&str, "%d,%d", getpid(), getppid()) >= 0);
|
|
|
|
threads = thread_map__new_str(str, NULL, 0, false);
|
|
|
|
TEST_ASSERT_VAL("failed to allocate thread_map",
|
|
threads);
|
|
|
|
if (verbose > 0)
|
|
thread_map__fprintf(threads, stderr);
|
|
|
|
TEST_ASSERT_VAL("failed to remove thread",
|
|
!thread_map__remove(threads, 0));
|
|
|
|
TEST_ASSERT_VAL("thread_map count != 1", threads->nr == 1);
|
|
|
|
if (verbose > 0)
|
|
thread_map__fprintf(threads, stderr);
|
|
|
|
TEST_ASSERT_VAL("failed to remove thread",
|
|
!thread_map__remove(threads, 0));
|
|
|
|
TEST_ASSERT_VAL("thread_map count != 0", threads->nr == 0);
|
|
|
|
if (verbose > 0)
|
|
thread_map__fprintf(threads, stderr);
|
|
|
|
TEST_ASSERT_VAL("failed to not remove thread",
|
|
thread_map__remove(threads, 0));
|
|
|
|
for (i = 0; i < threads->nr; i++)
|
|
zfree(&threads->map[i].comm);
|
|
|
|
free(threads);
|
|
return 0;
|
|
}
|