1
2#include <linux/elf.h>
3#include <asm/kexec.h>
4
5int arch_kexec_do_relocs(int r_type, void *loc, unsigned long val,
6 unsigned long addr)
7{
8 switch (r_type) {
9 case R_390_NONE:
10 break;
11 case R_390_8:
12 *(u8 *)loc = val;
13 break;
14 case R_390_12:
15 *(u16 *)loc &= 0xf000;
16 *(u16 *)loc |= val & 0xfff;
17 break;
18 case R_390_16:
19 *(u16 *)loc = val;
20 break;
21 case R_390_20:
22 *(u32 *)loc &= 0xf00000ff;
23 *(u32 *)loc |= (val & 0xfff) << 16;
24 *(u32 *)loc |= (val & 0xff000) >> 4;
25 break;
26 case R_390_32:
27 *(u32 *)loc = val;
28 break;
29 case R_390_64:
30 *(u64 *)loc = val;
31 break;
32 case R_390_PC16:
33 *(u16 *)loc = (val - addr);
34 break;
35 case R_390_PC16DBL:
36 *(u16 *)loc = (val - addr) >> 1;
37 break;
38 case R_390_PC32DBL:
39 *(u32 *)loc = (val - addr) >> 1;
40 break;
41 case R_390_PC32:
42 *(u32 *)loc = (val - addr);
43 break;
44 case R_390_PC64:
45 *(u64 *)loc = (val - addr);
46 break;
47 case R_390_RELATIVE:
48 *(unsigned long *) loc = val;
49 break;
50 default:
51 return 1;
52 }
53 return 0;
54}
55