1/* 2 * livepatch.c - x86-specific Kernel Live Patching Core 3 * 4 * Copyright (C) 2014 Seth Jennings <sjenning@redhat.com> 5 * Copyright (C) 2014 SUSE 6 * 7 * This program is free software; you can redistribute it and/or 8 * modify it under the terms of the GNU General Public License 9 * as published by the Free Software Foundation; either version 2 10 * of the License, or (at your option) any later version. 11 * 12 * This program is distributed in the hope that it will be useful, 13 * but WITHOUT ANY WARRANTY; without even the implied warranty of 14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 15 * GNU General Public License for more details. 16 * 17 * You should have received a copy of the GNU General Public License 18 * along with this program; if not, see <http://www.gnu.org/licenses/>. 19 */ 20 21#include <linux/module.h> 22#include <linux/uaccess.h> 23#include <asm/cacheflush.h> 24#include <asm/page_types.h> 25#include <asm/elf.h> 26#include <asm/livepatch.h> 27 28/** 29 * klp_write_module_reloc() - write a relocation in a module 30 * @mod: module in which the section to be modified is found 31 * @type: ELF relocation type (see asm/elf.h) 32 * @loc: address that the relocation should be written to 33 * @value: relocation value (sym address + addend) 34 * 35 * This function writes a relocation to the specified location for 36 * a particular module. 37 */ 38int klp_write_module_reloc(struct module *mod, unsigned long type, 39 unsigned long loc, unsigned long value) 40{ 41 int ret, numpages, size = 4; 42 bool readonly; 43 unsigned long val; 44 unsigned long core = (unsigned long)mod->module_core; 45 unsigned long core_size = mod->core_size; 46 47 switch (type) { 48 case R_X86_64_NONE: 49 return 0; 50 case R_X86_64_64: 51 val = value; 52 size = 8; 53 break; 54 case R_X86_64_32: 55 val = (u32)value; 56 break; 57 case R_X86_64_32S: 58 val = (s32)value; 59 break; 60 case R_X86_64_PC32: 61 val = (u32)(value - loc); 62 break; 63 default: 64 /* unsupported relocation type */ 65 return -EINVAL; 66 } 67 68 if (loc < core || loc >= core + core_size) 69 /* loc does not point to any symbol inside the module */ 70 return -EINVAL; 71 72 readonly = false; 73 74#ifdef CONFIG_DEBUG_SET_MODULE_RONX 75 if (loc < core + mod->core_ro_size) 76 readonly = true; 77#endif 78 79 /* determine if the relocation spans a page boundary */ 80 numpages = ((loc & PAGE_MASK) == ((loc + size) & PAGE_MASK)) ? 1 : 2; 81 82 if (readonly) 83 set_memory_rw(loc & PAGE_MASK, numpages); 84 85 ret = probe_kernel_write((void *)loc, &val, size); 86 87 if (readonly) 88 set_memory_ro(loc & PAGE_MASK, numpages); 89 90 return ret; 91} 92