kpatch/kpatch-build/gcc-plugins/ppc64le-plugin.c
Josh Poimboeuf 63a94b8323 gcc-plugin: create a ppc64le GCC plugin which inserts nops after local calls
This is in response to an upstream discussion for the following patch:

  https://lkml.kernel.org/r/1508217523-18885-1-git-send-email-kamalesh@linux.vnet.ibm.com

This should hopefully make it a lot easier for the ppc64le kernel module
code to support klp relocations.

The gcc-common.h and gcc-generate-rtl-pass.h header files are copied
from the upstream Linux source tree.

Signed-off-by: Josh Poimboeuf <jpoimboe@redhat.com>
2017-11-10 09:17:00 -06:00

87 lines
2.2 KiB
C

#include "gcc-common.h"
#include <error.h>
#define PLUGIN_NAME "ppc64le-plugin"
int plugin_is_GPL_compatible;
struct plugin_info plugin_info = {
.version = "1",
.help = PLUGIN_NAME ": insert nops after local calls\n",
};
static unsigned int ppc64le_plugin_execute(void)
{
rtx_insn *insn;
int code;
const char *name;
static int nonlocal_code = -1, local_code = -1,
value_nonlocal_code = -1, value_local_code = -1;
static bool initialized = false;
if (initialized)
goto found;
/* Find the rs6000.md code numbers for local and non-local calls */
initialized = true;
for (code = 0; code < 1000; code++) {
name = get_insn_name(code);
if (!name)
continue;
if (!strcmp(name , "*call_local_aixdi"))
local_code = code;
else if (!strcmp(name , "*call_nonlocal_aixdi"))
nonlocal_code = code;
else if (!strcmp(name, "*call_value_local_aixdi"))
value_local_code = code;
else if (!strcmp(name, "*call_value_nonlocal_aixdi"))
value_nonlocal_code = code;
if (nonlocal_code != -1 && local_code != -1 &&
value_nonlocal_code != -1 && value_local_code != -1)
goto found;
}
found:
if (nonlocal_code == -1 || local_code == -1 ||
value_nonlocal_code == -1 || value_local_code == -1) {
fprintf(stderr, PLUGIN_NAME ": can't find call instruction codes");
return 1;
}
/* Convert local calls to non-local */
for (insn = get_insns(); insn; insn = NEXT_INSN(insn)) {
if (GET_CODE(insn) == CALL_INSN) {
if (INSN_CODE(insn) == local_code)
INSN_CODE(insn) = nonlocal_code;
else if (INSN_CODE(insn) == value_local_code)
INSN_CODE(insn) = value_nonlocal_code;
}
}
return 0;
}
#define PASS_NAME ppc64le_plugin
#define NO_GATE
#include "gcc-generate-rtl-pass.h"
int plugin_init(struct plugin_name_args *plugin_info,
struct plugin_gcc_version *version)
{
const char * const plugin_name = plugin_info->base_name;
PASS_INFO(ppc64le_plugin, "dwarf2", 1, PASS_POS_INSERT_BEFORE);
if (!plugin_default_version_check(version, &gcc_version))
error(1, 0, PLUGIN_NAME ": incompatible gcc/plugin versions");
register_callback(plugin_name, PLUGIN_INFO, NULL, &plugin_info);
register_callback(plugin_name, PLUGIN_PASS_MANAGER_SETUP, NULL,
&ppc64le_plugin_pass_info);
return 0;
}