1936f094e1
We are using 'dscr_insn' as a label in inline asm to identify if a
SIGILL was generated by the mtspr instruction at that point. However,
with inline assembly, the compiler is still free to duplicate the asm
statement for optimization purposes, which results in the label being
defined twice with the error:
/tmp/ccerQCql.s:874: Error: symbol `dscr_insn' is already defined
With different compiler versions, we may also see:
/tmp/ccJzLDlN.o:(.toc+0x0): undefined reference to `dscr_insn'
Remove the use of the label in the inline assembly. Instead, just look
for the offending instruction in the signal handler.
Fixes: d2bf793237
("selftests/powerpc: Add test to verify rfi flush across a system call")
Reported-by: Breno Leitao <leitao@debian.org>
Signed-off-by: Naveen N. Rao <naveen.n.rao@linux.vnet.ibm.com>
Tested-by: Breno Leitao <leitao@debian.org>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
276 lines
4.9 KiB
C
276 lines
4.9 KiB
C
/*
|
|
* Copyright 2013-2015, Michael Ellerman, IBM Corp.
|
|
* Licensed under GPLv2.
|
|
*/
|
|
|
|
#define _GNU_SOURCE /* For CPU_ZERO etc. */
|
|
|
|
#include <elf.h>
|
|
#include <errno.h>
|
|
#include <fcntl.h>
|
|
#include <link.h>
|
|
#include <sched.h>
|
|
#include <signal.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <sys/ioctl.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/types.h>
|
|
#include <sys/utsname.h>
|
|
#include <unistd.h>
|
|
#include <asm/unistd.h>
|
|
#include <linux/limits.h>
|
|
|
|
#include "utils.h"
|
|
|
|
static char auxv[4096];
|
|
|
|
int read_auxv(char *buf, ssize_t buf_size)
|
|
{
|
|
ssize_t num;
|
|
int rc, fd;
|
|
|
|
fd = open("/proc/self/auxv", O_RDONLY);
|
|
if (fd == -1) {
|
|
perror("open");
|
|
return -errno;
|
|
}
|
|
|
|
num = read(fd, buf, buf_size);
|
|
if (num < 0) {
|
|
perror("read");
|
|
rc = -EIO;
|
|
goto out;
|
|
}
|
|
|
|
if (num > buf_size) {
|
|
printf("overflowed auxv buffer\n");
|
|
rc = -EOVERFLOW;
|
|
goto out;
|
|
}
|
|
|
|
rc = 0;
|
|
out:
|
|
close(fd);
|
|
return rc;
|
|
}
|
|
|
|
void *find_auxv_entry(int type, char *auxv)
|
|
{
|
|
ElfW(auxv_t) *p;
|
|
|
|
p = (ElfW(auxv_t) *)auxv;
|
|
|
|
while (p->a_type != AT_NULL) {
|
|
if (p->a_type == type)
|
|
return p;
|
|
|
|
p++;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
void *get_auxv_entry(int type)
|
|
{
|
|
ElfW(auxv_t) *p;
|
|
|
|
if (read_auxv(auxv, sizeof(auxv)))
|
|
return NULL;
|
|
|
|
p = find_auxv_entry(type, auxv);
|
|
if (p)
|
|
return (void *)p->a_un.a_val;
|
|
|
|
return NULL;
|
|
}
|
|
|
|
int pick_online_cpu(void)
|
|
{
|
|
cpu_set_t mask;
|
|
int cpu;
|
|
|
|
CPU_ZERO(&mask);
|
|
|
|
if (sched_getaffinity(0, sizeof(mask), &mask)) {
|
|
perror("sched_getaffinity");
|
|
return -1;
|
|
}
|
|
|
|
/* We prefer a primary thread, but skip 0 */
|
|
for (cpu = 8; cpu < CPU_SETSIZE; cpu += 8)
|
|
if (CPU_ISSET(cpu, &mask))
|
|
return cpu;
|
|
|
|
/* Search for anything, but in reverse */
|
|
for (cpu = CPU_SETSIZE - 1; cpu >= 0; cpu--)
|
|
if (CPU_ISSET(cpu, &mask))
|
|
return cpu;
|
|
|
|
printf("No cpus in affinity mask?!\n");
|
|
return -1;
|
|
}
|
|
|
|
bool is_ppc64le(void)
|
|
{
|
|
struct utsname uts;
|
|
int rc;
|
|
|
|
errno = 0;
|
|
rc = uname(&uts);
|
|
if (rc) {
|
|
perror("uname");
|
|
return false;
|
|
}
|
|
|
|
return strcmp(uts.machine, "ppc64le") == 0;
|
|
}
|
|
|
|
int read_debugfs_file(char *debugfs_file, int *result)
|
|
{
|
|
int rc = -1, fd;
|
|
char path[PATH_MAX];
|
|
char value[16];
|
|
|
|
strcpy(path, "/sys/kernel/debug/");
|
|
strncat(path, debugfs_file, PATH_MAX - strlen(path) - 1);
|
|
|
|
if ((fd = open(path, O_RDONLY)) < 0)
|
|
return rc;
|
|
|
|
if ((rc = read(fd, value, sizeof(value))) < 0)
|
|
return rc;
|
|
|
|
value[15] = 0;
|
|
*result = atoi(value);
|
|
close(fd);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int write_debugfs_file(char *debugfs_file, int result)
|
|
{
|
|
int rc = -1, fd;
|
|
char path[PATH_MAX];
|
|
char value[16];
|
|
|
|
strcpy(path, "/sys/kernel/debug/");
|
|
strncat(path, debugfs_file, PATH_MAX - strlen(path) - 1);
|
|
|
|
if ((fd = open(path, O_WRONLY)) < 0)
|
|
return rc;
|
|
|
|
snprintf(value, 16, "%d", result);
|
|
|
|
if ((rc = write(fd, value, strlen(value))) < 0)
|
|
return rc;
|
|
|
|
close(fd);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static long perf_event_open(struct perf_event_attr *hw_event, pid_t pid,
|
|
int cpu, int group_fd, unsigned long flags)
|
|
{
|
|
return syscall(__NR_perf_event_open, hw_event, pid, cpu,
|
|
group_fd, flags);
|
|
}
|
|
|
|
static void perf_event_attr_init(struct perf_event_attr *event_attr,
|
|
unsigned int type,
|
|
unsigned long config)
|
|
{
|
|
memset(event_attr, 0, sizeof(*event_attr));
|
|
|
|
event_attr->type = type;
|
|
event_attr->size = sizeof(struct perf_event_attr);
|
|
event_attr->config = config;
|
|
event_attr->read_format = PERF_FORMAT_GROUP;
|
|
event_attr->disabled = 1;
|
|
event_attr->exclude_kernel = 1;
|
|
event_attr->exclude_hv = 1;
|
|
event_attr->exclude_guest = 1;
|
|
}
|
|
|
|
int perf_event_open_counter(unsigned int type,
|
|
unsigned long config, int group_fd)
|
|
{
|
|
int fd;
|
|
struct perf_event_attr event_attr;
|
|
|
|
perf_event_attr_init(&event_attr, type, config);
|
|
|
|
fd = perf_event_open(&event_attr, 0, -1, group_fd, 0);
|
|
|
|
if (fd < 0)
|
|
perror("perf_event_open() failed");
|
|
|
|
return fd;
|
|
}
|
|
|
|
int perf_event_enable(int fd)
|
|
{
|
|
if (ioctl(fd, PERF_EVENT_IOC_ENABLE, PERF_IOC_FLAG_GROUP) == -1) {
|
|
perror("error while enabling perf events");
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int perf_event_disable(int fd)
|
|
{
|
|
if (ioctl(fd, PERF_EVENT_IOC_DISABLE, PERF_IOC_FLAG_GROUP) == -1) {
|
|
perror("error disabling perf events");
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int perf_event_reset(int fd)
|
|
{
|
|
if (ioctl(fd, PERF_EVENT_IOC_RESET, PERF_IOC_FLAG_GROUP) == -1) {
|
|
perror("error resetting perf events");
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void sigill_handler(int signr, siginfo_t *info, void *unused)
|
|
{
|
|
static int warned = 0;
|
|
ucontext_t *ctx = (ucontext_t *)unused;
|
|
unsigned long *pc = &UCONTEXT_NIA(ctx);
|
|
|
|
/* mtspr 3,RS to check for move to DSCR below */
|
|
if ((*((unsigned int *)*pc) & 0xfc1fffff) == 0x7c0303a6) {
|
|
if (!warned++)
|
|
printf("WARNING: Skipping over dscr setup. Consider running 'ppc64_cpu --dscr=1' manually.\n");
|
|
*pc += 4;
|
|
} else {
|
|
printf("SIGILL at %p\n", pc);
|
|
abort();
|
|
}
|
|
}
|
|
|
|
void set_dscr(unsigned long val)
|
|
{
|
|
static int init = 0;
|
|
struct sigaction sa;
|
|
|
|
if (!init) {
|
|
memset(&sa, 0, sizeof(sa));
|
|
sa.sa_sigaction = sigill_handler;
|
|
sa.sa_flags = SA_SIGINFO;
|
|
if (sigaction(SIGILL, &sa, NULL))
|
|
perror("sigill_handler");
|
|
init = 1;
|
|
}
|
|
|
|
asm volatile("mtspr %1,%0" : : "r" (val), "i" (SPRN_DSCR));
|
|
}
|